1 /*-------------------------------------------------------------------------
4 * A utility to "zero out" the xlog when it's corrupt beyond recovery.
5 * Can also rebuild pg_control if needed.
7 * The theory of operation is fairly simple:
8 * 1. Read the existing pg_control (which will include the last
9 * checkpoint record). If it is an old format then update to
11 * 2. If pg_control is corrupt, attempt to intuit reasonable values,
12 * by scanning the old xlog if necessary.
13 * 3. Modify pg_control to reflect a "shutdown" state with a checkpoint
14 * record at the start of xlog.
15 * 4. Flush the existing xlog files and write a new segment with
16 * just a checkpoint record in it. The new segment is positioned
17 * just past the end of the old xlog, so that existing LSNs in
18 * data pages will appear to be "in the past".
19 * This is all pretty straightforward except for the intuition part of
23 * Portions Copyright (c) 1996-2009, PostgreSQL Global Development Group
24 * Portions Copyright (c) 1994, Regents of the University of California
28 *-------------------------------------------------------------------------
32 * We have to use postgres.h not postgres_fe.h here, because there's so much
33 * backend-only stuff in the XLOG include files we need. But we need a
34 * frontend-ish environment otherwise. Hence this ugly hack.
51 #include "access/transam.h"
52 #include "access/tuptoaster.h"
53 #include "access/multixact.h"
54 #include "access/xlog_internal.h"
55 #include "catalog/catversion.h"
56 #include "catalog/pg_control.h"
62 static ControlFileData ControlFile
; /* pg_control values */
63 static uint32 newXlogId
,
64 newXlogSeg
; /* ID/Segment of new XLOG segment */
65 static bool guessed
= false; /* T if we had to guess at any values */
66 static const char *progname
;
68 static bool ReadControlFile(void);
69 static void GuessControlValues(void);
70 static void PrintControlValues(bool guessed
);
71 static void RewriteControlFile(void);
72 static void FindEndOfXLOG(void);
73 static void KillExistingXLOG(void);
74 static void KillExistingArchiveStatus(void);
75 static void WriteEmptyXLOG(void);
76 static void usage(void);
80 main(int argc
, char *argv
[])
84 bool noupdate
= false;
85 uint32 set_xid_epoch
= (uint32
) -1;
86 TransactionId set_xid
= 0;
88 MultiXactId set_mxid
= 0;
89 MultiXactOffset set_mxoff
= (MultiXactOffset
) -1;
90 uint32 minXlogTli
= 0,
100 set_pglocale_pgservice(argv
[0], PG_TEXTDOMAIN("pg_resetxlog"));
102 progname
= get_progname(argv
[0]);
106 if (strcmp(argv
[1], "--help") == 0 || strcmp(argv
[1], "-?") == 0)
111 if (strcmp(argv
[1], "--version") == 0 || strcmp(argv
[1], "-V") == 0)
113 puts("pg_resetxlog (PostgreSQL) " PG_VERSION
);
119 while ((c
= getopt(argc
, argv
, "fl:m:no:O:x:e:")) != -1)
132 set_xid_epoch
= strtoul(optarg
, &endptr
, 0);
133 if (endptr
== optarg
|| *endptr
!= '\0')
135 fprintf(stderr
, _("%s: invalid argument for option -e\n"), progname
);
136 fprintf(stderr
, _("Try \"%s --help\" for more information.\n"), progname
);
139 if (set_xid_epoch
== -1)
141 fprintf(stderr
, _("%s: transaction ID epoch (-e) must not be -1\n"), progname
);
147 set_xid
= strtoul(optarg
, &endptr
, 0);
148 if (endptr
== optarg
|| *endptr
!= '\0')
150 fprintf(stderr
, _("%s: invalid argument for option -x\n"), progname
);
151 fprintf(stderr
, _("Try \"%s --help\" for more information.\n"), progname
);
156 fprintf(stderr
, _("%s: transaction ID (-x) must not be 0\n"), progname
);
162 set_oid
= strtoul(optarg
, &endptr
, 0);
163 if (endptr
== optarg
|| *endptr
!= '\0')
165 fprintf(stderr
, _("%s: invalid argument for option -o\n"), progname
);
166 fprintf(stderr
, _("Try \"%s --help\" for more information.\n"), progname
);
171 fprintf(stderr
, _("%s: OID (-o) must not be 0\n"), progname
);
177 set_mxid
= strtoul(optarg
, &endptr
, 0);
178 if (endptr
== optarg
|| *endptr
!= '\0')
180 fprintf(stderr
, _("%s: invalid argument for option -m\n"), progname
);
181 fprintf(stderr
, _("Try \"%s --help\" for more information.\n"), progname
);
186 fprintf(stderr
, _("%s: multitransaction ID (-m) must not be 0\n"), progname
);
192 set_mxoff
= strtoul(optarg
, &endptr
, 0);
193 if (endptr
== optarg
|| *endptr
!= '\0')
195 fprintf(stderr
, _("%s: invalid argument for option -O\n"), progname
);
196 fprintf(stderr
, _("Try \"%s --help\" for more information.\n"), progname
);
201 fprintf(stderr
, _("%s: multitransaction offset (-O) must not be -1\n"), progname
);
207 minXlogTli
= strtoul(optarg
, &endptr
, 0);
208 if (endptr
== optarg
|| *endptr
!= ',')
210 fprintf(stderr
, _("%s: invalid argument for option -l\n"), progname
);
211 fprintf(stderr
, _("Try \"%s --help\" for more information.\n"), progname
);
214 minXlogId
= strtoul(endptr
+ 1, &endptr2
, 0);
215 if (endptr2
== endptr
+ 1 || *endptr2
!= ',')
217 fprintf(stderr
, _("%s: invalid argument for option -l\n"), progname
);
218 fprintf(stderr
, _("Try \"%s --help\" for more information.\n"), progname
);
221 minXlogSeg
= strtoul(endptr2
+ 1, &endptr3
, 0);
222 if (endptr3
== endptr2
+ 1 || *endptr3
!= '\0')
224 fprintf(stderr
, _("%s: invalid argument for option -l\n"), progname
);
225 fprintf(stderr
, _("Try \"%s --help\" for more information.\n"), progname
);
231 fprintf(stderr
, _("Try \"%s --help\" for more information.\n"), progname
);
238 fprintf(stderr
, _("%s: no data directory specified\n"), progname
);
239 fprintf(stderr
, _("Try \"%s --help\" for more information.\n"), progname
);
244 * Don't allow pg_resetxlog to be run as root, to avoid overwriting the
245 * ownership of files in the data directory. We need only check for root
246 * -- any other user won't have sufficient permissions to modify files in
247 * the data directory.
252 fprintf(stderr
, _("%s: cannot be executed by \"root\"\n"),
254 fprintf(stderr
, _("You must run %s as the PostgreSQL superuser.\n"),
260 DataDir
= argv
[optind
];
262 if (chdir(DataDir
) < 0)
264 fprintf(stderr
, _("%s: could not change directory to \"%s\": %s\n"),
265 progname
, DataDir
, strerror(errno
));
270 * Check for a postmaster lock file --- if there is one, refuse to
271 * proceed, on grounds we might be interfering with a live installation.
273 snprintf(path
, MAXPGPATH
, "%s/postmaster.pid", DataDir
);
275 if ((fd
= open(path
, O_RDONLY
, 0)) < 0)
279 fprintf(stderr
, _("%s: could not open file \"%s\" for reading: %s\n"), progname
, path
, strerror(errno
));
285 fprintf(stderr
, _("%s: lock file \"%s\" exists\n"
286 "Is a server running? If not, delete the lock file and try again.\n"),
292 * Attempt to read the existing pg_control file
294 if (!ReadControlFile())
295 GuessControlValues();
298 * Also look at existing segment files to set up newXlogId/newXlogSeg
303 * Adjust fields if required by switches. (Do this now so that printout,
304 * if any, includes these values.)
306 if (set_xid_epoch
!= -1)
307 ControlFile
.checkPointCopy
.nextXidEpoch
= set_xid_epoch
;
310 ControlFile
.checkPointCopy
.nextXid
= set_xid
;
313 ControlFile
.checkPointCopy
.nextOid
= set_oid
;
316 ControlFile
.checkPointCopy
.nextMulti
= set_mxid
;
319 ControlFile
.checkPointCopy
.nextMultiOffset
= set_mxoff
;
321 if (minXlogTli
> ControlFile
.checkPointCopy
.ThisTimeLineID
)
322 ControlFile
.checkPointCopy
.ThisTimeLineID
= minXlogTli
;
324 if (minXlogId
> newXlogId
||
325 (minXlogId
== newXlogId
&&
326 minXlogSeg
> newXlogSeg
))
328 newXlogId
= minXlogId
;
329 newXlogSeg
= minXlogSeg
;
333 * If we had to guess anything, and -f was not given, just print the
334 * guessed values and exit. Also print if -n is given.
336 if ((guessed
&& !force
) || noupdate
)
338 PrintControlValues(guessed
);
341 printf(_("\nIf these values seem acceptable, use -f to force reset.\n"));
349 * Don't reset from a dirty pg_control without -f, either.
351 if (ControlFile
.state
!= DB_SHUTDOWNED
&& !force
)
353 printf(_("The database server was not shut down cleanly.\n"
354 "Resetting the transaction log might cause data to be lost.\n"
355 "If you want to proceed anyway, use -f to force reset.\n"));
360 * Else, do the dirty deed.
362 RewriteControlFile();
364 KillExistingArchiveStatus();
367 printf(_("Transaction log reset\n"));
373 * Try to read the existing pg_control file.
375 * This routine is also responsible for updating old pg_control versions
376 * to the current format. (Currently we don't do anything of the sort.)
379 ReadControlFile(void)
386 if ((fd
= open(XLOG_CONTROL_FILE
, O_RDONLY
| PG_BINARY
, 0)) < 0)
389 * If pg_control is not there at all, or we can't read it, the odds
390 * are we've been handed a bad DataDir path, so give up. User can do
391 * "touch pg_control" to force us to proceed.
393 fprintf(stderr
, _("%s: could not open file \"%s\" for reading: %s\n"),
394 progname
, XLOG_CONTROL_FILE
, strerror(errno
));
396 fprintf(stderr
, _("If you are sure the data directory path is correct, execute\n"
403 /* Use malloc to ensure we have a maxaligned buffer */
404 buffer
= (char *) malloc(PG_CONTROL_SIZE
);
406 len
= read(fd
, buffer
, PG_CONTROL_SIZE
);
409 fprintf(stderr
, _("%s: could not read file \"%s\": %s\n"),
410 progname
, XLOG_CONTROL_FILE
, strerror(errno
));
415 if (len
>= sizeof(ControlFileData
) &&
416 ((ControlFileData
*) buffer
)->pg_control_version
== PG_CONTROL_VERSION
)
422 offsetof(ControlFileData
, crc
));
425 if (EQ_CRC32(crc
, ((ControlFileData
*) buffer
)->crc
))
428 memcpy(&ControlFile
, buffer
, sizeof(ControlFile
));
432 fprintf(stderr
, _("%s: pg_control exists but has invalid CRC; proceed with caution\n"),
434 /* We will use the data anyway, but treat it as guessed. */
435 memcpy(&ControlFile
, buffer
, sizeof(ControlFile
));
440 /* Looks like it's a mess. */
441 fprintf(stderr
, _("%s: pg_control exists but is broken or unknown version; ignoring it\n"),
448 * Guess at pg_control values when we can't read the old ones.
451 GuessControlValues(void)
453 uint64 sysidentifier
;
457 * Set up a completely default set of pg_control values.
460 memset(&ControlFile
, 0, sizeof(ControlFile
));
462 ControlFile
.pg_control_version
= PG_CONTROL_VERSION
;
463 ControlFile
.catalog_version_no
= CATALOG_VERSION_NO
;
466 * Create a new unique installation identifier, since we can no longer use
467 * any old XLOG records. See notes in xlog.c about the algorithm.
469 gettimeofday(&tv
, NULL
);
470 sysidentifier
= ((uint64
) tv
.tv_sec
) << 32;
471 sysidentifier
|= (uint32
) (tv
.tv_sec
| tv
.tv_usec
);
473 ControlFile
.system_identifier
= sysidentifier
;
475 ControlFile
.checkPointCopy
.redo
.xlogid
= 0;
476 ControlFile
.checkPointCopy
.redo
.xrecoff
= SizeOfXLogLongPHD
;
477 ControlFile
.checkPointCopy
.ThisTimeLineID
= 1;
478 ControlFile
.checkPointCopy
.nextXidEpoch
= 0;
479 ControlFile
.checkPointCopy
.nextXid
= (TransactionId
) 514; /* XXX */
480 ControlFile
.checkPointCopy
.nextOid
= FirstBootstrapObjectId
;
481 ControlFile
.checkPointCopy
.nextMulti
= FirstMultiXactId
;
482 ControlFile
.checkPointCopy
.nextMultiOffset
= 0;
483 ControlFile
.checkPointCopy
.time
= (pg_time_t
) time(NULL
);
485 ControlFile
.state
= DB_SHUTDOWNED
;
486 ControlFile
.time
= (pg_time_t
) time(NULL
);
487 ControlFile
.checkPoint
= ControlFile
.checkPointCopy
.redo
;
489 ControlFile
.maxAlign
= MAXIMUM_ALIGNOF
;
490 ControlFile
.floatFormat
= FLOATFORMAT_VALUE
;
491 ControlFile
.blcksz
= BLCKSZ
;
492 ControlFile
.relseg_size
= RELSEG_SIZE
;
493 ControlFile
.xlog_blcksz
= XLOG_BLCKSZ
;
494 ControlFile
.xlog_seg_size
= XLOG_SEG_SIZE
;
495 ControlFile
.nameDataLen
= NAMEDATALEN
;
496 ControlFile
.indexMaxKeys
= INDEX_MAX_KEYS
;
497 ControlFile
.toast_max_chunk_size
= TOAST_MAX_CHUNK_SIZE
;
498 #ifdef HAVE_INT64_TIMESTAMP
499 ControlFile
.enableIntTimes
= true;
501 ControlFile
.enableIntTimes
= false;
503 ControlFile
.float4ByVal
= FLOAT4PASSBYVAL
;
504 ControlFile
.float8ByVal
= FLOAT8PASSBYVAL
;
507 * XXX eventually, should try to grovel through old XLOG to develop more
508 * accurate values for TimeLineID, nextXID, etc.
514 * Print the guessed pg_control values when we had to guess.
516 * NB: this display should be just those fields that will not be
517 * reset by RewriteControlFile().
520 PrintControlValues(bool guessed
)
522 char sysident_str
[32];
525 printf(_("Guessed pg_control values:\n\n"));
527 printf(_("pg_control values:\n\n"));
530 * Format system_identifier separately to keep platform-dependent format
531 * code out of the translatable message string.
533 snprintf(sysident_str
, sizeof(sysident_str
), UINT64_FORMAT
,
534 ControlFile
.system_identifier
);
536 printf(_("First log file ID after reset: %u\n"),
538 printf(_("First log file segment after reset: %u\n"),
540 printf(_("pg_control version number: %u\n"),
541 ControlFile
.pg_control_version
);
542 printf(_("Catalog version number: %u\n"),
543 ControlFile
.catalog_version_no
);
544 printf(_("Database system identifier: %s\n"),
546 printf(_("Latest checkpoint's TimeLineID: %u\n"),
547 ControlFile
.checkPointCopy
.ThisTimeLineID
);
548 printf(_("Latest checkpoint's NextXID: %u/%u\n"),
549 ControlFile
.checkPointCopy
.nextXidEpoch
,
550 ControlFile
.checkPointCopy
.nextXid
);
551 printf(_("Latest checkpoint's NextOID: %u\n"),
552 ControlFile
.checkPointCopy
.nextOid
);
553 printf(_("Latest checkpoint's NextMultiXactId: %u\n"),
554 ControlFile
.checkPointCopy
.nextMulti
);
555 printf(_("Latest checkpoint's NextMultiOffset: %u\n"),
556 ControlFile
.checkPointCopy
.nextMultiOffset
);
557 printf(_("Maximum data alignment: %u\n"),
558 ControlFile
.maxAlign
);
559 /* we don't print floatFormat since can't say much useful about it */
560 printf(_("Database block size: %u\n"),
562 printf(_("Blocks per segment of large relation: %u\n"),
563 ControlFile
.relseg_size
);
564 printf(_("WAL block size: %u\n"),
565 ControlFile
.xlog_blcksz
);
566 printf(_("Bytes per WAL segment: %u\n"),
567 ControlFile
.xlog_seg_size
);
568 printf(_("Maximum length of identifiers: %u\n"),
569 ControlFile
.nameDataLen
);
570 printf(_("Maximum columns in an index: %u\n"),
571 ControlFile
.indexMaxKeys
);
572 printf(_("Maximum size of a TOAST chunk: %u\n"),
573 ControlFile
.toast_max_chunk_size
);
574 printf(_("Date/time type storage: %s\n"),
575 (ControlFile
.enableIntTimes
? _("64-bit integers") : _("floating-point numbers")));
576 printf(_("Float4 argument passing: %s\n"),
577 (ControlFile
.float4ByVal
? _("by value") : _("by reference")));
578 printf(_("Float8 argument passing: %s\n"),
579 (ControlFile
.float8ByVal
? _("by value") : _("by reference")));
584 * Write out the new pg_control file.
587 RewriteControlFile(void)
590 char buffer
[PG_CONTROL_SIZE
]; /* need not be aligned */
593 * Adjust fields as needed to force an empty XLOG starting at
594 * newXlogId/newXlogSeg.
596 ControlFile
.checkPointCopy
.redo
.xlogid
= newXlogId
;
597 ControlFile
.checkPointCopy
.redo
.xrecoff
=
598 newXlogSeg
* XLogSegSize
+ SizeOfXLogLongPHD
;
599 ControlFile
.checkPointCopy
.time
= (pg_time_t
) time(NULL
);
601 ControlFile
.state
= DB_SHUTDOWNED
;
602 ControlFile
.time
= (pg_time_t
) time(NULL
);
603 ControlFile
.checkPoint
= ControlFile
.checkPointCopy
.redo
;
604 ControlFile
.prevCheckPoint
.xlogid
= 0;
605 ControlFile
.prevCheckPoint
.xrecoff
= 0;
606 ControlFile
.minRecoveryPoint
.xlogid
= 0;
607 ControlFile
.minRecoveryPoint
.xrecoff
= 0;
609 /* Now we can force the recorded xlog seg size to the right thing. */
610 ControlFile
.xlog_seg_size
= XLogSegSize
;
612 /* Contents are protected with a CRC */
613 INIT_CRC32(ControlFile
.crc
);
614 COMP_CRC32(ControlFile
.crc
,
615 (char *) &ControlFile
,
616 offsetof(ControlFileData
, crc
));
617 FIN_CRC32(ControlFile
.crc
);
620 * We write out PG_CONTROL_SIZE bytes into pg_control, zero-padding the
621 * excess over sizeof(ControlFileData). This reduces the odds of
622 * premature-EOF errors when reading pg_control. We'll still fail when we
623 * check the contents of the file, but hopefully with a more specific
624 * error than "couldn't read pg_control".
626 if (sizeof(ControlFileData
) > PG_CONTROL_SIZE
)
629 _("%s: internal error -- sizeof(ControlFileData) is too large ... fix PG_CONTROL_SIZE\n"),
634 memset(buffer
, 0, PG_CONTROL_SIZE
);
635 memcpy(buffer
, &ControlFile
, sizeof(ControlFileData
));
637 unlink(XLOG_CONTROL_FILE
);
639 fd
= open(XLOG_CONTROL_FILE
,
640 O_RDWR
| O_CREAT
| O_EXCL
| PG_BINARY
,
644 fprintf(stderr
, _("%s: could not create pg_control file: %s\n"),
645 progname
, strerror(errno
));
650 if (write(fd
, buffer
, PG_CONTROL_SIZE
) != PG_CONTROL_SIZE
)
652 /* if write didn't set errno, assume problem is no disk space */
655 fprintf(stderr
, _("%s: could not write pg_control file: %s\n"),
656 progname
, strerror(errno
));
662 fprintf(stderr
, _("%s: fsync error: %s\n"), progname
, strerror(errno
));
671 * Scan existing XLOG files and determine the highest existing WAL address
673 * On entry, ControlFile.checkPointCopy.redo and ControlFile.xlog_seg_size
674 * are assumed valid (note that we allow the old xlog seg size to differ
675 * from what we're using). On exit, newXlogId and newXlogSeg are set to
676 * suitable values for the beginning of replacement WAL (in our seg size).
685 * Initialize the max() computation using the last checkpoint address from
686 * old pg_control. Note that for the moment we are working with segment
687 * numbering according to the old xlog seg size.
689 newXlogId
= ControlFile
.checkPointCopy
.redo
.xlogid
;
690 newXlogSeg
= ControlFile
.checkPointCopy
.redo
.xrecoff
/ ControlFile
.xlog_seg_size
;
693 * Scan the pg_xlog directory to find existing WAL segment files. We
694 * assume any present have been used; in most scenarios this should be
695 * conservative, because of xlog.c's attempts to pre-create files.
697 xldir
= opendir(XLOGDIR
);
700 fprintf(stderr
, _("%s: could not open directory \"%s\": %s\n"),
701 progname
, XLOGDIR
, strerror(errno
));
706 while ((xlde
= readdir(xldir
)) != NULL
)
708 if (strlen(xlde
->d_name
) == 24 &&
709 strspn(xlde
->d_name
, "0123456789ABCDEF") == 24)
715 sscanf(xlde
->d_name
, "%08X%08X%08X", &tli
, &log
, &seg
);
718 * Note: we take the max of all files found, regardless of their
719 * timelines. Another possibility would be to ignore files of
720 * timelines other than the target TLI, but this seems safer.
721 * Better too large a result than too small...
723 if (log
> newXlogId
||
724 (log
== newXlogId
&& seg
> newXlogSeg
))
735 * This fix is in mingw cvs (runtime/mingwex/dirent.c rev 1.4), but not in
738 if (GetLastError() == ERROR_NO_MORE_FILES
)
744 fprintf(stderr
, _("%s: could not read from directory \"%s\": %s\n"),
745 progname
, XLOGDIR
, strerror(errno
));
751 * Finally, convert to new xlog seg size, and advance by one to ensure we
752 * are in virgin territory.
754 newXlogSeg
*= ControlFile
.xlog_seg_size
;
755 newXlogSeg
= (newXlogSeg
+ XLogSegSize
- 1) / XLogSegSize
;
757 /* be sure we wrap around correctly at end of a logfile */
758 NextLogSeg(newXlogId
, newXlogSeg
);
763 * Remove existing XLOG files
766 KillExistingXLOG(void)
770 char path
[MAXPGPATH
];
772 xldir
= opendir(XLOGDIR
);
775 fprintf(stderr
, _("%s: could not open directory \"%s\": %s\n"),
776 progname
, XLOGDIR
, strerror(errno
));
781 while ((xlde
= readdir(xldir
)) != NULL
)
783 if (strlen(xlde
->d_name
) == 24 &&
784 strspn(xlde
->d_name
, "0123456789ABCDEF") == 24)
786 snprintf(path
, MAXPGPATH
, "%s/%s", XLOGDIR
, xlde
->d_name
);
787 if (unlink(path
) < 0)
789 fprintf(stderr
, _("%s: could not delete file \"%s\": %s\n"),
790 progname
, path
, strerror(errno
));
799 * This fix is in mingw cvs (runtime/mingwex/dirent.c rev 1.4), but not in
802 if (GetLastError() == ERROR_NO_MORE_FILES
)
808 fprintf(stderr
, _("%s: could not read from directory \"%s\": %s\n"),
809 progname
, XLOGDIR
, strerror(errno
));
817 * Remove existing archive status files
820 KillExistingArchiveStatus(void)
824 char path
[MAXPGPATH
];
826 #define ARCHSTATDIR XLOGDIR "/archive_status"
828 xldir
= opendir(ARCHSTATDIR
);
831 fprintf(stderr
, _("%s: could not open directory \"%s\": %s\n"),
832 progname
, ARCHSTATDIR
, strerror(errno
));
837 while ((xlde
= readdir(xldir
)) != NULL
)
839 if (strspn(xlde
->d_name
, "0123456789ABCDEF") == 24 &&
840 (strcmp(xlde
->d_name
+ 24, ".ready") == 0 ||
841 strcmp(xlde
->d_name
+ 24, ".done") == 0))
843 snprintf(path
, MAXPGPATH
, "%s/%s", ARCHSTATDIR
, xlde
->d_name
);
844 if (unlink(path
) < 0)
846 fprintf(stderr
, _("%s: could not delete file \"%s\": %s\n"),
847 progname
, path
, strerror(errno
));
856 * This fix is in mingw cvs (runtime/mingwex/dirent.c rev 1.4), but not in
859 if (GetLastError() == ERROR_NO_MORE_FILES
)
865 fprintf(stderr
, _("%s: could not read from directory \"%s\": %s\n"),
866 progname
, ARCHSTATDIR
, strerror(errno
));
874 * Write an empty XLOG file, containing only the checkpoint record
875 * already set up in ControlFile.
882 XLogLongPageHeader longpage
;
885 char path
[MAXPGPATH
];
889 /* Use malloc() to ensure buffer is MAXALIGNED */
890 buffer
= (char *) malloc(XLOG_BLCKSZ
);
891 page
= (XLogPageHeader
) buffer
;
892 memset(buffer
, 0, XLOG_BLCKSZ
);
894 /* Set up the XLOG page header */
895 page
->xlp_magic
= XLOG_PAGE_MAGIC
;
896 page
->xlp_info
= XLP_LONG_HEADER
;
897 page
->xlp_tli
= ControlFile
.checkPointCopy
.ThisTimeLineID
;
898 page
->xlp_pageaddr
.xlogid
=
899 ControlFile
.checkPointCopy
.redo
.xlogid
;
900 page
->xlp_pageaddr
.xrecoff
=
901 ControlFile
.checkPointCopy
.redo
.xrecoff
- SizeOfXLogLongPHD
;
902 longpage
= (XLogLongPageHeader
) page
;
903 longpage
->xlp_sysid
= ControlFile
.system_identifier
;
904 longpage
->xlp_seg_size
= XLogSegSize
;
905 longpage
->xlp_xlog_blcksz
= XLOG_BLCKSZ
;
907 /* Insert the initial checkpoint record */
908 record
= (XLogRecord
*) ((char *) page
+ SizeOfXLogLongPHD
);
909 record
->xl_prev
.xlogid
= 0;
910 record
->xl_prev
.xrecoff
= 0;
911 record
->xl_xid
= InvalidTransactionId
;
912 record
->xl_tot_len
= SizeOfXLogRecord
+ sizeof(CheckPoint
);
913 record
->xl_len
= sizeof(CheckPoint
);
914 record
->xl_info
= XLOG_CHECKPOINT_SHUTDOWN
;
915 record
->xl_rmid
= RM_XLOG_ID
;
916 memcpy(XLogRecGetData(record
), &ControlFile
.checkPointCopy
,
920 COMP_CRC32(crc
, &ControlFile
.checkPointCopy
, sizeof(CheckPoint
));
921 COMP_CRC32(crc
, (char *) record
+ sizeof(pg_crc32
),
922 SizeOfXLogRecord
- sizeof(pg_crc32
));
924 record
->xl_crc
= crc
;
926 /* Write the first page */
927 XLogFilePath(path
, ControlFile
.checkPointCopy
.ThisTimeLineID
,
928 newXlogId
, newXlogSeg
);
932 fd
= open(path
, O_RDWR
| O_CREAT
| O_EXCL
| PG_BINARY
,
936 fprintf(stderr
, _("%s: could not open file \"%s\": %s\n"),
937 progname
, path
, strerror(errno
));
942 if (write(fd
, buffer
, XLOG_BLCKSZ
) != XLOG_BLCKSZ
)
944 /* if write didn't set errno, assume problem is no disk space */
947 fprintf(stderr
, _("%s: could not write file \"%s\": %s\n"),
948 progname
, path
, strerror(errno
));
952 /* Fill the rest of the file with zeroes */
953 memset(buffer
, 0, XLOG_BLCKSZ
);
954 for (nbytes
= XLOG_BLCKSZ
; nbytes
< XLogSegSize
; nbytes
+= XLOG_BLCKSZ
)
957 if (write(fd
, buffer
, XLOG_BLCKSZ
) != XLOG_BLCKSZ
)
961 fprintf(stderr
, _("%s: could not write file \"%s\": %s\n"),
962 progname
, path
, strerror(errno
));
969 fprintf(stderr
, _("%s: fsync error: %s\n"), progname
, strerror(errno
));
980 printf(_("%s resets the PostgreSQL transaction log.\n\n"), progname
);
981 printf(_("Usage:\n %s [OPTION]... DATADIR\n\n"), progname
);
982 printf(_("Options:\n"));
983 printf(_(" -e XIDEPOCH set next transaction ID epoch\n"));
984 printf(_(" -f force update to be done\n"));
985 printf(_(" -l TLI,FILE,SEG force minimum WAL starting location for new transaction log\n"));
986 printf(_(" -m XID set next multitransaction ID\n"));
987 printf(_(" -n no update, just show extracted control values (for testing)\n"));
988 printf(_(" -o OID set next OID\n"));
989 printf(_(" -O OFFSET set next multitransaction offset\n"));
990 printf(_(" -x XID set next transaction ID\n"));
991 printf(_(" --help show this help, then exit\n"));
992 printf(_(" --version output version information, then exit\n"));
993 printf(_("\nReport bugs to <pgsql-bugs@postgresql.org>.\n"));