Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * heapam_xlog.c
4 : * WAL replay logic for heap access method.
5 : *
6 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
7 : * Portions Copyright (c) 1994, Regents of the University of California
8 : *
9 : *
10 : * IDENTIFICATION
11 : * src/backend/access/heap/heapam_xlog.c
12 : *
13 : *-------------------------------------------------------------------------
14 : */
15 : #include "postgres.h"
16 :
17 : #include "access/bufmask.h"
18 : #include "access/heapam.h"
19 : #include "access/visibilitymap.h"
20 : #include "access/xlog.h"
21 : #include "access/xlogutils.h"
22 : #include "storage/freespace.h"
23 : #include "storage/standby.h"
24 :
25 :
26 : /*
27 : * Replay XLOG_HEAP2_PRUNE_* records.
28 : */
29 : static void
30 17988 : heap_xlog_prune_freeze(XLogReaderState *record)
31 : {
32 17988 : XLogRecPtr lsn = record->EndRecPtr;
33 17988 : char *maindataptr = XLogRecGetData(record);
34 : xl_heap_prune xlrec;
35 : Buffer buffer;
36 : RelFileLocator rlocator;
37 : BlockNumber blkno;
38 : XLogRedoAction action;
39 :
40 17988 : XLogRecGetBlockTag(record, 0, &rlocator, NULL, &blkno);
41 17988 : memcpy(&xlrec, maindataptr, SizeOfHeapPrune);
42 17988 : maindataptr += SizeOfHeapPrune;
43 :
44 : /*
45 : * We will take an ordinary exclusive lock or a cleanup lock depending on
46 : * whether the XLHP_CLEANUP_LOCK flag is set. With an ordinary exclusive
47 : * lock, we better not be doing anything that requires moving existing
48 : * tuple data.
49 : */
50 : Assert((xlrec.flags & XLHP_CLEANUP_LOCK) != 0 ||
51 : (xlrec.flags & (XLHP_HAS_REDIRECTIONS | XLHP_HAS_DEAD_ITEMS)) == 0);
52 :
53 : /*
54 : * We are about to remove and/or freeze tuples. In Hot Standby mode,
55 : * ensure that there are no queries running for which the removed tuples
56 : * are still visible or which still consider the frozen xids as running.
57 : * The conflict horizon XID comes after xl_heap_prune.
58 : */
59 17988 : if ((xlrec.flags & XLHP_HAS_CONFLICT_HORIZON) != 0)
60 : {
61 : TransactionId snapshot_conflict_horizon;
62 :
63 : /* memcpy() because snapshot_conflict_horizon is stored unaligned */
64 14412 : memcpy(&snapshot_conflict_horizon, maindataptr, sizeof(TransactionId));
65 14412 : maindataptr += sizeof(TransactionId);
66 :
67 14412 : if (InHotStandby)
68 13942 : ResolveRecoveryConflictWithSnapshot(snapshot_conflict_horizon,
69 13942 : (xlrec.flags & XLHP_IS_CATALOG_REL) != 0,
70 : rlocator);
71 : }
72 :
73 : /*
74 : * If we have a full-page image, restore it and we're done.
75 : */
76 17988 : action = XLogReadBufferForRedoExtended(record, 0, RBM_NORMAL,
77 17988 : (xlrec.flags & XLHP_CLEANUP_LOCK) != 0,
78 : &buffer);
79 17988 : if (action == BLK_NEEDS_REDO)
80 : {
81 15688 : Page page = (Page) BufferGetPage(buffer);
82 : OffsetNumber *redirected;
83 : OffsetNumber *nowdead;
84 : OffsetNumber *nowunused;
85 : int nredirected;
86 : int ndead;
87 : int nunused;
88 : int nplans;
89 : Size datalen;
90 : xlhp_freeze_plan *plans;
91 : OffsetNumber *frz_offsets;
92 15688 : char *dataptr = XLogRecGetBlockData(record, 0, &datalen);
93 :
94 15688 : heap_xlog_deserialize_prune_and_freeze(dataptr, xlrec.flags,
95 : &nplans, &plans, &frz_offsets,
96 : &nredirected, &redirected,
97 : &ndead, &nowdead,
98 : &nunused, &nowunused);
99 :
100 : /*
101 : * Update all line pointers per the record, and repair fragmentation
102 : * if needed.
103 : */
104 15688 : if (nredirected > 0 || ndead > 0 || nunused > 0)
105 14724 : heap_page_prune_execute(buffer,
106 14724 : (xlrec.flags & XLHP_CLEANUP_LOCK) == 0,
107 : redirected, nredirected,
108 : nowdead, ndead,
109 : nowunused, nunused);
110 :
111 : /* Freeze tuples */
112 16844 : for (int p = 0; p < nplans; p++)
113 : {
114 : HeapTupleFreeze frz;
115 :
116 : /*
117 : * Convert freeze plan representation from WAL record into
118 : * per-tuple format used by heap_execute_freeze_tuple
119 : */
120 1156 : frz.xmax = plans[p].xmax;
121 1156 : frz.t_infomask2 = plans[p].t_infomask2;
122 1156 : frz.t_infomask = plans[p].t_infomask;
123 1156 : frz.frzflags = plans[p].frzflags;
124 1156 : frz.offset = InvalidOffsetNumber; /* unused, but be tidy */
125 :
126 102172 : for (int i = 0; i < plans[p].ntuples; i++)
127 : {
128 101016 : OffsetNumber offset = *(frz_offsets++);
129 : ItemId lp;
130 : HeapTupleHeader tuple;
131 :
132 101016 : lp = PageGetItemId(page, offset);
133 101016 : tuple = (HeapTupleHeader) PageGetItem(page, lp);
134 101016 : heap_execute_freeze_tuple(tuple, &frz);
135 : }
136 : }
137 :
138 : /* There should be no more data */
139 : Assert((char *) frz_offsets == dataptr + datalen);
140 :
141 : /*
142 : * Note: we don't worry about updating the page's prunability hints.
143 : * At worst this will cause an extra prune cycle to occur soon.
144 : */
145 :
146 15688 : PageSetLSN(page, lsn);
147 15688 : MarkBufferDirty(buffer);
148 : }
149 :
150 : /*
151 : * If we released any space or line pointers, update the free space map.
152 : *
153 : * Do this regardless of a full-page image being applied, since the FSM
154 : * data is not in the page anyway.
155 : */
156 17988 : if (BufferIsValid(buffer))
157 : {
158 17988 : if (xlrec.flags & (XLHP_HAS_REDIRECTIONS |
159 : XLHP_HAS_DEAD_ITEMS |
160 : XLHP_HAS_NOW_UNUSED_ITEMS))
161 : {
162 17020 : Size freespace = PageGetHeapFreeSpace(BufferGetPage(buffer));
163 :
164 17020 : UnlockReleaseBuffer(buffer);
165 :
166 17020 : XLogRecordPageWithFreeSpace(rlocator, blkno, freespace);
167 : }
168 : else
169 968 : UnlockReleaseBuffer(buffer);
170 : }
171 17988 : }
172 :
173 : /*
174 : * Replay XLOG_HEAP2_VISIBLE records.
175 : *
176 : * The critical integrity requirement here is that we must never end up with
177 : * a situation where the visibility map bit is set, and the page-level
178 : * PD_ALL_VISIBLE bit is clear. If that were to occur, then a subsequent
179 : * page modification would fail to clear the visibility map bit.
180 : */
181 : static void
182 7920 : heap_xlog_visible(XLogReaderState *record)
183 : {
184 7920 : XLogRecPtr lsn = record->EndRecPtr;
185 7920 : xl_heap_visible *xlrec = (xl_heap_visible *) XLogRecGetData(record);
186 7920 : Buffer vmbuffer = InvalidBuffer;
187 : Buffer buffer;
188 : Page page;
189 : RelFileLocator rlocator;
190 : BlockNumber blkno;
191 : XLogRedoAction action;
192 :
193 : Assert((xlrec->flags & VISIBILITYMAP_XLOG_VALID_BITS) == xlrec->flags);
194 :
195 7920 : XLogRecGetBlockTag(record, 1, &rlocator, NULL, &blkno);
196 :
197 : /*
198 : * If there are any Hot Standby transactions running that have an xmin
199 : * horizon old enough that this page isn't all-visible for them, they
200 : * might incorrectly decide that an index-only scan can skip a heap fetch.
201 : *
202 : * NB: It might be better to throw some kind of "soft" conflict here that
203 : * forces any index-only scan that is in flight to perform heap fetches,
204 : * rather than killing the transaction outright.
205 : */
206 7920 : if (InHotStandby)
207 7572 : ResolveRecoveryConflictWithSnapshot(xlrec->snapshotConflictHorizon,
208 7572 : xlrec->flags & VISIBILITYMAP_XLOG_CATALOG_REL,
209 : rlocator);
210 :
211 : /*
212 : * Read the heap page, if it still exists. If the heap file has dropped or
213 : * truncated later in recovery, we don't need to update the page, but we'd
214 : * better still update the visibility map.
215 : */
216 7920 : action = XLogReadBufferForRedo(record, 1, &buffer);
217 7920 : if (action == BLK_NEEDS_REDO)
218 : {
219 : /*
220 : * We don't bump the LSN of the heap page when setting the visibility
221 : * map bit (unless checksums or wal_hint_bits is enabled, in which
222 : * case we must). This exposes us to torn page hazards, but since
223 : * we're not inspecting the existing page contents in any way, we
224 : * don't care.
225 : */
226 5656 : page = BufferGetPage(buffer);
227 :
228 5656 : PageSetAllVisible(page);
229 :
230 5656 : if (XLogHintBitIsNeeded())
231 5656 : PageSetLSN(page, lsn);
232 :
233 5656 : MarkBufferDirty(buffer);
234 : }
235 : else if (action == BLK_RESTORED)
236 : {
237 : /*
238 : * If heap block was backed up, we already restored it and there's
239 : * nothing more to do. (This can only happen with checksums or
240 : * wal_log_hints enabled.)
241 : */
242 : }
243 :
244 7920 : if (BufferIsValid(buffer))
245 : {
246 7920 : Size space = PageGetFreeSpace(BufferGetPage(buffer));
247 :
248 7920 : UnlockReleaseBuffer(buffer);
249 :
250 : /*
251 : * Since FSM is not WAL-logged and only updated heuristically, it
252 : * easily becomes stale in standbys. If the standby is later promoted
253 : * and runs VACUUM, it will skip updating individual free space
254 : * figures for pages that became all-visible (or all-frozen, depending
255 : * on the vacuum mode,) which is troublesome when FreeSpaceMapVacuum
256 : * propagates too optimistic free space values to upper FSM layers;
257 : * later inserters try to use such pages only to find out that they
258 : * are unusable. This can cause long stalls when there are many such
259 : * pages.
260 : *
261 : * Forestall those problems by updating FSM's idea about a page that
262 : * is becoming all-visible or all-frozen.
263 : *
264 : * Do this regardless of a full-page image being applied, since the
265 : * FSM data is not in the page anyway.
266 : */
267 7920 : if (xlrec->flags & VISIBILITYMAP_VALID_BITS)
268 7920 : XLogRecordPageWithFreeSpace(rlocator, blkno, space);
269 : }
270 :
271 : /*
272 : * Even if we skipped the heap page update due to the LSN interlock, it's
273 : * still safe to update the visibility map. Any WAL record that clears
274 : * the visibility map bit does so before checking the page LSN, so any
275 : * bits that need to be cleared will still be cleared.
276 : */
277 7920 : if (XLogReadBufferForRedoExtended(record, 0, RBM_ZERO_ON_ERROR, false,
278 : &vmbuffer) == BLK_NEEDS_REDO)
279 : {
280 7476 : Page vmpage = BufferGetPage(vmbuffer);
281 : Relation reln;
282 : uint8 vmbits;
283 :
284 : /* initialize the page if it was read as zeros */
285 7476 : if (PageIsNew(vmpage))
286 0 : PageInit(vmpage, BLCKSZ, 0);
287 :
288 : /* remove VISIBILITYMAP_XLOG_* */
289 7476 : vmbits = xlrec->flags & VISIBILITYMAP_VALID_BITS;
290 :
291 : /*
292 : * XLogReadBufferForRedoExtended locked the buffer. But
293 : * visibilitymap_set will handle locking itself.
294 : */
295 7476 : LockBuffer(vmbuffer, BUFFER_LOCK_UNLOCK);
296 :
297 7476 : reln = CreateFakeRelcacheEntry(rlocator);
298 7476 : visibilitymap_pin(reln, blkno, &vmbuffer);
299 :
300 7476 : visibilitymap_set(reln, blkno, InvalidBuffer, lsn, vmbuffer,
301 : xlrec->snapshotConflictHorizon, vmbits);
302 :
303 7476 : ReleaseBuffer(vmbuffer);
304 7476 : FreeFakeRelcacheEntry(reln);
305 : }
306 444 : else if (BufferIsValid(vmbuffer))
307 444 : UnlockReleaseBuffer(vmbuffer);
308 7920 : }
309 :
310 : /*
311 : * Given an "infobits" field from an XLog record, set the correct bits in the
312 : * given infomask and infomask2 for the tuple touched by the record.
313 : *
314 : * (This is the reverse of compute_infobits).
315 : */
316 : static void
317 872948 : fix_infomask_from_infobits(uint8 infobits, uint16 *infomask, uint16 *infomask2)
318 : {
319 872948 : *infomask &= ~(HEAP_XMAX_IS_MULTI | HEAP_XMAX_LOCK_ONLY |
320 : HEAP_XMAX_KEYSHR_LOCK | HEAP_XMAX_EXCL_LOCK);
321 872948 : *infomask2 &= ~HEAP_KEYS_UPDATED;
322 :
323 872948 : if (infobits & XLHL_XMAX_IS_MULTI)
324 4 : *infomask |= HEAP_XMAX_IS_MULTI;
325 872948 : if (infobits & XLHL_XMAX_LOCK_ONLY)
326 108618 : *infomask |= HEAP_XMAX_LOCK_ONLY;
327 872948 : if (infobits & XLHL_XMAX_EXCL_LOCK)
328 107812 : *infomask |= HEAP_XMAX_EXCL_LOCK;
329 : /* note HEAP_XMAX_SHR_LOCK isn't considered here */
330 872948 : if (infobits & XLHL_XMAX_KEYSHR_LOCK)
331 822 : *infomask |= HEAP_XMAX_KEYSHR_LOCK;
332 :
333 872948 : if (infobits & XLHL_KEYS_UPDATED)
334 583730 : *infomask2 |= HEAP_KEYS_UPDATED;
335 872948 : }
336 :
337 : /*
338 : * Replay XLOG_HEAP_DELETE records.
339 : */
340 : static void
341 584916 : heap_xlog_delete(XLogReaderState *record)
342 : {
343 584916 : XLogRecPtr lsn = record->EndRecPtr;
344 584916 : xl_heap_delete *xlrec = (xl_heap_delete *) XLogRecGetData(record);
345 : Buffer buffer;
346 : Page page;
347 584916 : ItemId lp = NULL;
348 : HeapTupleHeader htup;
349 : BlockNumber blkno;
350 : RelFileLocator target_locator;
351 : ItemPointerData target_tid;
352 :
353 584916 : XLogRecGetBlockTag(record, 0, &target_locator, NULL, &blkno);
354 584916 : ItemPointerSetBlockNumber(&target_tid, blkno);
355 584916 : ItemPointerSetOffsetNumber(&target_tid, xlrec->offnum);
356 :
357 : /*
358 : * The visibility map may need to be fixed even if the heap page is
359 : * already up-to-date.
360 : */
361 584916 : if (xlrec->flags & XLH_DELETE_ALL_VISIBLE_CLEARED)
362 : {
363 348 : Relation reln = CreateFakeRelcacheEntry(target_locator);
364 348 : Buffer vmbuffer = InvalidBuffer;
365 :
366 348 : visibilitymap_pin(reln, blkno, &vmbuffer);
367 348 : visibilitymap_clear(reln, blkno, vmbuffer, VISIBILITYMAP_VALID_BITS);
368 348 : ReleaseBuffer(vmbuffer);
369 348 : FreeFakeRelcacheEntry(reln);
370 : }
371 :
372 584916 : if (XLogReadBufferForRedo(record, 0, &buffer) == BLK_NEEDS_REDO)
373 : {
374 581094 : page = BufferGetPage(buffer);
375 :
376 581094 : if (PageGetMaxOffsetNumber(page) >= xlrec->offnum)
377 581094 : lp = PageGetItemId(page, xlrec->offnum);
378 :
379 581094 : if (PageGetMaxOffsetNumber(page) < xlrec->offnum || !ItemIdIsNormal(lp))
380 0 : elog(PANIC, "invalid lp");
381 :
382 581094 : htup = (HeapTupleHeader) PageGetItem(page, lp);
383 :
384 581094 : htup->t_infomask &= ~(HEAP_XMAX_BITS | HEAP_MOVED);
385 581094 : htup->t_infomask2 &= ~HEAP_KEYS_UPDATED;
386 581094 : HeapTupleHeaderClearHotUpdated(htup);
387 581094 : fix_infomask_from_infobits(xlrec->infobits_set,
388 : &htup->t_infomask, &htup->t_infomask2);
389 581094 : if (!(xlrec->flags & XLH_DELETE_IS_SUPER))
390 581094 : HeapTupleHeaderSetXmax(htup, xlrec->xmax);
391 : else
392 0 : HeapTupleHeaderSetXmin(htup, InvalidTransactionId);
393 581094 : HeapTupleHeaderSetCmax(htup, FirstCommandId, false);
394 :
395 : /* Mark the page as a candidate for pruning */
396 581094 : PageSetPrunable(page, XLogRecGetXid(record));
397 :
398 581094 : if (xlrec->flags & XLH_DELETE_ALL_VISIBLE_CLEARED)
399 6 : PageClearAllVisible(page);
400 :
401 : /* Make sure t_ctid is set correctly */
402 581094 : if (xlrec->flags & XLH_DELETE_IS_PARTITION_MOVE)
403 278 : HeapTupleHeaderSetMovedPartitions(htup);
404 : else
405 580816 : htup->t_ctid = target_tid;
406 581094 : PageSetLSN(page, lsn);
407 581094 : MarkBufferDirty(buffer);
408 : }
409 584916 : if (BufferIsValid(buffer))
410 584916 : UnlockReleaseBuffer(buffer);
411 584916 : }
412 :
413 : /*
414 : * Replay XLOG_HEAP_INSERT records.
415 : */
416 : static void
417 2505260 : heap_xlog_insert(XLogReaderState *record)
418 : {
419 2505260 : XLogRecPtr lsn = record->EndRecPtr;
420 2505260 : xl_heap_insert *xlrec = (xl_heap_insert *) XLogRecGetData(record);
421 : Buffer buffer;
422 : Page page;
423 : union
424 : {
425 : HeapTupleHeaderData hdr;
426 : char data[MaxHeapTupleSize];
427 : } tbuf;
428 : HeapTupleHeader htup;
429 : xl_heap_header xlhdr;
430 : uint32 newlen;
431 2505260 : Size freespace = 0;
432 : RelFileLocator target_locator;
433 : BlockNumber blkno;
434 : ItemPointerData target_tid;
435 : XLogRedoAction action;
436 :
437 2505260 : XLogRecGetBlockTag(record, 0, &target_locator, NULL, &blkno);
438 2505260 : ItemPointerSetBlockNumber(&target_tid, blkno);
439 2505260 : ItemPointerSetOffsetNumber(&target_tid, xlrec->offnum);
440 :
441 : /*
442 : * The visibility map may need to be fixed even if the heap page is
443 : * already up-to-date.
444 : */
445 2505260 : if (xlrec->flags & XLH_INSERT_ALL_VISIBLE_CLEARED)
446 : {
447 1162 : Relation reln = CreateFakeRelcacheEntry(target_locator);
448 1162 : Buffer vmbuffer = InvalidBuffer;
449 :
450 1162 : visibilitymap_pin(reln, blkno, &vmbuffer);
451 1162 : visibilitymap_clear(reln, blkno, vmbuffer, VISIBILITYMAP_VALID_BITS);
452 1162 : ReleaseBuffer(vmbuffer);
453 1162 : FreeFakeRelcacheEntry(reln);
454 : }
455 :
456 : /*
457 : * If we inserted the first and only tuple on the page, re-initialize the
458 : * page from scratch.
459 : */
460 2505260 : if (XLogRecGetInfo(record) & XLOG_HEAP_INIT_PAGE)
461 : {
462 31676 : buffer = XLogInitBufferForRedo(record, 0);
463 31676 : page = BufferGetPage(buffer);
464 31676 : PageInit(page, BufferGetPageSize(buffer), 0);
465 31676 : action = BLK_NEEDS_REDO;
466 : }
467 : else
468 2473584 : action = XLogReadBufferForRedo(record, 0, &buffer);
469 2505260 : if (action == BLK_NEEDS_REDO)
470 : {
471 : Size datalen;
472 : char *data;
473 :
474 2500764 : page = BufferGetPage(buffer);
475 :
476 2500764 : if (PageGetMaxOffsetNumber(page) + 1 < xlrec->offnum)
477 0 : elog(PANIC, "invalid max offset number");
478 :
479 2500764 : data = XLogRecGetBlockData(record, 0, &datalen);
480 :
481 2500764 : newlen = datalen - SizeOfHeapHeader;
482 : Assert(datalen > SizeOfHeapHeader && newlen <= MaxHeapTupleSize);
483 2500764 : memcpy((char *) &xlhdr, data, SizeOfHeapHeader);
484 2500764 : data += SizeOfHeapHeader;
485 :
486 2500764 : htup = &tbuf.hdr;
487 2500764 : MemSet((char *) htup, 0, SizeofHeapTupleHeader);
488 : /* PG73FORMAT: get bitmap [+ padding] [+ oid] + data */
489 2500764 : memcpy((char *) htup + SizeofHeapTupleHeader,
490 : data,
491 : newlen);
492 2500764 : newlen += SizeofHeapTupleHeader;
493 2500764 : htup->t_infomask2 = xlhdr.t_infomask2;
494 2500764 : htup->t_infomask = xlhdr.t_infomask;
495 2500764 : htup->t_hoff = xlhdr.t_hoff;
496 2500764 : HeapTupleHeaderSetXmin(htup, XLogRecGetXid(record));
497 2500764 : HeapTupleHeaderSetCmin(htup, FirstCommandId);
498 2500764 : htup->t_ctid = target_tid;
499 :
500 2500764 : if (PageAddItem(page, (Item) htup, newlen, xlrec->offnum,
501 : true, true) == InvalidOffsetNumber)
502 0 : elog(PANIC, "failed to add tuple");
503 :
504 2500764 : freespace = PageGetHeapFreeSpace(page); /* needed to update FSM below */
505 :
506 2500764 : PageSetLSN(page, lsn);
507 :
508 2500764 : if (xlrec->flags & XLH_INSERT_ALL_VISIBLE_CLEARED)
509 336 : PageClearAllVisible(page);
510 :
511 : /* XLH_INSERT_ALL_FROZEN_SET implies that all tuples are visible */
512 2500764 : if (xlrec->flags & XLH_INSERT_ALL_FROZEN_SET)
513 0 : PageSetAllVisible(page);
514 :
515 2500764 : MarkBufferDirty(buffer);
516 : }
517 2505260 : if (BufferIsValid(buffer))
518 2505260 : UnlockReleaseBuffer(buffer);
519 :
520 : /*
521 : * If the page is running low on free space, update the FSM as well.
522 : * Arbitrarily, our definition of "low" is less than 20%. We can't do much
523 : * better than that without knowing the fill-factor for the table.
524 : *
525 : * XXX: Don't do this if the page was restored from full page image. We
526 : * don't bother to update the FSM in that case, it doesn't need to be
527 : * totally accurate anyway.
528 : */
529 2505260 : if (action == BLK_NEEDS_REDO && freespace < BLCKSZ / 5)
530 492424 : XLogRecordPageWithFreeSpace(target_locator, blkno, freespace);
531 2505260 : }
532 :
533 : /*
534 : * Replay XLOG_HEAP2_MULTI_INSERT records.
535 : */
536 : static void
537 107752 : heap_xlog_multi_insert(XLogReaderState *record)
538 : {
539 107752 : XLogRecPtr lsn = record->EndRecPtr;
540 : xl_heap_multi_insert *xlrec;
541 : RelFileLocator rlocator;
542 : BlockNumber blkno;
543 : Buffer buffer;
544 : Page page;
545 : union
546 : {
547 : HeapTupleHeaderData hdr;
548 : char data[MaxHeapTupleSize];
549 : } tbuf;
550 : HeapTupleHeader htup;
551 : uint32 newlen;
552 107752 : Size freespace = 0;
553 : int i;
554 107752 : bool isinit = (XLogRecGetInfo(record) & XLOG_HEAP_INIT_PAGE) != 0;
555 : XLogRedoAction action;
556 :
557 : /*
558 : * Insertion doesn't overwrite MVCC data, so no conflict processing is
559 : * required.
560 : */
561 107752 : xlrec = (xl_heap_multi_insert *) XLogRecGetData(record);
562 :
563 107752 : XLogRecGetBlockTag(record, 0, &rlocator, NULL, &blkno);
564 :
565 : /* check that the mutually exclusive flags are not both set */
566 : Assert(!((xlrec->flags & XLH_INSERT_ALL_VISIBLE_CLEARED) &&
567 : (xlrec->flags & XLH_INSERT_ALL_FROZEN_SET)));
568 :
569 : /*
570 : * The visibility map may need to be fixed even if the heap page is
571 : * already up-to-date.
572 : */
573 107752 : if (xlrec->flags & XLH_INSERT_ALL_VISIBLE_CLEARED)
574 : {
575 1038 : Relation reln = CreateFakeRelcacheEntry(rlocator);
576 1038 : Buffer vmbuffer = InvalidBuffer;
577 :
578 1038 : visibilitymap_pin(reln, blkno, &vmbuffer);
579 1038 : visibilitymap_clear(reln, blkno, vmbuffer, VISIBILITYMAP_VALID_BITS);
580 1038 : ReleaseBuffer(vmbuffer);
581 1038 : FreeFakeRelcacheEntry(reln);
582 : }
583 :
584 107752 : if (isinit)
585 : {
586 4098 : buffer = XLogInitBufferForRedo(record, 0);
587 4098 : page = BufferGetPage(buffer);
588 4098 : PageInit(page, BufferGetPageSize(buffer), 0);
589 4098 : action = BLK_NEEDS_REDO;
590 : }
591 : else
592 103654 : action = XLogReadBufferForRedo(record, 0, &buffer);
593 107752 : if (action == BLK_NEEDS_REDO)
594 : {
595 : char *tupdata;
596 : char *endptr;
597 : Size len;
598 :
599 : /* Tuples are stored as block data */
600 105714 : tupdata = XLogRecGetBlockData(record, 0, &len);
601 105714 : endptr = tupdata + len;
602 :
603 105714 : page = (Page) BufferGetPage(buffer);
604 :
605 508422 : for (i = 0; i < xlrec->ntuples; i++)
606 : {
607 : OffsetNumber offnum;
608 : xl_multi_insert_tuple *xlhdr;
609 :
610 : /*
611 : * If we're reinitializing the page, the tuples are stored in
612 : * order from FirstOffsetNumber. Otherwise there's an array of
613 : * offsets in the WAL record, and the tuples come after that.
614 : */
615 402708 : if (isinit)
616 199992 : offnum = FirstOffsetNumber + i;
617 : else
618 202716 : offnum = xlrec->offsets[i];
619 402708 : if (PageGetMaxOffsetNumber(page) + 1 < offnum)
620 0 : elog(PANIC, "invalid max offset number");
621 :
622 402708 : xlhdr = (xl_multi_insert_tuple *) SHORTALIGN(tupdata);
623 402708 : tupdata = ((char *) xlhdr) + SizeOfMultiInsertTuple;
624 :
625 402708 : newlen = xlhdr->datalen;
626 : Assert(newlen <= MaxHeapTupleSize);
627 402708 : htup = &tbuf.hdr;
628 402708 : MemSet((char *) htup, 0, SizeofHeapTupleHeader);
629 : /* PG73FORMAT: get bitmap [+ padding] [+ oid] + data */
630 402708 : memcpy((char *) htup + SizeofHeapTupleHeader,
631 : (char *) tupdata,
632 : newlen);
633 402708 : tupdata += newlen;
634 :
635 402708 : newlen += SizeofHeapTupleHeader;
636 402708 : htup->t_infomask2 = xlhdr->t_infomask2;
637 402708 : htup->t_infomask = xlhdr->t_infomask;
638 402708 : htup->t_hoff = xlhdr->t_hoff;
639 402708 : HeapTupleHeaderSetXmin(htup, XLogRecGetXid(record));
640 402708 : HeapTupleHeaderSetCmin(htup, FirstCommandId);
641 402708 : ItemPointerSetBlockNumber(&htup->t_ctid, blkno);
642 402708 : ItemPointerSetOffsetNumber(&htup->t_ctid, offnum);
643 :
644 402708 : offnum = PageAddItem(page, (Item) htup, newlen, offnum, true, true);
645 402708 : if (offnum == InvalidOffsetNumber)
646 0 : elog(PANIC, "failed to add tuple");
647 : }
648 105714 : if (tupdata != endptr)
649 0 : elog(PANIC, "total tuple length mismatch");
650 :
651 105714 : freespace = PageGetHeapFreeSpace(page); /* needed to update FSM below */
652 :
653 105714 : PageSetLSN(page, lsn);
654 :
655 105714 : if (xlrec->flags & XLH_INSERT_ALL_VISIBLE_CLEARED)
656 126 : PageClearAllVisible(page);
657 :
658 : /* XLH_INSERT_ALL_FROZEN_SET implies that all tuples are visible */
659 105714 : if (xlrec->flags & XLH_INSERT_ALL_FROZEN_SET)
660 8 : PageSetAllVisible(page);
661 :
662 105714 : MarkBufferDirty(buffer);
663 : }
664 107752 : if (BufferIsValid(buffer))
665 107752 : UnlockReleaseBuffer(buffer);
666 :
667 : /*
668 : * If the page is running low on free space, update the FSM as well.
669 : * Arbitrarily, our definition of "low" is less than 20%. We can't do much
670 : * better than that without knowing the fill-factor for the table.
671 : *
672 : * XXX: Don't do this if the page was restored from full page image. We
673 : * don't bother to update the FSM in that case, it doesn't need to be
674 : * totally accurate anyway.
675 : */
676 107752 : if (action == BLK_NEEDS_REDO && freespace < BLCKSZ / 5)
677 26806 : XLogRecordPageWithFreeSpace(rlocator, blkno, freespace);
678 107752 : }
679 :
680 : /*
681 : * Replay XLOG_HEAP_UPDATE and XLOG_HEAP_HOT_UPDATE records.
682 : */
683 : static void
684 183704 : heap_xlog_update(XLogReaderState *record, bool hot_update)
685 : {
686 183704 : XLogRecPtr lsn = record->EndRecPtr;
687 183704 : xl_heap_update *xlrec = (xl_heap_update *) XLogRecGetData(record);
688 : RelFileLocator rlocator;
689 : BlockNumber oldblk;
690 : BlockNumber newblk;
691 : ItemPointerData newtid;
692 : Buffer obuffer,
693 : nbuffer;
694 : Page page;
695 : OffsetNumber offnum;
696 183704 : ItemId lp = NULL;
697 : HeapTupleData oldtup;
698 : HeapTupleHeader htup;
699 183704 : uint16 prefixlen = 0,
700 183704 : suffixlen = 0;
701 : char *newp;
702 : union
703 : {
704 : HeapTupleHeaderData hdr;
705 : char data[MaxHeapTupleSize];
706 : } tbuf;
707 : xl_heap_header xlhdr;
708 : uint32 newlen;
709 183704 : Size freespace = 0;
710 : XLogRedoAction oldaction;
711 : XLogRedoAction newaction;
712 :
713 : /* initialize to keep the compiler quiet */
714 183704 : oldtup.t_data = NULL;
715 183704 : oldtup.t_len = 0;
716 :
717 183704 : XLogRecGetBlockTag(record, 0, &rlocator, NULL, &newblk);
718 183704 : if (XLogRecGetBlockTagExtended(record, 1, NULL, NULL, &oldblk, NULL))
719 : {
720 : /* HOT updates are never done across pages */
721 : Assert(!hot_update);
722 : }
723 : else
724 76592 : oldblk = newblk;
725 :
726 183704 : ItemPointerSet(&newtid, newblk, xlrec->new_offnum);
727 :
728 : /*
729 : * The visibility map may need to be fixed even if the heap page is
730 : * already up-to-date.
731 : */
732 183704 : if (xlrec->flags & XLH_UPDATE_OLD_ALL_VISIBLE_CLEARED)
733 : {
734 536 : Relation reln = CreateFakeRelcacheEntry(rlocator);
735 536 : Buffer vmbuffer = InvalidBuffer;
736 :
737 536 : visibilitymap_pin(reln, oldblk, &vmbuffer);
738 536 : visibilitymap_clear(reln, oldblk, vmbuffer, VISIBILITYMAP_VALID_BITS);
739 536 : ReleaseBuffer(vmbuffer);
740 536 : FreeFakeRelcacheEntry(reln);
741 : }
742 :
743 : /*
744 : * In normal operation, it is important to lock the two pages in
745 : * page-number order, to avoid possible deadlocks against other update
746 : * operations going the other way. However, during WAL replay there can
747 : * be no other update happening, so we don't need to worry about that. But
748 : * we *do* need to worry that we don't expose an inconsistent state to Hot
749 : * Standby queries --- so the original page can't be unlocked before we've
750 : * added the new tuple to the new page.
751 : */
752 :
753 : /* Deal with old tuple version */
754 183704 : oldaction = XLogReadBufferForRedo(record, (oldblk == newblk) ? 0 : 1,
755 : &obuffer);
756 183704 : if (oldaction == BLK_NEEDS_REDO)
757 : {
758 183236 : page = BufferGetPage(obuffer);
759 183236 : offnum = xlrec->old_offnum;
760 183236 : if (PageGetMaxOffsetNumber(page) >= offnum)
761 183236 : lp = PageGetItemId(page, offnum);
762 :
763 183236 : if (PageGetMaxOffsetNumber(page) < offnum || !ItemIdIsNormal(lp))
764 0 : elog(PANIC, "invalid lp");
765 :
766 183236 : htup = (HeapTupleHeader) PageGetItem(page, lp);
767 :
768 183236 : oldtup.t_data = htup;
769 183236 : oldtup.t_len = ItemIdGetLength(lp);
770 :
771 183236 : htup->t_infomask &= ~(HEAP_XMAX_BITS | HEAP_MOVED);
772 183236 : htup->t_infomask2 &= ~HEAP_KEYS_UPDATED;
773 183236 : if (hot_update)
774 70514 : HeapTupleHeaderSetHotUpdated(htup);
775 : else
776 112722 : HeapTupleHeaderClearHotUpdated(htup);
777 183236 : fix_infomask_from_infobits(xlrec->old_infobits_set, &htup->t_infomask,
778 : &htup->t_infomask2);
779 183236 : HeapTupleHeaderSetXmax(htup, xlrec->old_xmax);
780 183236 : HeapTupleHeaderSetCmax(htup, FirstCommandId, false);
781 : /* Set forward chain link in t_ctid */
782 183236 : htup->t_ctid = newtid;
783 :
784 : /* Mark the page as a candidate for pruning */
785 183236 : PageSetPrunable(page, XLogRecGetXid(record));
786 :
787 183236 : if (xlrec->flags & XLH_UPDATE_OLD_ALL_VISIBLE_CLEARED)
788 436 : PageClearAllVisible(page);
789 :
790 183236 : PageSetLSN(page, lsn);
791 183236 : MarkBufferDirty(obuffer);
792 : }
793 :
794 : /*
795 : * Read the page the new tuple goes into, if different from old.
796 : */
797 183704 : if (oldblk == newblk)
798 : {
799 76592 : nbuffer = obuffer;
800 76592 : newaction = oldaction;
801 : }
802 107112 : else if (XLogRecGetInfo(record) & XLOG_HEAP_INIT_PAGE)
803 : {
804 1226 : nbuffer = XLogInitBufferForRedo(record, 0);
805 1226 : page = (Page) BufferGetPage(nbuffer);
806 1226 : PageInit(page, BufferGetPageSize(nbuffer), 0);
807 1226 : newaction = BLK_NEEDS_REDO;
808 : }
809 : else
810 105886 : newaction = XLogReadBufferForRedo(record, 0, &nbuffer);
811 :
812 : /*
813 : * The visibility map may need to be fixed even if the heap page is
814 : * already up-to-date.
815 : */
816 183704 : if (xlrec->flags & XLH_UPDATE_NEW_ALL_VISIBLE_CLEARED)
817 : {
818 122 : Relation reln = CreateFakeRelcacheEntry(rlocator);
819 122 : Buffer vmbuffer = InvalidBuffer;
820 :
821 122 : visibilitymap_pin(reln, newblk, &vmbuffer);
822 122 : visibilitymap_clear(reln, newblk, vmbuffer, VISIBILITYMAP_VALID_BITS);
823 122 : ReleaseBuffer(vmbuffer);
824 122 : FreeFakeRelcacheEntry(reln);
825 : }
826 :
827 : /* Deal with new tuple */
828 183704 : if (newaction == BLK_NEEDS_REDO)
829 : {
830 : char *recdata;
831 : char *recdata_end;
832 : Size datalen;
833 : Size tuplen;
834 :
835 183070 : recdata = XLogRecGetBlockData(record, 0, &datalen);
836 183070 : recdata_end = recdata + datalen;
837 :
838 183070 : page = BufferGetPage(nbuffer);
839 :
840 183070 : offnum = xlrec->new_offnum;
841 183070 : if (PageGetMaxOffsetNumber(page) + 1 < offnum)
842 0 : elog(PANIC, "invalid max offset number");
843 :
844 183070 : if (xlrec->flags & XLH_UPDATE_PREFIX_FROM_OLD)
845 : {
846 : Assert(newblk == oldblk);
847 28974 : memcpy(&prefixlen, recdata, sizeof(uint16));
848 28974 : recdata += sizeof(uint16);
849 : }
850 183070 : if (xlrec->flags & XLH_UPDATE_SUFFIX_FROM_OLD)
851 : {
852 : Assert(newblk == oldblk);
853 66042 : memcpy(&suffixlen, recdata, sizeof(uint16));
854 66042 : recdata += sizeof(uint16);
855 : }
856 :
857 183070 : memcpy((char *) &xlhdr, recdata, SizeOfHeapHeader);
858 183070 : recdata += SizeOfHeapHeader;
859 :
860 183070 : tuplen = recdata_end - recdata;
861 : Assert(tuplen <= MaxHeapTupleSize);
862 :
863 183070 : htup = &tbuf.hdr;
864 183070 : MemSet((char *) htup, 0, SizeofHeapTupleHeader);
865 :
866 : /*
867 : * Reconstruct the new tuple using the prefix and/or suffix from the
868 : * old tuple, and the data stored in the WAL record.
869 : */
870 183070 : newp = (char *) htup + SizeofHeapTupleHeader;
871 183070 : if (prefixlen > 0)
872 : {
873 : int len;
874 :
875 : /* copy bitmap [+ padding] [+ oid] from WAL record */
876 28974 : len = xlhdr.t_hoff - SizeofHeapTupleHeader;
877 28974 : memcpy(newp, recdata, len);
878 28974 : recdata += len;
879 28974 : newp += len;
880 :
881 : /* copy prefix from old tuple */
882 28974 : memcpy(newp, (char *) oldtup.t_data + oldtup.t_data->t_hoff, prefixlen);
883 28974 : newp += prefixlen;
884 :
885 : /* copy new tuple data from WAL record */
886 28974 : len = tuplen - (xlhdr.t_hoff - SizeofHeapTupleHeader);
887 28974 : memcpy(newp, recdata, len);
888 28974 : recdata += len;
889 28974 : newp += len;
890 : }
891 : else
892 : {
893 : /*
894 : * copy bitmap [+ padding] [+ oid] + data from record, all in one
895 : * go
896 : */
897 154096 : memcpy(newp, recdata, tuplen);
898 154096 : recdata += tuplen;
899 154096 : newp += tuplen;
900 : }
901 : Assert(recdata == recdata_end);
902 :
903 : /* copy suffix from old tuple */
904 183070 : if (suffixlen > 0)
905 66042 : memcpy(newp, (char *) oldtup.t_data + oldtup.t_len - suffixlen, suffixlen);
906 :
907 183070 : newlen = SizeofHeapTupleHeader + tuplen + prefixlen + suffixlen;
908 183070 : htup->t_infomask2 = xlhdr.t_infomask2;
909 183070 : htup->t_infomask = xlhdr.t_infomask;
910 183070 : htup->t_hoff = xlhdr.t_hoff;
911 :
912 183070 : HeapTupleHeaderSetXmin(htup, XLogRecGetXid(record));
913 183070 : HeapTupleHeaderSetCmin(htup, FirstCommandId);
914 183070 : HeapTupleHeaderSetXmax(htup, xlrec->new_xmax);
915 : /* Make sure there is no forward chain link in t_ctid */
916 183070 : htup->t_ctid = newtid;
917 :
918 183070 : offnum = PageAddItem(page, (Item) htup, newlen, offnum, true, true);
919 183070 : if (offnum == InvalidOffsetNumber)
920 0 : elog(PANIC, "failed to add tuple");
921 :
922 183070 : if (xlrec->flags & XLH_UPDATE_NEW_ALL_VISIBLE_CLEARED)
923 42 : PageClearAllVisible(page);
924 :
925 183070 : freespace = PageGetHeapFreeSpace(page); /* needed to update FSM below */
926 :
927 183070 : PageSetLSN(page, lsn);
928 183070 : MarkBufferDirty(nbuffer);
929 : }
930 :
931 183704 : if (BufferIsValid(nbuffer) && nbuffer != obuffer)
932 107112 : UnlockReleaseBuffer(nbuffer);
933 183704 : if (BufferIsValid(obuffer))
934 183704 : UnlockReleaseBuffer(obuffer);
935 :
936 : /*
937 : * If the new page is running low on free space, update the FSM as well.
938 : * Arbitrarily, our definition of "low" is less than 20%. We can't do much
939 : * better than that without knowing the fill-factor for the table.
940 : *
941 : * However, don't update the FSM on HOT updates, because after crash
942 : * recovery, either the old or the new tuple will certainly be dead and
943 : * prunable. After pruning, the page will have roughly as much free space
944 : * as it did before the update, assuming the new tuple is about the same
945 : * size as the old one.
946 : *
947 : * XXX: Don't do this if the page was restored from full page image. We
948 : * don't bother to update the FSM in that case, it doesn't need to be
949 : * totally accurate anyway.
950 : */
951 183704 : if (newaction == BLK_NEEDS_REDO && !hot_update && freespace < BLCKSZ / 5)
952 22898 : XLogRecordPageWithFreeSpace(rlocator, newblk, freespace);
953 183704 : }
954 :
955 : /*
956 : * Replay XLOG_HEAP_CONFIRM records.
957 : */
958 : static void
959 154 : heap_xlog_confirm(XLogReaderState *record)
960 : {
961 154 : XLogRecPtr lsn = record->EndRecPtr;
962 154 : xl_heap_confirm *xlrec = (xl_heap_confirm *) XLogRecGetData(record);
963 : Buffer buffer;
964 : Page page;
965 : OffsetNumber offnum;
966 154 : ItemId lp = NULL;
967 : HeapTupleHeader htup;
968 :
969 154 : if (XLogReadBufferForRedo(record, 0, &buffer) == BLK_NEEDS_REDO)
970 : {
971 154 : page = BufferGetPage(buffer);
972 :
973 154 : offnum = xlrec->offnum;
974 154 : if (PageGetMaxOffsetNumber(page) >= offnum)
975 154 : lp = PageGetItemId(page, offnum);
976 :
977 154 : if (PageGetMaxOffsetNumber(page) < offnum || !ItemIdIsNormal(lp))
978 0 : elog(PANIC, "invalid lp");
979 :
980 154 : htup = (HeapTupleHeader) PageGetItem(page, lp);
981 :
982 : /*
983 : * Confirm tuple as actually inserted
984 : */
985 154 : ItemPointerSet(&htup->t_ctid, BufferGetBlockNumber(buffer), offnum);
986 :
987 154 : PageSetLSN(page, lsn);
988 154 : MarkBufferDirty(buffer);
989 : }
990 154 : if (BufferIsValid(buffer))
991 154 : UnlockReleaseBuffer(buffer);
992 154 : }
993 :
994 : /*
995 : * Replay XLOG_HEAP_LOCK records.
996 : */
997 : static void
998 108920 : heap_xlog_lock(XLogReaderState *record)
999 : {
1000 108920 : XLogRecPtr lsn = record->EndRecPtr;
1001 108920 : xl_heap_lock *xlrec = (xl_heap_lock *) XLogRecGetData(record);
1002 : Buffer buffer;
1003 : Page page;
1004 : OffsetNumber offnum;
1005 108920 : ItemId lp = NULL;
1006 : HeapTupleHeader htup;
1007 :
1008 : /*
1009 : * The visibility map may need to be fixed even if the heap page is
1010 : * already up-to-date.
1011 : */
1012 108920 : if (xlrec->flags & XLH_LOCK_ALL_FROZEN_CLEARED)
1013 : {
1014 : RelFileLocator rlocator;
1015 62 : Buffer vmbuffer = InvalidBuffer;
1016 : BlockNumber block;
1017 : Relation reln;
1018 :
1019 62 : XLogRecGetBlockTag(record, 0, &rlocator, NULL, &block);
1020 62 : reln = CreateFakeRelcacheEntry(rlocator);
1021 :
1022 62 : visibilitymap_pin(reln, block, &vmbuffer);
1023 62 : visibilitymap_clear(reln, block, vmbuffer, VISIBILITYMAP_ALL_FROZEN);
1024 :
1025 62 : ReleaseBuffer(vmbuffer);
1026 62 : FreeFakeRelcacheEntry(reln);
1027 : }
1028 :
1029 108920 : if (XLogReadBufferForRedo(record, 0, &buffer) == BLK_NEEDS_REDO)
1030 : {
1031 108618 : page = (Page) BufferGetPage(buffer);
1032 :
1033 108618 : offnum = xlrec->offnum;
1034 108618 : if (PageGetMaxOffsetNumber(page) >= offnum)
1035 108618 : lp = PageGetItemId(page, offnum);
1036 :
1037 108618 : if (PageGetMaxOffsetNumber(page) < offnum || !ItemIdIsNormal(lp))
1038 0 : elog(PANIC, "invalid lp");
1039 :
1040 108618 : htup = (HeapTupleHeader) PageGetItem(page, lp);
1041 :
1042 108618 : htup->t_infomask &= ~(HEAP_XMAX_BITS | HEAP_MOVED);
1043 108618 : htup->t_infomask2 &= ~HEAP_KEYS_UPDATED;
1044 108618 : fix_infomask_from_infobits(xlrec->infobits_set, &htup->t_infomask,
1045 : &htup->t_infomask2);
1046 :
1047 : /*
1048 : * Clear relevant update flags, but only if the modified infomask says
1049 : * there's no update.
1050 : */
1051 108618 : if (HEAP_XMAX_IS_LOCKED_ONLY(htup->t_infomask))
1052 : {
1053 108618 : HeapTupleHeaderClearHotUpdated(htup);
1054 : /* Make sure there is no forward chain link in t_ctid */
1055 108618 : ItemPointerSet(&htup->t_ctid,
1056 : BufferGetBlockNumber(buffer),
1057 : offnum);
1058 : }
1059 108618 : HeapTupleHeaderSetXmax(htup, xlrec->xmax);
1060 108618 : HeapTupleHeaderSetCmax(htup, FirstCommandId, false);
1061 108618 : PageSetLSN(page, lsn);
1062 108618 : MarkBufferDirty(buffer);
1063 : }
1064 108920 : if (BufferIsValid(buffer))
1065 108920 : UnlockReleaseBuffer(buffer);
1066 108920 : }
1067 :
1068 : /*
1069 : * Replay XLOG_HEAP2_LOCK_UPDATED records.
1070 : */
1071 : static void
1072 0 : heap_xlog_lock_updated(XLogReaderState *record)
1073 : {
1074 0 : XLogRecPtr lsn = record->EndRecPtr;
1075 : xl_heap_lock_updated *xlrec;
1076 : Buffer buffer;
1077 : Page page;
1078 : OffsetNumber offnum;
1079 0 : ItemId lp = NULL;
1080 : HeapTupleHeader htup;
1081 :
1082 0 : xlrec = (xl_heap_lock_updated *) XLogRecGetData(record);
1083 :
1084 : /*
1085 : * The visibility map may need to be fixed even if the heap page is
1086 : * already up-to-date.
1087 : */
1088 0 : if (xlrec->flags & XLH_LOCK_ALL_FROZEN_CLEARED)
1089 : {
1090 : RelFileLocator rlocator;
1091 0 : Buffer vmbuffer = InvalidBuffer;
1092 : BlockNumber block;
1093 : Relation reln;
1094 :
1095 0 : XLogRecGetBlockTag(record, 0, &rlocator, NULL, &block);
1096 0 : reln = CreateFakeRelcacheEntry(rlocator);
1097 :
1098 0 : visibilitymap_pin(reln, block, &vmbuffer);
1099 0 : visibilitymap_clear(reln, block, vmbuffer, VISIBILITYMAP_ALL_FROZEN);
1100 :
1101 0 : ReleaseBuffer(vmbuffer);
1102 0 : FreeFakeRelcacheEntry(reln);
1103 : }
1104 :
1105 0 : if (XLogReadBufferForRedo(record, 0, &buffer) == BLK_NEEDS_REDO)
1106 : {
1107 0 : page = BufferGetPage(buffer);
1108 :
1109 0 : offnum = xlrec->offnum;
1110 0 : if (PageGetMaxOffsetNumber(page) >= offnum)
1111 0 : lp = PageGetItemId(page, offnum);
1112 :
1113 0 : if (PageGetMaxOffsetNumber(page) < offnum || !ItemIdIsNormal(lp))
1114 0 : elog(PANIC, "invalid lp");
1115 :
1116 0 : htup = (HeapTupleHeader) PageGetItem(page, lp);
1117 :
1118 0 : htup->t_infomask &= ~(HEAP_XMAX_BITS | HEAP_MOVED);
1119 0 : htup->t_infomask2 &= ~HEAP_KEYS_UPDATED;
1120 0 : fix_infomask_from_infobits(xlrec->infobits_set, &htup->t_infomask,
1121 : &htup->t_infomask2);
1122 0 : HeapTupleHeaderSetXmax(htup, xlrec->xmax);
1123 :
1124 0 : PageSetLSN(page, lsn);
1125 0 : MarkBufferDirty(buffer);
1126 : }
1127 0 : if (BufferIsValid(buffer))
1128 0 : UnlockReleaseBuffer(buffer);
1129 0 : }
1130 :
1131 : /*
1132 : * Replay XLOG_HEAP_INPLACE records.
1133 : */
1134 : static void
1135 14048 : heap_xlog_inplace(XLogReaderState *record)
1136 : {
1137 14048 : XLogRecPtr lsn = record->EndRecPtr;
1138 14048 : xl_heap_inplace *xlrec = (xl_heap_inplace *) XLogRecGetData(record);
1139 : Buffer buffer;
1140 : Page page;
1141 : OffsetNumber offnum;
1142 14048 : ItemId lp = NULL;
1143 : HeapTupleHeader htup;
1144 : uint32 oldlen;
1145 : Size newlen;
1146 :
1147 14048 : if (XLogReadBufferForRedo(record, 0, &buffer) == BLK_NEEDS_REDO)
1148 : {
1149 13782 : char *newtup = XLogRecGetBlockData(record, 0, &newlen);
1150 :
1151 13782 : page = BufferGetPage(buffer);
1152 :
1153 13782 : offnum = xlrec->offnum;
1154 13782 : if (PageGetMaxOffsetNumber(page) >= offnum)
1155 13782 : lp = PageGetItemId(page, offnum);
1156 :
1157 13782 : if (PageGetMaxOffsetNumber(page) < offnum || !ItemIdIsNormal(lp))
1158 0 : elog(PANIC, "invalid lp");
1159 :
1160 13782 : htup = (HeapTupleHeader) PageGetItem(page, lp);
1161 :
1162 13782 : oldlen = ItemIdGetLength(lp) - htup->t_hoff;
1163 13782 : if (oldlen != newlen)
1164 0 : elog(PANIC, "wrong tuple length");
1165 :
1166 13782 : memcpy((char *) htup + htup->t_hoff, newtup, newlen);
1167 :
1168 13782 : PageSetLSN(page, lsn);
1169 13782 : MarkBufferDirty(buffer);
1170 : }
1171 14048 : if (BufferIsValid(buffer))
1172 14048 : UnlockReleaseBuffer(buffer);
1173 :
1174 14048 : ProcessCommittedInvalidationMessages(xlrec->msgs,
1175 : xlrec->nmsgs,
1176 14048 : xlrec->relcacheInitFileInval,
1177 : xlrec->dbId,
1178 : xlrec->tsId);
1179 14048 : }
1180 :
1181 : void
1182 3397006 : heap_redo(XLogReaderState *record)
1183 : {
1184 3397006 : uint8 info = XLogRecGetInfo(record) & ~XLR_INFO_MASK;
1185 :
1186 : /*
1187 : * These operations don't overwrite MVCC data so no conflict processing is
1188 : * required. The ones in heap2 rmgr do.
1189 : */
1190 :
1191 3397006 : switch (info & XLOG_HEAP_OPMASK)
1192 : {
1193 2505260 : case XLOG_HEAP_INSERT:
1194 2505260 : heap_xlog_insert(record);
1195 2505260 : break;
1196 584916 : case XLOG_HEAP_DELETE:
1197 584916 : heap_xlog_delete(record);
1198 584916 : break;
1199 112804 : case XLOG_HEAP_UPDATE:
1200 112804 : heap_xlog_update(record, false);
1201 112804 : break;
1202 4 : case XLOG_HEAP_TRUNCATE:
1203 :
1204 : /*
1205 : * TRUNCATE is a no-op because the actions are already logged as
1206 : * SMGR WAL records. TRUNCATE WAL record only exists for logical
1207 : * decoding.
1208 : */
1209 4 : break;
1210 70900 : case XLOG_HEAP_HOT_UPDATE:
1211 70900 : heap_xlog_update(record, true);
1212 70900 : break;
1213 154 : case XLOG_HEAP_CONFIRM:
1214 154 : heap_xlog_confirm(record);
1215 154 : break;
1216 108920 : case XLOG_HEAP_LOCK:
1217 108920 : heap_xlog_lock(record);
1218 108920 : break;
1219 14048 : case XLOG_HEAP_INPLACE:
1220 14048 : heap_xlog_inplace(record);
1221 14048 : break;
1222 0 : default:
1223 0 : elog(PANIC, "heap_redo: unknown op code %u", info);
1224 : }
1225 3397006 : }
1226 :
1227 : void
1228 135412 : heap2_redo(XLogReaderState *record)
1229 : {
1230 135412 : uint8 info = XLogRecGetInfo(record) & ~XLR_INFO_MASK;
1231 :
1232 135412 : switch (info & XLOG_HEAP_OPMASK)
1233 : {
1234 17988 : case XLOG_HEAP2_PRUNE_ON_ACCESS:
1235 : case XLOG_HEAP2_PRUNE_VACUUM_SCAN:
1236 : case XLOG_HEAP2_PRUNE_VACUUM_CLEANUP:
1237 17988 : heap_xlog_prune_freeze(record);
1238 17988 : break;
1239 7920 : case XLOG_HEAP2_VISIBLE:
1240 7920 : heap_xlog_visible(record);
1241 7920 : break;
1242 107752 : case XLOG_HEAP2_MULTI_INSERT:
1243 107752 : heap_xlog_multi_insert(record);
1244 107752 : break;
1245 0 : case XLOG_HEAP2_LOCK_UPDATED:
1246 0 : heap_xlog_lock_updated(record);
1247 0 : break;
1248 1752 : case XLOG_HEAP2_NEW_CID:
1249 :
1250 : /*
1251 : * Nothing to do on a real replay, only used during logical
1252 : * decoding.
1253 : */
1254 1752 : break;
1255 0 : case XLOG_HEAP2_REWRITE:
1256 0 : heap_xlog_logical_rewrite(record);
1257 0 : break;
1258 0 : default:
1259 0 : elog(PANIC, "heap2_redo: unknown op code %u", info);
1260 : }
1261 135412 : }
1262 :
1263 : /*
1264 : * Mask a heap page before performing consistency checks on it.
1265 : */
1266 : void
1267 5619396 : heap_mask(char *pagedata, BlockNumber blkno)
1268 : {
1269 5619396 : Page page = (Page) pagedata;
1270 : OffsetNumber off;
1271 :
1272 5619396 : mask_page_lsn_and_checksum(page);
1273 :
1274 5619396 : mask_page_hint_bits(page);
1275 5619396 : mask_unused_space(page);
1276 :
1277 460697620 : for (off = 1; off <= PageGetMaxOffsetNumber(page); off++)
1278 : {
1279 455078224 : ItemId iid = PageGetItemId(page, off);
1280 : char *page_item;
1281 :
1282 455078224 : page_item = (char *) (page + ItemIdGetOffset(iid));
1283 :
1284 455078224 : if (ItemIdIsNormal(iid))
1285 : {
1286 430928816 : HeapTupleHeader page_htup = (HeapTupleHeader) page_item;
1287 :
1288 : /*
1289 : * If xmin of a tuple is not yet frozen, we should ignore
1290 : * differences in hint bits, since they can be set without
1291 : * emitting WAL.
1292 : */
1293 430928816 : if (!HeapTupleHeaderXminFrozen(page_htup))
1294 419619716 : page_htup->t_infomask &= ~HEAP_XACT_MASK;
1295 : else
1296 : {
1297 : /* Still we need to mask xmax hint bits. */
1298 11309100 : page_htup->t_infomask &= ~HEAP_XMAX_INVALID;
1299 11309100 : page_htup->t_infomask &= ~HEAP_XMAX_COMMITTED;
1300 : }
1301 :
1302 : /*
1303 : * During replay, we set Command Id to FirstCommandId. Hence, mask
1304 : * it. See heap_xlog_insert() for details.
1305 : */
1306 430928816 : page_htup->t_choice.t_heap.t_field3.t_cid = MASK_MARKER;
1307 :
1308 : /*
1309 : * For a speculative tuple, heap_insert() does not set ctid in the
1310 : * caller-passed heap tuple itself, leaving the ctid field to
1311 : * contain a speculative token value - a per-backend monotonically
1312 : * increasing identifier. Besides, it does not WAL-log ctid under
1313 : * any circumstances.
1314 : *
1315 : * During redo, heap_xlog_insert() sets t_ctid to current block
1316 : * number and self offset number. It doesn't care about any
1317 : * speculative insertions on the primary. Hence, we set t_ctid to
1318 : * current block number and self offset number to ignore any
1319 : * inconsistency.
1320 : */
1321 430928816 : if (HeapTupleHeaderIsSpeculative(page_htup))
1322 156 : ItemPointerSet(&page_htup->t_ctid, blkno, off);
1323 :
1324 : /*
1325 : * NB: Not ignoring ctid changes due to the tuple having moved
1326 : * (i.e. HeapTupleHeaderIndicatesMovedPartitions), because that's
1327 : * important information that needs to be in-sync between primary
1328 : * and standby, and thus is WAL logged.
1329 : */
1330 : }
1331 :
1332 : /*
1333 : * Ignore any padding bytes after the tuple, when the length of the
1334 : * item is not MAXALIGNed.
1335 : */
1336 455078224 : if (ItemIdHasStorage(iid))
1337 : {
1338 430928816 : int len = ItemIdGetLength(iid);
1339 430928816 : int padlen = MAXALIGN(len) - len;
1340 :
1341 430928816 : if (padlen > 0)
1342 228159808 : memset(page_item + len, MASK_MARKER, padlen);
1343 : }
1344 : }
1345 5619396 : }
|