mirror of
https://github.com/postgres/postgres.git
synced 2025-11-10 17:42:29 +03:00
Reuse abbreviated keys in ordered [set] aggregates.
When processing ordered aggregates following a sort that could make use of the abbreviated key optimization, only call the equality operator to compare successive pairs of tuples when their abbreviated keys were not equal. Peter Geoghegan, reviewd by Andreas Karlsson and by me.
This commit is contained in:
@@ -1280,7 +1280,8 @@ tuplesort_putindextuplevalues(Tuplesortstate *state, Relation rel,
|
||||
* converter it won't expect NULL values, and cost model is not
|
||||
* required to account for NULL, so in that case we avoid calling
|
||||
* converter and just set datum1 to zeroed representation (to be
|
||||
* consistent).
|
||||
* consistent, and to support cheap inequality tests for NULL
|
||||
* abbreviated keys).
|
||||
*/
|
||||
stup.datum1 = original;
|
||||
}
|
||||
@@ -1349,7 +1350,11 @@ tuplesort_putdatum(Tuplesortstate *state, Datum val, bool isNull)
|
||||
|
||||
if (isNull || state->datumTypeByVal)
|
||||
{
|
||||
stup.datum1 = val;
|
||||
/*
|
||||
* Set datum1 to zeroed representation for NULLs (to be consistent, and
|
||||
* to support cheap inequality tests for NULL abbreviated keys).
|
||||
*/
|
||||
stup.datum1 = !isNull ? val : (Datum) 0;
|
||||
stup.isnull1 = isNull;
|
||||
stup.tuple = NULL; /* no separate storage */
|
||||
}
|
||||
@@ -1866,10 +1871,17 @@ tuplesort_gettuple_common(Tuplesortstate *state, bool forward,
|
||||
* Fetch the next tuple in either forward or back direction.
|
||||
* If successful, put tuple in slot and return TRUE; else, clear the slot
|
||||
* and return FALSE.
|
||||
*
|
||||
* Caller may optionally be passed back abbreviated value (on TRUE return
|
||||
* value) when abbreviation was used, which can be used to cheaply avoid
|
||||
* equality checks that might otherwise be required. Caller can safely make a
|
||||
* determination of "non-equal tuple" based on simple binary inequality. A
|
||||
* NULL value in leading attribute will set abbreviated value to zeroed
|
||||
* representation, which caller may rely on in abbreviated inequality check.
|
||||
*/
|
||||
bool
|
||||
tuplesort_gettupleslot(Tuplesortstate *state, bool forward,
|
||||
TupleTableSlot *slot)
|
||||
TupleTableSlot *slot, Datum *abbrev)
|
||||
{
|
||||
MemoryContext oldcontext = MemoryContextSwitchTo(state->sortcontext);
|
||||
SortTuple stup;
|
||||
@@ -1882,6 +1894,10 @@ tuplesort_gettupleslot(Tuplesortstate *state, bool forward,
|
||||
|
||||
if (stup.tuple)
|
||||
{
|
||||
/* Record abbreviated key for caller */
|
||||
if (state->sortKeys->abbrev_converter && abbrev)
|
||||
*abbrev = stup.datum1;
|
||||
|
||||
ExecStoreMinimalTuple((MinimalTuple) stup.tuple, slot, should_free);
|
||||
return true;
|
||||
}
|
||||
@@ -1937,10 +1953,17 @@ tuplesort_getindextuple(Tuplesortstate *state, bool forward,
|
||||
*
|
||||
* If the Datum is pass-by-ref type, the returned value is freshly palloc'd
|
||||
* and is now owned by the caller.
|
||||
*
|
||||
* Caller may optionally be passed back abbreviated value (on TRUE return
|
||||
* value) when abbreviation was used, which can be used to cheaply avoid
|
||||
* equality checks that might otherwise be required. Caller can safely make a
|
||||
* determination of "non-equal tuple" based on simple binary inequality. A
|
||||
* NULL value will have a zeroed abbreviated value representation, which caller
|
||||
* may rely on in abbreviated inequality check.
|
||||
*/
|
||||
bool
|
||||
tuplesort_getdatum(Tuplesortstate *state, bool forward,
|
||||
Datum *val, bool *isNull)
|
||||
Datum *val, bool *isNull, Datum *abbrev)
|
||||
{
|
||||
MemoryContext oldcontext = MemoryContextSwitchTo(state->sortcontext);
|
||||
SortTuple stup;
|
||||
@@ -1952,6 +1975,10 @@ tuplesort_getdatum(Tuplesortstate *state, bool forward,
|
||||
return false;
|
||||
}
|
||||
|
||||
/* Record abbreviated key for caller */
|
||||
if (state->sortKeys->abbrev_converter && abbrev)
|
||||
*abbrev = stup.datum1;
|
||||
|
||||
if (stup.isnull1 || state->datumTypeByVal)
|
||||
{
|
||||
*val = stup.datum1;
|
||||
@@ -2232,21 +2259,6 @@ mergeruns(Tuplesortstate *state)
|
||||
Assert(state->status == TSS_BUILDRUNS);
|
||||
Assert(state->memtupcount == 0);
|
||||
|
||||
/*
|
||||
* If we produced only one initial run (quite likely if the total data
|
||||
* volume is between 1X and 2X workMem), we can just use that tape as the
|
||||
* finished output, rather than doing a useless merge. (This obvious
|
||||
* optimization is not in Knuth's algorithm.)
|
||||
*/
|
||||
if (state->currentRun == 1)
|
||||
{
|
||||
state->result_tape = state->tp_tapenum[state->destTape];
|
||||
/* must freeze and rewind the finished output tape */
|
||||
LogicalTapeFreeze(state->tapeset, state->result_tape);
|
||||
state->status = TSS_SORTEDONTAPE;
|
||||
return;
|
||||
}
|
||||
|
||||
if (state->sortKeys != NULL && state->sortKeys->abbrev_converter != NULL)
|
||||
{
|
||||
/*
|
||||
@@ -2263,6 +2275,21 @@ mergeruns(Tuplesortstate *state)
|
||||
state->sortKeys->abbrev_full_comparator = NULL;
|
||||
}
|
||||
|
||||
/*
|
||||
* If we produced only one initial run (quite likely if the total data
|
||||
* volume is between 1X and 2X workMem), we can just use that tape as the
|
||||
* finished output, rather than doing a useless merge. (This obvious
|
||||
* optimization is not in Knuth's algorithm.)
|
||||
*/
|
||||
if (state->currentRun == 1)
|
||||
{
|
||||
state->result_tape = state->tp_tapenum[state->destTape];
|
||||
/* must freeze and rewind the finished output tape */
|
||||
LogicalTapeFreeze(state->tapeset, state->result_tape);
|
||||
state->status = TSS_SORTEDONTAPE;
|
||||
return;
|
||||
}
|
||||
|
||||
/* End of step D2: rewind all output tapes to prepare for merging */
|
||||
for (tapenum = 0; tapenum < state->tapeRange; tapenum++)
|
||||
LogicalTapeRewind(state->tapeset, tapenum, false);
|
||||
@@ -3164,7 +3191,8 @@ copytup_heap(Tuplesortstate *state, SortTuple *stup, void *tup)
|
||||
* converter it won't expect NULL values, and cost model is not
|
||||
* required to account for NULL, so in that case we avoid calling
|
||||
* converter and just set datum1 to zeroed representation (to be
|
||||
* consistent).
|
||||
* consistent, and to support cheap inequality tests for NULL
|
||||
* abbreviated keys).
|
||||
*/
|
||||
stup->datum1 = original;
|
||||
}
|
||||
@@ -3406,7 +3434,8 @@ copytup_cluster(Tuplesortstate *state, SortTuple *stup, void *tup)
|
||||
* converter it won't expect NULL values, and cost model is not
|
||||
* required to account for NULL, so in that case we avoid calling
|
||||
* converter and just set datum1 to zeroed representation (to be
|
||||
* consistent).
|
||||
* consistent, and to support cheap inequality tests for NULL
|
||||
* abbreviated keys).
|
||||
*/
|
||||
stup->datum1 = original;
|
||||
}
|
||||
@@ -3710,7 +3739,8 @@ copytup_index(Tuplesortstate *state, SortTuple *stup, void *tup)
|
||||
* converter it won't expect NULL values, and cost model is not
|
||||
* required to account for NULL, so in that case we avoid calling
|
||||
* converter and just set datum1 to zeroed representation (to be
|
||||
* consistent).
|
||||
* consistent, and to support cheap inequality tests for NULL
|
||||
* abbreviated keys).
|
||||
*/
|
||||
stup->datum1 = original;
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user