1 /*
2  * Copyright (c) 2009, Sun Microsystems, Inc.
3  * All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are met:
7  * - Redistributions of source code must retain the above copyright notice,
8  *   this list of conditions and the following disclaimer.
9  * - Redistributions in binary form must reproduce the above copyright notice,
10  *   this list of conditions and the following disclaimer in the documentation
11  *   and/or other materials provided with the distribution.
12  * - Neither the name of Sun Microsystems, Inc. nor the names of its
13  *   contributors may be used to endorse or promote products derived
14  *   from this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
17  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
18  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
19  * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
20  * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
21  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
22  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
23  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
24  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
25  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
26  * POSSIBILITY OF SUCH DAMAGE.
27  */
28 /*
29  * Copyright (c) 1986-1991 by Sun Microsystems Inc.
30  */
31 
32 //#include <sys/cdefs.h>
33 
34 /*
35  * rpc_generic.c, Miscl routines for RPC.
36  *
37  */
38 #include <wintirpc.h>
39 //#include <pthread.h>
40 #include <reentrant.h>
41 #include <sys/types.h>
42 //#include <sys/param.h>
43 //#include <sys/socket.h>
44 //#include <sys/time.h>
45 //#include <sys/un.h>
46 //#include <sys/resource.h>
47 //#include <netinet/in.h>
48 //#include <arpa/inet.h>
49 #include <rpc/rpc.h>
50 //#include <ctype.h>
51 //#include <stddef.h>
52 #include <stdio.h>
53 //#include <netdb.h>
54 #include <netconfig.h>
55 #include <stdlib.h>
56 #include <string.h>
57 //#include <syslog.h>
58 #include <rpc/nettype.h>
59 #include "rpc_com.h"
60 
61 struct handle {
62 	NCONF_HANDLE *nhandle;
63 	int nflag;		/* Whether NETPATH or NETCONFIG */
64 	int nettype;
65 };
66 
67 static const struct _rpcnettype {
68 	const char *name;
69 	const int type;
70 } _rpctypelist[] = {
71 	{ "netpath", _RPC_NETPATH },
72 	{ "visible", _RPC_VISIBLE },
73 	{ "circuit_v", _RPC_CIRCUIT_V },
74 	{ "datagram_v", _RPC_DATAGRAM_V },
75 	{ "circuit_n", _RPC_CIRCUIT_N },
76 	{ "datagram_n", _RPC_DATAGRAM_N },
77 	{ "tcp", _RPC_TCP },
78 	{ "udp", _RPC_UDP },
79 	{ 0, _RPC_NONE }
80 };
81 
82 struct netid_af {
83 	const char	*netid;
84 	ADDRESS_FAMILY		af;
85 	int		protocol;
86 };
87 
88 static const struct netid_af na_cvt[] = {
89 	{ "udp",  AF_INET,  IPPROTO_UDP },
90 	{ "tcp",  AF_INET,  IPPROTO_TCP },
91 #ifdef INET6
92 	{ "udp6", AF_INET6, IPPROTO_UDP },
93 	{ "tcp6", AF_INET6, IPPROTO_TCP },
94 #endif
95 #ifdef AF_LOCAL
96 	{ "local", AF_LOCAL, 0 }
97 #endif
98 };
99 
100 #if 0
101 static char *strlocase(char *);
102 #endif
103 static int getnettype(const char *);
104 
105 /*
106  * Cache the result of getrlimit(), so we don't have to do an
107  * expensive call every time.
108  */
109 int
110 __rpc_dtbsize()
111 {
112 #ifdef _WIN32
113 	return (WINSOCK_HANDLE_HASH_SIZE);
114 #else
115 
116 	static int tbsize;
117 	struct rlimit rl;
118 
119 	if (tbsize) {
120 		return (tbsize);
121 	}
122 	if (getrlimit(RLIMIT_NOFILE, &rl) == 0) {
123 		return (tbsize = (int)rl.rlim_max);
124 	}
125 	/*
126 	 * Something wrong.  I'll try to save face by returning a
127 	 * pessimistic number.
128 	 */
129 	return (32);
130 #endif
131 }
132 
133 
134 /*
135  * Find the appropriate buffer size
136  */
137 u_int
138 /*ARGSUSED*/
139 __rpc_get_t_size(af, proto, size)
140 	int af, proto;
141 	int size;	/* Size requested */
142 {
143 	int maxsize, defsize;
144 
145 	maxsize = 256 * 1024;	/* XXX */
146 	switch (proto) {
147 	case IPPROTO_TCP:
148 		defsize = 1024 * 1024;	/* XXX */
149 		break;
150 	case IPPROTO_UDP:
151 		defsize = UDPMSGSIZE;
152 		break;
153 	default:
154 		defsize = RPC_MAXDATASIZE;
155 		break;
156 	}
157 	if (size == 0)
158 		return defsize;
159 #if 1
160     /* cbodley- give us the size we ask for, or we'll get fragmented! */
161     return (u_int)size;
162 #else
163 	/* Check whether the value is within the upper max limit */
164 	return (size > maxsize ? (u_int)maxsize : (u_int)size);
165 #endif
166 }
167 
168 /*
169  * Find the appropriate address buffer size
170  */
171 u_int
172 __rpc_get_a_size(af)
173 	int af;
174 {
175 	switch (af) {
176 	case AF_INET:
177 		return sizeof (struct sockaddr_in);
178 #ifdef INET6
179 	case AF_INET6:
180 		return sizeof (struct sockaddr_in6);
181 #endif
182 #ifdef AF_LOCAL
183 	case AF_LOCAL:
184 		return sizeof (struct sockaddr_un);
185 #endif
186 	default:
187 		break;
188 	}
189 	return ((u_int)RPC_MAXADDRSIZE);
190 }
191 
192 #if 0
193 static char *
194 strlocase(p)
195 	char *p;
196 {
197 	char *t = p;
198 
199 	for (; *p; p++)
200 		if (isupper(*p))
201 			*p = tolower(*p);
202 	return (t);
203 }
204 #endif
205 
206 /*
207  * Returns the type of the network as defined in <rpc/nettype.h>
208  * If nettype is NULL, it defaults to NETPATH.
209  */
210 static int
211 getnettype(nettype)
212 	const char *nettype;
213 {
214 	int i;
215 
216 	if ((nettype == NULL) || (nettype[0] == 0)) {
217 		return (_RPC_NETPATH);	/* Default */
218 	}
219 
220 #if 0
221 	nettype = strlocase(nettype);
222 #endif
223 	for (i = 0; _rpctypelist[i].name; i++)
224 		if (strcasecmp(nettype, _rpctypelist[i].name) == 0) {
225 			return (_rpctypelist[i].type);
226 		}
227 	return (_rpctypelist[i].type);
228 }
229 
230 /*
231  * For the given nettype (tcp or udp only), return the first structure found.
232  * This should be freed by calling freenetconfigent()
233  */
234 struct netconfig *
235 __rpc_getconfip(nettype)
236 	const char *nettype;
237 {
238 	char *netid;
239 	char *netid_tcp = (char *) NULL;
240 	char *netid_udp = (char *) NULL;
241 	struct netconfig *dummy;
242 	extern thread_key_t tcp_key, udp_key;
243 	extern mutex_t tsd_lock;
244 
245 	if (tcp_key == -1) {
246 		mutex_lock(&tsd_lock);
247 		if (tcp_key == -1)
248 			tcp_key = TlsAlloc();	//thr_keycreate(&tcp_key, free);
249 		mutex_unlock(&tsd_lock);
250 	}
251 	netid_tcp = (char *)thr_getspecific(tcp_key);
252 	if (udp_key == -1) {
253 		mutex_lock(&tsd_lock);
254 		if (udp_key == -1)
255 			udp_key = TlsAlloc();	//thr_keycreate(&udp_key, free);
256 		mutex_unlock(&tsd_lock);
257 	}
258 	netid_udp = (char *)thr_getspecific(udp_key);
259 	if (!netid_udp && !netid_tcp) {
260 		struct netconfig *nconf;
261 		void *confighandle;
262 
263 		if (!(confighandle = setnetconfig())) {
264 			//syslog (LOG_ERR, "rpc: failed to open " NETCONFIG);
265 			return (NULL);
266 		}
267 		while ((nconf = getnetconfig(confighandle)) != NULL) {
268 			if (strcmp(nconf->nc_protofmly, NC_INET) == 0 ||
269 			    strcmp(nconf->nc_protofmly, NC_INET6) == 0) {
270 				if (strcmp(nconf->nc_proto, NC_TCP) == 0 &&
271 						netid_tcp == NULL) {
272 					netid_tcp = strdup(nconf->nc_netid);
273 					thr_setspecific(tcp_key,
274 							(void *) netid_tcp);
275 				} else
276 				if (strcmp(nconf->nc_proto, NC_UDP) == 0 &&
277 						netid_udp == NULL) {
278 					netid_udp = strdup(nconf->nc_netid);
279 					thr_setspecific(udp_key,
280 						(void *) netid_udp);
281 				}
282 			}
283 		}
284 		endnetconfig(confighandle);
285 	}
286 	if (strcmp(nettype, "udp") == 0)
287 		netid = netid_udp;
288 	else if (strcmp(nettype, "tcp") == 0)
289 		netid = netid_tcp;
290 	else {
291 		return (NULL);
292 	}
293 	if ((netid == NULL) || (netid[0] == 0)) {
294 		return (NULL);
295 	}
296 	dummy = getnetconfigent(netid);
297 	return (dummy);
298 }
299 
300 /*
301  * Returns the type of the nettype, which should then be used with
302  * __rpc_getconf().
303  */
304 void *
305 __rpc_setconf(nettype)
306 	const char *nettype;
307 {
308 	struct handle *handle;
309 
310 	handle = (struct handle *) malloc(sizeof (struct handle));
311 	if (handle == NULL) {
312 		return (NULL);
313 	}
314 	switch (handle->nettype = getnettype(nettype)) {
315 	case _RPC_NETPATH:
316 	case _RPC_CIRCUIT_N:
317 	case _RPC_DATAGRAM_N:
318 		if (!(handle->nhandle = setnetpath())) {
319 			free(handle);
320 			return (NULL);
321 		}
322 		handle->nflag = TRUE;
323 		break;
324 	case _RPC_VISIBLE:
325 	case _RPC_CIRCUIT_V:
326 	case _RPC_DATAGRAM_V:
327 	case _RPC_TCP:
328 	case _RPC_UDP:
329 		if (!(handle->nhandle = setnetconfig())) {
330 		        //syslog (LOG_ERR, "rpc: failed to open " NETCONFIG);
331 			free(handle);
332 			return (NULL);
333 		}
334 		handle->nflag = FALSE;
335 		break;
336 	default:
337 		return (NULL);
338 	}
339 
340 	return (handle);
341 }
342 
343 /*
344  * Returns the next netconfig struct for the given "net" type.
345  * __rpc_setconf() should have been called previously.
346  */
347 struct netconfig *
348 __rpc_getconf(vhandle)
349 	void *vhandle;
350 {
351 	struct handle *handle;
352 	struct netconfig *nconf;
353 
354 	handle = (struct handle *)vhandle;
355 	if (handle == NULL) {
356 		return (NULL);
357 	}
358 	for (;;) {
359 		if (handle->nflag)
360 			nconf = getnetpath(handle->nhandle);
361 		else
362 			nconf = getnetconfig(handle->nhandle);
363 		if (nconf == NULL)
364 			break;
365 		if ((nconf->nc_semantics != NC_TPI_CLTS) &&
366 			(nconf->nc_semantics != NC_TPI_COTS) &&
367 			(nconf->nc_semantics != NC_TPI_COTS_ORD))
368 			continue;
369 		switch (handle->nettype) {
370 		case _RPC_VISIBLE:
371 			if (!(nconf->nc_flag & NC_VISIBLE))
372 				continue;
373 			/* FALLTHROUGH */
374 		case _RPC_NETPATH:	/* Be happy */
375 			break;
376 		case _RPC_CIRCUIT_V:
377 			if (!(nconf->nc_flag & NC_VISIBLE))
378 				continue;
379 			/* FALLTHROUGH */
380 		case _RPC_CIRCUIT_N:
381 			if ((nconf->nc_semantics != NC_TPI_COTS) &&
382 				(nconf->nc_semantics != NC_TPI_COTS_ORD))
383 				continue;
384 			break;
385 		case _RPC_DATAGRAM_V:
386 			if (!(nconf->nc_flag & NC_VISIBLE))
387 				continue;
388 			/* FALLTHROUGH */
389 		case _RPC_DATAGRAM_N:
390 			if (nconf->nc_semantics != NC_TPI_CLTS)
391 				continue;
392 			break;
393 		case _RPC_TCP:
394 			if (((nconf->nc_semantics != NC_TPI_COTS) &&
395 				(nconf->nc_semantics != NC_TPI_COTS_ORD)) ||
396 				(strcmp(nconf->nc_protofmly, NC_INET)
397 #ifdef INET6
398 				 && strcmp(nconf->nc_protofmly, NC_INET6))
399 #else
400 				)
401 #endif
402 				||
403 				strcmp(nconf->nc_proto, NC_TCP))
404 				continue;
405 			break;
406 		case _RPC_UDP:
407 			if ((nconf->nc_semantics != NC_TPI_CLTS) ||
408 				(strcmp(nconf->nc_protofmly, NC_INET)
409 #ifdef INET6
410 				&& strcmp(nconf->nc_protofmly, NC_INET6))
411 #else
412 				)
413 #endif
414 				||
415 				strcmp(nconf->nc_proto, NC_UDP))
416 				continue;
417 			break;
418 		}
419 		break;
420 	}
421 	return (nconf);
422 }
423 
424 void
425 __rpc_endconf(vhandle)
426 	void * vhandle;
427 {
428 	struct handle *handle;
429 
430 	handle = (struct handle *) vhandle;
431 	if (handle == NULL) {
432 		return;
433 	}
434 	if (handle->nflag) {
435 		endnetpath(handle->nhandle);
436 	} else {
437 		endnetconfig(handle->nhandle);
438 	}
439 	free(handle);
440 }
441 
442 /*
443  * Used to ping the NULL procedure for clnt handle.
444  * Returns NULL if fails, else a non-NULL pointer.
445  */
446 void *
447 rpc_nullproc(clnt)
448 	CLIENT *clnt;
449 {
450 	struct timeval TIMEOUT = {25, 0};
451 
452 	if (clnt_call(clnt, NULLPROC, (xdrproc_t) xdr_void, NULL,
453 		(xdrproc_t) xdr_void, NULL, TIMEOUT) != RPC_SUCCESS) {
454 		return (NULL);
455 	}
456 	return ((void *) clnt);
457 }
458 
459 /*
460  * Try all possible transports until
461  * one succeeds in finding the netconf for the given fd.
462  */
463 struct netconfig *
464 __rpcgettp(fd)
465 	SOCKET fd;
466 {
467 	const char *netid;
468 	struct __rpc_sockinfo si;
469 
470 	if (!__rpc_fd2sockinfo(fd, &si))
471 		return NULL;
472 
473 	if (!__rpc_sockinfo2netid(&si, &netid))
474 		return NULL;
475 
476 	/*LINTED const castaway*/
477 	return getnetconfigent((char *)netid);
478 }
479 
480 int
481 __rpc_fd2sockinfo(SOCKET fd, struct __rpc_sockinfo *sip)
482 {
483 	socklen_t len;
484 	int type, proto;
485 	struct sockaddr_storage ss;
486 
487 #ifdef _WIN32
488 	WSAPROTOCOL_INFO proto_info;
489 	int proto_info_size = sizeof(proto_info);
490 	if (getsockopt(fd, SOL_SOCKET, SO_PROTOCOL_INFO, (char *)&proto_info, &proto_info_size) == SOCKET_ERROR) {
491 #ifndef __REACTOS__
492 		int err = WSAGetLastError();
493 #endif
494 		return 0;
495 	}
496 	len = proto_info.iMaxSockAddr;
497 	ss.ss_family = (ADDRESS_FAMILY)proto_info.iAddressFamily;
498 #else
499 	len = sizeof ss;
500 	if (getsockname(fd, (struct sockaddr *)&ss, &len) == SOCKET_ERROR) {
501 		return 0;
502 	}
503 #endif
504 	sip->si_alen = len;
505 
506 	len = sizeof type;
507 	if (getsockopt(fd, SOL_SOCKET, SO_TYPE, (char *)&type, &len) == SOCKET_ERROR) {
508 #ifndef __REACTOS__
509 		int err = WSAGetLastError();
510 #endif
511 		return 0;
512 	}
513 
514 	/* XXX */
515 #ifdef AF_LOCAL
516 	if (ss.ss_family != AF_LOCAL) {
517 #endif
518 		if (type == SOCK_STREAM)
519 			proto = IPPROTO_TCP;
520 		else if (type == SOCK_DGRAM)
521 			proto = IPPROTO_UDP;
522 		else
523 			return 0;
524 #ifdef AF_LOCAL
525 	} else
526 		proto = 0;
527 #endif
528 
529 	sip->si_af = ss.ss_family;
530 	sip->si_proto = proto;
531 	sip->si_socktype = type;
532 
533 	return 1;
534 }
535 
536 /*
537  * Linear search, but the number of entries is small.
538  */
539 int
540 __rpc_nconf2sockinfo(const struct netconfig *nconf, struct __rpc_sockinfo *sip)
541 {
542 	int i;
543 
544 	for (i = 0; i < (sizeof na_cvt) / (sizeof (struct netid_af)); i++)
545 		if (strcmp(na_cvt[i].netid, nconf->nc_netid) == 0 || (
546 		    strcmp(nconf->nc_netid, "unix") == 0 &&
547 		    strcmp(na_cvt[i].netid, "local") == 0)) {
548 			sip->si_af = na_cvt[i].af;
549 			sip->si_proto = na_cvt[i].protocol;
550 			sip->si_socktype =
551 			    __rpc_seman2socktype((int)nconf->nc_semantics);
552 			if (sip->si_socktype == -1)
553 				return 0;
554 			sip->si_alen = __rpc_get_a_size(sip->si_af);
555 			return 1;
556 		}
557 
558 	return 0;
559 }
560 
561 SOCKET
562 __rpc_nconf2fd(const struct netconfig *nconf)
563 {
564 	struct __rpc_sockinfo si;
565 	SOCKET fd;
566 
567 	if (!__rpc_nconf2sockinfo(nconf, &si))
568 		return 0;
569 
570 	if ((fd = socket(si.si_af, si.si_socktype, si.si_proto)) != INVALID_SOCKET &&
571 	    si.si_af == AF_INET6) {
572 		int val = 1;
573 
574 		setsockopt(fd, SOL_IPV6, IPV6_V6ONLY, (const char *)&val, sizeof(val));
575 	}
576 	return fd;
577 }
578 
579 int
580 __rpc_sockinfo2netid(struct __rpc_sockinfo *sip, const char **netid)
581 {
582 	int i;
583 	struct netconfig *nconf;
584 
585 	nconf = getnetconfigent("local");
586 
587 	for (i = 0; i < (sizeof na_cvt) / (sizeof (struct netid_af)); i++) {
588 		if (na_cvt[i].af == sip->si_af &&
589 		    na_cvt[i].protocol == sip->si_proto) {
590 			if (strcmp(na_cvt[i].netid, "local") == 0 && nconf == NULL) {
591 				if (netid)
592 					*netid = "unix";
593 			} else {
594 				if (netid)
595 					*netid = na_cvt[i].netid;
596 			}
597 			if (nconf != NULL)
598 				freenetconfigent(nconf);
599 			return 1;
600 		}
601 	}
602 	if (nconf != NULL)
603 		freenetconfigent(nconf);
604 
605 	return 0;
606 }
607 
608 char *
609 taddr2uaddr(const struct netconfig *nconf, const struct netbuf *nbuf)
610 {
611 	struct __rpc_sockinfo si;
612 
613 	if (!__rpc_nconf2sockinfo(nconf, &si))
614 		return NULL;
615 	return __rpc_taddr2uaddr_af(si.si_af, nbuf);
616 }
617 
618 struct netbuf *
619 uaddr2taddr(const struct netconfig *nconf, const char *uaddr)
620 {
621 	struct __rpc_sockinfo si;
622 
623 	if (!__rpc_nconf2sockinfo(nconf, &si))
624 		return NULL;
625 	return __rpc_uaddr2taddr_af(si.si_af, uaddr);
626 }
627 
628 void freeuaddr(char *uaddr)
629 {
630 	free(uaddr);
631 }
632 
633 void freenetbuf(struct netbuf *nbuf)
634 {
635 	if (nbuf) {
636 		free(nbuf->buf);
637 		free(nbuf);
638 	}
639 }
640 
641 #ifdef __REACTOS__
642 PCSTR
643 WSAAPI
644 inet_ntop(INT af, const VOID *src, PSTR dst, size_t cnt)
645 {
646 	struct in_addr in;
647 	char *text_addr;
648 
649 	if (af == AF_INET) {
650 		memcpy(&in.s_addr, src, sizeof(in.s_addr));
651 		text_addr = inet_ntoa(in);
652 		if (text_addr && dst) {
653 			strncpy(dst, text_addr, cnt);
654 			return dst;
655 		}
656 	}
657 
658 	return 0;
659 }
660 #endif
661 
662 char *
663 __rpc_taddr2uaddr_af(int af, const struct netbuf *nbuf)
664 {
665 	char *ret;
666 	struct sockaddr_in *sin;
667 #ifdef AF_LOCAL
668 	struct sockaddr_un *sun;
669 #endif
670 	char namebuf[INET_ADDRSTRLEN];
671 #ifdef INET6
672 	struct sockaddr_in6 *sin6;
673 	char namebuf6[INET6_ADDRSTRLEN];
674 #endif
675 	u_int16_t port;
676 
677 	if (nbuf->len <= 0)
678 		return NULL;
679 
680 	switch (af) {
681 	case AF_INET:
682 #ifdef __REACTOS__ // CVE-2017-8779
683 		if (nbuf->len < sizeof(*sin)) {
684 			return NULL;
685 		}
686 #endif
687 		sin = nbuf->buf;
688 		if (inet_ntop(af, &sin->sin_addr, namebuf, sizeof namebuf)
689 		    == NULL)
690 			return NULL;
691 		port = ntohs(sin->sin_port);
692 		if (asprintf(&ret, "%s.%u.%u", namebuf, ((u_int32_t)port) >> 8,
693 		    port & 0xff) < 0)
694 			return NULL;
695 		break;
696 #ifdef INET6
697 	case AF_INET6:
698 #ifdef __REACTOS__ // CVE-2017-8779
699 		if (nbuf->len < sizeof(*sin6)) {
700 			return NULL;
701 		}
702 #endif
703 		sin6 = nbuf->buf;
704 		if (inet_ntop(af, &sin6->sin6_addr, namebuf6, sizeof namebuf6)
705 		    == NULL)
706 			return NULL;
707 		port = ntohs(sin6->sin6_port);
708 		if (asprintf(&ret, "%s.%u.%u", namebuf6, ((u_int32_t)port) >> 8,
709 		    port & 0xff) < 0)
710 			return NULL;
711 		break;
712 #endif
713 #ifdef AF_LOCAL
714 	case AF_LOCAL:
715 		sun = nbuf->buf;
716 		/*	if (asprintf(&ret, "%.*s", (int)(sun->sun_len -
717 		    offsetof(struct sockaddr_un, sun_path)),
718 		    sun->sun_path) < 0)*/
719 		if (asprintf(&ret, "%.*s", (int)(sizeof(*sun) -
720 						 offsetof(struct sockaddr_un, sun_path)),
721 			     sun->sun_path) < 0)
722 
723 			return (NULL);
724 		break;
725 #endif
726 	default:
727 		return NULL;
728 	}
729 
730 	return ret;
731 }
732 
733 struct netbuf *
734 __rpc_uaddr2taddr_af(int af, const char *uaddr)
735 {
736 	struct netbuf *ret = NULL;
737 	char *addrstr, *p;
738 	unsigned short port, portlo, porthi;
739 	struct sockaddr_in *sin;
740 #ifdef INET6
741 	struct sockaddr_in6 *sin6;
742 #endif
743 #ifdef AF_LOCAL
744 	struct sockaddr_un *sun;
745 #endif
746 
747 	port = 0;
748 	sin = NULL;
749 #ifdef __REACTOS__ // CVE-2017-8779
750 	if (uaddr == NULL)
751 		return NULL;
752 #endif
753 	addrstr = strdup(uaddr);
754 	if (addrstr == NULL)
755 		return NULL;
756 
757 	/*
758 	 * AF_LOCAL addresses are expected to be absolute
759 	 * pathnames, anything else will be AF_INET or AF_INET6.
760 	 */
761 	if (*addrstr != '/') {
762 		p = strrchr(addrstr, '.');
763 		if (p == NULL)
764 			goto out;
765 		portlo = (unsigned)atoi(p + 1);
766 		*p = '\0';
767 
768 		p = strrchr(addrstr, '.');
769 		if (p == NULL)
770 			goto out;
771 		porthi = (unsigned)atoi(p + 1);
772 		*p = '\0';
773 		port = (porthi << 8) | portlo;
774 	}
775 
776 	ret = (struct netbuf *)malloc(sizeof *ret);
777 	if (ret == NULL)
778 		goto out;
779 
780 	switch (af) {
781 	case AF_INET:
782 		sin = (struct sockaddr_in *)malloc(sizeof *sin);
783 		if (sin == NULL)
784 			goto out;
785 		memset(sin, 0, sizeof *sin);
786 		sin->sin_family = AF_INET;
787 		sin->sin_port = htons(port);
788 #ifndef __REACTOS__
789 		if (inet_pton(AF_INET, addrstr, &sin->sin_addr) <= 0) {
790 #else
791 		sin->sin_addr.S_un.S_addr = inet_addr(addrstr);
792 		if (sin->sin_addr.S_un.S_addr == INADDR_NONE) {
793 #endif
794 			free(sin);
795 			free(ret);
796 			ret = NULL;
797 			goto out;
798 		}
799 		ret->maxlen = ret->len = sizeof *sin;
800 		ret->buf = sin;
801 		break;
802 #ifdef INET6
803 	case AF_INET6:
804 		sin6 = (struct sockaddr_in6 *)malloc(sizeof *sin6);
805 		if (sin6 == NULL)
806 			goto out;
807 		memset(sin6, 0, sizeof *sin6);
808 		sin6->sin6_family = AF_INET6;
809 		sin6->sin6_port = htons(port);
810 		if (inet_pton(AF_INET6, addrstr, &sin6->sin6_addr) <= 0) {
811 			free(sin6);
812 			free(ret);
813 			ret = NULL;
814 			goto out;
815 		}
816 		ret->maxlen = ret->len = sizeof *sin6;
817 		ret->buf = sin6;
818 		break;
819 #endif
820 #ifdef AF_LOCAL
821 	case AF_LOCAL:
822 		sun = (struct sockaddr_un *)malloc(sizeof *sun);
823 		if (sun == NULL)
824 			goto out;
825 		memset(sun, 0, sizeof *sun);
826 		sun->sun_family = AF_LOCAL;
827 		strncpy(sun->sun_path, addrstr, sizeof(sun->sun_path) - 1);
828 		ret->len = SUN_LEN(sun);
829 		ret->maxlen = sizeof(struct sockaddr_un);
830 		ret->buf = sun;
831 		break;
832 #endif
833 	default:
834 		break;
835 	}
836 out:
837 	free(addrstr);
838 	return ret;
839 }
840 
841 int
842 __rpc_seman2socktype(int semantics)
843 {
844 	switch (semantics) {
845 	case NC_TPI_CLTS:
846 		return SOCK_DGRAM;
847 	case NC_TPI_COTS_ORD:
848 		return SOCK_STREAM;
849 	case NC_TPI_RAW:
850 		return SOCK_RAW;
851 	default:
852 		break;
853 	}
854 
855 	return -1;
856 }
857 
858 int
859 __rpc_socktype2seman(int socktype)
860 {
861 	switch (socktype) {
862 	case SOCK_DGRAM:
863 		return NC_TPI_CLTS;
864 	case SOCK_STREAM:
865 		return NC_TPI_COTS_ORD;
866 	case SOCK_RAW:
867 		return NC_TPI_RAW;
868 	default:
869 		break;
870 	}
871 
872 	return -1;
873 }
874 
875 /*
876  * XXXX - IPv6 scope IDs can't be handled in universal addresses.
877  * Here, we compare the original server address to that of the RPC
878  * service we just received back from a call to rpcbind on the remote
879  * machine. If they are both "link local" or "site local", copy
880  * the scope id of the server address over to the service address.
881  */
882 int
883 __rpc_fixup_addr(struct netbuf *new, const struct netbuf *svc)
884 {
885 #ifdef INET6
886 	struct sockaddr *sa_new, *sa_svc;
887 	struct sockaddr_in6 *sin6_new, *sin6_svc;
888 
889 	sa_svc = (struct sockaddr *)svc->buf;
890 	sa_new = (struct sockaddr *)new->buf;
891 
892 	if (sa_new->sa_family == sa_svc->sa_family &&
893 	    sa_new->sa_family == AF_INET6) {
894 		sin6_new = (struct sockaddr_in6 *)new->buf;
895 		sin6_svc = (struct sockaddr_in6 *)svc->buf;
896 
897 		if ((IN6_IS_ADDR_LINKLOCAL(&sin6_new->sin6_addr) &&
898 		     IN6_IS_ADDR_LINKLOCAL(&sin6_svc->sin6_addr)) ||
899 		    (IN6_IS_ADDR_SITELOCAL(&sin6_new->sin6_addr) &&
900 		     IN6_IS_ADDR_SITELOCAL(&sin6_svc->sin6_addr))) {
901 			sin6_new->sin6_scope_id = sin6_svc->sin6_scope_id;
902 		}
903 	}
904 #endif
905 	return 1;
906 }
907 
908 int
909 __rpc_sockisbound(SOCKET fd)
910 {
911 	struct sockaddr_storage ss;
912 	union {
913 		struct sockaddr_in  sin;
914 		struct sockaddr_in6 sin6;
915 #ifdef AF_LOCAL
916 		struct sockaddr_un  usin;
917 #endif
918 	} u_addr;
919 	socklen_t slen;
920 
921 	slen = sizeof (struct sockaddr_storage);
922 	if (getsockname(fd, (struct sockaddr *)(void *)&ss, &slen) == SOCKET_ERROR)
923 		return 0;
924 
925 	switch (ss.ss_family) {
926 		case AF_INET:
927 			memcpy(&u_addr.sin, &ss, sizeof(u_addr.sin));
928 			return (u_addr.sin.sin_port != 0);
929 #ifdef INET6
930 		case AF_INET6:
931 			memcpy(&u_addr.sin6, &ss, sizeof(u_addr.sin6));
932 			return (u_addr.sin6.sin6_port != 0);
933 #endif
934 #ifdef AF_LOCAL
935 		case AF_LOCAL:
936 			/* XXX check this */
937 			memcpy(&u_addr.usin, &ss, sizeof(u_addr.usin));
938 			return (u_addr.usin.sun_path[0] != 0);
939 #endif
940 		default:
941 			break;
942 	}
943 
944 	return 0;
945 }
946 
947 /*
948  * Helper function to set up a netbuf
949  */
950 struct netbuf *
951 __rpc_set_netbuf(struct netbuf *nb, const void *ptr, size_t len)
952 {
953 	if (nb->len != len) {
954 		if (nb->len)
955 			mem_free(nb->buf, nb->len);
956 		nb->buf = mem_alloc(len);
957 		if (nb->buf == NULL)
958 			return NULL;
959 
960 		nb->maxlen = nb->len = len;
961 	}
962 	memcpy(nb->buf, ptr, len);
963 	return nb;
964 }
965