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
36ca987d46SWarner Losh /*
37ca987d46SWarner Losh * RPC functions used by NFS and bootparams.
38ca987d46SWarner Losh * Note that bootparams requires the ability to find out the
39ca987d46SWarner Losh * address of the server from which its response has come.
40ca987d46SWarner Losh * This is supported by keeping the IP/UDP headers in the
41ca987d46SWarner Losh * buffer space provided by the caller. (See rpc_fromaddr)
42ca987d46SWarner Losh */
43ca987d46SWarner Losh
44ca987d46SWarner Losh #include <sys/param.h>
45ca987d46SWarner Losh #include <sys/socket.h>
46ca987d46SWarner Losh
47ca987d46SWarner Losh #include <netinet/in.h>
48ca987d46SWarner Losh #include <netinet/in_systm.h>
49ca987d46SWarner Losh
50ca987d46SWarner Losh #include <string.h>
51ca987d46SWarner Losh
52ca987d46SWarner Losh #include "rpcv2.h"
53ca987d46SWarner Losh
54ca987d46SWarner Losh #include "stand.h"
55ca987d46SWarner Losh #include "net.h"
56ca987d46SWarner Losh #include "netif.h"
57ca987d46SWarner Losh #include "rpc.h"
58ca987d46SWarner Losh
59ca987d46SWarner Losh struct auth_info {
60ca987d46SWarner Losh int32_t authtype; /* auth type */
61*56e53cb8SWarner Losh uint32_t authlen; /* auth length */
62ca987d46SWarner Losh };
63ca987d46SWarner Losh
64ca987d46SWarner Losh struct auth_unix {
65ca987d46SWarner Losh int32_t ua_time;
66ca987d46SWarner Losh int32_t ua_hostname; /* null */
67ca987d46SWarner Losh int32_t ua_uid;
68ca987d46SWarner Losh int32_t ua_gid;
69ca987d46SWarner Losh int32_t ua_gidlist; /* null */
70ca987d46SWarner Losh };
71ca987d46SWarner Losh
72ca987d46SWarner Losh struct rpc_call {
73*56e53cb8SWarner Losh uint32_t rp_xid; /* request transaction id */
74ca987d46SWarner Losh int32_t rp_direction; /* call direction (0) */
75*56e53cb8SWarner Losh uint32_t rp_rpcvers; /* rpc version (2) */
76*56e53cb8SWarner Losh uint32_t rp_prog; /* program */
77*56e53cb8SWarner Losh uint32_t rp_vers; /* version */
78*56e53cb8SWarner Losh uint32_t rp_proc; /* procedure */
79ca987d46SWarner Losh };
80ca987d46SWarner Losh
81ca987d46SWarner Losh struct rpc_reply {
82*56e53cb8SWarner Losh uint32_t rp_xid; /* request transaction id */
83ca987d46SWarner Losh int32_t rp_direction; /* call direction (1) */
84ca987d46SWarner Losh int32_t rp_astatus; /* accept status (0: accepted) */
85ca987d46SWarner Losh union {
86*56e53cb8SWarner Losh uint32_t rpu_errno;
87ca987d46SWarner Losh struct {
88ca987d46SWarner Losh struct auth_info rok_auth;
89*56e53cb8SWarner Losh uint32_t rok_status;
90ca987d46SWarner Losh } rpu_rok;
91ca987d46SWarner Losh } rp_u;
92ca987d46SWarner Losh };
93ca987d46SWarner Losh
94ca987d46SWarner Losh /* Local forwards */
95c5b86c3bSKyle Evans static ssize_t recvrpc(struct iodesc *, void **, void **, time_t, void *);
96ca987d46SWarner Losh static int rpc_getport(struct iodesc *, n_long, n_long);
97ca987d46SWarner Losh
98ca987d46SWarner Losh int rpc_xid;
99ca987d46SWarner Losh int rpc_port = 0x400; /* predecrement */
100ca987d46SWarner Losh
101ca987d46SWarner Losh /*
102ca987d46SWarner Losh * Make a rpc call; return length of answer
103ca987d46SWarner Losh * Note: Caller must leave room for headers.
104ca987d46SWarner Losh */
105ca987d46SWarner Losh ssize_t
rpc_call(struct iodesc * d,n_long prog,n_long vers,n_long proc,void * sdata,size_t slen,void ** rdata,void ** pkt)106ca987d46SWarner Losh rpc_call(struct iodesc *d, n_long prog, n_long vers, n_long proc,
107ca987d46SWarner Losh void *sdata, size_t slen, void **rdata, void **pkt)
108ca987d46SWarner Losh {
109ca987d46SWarner Losh ssize_t cc, rsize;
110ca987d46SWarner Losh struct auth_info *auth;
111ca987d46SWarner Losh struct rpc_call *call;
112ca987d46SWarner Losh struct rpc_reply *reply;
113ca987d46SWarner Losh char *send_head, *send_tail;
114ca987d46SWarner Losh void *ptr;
115ca987d46SWarner Losh n_long x;
116ca987d46SWarner Losh int port; /* host order */
117ca987d46SWarner Losh
118ca987d46SWarner Losh #ifdef RPC_DEBUG
119ca987d46SWarner Losh if (debug)
120ca987d46SWarner Losh printf("rpc_call: prog=0x%x vers=%d proc=%d\n",
121ca987d46SWarner Losh prog, vers, proc);
122ca987d46SWarner Losh #endif
123ca987d46SWarner Losh
124ca987d46SWarner Losh port = rpc_getport(d, prog, vers);
125ca987d46SWarner Losh if (port == -1)
126ca987d46SWarner Losh return (-1);
127ca987d46SWarner Losh
128ca987d46SWarner Losh d->destport = htons(port);
129ca987d46SWarner Losh
130ca987d46SWarner Losh /*
131ca987d46SWarner Losh * Prepend authorization stuff and headers.
132ca987d46SWarner Losh * Note, must prepend things in reverse order.
133ca987d46SWarner Losh */
134ca987d46SWarner Losh send_head = sdata;
135ca987d46SWarner Losh send_tail = (char *)sdata + slen;
136ca987d46SWarner Losh
137ca987d46SWarner Losh /* Auth verifier is always auth_null */
138ca987d46SWarner Losh send_head -= sizeof(*auth);
139ca987d46SWarner Losh auth = (struct auth_info *)send_head;
140ca987d46SWarner Losh auth->authtype = htonl(RPCAUTH_NULL);
141ca987d46SWarner Losh auth->authlen = 0;
142ca987d46SWarner Losh
143ca987d46SWarner Losh /* Auth credentials: always auth unix (as root) */
144ca987d46SWarner Losh send_head -= sizeof(struct auth_unix);
145ca987d46SWarner Losh bzero(send_head, sizeof(struct auth_unix));
146ca987d46SWarner Losh send_head -= sizeof(*auth);
147ca987d46SWarner Losh auth = (struct auth_info *)send_head;
148ca987d46SWarner Losh auth->authtype = htonl(RPCAUTH_UNIX);
149ca987d46SWarner Losh auth->authlen = htonl(sizeof(struct auth_unix));
150ca987d46SWarner Losh
151ca987d46SWarner Losh /* RPC call structure. */
152ca987d46SWarner Losh send_head -= sizeof(*call);
153ca987d46SWarner Losh call = (struct rpc_call *)send_head;
154ca987d46SWarner Losh rpc_xid++;
155ca987d46SWarner Losh call->rp_xid = htonl(rpc_xid);
156ca987d46SWarner Losh call->rp_direction = htonl(RPC_CALL);
157ca987d46SWarner Losh call->rp_rpcvers = htonl(RPC_VER2);
158ca987d46SWarner Losh call->rp_prog = htonl(prog);
159ca987d46SWarner Losh call->rp_vers = htonl(vers);
160ca987d46SWarner Losh call->rp_proc = htonl(proc);
161ca987d46SWarner Losh
162ca987d46SWarner Losh ptr = NULL;
163ca987d46SWarner Losh cc = sendrecv(d,
164ca987d46SWarner Losh sendudp, send_head, send_tail - send_head,
165c5b86c3bSKyle Evans recvrpc, &ptr, (void **)&reply, NULL);
166ca987d46SWarner Losh
167ca987d46SWarner Losh #ifdef RPC_DEBUG
168ca987d46SWarner Losh if (debug)
169ca987d46SWarner Losh printf("callrpc: cc=%zd\n", cc);
170ca987d46SWarner Losh #endif
171ca987d46SWarner Losh if (cc == -1)
172ca987d46SWarner Losh return (-1);
173ca987d46SWarner Losh
174ca987d46SWarner Losh if (cc <= sizeof(*reply)) {
175ca987d46SWarner Losh errno = EBADRPC;
176ca987d46SWarner Losh free(ptr);
177ca987d46SWarner Losh return (-1);
178ca987d46SWarner Losh }
179ca987d46SWarner Losh
180ca987d46SWarner Losh /*
181ca987d46SWarner Losh * Check the RPC reply status.
182ca987d46SWarner Losh * The xid, dir, astatus were already checked.
183ca987d46SWarner Losh */
184ca987d46SWarner Losh auth = &reply->rp_u.rpu_rok.rok_auth;
185ca987d46SWarner Losh x = ntohl(auth->authlen);
186ca987d46SWarner Losh if (x != 0) {
187ca987d46SWarner Losh #ifdef RPC_DEBUG
188ca987d46SWarner Losh if (debug)
189ca987d46SWarner Losh printf("callrpc: reply auth != NULL\n");
190ca987d46SWarner Losh #endif
191ca987d46SWarner Losh errno = EBADRPC;
192ca987d46SWarner Losh free(ptr);
193ca987d46SWarner Losh return (-1);
194ca987d46SWarner Losh }
195ca987d46SWarner Losh x = ntohl(reply->rp_u.rpu_rok.rok_status);
196ca987d46SWarner Losh if (x != 0) {
197ca987d46SWarner Losh printf("callrpc: error = %ld\n", (long)x);
198ca987d46SWarner Losh errno = EBADRPC;
199ca987d46SWarner Losh free(ptr);
200ca987d46SWarner Losh return (-1);
201ca987d46SWarner Losh }
202ca987d46SWarner Losh
203ca987d46SWarner Losh rsize = cc - sizeof(*reply);
204ca987d46SWarner Losh *rdata = (void *)((uintptr_t)reply + sizeof(*reply));
205ca987d46SWarner Losh *pkt = ptr;
206ca987d46SWarner Losh return (rsize);
207ca987d46SWarner Losh }
208ca987d46SWarner Losh
209ca987d46SWarner Losh /*
210ca987d46SWarner Losh * Returns true if packet is the one we're waiting for.
211ca987d46SWarner Losh * This just checks the XID, direction, acceptance.
212ca987d46SWarner Losh * Remaining checks are done by callrpc
213ca987d46SWarner Losh */
214ca987d46SWarner Losh static ssize_t
recvrpc(struct iodesc * d,void ** pkt,void ** payload,time_t tleft,void * extra)215c5b86c3bSKyle Evans recvrpc(struct iodesc *d, void **pkt, void **payload, time_t tleft, void *extra)
216ca987d46SWarner Losh {
217ca987d46SWarner Losh void *ptr;
218ca987d46SWarner Losh struct rpc_reply *reply;
219ca987d46SWarner Losh ssize_t n;
220ca987d46SWarner Losh int x;
221ca987d46SWarner Losh
222ca987d46SWarner Losh errno = 0;
223ca987d46SWarner Losh #ifdef RPC_DEBUG
224ca987d46SWarner Losh if (debug)
225ca987d46SWarner Losh printf("recvrpc: called\n");
226ca987d46SWarner Losh #endif
227ca987d46SWarner Losh
228ca987d46SWarner Losh ptr = NULL;
229ca987d46SWarner Losh n = readudp(d, &ptr, (void **)&reply, tleft);
230ca987d46SWarner Losh if (n <= (4 * 4)) {
231ca987d46SWarner Losh free(ptr);
232ca987d46SWarner Losh return (-1);
233ca987d46SWarner Losh }
234ca987d46SWarner Losh
235ca987d46SWarner Losh x = ntohl(reply->rp_xid);
236ca987d46SWarner Losh if (x != rpc_xid) {
237ca987d46SWarner Losh #ifdef RPC_DEBUG
238ca987d46SWarner Losh if (debug)
239ca987d46SWarner Losh printf("recvrpc: rp_xid %d != xid %d\n", x, rpc_xid);
240ca987d46SWarner Losh #endif
241ca987d46SWarner Losh free(ptr);
242ca987d46SWarner Losh return (-1);
243ca987d46SWarner Losh }
244ca987d46SWarner Losh
245ca987d46SWarner Losh x = ntohl(reply->rp_direction);
246ca987d46SWarner Losh if (x != RPC_REPLY) {
247ca987d46SWarner Losh #ifdef RPC_DEBUG
248ca987d46SWarner Losh if (debug)
249ca987d46SWarner Losh printf("recvrpc: rp_direction %d != REPLY\n", x);
250ca987d46SWarner Losh #endif
251ca987d46SWarner Losh free(ptr);
252ca987d46SWarner Losh return (-1);
253ca987d46SWarner Losh }
254ca987d46SWarner Losh
255ca987d46SWarner Losh x = ntohl(reply->rp_astatus);
256ca987d46SWarner Losh if (x != RPC_MSGACCEPTED) {
257ca987d46SWarner Losh errno = ntohl(reply->rp_u.rpu_errno);
258ca987d46SWarner Losh printf("recvrpc: reject, astat=%d, errno=%d\n", x, errno);
259ca987d46SWarner Losh free(ptr);
260ca987d46SWarner Losh return (-1);
261ca987d46SWarner Losh }
262ca987d46SWarner Losh
263ca987d46SWarner Losh *pkt = ptr;
264ca987d46SWarner Losh *payload = reply;
265ca987d46SWarner Losh /* Return data count (thus indicating success) */
266ca987d46SWarner Losh return (n);
267ca987d46SWarner Losh }
268ca987d46SWarner Losh
269ca987d46SWarner Losh /*
270ca987d46SWarner Losh * Given a pointer to a reply just received,
271ca987d46SWarner Losh * dig out the IP address/port from the headers.
272ca987d46SWarner Losh */
273ca987d46SWarner Losh void
rpc_fromaddr(void * pkt,struct in_addr * addr,u_short * port)274ca987d46SWarner Losh rpc_fromaddr(void *pkt, struct in_addr *addr, u_short *port)
275ca987d46SWarner Losh {
276ca987d46SWarner Losh struct hackhdr {
277ca987d46SWarner Losh /* Tail of IP header: just IP addresses */
278ca987d46SWarner Losh n_long ip_src;
279ca987d46SWarner Losh n_long ip_dst;
280ca987d46SWarner Losh /* UDP header: */
281*56e53cb8SWarner Losh uint16_t uh_sport; /* source port */
282*56e53cb8SWarner Losh uint16_t uh_dport; /* destination port */
283ca987d46SWarner Losh int16_t uh_ulen; /* udp length */
284*56e53cb8SWarner Losh uint16_t uh_sum; /* udp checksum */
285ca987d46SWarner Losh /* RPC reply header: */
286ca987d46SWarner Losh struct rpc_reply rpc;
287ca987d46SWarner Losh } *hhdr;
288ca987d46SWarner Losh
289ca987d46SWarner Losh hhdr = ((struct hackhdr *)pkt) - 1;
290ca987d46SWarner Losh addr->s_addr = hhdr->ip_src;
291ca987d46SWarner Losh *port = hhdr->uh_sport;
292ca987d46SWarner Losh }
293ca987d46SWarner Losh
294ca987d46SWarner Losh /*
295ca987d46SWarner Losh * RPC Portmapper cache
296ca987d46SWarner Losh */
297ca987d46SWarner Losh #define PMAP_NUM 8 /* need at most 5 pmap entries */
298ca987d46SWarner Losh
299ca987d46SWarner Losh int rpc_pmap_num;
300ca987d46SWarner Losh struct pmap_list {
301ca987d46SWarner Losh struct in_addr addr; /* server, net order */
302ca987d46SWarner Losh u_int prog; /* host order */
303ca987d46SWarner Losh u_int vers; /* host order */
304ca987d46SWarner Losh int port; /* host order */
305ca987d46SWarner Losh } rpc_pmap_list[PMAP_NUM];
306ca987d46SWarner Losh
307ca987d46SWarner Losh /*
308ca987d46SWarner Losh * return port number in host order, or -1.
309ca987d46SWarner Losh * arguments are:
310ca987d46SWarner Losh * addr .. server, net order.
311ca987d46SWarner Losh * prog .. host order.
312ca987d46SWarner Losh * vers .. host order.
313ca987d46SWarner Losh */
314ca987d46SWarner Losh int
rpc_pmap_getcache(struct in_addr addr,u_int prog,u_int vers)315ca987d46SWarner Losh rpc_pmap_getcache(struct in_addr addr, u_int prog, u_int vers)
316ca987d46SWarner Losh {
317ca987d46SWarner Losh struct pmap_list *pl;
318ca987d46SWarner Losh
319ca987d46SWarner Losh for (pl = rpc_pmap_list; pl < &rpc_pmap_list[rpc_pmap_num]; pl++) {
320ca987d46SWarner Losh if (pl->addr.s_addr == addr.s_addr &&
321ca987d46SWarner Losh pl->prog == prog && pl->vers == vers )
322ca987d46SWarner Losh {
323ca987d46SWarner Losh return (pl->port);
324ca987d46SWarner Losh }
325ca987d46SWarner Losh }
326ca987d46SWarner Losh return (-1);
327ca987d46SWarner Losh }
328ca987d46SWarner Losh
329ca987d46SWarner Losh /*
330ca987d46SWarner Losh * arguments are:
331ca987d46SWarner Losh * addr .. server, net order.
332ca987d46SWarner Losh * prog .. host order.
333ca987d46SWarner Losh * vers .. host order.
334ca987d46SWarner Losh * port .. host order.
335ca987d46SWarner Losh */
336ca987d46SWarner Losh void
rpc_pmap_putcache(struct in_addr addr,u_int prog,u_int vers,int port)337ca987d46SWarner Losh rpc_pmap_putcache(struct in_addr addr, u_int prog, u_int vers, int port)
338ca987d46SWarner Losh {
339ca987d46SWarner Losh struct pmap_list *pl;
340ca987d46SWarner Losh
341ca987d46SWarner Losh /* Don't overflow cache... */
342ca987d46SWarner Losh if (rpc_pmap_num >= PMAP_NUM) {
343ca987d46SWarner Losh /* ... just re-use the last entry. */
344ca987d46SWarner Losh rpc_pmap_num = PMAP_NUM - 1;
345ca987d46SWarner Losh #ifdef RPC_DEBUG
346ca987d46SWarner Losh printf("rpc_pmap_putcache: cache overflow\n");
347ca987d46SWarner Losh #endif
348ca987d46SWarner Losh }
349ca987d46SWarner Losh
350ca987d46SWarner Losh pl = &rpc_pmap_list[rpc_pmap_num];
351ca987d46SWarner Losh rpc_pmap_num++;
352ca987d46SWarner Losh
353ca987d46SWarner Losh /* Cache answer */
354ca987d46SWarner Losh pl->addr = addr;
355ca987d46SWarner Losh pl->prog = prog;
356ca987d46SWarner Losh pl->vers = vers;
357ca987d46SWarner Losh pl->port = port;
358ca987d46SWarner Losh }
359ca987d46SWarner Losh
360ca987d46SWarner Losh
361ca987d46SWarner Losh /*
362ca987d46SWarner Losh * Request a port number from the port mapper.
363ca987d46SWarner Losh * Returns the port in host order.
364ca987d46SWarner Losh * prog and vers are host order.
365ca987d46SWarner Losh */
366ca987d46SWarner Losh int
rpc_getport(struct iodesc * d,n_long prog,n_long vers)367ca987d46SWarner Losh rpc_getport(struct iodesc *d, n_long prog, n_long vers)
368ca987d46SWarner Losh {
369ca987d46SWarner Losh struct args {
370ca987d46SWarner Losh n_long prog; /* call program */
371ca987d46SWarner Losh n_long vers; /* call version */
372ca987d46SWarner Losh n_long proto; /* call protocol */
373ca987d46SWarner Losh n_long port; /* call port (unused) */
374ca987d46SWarner Losh } *args;
375ca987d46SWarner Losh struct res {
376ca987d46SWarner Losh n_long port;
377ca987d46SWarner Losh } *res;
378ca987d46SWarner Losh struct {
379ca987d46SWarner Losh n_long h[RPC_HEADER_WORDS];
380ca987d46SWarner Losh struct args d;
381ca987d46SWarner Losh } sdata;
382ca987d46SWarner Losh void *pkt;
383ca987d46SWarner Losh ssize_t cc;
384ca987d46SWarner Losh int port;
385ca987d46SWarner Losh
386ca987d46SWarner Losh #ifdef RPC_DEBUG
387ca987d46SWarner Losh if (debug)
388ca987d46SWarner Losh printf("%s: prog=0x%x vers=%d\n", __func__, prog, vers);
389ca987d46SWarner Losh #endif
390ca987d46SWarner Losh
391ca987d46SWarner Losh /* This one is fixed forever. */
392ca987d46SWarner Losh if (prog == PMAPPROG) {
393ca987d46SWarner Losh port = PMAPPORT;
394ca987d46SWarner Losh goto out;
395ca987d46SWarner Losh }
396ca987d46SWarner Losh
397ca987d46SWarner Losh /* Try for cached answer first */
398ca987d46SWarner Losh port = rpc_pmap_getcache(d->destip, prog, vers);
399ca987d46SWarner Losh if (port != -1)
400ca987d46SWarner Losh goto out;
401ca987d46SWarner Losh
402ca987d46SWarner Losh args = &sdata.d;
403ca987d46SWarner Losh args->prog = htonl(prog);
404ca987d46SWarner Losh args->vers = htonl(vers);
405ca987d46SWarner Losh args->proto = htonl(IPPROTO_UDP);
406ca987d46SWarner Losh args->port = 0;
407ca987d46SWarner Losh pkt = NULL;
408ca987d46SWarner Losh
409ca987d46SWarner Losh cc = rpc_call(d, PMAPPROG, PMAPVERS, PMAPPROC_GETPORT,
410ca987d46SWarner Losh args, sizeof(*args), (void **)&res, &pkt);
411ca987d46SWarner Losh if (cc < sizeof(*res)) {
412ca987d46SWarner Losh printf("getport: %s", strerror(errno));
413ca987d46SWarner Losh errno = EBADRPC;
414ca987d46SWarner Losh free(pkt);
415ca987d46SWarner Losh return (-1);
416ca987d46SWarner Losh }
417ca987d46SWarner Losh port = (int)ntohl(res->port);
418ca987d46SWarner Losh free(pkt);
419ca987d46SWarner Losh
420ca987d46SWarner Losh rpc_pmap_putcache(d->destip, prog, vers, port);
421ca987d46SWarner Losh
422ca987d46SWarner Losh out:
423ca987d46SWarner Losh #ifdef RPC_DEBUG
424ca987d46SWarner Losh if (debug)
425ca987d46SWarner Losh printf("%s: port=%u\n", __func__, port);
426ca987d46SWarner Losh #endif
427ca987d46SWarner Losh return (port);
428ca987d46SWarner Losh }
429