LCOV - code coverage report
Current view: top level - src/backend/libpq - be-secure.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 78 84 92.9 %
Date: 2025-01-18 04:15:08 Functions: 9 9 100.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * be-secure.c
       4             :  *    functions related to setting up a secure connection to the frontend.
       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/backend/libpq/be-secure.c
      15             :  *
      16             :  *-------------------------------------------------------------------------
      17             :  */
      18             : 
      19             : #include "postgres.h"
      20             : 
      21             : #include <signal.h>
      22             : #include <fcntl.h>
      23             : #include <ctype.h>
      24             : #include <sys/socket.h>
      25             : #include <netdb.h>
      26             : #include <netinet/in.h>
      27             : #include <netinet/tcp.h>
      28             : #include <arpa/inet.h>
      29             : 
      30             : #include "libpq/libpq.h"
      31             : #include "miscadmin.h"
      32             : #include "tcop/tcopprot.h"
      33             : #include "utils/injection_point.h"
      34             : #include "utils/wait_event.h"
      35             : 
      36             : char       *ssl_library;
      37             : char       *ssl_cert_file;
      38             : char       *ssl_key_file;
      39             : char       *ssl_ca_file;
      40             : char       *ssl_crl_file;
      41             : char       *ssl_crl_dir;
      42             : char       *ssl_dh_params_file;
      43             : char       *ssl_passphrase_command;
      44             : bool        ssl_passphrase_command_supports_reload;
      45             : 
      46             : #ifdef USE_SSL
      47             : bool        ssl_loaded_verify_locations = false;
      48             : #endif
      49             : 
      50             : /* GUC variable controlling SSL cipher list */
      51             : char       *SSLCipherSuites = NULL;
      52             : char       *SSLCipherList = NULL;
      53             : 
      54             : /* GUC variable for default ECHD curve. */
      55             : char       *SSLECDHCurve;
      56             : 
      57             : /* GUC variable: if false, prefer client ciphers */
      58             : bool        SSLPreferServerCiphers;
      59             : 
      60             : int         ssl_min_protocol_version = PG_TLS1_2_VERSION;
      61             : int         ssl_max_protocol_version = PG_TLS_ANY;
      62             : 
      63             : /* ------------------------------------------------------------ */
      64             : /*           Procedures common to all secure sessions           */
      65             : /* ------------------------------------------------------------ */
      66             : 
      67             : /*
      68             :  *  Initialize global context.
      69             :  *
      70             :  * If isServerStart is true, report any errors as FATAL (so we don't return).
      71             :  * Otherwise, log errors at LOG level and return -1 to indicate trouble,
      72             :  * preserving the old SSL state if any.  Returns 0 if OK.
      73             :  */
      74             : int
      75          64 : secure_initialize(bool isServerStart)
      76             : {
      77             : #ifdef USE_SSL
      78          64 :     return be_tls_init(isServerStart);
      79             : #else
      80             :     return 0;
      81             : #endif
      82             : }
      83             : 
      84             : /*
      85             :  *  Destroy global context, if any.
      86             :  */
      87             : void
      88         246 : secure_destroy(void)
      89             : {
      90             : #ifdef USE_SSL
      91         246 :     be_tls_destroy();
      92             : #endif
      93         246 : }
      94             : 
      95             : /*
      96             :  * Indicate if we have loaded the root CA store to verify certificates
      97             :  */
      98             : bool
      99          60 : secure_loaded_verify_locations(void)
     100             : {
     101             : #ifdef USE_SSL
     102          60 :     return ssl_loaded_verify_locations;
     103             : #else
     104             :     return false;
     105             : #endif
     106             : }
     107             : 
     108             : /*
     109             :  *  Attempt to negotiate secure session.
     110             :  */
     111             : int
     112         250 : secure_open_server(Port *port)
     113             : {
     114             : #ifdef USE_SSL
     115         250 :     int         r = 0;
     116             :     ssize_t     len;
     117             : 
     118             :     /* push unencrypted buffered data back through SSL setup */
     119         250 :     len = pq_buffer_remaining_data();
     120         250 :     if (len > 0)
     121             :     {
     122           6 :         char       *buf = palloc(len);
     123             : 
     124           6 :         pq_startmsgread();
     125           6 :         if (pq_getbytes(buf, len) == EOF)
     126           0 :             return STATUS_ERROR;    /* shouldn't be possible */
     127           6 :         pq_endmsgread();
     128           6 :         port->raw_buf = buf;
     129           6 :         port->raw_buf_remaining = len;
     130           6 :         port->raw_buf_consumed = 0;
     131             :     }
     132             :     Assert(pq_buffer_remaining_data() == 0);
     133             : 
     134         250 :     INJECTION_POINT("backend-ssl-startup");
     135             : 
     136         248 :     r = be_tls_open_server(port);
     137             : 
     138         248 :     if (port->raw_buf_remaining > 0)
     139             :     {
     140             :         /*
     141             :          * This shouldn't be possible -- it would mean the client sent
     142             :          * encrypted data before we established a session key...
     143             :          */
     144           0 :         elog(LOG, "buffered unencrypted data remains after negotiating SSL connection");
     145           0 :         return STATUS_ERROR;
     146             :     }
     147         248 :     if (port->raw_buf != NULL)
     148             :     {
     149           6 :         pfree(port->raw_buf);
     150           6 :         port->raw_buf = NULL;
     151             :     }
     152             : 
     153         248 :     ereport(DEBUG2,
     154             :             (errmsg_internal("SSL connection from DN:\"%s\" CN:\"%s\"",
     155             :                              port->peer_dn ? port->peer_dn : "(anonymous)",
     156             :                              port->peer_cn ? port->peer_cn : "(anonymous)")));
     157         248 :     return r;
     158             : #else
     159             :     return 0;
     160             : #endif
     161             : }
     162             : 
     163             : /*
     164             :  *  Close secure session.
     165             :  */
     166             : void
     167       23018 : secure_close(Port *port)
     168             : {
     169             : #ifdef USE_SSL
     170       23018 :     if (port->ssl_in_use)
     171         248 :         be_tls_close(port);
     172             : #endif
     173       23018 : }
     174             : 
     175             : /*
     176             :  *  Read data from a secure connection.
     177             :  */
     178             : ssize_t
     179     3256334 : secure_read(Port *port, void *ptr, size_t len)
     180             : {
     181             :     ssize_t     n;
     182             :     int         waitfor;
     183             : 
     184             :     /* Deal with any already-pending interrupt condition. */
     185     3256334 :     ProcessClientReadInterrupt(false);
     186             : 
     187     3549772 : retry:
     188             : #ifdef USE_SSL
     189     3549772 :     waitfor = 0;
     190     3549772 :     if (port->ssl_in_use)
     191             :     {
     192         774 :         n = be_tls_read(port, ptr, len, &waitfor);
     193             :     }
     194             :     else
     195             : #endif
     196             : #ifdef ENABLE_GSS
     197             :     if (port->gss && port->gss->enc)
     198             :     {
     199             :         n = be_gssapi_read(port, ptr, len);
     200             :         waitfor = WL_SOCKET_READABLE;
     201             :     }
     202             :     else
     203             : #endif
     204             :     {
     205     3548998 :         n = secure_raw_read(port, ptr, len);
     206     3548998 :         waitfor = WL_SOCKET_READABLE;
     207             :     }
     208             : 
     209             :     /* In blocking mode, wait until the socket is ready */
     210     3549772 :     if (n < 0 && !port->noblock && (errno == EWOULDBLOCK || errno == EAGAIN))
     211             :     {
     212             :         WaitEvent   event;
     213             : 
     214             :         Assert(waitfor);
     215             : 
     216      293520 :         ModifyWaitEvent(FeBeWaitSet, FeBeWaitSetSocketPos, waitfor, NULL);
     217             : 
     218      293520 :         WaitEventSetWait(FeBeWaitSet, -1 /* no timeout */ , &event, 1,
     219             :                          WAIT_EVENT_CLIENT_READ);
     220             : 
     221             :         /*
     222             :          * If the postmaster has died, it's not safe to continue running,
     223             :          * because it is the postmaster's job to kill us if some other backend
     224             :          * exits uncleanly.  Moreover, we won't run very well in this state;
     225             :          * helper processes like walwriter and the bgwriter will exit, so
     226             :          * performance may be poor.  Finally, if we don't exit, pg_ctl will be
     227             :          * unable to restart the postmaster without manual intervention, so no
     228             :          * new connections can be accepted.  Exiting clears the deck for a
     229             :          * postmaster restart.
     230             :          *
     231             :          * (Note that we only make this check when we would otherwise sleep on
     232             :          * our latch.  We might still continue running for a while if the
     233             :          * postmaster is killed in mid-query, or even through multiple queries
     234             :          * if we never have to wait for read.  We don't want to burn too many
     235             :          * cycles checking for this very rare condition, and this should cause
     236             :          * us to exit quickly in most cases.)
     237             :          */
     238      293520 :         if (event.events & WL_POSTMASTER_DEATH)
     239           0 :             ereport(FATAL,
     240             :                     (errcode(ERRCODE_ADMIN_SHUTDOWN),
     241             :                      errmsg("terminating connection due to unexpected postmaster exit")));
     242             : 
     243             :         /* Handle interrupt. */
     244      293520 :         if (event.events & WL_LATCH_SET)
     245             :         {
     246       20436 :             ResetLatch(MyLatch);
     247       20436 :             ProcessClientReadInterrupt(true);
     248             : 
     249             :             /*
     250             :              * We'll retry the read. Most likely it will return immediately
     251             :              * because there's still no data available, and we'll wait for the
     252             :              * socket to become ready again.
     253             :              */
     254             :         }
     255      293492 :         goto retry;
     256             :     }
     257             : 
     258             :     /*
     259             :      * Process interrupts that happened during a successful (or non-blocking,
     260             :      * or hard-failed) read.
     261             :      */
     262     3256252 :     ProcessClientReadInterrupt(false);
     263             : 
     264     3256252 :     return n;
     265             : }
     266             : 
     267             : ssize_t
     268     3553076 : secure_raw_read(Port *port, void *ptr, size_t len)
     269             : {
     270             :     ssize_t     n;
     271             : 
     272             :     /* Read from the "unread" buffered data first. c.f. libpq-be.h */
     273     3553076 :     if (port->raw_buf_remaining > 0)
     274             :     {
     275             :         /* consume up to len bytes from the raw_buf */
     276          12 :         if (len > port->raw_buf_remaining)
     277           0 :             len = port->raw_buf_remaining;
     278             :         Assert(port->raw_buf);
     279          12 :         memcpy(ptr, port->raw_buf + port->raw_buf_consumed, len);
     280          12 :         port->raw_buf_consumed += len;
     281          12 :         port->raw_buf_remaining -= len;
     282          12 :         return len;
     283             :     }
     284             : 
     285             :     /*
     286             :      * Try to read from the socket without blocking. If it succeeds we're
     287             :      * done, otherwise we'll wait for the socket using the latch mechanism.
     288             :      */
     289             : #ifdef WIN32
     290             :     pgwin32_noblock = true;
     291             : #endif
     292     3553064 :     n = recv(port->sock, ptr, len, 0);
     293             : #ifdef WIN32
     294             :     pgwin32_noblock = false;
     295             : #endif
     296             : 
     297     3553064 :     return n;
     298             : }
     299             : 
     300             : 
     301             : /*
     302             :  *  Write data to a secure connection.
     303             :  */
     304             : ssize_t
     305     2060356 : secure_write(Port *port, void *ptr, size_t len)
     306             : {
     307             :     ssize_t     n;
     308             :     int         waitfor;
     309             : 
     310             :     /* Deal with any already-pending interrupt condition. */
     311     2060356 :     ProcessClientWriteInterrupt(false);
     312             : 
     313     2077514 : retry:
     314     2077514 :     waitfor = 0;
     315             : #ifdef USE_SSL
     316     2077514 :     if (port->ssl_in_use)
     317             :     {
     318         382 :         n = be_tls_write(port, ptr, len, &waitfor);
     319             :     }
     320             :     else
     321             : #endif
     322             : #ifdef ENABLE_GSS
     323             :     if (port->gss && port->gss->enc)
     324             :     {
     325             :         n = be_gssapi_write(port, ptr, len);
     326             :         waitfor = WL_SOCKET_WRITEABLE;
     327             :     }
     328             :     else
     329             : #endif
     330             :     {
     331     2077132 :         n = secure_raw_write(port, ptr, len);
     332     2077132 :         waitfor = WL_SOCKET_WRITEABLE;
     333             :     }
     334             : 
     335     2077514 :     if (n < 0 && !port->noblock && (errno == EWOULDBLOCK || errno == EAGAIN))
     336             :     {
     337             :         WaitEvent   event;
     338             : 
     339             :         Assert(waitfor);
     340             : 
     341       17158 :         ModifyWaitEvent(FeBeWaitSet, FeBeWaitSetSocketPos, waitfor, NULL);
     342             : 
     343       17158 :         WaitEventSetWait(FeBeWaitSet, -1 /* no timeout */ , &event, 1,
     344             :                          WAIT_EVENT_CLIENT_WRITE);
     345             : 
     346             :         /* See comments in secure_read. */
     347       17158 :         if (event.events & WL_POSTMASTER_DEATH)
     348           0 :             ereport(FATAL,
     349             :                     (errcode(ERRCODE_ADMIN_SHUTDOWN),
     350             :                      errmsg("terminating connection due to unexpected postmaster exit")));
     351             : 
     352             :         /* Handle interrupt. */
     353       17158 :         if (event.events & WL_LATCH_SET)
     354             :         {
     355           6 :             ResetLatch(MyLatch);
     356           6 :             ProcessClientWriteInterrupt(true);
     357             : 
     358             :             /*
     359             :              * We'll retry the write. Most likely it will return immediately
     360             :              * because there's still no buffer space available, and we'll wait
     361             :              * for the socket to become ready again.
     362             :              */
     363             :         }
     364       17158 :         goto retry;
     365             :     }
     366             : 
     367             :     /*
     368             :      * Process interrupts that happened during a successful (or non-blocking,
     369             :      * or hard-failed) write.
     370             :      */
     371     2060356 :     ProcessClientWriteInterrupt(false);
     372             : 
     373     2060356 :     return n;
     374             : }
     375             : 
     376             : ssize_t
     377     2078216 : secure_raw_write(Port *port, const void *ptr, size_t len)
     378             : {
     379             :     ssize_t     n;
     380             : 
     381             : #ifdef WIN32
     382             :     pgwin32_noblock = true;
     383             : #endif
     384     2078216 :     n = send(port->sock, ptr, len, 0);
     385             : #ifdef WIN32
     386             :     pgwin32_noblock = false;
     387             : #endif
     388             : 
     389     2078216 :     return n;
     390             : }

Generated by: LCOV version 1.14