Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * fe-secure.c
4 : * functions related to setting up a secure connection to the backend.
5 : * Secure connections are expected to provide confidentiality,
6 : * message integrity and endpoint authentication.
7 : *
8 : *
9 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
10 : * Portions Copyright (c) 1994, Regents of the University of California
11 : *
12 : *
13 : * IDENTIFICATION
14 : * src/interfaces/libpq/fe-secure.c
15 : *
16 : *-------------------------------------------------------------------------
17 : */
18 :
19 : #include "postgres_fe.h"
20 :
21 : #include <signal.h>
22 : #include <fcntl.h>
23 : #include <ctype.h>
24 :
25 : #ifdef WIN32
26 : #include "win32.h"
27 : #else
28 : #include <sys/socket.h>
29 : #include <unistd.h>
30 : #include <netdb.h>
31 : #include <netinet/in.h>
32 : #include <netinet/tcp.h>
33 : #include <arpa/inet.h>
34 : #endif
35 :
36 : #include <sys/stat.h>
37 :
38 : #ifdef WIN32
39 : #include "pthread-win32.h"
40 : #else
41 : #include <pthread.h>
42 : #endif
43 :
44 : #include "fe-auth.h"
45 : #include "libpq-fe.h"
46 : #include "libpq-int.h"
47 :
48 : /*
49 : * Macros to handle disabling and then restoring the state of SIGPIPE handling.
50 : * On Windows, these are all no-ops since there's no SIGPIPEs.
51 : */
52 :
53 : #ifndef WIN32
54 :
55 : #define SIGPIPE_MASKED(conn) ((conn)->sigpipe_so || (conn)->sigpipe_flag)
56 :
57 : struct sigpipe_info
58 : {
59 : sigset_t oldsigmask;
60 : bool sigpipe_pending;
61 : bool got_epipe;
62 : };
63 :
64 : #define DECLARE_SIGPIPE_INFO(spinfo) struct sigpipe_info spinfo
65 :
66 : #define DISABLE_SIGPIPE(conn, spinfo, failaction) \
67 : do { \
68 : (spinfo).got_epipe = false; \
69 : if (!SIGPIPE_MASKED(conn)) \
70 : { \
71 : if (pq_block_sigpipe(&(spinfo).oldsigmask, \
72 : &(spinfo).sigpipe_pending) < 0) \
73 : failaction; \
74 : } \
75 : } while (0)
76 :
77 : #define REMEMBER_EPIPE(spinfo, cond) \
78 : do { \
79 : if (cond) \
80 : (spinfo).got_epipe = true; \
81 : } while (0)
82 :
83 : #define RESTORE_SIGPIPE(conn, spinfo) \
84 : do { \
85 : if (!SIGPIPE_MASKED(conn)) \
86 : pq_reset_sigpipe(&(spinfo).oldsigmask, (spinfo).sigpipe_pending, \
87 : (spinfo).got_epipe); \
88 : } while (0)
89 : #else /* WIN32 */
90 :
91 : #define DECLARE_SIGPIPE_INFO(spinfo)
92 : #define DISABLE_SIGPIPE(conn, spinfo, failaction)
93 : #define REMEMBER_EPIPE(spinfo, cond)
94 : #define RESTORE_SIGPIPE(conn, spinfo)
95 : #endif /* WIN32 */
96 :
97 : /* ------------------------------------------------------------ */
98 : /* Procedures common to all secure sessions */
99 : /* ------------------------------------------------------------ */
100 :
101 :
102 : int
103 4 : PQsslInUse(PGconn *conn)
104 : {
105 4 : if (!conn)
106 0 : return 0;
107 4 : return conn->ssl_in_use;
108 : }
109 :
110 : /*
111 : * Exported function to allow application to tell us it's already initialized
112 : * OpenSSL. Since OpenSSL 1.1.0 it is no longer required to explicitly
113 : * initialize libssl and libcrypto, so this is a no-op. This function remains
114 : * for backwards API compatibility.
115 : */
116 : void
117 0 : PQinitSSL(int do_init)
118 : {
119 : /* no-op */
120 0 : }
121 :
122 : /*
123 : * Exported function to allow application to tell us it's already initialized
124 : * OpenSSL. Since OpenSSL 1.1.0 it is no longer required to explicitly
125 : * initialize libssl and libcrypto, so this is a no-op. This function remains
126 : * for backwards API compatibility.
127 : */
128 : void
129 0 : PQinitOpenSSL(int do_ssl, int do_crypto)
130 : {
131 : /* no-op */
132 0 : }
133 :
134 : /*
135 : * Begin or continue negotiating a secure session.
136 : */
137 : PostgresPollingStatusType
138 750 : pqsecure_open_client(PGconn *conn)
139 : {
140 : #ifdef USE_SSL
141 750 : return pgtls_open_client(conn);
142 : #else
143 : /* shouldn't get here */
144 : return PGRES_POLLING_FAILED;
145 : #endif
146 : }
147 :
148 : /*
149 : * Close secure session.
150 : */
151 : void
152 49498 : pqsecure_close(PGconn *conn)
153 : {
154 : #ifdef USE_SSL
155 49498 : pgtls_close(conn);
156 : #endif
157 49498 : }
158 :
159 : /*
160 : * Read data from a secure connection.
161 : *
162 : * On failure, this function is responsible for appending a suitable message
163 : * to conn->errorMessage. The caller must still inspect errno, but only
164 : * to determine whether to continue/retry after error.
165 : */
166 : ssize_t
167 2066358 : pqsecure_read(PGconn *conn, void *ptr, size_t len)
168 : {
169 : ssize_t n;
170 :
171 : #ifdef USE_SSL
172 2066358 : if (conn->ssl_in_use)
173 : {
174 562 : n = pgtls_read(conn, ptr, len);
175 : }
176 : else
177 : #endif
178 : #ifdef ENABLE_GSS
179 : if (conn->gssenc)
180 : {
181 : n = pg_GSS_read(conn, ptr, len);
182 : }
183 : else
184 : #endif
185 : {
186 2065796 : n = pqsecure_raw_read(conn, ptr, len);
187 : }
188 :
189 2066358 : return n;
190 : }
191 :
192 : ssize_t
193 2071036 : pqsecure_raw_read(PGconn *conn, void *ptr, size_t len)
194 : {
195 : ssize_t n;
196 2071036 : int result_errno = 0;
197 : char sebuf[PG_STRERROR_R_BUFLEN];
198 :
199 2071036 : SOCK_ERRNO_SET(0);
200 :
201 2071036 : n = recv(conn->sock, ptr, len, 0);
202 :
203 2071036 : if (n < 0)
204 : {
205 609334 : result_errno = SOCK_ERRNO;
206 :
207 : /* Set error message if appropriate */
208 609334 : switch (result_errno)
209 : {
210 : #ifdef EAGAIN
211 609318 : case EAGAIN:
212 : #endif
213 : #if defined(EWOULDBLOCK) && (!defined(EAGAIN) || (EWOULDBLOCK != EAGAIN))
214 : case EWOULDBLOCK:
215 : #endif
216 : case EINTR:
217 : /* no error message, caller is expected to retry */
218 609318 : break;
219 :
220 16 : case EPIPE:
221 : case ECONNRESET:
222 16 : libpq_append_conn_error(conn, "server closed the connection unexpectedly\n"
223 : "\tThis probably means the server terminated abnormally\n"
224 : "\tbefore or while processing the request.");
225 16 : break;
226 :
227 0 : case 0:
228 : /* If errno didn't get set, treat it as regular EOF */
229 0 : n = 0;
230 0 : break;
231 :
232 0 : default:
233 0 : libpq_append_conn_error(conn, "could not receive data from server: %s",
234 : SOCK_STRERROR(result_errno,
235 : sebuf, sizeof(sebuf)));
236 0 : break;
237 : }
238 1461702 : }
239 :
240 : /* ensure we return the intended errno to caller */
241 2071036 : SOCK_ERRNO_SET(result_errno);
242 :
243 2071036 : return n;
244 : }
245 :
246 : /*
247 : * Write data to a secure connection.
248 : *
249 : * Returns the number of bytes written, or a negative value (with errno
250 : * set) upon failure. The write count could be less than requested.
251 : *
252 : * Note that socket-level hard failures are masked from the caller,
253 : * instead setting conn->write_failed and storing an error message
254 : * in conn->write_err_msg; see pqsecure_raw_write. This allows us to
255 : * postpone reporting of write failures until we're sure no error
256 : * message is available from the server.
257 : *
258 : * However, errors detected in the SSL or GSS management level are reported
259 : * via a negative result, with message appended to conn->errorMessage.
260 : * It's frequently unclear whether such errors should be considered read or
261 : * write errors, so we don't attempt to postpone reporting them.
262 : *
263 : * The caller must still inspect errno upon failure, but only to determine
264 : * whether to continue/retry; a message has been saved someplace in any case.
265 : */
266 : ssize_t
267 707316 : pqsecure_write(PGconn *conn, const void *ptr, size_t len)
268 : {
269 : ssize_t n;
270 :
271 : #ifdef USE_SSL
272 707316 : if (conn->ssl_in_use)
273 : {
274 560 : n = pgtls_write(conn, ptr, len);
275 : }
276 : else
277 : #endif
278 : #ifdef ENABLE_GSS
279 : if (conn->gssenc)
280 : {
281 : n = pg_GSS_write(conn, ptr, len);
282 : }
283 : else
284 : #endif
285 : {
286 706756 : n = pqsecure_raw_write(conn, ptr, len);
287 : }
288 :
289 707316 : return n;
290 : }
291 :
292 : /*
293 : * Low-level implementation of pqsecure_write.
294 : *
295 : * This is used directly for an unencrypted connection. For encrypted
296 : * connections, this does the physical I/O on behalf of pgtls_write or
297 : * pg_GSS_write.
298 : *
299 : * This function reports failure (i.e., returns a negative result) only
300 : * for retryable errors such as EINTR. Looping for such cases is to be
301 : * handled at some outer level, maybe all the way up to the application.
302 : * For hard failures, we set conn->write_failed and store an error message
303 : * in conn->write_err_msg, but then claim to have written the data anyway.
304 : * This is because we don't want to report write failures so long as there
305 : * is a possibility of reading from the server and getting an error message
306 : * that could explain why the connection dropped. Many TCP stacks have
307 : * race conditions such that a write failure may or may not be reported
308 : * before all incoming data has been read.
309 : *
310 : * Note that this error behavior happens below the SSL management level when
311 : * we are using SSL. That's because at least some versions of OpenSSL are
312 : * too quick to report a write failure when there's still a possibility to
313 : * get a more useful error from the server.
314 : */
315 : ssize_t
316 708264 : pqsecure_raw_write(PGconn *conn, const void *ptr, size_t len)
317 : {
318 : ssize_t n;
319 708264 : int flags = 0;
320 708264 : int result_errno = 0;
321 : char msgbuf[1024];
322 : char sebuf[PG_STRERROR_R_BUFLEN];
323 :
324 : DECLARE_SIGPIPE_INFO(spinfo);
325 :
326 : /*
327 : * If we already had a write failure, we will never again try to send data
328 : * on that connection. Even if the kernel would let us, we've probably
329 : * lost message boundary sync with the server. conn->write_failed
330 : * therefore persists until the connection is reset, and we just discard
331 : * all data presented to be written.
332 : */
333 708264 : if (conn->write_failed)
334 0 : return len;
335 :
336 : #ifdef MSG_NOSIGNAL
337 708264 : if (conn->sigpipe_flag)
338 708264 : flags |= MSG_NOSIGNAL;
339 :
340 708264 : retry_masked:
341 : #endif /* MSG_NOSIGNAL */
342 :
343 708264 : DISABLE_SIGPIPE(conn, spinfo, return -1);
344 :
345 708264 : n = send(conn->sock, ptr, len, flags);
346 :
347 708264 : if (n < 0)
348 : {
349 66 : result_errno = SOCK_ERRNO;
350 :
351 : /*
352 : * If we see an EINVAL, it may be because MSG_NOSIGNAL isn't available
353 : * on this machine. So, clear sigpipe_flag so we don't try the flag
354 : * again, and retry the send().
355 : */
356 : #ifdef MSG_NOSIGNAL
357 66 : if (flags != 0 && result_errno == EINVAL)
358 : {
359 0 : conn->sigpipe_flag = false;
360 0 : flags = 0;
361 0 : goto retry_masked;
362 : }
363 : #endif /* MSG_NOSIGNAL */
364 :
365 : /* Set error message if appropriate */
366 66 : switch (result_errno)
367 : {
368 : #ifdef EAGAIN
369 34 : case EAGAIN:
370 : #endif
371 : #if defined(EWOULDBLOCK) && (!defined(EAGAIN) || (EWOULDBLOCK != EAGAIN))
372 : case EWOULDBLOCK:
373 : #endif
374 : case EINTR:
375 : /* no error message, caller is expected to retry */
376 34 : break;
377 :
378 32 : case EPIPE:
379 : /* Set flag for EPIPE */
380 32 : REMEMBER_EPIPE(spinfo, true);
381 :
382 : /* FALL THRU */
383 :
384 : case ECONNRESET:
385 32 : conn->write_failed = true;
386 : /* Store error message in conn->write_err_msg, if possible */
387 : /* (strdup failure is OK, we'll cope later) */
388 32 : snprintf(msgbuf, sizeof(msgbuf),
389 32 : libpq_gettext("server closed the connection unexpectedly\n"
390 : "\tThis probably means the server terminated abnormally\n"
391 : "\tbefore or while processing the request."));
392 : /* keep newline out of translated string */
393 32 : strlcat(msgbuf, "\n", sizeof(msgbuf));
394 32 : conn->write_err_msg = strdup(msgbuf);
395 : /* Now claim the write succeeded */
396 32 : n = len;
397 32 : break;
398 :
399 0 : default:
400 0 : conn->write_failed = true;
401 : /* Store error message in conn->write_err_msg, if possible */
402 : /* (strdup failure is OK, we'll cope later) */
403 0 : snprintf(msgbuf, sizeof(msgbuf),
404 0 : libpq_gettext("could not send data to server: %s"),
405 : SOCK_STRERROR(result_errno,
406 : sebuf, sizeof(sebuf)));
407 : /* keep newline out of translated string */
408 0 : strlcat(msgbuf, "\n", sizeof(msgbuf));
409 0 : conn->write_err_msg = strdup(msgbuf);
410 : /* Now claim the write succeeded */
411 0 : n = len;
412 0 : break;
413 : }
414 708198 : }
415 :
416 708264 : RESTORE_SIGPIPE(conn, spinfo);
417 :
418 : /* ensure we return the intended errno to caller */
419 708264 : SOCK_ERRNO_SET(result_errno);
420 :
421 708264 : return n;
422 : }
423 :
424 : /* Dummy versions of SSL info functions, when built without SSL support */
425 : #ifndef USE_SSL
426 :
427 : void *
428 : PQgetssl(PGconn *conn)
429 : {
430 : return NULL;
431 : }
432 :
433 : void *
434 : PQsslStruct(PGconn *conn, const char *struct_name)
435 : {
436 : return NULL;
437 : }
438 :
439 : const char *
440 : PQsslAttribute(PGconn *conn, const char *attribute_name)
441 : {
442 : return NULL;
443 : }
444 :
445 : const char *const *
446 : PQsslAttributeNames(PGconn *conn)
447 : {
448 : static const char *const result[] = {NULL};
449 :
450 : return result;
451 : }
452 : #endif /* USE_SSL */
453 :
454 : /*
455 : * Dummy versions of OpenSSL key password hook functions, when built without
456 : * OpenSSL.
457 : */
458 : #ifndef USE_OPENSSL
459 :
460 : PQsslKeyPassHook_OpenSSL_type
461 : PQgetSSLKeyPassHook_OpenSSL(void)
462 : {
463 : return NULL;
464 : }
465 :
466 : void
467 : PQsetSSLKeyPassHook_OpenSSL(PQsslKeyPassHook_OpenSSL_type hook)
468 : {
469 : return;
470 : }
471 :
472 : int
473 : PQdefaultSSLKeyPassHook_OpenSSL(char *buf, int size, PGconn *conn)
474 : {
475 : return 0;
476 : }
477 : #endif /* USE_OPENSSL */
478 :
479 : /* Dummy version of GSSAPI information functions, when built without GSS support */
480 : #ifndef ENABLE_GSS
481 :
482 : void *
483 0 : PQgetgssctx(PGconn *conn)
484 : {
485 0 : return NULL;
486 : }
487 :
488 : int
489 4 : PQgssEncInUse(PGconn *conn)
490 : {
491 4 : return 0;
492 : }
493 :
494 : #endif /* ENABLE_GSS */
495 :
496 :
497 : #if !defined(WIN32)
498 :
499 : /*
500 : * Block SIGPIPE for this thread. This prevents send()/write() from exiting
501 : * the application.
502 : */
503 : int
504 0 : pq_block_sigpipe(sigset_t *osigset, bool *sigpipe_pending)
505 : {
506 : sigset_t sigpipe_sigset;
507 : sigset_t sigset;
508 :
509 0 : sigemptyset(&sigpipe_sigset);
510 0 : sigaddset(&sigpipe_sigset, SIGPIPE);
511 :
512 : /* Block SIGPIPE and save previous mask for later reset */
513 0 : SOCK_ERRNO_SET(pthread_sigmask(SIG_BLOCK, &sigpipe_sigset, osigset));
514 0 : if (SOCK_ERRNO)
515 0 : return -1;
516 :
517 : /* We can have a pending SIGPIPE only if it was blocked before */
518 0 : if (sigismember(osigset, SIGPIPE))
519 : {
520 : /* Is there a pending SIGPIPE? */
521 0 : if (sigpending(&sigset) != 0)
522 0 : return -1;
523 :
524 0 : if (sigismember(&sigset, SIGPIPE))
525 0 : *sigpipe_pending = true;
526 : else
527 0 : *sigpipe_pending = false;
528 : }
529 : else
530 0 : *sigpipe_pending = false;
531 :
532 0 : return 0;
533 : }
534 :
535 : /*
536 : * Discard any pending SIGPIPE and reset the signal mask.
537 : *
538 : * Note: we are effectively assuming here that the C library doesn't queue
539 : * up multiple SIGPIPE events. If it did, then we'd accidentally leave
540 : * ours in the queue when an event was already pending and we got another.
541 : * As long as it doesn't queue multiple events, we're OK because the caller
542 : * can't tell the difference.
543 : *
544 : * The caller should say got_epipe = false if it is certain that it
545 : * didn't get an EPIPE error; in that case we'll skip the clear operation
546 : * and things are definitely OK, queuing or no. If it got one or might have
547 : * gotten one, pass got_epipe = true.
548 : *
549 : * We do not want this to change errno, since if it did that could lose
550 : * the error code from a preceding send(). We essentially assume that if
551 : * we were able to do pq_block_sigpipe(), this can't fail.
552 : */
553 : void
554 0 : pq_reset_sigpipe(sigset_t *osigset, bool sigpipe_pending, bool got_epipe)
555 : {
556 0 : int save_errno = SOCK_ERRNO;
557 : int signo;
558 : sigset_t sigset;
559 :
560 : /* Clear SIGPIPE only if none was pending */
561 0 : if (got_epipe && !sigpipe_pending)
562 : {
563 0 : if (sigpending(&sigset) == 0 &&
564 0 : sigismember(&sigset, SIGPIPE))
565 : {
566 : sigset_t sigpipe_sigset;
567 :
568 0 : sigemptyset(&sigpipe_sigset);
569 0 : sigaddset(&sigpipe_sigset, SIGPIPE);
570 :
571 0 : sigwait(&sigpipe_sigset, &signo);
572 : }
573 : }
574 :
575 : /* Restore saved block mask */
576 0 : pthread_sigmask(SIG_SETMASK, osigset, NULL);
577 :
578 0 : SOCK_ERRNO_SET(save_errno);
579 0 : }
580 :
581 : #endif /* !WIN32 */
|