Line data Source code
1 : /*-------------------------------------------------------------------------
2 : *
3 : * spginsert.c
4 : * Externally visible index creation/insertion routines
5 : *
6 : * All the actual insertion logic is in spgdoinsert.c.
7 : *
8 : * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
9 : * Portions Copyright (c) 1994, Regents of the University of California
10 : *
11 : * IDENTIFICATION
12 : * src/backend/access/spgist/spginsert.c
13 : *
14 : *-------------------------------------------------------------------------
15 : */
16 :
17 : #include "postgres.h"
18 :
19 : #include "access/genam.h"
20 : #include "access/spgist_private.h"
21 : #include "access/tableam.h"
22 : #include "access/xloginsert.h"
23 : #include "miscadmin.h"
24 : #include "nodes/execnodes.h"
25 : #include "storage/bufmgr.h"
26 : #include "storage/bulk_write.h"
27 : #include "utils/memutils.h"
28 : #include "utils/rel.h"
29 :
30 :
31 : typedef struct
32 : {
33 : SpGistState spgstate; /* SPGiST's working state */
34 : int64 indtuples; /* total number of tuples indexed */
35 : MemoryContext tmpCtx; /* per-tuple temporary context */
36 : } SpGistBuildState;
37 :
38 :
39 : /* Callback to process one heap tuple during table_index_build_scan */
40 : static void
41 562880 : spgistBuildCallback(Relation index, ItemPointer tid, Datum *values,
42 : bool *isnull, bool tupleIsAlive, void *state)
43 : {
44 562880 : SpGistBuildState *buildstate = (SpGistBuildState *) state;
45 : MemoryContext oldCtx;
46 :
47 : /* Work in temp context, and reset it after each tuple */
48 562880 : oldCtx = MemoryContextSwitchTo(buildstate->tmpCtx);
49 :
50 : /*
51 : * Even though no concurrent insertions can be happening, we still might
52 : * get a buffer-locking failure due to bgwriter or checkpointer taking a
53 : * lock on some buffer. So we need to be willing to retry. We can flush
54 : * any temp data when retrying.
55 : */
56 562880 : while (!spgdoinsert(index, &buildstate->spgstate, tid,
57 : values, isnull))
58 : {
59 0 : MemoryContextReset(buildstate->tmpCtx);
60 : }
61 :
62 : /* Update total tuple count */
63 562880 : buildstate->indtuples += 1;
64 :
65 562880 : MemoryContextSwitchTo(oldCtx);
66 562880 : MemoryContextReset(buildstate->tmpCtx);
67 562880 : }
68 :
69 : /*
70 : * Build an SP-GiST index.
71 : */
72 : IndexBuildResult *
73 208 : spgbuild(Relation heap, Relation index, IndexInfo *indexInfo)
74 : {
75 : IndexBuildResult *result;
76 : double reltuples;
77 : SpGistBuildState buildstate;
78 : Buffer metabuffer,
79 : rootbuffer,
80 : nullbuffer;
81 :
82 208 : if (RelationGetNumberOfBlocks(index) != 0)
83 0 : elog(ERROR, "index \"%s\" already contains data",
84 : RelationGetRelationName(index));
85 :
86 : /*
87 : * Initialize the meta page and root pages
88 : */
89 208 : metabuffer = SpGistNewBuffer(index);
90 208 : rootbuffer = SpGistNewBuffer(index);
91 208 : nullbuffer = SpGistNewBuffer(index);
92 :
93 : Assert(BufferGetBlockNumber(metabuffer) == SPGIST_METAPAGE_BLKNO);
94 : Assert(BufferGetBlockNumber(rootbuffer) == SPGIST_ROOT_BLKNO);
95 : Assert(BufferGetBlockNumber(nullbuffer) == SPGIST_NULL_BLKNO);
96 :
97 208 : START_CRIT_SECTION();
98 :
99 208 : SpGistInitMetapage(BufferGetPage(metabuffer));
100 208 : MarkBufferDirty(metabuffer);
101 208 : SpGistInitBuffer(rootbuffer, SPGIST_LEAF);
102 208 : MarkBufferDirty(rootbuffer);
103 208 : SpGistInitBuffer(nullbuffer, SPGIST_LEAF | SPGIST_NULLS);
104 208 : MarkBufferDirty(nullbuffer);
105 :
106 :
107 208 : END_CRIT_SECTION();
108 :
109 208 : UnlockReleaseBuffer(metabuffer);
110 208 : UnlockReleaseBuffer(rootbuffer);
111 208 : UnlockReleaseBuffer(nullbuffer);
112 :
113 : /*
114 : * Now insert all the heap data into the index
115 : */
116 208 : initSpGistState(&buildstate.spgstate, index);
117 208 : buildstate.spgstate.isBuild = true;
118 208 : buildstate.indtuples = 0;
119 :
120 208 : buildstate.tmpCtx = AllocSetContextCreate(CurrentMemoryContext,
121 : "SP-GiST build temporary context",
122 : ALLOCSET_DEFAULT_SIZES);
123 :
124 208 : reltuples = table_index_build_scan(heap, index, indexInfo, true, true,
125 : spgistBuildCallback, &buildstate,
126 : NULL);
127 :
128 208 : MemoryContextDelete(buildstate.tmpCtx);
129 :
130 208 : SpGistUpdateMetaPage(index);
131 :
132 : /*
133 : * We didn't write WAL records as we built the index, so if WAL-logging is
134 : * required, write all pages to the WAL now.
135 : */
136 208 : if (RelationNeedsWAL(index))
137 : {
138 74 : log_newpage_range(index, MAIN_FORKNUM,
139 : 0, RelationGetNumberOfBlocks(index),
140 : true);
141 : }
142 :
143 208 : result = (IndexBuildResult *) palloc0(sizeof(IndexBuildResult));
144 208 : result->heap_tuples = reltuples;
145 208 : result->index_tuples = buildstate.indtuples;
146 :
147 208 : return result;
148 : }
149 :
150 : /*
151 : * Build an empty SPGiST index in the initialization fork
152 : */
153 : void
154 8 : spgbuildempty(Relation index)
155 : {
156 : BulkWriteState *bulkstate;
157 : BulkWriteBuffer buf;
158 :
159 8 : bulkstate = smgr_bulk_start_rel(index, INIT_FORKNUM);
160 :
161 : /* Construct metapage. */
162 8 : buf = smgr_bulk_get_buf(bulkstate);
163 8 : SpGistInitMetapage((Page) buf);
164 8 : smgr_bulk_write(bulkstate, SPGIST_METAPAGE_BLKNO, buf, true);
165 :
166 : /* Likewise for the root page. */
167 8 : buf = smgr_bulk_get_buf(bulkstate);
168 8 : SpGistInitPage((Page) buf, SPGIST_LEAF);
169 8 : smgr_bulk_write(bulkstate, SPGIST_ROOT_BLKNO, buf, true);
170 :
171 : /* Likewise for the null-tuples root page. */
172 8 : buf = smgr_bulk_get_buf(bulkstate);
173 8 : SpGistInitPage((Page) buf, SPGIST_LEAF | SPGIST_NULLS);
174 8 : smgr_bulk_write(bulkstate, SPGIST_NULL_BLKNO, buf, true);
175 :
176 8 : smgr_bulk_finish(bulkstate);
177 8 : }
178 :
179 : /*
180 : * Insert one new tuple into an SPGiST index.
181 : */
182 : bool
183 242766 : spginsert(Relation index, Datum *values, bool *isnull,
184 : ItemPointer ht_ctid, Relation heapRel,
185 : IndexUniqueCheck checkUnique,
186 : bool indexUnchanged,
187 : IndexInfo *indexInfo)
188 : {
189 : SpGistState spgstate;
190 : MemoryContext oldCtx;
191 : MemoryContext insertCtx;
192 :
193 242766 : insertCtx = AllocSetContextCreate(CurrentMemoryContext,
194 : "SP-GiST insert temporary context",
195 : ALLOCSET_DEFAULT_SIZES);
196 242766 : oldCtx = MemoryContextSwitchTo(insertCtx);
197 :
198 242766 : initSpGistState(&spgstate, index);
199 :
200 : /*
201 : * We might have to repeat spgdoinsert() multiple times, if conflicts
202 : * occur with concurrent insertions. If so, reset the insertCtx each time
203 : * to avoid cumulative memory consumption. That means we also have to
204 : * redo initSpGistState(), but it's cheap enough not to matter.
205 : */
206 242898 : while (!spgdoinsert(index, &spgstate, ht_ctid, values, isnull))
207 : {
208 132 : MemoryContextReset(insertCtx);
209 132 : initSpGistState(&spgstate, index);
210 : }
211 :
212 242762 : SpGistUpdateMetaPage(index);
213 :
214 242762 : MemoryContextSwitchTo(oldCtx);
215 242762 : MemoryContextDelete(insertCtx);
216 :
217 : /* return false since we've not done any unique check */
218 242762 : return false;
219 : }
|