xref: /freebsd/stand/libsa/rpc.c (revision 56e53cb8ef000c3ef72337a4095987a932cdedef)
1ca987d46SWarner Losh /*	$NetBSD: rpc.c,v 1.18 1998/01/23 19:27:45 thorpej Exp $	*/
2ca987d46SWarner Losh 
3ca987d46SWarner Losh /*
4ca987d46SWarner Losh  * Copyright (c) 1992 Regents of the University of California.
5ca987d46SWarner Losh  * All rights reserved.
6ca987d46SWarner Losh  *
7ca987d46SWarner Losh  * This software was developed by the Computer Systems Engineering group
8ca987d46SWarner Losh  * at Lawrence Berkeley Laboratory under DARPA contract BG 91-66 and
9ca987d46SWarner Losh  * contributed to Berkeley.
10ca987d46SWarner Losh  *
11ca987d46SWarner Losh  * Redistribution and use in source and binary forms, with or without
12ca987d46SWarner Losh  * modification, are permitted provided that the following conditions
13ca987d46SWarner Losh  * are met:
14ca987d46SWarner Losh  * 1. Redistributions of source code must retain the above copyright
15ca987d46SWarner Losh  *    notice, this list of conditions and the following disclaimer.
16ca987d46SWarner Losh  * 2. Redistributions in binary form must reproduce the above copyright
17ca987d46SWarner Losh  *    notice, this list of conditions and the following disclaimer in the
18ca987d46SWarner Losh  *    documentation and/or other materials provided with the distribution.
19ca987d46SWarner Losh  * 3. Neither the name of the University nor the names of its contributors
20ca987d46SWarner Losh  *    may be used to endorse or promote products derived from this software
21ca987d46SWarner Losh  *    without specific prior written permission.
22ca987d46SWarner Losh  *
23ca987d46SWarner Losh  * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
24ca987d46SWarner Losh  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
25ca987d46SWarner Losh  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
26ca987d46SWarner Losh  * ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
27ca987d46SWarner Losh  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
28ca987d46SWarner Losh  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
29ca987d46SWarner Losh  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
30ca987d46SWarner Losh  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
31ca987d46SWarner Losh  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
32ca987d46SWarner Losh  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
33ca987d46SWarner Losh  * SUCH DAMAGE.
34ca987d46SWarner Losh  *
35ca987d46SWarner Losh  * @(#) Header: rpc.c,v 1.12 93/09/28 08:31:56 leres Exp  (LBL)
36ca987d46SWarner Losh  */
37ca987d46SWarner Losh 
38ca987d46SWarner Losh #include <sys/cdefs.h>
39ca987d46SWarner Losh __FBSDID("$FreeBSD$");
40ca987d46SWarner Losh 
41ca987d46SWarner Losh /*
42ca987d46SWarner Losh  * RPC functions used by NFS and bootparams.
43ca987d46SWarner Losh  * Note that bootparams requires the ability to find out the
44ca987d46SWarner Losh  * address of the server from which its response has come.
45ca987d46SWarner Losh  * This is supported by keeping the IP/UDP headers in the
46ca987d46SWarner Losh  * buffer space provided by the caller.  (See rpc_fromaddr)
47ca987d46SWarner Losh  */
48ca987d46SWarner Losh 
49ca987d46SWarner Losh #include <sys/param.h>
50ca987d46SWarner Losh #include <sys/socket.h>
51ca987d46SWarner Losh 
52ca987d46SWarner Losh #include <netinet/in.h>
53ca987d46SWarner Losh #include <netinet/in_systm.h>
54ca987d46SWarner Losh 
55ca987d46SWarner Losh #include <string.h>
56ca987d46SWarner Losh 
57ca987d46SWarner Losh #include "rpcv2.h"
58ca987d46SWarner Losh 
59ca987d46SWarner Losh #include "stand.h"
60ca987d46SWarner Losh #include "net.h"
61ca987d46SWarner Losh #include "netif.h"
62ca987d46SWarner Losh #include "rpc.h"
63ca987d46SWarner Losh 
64ca987d46SWarner Losh struct auth_info {
65ca987d46SWarner Losh 	int32_t 	authtype;	/* auth type */
66*56e53cb8SWarner Losh 	uint32_t	authlen;	/* auth length */
67ca987d46SWarner Losh };
68ca987d46SWarner Losh 
69ca987d46SWarner Losh struct auth_unix {
70ca987d46SWarner Losh 	int32_t   ua_time;
71ca987d46SWarner Losh 	int32_t   ua_hostname;	/* null */
72ca987d46SWarner Losh 	int32_t   ua_uid;
73ca987d46SWarner Losh 	int32_t   ua_gid;
74ca987d46SWarner Losh 	int32_t   ua_gidlist;	/* null */
75ca987d46SWarner Losh };
76ca987d46SWarner Losh 
77ca987d46SWarner Losh struct rpc_call {
78*56e53cb8SWarner Losh 	uint32_t	rp_xid;		/* request transaction id */
79ca987d46SWarner Losh 	int32_t 	rp_direction;	/* call direction (0) */
80*56e53cb8SWarner Losh 	uint32_t	rp_rpcvers;	/* rpc version (2) */
81*56e53cb8SWarner Losh 	uint32_t	rp_prog;	/* program */
82*56e53cb8SWarner Losh 	uint32_t	rp_vers;	/* version */
83*56e53cb8SWarner Losh 	uint32_t	rp_proc;	/* procedure */
84ca987d46SWarner Losh };
85ca987d46SWarner Losh 
86ca987d46SWarner Losh struct rpc_reply {
87*56e53cb8SWarner Losh 	uint32_t	rp_xid;		/* request transaction id */
88ca987d46SWarner Losh 	int32_t 	rp_direction;	/* call direction (1) */
89ca987d46SWarner Losh 	int32_t 	rp_astatus;	/* accept status (0: accepted) */
90ca987d46SWarner Losh 	union {
91*56e53cb8SWarner Losh 		uint32_t	rpu_errno;
92ca987d46SWarner Losh 		struct {
93ca987d46SWarner Losh 			struct auth_info rok_auth;
94*56e53cb8SWarner Losh 			uint32_t	rok_status;
95ca987d46SWarner Losh 		} rpu_rok;
96ca987d46SWarner Losh 	} rp_u;
97ca987d46SWarner Losh };
98ca987d46SWarner Losh 
99ca987d46SWarner Losh /* Local forwards */
100c5b86c3bSKyle Evans static	ssize_t recvrpc(struct iodesc *, void **, void **, time_t, void *);
101ca987d46SWarner Losh static	int rpc_getport(struct iodesc *, n_long, n_long);
102ca987d46SWarner Losh 
103ca987d46SWarner Losh int rpc_xid;
104ca987d46SWarner Losh int rpc_port = 0x400;	/* predecrement */
105ca987d46SWarner Losh 
106ca987d46SWarner Losh /*
107ca987d46SWarner Losh  * Make a rpc call; return length of answer
108ca987d46SWarner Losh  * Note: Caller must leave room for headers.
109ca987d46SWarner Losh  */
110ca987d46SWarner Losh ssize_t
111ca987d46SWarner Losh rpc_call(struct iodesc *d, n_long prog, n_long vers, n_long proc,
112ca987d46SWarner Losh 	void *sdata, size_t slen, void **rdata, void **pkt)
113ca987d46SWarner Losh {
114ca987d46SWarner Losh 	ssize_t cc, rsize;
115ca987d46SWarner Losh 	struct auth_info *auth;
116ca987d46SWarner Losh 	struct rpc_call *call;
117ca987d46SWarner Losh 	struct rpc_reply *reply;
118ca987d46SWarner Losh 	char *send_head, *send_tail;
119ca987d46SWarner Losh 	void *ptr;
120ca987d46SWarner Losh 	n_long x;
121ca987d46SWarner Losh 	int port;	/* host order */
122ca987d46SWarner Losh 
123ca987d46SWarner Losh #ifdef RPC_DEBUG
124ca987d46SWarner Losh 	if (debug)
125ca987d46SWarner Losh 		printf("rpc_call: prog=0x%x vers=%d proc=%d\n",
126ca987d46SWarner Losh 		    prog, vers, proc);
127ca987d46SWarner Losh #endif
128ca987d46SWarner Losh 
129ca987d46SWarner Losh 	port = rpc_getport(d, prog, vers);
130ca987d46SWarner Losh 	if (port == -1)
131ca987d46SWarner Losh 		return (-1);
132ca987d46SWarner Losh 
133ca987d46SWarner Losh 	d->destport = htons(port);
134ca987d46SWarner Losh 
135ca987d46SWarner Losh 	/*
136ca987d46SWarner Losh 	 * Prepend authorization stuff and headers.
137ca987d46SWarner Losh 	 * Note, must prepend things in reverse order.
138ca987d46SWarner Losh 	 */
139ca987d46SWarner Losh 	send_head = sdata;
140ca987d46SWarner Losh 	send_tail = (char *)sdata + slen;
141ca987d46SWarner Losh 
142ca987d46SWarner Losh 	/* Auth verifier is always auth_null */
143ca987d46SWarner Losh 	send_head -= sizeof(*auth);
144ca987d46SWarner Losh 	auth = (struct auth_info *)send_head;
145ca987d46SWarner Losh 	auth->authtype = htonl(RPCAUTH_NULL);
146ca987d46SWarner Losh 	auth->authlen = 0;
147ca987d46SWarner Losh 
148ca987d46SWarner Losh 	/* Auth credentials: always auth unix (as root) */
149ca987d46SWarner Losh 	send_head -= sizeof(struct auth_unix);
150ca987d46SWarner Losh 	bzero(send_head, sizeof(struct auth_unix));
151ca987d46SWarner Losh 	send_head -= sizeof(*auth);
152ca987d46SWarner Losh 	auth = (struct auth_info *)send_head;
153ca987d46SWarner Losh 	auth->authtype = htonl(RPCAUTH_UNIX);
154ca987d46SWarner Losh 	auth->authlen = htonl(sizeof(struct auth_unix));
155ca987d46SWarner Losh 
156ca987d46SWarner Losh 	/* RPC call structure. */
157ca987d46SWarner Losh 	send_head -= sizeof(*call);
158ca987d46SWarner Losh 	call = (struct rpc_call *)send_head;
159ca987d46SWarner Losh 	rpc_xid++;
160ca987d46SWarner Losh 	call->rp_xid       = htonl(rpc_xid);
161ca987d46SWarner Losh 	call->rp_direction = htonl(RPC_CALL);
162ca987d46SWarner Losh 	call->rp_rpcvers   = htonl(RPC_VER2);
163ca987d46SWarner Losh 	call->rp_prog = htonl(prog);
164ca987d46SWarner Losh 	call->rp_vers = htonl(vers);
165ca987d46SWarner Losh 	call->rp_proc = htonl(proc);
166ca987d46SWarner Losh 
167ca987d46SWarner Losh 	ptr = NULL;
168ca987d46SWarner Losh 	cc = sendrecv(d,
169ca987d46SWarner Losh 	    sendudp, send_head, send_tail - send_head,
170c5b86c3bSKyle Evans 	    recvrpc, &ptr, (void **)&reply, NULL);
171ca987d46SWarner Losh 
172ca987d46SWarner Losh #ifdef RPC_DEBUG
173ca987d46SWarner Losh 	if (debug)
174ca987d46SWarner Losh 		printf("callrpc: cc=%zd\n", cc);
175ca987d46SWarner Losh #endif
176ca987d46SWarner Losh 	if (cc == -1)
177ca987d46SWarner Losh 		return (-1);
178ca987d46SWarner Losh 
179ca987d46SWarner Losh 	if (cc <= sizeof(*reply)) {
180ca987d46SWarner Losh 		errno = EBADRPC;
181ca987d46SWarner Losh 		free(ptr);
182ca987d46SWarner Losh 		return (-1);
183ca987d46SWarner Losh 	}
184ca987d46SWarner Losh 
185ca987d46SWarner Losh 	/*
186ca987d46SWarner Losh 	 * Check the RPC reply status.
187ca987d46SWarner Losh 	 * The xid, dir, astatus were already checked.
188ca987d46SWarner Losh 	 */
189ca987d46SWarner Losh 	auth = &reply->rp_u.rpu_rok.rok_auth;
190ca987d46SWarner Losh 	x = ntohl(auth->authlen);
191ca987d46SWarner Losh 	if (x != 0) {
192ca987d46SWarner Losh #ifdef RPC_DEBUG
193ca987d46SWarner Losh 		if (debug)
194ca987d46SWarner Losh 			printf("callrpc: reply auth != NULL\n");
195ca987d46SWarner Losh #endif
196ca987d46SWarner Losh 		errno = EBADRPC;
197ca987d46SWarner Losh 		free(ptr);
198ca987d46SWarner Losh 		return (-1);
199ca987d46SWarner Losh 	}
200ca987d46SWarner Losh 	x = ntohl(reply->rp_u.rpu_rok.rok_status);
201ca987d46SWarner Losh 	if (x != 0) {
202ca987d46SWarner Losh 		printf("callrpc: error = %ld\n", (long)x);
203ca987d46SWarner Losh 		errno = EBADRPC;
204ca987d46SWarner Losh 		free(ptr);
205ca987d46SWarner Losh 		return (-1);
206ca987d46SWarner Losh 	}
207ca987d46SWarner Losh 
208ca987d46SWarner Losh 	rsize = cc - sizeof(*reply);
209ca987d46SWarner Losh 	*rdata = (void *)((uintptr_t)reply + sizeof(*reply));
210ca987d46SWarner Losh 	*pkt = ptr;
211ca987d46SWarner Losh 	return (rsize);
212ca987d46SWarner Losh }
213ca987d46SWarner Losh 
214ca987d46SWarner Losh /*
215ca987d46SWarner Losh  * Returns true if packet is the one we're waiting for.
216ca987d46SWarner Losh  * This just checks the XID, direction, acceptance.
217ca987d46SWarner Losh  * Remaining checks are done by callrpc
218ca987d46SWarner Losh  */
219ca987d46SWarner Losh static ssize_t
220c5b86c3bSKyle Evans recvrpc(struct iodesc *d, void **pkt, void **payload, time_t tleft, void *extra)
221ca987d46SWarner Losh {
222ca987d46SWarner Losh 	void *ptr;
223ca987d46SWarner Losh 	struct rpc_reply *reply;
224ca987d46SWarner Losh 	ssize_t	n;
225ca987d46SWarner Losh 	int	x;
226ca987d46SWarner Losh 
227ca987d46SWarner Losh 	errno = 0;
228ca987d46SWarner Losh #ifdef RPC_DEBUG
229ca987d46SWarner Losh 	if (debug)
230ca987d46SWarner Losh 		printf("recvrpc: called\n");
231ca987d46SWarner Losh #endif
232ca987d46SWarner Losh 
233ca987d46SWarner Losh 	ptr = NULL;
234ca987d46SWarner Losh 	n = readudp(d, &ptr, (void **)&reply, tleft);
235ca987d46SWarner Losh 	if (n <= (4 * 4)) {
236ca987d46SWarner Losh 		free(ptr);
237ca987d46SWarner Losh 		return (-1);
238ca987d46SWarner Losh 	}
239ca987d46SWarner Losh 
240ca987d46SWarner Losh 	x = ntohl(reply->rp_xid);
241ca987d46SWarner Losh 	if (x != rpc_xid) {
242ca987d46SWarner Losh #ifdef RPC_DEBUG
243ca987d46SWarner Losh 		if (debug)
244ca987d46SWarner Losh 			printf("recvrpc: rp_xid %d != xid %d\n", x, rpc_xid);
245ca987d46SWarner Losh #endif
246ca987d46SWarner Losh 		free(ptr);
247ca987d46SWarner Losh 		return (-1);
248ca987d46SWarner Losh 	}
249ca987d46SWarner Losh 
250ca987d46SWarner Losh 	x = ntohl(reply->rp_direction);
251ca987d46SWarner Losh 	if (x != RPC_REPLY) {
252ca987d46SWarner Losh #ifdef RPC_DEBUG
253ca987d46SWarner Losh 		if (debug)
254ca987d46SWarner Losh 			printf("recvrpc: rp_direction %d != REPLY\n", x);
255ca987d46SWarner Losh #endif
256ca987d46SWarner Losh 		free(ptr);
257ca987d46SWarner Losh 		return (-1);
258ca987d46SWarner Losh 	}
259ca987d46SWarner Losh 
260ca987d46SWarner Losh 	x = ntohl(reply->rp_astatus);
261ca987d46SWarner Losh 	if (x != RPC_MSGACCEPTED) {
262ca987d46SWarner Losh 		errno = ntohl(reply->rp_u.rpu_errno);
263ca987d46SWarner Losh 		printf("recvrpc: reject, astat=%d, errno=%d\n", x, errno);
264ca987d46SWarner Losh 		free(ptr);
265ca987d46SWarner Losh 		return (-1);
266ca987d46SWarner Losh 	}
267ca987d46SWarner Losh 
268ca987d46SWarner Losh 	*pkt = ptr;
269ca987d46SWarner Losh 	*payload = reply;
270ca987d46SWarner Losh 	/* Return data count (thus indicating success) */
271ca987d46SWarner Losh 	return (n);
272ca987d46SWarner Losh }
273ca987d46SWarner Losh 
274ca987d46SWarner Losh /*
275ca987d46SWarner Losh  * Given a pointer to a reply just received,
276ca987d46SWarner Losh  * dig out the IP address/port from the headers.
277ca987d46SWarner Losh  */
278ca987d46SWarner Losh void
279ca987d46SWarner Losh rpc_fromaddr(void *pkt, struct in_addr *addr, u_short *port)
280ca987d46SWarner Losh {
281ca987d46SWarner Losh 	struct hackhdr {
282ca987d46SWarner Losh 		/* Tail of IP header: just IP addresses */
283ca987d46SWarner Losh 		n_long ip_src;
284ca987d46SWarner Losh 		n_long ip_dst;
285ca987d46SWarner Losh 		/* UDP header: */
286*56e53cb8SWarner Losh 		uint16_t uh_sport;		/* source port */
287*56e53cb8SWarner Losh 		uint16_t uh_dport;		/* destination port */
288ca987d46SWarner Losh 		int16_t	  uh_ulen;		/* udp length */
289*56e53cb8SWarner Losh 		uint16_t uh_sum;		/* udp checksum */
290ca987d46SWarner Losh 		/* RPC reply header: */
291ca987d46SWarner Losh 		struct rpc_reply rpc;
292ca987d46SWarner Losh 	} *hhdr;
293ca987d46SWarner Losh 
294ca987d46SWarner Losh 	hhdr = ((struct hackhdr *)pkt) - 1;
295ca987d46SWarner Losh 	addr->s_addr = hhdr->ip_src;
296ca987d46SWarner Losh 	*port = hhdr->uh_sport;
297ca987d46SWarner Losh }
298ca987d46SWarner Losh 
299ca987d46SWarner Losh /*
300ca987d46SWarner Losh  * RPC Portmapper cache
301ca987d46SWarner Losh  */
302ca987d46SWarner Losh #define PMAP_NUM 8			/* need at most 5 pmap entries */
303ca987d46SWarner Losh 
304ca987d46SWarner Losh int rpc_pmap_num;
305ca987d46SWarner Losh struct pmap_list {
306ca987d46SWarner Losh 	struct in_addr	addr;	/* server, net order */
307ca987d46SWarner Losh 	u_int	prog;		/* host order */
308ca987d46SWarner Losh 	u_int	vers;		/* host order */
309ca987d46SWarner Losh 	int 	port;		/* host order */
310ca987d46SWarner Losh } rpc_pmap_list[PMAP_NUM];
311ca987d46SWarner Losh 
312ca987d46SWarner Losh /*
313ca987d46SWarner Losh  * return port number in host order, or -1.
314ca987d46SWarner Losh  * arguments are:
315ca987d46SWarner Losh  *  addr .. server, net order.
316ca987d46SWarner Losh  *  prog .. host order.
317ca987d46SWarner Losh  *  vers .. host order.
318ca987d46SWarner Losh  */
319ca987d46SWarner Losh int
320ca987d46SWarner Losh rpc_pmap_getcache(struct in_addr addr, u_int prog, u_int vers)
321ca987d46SWarner Losh {
322ca987d46SWarner Losh 	struct pmap_list *pl;
323ca987d46SWarner Losh 
324ca987d46SWarner Losh 	for (pl = rpc_pmap_list; pl < &rpc_pmap_list[rpc_pmap_num]; pl++) {
325ca987d46SWarner Losh 		if (pl->addr.s_addr == addr.s_addr &&
326ca987d46SWarner Losh 			pl->prog == prog && pl->vers == vers )
327ca987d46SWarner Losh 		{
328ca987d46SWarner Losh 			return (pl->port);
329ca987d46SWarner Losh 		}
330ca987d46SWarner Losh 	}
331ca987d46SWarner Losh 	return (-1);
332ca987d46SWarner Losh }
333ca987d46SWarner Losh 
334ca987d46SWarner Losh /*
335ca987d46SWarner Losh  * arguments are:
336ca987d46SWarner Losh  *  addr .. server, net order.
337ca987d46SWarner Losh  *  prog .. host order.
338ca987d46SWarner Losh  *  vers .. host order.
339ca987d46SWarner Losh  *  port .. host order.
340ca987d46SWarner Losh  */
341ca987d46SWarner Losh void
342ca987d46SWarner Losh rpc_pmap_putcache(struct in_addr addr, u_int prog, u_int vers, int port)
343ca987d46SWarner Losh {
344ca987d46SWarner Losh 	struct pmap_list *pl;
345ca987d46SWarner Losh 
346ca987d46SWarner Losh 	/* Don't overflow cache... */
347ca987d46SWarner Losh 	if (rpc_pmap_num >= PMAP_NUM) {
348ca987d46SWarner Losh 		/* ... just re-use the last entry. */
349ca987d46SWarner Losh 		rpc_pmap_num = PMAP_NUM - 1;
350ca987d46SWarner Losh #ifdef	RPC_DEBUG
351ca987d46SWarner Losh 		printf("rpc_pmap_putcache: cache overflow\n");
352ca987d46SWarner Losh #endif
353ca987d46SWarner Losh 	}
354ca987d46SWarner Losh 
355ca987d46SWarner Losh 	pl = &rpc_pmap_list[rpc_pmap_num];
356ca987d46SWarner Losh 	rpc_pmap_num++;
357ca987d46SWarner Losh 
358ca987d46SWarner Losh 	/* Cache answer */
359ca987d46SWarner Losh 	pl->addr = addr;
360ca987d46SWarner Losh 	pl->prog = prog;
361ca987d46SWarner Losh 	pl->vers = vers;
362ca987d46SWarner Losh 	pl->port = port;
363ca987d46SWarner Losh }
364ca987d46SWarner Losh 
365ca987d46SWarner Losh 
366ca987d46SWarner Losh /*
367ca987d46SWarner Losh  * Request a port number from the port mapper.
368ca987d46SWarner Losh  * Returns the port in host order.
369ca987d46SWarner Losh  * prog and vers are host order.
370ca987d46SWarner Losh  */
371ca987d46SWarner Losh int
372ca987d46SWarner Losh rpc_getport(struct iodesc *d, n_long prog, n_long vers)
373ca987d46SWarner Losh {
374ca987d46SWarner Losh 	struct args {
375ca987d46SWarner Losh 		n_long	prog;		/* call program */
376ca987d46SWarner Losh 		n_long	vers;		/* call version */
377ca987d46SWarner Losh 		n_long	proto;		/* call protocol */
378ca987d46SWarner Losh 		n_long	port;		/* call port (unused) */
379ca987d46SWarner Losh 	} *args;
380ca987d46SWarner Losh 	struct res {
381ca987d46SWarner Losh 		n_long port;
382ca987d46SWarner Losh 	} *res;
383ca987d46SWarner Losh 	struct {
384ca987d46SWarner Losh 		n_long	h[RPC_HEADER_WORDS];
385ca987d46SWarner Losh 		struct args d;
386ca987d46SWarner Losh 	} sdata;
387ca987d46SWarner Losh 	void *pkt;
388ca987d46SWarner Losh 	ssize_t cc;
389ca987d46SWarner Losh 	int port;
390ca987d46SWarner Losh 
391ca987d46SWarner Losh #ifdef RPC_DEBUG
392ca987d46SWarner Losh 	if (debug)
393ca987d46SWarner Losh 		printf("%s: prog=0x%x vers=%d\n", __func__, prog, vers);
394ca987d46SWarner Losh #endif
395ca987d46SWarner Losh 
396ca987d46SWarner Losh 	/* This one is fixed forever. */
397ca987d46SWarner Losh 	if (prog == PMAPPROG) {
398ca987d46SWarner Losh 		port = PMAPPORT;
399ca987d46SWarner Losh 		goto out;
400ca987d46SWarner Losh 	}
401ca987d46SWarner Losh 
402ca987d46SWarner Losh 	/* Try for cached answer first */
403ca987d46SWarner Losh 	port = rpc_pmap_getcache(d->destip, prog, vers);
404ca987d46SWarner Losh 	if (port != -1)
405ca987d46SWarner Losh 		goto out;
406ca987d46SWarner Losh 
407ca987d46SWarner Losh 	args = &sdata.d;
408ca987d46SWarner Losh 	args->prog = htonl(prog);
409ca987d46SWarner Losh 	args->vers = htonl(vers);
410ca987d46SWarner Losh 	args->proto = htonl(IPPROTO_UDP);
411ca987d46SWarner Losh 	args->port = 0;
412ca987d46SWarner Losh 	pkt = NULL;
413ca987d46SWarner Losh 
414ca987d46SWarner Losh 	cc = rpc_call(d, PMAPPROG, PMAPVERS, PMAPPROC_GETPORT,
415ca987d46SWarner Losh 	    args, sizeof(*args), (void **)&res, &pkt);
416ca987d46SWarner Losh 	if (cc < sizeof(*res)) {
417ca987d46SWarner Losh 		printf("getport: %s", strerror(errno));
418ca987d46SWarner Losh 		errno = EBADRPC;
419ca987d46SWarner Losh 		free(pkt);
420ca987d46SWarner Losh 		return (-1);
421ca987d46SWarner Losh 	}
422ca987d46SWarner Losh 	port = (int)ntohl(res->port);
423ca987d46SWarner Losh 	free(pkt);
424ca987d46SWarner Losh 
425ca987d46SWarner Losh 	rpc_pmap_putcache(d->destip, prog, vers, port);
426ca987d46SWarner Losh 
427ca987d46SWarner Losh out:
428ca987d46SWarner Losh #ifdef RPC_DEBUG
429ca987d46SWarner Losh 	if (debug)
430ca987d46SWarner Losh 		printf("%s: port=%u\n", __func__, port);
431ca987d46SWarner Losh #endif
432ca987d46SWarner Losh 	return (port);
433ca987d46SWarner Losh }
434