PostgreSQL Source Code  git master
execGrouping.c
Go to the documentation of this file.
1 /*-------------------------------------------------------------------------
2  *
3  * execGrouping.c
4  * executor utility routines for grouping, hashing, and aggregation
5  *
6  * Portions Copyright (c) 1996-2024, PostgreSQL Global Development Group
7  * Portions Copyright (c) 1994, Regents of the University of California
8  *
9  *
10  * IDENTIFICATION
11  * src/backend/executor/execGrouping.c
12  *
13  *-------------------------------------------------------------------------
14  */
15 #include "postgres.h"
16 
17 #include "access/parallel.h"
18 #include "common/hashfn.h"
19 #include "executor/executor.h"
20 #include "miscadmin.h"
21 #include "utils/lsyscache.h"
22 
23 static int TupleHashTableMatch(struct tuplehash_hash *tb, const MinimalTuple tuple1, const MinimalTuple tuple2);
24 static inline uint32 TupleHashTableHash_internal(struct tuplehash_hash *tb,
25  const MinimalTuple tuple);
27  TupleTableSlot *slot,
28  bool *isnew, uint32 hash);
29 
30 /*
31  * Define parameters for tuple hash table code generation. The interface is
32  * *also* declared in execnodes.h (to generate the types, which are externally
33  * visible).
34  */
35 #define SH_PREFIX tuplehash
36 #define SH_ELEMENT_TYPE TupleHashEntryData
37 #define SH_KEY_TYPE MinimalTuple
38 #define SH_KEY firstTuple
39 #define SH_HASH_KEY(tb, key) TupleHashTableHash_internal(tb, key)
40 #define SH_EQUAL(tb, a, b) TupleHashTableMatch(tb, a, b) == 0
41 #define SH_SCOPE extern
42 #define SH_STORE_HASH
43 #define SH_GET_HASH(tb, a) a->hash
44 #define SH_DEFINE
45 #include "lib/simplehash.h"
46 
47 
48 /*****************************************************************************
49  * Utility routines for grouping tuples together
50  *****************************************************************************/
51 
52 /*
53  * execTuplesMatchPrepare
54  * Build expression that can be evaluated using ExecQual(), returning
55  * whether an ExprContext's inner/outer tuples are NOT DISTINCT
56  */
57 ExprState *
59  int numCols,
60  const AttrNumber *keyColIdx,
61  const Oid *eqOperators,
62  const Oid *collations,
63  PlanState *parent)
64 {
65  Oid *eqFunctions = (Oid *) palloc(numCols * sizeof(Oid));
66  int i;
67  ExprState *expr;
68 
69  if (numCols == 0)
70  return NULL;
71 
72  /* lookup equality functions */
73  for (i = 0; i < numCols; i++)
74  eqFunctions[i] = get_opcode(eqOperators[i]);
75 
76  /* build actual expression */
77  expr = ExecBuildGroupingEqual(desc, desc, NULL, NULL,
78  numCols, keyColIdx, eqFunctions, collations,
79  parent);
80 
81  return expr;
82 }
83 
84 /*
85  * execTuplesHashPrepare
86  * Look up the equality and hashing functions needed for a TupleHashTable.
87  *
88  * This is similar to execTuplesMatchPrepare, but we also need to find the
89  * hash functions associated with the equality operators. *eqFunctions and
90  * *hashFunctions receive the palloc'd result arrays.
91  *
92  * Note: we expect that the given operators are not cross-type comparisons.
93  */
94 void
96  const Oid *eqOperators,
97  Oid **eqFuncOids,
98  FmgrInfo **hashFunctions)
99 {
100  int i;
101 
102  *eqFuncOids = (Oid *) palloc(numCols * sizeof(Oid));
103  *hashFunctions = (FmgrInfo *) palloc(numCols * sizeof(FmgrInfo));
104 
105  for (i = 0; i < numCols; i++)
106  {
107  Oid eq_opr = eqOperators[i];
108  Oid eq_function;
109  Oid left_hash_function;
110  Oid right_hash_function;
111 
112  eq_function = get_opcode(eq_opr);
113  if (!get_op_hash_functions(eq_opr,
114  &left_hash_function, &right_hash_function))
115  elog(ERROR, "could not find hash function for hash operator %u",
116  eq_opr);
117  /* We're not supporting cross-type cases here */
118  Assert(left_hash_function == right_hash_function);
119  (*eqFuncOids)[i] = eq_function;
120  fmgr_info(right_hash_function, &(*hashFunctions)[i]);
121  }
122 }
123 
124 
125 /*****************************************************************************
126  * Utility routines for all-in-memory hash tables
127  *
128  * These routines build hash tables for grouping tuples together (eg, for
129  * hash aggregation). There is one entry for each not-distinct set of tuples
130  * presented.
131  *****************************************************************************/
132 
133 /*
134  * Construct an empty TupleHashTable
135  *
136  * numCols, keyColIdx: identify the tuple fields to use as lookup key
137  * eqfunctions: equality comparison functions to use
138  * hashfunctions: datatype-specific hashing functions to use
139  * nbuckets: initial estimate of hashtable size
140  * additionalsize: size of data stored in ->additional
141  * metacxt: memory context for long-lived allocation, but not per-entry data
142  * tablecxt: memory context in which to store table entries
143  * tempcxt: short-lived context for evaluation hash and comparison functions
144  *
145  * The function arrays may be made with execTuplesHashPrepare(). Note they
146  * are not cross-type functions, but expect to see the table datatype(s)
147  * on both sides.
148  *
149  * Note that keyColIdx, eqfunctions, and hashfunctions must be allocated in
150  * storage that will live as long as the hashtable does.
151  */
154  TupleDesc inputDesc,
155  int numCols, AttrNumber *keyColIdx,
156  const Oid *eqfuncoids,
157  FmgrInfo *hashfunctions,
158  Oid *collations,
159  long nbuckets, Size additionalsize,
160  MemoryContext metacxt,
161  MemoryContext tablecxt,
162  MemoryContext tempcxt,
163  bool use_variable_hash_iv)
164 {
165  TupleHashTable hashtable;
166  Size entrysize = sizeof(TupleHashEntryData) + additionalsize;
167  Size hash_mem_limit;
168  MemoryContext oldcontext;
169  bool allow_jit;
170 
171  Assert(nbuckets > 0);
172 
173  /* Limit initial table size request to not more than hash_mem */
174  hash_mem_limit = get_hash_memory_limit() / entrysize;
175  if (nbuckets > hash_mem_limit)
176  nbuckets = hash_mem_limit;
177 
178  oldcontext = MemoryContextSwitchTo(metacxt);
179 
180  hashtable = (TupleHashTable) palloc(sizeof(TupleHashTableData));
181 
182  hashtable->numCols = numCols;
183  hashtable->keyColIdx = keyColIdx;
184  hashtable->tab_hash_funcs = hashfunctions;
185  hashtable->tab_collations = collations;
186  hashtable->tablecxt = tablecxt;
187  hashtable->tempcxt = tempcxt;
188  hashtable->entrysize = entrysize;
189  hashtable->tableslot = NULL; /* will be made on first lookup */
190  hashtable->inputslot = NULL;
191  hashtable->in_hash_funcs = NULL;
192  hashtable->cur_eq_func = NULL;
193 
194  /*
195  * If parallelism is in use, even if the leader backend is performing the
196  * scan itself, we don't want to create the hashtable exactly the same way
197  * in all workers. As hashtables are iterated over in keyspace-order,
198  * doing so in all processes in the same way is likely to lead to
199  * "unbalanced" hashtables when the table size initially is
200  * underestimated.
201  */
202  if (use_variable_hash_iv)
204  else
205  hashtable->hash_iv = 0;
206 
207  hashtable->hashtab = tuplehash_create(metacxt, nbuckets, hashtable);
208 
209  /*
210  * We copy the input tuple descriptor just for safety --- we assume all
211  * input tuples will have equivalent descriptors.
212  */
215 
216  /*
217  * If the old reset interface is used (i.e. BuildTupleHashTable, rather
218  * than BuildTupleHashTableExt), allowing JIT would lead to the generated
219  * functions to a) live longer than the query b) be re-generated each time
220  * the table is being reset. Therefore prevent JIT from being used in that
221  * case, by not providing a parent node (which prevents accessing the
222  * JitContext in the EState).
223  */
224  allow_jit = metacxt != tablecxt;
225 
226  /* build comparator for all columns */
227  /* XXX: should we support non-minimal tuples for the inputslot? */
228  hashtable->tab_eq_func = ExecBuildGroupingEqual(inputDesc, inputDesc,
230  numCols,
231  keyColIdx, eqfuncoids, collations,
232  allow_jit ? parent : NULL);
233 
234  /*
235  * While not pretty, it's ok to not shut down this context, but instead
236  * rely on the containing memory context being reset, as
237  * ExecBuildGroupingEqual() only builds a very simple expression calling
238  * functions (i.e. nothing that'd employ RegisterExprContextCallback()).
239  */
241 
242  MemoryContextSwitchTo(oldcontext);
243 
244  return hashtable;
245 }
246 
247 /*
248  * BuildTupleHashTable is a backwards-compatibility wrapper for
249  * BuildTupleHashTableExt(), that allocates the hashtable's metadata in
250  * tablecxt. Note that hashtables created this way cannot be reset leak-free
251  * with ResetTupleHashTable().
252  */
255  TupleDesc inputDesc,
256  int numCols, AttrNumber *keyColIdx,
257  const Oid *eqfuncoids,
258  FmgrInfo *hashfunctions,
259  Oid *collations,
260  long nbuckets, Size additionalsize,
261  MemoryContext tablecxt,
262  MemoryContext tempcxt,
263  bool use_variable_hash_iv)
264 {
265  return BuildTupleHashTableExt(parent,
266  inputDesc,
267  numCols, keyColIdx,
268  eqfuncoids,
269  hashfunctions,
270  collations,
271  nbuckets, additionalsize,
272  tablecxt,
273  tablecxt,
274  tempcxt,
275  use_variable_hash_iv);
276 }
277 
278 /*
279  * Reset contents of the hashtable to be empty, preserving all the non-content
280  * state. Note that the tablecxt passed to BuildTupleHashTableExt() should
281  * also be reset, otherwise there will be leaks.
282  */
283 void
285 {
286  tuplehash_reset(hashtable->hashtab);
287 }
288 
289 /*
290  * Find or create a hashtable entry for the tuple group containing the
291  * given tuple. The tuple must be the same type as the hashtable entries.
292  *
293  * If isnew is NULL, we do not create new entries; we return NULL if no
294  * match is found.
295  *
296  * If hash is not NULL, we set it to the calculated hash value. This allows
297  * callers access to the hash value even if no entry is returned.
298  *
299  * If isnew isn't NULL, then a new entry is created if no existing entry
300  * matches. On return, *isnew is true if the entry is newly created,
301  * false if it existed already. ->additional_data in the new entry has
302  * been zeroed.
303  */
306  bool *isnew, uint32 *hash)
307 {
308  TupleHashEntry entry;
309  MemoryContext oldContext;
310  uint32 local_hash;
311 
312  /* Need to run the hash functions in short-lived context */
313  oldContext = MemoryContextSwitchTo(hashtable->tempcxt);
314 
315  /* set up data needed by hash and match functions */
316  hashtable->inputslot = slot;
317  hashtable->in_hash_funcs = hashtable->tab_hash_funcs;
318  hashtable->cur_eq_func = hashtable->tab_eq_func;
319 
320  local_hash = TupleHashTableHash_internal(hashtable->hashtab, NULL);
321  entry = LookupTupleHashEntry_internal(hashtable, slot, isnew, local_hash);
322 
323  if (hash != NULL)
324  *hash = local_hash;
325 
326  Assert(entry == NULL || entry->hash == local_hash);
327 
328  MemoryContextSwitchTo(oldContext);
329 
330  return entry;
331 }
332 
333 /*
334  * Compute the hash value for a tuple
335  */
336 uint32
338 {
339  MemoryContext oldContext;
340  uint32 hash;
341 
342  hashtable->inputslot = slot;
343  hashtable->in_hash_funcs = hashtable->tab_hash_funcs;
344 
345  /* Need to run the hash functions in short-lived context */
346  oldContext = MemoryContextSwitchTo(hashtable->tempcxt);
347 
348  hash = TupleHashTableHash_internal(hashtable->hashtab, NULL);
349 
350  MemoryContextSwitchTo(oldContext);
351 
352  return hash;
353 }
354 
355 /*
356  * A variant of LookupTupleHashEntry for callers that have already computed
357  * the hash value.
358  */
361  bool *isnew, uint32 hash)
362 {
363  TupleHashEntry entry;
364  MemoryContext oldContext;
365 
366  /* Need to run the hash functions in short-lived context */
367  oldContext = MemoryContextSwitchTo(hashtable->tempcxt);
368 
369  /* set up data needed by hash and match functions */
370  hashtable->inputslot = slot;
371  hashtable->in_hash_funcs = hashtable->tab_hash_funcs;
372  hashtable->cur_eq_func = hashtable->tab_eq_func;
373 
374  entry = LookupTupleHashEntry_internal(hashtable, slot, isnew, hash);
375  Assert(entry == NULL || entry->hash == hash);
376 
377  MemoryContextSwitchTo(oldContext);
378 
379  return entry;
380 }
381 
382 /*
383  * Search for a hashtable entry matching the given tuple. No entry is
384  * created if there's not a match. This is similar to the non-creating
385  * case of LookupTupleHashEntry, except that it supports cross-type
386  * comparisons, in which the given tuple is not of the same type as the
387  * table entries. The caller must provide the hash functions to use for
388  * the input tuple, as well as the equality functions, since these may be
389  * different from the table's internal functions.
390  */
393  ExprState *eqcomp,
394  FmgrInfo *hashfunctions)
395 {
396  TupleHashEntry entry;
397  MemoryContext oldContext;
399 
400  /* Need to run the hash functions in short-lived context */
401  oldContext = MemoryContextSwitchTo(hashtable->tempcxt);
402 
403  /* Set up data needed by hash and match functions */
404  hashtable->inputslot = slot;
405  hashtable->in_hash_funcs = hashfunctions;
406  hashtable->cur_eq_func = eqcomp;
407 
408  /* Search the hash table */
409  key = NULL; /* flag to reference inputslot */
410  entry = tuplehash_lookup(hashtable->hashtab, key);
411  MemoryContextSwitchTo(oldContext);
412 
413  return entry;
414 }
415 
416 /*
417  * If tuple is NULL, use the input slot instead. This convention avoids the
418  * need to materialize virtual input tuples unless they actually need to get
419  * copied into the table.
420  *
421  * Also, the caller must select an appropriate memory context for running
422  * the hash functions. (dynahash.c doesn't change CurrentMemoryContext.)
423  */
424 static uint32
425 TupleHashTableHash_internal(struct tuplehash_hash *tb,
426  const MinimalTuple tuple)
427 {
428  TupleHashTable hashtable = (TupleHashTable) tb->private_data;
429  int numCols = hashtable->numCols;
430  AttrNumber *keyColIdx = hashtable->keyColIdx;
431  uint32 hashkey = hashtable->hash_iv;
432  TupleTableSlot *slot;
433  FmgrInfo *hashfunctions;
434  int i;
435 
436  if (tuple == NULL)
437  {
438  /* Process the current input tuple for the table */
439  slot = hashtable->inputslot;
440  hashfunctions = hashtable->in_hash_funcs;
441  }
442  else
443  {
444  /*
445  * Process a tuple already stored in the table.
446  *
447  * (this case never actually occurs due to the way simplehash.h is
448  * used, as the hash-value is stored in the entries)
449  */
450  slot = hashtable->tableslot;
451  ExecStoreMinimalTuple(tuple, slot, false);
452  hashfunctions = hashtable->tab_hash_funcs;
453  }
454 
455  for (i = 0; i < numCols; i++)
456  {
457  AttrNumber att = keyColIdx[i];
458  Datum attr;
459  bool isNull;
460 
461  /* combine successive hashkeys by rotating */
462  hashkey = pg_rotate_left32(hashkey, 1);
463 
464  attr = slot_getattr(slot, att, &isNull);
465 
466  if (!isNull) /* treat nulls as having hash key 0 */
467  {
468  uint32 hkey;
469 
470  hkey = DatumGetUInt32(FunctionCall1Coll(&hashfunctions[i],
471  hashtable->tab_collations[i],
472  attr));
473  hashkey ^= hkey;
474  }
475  }
476 
477  /*
478  * The way hashes are combined above, among each other and with the IV,
479  * doesn't lead to good bit perturbation. As the IV's goal is to lead to
480  * achieve that, perform a round of hashing of the combined hash -
481  * resulting in near perfect perturbation.
482  */
483  return murmurhash32(hashkey);
484 }
485 
486 /*
487  * Does the work of LookupTupleHashEntry and LookupTupleHashEntryHash. Useful
488  * so that we can avoid switching the memory context multiple times for
489  * LookupTupleHashEntry.
490  *
491  * NB: This function may or may not change the memory context. Caller is
492  * expected to change it back.
493  */
494 static inline TupleHashEntry
496  bool *isnew, uint32 hash)
497 {
498  TupleHashEntryData *entry;
499  bool found;
501 
502  key = NULL; /* flag to reference inputslot */
503 
504  if (isnew)
505  {
506  entry = tuplehash_insert_hash(hashtable->hashtab, key, hash, &found);
507 
508  if (found)
509  {
510  /* found pre-existing entry */
511  *isnew = false;
512  }
513  else
514  {
515  /* created new entry */
516  *isnew = true;
517  /* zero caller data */
518  entry->additional = NULL;
519  MemoryContextSwitchTo(hashtable->tablecxt);
520  /* Copy the first tuple into the table context */
521  entry->firstTuple = ExecCopySlotMinimalTuple(slot);
522  }
523  }
524  else
525  {
526  entry = tuplehash_lookup_hash(hashtable->hashtab, key, hash);
527  }
528 
529  return entry;
530 }
531 
532 /*
533  * See whether two tuples (presumably of the same hash value) match
534  */
535 static int
536 TupleHashTableMatch(struct tuplehash_hash *tb, const MinimalTuple tuple1, const MinimalTuple tuple2)
537 {
538  TupleTableSlot *slot1;
539  TupleTableSlot *slot2;
540  TupleHashTable hashtable = (TupleHashTable) tb->private_data;
541  ExprContext *econtext = hashtable->exprcontext;
542 
543  /*
544  * We assume that simplehash.h will only ever call us with the first
545  * argument being an actual table entry, and the second argument being
546  * LookupTupleHashEntry's dummy TupleHashEntryData. The other direction
547  * could be supported too, but is not currently required.
548  */
549  Assert(tuple1 != NULL);
550  slot1 = hashtable->tableslot;
551  ExecStoreMinimalTuple(tuple1, slot1, false);
552  Assert(tuple2 == NULL);
553  slot2 = hashtable->inputslot;
554 
555  /* For crosstype comparisons, the inputslot must be first */
556  econtext->ecxt_innertuple = slot2;
557  econtext->ecxt_outertuple = slot1;
558  return !ExecQualAndReset(hashtable->cur_eq_func, econtext);
559 }
int16 AttrNumber
Definition: attnum.h:21
int ParallelWorkerNumber
Definition: parallel.c:112
unsigned int uint32
Definition: c.h:506
#define Assert(condition)
Definition: c.h:858
size_t Size
Definition: c.h:605
#define ERROR
Definition: elog.h:39
#define elog(elevel,...)
Definition: elog.h:225
ExprState * ExecBuildGroupingEqual(TupleDesc ldesc, TupleDesc rdesc, const TupleTableSlotOps *lops, const TupleTableSlotOps *rops, int numCols, const AttrNumber *keyColIdx, const Oid *eqfunctions, const Oid *collations, PlanState *parent)
Definition: execExpr.c:4125
void execTuplesHashPrepare(int numCols, const Oid *eqOperators, Oid **eqFuncOids, FmgrInfo **hashFunctions)
Definition: execGrouping.c:95
TupleHashTable BuildTupleHashTable(PlanState *parent, TupleDesc inputDesc, int numCols, AttrNumber *keyColIdx, const Oid *eqfuncoids, FmgrInfo *hashfunctions, Oid *collations, long nbuckets, Size additionalsize, MemoryContext tablecxt, MemoryContext tempcxt, bool use_variable_hash_iv)
Definition: execGrouping.c:254
TupleHashEntry LookupTupleHashEntryHash(TupleHashTable hashtable, TupleTableSlot *slot, bool *isnew, uint32 hash)
Definition: execGrouping.c:360
TupleHashEntry LookupTupleHashEntry(TupleHashTable hashtable, TupleTableSlot *slot, bool *isnew, uint32 *hash)
Definition: execGrouping.c:305
static uint32 TupleHashTableHash_internal(struct tuplehash_hash *tb, const MinimalTuple tuple)
Definition: execGrouping.c:425
uint32 TupleHashTableHash(TupleHashTable hashtable, TupleTableSlot *slot)
Definition: execGrouping.c:337
static int TupleHashTableMatch(struct tuplehash_hash *tb, const MinimalTuple tuple1, const MinimalTuple tuple2)
Definition: execGrouping.c:536
TupleHashEntry FindTupleHashEntry(TupleHashTable hashtable, TupleTableSlot *slot, ExprState *eqcomp, FmgrInfo *hashfunctions)
Definition: execGrouping.c:392
TupleHashTable BuildTupleHashTableExt(PlanState *parent, TupleDesc inputDesc, int numCols, AttrNumber *keyColIdx, const Oid *eqfuncoids, FmgrInfo *hashfunctions, Oid *collations, long nbuckets, Size additionalsize, MemoryContext metacxt, MemoryContext tablecxt, MemoryContext tempcxt, bool use_variable_hash_iv)
Definition: execGrouping.c:153
void ResetTupleHashTable(TupleHashTable hashtable)
Definition: execGrouping.c:284
ExprState * execTuplesMatchPrepare(TupleDesc desc, int numCols, const AttrNumber *keyColIdx, const Oid *eqOperators, const Oid *collations, PlanState *parent)
Definition: execGrouping.c:58
static TupleHashEntry LookupTupleHashEntry_internal(TupleHashTable hashtable, TupleTableSlot *slot, bool *isnew, uint32 hash)
Definition: execGrouping.c:495
TupleTableSlot * ExecStoreMinimalTuple(MinimalTuple mtup, TupleTableSlot *slot, bool shouldFree)
Definition: execTuples.c:1533
const TupleTableSlotOps TTSOpsMinimalTuple
Definition: execTuples.c:86
TupleTableSlot * MakeSingleTupleTableSlot(TupleDesc tupdesc, const TupleTableSlotOps *tts_ops)
Definition: execTuples.c:1325
ExprContext * CreateStandaloneExprContext(void)
Definition: execUtils.c:355
struct TupleHashTableData * TupleHashTable
Definition: execnodes.h:801
struct TupleHashEntryData TupleHashEntryData
static bool ExecQualAndReset(ExprState *state, ExprContext *econtext)
Definition: executor.h:451
void fmgr_info(Oid functionId, FmgrInfo *finfo)
Definition: fmgr.c:127
Datum FunctionCall1Coll(FmgrInfo *flinfo, Oid collation, Datum arg1)
Definition: fmgr.c:1129
static uint32 murmurhash32(uint32 data)
Definition: hashfn.h:92
int i
Definition: isn.c:73
if(TABLE==NULL||TABLE_index==NULL)
Definition: isn.c:77
RegProcedure get_opcode(Oid opno)
Definition: lsyscache.c:1285
bool get_op_hash_functions(Oid opno, RegProcedure *lhs_procno, RegProcedure *rhs_procno)
Definition: lsyscache.c:510
void * palloc(Size size)
Definition: mcxt.c:1317
size_t get_hash_memory_limit(void)
Definition: nodeHash.c:3481
static uint32 pg_rotate_left32(uint32 word, int n)
Definition: pg_bitutils.h:404
static uint32 DatumGetUInt32(Datum X)
Definition: postgres.h:222
uintptr_t Datum
Definition: postgres.h:64
unsigned int Oid
Definition: postgres_ext.h:31
MemoryContextSwitchTo(old_ctx)
static unsigned hash(unsigned *uv, int n)
Definition: rege_dfa.c:715
Definition: fmgr.h:57
MinimalTuple firstTuple
Definition: execnodes.h:805
AttrNumber * keyColIdx
Definition: execnodes.h:823
FmgrInfo * tab_hash_funcs
Definition: execnodes.h:824
tuplehash_hash * hashtab
Definition: execnodes.h:821
MemoryContext tempcxt
Definition: execnodes.h:828
ExprState * tab_eq_func
Definition: execnodes.h:825
TupleTableSlot * tableslot
Definition: execnodes.h:830
ExprContext * exprcontext
Definition: execnodes.h:836
MemoryContext tablecxt
Definition: execnodes.h:827
TupleTableSlot * inputslot
Definition: execnodes.h:832
ExprState * cur_eq_func
Definition: execnodes.h:834
FmgrInfo * in_hash_funcs
Definition: execnodes.h:833
TupleDesc CreateTupleDescCopy(TupleDesc tupdesc)
Definition: tupdesc.c:133
static MinimalTuple ExecCopySlotMinimalTuple(TupleTableSlot *slot)
Definition: tuptable.h:492
static Datum slot_getattr(TupleTableSlot *slot, int attnum, bool *isnull)
Definition: tuptable.h:395