Home | History | Annotate | Line # | Download | only in rpc
svc.c revision 1.25
      1  1.25  christos /*	$NetBSD: svc.c,v 1.25 2005/09/09 15:41:27 christos Exp $	*/
      2   1.7       cgd 
      3   1.1       cgd /*
      4   1.1       cgd  * Sun RPC is a product of Sun Microsystems, Inc. and is provided for
      5   1.1       cgd  * unrestricted use provided that this legend is included on all tape
      6   1.1       cgd  * media and as a part of the software program in whole or part.  Users
      7   1.1       cgd  * may copy or modify Sun RPC without charge, but are not authorized
      8   1.1       cgd  * to license or distribute it to anyone else except as part of a product or
      9   1.1       cgd  * program developed by the user.
     10   1.1       cgd  *
     11   1.1       cgd  * SUN RPC IS PROVIDED AS IS WITH NO WARRANTIES OF ANY KIND INCLUDING THE
     12   1.1       cgd  * WARRANTIES OF DESIGN, MERCHANTIBILITY AND FITNESS FOR A PARTICULAR
     13   1.1       cgd  * PURPOSE, OR ARISING FROM A COURSE OF DEALING, USAGE OR TRADE PRACTICE.
     14   1.1       cgd  *
     15   1.1       cgd  * Sun RPC is provided with no support and without any obligation on the
     16   1.1       cgd  * part of Sun Microsystems, Inc. to assist in its use, correction,
     17   1.1       cgd  * modification or enhancement.
     18   1.1       cgd  *
     19   1.1       cgd  * SUN MICROSYSTEMS, INC. SHALL HAVE NO LIABILITY WITH RESPECT TO THE
     20   1.1       cgd  * INFRINGEMENT OF COPYRIGHTS, TRADE SECRETS OR ANY PATENTS BY SUN RPC
     21   1.1       cgd  * OR ANY PART THEREOF.
     22   1.1       cgd  *
     23   1.1       cgd  * In no event will Sun Microsystems, Inc. be liable for any lost revenue
     24   1.1       cgd  * or profits or other special, indirect and consequential damages, even if
     25   1.1       cgd  * Sun has been advised of the possibility of such damages.
     26   1.1       cgd  *
     27   1.1       cgd  * Sun Microsystems, Inc.
     28   1.1       cgd  * 2550 Garcia Avenue
     29   1.1       cgd  * Mountain View, California  94043
     30   1.1       cgd  */
     31   1.1       cgd 
     32  1.10  christos #include <sys/cdefs.h>
     33   1.1       cgd #if defined(LIBC_SCCS) && !defined(lint)
     34  1.10  christos #if 0
     35  1.10  christos static char *sccsid = "@(#)svc.c 1.44 88/02/08 Copyr 1984 Sun Micro";
     36  1.10  christos static char *sccsid = "@(#)svc.c	2.4 88/08/11 4.0 RPCSRC";
     37  1.10  christos #else
     38  1.25  christos __RCSID("$NetBSD: svc.c,v 1.25 2005/09/09 15:41:27 christos Exp $");
     39  1.10  christos #endif
     40   1.1       cgd #endif
     41   1.1       cgd 
     42   1.1       cgd /*
     43   1.1       cgd  * svc.c, Server-side remote procedure call interface.
     44   1.1       cgd  *
     45   1.1       cgd  * There are two sets of procedures here.  The xprt routines are
     46   1.1       cgd  * for handling transport handles.  The svc routines handle the
     47   1.1       cgd  * list of service routines.
     48   1.1       cgd  *
     49   1.1       cgd  * Copyright (C) 1984, Sun Microsystems, Inc.
     50   1.1       cgd  */
     51   1.1       cgd 
     52  1.11       jtc #include "namespace.h"
     53  1.20      fvdl #include "reentrant.h"
     54  1.20      fvdl #include <sys/types.h>
     55  1.20      fvdl #include <sys/poll.h>
     56  1.17     lukem #include <assert.h>
     57  1.14     lukem #include <errno.h>
     58   1.4       cgd #include <stdlib.h>
     59   1.8        pk #include <string.h>
     60   1.4       cgd 
     61   1.1       cgd #include <rpc/rpc.h>
     62  1.20      fvdl #ifdef PORTMAP
     63   1.1       cgd #include <rpc/pmap_clnt.h>
     64  1.20      fvdl #endif
     65  1.20      fvdl 
     66  1.23      fvdl #include "rpc_internal.h"
     67  1.11       jtc 
     68  1.11       jtc #ifdef __weak_alias
     69  1.19   mycroft __weak_alias(svc_getreq,_svc_getreq)
     70  1.19   mycroft __weak_alias(svc_getreqset,_svc_getreqset)
     71  1.20      fvdl __weak_alias(svc_getreq_common,_svc_getreq_common)
     72  1.19   mycroft __weak_alias(svc_register,_svc_register)
     73  1.20      fvdl __weak_alias(svc_reg,_svc_reg)
     74  1.20      fvdl __weak_alias(svc_unreg,_svc_unreg)
     75  1.19   mycroft __weak_alias(svc_sendreply,_svc_sendreply)
     76  1.19   mycroft __weak_alias(svc_unregister,_svc_unregister)
     77  1.19   mycroft __weak_alias(svcerr_auth,_svcerr_auth)
     78  1.19   mycroft __weak_alias(svcerr_decode,_svcerr_decode)
     79  1.19   mycroft __weak_alias(svcerr_noproc,_svcerr_noproc)
     80  1.19   mycroft __weak_alias(svcerr_noprog,_svcerr_noprog)
     81  1.19   mycroft __weak_alias(svcerr_progvers,_svcerr_progvers)
     82  1.19   mycroft __weak_alias(svcerr_systemerr,_svcerr_systemerr)
     83  1.19   mycroft __weak_alias(svcerr_weakauth,_svcerr_weakauth)
     84  1.19   mycroft __weak_alias(xprt_register,_xprt_register)
     85  1.19   mycroft __weak_alias(xprt_unregister,_xprt_unregister)
     86  1.23      fvdl __weak_alias(rpc_control,_rpc_control)
     87  1.11       jtc #endif
     88   1.1       cgd 
     89  1.23      fvdl SVCXPRT **__svc_xports;
     90  1.23      fvdl int __svc_maxrec;
     91   1.1       cgd 
     92   1.1       cgd #define	RQCRED_SIZE	400		/* this size is excessive */
     93   1.1       cgd 
     94  1.20      fvdl #define SVC_VERSQUIET 0x0001		/* keep quiet about vers mismatch */
     95  1.20      fvdl #define version_keepquiet(xp) ((u_long)(xp)->xp_p3 & SVC_VERSQUIET)
     96  1.20      fvdl 
     97   1.2   deraadt #define max(a, b) (a > b ? a : b)
     98   1.2   deraadt 
     99   1.1       cgd /*
    100   1.1       cgd  * The services list
    101   1.1       cgd  * Each entry represents a set of procedures (an rpc program).
    102   1.1       cgd  * The dispatch routine takes request structs and runs the
    103   1.1       cgd  * apropriate procedure.
    104   1.1       cgd  */
    105   1.1       cgd static struct svc_callout {
    106   1.1       cgd 	struct svc_callout *sc_next;
    107  1.20      fvdl 	rpcprog_t	    sc_prog;
    108  1.20      fvdl 	rpcvers_t	    sc_vers;
    109  1.20      fvdl 	char		   *sc_netid;
    110  1.10  christos 	void		    (*sc_dispatch) __P((struct svc_req *, SVCXPRT *));
    111   1.1       cgd } *svc_head;
    112   1.1       cgd 
    113  1.24   thorpej #ifdef _REENTRANT
    114  1.20      fvdl extern rwlock_t svc_lock;
    115  1.20      fvdl extern rwlock_t svc_fd_lock;
    116  1.20      fvdl #endif
    117  1.20      fvdl 
    118  1.20      fvdl static struct svc_callout *svc_find __P((rpcprog_t, rpcvers_t,
    119  1.20      fvdl 					 struct svc_callout **, char *));
    120  1.23      fvdl static void __xprt_do_unregister __P((SVCXPRT *xprt, bool_t dolock));
    121   1.1       cgd 
    122   1.1       cgd /* ***************  SVCXPRT related stuff **************** */
    123   1.3   deraadt 
    124   1.1       cgd /*
    125   1.1       cgd  * Activate a transport handle.
    126   1.1       cgd  */
    127   1.1       cgd void
    128   1.1       cgd xprt_register(xprt)
    129   1.1       cgd 	SVCXPRT *xprt;
    130   1.1       cgd {
    131  1.17     lukem 	int sock;
    132  1.17     lukem 
    133  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    134  1.17     lukem 
    135  1.20      fvdl 	sock = xprt->xp_fd;
    136   1.1       cgd 
    137  1.20      fvdl 	rwlock_wrlock(&svc_fd_lock);
    138  1.23      fvdl 	if (__svc_xports == NULL) {
    139  1.23      fvdl 		__svc_xports = (SVCXPRT **)
    140   1.1       cgd 			mem_alloc(FD_SETSIZE * sizeof(SVCXPRT *));
    141  1.23      fvdl 		if (__svc_xports == NULL)
    142  1.16     lukem 			return;
    143  1.23      fvdl 		memset(__svc_xports, '\0', FD_SETSIZE * sizeof(SVCXPRT *));
    144   1.1       cgd 	}
    145   1.2   deraadt 	if (sock < FD_SETSIZE) {
    146  1.23      fvdl 		__svc_xports[sock] = xprt;
    147   1.1       cgd 		FD_SET(sock, &svc_fdset);
    148   1.2   deraadt 		svc_maxfd = max(svc_maxfd, sock);
    149   1.1       cgd 	}
    150  1.20      fvdl 	rwlock_unlock(&svc_fd_lock);
    151   1.1       cgd }
    152   1.1       cgd 
    153  1.23      fvdl void
    154  1.23      fvdl xprt_unregister(SVCXPRT *xprt)
    155  1.23      fvdl {
    156  1.23      fvdl 	__xprt_do_unregister(xprt, TRUE);
    157  1.23      fvdl }
    158  1.23      fvdl 
    159  1.23      fvdl void
    160  1.23      fvdl __xprt_unregister_unlocked(SVCXPRT *xprt)
    161  1.23      fvdl {
    162  1.23      fvdl 	__xprt_do_unregister(xprt, FALSE);
    163  1.23      fvdl }
    164  1.23      fvdl 
    165   1.1       cgd /*
    166   1.1       cgd  * De-activate a transport handle.
    167   1.1       cgd  */
    168  1.23      fvdl static void
    169  1.23      fvdl __xprt_do_unregister(xprt, dolock)
    170   1.1       cgd 	SVCXPRT *xprt;
    171  1.23      fvdl 	bool_t dolock;
    172   1.1       cgd {
    173  1.17     lukem 	int sock;
    174  1.17     lukem 
    175  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    176  1.17     lukem 
    177  1.20      fvdl 	sock = xprt->xp_fd;
    178   1.1       cgd 
    179  1.23      fvdl 	if (dolock)
    180  1.23      fvdl 		rwlock_wrlock(&svc_fd_lock);
    181  1.23      fvdl 	if ((sock < FD_SETSIZE) && (__svc_xports[sock] == xprt)) {
    182  1.23      fvdl 		__svc_xports[sock] = NULL;
    183   1.1       cgd 		FD_CLR(sock, &svc_fdset);
    184  1.20      fvdl 		if (sock >= svc_maxfd) {
    185   1.3   deraadt 			for (svc_maxfd--; svc_maxfd>=0; svc_maxfd--)
    186  1.23      fvdl 				if (__svc_xports[svc_maxfd])
    187   1.3   deraadt 					break;
    188   1.3   deraadt 		}
    189   1.1       cgd 	}
    190  1.23      fvdl 	if (dolock)
    191  1.23      fvdl 		rwlock_unlock(&svc_fd_lock);
    192   1.1       cgd }
    193   1.1       cgd 
    194  1.20      fvdl /*
    195  1.20      fvdl  * Add a service program to the callout list.
    196  1.20      fvdl  * The dispatch routine will be called when a rpc request for this
    197  1.20      fvdl  * program number comes in.
    198  1.20      fvdl  */
    199  1.20      fvdl bool_t
    200  1.20      fvdl svc_reg(xprt, prog, vers, dispatch, nconf)
    201  1.20      fvdl 	SVCXPRT *xprt;
    202  1.20      fvdl 	const rpcprog_t prog;
    203  1.20      fvdl 	const rpcvers_t vers;
    204  1.20      fvdl 	void (*dispatch) __P((struct svc_req *, SVCXPRT *));
    205  1.20      fvdl 	const struct netconfig *nconf;
    206  1.20      fvdl {
    207  1.20      fvdl 	bool_t dummy;
    208  1.20      fvdl 	struct svc_callout *prev;
    209  1.21  christos 	struct svc_callout *s;
    210  1.20      fvdl 	struct netconfig *tnconf;
    211  1.21  christos 	char *netid = NULL;
    212  1.20      fvdl 	int flag = 0;
    213  1.20      fvdl 
    214  1.22     lukem 	_DIAGASSERT(xprt != NULL);
    215  1.22     lukem 	/* XXX: dispatch may be NULL ??? */
    216  1.22     lukem 
    217  1.20      fvdl /* VARIABLES PROTECTED BY svc_lock: s, prev, svc_head */
    218  1.20      fvdl 
    219  1.20      fvdl 	if (xprt->xp_netid) {
    220  1.20      fvdl 		netid = strdup(xprt->xp_netid);
    221  1.20      fvdl 		flag = 1;
    222  1.20      fvdl 	} else if (nconf && nconf->nc_netid) {
    223  1.20      fvdl 		netid = strdup(nconf->nc_netid);
    224  1.20      fvdl 		flag = 1;
    225  1.20      fvdl 	} else if ((tnconf = __rpcgettp(xprt->xp_fd)) != NULL) {
    226  1.20      fvdl 		netid = strdup(tnconf->nc_netid);
    227  1.20      fvdl 		flag = 1;
    228  1.20      fvdl 		freenetconfigent(tnconf);
    229  1.20      fvdl 	} /* must have been created with svc_raw_create */
    230  1.20      fvdl 	if ((netid == NULL) && (flag == 1)) {
    231  1.20      fvdl 		return (FALSE);
    232  1.20      fvdl 	}
    233  1.20      fvdl 
    234  1.20      fvdl 	rwlock_wrlock(&svc_lock);
    235  1.21  christos 	if ((s = svc_find(prog, vers, &prev, netid)) != NULL) {
    236  1.20      fvdl 		if (netid)
    237  1.20      fvdl 			free(netid);
    238  1.20      fvdl 		if (s->sc_dispatch == dispatch)
    239  1.20      fvdl 			goto rpcb_it; /* he is registering another xptr */
    240  1.20      fvdl 		rwlock_unlock(&svc_lock);
    241  1.20      fvdl 		return (FALSE);
    242  1.20      fvdl 	}
    243  1.21  christos 	s = mem_alloc(sizeof (struct svc_callout));
    244  1.21  christos 	if (s == NULL) {
    245  1.20      fvdl 		if (netid)
    246  1.20      fvdl 			free(netid);
    247  1.20      fvdl 		rwlock_unlock(&svc_lock);
    248  1.20      fvdl 		return (FALSE);
    249  1.20      fvdl 	}
    250  1.20      fvdl 
    251  1.20      fvdl 	s->sc_prog = prog;
    252  1.20      fvdl 	s->sc_vers = vers;
    253  1.20      fvdl 	s->sc_dispatch = dispatch;
    254  1.20      fvdl 	s->sc_netid = netid;
    255  1.20      fvdl 	s->sc_next = svc_head;
    256  1.20      fvdl 	svc_head = s;
    257  1.20      fvdl 
    258  1.20      fvdl 	if ((xprt->xp_netid == NULL) && (flag == 1) && netid)
    259  1.20      fvdl 		((SVCXPRT *) xprt)->xp_netid = strdup(netid);
    260  1.20      fvdl 
    261  1.20      fvdl rpcb_it:
    262  1.20      fvdl 	rwlock_unlock(&svc_lock);
    263  1.20      fvdl 	/* now register the information with the local binder service */
    264  1.20      fvdl 	if (nconf) {
    265  1.21  christos 		/*LINTED const castaway*/
    266  1.20      fvdl 		dummy = rpcb_set(prog, vers, (struct netconfig *) nconf,
    267  1.20      fvdl 		&((SVCXPRT *) xprt)->xp_ltaddr);
    268  1.20      fvdl 		return (dummy);
    269  1.20      fvdl 	}
    270  1.20      fvdl 	return (TRUE);
    271  1.20      fvdl }
    272  1.20      fvdl 
    273  1.20      fvdl /*
    274  1.20      fvdl  * Remove a service program from the callout list.
    275  1.20      fvdl  */
    276  1.20      fvdl void
    277  1.20      fvdl svc_unreg(prog, vers)
    278  1.20      fvdl 	const rpcprog_t prog;
    279  1.20      fvdl 	const rpcvers_t vers;
    280  1.20      fvdl {
    281  1.20      fvdl 	struct svc_callout *prev;
    282  1.21  christos 	struct svc_callout *s;
    283  1.20      fvdl 
    284  1.20      fvdl 	/* unregister the information anyway */
    285  1.20      fvdl 	(void) rpcb_unset(prog, vers, NULL);
    286  1.20      fvdl 	rwlock_wrlock(&svc_lock);
    287  1.21  christos 	while ((s = svc_find(prog, vers, &prev, NULL)) != NULL) {
    288  1.21  christos 		if (prev == NULL) {
    289  1.20      fvdl 			svc_head = s->sc_next;
    290  1.20      fvdl 		} else {
    291  1.20      fvdl 			prev->sc_next = s->sc_next;
    292  1.20      fvdl 		}
    293  1.21  christos 		s->sc_next = NULL;
    294  1.20      fvdl 		if (s->sc_netid)
    295  1.21  christos 			mem_free(s->sc_netid, sizeof (s->sc_netid) + 1);
    296  1.21  christos 		mem_free(s, sizeof (struct svc_callout));
    297  1.20      fvdl 	}
    298  1.20      fvdl 	rwlock_unlock(&svc_lock);
    299  1.20      fvdl }
    300   1.1       cgd 
    301   1.1       cgd /* ********************** CALLOUT list related stuff ************* */
    302   1.1       cgd 
    303  1.20      fvdl #ifdef PORTMAP
    304   1.1       cgd /*
    305   1.1       cgd  * Add a service program to the callout list.
    306   1.1       cgd  * The dispatch routine will be called when a rpc request for this
    307   1.1       cgd  * program number comes in.
    308   1.1       cgd  */
    309   1.1       cgd bool_t
    310   1.1       cgd svc_register(xprt, prog, vers, dispatch, protocol)
    311   1.1       cgd 	SVCXPRT *xprt;
    312  1.13     lukem 	u_long prog;
    313  1.13     lukem 	u_long vers;
    314  1.10  christos 	void (*dispatch) __P((struct svc_req *, SVCXPRT *));
    315   1.1       cgd 	int protocol;
    316   1.1       cgd {
    317   1.1       cgd 	struct svc_callout *prev;
    318  1.14     lukem 	struct svc_callout *s;
    319   1.1       cgd 
    320  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    321  1.17     lukem 	_DIAGASSERT(dispatch != NULL);
    322  1.17     lukem 
    323  1.21  christos 	if ((s = svc_find((rpcprog_t)prog, (rpcvers_t)vers, &prev, NULL)) !=
    324  1.21  christos 	    NULL) {
    325   1.1       cgd 		if (s->sc_dispatch == dispatch)
    326   1.1       cgd 			goto pmap_it;  /* he is registering another xptr */
    327   1.1       cgd 		return (FALSE);
    328   1.1       cgd 	}
    329  1.21  christos 	s = mem_alloc(sizeof(struct svc_callout));
    330  1.21  christos 	if (s == NULL) {
    331   1.1       cgd 		return (FALSE);
    332   1.1       cgd 	}
    333  1.21  christos 	s->sc_prog = (rpcprog_t)prog;
    334  1.21  christos 	s->sc_vers = (rpcvers_t)vers;
    335   1.1       cgd 	s->sc_dispatch = dispatch;
    336   1.1       cgd 	s->sc_next = svc_head;
    337   1.1       cgd 	svc_head = s;
    338   1.1       cgd pmap_it:
    339   1.1       cgd 	/* now register the information with the local binder service */
    340   1.1       cgd 	if (protocol) {
    341   1.1       cgd 		return (pmap_set(prog, vers, protocol, xprt->xp_port));
    342   1.1       cgd 	}
    343   1.1       cgd 	return (TRUE);
    344   1.1       cgd }
    345   1.1       cgd 
    346   1.1       cgd /*
    347   1.1       cgd  * Remove a service program from the callout list.
    348   1.1       cgd  */
    349   1.1       cgd void
    350   1.1       cgd svc_unregister(prog, vers)
    351  1.13     lukem 	u_long prog;
    352  1.13     lukem 	u_long vers;
    353   1.1       cgd {
    354   1.1       cgd 	struct svc_callout *prev;
    355  1.14     lukem 	struct svc_callout *s;
    356   1.1       cgd 
    357  1.21  christos 	if ((s = svc_find((rpcprog_t)prog, (rpcvers_t)vers, &prev, NULL)) ==
    358  1.21  christos 	    NULL)
    359   1.1       cgd 		return;
    360  1.21  christos 	if (prev == NULL) {
    361   1.1       cgd 		svc_head = s->sc_next;
    362   1.1       cgd 	} else {
    363   1.1       cgd 		prev->sc_next = s->sc_next;
    364   1.1       cgd 	}
    365  1.21  christos 	s->sc_next = NULL;
    366  1.15  christos 	mem_free(s, sizeof(struct svc_callout));
    367   1.1       cgd 	/* now unregister the information with the local binder service */
    368   1.1       cgd 	(void)pmap_unset(prog, vers);
    369   1.1       cgd }
    370  1.20      fvdl #endif /* PORTMAP */
    371   1.1       cgd 
    372   1.1       cgd /*
    373   1.1       cgd  * Search the callout list for a program number, return the callout
    374   1.1       cgd  * struct.
    375   1.1       cgd  */
    376   1.1       cgd static struct svc_callout *
    377  1.20      fvdl svc_find(prog, vers, prev, netid)
    378  1.20      fvdl 	rpcprog_t prog;
    379  1.20      fvdl 	rpcvers_t vers;
    380   1.1       cgd 	struct svc_callout **prev;
    381  1.20      fvdl 	char *netid;
    382   1.1       cgd {
    383  1.14     lukem 	struct svc_callout *s, *p;
    384   1.1       cgd 
    385  1.17     lukem 	_DIAGASSERT(prev != NULL);
    386  1.22     lukem 	/* netid is handled below */
    387  1.17     lukem 
    388  1.21  christos 	p = NULL;
    389  1.21  christos 	for (s = svc_head; s != NULL; s = s->sc_next) {
    390  1.20      fvdl 		if (((s->sc_prog == prog) && (s->sc_vers == vers)) &&
    391  1.20      fvdl 		    ((netid == NULL) || (s->sc_netid == NULL) ||
    392  1.20      fvdl 		    (strcmp(netid, s->sc_netid) == 0)))
    393  1.20      fvdl 			break;
    394   1.1       cgd 		p = s;
    395   1.1       cgd 	}
    396   1.1       cgd 	*prev = p;
    397   1.1       cgd 	return (s);
    398   1.1       cgd }
    399   1.1       cgd 
    400   1.1       cgd /* ******************* REPLY GENERATION ROUTINES  ************ */
    401   1.1       cgd 
    402   1.1       cgd /*
    403   1.1       cgd  * Send a reply to an rpc request
    404   1.1       cgd  */
    405   1.1       cgd bool_t
    406   1.1       cgd svc_sendreply(xprt, xdr_results, xdr_location)
    407  1.14     lukem 	SVCXPRT *xprt;
    408   1.1       cgd 	xdrproc_t xdr_results;
    409  1.13     lukem 	caddr_t xdr_location;
    410   1.1       cgd {
    411   1.1       cgd 	struct rpc_msg rply;
    412   1.1       cgd 
    413  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    414  1.17     lukem 
    415   1.1       cgd 	rply.rm_direction = REPLY;
    416   1.1       cgd 	rply.rm_reply.rp_stat = MSG_ACCEPTED;
    417   1.1       cgd 	rply.acpted_rply.ar_verf = xprt->xp_verf;
    418   1.1       cgd 	rply.acpted_rply.ar_stat = SUCCESS;
    419   1.1       cgd 	rply.acpted_rply.ar_results.where = xdr_location;
    420   1.1       cgd 	rply.acpted_rply.ar_results.proc = xdr_results;
    421   1.1       cgd 	return (SVC_REPLY(xprt, &rply));
    422   1.1       cgd }
    423   1.1       cgd 
    424   1.1       cgd /*
    425   1.1       cgd  * No procedure error reply
    426   1.1       cgd  */
    427   1.1       cgd void
    428   1.1       cgd svcerr_noproc(xprt)
    429  1.14     lukem 	SVCXPRT *xprt;
    430   1.1       cgd {
    431   1.1       cgd 	struct rpc_msg rply;
    432   1.1       cgd 
    433  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    434  1.17     lukem 
    435   1.1       cgd 	rply.rm_direction = REPLY;
    436   1.1       cgd 	rply.rm_reply.rp_stat = MSG_ACCEPTED;
    437   1.1       cgd 	rply.acpted_rply.ar_verf = xprt->xp_verf;
    438   1.1       cgd 	rply.acpted_rply.ar_stat = PROC_UNAVAIL;
    439   1.1       cgd 	SVC_REPLY(xprt, &rply);
    440   1.1       cgd }
    441   1.1       cgd 
    442   1.1       cgd /*
    443   1.1       cgd  * Can't decode args error reply
    444   1.1       cgd  */
    445   1.1       cgd void
    446   1.1       cgd svcerr_decode(xprt)
    447  1.14     lukem 	SVCXPRT *xprt;
    448   1.1       cgd {
    449   1.1       cgd 	struct rpc_msg rply;
    450   1.1       cgd 
    451  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    452  1.17     lukem 
    453   1.1       cgd 	rply.rm_direction = REPLY;
    454   1.1       cgd 	rply.rm_reply.rp_stat = MSG_ACCEPTED;
    455   1.1       cgd 	rply.acpted_rply.ar_verf = xprt->xp_verf;
    456   1.1       cgd 	rply.acpted_rply.ar_stat = GARBAGE_ARGS;
    457   1.1       cgd 	SVC_REPLY(xprt, &rply);
    458   1.1       cgd }
    459   1.1       cgd 
    460   1.1       cgd /*
    461   1.1       cgd  * Some system error
    462   1.1       cgd  */
    463   1.1       cgd void
    464   1.1       cgd svcerr_systemerr(xprt)
    465  1.14     lukem 	SVCXPRT *xprt;
    466   1.1       cgd {
    467   1.1       cgd 	struct rpc_msg rply;
    468   1.1       cgd 
    469  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    470  1.17     lukem 
    471   1.1       cgd 	rply.rm_direction = REPLY;
    472   1.1       cgd 	rply.rm_reply.rp_stat = MSG_ACCEPTED;
    473   1.1       cgd 	rply.acpted_rply.ar_verf = xprt->xp_verf;
    474   1.1       cgd 	rply.acpted_rply.ar_stat = SYSTEM_ERR;
    475   1.1       cgd 	SVC_REPLY(xprt, &rply);
    476   1.1       cgd }
    477   1.1       cgd 
    478  1.20      fvdl #if 0
    479  1.20      fvdl /*
    480  1.20      fvdl  * Tell RPC package to not complain about version errors to the client.	 This
    481  1.20      fvdl  * is useful when revving broadcast protocols that sit on a fixed address.
    482  1.20      fvdl  * There is really one (or should be only one) example of this kind of
    483  1.20      fvdl  * protocol: the portmapper (or rpc binder).
    484  1.20      fvdl  */
    485  1.20      fvdl void
    486  1.20      fvdl __svc_versquiet_on(xprt)
    487  1.21  christos 	SVCXPRT *xprt;
    488  1.20      fvdl {
    489  1.20      fvdl 	u_long	tmp;
    490  1.20      fvdl 
    491  1.22     lukem 	_DIAGASSERT(xprt != NULL);
    492  1.22     lukem 
    493  1.20      fvdl 	tmp = ((u_long) xprt->xp_p3) | SVC_VERSQUIET;
    494  1.20      fvdl 	xprt->xp_p3 = (caddr_t) tmp;
    495  1.20      fvdl }
    496  1.20      fvdl 
    497  1.20      fvdl void
    498  1.20      fvdl __svc_versquiet_off(xprt)
    499  1.21  christos 	SVCXPRT *xprt;
    500  1.20      fvdl {
    501  1.20      fvdl 	u_long	tmp;
    502  1.20      fvdl 
    503  1.22     lukem 	_DIAGASSERT(xprt != NULL);
    504  1.22     lukem 
    505  1.20      fvdl 	tmp = ((u_long) xprt->xp_p3) & ~SVC_VERSQUIET;
    506  1.20      fvdl 	xprt->xp_p3 = (caddr_t) tmp;
    507  1.20      fvdl }
    508  1.20      fvdl 
    509  1.20      fvdl void
    510  1.20      fvdl svc_versquiet(xprt)
    511  1.21  christos 	SVCXPRT *xprt;
    512  1.20      fvdl {
    513  1.20      fvdl 	__svc_versquiet_on(xprt);
    514  1.20      fvdl }
    515  1.20      fvdl 
    516  1.20      fvdl int
    517  1.20      fvdl __svc_versquiet_get(xprt)
    518  1.21  christos 	SVCXPRT *xprt;
    519  1.20      fvdl {
    520  1.22     lukem 
    521  1.22     lukem 	_DIAGASSERT(xprt != NULL);
    522  1.22     lukem 
    523  1.20      fvdl 	return ((int) xprt->xp_p3) & SVC_VERSQUIET;
    524  1.20      fvdl }
    525  1.20      fvdl #endif
    526  1.20      fvdl 
    527   1.1       cgd /*
    528   1.1       cgd  * Authentication error reply
    529   1.1       cgd  */
    530   1.1       cgd void
    531   1.1       cgd svcerr_auth(xprt, why)
    532   1.1       cgd 	SVCXPRT *xprt;
    533   1.1       cgd 	enum auth_stat why;
    534   1.1       cgd {
    535   1.1       cgd 	struct rpc_msg rply;
    536   1.1       cgd 
    537  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    538  1.17     lukem 
    539   1.1       cgd 	rply.rm_direction = REPLY;
    540   1.1       cgd 	rply.rm_reply.rp_stat = MSG_DENIED;
    541   1.1       cgd 	rply.rjcted_rply.rj_stat = AUTH_ERROR;
    542   1.1       cgd 	rply.rjcted_rply.rj_why = why;
    543   1.1       cgd 	SVC_REPLY(xprt, &rply);
    544   1.1       cgd }
    545   1.1       cgd 
    546   1.1       cgd /*
    547   1.1       cgd  * Auth too weak error reply
    548   1.1       cgd  */
    549   1.1       cgd void
    550   1.1       cgd svcerr_weakauth(xprt)
    551   1.1       cgd 	SVCXPRT *xprt;
    552   1.1       cgd {
    553   1.1       cgd 
    554  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    555  1.17     lukem 
    556   1.1       cgd 	svcerr_auth(xprt, AUTH_TOOWEAK);
    557   1.1       cgd }
    558   1.1       cgd 
    559   1.1       cgd /*
    560   1.1       cgd  * Program unavailable error reply
    561   1.1       cgd  */
    562   1.1       cgd void
    563   1.1       cgd svcerr_noprog(xprt)
    564  1.14     lukem 	SVCXPRT *xprt;
    565   1.1       cgd {
    566   1.1       cgd 	struct rpc_msg rply;
    567   1.1       cgd 
    568  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    569  1.17     lukem 
    570   1.1       cgd 	rply.rm_direction = REPLY;
    571   1.1       cgd 	rply.rm_reply.rp_stat = MSG_ACCEPTED;
    572   1.1       cgd 	rply.acpted_rply.ar_verf = xprt->xp_verf;
    573   1.1       cgd 	rply.acpted_rply.ar_stat = PROG_UNAVAIL;
    574   1.1       cgd 	SVC_REPLY(xprt, &rply);
    575   1.1       cgd }
    576   1.1       cgd 
    577   1.1       cgd /*
    578   1.1       cgd  * Program version mismatch error reply
    579   1.1       cgd  */
    580   1.1       cgd void
    581   1.1       cgd svcerr_progvers(xprt, low_vers, high_vers)
    582  1.14     lukem 	SVCXPRT *xprt;
    583  1.20      fvdl 	rpcvers_t low_vers;
    584  1.20      fvdl 	rpcvers_t high_vers;
    585   1.1       cgd {
    586   1.1       cgd 	struct rpc_msg rply;
    587   1.1       cgd 
    588  1.17     lukem 	_DIAGASSERT(xprt != NULL);
    589  1.17     lukem 
    590   1.1       cgd 	rply.rm_direction = REPLY;
    591   1.1       cgd 	rply.rm_reply.rp_stat = MSG_ACCEPTED;
    592   1.1       cgd 	rply.acpted_rply.ar_verf = xprt->xp_verf;
    593   1.1       cgd 	rply.acpted_rply.ar_stat = PROG_MISMATCH;
    594  1.15  christos 	rply.acpted_rply.ar_vers.low = (u_int32_t)low_vers;
    595  1.15  christos 	rply.acpted_rply.ar_vers.high = (u_int32_t)high_vers;
    596   1.1       cgd 	SVC_REPLY(xprt, &rply);
    597   1.1       cgd }
    598   1.1       cgd 
    599   1.1       cgd /* ******************* SERVER INPUT STUFF ******************* */
    600   1.1       cgd 
    601   1.1       cgd /*
    602   1.1       cgd  * Get server side input from some transport.
    603   1.1       cgd  *
    604   1.1       cgd  * Statement of authentication parameters management:
    605   1.1       cgd  * This function owns and manages all authentication parameters, specifically
    606   1.1       cgd  * the "raw" parameters (msg.rm_call.cb_cred and msg.rm_call.cb_verf) and
    607   1.1       cgd  * the "cooked" credentials (rqst->rq_clntcred).
    608   1.1       cgd  * However, this function does not know the structure of the cooked
    609   1.1       cgd  * credentials, so it make the following assumptions:
    610   1.1       cgd  *   a) the structure is contiguous (no pointers), and
    611   1.1       cgd  *   b) the cred structure size does not exceed RQCRED_SIZE bytes.
    612   1.1       cgd  * In all events, all three parameters are freed upon exit from this routine.
    613   1.1       cgd  * The storage is trivially management on the call stack in user land, but
    614   1.1       cgd  * is mallocated in kernel land.
    615   1.1       cgd  */
    616   1.1       cgd 
    617   1.1       cgd void
    618   1.1       cgd svc_getreq(rdfds)
    619   1.1       cgd 	int rdfds;
    620   1.1       cgd {
    621   1.1       cgd 	fd_set readfds;
    622   1.1       cgd 
    623   1.1       cgd 	FD_ZERO(&readfds);
    624   1.1       cgd 	readfds.fds_bits[0] = rdfds;
    625   1.1       cgd 	svc_getreqset(&readfds);
    626   1.1       cgd }
    627   1.1       cgd 
    628   1.1       cgd void
    629   1.1       cgd svc_getreqset(readfds)
    630   1.1       cgd 	fd_set *readfds;
    631   1.1       cgd {
    632  1.20      fvdl 	int bit, fd;
    633  1.15  christos 	int32_t mask, *maskp;
    634  1.14     lukem 	int sock;
    635   1.1       cgd 
    636  1.17     lukem 	_DIAGASSERT(readfds != NULL);
    637   1.1       cgd 
    638   1.4       cgd 	maskp = readfds->fds_bits;
    639   1.2   deraadt 	for (sock = 0; sock < FD_SETSIZE; sock += NFDBITS) {
    640  1.10  christos 	    for (mask = *maskp++; (bit = ffs(mask)) != 0;
    641  1.10  christos 		mask ^= (1 << (bit - 1))) {
    642   1.1       cgd 		/* sock has input waiting */
    643  1.20      fvdl 		fd = sock + bit - 1;
    644  1.20      fvdl 		svc_getreq_common(fd);
    645  1.20      fvdl 	    }
    646  1.20      fvdl 	}
    647  1.20      fvdl }
    648  1.20      fvdl 
    649  1.20      fvdl void
    650  1.20      fvdl svc_getreq_common(fd)
    651  1.20      fvdl 	int fd;
    652  1.20      fvdl {
    653  1.20      fvdl 	SVCXPRT *xprt;
    654  1.20      fvdl 	struct svc_req r;
    655  1.20      fvdl 	struct rpc_msg msg;
    656  1.20      fvdl 	int prog_found;
    657  1.20      fvdl 	rpcvers_t low_vers;
    658  1.20      fvdl 	rpcvers_t high_vers;
    659  1.20      fvdl 	enum xprt_stat stat;
    660  1.20      fvdl 	char cred_area[2*MAX_AUTH_BYTES + RQCRED_SIZE];
    661  1.20      fvdl 
    662  1.20      fvdl 	msg.rm_call.cb_cred.oa_base = cred_area;
    663  1.20      fvdl 	msg.rm_call.cb_verf.oa_base = &(cred_area[MAX_AUTH_BYTES]);
    664  1.20      fvdl 	r.rq_clntcred = &(cred_area[2*MAX_AUTH_BYTES]);
    665  1.20      fvdl 
    666  1.20      fvdl 	rwlock_rdlock(&svc_fd_lock);
    667  1.23      fvdl 	xprt = __svc_xports[fd];
    668  1.20      fvdl 	rwlock_unlock(&svc_fd_lock);
    669  1.20      fvdl 	if (xprt == NULL)
    670  1.20      fvdl 		/* But do we control sock? */
    671  1.20      fvdl 		return;
    672  1.20      fvdl 	/* now receive msgs from xprtprt (support batch calls) */
    673  1.20      fvdl 	do {
    674  1.20      fvdl 		if (SVC_RECV(xprt, &msg)) {
    675  1.20      fvdl 
    676  1.20      fvdl 			/* now find the exported program and call it */
    677  1.20      fvdl 			struct svc_callout *s;
    678  1.20      fvdl 			enum auth_stat why;
    679  1.20      fvdl 
    680  1.20      fvdl 			r.rq_xprt = xprt;
    681  1.20      fvdl 			r.rq_prog = msg.rm_call.cb_prog;
    682  1.20      fvdl 			r.rq_vers = msg.rm_call.cb_vers;
    683  1.20      fvdl 			r.rq_proc = msg.rm_call.cb_proc;
    684  1.20      fvdl 			r.rq_cred = msg.rm_call.cb_cred;
    685  1.20      fvdl 			/* first authenticate the message */
    686  1.20      fvdl 			if ((why = _authenticate(&r, &msg)) != AUTH_OK) {
    687  1.20      fvdl 				svcerr_auth(xprt, why);
    688  1.20      fvdl 				goto call_done;
    689   1.1       cgd 			}
    690  1.20      fvdl 			/* now match message with a registered service*/
    691  1.20      fvdl 			prog_found = FALSE;
    692  1.20      fvdl 			low_vers = (rpcvers_t) -1L;
    693  1.20      fvdl 			high_vers = (rpcvers_t) 0L;
    694  1.21  christos 			for (s = svc_head; s != NULL; s = s->sc_next) {
    695  1.20      fvdl 				if (s->sc_prog == r.rq_prog) {
    696  1.20      fvdl 					if (s->sc_vers == r.rq_vers) {
    697  1.20      fvdl 						(*s->sc_dispatch)(&r, xprt);
    698  1.20      fvdl 						goto call_done;
    699  1.20      fvdl 					}  /* found correct version */
    700  1.20      fvdl 					prog_found = TRUE;
    701  1.20      fvdl 					if (s->sc_vers < low_vers)
    702  1.20      fvdl 						low_vers = s->sc_vers;
    703  1.20      fvdl 					if (s->sc_vers > high_vers)
    704  1.20      fvdl 						high_vers = s->sc_vers;
    705  1.20      fvdl 				}   /* found correct program */
    706   1.1       cgd 			}
    707  1.20      fvdl 			/*
    708  1.20      fvdl 			 * if we got here, the program or version
    709  1.20      fvdl 			 * is not served ...
    710  1.20      fvdl 			 */
    711  1.20      fvdl 			if (prog_found)
    712  1.20      fvdl 				svcerr_progvers(xprt, low_vers, high_vers);
    713  1.20      fvdl 			else
    714  1.20      fvdl 				 svcerr_noprog(xprt);
    715  1.20      fvdl 			/* Fall through to ... */
    716  1.20      fvdl 		}
    717  1.20      fvdl 		/*
    718  1.20      fvdl 		 * Check if the xprt has been disconnected in a
    719  1.20      fvdl 		 * recursive call in the service dispatch routine.
    720  1.20      fvdl 		 * If so, then break.
    721  1.20      fvdl 		 */
    722  1.20      fvdl 		rwlock_rdlock(&svc_fd_lock);
    723  1.23      fvdl 		if (xprt != __svc_xports[fd]) {
    724  1.20      fvdl 			rwlock_unlock(&svc_fd_lock);
    725  1.20      fvdl 			break;
    726  1.20      fvdl 		}
    727  1.20      fvdl 		rwlock_unlock(&svc_fd_lock);
    728  1.20      fvdl call_done:
    729  1.20      fvdl 		if ((stat = SVC_STAT(xprt)) == XPRT_DIED){
    730  1.20      fvdl 			SVC_DESTROY(xprt);
    731  1.20      fvdl 			break;
    732  1.20      fvdl 		}
    733  1.20      fvdl 	} while (stat == XPRT_MOREREQS);
    734  1.20      fvdl }
    735  1.20      fvdl 
    736  1.20      fvdl 
    737  1.20      fvdl void
    738  1.20      fvdl svc_getreq_poll(pfdp, pollretval)
    739  1.20      fvdl 	struct pollfd	*pfdp;
    740  1.20      fvdl 	int	pollretval;
    741  1.20      fvdl {
    742  1.20      fvdl 	int i;
    743  1.20      fvdl 	int fds_found;
    744  1.22     lukem 
    745  1.22     lukem 	_DIAGASSERT(pfdp != NULL);
    746  1.20      fvdl 
    747  1.20      fvdl 	for (i = fds_found = 0; fds_found < pollretval; i++) {
    748  1.21  christos 		struct pollfd *p = &pfdp[i];
    749  1.20      fvdl 
    750  1.20      fvdl 		if (p->revents) {
    751  1.20      fvdl 			/* fd has input waiting */
    752  1.20      fvdl 			fds_found++;
    753  1.20      fvdl 			/*
    754  1.20      fvdl 			 *	We assume that this function is only called
    755  1.20      fvdl 			 *	via someone select()ing from svc_fdset or
    756  1.25  christos 			 *	pollts()ing from svc_pollset[].  Thus it's safe
    757  1.20      fvdl 			 *	to handle the POLLNVAL event by simply turning
    758  1.20      fvdl 			 *	the corresponding bit off in svc_fdset.  The
    759  1.20      fvdl 			 *	svc_pollset[] array is derived from svc_fdset
    760  1.20      fvdl 			 *	and so will also be updated eventually.
    761  1.20      fvdl 			 *
    762  1.20      fvdl 			 *	XXX Should we do an xprt_unregister() instead?
    763  1.20      fvdl 			 */
    764  1.20      fvdl 			if (p->revents & POLLNVAL) {
    765  1.20      fvdl 				rwlock_wrlock(&svc_fd_lock);
    766  1.20      fvdl 				FD_CLR(p->fd, &svc_fdset);
    767  1.20      fvdl 				rwlock_unlock(&svc_fd_lock);
    768  1.20      fvdl 			} else
    769  1.20      fvdl 				svc_getreq_common(p->fd);
    770  1.20      fvdl 		}
    771   1.1       cgd 	}
    772  1.23      fvdl }
    773  1.23      fvdl 
    774  1.23      fvdl bool_t
    775  1.23      fvdl rpc_control(int what, void *arg)
    776  1.23      fvdl {
    777  1.23      fvdl 	int val;
    778  1.23      fvdl 
    779  1.23      fvdl 	switch (what) {
    780  1.23      fvdl 	case RPC_SVC_CONNMAXREC_SET:
    781  1.23      fvdl 		val = *(int *)arg;
    782  1.23      fvdl 		if (val <= 0)
    783  1.23      fvdl 			return FALSE;
    784  1.23      fvdl 		__svc_maxrec = val;
    785  1.23      fvdl 		return TRUE;
    786  1.23      fvdl 	case RPC_SVC_CONNMAXREC_GET:
    787  1.23      fvdl 		*(int *)arg = __svc_maxrec;
    788  1.23      fvdl 		return TRUE;
    789  1.23      fvdl 	default:
    790  1.23      fvdl 		break;
    791  1.23      fvdl 	}
    792  1.23      fvdl 	return FALSE;
    793   1.1       cgd }
    794