freebsd-skq/sys/i386/ibcs2/ibcs2_socksys.c

1394 lines
31 KiB
C

/*-
* Copyright (c) 1994 Mostyn Lewis
* All rights reserved.
*
* This software is based on code which is:
* Copyright (c) 1994 Mike Jagdis (jaggy@purplet.demon.co.uk)
*
* 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
* derived from this software withough specific prior written permission
*
* 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.
*
* $Id: ibcs2_socksys.c,v 1.1 1994/10/14 08:53:08 sos Exp $
*/
#include <sys/param.h>
#include <sys/systm.h>
#include <sys/filedesc.h>
#include <sys/file.h>
#include <sys/kernel.h>
#include <sys/mbuf.h>
#include <sys/protosw.h>
#include <sys/socket.h>
#include <sys/socketvar.h>
#include <sys/ioctl.h>
#include <sys/sockio.h>
#include <sys/proc.h>
#include <sys/exec.h>
#include <sys/stat.h>
#include <sys/syscall.h>
#include <sys/sysent.h>
#include <sys/malloc.h>
#include <sys/un.h>
#include <net/if.h>
#include <net/if_arp.h>
#include <net/route.h>
#include <netinet/in.h>
#include <vm/vm.h>
#include <i386/ibcs2/ibcs2.h>
#include <i386/ibcs2/ibcs2_socksys.h>
/* Socksys pseudo driver entry points */
int sockopen (dev_t dev, int mode, int devtype, struct proc *p);
int sockioctl(dev_t dev, int cmd, caddr_t arg, int fflag, struct proc *p);
int sockclose(dev_t dev, int flag, int mode, struct proc *p);
/* Socksys internal functions */
static void put_socket_fops(struct proc *p, int fd);
static int ss_fop_close(struct file *fp, struct proc *p);
static int ss_fop_ioctl(struct file*fp, int cmd, caddr_t arg, struct proc *p);
static int ss_syscall(caddr_t arg, struct proc *p);
/*
* This structure is setup on first usage. Its address is planted
* into a socket's file structure fileops pointer after a successful
* socket creation or accept.
*/
static struct fileops ss_socket_fops = {
NULL, /* normal socket read */
NULL, /* normal socket write */
NULL, /* socksys ioctl */
NULL, /* normal socket select */
NULL, /* socksys close */
};
static int (*close_s)__P((struct file *fp, struct proc *p));
static int (*ioctl_s)__P((struct file *fp, int cmd, caddr_t data, struct proc *p));
int ss_debug = 10;
static int
ss_syscall(arg, p)
caddr_t arg;
struct proc *p;
{
int cmd;
int error;
int retval[2];
retval[0] = retval[1] = 0;
cmd = ((struct ss_call *)arg)->arg[0];
if(ss_debug) {
static char *ss_syscall_strings[] = {
"0?", "accept", "bind", "connect", "getpeername",
"getsockname", "getsockopt", "listen", "recv(from)",
"recvfrom", "send(to)", "sendto", "setsockopt", "shutdown",
"socket", "select", "getipdomain", "setipdomain",
"adjtime", "setreuid", "setregid", "gettimeofday",
"settimeofday", "getitimer", "setitimer",
};
printf("ss_syscall: [%d] ",p->p_pid);
if(cmd < 0 || (cmd > CMD_SO_SETITIMER && cmd != CMD_SO_SS_DEBUG) )
printf("? ");
else {
if(cmd == CMD_SO_SS_DEBUG)
printf("%s ","ss_debug");
else
printf("%s ",ss_syscall_strings[cmd]);
}
printf("(%d) <0x%x,0x%x,0x%x,0x%x,0x%x,0x%x>\n",
cmd,
((struct ss_call *)arg)->arg[1],
((struct ss_call *)arg)->arg[2],
((struct ss_call *)arg)->arg[3],
((struct ss_call *)arg)->arg[4],
((struct ss_call *)arg)->arg[5],
((struct ss_call *)arg)->arg[6]);
}
error = 0;
switch (cmd) {
case CMD_SO_SS_DEBUG:
/* ss_debug = ((struct ss_call *)arg)->arg[1]; */
break;
case CMD_SO_SOCKET: { /* NO CONV */
if(ss_debug > 1)
printf("SO_SOCKET af in %d\n",
((struct ss_call *)arg)->arg[1]);
((struct ss_call *)arg)->arg[1] = ss_convert(
af_whatevers,
&(((struct ss_call *)arg)->arg[1]),
0);
if(ss_debug > 1) {
printf("SO_SOCKET af out %d\n",
((struct ss_call *)arg)->arg[1]);
printf("SO_SOCKET type in %d\n",
((struct ss_call *)arg)->arg[2]);
}
((struct ss_call *)arg)->arg[2] = ss_convert(
type_whatevers,
&(((struct ss_call *)arg)->arg[2]),
0);
if(ss_debug > 1)
printf("SO_SOCKET type out %d\n",
((struct ss_call *)arg)->arg[2]);
SYSCALL(SYS_socket, 0, 0);
if(ss_debug)
printf("ss_syscall: [%d] socket fd=%d\n",
p->p_pid, retval[0]);
put_socket_fops(p,retval[0]);
break;
}
case CMD_SO_ACCEPT: { /* CONVERSION in arg 2 */
SYSCALL(SYS_accept, 2, SS_STRUCT_SOCKADDR);
if(ss_debug)
printf("ss_syscall: [%d] accept fd=%d\n",
p->p_pid, retval[0]);
put_socket_fops(p,retval[0]);
break;
}
case CMD_SO_BIND:
SYSCALL(SYS_bind, 2, SS_STRUCT_SOCKADDR);
break;
case CMD_SO_CONNECT: {
struct alien_sockaddr *sa;
unsigned short family;
/* Remap any INADDR_ANY (0.0.0.0) to localhost */
sa = (struct alien_sockaddr *)((struct ss_call *)arg)->arg[1];
if(error = copyin((caddr_t)&sa->sa_family,
(caddr_t)&family, sizeof(short)))
return(error);
if (family == AF_INET) {
unsigned long *addr;
unsigned long saddr;
addr = &(((struct alien_sockaddr_in *)sa)->sin_addr.s_addr);
if(error = copyin((caddr_t)addr, (caddr_t)&saddr, sizeof(long)))
return(error);
if (saddr == INADDR_ANY) {
/* 0x0100007f is 127.0.0.1 reversed */
saddr = 0x0100007f;
if(error = copyout((caddr_t)&saddr,
(caddr_t)addr, sizeof(long)))
return(error);
if (ss_debug)
printf("ss_syscall: remapped INADDR_ANY to localhost\n");
}
}
SYSCALL(SYS_connect, 2, SS_STRUCT_SOCKADDR);
break;
}
case CMD_SO_GETPEERNAME:
SYSCALL(SYS_getpeername, 2, SS_STRUCT_SOCKADDR);
break;
case CMD_SO_GETSOCKNAME:
SYSCALL(SYS_getsockname, 2, SS_STRUCT_SOCKADDR);
break;
case CMD_SO_GETSOCKOPT:
if(error = ss_getsockopt((caddr_t)(((int *)arg) + 1),retval,p))
return(error);
break;
case CMD_SO_LISTEN:
SYSCALL(SYS_listen, 0, 0);
break;
case CMD_SO_RECV:
((struct ss_call *)arg)->arg[5] = (int)((struct sockaddr *)NULL);
((struct ss_call *)arg)->arg[6] = 0;
SYSCALL(SYS_recvfrom, 0, 0);
break;
case CMD_SO_RECVFROM:
SYSCALL(SYS_recvfrom, 5, SS_STRUCT_SOCKADDR);
break;
case CMD_SO_SEND:
((struct ss_call *)arg)->arg[5] = (int)((struct sockaddr *)NULL);
((struct ss_call *)arg)->arg[6] = 0;
SYSCALL(SYS_sendto, 0, 0);
break;
case CMD_SO_SENDTO:
SYSCALL(SYS_sendto, 5, SS_STRUCT_SOCKADDR);
break;
case CMD_SO_SETSOCKOPT:
if(error = ss_setsockopt((caddr_t)(((int *)arg) + 1),retval,p))
return(error);
case CMD_SO_SHUTDOWN:
SYSCALL(SYS_shutdown, 0, 0);
break;
case CMD_SO_GETIPDOMAIN:
SYSCALL(SYS_getdomainname, 0, 0);
break;
case CMD_SO_SETIPDOMAIN: /* Note check on BSD utsname no change? */
SYSCALL(SYS_setdomainname, 0, 0);
break;
case CMD_SO_SETREUID:
SYSCALL(126/*SYS_setreuid*/, 0, 0);
break;
case CMD_SO_SETREGID:
SYSCALL(127/*SYS_setregid*/, 0, 0);
break;
case CMD_SO_GETTIME:
SYSCALL(SYS_gettimeofday, 0, 0);
break;
case CMD_SO_SETTIME:
SYSCALL(SYS_settimeofday, 0, 0);
break;
case CMD_SO_GETITIMER:
SYSCALL(SYS_getitimer, 0, 0);
break;
case CMD_SO_SETITIMER:
SYSCALL(SYS_setitimer, 0, 0);
break;
case CMD_SO_SELECT:
SYSCALL(SYS_select, 0, 0);
break;
case CMD_SO_ADJTIME:
SYSCALL(SYS_adjtime, 0, 0);
break;
default:
printf("ss_syscall: default 0x%x\n",cmd);
return (EINVAL);
}
IBCS2_MAGIC_RETURN(arg);
}
static int
ss_fop_ioctl(fp, cmd, arg, p)
struct file *fp;
int cmd;
caddr_t arg;
struct proc *p;
{
int error;
int retval[2];
if(ss_debug) {
static char **ioctl_strings;
int fd;
struct filedesc *fdp;
unsigned int ioctl_type;
unsigned int ioctl_len;
char cmd_type;
int cmd_ordinal;
static char *ioctl_type_strings[] = {
"0?", "SS_IO", "SS_IOR", "3?", "SS_IOW", "5?", "SS_IOWR"
};
static char *ioctl_S_strings[] = {
"0?", "SIOCSHIWAT", "SIOCGHIWAT", "SIOCSLOWAT", "SIOCGLOWAT",
"SIOCATMARK", "SIOCSPGRP", "SIOCGPGRP", "FIONREAD",
"FIONBIO", "FIOASYNC", "SIOCPROTO", "SIOCGETNAME",
"SIOCGETPEER", "IF_UNITSEL", "SIOCXPROTO"
};
static char *ioctl_R_strings[] = {
"0?", "1?", "2?", "3?", "4?", "5?", "6?", "7?", "8?",
"SIOCADDRT", "SIOCDELRT"
};
static char *ioctl_I_strings[] = {
"0?", "1?", "2?", "3?", "4?", "5?", "6?", "7?", "8?",
"9?", "10?", "SIOCSIFADDR", "SIOCGIFADDR", "SIOCSIFDSTADDR",
"SIOCGIFDSTADDR", "SIOCSIFFLAGS", "SIOCGIFFLAGS",
"SIOCGIFCONF", "18?", "19?", "20?", "SIOCSIFMTU",
"SIOCGIFMTU", "23?", "24?", "25?", "SIOCIFDETACH",
"SIOCGENPSTATS", "28?", "SIOCX25XMT", "SS_SIOCX25RCV",
"SS_SIOCX25TBL", "SIOCGIFBRDADDR" ,"SIOCSIFBRDADDR",
"SIOCGIFNETMASK", "SIOCSIFNETMASK", "SIOCGIFMETRIC",
"SIOCSIFMETRIC", "SIOCSARP", "SIOCGARP", "SIOCDARP",
"SIOCSIFNAME", "SIOCGIFONEP", "SIOCSIFONEP ",
"44?", "45?", "46?", "47?", "48?", "49?", "50?", "51?",
"52?", "53?", "54?", "55?", "56?", "57?", "58?", "59?",
"60?", "61?", "62?", "63?", "64?", "SIOCGENADDR",
"SIOCSOCKSYS"
};
cmd_type = (cmd >> 8) & 0xff;
cmd_ordinal = cmd & 0xff;
switch (cmd_type) {
case 'S':
ioctl_strings = ioctl_S_strings;
if (cmd_ordinal > 15)
cmd_ordinal = -1;
break;
case 'R':
ioctl_strings = ioctl_R_strings;
if (cmd_ordinal > 10)
cmd_ordinal = -1;
break;
case 'I':
ioctl_strings = ioctl_I_strings;
if (cmd_ordinal > 66)
cmd_ordinal = -1;
break;
default:
cmd_type = '?';
break;
}
fdp = p->p_fd;
fd = -1;
while(++fd < NOFILE)
if ( fp == fdp->fd_ofiles[fd] )
break;
ioctl_type = (0xe0000000 & cmd) >> 29;
ioctl_len = (cmd >> 16) & SS_IOCPARM_MASK;
printf("ss_fop_ioctl: [%d] fd=%d ",p->p_pid, fd);
if(cmd_type != '?'){
if(cmd_ordinal != -1)
printf("%s %s('%c',%d,l=%d) ",ioctl_strings[cmd_ordinal],
ioctl_type_strings[ioctl_type],
cmd_type,
cmd_ordinal,
ioctl_len);
else {
cmd_ordinal = cmd & 0xff;
printf("[unknown ordinal %d] %s('%c',%d,l=%d) ",cmd_ordinal,
ioctl_type_strings[ioctl_type],
cmd_type,
cmd_ordinal,
ioctl_len);
}
}
else {
printf("? %s('%c',%d,l=%d) ",
ioctl_type_strings[ioctl_type],
cmd_type,
cmd_ordinal,
ioctl_len);
}
printf("0x%x (0x%x) <0x%x>\n",
fp, cmd, arg);
}
/* No dogs allowed */
if(*(((int *)arg) - 3) != IBCS2_MAGIC_IN){
printf("ss_fop_ioctl: bad magic (sys_generic.c has no socksys mods?)\n");
return(EINVAL);
}
if(fp->f_type != DTYPE_SOCKET)
return (ENOTSOCK);
retval[0] = retval[1] = 0;
error = 0;
switch (cmd) {
case SS_SIOCSOCKSYS: /* ss syscall */
return ss_syscall(arg, p);
case SS_SIOCSHIWAT: /* set high watermark */
case SS_SIOCSLOWAT: /* set low watermark */
break; /* return value of 0 and no error */
case SS_SIOCGHIWAT: /* get high watermark */
case SS_SIOCGLOWAT: /* get low watermark */
break; /* return value of 0 and no error */
case SS_SIOCATMARK: /* at oob mark */
IOCTL(SIOCATMARK);
break;
case SS_SIOCSPGRP: /* set process group */
IOCTL(SIOCSPGRP);
break;
case SS_SIOCGPGRP: /* get process group */
IOCTL(SIOCGPGRP);
break;
case FIONREAD:
case SS_FIONREAD: /* get # bytes to read */
IOCTL(FIONREAD);
break;
case SS_FIONBIO: /* set/clear non-blocking i/o */
IOCTL(FIONBIO);
break;
case SS_FIOASYNC: /* set/clear async i/o */
IOCTL(FIOASYNC);
break;
case SS_SIOCADDRT: /* add route - uses struct ortentry */
IOCTL(SIOCADDRT);
break;
case SS_SIOCDELRT: /* delete route - uses struct ortentry */
IOCTL(SIOCDELRT);
break;
case SS_SIOCSIFADDR: /* set ifnet address */
IOCTL(SIOCSIFADDR);
break;
case SS_SIOCGIFADDR: /* get ifnet address */
IOCTL(SIOCGIFADDR);
break;
case SS_SIOCSIFDSTADDR: /* set p-p address */
IOCTL(SIOCSIFDSTADDR);
break;
case SS_SIOCGIFDSTADDR: /* get p-p address */
IOCTL(SIOCGIFDSTADDR);
break;
case SS_SIOCSIFFLAGS: /* set ifnet flags */
IOCTL(SIOCSIFFLAGS);
break;
case SS_SIOCGIFFLAGS: /* get ifnet flags */
IOCTL(SIOCGIFFLAGS);
break;
case SS_SIOCGIFCONF: /* get ifnet ltst */
IOCTL(SIOCGIFCONF);
break;
case SS_SIOCGIFBRDADDR: /* get broadcast addr */
IOCTL(SIOCGIFBRDADDR);
break;
case SS_SIOCSIFBRDADDR: /* set broadcast addr */
IOCTL(SIOCSIFBRDADDR);
break;
case SS_SIOCGIFNETMASK: /* get net addr mask */
IOCTL(SIOCGIFNETMASK);
break;
case SS_SIOCSIFNETMASK: /* set net addr mask */
IOCTL(SIOCSIFNETMASK);
break;
case SS_SIOCGIFMETRIC: /* get IF metric */
IOCTL(SIOCGIFMETRIC);
break;
case SS_SIOCSIFMETRIC: /* set IF metric */
IOCTL(SIOCSIFMETRIC);
break;
/* FreeBSD 2.0 does not have socket ARPs */
#ifdef SIOCSARP
case SS_SIOCSARP: /* set arp entry */
IOCTL(SIOCSARP);
break;
case SS_SIOCGARP: /* get arp entry */
IOCTL(SIOCGARP);
break;
case SS_SIOCDARP: /* delete arp entry */
IOCTL(SIOCDARP);
break;
#else /* SIOCSARP */
case SS_SIOCSARP: /* set arp entry */
return(EINVAL);
case SS_SIOCGARP: /* get arp entry */
return(EINVAL);
case SS_SIOCDARP: /* delete arp entry */
return(EINVAL);
#endif /* SIOCSARP */
case SS_SIOCGENADDR: /* Get ethernet addr XXX */
return (EINVAL);
/* return (error = ioctl_s(fp, SIOCGIFHWADDR, arg, p)); */
case SS_SIOCSIFMTU: /* get if_mtu */
IOCTL(SIOCSIFMTU);
break;
case SS_SIOCGIFMTU: /* set if_mtu */
IOCTL(SIOCGIFMTU);
break;
case SS_SIOCGETNAME: /* getsockname XXX */
return (EINVAL);
/* return (ioctl_s(fp, SIOCGIFNAME, arg, p)); MMM */
case SS_SIOCGETPEER: { /* getpeername */
struct moose {
int fd;
caddr_t asa;
int *alen;
int compat_43;
} args;
struct alien_sockaddr uaddr;
struct sockaddr nuaddr;
int nuaddr_len = sizeof(struct sockaddr);
struct filedesc *fdp;
if(fp->f_type != DTYPE_SOCKET)
return (ENOTSOCK);
bzero((caddr_t)&nuaddr, sizeof(struct sockaddr));
fdp = p->p_fd;
args.fd = -1;
while(++args.fd < NOFILE)
if ( fp == fdp->fd_ofiles[args.fd] )
break;
if(args.fd == NOFILE){
printf("ss_fop_ioctl: [%d] SS_SIOCGETPEER args.fd > NOFILE\n", p->p_pid);
return(EBADF);
}
args.asa = (caddr_t)&nuaddr;
args.alen = &nuaddr_len;
args.compat_43 = 0;
error = SYSCALLX(SYS_getpeername, &args);
if(error)
return(error);
bzero((caddr_t)&uaddr, sizeof(struct alien_sockaddr));
uaddr.sa_family = (unsigned short)nuaddr.sa_family;
bcopy(&nuaddr.sa_data, &uaddr.sa_data, __ALIEN_SOCK_SIZE__ - sizeof(unsigned short));
error = copyout((caddr_t)&uaddr, (caddr_t)arg, sizeof(struct alien_sockaddr));
return error;
}
default:
printf("ss_fop_ioctl: [%d] %lx unknown ioctl 0x%x, 0x%lx\n",
p->p_pid, (unsigned long)fp,
cmd, (unsigned long)arg);
return (EINVAL);
}
IBCS2_MAGIC_RETURN(arg);
}
int
sockioctl(dev, cmd, arg, fflag, p)
dev_t dev;
int cmd;
caddr_t arg;
int fflag;
struct proc *p;
{
int error;
int retval[2];
if(ss_debug) {
char cmd_type;
int cmd_ordinal;
static char **ioctl_strings;
unsigned int ioctl_type;
unsigned int ioctl_len;
static char *ioctl_type_strings[] = {
"NIOCxx", "SS_IO", "SS_IOR", "3?", "SS_IOW", "5?", "SS_IOWR"
};
static char *ioctl_S_strings[] = {
"0?", "SIOCSHIWAT", "SIOCGHIWAT", "SIOCSLOWAT", "SIOCGLOWAT",
"SIOCATMARK", "SIOCSPGRP", "SIOCGPGRP", "FIONREAD",
"FIONBIO", "FIOASYNC", "SIOCPROTO", "SIOCGETNAME",
"SIOCGETPEER", "IF_UNITSEL", "SIOCXPROTO"
};
static char *ioctl_R_strings[] = {
"0?", "1?", "2?", "3?", "4?", "5?", "6?", "7?", "8?",
"SIOCADDRT", "SIOCDELRT"
};
static char *ioctl_I_strings[] = {
"0?", "1?", "2?", "3?", "4?", "5?", "6?", "7?", "8?",
"9?", "10?", "SIOCSIFADDR", "SIOCGIFADDR", "SIOCSIFDSTADDR",
"SIOCGIFDSTADDR", "SIOCSIFFLAGS", "SIOCGIFFLAGS",
"SIOCGIFCONF", "18?", "19?", "20?", "SIOCSIFMTU",
"SIOCGIFMTU", "23?", "24?", "25?", "SIOCIFDETACH",
"SIOCGENPSTATS", "28?", "SIOCX25XMT", "SS_SIOCX25RCV",
"SS_SIOCX25TBL", "SIOCGIFBRDADDR" ,"SIOCSIFBRDADDR",
"SIOCGIFNETMASK", "SIOCSIFNETMASK", "SIOCGIFMETRIC",
"SIOCSIFMETRIC", "SIOCSARP", "SIOCGARP", "SIOCDARP",
"SIOCSIFNAME", "SIOCGIFONEP", "SIOCSIFONEP ",
"44?", "45?", "46?", "47?", "48?", "49?", "50?", "51?",
"52?", "53?", "54?", "55?", "56?", "57?", "58?", "59?",
"60?", "61?", "62?", "63?", "64?", "SIOCGENADDR",
"SIOCSOCKSYS"
};
static char *ioctl_NIOC_strings[] = {
"0?", "NIOCNFSD", "NIOCOLDGETFH", "NIOCASYNCD",
"NIOCSETDOMNAM", "NIOCGETDOMNAM", "NIOCCLNTHAND",
"NIOCEXPORTFS", "NIOCGETFH", "NIOCLSTAT"
};
cmd_ordinal = cmd & 0xff;
cmd_type = (cmd >> 8) & 0xff;
switch (cmd_type) {
case 0:
ioctl_strings = ioctl_NIOC_strings;
cmd_type = ' ';
if (cmd_ordinal > 9)
cmd_ordinal = -1;
break;
case 'S':
ioctl_strings = ioctl_S_strings;
if (cmd_ordinal > 15)
cmd_ordinal = -1;
break;
case 'R':
ioctl_strings = ioctl_R_strings;
if (cmd_ordinal > 10)
cmd_ordinal = -1;
break;
case 'I':
ioctl_strings = ioctl_I_strings;
if (cmd_ordinal > 66)
cmd_ordinal = -1;
break;
default:
cmd_type = '?';
break;
}
ioctl_type = (0xe0000000 & cmd) >> 29;
ioctl_len = (cmd >> 16) & SS_IOCPARM_MASK;
printf("sockioctl: [%d] ",p->p_pid);
if(cmd_type != '?'){
if(cmd_ordinal != -1)
printf("%s %s('%c',%d,l=%d) ",ioctl_strings[cmd_ordinal],
ioctl_type_strings[ioctl_type],
cmd_type,
cmd_ordinal,
ioctl_len);
else {
cmd_ordinal = cmd & 0xff;
printf("[unknown ordinal %d] %s('%c',%d,l=%d) ",cmd_ordinal,
ioctl_type_strings[ioctl_type],
cmd_type,
cmd_ordinal,
ioctl_len);
}
}
else {
printf("? %s('%c',%d,l=%d) ",
ioctl_type_strings[ioctl_type],
cmd_type,
cmd_ordinal,
ioctl_len);
}
printf("0x%x (0x%x) <0x%x>\n",
dev, cmd, arg);
}
if(*(((int *)arg) - 3) != IBCS2_MAGIC_IN){
printf("sockioctl: bad magic (sys_generic.c has no socksys mods?)\n");
return(EINVAL);
}
switch (cmd) {
case SS_SIOCSOCKSYS: /* ss syscall */
return ss_syscall(arg, p);
/* NIOCxx: These ioctls are really just integers
* (no other information to go on).
*/
case NIOCSETDOMNAM: {
struct sgdomarg domargs;
if(error = copyin((caddr_t)*((caddr_t *)arg), (caddr_t)&domargs, sizeof(struct sgdomarg)))
return(error);
arg = (caddr_t)&domargs;
SYSCALL_N(SYS_setdomainname, 0, 0);
break;
}
case NIOCGETDOMNAM: {
struct sgdomarg domargs;
if(error = copyin((caddr_t)*((caddr_t *)arg), (caddr_t)&domargs, sizeof(struct sgdomarg)))
return(error);
arg = (caddr_t)&domargs;
SYSCALL_N(SYS_getdomainname, 0, 0);
break;
}
case NIOCLSTAT: {
struct lstatarg st;
if(error = copyin((caddr_t)*((caddr_t *)arg), (caddr_t)&st, sizeof(struct lstatarg)))
return(error);
/* DO WE HAVE A FOREIGN LSTAT */
/* return mumbo_lstat(st.fname, st.statb); */
return (EINVAL);
}
case NIOCNFSD:
case NIOCOLDGETFH:
case NIOCASYNCD:
case NIOCCLNTHAND:
case NIOCEXPORTFS:
case NIOCGETFH:
return (EINVAL);
case SS_IF_UNITSEL: /* set unit number */
case SS_SIOCXPROTO: /* empty proto table */
case SS_SIOCIFDETACH: /* detach interface */
case SS_SIOCGENPSTATS: /* get ENP stats */
case SS_SIOCSIFNAME: /* set interface name */
case SS_SIOCGIFONEP: /* get one-packet params */
case SS_SIOCSIFONEP: /* set one-packet params */
case SS_SIOCPROTO: /* link proto */
case SS_SIOCX25XMT:
case SS_SIOCX25RCV:
case SS_SIOCX25TBL:
printf("sockioctl: [%d] unsupported ioctl 0x%x , 0x%lx\n",
p->p_pid,
cmd, (unsigned long)arg);
return (EINVAL);
default:
printf("sockioctl: [%d] unknown ioctl 0x%x , 0x%lx\n",
p->p_pid,
cmd, (unsigned long)arg);
return (EINVAL);
}
IBCS2_MAGIC_RETURN(arg);
}
int sockopen(dev, mode, devtype, p)
dev_t dev;
int mode;
int devtype;
struct proc *p;
{
if(ss_debug)
printf("sockopen: [%d] 0x%x\n", p->p_pid, dev);
/* minor = 0 is the socksys device itself. No special handling
* will be needed as it is controlled by the application
* via ioctls.
*/
if (minor(dev) == 0)
return 0;
/* minor = 1 is the spx device. This is the client side of a
* streams pipe to the X server. Under SCO and friends
* the library code messes around setting the connection
* up itself. We do it ourselves - this means we don't
* need to worry about the implementation of the server
* side (/dev/X0R - which must exist but can be a link
* to /dev/null) nor do we need to actually implement
* getmsg/putmsg.
*/
{ /* SPX */
int fd, error, args[3];
int retval[2];
#define SUN_LEN(su) \
(sizeof(*(su)) - sizeof((su)->sun_path) + strlen((su)->sun_path)) + 1
struct sockaddr_un *Xaddr = (struct sockaddr_un *)UA_ALLOC();
retval[0] = retval[1] = 0;
if(ss_debug)
printf("sockopen: SPX: [%d] opening\n", p->p_pid);
/* Grab a socket. */
if(ss_debug)
printf("sockopen: SPX: [%d] get a unix domain socket\n",
p->p_pid);
args[0] = AF_UNIX;
args[1] = SOCK_STREAM;
args[2] = 0;
error = SYSCALLX(SYS_socket, args);
if (error)
return error;
fd = retval[0];
if(fd < 1) {
printf("sockopen: SPX: [%d] unexpected fd of %d\n",
p->p_pid, fd);
return(EOPNOTSUPP); /* MRL whatever */
}
/* Connect the socket to X. */
if(ss_debug)
printf("sockopen: SPX: [%d] connect to /tmp/X11-unix/X0\n",
p->p_pid);
args[0] = fd;
Xaddr->sun_family = AF_UNIX;
copyout("/tmp/.X11-unix/X0", Xaddr->sun_path, 18);
Xaddr->sun_len = SUN_LEN(Xaddr);
args[1] = (int)Xaddr;
args[2] = sizeof(struct sockaddr_un);
error = SYSCALLX(SYS_connect, args);
if (error) {
(void)SYSCALLX(SYS_close, &fd);
return error;
}
put_socket_fops(p,fd);
return 0;
} /* SPX */
}
int sockclose(dev, flag, mode, p)
dev_t dev;
int flag;
int mode;
struct proc *p;
{
if(ss_debug)
printf("sockclose: [%d] 0x%x\n", p->p_pid, dev);
return(0);
}
static
int ss_fop_close(struct file *fp, struct proc *p)
{
int fd;
struct filedesc *fdp;
if(ss_debug){
fdp = p->p_fd;
fd = -1;
while(++fd < NOFILE)
if ( fp == fdp->fd_ofiles[fd] )
break;
printf("ss_fop_close: [%d] fd=%d ", p->p_pid, fd);
}
if(fp->f_type == DTYPE_SOCKET) {
if(ss_debug)
printf("is a socket\n");
return(close_s(fp, p));
}
else {
if(ss_debug)
printf("is not a socket\n");
return(ENOTSOCK);
}
}
void put_socket_fops(struct proc *p, int fd)
{
struct filedesc *fdp;
struct file *fp;
fdp = p->p_fd;
fp = fdp->fd_ofiles[fd];
if (ss_socket_fops.fo_ioctl != fp->f_ops->fo_ioctl) {
bcopy(fp->f_ops, &ss_socket_fops, sizeof(struct fileops));
ioctl_s = ss_socket_fops.fo_ioctl; /* save standard ioctl */
close_s = ss_socket_fops.fo_close; /* save standard close */
ss_socket_fops.fo_ioctl = ss_fop_ioctl;
ss_socket_fops.fo_close = ss_fop_close;
}
fp->f_ops = &ss_socket_fops;
return;
}
int ss_SYSCALL(n,convert_arg,indicator,arg,p,retval)
int n; /* syscall ordinal */
int convert_arg; /* if not 0, argument to convert */
int indicator; /* type of argument to convert */
int *arg; /* address of alien arg */
struct proc *p;
int *retval;
{
int error;
int rc;
if(convert_arg){
if(rc = ss_convert_struct( (caddr_t)*(arg + convert_arg),
indicator,
SS_ALIEN_TO_NATIVE))
return(rc);
error = (*sysent[n].sy_call)(p, arg + 1, retval);
rc = ss_convert_struct( (caddr_t)*(arg + convert_arg),
indicator,
SS_NATIVE_TO_ALIEN);
if(ss_debug)
printf("ss_SYSCALL: [%d] error=%d, rc=%d\n",
p->p_pid, error, rc);
}
else {
rc = 0;
error = (*sysent[n].sy_call)(p, arg + 1, retval);
if(ss_debug)
printf("ss_SYSCALL: [%d] error=%d\n",p->p_pid, error);
}
return(error ? error : rc);
}
int ss_IOCTL(fp, cmd, arg, p)
struct file *fp;
int cmd;
int *arg; /* address of alien arg */
struct proc *p;
{
int error, rc;
int these[2];
char cmd_type;
int cmd_ordinal;
int indicator;
cmd_type = (cmd >> 8) & 0xff;
cmd_ordinal = cmd & 0xff;
these[0] = cmd_type;
these[1] = cmd_ordinal;
if(ss_debug > 1)
printf("ss_IOCTL: calling ss_convert with %d(%c) %d\n",
these[0],these[0],these[1]);
indicator = ss_convert( struct_whatevers, these, 0);
if(ss_debug > 1)
printf("ss_IOCTL: ss_convert returns indicator %d\n",indicator);
if(indicator){
error = ss_convert_struct((caddr_t)*(arg + 2),
indicator,
SS_ALIEN_TO_NATIVE);
if(ss_debug > 1)
printf("ss_IOCTL: ss_convert_struct returns %d\n",error);
if(error)
return(error);
/* change len in ioctl now - in the general case */
error = ioctl_s(fp, cmd, (caddr_t)arg, p);
rc = ss_convert_struct( (caddr_t)*(arg + 2),
indicator,
SS_NATIVE_TO_ALIEN);
if(ss_debug)
printf("ss_IOCTL: [%d] error=%d, rc=%d\n",p->p_pid,
error, rc);
}
else {
rc = 0;
error = ioctl_s(fp, cmd, (caddr_t)arg, p);
if(ss_debug)
printf("ss_IOCTL: [%d] error=%d\n",p->p_pid, error);
}
return(error ? error : rc);
}
struct ss_socketopt_args {
int s;
int level;
int name;
caddr_t val;
int valsize;
};
int
ss_setsockopt(arg, ret, p)
struct ss_socketopt_args *arg;
int *ret;
struct proc *p;
{
int error, optname;
int retval[2];
if (arg->level != 0xffff) /* FreeBSD, SCO and ? */
return (ENOPROTOOPT);
optname = ss_convert(sopt_whatevers, &arg->name, 0);
switch (optname) {
case SO_ACCEPTCONN:
case SO_BROADCAST:
case SO_DEBUG:
case SO_DONTROUTE:
case SO_LINGER:
case SO_KEEPALIVE:
case SO_OOBINLINE:
case SO_RCVBUF:
case SO_RCVLOWAT:
case SO_RCVTIMEO:
case SO_REUSEADDR:
case SO_SNDBUF:
case SO_SNDLOWAT:
case SO_SNDTIMEO:
case SO_USELOOPBACK:
error = SYSCALLX(SYS_setsockopt, arg);
*ret = retval[0];
*(ret + 1) = retval[1];
return(error);
case SO_ERROR:
case SO_IMASOCKET:
case SO_NO_CHECK:
case SO_ORDREL:
case SO_PRIORITY:
case SO_PROTOTYPE:
case SO_TYPE:
return (ENOPROTOOPT);
}
return (ENOPROTOOPT);
}
int
ss_getsockopt(arg, ret, p)
struct ss_socketopt_args *arg;
int *ret;
struct proc *p;
{
int error, optname;
int retval[2];
if (arg->level != 0xffff) /* FreeBSD, SCO and ? */
return (ENOPROTOOPT);
optname = ss_convert(sopt_whatevers, &arg->name, 0);
switch (optname) {
case SO_ACCEPTCONN:
case SO_BROADCAST:
case SO_DEBUG:
case SO_DONTROUTE:
case SO_ERROR:
case SO_KEEPALIVE:
case SO_LINGER:
case SO_OOBINLINE:
case SO_RCVBUF:
case SO_RCVLOWAT:
case SO_RCVTIMEO:
case SO_REUSEADDR:
case SO_SNDBUF:
case SO_SNDLOWAT:
case SO_SNDTIMEO:
case SO_TYPE:
case SO_USELOOPBACK:
error = SYSCALLX(SYS_getsockopt, arg);
*ret = retval[0];
*(ret + 1) = retval[1];
return(error);
case SO_PROTOTYPE: {
int value = 0;
error = copyout((caddr_t)&value, (caddr_t)arg->s, sizeof(int));
return(error);
}
case SO_IMASOCKET: {
int value = 1;
error = copyout((caddr_t)&value, (caddr_t)arg->s, sizeof(int));
return(error);
}
case SO_NO_CHECK:
case SO_ORDREL:
case SO_PRIORITY:
return (ENOPROTOOPT);
}
return (ENOPROTOOPT);
}
#define SS_CONVERT
int system_type = SS_FREEBSD; /* FreeBSD */
int
ss_convert(what, this, otherwise)
struct whatever **what;
int *this;
int otherwise;
{
struct whatever *specific;
if(!(specific = what[system_type]))
return *this;
for (; specific->from != -1; specific++)
if(specific->from <= *this && *this <= specific->to)
if(specific->from == specific->to){
if(specific->more){
specific = specific->more;
this++;
continue;
}
else {
return((int)specific->conversion);
}
}
else {
return(specific->conversion ? (
specific->all_the_same ? (int)specific->conversion : specific->conversion[*this - specific->from] ) : *this);
}
return otherwise;
}
/* Returns 0 - no conversion, no pointer modification
1 - converted, relevant pointer modification
-1 - error
*/
int
ss_convert_struct(alien, indicator, direction)
char *alien;
int indicator;
int direction;
{
int error, len;
switch (system_type) {
case SS_FREEBSD:
return(0);
case SS_SYSVR4:
case SS_SYSVR3:
case SS_SCO_32:
case SS_WYSE_321:
case SS_ISC:
case SS_LINUX:
switch(direction){
case SS_ALIEN_TO_NATIVE:
error = ss_atn(alien, indicator);
if(ss_debug > 1)
printf("ss_convert: ATN ss_atn error %d\n",error);
return(error);
case SS_NATIVE_TO_ALIEN:
error = ss_nta(alien, indicator);
if(ss_debug > 1)
printf("ss_convert: NTA ss_nta error %d\n",error);
return(error);
}
default:
printf("ss_convert_struct: not expecting system_type %d\n", system_type);
break;
}
return(EINVAL);
}
/* note sockaddr_un linux unsigned short fam, 108 path
BSD uchar , uchar 104 */
int
ss_atn(alien, indicator)
char *alien;
int indicator;
{
int error;
switch (indicator) {
case SS_STRUCT_ARPREQ:
/* compatible */
return(0);
case SS_STRUCT_IFCONF:
/* compatible */
return(0);
case SS_STRUCT_IFREQ:
/* length OK - more unions - function dependent */
return(0);
case SS_STRUCT_ORTENTRY:
/* compatible */
return(0);
case SS_STRUCT_SOCKADDR:{
struct native_hdr {
u_char len;
u_char family;
};
union hdr_part {
struct native_hdr native;
u_short alien_family;
} hdr;
if(error = copyin((caddr_t)alien,(caddr_t)&hdr,sizeof(hdr)))
return(error);
if(ss_debug > 1)
printf("ss_atn:copyin 0x%x\n",hdr.alien_family);
if( hdr.alien_family < AF_MAX){
hdr.native.family = hdr.alien_family >> 8; /* 386 endianess */
/* OR LEN FOM A PARAM ? */
hdr.native.len = sizeof(struct sockaddr);
if(ss_debug > 1)
printf("ss_atn:copyout 0x%x\n",hdr.alien_family);
error = copyout((caddr_t)&hdr,(caddr_t)alien,sizeof(hdr));
return(error);
}
else {
printf("ss_atn: sa_family = %d\n", hdr.alien_family);
return(EINVAL);
}
}
case SS_STRUCT_SOCKNEWPROTO:
/* don't have */
printf("ss_atn: not expecting SS_STRUCT_SOCKNEWPROTO\n");
return(EINVAL);
default:
printf("ss_atn: not expecting case %d\n",indicator);
return(EINVAL);
}
}
/* note sockaddr_un linux unsigned short fam, 108 path
BSD uchar , uchar 104 */
int
ss_nta(alien, indicator)
char *alien;
int indicator;
{
int error;
switch (indicator) {
case SS_STRUCT_ARPREQ:
/* compatible */
return(0);
case SS_STRUCT_IFCONF:
/* compatible */
return(0);
case SS_STRUCT_IFREQ:
/* length OK - more unions - function dependent */
return(0);
case SS_STRUCT_ORTENTRY:
/* compatible */
return(0);
case SS_STRUCT_SOCKADDR:{
struct native_hdr {
u_char len;
u_char family;
};
union hdr_part {
struct native_hdr native;
u_short alien_family;
} hdr;
if(error = copyin((caddr_t)alien,(caddr_t)&hdr,sizeof(hdr)))
return(error);
if(ss_debug > 1)
printf("ss_nta:copyin 0x%x\n",hdr.alien_family);
hdr.alien_family = hdr.native.family;
if(ss_debug > 1)
printf("ss_nta:copyout 0x%x\n",hdr.alien_family);
error = copyout((caddr_t)&hdr,(caddr_t)alien,sizeof(hdr));
return(error);
}
case SS_STRUCT_SOCKNEWPROTO:
/* don't have */
printf("ss_nta: not expecting SS_STRUCT_SOCKNEWPROTO\n");
return(EINVAL);
default:
printf("ss_nta: not expecting case %d\n",indicator);
return(EINVAL);
}
}