Commit c6a11b89 authored by Tom Lane's avatar Tom Lane

Teach SPGiST to store nulls and do whole-index scans.

This patch fixes the other major compatibility-breaking limitation of
SPGiST, that it didn't store anything for null values of the indexed
column, and so could not support whole-index scans or "x IS NULL"
tests.  The approach is to create a wholly separate search tree for
the null entries, and use fixed "allTheSame" insertion and search
rules when processing this tree, instead of calling the index opclass
methods.  This way the opclass methods do not need to worry about
dealing with nulls.

Catversion bump is for pg_am updates as well as the change in on-disk
format of SPGiST indexes; there are some tweaks in SPGiST WAL records
as well.

Heavily rewritten version of a patch by Oleg Bartunov and Teodor Sigaev.
(The original also stored nulls separately, but it reused GIN code to do
so; which required undesirable compromises in the on-disk format, and
would likely lead to bugs due to the GIN code being required to work in
two very different contexts.)
parent fc227a4e
...@@ -100,6 +100,18 @@ ...@@ -100,6 +100,18 @@
value when that is needed. value when that is needed.
</para> </para>
<note>
<para>
The <acronym>SP-GiST</acronym> core code takes care of NULL entries.
Although <acronym>SP-GiST</acronym> indexes do store entries for nulls
in indexed columns, this is hidden from the index operator class code:
no null index entries or search conditions will ever be passed to the
operator class methods. (It is assumed that <acronym>SP-GiST</acronym>
operators are strict and so cannot succeed for NULL values.) NULLs
are therefore not discussed further here.
</para>
</note>
<para> <para>
There are five user-defined methods that an index operator class for There are five user-defined methods that an index operator class for
<acronym>SP-GiST</acronym> must provide. All five follow the convention <acronym>SP-GiST</acronym> must provide. All five follow the convention
......
...@@ -11,6 +11,7 @@ should have a high fanout to minimize I/O. The challenge is to map tree ...@@ -11,6 +11,7 @@ should have a high fanout to minimize I/O. The challenge is to map tree
nodes to disk pages in such a way that the search algorithm accesses only a nodes to disk pages in such a way that the search algorithm accesses only a
few disk pages, even if it traverses many nodes. few disk pages, even if it traverses many nodes.
COMMON STRUCTURE DESCRIPTION COMMON STRUCTURE DESCRIPTION
Logically, an SP-GiST tree is a set of tuples, each of which can be either Logically, an SP-GiST tree is a set of tuples, each of which can be either
...@@ -71,6 +72,21 @@ Leaf tuple consists of: ...@@ -71,6 +72,21 @@ Leaf tuple consists of:
ItemPointer to the heap ItemPointer to the heap
NULLS HANDLING
We assume that SPGiST-indexable operators are strict (can never succeed for
null inputs). It is still desirable to index nulls, so that whole-table
indexscans are possible and so that "x IS NULL" can be implemented by an
SPGiST indexscan. However, we prefer that SPGiST index opclasses not have
to cope with nulls. Therefore, the main tree of an SPGiST index does not
include any null entries. We store null entries in a separate SPGiST tree
occupying a disjoint set of pages (in particular, its own root page).
Insertions and searches in the nulls tree do not use any of the
opclass-supplied functions, but just use hardwired logic comparable to
AllTheSame cases in the normal tree.
INSERTION ALGORITHM INSERTION ALGORITHM
Insertion algorithm is designed to keep the tree in a consistent state at Insertion algorithm is designed to keep the tree in a consistent state at
...@@ -181,6 +197,7 @@ described in (5). ...@@ -181,6 +197,7 @@ described in (5).
and a new tuple to another page, if the list is short enough. This improves and a new tuple to another page, if the list is short enough. This improves
space utilization, but doesn't change the basis of the algorithm. space utilization, but doesn't change the basis of the algorithm.
CONCURRENCY CONCURRENCY
While descending the tree, the insertion algorithm holds exclusive lock on While descending the tree, the insertion algorithm holds exclusive lock on
...@@ -218,6 +235,7 @@ scan that had already visited the parent level could possibly reach such a ...@@ -218,6 +235,7 @@ scan that had already visited the parent level could possibly reach such a
redirect tuple, so we can remove redirects once all active transactions have redirect tuple, so we can remove redirects once all active transactions have
been flushed out of the system. been flushed out of the system.
DEAD TUPLES DEAD TUPLES
Tuples on leaf pages can be in one of four states: Tuples on leaf pages can be in one of four states:
...@@ -269,6 +287,7 @@ to PLACEHOLDER status by VACUUM, and are then candidates for replacement. ...@@ -269,6 +287,7 @@ to PLACEHOLDER status by VACUUM, and are then candidates for replacement.
DEAD state is not currently possible, since VACUUM does not attempt to DEAD state is not currently possible, since VACUUM does not attempt to
remove unused inner tuples. remove unused inner tuples.
VACUUM VACUUM
VACUUM (or more precisely, spgbulkdelete) performs a single sequential scan VACUUM (or more precisely, spgbulkdelete) performs a single sequential scan
...@@ -302,13 +321,16 @@ performed; otherwise, it does an spgbulkdelete scan with an empty target ...@@ -302,13 +321,16 @@ performed; otherwise, it does an spgbulkdelete scan with an empty target
list, so as to clean up redirections and placeholders, update the free list, so as to clean up redirections and placeholders, update the free
space map, and gather statistics. space map, and gather statistics.
LAST USED PAGE MANAGEMENT LAST USED PAGE MANAGEMENT
List of last used pages contains four pages - a leaf page and three inner The list of last used pages contains four pages - a leaf page and three
pages, one from each "triple parity" group. This list is stored between inner pages, one from each "triple parity" group. (Actually, there's one
calls on the index meta page, but updates are never WAL-logged to decrease such list for the main tree and a separate one for the nulls tree.) This
WAL traffic. Incorrect data on meta page isn't critical, because we could list is stored between calls on the index meta page, but updates are never
allocate a new page at any moment. WAL-logged to decrease WAL traffic. Incorrect data on meta page isn't
critical, because we could allocate a new page at any moment.
AUTHORS AUTHORS
......
This diff is collapsed.
...@@ -38,18 +38,15 @@ spgistBuildCallback(Relation index, HeapTuple htup, Datum *values, ...@@ -38,18 +38,15 @@ spgistBuildCallback(Relation index, HeapTuple htup, Datum *values,
bool *isnull, bool tupleIsAlive, void *state) bool *isnull, bool tupleIsAlive, void *state)
{ {
SpGistBuildState *buildstate = (SpGistBuildState *) state; SpGistBuildState *buildstate = (SpGistBuildState *) state;
MemoryContext oldCtx;
/* SPGiST doesn't index nulls */
if (*isnull == false)
{
/* Work in temp context, and reset it after each tuple */ /* Work in temp context, and reset it after each tuple */
MemoryContext oldCtx = MemoryContextSwitchTo(buildstate->tmpCtx); oldCtx = MemoryContextSwitchTo(buildstate->tmpCtx);
spgdoinsert(index, &buildstate->spgstate, &htup->t_self, *values); spgdoinsert(index, &buildstate->spgstate, &htup->t_self, *values, *isnull);
MemoryContextSwitchTo(oldCtx); MemoryContextSwitchTo(oldCtx);
MemoryContextReset(buildstate->tmpCtx); MemoryContextReset(buildstate->tmpCtx);
}
} }
/* /*
...@@ -65,20 +62,23 @@ spgbuild(PG_FUNCTION_ARGS) ...@@ -65,20 +62,23 @@ spgbuild(PG_FUNCTION_ARGS)
double reltuples; double reltuples;
SpGistBuildState buildstate; SpGistBuildState buildstate;
Buffer metabuffer, Buffer metabuffer,
rootbuffer; rootbuffer,
nullbuffer;
if (RelationGetNumberOfBlocks(index) != 0) if (RelationGetNumberOfBlocks(index) != 0)
elog(ERROR, "index \"%s\" already contains data", elog(ERROR, "index \"%s\" already contains data",
RelationGetRelationName(index)); RelationGetRelationName(index));
/* /*
* Initialize the meta page and root page * Initialize the meta page and root pages
*/ */
metabuffer = SpGistNewBuffer(index); metabuffer = SpGistNewBuffer(index);
rootbuffer = SpGistNewBuffer(index); rootbuffer = SpGistNewBuffer(index);
nullbuffer = SpGistNewBuffer(index);
Assert(BufferGetBlockNumber(metabuffer) == SPGIST_METAPAGE_BLKNO); Assert(BufferGetBlockNumber(metabuffer) == SPGIST_METAPAGE_BLKNO);
Assert(BufferGetBlockNumber(rootbuffer) == SPGIST_HEAD_BLKNO); Assert(BufferGetBlockNumber(rootbuffer) == SPGIST_ROOT_BLKNO);
Assert(BufferGetBlockNumber(nullbuffer) == SPGIST_NULL_BLKNO);
START_CRIT_SECTION(); START_CRIT_SECTION();
...@@ -86,6 +86,8 @@ spgbuild(PG_FUNCTION_ARGS) ...@@ -86,6 +86,8 @@ spgbuild(PG_FUNCTION_ARGS)
MarkBufferDirty(metabuffer); MarkBufferDirty(metabuffer);
SpGistInitBuffer(rootbuffer, SPGIST_LEAF); SpGistInitBuffer(rootbuffer, SPGIST_LEAF);
MarkBufferDirty(rootbuffer); MarkBufferDirty(rootbuffer);
SpGistInitBuffer(nullbuffer, SPGIST_LEAF | SPGIST_NULLS);
MarkBufferDirty(nullbuffer);
if (RelationNeedsWAL(index)) if (RelationNeedsWAL(index))
{ {
...@@ -104,12 +106,15 @@ spgbuild(PG_FUNCTION_ARGS) ...@@ -104,12 +106,15 @@ spgbuild(PG_FUNCTION_ARGS)
PageSetTLI(BufferGetPage(metabuffer), ThisTimeLineID); PageSetTLI(BufferGetPage(metabuffer), ThisTimeLineID);
PageSetLSN(BufferGetPage(rootbuffer), recptr); PageSetLSN(BufferGetPage(rootbuffer), recptr);
PageSetTLI(BufferGetPage(rootbuffer), ThisTimeLineID); PageSetTLI(BufferGetPage(rootbuffer), ThisTimeLineID);
PageSetLSN(BufferGetPage(nullbuffer), recptr);
PageSetTLI(BufferGetPage(nullbuffer), ThisTimeLineID);
} }
END_CRIT_SECTION(); END_CRIT_SECTION();
UnlockReleaseBuffer(metabuffer); UnlockReleaseBuffer(metabuffer);
UnlockReleaseBuffer(rootbuffer); UnlockReleaseBuffer(rootbuffer);
UnlockReleaseBuffer(nullbuffer);
/* /*
* Now insert all the heap data into the index * Now insert all the heap data into the index
...@@ -159,11 +164,20 @@ spgbuildempty(PG_FUNCTION_ARGS) ...@@ -159,11 +164,20 @@ spgbuildempty(PG_FUNCTION_ARGS)
/* Likewise for the root page. */ /* Likewise for the root page. */
SpGistInitPage(page, SPGIST_LEAF); SpGistInitPage(page, SPGIST_LEAF);
smgrwrite(index->rd_smgr, INIT_FORKNUM, SPGIST_HEAD_BLKNO, smgrwrite(index->rd_smgr, INIT_FORKNUM, SPGIST_ROOT_BLKNO,
(char *) page, true);
if (XLogIsNeeded())
log_newpage(&index->rd_smgr->smgr_rnode.node, INIT_FORKNUM,
SPGIST_ROOT_BLKNO, page);
/* Likewise for the null-tuples root page. */
SpGistInitPage(page, SPGIST_LEAF | SPGIST_NULLS);
smgrwrite(index->rd_smgr, INIT_FORKNUM, SPGIST_NULL_BLKNO,
(char *) page, true); (char *) page, true);
if (XLogIsNeeded()) if (XLogIsNeeded())
log_newpage(&index->rd_smgr->smgr_rnode.node, INIT_FORKNUM, log_newpage(&index->rd_smgr->smgr_rnode.node, INIT_FORKNUM,
SPGIST_HEAD_BLKNO, page); SPGIST_NULL_BLKNO, page);
/* /*
* An immediate sync is required even if we xlog'd the pages, because the * An immediate sync is required even if we xlog'd the pages, because the
...@@ -194,10 +208,6 @@ spginsert(PG_FUNCTION_ARGS) ...@@ -194,10 +208,6 @@ spginsert(PG_FUNCTION_ARGS)
MemoryContext oldCtx; MemoryContext oldCtx;
MemoryContext insertCtx; MemoryContext insertCtx;
/* SPGiST doesn't index nulls */
if (*isnull)
PG_RETURN_BOOL(false);
insertCtx = AllocSetContextCreate(CurrentMemoryContext, insertCtx = AllocSetContextCreate(CurrentMemoryContext,
"SP-GiST insert temporary context", "SP-GiST insert temporary context",
ALLOCSET_DEFAULT_MINSIZE, ALLOCSET_DEFAULT_MINSIZE,
...@@ -207,7 +217,7 @@ spginsert(PG_FUNCTION_ARGS) ...@@ -207,7 +217,7 @@ spginsert(PG_FUNCTION_ARGS)
initSpGistState(&spgstate, index); initSpGistState(&spgstate, index);
spgdoinsert(index, &spgstate, ht_ctid, *values); spgdoinsert(index, &spgstate, ht_ctid, *values, *isnull);
SpGistUpdateMetaPage(index); SpGistUpdateMetaPage(index);
......
...@@ -23,6 +23,9 @@ ...@@ -23,6 +23,9 @@
#include "utils/memutils.h" #include "utils/memutils.h"
typedef void (*storeRes_func) (SpGistScanOpaque so, ItemPointer heapPtr,
Datum leafValue, bool isnull, bool recheck);
typedef struct ScanStackEntry typedef struct ScanStackEntry
{ {
Datum reconstructedValue; /* value reconstructed from parent */ Datum reconstructedValue; /* value reconstructed from parent */
...@@ -66,14 +69,20 @@ resetSpGistScanOpaque(SpGistScanOpaque so) ...@@ -66,14 +69,20 @@ resetSpGistScanOpaque(SpGistScanOpaque so)
freeScanStack(so); freeScanStack(so);
Assert(!so->searchNulls); /* XXX fixme */ if (so->searchNulls)
{
/* Stack a work item to scan the null index entries */
startEntry = (ScanStackEntry *) palloc0(sizeof(ScanStackEntry));
ItemPointerSet(&startEntry->ptr, SPGIST_NULL_BLKNO, FirstOffsetNumber);
so->scanStack = lappend(so->scanStack, startEntry);
}
if (so->searchNonNulls) if (so->searchNonNulls)
{ {
/* Stack a work item to scan the non-null index entries */ /* Stack a work item to scan the non-null index entries */
startEntry = (ScanStackEntry *) palloc0(sizeof(ScanStackEntry)); startEntry = (ScanStackEntry *) palloc0(sizeof(ScanStackEntry));
ItemPointerSet(&startEntry->ptr, SPGIST_HEAD_BLKNO, FirstOffsetNumber); ItemPointerSet(&startEntry->ptr, SPGIST_ROOT_BLKNO, FirstOffsetNumber);
so->scanStack = list_make1(startEntry); so->scanStack = lappend(so->scanStack, startEntry);
} }
if (so->want_itup) if (so->want_itup)
...@@ -243,22 +252,35 @@ spgrestrpos(PG_FUNCTION_ARGS) ...@@ -243,22 +252,35 @@ spgrestrpos(PG_FUNCTION_ARGS)
} }
/* /*
* Test whether a leaf datum satisfies all the scan keys * Test whether a leaf tuple satisfies all the scan keys
* *
* *leafValue is set to the reconstructed datum, if provided * *leafValue is set to the reconstructed datum, if provided
* *recheck is set true if any of the operators are lossy * *recheck is set true if any of the operators are lossy
*/ */
static bool static bool
spgLeafTest(Relation index, SpGistScanOpaque so, Datum leafDatum, spgLeafTest(Relation index, SpGistScanOpaque so,
SpGistLeafTuple leafTuple, bool isnull,
int level, Datum reconstructedValue, int level, Datum reconstructedValue,
Datum *leafValue, bool *recheck) Datum *leafValue, bool *recheck)
{ {
bool result; bool result;
Datum leafDatum;
spgLeafConsistentIn in; spgLeafConsistentIn in;
spgLeafConsistentOut out; spgLeafConsistentOut out;
FmgrInfo *procinfo; FmgrInfo *procinfo;
MemoryContext oldCtx; MemoryContext oldCtx;
if (isnull)
{
/* Should not have arrived on a nulls page unless nulls are wanted */
Assert(so->searchNulls);
*leafValue = (Datum) 0;
*recheck = false;
return true;
}
leafDatum = SGLTDATUM(leafTuple, &so->state);
/* use temp context for calling leaf_consistent */ /* use temp context for calling leaf_consistent */
oldCtx = MemoryContextSwitchTo(so->tempCxt); oldCtx = MemoryContextSwitchTo(so->tempCxt);
...@@ -295,7 +317,7 @@ spgLeafTest(Relation index, SpGistScanOpaque so, Datum leafDatum, ...@@ -295,7 +317,7 @@ spgLeafTest(Relation index, SpGistScanOpaque so, Datum leafDatum,
*/ */
static void static void
spgWalk(Relation index, SpGistScanOpaque so, bool scanWholeIndex, spgWalk(Relation index, SpGistScanOpaque so, bool scanWholeIndex,
void (*storeRes) (SpGistScanOpaque, ItemPointer, Datum, bool)) storeRes_func storeRes)
{ {
Buffer buffer = InvalidBuffer; Buffer buffer = InvalidBuffer;
bool reportedSome = false; bool reportedSome = false;
...@@ -306,6 +328,7 @@ spgWalk(Relation index, SpGistScanOpaque so, bool scanWholeIndex, ...@@ -306,6 +328,7 @@ spgWalk(Relation index, SpGistScanOpaque so, bool scanWholeIndex,
BlockNumber blkno; BlockNumber blkno;
OffsetNumber offset; OffsetNumber offset;
Page page; Page page;
bool isnull;
/* Pull next to-do item from the list */ /* Pull next to-do item from the list */
if (so->scanStack == NIL) if (so->scanStack == NIL)
...@@ -336,6 +359,8 @@ redirect: ...@@ -336,6 +359,8 @@ redirect:
page = BufferGetPage(buffer); page = BufferGetPage(buffer);
isnull = SpGistPageStoresNulls(page) ? true : false;
if (SpGistPageIsLeaf(page)) if (SpGistPageIsLeaf(page))
{ {
SpGistLeafTuple leafTuple; SpGistLeafTuple leafTuple;
...@@ -343,7 +368,7 @@ redirect: ...@@ -343,7 +368,7 @@ redirect:
Datum leafValue = (Datum) 0; Datum leafValue = (Datum) 0;
bool recheck = false; bool recheck = false;
if (blkno == SPGIST_HEAD_BLKNO) if (SpGistBlockIsRoot(blkno))
{ {
/* When root is a leaf, examine all its tuples */ /* When root is a leaf, examine all its tuples */
for (offset = FirstOffsetNumber; offset <= max; offset++) for (offset = FirstOffsetNumber; offset <= max; offset++)
...@@ -359,13 +384,14 @@ redirect: ...@@ -359,13 +384,14 @@ redirect:
Assert(ItemPointerIsValid(&leafTuple->heapPtr)); Assert(ItemPointerIsValid(&leafTuple->heapPtr));
if (spgLeafTest(index, so, if (spgLeafTest(index, so,
SGLTDATUM(leafTuple, &so->state), leafTuple, isnull,
stackEntry->level, stackEntry->level,
stackEntry->reconstructedValue, stackEntry->reconstructedValue,
&leafValue, &leafValue,
&recheck)) &recheck))
{ {
storeRes(so, &leafTuple->heapPtr, leafValue, recheck); storeRes(so, &leafTuple->heapPtr,
leafValue, isnull, recheck);
reportedSome = true; reportedSome = true;
} }
} }
...@@ -404,13 +430,14 @@ redirect: ...@@ -404,13 +430,14 @@ redirect:
Assert(ItemPointerIsValid(&leafTuple->heapPtr)); Assert(ItemPointerIsValid(&leafTuple->heapPtr));
if (spgLeafTest(index, so, if (spgLeafTest(index, so,
SGLTDATUM(leafTuple, &so->state), leafTuple, isnull,
stackEntry->level, stackEntry->level,
stackEntry->reconstructedValue, stackEntry->reconstructedValue,
&leafValue, &leafValue,
&recheck)) &recheck))
{ {
storeRes(so, &leafTuple->heapPtr, leafValue, recheck); storeRes(so, &leafTuple->heapPtr,
leafValue, isnull, recheck);
reportedSome = true; reportedSome = true;
} }
...@@ -468,11 +495,23 @@ redirect: ...@@ -468,11 +495,23 @@ redirect:
memset(&out, 0, sizeof(out)); memset(&out, 0, sizeof(out));
if (!isnull)
{
/* use user-defined inner consistent method */
procinfo = index_getprocinfo(index, 1, SPGIST_INNER_CONSISTENT_PROC); procinfo = index_getprocinfo(index, 1, SPGIST_INNER_CONSISTENT_PROC);
FunctionCall2Coll(procinfo, FunctionCall2Coll(procinfo,
index->rd_indcollation[0], index->rd_indcollation[0],
PointerGetDatum(&in), PointerGetDatum(&in),
PointerGetDatum(&out)); PointerGetDatum(&out));
}
else
{
/* force all children to be visited */
out.nNodes = in.nNodes;
out.nodeNumbers = (int *) palloc(sizeof(int) * in.nNodes);
for (i = 0; i < in.nNodes; i++)
out.nodeNumbers[i] = i;
}
MemoryContextSwitchTo(oldCtx); MemoryContextSwitchTo(oldCtx);
...@@ -524,7 +563,7 @@ redirect: ...@@ -524,7 +563,7 @@ redirect:
/* storeRes subroutine for getbitmap case */ /* storeRes subroutine for getbitmap case */
static void static void
storeBitmap(SpGistScanOpaque so, ItemPointer heapPtr, storeBitmap(SpGistScanOpaque so, ItemPointer heapPtr,
Datum leafValue, bool recheck) Datum leafValue, bool isnull, bool recheck)
{ {
tbm_add_tuples(so->tbm, heapPtr, 1, recheck); tbm_add_tuples(so->tbm, heapPtr, 1, recheck);
so->ntids++; so->ntids++;
...@@ -551,7 +590,7 @@ spggetbitmap(PG_FUNCTION_ARGS) ...@@ -551,7 +590,7 @@ spggetbitmap(PG_FUNCTION_ARGS)
/* storeRes subroutine for gettuple case */ /* storeRes subroutine for gettuple case */
static void static void
storeGettuple(SpGistScanOpaque so, ItemPointer heapPtr, storeGettuple(SpGistScanOpaque so, ItemPointer heapPtr,
Datum leafValue, bool recheck) Datum leafValue, bool isnull, bool recheck)
{ {
Assert(so->nPtrs < MaxIndexTuplesPerPage); Assert(so->nPtrs < MaxIndexTuplesPerPage);
so->heapPtrs[so->nPtrs] = *heapPtr; so->heapPtrs[so->nPtrs] = *heapPtr;
...@@ -562,8 +601,6 @@ storeGettuple(SpGistScanOpaque so, ItemPointer heapPtr, ...@@ -562,8 +601,6 @@ storeGettuple(SpGistScanOpaque so, ItemPointer heapPtr,
* Reconstruct desired IndexTuple. We have to copy the datum out of * Reconstruct desired IndexTuple. We have to copy the datum out of
* the temp context anyway, so we may as well create the tuple here. * the temp context anyway, so we may as well create the tuple here.
*/ */
bool isnull = false;
so->indexTups[so->nPtrs] = index_form_tuple(so->indexTupDesc, so->indexTups[so->nPtrs] = index_form_tuple(so->indexTupDesc,
&leafValue, &leafValue,
&isnull); &isnull);
......
...@@ -148,10 +148,10 @@ SpGistNewBuffer(Relation index) ...@@ -148,10 +148,10 @@ SpGistNewBuffer(Relation index)
break; /* nothing known to FSM */ break; /* nothing known to FSM */
/* /*
* The root page shouldn't ever be listed in FSM, but just in case it * The fixed pages shouldn't ever be listed in FSM, but just in case
* is, ignore it. * one is, ignore it.
*/ */
if (blkno == SPGIST_HEAD_BLKNO) if (SpGistBlockIsFixed(blkno))
continue; continue;
buffer = ReadBuffer(index, blkno); buffer = ReadBuffer(index, blkno);
...@@ -226,9 +226,8 @@ SpGistUpdateMetaPage(Relation index) ...@@ -226,9 +226,8 @@ SpGistUpdateMetaPage(Relation index)
} }
/* Macro to select proper element of lastUsedPages cache depending on flags */ /* Macro to select proper element of lastUsedPages cache depending on flags */
#define GET_LUP(c, f) (((f) & GBUF_LEAF) ? \ /* Masking flags with SPGIST_CACHED_PAGES is just for paranoia's sake */
&(c)->lastUsedPages.leafPage : \ #define GET_LUP(c, f) (&(c)->lastUsedPages.cachedPage[((unsigned int) (f)) % SPGIST_CACHED_PAGES])
&(c)->lastUsedPages.innerPage[(f) & GBUF_PARITY_MASK])
/* /*
* Allocate and initialize a new buffer of the type and parity specified by * Allocate and initialize a new buffer of the type and parity specified by
...@@ -254,15 +253,21 @@ static Buffer ...@@ -254,15 +253,21 @@ static Buffer
allocNewBuffer(Relation index, int flags) allocNewBuffer(Relation index, int flags)
{ {
SpGistCache *cache = spgGetCache(index); SpGistCache *cache = spgGetCache(index);
uint16 pageflags = 0;
if (GBUF_REQ_LEAF(flags))
pageflags |= SPGIST_LEAF;
if (GBUF_REQ_NULLS(flags))
pageflags |= SPGIST_NULLS;
for (;;) for (;;)
{ {
Buffer buffer; Buffer buffer;
buffer = SpGistNewBuffer(index); buffer = SpGistNewBuffer(index);
SpGistInitBuffer(buffer, (flags & GBUF_LEAF) ? SPGIST_LEAF : 0); SpGistInitBuffer(buffer, pageflags);
if (flags & GBUF_LEAF) if (pageflags & SPGIST_LEAF)
{ {
/* Leaf pages have no parity concerns, so just use it */ /* Leaf pages have no parity concerns, so just use it */
return buffer; return buffer;
...@@ -270,9 +275,9 @@ allocNewBuffer(Relation index, int flags) ...@@ -270,9 +275,9 @@ allocNewBuffer(Relation index, int flags)
else else
{ {
BlockNumber blkno = BufferGetBlockNumber(buffer); BlockNumber blkno = BufferGetBlockNumber(buffer);
int blkParity = blkno % 3; int blkFlags = GBUF_INNER_PARITY(blkno);
if ((flags & GBUF_PARITY_MASK) == blkParity) if ((flags & GBUF_PARITY_MASK) == blkFlags)
{ {
/* Page has right parity, use it */ /* Page has right parity, use it */
return buffer; return buffer;
...@@ -280,8 +285,10 @@ allocNewBuffer(Relation index, int flags) ...@@ -280,8 +285,10 @@ allocNewBuffer(Relation index, int flags)
else else
{ {
/* Page has wrong parity, record it in cache and try again */ /* Page has wrong parity, record it in cache and try again */
cache->lastUsedPages.innerPage[blkParity].blkno = blkno; if (pageflags & SPGIST_NULLS)
cache->lastUsedPages.innerPage[blkParity].freeSpace = blkFlags |= GBUF_NULLS;
cache->lastUsedPages.cachedPage[blkFlags].blkno = blkno;
cache->lastUsedPages.cachedPage[blkFlags].freeSpace =
PageGetExactFreeSpace(BufferGetPage(buffer)); PageGetExactFreeSpace(BufferGetPage(buffer));
UnlockReleaseBuffer(buffer); UnlockReleaseBuffer(buffer);
} }
...@@ -329,8 +336,8 @@ SpGistGetBuffer(Relation index, int flags, int needSpace, bool *isNew) ...@@ -329,8 +336,8 @@ SpGistGetBuffer(Relation index, int flags, int needSpace, bool *isNew)
return allocNewBuffer(index, flags); return allocNewBuffer(index, flags);
} }
/* root page should never be in cache */ /* fixed pages should never be in cache */
Assert(lup->blkno != SPGIST_HEAD_BLKNO); Assert(!SpGistBlockIsFixed(lup->blkno));
/* If cached freeSpace isn't enough, don't bother looking at the page */ /* If cached freeSpace isn't enough, don't bother looking at the page */
if (lup->freeSpace >= needSpace) if (lup->freeSpace >= needSpace)
...@@ -355,7 +362,13 @@ SpGistGetBuffer(Relation index, int flags, int needSpace, bool *isNew) ...@@ -355,7 +362,13 @@ SpGistGetBuffer(Relation index, int flags, int needSpace, bool *isNew)
if (PageIsNew(page) || SpGistPageIsDeleted(page) || PageIsEmpty(page)) if (PageIsNew(page) || SpGistPageIsDeleted(page) || PageIsEmpty(page))
{ {
/* OK to initialize the page */ /* OK to initialize the page */
SpGistInitBuffer(buffer, (flags & GBUF_LEAF) ? SPGIST_LEAF : 0); uint16 pageflags = 0;
if (GBUF_REQ_LEAF(flags))
pageflags |= SPGIST_LEAF;
if (GBUF_REQ_NULLS(flags))
pageflags |= SPGIST_NULLS;
SpGistInitBuffer(buffer, pageflags);
lup->freeSpace = PageGetExactFreeSpace(page) - needSpace; lup->freeSpace = PageGetExactFreeSpace(page) - needSpace;
*isNew = true; *isNew = true;
return buffer; return buffer;
...@@ -365,8 +378,8 @@ SpGistGetBuffer(Relation index, int flags, int needSpace, bool *isNew) ...@@ -365,8 +378,8 @@ SpGistGetBuffer(Relation index, int flags, int needSpace, bool *isNew)
* Check that page is of right type and has enough space. We must * Check that page is of right type and has enough space. We must
* recheck this since our cache isn't necessarily up to date. * recheck this since our cache isn't necessarily up to date.
*/ */
if ((flags & GBUF_LEAF) ? SpGistPageIsLeaf(page) : if ((GBUF_REQ_LEAF(flags) ? SpGistPageIsLeaf(page) : !SpGistPageIsLeaf(page)) &&
!SpGistPageIsLeaf(page)) (GBUF_REQ_NULLS(flags) ? SpGistPageStoresNulls(page) : !SpGistPageStoresNulls(page)))
{ {
int freeSpace = PageGetExactFreeSpace(page); int freeSpace = PageGetExactFreeSpace(page);
...@@ -407,14 +420,16 @@ SpGistSetLastUsedPage(Relation index, Buffer buffer) ...@@ -407,14 +420,16 @@ SpGistSetLastUsedPage(Relation index, Buffer buffer)
BlockNumber blkno = BufferGetBlockNumber(buffer); BlockNumber blkno = BufferGetBlockNumber(buffer);
int flags; int flags;
/* Never enter the root page in cache, though */ /* Never enter fixed pages (root pages) in cache, though */
if (blkno == SPGIST_HEAD_BLKNO) if (SpGistBlockIsFixed(blkno))
return; return;
if (SpGistPageIsLeaf(page)) if (SpGistPageIsLeaf(page))
flags = GBUF_LEAF; flags = GBUF_LEAF;
else else
flags = GBUF_INNER_PARITY(blkno); flags = GBUF_INNER_PARITY(blkno);
if (SpGistPageStoresNulls(page))
flags |= GBUF_NULLS;
lup = GET_LUP(cache, flags); lup = GET_LUP(cache, flags);
...@@ -459,6 +474,7 @@ void ...@@ -459,6 +474,7 @@ void
SpGistInitMetapage(Page page) SpGistInitMetapage(Page page)
{ {
SpGistMetaPageData *metadata; SpGistMetaPageData *metadata;
int i;
SpGistInitPage(page, SPGIST_META); SpGistInitPage(page, SPGIST_META);
metadata = SpGistPageGetMeta(page); metadata = SpGistPageGetMeta(page);
...@@ -466,10 +482,8 @@ SpGistInitMetapage(Page page) ...@@ -466,10 +482,8 @@ SpGistInitMetapage(Page page)
metadata->magicNumber = SPGIST_MAGIC_NUMBER; metadata->magicNumber = SPGIST_MAGIC_NUMBER;
/* initialize last-used-page cache to empty */ /* initialize last-used-page cache to empty */
metadata->lastUsedPages.innerPage[0].blkno = InvalidBlockNumber; for (i = 0; i < SPGIST_CACHED_PAGES; i++)
metadata->lastUsedPages.innerPage[1].blkno = InvalidBlockNumber; metadata->lastUsedPages.cachedPage[i].blkno = InvalidBlockNumber;
metadata->lastUsedPages.innerPage[2].blkno = InvalidBlockNumber;
metadata->lastUsedPages.leafPage.blkno = InvalidBlockNumber;
} }
/* /*
...@@ -490,7 +504,7 @@ spgoptions(PG_FUNCTION_ARGS) ...@@ -490,7 +504,7 @@ spgoptions(PG_FUNCTION_ARGS)
} }
/* /*
* Get the space needed to store a datum of the indicated type. * Get the space needed to store a non-null datum of the indicated type.
* Note the result is already rounded up to a MAXALIGN boundary. * Note the result is already rounded up to a MAXALIGN boundary.
* Also, we follow the SPGiST convention that pass-by-val types are * Also, we follow the SPGiST convention that pass-by-val types are
* just stored in their Datum representation (compare memcpyDatum). * just stored in their Datum representation (compare memcpyDatum).
...@@ -511,7 +525,7 @@ SpGistGetTypeSize(SpGistTypeDesc *att, Datum datum) ...@@ -511,7 +525,7 @@ SpGistGetTypeSize(SpGistTypeDesc *att, Datum datum)
} }
/* /*
* Copy the given datum to *target * Copy the given non-null datum to *target
*/ */
static void static void
memcpyDatum(void *target, SpGistTypeDesc *att, Datum datum) memcpyDatum(void *target, SpGistTypeDesc *att, Datum datum)
...@@ -533,17 +547,20 @@ memcpyDatum(void *target, SpGistTypeDesc *att, Datum datum) ...@@ -533,17 +547,20 @@ memcpyDatum(void *target, SpGistTypeDesc *att, Datum datum)
* Construct a leaf tuple containing the given heap TID and datum value * Construct a leaf tuple containing the given heap TID and datum value
*/ */
SpGistLeafTuple SpGistLeafTuple
spgFormLeafTuple(SpGistState *state, ItemPointer heapPtr, Datum datum) spgFormLeafTuple(SpGistState *state, ItemPointer heapPtr,
Datum datum, bool isnull)
{ {
SpGistLeafTuple tup; SpGistLeafTuple tup;
unsigned int size; unsigned int size;
/* compute space needed (note result is already maxaligned) */ /* compute space needed (note result is already maxaligned) */
size = SGLTHDRSZ + SpGistGetTypeSize(&state->attType, datum); size = SGLTHDRSZ;
if (!isnull)
size += SpGistGetTypeSize(&state->attType, datum);
/* /*
* Ensure that we can replace the tuple with a dead tuple later. This * Ensure that we can replace the tuple with a dead tuple later. This
* test is unnecessary given current tuple layouts, but let's be safe. * test is unnecessary when !isnull, but let's be safe.
*/ */
if (size < SGDTSIZE) if (size < SGDTSIZE)
size = SGDTSIZE; size = SGDTSIZE;
...@@ -554,6 +571,7 @@ spgFormLeafTuple(SpGistState *state, ItemPointer heapPtr, Datum datum) ...@@ -554,6 +571,7 @@ spgFormLeafTuple(SpGistState *state, ItemPointer heapPtr, Datum datum)
tup->size = size; tup->size = size;
tup->nextOffset = InvalidOffsetNumber; tup->nextOffset = InvalidOffsetNumber;
tup->heapPtr = *heapPtr; tup->heapPtr = *heapPtr;
if (!isnull)
memcpyDatum(SGLTDATAPTR(tup), &state->attType, datum); memcpyDatum(SGLTDATAPTR(tup), &state->attType, datum);
return tup; return tup;
......
...@@ -307,7 +307,7 @@ vacuumLeafPage(spgBulkDeleteState *bds, Relation index, Buffer buffer) ...@@ -307,7 +307,7 @@ vacuumLeafPage(spgBulkDeleteState *bds, Relation index, Buffer buffer)
} }
/* /*
* Vacuum the root page when it is a leaf * Vacuum a root page when it is also a leaf
* *
* On the root, we just delete any dead leaf tuples; no fancy business * On the root, we just delete any dead leaf tuples; no fancy business
*/ */
...@@ -321,6 +321,7 @@ vacuumLeafRoot(spgBulkDeleteState *bds, Relation index, Buffer buffer) ...@@ -321,6 +321,7 @@ vacuumLeafRoot(spgBulkDeleteState *bds, Relation index, Buffer buffer)
OffsetNumber i, OffsetNumber i,
max = PageGetMaxOffsetNumber(page); max = PageGetMaxOffsetNumber(page);
xlrec.blkno = BufferGetBlockNumber(buffer);
xlrec.nDelete = 0; xlrec.nDelete = 0;
/* Scan page, identify tuples to delete, accumulate stats */ /* Scan page, identify tuples to delete, accumulate stats */
...@@ -537,7 +538,7 @@ spgvacuumpage(spgBulkDeleteState *bds, BlockNumber blkno) ...@@ -537,7 +538,7 @@ spgvacuumpage(spgBulkDeleteState *bds, BlockNumber blkno)
} }
else if (SpGistPageIsLeaf(page)) else if (SpGistPageIsLeaf(page))
{ {
if (blkno == SPGIST_HEAD_BLKNO) if (SpGistBlockIsRoot(blkno))
{ {
vacuumLeafRoot(bds, index, buffer); vacuumLeafRoot(bds, index, buffer);
/* no need for vacuumRedirectAndPlaceholder */ /* no need for vacuumRedirectAndPlaceholder */
...@@ -560,7 +561,7 @@ spgvacuumpage(spgBulkDeleteState *bds, BlockNumber blkno) ...@@ -560,7 +561,7 @@ spgvacuumpage(spgBulkDeleteState *bds, BlockNumber blkno)
* put a new tuple. Otherwise, check for empty/deletable page, and * put a new tuple. Otherwise, check for empty/deletable page, and
* make sure FSM knows about it. * make sure FSM knows about it.
*/ */
if (blkno != SPGIST_HEAD_BLKNO) if (!SpGistBlockIsRoot(blkno))
{ {
/* If page is now empty, mark it deleted */ /* If page is now empty, mark it deleted */
if (PageIsEmpty(page) && !SpGistPageIsDeleted(page)) if (PageIsEmpty(page) && !SpGistPageIsDeleted(page))
...@@ -598,7 +599,7 @@ spgvacuumscan(spgBulkDeleteState *bds) ...@@ -598,7 +599,7 @@ spgvacuumscan(spgBulkDeleteState *bds)
/* Finish setting up spgBulkDeleteState */ /* Finish setting up spgBulkDeleteState */
initSpGistState(&bds->spgstate, index); initSpGistState(&bds->spgstate, index);
bds->OldestXmin = GetOldestXmin(true, false); bds->OldestXmin = GetOldestXmin(true, false);
bds->lastFilledBlock = SPGIST_HEAD_BLKNO; bds->lastFilledBlock = SPGIST_LAST_FIXED_BLKNO;
/* /*
* Reset counts that will be incremented during the scan; needed in case * Reset counts that will be incremented during the scan; needed in case
...@@ -619,7 +620,7 @@ spgvacuumscan(spgBulkDeleteState *bds) ...@@ -619,7 +620,7 @@ spgvacuumscan(spgBulkDeleteState *bds)
* delete some deletable tuples. See more extensive comments about * delete some deletable tuples. See more extensive comments about
* this in btvacuumscan(). * this in btvacuumscan().
*/ */
blkno = SPGIST_HEAD_BLKNO; blkno = SPGIST_METAPAGE_BLKNO + 1;
for (;;) for (;;)
{ {
/* Get the current relation length */ /* Get the current relation length */
...@@ -648,6 +649,12 @@ spgvacuumscan(spgBulkDeleteState *bds) ...@@ -648,6 +649,12 @@ spgvacuumscan(spgBulkDeleteState *bds)
* XXX disabled because it's unsafe due to possible concurrent inserts. * XXX disabled because it's unsafe due to possible concurrent inserts.
* We'd have to rescan the pages to make sure they're still empty, and it * We'd have to rescan the pages to make sure they're still empty, and it
* doesn't seem worth it. Note that btree doesn't do this either. * doesn't seem worth it. Note that btree doesn't do this either.
*
* Another reason not to truncate is that it could invalidate the cached
* pages-with-freespace pointers in the metapage and other backends'
* relation caches, that is leave them pointing to nonexistent pages.
* Adding RelationGetNumberOfBlocks calls to protect the places that use
* those pointers would be unduly expensive.
*/ */
#ifdef NOT_USED #ifdef NOT_USED
if (num_pages > bds->lastFilledBlock + 1) if (num_pages > bds->lastFilledBlock + 1)
......
...@@ -84,7 +84,7 @@ spgRedoCreateIndex(XLogRecPtr lsn, XLogRecord *record) ...@@ -84,7 +84,7 @@ spgRedoCreateIndex(XLogRecPtr lsn, XLogRecord *record)
MarkBufferDirty(buffer); MarkBufferDirty(buffer);
UnlockReleaseBuffer(buffer); UnlockReleaseBuffer(buffer);
buffer = XLogReadBuffer(*node, SPGIST_HEAD_BLKNO, true); buffer = XLogReadBuffer(*node, SPGIST_ROOT_BLKNO, true);
Assert(BufferIsValid(buffer)); Assert(BufferIsValid(buffer));
SpGistInitBuffer(buffer, SPGIST_LEAF); SpGistInitBuffer(buffer, SPGIST_LEAF);
page = (Page) BufferGetPage(buffer); page = (Page) BufferGetPage(buffer);
...@@ -92,6 +92,15 @@ spgRedoCreateIndex(XLogRecPtr lsn, XLogRecord *record) ...@@ -92,6 +92,15 @@ spgRedoCreateIndex(XLogRecPtr lsn, XLogRecord *record)
PageSetTLI(page, ThisTimeLineID); PageSetTLI(page, ThisTimeLineID);
MarkBufferDirty(buffer); MarkBufferDirty(buffer);
UnlockReleaseBuffer(buffer); UnlockReleaseBuffer(buffer);
buffer = XLogReadBuffer(*node, SPGIST_NULL_BLKNO, true);
Assert(BufferIsValid(buffer));
SpGistInitBuffer(buffer, SPGIST_LEAF | SPGIST_NULLS);
page = (Page) BufferGetPage(buffer);
PageSetLSN(page, lsn);
PageSetTLI(page, ThisTimeLineID);
MarkBufferDirty(buffer);
UnlockReleaseBuffer(buffer);
} }
static void static void
...@@ -116,7 +125,8 @@ spgRedoAddLeaf(XLogRecPtr lsn, XLogRecord *record) ...@@ -116,7 +125,8 @@ spgRedoAddLeaf(XLogRecPtr lsn, XLogRecord *record)
page = BufferGetPage(buffer); page = BufferGetPage(buffer);
if (xldata->newPage) if (xldata->newPage)
SpGistInitBuffer(buffer, SPGIST_LEAF); SpGistInitBuffer(buffer,
SPGIST_LEAF | (xldata->storesNulls ? SPGIST_NULLS : 0));
if (!XLByteLE(lsn, PageGetLSN(page))) if (!XLByteLE(lsn, PageGetLSN(page)))
{ {
...@@ -218,7 +228,8 @@ spgRedoMoveLeafs(XLogRecPtr lsn, XLogRecord *record) ...@@ -218,7 +228,8 @@ spgRedoMoveLeafs(XLogRecPtr lsn, XLogRecord *record)
page = BufferGetPage(buffer); page = BufferGetPage(buffer);
if (xldata->newPage) if (xldata->newPage)
SpGistInitBuffer(buffer, SPGIST_LEAF); SpGistInitBuffer(buffer,
SPGIST_LEAF | (xldata->storesNulls ? SPGIST_NULLS : 0));
if (!XLByteLE(lsn, PageGetLSN(page))) if (!XLByteLE(lsn, PageGetLSN(page)))
{ {
...@@ -344,6 +355,7 @@ spgRedoAddNode(XLogRecPtr lsn, XLogRecord *record) ...@@ -344,6 +355,7 @@ spgRedoAddNode(XLogRecPtr lsn, XLogRecord *record)
{ {
page = BufferGetPage(buffer); page = BufferGetPage(buffer);
/* AddNode is not used for nulls pages */
if (xldata->newPage) if (xldata->newPage)
SpGistInitBuffer(buffer, 0); SpGistInitBuffer(buffer, 0);
...@@ -464,6 +476,7 @@ spgRedoSplitTuple(XLogRecPtr lsn, XLogRecord *record) ...@@ -464,6 +476,7 @@ spgRedoSplitTuple(XLogRecPtr lsn, XLogRecord *record)
{ {
page = BufferGetPage(buffer); page = BufferGetPage(buffer);
/* SplitTuple is not used for nulls pages */
if (xldata->newPage) if (xldata->newPage)
SpGistInitBuffer(buffer, 0); SpGistInitBuffer(buffer, 0);
...@@ -545,7 +558,7 @@ spgRedoPickSplit(XLogRecPtr lsn, XLogRecord *record) ...@@ -545,7 +558,7 @@ spgRedoPickSplit(XLogRecPtr lsn, XLogRecord *record)
*/ */
bbi = 0; bbi = 0;
if (xldata->blknoSrc == SPGIST_HEAD_BLKNO) if (SpGistBlockIsRoot(xldata->blknoSrc))
{ {
/* when splitting root, we touch it only in the guise of new inner */ /* when splitting root, we touch it only in the guise of new inner */
srcBuffer = InvalidBuffer; srcBuffer = InvalidBuffer;
...@@ -557,7 +570,8 @@ spgRedoPickSplit(XLogRecPtr lsn, XLogRecord *record) ...@@ -557,7 +570,8 @@ spgRedoPickSplit(XLogRecPtr lsn, XLogRecord *record)
Assert(BufferIsValid(srcBuffer)); Assert(BufferIsValid(srcBuffer));
page = (Page) BufferGetPage(srcBuffer); page = (Page) BufferGetPage(srcBuffer);
SpGistInitBuffer(srcBuffer, SPGIST_LEAF); SpGistInitBuffer(srcBuffer,
SPGIST_LEAF | (xldata->storesNulls ? SPGIST_NULLS : 0));
/* don't update LSN etc till we're done with it */ /* don't update LSN etc till we're done with it */
} }
else else
...@@ -612,7 +626,8 @@ spgRedoPickSplit(XLogRecPtr lsn, XLogRecord *record) ...@@ -612,7 +626,8 @@ spgRedoPickSplit(XLogRecPtr lsn, XLogRecord *record)
Assert(BufferIsValid(destBuffer)); Assert(BufferIsValid(destBuffer));
page = (Page) BufferGetPage(destBuffer); page = (Page) BufferGetPage(destBuffer);
SpGistInitBuffer(destBuffer, SPGIST_LEAF); SpGistInitBuffer(destBuffer,
SPGIST_LEAF | (xldata->storesNulls ? SPGIST_NULLS : 0));
/* don't update LSN etc till we're done with it */ /* don't update LSN etc till we're done with it */
} }
else else
...@@ -678,7 +693,8 @@ spgRedoPickSplit(XLogRecPtr lsn, XLogRecord *record) ...@@ -678,7 +693,8 @@ spgRedoPickSplit(XLogRecPtr lsn, XLogRecord *record)
page = BufferGetPage(buffer); page = BufferGetPage(buffer);
if (xldata->initInner) if (xldata->initInner)
SpGistInitBuffer(buffer, 0); SpGistInitBuffer(buffer,
(xldata->storesNulls ? SPGIST_NULLS : 0));
if (!XLByteLE(lsn, PageGetLSN(page))) if (!XLByteLE(lsn, PageGetLSN(page)))
{ {
...@@ -709,7 +725,7 @@ spgRedoPickSplit(XLogRecPtr lsn, XLogRecord *record) ...@@ -709,7 +725,7 @@ spgRedoPickSplit(XLogRecPtr lsn, XLogRecord *record)
if (xldata->blknoParent == InvalidBlockNumber) if (xldata->blknoParent == InvalidBlockNumber)
{ {
/* no parent cause we split the root */ /* no parent cause we split the root */
Assert(xldata->blknoInner == SPGIST_HEAD_BLKNO); Assert(SpGistBlockIsRoot(xldata->blknoInner));
} }
else if (xldata->blknoInner != xldata->blknoParent) else if (xldata->blknoInner != xldata->blknoParent)
{ {
...@@ -842,7 +858,7 @@ spgRedoVacuumRoot(XLogRecPtr lsn, XLogRecord *record) ...@@ -842,7 +858,7 @@ spgRedoVacuumRoot(XLogRecPtr lsn, XLogRecord *record)
if (!(record->xl_info & XLR_BKP_BLOCK_1)) if (!(record->xl_info & XLR_BKP_BLOCK_1))
{ {
buffer = XLogReadBuffer(xldata->node, SPGIST_HEAD_BLKNO, false); buffer = XLogReadBuffer(xldata->node, xldata->blkno, false);
if (BufferIsValid(buffer)) if (BufferIsValid(buffer))
{ {
page = BufferGetPage(buffer); page = BufferGetPage(buffer);
...@@ -1039,7 +1055,8 @@ spg_desc(StringInfo buf, uint8 xl_info, char *rec) ...@@ -1039,7 +1055,8 @@ spg_desc(StringInfo buf, uint8 xl_info, char *rec)
break; break;
case XLOG_SPGIST_VACUUM_ROOT: case XLOG_SPGIST_VACUUM_ROOT:
out_target(buf, ((spgxlogVacuumRoot *) rec)->node); out_target(buf, ((spgxlogVacuumRoot *) rec)->node);
appendStringInfo(buf, "vacuum leaf tuples on root page"); appendStringInfo(buf, "vacuum leaf tuples on root page %u",
((spgxlogVacuumRoot *) rec)->blkno);
break; break;
case XLOG_SPGIST_VACUUM_REDIRECT: case XLOG_SPGIST_VACUUM_REDIRECT:
out_target(buf, ((spgxlogVacuumRedirect *) rec)->node); out_target(buf, ((spgxlogVacuumRedirect *) rec)->node);
......
...@@ -21,8 +21,15 @@ ...@@ -21,8 +21,15 @@
/* Page numbers of fixed-location pages */ /* Page numbers of fixed-location pages */
#define SPGIST_METAPAGE_BLKNO (0) #define SPGIST_METAPAGE_BLKNO (0) /* metapage */
#define SPGIST_HEAD_BLKNO (1) #define SPGIST_ROOT_BLKNO (1) /* root for normal entries */
#define SPGIST_NULL_BLKNO (2) /* root for null-value entries */
#define SPGIST_LAST_FIXED_BLKNO SPGIST_NULL_BLKNO
#define SpGistBlockIsRoot(blkno) \
((blkno) == SPGIST_ROOT_BLKNO || (blkno) == SPGIST_NULL_BLKNO)
#define SpGistBlockIsFixed(blkno) \
((BlockNumber) (blkno) <= (BlockNumber) SPGIST_LAST_FIXED_BLKNO)
/* /*
* Contents of page special space on SPGiST index pages * Contents of page special space on SPGiST index pages
...@@ -42,15 +49,14 @@ typedef SpGistPageOpaqueData *SpGistPageOpaque; ...@@ -42,15 +49,14 @@ typedef SpGistPageOpaqueData *SpGistPageOpaque;
#define SPGIST_META (1<<0) #define SPGIST_META (1<<0)
#define SPGIST_DELETED (1<<1) #define SPGIST_DELETED (1<<1)
#define SPGIST_LEAF (1<<2) #define SPGIST_LEAF (1<<2)
#define SPGIST_NULLS (1<<3)
#define SpGistPageGetOpaque(page) ((SpGistPageOpaque) PageGetSpecialPointer(page)) #define SpGistPageGetOpaque(page) ((SpGistPageOpaque) PageGetSpecialPointer(page))
#define SpGistPageIsMeta(page) (SpGistPageGetOpaque(page)->flags & SPGIST_META) #define SpGistPageIsMeta(page) (SpGistPageGetOpaque(page)->flags & SPGIST_META)
#define SpGistPageIsDeleted(page) (SpGistPageGetOpaque(page)->flags & SPGIST_DELETED) #define SpGistPageIsDeleted(page) (SpGistPageGetOpaque(page)->flags & SPGIST_DELETED)
#define SpGistPageSetDeleted(page) (SpGistPageGetOpaque(page)->flags |= SPGIST_DELETED) #define SpGistPageSetDeleted(page) (SpGistPageGetOpaque(page)->flags |= SPGIST_DELETED)
#define SpGistPageSetNonDeleted(page) (SpGistPageGetOpaque(page)->flags &= ~SPGIST_DELETED)
#define SpGistPageIsLeaf(page) (SpGistPageGetOpaque(page)->flags & SPGIST_LEAF) #define SpGistPageIsLeaf(page) (SpGistPageGetOpaque(page)->flags & SPGIST_LEAF)
#define SpGistPageSetLeaf(page) (SpGistPageGetOpaque(page)->flags |= SPGIST_LEAF) #define SpGistPageStoresNulls(page) (SpGistPageGetOpaque(page)->flags & SPGIST_NULLS)
#define SpGistPageSetInner(page) (SpGistPageGetOpaque(page)->flags &= ~SPGIST_LEAF)
/* /*
* The page ID is for the convenience of pg_filedump and similar utilities, * The page ID is for the convenience of pg_filedump and similar utilities,
...@@ -67,14 +73,16 @@ typedef SpGistPageOpaqueData *SpGistPageOpaque; ...@@ -67,14 +73,16 @@ typedef SpGistPageOpaqueData *SpGistPageOpaque;
*/ */
typedef struct SpGistLastUsedPage typedef struct SpGistLastUsedPage
{ {
BlockNumber blkno; /* block number of described page */ BlockNumber blkno; /* block number, or InvalidBlockNumber */
int freeSpace; /* its free space (could be obsolete!) */ int freeSpace; /* page's free space (could be obsolete!) */
} SpGistLastUsedPage; } SpGistLastUsedPage;
/* Note: indexes in cachedPage[] match flag assignments for SpGistGetBuffer */
#define SPGIST_CACHED_PAGES 8
typedef struct SpGistLUPCache typedef struct SpGistLUPCache
{ {
SpGistLastUsedPage innerPage[3]; /* one per triple-parity group */ SpGistLastUsedPage cachedPage[SPGIST_CACHED_PAGES];
SpGistLastUsedPage leafPage;
} SpGistLUPCache; } SpGistLUPCache;
/* /*
...@@ -86,7 +94,7 @@ typedef struct SpGistMetaPageData ...@@ -86,7 +94,7 @@ typedef struct SpGistMetaPageData
SpGistLUPCache lastUsedPages; /* shared storage of last-used info */ SpGistLUPCache lastUsedPages; /* shared storage of last-used info */
} SpGistMetaPageData; } SpGistMetaPageData;
#define SPGIST_MAGIC_NUMBER (0xBA0BABED) #define SPGIST_MAGIC_NUMBER (0xBA0BABEE)
#define SpGistPageGetMeta(p) \ #define SpGistPageGetMeta(p) \
((SpGistMetaPageData *) PageGetContents(p)) ((SpGistMetaPageData *) PageGetContents(p))
...@@ -266,7 +274,15 @@ typedef SpGistNodeTupleData *SpGistNodeTuple; ...@@ -266,7 +274,15 @@ typedef SpGistNodeTupleData *SpGistNodeTuple;
* node (which must be on the same page). But when the root page is a leaf * node (which must be on the same page). But when the root page is a leaf
* page, we don't chain its tuples, so nextOffset is always 0 on the root. * page, we don't chain its tuples, so nextOffset is always 0 on the root.
* *
* size must be a multiple of MAXALIGN * size must be a multiple of MAXALIGN; also, it must be at least SGDTSIZE
* so that the tuple can be converted to REDIRECT status later. (This
* restriction only adds bytes for the null-datum case, otherwise alignment
* restrictions force it anyway.)
*
* In a leaf tuple for a NULL indexed value, there's no useful datum value;
* however, the SGDTSIZE limit ensures that's there's a Datum word there
* anyway, so SGLTDATUM can be applied safely as long as you don't do
* anything with the result.
*/ */
typedef struct SpGistLeafTupleData typedef struct SpGistLeafTupleData
{ {
...@@ -397,6 +413,7 @@ typedef struct spgxlogAddLeaf ...@@ -397,6 +413,7 @@ typedef struct spgxlogAddLeaf
BlockNumber blknoLeaf; /* destination page for leaf tuple */ BlockNumber blknoLeaf; /* destination page for leaf tuple */
bool newPage; /* init dest page? */ bool newPage; /* init dest page? */
bool storesNulls; /* page is in the nulls tree? */
OffsetNumber offnumLeaf; /* offset where leaf tuple gets placed */ OffsetNumber offnumLeaf; /* offset where leaf tuple gets placed */
OffsetNumber offnumHeadLeaf; /* offset of head tuple in chain, if any */ OffsetNumber offnumHeadLeaf; /* offset of head tuple in chain, if any */
...@@ -419,6 +436,7 @@ typedef struct spgxlogMoveLeafs ...@@ -419,6 +436,7 @@ typedef struct spgxlogMoveLeafs
uint16 nMoves; /* number of tuples moved from source page */ uint16 nMoves; /* number of tuples moved from source page */
bool newPage; /* init dest page? */ bool newPage; /* init dest page? */
bool replaceDead; /* are we replacing a DEAD source tuple? */ bool replaceDead; /* are we replacing a DEAD source tuple? */
bool storesNulls; /* pages are in the nulls tree? */
BlockNumber blknoParent; /* where the parent downlink is */ BlockNumber blknoParent; /* where the parent downlink is */
OffsetNumber offnumParent; OffsetNumber offnumParent;
...@@ -502,6 +520,8 @@ typedef struct spgxlogPickSplit ...@@ -502,6 +520,8 @@ typedef struct spgxlogPickSplit
OffsetNumber offnumInner; OffsetNumber offnumInner;
bool initInner; /* re-init the Inner page? */ bool initInner; /* re-init the Inner page? */
bool storesNulls; /* pages are in the nulls tree? */
BlockNumber blknoParent; /* where the parent downlink is, if any */ BlockNumber blknoParent; /* where the parent downlink is, if any */
OffsetNumber offnumParent; OffsetNumber offnumParent;
uint16 nodeI; uint16 nodeI;
...@@ -553,9 +573,10 @@ typedef struct spgxlogVacuumLeaf ...@@ -553,9 +573,10 @@ typedef struct spgxlogVacuumLeaf
typedef struct spgxlogVacuumRoot typedef struct spgxlogVacuumRoot
{ {
/* vacuum root page when it is a leaf */ /* vacuum a root page when it is also a leaf */
RelFileNode node; RelFileNode node;
BlockNumber blkno; /* block number to clean */
uint16 nDelete; /* number of tuples to delete */ uint16 nDelete; /* number of tuples to delete */
spgxlogState stateSrc; spgxlogState stateSrc;
...@@ -580,10 +601,18 @@ typedef struct spgxlogVacuumRedirect ...@@ -580,10 +601,18 @@ typedef struct spgxlogVacuumRedirect
* page in the same triple-parity group as the specified block number. * page in the same triple-parity group as the specified block number.
* (Typically, this should be GBUF_INNER_PARITY(parentBlockNumber + 1) * (Typically, this should be GBUF_INNER_PARITY(parentBlockNumber + 1)
* to follow the rule described in spgist/README.) * to follow the rule described in spgist/README.)
* In addition, GBUF_NULLS can be OR'd in to get a page for storage of
* null-valued tuples.
*
* Note: these flag values are used as indexes into lastUsedPages.
*/ */
#define GBUF_PARITY_MASK 0x03 #define GBUF_LEAF 0x03
#define GBUF_LEAF 0x04
#define GBUF_INNER_PARITY(x) ((x) % 3) #define GBUF_INNER_PARITY(x) ((x) % 3)
#define GBUF_NULLS 0x04
#define GBUF_PARITY_MASK 0x03
#define GBUF_REQ_LEAF(flags) (((flags) & GBUF_PARITY_MASK) == GBUF_LEAF)
#define GBUF_REQ_NULLS(flags) ((flags) & GBUF_NULLS)
/* spgutils.c */ /* spgutils.c */
extern SpGistCache *spgGetCache(Relation index); extern SpGistCache *spgGetCache(Relation index);
...@@ -598,7 +627,8 @@ extern void SpGistInitBuffer(Buffer b, uint16 f); ...@@ -598,7 +627,8 @@ extern void SpGistInitBuffer(Buffer b, uint16 f);
extern void SpGistInitMetapage(Page page); extern void SpGistInitMetapage(Page page);
extern unsigned int SpGistGetTypeSize(SpGistTypeDesc *att, Datum datum); extern unsigned int SpGistGetTypeSize(SpGistTypeDesc *att, Datum datum);
extern SpGistLeafTuple spgFormLeafTuple(SpGistState *state, extern SpGistLeafTuple spgFormLeafTuple(SpGistState *state,
ItemPointer heapPtr, Datum datum); ItemPointer heapPtr,
Datum datum, bool isnull);
extern SpGistNodeTuple spgFormNodeTuple(SpGistState *state, extern SpGistNodeTuple spgFormNodeTuple(SpGistState *state,
Datum label, bool isnull); Datum label, bool isnull);
extern SpGistInnerTuple spgFormInnerTuple(SpGistState *state, extern SpGistInnerTuple spgFormInnerTuple(SpGistState *state,
...@@ -621,6 +651,6 @@ extern void spgPageIndexMultiDelete(SpGistState *state, Page page, ...@@ -621,6 +651,6 @@ extern void spgPageIndexMultiDelete(SpGistState *state, Page page,
int firststate, int reststate, int firststate, int reststate,
BlockNumber blkno, OffsetNumber offnum); BlockNumber blkno, OffsetNumber offnum);
extern void spgdoinsert(Relation index, SpGistState *state, extern void spgdoinsert(Relation index, SpGistState *state,
ItemPointer heapPtr, Datum datum); ItemPointer heapPtr, Datum datum, bool isnull);
#endif /* SPGIST_PRIVATE_H */ #endif /* SPGIST_PRIVATE_H */
...@@ -53,6 +53,6 @@ ...@@ -53,6 +53,6 @@
*/ */
/* yyyymmddN */ /* yyyymmddN */
#define CATALOG_VERSION_NO 201203041 #define CATALOG_VERSION_NO 201203111
#endif #endif
...@@ -129,7 +129,7 @@ DESCR("GiST index access method"); ...@@ -129,7 +129,7 @@ DESCR("GiST index access method");
DATA(insert OID = 2742 ( gin 0 5 f f f f t t f f t f f 0 gininsert ginbeginscan - gingetbitmap ginrescan ginendscan ginmarkpos ginrestrpos ginbuild ginbuildempty ginbulkdelete ginvacuumcleanup - gincostestimate ginoptions )); DATA(insert OID = 2742 ( gin 0 5 f f f f t t f f t f f 0 gininsert ginbeginscan - gingetbitmap ginrescan ginendscan ginmarkpos ginrestrpos ginbuild ginbuildempty ginbulkdelete ginvacuumcleanup - gincostestimate ginoptions ));
DESCR("GIN index access method"); DESCR("GIN index access method");
#define GIN_AM_OID 2742 #define GIN_AM_OID 2742
DATA(insert OID = 4000 ( spgist 0 5 f f f f f f f f f f f 0 spginsert spgbeginscan spggettuple spggetbitmap spgrescan spgendscan spgmarkpos spgrestrpos spgbuild spgbuildempty spgbulkdelete spgvacuumcleanup spgcanreturn spgcostestimate spgoptions )); DATA(insert OID = 4000 ( spgist 0 5 f f f f f t f t f f f 0 spginsert spgbeginscan spggettuple spggetbitmap spgrescan spgendscan spgmarkpos spgrestrpos spgbuild spgbuildempty spgbulkdelete spgvacuumcleanup spgcanreturn spgcostestimate spgoptions ));
DESCR("SP-GiST index access method"); DESCR("SP-GiST index access method");
#define SPGIST_AM_OID 4000 #define SPGIST_AM_OID 4000
......
...@@ -68,6 +68,7 @@ CREATE TABLE quad_point_tbl AS ...@@ -68,6 +68,7 @@ CREATE TABLE quad_point_tbl AS
SELECT point(unique1,unique2) AS p FROM tenk1; SELECT point(unique1,unique2) AS p FROM tenk1;
INSERT INTO quad_point_tbl INSERT INTO quad_point_tbl
SELECT '(333.0,400.0)'::point FROM generate_series(1,1000); SELECT '(333.0,400.0)'::point FROM generate_series(1,1000);
INSERT INTO quad_point_tbl VALUES (NULL), (NULL), (NULL);
CREATE INDEX sp_quad_ind ON quad_point_tbl USING spgist (p); CREATE INDEX sp_quad_ind ON quad_point_tbl USING spgist (p);
CREATE TABLE kd_point_tbl AS SELECT * FROM quad_point_tbl; CREATE TABLE kd_point_tbl AS SELECT * FROM quad_point_tbl;
CREATE INDEX sp_kd_ind ON kd_point_tbl USING spgist (p kd_point_ops); CREATE INDEX sp_kd_ind ON kd_point_tbl USING spgist (p kd_point_ops);
...@@ -227,6 +228,24 @@ SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0 ...@@ -227,6 +228,24 @@ SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0
(10,10) (10,10)
(4 rows) (4 rows)
SELECT count(*) FROM quad_point_tbl WHERE p IS NULL;
count
-------
3
(1 row)
SELECT count(*) FROM quad_point_tbl WHERE p IS NOT NULL;
count
-------
11000
(1 row)
SELECT count(*) FROM quad_point_tbl;
count
-------
11003
(1 row)
SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)'; SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)';
count count
------- -------
...@@ -678,6 +697,50 @@ SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0 ...@@ -678,6 +697,50 @@ SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0
(10,10) (10,10)
(4 rows) (4 rows)
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p IS NULL;
QUERY PLAN
-----------------------------------------------------------
Aggregate
-> Index Only Scan using sp_quad_ind on quad_point_tbl
Index Cond: (p IS NULL)
(3 rows)
SELECT count(*) FROM quad_point_tbl WHERE p IS NULL;
count
-------
3
(1 row)
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p IS NOT NULL;
QUERY PLAN
-----------------------------------------------------------
Aggregate
-> Index Only Scan using sp_quad_ind on quad_point_tbl
Index Cond: (p IS NOT NULL)
(3 rows)
SELECT count(*) FROM quad_point_tbl WHERE p IS NOT NULL;
count
-------
11000
(1 row)
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl;
QUERY PLAN
-----------------------------------------------------------
Aggregate
-> Index Only Scan using sp_quad_ind on quad_point_tbl
(2 rows)
SELECT count(*) FROM quad_point_tbl;
count
-------
11003
(1 row)
EXPLAIN (COSTS OFF) EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)'; SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)';
QUERY PLAN QUERY PLAN
...@@ -1108,6 +1171,55 @@ SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0 ...@@ -1108,6 +1171,55 @@ SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0
(10,10) (10,10)
(4 rows) (4 rows)
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p IS NULL;
QUERY PLAN
----------------------------------------------
Aggregate
-> Bitmap Heap Scan on quad_point_tbl
Recheck Cond: (p IS NULL)
-> Bitmap Index Scan on sp_quad_ind
Index Cond: (p IS NULL)
(5 rows)
SELECT count(*) FROM quad_point_tbl WHERE p IS NULL;
count
-------
3
(1 row)
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p IS NOT NULL;
QUERY PLAN
----------------------------------------------
Aggregate
-> Bitmap Heap Scan on quad_point_tbl
Recheck Cond: (p IS NOT NULL)
-> Bitmap Index Scan on sp_quad_ind
Index Cond: (p IS NOT NULL)
(5 rows)
SELECT count(*) FROM quad_point_tbl WHERE p IS NOT NULL;
count
-------
11000
(1 row)
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl;
QUERY PLAN
----------------------------------------------
Aggregate
-> Bitmap Heap Scan on quad_point_tbl
-> Bitmap Index Scan on sp_quad_ind
(3 rows)
SELECT count(*) FROM quad_point_tbl;
count
-------
11003
(1 row)
EXPLAIN (COSTS OFF) EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)'; SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)';
QUERY PLAN QUERY PLAN
......
...@@ -102,6 +102,8 @@ CREATE TABLE quad_point_tbl AS ...@@ -102,6 +102,8 @@ CREATE TABLE quad_point_tbl AS
INSERT INTO quad_point_tbl INSERT INTO quad_point_tbl
SELECT '(333.0,400.0)'::point FROM generate_series(1,1000); SELECT '(333.0,400.0)'::point FROM generate_series(1,1000);
INSERT INTO quad_point_tbl VALUES (NULL), (NULL), (NULL);
CREATE INDEX sp_quad_ind ON quad_point_tbl USING spgist (p); CREATE INDEX sp_quad_ind ON quad_point_tbl USING spgist (p);
CREATE TABLE kd_point_tbl AS SELECT * FROM quad_point_tbl; CREATE TABLE kd_point_tbl AS SELECT * FROM quad_point_tbl;
...@@ -172,6 +174,12 @@ SELECT * FROM point_tbl WHERE f1 IS NOT NULL ORDER BY f1 <-> '0,1'; ...@@ -172,6 +174,12 @@ SELECT * FROM point_tbl WHERE f1 IS NOT NULL ORDER BY f1 <-> '0,1';
SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0,1'; SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0,1';
SELECT count(*) FROM quad_point_tbl WHERE p IS NULL;
SELECT count(*) FROM quad_point_tbl WHERE p IS NOT NULL;
SELECT count(*) FROM quad_point_tbl;
SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)'; SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)';
SELECT count(*) FROM quad_point_tbl WHERE box '(200,200,1000,1000)' @> p; SELECT count(*) FROM quad_point_tbl WHERE box '(200,200,1000,1000)' @> p;
...@@ -305,6 +313,18 @@ EXPLAIN (COSTS OFF) ...@@ -305,6 +313,18 @@ EXPLAIN (COSTS OFF)
SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0,1'; SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0,1';
SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0,1'; SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0,1';
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p IS NULL;
SELECT count(*) FROM quad_point_tbl WHERE p IS NULL;
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p IS NOT NULL;
SELECT count(*) FROM quad_point_tbl WHERE p IS NOT NULL;
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl;
SELECT count(*) FROM quad_point_tbl;
EXPLAIN (COSTS OFF) EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)'; SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)';
SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)'; SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)';
...@@ -422,6 +442,18 @@ EXPLAIN (COSTS OFF) ...@@ -422,6 +442,18 @@ EXPLAIN (COSTS OFF)
SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0,1'; SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0,1';
SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0,1'; SELECT * FROM point_tbl WHERE f1 <@ '(-10,-10),(10,10)':: box ORDER BY f1 <-> '0,1';
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p IS NULL;
SELECT count(*) FROM quad_point_tbl WHERE p IS NULL;
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p IS NOT NULL;
SELECT count(*) FROM quad_point_tbl WHERE p IS NOT NULL;
EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl;
SELECT count(*) FROM quad_point_tbl;
EXPLAIN (COSTS OFF) EXPLAIN (COSTS OFF)
SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)'; SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)';
SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)'; SELECT count(*) FROM quad_point_tbl WHERE p <@ box '(200,200,1000,1000)';
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment