[comp.sys.att] uipc socket patch#1

bes@holin.ATT.COM (Bradley Smith) (08/14/89)

Michael Gersten, sent me a nice piece of mail stating that he done
some select work and that his code didn't have the select return
when a close was called on the other side.  Well mine didn't either,
so here are some new update files (linesw.c, syscalls.c and interface.c)
which now allow close to get signaled on sockets and named pipes.

I haven't tried to close on ptys to see what happens, and I know that
you must call accept before select will work.  I am going to look at
this so one could use select to figure out when one might need to call
accept.  Also I need to work on the write bitmask.

=======cut here====
: This is a shar archive.  Extract with sh, not csh.
: This archive ends with exit, so do not worry about trailing junk.
echo 'Extracting interface.c'
sed 's/^X//' > interface.c << '+ END-OF-FILE interface.c'
X#ifndef LINT
Xstatic char * sccsdef = "@(#)interface.c	1.1	(Alex Crain) 6/20/89";
X#endif
X
X/*
X *  interface.c - loadable driver interface.
X *
X *  Written By Alex Crain.
X *
X *  This file contains the driver interface routines for the uipc driver, as
X *    discribed in drivers(7). These interface routines are:
X *	uipcinit ();
X *
X */
X
X#include <sys/types.h>
X#include <sys/systm.h>
X#include <sys/errno.h>
X#include <sys/user.h>
X#include <sys/conf.h>
X#include <uipc/socketvar.h>
X#include <uipc/mbuf.h>
X#include <uipc/protosw.h>
X#include <uipc/domain.h>
X#include <uipc/fproto.h>
X#include <uipc/pty.h>
X
X#include "sysent.h"
X
Xextern void (* sock_read)();
Xextern void (* sock_write)();
Xextern void (* sock_close)();
X
X/* stuff to wake up on */
Xextern int select_sleep, select_sleep_addr;
X
Xvoid
Xptyinit ()
X{
X   int i;
X
X   /*
X    * add our own system call processor.
X    */
X
X   sysent[SYSL_LOCSYS].sy_call = (int (*)()) dosyscall;
X   sysent[SYSL_LOCSYS].sy_narg = 3;
X
X   /*
X    *  link to the existing hooks in the kernal.
X    */
X
X   sock_read = uipc_read;
X   sock_write = uipc_write;
X   sock_close = uipc_close;
X
X   /*
X    *  Initialize the system.
X    */
X
X   mbinit ();
X   domaininit ();
X
X   /*
X    * find out where the pty major device is
X    */
X   for(i=0;i<cdevcnt;i++) {
X	if(cdevsw[i].d_open == ptyopen)
X		pty_major = i; /* got it */
X   }
X   so_linesw_setup();
X}
X
Xvoid
Xdosyscall()
X{
X   int index = u.u_ap[0] - SYSL_FIRST;
X   /*
X    * Intercept our calls
X    */
X
X   if (index >= 0 && index <= (SYSL_LAST - SYSL_FIRST))
X    {
X       /*
X	*  syscall arguments are available via the users %sp, at u.u_ar0[15].
X	*  These arguments must be copied to the argument vector u.u_arg[]
X	*  for access by the kernal. Noting that the stack looks like:
X	*	%sp -> [ frame link, &67, arg1, arg2 ... ]
X	*/
X       int arg = 0;
X       int * ap = (int *) (u.u_ar0[15]) + 2;
X       while (arg < sysentries[index].sy_narg)
X	   u.u_arg[arg++] = fuword(ap++);
X
X       /*
X	*  Perform the call.
X	*/
X       (* sysentries[index].sy_call) ();
X    }
X   else
X       locsys ();
X}
X
Xint
Xuipcrelease ()
X{
X   int mbmem_ref;
X   int s = spl5();
X   struct mbuf * m;
X   
X   for (m = mbmem; m < &mbmem[NMBUF+1]; m++)
X       if (m->m_type == MT_SOCKET)
X	   if (soclose (mtod (m, struct socket *)))
X	        return EBUSY;
X   for (mbmem_ref =0, m = mbmem; m < &mbmem[NMBUF+1]; m++, mbmem_ref++)
X       if (m->m_type != MT_FREE)
X	{
X	   int * i;
X	   (void) printf ("\n\n\nUIPC: Illegal mbuf type %d.\n", m->m_type);
X	   (void) printf ("m = ([*|%x] [m_next|%x] [m_len|%x] [m_type|%x])\n",
X			  m, m->m_next, m->m_len, m->m_type);
X	   (void) printf ("mbmem_ref=%d, NMBUF=%d\n", mbmem_ref, NMBUF);
X	   for (i = mtod (m, int *); 
X		i < ((int *) ((caddr_t) m + MSIZE - MTAIL));
X		i+=4)
X	       (void) printf ("%x %x %x %x\n", *i, *(i+1), *(i+2), *(i+3));
X	   panic ("uipc_release");
X	}
X
X   splx (s);
X   so_linesw_release();
X   return 0;
X}
X
X/*
X *  rdwr() uses this for reading sockets. 
X *
X *  There are no arguments, pertinant info is available as:
X *	u.u_ap[0] -	The file descriptor number
X *	u.u_base - 	IO buffer address
X *	u.u_count -	size of buffer.
X *	u.u_segflg -	IO buffer location
X *
X *  Errors do not return, rather an error condition is handled with a longjmp
X *  to u.u_qsav, with some non-zero argument. the call will return -1 to the 
X *  user, passing the error in errno (u.u_error). Since the jump returns
X *  directly to trap(), we need to do any houskeeping here.
X */
X
Xvoid
Xuipc_write ()
X{
X   struct file * fp = getf (u.u_ap[0]);
X
X   u.u_error = sosend (filesock (fp), (struct mbuf *) 0, 0, (struct mbuf *) 0);
X   
X   /*
X    *  process errors
X    */
X
X   if (u.u_error)
X       longjmp (u.u_qsav, 1);
X}
X
Xvoid
Xuipc_read ()
X{
X   struct file * fp = getf (u.u_ap[0]);
X
X   u.u_error = 
X       soreceive (filesock (fp), (struct mbuf **) 0, 0, (struct mbuf **) 0);
X			  
X			  
X   
X   /*
X    *  process errors
X    */
X
X   if (u.u_error)
X       longjmp (u.u_qsav, 1);
X}
X
Xvoid
Xuipc_close (sp)
X  off_t sp;
X{
X   u.u_error = soclose (mtod (ptom (sp), struct socket *));
X
X   /* next check us for wakeup maybe? */
X   if(select_sleep) {
X	select_sleep = 0;
X	wakeup((caddr_t) &select_sleep_addr);
X   }
X   /*
X    *  process errors
X    */
X
X   if (u.u_error)
X       longjmp (u.u_qsav, 1);
X}
X
X
X/*
X *  General utilities for the BSD<->sysV mix.
X */
X
X#ifdef unixpc
X
Xasm("	global bzero	");
Xasm("bzero:		");
Xasm("	mov.l	4(%sp),%a0");
Xasm("	mov.w	10(%sp),%d0");
Xasm("	sub.w	&1,%d0	");
Xasm("	bmi	end	");
Xasm("loop:		");
Xasm("	mov.b	&0,(%a0)+");
Xasm("	dbf	%d0,loop");
Xasm("end:		");
Xasm("	rts		");
X
X#else
X
Xvoid
Xbzero (s, n)
X  char * s; 
X  int n;
X{
X   while (n--)
X       *s++ = '\0';
X}
X
X#endif
X
Xint
Xufavail ()
X{
X   int avail = 0, fd = 0;
X
X   for (fd = 0; fd < 80; fd++)
X       if (u.u_ofile[fd] == 0)
X	   avail++;
X
X   return avail;
X}
+ END-OF-FILE interface.c
chmod 'u=rw,g=r,o=r' 'interface.c'
echo '	-rw-r--r--   1 bes      HSJ         4757 Aug 13 13:08 interface.c        (as sent)'
echo '	\c'
/bin/ls -l interface.c
echo 'Extracting linesw.c'
sed 's/^X//' > linesw.c << '+ END-OF-FILE linesw.c'
X#include	<sys/types.h>
X#include	<sys/sysmacros.h>
X#include	<sys/conf.h>
X#include	<sys/tty.h>
X
Xint (*real_linesw_read)(), (*real_linesw_write)();
Xint (*real_linesw_l_input)();
Xint (*real_linesw_close)();
Xint so_linesw_read(), so_linesw_write();
Xint so_linesw_l_input();
Xint so_linesw_close();
X
X/* stuff to wake up on */
Xextern int select_sleep, select_sleep_addr;
X
X/* unsigned addr_win_tty = 384636; /* */
X
Xso_linesw_read(tp)
Xstruct tty *tp;
X{
X	extern int (*real_linesw_read) ();
X
X	/* next check us for wakeup maybe? */
X	if(select_sleep) {
X		select_sleep = 0;
X		wakeup((caddr_t) &select_sleep_addr);
X	}
X	(*real_linesw_read)(tp);
X}
Xso_linesw_close(tp)
Xstruct tty *tp;
X{
X	extern int (*real_linesw_close) ();
X
X	/* next check us for wakeup maybe? */
X	if(select_sleep) {
X		select_sleep = 0;
X		wakeup((caddr_t) &select_sleep_addr);
X	}
X	(*real_linesw_close)(tp);
X}
Xso_linesw_write(tp)
Xstruct tty *tp;
X{
X	extern int (*real_linesw_write) ();
X
X	/* next check us for wakeup maybe? */
X	if(select_sleep) {
X		select_sleep = 0;
X		wakeup((caddr_t) &select_sleep_addr);
X	}
X	(*real_linesw_write)(tp);
X}
Xso_linesw_l_input(tp)
Xstruct tty *tp;
X{
X
X	/* next check us for wakeup maybe? */
X	if(select_sleep) {
X		select_sleep = 0;
X		wakeup((caddr_t) &select_sleep_addr);
X	}
X	/* first call real input routine */
X	(*real_linesw_l_input)(tp);
X}
Xso_linesw_setup()
X{
X	extern int (*real_linesw_write) ();
X	extern int (*real_linesw_read) ();
X	extern int (*real_linesw_close) ();
X	extern int (*real_linesw_l_input) ();
X
X	/* setup to route tty ouput to here */
X	real_linesw_read = linesw[0].l_read;
X	linesw[0].l_read = so_linesw_read;
X
X	real_linesw_write = linesw[0].l_write;
X	linesw[0].l_write = so_linesw_write;
X
X	real_linesw_l_input = linesw[0].l_input;
X	linesw[0].l_input = so_linesw_l_input;
X
X	real_linesw_close = linesw[0].l_close;
X	linesw[0].l_close = so_linesw_close;
X
X
X}
Xso_linesw_release()
X{
X
X	/* setup to route tty ouput to here */
X	linesw[0].l_read = real_linesw_read;
X
X	linesw[0].l_write = real_linesw_write;
X
X	linesw[0].l_input = real_linesw_l_input;
X
X	linesw[0].l_close = real_linesw_close;
+ END-OF-FILE linesw.c
chmod 'u=rw,g=r,o=r' 'linesw.c'
echo '	-rw-r--r--   1 bes      HSJ         2071 Aug 13 13:23 linesw.c        (as sent)'
echo '	\c'
/bin/ls -l linesw.c
echo 'Extracting syscalls.c'
sed 's/^X//' > syscalls.c << '+ END-OF-FILE syscalls.c'
X#ifndef LINT
Xstatic char * sccsdef = "@(#)syscalls.c	1.1	(Alex Crain) 6/20/89";
X#endif
X
X/*
X *  syscalls.c - system call kernal interface routines.
X *
X *  Written by Alex Crain.
X *
X *  This file is based in the Berkeley file uipc_syscalls.c,
X *  but is *not* guarenteed to be in any way compatable. It is
X *  close enough to the Berkeley code that the following applies...
X *
X *  Copyright (c) 1982, 1986, 1988 Regents of the University of California.
X *  All rights reserved.
X * 
X *  Redistribution and use in source and binary forms are permitted
X *  provided that this notice is preserved and that due credit is given
X *  to the University of California at Berkeley. The name of the University
X *  may not be used to endorse or promote products derived from this
X *  software without specific prior written permission. This software
X *  is provided "as is" without express or implied warranty.
X *
X */
X
X#include <sys/types.h>
X#include <sys/param.h>
X#include <sys/systm.h>
X#include <sys/user.h>
X#include <sys/file.h>
X#include <sys/buf.h>
X#include <sys/errno.h>
X#include <sys/systm.h>
X#include <sys/tty.h>
X#include <uipc/mbuf.h>
X#include <uipc/socket.h>
X#include <uipc/socketvar.h>
X#include <uipc/domain.h>
X#include <uipc/protosw.h>
X#include <uipc/un.h>
X#include <uipc/fproto.h>
X#include <uipc/pty.h>
X
Xstruct file * getsock ();
X
X/*
X * socket (domain, type, protocol)
X *
X *   Create a socket and add it to the processes open file table. This involves
X * some creativity, because the file structure is really too small for our 
X * uses. The kernal only knows about inodes, so there is no f_type slot. 
X * Instead, the kernal looks for a NULL f_inode, which means that we are a 
X * socket. Unfortunately, this means that there is no room in the file 
X * structure for the socket address, so we keep all of our sockets in a linear
X * table, and store the table offset in f_offset, which has no meaning here 
X * anyway. (see the macros filesock() and sockoffet() in conf.h).
X */
X
Xint	select_sleep; /* value to say if we need to wakeup */
Xint	select_sleep_addr; /* address we select on if wait in select */
X/*
X * this is hard coded righ now....for test only...soon it will be
X * loaded in at boot time so that it is not hard coded
X */
Xint	so_win_major = 0; /* major device of window */
Xunsigned int so_win_tty = 0; /* address of wintty */
X
Xsosetup() /* setup variables */
X{
X   register struct a {
X	int w_major;
X	unsigned int w_tty;
X   } * uap = (struct a *) u.u_ap;
X
X   so_win_major = uap->w_major;
X   so_win_tty = uap->w_tty;
X}
X
Xsoselect ()
X{
X   register struct a {
X      int	nfds;
X      int	*readfds;
X      int	*writefds;
X      int	*execptfds;
X   } * uap = (struct a *) u.u_ap;
X   int i,mask, cnt, *rds, *wds;
X   int k,l, j;
X   struct file *fp;
X   struct inode *ip;
X   struct tty *tp;
X   struct socket *so;
X
X   rds = uap->readfds;
X   wds = uap->writefds;
X   if(rds) { /* have readbitmask */
X      for(cnt=0,i=0;i < uap->nfds; i++) {
X	u.u_error = 0; /* reset it */
X	mask = 1 << i;
X	if(*rds & mask) { /* is this one? */
X		fp = getsock(i);
X		if(fp != 0) { /* valid socket */
X			so = filesock(fp);
X			j = (SS_CANTRCVMORE | SS_CANTSENDMORE);
X		 	k = SS_ISCONNECTED;
X			/* first check for closed sockets
X			 * closed sockets appear only to have j set
X			 */
X			if((so->so_state & ~j) == 0) {
X				/* socket is close. mark as having
X				 * data on it, so a read will fail 
X				 */
X				cnt++;
X			}
X			/* next we check for a socket in accept state
X			 */
X			/* this really doesn't work */
X			else if( so->so_state  == 0 ) {
X				/* here we want to sleep */
X				*rds &= ~mask;
X			}
X			/* next we check for a state of incomming ie
X			 * accept needs to be called
X			 */
X
X			/* this doesn't work right now */
X
X			/* and this is for regular sockets already
X			 * connected
X			 */
X			else if( (so->so_rcv.sb_mb != 0) &&
X			    (so->so_rcv.sb_cc != 0)) {
X				/* has buffer & has chars */
X				cnt++;
X			} else {
X				*rds &= ~mask;
X			}
X		} else if((fp = getf(i)) != 0) { /* valid open file */
X			ip = fp->f_inode;
X			if(major(ip->i_rdev) == pty_major) {
X				/* got a pty file descriptor */
X				if(Master(ip->i_rdev) == True) {
X					/* get slot in tty table */
X					k = minor(ip->i_rdev) - PTYCNT;
X					tp = &pts_tty[k]; /* ok */
X				}else { /* normal slot */
X					k = minor(ip->i_rdev) - PTYCNT;
X					tp = &pts_tty[k]; /* ok */
X				}
X				/* check buffer address */
X				if(tp->t_tbuf.c_count) {
X					/* ok to read */
X					cnt++;
X				}else {
X					*rds &= ~mask;
X				}
X			}else if(major(ip->i_rdev) == so_win_major) {
X				/* got a window file descriptor */
X				/* take off 1 for aligment */
X				k = minor(ip->i_rdev) - 1;
X				k *= sizeof(struct tty);
X				tp = (struct tty *) (unsigned) (so_win_tty + k);
X				/* check buffer */
X				k = tp->t_rawq.c_cc;
X				if(k)
X					cnt++;
X				else
X					*rds &= ~mask;
X			}
X				
X			/* here we have named pipes */
X			else if(ip->i_mode & IFIFO) {
X				/* if i_count == 1 no opens left
X				 * we notify select of this
X				 */
X				if (ip->i_count == 1) 
X					cnt++;
X				else if (ip->i_fwptr) 
X					cnt++;
X				else
X					*rds &= ~mask;
X			} else {
X				*rds &= ~mask;
X			}
X		} else {
X			*rds &= ~mask;
X		}
X	}
X      }
X   }
X   if(cnt) {
X	   u.u_rval1 = cnt;
X	   return;
X   }
X   select_sleep = 1;
X   /* sleep until worked up */
X   sleep( (caddr_t) &select_sleep_addr, PZERO+1 );
X   /* we are here so we let the user level know that we are ready */
X   u.u_rval1 = 0;
X   return;
X}
X
Xsocket ()
X{
X   register struct a {
X      int	domain;
X      int 	type;
X      int	proto;
X   } * uap = (struct a *) u.u_ap;
X
X   struct socket * so;
X   struct file *fp;
X
X   if ((fp = falloc ((struct inode *) 0, FREAD| FWRITE)) == NULL)
X       return;
X
X   if (u.u_error = socreate (uap->domain, &so, uap->type, uap->proto))
X       goto bad;
X
X   fp->f_offset = sockoffset (so);
X   return;
X
X bad:
X   u.u_ofile[u.u_rval1] = 0;
X   fp->f_count = 0;
X   fp->f_next = ffreelist;
X   ffreelist = fp;
X}
X
Xbind ()
X{
X   struct a {
X      int	s;
X      caddr_t	name;
X      int	namelen;
X   } * uap = (struct a *) u.u_ap;
X
X   struct file * fp;
X   struct mbuf * nam;
X
X   if ((fp = getsock (uap->s)) == 0)
X       return;
X
X   if (u.u_error = sockargs (&nam, uap->name, uap->namelen, MT_SONAME))
X       return;
X
X   u.u_error = sobind (filesock (fp), nam);
X   m_freem (nam);
X}
X
Xlisten ()
X{
X   struct a {
X      int	s;
X      int	backlog;
X   } * uap = (struct a *) u.u_ap;
X   struct file * fp;
X
X   if ((fp = getsock (uap->s)) == 0)
X       return;
X
X   u.u_error = solisten (filesock (fp), uap->backlog);
X}
X
Xaccept ()
X{
X   struct a {
X      int	s;
X      caddr_t	name;
X      int	* anamelen;
X   } * uap = (struct a *) u.u_ap;
X   struct file * fp;
X   struct mbuf * nam;
X   int namelen;
X   int s;
X   struct socket * so;
X
X   if (uap->name == 0)
X       goto noname;
X
X   if (u.u_error = copyin ((caddr_t) uap->anamelen, (caddr_t) &namelen, 
X			   sizeof (namelen)))
X       return;
X
X   if (useracc ((caddr_t) uap->name, (u_int) namelen, UACC_WRITE) == 0)
X    {
X       u.u_error = EFAULT;
X       return;
X    }
X
X noname:
X   if ((fp = getsock (uap->s)) == 0)
X       return;
X   s = splnet ();
X   so = filesock (fp);
X   if ((so->so_options & SO_ACCEPTCONN) == 0)
X    {
X       u.u_error = EINVAL;
X       goto bad;
X    }
X   if ((so->so_state & SS_NBIO) && so->so_qlen == 0)
X    {
X       u.u_error = EWOULDBLOCK;
X       goto bad;
X    }
X   while (so->so_qlen == 0 && so->so_error == 0)
X    {
X       if (so->so_state & SS_CANTRCVMORE)
X	{
X	   so->so_error = ECONNABORTED;
X	   break;
X	}
X       sleep ((caddr_t) &so->so_timeo, PZERO+1);
X    }
X
X   if (so->so_error)
X    {
X       u.u_error = so->so_error;
X       so->so_error = 0;
X       goto bad;
X    }
X
X   if ((fp = falloc ((struct inode *) 0, FREAD| FWRITE)) == 0)
X       goto bad;
X   else
X    {
X       struct socket * so2 = so->so_q;
X       if (soqremque (so2, 1) == 0)
X	   panic ("accept");
X       so = so2;
X    }
X
X   fp->f_offset = sockoffset (so);
X   nam = m_get (M_WAIT, MT_SONAME);
X   (void) soaccept (so, nam);
X
X   if (uap->name)
X    {
X       if (namelen > nam->m_len)
X	   namelen = nam->m_len;
X       (void) copyout (mtod (nam, caddr_t), (caddr_t) uap->name, 
X		       (u_int) namelen);
X       (void) copyout ((caddr_t) &namelen, (caddr_t) uap->anamelen,
X		       sizeof (*uap->anamelen));
X    }
X   m_freem (nam);
X
X bad:
X   splx (s);
X   return;
X}
X
Xconnect ()
X{
X   struct a {
X      int	s;
X      caddr_t	name;
X      int	namelen;
X   } * uap = (struct a *) u.u_ap;
X   struct file * fp;
X   struct socket * so;
X   struct mbuf * nam;
X   int s;
X
X   if ((fp = getsock (uap->s)) == 0)
X       return;
X
X   so = filesock (fp);
X
X   if ((so->so_state & SS_NBIO) &&
X       (so->so_state & SS_ISCONNECTING))
X    {
X       u.u_error = EALREADY;
X       return;
X    }
X
X   if (u.u_error = sockargs (&nam, uap->name, uap->namelen, MT_SONAME))
X       return;
X
X   if (u.u_error = soconnect (so, nam))
X       goto bad;
X
X   if ((so->so_state & SS_NBIO) &&
X       (so->so_state & SS_ISCONNECTING))
X    {
X       u.u_error = EINPROGRESS;
X       m_freem (nam);
X       return;
X    }
X
X   s = splnet ();
X
X   if (setjmp (u.u_qsav))
X    {
X       if (u.u_error == 0)
X	   u.u_error = EINTR;
X       goto bad2;
X    }
X
X   while ((so->so_state & SS_ISCONNECTING) && so->so_error == 0)
X       sleep ((caddr_t) &so->so_timeo, PZERO + 1);
X
X   u.u_error = so->so_error;
X   so->so_error = 0;
X
X bad2:
X   splx (s);
X
X bad:
X   so->so_state &= ~SS_ISCONNECTING;
X   m_freem (nam);
X}
X
Xsocketpair ()
X{
X   struct a {
X      int	domain;
X      int	type;
X      int	proto;
X      int	* rsv;
X   } * uap = (struct a *) u.u_ap;
X
X   register struct file * fp1, * fp2;
X   struct socket * so1, * so2;
X   int sv[2];
X
X   /*
X    *  verify that uap->rsv is in the users address space & writeable.
X    *  UACC_READ and UACC_WRITE are defined in <uipc/conf.h>.
X    */
X   if (useracc ((caddr_t) uap->rsv, sizeof (int) * 2, UACC_WRITE) == 0)
X    {
X       u.u_error = EFAULT;
X       return;
X    }
X
X   /*
X    *  Create some sockets (2).
X    */
X   if (u.u_error = socreate (uap->domain, &so1, uap->type, uap->proto))
X       return;
X
X   if (u.u_error = socreate (uap->domain, &so2, uap->type, uap->proto))
X       goto free1;
X
X   /*
X    *  assign them to file structures in the open file table.
X    */
X   if ((fp1 = falloc ((struct inode *) 0, FREAD | FWRITE)) == NULL)
X       goto free2;
X   sv[0] = u.u_rval1;
X   fp1->f_offset = sockoffset (so1);
X
X   if ((fp2 = falloc ((struct inode *) 0, FREAD | FWRITE)) == NULL)   
X       goto free3;
X   sv[1] = u.u_rval1;
X   fp2->f_offset = sockoffset (so2);
X
X   /* 
X    *  Connect them together.
X    */
X
X   if (u.u_error = soconnect2 (so1, so2))
X       goto free4;
X
X   /*
X    *  DATAGRAMS need to be connected both ways
X    */
X   if (uap->type == SOCK_DGRAM)
X       if (u.u_error = soconnect2 (so2, so1))
X	   goto free4;
X
X   /*
X    *  done, return 0 and pass the file descriptors back.
X    */
X   u.u_rval1 = 0;
X   copyout ((caddr_t) sv, (caddr_t) uap->rsv, 2 * sizeof (int));
X   return;
X
X free4:
X   fp2->f_count = 0;
X   fp2->f_next = ffreelist;
X   ffreelist = fp2;
X
X free3:
X   fp1->f_count = 0;
X   fp1->f_next = ffreelist;
X   ffreelist = fp1;
X
X free2:
X   (void) soclose (so2);
X   
X free1:
X   (void) soclose (so1);
X}
X
Xsendto ()
X{
X   struct a {
X      int	s;
X      caddr_t	buf;
X      int	len;
X      int	flags;
X      caddr_t	to;
X      int	tolen;
X   } * uap = (struct a *) u.u_ap;
X
X   struct msghdr msg;
X   
X   msg.msg_name = uap->to;
X   msg.msg_namelen = uap->tolen;
X   msg.msg_accrights = (caddr_t) 0;
X   msg.msg_accrightslen = 0;
X
X   u.u_base = uap->buf;
X   u.u_count = uap->len;
X   u.u_segflg = 0;
X
X   sendit (uap->s, &msg, uap->flags);
X}
X
Xsend ()
X{
X   struct a {
X      int	s;
X      caddr_t	buf;
X      int	len;
X      int	flags;
X   } * uap = (struct a *) u.u_ap;
X
X   struct msghdr msg;
X
X   msg.msg_name = (caddr_t) 0;
X   msg.msg_namelen = 0;
X   msg.msg_accrights = (caddr_t) 0;
X   msg.msg_accrightslen = 0;
X
X   u.u_base = uap->buf;
X   u.u_count = uap->len;
X   u.u_segflg = 0;
X      
X   sendit (uap->s, &msg, uap->flags);
X}
X
Xvoid
Xsendit (s, mp, flags)
X  int s;
X  struct msghdr * mp;
X  int flags;
X{
X   struct file * fp;
X   struct mbuf * to, * rights;
X
X   if ((fp = getsock (s)) == 0)
X       return;
X
X   if (u.u_count != 0 && useracc (u.u_base, u.u_count, UACC_READ) == 0)
X    {
X       u.u_error = EFAULT;
X       return;
X    }
X
X   if (mp->msg_name)
X    {
X       if (u.u_error = sockargs (&to, mp->msg_name, mp->msg_namelen,MT_SONAME))
X	   return; 
X    }
X   else
X      to = (struct mbuf *) 0;
X
X   if (mp->msg_accrights)
X    {
X       if (u.u_error = sockargs (&to, mp->msg_accrights, mp->msg_accrightslen,
X				 MT_SONAME))
X	   goto bad;
X    }
X   else
X       rights = (struct mbuf *) 0;
X
X   u.u_error = sosend (filesock (fp), to, flags, rights);
X
X   if (rights)
X       m_freem (rights);
X
X bad:
X   if (to)
X       m_freem (to);
X}
X
Xrecvfrom ()
X{
X   struct a {
X      int	s;
X      caddr_t	buf;
X      int	len;
X      int	flags;
X      caddr_t	from;
X      int	* fromlenaddr;
X   } * uap = (struct a *) u.u_ap;
X
X   struct msghdr msg;
X
X   msg.msg_name = uap->from;
X   if (u.u_error = copyin ((caddr_t) uap->fromlenaddr, 
X			   (caddr_t) &msg.msg_namelen, 
X			   sizeof (msg.msg_namelen)))
X       return;
X
X   msg.msg_accrights = (caddr_t) 0;
X   msg.msg_accrightslen = 0;
X
X   u.u_base = uap->buf;
X   u.u_count = uap->len;
X   u.u_segflg = 0;
X
X   recvit (uap->s, &msg, uap->flags, (caddr_t) uap->fromlenaddr, (caddr_t) 0);
X}
X
Xrecv ()
X{
X   struct a {
X      int	s;
X      caddr_t	buf;
X      int	len;
X      int	flags;
X   } * uap = (struct a *) u.u_ap;
X
X   struct msghdr msg;
X
X   msg.msg_name = (caddr_t) 0;
X   msg.msg_namelen = 0;
X   msg.msg_accrights = (caddr_t) 0;
X   msg.msg_accrightslen = 0;
X
X   u.u_base = uap->buf;
X   u.u_count = uap->len;
X   u.u_segflg = 0;
X
X   recvit (uap->s, &msg, uap->flags, (caddr_t) 0, (caddr_t) 0);
X}
X
Xvoid
Xrecvit (s, mp, flags, namelenp, rightslenp)
X  int s;
X  struct msghdr * mp;
X  int flags;
X  caddr_t namelenp, rightslenp;
X{
X   struct file * fp;
X   struct mbuf * from, * rights;
X   int len;
X
X   if ((fp = getsock (s)) == 0)
X       return;
X
X   if (u.u_count != 0 && useracc (u.u_base, u.u_count, UACC_WRITE) == 0)
X    {
X       u.u_error = EFAULT;
X       return;
X    }
X
X   u.u_error = soreceive (filesock (fp), &from, flags, &rights);
X
X   if (mp->msg_name)
X    {
X       len = mp->msg_namelen;
X       if (len <= 0 || from == (struct mbuf *) 0)
X	   len = 0;
X       else
X	{
X	   if (len > from->m_len)
X	        len = from->m_len;
X	   (void) copyout ((caddr_t) mtod (from, caddr_t), 
X			   (caddr_t) mp->msg_name, (unsigned) len);
X	}
X       (void) copyout ((caddr_t) &len, namelenp, sizeof (int));
X    }
X
X   if (mp->msg_accrights)
X    {
X       len = mp->msg_accrightslen;
X       if (len <= 0 || rights == (struct mbuf *) 0)
X	   len = 0;
X       else
X	{
X	   if (len > rights->m_len)
X	        len = rights->m_len;
X	   (void) copyout ((caddr_t) mtod (rights, caddr_t), 
X			   (caddr_t) mp->msg_accrights, (unsigned) len);
X	}
X       (void) copyout ((caddr_t) &len, rightslenp, sizeof (int));
X    }
X
X   if (rights)
X       m_freem (rights);
X   if (from)
X       m_freem (from);
X}
X
Xsetsockopt ()
X{
X   struct a {
X      int	s;
X      int	level;
X      int	name;
X      caddr_t	val;
X      int	valsize;
X   } * uap = (struct a *) u.u_ap;
X   struct file * fp;
X   struct mbuf * m = (struct mbuf *) 0;
X
X   if ((fp = getsock (uap->s)) == 0)
X       return;
X
X   if (uap->valsize > MLEN)
X    {
X       u.u_error = EINVAL;
X       return;
X    }
X   if (uap->val)
X    {
X       m = m_get (M_WAIT, MT_SOOPTS);
X       if (m == (struct mbuf *) 0)
X	{
X	   u.u_error = ENOBUFS;
X	   return;
X	}
X       if (u.u_error = copyin (uap->val, mtod (m, caddr_t), 
X			       (u_int) uap->valsize))
X	{
X	   (void) m_freem (m);
X	   return;
X	}
X       m->m_len = uap->valsize;
X    }
X   u.u_error = sosetopt (filesock (fp), uap->level, uap->name, m);
X}
X
Xgetsockopt ()
X{
X   struct a {
X      int	s;
X      int	level;
X      int	name;
X      caddr_t	val;
X      int	* avalsize;
X   } * uap = (struct a *) u.u_ap;
X   struct file * fp;
X   struct mbuf * m = (struct mbuf *) 0;
X   int valsize;
X
X   if ((fp = getsock (uap->s)) == 0)
X       return;
X
X   if (uap->val)
X    {
X       if (u.u_error = copyin ((caddr_t) uap->avalsize, (caddr_t) &valsize,
X			   sizeof (valsize)))
X	   return;
X    }
X   else
X       valsize = 0;
X
X   if (u.u_error = sogetopt (filesock (fp), uap->level, uap->name, &m))
X       goto bad;
X
X   if (uap->val && valsize && m != (struct mbuf *) 0)
X    {
X       if (valsize > m->m_len)
X	   valsize = m->m_len;
X       if (u.u_error = copyout (mtod (m, caddr_t), uap->val, (u_int) valsize))
X	   goto bad;
X       u.u_error = copyout ((caddr_t) &valsize, (caddr_t) uap->avalsize,
X			    sizeof (valsize));
X     }
X bad:
X   if (m != (struct mbuf *) 0)
X       (void) m_freem (m);
X}
X
Xsockpipe ()
X{
X   register struct file * fpr, * fpw;
X   struct socket * sor, * sow;
X   int r;
X
X   /*
X    *  Create some sockets (2).
X    */
X   if (u.u_error = socreate (AF_UNIX, &sor, SOCK_STREAM, 0))
X       return;
X
X   if (u.u_error = socreate (AF_UNIX, &sow, SOCK_STREAM, 0))
X       goto free1;
X
X   /*
X    *  assign them to file structures in the open file table.
X    */
X   if ((fpr = falloc ((struct inode *) 0, FREAD | FWRITE)) == NULL)
X       goto free2;
X   fpr->f_offset = sockoffset (sor);
X   r = u.u_rval1;
X
X   if ((fpw = falloc ((struct inode *) 0, FREAD | FWRITE)) == NULL)   
X       goto free3;
X   fpw->f_offset = sockoffset (sow);
X   u.u_rval2 = u.u_rval1;
X   u.u_rval1 = r;
X
X   /* 
X    *  Connect them together.
X    */
X
X   if (u.u_error = unp_connect2 (sow, sor))
X       goto free4;
X
X   /*
X    *  Close one direction.
X    */
X
X   sor->so_state |= SS_CANTSENDMORE;
X   sow->so_state |= SS_CANTRCVMORE;
X   return;
X
X free4:
X   fpw->f_count = 0;
X   fpw->f_next = ffreelist;
X   ffreelist = fpw;
X
X free3:
X   fpr->f_count = 0;
X   fpr->f_next = ffreelist;
X   ffreelist = fpr;
X
X free2:
X   (void) soclose (sow);
X   
X free1:
X   (void) soclose (sor);
X}
X
Xvoid
Xgetsockname ()
X{
X   struct a {
X      int	fdes;
X      caddr_t	asa;
X      int	* alen;
X   } * uap = (struct a *) u.u_ap;
X   struct file * fp;
X   struct socket * so;
X   struct mbuf * m;
X   int len;
X
X   if ((fp = getsock (uap->fdes)) == 0)
X       return;
X   if (u.u_error = copyin ((caddr_t) uap->alen, (caddr_t) &len, sizeof (len)))
X       return;
X   so = filesock (fp);
X   if ((m = m_getclr (M_WAIT, MT_SONAME)) == (struct mbuf *) 0)
X    {
X       u.u_error = ENOBUFS;
X       return;
X    }
X   if (u.u_error = (* so->so_proto->pr_usrreq) (so, PRU_SOCKADDR,
X		       (struct mbuf *) 0, m, (struct mbuf *) 0))
X       goto bad;
X   if (len > m->m_len)
X       len = m->m_len;
X   if (u.u_error = copyout (mtod (m, caddr_t), (caddr_t) uap->asa,
X			    (u_int) len))
X       goto bad;
X   u.u_error = copyout ((caddr_t) &len, (caddr_t) uap->alen, 
X			    sizeof (len));
X bad:
X   m_freem (m);
X}
X
X      
X/*
X *  System call helper functions
X */
X
Xint
Xsockargs (aname, name, namelen, type)
X  struct mbuf ** aname;
X  caddr_t name;
X  int namelen, type;
X{
X   struct mbuf * m;
X   int error;
X
X   if (namelen > MLEN)
X       return EINVAL;
X
X   if ((m = m_get (M_WAIT, type)) == NULL)
X       return ENOBUFS;
X
X   m->m_len = namelen;
X
X   if (error = copyin (name, mtod (m, caddr_t), (u_int) namelen))
X       (void) m_free (m);
X   else
X       * aname = m;
X
X   return error;
X}
X
X/* given a file descriptor see if it is a socket file descriptor */
Xstruct file *
Xgetsock (fd)
X  int fd;
X{
X   struct file * fp;
X
X   /* given an fd, see if it is a valid fd, ie in file table*/
X   if ((fp = getf (fd)) == NULL)
X       return 0;
X   
X   if (fp->f_inode)
X    {
X       u.u_error = ENOTSOCK;
X       return 0;
X    }
X   
X   return fp;
X}
+ END-OF-FILE syscalls.c
chmod 'u=rw,g=r,o=r' 'syscalls.c'
echo '	-rw-r--r--   1 bes      HSJ        19412 Aug 13 13:18 syscalls.c        (as sent)'
echo '	\c'
/bin/ls -l syscalls.c
exit 0
-- 
Bradley Smith
Computer Systems Offer Integration Laboratory
AT&T Bell Labs, Holmdel, NJ 
201-949-0090 att!holin!bes or bes@holin.ATT.COM