xref: /freebsd/usr.sbin/rpc.tlsservd/rpc.tlscommon.c (revision e32fecd0c2c3ee37c47ee100f169e7eb0282a873)
1 /*-
2  * SPDX-License-Identifier: BSD-2-Clause-FreeBSD
3  *
4  * Copyright (c) 2021 Rick Macklem
5  *
6  * Redistribution and use in source and binary forms, with or without
7  * modification, are permitted provided that the following conditions
8  * are met:
9  * 1. Redistributions of source code must retain the above copyright
10  *    notice, this list of conditions and the following disclaimer.
11  * 2. Redistributions in binary form must reproduce the above copyright
12  *    notice, this list of conditions and the following disclaimer in the
13  *    documentation and/or other materials provided with the distribution.
14  *
15  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
16  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
17  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
18  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
19  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
20  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
21  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
22  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
23  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
24  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
25  * SUCH DAMAGE.
26  *
27  */
28 
29 #include <sys/cdefs.h>
30 __FBSDID("$FreeBSD$");
31 
32 #include <sys/queue.h>
33 #include <sys/syslog.h>
34 #include <sys/select.h>
35 #include <sys/time.h>
36 
37 #include <netdb.h>
38 #include <signal.h>
39 #include <stdarg.h>
40 #include <stdbool.h>
41 #include <string.h>
42 
43 #include <rpc/rpc.h>
44 
45 #include <openssl/opensslconf.h>
46 #include <openssl/bio.h>
47 #include <openssl/ssl.h>
48 #include <openssl/err.h>
49 #include <openssl/x509v3.h>
50 
51 #include "rpc.tlscommon.h"
52 
53 /*
54  * How long to delay a reload of the CRL when there are RPC request(s)
55  * to process, in usec.  Must be less than 1second.
56  */
57 #define	RELOADDELAY	250000
58 
59 void
60 rpctls_svc_run(void)
61 {
62 	int ret;
63 	struct timeval tv;
64 	fd_set readfds;
65 	uint64_t curtime, nexttime;
66 	struct timespec tp;
67 	sigset_t sighup_mask;
68 
69 	/* Expand svc_run() here so that we can call rpctls_loadcrlfile(). */
70 	curtime = nexttime = 0;
71 	sigemptyset(&sighup_mask);
72 	sigaddset(&sighup_mask, SIGHUP);
73 	for (;;) {
74 		clock_gettime(CLOCK_MONOTONIC, &tp);
75 		curtime = tp.tv_sec;
76 		curtime = curtime * 1000000 + tp.tv_nsec / 1000;
77 		sigprocmask(SIG_BLOCK, &sighup_mask, NULL);
78 		if (rpctls_gothup && curtime >= nexttime) {
79 			rpctls_gothup = false;
80 			sigprocmask(SIG_UNBLOCK, &sighup_mask, NULL);
81 			ret = rpctls_loadcrlfile(rpctls_ctx);
82 			if (ret != 0)
83 				rpctls_checkcrl();
84 			else
85 				rpctls_verbose_out("rpc.tlsservd: Can't "
86 				    "reload CRLfile\n");
87 			clock_gettime(CLOCK_MONOTONIC, &tp);
88 			nexttime = tp.tv_sec;
89 			nexttime = nexttime * 1000000 + tp.tv_nsec / 1000 +
90 			    RELOADDELAY;
91 		} else
92 			sigprocmask(SIG_UNBLOCK, &sighup_mask, NULL);
93 
94 		/*
95 		 * If a reload is pending, poll for received request(s),
96 		 * otherwise set a RELOADDELAY timeout, since a SIGHUP
97 		 * could be processed between the got_sighup test and
98 		 * the select() system call.
99 		 */
100 		tv.tv_sec = 0;
101 		if (rpctls_gothup)
102 			tv.tv_usec = 0;
103 		else
104 			tv.tv_usec = RELOADDELAY;
105 		readfds = svc_fdset;
106 		switch (select(svc_maxfd + 1, &readfds, NULL, NULL, &tv)) {
107 		case -1:
108 			if (errno == EINTR) {
109 				/* Allow a reload now. */
110 				nexttime = 0;
111 				continue;
112 			}
113 			syslog(LOG_ERR, "rpc.tls daemon died: select: %m");
114 			exit(1);
115 		case 0:
116 			/* Allow a reload now. */
117 			nexttime = 0;
118 			continue;
119 		default:
120 			svc_getreqset(&readfds);
121 		}
122 	}
123 }
124 
125 /*
126  * (re)load the CRLfile into the certificate verification store.
127  */
128 int
129 rpctls_loadcrlfile(SSL_CTX *ctx)
130 {
131 	X509_STORE *certstore;
132 	X509_LOOKUP *certlookup;
133 	int ret;
134 
135 	if ((rpctls_verify_cafile != NULL ||
136 	    rpctls_verify_capath != NULL) &&
137 	    rpctls_crlfile != NULL) {
138 		certstore = SSL_CTX_get_cert_store(ctx);
139 		certlookup = X509_STORE_add_lookup(
140 		    certstore, X509_LOOKUP_file());
141 		ret = 0;
142 		if (certlookup != NULL)
143 			ret = X509_load_crl_file(certlookup,
144 			    rpctls_crlfile, X509_FILETYPE_PEM);
145 		if (ret != 0)
146 			ret = X509_STORE_set_flags(certstore,
147 			    X509_V_FLAG_CRL_CHECK |
148 			    X509_V_FLAG_CRL_CHECK_ALL);
149 		if (ret == 0) {
150 			rpctls_verbose_out(
151 			    "rpctls_loadcrlfile: Can't"
152 			    " load CRLfile=%s\n",
153 			    rpctls_crlfile);
154 			return (ret);
155 		}
156 	}
157 	return (1);
158 }
159 
160 /*
161  * Read the CRL file and check for any extant connections
162  * that might now be revoked.
163  */
164 void
165 rpctls_checkcrl(void)
166 {
167 	struct ssl_entry *slp;
168 	BIO *infile;
169 	X509_CRL *crl;
170 	X509_REVOKED *revoked;
171 	char *cp, *cp2, nullstr[1];
172 	int ret;
173 
174 	if (rpctls_crlfile == NULL || (rpctls_verify_cafile == NULL &&
175 	    rpctls_verify_capath == NULL))
176 		return;
177 	infile = BIO_new(BIO_s_file());
178 	if (infile == NULL) {
179 		rpctls_verbose_out("rpctls_checkcrl: Cannot BIO_new\n");
180 		return;
181 	}
182 	ret = BIO_read_filename(infile, rpctls_crlfile);
183 	if (ret != 1) {
184 		rpctls_verbose_out("rpctls_checkcrl: Cannot read CRL file\n");
185 		BIO_free(infile);
186 		return;
187 	}
188 
189 	nullstr[0] = '\0';
190 	for (crl = PEM_read_bio_X509_CRL(infile, NULL, NULL, nullstr);
191 	    crl != NULL; crl = PEM_read_bio_X509_CRL(infile, NULL, NULL,
192 	    nullstr)) {
193 		LIST_FOREACH(slp, &rpctls_ssllist, next) {
194 			if (slp->cert != NULL) {
195 				ret = X509_CRL_get0_by_cert(crl, &revoked,
196 				    slp->cert);
197 				/*
198 				 * Do a shutdown on the socket, so that it
199 				 * can no longer be used.  The kernel RPC
200 				 * code will notice the socket is disabled
201 				 * and will do a disconnect upcall, which will
202 				 * close the socket.
203 				 */
204 				if (ret == 1) {
205 					cp2 = X509_NAME_oneline(
206 					    X509_get_subject_name(slp->cert),
207 					    NULL, 0);
208 					cp = X509_NAME_oneline(
209 					    X509_get_issuer_name(slp->cert),
210 					    NULL, 0);
211 					if (rpctls_debug_level == 0)
212 						syslog(LOG_INFO | LOG_DAEMON,
213 						    "rpctls_daemon: Certificate"
214 						    " Revoked "
215 						    "issuerName=%s "
216 						    "subjectName=%s: "
217 						    "TCP connection closed",
218 						    cp, cp2);
219 					else
220 						fprintf(stderr,
221 						    "rpctls_daemon: Certificate"
222 						    " Revoked "
223 						    "issuerName=%s "
224 						    "subjectName=%s: "
225 						    "TCP connection closed",
226 						    cp, cp2);
227 					shutdown(slp->s, SHUT_WR);
228 					slp->shutoff = true;
229 				}
230 			}
231 		}
232 		X509_CRL_free(crl);
233 	}
234 	BIO_free(infile);
235 }
236 
237 void
238 rpctls_verbose_out(const char *fmt, ...)
239 {
240 	va_list ap;
241 
242 	if (rpctls_verbose) {
243 		va_start(ap, fmt);
244 		if (rpctls_debug_level == 0)
245 			vsyslog(LOG_INFO | LOG_DAEMON, fmt, ap);
246 		else
247 			vfprintf(stderr, fmt, ap);
248 		va_end(ap);
249 	}
250 }
251 
252 /*
253  * Check a IP address against any host address in the
254  * certificate.  Basically getnameinfo(3) and
255  * X509_check_host().
256  */
257 int
258 rpctls_checkhost(struct sockaddr *sad, X509 *cert, unsigned int wildcard)
259 {
260 	char hostnam[NI_MAXHOST];
261 	int ret;
262 
263 	if (getnameinfo((const struct sockaddr *)sad,
264 	    sad->sa_len, hostnam, sizeof(hostnam),
265 	    NULL, 0, NI_NAMEREQD) != 0)
266 		return (0);
267 	rpctls_verbose_out("rpctls_checkhost: DNS %s\n",
268 	    hostnam);
269 	ret = X509_check_host(cert, hostnam, strlen(hostnam),
270 	    wildcard, NULL);
271 	return (ret);
272 }
273 
274 /*
275  * Get the peer's IP address.
276  */
277 int
278 rpctls_gethost(int s, struct sockaddr *sad, char *hostip, size_t hostlen)
279 {
280 	socklen_t slen;
281 	int ret;
282 
283 	slen = sizeof(struct sockaddr_storage);
284 	if (getpeername(s, sad, &slen) < 0)
285 		return (0);
286 	ret = 0;
287 	if (getnameinfo((const struct sockaddr *)sad,
288 	    sad->sa_len, hostip, hostlen,
289 	    NULL, 0, NI_NUMERICHOST) == 0) {
290 		rpctls_verbose_out("rpctls_gethost: %s\n",
291 		    hostip);
292 		ret = 1;
293 	}
294 	return (ret);
295 }
296