bool use_variable_hash_iv)
 {
        TupleHashTable hashtable;
-       Size            entrysize = sizeof(TupleHashEntryData) + additionalsize;
+       Size            entrysize;
        Size            hash_mem_limit;
        MemoryContext oldcontext;
        bool            allow_jit;
        uint32          hash_iv = 0;
 
        Assert(nbuckets > 0);
+       additionalsize = MAXALIGN(additionalsize);
+       entrysize = sizeof(TupleHashEntryData) + additionalsize;
 
        /* Limit initial table size request to not more than hash_mem */
        hash_mem_limit = get_hash_memory_limit() / entrysize;
        hashtable->tab_collations = collations;
        hashtable->tablecxt = tablecxt;
        hashtable->tempcxt = tempcxt;
+       hashtable->additionalsize = additionalsize;
        hashtable->tableslot = NULL;    /* will be made on first lookup */
        hashtable->inputslot = NULL;
        hashtable->in_hash_expr = NULL;
                {
                        /* created new entry */
                        *isnew = true;
-                       /* zero caller data */
-                       entry->additional = NULL;
+
                        MemoryContextSwitchTo(hashtable->tablecxt);
-                       /* Copy the first tuple into the table context */
+
                        entry->firstTuple = ExecCopySlotMinimalTuple(slot);
+                       if (hashtable->additionalsize > 0)
+                               entry->additional = palloc0(hashtable->additionalsize);
+                       else
+                               entry->additional = NULL;
                }
        }
        else
 
 #ifdef USE_INJECTION_POINTS
                if (IS_INJECTION_POINT_ATTACHED("hash-aggregate-oversize-table"))
                {
-                       nbuckets = memory / sizeof(TupleHashEntryData);
+                       nbuckets = memory / TupleHashEntrySize();
                        INJECTION_POINT_CACHED("hash-aggregate-oversize-table");
                }
 #endif
                transitionChunkSize = 0;
 
        return
-               sizeof(TupleHashEntryData) +
+               TupleHashEntrySize() +
                tupleChunkSize +
                pergroupChunkSize +
                transitionChunkSize;
        if (aggstate->hash_ngroups_current > 0)
        {
                aggstate->hashentrysize =
-                       sizeof(TupleHashEntryData) +
+                       TupleHashEntrySize() +
                        (hashkey_mem / (double) aggstate->hash_ngroups_current);
        }
 }
        if (aggstate->numtrans == 0)
                return;
 
-       pergroup = (AggStatePerGroup)
-               MemoryContextAlloc(hashtable->tablecxt,
-                                                  sizeof(AggStatePerGroupData) * aggstate->numtrans);
-
-       entry->additional = pergroup;
+       pergroup = (AggStatePerGroup) TupleHashEntryGetAdditional(hashtable, entry);
 
        /*
         * Initialize aggregates for new tuple group, lookup_hash_entries()
                {
                        if (isnew)
                                initialize_hash_entry(aggstate, hashtable, entry);
-                       pergroup[setno] = entry->additional;
+                       pergroup[setno] = TupleHashEntryGetAdditional(hashtable, entry);
                }
                else
                {
        {
                TupleTableSlot *spillslot = aggstate->hash_spill_rslot;
                TupleTableSlot *hashslot = perhash->hashslot;
+               TupleHashTable hashtable = perhash->hashtable;
                TupleHashEntry entry;
                MinimalTuple tuple;
                uint32          hash;
                prepare_hash_slot(perhash,
                                                  aggstate->tmpcontext->ecxt_outertuple,
                                                  hashslot);
-               entry = LookupTupleHashEntryHash(perhash->hashtable, hashslot,
+               entry = LookupTupleHashEntryHash(hashtable, hashslot,
                                                                                 p_isnew, hash);
 
                if (entry != NULL)
                {
                        if (isnew)
-                               initialize_hash_entry(aggstate, perhash->hashtable, entry);
-                       aggstate->hash_pergroup[batch->setno] = entry->additional;
+                               initialize_hash_entry(aggstate, hashtable, entry);
+                       aggstate->hash_pergroup[batch->setno] = TupleHashEntryGetAdditional(hashtable, entry);
                        advance_aggregates(aggstate);
                }
                else
        ExprContext *econtext;
        AggStatePerAgg peragg;
        AggStatePerGroup pergroup;
-       TupleHashEntryData *entry;
+       TupleHashEntry entry;
        TupleTableSlot *firstSlot;
        TupleTableSlot *result;
        AggStatePerHash perhash;
        for (;;)
        {
                TupleTableSlot *hashslot = perhash->hashslot;
+               TupleHashTable hashtable = perhash->hashtable;
                int                     i;
 
                CHECK_FOR_INTERRUPTS();
                /*
                 * Find the next entry in the hash table
                 */
-               entry = ScanTupleHashTable(perhash->hashtable, &perhash->hashiter);
+               entry = ScanTupleHashTable(hashtable, &perhash->hashiter);
                if (entry == NULL)
                {
                        int                     nextset = aggstate->current_set + 1;
 
                                perhash = &aggstate->perhash[aggstate->current_set];
 
-                               ResetTupleHashIterator(perhash->hashtable, &perhash->hashiter);
+                               ResetTupleHashIterator(hashtable, &perhash->hashiter);
 
                                continue;
                        }
                 * Transform representative tuple back into one with the right
                 * columns.
                 */
-               ExecStoreMinimalTuple(entry->firstTuple, hashslot, false);
+               ExecStoreMinimalTuple(TupleHashEntryGetTuple(entry), hashslot, false);
                slot_getallattrs(hashslot);
 
                ExecClearTuple(firstSlot);
                }
                ExecStoreVirtualTuple(firstSlot);
 
-               pergroup = (AggStatePerGroup) entry->additional;
+               pergroup = (AggStatePerGroup) TupleHashEntryGetAdditional(hashtable, entry);
 
                /*
                 * Use the representative input tuple for any references to
 
        for (;;)
        {
                TupleTableSlot *outerslot;
+               TupleHashTable hashtable = setopstate->hashtable;
                TupleHashEntryData *entry;
+               SetOpStatePerGroup pergroup;
                bool            isnew;
 
                outerslot = ExecProcNode(outerPlan);
                have_tuples = true;
 
                /* Find or build hashtable entry for this tuple's group */
-               entry = LookupTupleHashEntry(setopstate->hashtable,
+               entry = LookupTupleHashEntry(hashtable,
                                                                         outerslot,
                                                                         &isnew, NULL);
 
+               pergroup = TupleHashEntryGetAdditional(hashtable, entry);
                /* If new tuple group, initialize counts to zero */
                if (isnew)
                {
-                       entry->additional = (SetOpStatePerGroup)
-                               MemoryContextAllocZero(setopstate->hashtable->tablecxt,
-                                                                          sizeof(SetOpStatePerGroupData));
+                       pergroup->numLeft = 0;
+                       pergroup->numRight = 0;
                }
 
                /* Advance the counts */
-               ((SetOpStatePerGroup) entry->additional)->numLeft++;
+               pergroup->numLeft++;
 
                /* Must reset expression context after each hashtable lookup */
                ResetExprContext(econtext);
                for (;;)
                {
                        TupleTableSlot *innerslot;
+                       TupleHashTable hashtable = setopstate->hashtable;
                        TupleHashEntryData *entry;
 
                        innerslot = ExecProcNode(innerPlan);
                                break;
 
                        /* For tuples not seen previously, do not make hashtable entry */
-                       entry = LookupTupleHashEntry(setopstate->hashtable,
+                       entry = LookupTupleHashEntry(hashtable,
                                                                                 innerslot,
                                                                                 NULL, NULL);
 
                        /* Advance the counts if entry is already present */
                        if (entry)
-                               ((SetOpStatePerGroup) entry->additional)->numRight++;
+                       {
+                               SetOpStatePerGroup pergroup = TupleHashEntryGetAdditional(hashtable, entry);
+
+                               pergroup->numRight++;
+                       }
 
                        /* Must reset expression context after each hashtable lookup */
                        ResetExprContext(econtext);
 static TupleTableSlot *
 setop_retrieve_hash_table(SetOpState *setopstate)
 {
-       TupleHashEntryData *entry;
+       TupleHashEntry entry;
        TupleTableSlot *resultTupleSlot;
 
        /*
         */
        while (!setopstate->setop_done)
        {
+               TupleHashTable hashtable = setopstate->hashtable;
+               SetOpStatePerGroup pergroup;
+
                CHECK_FOR_INTERRUPTS();
 
                /*
                 * Find the next entry in the hash table
                 */
-               entry = ScanTupleHashTable(setopstate->hashtable, &setopstate->hashiter);
+               entry = ScanTupleHashTable(hashtable, &setopstate->hashiter);
                if (entry == NULL)
                {
                        /* No more entries in hashtable, so done */
                 * See if we should emit any copies of this tuple, and if so return
                 * the first copy.
                 */
-               set_output_count(setopstate, (SetOpStatePerGroup) entry->additional);
+               pergroup = TupleHashEntryGetAdditional(hashtable, entry);
+               set_output_count(setopstate, pergroup);
 
                if (setopstate->numOutput > 0)
                {
                        setopstate->numOutput--;
-                       return ExecStoreMinimalTuple(entry->firstTuple,
+                       return ExecStoreMinimalTuple(TupleHashEntryGetTuple(entry),
                                                                                 resultTupleSlot,
                                                                                 false);
                }
 
        {
                CHECK_FOR_INTERRUPTS();
 
-               ExecStoreMinimalTuple(entry->firstTuple, hashtable->tableslot, false);
+               ExecStoreMinimalTuple(TupleHashEntryGetTuple(entry), hashtable->tableslot, false);
                if (!execTuplesUnequal(slot, hashtable->tableslot,
                                                           numCols, keyColIdx,
                                                           eqfunctions,
 
                                                                                 ExprState *hashexpr);
 extern void ResetTupleHashTable(TupleHashTable hashtable);
 
+#ifndef FRONTEND
+/*
+ * Return size of the hash bucket. Useful for estimating memory usage.
+ */
+static inline size_t
+TupleHashEntrySize(void)
+{
+       return sizeof(TupleHashEntryData);
+}
+
+/*
+ * Return tuple from hash entry.
+ */
+static inline MinimalTuple
+TupleHashEntryGetTuple(TupleHashEntry entry)
+{
+       return entry->firstTuple;
+}
+
+/*
+ * Get a pointer into the additional space allocated for this entry. The
+ * memory will be maxaligned and zeroed.
+ *
+ * The amount of space available is the additionalsize requested in the call
+ * to BuildTupleHashTable(). If additionalsize was specified as zero, return
+ * NULL.
+ */
+static inline void *
+TupleHashEntryGetAdditional(TupleHashTable hashtable, TupleHashEntry entry)
+{
+       return entry->additional;
+}
+#endif
+
 /*
  * prototypes from functions in execJunk.c
  */
 
        Oid                *tab_collations; /* collations for hash and comparison */
        MemoryContext tablecxt;         /* memory context containing table */
        MemoryContext tempcxt;          /* context for function evaluations */
+       Size            additionalsize; /* size of additional data */
        TupleTableSlot *tableslot;      /* slot for referencing table entries */
        /* The following fields are set transiently for each table search: */
        TupleTableSlot *inputslot;      /* current input tuple's slot */