|  | /* | 
|  | * clnt.h - Client side remote procedure call interface. | 
|  | * | 
|  | * Copyright (c) 2010, Oracle America, Inc. | 
|  | * | 
|  | * Redistribution and use in source and binary forms, with or without | 
|  | * modification, are permitted provided that the following conditions are | 
|  | * met: | 
|  | * | 
|  | *     * Redistributions of source code must retain the above copyright | 
|  | *       notice, this list of conditions and the following disclaimer. | 
|  | *     * Redistributions in binary form must reproduce the above | 
|  | *       copyright notice, this list of conditions and the following | 
|  | *       disclaimer in the documentation and/or other materials | 
|  | *       provided with the distribution. | 
|  | *     * Neither the name of the "Oracle America, Inc." nor the names of its | 
|  | *       contributors may be used to endorse or promote products derived | 
|  | *       from this software without specific prior written permission. | 
|  | * | 
|  | *   THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | 
|  | *   "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | 
|  | *   LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS | 
|  | *   FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE | 
|  | *   COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, | 
|  | *   INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL | 
|  | *   DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE | 
|  | *   GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS | 
|  | *   INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, | 
|  | *   WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING | 
|  | *   NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE | 
|  | *   OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | 
|  | */ | 
|  |  | 
|  | #ifndef _RPC_CLNT_H | 
|  | #define _RPC_CLNT_H	1 | 
|  |  | 
|  | #include <features.h> | 
|  | #include <sys/types.h> | 
|  | #include <rpc/types.h> | 
|  | #include <rpc/auth.h> | 
|  | #include <sys/un.h> | 
|  |  | 
|  | __BEGIN_DECLS | 
|  |  | 
|  | /* | 
|  | * Rpc calls return an enum clnt_stat.  This should be looked at more, | 
|  | * since each implementation is required to live with this (implementation | 
|  | * independent) list of errors. | 
|  | */ | 
|  | enum clnt_stat { | 
|  | RPC_SUCCESS=0,			/* call succeeded */ | 
|  | /* | 
|  | * local errors | 
|  | */ | 
|  | RPC_CANTENCODEARGS=1,		/* can't encode arguments */ | 
|  | RPC_CANTDECODERES=2,		/* can't decode results */ | 
|  | RPC_CANTSEND=3,			/* failure in sending call */ | 
|  | RPC_CANTRECV=4,			/* failure in receiving result */ | 
|  | RPC_TIMEDOUT=5,			/* call timed out */ | 
|  | /* | 
|  | * remote errors | 
|  | */ | 
|  | RPC_VERSMISMATCH=6,		/* rpc versions not compatible */ | 
|  | RPC_AUTHERROR=7,		/* authentication error */ | 
|  | RPC_PROGUNAVAIL=8,		/* program not available */ | 
|  | RPC_PROGVERSMISMATCH=9,		/* program version mismatched */ | 
|  | RPC_PROCUNAVAIL=10,		/* procedure unavailable */ | 
|  | RPC_CANTDECODEARGS=11,		/* decode arguments error */ | 
|  | RPC_SYSTEMERROR=12,		/* generic "other problem" */ | 
|  | RPC_NOBROADCAST = 21,		/* Broadcasting not supported */ | 
|  | /* | 
|  | * callrpc & clnt_create errors | 
|  | */ | 
|  | RPC_UNKNOWNHOST=13,		/* unknown host name */ | 
|  | RPC_UNKNOWNPROTO=17,		/* unknown protocol */ | 
|  | RPC_UNKNOWNADDR = 19,		/* Remote address unknown */ | 
|  |  | 
|  | /* | 
|  | * rpcbind errors | 
|  | */ | 
|  | RPC_RPCBFAILURE=14,		/* portmapper failed in its call */ | 
|  | #define RPC_PMAPFAILURE RPC_RPCBFAILURE | 
|  | RPC_PROGNOTREGISTERED=15,	/* remote program is not registered */ | 
|  | RPC_N2AXLATEFAILURE = 22,	/* Name to addr translation failed */ | 
|  | /* | 
|  | * unspecified error | 
|  | */ | 
|  | RPC_FAILED=16, | 
|  | RPC_INTR=18, | 
|  | RPC_TLIERROR=20, | 
|  | RPC_UDERROR=23, | 
|  | /* | 
|  | * asynchronous errors | 
|  | */ | 
|  | RPC_INPROGRESS = 24, | 
|  | RPC_STALERACHANDLE = 25 | 
|  | }; | 
|  |  | 
|  |  | 
|  | /* | 
|  | * Error info. | 
|  | */ | 
|  | struct rpc_err { | 
|  | enum clnt_stat re_status; | 
|  | union { | 
|  | int RE_errno;		/* related system error */ | 
|  | enum auth_stat RE_why;	/* why the auth error occurred */ | 
|  | struct { | 
|  | u_long low;		/* lowest verion supported */ | 
|  | u_long high;		/* highest verion supported */ | 
|  | } RE_vers; | 
|  | struct {			/* maybe meaningful if RPC_FAILED */ | 
|  | long s1; | 
|  | long s2; | 
|  | } RE_lb;			/* life boot & debugging only */ | 
|  | } ru; | 
|  | #define	re_errno	ru.RE_errno | 
|  | #define	re_why		ru.RE_why | 
|  | #define	re_vers		ru.RE_vers | 
|  | #define	re_lb		ru.RE_lb | 
|  | }; | 
|  |  | 
|  |  | 
|  | /* | 
|  | * Client rpc handle. | 
|  | * Created by individual implementations, see e.g. rpc_udp.c. | 
|  | * Client is responsible for initializing auth, see e.g. auth_none.c. | 
|  | */ | 
|  | typedef struct CLIENT CLIENT; | 
|  | struct CLIENT { | 
|  | AUTH	*cl_auth;		 /* authenticator */ | 
|  | struct clnt_ops { | 
|  | enum clnt_stat (*cl_call) (CLIENT *, u_long, xdrproc_t, caddr_t, xdrproc_t, | 
|  | caddr_t, struct timeval); | 
|  | /* call remote procedure */ | 
|  | void (*cl_abort) (void);	/* abort a call */ | 
|  | void (*cl_geterr) (CLIENT *, struct rpc_err *); | 
|  | /* get specific error code */ | 
|  | bool_t (*cl_freeres) (CLIENT *, xdrproc_t, caddr_t); | 
|  | /* frees results */ | 
|  | void (*cl_destroy) (CLIENT *); /* destroy this structure */ | 
|  | bool_t (*cl_control) (CLIENT *, int, char *); | 
|  | /* the ioctl() of rpc */ | 
|  | } *cl_ops; | 
|  | caddr_t cl_private;		/* private stuff */ | 
|  | }; | 
|  |  | 
|  |  | 
|  | /* | 
|  | * client side rpc interface ops | 
|  | * | 
|  | * Parameter types are: | 
|  | * | 
|  | */ | 
|  |  | 
|  | /* | 
|  | * enum clnt_stat | 
|  | * CLNT_CALL(rh, proc, xargs, argsp, xres, resp, timeout) | 
|  | * 	CLIENT *rh; | 
|  | *	u_long proc; | 
|  | *	xdrproc_t xargs; | 
|  | *	caddr_t argsp; | 
|  | *	xdrproc_t xres; | 
|  | *	caddr_t resp; | 
|  | *	struct timeval timeout; | 
|  | */ | 
|  | #define	CLNT_CALL(rh, proc, xargs, argsp, xres, resp, secs)	\ | 
|  | ((*(rh)->cl_ops->cl_call)(rh, proc, xargs, argsp, xres, resp, secs)) | 
|  | #define	clnt_call(rh, proc, xargs, argsp, xres, resp, secs)	\ | 
|  | ((*(rh)->cl_ops->cl_call)(rh, proc, xargs, argsp, xres, resp, secs)) | 
|  |  | 
|  | /* | 
|  | * void | 
|  | * CLNT_ABORT(rh); | 
|  | * 	CLIENT *rh; | 
|  | */ | 
|  | #define	CLNT_ABORT(rh)	((*(rh)->cl_ops->cl_abort)(rh)) | 
|  | #define	clnt_abort(rh)	((*(rh)->cl_ops->cl_abort)(rh)) | 
|  |  | 
|  | /* | 
|  | * struct rpc_err | 
|  | * CLNT_GETERR(rh); | 
|  | * 	CLIENT *rh; | 
|  | */ | 
|  | #define	CLNT_GETERR(rh,errp)	((*(rh)->cl_ops->cl_geterr)(rh, errp)) | 
|  | #define	clnt_geterr(rh,errp)	((*(rh)->cl_ops->cl_geterr)(rh, errp)) | 
|  |  | 
|  |  | 
|  | /* | 
|  | * bool_t | 
|  | * CLNT_FREERES(rh, xres, resp); | 
|  | * 	CLIENT *rh; | 
|  | *	xdrproc_t xres; | 
|  | *	caddr_t resp; | 
|  | */ | 
|  | #define	CLNT_FREERES(rh,xres,resp) ((*(rh)->cl_ops->cl_freeres)(rh,xres,resp)) | 
|  | #define	clnt_freeres(rh,xres,resp) ((*(rh)->cl_ops->cl_freeres)(rh,xres,resp)) | 
|  |  | 
|  | /* | 
|  | * bool_t | 
|  | * CLNT_CONTROL(cl, request, info) | 
|  | *      CLIENT *cl; | 
|  | *      u_int request; | 
|  | *      char *info; | 
|  | */ | 
|  | #define	CLNT_CONTROL(cl,rq,in) ((*(cl)->cl_ops->cl_control)(cl,rq,in)) | 
|  | #define	clnt_control(cl,rq,in) ((*(cl)->cl_ops->cl_control)(cl,rq,in)) | 
|  |  | 
|  | /* | 
|  | * control operations that apply to all transports | 
|  | * | 
|  | * Note: options marked XXX are no-ops in this implementation of RPC. | 
|  | * The are present in TI-RPC but can't be implemented here since they | 
|  | * depend on the presence of STREAMS/TLI, which we don't have. | 
|  | */ | 
|  | #define CLSET_TIMEOUT        1    /* set timeout (timeval) */ | 
|  | #define CLGET_TIMEOUT        2    /* get timeout (timeval) */ | 
|  | #define CLGET_SERVER_ADDR    3    /* get server's address (sockaddr) */ | 
|  | #define CLGET_FD             6    /* get connections file descriptor */ | 
|  | #define CLGET_SVC_ADDR       7    /* get server's address (netbuf)      XXX */ | 
|  | #define CLSET_FD_CLOSE       8    /* close fd while clnt_destroy */ | 
|  | #define CLSET_FD_NCLOSE      9    /* Do not close fd while clnt_destroy*/ | 
|  | #define CLGET_XID            10   /* Get xid */ | 
|  | #define CLSET_XID            11   /* Set xid */ | 
|  | #define CLGET_VERS           12   /* Get version number */ | 
|  | #define CLSET_VERS           13   /* Set version number */ | 
|  | #define CLGET_PROG           14   /* Get program number */ | 
|  | #define CLSET_PROG           15   /* Set program number */ | 
|  | #define CLSET_SVC_ADDR       16   /* get server's address (netbuf)      XXX */ | 
|  | #define CLSET_PUSH_TIMOD     17   /* push timod if not already present  XXX */ | 
|  | #define CLSET_POP_TIMOD      18   /* pop timod                          XXX */ | 
|  | /* | 
|  | * Connectionless only control operations | 
|  | */ | 
|  | #define CLSET_RETRY_TIMEOUT	4	/* set retry timeout (timeval) */ | 
|  | #define CLGET_RETRY_TIMEOUT	5	/* get retry timeout (timeval) */ | 
|  |  | 
|  | /* | 
|  | * void | 
|  | * CLNT_DESTROY(rh); | 
|  | * 	CLIENT *rh; | 
|  | */ | 
|  | #define	CLNT_DESTROY(rh)	((*(rh)->cl_ops->cl_destroy)(rh)) | 
|  | #define	clnt_destroy(rh)	((*(rh)->cl_ops->cl_destroy)(rh)) | 
|  |  | 
|  |  | 
|  | /* | 
|  | * RPCTEST is a test program which is accessible on every rpc | 
|  | * transport/port.  It is used for testing, performance evaluation, | 
|  | * and network administration. | 
|  | */ | 
|  |  | 
|  | #define RPCTEST_PROGRAM		((u_long)1) | 
|  | #define RPCTEST_VERSION		((u_long)1) | 
|  | #define RPCTEST_NULL_PROC	((u_long)2) | 
|  | #define RPCTEST_NULL_BATCH_PROC	((u_long)3) | 
|  |  | 
|  | /* | 
|  | * By convention, procedure 0 takes null arguments and returns them | 
|  | */ | 
|  |  | 
|  | #define NULLPROC ((u_long)0) | 
|  |  | 
|  | /* | 
|  | * Below are the client handle creation routines for the various | 
|  | * implementations of client side rpc.  They can return NULL if a | 
|  | * creation failure occurs. | 
|  | */ | 
|  |  | 
|  | /* | 
|  | * Memory based rpc (for speed check and testing) | 
|  | * CLIENT * | 
|  | * clntraw_create(prog, vers) | 
|  | *	u_long prog; | 
|  | *	u_long vers; | 
|  | */ | 
|  | extern CLIENT *clntraw_create (const u_long __prog, const u_long __vers) | 
|  | __THROW; | 
|  |  | 
|  |  | 
|  | /* | 
|  | * Generic client creation routine. Supported protocols are "udp", "tcp" and | 
|  | * "unix" | 
|  | * CLIENT * | 
|  | * clnt_create(host, prog, vers, prot) | 
|  | *	char *host; 	-- hostname | 
|  | *	u_long prog;	-- program number | 
|  | *	u_ong vers;	-- version number | 
|  | *	char *prot;	-- protocol | 
|  | */ | 
|  | extern CLIENT *clnt_create (const char *__host, const u_long __prog, | 
|  | const u_long __vers, const char *__prot) | 
|  | __THROW; | 
|  |  | 
|  |  | 
|  | /* | 
|  | * TCP based rpc | 
|  | * CLIENT * | 
|  | * clnttcp_create(raddr, prog, vers, sockp, sendsz, recvsz) | 
|  | *	struct sockaddr_in *raddr; | 
|  | *	u_long prog; | 
|  | *	u_long version; | 
|  | *	register int *sockp; | 
|  | *	u_int sendsz; | 
|  | *	u_int recvsz; | 
|  | */ | 
|  | extern CLIENT *clnttcp_create (struct sockaddr_in *__raddr, u_long __prog, | 
|  | u_long __version, int *__sockp, u_int __sendsz, | 
|  | u_int __recvsz) __THROW; | 
|  |  | 
|  | /* | 
|  | * UDP based rpc. | 
|  | * CLIENT * | 
|  | * clntudp_create(raddr, program, version, wait, sockp) | 
|  | *	struct sockaddr_in *raddr; | 
|  | *	u_long program; | 
|  | *	u_long version; | 
|  | *	struct timeval wait_resend; | 
|  | *	int *sockp; | 
|  | * | 
|  | * Same as above, but you specify max packet sizes. | 
|  | * CLIENT * | 
|  | * clntudp_bufcreate(raddr, program, version, wait, sockp, sendsz, recvsz) | 
|  | *	struct sockaddr_in *raddr; | 
|  | *	u_long program; | 
|  | *	u_long version; | 
|  | *	struct timeval wait_resend; | 
|  | *	int *sockp; | 
|  | *	u_int sendsz; | 
|  | *	u_int recvsz; | 
|  | */ | 
|  | extern CLIENT *clntudp_create (struct sockaddr_in *__raddr, u_long __program, | 
|  | u_long __version, struct timeval __wait_resend, | 
|  | int *__sockp) __THROW; | 
|  | extern CLIENT *clntudp_bufcreate (struct sockaddr_in *__raddr, | 
|  | u_long __program, u_long __version, | 
|  | struct timeval __wait_resend, int *__sockp, | 
|  | u_int __sendsz, u_int __recvsz) __THROW; | 
|  |  | 
|  |  | 
|  |  | 
|  |  | 
|  | /* | 
|  | * AF_UNIX based rpc | 
|  | * CLIENT * | 
|  | * clntunix_create(raddr, prog, vers, sockp, sendsz, recvsz) | 
|  | *      struct sockaddr_un *raddr; | 
|  | *      u_long prog; | 
|  | *      u_long version; | 
|  | *      register int *sockp; | 
|  | *      u_int sendsz; | 
|  | *      u_int recvsz; | 
|  | */ | 
|  | extern CLIENT *clntunix_create  (struct sockaddr_un *__raddr, u_long __program, | 
|  | u_long __version, int *__sockp, | 
|  | u_int __sendsz, u_int __recvsz) __THROW; | 
|  |  | 
|  |  | 
|  | extern int callrpc (const char *__host, const u_long __prognum, | 
|  | const u_long __versnum, const u_long __procnum, | 
|  | const xdrproc_t __inproc, const char *__in, | 
|  | const xdrproc_t __outproc, char *__out) __THROW; | 
|  | extern int _rpc_dtablesize (void) __THROW; | 
|  |  | 
|  | /* | 
|  | * Print why creation failed | 
|  | */ | 
|  | extern void clnt_pcreateerror (const char *__msg);	/* stderr */ | 
|  | extern char *clnt_spcreateerror(const char *__msg) __THROW;	/* string */ | 
|  |  | 
|  | /* | 
|  | * Like clnt_perror(), but is more verbose in its output | 
|  | */ | 
|  | extern void clnt_perrno (enum clnt_stat __num);		/* stderr */ | 
|  |  | 
|  | /* | 
|  | * Print an English error message, given the client error code | 
|  | */ | 
|  | extern void clnt_perror (CLIENT *__clnt, const char *__msg); | 
|  | /* stderr */ | 
|  | extern char *clnt_sperror (CLIENT *__clnt, const char *__msg) __THROW; | 
|  | /* string */ | 
|  |  | 
|  | /* | 
|  | * If a creation fails, the following allows the user to figure out why. | 
|  | */ | 
|  | struct rpc_createerr { | 
|  | enum clnt_stat cf_stat; | 
|  | struct rpc_err cf_error; /* useful when cf_stat == RPC_PMAPFAILURE */ | 
|  | }; | 
|  |  | 
|  | extern struct rpc_createerr rpc_createerr; | 
|  |  | 
|  |  | 
|  |  | 
|  | /* | 
|  | * Copy error message to buffer. | 
|  | */ | 
|  | extern char *clnt_sperrno (enum clnt_stat __num) __THROW;	/* string */ | 
|  |  | 
|  | /* | 
|  | * get the port number on the host for the rpc program,version and proto | 
|  | */ | 
|  | extern int getrpcport (const char * __host, u_long __prognum, | 
|  | u_long __versnum, u_int __proto) __THROW; | 
|  |  | 
|  | /* | 
|  | * get the local host's IP address without consulting | 
|  | * name service library functions | 
|  | */ | 
|  | extern void get_myaddress (struct sockaddr_in *) __THROW; | 
|  |  | 
|  | #define UDPMSGSIZE	8800	/* rpc imposed limit on udp msg size */ | 
|  | #define RPCSMALLMSGSIZE	400	/* a more reasonable packet size */ | 
|  |  | 
|  | __END_DECLS | 
|  |  | 
|  | #endif /* rpc/clnt.h */ |