PostgreSQL Source Code git master
Loading...
Searching...
No Matches
ps_status.c
Go to the documentation of this file.
1/*--------------------------------------------------------------------
2 * ps_status.c
3 *
4 * Routines to support changing the ps display of PostgreSQL backends
5 * to contain some useful information. Mechanism differs wildly across
6 * platforms.
7 *
8 * src/backend/utils/misc/ps_status.c
9 *
10 * Copyright (c) 2000-2026, PostgreSQL Global Development Group
11 * various details abducted from various places
12 *--------------------------------------------------------------------
13 */
14
15#include "postgres.h"
16
17#include <unistd.h>
18#if defined(__darwin__)
19#include <crt_externs.h>
20#endif
21
22#include "miscadmin.h"
23#include "utils/guc.h"
24#include "utils/ps_status.h"
25
26#if !defined(WIN32)
27extern char **environ;
28#endif
29
30/* GUC variable */
32
33/*
34 * Alternative ways of updating ps display:
35 *
36 * PS_USE_SETPROCTITLE_FAST
37 * use the function setproctitle_fast(const char *, ...)
38 * (FreeBSD)
39 * PS_USE_SETPROCTITLE
40 * use the function setproctitle(const char *, ...)
41 * (other BSDs)
42 * PS_USE_CLOBBER_ARGV
43 * write over the argv and environment area
44 * (Linux and most SysV-like systems)
45 * PS_USE_WIN32
46 * push the string out as the name of a Windows event
47 * PS_USE_NONE
48 * don't update ps display
49 * (This is the default, as it is safest.)
50 */
51#if defined(HAVE_SETPROCTITLE_FAST)
52#define PS_USE_SETPROCTITLE_FAST
53#elif defined(HAVE_SETPROCTITLE)
54#define PS_USE_SETPROCTITLE
55#elif defined(__linux__) || defined(_AIX) || defined(__sun) || defined(__darwin__) || defined(__GNU__)
56#define PS_USE_CLOBBER_ARGV
57#elif defined(WIN32)
58#define PS_USE_WIN32
59#else
60#define PS_USE_NONE
61#endif
62
63
64/* Different systems want the buffer padded differently */
65#if defined(_AIX) || defined(__linux__) || defined(__darwin__) || defined(__GNU__)
66#define PS_PADDING '\0'
67#else
68#define PS_PADDING ' '
69#endif
70
71
72#ifndef PS_USE_NONE
73
74#ifndef PS_USE_CLOBBER_ARGV
75/* all but one option need a buffer to write their ps line in */
76#define PS_BUFFER_SIZE 256
77static char ps_buffer[PS_BUFFER_SIZE];
78static const size_t ps_buffer_size = PS_BUFFER_SIZE;
79#else /* PS_USE_CLOBBER_ARGV */
80static char *ps_buffer; /* will point to argv area */
81static size_t ps_buffer_size; /* space determined at run time */
82static size_t last_status_len; /* use to minimize length of clobber */
83#endif /* PS_USE_CLOBBER_ARGV */
84
85static size_t ps_buffer_cur_len; /* nominal strlen(ps_buffer) */
86
87static size_t ps_buffer_fixed_size; /* size of the constant prefix */
88
89/*
90 * Length of ps_buffer before the suffix was appended to the end, or 0 if we
91 * didn't set a suffix.
92 */
93static size_t ps_buffer_nosuffix_len;
94
95static void flush_ps_display(void);
96
97#endif /* not PS_USE_NONE */
98
99/* save the original argv[] location here */
100static int save_argc;
101static char **save_argv;
102
103/*
104 * Valgrind seems not to consider the global "environ" variable as a valid
105 * root pointer; so when we allocate a new environment array, it claims that
106 * data is leaked. To fix that, keep our own statically-allocated copy of the
107 * pointer. (Oddly, this doesn't seem to be a problem for "argv".)
108 */
109#if defined(PS_USE_CLOBBER_ARGV) && defined(USE_VALGRIND)
110extern char **ps_status_new_environ;
112#endif
113
114
115/*
116 * Call this early in startup to save the original argc/argv values.
117 * If needed, we make a copy of the original argv[] array to preserve it
118 * from being clobbered by subsequent ps_display actions.
119 *
120 * (The original argv[] will not be overwritten by this routine, but may be
121 * overwritten during init_ps_display. Also, the physical location of the
122 * environment strings may be moved, so this should be called before any code
123 * that might try to hang onto a getenv() result. But see hack for musl
124 * within.)
125 *
126 * Note that in case of failure this cannot call elog() as that is not
127 * initialized yet. We rely on write_stderr() instead.
128 */
129char **
130save_ps_display_args(int argc, char **argv)
131{
132 save_argc = argc;
133 save_argv = argv;
134
135#if defined(PS_USE_CLOBBER_ARGV)
136
137 /*
138 * If we're going to overwrite the argv area, count the available space.
139 * Also move the environment strings to make additional room.
140 */
141 {
142 char *end_of_area = NULL;
143 char **new_environ;
144 int i;
145
146 /*
147 * check for contiguous argv strings
148 */
149 for (i = 0; i < argc; i++)
150 {
151 if (i == 0 || end_of_area + 1 == argv[i])
152 end_of_area = argv[i] + strlen(argv[i]);
153 }
154
155 if (end_of_area == NULL) /* probably can't happen? */
156 {
157 ps_buffer = NULL;
158 ps_buffer_size = 0;
159 return argv;
160 }
161
162 /*
163 * check for contiguous environ strings following argv
164 */
165 for (i = 0; environ[i] != NULL; i++)
166 {
167 if (end_of_area + 1 == environ[i])
168 {
169 /*
170 * The musl dynamic linker keeps a static pointer to the
171 * initial value of LD_LIBRARY_PATH, if that is defined in the
172 * process's environment. Therefore, we must not overwrite the
173 * value of that setting and thus cannot advance end_of_area
174 * beyond it. Musl does not define any identifying compiler
175 * symbol, so we have to do this unless we see a symbol
176 * identifying a Linux libc we know is safe.
177 */
178#if defined(__linux__) && (!defined(__GLIBC__) && !defined(__UCLIBC__))
179 if (strncmp(environ[i], "LD_LIBRARY_PATH=", 16) == 0)
180 {
181 /*
182 * We can overwrite the name, but stop at the equals sign.
183 * Future loop iterations will not find any more
184 * contiguous space, but we don't break early because we
185 * need to count the total number of environ[] entries.
186 */
187 end_of_area = environ[i] + 15;
188 }
189 else
190#endif
191 {
193 }
194 }
195 }
196
197 ps_buffer = argv[0];
199
200 /*
201 * move the environment out of the way
202 */
203 new_environ = (char **) malloc((i + 1) * sizeof(char *));
204 if (!new_environ)
205 {
206 write_stderr("out of memory\n");
207 exit(1);
208 }
209 for (i = 0; environ[i] != NULL; i++)
210 {
212 if (!new_environ[i])
213 {
214 write_stderr("out of memory\n");
215 exit(1);
216 }
217 }
218 new_environ[i] = NULL;
220
221 /* See notes about Valgrind above. */
222#ifdef USE_VALGRIND
224#endif
225 }
226
227 /*
228 * If we're going to change the original argv[] then make a copy for
229 * argument parsing purposes.
230 *
231 * NB: do NOT think to remove the copying of argv[], even though
232 * postmaster.c finishes looking at argv[] long before we ever consider
233 * changing the ps display. On some platforms, getopt() keeps pointers
234 * into the argv array, and will get horribly confused when it is
235 * re-called to analyze a subprocess' argument string if the argv storage
236 * has been clobbered meanwhile. Other platforms have other dependencies
237 * on argv[]. (We use custom pg_getopt_start/next() functions nowadays
238 * that don't do that, but those other dependencies might still exist.)
239 */
240 {
241 char **new_argv;
242 int i;
243
244 new_argv = (char **) malloc((argc + 1) * sizeof(char *));
245 if (!new_argv)
246 {
247 write_stderr("out of memory\n");
248 exit(1);
249 }
250 for (i = 0; i < argc; i++)
251 {
252 new_argv[i] = strdup(argv[i]);
253 if (!new_argv[i])
254 {
255 write_stderr("out of memory\n");
256 exit(1);
257 }
258 }
259 new_argv[argc] = NULL;
260
261#if defined(__darwin__)
262
263 /*
264 * macOS has a static copy of the argv pointer, which we may fix like
265 * so:
266 */
267 *_NSGetArgv() = new_argv;
268#endif
269
270 argv = new_argv;
271 }
272#endif /* PS_USE_CLOBBER_ARGV */
273
274 return argv;
275}
276
277/*
278 * Call this once during subprocess startup to set the identification
279 * values.
280 *
281 * If fixed_part is NULL, a default will be obtained from MyBackendType.
282 *
283 * At this point, the original argv[] array may be overwritten.
284 */
285void
287{
288#ifndef PS_USE_NONE
290#endif
291
293 if (!fixed_part)
295
296#ifndef PS_USE_NONE
297 /* no ps display for stand-alone backend */
299 return;
300
301 /* no ps display if you didn't call save_ps_display_args() */
302 if (!save_argv)
303 return;
304
305#ifdef PS_USE_CLOBBER_ARGV
306 /* If ps_buffer is a pointer, it might still be null */
307 if (!ps_buffer)
308 return;
309
310 /* make extra argv slots point at end_of_area (a NUL) */
311 for (int i = 1; i < save_argc; i++)
313#endif /* PS_USE_CLOBBER_ARGV */
314
315 /*
316 * Make fixed prefix of ps display.
317 */
318
319#if defined(PS_USE_SETPROCTITLE) || defined(PS_USE_SETPROCTITLE_FAST)
320
321 /*
322 * apparently setproctitle() already adds a `progname:' prefix to the ps
323 * line
324 */
325#define PROGRAM_NAME_PREFIX ""
326#else
327#define PROGRAM_NAME_PREFIX "postgres: "
328#endif
329
330 if (*cluster_name == '\0')
331 {
334 fixed_part);
335 }
336 else
337 {
339 PROGRAM_NAME_PREFIX "%s: %s ",
341 }
342
344
345 /*
346 * On the first run, force the update.
347 */
350 set_ps_display("");
352#endif /* not PS_USE_NONE */
353}
354
355#ifndef PS_USE_NONE
356/*
357 * update_ps_display_precheck
358 * Helper function to determine if updating the process title is
359 * something that we need to do.
360 */
361static bool
363{
364 /* update_process_title=off disables updates */
366 return false;
367
368 /* no ps display for stand-alone backend */
370 return false;
371
372#ifdef PS_USE_CLOBBER_ARGV
373 /* If ps_buffer is a pointer, it might still be null */
374 if (!ps_buffer)
375 return false;
376#endif
377
378 return true;
379}
380#endif /* not PS_USE_NONE */
381
382/*
383 * set_ps_display_suffix
384 * Adjust the process title to append 'suffix' onto the end with a space
385 * between it and the current process title.
386 */
387void
388set_ps_display_suffix(const char *suffix)
389{
390#ifndef PS_USE_NONE
391 size_t len;
392
393 /* first, check if we need to update the process title */
395 return;
396
397 /* if there's already a suffix, overwrite it */
400 else
402
403 len = strlen(suffix);
404
405 /* check if we have enough space to append the suffix */
407 {
408 /* not enough space. Check the buffer isn't full already */
410 {
411 /* append a space before the suffix */
413
414 /* just add what we can and fill the ps_buffer */
417 ps_buffer[ps_buffer_size - 1] = '\0';
419 }
420 }
421 else
422 {
424 memcpy(ps_buffer + ps_buffer_cur_len, suffix, len + 1);
426 }
427
429
430 /* and set the new title */
432#endif /* not PS_USE_NONE */
433}
434
435/*
436 * set_ps_display_remove_suffix
437 * Remove the process display suffix added by set_ps_display_suffix
438 */
439void
441{
442#ifndef PS_USE_NONE
443 /* first, check if we need to update the process title */
445 return;
446
447 /* check we added a suffix */
448 if (ps_buffer_nosuffix_len == 0)
449 return; /* no suffix */
450
451 /* remove the suffix from ps_buffer */
455
457
458 /* and set the new title */
460#endif /* not PS_USE_NONE */
461}
462
463/*
464 * Call this to update the ps status display to a fixed prefix plus an
465 * indication of what you're currently doing passed in the argument.
466 *
467 * 'len' must be the same as strlen(activity)
468 */
469void
471{
473
474#ifndef PS_USE_NONE
475 /* first, check if we need to update the process title */
477 return;
478
479 /* wipe out any suffix when the title is completely changed */
481
482 /* Update ps_buffer to contain both fixed part and activity */
484 {
485 /* handle the case where ps_buffer doesn't have enough space */
488 ps_buffer[ps_buffer_size - 1] = '\0';
490 }
491 else
492 {
495 }
497
498 /* Transmit new setting to kernel, if necessary */
500#endif /* not PS_USE_NONE */
501}
502
503#ifndef PS_USE_NONE
504static void
506{
507#ifdef PS_USE_SETPROCTITLE
508 setproctitle("%s", ps_buffer);
509#elif defined(PS_USE_SETPROCTITLE_FAST)
511#endif
512
513#ifdef PS_USE_CLOBBER_ARGV
514 /* pad unused memory; need only clobber remainder of old status string */
519#endif /* PS_USE_CLOBBER_ARGV */
520
521#ifdef PS_USE_WIN32
522 {
523 /*
524 * Win32 does not support showing any changed arguments. To make it at
525 * all possible to track which backend is doing what, we create a
526 * named object that can be viewed with for example Process Explorer.
527 */
529 char name[PS_BUFFER_SIZE + 32];
530
533
534 sprintf(name, "pgident(%d): %s", MyProcPid, ps_buffer);
535
537 }
538#endif /* PS_USE_WIN32 */
539}
540#endif /* not PS_USE_NONE */
541
542/*
543 * Returns what's currently in the ps display, in case someone needs
544 * it. Note that only the activity part is returned. On some platforms
545 * the string will not be null-terminated, so return the effective
546 * length into *displen.
547 */
548const char *
550{
551#ifdef PS_USE_CLOBBER_ARGV
552 /* If ps_buffer is a pointer, it might still be null */
553 if (!ps_buffer)
554 {
555 *displen = 0;
556 return "";
557 }
558#endif
559
560#ifndef PS_USE_NONE
562
564#else
565 *displen = 0;
566 return "";
567#endif
568}
#define write_stderr(str)
Definition parallel.c:186
#define Assert(condition)
Definition c.h:943
#define MemSet(start, val, len)
Definition c.h:1107
memcpy(sums, checksumBaseOffsets, sizeof(checksumBaseOffsets))
int MyProcPid
Definition globals.c:49
bool IsUnderPostmaster
Definition globals.c:122
char * cluster_name
Definition guc_tables.c:582
int i
Definition isn.c:77
const char * GetBackendTypeDesc(BackendType backendType)
Definition miscinit.c:264
BackendType MyBackendType
Definition miscinit.c:65
const void size_t len
#define sprintf
Definition port.h:262
#define snprintf
Definition port.h:260
static int fb(int x)
#define PS_PADDING
Definition ps_status.c:68
const char * get_ps_display(int *displen)
Definition ps_status.c:549
static int save_argc
Definition ps_status.c:100
void set_ps_display_remove_suffix(void)
Definition ps_status.c:440
void set_ps_display_with_len(const char *activity, size_t len)
Definition ps_status.c:470
void init_ps_display(const char *fixed_part)
Definition ps_status.c:286
void set_ps_display_suffix(const char *suffix)
Definition ps_status.c:388
char ** environ
bool update_process_title
Definition ps_status.c:31
char ** save_ps_display_args(int argc, char **argv)
Definition ps_status.c:130
static char ** save_argv
Definition ps_status.c:101
static void set_ps_display(const char *activity)
Definition ps_status.h:40
#define DEFAULT_UPDATE_PROCESS_TITLE
Definition ps_status.h:19
#define malloc(a)
const char * name