Commit 72ccf55c authored by Tomas Vondra's avatar Tomas Vondra

Move IS [NOT] NULL handling from BRIN support functions

The handling of IS [NOT] NULL clauses is independent of an opclass, and
most of the code was exactly the same in both minmax and inclusion. So
instead move the code from support procedures to the AM.

This simplifies the code - especially the support procedures - quite a
bit, as they don't need to care about NULL values and flags at all. It
also means the IS [NOT] NULL clauses can be evaluated without invoking
the support procedure.

Author: Tomas Vondra <tomas.vondra@postgresql.org>
Author: Nikita Glukhov <n.gluhov@postgrespro.ru>
Reviewed-by: default avatarNikita Glukhov <n.gluhov@postgrespro.ru>
Reviewed-by: default avatarMark Dilger <hornschnorter@gmail.com>
Reviewed-by: default avatarAlexander Korotkov <aekorotkov@gmail.com>
Reviewed-by: default avatarMasahiko Sawada <masahiko.sawada@enterprisedb.com>
Reviewed-by: default avatarJohn Naylor <john.naylor@enterprisedb.com>
Discussion: https://postgr.es/m/c1138ead-7668-f0e1-0638-c3be3237e812@2ndquadrant.com
parent a1c649d8
...@@ -35,6 +35,7 @@ ...@@ -35,6 +35,7 @@
#include "storage/freespace.h" #include "storage/freespace.h"
#include "utils/acl.h" #include "utils/acl.h"
#include "utils/builtins.h" #include "utils/builtins.h"
#include "utils/datum.h"
#include "utils/index_selfuncs.h" #include "utils/index_selfuncs.h"
#include "utils/memutils.h" #include "utils/memutils.h"
#include "utils/rel.h" #include "utils/rel.h"
...@@ -77,7 +78,9 @@ static void form_and_insert_tuple(BrinBuildState *state); ...@@ -77,7 +78,9 @@ static void form_and_insert_tuple(BrinBuildState *state);
static void union_tuples(BrinDesc *bdesc, BrinMemTuple *a, static void union_tuples(BrinDesc *bdesc, BrinMemTuple *a,
BrinTuple *b); BrinTuple *b);
static void brin_vacuum_scan(Relation idxrel, BufferAccessStrategy strategy); static void brin_vacuum_scan(Relation idxrel, BufferAccessStrategy strategy);
static bool add_values_to_range(Relation idxRel, BrinDesc *bdesc,
BrinMemTuple *dtup, Datum *values, bool *nulls);
static bool check_null_keys(BrinValues *bval, ScanKey *nullkeys, int nnullkeys);
/* /*
* BRIN handler function: return IndexAmRoutine with access method parameters * BRIN handler function: return IndexAmRoutine with access method parameters
...@@ -179,7 +182,6 @@ brininsert(Relation idxRel, Datum *values, bool *nulls, ...@@ -179,7 +182,6 @@ brininsert(Relation idxRel, Datum *values, bool *nulls,
OffsetNumber off; OffsetNumber off;
BrinTuple *brtup; BrinTuple *brtup;
BrinMemTuple *dtup; BrinMemTuple *dtup;
int keyno;
CHECK_FOR_INTERRUPTS(); CHECK_FOR_INTERRUPTS();
...@@ -243,31 +245,7 @@ brininsert(Relation idxRel, Datum *values, bool *nulls, ...@@ -243,31 +245,7 @@ brininsert(Relation idxRel, Datum *values, bool *nulls,
dtup = brin_deform_tuple(bdesc, brtup, NULL); dtup = brin_deform_tuple(bdesc, brtup, NULL);
/* need_insert = add_values_to_range(idxRel, bdesc, dtup, values, nulls);
* Compare the key values of the new tuple to the stored index values;
* our deformed tuple will get updated if the new tuple doesn't fit
* the original range (note this means we can't break out of the loop
* early). Make a note of whether this happens, so that we know to
* insert the modified tuple later.
*/
for (keyno = 0; keyno < bdesc->bd_tupdesc->natts; keyno++)
{
Datum result;
BrinValues *bval;
FmgrInfo *addValue;
bval = &dtup->bt_columns[keyno];
addValue = index_getprocinfo(idxRel, keyno + 1,
BRIN_PROCNUM_ADDVALUE);
result = FunctionCall4Coll(addValue,
idxRel->rd_indcollation[keyno],
PointerGetDatum(bdesc),
PointerGetDatum(bval),
values[keyno],
nulls[keyno]);
/* if that returned true, we need to insert the updated tuple */
need_insert |= DatumGetBool(result);
}
if (!need_insert) if (!need_insert)
{ {
...@@ -390,8 +368,10 @@ bringetbitmap(IndexScanDesc scan, TIDBitmap *tbm) ...@@ -390,8 +368,10 @@ bringetbitmap(IndexScanDesc scan, TIDBitmap *tbm)
BrinMemTuple *dtup; BrinMemTuple *dtup;
BrinTuple *btup = NULL; BrinTuple *btup = NULL;
Size btupsz = 0; Size btupsz = 0;
ScanKey **keys; ScanKey **keys,
int *nkeys; **nullkeys;
int *nkeys,
*nnullkeys;
int keyno; int keyno;
opaque = (BrinOpaque *) scan->opaque; opaque = (BrinOpaque *) scan->opaque;
...@@ -420,13 +400,18 @@ bringetbitmap(IndexScanDesc scan, TIDBitmap *tbm) ...@@ -420,13 +400,18 @@ bringetbitmap(IndexScanDesc scan, TIDBitmap *tbm)
* keys, so we allocate space for all attributes. That may use more memory * keys, so we allocate space for all attributes. That may use more memory
* but it's probably cheaper than determining which attributes are used. * but it's probably cheaper than determining which attributes are used.
* *
* We keep null and regular keys separate, so that we can pass just the
* regular keys to the consistent function easily.
*
* XXX The widest index can have 32 attributes, so the amount of wasted * XXX The widest index can have 32 attributes, so the amount of wasted
* memory is negligible. We could invent a more compact approach (with * memory is negligible. We could invent a more compact approach (with
* just space for used attributes) but that would make the matching more * just space for used attributes) but that would make the matching more
* complex so it's not a good trade-off. * complex so it's not a good trade-off.
*/ */
keys = palloc0(sizeof(ScanKey *) * bdesc->bd_tupdesc->natts); keys = palloc0(sizeof(ScanKey *) * bdesc->bd_tupdesc->natts);
nullkeys = palloc0(sizeof(ScanKey *) * bdesc->bd_tupdesc->natts);
nkeys = palloc0(sizeof(int) * bdesc->bd_tupdesc->natts); nkeys = palloc0(sizeof(int) * bdesc->bd_tupdesc->natts);
nnullkeys = palloc0(sizeof(int) * bdesc->bd_tupdesc->natts);
/* Preprocess the scan keys - split them into per-attribute arrays. */ /* Preprocess the scan keys - split them into per-attribute arrays. */
for (keyno = 0; keyno < scan->numberOfKeys; keyno++) for (keyno = 0; keyno < scan->numberOfKeys; keyno++)
...@@ -444,23 +429,23 @@ bringetbitmap(IndexScanDesc scan, TIDBitmap *tbm) ...@@ -444,23 +429,23 @@ bringetbitmap(IndexScanDesc scan, TIDBitmap *tbm)
TupleDescAttr(bdesc->bd_tupdesc, TupleDescAttr(bdesc->bd_tupdesc,
keyattno - 1)->attcollation)); keyattno - 1)->attcollation));
/* First time we see this attribute, so init the array of keys. */ /*
if (!keys[keyattno - 1]) * First time we see this index attribute, so init as needed.
*
* This is a bit of an overkill - we don't know how many scan keys are
* there for this attribute, so we simply allocate the largest number
* possible (as if all keys were for this attribute). This may waste a
* bit of memory, but we only expect small number of scan keys in
* general, so this should be negligible, and repeated repalloc calls
* are not free either.
*/
if (consistentFn[keyattno - 1].fn_oid == InvalidOid)
{ {
FmgrInfo *tmp; FmgrInfo *tmp;
/* /* No key/null arrays for this attribute. */
* This is a bit of an overkill - we don't know how many scan keys Assert((keys[keyattno - 1] == NULL) && (nkeys[keyattno - 1] == 0));
* are there for this attribute, so we simply allocate the largest Assert((nullkeys[keyattno - 1] == NULL) && (nnullkeys[keyattno - 1] == 0));
* number possible (as if all keys were for this attribute). This
* may waste a bit of memory, but we only expect small number of
* scan keys in general, so this should be negligible, and
* repeated repalloc calls are not free either.
*/
keys[keyattno - 1] = palloc0(sizeof(ScanKey) * scan->numberOfKeys);
/* First time this column, so look up consistent function */
Assert(consistentFn[keyattno - 1].fn_oid == InvalidOid);
tmp = index_getprocinfo(idxRel, keyattno, tmp = index_getprocinfo(idxRel, keyattno,
BRIN_PROCNUM_CONSISTENT); BRIN_PROCNUM_CONSISTENT);
...@@ -468,10 +453,24 @@ bringetbitmap(IndexScanDesc scan, TIDBitmap *tbm) ...@@ -468,10 +453,24 @@ bringetbitmap(IndexScanDesc scan, TIDBitmap *tbm)
CurrentMemoryContext); CurrentMemoryContext);
} }
/* Add key to the per-attribute array. */ /* Add key to the proper per-attribute array. */
if (key->sk_flags & SK_ISNULL)
{
if (!nullkeys[keyattno - 1])
nullkeys[keyattno - 1] = palloc0(sizeof(ScanKey) * scan->numberOfKeys);
nullkeys[keyattno - 1][nnullkeys[keyattno - 1]] = key;
nnullkeys[keyattno - 1]++;
}
else
{
if (!keys[keyattno - 1])
keys[keyattno - 1] = palloc0(sizeof(ScanKey) * scan->numberOfKeys);
keys[keyattno - 1][nkeys[keyattno - 1]] = key; keys[keyattno - 1][nkeys[keyattno - 1]] = key;
nkeys[keyattno - 1]++; nkeys[keyattno - 1]++;
} }
}
/* allocate an initial in-memory tuple, out of the per-range memcxt */ /* allocate an initial in-memory tuple, out of the per-range memcxt */
dtup = brin_new_memtuple(bdesc); dtup = brin_new_memtuple(bdesc);
...@@ -549,15 +548,58 @@ bringetbitmap(IndexScanDesc scan, TIDBitmap *tbm) ...@@ -549,15 +548,58 @@ bringetbitmap(IndexScanDesc scan, TIDBitmap *tbm)
Datum add; Datum add;
Oid collation; Oid collation;
/* skip attributes without any scan keys */ /*
if (nkeys[attno - 1] == 0) * skip attributes without any scan keys (both regular and
* IS [NOT] NULL)
*/
if (nkeys[attno - 1] == 0 && nnullkeys[attno - 1] == 0)
continue; continue;
bval = &dtup->bt_columns[attno - 1]; bval = &dtup->bt_columns[attno - 1];
/*
* First check if there are any IS [NOT] NULL scan keys,
* and if we're violating them. In that case we can
* terminate early, without invoking the support function.
*
* As there may be more keys, we can only detemine
* mismatch within this loop.
*/
if (bdesc->bd_info[attno - 1]->oi_regular_nulls &&
!check_null_keys(bval, nullkeys[attno - 1],
nnullkeys[attno - 1]))
{
/*
* If any of the IS [NOT] NULL keys failed, the page
* range as a whole can't pass. So terminate the loop.
*/
addrange = false;
break;
}
/*
* So either there are no IS [NOT] NULL keys, or all
* passed. If there are no regular scan keys, we're done -
* the page range matches. If there are regular keys, but
* the page range is marked as 'all nulls' it can't
* possibly pass (we're assuming the operators are
* strict).
*/
/* No regular scan keys - page range as a whole passes. */
if (!nkeys[attno - 1])
continue;
Assert((nkeys[attno - 1] > 0) && Assert((nkeys[attno - 1] > 0) &&
(nkeys[attno - 1] <= scan->numberOfKeys)); (nkeys[attno - 1] <= scan->numberOfKeys));
/* If it is all nulls, it cannot possibly be consistent. */
if (bval->bv_allnulls)
{
addrange = false;
break;
}
/* /*
* Check whether the scan key is consistent with the page * Check whether the scan key is consistent with the page
* range values; if so, have the pages in the range added * range values; if so, have the pages in the range added
...@@ -663,7 +705,6 @@ brinbuildCallback(Relation index, ...@@ -663,7 +705,6 @@ brinbuildCallback(Relation index,
{ {
BrinBuildState *state = (BrinBuildState *) brstate; BrinBuildState *state = (BrinBuildState *) brstate;
BlockNumber thisblock; BlockNumber thisblock;
int i;
thisblock = ItemPointerGetBlockNumber(tid); thisblock = ItemPointerGetBlockNumber(tid);
...@@ -692,25 +733,8 @@ brinbuildCallback(Relation index, ...@@ -692,25 +733,8 @@ brinbuildCallback(Relation index,
} }
/* Accumulate the current tuple into the running state */ /* Accumulate the current tuple into the running state */
for (i = 0; i < state->bs_bdesc->bd_tupdesc->natts; i++) (void) add_values_to_range(index, state->bs_bdesc, state->bs_dtuple,
{ values, isnull);
FmgrInfo *addValue;
BrinValues *col;
Form_pg_attribute attr = TupleDescAttr(state->bs_bdesc->bd_tupdesc, i);
col = &state->bs_dtuple->bt_columns[i];
addValue = index_getprocinfo(index, i + 1,
BRIN_PROCNUM_ADDVALUE);
/*
* Update dtuple state, if and as necessary.
*/
FunctionCall4Coll(addValue,
attr->attcollation,
PointerGetDatum(state->bs_bdesc),
PointerGetDatum(col),
values[i], isnull[i]);
}
} }
/* /*
...@@ -1489,6 +1513,39 @@ union_tuples(BrinDesc *bdesc, BrinMemTuple *a, BrinTuple *b) ...@@ -1489,6 +1513,39 @@ union_tuples(BrinDesc *bdesc, BrinMemTuple *a, BrinTuple *b)
FmgrInfo *unionFn; FmgrInfo *unionFn;
BrinValues *col_a = &a->bt_columns[keyno]; BrinValues *col_a = &a->bt_columns[keyno];
BrinValues *col_b = &db->bt_columns[keyno]; BrinValues *col_b = &db->bt_columns[keyno];
BrinOpcInfo *opcinfo = bdesc->bd_info[keyno];
if (opcinfo->oi_regular_nulls)
{
/* Adjust "hasnulls". */
if (!col_a->bv_hasnulls && col_b->bv_hasnulls)
col_a->bv_hasnulls = true;
/* If there are no values in B, there's nothing left to do. */
if (col_b->bv_allnulls)
continue;
/*
* Adjust "allnulls". If A doesn't have values, just copy the
* values from B into A, and we're done. We cannot run the
* operators in this case, because values in A might contain
* garbage. Note we already established that B contains values.
*/
if (col_a->bv_allnulls)
{
int i;
col_a->bv_allnulls = false;
for (i = 0; i < opcinfo->oi_nstored; i++)
col_a->bv_values[i] =
datumCopy(col_b->bv_values[i],
opcinfo->oi_typcache[i]->typbyval,
opcinfo->oi_typcache[i]->typlen);
continue;
}
}
unionFn = index_getprocinfo(bdesc->bd_index, keyno + 1, unionFn = index_getprocinfo(bdesc->bd_index, keyno + 1,
BRIN_PROCNUM_UNION); BRIN_PROCNUM_UNION);
...@@ -1542,3 +1599,103 @@ brin_vacuum_scan(Relation idxrel, BufferAccessStrategy strategy) ...@@ -1542,3 +1599,103 @@ brin_vacuum_scan(Relation idxrel, BufferAccessStrategy strategy)
*/ */
FreeSpaceMapVacuum(idxrel); FreeSpaceMapVacuum(idxrel);
} }
static bool
add_values_to_range(Relation idxRel, BrinDesc *bdesc, BrinMemTuple *dtup,
Datum *values, bool *nulls)
{
int keyno;
bool modified = false;
/*
* Compare the key values of the new tuple to the stored index values; our
* deformed tuple will get updated if the new tuple doesn't fit the
* original range (note this means we can't break out of the loop early).
* Make a note of whether this happens, so that we know to insert the
* modified tuple later.
*/
for (keyno = 0; keyno < bdesc->bd_tupdesc->natts; keyno++)
{
Datum result;
BrinValues *bval;
FmgrInfo *addValue;
bval = &dtup->bt_columns[keyno];
if (bdesc->bd_info[keyno]->oi_regular_nulls && nulls[keyno])
{
/*
* If the new value is null, we record that we saw it if it's the
* first one; otherwise, there's nothing to do.
*/
if (!bval->bv_hasnulls)
{
bval->bv_hasnulls = true;
modified = true;
}
continue;
}
addValue = index_getprocinfo(idxRel, keyno + 1,
BRIN_PROCNUM_ADDVALUE);
result = FunctionCall4Coll(addValue,
idxRel->rd_indcollation[keyno],
PointerGetDatum(bdesc),
PointerGetDatum(bval),
values[keyno],
nulls[keyno]);
/* if that returned true, we need to insert the updated tuple */
modified |= DatumGetBool(result);
}
return modified;
}
static bool
check_null_keys(BrinValues *bval, ScanKey *nullkeys, int nnullkeys)
{
int keyno;
/*
* First check if there are any IS [NOT] NULL scan keys, and if we're
* violating them.
*/
for (keyno = 0; keyno < nnullkeys; keyno++)
{
ScanKey key = nullkeys[keyno];
Assert(key->sk_attno == bval->bv_attno);
/* Handle only IS NULL/IS NOT NULL tests */
if (!(key->sk_flags & SK_ISNULL))
continue;
if (key->sk_flags & SK_SEARCHNULL)
{
/* IS NULL scan key, but range has no NULLs */
if (!bval->bv_allnulls && !bval->bv_hasnulls)
return false;
}
else if (key->sk_flags & SK_SEARCHNOTNULL)
{
/*
* For IS NOT NULL, we can only skip ranges that are known to have
* only nulls.
*/
if (bval->bv_allnulls)
return false;
}
else
{
/*
* Neither IS NULL nor IS NOT NULL was used; assume all indexable
* operators are strict and thus return false with NULL value in
* the scan key.
*/
return false;
}
}
return true;
}
...@@ -110,6 +110,7 @@ brin_inclusion_opcinfo(PG_FUNCTION_ARGS) ...@@ -110,6 +110,7 @@ brin_inclusion_opcinfo(PG_FUNCTION_ARGS)
*/ */
result = palloc0(MAXALIGN(SizeofBrinOpcInfo(3)) + sizeof(InclusionOpaque)); result = palloc0(MAXALIGN(SizeofBrinOpcInfo(3)) + sizeof(InclusionOpaque));
result->oi_nstored = 3; result->oi_nstored = 3;
result->oi_regular_nulls = true;
result->oi_opaque = (InclusionOpaque *) result->oi_opaque = (InclusionOpaque *)
MAXALIGN((char *) result + SizeofBrinOpcInfo(3)); MAXALIGN((char *) result + SizeofBrinOpcInfo(3));
...@@ -141,7 +142,7 @@ brin_inclusion_add_value(PG_FUNCTION_ARGS) ...@@ -141,7 +142,7 @@ brin_inclusion_add_value(PG_FUNCTION_ARGS)
BrinDesc *bdesc = (BrinDesc *) PG_GETARG_POINTER(0); BrinDesc *bdesc = (BrinDesc *) PG_GETARG_POINTER(0);
BrinValues *column = (BrinValues *) PG_GETARG_POINTER(1); BrinValues *column = (BrinValues *) PG_GETARG_POINTER(1);
Datum newval = PG_GETARG_DATUM(2); Datum newval = PG_GETARG_DATUM(2);
bool isnull = PG_GETARG_BOOL(3); bool isnull PG_USED_FOR_ASSERTS_ONLY = PG_GETARG_BOOL(3);
Oid colloid = PG_GET_COLLATION(); Oid colloid = PG_GET_COLLATION();
FmgrInfo *finfo; FmgrInfo *finfo;
Datum result; Datum result;
...@@ -149,18 +150,7 @@ brin_inclusion_add_value(PG_FUNCTION_ARGS) ...@@ -149,18 +150,7 @@ brin_inclusion_add_value(PG_FUNCTION_ARGS)
AttrNumber attno; AttrNumber attno;
Form_pg_attribute attr; Form_pg_attribute attr;
/* Assert(!isnull);
* If the new value is null, we record that we saw it if it's the first
* one; otherwise, there's nothing to do.
*/
if (isnull)
{
if (column->bv_hasnulls)
PG_RETURN_BOOL(false);
column->bv_hasnulls = true;
PG_RETURN_BOOL(true);
}
attno = column->bv_attno; attno = column->bv_attno;
attr = TupleDescAttr(bdesc->bd_tupdesc, attno - 1); attr = TupleDescAttr(bdesc->bd_tupdesc, attno - 1);
...@@ -268,52 +258,9 @@ brin_inclusion_consistent(PG_FUNCTION_ARGS) ...@@ -268,52 +258,9 @@ brin_inclusion_consistent(PG_FUNCTION_ARGS)
int nkeys = PG_GETARG_INT32(3); int nkeys = PG_GETARG_INT32(3);
Oid colloid = PG_GET_COLLATION(); Oid colloid = PG_GET_COLLATION();
int keyno; int keyno;
bool has_regular_keys = false;
/* Handle IS NULL/IS NOT NULL tests */
for (keyno = 0; keyno < nkeys; keyno++)
{
ScanKey key = keys[keyno];
Assert(key->sk_attno == column->bv_attno);
/* Skip regular scan keys (and remember that we have some). */
if ((!key->sk_flags & SK_ISNULL))
{
has_regular_keys = true;
continue;
}
if (key->sk_flags & SK_SEARCHNULL)
{
if (column->bv_allnulls || column->bv_hasnulls)
continue; /* this key is fine, continue */
PG_RETURN_BOOL(false);
}
/*
* For IS NOT NULL, we can only skip ranges that are known to have
* only nulls.
*/
if (key->sk_flags & SK_SEARCHNOTNULL)
{
if (column->bv_allnulls)
PG_RETURN_BOOL(false);
continue;
}
/*
* Neither IS NULL nor IS NOT NULL was used; assume all indexable
* operators are strict and return false.
*/
PG_RETURN_BOOL(false);
}
/* If there are no regular keys, the page range is considered consistent. */ /* make sure we got some scan keys */
if (!has_regular_keys) Assert((nkeys > 0) && (keys != NULL));
PG_RETURN_BOOL(true);
/* /*
* If is all nulls, it cannot possibly be consistent (at this point we * If is all nulls, it cannot possibly be consistent (at this point we
...@@ -331,9 +278,8 @@ brin_inclusion_consistent(PG_FUNCTION_ARGS) ...@@ -331,9 +278,8 @@ brin_inclusion_consistent(PG_FUNCTION_ARGS)
{ {
ScanKey key = keys[keyno]; ScanKey key = keys[keyno];
/* Skip IS NULL/IS NOT NULL keys (already handled above). */ /* NULL keys are handled and filtered-out in bringetbitmap */
if (key->sk_flags & SK_ISNULL) Assert(!(key->sk_flags & SK_ISNULL));
continue;
/* /*
* When there are multiple scan keys, failure to meet the criteria for * When there are multiple scan keys, failure to meet the criteria for
...@@ -574,37 +520,11 @@ brin_inclusion_union(PG_FUNCTION_ARGS) ...@@ -574,37 +520,11 @@ brin_inclusion_union(PG_FUNCTION_ARGS)
Datum result; Datum result;
Assert(col_a->bv_attno == col_b->bv_attno); Assert(col_a->bv_attno == col_b->bv_attno);
Assert(!col_a->bv_allnulls && !col_b->bv_allnulls);
/* Adjust "hasnulls". */
if (!col_a->bv_hasnulls && col_b->bv_hasnulls)
col_a->bv_hasnulls = true;
/* If there are no values in B, there's nothing left to do. */
if (col_b->bv_allnulls)
PG_RETURN_VOID();
attno = col_a->bv_attno; attno = col_a->bv_attno;
attr = TupleDescAttr(bdesc->bd_tupdesc, attno - 1); attr = TupleDescAttr(bdesc->bd_tupdesc, attno - 1);
/*
* Adjust "allnulls". If A doesn't have values, just copy the values from
* B into A, and we're done. We cannot run the operators in this case,
* because values in A might contain garbage. Note we already established
* that B contains values.
*/
if (col_a->bv_allnulls)
{
col_a->bv_allnulls = false;
col_a->bv_values[INCLUSION_UNION] =
datumCopy(col_b->bv_values[INCLUSION_UNION],
attr->attbyval, attr->attlen);
col_a->bv_values[INCLUSION_UNMERGEABLE] =
col_b->bv_values[INCLUSION_UNMERGEABLE];
col_a->bv_values[INCLUSION_CONTAINS_EMPTY] =
col_b->bv_values[INCLUSION_CONTAINS_EMPTY];
PG_RETURN_VOID();
}
/* If B includes empty elements, mark A similarly, if needed. */ /* If B includes empty elements, mark A similarly, if needed. */
if (!DatumGetBool(col_a->bv_values[INCLUSION_CONTAINS_EMPTY]) && if (!DatumGetBool(col_a->bv_values[INCLUSION_CONTAINS_EMPTY]) &&
DatumGetBool(col_b->bv_values[INCLUSION_CONTAINS_EMPTY])) DatumGetBool(col_b->bv_values[INCLUSION_CONTAINS_EMPTY]))
......
...@@ -48,6 +48,7 @@ brin_minmax_opcinfo(PG_FUNCTION_ARGS) ...@@ -48,6 +48,7 @@ brin_minmax_opcinfo(PG_FUNCTION_ARGS)
result = palloc0(MAXALIGN(SizeofBrinOpcInfo(2)) + result = palloc0(MAXALIGN(SizeofBrinOpcInfo(2)) +
sizeof(MinmaxOpaque)); sizeof(MinmaxOpaque));
result->oi_nstored = 2; result->oi_nstored = 2;
result->oi_regular_nulls = true;
result->oi_opaque = (MinmaxOpaque *) result->oi_opaque = (MinmaxOpaque *)
MAXALIGN((char *) result + SizeofBrinOpcInfo(2)); MAXALIGN((char *) result + SizeofBrinOpcInfo(2));
result->oi_typcache[0] = result->oi_typcache[1] = result->oi_typcache[0] = result->oi_typcache[1] =
...@@ -69,7 +70,7 @@ brin_minmax_add_value(PG_FUNCTION_ARGS) ...@@ -69,7 +70,7 @@ brin_minmax_add_value(PG_FUNCTION_ARGS)
BrinDesc *bdesc = (BrinDesc *) PG_GETARG_POINTER(0); BrinDesc *bdesc = (BrinDesc *) PG_GETARG_POINTER(0);
BrinValues *column = (BrinValues *) PG_GETARG_POINTER(1); BrinValues *column = (BrinValues *) PG_GETARG_POINTER(1);
Datum newval = PG_GETARG_DATUM(2); Datum newval = PG_GETARG_DATUM(2);
bool isnull = PG_GETARG_DATUM(3); bool isnull PG_USED_FOR_ASSERTS_ONLY = PG_GETARG_DATUM(3);
Oid colloid = PG_GET_COLLATION(); Oid colloid = PG_GET_COLLATION();
FmgrInfo *cmpFn; FmgrInfo *cmpFn;
Datum compar; Datum compar;
...@@ -77,18 +78,7 @@ brin_minmax_add_value(PG_FUNCTION_ARGS) ...@@ -77,18 +78,7 @@ brin_minmax_add_value(PG_FUNCTION_ARGS)
Form_pg_attribute attr; Form_pg_attribute attr;
AttrNumber attno; AttrNumber attno;
/* Assert(!isnull);
* If the new value is null, we record that we saw it if it's the first
* one; otherwise, there's nothing to do.
*/
if (isnull)
{
if (column->bv_hasnulls)
PG_RETURN_BOOL(false);
column->bv_hasnulls = true;
PG_RETURN_BOOL(true);
}
attno = column->bv_attno; attno = column->bv_attno;
attr = TupleDescAttr(bdesc->bd_tupdesc, attno - 1); attr = TupleDescAttr(bdesc->bd_tupdesc, attno - 1);
...@@ -156,52 +146,9 @@ brin_minmax_consistent(PG_FUNCTION_ARGS) ...@@ -156,52 +146,9 @@ brin_minmax_consistent(PG_FUNCTION_ARGS)
int nkeys = PG_GETARG_INT32(3); int nkeys = PG_GETARG_INT32(3);
Oid colloid = PG_GET_COLLATION(); Oid colloid = PG_GET_COLLATION();
int keyno; int keyno;
bool has_regular_keys = false;
/* handle IS NULL/IS NOT NULL tests */
for (keyno = 0; keyno < nkeys; keyno++)
{
ScanKey key = keys[keyno];
Assert(key->sk_attno == column->bv_attno);
/* Skip regular scan keys (and remember that we have some). */ /* make sure we got some scan keys */
if ((!key->sk_flags & SK_ISNULL)) Assert((nkeys > 0) && (keys != NULL));
{
has_regular_keys = true;
continue;
}
if (key->sk_flags & SK_SEARCHNULL)
{
if (column->bv_allnulls || column->bv_hasnulls)
continue; /* this key is fine, continue */
PG_RETURN_BOOL(false);
}
/*
* For IS NOT NULL, we can only skip ranges that are known to have
* only nulls.
*/
if (key->sk_flags & SK_SEARCHNOTNULL)
{
if (column->bv_allnulls)
PG_RETURN_BOOL(false);
continue;
}
/*
* Neither IS NULL nor IS NOT NULL was used; assume all indexable
* operators are strict and return false.
*/
PG_RETURN_BOOL(false);
}
/* If there are no regular keys, the page range is considered consistent. */
if (!has_regular_keys)
PG_RETURN_BOOL(true);
/* /*
* If is all nulls, it cannot possibly be consistent (at this point we * If is all nulls, it cannot possibly be consistent (at this point we
...@@ -215,9 +162,8 @@ brin_minmax_consistent(PG_FUNCTION_ARGS) ...@@ -215,9 +162,8 @@ brin_minmax_consistent(PG_FUNCTION_ARGS)
{ {
ScanKey key = keys[keyno]; ScanKey key = keys[keyno];
/* ignore IS NULL/IS NOT NULL tests handled above */ /* NULL keys are handled and filtered-out in bringetbitmap */
if (key->sk_flags & SK_ISNULL) Assert(!(key->sk_flags & SK_ISNULL));
continue;
/* /*
* When there are multiple scan keys, failure to meet the criteria for * When there are multiple scan keys, failure to meet the criteria for
...@@ -307,34 +253,11 @@ brin_minmax_union(PG_FUNCTION_ARGS) ...@@ -307,34 +253,11 @@ brin_minmax_union(PG_FUNCTION_ARGS)
bool needsadj; bool needsadj;
Assert(col_a->bv_attno == col_b->bv_attno); Assert(col_a->bv_attno == col_b->bv_attno);
Assert(!col_a->bv_allnulls && !col_b->bv_allnulls);
/* Adjust "hasnulls" */
if (!col_a->bv_hasnulls && col_b->bv_hasnulls)
col_a->bv_hasnulls = true;
/* If there are no values in B, there's nothing left to do */
if (col_b->bv_allnulls)
PG_RETURN_VOID();
attno = col_a->bv_attno; attno = col_a->bv_attno;
attr = TupleDescAttr(bdesc->bd_tupdesc, attno - 1); attr = TupleDescAttr(bdesc->bd_tupdesc, attno - 1);
/*
* Adjust "allnulls". If A doesn't have values, just copy the values from
* B into A, and we're done. We cannot run the operators in this case,
* because values in A might contain garbage. Note we already established
* that B contains values.
*/
if (col_a->bv_allnulls)
{
col_a->bv_allnulls = false;
col_a->bv_values[0] = datumCopy(col_b->bv_values[0],
attr->attbyval, attr->attlen);
col_a->bv_values[1] = datumCopy(col_b->bv_values[1],
attr->attbyval, attr->attlen);
PG_RETURN_VOID();
}
/* Adjust minimum, if B's min is less than A's min */ /* Adjust minimum, if B's min is less than A's min */
finfo = minmax_get_strategy_procinfo(bdesc, attno, attr->atttypid, finfo = minmax_get_strategy_procinfo(bdesc, attno, attr->atttypid,
BTLessStrategyNumber); BTLessStrategyNumber);
......
...@@ -27,6 +27,9 @@ typedef struct BrinOpcInfo ...@@ -27,6 +27,9 @@ typedef struct BrinOpcInfo
/* Number of columns stored in an index column of this opclass */ /* Number of columns stored in an index column of this opclass */
uint16 oi_nstored; uint16 oi_nstored;
/* Regular processing of NULLs in BrinValues? */
bool oi_regular_nulls;
/* Opaque pointer for the opclass' private use */ /* Opaque pointer for the opclass' private use */
void *oi_opaque; void *oi_opaque;
......
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