PostgreSQL Source Code  git master
gistsplit.c
Go to the documentation of this file.
1 /*-------------------------------------------------------------------------
2  *
3  * gistsplit.c
4  * Multi-column page splitting algorithm
5  *
6  * This file is concerned with making good page-split decisions in multi-column
7  * GiST indexes. The opclass-specific picksplit functions can only be expected
8  * to produce answers based on a single column. We first run the picksplit
9  * function for column 1; then, if there are more columns, we check if any of
10  * the tuples are "don't cares" so far as the column 1 split is concerned
11  * (that is, they could go to either side for no additional penalty). If so,
12  * we try to redistribute those tuples on the basis of the next column.
13  * Repeat till we're out of columns.
14  *
15  * gistSplitByKey() is the entry point to this file.
16  *
17  *
18  * Portions Copyright (c) 1996-2024, PostgreSQL Global Development Group
19  * Portions Copyright (c) 1994, Regents of the University of California
20  *
21  * IDENTIFICATION
22  * src/backend/access/gist/gistsplit.c
23  *
24  *-------------------------------------------------------------------------
25  */
26 #include "postgres.h"
27 
28 #include "access/gist_private.h"
29 #include "utils/rel.h"
30 
31 typedef struct
32 {
34  int len;
36  bool *isnull;
37  bool *dontcare;
39 
40 
41 /*
42  * Form unions of subkeys in itvec[] entries listed in gsvp->entries[],
43  * ignoring any tuples that are marked in gsvp->dontcare[]. Subroutine for
44  * gistunionsubkey.
45  */
46 static void
48  GistSplitUnion *gsvp)
49 {
50  IndexTuple *cleanedItVec;
51  int i,
52  cleanedLen = 0;
53 
54  cleanedItVec = (IndexTuple *) palloc(sizeof(IndexTuple) * gsvp->len);
55 
56  for (i = 0; i < gsvp->len; i++)
57  {
58  if (gsvp->dontcare && gsvp->dontcare[gsvp->entries[i]])
59  continue;
60 
61  cleanedItVec[cleanedLen++] = itvec[gsvp->entries[i] - 1];
62  }
63 
64  gistMakeUnionItVec(giststate, cleanedItVec, cleanedLen,
65  gsvp->attr, gsvp->isnull);
66 
67  pfree(cleanedItVec);
68 }
69 
70 /*
71  * Recompute unions of left- and right-side subkeys after a page split,
72  * ignoring any tuples that are marked in spl->spl_dontcare[].
73  *
74  * Note: we always recompute union keys for all index columns. In some cases
75  * this might represent duplicate work for the leftmost column(s), but it's
76  * not safe to assume that "zero penalty to move a tuple" means "the union
77  * key doesn't change at all". Penalty functions aren't 100% accurate.
78  */
79 static void
81 {
82  GistSplitUnion gsvp;
83 
84  gsvp.dontcare = spl->spl_dontcare;
85 
86  gsvp.entries = spl->splitVector.spl_left;
87  gsvp.len = spl->splitVector.spl_nleft;
88  gsvp.attr = spl->spl_lattr;
89  gsvp.isnull = spl->spl_lisnull;
90 
91  gistunionsubkeyvec(giststate, itvec, &gsvp);
92 
93  gsvp.entries = spl->splitVector.spl_right;
94  gsvp.len = spl->splitVector.spl_nright;
95  gsvp.attr = spl->spl_rattr;
96  gsvp.isnull = spl->spl_risnull;
97 
98  gistunionsubkeyvec(giststate, itvec, &gsvp);
99 }
100 
101 /*
102  * Find tuples that are "don't cares", that is could be moved to the other
103  * side of the split with zero penalty, so far as the attno column is
104  * concerned.
105  *
106  * Don't-care tuples are marked by setting the corresponding entry in
107  * spl->spl_dontcare[] to "true". Caller must have initialized that array
108  * to zeroes.
109  *
110  * Returns number of don't-cares found.
111  */
112 static int
113 findDontCares(Relation r, GISTSTATE *giststate, GISTENTRY *valvec,
114  GistSplitVector *spl, int attno)
115 {
116  int i;
117  GISTENTRY entry;
118  int NumDontCare = 0;
119 
120  /*
121  * First, search the left-side tuples to see if any have zero penalty to
122  * be added to the right-side union key.
123  *
124  * attno column is known all-not-null (see gistSplitByKey), so we need not
125  * check for nulls
126  */
127  gistentryinit(entry, spl->splitVector.spl_rdatum, r, NULL,
128  (OffsetNumber) 0, false);
129  for (i = 0; i < spl->splitVector.spl_nleft; i++)
130  {
131  int j = spl->splitVector.spl_left[i];
132  float penalty = gistpenalty(giststate, attno, &entry, false,
133  &valvec[j], false);
134 
135  if (penalty == 0.0)
136  {
137  spl->spl_dontcare[j] = true;
138  NumDontCare++;
139  }
140  }
141 
142  /* And conversely for the right-side tuples */
143  gistentryinit(entry, spl->splitVector.spl_ldatum, r, NULL,
144  (OffsetNumber) 0, false);
145  for (i = 0; i < spl->splitVector.spl_nright; i++)
146  {
147  int j = spl->splitVector.spl_right[i];
148  float penalty = gistpenalty(giststate, attno, &entry, false,
149  &valvec[j], false);
150 
151  if (penalty == 0.0)
152  {
153  spl->spl_dontcare[j] = true;
154  NumDontCare++;
155  }
156  }
157 
158  return NumDontCare;
159 }
160 
161 /*
162  * Remove tuples that are marked don't-cares from the tuple index array a[]
163  * of length *len. This is applied separately to the spl_left and spl_right
164  * arrays.
165  */
166 static void
167 removeDontCares(OffsetNumber *a, int *len, const bool *dontcare)
168 {
169  int origlen,
170  newlen,
171  i;
172  OffsetNumber *curwpos;
173 
174  origlen = newlen = *len;
175  curwpos = a;
176  for (i = 0; i < origlen; i++)
177  {
178  OffsetNumber ai = a[i];
179 
180  if (dontcare[ai] == false)
181  {
182  /* re-emit item into a[] */
183  *curwpos = ai;
184  curwpos++;
185  }
186  else
187  newlen--;
188  }
189 
190  *len = newlen;
191 }
192 
193 /*
194  * Place a single don't-care tuple into either the left or right side of the
195  * split, according to which has least penalty for merging the tuple into
196  * the previously-computed union keys. We need consider only columns starting
197  * at attno.
198  */
199 static void
201  IndexTuple itup, OffsetNumber off, int attno)
202 {
203  GISTENTRY identry[INDEX_MAX_KEYS];
204  bool isnull[INDEX_MAX_KEYS];
205  bool toLeft = true;
206 
207  gistDeCompressAtt(giststate, r, itup, NULL, (OffsetNumber) 0,
208  identry, isnull);
209 
210  for (; attno < giststate->nonLeafTupdesc->natts; attno++)
211  {
212  float lpenalty,
213  rpenalty;
214  GISTENTRY entry;
215 
216  gistentryinit(entry, v->spl_lattr[attno], r, NULL, 0, false);
217  lpenalty = gistpenalty(giststate, attno, &entry, v->spl_lisnull[attno],
218  identry + attno, isnull[attno]);
219  gistentryinit(entry, v->spl_rattr[attno], r, NULL, 0, false);
220  rpenalty = gistpenalty(giststate, attno, &entry, v->spl_risnull[attno],
221  identry + attno, isnull[attno]);
222 
223  if (lpenalty != rpenalty)
224  {
225  if (lpenalty > rpenalty)
226  toLeft = false;
227  break;
228  }
229  }
230 
231  if (toLeft)
232  v->splitVector.spl_left[v->splitVector.spl_nleft++] = off;
233  else
235 }
236 
237 #define SWAPVAR( s, d, t ) \
238 do { \
239  (t) = (s); \
240  (s) = (d); \
241  (d) = (t); \
242 } while(0)
243 
244 /*
245  * Clean up when we did a secondary split but the user-defined PickSplit
246  * method didn't support it (leaving spl_ldatum_exists or spl_rdatum_exists
247  * true).
248  *
249  * We consider whether to swap the left and right outputs of the secondary
250  * split; this can be worthwhile if the penalty for merging those tuples into
251  * the previously chosen sets is less that way.
252  *
253  * In any case we must update the union datums for the current column by
254  * adding in the previous union keys (oldL/oldR), since the user-defined
255  * PickSplit method didn't do so.
256  */
257 static void
258 supportSecondarySplit(Relation r, GISTSTATE *giststate, int attno,
259  GIST_SPLITVEC *sv, Datum oldL, Datum oldR)
260 {
261  bool leaveOnLeft = true,
262  tmpBool;
263  GISTENTRY entryL,
264  entryR,
265  entrySL,
266  entrySR;
267 
268  gistentryinit(entryL, oldL, r, NULL, 0, false);
269  gistentryinit(entryR, oldR, r, NULL, 0, false);
270  gistentryinit(entrySL, sv->spl_ldatum, r, NULL, 0, false);
271  gistentryinit(entrySR, sv->spl_rdatum, r, NULL, 0, false);
272 
273  if (sv->spl_ldatum_exists && sv->spl_rdatum_exists)
274  {
275  float penalty1,
276  penalty2;
277 
278  penalty1 = gistpenalty(giststate, attno, &entryL, false, &entrySL, false) +
279  gistpenalty(giststate, attno, &entryR, false, &entrySR, false);
280  penalty2 = gistpenalty(giststate, attno, &entryL, false, &entrySR, false) +
281  gistpenalty(giststate, attno, &entryR, false, &entrySL, false);
282 
283  if (penalty1 > penalty2)
284  leaveOnLeft = false;
285  }
286  else
287  {
288  GISTENTRY *entry1 = (sv->spl_ldatum_exists) ? &entryL : &entryR;
289  float penalty1,
290  penalty2;
291 
292  /*
293  * There is only one previously defined union, so we just choose swap
294  * or not by lowest penalty for that side. We can only get here if a
295  * secondary split happened to have all NULLs in its column in the
296  * tuples that the outer recursion level had assigned to one side.
297  * (Note that the null checks in gistSplitByKey don't prevent the
298  * case, because they'll only be checking tuples that were considered
299  * don't-cares at the outer recursion level, not the tuples that went
300  * into determining the passed-down left and right union keys.)
301  */
302  penalty1 = gistpenalty(giststate, attno, entry1, false, &entrySL, false);
303  penalty2 = gistpenalty(giststate, attno, entry1, false, &entrySR, false);
304 
305  if (penalty1 < penalty2)
306  leaveOnLeft = sv->spl_ldatum_exists;
307  else
308  leaveOnLeft = sv->spl_rdatum_exists;
309  }
310 
311  if (leaveOnLeft == false)
312  {
313  /*
314  * swap left and right
315  */
316  OffsetNumber *off,
317  noff;
318  Datum datum;
319 
320  SWAPVAR(sv->spl_left, sv->spl_right, off);
321  SWAPVAR(sv->spl_nleft, sv->spl_nright, noff);
322  SWAPVAR(sv->spl_ldatum, sv->spl_rdatum, datum);
323  gistentryinit(entrySL, sv->spl_ldatum, r, NULL, 0, false);
324  gistentryinit(entrySR, sv->spl_rdatum, r, NULL, 0, false);
325  }
326 
327  if (sv->spl_ldatum_exists)
328  gistMakeUnionKey(giststate, attno, &entryL, false, &entrySL, false,
329  &sv->spl_ldatum, &tmpBool);
330 
331  if (sv->spl_rdatum_exists)
332  gistMakeUnionKey(giststate, attno, &entryR, false, &entrySR, false,
333  &sv->spl_rdatum, &tmpBool);
334 
335  sv->spl_ldatum_exists = sv->spl_rdatum_exists = false;
336 }
337 
338 /*
339  * Trivial picksplit implementation. Function called only
340  * if user-defined picksplit puts all keys on the same side of the split.
341  * That is a bug of user-defined picksplit but we don't want to fail.
342  */
343 static void
344 genericPickSplit(GISTSTATE *giststate, GistEntryVector *entryvec, GIST_SPLITVEC *v, int attno)
345 {
346  OffsetNumber i,
347  maxoff;
348  int nbytes;
349  GistEntryVector *evec;
350 
351  maxoff = entryvec->n - 1;
352 
353  nbytes = (maxoff + 2) * sizeof(OffsetNumber);
354 
355  v->spl_left = (OffsetNumber *) palloc(nbytes);
356  v->spl_right = (OffsetNumber *) palloc(nbytes);
357  v->spl_nleft = v->spl_nright = 0;
358 
359  for (i = FirstOffsetNumber; i <= maxoff; i = OffsetNumberNext(i))
360  {
361  if (i <= (maxoff - FirstOffsetNumber + 1) / 2)
362  {
363  v->spl_left[v->spl_nleft] = i;
364  v->spl_nleft++;
365  }
366  else
367  {
368  v->spl_right[v->spl_nright] = i;
369  v->spl_nright++;
370  }
371  }
372 
373  /*
374  * Form union datums for each side
375  */
376  evec = palloc(sizeof(GISTENTRY) * entryvec->n + GEVHDRSZ);
377 
378  evec->n = v->spl_nleft;
379  memcpy(evec->vector, entryvec->vector + FirstOffsetNumber,
380  sizeof(GISTENTRY) * evec->n);
381  v->spl_ldatum = FunctionCall2Coll(&giststate->unionFn[attno],
382  giststate->supportCollation[attno],
383  PointerGetDatum(evec),
384  PointerGetDatum(&nbytes));
385 
386  evec->n = v->spl_nright;
387  memcpy(evec->vector, entryvec->vector + FirstOffsetNumber + v->spl_nleft,
388  sizeof(GISTENTRY) * evec->n);
389  v->spl_rdatum = FunctionCall2Coll(&giststate->unionFn[attno],
390  giststate->supportCollation[attno],
391  PointerGetDatum(evec),
392  PointerGetDatum(&nbytes));
393 }
394 
395 /*
396  * Calls user picksplit method for attno column to split tuples into
397  * two vectors.
398  *
399  * Returns false if split is complete (there are no more index columns, or
400  * there is no need to consider them because split is optimal already).
401  *
402  * Returns true and v->spl_dontcare = NULL if the picksplit result is
403  * degenerate (all tuples seem to be don't-cares), so we should just
404  * disregard this column and split on the next column(s) instead.
405  *
406  * Returns true and v->spl_dontcare != NULL if there are don't-care tuples
407  * that could be relocated based on the next column(s). The don't-care
408  * tuples have been removed from the split and must be reinserted by caller.
409  * There is at least one non-don't-care tuple on each side of the split,
410  * and union keys for all columns are updated to include just those tuples.
411  *
412  * A true result implies there is at least one more index column.
413  */
414 static bool
416  IndexTuple *itup, int len, GISTSTATE *giststate)
417 {
418  GIST_SPLITVEC *sv = &v->splitVector;
419 
420  /*
421  * Prepare spl_ldatum/spl_rdatum/spl_ldatum_exists/spl_rdatum_exists in
422  * case we are doing a secondary split (see comments in gist.h).
423  */
424  sv->spl_ldatum_exists = !(v->spl_lisnull[attno]);
425  sv->spl_rdatum_exists = !(v->spl_risnull[attno]);
426  sv->spl_ldatum = v->spl_lattr[attno];
427  sv->spl_rdatum = v->spl_rattr[attno];
428 
429  /*
430  * Let the opclass-specific PickSplit method do its thing. Note that at
431  * this point we know there are no null keys in the entryvec.
432  */
433  FunctionCall2Coll(&giststate->picksplitFn[attno],
434  giststate->supportCollation[attno],
435  PointerGetDatum(entryvec),
436  PointerGetDatum(sv));
437 
438  if (sv->spl_nleft == 0 || sv->spl_nright == 0)
439  {
440  /*
441  * User-defined picksplit failed to create an actual split, ie it put
442  * everything on the same side. Complain but cope.
443  */
444  ereport(DEBUG1,
445  (errcode(ERRCODE_INTERNAL_ERROR),
446  errmsg("picksplit method for column %d of index \"%s\" failed",
447  attno + 1, RelationGetRelationName(r)),
448  errhint("The index is not optimal. To optimize it, contact a developer, or try to use the column as the second one in the CREATE INDEX command.")));
449 
450  /*
451  * Reinit GIST_SPLITVEC. Although these fields are not used by
452  * genericPickSplit(), set them up for further processing
453  */
454  sv->spl_ldatum_exists = !(v->spl_lisnull[attno]);
455  sv->spl_rdatum_exists = !(v->spl_risnull[attno]);
456  sv->spl_ldatum = v->spl_lattr[attno];
457  sv->spl_rdatum = v->spl_rattr[attno];
458 
459  /* Do a generic split */
460  genericPickSplit(giststate, entryvec, sv, attno);
461  }
462  else
463  {
464  /* hack for compatibility with old picksplit API */
465  if (sv->spl_left[sv->spl_nleft - 1] == InvalidOffsetNumber)
466  sv->spl_left[sv->spl_nleft - 1] = (OffsetNumber) (entryvec->n - 1);
467  if (sv->spl_right[sv->spl_nright - 1] == InvalidOffsetNumber)
468  sv->spl_right[sv->spl_nright - 1] = (OffsetNumber) (entryvec->n - 1);
469  }
470 
471  /* Clean up if PickSplit didn't take care of a secondary split */
472  if (sv->spl_ldatum_exists || sv->spl_rdatum_exists)
473  supportSecondarySplit(r, giststate, attno, sv,
474  v->spl_lattr[attno], v->spl_rattr[attno]);
475 
476  /* emit union datums computed by PickSplit back to v arrays */
477  v->spl_lattr[attno] = sv->spl_ldatum;
478  v->spl_rattr[attno] = sv->spl_rdatum;
479  v->spl_lisnull[attno] = false;
480  v->spl_risnull[attno] = false;
481 
482  /*
483  * If index columns remain, then consider whether we can improve the split
484  * by using them.
485  */
486  v->spl_dontcare = NULL;
487 
488  if (attno + 1 < giststate->nonLeafTupdesc->natts)
489  {
490  int NumDontCare;
491 
492  /*
493  * Make a quick check to see if left and right union keys are equal;
494  * if so, the split is certainly degenerate, so tell caller to
495  * re-split with the next column.
496  */
497  if (gistKeyIsEQ(giststate, attno, sv->spl_ldatum, sv->spl_rdatum))
498  return true;
499 
500  /*
501  * Locate don't-care tuples, if any. If there are none, the split is
502  * optimal, so just fall out and return false.
503  */
504  v->spl_dontcare = (bool *) palloc0(sizeof(bool) * (entryvec->n + 1));
505 
506  NumDontCare = findDontCares(r, giststate, entryvec->vector, v, attno);
507 
508  if (NumDontCare > 0)
509  {
510  /*
511  * Remove don't-cares from spl_left[] and spl_right[].
512  */
515 
516  /*
517  * If all tuples on either side were don't-cares, the split is
518  * degenerate, and we're best off to ignore it and split on the
519  * next column. (We used to try to press on with a secondary
520  * split by forcing a random tuple on each side to be treated as
521  * non-don't-care, but it seems unlikely that that technique
522  * really gives a better result. Note that we don't want to try a
523  * secondary split with empty left or right primary split sides,
524  * because then there is no union key on that side for the
525  * PickSplit function to try to expand, so it can have no good
526  * figure of merit for what it's doing. Also note that this check
527  * ensures we can't produce a bogus one-side-only split in the
528  * NumDontCare == 1 special case below.)
529  */
530  if (sv->spl_nleft == 0 || sv->spl_nright == 0)
531  {
532  v->spl_dontcare = NULL;
533  return true;
534  }
535 
536  /*
537  * Recompute union keys, considering only non-don't-care tuples.
538  * NOTE: this will set union keys for remaining index columns,
539  * which will cause later calls of gistUserPicksplit to pass those
540  * values down to user-defined PickSplit methods with
541  * spl_ldatum_exists/spl_rdatum_exists set true.
542  */
543  gistunionsubkey(giststate, itup, v);
544 
545  if (NumDontCare == 1)
546  {
547  /*
548  * If there's only one don't-care tuple then we can't do a
549  * PickSplit on it, so just choose whether to send it left or
550  * right by comparing penalties. We needed the
551  * gistunionsubkey step anyway so that we have appropriate
552  * union keys for figuring the penalties.
553  */
554  OffsetNumber toMove;
555 
556  /* find it ... */
557  for (toMove = FirstOffsetNumber; toMove < entryvec->n; toMove++)
558  {
559  if (v->spl_dontcare[toMove])
560  break;
561  }
562  Assert(toMove < entryvec->n);
563 
564  /* ... and assign it to cheaper side */
565  placeOne(r, giststate, v, itup[toMove - 1], toMove, attno + 1);
566 
567  /*
568  * At this point the union keys are wrong, but we don't care
569  * because we're done splitting. The outermost recursion
570  * level of gistSplitByKey will fix things before returning.
571  */
572  }
573  else
574  return true;
575  }
576  }
577 
578  return false;
579 }
580 
581 /*
582  * simply split page in half
583  */
584 static void
586 {
587  int i;
588 
589  v->spl_nright = v->spl_nleft = 0;
590  v->spl_left = (OffsetNumber *) palloc(len * sizeof(OffsetNumber));
591  v->spl_right = (OffsetNumber *) palloc(len * sizeof(OffsetNumber));
592  for (i = 1; i <= len; i++)
593  if (i < len / 2)
594  v->spl_right[v->spl_nright++] = i;
595  else
596  v->spl_left[v->spl_nleft++] = i;
597 
598  /* we need not compute union keys, caller took care of it */
599 }
600 
601 /*
602  * gistSplitByKey: main entry point for page-splitting algorithm
603  *
604  * r: index relation
605  * page: page being split
606  * itup: array of IndexTuples to be processed
607  * len: number of IndexTuples to be processed (must be at least 2)
608  * giststate: additional info about index
609  * v: working state and output area
610  * attno: column we are working on (zero-based index)
611  *
612  * Outside caller must initialize v->spl_lisnull and v->spl_risnull arrays
613  * to all-true. On return, spl_left/spl_nleft contain indexes of tuples
614  * to go left, spl_right/spl_nright contain indexes of tuples to go right,
615  * spl_lattr/spl_lisnull contain left-side union key values, and
616  * spl_rattr/spl_risnull contain right-side union key values. Other fields
617  * in this struct are workspace for this file.
618  *
619  * Outside caller must pass zero for attno. The function may internally
620  * recurse to the next column by passing attno+1.
621  */
622 void
624  GISTSTATE *giststate, GistSplitVector *v, int attno)
625 {
626  GistEntryVector *entryvec;
627  OffsetNumber *offNullTuples;
628  int nOffNullTuples = 0;
629  int i;
630 
631  /* generate the item array, and identify tuples with null keys */
632  /* note that entryvec->vector[0] goes unused in this code */
633  entryvec = palloc(GEVHDRSZ + (len + 1) * sizeof(GISTENTRY));
634  entryvec->n = len + 1;
635  offNullTuples = (OffsetNumber *) palloc(len * sizeof(OffsetNumber));
636 
637  for (i = 1; i <= len; i++)
638  {
639  Datum datum;
640  bool IsNull;
641 
642  datum = index_getattr(itup[i - 1], attno + 1, giststate->leafTupdesc,
643  &IsNull);
644  gistdentryinit(giststate, attno, &(entryvec->vector[i]),
645  datum, r, page, i,
646  false, IsNull);
647  if (IsNull)
648  offNullTuples[nOffNullTuples++] = i;
649  }
650 
651  if (nOffNullTuples == len)
652  {
653  /*
654  * Corner case: All keys in attno column are null, so just transfer
655  * our attention to the next column. If there's no next column, just
656  * split page in half.
657  */
658  v->spl_risnull[attno] = v->spl_lisnull[attno] = true;
659 
660  if (attno + 1 < giststate->nonLeafTupdesc->natts)
661  gistSplitByKey(r, page, itup, len, giststate, v, attno + 1);
662  else
664  }
665  else if (nOffNullTuples > 0)
666  {
667  int j = 0;
668 
669  /*
670  * We don't want to mix NULL and not-NULL keys on one page, so split
671  * nulls to right page and not-nulls to left.
672  */
673  v->splitVector.spl_right = offNullTuples;
674  v->splitVector.spl_nright = nOffNullTuples;
675  v->spl_risnull[attno] = true;
676 
678  v->splitVector.spl_nleft = 0;
679  for (i = 1; i <= len; i++)
680  if (j < v->splitVector.spl_nright && offNullTuples[j] == i)
681  j++;
682  else
684 
685  /* Compute union keys, unless outer recursion level will handle it */
686  if (attno == 0 && giststate->nonLeafTupdesc->natts == 1)
687  {
688  v->spl_dontcare = NULL;
689  gistunionsubkey(giststate, itup, v);
690  }
691  }
692  else
693  {
694  /*
695  * All keys are not-null, so apply user-defined PickSplit method
696  */
697  if (gistUserPicksplit(r, entryvec, attno, v, itup, len, giststate))
698  {
699  /*
700  * Splitting on attno column is not optimal, so consider
701  * redistributing don't-care tuples according to the next column
702  */
703  Assert(attno + 1 < giststate->nonLeafTupdesc->natts);
704 
705  if (v->spl_dontcare == NULL)
706  {
707  /*
708  * This split was actually degenerate, so ignore it altogether
709  * and just split according to the next column.
710  */
711  gistSplitByKey(r, page, itup, len, giststate, v, attno + 1);
712  }
713  else
714  {
715  /*
716  * Form an array of just the don't-care tuples to pass to a
717  * recursive invocation of this function for the next column.
718  */
719  IndexTuple *newitup = (IndexTuple *) palloc(len * sizeof(IndexTuple));
720  OffsetNumber *map = (OffsetNumber *) palloc(len * sizeof(OffsetNumber));
721  int newlen = 0;
722  GIST_SPLITVEC backupSplit;
723 
724  for (i = 0; i < len; i++)
725  {
726  if (v->spl_dontcare[i + 1])
727  {
728  newitup[newlen] = itup[i];
729  map[newlen] = i + 1;
730  newlen++;
731  }
732  }
733 
734  Assert(newlen > 0);
735 
736  /*
737  * Make a backup copy of v->splitVector, since the recursive
738  * call will overwrite that with its own result.
739  */
740  backupSplit = v->splitVector;
741  backupSplit.spl_left = (OffsetNumber *) palloc(sizeof(OffsetNumber) * len);
742  memcpy(backupSplit.spl_left, v->splitVector.spl_left, sizeof(OffsetNumber) * v->splitVector.spl_nleft);
743  backupSplit.spl_right = (OffsetNumber *) palloc(sizeof(OffsetNumber) * len);
744  memcpy(backupSplit.spl_right, v->splitVector.spl_right, sizeof(OffsetNumber) * v->splitVector.spl_nright);
745 
746  /* Recursively decide how to split the don't-care tuples */
747  gistSplitByKey(r, page, newitup, newlen, giststate, v, attno + 1);
748 
749  /* Merge result of subsplit with non-don't-care tuples */
750  for (i = 0; i < v->splitVector.spl_nleft; i++)
751  backupSplit.spl_left[backupSplit.spl_nleft++] = map[v->splitVector.spl_left[i] - 1];
752  for (i = 0; i < v->splitVector.spl_nright; i++)
753  backupSplit.spl_right[backupSplit.spl_nright++] = map[v->splitVector.spl_right[i] - 1];
754 
755  v->splitVector = backupSplit;
756  }
757  }
758  }
759 
760  /*
761  * If we're handling a multicolumn index, at the end of the recursion
762  * recompute the left and right union datums for all index columns. This
763  * makes sure we hand back correct union datums in all corner cases,
764  * including when we haven't processed all columns to start with, or when
765  * a secondary split moved "don't care" tuples from one side to the other
766  * (we really shouldn't assume that that didn't change the union datums).
767  *
768  * Note: when we're in an internal recursion (attno > 0), we do not worry
769  * about whether the union datums we return with are sensible, since
770  * calling levels won't care. Also, in a single-column index, we expect
771  * that PickSplit (or the special cases above) produced correct union
772  * datums.
773  */
774  if (attno == 0 && giststate->nonLeafTupdesc->natts > 1)
775  {
776  v->spl_dontcare = NULL;
777  gistunionsubkey(giststate, itup, v);
778  }
779 }
Pointer Page
Definition: bufpage.h:81
#define Assert(condition)
Definition: c.h:858
int errhint(const char *fmt,...)
Definition: elog.c:1317
int errcode(int sqlerrcode)
Definition: elog.c:853
int errmsg(const char *fmt,...)
Definition: elog.c:1070
#define DEBUG1
Definition: elog.h:30
#define ereport(elevel,...)
Definition: elog.h:149
Datum FunctionCall2Coll(FmgrInfo *flinfo, Oid collation, Datum arg1, Datum arg2)
Definition: fmgr.c:1149
#define GEVHDRSZ
Definition: gist.h:239
#define gistentryinit(e, k, r, pg, o, l)
Definition: gist.h:244
static void removeDontCares(OffsetNumber *a, int *len, const bool *dontcare)
Definition: gistsplit.c:167
static void supportSecondarySplit(Relation r, GISTSTATE *giststate, int attno, GIST_SPLITVEC *sv, Datum oldL, Datum oldR)
Definition: gistsplit.c:258
static void genericPickSplit(GISTSTATE *giststate, GistEntryVector *entryvec, GIST_SPLITVEC *v, int attno)
Definition: gistsplit.c:344
static void placeOne(Relation r, GISTSTATE *giststate, GistSplitVector *v, IndexTuple itup, OffsetNumber off, int attno)
Definition: gistsplit.c:200
static void gistunionsubkey(GISTSTATE *giststate, IndexTuple *itvec, GistSplitVector *spl)
Definition: gistsplit.c:80
void gistSplitByKey(Relation r, Page page, IndexTuple *itup, int len, GISTSTATE *giststate, GistSplitVector *v, int attno)
Definition: gistsplit.c:623
static bool gistUserPicksplit(Relation r, GistEntryVector *entryvec, int attno, GistSplitVector *v, IndexTuple *itup, int len, GISTSTATE *giststate)
Definition: gistsplit.c:415
static int findDontCares(Relation r, GISTSTATE *giststate, GISTENTRY *valvec, GistSplitVector *spl, int attno)
Definition: gistsplit.c:113
static void gistunionsubkeyvec(GISTSTATE *giststate, IndexTuple *itvec, GistSplitUnion *gsvp)
Definition: gistsplit.c:47
static void gistSplitHalf(GIST_SPLITVEC *v, int len)
Definition: gistsplit.c:585
#define SWAPVAR(s, d, t)
Definition: gistsplit.c:237
void gistDeCompressAtt(GISTSTATE *giststate, Relation r, IndexTuple tuple, Page p, OffsetNumber o, GISTENTRY *attdata, bool *isnull)
Definition: gistutil.c:296
void gistMakeUnionKey(GISTSTATE *giststate, int attno, GISTENTRY *entry1, bool isnull1, GISTENTRY *entry2, bool isnull2, Datum *dst, bool *dstisnull)
Definition: gistutil.c:233
void gistMakeUnionItVec(GISTSTATE *giststate, IndexTuple *itvec, int len, Datum *attr, bool *isnull)
Definition: gistutil.c:155
bool gistKeyIsEQ(GISTSTATE *giststate, int attno, Datum a, Datum b)
Definition: gistutil.c:281
void gistdentryinit(GISTSTATE *giststate, int nkey, GISTENTRY *e, Datum k, Relation r, Page pg, OffsetNumber o, bool l, bool isNull)
Definition: gistutil.c:547
float gistpenalty(GISTSTATE *giststate, int attno, GISTENTRY *orig, bool isNullOrig, GISTENTRY *add, bool isNullAdd)
Definition: gistutil.c:724
for(;;)
int a
Definition: isn.c:69
int j
Definition: isn.c:74
int i
Definition: isn.c:73
static Datum index_getattr(IndexTuple tup, int attnum, TupleDesc tupleDesc, bool *isnull)
Definition: itup.h:117
void pfree(void *pointer)
Definition: mcxt.c:1521
void * palloc0(Size size)
Definition: mcxt.c:1347
void * palloc(Size size)
Definition: mcxt.c:1317
#define InvalidOffsetNumber
Definition: off.h:26
#define OffsetNumberNext(offsetNumber)
Definition: off.h:52
uint16 OffsetNumber
Definition: off.h:24
#define FirstOffsetNumber
Definition: off.h:27
#define INDEX_MAX_KEYS
const void size_t len
static Datum PointerGetDatum(const void *X)
Definition: postgres.h:322
uintptr_t Datum
Definition: postgres.h:64
#define RelationGetRelationName(relation)
Definition: rel.h:539
TupleDesc leafTupdesc
Definition: gist_private.h:80
TupleDesc nonLeafTupdesc
Definition: gist_private.h:81
Oid supportCollation[INDEX_MAX_KEYS]
Definition: gist_private.h:97
FmgrInfo unionFn[INDEX_MAX_KEYS]
Definition: gist_private.h:87
FmgrInfo picksplitFn[INDEX_MAX_KEYS]
Definition: gist_private.h:91
bool spl_ldatum_exists
Definition: gist.h:145
bool spl_rdatum_exists
Definition: gist.h:150
int spl_nleft
Definition: gist.h:143
OffsetNumber * spl_right
Definition: gist.h:147
Datum spl_ldatum
Definition: gist.h:144
Datum spl_rdatum
Definition: gist.h:149
int spl_nright
Definition: gist.h:148
OffsetNumber * spl_left
Definition: gist.h:142
GISTENTRY vector[FLEXIBLE_ARRAY_MEMBER]
Definition: gist.h:236
int32 n
Definition: gist.h:235
OffsetNumber * entries
Definition: gistsplit.c:33
bool * isnull
Definition: gistsplit.c:36
bool * dontcare
Definition: gistsplit.c:37
Datum * attr
Definition: gistsplit.c:35
GIST_SPLITVEC splitVector
Definition: gist_private.h:237
Datum spl_lattr[INDEX_MAX_KEYS]
Definition: gist_private.h:239
bool spl_lisnull[INDEX_MAX_KEYS]
Definition: gist_private.h:241
Datum spl_rattr[INDEX_MAX_KEYS]
Definition: gist_private.h:243
bool spl_risnull[INDEX_MAX_KEYS]
Definition: gist_private.h:245