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 59acbbeafSnn35248 * Common Development and Distribution License (the "License"). 69acbbeafSnn35248 * 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*9ff75adeSSurya Prakki * Copyright (c) 1989, 2010, Oracle and/or its affiliates. All rights reserved. 247c478bd9Sstevel@tonic-gate */ 257c478bd9Sstevel@tonic-gate /* 267c478bd9Sstevel@tonic-gate * Copyright 1993 OpenVision Technologies, Inc., All Rights Reserved. 277c478bd9Sstevel@tonic-gate */ 287c478bd9Sstevel@tonic-gate /* Copyright (c) 1983, 1984, 1985, 1986, 1987, 1988, 1989 AT&T */ 297c478bd9Sstevel@tonic-gate /* All Rights Reserved */ 307c478bd9Sstevel@tonic-gate /* 317c478bd9Sstevel@tonic-gate * Portions of this source code were derived from Berkeley 327c478bd9Sstevel@tonic-gate * 4.3 BSD under license from the Regents of the University of 337c478bd9Sstevel@tonic-gate * California. 347c478bd9Sstevel@tonic-gate */ 357c478bd9Sstevel@tonic-gate 367c478bd9Sstevel@tonic-gate /* 377c478bd9Sstevel@tonic-gate * svc.c, Server-side remote procedure call interface. 387c478bd9Sstevel@tonic-gate * 397c478bd9Sstevel@tonic-gate * There are two sets of procedures here. The xprt routines are 407c478bd9Sstevel@tonic-gate * for handling transport handles. The svc routines handle the 417c478bd9Sstevel@tonic-gate * list of service routines. 427c478bd9Sstevel@tonic-gate * 437c478bd9Sstevel@tonic-gate */ 447c478bd9Sstevel@tonic-gate 457c478bd9Sstevel@tonic-gate 467c478bd9Sstevel@tonic-gate #include "mt.h" 477c478bd9Sstevel@tonic-gate #include "rpc_mt.h" 487c478bd9Sstevel@tonic-gate #include <assert.h> 497c478bd9Sstevel@tonic-gate #include <errno.h> 507c478bd9Sstevel@tonic-gate #include <sys/types.h> 517c478bd9Sstevel@tonic-gate #include <stropts.h> 527c478bd9Sstevel@tonic-gate #include <sys/conf.h> 537c478bd9Sstevel@tonic-gate #include <rpc/rpc.h> 547c478bd9Sstevel@tonic-gate #ifdef PORTMAP 557c478bd9Sstevel@tonic-gate #include <rpc/pmap_clnt.h> 567c478bd9Sstevel@tonic-gate #endif 577c478bd9Sstevel@tonic-gate #include <sys/poll.h> 587c478bd9Sstevel@tonic-gate #include <netconfig.h> 597c478bd9Sstevel@tonic-gate #include <syslog.h> 607c478bd9Sstevel@tonic-gate #include <stdlib.h> 617c478bd9Sstevel@tonic-gate #include <unistd.h> 627c478bd9Sstevel@tonic-gate #include <string.h> 637c478bd9Sstevel@tonic-gate #include <limits.h> 647c478bd9Sstevel@tonic-gate 657c478bd9Sstevel@tonic-gate extern bool_t __svc_get_door_cred(); 667c478bd9Sstevel@tonic-gate extern bool_t __rpc_get_local_cred(); 677c478bd9Sstevel@tonic-gate 687c478bd9Sstevel@tonic-gate SVCXPRT **svc_xports; 697c478bd9Sstevel@tonic-gate static int nsvc_xports; /* total number of svc_xports allocated */ 707c478bd9Sstevel@tonic-gate 717c478bd9Sstevel@tonic-gate XDR **svc_xdrs; /* common XDR receive area */ 727c478bd9Sstevel@tonic-gate int nsvc_xdrs; /* total number of svc_xdrs allocated */ 737c478bd9Sstevel@tonic-gate 747c478bd9Sstevel@tonic-gate int __rpc_use_pollfd_done; /* to unlimit the number of connections */ 757c478bd9Sstevel@tonic-gate 767c478bd9Sstevel@tonic-gate #define NULL_SVC ((struct svc_callout *)0) 777c478bd9Sstevel@tonic-gate #define RQCRED_SIZE 400 /* this size is excessive */ 787c478bd9Sstevel@tonic-gate 797c478bd9Sstevel@tonic-gate /* 807c478bd9Sstevel@tonic-gate * The services list 817c478bd9Sstevel@tonic-gate * Each entry represents a set of procedures (an rpc program). 827c478bd9Sstevel@tonic-gate * The dispatch routine takes request structs and runs the 837c478bd9Sstevel@tonic-gate * appropriate procedure. 847c478bd9Sstevel@tonic-gate */ 857c478bd9Sstevel@tonic-gate static struct svc_callout { 867c478bd9Sstevel@tonic-gate struct svc_callout *sc_next; 877c478bd9Sstevel@tonic-gate rpcprog_t sc_prog; 887c478bd9Sstevel@tonic-gate rpcvers_t sc_vers; 897c478bd9Sstevel@tonic-gate char *sc_netid; 907c478bd9Sstevel@tonic-gate void (*sc_dispatch)(); 917c478bd9Sstevel@tonic-gate } *svc_head; 927c478bd9Sstevel@tonic-gate extern rwlock_t svc_lock; 937c478bd9Sstevel@tonic-gate 947c478bd9Sstevel@tonic-gate static struct svc_callout *svc_find(); 957c478bd9Sstevel@tonic-gate int _svc_prog_dispatch(); 967c478bd9Sstevel@tonic-gate void svc_getreq_common(); 977c478bd9Sstevel@tonic-gate char *strdup(); 987c478bd9Sstevel@tonic-gate 997c478bd9Sstevel@tonic-gate extern mutex_t svc_door_mutex; 1007c478bd9Sstevel@tonic-gate extern cond_t svc_door_waitcv; 1017c478bd9Sstevel@tonic-gate extern int svc_ndoorfds; 1027c478bd9Sstevel@tonic-gate extern SVCXPRT_LIST *_svc_xprtlist; 1037c478bd9Sstevel@tonic-gate extern mutex_t xprtlist_lock; 1047c478bd9Sstevel@tonic-gate extern void __svc_rm_from_xlist(); 1057c478bd9Sstevel@tonic-gate 1067c478bd9Sstevel@tonic-gate extern fd_set _new_svc_fdset; 1077c478bd9Sstevel@tonic-gate 1087c478bd9Sstevel@tonic-gate /* 1097c478bd9Sstevel@tonic-gate * If the allocated array of reactor is too small, this value is used as a 1107c478bd9Sstevel@tonic-gate * margin. This reduces the number of allocations. 1117c478bd9Sstevel@tonic-gate */ 1127c478bd9Sstevel@tonic-gate #define USER_FD_INCREMENT 5 1137c478bd9Sstevel@tonic-gate 1147c478bd9Sstevel@tonic-gate static void add_pollfd(int fd, short events); 1157c478bd9Sstevel@tonic-gate static void remove_pollfd(int fd); 1167c478bd9Sstevel@tonic-gate static void __svc_remove_input_of_fd(int fd); 1177c478bd9Sstevel@tonic-gate 1187c478bd9Sstevel@tonic-gate 1197c478bd9Sstevel@tonic-gate /* 1207c478bd9Sstevel@tonic-gate * Data used to handle reactor: 1217c478bd9Sstevel@tonic-gate * - one file descriptor we listen to, 1227c478bd9Sstevel@tonic-gate * - one callback we call if the fd pops, 1237c478bd9Sstevel@tonic-gate * - and a cookie passed as a parameter to the callback. 1247c478bd9Sstevel@tonic-gate * 1257c478bd9Sstevel@tonic-gate * The structure is an array indexed on the file descriptor. Each entry is 1267c478bd9Sstevel@tonic-gate * pointing to the first element of a double-linked list of callback. 1277c478bd9Sstevel@tonic-gate * only one callback may be associated to a couple (fd, event). 1287c478bd9Sstevel@tonic-gate */ 1297c478bd9Sstevel@tonic-gate 1307c478bd9Sstevel@tonic-gate struct _svc_user_fd_head; 1317c478bd9Sstevel@tonic-gate 13261961e0fSrobinson typedef struct { 1337c478bd9Sstevel@tonic-gate struct _svc_user_fd_node *next; 1347c478bd9Sstevel@tonic-gate struct _svc_user_fd_node *previous; 1357c478bd9Sstevel@tonic-gate } _svc_user_link; 1367c478bd9Sstevel@tonic-gate 13761961e0fSrobinson typedef struct _svc_user_fd_node { 1387c478bd9Sstevel@tonic-gate /* The lnk field must be the first field. */ 1397c478bd9Sstevel@tonic-gate _svc_user_link lnk; 1407c478bd9Sstevel@tonic-gate svc_input_id_t id; 1417c478bd9Sstevel@tonic-gate int fd; 1427c478bd9Sstevel@tonic-gate unsigned int events; 1437c478bd9Sstevel@tonic-gate svc_callback_t callback; 1447c478bd9Sstevel@tonic-gate void* cookie; 1457c478bd9Sstevel@tonic-gate } _svc_user_fd_node; 1467c478bd9Sstevel@tonic-gate 14761961e0fSrobinson typedef struct _svc_user_fd_head { 1487c478bd9Sstevel@tonic-gate /* The lnk field must be the first field. */ 1497c478bd9Sstevel@tonic-gate _svc_user_link lnk; 1507c478bd9Sstevel@tonic-gate unsigned int mask; /* logical OR of all sub-masks */ 1517c478bd9Sstevel@tonic-gate } _svc_user_fd_head; 1527c478bd9Sstevel@tonic-gate 1537c478bd9Sstevel@tonic-gate 1547c478bd9Sstevel@tonic-gate /* Define some macros to manage the linked list. */ 1557c478bd9Sstevel@tonic-gate #define LIST_ISEMPTY(l) ((_svc_user_fd_node *) &(l.lnk) == l.lnk.next) 1567c478bd9Sstevel@tonic-gate #define LIST_CLR(l) \ 1577c478bd9Sstevel@tonic-gate (l.lnk.previous = l.lnk.next = (_svc_user_fd_node *) &(l.lnk)) 1587c478bd9Sstevel@tonic-gate 1597c478bd9Sstevel@tonic-gate /* Array of defined reactor - indexed on file descriptor */ 1607c478bd9Sstevel@tonic-gate static _svc_user_fd_head *svc_userfds = NULL; 1617c478bd9Sstevel@tonic-gate 1627c478bd9Sstevel@tonic-gate /* current size of file descriptor */ 1637c478bd9Sstevel@tonic-gate static int svc_nuserfds = 0; 1647c478bd9Sstevel@tonic-gate 1657c478bd9Sstevel@tonic-gate /* Mutex to ensure MT safe operations for user fds callbacks. */ 1667c478bd9Sstevel@tonic-gate static mutex_t svc_userfds_lock = DEFAULTMUTEX; 1677c478bd9Sstevel@tonic-gate 1687c478bd9Sstevel@tonic-gate 1697c478bd9Sstevel@tonic-gate /* 1707c478bd9Sstevel@tonic-gate * This structure is used to have constant time alogrithms. There is an array 1717c478bd9Sstevel@tonic-gate * of this structure as large as svc_nuserfds. When the user is registering a 1727c478bd9Sstevel@tonic-gate * new callback, the address of the created structure is stored in a cell of 1737c478bd9Sstevel@tonic-gate * this array. The address of this cell is the returned unique identifier. 1747c478bd9Sstevel@tonic-gate * 1757c478bd9Sstevel@tonic-gate * On removing, the id is given by the user, then we know if this cell is 1767c478bd9Sstevel@tonic-gate * filled or not (with free). If it is free, we return an error. Otherwise, 1777c478bd9Sstevel@tonic-gate * we can free the structure pointed by fd_node. 1787c478bd9Sstevel@tonic-gate * 1797c478bd9Sstevel@tonic-gate * On insertion, we use the linked list created by (first_free, 1807c478bd9Sstevel@tonic-gate * next_free). In this way with a constant time computation, we can give a 1817c478bd9Sstevel@tonic-gate * correct index to the user. 1827c478bd9Sstevel@tonic-gate */ 1837c478bd9Sstevel@tonic-gate 18461961e0fSrobinson typedef struct _svc_management_user_fd { 1857c478bd9Sstevel@tonic-gate bool_t free; 1867c478bd9Sstevel@tonic-gate union { 1877c478bd9Sstevel@tonic-gate svc_input_id_t next_free; 1887c478bd9Sstevel@tonic-gate _svc_user_fd_node *fd_node; 1897c478bd9Sstevel@tonic-gate } data; 1907c478bd9Sstevel@tonic-gate } _svc_management_user_fd; 1917c478bd9Sstevel@tonic-gate 1927c478bd9Sstevel@tonic-gate /* index to the first free elem */ 1937c478bd9Sstevel@tonic-gate static svc_input_id_t first_free = (svc_input_id_t)-1; 1947c478bd9Sstevel@tonic-gate /* the size of this array is the same as svc_nuserfds */ 1957c478bd9Sstevel@tonic-gate static _svc_management_user_fd* user_fd_mgt_array = NULL; 1967c478bd9Sstevel@tonic-gate 1977c478bd9Sstevel@tonic-gate /* current size of user_fd_mgt_array */ 1987c478bd9Sstevel@tonic-gate static int svc_nmgtuserfds = 0; 1997c478bd9Sstevel@tonic-gate 2007c478bd9Sstevel@tonic-gate 2017c478bd9Sstevel@tonic-gate /* Define some macros to access data associated to registration ids. */ 2027c478bd9Sstevel@tonic-gate #define node_from_id(id) (user_fd_mgt_array[(int)id].data.fd_node) 2037c478bd9Sstevel@tonic-gate #define is_free_id(id) (user_fd_mgt_array[(int)id].free) 2047c478bd9Sstevel@tonic-gate 2057c478bd9Sstevel@tonic-gate #ifndef POLLSTANDARD 2067c478bd9Sstevel@tonic-gate #define POLLSTANDARD \ 2077c478bd9Sstevel@tonic-gate (POLLIN|POLLPRI|POLLOUT|POLLRDNORM|POLLRDBAND| \ 2087c478bd9Sstevel@tonic-gate POLLWRBAND|POLLERR|POLLHUP|POLLNVAL) 2097c478bd9Sstevel@tonic-gate #endif 2107c478bd9Sstevel@tonic-gate 2117c478bd9Sstevel@tonic-gate /* 2127c478bd9Sstevel@tonic-gate * To free an Id, we set the cell as free and insert its address in the list 2137c478bd9Sstevel@tonic-gate * of free cell. 2147c478bd9Sstevel@tonic-gate */ 2157c478bd9Sstevel@tonic-gate 2167c478bd9Sstevel@tonic-gate static void 2177c478bd9Sstevel@tonic-gate _svc_free_id(const svc_input_id_t id) 2187c478bd9Sstevel@tonic-gate { 2197c478bd9Sstevel@tonic-gate assert(((int)id >= 0) && ((int)id < svc_nmgtuserfds)); 2207c478bd9Sstevel@tonic-gate user_fd_mgt_array[(int)id].free = TRUE; 2217c478bd9Sstevel@tonic-gate user_fd_mgt_array[(int)id].data.next_free = first_free; 2227c478bd9Sstevel@tonic-gate first_free = id; 2237c478bd9Sstevel@tonic-gate } 2247c478bd9Sstevel@tonic-gate 2257c478bd9Sstevel@tonic-gate /* 2267c478bd9Sstevel@tonic-gate * To get a free cell, we just have to take it from the free linked list and 2277c478bd9Sstevel@tonic-gate * set the flag to "not free". This function also allocates new memory if 2287c478bd9Sstevel@tonic-gate * necessary 2297c478bd9Sstevel@tonic-gate */ 2307c478bd9Sstevel@tonic-gate static svc_input_id_t 2317c478bd9Sstevel@tonic-gate _svc_attribute_new_id(_svc_user_fd_node *node) 2327c478bd9Sstevel@tonic-gate { 2337c478bd9Sstevel@tonic-gate int selected_index = (int)first_free; 2347c478bd9Sstevel@tonic-gate assert(node != NULL); 2357c478bd9Sstevel@tonic-gate 2367c478bd9Sstevel@tonic-gate if (selected_index == -1) { 2377c478bd9Sstevel@tonic-gate /* Allocate new entries */ 2387c478bd9Sstevel@tonic-gate int L_inOldSize = svc_nmgtuserfds; 2397c478bd9Sstevel@tonic-gate int i; 2407c478bd9Sstevel@tonic-gate 2417c478bd9Sstevel@tonic-gate svc_nmgtuserfds += USER_FD_INCREMENT; 2427c478bd9Sstevel@tonic-gate 2437c478bd9Sstevel@tonic-gate user_fd_mgt_array = (_svc_management_user_fd *) 2447c478bd9Sstevel@tonic-gate realloc(user_fd_mgt_array, svc_nmgtuserfds 2457c478bd9Sstevel@tonic-gate * sizeof (_svc_management_user_fd)); 2467c478bd9Sstevel@tonic-gate 2477c478bd9Sstevel@tonic-gate if (user_fd_mgt_array == NULL) { 2487c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "_svc_attribute_new_id: out of memory"); 2497c478bd9Sstevel@tonic-gate errno = ENOMEM; 2507c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 2517c478bd9Sstevel@tonic-gate } 2527c478bd9Sstevel@tonic-gate 2537c478bd9Sstevel@tonic-gate for (i = svc_nmgtuserfds - 1; i >= L_inOldSize; i--) 2547c478bd9Sstevel@tonic-gate _svc_free_id((svc_input_id_t)i); 2557c478bd9Sstevel@tonic-gate selected_index = (int)first_free; 2567c478bd9Sstevel@tonic-gate } 2577c478bd9Sstevel@tonic-gate 2587c478bd9Sstevel@tonic-gate node->id = (svc_input_id_t)selected_index; 2597c478bd9Sstevel@tonic-gate first_free = user_fd_mgt_array[selected_index].data.next_free; 2607c478bd9Sstevel@tonic-gate 2617c478bd9Sstevel@tonic-gate user_fd_mgt_array[selected_index].data.fd_node = node; 2627c478bd9Sstevel@tonic-gate user_fd_mgt_array[selected_index].free = FALSE; 2637c478bd9Sstevel@tonic-gate 2647c478bd9Sstevel@tonic-gate return ((svc_input_id_t)selected_index); 2657c478bd9Sstevel@tonic-gate } 2667c478bd9Sstevel@tonic-gate 2677c478bd9Sstevel@tonic-gate /* 2687c478bd9Sstevel@tonic-gate * Access to a pollfd treatment. Scan all the associated callbacks that have 2697c478bd9Sstevel@tonic-gate * at least one bit in their mask that masks a received event. 2707c478bd9Sstevel@tonic-gate * 2717c478bd9Sstevel@tonic-gate * If event POLLNVAL is received, we check that one callback processes it, if 2727c478bd9Sstevel@tonic-gate * not, then remove the file descriptor from the poll. If there is one, let 2737c478bd9Sstevel@tonic-gate * the user do the work. 2747c478bd9Sstevel@tonic-gate */ 2757c478bd9Sstevel@tonic-gate void 2767c478bd9Sstevel@tonic-gate __svc_getreq_user(struct pollfd *pfd) 2777c478bd9Sstevel@tonic-gate { 2787c478bd9Sstevel@tonic-gate int fd = pfd->fd; 2797c478bd9Sstevel@tonic-gate short revents = pfd->revents; 2807c478bd9Sstevel@tonic-gate bool_t invalHandled = FALSE; 2817c478bd9Sstevel@tonic-gate _svc_user_fd_node *node; 2827c478bd9Sstevel@tonic-gate 28361961e0fSrobinson (void) mutex_lock(&svc_userfds_lock); 2847c478bd9Sstevel@tonic-gate 2857c478bd9Sstevel@tonic-gate if ((fd < 0) || (fd >= svc_nuserfds)) { 28661961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 2877c478bd9Sstevel@tonic-gate return; 2887c478bd9Sstevel@tonic-gate } 2897c478bd9Sstevel@tonic-gate 2907c478bd9Sstevel@tonic-gate node = svc_userfds[fd].lnk.next; 2917c478bd9Sstevel@tonic-gate 2927c478bd9Sstevel@tonic-gate /* check if at least one mask fits */ 2937c478bd9Sstevel@tonic-gate if (0 == (revents & svc_userfds[fd].mask)) { 29461961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 2957c478bd9Sstevel@tonic-gate return; 2967c478bd9Sstevel@tonic-gate } 2977c478bd9Sstevel@tonic-gate 2987c478bd9Sstevel@tonic-gate while ((svc_userfds[fd].mask != 0) && 2997c478bd9Sstevel@tonic-gate ((_svc_user_link *)node != &(svc_userfds[fd].lnk))) { 3007c478bd9Sstevel@tonic-gate /* 3017c478bd9Sstevel@tonic-gate * If one of the received events maps the ones the node listens 3027c478bd9Sstevel@tonic-gate * to 3037c478bd9Sstevel@tonic-gate */ 3047c478bd9Sstevel@tonic-gate _svc_user_fd_node *next = node->lnk.next; 3057c478bd9Sstevel@tonic-gate 3067c478bd9Sstevel@tonic-gate if (node->callback != NULL) { 3077c478bd9Sstevel@tonic-gate if (node->events & revents) { 3087c478bd9Sstevel@tonic-gate if (revents & POLLNVAL) { 3097c478bd9Sstevel@tonic-gate invalHandled = TRUE; 3107c478bd9Sstevel@tonic-gate } 3117c478bd9Sstevel@tonic-gate 3127c478bd9Sstevel@tonic-gate /* 3137c478bd9Sstevel@tonic-gate * The lock must be released before calling the 3147c478bd9Sstevel@tonic-gate * user function, as this function can call 3157c478bd9Sstevel@tonic-gate * svc_remove_input() for example. 3167c478bd9Sstevel@tonic-gate */ 31761961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 3187c478bd9Sstevel@tonic-gate node->callback(node->id, node->fd, 3197c478bd9Sstevel@tonic-gate node->events & revents, node->cookie); 3207c478bd9Sstevel@tonic-gate /* 3217c478bd9Sstevel@tonic-gate * Do not use the node structure anymore, as it 3227c478bd9Sstevel@tonic-gate * could have been deallocated by the previous 3237c478bd9Sstevel@tonic-gate * callback. 3247c478bd9Sstevel@tonic-gate */ 32561961e0fSrobinson (void) mutex_lock(&svc_userfds_lock); 3267c478bd9Sstevel@tonic-gate } 3277c478bd9Sstevel@tonic-gate } 3287c478bd9Sstevel@tonic-gate node = next; 3297c478bd9Sstevel@tonic-gate } 3307c478bd9Sstevel@tonic-gate 3317c478bd9Sstevel@tonic-gate if ((revents & POLLNVAL) && !invalHandled) 3327c478bd9Sstevel@tonic-gate __svc_remove_input_of_fd(fd); 33361961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 3347c478bd9Sstevel@tonic-gate } 3357c478bd9Sstevel@tonic-gate 3367c478bd9Sstevel@tonic-gate 3377c478bd9Sstevel@tonic-gate /* 3387c478bd9Sstevel@tonic-gate * Check if a file descriptor is associated with a user reactor. 3397c478bd9Sstevel@tonic-gate * To do this, just check that the array indexed on fd has a non-void linked 3407c478bd9Sstevel@tonic-gate * list (ie. first element is not NULL) 3417c478bd9Sstevel@tonic-gate */ 3427c478bd9Sstevel@tonic-gate bool_t 3437c478bd9Sstevel@tonic-gate __is_a_userfd(int fd) 3447c478bd9Sstevel@tonic-gate { 3457c478bd9Sstevel@tonic-gate /* Checks argument */ 3467c478bd9Sstevel@tonic-gate if ((fd < 0) || (fd >= svc_nuserfds)) 3477c478bd9Sstevel@tonic-gate return (FALSE); 3487c478bd9Sstevel@tonic-gate return ((svc_userfds[fd].mask == 0x0000)? FALSE:TRUE); 3497c478bd9Sstevel@tonic-gate } 3507c478bd9Sstevel@tonic-gate 3517c478bd9Sstevel@tonic-gate /* free everything concerning user fd */ 3527c478bd9Sstevel@tonic-gate /* used in svc_run.c => no static */ 3537c478bd9Sstevel@tonic-gate 3547c478bd9Sstevel@tonic-gate void 35561961e0fSrobinson __destroy_userfd(void) 3567c478bd9Sstevel@tonic-gate { 3577c478bd9Sstevel@tonic-gate int one_fd; 3587c478bd9Sstevel@tonic-gate /* Clean user fd */ 3597c478bd9Sstevel@tonic-gate if (svc_userfds != NULL) { 3607c478bd9Sstevel@tonic-gate for (one_fd = 0; one_fd < svc_nuserfds; one_fd++) { 3617c478bd9Sstevel@tonic-gate _svc_user_fd_node *node; 3627c478bd9Sstevel@tonic-gate 3637c478bd9Sstevel@tonic-gate node = svc_userfds[one_fd].lnk.next; 3647c478bd9Sstevel@tonic-gate while ((_svc_user_link *) node 3657c478bd9Sstevel@tonic-gate != (_svc_user_link *) &(svc_userfds[one_fd])) { 3667c478bd9Sstevel@tonic-gate _svc_free_id(node->id); 3677c478bd9Sstevel@tonic-gate node = node->lnk.next; 3687c478bd9Sstevel@tonic-gate free(node->lnk.previous); 3697c478bd9Sstevel@tonic-gate } 3707c478bd9Sstevel@tonic-gate } 3717c478bd9Sstevel@tonic-gate 3727c478bd9Sstevel@tonic-gate free(user_fd_mgt_array); 3737c478bd9Sstevel@tonic-gate user_fd_mgt_array = NULL; 3747c478bd9Sstevel@tonic-gate first_free = (svc_input_id_t)-1; 3757c478bd9Sstevel@tonic-gate 3767c478bd9Sstevel@tonic-gate free(svc_userfds); 3777c478bd9Sstevel@tonic-gate svc_userfds = NULL; 3787c478bd9Sstevel@tonic-gate svc_nuserfds = 0; 3797c478bd9Sstevel@tonic-gate } 3807c478bd9Sstevel@tonic-gate } 3817c478bd9Sstevel@tonic-gate 3827c478bd9Sstevel@tonic-gate /* 3837c478bd9Sstevel@tonic-gate * Remove all the callback associated with a fd => useful when the fd is 3847c478bd9Sstevel@tonic-gate * closed for instance 3857c478bd9Sstevel@tonic-gate */ 3867c478bd9Sstevel@tonic-gate static void 3877c478bd9Sstevel@tonic-gate __svc_remove_input_of_fd(int fd) 3887c478bd9Sstevel@tonic-gate { 3897c478bd9Sstevel@tonic-gate _svc_user_fd_node *one_node; 3907c478bd9Sstevel@tonic-gate 3917c478bd9Sstevel@tonic-gate if ((fd < 0) || (fd >= svc_nuserfds)) 3927c478bd9Sstevel@tonic-gate return; 3937c478bd9Sstevel@tonic-gate 3947c478bd9Sstevel@tonic-gate one_node = svc_userfds[fd].lnk.next; 3957c478bd9Sstevel@tonic-gate while ((_svc_user_link *) one_node 3967c478bd9Sstevel@tonic-gate != (_svc_user_link *) &(svc_userfds[fd].lnk)) { 3977c478bd9Sstevel@tonic-gate _svc_free_id(one_node->id); 3987c478bd9Sstevel@tonic-gate one_node = one_node->lnk.next; 3997c478bd9Sstevel@tonic-gate free(one_node->lnk.previous); 4007c478bd9Sstevel@tonic-gate } 4017c478bd9Sstevel@tonic-gate 4027c478bd9Sstevel@tonic-gate LIST_CLR(svc_userfds[fd]); 4037c478bd9Sstevel@tonic-gate svc_userfds[fd].mask = 0; 4047c478bd9Sstevel@tonic-gate } 4057c478bd9Sstevel@tonic-gate 4067c478bd9Sstevel@tonic-gate /* 4077c478bd9Sstevel@tonic-gate * Allow user to add an fd in the poll list. If it does not succeed, return 4087c478bd9Sstevel@tonic-gate * -1. Otherwise, return a svc_id 4097c478bd9Sstevel@tonic-gate */ 4107c478bd9Sstevel@tonic-gate 4117c478bd9Sstevel@tonic-gate svc_input_id_t 4127c478bd9Sstevel@tonic-gate svc_add_input(int user_fd, unsigned int events, 4137c478bd9Sstevel@tonic-gate svc_callback_t user_callback, void *cookie) 4147c478bd9Sstevel@tonic-gate { 4157c478bd9Sstevel@tonic-gate _svc_user_fd_node *new_node; 4167c478bd9Sstevel@tonic-gate 4177c478bd9Sstevel@tonic-gate if (user_fd < 0) { 4187c478bd9Sstevel@tonic-gate errno = EINVAL; 4197c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 4207c478bd9Sstevel@tonic-gate } 4217c478bd9Sstevel@tonic-gate 4227c478bd9Sstevel@tonic-gate if ((events == 0x0000) || 4237c478bd9Sstevel@tonic-gate (events & ~(POLLIN|POLLPRI|POLLOUT|POLLRDNORM|POLLRDBAND|\ 4247c478bd9Sstevel@tonic-gate POLLWRBAND|POLLERR|POLLHUP|POLLNVAL))) { 4257c478bd9Sstevel@tonic-gate errno = EINVAL; 4267c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 4277c478bd9Sstevel@tonic-gate } 4287c478bd9Sstevel@tonic-gate 42961961e0fSrobinson (void) mutex_lock(&svc_userfds_lock); 4307c478bd9Sstevel@tonic-gate 4317c478bd9Sstevel@tonic-gate if ((user_fd < svc_nuserfds) && 4327c478bd9Sstevel@tonic-gate (svc_userfds[user_fd].mask & events) != 0) { 4337c478bd9Sstevel@tonic-gate /* Already registrated call-back */ 4347c478bd9Sstevel@tonic-gate errno = EEXIST; 43561961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 4367c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 4377c478bd9Sstevel@tonic-gate } 4387c478bd9Sstevel@tonic-gate 4397c478bd9Sstevel@tonic-gate /* Handle memory allocation. */ 4407c478bd9Sstevel@tonic-gate if (user_fd >= svc_nuserfds) { 4417c478bd9Sstevel@tonic-gate int oldSize = svc_nuserfds; 4427c478bd9Sstevel@tonic-gate int i; 4437c478bd9Sstevel@tonic-gate 4447c478bd9Sstevel@tonic-gate svc_nuserfds = (user_fd + 1) + USER_FD_INCREMENT; 4457c478bd9Sstevel@tonic-gate 4467c478bd9Sstevel@tonic-gate svc_userfds = (_svc_user_fd_head *) 4477c478bd9Sstevel@tonic-gate realloc(svc_userfds, 4487c478bd9Sstevel@tonic-gate svc_nuserfds * sizeof (_svc_user_fd_head)); 4497c478bd9Sstevel@tonic-gate 4507c478bd9Sstevel@tonic-gate if (svc_userfds == NULL) { 4517c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "svc_add_input: out of memory"); 4527c478bd9Sstevel@tonic-gate errno = ENOMEM; 45361961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 4547c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 4557c478bd9Sstevel@tonic-gate } 4567c478bd9Sstevel@tonic-gate 4577c478bd9Sstevel@tonic-gate for (i = oldSize; i < svc_nuserfds; i++) { 4587c478bd9Sstevel@tonic-gate LIST_CLR(svc_userfds[i]); 4597c478bd9Sstevel@tonic-gate svc_userfds[i].mask = 0; 4607c478bd9Sstevel@tonic-gate } 4617c478bd9Sstevel@tonic-gate } 4627c478bd9Sstevel@tonic-gate 46361961e0fSrobinson new_node = malloc(sizeof (_svc_user_fd_node)); 4647c478bd9Sstevel@tonic-gate if (new_node == NULL) { 4657c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "svc_add_input: out of memory"); 4667c478bd9Sstevel@tonic-gate errno = ENOMEM; 46761961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 4687c478bd9Sstevel@tonic-gate return ((svc_input_id_t)-1); 4697c478bd9Sstevel@tonic-gate } 4707c478bd9Sstevel@tonic-gate 4717c478bd9Sstevel@tonic-gate /* create a new node */ 4727c478bd9Sstevel@tonic-gate new_node->fd = user_fd; 4737c478bd9Sstevel@tonic-gate new_node->events = events; 4747c478bd9Sstevel@tonic-gate new_node->callback = user_callback; 4757c478bd9Sstevel@tonic-gate new_node->cookie = cookie; 4767c478bd9Sstevel@tonic-gate 4777c478bd9Sstevel@tonic-gate (void) _svc_attribute_new_id(new_node); 4787c478bd9Sstevel@tonic-gate 4797c478bd9Sstevel@tonic-gate /* Add the new element at the beginning of the list. */ 4807c478bd9Sstevel@tonic-gate if (LIST_ISEMPTY(svc_userfds[user_fd])) { 4817c478bd9Sstevel@tonic-gate svc_userfds[user_fd].lnk.previous = new_node; 4827c478bd9Sstevel@tonic-gate } 4837c478bd9Sstevel@tonic-gate new_node->lnk.next = svc_userfds[user_fd].lnk.next; 4847c478bd9Sstevel@tonic-gate new_node->lnk.previous = (_svc_user_fd_node *)&(svc_userfds[user_fd]); 4857c478bd9Sstevel@tonic-gate 4867c478bd9Sstevel@tonic-gate svc_userfds[user_fd].lnk.next = new_node; 4877c478bd9Sstevel@tonic-gate 4887c478bd9Sstevel@tonic-gate /* refresh global mask for this file desciptor */ 4897c478bd9Sstevel@tonic-gate svc_userfds[user_fd].mask |= events; 4907c478bd9Sstevel@tonic-gate 4917c478bd9Sstevel@tonic-gate /* refresh mask for the poll */ 4927c478bd9Sstevel@tonic-gate add_pollfd(user_fd, (svc_userfds[user_fd].mask)); 4937c478bd9Sstevel@tonic-gate 49461961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 4957c478bd9Sstevel@tonic-gate return (new_node->id); 4967c478bd9Sstevel@tonic-gate } 4977c478bd9Sstevel@tonic-gate 4987c478bd9Sstevel@tonic-gate 4997c478bd9Sstevel@tonic-gate int 5007c478bd9Sstevel@tonic-gate svc_remove_input(svc_input_id_t id) 5017c478bd9Sstevel@tonic-gate { 5027c478bd9Sstevel@tonic-gate _svc_user_fd_node* node; 5037c478bd9Sstevel@tonic-gate _svc_user_fd_node* next; 5047c478bd9Sstevel@tonic-gate _svc_user_fd_node* previous; 5057c478bd9Sstevel@tonic-gate int fd; /* caching optim */ 5067c478bd9Sstevel@tonic-gate 50761961e0fSrobinson (void) mutex_lock(&svc_userfds_lock); 5087c478bd9Sstevel@tonic-gate 5097c478bd9Sstevel@tonic-gate /* Immediately update data for id management */ 5107c478bd9Sstevel@tonic-gate if (user_fd_mgt_array == NULL || id >= svc_nmgtuserfds || 5117c478bd9Sstevel@tonic-gate is_free_id(id)) { 5127c478bd9Sstevel@tonic-gate errno = EINVAL; 51361961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 5147c478bd9Sstevel@tonic-gate return (-1); 5157c478bd9Sstevel@tonic-gate } 5167c478bd9Sstevel@tonic-gate 5177c478bd9Sstevel@tonic-gate node = node_from_id(id); 5187c478bd9Sstevel@tonic-gate assert(node != NULL); 5197c478bd9Sstevel@tonic-gate 5207c478bd9Sstevel@tonic-gate _svc_free_id(id); 5217c478bd9Sstevel@tonic-gate next = node->lnk.next; 5227c478bd9Sstevel@tonic-gate previous = node->lnk.previous; 5237c478bd9Sstevel@tonic-gate fd = node->fd; /* caching optim */ 5247c478bd9Sstevel@tonic-gate 5257c478bd9Sstevel@tonic-gate /* Remove this node from the list. */ 5267c478bd9Sstevel@tonic-gate previous->lnk.next = next; 5277c478bd9Sstevel@tonic-gate next->lnk.previous = previous; 5287c478bd9Sstevel@tonic-gate 5297c478bd9Sstevel@tonic-gate /* Remove the node flags from the global mask */ 5307c478bd9Sstevel@tonic-gate svc_userfds[fd].mask ^= node->events; 5317c478bd9Sstevel@tonic-gate 5327c478bd9Sstevel@tonic-gate free(node); 5337c478bd9Sstevel@tonic-gate if (svc_userfds[fd].mask == 0) { 5347c478bd9Sstevel@tonic-gate LIST_CLR(svc_userfds[fd]); 5357c478bd9Sstevel@tonic-gate assert(LIST_ISEMPTY(svc_userfds[fd])); 5367c478bd9Sstevel@tonic-gate remove_pollfd(fd); 5377c478bd9Sstevel@tonic-gate } 5387c478bd9Sstevel@tonic-gate /* <=> CLEAN NEEDED TO SHRINK MEMORY USAGE */ 5397c478bd9Sstevel@tonic-gate 54061961e0fSrobinson (void) mutex_unlock(&svc_userfds_lock); 5417c478bd9Sstevel@tonic-gate return (0); 5427c478bd9Sstevel@tonic-gate } 5437c478bd9Sstevel@tonic-gate 5447c478bd9Sstevel@tonic-gate 5457c478bd9Sstevel@tonic-gate /* 5467c478bd9Sstevel@tonic-gate * Provides default service-side functions for authentication flavors 5477c478bd9Sstevel@tonic-gate * that do not use all the fields in struct svc_auth_ops. 5487c478bd9Sstevel@tonic-gate */ 5497c478bd9Sstevel@tonic-gate 5507c478bd9Sstevel@tonic-gate /*ARGSUSED*/ 5517c478bd9Sstevel@tonic-gate static int 55261961e0fSrobinson authany_wrap(AUTH *auth, XDR *xdrs, xdrproc_t xfunc, caddr_t xwhere) 5537c478bd9Sstevel@tonic-gate { 5547c478bd9Sstevel@tonic-gate return (*xfunc)(xdrs, xwhere); 5557c478bd9Sstevel@tonic-gate } 5567c478bd9Sstevel@tonic-gate 5577c478bd9Sstevel@tonic-gate struct svc_auth_ops svc_auth_any_ops = { 5587c478bd9Sstevel@tonic-gate authany_wrap, 5597c478bd9Sstevel@tonic-gate authany_wrap, 5607c478bd9Sstevel@tonic-gate }; 5617c478bd9Sstevel@tonic-gate 5627c478bd9Sstevel@tonic-gate /* 5637c478bd9Sstevel@tonic-gate * Return pointer to server authentication structure. 5647c478bd9Sstevel@tonic-gate */ 5657c478bd9Sstevel@tonic-gate SVCAUTH * 56661961e0fSrobinson __svc_get_svcauth(SVCXPRT *xprt) 5677c478bd9Sstevel@tonic-gate { 5687c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 5697c478bd9Sstevel@tonic-gate return (&SVC_XP_AUTH(xprt)); 5707c478bd9Sstevel@tonic-gate } 5717c478bd9Sstevel@tonic-gate 5727c478bd9Sstevel@tonic-gate /* 5737c478bd9Sstevel@tonic-gate * A callback routine to cleanup after a procedure is executed. 5747c478bd9Sstevel@tonic-gate */ 5757c478bd9Sstevel@tonic-gate void (*__proc_cleanup_cb)() = NULL; 5767c478bd9Sstevel@tonic-gate 5777c478bd9Sstevel@tonic-gate void * 57861961e0fSrobinson __svc_set_proc_cleanup_cb(void *cb) 5797c478bd9Sstevel@tonic-gate { 5807c478bd9Sstevel@tonic-gate void *tmp = (void *)__proc_cleanup_cb; 5817c478bd9Sstevel@tonic-gate 5827c478bd9Sstevel@tonic-gate __proc_cleanup_cb = (void (*)())cb; 5837c478bd9Sstevel@tonic-gate return (tmp); 5847c478bd9Sstevel@tonic-gate } 5857c478bd9Sstevel@tonic-gate 5867c478bd9Sstevel@tonic-gate /* *************** SVCXPRT related stuff **************** */ 5877c478bd9Sstevel@tonic-gate 5887c478bd9Sstevel@tonic-gate 5897c478bd9Sstevel@tonic-gate static int pollfd_shrinking = 1; 5907c478bd9Sstevel@tonic-gate 5917c478bd9Sstevel@tonic-gate 5927c478bd9Sstevel@tonic-gate /* 5937c478bd9Sstevel@tonic-gate * Add fd to svc_pollfd 5947c478bd9Sstevel@tonic-gate */ 5957c478bd9Sstevel@tonic-gate static void 5967c478bd9Sstevel@tonic-gate add_pollfd(int fd, short events) 5977c478bd9Sstevel@tonic-gate { 5987c478bd9Sstevel@tonic-gate if (fd < FD_SETSIZE) { 5997c478bd9Sstevel@tonic-gate FD_SET(fd, &svc_fdset); 6007c478bd9Sstevel@tonic-gate #if !defined(_LP64) 6017c478bd9Sstevel@tonic-gate FD_SET(fd, &_new_svc_fdset); 6027c478bd9Sstevel@tonic-gate #endif 6037c478bd9Sstevel@tonic-gate svc_nfds++; 6047c478bd9Sstevel@tonic-gate svc_nfds_set++; 6057c478bd9Sstevel@tonic-gate if (fd >= svc_max_fd) 6067c478bd9Sstevel@tonic-gate svc_max_fd = fd + 1; 6077c478bd9Sstevel@tonic-gate } 6087c478bd9Sstevel@tonic-gate if (fd >= svc_max_pollfd) 6097c478bd9Sstevel@tonic-gate svc_max_pollfd = fd + 1; 6107c478bd9Sstevel@tonic-gate if (svc_max_pollfd > svc_pollfd_allocd) { 6117c478bd9Sstevel@tonic-gate int i = svc_pollfd_allocd; 6127c478bd9Sstevel@tonic-gate pollfd_t *tmp; 6137c478bd9Sstevel@tonic-gate do { 6147c478bd9Sstevel@tonic-gate svc_pollfd_allocd += POLLFD_EXTEND; 6157c478bd9Sstevel@tonic-gate } while (svc_max_pollfd > svc_pollfd_allocd); 6167c478bd9Sstevel@tonic-gate tmp = realloc(svc_pollfd, 6177c478bd9Sstevel@tonic-gate sizeof (pollfd_t) * svc_pollfd_allocd); 6187c478bd9Sstevel@tonic-gate if (tmp != NULL) { 6197c478bd9Sstevel@tonic-gate svc_pollfd = tmp; 6207c478bd9Sstevel@tonic-gate for (; i < svc_pollfd_allocd; i++) 6217c478bd9Sstevel@tonic-gate POLLFD_CLR(i, tmp); 6227c478bd9Sstevel@tonic-gate } else { 6237c478bd9Sstevel@tonic-gate /* 6247c478bd9Sstevel@tonic-gate * give an error message; undo fdset setting 6257c478bd9Sstevel@tonic-gate * above; reset the pollfd_shrinking flag. 6267c478bd9Sstevel@tonic-gate * because of this poll will not be done 6277c478bd9Sstevel@tonic-gate * on these fds. 6287c478bd9Sstevel@tonic-gate */ 6297c478bd9Sstevel@tonic-gate if (fd < FD_SETSIZE) { 6307c478bd9Sstevel@tonic-gate FD_CLR(fd, &svc_fdset); 6317c478bd9Sstevel@tonic-gate #if !defined(_LP64) 6327c478bd9Sstevel@tonic-gate FD_CLR(fd, &_new_svc_fdset); 6337c478bd9Sstevel@tonic-gate #endif 6347c478bd9Sstevel@tonic-gate svc_nfds--; 6357c478bd9Sstevel@tonic-gate svc_nfds_set--; 6367c478bd9Sstevel@tonic-gate if (fd == (svc_max_fd - 1)) 6377c478bd9Sstevel@tonic-gate svc_max_fd--; 6387c478bd9Sstevel@tonic-gate } 6397c478bd9Sstevel@tonic-gate if (fd == (svc_max_pollfd - 1)) 6407c478bd9Sstevel@tonic-gate svc_max_pollfd--; 6417c478bd9Sstevel@tonic-gate pollfd_shrinking = 0; 6427c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "add_pollfd: out of memory"); 6437c478bd9Sstevel@tonic-gate _exit(1); 6447c478bd9Sstevel@tonic-gate } 6457c478bd9Sstevel@tonic-gate } 6467c478bd9Sstevel@tonic-gate svc_pollfd[fd].fd = fd; 6477c478bd9Sstevel@tonic-gate svc_pollfd[fd].events = events; 6487c478bd9Sstevel@tonic-gate svc_npollfds++; 6497c478bd9Sstevel@tonic-gate svc_npollfds_set++; 6507c478bd9Sstevel@tonic-gate } 6517c478bd9Sstevel@tonic-gate 6527c478bd9Sstevel@tonic-gate /* 6537c478bd9Sstevel@tonic-gate * the fd is still active but only the bit in fdset is cleared. 6547c478bd9Sstevel@tonic-gate * do not subtract svc_nfds or svc_npollfds 6557c478bd9Sstevel@tonic-gate */ 6567c478bd9Sstevel@tonic-gate void 6577c478bd9Sstevel@tonic-gate clear_pollfd(int fd) 6587c478bd9Sstevel@tonic-gate { 6597c478bd9Sstevel@tonic-gate if (fd < FD_SETSIZE && FD_ISSET(fd, &svc_fdset)) { 6607c478bd9Sstevel@tonic-gate FD_CLR(fd, &svc_fdset); 6617c478bd9Sstevel@tonic-gate #if !defined(_LP64) 6627c478bd9Sstevel@tonic-gate FD_CLR(fd, &_new_svc_fdset); 6637c478bd9Sstevel@tonic-gate #endif 6647c478bd9Sstevel@tonic-gate svc_nfds_set--; 6657c478bd9Sstevel@tonic-gate } 6667c478bd9Sstevel@tonic-gate if (fd < svc_pollfd_allocd && POLLFD_ISSET(fd, svc_pollfd)) { 6677c478bd9Sstevel@tonic-gate POLLFD_CLR(fd, svc_pollfd); 6687c478bd9Sstevel@tonic-gate svc_npollfds_set--; 6697c478bd9Sstevel@tonic-gate } 6707c478bd9Sstevel@tonic-gate } 6717c478bd9Sstevel@tonic-gate 6727c478bd9Sstevel@tonic-gate /* 6737c478bd9Sstevel@tonic-gate * sets the bit in fdset for an active fd so that poll() is done for that 6747c478bd9Sstevel@tonic-gate */ 6757c478bd9Sstevel@tonic-gate void 6767c478bd9Sstevel@tonic-gate set_pollfd(int fd, short events) 6777c478bd9Sstevel@tonic-gate { 6787c478bd9Sstevel@tonic-gate if (fd < FD_SETSIZE) { 6797c478bd9Sstevel@tonic-gate FD_SET(fd, &svc_fdset); 6807c478bd9Sstevel@tonic-gate #if !defined(_LP64) 6817c478bd9Sstevel@tonic-gate FD_SET(fd, &_new_svc_fdset); 6827c478bd9Sstevel@tonic-gate #endif 6837c478bd9Sstevel@tonic-gate svc_nfds_set++; 6847c478bd9Sstevel@tonic-gate } 6857c478bd9Sstevel@tonic-gate if (fd < svc_pollfd_allocd) { 6867c478bd9Sstevel@tonic-gate svc_pollfd[fd].fd = fd; 6877c478bd9Sstevel@tonic-gate svc_pollfd[fd].events = events; 6887c478bd9Sstevel@tonic-gate svc_npollfds_set++; 6897c478bd9Sstevel@tonic-gate } 6907c478bd9Sstevel@tonic-gate } 6917c478bd9Sstevel@tonic-gate 6927c478bd9Sstevel@tonic-gate /* 6937c478bd9Sstevel@tonic-gate * remove a svc_pollfd entry; it does not shrink the memory 6947c478bd9Sstevel@tonic-gate */ 6957c478bd9Sstevel@tonic-gate static void 69661961e0fSrobinson remove_pollfd(int fd) 6977c478bd9Sstevel@tonic-gate { 6987c478bd9Sstevel@tonic-gate clear_pollfd(fd); 6997c478bd9Sstevel@tonic-gate if (fd == (svc_max_fd - 1)) 7007c478bd9Sstevel@tonic-gate svc_max_fd--; 7017c478bd9Sstevel@tonic-gate svc_nfds--; 7027c478bd9Sstevel@tonic-gate if (fd == (svc_max_pollfd - 1)) 7037c478bd9Sstevel@tonic-gate svc_max_pollfd--; 7047c478bd9Sstevel@tonic-gate svc_npollfds--; 7057c478bd9Sstevel@tonic-gate } 7067c478bd9Sstevel@tonic-gate 7077c478bd9Sstevel@tonic-gate /* 7087c478bd9Sstevel@tonic-gate * delete a svc_pollfd entry; it shrinks the memory 7097c478bd9Sstevel@tonic-gate * use remove_pollfd if you do not want to shrink 7107c478bd9Sstevel@tonic-gate */ 7117c478bd9Sstevel@tonic-gate static void 7127c478bd9Sstevel@tonic-gate delete_pollfd(int fd) 7137c478bd9Sstevel@tonic-gate { 7147c478bd9Sstevel@tonic-gate remove_pollfd(fd); 7157c478bd9Sstevel@tonic-gate if (pollfd_shrinking && svc_max_pollfd < 7167c478bd9Sstevel@tonic-gate (svc_pollfd_allocd - POLLFD_SHRINK)) { 7177c478bd9Sstevel@tonic-gate do { 7187c478bd9Sstevel@tonic-gate svc_pollfd_allocd -= POLLFD_SHRINK; 7197c478bd9Sstevel@tonic-gate } while (svc_max_pollfd < (svc_pollfd_allocd - POLLFD_SHRINK)); 7207c478bd9Sstevel@tonic-gate svc_pollfd = realloc(svc_pollfd, 7217c478bd9Sstevel@tonic-gate sizeof (pollfd_t) * svc_pollfd_allocd); 7227c478bd9Sstevel@tonic-gate if (svc_pollfd == NULL) { 7237c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "delete_pollfd: out of memory"); 7247c478bd9Sstevel@tonic-gate _exit(1); 7257c478bd9Sstevel@tonic-gate } 7267c478bd9Sstevel@tonic-gate } 7277c478bd9Sstevel@tonic-gate } 7287c478bd9Sstevel@tonic-gate 7297c478bd9Sstevel@tonic-gate 7307c478bd9Sstevel@tonic-gate /* 7317c478bd9Sstevel@tonic-gate * Activate a transport handle. 7327c478bd9Sstevel@tonic-gate */ 7337c478bd9Sstevel@tonic-gate void 73461961e0fSrobinson xprt_register(const SVCXPRT *xprt) 7357c478bd9Sstevel@tonic-gate { 7367c478bd9Sstevel@tonic-gate int fd = xprt->xp_fd; 7377c478bd9Sstevel@tonic-gate #ifdef CALLBACK 7387c478bd9Sstevel@tonic-gate extern void (*_svc_getreqset_proc)(); 7397c478bd9Sstevel@tonic-gate #endif 7407c478bd9Sstevel@tonic-gate /* VARIABLES PROTECTED BY svc_fd_lock: svc_xports, svc_fdset */ 7417c478bd9Sstevel@tonic-gate 74261961e0fSrobinson (void) rw_wrlock(&svc_fd_lock); 7437c478bd9Sstevel@tonic-gate if (svc_xports == NULL) { 7447c478bd9Sstevel@tonic-gate /* allocate some small amount first */ 7457c478bd9Sstevel@tonic-gate svc_xports = calloc(FD_INCREMENT, sizeof (SVCXPRT *)); 7467c478bd9Sstevel@tonic-gate if (svc_xports == NULL) { 7477c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "xprt_register: out of memory"); 7487c478bd9Sstevel@tonic-gate _exit(1); 7497c478bd9Sstevel@tonic-gate } 7507c478bd9Sstevel@tonic-gate nsvc_xports = FD_INCREMENT; 7517c478bd9Sstevel@tonic-gate 7527c478bd9Sstevel@tonic-gate #ifdef CALLBACK 7537c478bd9Sstevel@tonic-gate /* 7547c478bd9Sstevel@tonic-gate * XXX: This code does not keep track of the server state. 7557c478bd9Sstevel@tonic-gate * 7567c478bd9Sstevel@tonic-gate * This provides for callback support. When a client 7577c478bd9Sstevel@tonic-gate * recv's a call from another client on the server fd's, 7587c478bd9Sstevel@tonic-gate * it calls _svc_getreqset_proc() which would return 7597c478bd9Sstevel@tonic-gate * after serving all the server requests. Also look under 7607c478bd9Sstevel@tonic-gate * clnt_dg.c and clnt_vc.c (clnt_call part of it) 7617c478bd9Sstevel@tonic-gate */ 7627c478bd9Sstevel@tonic-gate _svc_getreqset_proc = svc_getreq_poll; 7637c478bd9Sstevel@tonic-gate #endif 7647c478bd9Sstevel@tonic-gate } 7657c478bd9Sstevel@tonic-gate 7667c478bd9Sstevel@tonic-gate while (fd >= nsvc_xports) { 7677c478bd9Sstevel@tonic-gate SVCXPRT **tmp_xprts = svc_xports; 7687c478bd9Sstevel@tonic-gate 7697c478bd9Sstevel@tonic-gate /* time to expand svc_xprts */ 7707c478bd9Sstevel@tonic-gate tmp_xprts = realloc(svc_xports, 7717c478bd9Sstevel@tonic-gate sizeof (SVCXPRT *) * (nsvc_xports + FD_INCREMENT)); 7727c478bd9Sstevel@tonic-gate if (tmp_xprts == NULL) { 7737c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "xprt_register : out of memory."); 7747c478bd9Sstevel@tonic-gate _exit(1); 7757c478bd9Sstevel@tonic-gate } 7767c478bd9Sstevel@tonic-gate 7777c478bd9Sstevel@tonic-gate svc_xports = tmp_xprts; 7787c478bd9Sstevel@tonic-gate (void) memset(&svc_xports[nsvc_xports], 0, 7797c478bd9Sstevel@tonic-gate sizeof (SVCXPRT *) * FD_INCREMENT); 7807c478bd9Sstevel@tonic-gate nsvc_xports += FD_INCREMENT; 7817c478bd9Sstevel@tonic-gate } 7827c478bd9Sstevel@tonic-gate 7837c478bd9Sstevel@tonic-gate svc_xports[fd] = (SVCXPRT *)xprt; 7847c478bd9Sstevel@tonic-gate 7857c478bd9Sstevel@tonic-gate add_pollfd(fd, MASKVAL); 7867c478bd9Sstevel@tonic-gate 7877c478bd9Sstevel@tonic-gate if (svc_polling) { 7887c478bd9Sstevel@tonic-gate char dummy; 7897c478bd9Sstevel@tonic-gate 7907c478bd9Sstevel@tonic-gate /* 7917c478bd9Sstevel@tonic-gate * This happens only in one of the MT modes. 7927c478bd9Sstevel@tonic-gate * Wake up poller. 7937c478bd9Sstevel@tonic-gate */ 79461961e0fSrobinson (void) write(svc_pipe[1], &dummy, sizeof (dummy)); 7957c478bd9Sstevel@tonic-gate } 7967c478bd9Sstevel@tonic-gate /* 7977c478bd9Sstevel@tonic-gate * If already dispatching door based services, start 7987c478bd9Sstevel@tonic-gate * dispatching TLI based services now. 7997c478bd9Sstevel@tonic-gate */ 80061961e0fSrobinson (void) mutex_lock(&svc_door_mutex); 8017c478bd9Sstevel@tonic-gate if (svc_ndoorfds > 0) 80261961e0fSrobinson (void) cond_signal(&svc_door_waitcv); 80361961e0fSrobinson (void) mutex_unlock(&svc_door_mutex); 8047c478bd9Sstevel@tonic-gate 8057c478bd9Sstevel@tonic-gate if (svc_xdrs == NULL) { 8067c478bd9Sstevel@tonic-gate /* allocate initial chunk */ 8077c478bd9Sstevel@tonic-gate svc_xdrs = calloc(FD_INCREMENT, sizeof (XDR *)); 8087c478bd9Sstevel@tonic-gate if (svc_xdrs != NULL) 8097c478bd9Sstevel@tonic-gate nsvc_xdrs = FD_INCREMENT; 8107c478bd9Sstevel@tonic-gate else { 8117c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "xprt_register : out of memory."); 8127c478bd9Sstevel@tonic-gate _exit(1); 8137c478bd9Sstevel@tonic-gate } 8147c478bd9Sstevel@tonic-gate } 81561961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 8167c478bd9Sstevel@tonic-gate } 8177c478bd9Sstevel@tonic-gate 8187c478bd9Sstevel@tonic-gate /* 8197c478bd9Sstevel@tonic-gate * De-activate a transport handle. 8207c478bd9Sstevel@tonic-gate */ 8217c478bd9Sstevel@tonic-gate void 8227c478bd9Sstevel@tonic-gate __xprt_unregister_private(const SVCXPRT *xprt, bool_t lock_not_held) 8237c478bd9Sstevel@tonic-gate { 8247c478bd9Sstevel@tonic-gate int fd = xprt->xp_fd; 8257c478bd9Sstevel@tonic-gate 8267c478bd9Sstevel@tonic-gate if (lock_not_held) 82761961e0fSrobinson (void) rw_wrlock(&svc_fd_lock); 8287c478bd9Sstevel@tonic-gate if ((fd < nsvc_xports) && (svc_xports[fd] == xprt)) { 82961961e0fSrobinson svc_xports[fd] = NULL; 8307c478bd9Sstevel@tonic-gate delete_pollfd(fd); 8317c478bd9Sstevel@tonic-gate } 8327c478bd9Sstevel@tonic-gate if (lock_not_held) 83361961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 8347c478bd9Sstevel@tonic-gate __svc_rm_from_xlist(&_svc_xprtlist, xprt, &xprtlist_lock); 8357c478bd9Sstevel@tonic-gate } 8367c478bd9Sstevel@tonic-gate 8377c478bd9Sstevel@tonic-gate void 83861961e0fSrobinson xprt_unregister(const SVCXPRT *xprt) 8397c478bd9Sstevel@tonic-gate { 8407c478bd9Sstevel@tonic-gate __xprt_unregister_private(xprt, TRUE); 8417c478bd9Sstevel@tonic-gate } 8427c478bd9Sstevel@tonic-gate 8437c478bd9Sstevel@tonic-gate /* ********************** CALLOUT list related stuff ************* */ 8447c478bd9Sstevel@tonic-gate 8457c478bd9Sstevel@tonic-gate /* 8467c478bd9Sstevel@tonic-gate * Add a service program to the callout list. 8477c478bd9Sstevel@tonic-gate * The dispatch routine will be called when a rpc request for this 8487c478bd9Sstevel@tonic-gate * program number comes in. 8497c478bd9Sstevel@tonic-gate */ 8507c478bd9Sstevel@tonic-gate bool_t 85161961e0fSrobinson svc_reg(const SVCXPRT *xprt, const rpcprog_t prog, const rpcvers_t vers, 85261961e0fSrobinson void (*dispatch)(), const struct netconfig *nconf) 8537c478bd9Sstevel@tonic-gate { 8547c478bd9Sstevel@tonic-gate struct svc_callout *prev; 8557c478bd9Sstevel@tonic-gate struct svc_callout *s, **s2; 8567c478bd9Sstevel@tonic-gate struct netconfig *tnconf; 8577c478bd9Sstevel@tonic-gate char *netid = NULL; 8587c478bd9Sstevel@tonic-gate int flag = 0; 8597c478bd9Sstevel@tonic-gate 8607c478bd9Sstevel@tonic-gate /* VARIABLES PROTECTED BY svc_lock: s, prev, svc_head */ 8617c478bd9Sstevel@tonic-gate 8627c478bd9Sstevel@tonic-gate if (xprt->xp_netid) { 8637c478bd9Sstevel@tonic-gate netid = strdup(xprt->xp_netid); 8647c478bd9Sstevel@tonic-gate flag = 1; 8657c478bd9Sstevel@tonic-gate } else if (nconf && nconf->nc_netid) { 8667c478bd9Sstevel@tonic-gate netid = strdup(nconf->nc_netid); 8677c478bd9Sstevel@tonic-gate flag = 1; 8687c478bd9Sstevel@tonic-gate } else if ((tnconf = __rpcfd_to_nconf(xprt->xp_fd, xprt->xp_type)) 8697c478bd9Sstevel@tonic-gate != NULL) { 8707c478bd9Sstevel@tonic-gate netid = strdup(tnconf->nc_netid); 8717c478bd9Sstevel@tonic-gate flag = 1; 8727c478bd9Sstevel@tonic-gate freenetconfigent(tnconf); 8737c478bd9Sstevel@tonic-gate } /* must have been created with svc_raw_create */ 87461961e0fSrobinson if ((netid == NULL) && (flag == 1)) 8757c478bd9Sstevel@tonic-gate return (FALSE); 8767c478bd9Sstevel@tonic-gate 87761961e0fSrobinson (void) rw_wrlock(&svc_lock); 8787c478bd9Sstevel@tonic-gate if ((s = svc_find(prog, vers, &prev, netid)) != NULL_SVC) { 8797c478bd9Sstevel@tonic-gate if (netid) 8807c478bd9Sstevel@tonic-gate free(netid); 8817c478bd9Sstevel@tonic-gate if (s->sc_dispatch == dispatch) 8827c478bd9Sstevel@tonic-gate goto rpcb_it; /* he is registering another xptr */ 88361961e0fSrobinson (void) rw_unlock(&svc_lock); 8847c478bd9Sstevel@tonic-gate return (FALSE); 8857c478bd9Sstevel@tonic-gate } 88661961e0fSrobinson s = malloc(sizeof (struct svc_callout)); 88761961e0fSrobinson if (s == NULL) { 8887c478bd9Sstevel@tonic-gate if (netid) 8897c478bd9Sstevel@tonic-gate free(netid); 89061961e0fSrobinson (void) rw_unlock(&svc_lock); 8917c478bd9Sstevel@tonic-gate return (FALSE); 8927c478bd9Sstevel@tonic-gate } 8937c478bd9Sstevel@tonic-gate 8947c478bd9Sstevel@tonic-gate s->sc_prog = prog; 8957c478bd9Sstevel@tonic-gate s->sc_vers = vers; 8967c478bd9Sstevel@tonic-gate s->sc_dispatch = dispatch; 8977c478bd9Sstevel@tonic-gate s->sc_netid = netid; 8987c478bd9Sstevel@tonic-gate s->sc_next = NULL; 8997c478bd9Sstevel@tonic-gate 9007c478bd9Sstevel@tonic-gate /* 9017c478bd9Sstevel@tonic-gate * The ordering of transports is such that the most frequently used 9027c478bd9Sstevel@tonic-gate * one appears first. So add the new entry to the end of the list. 9037c478bd9Sstevel@tonic-gate */ 9047c478bd9Sstevel@tonic-gate for (s2 = &svc_head; *s2 != NULL; s2 = &(*s2)->sc_next) 9057c478bd9Sstevel@tonic-gate ; 9067c478bd9Sstevel@tonic-gate *s2 = s; 9077c478bd9Sstevel@tonic-gate 9087c478bd9Sstevel@tonic-gate if ((xprt->xp_netid == NULL) && (flag == 1) && netid) 9097c478bd9Sstevel@tonic-gate if ((((SVCXPRT *)xprt)->xp_netid = strdup(netid)) == NULL) { 9107c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "svc_reg : strdup failed."); 9117c478bd9Sstevel@tonic-gate free(netid); 91261961e0fSrobinson free(s); 9137c478bd9Sstevel@tonic-gate *s2 = NULL; 91461961e0fSrobinson (void) rw_unlock(&svc_lock); 9157c478bd9Sstevel@tonic-gate return (FALSE); 9167c478bd9Sstevel@tonic-gate } 9177c478bd9Sstevel@tonic-gate 9187c478bd9Sstevel@tonic-gate rpcb_it: 91961961e0fSrobinson (void) rw_unlock(&svc_lock); 9209acbbeafSnn35248 9219acbbeafSnn35248 /* now register the information with the local binder service */ 922*9ff75adeSSurya Prakki if (nconf) 9239acbbeafSnn35248 return (rpcb_set(prog, vers, nconf, &xprt->xp_ltaddr)); 924*9ff75adeSSurya Prakki return (TRUE); 9259acbbeafSnn35248 /*NOTREACHED*/ 9267c478bd9Sstevel@tonic-gate } 9277c478bd9Sstevel@tonic-gate 9287c478bd9Sstevel@tonic-gate /* 9297c478bd9Sstevel@tonic-gate * Remove a service program from the callout list. 9307c478bd9Sstevel@tonic-gate */ 9317c478bd9Sstevel@tonic-gate void 93261961e0fSrobinson svc_unreg(const rpcprog_t prog, const rpcvers_t vers) 9337c478bd9Sstevel@tonic-gate { 9347c478bd9Sstevel@tonic-gate struct svc_callout *prev; 9357c478bd9Sstevel@tonic-gate struct svc_callout *s; 9367c478bd9Sstevel@tonic-gate 9377c478bd9Sstevel@tonic-gate /* unregister the information anyway */ 9387c478bd9Sstevel@tonic-gate (void) rpcb_unset(prog, vers, NULL); 939*9ff75adeSSurya Prakki 94061961e0fSrobinson (void) rw_wrlock(&svc_lock); 9417c478bd9Sstevel@tonic-gate while ((s = svc_find(prog, vers, &prev, NULL)) != NULL_SVC) { 9427c478bd9Sstevel@tonic-gate if (prev == NULL_SVC) { 9437c478bd9Sstevel@tonic-gate svc_head = s->sc_next; 9447c478bd9Sstevel@tonic-gate } else { 9457c478bd9Sstevel@tonic-gate prev->sc_next = s->sc_next; 9467c478bd9Sstevel@tonic-gate } 9477c478bd9Sstevel@tonic-gate s->sc_next = NULL_SVC; 9487c478bd9Sstevel@tonic-gate if (s->sc_netid) 94961961e0fSrobinson free(s->sc_netid); 95061961e0fSrobinson free(s); 9517c478bd9Sstevel@tonic-gate } 95261961e0fSrobinson (void) rw_unlock(&svc_lock); 9537c478bd9Sstevel@tonic-gate } 9547c478bd9Sstevel@tonic-gate 9557c478bd9Sstevel@tonic-gate #ifdef PORTMAP 9567c478bd9Sstevel@tonic-gate /* 9577c478bd9Sstevel@tonic-gate * Add a service program to the callout list. 9587c478bd9Sstevel@tonic-gate * The dispatch routine will be called when a rpc request for this 9597c478bd9Sstevel@tonic-gate * program number comes in. 9607c478bd9Sstevel@tonic-gate * For version 2 portmappers. 9617c478bd9Sstevel@tonic-gate */ 9627c478bd9Sstevel@tonic-gate bool_t 96361961e0fSrobinson svc_register(SVCXPRT *xprt, rpcprog_t prog, rpcvers_t vers, 96461961e0fSrobinson void (*dispatch)(), int protocol) 9657c478bd9Sstevel@tonic-gate { 9667c478bd9Sstevel@tonic-gate struct svc_callout *prev; 9677c478bd9Sstevel@tonic-gate struct svc_callout *s; 9687c478bd9Sstevel@tonic-gate struct netconfig *nconf; 9697c478bd9Sstevel@tonic-gate char *netid = NULL; 9707c478bd9Sstevel@tonic-gate int flag = 0; 9717c478bd9Sstevel@tonic-gate 9727c478bd9Sstevel@tonic-gate if (xprt->xp_netid) { 9737c478bd9Sstevel@tonic-gate netid = strdup(xprt->xp_netid); 9747c478bd9Sstevel@tonic-gate flag = 1; 9757c478bd9Sstevel@tonic-gate } else if ((ioctl(xprt->xp_fd, I_FIND, "timod") > 0) && ((nconf = 9767c478bd9Sstevel@tonic-gate __rpcfd_to_nconf(xprt->xp_fd, xprt->xp_type)) != NULL)) { 9777c478bd9Sstevel@tonic-gate /* fill in missing netid field in SVCXPRT */ 9787c478bd9Sstevel@tonic-gate netid = strdup(nconf->nc_netid); 9797c478bd9Sstevel@tonic-gate flag = 1; 9807c478bd9Sstevel@tonic-gate freenetconfigent(nconf); 9817c478bd9Sstevel@tonic-gate } /* must be svc_raw_create */ 9827c478bd9Sstevel@tonic-gate 98361961e0fSrobinson if ((netid == NULL) && (flag == 1)) 9847c478bd9Sstevel@tonic-gate return (FALSE); 9857c478bd9Sstevel@tonic-gate 98661961e0fSrobinson (void) rw_wrlock(&svc_lock); 9877c478bd9Sstevel@tonic-gate if ((s = svc_find(prog, vers, &prev, netid)) != NULL_SVC) { 9887c478bd9Sstevel@tonic-gate if (netid) 9897c478bd9Sstevel@tonic-gate free(netid); 9907c478bd9Sstevel@tonic-gate if (s->sc_dispatch == dispatch) 9917c478bd9Sstevel@tonic-gate goto pmap_it; /* he is registering another xptr */ 99261961e0fSrobinson (void) rw_unlock(&svc_lock); 9937c478bd9Sstevel@tonic-gate return (FALSE); 9947c478bd9Sstevel@tonic-gate } 99561961e0fSrobinson s = malloc(sizeof (struct svc_callout)); 9967c478bd9Sstevel@tonic-gate if (s == (struct svc_callout *)0) { 9977c478bd9Sstevel@tonic-gate if (netid) 9987c478bd9Sstevel@tonic-gate free(netid); 99961961e0fSrobinson (void) rw_unlock(&svc_lock); 10007c478bd9Sstevel@tonic-gate return (FALSE); 10017c478bd9Sstevel@tonic-gate } 10027c478bd9Sstevel@tonic-gate s->sc_prog = prog; 10037c478bd9Sstevel@tonic-gate s->sc_vers = vers; 10047c478bd9Sstevel@tonic-gate s->sc_dispatch = dispatch; 10057c478bd9Sstevel@tonic-gate s->sc_netid = netid; 10067c478bd9Sstevel@tonic-gate s->sc_next = svc_head; 10077c478bd9Sstevel@tonic-gate svc_head = s; 10087c478bd9Sstevel@tonic-gate 10097c478bd9Sstevel@tonic-gate if ((xprt->xp_netid == NULL) && (flag == 1) && netid) 10107c478bd9Sstevel@tonic-gate if ((xprt->xp_netid = strdup(netid)) == NULL) { 10117c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "svc_register : strdup failed."); 10127c478bd9Sstevel@tonic-gate free(netid); 10137c478bd9Sstevel@tonic-gate svc_head = s->sc_next; 101461961e0fSrobinson free(s); 101561961e0fSrobinson (void) rw_unlock(&svc_lock); 10167c478bd9Sstevel@tonic-gate return (FALSE); 10177c478bd9Sstevel@tonic-gate } 10187c478bd9Sstevel@tonic-gate 10197c478bd9Sstevel@tonic-gate pmap_it: 102061961e0fSrobinson (void) rw_unlock(&svc_lock); 10217c478bd9Sstevel@tonic-gate /* now register the information with the local binder service */ 102261961e0fSrobinson if (protocol) 102361961e0fSrobinson return (pmap_set(prog, vers, protocol, xprt->xp_port)); 10247c478bd9Sstevel@tonic-gate return (TRUE); 10257c478bd9Sstevel@tonic-gate } 10267c478bd9Sstevel@tonic-gate 10277c478bd9Sstevel@tonic-gate /* 10287c478bd9Sstevel@tonic-gate * Remove a service program from the callout list. 10297c478bd9Sstevel@tonic-gate * For version 2 portmappers. 10307c478bd9Sstevel@tonic-gate */ 10317c478bd9Sstevel@tonic-gate void 103261961e0fSrobinson svc_unregister(rpcprog_t prog, rpcvers_t vers) 10337c478bd9Sstevel@tonic-gate { 10347c478bd9Sstevel@tonic-gate struct svc_callout *prev; 10357c478bd9Sstevel@tonic-gate struct svc_callout *s; 10367c478bd9Sstevel@tonic-gate 103761961e0fSrobinson (void) rw_wrlock(&svc_lock); 10387c478bd9Sstevel@tonic-gate while ((s = svc_find(prog, vers, &prev, NULL)) != NULL_SVC) { 10397c478bd9Sstevel@tonic-gate if (prev == NULL_SVC) { 10407c478bd9Sstevel@tonic-gate svc_head = s->sc_next; 10417c478bd9Sstevel@tonic-gate } else { 10427c478bd9Sstevel@tonic-gate prev->sc_next = s->sc_next; 10437c478bd9Sstevel@tonic-gate } 10447c478bd9Sstevel@tonic-gate s->sc_next = NULL_SVC; 10457c478bd9Sstevel@tonic-gate if (s->sc_netid) 104661961e0fSrobinson free(s->sc_netid); 104761961e0fSrobinson free(s); 10487c478bd9Sstevel@tonic-gate /* unregister the information with the local binder service */ 10497c478bd9Sstevel@tonic-gate (void) pmap_unset(prog, vers); 10507c478bd9Sstevel@tonic-gate } 105161961e0fSrobinson (void) rw_unlock(&svc_lock); 10527c478bd9Sstevel@tonic-gate } 10537c478bd9Sstevel@tonic-gate #endif /* PORTMAP */ 105461961e0fSrobinson 10557c478bd9Sstevel@tonic-gate /* 10567c478bd9Sstevel@tonic-gate * Search the callout list for a program number, return the callout 10577c478bd9Sstevel@tonic-gate * struct. 10587c478bd9Sstevel@tonic-gate * Also check for transport as well. Many routines such as svc_unreg 10597c478bd9Sstevel@tonic-gate * dont give any corresponding transport, so dont check for transport if 10607c478bd9Sstevel@tonic-gate * netid == NULL 10617c478bd9Sstevel@tonic-gate */ 10627c478bd9Sstevel@tonic-gate static struct svc_callout * 106361961e0fSrobinson svc_find(rpcprog_t prog, rpcvers_t vers, struct svc_callout **prev, char *netid) 10647c478bd9Sstevel@tonic-gate { 10657c478bd9Sstevel@tonic-gate struct svc_callout *s, *p; 10667c478bd9Sstevel@tonic-gate 10677c478bd9Sstevel@tonic-gate /* WRITE LOCK HELD ON ENTRY: svc_lock */ 10687c478bd9Sstevel@tonic-gate 10697c478bd9Sstevel@tonic-gate /* assert(RW_WRITE_HELD(&svc_lock)); */ 10707c478bd9Sstevel@tonic-gate p = NULL_SVC; 10717c478bd9Sstevel@tonic-gate for (s = svc_head; s != NULL_SVC; s = s->sc_next) { 10727c478bd9Sstevel@tonic-gate if (((s->sc_prog == prog) && (s->sc_vers == vers)) && 10737c478bd9Sstevel@tonic-gate ((netid == NULL) || (s->sc_netid == NULL) || 10747c478bd9Sstevel@tonic-gate (strcmp(netid, s->sc_netid) == 0))) 10757c478bd9Sstevel@tonic-gate break; 10767c478bd9Sstevel@tonic-gate p = s; 10777c478bd9Sstevel@tonic-gate } 10787c478bd9Sstevel@tonic-gate *prev = p; 10797c478bd9Sstevel@tonic-gate return (s); 10807c478bd9Sstevel@tonic-gate } 10817c478bd9Sstevel@tonic-gate 10827c478bd9Sstevel@tonic-gate 10837c478bd9Sstevel@tonic-gate /* ******************* REPLY GENERATION ROUTINES ************ */ 10847c478bd9Sstevel@tonic-gate 10857c478bd9Sstevel@tonic-gate /* 10867c478bd9Sstevel@tonic-gate * Send a reply to an rpc request 10877c478bd9Sstevel@tonic-gate */ 10887c478bd9Sstevel@tonic-gate bool_t 108961961e0fSrobinson svc_sendreply(const SVCXPRT *xprt, const xdrproc_t xdr_results, 109061961e0fSrobinson const caddr_t xdr_location) 10917c478bd9Sstevel@tonic-gate { 10927c478bd9Sstevel@tonic-gate struct rpc_msg rply; 10937c478bd9Sstevel@tonic-gate 10947c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 10957c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 10967c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 10977c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = SUCCESS; 10987c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_results.where = xdr_location; 10997c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_results.proc = xdr_results; 110061961e0fSrobinson return (SVC_REPLY((SVCXPRT *)xprt, &rply)); 11017c478bd9Sstevel@tonic-gate } 11027c478bd9Sstevel@tonic-gate 11037c478bd9Sstevel@tonic-gate /* 11047c478bd9Sstevel@tonic-gate * No procedure error reply 11057c478bd9Sstevel@tonic-gate */ 11067c478bd9Sstevel@tonic-gate void 110761961e0fSrobinson svcerr_noproc(const SVCXPRT *xprt) 11087c478bd9Sstevel@tonic-gate { 11097c478bd9Sstevel@tonic-gate struct rpc_msg rply; 11107c478bd9Sstevel@tonic-gate 11117c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 11127c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 11137c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 11147c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = PROC_UNAVAIL; 11157c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 11167c478bd9Sstevel@tonic-gate } 11177c478bd9Sstevel@tonic-gate 11187c478bd9Sstevel@tonic-gate /* 11197c478bd9Sstevel@tonic-gate * Can't decode args error reply 11207c478bd9Sstevel@tonic-gate */ 11217c478bd9Sstevel@tonic-gate void 112261961e0fSrobinson svcerr_decode(const SVCXPRT *xprt) 11237c478bd9Sstevel@tonic-gate { 11247c478bd9Sstevel@tonic-gate struct rpc_msg rply; 11257c478bd9Sstevel@tonic-gate 11267c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 11277c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 11287c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 11297c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = GARBAGE_ARGS; 11307c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 11317c478bd9Sstevel@tonic-gate } 11327c478bd9Sstevel@tonic-gate 11337c478bd9Sstevel@tonic-gate /* 11347c478bd9Sstevel@tonic-gate * Some system error 11357c478bd9Sstevel@tonic-gate */ 11367c478bd9Sstevel@tonic-gate void 113761961e0fSrobinson svcerr_systemerr(const SVCXPRT *xprt) 11387c478bd9Sstevel@tonic-gate { 11397c478bd9Sstevel@tonic-gate struct rpc_msg rply; 11407c478bd9Sstevel@tonic-gate 11417c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 11427c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 11437c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 11447c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = SYSTEM_ERR; 11457c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 11467c478bd9Sstevel@tonic-gate } 11477c478bd9Sstevel@tonic-gate 11487c478bd9Sstevel@tonic-gate /* 11497c478bd9Sstevel@tonic-gate * Tell RPC package to not complain about version errors to the client. This 11507c478bd9Sstevel@tonic-gate * is useful when revving broadcast protocols that sit on a fixed address. 11517c478bd9Sstevel@tonic-gate * There is really one (or should be only one) example of this kind of 11527c478bd9Sstevel@tonic-gate * protocol: the portmapper (or rpc binder). 11537c478bd9Sstevel@tonic-gate */ 11547c478bd9Sstevel@tonic-gate void 115561961e0fSrobinson __svc_versquiet_on(const SVCXPRT *xprt) 11567c478bd9Sstevel@tonic-gate { 11577c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 11587c478bd9Sstevel@tonic-gate svc_flags(xprt) |= SVC_VERSQUIET; 11597c478bd9Sstevel@tonic-gate } 11607c478bd9Sstevel@tonic-gate 11617c478bd9Sstevel@tonic-gate void 116261961e0fSrobinson __svc_versquiet_off(const SVCXPRT *xprt) 11637c478bd9Sstevel@tonic-gate { 11647c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 11657c478bd9Sstevel@tonic-gate svc_flags(xprt) &= ~SVC_VERSQUIET; 11667c478bd9Sstevel@tonic-gate } 11677c478bd9Sstevel@tonic-gate 11687c478bd9Sstevel@tonic-gate void 116961961e0fSrobinson svc_versquiet(const SVCXPRT *xprt) 11707c478bd9Sstevel@tonic-gate { 11717c478bd9Sstevel@tonic-gate __svc_versquiet_on(xprt); 11727c478bd9Sstevel@tonic-gate } 11737c478bd9Sstevel@tonic-gate 11747c478bd9Sstevel@tonic-gate int 117561961e0fSrobinson __svc_versquiet_get(const SVCXPRT *xprt) 11767c478bd9Sstevel@tonic-gate { 11777c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 11787c478bd9Sstevel@tonic-gate return (svc_flags(xprt) & SVC_VERSQUIET); 11797c478bd9Sstevel@tonic-gate } 11807c478bd9Sstevel@tonic-gate 11817c478bd9Sstevel@tonic-gate /* 11827c478bd9Sstevel@tonic-gate * Authentication error reply 11837c478bd9Sstevel@tonic-gate */ 11847c478bd9Sstevel@tonic-gate void 118561961e0fSrobinson svcerr_auth(const SVCXPRT *xprt, const enum auth_stat why) 11867c478bd9Sstevel@tonic-gate { 11877c478bd9Sstevel@tonic-gate struct rpc_msg rply; 11887c478bd9Sstevel@tonic-gate 11897c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 11907c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_DENIED; 11917c478bd9Sstevel@tonic-gate rply.rjcted_rply.rj_stat = AUTH_ERROR; 11927c478bd9Sstevel@tonic-gate rply.rjcted_rply.rj_why = why; 11937c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 11947c478bd9Sstevel@tonic-gate } 11957c478bd9Sstevel@tonic-gate 11967c478bd9Sstevel@tonic-gate /* 11977c478bd9Sstevel@tonic-gate * Auth too weak error reply 11987c478bd9Sstevel@tonic-gate */ 11997c478bd9Sstevel@tonic-gate void 120061961e0fSrobinson svcerr_weakauth(const SVCXPRT *xprt) 12017c478bd9Sstevel@tonic-gate { 12027c478bd9Sstevel@tonic-gate svcerr_auth(xprt, AUTH_TOOWEAK); 12037c478bd9Sstevel@tonic-gate } 12047c478bd9Sstevel@tonic-gate 12057c478bd9Sstevel@tonic-gate /* 12067c478bd9Sstevel@tonic-gate * Program unavailable error reply 12077c478bd9Sstevel@tonic-gate */ 12087c478bd9Sstevel@tonic-gate void 120961961e0fSrobinson svcerr_noprog(const SVCXPRT *xprt) 12107c478bd9Sstevel@tonic-gate { 12117c478bd9Sstevel@tonic-gate struct rpc_msg rply; 12127c478bd9Sstevel@tonic-gate 12137c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 12147c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 12157c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 12167c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = PROG_UNAVAIL; 12177c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 12187c478bd9Sstevel@tonic-gate } 12197c478bd9Sstevel@tonic-gate 12207c478bd9Sstevel@tonic-gate /* 12217c478bd9Sstevel@tonic-gate * Program version mismatch error reply 12227c478bd9Sstevel@tonic-gate */ 12237c478bd9Sstevel@tonic-gate void 122461961e0fSrobinson svcerr_progvers(const SVCXPRT *xprt, const rpcvers_t low_vers, 122561961e0fSrobinson const rpcvers_t high_vers) 12267c478bd9Sstevel@tonic-gate { 12277c478bd9Sstevel@tonic-gate struct rpc_msg rply; 12287c478bd9Sstevel@tonic-gate 12297c478bd9Sstevel@tonic-gate rply.rm_direction = REPLY; 12307c478bd9Sstevel@tonic-gate rply.rm_reply.rp_stat = MSG_ACCEPTED; 12317c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_verf = xprt->xp_verf; 12327c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_stat = PROG_MISMATCH; 12337c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_vers.low = low_vers; 12347c478bd9Sstevel@tonic-gate rply.acpted_rply.ar_vers.high = high_vers; 12357c478bd9Sstevel@tonic-gate SVC_REPLY((SVCXPRT *)xprt, &rply); 12367c478bd9Sstevel@tonic-gate } 12377c478bd9Sstevel@tonic-gate 12387c478bd9Sstevel@tonic-gate /* ******************* SERVER INPUT STUFF ******************* */ 12397c478bd9Sstevel@tonic-gate 12407c478bd9Sstevel@tonic-gate /* 12417c478bd9Sstevel@tonic-gate * Get server side input from some transport. 12427c478bd9Sstevel@tonic-gate * 12437c478bd9Sstevel@tonic-gate * Statement of authentication parameters management: 12447c478bd9Sstevel@tonic-gate * This function owns and manages all authentication parameters, specifically 12457c478bd9Sstevel@tonic-gate * the "raw" parameters (msg.rm_call.cb_cred and msg.rm_call.cb_verf) and 12467c478bd9Sstevel@tonic-gate * the "cooked" credentials (rqst->rq_clntcred). 12477c478bd9Sstevel@tonic-gate * However, this function does not know the structure of the cooked 12487c478bd9Sstevel@tonic-gate * credentials, so it make the following assumptions: 12497c478bd9Sstevel@tonic-gate * a) the structure is contiguous (no pointers), and 12507c478bd9Sstevel@tonic-gate * b) the cred structure size does not exceed RQCRED_SIZE bytes. 12517c478bd9Sstevel@tonic-gate * In all events, all three parameters are freed upon exit from this routine. 12527c478bd9Sstevel@tonic-gate * The storage is trivially management on the call stack in user land, but 12537c478bd9Sstevel@tonic-gate * is mallocated in kernel land. 12547c478bd9Sstevel@tonic-gate */ 12557c478bd9Sstevel@tonic-gate 12567c478bd9Sstevel@tonic-gate void 125761961e0fSrobinson svc_getreq(int rdfds) 12587c478bd9Sstevel@tonic-gate { 12597c478bd9Sstevel@tonic-gate fd_set readfds; 12607c478bd9Sstevel@tonic-gate 12617c478bd9Sstevel@tonic-gate FD_ZERO(&readfds); 12627c478bd9Sstevel@tonic-gate readfds.fds_bits[0] = rdfds; 12637c478bd9Sstevel@tonic-gate svc_getreqset(&readfds); 12647c478bd9Sstevel@tonic-gate } 12657c478bd9Sstevel@tonic-gate 12667c478bd9Sstevel@tonic-gate void 126761961e0fSrobinson svc_getreqset(fd_set *readfds) 12687c478bd9Sstevel@tonic-gate { 12697c478bd9Sstevel@tonic-gate int i; 12707c478bd9Sstevel@tonic-gate 12717c478bd9Sstevel@tonic-gate for (i = 0; i < svc_max_fd; i++) { 12727c478bd9Sstevel@tonic-gate /* fd has input waiting */ 12737c478bd9Sstevel@tonic-gate if (FD_ISSET(i, readfds)) 12747c478bd9Sstevel@tonic-gate svc_getreq_common(i); 12757c478bd9Sstevel@tonic-gate } 12767c478bd9Sstevel@tonic-gate } 12777c478bd9Sstevel@tonic-gate 12787c478bd9Sstevel@tonic-gate void 127961961e0fSrobinson svc_getreq_poll(struct pollfd *pfdp, const int pollretval) 12807c478bd9Sstevel@tonic-gate { 12817c478bd9Sstevel@tonic-gate int i; 12827c478bd9Sstevel@tonic-gate int fds_found; 12837c478bd9Sstevel@tonic-gate 12847c478bd9Sstevel@tonic-gate for (i = fds_found = 0; fds_found < pollretval; i++) { 12857c478bd9Sstevel@tonic-gate struct pollfd *p = &pfdp[i]; 12867c478bd9Sstevel@tonic-gate 12877c478bd9Sstevel@tonic-gate if (p->revents) { 12887c478bd9Sstevel@tonic-gate /* fd has input waiting */ 12897c478bd9Sstevel@tonic-gate fds_found++; 12907c478bd9Sstevel@tonic-gate /* 12917c478bd9Sstevel@tonic-gate * We assume that this function is only called 12927c478bd9Sstevel@tonic-gate * via someone select()ing from svc_fdset or 12937c478bd9Sstevel@tonic-gate * poll()ing from svc_pollset[]. Thus it's safe 12947c478bd9Sstevel@tonic-gate * to handle the POLLNVAL event by simply turning 12957c478bd9Sstevel@tonic-gate * the corresponding bit off in svc_fdset. The 12967c478bd9Sstevel@tonic-gate * svc_pollset[] array is derived from svc_fdset 12977c478bd9Sstevel@tonic-gate * and so will also be updated eventually. 12987c478bd9Sstevel@tonic-gate * 12997c478bd9Sstevel@tonic-gate * XXX Should we do an xprt_unregister() instead? 13007c478bd9Sstevel@tonic-gate */ 13017c478bd9Sstevel@tonic-gate /* Handle user callback */ 13027c478bd9Sstevel@tonic-gate if (__is_a_userfd(p->fd) == TRUE) { 130361961e0fSrobinson (void) rw_rdlock(&svc_fd_lock); 13047c478bd9Sstevel@tonic-gate __svc_getreq_user(p); 130561961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13067c478bd9Sstevel@tonic-gate } else { 13077c478bd9Sstevel@tonic-gate if (p->revents & POLLNVAL) { 130861961e0fSrobinson (void) rw_wrlock(&svc_fd_lock); 13097c478bd9Sstevel@tonic-gate remove_pollfd(p->fd); /* XXX */ 131061961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 131161961e0fSrobinson } else { 13127c478bd9Sstevel@tonic-gate svc_getreq_common(p->fd); 13137c478bd9Sstevel@tonic-gate } 13147c478bd9Sstevel@tonic-gate } 13157c478bd9Sstevel@tonic-gate } 131661961e0fSrobinson } 13177c478bd9Sstevel@tonic-gate } 13187c478bd9Sstevel@tonic-gate 13197c478bd9Sstevel@tonic-gate void 132061961e0fSrobinson svc_getreq_common(const int fd) 13217c478bd9Sstevel@tonic-gate { 13227c478bd9Sstevel@tonic-gate SVCXPRT *xprt; 13237c478bd9Sstevel@tonic-gate enum xprt_stat stat; 13247c478bd9Sstevel@tonic-gate struct rpc_msg *msg; 13257c478bd9Sstevel@tonic-gate struct svc_req *r; 13267c478bd9Sstevel@tonic-gate char *cred_area; 13277c478bd9Sstevel@tonic-gate 132861961e0fSrobinson (void) rw_rdlock(&svc_fd_lock); 13297c478bd9Sstevel@tonic-gate 13307c478bd9Sstevel@tonic-gate /* HANDLE USER CALLBACK */ 13317c478bd9Sstevel@tonic-gate if (__is_a_userfd(fd) == TRUE) { 13327c478bd9Sstevel@tonic-gate struct pollfd virtual_fd; 13337c478bd9Sstevel@tonic-gate 13347c478bd9Sstevel@tonic-gate virtual_fd.events = virtual_fd.revents = (short)0xFFFF; 13357c478bd9Sstevel@tonic-gate virtual_fd.fd = fd; 13367c478bd9Sstevel@tonic-gate __svc_getreq_user(&virtual_fd); 133761961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13387c478bd9Sstevel@tonic-gate return; 13397c478bd9Sstevel@tonic-gate } 13407c478bd9Sstevel@tonic-gate 13417c478bd9Sstevel@tonic-gate /* 13427c478bd9Sstevel@tonic-gate * The transport associated with this fd could have been 13437c478bd9Sstevel@tonic-gate * removed from svc_timeout_nonblock_xprt_and_LRU, for instance. 13447c478bd9Sstevel@tonic-gate * This can happen if two or more fds get read events and are 13457c478bd9Sstevel@tonic-gate * passed to svc_getreq_poll/set, the first fd is seviced by 13467c478bd9Sstevel@tonic-gate * the dispatch routine and cleans up any dead transports. If 13477c478bd9Sstevel@tonic-gate * one of the dead transports removed is the other fd that 13487c478bd9Sstevel@tonic-gate * had a read event then svc_getreq_common() will be called with no 13497c478bd9Sstevel@tonic-gate * xprt associated with the fd that had the original read event. 13507c478bd9Sstevel@tonic-gate */ 13517c478bd9Sstevel@tonic-gate if ((fd >= nsvc_xports) || (xprt = svc_xports[fd]) == NULL) { 135261961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13537c478bd9Sstevel@tonic-gate return; 13547c478bd9Sstevel@tonic-gate } 135561961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13567c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 13577c478bd9Sstevel@tonic-gate msg = SVCEXT(xprt)->msg; 13587c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 13597c478bd9Sstevel@tonic-gate r = SVCEXT(xprt)->req; 13607c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 13617c478bd9Sstevel@tonic-gate cred_area = SVCEXT(xprt)->cred_area; 13627c478bd9Sstevel@tonic-gate msg->rm_call.cb_cred.oa_base = cred_area; 13637c478bd9Sstevel@tonic-gate msg->rm_call.cb_verf.oa_base = &(cred_area[MAX_AUTH_BYTES]); 13647c478bd9Sstevel@tonic-gate r->rq_clntcred = &(cred_area[2 * MAX_AUTH_BYTES]); 13657c478bd9Sstevel@tonic-gate 13667c478bd9Sstevel@tonic-gate /* receive msgs from xprtprt (support batch calls) */ 13677c478bd9Sstevel@tonic-gate do { 13687c478bd9Sstevel@tonic-gate bool_t dispatch; 13697c478bd9Sstevel@tonic-gate 13707c478bd9Sstevel@tonic-gate if (dispatch = SVC_RECV(xprt, msg)) 13717c478bd9Sstevel@tonic-gate (void) _svc_prog_dispatch(xprt, msg, r); 13727c478bd9Sstevel@tonic-gate /* 13737c478bd9Sstevel@tonic-gate * Check if the xprt has been disconnected in a recursive call 13747c478bd9Sstevel@tonic-gate * in the service dispatch routine. If so, then break 13757c478bd9Sstevel@tonic-gate */ 137661961e0fSrobinson (void) rw_rdlock(&svc_fd_lock); 13777c478bd9Sstevel@tonic-gate if (xprt != svc_xports[fd]) { 137861961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13797c478bd9Sstevel@tonic-gate break; 13807c478bd9Sstevel@tonic-gate } 138161961e0fSrobinson (void) rw_unlock(&svc_fd_lock); 13827c478bd9Sstevel@tonic-gate 13837c478bd9Sstevel@tonic-gate /* 13847c478bd9Sstevel@tonic-gate * Call cleanup procedure if set. 13857c478bd9Sstevel@tonic-gate */ 13867c478bd9Sstevel@tonic-gate if (__proc_cleanup_cb != NULL && dispatch) 13877c478bd9Sstevel@tonic-gate (*__proc_cleanup_cb)(xprt); 13887c478bd9Sstevel@tonic-gate 13897c478bd9Sstevel@tonic-gate if ((stat = SVC_STAT(xprt)) == XPRT_DIED) { 13907c478bd9Sstevel@tonic-gate SVC_DESTROY(xprt); 13917c478bd9Sstevel@tonic-gate break; 13927c478bd9Sstevel@tonic-gate } 13937c478bd9Sstevel@tonic-gate } while (stat == XPRT_MOREREQS); 13947c478bd9Sstevel@tonic-gate } 13957c478bd9Sstevel@tonic-gate 13967c478bd9Sstevel@tonic-gate int 139761961e0fSrobinson _svc_prog_dispatch(SVCXPRT *xprt, struct rpc_msg *msg, struct svc_req *r) 13987c478bd9Sstevel@tonic-gate { 13997c478bd9Sstevel@tonic-gate struct svc_callout *s; 14007c478bd9Sstevel@tonic-gate enum auth_stat why; 14017c478bd9Sstevel@tonic-gate int prog_found; 14027c478bd9Sstevel@tonic-gate rpcvers_t low_vers; 14037c478bd9Sstevel@tonic-gate rpcvers_t high_vers; 14047c478bd9Sstevel@tonic-gate void (*disp_fn)(); 14057c478bd9Sstevel@tonic-gate 14067c478bd9Sstevel@tonic-gate r->rq_xprt = xprt; 14077c478bd9Sstevel@tonic-gate r->rq_prog = msg->rm_call.cb_prog; 14087c478bd9Sstevel@tonic-gate r->rq_vers = msg->rm_call.cb_vers; 14097c478bd9Sstevel@tonic-gate r->rq_proc = msg->rm_call.cb_proc; 14107c478bd9Sstevel@tonic-gate r->rq_cred = msg->rm_call.cb_cred; 14117c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 14127c478bd9Sstevel@tonic-gate SVC_XP_AUTH(r->rq_xprt).svc_ah_ops = svc_auth_any_ops; 14137c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 14147c478bd9Sstevel@tonic-gate SVC_XP_AUTH(r->rq_xprt).svc_ah_private = NULL; 14157c478bd9Sstevel@tonic-gate 14167c478bd9Sstevel@tonic-gate /* first authenticate the message */ 14177c478bd9Sstevel@tonic-gate /* Check for null flavor and bypass these calls if possible */ 14187c478bd9Sstevel@tonic-gate 14197c478bd9Sstevel@tonic-gate if (msg->rm_call.cb_cred.oa_flavor == AUTH_NULL) { 14207c478bd9Sstevel@tonic-gate r->rq_xprt->xp_verf.oa_flavor = _null_auth.oa_flavor; 14217c478bd9Sstevel@tonic-gate r->rq_xprt->xp_verf.oa_length = 0; 14227c478bd9Sstevel@tonic-gate } else { 14237c478bd9Sstevel@tonic-gate bool_t no_dispatch; 14247c478bd9Sstevel@tonic-gate 14257c478bd9Sstevel@tonic-gate if ((why = __gss_authenticate(r, msg, 14267c478bd9Sstevel@tonic-gate &no_dispatch)) != AUTH_OK) { 14277c478bd9Sstevel@tonic-gate svcerr_auth(xprt, why); 14287c478bd9Sstevel@tonic-gate return (0); 14297c478bd9Sstevel@tonic-gate } 14307c478bd9Sstevel@tonic-gate if (no_dispatch) 14317c478bd9Sstevel@tonic-gate return (0); 14327c478bd9Sstevel@tonic-gate } 14337c478bd9Sstevel@tonic-gate /* match message with a registered service */ 14347c478bd9Sstevel@tonic-gate prog_found = FALSE; 14357c478bd9Sstevel@tonic-gate low_vers = (rpcvers_t)(0 - 1); 14367c478bd9Sstevel@tonic-gate high_vers = 0; 143761961e0fSrobinson (void) rw_rdlock(&svc_lock); 14387c478bd9Sstevel@tonic-gate for (s = svc_head; s != NULL_SVC; s = s->sc_next) { 14397c478bd9Sstevel@tonic-gate if (s->sc_prog == r->rq_prog) { 14407c478bd9Sstevel@tonic-gate prog_found = TRUE; 14417c478bd9Sstevel@tonic-gate if (s->sc_vers == r->rq_vers) { 14427c478bd9Sstevel@tonic-gate if ((xprt->xp_netid == NULL) || 14437c478bd9Sstevel@tonic-gate (s->sc_netid == NULL) || 14447c478bd9Sstevel@tonic-gate (strcmp(xprt->xp_netid, 14457c478bd9Sstevel@tonic-gate s->sc_netid) == 0)) { 14467c478bd9Sstevel@tonic-gate disp_fn = (*s->sc_dispatch); 144761961e0fSrobinson (void) rw_unlock(&svc_lock); 14487c478bd9Sstevel@tonic-gate disp_fn(r, xprt); 14497c478bd9Sstevel@tonic-gate return (1); 14507c478bd9Sstevel@tonic-gate } 145161961e0fSrobinson prog_found = FALSE; 14527c478bd9Sstevel@tonic-gate } 14537c478bd9Sstevel@tonic-gate if (s->sc_vers < low_vers) 14547c478bd9Sstevel@tonic-gate low_vers = s->sc_vers; 14557c478bd9Sstevel@tonic-gate if (s->sc_vers > high_vers) 14567c478bd9Sstevel@tonic-gate high_vers = s->sc_vers; 14577c478bd9Sstevel@tonic-gate } /* found correct program */ 14587c478bd9Sstevel@tonic-gate } 145961961e0fSrobinson (void) rw_unlock(&svc_lock); 14607c478bd9Sstevel@tonic-gate 14617c478bd9Sstevel@tonic-gate /* 14627c478bd9Sstevel@tonic-gate * if we got here, the program or version 14637c478bd9Sstevel@tonic-gate * is not served ... 14647c478bd9Sstevel@tonic-gate */ 14657c478bd9Sstevel@tonic-gate if (prog_found) { 14667c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 14677c478bd9Sstevel@tonic-gate if (!version_keepquiet(xprt)) 14687c478bd9Sstevel@tonic-gate svcerr_progvers(xprt, low_vers, high_vers); 14697c478bd9Sstevel@tonic-gate } else { 14707c478bd9Sstevel@tonic-gate svcerr_noprog(xprt); 14717c478bd9Sstevel@tonic-gate } 14727c478bd9Sstevel@tonic-gate return (0); 14737c478bd9Sstevel@tonic-gate } 14747c478bd9Sstevel@tonic-gate 14757c478bd9Sstevel@tonic-gate /* ******************* SVCXPRT allocation and deallocation ***************** */ 14767c478bd9Sstevel@tonic-gate 14777c478bd9Sstevel@tonic-gate /* 14787c478bd9Sstevel@tonic-gate * svc_xprt_alloc() - allocate a service transport handle 14797c478bd9Sstevel@tonic-gate */ 14807c478bd9Sstevel@tonic-gate SVCXPRT * 148161961e0fSrobinson svc_xprt_alloc(void) 14827c478bd9Sstevel@tonic-gate { 14837c478bd9Sstevel@tonic-gate SVCXPRT *xprt = NULL; 14847c478bd9Sstevel@tonic-gate SVCXPRT_EXT *xt = NULL; 14857c478bd9Sstevel@tonic-gate SVCXPRT_LIST *xlist = NULL; 14867c478bd9Sstevel@tonic-gate struct rpc_msg *msg = NULL; 14877c478bd9Sstevel@tonic-gate struct svc_req *req = NULL; 14887c478bd9Sstevel@tonic-gate char *cred_area = NULL; 14897c478bd9Sstevel@tonic-gate 149061961e0fSrobinson if ((xprt = calloc(1, sizeof (SVCXPRT))) == NULL) 14917c478bd9Sstevel@tonic-gate goto err_exit; 14927c478bd9Sstevel@tonic-gate 149361961e0fSrobinson if ((xt = calloc(1, sizeof (SVCXPRT_EXT))) == NULL) 14947c478bd9Sstevel@tonic-gate goto err_exit; 14957c478bd9Sstevel@tonic-gate xprt->xp_p3 = (caddr_t)xt; /* SVCEXT(xprt) = xt */ 14967c478bd9Sstevel@tonic-gate 149761961e0fSrobinson if ((xlist = calloc(1, sizeof (SVCXPRT_LIST))) == NULL) 14987c478bd9Sstevel@tonic-gate goto err_exit; 14997c478bd9Sstevel@tonic-gate xt->my_xlist = xlist; 15007c478bd9Sstevel@tonic-gate xlist->xprt = xprt; 15017c478bd9Sstevel@tonic-gate 150261961e0fSrobinson if ((msg = malloc(sizeof (struct rpc_msg))) == NULL) 15037c478bd9Sstevel@tonic-gate goto err_exit; 15047c478bd9Sstevel@tonic-gate xt->msg = msg; 15057c478bd9Sstevel@tonic-gate 150661961e0fSrobinson if ((req = malloc(sizeof (struct svc_req))) == NULL) 15077c478bd9Sstevel@tonic-gate goto err_exit; 15087c478bd9Sstevel@tonic-gate xt->req = req; 15097c478bd9Sstevel@tonic-gate 151061961e0fSrobinson if ((cred_area = malloc(2*MAX_AUTH_BYTES + RQCRED_SIZE)) == NULL) 15117c478bd9Sstevel@tonic-gate goto err_exit; 15127c478bd9Sstevel@tonic-gate xt->cred_area = cred_area; 15137c478bd9Sstevel@tonic-gate 15147c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 151561961e0fSrobinson (void) mutex_init(&svc_send_mutex(xprt), USYNC_THREAD, (void *)0); 15167c478bd9Sstevel@tonic-gate return (xprt); 15177c478bd9Sstevel@tonic-gate 15187c478bd9Sstevel@tonic-gate err_exit: 15197c478bd9Sstevel@tonic-gate svc_xprt_free(xprt); 15207c478bd9Sstevel@tonic-gate return (NULL); 15217c478bd9Sstevel@tonic-gate } 15227c478bd9Sstevel@tonic-gate 15237c478bd9Sstevel@tonic-gate 15247c478bd9Sstevel@tonic-gate /* 15257c478bd9Sstevel@tonic-gate * svc_xprt_free() - free a service handle 15267c478bd9Sstevel@tonic-gate */ 15277c478bd9Sstevel@tonic-gate void 152861961e0fSrobinson svc_xprt_free(SVCXPRT *xprt) 15297c478bd9Sstevel@tonic-gate { 15307c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15317c478bd9Sstevel@tonic-gate SVCXPRT_EXT *xt = xprt ? SVCEXT(xprt) : NULL; 15327c478bd9Sstevel@tonic-gate SVCXPRT_LIST *my_xlist = xt ? xt->my_xlist: NULL; 15337c478bd9Sstevel@tonic-gate struct rpc_msg *msg = xt ? xt->msg : NULL; 15347c478bd9Sstevel@tonic-gate struct svc_req *req = xt ? xt->req : NULL; 15357c478bd9Sstevel@tonic-gate char *cred_area = xt ? xt->cred_area : NULL; 15367c478bd9Sstevel@tonic-gate 15377c478bd9Sstevel@tonic-gate if (xprt) 153861961e0fSrobinson free(xprt); 15397c478bd9Sstevel@tonic-gate if (xt) 154061961e0fSrobinson free(xt); 15417c478bd9Sstevel@tonic-gate if (my_xlist) 154261961e0fSrobinson free(my_xlist); 15437c478bd9Sstevel@tonic-gate if (msg) 154461961e0fSrobinson free(msg); 15457c478bd9Sstevel@tonic-gate if (req) 154661961e0fSrobinson free(req); 15477c478bd9Sstevel@tonic-gate if (cred_area) 154861961e0fSrobinson free(cred_area); 15497c478bd9Sstevel@tonic-gate } 15507c478bd9Sstevel@tonic-gate 15517c478bd9Sstevel@tonic-gate 15527c478bd9Sstevel@tonic-gate /* 15537c478bd9Sstevel@tonic-gate * svc_xprt_destroy() - free parent and child xprt list 15547c478bd9Sstevel@tonic-gate */ 15557c478bd9Sstevel@tonic-gate void 155661961e0fSrobinson svc_xprt_destroy(SVCXPRT *xprt) 15577c478bd9Sstevel@tonic-gate { 15587c478bd9Sstevel@tonic-gate SVCXPRT_LIST *xlist, *xnext = NULL; 15597c478bd9Sstevel@tonic-gate int type; 15607c478bd9Sstevel@tonic-gate 15617c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15627c478bd9Sstevel@tonic-gate if (SVCEXT(xprt)->parent) 15637c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15647c478bd9Sstevel@tonic-gate xprt = SVCEXT(xprt)->parent; 15657c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15667c478bd9Sstevel@tonic-gate type = svc_type(xprt); 15677c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15687c478bd9Sstevel@tonic-gate for (xlist = SVCEXT(xprt)->my_xlist; xlist != NULL; xlist = xnext) { 15697c478bd9Sstevel@tonic-gate xnext = xlist->next; 15707c478bd9Sstevel@tonic-gate xprt = xlist->xprt; 15717c478bd9Sstevel@tonic-gate switch (type) { 15727c478bd9Sstevel@tonic-gate case SVC_DGRAM: 15737c478bd9Sstevel@tonic-gate svc_dg_xprtfree(xprt); 15747c478bd9Sstevel@tonic-gate break; 15757c478bd9Sstevel@tonic-gate case SVC_RENDEZVOUS: 15767c478bd9Sstevel@tonic-gate svc_vc_xprtfree(xprt); 15777c478bd9Sstevel@tonic-gate break; 15787c478bd9Sstevel@tonic-gate case SVC_CONNECTION: 15797c478bd9Sstevel@tonic-gate svc_fd_xprtfree(xprt); 15807c478bd9Sstevel@tonic-gate break; 15817c478bd9Sstevel@tonic-gate case SVC_DOOR: 15827c478bd9Sstevel@tonic-gate svc_door_xprtfree(xprt); 15837c478bd9Sstevel@tonic-gate break; 15847c478bd9Sstevel@tonic-gate } 15857c478bd9Sstevel@tonic-gate } 15867c478bd9Sstevel@tonic-gate } 15877c478bd9Sstevel@tonic-gate 15887c478bd9Sstevel@tonic-gate 15897c478bd9Sstevel@tonic-gate /* 15907c478bd9Sstevel@tonic-gate * svc_copy() - make a copy of parent 15917c478bd9Sstevel@tonic-gate */ 15927c478bd9Sstevel@tonic-gate SVCXPRT * 159361961e0fSrobinson svc_copy(SVCXPRT *xprt) 15947c478bd9Sstevel@tonic-gate { 15957c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 15967c478bd9Sstevel@tonic-gate switch (svc_type(xprt)) { 15977c478bd9Sstevel@tonic-gate case SVC_DGRAM: 15987c478bd9Sstevel@tonic-gate return (svc_dg_xprtcopy(xprt)); 15997c478bd9Sstevel@tonic-gate case SVC_RENDEZVOUS: 16007c478bd9Sstevel@tonic-gate return (svc_vc_xprtcopy(xprt)); 16017c478bd9Sstevel@tonic-gate case SVC_CONNECTION: 16027c478bd9Sstevel@tonic-gate return (svc_fd_xprtcopy(xprt)); 16037c478bd9Sstevel@tonic-gate } 160461961e0fSrobinson return (NULL); 16057c478bd9Sstevel@tonic-gate } 16067c478bd9Sstevel@tonic-gate 16077c478bd9Sstevel@tonic-gate 16087c478bd9Sstevel@tonic-gate /* 16097c478bd9Sstevel@tonic-gate * _svc_destroy_private() - private SVC_DESTROY interface 16107c478bd9Sstevel@tonic-gate */ 16117c478bd9Sstevel@tonic-gate void 161261961e0fSrobinson _svc_destroy_private(SVCXPRT *xprt) 16137c478bd9Sstevel@tonic-gate { 16147c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 16157c478bd9Sstevel@tonic-gate switch (svc_type(xprt)) { 16167c478bd9Sstevel@tonic-gate case SVC_DGRAM: 16177c478bd9Sstevel@tonic-gate _svc_dg_destroy_private(xprt); 16187c478bd9Sstevel@tonic-gate break; 16197c478bd9Sstevel@tonic-gate case SVC_RENDEZVOUS: 16207c478bd9Sstevel@tonic-gate case SVC_CONNECTION: 16217c478bd9Sstevel@tonic-gate _svc_vc_destroy_private(xprt, TRUE); 16227c478bd9Sstevel@tonic-gate break; 16237c478bd9Sstevel@tonic-gate } 16247c478bd9Sstevel@tonic-gate } 16257c478bd9Sstevel@tonic-gate 16267c478bd9Sstevel@tonic-gate /* 16277c478bd9Sstevel@tonic-gate * svc_get_local_cred() - fetch local user credentials. This always 16287c478bd9Sstevel@tonic-gate * works over doors based transports. For local transports, this 16297c478bd9Sstevel@tonic-gate * does not yield correct results unless the __rpc_negotiate_uid() 16307c478bd9Sstevel@tonic-gate * call has been invoked to enable this feature. 16317c478bd9Sstevel@tonic-gate */ 16327c478bd9Sstevel@tonic-gate bool_t 163361961e0fSrobinson svc_get_local_cred(SVCXPRT *xprt, svc_local_cred_t *lcred) 16347c478bd9Sstevel@tonic-gate { 16357c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 16367c478bd9Sstevel@tonic-gate if (svc_type(xprt) == SVC_DOOR) 16377c478bd9Sstevel@tonic-gate return (__svc_get_door_cred(xprt, lcred)); 16387c478bd9Sstevel@tonic-gate return (__rpc_get_local_cred(xprt, lcred)); 16397c478bd9Sstevel@tonic-gate } 16407c478bd9Sstevel@tonic-gate 16417c478bd9Sstevel@tonic-gate 16427c478bd9Sstevel@tonic-gate /* ******************* DUPLICATE ENTRY HANDLING ROUTINES ************** */ 16437c478bd9Sstevel@tonic-gate 16447c478bd9Sstevel@tonic-gate /* 16457c478bd9Sstevel@tonic-gate * the dup cacheing routines below provide a cache of received 16467c478bd9Sstevel@tonic-gate * transactions. rpc service routines can use this to detect 16477c478bd9Sstevel@tonic-gate * retransmissions and re-send a non-failure response. Uses a 16487c478bd9Sstevel@tonic-gate * lru scheme to find entries to get rid of entries in the cache, 16497c478bd9Sstevel@tonic-gate * though only DUP_DONE entries are placed on the lru list. 16507c478bd9Sstevel@tonic-gate * the routines were written towards development of a generic 16517c478bd9Sstevel@tonic-gate * SVC_DUP() interface, which can be expanded to encompass the 16527c478bd9Sstevel@tonic-gate * svc_dg_enablecache() routines as well. the cache is currently 16537c478bd9Sstevel@tonic-gate * private to the automounter. 16547c478bd9Sstevel@tonic-gate */ 16557c478bd9Sstevel@tonic-gate 16567c478bd9Sstevel@tonic-gate 16577c478bd9Sstevel@tonic-gate /* dupcache header contains xprt specific information */ 165861961e0fSrobinson struct dupcache { 16597c478bd9Sstevel@tonic-gate rwlock_t dc_lock; 16607c478bd9Sstevel@tonic-gate time_t dc_time; 16617c478bd9Sstevel@tonic-gate int dc_buckets; 16627c478bd9Sstevel@tonic-gate int dc_maxsz; 16637c478bd9Sstevel@tonic-gate int dc_basis; 16647c478bd9Sstevel@tonic-gate struct dupreq *dc_mru; 16657c478bd9Sstevel@tonic-gate struct dupreq **dc_hashtbl; 16667c478bd9Sstevel@tonic-gate }; 16677c478bd9Sstevel@tonic-gate 16687c478bd9Sstevel@tonic-gate /* 16697c478bd9Sstevel@tonic-gate * private duplicate cache request routines 16707c478bd9Sstevel@tonic-gate */ 16717c478bd9Sstevel@tonic-gate static int __svc_dupcache_check(struct svc_req *, caddr_t *, uint_t *, 16727c478bd9Sstevel@tonic-gate struct dupcache *, uint32_t, uint32_t); 16737c478bd9Sstevel@tonic-gate static struct dupreq *__svc_dupcache_victim(struct dupcache *, time_t); 16747c478bd9Sstevel@tonic-gate static int __svc_dupcache_enter(struct svc_req *, struct dupreq *, 16757c478bd9Sstevel@tonic-gate struct dupcache *, uint32_t, uint32_t, time_t); 16767c478bd9Sstevel@tonic-gate static int __svc_dupcache_update(struct svc_req *, caddr_t, uint_t, int, 16777c478bd9Sstevel@tonic-gate struct dupcache *, uint32_t, uint32_t); 16787c478bd9Sstevel@tonic-gate #ifdef DUP_DEBUG 16797c478bd9Sstevel@tonic-gate static void __svc_dupcache_debug(struct dupcache *); 16807c478bd9Sstevel@tonic-gate #endif /* DUP_DEBUG */ 16817c478bd9Sstevel@tonic-gate 16827c478bd9Sstevel@tonic-gate /* default parameters for the dupcache */ 16837c478bd9Sstevel@tonic-gate #define DUPCACHE_BUCKETS 257 16847c478bd9Sstevel@tonic-gate #define DUPCACHE_TIME 900 16857c478bd9Sstevel@tonic-gate #define DUPCACHE_MAXSZ INT_MAX 16867c478bd9Sstevel@tonic-gate 16877c478bd9Sstevel@tonic-gate /* 16887c478bd9Sstevel@tonic-gate * __svc_dupcache_init(void *condition, int basis, char *xprt_cache) 16897c478bd9Sstevel@tonic-gate * initialize the duprequest cache and assign it to the xprt_cache 16907c478bd9Sstevel@tonic-gate * Use default values depending on the cache condition and basis. 16917c478bd9Sstevel@tonic-gate * return TRUE on success and FALSE on failure 16927c478bd9Sstevel@tonic-gate */ 16937c478bd9Sstevel@tonic-gate bool_t 16947c478bd9Sstevel@tonic-gate __svc_dupcache_init(void *condition, int basis, char **xprt_cache) 16957c478bd9Sstevel@tonic-gate { 16967c478bd9Sstevel@tonic-gate static mutex_t initdc_lock = DEFAULTMUTEX; 16977c478bd9Sstevel@tonic-gate int i; 16987c478bd9Sstevel@tonic-gate struct dupcache *dc; 16997c478bd9Sstevel@tonic-gate 170061961e0fSrobinson (void) mutex_lock(&initdc_lock); 17017c478bd9Sstevel@tonic-gate if (*xprt_cache != NULL) { /* do only once per xprt */ 170261961e0fSrobinson (void) mutex_unlock(&initdc_lock); 17037c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 17047c478bd9Sstevel@tonic-gate "__svc_dupcache_init: multiply defined dup cache"); 17057c478bd9Sstevel@tonic-gate return (FALSE); 17067c478bd9Sstevel@tonic-gate } 17077c478bd9Sstevel@tonic-gate 17087c478bd9Sstevel@tonic-gate switch (basis) { 17097c478bd9Sstevel@tonic-gate case DUPCACHE_FIXEDTIME: 171061961e0fSrobinson dc = malloc(sizeof (struct dupcache)); 17117c478bd9Sstevel@tonic-gate if (dc == NULL) { 171261961e0fSrobinson (void) mutex_unlock(&initdc_lock); 17137c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 17147c478bd9Sstevel@tonic-gate "__svc_dupcache_init: memory alloc failed"); 17157c478bd9Sstevel@tonic-gate return (FALSE); 17167c478bd9Sstevel@tonic-gate } 171761961e0fSrobinson (void) rwlock_init(&(dc->dc_lock), USYNC_THREAD, NULL); 17187c478bd9Sstevel@tonic-gate if (condition != NULL) 17197c478bd9Sstevel@tonic-gate dc->dc_time = *((time_t *)condition); 17207c478bd9Sstevel@tonic-gate else 17217c478bd9Sstevel@tonic-gate dc->dc_time = DUPCACHE_TIME; 17227c478bd9Sstevel@tonic-gate dc->dc_buckets = DUPCACHE_BUCKETS; 17237c478bd9Sstevel@tonic-gate dc->dc_maxsz = DUPCACHE_MAXSZ; 17247c478bd9Sstevel@tonic-gate dc->dc_basis = basis; 17257c478bd9Sstevel@tonic-gate dc->dc_mru = NULL; 172661961e0fSrobinson dc->dc_hashtbl = malloc(dc->dc_buckets * 17277c478bd9Sstevel@tonic-gate sizeof (struct dupreq *)); 17287c478bd9Sstevel@tonic-gate if (dc->dc_hashtbl == NULL) { 172961961e0fSrobinson free(dc); 173061961e0fSrobinson (void) mutex_unlock(&initdc_lock); 17317c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 17327c478bd9Sstevel@tonic-gate "__svc_dupcache_init: memory alloc failed"); 17337c478bd9Sstevel@tonic-gate return (FALSE); 17347c478bd9Sstevel@tonic-gate } 17357c478bd9Sstevel@tonic-gate for (i = 0; i < DUPCACHE_BUCKETS; i++) 17367c478bd9Sstevel@tonic-gate dc->dc_hashtbl[i] = NULL; 17377c478bd9Sstevel@tonic-gate *xprt_cache = (char *)dc; 17387c478bd9Sstevel@tonic-gate break; 17397c478bd9Sstevel@tonic-gate default: 174061961e0fSrobinson (void) mutex_unlock(&initdc_lock); 17417c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 17427c478bd9Sstevel@tonic-gate "__svc_dupcache_init: undefined dup cache basis"); 17437c478bd9Sstevel@tonic-gate return (FALSE); 17447c478bd9Sstevel@tonic-gate } 17457c478bd9Sstevel@tonic-gate 174661961e0fSrobinson (void) mutex_unlock(&initdc_lock); 17477c478bd9Sstevel@tonic-gate 17487c478bd9Sstevel@tonic-gate return (TRUE); 17497c478bd9Sstevel@tonic-gate } 17507c478bd9Sstevel@tonic-gate 17517c478bd9Sstevel@tonic-gate /* 17527c478bd9Sstevel@tonic-gate * __svc_dup(struct svc_req *req, caddr_t *resp_buf, uint_t *resp_bufsz, 17537c478bd9Sstevel@tonic-gate * char *xprt_cache) 17547c478bd9Sstevel@tonic-gate * searches the request cache. Creates an entry and returns DUP_NEW if 17557c478bd9Sstevel@tonic-gate * the request is not found in the cache. If it is found, then it 17567c478bd9Sstevel@tonic-gate * returns the state of the request (in progress, drop, or done) and 17577c478bd9Sstevel@tonic-gate * also allocates, and passes back results to the user (if any) in 17587c478bd9Sstevel@tonic-gate * resp_buf, and its length in resp_bufsz. DUP_ERROR is returned on error. 17597c478bd9Sstevel@tonic-gate */ 17607c478bd9Sstevel@tonic-gate int 17617c478bd9Sstevel@tonic-gate __svc_dup(struct svc_req *req, caddr_t *resp_buf, uint_t *resp_bufsz, 17627c478bd9Sstevel@tonic-gate char *xprt_cache) 17637c478bd9Sstevel@tonic-gate { 17647c478bd9Sstevel@tonic-gate uint32_t drxid, drhash; 17657c478bd9Sstevel@tonic-gate int rc; 17667c478bd9Sstevel@tonic-gate struct dupreq *dr = NULL; 17677c478bd9Sstevel@tonic-gate time_t timenow = time(NULL); 17687c478bd9Sstevel@tonic-gate 17697c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 17707c478bd9Sstevel@tonic-gate struct dupcache *dc = (struct dupcache *)xprt_cache; 17717c478bd9Sstevel@tonic-gate 17727c478bd9Sstevel@tonic-gate if (dc == NULL) { 17737c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dup: undefined cache"); 17747c478bd9Sstevel@tonic-gate return (DUP_ERROR); 17757c478bd9Sstevel@tonic-gate } 17767c478bd9Sstevel@tonic-gate 17777c478bd9Sstevel@tonic-gate /* get the xid of the request */ 17787c478bd9Sstevel@tonic-gate if (SVC_CONTROL(req->rq_xprt, SVCGET_XID, (void*)&drxid) == FALSE) { 17797c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dup: xid error"); 17807c478bd9Sstevel@tonic-gate return (DUP_ERROR); 17817c478bd9Sstevel@tonic-gate } 17827c478bd9Sstevel@tonic-gate drhash = drxid % dc->dc_buckets; 17837c478bd9Sstevel@tonic-gate 17847c478bd9Sstevel@tonic-gate if ((rc = __svc_dupcache_check(req, resp_buf, resp_bufsz, dc, drxid, 17857c478bd9Sstevel@tonic-gate drhash)) != DUP_NEW) 17867c478bd9Sstevel@tonic-gate return (rc); 17877c478bd9Sstevel@tonic-gate 17887c478bd9Sstevel@tonic-gate if ((dr = __svc_dupcache_victim(dc, timenow)) == NULL) 17897c478bd9Sstevel@tonic-gate return (DUP_ERROR); 17907c478bd9Sstevel@tonic-gate 17917c478bd9Sstevel@tonic-gate if ((rc = __svc_dupcache_enter(req, dr, dc, drxid, drhash, timenow)) 17927c478bd9Sstevel@tonic-gate == DUP_ERROR) 17937c478bd9Sstevel@tonic-gate return (rc); 17947c478bd9Sstevel@tonic-gate 17957c478bd9Sstevel@tonic-gate return (DUP_NEW); 17967c478bd9Sstevel@tonic-gate } 17977c478bd9Sstevel@tonic-gate 17987c478bd9Sstevel@tonic-gate 17997c478bd9Sstevel@tonic-gate 18007c478bd9Sstevel@tonic-gate /* 18017c478bd9Sstevel@tonic-gate * __svc_dupcache_check(struct svc_req *req, caddr_t *resp_buf, 18027c478bd9Sstevel@tonic-gate * uint_t *resp_bufsz,truct dupcache *dc, uint32_t drxid, 18037c478bd9Sstevel@tonic-gate * uint32_t drhash) 18047c478bd9Sstevel@tonic-gate * Checks to see whether an entry already exists in the cache. If it does 18057c478bd9Sstevel@tonic-gate * copy back into the resp_buf, if appropriate. Return the status of 18067c478bd9Sstevel@tonic-gate * the request, or DUP_NEW if the entry is not in the cache 18077c478bd9Sstevel@tonic-gate */ 18087c478bd9Sstevel@tonic-gate static int 18097c478bd9Sstevel@tonic-gate __svc_dupcache_check(struct svc_req *req, caddr_t *resp_buf, uint_t *resp_bufsz, 18107c478bd9Sstevel@tonic-gate struct dupcache *dc, uint32_t drxid, uint32_t drhash) 18117c478bd9Sstevel@tonic-gate { 18127c478bd9Sstevel@tonic-gate struct dupreq *dr = NULL; 18137c478bd9Sstevel@tonic-gate 181461961e0fSrobinson (void) rw_rdlock(&(dc->dc_lock)); 18157c478bd9Sstevel@tonic-gate dr = dc->dc_hashtbl[drhash]; 18167c478bd9Sstevel@tonic-gate while (dr != NULL) { 18177c478bd9Sstevel@tonic-gate if (dr->dr_xid == drxid && 18187c478bd9Sstevel@tonic-gate dr->dr_proc == req->rq_proc && 18197c478bd9Sstevel@tonic-gate dr->dr_prog == req->rq_prog && 18207c478bd9Sstevel@tonic-gate dr->dr_vers == req->rq_vers && 18217c478bd9Sstevel@tonic-gate dr->dr_addr.len == req->rq_xprt->xp_rtaddr.len && 182261961e0fSrobinson memcmp(dr->dr_addr.buf, 182361961e0fSrobinson req->rq_xprt->xp_rtaddr.buf, 18247c478bd9Sstevel@tonic-gate dr->dr_addr.len) == 0) { /* entry found */ 18257c478bd9Sstevel@tonic-gate if (dr->dr_hash != drhash) { 18267c478bd9Sstevel@tonic-gate /* sanity check */ 182761961e0fSrobinson (void) rw_unlock((&dc->dc_lock)); 18287c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 18297c478bd9Sstevel@tonic-gate "\n__svc_dupdone: hashing error"); 18307c478bd9Sstevel@tonic-gate return (DUP_ERROR); 18317c478bd9Sstevel@tonic-gate } 18327c478bd9Sstevel@tonic-gate 18337c478bd9Sstevel@tonic-gate /* 18347c478bd9Sstevel@tonic-gate * return results for requests on lru list, if 18357c478bd9Sstevel@tonic-gate * appropriate requests must be DUP_DROP or DUP_DONE 18367c478bd9Sstevel@tonic-gate * to have a result. A NULL buffer in the cache 18377c478bd9Sstevel@tonic-gate * implies no results were sent during dupdone. 18387c478bd9Sstevel@tonic-gate * A NULL buffer in the call implies not interested 18397c478bd9Sstevel@tonic-gate * in results. 18407c478bd9Sstevel@tonic-gate */ 18417c478bd9Sstevel@tonic-gate if (((dr->dr_status == DUP_DONE) || 18427c478bd9Sstevel@tonic-gate (dr->dr_status == DUP_DROP)) && 18437c478bd9Sstevel@tonic-gate resp_buf != NULL && 18447c478bd9Sstevel@tonic-gate dr->dr_resp.buf != NULL) { 184561961e0fSrobinson *resp_buf = malloc(dr->dr_resp.len); 18467c478bd9Sstevel@tonic-gate if (*resp_buf == NULL) { 18477c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 18487c478bd9Sstevel@tonic-gate "__svc_dupcache_check: malloc failed"); 184961961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 18507c478bd9Sstevel@tonic-gate return (DUP_ERROR); 18517c478bd9Sstevel@tonic-gate } 185261961e0fSrobinson (void) memset(*resp_buf, 0, dr->dr_resp.len); 185361961e0fSrobinson (void) memcpy(*resp_buf, dr->dr_resp.buf, 18547c478bd9Sstevel@tonic-gate dr->dr_resp.len); 18557c478bd9Sstevel@tonic-gate *resp_bufsz = dr->dr_resp.len; 18567c478bd9Sstevel@tonic-gate } else { 18577c478bd9Sstevel@tonic-gate /* no result */ 18587c478bd9Sstevel@tonic-gate if (resp_buf) 18597c478bd9Sstevel@tonic-gate *resp_buf = NULL; 18607c478bd9Sstevel@tonic-gate if (resp_bufsz) 18617c478bd9Sstevel@tonic-gate *resp_bufsz = 0; 18627c478bd9Sstevel@tonic-gate } 186361961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 18647c478bd9Sstevel@tonic-gate return (dr->dr_status); 18657c478bd9Sstevel@tonic-gate } 18667c478bd9Sstevel@tonic-gate dr = dr->dr_chain; 18677c478bd9Sstevel@tonic-gate } 186861961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 18697c478bd9Sstevel@tonic-gate return (DUP_NEW); 18707c478bd9Sstevel@tonic-gate } 18717c478bd9Sstevel@tonic-gate 18727c478bd9Sstevel@tonic-gate /* 18737c478bd9Sstevel@tonic-gate * __svc_dupcache_victim(struct dupcache *dc, time_t timenow) 18747c478bd9Sstevel@tonic-gate * Return a victim dupreq entry to the caller, depending on cache policy. 18757c478bd9Sstevel@tonic-gate */ 18767c478bd9Sstevel@tonic-gate static struct dupreq * 18777c478bd9Sstevel@tonic-gate __svc_dupcache_victim(struct dupcache *dc, time_t timenow) 18787c478bd9Sstevel@tonic-gate { 18797c478bd9Sstevel@tonic-gate struct dupreq *dr = NULL; 18807c478bd9Sstevel@tonic-gate 18817c478bd9Sstevel@tonic-gate switch (dc->dc_basis) { 18827c478bd9Sstevel@tonic-gate case DUPCACHE_FIXEDTIME: 18837c478bd9Sstevel@tonic-gate /* 18847c478bd9Sstevel@tonic-gate * The hash policy is to free up a bit of the hash 18857c478bd9Sstevel@tonic-gate * table before allocating a new entry as the victim. 18867c478bd9Sstevel@tonic-gate * Freeing up the hash table each time should split 18877c478bd9Sstevel@tonic-gate * the cost of keeping the hash table clean among threads. 18887c478bd9Sstevel@tonic-gate * Note that only DONE or DROPPED entries are on the lru 18897c478bd9Sstevel@tonic-gate * list but we do a sanity check anyway. 18907c478bd9Sstevel@tonic-gate */ 189161961e0fSrobinson (void) rw_wrlock(&(dc->dc_lock)); 18927c478bd9Sstevel@tonic-gate while ((dc->dc_mru) && (dr = dc->dc_mru->dr_next) && 18937c478bd9Sstevel@tonic-gate ((timenow - dr->dr_time) > dc->dc_time)) { 18947c478bd9Sstevel@tonic-gate /* clean and then free the entry */ 18957c478bd9Sstevel@tonic-gate if (dr->dr_status != DUP_DONE && 18967c478bd9Sstevel@tonic-gate dr->dr_status != DUP_DROP) { 18977c478bd9Sstevel@tonic-gate /* 18987c478bd9Sstevel@tonic-gate * The LRU list can't contain an 18997c478bd9Sstevel@tonic-gate * entry where the status is other than 19007c478bd9Sstevel@tonic-gate * DUP_DONE or DUP_DROP. 19017c478bd9Sstevel@tonic-gate */ 19027c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 19037c478bd9Sstevel@tonic-gate "__svc_dupcache_victim: bad victim"); 19047c478bd9Sstevel@tonic-gate #ifdef DUP_DEBUG 19057c478bd9Sstevel@tonic-gate /* 19067c478bd9Sstevel@tonic-gate * Need to hold the reader/writers lock to 19077c478bd9Sstevel@tonic-gate * print the cache info, since we already 19087c478bd9Sstevel@tonic-gate * hold the writers lock, we shall continue 19097c478bd9Sstevel@tonic-gate * calling __svc_dupcache_debug() 19107c478bd9Sstevel@tonic-gate */ 19117c478bd9Sstevel@tonic-gate __svc_dupcache_debug(dc); 19127c478bd9Sstevel@tonic-gate #endif /* DUP_DEBUG */ 191361961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 19147c478bd9Sstevel@tonic-gate return (NULL); 19157c478bd9Sstevel@tonic-gate } 19167c478bd9Sstevel@tonic-gate /* free buffers */ 19177c478bd9Sstevel@tonic-gate if (dr->dr_resp.buf) { 191861961e0fSrobinson free(dr->dr_resp.buf); 19197c478bd9Sstevel@tonic-gate dr->dr_resp.buf = NULL; 19207c478bd9Sstevel@tonic-gate } 19217c478bd9Sstevel@tonic-gate if (dr->dr_addr.buf) { 192261961e0fSrobinson free(dr->dr_addr.buf); 19237c478bd9Sstevel@tonic-gate dr->dr_addr.buf = NULL; 19247c478bd9Sstevel@tonic-gate } 19257c478bd9Sstevel@tonic-gate 19267c478bd9Sstevel@tonic-gate /* unhash the entry */ 19277c478bd9Sstevel@tonic-gate if (dr->dr_chain) 19287c478bd9Sstevel@tonic-gate dr->dr_chain->dr_prevchain = dr->dr_prevchain; 19297c478bd9Sstevel@tonic-gate if (dr->dr_prevchain) 19307c478bd9Sstevel@tonic-gate dr->dr_prevchain->dr_chain = dr->dr_chain; 19317c478bd9Sstevel@tonic-gate if (dc->dc_hashtbl[dr->dr_hash] == dr) 19327c478bd9Sstevel@tonic-gate dc->dc_hashtbl[dr->dr_hash] = dr->dr_chain; 19337c478bd9Sstevel@tonic-gate 19347c478bd9Sstevel@tonic-gate /* modify the lru pointers */ 193561961e0fSrobinson if (dc->dc_mru == dr) { 19367c478bd9Sstevel@tonic-gate dc->dc_mru = NULL; 193761961e0fSrobinson } else { 19387c478bd9Sstevel@tonic-gate dc->dc_mru->dr_next = dr->dr_next; 19397c478bd9Sstevel@tonic-gate dr->dr_next->dr_prev = dc->dc_mru; 19407c478bd9Sstevel@tonic-gate } 194161961e0fSrobinson free(dr); 19427c478bd9Sstevel@tonic-gate dr = NULL; 19437c478bd9Sstevel@tonic-gate } 194461961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 19457c478bd9Sstevel@tonic-gate 19467c478bd9Sstevel@tonic-gate /* 19477c478bd9Sstevel@tonic-gate * Allocate and return new clean entry as victim 19487c478bd9Sstevel@tonic-gate */ 194961961e0fSrobinson if ((dr = malloc(sizeof (*dr))) == NULL) { 19507c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 195161961e0fSrobinson "__svc_dupcache_victim: malloc failed"); 19527c478bd9Sstevel@tonic-gate return (NULL); 19537c478bd9Sstevel@tonic-gate } 195461961e0fSrobinson (void) memset(dr, 0, sizeof (*dr)); 19557c478bd9Sstevel@tonic-gate return (dr); 19567c478bd9Sstevel@tonic-gate default: 19577c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 19587c478bd9Sstevel@tonic-gate "__svc_dupcache_victim: undefined dup cache_basis"); 19597c478bd9Sstevel@tonic-gate return (NULL); 19607c478bd9Sstevel@tonic-gate } 19617c478bd9Sstevel@tonic-gate } 19627c478bd9Sstevel@tonic-gate 19637c478bd9Sstevel@tonic-gate /* 19647c478bd9Sstevel@tonic-gate * __svc_dupcache_enter(struct svc_req *req, struct dupreq *dr, 19657c478bd9Sstevel@tonic-gate * struct dupcache *dc, uint32_t drxid, uint32_t drhash, time_t timenow) 19667c478bd9Sstevel@tonic-gate * build new duprequest entry and then insert into the cache 19677c478bd9Sstevel@tonic-gate */ 19687c478bd9Sstevel@tonic-gate static int 19697c478bd9Sstevel@tonic-gate __svc_dupcache_enter(struct svc_req *req, struct dupreq *dr, 19707c478bd9Sstevel@tonic-gate struct dupcache *dc, uint32_t drxid, uint32_t drhash, time_t timenow) 19717c478bd9Sstevel@tonic-gate { 19727c478bd9Sstevel@tonic-gate dr->dr_xid = drxid; 19737c478bd9Sstevel@tonic-gate dr->dr_prog = req->rq_prog; 19747c478bd9Sstevel@tonic-gate dr->dr_vers = req->rq_vers; 19757c478bd9Sstevel@tonic-gate dr->dr_proc = req->rq_proc; 19767c478bd9Sstevel@tonic-gate dr->dr_addr.maxlen = req->rq_xprt->xp_rtaddr.len; 19777c478bd9Sstevel@tonic-gate dr->dr_addr.len = dr->dr_addr.maxlen; 197861961e0fSrobinson if ((dr->dr_addr.buf = malloc(dr->dr_addr.maxlen)) == NULL) { 197961961e0fSrobinson syslog(LOG_ERR, "__svc_dupcache_enter: malloc failed"); 198061961e0fSrobinson free(dr); 19817c478bd9Sstevel@tonic-gate return (DUP_ERROR); 19827c478bd9Sstevel@tonic-gate } 198361961e0fSrobinson (void) memset(dr->dr_addr.buf, 0, dr->dr_addr.len); 198461961e0fSrobinson (void) memcpy(dr->dr_addr.buf, req->rq_xprt->xp_rtaddr.buf, 198561961e0fSrobinson dr->dr_addr.len); 19867c478bd9Sstevel@tonic-gate dr->dr_resp.buf = NULL; 19877c478bd9Sstevel@tonic-gate dr->dr_resp.maxlen = 0; 19887c478bd9Sstevel@tonic-gate dr->dr_resp.len = 0; 19897c478bd9Sstevel@tonic-gate dr->dr_status = DUP_INPROGRESS; 19907c478bd9Sstevel@tonic-gate dr->dr_time = timenow; 19917c478bd9Sstevel@tonic-gate dr->dr_hash = drhash; /* needed for efficient victim cleanup */ 19927c478bd9Sstevel@tonic-gate 19937c478bd9Sstevel@tonic-gate /* place entry at head of hash table */ 199461961e0fSrobinson (void) rw_wrlock(&(dc->dc_lock)); 19957c478bd9Sstevel@tonic-gate dr->dr_chain = dc->dc_hashtbl[drhash]; 19967c478bd9Sstevel@tonic-gate dr->dr_prevchain = NULL; 19977c478bd9Sstevel@tonic-gate if (dc->dc_hashtbl[drhash] != NULL) 19987c478bd9Sstevel@tonic-gate dc->dc_hashtbl[drhash]->dr_prevchain = dr; 19997c478bd9Sstevel@tonic-gate dc->dc_hashtbl[drhash] = dr; 200061961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 20017c478bd9Sstevel@tonic-gate return (DUP_NEW); 20027c478bd9Sstevel@tonic-gate } 20037c478bd9Sstevel@tonic-gate 20047c478bd9Sstevel@tonic-gate /* 20057c478bd9Sstevel@tonic-gate * __svc_dupdone(struct svc_req *req, caddr_t resp_buf, uint_t resp_bufsz, 20067c478bd9Sstevel@tonic-gate * int status, char *xprt_cache) 20077c478bd9Sstevel@tonic-gate * Marks the request done (DUP_DONE or DUP_DROP) and stores the response. 20087c478bd9Sstevel@tonic-gate * Only DONE and DROP requests can be marked as done. Sets the lru pointers 20097c478bd9Sstevel@tonic-gate * to make the entry the most recently used. Returns DUP_ERROR or status. 20107c478bd9Sstevel@tonic-gate */ 20117c478bd9Sstevel@tonic-gate int 20127c478bd9Sstevel@tonic-gate __svc_dupdone(struct svc_req *req, caddr_t resp_buf, uint_t resp_bufsz, 20137c478bd9Sstevel@tonic-gate int status, char *xprt_cache) 20147c478bd9Sstevel@tonic-gate { 20157c478bd9Sstevel@tonic-gate uint32_t drxid, drhash; 20167c478bd9Sstevel@tonic-gate int rc; 20177c478bd9Sstevel@tonic-gate 20187c478bd9Sstevel@tonic-gate /* LINTED pointer alignment */ 20197c478bd9Sstevel@tonic-gate struct dupcache *dc = (struct dupcache *)xprt_cache; 20207c478bd9Sstevel@tonic-gate 20217c478bd9Sstevel@tonic-gate if (dc == NULL) { 20227c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dupdone: undefined cache"); 20237c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20247c478bd9Sstevel@tonic-gate } 20257c478bd9Sstevel@tonic-gate 20267c478bd9Sstevel@tonic-gate if (status != DUP_DONE && status != DUP_DROP) { 20277c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dupdone: invalid dupdone status"); 20287c478bd9Sstevel@tonic-gate syslog(LOG_ERR, " must be DUP_DONE or DUP_DROP"); 20297c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20307c478bd9Sstevel@tonic-gate } 20317c478bd9Sstevel@tonic-gate 20327c478bd9Sstevel@tonic-gate /* find the xid of the entry in the cache */ 20337c478bd9Sstevel@tonic-gate if (SVC_CONTROL(req->rq_xprt, SVCGET_XID, (void*)&drxid) == FALSE) { 20347c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dup: xid error"); 20357c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20367c478bd9Sstevel@tonic-gate } 20377c478bd9Sstevel@tonic-gate drhash = drxid % dc->dc_buckets; 20387c478bd9Sstevel@tonic-gate 20397c478bd9Sstevel@tonic-gate /* update the status of the entry and result buffers, if required */ 20407c478bd9Sstevel@tonic-gate if ((rc = __svc_dupcache_update(req, resp_buf, resp_bufsz, status, 20417c478bd9Sstevel@tonic-gate dc, drxid, drhash)) == DUP_ERROR) { 20427c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dupdone: cache entry error"); 20437c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20447c478bd9Sstevel@tonic-gate } 20457c478bd9Sstevel@tonic-gate 20467c478bd9Sstevel@tonic-gate return (rc); 20477c478bd9Sstevel@tonic-gate } 20487c478bd9Sstevel@tonic-gate 20497c478bd9Sstevel@tonic-gate /* 20507c478bd9Sstevel@tonic-gate * __svc_dupcache_update(struct svc_req *req, caddr_t resp_buf, 20517c478bd9Sstevel@tonic-gate * uint_t resp_bufsz, int status, struct dupcache *dc, uint32_t drxid, 20527c478bd9Sstevel@tonic-gate * uint32_t drhash) 20537c478bd9Sstevel@tonic-gate * Check if entry exists in the dupcacache. If it does, update its status 20547c478bd9Sstevel@tonic-gate * and time and also its buffer, if appropriate. Its possible, but unlikely 20557c478bd9Sstevel@tonic-gate * for DONE requests to not exist in the cache. Return DUP_ERROR or status. 20567c478bd9Sstevel@tonic-gate */ 20577c478bd9Sstevel@tonic-gate static int 20587c478bd9Sstevel@tonic-gate __svc_dupcache_update(struct svc_req *req, caddr_t resp_buf, uint_t resp_bufsz, 20597c478bd9Sstevel@tonic-gate int status, struct dupcache *dc, uint32_t drxid, uint32_t drhash) 20607c478bd9Sstevel@tonic-gate { 20617c478bd9Sstevel@tonic-gate struct dupreq *dr = NULL; 20627c478bd9Sstevel@tonic-gate time_t timenow = time(NULL); 20637c478bd9Sstevel@tonic-gate 206461961e0fSrobinson (void) rw_wrlock(&(dc->dc_lock)); 20657c478bd9Sstevel@tonic-gate dr = dc->dc_hashtbl[drhash]; 20667c478bd9Sstevel@tonic-gate while (dr != NULL) { 20677c478bd9Sstevel@tonic-gate if (dr->dr_xid == drxid && 20687c478bd9Sstevel@tonic-gate dr->dr_proc == req->rq_proc && 20697c478bd9Sstevel@tonic-gate dr->dr_prog == req->rq_prog && 20707c478bd9Sstevel@tonic-gate dr->dr_vers == req->rq_vers && 20717c478bd9Sstevel@tonic-gate dr->dr_addr.len == req->rq_xprt->xp_rtaddr.len && 207261961e0fSrobinson memcmp(dr->dr_addr.buf, 207361961e0fSrobinson req->rq_xprt->xp_rtaddr.buf, 20747c478bd9Sstevel@tonic-gate dr->dr_addr.len) == 0) { /* entry found */ 20757c478bd9Sstevel@tonic-gate if (dr->dr_hash != drhash) { 20767c478bd9Sstevel@tonic-gate /* sanity check */ 207761961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 20787c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 20797c478bd9Sstevel@tonic-gate "\n__svc_dupdone: hashing error"); 20807c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20817c478bd9Sstevel@tonic-gate } 20827c478bd9Sstevel@tonic-gate 20837c478bd9Sstevel@tonic-gate /* store the results if bufer is not NULL */ 20847c478bd9Sstevel@tonic-gate if (resp_buf != NULL) { 208561961e0fSrobinson if ((dr->dr_resp.buf = 208661961e0fSrobinson malloc(resp_bufsz)) == NULL) { 208761961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 20887c478bd9Sstevel@tonic-gate syslog(LOG_ERR, 208961961e0fSrobinson "__svc_dupdone: malloc failed"); 20907c478bd9Sstevel@tonic-gate return (DUP_ERROR); 20917c478bd9Sstevel@tonic-gate } 209261961e0fSrobinson (void) memset(dr->dr_resp.buf, 0, resp_bufsz); 209361961e0fSrobinson (void) memcpy(dr->dr_resp.buf, resp_buf, 20947c478bd9Sstevel@tonic-gate (uint_t)resp_bufsz); 20957c478bd9Sstevel@tonic-gate dr->dr_resp.len = resp_bufsz; 20967c478bd9Sstevel@tonic-gate } 20977c478bd9Sstevel@tonic-gate 20987c478bd9Sstevel@tonic-gate /* update status and done time */ 20997c478bd9Sstevel@tonic-gate dr->dr_status = status; 21007c478bd9Sstevel@tonic-gate dr->dr_time = timenow; 21017c478bd9Sstevel@tonic-gate 21027c478bd9Sstevel@tonic-gate /* move the entry to the mru position */ 21037c478bd9Sstevel@tonic-gate if (dc->dc_mru == NULL) { 21047c478bd9Sstevel@tonic-gate dr->dr_next = dr; 21057c478bd9Sstevel@tonic-gate dr->dr_prev = dr; 21067c478bd9Sstevel@tonic-gate } else { 21077c478bd9Sstevel@tonic-gate dr->dr_next = dc->dc_mru->dr_next; 21087c478bd9Sstevel@tonic-gate dc->dc_mru->dr_next->dr_prev = dr; 21097c478bd9Sstevel@tonic-gate dr->dr_prev = dc->dc_mru; 21107c478bd9Sstevel@tonic-gate dc->dc_mru->dr_next = dr; 21117c478bd9Sstevel@tonic-gate } 21127c478bd9Sstevel@tonic-gate dc->dc_mru = dr; 21137c478bd9Sstevel@tonic-gate 211461961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 21157c478bd9Sstevel@tonic-gate return (status); 21167c478bd9Sstevel@tonic-gate } 21177c478bd9Sstevel@tonic-gate dr = dr->dr_chain; 21187c478bd9Sstevel@tonic-gate } 211961961e0fSrobinson (void) rw_unlock(&(dc->dc_lock)); 21207c478bd9Sstevel@tonic-gate syslog(LOG_ERR, "__svc_dupdone: entry not in dup cache"); 21217c478bd9Sstevel@tonic-gate return (DUP_ERROR); 21227c478bd9Sstevel@tonic-gate } 21237c478bd9Sstevel@tonic-gate 21247c478bd9Sstevel@tonic-gate #ifdef DUP_DEBUG 21257c478bd9Sstevel@tonic-gate /* 21267c478bd9Sstevel@tonic-gate * __svc_dupcache_debug(struct dupcache *dc) 21277c478bd9Sstevel@tonic-gate * print out the hash table stuff 21287c478bd9Sstevel@tonic-gate * 21297c478bd9Sstevel@tonic-gate * This function requires the caller to hold the reader 21307c478bd9Sstevel@tonic-gate * or writer version of the duplicate request cache lock (dc_lock). 21317c478bd9Sstevel@tonic-gate */ 21327c478bd9Sstevel@tonic-gate static void 21337c478bd9Sstevel@tonic-gate __svc_dupcache_debug(struct dupcache *dc) 21347c478bd9Sstevel@tonic-gate { 21357c478bd9Sstevel@tonic-gate struct dupreq *dr = NULL; 21367c478bd9Sstevel@tonic-gate int i; 21377c478bd9Sstevel@tonic-gate bool_t bval; 21387c478bd9Sstevel@tonic-gate 21397c478bd9Sstevel@tonic-gate fprintf(stderr, " HASHTABLE\n"); 21407c478bd9Sstevel@tonic-gate for (i = 0; i < dc->dc_buckets; i++) { 21417c478bd9Sstevel@tonic-gate bval = FALSE; 21427c478bd9Sstevel@tonic-gate dr = dc->dc_hashtbl[i]; 21437c478bd9Sstevel@tonic-gate while (dr != NULL) { 21447c478bd9Sstevel@tonic-gate if (!bval) { /* ensures bucket printed only once */ 21457c478bd9Sstevel@tonic-gate fprintf(stderr, " bucket : %d\n", i); 21467c478bd9Sstevel@tonic-gate bval = TRUE; 21477c478bd9Sstevel@tonic-gate } 21487c478bd9Sstevel@tonic-gate fprintf(stderr, "\txid: %u status: %d time: %ld", 21497c478bd9Sstevel@tonic-gate dr->dr_xid, dr->dr_status, dr->dr_time); 21507c478bd9Sstevel@tonic-gate fprintf(stderr, " dr: %x chain: %x prevchain: %x\n", 21517c478bd9Sstevel@tonic-gate dr, dr->dr_chain, dr->dr_prevchain); 21527c478bd9Sstevel@tonic-gate dr = dr->dr_chain; 21537c478bd9Sstevel@tonic-gate } 21547c478bd9Sstevel@tonic-gate } 21557c478bd9Sstevel@tonic-gate 21567c478bd9Sstevel@tonic-gate fprintf(stderr, " LRU\n"); 21577c478bd9Sstevel@tonic-gate if (dc->dc_mru) { 21587c478bd9Sstevel@tonic-gate dr = dc->dc_mru->dr_next; /* lru */ 21597c478bd9Sstevel@tonic-gate while (dr != dc->dc_mru) { 21607c478bd9Sstevel@tonic-gate fprintf(stderr, "\txid: %u status : %d time : %ld", 21617c478bd9Sstevel@tonic-gate dr->dr_xid, dr->dr_status, dr->dr_time); 21627c478bd9Sstevel@tonic-gate fprintf(stderr, " dr: %x next: %x prev: %x\n", 21637c478bd9Sstevel@tonic-gate dr, dr->dr_next, dr->dr_prev); 21647c478bd9Sstevel@tonic-gate dr = dr->dr_next; 21657c478bd9Sstevel@tonic-gate } 21667c478bd9Sstevel@tonic-gate fprintf(stderr, "\txid: %u status: %d time: %ld", 21677c478bd9Sstevel@tonic-gate dr->dr_xid, dr->dr_status, dr->dr_time); 21687c478bd9Sstevel@tonic-gate fprintf(stderr, " dr: %x next: %x prev: %x\n", dr, 21697c478bd9Sstevel@tonic-gate dr->dr_next, dr->dr_prev); 21707c478bd9Sstevel@tonic-gate } 21717c478bd9Sstevel@tonic-gate } 21727c478bd9Sstevel@tonic-gate #endif /* DUP_DEBUG */ 2173