Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * nbtpage.c
4 : * BTree-specific page management code for the Postgres btree access
5 : * method.
6 : *
7 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
8 : * Portions Copyright (c) 1994, Regents of the University of California
9 : *
10 : *
11 : * IDENTIFICATION
12 : * src/backend/access/nbtree/nbtpage.c
13 : *
14 : * NOTES
15 : * Postgres btree pages look like ordinary relation pages. The opaque
16 : * data at high addresses includes pointers to left and right siblings
17 : * and flag data describing page state. The first page in a btree, page
18 : * zero, is special -- it stores meta-information describing the tree.
19 : * Pages one and higher store the actual tree data.
20 : *
21 : *-------------------------------------------------------------------------
22 : */
23 : #include "postgres.h"
24 :
25 : #include "access/nbtree.h"
26 : #include "access/nbtxlog.h"
27 : #include "access/tableam.h"
28 : #include "access/transam.h"
29 : #include "access/xlog.h"
30 : #include "access/xloginsert.h"
31 : #include "common/int.h"
32 : #include "miscadmin.h"
33 : #include "storage/indexfsm.h"
34 : #include "storage/predicate.h"
35 : #include "storage/procarray.h"
36 : #include "utils/memdebug.h"
37 : #include "utils/memutils.h"
38 : #include "utils/snapmgr.h"
39 :
40 : static BTMetaPageData *_bt_getmeta(Relation rel, Buffer metabuf);
41 : static void _bt_delitems_delete(Relation rel, Buffer buf,
42 : TransactionId snapshotConflictHorizon,
43 : bool isCatalogRel,
44 : OffsetNumber *deletable, int ndeletable,
45 : BTVacuumPosting *updatable, int nupdatable);
46 : static char *_bt_delitems_update(BTVacuumPosting *updatable, int nupdatable,
47 : OffsetNumber *updatedoffsets,
48 : Size *updatedbuflen, bool needswal);
49 : static bool _bt_mark_page_halfdead(Relation rel, Relation heaprel,
50 : Buffer leafbuf, BTStack stack);
51 : static bool _bt_unlink_halfdead_page(Relation rel, Buffer leafbuf,
52 : BlockNumber scanblkno,
53 : bool *rightsib_empty,
54 : BTVacState *vstate);
55 : static bool _bt_lock_subtree_parent(Relation rel, Relation heaprel,
56 : BlockNumber child, BTStack stack,
57 : Buffer *subtreeparent, OffsetNumber *poffset,
58 : BlockNumber *topparent,
59 : BlockNumber *topparentrightsib);
60 : static void _bt_pendingfsm_add(BTVacState *vstate, BlockNumber target,
61 : FullTransactionId safexid);
62 :
63 : /*
64 : * _bt_initmetapage() -- Fill a page buffer with a correct metapage image
65 : */
66 : void
67 47816 : _bt_initmetapage(Page page, BlockNumber rootbknum, uint32 level,
68 : bool allequalimage)
69 : {
70 : BTMetaPageData *metad;
71 : BTPageOpaque metaopaque;
72 :
73 47816 : _bt_pageinit(page, BLCKSZ);
74 :
75 47816 : metad = BTPageGetMeta(page);
76 47816 : metad->btm_magic = BTREE_MAGIC;
77 47816 : metad->btm_version = BTREE_VERSION;
78 47816 : metad->btm_root = rootbknum;
79 47816 : metad->btm_level = level;
80 47816 : metad->btm_fastroot = rootbknum;
81 47816 : metad->btm_fastlevel = level;
82 47816 : metad->btm_last_cleanup_num_delpages = 0;
83 47816 : metad->btm_last_cleanup_num_heap_tuples = -1.0;
84 47816 : metad->btm_allequalimage = allequalimage;
85 :
86 47816 : metaopaque = BTPageGetOpaque(page);
87 47816 : metaopaque->btpo_flags = BTP_META;
88 :
89 : /*
90 : * Set pd_lower just past the end of the metadata. This is essential,
91 : * because without doing so, metadata will be lost if xlog.c compresses
92 : * the page.
93 : */
94 47816 : ((PageHeader) page)->pd_lower =
95 47816 : ((char *) metad + sizeof(BTMetaPageData)) - (char *) page;
96 47816 : }
97 :
98 : /*
99 : * _bt_upgrademetapage() -- Upgrade a meta-page from an old format to version
100 : * 3, the last version that can be updated without broadly affecting
101 : * on-disk compatibility. (A REINDEX is required to upgrade to v4.)
102 : *
103 : * This routine does purely in-memory image upgrade. Caller is
104 : * responsible for locking, WAL-logging etc.
105 : */
106 : void
107 0 : _bt_upgrademetapage(Page page)
108 : {
109 : BTMetaPageData *metad;
110 : BTPageOpaque metaopaque PG_USED_FOR_ASSERTS_ONLY;
111 :
112 0 : metad = BTPageGetMeta(page);
113 0 : metaopaque = BTPageGetOpaque(page);
114 :
115 : /* It must be really a meta page of upgradable version */
116 : Assert(metaopaque->btpo_flags & BTP_META);
117 : Assert(metad->btm_version < BTREE_NOVAC_VERSION);
118 : Assert(metad->btm_version >= BTREE_MIN_VERSION);
119 :
120 : /* Set version number and fill extra fields added into version 3 */
121 0 : metad->btm_version = BTREE_NOVAC_VERSION;
122 0 : metad->btm_last_cleanup_num_delpages = 0;
123 0 : metad->btm_last_cleanup_num_heap_tuples = -1.0;
124 : /* Only a REINDEX can set this field */
125 : Assert(!metad->btm_allequalimage);
126 0 : metad->btm_allequalimage = false;
127 :
128 : /* Adjust pd_lower (see _bt_initmetapage() for details) */
129 0 : ((PageHeader) page)->pd_lower =
130 0 : ((char *) metad + sizeof(BTMetaPageData)) - (char *) page;
131 0 : }
132 :
133 : /*
134 : * Get metadata from share-locked buffer containing metapage, while performing
135 : * standard sanity checks.
136 : *
137 : * Callers that cache data returned here in local cache should note that an
138 : * on-the-fly upgrade using _bt_upgrademetapage() can change the version field
139 : * and BTREE_NOVAC_VERSION specific fields without invalidating local cache.
140 : */
141 : static BTMetaPageData *
142 1688032 : _bt_getmeta(Relation rel, Buffer metabuf)
143 : {
144 : Page metapg;
145 : BTPageOpaque metaopaque;
146 : BTMetaPageData *metad;
147 :
148 1688032 : metapg = BufferGetPage(metabuf);
149 1688032 : metaopaque = BTPageGetOpaque(metapg);
150 1688032 : metad = BTPageGetMeta(metapg);
151 :
152 : /* sanity-check the metapage */
153 1688032 : if (!P_ISMETA(metaopaque) ||
154 1688032 : metad->btm_magic != BTREE_MAGIC)
155 0 : ereport(ERROR,
156 : (errcode(ERRCODE_INDEX_CORRUPTED),
157 : errmsg("index \"%s\" is not a btree",
158 : RelationGetRelationName(rel))));
159 :
160 1688032 : if (metad->btm_version < BTREE_MIN_VERSION ||
161 1688032 : metad->btm_version > BTREE_VERSION)
162 0 : ereport(ERROR,
163 : (errcode(ERRCODE_INDEX_CORRUPTED),
164 : errmsg("version mismatch in index \"%s\": file version %d, "
165 : "current version %d, minimal supported version %d",
166 : RelationGetRelationName(rel),
167 : metad->btm_version, BTREE_VERSION, BTREE_MIN_VERSION)));
168 :
169 1688032 : return metad;
170 : }
171 :
172 : /*
173 : * _bt_vacuum_needs_cleanup() -- Checks if index needs cleanup
174 : *
175 : * Called by btvacuumcleanup when btbulkdelete was never called because no
176 : * index tuples needed to be deleted.
177 : */
178 : bool
179 138470 : _bt_vacuum_needs_cleanup(Relation rel)
180 : {
181 : Buffer metabuf;
182 : Page metapg;
183 : BTMetaPageData *metad;
184 : uint32 btm_version;
185 : BlockNumber prev_num_delpages;
186 :
187 : /*
188 : * Copy details from metapage to local variables quickly.
189 : *
190 : * Note that we deliberately avoid using cached version of metapage here.
191 : */
192 138470 : metabuf = _bt_getbuf(rel, BTREE_METAPAGE, BT_READ);
193 138470 : metapg = BufferGetPage(metabuf);
194 138470 : metad = BTPageGetMeta(metapg);
195 138470 : btm_version = metad->btm_version;
196 :
197 138470 : if (btm_version < BTREE_NOVAC_VERSION)
198 : {
199 : /*
200 : * Metapage needs to be dynamically upgraded to store fields that are
201 : * only present when btm_version >= BTREE_NOVAC_VERSION
202 : */
203 0 : _bt_relbuf(rel, metabuf);
204 0 : return true;
205 : }
206 :
207 138470 : prev_num_delpages = metad->btm_last_cleanup_num_delpages;
208 138470 : _bt_relbuf(rel, metabuf);
209 :
210 : /*
211 : * Trigger cleanup in rare cases where prev_num_delpages exceeds 5% of the
212 : * total size of the index. We can reasonably expect (though are not
213 : * guaranteed) to be able to recycle this many pages if we decide to do a
214 : * btvacuumscan call during the ongoing btvacuumcleanup. For further
215 : * details see the nbtree/README section on placing deleted pages in the
216 : * FSM.
217 : */
218 138470 : if (prev_num_delpages > 0 &&
219 12 : prev_num_delpages > RelationGetNumberOfBlocks(rel) / 20)
220 12 : return true;
221 :
222 138458 : return false;
223 : }
224 :
225 : /*
226 : * _bt_set_cleanup_info() -- Update metapage for btvacuumcleanup.
227 : *
228 : * Called at the end of btvacuumcleanup, when num_delpages value has been
229 : * finalized.
230 : */
231 : void
232 1956 : _bt_set_cleanup_info(Relation rel, BlockNumber num_delpages)
233 : {
234 : Buffer metabuf;
235 : Page metapg;
236 : BTMetaPageData *metad;
237 :
238 : /*
239 : * On-disk compatibility note: The btm_last_cleanup_num_delpages metapage
240 : * field started out as a TransactionId field called btm_oldest_btpo_xact.
241 : * Both "versions" are just uint32 fields. It was convenient to repurpose
242 : * the field when we began to use 64-bit XIDs in deleted pages.
243 : *
244 : * It's possible that a pg_upgrade'd database will contain an XID value in
245 : * what is now recognized as the metapage's btm_last_cleanup_num_delpages
246 : * field. _bt_vacuum_needs_cleanup() may even believe that this value
247 : * indicates that there are lots of pages that it needs to recycle, when
248 : * in reality there are only one or two. The worst that can happen is
249 : * that there will be a call to btvacuumscan a little earlier, which will
250 : * set btm_last_cleanup_num_delpages to a sane value when we're called.
251 : *
252 : * Note also that the metapage's btm_last_cleanup_num_heap_tuples field is
253 : * no longer used as of PostgreSQL 14. We set it to -1.0 on rewrite, just
254 : * to be consistent.
255 : */
256 1956 : metabuf = _bt_getbuf(rel, BTREE_METAPAGE, BT_READ);
257 1956 : metapg = BufferGetPage(metabuf);
258 1956 : metad = BTPageGetMeta(metapg);
259 :
260 : /* Don't miss chance to upgrade index/metapage when BTREE_MIN_VERSION */
261 1956 : if (metad->btm_version >= BTREE_NOVAC_VERSION &&
262 1956 : metad->btm_last_cleanup_num_delpages == num_delpages)
263 : {
264 : /* Usually means index continues to have num_delpages of 0 */
265 1826 : _bt_relbuf(rel, metabuf);
266 1826 : return;
267 : }
268 :
269 : /* trade in our read lock for a write lock */
270 130 : _bt_unlockbuf(rel, metabuf);
271 130 : _bt_lockbuf(rel, metabuf, BT_WRITE);
272 :
273 130 : START_CRIT_SECTION();
274 :
275 : /* upgrade meta-page if needed */
276 130 : if (metad->btm_version < BTREE_NOVAC_VERSION)
277 0 : _bt_upgrademetapage(metapg);
278 :
279 : /* update cleanup-related information */
280 130 : metad->btm_last_cleanup_num_delpages = num_delpages;
281 130 : metad->btm_last_cleanup_num_heap_tuples = -1.0;
282 130 : MarkBufferDirty(metabuf);
283 :
284 : /* write wal record if needed */
285 130 : if (RelationNeedsWAL(rel))
286 : {
287 : xl_btree_metadata md;
288 : XLogRecPtr recptr;
289 :
290 130 : XLogBeginInsert();
291 130 : XLogRegisterBuffer(0, metabuf, REGBUF_WILL_INIT | REGBUF_STANDARD);
292 :
293 : Assert(metad->btm_version >= BTREE_NOVAC_VERSION);
294 130 : md.version = metad->btm_version;
295 130 : md.root = metad->btm_root;
296 130 : md.level = metad->btm_level;
297 130 : md.fastroot = metad->btm_fastroot;
298 130 : md.fastlevel = metad->btm_fastlevel;
299 130 : md.last_cleanup_num_delpages = num_delpages;
300 130 : md.allequalimage = metad->btm_allequalimage;
301 :
302 130 : XLogRegisterBufData(0, (char *) &md, sizeof(xl_btree_metadata));
303 :
304 130 : recptr = XLogInsert(RM_BTREE_ID, XLOG_BTREE_META_CLEANUP);
305 :
306 130 : PageSetLSN(metapg, recptr);
307 : }
308 :
309 130 : END_CRIT_SECTION();
310 :
311 130 : _bt_relbuf(rel, metabuf);
312 : }
313 :
314 : /*
315 : * _bt_getroot() -- Get the root page of the btree.
316 : *
317 : * Since the root page can move around the btree file, we have to read
318 : * its location from the metadata page, and then read the root page
319 : * itself. If no root page exists yet, we have to create one.
320 : *
321 : * The access type parameter (BT_READ or BT_WRITE) controls whether
322 : * a new root page will be created or not. If access = BT_READ,
323 : * and no root page exists, we just return InvalidBuffer. For
324 : * BT_WRITE, we try to create the root page if it doesn't exist.
325 : * NOTE that the returned root page will have only a read lock set
326 : * on it even if access = BT_WRITE!
327 : *
328 : * If access = BT_WRITE, heaprel must be set; otherwise caller can just
329 : * pass NULL. See _bt_allocbuf for an explanation.
330 : *
331 : * The returned page is not necessarily the true root --- it could be
332 : * a "fast root" (a page that is alone in its level due to deletions).
333 : * Also, if the root page is split while we are "in flight" to it,
334 : * what we will return is the old root, which is now just the leftmost
335 : * page on a probably-not-very-wide level. For most purposes this is
336 : * as good as or better than the true root, so we do not bother to
337 : * insist on finding the true root. We do, however, guarantee to
338 : * return a live (not deleted or half-dead) page.
339 : *
340 : * On successful return, the root page is pinned and read-locked.
341 : * The metadata page is not locked or pinned on exit.
342 : */
343 : Buffer
344 20327338 : _bt_getroot(Relation rel, Relation heaprel, int access)
345 : {
346 : Buffer metabuf;
347 : Buffer rootbuf;
348 : Page rootpage;
349 : BTPageOpaque rootopaque;
350 : BlockNumber rootblkno;
351 : uint32 rootlevel;
352 : BTMetaPageData *metad;
353 :
354 : Assert(access == BT_READ || heaprel != NULL);
355 :
356 : /*
357 : * Try to use previously-cached metapage data to find the root. This
358 : * normally saves one buffer access per index search, which is a very
359 : * helpful savings in bufmgr traffic and hence contention.
360 : */
361 20327338 : if (rel->rd_amcache != NULL)
362 : {
363 19800138 : metad = (BTMetaPageData *) rel->rd_amcache;
364 : /* We shouldn't have cached it if any of these fail */
365 : Assert(metad->btm_magic == BTREE_MAGIC);
366 : Assert(metad->btm_version >= BTREE_MIN_VERSION);
367 : Assert(metad->btm_version <= BTREE_VERSION);
368 : Assert(!metad->btm_allequalimage ||
369 : metad->btm_version > BTREE_NOVAC_VERSION);
370 : Assert(metad->btm_root != P_NONE);
371 :
372 19800138 : rootblkno = metad->btm_fastroot;
373 : Assert(rootblkno != P_NONE);
374 19800138 : rootlevel = metad->btm_fastlevel;
375 :
376 19800138 : rootbuf = _bt_getbuf(rel, rootblkno, BT_READ);
377 19800138 : rootpage = BufferGetPage(rootbuf);
378 19800138 : rootopaque = BTPageGetOpaque(rootpage);
379 :
380 : /*
381 : * Since the cache might be stale, we check the page more carefully
382 : * here than normal. We *must* check that it's not deleted. If it's
383 : * not alone on its level, then we reject too --- this may be overly
384 : * paranoid but better safe than sorry. Note we don't check P_ISROOT,
385 : * because that's not set in a "fast root".
386 : */
387 19800138 : if (!P_IGNORE(rootopaque) &&
388 19800138 : rootopaque->btpo_level == rootlevel &&
389 19800138 : P_LEFTMOST(rootopaque) &&
390 19800138 : P_RIGHTMOST(rootopaque))
391 : {
392 : /* OK, accept cached page as the root */
393 19798690 : return rootbuf;
394 : }
395 1448 : _bt_relbuf(rel, rootbuf);
396 : /* Cache is stale, throw it away */
397 1448 : if (rel->rd_amcache)
398 1448 : pfree(rel->rd_amcache);
399 1448 : rel->rd_amcache = NULL;
400 : }
401 :
402 528648 : metabuf = _bt_getbuf(rel, BTREE_METAPAGE, BT_READ);
403 528648 : metad = _bt_getmeta(rel, metabuf);
404 :
405 : /* if no root page initialized yet, do it */
406 528648 : if (metad->btm_root == P_NONE)
407 : {
408 : Page metapg;
409 :
410 : /* If access = BT_READ, caller doesn't want us to create root yet */
411 526718 : if (access == BT_READ)
412 : {
413 515158 : _bt_relbuf(rel, metabuf);
414 515158 : return InvalidBuffer;
415 : }
416 :
417 : /* trade in our read lock for a write lock */
418 11560 : _bt_unlockbuf(rel, metabuf);
419 11560 : _bt_lockbuf(rel, metabuf, BT_WRITE);
420 :
421 : /*
422 : * Race condition: if someone else initialized the metadata between
423 : * the time we released the read lock and acquired the write lock, we
424 : * must avoid doing it again.
425 : */
426 11560 : if (metad->btm_root != P_NONE)
427 : {
428 : /*
429 : * Metadata initialized by someone else. In order to guarantee no
430 : * deadlocks, we have to release the metadata page and start all
431 : * over again. (Is that really true? But it's hardly worth trying
432 : * to optimize this case.)
433 : */
434 0 : _bt_relbuf(rel, metabuf);
435 0 : return _bt_getroot(rel, heaprel, access);
436 : }
437 :
438 : /*
439 : * Get, initialize, write, and leave a lock of the appropriate type on
440 : * the new root page. Since this is the first page in the tree, it's
441 : * a leaf as well as the root.
442 : */
443 11560 : rootbuf = _bt_allocbuf(rel, heaprel);
444 11560 : rootblkno = BufferGetBlockNumber(rootbuf);
445 11560 : rootpage = BufferGetPage(rootbuf);
446 11560 : rootopaque = BTPageGetOpaque(rootpage);
447 11560 : rootopaque->btpo_prev = rootopaque->btpo_next = P_NONE;
448 11560 : rootopaque->btpo_flags = (BTP_LEAF | BTP_ROOT);
449 11560 : rootopaque->btpo_level = 0;
450 11560 : rootopaque->btpo_cycleid = 0;
451 : /* Get raw page pointer for metapage */
452 11560 : metapg = BufferGetPage(metabuf);
453 :
454 : /* NO ELOG(ERROR) till meta is updated */
455 11560 : START_CRIT_SECTION();
456 :
457 : /* upgrade metapage if needed */
458 11560 : if (metad->btm_version < BTREE_NOVAC_VERSION)
459 0 : _bt_upgrademetapage(metapg);
460 :
461 11560 : metad->btm_root = rootblkno;
462 11560 : metad->btm_level = 0;
463 11560 : metad->btm_fastroot = rootblkno;
464 11560 : metad->btm_fastlevel = 0;
465 11560 : metad->btm_last_cleanup_num_delpages = 0;
466 11560 : metad->btm_last_cleanup_num_heap_tuples = -1.0;
467 :
468 11560 : MarkBufferDirty(rootbuf);
469 11560 : MarkBufferDirty(metabuf);
470 :
471 : /* XLOG stuff */
472 11560 : if (RelationNeedsWAL(rel))
473 : {
474 : xl_btree_newroot xlrec;
475 : XLogRecPtr recptr;
476 : xl_btree_metadata md;
477 :
478 11102 : XLogBeginInsert();
479 11102 : XLogRegisterBuffer(0, rootbuf, REGBUF_WILL_INIT);
480 11102 : XLogRegisterBuffer(2, metabuf, REGBUF_WILL_INIT | REGBUF_STANDARD);
481 :
482 : Assert(metad->btm_version >= BTREE_NOVAC_VERSION);
483 11102 : md.version = metad->btm_version;
484 11102 : md.root = rootblkno;
485 11102 : md.level = 0;
486 11102 : md.fastroot = rootblkno;
487 11102 : md.fastlevel = 0;
488 11102 : md.last_cleanup_num_delpages = 0;
489 11102 : md.allequalimage = metad->btm_allequalimage;
490 :
491 11102 : XLogRegisterBufData(2, (char *) &md, sizeof(xl_btree_metadata));
492 :
493 11102 : xlrec.rootblk = rootblkno;
494 11102 : xlrec.level = 0;
495 :
496 11102 : XLogRegisterData((char *) &xlrec, SizeOfBtreeNewroot);
497 :
498 11102 : recptr = XLogInsert(RM_BTREE_ID, XLOG_BTREE_NEWROOT);
499 :
500 11102 : PageSetLSN(rootpage, recptr);
501 11102 : PageSetLSN(metapg, recptr);
502 : }
503 :
504 11560 : END_CRIT_SECTION();
505 :
506 : /*
507 : * swap root write lock for read lock. There is no danger of anyone
508 : * else accessing the new root page while it's unlocked, since no one
509 : * else knows where it is yet.
510 : */
511 11560 : _bt_unlockbuf(rel, rootbuf);
512 11560 : _bt_lockbuf(rel, rootbuf, BT_READ);
513 :
514 : /* okay, metadata is correct, release lock on it without caching */
515 11560 : _bt_relbuf(rel, metabuf);
516 : }
517 : else
518 : {
519 1930 : rootblkno = metad->btm_fastroot;
520 : Assert(rootblkno != P_NONE);
521 1930 : rootlevel = metad->btm_fastlevel;
522 :
523 : /*
524 : * Cache the metapage data for next time
525 : */
526 1930 : rel->rd_amcache = MemoryContextAlloc(rel->rd_indexcxt,
527 : sizeof(BTMetaPageData));
528 1930 : memcpy(rel->rd_amcache, metad, sizeof(BTMetaPageData));
529 :
530 : /*
531 : * We are done with the metapage; arrange to release it via first
532 : * _bt_relandgetbuf call
533 : */
534 1930 : rootbuf = metabuf;
535 :
536 : for (;;)
537 : {
538 1930 : rootbuf = _bt_relandgetbuf(rel, rootbuf, rootblkno, BT_READ);
539 1930 : rootpage = BufferGetPage(rootbuf);
540 1930 : rootopaque = BTPageGetOpaque(rootpage);
541 :
542 1930 : if (!P_IGNORE(rootopaque))
543 1930 : break;
544 :
545 : /* it's dead, Jim. step right one page */
546 0 : if (P_RIGHTMOST(rootopaque))
547 0 : elog(ERROR, "no live root page found in index \"%s\"",
548 : RelationGetRelationName(rel));
549 0 : rootblkno = rootopaque->btpo_next;
550 : }
551 :
552 1930 : if (rootopaque->btpo_level != rootlevel)
553 0 : elog(ERROR, "root page %u of index \"%s\" has level %u, expected %u",
554 : rootblkno, RelationGetRelationName(rel),
555 : rootopaque->btpo_level, rootlevel);
556 : }
557 :
558 : /*
559 : * By here, we have a pin and read lock on the root page, and no lock set
560 : * on the metadata page. Return the root page's buffer.
561 : */
562 13490 : return rootbuf;
563 : }
564 :
565 : /*
566 : * _bt_gettrueroot() -- Get the true root page of the btree.
567 : *
568 : * This is the same as the BT_READ case of _bt_getroot(), except
569 : * we follow the true-root link not the fast-root link.
570 : *
571 : * By the time we acquire lock on the root page, it might have been split and
572 : * not be the true root anymore. This is okay for the present uses of this
573 : * routine; we only really need to be able to move up at least one tree level
574 : * from whatever non-root page we were at. If we ever do need to lock the
575 : * one true root page, we could loop here, re-reading the metapage on each
576 : * failure. (Note that it wouldn't do to hold the lock on the metapage while
577 : * moving to the root --- that'd deadlock against any concurrent root split.)
578 : */
579 : Buffer
580 24 : _bt_gettrueroot(Relation rel)
581 : {
582 : Buffer metabuf;
583 : Page metapg;
584 : BTPageOpaque metaopaque;
585 : Buffer rootbuf;
586 : Page rootpage;
587 : BTPageOpaque rootopaque;
588 : BlockNumber rootblkno;
589 : uint32 rootlevel;
590 : BTMetaPageData *metad;
591 :
592 : /*
593 : * We don't try to use cached metapage data here, since (a) this path is
594 : * not performance-critical, and (b) if we are here it suggests our cache
595 : * is out-of-date anyway. In light of point (b), it's probably safest to
596 : * actively flush any cached metapage info.
597 : */
598 24 : if (rel->rd_amcache)
599 24 : pfree(rel->rd_amcache);
600 24 : rel->rd_amcache = NULL;
601 :
602 24 : metabuf = _bt_getbuf(rel, BTREE_METAPAGE, BT_READ);
603 24 : metapg = BufferGetPage(metabuf);
604 24 : metaopaque = BTPageGetOpaque(metapg);
605 24 : metad = BTPageGetMeta(metapg);
606 :
607 24 : if (!P_ISMETA(metaopaque) ||
608 24 : metad->btm_magic != BTREE_MAGIC)
609 0 : ereport(ERROR,
610 : (errcode(ERRCODE_INDEX_CORRUPTED),
611 : errmsg("index \"%s\" is not a btree",
612 : RelationGetRelationName(rel))));
613 :
614 24 : if (metad->btm_version < BTREE_MIN_VERSION ||
615 24 : metad->btm_version > BTREE_VERSION)
616 0 : ereport(ERROR,
617 : (errcode(ERRCODE_INDEX_CORRUPTED),
618 : errmsg("version mismatch in index \"%s\": file version %d, "
619 : "current version %d, minimal supported version %d",
620 : RelationGetRelationName(rel),
621 : metad->btm_version, BTREE_VERSION, BTREE_MIN_VERSION)));
622 :
623 : /* if no root page initialized yet, fail */
624 24 : if (metad->btm_root == P_NONE)
625 : {
626 0 : _bt_relbuf(rel, metabuf);
627 0 : return InvalidBuffer;
628 : }
629 :
630 24 : rootblkno = metad->btm_root;
631 24 : rootlevel = metad->btm_level;
632 :
633 : /*
634 : * We are done with the metapage; arrange to release it via first
635 : * _bt_relandgetbuf call
636 : */
637 24 : rootbuf = metabuf;
638 :
639 : for (;;)
640 : {
641 24 : rootbuf = _bt_relandgetbuf(rel, rootbuf, rootblkno, BT_READ);
642 24 : rootpage = BufferGetPage(rootbuf);
643 24 : rootopaque = BTPageGetOpaque(rootpage);
644 :
645 24 : if (!P_IGNORE(rootopaque))
646 24 : break;
647 :
648 : /* it's dead, Jim. step right one page */
649 0 : if (P_RIGHTMOST(rootopaque))
650 0 : elog(ERROR, "no live root page found in index \"%s\"",
651 : RelationGetRelationName(rel));
652 0 : rootblkno = rootopaque->btpo_next;
653 : }
654 :
655 24 : if (rootopaque->btpo_level != rootlevel)
656 0 : elog(ERROR, "root page %u of index \"%s\" has level %u, expected %u",
657 : rootblkno, RelationGetRelationName(rel),
658 : rootopaque->btpo_level, rootlevel);
659 :
660 24 : return rootbuf;
661 : }
662 :
663 : /*
664 : * _bt_getrootheight() -- Get the height of the btree search tree.
665 : *
666 : * We return the level (counting from zero) of the current fast root.
667 : * This represents the number of tree levels we'd have to descend through
668 : * to start any btree index search.
669 : *
670 : * This is used by the planner for cost-estimation purposes. Since it's
671 : * only an estimate, slightly-stale data is fine, hence we don't worry
672 : * about updating previously cached data.
673 : */
674 : int
675 4319416 : _bt_getrootheight(Relation rel)
676 : {
677 : BTMetaPageData *metad;
678 :
679 4319416 : if (rel->rd_amcache == NULL)
680 : {
681 : Buffer metabuf;
682 :
683 68020 : metabuf = _bt_getbuf(rel, BTREE_METAPAGE, BT_READ);
684 68020 : metad = _bt_getmeta(rel, metabuf);
685 :
686 : /*
687 : * If there's no root page yet, _bt_getroot() doesn't expect a cache
688 : * to be made, so just stop here and report the index height is zero.
689 : * (XXX perhaps _bt_getroot() should be changed to allow this case.)
690 : */
691 68020 : if (metad->btm_root == P_NONE)
692 : {
693 29206 : _bt_relbuf(rel, metabuf);
694 29206 : return 0;
695 : }
696 :
697 : /*
698 : * Cache the metapage data for next time
699 : */
700 38814 : rel->rd_amcache = MemoryContextAlloc(rel->rd_indexcxt,
701 : sizeof(BTMetaPageData));
702 38814 : memcpy(rel->rd_amcache, metad, sizeof(BTMetaPageData));
703 38814 : _bt_relbuf(rel, metabuf);
704 : }
705 :
706 : /* Get cached page */
707 4290210 : metad = (BTMetaPageData *) rel->rd_amcache;
708 : /* We shouldn't have cached it if any of these fail */
709 : Assert(metad->btm_magic == BTREE_MAGIC);
710 : Assert(metad->btm_version >= BTREE_MIN_VERSION);
711 : Assert(metad->btm_version <= BTREE_VERSION);
712 : Assert(!metad->btm_allequalimage ||
713 : metad->btm_version > BTREE_NOVAC_VERSION);
714 : Assert(metad->btm_fastroot != P_NONE);
715 :
716 4290210 : return metad->btm_fastlevel;
717 : }
718 :
719 : /*
720 : * _bt_metaversion() -- Get version/status info from metapage.
721 : *
722 : * Sets caller's *heapkeyspace and *allequalimage arguments using data
723 : * from the B-Tree metapage (could be locally-cached version). This
724 : * information needs to be stashed in insertion scankey, so we provide a
725 : * single function that fetches both at once.
726 : *
727 : * This is used to determine the rules that must be used to descend a
728 : * btree. Version 4 indexes treat heap TID as a tiebreaker attribute.
729 : * pg_upgrade'd version 3 indexes need extra steps to preserve reasonable
730 : * performance when inserting a new BTScanInsert-wise duplicate tuple
731 : * among many leaf pages already full of such duplicates.
732 : *
733 : * Also sets allequalimage field, which indicates whether or not it is
734 : * safe to apply deduplication. We rely on the assumption that
735 : * btm_allequalimage will be zero'ed on heapkeyspace indexes that were
736 : * pg_upgrade'd from Postgres 12.
737 : */
738 : void
739 24354100 : _bt_metaversion(Relation rel, bool *heapkeyspace, bool *allequalimage)
740 : {
741 : BTMetaPageData *metad;
742 :
743 24354100 : if (rel->rd_amcache == NULL)
744 : {
745 : Buffer metabuf;
746 :
747 1091364 : metabuf = _bt_getbuf(rel, BTREE_METAPAGE, BT_READ);
748 1091364 : metad = _bt_getmeta(rel, metabuf);
749 :
750 : /*
751 : * If there's no root page yet, _bt_getroot() doesn't expect a cache
752 : * to be made, so just stop here. (XXX perhaps _bt_getroot() should
753 : * be changed to allow this case.)
754 : */
755 1091364 : if (metad->btm_root == P_NONE)
756 : {
757 518480 : *heapkeyspace = metad->btm_version > BTREE_NOVAC_VERSION;
758 518480 : *allequalimage = metad->btm_allequalimage;
759 :
760 518480 : _bt_relbuf(rel, metabuf);
761 518480 : return;
762 : }
763 :
764 : /*
765 : * Cache the metapage data for next time
766 : *
767 : * An on-the-fly version upgrade performed by _bt_upgrademetapage()
768 : * can change the nbtree version for an index without invalidating any
769 : * local cache. This is okay because it can only happen when moving
770 : * from version 2 to version 3, both of which are !heapkeyspace
771 : * versions.
772 : */
773 572884 : rel->rd_amcache = MemoryContextAlloc(rel->rd_indexcxt,
774 : sizeof(BTMetaPageData));
775 572884 : memcpy(rel->rd_amcache, metad, sizeof(BTMetaPageData));
776 572884 : _bt_relbuf(rel, metabuf);
777 : }
778 :
779 : /* Get cached page */
780 23835620 : metad = (BTMetaPageData *) rel->rd_amcache;
781 : /* We shouldn't have cached it if any of these fail */
782 : Assert(metad->btm_magic == BTREE_MAGIC);
783 : Assert(metad->btm_version >= BTREE_MIN_VERSION);
784 : Assert(metad->btm_version <= BTREE_VERSION);
785 : Assert(!metad->btm_allequalimage ||
786 : metad->btm_version > BTREE_NOVAC_VERSION);
787 : Assert(metad->btm_fastroot != P_NONE);
788 :
789 23835620 : *heapkeyspace = metad->btm_version > BTREE_NOVAC_VERSION;
790 23835620 : *allequalimage = metad->btm_allequalimage;
791 : }
792 :
793 : /*
794 : * _bt_checkpage() -- Verify that a freshly-read page looks sane.
795 : */
796 : void
797 38309516 : _bt_checkpage(Relation rel, Buffer buf)
798 : {
799 38309516 : Page page = BufferGetPage(buf);
800 :
801 : /*
802 : * ReadBuffer verifies that every newly-read page passes
803 : * PageHeaderIsValid, which means it either contains a reasonably sane
804 : * page header or is all-zero. We have to defend against the all-zero
805 : * case, however.
806 : */
807 38309516 : if (PageIsNew(page))
808 0 : ereport(ERROR,
809 : (errcode(ERRCODE_INDEX_CORRUPTED),
810 : errmsg("index \"%s\" contains unexpected zero page at block %u",
811 : RelationGetRelationName(rel),
812 : BufferGetBlockNumber(buf)),
813 : errhint("Please REINDEX it.")));
814 :
815 : /*
816 : * Additionally check that the special area looks sane.
817 : */
818 38309516 : if (PageGetSpecialSize(page) != MAXALIGN(sizeof(BTPageOpaqueData)))
819 0 : ereport(ERROR,
820 : (errcode(ERRCODE_INDEX_CORRUPTED),
821 : errmsg("index \"%s\" contains corrupted page at block %u",
822 : RelationGetRelationName(rel),
823 : BufferGetBlockNumber(buf)),
824 : errhint("Please REINDEX it.")));
825 38309516 : }
826 :
827 : /*
828 : * _bt_getbuf() -- Get an existing block in a buffer, for read or write.
829 : *
830 : * The general rule in nbtree is that it's never okay to access a
831 : * page without holding both a buffer pin and a buffer lock on
832 : * the page's buffer.
833 : *
834 : * When this routine returns, the appropriate lock is set on the
835 : * requested buffer and its reference count has been incremented
836 : * (ie, the buffer is "locked and pinned"). Also, we apply
837 : * _bt_checkpage to sanity-check the page, and perform Valgrind
838 : * client requests that help Valgrind detect unsafe page accesses.
839 : *
840 : * Note: raw LockBuffer() calls are disallowed in nbtree; all
841 : * buffer lock requests need to go through wrapper functions such
842 : * as _bt_lockbuf().
843 : */
844 : Buffer
845 21831814 : _bt_getbuf(Relation rel, BlockNumber blkno, int access)
846 : {
847 : Buffer buf;
848 :
849 : Assert(BlockNumberIsValid(blkno));
850 :
851 : /* Read an existing block of the relation */
852 21831814 : buf = ReadBuffer(rel, blkno);
853 21831814 : _bt_lockbuf(rel, buf, access);
854 21831814 : _bt_checkpage(rel, buf);
855 :
856 21831814 : return buf;
857 : }
858 :
859 : /*
860 : * _bt_allocbuf() -- Allocate a new block/page.
861 : *
862 : * Returns a write-locked buffer containing an unallocated nbtree page.
863 : *
864 : * Callers are required to pass a valid heaprel. We need heaprel so that we
865 : * can handle generating a snapshotConflictHorizon that makes reusing a page
866 : * from the FSM safe for queries that may be running on standbys.
867 : */
868 : Buffer
869 34392 : _bt_allocbuf(Relation rel, Relation heaprel)
870 : {
871 : Buffer buf;
872 : BlockNumber blkno;
873 : Page page;
874 :
875 : Assert(heaprel != NULL);
876 :
877 : /*
878 : * First see if the FSM knows of any free pages.
879 : *
880 : * We can't trust the FSM's report unreservedly; we have to check that the
881 : * page is still free. (For example, an already-free page could have been
882 : * re-used between the time the last VACUUM scanned it and the time the
883 : * VACUUM made its FSM updates.)
884 : *
885 : * In fact, it's worse than that: we can't even assume that it's safe to
886 : * take a lock on the reported page. If somebody else has a lock on it,
887 : * or even worse our own caller does, we could deadlock. (The own-caller
888 : * scenario is actually not improbable. Consider an index on a serial or
889 : * timestamp column. Nearly all splits will be at the rightmost page, so
890 : * it's entirely likely that _bt_split will call us while holding a lock
891 : * on the page most recently acquired from FSM. A VACUUM running
892 : * concurrently with the previous split could well have placed that page
893 : * back in FSM.)
894 : *
895 : * To get around that, we ask for only a conditional lock on the reported
896 : * page. If we fail, then someone else is using the page, and we may
897 : * reasonably assume it's not free. (If we happen to be wrong, the worst
898 : * consequence is the page will be lost to use till the next VACUUM, which
899 : * is no big problem.)
900 : */
901 : for (;;)
902 : {
903 34392 : blkno = GetFreeIndexPage(rel);
904 34392 : if (blkno == InvalidBlockNumber)
905 34282 : break;
906 110 : buf = ReadBuffer(rel, blkno);
907 110 : if (_bt_conditionallockbuf(rel, buf))
908 : {
909 110 : page = BufferGetPage(buf);
910 :
911 : /*
912 : * It's possible to find an all-zeroes page in an index. For
913 : * example, a backend might successfully extend the relation one
914 : * page and then crash before it is able to make a WAL entry for
915 : * adding the page. If we find a zeroed page then reclaim it
916 : * immediately.
917 : */
918 110 : if (PageIsNew(page))
919 : {
920 : /* Okay to use page. Initialize and return it. */
921 0 : _bt_pageinit(page, BufferGetPageSize(buf));
922 0 : return buf;
923 : }
924 :
925 110 : if (BTPageIsRecyclable(page, heaprel))
926 : {
927 : /*
928 : * If we are generating WAL for Hot Standby then create a WAL
929 : * record that will allow us to conflict with queries running
930 : * on standby, in case they have snapshots older than safexid
931 : * value
932 : */
933 110 : if (RelationNeedsWAL(rel) && XLogStandbyInfoActive())
934 : {
935 : xl_btree_reuse_page xlrec_reuse;
936 :
937 : /*
938 : * Note that we don't register the buffer with the record,
939 : * because this operation doesn't modify the page (that
940 : * already happened, back when VACUUM deleted the page).
941 : * This record only exists to provide a conflict point for
942 : * Hot Standby. See record REDO routine comments.
943 : */
944 110 : xlrec_reuse.locator = rel->rd_locator;
945 110 : xlrec_reuse.block = blkno;
946 110 : xlrec_reuse.snapshotConflictHorizon = BTPageGetDeleteXid(page);
947 110 : xlrec_reuse.isCatalogRel =
948 110 : RelationIsAccessibleInLogicalDecoding(heaprel);
949 :
950 110 : XLogBeginInsert();
951 110 : XLogRegisterData((char *) &xlrec_reuse, SizeOfBtreeReusePage);
952 :
953 110 : XLogInsert(RM_BTREE_ID, XLOG_BTREE_REUSE_PAGE);
954 : }
955 :
956 : /* Okay to use page. Re-initialize and return it. */
957 110 : _bt_pageinit(page, BufferGetPageSize(buf));
958 110 : return buf;
959 : }
960 0 : elog(DEBUG2, "FSM returned nonrecyclable page");
961 0 : _bt_relbuf(rel, buf);
962 : }
963 : else
964 : {
965 0 : elog(DEBUG2, "FSM returned nonlockable page");
966 : /* couldn't get lock, so just drop pin */
967 0 : ReleaseBuffer(buf);
968 : }
969 : }
970 :
971 : /*
972 : * Extend the relation by one page. Need to use RBM_ZERO_AND_LOCK or we
973 : * risk a race condition against btvacuumscan --- see comments therein.
974 : * This forces us to repeat the valgrind request that _bt_lockbuf()
975 : * otherwise would make, as we can't use _bt_lockbuf() without introducing
976 : * a race.
977 : */
978 34282 : buf = ExtendBufferedRel(BMR_REL(rel), MAIN_FORKNUM, NULL, EB_LOCK_FIRST);
979 34282 : if (!RelationUsesLocalBuffers(rel))
980 : VALGRIND_MAKE_MEM_DEFINED(BufferGetPage(buf), BLCKSZ);
981 :
982 : /* Initialize the new page before returning it */
983 34282 : page = BufferGetPage(buf);
984 : Assert(PageIsNew(page));
985 34282 : _bt_pageinit(page, BufferGetPageSize(buf));
986 :
987 34282 : return buf;
988 : }
989 :
990 : /*
991 : * _bt_relandgetbuf() -- release a locked buffer and get another one.
992 : *
993 : * This is equivalent to _bt_relbuf followed by _bt_getbuf. Also, if obuf is
994 : * InvalidBuffer then it reduces to just _bt_getbuf; allowing this case
995 : * simplifies some callers.
996 : *
997 : * The original motivation for using this was to avoid two entries to the
998 : * bufmgr when one would do. However, now it's mainly just a notational
999 : * convenience. The only case where it saves work over _bt_relbuf/_bt_getbuf
1000 : * is when the target page is the same one already in the buffer.
1001 : */
1002 : Buffer
1003 16348120 : _bt_relandgetbuf(Relation rel, Buffer obuf, BlockNumber blkno, int access)
1004 : {
1005 : Buffer buf;
1006 :
1007 : Assert(BlockNumberIsValid(blkno));
1008 16348120 : if (BufferIsValid(obuf))
1009 16332316 : _bt_unlockbuf(rel, obuf);
1010 16348120 : buf = ReleaseAndReadBuffer(obuf, rel, blkno);
1011 16348120 : _bt_lockbuf(rel, buf, access);
1012 :
1013 16348120 : _bt_checkpage(rel, buf);
1014 16348120 : return buf;
1015 : }
1016 :
1017 : /*
1018 : * _bt_relbuf() -- release a locked buffer.
1019 : *
1020 : * Lock and pin (refcount) are both dropped.
1021 : */
1022 : void
1023 9109486 : _bt_relbuf(Relation rel, Buffer buf)
1024 : {
1025 9109486 : _bt_unlockbuf(rel, buf);
1026 9109486 : ReleaseBuffer(buf);
1027 9109486 : }
1028 :
1029 : /*
1030 : * _bt_lockbuf() -- lock a pinned buffer.
1031 : *
1032 : * Lock is acquired without acquiring another pin. This is like a raw
1033 : * LockBuffer() call, but performs extra steps needed by Valgrind.
1034 : *
1035 : * Note: Caller may need to call _bt_checkpage() with buf when pin on buf
1036 : * wasn't originally acquired in _bt_getbuf() or _bt_relandgetbuf().
1037 : */
1038 : void
1039 39090488 : _bt_lockbuf(Relation rel, Buffer buf, int access)
1040 : {
1041 : /* LockBuffer() asserts that pin is held by this backend */
1042 39090488 : LockBuffer(buf, access);
1043 :
1044 : /*
1045 : * It doesn't matter that _bt_unlockbuf() won't get called in the event of
1046 : * an nbtree error (e.g. a unique violation error). That won't cause
1047 : * Valgrind false positives.
1048 : *
1049 : * The nbtree client requests are superimposed on top of the bufmgr.c
1050 : * buffer pin client requests. In the event of an nbtree error the buffer
1051 : * will certainly get marked as defined when the backend once again
1052 : * acquires its first pin on the buffer. (Of course, if the backend never
1053 : * touches the buffer again then it doesn't matter that it remains
1054 : * non-accessible to Valgrind.)
1055 : *
1056 : * Note: When an IndexTuple C pointer gets computed using an ItemId read
1057 : * from a page while a lock was held, the C pointer becomes unsafe to
1058 : * dereference forever as soon as the lock is released. Valgrind can only
1059 : * detect cases where the pointer gets dereferenced with no _current_
1060 : * lock/pin held, though.
1061 : */
1062 39090488 : if (!RelationUsesLocalBuffers(rel))
1063 : VALGRIND_MAKE_MEM_DEFINED(BufferGetPage(buf), BLCKSZ);
1064 39090488 : }
1065 :
1066 : /*
1067 : * _bt_unlockbuf() -- unlock a pinned buffer.
1068 : */
1069 : void
1070 39188114 : _bt_unlockbuf(Relation rel, Buffer buf)
1071 : {
1072 : /*
1073 : * Buffer is pinned and locked, which means that it is expected to be
1074 : * defined and addressable. Check that proactively.
1075 : */
1076 : VALGRIND_CHECK_MEM_IS_DEFINED(BufferGetPage(buf), BLCKSZ);
1077 :
1078 : /* LockBuffer() asserts that pin is held by this backend */
1079 39188114 : LockBuffer(buf, BUFFER_LOCK_UNLOCK);
1080 :
1081 39188114 : if (!RelationUsesLocalBuffers(rel))
1082 : VALGRIND_MAKE_MEM_NOACCESS(BufferGetPage(buf), BLCKSZ);
1083 39188114 : }
1084 :
1085 : /*
1086 : * _bt_conditionallockbuf() -- conditionally BT_WRITE lock pinned
1087 : * buffer.
1088 : *
1089 : * Note: Caller may need to call _bt_checkpage() with buf when pin on buf
1090 : * wasn't originally acquired in _bt_getbuf() or _bt_relandgetbuf().
1091 : */
1092 : bool
1093 64534 : _bt_conditionallockbuf(Relation rel, Buffer buf)
1094 : {
1095 : /* ConditionalLockBuffer() asserts that pin is held by this backend */
1096 64534 : if (!ConditionalLockBuffer(buf))
1097 1176 : return false;
1098 :
1099 63358 : if (!RelationUsesLocalBuffers(rel))
1100 : VALGRIND_MAKE_MEM_DEFINED(BufferGetPage(buf), BLCKSZ);
1101 :
1102 63358 : return true;
1103 : }
1104 :
1105 : /*
1106 : * _bt_upgradelockbufcleanup() -- upgrade lock to a full cleanup lock.
1107 : */
1108 : void
1109 20522 : _bt_upgradelockbufcleanup(Relation rel, Buffer buf)
1110 : {
1111 : /*
1112 : * Buffer is pinned and locked, which means that it is expected to be
1113 : * defined and addressable. Check that proactively.
1114 : */
1115 : VALGRIND_CHECK_MEM_IS_DEFINED(BufferGetPage(buf), BLCKSZ);
1116 :
1117 : /* LockBuffer() asserts that pin is held by this backend */
1118 20522 : LockBuffer(buf, BUFFER_LOCK_UNLOCK);
1119 20522 : LockBufferForCleanup(buf);
1120 20522 : }
1121 :
1122 : /*
1123 : * _bt_pageinit() -- Initialize a new page.
1124 : *
1125 : * On return, the page header is initialized; data space is empty;
1126 : * special space is zeroed out.
1127 : */
1128 : void
1129 163534 : _bt_pageinit(Page page, Size size)
1130 : {
1131 163534 : PageInit(page, size, sizeof(BTPageOpaqueData));
1132 163534 : }
1133 :
1134 : /*
1135 : * Delete item(s) from a btree leaf page during VACUUM.
1136 : *
1137 : * This routine assumes that the caller already has a full cleanup lock on
1138 : * the buffer. Also, the given deletable and updatable arrays *must* be
1139 : * sorted in ascending order.
1140 : *
1141 : * Routine deals with deleting TIDs when some (but not all) of the heap TIDs
1142 : * in an existing posting list item are to be removed. This works by
1143 : * updating/overwriting an existing item with caller's new version of the item
1144 : * (a version that lacks the TIDs that are to be deleted).
1145 : *
1146 : * We record VACUUMs and b-tree deletes differently in WAL. Deletes must
1147 : * generate their own snapshotConflictHorizon directly from the tableam,
1148 : * whereas VACUUMs rely on the initial VACUUM table scan performing
1149 : * WAL-logging that takes care of the issue for the table's indexes
1150 : * indirectly. Also, we remove the VACUUM cycle ID from pages, which b-tree
1151 : * deletes don't do.
1152 : */
1153 : void
1154 13274 : _bt_delitems_vacuum(Relation rel, Buffer buf,
1155 : OffsetNumber *deletable, int ndeletable,
1156 : BTVacuumPosting *updatable, int nupdatable)
1157 : {
1158 13274 : Page page = BufferGetPage(buf);
1159 : BTPageOpaque opaque;
1160 13274 : bool needswal = RelationNeedsWAL(rel);
1161 13274 : char *updatedbuf = NULL;
1162 13274 : Size updatedbuflen = 0;
1163 : OffsetNumber updatedoffsets[MaxIndexTuplesPerPage];
1164 :
1165 : /* Shouldn't be called unless there's something to do */
1166 : Assert(ndeletable > 0 || nupdatable > 0);
1167 :
1168 : /* Generate new version of posting lists without deleted TIDs */
1169 13274 : if (nupdatable > 0)
1170 1218 : updatedbuf = _bt_delitems_update(updatable, nupdatable,
1171 : updatedoffsets, &updatedbuflen,
1172 : needswal);
1173 :
1174 : /* No ereport(ERROR) until changes are logged */
1175 13274 : START_CRIT_SECTION();
1176 :
1177 : /*
1178 : * Handle posting tuple updates.
1179 : *
1180 : * Deliberately do this before handling simple deletes. If we did it the
1181 : * other way around (i.e. WAL record order -- simple deletes before
1182 : * updates) then we'd have to make compensating changes to the 'updatable'
1183 : * array of offset numbers.
1184 : *
1185 : * PageIndexTupleOverwrite() won't unset each item's LP_DEAD bit when it
1186 : * happens to already be set. It's important that we not interfere with
1187 : * any future simple index tuple deletion operations.
1188 : */
1189 41914 : for (int i = 0; i < nupdatable; i++)
1190 : {
1191 28640 : OffsetNumber updatedoffset = updatedoffsets[i];
1192 : IndexTuple itup;
1193 : Size itemsz;
1194 :
1195 28640 : itup = updatable[i]->itup;
1196 28640 : itemsz = MAXALIGN(IndexTupleSize(itup));
1197 28640 : if (!PageIndexTupleOverwrite(page, updatedoffset, (Item) itup,
1198 : itemsz))
1199 0 : elog(PANIC, "failed to update partially dead item in block %u of index \"%s\"",
1200 : BufferGetBlockNumber(buf), RelationGetRelationName(rel));
1201 : }
1202 :
1203 : /* Now handle simple deletes of entire tuples */
1204 13274 : if (ndeletable > 0)
1205 12968 : PageIndexMultiDelete(page, deletable, ndeletable);
1206 :
1207 : /*
1208 : * We can clear the vacuum cycle ID since this page has certainly been
1209 : * processed by the current vacuum scan.
1210 : */
1211 13274 : opaque = BTPageGetOpaque(page);
1212 13274 : opaque->btpo_cycleid = 0;
1213 :
1214 : /*
1215 : * Clear the BTP_HAS_GARBAGE page flag.
1216 : *
1217 : * This flag indicates the presence of LP_DEAD items on the page (though
1218 : * not reliably). Note that we only rely on it with pg_upgrade'd
1219 : * !heapkeyspace indexes. That's why clearing it here won't usually
1220 : * interfere with simple index tuple deletion.
1221 : */
1222 13274 : opaque->btpo_flags &= ~BTP_HAS_GARBAGE;
1223 :
1224 13274 : MarkBufferDirty(buf);
1225 :
1226 : /* XLOG stuff */
1227 13274 : if (needswal)
1228 : {
1229 : XLogRecPtr recptr;
1230 : xl_btree_vacuum xlrec_vacuum;
1231 :
1232 13272 : xlrec_vacuum.ndeleted = ndeletable;
1233 13272 : xlrec_vacuum.nupdated = nupdatable;
1234 :
1235 13272 : XLogBeginInsert();
1236 13272 : XLogRegisterBuffer(0, buf, REGBUF_STANDARD);
1237 13272 : XLogRegisterData((char *) &xlrec_vacuum, SizeOfBtreeVacuum);
1238 :
1239 13272 : if (ndeletable > 0)
1240 12966 : XLogRegisterBufData(0, (char *) deletable,
1241 : ndeletable * sizeof(OffsetNumber));
1242 :
1243 13272 : if (nupdatable > 0)
1244 : {
1245 1218 : XLogRegisterBufData(0, (char *) updatedoffsets,
1246 : nupdatable * sizeof(OffsetNumber));
1247 1218 : XLogRegisterBufData(0, updatedbuf, updatedbuflen);
1248 : }
1249 :
1250 13272 : recptr = XLogInsert(RM_BTREE_ID, XLOG_BTREE_VACUUM);
1251 :
1252 13272 : PageSetLSN(page, recptr);
1253 : }
1254 :
1255 13274 : END_CRIT_SECTION();
1256 :
1257 : /* can't leak memory here */
1258 13274 : if (updatedbuf != NULL)
1259 1218 : pfree(updatedbuf);
1260 : /* free tuples allocated within _bt_delitems_update() */
1261 41914 : for (int i = 0; i < nupdatable; i++)
1262 28640 : pfree(updatable[i]->itup);
1263 13274 : }
1264 :
1265 : /*
1266 : * Delete item(s) from a btree leaf page during single-page cleanup.
1267 : *
1268 : * This routine assumes that the caller has pinned and write locked the
1269 : * buffer. Also, the given deletable and updatable arrays *must* be sorted in
1270 : * ascending order.
1271 : *
1272 : * Routine deals with deleting TIDs when some (but not all) of the heap TIDs
1273 : * in an existing posting list item are to be removed. This works by
1274 : * updating/overwriting an existing item with caller's new version of the item
1275 : * (a version that lacks the TIDs that are to be deleted).
1276 : *
1277 : * This is nearly the same as _bt_delitems_vacuum as far as what it does to
1278 : * the page, but it needs its own snapshotConflictHorizon and isCatalogRel
1279 : * (from the tableam). This is used by the REDO routine to generate recovery
1280 : * conflicts. The other difference is that only _bt_delitems_vacuum will
1281 : * clear page's VACUUM cycle ID.
1282 : */
1283 : static void
1284 7516 : _bt_delitems_delete(Relation rel, Buffer buf,
1285 : TransactionId snapshotConflictHorizon, bool isCatalogRel,
1286 : OffsetNumber *deletable, int ndeletable,
1287 : BTVacuumPosting *updatable, int nupdatable)
1288 : {
1289 7516 : Page page = BufferGetPage(buf);
1290 : BTPageOpaque opaque;
1291 7516 : bool needswal = RelationNeedsWAL(rel);
1292 7516 : char *updatedbuf = NULL;
1293 7516 : Size updatedbuflen = 0;
1294 : OffsetNumber updatedoffsets[MaxIndexTuplesPerPage];
1295 :
1296 : /* Shouldn't be called unless there's something to do */
1297 : Assert(ndeletable > 0 || nupdatable > 0);
1298 :
1299 : /* Generate new versions of posting lists without deleted TIDs */
1300 7516 : if (nupdatable > 0)
1301 674 : updatedbuf = _bt_delitems_update(updatable, nupdatable,
1302 : updatedoffsets, &updatedbuflen,
1303 : needswal);
1304 :
1305 : /* No ereport(ERROR) until changes are logged */
1306 7516 : START_CRIT_SECTION();
1307 :
1308 : /* Handle updates and deletes just like _bt_delitems_vacuum */
1309 17274 : for (int i = 0; i < nupdatable; i++)
1310 : {
1311 9758 : OffsetNumber updatedoffset = updatedoffsets[i];
1312 : IndexTuple itup;
1313 : Size itemsz;
1314 :
1315 9758 : itup = updatable[i]->itup;
1316 9758 : itemsz = MAXALIGN(IndexTupleSize(itup));
1317 9758 : if (!PageIndexTupleOverwrite(page, updatedoffset, (Item) itup,
1318 : itemsz))
1319 0 : elog(PANIC, "failed to update partially dead item in block %u of index \"%s\"",
1320 : BufferGetBlockNumber(buf), RelationGetRelationName(rel));
1321 : }
1322 :
1323 7516 : if (ndeletable > 0)
1324 7434 : PageIndexMultiDelete(page, deletable, ndeletable);
1325 :
1326 : /*
1327 : * Unlike _bt_delitems_vacuum, we *must not* clear the vacuum cycle ID at
1328 : * this point. The VACUUM command alone controls vacuum cycle IDs.
1329 : */
1330 7516 : opaque = BTPageGetOpaque(page);
1331 :
1332 : /*
1333 : * Clear the BTP_HAS_GARBAGE page flag.
1334 : *
1335 : * This flag indicates the presence of LP_DEAD items on the page (though
1336 : * not reliably). Note that we only rely on it with pg_upgrade'd
1337 : * !heapkeyspace indexes.
1338 : */
1339 7516 : opaque->btpo_flags &= ~BTP_HAS_GARBAGE;
1340 :
1341 7516 : MarkBufferDirty(buf);
1342 :
1343 : /* XLOG stuff */
1344 7516 : if (needswal)
1345 : {
1346 : XLogRecPtr recptr;
1347 : xl_btree_delete xlrec_delete;
1348 :
1349 7468 : xlrec_delete.snapshotConflictHorizon = snapshotConflictHorizon;
1350 7468 : xlrec_delete.ndeleted = ndeletable;
1351 7468 : xlrec_delete.nupdated = nupdatable;
1352 7468 : xlrec_delete.isCatalogRel = isCatalogRel;
1353 :
1354 7468 : XLogBeginInsert();
1355 7468 : XLogRegisterBuffer(0, buf, REGBUF_STANDARD);
1356 7468 : XLogRegisterData((char *) &xlrec_delete, SizeOfBtreeDelete);
1357 :
1358 7468 : if (ndeletable > 0)
1359 7386 : XLogRegisterBufData(0, (char *) deletable,
1360 : ndeletable * sizeof(OffsetNumber));
1361 :
1362 7468 : if (nupdatable > 0)
1363 : {
1364 674 : XLogRegisterBufData(0, (char *) updatedoffsets,
1365 : nupdatable * sizeof(OffsetNumber));
1366 674 : XLogRegisterBufData(0, updatedbuf, updatedbuflen);
1367 : }
1368 :
1369 7468 : recptr = XLogInsert(RM_BTREE_ID, XLOG_BTREE_DELETE);
1370 :
1371 7468 : PageSetLSN(page, recptr);
1372 : }
1373 :
1374 7516 : END_CRIT_SECTION();
1375 :
1376 : /* can't leak memory here */
1377 7516 : if (updatedbuf != NULL)
1378 674 : pfree(updatedbuf);
1379 : /* free tuples allocated within _bt_delitems_update() */
1380 17274 : for (int i = 0; i < nupdatable; i++)
1381 9758 : pfree(updatable[i]->itup);
1382 7516 : }
1383 :
1384 : /*
1385 : * Set up state needed to delete TIDs from posting list tuples via "updating"
1386 : * the tuple. Performs steps common to both _bt_delitems_vacuum and
1387 : * _bt_delitems_delete. These steps must take place before each function's
1388 : * critical section begins.
1389 : *
1390 : * updatable and nupdatable are inputs, though note that we will use
1391 : * _bt_update_posting() to replace the original itup with a pointer to a final
1392 : * version in palloc()'d memory. Caller should free the tuples when its done.
1393 : *
1394 : * The first nupdatable entries from updatedoffsets are set to the page offset
1395 : * number for posting list tuples that caller updates. This is mostly useful
1396 : * because caller may need to WAL-log the page offsets (though we always do
1397 : * this for caller out of convenience).
1398 : *
1399 : * Returns buffer consisting of an array of xl_btree_update structs that
1400 : * describe the steps we perform here for caller (though only when needswal is
1401 : * true). Also sets *updatedbuflen to the final size of the buffer. This
1402 : * buffer is used by caller when WAL logging is required.
1403 : */
1404 : static char *
1405 1892 : _bt_delitems_update(BTVacuumPosting *updatable, int nupdatable,
1406 : OffsetNumber *updatedoffsets, Size *updatedbuflen,
1407 : bool needswal)
1408 : {
1409 1892 : char *updatedbuf = NULL;
1410 1892 : Size buflen = 0;
1411 :
1412 : /* Shouldn't be called unless there's something to do */
1413 : Assert(nupdatable > 0);
1414 :
1415 40290 : for (int i = 0; i < nupdatable; i++)
1416 : {
1417 38398 : BTVacuumPosting vacposting = updatable[i];
1418 : Size itemsz;
1419 :
1420 : /* Replace work area IndexTuple with updated version */
1421 38398 : _bt_update_posting(vacposting);
1422 :
1423 : /* Keep track of size of xl_btree_update for updatedbuf in passing */
1424 38398 : itemsz = SizeOfBtreeUpdate + vacposting->ndeletedtids * sizeof(uint16);
1425 38398 : buflen += itemsz;
1426 :
1427 : /* Build updatedoffsets buffer in passing */
1428 38398 : updatedoffsets[i] = vacposting->updatedoffset;
1429 : }
1430 :
1431 : /* XLOG stuff */
1432 1892 : if (needswal)
1433 : {
1434 1892 : Size offset = 0;
1435 :
1436 : /* Allocate, set final size for caller */
1437 1892 : updatedbuf = palloc(buflen);
1438 1892 : *updatedbuflen = buflen;
1439 40290 : for (int i = 0; i < nupdatable; i++)
1440 : {
1441 38398 : BTVacuumPosting vacposting = updatable[i];
1442 : Size itemsz;
1443 : xl_btree_update update;
1444 :
1445 38398 : update.ndeletedtids = vacposting->ndeletedtids;
1446 38398 : memcpy(updatedbuf + offset, &update.ndeletedtids,
1447 : SizeOfBtreeUpdate);
1448 38398 : offset += SizeOfBtreeUpdate;
1449 :
1450 38398 : itemsz = update.ndeletedtids * sizeof(uint16);
1451 38398 : memcpy(updatedbuf + offset, vacposting->deletetids, itemsz);
1452 38398 : offset += itemsz;
1453 : }
1454 : }
1455 :
1456 1892 : return updatedbuf;
1457 : }
1458 :
1459 : /*
1460 : * Comparator used by _bt_delitems_delete_check() to restore deltids array
1461 : * back to its original leaf-page-wise sort order
1462 : */
1463 : static int
1464 4826950 : _bt_delitems_cmp(const void *a, const void *b)
1465 : {
1466 4826950 : TM_IndexDelete *indexdelete1 = (TM_IndexDelete *) a;
1467 4826950 : TM_IndexDelete *indexdelete2 = (TM_IndexDelete *) b;
1468 :
1469 : Assert(indexdelete1->id != indexdelete2->id);
1470 :
1471 4826950 : return pg_cmp_s16(indexdelete1->id, indexdelete2->id);
1472 : }
1473 :
1474 : /*
1475 : * Try to delete item(s) from a btree leaf page during single-page cleanup.
1476 : *
1477 : * nbtree interface to table_index_delete_tuples(). Deletes a subset of index
1478 : * tuples from caller's deltids array: those whose TIDs are found safe to
1479 : * delete by the tableam (or already marked LP_DEAD in index, and so already
1480 : * known to be deletable by our simple index deletion caller). We physically
1481 : * delete index tuples from buf leaf page last of all (for index tuples where
1482 : * that is known to be safe following our table_index_delete_tuples() call).
1483 : *
1484 : * Simple index deletion caller only includes TIDs from index tuples marked
1485 : * LP_DEAD, as well as extra TIDs it found on the same leaf page that can be
1486 : * included without increasing the total number of distinct table blocks for
1487 : * the deletion operation as a whole. This approach often allows us to delete
1488 : * some extra index tuples that were practically free for tableam to check in
1489 : * passing (when they actually turn out to be safe to delete). It probably
1490 : * only makes sense for the tableam to go ahead with these extra checks when
1491 : * it is block-oriented (otherwise the checks probably won't be practically
1492 : * free, which we rely on). The tableam interface requires the tableam side
1493 : * to handle the problem, though, so this is okay (we as an index AM are free
1494 : * to make the simplifying assumption that all tableams must be block-based).
1495 : *
1496 : * Bottom-up index deletion caller provides all the TIDs from the leaf page,
1497 : * without expecting that tableam will check most of them. The tableam has
1498 : * considerable discretion around which entries/blocks it checks. Our role in
1499 : * costing the bottom-up deletion operation is strictly advisory.
1500 : *
1501 : * Note: Caller must have added deltids entries (i.e. entries that go in
1502 : * delstate's main array) in leaf-page-wise order: page offset number order,
1503 : * TID order among entries taken from the same posting list tuple (tiebreak on
1504 : * TID). This order is convenient to work with here.
1505 : *
1506 : * Note: We also rely on the id field of each deltids element "capturing" this
1507 : * original leaf-page-wise order. That is, we expect to be able to get back
1508 : * to the original leaf-page-wise order just by sorting deltids on the id
1509 : * field (tableam will sort deltids for its own reasons, so we'll need to put
1510 : * it back in leaf-page-wise order afterwards).
1511 : */
1512 : void
1513 10440 : _bt_delitems_delete_check(Relation rel, Buffer buf, Relation heapRel,
1514 : TM_IndexDeleteOp *delstate)
1515 : {
1516 10440 : Page page = BufferGetPage(buf);
1517 : TransactionId snapshotConflictHorizon;
1518 : bool isCatalogRel;
1519 10440 : OffsetNumber postingidxoffnum = InvalidOffsetNumber;
1520 10440 : int ndeletable = 0,
1521 10440 : nupdatable = 0;
1522 : OffsetNumber deletable[MaxIndexTuplesPerPage];
1523 : BTVacuumPosting updatable[MaxIndexTuplesPerPage];
1524 :
1525 : /* Use tableam interface to determine which tuples to delete first */
1526 10440 : snapshotConflictHorizon = table_index_delete_tuples(heapRel, delstate);
1527 10440 : isCatalogRel = RelationIsAccessibleInLogicalDecoding(heapRel);
1528 :
1529 : /* Should not WAL-log snapshotConflictHorizon unless it's required */
1530 10440 : if (!XLogStandbyInfoActive())
1531 3048 : snapshotConflictHorizon = InvalidTransactionId;
1532 :
1533 : /*
1534 : * Construct a leaf-page-wise description of what _bt_delitems_delete()
1535 : * needs to do to physically delete index tuples from the page.
1536 : *
1537 : * Must sort deltids array to restore leaf-page-wise order (original order
1538 : * before call to tableam). This is the order that the loop expects.
1539 : *
1540 : * Note that deltids array might be a lot smaller now. It might even have
1541 : * no entries at all (with bottom-up deletion caller), in which case there
1542 : * is nothing left to do.
1543 : */
1544 10440 : qsort(delstate->deltids, delstate->ndeltids, sizeof(TM_IndexDelete),
1545 : _bt_delitems_cmp);
1546 10440 : if (delstate->ndeltids == 0)
1547 : {
1548 : Assert(delstate->bottomup);
1549 2924 : return;
1550 : }
1551 :
1552 : /* We definitely have to delete at least one index tuple (or one TID) */
1553 703332 : for (int i = 0; i < delstate->ndeltids; i++)
1554 : {
1555 695816 : TM_IndexStatus *dstatus = delstate->status + delstate->deltids[i].id;
1556 695816 : OffsetNumber idxoffnum = dstatus->idxoffnum;
1557 695816 : ItemId itemid = PageGetItemId(page, idxoffnum);
1558 695816 : IndexTuple itup = (IndexTuple) PageGetItem(page, itemid);
1559 : int nestedi,
1560 : nitem;
1561 : BTVacuumPosting vacposting;
1562 :
1563 : Assert(OffsetNumberIsValid(idxoffnum));
1564 :
1565 695816 : if (idxoffnum == postingidxoffnum)
1566 : {
1567 : /*
1568 : * This deltid entry is a TID from a posting list tuple that has
1569 : * already been completely processed
1570 : */
1571 : Assert(BTreeTupleIsPosting(itup));
1572 : Assert(ItemPointerCompare(BTreeTupleGetHeapTID(itup),
1573 : &delstate->deltids[i].tid) < 0);
1574 : Assert(ItemPointerCompare(BTreeTupleGetMaxHeapTID(itup),
1575 : &delstate->deltids[i].tid) >= 0);
1576 27132 : continue;
1577 : }
1578 :
1579 668684 : if (!BTreeTupleIsPosting(itup))
1580 : {
1581 : /* Plain non-pivot tuple */
1582 : Assert(ItemPointerEquals(&itup->t_tid, &delstate->deltids[i].tid));
1583 643672 : if (dstatus->knowndeletable)
1584 528230 : deletable[ndeletable++] = idxoffnum;
1585 643672 : continue;
1586 : }
1587 :
1588 : /*
1589 : * itup is a posting list tuple whose lowest deltids entry (which may
1590 : * or may not be for the first TID from itup) is considered here now.
1591 : * We should process all of the deltids entries for the posting list
1592 : * together now, though (not just the lowest). Remember to skip over
1593 : * later itup-related entries during later iterations of outermost
1594 : * loop.
1595 : */
1596 25012 : postingidxoffnum = idxoffnum; /* Remember work in outermost loop */
1597 25012 : nestedi = i; /* Initialize for first itup deltids entry */
1598 25012 : vacposting = NULL; /* Describes final action for itup */
1599 25012 : nitem = BTreeTupleGetNPosting(itup);
1600 106378 : for (int p = 0; p < nitem; p++)
1601 : {
1602 81366 : ItemPointer ptid = BTreeTupleGetPostingN(itup, p);
1603 81366 : int ptidcmp = -1;
1604 :
1605 : /*
1606 : * This nested loop reuses work across ptid TIDs taken from itup.
1607 : * We take advantage of the fact that both itup's TIDs and deltids
1608 : * entries (within a single itup/posting list grouping) must both
1609 : * be in ascending TID order.
1610 : */
1611 120056 : for (; nestedi < delstate->ndeltids; nestedi++)
1612 : {
1613 115580 : TM_IndexDelete *tcdeltid = &delstate->deltids[nestedi];
1614 115580 : TM_IndexStatus *tdstatus = (delstate->status + tcdeltid->id);
1615 :
1616 : /* Stop once we get past all itup related deltids entries */
1617 : Assert(tdstatus->idxoffnum >= idxoffnum);
1618 115580 : if (tdstatus->idxoffnum != idxoffnum)
1619 23778 : break;
1620 :
1621 : /* Skip past non-deletable itup related entries up front */
1622 91802 : if (!tdstatus->knowndeletable)
1623 8094 : continue;
1624 :
1625 : /* Entry is first partial ptid match (or an exact match)? */
1626 83708 : ptidcmp = ItemPointerCompare(&tcdeltid->tid, ptid);
1627 83708 : if (ptidcmp >= 0)
1628 : {
1629 : /* Greater than or equal (partial or exact) match... */
1630 53112 : break;
1631 : }
1632 : }
1633 :
1634 : /* ...exact ptid match to a deletable deltids entry? */
1635 81366 : if (ptidcmp != 0)
1636 37316 : continue;
1637 :
1638 : /* Exact match for deletable deltids entry -- ptid gets deleted */
1639 44050 : if (vacposting == NULL)
1640 : {
1641 22500 : vacposting = palloc(offsetof(BTVacuumPostingData, deletetids) +
1642 : nitem * sizeof(uint16));
1643 22500 : vacposting->itup = itup;
1644 22500 : vacposting->updatedoffset = idxoffnum;
1645 22500 : vacposting->ndeletedtids = 0;
1646 : }
1647 44050 : vacposting->deletetids[vacposting->ndeletedtids++] = p;
1648 : }
1649 :
1650 : /* Final decision on itup, a posting list tuple */
1651 :
1652 25012 : if (vacposting == NULL)
1653 : {
1654 : /* No TIDs to delete from itup -- do nothing */
1655 : }
1656 22500 : else if (vacposting->ndeletedtids == nitem)
1657 : {
1658 : /* Straight delete of itup (to delete all TIDs) */
1659 12742 : deletable[ndeletable++] = idxoffnum;
1660 : /* Turns out we won't need granular information */
1661 12742 : pfree(vacposting);
1662 : }
1663 : else
1664 : {
1665 : /* Delete some (but not all) TIDs from itup */
1666 : Assert(vacposting->ndeletedtids > 0 &&
1667 : vacposting->ndeletedtids < nitem);
1668 9758 : updatable[nupdatable++] = vacposting;
1669 : }
1670 : }
1671 :
1672 : /* Physically delete tuples (or TIDs) using deletable (or updatable) */
1673 7516 : _bt_delitems_delete(rel, buf, snapshotConflictHorizon, isCatalogRel,
1674 : deletable, ndeletable, updatable, nupdatable);
1675 :
1676 : /* be tidy */
1677 17274 : for (int i = 0; i < nupdatable; i++)
1678 9758 : pfree(updatable[i]);
1679 : }
1680 :
1681 : /*
1682 : * Check that leftsib page (the btpo_prev of target page) is not marked with
1683 : * INCOMPLETE_SPLIT flag. Used during page deletion.
1684 : *
1685 : * Returning true indicates that page flag is set in leftsib (which is
1686 : * definitely still the left sibling of target). When that happens, the
1687 : * target doesn't have a downlink in parent, and the page deletion algorithm
1688 : * isn't prepared to handle that. Deletion of the target page (or the whole
1689 : * subtree that contains the target page) cannot take place.
1690 : *
1691 : * Caller should not have a lock on the target page itself, since pages on the
1692 : * same level must always be locked left to right to avoid deadlocks.
1693 : */
1694 : static bool
1695 5626 : _bt_leftsib_splitflag(Relation rel, BlockNumber leftsib, BlockNumber target)
1696 : {
1697 : Buffer buf;
1698 : Page page;
1699 : BTPageOpaque opaque;
1700 : bool result;
1701 :
1702 : /* Easy case: No left sibling */
1703 5626 : if (leftsib == P_NONE)
1704 4448 : return false;
1705 :
1706 1178 : buf = _bt_getbuf(rel, leftsib, BT_READ);
1707 1178 : page = BufferGetPage(buf);
1708 1178 : opaque = BTPageGetOpaque(page);
1709 :
1710 : /*
1711 : * If the left sibling was concurrently split, so that its next-pointer
1712 : * doesn't point to the current page anymore, the split that created
1713 : * target must be completed. Caller can reasonably expect that there will
1714 : * be a downlink to the target page that it can relocate using its stack.
1715 : * (We don't allow splitting an incompletely split page again until the
1716 : * previous split has been completed.)
1717 : */
1718 1178 : result = (opaque->btpo_next == target && P_INCOMPLETE_SPLIT(opaque));
1719 1178 : _bt_relbuf(rel, buf);
1720 :
1721 1178 : return result;
1722 : }
1723 :
1724 : /*
1725 : * Check that leafrightsib page (the btpo_next of target leaf page) is not
1726 : * marked with ISHALFDEAD flag. Used during page deletion.
1727 : *
1728 : * Returning true indicates that page flag is set in leafrightsib, so page
1729 : * deletion cannot go ahead. Our caller is not prepared to deal with the case
1730 : * where the parent page does not have a pivot tuples whose downlink points to
1731 : * leafrightsib (due to an earlier interrupted VACUUM operation). It doesn't
1732 : * seem worth going to the trouble of teaching our caller to deal with it.
1733 : * The situation will be resolved after VACUUM finishes the deletion of the
1734 : * half-dead page (when a future VACUUM operation reaches the target page
1735 : * again).
1736 : *
1737 : * _bt_leftsib_splitflag() is called for both leaf pages and internal pages.
1738 : * _bt_rightsib_halfdeadflag() is only called for leaf pages, though. This is
1739 : * okay because of the restriction on deleting pages that are the rightmost
1740 : * page of their parent (i.e. that such deletions can only take place when the
1741 : * entire subtree must be deleted). The leaf level check made here will apply
1742 : * to a right "cousin" leaf page rather than a simple right sibling leaf page
1743 : * in cases where caller actually goes on to attempt deleting pages that are
1744 : * above the leaf page. The right cousin leaf page is representative of the
1745 : * left edge of the subtree to the right of the to-be-deleted subtree as a
1746 : * whole, which is exactly the condition that our caller cares about.
1747 : * (Besides, internal pages are never marked half-dead, so it isn't even
1748 : * possible to _directly_ assess if an internal page is part of some other
1749 : * to-be-deleted subtree.)
1750 : */
1751 : static bool
1752 5488 : _bt_rightsib_halfdeadflag(Relation rel, BlockNumber leafrightsib)
1753 : {
1754 : Buffer buf;
1755 : Page page;
1756 : BTPageOpaque opaque;
1757 : bool result;
1758 :
1759 : Assert(leafrightsib != P_NONE);
1760 :
1761 5488 : buf = _bt_getbuf(rel, leafrightsib, BT_READ);
1762 5488 : page = BufferGetPage(buf);
1763 5488 : opaque = BTPageGetOpaque(page);
1764 :
1765 : Assert(P_ISLEAF(opaque) && !P_ISDELETED(opaque));
1766 5488 : result = P_ISHALFDEAD(opaque);
1767 5488 : _bt_relbuf(rel, buf);
1768 :
1769 5488 : return result;
1770 : }
1771 :
1772 : /*
1773 : * _bt_pagedel() -- Delete a leaf page from the b-tree, if legal to do so.
1774 : *
1775 : * This action unlinks the leaf page from the b-tree structure, removing all
1776 : * pointers leading to it --- but not touching its own left and right links.
1777 : * The page cannot be physically reclaimed right away, since other processes
1778 : * may currently be trying to follow links leading to the page; they have to
1779 : * be allowed to use its right-link to recover. See nbtree/README.
1780 : *
1781 : * On entry, the target buffer must be pinned and locked (either read or write
1782 : * lock is OK). The page must be an empty leaf page, which may be half-dead
1783 : * already (a half-dead page should only be passed to us when an earlier
1784 : * VACUUM operation was interrupted, though). Note in particular that caller
1785 : * should never pass a buffer containing an existing deleted page here. The
1786 : * lock and pin on caller's buffer will be dropped before we return.
1787 : *
1788 : * Maintains bulk delete stats for caller, which are taken from vstate. We
1789 : * need to cooperate closely with caller here so that whole VACUUM operation
1790 : * reliably avoids any double counting of subsidiary-to-leafbuf pages that we
1791 : * delete in passing. If such pages happen to be from a block number that is
1792 : * ahead of the current scanblkno position, then caller is expected to count
1793 : * them directly later on. It's simpler for us to understand caller's
1794 : * requirements than it would be for caller to understand when or how a
1795 : * deleted page became deleted after the fact.
1796 : *
1797 : * NOTE: this leaks memory. Rather than trying to clean up everything
1798 : * carefully, it's better to run it in a temp context that can be reset
1799 : * frequently.
1800 : */
1801 : void
1802 5690 : _bt_pagedel(Relation rel, Buffer leafbuf, BTVacState *vstate)
1803 : {
1804 : BlockNumber rightsib;
1805 : bool rightsib_empty;
1806 : Page page;
1807 : BTPageOpaque opaque;
1808 :
1809 : /*
1810 : * Save original leafbuf block number from caller. Only deleted blocks
1811 : * that are <= scanblkno are added to bulk delete stat's pages_deleted
1812 : * count.
1813 : */
1814 5690 : BlockNumber scanblkno = BufferGetBlockNumber(leafbuf);
1815 :
1816 : /*
1817 : * "stack" is a search stack leading (approximately) to the target page.
1818 : * It is initially NULL, but when iterating, we keep it to avoid
1819 : * duplicated search effort.
1820 : *
1821 : * Also, when "stack" is not NULL, we have already checked that the
1822 : * current page is not the right half of an incomplete split, i.e. the
1823 : * left sibling does not have its INCOMPLETE_SPLIT flag set, including
1824 : * when the current target page is to the right of caller's initial page
1825 : * (the scanblkno page).
1826 : */
1827 5690 : BTStack stack = NULL;
1828 :
1829 : for (;;)
1830 : {
1831 11178 : page = BufferGetPage(leafbuf);
1832 11178 : opaque = BTPageGetOpaque(page);
1833 :
1834 : /*
1835 : * Internal pages are never deleted directly, only as part of deleting
1836 : * the whole subtree all the way down to leaf level.
1837 : *
1838 : * Also check for deleted pages here. Caller never passes us a fully
1839 : * deleted page. Only VACUUM can delete pages, so there can't have
1840 : * been a concurrent deletion. Assume that we reached any deleted
1841 : * page encountered here by following a sibling link, and that the
1842 : * index is corrupt.
1843 : */
1844 : Assert(!P_ISDELETED(opaque));
1845 11178 : if (!P_ISLEAF(opaque) || P_ISDELETED(opaque))
1846 : {
1847 : /*
1848 : * Pre-9.4 page deletion only marked internal pages as half-dead,
1849 : * but now we only use that flag on leaf pages. The old algorithm
1850 : * was never supposed to leave half-dead pages in the tree, it was
1851 : * just a transient state, but it was nevertheless possible in
1852 : * error scenarios. We don't know how to deal with them here. They
1853 : * are harmless as far as searches are considered, but inserts
1854 : * into the deleted keyspace could add out-of-order downlinks in
1855 : * the upper levels. Log a notice, hopefully the admin will notice
1856 : * and reindex.
1857 : */
1858 0 : if (P_ISHALFDEAD(opaque))
1859 0 : ereport(LOG,
1860 : (errcode(ERRCODE_INDEX_CORRUPTED),
1861 : errmsg("index \"%s\" contains a half-dead internal page",
1862 : RelationGetRelationName(rel)),
1863 : errhint("This can be caused by an interrupted VACUUM in version 9.3 or older, before upgrade. Please REINDEX it.")));
1864 :
1865 0 : if (P_ISDELETED(opaque))
1866 0 : ereport(LOG,
1867 : (errcode(ERRCODE_INDEX_CORRUPTED),
1868 : errmsg_internal("found deleted block %u while following right link from block %u in index \"%s\"",
1869 : BufferGetBlockNumber(leafbuf),
1870 : scanblkno,
1871 : RelationGetRelationName(rel))));
1872 :
1873 0 : _bt_relbuf(rel, leafbuf);
1874 206 : return;
1875 : }
1876 :
1877 : /*
1878 : * We can never delete rightmost pages nor root pages. While at it,
1879 : * check that page is empty, since it's possible that the leafbuf page
1880 : * was empty a moment ago, but has since had some inserts.
1881 : *
1882 : * To keep the algorithm simple, we also never delete an incompletely
1883 : * split page (they should be rare enough that this doesn't make any
1884 : * meaningful difference to disk usage):
1885 : *
1886 : * The INCOMPLETE_SPLIT flag on the page tells us if the page is the
1887 : * left half of an incomplete split, but ensuring that it's not the
1888 : * right half is more complicated. For that, we have to check that
1889 : * the left sibling doesn't have its INCOMPLETE_SPLIT flag set using
1890 : * _bt_leftsib_splitflag(). On the first iteration, we temporarily
1891 : * release the lock on scanblkno/leafbuf, check the left sibling, and
1892 : * construct a search stack to scanblkno. On subsequent iterations,
1893 : * we know we stepped right from a page that passed these tests, so
1894 : * it's OK.
1895 : */
1896 11178 : if (P_RIGHTMOST(opaque) || P_ISROOT(opaque) ||
1897 10974 : P_FIRSTDATAKEY(opaque) <= PageGetMaxOffsetNumber(page) ||
1898 10974 : P_INCOMPLETE_SPLIT(opaque))
1899 : {
1900 : /* Should never fail to delete a half-dead page */
1901 : Assert(!P_ISHALFDEAD(opaque));
1902 :
1903 204 : _bt_relbuf(rel, leafbuf);
1904 204 : return;
1905 : }
1906 :
1907 : /*
1908 : * First, remove downlink pointing to the page (or a parent of the
1909 : * page, if we are going to delete a taller subtree), and mark the
1910 : * leafbuf page half-dead
1911 : */
1912 10974 : if (!P_ISHALFDEAD(opaque))
1913 : {
1914 : /*
1915 : * We need an approximate pointer to the page's parent page. We
1916 : * use a variant of the standard search mechanism to search for
1917 : * the page's high key; this will give us a link to either the
1918 : * current parent or someplace to its left (if there are multiple
1919 : * equal high keys, which is possible with !heapkeyspace indexes).
1920 : *
1921 : * Also check if this is the right-half of an incomplete split
1922 : * (see comment above).
1923 : */
1924 10974 : if (!stack)
1925 : {
1926 : BTScanInsert itup_key;
1927 : ItemId itemid;
1928 : IndexTuple targetkey;
1929 : BlockNumber leftsib,
1930 : leafblkno;
1931 : Buffer sleafbuf;
1932 :
1933 5486 : itemid = PageGetItemId(page, P_HIKEY);
1934 5486 : targetkey = CopyIndexTuple((IndexTuple) PageGetItem(page, itemid));
1935 :
1936 5486 : leftsib = opaque->btpo_prev;
1937 5486 : leafblkno = BufferGetBlockNumber(leafbuf);
1938 :
1939 : /*
1940 : * To avoid deadlocks, we'd better drop the leaf page lock
1941 : * before going further.
1942 : */
1943 5486 : _bt_unlockbuf(rel, leafbuf);
1944 :
1945 : /*
1946 : * Check that the left sibling of leafbuf (if any) is not
1947 : * marked with INCOMPLETE_SPLIT flag before proceeding
1948 : */
1949 : Assert(leafblkno == scanblkno);
1950 5486 : if (_bt_leftsib_splitflag(rel, leftsib, leafblkno))
1951 : {
1952 0 : ReleaseBuffer(leafbuf);
1953 0 : return;
1954 : }
1955 :
1956 : /*
1957 : * We need an insertion scan key, so build one.
1958 : *
1959 : * _bt_search searches for the leaf page that contains any
1960 : * matching non-pivot tuples, but we need it to "search" for
1961 : * the high key pivot from the page that we're set to delete.
1962 : * Compensate for the mismatch by having _bt_search locate the
1963 : * last position < equal-to-untruncated-prefix non-pivots.
1964 : */
1965 5486 : itup_key = _bt_mkscankey(rel, targetkey);
1966 :
1967 : /* Set up a BTLessStrategyNumber-like insertion scan key */
1968 5486 : itup_key->nextkey = false;
1969 5486 : itup_key->backward = true;
1970 5486 : stack = _bt_search(rel, NULL, itup_key, &sleafbuf, BT_READ);
1971 : /* won't need a second lock or pin on leafbuf */
1972 5486 : _bt_relbuf(rel, sleafbuf);
1973 :
1974 : /*
1975 : * Re-lock the leaf page, and start over to use our stack
1976 : * within _bt_mark_page_halfdead. We must do it that way
1977 : * because it's possible that leafbuf can no longer be
1978 : * deleted. We need to recheck.
1979 : *
1980 : * Note: We can't simply hold on to the sleafbuf lock instead,
1981 : * because it's barely possible that sleafbuf is not the same
1982 : * page as leafbuf. This happens when leafbuf split after our
1983 : * original lock was dropped, but before _bt_search finished
1984 : * its descent. We rely on the assumption that we'll find
1985 : * leafbuf isn't safe to delete anymore in this scenario.
1986 : * (Page deletion can cope with the stack being to the left of
1987 : * leafbuf, but not to the right of leafbuf.)
1988 : */
1989 5486 : _bt_lockbuf(rel, leafbuf, BT_WRITE);
1990 5486 : continue;
1991 : }
1992 :
1993 : /*
1994 : * See if it's safe to delete the leaf page, and determine how
1995 : * many parent/internal pages above the leaf level will be
1996 : * deleted. If it's safe then _bt_mark_page_halfdead will also
1997 : * perform the first phase of deletion, which includes marking the
1998 : * leafbuf page half-dead.
1999 : */
2000 : Assert(P_ISLEAF(opaque) && !P_IGNORE(opaque));
2001 5488 : if (!_bt_mark_page_halfdead(rel, vstate->info->heaprel, leafbuf,
2002 : stack))
2003 : {
2004 2 : _bt_relbuf(rel, leafbuf);
2005 2 : return;
2006 : }
2007 : }
2008 :
2009 : /*
2010 : * Then unlink it from its siblings. Each call to
2011 : * _bt_unlink_halfdead_page unlinks the topmost page from the subtree,
2012 : * making it shallower. Iterate until the leafbuf page is deleted.
2013 : */
2014 5486 : rightsib_empty = false;
2015 : Assert(P_ISLEAF(opaque) && P_ISHALFDEAD(opaque));
2016 11108 : while (P_ISHALFDEAD(opaque))
2017 : {
2018 : /* Check for interrupts in _bt_unlink_halfdead_page */
2019 5622 : if (!_bt_unlink_halfdead_page(rel, leafbuf, scanblkno,
2020 : &rightsib_empty, vstate))
2021 : {
2022 : /*
2023 : * _bt_unlink_halfdead_page should never fail, since we
2024 : * established that deletion is generally safe in
2025 : * _bt_mark_page_halfdead -- index must be corrupt.
2026 : *
2027 : * Note that _bt_unlink_halfdead_page already released the
2028 : * lock and pin on leafbuf for us.
2029 : */
2030 : Assert(false);
2031 0 : return;
2032 : }
2033 : }
2034 :
2035 : Assert(P_ISLEAF(opaque) && P_ISDELETED(opaque));
2036 :
2037 5486 : rightsib = opaque->btpo_next;
2038 :
2039 5486 : _bt_relbuf(rel, leafbuf);
2040 :
2041 : /*
2042 : * Check here, as calling loops will have locks held, preventing
2043 : * interrupts from being processed.
2044 : */
2045 5486 : CHECK_FOR_INTERRUPTS();
2046 :
2047 : /*
2048 : * The page has now been deleted. If its right sibling is completely
2049 : * empty, it's possible that the reason we haven't deleted it earlier
2050 : * is that it was the rightmost child of the parent. Now that we
2051 : * removed the downlink for this page, the right sibling might now be
2052 : * the only child of the parent, and could be removed. It would be
2053 : * picked up by the next vacuum anyway, but might as well try to
2054 : * remove it now, so loop back to process the right sibling.
2055 : *
2056 : * Note: This relies on the assumption that _bt_getstackbuf() will be
2057 : * able to reuse our original descent stack with a different child
2058 : * block (provided that the child block is to the right of the
2059 : * original leaf page reached by _bt_search()). It will even update
2060 : * the descent stack each time we loop around, avoiding repeated work.
2061 : */
2062 5486 : if (!rightsib_empty)
2063 5484 : break;
2064 :
2065 2 : leafbuf = _bt_getbuf(rel, rightsib, BT_WRITE);
2066 : }
2067 : }
2068 :
2069 : /*
2070 : * First stage of page deletion.
2071 : *
2072 : * Establish the height of the to-be-deleted subtree with leafbuf at its
2073 : * lowest level, remove the downlink to the subtree, and mark leafbuf
2074 : * half-dead. The final to-be-deleted subtree is usually just leafbuf itself,
2075 : * but may include additional internal pages (at most one per level of the
2076 : * tree below the root).
2077 : *
2078 : * Caller must pass a valid heaprel, since it's just about possible that our
2079 : * call to _bt_lock_subtree_parent will need to allocate a new index page to
2080 : * complete a page split. Every call to _bt_allocbuf needs to pass a heaprel.
2081 : *
2082 : * Returns 'false' if leafbuf is unsafe to delete, usually because leafbuf is
2083 : * the rightmost child of its parent (and parent has more than one downlink).
2084 : * Returns 'true' when the first stage of page deletion completed
2085 : * successfully.
2086 : */
2087 : static bool
2088 5488 : _bt_mark_page_halfdead(Relation rel, Relation heaprel, Buffer leafbuf,
2089 : BTStack stack)
2090 : {
2091 : BlockNumber leafblkno;
2092 : BlockNumber leafrightsib;
2093 : BlockNumber topparent;
2094 : BlockNumber topparentrightsib;
2095 : ItemId itemid;
2096 : Page page;
2097 : BTPageOpaque opaque;
2098 : Buffer subtreeparent;
2099 : OffsetNumber poffset;
2100 : OffsetNumber nextoffset;
2101 : IndexTuple itup;
2102 : IndexTupleData trunctuple;
2103 :
2104 5488 : page = BufferGetPage(leafbuf);
2105 5488 : opaque = BTPageGetOpaque(page);
2106 :
2107 : Assert(!P_RIGHTMOST(opaque) && !P_ISROOT(opaque) &&
2108 : P_ISLEAF(opaque) && !P_IGNORE(opaque) &&
2109 : P_FIRSTDATAKEY(opaque) > PageGetMaxOffsetNumber(page));
2110 : Assert(heaprel != NULL);
2111 :
2112 : /*
2113 : * Save info about the leaf page.
2114 : */
2115 5488 : leafblkno = BufferGetBlockNumber(leafbuf);
2116 5488 : leafrightsib = opaque->btpo_next;
2117 :
2118 : /*
2119 : * Before attempting to lock the parent page, check that the right sibling
2120 : * is not in half-dead state. A half-dead right sibling would have no
2121 : * downlink in the parent, which would be highly confusing later when we
2122 : * delete the downlink. It would fail the "right sibling of target page
2123 : * is also the next child in parent page" cross-check below.
2124 : */
2125 5488 : if (_bt_rightsib_halfdeadflag(rel, leafrightsib))
2126 : {
2127 0 : elog(DEBUG1, "could not delete page %u because its right sibling %u is half-dead",
2128 : leafblkno, leafrightsib);
2129 0 : return false;
2130 : }
2131 :
2132 : /*
2133 : * We cannot delete a page that is the rightmost child of its immediate
2134 : * parent, unless it is the only child --- in which case the parent has to
2135 : * be deleted too, and the same condition applies recursively to it. We
2136 : * have to check this condition all the way up before trying to delete,
2137 : * and lock the parent of the root of the to-be-deleted subtree (the
2138 : * "subtree parent"). _bt_lock_subtree_parent() locks the subtree parent
2139 : * for us. We remove the downlink to the "top parent" page (subtree root
2140 : * page) from the subtree parent page below.
2141 : *
2142 : * Initialize topparent to be leafbuf page now. The final to-be-deleted
2143 : * subtree is often a degenerate one page subtree consisting only of the
2144 : * leafbuf page. When that happens, the leafbuf page is the final subtree
2145 : * root page/top parent page.
2146 : */
2147 5488 : topparent = leafblkno;
2148 5488 : topparentrightsib = leafrightsib;
2149 5488 : if (!_bt_lock_subtree_parent(rel, heaprel, leafblkno, stack,
2150 : &subtreeparent, &poffset,
2151 : &topparent, &topparentrightsib))
2152 2 : return false;
2153 :
2154 5486 : page = BufferGetPage(subtreeparent);
2155 5486 : opaque = BTPageGetOpaque(page);
2156 :
2157 : #ifdef USE_ASSERT_CHECKING
2158 :
2159 : /*
2160 : * This is just an assertion because _bt_lock_subtree_parent should have
2161 : * guaranteed tuple has the expected contents
2162 : */
2163 : itemid = PageGetItemId(page, poffset);
2164 : itup = (IndexTuple) PageGetItem(page, itemid);
2165 : Assert(BTreeTupleGetDownLink(itup) == topparent);
2166 : #endif
2167 :
2168 5486 : nextoffset = OffsetNumberNext(poffset);
2169 5486 : itemid = PageGetItemId(page, nextoffset);
2170 5486 : itup = (IndexTuple) PageGetItem(page, itemid);
2171 :
2172 : /*
2173 : * Check that the parent-page index items we're about to delete/overwrite
2174 : * in subtree parent page contain what we expect. This can fail if the
2175 : * index has become corrupt for some reason. When that happens we back
2176 : * out of deletion of the leafbuf subtree. (This is just like the case
2177 : * where _bt_lock_subtree_parent() cannot "re-find" leafbuf's downlink.)
2178 : */
2179 5486 : if (BTreeTupleGetDownLink(itup) != topparentrightsib)
2180 : {
2181 0 : ereport(LOG,
2182 : (errcode(ERRCODE_INDEX_CORRUPTED),
2183 : errmsg_internal("right sibling %u of block %u is not next child %u of block %u in index \"%s\"",
2184 : topparentrightsib, topparent,
2185 : BTreeTupleGetDownLink(itup),
2186 : BufferGetBlockNumber(subtreeparent),
2187 : RelationGetRelationName(rel))));
2188 :
2189 0 : _bt_relbuf(rel, subtreeparent);
2190 : Assert(false);
2191 0 : return false;
2192 : }
2193 :
2194 : /*
2195 : * Any insert which would have gone on the leaf block will now go to its
2196 : * right sibling. In other words, the key space moves right.
2197 : */
2198 5486 : PredicateLockPageCombine(rel, leafblkno, leafrightsib);
2199 :
2200 : /* No ereport(ERROR) until changes are logged */
2201 5486 : START_CRIT_SECTION();
2202 :
2203 : /*
2204 : * Update parent of subtree. We want to delete the downlink to the top
2205 : * parent page/root of the subtree, and the *following* key. Easiest way
2206 : * is to copy the right sibling's downlink over the downlink that points
2207 : * to top parent page, and then delete the right sibling's original pivot
2208 : * tuple.
2209 : *
2210 : * Lanin and Shasha make the key space move left when deleting a page,
2211 : * whereas the key space moves right here. That's why we cannot simply
2212 : * delete the pivot tuple with the downlink to the top parent page. See
2213 : * nbtree/README.
2214 : */
2215 5486 : page = BufferGetPage(subtreeparent);
2216 5486 : opaque = BTPageGetOpaque(page);
2217 :
2218 5486 : itemid = PageGetItemId(page, poffset);
2219 5486 : itup = (IndexTuple) PageGetItem(page, itemid);
2220 5486 : BTreeTupleSetDownLink(itup, topparentrightsib);
2221 :
2222 5486 : nextoffset = OffsetNumberNext(poffset);
2223 5486 : PageIndexTupleDelete(page, nextoffset);
2224 :
2225 : /*
2226 : * Mark the leaf page as half-dead, and stamp it with a link to the top
2227 : * parent page. When the leaf page is also the top parent page, the link
2228 : * is set to InvalidBlockNumber.
2229 : */
2230 5486 : page = BufferGetPage(leafbuf);
2231 5486 : opaque = BTPageGetOpaque(page);
2232 5486 : opaque->btpo_flags |= BTP_HALF_DEAD;
2233 :
2234 : Assert(PageGetMaxOffsetNumber(page) == P_HIKEY);
2235 5486 : MemSet(&trunctuple, 0, sizeof(IndexTupleData));
2236 5486 : trunctuple.t_info = sizeof(IndexTupleData);
2237 5486 : if (topparent != leafblkno)
2238 70 : BTreeTupleSetTopParent(&trunctuple, topparent);
2239 : else
2240 5416 : BTreeTupleSetTopParent(&trunctuple, InvalidBlockNumber);
2241 :
2242 5486 : if (!PageIndexTupleOverwrite(page, P_HIKEY, (Item) &trunctuple,
2243 5486 : IndexTupleSize(&trunctuple)))
2244 0 : elog(ERROR, "could not overwrite high key in half-dead page");
2245 :
2246 : /* Must mark buffers dirty before XLogInsert */
2247 5486 : MarkBufferDirty(subtreeparent);
2248 5486 : MarkBufferDirty(leafbuf);
2249 :
2250 : /* XLOG stuff */
2251 5486 : if (RelationNeedsWAL(rel))
2252 : {
2253 : xl_btree_mark_page_halfdead xlrec;
2254 : XLogRecPtr recptr;
2255 :
2256 5486 : xlrec.poffset = poffset;
2257 5486 : xlrec.leafblk = leafblkno;
2258 5486 : if (topparent != leafblkno)
2259 70 : xlrec.topparent = topparent;
2260 : else
2261 5416 : xlrec.topparent = InvalidBlockNumber;
2262 :
2263 5486 : XLogBeginInsert();
2264 5486 : XLogRegisterBuffer(0, leafbuf, REGBUF_WILL_INIT);
2265 5486 : XLogRegisterBuffer(1, subtreeparent, REGBUF_STANDARD);
2266 :
2267 5486 : page = BufferGetPage(leafbuf);
2268 5486 : opaque = BTPageGetOpaque(page);
2269 5486 : xlrec.leftblk = opaque->btpo_prev;
2270 5486 : xlrec.rightblk = opaque->btpo_next;
2271 :
2272 5486 : XLogRegisterData((char *) &xlrec, SizeOfBtreeMarkPageHalfDead);
2273 :
2274 5486 : recptr = XLogInsert(RM_BTREE_ID, XLOG_BTREE_MARK_PAGE_HALFDEAD);
2275 :
2276 5486 : page = BufferGetPage(subtreeparent);
2277 5486 : PageSetLSN(page, recptr);
2278 5486 : page = BufferGetPage(leafbuf);
2279 5486 : PageSetLSN(page, recptr);
2280 : }
2281 :
2282 5486 : END_CRIT_SECTION();
2283 :
2284 5486 : _bt_relbuf(rel, subtreeparent);
2285 5486 : return true;
2286 : }
2287 :
2288 : /*
2289 : * Second stage of page deletion.
2290 : *
2291 : * Unlinks a single page (in the subtree undergoing deletion) from its
2292 : * siblings. Also marks the page deleted.
2293 : *
2294 : * To get rid of the whole subtree, including the leaf page itself, call here
2295 : * until the leaf page is deleted. The original "top parent" established in
2296 : * the first stage of deletion is deleted in the first call here, while the
2297 : * leaf page is deleted in the last call here. Note that the leaf page itself
2298 : * is often the initial top parent page.
2299 : *
2300 : * Returns 'false' if the page could not be unlinked (shouldn't happen). If
2301 : * the right sibling of the current target page is empty, *rightsib_empty is
2302 : * set to true, allowing caller to delete the target's right sibling page in
2303 : * passing. Note that *rightsib_empty is only actually used by caller when
2304 : * target page is leafbuf, following last call here for leafbuf/the subtree
2305 : * containing leafbuf. (We always set *rightsib_empty for caller, just to be
2306 : * consistent.)
2307 : *
2308 : * Must hold pin and lock on leafbuf at entry (read or write doesn't matter).
2309 : * On success exit, we'll be holding pin and write lock. On failure exit,
2310 : * we'll release both pin and lock before returning (we define it that way
2311 : * to avoid having to reacquire a lock we already released).
2312 : */
2313 : static bool
2314 5622 : _bt_unlink_halfdead_page(Relation rel, Buffer leafbuf, BlockNumber scanblkno,
2315 : bool *rightsib_empty, BTVacState *vstate)
2316 : {
2317 5622 : BlockNumber leafblkno = BufferGetBlockNumber(leafbuf);
2318 5622 : IndexBulkDeleteResult *stats = vstate->stats;
2319 : BlockNumber leafleftsib;
2320 : BlockNumber leafrightsib;
2321 : BlockNumber target;
2322 : BlockNumber leftsib;
2323 : BlockNumber rightsib;
2324 5622 : Buffer lbuf = InvalidBuffer;
2325 : Buffer buf;
2326 : Buffer rbuf;
2327 5622 : Buffer metabuf = InvalidBuffer;
2328 5622 : Page metapg = NULL;
2329 5622 : BTMetaPageData *metad = NULL;
2330 : ItemId itemid;
2331 : Page page;
2332 : BTPageOpaque opaque;
2333 : FullTransactionId safexid;
2334 : bool rightsib_is_rightmost;
2335 : uint32 targetlevel;
2336 : IndexTuple leafhikey;
2337 : BlockNumber leaftopparent;
2338 :
2339 5622 : page = BufferGetPage(leafbuf);
2340 5622 : opaque = BTPageGetOpaque(page);
2341 :
2342 : Assert(P_ISLEAF(opaque) && !P_ISDELETED(opaque) && P_ISHALFDEAD(opaque));
2343 :
2344 : /*
2345 : * Remember some information about the leaf page.
2346 : */
2347 5622 : itemid = PageGetItemId(page, P_HIKEY);
2348 5622 : leafhikey = (IndexTuple) PageGetItem(page, itemid);
2349 5622 : target = BTreeTupleGetTopParent(leafhikey);
2350 5622 : leafleftsib = opaque->btpo_prev;
2351 5622 : leafrightsib = opaque->btpo_next;
2352 :
2353 5622 : _bt_unlockbuf(rel, leafbuf);
2354 :
2355 : /*
2356 : * Check here, as calling loops will have locks held, preventing
2357 : * interrupts from being processed.
2358 : */
2359 5622 : CHECK_FOR_INTERRUPTS();
2360 :
2361 : /* Unlink the current top parent of the subtree */
2362 5622 : if (!BlockNumberIsValid(target))
2363 : {
2364 : /* Target is leaf page (or leaf page is top parent, if you prefer) */
2365 5486 : target = leafblkno;
2366 :
2367 5486 : buf = leafbuf;
2368 5486 : leftsib = leafleftsib;
2369 5486 : targetlevel = 0;
2370 : }
2371 : else
2372 : {
2373 : /* Target is the internal page taken from leaf's top parent link */
2374 : Assert(target != leafblkno);
2375 :
2376 : /* Fetch the block number of the target's left sibling */
2377 136 : buf = _bt_getbuf(rel, target, BT_READ);
2378 136 : page = BufferGetPage(buf);
2379 136 : opaque = BTPageGetOpaque(page);
2380 136 : leftsib = opaque->btpo_prev;
2381 136 : targetlevel = opaque->btpo_level;
2382 : Assert(targetlevel > 0);
2383 :
2384 : /*
2385 : * To avoid deadlocks, we'd better drop the target page lock before
2386 : * going further.
2387 : */
2388 136 : _bt_unlockbuf(rel, buf);
2389 : }
2390 :
2391 : /*
2392 : * We have to lock the pages we need to modify in the standard order:
2393 : * moving right, then up. Else we will deadlock against other writers.
2394 : *
2395 : * So, first lock the leaf page, if it's not the target. Then find and
2396 : * write-lock the current left sibling of the target page. The sibling
2397 : * that was current a moment ago could have split, so we may have to move
2398 : * right.
2399 : */
2400 5622 : if (target != leafblkno)
2401 136 : _bt_lockbuf(rel, leafbuf, BT_WRITE);
2402 5622 : if (leftsib != P_NONE)
2403 : {
2404 1172 : lbuf = _bt_getbuf(rel, leftsib, BT_WRITE);
2405 1172 : page = BufferGetPage(lbuf);
2406 1172 : opaque = BTPageGetOpaque(page);
2407 1172 : while (P_ISDELETED(opaque) || opaque->btpo_next != target)
2408 : {
2409 0 : bool leftsibvalid = true;
2410 :
2411 : /*
2412 : * Before we follow the link from the page that was the left
2413 : * sibling mere moments ago, validate its right link. This
2414 : * reduces the opportunities for loop to fail to ever make any
2415 : * progress in the presence of index corruption.
2416 : *
2417 : * Note: we rely on the assumption that there can only be one
2418 : * vacuum process running at a time (against the same index).
2419 : */
2420 0 : if (P_RIGHTMOST(opaque) || P_ISDELETED(opaque) ||
2421 0 : leftsib == opaque->btpo_next)
2422 0 : leftsibvalid = false;
2423 :
2424 0 : leftsib = opaque->btpo_next;
2425 0 : _bt_relbuf(rel, lbuf);
2426 :
2427 0 : if (!leftsibvalid)
2428 : {
2429 : /*
2430 : * This is known to fail in the field; sibling link corruption
2431 : * is relatively common. Press on with vacuuming rather than
2432 : * just throwing an ERROR.
2433 : */
2434 0 : ereport(LOG,
2435 : (errcode(ERRCODE_INDEX_CORRUPTED),
2436 : errmsg_internal("valid left sibling for deletion target could not be located: "
2437 : "left sibling %u of target %u with leafblkno %u and scanblkno %u on level %u of index \"%s\"",
2438 : leftsib, target, leafblkno, scanblkno,
2439 : targetlevel, RelationGetRelationName(rel))));
2440 :
2441 : /* Must release all pins and locks on failure exit */
2442 0 : ReleaseBuffer(buf);
2443 0 : if (target != leafblkno)
2444 0 : _bt_relbuf(rel, leafbuf);
2445 :
2446 0 : return false;
2447 : }
2448 :
2449 0 : CHECK_FOR_INTERRUPTS();
2450 :
2451 : /* step right one page */
2452 0 : lbuf = _bt_getbuf(rel, leftsib, BT_WRITE);
2453 0 : page = BufferGetPage(lbuf);
2454 0 : opaque = BTPageGetOpaque(page);
2455 : }
2456 : }
2457 : else
2458 4450 : lbuf = InvalidBuffer;
2459 :
2460 : /* Next write-lock the target page itself */
2461 5622 : _bt_lockbuf(rel, buf, BT_WRITE);
2462 5622 : page = BufferGetPage(buf);
2463 5622 : opaque = BTPageGetOpaque(page);
2464 :
2465 : /*
2466 : * Check page is still empty etc, else abandon deletion. This is just for
2467 : * paranoia's sake; a half-dead page cannot resurrect because there can be
2468 : * only one vacuum process running at a time.
2469 : */
2470 5622 : if (P_RIGHTMOST(opaque) || P_ISROOT(opaque) || P_ISDELETED(opaque))
2471 0 : elog(ERROR, "target page changed status unexpectedly in block %u of index \"%s\"",
2472 : target, RelationGetRelationName(rel));
2473 :
2474 5622 : if (opaque->btpo_prev != leftsib)
2475 0 : ereport(ERROR,
2476 : (errcode(ERRCODE_INDEX_CORRUPTED),
2477 : errmsg_internal("target page left link unexpectedly changed from %u to %u in block %u of index \"%s\"",
2478 : leftsib, opaque->btpo_prev, target,
2479 : RelationGetRelationName(rel))));
2480 :
2481 5622 : if (target == leafblkno)
2482 : {
2483 5486 : if (P_FIRSTDATAKEY(opaque) <= PageGetMaxOffsetNumber(page) ||
2484 5486 : !P_ISLEAF(opaque) || !P_ISHALFDEAD(opaque))
2485 0 : elog(ERROR, "target leaf page changed status unexpectedly in block %u of index \"%s\"",
2486 : target, RelationGetRelationName(rel));
2487 :
2488 : /* Leaf page is also target page: don't set leaftopparent */
2489 5486 : leaftopparent = InvalidBlockNumber;
2490 : }
2491 : else
2492 : {
2493 : IndexTuple finaldataitem;
2494 :
2495 136 : if (P_FIRSTDATAKEY(opaque) != PageGetMaxOffsetNumber(page) ||
2496 136 : P_ISLEAF(opaque))
2497 0 : elog(ERROR, "target internal page on level %u changed status unexpectedly in block %u of index \"%s\"",
2498 : targetlevel, target, RelationGetRelationName(rel));
2499 :
2500 : /* Target is internal: set leaftopparent for next call here... */
2501 136 : itemid = PageGetItemId(page, P_FIRSTDATAKEY(opaque));
2502 136 : finaldataitem = (IndexTuple) PageGetItem(page, itemid);
2503 136 : leaftopparent = BTreeTupleGetDownLink(finaldataitem);
2504 : /* ...except when it would be a redundant pointer-to-self */
2505 136 : if (leaftopparent == leafblkno)
2506 70 : leaftopparent = InvalidBlockNumber;
2507 : }
2508 :
2509 : /* No leaftopparent for level 0 (leaf page) or level 1 target */
2510 : Assert(!BlockNumberIsValid(leaftopparent) || targetlevel > 1);
2511 :
2512 : /*
2513 : * And next write-lock the (current) right sibling.
2514 : */
2515 5622 : rightsib = opaque->btpo_next;
2516 5622 : rbuf = _bt_getbuf(rel, rightsib, BT_WRITE);
2517 5622 : page = BufferGetPage(rbuf);
2518 5622 : opaque = BTPageGetOpaque(page);
2519 :
2520 : /*
2521 : * Validate target's right sibling page. Its left link must point back to
2522 : * the target page.
2523 : */
2524 5622 : if (opaque->btpo_prev != target)
2525 : {
2526 : /*
2527 : * This is known to fail in the field; sibling link corruption is
2528 : * relatively common. Press on with vacuuming rather than just
2529 : * throwing an ERROR (same approach used for left-sibling's-right-link
2530 : * validation check a moment ago).
2531 : */
2532 0 : ereport(LOG,
2533 : (errcode(ERRCODE_INDEX_CORRUPTED),
2534 : errmsg_internal("right sibling's left-link doesn't match: "
2535 : "right sibling %u of target %u with leafblkno %u "
2536 : "and scanblkno %u spuriously links to non-target %u "
2537 : "on level %u of index \"%s\"",
2538 : rightsib, target, leafblkno,
2539 : scanblkno, opaque->btpo_prev,
2540 : targetlevel, RelationGetRelationName(rel))));
2541 :
2542 : /* Must release all pins and locks on failure exit */
2543 0 : if (BufferIsValid(lbuf))
2544 0 : _bt_relbuf(rel, lbuf);
2545 0 : _bt_relbuf(rel, rbuf);
2546 0 : _bt_relbuf(rel, buf);
2547 0 : if (target != leafblkno)
2548 0 : _bt_relbuf(rel, leafbuf);
2549 :
2550 0 : return false;
2551 : }
2552 :
2553 5622 : rightsib_is_rightmost = P_RIGHTMOST(opaque);
2554 5622 : *rightsib_empty = (P_FIRSTDATAKEY(opaque) > PageGetMaxOffsetNumber(page));
2555 :
2556 : /*
2557 : * If we are deleting the next-to-last page on the target's level, then
2558 : * the rightsib is a candidate to become the new fast root. (In theory, it
2559 : * might be possible to push the fast root even further down, but the odds
2560 : * of doing so are slim, and the locking considerations daunting.)
2561 : *
2562 : * We can safely acquire a lock on the metapage here --- see comments for
2563 : * _bt_newlevel().
2564 : */
2565 5622 : if (leftsib == P_NONE && rightsib_is_rightmost)
2566 : {
2567 48 : page = BufferGetPage(rbuf);
2568 48 : opaque = BTPageGetOpaque(page);
2569 48 : if (P_RIGHTMOST(opaque))
2570 : {
2571 : /* rightsib will be the only one left on the level */
2572 48 : metabuf = _bt_getbuf(rel, BTREE_METAPAGE, BT_WRITE);
2573 48 : metapg = BufferGetPage(metabuf);
2574 48 : metad = BTPageGetMeta(metapg);
2575 :
2576 : /*
2577 : * The expected case here is btm_fastlevel == targetlevel+1; if
2578 : * the fastlevel is <= targetlevel, something is wrong, and we
2579 : * choose to overwrite it to fix it.
2580 : */
2581 48 : if (metad->btm_fastlevel > targetlevel + 1)
2582 : {
2583 : /* no update wanted */
2584 0 : _bt_relbuf(rel, metabuf);
2585 0 : metabuf = InvalidBuffer;
2586 : }
2587 : }
2588 : }
2589 :
2590 : /*
2591 : * Here we begin doing the deletion.
2592 : */
2593 :
2594 : /* No ereport(ERROR) until changes are logged */
2595 5622 : START_CRIT_SECTION();
2596 :
2597 : /*
2598 : * Update siblings' side-links. Note the target page's side-links will
2599 : * continue to point to the siblings. Asserts here are just rechecking
2600 : * things we already verified above.
2601 : */
2602 5622 : if (BufferIsValid(lbuf))
2603 : {
2604 1172 : page = BufferGetPage(lbuf);
2605 1172 : opaque = BTPageGetOpaque(page);
2606 : Assert(opaque->btpo_next == target);
2607 1172 : opaque->btpo_next = rightsib;
2608 : }
2609 5622 : page = BufferGetPage(rbuf);
2610 5622 : opaque = BTPageGetOpaque(page);
2611 : Assert(opaque->btpo_prev == target);
2612 5622 : opaque->btpo_prev = leftsib;
2613 :
2614 : /*
2615 : * If we deleted a parent of the targeted leaf page, instead of the leaf
2616 : * itself, update the leaf to point to the next remaining child in the
2617 : * subtree.
2618 : *
2619 : * Note: We rely on the fact that a buffer pin on the leaf page has been
2620 : * held since leafhikey was initialized. This is safe, though only
2621 : * because the page was already half-dead at that point. The leaf page
2622 : * cannot have been modified by any other backend during the period when
2623 : * no lock was held.
2624 : */
2625 5622 : if (target != leafblkno)
2626 136 : BTreeTupleSetTopParent(leafhikey, leaftopparent);
2627 :
2628 : /*
2629 : * Mark the page itself deleted. It can be recycled when all current
2630 : * transactions are gone. Storing GetTopTransactionId() would work, but
2631 : * we're in VACUUM and would not otherwise have an XID. Having already
2632 : * updated links to the target, ReadNextFullTransactionId() suffices as an
2633 : * upper bound. Any scan having retained a now-stale link is advertising
2634 : * in its PGPROC an xmin less than or equal to the value we read here. It
2635 : * will continue to do so, holding back the xmin horizon, for the duration
2636 : * of that scan.
2637 : */
2638 5622 : page = BufferGetPage(buf);
2639 5622 : opaque = BTPageGetOpaque(page);
2640 : Assert(P_ISHALFDEAD(opaque) || !P_ISLEAF(opaque));
2641 :
2642 : /*
2643 : * Store upper bound XID that's used to determine when deleted page is no
2644 : * longer needed as a tombstone
2645 : */
2646 5622 : safexid = ReadNextFullTransactionId();
2647 5622 : BTPageSetDeleted(page, safexid);
2648 5622 : opaque->btpo_cycleid = 0;
2649 :
2650 : /* And update the metapage, if needed */
2651 5622 : if (BufferIsValid(metabuf))
2652 : {
2653 : /* upgrade metapage if needed */
2654 48 : if (metad->btm_version < BTREE_NOVAC_VERSION)
2655 0 : _bt_upgrademetapage(metapg);
2656 48 : metad->btm_fastroot = rightsib;
2657 48 : metad->btm_fastlevel = targetlevel;
2658 48 : MarkBufferDirty(metabuf);
2659 : }
2660 :
2661 : /* Must mark buffers dirty before XLogInsert */
2662 5622 : MarkBufferDirty(rbuf);
2663 5622 : MarkBufferDirty(buf);
2664 5622 : if (BufferIsValid(lbuf))
2665 1172 : MarkBufferDirty(lbuf);
2666 5622 : if (target != leafblkno)
2667 136 : MarkBufferDirty(leafbuf);
2668 :
2669 : /* XLOG stuff */
2670 5622 : if (RelationNeedsWAL(rel))
2671 : {
2672 : xl_btree_unlink_page xlrec;
2673 : xl_btree_metadata xlmeta;
2674 : uint8 xlinfo;
2675 : XLogRecPtr recptr;
2676 :
2677 5622 : XLogBeginInsert();
2678 :
2679 5622 : XLogRegisterBuffer(0, buf, REGBUF_WILL_INIT);
2680 5622 : if (BufferIsValid(lbuf))
2681 1172 : XLogRegisterBuffer(1, lbuf, REGBUF_STANDARD);
2682 5622 : XLogRegisterBuffer(2, rbuf, REGBUF_STANDARD);
2683 5622 : if (target != leafblkno)
2684 136 : XLogRegisterBuffer(3, leafbuf, REGBUF_WILL_INIT);
2685 :
2686 : /* information stored on the target/to-be-unlinked block */
2687 5622 : xlrec.leftsib = leftsib;
2688 5622 : xlrec.rightsib = rightsib;
2689 5622 : xlrec.level = targetlevel;
2690 5622 : xlrec.safexid = safexid;
2691 :
2692 : /* information needed to recreate the leaf block (if not the target) */
2693 5622 : xlrec.leafleftsib = leafleftsib;
2694 5622 : xlrec.leafrightsib = leafrightsib;
2695 5622 : xlrec.leaftopparent = leaftopparent;
2696 :
2697 5622 : XLogRegisterData((char *) &xlrec, SizeOfBtreeUnlinkPage);
2698 :
2699 5622 : if (BufferIsValid(metabuf))
2700 : {
2701 48 : XLogRegisterBuffer(4, metabuf, REGBUF_WILL_INIT | REGBUF_STANDARD);
2702 :
2703 : Assert(metad->btm_version >= BTREE_NOVAC_VERSION);
2704 48 : xlmeta.version = metad->btm_version;
2705 48 : xlmeta.root = metad->btm_root;
2706 48 : xlmeta.level = metad->btm_level;
2707 48 : xlmeta.fastroot = metad->btm_fastroot;
2708 48 : xlmeta.fastlevel = metad->btm_fastlevel;
2709 48 : xlmeta.last_cleanup_num_delpages = metad->btm_last_cleanup_num_delpages;
2710 48 : xlmeta.allequalimage = metad->btm_allequalimage;
2711 :
2712 48 : XLogRegisterBufData(4, (char *) &xlmeta, sizeof(xl_btree_metadata));
2713 48 : xlinfo = XLOG_BTREE_UNLINK_PAGE_META;
2714 : }
2715 : else
2716 5574 : xlinfo = XLOG_BTREE_UNLINK_PAGE;
2717 :
2718 5622 : recptr = XLogInsert(RM_BTREE_ID, xlinfo);
2719 :
2720 5622 : if (BufferIsValid(metabuf))
2721 : {
2722 48 : PageSetLSN(metapg, recptr);
2723 : }
2724 5622 : page = BufferGetPage(rbuf);
2725 5622 : PageSetLSN(page, recptr);
2726 5622 : page = BufferGetPage(buf);
2727 5622 : PageSetLSN(page, recptr);
2728 5622 : if (BufferIsValid(lbuf))
2729 : {
2730 1172 : page = BufferGetPage(lbuf);
2731 1172 : PageSetLSN(page, recptr);
2732 : }
2733 5622 : if (target != leafblkno)
2734 : {
2735 136 : page = BufferGetPage(leafbuf);
2736 136 : PageSetLSN(page, recptr);
2737 : }
2738 : }
2739 :
2740 5622 : END_CRIT_SECTION();
2741 :
2742 : /* release metapage */
2743 5622 : if (BufferIsValid(metabuf))
2744 48 : _bt_relbuf(rel, metabuf);
2745 :
2746 : /* release siblings */
2747 5622 : if (BufferIsValid(lbuf))
2748 1172 : _bt_relbuf(rel, lbuf);
2749 5622 : _bt_relbuf(rel, rbuf);
2750 :
2751 : /* If the target is not leafbuf, we're done with it now -- release it */
2752 5622 : if (target != leafblkno)
2753 136 : _bt_relbuf(rel, buf);
2754 :
2755 : /*
2756 : * Maintain pages_newly_deleted, which is simply the number of pages
2757 : * deleted by the ongoing VACUUM operation.
2758 : *
2759 : * Maintain pages_deleted in a way that takes into account how
2760 : * btvacuumpage() will count deleted pages that have yet to become
2761 : * scanblkno -- only count page when it's not going to get that treatment
2762 : * later on.
2763 : */
2764 5622 : stats->pages_newly_deleted++;
2765 5622 : if (target <= scanblkno)
2766 5500 : stats->pages_deleted++;
2767 :
2768 : /*
2769 : * Remember information about the target page (now a newly deleted page)
2770 : * in dedicated vstate space for later. The page will be considered as a
2771 : * candidate to place in the FSM at the end of the current btvacuumscan()
2772 : * call.
2773 : */
2774 5622 : _bt_pendingfsm_add(vstate, target, safexid);
2775 :
2776 : /* Success - hold on to lock on leafbuf (might also have been target) */
2777 5622 : return true;
2778 : }
2779 :
2780 : /*
2781 : * Establish how tall the to-be-deleted subtree will be during the first stage
2782 : * of page deletion.
2783 : *
2784 : * Caller's child argument is the block number of the page caller wants to
2785 : * delete (this is leafbuf's block number, except when we're called
2786 : * recursively). stack is a search stack leading to it. Note that we will
2787 : * update the stack entry(s) to reflect current downlink positions --- this is
2788 : * similar to the corresponding point in page split handling.
2789 : *
2790 : * If "first stage" caller cannot go ahead with deleting _any_ pages, returns
2791 : * false. Returns true on success, in which case caller can use certain
2792 : * details established here to perform the first stage of deletion. This
2793 : * function is the last point at which page deletion may be deemed unsafe
2794 : * (barring index corruption, or unexpected concurrent page deletions).
2795 : *
2796 : * We write lock the parent of the root of the to-be-deleted subtree for
2797 : * caller on success (i.e. we leave our lock on the *subtreeparent buffer for
2798 : * caller). Caller will have to remove a downlink from *subtreeparent. We
2799 : * also set a *subtreeparent offset number in *poffset, to indicate the
2800 : * location of the pivot tuple that contains the relevant downlink.
2801 : *
2802 : * The root of the to-be-deleted subtree is called the "top parent". Note
2803 : * that the leafbuf page is often the final "top parent" page (you can think
2804 : * of the leafbuf page as a degenerate single page subtree when that happens).
2805 : * Caller should initialize *topparent to the target leafbuf page block number
2806 : * (while *topparentrightsib should be set to leafbuf's right sibling block
2807 : * number). We will update *topparent (and *topparentrightsib) for caller
2808 : * here, though only when it turns out that caller will delete at least one
2809 : * internal page (i.e. only when caller needs to store a valid link to the top
2810 : * parent block in the leafbuf page using BTreeTupleSetTopParent()).
2811 : */
2812 : static bool
2813 5628 : _bt_lock_subtree_parent(Relation rel, Relation heaprel, BlockNumber child,
2814 : BTStack stack, Buffer *subtreeparent,
2815 : OffsetNumber *poffset, BlockNumber *topparent,
2816 : BlockNumber *topparentrightsib)
2817 : {
2818 : BlockNumber parent,
2819 : leftsibparent;
2820 : OffsetNumber parentoffset,
2821 : maxoff;
2822 : Buffer pbuf;
2823 : Page page;
2824 : BTPageOpaque opaque;
2825 :
2826 : /*
2827 : * Locate the pivot tuple whose downlink points to "child". Write lock
2828 : * the parent page itself.
2829 : */
2830 5628 : pbuf = _bt_getstackbuf(rel, heaprel, stack, child);
2831 5628 : if (pbuf == InvalidBuffer)
2832 : {
2833 : /*
2834 : * Failed to "re-find" a pivot tuple whose downlink matched our child
2835 : * block number on the parent level -- the index must be corrupt.
2836 : * Don't even try to delete the leafbuf subtree. Just report the
2837 : * issue and press on with vacuuming the index.
2838 : *
2839 : * Note: _bt_getstackbuf() recovers from concurrent page splits that
2840 : * take place on the parent level. Its approach is a near-exhaustive
2841 : * linear search. This also gives it a surprisingly good chance of
2842 : * recovering in the event of a buggy or inconsistent opclass. But we
2843 : * don't rely on that here.
2844 : */
2845 0 : ereport(LOG,
2846 : (errcode(ERRCODE_INDEX_CORRUPTED),
2847 : errmsg_internal("failed to re-find parent key in index \"%s\" for deletion target page %u",
2848 : RelationGetRelationName(rel), child)));
2849 : Assert(false);
2850 0 : return false;
2851 : }
2852 :
2853 5628 : parent = stack->bts_blkno;
2854 5628 : parentoffset = stack->bts_offset;
2855 :
2856 5628 : page = BufferGetPage(pbuf);
2857 5628 : opaque = BTPageGetOpaque(page);
2858 5628 : maxoff = PageGetMaxOffsetNumber(page);
2859 5628 : leftsibparent = opaque->btpo_prev;
2860 :
2861 : /*
2862 : * _bt_getstackbuf() completes page splits on returned parent buffer when
2863 : * required.
2864 : *
2865 : * In general it's a bad idea for VACUUM to use up more disk space, which
2866 : * is why page deletion does not finish incomplete page splits most of the
2867 : * time. We allow this limited exception because the risk is much lower,
2868 : * and the potential downside of not proceeding is much higher: A single
2869 : * internal page with the INCOMPLETE_SPLIT flag set might otherwise
2870 : * prevent us from deleting hundreds of empty leaf pages from one level
2871 : * down.
2872 : */
2873 : Assert(!P_INCOMPLETE_SPLIT(opaque));
2874 :
2875 5628 : if (parentoffset < maxoff)
2876 : {
2877 : /*
2878 : * Child is not the rightmost child in parent, so it's safe to delete
2879 : * the subtree whose root/topparent is child page
2880 : */
2881 5486 : *subtreeparent = pbuf;
2882 5486 : *poffset = parentoffset;
2883 5486 : return true;
2884 : }
2885 :
2886 : /*
2887 : * Child is the rightmost child of parent.
2888 : *
2889 : * Since it's the rightmost child of parent, deleting the child (or
2890 : * deleting the subtree whose root/topparent is the child page) is only
2891 : * safe when it's also possible to delete the parent.
2892 : */
2893 : Assert(parentoffset == maxoff);
2894 142 : if (parentoffset != P_FIRSTDATAKEY(opaque) || P_RIGHTMOST(opaque))
2895 : {
2896 : /*
2897 : * Child isn't parent's only child, or parent is rightmost on its
2898 : * entire level. Definitely cannot delete any pages.
2899 : */
2900 2 : _bt_relbuf(rel, pbuf);
2901 2 : return false;
2902 : }
2903 :
2904 : /*
2905 : * Now make sure that the parent deletion is itself safe by examining the
2906 : * child's grandparent page. Recurse, passing the parent page as the
2907 : * child page (child's grandparent is the parent on the next level up). If
2908 : * parent deletion is unsafe, then child deletion must also be unsafe (in
2909 : * which case caller cannot delete any pages at all).
2910 : */
2911 140 : *topparent = parent;
2912 140 : *topparentrightsib = opaque->btpo_next;
2913 :
2914 : /*
2915 : * Release lock on parent before recursing.
2916 : *
2917 : * It's OK to release page locks on parent before recursive call locks
2918 : * grandparent. An internal page can only acquire an entry if the child
2919 : * is split, but that cannot happen as long as we still hold a lock on the
2920 : * leafbuf page.
2921 : */
2922 140 : _bt_relbuf(rel, pbuf);
2923 :
2924 : /*
2925 : * Before recursing, check that the left sibling of parent (if any) is not
2926 : * marked with INCOMPLETE_SPLIT flag first (must do so after we drop the
2927 : * parent lock).
2928 : *
2929 : * Note: We deliberately avoid completing incomplete splits here.
2930 : */
2931 140 : if (_bt_leftsib_splitflag(rel, leftsibparent, parent))
2932 0 : return false;
2933 :
2934 : /* Recurse to examine child page's grandparent page */
2935 140 : return _bt_lock_subtree_parent(rel, heaprel, parent, stack->bts_parent,
2936 : subtreeparent, poffset,
2937 : topparent, topparentrightsib);
2938 : }
2939 :
2940 : /*
2941 : * Initialize local memory state used by VACUUM for _bt_pendingfsm_finalize
2942 : * optimization.
2943 : *
2944 : * Called at the start of a btvacuumscan(). Caller's cleanuponly argument
2945 : * indicates if ongoing VACUUM has not (and will not) call btbulkdelete().
2946 : *
2947 : * We expect to allocate memory inside VACUUM's top-level memory context here.
2948 : * The working buffer is subject to a limit based on work_mem. Our strategy
2949 : * when the array can no longer grow within the bounds of that limit is to
2950 : * stop saving additional newly deleted pages, while proceeding as usual with
2951 : * the pages that we can fit.
2952 : */
2953 : void
2954 2566 : _bt_pendingfsm_init(Relation rel, BTVacState *vstate, bool cleanuponly)
2955 : {
2956 : int64 maxbufsize;
2957 :
2958 : /*
2959 : * Don't bother with optimization in cleanup-only case -- we don't expect
2960 : * any newly deleted pages. Besides, cleanup-only calls to btvacuumscan()
2961 : * can only take place because this optimization didn't work out during
2962 : * the last VACUUM.
2963 : */
2964 2566 : if (cleanuponly)
2965 12 : return;
2966 :
2967 : /*
2968 : * Cap maximum size of array so that we always respect work_mem. Avoid
2969 : * int overflow here.
2970 : */
2971 2554 : vstate->bufsize = 256;
2972 2554 : maxbufsize = (work_mem * 1024L) / sizeof(BTPendingFSM);
2973 2554 : maxbufsize = Min(maxbufsize, INT_MAX);
2974 2554 : maxbufsize = Min(maxbufsize, MaxAllocSize / sizeof(BTPendingFSM));
2975 : /* Stay sane with small work_mem */
2976 2554 : maxbufsize = Max(maxbufsize, vstate->bufsize);
2977 2554 : vstate->maxbufsize = maxbufsize;
2978 :
2979 : /* Allocate buffer, indicate that there are currently 0 pending pages */
2980 2554 : vstate->pendingpages = palloc(sizeof(BTPendingFSM) * vstate->bufsize);
2981 2554 : vstate->npendingpages = 0;
2982 : }
2983 :
2984 : /*
2985 : * Place any newly deleted pages (i.e. pages that _bt_pagedel() deleted during
2986 : * the ongoing VACUUM operation) into the free space map -- though only when
2987 : * it is actually safe to do so by now.
2988 : *
2989 : * Called at the end of a btvacuumscan(), just before free space map vacuuming
2990 : * takes place.
2991 : *
2992 : * Frees memory allocated by _bt_pendingfsm_init(), if any.
2993 : */
2994 : void
2995 2566 : _bt_pendingfsm_finalize(Relation rel, BTVacState *vstate)
2996 : {
2997 2566 : IndexBulkDeleteResult *stats = vstate->stats;
2998 2566 : Relation heaprel = vstate->info->heaprel;
2999 :
3000 : Assert(stats->pages_newly_deleted >= vstate->npendingpages);
3001 : Assert(heaprel != NULL);
3002 :
3003 2566 : if (vstate->npendingpages == 0)
3004 : {
3005 : /* Just free memory when nothing to do */
3006 2448 : if (vstate->pendingpages)
3007 2436 : pfree(vstate->pendingpages);
3008 :
3009 2448 : return;
3010 : }
3011 :
3012 : #ifdef DEBUG_BTREE_PENDING_FSM
3013 :
3014 : /*
3015 : * Debugging aid: Sleep for 5 seconds to greatly increase the chances of
3016 : * placing pending pages in the FSM. Note that the optimization will
3017 : * never be effective without some other backend concurrently consuming an
3018 : * XID.
3019 : */
3020 : pg_usleep(5000000L);
3021 : #endif
3022 :
3023 : /*
3024 : * Recompute VACUUM XID boundaries.
3025 : *
3026 : * We don't actually care about the oldest non-removable XID. Computing
3027 : * the oldest such XID has a useful side-effect that we rely on: it
3028 : * forcibly updates the XID horizon state for this backend. This step is
3029 : * essential; GlobalVisCheckRemovableFullXid() will not reliably recognize
3030 : * that it is now safe to recycle newly deleted pages without this step.
3031 : */
3032 118 : GetOldestNonRemovableTransactionId(heaprel);
3033 :
3034 140 : for (int i = 0; i < vstate->npendingpages; i++)
3035 : {
3036 140 : BlockNumber target = vstate->pendingpages[i].target;
3037 140 : FullTransactionId safexid = vstate->pendingpages[i].safexid;
3038 :
3039 : /*
3040 : * Do the equivalent of checking BTPageIsRecyclable(), but without
3041 : * accessing the page again a second time.
3042 : *
3043 : * Give up on finding the first non-recyclable page -- all later pages
3044 : * must be non-recyclable too, since _bt_pendingfsm_add() adds pages
3045 : * to the array in safexid order.
3046 : */
3047 140 : if (!GlobalVisCheckRemovableFullXid(heaprel, safexid))
3048 118 : break;
3049 :
3050 22 : RecordFreeIndexPage(rel, target);
3051 22 : stats->pages_free++;
3052 : }
3053 :
3054 118 : pfree(vstate->pendingpages);
3055 : }
3056 :
3057 : /*
3058 : * Maintain array of pages that were deleted during current btvacuumscan()
3059 : * call, for use in _bt_pendingfsm_finalize()
3060 : */
3061 : static void
3062 5622 : _bt_pendingfsm_add(BTVacState *vstate,
3063 : BlockNumber target,
3064 : FullTransactionId safexid)
3065 : {
3066 : Assert(vstate->npendingpages <= vstate->bufsize);
3067 : Assert(vstate->bufsize <= vstate->maxbufsize);
3068 :
3069 : #ifdef USE_ASSERT_CHECKING
3070 :
3071 : /*
3072 : * Verify an assumption made by _bt_pendingfsm_finalize(): pages from the
3073 : * array will always be in safexid order (since that is the order that we
3074 : * save them in here)
3075 : */
3076 : if (vstate->npendingpages > 0)
3077 : {
3078 : FullTransactionId lastsafexid =
3079 : vstate->pendingpages[vstate->npendingpages - 1].safexid;
3080 :
3081 : Assert(FullTransactionIdFollowsOrEquals(safexid, lastsafexid));
3082 : }
3083 : #endif
3084 :
3085 : /*
3086 : * If temp buffer reaches maxbufsize/work_mem capacity then we discard
3087 : * information about this page.
3088 : *
3089 : * Note that this also covers the case where we opted to not use the
3090 : * optimization in _bt_pendingfsm_init().
3091 : */
3092 5622 : if (vstate->npendingpages == vstate->maxbufsize)
3093 0 : return;
3094 :
3095 : /* Consider enlarging buffer */
3096 5622 : if (vstate->npendingpages == vstate->bufsize)
3097 : {
3098 8 : int newbufsize = vstate->bufsize * 2;
3099 :
3100 : /* Respect work_mem */
3101 8 : if (newbufsize > vstate->maxbufsize)
3102 0 : newbufsize = vstate->maxbufsize;
3103 :
3104 8 : vstate->bufsize = newbufsize;
3105 8 : vstate->pendingpages =
3106 8 : repalloc(vstate->pendingpages,
3107 8 : sizeof(BTPendingFSM) * vstate->bufsize);
3108 : }
3109 :
3110 : /* Save metadata for newly deleted page */
3111 5622 : vstate->pendingpages[vstate->npendingpages].target = target;
3112 5622 : vstate->pendingpages[vstate->npendingpages].safexid = safexid;
3113 5622 : vstate->npendingpages++;
3114 : }
|