Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * backend_startup.c
4 : * Backend startup code
5 : *
6 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
7 : * Portions Copyright (c) 1994, Regents of the University of California
8 : *
9 : *
10 : * IDENTIFICATION
11 : * src/backend/tcop/backend_startup.c
12 : *
13 : *-------------------------------------------------------------------------
14 : */
15 :
16 : #include "postgres.h"
17 :
18 : #include <unistd.h>
19 :
20 : #include "access/xlog.h"
21 : #include "common/ip.h"
22 : #include "common/string.h"
23 : #include "libpq/libpq.h"
24 : #include "libpq/libpq-be.h"
25 : #include "libpq/pqformat.h"
26 : #include "libpq/pqsignal.h"
27 : #include "miscadmin.h"
28 : #include "postmaster/postmaster.h"
29 : #include "replication/walsender.h"
30 : #include "storage/fd.h"
31 : #include "storage/ipc.h"
32 : #include "storage/procsignal.h"
33 : #include "storage/proc.h"
34 : #include "tcop/backend_startup.h"
35 : #include "tcop/tcopprot.h"
36 : #include "utils/builtins.h"
37 : #include "utils/injection_point.h"
38 : #include "utils/memutils.h"
39 : #include "utils/ps_status.h"
40 : #include "utils/timeout.h"
41 :
42 : /* GUCs */
43 : bool Trace_connection_negotiation = false;
44 :
45 : static void BackendInitialize(ClientSocket *client_sock, CAC_state cac);
46 : static int ProcessSSLStartup(Port *port);
47 : static int ProcessStartupPacket(Port *port, bool ssl_done, bool gss_done);
48 : static void SendNegotiateProtocolVersion(List *unrecognized_protocol_options);
49 : static void process_startup_packet_die(SIGNAL_ARGS);
50 : static void StartupPacketTimeoutHandler(void);
51 :
52 : /*
53 : * Entry point for a new backend process.
54 : *
55 : * Initialize the connection, read the startup packet, authenticate the
56 : * client, and start the main processing loop.
57 : */
58 : void
59 23062 : BackendMain(char *startup_data, size_t startup_data_len)
60 : {
61 23062 : BackendStartupData *bsdata = (BackendStartupData *) startup_data;
62 :
63 : Assert(startup_data_len == sizeof(BackendStartupData));
64 : Assert(MyClientSocket != NULL);
65 :
66 : #ifdef EXEC_BACKEND
67 :
68 : /*
69 : * Need to reinitialize the SSL library in the backend, since the context
70 : * structures contain function pointers and cannot be passed through the
71 : * parameter file.
72 : *
73 : * If for some reason reload fails (maybe the user installed broken key
74 : * files), soldier on without SSL; that's better than all connections
75 : * becoming impossible.
76 : *
77 : * XXX should we do this in all child processes? For the moment it's
78 : * enough to do it in backend children.
79 : */
80 : #ifdef USE_SSL
81 : if (EnableSSL)
82 : {
83 : if (secure_initialize(false) == 0)
84 : LoadedSSL = true;
85 : else
86 : ereport(LOG,
87 : (errmsg("SSL configuration could not be loaded in child process")));
88 : }
89 : #endif
90 : #endif
91 :
92 : /* Perform additional initialization and collect startup packet */
93 23062 : BackendInitialize(MyClientSocket, bsdata->canAcceptConnections);
94 :
95 : /*
96 : * Create a per-backend PGPROC struct in shared memory. We must do this
97 : * before we can use LWLocks or access any shared memory.
98 : */
99 22748 : InitProcess();
100 :
101 : /*
102 : * Make sure we aren't in PostmasterContext anymore. (We can't delete it
103 : * just yet, though, because InitPostgres will need the HBA data.)
104 : */
105 22742 : MemoryContextSwitchTo(TopMemoryContext);
106 :
107 22742 : PostgresMain(MyProcPort->database_name, MyProcPort->user_name);
108 : }
109 :
110 :
111 : /*
112 : * BackendInitialize -- initialize an interactive (postmaster-child)
113 : * backend process, and collect the client's startup packet.
114 : *
115 : * returns: nothing. Will not return at all if there's any failure.
116 : *
117 : * Note: this code does not depend on having any access to shared memory.
118 : * Indeed, our approach to SIGTERM/timeout handling *requires* that
119 : * shared memory not have been touched yet; see comments within.
120 : * In the EXEC_BACKEND case, we are physically attached to shared memory
121 : * but have not yet set up most of our local pointers to shmem structures.
122 : */
123 : static void
124 23062 : BackendInitialize(ClientSocket *client_sock, CAC_state cac)
125 : {
126 : int status;
127 : int ret;
128 : Port *port;
129 : char remote_host[NI_MAXHOST];
130 : char remote_port[NI_MAXSERV];
131 : StringInfoData ps_data;
132 : MemoryContext oldcontext;
133 :
134 : /* Tell fd.c about the long-lived FD associated with the client_sock */
135 23062 : ReserveExternalFD();
136 :
137 : /*
138 : * PreAuthDelay is a debugging aid for investigating problems in the
139 : * authentication cycle: it can be set in postgresql.conf to allow time to
140 : * attach to the newly-forked backend with a debugger. (See also
141 : * PostAuthDelay, which we allow clients to pass through PGOPTIONS, but it
142 : * is not honored until after authentication.)
143 : */
144 23062 : if (PreAuthDelay > 0)
145 0 : pg_usleep(PreAuthDelay * 1000000L);
146 :
147 : /* This flag will remain set until InitPostgres finishes authentication */
148 23062 : ClientAuthInProgress = true; /* limit visibility of log messages */
149 :
150 : /*
151 : * Initialize libpq and enable reporting of ereport errors to the client.
152 : * Must do this now because authentication uses libpq to send messages.
153 : *
154 : * The Port structure and all data structures attached to it are allocated
155 : * in TopMemoryContext, so that they survive into PostgresMain execution.
156 : * We need not worry about leaking this storage on failure, since we
157 : * aren't in the postmaster process anymore.
158 : */
159 23062 : oldcontext = MemoryContextSwitchTo(TopMemoryContext);
160 23062 : port = MyProcPort = pq_init(client_sock);
161 23062 : MemoryContextSwitchTo(oldcontext);
162 :
163 23062 : whereToSendOutput = DestRemote; /* now safe to ereport to client */
164 :
165 : /* set these to empty in case they are needed before we set them up */
166 23062 : port->remote_host = "";
167 23062 : port->remote_port = "";
168 :
169 : /*
170 : * We arrange to do _exit(1) if we receive SIGTERM or timeout while trying
171 : * to collect the startup packet; while SIGQUIT results in _exit(2).
172 : * Otherwise the postmaster cannot shutdown the database FAST or IMMED
173 : * cleanly if a buggy client fails to send the packet promptly.
174 : *
175 : * Exiting with _exit(1) is only possible because we have not yet touched
176 : * shared memory; therefore no outside-the-process state needs to get
177 : * cleaned up.
178 : */
179 23062 : pqsignal(SIGTERM, process_startup_packet_die);
180 : /* SIGQUIT handler was already set up by InitPostmasterChild */
181 23062 : InitializeTimeouts(); /* establishes SIGALRM handler */
182 23062 : sigprocmask(SIG_SETMASK, &StartupBlockSig, NULL);
183 :
184 : /*
185 : * Get the remote host name and port for logging and status display.
186 : */
187 23062 : remote_host[0] = '\0';
188 23062 : remote_port[0] = '\0';
189 23062 : if ((ret = pg_getnameinfo_all(&port->raddr.addr, port->raddr.salen,
190 : remote_host, sizeof(remote_host),
191 : remote_port, sizeof(remote_port),
192 : (log_hostname ? 0 : NI_NUMERICHOST) | NI_NUMERICSERV)) != 0)
193 0 : ereport(WARNING,
194 : (errmsg_internal("pg_getnameinfo_all() failed: %s",
195 : gai_strerror(ret))));
196 :
197 : /*
198 : * Save remote_host and remote_port in port structure (after this, they
199 : * will appear in log_line_prefix data for log messages).
200 : */
201 23062 : port->remote_host = MemoryContextStrdup(TopMemoryContext, remote_host);
202 23062 : port->remote_port = MemoryContextStrdup(TopMemoryContext, remote_port);
203 :
204 : /* And now we can issue the Log_connections message, if wanted */
205 23062 : if (Log_connections)
206 : {
207 806 : if (remote_port[0])
208 290 : ereport(LOG,
209 : (errmsg("connection received: host=%s port=%s",
210 : remote_host,
211 : remote_port)));
212 : else
213 516 : ereport(LOG,
214 : (errmsg("connection received: host=%s",
215 : remote_host)));
216 : }
217 :
218 : /* For testing client error handling */
219 : #ifdef USE_INJECTION_POINTS
220 23062 : INJECTION_POINT("backend-initialize");
221 23062 : if (IS_INJECTION_POINT_ATTACHED("backend-initialize-v2-error"))
222 : {
223 : /*
224 : * This simulates an early error from a pre-v14 server, which used the
225 : * version 2 protocol for any errors that occurred before processing
226 : * the startup packet.
227 : */
228 2 : FrontendProtocol = PG_PROTOCOL(2, 0);
229 2 : elog(FATAL, "protocol version 2 error triggered");
230 : }
231 : #endif
232 :
233 : /*
234 : * If we did a reverse lookup to name, we might as well save the results
235 : * rather than possibly repeating the lookup during authentication.
236 : *
237 : * Note that we don't want to specify NI_NAMEREQD above, because then we'd
238 : * get nothing useful for a client without an rDNS entry. Therefore, we
239 : * must check whether we got a numeric IPv4 or IPv6 address, and not save
240 : * it into remote_hostname if so. (This test is conservative and might
241 : * sometimes classify a hostname as numeric, but an error in that
242 : * direction is safe; it only results in a possible extra lookup.)
243 : */
244 23060 : if (log_hostname &&
245 230 : ret == 0 &&
246 230 : strspn(remote_host, "0123456789.") < strlen(remote_host) &&
247 230 : strspn(remote_host, "0123456789ABCDEFabcdef:") < strlen(remote_host))
248 : {
249 230 : port->remote_hostname = MemoryContextStrdup(TopMemoryContext, remote_host);
250 : }
251 :
252 : /*
253 : * Ready to begin client interaction. We will give up and _exit(1) after
254 : * a time delay, so that a broken client can't hog a connection
255 : * indefinitely. PreAuthDelay and any DNS interactions above don't count
256 : * against the time limit.
257 : *
258 : * Note: AuthenticationTimeout is applied here while waiting for the
259 : * startup packet, and then again in InitPostgres for the duration of any
260 : * authentication operations. So a hostile client could tie up the
261 : * process for nearly twice AuthenticationTimeout before we kick him off.
262 : *
263 : * Note: because PostgresMain will call InitializeTimeouts again, the
264 : * registration of STARTUP_PACKET_TIMEOUT will be lost. This is okay
265 : * since we never use it again after this function.
266 : */
267 23060 : RegisterTimeout(STARTUP_PACKET_TIMEOUT, StartupPacketTimeoutHandler);
268 23060 : enable_timeout_after(STARTUP_PACKET_TIMEOUT, AuthenticationTimeout * 1000);
269 :
270 : /* Handle direct SSL handshake */
271 23060 : status = ProcessSSLStartup(port);
272 :
273 : /*
274 : * Receive the startup packet (which might turn out to be a cancel request
275 : * packet).
276 : */
277 23060 : if (status == STATUS_OK)
278 23050 : status = ProcessStartupPacket(port, false, false);
279 :
280 : /*
281 : * If we're going to reject the connection due to database state, say so
282 : * now instead of wasting cycles on an authentication exchange. (This also
283 : * allows a pg_ping utility to be written.)
284 : */
285 23058 : if (status == STATUS_OK)
286 : {
287 22956 : switch (cac)
288 : {
289 194 : case CAC_STARTUP:
290 194 : ereport(FATAL,
291 : (errcode(ERRCODE_CANNOT_CONNECT_NOW),
292 : errmsg("the database system is starting up")));
293 : break;
294 10 : case CAC_NOTCONSISTENT:
295 10 : if (EnableHotStandby)
296 10 : ereport(FATAL,
297 : (errcode(ERRCODE_CANNOT_CONNECT_NOW),
298 : errmsg("the database system is not yet accepting connections"),
299 : errdetail("Consistent recovery state has not been yet reached.")));
300 : else
301 0 : ereport(FATAL,
302 : (errcode(ERRCODE_CANNOT_CONNECT_NOW),
303 : errmsg("the database system is not accepting connections"),
304 : errdetail("Hot standby mode is disabled.")));
305 : break;
306 2 : case CAC_SHUTDOWN:
307 2 : ereport(FATAL,
308 : (errcode(ERRCODE_CANNOT_CONNECT_NOW),
309 : errmsg("the database system is shutting down")));
310 : break;
311 0 : case CAC_RECOVERY:
312 0 : ereport(FATAL,
313 : (errcode(ERRCODE_CANNOT_CONNECT_NOW),
314 : errmsg("the database system is in recovery mode")));
315 : break;
316 2 : case CAC_TOOMANY:
317 2 : ereport(FATAL,
318 : (errcode(ERRCODE_TOO_MANY_CONNECTIONS),
319 : errmsg("sorry, too many clients already")));
320 : break;
321 22748 : case CAC_OK:
322 22748 : break;
323 : }
324 102 : }
325 :
326 : /*
327 : * Disable the timeout, and prevent SIGTERM again.
328 : */
329 22850 : disable_timeout(STARTUP_PACKET_TIMEOUT, false);
330 22850 : sigprocmask(SIG_SETMASK, &BlockSig, NULL);
331 :
332 : /*
333 : * As a safety check that nothing in startup has yet performed
334 : * shared-memory modifications that would need to be undone if we had
335 : * exited through SIGTERM or timeout above, check that no on_shmem_exit
336 : * handlers have been registered yet. (This isn't terribly bulletproof,
337 : * since someone might misuse an on_proc_exit handler for shmem cleanup,
338 : * but it's a cheap and helpful check. We cannot disallow on_proc_exit
339 : * handlers unfortunately, since pq_init() already registered one.)
340 : */
341 22850 : check_on_shmem_exit_lists_are_empty();
342 :
343 : /*
344 : * Stop here if it was bad or a cancel packet. ProcessStartupPacket
345 : * already did any appropriate error reporting.
346 : */
347 22850 : if (status != STATUS_OK)
348 102 : proc_exit(0);
349 :
350 : /*
351 : * Now that we have the user and database name, we can set the process
352 : * title for ps. It's good to do this as early as possible in startup.
353 : */
354 22748 : initStringInfo(&ps_data);
355 22748 : if (am_walsender)
356 2128 : appendStringInfo(&ps_data, "%s ", GetBackendTypeDesc(B_WAL_SENDER));
357 22748 : appendStringInfo(&ps_data, "%s ", port->user_name);
358 22748 : if (port->database_name[0] != '\0')
359 21852 : appendStringInfo(&ps_data, "%s ", port->database_name);
360 22748 : appendStringInfoString(&ps_data, port->remote_host);
361 22748 : if (port->remote_port[0] != '\0')
362 502 : appendStringInfo(&ps_data, "(%s)", port->remote_port);
363 :
364 22748 : init_ps_display(ps_data.data);
365 22748 : pfree(ps_data.data);
366 :
367 22748 : set_ps_display("initializing");
368 22748 : }
369 :
370 : /*
371 : * Check for a direct SSL connection.
372 : *
373 : * This happens before the startup packet so we are careful not to actually
374 : * read any bytes from the stream if it's not a direct SSL connection.
375 : */
376 : static int
377 23060 : ProcessSSLStartup(Port *port)
378 : {
379 : int firstbyte;
380 :
381 : Assert(!port->ssl_in_use);
382 :
383 23060 : pq_startmsgread();
384 23060 : firstbyte = pq_peekbyte();
385 23060 : pq_endmsgread();
386 23060 : if (firstbyte == EOF)
387 : {
388 : /*
389 : * Like in ProcessStartupPacket, if we get no data at all, don't
390 : * clutter the log with a complaint.
391 : */
392 6 : return STATUS_ERROR;
393 : }
394 :
395 23054 : if (firstbyte != 0x16)
396 : {
397 : /* Not an SSL handshake message */
398 23044 : return STATUS_OK;
399 : }
400 :
401 : /*
402 : * First byte indicates standard SSL handshake message
403 : *
404 : * (It can't be a Postgres startup length because in network byte order
405 : * that would be a startup packet hundreds of megabytes long)
406 : */
407 :
408 : #ifdef USE_SSL
409 10 : if (!LoadedSSL || port->laddr.addr.ss_family == AF_UNIX)
410 : {
411 : /* SSL not supported */
412 4 : goto reject;
413 : }
414 :
415 6 : if (secure_open_server(port) == -1)
416 : {
417 : /*
418 : * we assume secure_open_server() sent an appropriate TLS alert
419 : * already
420 : */
421 0 : goto reject;
422 : }
423 : Assert(port->ssl_in_use);
424 :
425 6 : if (!port->alpn_used)
426 : {
427 0 : ereport(COMMERROR,
428 : (errcode(ERRCODE_PROTOCOL_VIOLATION),
429 : errmsg("received direct SSL connection request without ALPN protocol negotiation extension")));
430 0 : goto reject;
431 : }
432 :
433 6 : if (Trace_connection_negotiation)
434 6 : ereport(LOG,
435 : (errmsg("direct SSL connection accepted")));
436 6 : return STATUS_OK;
437 : #else
438 : /* SSL not supported by this build */
439 : goto reject;
440 : #endif
441 :
442 4 : reject:
443 4 : if (Trace_connection_negotiation)
444 4 : ereport(LOG,
445 : (errmsg("direct SSL connection rejected")));
446 4 : return STATUS_ERROR;
447 : }
448 :
449 : /*
450 : * Read a client's startup packet and do something according to it.
451 : *
452 : * Returns STATUS_OK or STATUS_ERROR, or might call ereport(FATAL) and
453 : * not return at all.
454 : *
455 : * (Note that ereport(FATAL) stuff is sent to the client, so only use it
456 : * if that's what you want. Return STATUS_ERROR if you don't want to
457 : * send anything to the client, which would typically be appropriate
458 : * if we detect a communications failure.)
459 : *
460 : * Set ssl_done and/or gss_done when negotiation of an encrypted layer
461 : * (currently, TLS or GSSAPI) is completed. A successful negotiation of either
462 : * encryption layer sets both flags, but a rejected negotiation sets only the
463 : * flag for that layer, since the client may wish to try the other one. We
464 : * should make no assumption here about the order in which the client may make
465 : * requests.
466 : */
467 : static int
468 23558 : ProcessStartupPacket(Port *port, bool ssl_done, bool gss_done)
469 : {
470 : int32 len;
471 : char *buf;
472 : ProtocolVersion proto;
473 : MemoryContext oldcontext;
474 :
475 23558 : pq_startmsgread();
476 :
477 : /*
478 : * Grab the first byte of the length word separately, so that we can tell
479 : * whether we have no data at all or an incomplete packet. (This might
480 : * sound inefficient, but it's not really, because of buffering in
481 : * pqcomm.c.)
482 : */
483 23558 : if (pq_getbytes((char *) &len, 1) == EOF)
484 : {
485 : /*
486 : * If we get no data at all, don't clutter the log with a complaint;
487 : * such cases often occur for legitimate reasons. An example is that
488 : * we might be here after responding to NEGOTIATE_SSL_CODE, and if the
489 : * client didn't like our response, it'll probably just drop the
490 : * connection. Service-monitoring software also often just opens and
491 : * closes a connection without sending anything. (So do port
492 : * scanners, which may be less benign, but it's not really our job to
493 : * notice those.)
494 : */
495 36 : return STATUS_ERROR;
496 : }
497 :
498 23522 : if (pq_getbytes(((char *) &len) + 1, 3) == EOF)
499 : {
500 : /* Got a partial length word, so bleat about that */
501 0 : if (!ssl_done && !gss_done)
502 0 : ereport(COMMERROR,
503 : (errcode(ERRCODE_PROTOCOL_VIOLATION),
504 : errmsg("incomplete startup packet")));
505 0 : return STATUS_ERROR;
506 : }
507 :
508 23522 : len = pg_ntoh32(len);
509 23522 : len -= 4;
510 :
511 23522 : if (len < (int32) sizeof(ProtocolVersion) ||
512 23522 : len > MAX_STARTUP_PACKET_LENGTH)
513 : {
514 0 : ereport(COMMERROR,
515 : (errcode(ERRCODE_PROTOCOL_VIOLATION),
516 : errmsg("invalid length of startup packet")));
517 0 : return STATUS_ERROR;
518 : }
519 :
520 : /*
521 : * Allocate space to hold the startup packet, plus one extra byte that's
522 : * initialized to be zero. This ensures we will have null termination of
523 : * all strings inside the packet.
524 : */
525 23522 : buf = palloc(len + 1);
526 23522 : buf[len] = '\0';
527 :
528 23522 : if (pq_getbytes(buf, len) == EOF)
529 : {
530 0 : ereport(COMMERROR,
531 : (errcode(ERRCODE_PROTOCOL_VIOLATION),
532 : errmsg("incomplete startup packet")));
533 0 : return STATUS_ERROR;
534 : }
535 23522 : pq_endmsgread();
536 :
537 : /*
538 : * The first field is either a protocol version number or a special
539 : * request code.
540 : */
541 23522 : port->proto = proto = pg_ntoh32(*((ProtocolVersion *) buf));
542 :
543 23522 : if (proto == CANCEL_REQUEST_CODE)
544 : {
545 : /*
546 : * The client has sent a cancel request packet, not a normal
547 : * start-a-new-connection packet. Perform the necessary processing.
548 : * Nothing is sent back to the client.
549 : */
550 : CancelRequestPacket *canc;
551 : int backendPID;
552 : int32 cancelAuthCode;
553 :
554 20 : if (len != sizeof(CancelRequestPacket))
555 : {
556 0 : ereport(COMMERROR,
557 : (errcode(ERRCODE_PROTOCOL_VIOLATION),
558 : errmsg("invalid length of startup packet")));
559 0 : return STATUS_ERROR;
560 : }
561 20 : canc = (CancelRequestPacket *) buf;
562 20 : backendPID = (int) pg_ntoh32(canc->backendPID);
563 20 : cancelAuthCode = (int32) pg_ntoh32(canc->cancelAuthCode);
564 :
565 20 : if (backendPID != 0)
566 20 : SendCancelRequest(backendPID, cancelAuthCode);
567 : /* Not really an error, but we don't want to proceed further */
568 20 : return STATUS_ERROR;
569 : }
570 :
571 23502 : if (proto == NEGOTIATE_SSL_CODE && !ssl_done)
572 : {
573 : char SSLok;
574 :
575 : #ifdef USE_SSL
576 :
577 : /*
578 : * No SSL when disabled or on Unix sockets.
579 : *
580 : * Also no SSL negotiation if we already have a direct SSL connection
581 : */
582 546 : if (!LoadedSSL || port->laddr.addr.ss_family == AF_UNIX || port->ssl_in_use)
583 302 : SSLok = 'N';
584 : else
585 244 : SSLok = 'S'; /* Support for SSL */
586 : #else
587 : SSLok = 'N'; /* No support for SSL */
588 : #endif
589 :
590 546 : if (Trace_connection_negotiation)
591 : {
592 24 : if (SSLok == 'S')
593 16 : ereport(LOG,
594 : (errmsg("SSLRequest accepted")));
595 : else
596 8 : ereport(LOG,
597 : (errmsg("SSLRequest rejected")));
598 : }
599 :
600 546 : while (secure_write(port, &SSLok, 1) != 1)
601 : {
602 0 : if (errno == EINTR)
603 0 : continue; /* if interrupted, just retry */
604 0 : ereport(COMMERROR,
605 : (errcode_for_socket_access(),
606 : errmsg("failed to send SSL negotiation response: %m")));
607 0 : return STATUS_ERROR; /* close the connection */
608 : }
609 :
610 : #ifdef USE_SSL
611 546 : if (SSLok == 'S' && secure_open_server(port) == -1)
612 36 : return STATUS_ERROR;
613 : #endif
614 :
615 : /*
616 : * At this point we should have no data already buffered. If we do,
617 : * it was received before we performed the SSL handshake, so it wasn't
618 : * encrypted and indeed may have been injected by a man-in-the-middle.
619 : * We report this case to the client.
620 : */
621 508 : if (pq_buffer_remaining_data() > 0)
622 0 : ereport(FATAL,
623 : (errcode(ERRCODE_PROTOCOL_VIOLATION),
624 : errmsg("received unencrypted data after SSL request"),
625 : errdetail("This could be either a client-software bug or evidence of an attempted man-in-the-middle attack.")));
626 :
627 : /*
628 : * regular startup packet, cancel, etc packet should follow, but not
629 : * another SSL negotiation request, and a GSS request should only
630 : * follow if SSL was rejected (client may negotiate in either order)
631 : */
632 508 : return ProcessStartupPacket(port, true, SSLok == 'S');
633 : }
634 22956 : else if (proto == NEGOTIATE_GSS_CODE && !gss_done)
635 : {
636 0 : char GSSok = 'N';
637 :
638 : #ifdef ENABLE_GSS
639 : /* No GSSAPI encryption when on Unix socket */
640 : if (port->laddr.addr.ss_family != AF_UNIX)
641 : GSSok = 'G';
642 : #endif
643 :
644 0 : if (Trace_connection_negotiation)
645 : {
646 0 : if (GSSok == 'G')
647 0 : ereport(LOG,
648 : (errmsg("GSSENCRequest accepted")));
649 : else
650 0 : ereport(LOG,
651 : (errmsg("GSSENCRequest rejected")));
652 : }
653 :
654 0 : while (secure_write(port, &GSSok, 1) != 1)
655 : {
656 0 : if (errno == EINTR)
657 0 : continue;
658 0 : ereport(COMMERROR,
659 : (errcode_for_socket_access(),
660 : errmsg("failed to send GSSAPI negotiation response: %m")));
661 0 : return STATUS_ERROR; /* close the connection */
662 : }
663 :
664 : #ifdef ENABLE_GSS
665 : if (GSSok == 'G' && secure_open_gssapi(port) == -1)
666 : return STATUS_ERROR;
667 : #endif
668 :
669 : /*
670 : * At this point we should have no data already buffered. If we do,
671 : * it was received before we performed the GSS handshake, so it wasn't
672 : * encrypted and indeed may have been injected by a man-in-the-middle.
673 : * We report this case to the client.
674 : */
675 0 : if (pq_buffer_remaining_data() > 0)
676 0 : ereport(FATAL,
677 : (errcode(ERRCODE_PROTOCOL_VIOLATION),
678 : errmsg("received unencrypted data after GSSAPI encryption request"),
679 : errdetail("This could be either a client-software bug or evidence of an attempted man-in-the-middle attack.")));
680 :
681 : /*
682 : * regular startup packet, cancel, etc packet should follow, but not
683 : * another GSS negotiation request, and an SSL request should only
684 : * follow if GSS was rejected (client may negotiate in either order)
685 : */
686 0 : return ProcessStartupPacket(port, GSSok == 'G', true);
687 : }
688 :
689 : /* Could add additional special packet types here */
690 :
691 : /*
692 : * Set FrontendProtocol now so that ereport() knows what format to send if
693 : * we fail during startup. We use the protocol version requested by the
694 : * client unless it's higher than the latest version we support. It's
695 : * possible that error message fields might look different in newer
696 : * protocol versions, but that's something those new clients should be
697 : * able to deal with.
698 : */
699 22956 : FrontendProtocol = Min(proto, PG_PROTOCOL_LATEST);
700 :
701 : /* Check that the major protocol version is in range. */
702 22956 : if (PG_PROTOCOL_MAJOR(proto) < PG_PROTOCOL_MAJOR(PG_PROTOCOL_EARLIEST) ||
703 22956 : PG_PROTOCOL_MAJOR(proto) > PG_PROTOCOL_MAJOR(PG_PROTOCOL_LATEST))
704 0 : ereport(FATAL,
705 : (errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
706 : errmsg("unsupported frontend protocol %u.%u: server supports %u.0 to %u.%u",
707 : PG_PROTOCOL_MAJOR(proto), PG_PROTOCOL_MINOR(proto),
708 : PG_PROTOCOL_MAJOR(PG_PROTOCOL_EARLIEST),
709 : PG_PROTOCOL_MAJOR(PG_PROTOCOL_LATEST),
710 : PG_PROTOCOL_MINOR(PG_PROTOCOL_LATEST))));
711 :
712 : /*
713 : * Now fetch parameters out of startup packet and save them into the Port
714 : * structure.
715 : */
716 22956 : oldcontext = MemoryContextSwitchTo(TopMemoryContext);
717 :
718 : /* Handle protocol version 3 startup packet */
719 : {
720 22956 : int32 offset = sizeof(ProtocolVersion);
721 22956 : List *unrecognized_protocol_options = NIL;
722 :
723 : /*
724 : * Scan packet body for name/option pairs. We can assume any string
725 : * beginning within the packet body is null-terminated, thanks to
726 : * zeroing extra byte above.
727 : */
728 22956 : port->guc_options = NIL;
729 :
730 110730 : while (offset < len)
731 : {
732 110730 : char *nameptr = buf + offset;
733 : int32 valoffset;
734 : char *valptr;
735 :
736 110730 : if (*nameptr == '\0')
737 22956 : break; /* found packet terminator */
738 87774 : valoffset = offset + strlen(nameptr) + 1;
739 87774 : if (valoffset >= len)
740 0 : break; /* missing value, will complain below */
741 87774 : valptr = buf + valoffset;
742 :
743 87774 : if (strcmp(nameptr, "database") == 0)
744 22956 : port->database_name = pstrdup(valptr);
745 64818 : else if (strcmp(nameptr, "user") == 0)
746 22956 : port->user_name = pstrdup(valptr);
747 41862 : else if (strcmp(nameptr, "options") == 0)
748 6414 : port->cmdline_options = pstrdup(valptr);
749 35448 : else if (strcmp(nameptr, "replication") == 0)
750 : {
751 : /*
752 : * Due to backward compatibility concerns the replication
753 : * parameter is a hybrid beast which allows the value to be
754 : * either boolean or the string 'database'. The latter
755 : * connects to a specific database which is e.g. required for
756 : * logical decoding while.
757 : */
758 2140 : if (strcmp(valptr, "database") == 0)
759 : {
760 1232 : am_walsender = true;
761 1232 : am_db_walsender = true;
762 : }
763 908 : else if (!parse_bool(valptr, &am_walsender))
764 0 : ereport(FATAL,
765 : (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
766 : errmsg("invalid value for parameter \"%s\": \"%s\"",
767 : "replication",
768 : valptr),
769 : errhint("Valid values are: \"false\", 0, \"true\", 1, \"database\".")));
770 : }
771 33308 : else if (strncmp(nameptr, "_pq_.", 5) == 0)
772 : {
773 : /*
774 : * Any option beginning with _pq_. is reserved for use as a
775 : * protocol-level option, but at present no such options are
776 : * defined.
777 : */
778 : unrecognized_protocol_options =
779 0 : lappend(unrecognized_protocol_options, pstrdup(nameptr));
780 : }
781 : else
782 : {
783 : /* Assume it's a generic GUC option */
784 33308 : port->guc_options = lappend(port->guc_options,
785 33308 : pstrdup(nameptr));
786 33308 : port->guc_options = lappend(port->guc_options,
787 33308 : pstrdup(valptr));
788 :
789 : /*
790 : * Copy application_name to port if we come across it. This
791 : * is done so we can log the application_name in the
792 : * connection authorization message. Note that the GUC would
793 : * be used but we haven't gone through GUC setup yet.
794 : */
795 33308 : if (strcmp(nameptr, "application_name") == 0)
796 : {
797 22954 : port->application_name = pg_clean_ascii(valptr, 0);
798 : }
799 : }
800 87774 : offset = valoffset + strlen(valptr) + 1;
801 : }
802 :
803 : /*
804 : * If we didn't find a packet terminator exactly at the end of the
805 : * given packet length, complain.
806 : */
807 22956 : if (offset != len - 1)
808 0 : ereport(FATAL,
809 : (errcode(ERRCODE_PROTOCOL_VIOLATION),
810 : errmsg("invalid startup packet layout: expected terminator as last byte")));
811 :
812 : /*
813 : * If the client requested a newer protocol version or if the client
814 : * requested any protocol options we didn't recognize, let them know
815 : * the newest minor protocol version we do support and the names of
816 : * any unrecognized options.
817 : */
818 22956 : if (PG_PROTOCOL_MINOR(proto) > PG_PROTOCOL_MINOR(PG_PROTOCOL_LATEST) ||
819 : unrecognized_protocol_options != NIL)
820 0 : SendNegotiateProtocolVersion(unrecognized_protocol_options);
821 : }
822 :
823 : /* Check a user name was given. */
824 22956 : if (port->user_name == NULL || port->user_name[0] == '\0')
825 0 : ereport(FATAL,
826 : (errcode(ERRCODE_INVALID_AUTHORIZATION_SPECIFICATION),
827 : errmsg("no PostgreSQL user name specified in startup packet")));
828 :
829 : /* The database defaults to the user name. */
830 22956 : if (port->database_name == NULL || port->database_name[0] == '\0')
831 0 : port->database_name = pstrdup(port->user_name);
832 :
833 : /*
834 : * Truncate given database and user names to length of a Postgres name.
835 : * This avoids lookup failures when overlength names are given.
836 : */
837 22956 : if (strlen(port->database_name) >= NAMEDATALEN)
838 0 : port->database_name[NAMEDATALEN - 1] = '\0';
839 22956 : if (strlen(port->user_name) >= NAMEDATALEN)
840 0 : port->user_name[NAMEDATALEN - 1] = '\0';
841 :
842 22956 : if (am_walsender)
843 2140 : MyBackendType = B_WAL_SENDER;
844 : else
845 20816 : MyBackendType = B_BACKEND;
846 :
847 : /*
848 : * Normal walsender backends, e.g. for streaming replication, are not
849 : * connected to a particular database. But walsenders used for logical
850 : * replication need to connect to a specific database. We allow streaming
851 : * replication commands to be issued even if connected to a database as it
852 : * can make sense to first make a basebackup and then stream changes
853 : * starting from that.
854 : */
855 22956 : if (am_walsender && !am_db_walsender)
856 908 : port->database_name[0] = '\0';
857 :
858 : /*
859 : * Done filling the Port structure
860 : */
861 22956 : MemoryContextSwitchTo(oldcontext);
862 :
863 22956 : return STATUS_OK;
864 : }
865 :
866 : /*
867 : * Send a NegotiateProtocolVersion to the client. This lets the client know
868 : * that they have either requested a newer minor protocol version than we are
869 : * able to speak, or at least one protocol option that we don't understand, or
870 : * possibly both. FrontendProtocol has already been set to the version
871 : * requested by the client or the highest version we know how to speak,
872 : * whichever is older. If the highest version that we know how to speak is too
873 : * old for the client, it can abandon the connection.
874 : *
875 : * We also include in the response a list of protocol options we didn't
876 : * understand. This allows clients to include optional parameters that might
877 : * be present either in newer protocol versions or third-party protocol
878 : * extensions without fear of having to reconnect if those options are not
879 : * understood, while at the same time making certain that the client is aware
880 : * of which options were actually accepted.
881 : */
882 : static void
883 0 : SendNegotiateProtocolVersion(List *unrecognized_protocol_options)
884 : {
885 : StringInfoData buf;
886 : ListCell *lc;
887 :
888 0 : pq_beginmessage(&buf, PqMsg_NegotiateProtocolVersion);
889 0 : pq_sendint32(&buf, FrontendProtocol);
890 0 : pq_sendint32(&buf, list_length(unrecognized_protocol_options));
891 0 : foreach(lc, unrecognized_protocol_options)
892 0 : pq_sendstring(&buf, lfirst(lc));
893 0 : pq_endmessage(&buf);
894 :
895 : /* no need to flush, some other message will follow */
896 0 : }
897 :
898 :
899 : /*
900 : * SIGTERM while processing startup packet.
901 : *
902 : * Running proc_exit() from a signal handler would be quite unsafe.
903 : * However, since we have not yet touched shared memory, we can just
904 : * pull the plug and exit without running any atexit handlers.
905 : *
906 : * One might be tempted to try to send a message, or log one, indicating
907 : * why we are disconnecting. However, that would be quite unsafe in itself.
908 : * Also, it seems undesirable to provide clues about the database's state
909 : * to a client that has not yet completed authentication, or even sent us
910 : * a startup packet.
911 : */
912 : static void
913 0 : process_startup_packet_die(SIGNAL_ARGS)
914 : {
915 0 : _exit(1);
916 : }
917 :
918 : /*
919 : * Timeout while processing startup packet.
920 : * As for process_startup_packet_die(), we exit via _exit(1).
921 : */
922 : static void
923 0 : StartupPacketTimeoutHandler(void)
924 : {
925 0 : _exit(1);
926 : }
|