LCOV - differential code coverage report
Current view: top level - src/bin/pg_resetwal - pg_resetwal.c (source / functions) Coverage Total Hit UNC UBC GBC GNC CBC DUB DCB
Current: Differential Code Coverage 16@8cea358b128 vs 17@8cea358b128 Lines: 89.8 % 522 469 7 46 65 40 364 14 12
Current Date: 2024-04-14 14:21:10 Functions: 100.0 % 13 13 3 10
Baseline: 16@8cea358b128 Branches: 72.9 % 303 221 12 70 39 26 156
Baseline Date: 2024-04-14 14:21:09 Line coverage date bins:
Legend: Lines: hit not hit | Branches: + taken - not taken # not executed (60,120] days: 56.2 % 16 9 7 9
(180,240] days: 97.0 % 33 32 1 30 2
(240..) days: 90.5 % 473 428 45 65 1 362
Function coverage date bins:
(60,120] days: 100.0 % 1 1 1
(240..) days: 100.0 % 12 12 2 10
Branch coverage date bins:
(60,120] days: 42.9 % 14 6 8 6
(180,240] days: 80.8 % 26 21 4 1 20 1
(240..) days: 73.8 % 263 194 69 39 155

 Age         Owner                    Branch data    TLA  Line data    Source code
                                  1                 :                : /*-------------------------------------------------------------------------
                                  2                 :                :  *
                                  3                 :                :  * pg_resetwal.c
                                  4                 :                :  *    A utility to "zero out" the xlog when it's corrupt beyond recovery.
                                  5                 :                :  *    Can also rebuild pg_control if needed.
                                  6                 :                :  *
                                  7                 :                :  * The theory of operation is fairly simple:
                                  8                 :                :  *    1. Read the existing pg_control (which will include the last
                                  9                 :                :  *       checkpoint record).
                                 10                 :                :  *    2. If pg_control is corrupt, attempt to intuit reasonable values,
                                 11                 :                :  *       by scanning the old xlog if necessary.
                                 12                 :                :  *    3. Modify pg_control to reflect a "shutdown" state with a checkpoint
                                 13                 :                :  *       record at the start of xlog.
                                 14                 :                :  *    4. Flush the existing xlog files and write a new segment with
                                 15                 :                :  *       just a checkpoint record in it.  The new segment is positioned
                                 16                 :                :  *       just past the end of the old xlog, so that existing LSNs in
                                 17                 :                :  *       data pages will appear to be "in the past".
                                 18                 :                :  * This is all pretty straightforward except for the intuition part of
                                 19                 :                :  * step 2 ...
                                 20                 :                :  *
                                 21                 :                :  *
                                 22                 :                :  * Portions Copyright (c) 1996-2024, PostgreSQL Global Development Group
                                 23                 :                :  * Portions Copyright (c) 1994, Regents of the University of California
                                 24                 :                :  *
                                 25                 :                :  * src/bin/pg_resetwal/pg_resetwal.c
                                 26                 :                :  *
                                 27                 :                :  *-------------------------------------------------------------------------
                                 28                 :                :  */
                                 29                 :                : 
                                 30                 :                : /*
                                 31                 :                :  * We have to use postgres.h not postgres_fe.h here, because there's so much
                                 32                 :                :  * backend-only stuff in the XLOG include files we need.  But we need a
                                 33                 :                :  * frontend-ish environment otherwise.  Hence this ugly hack.
                                 34                 :                :  */
                                 35                 :                : #define FRONTEND 1
                                 36                 :                : 
                                 37                 :                : #include "postgres.h"
                                 38                 :                : 
                                 39                 :                : #include <dirent.h>
                                 40                 :                : #include <fcntl.h>
                                 41                 :                : #include <sys/stat.h>
                                 42                 :                : #include <sys/time.h>
                                 43                 :                : #include <time.h>
                                 44                 :                : #include <unistd.h>
                                 45                 :                : 
                                 46                 :                : #include "access/heaptoast.h"
                                 47                 :                : #include "access/multixact.h"
                                 48                 :                : #include "access/transam.h"
                                 49                 :                : #include "access/xlog.h"
                                 50                 :                : #include "access/xlog_internal.h"
                                 51                 :                : #include "common/controldata_utils.h"
                                 52                 :                : #include "common/fe_memutils.h"
                                 53                 :                : #include "common/file_perm.h"
                                 54                 :                : #include "common/logging.h"
                                 55                 :                : #include "common/restricted_token.h"
                                 56                 :                : #include "common/string.h"
                                 57                 :                : #include "fe_utils/option_utils.h"
                                 58                 :                : #include "getopt_long.h"
                                 59                 :                : #include "pg_getopt.h"
                                 60                 :                : #include "storage/large_object.h"
                                 61                 :                : 
                                 62                 :                : static ControlFileData ControlFile; /* pg_control values */
                                 63                 :                : static XLogSegNo newXlogSegNo;  /* new XLOG segment # */
                                 64                 :                : static bool guessed = false;    /* T if we had to guess at any values */
                                 65                 :                : static const char *progname;
                                 66                 :                : static uint32 set_xid_epoch = (uint32) -1;
                                 67                 :                : static TransactionId set_oldest_xid = 0;
                                 68                 :                : static TransactionId set_xid = 0;
                                 69                 :                : static TransactionId set_oldest_commit_ts_xid = 0;
                                 70                 :                : static TransactionId set_newest_commit_ts_xid = 0;
                                 71                 :                : static Oid  set_oid = 0;
                                 72                 :                : static MultiXactId set_mxid = 0;
                                 73                 :                : static MultiXactOffset set_mxoff = (MultiXactOffset) -1;
                                 74                 :                : static TimeLineID minXlogTli = 0;
                                 75                 :                : static XLogSegNo minXlogSegNo = 0;
                                 76                 :                : static int  WalSegSz;
                                 77                 :                : static int  set_wal_segsize;
                                 78                 :                : 
                                 79                 :                : static void CheckDataVersion(void);
                                 80                 :                : static bool read_controlfile(void);
                                 81                 :                : static void GuessControlValues(void);
                                 82                 :                : static void PrintControlValues(bool guessed);
                                 83                 :                : static void PrintNewControlValues(void);
                                 84                 :                : static void RewriteControlFile(void);
                                 85                 :                : static void FindEndOfXLOG(void);
                                 86                 :                : static void KillExistingXLOG(void);
                                 87                 :                : static void KillExistingArchiveStatus(void);
                                 88                 :                : static void KillExistingWALSummaries(void);
                                 89                 :                : static void WriteEmptyXLOG(void);
                                 90                 :                : static void usage(void);
                                 91                 :                : 
                                 92                 :                : 
                                 93                 :                : int
 7899 peter_e@gmx.net            94                 :CBC         108 : main(int argc, char *argv[])
                                 95                 :                : {
                                 96                 :                :     static struct option long_options[] = {
                                 97                 :                :         {"commit-timestamp-ids", required_argument, NULL, 'c'},
                                 98                 :                :         {"pgdata", required_argument, NULL, 'D'},
                                 99                 :                :         {"epoch", required_argument, NULL, 'e'},
                                100                 :                :         {"force", no_argument, NULL, 'f'},
                                101                 :                :         {"next-wal-file", required_argument, NULL, 'l'},
                                102                 :                :         {"multixact-ids", required_argument, NULL, 'm'},
                                103                 :                :         {"dry-run", no_argument, NULL, 'n'},
                                104                 :                :         {"next-oid", required_argument, NULL, 'o'},
                                105                 :                :         {"multixact-offset", required_argument, NULL, 'O'},
                                106                 :                :         {"oldest-transaction-id", required_argument, NULL, 'u'},
                                107                 :                :         {"next-transaction-id", required_argument, NULL, 'x'},
                                108                 :                :         {"wal-segsize", required_argument, NULL, 1},
                                109                 :                :         {NULL, 0, NULL, 0}
                                110                 :                :     };
                                111                 :                : 
                                112                 :                :     int         c;
                                113                 :            108 :     bool        force = false;
                                114                 :            108 :     bool        noupdate = false;
 4099 alvherre@alvh.no-ip.      115                 :            108 :     MultiXactId set_oldestmxid = 0;
                                116                 :                :     char       *endptr;
                                117                 :                :     char       *endptr2;
 3489 heikki.linnakangas@i      118                 :            108 :     char       *DataDir = NULL;
 2399 andres@anarazel.de        119                 :            108 :     char       *log_fname = NULL;
                                120                 :                :     int         fd;
                                121                 :                : 
 1840 peter@eisentraut.org      122                 :            108 :     pg_logging_init(argv[0]);
 2621 rhaas@postgresql.org      123                 :            108 :     set_pglocale_pgservice(argv[0], PG_TEXTDOMAIN("pg_resetwal"));
 7681 bruce@momjian.us          124                 :            108 :     progname = get_progname(argv[0]);
                                125                 :                : 
 7899 peter_e@gmx.net           126         [ +  + ]:            108 :     if (argc > 1)
                                127                 :                :     {
                                128   [ +  +  -  + ]:            107 :         if (strcmp(argv[1], "--help") == 0 || strcmp(argv[1], "-?") == 0)
                                129                 :                :         {
                                130                 :              1 :             usage();
                                131                 :              1 :             exit(0);
                                132                 :                :         }
                                133   [ +  +  +  + ]:            106 :         if (strcmp(argv[1], "--version") == 0 || strcmp(argv[1], "-V") == 0)
                                134                 :                :         {
 2621 rhaas@postgresql.org      135                 :             28 :             puts("pg_resetwal (PostgreSQL) " PG_VERSION);
 7899 peter_e@gmx.net           136                 :             28 :             exit(0);
                                137                 :                :         }
                                138                 :                :     }
                                139                 :                : 
                                140                 :                : 
  993 bruce@momjian.us          141         [ +  + ]:            164 :     while ((c = getopt_long(argc, argv, "c:D:e:fl:m:no:O:u:x:", long_options, NULL)) != -1)
                                142                 :                :     {
 7899 peter_e@gmx.net           143   [ +  +  +  +  :            107 :         switch (c)
                                     +  +  +  +  +  
                                        +  +  +  + ]
                                144                 :                :         {
 3489 heikki.linnakangas@i      145                 :GBC           4 :             case 'D':
                                146                 :              4 :                 DataDir = optarg;
                                147                 :              4 :                 break;
                                148                 :                : 
 7899 peter_e@gmx.net           149                 :CBC          14 :             case 'f':
                                150                 :             14 :                 force = true;
                                151                 :             14 :                 break;
                                152                 :                : 
                                153                 :             24 :             case 'n':
                                154                 :             24 :                 noupdate = true;
                                155                 :             24 :                 break;
                                156                 :                : 
 6446 tgl@sss.pgh.pa.us         157                 :              8 :             case 'e':
  968 peter@eisentraut.org      158                 :              8 :                 errno = 0;
 6446 tgl@sss.pgh.pa.us         159                 :              8 :                 set_xid_epoch = strtoul(optarg, &endptr, 0);
  968 peter@eisentraut.org      160   [ +  +  +  -  :              8 :                 if (endptr == optarg || *endptr != '\0' || errno != 0)
                                              -  + ]
                                161                 :                :                 {
                                162                 :                :                     /*------
                                163                 :                :                       translator: the second %s is a command line argument (-e, etc) */
 1840 peter@eisentraut.org      164                 :GBC           1 :                     pg_log_error("invalid argument for option %s", "-e");
  737 tgl@sss.pgh.pa.us         165                 :              1 :                     pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 6446                           166                 :              1 :                     exit(1);
                                167                 :                :                 }
 6446 tgl@sss.pgh.pa.us         168         [ +  + ]:CBC           7 :                 if (set_xid_epoch == -1)
  737 tgl@sss.pgh.pa.us         169                 :GBC           1 :                     pg_fatal("transaction ID epoch (-e) must not be -1");
 6446 tgl@sss.pgh.pa.us         170                 :CBC           6 :                 break;
                                171                 :                : 
  993 bruce@momjian.us          172                 :              7 :             case 'u':
  968 peter@eisentraut.org      173                 :              7 :                 errno = 0;
  993 bruce@momjian.us          174                 :              7 :                 set_oldest_xid = strtoul(optarg, &endptr, 0);
  968 peter@eisentraut.org      175   [ +  +  +  -  :              7 :                 if (endptr == optarg || *endptr != '\0' || errno != 0)
                                              -  + ]
                                176                 :                :                 {
  993 bruce@momjian.us          177                 :GBC           1 :                     pg_log_error("invalid argument for option %s", "-u");
  737 tgl@sss.pgh.pa.us         178                 :              1 :                     pg_log_error_hint("Try \"%s --help\" for more information.", progname);
  993 bruce@momjian.us          179                 :              1 :                     exit(1);
                                180                 :                :                 }
  993 bruce@momjian.us          181         [ +  + ]:CBC           6 :                 if (!TransactionIdIsNormal(set_oldest_xid))
  737 tgl@sss.pgh.pa.us         182                 :GBC           1 :                     pg_fatal("oldest transaction ID (-u) must be greater than or equal to %u", FirstNormalTransactionId);
  993 bruce@momjian.us          183                 :CBC           5 :                 break;
                                184                 :                : 
 7899 peter_e@gmx.net           185                 :              7 :             case 'x':
  968 peter@eisentraut.org      186                 :              7 :                 errno = 0;
 7865 tgl@sss.pgh.pa.us         187                 :              7 :                 set_xid = strtoul(optarg, &endptr, 0);
  968 peter@eisentraut.org      188   [ +  +  +  -  :              7 :                 if (endptr == optarg || *endptr != '\0' || errno != 0)
                                              -  + ]
                                189                 :                :                 {
 1840 peter@eisentraut.org      190                 :GBC           1 :                     pg_log_error("invalid argument for option %s", "-x");
  737 tgl@sss.pgh.pa.us         191                 :              1 :                     pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 7865                           192                 :              1 :                     exit(1);
                                193                 :                :                 }
  993 bruce@momjian.us          194         [ +  + ]:CBC           6 :                 if (!TransactionIdIsNormal(set_xid))
  737 tgl@sss.pgh.pa.us         195                 :GBC           1 :                     pg_fatal("transaction ID (-x) must be greater than or equal to %u", FirstNormalTransactionId);
 7899 peter_e@gmx.net           196                 :CBC           5 :                 break;
                                197                 :                : 
 3420 alvherre@alvh.no-ip.      198                 :              9 :             case 'c':
  968 peter@eisentraut.org      199                 :              9 :                 errno = 0;
 3030 mail@joeconway.com        200                 :              9 :                 set_oldest_commit_ts_xid = strtoul(optarg, &endptr, 0);
  968 peter@eisentraut.org      201   [ +  +  +  -  :              9 :                 if (endptr == optarg || *endptr != ',' || errno != 0)
                                              -  + ]
                                202                 :                :                 {
 1840 peter@eisentraut.org      203                 :GBC           1 :                     pg_log_error("invalid argument for option %s", "-c");
  737 tgl@sss.pgh.pa.us         204                 :              1 :                     pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 3420 alvherre@alvh.no-ip.      205                 :              1 :                     exit(1);
                                206                 :                :                 }
 3030 mail@joeconway.com        207                 :CBC           8 :                 set_newest_commit_ts_xid = strtoul(endptr + 1, &endptr2, 0);
  968 peter@eisentraut.org      208   [ +  +  +  -  :              8 :                 if (endptr2 == endptr + 1 || *endptr2 != '\0' || errno != 0)
                                              -  + ]
                                209                 :                :                 {
 1840 peter@eisentraut.org      210                 :GBC           1 :                     pg_log_error("invalid argument for option %s", "-c");
  737 tgl@sss.pgh.pa.us         211                 :              1 :                     pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 3420 alvherre@alvh.no-ip.      212                 :              1 :                     exit(1);
                                213                 :                :                 }
                                214                 :                : 
  187 peter@eisentraut.org      215         [ +  + ]:GNC           7 :                 if (set_oldest_commit_ts_xid < FirstNormalTransactionId &&
                                216         [ +  - ]:              1 :                     set_oldest_commit_ts_xid != InvalidTransactionId)
                                217                 :              1 :                     pg_fatal("transaction ID (-c) must be either %u or greater than or equal to %u", InvalidTransactionId, FirstNormalTransactionId);
                                218                 :                : 
                                219         [ +  + ]:              6 :                 if (set_newest_commit_ts_xid < FirstNormalTransactionId &&
                                220         [ +  + ]:              3 :                     set_newest_commit_ts_xid != InvalidTransactionId)
                                221                 :              1 :                     pg_fatal("transaction ID (-c) must be either %u or greater than or equal to %u", InvalidTransactionId, FirstNormalTransactionId);
 3420 alvherre@alvh.no-ip.      222                 :CBC           5 :                 break;
                                223                 :                : 
 7865 tgl@sss.pgh.pa.us         224                 :              7 :             case 'o':
  968 peter@eisentraut.org      225                 :              7 :                 errno = 0;
 7865 tgl@sss.pgh.pa.us         226                 :              7 :                 set_oid = strtoul(optarg, &endptr, 0);
  968 peter@eisentraut.org      227   [ +  +  +  -  :              7 :                 if (endptr == optarg || *endptr != '\0' || errno != 0)
                                              -  + ]
                                228                 :                :                 {
 1840 peter@eisentraut.org      229                 :GBC           1 :                     pg_log_error("invalid argument for option %s", "-o");
  737 tgl@sss.pgh.pa.us         230                 :              1 :                     pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 7865                           231                 :              1 :                     exit(1);
                                232                 :                :                 }
 7865 tgl@sss.pgh.pa.us         233         [ +  + ]:CBC           6 :                 if (set_oid == 0)
  737 tgl@sss.pgh.pa.us         234                 :GBC           1 :                     pg_fatal("OID (-o) must not be 0");
 7865 tgl@sss.pgh.pa.us         235                 :CBC           5 :                 break;
                                236                 :                : 
 6926                           237                 :              9 :             case 'm':
  968 peter@eisentraut.org      238                 :              9 :                 errno = 0;
 6926 tgl@sss.pgh.pa.us         239                 :              9 :                 set_mxid = strtoul(optarg, &endptr, 0);
  968 peter@eisentraut.org      240   [ +  +  +  -  :              9 :                 if (endptr == optarg || *endptr != ',' || errno != 0)
                                              -  + ]
                                241                 :                :                 {
 1840 peter@eisentraut.org      242                 :GBC           1 :                     pg_log_error("invalid argument for option %s", "-m");
  737 tgl@sss.pgh.pa.us         243                 :              1 :                     pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 4099 alvherre@alvh.no-ip.      244                 :              1 :                     exit(1);
                                245                 :                :                 }
                                246                 :                : 
 4099 alvherre@alvh.no-ip.      247                 :CBC           8 :                 set_oldestmxid = strtoul(endptr + 1, &endptr2, 0);
  968 peter@eisentraut.org      248   [ +  +  +  -  :              8 :                 if (endptr2 == endptr + 1 || *endptr2 != '\0' || errno != 0)
                                              -  + ]
                                249                 :                :                 {
 1840 peter@eisentraut.org      250                 :GBC           1 :                     pg_log_error("invalid argument for option %s", "-m");
  737 tgl@sss.pgh.pa.us         251                 :              1 :                     pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 6926                           252                 :              1 :                     exit(1);
                                253                 :                :                 }
 6926 tgl@sss.pgh.pa.us         254         [ +  + ]:CBC           7 :                 if (set_mxid == 0)
  737 tgl@sss.pgh.pa.us         255                 :GBC           1 :                     pg_fatal("multitransaction ID (-m) must not be 0");
                                256                 :                : 
                                257                 :                :                 /*
                                258                 :                :                  * XXX It'd be nice to have more sanity checks here, e.g. so
                                259                 :                :                  * that oldest is not wrapped around w.r.t. nextMulti.
                                260                 :                :                  */
 4099 alvherre@alvh.no-ip.      261         [ +  + ]:CBC           6 :                 if (set_oldestmxid == 0)
  737 tgl@sss.pgh.pa.us         262                 :GBC           1 :                     pg_fatal("oldest multitransaction ID (-m) must not be 0");
 6926 tgl@sss.pgh.pa.us         263                 :CBC           5 :                 break;
                                264                 :                : 
 6885                           265                 :              7 :             case 'O':
  968 peter@eisentraut.org      266                 :              7 :                 errno = 0;
 6885 tgl@sss.pgh.pa.us         267                 :              7 :                 set_mxoff = strtoul(optarg, &endptr, 0);
  968 peter@eisentraut.org      268   [ +  +  +  -  :              7 :                 if (endptr == optarg || *endptr != '\0' || errno != 0)
                                              -  + ]
                                269                 :                :                 {
 1840 peter@eisentraut.org      270                 :GBC           1 :                     pg_log_error("invalid argument for option %s", "-O");
  737 tgl@sss.pgh.pa.us         271                 :              1 :                     pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 6885                           272                 :              1 :                     exit(1);
                                273                 :                :                 }
 6885 tgl@sss.pgh.pa.us         274         [ +  + ]:CBC           6 :                 if (set_mxoff == -1)
  737 tgl@sss.pgh.pa.us         275                 :GBC           1 :                     pg_fatal("multitransaction offset (-O) must not be -1");
 6885 tgl@sss.pgh.pa.us         276                 :CBC           5 :                 break;
                                277                 :                : 
 7899 peter_e@gmx.net           278                 :              6 :             case 'l':
 3209 fujii@postgresql.org      279         [ +  + ]:              6 :                 if (strspn(optarg, "01234567890ABCDEFabcdef") != XLOG_FNAME_LEN)
                                280                 :                :                 {
 1840 peter@eisentraut.org      281                 :GBC           1 :                     pg_log_error("invalid argument for option %s", "-l");
  737 tgl@sss.pgh.pa.us         282                 :              1 :                     pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 7899 peter_e@gmx.net           283                 :              1 :                     exit(1);
                                284                 :                :                 }
                                285                 :                : 
                                286                 :                :                 /*
                                287                 :                :                  * XLogFromFileName requires wal segment size which is not yet
                                288                 :                :                  * set. Hence wal details are set later on.
                                289                 :                :                  */
 2399 andres@anarazel.de        290                 :CBC           5 :                 log_fname = pg_strdup(optarg);
 7899 peter_e@gmx.net           291                 :              5 :                 break;
                                292                 :                : 
 2212 peter_e@gmx.net           293                 :GBC           4 :             case 1:
                                294                 :                :                 {
                                295                 :                :                     int         wal_segsize_mb;
                                296                 :                : 
  230 peter@eisentraut.org      297         [ +  + ]:GNC           4 :                     if (!option_parse_int(optarg, "--wal-segsize", 1, 1024, &wal_segsize_mb))
                                298                 :              1 :                         exit(1);
                                299                 :              3 :                     set_wal_segsize = wal_segsize_mb * 1024 * 1024;
                                300   [ +  -  +  +  :              3 :                     if (!IsValidWalSegSize(set_wal_segsize))
                                        +  -  -  + ]
  229 dgustafsson@postgres      301                 :              1 :                         pg_fatal("argument of %s must be a power of two between 1 and 1024", "--wal-segsize");
  230 peter@eisentraut.org      302                 :              2 :                     break;
                                303                 :                :                 }
                                304                 :                : 
 7899 peter_e@gmx.net           305                 :CBC           1 :             default:
                                306                 :                :                 /* getopt_long already emitted a complaint */
  737 tgl@sss.pgh.pa.us         307                 :              1 :                 pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 7899 peter_e@gmx.net           308                 :              1 :                 exit(1);
                                309                 :                :         }
                                310                 :                :     }
                                311                 :                : 
 3460 heikki.linnakangas@i      312   [ +  +  +  + ]:             57 :     if (DataDir == NULL && optind < argc)
                                313                 :             52 :         DataDir = argv[optind++];
                                314                 :                : 
                                315                 :                :     /* Complain if any arguments remain */
                                316         [ +  + ]:             57 :     if (optind < argc)
                                317                 :                :     {
 1840 peter@eisentraut.org      318                 :GBC           1 :         pg_log_error("too many command-line arguments (first is \"%s\")",
                                319                 :                :                      argv[optind]);
  737 tgl@sss.pgh.pa.us         320                 :              1 :         pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 3460 heikki.linnakangas@i      321                 :              1 :         exit(1);
                                322                 :                :     }
                                323                 :                : 
 3460 heikki.linnakangas@i      324         [ +  + ]:CBC          56 :     if (DataDir == NULL)
                                325                 :                :     {
 1840 peter@eisentraut.org      326                 :GBC           1 :         pg_log_error("no data directory specified");
  737 tgl@sss.pgh.pa.us         327                 :              1 :         pg_log_error_hint("Try \"%s --help\" for more information.", progname);
 7899 peter_e@gmx.net           328                 :              1 :         exit(1);
                                329                 :                :     }
                                330                 :                : 
                                331                 :                :     /*
                                332                 :                :      * Don't allow pg_resetwal to be run as root, to avoid overwriting the
                                333                 :                :      * ownership of files in the data directory. We need only check for root
                                334                 :                :      * -- any other user won't have sufficient permissions to modify files in
                                335                 :                :      * the data directory.
                                336                 :                :      */
                                337                 :                : #ifndef WIN32
 7061 neilc@samurai.com         338         [ -  + ]:CBC          55 :     if (geteuid() == 0)
                                339                 :                :     {
 1840 peter@eisentraut.org      340                 :UBC           0 :         pg_log_error("cannot be executed by \"root\"");
  737 tgl@sss.pgh.pa.us         341                 :              0 :         pg_log_error_hint("You must run %s as the PostgreSQL superuser.",
                                342                 :                :                           progname);
 7061 neilc@samurai.com         343                 :              0 :         exit(1);
                                344                 :                :     }
                                345                 :                : #endif
                                346                 :                : 
 1840 peter@eisentraut.org      347                 :CBC          55 :     get_restricted_token();
                                348                 :                : 
                                349                 :                :     /* Set mask based on PGDATA permissions */
 2199 sfrost@snowman.net        350         [ +  + ]:             55 :     if (!GetDataDirectoryCreatePerm(DataDir))
  737 tgl@sss.pgh.pa.us         351                 :GBC           1 :         pg_fatal("could not read permissions of directory \"%s\": %m",
                                352                 :                :                  DataDir);
                                353                 :                : 
 2199 sfrost@snowman.net        354                 :CBC          54 :     umask(pg_mode_mask);
                                355                 :                : 
  198 peter@eisentraut.org      356         [ -  + ]:             54 :     if (chdir(DataDir) < 0)
  198 peter@eisentraut.org      357                 :UBC           0 :         pg_fatal("could not change directory to \"%s\": %m",
                                358                 :                :                  DataDir);
                                359                 :                : 
                                360                 :                :     /* Check that data directory matches our server version */
 2512 tgl@sss.pgh.pa.us         361                 :CBC          54 :     CheckDataVersion();
                                362                 :                : 
                                363                 :                :     /*
                                364                 :                :      * Check for a postmaster lock file --- if there is one, refuse to
                                365                 :                :      * proceed, on grounds we might be interfering with a live installation.
                                366                 :                :      */
 4161                           367         [ +  + ]:             54 :     if ((fd = open("postmaster.pid", O_RDONLY, 0)) < 0)
                                368                 :                :     {
 7899 peter_e@gmx.net           369         [ -  + ]:             53 :         if (errno != ENOENT)
  737 tgl@sss.pgh.pa.us         370                 :UBC           0 :             pg_fatal("could not open file \"%s\" for reading: %m",
                                371                 :                :                      "postmaster.pid");
                                372                 :                :     }
                                373                 :                :     else
                                374                 :                :     {
 1840 peter@eisentraut.org      375                 :GBC           1 :         pg_log_error("lock file \"%s\" exists", "postmaster.pid");
  737 tgl@sss.pgh.pa.us         376                 :              1 :         pg_log_error_hint("Is a server running?  If not, delete the lock file and try again.");
 6525 bruce@momjian.us          377                 :              1 :         exit(1);
                                378                 :                :     }
                                379                 :                : 
                                380                 :                :     /*
                                381                 :                :      * Attempt to read the existing pg_control file
                                382                 :                :      */
 1518 peter@eisentraut.org      383         [ +  + ]:CBC          53 :     if (!read_controlfile())
 6525 bruce@momjian.us          384                 :              4 :         GuessControlValues();
                                385                 :                : 
                                386                 :                :     /*
                                387                 :                :      * If no new WAL segment size was specified, use the control file value.
                                388                 :                :      */
 2212 peter_e@gmx.net           389         [ +  + ]:             53 :     if (set_wal_segsize != 0)
 2212 peter_e@gmx.net           390                 :GBC           2 :         WalSegSz = set_wal_segsize;
                                391                 :                :     else
 2212 peter_e@gmx.net           392                 :CBC          51 :         WalSegSz = ControlFile.xlog_seg_size;
                                393                 :                : 
 2399 andres@anarazel.de        394         [ +  + ]:             53 :     if (log_fname != NULL)
                                395                 :              5 :         XLogFromFileName(log_fname, &minXlogTli, &minXlogSegNo, WalSegSz);
                                396                 :                : 
                                397                 :                :     /*
                                398                 :                :      * Also look at existing segment files to set up newXlogSegNo
                                399                 :                :      */
 6337 tgl@sss.pgh.pa.us         400                 :             53 :     FindEndOfXLOG();
                                401                 :                : 
                                402                 :                :     /*
                                403                 :                :      * If we're not going to proceed with the reset, print the current control
                                404                 :                :      * file parameters.
                                405                 :                :      */
 3776 heikki.linnakangas@i      406   [ +  +  +  +  :             53 :     if ((guessed && !force) || noupdate)
                                              +  + ]
                                407                 :             25 :         PrintControlValues(guessed);
                                408                 :                : 
                                409                 :                :     /*
                                410                 :                :      * Adjust fields if required by switches.  (Do this now so that printout,
                                411                 :                :      * if any, includes these values.)
                                412                 :                :      */
 6446 tgl@sss.pgh.pa.us         413         [ +  + ]:             53 :     if (set_xid_epoch != -1)
                                414                 :                :         ControlFile.checkPointCopy.nextXid =
 1844 tmunro@postgresql.or      415                 :              6 :             FullTransactionIdFromEpochAndXid(set_xid_epoch,
 1342 andres@anarazel.de        416                 :              6 :                                              XidFromFullTransactionId(ControlFile.checkPointCopy.nextXid));
                                417                 :                : 
  993 bruce@momjian.us          418         [ +  + ]:             53 :     if (set_oldest_xid != 0)
                                419                 :                :     {
                                420                 :              5 :         ControlFile.checkPointCopy.oldestXid = set_oldest_xid;
                                421                 :              5 :         ControlFile.checkPointCopy.oldestXidDB = InvalidOid;
                                422                 :                :     }
                                423                 :                : 
                                424         [ +  + ]:             53 :     if (set_xid != 0)
                                425                 :                :         ControlFile.checkPointCopy.nextXid =
 1342 andres@anarazel.de        426                 :              5 :             FullTransactionIdFromEpochAndXid(EpochFromFullTransactionId(ControlFile.checkPointCopy.nextXid),
                                427                 :                :                                              set_xid);
                                428                 :                : 
 3030 mail@joeconway.com        429         [ +  + ]:             53 :     if (set_oldest_commit_ts_xid != 0)
                                430                 :              5 :         ControlFile.checkPointCopy.oldestCommitTsXid = set_oldest_commit_ts_xid;
                                431         [ +  + ]:             53 :     if (set_newest_commit_ts_xid != 0)
                                432                 :              3 :         ControlFile.checkPointCopy.newestCommitTsXid = set_newest_commit_ts_xid;
                                433                 :                : 
 7865 tgl@sss.pgh.pa.us         434         [ +  + ]:             53 :     if (set_oid != 0)
                                435                 :              5 :         ControlFile.checkPointCopy.nextOid = set_oid;
                                436                 :                : 
 6926                           437         [ +  + ]:             53 :     if (set_mxid != 0)
                                438                 :                :     {
                                439                 :              5 :         ControlFile.checkPointCopy.nextMulti = set_mxid;
                                440                 :                : 
 4099 alvherre@alvh.no-ip.      441                 :              5 :         ControlFile.checkPointCopy.oldestMulti = set_oldestmxid;
                                442         [ -  + ]:              5 :         if (ControlFile.checkPointCopy.oldestMulti < FirstMultiXactId)
 4099 alvherre@alvh.no-ip.      443                 :UBC           0 :             ControlFile.checkPointCopy.oldestMulti += FirstMultiXactId;
 4099 alvherre@alvh.no-ip.      444                 :CBC           5 :         ControlFile.checkPointCopy.oldestMultiDB = InvalidOid;
                                445                 :                :     }
                                446                 :                : 
 6885 tgl@sss.pgh.pa.us         447         [ +  + ]:             53 :     if (set_mxoff != -1)
                                448                 :              5 :         ControlFile.checkPointCopy.nextMultiOffset = set_mxoff;
                                449                 :                : 
 7055                           450         [ -  + ]:             53 :     if (minXlogTli > ControlFile.checkPointCopy.ThisTimeLineID)
                                451                 :                :     {
 7055 tgl@sss.pgh.pa.us         452                 :UBC           0 :         ControlFile.checkPointCopy.ThisTimeLineID = minXlogTli;
 4080 heikki.linnakangas@i      453                 :              0 :         ControlFile.checkPointCopy.PrevTimeLineID = minXlogTli;
                                454                 :                :     }
                                455                 :                : 
 2212 peter_e@gmx.net           456         [ +  + ]:CBC          53 :     if (set_wal_segsize != 0)
 2212 peter_e@gmx.net           457                 :GBC           2 :         ControlFile.xlog_seg_size = WalSegSz;
                                458                 :                : 
 4312 heikki.linnakangas@i      459         [ +  + ]:CBC          53 :     if (minXlogSegNo > newXlogSegNo)
                                460                 :              3 :         newXlogSegNo = minXlogSegNo;
                                461                 :                : 
  199 peter@eisentraut.org      462         [ +  + ]:GNC          53 :     if (noupdate)
                                463                 :                :     {
  199 peter@eisentraut.org      464                 :CBC          24 :         PrintNewControlValues();
  199 peter@eisentraut.org      465                 :GNC          24 :         exit(0);
                                466                 :                :     }
                                467                 :                : 
                                468                 :                :     /*
                                469                 :                :      * If we had to guess anything, and -f was not given, just print the
                                470                 :                :      * guessed values and exit.
                                471                 :                :      */
                                472   [ +  +  +  + ]:             29 :     if (guessed && !force)
                                473                 :                :     {
 3776 heikki.linnakangas@i      474                 :              1 :         PrintNewControlValues();
  199 peter@eisentraut.org      475                 :              1 :         pg_log_error("not proceeding because control file values were guessed");
                                476                 :              1 :         pg_log_error_hint("If these values seem acceptable, use -f to force reset.");
                                477                 :              1 :         exit(1);
                                478                 :                :     }
                                479                 :                : 
                                480                 :                :     /*
                                481                 :                :      * Don't reset from a dirty pg_control without -f, either.
                                482                 :                :      */
 6525 bruce@momjian.us          483   [ +  +  +  + ]:CBC          28 :     if (ControlFile.state != DB_SHUTDOWNED && !force)
                                484                 :                :     {
  199 peter@eisentraut.org      485                 :GNC           1 :         pg_log_error("database server was not shut down cleanly");
                                486                 :              1 :         pg_log_error_detail("Resetting the write-ahead log might cause data to be lost.");
                                487                 :              1 :         pg_log_error_hint("If you want to proceed anyway, use -f to force reset.");
 7899 peter_e@gmx.net           488                 :GBC           1 :         exit(1);
                                489                 :                :     }
                                490                 :                : 
                                491                 :                :     /*
                                492                 :                :      * Else, do the dirty deed.
                                493                 :                :      */
 7899 peter_e@gmx.net           494                 :CBC          27 :     RewriteControlFile();
                                495                 :             27 :     KillExistingXLOG();
 5460 tgl@sss.pgh.pa.us         496                 :             27 :     KillExistingArchiveStatus();
  116 rhaas@postgresql.org      497                 :GNC          27 :     KillExistingWALSummaries();
 7899 peter_e@gmx.net           498                 :CBC          27 :     WriteEmptyXLOG();
                                499                 :                : 
 2529                           500                 :             27 :     printf(_("Write-ahead log reset\n"));
 7899                           501                 :             27 :     return 0;
                                502                 :                : }
                                503                 :                : 
                                504                 :                : 
                                505                 :                : /*
                                506                 :                :  * Look at the version string stored in PG_VERSION and decide if this utility
                                507                 :                :  * can be run safely or not.
                                508                 :                :  *
                                509                 :                :  * We don't want to inject pg_control and WAL files that are for a different
                                510                 :                :  * major version; that can't do anything good.  Note that we don't treat
                                511                 :                :  * mismatching version info in pg_control as a reason to bail out, because
                                512                 :                :  * recovering from a corrupted pg_control is one of the main reasons for this
                                513                 :                :  * program to exist at all.  However, PG_VERSION is unlikely to get corrupted,
                                514                 :                :  * and if it were it would be easy to fix by hand.  So let's make this check
                                515                 :                :  * to prevent simple user errors.
                                516                 :                :  */
                                517                 :                : static void
 2512 tgl@sss.pgh.pa.us         518                 :             54 : CheckDataVersion(void)
                                519                 :                : {
                                520                 :             54 :     const char *ver_file = "PG_VERSION";
                                521                 :                :     FILE       *ver_fd;
                                522                 :                :     char        rawline[64];
                                523                 :                : 
                                524         [ -  + ]:             54 :     if ((ver_fd = fopen(ver_file, "r")) == NULL)
  737 tgl@sss.pgh.pa.us         525                 :UBC           0 :         pg_fatal("could not open file \"%s\" for reading: %m",
                                526                 :                :                  ver_file);
                                527                 :                : 
                                528                 :                :     /* version number has to be the first line read */
 2512 tgl@sss.pgh.pa.us         529         [ -  + ]:CBC          54 :     if (!fgets(rawline, sizeof(rawline), ver_fd))
                                530                 :                :     {
 2512 tgl@sss.pgh.pa.us         531         [ #  # ]:UBC           0 :         if (!ferror(ver_fd))
  737                           532                 :              0 :             pg_fatal("unexpected empty file \"%s\"", ver_file);
                                533                 :                :         else
                                534                 :              0 :             pg_fatal("could not read file \"%s\": %m", ver_file);
                                535                 :                :     }
                                536                 :                : 
                                537                 :                :     /* strip trailing newline and carriage return */
 1710 michael@paquier.xyz       538                 :CBC          54 :     (void) pg_strip_crlf(rawline);
                                539                 :                : 
 2512 tgl@sss.pgh.pa.us         540         [ -  + ]:             54 :     if (strcmp(rawline, PG_MAJORVERSION) != 0)
                                541                 :                :     {
 1840 peter@eisentraut.org      542                 :UBC           0 :         pg_log_error("data directory is of wrong version");
  737 tgl@sss.pgh.pa.us         543                 :              0 :         pg_log_error_detail("File \"%s\" contains \"%s\", which is not compatible with this program's version \"%s\".",
                                544                 :                :                             ver_file, rawline, PG_MAJORVERSION);
 2512                           545                 :              0 :         exit(1);
                                546                 :                :     }
                                547                 :                : 
 2512 tgl@sss.pgh.pa.us         548                 :CBC          54 :     fclose(ver_fd);
                                549                 :             54 : }
                                550                 :                : 
                                551                 :                : 
                                552                 :                : /*
                                553                 :                :  * Try to read the existing pg_control file.
                                554                 :                :  *
                                555                 :                :  * This routine is also responsible for updating old pg_control versions
                                556                 :                :  * to the current format.  (Currently we don't do anything of the sort.)
                                557                 :                :  */
                                558                 :                : static bool
 1518 peter@eisentraut.org      559                 :             53 : read_controlfile(void)
                                560                 :                : {
                                561                 :                :     int         fd;
                                562                 :                :     int         len;
                                563                 :                :     char       *buffer;
                                564                 :                :     pg_crc32c   crc;
                                565                 :                : 
 5681 magnus@hagander.net       566         [ -  + ]:             53 :     if ((fd = open(XLOG_CONTROL_FILE, O_RDONLY | PG_BINARY, 0)) < 0)
                                567                 :                :     {
                                568                 :                :         /*
                                569                 :                :          * If pg_control is not there at all, or we can't read it, the odds
                                570                 :                :          * are we've been handed a bad DataDir path, so give up. User can do
                                571                 :                :          * "touch pg_control" to force us to proceed.
                                572                 :                :          */
 1840 peter@eisentraut.org      573                 :UBC           0 :         pg_log_error("could not open file \"%s\" for reading: %m",
                                574                 :                :                      XLOG_CONTROL_FILE);
 7911 bruce@momjian.us          575         [ #  # ]:              0 :         if (errno == ENOENT)
  737 tgl@sss.pgh.pa.us         576                 :              0 :             pg_log_error_hint("If you are sure the data directory path is correct, execute\n"
                                577                 :                :                               "  touch %s\n"
                                578                 :                :                               "and try again.",
                                579                 :                :                               XLOG_CONTROL_FILE);
 7911 bruce@momjian.us          580                 :              0 :         exit(1);
                                581                 :                :     }
                                582                 :                : 
                                583                 :                :     /* Use malloc to ensure we have a maxaligned buffer */
 2461 tgl@sss.pgh.pa.us         584                 :CBC          53 :     buffer = (char *) pg_malloc(PG_CONTROL_FILE_SIZE);
                                585                 :                : 
                                586                 :             53 :     len = read(fd, buffer, PG_CONTROL_FILE_SIZE);
 7911 bruce@momjian.us          587         [ -  + ]:             53 :     if (len < 0)
  737 tgl@sss.pgh.pa.us         588                 :UBC           0 :         pg_fatal("could not read file \"%s\": %m", XLOG_CONTROL_FILE);
 7911 bruce@momjian.us          589                 :CBC          53 :     close(fd);
                                590                 :                : 
                                591         [ +  - ]:             53 :     if (len >= sizeof(ControlFileData) &&
 2489 tgl@sss.pgh.pa.us         592         [ +  + ]:             53 :         ((ControlFileData *) buffer)->pg_control_version == PG_CONTROL_VERSION)
                                593                 :                :     {
                                594                 :                :         /* Check the CRC. */
 3449 heikki.linnakangas@i      595                 :             52 :         INIT_CRC32C(crc);
                                596                 :             52 :         COMP_CRC32C(crc,
                                597                 :                :                     buffer,
                                598                 :                :                     offsetof(ControlFileData, crc));
                                599                 :             52 :         FIN_CRC32C(crc);
                                600                 :                : 
 2399 andres@anarazel.de        601         [ +  + ]:             52 :         if (!EQ_CRC32C(crc, ((ControlFileData *) buffer)->crc))
                                602                 :                :         {
                                603                 :                :             /* We will use the data but treat it as guessed. */
 1840 peter@eisentraut.org      604                 :              3 :             pg_log_warning("pg_control exists but has invalid CRC; proceed with caution");
 2399 andres@anarazel.de        605                 :              3 :             guessed = true;
                                606                 :                :         }
                                607                 :                : 
 7911 bruce@momjian.us          608                 :             52 :         memcpy(&ControlFile, buffer, sizeof(ControlFile));
                                609                 :                : 
                                610                 :                :         /* return false if WAL segment size is not valid */
 2212 peter_e@gmx.net           611   [ +  +  +  -  :             52 :         if (!IsValidWalSegSize(ControlFile.xlog_seg_size))
                                        +  -  -  + ]
                                612                 :                :         {
 1840 peter@eisentraut.org      613                 :              3 :             pg_log_warning(ngettext("pg_control specifies invalid WAL segment size (%d byte); proceed with caution",
                                614                 :                :                                     "pg_control specifies invalid WAL segment size (%d bytes); proceed with caution",
                                615                 :                :                                     ControlFile.xlog_seg_size),
                                616                 :                :                            ControlFile.xlog_seg_size);
 2214 peter_e@gmx.net           617                 :              3 :             return false;
                                618                 :                :         }
                                619                 :                : 
 7911 bruce@momjian.us          620                 :             49 :         return true;
                                621                 :                :     }
                                622                 :                : 
                                623                 :                :     /* Looks like it's a mess. */
 1840 peter@eisentraut.org      624                 :              1 :     pg_log_warning("pg_control exists but is broken or wrong version; ignoring it");
 7911 bruce@momjian.us          625                 :              1 :     return false;
                                626                 :                : }
                                627                 :                : 
                                628                 :                : 
                                629                 :                : /*
                                630                 :                :  * Guess at pg_control values when we can't read the old ones.
                                631                 :                :  */
                                632                 :                : static void
 6525                           633                 :              4 : GuessControlValues(void)
                                634                 :                : {
                                635                 :                :     uint64      sysidentifier;
                                636                 :                :     struct timeval tv;
                                637                 :                : 
                                638                 :                :     /*
                                639                 :                :      * Set up a completely default set of pg_control values.
                                640                 :                :      */
                                641                 :              4 :     guessed = true;
 7911                           642                 :              4 :     memset(&ControlFile, 0, sizeof(ControlFile));
                                643                 :                : 
                                644                 :              4 :     ControlFile.pg_control_version = PG_CONTROL_VERSION;
                                645                 :              4 :     ControlFile.catalog_version_no = CATALOG_VERSION_NO;
                                646                 :                : 
                                647                 :                :     /*
                                648                 :                :      * Create a new unique installation identifier, since we can no longer use
                                649                 :                :      * any old XLOG records.  See notes in xlog.c about the algorithm.
                                650                 :                :      */
 6525                           651                 :              4 :     gettimeofday(&tv, NULL);
                                652                 :              4 :     sysidentifier = ((uint64) tv.tv_sec) << 32;
 3610 tgl@sss.pgh.pa.us         653                 :              4 :     sysidentifier |= ((uint64) tv.tv_usec) << 12;
                                654                 :              4 :     sysidentifier |= getpid() & 0xFFF;
                                655                 :                : 
 6525 bruce@momjian.us          656                 :              4 :     ControlFile.system_identifier = sysidentifier;
                                657                 :                : 
 4312 heikki.linnakangas@i      658                 :              4 :     ControlFile.checkPointCopy.redo = SizeOfXLogLongPHD;
 6525 bruce@momjian.us          659                 :              4 :     ControlFile.checkPointCopy.ThisTimeLineID = 1;
 4080 heikki.linnakangas@i      660                 :              4 :     ControlFile.checkPointCopy.PrevTimeLineID = 1;
 4463 simon@2ndQuadrant.co      661                 :              4 :     ControlFile.checkPointCopy.fullPageWrites = false;
                                662                 :                :     ControlFile.checkPointCopy.nextXid =
 1844 tmunro@postgresql.or      663                 :              4 :         FullTransactionIdFromEpochAndXid(0, FirstNormalTransactionId);
 1004 tgl@sss.pgh.pa.us         664                 :              4 :     ControlFile.checkPointCopy.nextOid = FirstGenbkiObjectId;
 6525 bruce@momjian.us          665                 :              4 :     ControlFile.checkPointCopy.nextMulti = FirstMultiXactId;
                                666                 :              4 :     ControlFile.checkPointCopy.nextMultiOffset = 0;
 5340 tgl@sss.pgh.pa.us         667                 :              4 :     ControlFile.checkPointCopy.oldestXid = FirstNormalTransactionId;
                                668                 :              4 :     ControlFile.checkPointCopy.oldestXidDB = InvalidOid;
 4099 alvherre@alvh.no-ip.      669                 :              4 :     ControlFile.checkPointCopy.oldestMulti = FirstMultiXactId;
                                670                 :              4 :     ControlFile.checkPointCopy.oldestMultiDB = InvalidOid;
 5901 tgl@sss.pgh.pa.us         671                 :              4 :     ControlFile.checkPointCopy.time = (pg_time_t) time(NULL);
 5100                           672                 :              4 :     ControlFile.checkPointCopy.oldestActiveXid = InvalidTransactionId;
                                673                 :                : 
 6525 bruce@momjian.us          674                 :              4 :     ControlFile.state = DB_SHUTDOWNED;
 5901 tgl@sss.pgh.pa.us         675                 :              4 :     ControlFile.time = (pg_time_t) time(NULL);
 6525 bruce@momjian.us          676                 :              4 :     ControlFile.checkPoint = ControlFile.checkPointCopy.redo;
 1631 michael@paquier.xyz       677                 :              4 :     ControlFile.unloggedLSN = FirstNormalUnloggedLSN;
                                678                 :                : 
                                679                 :                :     /* minRecoveryPoint, backupStartPoint and backupEndPoint can be left zero */
                                680                 :                : 
 5100 tgl@sss.pgh.pa.us         681                 :              4 :     ControlFile.wal_level = WAL_LEVEL_MINIMAL;
 3767 fujii@postgresql.org      682                 :              4 :     ControlFile.wal_log_hints = false;
 3420 alvherre@alvh.no-ip.      683                 :              4 :     ControlFile.track_commit_timestamp = false;
 5100 tgl@sss.pgh.pa.us         684                 :              4 :     ControlFile.MaxConnections = 100;
 1888 michael@paquier.xyz       685                 :              4 :     ControlFile.max_wal_senders = 10;
 2687 rhaas@postgresql.org      686                 :              4 :     ControlFile.max_worker_processes = 8;
 5100 tgl@sss.pgh.pa.us         687                 :              4 :     ControlFile.max_prepared_xacts = 0;
                                688                 :              4 :     ControlFile.max_locks_per_xact = 64;
                                689                 :                : 
 6768                           690                 :              4 :     ControlFile.maxAlign = MAXIMUM_ALIGNOF;
                                691                 :              4 :     ControlFile.floatFormat = FLOATFORMAT_VALUE;
 7911 bruce@momjian.us          692                 :              4 :     ControlFile.blcksz = BLCKSZ;
                                693                 :              4 :     ControlFile.relseg_size = RELSEG_SIZE;
 2212 peter_e@gmx.net           694                 :              4 :     ControlFile.xlog_blcksz = XLOG_BLCKSZ;
                                695                 :              4 :     ControlFile.xlog_seg_size = DEFAULT_XLOG_SEG_SIZE;
 7865 tgl@sss.pgh.pa.us         696                 :              4 :     ControlFile.nameDataLen = NAMEDATALEN;
 6956                           697                 :              4 :     ControlFile.indexMaxKeys = INDEX_MAX_KEYS;
 6221                           698                 :              4 :     ControlFile.toast_max_chunk_size = TOAST_MAX_CHUNK_SIZE;
 3601                           699                 :              4 :     ControlFile.loblksize = LOBLKSIZE;
 5837                           700                 :              4 :     ControlFile.float8ByVal = FLOAT8PASSBYVAL;
                                701                 :                : 
                                702                 :                :     /*
                                703                 :                :      * XXX eventually, should try to grovel through old XLOG to develop more
                                704                 :                :      * accurate values for TimeLineID, nextXID, etc.
                                705                 :                :      */
 7911 bruce@momjian.us          706                 :              4 : }
                                707                 :                : 
                                708                 :                : 
                                709                 :                : /*
                                710                 :                :  * Print the guessed pg_control values when we had to guess.
                                711                 :                :  *
                                712                 :                :  * NB: this display should be just those fields that will not be
                                713                 :                :  * reset by RewriteControlFile().
                                714                 :                :  */
                                715                 :                : static void
 6525                           716                 :             25 : PrintControlValues(bool guessed)
                                717                 :                : {
                                718         [ +  + ]:             25 :     if (guessed)
                                719                 :              3 :         printf(_("Guessed pg_control values:\n\n"));
                                720                 :                :     else
 3776 heikki.linnakangas@i      721                 :             22 :         printf(_("Current pg_control values:\n\n"));
                                722                 :                : 
 6446 tgl@sss.pgh.pa.us         723                 :             25 :     printf(_("pg_control version number:            %u\n"),
                                724                 :                :            ControlFile.pg_control_version);
                                725                 :             25 :     printf(_("Catalog version number:               %u\n"),
                                726                 :                :            ControlFile.catalog_version_no);
 1774 peter@eisentraut.org      727                 :             25 :     printf(_("Database system identifier:           %llu\n"),
                                728                 :                :            (unsigned long long) ControlFile.system_identifier);
 6446 tgl@sss.pgh.pa.us         729                 :             25 :     printf(_("Latest checkpoint's TimeLineID:       %u\n"),
                                730                 :                :            ControlFile.checkPointCopy.ThisTimeLineID);
 4329 peter_e@gmx.net           731         [ +  + ]:             25 :     printf(_("Latest checkpoint's full_page_writes: %s\n"),
                                732                 :                :            ControlFile.checkPointCopy.fullPageWrites ? _("on") : _("off"));
 2984 mail@joeconway.com        733                 :             25 :     printf(_("Latest checkpoint's NextXID:          %u:%u\n"),
                                734                 :                :            EpochFromFullTransactionId(ControlFile.checkPointCopy.nextXid),
                                735                 :                :            XidFromFullTransactionId(ControlFile.checkPointCopy.nextXid));
 6446 tgl@sss.pgh.pa.us         736                 :             25 :     printf(_("Latest checkpoint's NextOID:          %u\n"),
                                737                 :                :            ControlFile.checkPointCopy.nextOid);
                                738                 :             25 :     printf(_("Latest checkpoint's NextMultiXactId:  %u\n"),
                                739                 :                :            ControlFile.checkPointCopy.nextMulti);
                                740                 :             25 :     printf(_("Latest checkpoint's NextMultiOffset:  %u\n"),
                                741                 :                :            ControlFile.checkPointCopy.nextMultiOffset);
 5340                           742                 :             25 :     printf(_("Latest checkpoint's oldestXID:        %u\n"),
                                743                 :                :            ControlFile.checkPointCopy.oldestXid);
                                744                 :             25 :     printf(_("Latest checkpoint's oldestXID's DB:   %u\n"),
                                745                 :                :            ControlFile.checkPointCopy.oldestXidDB);
 5100                           746                 :             25 :     printf(_("Latest checkpoint's oldestActiveXID:  %u\n"),
                                747                 :                :            ControlFile.checkPointCopy.oldestActiveXid);
 4099 alvherre@alvh.no-ip.      748                 :             25 :     printf(_("Latest checkpoint's oldestMultiXid:   %u\n"),
                                749                 :                :            ControlFile.checkPointCopy.oldestMulti);
                                750                 :             25 :     printf(_("Latest checkpoint's oldestMulti's DB: %u\n"),
                                751                 :                :            ControlFile.checkPointCopy.oldestMultiDB);
 3030 mail@joeconway.com        752                 :             25 :     printf(_("Latest checkpoint's oldestCommitTsXid:%u\n"),
                                753                 :                :            ControlFile.checkPointCopy.oldestCommitTsXid);
                                754                 :             25 :     printf(_("Latest checkpoint's newestCommitTsXid:%u\n"),
                                755                 :                :            ControlFile.checkPointCopy.newestCommitTsXid);
 6446 tgl@sss.pgh.pa.us         756                 :             25 :     printf(_("Maximum data alignment:               %u\n"),
                                757                 :                :            ControlFile.maxAlign);
                                758                 :                :     /* we don't print floatFormat since can't say much useful about it */
                                759                 :             25 :     printf(_("Database block size:                  %u\n"),
                                760                 :                :            ControlFile.blcksz);
                                761                 :             25 :     printf(_("Blocks per segment of large relation: %u\n"),
                                762                 :                :            ControlFile.relseg_size);
                                763                 :             25 :     printf(_("WAL block size:                       %u\n"),
                                764                 :                :            ControlFile.xlog_blcksz);
                                765                 :             25 :     printf(_("Bytes per WAL segment:                %u\n"),
                                766                 :                :            ControlFile.xlog_seg_size);
                                767                 :             25 :     printf(_("Maximum length of identifiers:        %u\n"),
                                768                 :                :            ControlFile.nameDataLen);
                                769                 :             25 :     printf(_("Maximum columns in an index:          %u\n"),
                                770                 :                :            ControlFile.indexMaxKeys);
 6221                           771                 :             25 :     printf(_("Maximum size of a TOAST chunk:        %u\n"),
                                772                 :                :            ControlFile.toast_max_chunk_size);
 3601                           773                 :             25 :     printf(_("Size of a large-object chunk:         %u\n"),
                                774                 :                :            ControlFile.loblksize);
                                775                 :                :     /* This is no longer configurable, but users may still expect to see it: */
 7865                           776                 :             25 :     printf(_("Date/time type storage:               %s\n"),
                                777                 :                :            _("64-bit integers"));
 5837                           778         [ +  - ]:             25 :     printf(_("Float8 argument passing:              %s\n"),
                                779                 :                :            (ControlFile.float8ByVal ? _("by value") : _("by reference")));
 4002 simon@2ndQuadrant.co      780                 :             25 :     printf(_("Data page checksum version:           %u\n"),
                                781                 :                :            ControlFile.data_checksum_version);
 7911 bruce@momjian.us          782                 :             25 : }
                                783                 :                : 
                                784                 :                : 
                                785                 :                : /*
                                786                 :                :  * Print the values to be changed.
                                787                 :                :  */
                                788                 :                : static void
 3165 andres@anarazel.de        789                 :             25 : PrintNewControlValues(void)
                                790                 :                : {
                                791                 :                :     char        fname[MAXFNAMELEN];
                                792                 :                : 
                                793                 :                :     /* This will be always printed in order to keep format same. */
 3776 heikki.linnakangas@i      794                 :             25 :     printf(_("\n\nValues to be changed:\n\n"));
                                795                 :                : 
 2399 andres@anarazel.de        796                 :             25 :     XLogFileName(fname, ControlFile.checkPointCopy.ThisTimeLineID,
                                797                 :                :                  newXlogSegNo, WalSegSz);
  577 tgl@sss.pgh.pa.us         798                 :             25 :     printf(_("First log segment after reset:        %s\n"), fname);
                                799                 :                : 
 3776 heikki.linnakangas@i      800         [ +  + ]:             25 :     if (set_mxid != 0)
                                801                 :                :     {
 3776 heikki.linnakangas@i      802                 :GBC           1 :         printf(_("NextMultiXactId:                      %u\n"),
                                803                 :                :                ControlFile.checkPointCopy.nextMulti);
                                804                 :              1 :         printf(_("OldestMultiXid:                       %u\n"),
                                805                 :                :                ControlFile.checkPointCopy.oldestMulti);
                                806                 :              1 :         printf(_("OldestMulti's DB:                     %u\n"),
                                807                 :                :                ControlFile.checkPointCopy.oldestMultiDB);
                                808                 :                :     }
                                809                 :                : 
 3776 heikki.linnakangas@i      810         [ +  + ]:CBC          25 :     if (set_mxoff != -1)
                                811                 :                :     {
 3776 heikki.linnakangas@i      812                 :GBC           1 :         printf(_("NextMultiOffset:                      %u\n"),
                                813                 :                :                ControlFile.checkPointCopy.nextMultiOffset);
                                814                 :                :     }
                                815                 :                : 
 3776 heikki.linnakangas@i      816         [ +  + ]:CBC          25 :     if (set_oid != 0)
                                817                 :                :     {
 3776 heikki.linnakangas@i      818                 :GBC           1 :         printf(_("NextOID:                              %u\n"),
                                819                 :                :                ControlFile.checkPointCopy.nextOid);
                                820                 :                :     }
                                821                 :                : 
 3776 heikki.linnakangas@i      822         [ +  + ]:CBC          25 :     if (set_xid != 0)
                                823                 :                :     {
 3776 heikki.linnakangas@i      824                 :GBC           1 :         printf(_("NextXID:                              %u\n"),
                                825                 :                :                XidFromFullTransactionId(ControlFile.checkPointCopy.nextXid));
                                826                 :              1 :         printf(_("OldestXID:                            %u\n"),
                                827                 :                :                ControlFile.checkPointCopy.oldestXid);
                                828                 :              1 :         printf(_("OldestXID's DB:                       %u\n"),
                                829                 :                :                ControlFile.checkPointCopy.oldestXidDB);
                                830                 :                :     }
                                831                 :                : 
 3776 heikki.linnakangas@i      832         [ +  + ]:CBC          25 :     if (set_xid_epoch != -1)
                                833                 :                :     {
 3536 peter_e@gmx.net           834                 :GBC           1 :         printf(_("NextXID epoch:                        %u\n"),
                                835                 :                :                EpochFromFullTransactionId(ControlFile.checkPointCopy.nextXid));
                                836                 :                :     }
                                837                 :                : 
 3030 mail@joeconway.com        838         [ +  + ]:CBC          25 :     if (set_oldest_commit_ts_xid != 0)
                                839                 :                :     {
 3030 mail@joeconway.com        840                 :GBC           1 :         printf(_("oldestCommitTsXid:                    %u\n"),
                                841                 :                :                ControlFile.checkPointCopy.oldestCommitTsXid);
                                842                 :                :     }
 3030 mail@joeconway.com        843         [ -  + ]:CBC          25 :     if (set_newest_commit_ts_xid != 0)
                                844                 :                :     {
 3030 mail@joeconway.com        845                 :UBC           0 :         printf(_("newestCommitTsXid:                    %u\n"),
                                846                 :                :                ControlFile.checkPointCopy.newestCommitTsXid);
                                847                 :                :     }
                                848                 :                : 
 2212 peter_e@gmx.net           849         [ +  + ]:CBC          25 :     if (set_wal_segsize != 0)
                                850                 :                :     {
 2212 peter_e@gmx.net           851                 :GBC           1 :         printf(_("Bytes per WAL segment:                %u\n"),
                                852                 :                :                ControlFile.xlog_seg_size);
                                853                 :                :     }
 3776 heikki.linnakangas@i      854                 :CBC          25 : }
                                855                 :                : 
                                856                 :                : 
                                857                 :                : /*
                                858                 :                :  * Write out the new pg_control file.
                                859                 :                :  */
                                860                 :                : static void
 6525 bruce@momjian.us          861                 :             27 : RewriteControlFile(void)
                                862                 :                : {
                                863                 :                :     /*
                                864                 :                :      * Adjust fields as needed to force an empty XLOG starting at
                                865                 :                :      * newXlogSegNo.
                                866                 :                :      */
 2106 alvherre@alvh.no-ip.      867                 :             27 :     XLogSegNoOffsetToRecPtr(newXlogSegNo, SizeOfXLogLongPHD, WalSegSz,
                                868                 :                :                             ControlFile.checkPointCopy.redo);
 5901 tgl@sss.pgh.pa.us         869                 :             27 :     ControlFile.checkPointCopy.time = (pg_time_t) time(NULL);
                                870                 :                : 
 7911 bruce@momjian.us          871                 :             27 :     ControlFile.state = DB_SHUTDOWNED;
                                872                 :             27 :     ControlFile.checkPoint = ControlFile.checkPointCopy.redo;
 4312 heikki.linnakangas@i      873                 :             27 :     ControlFile.minRecoveryPoint = 0;
 4149                           874                 :             27 :     ControlFile.minRecoveryPointTLI = 0;
 4312                           875                 :             27 :     ControlFile.backupStartPoint = 0;
                                876                 :             27 :     ControlFile.backupEndPoint = 0;
 4624                           877                 :             27 :     ControlFile.backupEndRequired = false;
                                878                 :                : 
                                879                 :                :     /*
                                880                 :                :      * Force the defaults for max_* settings. The values don't really matter
                                881                 :                :      * as long as wal_level='minimal'; the postmaster will reset these fields
                                882                 :                :      * anyway at startup.
                                883                 :                :      */
 5100 tgl@sss.pgh.pa.us         884                 :             27 :     ControlFile.wal_level = WAL_LEVEL_MINIMAL;
 3767 fujii@postgresql.org      885                 :             27 :     ControlFile.wal_log_hints = false;
 3420 alvherre@alvh.no-ip.      886                 :             27 :     ControlFile.track_commit_timestamp = false;
 5100 heikki.linnakangas@i      887                 :             27 :     ControlFile.MaxConnections = 100;
 1888 michael@paquier.xyz       888                 :             27 :     ControlFile.max_wal_senders = 10;
 2687 rhaas@postgresql.org      889                 :             27 :     ControlFile.max_worker_processes = 8;
 5100 heikki.linnakangas@i      890                 :             27 :     ControlFile.max_prepared_xacts = 0;
                                891                 :             27 :     ControlFile.max_locks_per_xact = 64;
                                892                 :                : 
                                893                 :                :     /* The control file gets flushed here. */
 1840 peter@eisentraut.org      894                 :             27 :     update_controlfile(".", &ControlFile, true);
 7911 bruce@momjian.us          895                 :             27 : }
                                896                 :                : 
                                897                 :                : 
                                898                 :                : /*
                                899                 :                :  * Scan existing XLOG files and determine the highest existing WAL address
                                900                 :                :  *
                                901                 :                :  * On entry, ControlFile.checkPointCopy.redo and ControlFile.xlog_seg_size
                                902                 :                :  * are assumed valid (note that we allow the old xlog seg size to differ
                                903                 :                :  * from what we're using).  On exit, newXlogSegNo is set to suitable
                                904                 :                :  * value for the beginning of replacement WAL (in our seg size).
                                905                 :                :  */
                                906                 :                : static void
 6337 tgl@sss.pgh.pa.us         907                 :             53 : FindEndOfXLOG(void)
                                908                 :                : {
                                909                 :                :     DIR        *xldir;
                                910                 :                :     struct dirent *xlde;
                                911                 :                :     uint64      xlogbytepos;
                                912                 :                : 
                                913                 :                :     /*
                                914                 :                :      * Initialize the max() computation using the last checkpoint address from
                                915                 :                :      * old pg_control.  Note that for the moment we are working with segment
                                916                 :                :      * numbering according to the old xlog seg size.
                                917                 :                :      */
  557 michael@paquier.xyz       918                 :             53 :     XLByteToSeg(ControlFile.checkPointCopy.redo, newXlogSegNo,
                                919                 :                :                 ControlFile.xlog_seg_size);
                                920                 :                : 
                                921                 :                :     /*
                                922                 :                :      * Scan the pg_wal directory to find existing WAL segment files. We assume
                                923                 :                :      * any present have been used; in most scenarios this should be
                                924                 :                :      * conservative, because of xlog.c's attempts to pre-create files.
                                925                 :                :      */
 6337 tgl@sss.pgh.pa.us         926                 :             53 :     xldir = opendir(XLOGDIR);
                                927         [ -  + ]:             53 :     if (xldir == NULL)
  737 tgl@sss.pgh.pa.us         928                 :UBC           0 :         pg_fatal("could not open directory \"%s\": %m", XLOGDIR);
                                929                 :                : 
 3677 bruce@momjian.us          930         [ +  + ]:CBC         519 :     while (errno = 0, (xlde = readdir(xldir)) != NULL)
                                931                 :                :     {
 3208 fujii@postgresql.org      932   [ +  +  -  + ]:            679 :         if (IsXLogFileName(xlde->d_name) ||
                                933                 :            213 :             IsPartialXLogFileName(xlde->d_name))
                                934                 :                :         {
                                935                 :                :             TimeLineID  tli;
                                936                 :                :             XLogSegNo   segno;
                                937                 :                : 
                                938                 :                :             /* Use the segment size from the control file */
  557 michael@paquier.xyz       939                 :            253 :             XLogFromFileName(xlde->d_name, &tli, &segno,
                                940                 :            253 :                              ControlFile.xlog_seg_size);
                                941                 :                : 
                                942                 :                :             /*
                                943                 :                :              * Note: we take the max of all files found, regardless of their
                                944                 :                :              * timelines.  Another possibility would be to ignore files of
                                945                 :                :              * timelines other than the target TLI, but this seems safer.
                                946                 :                :              * Better too large a result than too small...
                                947                 :                :              */
 4312 heikki.linnakangas@i      948         [ +  + ]:            253 :             if (segno > newXlogSegNo)
                                949                 :             86 :                 newXlogSegNo = segno;
                                950                 :                :         }
                                951                 :                :     }
                                952                 :                : 
 6337 tgl@sss.pgh.pa.us         953         [ -  + ]:             53 :     if (errno)
  737 tgl@sss.pgh.pa.us         954                 :UBC           0 :         pg_fatal("could not read directory \"%s\": %m", XLOGDIR);
                                955                 :                : 
 3677 bruce@momjian.us          956         [ -  + ]:CBC          53 :     if (closedir(xldir))
  737 tgl@sss.pgh.pa.us         957                 :UBC           0 :         pg_fatal("could not close directory \"%s\": %m", XLOGDIR);
                                958                 :                : 
                                959                 :                :     /*
                                960                 :                :      * Finally, convert to new xlog seg size, and advance by one to ensure we
                                961                 :                :      * are in virgin territory.
                                962                 :                :      */
 4312 heikki.linnakangas@i      963                 :CBC          53 :     xlogbytepos = newXlogSegNo * ControlFile.xlog_seg_size;
 2212 peter_e@gmx.net           964                 :             53 :     newXlogSegNo = (xlogbytepos + ControlFile.xlog_seg_size - 1) / WalSegSz;
 4312 heikki.linnakangas@i      965                 :             53 :     newXlogSegNo++;
 6337 tgl@sss.pgh.pa.us         966                 :             53 : }
                                967                 :                : 
                                968                 :                : 
                                969                 :                : /*
                                970                 :                :  * Remove existing XLOG files
                                971                 :                :  */
                                972                 :                : static void
 7911 bruce@momjian.us          973                 :             27 : KillExistingXLOG(void)
                                974                 :                : {
                                975                 :                :     DIR        *xldir;
                                976                 :                :     struct dirent *xlde;
                                977                 :                :     char        path[MAXPGPATH + sizeof(XLOGDIR)];
                                978                 :                : 
 6859 tgl@sss.pgh.pa.us         979                 :             27 :     xldir = opendir(XLOGDIR);
 7911 bruce@momjian.us          980         [ -  + ]:             27 :     if (xldir == NULL)
  737 tgl@sss.pgh.pa.us         981                 :UBC           0 :         pg_fatal("could not open directory \"%s\": %m", XLOGDIR);
                                982                 :                : 
 3677 bruce@momjian.us          983         [ +  + ]:CBC         219 :     while (errno = 0, (xlde = readdir(xldir)) != NULL)
                                984                 :                :     {
 3208 fujii@postgresql.org      985   [ +  +  -  + ]:            301 :         if (IsXLogFileName(xlde->d_name) ||
                                986                 :            109 :             IsPartialXLogFileName(xlde->d_name))
                                987                 :                :         {
 2560 peter_e@gmx.net           988                 :             83 :             snprintf(path, sizeof(path), "%s/%s", XLOGDIR, xlde->d_name);
 7911 bruce@momjian.us          989         [ -  + ]:             83 :             if (unlink(path) < 0)
  737 tgl@sss.pgh.pa.us         990                 :UBC           0 :                 pg_fatal("could not delete file \"%s\": %m", path);
                                991                 :                :         }
                                992                 :                :     }
                                993                 :                : 
 7911 bruce@momjian.us          994         [ -  + ]:CBC          27 :     if (errno)
  737 tgl@sss.pgh.pa.us         995                 :UBC           0 :         pg_fatal("could not read directory \"%s\": %m", XLOGDIR);
                                996                 :                : 
 3677 bruce@momjian.us          997         [ -  + ]:CBC          27 :     if (closedir(xldir))
  737 tgl@sss.pgh.pa.us         998                 :UBC           0 :         pg_fatal("could not close directory \"%s\": %m", XLOGDIR);
 7911 bruce@momjian.us          999                 :CBC          27 : }
                               1000                 :                : 
                               1001                 :                : 
                               1002                 :                : /*
                               1003                 :                :  * Remove existing archive status files
                               1004                 :                :  */
                               1005                 :                : static void
 5460 tgl@sss.pgh.pa.us        1006                 :             27 : KillExistingArchiveStatus(void)
                               1007                 :                : {
                               1008                 :                : #define ARCHSTATDIR XLOGDIR "/archive_status"
                               1009                 :                : 
                               1010                 :                :     DIR        *xldir;
                               1011                 :                :     struct dirent *xlde;
                               1012                 :                :     char        path[MAXPGPATH + sizeof(ARCHSTATDIR)];
                               1013                 :                : 
                               1014                 :             27 :     xldir = opendir(ARCHSTATDIR);
                               1015         [ -  + ]:             27 :     if (xldir == NULL)
  737 tgl@sss.pgh.pa.us        1016                 :UBC           0 :         pg_fatal("could not open directory \"%s\": %m", ARCHSTATDIR);
                               1017                 :                : 
 3677 bruce@momjian.us         1018         [ +  + ]:CBC          81 :     while (errno = 0, (xlde = readdir(xldir)) != NULL)
                               1019                 :                :     {
 3209 fujii@postgresql.org     1020         [ -  + ]:             54 :         if (strspn(xlde->d_name, "0123456789ABCDEF") == XLOG_FNAME_LEN &&
 3209 fujii@postgresql.org     1021         [ #  # ]:UBC           0 :             (strcmp(xlde->d_name + XLOG_FNAME_LEN, ".ready") == 0 ||
 3208                          1022         [ #  # ]:              0 :              strcmp(xlde->d_name + XLOG_FNAME_LEN, ".done") == 0 ||
                               1023         [ #  # ]:              0 :              strcmp(xlde->d_name + XLOG_FNAME_LEN, ".partial.ready") == 0 ||
                               1024         [ #  # ]:              0 :              strcmp(xlde->d_name + XLOG_FNAME_LEN, ".partial.done") == 0))
                               1025                 :                :         {
 2560 peter_e@gmx.net          1026                 :              0 :             snprintf(path, sizeof(path), "%s/%s", ARCHSTATDIR, xlde->d_name);
 5460 tgl@sss.pgh.pa.us        1027         [ #  # ]:              0 :             if (unlink(path) < 0)
  737                          1028                 :              0 :                 pg_fatal("could not delete file \"%s\": %m", path);
                               1029                 :                :         }
                               1030                 :                :     }
                               1031                 :                : 
 5460 tgl@sss.pgh.pa.us        1032         [ -  + ]:CBC          27 :     if (errno)
  737 tgl@sss.pgh.pa.us        1033                 :UBC           0 :         pg_fatal("could not read directory \"%s\": %m", ARCHSTATDIR);
                               1034                 :                : 
 3677 bruce@momjian.us         1035         [ -  + ]:CBC          27 :     if (closedir(xldir))
  737 tgl@sss.pgh.pa.us        1036                 :UBC           0 :         pg_fatal("could not close directory \"%s\": %m", ARCHSTATDIR);
 5460 tgl@sss.pgh.pa.us        1037                 :CBC          27 : }
                               1038                 :                : 
                               1039                 :                : /*
                               1040                 :                :  * Remove existing WAL summary files
                               1041                 :                :  */
                               1042                 :                : static void
  116 rhaas@postgresql.org     1043                 :GNC          27 : KillExistingWALSummaries(void)
                               1044                 :                : {
                               1045                 :                : #define WALSUMMARYDIR XLOGDIR   "/summaries"
                               1046                 :                : #define WALSUMMARY_NHEXCHARS    40
                               1047                 :                : 
                               1048                 :                :     DIR        *xldir;
                               1049                 :                :     struct dirent *xlde;
                               1050                 :                :     char        path[MAXPGPATH + sizeof(WALSUMMARYDIR)];
                               1051                 :                : 
                               1052                 :             27 :     xldir = opendir(WALSUMMARYDIR);
                               1053         [ -  + ]:             27 :     if (xldir == NULL)
  116 rhaas@postgresql.org     1054                 :UNC           0 :         pg_fatal("could not open directory \"%s\": %m", WALSUMMARYDIR);
                               1055                 :                : 
  116 rhaas@postgresql.org     1056         [ +  + ]:GNC          81 :     while (errno = 0, (xlde = readdir(xldir)) != NULL)
                               1057                 :                :     {
                               1058         [ -  + ]:             54 :         if (strspn(xlde->d_name, "0123456789ABCDEF") == WALSUMMARY_NHEXCHARS &&
  116 rhaas@postgresql.org     1059         [ #  # ]:UNC           0 :             strcmp(xlde->d_name + WALSUMMARY_NHEXCHARS, ".summary") == 0)
                               1060                 :                :         {
                               1061                 :              0 :             snprintf(path, sizeof(path), "%s/%s", WALSUMMARYDIR, xlde->d_name);
                               1062         [ #  # ]:              0 :             if (unlink(path) < 0)
                               1063                 :              0 :                 pg_fatal("could not delete file \"%s\": %m", path);
                               1064                 :                :         }
                               1065                 :                :     }
                               1066                 :                : 
  116 rhaas@postgresql.org     1067         [ -  + ]:GNC          27 :     if (errno)
  116 rhaas@postgresql.org     1068                 :UNC           0 :         pg_fatal("could not read directory \"%s\": %m", WALSUMMARYDIR);
                               1069                 :                : 
  116 rhaas@postgresql.org     1070         [ -  + ]:GNC          27 :     if (closedir(xldir))
  116 rhaas@postgresql.org     1071                 :UNC           0 :         pg_fatal("could not close directory \"%s\": %m", ARCHSTATDIR);
  116 rhaas@postgresql.org     1072                 :GNC          27 : }
                               1073                 :                : 
                               1074                 :                : /*
                               1075                 :                :  * Write an empty XLOG file, containing only the checkpoint record
                               1076                 :                :  * already set up in ControlFile.
                               1077                 :                :  */
                               1078                 :                : static void
 7911 bruce@momjian.us         1079                 :CBC          27 : WriteEmptyXLOG(void)
                               1080                 :                : {
                               1081                 :                :     PGAlignedXLogBlock buffer;
                               1082                 :                :     XLogPageHeader page;
                               1083                 :                :     XLogLongPageHeader longpage;
                               1084                 :                :     XLogRecord *record;
                               1085                 :                :     pg_crc32c   crc;
                               1086                 :                :     char        path[MAXPGPATH];
                               1087                 :                :     int         fd;
                               1088                 :                :     int         nbytes;
                               1089                 :                :     char       *recptr;
                               1090                 :                : 
 2052 tgl@sss.pgh.pa.us        1091                 :             27 :     memset(buffer.data, 0, XLOG_BLCKSZ);
                               1092                 :                : 
                               1093                 :                :     /* Set up the XLOG page header */
                               1094                 :             27 :     page = (XLogPageHeader) buffer.data;
 7911 bruce@momjian.us         1095                 :             27 :     page->xlp_magic = XLOG_PAGE_MAGIC;
 7207 tgl@sss.pgh.pa.us        1096                 :             27 :     page->xlp_info = XLP_LONG_HEADER;
                               1097                 :             27 :     page->xlp_tli = ControlFile.checkPointCopy.ThisTimeLineID;
 4312 heikki.linnakangas@i     1098                 :             27 :     page->xlp_pageaddr = ControlFile.checkPointCopy.redo - SizeOfXLogLongPHD;
 7207 tgl@sss.pgh.pa.us        1099                 :             27 :     longpage = (XLogLongPageHeader) page;
                               1100                 :             27 :     longpage->xlp_sysid = ControlFile.system_identifier;
 2399 andres@anarazel.de       1101                 :             27 :     longpage->xlp_seg_size = WalSegSz;
 6584 tgl@sss.pgh.pa.us        1102                 :             27 :     longpage->xlp_xlog_blcksz = XLOG_BLCKSZ;
                               1103                 :                : 
                               1104                 :                :     /* Insert the initial checkpoint record */
 3433 heikki.linnakangas@i     1105                 :             27 :     recptr = (char *) page + SizeOfXLogLongPHD;
                               1106                 :             27 :     record = (XLogRecord *) recptr;
 4312                          1107                 :             27 :     record->xl_prev = 0;
 7368 tgl@sss.pgh.pa.us        1108                 :             27 :     record->xl_xid = InvalidTransactionId;
 3433 heikki.linnakangas@i     1109                 :             27 :     record->xl_tot_len = SizeOfXLogRecord + SizeOfXLogRecordDataHeaderShort + sizeof(CheckPoint);
 7911 bruce@momjian.us         1110                 :             27 :     record->xl_info = XLOG_CHECKPOINT_SHUTDOWN;
                               1111                 :             27 :     record->xl_rmid = RM_XLOG_ID;
                               1112                 :                : 
 3433 heikki.linnakangas@i     1113                 :             27 :     recptr += SizeOfXLogRecord;
 2574 tgl@sss.pgh.pa.us        1114                 :             27 :     *(recptr++) = (char) XLR_BLOCK_ID_DATA_SHORT;
 3433 heikki.linnakangas@i     1115                 :             27 :     *(recptr++) = sizeof(CheckPoint);
                               1116                 :             27 :     memcpy(recptr, &ControlFile.checkPointCopy,
                               1117                 :                :            sizeof(CheckPoint));
                               1118                 :                : 
 3449                          1119                 :             27 :     INIT_CRC32C(crc);
 3433                          1120                 :             27 :     COMP_CRC32C(crc, ((char *) record) + SizeOfXLogRecord, record->xl_tot_len - SizeOfXLogRecord);
 3449                          1121                 :             27 :     COMP_CRC32C(crc, (char *) record, offsetof(XLogRecord, xl_crc));
                               1122                 :             27 :     FIN_CRC32C(crc);
 7911 bruce@momjian.us         1123                 :             27 :     record->xl_crc = crc;
                               1124                 :                : 
                               1125                 :                :     /* Write the first page */
 2399 andres@anarazel.de       1126                 :             27 :     XLogFilePath(path, ControlFile.checkPointCopy.ThisTimeLineID,
                               1127                 :                :                  newXlogSegNo, WalSegSz);
                               1128                 :                : 
 7911 bruce@momjian.us         1129                 :             27 :     unlink(path);
                               1130                 :                : 
                               1131                 :             27 :     fd = open(path, O_RDWR | O_CREAT | O_EXCL | PG_BINARY,
                               1132                 :                :               pg_file_create_mode);
                               1133         [ -  + ]:             27 :     if (fd < 0)
  737 tgl@sss.pgh.pa.us        1134                 :UBC           0 :         pg_fatal("could not open file \"%s\": %m", path);
                               1135                 :                : 
 7911 bruce@momjian.us         1136                 :CBC          27 :     errno = 0;
 2052 tgl@sss.pgh.pa.us        1137         [ -  + ]:             27 :     if (write(fd, buffer.data, XLOG_BLCKSZ) != XLOG_BLCKSZ)
                               1138                 :                :     {
                               1139                 :                :         /* if write didn't set errno, assume problem is no disk space */
 7911 bruce@momjian.us         1140         [ #  # ]:UBC           0 :         if (errno == 0)
                               1141                 :              0 :             errno = ENOSPC;
  737 tgl@sss.pgh.pa.us        1142                 :              0 :         pg_fatal("could not write file \"%s\": %m", path);
                               1143                 :                :     }
                               1144                 :                : 
                               1145                 :                :     /* Fill the rest of the file with zeroes */
 2052 tgl@sss.pgh.pa.us        1146                 :CBC          27 :     memset(buffer.data, 0, XLOG_BLCKSZ);
 2399 andres@anarazel.de       1147         [ +  + ]:          39936 :     for (nbytes = XLOG_BLCKSZ; nbytes < WalSegSz; nbytes += XLOG_BLCKSZ)
                               1148                 :                :     {
 7911 bruce@momjian.us         1149                 :          39909 :         errno = 0;
 2052 tgl@sss.pgh.pa.us        1150         [ -  + ]:          39909 :         if (write(fd, buffer.data, XLOG_BLCKSZ) != XLOG_BLCKSZ)
                               1151                 :                :         {
 7911 bruce@momjian.us         1152         [ #  # ]:UBC           0 :             if (errno == 0)
                               1153                 :              0 :                 errno = ENOSPC;
  737 tgl@sss.pgh.pa.us        1154                 :              0 :             pg_fatal("could not write file \"%s\": %m", path);
                               1155                 :                :         }
                               1156                 :                :     }
                               1157                 :                : 
 7911 bruce@momjian.us         1158         [ -  + ]:CBC          27 :     if (fsync(fd) != 0)
  737 tgl@sss.pgh.pa.us        1159                 :UBC           0 :         pg_fatal("fsync error: %m");
                               1160                 :                : 
 7911 bruce@momjian.us         1161                 :CBC          27 :     close(fd);
                               1162                 :             27 : }
                               1163                 :                : 
                               1164                 :                : 
                               1165                 :                : static void
                               1166                 :              1 : usage(void)
                               1167                 :                : {
 2529 peter_e@gmx.net          1168                 :              1 :     printf(_("%s resets the PostgreSQL write-ahead log.\n\n"), progname);
  199 peter@eisentraut.org     1169                 :GNC           1 :     printf(_("Usage:\n"));
                               1170                 :              1 :     printf(_("  %s [OPTION]... DATADIR\n"), progname);
                               1171                 :                : 
                               1172                 :              1 :     printf(_("\nOptions:\n"));
                               1173                 :              1 :     printf(_(" [-D, --pgdata=]DATADIR  data directory\n"));
                               1174                 :              1 :     printf(_("  -f, --force            force update to be done even after unclean shutdown or\n"
                               1175                 :                :              "                         if pg_control values had to be guessed\n"));
                               1176                 :              1 :     printf(_("  -n, --dry-run          no update, just show what would be done\n"));
                               1177                 :              1 :     printf(_("  -V, --version          output version information, then exit\n"));
                               1178                 :              1 :     printf(_("  -?, --help             show this help, then exit\n"));
                               1179                 :                : 
                               1180                 :              1 :     printf(_("\nOptions to override control file values:\n"));
 2213 peter_e@gmx.net          1181                 :CBC           1 :     printf(_("  -c, --commit-timestamp-ids=XID,XID\n"
                               1182                 :                :              "                                   set oldest and newest transactions bearing\n"
                               1183                 :                :              "                                   commit timestamp (zero means no change)\n"));
  993 bruce@momjian.us         1184                 :              1 :     printf(_("  -e, --epoch=XIDEPOCH             set next transaction ID epoch\n"));
                               1185                 :              1 :     printf(_("  -l, --next-wal-file=WALFILE      set minimum starting location for new WAL\n"));
                               1186                 :              1 :     printf(_("  -m, --multixact-ids=MXID,MXID    set next and oldest multitransaction ID\n"));
                               1187                 :              1 :     printf(_("  -o, --next-oid=OID               set next OID\n"));
                               1188                 :              1 :     printf(_("  -O, --multixact-offset=OFFSET    set next multitransaction offset\n"));
                               1189                 :              1 :     printf(_("  -u, --oldest-transaction-id=XID  set oldest transaction ID\n"));
                               1190                 :              1 :     printf(_("  -x, --next-transaction-id=XID    set next transaction ID\n"));
                               1191                 :              1 :     printf(_("      --wal-segsize=SIZE           size of WAL segments, in megabytes\n"));
                               1192                 :                : 
 1507 peter@eisentraut.org     1193                 :              1 :     printf(_("\nReport bugs to <%s>.\n"), PACKAGE_BUGREPORT);
                               1194                 :              1 :     printf(_("%s home page: <%s>\n"), PACKAGE_NAME, PACKAGE_URL);
 7911 bruce@momjian.us         1195                 :              1 : }
        

Generated by: LCOV version 2.1-beta2-3-g6141622