Line data Source code
1 : /*--------------------------------------------------------------------------
2 : *
3 : * test_tidstore.c
4 : * Test TidStore data structure.
5 : *
6 : * Note: all locking in this test module is useless since there is only
7 : * a single process to use the TidStore. It is meant to be an example of
8 : * usage.
9 : *
10 : * Copyright (c) 2024-2025, PostgreSQL Global Development Group
11 : *
12 : * IDENTIFICATION
13 : * src/test/modules/test_tidstore/test_tidstore.c
14 : *
15 : * -------------------------------------------------------------------------
16 : */
17 : #include "postgres.h"
18 :
19 : #include "access/tidstore.h"
20 : #include "fmgr.h"
21 : #include "storage/block.h"
22 : #include "storage/itemptr.h"
23 : #include "storage/lwlock.h"
24 : #include "utils/array.h"
25 : #include "utils/memutils.h"
26 :
27 2 : PG_MODULE_MAGIC;
28 :
29 4 : PG_FUNCTION_INFO_V1(test_create);
30 4 : PG_FUNCTION_INFO_V1(do_set_block_offsets);
31 4 : PG_FUNCTION_INFO_V1(check_set_block_offsets);
32 4 : PG_FUNCTION_INFO_V1(test_is_full);
33 4 : PG_FUNCTION_INFO_V1(test_destroy);
34 :
35 : static TidStore *tidstore = NULL;
36 : static size_t tidstore_empty_size;
37 :
38 : /* array for verification of some tests */
39 : typedef struct ItemArray
40 : {
41 : ItemPointerData *insert_tids;
42 : ItemPointerData *lookup_tids;
43 : ItemPointerData *iter_tids;
44 : int max_tids;
45 : int num_tids;
46 : } ItemArray;
47 :
48 : static ItemArray items;
49 :
50 : /* comparator routine for ItemPointer */
51 : static int
52 416088 : itemptr_cmp(const void *left, const void *right)
53 : {
54 : BlockNumber lblk,
55 : rblk;
56 : OffsetNumber loff,
57 : roff;
58 :
59 416088 : lblk = ItemPointerGetBlockNumber((ItemPointer) left);
60 416088 : rblk = ItemPointerGetBlockNumber((ItemPointer) right);
61 :
62 416088 : if (lblk < rblk)
63 128168 : return -1;
64 287920 : if (lblk > rblk)
65 129020 : return 1;
66 :
67 158900 : loff = ItemPointerGetOffsetNumber((ItemPointer) left);
68 158900 : roff = ItemPointerGetOffsetNumber((ItemPointer) right);
69 :
70 158900 : if (loff < roff)
71 70216 : return -1;
72 88684 : if (loff > roff)
73 27684 : return 1;
74 :
75 61000 : return 0;
76 : }
77 :
78 : /*
79 : * Create a TidStore. If shared is false, the tidstore is created
80 : * on TopMemoryContext, otherwise on DSA. Although the tidstore
81 : * is created on DSA, only the same process can subsequently use
82 : * the tidstore. The tidstore handle is not shared anywhere.
83 : */
84 : Datum
85 6 : test_create(PG_FUNCTION_ARGS)
86 : {
87 6 : bool shared = PG_GETARG_BOOL(0);
88 : MemoryContext old_ctx;
89 :
90 : /* doesn't really matter, since it's just a hint */
91 6 : size_t tidstore_max_size = 2 * 1024 * 1024;
92 6 : size_t array_init_size = 1024;
93 :
94 : Assert(tidstore == NULL);
95 :
96 : /*
97 : * Create the TidStore on TopMemoryContext so that the same process use it
98 : * for subsequent tests.
99 : */
100 6 : old_ctx = MemoryContextSwitchTo(TopMemoryContext);
101 :
102 6 : if (shared)
103 : {
104 : int tranche_id;
105 :
106 2 : tranche_id = LWLockNewTrancheId();
107 2 : LWLockRegisterTranche(tranche_id, "test_tidstore");
108 :
109 2 : tidstore = TidStoreCreateShared(tidstore_max_size, tranche_id);
110 :
111 : /*
112 : * Remain attached until end of backend or explicitly detached so that
113 : * the same process use the tidstore for subsequent tests.
114 : */
115 2 : dsa_pin_mapping(TidStoreGetDSA(tidstore));
116 : }
117 : else
118 : /* VACUUM uses insert only, so we test the other option. */
119 4 : tidstore = TidStoreCreateLocal(tidstore_max_size, false);
120 :
121 6 : tidstore_empty_size = TidStoreMemoryUsage(tidstore);
122 :
123 6 : items.num_tids = 0;
124 6 : items.max_tids = array_init_size / sizeof(ItemPointerData);
125 6 : items.insert_tids = (ItemPointerData *) palloc0(array_init_size);
126 6 : items.lookup_tids = (ItemPointerData *) palloc0(array_init_size);
127 6 : items.iter_tids = (ItemPointerData *) palloc0(array_init_size);
128 :
129 6 : MemoryContextSwitchTo(old_ctx);
130 :
131 6 : PG_RETURN_VOID();
132 : }
133 :
134 : static void
135 2242 : sanity_check_array(ArrayType *ta)
136 : {
137 2242 : if (ARR_HASNULL(ta) && array_contains_nulls(ta))
138 0 : ereport(ERROR,
139 : (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
140 : errmsg("array must not contain nulls")));
141 :
142 2242 : if (ARR_NDIM(ta) > 1)
143 0 : ereport(ERROR,
144 : (errcode(ERRCODE_DATA_EXCEPTION),
145 : errmsg("argument must be empty or one-dimensional array")));
146 2242 : }
147 :
148 : static void
149 2276 : check_tidstore_available(void)
150 : {
151 2276 : if (tidstore == NULL)
152 0 : elog(ERROR, "tidstore is not created");
153 2276 : }
154 :
155 : static void
156 2240 : purge_from_verification_array(BlockNumber blkno)
157 : {
158 2240 : int dst = 0;
159 :
160 6987408 : for (int src = 0; src < items.num_tids; src++)
161 6985168 : if (ItemPointerGetBlockNumber(&items.insert_tids[src]) != blkno)
162 6985120 : items.insert_tids[dst++] = items.insert_tids[src];
163 2240 : items.num_tids = dst;
164 2240 : }
165 :
166 :
167 : /* Set the given block and offsets pairs */
168 : Datum
169 2242 : do_set_block_offsets(PG_FUNCTION_ARGS)
170 : {
171 2242 : BlockNumber blkno = PG_GETARG_INT64(0);
172 2242 : ArrayType *ta = PG_GETARG_ARRAYTYPE_P_COPY(1);
173 : OffsetNumber *offs;
174 : int noffs;
175 :
176 2242 : check_tidstore_available();
177 2242 : sanity_check_array(ta);
178 :
179 2242 : noffs = ArrayGetNItems(ARR_NDIM(ta), ARR_DIMS(ta));
180 2242 : offs = ((OffsetNumber *) ARR_DATA_PTR(ta));
181 :
182 : /* Set TIDs in the store */
183 2242 : TidStoreLockExclusive(tidstore);
184 2242 : TidStoreSetBlockOffsets(tidstore, blkno, offs, noffs);
185 2240 : TidStoreUnlock(tidstore);
186 :
187 : /* Remove the existing items of blkno from the verification array */
188 2240 : purge_from_verification_array(blkno);
189 :
190 : /* Set TIDs in verification array */
191 32740 : for (int i = 0; i < noffs; i++)
192 : {
193 : ItemPointer tid;
194 30500 : int idx = items.num_tids + i;
195 :
196 : /* Enlarge the TID arrays if necessary */
197 30500 : if (idx >= items.max_tids)
198 : {
199 24 : items.max_tids *= 2;
200 24 : items.insert_tids = repalloc(items.insert_tids, sizeof(ItemPointerData) * items.max_tids);
201 24 : items.lookup_tids = repalloc(items.lookup_tids, sizeof(ItemPointerData) * items.max_tids);
202 24 : items.iter_tids = repalloc(items.iter_tids, sizeof(ItemPointerData) * items.max_tids);
203 : }
204 :
205 30500 : tid = &(items.insert_tids[idx]);
206 30500 : ItemPointerSet(tid, blkno, offs[i]);
207 : }
208 :
209 : /* Update statistics */
210 2240 : items.num_tids += noffs;
211 :
212 2240 : PG_RETURN_INT64(blkno);
213 : }
214 :
215 : /*
216 : * Verify TIDs in store against the array.
217 : */
218 : Datum
219 24 : check_set_block_offsets(PG_FUNCTION_ARGS)
220 : {
221 : TidStoreIter *iter;
222 : TidStoreIterResult *iter_result;
223 24 : int num_iter_tids = 0;
224 24 : int num_lookup_tids = 0;
225 24 : BlockNumber prevblkno = 0;
226 :
227 24 : check_tidstore_available();
228 :
229 : /* lookup each member in the verification array */
230 30524 : for (int i = 0; i < items.num_tids; i++)
231 30500 : if (!TidStoreIsMember(tidstore, &items.insert_tids[i]))
232 0 : elog(ERROR, "missing TID with block %u, offset %u",
233 : ItemPointerGetBlockNumber(&items.insert_tids[i]),
234 : ItemPointerGetOffsetNumber(&items.insert_tids[i]));
235 :
236 : /*
237 : * Lookup all possible TIDs for each distinct block in the verification
238 : * array and save successful lookups in the lookup array.
239 : */
240 :
241 30524 : for (int i = 0; i < items.num_tids; i++)
242 : {
243 30500 : BlockNumber blkno = ItemPointerGetBlockNumber(&items.insert_tids[i]);
244 :
245 30500 : if (i > 0 && blkno == prevblkno)
246 28260 : continue;
247 :
248 4587520 : for (OffsetNumber offset = FirstOffsetNumber; offset < MaxOffsetNumber; offset++)
249 : {
250 : ItemPointerData tid;
251 :
252 4585280 : ItemPointerSet(&tid, blkno, offset);
253 :
254 4585280 : TidStoreLockShare(tidstore);
255 4585280 : if (TidStoreIsMember(tidstore, &tid))
256 30500 : ItemPointerSet(&items.lookup_tids[num_lookup_tids++], blkno, offset);
257 4585280 : TidStoreUnlock(tidstore);
258 : }
259 :
260 2240 : prevblkno = blkno;
261 : }
262 :
263 : /* Collect TIDs stored in the tidstore, in order */
264 :
265 24 : TidStoreLockShare(tidstore);
266 24 : iter = TidStoreBeginIterate(tidstore);
267 2264 : while ((iter_result = TidStoreIterateNext(iter)) != NULL)
268 : {
269 : OffsetNumber offsets[MaxOffsetNumber];
270 : int num_offsets;
271 :
272 2240 : num_offsets = TidStoreGetBlockOffsets(iter_result, offsets, lengthof(offsets));
273 : Assert(num_offsets <= lengthof(offsets));
274 32740 : for (int i = 0; i < num_offsets; i++)
275 30500 : ItemPointerSet(&(items.iter_tids[num_iter_tids++]), iter_result->blkno,
276 30500 : offsets[i]);
277 : }
278 24 : TidStoreEndIterate(iter);
279 24 : TidStoreUnlock(tidstore);
280 :
281 : /*
282 : * Sort verification and lookup arrays and test that all arrays are the
283 : * same.
284 : */
285 :
286 24 : if (num_lookup_tids != items.num_tids)
287 0 : elog(ERROR, "should have %d TIDs, have %d", items.num_tids, num_lookup_tids);
288 24 : if (num_iter_tids != items.num_tids)
289 0 : elog(ERROR, "should have %d TIDs, have %d", items.num_tids, num_iter_tids);
290 :
291 24 : qsort(items.insert_tids, items.num_tids, sizeof(ItemPointerData), itemptr_cmp);
292 24 : qsort(items.lookup_tids, items.num_tids, sizeof(ItemPointerData), itemptr_cmp);
293 30524 : for (int i = 0; i < items.num_tids; i++)
294 : {
295 30500 : if (itemptr_cmp(&items.insert_tids[i], &items.iter_tids[i]) != 0)
296 0 : elog(ERROR, "TID iter array doesn't match verification array, got (%u,%u) expected (%u,%u)",
297 : ItemPointerGetBlockNumber(&items.iter_tids[i]),
298 : ItemPointerGetOffsetNumber(&items.iter_tids[i]),
299 : ItemPointerGetBlockNumber(&items.insert_tids[i]),
300 : ItemPointerGetOffsetNumber(&items.insert_tids[i]));
301 30500 : if (itemptr_cmp(&items.insert_tids[i], &items.lookup_tids[i]) != 0)
302 0 : elog(ERROR, "TID lookup array doesn't match verification array, got (%u,%u) expected (%u,%u)",
303 : ItemPointerGetBlockNumber(&items.lookup_tids[i]),
304 : ItemPointerGetOffsetNumber(&items.lookup_tids[i]),
305 : ItemPointerGetBlockNumber(&items.insert_tids[i]),
306 : ItemPointerGetOffsetNumber(&items.insert_tids[i]));
307 : }
308 :
309 24 : PG_RETURN_VOID();
310 : }
311 :
312 : /*
313 : * In real world use, we care if the memory usage is greater than
314 : * some configured limit. Here we just want to verify that
315 : * TidStoreMemoryUsage is not broken.
316 : */
317 : Datum
318 4 : test_is_full(PG_FUNCTION_ARGS)
319 : {
320 : bool is_full;
321 :
322 4 : check_tidstore_available();
323 :
324 4 : is_full = (TidStoreMemoryUsage(tidstore) > tidstore_empty_size);
325 :
326 4 : PG_RETURN_BOOL(is_full);
327 : }
328 :
329 : /* Free the tidstore */
330 : Datum
331 6 : test_destroy(PG_FUNCTION_ARGS)
332 : {
333 6 : check_tidstore_available();
334 :
335 6 : TidStoreDestroy(tidstore);
336 6 : tidstore = NULL;
337 6 : items.num_tids = 0;
338 6 : pfree(items.insert_tids);
339 6 : pfree(items.lookup_tids);
340 6 : pfree(items.iter_tids);
341 :
342 6 : PG_RETURN_VOID();
343 : }
|