xref: /netbsd-src/tests/lib/libc/rpc/t_rpc.c (revision bd9cab8c091d3ca3651d0888d2e5c68b0c7f589e)
1*bd9cab8cSmrg /*	$NetBSD: t_rpc.c,v 1.11 2019/02/04 04:20:13 mrg Exp $	*/
2d7abae09Schristos 
3d7abae09Schristos #include <sys/cdefs.h>
4*bd9cab8cSmrg __RCSID("$NetBSD: t_rpc.c,v 1.11 2019/02/04 04:20:13 mrg Exp $");
5d7abae09Schristos 
6d7abae09Schristos #include <sys/types.h>
7d7abae09Schristos #include <sys/socket.h>
8d7abae09Schristos #include <rpc/rpc.h>
9d7abae09Schristos #include <stdlib.h>
105c8014d1Schristos #include <string.h>
11d7abae09Schristos #include <err.h>
12d7abae09Schristos #include <netdb.h>
13d7abae09Schristos #include <stdio.h>
145c8014d1Schristos #include <errno.h>
15d7abae09Schristos #include <unistd.h>
16d7abae09Schristos 
17d7abae09Schristos #ifndef TEST
18d7abae09Schristos #include <atf-c.h>
19d7abae09Schristos 
20*bd9cab8cSmrg #define ERRX(ev, msg, ...)	do {			\
21*bd9cab8cSmrg 	ATF_REQUIRE_MSG(0, msg, __VA_ARGS__);		\
22*bd9cab8cSmrg 	return ev;					\
23*bd9cab8cSmrg } while(/*CONSTCOND*/0)
241b48e11eSchristos 
251b48e11eSchristos #define SKIPX(ev, msg, ...)	do {			\
261b48e11eSchristos 	atf_tc_skip(msg, __VA_ARGS__);			\
277e1ddeefSchristos 	return ev;					\
281b48e11eSchristos } while(/*CONSTCOND*/0)
291b48e11eSchristos 
30d7abae09Schristos #else
317e1ddeefSchristos #define ERRX(ev, msg, ...)	errx(EXIT_FAILURE, msg, __VA_ARGS__)
327e1ddeefSchristos #define SKIPX(ev, msg, ...)	errx(EXIT_FAILURE, msg, __VA_ARGS__)
33d7abae09Schristos #endif
34d7abae09Schristos 
356e1ae275Schristos #ifdef DEBUG
366e1ae275Schristos #define DPRINTF(...)	printf(__VA_ARGS__)
376e1ae275Schristos #else
386e1ae275Schristos #define DPRINTF(...)
396e1ae275Schristos #endif
406e1ae275Schristos 
41d7abae09Schristos 
42d7abae09Schristos #define RPCBPROC_NULL 0
43d7abae09Schristos 
44d7abae09Schristos static int
reply(caddr_t replyp,struct netbuf * raddrp,struct netconfig * nconf)45d7abae09Schristos reply(caddr_t replyp, struct netbuf * raddrp, struct netconfig * nconf)
46d7abae09Schristos {
47d7abae09Schristos 	char host[NI_MAXHOST];
48d7abae09Schristos 	struct sockaddr *sock = raddrp->buf;
49d7abae09Schristos 	int error;
50d7abae09Schristos 
51d7abae09Schristos 
52d7abae09Schristos 	error = getnameinfo(sock, sock->sa_len, host, sizeof(host), NULL, 0, 0);
53d7abae09Schristos 	if (error)
54d7abae09Schristos 		warnx("Cannot resolve address (%s)", gai_strerror(error));
55d7abae09Schristos 	else
56d7abae09Schristos 		printf("response from: %s\n", host);
57d7abae09Schristos 	return 0;
58d7abae09Schristos }
59d7abae09Schristos 
606e1ae275Schristos extern bool_t __rpc_control(int, void *);
611b48e11eSchristos 
62d7abae09Schristos static void
onehost(const char * host,const char * transp)63d7abae09Schristos onehost(const char *host, const char *transp)
64d7abae09Schristos {
65d7abae09Schristos 	CLIENT         *clnt;
66d7abae09Schristos 	struct netbuf   addr;
67d7abae09Schristos 	struct timeval  tv;
68d7abae09Schristos 
691b48e11eSchristos 	/*
701b48e11eSchristos 	 * Magic!
711b48e11eSchristos 	 */
721b48e11eSchristos 	tv.tv_sec = 0;
731b48e11eSchristos 	tv.tv_usec = 500000;
741b48e11eSchristos #define CLCR_SET_RPCB_TIMEOUT   2
751b48e11eSchristos 	__rpc_control(CLCR_SET_RPCB_TIMEOUT, &tv);
76d7abae09Schristos 
77d7abae09Schristos 	if ((clnt = clnt_create(host, RPCBPROG, RPCBVERS, transp)) == NULL)
787e1ddeefSchristos 		SKIPX(, "clnt_create (%s)", clnt_spcreateerror(""));
79d7abae09Schristos 
80d7abae09Schristos 	tv.tv_sec = 1;
81d7abae09Schristos 	tv.tv_usec = 0;
82d7abae09Schristos 	if (clnt_call(clnt, RPCBPROC_NULL, xdr_void, NULL, xdr_void, NULL, tv)
83d7abae09Schristos 	    != RPC_SUCCESS)
847e1ddeefSchristos 		ERRX(, "clnt_call (%s)", clnt_sperror(clnt, ""));
85d7abae09Schristos 	clnt_control(clnt, CLGET_SVC_ADDR, (char *) &addr);
86d7abae09Schristos 	reply(NULL, &addr, NULL);
87d7abae09Schristos }
88d7abae09Schristos 
896e1ae275Schristos #define PROGNUM 0x81
906e1ae275Schristos #define VERSNUM 0x01
916e1ae275Schristos #define PLUSONE 1
925c8014d1Schristos #define DESTROY 2
936e1ae275Schristos 
945c8014d1Schristos static struct timeval 	tout = {1, 0};
956e1ae275Schristos 
966e1ae275Schristos static void
server(struct svc_req * rqstp,SVCXPRT * transp)976e1ae275Schristos server(struct svc_req *rqstp, SVCXPRT *transp)
986e1ae275Schristos {
996e1ae275Schristos 	int num;
1006e1ae275Schristos 
1016e1ae275Schristos 	DPRINTF("Starting server\n");
1026e1ae275Schristos 
1036e1ae275Schristos 	switch (rqstp->rq_proc) {
1046e1ae275Schristos 	case NULLPROC:
1056e1ae275Schristos 		if (!svc_sendreply(transp, (xdrproc_t)xdr_void, NULL))
1067e1ddeefSchristos 			ERRX(, "svc_sendreply failed %d", 0);
1076e1ae275Schristos 		return;
1086e1ae275Schristos 	case PLUSONE:
1096e1ae275Schristos 		break;
1105c8014d1Schristos 	case DESTROY:
1115c8014d1Schristos 		if (!svc_sendreply(transp, (xdrproc_t)xdr_void, NULL))
1127e1ddeefSchristos 			ERRX(, "svc_sendreply failed %d", 0);
1135c8014d1Schristos 		svc_destroy(transp);
1145c8014d1Schristos 		exit(0);
1156e1ae275Schristos 	default:
1166e1ae275Schristos 		svcerr_noproc(transp);
1176e1ae275Schristos 		return;
1186e1ae275Schristos 	}
1196e1ae275Schristos 
1206e1ae275Schristos 	if (!svc_getargs(transp, (xdrproc_t)xdr_int, (void *)&num)) {
1216e1ae275Schristos 		svcerr_decode(transp);
1226e1ae275Schristos 		return;
1236e1ae275Schristos 	}
1246e1ae275Schristos 	DPRINTF("About to increment\n");
1256e1ae275Schristos 	num++;
1266e1ae275Schristos 	if (!svc_sendreply(transp, (xdrproc_t)xdr_int, (void *)&num))
1277e1ddeefSchristos 		ERRX(, "svc_sendreply failed %d", 1);
1286e1ae275Schristos 	DPRINTF("Leaving server procedure.\n");
1296e1ae275Schristos }
1306e1ae275Schristos 
131b2d56358Schristos static int
rawtest(const char * arg)1326e1ae275Schristos rawtest(const char *arg)
1336e1ae275Schristos {
1346e1ae275Schristos 	CLIENT         *clnt;
1356e1ae275Schristos 	SVCXPRT        *svc;
1366e1ae275Schristos 	int 		num, resp;
1376e1ae275Schristos 	enum clnt_stat  rv;
1386e1ae275Schristos 
1396e1ae275Schristos 	if (arg)
1406e1ae275Schristos 		num = atoi(arg);
1416e1ae275Schristos 	else
1426e1ae275Schristos 		num = 0;
1436e1ae275Schristos 
1446e1ae275Schristos 	svc = svc_raw_create();
1456e1ae275Schristos 	if (svc == NULL)
1467b88a1a0Schristos 		ERRX(EXIT_FAILURE, "Cannot create server %d", num);
1476e1ae275Schristos 	if (!svc_reg(svc, PROGNUM, VERSNUM, server, NULL))
1487b88a1a0Schristos 		ERRX(EXIT_FAILURE, "Cannot register server %d", num);
1496e1ae275Schristos 
1506e1ae275Schristos 	clnt = clnt_raw_create(PROGNUM, VERSNUM);
1516e1ae275Schristos 	if (clnt == NULL)
1526e1ae275Schristos 		ERRX(EXIT_FAILURE, "%s",
1536e1ae275Schristos 		    clnt_spcreateerror("clnt_raw_create"));
1546e1ae275Schristos 	rv = clnt_call(clnt, PLUSONE, (xdrproc_t)xdr_int, (void *)&num,
1556e1ae275Schristos 	    (xdrproc_t)xdr_int, (void *)&resp, tout);
1566e1ae275Schristos 	if (rv != RPC_SUCCESS)
1576e1ae275Schristos 		ERRX(EXIT_FAILURE, "clnt_call: %s", clnt_sperrno(rv));
1586e1ae275Schristos 	DPRINTF("Got %d\n", resp);
1596e1ae275Schristos 	clnt_destroy(clnt);
1606e1ae275Schristos 	svc_destroy(svc);
1616e1ae275Schristos 	if (++num != resp)
1626e1ae275Schristos 		ERRX(EXIT_FAILURE, "expected %d got %d", num, resp);
1636e1ae275Schristos 
1646e1ae275Schristos 	return EXIT_SUCCESS;
1656e1ae275Schristos }
1666e1ae275Schristos 
1675c8014d1Schristos static int
regtest(const char * hostname,const char * transp,const char * arg,int p)1685c8014d1Schristos regtest(const char *hostname, const char *transp, const char *arg, int p)
1695c8014d1Schristos {
1705c8014d1Schristos 	CLIENT         *clnt;
1715c8014d1Schristos 	int 		num, resp;
1725c8014d1Schristos 	enum clnt_stat  rv;
1735c8014d1Schristos 	pid_t		pid;
1745c8014d1Schristos 
1755c8014d1Schristos 	if (arg)
1765c8014d1Schristos 		num = atoi(arg);
1775c8014d1Schristos 	else
1785c8014d1Schristos 		num = 0;
1795c8014d1Schristos 
1805c8014d1Schristos 	svc_fdset_init(p ? SVC_FDSET_POLL : 0);
1815c8014d1Schristos 	if (!svc_create(server, PROGNUM, VERSNUM, transp))
1827e1ddeefSchristos 		SKIPX(EXIT_FAILURE, "Cannot create server %d", num);
1835c8014d1Schristos 
1845c8014d1Schristos 	switch ((pid = fork())) {
1855c8014d1Schristos 	case 0:
1865c8014d1Schristos 		DPRINTF("Calling svc_run\n");
1875c8014d1Schristos 		svc_run();
188e700dcfbSchristos 		ERRX(EXIT_FAILURE, "svc_run returned %d!", num);
1895c8014d1Schristos 	case -1:
1905c8014d1Schristos 		ERRX(EXIT_FAILURE, "Fork failed (%s)", strerror(errno));
1915c8014d1Schristos 	default:
1925c8014d1Schristos 		sleep(1);
1935c8014d1Schristos 		break;
1945c8014d1Schristos 	}
1955c8014d1Schristos 
1965c8014d1Schristos 	DPRINTF("Initializing client\n");
1975c8014d1Schristos 	clnt = clnt_create(hostname, PROGNUM, VERSNUM, transp);
1985c8014d1Schristos 	if (clnt == NULL)
1995c8014d1Schristos 		ERRX(EXIT_FAILURE, "%s",
2005c8014d1Schristos 		    clnt_spcreateerror("clnt_raw_create"));
2015c8014d1Schristos 	rv = clnt_call(clnt, PLUSONE, (xdrproc_t)xdr_int, (void *)&num,
2025c8014d1Schristos 	    (xdrproc_t)xdr_int, (void *)&resp, tout);
2035c8014d1Schristos 	if (rv != RPC_SUCCESS)
2045c8014d1Schristos 		ERRX(EXIT_FAILURE, "clnt_call: %s", clnt_sperrno(rv));
2055c8014d1Schristos 	DPRINTF("Got %d\n", resp);
2065c8014d1Schristos 	if (++num != resp)
2075c8014d1Schristos 		ERRX(EXIT_FAILURE, "expected %d got %d", num, resp);
2085c8014d1Schristos 	rv = clnt_call(clnt, DESTROY, (xdrproc_t)xdr_void, NULL,
2095c8014d1Schristos 	    (xdrproc_t)xdr_void, NULL, tout);
2105c8014d1Schristos 	if (rv != RPC_SUCCESS)
2115c8014d1Schristos 		ERRX(EXIT_FAILURE, "clnt_call: %s", clnt_sperrno(rv));
2125c8014d1Schristos 	clnt_destroy(clnt);
2135c8014d1Schristos 
2145c8014d1Schristos 	return EXIT_SUCCESS;
2155c8014d1Schristos }
2165c8014d1Schristos 
2176e1ae275Schristos 
218d7abae09Schristos #ifdef TEST
219d7abae09Schristos static void
allhosts(const char * transp)2206e1ae275Schristos allhosts(const char *transp)
221d7abae09Schristos {
222d7abae09Schristos 	enum clnt_stat  clnt_stat;
223d7abae09Schristos 
224d7abae09Schristos 	clnt_stat = rpc_broadcast(RPCBPROG, RPCBVERS, RPCBPROC_NULL,
225d7abae09Schristos 	    (xdrproc_t)xdr_void, NULL, (xdrproc_t)xdr_void,
226d7abae09Schristos 	    NULL, (resultproc_t)reply, transp);
227d7abae09Schristos 	if (clnt_stat != RPC_SUCCESS && clnt_stat != RPC_TIMEDOUT)
228d7abae09Schristos 		ERRX(EXIT_FAILURE, "%s", clnt_sperrno(clnt_stat));
229d7abae09Schristos }
230d7abae09Schristos 
231d7abae09Schristos int
main(int argc,char * argv[])232d7abae09Schristos main(int argc, char *argv[])
233d7abae09Schristos {
234d7abae09Schristos 	int             ch;
2355c8014d1Schristos 	int		s, p;
236d7abae09Schristos 	const char     *transp = "udp";
237d7abae09Schristos 
2385c8014d1Schristos 	p = s = 0;
2395c8014d1Schristos 	while ((ch = getopt(argc, argv, "prstu")) != -1)
240d7abae09Schristos 		switch (ch) {
2415c8014d1Schristos 		case 'p':
2425c8014d1Schristos 			p = 1;
2435c8014d1Schristos 			break;
2445c8014d1Schristos 		case 's':
2455c8014d1Schristos 			s = 1;
2465c8014d1Schristos 			break;
247d7abae09Schristos 		case 't':
248d7abae09Schristos 			transp = "tcp";
249d7abae09Schristos 			break;
250d7abae09Schristos 		case 'u':
251d7abae09Schristos 			transp = "udp";
252d7abae09Schristos 			break;
2536e1ae275Schristos 		case 'r':
2546e1ae275Schristos 			transp = NULL;
2556e1ae275Schristos 			break;
256d7abae09Schristos 		default:
2575c8014d1Schristos 			fprintf(stderr,
2585c8014d1Schristos 			    "Usage: %s -[r|s|t|u] [<hostname>...]\n",
259d7abae09Schristos 			    getprogname());
260d7abae09Schristos 			return EXIT_FAILURE;
261d7abae09Schristos 		}
262d7abae09Schristos 
2636e1ae275Schristos 	if (argc == optind) {
2646e1ae275Schristos 		if  (transp)
2656e1ae275Schristos 			allhosts(transp);
266d7abae09Schristos 		else
2676e1ae275Schristos 			rawtest(NULL);
2686e1ae275Schristos 	} else {
2696e1ae275Schristos 		for (; optind < argc; optind++) {
2706e1ae275Schristos 			if (transp)
2715c8014d1Schristos 				s == 0 ?
2725c8014d1Schristos 				    onehost(argv[optind], transp) :
2735c8014d1Schristos 				    regtest(argv[optind], transp, "1", p);
2746e1ae275Schristos 			else
2756e1ae275Schristos 				rawtest(argv[optind]);
2766e1ae275Schristos 		}
2776e1ae275Schristos 	}
278d7abae09Schristos 
279d7abae09Schristos 	return EXIT_SUCCESS;
280d7abae09Schristos }
281d7abae09Schristos 
282d7abae09Schristos #else
283d7abae09Schristos 
284d7abae09Schristos ATF_TC(get_svc_addr_tcp);
ATF_TC_HEAD(get_svc_addr_tcp,tc)285d7abae09Schristos ATF_TC_HEAD(get_svc_addr_tcp, tc)
286d7abae09Schristos {
287d7abae09Schristos 	atf_tc_set_md_var(tc, "descr", "Checks CLGET_SVC_ADDR for tcp");
288d7abae09Schristos 
289d7abae09Schristos }
290d7abae09Schristos 
ATF_TC_BODY(get_svc_addr_tcp,tc)291d7abae09Schristos ATF_TC_BODY(get_svc_addr_tcp, tc)
292d7abae09Schristos {
293d7abae09Schristos 	onehost("localhost", "tcp");
294d7abae09Schristos 
295d7abae09Schristos }
296d7abae09Schristos 
297d7abae09Schristos ATF_TC(get_svc_addr_udp);
ATF_TC_HEAD(get_svc_addr_udp,tc)298d7abae09Schristos ATF_TC_HEAD(get_svc_addr_udp, tc)
299d7abae09Schristos {
300d7abae09Schristos 	atf_tc_set_md_var(tc, "descr", "Checks CLGET_SVC_ADDR for udp");
301d7abae09Schristos }
302d7abae09Schristos 
ATF_TC_BODY(get_svc_addr_udp,tc)303d7abae09Schristos ATF_TC_BODY(get_svc_addr_udp, tc)
304d7abae09Schristos {
305d7abae09Schristos 	onehost("localhost", "udp");
306d7abae09Schristos 
307d7abae09Schristos }
308d7abae09Schristos 
3096e1ae275Schristos ATF_TC(raw);
ATF_TC_HEAD(raw,tc)3106e1ae275Schristos ATF_TC_HEAD(raw, tc)
3116e1ae275Schristos {
3126e1ae275Schristos 	atf_tc_set_md_var(tc, "descr", "Checks svc raw");
3136e1ae275Schristos }
3146e1ae275Schristos 
ATF_TC_BODY(raw,tc)3156e1ae275Schristos ATF_TC_BODY(raw, tc)
3166e1ae275Schristos {
3176e1ae275Schristos 	rawtest(NULL);
3186e1ae275Schristos 
3196e1ae275Schristos }
3205c8014d1Schristos 
3215c8014d1Schristos ATF_TC(tcp);
ATF_TC_HEAD(tcp,tc)3225c8014d1Schristos ATF_TC_HEAD(tcp, tc)
3235c8014d1Schristos {
3245c8014d1Schristos 	atf_tc_set_md_var(tc, "descr", "Checks svc tcp (select)");
3255c8014d1Schristos }
3265c8014d1Schristos 
ATF_TC_BODY(tcp,tc)3275c8014d1Schristos ATF_TC_BODY(tcp, tc)
3285c8014d1Schristos {
329871846c7Schristos 	regtest("localhost", "tcp", "1", 0);
3305c8014d1Schristos 
3315c8014d1Schristos }
3325c8014d1Schristos 
3335c8014d1Schristos ATF_TC(udp);
ATF_TC_HEAD(udp,tc)3345c8014d1Schristos ATF_TC_HEAD(udp, tc)
3355c8014d1Schristos {
3365c8014d1Schristos 	atf_tc_set_md_var(tc, "descr", "Checks svc udp (select)");
3375c8014d1Schristos }
3385c8014d1Schristos 
ATF_TC_BODY(udp,tc)3395c8014d1Schristos ATF_TC_BODY(udp, tc)
3405c8014d1Schristos {
341871846c7Schristos 	regtest("localhost", "udp", "1", 0);
3425c8014d1Schristos 
3435c8014d1Schristos }
3445c8014d1Schristos 
3455c8014d1Schristos ATF_TC(tcp_poll);
ATF_TC_HEAD(tcp_poll,tc)3465c8014d1Schristos ATF_TC_HEAD(tcp_poll, tc)
3475c8014d1Schristos {
3485c8014d1Schristos 	atf_tc_set_md_var(tc, "descr", "Checks svc tcp (poll)");
3495c8014d1Schristos }
3505c8014d1Schristos 
ATF_TC_BODY(tcp_poll,tc)3515c8014d1Schristos ATF_TC_BODY(tcp_poll, tc)
3525c8014d1Schristos {
353871846c7Schristos 	regtest("localhost", "tcp", "1", 1);
3545c8014d1Schristos 
3555c8014d1Schristos }
3565c8014d1Schristos 
3575c8014d1Schristos ATF_TC(udp_poll);
ATF_TC_HEAD(udp_poll,tc)3585c8014d1Schristos ATF_TC_HEAD(udp_poll, tc)
3595c8014d1Schristos {
3605c8014d1Schristos 	atf_tc_set_md_var(tc, "descr", "Checks svc udp (poll)");
3615c8014d1Schristos }
3625c8014d1Schristos 
ATF_TC_BODY(udp_poll,tc)3635c8014d1Schristos ATF_TC_BODY(udp_poll, tc)
3645c8014d1Schristos {
365871846c7Schristos 	regtest("localhost", "udp", "1", 1);
3665c8014d1Schristos 
3675c8014d1Schristos }
3685c8014d1Schristos 
ATF_TP_ADD_TCS(tp)369d7abae09Schristos ATF_TP_ADD_TCS(tp)
370d7abae09Schristos {
371d7abae09Schristos 	ATF_TP_ADD_TC(tp, get_svc_addr_udp);
372d7abae09Schristos 	ATF_TP_ADD_TC(tp, get_svc_addr_tcp);
3736e1ae275Schristos 	ATF_TP_ADD_TC(tp, raw);
3745c8014d1Schristos 	ATF_TP_ADD_TC(tp, tcp);
3755c8014d1Schristos 	ATF_TP_ADD_TC(tp, udp);
3765c8014d1Schristos 	ATF_TP_ADD_TC(tp, tcp_poll);
3775c8014d1Schristos 	ATF_TP_ADD_TC(tp, udp_poll);
378d7abae09Schristos 
379d7abae09Schristos 	return atf_no_error();
380d7abae09Schristos }
381d7abae09Schristos 
382d7abae09Schristos #endif
383