17c478bd9Sstevel@tonic-gate /* 27c478bd9Sstevel@tonic-gate * CDDL HEADER START 37c478bd9Sstevel@tonic-gate * 47c478bd9Sstevel@tonic-gate * The contents of this file are subject to the terms of the 5*9acbbeafSnn35248 * Common Development and Distribution License (the "License"). 6*9acbbeafSnn35248 * You may not use this file except in compliance with the License. 77c478bd9Sstevel@tonic-gate * 87c478bd9Sstevel@tonic-gate * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE 97c478bd9Sstevel@tonic-gate * or http://www.opensolaris.org/os/licensing. 107c478bd9Sstevel@tonic-gate * See the License for the specific language governing permissions 117c478bd9Sstevel@tonic-gate * and limitations under the License. 127c478bd9Sstevel@tonic-gate * 137c478bd9Sstevel@tonic-gate * When distributing Covered Code, include this CDDL HEADER in each 147c478bd9Sstevel@tonic-gate * file and include the License file at usr/src/OPENSOLARIS.LICENSE. 157c478bd9Sstevel@tonic-gate * If applicable, add the following below this CDDL HEADER, with the 167c478bd9Sstevel@tonic-gate * fields enclosed by brackets "[]" replaced with your own identifying 177c478bd9Sstevel@tonic-gate * information: Portions Copyright [yyyy] [name of copyright owner] 187c478bd9Sstevel@tonic-gate * 197c478bd9Sstevel@tonic-gate * CDDL HEADER END 207c478bd9Sstevel@tonic-gate */ 2161961e0fSrobinson 227c478bd9Sstevel@tonic-gate /* 23*9acbbeafSnn35248 * Copyright 2006 Sun Microsystems, Inc. All rights reserved. 247c478bd9Sstevel@tonic-gate * Use is subject to license terms. 257c478bd9Sstevel@tonic-gate */ 267c478bd9Sstevel@tonic-gate /* 277c478bd9Sstevel@tonic-gate * Copyright 1993 OpenVision Technologies, Inc., All Rights Reserved. 287c478bd9Sstevel@tonic-gate */ 297c478bd9Sstevel@tonic-gate /* Copyright (c) 1983, 1984, 1985, 1986, 1987, 1988, 1989 AT&T */ 307c478bd9Sstevel@tonic-gate /* All Rights Reserved */ 317c478bd9Sstevel@tonic-gate /* 327c478bd9Sstevel@tonic-gate * Portions of this source code were derived from Berkeley 337c478bd9Sstevel@tonic-gate * 4.3 BSD under license from the Regents of the University of 347c478bd9Sstevel@tonic-gate * California. 357c478bd9Sstevel@tonic-gate */ 367c478bd9Sstevel@tonic-gate 377c478bd9Sstevel@tonic-gate #pragma ident "%Z%%M% %I% %E% SMI" 387c478bd9Sstevel@tonic-gate 397c478bd9Sstevel@tonic-gate /* 407c478bd9Sstevel@tonic-gate * svc.c, Server-side remote procedure call interface. 417c478bd9Sstevel@tonic-gate * 427c478bd9Sstevel@tonic-gate * There are two sets of procedures here. The xprt routines are 437c478bd9Sstevel@tonic-gate * for handling transport handles. The svc routines handle the 447c478bd9Sstevel@tonic-gate * list of service routines. 457c478bd9Sstevel@tonic-gate * 467c478bd9Sstevel@tonic-gate */ 477c478bd9Sstevel@tonic-gate 487c478bd9Sstevel@tonic-gate 497c478bd9Sstevel@tonic-gate #include "mt.h" 507c478bd9Sstevel@tonic-gate #include "rpc_mt.h" 517c478bd9Sstevel@tonic-gate #include <assert.h> 527c478bd9Sstevel@tonic-gate #include <errno.h> 537c478bd9Sstevel@tonic-gate #include <sys/types.h> 547c478bd9Sstevel@tonic-gate #include <stropts.h> 557c478bd9Sstevel@tonic-gate #include <sys/conf.h> 567c478bd9Sstevel@tonic-gate #include <rpc/rpc.h> 577c478bd9Sstevel@tonic-gate #ifdef PORTMAP 587c478bd9Sstevel@tonic-gate #include <rpc/pmap_clnt.h> 597c478bd9Sstevel@tonic-gate #endif 607c478bd9Sstevel@tonic-gate #include <sys/poll.h> 617c478bd9Sstevel@tonic-gate #include <netconfig.h> 627c478bd9Sstevel@tonic-gate #include <syslog.h> 637c478bd9Sstevel@tonic-gate #include <stdlib.h> 647c478bd9Sstevel@tonic-gate #include <unistd.h> 657c478bd9Sstevel@tonic-gate #include <string.h> 667c478bd9Sstevel@tonic-gate #include <limits.h> 677c478bd9Sstevel@tonic-gate 687c478bd9Sstevel@tonic-gate extern bool_t __svc_get_door_cred(); 697c478bd9Sstevel@tonic-gate extern bool_t __rpc_get_local_cred(); 707c478bd9Sstevel@tonic-gate 71*9acbbeafSnn35248 extern int use_portmapper; 72*9acbbeafSnn35248 extern bool_t __pmap_set(const rpcprog_t, const rpcvers_t, 73*9acbbeafSnn35248 const struct netconfig *, const struct netbuf *); 74*9acbbeafSnn35248 extern bool_t __pmap_unset(const rpcprog_t, const rpcvers_t); 75*9acbbeafSnn35248 767c478bd9Sstevel@tonic-gate SVCXPRT **svc_xports; 777c478bd9Sstevel@tonic-gate static int nsvc_xports; /* total number of svc_xports allocated */ 787c478bd9Sstevel@tonic-gate 797c478bd9Sstevel@tonic-gate XDR **svc_xdrs; /* common XDR receive area */ 807c478bd9Sstevel@tonic-gate int nsvc_xdrs; /* total number of svc_xdrs allocated */ 817c478bd9Sstevel@tonic-gate 827c478bd9Sstevel@tonic-gate int __rpc_use_pollfd_done; /* to unlimit the number of connections */ 837c478bd9Sstevel@tonic-gate 847c478bd9Sstevel@tonic-gate #define NULL_SVC ((struct svc_callout *)0) 857c478bd9Sstevel@tonic-gate #define RQCRED_SIZE 400 /* this size is excessive */ 867c478bd9Sstevel@tonic-gate 877c478bd9Sstevel@tonic-gate /* 887c478bd9Sstevel@tonic-gate * The services list 897c478bd9Sstevel@tonic-gate * Each entry represents a set of procedures (an rpc program). 907c478bd9Sstevel@tonic-gate * The dispatch routine takes request structs and runs the 917c478bd9Sstevel@tonic-gate * appropriate procedure. 927c478bd9Sstevel@tonic-gate */ 937c478bd9Sstevel@tonic-gate static struct svc_callout { 947c478bd9Sstevel@tonic-gate struct svc_callout *sc_next; 957c478bd9Sstevel@tonic-gate rpcprog_t sc_prog; 967c478bd9Sstevel@tonic-gate rpcvers_t sc_vers; 977c478bd9Sstevel@tonic-gate char *sc_netid; 987c478bd9Sstevel@tonic-gate void (*sc_dispatch)(); 997c478bd9Sstevel@tonic-gate } *svc_head; 1007c478bd9Sstevel@tonic-gate extern rwlock_t svc_lock; 1017c478bd9Sstevel@tonic-gate 1027c478bd9Sstevel@tonic-gate static struct svc_callout *svc_find(); 1037c478bd9Sstevel@tonic-gate int _svc_prog_dispatch(); 1047c478bd9Sstevel@tonic-gate void svc_getreq_common(); 1057c478bd9Sstevel@tonic-gate char *strdup(); 1067c478bd9Sstevel@tonic-gate 1077c478bd9Sstevel@tonic-gate extern mutex_t svc_door_mutex; 1087c478bd9Sstevel@tonic-gate extern cond_t svc_door_waitcv; 1097c478bd9Sstevel@tonic-gate extern int svc_ndoorfds; 1107c478bd9Sstevel@tonic-gate extern SVCXPRT_LIST *_svc_xprtlist; 1117c478bd9Sstevel@tonic-gate extern mutex_t xprtlist_lock; 1127c478bd9Sstevel@tonic-gate extern void __svc_rm_from_xlist(); 1137c478bd9Sstevel@tonic-gate 1147c478bd9Sstevel@tonic-gate extern fd_set _new_svc_fdset; 1157c478bd9Sstevel@tonic-gate 1167c478bd9Sstevel@tonic-gate /* 1177c478bd9Sstevel@tonic-gate * If the allocated array of reactor is too small, this value is used as a 1187c478bd9Sstevel@tonic-gate * margin. This reduces the number of allocations. 1197c478bd9Sstevel@tonic-gate */ 1207c478bd9Sstevel@tonic-gate #define USER_FD_INCREMENT 5 1217c478bd9Sstevel@tonic-gate 1227c478bd9Sstevel@tonic-gate static void add_pollfd(int fd, short events); 1237c478bd9Sstevel@tonic-gate static void remove_pollfd(int fd); 1247c478bd9Sstevel@tonic-gate static void __svc_remove_input_of_fd(int fd); 1257c478bd9Sstevel@tonic-gate 1267c478bd9Sstevel@tonic-gate 1277c478bd9Sstevel@tonic-gate /* 1287c478bd9Sstevel@tonic-gate * Data used to handle reactor: 1297c478bd9Sstevel@tonic-gate * - one file descriptor we listen to, 1307c478bd9Sstevel@tonic-gate * - one callback we call if the fd pops, 1317c478bd9Sstevel@tonic-gate * - and a cookie passed as a parameter to the callback. 1327c478bd9Sstevel@tonic-gate * 1337c478bd9Sstevel@tonic-gate * The structure is an array indexed on the file descriptor. Each entry is 1347c478bd9Sstevel@tonic-gate * pointing to the first element of a double-linked list of callback. 1357c478bd9Sstevel@tonic-gate * only one callback may be associated to a couple (fd, event). 1367c478bd9Sstevel@tonic-gate */ 1377c478bd9Sstevel@tonic-gate 1387c478bd9Sstevel@tonic-gate struct _svc_user_fd_head; 1397c478bd9Sstevel@tonic-gate 14061961e0fSrobinson typedef struct { 1417c478bd9Sstevel@tonic-gate struct _svc_user_fd_node *next; 1427c478bd9Sstevel@tonic-gate struct _svc_user_fd_node *previous; 1437c478bd9Sstevel@tonic-gate } _svc_user_link; 1447c478bd9Sstevel@tonic-gate 14561961e0fSrobinson typedef struct _svc_user_fd_node { 1467c478bd9Sstevel@tonic-gate /* The lnk field must be the first field. */ 1477c478bd9Sstevel@tonic-gate _svc_user_link lnk; 1487c478bd9Sstevel@tonic-gate svc_input_id_t id; 1497c478bd9Sstevel@tonic-gate int fd; 1507c478bd9Sstevel@tonic-gate unsigned int events; 1517c478bd9Sstevel@tonic-gate svc_callback_t callback; 1527c478bd9Sstevel@tonic-gate void* cookie; 1537c478bd9Sstevel@tonic-gate } _svc_user_fd_node; 1547c478bd9Sstevel@tonic-gate 15561961e0fSrobinson typedef struct _svc_user_fd_head { 1567c478bd9Sstevel@tonic-gate /* The lnk field must be the first field. */ 1577c478bd9Sstevel@tonic-gate _svc_user_link lnk; 1587c478bd9Sstevel@tonic-gate unsigned int mask; /* logical OR of all sub-masks */ 1597c478bd9Sstevel@tonic-gate } _svc_user_fd_head; 1607c478bd9Sstevel@tonic-gate 1617c478bd9Sstevel@tonic-gate 1627c478bd9Sstevel@tonic-gate /* Define some macros to manage the linked list. */ 1637c478bd9Sstevel@tonic-gate #define LIST_ISEMPTY(l) ((_svc_user_fd_node *) &(l.lnk) == l.lnk.next) 1647c478bd9Sstevel@tonic-gate #define LIST_CLR(l) \ 1657c478bd9Sstevel@tonic-gate (l.lnk.previous = l.lnk.next = (_svc_user_fd_node *) &(l.lnk)) 1667c478bd9Sstevel@tonic-gate 1677c478bd9Sstevel@tonic-gate /* Array of defined reactor - indexed on file descriptor */ 1687c478bd9Sstevel@tonic-gate static _svc_user_fd_head *svc_userfds = NULL; 1697c478bd9Sstevel@tonic-gate 1707c478bd9Sstevel@tonic-gate /* current size of file descriptor */ 1717c478bd9Sstevel@tonic-gate static int svc_nuserfds = 0; 1727c478bd9Sstevel@tonic-gate 1737c478bd9Sstevel@tonic-gate /* Mutex to ensure MT safe operations for user fds callbacks. */ 1747c478bd9Sstevel@tonic-gate static mutex_t svc_userfds_lock = DEFAULTMUTEX; 1757c478bd9Sstevel@tonic-gate 1767c478bd9Sstevel@tonic-gate 1777c478bd9Sstevel@tonic-gate /* 1787c478bd9Sstevel@tonic-gate * This structure is used to have constant time alogrithms. There is an array 1797c478bd9Sstevel@tonic-gate * of this structure as large as svc_nuserfds. When the user is registering a 1807c478bd9Sstevel@tonic-gate * new callback, the address of the created structure is stored in a cell of 1817c478bd9Sstevel@tonic-gate * this array. The address of this cell is the returned unique identifier. 1827c478bd9Sstevel@tonic-gate * 1837c478bd9Sstevel@tonic-gate * On removing, the id is given by the user, then we know if this cell is 1847c478bd9Sstevel@tonic-gate * filled or not (with free). If it is free, we return an error. Otherwise, 1857c478bd9Sstevel@tonic-gate * we can free the structure pointed by fd_node. 1867c478bd9Sstevel@tonic-gate * 1877c478bd9Sstevel@tonic-gate * On insertion, we use the linked list created by (first_free, 1887c478bd9Sstevel@tonic-gate * next_free). In this way with a constant time computation, we can give a 1897c478bd9Sstevel@tonic-gate * correct index to the user. 1907c478bd9Sstevel@tonic-gate */ 1917c478bd9Sstevel@tonic-gate 19261961e0fSrobinson typedef struct _svc_management_user_fd { 1937c478bd9Sstevel@tonic-gate bool_t free; 1947c478bd9Sstevel@tonic-gate union { 1957c478bd9Sstevel@tonic-gate svc_input_id_t next_free; 1967c478bd9Sstevel@tonic-gate _svc_user_fd_node *fd_node; 1977c478bd9Sstevel@tonic-gate } data; 1987c478bd9Sstevel@tonic-gate } _svc_management_user_fd; 1997c478bd9Sstevel@tonic-gate 2007c478bd9Sstevel@tonic-gate /* index to the first free elem */ 2017c478bd9Sstevel@tonic-gate static svc_input_id_t first_free = (svc_input_id_t)-1; 2027c478bd9Sstevel@tonic-gate /* the size of this array is the same as svc_nuserfds */ 2037c478bd9Sstevel@tonic-gate static _svc_management_user_fd* user_fd_mgt_array = NULL; 2047c478bd9Sstevel@tonic-gate 2057c478bd9Sstevel@tonic-gate /* current size of user_fd_mgt_array */ 2067c478bd9Sstevel@tonic-gate static int svc_nmgtuserfds = 0; 2077c478bd9Sstevel@tonic-gate 2087c478bd9Sstevel@tonic-gate 2097c478bd9Sstevel@tonic-gate /* Define some macros to access data associated to registration ids. */ 2107c478bd9Sstevel@tonic-gate #define node_from_id(id) (user_fd_mgt_array[(int)id].data.fd_node) 2117c478bd9Sstevel@tonic-gate #define is_free_id(id) (user_fd_mgt_array[(int)id].free) 2127c478bd9Sstevel@tonic-gate 2137c478bd9Sstevel@tonic-gate #ifndef POLLSTANDARD 2147c478bd9Sstevel@tonic-gate #define POLLSTANDARD \ 2157c478bd9Sstevel@tonic-gate (POLLIN|POLLPRI|POLLOUT|POLLRDNORM|POLLRDBAND| \ 2167c478bd9Sstevel@tonic-gate POLLWRBAND|POLLERR|POLLHUP|POLLNVAL) 2177c478bd9Sstevel@tonic-gate #endif 2187c478bd9Sstevel@tonic-gate 2197c478bd9Sstevel@tonic-gate /* 2207c478bd9Sstevel@tonic-gate * To free an Id, we set the cell as free and insert its address in the list 2217c478bd9Sstevel@tonic-gate * of free cell. 2227c478bd9Sstevel@tonic-gate */ 2237c478bd9Sstevel@tonic-gate 2247c478bd9Sstevel@tonic-gate static void 2257c478bd9Sstevel@tonic-gate _svc_free_id(const svc_input_id_t id) 2267c478bd9Sstevel@tonic-gate { 2277c478bd9Sstevel@tonic-gate assert(((int)id >= 0) && ((int)id < svc_nmgtuserfds)); 2287c478bd9Sstevel@tonic-gate user_fd_mgt_array[(int)id].free = TRUE; 2297c478bd9Sstevel@tonic-gate user_fd_mgt_array[(int)id].data.next_free = first_free; 2307c478bd9Sstevel@tonic-gate first_free = id; 2317c478bd9Sstevel@tonic-gate } 2327c478bd9Sstevel@tonic-gate 2337c478bd9Sstevel@tonic-gate /* 2347c478bd9Sstevel@tonic-gate * To get a free cell, we just have to take it from the free linked list and 2357c478bd9Sstevel@tonic-gate * set the flag to "not free". This function also allocates new memory if 2367c478bd9Sstevel@tonic-gate * necessary 2377c478bd9Sstevel@tonic-gate */ 2387c478bd9Sstevel@tonic-gate static svc_input_id_t 2397c478bd9Sstevel@tonic-gate _svc_attribute_new_id(_svc_user_fd_node *node) 2407c478bd9Sstevel@tonic-gate { 2417c478bd9Sstevel@tonic-gate int selected_index = (int)first_free; 2427c478bd9Sstevel@tonic-gate assert(node != NULL); 2437c478bd9Sstevel@tonic-gate 2447c478bd9Sstevel@tonic-gate if (selected_index == -1) { 2457c478bd9Sstevel@tonic-gate /* Allocate new entries */ 2467c478bd9Sstevel@tonic-gate int L_inOldSize = svc_nmgtuserfds; 2477c478bd9Sstevel@tonic-gate int i; 2487c478bd9Sstevel@tonic-gate 2497c478bd9Sstevel@tonic-gate svc_nmgtuserfds += USER_FD_INCREMENT; 2507c478bd9Sstevel@tonic-gate 2517c478bd9Sstevel@tonic-gate user_fd_mgt_array = (_svc_management_user_fd *) 2527c478bd9Sstevel@tonic-gate realloc(user_fd_mgt_array, svc_nmgtuserfds 2537c478bd9Sstevel@tonic-gate * sizeof (_svc_management_user_fd)); 2547c478bd9Sstevel@tonic-gate 2557c478bd9Sstevel@tonic-gate if (user_fd_mgt_array == NULL) { 2567c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "_svc_attribute_new_id: out of memory"); 2577c478bd9Sstevel@tonic-gate errno = ENOMEM; 2587c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 2597c478bd9Sstevel@tonic-gate } 2607c478bd9Sstevel@tonic-gate 2617c478bd9Sstevel@tonic-gate for (i = svc_nmgtuserfds - 1; i >= L_inOldSize; i--) 2627c478bd9Sstevel@tonic-gate _svc_free_id((svc_input_id_t)i); 2637c478bd9Sstevel@tonic-gate selected_index = (int)first_free; 2647c478bd9Sstevel@tonic-gate } 2657c478bd9Sstevel@tonic-gate 2667c478bd9Sstevel@tonic-gate node->id = (svc_input_id_t)selected_index; 2677c478bd9Sstevel@tonic-gate first_free = user_fd_mgt_array[selected_index].data.next_free; 2687c478bd9Sstevel@tonic-gate 2697c478bd9Sstevel@tonic-gate user_fd_mgt_array[selected_index].data.fd_node = node; 2707c478bd9Sstevel@tonic-gate user_fd_mgt_array[selected_index].free = FALSE; 2717c478bd9Sstevel@tonic-gate 2727c478bd9Sstevel@tonic-gate return ((svc_input_id_t)selected_index); 2737c478bd9Sstevel@tonic-gate } 2747c478bd9Sstevel@tonic-gate 2757c478bd9Sstevel@tonic-gate /* 2767c478bd9Sstevel@tonic-gate * Access to a pollfd treatment. Scan all the associated callbacks that have 2777c478bd9Sstevel@tonic-gate * at least one bit in their mask that masks a received event. 2787c478bd9Sstevel@tonic-gate * 2797c478bd9Sstevel@tonic-gate * If event POLLNVAL is received, we check that one callback processes it, if 2807c478bd9Sstevel@tonic-gate * not, then remove the file descriptor from the poll. If there is one, let 2817c478bd9Sstevel@tonic-gate * the user do the work. 2827c478bd9Sstevel@tonic-gate */ 2837c478bd9Sstevel@tonic-gate void 2847c478bd9Sstevel@tonic-gate __svc_getreq_user(struct pollfd *pfd) 2857c478bd9Sstevel@tonic-gate { 2867c478bd9Sstevel@tonic-gate int fd = pfd->fd; 2877c478bd9Sstevel@tonic-gate short revents = pfd->revents; 2887c478bd9Sstevel@tonic-gate bool_t invalHandled = FALSE; 2897c478bd9Sstevel@tonic-gate _svc_user_fd_node *node; 2907c478bd9Sstevel@tonic-gate 29161961e0fSrobinson (void) mutex_lock(&svc_userfds_lock); 2927c478bd9Sstevel@tonic-gate 2937c478bd9Sstevel@tonic-gate if ((fd < 0) || (fd >= svc_nuserfds)) { 29461961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 2957c478bd9Sstevel@tonic-gate return; 2967c478bd9Sstevel@tonic-gate } 2977c478bd9Sstevel@tonic-gate 2987c478bd9Sstevel@tonic-gate node = svc_userfds[fd].lnk.next; 2997c478bd9Sstevel@tonic-gate 3007c478bd9Sstevel@tonic-gate /* check if at least one mask fits */ 3017c478bd9Sstevel@tonic-gate if (0 == (revents & svc_userfds[fd].mask)) { 30261961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 3037c478bd9Sstevel@tonic-gate return; 3047c478bd9Sstevel@tonic-gate } 3057c478bd9Sstevel@tonic-gate 3067c478bd9Sstevel@tonic-gate while ((svc_userfds[fd].mask != 0) && 3077c478bd9Sstevel@tonic-gate ((_svc_user_link *)node != &(svc_userfds[fd].lnk))) { 3087c478bd9Sstevel@tonic-gate /* 3097c478bd9Sstevel@tonic-gate * If one of the received events maps the ones the node listens 3107c478bd9Sstevel@tonic-gate * to 3117c478bd9Sstevel@tonic-gate */ 3127c478bd9Sstevel@tonic-gate _svc_user_fd_node *next = node->lnk.next; 3137c478bd9Sstevel@tonic-gate 3147c478bd9Sstevel@tonic-gate if (node->callback != NULL) { 3157c478bd9Sstevel@tonic-gate if (node->events & revents) { 3167c478bd9Sstevel@tonic-gate if (revents & POLLNVAL) { 3177c478bd9Sstevel@tonic-gate invalHandled = TRUE; 3187c478bd9Sstevel@tonic-gate } 3197c478bd9Sstevel@tonic-gate 3207c478bd9Sstevel@tonic-gate /* 3217c478bd9Sstevel@tonic-gate * The lock must be released before calling the 3227c478bd9Sstevel@tonic-gate * user function, as this function can call 3237c478bd9Sstevel@tonic-gate * svc_remove_input() for example. 3247c478bd9Sstevel@tonic-gate */ 32561961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 3267c478bd9Sstevel@tonic-gate node->callback(node->id, node->fd, 3277c478bd9Sstevel@tonic-gate node->events & revents, node->cookie); 3287c478bd9Sstevel@tonic-gate /* 3297c478bd9Sstevel@tonic-gate * Do not use the node structure anymore, as it 3307c478bd9Sstevel@tonic-gate * could have been deallocated by the previous 3317c478bd9Sstevel@tonic-gate * callback. 3327c478bd9Sstevel@tonic-gate */ 33361961e0fSrobinson (void) mutex_lock(&svc_userfds_lock); 3347c478bd9Sstevel@tonic-gate } 3357c478bd9Sstevel@tonic-gate } 3367c478bd9Sstevel@tonic-gate node = next; 3377c478bd9Sstevel@tonic-gate } 3387c478bd9Sstevel@tonic-gate 3397c478bd9Sstevel@tonic-gate if ((revents & POLLNVAL) && !invalHandled) 3407c478bd9Sstevel@tonic-gate __svc_remove_input_of_fd(fd); 34161961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 3427c478bd9Sstevel@tonic-gate } 3437c478bd9Sstevel@tonic-gate 3447c478bd9Sstevel@tonic-gate 3457c478bd9Sstevel@tonic-gate /* 3467c478bd9Sstevel@tonic-gate * Check if a file descriptor is associated with a user reactor. 3477c478bd9Sstevel@tonic-gate * To do this, just check that the array indexed on fd has a non-void linked 3487c478bd9Sstevel@tonic-gate * list (ie. first element is not NULL) 3497c478bd9Sstevel@tonic-gate */ 3507c478bd9Sstevel@tonic-gate bool_t 3517c478bd9Sstevel@tonic-gate __is_a_userfd(int fd) 3527c478bd9Sstevel@tonic-gate { 3537c478bd9Sstevel@tonic-gate /* Checks argument */ 3547c478bd9Sstevel@tonic-gate if ((fd < 0) || (fd >= svc_nuserfds)) 3557c478bd9Sstevel@tonic-gate return (FALSE); 3567c478bd9Sstevel@tonic-gate return ((svc_userfds[fd].mask == 0x0000)? FALSE:TRUE); 3577c478bd9Sstevel@tonic-gate } 3587c478bd9Sstevel@tonic-gate 3597c478bd9Sstevel@tonic-gate /* free everything concerning user fd */ 3607c478bd9Sstevel@tonic-gate /* used in svc_run.c => no static */ 3617c478bd9Sstevel@tonic-gate 3627c478bd9Sstevel@tonic-gate void 36361961e0fSrobinson __destroy_userfd(void) 3647c478bd9Sstevel@tonic-gate { 3657c478bd9Sstevel@tonic-gate int one_fd; 3667c478bd9Sstevel@tonic-gate /* Clean user fd */ 3677c478bd9Sstevel@tonic-gate if (svc_userfds != NULL) { 3687c478bd9Sstevel@tonic-gate for (one_fd = 0; one_fd < svc_nuserfds; one_fd++) { 3697c478bd9Sstevel@tonic-gate _svc_user_fd_node *node; 3707c478bd9Sstevel@tonic-gate 3717c478bd9Sstevel@tonic-gate node = svc_userfds[one_fd].lnk.next; 3727c478bd9Sstevel@tonic-gate while ((_svc_user_link *) node 3737c478bd9Sstevel@tonic-gate != (_svc_user_link *) &(svc_userfds[one_fd])) { 3747c478bd9Sstevel@tonic-gate _svc_free_id(node->id); 3757c478bd9Sstevel@tonic-gate node = node->lnk.next; 3767c478bd9Sstevel@tonic-gate free(node->lnk.previous); 3777c478bd9Sstevel@tonic-gate } 3787c478bd9Sstevel@tonic-gate } 3797c478bd9Sstevel@tonic-gate 3807c478bd9Sstevel@tonic-gate free(user_fd_mgt_array); 3817c478bd9Sstevel@tonic-gate user_fd_mgt_array = NULL; 3827c478bd9Sstevel@tonic-gate first_free = (svc_input_id_t)-1; 3837c478bd9Sstevel@tonic-gate 3847c478bd9Sstevel@tonic-gate free(svc_userfds); 3857c478bd9Sstevel@tonic-gate svc_userfds = NULL; 3867c478bd9Sstevel@tonic-gate svc_nuserfds = 0; 3877c478bd9Sstevel@tonic-gate } 3887c478bd9Sstevel@tonic-gate } 3897c478bd9Sstevel@tonic-gate 3907c478bd9Sstevel@tonic-gate /* 3917c478bd9Sstevel@tonic-gate * Remove all the callback associated with a fd => useful when the fd is 3927c478bd9Sstevel@tonic-gate * closed for instance 3937c478bd9Sstevel@tonic-gate */ 3947c478bd9Sstevel@tonic-gate static void 3957c478bd9Sstevel@tonic-gate __svc_remove_input_of_fd(int fd) 3967c478bd9Sstevel@tonic-gate { 3977c478bd9Sstevel@tonic-gate _svc_user_fd_node *one_node; 3987c478bd9Sstevel@tonic-gate 3997c478bd9Sstevel@tonic-gate if ((fd < 0) || (fd >= svc_nuserfds)) 4007c478bd9Sstevel@tonic-gate return; 4017c478bd9Sstevel@tonic-gate 4027c478bd9Sstevel@tonic-gate one_node = svc_userfds[fd].lnk.next; 4037c478bd9Sstevel@tonic-gate while ((_svc_user_link *) one_node 4047c478bd9Sstevel@tonic-gate != (_svc_user_link *) &(svc_userfds[fd].lnk)) { 4057c478bd9Sstevel@tonic-gate _svc_free_id(one_node->id); 4067c478bd9Sstevel@tonic-gate one_node = one_node->lnk.next; 4077c478bd9Sstevel@tonic-gate free(one_node->lnk.previous); 4087c478bd9Sstevel@tonic-gate } 4097c478bd9Sstevel@tonic-gate 4107c478bd9Sstevel@tonic-gate LIST_CLR(svc_userfds[fd]); 4117c478bd9Sstevel@tonic-gate svc_userfds[fd].mask = 0; 4127c478bd9Sstevel@tonic-gate } 4137c478bd9Sstevel@tonic-gate 4147c478bd9Sstevel@tonic-gate /* 4157c478bd9Sstevel@tonic-gate * Allow user to add an fd in the poll list. If it does not succeed, return 4167c478bd9Sstevel@tonic-gate * -1. Otherwise, return a svc_id 4177c478bd9Sstevel@tonic-gate */ 4187c478bd9Sstevel@tonic-gate 4197c478bd9Sstevel@tonic-gate svc_input_id_t 4207c478bd9Sstevel@tonic-gate svc_add_input(int user_fd, unsigned int events, 4217c478bd9Sstevel@tonic-gate svc_callback_t user_callback, void *cookie) 4227c478bd9Sstevel@tonic-gate { 4237c478bd9Sstevel@tonic-gate _svc_user_fd_node *new_node; 4247c478bd9Sstevel@tonic-gate 4257c478bd9Sstevel@tonic-gate if (user_fd < 0) { 4267c478bd9Sstevel@tonic-gate errno = EINVAL; 4277c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 4287c478bd9Sstevel@tonic-gate } 4297c478bd9Sstevel@tonic-gate 4307c478bd9Sstevel@tonic-gate if ((events == 0x0000) || 4317c478bd9Sstevel@tonic-gate (events & ~(POLLIN|POLLPRI|POLLOUT|POLLRDNORM|POLLRDBAND|\ 4327c478bd9Sstevel@tonic-gate POLLWRBAND|POLLERR|POLLHUP|POLLNVAL))) { 4337c478bd9Sstevel@tonic-gate errno = EINVAL; 4347c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 4357c478bd9Sstevel@tonic-gate } 4367c478bd9Sstevel@tonic-gate 43761961e0fSrobinson (void) mutex_lock(&svc_userfds_lock); 4387c478bd9Sstevel@tonic-gate 4397c478bd9Sstevel@tonic-gate if ((user_fd < svc_nuserfds) && 4407c478bd9Sstevel@tonic-gate (svc_userfds[user_fd].mask & events) != 0) { 4417c478bd9Sstevel@tonic-gate /* Already registrated call-back */ 4427c478bd9Sstevel@tonic-gate errno = EEXIST; 44361961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 4447c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 4457c478bd9Sstevel@tonic-gate } 4467c478bd9Sstevel@tonic-gate 4477c478bd9Sstevel@tonic-gate /* Handle memory allocation. */ 4487c478bd9Sstevel@tonic-gate if (user_fd >= svc_nuserfds) { 4497c478bd9Sstevel@tonic-gate int oldSize = svc_nuserfds; 4507c478bd9Sstevel@tonic-gate int i; 4517c478bd9Sstevel@tonic-gate 4527c478bd9Sstevel@tonic-gate svc_nuserfds = (user_fd + 1) + USER_FD_INCREMENT; 4537c478bd9Sstevel@tonic-gate 4547c478bd9Sstevel@tonic-gate svc_userfds = (_svc_user_fd_head *) 4557c478bd9Sstevel@tonic-gate realloc(svc_userfds, 4567c478bd9Sstevel@tonic-gate svc_nuserfds * sizeof (_svc_user_fd_head)); 4577c478bd9Sstevel@tonic-gate 4587c478bd9Sstevel@tonic-gate if (svc_userfds == NULL) { 4597c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "svc_add_input: out of memory"); 4607c478bd9Sstevel@tonic-gate errno = ENOMEM; 46161961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 4627c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 4637c478bd9Sstevel@tonic-gate } 4647c478bd9Sstevel@tonic-gate 4657c478bd9Sstevel@tonic-gate for (i = oldSize; i < svc_nuserfds; i++) { 4667c478bd9Sstevel@tonic-gate LIST_CLR(svc_userfds[i]); 4677c478bd9Sstevel@tonic-gate svc_userfds[i].mask = 0; 4687c478bd9Sstevel@tonic-gate } 4697c478bd9Sstevel@tonic-gate } 4707c478bd9Sstevel@tonic-gate 47161961e0fSrobinson new_node = malloc(sizeof (_svc_user_fd_node)); 4727c478bd9Sstevel@tonic-gate if (new_node == NULL) { 4737c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "svc_add_input: out of memory"); 4747c478bd9Sstevel@tonic-gate errno = ENOMEM; 47561961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 4767c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 4777c478bd9Sstevel@tonic-gate } 4787c478bd9Sstevel@tonic-gate 4797c478bd9Sstevel@tonic-gate /* create a new node */ 4807c478bd9Sstevel@tonic-gate new_node->fd = user_fd; 4817c478bd9Sstevel@tonic-gate new_node->events = events; 4827c478bd9Sstevel@tonic-gate new_node->callback = user_callback; 4837c478bd9Sstevel@tonic-gate new_node->cookie = cookie; 4847c478bd9Sstevel@tonic-gate 4857c478bd9Sstevel@tonic-gate (void) _svc_attribute_new_id(new_node); 4867c478bd9Sstevel@tonic-gate 4877c478bd9Sstevel@tonic-gate /* Add the new element at the beginning of the list. */ 4887c478bd9Sstevel@tonic-gate if (LIST_ISEMPTY(svc_userfds[user_fd])) { 4897c478bd9Sstevel@tonic-gate svc_userfds[user_fd].lnk.previous = new_node; 4907c478bd9Sstevel@tonic-gate } 4917c478bd9Sstevel@tonic-gate new_node->lnk.next = svc_userfds[user_fd].lnk.next; 4927c478bd9Sstevel@tonic-gate new_node->lnk.previous = (_svc_user_fd_node *)&(svc_userfds[user_fd]); 4937c478bd9Sstevel@tonic-gate 4947c478bd9Sstevel@tonic-gate svc_userfds[user_fd].lnk.next = new_node; 4957c478bd9Sstevel@tonic-gate 4967c478bd9Sstevel@tonic-gate /* refresh global mask for this file desciptor */ 4977c478bd9Sstevel@tonic-gate svc_userfds[user_fd].mask |= events; 4987c478bd9Sstevel@tonic-gate 4997c478bd9Sstevel@tonic-gate /* refresh mask for the poll */ 5007c478bd9Sstevel@tonic-gate add_pollfd(user_fd, (svc_userfds[user_fd].mask)); 5017c478bd9Sstevel@tonic-gate 50261961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 5037c478bd9Sstevel@tonic-gate return (new_node->id); 5047c478bd9Sstevel@tonic-gate } 5057c478bd9Sstevel@tonic-gate 5067c478bd9Sstevel@tonic-gate 5077c478bd9Sstevel@tonic-gate int 5087c478bd9Sstevel@tonic-gate svc_remove_input(svc_input_id_t id) 5097c478bd9Sstevel@tonic-gate { 5107c478bd9Sstevel@tonic-gate _svc_user_fd_node* node; 5117c478bd9Sstevel@tonic-gate _svc_user_fd_node* next; 5127c478bd9Sstevel@tonic-gate _svc_user_fd_node* previous; 5137c478bd9Sstevel@tonic-gate int fd; /* caching optim */ 5147c478bd9Sstevel@tonic-gate 51561961e0fSrobinson (void) mutex_lock(&svc_userfds_lock); 5167c478bd9Sstevel@tonic-gate 5177c478bd9Sstevel@tonic-gate /* Immediately update data for id management */ 5187c478bd9Sstevel@tonic-gate if (user_fd_mgt_array == NULL || id >= svc_nmgtuserfds || 5197c478bd9Sstevel@tonic-gate is_free_id(id)) { 5207c478bd9Sstevel@tonic-gate errno = EINVAL; 52161961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 5227c478bd9Sstevel@tonic-gate return (-1); 5237c478bd9Sstevel@tonic-gate } 5247c478bd9Sstevel@tonic-gate 5257c478bd9Sstevel@tonic-gate node = node_from_id(id); 5267c478bd9Sstevel@tonic-gate assert(node != NULL); 5277c478bd9Sstevel@tonic-gate 5287c478bd9Sstevel@tonic-gate _svc_free_id(id); 5297c478bd9Sstevel@tonic-gate next = node->lnk.next; 5307c478bd9Sstevel@tonic-gate previous = node->lnk.previous; 5317c478bd9Sstevel@tonic-gate fd = node->fd; /* caching optim */ 5327c478bd9Sstevel@tonic-gate 5337c478bd9Sstevel@tonic-gate /* Remove this node from the list. */ 5347c478bd9Sstevel@tonic-gate previous->lnk.next = next; 5357c478bd9Sstevel@tonic-gate next->lnk.previous = previous; 5367c478bd9Sstevel@tonic-gate 5377c478bd9Sstevel@tonic-gate /* Remove the node flags from the global mask */ 5387c478bd9Sstevel@tonic-gate svc_userfds[fd].mask ^= node->events; 5397c478bd9Sstevel@tonic-gate 5407c478bd9Sstevel@tonic-gate free(node); 5417c478bd9Sstevel@tonic-gate if (svc_userfds[fd].mask == 0) { 5427c478bd9Sstevel@tonic-gate LIST_CLR(svc_userfds[fd]); 5437c478bd9Sstevel@tonic-gate assert(LIST_ISEMPTY(svc_userfds[fd])); 5447c478bd9Sstevel@tonic-gate remove_pollfd(fd); 5457c478bd9Sstevel@tonic-gate } 5467c478bd9Sstevel@tonic-gate /* <=> CLEAN NEEDED TO SHRINK MEMORY USAGE */ 5477c478bd9Sstevel@tonic-gate 54861961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 5497c478bd9Sstevel@tonic-gate return (0); 5507c478bd9Sstevel@tonic-gate } 5517c478bd9Sstevel@tonic-gate 5527c478bd9Sstevel@tonic-gate 5537c478bd9Sstevel@tonic-gate /* 5547c478bd9Sstevel@tonic-gate * Provides default service-side functions for authentication flavors 5557c478bd9Sstevel@tonic-gate * that do not use all the fields in struct svc_auth_ops. 5567c478bd9Sstevel@tonic-gate */ 5577c478bd9Sstevel@tonic-gate 5587c478bd9Sstevel@tonic-gate /*ARGSUSED*/ 5597c478bd9Sstevel@tonic-gate static int 56061961e0fSrobinson authany_wrap(AUTH *auth, XDR *xdrs, xdrproc_t xfunc, caddr_t xwhere) 5617c478bd9Sstevel@tonic-gate { 5627c478bd9Sstevel@tonic-gate return (*xfunc)(xdrs, xwhere); 5637c478bd9Sstevel@tonic-gate } 5647c478bd9Sstevel@tonic-gate 5657c478bd9Sstevel@tonic-gate struct svc_auth_ops svc_auth_any_ops = { 5667c478bd9Sstevel@tonic-gate authany_wrap, 5677c478bd9Sstevel@tonic-gate authany_wrap, 5687c478bd9Sstevel@tonic-gate }; 5697c478bd9Sstevel@tonic-gate 5707c478bd9Sstevel@tonic-gate /* 5717c478bd9Sstevel@tonic-gate * Return pointer to server authentication structure. 5727c478bd9Sstevel@tonic-gate */ 5737c478bd9Sstevel@tonic-gate SVCAUTH * 57461961e0fSrobinson __svc_get_svcauth(SVCXPRT *xprt) 5757c478bd9Sstevel@tonic-gate { 5767c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 5777c478bd9Sstevel@tonic-gate return (&SVC_XP_AUTH(xprt)); 5787c478bd9Sstevel@tonic-gate } 5797c478bd9Sstevel@tonic-gate 5807c478bd9Sstevel@tonic-gate /* 5817c478bd9Sstevel@tonic-gate * A callback routine to cleanup after a procedure is executed. 5827c478bd9Sstevel@tonic-gate */ 5837c478bd9Sstevel@tonic-gate void (*__proc_cleanup_cb)() = NULL; 5847c478bd9Sstevel@tonic-gate 5857c478bd9Sstevel@tonic-gate void * 58661961e0fSrobinson __svc_set_proc_cleanup_cb(void *cb) 5877c478bd9Sstevel@tonic-gate { 5887c478bd9Sstevel@tonic-gate void *tmp = (void *)__proc_cleanup_cb; 5897c478bd9Sstevel@tonic-gate 5907c478bd9Sstevel@tonic-gate __proc_cleanup_cb = (void (*)())cb; 5917c478bd9Sstevel@tonic-gate return (tmp); 5927c478bd9Sstevel@tonic-gate } 5937c478bd9Sstevel@tonic-gate 5947c478bd9Sstevel@tonic-gate /* *************** SVCXPRT related stuff **************** */ 5957c478bd9Sstevel@tonic-gate 5967c478bd9Sstevel@tonic-gate 5977c478bd9Sstevel@tonic-gate static int pollfd_shrinking = 1; 5987c478bd9Sstevel@tonic-gate 5997c478bd9Sstevel@tonic-gate 6007c478bd9Sstevel@tonic-gate /* 6017c478bd9Sstevel@tonic-gate * Add fd to svc_pollfd 6027c478bd9Sstevel@tonic-gate */ 6037c478bd9Sstevel@tonic-gate static void 6047c478bd9Sstevel@tonic-gate add_pollfd(int fd, short events) 6057c478bd9Sstevel@tonic-gate { 6067c478bd9Sstevel@tonic-gate if (fd < FD_SETSIZE) { 6077c478bd9Sstevel@tonic-gate FD_SET(fd, &svc_fdset); 6087c478bd9Sstevel@tonic-gate #if !defined(_LP64) 6097c478bd9Sstevel@tonic-gate FD_SET(fd, &_new_svc_fdset); 6107c478bd9Sstevel@tonic-gate #endif 6117c478bd9Sstevel@tonic-gate svc_nfds++; 6127c478bd9Sstevel@tonic-gate svc_nfds_set++; 6137c478bd9Sstevel@tonic-gate if (fd >= svc_max_fd) 6147c478bd9Sstevel@tonic-gate svc_max_fd = fd + 1; 6157c478bd9Sstevel@tonic-gate } 6167c478bd9Sstevel@tonic-gate if (fd >= svc_max_pollfd) 6177c478bd9Sstevel@tonic-gate svc_max_pollfd = fd + 1; 6187c478bd9Sstevel@tonic-gate if (svc_max_pollfd > svc_pollfd_allocd) { 6197c478bd9Sstevel@tonic-gate int i = svc_pollfd_allocd; 6207c478bd9Sstevel@tonic-gate pollfd_t *tmp; 6217c478bd9Sstevel@tonic-gate do { 6227c478bd9Sstevel@tonic-gate svc_pollfd_allocd += POLLFD_EXTEND; 6237c478bd9Sstevel@tonic-gate } while (svc_max_pollfd > svc_pollfd_allocd); 6247c478bd9Sstevel@tonic-gate tmp = realloc(svc_pollfd, 6257c478bd9Sstevel@tonic-gate sizeof (pollfd_t) * svc_pollfd_allocd); 6267c478bd9Sstevel@tonic-gate if (tmp != NULL) { 6277c478bd9Sstevel@tonic-gate svc_pollfd = tmp; 6287c478bd9Sstevel@tonic-gate for (; i < svc_pollfd_allocd; i++) 6297c478bd9Sstevel@tonic-gate POLLFD_CLR(i, tmp); 6307c478bd9Sstevel@tonic-gate } else { 6317c478bd9Sstevel@tonic-gate /* 6327c478bd9Sstevel@tonic-gate * give an error message; undo fdset setting 6337c478bd9Sstevel@tonic-gate * above; reset the pollfd_shrinking flag. 6347c478bd9Sstevel@tonic-gate * because of this poll will not be done 6357c478bd9Sstevel@tonic-gate * on these fds. 6367c478bd9Sstevel@tonic-gate */ 6377c478bd9Sstevel@tonic-gate if (fd < FD_SETSIZE) { 6387c478bd9Sstevel@tonic-gate FD_CLR(fd, &svc_fdset); 6397c478bd9Sstevel@tonic-gate #if !defined(_LP64) 6407c478bd9Sstevel@tonic-gate FD_CLR(fd, &_new_svc_fdset); 6417c478bd9Sstevel@tonic-gate #endif 6427c478bd9Sstevel@tonic-gate svc_nfds--; 6437c478bd9Sstevel@tonic-gate svc_nfds_set--; 6447c478bd9Sstevel@tonic-gate if (fd == (svc_max_fd - 1)) 6457c478bd9Sstevel@tonic-gate svc_max_fd--; 6467c478bd9Sstevel@tonic-gate } 6477c478bd9Sstevel@tonic-gate if (fd == (svc_max_pollfd - 1)) 6487c478bd9Sstevel@tonic-gate svc_max_pollfd--; 6497c478bd9Sstevel@tonic-gate pollfd_shrinking = 0; 6507c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "add_pollfd: out of memory"); 6517c478bd9Sstevel@tonic-gate _exit(1); 6527c478bd9Sstevel@tonic-gate } 6537c478bd9Sstevel@tonic-gate } 6547c478bd9Sstevel@tonic-gate svc_pollfd[fd].fd = fd; 6557c478bd9Sstevel@tonic-gate svc_pollfd[fd].events = events; 6567c478bd9Sstevel@tonic-gate svc_npollfds++; 6577c478bd9Sstevel@tonic-gate svc_npollfds_set++; 6587c478bd9Sstevel@tonic-gate } 6597c478bd9Sstevel@tonic-gate 6607c478bd9Sstevel@tonic-gate /* 6617c478bd9Sstevel@tonic-gate * the fd is still active but only the bit in fdset is cleared. 6627c478bd9Sstevel@tonic-gate * do not subtract svc_nfds or svc_npollfds 6637c478bd9Sstevel@tonic-gate */ 6647c478bd9Sstevel@tonic-gate void 6657c478bd9Sstevel@tonic-gate clear_pollfd(int fd) 6667c478bd9Sstevel@tonic-gate { 6677c478bd9Sstevel@tonic-gate if (fd < FD_SETSIZE && FD_ISSET(fd, &svc_fdset)) { 6687c478bd9Sstevel@tonic-gate FD_CLR(fd, &svc_fdset); 6697c478bd9Sstevel@tonic-gate #if !defined(_LP64) 6707c478bd9Sstevel@tonic-gate FD_CLR(fd, &_new_svc_fdset); 6717c478bd9Sstevel@tonic-gate #endif 6727c478bd9Sstevel@tonic-gate svc_nfds_set--; 6737c478bd9Sstevel@tonic-gate } 6747c478bd9Sstevel@tonic-gate if (fd < svc_pollfd_allocd && POLLFD_ISSET(fd, svc_pollfd)) { 6757c478bd9Sstevel@tonic-gate POLLFD_CLR(fd, svc_pollfd); 6767c478bd9Sstevel@tonic-gate svc_npollfds_set--; 6777c478bd9Sstevel@tonic-gate } 6787c478bd9Sstevel@tonic-gate } 6797c478bd9Sstevel@tonic-gate 6807c478bd9Sstevel@tonic-gate /* 6817c478bd9Sstevel@tonic-gate * sets the bit in fdset for an active fd so that poll() is done for that 6827c478bd9Sstevel@tonic-gate */ 6837c478bd9Sstevel@tonic-gate void 6847c478bd9Sstevel@tonic-gate set_pollfd(int fd, short events) 6857c478bd9Sstevel@tonic-gate { 6867c478bd9Sstevel@tonic-gate if (fd < FD_SETSIZE) { 6877c478bd9Sstevel@tonic-gate FD_SET(fd, &svc_fdset); 6887c478bd9Sstevel@tonic-gate #if !defined(_LP64) 6897c478bd9Sstevel@tonic-gate FD_SET(fd, &_new_svc_fdset); 6907c478bd9Sstevel@tonic-gate #endif 6917c478bd9Sstevel@tonic-gate svc_nfds_set++; 6927c478bd9Sstevel@tonic-gate } 6937c478bd9Sstevel@tonic-gate if (fd < svc_pollfd_allocd) { 6947c478bd9Sstevel@tonic-gate svc_pollfd[fd].fd = fd; 6957c478bd9Sstevel@tonic-gate svc_pollfd[fd].events = events; 6967c478bd9Sstevel@tonic-gate svc_npollfds_set++; 6977c478bd9Sstevel@tonic-gate } 6987c478bd9Sstevel@tonic-gate } 6997c478bd9Sstevel@tonic-gate 7007c478bd9Sstevel@tonic-gate /* 7017c478bd9Sstevel@tonic-gate * remove a svc_pollfd entry; it does not shrink the memory 7027c478bd9Sstevel@tonic-gate */ 7037c478bd9Sstevel@tonic-gate static void 70461961e0fSrobinson remove_pollfd(int fd) 7057c478bd9Sstevel@tonic-gate { 7067c478bd9Sstevel@tonic-gate clear_pollfd(fd); 7077c478bd9Sstevel@tonic-gate if (fd == (svc_max_fd - 1)) 7087c478bd9Sstevel@tonic-gate svc_max_fd--; 7097c478bd9Sstevel@tonic-gate svc_nfds--; 7107c478bd9Sstevel@tonic-gate if (fd == (svc_max_pollfd - 1)) 7117c478bd9Sstevel@tonic-gate svc_max_pollfd--; 7127c478bd9Sstevel@tonic-gate svc_npollfds--; 7137c478bd9Sstevel@tonic-gate } 7147c478bd9Sstevel@tonic-gate 7157c478bd9Sstevel@tonic-gate /* 7167c478bd9Sstevel@tonic-gate * delete a svc_pollfd entry; it shrinks the memory 7177c478bd9Sstevel@tonic-gate * use remove_pollfd if you do not want to shrink 7187c478bd9Sstevel@tonic-gate */ 7197c478bd9Sstevel@tonic-gate static void 7207c478bd9Sstevel@tonic-gate delete_pollfd(int fd) 7217c478bd9Sstevel@tonic-gate { 7227c478bd9Sstevel@tonic-gate remove_pollfd(fd); 7237c478bd9Sstevel@tonic-gate if (pollfd_shrinking && svc_max_pollfd < 7247c478bd9Sstevel@tonic-gate (svc_pollfd_allocd - POLLFD_SHRINK)) { 7257c478bd9Sstevel@tonic-gate do { 7267c478bd9Sstevel@tonic-gate svc_pollfd_allocd -= POLLFD_SHRINK; 7277c478bd9Sstevel@tonic-gate } while (svc_max_pollfd < (svc_pollfd_allocd - POLLFD_SHRINK)); 7287c478bd9Sstevel@tonic-gate svc_pollfd = realloc(svc_pollfd, 7297c478bd9Sstevel@tonic-gate sizeof (pollfd_t) * svc_pollfd_allocd); 7307c478bd9Sstevel@tonic-gate if (svc_pollfd == NULL) { 7317c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "delete_pollfd: out of memory"); 7327c478bd9Sstevel@tonic-gate _exit(1); 7337c478bd9Sstevel@tonic-gate } 7347c478bd9Sstevel@tonic-gate } 7357c478bd9Sstevel@tonic-gate } 7367c478bd9Sstevel@tonic-gate 7377c478bd9Sstevel@tonic-gate 7387c478bd9Sstevel@tonic-gate /* 7397c478bd9Sstevel@tonic-gate * Activate a transport handle. 7407c478bd9Sstevel@tonic-gate */ 7417c478bd9Sstevel@tonic-gate void 74261961e0fSrobinson xprt_register(const SVCXPRT *xprt) 7437c478bd9Sstevel@tonic-gate { 7447c478bd9Sstevel@tonic-gate int fd = xprt->xp_fd; 7457c478bd9Sstevel@tonic-gate #ifdef CALLBACK 7467c478bd9Sstevel@tonic-gate extern void (*_svc_getreqset_proc)(); 7477c478bd9Sstevel@tonic-gate #endif 7487c478bd9Sstevel@tonic-gate /* VARIABLES PROTECTED BY svc_fd_lock: svc_xports, svc_fdset */ 7497c478bd9Sstevel@tonic-gate 75061961e0fSrobinson (void) rw_wrlock(&svc_fd_lock); 7517c478bd9Sstevel@tonic-gate if (svc_xports == NULL) { 7527c478bd9Sstevel@tonic-gate /* allocate some small amount first */ 7537c478bd9Sstevel@tonic-gate svc_xports = calloc(FD_INCREMENT, sizeof (SVCXPRT *)); 7547c478bd9Sstevel@tonic-gate if (svc_xports == NULL) { 7557c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "xprt_register: out of memory"); 7567c478bd9Sstevel@tonic-gate _exit(1); 7577c478bd9Sstevel@tonic-gate } 7587c478bd9Sstevel@tonic-gate nsvc_xports = FD_INCREMENT; 7597c478bd9Sstevel@tonic-gate 7607c478bd9Sstevel@tonic-gate #ifdef CALLBACK 7617c478bd9Sstevel@tonic-gate /* 7627c478bd9Sstevel@tonic-gate * XXX: This code does not keep track of the server state. 7637c478bd9Sstevel@tonic-gate * 7647c478bd9Sstevel@tonic-gate * This provides for callback support. When a client 7657c478bd9Sstevel@tonic-gate * recv's a call from another client on the server fd's, 7667c478bd9Sstevel@tonic-gate * it calls _svc_getreqset_proc() which would return 7677c478bd9Sstevel@tonic-gate * after serving all the server requests. Also look under 7687c478bd9Sstevel@tonic-gate * clnt_dg.c and clnt_vc.c (clnt_call part of it) 7697c478bd9Sstevel@tonic-gate */ 7707c478bd9Sstevel@tonic-gate _svc_getreqset_proc = svc_getreq_poll; 7717c478bd9Sstevel@tonic-gate #endif 7727c478bd9Sstevel@tonic-gate } 7737c478bd9Sstevel@tonic-gate 7747c478bd9Sstevel@tonic-gate while (fd >= nsvc_xports) { 7757c478bd9Sstevel@tonic-gate SVCXPRT **tmp_xprts = svc_xports; 7767c478bd9Sstevel@tonic-gate 7777c478bd9Sstevel@tonic-gate /* time to expand svc_xprts */ 7787c478bd9Sstevel@tonic-gate tmp_xprts = realloc(svc_xports, 7797c478bd9Sstevel@tonic-gate sizeof (SVCXPRT *) * (nsvc_xports + FD_INCREMENT)); 7807c478bd9Sstevel@tonic-gate if (tmp_xprts == NULL) { 7817c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "xprt_register : out of memory."); 7827c478bd9Sstevel@tonic-gate _exit(1); 7837c478bd9Sstevel@tonic-gate } 7847c478bd9Sstevel@tonic-gate 7857c478bd9Sstevel@tonic-gate svc_xports = tmp_xprts; 7867c478bd9Sstevel@tonic-gate (void) memset(&svc_xports[nsvc_xports], 0, 7877c478bd9Sstevel@tonic-gate sizeof (SVCXPRT *) * FD_INCREMENT); 7887c478bd9Sstevel@tonic-gate nsvc_xports += FD_INCREMENT; 7897c478bd9Sstevel@tonic-gate } 7907c478bd9Sstevel@tonic-gate 7917c478bd9Sstevel@tonic-gate svc_xports[fd] = (SVCXPRT *)xprt; 7927c478bd9Sstevel@tonic-gate 7937c478bd9Sstevel@tonic-gate add_pollfd(fd, MASKVAL); 7947c478bd9Sstevel@tonic-gate 7957c478bd9Sstevel@tonic-gate if (svc_polling) { 7967c478bd9Sstevel@tonic-gate char dummy; 7977c478bd9Sstevel@tonic-gate 7987c478bd9Sstevel@tonic-gate /* 7997c478bd9Sstevel@tonic-gate * This happens only in one of the MT modes. 8007c478bd9Sstevel@tonic-gate * Wake up poller. 8017c478bd9Sstevel@tonic-gate */ 80261961e0fSrobinson (void) write(svc_pipe[1], &dummy, sizeof (dummy)); 8037c478bd9Sstevel@tonic-gate } 8047c478bd9Sstevel@tonic-gate /* 8057c478bd9Sstevel@tonic-gate * If already dispatching door based services, start 8067c478bd9Sstevel@tonic-gate * dispatching TLI based services now. 8077c478bd9Sstevel@tonic-gate */ 80861961e0fSrobinson (void) mutex_lock(&svc_door_mutex); 8097c478bd9Sstevel@tonic-gate if (svc_ndoorfds > 0) 81061961e0fSrobinson (void) cond_signal(&svc_door_waitcv); 81161961e0fSrobinson (void) mutex_unlock(&svc_door_mutex); 8127c478bd9Sstevel@tonic-gate 8137c478bd9Sstevel@tonic-gate if (svc_xdrs == NULL) { 8147c478bd9Sstevel@tonic-gate /* allocate initial chunk */ 8157c478bd9Sstevel@tonic-gate svc_xdrs = calloc(FD_INCREMENT, sizeof (XDR *)); 8167c478bd9Sstevel@tonic-gate if (svc_xdrs != NULL) 8177c478bd9Sstevel@tonic-gate nsvc_xdrs = FD_INCREMENT; 8187c478bd9Sstevel@tonic-gate else { 8197c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "xprt_register : out of memory."); 8207c478bd9Sstevel@tonic-gate _exit(1); 8217c478bd9Sstevel@tonic-gate } 8227c478bd9Sstevel@tonic-gate } 82361961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 8247c478bd9Sstevel@tonic-gate } 8257c478bd9Sstevel@tonic-gate 8267c478bd9Sstevel@tonic-gate /* 8277c478bd9Sstevel@tonic-gate * De-activate a transport handle. 8287c478bd9Sstevel@tonic-gate */ 8297c478bd9Sstevel@tonic-gate void 8307c478bd9Sstevel@tonic-gate __xprt_unregister_private(const SVCXPRT *xprt, bool_t lock_not_held) 8317c478bd9Sstevel@tonic-gate { 8327c478bd9Sstevel@tonic-gate int fd = xprt->xp_fd; 8337c478bd9Sstevel@tonic-gate 8347c478bd9Sstevel@tonic-gate if (lock_not_held) 83561961e0fSrobinson (void) rw_wrlock(&svc_fd_lock); 8367c478bd9Sstevel@tonic-gate if ((fd < nsvc_xports) && (svc_xports[fd] == xprt)) { 83761961e0fSrobinson svc_xports[fd] = NULL; 8387c478bd9Sstevel@tonic-gate delete_pollfd(fd); 8397c478bd9Sstevel@tonic-gate } 8407c478bd9Sstevel@tonic-gate if (lock_not_held) 84161961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 8427c478bd9Sstevel@tonic-gate __svc_rm_from_xlist(&_svc_xprtlist, xprt, &xprtlist_lock); 8437c478bd9Sstevel@tonic-gate } 8447c478bd9Sstevel@tonic-gate 8457c478bd9Sstevel@tonic-gate void 84661961e0fSrobinson xprt_unregister(const SVCXPRT *xprt) 8477c478bd9Sstevel@tonic-gate { 8487c478bd9Sstevel@tonic-gate __xprt_unregister_private(xprt, TRUE); 8497c478bd9Sstevel@tonic-gate } 8507c478bd9Sstevel@tonic-gate 8517c478bd9Sstevel@tonic-gate /* ********************** CALLOUT list related stuff ************* */ 8527c478bd9Sstevel@tonic-gate 8537c478bd9Sstevel@tonic-gate /* 8547c478bd9Sstevel@tonic-gate * Add a service program to the callout list. 8557c478bd9Sstevel@tonic-gate * The dispatch routine will be called when a rpc request for this 8567c478bd9Sstevel@tonic-gate * program number comes in. 8577c478bd9Sstevel@tonic-gate */ 8587c478bd9Sstevel@tonic-gate bool_t 85961961e0fSrobinson svc_reg(const SVCXPRT *xprt, const rpcprog_t prog, const rpcvers_t vers, 86061961e0fSrobinson void (*dispatch)(), const struct netconfig *nconf) 8617c478bd9Sstevel@tonic-gate { 8627c478bd9Sstevel@tonic-gate struct svc_callout *prev; 8637c478bd9Sstevel@tonic-gate struct svc_callout *s, **s2; 8647c478bd9Sstevel@tonic-gate struct netconfig *tnconf; 8657c478bd9Sstevel@tonic-gate char *netid = NULL; 8667c478bd9Sstevel@tonic-gate int flag = 0; 8677c478bd9Sstevel@tonic-gate 8687c478bd9Sstevel@tonic-gate /* VARIABLES PROTECTED BY svc_lock: s, prev, svc_head */ 8697c478bd9Sstevel@tonic-gate 8707c478bd9Sstevel@tonic-gate if (xprt->xp_netid) { 8717c478bd9Sstevel@tonic-gate netid = strdup(xprt->xp_netid); 8727c478bd9Sstevel@tonic-gate flag = 1; 8737c478bd9Sstevel@tonic-gate } else if (nconf && nconf->nc_netid) { 8747c478bd9Sstevel@tonic-gate netid = strdup(nconf->nc_netid); 8757c478bd9Sstevel@tonic-gate flag = 1; 8767c478bd9Sstevel@tonic-gate } else if ((tnconf = __rpcfd_to_nconf(xprt->xp_fd, xprt->xp_type)) 8777c478bd9Sstevel@tonic-gate != NULL) { 8787c478bd9Sstevel@tonic-gate netid = strdup(tnconf->nc_netid); 8797c478bd9Sstevel@tonic-gate flag = 1; 8807c478bd9Sstevel@tonic-gate freenetconfigent(tnconf); 8817c478bd9Sstevel@tonic-gate } /* must have been created with svc_raw_create */ 88261961e0fSrobinson if ((netid == NULL) && (flag == 1)) 8837c478bd9Sstevel@tonic-gate return (FALSE); 8847c478bd9Sstevel@tonic-gate 88561961e0fSrobinson (void) rw_wrlock(&svc_lock); 8867c478bd9Sstevel@tonic-gate if ((s = svc_find(prog, vers, &prev, netid)) != NULL_SVC) { 8877c478bd9Sstevel@tonic-gate if (netid) 8887c478bd9Sstevel@tonic-gate free(netid); 8897c478bd9Sstevel@tonic-gate if (s->sc_dispatch == dispatch) 8907c478bd9Sstevel@tonic-gate goto rpcb_it; /* he is registering another xptr */ 89161961e0fSrobinson (void) rw_unlock(&svc_lock); 8927c478bd9Sstevel@tonic-gate return (FALSE); 8937c478bd9Sstevel@tonic-gate } 89461961e0fSrobinson s = malloc(sizeof (struct svc_callout)); 89561961e0fSrobinson if (s == NULL) { 8967c478bd9Sstevel@tonic-gate if (netid) 8977c478bd9Sstevel@tonic-gate free(netid); 89861961e0fSrobinson (void) rw_unlock(&svc_lock); 8997c478bd9Sstevel@tonic-gate return (FALSE); 9007c478bd9Sstevel@tonic-gate } 9017c478bd9Sstevel@tonic-gate 9027c478bd9Sstevel@tonic-gate s->sc_prog = prog; 9037c478bd9Sstevel@tonic-gate s->sc_vers = vers; 9047c478bd9Sstevel@tonic-gate s->sc_dispatch = dispatch; 9057c478bd9Sstevel@tonic-gate s->sc_netid = netid; 9067c478bd9Sstevel@tonic-gate s->sc_next = NULL; 9077c478bd9Sstevel@tonic-gate 9087c478bd9Sstevel@tonic-gate /* 9097c478bd9Sstevel@tonic-gate * The ordering of transports is such that the most frequently used 9107c478bd9Sstevel@tonic-gate * one appears first. So add the new entry to the end of the list. 9117c478bd9Sstevel@tonic-gate */ 9127c478bd9Sstevel@tonic-gate for (s2 = &svc_head; *s2 != NULL; s2 = &(*s2)->sc_next) 9137c478bd9Sstevel@tonic-gate ; 9147c478bd9Sstevel@tonic-gate *s2 = s; 9157c478bd9Sstevel@tonic-gate 9167c478bd9Sstevel@tonic-gate if ((xprt->xp_netid == NULL) && (flag == 1) && netid) 9177c478bd9Sstevel@tonic-gate if ((((SVCXPRT *)xprt)->xp_netid = strdup(netid)) == NULL) { 9187c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "svc_reg : strdup failed."); 9197c478bd9Sstevel@tonic-gate free(netid); 92061961e0fSrobinson free(s); 9217c478bd9Sstevel@tonic-gate *s2 = NULL; 92261961e0fSrobinson (void) rw_unlock(&svc_lock); 9237c478bd9Sstevel@tonic-gate return (FALSE); 9247c478bd9Sstevel@tonic-gate } 9257c478bd9Sstevel@tonic-gate 9267c478bd9Sstevel@tonic-gate rpcb_it: 92761961e0fSrobinson (void) rw_unlock(&svc_lock); 928*9acbbeafSnn35248 if (!nconf) 9297c478bd9Sstevel@tonic-gate return (TRUE); 930*9acbbeafSnn35248 931*9acbbeafSnn35248 /* now register the information with the local binder service */ 932*9acbbeafSnn35248 if (!use_portmapper) 933*9acbbeafSnn35248 return (rpcb_set(prog, vers, nconf, &xprt->xp_ltaddr)); 934*9acbbeafSnn35248 else 935*9acbbeafSnn35248 return (__pmap_set(prog, vers, nconf, &xprt->xp_ltaddr)); 936*9acbbeafSnn35248 /*NOTREACHED*/ 9377c478bd9Sstevel@tonic-gate } 9387c478bd9Sstevel@tonic-gate 9397c478bd9Sstevel@tonic-gate /* 9407c478bd9Sstevel@tonic-gate * Remove a service program from the callout list. 9417c478bd9Sstevel@tonic-gate */ 9427c478bd9Sstevel@tonic-gate void 94361961e0fSrobinson svc_unreg(const rpcprog_t prog, const rpcvers_t vers) 9447c478bd9Sstevel@tonic-gate { 9457c478bd9Sstevel@tonic-gate struct svc_callout *prev; 9467c478bd9Sstevel@tonic-gate struct svc_callout *s; 9477c478bd9Sstevel@tonic-gate 9487c478bd9Sstevel@tonic-gate /* unregister the information anyway */ 949*9acbbeafSnn35248 if (!use_portmapper) 9507c478bd9Sstevel@tonic-gate (void) rpcb_unset(prog, vers, NULL); 951*9acbbeafSnn35248 else 952*9acbbeafSnn35248 (void) __pmap_unset(prog, vers); 95361961e0fSrobinson (void) rw_wrlock(&svc_lock); 9547c478bd9Sstevel@tonic-gate while ((s = svc_find(prog, vers, &prev, NULL)) != NULL_SVC) { 9557c478bd9Sstevel@tonic-gate if (prev == NULL_SVC) { 9567c478bd9Sstevel@tonic-gate svc_head = s->sc_next; 9577c478bd9Sstevel@tonic-gate } else { 9587c478bd9Sstevel@tonic-gate prev->sc_next = s->sc_next; 9597c478bd9Sstevel@tonic-gate } 9607c478bd9Sstevel@tonic-gate s->sc_next = NULL_SVC; 9617c478bd9Sstevel@tonic-gate if (s->sc_netid) 96261961e0fSrobinson free(s->sc_netid); 96361961e0fSrobinson free(s); 9647c478bd9Sstevel@tonic-gate } 96561961e0fSrobinson (void) rw_unlock(&svc_lock); 9667c478bd9Sstevel@tonic-gate } 9677c478bd9Sstevel@tonic-gate 9687c478bd9Sstevel@tonic-gate #ifdef PORTMAP 9697c478bd9Sstevel@tonic-gate /* 9707c478bd9Sstevel@tonic-gate * Add a service program to the callout list. 9717c478bd9Sstevel@tonic-gate * The dispatch routine will be called when a rpc request for this 9727c478bd9Sstevel@tonic-gate * program number comes in. 9737c478bd9Sstevel@tonic-gate * For version 2 portmappers. 9747c478bd9Sstevel@tonic-gate */ 9757c478bd9Sstevel@tonic-gate bool_t 97661961e0fSrobinson svc_register(SVCXPRT *xprt, rpcprog_t prog, rpcvers_t vers, 97761961e0fSrobinson void (*dispatch)(), int protocol) 9787c478bd9Sstevel@tonic-gate { 9797c478bd9Sstevel@tonic-gate struct svc_callout *prev; 9807c478bd9Sstevel@tonic-gate struct svc_callout *s; 9817c478bd9Sstevel@tonic-gate struct netconfig *nconf; 9827c478bd9Sstevel@tonic-gate char *netid = NULL; 9837c478bd9Sstevel@tonic-gate int flag = 0; 9847c478bd9Sstevel@tonic-gate 9857c478bd9Sstevel@tonic-gate if (xprt->xp_netid) { 9867c478bd9Sstevel@tonic-gate netid = strdup(xprt->xp_netid); 9877c478bd9Sstevel@tonic-gate flag = 1; 9887c478bd9Sstevel@tonic-gate } else if ((ioctl(xprt->xp_fd, I_FIND, "timod") > 0) && ((nconf = 9897c478bd9Sstevel@tonic-gate __rpcfd_to_nconf(xprt->xp_fd, xprt->xp_type)) != NULL)) { 9907c478bd9Sstevel@tonic-gate /* fill in missing netid field in SVCXPRT */ 9917c478bd9Sstevel@tonic-gate netid = strdup(nconf->nc_netid); 9927c478bd9Sstevel@tonic-gate flag = 1; 9937c478bd9Sstevel@tonic-gate freenetconfigent(nconf); 9947c478bd9Sstevel@tonic-gate } /* must be svc_raw_create */ 9957c478bd9Sstevel@tonic-gate 99661961e0fSrobinson if ((netid == NULL) && (flag == 1)) 9977c478bd9Sstevel@tonic-gate return (FALSE); 9987c478bd9Sstevel@tonic-gate 99961961e0fSrobinson (void) rw_wrlock(&svc_lock); 10007c478bd9Sstevel@tonic-gate if ((s = svc_find(prog, vers, &prev, netid)) != NULL_SVC) { 10017c478bd9Sstevel@tonic-gate if (netid) 10027c478bd9Sstevel@tonic-gate free(netid); 10037c478bd9Sstevel@tonic-gate if (s->sc_dispatch == dispatch) 10047c478bd9Sstevel@tonic-gate goto pmap_it; /* he is registering another xptr */ 100561961e0fSrobinson (void) rw_unlock(&svc_lock); 10067c478bd9Sstevel@tonic-gate return (FALSE); 10077c478bd9Sstevel@tonic-gate } 100861961e0fSrobinson s = malloc(sizeof (struct svc_callout)); 10097c478bd9Sstevel@tonic-gate if (s == (struct svc_callout *)0) { 10107c478bd9Sstevel@tonic-gate if (netid) 10117c478bd9Sstevel@tonic-gate free(netid); 101261961e0fSrobinson (void) rw_unlock(&svc_lock); 10137c478bd9Sstevel@tonic-gate return (FALSE); 10147c478bd9Sstevel@tonic-gate } 10157c478bd9Sstevel@tonic-gate s->sc_prog = prog; 10167c478bd9Sstevel@tonic-gate s->sc_vers = vers; 10177c478bd9Sstevel@tonic-gate s->sc_dispatch = dispatch; 10187c478bd9Sstevel@tonic-gate s->sc_netid = netid; 10197c478bd9Sstevel@tonic-gate s->sc_next = svc_head; 10207c478bd9Sstevel@tonic-gate svc_head = s; 10217c478bd9Sstevel@tonic-gate 10227c478bd9Sstevel@tonic-gate if ((xprt->xp_netid == NULL) && (flag == 1) && netid) 10237c478bd9Sstevel@tonic-gate if ((xprt->xp_netid = strdup(netid)) == NULL) { 10247c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "svc_register : strdup failed."); 10257c478bd9Sstevel@tonic-gate free(netid); 10267c478bd9Sstevel@tonic-gate svc_head = s->sc_next; 102761961e0fSrobinson free(s); 102861961e0fSrobinson (void) rw_unlock(&svc_lock); 10297c478bd9Sstevel@tonic-gate return (FALSE); 10307c478bd9Sstevel@tonic-gate } 10317c478bd9Sstevel@tonic-gate 10327c478bd9Sstevel@tonic-gate pmap_it: 103361961e0fSrobinson (void) rw_unlock(&svc_lock); 10347c478bd9Sstevel@tonic-gate /* now register the information with the local binder service */ 103561961e0fSrobinson if (protocol) 103661961e0fSrobinson return (pmap_set(prog, vers, protocol, xprt->xp_port)); 10377c478bd9Sstevel@tonic-gate return (TRUE); 10387c478bd9Sstevel@tonic-gate } 10397c478bd9Sstevel@tonic-gate 10407c478bd9Sstevel@tonic-gate /* 10417c478bd9Sstevel@tonic-gate * Remove a service program from the callout list. 10427c478bd9Sstevel@tonic-gate * For version 2 portmappers. 10437c478bd9Sstevel@tonic-gate */ 10447c478bd9Sstevel@tonic-gate void 104561961e0fSrobinson svc_unregister(rpcprog_t prog, rpcvers_t vers) 10467c478bd9Sstevel@tonic-gate { 10477c478bd9Sstevel@tonic-gate struct svc_callout *prev; 10487c478bd9Sstevel@tonic-gate struct svc_callout *s; 10497c478bd9Sstevel@tonic-gate 105061961e0fSrobinson (void) rw_wrlock(&svc_lock); 10517c478bd9Sstevel@tonic-gate while ((s = svc_find(prog, vers, &prev, NULL)) != NULL_SVC) { 10527c478bd9Sstevel@tonic-gate if (prev == NULL_SVC) { 10537c478bd9Sstevel@tonic-gate svc_head = s->sc_next; 10547c478bd9Sstevel@tonic-gate } else { 10557c478bd9Sstevel@tonic-gate prev->sc_next = s->sc_next; 10567c478bd9Sstevel@tonic-gate } 10577c478bd9Sstevel@tonic-gate s->sc_next = NULL_SVC; 10587c478bd9Sstevel@tonic-gate if (s->sc_netid) 105961961e0fSrobinson free(s->sc_netid); 106061961e0fSrobinson free(s); 10617c478bd9Sstevel@tonic-gate /* unregister the information with the local binder service */ 10627c478bd9Sstevel@tonic-gate (void) pmap_unset(prog, vers); 10637c478bd9Sstevel@tonic-gate } 106461961e0fSrobinson (void) rw_unlock(&svc_lock); 10657c478bd9Sstevel@tonic-gate } 10667c478bd9Sstevel@tonic-gate #endif /* PORTMAP */ 106761961e0fSrobinson 10687c478bd9Sstevel@tonic-gate /* 10697c478bd9Sstevel@tonic-gate * Search the callout list for a program number, return the callout 10707c478bd9Sstevel@tonic-gate * struct. 10717c478bd9Sstevel@tonic-gate * Also check for transport as well. Many routines such as svc_unreg 10727c478bd9Sstevel@tonic-gate * dont give any corresponding transport, so dont check for transport if 10737c478bd9Sstevel@tonic-gate * netid == NULL 10747c478bd9Sstevel@tonic-gate */ 10757c478bd9Sstevel@tonic-gate static struct svc_callout * 107661961e0fSrobinson svc_find(rpcprog_t prog, rpcvers_t vers, struct svc_callout **prev, char *netid) 10777c478bd9Sstevel@tonic-gate { 10787c478bd9Sstevel@tonic-gate struct svc_callout *s, *p; 10797c478bd9Sstevel@tonic-gate 10807c478bd9Sstevel@tonic-gate /* WRITE LOCK HELD ON ENTRY: svc_lock */ 10817c478bd9Sstevel@tonic-gate 10827c478bd9Sstevel@tonic-gate /* assert(RW_WRITE_HELD(&svc_lock)); */ 10837c478bd9Sstevel@tonic-gate p = NULL_SVC; 10847c478bd9Sstevel@tonic-gate for (s = svc_head; s != NULL_SVC; s = s->sc_next) { 10857c478bd9Sstevel@tonic-gate if (((s->sc_prog == prog) && (s->sc_vers == vers)) && 10867c478bd9Sstevel@tonic-gate ((netid == NULL) || (s->sc_netid == NULL) || 10877c478bd9Sstevel@tonic-gate (strcmp(netid, s->sc_netid) == 0))) 10887c478bd9Sstevel@tonic-gate break; 10897c478bd9Sstevel@tonic-gate p = s; 10907c478bd9Sstevel@tonic-gate } 10917c478bd9Sstevel@tonic-gate *prev = p; 10927c478bd9Sstevel@tonic-gate return (s); 10937c478bd9Sstevel@tonic-gate } 10947c478bd9Sstevel@tonic-gate 10957c478bd9Sstevel@tonic-gate 10967c478bd9Sstevel@tonic-gate /* ******************* REPLY GENERATION ROUTINES ************ */ 10977c478bd9Sstevel@tonic-gate 10987c478bd9Sstevel@tonic-gate /* 10997c478bd9Sstevel@tonic-gate * Send a reply to an rpc request 11007c478bd9Sstevel@tonic-gate */ 11017c478bd9Sstevel@tonic-gate bool_t 110261961e0fSrobinson svc_sendreply(const SVCXPRT *xprt, const xdrproc_t xdr_results, 110361961e0fSrobinson const caddr_t xdr_location) 11047c478bd9Sstevel@tonic-gate { 11057c478bd9Sstevel@tonic-gate struct rpc_msg rply; 11067c478bd9Sstevel@tonic-gate 11077c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 11087c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 11097c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 11107c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = SUCCESS; 11117c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_results.where = xdr_location; 11127c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_results.proc = xdr_results; 111361961e0fSrobinson return (SVC_REPLY((SVCXPRT *)xprt, &rply)); 11147c478bd9Sstevel@tonic-gate } 11157c478bd9Sstevel@tonic-gate 11167c478bd9Sstevel@tonic-gate /* 11177c478bd9Sstevel@tonic-gate * No procedure error reply 11187c478bd9Sstevel@tonic-gate */ 11197c478bd9Sstevel@tonic-gate void 112061961e0fSrobinson svcerr_noproc(const SVCXPRT *xprt) 11217c478bd9Sstevel@tonic-gate { 11227c478bd9Sstevel@tonic-gate struct rpc_msg rply; 11237c478bd9Sstevel@tonic-gate 11247c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 11257c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 11267c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 11277c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = PROC_UNAVAIL; 11287c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 11297c478bd9Sstevel@tonic-gate } 11307c478bd9Sstevel@tonic-gate 11317c478bd9Sstevel@tonic-gate /* 11327c478bd9Sstevel@tonic-gate * Can't decode args error reply 11337c478bd9Sstevel@tonic-gate */ 11347c478bd9Sstevel@tonic-gate void 113561961e0fSrobinson svcerr_decode(const SVCXPRT *xprt) 11367c478bd9Sstevel@tonic-gate { 11377c478bd9Sstevel@tonic-gate struct rpc_msg rply; 11387c478bd9Sstevel@tonic-gate 11397c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 11407c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 11417c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 11427c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = GARBAGE_ARGS; 11437c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 11447c478bd9Sstevel@tonic-gate } 11457c478bd9Sstevel@tonic-gate 11467c478bd9Sstevel@tonic-gate /* 11477c478bd9Sstevel@tonic-gate * Some system error 11487c478bd9Sstevel@tonic-gate */ 11497c478bd9Sstevel@tonic-gate void 115061961e0fSrobinson svcerr_systemerr(const SVCXPRT *xprt) 11517c478bd9Sstevel@tonic-gate { 11527c478bd9Sstevel@tonic-gate struct rpc_msg rply; 11537c478bd9Sstevel@tonic-gate 11547c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 11557c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 11567c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 11577c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = SYSTEM_ERR; 11587c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 11597c478bd9Sstevel@tonic-gate } 11607c478bd9Sstevel@tonic-gate 11617c478bd9Sstevel@tonic-gate /* 11627c478bd9Sstevel@tonic-gate * Tell RPC package to not complain about version errors to the client. This 11637c478bd9Sstevel@tonic-gate * is useful when revving broadcast protocols that sit on a fixed address. 11647c478bd9Sstevel@tonic-gate * There is really one (or should be only one) example of this kind of 11657c478bd9Sstevel@tonic-gate * protocol: the portmapper (or rpc binder). 11667c478bd9Sstevel@tonic-gate */ 11677c478bd9Sstevel@tonic-gate void 116861961e0fSrobinson __svc_versquiet_on(const SVCXPRT *xprt) 11697c478bd9Sstevel@tonic-gate { 11707c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 11717c478bd9Sstevel@tonic-gate svc_flags(xprt) |= SVC_VERSQUIET; 11727c478bd9Sstevel@tonic-gate } 11737c478bd9Sstevel@tonic-gate 11747c478bd9Sstevel@tonic-gate void 117561961e0fSrobinson __svc_versquiet_off(const SVCXPRT *xprt) 11767c478bd9Sstevel@tonic-gate { 11777c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 11787c478bd9Sstevel@tonic-gate svc_flags(xprt) &= ~SVC_VERSQUIET; 11797c478bd9Sstevel@tonic-gate } 11807c478bd9Sstevel@tonic-gate 11817c478bd9Sstevel@tonic-gate void 118261961e0fSrobinson svc_versquiet(const SVCXPRT *xprt) 11837c478bd9Sstevel@tonic-gate { 11847c478bd9Sstevel@tonic-gate __svc_versquiet_on(xprt); 11857c478bd9Sstevel@tonic-gate } 11867c478bd9Sstevel@tonic-gate 11877c478bd9Sstevel@tonic-gate int 118861961e0fSrobinson __svc_versquiet_get(const SVCXPRT *xprt) 11897c478bd9Sstevel@tonic-gate { 11907c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 11917c478bd9Sstevel@tonic-gate return (svc_flags(xprt) & SVC_VERSQUIET); 11927c478bd9Sstevel@tonic-gate } 11937c478bd9Sstevel@tonic-gate 11947c478bd9Sstevel@tonic-gate /* 11957c478bd9Sstevel@tonic-gate * Authentication error reply 11967c478bd9Sstevel@tonic-gate */ 11977c478bd9Sstevel@tonic-gate void 119861961e0fSrobinson svcerr_auth(const SVCXPRT *xprt, const enum auth_stat why) 11997c478bd9Sstevel@tonic-gate { 12007c478bd9Sstevel@tonic-gate struct rpc_msg rply; 12017c478bd9Sstevel@tonic-gate 12027c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 12037c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_DENIED; 12047c478bd9Sstevel@tonic-gate rply.rjcted_rply.rj_stat = AUTH_ERROR; 12057c478bd9Sstevel@tonic-gate rply.rjcted_rply.rj_why = why; 12067c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 12077c478bd9Sstevel@tonic-gate } 12087c478bd9Sstevel@tonic-gate 12097c478bd9Sstevel@tonic-gate /* 12107c478bd9Sstevel@tonic-gate * Auth too weak error reply 12117c478bd9Sstevel@tonic-gate */ 12127c478bd9Sstevel@tonic-gate void 121361961e0fSrobinson svcerr_weakauth(const SVCXPRT *xprt) 12147c478bd9Sstevel@tonic-gate { 12157c478bd9Sstevel@tonic-gate svcerr_auth(xprt, AUTH_TOOWEAK); 12167c478bd9Sstevel@tonic-gate } 12177c478bd9Sstevel@tonic-gate 12187c478bd9Sstevel@tonic-gate /* 12197c478bd9Sstevel@tonic-gate * Program unavailable error reply 12207c478bd9Sstevel@tonic-gate */ 12217c478bd9Sstevel@tonic-gate void 122261961e0fSrobinson svcerr_noprog(const SVCXPRT *xprt) 12237c478bd9Sstevel@tonic-gate { 12247c478bd9Sstevel@tonic-gate struct rpc_msg rply; 12257c478bd9Sstevel@tonic-gate 12267c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 12277c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 12287c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 12297c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = PROG_UNAVAIL; 12307c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 12317c478bd9Sstevel@tonic-gate } 12327c478bd9Sstevel@tonic-gate 12337c478bd9Sstevel@tonic-gate /* 12347c478bd9Sstevel@tonic-gate * Program version mismatch error reply 12357c478bd9Sstevel@tonic-gate */ 12367c478bd9Sstevel@tonic-gate void 123761961e0fSrobinson svcerr_progvers(const SVCXPRT *xprt, const rpcvers_t low_vers, 123861961e0fSrobinson const rpcvers_t high_vers) 12397c478bd9Sstevel@tonic-gate { 12407c478bd9Sstevel@tonic-gate struct rpc_msg rply; 12417c478bd9Sstevel@tonic-gate 12427c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 12437c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 12447c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 12457c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = PROG_MISMATCH; 12467c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_vers.low = low_vers; 12477c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_vers.high = high_vers; 12487c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 12497c478bd9Sstevel@tonic-gate } 12507c478bd9Sstevel@tonic-gate 12517c478bd9Sstevel@tonic-gate /* ******************* SERVER INPUT STUFF ******************* */ 12527c478bd9Sstevel@tonic-gate 12537c478bd9Sstevel@tonic-gate /* 12547c478bd9Sstevel@tonic-gate * Get server side input from some transport. 12557c478bd9Sstevel@tonic-gate * 12567c478bd9Sstevel@tonic-gate * Statement of authentication parameters management: 12577c478bd9Sstevel@tonic-gate * This function owns and manages all authentication parameters, specifically 12587c478bd9Sstevel@tonic-gate * the "raw" parameters (msg.rm_call.cb_cred and msg.rm_call.cb_verf) and 12597c478bd9Sstevel@tonic-gate * the "cooked" credentials (rqst->rq_clntcred). 12607c478bd9Sstevel@tonic-gate * However, this function does not know the structure of the cooked 12617c478bd9Sstevel@tonic-gate * credentials, so it make the following assumptions: 12627c478bd9Sstevel@tonic-gate * a) the structure is contiguous (no pointers), and 12637c478bd9Sstevel@tonic-gate * b) the cred structure size does not exceed RQCRED_SIZE bytes. 12647c478bd9Sstevel@tonic-gate * In all events, all three parameters are freed upon exit from this routine. 12657c478bd9Sstevel@tonic-gate * The storage is trivially management on the call stack in user land, but 12667c478bd9Sstevel@tonic-gate * is mallocated in kernel land. 12677c478bd9Sstevel@tonic-gate */ 12687c478bd9Sstevel@tonic-gate 12697c478bd9Sstevel@tonic-gate void 127061961e0fSrobinson svc_getreq(int rdfds) 12717c478bd9Sstevel@tonic-gate { 12727c478bd9Sstevel@tonic-gate fd_set readfds; 12737c478bd9Sstevel@tonic-gate 12747c478bd9Sstevel@tonic-gate FD_ZERO(&readfds); 12757c478bd9Sstevel@tonic-gate readfds.fds_bits[0] = rdfds; 12767c478bd9Sstevel@tonic-gate svc_getreqset(&readfds); 12777c478bd9Sstevel@tonic-gate } 12787c478bd9Sstevel@tonic-gate 12797c478bd9Sstevel@tonic-gate void 128061961e0fSrobinson svc_getreqset(fd_set *readfds) 12817c478bd9Sstevel@tonic-gate { 12827c478bd9Sstevel@tonic-gate int i; 12837c478bd9Sstevel@tonic-gate 12847c478bd9Sstevel@tonic-gate for (i = 0; i < svc_max_fd; i++) { 12857c478bd9Sstevel@tonic-gate /* fd has input waiting */ 12867c478bd9Sstevel@tonic-gate if (FD_ISSET(i, readfds)) 12877c478bd9Sstevel@tonic-gate svc_getreq_common(i); 12887c478bd9Sstevel@tonic-gate } 12897c478bd9Sstevel@tonic-gate } 12907c478bd9Sstevel@tonic-gate 12917c478bd9Sstevel@tonic-gate void 129261961e0fSrobinson svc_getreq_poll(struct pollfd *pfdp, const int pollretval) 12937c478bd9Sstevel@tonic-gate { 12947c478bd9Sstevel@tonic-gate int i; 12957c478bd9Sstevel@tonic-gate int fds_found; 12967c478bd9Sstevel@tonic-gate 12977c478bd9Sstevel@tonic-gate for (i = fds_found = 0; fds_found < pollretval; i++) { 12987c478bd9Sstevel@tonic-gate struct pollfd *p = &pfdp[i]; 12997c478bd9Sstevel@tonic-gate 13007c478bd9Sstevel@tonic-gate if (p->revents) { 13017c478bd9Sstevel@tonic-gate /* fd has input waiting */ 13027c478bd9Sstevel@tonic-gate fds_found++; 13037c478bd9Sstevel@tonic-gate /* 13047c478bd9Sstevel@tonic-gate * We assume that this function is only called 13057c478bd9Sstevel@tonic-gate * via someone select()ing from svc_fdset or 13067c478bd9Sstevel@tonic-gate * poll()ing from svc_pollset[]. Thus it's safe 13077c478bd9Sstevel@tonic-gate * to handle the POLLNVAL event by simply turning 13087c478bd9Sstevel@tonic-gate * the corresponding bit off in svc_fdset. The 13097c478bd9Sstevel@tonic-gate * svc_pollset[] array is derived from svc_fdset 13107c478bd9Sstevel@tonic-gate * and so will also be updated eventually. 13117c478bd9Sstevel@tonic-gate * 13127c478bd9Sstevel@tonic-gate * XXX Should we do an xprt_unregister() instead? 13137c478bd9Sstevel@tonic-gate */ 13147c478bd9Sstevel@tonic-gate /* Handle user callback */ 13157c478bd9Sstevel@tonic-gate if (__is_a_userfd(p->fd) == TRUE) { 131661961e0fSrobinson (void) rw_rdlock(&svc_fd_lock); 13177c478bd9Sstevel@tonic-gate __svc_getreq_user(p); 131861961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13197c478bd9Sstevel@tonic-gate } else { 13207c478bd9Sstevel@tonic-gate if (p->revents & POLLNVAL) { 132161961e0fSrobinson (void) rw_wrlock(&svc_fd_lock); 13227c478bd9Sstevel@tonic-gate remove_pollfd(p->fd); /* XXX */ 132361961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 132461961e0fSrobinson } else { 13257c478bd9Sstevel@tonic-gate svc_getreq_common(p->fd); 13267c478bd9Sstevel@tonic-gate } 13277c478bd9Sstevel@tonic-gate } 13287c478bd9Sstevel@tonic-gate } 132961961e0fSrobinson } 13307c478bd9Sstevel@tonic-gate } 13317c478bd9Sstevel@tonic-gate 13327c478bd9Sstevel@tonic-gate void 133361961e0fSrobinson svc_getreq_common(const int fd) 13347c478bd9Sstevel@tonic-gate { 13357c478bd9Sstevel@tonic-gate SVCXPRT *xprt; 13367c478bd9Sstevel@tonic-gate enum xprt_stat stat; 13377c478bd9Sstevel@tonic-gate struct rpc_msg *msg; 13387c478bd9Sstevel@tonic-gate struct svc_req *r; 13397c478bd9Sstevel@tonic-gate char *cred_area; 13407c478bd9Sstevel@tonic-gate 134161961e0fSrobinson (void) rw_rdlock(&svc_fd_lock); 13427c478bd9Sstevel@tonic-gate 13437c478bd9Sstevel@tonic-gate /* HANDLE USER CALLBACK */ 13447c478bd9Sstevel@tonic-gate if (__is_a_userfd(fd) == TRUE) { 13457c478bd9Sstevel@tonic-gate struct pollfd virtual_fd; 13467c478bd9Sstevel@tonic-gate 13477c478bd9Sstevel@tonic-gate virtual_fd.events = virtual_fd.revents = (short)0xFFFF; 13487c478bd9Sstevel@tonic-gate virtual_fd.fd = fd; 13497c478bd9Sstevel@tonic-gate __svc_getreq_user(&virtual_fd); 135061961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13517c478bd9Sstevel@tonic-gate return; 13527c478bd9Sstevel@tonic-gate } 13537c478bd9Sstevel@tonic-gate 13547c478bd9Sstevel@tonic-gate /* 13557c478bd9Sstevel@tonic-gate * The transport associated with this fd could have been 13567c478bd9Sstevel@tonic-gate * removed from svc_timeout_nonblock_xprt_and_LRU, for instance. 13577c478bd9Sstevel@tonic-gate * This can happen if two or more fds get read events and are 13587c478bd9Sstevel@tonic-gate * passed to svc_getreq_poll/set, the first fd is seviced by 13597c478bd9Sstevel@tonic-gate * the dispatch routine and cleans up any dead transports. If 13607c478bd9Sstevel@tonic-gate * one of the dead transports removed is the other fd that 13617c478bd9Sstevel@tonic-gate * had a read event then svc_getreq_common() will be called with no 13627c478bd9Sstevel@tonic-gate * xprt associated with the fd that had the original read event. 13637c478bd9Sstevel@tonic-gate */ 13647c478bd9Sstevel@tonic-gate if ((fd >= nsvc_xports) || (xprt = svc_xports[fd]) == NULL) { 136561961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13667c478bd9Sstevel@tonic-gate return; 13677c478bd9Sstevel@tonic-gate } 136861961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13697c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 13707c478bd9Sstevel@tonic-gate msg = SVCEXT(xprt)->msg; 13717c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 13727c478bd9Sstevel@tonic-gate r = SVCEXT(xprt)->req; 13737c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 13747c478bd9Sstevel@tonic-gate cred_area = SVCEXT(xprt)->cred_area; 13757c478bd9Sstevel@tonic-gate msg->rm_call.cb_cred.oa_base = cred_area; 13767c478bd9Sstevel@tonic-gate msg->rm_call.cb_verf.oa_base = &(cred_area[MAX_AUTH_BYTES]); 13777c478bd9Sstevel@tonic-gate r->rq_clntcred = &(cred_area[2 * MAX_AUTH_BYTES]); 13787c478bd9Sstevel@tonic-gate 13797c478bd9Sstevel@tonic-gate /* receive msgs from xprtprt (support batch calls) */ 13807c478bd9Sstevel@tonic-gate do { 13817c478bd9Sstevel@tonic-gate bool_t dispatch; 13827c478bd9Sstevel@tonic-gate 13837c478bd9Sstevel@tonic-gate if (dispatch = SVC_RECV(xprt, msg)) 13847c478bd9Sstevel@tonic-gate (void) _svc_prog_dispatch(xprt, msg, r); 13857c478bd9Sstevel@tonic-gate /* 13867c478bd9Sstevel@tonic-gate * Check if the xprt has been disconnected in a recursive call 13877c478bd9Sstevel@tonic-gate * in the service dispatch routine. If so, then break 13887c478bd9Sstevel@tonic-gate */ 138961961e0fSrobinson (void) rw_rdlock(&svc_fd_lock); 13907c478bd9Sstevel@tonic-gate if (xprt != svc_xports[fd]) { 139161961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13927c478bd9Sstevel@tonic-gate break; 13937c478bd9Sstevel@tonic-gate } 139461961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13957c478bd9Sstevel@tonic-gate 13967c478bd9Sstevel@tonic-gate /* 13977c478bd9Sstevel@tonic-gate * Call cleanup procedure if set. 13987c478bd9Sstevel@tonic-gate */ 13997c478bd9Sstevel@tonic-gate if (__proc_cleanup_cb != NULL && dispatch) 14007c478bd9Sstevel@tonic-gate (*__proc_cleanup_cb)(xprt); 14017c478bd9Sstevel@tonic-gate 14027c478bd9Sstevel@tonic-gate if ((stat = SVC_STAT(xprt)) == XPRT_DIED) { 14037c478bd9Sstevel@tonic-gate SVC_DESTROY(xprt); 14047c478bd9Sstevel@tonic-gate break; 14057c478bd9Sstevel@tonic-gate } 14067c478bd9Sstevel@tonic-gate } while (stat == XPRT_MOREREQS); 14077c478bd9Sstevel@tonic-gate } 14087c478bd9Sstevel@tonic-gate 14097c478bd9Sstevel@tonic-gate int 141061961e0fSrobinson _svc_prog_dispatch(SVCXPRT *xprt, struct rpc_msg *msg, struct svc_req *r) 14117c478bd9Sstevel@tonic-gate { 14127c478bd9Sstevel@tonic-gate struct svc_callout *s; 14137c478bd9Sstevel@tonic-gate enum auth_stat why; 14147c478bd9Sstevel@tonic-gate int prog_found; 14157c478bd9Sstevel@tonic-gate rpcvers_t low_vers; 14167c478bd9Sstevel@tonic-gate rpcvers_t high_vers; 14177c478bd9Sstevel@tonic-gate void (*disp_fn)(); 14187c478bd9Sstevel@tonic-gate 14197c478bd9Sstevel@tonic-gate r->rq_xprt = xprt; 14207c478bd9Sstevel@tonic-gate r->rq_prog = msg->rm_call.cb_prog; 14217c478bd9Sstevel@tonic-gate r->rq_vers = msg->rm_call.cb_vers; 14227c478bd9Sstevel@tonic-gate r->rq_proc = msg->rm_call.cb_proc; 14237c478bd9Sstevel@tonic-gate r->rq_cred = msg->rm_call.cb_cred; 14247c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 14257c478bd9Sstevel@tonic-gate SVC_XP_AUTH(r->rq_xprt).svc_ah_ops = svc_auth_any_ops; 14267c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 14277c478bd9Sstevel@tonic-gate SVC_XP_AUTH(r->rq_xprt).svc_ah_private = NULL; 14287c478bd9Sstevel@tonic-gate 14297c478bd9Sstevel@tonic-gate /* first authenticate the message */ 14307c478bd9Sstevel@tonic-gate /* Check for null flavor and bypass these calls if possible */ 14317c478bd9Sstevel@tonic-gate 14327c478bd9Sstevel@tonic-gate if (msg->rm_call.cb_cred.oa_flavor == AUTH_NULL) { 14337c478bd9Sstevel@tonic-gate r->rq_xprt->xp_verf.oa_flavor = _null_auth.oa_flavor; 14347c478bd9Sstevel@tonic-gate r->rq_xprt->xp_verf.oa_length = 0; 14357c478bd9Sstevel@tonic-gate } else { 14367c478bd9Sstevel@tonic-gate bool_t no_dispatch; 14377c478bd9Sstevel@tonic-gate 14387c478bd9Sstevel@tonic-gate if ((why = __gss_authenticate(r, msg, 14397c478bd9Sstevel@tonic-gate &no_dispatch)) != AUTH_OK) { 14407c478bd9Sstevel@tonic-gate svcerr_auth(xprt, why); 14417c478bd9Sstevel@tonic-gate return (0); 14427c478bd9Sstevel@tonic-gate } 14437c478bd9Sstevel@tonic-gate if (no_dispatch) 14447c478bd9Sstevel@tonic-gate return (0); 14457c478bd9Sstevel@tonic-gate } 14467c478bd9Sstevel@tonic-gate /* match message with a registered service */ 14477c478bd9Sstevel@tonic-gate prog_found = FALSE; 14487c478bd9Sstevel@tonic-gate low_vers = (rpcvers_t)(0 - 1); 14497c478bd9Sstevel@tonic-gate high_vers = 0; 145061961e0fSrobinson (void) rw_rdlock(&svc_lock); 14517c478bd9Sstevel@tonic-gate for (s = svc_head; s != NULL_SVC; s = s->sc_next) { 14527c478bd9Sstevel@tonic-gate if (s->sc_prog == r->rq_prog) { 14537c478bd9Sstevel@tonic-gate prog_found = TRUE; 14547c478bd9Sstevel@tonic-gate if (s->sc_vers == r->rq_vers) { 14557c478bd9Sstevel@tonic-gate if ((xprt->xp_netid == NULL) || 14567c478bd9Sstevel@tonic-gate (s->sc_netid == NULL) || 14577c478bd9Sstevel@tonic-gate (strcmp(xprt->xp_netid, 14587c478bd9Sstevel@tonic-gate s->sc_netid) == 0)) { 14597c478bd9Sstevel@tonic-gate disp_fn = (*s->sc_dispatch); 146061961e0fSrobinson (void) rw_unlock(&svc_lock); 14617c478bd9Sstevel@tonic-gate disp_fn(r, xprt); 14627c478bd9Sstevel@tonic-gate return (1); 14637c478bd9Sstevel@tonic-gate } 146461961e0fSrobinson prog_found = FALSE; 14657c478bd9Sstevel@tonic-gate } 14667c478bd9Sstevel@tonic-gate if (s->sc_vers < low_vers) 14677c478bd9Sstevel@tonic-gate low_vers = s->sc_vers; 14687c478bd9Sstevel@tonic-gate if (s->sc_vers > high_vers) 14697c478bd9Sstevel@tonic-gate high_vers = s->sc_vers; 14707c478bd9Sstevel@tonic-gate } /* found correct program */ 14717c478bd9Sstevel@tonic-gate } 147261961e0fSrobinson (void) rw_unlock(&svc_lock); 14737c478bd9Sstevel@tonic-gate 14747c478bd9Sstevel@tonic-gate /* 14757c478bd9Sstevel@tonic-gate * if we got here, the program or version 14767c478bd9Sstevel@tonic-gate * is not served ... 14777c478bd9Sstevel@tonic-gate */ 14787c478bd9Sstevel@tonic-gate if (prog_found) { 14797c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 14807c478bd9Sstevel@tonic-gate if (!version_keepquiet(xprt)) 14817c478bd9Sstevel@tonic-gate svcerr_progvers(xprt, low_vers, high_vers); 14827c478bd9Sstevel@tonic-gate } else { 14837c478bd9Sstevel@tonic-gate svcerr_noprog(xprt); 14847c478bd9Sstevel@tonic-gate } 14857c478bd9Sstevel@tonic-gate return (0); 14867c478bd9Sstevel@tonic-gate } 14877c478bd9Sstevel@tonic-gate 14887c478bd9Sstevel@tonic-gate /* ******************* SVCXPRT allocation and deallocation ***************** */ 14897c478bd9Sstevel@tonic-gate 14907c478bd9Sstevel@tonic-gate /* 14917c478bd9Sstevel@tonic-gate * svc_xprt_alloc() - allocate a service transport handle 14927c478bd9Sstevel@tonic-gate */ 14937c478bd9Sstevel@tonic-gate SVCXPRT * 149461961e0fSrobinson svc_xprt_alloc(void) 14957c478bd9Sstevel@tonic-gate { 14967c478bd9Sstevel@tonic-gate SVCXPRT *xprt = NULL; 14977c478bd9Sstevel@tonic-gate SVCXPRT_EXT *xt = NULL; 14987c478bd9Sstevel@tonic-gate SVCXPRT_LIST *xlist = NULL; 14997c478bd9Sstevel@tonic-gate struct rpc_msg *msg = NULL; 15007c478bd9Sstevel@tonic-gate struct svc_req *req = NULL; 15017c478bd9Sstevel@tonic-gate char *cred_area = NULL; 15027c478bd9Sstevel@tonic-gate 150361961e0fSrobinson if ((xprt = calloc(1, sizeof (SVCXPRT))) == NULL) 15047c478bd9Sstevel@tonic-gate goto err_exit; 15057c478bd9Sstevel@tonic-gate 150661961e0fSrobinson if ((xt = calloc(1, sizeof (SVCXPRT_EXT))) == NULL) 15077c478bd9Sstevel@tonic-gate goto err_exit; 15087c478bd9Sstevel@tonic-gate xprt->xp_p3 = (caddr_t)xt; /* SVCEXT(xprt) = xt */ 15097c478bd9Sstevel@tonic-gate 151061961e0fSrobinson if ((xlist = calloc(1, sizeof (SVCXPRT_LIST))) == NULL) 15117c478bd9Sstevel@tonic-gate goto err_exit; 15127c478bd9Sstevel@tonic-gate xt->my_xlist = xlist; 15137c478bd9Sstevel@tonic-gate xlist->xprt = xprt; 15147c478bd9Sstevel@tonic-gate 151561961e0fSrobinson if ((msg = malloc(sizeof (struct rpc_msg))) == NULL) 15167c478bd9Sstevel@tonic-gate goto err_exit; 15177c478bd9Sstevel@tonic-gate xt->msg = msg; 15187c478bd9Sstevel@tonic-gate 151961961e0fSrobinson if ((req = malloc(sizeof (struct svc_req))) == NULL) 15207c478bd9Sstevel@tonic-gate goto err_exit; 15217c478bd9Sstevel@tonic-gate xt->req = req; 15227c478bd9Sstevel@tonic-gate 152361961e0fSrobinson if ((cred_area = malloc(2*MAX_AUTH_BYTES + RQCRED_SIZE)) == NULL) 15247c478bd9Sstevel@tonic-gate goto err_exit; 15257c478bd9Sstevel@tonic-gate xt->cred_area = cred_area; 15267c478bd9Sstevel@tonic-gate 15277c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 152861961e0fSrobinson (void) mutex_init(&svc_send_mutex(xprt), USYNC_THREAD, (void *)0); 15297c478bd9Sstevel@tonic-gate return (xprt); 15307c478bd9Sstevel@tonic-gate 15317c478bd9Sstevel@tonic-gate err_exit: 15327c478bd9Sstevel@tonic-gate svc_xprt_free(xprt); 15337c478bd9Sstevel@tonic-gate return (NULL); 15347c478bd9Sstevel@tonic-gate } 15357c478bd9Sstevel@tonic-gate 15367c478bd9Sstevel@tonic-gate 15377c478bd9Sstevel@tonic-gate /* 15387c478bd9Sstevel@tonic-gate * svc_xprt_free() - free a service handle 15397c478bd9Sstevel@tonic-gate */ 15407c478bd9Sstevel@tonic-gate void 154161961e0fSrobinson svc_xprt_free(SVCXPRT *xprt) 15427c478bd9Sstevel@tonic-gate { 15437c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15447c478bd9Sstevel@tonic-gate SVCXPRT_EXT *xt = xprt ? SVCEXT(xprt) : NULL; 15457c478bd9Sstevel@tonic-gate SVCXPRT_LIST *my_xlist = xt ? xt->my_xlist: NULL; 15467c478bd9Sstevel@tonic-gate struct rpc_msg *msg = xt ? xt->msg : NULL; 15477c478bd9Sstevel@tonic-gate struct svc_req *req = xt ? xt->req : NULL; 15487c478bd9Sstevel@tonic-gate char *cred_area = xt ? xt->cred_area : NULL; 15497c478bd9Sstevel@tonic-gate 15507c478bd9Sstevel@tonic-gate if (xprt) 155161961e0fSrobinson free(xprt); 15527c478bd9Sstevel@tonic-gate if (xt) 155361961e0fSrobinson free(xt); 15547c478bd9Sstevel@tonic-gate if (my_xlist) 155561961e0fSrobinson free(my_xlist); 15567c478bd9Sstevel@tonic-gate if (msg) 155761961e0fSrobinson free(msg); 15587c478bd9Sstevel@tonic-gate if (req) 155961961e0fSrobinson free(req); 15607c478bd9Sstevel@tonic-gate if (cred_area) 156161961e0fSrobinson free(cred_area); 15627c478bd9Sstevel@tonic-gate } 15637c478bd9Sstevel@tonic-gate 15647c478bd9Sstevel@tonic-gate 15657c478bd9Sstevel@tonic-gate /* 15667c478bd9Sstevel@tonic-gate * svc_xprt_destroy() - free parent and child xprt list 15677c478bd9Sstevel@tonic-gate */ 15687c478bd9Sstevel@tonic-gate void 156961961e0fSrobinson svc_xprt_destroy(SVCXPRT *xprt) 15707c478bd9Sstevel@tonic-gate { 15717c478bd9Sstevel@tonic-gate SVCXPRT_LIST *xlist, *xnext = NULL; 15727c478bd9Sstevel@tonic-gate int type; 15737c478bd9Sstevel@tonic-gate 15747c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15757c478bd9Sstevel@tonic-gate if (SVCEXT(xprt)->parent) 15767c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15777c478bd9Sstevel@tonic-gate xprt = SVCEXT(xprt)->parent; 15787c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15797c478bd9Sstevel@tonic-gate type = svc_type(xprt); 15807c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15817c478bd9Sstevel@tonic-gate for (xlist = SVCEXT(xprt)->my_xlist; xlist != NULL; xlist = xnext) { 15827c478bd9Sstevel@tonic-gate xnext = xlist->next; 15837c478bd9Sstevel@tonic-gate xprt = xlist->xprt; 15847c478bd9Sstevel@tonic-gate switch (type) { 15857c478bd9Sstevel@tonic-gate case SVC_DGRAM: 15867c478bd9Sstevel@tonic-gate svc_dg_xprtfree(xprt); 15877c478bd9Sstevel@tonic-gate break; 15887c478bd9Sstevel@tonic-gate case SVC_RENDEZVOUS: 15897c478bd9Sstevel@tonic-gate svc_vc_xprtfree(xprt); 15907c478bd9Sstevel@tonic-gate break; 15917c478bd9Sstevel@tonic-gate case SVC_CONNECTION: 15927c478bd9Sstevel@tonic-gate svc_fd_xprtfree(xprt); 15937c478bd9Sstevel@tonic-gate break; 15947c478bd9Sstevel@tonic-gate case SVC_DOOR: 15957c478bd9Sstevel@tonic-gate svc_door_xprtfree(xprt); 15967c478bd9Sstevel@tonic-gate break; 15977c478bd9Sstevel@tonic-gate } 15987c478bd9Sstevel@tonic-gate } 15997c478bd9Sstevel@tonic-gate } 16007c478bd9Sstevel@tonic-gate 16017c478bd9Sstevel@tonic-gate 16027c478bd9Sstevel@tonic-gate /* 16037c478bd9Sstevel@tonic-gate * svc_copy() - make a copy of parent 16047c478bd9Sstevel@tonic-gate */ 16057c478bd9Sstevel@tonic-gate SVCXPRT * 160661961e0fSrobinson svc_copy(SVCXPRT *xprt) 16077c478bd9Sstevel@tonic-gate { 16087c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 16097c478bd9Sstevel@tonic-gate switch (svc_type(xprt)) { 16107c478bd9Sstevel@tonic-gate case SVC_DGRAM: 16117c478bd9Sstevel@tonic-gate return (svc_dg_xprtcopy(xprt)); 16127c478bd9Sstevel@tonic-gate case SVC_RENDEZVOUS: 16137c478bd9Sstevel@tonic-gate return (svc_vc_xprtcopy(xprt)); 16147c478bd9Sstevel@tonic-gate case SVC_CONNECTION: 16157c478bd9Sstevel@tonic-gate return (svc_fd_xprtcopy(xprt)); 16167c478bd9Sstevel@tonic-gate } 161761961e0fSrobinson return (NULL); 16187c478bd9Sstevel@tonic-gate } 16197c478bd9Sstevel@tonic-gate 16207c478bd9Sstevel@tonic-gate 16217c478bd9Sstevel@tonic-gate /* 16227c478bd9Sstevel@tonic-gate * _svc_destroy_private() - private SVC_DESTROY interface 16237c478bd9Sstevel@tonic-gate */ 16247c478bd9Sstevel@tonic-gate void 162561961e0fSrobinson _svc_destroy_private(SVCXPRT *xprt) 16267c478bd9Sstevel@tonic-gate { 16277c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 16287c478bd9Sstevel@tonic-gate switch (svc_type(xprt)) { 16297c478bd9Sstevel@tonic-gate case SVC_DGRAM: 16307c478bd9Sstevel@tonic-gate _svc_dg_destroy_private(xprt); 16317c478bd9Sstevel@tonic-gate break; 16327c478bd9Sstevel@tonic-gate case SVC_RENDEZVOUS: 16337c478bd9Sstevel@tonic-gate case SVC_CONNECTION: 16347c478bd9Sstevel@tonic-gate _svc_vc_destroy_private(xprt, TRUE); 16357c478bd9Sstevel@tonic-gate break; 16367c478bd9Sstevel@tonic-gate } 16377c478bd9Sstevel@tonic-gate } 16387c478bd9Sstevel@tonic-gate 16397c478bd9Sstevel@tonic-gate /* 16407c478bd9Sstevel@tonic-gate * svc_get_local_cred() - fetch local user credentials. This always 16417c478bd9Sstevel@tonic-gate * works over doors based transports. For local transports, this 16427c478bd9Sstevel@tonic-gate * does not yield correct results unless the __rpc_negotiate_uid() 16437c478bd9Sstevel@tonic-gate * call has been invoked to enable this feature. 16447c478bd9Sstevel@tonic-gate */ 16457c478bd9Sstevel@tonic-gate bool_t 164661961e0fSrobinson svc_get_local_cred(SVCXPRT *xprt, svc_local_cred_t *lcred) 16477c478bd9Sstevel@tonic-gate { 16487c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 16497c478bd9Sstevel@tonic-gate if (svc_type(xprt) == SVC_DOOR) 16507c478bd9Sstevel@tonic-gate return (__svc_get_door_cred(xprt, lcred)); 16517c478bd9Sstevel@tonic-gate return (__rpc_get_local_cred(xprt, lcred)); 16527c478bd9Sstevel@tonic-gate } 16537c478bd9Sstevel@tonic-gate 16547c478bd9Sstevel@tonic-gate 16557c478bd9Sstevel@tonic-gate /* ******************* DUPLICATE ENTRY HANDLING ROUTINES ************** */ 16567c478bd9Sstevel@tonic-gate 16577c478bd9Sstevel@tonic-gate /* 16587c478bd9Sstevel@tonic-gate * the dup cacheing routines below provide a cache of received 16597c478bd9Sstevel@tonic-gate * transactions. rpc service routines can use this to detect 16607c478bd9Sstevel@tonic-gate * retransmissions and re-send a non-failure response. Uses a 16617c478bd9Sstevel@tonic-gate * lru scheme to find entries to get rid of entries in the cache, 16627c478bd9Sstevel@tonic-gate * though only DUP_DONE entries are placed on the lru list. 16637c478bd9Sstevel@tonic-gate * the routines were written towards development of a generic 16647c478bd9Sstevel@tonic-gate * SVC_DUP() interface, which can be expanded to encompass the 16657c478bd9Sstevel@tonic-gate * svc_dg_enablecache() routines as well. the cache is currently 16667c478bd9Sstevel@tonic-gate * private to the automounter. 16677c478bd9Sstevel@tonic-gate */ 16687c478bd9Sstevel@tonic-gate 16697c478bd9Sstevel@tonic-gate 16707c478bd9Sstevel@tonic-gate /* dupcache header contains xprt specific information */ 167161961e0fSrobinson struct dupcache { 16727c478bd9Sstevel@tonic-gate rwlock_t dc_lock; 16737c478bd9Sstevel@tonic-gate time_t dc_time; 16747c478bd9Sstevel@tonic-gate int dc_buckets; 16757c478bd9Sstevel@tonic-gate int dc_maxsz; 16767c478bd9Sstevel@tonic-gate int dc_basis; 16777c478bd9Sstevel@tonic-gate struct dupreq *dc_mru; 16787c478bd9Sstevel@tonic-gate struct dupreq **dc_hashtbl; 16797c478bd9Sstevel@tonic-gate }; 16807c478bd9Sstevel@tonic-gate 16817c478bd9Sstevel@tonic-gate /* 16827c478bd9Sstevel@tonic-gate * private duplicate cache request routines 16837c478bd9Sstevel@tonic-gate */ 16847c478bd9Sstevel@tonic-gate static int __svc_dupcache_check(struct svc_req *, caddr_t *, uint_t *, 16857c478bd9Sstevel@tonic-gate struct dupcache *, uint32_t, uint32_t); 16867c478bd9Sstevel@tonic-gate static struct dupreq *__svc_dupcache_victim(struct dupcache *, time_t); 16877c478bd9Sstevel@tonic-gate static int __svc_dupcache_enter(struct svc_req *, struct dupreq *, 16887c478bd9Sstevel@tonic-gate struct dupcache *, uint32_t, uint32_t, time_t); 16897c478bd9Sstevel@tonic-gate static int __svc_dupcache_update(struct svc_req *, caddr_t, uint_t, int, 16907c478bd9Sstevel@tonic-gate struct dupcache *, uint32_t, uint32_t); 16917c478bd9Sstevel@tonic-gate #ifdef DUP_DEBUG 16927c478bd9Sstevel@tonic-gate static void __svc_dupcache_debug(struct dupcache *); 16937c478bd9Sstevel@tonic-gate #endif /* DUP_DEBUG */ 16947c478bd9Sstevel@tonic-gate 16957c478bd9Sstevel@tonic-gate /* default parameters for the dupcache */ 16967c478bd9Sstevel@tonic-gate #define DUPCACHE_BUCKETS 257 16977c478bd9Sstevel@tonic-gate #define DUPCACHE_TIME 900 16987c478bd9Sstevel@tonic-gate #define DUPCACHE_MAXSZ INT_MAX 16997c478bd9Sstevel@tonic-gate 17007c478bd9Sstevel@tonic-gate /* 17017c478bd9Sstevel@tonic-gate * __svc_dupcache_init(void *condition, int basis, char *xprt_cache) 17027c478bd9Sstevel@tonic-gate * initialize the duprequest cache and assign it to the xprt_cache 17037c478bd9Sstevel@tonic-gate * Use default values depending on the cache condition and basis. 17047c478bd9Sstevel@tonic-gate * return TRUE on success and FALSE on failure 17057c478bd9Sstevel@tonic-gate */ 17067c478bd9Sstevel@tonic-gate bool_t 17077c478bd9Sstevel@tonic-gate __svc_dupcache_init(void *condition, int basis, char **xprt_cache) 17087c478bd9Sstevel@tonic-gate { 17097c478bd9Sstevel@tonic-gate static mutex_t initdc_lock = DEFAULTMUTEX; 17107c478bd9Sstevel@tonic-gate int i; 17117c478bd9Sstevel@tonic-gate struct dupcache *dc; 17127c478bd9Sstevel@tonic-gate 171361961e0fSrobinson (void) mutex_lock(&initdc_lock); 17147c478bd9Sstevel@tonic-gate if (*xprt_cache != NULL) { /* do only once per xprt */ 171561961e0fSrobinson (void) mutex_unlock(&initdc_lock); 17167c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 17177c478bd9Sstevel@tonic-gate "__svc_dupcache_init: multiply defined dup cache"); 17187c478bd9Sstevel@tonic-gate return (FALSE); 17197c478bd9Sstevel@tonic-gate } 17207c478bd9Sstevel@tonic-gate 17217c478bd9Sstevel@tonic-gate switch (basis) { 17227c478bd9Sstevel@tonic-gate case DUPCACHE_FIXEDTIME: 172361961e0fSrobinson dc = malloc(sizeof (struct dupcache)); 17247c478bd9Sstevel@tonic-gate if (dc == NULL) { 172561961e0fSrobinson (void) mutex_unlock(&initdc_lock); 17267c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 17277c478bd9Sstevel@tonic-gate "__svc_dupcache_init: memory alloc failed"); 17287c478bd9Sstevel@tonic-gate return (FALSE); 17297c478bd9Sstevel@tonic-gate } 173061961e0fSrobinson (void) rwlock_init(&(dc->dc_lock), USYNC_THREAD, NULL); 17317c478bd9Sstevel@tonic-gate if (condition != NULL) 17327c478bd9Sstevel@tonic-gate dc->dc_time = *((time_t *)condition); 17337c478bd9Sstevel@tonic-gate else 17347c478bd9Sstevel@tonic-gate dc->dc_time = DUPCACHE_TIME; 17357c478bd9Sstevel@tonic-gate dc->dc_buckets = DUPCACHE_BUCKETS; 17367c478bd9Sstevel@tonic-gate dc->dc_maxsz = DUPCACHE_MAXSZ; 17377c478bd9Sstevel@tonic-gate dc->dc_basis = basis; 17387c478bd9Sstevel@tonic-gate dc->dc_mru = NULL; 173961961e0fSrobinson dc->dc_hashtbl = malloc(dc->dc_buckets * 17407c478bd9Sstevel@tonic-gate sizeof (struct dupreq *)); 17417c478bd9Sstevel@tonic-gate if (dc->dc_hashtbl == NULL) { 174261961e0fSrobinson free(dc); 174361961e0fSrobinson (void) mutex_unlock(&initdc_lock); 17447c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 17457c478bd9Sstevel@tonic-gate "__svc_dupcache_init: memory alloc failed"); 17467c478bd9Sstevel@tonic-gate return (FALSE); 17477c478bd9Sstevel@tonic-gate } 17487c478bd9Sstevel@tonic-gate for (i = 0; i < DUPCACHE_BUCKETS; i++) 17497c478bd9Sstevel@tonic-gate dc->dc_hashtbl[i] = NULL; 17507c478bd9Sstevel@tonic-gate *xprt_cache = (char *)dc; 17517c478bd9Sstevel@tonic-gate break; 17527c478bd9Sstevel@tonic-gate default: 175361961e0fSrobinson (void) mutex_unlock(&initdc_lock); 17547c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 17557c478bd9Sstevel@tonic-gate "__svc_dupcache_init: undefined dup cache basis"); 17567c478bd9Sstevel@tonic-gate return (FALSE); 17577c478bd9Sstevel@tonic-gate } 17587c478bd9Sstevel@tonic-gate 175961961e0fSrobinson (void) mutex_unlock(&initdc_lock); 17607c478bd9Sstevel@tonic-gate 17617c478bd9Sstevel@tonic-gate return (TRUE); 17627c478bd9Sstevel@tonic-gate } 17637c478bd9Sstevel@tonic-gate 17647c478bd9Sstevel@tonic-gate /* 17657c478bd9Sstevel@tonic-gate * __svc_dup(struct svc_req *req, caddr_t *resp_buf, uint_t *resp_bufsz, 17667c478bd9Sstevel@tonic-gate * char *xprt_cache) 17677c478bd9Sstevel@tonic-gate * searches the request cache. Creates an entry and returns DUP_NEW if 17687c478bd9Sstevel@tonic-gate * the request is not found in the cache. If it is found, then it 17697c478bd9Sstevel@tonic-gate * returns the state of the request (in progress, drop, or done) and 17707c478bd9Sstevel@tonic-gate * also allocates, and passes back results to the user (if any) in 17717c478bd9Sstevel@tonic-gate * resp_buf, and its length in resp_bufsz. DUP_ERROR is returned on error. 17727c478bd9Sstevel@tonic-gate */ 17737c478bd9Sstevel@tonic-gate int 17747c478bd9Sstevel@tonic-gate __svc_dup(struct svc_req *req, caddr_t *resp_buf, uint_t *resp_bufsz, 17757c478bd9Sstevel@tonic-gate char *xprt_cache) 17767c478bd9Sstevel@tonic-gate { 17777c478bd9Sstevel@tonic-gate uint32_t drxid, drhash; 17787c478bd9Sstevel@tonic-gate int rc; 17797c478bd9Sstevel@tonic-gate struct dupreq *dr = NULL; 17807c478bd9Sstevel@tonic-gate time_t timenow = time(NULL); 17817c478bd9Sstevel@tonic-gate 17827c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 17837c478bd9Sstevel@tonic-gate struct dupcache *dc = (struct dupcache *)xprt_cache; 17847c478bd9Sstevel@tonic-gate 17857c478bd9Sstevel@tonic-gate if (dc == NULL) { 17867c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dup: undefined cache"); 17877c478bd9Sstevel@tonic-gate return (DUP_ERROR); 17887c478bd9Sstevel@tonic-gate } 17897c478bd9Sstevel@tonic-gate 17907c478bd9Sstevel@tonic-gate /* get the xid of the request */ 17917c478bd9Sstevel@tonic-gate if (SVC_CONTROL(req->rq_xprt, SVCGET_XID, (void*)&drxid) == FALSE) { 17927c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dup: xid error"); 17937c478bd9Sstevel@tonic-gate return (DUP_ERROR); 17947c478bd9Sstevel@tonic-gate } 17957c478bd9Sstevel@tonic-gate drhash = drxid % dc->dc_buckets; 17967c478bd9Sstevel@tonic-gate 17977c478bd9Sstevel@tonic-gate if ((rc = __svc_dupcache_check(req, resp_buf, resp_bufsz, dc, drxid, 17987c478bd9Sstevel@tonic-gate drhash)) != DUP_NEW) 17997c478bd9Sstevel@tonic-gate return (rc); 18007c478bd9Sstevel@tonic-gate 18017c478bd9Sstevel@tonic-gate if ((dr = __svc_dupcache_victim(dc, timenow)) == NULL) 18027c478bd9Sstevel@tonic-gate return (DUP_ERROR); 18037c478bd9Sstevel@tonic-gate 18047c478bd9Sstevel@tonic-gate if ((rc = __svc_dupcache_enter(req, dr, dc, drxid, drhash, timenow)) 18057c478bd9Sstevel@tonic-gate == DUP_ERROR) 18067c478bd9Sstevel@tonic-gate return (rc); 18077c478bd9Sstevel@tonic-gate 18087c478bd9Sstevel@tonic-gate return (DUP_NEW); 18097c478bd9Sstevel@tonic-gate } 18107c478bd9Sstevel@tonic-gate 18117c478bd9Sstevel@tonic-gate 18127c478bd9Sstevel@tonic-gate 18137c478bd9Sstevel@tonic-gate /* 18147c478bd9Sstevel@tonic-gate * __svc_dupcache_check(struct svc_req *req, caddr_t *resp_buf, 18157c478bd9Sstevel@tonic-gate * uint_t *resp_bufsz,truct dupcache *dc, uint32_t drxid, 18167c478bd9Sstevel@tonic-gate * uint32_t drhash) 18177c478bd9Sstevel@tonic-gate * Checks to see whether an entry already exists in the cache. If it does 18187c478bd9Sstevel@tonic-gate * copy back into the resp_buf, if appropriate. Return the status of 18197c478bd9Sstevel@tonic-gate * the request, or DUP_NEW if the entry is not in the cache 18207c478bd9Sstevel@tonic-gate */ 18217c478bd9Sstevel@tonic-gate static int 18227c478bd9Sstevel@tonic-gate __svc_dupcache_check(struct svc_req *req, caddr_t *resp_buf, uint_t *resp_bufsz, 18237c478bd9Sstevel@tonic-gate struct dupcache *dc, uint32_t drxid, uint32_t drhash) 18247c478bd9Sstevel@tonic-gate { 18257c478bd9Sstevel@tonic-gate struct dupreq *dr = NULL; 18267c478bd9Sstevel@tonic-gate 182761961e0fSrobinson (void) rw_rdlock(&(dc->dc_lock)); 18287c478bd9Sstevel@tonic-gate dr = dc->dc_hashtbl[drhash]; 18297c478bd9Sstevel@tonic-gate while (dr != NULL) { 18307c478bd9Sstevel@tonic-gate if (dr->dr_xid == drxid && 18317c478bd9Sstevel@tonic-gate dr->dr_proc == req->rq_proc && 18327c478bd9Sstevel@tonic-gate dr->dr_prog == req->rq_prog && 18337c478bd9Sstevel@tonic-gate dr->dr_vers == req->rq_vers && 18347c478bd9Sstevel@tonic-gate dr->dr_addr.len == req->rq_xprt->xp_rtaddr.len && 183561961e0fSrobinson memcmp(dr->dr_addr.buf, 183661961e0fSrobinson req->rq_xprt->xp_rtaddr.buf, 18377c478bd9Sstevel@tonic-gate dr->dr_addr.len) == 0) { /* entry found */ 18387c478bd9Sstevel@tonic-gate if (dr->dr_hash != drhash) { 18397c478bd9Sstevel@tonic-gate /* sanity check */ 184061961e0fSrobinson (void) rw_unlock((&dc->dc_lock)); 18417c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 18427c478bd9Sstevel@tonic-gate "\n__svc_dupdone: hashing error"); 18437c478bd9Sstevel@tonic-gate return (DUP_ERROR); 18447c478bd9Sstevel@tonic-gate } 18457c478bd9Sstevel@tonic-gate 18467c478bd9Sstevel@tonic-gate /* 18477c478bd9Sstevel@tonic-gate * return results for requests on lru list, if 18487c478bd9Sstevel@tonic-gate * appropriate requests must be DUP_DROP or DUP_DONE 18497c478bd9Sstevel@tonic-gate * to have a result. A NULL buffer in the cache 18507c478bd9Sstevel@tonic-gate * implies no results were sent during dupdone. 18517c478bd9Sstevel@tonic-gate * A NULL buffer in the call implies not interested 18527c478bd9Sstevel@tonic-gate * in results. 18537c478bd9Sstevel@tonic-gate */ 18547c478bd9Sstevel@tonic-gate if (((dr->dr_status == DUP_DONE) || 18557c478bd9Sstevel@tonic-gate (dr->dr_status == DUP_DROP)) && 18567c478bd9Sstevel@tonic-gate resp_buf != NULL && 18577c478bd9Sstevel@tonic-gate dr->dr_resp.buf != NULL) { 185861961e0fSrobinson *resp_buf = malloc(dr->dr_resp.len); 18597c478bd9Sstevel@tonic-gate if (*resp_buf == NULL) { 18607c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 18617c478bd9Sstevel@tonic-gate "__svc_dupcache_check: malloc failed"); 186261961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 18637c478bd9Sstevel@tonic-gate return (DUP_ERROR); 18647c478bd9Sstevel@tonic-gate } 186561961e0fSrobinson (void) memset(*resp_buf, 0, dr->dr_resp.len); 186661961e0fSrobinson (void) memcpy(*resp_buf, dr->dr_resp.buf, 18677c478bd9Sstevel@tonic-gate dr->dr_resp.len); 18687c478bd9Sstevel@tonic-gate *resp_bufsz = dr->dr_resp.len; 18697c478bd9Sstevel@tonic-gate } else { 18707c478bd9Sstevel@tonic-gate /* no result */ 18717c478bd9Sstevel@tonic-gate if (resp_buf) 18727c478bd9Sstevel@tonic-gate *resp_buf = NULL; 18737c478bd9Sstevel@tonic-gate if (resp_bufsz) 18747c478bd9Sstevel@tonic-gate *resp_bufsz = 0; 18757c478bd9Sstevel@tonic-gate } 187661961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 18777c478bd9Sstevel@tonic-gate return (dr->dr_status); 18787c478bd9Sstevel@tonic-gate } 18797c478bd9Sstevel@tonic-gate dr = dr->dr_chain; 18807c478bd9Sstevel@tonic-gate } 188161961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 18827c478bd9Sstevel@tonic-gate return (DUP_NEW); 18837c478bd9Sstevel@tonic-gate } 18847c478bd9Sstevel@tonic-gate 18857c478bd9Sstevel@tonic-gate /* 18867c478bd9Sstevel@tonic-gate * __svc_dupcache_victim(struct dupcache *dc, time_t timenow) 18877c478bd9Sstevel@tonic-gate * Return a victim dupreq entry to the caller, depending on cache policy. 18887c478bd9Sstevel@tonic-gate */ 18897c478bd9Sstevel@tonic-gate static struct dupreq * 18907c478bd9Sstevel@tonic-gate __svc_dupcache_victim(struct dupcache *dc, time_t timenow) 18917c478bd9Sstevel@tonic-gate { 18927c478bd9Sstevel@tonic-gate struct dupreq *dr = NULL; 18937c478bd9Sstevel@tonic-gate 18947c478bd9Sstevel@tonic-gate switch (dc->dc_basis) { 18957c478bd9Sstevel@tonic-gate case DUPCACHE_FIXEDTIME: 18967c478bd9Sstevel@tonic-gate /* 18977c478bd9Sstevel@tonic-gate * The hash policy is to free up a bit of the hash 18987c478bd9Sstevel@tonic-gate * table before allocating a new entry as the victim. 18997c478bd9Sstevel@tonic-gate * Freeing up the hash table each time should split 19007c478bd9Sstevel@tonic-gate * the cost of keeping the hash table clean among threads. 19017c478bd9Sstevel@tonic-gate * Note that only DONE or DROPPED entries are on the lru 19027c478bd9Sstevel@tonic-gate * list but we do a sanity check anyway. 19037c478bd9Sstevel@tonic-gate */ 190461961e0fSrobinson (void) rw_wrlock(&(dc->dc_lock)); 19057c478bd9Sstevel@tonic-gate while ((dc->dc_mru) && (dr = dc->dc_mru->dr_next) && 19067c478bd9Sstevel@tonic-gate ((timenow - dr->dr_time) > dc->dc_time)) { 19077c478bd9Sstevel@tonic-gate /* clean and then free the entry */ 19087c478bd9Sstevel@tonic-gate if (dr->dr_status != DUP_DONE && 19097c478bd9Sstevel@tonic-gate dr->dr_status != DUP_DROP) { 19107c478bd9Sstevel@tonic-gate /* 19117c478bd9Sstevel@tonic-gate * The LRU list can't contain an 19127c478bd9Sstevel@tonic-gate * entry where the status is other than 19137c478bd9Sstevel@tonic-gate * DUP_DONE or DUP_DROP. 19147c478bd9Sstevel@tonic-gate */ 19157c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 19167c478bd9Sstevel@tonic-gate "__svc_dupcache_victim: bad victim"); 19177c478bd9Sstevel@tonic-gate #ifdef DUP_DEBUG 19187c478bd9Sstevel@tonic-gate /* 19197c478bd9Sstevel@tonic-gate * Need to hold the reader/writers lock to 19207c478bd9Sstevel@tonic-gate * print the cache info, since we already 19217c478bd9Sstevel@tonic-gate * hold the writers lock, we shall continue 19227c478bd9Sstevel@tonic-gate * calling __svc_dupcache_debug() 19237c478bd9Sstevel@tonic-gate */ 19247c478bd9Sstevel@tonic-gate __svc_dupcache_debug(dc); 19257c478bd9Sstevel@tonic-gate #endif /* DUP_DEBUG */ 192661961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 19277c478bd9Sstevel@tonic-gate return (NULL); 19287c478bd9Sstevel@tonic-gate } 19297c478bd9Sstevel@tonic-gate /* free buffers */ 19307c478bd9Sstevel@tonic-gate if (dr->dr_resp.buf) { 193161961e0fSrobinson free(dr->dr_resp.buf); 19327c478bd9Sstevel@tonic-gate dr->dr_resp.buf = NULL; 19337c478bd9Sstevel@tonic-gate } 19347c478bd9Sstevel@tonic-gate if (dr->dr_addr.buf) { 193561961e0fSrobinson free(dr->dr_addr.buf); 19367c478bd9Sstevel@tonic-gate dr->dr_addr.buf = NULL; 19377c478bd9Sstevel@tonic-gate } 19387c478bd9Sstevel@tonic-gate 19397c478bd9Sstevel@tonic-gate /* unhash the entry */ 19407c478bd9Sstevel@tonic-gate if (dr->dr_chain) 19417c478bd9Sstevel@tonic-gate dr->dr_chain->dr_prevchain = dr->dr_prevchain; 19427c478bd9Sstevel@tonic-gate if (dr->dr_prevchain) 19437c478bd9Sstevel@tonic-gate dr->dr_prevchain->dr_chain = dr->dr_chain; 19447c478bd9Sstevel@tonic-gate if (dc->dc_hashtbl[dr->dr_hash] == dr) 19457c478bd9Sstevel@tonic-gate dc->dc_hashtbl[dr->dr_hash] = dr->dr_chain; 19467c478bd9Sstevel@tonic-gate 19477c478bd9Sstevel@tonic-gate /* modify the lru pointers */ 194861961e0fSrobinson if (dc->dc_mru == dr) { 19497c478bd9Sstevel@tonic-gate dc->dc_mru = NULL; 195061961e0fSrobinson } else { 19517c478bd9Sstevel@tonic-gate dc->dc_mru->dr_next = dr->dr_next; 19527c478bd9Sstevel@tonic-gate dr->dr_next->dr_prev = dc->dc_mru; 19537c478bd9Sstevel@tonic-gate } 195461961e0fSrobinson free(dr); 19557c478bd9Sstevel@tonic-gate dr = NULL; 19567c478bd9Sstevel@tonic-gate } 195761961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 19587c478bd9Sstevel@tonic-gate 19597c478bd9Sstevel@tonic-gate /* 19607c478bd9Sstevel@tonic-gate * Allocate and return new clean entry as victim 19617c478bd9Sstevel@tonic-gate */ 196261961e0fSrobinson if ((dr = malloc(sizeof (*dr))) == NULL) { 19637c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 196461961e0fSrobinson "__svc_dupcache_victim: malloc failed"); 19657c478bd9Sstevel@tonic-gate return (NULL); 19667c478bd9Sstevel@tonic-gate } 196761961e0fSrobinson (void) memset(dr, 0, sizeof (*dr)); 19687c478bd9Sstevel@tonic-gate return (dr); 19697c478bd9Sstevel@tonic-gate default: 19707c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 19717c478bd9Sstevel@tonic-gate "__svc_dupcache_victim: undefined dup cache_basis"); 19727c478bd9Sstevel@tonic-gate return (NULL); 19737c478bd9Sstevel@tonic-gate } 19747c478bd9Sstevel@tonic-gate } 19757c478bd9Sstevel@tonic-gate 19767c478bd9Sstevel@tonic-gate /* 19777c478bd9Sstevel@tonic-gate * __svc_dupcache_enter(struct svc_req *req, struct dupreq *dr, 19787c478bd9Sstevel@tonic-gate * struct dupcache *dc, uint32_t drxid, uint32_t drhash, time_t timenow) 19797c478bd9Sstevel@tonic-gate * build new duprequest entry and then insert into the cache 19807c478bd9Sstevel@tonic-gate */ 19817c478bd9Sstevel@tonic-gate static int 19827c478bd9Sstevel@tonic-gate __svc_dupcache_enter(struct svc_req *req, struct dupreq *dr, 19837c478bd9Sstevel@tonic-gate struct dupcache *dc, uint32_t drxid, uint32_t drhash, time_t timenow) 19847c478bd9Sstevel@tonic-gate { 19857c478bd9Sstevel@tonic-gate dr->dr_xid = drxid; 19867c478bd9Sstevel@tonic-gate dr->dr_prog = req->rq_prog; 19877c478bd9Sstevel@tonic-gate dr->dr_vers = req->rq_vers; 19887c478bd9Sstevel@tonic-gate dr->dr_proc = req->rq_proc; 19897c478bd9Sstevel@tonic-gate dr->dr_addr.maxlen = req->rq_xprt->xp_rtaddr.len; 19907c478bd9Sstevel@tonic-gate dr->dr_addr.len = dr->dr_addr.maxlen; 199161961e0fSrobinson if ((dr->dr_addr.buf = malloc(dr->dr_addr.maxlen)) == NULL) { 199261961e0fSrobinson syslog(LOG_ERR, "__svc_dupcache_enter: malloc failed"); 199361961e0fSrobinson free(dr); 19947c478bd9Sstevel@tonic-gate return (DUP_ERROR); 19957c478bd9Sstevel@tonic-gate } 199661961e0fSrobinson (void) memset(dr->dr_addr.buf, 0, dr->dr_addr.len); 199761961e0fSrobinson (void) memcpy(dr->dr_addr.buf, req->rq_xprt->xp_rtaddr.buf, 199861961e0fSrobinson dr->dr_addr.len); 19997c478bd9Sstevel@tonic-gate dr->dr_resp.buf = NULL; 20007c478bd9Sstevel@tonic-gate dr->dr_resp.maxlen = 0; 20017c478bd9Sstevel@tonic-gate dr->dr_resp.len = 0; 20027c478bd9Sstevel@tonic-gate dr->dr_status = DUP_INPROGRESS; 20037c478bd9Sstevel@tonic-gate dr->dr_time = timenow; 20047c478bd9Sstevel@tonic-gate dr->dr_hash = drhash; /* needed for efficient victim cleanup */ 20057c478bd9Sstevel@tonic-gate 20067c478bd9Sstevel@tonic-gate /* place entry at head of hash table */ 200761961e0fSrobinson (void) rw_wrlock(&(dc->dc_lock)); 20087c478bd9Sstevel@tonic-gate dr->dr_chain = dc->dc_hashtbl[drhash]; 20097c478bd9Sstevel@tonic-gate dr->dr_prevchain = NULL; 20107c478bd9Sstevel@tonic-gate if (dc->dc_hashtbl[drhash] != NULL) 20117c478bd9Sstevel@tonic-gate dc->dc_hashtbl[drhash]->dr_prevchain = dr; 20127c478bd9Sstevel@tonic-gate dc->dc_hashtbl[drhash] = dr; 201361961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 20147c478bd9Sstevel@tonic-gate return (DUP_NEW); 20157c478bd9Sstevel@tonic-gate } 20167c478bd9Sstevel@tonic-gate 20177c478bd9Sstevel@tonic-gate /* 20187c478bd9Sstevel@tonic-gate * __svc_dupdone(struct svc_req *req, caddr_t resp_buf, uint_t resp_bufsz, 20197c478bd9Sstevel@tonic-gate * int status, char *xprt_cache) 20207c478bd9Sstevel@tonic-gate * Marks the request done (DUP_DONE or DUP_DROP) and stores the response. 20217c478bd9Sstevel@tonic-gate * Only DONE and DROP requests can be marked as done. Sets the lru pointers 20227c478bd9Sstevel@tonic-gate * to make the entry the most recently used. Returns DUP_ERROR or status. 20237c478bd9Sstevel@tonic-gate */ 20247c478bd9Sstevel@tonic-gate int 20257c478bd9Sstevel@tonic-gate __svc_dupdone(struct svc_req *req, caddr_t resp_buf, uint_t resp_bufsz, 20267c478bd9Sstevel@tonic-gate int status, char *xprt_cache) 20277c478bd9Sstevel@tonic-gate { 20287c478bd9Sstevel@tonic-gate uint32_t drxid, drhash; 20297c478bd9Sstevel@tonic-gate int rc; 20307c478bd9Sstevel@tonic-gate 20317c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 20327c478bd9Sstevel@tonic-gate struct dupcache *dc = (struct dupcache *)xprt_cache; 20337c478bd9Sstevel@tonic-gate 20347c478bd9Sstevel@tonic-gate if (dc == NULL) { 20357c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dupdone: undefined cache"); 20367c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20377c478bd9Sstevel@tonic-gate } 20387c478bd9Sstevel@tonic-gate 20397c478bd9Sstevel@tonic-gate if (status != DUP_DONE && status != DUP_DROP) { 20407c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dupdone: invalid dupdone status"); 20417c478bd9Sstevel@tonic-gate syslog(LOG_ERR, " must be DUP_DONE or DUP_DROP"); 20427c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20437c478bd9Sstevel@tonic-gate } 20447c478bd9Sstevel@tonic-gate 20457c478bd9Sstevel@tonic-gate /* find the xid of the entry in the cache */ 20467c478bd9Sstevel@tonic-gate if (SVC_CONTROL(req->rq_xprt, SVCGET_XID, (void*)&drxid) == FALSE) { 20477c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dup: xid error"); 20487c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20497c478bd9Sstevel@tonic-gate } 20507c478bd9Sstevel@tonic-gate drhash = drxid % dc->dc_buckets; 20517c478bd9Sstevel@tonic-gate 20527c478bd9Sstevel@tonic-gate /* update the status of the entry and result buffers, if required */ 20537c478bd9Sstevel@tonic-gate if ((rc = __svc_dupcache_update(req, resp_buf, resp_bufsz, status, 20547c478bd9Sstevel@tonic-gate dc, drxid, drhash)) == DUP_ERROR) { 20557c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dupdone: cache entry error"); 20567c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20577c478bd9Sstevel@tonic-gate } 20587c478bd9Sstevel@tonic-gate 20597c478bd9Sstevel@tonic-gate return (rc); 20607c478bd9Sstevel@tonic-gate } 20617c478bd9Sstevel@tonic-gate 20627c478bd9Sstevel@tonic-gate /* 20637c478bd9Sstevel@tonic-gate * __svc_dupcache_update(struct svc_req *req, caddr_t resp_buf, 20647c478bd9Sstevel@tonic-gate * uint_t resp_bufsz, int status, struct dupcache *dc, uint32_t drxid, 20657c478bd9Sstevel@tonic-gate * uint32_t drhash) 20667c478bd9Sstevel@tonic-gate * Check if entry exists in the dupcacache. If it does, update its status 20677c478bd9Sstevel@tonic-gate * and time and also its buffer, if appropriate. Its possible, but unlikely 20687c478bd9Sstevel@tonic-gate * for DONE requests to not exist in the cache. Return DUP_ERROR or status. 20697c478bd9Sstevel@tonic-gate */ 20707c478bd9Sstevel@tonic-gate static int 20717c478bd9Sstevel@tonic-gate __svc_dupcache_update(struct svc_req *req, caddr_t resp_buf, uint_t resp_bufsz, 20727c478bd9Sstevel@tonic-gate int status, struct dupcache *dc, uint32_t drxid, uint32_t drhash) 20737c478bd9Sstevel@tonic-gate { 20747c478bd9Sstevel@tonic-gate struct dupreq *dr = NULL; 20757c478bd9Sstevel@tonic-gate time_t timenow = time(NULL); 20767c478bd9Sstevel@tonic-gate 207761961e0fSrobinson (void) rw_wrlock(&(dc->dc_lock)); 20787c478bd9Sstevel@tonic-gate dr = dc->dc_hashtbl[drhash]; 20797c478bd9Sstevel@tonic-gate while (dr != NULL) { 20807c478bd9Sstevel@tonic-gate if (dr->dr_xid == drxid && 20817c478bd9Sstevel@tonic-gate dr->dr_proc == req->rq_proc && 20827c478bd9Sstevel@tonic-gate dr->dr_prog == req->rq_prog && 20837c478bd9Sstevel@tonic-gate dr->dr_vers == req->rq_vers && 20847c478bd9Sstevel@tonic-gate dr->dr_addr.len == req->rq_xprt->xp_rtaddr.len && 208561961e0fSrobinson memcmp(dr->dr_addr.buf, 208661961e0fSrobinson req->rq_xprt->xp_rtaddr.buf, 20877c478bd9Sstevel@tonic-gate dr->dr_addr.len) == 0) { /* entry found */ 20887c478bd9Sstevel@tonic-gate if (dr->dr_hash != drhash) { 20897c478bd9Sstevel@tonic-gate /* sanity check */ 209061961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 20917c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 20927c478bd9Sstevel@tonic-gate "\n__svc_dupdone: hashing error"); 20937c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20947c478bd9Sstevel@tonic-gate } 20957c478bd9Sstevel@tonic-gate 20967c478bd9Sstevel@tonic-gate /* store the results if bufer is not NULL */ 20977c478bd9Sstevel@tonic-gate if (resp_buf != NULL) { 209861961e0fSrobinson if ((dr->dr_resp.buf = 209961961e0fSrobinson malloc(resp_bufsz)) == NULL) { 210061961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 21017c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 210261961e0fSrobinson "__svc_dupdone: malloc failed"); 21037c478bd9Sstevel@tonic-gate return (DUP_ERROR); 21047c478bd9Sstevel@tonic-gate } 210561961e0fSrobinson (void) memset(dr->dr_resp.buf, 0, resp_bufsz); 210661961e0fSrobinson (void) memcpy(dr->dr_resp.buf, resp_buf, 21077c478bd9Sstevel@tonic-gate (uint_t)resp_bufsz); 21087c478bd9Sstevel@tonic-gate dr->dr_resp.len = resp_bufsz; 21097c478bd9Sstevel@tonic-gate } 21107c478bd9Sstevel@tonic-gate 21117c478bd9Sstevel@tonic-gate /* update status and done time */ 21127c478bd9Sstevel@tonic-gate dr->dr_status = status; 21137c478bd9Sstevel@tonic-gate dr->dr_time = timenow; 21147c478bd9Sstevel@tonic-gate 21157c478bd9Sstevel@tonic-gate /* move the entry to the mru position */ 21167c478bd9Sstevel@tonic-gate if (dc->dc_mru == NULL) { 21177c478bd9Sstevel@tonic-gate dr->dr_next = dr; 21187c478bd9Sstevel@tonic-gate dr->dr_prev = dr; 21197c478bd9Sstevel@tonic-gate } else { 21207c478bd9Sstevel@tonic-gate dr->dr_next = dc->dc_mru->dr_next; 21217c478bd9Sstevel@tonic-gate dc->dc_mru->dr_next->dr_prev = dr; 21227c478bd9Sstevel@tonic-gate dr->dr_prev = dc->dc_mru; 21237c478bd9Sstevel@tonic-gate dc->dc_mru->dr_next = dr; 21247c478bd9Sstevel@tonic-gate } 21257c478bd9Sstevel@tonic-gate dc->dc_mru = dr; 21267c478bd9Sstevel@tonic-gate 212761961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 21287c478bd9Sstevel@tonic-gate return (status); 21297c478bd9Sstevel@tonic-gate } 21307c478bd9Sstevel@tonic-gate dr = dr->dr_chain; 21317c478bd9Sstevel@tonic-gate } 213261961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 21337c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dupdone: entry not in dup cache"); 21347c478bd9Sstevel@tonic-gate return (DUP_ERROR); 21357c478bd9Sstevel@tonic-gate } 21367c478bd9Sstevel@tonic-gate 21377c478bd9Sstevel@tonic-gate #ifdef DUP_DEBUG 21387c478bd9Sstevel@tonic-gate /* 21397c478bd9Sstevel@tonic-gate * __svc_dupcache_debug(struct dupcache *dc) 21407c478bd9Sstevel@tonic-gate * print out the hash table stuff 21417c478bd9Sstevel@tonic-gate * 21427c478bd9Sstevel@tonic-gate * This function requires the caller to hold the reader 21437c478bd9Sstevel@tonic-gate * or writer version of the duplicate request cache lock (dc_lock). 21447c478bd9Sstevel@tonic-gate */ 21457c478bd9Sstevel@tonic-gate static void 21467c478bd9Sstevel@tonic-gate __svc_dupcache_debug(struct dupcache *dc) 21477c478bd9Sstevel@tonic-gate { 21487c478bd9Sstevel@tonic-gate struct dupreq *dr = NULL; 21497c478bd9Sstevel@tonic-gate int i; 21507c478bd9Sstevel@tonic-gate bool_t bval; 21517c478bd9Sstevel@tonic-gate 21527c478bd9Sstevel@tonic-gate fprintf(stderr, " HASHTABLE\n"); 21537c478bd9Sstevel@tonic-gate for (i = 0; i < dc->dc_buckets; i++) { 21547c478bd9Sstevel@tonic-gate bval = FALSE; 21557c478bd9Sstevel@tonic-gate dr = dc->dc_hashtbl[i]; 21567c478bd9Sstevel@tonic-gate while (dr != NULL) { 21577c478bd9Sstevel@tonic-gate if (!bval) { /* ensures bucket printed only once */ 21587c478bd9Sstevel@tonic-gate fprintf(stderr, " bucket : %d\n", i); 21597c478bd9Sstevel@tonic-gate bval = TRUE; 21607c478bd9Sstevel@tonic-gate } 21617c478bd9Sstevel@tonic-gate fprintf(stderr, "\txid: %u status: %d time: %ld", 21627c478bd9Sstevel@tonic-gate dr->dr_xid, dr->dr_status, dr->dr_time); 21637c478bd9Sstevel@tonic-gate fprintf(stderr, " dr: %x chain: %x prevchain: %x\n", 21647c478bd9Sstevel@tonic-gate dr, dr->dr_chain, dr->dr_prevchain); 21657c478bd9Sstevel@tonic-gate dr = dr->dr_chain; 21667c478bd9Sstevel@tonic-gate } 21677c478bd9Sstevel@tonic-gate } 21687c478bd9Sstevel@tonic-gate 21697c478bd9Sstevel@tonic-gate fprintf(stderr, " LRU\n"); 21707c478bd9Sstevel@tonic-gate if (dc->dc_mru) { 21717c478bd9Sstevel@tonic-gate dr = dc->dc_mru->dr_next; /* lru */ 21727c478bd9Sstevel@tonic-gate while (dr != dc->dc_mru) { 21737c478bd9Sstevel@tonic-gate fprintf(stderr, "\txid: %u status : %d time : %ld", 21747c478bd9Sstevel@tonic-gate dr->dr_xid, dr->dr_status, dr->dr_time); 21757c478bd9Sstevel@tonic-gate fprintf(stderr, " dr: %x next: %x prev: %x\n", 21767c478bd9Sstevel@tonic-gate dr, dr->dr_next, dr->dr_prev); 21777c478bd9Sstevel@tonic-gate dr = dr->dr_next; 21787c478bd9Sstevel@tonic-gate } 21797c478bd9Sstevel@tonic-gate fprintf(stderr, "\txid: %u status: %d time: %ld", 21807c478bd9Sstevel@tonic-gate dr->dr_xid, dr->dr_status, dr->dr_time); 21817c478bd9Sstevel@tonic-gate fprintf(stderr, " dr: %x next: %x prev: %x\n", dr, 21827c478bd9Sstevel@tonic-gate dr->dr_next, dr->dr_prev); 21837c478bd9Sstevel@tonic-gate } 21847c478bd9Sstevel@tonic-gate } 21857c478bd9Sstevel@tonic-gate #endif /* DUP_DEBUG */ 2186