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