Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * walsummary.c
4 : * Functions for accessing and managing WAL summary data.
5 : *
6 : * Portions Copyright (c) 2010-2025, PostgreSQL Global Development Group
7 : *
8 : * src/backend/backup/walsummary.c
9 : *
10 : *-------------------------------------------------------------------------
11 : */
12 :
13 : #include "postgres.h"
14 :
15 : #include <sys/stat.h>
16 : #include <unistd.h>
17 :
18 : #include "access/xlog_internal.h"
19 : #include "backup/walsummary.h"
20 : #include "common/int.h"
21 : #include "utils/wait_event.h"
22 :
23 : static bool IsWalSummaryFilename(char *filename);
24 : static int ListComparatorForWalSummaryFiles(const ListCell *a,
25 : const ListCell *b);
26 :
27 : /*
28 : * Get a list of WAL summaries.
29 : *
30 : * If tli != 0, only WAL summaries with the indicated TLI will be included.
31 : *
32 : * If start_lsn != InvalidXLogRecPtr, only summaries that end after the
33 : * indicated LSN will be included.
34 : *
35 : * If end_lsn != InvalidXLogRecPtr, only summaries that start before the
36 : * indicated LSN will be included.
37 : *
38 : * The intent is that you can call GetWalSummaries(tli, start_lsn, end_lsn)
39 : * to get all WAL summaries on the indicated timeline that overlap the
40 : * specified LSN range.
41 : */
42 : List *
43 30 : GetWalSummaries(TimeLineID tli, XLogRecPtr start_lsn, XLogRecPtr end_lsn)
44 : {
45 : DIR *sdir;
46 : struct dirent *dent;
47 30 : List *result = NIL;
48 :
49 30 : sdir = AllocateDir(XLOGDIR "/summaries");
50 308 : while ((dent = ReadDir(sdir, XLOGDIR "/summaries")) != NULL)
51 : {
52 : WalSummaryFile *ws;
53 : uint32 tmp[5];
54 : TimeLineID file_tli;
55 : XLogRecPtr file_start_lsn;
56 : XLogRecPtr file_end_lsn;
57 :
58 : /* Decode filename, or skip if it's not in the expected format. */
59 278 : if (!IsWalSummaryFilename(dent->d_name))
60 178 : continue;
61 218 : sscanf(dent->d_name, "%08X%08X%08X%08X%08X",
62 : &tmp[0], &tmp[1], &tmp[2], &tmp[3], &tmp[4]);
63 218 : file_tli = tmp[0];
64 218 : file_start_lsn = ((uint64) tmp[1]) << 32 | tmp[2];
65 218 : file_end_lsn = ((uint64) tmp[3]) << 32 | tmp[4];
66 :
67 : /* Skip if it doesn't match the filter criteria. */
68 218 : if (tli != 0 && tli != file_tli)
69 0 : continue;
70 218 : if (!XLogRecPtrIsInvalid(start_lsn) && start_lsn >= file_end_lsn)
71 118 : continue;
72 100 : if (!XLogRecPtrIsInvalid(end_lsn) && end_lsn <= file_start_lsn)
73 0 : continue;
74 :
75 : /* Add it to the list. */
76 100 : ws = palloc(sizeof(WalSummaryFile));
77 100 : ws->tli = file_tli;
78 100 : ws->start_lsn = file_start_lsn;
79 100 : ws->end_lsn = file_end_lsn;
80 100 : result = lappend(result, ws);
81 : }
82 30 : FreeDir(sdir);
83 :
84 30 : return result;
85 : }
86 :
87 : /*
88 : * Build a new list of WAL summaries based on an existing list, but filtering
89 : * out summaries that don't match the search parameters.
90 : *
91 : * If tli != 0, only WAL summaries with the indicated TLI will be included.
92 : *
93 : * If start_lsn != InvalidXLogRecPtr, only summaries that end after the
94 : * indicated LSN will be included.
95 : *
96 : * If end_lsn != InvalidXLogRecPtr, only summaries that start before the
97 : * indicated LSN will be included.
98 : */
99 : List *
100 20 : FilterWalSummaries(List *wslist, TimeLineID tli,
101 : XLogRecPtr start_lsn, XLogRecPtr end_lsn)
102 : {
103 20 : List *result = NIL;
104 : ListCell *lc;
105 :
106 : /* Loop over input. */
107 62 : foreach(lc, wslist)
108 : {
109 42 : WalSummaryFile *ws = lfirst(lc);
110 :
111 : /* Skip if it doesn't match the filter criteria. */
112 42 : if (tli != 0 && tli != ws->tli)
113 8 : continue;
114 34 : if (!XLogRecPtrIsInvalid(start_lsn) && start_lsn > ws->end_lsn)
115 0 : continue;
116 34 : if (!XLogRecPtrIsInvalid(end_lsn) && end_lsn < ws->start_lsn)
117 0 : continue;
118 :
119 : /* Add it to the result list. */
120 34 : result = lappend(result, ws);
121 : }
122 :
123 20 : return result;
124 : }
125 :
126 : /*
127 : * Check whether the supplied list of WalSummaryFile objects covers the
128 : * whole range of LSNs from start_lsn to end_lsn. This function ignores
129 : * timelines, so the caller should probably filter using the appropriate
130 : * timeline before calling this.
131 : *
132 : * If the whole range of LSNs is covered, returns true, otherwise false.
133 : * If false is returned, *missing_lsn is set either to InvalidXLogRecPtr
134 : * if there are no WAL summary files in the input list, or to the first LSN
135 : * in the range that is not covered by a WAL summary file in the input list.
136 : */
137 : bool
138 20 : WalSummariesAreComplete(List *wslist, XLogRecPtr start_lsn,
139 : XLogRecPtr end_lsn, XLogRecPtr *missing_lsn)
140 : {
141 20 : XLogRecPtr current_lsn = start_lsn;
142 : ListCell *lc;
143 :
144 : /* Special case for empty list. */
145 20 : if (wslist == NIL)
146 : {
147 0 : *missing_lsn = InvalidXLogRecPtr;
148 0 : return false;
149 : }
150 :
151 : /* Make a private copy of the list and sort it by start LSN. */
152 20 : wslist = list_copy(wslist);
153 20 : list_sort(wslist, ListComparatorForWalSummaryFiles);
154 :
155 : /*
156 : * Consider summary files in order of increasing start_lsn, advancing the
157 : * known-summarized range from start_lsn toward end_lsn.
158 : *
159 : * Normally, the summary files should cover non-overlapping WAL ranges,
160 : * but this algorithm is intended to be correct even in case of overlap.
161 : */
162 34 : foreach(lc, wslist)
163 : {
164 34 : WalSummaryFile *ws = lfirst(lc);
165 :
166 34 : if (ws->start_lsn > current_lsn)
167 : {
168 : /* We found a gap. */
169 0 : break;
170 : }
171 34 : if (ws->end_lsn > current_lsn)
172 : {
173 : /*
174 : * Next summary extends beyond end of previous summary, so extend
175 : * the end of the range known to be summarized.
176 : */
177 34 : current_lsn = ws->end_lsn;
178 :
179 : /*
180 : * If the range we know to be summarized has reached the required
181 : * end LSN, we have proved completeness.
182 : */
183 34 : if (current_lsn >= end_lsn)
184 20 : return true;
185 : }
186 : }
187 :
188 : /*
189 : * We either ran out of summary files without reaching the end LSN, or we
190 : * hit a gap in the sequence that resulted in us bailing out of the loop
191 : * above.
192 : */
193 0 : *missing_lsn = current_lsn;
194 0 : return false;
195 : }
196 :
197 : /*
198 : * Open a WAL summary file.
199 : *
200 : * This will throw an error in case of trouble. As an exception, if
201 : * missing_ok = true and the trouble is specifically that the file does
202 : * not exist, it will not throw an error and will return a value less than 0.
203 : */
204 : File
205 34 : OpenWalSummaryFile(WalSummaryFile *ws, bool missing_ok)
206 : {
207 : char path[MAXPGPATH];
208 : File file;
209 :
210 34 : snprintf(path, MAXPGPATH,
211 : XLOGDIR "/summaries/%08X%08X%08X%08X%08X.summary",
212 : ws->tli,
213 34 : LSN_FORMAT_ARGS(ws->start_lsn),
214 34 : LSN_FORMAT_ARGS(ws->end_lsn));
215 :
216 34 : file = PathNameOpenFile(path, O_RDONLY);
217 34 : if (file < 0 && (errno != EEXIST || !missing_ok))
218 0 : ereport(ERROR,
219 : (errcode_for_file_access(),
220 : errmsg("could not open file \"%s\": %m", path)));
221 :
222 34 : return file;
223 : }
224 :
225 : /*
226 : * Remove a WAL summary file if the last modification time precedes the
227 : * cutoff time.
228 : */
229 : void
230 0 : RemoveWalSummaryIfOlderThan(WalSummaryFile *ws, time_t cutoff_time)
231 : {
232 : char path[MAXPGPATH];
233 : struct stat statbuf;
234 :
235 0 : snprintf(path, MAXPGPATH,
236 : XLOGDIR "/summaries/%08X%08X%08X%08X%08X.summary",
237 : ws->tli,
238 0 : LSN_FORMAT_ARGS(ws->start_lsn),
239 0 : LSN_FORMAT_ARGS(ws->end_lsn));
240 :
241 0 : if (lstat(path, &statbuf) != 0)
242 : {
243 0 : if (errno == ENOENT)
244 0 : return;
245 0 : ereport(ERROR,
246 : (errcode_for_file_access(),
247 : errmsg("could not stat file \"%s\": %m", path)));
248 : }
249 0 : if (statbuf.st_mtime >= cutoff_time)
250 0 : return;
251 0 : if (unlink(path) != 0)
252 0 : ereport(ERROR,
253 : (errcode_for_file_access(),
254 : errmsg("could not stat file \"%s\": %m", path)));
255 0 : ereport(DEBUG2,
256 : (errmsg_internal("removing file \"%s\"", path)));
257 : }
258 :
259 : /*
260 : * Test whether a filename looks like a WAL summary file.
261 : */
262 : static bool
263 278 : IsWalSummaryFilename(char *filename)
264 : {
265 496 : return strspn(filename, "0123456789ABCDEF") == 40 &&
266 218 : strcmp(filename + 40, ".summary") == 0;
267 : }
268 :
269 : /*
270 : * Data read callback for use with CreateBlockRefTableReader.
271 : */
272 : int
273 34 : ReadWalSummary(void *wal_summary_io, void *data, int length)
274 : {
275 34 : WalSummaryIO *io = wal_summary_io;
276 : int nbytes;
277 :
278 34 : nbytes = FileRead(io->file, data, length, io->filepos,
279 : WAIT_EVENT_WAL_SUMMARY_READ);
280 34 : if (nbytes < 0)
281 0 : ereport(ERROR,
282 : (errcode_for_file_access(),
283 : errmsg("could not read file \"%s\": %m",
284 : FilePathName(io->file))));
285 :
286 34 : io->filepos += nbytes;
287 34 : return nbytes;
288 : }
289 :
290 : /*
291 : * Data write callback for use with WriteBlockRefTable.
292 : */
293 : int
294 16 : WriteWalSummary(void *wal_summary_io, void *data, int length)
295 : {
296 16 : WalSummaryIO *io = wal_summary_io;
297 : int nbytes;
298 :
299 16 : nbytes = FileWrite(io->file, data, length, io->filepos,
300 : WAIT_EVENT_WAL_SUMMARY_WRITE);
301 16 : if (nbytes < 0)
302 0 : ereport(ERROR,
303 : (errcode_for_file_access(),
304 : errmsg("could not write file \"%s\": %m",
305 : FilePathName(io->file))));
306 16 : if (nbytes != length)
307 0 : ereport(ERROR,
308 : (errcode_for_file_access(),
309 : errmsg("could not write file \"%s\": wrote only %d of %d bytes at offset %u",
310 : FilePathName(io->file), nbytes,
311 : length, (unsigned) io->filepos),
312 : errhint("Check free disk space.")));
313 :
314 16 : io->filepos += nbytes;
315 16 : return nbytes;
316 : }
317 :
318 : /*
319 : * Error-reporting callback for use with CreateBlockRefTableReader.
320 : */
321 : void
322 0 : ReportWalSummaryError(void *callback_arg, char *fmt,...)
323 : {
324 : StringInfoData buf;
325 : va_list ap;
326 : int needed;
327 :
328 0 : initStringInfo(&buf);
329 : for (;;)
330 : {
331 0 : va_start(ap, fmt);
332 0 : needed = appendStringInfoVA(&buf, fmt, ap);
333 0 : va_end(ap);
334 0 : if (needed == 0)
335 0 : break;
336 0 : enlargeStringInfo(&buf, needed);
337 : }
338 0 : ereport(ERROR,
339 : errcode(ERRCODE_DATA_CORRUPTED),
340 : errmsg_internal("%s", buf.data));
341 : }
342 :
343 : /*
344 : * Comparator to sort a List of WalSummaryFile objects by start_lsn.
345 : */
346 : static int
347 18 : ListComparatorForWalSummaryFiles(const ListCell *a, const ListCell *b)
348 : {
349 18 : WalSummaryFile *ws1 = lfirst(a);
350 18 : WalSummaryFile *ws2 = lfirst(b);
351 :
352 18 : return pg_cmp_u64(ws1->start_lsn, ws2->start_lsn);
353 : }
|