2005-01-05 22:34:37 +00:00
|
|
|
|
/*-
|
1995-06-25 17:32:43 +00:00
|
|
|
|
* Copyright (c) 1994-1995 S<EFBFBD>ren Schmidt
|
|
|
|
|
* All rights reserved.
|
|
|
|
|
*
|
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
|
* are met:
|
|
|
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
|
|
|
* notice, this list of conditions and the following disclaimer
|
|
|
|
|
* in this position and unchanged.
|
|
|
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
|
* documentation and/or other materials provided with the distribution.
|
|
|
|
|
* 3. The name of the author may not be used to endorse or promote products
|
2002-06-02 20:05:59 +00:00
|
|
|
|
* derived from this software without specific prior written permission
|
1995-06-25 17:32:43 +00:00
|
|
|
|
*
|
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
|
* IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
|
* OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
|
* IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
|
* NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
|
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
|
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
|
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
|
*/
|
|
|
|
|
|
2008-08-25 04:55:29 +00:00
|
|
|
|
#include "opt_compat.h"
|
|
|
|
|
|
2003-06-10 21:29:12 +00:00
|
|
|
|
#include <sys/cdefs.h>
|
|
|
|
|
__FBSDID("$FreeBSD$");
|
|
|
|
|
|
1995-06-25 17:32:43 +00:00
|
|
|
|
#include <sys/param.h>
|
|
|
|
|
#include <sys/systm.h>
|
1995-11-22 07:43:53 +00:00
|
|
|
|
#include <sys/sysproto.h>
|
1999-04-29 04:37:57 +00:00
|
|
|
|
#include <sys/cdio.h>
|
2001-11-18 18:49:07 +00:00
|
|
|
|
#include <sys/dvdio.h>
|
2003-04-01 12:34:47 +00:00
|
|
|
|
#include <sys/conf.h>
|
2003-03-28 08:58:11 +00:00
|
|
|
|
#include <sys/disk.h>
|
2000-10-08 21:34:00 +00:00
|
|
|
|
#include <sys/consio.h>
|
2001-10-15 20:52:17 +00:00
|
|
|
|
#include <sys/ctype.h>
|
1997-03-23 03:37:54 +00:00
|
|
|
|
#include <sys/fcntl.h>
|
1995-06-25 17:32:43 +00:00
|
|
|
|
#include <sys/file.h>
|
|
|
|
|
#include <sys/filedesc.h>
|
1997-03-24 11:37:53 +00:00
|
|
|
|
#include <sys/filio.h>
|
2009-06-17 15:01:01 +00:00
|
|
|
|
#include <sys/jail.h>
|
2000-10-08 21:34:00 +00:00
|
|
|
|
#include <sys/kbio.h>
|
2006-07-06 21:42:36 +00:00
|
|
|
|
#include <sys/kernel.h>
|
1999-12-04 11:10:22 +00:00
|
|
|
|
#include <sys/linker_set.h>
|
2006-07-06 21:42:36 +00:00
|
|
|
|
#include <sys/lock.h>
|
1999-12-04 11:10:22 +00:00
|
|
|
|
#include <sys/malloc.h>
|
2001-10-15 20:52:17 +00:00
|
|
|
|
#include <sys/proc.h>
|
2005-08-27 14:44:10 +00:00
|
|
|
|
#include <sys/sbuf.h>
|
Mega-commit for Linux emulator update.. This has been stress tested under
netscape-2.0 for Linux running all the Java stuff. The scrollbars are now
working, at least on my machine. (whew! :-)
I'm uncomfortable with the size of this commit, but it's too
inter-dependant to easily seperate out.
The main changes:
COMPAT_LINUX is *GONE*. Most of the code has been moved out of the i386
machine dependent section into the linux emulator itself. The int 0x80
syscall code was almost identical to the lcall 7,0 code and a minor tweak
allows them to both be used with the same C code. All kernels can now
just modload the lkm and it'll DTRT without having to rebuild the kernel
first. Like IBCS2, you can statically compile it in with "options LINUX".
A pile of new syscalls implemented, including getdents(), llseek(),
readv(), writev(), msync(), personality(). The Linux-ELF libraries want
to use some of these.
linux_select() now obeys Linux semantics, ie: returns the time remaining
of the timeout value rather than leaving it the original value.
Quite a few bugs removed, including incorrect arguments being used in
syscalls.. eg: mixups between passing the sigset as an int, vs passing
it as a pointer and doing a copyin(), missing return values, unhandled
cases, SIOC* ioctls, etc.
The build for the code has changed. i386/conf/files now knows how
to build linux_genassym and generate linux_assym.h on the fly.
Supporting changes elsewhere in the kernel:
The user-mode signal trampoline has moved from the U area to immediately
below the top of the stack (below PS_STRINGS). This allows the different
binary emulations to have their own signal trampoline code (which gets rid
of the hardwired syscall 103 (sigreturn on BSD, syslog on Linux)) and so
that the emulator can provide the exact "struct sigcontext *" argument to
the program's signal handlers.
The sigstack's "ss_flags" now uses SS_DISABLE and SS_ONSTACK flags, which
have the same values as the re-used SA_DISABLE and SA_ONSTACK which are
intended for sigaction only. This enables the support of a SA_RESETHAND
flag to sigaction to implement the gross SYSV and Linux SA_ONESHOT signal
semantics where the signal handler is reset when it's triggered.
makesyscalls.sh no longer appends the struct sysentvec on the end of the
generated init_sysent.c code. It's a lot saner to have it in a seperate
file rather than trying to update the structure inside the awk script. :-)
At exec time, the dozen bytes or so of signal trampoline code are copied
to the top of the user's stack, rather than obtaining the trampoline code
the old way by getting a clone of the parent's user area. This allows
Linux and native binaries to freely exec each other without getting
trampolines mixed up.
1996-03-02 19:38:20 +00:00
|
|
|
|
#include <sys/socket.h>
|
2001-10-15 20:52:17 +00:00
|
|
|
|
#include <sys/sockio.h>
|
|
|
|
|
#include <sys/soundcard.h>
|
2004-06-14 07:26:23 +00:00
|
|
|
|
#include <sys/stdint.h>
|
2006-07-06 21:42:36 +00:00
|
|
|
|
#include <sys/sx.h>
|
2001-10-15 20:52:17 +00:00
|
|
|
|
#include <sys/tty.h>
|
|
|
|
|
#include <sys/uio.h>
|
Commit step 1 of the vimage project, (network stack)
virtualization work done by Marko Zec (zec@).
This is the first in a series of commits over the course
of the next few weeks.
Mark all uses of global variables to be virtualized
with a V_ prefix.
Use macros to map them back to their global names for
now, so this is a NOP change only.
We hope to have caught at least 85-90% of what is needed
so we do not invalidate a lot of outstanding patches again.
Obtained from: //depot/projects/vimage-commit2/...
Reviewed by: brooks, des, ed, mav, julian,
jamie, kris, rwatson, zec, ...
(various people I forgot, different versions)
md5 (with a bit of help)
Sponsored by: NLnet Foundation, The FreeBSD Foundation
X-MFC after: never
V_Commit_Message_Reviewed_By: more people than the patch
2008-08-17 23:27:27 +00:00
|
|
|
|
|
Mega-commit for Linux emulator update.. This has been stress tested under
netscape-2.0 for Linux running all the Java stuff. The scrollbars are now
working, at least on my machine. (whew! :-)
I'm uncomfortable with the size of this commit, but it's too
inter-dependant to easily seperate out.
The main changes:
COMPAT_LINUX is *GONE*. Most of the code has been moved out of the i386
machine dependent section into the linux emulator itself. The int 0x80
syscall code was almost identical to the lcall 7,0 code and a minor tweak
allows them to both be used with the same C code. All kernels can now
just modload the lkm and it'll DTRT without having to rebuild the kernel
first. Like IBCS2, you can statically compile it in with "options LINUX".
A pile of new syscalls implemented, including getdents(), llseek(),
readv(), writev(), msync(), personality(). The Linux-ELF libraries want
to use some of these.
linux_select() now obeys Linux semantics, ie: returns the time remaining
of the timeout value rather than leaving it the original value.
Quite a few bugs removed, including incorrect arguments being used in
syscalls.. eg: mixups between passing the sigset as an int, vs passing
it as a pointer and doing a copyin(), missing return values, unhandled
cases, SIOC* ioctls, etc.
The build for the code has changed. i386/conf/files now knows how
to build linux_genassym and generate linux_assym.h on the fly.
Supporting changes elsewhere in the kernel:
The user-mode signal trampoline has moved from the U area to immediately
below the top of the stack (below PS_STRINGS). This allows the different
binary emulations to have their own signal trampoline code (which gets rid
of the hardwired syscall 103 (sigreturn on BSD, syslog on Linux)) and so
that the emulator can provide the exact "struct sigcontext *" argument to
the program's signal handlers.
The sigstack's "ss_flags" now uses SS_DISABLE and SS_ONSTACK flags, which
have the same values as the re-used SA_DISABLE and SA_ONSTACK which are
intended for sigaction only. This enables the support of a SA_RESETHAND
flag to sigaction to implement the gross SYSV and Linux SA_ONESHOT signal
semantics where the signal handler is reset when it's triggered.
makesyscalls.sh no longer appends the struct sysentvec on the end of the
generated init_sysent.c code. It's a lot saner to have it in a seperate
file rather than trying to update the structure inside the awk script. :-)
At exec time, the dozen bytes or so of signal trampoline code are copied
to the top of the user's stack, rather than obtaining the trampoline code
the old way by getting a clone of the parent's user area. This allows
Linux and native binaries to freely exec each other without getting
trampolines mixed up.
1996-03-02 19:38:20 +00:00
|
|
|
|
#include <net/if.h>
|
1997-06-02 06:31:49 +00:00
|
|
|
|
#include <net/if_dl.h>
|
|
|
|
|
#include <net/if_types.h>
|
2008-12-02 21:37:28 +00:00
|
|
|
|
#include <net/vnet.h>
|
1995-11-22 07:43:53 +00:00
|
|
|
|
|
2005-01-14 04:44:56 +00:00
|
|
|
|
#ifdef COMPAT_LINUX32
|
2004-08-16 07:28:16 +00:00
|
|
|
|
#include <machine/../linux32/linux.h>
|
|
|
|
|
#include <machine/../linux32/linux32_proto.h>
|
2005-01-14 04:44:56 +00:00
|
|
|
|
#else
|
|
|
|
|
#include <machine/../linux/linux.h>
|
|
|
|
|
#include <machine/../linux/linux_proto.h>
|
2004-08-16 07:28:16 +00:00
|
|
|
|
#endif
|
2000-11-01 19:48:35 +00:00
|
|
|
|
|
2000-08-22 01:32:14 +00:00
|
|
|
|
#include <compat/linux/linux_ioctl.h>
|
|
|
|
|
#include <compat/linux/linux_mib.h>
|
2009-05-07 09:39:20 +00:00
|
|
|
|
#include <compat/linux/linux_socket.h>
|
2000-08-22 01:32:14 +00:00
|
|
|
|
#include <compat/linux/linux_util.h>
|
1995-06-25 17:32:43 +00:00
|
|
|
|
|
2005-03-24 19:26:50 +00:00
|
|
|
|
CTASSERT(LINUX_IFNAMSIZ == IFNAMSIZ);
|
2005-03-24 17:51:15 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
static linux_ioctl_function_t linux_ioctl_cdrom;
|
2002-10-11 11:43:09 +00:00
|
|
|
|
static linux_ioctl_function_t linux_ioctl_vfat;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
static linux_ioctl_function_t linux_ioctl_console;
|
2004-06-14 07:26:23 +00:00
|
|
|
|
static linux_ioctl_function_t linux_ioctl_hdio;
|
2000-03-13 15:27:19 +00:00
|
|
|
|
static linux_ioctl_function_t linux_ioctl_disk;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
static linux_ioctl_function_t linux_ioctl_socket;
|
|
|
|
|
static linux_ioctl_function_t linux_ioctl_sound;
|
|
|
|
|
static linux_ioctl_function_t linux_ioctl_termio;
|
2001-10-19 01:38:10 +00:00
|
|
|
|
static linux_ioctl_function_t linux_ioctl_private;
|
2003-04-24 23:36:35 +00:00
|
|
|
|
static linux_ioctl_function_t linux_ioctl_drm;
|
2007-04-07 19:40:58 +00:00
|
|
|
|
static linux_ioctl_function_t linux_ioctl_sg;
|
2001-11-19 15:43:50 +00:00
|
|
|
|
static linux_ioctl_function_t linux_ioctl_special;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
static struct linux_ioctl_handler cdrom_handler =
|
|
|
|
|
{ linux_ioctl_cdrom, LINUX_IOCTL_CDROM_MIN, LINUX_IOCTL_CDROM_MAX };
|
2002-10-11 11:43:09 +00:00
|
|
|
|
static struct linux_ioctl_handler vfat_handler =
|
|
|
|
|
{ linux_ioctl_vfat, LINUX_IOCTL_VFAT_MIN, LINUX_IOCTL_VFAT_MAX };
|
1999-12-04 11:10:22 +00:00
|
|
|
|
static struct linux_ioctl_handler console_handler =
|
|
|
|
|
{ linux_ioctl_console, LINUX_IOCTL_CONSOLE_MIN, LINUX_IOCTL_CONSOLE_MAX };
|
2004-06-14 07:26:23 +00:00
|
|
|
|
static struct linux_ioctl_handler hdio_handler =
|
|
|
|
|
{ linux_ioctl_hdio, LINUX_IOCTL_HDIO_MIN, LINUX_IOCTL_HDIO_MAX };
|
2000-03-13 15:27:19 +00:00
|
|
|
|
static struct linux_ioctl_handler disk_handler =
|
|
|
|
|
{ linux_ioctl_disk, LINUX_IOCTL_DISK_MIN, LINUX_IOCTL_DISK_MAX };
|
1999-12-04 11:10:22 +00:00
|
|
|
|
static struct linux_ioctl_handler socket_handler =
|
|
|
|
|
{ linux_ioctl_socket, LINUX_IOCTL_SOCKET_MIN, LINUX_IOCTL_SOCKET_MAX };
|
|
|
|
|
static struct linux_ioctl_handler sound_handler =
|
|
|
|
|
{ linux_ioctl_sound, LINUX_IOCTL_SOUND_MIN, LINUX_IOCTL_SOUND_MAX };
|
|
|
|
|
static struct linux_ioctl_handler termio_handler =
|
|
|
|
|
{ linux_ioctl_termio, LINUX_IOCTL_TERMIO_MIN, LINUX_IOCTL_TERMIO_MAX };
|
2001-10-19 01:38:10 +00:00
|
|
|
|
static struct linux_ioctl_handler private_handler =
|
|
|
|
|
{ linux_ioctl_private, LINUX_IOCTL_PRIVATE_MIN, LINUX_IOCTL_PRIVATE_MAX };
|
2003-04-24 23:36:35 +00:00
|
|
|
|
static struct linux_ioctl_handler drm_handler =
|
|
|
|
|
{ linux_ioctl_drm, LINUX_IOCTL_DRM_MIN, LINUX_IOCTL_DRM_MAX };
|
2007-04-07 19:40:58 +00:00
|
|
|
|
static struct linux_ioctl_handler sg_handler =
|
|
|
|
|
{ linux_ioctl_sg, LINUX_IOCTL_SG_MIN, LINUX_IOCTL_SG_MAX };
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
DATA_SET(linux_ioctl_handler_set, cdrom_handler);
|
2002-10-11 11:43:09 +00:00
|
|
|
|
DATA_SET(linux_ioctl_handler_set, vfat_handler);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
DATA_SET(linux_ioctl_handler_set, console_handler);
|
2004-06-14 07:26:23 +00:00
|
|
|
|
DATA_SET(linux_ioctl_handler_set, hdio_handler);
|
2000-03-13 15:27:19 +00:00
|
|
|
|
DATA_SET(linux_ioctl_handler_set, disk_handler);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
DATA_SET(linux_ioctl_handler_set, socket_handler);
|
|
|
|
|
DATA_SET(linux_ioctl_handler_set, sound_handler);
|
|
|
|
|
DATA_SET(linux_ioctl_handler_set, termio_handler);
|
2001-10-19 01:38:10 +00:00
|
|
|
|
DATA_SET(linux_ioctl_handler_set, private_handler);
|
2003-04-24 23:36:35 +00:00
|
|
|
|
DATA_SET(linux_ioctl_handler_set, drm_handler);
|
2007-04-07 19:40:58 +00:00
|
|
|
|
DATA_SET(linux_ioctl_handler_set, sg_handler);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
2003-03-02 15:56:49 +00:00
|
|
|
|
struct handler_element
|
1999-12-04 11:10:22 +00:00
|
|
|
|
{
|
2000-05-26 02:09:24 +00:00
|
|
|
|
TAILQ_ENTRY(handler_element) list;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
int (*func)(struct thread *, struct linux_ioctl_args *);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
int low, high, span;
|
|
|
|
|
};
|
|
|
|
|
|
2000-05-26 02:09:24 +00:00
|
|
|
|
static TAILQ_HEAD(, handler_element) handlers =
|
1999-12-04 11:10:22 +00:00
|
|
|
|
TAILQ_HEAD_INITIALIZER(handlers);
|
2006-07-06 21:42:36 +00:00
|
|
|
|
static struct sx linux_ioctl_sx;
|
|
|
|
|
SX_SYSINIT(linux_ioctl, &linux_ioctl_sx, "linux ioctl handlers");
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
2004-06-14 07:26:23 +00:00
|
|
|
|
/*
|
|
|
|
|
* hdio related ioctls for VMWare support
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
struct linux_hd_geometry {
|
|
|
|
|
u_int8_t heads;
|
|
|
|
|
u_int8_t sectors;
|
|
|
|
|
u_int16_t cylinders;
|
|
|
|
|
u_int32_t start;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct linux_hd_big_geometry {
|
|
|
|
|
u_int8_t heads;
|
|
|
|
|
u_int8_t sectors;
|
|
|
|
|
u_int32_t cylinders;
|
|
|
|
|
u_int32_t start;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
linux_ioctl_hdio(struct thread *td, struct linux_ioctl_args *args)
|
|
|
|
|
{
|
|
|
|
|
struct file *fp;
|
|
|
|
|
int error;
|
|
|
|
|
u_int sectorsize, fwcylinders, fwheads, fwsectors;
|
|
|
|
|
off_t mediasize, bytespercyl;
|
|
|
|
|
|
|
|
|
|
if ((error = fget(td, args->fd, &fp)) != 0)
|
|
|
|
|
return (error);
|
|
|
|
|
switch (args->cmd & 0xffff) {
|
|
|
|
|
case LINUX_HDIO_GET_GEO:
|
|
|
|
|
case LINUX_HDIO_GET_GEO_BIG:
|
|
|
|
|
error = fo_ioctl(fp, DIOCGMEDIASIZE,
|
|
|
|
|
(caddr_t)&mediasize, td->td_ucred, td);
|
|
|
|
|
if (!error)
|
|
|
|
|
error = fo_ioctl(fp, DIOCGSECTORSIZE,
|
|
|
|
|
(caddr_t)§orsize, td->td_ucred, td);
|
|
|
|
|
if (!error)
|
|
|
|
|
error = fo_ioctl(fp, DIOCGFWHEADS,
|
|
|
|
|
(caddr_t)&fwheads, td->td_ucred, td);
|
|
|
|
|
if (!error)
|
|
|
|
|
error = fo_ioctl(fp, DIOCGFWSECTORS,
|
|
|
|
|
(caddr_t)&fwsectors, td->td_ucred, td);
|
|
|
|
|
/*
|
|
|
|
|
* XXX: DIOCGFIRSTOFFSET is not yet implemented, so
|
|
|
|
|
* so pretend that GEOM always says 0. This is NOT VALID
|
|
|
|
|
* for slices or partitions, only the per-disk raw devices.
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
fdrop(fp, td);
|
|
|
|
|
if (error)
|
|
|
|
|
return (error);
|
|
|
|
|
/*
|
|
|
|
|
* 1. Calculate the number of bytes in a cylinder,
|
|
|
|
|
* given the firmware's notion of heads and sectors
|
|
|
|
|
* per cylinder.
|
|
|
|
|
* 2. Calculate the number of cylinders, given the total
|
|
|
|
|
* size of the media.
|
|
|
|
|
* All internal calculations should have 64-bit precision.
|
|
|
|
|
*/
|
|
|
|
|
bytespercyl = (off_t) sectorsize * fwheads * fwsectors;
|
|
|
|
|
fwcylinders = mediasize / bytespercyl;
|
|
|
|
|
#if defined(DEBUG)
|
|
|
|
|
linux_msg(td, "HDIO_GET_GEO: mediasize %jd, c/h/s %d/%d/%d, "
|
|
|
|
|
"bpc %jd",
|
|
|
|
|
(intmax_t)mediasize, fwcylinders, fwheads, fwsectors,
|
|
|
|
|
(intmax_t)bytespercyl);
|
|
|
|
|
#endif
|
|
|
|
|
if ((args->cmd & 0xffff) == LINUX_HDIO_GET_GEO) {
|
|
|
|
|
struct linux_hd_geometry hdg;
|
|
|
|
|
|
|
|
|
|
hdg.cylinders = fwcylinders;
|
|
|
|
|
hdg.heads = fwheads;
|
|
|
|
|
hdg.sectors = fwsectors;
|
|
|
|
|
hdg.start = 0;
|
|
|
|
|
error = copyout(&hdg, (void *)args->arg, sizeof(hdg));
|
|
|
|
|
} else if ((args->cmd & 0xffff) == LINUX_HDIO_GET_GEO_BIG) {
|
|
|
|
|
struct linux_hd_big_geometry hdbg;
|
|
|
|
|
|
|
|
|
|
hdbg.cylinders = fwcylinders;
|
|
|
|
|
hdbg.heads = fwheads;
|
|
|
|
|
hdbg.sectors = fwsectors;
|
|
|
|
|
hdbg.start = 0;
|
|
|
|
|
error = copyout(&hdbg, (void *)args->arg, sizeof(hdbg));
|
|
|
|
|
}
|
|
|
|
|
return (error);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
/* XXX */
|
|
|
|
|
linux_msg(td,
|
|
|
|
|
"ioctl fd=%d, cmd=0x%x ('%c',%d) is not implemented",
|
|
|
|
|
args->fd, (int)(args->cmd & 0xffff),
|
|
|
|
|
(int)(args->cmd & 0xff00) >> 8,
|
|
|
|
|
(int)(args->cmd & 0xff));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
fdrop(fp, td);
|
|
|
|
|
return (ENOIOCTL);
|
|
|
|
|
}
|
|
|
|
|
|
2000-03-09 15:14:14 +00:00
|
|
|
|
static int
|
2001-09-12 08:38:13 +00:00
|
|
|
|
linux_ioctl_disk(struct thread *td, struct linux_ioctl_args *args)
|
2000-03-09 15:14:14 +00:00
|
|
|
|
{
|
2002-01-13 11:58:06 +00:00
|
|
|
|
struct file *fp;
|
2000-03-09 15:14:14 +00:00
|
|
|
|
int error;
|
2003-03-28 08:58:11 +00:00
|
|
|
|
u_int sectorsize;
|
|
|
|
|
off_t mediasize;
|
2000-03-09 15:14:14 +00:00
|
|
|
|
|
2002-01-14 00:13:45 +00:00
|
|
|
|
if ((error = fget(td, args->fd, &fp)) != 0)
|
|
|
|
|
return (error);
|
2000-03-09 15:14:14 +00:00
|
|
|
|
switch (args->cmd & 0xffff) {
|
|
|
|
|
case LINUX_BLKGETSIZE:
|
2003-03-28 08:58:11 +00:00
|
|
|
|
error = fo_ioctl(fp, DIOCGSECTORSIZE,
|
|
|
|
|
(caddr_t)§orsize, td->td_ucred, td);
|
|
|
|
|
if (!error)
|
|
|
|
|
error = fo_ioctl(fp, DIOCGMEDIASIZE,
|
|
|
|
|
(caddr_t)&mediasize, td->td_ucred, td);
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
2000-03-09 15:14:14 +00:00
|
|
|
|
if (error)
|
|
|
|
|
return (error);
|
2003-03-28 08:58:11 +00:00
|
|
|
|
sectorsize = mediasize / sectorsize;
|
|
|
|
|
/*
|
|
|
|
|
* XXX: How do we know we return the right size of integer ?
|
|
|
|
|
*/
|
|
|
|
|
return (copyout(§orsize, (void *)args->arg,
|
|
|
|
|
sizeof(sectorsize)));
|
2004-06-14 07:26:23 +00:00
|
|
|
|
break;
|
2000-03-09 15:14:14 +00:00
|
|
|
|
}
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
2000-03-09 15:14:14 +00:00
|
|
|
|
return (ENOIOCTL);
|
|
|
|
|
}
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
/*
|
|
|
|
|
* termio related ioctls
|
|
|
|
|
*/
|
1998-08-31 06:55:02 +00:00
|
|
|
|
|
1996-03-03 19:07:50 +00:00
|
|
|
|
struct linux_termio {
|
1999-12-04 11:10:22 +00:00
|
|
|
|
unsigned short c_iflag;
|
|
|
|
|
unsigned short c_oflag;
|
|
|
|
|
unsigned short c_cflag;
|
|
|
|
|
unsigned short c_lflag;
|
|
|
|
|
unsigned char c_line;
|
|
|
|
|
unsigned char c_cc[LINUX_NCC];
|
1996-03-03 19:07:50 +00:00
|
|
|
|
};
|
|
|
|
|
|
1995-06-25 17:32:43 +00:00
|
|
|
|
struct linux_termios {
|
1999-12-04 11:10:22 +00:00
|
|
|
|
unsigned int c_iflag;
|
|
|
|
|
unsigned int c_oflag;
|
|
|
|
|
unsigned int c_cflag;
|
|
|
|
|
unsigned int c_lflag;
|
|
|
|
|
unsigned char c_line;
|
|
|
|
|
unsigned char c_cc[LINUX_NCCS];
|
1995-06-25 17:32:43 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct linux_winsize {
|
1999-12-04 11:10:22 +00:00
|
|
|
|
unsigned short ws_row, ws_col;
|
|
|
|
|
unsigned short ws_xpixel, ws_ypixel;
|
1995-06-25 17:32:43 +00:00
|
|
|
|
};
|
|
|
|
|
|
Integrate the new MPSAFE TTY layer to the FreeBSD operating system.
The last half year I've been working on a replacement TTY layer for the
FreeBSD kernel. The new TTY layer was designed to improve the following:
- Improved driver model:
The old TTY layer has a driver model that is not abstract enough to
make it friendly to use. A good example is the output path, where the
device drivers directly access the output buffers. This means that an
in-kernel PPP implementation must always convert network buffers into
TTY buffers.
If a PPP implementation would be built on top of the new TTY layer
(still needs a hooks layer, though), it would allow the PPP
implementation to directly hand the data to the TTY driver.
- Improved hotplugging:
With the old TTY layer, it isn't entirely safe to destroy TTY's from
the system. This implementation has a two-step destructing design,
where the driver first abandons the TTY. After all threads have left
the TTY, the TTY layer calls a routine in the driver, which can be
used to free resources (unit numbers, etc).
The pts(4) driver also implements this feature, which means
posix_openpt() will now return PTY's that are created on the fly.
- Improved performance:
One of the major improvements is the per-TTY mutex, which is expected
to improve scalability when compared to the old Giant locking.
Another change is the unbuffered copying to userspace, which is both
used on TTY device nodes and PTY masters.
Upgrading should be quite straightforward. Unlike previous versions,
existing kernel configuration files do not need to be changed, except
when they reference device drivers that are listed in UPDATING.
Obtained from: //depot/projects/mpsafetty/...
Approved by: philip (ex-mentor)
Discussed: on the lists, at BSDCan, at the DevSummit
Sponsored by: Snow B.V., the Netherlands
dcons(4) fixed by: kan
2008-08-20 08:31:58 +00:00
|
|
|
|
struct speedtab {
|
|
|
|
|
int sp_speed; /* Speed. */
|
|
|
|
|
int sp_code; /* Code. */
|
|
|
|
|
};
|
|
|
|
|
|
1995-06-25 17:32:43 +00:00
|
|
|
|
static struct speedtab sptab[] = {
|
1999-12-04 11:10:22 +00:00
|
|
|
|
{ B0, LINUX_B0 }, { B50, LINUX_B50 },
|
|
|
|
|
{ B75, LINUX_B75 }, { B110, LINUX_B110 },
|
|
|
|
|
{ B134, LINUX_B134 }, { B150, LINUX_B150 },
|
|
|
|
|
{ B200, LINUX_B200 }, { B300, LINUX_B300 },
|
|
|
|
|
{ B600, LINUX_B600 }, { B1200, LINUX_B1200 },
|
|
|
|
|
{ B1800, LINUX_B1800 }, { B2400, LINUX_B2400 },
|
|
|
|
|
{ B4800, LINUX_B4800 }, { B9600, LINUX_B9600 },
|
|
|
|
|
{ B19200, LINUX_B19200 }, { B38400, LINUX_B38400 },
|
|
|
|
|
{ B57600, LINUX_B57600 }, { B115200, LINUX_B115200 },
|
|
|
|
|
{-1, -1 }
|
1995-06-25 17:32:43 +00:00
|
|
|
|
};
|
|
|
|
|
|
1996-03-10 22:30:53 +00:00
|
|
|
|
struct linux_serial_struct {
|
1999-12-04 11:10:22 +00:00
|
|
|
|
int type;
|
|
|
|
|
int line;
|
|
|
|
|
int port;
|
|
|
|
|
int irq;
|
|
|
|
|
int flags;
|
|
|
|
|
int xmit_fifo_size;
|
|
|
|
|
int custom_divisor;
|
|
|
|
|
int baud_base;
|
|
|
|
|
unsigned short close_delay;
|
|
|
|
|
char reserved_char[2];
|
|
|
|
|
int hub6;
|
|
|
|
|
unsigned short closing_wait;
|
|
|
|
|
unsigned short closing_wait2;
|
|
|
|
|
int reserved[4];
|
1996-03-10 22:30:53 +00:00
|
|
|
|
};
|
|
|
|
|
|
1995-06-25 17:32:43 +00:00
|
|
|
|
static int
|
|
|
|
|
linux_to_bsd_speed(int code, struct speedtab *table)
|
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
for ( ; table->sp_code != -1; table++)
|
|
|
|
|
if (table->sp_code == code)
|
|
|
|
|
return (table->sp_speed);
|
|
|
|
|
return -1;
|
1995-06-25 17:32:43 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
bsd_to_linux_speed(int speed, struct speedtab *table)
|
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
for ( ; table->sp_speed != -1; table++)
|
|
|
|
|
if (table->sp_speed == speed)
|
|
|
|
|
return (table->sp_code);
|
|
|
|
|
return -1;
|
1995-06-25 17:32:43 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
1999-12-04 11:10:22 +00:00
|
|
|
|
bsd_to_linux_termios(struct termios *bios, struct linux_termios *lios)
|
1995-06-25 17:32:43 +00:00
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
int i;
|
1995-06-25 17:32:43 +00:00
|
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
2001-02-16 16:40:43 +00:00
|
|
|
|
if (ldebug(ioctl)) {
|
|
|
|
|
printf("LINUX: BSD termios structure (input):\n");
|
|
|
|
|
printf("i=%08x o=%08x c=%08x l=%08x ispeed=%d ospeed=%d\n",
|
|
|
|
|
bios->c_iflag, bios->c_oflag, bios->c_cflag, bios->c_lflag,
|
|
|
|
|
bios->c_ispeed, bios->c_ospeed);
|
|
|
|
|
printf("c_cc ");
|
|
|
|
|
for (i=0; i<NCCS; i++)
|
|
|
|
|
printf("%02x ", bios->c_cc[i]);
|
|
|
|
|
printf("\n");
|
|
|
|
|
}
|
1995-06-25 17:32:43 +00:00
|
|
|
|
#endif
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
lios->c_iflag = 0;
|
|
|
|
|
if (bios->c_iflag & IGNBRK)
|
|
|
|
|
lios->c_iflag |= LINUX_IGNBRK;
|
|
|
|
|
if (bios->c_iflag & BRKINT)
|
|
|
|
|
lios->c_iflag |= LINUX_BRKINT;
|
|
|
|
|
if (bios->c_iflag & IGNPAR)
|
|
|
|
|
lios->c_iflag |= LINUX_IGNPAR;
|
|
|
|
|
if (bios->c_iflag & PARMRK)
|
|
|
|
|
lios->c_iflag |= LINUX_PARMRK;
|
|
|
|
|
if (bios->c_iflag & INPCK)
|
|
|
|
|
lios->c_iflag |= LINUX_INPCK;
|
|
|
|
|
if (bios->c_iflag & ISTRIP)
|
|
|
|
|
lios->c_iflag |= LINUX_ISTRIP;
|
|
|
|
|
if (bios->c_iflag & INLCR)
|
|
|
|
|
lios->c_iflag |= LINUX_INLCR;
|
|
|
|
|
if (bios->c_iflag & IGNCR)
|
|
|
|
|
lios->c_iflag |= LINUX_IGNCR;
|
|
|
|
|
if (bios->c_iflag & ICRNL)
|
|
|
|
|
lios->c_iflag |= LINUX_ICRNL;
|
|
|
|
|
if (bios->c_iflag & IXON)
|
|
|
|
|
lios->c_iflag |= LINUX_IXON;
|
|
|
|
|
if (bios->c_iflag & IXANY)
|
|
|
|
|
lios->c_iflag |= LINUX_IXANY;
|
|
|
|
|
if (bios->c_iflag & IXOFF)
|
|
|
|
|
lios->c_iflag |= LINUX_IXOFF;
|
|
|
|
|
if (bios->c_iflag & IMAXBEL)
|
|
|
|
|
lios->c_iflag |= LINUX_IMAXBEL;
|
|
|
|
|
|
|
|
|
|
lios->c_oflag = 0;
|
|
|
|
|
if (bios->c_oflag & OPOST)
|
|
|
|
|
lios->c_oflag |= LINUX_OPOST;
|
|
|
|
|
if (bios->c_oflag & ONLCR)
|
|
|
|
|
lios->c_oflag |= LINUX_ONLCR;
|
Integrate the new MPSAFE TTY layer to the FreeBSD operating system.
The last half year I've been working on a replacement TTY layer for the
FreeBSD kernel. The new TTY layer was designed to improve the following:
- Improved driver model:
The old TTY layer has a driver model that is not abstract enough to
make it friendly to use. A good example is the output path, where the
device drivers directly access the output buffers. This means that an
in-kernel PPP implementation must always convert network buffers into
TTY buffers.
If a PPP implementation would be built on top of the new TTY layer
(still needs a hooks layer, though), it would allow the PPP
implementation to directly hand the data to the TTY driver.
- Improved hotplugging:
With the old TTY layer, it isn't entirely safe to destroy TTY's from
the system. This implementation has a two-step destructing design,
where the driver first abandons the TTY. After all threads have left
the TTY, the TTY layer calls a routine in the driver, which can be
used to free resources (unit numbers, etc).
The pts(4) driver also implements this feature, which means
posix_openpt() will now return PTY's that are created on the fly.
- Improved performance:
One of the major improvements is the per-TTY mutex, which is expected
to improve scalability when compared to the old Giant locking.
Another change is the unbuffered copying to userspace, which is both
used on TTY device nodes and PTY masters.
Upgrading should be quite straightforward. Unlike previous versions,
existing kernel configuration files do not need to be changed, except
when they reference device drivers that are listed in UPDATING.
Obtained from: //depot/projects/mpsafetty/...
Approved by: philip (ex-mentor)
Discussed: on the lists, at BSDCan, at the DevSummit
Sponsored by: Snow B.V., the Netherlands
dcons(4) fixed by: kan
2008-08-20 08:31:58 +00:00
|
|
|
|
if (bios->c_oflag & TAB3)
|
1999-12-04 11:10:22 +00:00
|
|
|
|
lios->c_oflag |= LINUX_XTABS;
|
|
|
|
|
|
|
|
|
|
lios->c_cflag = bsd_to_linux_speed(bios->c_ispeed, sptab);
|
|
|
|
|
lios->c_cflag |= (bios->c_cflag & CSIZE) >> 4;
|
|
|
|
|
if (bios->c_cflag & CSTOPB)
|
|
|
|
|
lios->c_cflag |= LINUX_CSTOPB;
|
|
|
|
|
if (bios->c_cflag & CREAD)
|
|
|
|
|
lios->c_cflag |= LINUX_CREAD;
|
|
|
|
|
if (bios->c_cflag & PARENB)
|
|
|
|
|
lios->c_cflag |= LINUX_PARENB;
|
|
|
|
|
if (bios->c_cflag & PARODD)
|
|
|
|
|
lios->c_cflag |= LINUX_PARODD;
|
|
|
|
|
if (bios->c_cflag & HUPCL)
|
|
|
|
|
lios->c_cflag |= LINUX_HUPCL;
|
|
|
|
|
if (bios->c_cflag & CLOCAL)
|
|
|
|
|
lios->c_cflag |= LINUX_CLOCAL;
|
|
|
|
|
if (bios->c_cflag & CRTSCTS)
|
|
|
|
|
lios->c_cflag |= LINUX_CRTSCTS;
|
|
|
|
|
|
|
|
|
|
lios->c_lflag = 0;
|
|
|
|
|
if (bios->c_lflag & ISIG)
|
|
|
|
|
lios->c_lflag |= LINUX_ISIG;
|
|
|
|
|
if (bios->c_lflag & ICANON)
|
|
|
|
|
lios->c_lflag |= LINUX_ICANON;
|
|
|
|
|
if (bios->c_lflag & ECHO)
|
|
|
|
|
lios->c_lflag |= LINUX_ECHO;
|
|
|
|
|
if (bios->c_lflag & ECHOE)
|
|
|
|
|
lios->c_lflag |= LINUX_ECHOE;
|
|
|
|
|
if (bios->c_lflag & ECHOK)
|
|
|
|
|
lios->c_lflag |= LINUX_ECHOK;
|
|
|
|
|
if (bios->c_lflag & ECHONL)
|
|
|
|
|
lios->c_lflag |= LINUX_ECHONL;
|
|
|
|
|
if (bios->c_lflag & NOFLSH)
|
|
|
|
|
lios->c_lflag |= LINUX_NOFLSH;
|
|
|
|
|
if (bios->c_lflag & TOSTOP)
|
|
|
|
|
lios->c_lflag |= LINUX_TOSTOP;
|
|
|
|
|
if (bios->c_lflag & ECHOCTL)
|
|
|
|
|
lios->c_lflag |= LINUX_ECHOCTL;
|
|
|
|
|
if (bios->c_lflag & ECHOPRT)
|
|
|
|
|
lios->c_lflag |= LINUX_ECHOPRT;
|
|
|
|
|
if (bios->c_lflag & ECHOKE)
|
|
|
|
|
lios->c_lflag |= LINUX_ECHOKE;
|
|
|
|
|
if (bios->c_lflag & FLUSHO)
|
|
|
|
|
lios->c_lflag |= LINUX_FLUSHO;
|
|
|
|
|
if (bios->c_lflag & PENDIN)
|
|
|
|
|
lios->c_lflag |= LINUX_PENDIN;
|
|
|
|
|
if (bios->c_lflag & IEXTEN)
|
|
|
|
|
lios->c_lflag |= LINUX_IEXTEN;
|
|
|
|
|
|
|
|
|
|
for (i=0; i<LINUX_NCCS; i++)
|
|
|
|
|
lios->c_cc[i] = LINUX_POSIX_VDISABLE;
|
|
|
|
|
lios->c_cc[LINUX_VINTR] = bios->c_cc[VINTR];
|
|
|
|
|
lios->c_cc[LINUX_VQUIT] = bios->c_cc[VQUIT];
|
|
|
|
|
lios->c_cc[LINUX_VERASE] = bios->c_cc[VERASE];
|
|
|
|
|
lios->c_cc[LINUX_VKILL] = bios->c_cc[VKILL];
|
|
|
|
|
lios->c_cc[LINUX_VEOF] = bios->c_cc[VEOF];
|
|
|
|
|
lios->c_cc[LINUX_VEOL] = bios->c_cc[VEOL];
|
|
|
|
|
lios->c_cc[LINUX_VMIN] = bios->c_cc[VMIN];
|
|
|
|
|
lios->c_cc[LINUX_VTIME] = bios->c_cc[VTIME];
|
|
|
|
|
lios->c_cc[LINUX_VEOL2] = bios->c_cc[VEOL2];
|
|
|
|
|
lios->c_cc[LINUX_VSUSP] = bios->c_cc[VSUSP];
|
|
|
|
|
lios->c_cc[LINUX_VSTART] = bios->c_cc[VSTART];
|
|
|
|
|
lios->c_cc[LINUX_VSTOP] = bios->c_cc[VSTOP];
|
|
|
|
|
lios->c_cc[LINUX_VREPRINT] = bios->c_cc[VREPRINT];
|
|
|
|
|
lios->c_cc[LINUX_VDISCARD] = bios->c_cc[VDISCARD];
|
|
|
|
|
lios->c_cc[LINUX_VWERASE] = bios->c_cc[VWERASE];
|
|
|
|
|
lios->c_cc[LINUX_VLNEXT] = bios->c_cc[VLNEXT];
|
|
|
|
|
|
|
|
|
|
for (i=0; i<LINUX_NCCS; i++) {
|
2003-06-28 19:32:07 +00:00
|
|
|
|
if (i != LINUX_VMIN && i != LINUX_VTIME &&
|
|
|
|
|
lios->c_cc[i] == _POSIX_VDISABLE)
|
1999-12-04 11:10:22 +00:00
|
|
|
|
lios->c_cc[i] = LINUX_POSIX_VDISABLE;
|
|
|
|
|
}
|
|
|
|
|
lios->c_line = 0;
|
|
|
|
|
|
1995-06-25 17:32:43 +00:00
|
|
|
|
#ifdef DEBUG
|
2001-02-16 16:40:43 +00:00
|
|
|
|
if (ldebug(ioctl)) {
|
|
|
|
|
printf("LINUX: LINUX termios structure (output):\n");
|
|
|
|
|
printf("i=%08x o=%08x c=%08x l=%08x line=%d\n",
|
|
|
|
|
lios->c_iflag, lios->c_oflag, lios->c_cflag,
|
|
|
|
|
lios->c_lflag, (int)lios->c_line);
|
|
|
|
|
printf("c_cc ");
|
2003-03-02 15:56:49 +00:00
|
|
|
|
for (i=0; i<LINUX_NCCS; i++)
|
2001-02-16 16:40:43 +00:00
|
|
|
|
printf("%02x ", lios->c_cc[i]);
|
|
|
|
|
printf("\n");
|
|
|
|
|
}
|
1995-06-25 17:32:43 +00:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
1999-12-04 11:10:22 +00:00
|
|
|
|
linux_to_bsd_termios(struct linux_termios *lios, struct termios *bios)
|
1995-06-25 17:32:43 +00:00
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
int i;
|
|
|
|
|
|
1995-06-25 17:32:43 +00:00
|
|
|
|
#ifdef DEBUG
|
2001-02-16 16:40:43 +00:00
|
|
|
|
if (ldebug(ioctl)) {
|
|
|
|
|
printf("LINUX: LINUX termios structure (input):\n");
|
2003-03-02 15:56:49 +00:00
|
|
|
|
printf("i=%08x o=%08x c=%08x l=%08x line=%d\n",
|
2001-02-16 16:40:43 +00:00
|
|
|
|
lios->c_iflag, lios->c_oflag, lios->c_cflag,
|
|
|
|
|
lios->c_lflag, (int)lios->c_line);
|
|
|
|
|
printf("c_cc ");
|
|
|
|
|
for (i=0; i<LINUX_NCCS; i++)
|
|
|
|
|
printf("%02x ", lios->c_cc[i]);
|
|
|
|
|
printf("\n");
|
|
|
|
|
}
|
1995-06-25 17:32:43 +00:00
|
|
|
|
#endif
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
bios->c_iflag = 0;
|
|
|
|
|
if (lios->c_iflag & LINUX_IGNBRK)
|
|
|
|
|
bios->c_iflag |= IGNBRK;
|
|
|
|
|
if (lios->c_iflag & LINUX_BRKINT)
|
|
|
|
|
bios->c_iflag |= BRKINT;
|
|
|
|
|
if (lios->c_iflag & LINUX_IGNPAR)
|
|
|
|
|
bios->c_iflag |= IGNPAR;
|
|
|
|
|
if (lios->c_iflag & LINUX_PARMRK)
|
|
|
|
|
bios->c_iflag |= PARMRK;
|
|
|
|
|
if (lios->c_iflag & LINUX_INPCK)
|
|
|
|
|
bios->c_iflag |= INPCK;
|
|
|
|
|
if (lios->c_iflag & LINUX_ISTRIP)
|
|
|
|
|
bios->c_iflag |= ISTRIP;
|
|
|
|
|
if (lios->c_iflag & LINUX_INLCR)
|
|
|
|
|
bios->c_iflag |= INLCR;
|
|
|
|
|
if (lios->c_iflag & LINUX_IGNCR)
|
|
|
|
|
bios->c_iflag |= IGNCR;
|
|
|
|
|
if (lios->c_iflag & LINUX_ICRNL)
|
|
|
|
|
bios->c_iflag |= ICRNL;
|
|
|
|
|
if (lios->c_iflag & LINUX_IXON)
|
|
|
|
|
bios->c_iflag |= IXON;
|
|
|
|
|
if (lios->c_iflag & LINUX_IXANY)
|
|
|
|
|
bios->c_iflag |= IXANY;
|
|
|
|
|
if (lios->c_iflag & LINUX_IXOFF)
|
|
|
|
|
bios->c_iflag |= IXOFF;
|
|
|
|
|
if (lios->c_iflag & LINUX_IMAXBEL)
|
|
|
|
|
bios->c_iflag |= IMAXBEL;
|
|
|
|
|
|
|
|
|
|
bios->c_oflag = 0;
|
|
|
|
|
if (lios->c_oflag & LINUX_OPOST)
|
|
|
|
|
bios->c_oflag |= OPOST;
|
|
|
|
|
if (lios->c_oflag & LINUX_ONLCR)
|
|
|
|
|
bios->c_oflag |= ONLCR;
|
|
|
|
|
if (lios->c_oflag & LINUX_XTABS)
|
Integrate the new MPSAFE TTY layer to the FreeBSD operating system.
The last half year I've been working on a replacement TTY layer for the
FreeBSD kernel. The new TTY layer was designed to improve the following:
- Improved driver model:
The old TTY layer has a driver model that is not abstract enough to
make it friendly to use. A good example is the output path, where the
device drivers directly access the output buffers. This means that an
in-kernel PPP implementation must always convert network buffers into
TTY buffers.
If a PPP implementation would be built on top of the new TTY layer
(still needs a hooks layer, though), it would allow the PPP
implementation to directly hand the data to the TTY driver.
- Improved hotplugging:
With the old TTY layer, it isn't entirely safe to destroy TTY's from
the system. This implementation has a two-step destructing design,
where the driver first abandons the TTY. After all threads have left
the TTY, the TTY layer calls a routine in the driver, which can be
used to free resources (unit numbers, etc).
The pts(4) driver also implements this feature, which means
posix_openpt() will now return PTY's that are created on the fly.
- Improved performance:
One of the major improvements is the per-TTY mutex, which is expected
to improve scalability when compared to the old Giant locking.
Another change is the unbuffered copying to userspace, which is both
used on TTY device nodes and PTY masters.
Upgrading should be quite straightforward. Unlike previous versions,
existing kernel configuration files do not need to be changed, except
when they reference device drivers that are listed in UPDATING.
Obtained from: //depot/projects/mpsafetty/...
Approved by: philip (ex-mentor)
Discussed: on the lists, at BSDCan, at the DevSummit
Sponsored by: Snow B.V., the Netherlands
dcons(4) fixed by: kan
2008-08-20 08:31:58 +00:00
|
|
|
|
bios->c_oflag |= TAB3;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
bios->c_cflag = (lios->c_cflag & LINUX_CSIZE) << 4;
|
|
|
|
|
if (lios->c_cflag & LINUX_CSTOPB)
|
|
|
|
|
bios->c_cflag |= CSTOPB;
|
|
|
|
|
if (lios->c_cflag & LINUX_CREAD)
|
|
|
|
|
bios->c_cflag |= CREAD;
|
|
|
|
|
if (lios->c_cflag & LINUX_PARENB)
|
|
|
|
|
bios->c_cflag |= PARENB;
|
|
|
|
|
if (lios->c_cflag & LINUX_PARODD)
|
|
|
|
|
bios->c_cflag |= PARODD;
|
|
|
|
|
if (lios->c_cflag & LINUX_HUPCL)
|
|
|
|
|
bios->c_cflag |= HUPCL;
|
|
|
|
|
if (lios->c_cflag & LINUX_CLOCAL)
|
|
|
|
|
bios->c_cflag |= CLOCAL;
|
|
|
|
|
if (lios->c_cflag & LINUX_CRTSCTS)
|
|
|
|
|
bios->c_cflag |= CRTSCTS;
|
|
|
|
|
|
|
|
|
|
bios->c_lflag = 0;
|
|
|
|
|
if (lios->c_lflag & LINUX_ISIG)
|
|
|
|
|
bios->c_lflag |= ISIG;
|
|
|
|
|
if (lios->c_lflag & LINUX_ICANON)
|
|
|
|
|
bios->c_lflag |= ICANON;
|
|
|
|
|
if (lios->c_lflag & LINUX_ECHO)
|
|
|
|
|
bios->c_lflag |= ECHO;
|
|
|
|
|
if (lios->c_lflag & LINUX_ECHOE)
|
|
|
|
|
bios->c_lflag |= ECHOE;
|
|
|
|
|
if (lios->c_lflag & LINUX_ECHOK)
|
|
|
|
|
bios->c_lflag |= ECHOK;
|
|
|
|
|
if (lios->c_lflag & LINUX_ECHONL)
|
|
|
|
|
bios->c_lflag |= ECHONL;
|
|
|
|
|
if (lios->c_lflag & LINUX_NOFLSH)
|
|
|
|
|
bios->c_lflag |= NOFLSH;
|
|
|
|
|
if (lios->c_lflag & LINUX_TOSTOP)
|
|
|
|
|
bios->c_lflag |= TOSTOP;
|
|
|
|
|
if (lios->c_lflag & LINUX_ECHOCTL)
|
|
|
|
|
bios->c_lflag |= ECHOCTL;
|
|
|
|
|
if (lios->c_lflag & LINUX_ECHOPRT)
|
|
|
|
|
bios->c_lflag |= ECHOPRT;
|
|
|
|
|
if (lios->c_lflag & LINUX_ECHOKE)
|
|
|
|
|
bios->c_lflag |= ECHOKE;
|
|
|
|
|
if (lios->c_lflag & LINUX_FLUSHO)
|
|
|
|
|
bios->c_lflag |= FLUSHO;
|
|
|
|
|
if (lios->c_lflag & LINUX_PENDIN)
|
|
|
|
|
bios->c_lflag |= PENDIN;
|
|
|
|
|
if (lios->c_lflag & LINUX_IEXTEN)
|
|
|
|
|
bios->c_lflag |= IEXTEN;
|
|
|
|
|
|
|
|
|
|
for (i=0; i<NCCS; i++)
|
|
|
|
|
bios->c_cc[i] = _POSIX_VDISABLE;
|
|
|
|
|
bios->c_cc[VINTR] = lios->c_cc[LINUX_VINTR];
|
|
|
|
|
bios->c_cc[VQUIT] = lios->c_cc[LINUX_VQUIT];
|
|
|
|
|
bios->c_cc[VERASE] = lios->c_cc[LINUX_VERASE];
|
|
|
|
|
bios->c_cc[VKILL] = lios->c_cc[LINUX_VKILL];
|
|
|
|
|
bios->c_cc[VEOF] = lios->c_cc[LINUX_VEOF];
|
|
|
|
|
bios->c_cc[VEOL] = lios->c_cc[LINUX_VEOL];
|
|
|
|
|
bios->c_cc[VMIN] = lios->c_cc[LINUX_VMIN];
|
|
|
|
|
bios->c_cc[VTIME] = lios->c_cc[LINUX_VTIME];
|
|
|
|
|
bios->c_cc[VEOL2] = lios->c_cc[LINUX_VEOL2];
|
|
|
|
|
bios->c_cc[VSUSP] = lios->c_cc[LINUX_VSUSP];
|
|
|
|
|
bios->c_cc[VSTART] = lios->c_cc[LINUX_VSTART];
|
|
|
|
|
bios->c_cc[VSTOP] = lios->c_cc[LINUX_VSTOP];
|
|
|
|
|
bios->c_cc[VREPRINT] = lios->c_cc[LINUX_VREPRINT];
|
|
|
|
|
bios->c_cc[VDISCARD] = lios->c_cc[LINUX_VDISCARD];
|
|
|
|
|
bios->c_cc[VWERASE] = lios->c_cc[LINUX_VWERASE];
|
|
|
|
|
bios->c_cc[VLNEXT] = lios->c_cc[LINUX_VLNEXT];
|
|
|
|
|
|
|
|
|
|
for (i=0; i<NCCS; i++) {
|
2003-06-28 19:32:07 +00:00
|
|
|
|
if (i != VMIN && i != VTIME &&
|
|
|
|
|
bios->c_cc[i] == LINUX_POSIX_VDISABLE)
|
1999-12-04 11:10:22 +00:00
|
|
|
|
bios->c_cc[i] = _POSIX_VDISABLE;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bios->c_ispeed = bios->c_ospeed =
|
|
|
|
|
linux_to_bsd_speed(lios->c_cflag & LINUX_CBAUD, sptab);
|
|
|
|
|
|
1995-06-25 17:32:43 +00:00
|
|
|
|
#ifdef DEBUG
|
2001-02-16 16:40:43 +00:00
|
|
|
|
if (ldebug(ioctl)) {
|
|
|
|
|
printf("LINUX: BSD termios structure (output):\n");
|
|
|
|
|
printf("i=%08x o=%08x c=%08x l=%08x ispeed=%d ospeed=%d\n",
|
|
|
|
|
bios->c_iflag, bios->c_oflag, bios->c_cflag, bios->c_lflag,
|
|
|
|
|
bios->c_ispeed, bios->c_ospeed);
|
|
|
|
|
printf("c_cc ");
|
2003-03-02 15:56:49 +00:00
|
|
|
|
for (i=0; i<NCCS; i++)
|
2001-02-16 16:40:43 +00:00
|
|
|
|
printf("%02x ", bios->c_cc[i]);
|
|
|
|
|
printf("\n");
|
|
|
|
|
}
|
1995-06-25 17:32:43 +00:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
1996-03-03 19:07:50 +00:00
|
|
|
|
static void
|
1999-12-04 11:10:22 +00:00
|
|
|
|
bsd_to_linux_termio(struct termios *bios, struct linux_termio *lio)
|
1996-03-03 19:07:50 +00:00
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
struct linux_termios lios;
|
|
|
|
|
|
|
|
|
|
bsd_to_linux_termios(bios, &lios);
|
|
|
|
|
lio->c_iflag = lios.c_iflag;
|
|
|
|
|
lio->c_oflag = lios.c_oflag;
|
|
|
|
|
lio->c_cflag = lios.c_cflag;
|
|
|
|
|
lio->c_lflag = lios.c_lflag;
|
|
|
|
|
lio->c_line = lios.c_line;
|
|
|
|
|
memcpy(lio->c_cc, lios.c_cc, LINUX_NCC);
|
1996-03-03 19:07:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
1999-12-04 11:10:22 +00:00
|
|
|
|
linux_to_bsd_termio(struct linux_termio *lio, struct termios *bios)
|
|
|
|
|
{
|
|
|
|
|
struct linux_termios lios;
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
lios.c_iflag = lio->c_iflag;
|
|
|
|
|
lios.c_oflag = lio->c_oflag;
|
|
|
|
|
lios.c_cflag = lio->c_cflag;
|
|
|
|
|
lios.c_lflag = lio->c_lflag;
|
|
|
|
|
for (i=LINUX_NCC; i<LINUX_NCCS; i++)
|
|
|
|
|
lios.c_cc[i] = LINUX_POSIX_VDISABLE;
|
|
|
|
|
memcpy(lios.c_cc, lio->c_cc, LINUX_NCC);
|
|
|
|
|
linux_to_bsd_termios(&lios, bios);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
2001-09-12 08:38:13 +00:00
|
|
|
|
linux_ioctl_termio(struct thread *td, struct linux_ioctl_args *args)
|
1996-03-03 19:07:50 +00:00
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
struct termios bios;
|
|
|
|
|
struct linux_termios lios;
|
|
|
|
|
struct linux_termio lio;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
struct file *fp;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
int error;
|
|
|
|
|
|
2002-01-14 00:13:45 +00:00
|
|
|
|
if ((error = fget(td, args->fd, &fp)) != 0)
|
|
|
|
|
return (error);
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
switch (args->cmd & 0xffff) {
|
|
|
|
|
|
|
|
|
|
case LINUX_TCGETS:
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = fo_ioctl(fp, TIOCGETA, (caddr_t)&bios, td->td_ucred,
|
|
|
|
|
td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
bsd_to_linux_termios(&bios, &lios);
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyout(&lios, (void *)args->arg, sizeof(lios));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TCSETS:
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin((void *)args->arg, &lios, sizeof(lios));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
linux_to_bsd_termios(&lios, &bios);
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = (fo_ioctl(fp, TIOCSETA, (caddr_t)&bios, td->td_ucred,
|
|
|
|
|
td));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TCSETSW:
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin((void *)args->arg, &lios, sizeof(lios));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
linux_to_bsd_termios(&lios, &bios);
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = (fo_ioctl(fp, TIOCSETAW, (caddr_t)&bios, td->td_ucred,
|
|
|
|
|
td));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TCSETSF:
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin((void *)args->arg, &lios, sizeof(lios));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
linux_to_bsd_termios(&lios, &bios);
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = (fo_ioctl(fp, TIOCSETAF, (caddr_t)&bios, td->td_ucred,
|
|
|
|
|
td));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TCGETA:
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = fo_ioctl(fp, TIOCGETA, (caddr_t)&bios, td->td_ucred,
|
|
|
|
|
td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
bsd_to_linux_termio(&bios, &lio);
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = (copyout(&lio, (void *)args->arg, sizeof(lio)));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TCSETA:
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin((void *)args->arg, &lio, sizeof(lio));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
linux_to_bsd_termio(&lio, &bios);
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = (fo_ioctl(fp, TIOCSETA, (caddr_t)&bios, td->td_ucred,
|
|
|
|
|
td));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TCSETAW:
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin((void *)args->arg, &lio, sizeof(lio));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
linux_to_bsd_termio(&lio, &bios);
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = (fo_ioctl(fp, TIOCSETAW, (caddr_t)&bios, td->td_ucred,
|
|
|
|
|
td));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TCSETAF:
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin((void *)args->arg, &lio, sizeof(lio));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
linux_to_bsd_termio(&lio, &bios);
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = (fo_ioctl(fp, TIOCSETAF, (caddr_t)&bios, td->td_ucred,
|
|
|
|
|
td));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
/* LINUX_TCSBRK */
|
|
|
|
|
|
|
|
|
|
case LINUX_TCXONC: {
|
|
|
|
|
switch (args->arg) {
|
|
|
|
|
case LINUX_TCOOFF:
|
|
|
|
|
args->cmd = TIOCSTOP;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_TCOON:
|
|
|
|
|
args->cmd = TIOCSTART;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_TCIOFF:
|
|
|
|
|
case LINUX_TCION: {
|
|
|
|
|
int c;
|
|
|
|
|
struct write_args wr;
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = fo_ioctl(fp, TIOCGETA, (caddr_t)&bios,
|
|
|
|
|
td->td_ucred, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
|
|
|
|
fdrop(fp, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
c = (args->arg == LINUX_TCIOFF) ? VSTOP : VSTART;
|
|
|
|
|
c = bios.c_cc[c];
|
|
|
|
|
if (c != _POSIX_VDISABLE) {
|
|
|
|
|
wr.fd = args->fd;
|
|
|
|
|
wr.buf = &c;
|
|
|
|
|
wr.nbyte = sizeof(c);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (write(td, &wr));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
} else
|
|
|
|
|
return (0);
|
|
|
|
|
}
|
|
|
|
|
default:
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|
|
|
|
|
args->arg = 0;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case LINUX_TCFLSH: {
|
2005-12-13 15:32:52 +00:00
|
|
|
|
int val;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
switch (args->arg) {
|
|
|
|
|
case LINUX_TCIFLUSH:
|
2005-12-13 15:32:52 +00:00
|
|
|
|
val = FREAD;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
break;
|
|
|
|
|
case LINUX_TCOFLUSH:
|
2005-12-13 15:32:52 +00:00
|
|
|
|
val = FWRITE;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
break;
|
|
|
|
|
case LINUX_TCIOFLUSH:
|
2005-12-13 15:32:52 +00:00
|
|
|
|
val = FREAD | FWRITE;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|
2005-12-13 15:32:52 +00:00
|
|
|
|
error = (fo_ioctl(fp,TIOCFLUSH,(caddr_t)&val,td->td_ucred,td));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case LINUX_TIOCEXCL:
|
|
|
|
|
args->cmd = TIOCEXCL;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TIOCNXCL:
|
|
|
|
|
args->cmd = TIOCNXCL;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
2002-09-05 02:51:25 +00:00
|
|
|
|
case LINUX_TIOCSCTTY:
|
|
|
|
|
args->cmd = TIOCSCTTY;
|
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TIOCGPGRP:
|
|
|
|
|
args->cmd = TIOCGPGRP;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TIOCSPGRP:
|
|
|
|
|
args->cmd = TIOCSPGRP;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
/* LINUX_TIOCOUTQ */
|
|
|
|
|
/* LINUX_TIOCSTI */
|
|
|
|
|
|
|
|
|
|
case LINUX_TIOCGWINSZ:
|
|
|
|
|
args->cmd = TIOCGWINSZ;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TIOCSWINSZ:
|
|
|
|
|
args->cmd = TIOCSWINSZ;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TIOCMGET:
|
|
|
|
|
args->cmd = TIOCMGET;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1996-03-03 19:07:50 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_TIOCMBIS:
|
|
|
|
|
args->cmd = TIOCMBIS;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1996-03-03 19:07:50 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_TIOCMBIC:
|
|
|
|
|
args->cmd = TIOCMBIC;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1996-03-03 19:07:50 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_TIOCMSET:
|
|
|
|
|
args->cmd = TIOCMSET;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
/* TIOCGSOFTCAR */
|
|
|
|
|
/* TIOCSSOFTCAR */
|
|
|
|
|
|
|
|
|
|
case LINUX_FIONREAD: /* LINUX_TIOCINQ */
|
|
|
|
|
args->cmd = FIONREAD;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
/* LINUX_TIOCLINUX */
|
|
|
|
|
|
|
|
|
|
case LINUX_TIOCCONS:
|
|
|
|
|
args->cmd = TIOCCONS;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TIOCGSERIAL: {
|
|
|
|
|
struct linux_serial_struct lss;
|
|
|
|
|
lss.type = LINUX_PORT_16550A;
|
|
|
|
|
lss.flags = 0;
|
|
|
|
|
lss.close_delay = 0;
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyout(&lss, (void *)args->arg, sizeof(lss));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case LINUX_TIOCSSERIAL: {
|
|
|
|
|
struct linux_serial_struct lss;
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin((void *)args->arg, &lss, sizeof(lss));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
/* XXX - It really helps to have an implementation that
|
|
|
|
|
* does nothing. NOT!
|
|
|
|
|
*/
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = 0;
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
2008-07-23 17:47:44 +00:00
|
|
|
|
case LINUX_TIOCPKT:
|
|
|
|
|
args->cmd = TIOCPKT;
|
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_FIONBIO:
|
|
|
|
|
args->cmd = FIONBIO;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TIOCNOTTY:
|
|
|
|
|
args->cmd = TIOCNOTTY;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_TIOCSETD: {
|
|
|
|
|
int line;
|
|
|
|
|
switch (args->arg) {
|
|
|
|
|
case LINUX_N_TTY:
|
|
|
|
|
line = TTYDISC;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_N_SLIP:
|
|
|
|
|
line = SLIPDISC;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_N_PPP:
|
|
|
|
|
line = PPPDISC;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = (fo_ioctl(fp, TIOCSETD, (caddr_t)&line, td->td_ucred,
|
|
|
|
|
td));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case LINUX_TIOCGETD: {
|
|
|
|
|
int linux_line;
|
|
|
|
|
int bsd_line = TTYDISC;
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = fo_ioctl(fp, TIOCGETD, (caddr_t)&bsd_line,
|
|
|
|
|
td->td_ucred, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
|
|
|
|
return (error);
|
|
|
|
|
switch (bsd_line) {
|
|
|
|
|
case TTYDISC:
|
|
|
|
|
linux_line = LINUX_N_TTY;
|
|
|
|
|
break;
|
|
|
|
|
case SLIPDISC:
|
|
|
|
|
linux_line = LINUX_N_SLIP;
|
|
|
|
|
break;
|
|
|
|
|
case PPPDISC:
|
|
|
|
|
linux_line = LINUX_N_PPP;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = (copyout(&linux_line, (void *)args->arg, sizeof(int)));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* LINUX_TCSBRKP */
|
|
|
|
|
/* LINUX_TIOCTTYGSTRUCT */
|
|
|
|
|
|
|
|
|
|
case LINUX_FIONCLEX:
|
|
|
|
|
args->cmd = FIONCLEX;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_FIOCLEX:
|
|
|
|
|
args->cmd = FIOCLEX;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_FIOASYNC:
|
|
|
|
|
args->cmd = FIOASYNC;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
/* LINUX_TIOCSERCONFIG */
|
|
|
|
|
/* LINUX_TIOCSERGWILD */
|
|
|
|
|
/* LINUX_TIOCSERSWILD */
|
|
|
|
|
/* LINUX_TIOCGLCKTRMIOS */
|
|
|
|
|
/* LINUX_TIOCSLCKTRMIOS */
|
|
|
|
|
|
2004-02-19 12:38:12 +00:00
|
|
|
|
case LINUX_TIOCSBRK:
|
|
|
|
|
args->cmd = TIOCSBRK;
|
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case LINUX_TIOCCBRK:
|
|
|
|
|
args->cmd = TIOCCBRK;
|
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
2006-01-26 01:32:46 +00:00
|
|
|
|
case LINUX_TIOCGPTN: {
|
|
|
|
|
int nb;
|
|
|
|
|
|
|
|
|
|
error = fo_ioctl(fp, TIOCGPTN, (caddr_t)&nb, td->td_ucred, td);
|
|
|
|
|
if (!error)
|
|
|
|
|
error = copyout(&nb, (void *)args->arg,
|
|
|
|
|
sizeof(int));
|
|
|
|
|
break;
|
|
|
|
|
}
|
2008-07-23 17:47:44 +00:00
|
|
|
|
case LINUX_TIOCSPTLCK:
|
|
|
|
|
/* Our unlockpt() does nothing. */
|
|
|
|
|
error = 0;
|
|
|
|
|
break;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
default:
|
|
|
|
|
error = ENOIOCTL;
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
|
|
|
|
return (error);
|
1996-03-03 19:07:50 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
/*
|
|
|
|
|
* CDROM related ioctls
|
|
|
|
|
*/
|
|
|
|
|
|
1999-04-29 04:37:57 +00:00
|
|
|
|
struct linux_cdrom_msf
|
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
u_char cdmsf_min0;
|
|
|
|
|
u_char cdmsf_sec0;
|
|
|
|
|
u_char cdmsf_frame0;
|
|
|
|
|
u_char cdmsf_min1;
|
|
|
|
|
u_char cdmsf_sec1;
|
|
|
|
|
u_char cdmsf_frame1;
|
1999-04-29 04:37:57 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct linux_cdrom_tochdr
|
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
u_char cdth_trk0;
|
|
|
|
|
u_char cdth_trk1;
|
1999-04-29 04:37:57 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
union linux_cdrom_addr
|
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
struct {
|
|
|
|
|
u_char minute;
|
|
|
|
|
u_char second;
|
|
|
|
|
u_char frame;
|
|
|
|
|
} msf;
|
|
|
|
|
int lba;
|
1999-04-29 04:37:57 +00:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct linux_cdrom_tocentry
|
|
|
|
|
{
|
2003-03-02 15:56:49 +00:00
|
|
|
|
u_char cdte_track;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
u_char cdte_adr:4;
|
|
|
|
|
u_char cdte_ctrl:4;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
u_char cdte_format;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
union linux_cdrom_addr cdte_addr;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
u_char cdte_datamode;
|
1999-04-29 04:37:57 +00:00
|
|
|
|
};
|
|
|
|
|
|
1999-08-13 14:44:13 +00:00
|
|
|
|
struct linux_cdrom_subchnl
|
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
u_char cdsc_format;
|
|
|
|
|
u_char cdsc_audiostatus;
|
|
|
|
|
u_char cdsc_adr:4;
|
|
|
|
|
u_char cdsc_ctrl:4;
|
|
|
|
|
u_char cdsc_trk;
|
|
|
|
|
u_char cdsc_ind;
|
|
|
|
|
union linux_cdrom_addr cdsc_absaddr;
|
|
|
|
|
union linux_cdrom_addr cdsc_reladdr;
|
1999-08-13 14:44:13 +00:00
|
|
|
|
};
|
|
|
|
|
|
2002-10-19 21:11:43 +00:00
|
|
|
|
struct l_cdrom_read_audio {
|
|
|
|
|
union linux_cdrom_addr addr;
|
|
|
|
|
u_char addr_format;
|
|
|
|
|
l_int nframes;
|
|
|
|
|
u_char *buf;
|
|
|
|
|
};
|
|
|
|
|
|
2001-11-18 18:49:07 +00:00
|
|
|
|
struct l_dvd_layer {
|
|
|
|
|
u_char book_version:4;
|
|
|
|
|
u_char book_type:4;
|
|
|
|
|
u_char min_rate:4;
|
|
|
|
|
u_char disc_size:4;
|
|
|
|
|
u_char layer_type:4;
|
|
|
|
|
u_char track_path:1;
|
|
|
|
|
u_char nlayers:2;
|
|
|
|
|
u_char track_density:4;
|
|
|
|
|
u_char linear_density:4;
|
|
|
|
|
u_char bca:1;
|
|
|
|
|
u_int32_t start_sector;
|
|
|
|
|
u_int32_t end_sector;
|
|
|
|
|
u_int32_t end_sector_l0;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_physical {
|
|
|
|
|
u_char type;
|
|
|
|
|
u_char layer_num;
|
|
|
|
|
struct l_dvd_layer layer[4];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_copyright {
|
|
|
|
|
u_char type;
|
|
|
|
|
u_char layer_num;
|
|
|
|
|
u_char cpst;
|
|
|
|
|
u_char rmi;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_disckey {
|
|
|
|
|
u_char type;
|
|
|
|
|
l_uint agid:2;
|
|
|
|
|
u_char value[2048];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_bca {
|
|
|
|
|
u_char type;
|
|
|
|
|
l_int len;
|
|
|
|
|
u_char value[188];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_manufact {
|
|
|
|
|
u_char type;
|
|
|
|
|
u_char layer_num;
|
|
|
|
|
l_int len;
|
|
|
|
|
u_char value[2048];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
typedef union {
|
|
|
|
|
u_char type;
|
|
|
|
|
struct l_dvd_physical physical;
|
|
|
|
|
struct l_dvd_copyright copyright;
|
|
|
|
|
struct l_dvd_disckey disckey;
|
|
|
|
|
struct l_dvd_bca bca;
|
|
|
|
|
struct l_dvd_manufact manufact;
|
|
|
|
|
} l_dvd_struct;
|
|
|
|
|
|
|
|
|
|
typedef u_char l_dvd_key[5];
|
|
|
|
|
typedef u_char l_dvd_challenge[10];
|
|
|
|
|
|
|
|
|
|
struct l_dvd_lu_send_agid {
|
|
|
|
|
u_char type;
|
|
|
|
|
l_uint agid:2;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_host_send_challenge {
|
|
|
|
|
u_char type;
|
|
|
|
|
l_uint agid:2;
|
|
|
|
|
l_dvd_challenge chal;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_send_key {
|
|
|
|
|
u_char type;
|
|
|
|
|
l_uint agid:2;
|
|
|
|
|
l_dvd_key key;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_lu_send_challenge {
|
|
|
|
|
u_char type;
|
|
|
|
|
l_uint agid:2;
|
|
|
|
|
l_dvd_challenge chal;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_lu_send_title_key {
|
|
|
|
|
u_char type;
|
|
|
|
|
l_uint agid:2;
|
|
|
|
|
l_dvd_key title_key;
|
|
|
|
|
l_int lba;
|
|
|
|
|
l_uint cpm:1;
|
|
|
|
|
l_uint cp_sec:1;
|
|
|
|
|
l_uint cgms:2;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_lu_send_asf {
|
|
|
|
|
u_char type;
|
|
|
|
|
l_uint agid:2;
|
|
|
|
|
l_uint asf:1;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_host_send_rpcstate {
|
|
|
|
|
u_char type;
|
|
|
|
|
u_char pdrc;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct l_dvd_lu_send_rpcstate {
|
|
|
|
|
u_char type:2;
|
|
|
|
|
u_char vra:3;
|
|
|
|
|
u_char ucca:3;
|
|
|
|
|
u_char region_mask;
|
|
|
|
|
u_char rpc_scheme;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
typedef union {
|
|
|
|
|
u_char type;
|
|
|
|
|
struct l_dvd_lu_send_agid lsa;
|
|
|
|
|
struct l_dvd_host_send_challenge hsc;
|
|
|
|
|
struct l_dvd_send_key lsk;
|
|
|
|
|
struct l_dvd_lu_send_challenge lsc;
|
|
|
|
|
struct l_dvd_send_key hsk;
|
|
|
|
|
struct l_dvd_lu_send_title_key lstk;
|
|
|
|
|
struct l_dvd_lu_send_asf lsasf;
|
|
|
|
|
struct l_dvd_host_send_rpcstate hrpcs;
|
|
|
|
|
struct l_dvd_lu_send_rpcstate lrpcs;
|
|
|
|
|
} l_dvd_authinfo;
|
|
|
|
|
|
1999-04-29 04:37:57 +00:00
|
|
|
|
static void
|
1999-12-04 11:10:22 +00:00
|
|
|
|
bsd_to_linux_msf_lba(u_char af, union msf_lba *bp, union linux_cdrom_addr *lp)
|
1999-04-29 04:37:57 +00:00
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (af == CD_LBA_FORMAT)
|
|
|
|
|
lp->lba = bp->lba;
|
|
|
|
|
else {
|
|
|
|
|
lp->msf.minute = bp->msf.minute;
|
|
|
|
|
lp->msf.second = bp->msf.second;
|
|
|
|
|
lp->msf.frame = bp->msf.frame;
|
|
|
|
|
}
|
1999-04-29 04:37:57 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
1999-12-04 11:10:22 +00:00
|
|
|
|
set_linux_cdrom_addr(union linux_cdrom_addr *addr, int format, int lba)
|
1999-04-29 04:37:57 +00:00
|
|
|
|
{
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (format == LINUX_CDROM_MSF) {
|
|
|
|
|
addr->msf.frame = lba % 75;
|
|
|
|
|
lba /= 75;
|
|
|
|
|
lba += 2;
|
|
|
|
|
addr->msf.second = lba % 60;
|
|
|
|
|
addr->msf.minute = lba / 60;
|
|
|
|
|
} else
|
|
|
|
|
addr->lba = lba;
|
1999-04-29 04:37:57 +00:00
|
|
|
|
}
|
|
|
|
|
|
2001-11-18 18:49:07 +00:00
|
|
|
|
static int
|
|
|
|
|
linux_to_bsd_dvd_struct(l_dvd_struct *lp, struct dvd_struct *bp)
|
|
|
|
|
{
|
|
|
|
|
bp->format = lp->type;
|
|
|
|
|
switch (bp->format) {
|
|
|
|
|
case DVD_STRUCT_PHYSICAL:
|
|
|
|
|
if (bp->layer_num >= 4)
|
|
|
|
|
return (EINVAL);
|
|
|
|
|
bp->layer_num = lp->physical.layer_num;
|
|
|
|
|
break;
|
|
|
|
|
case DVD_STRUCT_COPYRIGHT:
|
|
|
|
|
bp->layer_num = lp->copyright.layer_num;
|
|
|
|
|
break;
|
|
|
|
|
case DVD_STRUCT_DISCKEY:
|
|
|
|
|
bp->agid = lp->disckey.agid;
|
|
|
|
|
break;
|
|
|
|
|
case DVD_STRUCT_BCA:
|
|
|
|
|
case DVD_STRUCT_MANUFACT:
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|
|
|
|
|
return (0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
bsd_to_linux_dvd_struct(struct dvd_struct *bp, l_dvd_struct *lp)
|
|
|
|
|
{
|
|
|
|
|
switch (bp->format) {
|
|
|
|
|
case DVD_STRUCT_PHYSICAL: {
|
|
|
|
|
struct dvd_layer *blp = (struct dvd_layer *)bp->data;
|
|
|
|
|
struct l_dvd_layer *llp = &lp->physical.layer[bp->layer_num];
|
|
|
|
|
memset(llp, 0, sizeof(*llp));
|
|
|
|
|
llp->book_version = blp->book_version;
|
|
|
|
|
llp->book_type = blp->book_type;
|
|
|
|
|
llp->min_rate = blp->max_rate;
|
|
|
|
|
llp->disc_size = blp->disc_size;
|
|
|
|
|
llp->layer_type = blp->layer_type;
|
|
|
|
|
llp->track_path = blp->track_path;
|
|
|
|
|
llp->nlayers = blp->nlayers;
|
|
|
|
|
llp->track_density = blp->track_density;
|
|
|
|
|
llp->linear_density = blp->linear_density;
|
|
|
|
|
llp->bca = blp->bca;
|
|
|
|
|
llp->start_sector = blp->start_sector;
|
|
|
|
|
llp->end_sector = blp->end_sector;
|
|
|
|
|
llp->end_sector_l0 = blp->end_sector_l0;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case DVD_STRUCT_COPYRIGHT:
|
|
|
|
|
lp->copyright.cpst = bp->cpst;
|
|
|
|
|
lp->copyright.rmi = bp->rmi;
|
|
|
|
|
break;
|
|
|
|
|
case DVD_STRUCT_DISCKEY:
|
|
|
|
|
memcpy(lp->disckey.value, bp->data, sizeof(lp->disckey.value));
|
|
|
|
|
break;
|
|
|
|
|
case DVD_STRUCT_BCA:
|
|
|
|
|
lp->bca.len = bp->length;
|
|
|
|
|
memcpy(lp->bca.value, bp->data, sizeof(lp->bca.value));
|
|
|
|
|
break;
|
|
|
|
|
case DVD_STRUCT_MANUFACT:
|
|
|
|
|
lp->manufact.len = bp->length;
|
|
|
|
|
memcpy(lp->manufact.value, bp->data,
|
|
|
|
|
sizeof(lp->manufact.value));
|
|
|
|
|
/* lp->manufact.layer_num is unused in linux (redhat 7.0) */
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|
|
|
|
|
return (0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
linux_to_bsd_dvd_authinfo(l_dvd_authinfo *lp, int *bcode,
|
|
|
|
|
struct dvd_authinfo *bp)
|
|
|
|
|
{
|
|
|
|
|
switch (lp->type) {
|
|
|
|
|
case LINUX_DVD_LU_SEND_AGID:
|
|
|
|
|
*bcode = DVDIOCREPORTKEY;
|
|
|
|
|
bp->format = DVD_REPORT_AGID;
|
|
|
|
|
bp->agid = lp->lsa.agid;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_HOST_SEND_CHALLENGE:
|
|
|
|
|
*bcode = DVDIOCSENDKEY;
|
|
|
|
|
bp->format = DVD_SEND_CHALLENGE;
|
|
|
|
|
bp->agid = lp->hsc.agid;
|
|
|
|
|
memcpy(bp->keychal, lp->hsc.chal, 10);
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_LU_SEND_KEY1:
|
|
|
|
|
*bcode = DVDIOCREPORTKEY;
|
|
|
|
|
bp->format = DVD_REPORT_KEY1;
|
|
|
|
|
bp->agid = lp->lsk.agid;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_LU_SEND_CHALLENGE:
|
|
|
|
|
*bcode = DVDIOCREPORTKEY;
|
|
|
|
|
bp->format = DVD_REPORT_CHALLENGE;
|
|
|
|
|
bp->agid = lp->lsc.agid;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_HOST_SEND_KEY2:
|
|
|
|
|
*bcode = DVDIOCSENDKEY;
|
|
|
|
|
bp->format = DVD_SEND_KEY2;
|
|
|
|
|
bp->agid = lp->hsk.agid;
|
|
|
|
|
memcpy(bp->keychal, lp->hsk.key, 5);
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_LU_SEND_TITLE_KEY:
|
|
|
|
|
*bcode = DVDIOCREPORTKEY;
|
|
|
|
|
bp->format = DVD_REPORT_TITLE_KEY;
|
|
|
|
|
bp->agid = lp->lstk.agid;
|
|
|
|
|
bp->lba = lp->lstk.lba;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_LU_SEND_ASF:
|
|
|
|
|
*bcode = DVDIOCREPORTKEY;
|
|
|
|
|
bp->format = DVD_REPORT_ASF;
|
|
|
|
|
bp->agid = lp->lsasf.agid;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_INVALIDATE_AGID:
|
|
|
|
|
*bcode = DVDIOCREPORTKEY;
|
|
|
|
|
bp->format = DVD_INVALIDATE_AGID;
|
|
|
|
|
bp->agid = lp->lsa.agid;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_LU_SEND_RPC_STATE:
|
|
|
|
|
*bcode = DVDIOCREPORTKEY;
|
|
|
|
|
bp->format = DVD_REPORT_RPC;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_HOST_SEND_RPC_STATE:
|
|
|
|
|
*bcode = DVDIOCSENDKEY;
|
|
|
|
|
bp->format = DVD_SEND_RPC;
|
|
|
|
|
bp->region = lp->hrpcs.pdrc;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|
|
|
|
|
return (0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
bsd_to_linux_dvd_authinfo(struct dvd_authinfo *bp, l_dvd_authinfo *lp)
|
|
|
|
|
{
|
|
|
|
|
switch (lp->type) {
|
|
|
|
|
case LINUX_DVD_LU_SEND_AGID:
|
|
|
|
|
lp->lsa.agid = bp->agid;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_HOST_SEND_CHALLENGE:
|
|
|
|
|
lp->type = LINUX_DVD_LU_SEND_KEY1;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_LU_SEND_KEY1:
|
|
|
|
|
memcpy(lp->lsk.key, bp->keychal, sizeof(lp->lsk.key));
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_LU_SEND_CHALLENGE:
|
|
|
|
|
memcpy(lp->lsc.chal, bp->keychal, sizeof(lp->lsc.chal));
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_HOST_SEND_KEY2:
|
|
|
|
|
lp->type = LINUX_DVD_AUTH_ESTABLISHED;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_LU_SEND_TITLE_KEY:
|
|
|
|
|
memcpy(lp->lstk.title_key, bp->keychal,
|
|
|
|
|
sizeof(lp->lstk.title_key));
|
|
|
|
|
lp->lstk.cpm = bp->cpm;
|
|
|
|
|
lp->lstk.cp_sec = bp->cp_sec;
|
|
|
|
|
lp->lstk.cgms = bp->cgms;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_LU_SEND_ASF:
|
|
|
|
|
lp->lsasf.asf = bp->asf;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_INVALIDATE_AGID:
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_LU_SEND_RPC_STATE:
|
|
|
|
|
lp->lrpcs.type = bp->reg_type;
|
|
|
|
|
lp->lrpcs.vra = bp->vend_rsts;
|
|
|
|
|
lp->lrpcs.ucca = bp->user_rsts;
|
|
|
|
|
lp->lrpcs.region_mask = bp->region;
|
|
|
|
|
lp->lrpcs.rpc_scheme = bp->rpc_scheme;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_DVD_HOST_SEND_RPC_STATE:
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|
|
|
|
|
return (0);
|
|
|
|
|
}
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
static int
|
2001-09-12 08:38:13 +00:00
|
|
|
|
linux_ioctl_cdrom(struct thread *td, struct linux_ioctl_args *args)
|
1999-08-13 14:44:13 +00:00
|
|
|
|
{
|
2002-01-13 11:58:06 +00:00
|
|
|
|
struct file *fp;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
int error;
|
|
|
|
|
|
2002-01-14 00:13:45 +00:00
|
|
|
|
if ((error = fget(td, args->fd, &fp)) != 0)
|
|
|
|
|
return (error);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
switch (args->cmd & 0xffff) {
|
|
|
|
|
|
|
|
|
|
case LINUX_CDROMPAUSE:
|
|
|
|
|
args->cmd = CDIOCPAUSE;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_CDROMRESUME:
|
|
|
|
|
args->cmd = CDIOCRESUME;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_CDROMPLAYMSF:
|
|
|
|
|
args->cmd = CDIOCPLAYMSF;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_CDROMPLAYTRKIND:
|
|
|
|
|
args->cmd = CDIOCPLAYTRACKS;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_CDROMREADTOCHDR: {
|
|
|
|
|
struct ioc_toc_header th;
|
|
|
|
|
struct linux_cdrom_tochdr lth;
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = fo_ioctl(fp, CDIOREADTOCHEADER, (caddr_t)&th,
|
|
|
|
|
td->td_ucred, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (!error) {
|
|
|
|
|
lth.cdth_trk0 = th.starting_track;
|
|
|
|
|
lth.cdth_trk1 = th.ending_track;
|
2003-03-03 09:14:26 +00:00
|
|
|
|
copyout(<h, (void *)args->arg, sizeof(lth));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case LINUX_CDROMREADTOCENTRY: {
|
2004-06-18 13:36:30 +00:00
|
|
|
|
struct linux_cdrom_tocentry lte;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
struct ioc_read_toc_single_entry irtse;
|
2004-06-18 13:36:30 +00:00
|
|
|
|
|
|
|
|
|
error = copyin((void *)args->arg, <e, sizeof(lte));
|
|
|
|
|
if (error)
|
|
|
|
|
break;
|
|
|
|
|
irtse.address_format = lte.cdte_format;
|
|
|
|
|
irtse.track = lte.cdte_track;
|
|
|
|
|
error = fo_ioctl(fp, CDIOREADTOCENTRY, (caddr_t)&irtse,
|
|
|
|
|
td->td_ucred, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (!error) {
|
|
|
|
|
lte.cdte_ctrl = irtse.entry.control;
|
|
|
|
|
lte.cdte_adr = irtse.entry.addr_type;
|
|
|
|
|
bsd_to_linux_msf_lba(irtse.address_format,
|
|
|
|
|
&irtse.entry.addr, <e.cdte_addr);
|
2004-06-18 13:36:30 +00:00
|
|
|
|
error = copyout(<e, (void *)args->arg, sizeof(lte));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
case LINUX_CDROMSTOP:
|
|
|
|
|
args->cmd = CDIOCSTOP;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_CDROMSTART:
|
|
|
|
|
args->cmd = CDIOCSTART;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_CDROMEJECT:
|
|
|
|
|
args->cmd = CDIOCEJECT;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
/* LINUX_CDROMVOLCTRL */
|
|
|
|
|
|
|
|
|
|
case LINUX_CDROMSUBCHNL: {
|
|
|
|
|
struct linux_cdrom_subchnl sc;
|
|
|
|
|
struct ioc_read_subchannel bsdsc;
|
2005-01-30 08:12:37 +00:00
|
|
|
|
struct cd_sub_channel_info bsdinfo;
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
bsdsc.address_format = CD_LBA_FORMAT;
|
|
|
|
|
bsdsc.data_format = CD_CURRENT_POSITION;
|
2000-02-01 16:20:24 +00:00
|
|
|
|
bsdsc.track = 0;
|
2005-01-30 08:12:37 +00:00
|
|
|
|
bsdsc.data_len = sizeof(bsdinfo);
|
|
|
|
|
bsdsc.data = &bsdinfo;
|
|
|
|
|
error = fo_ioctl(fp, CDIOCREADSUBCHANNEL_SYSSPACE,
|
|
|
|
|
(caddr_t)&bsdsc, td->td_ucred, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin((void *)args->arg, &sc, sizeof(sc));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
2005-01-30 08:12:37 +00:00
|
|
|
|
sc.cdsc_audiostatus = bsdinfo.header.audio_status;
|
|
|
|
|
sc.cdsc_adr = bsdinfo.what.position.addr_type;
|
|
|
|
|
sc.cdsc_ctrl = bsdinfo.what.position.control;
|
|
|
|
|
sc.cdsc_trk = bsdinfo.what.position.track_number;
|
|
|
|
|
sc.cdsc_ind = bsdinfo.what.position.index_number;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
set_linux_cdrom_addr(&sc.cdsc_absaddr, sc.cdsc_format,
|
2005-01-30 08:12:37 +00:00
|
|
|
|
bsdinfo.what.position.absaddr.lba);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
set_linux_cdrom_addr(&sc.cdsc_reladdr, sc.cdsc_format,
|
2005-01-30 08:12:37 +00:00
|
|
|
|
bsdinfo.what.position.reladdr.lba);
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyout(&sc, (void *)args->arg, sizeof(sc));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* LINUX_CDROMREADMODE2 */
|
|
|
|
|
/* LINUX_CDROMREADMODE1 */
|
2003-10-20 09:51:00 +00:00
|
|
|
|
/* LINUX_CDROMREADAUDIO */
|
1999-12-04 11:10:22 +00:00
|
|
|
|
/* LINUX_CDROMEJECT_SW */
|
|
|
|
|
/* LINUX_CDROMMULTISESSION */
|
|
|
|
|
/* LINUX_CDROM_GET_UPC */
|
|
|
|
|
|
|
|
|
|
case LINUX_CDROMRESET:
|
|
|
|
|
args->cmd = CDIOCRESET;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
/* LINUX_CDROMVOLREAD */
|
|
|
|
|
/* LINUX_CDROMREADRAW */
|
|
|
|
|
/* LINUX_CDROMREADCOOKED */
|
|
|
|
|
/* LINUX_CDROMSEEK */
|
|
|
|
|
/* LINUX_CDROMPLAYBLK */
|
|
|
|
|
/* LINUX_CDROMREADALL */
|
|
|
|
|
/* LINUX_CDROMCLOSETRAY */
|
|
|
|
|
/* LINUX_CDROMLOADFROMSLOT */
|
2001-11-18 18:49:07 +00:00
|
|
|
|
/* LINUX_CDROMGETSPINDOWN */
|
|
|
|
|
/* LINUX_CDROMSETSPINDOWN */
|
|
|
|
|
/* LINUX_CDROM_SET_OPTIONS */
|
|
|
|
|
/* LINUX_CDROM_CLEAR_OPTIONS */
|
|
|
|
|
/* LINUX_CDROM_SELECT_SPEED */
|
|
|
|
|
/* LINUX_CDROM_SELECT_DISC */
|
|
|
|
|
/* LINUX_CDROM_MEDIA_CHANGED */
|
|
|
|
|
/* LINUX_CDROM_DRIVE_STATUS */
|
|
|
|
|
/* LINUX_CDROM_DISC_STATUS */
|
|
|
|
|
/* LINUX_CDROM_CHANGER_NSLOTS */
|
|
|
|
|
/* LINUX_CDROM_LOCKDOOR */
|
|
|
|
|
/* LINUX_CDROM_DEBUG */
|
|
|
|
|
/* LINUX_CDROM_GET_CAPABILITY */
|
|
|
|
|
/* LINUX_CDROMAUDIOBUFSIZ */
|
|
|
|
|
|
|
|
|
|
case LINUX_DVD_READ_STRUCT: {
|
2009-05-27 15:23:12 +00:00
|
|
|
|
l_dvd_struct *lds;
|
|
|
|
|
struct dvd_struct *bds;
|
2001-11-18 18:49:07 +00:00
|
|
|
|
|
2009-05-27 15:23:12 +00:00
|
|
|
|
lds = malloc(sizeof(*lds), M_LINUX, M_WAITOK);
|
|
|
|
|
bds = malloc(sizeof(*bds), M_LINUX, M_WAITOK);
|
|
|
|
|
error = copyin((void *)args->arg, lds, sizeof(*lds));
|
2001-11-18 18:49:07 +00:00
|
|
|
|
if (error)
|
2009-05-27 15:23:12 +00:00
|
|
|
|
goto out;
|
|
|
|
|
error = linux_to_bsd_dvd_struct(lds, bds);
|
2001-11-18 18:49:07 +00:00
|
|
|
|
if (error)
|
2009-05-27 15:23:12 +00:00
|
|
|
|
goto out;
|
|
|
|
|
error = fo_ioctl(fp, DVDIOCREADSTRUCTURE, (caddr_t)bds,
|
2002-08-17 02:36:16 +00:00
|
|
|
|
td->td_ucred, td);
|
2001-11-18 18:49:07 +00:00
|
|
|
|
if (error)
|
2009-05-27 15:23:12 +00:00
|
|
|
|
goto out;
|
|
|
|
|
error = bsd_to_linux_dvd_struct(bds, lds);
|
2001-11-18 18:49:07 +00:00
|
|
|
|
if (error)
|
2009-05-27 15:23:12 +00:00
|
|
|
|
goto out;
|
|
|
|
|
error = copyout(lds, (void *)args->arg, sizeof(*lds));
|
|
|
|
|
out:
|
|
|
|
|
free(bds, M_LINUX);
|
|
|
|
|
free(lds, M_LINUX);
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
2001-11-18 18:49:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* LINUX_DVD_WRITE_STRUCT */
|
|
|
|
|
|
|
|
|
|
case LINUX_DVD_AUTH: {
|
|
|
|
|
l_dvd_authinfo lda;
|
|
|
|
|
struct dvd_authinfo bda;
|
|
|
|
|
int bcode;
|
|
|
|
|
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin((void *)args->arg, &lda, sizeof(lda));
|
2001-11-18 18:49:07 +00:00
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
2001-11-18 18:49:07 +00:00
|
|
|
|
error = linux_to_bsd_dvd_authinfo(&lda, &bcode, &bda);
|
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = fo_ioctl(fp, bcode, (caddr_t)&bda, td->td_ucred,
|
|
|
|
|
td);
|
2001-11-18 18:49:07 +00:00
|
|
|
|
if (error) {
|
|
|
|
|
if (lda.type == LINUX_DVD_HOST_SEND_KEY2) {
|
|
|
|
|
lda.type = LINUX_DVD_AUTH_FAILURE;
|
2003-03-03 09:14:26 +00:00
|
|
|
|
copyout(&lda, (void *)args->arg, sizeof(lda));
|
2001-11-18 18:49:07 +00:00
|
|
|
|
}
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
2001-11-18 18:49:07 +00:00
|
|
|
|
}
|
|
|
|
|
error = bsd_to_linux_dvd_authinfo(&bda, &lda);
|
|
|
|
|
if (error)
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyout(&lda, (void *)args->arg, sizeof(lda));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
2001-11-18 18:49:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
2007-04-07 19:40:58 +00:00
|
|
|
|
case LINUX_SCSI_GET_BUS_NUMBER:
|
|
|
|
|
case LINUX_SCSI_GET_IDLUN:
|
|
|
|
|
error = linux_ioctl_sg(td, args);
|
|
|
|
|
break;
|
|
|
|
|
|
2001-11-18 18:49:07 +00:00
|
|
|
|
/* LINUX_CDROM_SEND_PACKET */
|
|
|
|
|
/* LINUX_CDROM_NEXT_WRITABLE */
|
|
|
|
|
/* LINUX_CDROM_LAST_WRITTEN */
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
2002-01-13 11:58:06 +00:00
|
|
|
|
default:
|
|
|
|
|
error = ENOIOCTL;
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
|
|
|
|
return (error);
|
1999-08-13 14:44:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
2002-10-11 11:43:09 +00:00
|
|
|
|
static int
|
|
|
|
|
linux_ioctl_vfat(struct thread *td, struct linux_ioctl_args *args)
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
return (ENOTTY);
|
|
|
|
|
}
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
/*
|
|
|
|
|
* Sound related ioctls
|
|
|
|
|
*/
|
|
|
|
|
|
2005-04-13 04:33:06 +00:00
|
|
|
|
struct linux_mixer_info {
|
|
|
|
|
char id[16];
|
|
|
|
|
char name[32];
|
|
|
|
|
int modify_counter;
|
|
|
|
|
int fillers[10];
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct linux_old_mixer_info {
|
|
|
|
|
char id[16];
|
|
|
|
|
char name[32];
|
|
|
|
|
};
|
|
|
|
|
|
2000-11-01 19:48:35 +00:00
|
|
|
|
static u_int32_t dirbits[4] = { IOC_VOID, IOC_IN, IOC_OUT, IOC_INOUT };
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
#define SETDIR(c) (((c) & ~IOC_DIRMASK) | dirbits[args->cmd >> 30])
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
static int
|
2001-09-12 08:38:13 +00:00
|
|
|
|
linux_ioctl_sound(struct thread *td, struct linux_ioctl_args *args)
|
1995-06-25 17:32:43 +00:00
|
|
|
|
{
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
switch (args->cmd & 0xffff) {
|
1997-06-02 06:31:49 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_VOLUME:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_VOLUME);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1997-06-02 06:31:49 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_BASS:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_BASS);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1997-06-02 06:31:49 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_TREBLE:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_TREBLE);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_SYNTH:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_SYNTH);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_PCM:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_PCM);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_SPEAKER:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_SPEAKER);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_LINE:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_LINE);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_MIC:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_MIC);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-06-25 17:32:43 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_CD:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_CD);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_IMIX:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_IMIX);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_ALTPCM:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_ALTPCM);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_RECLEV:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_RECLEV);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_IGAIN:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_IGAIN);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_OGAIN:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_OGAIN);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_LINE1:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_LINE1);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_LINE2:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_LINE2);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_LINE3:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_LINE3);
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
2005-04-13 04:33:06 +00:00
|
|
|
|
case LINUX_SOUND_MIXER_INFO: {
|
|
|
|
|
/* Key on encoded length */
|
|
|
|
|
switch ((args->cmd >> 16) & 0x1fff) {
|
|
|
|
|
case 0x005c: { /* SOUND_MIXER_INFO */
|
|
|
|
|
struct linux_mixer_info info;
|
|
|
|
|
bzero(&info, sizeof(info));
|
|
|
|
|
strncpy(info.id, "OSS", sizeof(info.id) - 1);
|
|
|
|
|
strncpy(info.name, "FreeBSD OSS Mixer", sizeof(info.name) - 1);
|
|
|
|
|
copyout(&info, (void *)args->arg, sizeof(info));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
case 0x0030: { /* SOUND_OLD_MIXER_INFO */
|
|
|
|
|
struct linux_old_mixer_info info;
|
|
|
|
|
bzero(&info, sizeof(info));
|
|
|
|
|
strncpy(info.id, "OSS", sizeof(info.id) - 1);
|
|
|
|
|
strncpy(info.name, "FreeBSD OSS Mixer", sizeof(info.name) - 1);
|
|
|
|
|
copyout(&info, (void *)args->arg, sizeof(info));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
default:
|
|
|
|
|
return (ENOIOCTL);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
}
|
2004-06-18 14:25:44 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_OSS_GETVERSION: {
|
2003-03-13 22:45:43 +00:00
|
|
|
|
int version = linux_get_oss_version(td);
|
2003-03-03 09:14:26 +00:00
|
|
|
|
return (copyout(&version, (void *)args->arg, sizeof(int)));
|
1995-06-25 17:32:43 +00:00
|
|
|
|
}
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
2002-10-11 11:43:09 +00:00
|
|
|
|
case LINUX_SOUND_MIXER_READ_STEREODEVS:
|
|
|
|
|
args->cmd = SOUND_MIXER_READ_STEREODEVS;
|
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
|
2004-06-18 14:36:24 +00:00
|
|
|
|
case LINUX_SOUND_MIXER_READ_RECMASK:
|
|
|
|
|
args->cmd = SOUND_MIXER_READ_RECMASK;
|
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SOUND_MIXER_READ_DEVMASK:
|
|
|
|
|
args->cmd = SOUND_MIXER_READ_DEVMASK;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
2001-11-18 06:36:18 +00:00
|
|
|
|
case LINUX_SOUND_MIXER_WRITE_RECSRC:
|
|
|
|
|
args->cmd = SETDIR(SOUND_MIXER_WRITE_RECSRC);
|
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_RESET:
|
|
|
|
|
args->cmd = SNDCTL_DSP_RESET;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_SYNC:
|
|
|
|
|
args->cmd = SNDCTL_DSP_SYNC;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_SPEED:
|
|
|
|
|
args->cmd = SNDCTL_DSP_SPEED;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_STEREO:
|
|
|
|
|
args->cmd = SNDCTL_DSP_STEREO;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_GETBLKSIZE: /* LINUX_SNDCTL_DSP_SETBLKSIZE */
|
|
|
|
|
args->cmd = SNDCTL_DSP_GETBLKSIZE;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_SETFMT:
|
|
|
|
|
args->cmd = SNDCTL_DSP_SETFMT;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SOUND_PCM_WRITE_CHANNELS:
|
|
|
|
|
args->cmd = SOUND_PCM_WRITE_CHANNELS;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SOUND_PCM_WRITE_FILTER:
|
|
|
|
|
args->cmd = SOUND_PCM_WRITE_FILTER;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_POST:
|
|
|
|
|
args->cmd = SNDCTL_DSP_POST;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_SUBDIVIDE:
|
|
|
|
|
args->cmd = SNDCTL_DSP_SUBDIVIDE;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_SETFRAGMENT:
|
|
|
|
|
args->cmd = SNDCTL_DSP_SETFRAGMENT;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_GETFMTS:
|
|
|
|
|
args->cmd = SNDCTL_DSP_GETFMTS;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_GETOSPACE:
|
|
|
|
|
args->cmd = SNDCTL_DSP_GETOSPACE;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_GETISPACE:
|
|
|
|
|
args->cmd = SNDCTL_DSP_GETISPACE;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1998-11-12 00:42:08 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_NONBLOCK:
|
|
|
|
|
args->cmd = SNDCTL_DSP_NONBLOCK;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1997-11-17 04:00:32 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_GETCAPS:
|
|
|
|
|
args->cmd = SNDCTL_DSP_GETCAPS;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1997-11-17 04:00:32 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_SETTRIGGER: /* LINUX_SNDCTL_GETTRIGGER */
|
|
|
|
|
args->cmd = SNDCTL_DSP_SETTRIGGER;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1997-11-17 04:00:32 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_GETIPTR:
|
|
|
|
|
args->cmd = SNDCTL_DSP_GETIPTR;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1997-11-17 04:00:32 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_GETOPTR:
|
|
|
|
|
args->cmd = SNDCTL_DSP_GETOPTR;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
2004-07-02 15:31:44 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_SETDUPLEX:
|
|
|
|
|
args->cmd = SNDCTL_DSP_SETDUPLEX;
|
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_DSP_GETODELAY:
|
|
|
|
|
args->cmd = SNDCTL_DSP_GETODELAY;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_RESET:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_RESET;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_SYNC:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_SYNC;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SYNTH_INFO:
|
|
|
|
|
args->cmd = SNDCTL_SYNTH_INFO;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_CTRLRATE:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_CTRLRATE;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-11-29 23:03:34 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_GETOUTCOUNT:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_GETOUTCOUNT;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_GETINCOUNT:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_GETINCOUNT;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_PERCMODE:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_PERCMODE;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_FM_LOAD_INSTR:
|
|
|
|
|
args->cmd = SNDCTL_FM_LOAD_INSTR;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_TESTMIDI:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_TESTMIDI;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1995-12-29 22:12:14 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_RESETSAMPLES:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_RESETSAMPLES;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1996-11-05 02:04:37 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_NRSYNTHS:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_NRSYNTHS;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1996-11-05 02:04:37 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_NRMIDIS:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_NRMIDIS;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1996-11-05 02:04:37 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_MIDI_INFO:
|
|
|
|
|
args->cmd = SNDCTL_MIDI_INFO;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1996-11-05 02:04:37 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SEQ_TRESHOLD:
|
|
|
|
|
args->cmd = SNDCTL_SEQ_TRESHOLD;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1996-11-05 02:04:37 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SNDCTL_SYNTH_MEMAVL:
|
|
|
|
|
args->cmd = SNDCTL_SYNTH_MEMAVL;
|
2001-09-12 08:38:13 +00:00
|
|
|
|
return (ioctl(td, (struct ioctl_args *)args));
|
1999-08-14 13:26:44 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
1999-08-14 13:26:44 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (ENOIOCTL);
|
|
|
|
|
}
|
1996-03-10 22:30:53 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
/*
|
|
|
|
|
* Console related ioctls
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
#define ISSIGVALID(sig) ((sig) > 0 && (sig) < NSIG)
|
|
|
|
|
|
|
|
|
|
static int
|
2001-09-12 08:38:13 +00:00
|
|
|
|
linux_ioctl_console(struct thread *td, struct linux_ioctl_args *args)
|
1999-12-04 11:10:22 +00:00
|
|
|
|
{
|
2002-01-13 11:58:06 +00:00
|
|
|
|
struct file *fp;
|
|
|
|
|
int error;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
2002-01-14 00:13:45 +00:00
|
|
|
|
if ((error = fget(td, args->fd, &fp)) != 0)
|
|
|
|
|
return (error);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
switch (args->cmd & 0xffff) {
|
|
|
|
|
|
|
|
|
|
case LINUX_KIOCSOUND:
|
|
|
|
|
args->cmd = KIOCSOUND;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_KDMKTONE:
|
|
|
|
|
args->cmd = KDMKTONE;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_KDGETLED:
|
|
|
|
|
args->cmd = KDGETLED;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_KDSETLED:
|
|
|
|
|
args->cmd = KDSETLED;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_KDSETMODE:
|
|
|
|
|
args->cmd = KDSETMODE;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_KDGETMODE:
|
|
|
|
|
args->cmd = KDGETMODE;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_KDGKBMODE:
|
|
|
|
|
args->cmd = KDGKBMODE;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_KDSKBMODE: {
|
|
|
|
|
int kbdmode;
|
|
|
|
|
switch (args->arg) {
|
|
|
|
|
case LINUX_KBD_RAW:
|
|
|
|
|
kbdmode = K_RAW;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_KBD_XLATE:
|
|
|
|
|
kbdmode = K_XLATE;
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_KBD_MEDIUMRAW:
|
|
|
|
|
kbdmode = K_RAW;
|
|
|
|
|
break;
|
|
|
|
|
default:
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|
2002-08-17 02:36:16 +00:00
|
|
|
|
error = (fo_ioctl(fp, KDSKBMODE, (caddr_t)&kbdmode,
|
|
|
|
|
td->td_ucred, td));
|
2002-01-13 11:58:06 +00:00
|
|
|
|
break;
|
1997-12-15 06:09:11 +00:00
|
|
|
|
}
|
1998-08-31 06:55:02 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_VT_OPENQRY:
|
|
|
|
|
args->cmd = VT_OPENQRY;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_VT_GETMODE:
|
|
|
|
|
args->cmd = VT_GETMODE;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_VT_SETMODE: {
|
2004-06-18 13:36:30 +00:00
|
|
|
|
struct vt_mode mode;
|
|
|
|
|
if ((error = copyin((void *)args->arg, &mode, sizeof(mode))))
|
|
|
|
|
break;
|
|
|
|
|
if (!ISSIGVALID(mode.frsig) && ISSIGVALID(mode.acqsig))
|
|
|
|
|
mode.frsig = mode.acqsig;
|
|
|
|
|
if ((error = copyout(&mode, (void *)args->arg, sizeof(mode))))
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
args->cmd = VT_SETMODE;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
1998-08-31 06:55:02 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_VT_GETSTATE:
|
|
|
|
|
args->cmd = VT_GETACTIVE;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1998-08-31 06:55:02 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_VT_RELDISP:
|
|
|
|
|
args->cmd = VT_RELDISP;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1998-08-31 06:55:02 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_VT_ACTIVATE:
|
|
|
|
|
args->cmd = VT_ACTIVATE;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1998-08-31 06:55:02 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_VT_WAITACTIVE:
|
|
|
|
|
args->cmd = VT_WAITACTIVE;
|
2002-01-13 11:58:06 +00:00
|
|
|
|
error = (ioctl(td, (struct ioctl_args *)args));
|
|
|
|
|
break;
|
1998-08-31 06:55:02 +00:00
|
|
|
|
|
2002-01-13 11:58:06 +00:00
|
|
|
|
default:
|
|
|
|
|
error = ENOIOCTL;
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
|
|
|
|
return (error);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
2001-11-19 15:43:50 +00:00
|
|
|
|
/*
|
|
|
|
|
* Criteria for interface name translation
|
|
|
|
|
*/
|
|
|
|
|
#define IFP_IS_ETH(ifp) (ifp->if_type == IFT_ETHER)
|
2001-10-20 00:01:26 +00:00
|
|
|
|
|
2002-01-15 03:10:33 +00:00
|
|
|
|
/*
|
|
|
|
|
* Interface function used by linprocfs (at the time of writing). It's not
|
|
|
|
|
* used by the Linuxulator itself.
|
|
|
|
|
*/
|
|
|
|
|
int
|
|
|
|
|
linux_ifname(struct ifnet *ifp, char *buffer, size_t buflen)
|
|
|
|
|
{
|
|
|
|
|
struct ifnet *ifscan;
|
|
|
|
|
int ethno;
|
|
|
|
|
|
2009-08-23 20:40:19 +00:00
|
|
|
|
IFNET_RLOCK_ASSERT();
|
|
|
|
|
|
2002-01-15 03:10:33 +00:00
|
|
|
|
/* Short-circuit non ethernet interfaces */
|
|
|
|
|
if (!IFP_IS_ETH(ifp))
|
2003-10-31 18:32:15 +00:00
|
|
|
|
return (strlcpy(buffer, ifp->if_xname, buflen));
|
2002-01-15 03:10:33 +00:00
|
|
|
|
|
|
|
|
|
/* Determine the (relative) unit number for ethernet interfaces */
|
|
|
|
|
ethno = 0;
|
Commit step 1 of the vimage project, (network stack)
virtualization work done by Marko Zec (zec@).
This is the first in a series of commits over the course
of the next few weeks.
Mark all uses of global variables to be virtualized
with a V_ prefix.
Use macros to map them back to their global names for
now, so this is a NOP change only.
We hope to have caught at least 85-90% of what is needed
so we do not invalidate a lot of outstanding patches again.
Obtained from: //depot/projects/vimage-commit2/...
Reviewed by: brooks, des, ed, mav, julian,
jamie, kris, rwatson, zec, ...
(various people I forgot, different versions)
md5 (with a bit of help)
Sponsored by: NLnet Foundation, The FreeBSD Foundation
X-MFC after: never
V_Commit_Message_Reviewed_By: more people than the patch
2008-08-17 23:27:27 +00:00
|
|
|
|
TAILQ_FOREACH(ifscan, &V_ifnet, if_link) {
|
2009-08-23 20:40:19 +00:00
|
|
|
|
if (ifscan == ifp)
|
2002-01-15 03:10:33 +00:00
|
|
|
|
return (snprintf(buffer, buflen, "eth%d", ethno));
|
|
|
|
|
if (IFP_IS_ETH(ifscan))
|
|
|
|
|
ethno++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (0);
|
|
|
|
|
}
|
|
|
|
|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
/*
|
|
|
|
|
* Translate a Linux interface name to a FreeBSD interface name,
|
|
|
|
|
* and return the associated ifnet structure
|
|
|
|
|
* bsdname and lxname need to be least IFNAMSIZ bytes long, but
|
|
|
|
|
* can point to the same buffer.
|
|
|
|
|
*/
|
2001-10-20 00:01:26 +00:00
|
|
|
|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
static struct ifnet *
|
2009-06-15 19:01:53 +00:00
|
|
|
|
ifname_linux_to_bsd(struct thread *td, const char *lxname, char *bsdname)
|
2001-10-15 20:52:17 +00:00
|
|
|
|
{
|
|
|
|
|
struct ifnet *ifp;
|
|
|
|
|
int len, unit;
|
|
|
|
|
char *ep;
|
2001-12-04 03:55:10 +00:00
|
|
|
|
int is_eth, index;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
|
|
|
|
|
for (len = 0; len < LINUX_IFNAMSIZ; ++len)
|
|
|
|
|
if (!isalpha(lxname[len]))
|
|
|
|
|
break;
|
|
|
|
|
if (len == 0 || len == LINUX_IFNAMSIZ)
|
|
|
|
|
return (NULL);
|
|
|
|
|
unit = (int)strtoul(lxname + len, &ep, 10);
|
|
|
|
|
if (ep == NULL || ep == lxname + len || ep >= lxname + LINUX_IFNAMSIZ)
|
|
|
|
|
return (NULL);
|
2001-12-04 03:55:10 +00:00
|
|
|
|
index = 0;
|
|
|
|
|
is_eth = (len == 3 && !strncmp(lxname, "eth", len)) ? 1 : 0;
|
2009-08-28 22:51:07 +00:00
|
|
|
|
CURVNET_SET(TD_TO_VNET(td));
|
2002-12-22 05:35:03 +00:00
|
|
|
|
IFNET_RLOCK();
|
Commit step 1 of the vimage project, (network stack)
virtualization work done by Marko Zec (zec@).
This is the first in a series of commits over the course
of the next few weeks.
Mark all uses of global variables to be virtualized
with a V_ prefix.
Use macros to map them back to their global names for
now, so this is a NOP change only.
We hope to have caught at least 85-90% of what is needed
so we do not invalidate a lot of outstanding patches again.
Obtained from: //depot/projects/vimage-commit2/...
Reviewed by: brooks, des, ed, mav, julian,
jamie, kris, rwatson, zec, ...
(various people I forgot, different versions)
md5 (with a bit of help)
Sponsored by: NLnet Foundation, The FreeBSD Foundation
X-MFC after: never
V_Commit_Message_Reviewed_By: more people than the patch
2008-08-17 23:27:27 +00:00
|
|
|
|
TAILQ_FOREACH(ifp, &V_ifnet, if_link) {
|
2001-12-04 03:55:10 +00:00
|
|
|
|
/*
|
|
|
|
|
* Allow Linux programs to use FreeBSD names. Don't presume
|
|
|
|
|
* we never have an interface named "eth", so don't make
|
|
|
|
|
* the test optional based on is_eth.
|
|
|
|
|
*/
|
2003-10-31 18:32:15 +00:00
|
|
|
|
if (strncmp(ifp->if_xname, lxname, LINUX_IFNAMSIZ) == 0)
|
2001-10-15 20:52:17 +00:00
|
|
|
|
break;
|
2001-12-04 03:55:10 +00:00
|
|
|
|
if (is_eth && IFP_IS_ETH(ifp) && unit == index++)
|
2001-10-15 20:52:17 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
2002-12-22 05:35:03 +00:00
|
|
|
|
IFNET_RUNLOCK();
|
2009-08-28 22:51:07 +00:00
|
|
|
|
CURVNET_RESTORE();
|
2001-10-15 20:52:17 +00:00
|
|
|
|
if (ifp != NULL)
|
2003-10-31 18:32:15 +00:00
|
|
|
|
strlcpy(bsdname, ifp->if_xname, IFNAMSIZ);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
return (ifp);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Implement the SIOCGIFCONF ioctl
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
linux_ifconf(struct thread *td, struct ifconf *uifc)
|
|
|
|
|
{
|
2005-01-14 04:44:56 +00:00
|
|
|
|
#ifdef COMPAT_LINUX32
|
2004-08-16 07:28:16 +00:00
|
|
|
|
struct l_ifconf ifc;
|
|
|
|
|
#else
|
2001-10-15 20:52:17 +00:00
|
|
|
|
struct ifconf ifc;
|
2004-08-16 07:28:16 +00:00
|
|
|
|
#endif
|
2001-10-15 20:52:17 +00:00
|
|
|
|
struct l_ifreq ifr;
|
|
|
|
|
struct ifnet *ifp;
|
2002-01-29 06:00:11 +00:00
|
|
|
|
struct ifaddr *ifa;
|
2005-08-27 14:44:10 +00:00
|
|
|
|
struct sbuf *sb;
|
|
|
|
|
int error, ethno, full = 0, valid_len, max_len;
|
2002-01-10 05:36:36 +00:00
|
|
|
|
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin(uifc, &ifc, sizeof(ifc));
|
2001-10-15 20:52:17 +00:00
|
|
|
|
if (error != 0)
|
|
|
|
|
return (error);
|
|
|
|
|
|
2005-08-27 14:44:10 +00:00
|
|
|
|
max_len = MAXPHYS - 1;
|
|
|
|
|
|
2009-08-28 22:51:07 +00:00
|
|
|
|
CURVNET_SET(TD_TO_VNET(td));
|
2004-06-18 14:06:46 +00:00
|
|
|
|
/* handle the 'request buffer size' case */
|
2004-08-16 07:28:16 +00:00
|
|
|
|
if (ifc.ifc_buf == PTROUT(NULL)) {
|
2004-06-18 14:06:46 +00:00
|
|
|
|
ifc.ifc_len = 0;
|
Commit step 1 of the vimage project, (network stack)
virtualization work done by Marko Zec (zec@).
This is the first in a series of commits over the course
of the next few weeks.
Mark all uses of global variables to be virtualized
with a V_ prefix.
Use macros to map them back to their global names for
now, so this is a NOP change only.
We hope to have caught at least 85-90% of what is needed
so we do not invalidate a lot of outstanding patches again.
Obtained from: //depot/projects/vimage-commit2/...
Reviewed by: brooks, des, ed, mav, julian,
jamie, kris, rwatson, zec, ...
(various people I forgot, different versions)
md5 (with a bit of help)
Sponsored by: NLnet Foundation, The FreeBSD Foundation
X-MFC after: never
V_Commit_Message_Reviewed_By: more people than the patch
2008-08-17 23:27:27 +00:00
|
|
|
|
TAILQ_FOREACH(ifp, &V_ifnet, if_link) {
|
2004-06-18 14:06:46 +00:00
|
|
|
|
TAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) {
|
|
|
|
|
struct sockaddr *sa = ifa->ifa_addr;
|
|
|
|
|
if (sa->sa_family == AF_INET)
|
|
|
|
|
ifc.ifc_len += sizeof(ifr);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
error = copyout(&ifc, uifc, sizeof(ifc));
|
2009-08-28 22:51:07 +00:00
|
|
|
|
CURVNET_RESTORE();
|
2004-06-18 14:06:46 +00:00
|
|
|
|
return (error);
|
|
|
|
|
}
|
|
|
|
|
|
2009-08-28 22:51:07 +00:00
|
|
|
|
if (ifc.ifc_len <= 0) {
|
|
|
|
|
CURVNET_RESTORE();
|
2005-08-27 14:44:10 +00:00
|
|
|
|
return (EINVAL);
|
2009-08-28 22:51:07 +00:00
|
|
|
|
}
|
2001-10-15 20:52:17 +00:00
|
|
|
|
|
2005-08-27 14:44:10 +00:00
|
|
|
|
again:
|
2002-01-10 05:36:36 +00:00
|
|
|
|
/* Keep track of eth interfaces */
|
|
|
|
|
ethno = 0;
|
2005-08-27 14:44:10 +00:00
|
|
|
|
if (ifc.ifc_len <= max_len) {
|
|
|
|
|
max_len = ifc.ifc_len;
|
|
|
|
|
full = 1;
|
|
|
|
|
}
|
|
|
|
|
sb = sbuf_new(NULL, NULL, max_len + 1, SBUF_FIXEDLEN);
|
|
|
|
|
max_len = 0;
|
|
|
|
|
valid_len = 0;
|
2002-01-10 05:36:36 +00:00
|
|
|
|
|
2002-01-29 06:00:11 +00:00
|
|
|
|
/* Return all AF_INET addresses of all interfaces */
|
2009-08-23 20:40:19 +00:00
|
|
|
|
IFNET_RLOCK();
|
Commit step 1 of the vimage project, (network stack)
virtualization work done by Marko Zec (zec@).
This is the first in a series of commits over the course
of the next few weeks.
Mark all uses of global variables to be virtualized
with a V_ prefix.
Use macros to map them back to their global names for
now, so this is a NOP change only.
We hope to have caught at least 85-90% of what is needed
so we do not invalidate a lot of outstanding patches again.
Obtained from: //depot/projects/vimage-commit2/...
Reviewed by: brooks, des, ed, mav, julian,
jamie, kris, rwatson, zec, ...
(various people I forgot, different versions)
md5 (with a bit of help)
Sponsored by: NLnet Foundation, The FreeBSD Foundation
X-MFC after: never
V_Commit_Message_Reviewed_By: more people than the patch
2008-08-17 23:27:27 +00:00
|
|
|
|
TAILQ_FOREACH(ifp, &V_ifnet, if_link) {
|
2005-08-27 14:44:10 +00:00
|
|
|
|
int addrs = 0;
|
2002-01-29 06:00:11 +00:00
|
|
|
|
|
2003-03-03 09:14:26 +00:00
|
|
|
|
bzero(&ifr, sizeof(ifr));
|
2002-01-10 05:36:36 +00:00
|
|
|
|
if (IFP_IS_ETH(ifp))
|
|
|
|
|
snprintf(ifr.ifr_name, LINUX_IFNAMSIZ, "eth%d",
|
|
|
|
|
ethno++);
|
|
|
|
|
else
|
2003-10-31 18:32:15 +00:00
|
|
|
|
strlcpy(ifr.ifr_name, ifp->if_xname, LINUX_IFNAMSIZ);
|
2002-01-29 06:00:11 +00:00
|
|
|
|
|
|
|
|
|
/* Walk the address list */
|
|
|
|
|
TAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) {
|
|
|
|
|
struct sockaddr *sa = ifa->ifa_addr;
|
|
|
|
|
|
|
|
|
|
if (sa->sa_family == AF_INET) {
|
|
|
|
|
ifr.ifr_addr.sa_family = LINUX_AF_INET;
|
|
|
|
|
memcpy(ifr.ifr_addr.sa_data, sa->sa_data,
|
|
|
|
|
sizeof(ifr.ifr_addr.sa_data));
|
2005-08-27 14:44:10 +00:00
|
|
|
|
sbuf_bcat(sb, &ifr, sizeof(ifr));
|
|
|
|
|
max_len += sizeof(ifr);
|
|
|
|
|
addrs++;
|
2002-01-29 06:00:11 +00:00
|
|
|
|
}
|
2005-08-27 14:44:10 +00:00
|
|
|
|
|
|
|
|
|
if (!sbuf_overflowed(sb))
|
|
|
|
|
valid_len = sbuf_len(sb);
|
|
|
|
|
}
|
|
|
|
|
if (addrs == 0) {
|
|
|
|
|
bzero((caddr_t)&ifr.ifr_addr, sizeof(ifr.ifr_addr));
|
|
|
|
|
sbuf_bcat(sb, &ifr, sizeof(ifr));
|
|
|
|
|
max_len += sizeof(ifr);
|
|
|
|
|
|
|
|
|
|
if (!sbuf_overflowed(sb))
|
|
|
|
|
valid_len = sbuf_len(sb);
|
2002-01-29 06:00:11 +00:00
|
|
|
|
}
|
2001-10-15 20:52:17 +00:00
|
|
|
|
}
|
2002-12-22 05:35:03 +00:00
|
|
|
|
IFNET_RUNLOCK();
|
2001-10-15 20:52:17 +00:00
|
|
|
|
|
2005-08-27 14:44:10 +00:00
|
|
|
|
if (valid_len != max_len && !full) {
|
|
|
|
|
sbuf_delete(sb);
|
|
|
|
|
goto again;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ifc.ifc_len = valid_len;
|
|
|
|
|
sbuf_finish(sb);
|
2005-08-28 13:11:08 +00:00
|
|
|
|
memcpy(PTRIN(ifc.ifc_buf), sbuf_data(sb), ifc.ifc_len);
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyout(&ifc, uifc, sizeof(ifc));
|
2005-08-27 14:44:10 +00:00
|
|
|
|
sbuf_delete(sb);
|
2009-08-28 22:51:07 +00:00
|
|
|
|
CURVNET_RESTORE();
|
2002-01-10 05:36:36 +00:00
|
|
|
|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
return (error);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
linux_gifflags(struct thread *td, struct ifnet *ifp, struct l_ifreq *ifr)
|
|
|
|
|
{
|
|
|
|
|
l_short flags;
|
|
|
|
|
|
2005-08-09 10:20:02 +00:00
|
|
|
|
flags = (ifp->if_flags | ifp->if_drv_flags) & 0xffff;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
/* these flags have no Linux equivalent */
|
2005-08-09 10:20:02 +00:00
|
|
|
|
flags &= ~(IFF_SMART|IFF_DRV_OACTIVE|IFF_SIMPLEX|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
IFF_LINK0|IFF_LINK1|IFF_LINK2);
|
|
|
|
|
/* Linux' multicast flag is in a different bit */
|
|
|
|
|
if (flags & IFF_MULTICAST) {
|
|
|
|
|
flags &= ~IFF_MULTICAST;
|
|
|
|
|
flags |= 0x1000;
|
|
|
|
|
}
|
|
|
|
|
|
2003-03-03 09:14:26 +00:00
|
|
|
|
return (copyout(&flags, &ifr->ifr_flags, sizeof(flags)));
|
2001-10-15 20:52:17 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define ARPHRD_ETHER 1
|
|
|
|
|
#define ARPHRD_LOOPBACK 772
|
|
|
|
|
|
|
|
|
|
static int
|
2001-10-20 00:01:26 +00:00
|
|
|
|
linux_gifhwaddr(struct ifnet *ifp, struct l_ifreq *ifr)
|
2001-10-15 20:52:17 +00:00
|
|
|
|
{
|
|
|
|
|
struct ifaddr *ifa;
|
|
|
|
|
struct sockaddr_dl *sdl;
|
|
|
|
|
struct l_sockaddr lsa;
|
|
|
|
|
|
|
|
|
|
if (ifp->if_type == IFT_LOOP) {
|
2003-03-03 09:14:26 +00:00
|
|
|
|
bzero(&lsa, sizeof(lsa));
|
2001-10-15 20:52:17 +00:00
|
|
|
|
lsa.sa_family = ARPHRD_LOOPBACK;
|
2003-03-03 09:14:26 +00:00
|
|
|
|
return (copyout(&lsa, &ifr->ifr_hwaddr, sizeof(lsa)));
|
2001-10-15 20:52:17 +00:00
|
|
|
|
}
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
if (ifp->if_type != IFT_ETHER)
|
|
|
|
|
return (ENOENT);
|
|
|
|
|
|
|
|
|
|
TAILQ_FOREACH(ifa, &ifp->if_addrhead, ifa_link) {
|
|
|
|
|
sdl = (struct sockaddr_dl*)ifa->ifa_addr;
|
|
|
|
|
if (sdl != NULL && (sdl->sdl_family == AF_LINK) &&
|
|
|
|
|
(sdl->sdl_type == IFT_ETHER)) {
|
2003-03-03 09:14:26 +00:00
|
|
|
|
bzero(&lsa, sizeof(lsa));
|
2001-10-15 20:52:17 +00:00
|
|
|
|
lsa.sa_family = ARPHRD_ETHER;
|
|
|
|
|
bcopy(LLADDR(sdl), lsa.sa_data, LINUX_IFHWADDRLEN);
|
2003-03-03 09:14:26 +00:00
|
|
|
|
return (copyout(&lsa, &ifr->ifr_hwaddr, sizeof(lsa)));
|
2001-10-15 20:52:17 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
return (ENOENT);
|
|
|
|
|
}
|
|
|
|
|
|
2006-03-18 18:20:17 +00:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* If we fault in bsd_to_linux_ifreq() then we will fault when we call
|
|
|
|
|
* the native ioctl(). Thus, we don't really need to check the return
|
|
|
|
|
* value of this function.
|
|
|
|
|
*/
|
|
|
|
|
static int
|
|
|
|
|
bsd_to_linux_ifreq(struct ifreq *arg)
|
|
|
|
|
{
|
|
|
|
|
struct ifreq ifr;
|
|
|
|
|
size_t ifr_len = sizeof(struct ifreq);
|
|
|
|
|
int error;
|
|
|
|
|
|
|
|
|
|
if ((error = copyin(arg, &ifr, ifr_len)))
|
|
|
|
|
return (error);
|
|
|
|
|
|
|
|
|
|
*(u_short *)&ifr.ifr_addr = ifr.ifr_addr.sa_family;
|
|
|
|
|
|
|
|
|
|
error = copyout(&ifr, arg, ifr_len);
|
|
|
|
|
|
|
|
|
|
return (error);
|
|
|
|
|
}
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
/*
|
|
|
|
|
* Socket related ioctls
|
|
|
|
|
*/
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
static int
|
2001-09-12 08:38:13 +00:00
|
|
|
|
linux_ioctl_socket(struct thread *td, struct linux_ioctl_args *args)
|
1999-12-04 11:10:22 +00:00
|
|
|
|
{
|
2001-10-15 20:52:17 +00:00
|
|
|
|
char lifname[LINUX_IFNAMSIZ], ifname[IFNAMSIZ];
|
|
|
|
|
struct ifnet *ifp;
|
2001-11-19 15:43:50 +00:00
|
|
|
|
struct file *fp;
|
|
|
|
|
int error, type;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
|
|
|
|
|
ifp = NULL;
|
|
|
|
|
error = 0;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2002-06-26 15:53:11 +00:00
|
|
|
|
if ((error = fget(td, args->fd, &fp)) != 0)
|
2001-11-19 15:43:50 +00:00
|
|
|
|
return (error);
|
|
|
|
|
type = fp->f_type;
|
|
|
|
|
fdrop(fp, td);
|
|
|
|
|
if (type != DTYPE_SOCKET) {
|
|
|
|
|
/* not a socket - probably a tap / vmnet device */
|
|
|
|
|
switch (args->cmd) {
|
|
|
|
|
case LINUX_SIOCGIFADDR:
|
|
|
|
|
case LINUX_SIOCSIFADDR:
|
|
|
|
|
case LINUX_SIOCGIFFLAGS:
|
|
|
|
|
return (linux_ioctl_special(td, args));
|
|
|
|
|
default:
|
|
|
|
|
return (ENOIOCTL);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
switch (args->cmd & 0xffff) {
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
case LINUX_FIOGETOWN:
|
|
|
|
|
case LINUX_FIOSETOWN:
|
|
|
|
|
case LINUX_SIOCADDMULTI:
|
|
|
|
|
case LINUX_SIOCATMARK:
|
|
|
|
|
case LINUX_SIOCDELMULTI:
|
|
|
|
|
case LINUX_SIOCGIFCONF:
|
|
|
|
|
case LINUX_SIOCGPGRP:
|
|
|
|
|
case LINUX_SIOCSPGRP:
|
2007-11-07 16:42:52 +00:00
|
|
|
|
case LINUX_SIOCGIFCOUNT:
|
2001-10-15 20:52:17 +00:00
|
|
|
|
/* these ioctls don't take an interface name */
|
|
|
|
|
#ifdef DEBUG
|
2001-12-10 08:09:49 +00:00
|
|
|
|
printf("%s(): ioctl %d\n", __func__,
|
2001-10-15 20:52:17 +00:00
|
|
|
|
args->cmd & 0xffff);
|
|
|
|
|
#endif
|
|
|
|
|
break;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2001-10-20 00:01:26 +00:00
|
|
|
|
case LINUX_SIOCGIFFLAGS:
|
2001-10-15 20:52:17 +00:00
|
|
|
|
case LINUX_SIOCGIFADDR:
|
2001-11-19 15:43:50 +00:00
|
|
|
|
case LINUX_SIOCSIFADDR:
|
2001-10-15 20:52:17 +00:00
|
|
|
|
case LINUX_SIOCGIFDSTADDR:
|
2001-10-20 00:01:26 +00:00
|
|
|
|
case LINUX_SIOCGIFBRDADDR:
|
2001-10-15 20:52:17 +00:00
|
|
|
|
case LINUX_SIOCGIFNETMASK:
|
2001-10-20 00:01:26 +00:00
|
|
|
|
case LINUX_SIOCSIFNETMASK:
|
|
|
|
|
case LINUX_SIOCGIFMTU:
|
|
|
|
|
case LINUX_SIOCSIFMTU:
|
|
|
|
|
case LINUX_SIOCSIFNAME:
|
|
|
|
|
case LINUX_SIOCGIFHWADDR:
|
|
|
|
|
case LINUX_SIOCSIFHWADDR:
|
|
|
|
|
case LINUX_SIOCDEVPRIVATE:
|
|
|
|
|
case LINUX_SIOCDEVPRIVATE+1:
|
2007-11-07 16:42:52 +00:00
|
|
|
|
case LINUX_SIOCGIFINDEX:
|
2001-10-15 20:52:17 +00:00
|
|
|
|
/* copy in the interface name and translate it. */
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyin((void *)args->arg, lifname, LINUX_IFNAMSIZ);
|
2001-10-20 00:01:26 +00:00
|
|
|
|
if (error != 0)
|
|
|
|
|
return (error);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
#ifdef DEBUG
|
2001-12-10 08:09:49 +00:00
|
|
|
|
printf("%s(): ioctl %d on %.*s\n", __func__,
|
2001-10-15 20:52:17 +00:00
|
|
|
|
args->cmd & 0xffff, LINUX_IFNAMSIZ, lifname);
|
|
|
|
|
#endif
|
2009-06-15 19:01:53 +00:00
|
|
|
|
ifp = ifname_linux_to_bsd(td, lifname, ifname);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
if (ifp == NULL)
|
|
|
|
|
return (EINVAL);
|
|
|
|
|
/*
|
|
|
|
|
* We need to copy it back out in case we pass the
|
|
|
|
|
* request on to our native ioctl(), which will expect
|
|
|
|
|
* the ifreq to be in user space and have the correct
|
|
|
|
|
* interface name.
|
|
|
|
|
*/
|
2003-03-03 09:14:26 +00:00
|
|
|
|
error = copyout(ifname, (void *)args->arg, IFNAMSIZ);
|
2001-10-20 00:01:26 +00:00
|
|
|
|
if (error != 0)
|
|
|
|
|
return (error);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
#ifdef DEBUG
|
2001-12-10 08:09:49 +00:00
|
|
|
|
printf("%s(): %s translated to %s\n", __func__,
|
2001-10-15 20:52:17 +00:00
|
|
|
|
lifname, ifname);
|
|
|
|
|
#endif
|
|
|
|
|
break;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
default:
|
|
|
|
|
return (ENOIOCTL);
|
|
|
|
|
}
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
switch (args->cmd & 0xffff) {
|
|
|
|
|
|
|
|
|
|
case LINUX_FIOSETOWN:
|
|
|
|
|
args->cmd = FIOSETOWN;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SIOCSPGRP:
|
|
|
|
|
args->cmd = SIOCSPGRP;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_FIOGETOWN:
|
|
|
|
|
args->cmd = FIOGETOWN;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SIOCGPGRP:
|
|
|
|
|
args->cmd = SIOCGPGRP;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SIOCATMARK:
|
|
|
|
|
args->cmd = SIOCATMARK;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
/* LINUX_SIOCGSTAMP */
|
|
|
|
|
|
|
|
|
|
case LINUX_SIOCGIFCONF:
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = linux_ifconf(td, (struct ifconf *)args->arg);
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SIOCGIFFLAGS:
|
|
|
|
|
args->cmd = SIOCGIFFLAGS;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = linux_gifflags(td, ifp, (struct l_ifreq *)args->arg);
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SIOCGIFADDR:
|
2006-03-18 18:20:17 +00:00
|
|
|
|
args->cmd = SIOCGIFADDR;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
2006-03-18 18:20:17 +00:00
|
|
|
|
bsd_to_linux_ifreq((struct ifreq *)args->arg);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
2001-11-19 15:43:50 +00:00
|
|
|
|
case LINUX_SIOCSIFADDR:
|
|
|
|
|
/* XXX probably doesn't work, included for completeness */
|
|
|
|
|
args->cmd = SIOCSIFADDR;
|
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SIOCGIFDSTADDR:
|
2006-03-18 18:20:17 +00:00
|
|
|
|
args->cmd = SIOCGIFDSTADDR;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
2006-03-18 18:20:17 +00:00
|
|
|
|
bsd_to_linux_ifreq((struct ifreq *)args->arg);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SIOCGIFBRDADDR:
|
2006-03-18 18:20:17 +00:00
|
|
|
|
args->cmd = SIOCGIFBRDADDR;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
2006-03-18 18:20:17 +00:00
|
|
|
|
bsd_to_linux_ifreq((struct ifreq *)args->arg);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
case LINUX_SIOCGIFNETMASK:
|
2006-03-18 18:20:17 +00:00
|
|
|
|
args->cmd = SIOCGIFNETMASK;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
2006-03-18 18:20:17 +00:00
|
|
|
|
bsd_to_linux_ifreq((struct ifreq *)args->arg);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
2001-10-20 00:01:26 +00:00
|
|
|
|
case LINUX_SIOCSIFNETMASK:
|
|
|
|
|
error = ENOIOCTL;
|
|
|
|
|
break;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2001-10-20 00:01:26 +00:00
|
|
|
|
case LINUX_SIOCGIFMTU:
|
|
|
|
|
args->cmd = SIOCGIFMTU;
|
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2001-10-20 00:01:26 +00:00
|
|
|
|
case LINUX_SIOCSIFMTU:
|
|
|
|
|
args->cmd = SIOCSIFMTU;
|
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2001-10-20 00:01:26 +00:00
|
|
|
|
case LINUX_SIOCSIFNAME:
|
|
|
|
|
error = ENOIOCTL;
|
|
|
|
|
break;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
case LINUX_SIOCGIFHWADDR:
|
2001-10-20 00:01:26 +00:00
|
|
|
|
error = linux_gifhwaddr(ifp, (struct l_ifreq *)args->arg);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
break;
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
2001-10-20 00:01:26 +00:00
|
|
|
|
case LINUX_SIOCSIFHWADDR:
|
|
|
|
|
error = ENOIOCTL;
|
|
|
|
|
break;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SIOCADDMULTI:
|
|
|
|
|
args->cmd = SIOCADDMULTI;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
case LINUX_SIOCDELMULTI:
|
|
|
|
|
args->cmd = SIOCDELMULTI;
|
2001-10-15 20:52:17 +00:00
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
2001-10-19 01:38:10 +00:00
|
|
|
|
|
2007-11-07 16:42:52 +00:00
|
|
|
|
case LINUX_SIOCGIFINDEX:
|
|
|
|
|
args->cmd = SIOCGIFINDEX;
|
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case LINUX_SIOCGIFCOUNT:
|
|
|
|
|
error = 0;
|
|
|
|
|
break;
|
|
|
|
|
|
2001-10-19 01:38:10 +00:00
|
|
|
|
/*
|
|
|
|
|
* XXX This is slightly bogus, but these ioctls are currently
|
|
|
|
|
* XXX only used by the aironet (if_an) network driver.
|
|
|
|
|
*/
|
2001-10-20 00:01:26 +00:00
|
|
|
|
case LINUX_SIOCDEVPRIVATE:
|
|
|
|
|
args->cmd = SIOCGPRIVATE_0;
|
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
2001-10-20 00:01:26 +00:00
|
|
|
|
case LINUX_SIOCDEVPRIVATE+1:
|
|
|
|
|
args->cmd = SIOCGPRIVATE_1;
|
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
2001-10-15 20:52:17 +00:00
|
|
|
|
if (ifp != NULL)
|
|
|
|
|
/* restore the original interface name */
|
2003-03-03 09:14:26 +00:00
|
|
|
|
copyout(lifname, (void *)args->arg, LINUX_IFNAMSIZ);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
2001-12-10 08:09:49 +00:00
|
|
|
|
printf("%s(): returning %d\n", __func__, error);
|
2001-10-15 20:52:17 +00:00
|
|
|
|
#endif
|
|
|
|
|
return (error);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
2001-10-19 01:38:10 +00:00
|
|
|
|
/*
|
|
|
|
|
* Device private ioctl handler
|
|
|
|
|
*/
|
|
|
|
|
static int
|
|
|
|
|
linux_ioctl_private(struct thread *td, struct linux_ioctl_args *args)
|
|
|
|
|
{
|
|
|
|
|
struct file *fp;
|
2001-11-19 15:43:50 +00:00
|
|
|
|
int error, type;
|
2001-10-19 01:38:10 +00:00
|
|
|
|
|
2002-06-26 15:53:11 +00:00
|
|
|
|
if ((error = fget(td, args->fd, &fp)) != 0)
|
2001-11-19 15:43:50 +00:00
|
|
|
|
return (error);
|
|
|
|
|
type = fp->f_type;
|
|
|
|
|
fdrop(fp, td);
|
2001-10-19 01:38:10 +00:00
|
|
|
|
if (type == DTYPE_SOCKET)
|
|
|
|
|
return (linux_ioctl_socket(td, args));
|
2001-11-19 15:43:50 +00:00
|
|
|
|
return (ENOIOCTL);
|
|
|
|
|
}
|
|
|
|
|
|
2003-04-24 23:36:35 +00:00
|
|
|
|
/*
|
|
|
|
|
* DRM ioctl handler (sys/dev/drm)
|
|
|
|
|
*/
|
|
|
|
|
static int
|
|
|
|
|
linux_ioctl_drm(struct thread *td, struct linux_ioctl_args *args)
|
|
|
|
|
{
|
|
|
|
|
args->cmd = SETDIR(args->cmd);
|
|
|
|
|
return ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
}
|
|
|
|
|
|
2007-04-07 19:40:58 +00:00
|
|
|
|
static int
|
|
|
|
|
linux_ioctl_sg(struct thread *td, struct linux_ioctl_args *args)
|
|
|
|
|
{
|
|
|
|
|
struct file *fp;
|
|
|
|
|
u_long cmd;
|
|
|
|
|
int error;
|
|
|
|
|
|
|
|
|
|
if ((error = fget(td, args->fd, &fp)) != 0) {
|
|
|
|
|
printf("sg_linux_ioctl: fget returned %d\n", error);
|
|
|
|
|
return (error);
|
|
|
|
|
}
|
|
|
|
|
cmd = args->cmd;
|
|
|
|
|
|
|
|
|
|
error = (fo_ioctl(fp, cmd, (caddr_t)args->arg, td->td_ucred, td));
|
|
|
|
|
fdrop(fp, td);
|
|
|
|
|
return (error);
|
|
|
|
|
}
|
|
|
|
|
|
2001-11-19 15:43:50 +00:00
|
|
|
|
/*
|
|
|
|
|
* Special ioctl handler
|
|
|
|
|
*/
|
|
|
|
|
static int
|
|
|
|
|
linux_ioctl_special(struct thread *td, struct linux_ioctl_args *args)
|
|
|
|
|
{
|
|
|
|
|
int error;
|
|
|
|
|
|
|
|
|
|
switch (args->cmd) {
|
|
|
|
|
case LINUX_SIOCGIFADDR:
|
|
|
|
|
args->cmd = SIOCGIFADDR;
|
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_SIOCSIFADDR:
|
|
|
|
|
args->cmd = SIOCSIFADDR;
|
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
|
|
|
|
case LINUX_SIOCGIFFLAGS:
|
|
|
|
|
args->cmd = SIOCGIFFLAGS;
|
|
|
|
|
error = ioctl(td, (struct ioctl_args *)args);
|
|
|
|
|
break;
|
|
|
|
|
default:
|
|
|
|
|
error = ENOIOCTL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (error);
|
2001-10-19 01:38:10 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
/*
|
|
|
|
|
* main ioctl syscall function
|
|
|
|
|
*/
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
int
|
2001-09-12 08:38:13 +00:00
|
|
|
|
linux_ioctl(struct thread *td, struct linux_ioctl_args *args)
|
1999-12-04 11:10:22 +00:00
|
|
|
|
{
|
|
|
|
|
struct file *fp;
|
|
|
|
|
struct handler_element *he;
|
|
|
|
|
int error, cmd;
|
|
|
|
|
|
|
|
|
|
#ifdef DEBUG
|
2001-02-16 16:40:43 +00:00
|
|
|
|
if (ldebug(ioctl))
|
2001-11-17 01:20:02 +00:00
|
|
|
|
printf(ARGS(ioctl, "%d, %04lx, *"), args->fd,
|
|
|
|
|
(unsigned long)args->cmd);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
2002-01-14 00:13:45 +00:00
|
|
|
|
if ((error = fget(td, args->fd, &fp)) != 0)
|
|
|
|
|
return (error);
|
2002-01-13 11:58:06 +00:00
|
|
|
|
if ((fp->f_flag & (FREAD|FWRITE)) == 0) {
|
|
|
|
|
fdrop(fp, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (EBADF);
|
2002-01-13 11:58:06 +00:00
|
|
|
|
}
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
/* Iterate over the ioctl handlers */
|
|
|
|
|
cmd = args->cmd & 0xffff;
|
2006-07-06 21:42:36 +00:00
|
|
|
|
sx_slock(&linux_ioctl_sx);
|
|
|
|
|
mtx_lock(&Giant);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
TAILQ_FOREACH(he, &handlers, list) {
|
|
|
|
|
if (cmd >= he->low && cmd <= he->high) {
|
2001-09-12 08:38:13 +00:00
|
|
|
|
error = (*he->func)(td, args);
|
2002-02-20 17:06:37 +00:00
|
|
|
|
if (error != ENOIOCTL) {
|
2006-07-06 21:42:36 +00:00
|
|
|
|
mtx_unlock(&Giant);
|
|
|
|
|
sx_sunlock(&linux_ioctl_sx);
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (error);
|
2002-02-20 17:06:37 +00:00
|
|
|
|
}
|
1999-12-04 11:10:22 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
2006-07-06 21:42:36 +00:00
|
|
|
|
mtx_unlock(&Giant);
|
|
|
|
|
sx_sunlock(&linux_ioctl_sx);
|
2002-01-13 11:58:06 +00:00
|
|
|
|
fdrop(fp, td);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
2003-01-02 02:19:10 +00:00
|
|
|
|
linux_msg(td, "ioctl fd=%d, cmd=0x%x ('%c',%d) is not implemented",
|
1999-12-04 11:10:22 +00:00
|
|
|
|
args->fd, (int)(args->cmd & 0xffff),
|
|
|
|
|
(int)(args->cmd & 0xff00) >> 8, (int)(args->cmd & 0xff));
|
|
|
|
|
|
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
linux_ioctl_register_handler(struct linux_ioctl_handler *h)
|
|
|
|
|
{
|
|
|
|
|
struct handler_element *he, *cur;
|
|
|
|
|
|
|
|
|
|
if (h == NULL || h->func == NULL)
|
|
|
|
|
return (EINVAL);
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* Reuse the element if the handler is already on the list, otherwise
|
|
|
|
|
* create a new element.
|
|
|
|
|
*/
|
2006-07-06 21:42:36 +00:00
|
|
|
|
sx_xlock(&linux_ioctl_sx);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
TAILQ_FOREACH(he, &handlers, list) {
|
|
|
|
|
if (he->func == h->func)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (he == NULL) {
|
2008-10-23 15:53:51 +00:00
|
|
|
|
he = malloc(sizeof(*he),
|
2003-02-19 05:47:46 +00:00
|
|
|
|
M_LINUX, M_WAITOK);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
he->func = h->func;
|
|
|
|
|
} else
|
|
|
|
|
TAILQ_REMOVE(&handlers, he, list);
|
2003-03-02 15:56:49 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
/* Initialize range information. */
|
|
|
|
|
he->low = h->low;
|
|
|
|
|
he->high = h->high;
|
|
|
|
|
he->span = h->high - h->low + 1;
|
|
|
|
|
|
|
|
|
|
/* Add the element to the list, sorted on span. */
|
|
|
|
|
TAILQ_FOREACH(cur, &handlers, list) {
|
|
|
|
|
if (cur->span > he->span) {
|
|
|
|
|
TAILQ_INSERT_BEFORE(cur, he, list);
|
2006-07-06 21:42:36 +00:00
|
|
|
|
sx_xunlock(&linux_ioctl_sx);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (0);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
TAILQ_INSERT_TAIL(&handlers, he, list);
|
2006-07-06 21:42:36 +00:00
|
|
|
|
sx_xunlock(&linux_ioctl_sx);
|
1999-04-29 04:37:57 +00:00
|
|
|
|
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
linux_ioctl_unregister_handler(struct linux_ioctl_handler *h)
|
|
|
|
|
{
|
|
|
|
|
struct handler_element *he;
|
|
|
|
|
|
|
|
|
|
if (h == NULL || h->func == NULL)
|
|
|
|
|
return (EINVAL);
|
|
|
|
|
|
2006-07-06 21:42:36 +00:00
|
|
|
|
sx_xlock(&linux_ioctl_sx);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
TAILQ_FOREACH(he, &handlers, list) {
|
|
|
|
|
if (he->func == h->func) {
|
|
|
|
|
TAILQ_REMOVE(&handlers, he, list);
|
2006-07-06 21:42:36 +00:00
|
|
|
|
sx_xunlock(&linux_ioctl_sx);
|
2008-10-23 15:53:51 +00:00
|
|
|
|
free(he, M_LINUX);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
return (0);
|
|
|
|
|
}
|
1999-04-29 04:37:57 +00:00
|
|
|
|
}
|
2006-07-06 21:42:36 +00:00
|
|
|
|
sx_xunlock(&linux_ioctl_sx);
|
1999-12-04 11:10:22 +00:00
|
|
|
|
|
|
|
|
|
return (EINVAL);
|
|
|
|
|
}
|