[HACKERS] Safer and faster get_attstatsslot() - Mailing list pgsql-hackers

From Tom Lane
Subject [HACKERS] Safer and faster get_attstatsslot()
Date
Msg-id 16364.1494520862@sss.pgh.pa.us
Whole thread Raw
Responses Re: [HACKERS] Safer and faster get_attstatsslot()  (Simon Riggs <simon@2ndquadrant.com>)
List pgsql-hackers
Monday's round of security patches was a lot more exciting than I would
have liked, because code that worked fine for Peter and me failed
erratically in the buildfarm.  What eventually emerged was that I'd added
some missing free_attstatsslot() calls in rangetypes_selfuncs.c, and
naively copied the first argument (atttype) from the matching
get_attstatsslot() calls.  One of those atttype values was in fact
wrong for the slot in question; this had been missed for years because
get_attstatsslot() doesn't actually do anything with that argument.
I think that at one point we had, or at least in the original conception
intended to have, an Assert that the atttype matched the actual stats
array element type found in the pg_statistic row; but we had to remove
it because in some cases the type in pg_statistic is only
binary-compatible with the datatype the applied operator is expecting.

So the existing API for get_attstatsslot()/free_attstatsslot() is just
seriously bug-prone.  It would be better if the caller did not have
to supply any type information; indeed really what we'd want is for
get_attstatsslot() to pass back the actual type it found in pg_statistic.

I also realized as I looked at the code that it's exceedingly inefficient
if the array element type is pass-by-reference --- then it'll incur a
separate palloc, copy, and pfree for each element.  We'd be a lot better
off to copy the stats array as a whole, especially since that would come
for free in the probably-common case that the array has to be detoasted.
This code was written with very small stats targets in mind, like about
10, and it just doesn't look very good when you're imagining 1000 or more
entries in the stats array.

So attached is a proposed redesign that makes the API for
get_attstatsslot()/free_attstatsslot() simpler and hopefully more
foolproof.  I've not made any particular attempt to performance-test
it, but it really ought to be a significant win for pass-by-ref element
types.  It will add an array-copying step that wasn't there before when
the element type is pass-by-value and the array isn't toasted, but that
seems like an acceptable price.

BTW, this patch makes the skewColType and skewColTypmod fields of Hash
plan nodes unnecessary, as the only reason for them was to satisfy
get_attstatsslot().  I didn't remove them here but it would make sense
to do so.

Comments?  Is this something that'd be OK to push now, or had I better
sit on it till v11?  Being freshly burned, I kinda want to fix it now,
but I recognize that my judgment may be colored by that.

            regards, tom lane

diff --git a/contrib/intarray/_int_selfuncs.c b/contrib/intarray/_int_selfuncs.c
index 9b4a22f..3d92025 100644
*** a/contrib/intarray/_int_selfuncs.c
--- b/contrib/intarray/_int_selfuncs.c
*************** _int_matchsel(PG_FUNCTION_ARGS)
*** 136,146 ****
      int            nmcelems = 0;
      float4        minfreq = 0.0;
      float4        nullfrac = 0.0;
!     Form_pg_statistic stats;
!     Datum       *values = NULL;
!     int            nvalues = 0;
!     float4       *numbers = NULL;
!     int            nnumbers = 0;

      /*
       * If expression is not "variable @@ something" or "something @@ variable"
--- 136,142 ----
      int            nmcelems = 0;
      float4        minfreq = 0.0;
      float4        nullfrac = 0.0;
!     AttStatsSlot sslot;

      /*
       * If expression is not "variable @@ something" or "something @@ variable"
*************** _int_matchsel(PG_FUNCTION_ARGS)
*** 193,198 ****
--- 189,196 ----
       */
      if (HeapTupleIsValid(vardata.statsTuple))
      {
+         Form_pg_statistic stats;
+
          stats = (Form_pg_statistic) GETSTRUCT(vardata.statsTuple);
          nullfrac = stats->stanullfrac;

*************** _int_matchsel(PG_FUNCTION_ARGS)
*** 200,228 ****
           * For an int4 array, the default array type analyze function will
           * collect a Most Common Elements list, which is an array of int4s.
           */
!         if (get_attstatsslot(vardata.statsTuple,
!                              INT4OID, -1,
                               STATISTIC_KIND_MCELEM, InvalidOid,
!                              NULL,
!                              &values, &nvalues,
!                              &numbers, &nnumbers))
          {
              /*
               * There should be three more Numbers than Values, because the
               * last three (for intarray) cells are taken for minimal, maximal
               * and nulls frequency. Punt if not.
               */
!             if (nnumbers == nvalues + 3)
              {
                  /* Grab the lowest frequency. */
!                 minfreq = numbers[nnumbers - (nnumbers - nvalues)];

!                 mcelems = values;
!                 mcefreqs = numbers;
!                 nmcelems = nvalues;
              }
          }
      }

      /* Process the logical expression in the query, using the stats */
      selec = int_query_opr_selec(GETQUERY(query) + query->size - 1,
--- 198,227 ----
           * For an int4 array, the default array type analyze function will
           * collect a Most Common Elements list, which is an array of int4s.
           */
!         if (get_attstatsslot(&sslot, vardata.statsTuple,
                               STATISTIC_KIND_MCELEM, InvalidOid,
!                              ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS))
          {
+             Assert(sslot.valuetype == INT4OID);
+
              /*
               * There should be three more Numbers than Values, because the
               * last three (for intarray) cells are taken for minimal, maximal
               * and nulls frequency. Punt if not.
               */
!             if (sslot.nnumbers == sslot.nvalues + 3)
              {
                  /* Grab the lowest frequency. */
!                 minfreq = sslot.numbers[sslot.nnumbers - (sslot.nnumbers - sslot.nvalues)];

!                 mcelems = sslot.values;
!                 mcefreqs = sslot.numbers;
!                 nmcelems = sslot.nvalues;
              }
          }
      }
+     else
+         memset(&sslot, 0, sizeof(sslot));

      /* Process the logical expression in the query, using the stats */
      selec = int_query_opr_selec(GETQUERY(query) + query->size - 1,
*************** _int_matchsel(PG_FUNCTION_ARGS)
*** 231,237 ****
      /* MCE stats count only non-null rows, so adjust for null rows. */
      selec *= (1.0 - nullfrac);

!     free_attstatsslot(INT4OID, values, nvalues, numbers, nnumbers);
      ReleaseVariableStats(vardata);

      CLAMP_PROBABILITY(selec);
--- 230,236 ----
      /* MCE stats count only non-null rows, so adjust for null rows. */
      selec *= (1.0 - nullfrac);

!     free_attstatsslot(&sslot);
      ReleaseVariableStats(vardata);

      CLAMP_PROBABILITY(selec);
diff --git a/src/backend/executor/nodeHash.c b/src/backend/executor/nodeHash.c
index cfc6b96..d9789d0 100644
*** a/src/backend/executor/nodeHash.c
--- b/src/backend/executor/nodeHash.c
*************** static void
*** 1283,1292 ****
  ExecHashBuildSkewHash(HashJoinTable hashtable, Hash *node, int mcvsToUse)
  {
      HeapTupleData *statsTuple;
!     Datum       *values;
!     int            nvalues;
!     float4       *numbers;
!     int            nnumbers;

      /* Do nothing if planner didn't identify the outer relation's join key */
      if (!OidIsValid(node->skewTable))
--- 1283,1289 ----
  ExecHashBuildSkewHash(HashJoinTable hashtable, Hash *node, int mcvsToUse)
  {
      HeapTupleData *statsTuple;
!     AttStatsSlot sslot;

      /* Do nothing if planner didn't identify the outer relation's join key */
      if (!OidIsValid(node->skewTable))
*************** ExecHashBuildSkewHash(HashJoinTable hash
*** 1305,1323 ****
      if (!HeapTupleIsValid(statsTuple))
          return;

!     if (get_attstatsslot(statsTuple, node->skewColType, node->skewColTypmod,
                           STATISTIC_KIND_MCV, InvalidOid,
!                          NULL,
!                          &values, &nvalues,
!                          &numbers, &nnumbers))
      {
          double        frac;
          int            nbuckets;
          FmgrInfo   *hashfunctions;
          int            i;

!         if (mcvsToUse > nvalues)
!             mcvsToUse = nvalues;

          /*
           * Calculate the expected fraction of outer relation that will
--- 1302,1318 ----
      if (!HeapTupleIsValid(statsTuple))
          return;

!     if (get_attstatsslot(&sslot, statsTuple,
                           STATISTIC_KIND_MCV, InvalidOid,
!                          ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS))
      {
          double        frac;
          int            nbuckets;
          FmgrInfo   *hashfunctions;
          int            i;

!         if (mcvsToUse > sslot.nvalues)
!             mcvsToUse = sslot.nvalues;

          /*
           * Calculate the expected fraction of outer relation that will
*************** ExecHashBuildSkewHash(HashJoinTable hash
*** 1326,1336 ****
           */
          frac = 0;
          for (i = 0; i < mcvsToUse; i++)
!             frac += numbers[i];
          if (frac < SKEW_MIN_OUTER_FRACTION)
          {
!             free_attstatsslot(node->skewColType,
!                               values, nvalues, numbers, nnumbers);
              ReleaseSysCache(statsTuple);
              return;
          }
--- 1321,1330 ----
           */
          frac = 0;
          for (i = 0; i < mcvsToUse; i++)
!             frac += sslot.numbers[i];
          if (frac < SKEW_MIN_OUTER_FRACTION)
          {
!             free_attstatsslot(&sslot);
              ReleaseSysCache(statsTuple);
              return;
          }
*************** ExecHashBuildSkewHash(HashJoinTable hash
*** 1392,1398 ****
              int            bucket;

              hashvalue = DatumGetUInt32(FunctionCall1(&hashfunctions[0],
!                                                      values[i]));

              /*
               * While we have not hit a hole in the hashtable and have not hit
--- 1386,1392 ----
              int            bucket;

              hashvalue = DatumGetUInt32(FunctionCall1(&hashfunctions[0],
!                                                      sslot.values[i]));

              /*
               * While we have not hit a hole in the hashtable and have not hit
*************** ExecHashBuildSkewHash(HashJoinTable hash
*** 1426,1433 ****
                  hashtable->spacePeak = hashtable->spaceUsed;
          }

!         free_attstatsslot(node->skewColType,
!                           values, nvalues, numbers, nnumbers);
      }

      ReleaseSysCache(statsTuple);
--- 1420,1426 ----
                  hashtable->spacePeak = hashtable->spaceUsed;
          }

!         free_attstatsslot(&sslot);
      }

      ReleaseSysCache(statsTuple);
diff --git a/src/backend/tsearch/ts_selfuncs.c b/src/backend/tsearch/ts_selfuncs.c
index 904d884..046f543 100644
*** a/src/backend/tsearch/ts_selfuncs.c
--- b/src/backend/tsearch/ts_selfuncs.c
*************** tsquerysel(VariableStatData *vardata, Da
*** 163,190 ****
      if (HeapTupleIsValid(vardata->statsTuple))
      {
          Form_pg_statistic stats;
!         Datum       *values;
!         int            nvalues;
!         float4       *numbers;
!         int            nnumbers;

          stats = (Form_pg_statistic) GETSTRUCT(vardata->statsTuple);

          /* MCELEM will be an array of TEXT elements for a tsvector column */
!         if (get_attstatsslot(vardata->statsTuple,
!                              TEXTOID, -1,
                               STATISTIC_KIND_MCELEM, InvalidOid,
!                              NULL,
!                              &values, &nvalues,
!                              &numbers, &nnumbers))
          {
              /*
               * There is a most-common-elements slot for the tsvector Var, so
               * use that.
               */
!             selec = mcelem_tsquery_selec(query, values, nvalues,
!                                          numbers, nnumbers);
!             free_attstatsslot(TEXTOID, values, nvalues, numbers, nnumbers);
          }
          else
          {
--- 163,184 ----
      if (HeapTupleIsValid(vardata->statsTuple))
      {
          Form_pg_statistic stats;
!         AttStatsSlot sslot;

          stats = (Form_pg_statistic) GETSTRUCT(vardata->statsTuple);

          /* MCELEM will be an array of TEXT elements for a tsvector column */
!         if (get_attstatsslot(&sslot, vardata->statsTuple,
                               STATISTIC_KIND_MCELEM, InvalidOid,
!                              ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS))
          {
              /*
               * There is a most-common-elements slot for the tsvector Var, so
               * use that.
               */
!             selec = mcelem_tsquery_selec(query, sslot.values, sslot.nvalues,
!                                          sslot.numbers, sslot.nnumbers);
!             free_attstatsslot(&sslot);
          }
          else
          {
diff --git a/src/backend/utils/adt/array_selfuncs.c b/src/backend/utils/adt/array_selfuncs.c
index cfaf873..3ae6018 100644
*** a/src/backend/utils/adt/array_selfuncs.c
--- b/src/backend/utils/adt/array_selfuncs.c
*************** scalararraysel_containment(PlannerInfo *
*** 137,171 ****
          statistic_proc_security_check(&vardata, cmpfunc->fn_oid))
      {
          Form_pg_statistic stats;
!         Datum       *values;
!         int            nvalues;
!         float4       *numbers;
!         int            nnumbers;
!         float4       *hist;
!         int            nhist;

          stats = (Form_pg_statistic) GETSTRUCT(vardata.statsTuple);

          /* MCELEM will be an array of same type as element */
!         if (get_attstatsslot(vardata.statsTuple,
!                              elemtype, vardata.atttypmod,
                               STATISTIC_KIND_MCELEM, InvalidOid,
!                              NULL,
!                              &values, &nvalues,
!                              &numbers, &nnumbers))
          {
              /* For ALL case, also get histogram of distinct-element counts */
              if (useOr ||
!                 !get_attstatsslot(vardata.statsTuple,
!                                   elemtype, vardata.atttypmod,
                                    STATISTIC_KIND_DECHIST, InvalidOid,
!                                   NULL,
!                                   NULL, NULL,
!                                   &hist, &nhist))
!             {
!                 hist = NULL;
!                 nhist = 0;
!             }

              /*
               * For = ANY, estimate as var @> ARRAY[const].
--- 137,158 ----
          statistic_proc_security_check(&vardata, cmpfunc->fn_oid))
      {
          Form_pg_statistic stats;
!         AttStatsSlot sslot;
!         AttStatsSlot hslot;

          stats = (Form_pg_statistic) GETSTRUCT(vardata.statsTuple);

          /* MCELEM will be an array of same type as element */
!         if (get_attstatsslot(&sslot, vardata.statsTuple,
                               STATISTIC_KIND_MCELEM, InvalidOid,
!                              ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS))
          {
              /* For ALL case, also get histogram of distinct-element counts */
              if (useOr ||
!                 !get_attstatsslot(&hslot, vardata.statsTuple,
                                    STATISTIC_KIND_DECHIST, InvalidOid,
!                                   ATTSTATSSLOT_NUMBERS))
!                 memset(&hslot, 0, sizeof(hslot));

              /*
               * For = ANY, estimate as var @> ARRAY[const].
*************** scalararraysel_containment(PlannerInfo *
*** 173,194 ****
               * For = ALL, estimate as var <@ ARRAY[const].
               */
              if (useOr)
!                 selec = mcelem_array_contain_overlap_selec(values, nvalues,
!                                                            numbers, nnumbers,
                                                             &constval, 1,
                                                         OID_ARRAY_CONTAINS_OP,
                                                             cmpfunc);
              else
!                 selec = mcelem_array_contained_selec(values, nvalues,
!                                                      numbers, nnumbers,
                                                       &constval, 1,
!                                                      hist, nhist,
                                                       OID_ARRAY_CONTAINED_OP,
                                                       cmpfunc);

!             if (hist)
!                 free_attstatsslot(elemtype, NULL, 0, hist, nhist);
!             free_attstatsslot(elemtype, values, nvalues, numbers, nnumbers);
          }
          else
          {
--- 160,185 ----
               * For = ALL, estimate as var <@ ARRAY[const].
               */
              if (useOr)
!                 selec = mcelem_array_contain_overlap_selec(sslot.values,
!                                                            sslot.nvalues,
!                                                            sslot.numbers,
!                                                            sslot.nnumbers,
                                                             &constval, 1,
                                                         OID_ARRAY_CONTAINS_OP,
                                                             cmpfunc);
              else
!                 selec = mcelem_array_contained_selec(sslot.values,
!                                                      sslot.nvalues,
!                                                      sslot.numbers,
!                                                      sslot.nnumbers,
                                                       &constval, 1,
!                                                      hslot.numbers,
!                                                      hslot.nnumbers,
                                                       OID_ARRAY_CONTAINED_OP,
                                                       cmpfunc);

!             free_attstatsslot(&hslot);
!             free_attstatsslot(&sslot);
          }
          else
          {
*************** calc_arraycontsel(VariableStatData *vard
*** 369,417 ****
          statistic_proc_security_check(vardata, cmpfunc->fn_oid))
      {
          Form_pg_statistic stats;
!         Datum       *values;
!         int            nvalues;
!         float4       *numbers;
!         int            nnumbers;
!         float4       *hist;
!         int            nhist;

          stats = (Form_pg_statistic) GETSTRUCT(vardata->statsTuple);

          /* MCELEM will be an array of same type as column */
!         if (get_attstatsslot(vardata->statsTuple,
!                              elemtype, vardata->atttypmod,
                               STATISTIC_KIND_MCELEM, InvalidOid,
!                              NULL,
!                              &values, &nvalues,
!                              &numbers, &nnumbers))
          {
              /*
               * For "array <@ const" case we also need histogram of distinct
               * element counts.
               */
              if (operator != OID_ARRAY_CONTAINED_OP ||
!                 !get_attstatsslot(vardata->statsTuple,
!                                   elemtype, vardata->atttypmod,
                                    STATISTIC_KIND_DECHIST, InvalidOid,
!                                   NULL,
!                                   NULL, NULL,
!                                   &hist, &nhist))
!             {
!                 hist = NULL;
!                 nhist = 0;
!             }

              /* Use the most-common-elements slot for the array Var. */
              selec = mcelem_array_selec(array, typentry,
!                                        values, nvalues,
!                                        numbers, nnumbers,
!                                        hist, nhist,
                                         operator, cmpfunc);

!             if (hist)
!                 free_attstatsslot(elemtype, NULL, 0, hist, nhist);
!             free_attstatsslot(elemtype, values, nvalues, numbers, nnumbers);
          }
          else
          {
--- 360,394 ----
          statistic_proc_security_check(vardata, cmpfunc->fn_oid))
      {
          Form_pg_statistic stats;
!         AttStatsSlot sslot;
!         AttStatsSlot hslot;

          stats = (Form_pg_statistic) GETSTRUCT(vardata->statsTuple);

          /* MCELEM will be an array of same type as column */
!         if (get_attstatsslot(&sslot, vardata->statsTuple,
                               STATISTIC_KIND_MCELEM, InvalidOid,
!                              ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS))
          {
              /*
               * For "array <@ const" case we also need histogram of distinct
               * element counts.
               */
              if (operator != OID_ARRAY_CONTAINED_OP ||
!                 !get_attstatsslot(&hslot, vardata->statsTuple,
                                    STATISTIC_KIND_DECHIST, InvalidOid,
!                                   ATTSTATSSLOT_NUMBERS))
!                 memset(&hslot, 0, sizeof(hslot));

              /* Use the most-common-elements slot for the array Var. */
              selec = mcelem_array_selec(array, typentry,
!                                        sslot.values, sslot.nvalues,
!                                        sslot.numbers, sslot.nnumbers,
!                                        hslot.numbers, hslot.nnumbers,
                                         operator, cmpfunc);

!             free_attstatsslot(&hslot);
!             free_attstatsslot(&sslot);
          }
          else
          {
diff --git a/src/backend/utils/adt/network_selfuncs.c b/src/backend/utils/adt/network_selfuncs.c
index bcdd902..1d29ecd 100644
*** a/src/backend/utils/adt/network_selfuncs.c
--- b/src/backend/utils/adt/network_selfuncs.c
*************** networksel(PG_FUNCTION_ARGS)
*** 88,97 ****
      Selectivity selec,
                  mcv_selec,
                  non_mcv_selec;
!     Datum        constvalue,
!                *hist_values;
!     int            hist_nvalues;
      Form_pg_statistic stats;
      double        sumcommon,
                  nullfrac;
      FmgrInfo    proc;
--- 88,96 ----
      Selectivity selec,
                  mcv_selec,
                  non_mcv_selec;
!     Datum        constvalue;
      Form_pg_statistic stats;
+     AttStatsSlot hslot;
      double        sumcommon,
                  nullfrac;
      FmgrInfo    proc;
*************** networksel(PG_FUNCTION_ARGS)
*** 146,167 ****
       * non-MCV population that satisfies the clause.  If we don't, apply the
       * default selectivity to that population.
       */
!     if (get_attstatsslot(vardata.statsTuple,
!                          vardata.atttype, vardata.atttypmod,
                           STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                          NULL,
!                          &hist_values, &hist_nvalues,
!                          NULL, NULL))
      {
          int            opr_codenum = inet_opr_codenum(operator);

          /* Commute if needed, so we can consider histogram to be on the left */
          if (!varonleft)
              opr_codenum = -opr_codenum;
!         non_mcv_selec = inet_hist_value_sel(hist_values, hist_nvalues,
                                              constvalue, opr_codenum);

!         free_attstatsslot(vardata.atttype, hist_values, hist_nvalues, NULL, 0);
      }
      else
          non_mcv_selec = DEFAULT_SEL(operator);
--- 145,163 ----
       * non-MCV population that satisfies the clause.  If we don't, apply the
       * default selectivity to that population.
       */
!     if (get_attstatsslot(&hslot, vardata.statsTuple,
                           STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                          ATTSTATSSLOT_VALUES))
      {
          int            opr_codenum = inet_opr_codenum(operator);

          /* Commute if needed, so we can consider histogram to be on the left */
          if (!varonleft)
              opr_codenum = -opr_codenum;
!         non_mcv_selec = inet_hist_value_sel(hslot.values, hslot.nvalues,
                                              constvalue, opr_codenum);

!         free_attstatsslot(&hslot);
      }
      else
          non_mcv_selec = DEFAULT_SEL(operator);
*************** networkjoinsel_inner(Oid operator,
*** 277,318 ****
                  hist1_exists = false,
                  hist2_exists = false;
      int            opr_codenum;
!     int            mcv1_nvalues,
!                 mcv2_nvalues,
!                 mcv1_nnumbers,
!                 mcv2_nnumbers,
!                 hist1_nvalues,
!                 hist2_nvalues,
!                 mcv1_length = 0,
                  mcv2_length = 0;
!     Datum       *mcv1_values,
!                *mcv2_values,
!                *hist1_values,
!                *hist2_values;
!     float4       *mcv1_numbers,
!                *mcv2_numbers;

      if (HeapTupleIsValid(vardata1->statsTuple))
      {
          stats = (Form_pg_statistic) GETSTRUCT(vardata1->statsTuple);
          nullfrac1 = stats->stanullfrac;

!         mcv1_exists = get_attstatsslot(vardata1->statsTuple,
!                                        vardata1->atttype, vardata1->atttypmod,
                                         STATISTIC_KIND_MCV, InvalidOid,
!                                        NULL,
!                                        &mcv1_values, &mcv1_nvalues,
!                                        &mcv1_numbers, &mcv1_nnumbers);
!         hist1_exists = get_attstatsslot(vardata1->statsTuple,
!                                       vardata1->atttype, vardata1->atttypmod,
                                          STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                                         NULL,
!                                         &hist1_values, &hist1_nvalues,
!                                         NULL, NULL);
          /* Arbitrarily limit number of MCVs considered */
!         mcv1_length = Min(mcv1_nvalues, MAX_CONSIDERED_ELEMS);
          if (mcv1_exists)
!             sumcommon1 = mcv_population(mcv1_numbers, mcv1_length);
      }

      if (HeapTupleIsValid(vardata2->statsTuple))
--- 273,305 ----
                  hist1_exists = false,
                  hist2_exists = false;
      int            opr_codenum;
!     int            mcv1_length = 0,
                  mcv2_length = 0;
!     AttStatsSlot mcv1_slot;
!     AttStatsSlot mcv2_slot;
!     AttStatsSlot hist1_slot;
!     AttStatsSlot hist2_slot;

      if (HeapTupleIsValid(vardata1->statsTuple))
      {
          stats = (Form_pg_statistic) GETSTRUCT(vardata1->statsTuple);
          nullfrac1 = stats->stanullfrac;

!         mcv1_exists = get_attstatsslot(&mcv1_slot, vardata1->statsTuple,
                                         STATISTIC_KIND_MCV, InvalidOid,
!                                  ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS);
!         hist1_exists = get_attstatsslot(&hist1_slot, vardata1->statsTuple,
                                          STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                                         ATTSTATSSLOT_VALUES);
          /* Arbitrarily limit number of MCVs considered */
!         mcv1_length = Min(mcv1_slot.nvalues, MAX_CONSIDERED_ELEMS);
          if (mcv1_exists)
!             sumcommon1 = mcv_population(mcv1_slot.numbers, mcv1_length);
!     }
!     else
!     {
!         memset(&mcv1_slot, 0, sizeof(mcv1_slot));
!         memset(&hist1_slot, 0, sizeof(hist1_slot));
      }

      if (HeapTupleIsValid(vardata2->statsTuple))
*************** networkjoinsel_inner(Oid operator,
*** 320,341 ****
          stats = (Form_pg_statistic) GETSTRUCT(vardata2->statsTuple);
          nullfrac2 = stats->stanullfrac;

!         mcv2_exists = get_attstatsslot(vardata2->statsTuple,
!                                        vardata2->atttype, vardata2->atttypmod,
                                         STATISTIC_KIND_MCV, InvalidOid,
!                                        NULL,
!                                        &mcv2_values, &mcv2_nvalues,
!                                        &mcv2_numbers, &mcv2_nnumbers);
!         hist2_exists = get_attstatsslot(vardata2->statsTuple,
!                                       vardata2->atttype, vardata2->atttypmod,
                                          STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                                         NULL,
!                                         &hist2_values, &hist2_nvalues,
!                                         NULL, NULL);
          /* Arbitrarily limit number of MCVs considered */
!         mcv2_length = Min(mcv2_nvalues, MAX_CONSIDERED_ELEMS);
          if (mcv2_exists)
!             sumcommon2 = mcv_population(mcv2_numbers, mcv2_length);
      }

      opr_codenum = inet_opr_codenum(operator);
--- 307,327 ----
          stats = (Form_pg_statistic) GETSTRUCT(vardata2->statsTuple);
          nullfrac2 = stats->stanullfrac;

!         mcv2_exists = get_attstatsslot(&mcv2_slot, vardata2->statsTuple,
                                         STATISTIC_KIND_MCV, InvalidOid,
!                                  ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS);
!         hist2_exists = get_attstatsslot(&hist2_slot, vardata2->statsTuple,
                                          STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                                         ATTSTATSSLOT_VALUES);
          /* Arbitrarily limit number of MCVs considered */
!         mcv2_length = Min(mcv2_slot.nvalues, MAX_CONSIDERED_ELEMS);
          if (mcv2_exists)
!             sumcommon2 = mcv_population(mcv2_slot.numbers, mcv2_length);
!     }
!     else
!     {
!         memset(&mcv2_slot, 0, sizeof(mcv2_slot));
!         memset(&hist2_slot, 0, sizeof(hist2_slot));
      }

      opr_codenum = inet_opr_codenum(operator);
*************** networkjoinsel_inner(Oid operator,
*** 344,351 ****
       * Calculate selectivity for MCV vs MCV matches.
       */
      if (mcv1_exists && mcv2_exists)
!         selec += inet_mcv_join_sel(mcv1_values, mcv1_numbers, mcv1_length,
!                                    mcv2_values, mcv2_numbers, mcv2_length,
                                     operator);

      /*
--- 330,339 ----
       * Calculate selectivity for MCV vs MCV matches.
       */
      if (mcv1_exists && mcv2_exists)
!         selec += inet_mcv_join_sel(mcv1_slot.values, mcv1_slot.numbers,
!                                    mcv1_length,
!                                    mcv2_slot.values, mcv2_slot.numbers,
!                                    mcv2_length,
                                     operator);

      /*
*************** networkjoinsel_inner(Oid operator,
*** 355,367 ****
       */
      if (mcv1_exists && hist2_exists)
          selec += (1.0 - nullfrac2 - sumcommon2) *
!             inet_mcv_hist_sel(mcv1_values, mcv1_numbers, mcv1_length,
!                               hist2_values, hist2_nvalues,
                                opr_codenum);
      if (mcv2_exists && hist1_exists)
          selec += (1.0 - nullfrac1 - sumcommon1) *
!             inet_mcv_hist_sel(mcv2_values, mcv2_numbers, mcv2_length,
!                               hist1_values, hist1_nvalues,
                                -opr_codenum);

      /*
--- 343,355 ----
       */
      if (mcv1_exists && hist2_exists)
          selec += (1.0 - nullfrac2 - sumcommon2) *
!             inet_mcv_hist_sel(mcv1_slot.values, mcv1_slot.numbers, mcv1_length,
!                               hist2_slot.values, hist2_slot.nvalues,
                                opr_codenum);
      if (mcv2_exists && hist1_exists)
          selec += (1.0 - nullfrac1 - sumcommon1) *
!             inet_mcv_hist_sel(mcv2_slot.values, mcv2_slot.numbers, mcv2_length,
!                               hist1_slot.values, hist1_slot.nvalues,
                                -opr_codenum);

      /*
*************** networkjoinsel_inner(Oid operator,
*** 371,378 ****
      if (hist1_exists && hist2_exists)
          selec += (1.0 - nullfrac1 - sumcommon1) *
              (1.0 - nullfrac2 - sumcommon2) *
!             inet_hist_inclusion_join_sel(hist1_values, hist1_nvalues,
!                                          hist2_values, hist2_nvalues,
                                           opr_codenum);

      /*
--- 359,366 ----
      if (hist1_exists && hist2_exists)
          selec += (1.0 - nullfrac1 - sumcommon1) *
              (1.0 - nullfrac2 - sumcommon2) *
!             inet_hist_inclusion_join_sel(hist1_slot.values, hist1_slot.nvalues,
!                                        hist2_slot.values, hist2_slot.nvalues,
                                           opr_codenum);

      /*
*************** networkjoinsel_inner(Oid operator,
*** 383,400 ****
          selec = (1.0 - nullfrac1) * (1.0 - nullfrac2) * DEFAULT_SEL(operator);

      /* Release stats. */
!     if (mcv1_exists)
!         free_attstatsslot(vardata1->atttype, mcv1_values, mcv1_nvalues,
!                           mcv1_numbers, mcv1_nnumbers);
!     if (mcv2_exists)
!         free_attstatsslot(vardata2->atttype, mcv2_values, mcv2_nvalues,
!                           mcv2_numbers, mcv2_nnumbers);
!     if (hist1_exists)
!         free_attstatsslot(vardata1->atttype, hist1_values, hist1_nvalues,
!                           NULL, 0);
!     if (hist2_exists)
!         free_attstatsslot(vardata2->atttype, hist2_values, hist2_nvalues,
!                           NULL, 0);

      return selec;
  }
--- 371,380 ----
          selec = (1.0 - nullfrac1) * (1.0 - nullfrac2) * DEFAULT_SEL(operator);

      /* Release stats. */
!     free_attstatsslot(&mcv1_slot);
!     free_attstatsslot(&mcv2_slot);
!     free_attstatsslot(&hist1_slot);
!     free_attstatsslot(&hist2_slot);

      return selec;
  }
*************** networkjoinsel_semi(Oid operator,
*** 423,464 ****
      int            opr_codenum;
      FmgrInfo    proc;
      int            i,
-                 mcv1_nvalues,
-                 mcv2_nvalues,
-                 mcv1_nnumbers,
-                 mcv2_nnumbers,
-                 hist1_nvalues,
-                 hist2_nvalues,
                  mcv1_length = 0,
                  mcv2_length = 0;
!     Datum       *mcv1_values,
!                *mcv2_values,
!                *hist1_values,
!                *hist2_values;
!     float4       *mcv1_numbers,
!                *mcv2_numbers;

      if (HeapTupleIsValid(vardata1->statsTuple))
      {
          stats = (Form_pg_statistic) GETSTRUCT(vardata1->statsTuple);
          nullfrac1 = stats->stanullfrac;

!         mcv1_exists = get_attstatsslot(vardata1->statsTuple,
!                                        vardata1->atttype, vardata1->atttypmod,
                                         STATISTIC_KIND_MCV, InvalidOid,
!                                        NULL,
!                                        &mcv1_values, &mcv1_nvalues,
!                                        &mcv1_numbers, &mcv1_nnumbers);
!         hist1_exists = get_attstatsslot(vardata1->statsTuple,
!                                       vardata1->atttype, vardata1->atttypmod,
                                          STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                                         NULL,
!                                         &hist1_values, &hist1_nvalues,
!                                         NULL, NULL);
          /* Arbitrarily limit number of MCVs considered */
!         mcv1_length = Min(mcv1_nvalues, MAX_CONSIDERED_ELEMS);
          if (mcv1_exists)
!             sumcommon1 = mcv_population(mcv1_numbers, mcv1_length);
      }

      if (HeapTupleIsValid(vardata2->statsTuple))
--- 403,435 ----
      int            opr_codenum;
      FmgrInfo    proc;
      int            i,
                  mcv1_length = 0,
                  mcv2_length = 0;
!     AttStatsSlot mcv1_slot;
!     AttStatsSlot mcv2_slot;
!     AttStatsSlot hist1_slot;
!     AttStatsSlot hist2_slot;

      if (HeapTupleIsValid(vardata1->statsTuple))
      {
          stats = (Form_pg_statistic) GETSTRUCT(vardata1->statsTuple);
          nullfrac1 = stats->stanullfrac;

!         mcv1_exists = get_attstatsslot(&mcv1_slot, vardata1->statsTuple,
                                         STATISTIC_KIND_MCV, InvalidOid,
!                                  ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS);
!         hist1_exists = get_attstatsslot(&hist1_slot, vardata1->statsTuple,
                                          STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                                         ATTSTATSSLOT_VALUES);
          /* Arbitrarily limit number of MCVs considered */
!         mcv1_length = Min(mcv1_slot.nvalues, MAX_CONSIDERED_ELEMS);
          if (mcv1_exists)
!             sumcommon1 = mcv_population(mcv1_slot.numbers, mcv1_length);
!     }
!     else
!     {
!         memset(&mcv1_slot, 0, sizeof(mcv1_slot));
!         memset(&hist1_slot, 0, sizeof(hist1_slot));
      }

      if (HeapTupleIsValid(vardata2->statsTuple))
*************** networkjoinsel_semi(Oid operator,
*** 466,487 ****
          stats = (Form_pg_statistic) GETSTRUCT(vardata2->statsTuple);
          nullfrac2 = stats->stanullfrac;

!         mcv2_exists = get_attstatsslot(vardata2->statsTuple,
!                                        vardata2->atttype, vardata2->atttypmod,
                                         STATISTIC_KIND_MCV, InvalidOid,
!                                        NULL,
!                                        &mcv2_values, &mcv2_nvalues,
!                                        &mcv2_numbers, &mcv2_nnumbers);
!         hist2_exists = get_attstatsslot(vardata2->statsTuple,
!                                       vardata2->atttype, vardata2->atttypmod,
                                          STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                                         NULL,
!                                         &hist2_values, &hist2_nvalues,
!                                         NULL, NULL);
          /* Arbitrarily limit number of MCVs considered */
!         mcv2_length = Min(mcv2_nvalues, MAX_CONSIDERED_ELEMS);
          if (mcv2_exists)
!             sumcommon2 = mcv_population(mcv2_numbers, mcv2_length);
      }

      opr_codenum = inet_opr_codenum(operator);
--- 437,457 ----
          stats = (Form_pg_statistic) GETSTRUCT(vardata2->statsTuple);
          nullfrac2 = stats->stanullfrac;

!         mcv2_exists = get_attstatsslot(&mcv2_slot, vardata2->statsTuple,
                                         STATISTIC_KIND_MCV, InvalidOid,
!                                  ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS);
!         hist2_exists = get_attstatsslot(&hist2_slot, vardata2->statsTuple,
                                          STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                                         ATTSTATSSLOT_VALUES);
          /* Arbitrarily limit number of MCVs considered */
!         mcv2_length = Min(mcv2_slot.nvalues, MAX_CONSIDERED_ELEMS);
          if (mcv2_exists)
!             sumcommon2 = mcv_population(mcv2_slot.numbers, mcv2_length);
!     }
!     else
!     {
!         memset(&mcv2_slot, 0, sizeof(mcv2_slot));
!         memset(&hist2_slot, 0, sizeof(hist2_slot));
      }

      opr_codenum = inet_opr_codenum(operator);
*************** networkjoinsel_semi(Oid operator,
*** 499,508 ****
      {
          for (i = 0; i < mcv1_length; i++)
          {
!             selec += mcv1_numbers[i] *
!                 inet_semi_join_sel(mcv1_values[i],
!                                    mcv2_exists, mcv2_values, mcv2_length,
!                                    hist2_exists, hist2_values, hist2_nvalues,
                                     hist2_weight,
                                     &proc, opr_codenum);
          }
--- 469,479 ----
      {
          for (i = 0; i < mcv1_length; i++)
          {
!             selec += mcv1_slot.numbers[i] *
!                 inet_semi_join_sel(mcv1_slot.values[i],
!                                    mcv2_exists, mcv2_slot.values, mcv2_length,
!                                    hist2_exists,
!                                    hist2_slot.values, hist2_slot.nvalues,
                                     hist2_weight,
                                     &proc, opr_codenum);
          }
*************** networkjoinsel_semi(Oid operator,
*** 519,539 ****
       *
       * If there are too many histogram elements, decimate to limit runtime.
       */
!     if (hist1_exists && hist1_nvalues > 2 && (mcv2_exists || hist2_exists))
      {
          double        hist_selec_sum = 0.0;
          int            k,
                      n;

!         k = (hist1_nvalues - 3) / MAX_CONSIDERED_ELEMS + 1;

          n = 0;
!         for (i = 1; i < hist1_nvalues - 1; i += k)
          {
              hist_selec_sum +=
!                 inet_semi_join_sel(hist1_values[i],
!                                    mcv2_exists, mcv2_values, mcv2_length,
!                                    hist2_exists, hist2_values, hist2_nvalues,
                                     hist2_weight,
                                     &proc, opr_codenum);
              n++;
--- 490,511 ----
       *
       * If there are too many histogram elements, decimate to limit runtime.
       */
!     if (hist1_exists && hist1_slot.nvalues > 2 && (mcv2_exists || hist2_exists))
      {
          double        hist_selec_sum = 0.0;
          int            k,
                      n;

!         k = (hist1_slot.nvalues - 3) / MAX_CONSIDERED_ELEMS + 1;

          n = 0;
!         for (i = 1; i < hist1_slot.nvalues - 1; i += k)
          {
              hist_selec_sum +=
!                 inet_semi_join_sel(hist1_slot.values[i],
!                                    mcv2_exists, mcv2_slot.values, mcv2_length,
!                                    hist2_exists,
!                                    hist2_slot.values, hist2_slot.nvalues,
                                     hist2_weight,
                                     &proc, opr_codenum);
              n++;
*************** networkjoinsel_semi(Oid operator,
*** 550,567 ****
          selec = (1.0 - nullfrac1) * (1.0 - nullfrac2) * DEFAULT_SEL(operator);

      /* Release stats. */
!     if (mcv1_exists)
!         free_attstatsslot(vardata1->atttype, mcv1_values, mcv1_nvalues,
!                           mcv1_numbers, mcv1_nnumbers);
!     if (mcv2_exists)
!         free_attstatsslot(vardata2->atttype, mcv2_values, mcv2_nvalues,
!                           mcv2_numbers, mcv2_nnumbers);
!     if (hist1_exists)
!         free_attstatsslot(vardata1->atttype, hist1_values, hist1_nvalues,
!                           NULL, 0);
!     if (hist2_exists)
!         free_attstatsslot(vardata2->atttype, hist2_values, hist2_nvalues,
!                           NULL, 0);

      return selec;
  }
--- 522,531 ----
          selec = (1.0 - nullfrac1) * (1.0 - nullfrac2) * DEFAULT_SEL(operator);

      /* Release stats. */
!     free_attstatsslot(&mcv1_slot);
!     free_attstatsslot(&mcv2_slot);
!     free_attstatsslot(&hist1_slot);
!     free_attstatsslot(&hist2_slot);

      return selec;
  }
diff --git a/src/backend/utils/adt/rangetypes_selfuncs.c b/src/backend/utils/adt/rangetypes_selfuncs.c
index dbf1929..c4c5496 100644
*** a/src/backend/utils/adt/rangetypes_selfuncs.c
--- b/src/backend/utils/adt/rangetypes_selfuncs.c
*************** calc_rangesel(TypeCacheEntry *typcache,
*** 239,263 ****
      if (HeapTupleIsValid(vardata->statsTuple))
      {
          Form_pg_statistic stats;
!         float4       *numbers;
!         int            nnumbers;

          stats = (Form_pg_statistic) GETSTRUCT(vardata->statsTuple);
          null_frac = stats->stanullfrac;

          /* Try to get fraction of empty ranges */
!         if (get_attstatsslot(vardata->statsTuple,
!                              FLOAT8OID, -1,
                               STATISTIC_KIND_RANGE_LENGTH_HISTOGRAM,
                               InvalidOid,
!                              NULL,
!                              NULL, NULL,
!                              &numbers, &nnumbers))
          {
!             if (nnumbers != 1)
                  elog(ERROR, "invalid empty fraction statistic");        /* shouldn't happen */
!             empty_frac = numbers[0];
!             free_attstatsslot(FLOAT8OID, NULL, 0, numbers, nnumbers);
          }
          else
          {
--- 239,259 ----
      if (HeapTupleIsValid(vardata->statsTuple))
      {
          Form_pg_statistic stats;
!         AttStatsSlot sslot;

          stats = (Form_pg_statistic) GETSTRUCT(vardata->statsTuple);
          null_frac = stats->stanullfrac;

          /* Try to get fraction of empty ranges */
!         if (get_attstatsslot(&sslot, vardata->statsTuple,
                               STATISTIC_KIND_RANGE_LENGTH_HISTOGRAM,
                               InvalidOid,
!                              ATTSTATSSLOT_NUMBERS))
          {
!             if (sslot.nnumbers != 1)
                  elog(ERROR, "invalid empty fraction statistic");        /* shouldn't happen */
!             empty_frac = sslot.numbers[0];
!             free_attstatsslot(&sslot);
          }
          else
          {
*************** static double
*** 374,383 ****
  calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata,
                        RangeType *constval, Oid operator)
  {
!     Datum       *hist_values;
      int            nhist;
-     Datum       *length_hist_values = NULL;
-     int            length_nhist = 0;
      RangeBound *hist_lower;
      RangeBound *hist_upper;
      int            i;
--- 370,378 ----
  calc_hist_selectivity(TypeCacheEntry *typcache, VariableStatData *vardata,
                        RangeType *constval, Oid operator)
  {
!     AttStatsSlot hslot;
!     AttStatsSlot lslot;
      int            nhist;
      RangeBound *hist_lower;
      RangeBound *hist_upper;
      int            i;
*************** calc_hist_selectivity(TypeCacheEntry *ty
*** 397,419 ****

      /* Try to get histogram of ranges */
      if (!(HeapTupleIsValid(vardata->statsTuple) &&
!           get_attstatsslot(vardata->statsTuple,
!                            vardata->atttype, vardata->atttypmod,
                             STATISTIC_KIND_BOUNDS_HISTOGRAM, InvalidOid,
!                            NULL,
!                            &hist_values, &nhist,
!                            NULL, NULL)))
          return -1.0;

      /*
       * Convert histogram of ranges into histograms of its lower and upper
       * bounds.
       */
      hist_lower = (RangeBound *) palloc(sizeof(RangeBound) * nhist);
      hist_upper = (RangeBound *) palloc(sizeof(RangeBound) * nhist);
      for (i = 0; i < nhist; i++)
      {
!         range_deserialize(typcache, DatumGetRangeType(hist_values[i]),
                            &hist_lower[i], &hist_upper[i], &empty);
          /* The histogram should not contain any empty ranges */
          if (empty)
--- 392,412 ----

      /* Try to get histogram of ranges */
      if (!(HeapTupleIsValid(vardata->statsTuple) &&
!           get_attstatsslot(&hslot, vardata->statsTuple,
                             STATISTIC_KIND_BOUNDS_HISTOGRAM, InvalidOid,
!                            ATTSTATSSLOT_VALUES)))
          return -1.0;

      /*
       * Convert histogram of ranges into histograms of its lower and upper
       * bounds.
       */
+     nhist = hslot.nvalues;
      hist_lower = (RangeBound *) palloc(sizeof(RangeBound) * nhist);
      hist_upper = (RangeBound *) palloc(sizeof(RangeBound) * nhist);
      for (i = 0; i < nhist; i++)
      {
!         range_deserialize(typcache, DatumGetRangeType(hslot.values[i]),
                            &hist_lower[i], &hist_upper[i], &empty);
          /* The histogram should not contain any empty ranges */
          if (empty)
*************** calc_hist_selectivity(TypeCacheEntry *ty
*** 425,451 ****
          operator == OID_RANGE_CONTAINED_OP)
      {
          if (!(HeapTupleIsValid(vardata->statsTuple) &&
!               get_attstatsslot(vardata->statsTuple,
!                                FLOAT8OID, -1,
                                 STATISTIC_KIND_RANGE_LENGTH_HISTOGRAM,
                                 InvalidOid,
!                                NULL,
!                                &length_hist_values, &length_nhist,
!                                NULL, NULL)))
          {
!             free_attstatsslot(vardata->atttype, hist_values, nhist, NULL, 0);
              return -1.0;
          }

          /* check that it's a histogram, not just a dummy entry */
!         if (length_nhist < 2)
          {
!             free_attstatsslot(FLOAT8OID,
!                               length_hist_values, length_nhist, NULL, 0);
!             free_attstatsslot(vardata->atttype, hist_values, nhist, NULL, 0);
              return -1.0;
          }
      }

      /* Extract the bounds of the constant value. */
      range_deserialize(typcache, constval, &const_lower, &const_upper, &empty);
--- 418,442 ----
          operator == OID_RANGE_CONTAINED_OP)
      {
          if (!(HeapTupleIsValid(vardata->statsTuple) &&
!               get_attstatsslot(&lslot, vardata->statsTuple,
                                 STATISTIC_KIND_RANGE_LENGTH_HISTOGRAM,
                                 InvalidOid,
!                                ATTSTATSSLOT_VALUES)))
          {
!             free_attstatsslot(&hslot);
              return -1.0;
          }

          /* check that it's a histogram, not just a dummy entry */
!         if (lslot.nvalues < 2)
          {
!             free_attstatsslot(&lslot);
!             free_attstatsslot(&hslot);
              return -1.0;
          }
      }
+     else
+         memset(&lslot, 0, sizeof(lslot));

      /* Extract the bounds of the constant value. */
      range_deserialize(typcache, constval, &const_lower, &const_upper, &empty);
*************** calc_hist_selectivity(TypeCacheEntry *ty
*** 545,551 ****
              hist_selec =
                  calc_hist_selectivity_contains(typcache, &const_lower,
                                               &const_upper, hist_lower, nhist,
!                                            length_hist_values, length_nhist);
              break;

          case OID_RANGE_CONTAINED_OP:
--- 536,542 ----
              hist_selec =
                  calc_hist_selectivity_contains(typcache, &const_lower,
                                               &const_upper, hist_lower, nhist,
!                                                lslot.values, lslot.nvalues);
              break;

          case OID_RANGE_CONTAINED_OP:
*************** calc_hist_selectivity(TypeCacheEntry *ty
*** 570,576 ****
                  hist_selec =
                      calc_hist_selectivity_contained(typcache, &const_lower,
                                               &const_upper, hist_lower, nhist,
!                                            length_hist_values, length_nhist);
              }
              break;

--- 561,567 ----
                  hist_selec =
                      calc_hist_selectivity_contained(typcache, &const_lower,
                                               &const_upper, hist_lower, nhist,
!                                                 lslot.values, lslot.nvalues);
              }
              break;

*************** calc_hist_selectivity(TypeCacheEntry *ty
*** 580,588 ****
              break;
      }

!     free_attstatsslot(FLOAT8OID,
!                       length_hist_values, length_nhist, NULL, 0);
!     free_attstatsslot(vardata->atttype, hist_values, nhist, NULL, 0);

      return hist_selec;
  }
--- 571,578 ----
              break;
      }

!     free_attstatsslot(&lslot);
!     free_attstatsslot(&hslot);

      return hist_selec;
  }
diff --git a/src/backend/utils/adt/selfuncs.c b/src/backend/utils/adt/selfuncs.c
index 9b157f4..6c4cef9 100644
*** a/src/backend/utils/adt/selfuncs.c
--- b/src/backend/utils/adt/selfuncs.c
*************** var_eq_const(VariableStatData *vardata,
*** 299,308 ****
                                        (opfuncoid = get_opcode(operator))))
      {
          Form_pg_statistic stats;
!         Datum       *values;
!         int            nvalues;
!         float4       *numbers;
!         int            nnumbers;
          bool        match = false;
          int            i;

--- 299,305 ----
                                        (opfuncoid = get_opcode(operator))))
      {
          Form_pg_statistic stats;
!         AttStatsSlot sslot;
          bool        match = false;
          int            i;

*************** var_eq_const(VariableStatData *vardata,
*** 315,344 ****
           * don't like this, maybe you shouldn't be using eqsel for your
           * operator...)
           */
!         if (get_attstatsslot(vardata->statsTuple,
!                              vardata->atttype, vardata->atttypmod,
                               STATISTIC_KIND_MCV, InvalidOid,
!                              NULL,
!                              &values, &nvalues,
!                              &numbers, &nnumbers))
          {
              FmgrInfo    eqproc;

              fmgr_info(opfuncoid, &eqproc);

!             for (i = 0; i < nvalues; i++)
              {
                  /* be careful to apply operator right way 'round */
                  if (varonleft)
                      match = DatumGetBool(FunctionCall2Coll(&eqproc,
                                                         DEFAULT_COLLATION_OID,
!                                                            values[i],
                                                             constval));
                  else
                      match = DatumGetBool(FunctionCall2Coll(&eqproc,
                                                         DEFAULT_COLLATION_OID,
                                                             constval,
!                                                            values[i]));
                  if (match)
                      break;
              }
--- 312,338 ----
           * don't like this, maybe you shouldn't be using eqsel for your
           * operator...)
           */
!         if (get_attstatsslot(&sslot, vardata->statsTuple,
                               STATISTIC_KIND_MCV, InvalidOid,
!                              ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS))
          {
              FmgrInfo    eqproc;

              fmgr_info(opfuncoid, &eqproc);

!             for (i = 0; i < sslot.nvalues; i++)
              {
                  /* be careful to apply operator right way 'round */
                  if (varonleft)
                      match = DatumGetBool(FunctionCall2Coll(&eqproc,
                                                         DEFAULT_COLLATION_OID,
!                                                            sslot.values[i],
                                                             constval));
                  else
                      match = DatumGetBool(FunctionCall2Coll(&eqproc,
                                                         DEFAULT_COLLATION_OID,
                                                             constval,
!                                                            sslot.values[i]));
                  if (match)
                      break;
              }
*************** var_eq_const(VariableStatData *vardata,
*** 346,354 ****
          else
          {
              /* no most-common-value info available */
!             values = NULL;
!             numbers = NULL;
!             i = nvalues = nnumbers = 0;
          }

          if (match)
--- 340,346 ----
          else
          {
              /* no most-common-value info available */
!             i = 0;                /* keep compiler quiet */
          }

          if (match)
*************** var_eq_const(VariableStatData *vardata,
*** 357,363 ****
               * Constant is "=" to this common value.  We know selectivity
               * exactly (or as exactly as ANALYZE could calculate it, anyway).
               */
!             selec = numbers[i];
          }
          else
          {
--- 349,355 ----
               * Constant is "=" to this common value.  We know selectivity
               * exactly (or as exactly as ANALYZE could calculate it, anyway).
               */
!             selec = sslot.numbers[i];
          }
          else
          {
*************** var_eq_const(VariableStatData *vardata,
*** 369,376 ****
              double        sumcommon = 0.0;
              double        otherdistinct;

!             for (i = 0; i < nnumbers; i++)
!                 sumcommon += numbers[i];
              selec = 1.0 - sumcommon - stats->stanullfrac;
              CLAMP_PROBABILITY(selec);

--- 361,368 ----
              double        sumcommon = 0.0;
              double        otherdistinct;

!             for (i = 0; i < sslot.nnumbers; i++)
!                 sumcommon += sslot.numbers[i];
              selec = 1.0 - sumcommon - stats->stanullfrac;
              CLAMP_PROBABILITY(selec);

*************** var_eq_const(VariableStatData *vardata,
*** 379,385 ****
               * all the not-common values share this remaining fraction
               * equally, so we divide by the number of other distinct values.
               */
!             otherdistinct = get_variable_numdistinct(vardata, &isdefault) - nnumbers;
              if (otherdistinct > 1)
                  selec /= otherdistinct;

--- 371,378 ----
               * all the not-common values share this remaining fraction
               * equally, so we divide by the number of other distinct values.
               */
!             otherdistinct = get_variable_numdistinct(vardata, &isdefault) -
!                 sslot.nnumbers;
              if (otherdistinct > 1)
                  selec /= otherdistinct;

*************** var_eq_const(VariableStatData *vardata,
*** 387,398 ****
               * Another cross-check: selectivity shouldn't be estimated as more
               * than the least common "most common value".
               */
!             if (nnumbers > 0 && selec > numbers[nnumbers - 1])
!                 selec = numbers[nnumbers - 1];
          }

!         free_attstatsslot(vardata->atttype, values, nvalues,
!                           numbers, nnumbers);
      }
      else
      {
--- 380,390 ----
               * Another cross-check: selectivity shouldn't be estimated as more
               * than the least common "most common value".
               */
!             if (sslot.nnumbers > 0 && selec > sslot.numbers[sslot.nnumbers - 1])
!                 selec = sslot.numbers[sslot.nnumbers - 1];
          }

!         free_attstatsslot(&sslot);
      }
      else
      {
*************** var_eq_non_const(VariableStatData *varda
*** 435,442 ****
      {
          Form_pg_statistic stats;
          double        ndistinct;
!         float4       *numbers;
!         int            nnumbers;

          stats = (Form_pg_statistic) GETSTRUCT(vardata->statsTuple);

--- 427,433 ----
      {
          Form_pg_statistic stats;
          double        ndistinct;
!         AttStatsSlot sslot;

          stats = (Form_pg_statistic) GETSTRUCT(vardata->statsTuple);

*************** var_eq_non_const(VariableStatData *varda
*** 459,474 ****
           * Cross-check: selectivity should never be estimated as more than the
           * most common value's.
           */
!         if (get_attstatsslot(vardata->statsTuple,
!                              vardata->atttype, vardata->atttypmod,
                               STATISTIC_KIND_MCV, InvalidOid,
!                              NULL,
!                              NULL, NULL,
!                              &numbers, &nnumbers))
          {
!             if (nnumbers > 0 && selec > numbers[0])
!                 selec = numbers[0];
!             free_attstatsslot(vardata->atttype, NULL, 0, numbers, nnumbers);
          }
      }
      else
--- 450,462 ----
           * Cross-check: selectivity should never be estimated as more than the
           * most common value's.
           */
!         if (get_attstatsslot(&sslot, vardata->statsTuple,
                               STATISTIC_KIND_MCV, InvalidOid,
!                              ATTSTATSSLOT_NUMBERS))
          {
!             if (sslot.nnumbers > 0 && selec > sslot.numbers[0])
!                 selec = sslot.numbers[0];
!             free_attstatsslot(&sslot);
          }
      }
      else
*************** mcv_selectivity(VariableStatData *vardat
*** 620,629 ****
  {
      double        mcv_selec,
                  sumcommon;
!     Datum       *values;
!     int            nvalues;
!     float4       *numbers;
!     int            nnumbers;
      int            i;

      mcv_selec = 0.0;
--- 608,614 ----
  {
      double        mcv_selec,
                  sumcommon;
!     AttStatsSlot sslot;
      int            i;

      mcv_selec = 0.0;
*************** mcv_selectivity(VariableStatData *vardat
*** 631,659 ****

      if (HeapTupleIsValid(vardata->statsTuple) &&
          statistic_proc_security_check(vardata, opproc->fn_oid) &&
!         get_attstatsslot(vardata->statsTuple,
!                          vardata->atttype, vardata->atttypmod,
                           STATISTIC_KIND_MCV, InvalidOid,
!                          NULL,
!                          &values, &nvalues,
!                          &numbers, &nnumbers))
      {
!         for (i = 0; i < nvalues; i++)
          {
              if (varonleft ?
                  DatumGetBool(FunctionCall2Coll(opproc,
                                                 DEFAULT_COLLATION_OID,
!                                                values[i],
                                                 constval)) :
                  DatumGetBool(FunctionCall2Coll(opproc,
                                                 DEFAULT_COLLATION_OID,
                                                 constval,
!                                                values[i])))
!                 mcv_selec += numbers[i];
!             sumcommon += numbers[i];
          }
!         free_attstatsslot(vardata->atttype, values, nvalues,
!                           numbers, nnumbers);
      }

      *sumcommonp = sumcommon;
--- 616,640 ----

      if (HeapTupleIsValid(vardata->statsTuple) &&
          statistic_proc_security_check(vardata, opproc->fn_oid) &&
!         get_attstatsslot(&sslot, vardata->statsTuple,
                           STATISTIC_KIND_MCV, InvalidOid,
!                          ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS))
      {
!         for (i = 0; i < sslot.nvalues; i++)
          {
              if (varonleft ?
                  DatumGetBool(FunctionCall2Coll(opproc,
                                                 DEFAULT_COLLATION_OID,
!                                                sslot.values[i],
                                                 constval)) :
                  DatumGetBool(FunctionCall2Coll(opproc,
                                                 DEFAULT_COLLATION_OID,
                                                 constval,
!                                                sslot.values[i])))
!                 mcv_selec += sslot.numbers[i];
!             sumcommon += sslot.numbers[i];
          }
!         free_attstatsslot(&sslot);
      }

      *sumcommonp = sumcommon;
*************** histogram_selectivity(VariableStatData *
*** 699,706 ****
                        int *hist_size)
  {
      double        result;
!     Datum       *values;
!     int            nvalues;

      /* check sanity of parameters */
      Assert(n_skip >= 0);
--- 680,686 ----
                        int *hist_size)
  {
      double        result;
!     AttStatsSlot sslot;

      /* check sanity of parameters */
      Assert(n_skip >= 0);
*************** histogram_selectivity(VariableStatData *
*** 708,744 ****

      if (HeapTupleIsValid(vardata->statsTuple) &&
          statistic_proc_security_check(vardata, opproc->fn_oid) &&
!         get_attstatsslot(vardata->statsTuple,
!                          vardata->atttype, vardata->atttypmod,
                           STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                          NULL,
!                          &values, &nvalues,
!                          NULL, NULL))
      {
!         *hist_size = nvalues;
!         if (nvalues >= min_hist_size)
          {
              int            nmatch = 0;
              int            i;

!             for (i = n_skip; i < nvalues - n_skip; i++)
              {
                  if (varonleft ?
                      DatumGetBool(FunctionCall2Coll(opproc,
                                                     DEFAULT_COLLATION_OID,
!                                                    values[i],
                                                     constval)) :
                      DatumGetBool(FunctionCall2Coll(opproc,
                                                     DEFAULT_COLLATION_OID,
                                                     constval,
!                                                    values[i])))
                      nmatch++;
              }
!             result = ((double) nmatch) / ((double) (nvalues - 2 * n_skip));
          }
          else
              result = -1;
!         free_attstatsslot(vardata->atttype, values, nvalues, NULL, 0);
      }
      else
      {
--- 688,721 ----

      if (HeapTupleIsValid(vardata->statsTuple) &&
          statistic_proc_security_check(vardata, opproc->fn_oid) &&
!         get_attstatsslot(&sslot, vardata->statsTuple,
                           STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                          ATTSTATSSLOT_VALUES))
      {
!         *hist_size = sslot.nvalues;
!         if (sslot.nvalues >= min_hist_size)
          {
              int            nmatch = 0;
              int            i;

!             for (i = n_skip; i < sslot.nvalues - n_skip; i++)
              {
                  if (varonleft ?
                      DatumGetBool(FunctionCall2Coll(opproc,
                                                     DEFAULT_COLLATION_OID,
!                                                    sslot.values[i],
                                                     constval)) :
                      DatumGetBool(FunctionCall2Coll(opproc,
                                                     DEFAULT_COLLATION_OID,
                                                     constval,
!                                                    sslot.values[i])))
                      nmatch++;
              }
!             result = ((double) nmatch) / ((double) (sslot.nvalues - 2 * n_skip));
          }
          else
              result = -1;
!         free_attstatsslot(&sslot);
      }
      else
      {
*************** ineq_histogram_selectivity(PlannerInfo *
*** 768,776 ****
                             Datum constval, Oid consttype)
  {
      double        hist_selec;
!     Oid            hist_op;
!     Datum       *values;
!     int            nvalues;

      hist_selec = -1.0;

--- 745,751 ----
                             Datum constval, Oid consttype)
  {
      double        hist_selec;
!     AttStatsSlot sslot;

      hist_selec = -1.0;

*************** ineq_histogram_selectivity(PlannerInfo *
*** 786,799 ****
       */
      if (HeapTupleIsValid(vardata->statsTuple) &&
          statistic_proc_security_check(vardata, opproc->fn_oid) &&
!         get_attstatsslot(vardata->statsTuple,
!                          vardata->atttype, vardata->atttypmod,
                           STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                          &hist_op,
!                          &values, &nvalues,
!                          NULL, NULL))
      {
!         if (nvalues > 1)
          {
              /*
               * Use binary search to find proper location, ie, the first slot
--- 761,771 ----
       */
      if (HeapTupleIsValid(vardata->statsTuple) &&
          statistic_proc_security_check(vardata, opproc->fn_oid) &&
!         get_attstatsslot(&sslot, vardata->statsTuple,
                           STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                          ATTSTATSSLOT_VALUES))
      {
!         if (sslot.nvalues > 1)
          {
              /*
               * Use binary search to find proper location, ie, the first slot
*************** ineq_histogram_selectivity(PlannerInfo *
*** 812,818 ****
               */
              double        histfrac;
              int            lobound = 0;    /* first possible slot to search */
!             int            hibound = nvalues;        /* last+1 slot to search */
              bool        have_end = false;

              /*
--- 784,790 ----
               */
              double        histfrac;
              int            lobound = 0;    /* first possible slot to search */
!             int            hibound = sslot.nvalues;        /* last+1 slot to search */
              bool        have_end = false;

              /*
*************** ineq_histogram_selectivity(PlannerInfo *
*** 821,832 ****
               * one of them to be updated, so we deal with that within the
               * loop.)
               */
!             if (nvalues == 2)
                  have_end = get_actual_variable_range(root,
                                                       vardata,
!                                                      hist_op,
!                                                      &values[0],
!                                                      &values[1]);

              while (lobound < hibound)
              {
--- 793,804 ----
               * one of them to be updated, so we deal with that within the
               * loop.)
               */
!             if (sslot.nvalues == 2)
                  have_end = get_actual_variable_range(root,
                                                       vardata,
!                                                      sslot.staop,
!                                                      &sslot.values[0],
!                                                      &sslot.values[1]);

              while (lobound < hibound)
              {
*************** ineq_histogram_selectivity(PlannerInfo *
*** 838,859 ****
                   * histogram entry, first try to replace it with the actual
                   * current min or max (unless we already did so above).
                   */
!                 if (probe == 0 && nvalues > 2)
                      have_end = get_actual_variable_range(root,
                                                           vardata,
!                                                          hist_op,
!                                                          &values[0],
                                                           NULL);
!                 else if (probe == nvalues - 1 && nvalues > 2)
                      have_end = get_actual_variable_range(root,
                                                           vardata,
!                                                          hist_op,
                                                           NULL,
!                                                          &values[probe]);

                  ltcmp = DatumGetBool(FunctionCall2Coll(opproc,
                                                         DEFAULT_COLLATION_OID,
!                                                        values[probe],
                                                         constval));
                  if (isgt)
                      ltcmp = !ltcmp;
--- 810,831 ----
                   * histogram entry, first try to replace it with the actual
                   * current min or max (unless we already did so above).
                   */
!                 if (probe == 0 && sslot.nvalues > 2)
                      have_end = get_actual_variable_range(root,
                                                           vardata,
!                                                          sslot.staop,
!                                                          &sslot.values[0],
                                                           NULL);
!                 else if (probe == sslot.nvalues - 1 && sslot.nvalues > 2)
                      have_end = get_actual_variable_range(root,
                                                           vardata,
!                                                          sslot.staop,
                                                           NULL,
!                                                        &sslot.values[probe]);

                  ltcmp = DatumGetBool(FunctionCall2Coll(opproc,
                                                         DEFAULT_COLLATION_OID,
!                                                        sslot.values[probe],
                                                         constval));
                  if (isgt)
                      ltcmp = !ltcmp;
*************** ineq_histogram_selectivity(PlannerInfo *
*** 868,874 ****
                  /* Constant is below lower histogram boundary. */
                  histfrac = 0.0;
              }
!             else if (lobound >= nvalues)
              {
                  /* Constant is above upper histogram boundary. */
                  histfrac = 1.0;
--- 840,846 ----
                  /* Constant is below lower histogram boundary. */
                  histfrac = 0.0;
              }
!             else if (lobound >= sslot.nvalues)
              {
                  /* Constant is above upper histogram boundary. */
                  histfrac = 1.0;
*************** ineq_histogram_selectivity(PlannerInfo *
*** 889,895 ****
                   * interpolation within this bin.
                   */
                  if (convert_to_scalar(constval, consttype, &val,
!                                       values[i - 1], values[i],
                                        vardata->vartype,
                                        &low, &high))
                  {
--- 861,867 ----
                   * interpolation within this bin.
                   */
                  if (convert_to_scalar(constval, consttype, &val,
!                                       sslot.values[i - 1], sslot.values[i],
                                        vardata->vartype,
                                        &low, &high))
                  {
*************** ineq_histogram_selectivity(PlannerInfo *
*** 936,942 ****
                   * binfrac partial bin below the constant.
                   */
                  histfrac = (double) (i - 1) + binfrac;
!                 histfrac /= (double) (nvalues - 1);
              }

              /*
--- 908,914 ----
                   * binfrac partial bin below the constant.
                   */
                  histfrac = (double) (i - 1) + binfrac;
!                 histfrac /= (double) (sslot.nvalues - 1);
              }

              /*
*************** ineq_histogram_selectivity(PlannerInfo *
*** 964,970 ****
              }
          }

!         free_attstatsslot(vardata->atttype, values, nvalues, NULL, 0);
      }

      return hist_selec;
--- 936,942 ----
              }
          }

!         free_attstatsslot(&sslot);
      }

      return hist_selec;
*************** booltestsel(PlannerInfo *root, BoolTestT
*** 1517,1537 ****
      {
          Form_pg_statistic stats;
          double        freq_null;
!         Datum       *values;
!         int            nvalues;
!         float4       *numbers;
!         int            nnumbers;

          stats = (Form_pg_statistic) GETSTRUCT(vardata.statsTuple);
          freq_null = stats->stanullfrac;

!         if (get_attstatsslot(vardata.statsTuple,
!                              vardata.atttype, vardata.atttypmod,
                               STATISTIC_KIND_MCV, InvalidOid,
!                              NULL,
!                              &values, &nvalues,
!                              &numbers, &nnumbers)
!             && nnumbers > 0)
          {
              double        freq_true;
              double        freq_false;
--- 1489,1503 ----
      {
          Form_pg_statistic stats;
          double        freq_null;
!         AttStatsSlot sslot;

          stats = (Form_pg_statistic) GETSTRUCT(vardata.statsTuple);
          freq_null = stats->stanullfrac;

!         if (get_attstatsslot(&sslot, vardata.statsTuple,
                               STATISTIC_KIND_MCV, InvalidOid,
!                              ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS)
!             && sslot.nnumbers > 0)
          {
              double        freq_true;
              double        freq_false;
*************** booltestsel(PlannerInfo *root, BoolTestT
*** 1539,1548 ****
              /*
               * Get first MCV frequency and derive frequency for true.
               */
!             if (DatumGetBool(values[0]))
!                 freq_true = numbers[0];
              else
!                 freq_true = 1.0 - numbers[0] - freq_null;

              /*
               * Next derive frequency for false. Then use these as appropriate
--- 1505,1514 ----
              /*
               * Get first MCV frequency and derive frequency for true.
               */
!             if (DatumGetBool(sslot.values[0]))
!                 freq_true = sslot.numbers[0];
              else
!                 freq_true = 1.0 - sslot.numbers[0] - freq_null;

              /*
               * Next derive frequency for false. Then use these as appropriate
*************** booltestsel(PlannerInfo *root, BoolTestT
*** 1583,1590 ****
                      break;
              }

!             free_attstatsslot(vardata.atttype, values, nvalues,
!                               numbers, nnumbers);
          }
          else
          {
--- 1549,1555 ----
                      break;
              }

!             free_attstatsslot(&sslot);
          }
          else
          {
*************** eqjoinsel_inner(Oid operator,
*** 2274,2307 ****
      Form_pg_statistic stats1 = NULL;
      Form_pg_statistic stats2 = NULL;
      bool        have_mcvs1 = false;
-     Datum       *values1 = NULL;
-     int            nvalues1 = 0;
-     float4       *numbers1 = NULL;
-     int            nnumbers1 = 0;
      bool        have_mcvs2 = false;
!     Datum       *values2 = NULL;
!     int            nvalues2 = 0;
!     float4       *numbers2 = NULL;
!     int            nnumbers2 = 0;

      nd1 = get_variable_numdistinct(vardata1, &isdefault1);
      nd2 = get_variable_numdistinct(vardata2, &isdefault2);

      opfuncoid = get_opcode(operator);

      if (HeapTupleIsValid(vardata1->statsTuple))
      {
          /* note we allow use of nullfrac regardless of security check */
          stats1 = (Form_pg_statistic) GETSTRUCT(vardata1->statsTuple);
          if (statistic_proc_security_check(vardata1, opfuncoid))
!             have_mcvs1 = get_attstatsslot(vardata1->statsTuple,
!                                           vardata1->atttype,
!                                           vardata1->atttypmod,
!                                           STATISTIC_KIND_MCV,
!                                           InvalidOid,
!                                           NULL,
!                                           &values1, &nvalues1,
!                                           &numbers1, &nnumbers1);
      }

      if (HeapTupleIsValid(vardata2->statsTuple))
--- 2239,2264 ----
      Form_pg_statistic stats1 = NULL;
      Form_pg_statistic stats2 = NULL;
      bool        have_mcvs1 = false;
      bool        have_mcvs2 = false;
!     AttStatsSlot sslot1;
!     AttStatsSlot sslot2;

      nd1 = get_variable_numdistinct(vardata1, &isdefault1);
      nd2 = get_variable_numdistinct(vardata2, &isdefault2);

      opfuncoid = get_opcode(operator);

+     memset(&sslot1, 0, sizeof(sslot1));
+     memset(&sslot2, 0, sizeof(sslot2));
+
      if (HeapTupleIsValid(vardata1->statsTuple))
      {
          /* note we allow use of nullfrac regardless of security check */
          stats1 = (Form_pg_statistic) GETSTRUCT(vardata1->statsTuple);
          if (statistic_proc_security_check(vardata1, opfuncoid))
!             have_mcvs1 = get_attstatsslot(&sslot1, vardata1->statsTuple,
!                                           STATISTIC_KIND_MCV, InvalidOid,
!                                  ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS);
      }

      if (HeapTupleIsValid(vardata2->statsTuple))
*************** eqjoinsel_inner(Oid operator,
*** 2309,2322 ****
          /* note we allow use of nullfrac regardless of security check */
          stats2 = (Form_pg_statistic) GETSTRUCT(vardata2->statsTuple);
          if (statistic_proc_security_check(vardata2, opfuncoid))
!             have_mcvs2 = get_attstatsslot(vardata2->statsTuple,
!                                           vardata2->atttype,
!                                           vardata2->atttypmod,
!                                           STATISTIC_KIND_MCV,
!                                           InvalidOid,
!                                           NULL,
!                                           &values2, &nvalues2,
!                                           &numbers2, &nnumbers2);
      }

      if (have_mcvs1 && have_mcvs2)
--- 2266,2274 ----
          /* note we allow use of nullfrac regardless of security check */
          stats2 = (Form_pg_statistic) GETSTRUCT(vardata2->statsTuple);
          if (statistic_proc_security_check(vardata2, opfuncoid))
!             have_mcvs2 = get_attstatsslot(&sslot2, vardata2->statsTuple,
!                                           STATISTIC_KIND_MCV, InvalidOid,
!                                  ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS);
      }

      if (have_mcvs1 && have_mcvs2)
*************** eqjoinsel_inner(Oid operator,
*** 2351,2358 ****
                      nmatches;

          fmgr_info(opfuncoid, &eqproc);
!         hasmatch1 = (bool *) palloc0(nvalues1 * sizeof(bool));
!         hasmatch2 = (bool *) palloc0(nvalues2 * sizeof(bool));

          /*
           * Note we assume that each MCV will match at most one member of the
--- 2303,2310 ----
                      nmatches;

          fmgr_info(opfuncoid, &eqproc);
!         hasmatch1 = (bool *) palloc0(sslot1.nvalues * sizeof(bool));
!         hasmatch2 = (bool *) palloc0(sslot2.nvalues * sizeof(bool));

          /*
           * Note we assume that each MCV will match at most one member of the
*************** eqjoinsel_inner(Oid operator,
*** 2362,2382 ****
           */
          matchprodfreq = 0.0;
          nmatches = 0;
!         for (i = 0; i < nvalues1; i++)
          {
              int            j;

!             for (j = 0; j < nvalues2; j++)
              {
                  if (hasmatch2[j])
                      continue;
                  if (DatumGetBool(FunctionCall2Coll(&eqproc,
                                                     DEFAULT_COLLATION_OID,
!                                                    values1[i],
!                                                    values2[j])))
                  {
                      hasmatch1[i] = hasmatch2[j] = true;
!                     matchprodfreq += numbers1[i] * numbers2[j];
                      nmatches++;
                      break;
                  }
--- 2314,2334 ----
           */
          matchprodfreq = 0.0;
          nmatches = 0;
!         for (i = 0; i < sslot1.nvalues; i++)
          {
              int            j;

!             for (j = 0; j < sslot2.nvalues; j++)
              {
                  if (hasmatch2[j])
                      continue;
                  if (DatumGetBool(FunctionCall2Coll(&eqproc,
                                                     DEFAULT_COLLATION_OID,
!                                                    sslot1.values[i],
!                                                    sslot2.values[j])))
                  {
                      hasmatch1[i] = hasmatch2[j] = true;
!                     matchprodfreq += sslot1.numbers[i] * sslot2.numbers[j];
                      nmatches++;
                      break;
                  }
*************** eqjoinsel_inner(Oid operator,
*** 2385,2406 ****
          CLAMP_PROBABILITY(matchprodfreq);
          /* Sum up frequencies of matched and unmatched MCVs */
          matchfreq1 = unmatchfreq1 = 0.0;
!         for (i = 0; i < nvalues1; i++)
          {
              if (hasmatch1[i])
!                 matchfreq1 += numbers1[i];
              else
!                 unmatchfreq1 += numbers1[i];
          }
          CLAMP_PROBABILITY(matchfreq1);
          CLAMP_PROBABILITY(unmatchfreq1);
          matchfreq2 = unmatchfreq2 = 0.0;
!         for (i = 0; i < nvalues2; i++)
          {
              if (hasmatch2[i])
!                 matchfreq2 += numbers2[i];
              else
!                 unmatchfreq2 += numbers2[i];
          }
          CLAMP_PROBABILITY(matchfreq2);
          CLAMP_PROBABILITY(unmatchfreq2);
--- 2337,2358 ----
          CLAMP_PROBABILITY(matchprodfreq);
          /* Sum up frequencies of matched and unmatched MCVs */
          matchfreq1 = unmatchfreq1 = 0.0;
!         for (i = 0; i < sslot1.nvalues; i++)
          {
              if (hasmatch1[i])
!                 matchfreq1 += sslot1.numbers[i];
              else
!                 unmatchfreq1 += sslot1.numbers[i];
          }
          CLAMP_PROBABILITY(matchfreq1);
          CLAMP_PROBABILITY(unmatchfreq1);
          matchfreq2 = unmatchfreq2 = 0.0;
!         for (i = 0; i < sslot2.nvalues; i++)
          {
              if (hasmatch2[i])
!                 matchfreq2 += sslot2.numbers[i];
              else
!                 unmatchfreq2 += sslot2.numbers[i];
          }
          CLAMP_PROBABILITY(matchfreq2);
          CLAMP_PROBABILITY(unmatchfreq2);
*************** eqjoinsel_inner(Oid operator,
*** 2425,2439 ****
           * MCVs plus non-MCV values.
           */
          totalsel1 = matchprodfreq;
!         if (nd2 > nvalues2)
!             totalsel1 += unmatchfreq1 * otherfreq2 / (nd2 - nvalues2);
          if (nd2 > nmatches)
              totalsel1 += otherfreq1 * (otherfreq2 + unmatchfreq2) /
                  (nd2 - nmatches);
          /* Same estimate from the point of view of relation 2. */
          totalsel2 = matchprodfreq;
!         if (nd1 > nvalues1)
!             totalsel2 += unmatchfreq2 * otherfreq1 / (nd1 - nvalues1);
          if (nd1 > nmatches)
              totalsel2 += otherfreq2 * (otherfreq1 + unmatchfreq1) /
                  (nd1 - nmatches);
--- 2377,2391 ----
           * MCVs plus non-MCV values.
           */
          totalsel1 = matchprodfreq;
!         if (nd2 > sslot2.nvalues)
!             totalsel1 += unmatchfreq1 * otherfreq2 / (nd2 - sslot2.nvalues);
          if (nd2 > nmatches)
              totalsel1 += otherfreq1 * (otherfreq2 + unmatchfreq2) /
                  (nd2 - nmatches);
          /* Same estimate from the point of view of relation 2. */
          totalsel2 = matchprodfreq;
!         if (nd1 > sslot1.nvalues)
!             totalsel2 += unmatchfreq2 * otherfreq1 / (nd1 - sslot1.nvalues);
          if (nd1 > nmatches)
              totalsel2 += otherfreq2 * (otherfreq1 + unmatchfreq1) /
                  (nd1 - nmatches);
*************** eqjoinsel_inner(Oid operator,
*** 2478,2489 ****
              selec /= nd2;
      }

!     if (have_mcvs1)
!         free_attstatsslot(vardata1->atttype, values1, nvalues1,
!                           numbers1, nnumbers1);
!     if (have_mcvs2)
!         free_attstatsslot(vardata2->atttype, values2, nvalues2,
!                           numbers2, nnumbers2);

      return selec;
  }
--- 2430,2437 ----
              selec /= nd2;
      }

!     free_attstatsslot(&sslot1);
!     free_attstatsslot(&sslot2);

      return selec;
  }
*************** eqjoinsel_semi(Oid operator,
*** 2508,2528 ****
      Oid            opfuncoid;
      Form_pg_statistic stats1 = NULL;
      bool        have_mcvs1 = false;
-     Datum       *values1 = NULL;
-     int            nvalues1 = 0;
-     float4       *numbers1 = NULL;
-     int            nnumbers1 = 0;
      bool        have_mcvs2 = false;
!     Datum       *values2 = NULL;
!     int            nvalues2 = 0;
!     float4       *numbers2 = NULL;
!     int            nnumbers2 = 0;

      nd1 = get_variable_numdistinct(vardata1, &isdefault1);
      nd2 = get_variable_numdistinct(vardata2, &isdefault2);

      opfuncoid = OidIsValid(operator) ? get_opcode(operator) : InvalidOid;

      /*
       * We clamp nd2 to be not more than what we estimate the inner relation's
       * size to be.  This is intuitively somewhat reasonable since obviously
--- 2456,2473 ----
      Oid            opfuncoid;
      Form_pg_statistic stats1 = NULL;
      bool        have_mcvs1 = false;
      bool        have_mcvs2 = false;
!     AttStatsSlot sslot1;
!     AttStatsSlot sslot2;

      nd1 = get_variable_numdistinct(vardata1, &isdefault1);
      nd2 = get_variable_numdistinct(vardata2, &isdefault2);

      opfuncoid = OidIsValid(operator) ? get_opcode(operator) : InvalidOid;

+     memset(&sslot1, 0, sizeof(sslot1));
+     memset(&sslot2, 0, sizeof(sslot2));
+
      /*
       * We clamp nd2 to be not more than what we estimate the inner relation's
       * size to be.  This is intuitively somewhat reasonable since obviously
*************** eqjoinsel_semi(Oid operator,
*** 2561,2587 ****
          /* note we allow use of nullfrac regardless of security check */
          stats1 = (Form_pg_statistic) GETSTRUCT(vardata1->statsTuple);
          if (statistic_proc_security_check(vardata1, opfuncoid))
!             have_mcvs1 = get_attstatsslot(vardata1->statsTuple,
!                                           vardata1->atttype,
!                                           vardata1->atttypmod,
!                                           STATISTIC_KIND_MCV,
!                                           InvalidOid,
!                                           NULL,
!                                           &values1, &nvalues1,
!                                           &numbers1, &nnumbers1);
      }

      if (HeapTupleIsValid(vardata2->statsTuple) &&
          statistic_proc_security_check(vardata2, opfuncoid))
      {
!         have_mcvs2 = get_attstatsslot(vardata2->statsTuple,
!                                       vardata2->atttype,
!                                       vardata2->atttypmod,
!                                       STATISTIC_KIND_MCV,
!                                       InvalidOid,
!                                       NULL,
!                                       &values2, &nvalues2,
!                                       &numbers2, &nnumbers2);
      }

      if (have_mcvs1 && have_mcvs2 && OidIsValid(operator))
--- 2506,2523 ----
          /* note we allow use of nullfrac regardless of security check */
          stats1 = (Form_pg_statistic) GETSTRUCT(vardata1->statsTuple);
          if (statistic_proc_security_check(vardata1, opfuncoid))
!             have_mcvs1 = get_attstatsslot(&sslot1, vardata1->statsTuple,
!                                           STATISTIC_KIND_MCV, InvalidOid,
!                                  ATTSTATSSLOT_VALUES | ATTSTATSSLOT_NUMBERS);
      }

      if (HeapTupleIsValid(vardata2->statsTuple) &&
          statistic_proc_security_check(vardata2, opfuncoid))
      {
!         have_mcvs2 = get_attstatsslot(&sslot2, vardata2->statsTuple,
!                                       STATISTIC_KIND_MCV, InvalidOid,
!                                       ATTSTATSSLOT_VALUES);
!         /* note: currently don't need stanumbers from RHS */
      }

      if (have_mcvs1 && have_mcvs2 && OidIsValid(operator))
*************** eqjoinsel_semi(Oid operator,
*** 2607,2621 ****

          /*
           * The clamping above could have resulted in nd2 being less than
!          * nvalues2; in which case, we assume that precisely the nd2 most
!          * common values in the relation will appear in the join input, and so
!          * compare to only the first nd2 members of the MCV list.  Of course
!          * this is frequently wrong, but it's the best bet we can make.
           */
!         clamped_nvalues2 = Min(nvalues2, nd2);

          fmgr_info(opfuncoid, &eqproc);
!         hasmatch1 = (bool *) palloc0(nvalues1 * sizeof(bool));
          hasmatch2 = (bool *) palloc0(clamped_nvalues2 * sizeof(bool));

          /*
--- 2543,2557 ----

          /*
           * The clamping above could have resulted in nd2 being less than
!          * sslot2.nvalues; in which case, we assume that precisely the nd2
!          * most common values in the relation will appear in the join input,
!          * and so compare to only the first nd2 members of the MCV list.  Of
!          * course this is frequently wrong, but it's the best bet we can make.
           */
!         clamped_nvalues2 = Min(sslot2.nvalues, nd2);

          fmgr_info(opfuncoid, &eqproc);
!         hasmatch1 = (bool *) palloc0(sslot1.nvalues * sizeof(bool));
          hasmatch2 = (bool *) palloc0(clamped_nvalues2 * sizeof(bool));

          /*
*************** eqjoinsel_semi(Oid operator,
*** 2625,2631 ****
           * and because the math wouldn't add up...
           */
          nmatches = 0;
!         for (i = 0; i < nvalues1; i++)
          {
              int            j;

--- 2561,2567 ----
           * and because the math wouldn't add up...
           */
          nmatches = 0;
!         for (i = 0; i < sslot1.nvalues; i++)
          {
              int            j;

*************** eqjoinsel_semi(Oid operator,
*** 2635,2642 ****
                      continue;
                  if (DatumGetBool(FunctionCall2Coll(&eqproc,
                                                     DEFAULT_COLLATION_OID,
!                                                    values1[i],
!                                                    values2[j])))
                  {
                      hasmatch1[i] = hasmatch2[j] = true;
                      nmatches++;
--- 2571,2578 ----
                      continue;
                  if (DatumGetBool(FunctionCall2Coll(&eqproc,
                                                     DEFAULT_COLLATION_OID,
!                                                    sslot1.values[i],
!                                                    sslot2.values[j])))
                  {
                      hasmatch1[i] = hasmatch2[j] = true;
                      nmatches++;
*************** eqjoinsel_semi(Oid operator,
*** 2646,2655 ****
          }
          /* Sum up frequencies of matched MCVs */
          matchfreq1 = 0.0;
!         for (i = 0; i < nvalues1; i++)
          {
              if (hasmatch1[i])
!                 matchfreq1 += numbers1[i];
          }
          CLAMP_PROBABILITY(matchfreq1);
          pfree(hasmatch1);
--- 2582,2591 ----
          }
          /* Sum up frequencies of matched MCVs */
          matchfreq1 = 0.0;
!         for (i = 0; i < sslot1.nvalues; i++)
          {
              if (hasmatch1[i])
!                 matchfreq1 += sslot1.numbers[i];
          }
          CLAMP_PROBABILITY(matchfreq1);
          pfree(hasmatch1);
*************** eqjoinsel_semi(Oid operator,
*** 2704,2715 ****
              selec = 0.5 * (1.0 - nullfrac1);
      }

!     if (have_mcvs1)
!         free_attstatsslot(vardata1->atttype, values1, nvalues1,
!                           numbers1, nnumbers1);
!     if (have_mcvs2)
!         free_attstatsslot(vardata2->atttype, values2, nvalues2,
!                           numbers2, nnumbers2);

      return selec;
  }
--- 2640,2647 ----
              selec = 0.5 * (1.0 - nullfrac1);
      }

!     free_attstatsslot(&sslot1);
!     free_attstatsslot(&sslot2);

      return selec;
  }
*************** estimate_hash_bucketsize(PlannerInfo *ro
*** 3629,3636 ****
                  mcvfreq,
                  avgfreq;
      bool        isdefault;
!     float4       *numbers;
!     int            nnumbers;

      examine_variable(root, hashkey, 0, &vardata);

--- 3561,3567 ----
                  mcvfreq,
                  avgfreq;
      bool        isdefault;
!     AttStatsSlot sslot;

      examine_variable(root, hashkey, 0, &vardata);

*************** estimate_hash_bucketsize(PlannerInfo *ro
*** 3689,3708 ****

      if (HeapTupleIsValid(vardata.statsTuple))
      {
!         if (get_attstatsslot(vardata.statsTuple,
!                              vardata.atttype, vardata.atttypmod,
                               STATISTIC_KIND_MCV, InvalidOid,
!                              NULL,
!                              NULL, NULL,
!                              &numbers, &nnumbers))
          {
              /*
               * The first MCV stat is for the most common value.
               */
!             if (nnumbers > 0)
!                 mcvfreq = numbers[0];
!             free_attstatsslot(vardata.atttype, NULL, 0,
!                               numbers, nnumbers);
          }
      }

--- 3620,3635 ----

      if (HeapTupleIsValid(vardata.statsTuple))
      {
!         if (get_attstatsslot(&sslot, vardata.statsTuple,
                               STATISTIC_KIND_MCV, InvalidOid,
!                              ATTSTATSSLOT_NUMBERS))
          {
              /*
               * The first MCV stat is for the most common value.
               */
!             if (sslot.nnumbers > 0)
!                 mcvfreq = sslot.numbers[0];
!             free_attstatsslot(&sslot);
          }
      }

*************** get_join_variables(PlannerInfo *root, Li
*** 4572,4578 ****
   *    freefunc: pointer to a function to release statsTuple with.
   *    vartype: exposed type of the expression; this should always match
   *        the declared input type of the operator we are estimating for.
!  *    atttype, atttypmod: type data to pass to get_attstatsslot().  This is
   *        commonly the same as the exposed type of the variable argument,
   *        but can be different in binary-compatible-type cases.
   *    isunique: TRUE if we were able to match the var to a unique index or a
--- 4499,4505 ----
   *    freefunc: pointer to a function to release statsTuple with.
   *    vartype: exposed type of the expression; this should always match
   *        the declared input type of the operator we are estimating for.
!  *    atttype, atttypmod: actual type/typmod of the "var" expression.  This is
   *        commonly the same as the exposed type of the variable argument,
   *        but can be different in binary-compatible-type cases.
   *    isunique: TRUE if we were able to match the var to a unique index or a
*************** get_variable_range(PlannerInfo *root, Va
*** 5125,5132 ****
      int16        typLen;
      bool        typByVal;
      Oid            opfuncoid;
!     Datum       *values;
!     int            nvalues;
      int            i;

      /*
--- 5052,5058 ----
      int16        typLen;
      bool        typByVal;
      Oid            opfuncoid;
!     AttStatsSlot sslot;
      int            i;

      /*
*************** get_variable_range(PlannerInfo *root, Va
*** 5167,5195 ****
       * the one we want, fail --- this suggests that there is data we can't
       * use.
       */
!     if (get_attstatsslot(vardata->statsTuple,
!                          vardata->atttype, vardata->atttypmod,
                           STATISTIC_KIND_HISTOGRAM, sortop,
!                          NULL,
!                          &values, &nvalues,
!                          NULL, NULL))
      {
!         if (nvalues > 0)
          {
!             tmin = datumCopy(values[0], typByVal, typLen);
!             tmax = datumCopy(values[nvalues - 1], typByVal, typLen);
              have_data = true;
          }
!         free_attstatsslot(vardata->atttype, values, nvalues, NULL, 0);
      }
!     else if (get_attstatsslot(vardata->statsTuple,
!                               vardata->atttype, vardata->atttypmod,
                                STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                               NULL,
!                               &values, &nvalues,
!                               NULL, NULL))
      {
!         free_attstatsslot(vardata->atttype, values, nvalues, NULL, 0);
          return false;
      }

--- 5093,5115 ----
       * the one we want, fail --- this suggests that there is data we can't
       * use.
       */
!     if (get_attstatsslot(&sslot, vardata->statsTuple,
                           STATISTIC_KIND_HISTOGRAM, sortop,
!                          ATTSTATSSLOT_VALUES))
      {
!         if (sslot.nvalues > 0)
          {
!             tmin = datumCopy(sslot.values[0], typByVal, typLen);
!             tmax = datumCopy(sslot.values[sslot.nvalues - 1], typByVal, typLen);
              have_data = true;
          }
!         free_attstatsslot(&sslot);
      }
!     else if (get_attstatsslot(&sslot, vardata->statsTuple,
                                STATISTIC_KIND_HISTOGRAM, InvalidOid,
!                               0))
      {
!         free_attstatsslot(&sslot);
          return false;
      }

*************** get_variable_range(PlannerInfo *root, Va
*** 5199,5210 ****
       * the MCVs.  However, usually the MCVs will not be the extreme values, so
       * avoid unnecessary data copying.
       */
!     if (get_attstatsslot(vardata->statsTuple,
!                          vardata->atttype, vardata->atttypmod,
                           STATISTIC_KIND_MCV, InvalidOid,
!                          NULL,
!                          &values, &nvalues,
!                          NULL, NULL))
      {
          bool        tmin_is_mcv = false;
          bool        tmax_is_mcv = false;
--- 5119,5127 ----
       * the MCVs.  However, usually the MCVs will not be the extreme values, so
       * avoid unnecessary data copying.
       */
!     if (get_attstatsslot(&sslot, vardata->statsTuple,
                           STATISTIC_KIND_MCV, InvalidOid,
!                          ATTSTATSSLOT_VALUES))
      {
          bool        tmin_is_mcv = false;
          bool        tmax_is_mcv = false;
*************** get_variable_range(PlannerInfo *root, Va
*** 5212,5237 ****

          fmgr_info(opfuncoid, &opproc);

!         for (i = 0; i < nvalues; i++)
          {
              if (!have_data)
              {
!                 tmin = tmax = values[i];
                  tmin_is_mcv = tmax_is_mcv = have_data = true;
                  continue;
              }
              if (DatumGetBool(FunctionCall2Coll(&opproc,
                                                 DEFAULT_COLLATION_OID,
!                                                values[i], tmin)))
              {
!                 tmin = values[i];
                  tmin_is_mcv = true;
              }
              if (DatumGetBool(FunctionCall2Coll(&opproc,
                                                 DEFAULT_COLLATION_OID,
!                                                tmax, values[i])))
              {
!                 tmax = values[i];
                  tmax_is_mcv = true;
              }
          }
--- 5129,5154 ----

          fmgr_info(opfuncoid, &opproc);

!         for (i = 0; i < sslot.nvalues; i++)
          {
              if (!have_data)
              {
!                 tmin = tmax = sslot.values[i];
                  tmin_is_mcv = tmax_is_mcv = have_data = true;
                  continue;
              }
              if (DatumGetBool(FunctionCall2Coll(&opproc,
                                                 DEFAULT_COLLATION_OID,
!                                                sslot.values[i], tmin)))
              {
!                 tmin = sslot.values[i];
                  tmin_is_mcv = true;
              }
              if (DatumGetBool(FunctionCall2Coll(&opproc,
                                                 DEFAULT_COLLATION_OID,
!                                                tmax, sslot.values[i])))
              {
!                 tmax = sslot.values[i];
                  tmax_is_mcv = true;
              }
          }
*************** get_variable_range(PlannerInfo *root, Va
*** 5239,5245 ****
              tmin = datumCopy(tmin, typByVal, typLen);
          if (tmax_is_mcv)
              tmax = datumCopy(tmax, typByVal, typLen);
!         free_attstatsslot(vardata->atttype, values, nvalues, NULL, 0);
      }

      *min = tmin;
--- 5156,5162 ----
              tmin = datumCopy(tmin, typByVal, typLen);
          if (tmax_is_mcv)
              tmax = datumCopy(tmax, typByVal, typLen);
!         free_attstatsslot(&sslot);
      }

      *min = tmin;
*************** btcostestimate(PlannerInfo *root, IndexP
*** 6979,7003 ****
      if (HeapTupleIsValid(vardata.statsTuple))
      {
          Oid            sortop;
!         float4       *numbers;
!         int            nnumbers;

          sortop = get_opfamily_member(index->opfamily[0],
                                       index->opcintype[0],
                                       index->opcintype[0],
                                       BTLessStrategyNumber);
          if (OidIsValid(sortop) &&
!             get_attstatsslot(vardata.statsTuple, InvalidOid, 0,
!                              STATISTIC_KIND_CORRELATION,
!                              sortop,
!                              NULL,
!                              NULL, NULL,
!                              &numbers, &nnumbers))
          {
              double        varCorrelation;

!             Assert(nnumbers == 1);
!             varCorrelation = numbers[0];

              if (index->reverse_sort[0])
                  varCorrelation = -varCorrelation;
--- 6896,6916 ----
      if (HeapTupleIsValid(vardata.statsTuple))
      {
          Oid            sortop;
!         AttStatsSlot sslot;

          sortop = get_opfamily_member(index->opfamily[0],
                                       index->opcintype[0],
                                       index->opcintype[0],
                                       BTLessStrategyNumber);
          if (OidIsValid(sortop) &&
!             get_attstatsslot(&sslot, vardata.statsTuple,
!                              STATISTIC_KIND_CORRELATION, sortop,
!                              ATTSTATSSLOT_NUMBERS))
          {
              double        varCorrelation;

!             Assert(sslot.nnumbers == 1);
!             varCorrelation = sslot.numbers[0];

              if (index->reverse_sort[0])
                  varCorrelation = -varCorrelation;
*************** btcostestimate(PlannerInfo *root, IndexP
*** 7007,7013 ****
              else
                  costs.indexCorrelation = varCorrelation;

!             free_attstatsslot(InvalidOid, NULL, 0, numbers, nnumbers);
          }
      }

--- 6920,6926 ----
              else
                  costs.indexCorrelation = varCorrelation;

!             free_attstatsslot(&sslot);
          }
      }

*************** brincostestimate(PlannerInfo *root, Inde
*** 7920,7944 ****

          if (HeapTupleIsValid(vardata.statsTuple))
          {
!             float4       *numbers;
!             int            nnumbers;

!             if (get_attstatsslot(vardata.statsTuple, InvalidOid, 0,
!                                  STATISTIC_KIND_CORRELATION,
!                                  InvalidOid,
!                                  NULL,
!                                  NULL, NULL,
!                                  &numbers, &nnumbers))
              {
                  double        varCorrelation = 0.0;

!                 if (nnumbers > 0)
!                     varCorrelation = Abs(numbers[0]);

                  if (varCorrelation > *indexCorrelation)
                      *indexCorrelation = varCorrelation;

!                 free_attstatsslot(InvalidOid, NULL, 0, numbers, nnumbers);
              }
          }

--- 7833,7853 ----

          if (HeapTupleIsValid(vardata.statsTuple))
          {
!             AttStatsSlot sslot;

!             if (get_attstatsslot(&sslot, vardata.statsTuple,
!                                  STATISTIC_KIND_CORRELATION, InvalidOid,
!                                  ATTSTATSSLOT_NUMBERS))
              {
                  double        varCorrelation = 0.0;

!                 if (sslot.nnumbers > 0)
!                     varCorrelation = Abs(sslot.numbers[0]);

                  if (varCorrelation > *indexCorrelation)
                      *indexCorrelation = varCorrelation;

!                 free_attstatsslot(&sslot);
              }
          }

diff --git a/src/backend/utils/cache/lsyscache.c b/src/backend/utils/cache/lsyscache.c
index 236d876..b94d475 100644
*** a/src/backend/utils/cache/lsyscache.c
--- b/src/backend/utils/cache/lsyscache.c
*************** get_attavgwidth(Oid relid, AttrNumber at
*** 2864,2898 ****
   * that have been provided by a stats hook and didn't really come from
   * pg_statistic.
   *
   * statstuple: pg_statistic tuple to be examined.
-  * atttype: type OID of slot's stavalues (can be InvalidOid if values == NULL).
-  * atttypmod: typmod of slot's stavalues (can be 0 if values == NULL).
   * reqkind: STAKIND code for desired statistics slot kind.
   * reqop: STAOP value wanted, or InvalidOid if don't care.
!  * actualop: if not NULL, *actualop receives the actual STAOP value.
!  * values, nvalues: if not NULL, the slot's stavalues are extracted.
!  * numbers, nnumbers: if not NULL, the slot's stanumbers are extracted.
   *
!  * If assigned, values and numbers are set to point to palloc'd arrays.
!  * If the stavalues datatype is pass-by-reference, the values referenced by
!  * the values array are themselves palloc'd.  The palloc'd stuff can be
!  * freed by calling free_attstatsslot.
   *
!  * Note: at present, atttype/atttypmod aren't actually used here at all.
!  * But the caller must have the correct (or at least binary-compatible)
!  * type ID to pass to free_attstatsslot later.
   */
  bool
! get_attstatsslot(HeapTuple statstuple,
!                  Oid atttype, int32 atttypmod,
!                  int reqkind, Oid reqop,
!                  Oid *actualop,
!                  Datum **values, int *nvalues,
!                  float4 **numbers, int *nnumbers)
  {
      Form_pg_statistic stats = (Form_pg_statistic) GETSTRUCT(statstuple);
!     int            i,
!                 j;
      Datum        val;
      bool        isnull;
      ArrayType  *statarray;
--- 2864,2902 ----
   * that have been provided by a stats hook and didn't really come from
   * pg_statistic.
   *
+  * sslot: pointer to output area (typically, a local variable in the caller).
   * statstuple: pg_statistic tuple to be examined.
   * reqkind: STAKIND code for desired statistics slot kind.
   * reqop: STAOP value wanted, or InvalidOid if don't care.
!  * flags: bitmask of ATTSTATSSLOT_VALUES and/or ATTSTATSSLOT_NUMBERS.
   *
!  * If a matching slot is found, TRUE is returned, and *sslot is filled thus:
!  * staop: receives the actual STAOP value.
!  * valuetype: receives actual datatype of the elements of stavalues.
!  * values: receives pointer to an array of the slot's stavalues.
!  * nvalues: receives number of stavalues.
!  * numbers: receives pointer to an array of the slot's stanumbers (as float4).
!  * nnumbers: receives number of stanumbers.
   *
!  * valuetype/values/nvalues are InvalidOid/NULL/0 if ATTSTATSSLOT_VALUES
!  * wasn't specified.  Likewise, numbers/nnumbers are NULL/0 if
!  * ATTSTATSSLOT_NUMBERS wasn't specified.
!  *
!  * If no matching slot is found, FALSE is returned, and *sslot is zeroed.
!  *
!  * The data referred to by the fields of sslot is locally palloc'd and
!  * is independent of the original pg_statistic tuple.  When the caller
!  * is done with it, call free_attstatsslot to release the palloc'd data.
!  *
!  * If it's desirable to call free_attstatsslot when get_attstatsslot might
!  * not have been called, memset'ing sslot to zeroes will allow that.
   */
  bool
! get_attstatsslot(AttStatsSlot *sslot, HeapTuple statstuple,
!                  int reqkind, Oid reqop, int flags)
  {
      Form_pg_statistic stats = (Form_pg_statistic) GETSTRUCT(statstuple);
!     int            i;
      Datum        val;
      bool        isnull;
      ArrayType  *statarray;
*************** get_attstatsslot(HeapTuple statstuple,
*** 2901,2906 ****
--- 2905,2913 ----
      HeapTuple    typeTuple;
      Form_pg_type typeForm;

+     /* initialize *sslot properly */
+     memset(sslot, 0, sizeof(AttStatsSlot));
+
      for (i = 0; i < STATISTIC_NUM_SLOTS; i++)
      {
          if ((&stats->stakind1)[i] == reqkind &&
*************** get_attstatsslot(HeapTuple statstuple,
*** 2910,2935 ****
      if (i >= STATISTIC_NUM_SLOTS)
          return false;            /* not there */

!     if (actualop)
!         *actualop = (&stats->staop1)[i];

!     if (values)
      {
          val = SysCacheGetAttr(STATRELATTINH, statstuple,
                                Anum_pg_statistic_stavalues1 + i,
                                &isnull);
          if (isnull)
              elog(ERROR, "stavalues is null");
-         statarray = DatumGetArrayTypeP(val);

          /*
!          * Need to get info about the array element type.  We look at the
!          * actual element type embedded in the array, which might be only
!          * binary-compatible with the passed-in atttype.  The info we extract
!          * here should be the same either way, but deconstruct_array is picky
!          * about having an exact type OID match.
           */
!         arrayelemtype = ARR_ELEMTYPE(statarray);
          typeTuple = SearchSysCache1(TYPEOID, ObjectIdGetDatum(arrayelemtype));
          if (!HeapTupleIsValid(typeTuple))
              elog(ERROR, "cache lookup failed for type %u", arrayelemtype);
--- 2917,2945 ----
      if (i >= STATISTIC_NUM_SLOTS)
          return false;            /* not there */

!     sslot->staop = (&stats->staop1)[i];

!     if (flags & ATTSTATSSLOT_VALUES)
      {
          val = SysCacheGetAttr(STATRELATTINH, statstuple,
                                Anum_pg_statistic_stavalues1 + i,
                                &isnull);
          if (isnull)
              elog(ERROR, "stavalues is null");

          /*
!          * Detoast the array if needed, and in any case make a copy that's
!          * under control of this AttStatsSlot.
           */
!         statarray = DatumGetArrayTypePCopy(val);
!
!         /*
!          * Extract the actual array element type, and pass it back in case the
!          * caller needs it.
!          */
!         sslot->valuetype = arrayelemtype = ARR_ELEMTYPE(statarray);
!
!         /* Need info about element type */
          typeTuple = SearchSysCache1(TYPEOID, ObjectIdGetDatum(arrayelemtype));
          if (!HeapTupleIsValid(typeTuple))
              elog(ERROR, "cache lookup failed for type %u", arrayelemtype);
*************** get_attstatsslot(HeapTuple statstuple,
*** 2941,2980 ****
                            typeForm->typlen,
                            typeForm->typbyval,
                            typeForm->typalign,
!                           values, NULL, nvalues);

          /*
           * If the element type is pass-by-reference, we now have a bunch of
!          * Datums that are pointers into the syscache value.  Copy them to
!          * avoid problems if syscache decides to drop the entry.
           */
          if (!typeForm->typbyval)
!         {
!             for (j = 0; j < *nvalues; j++)
!             {
!                 (*values)[j] = datumCopy((*values)[j],
!                                          typeForm->typbyval,
!                                          typeForm->typlen);
!             }
!         }

          ReleaseSysCache(typeTuple);
-
-         /*
-          * Free statarray if it's a detoasted copy.
-          */
-         if ((Pointer) statarray != DatumGetPointer(val))
-             pfree(statarray);
      }

!     if (numbers)
      {
          val = SysCacheGetAttr(STATRELATTINH, statstuple,
                                Anum_pg_statistic_stanumbers1 + i,
                                &isnull);
          if (isnull)
              elog(ERROR, "stanumbers is null");
!         statarray = DatumGetArrayTypeP(val);

          /*
           * We expect the array to be a 1-D float4 array; verify that. We don't
--- 2951,2985 ----
                            typeForm->typlen,
                            typeForm->typbyval,
                            typeForm->typalign,
!                           &sslot->values, NULL, &sslot->nvalues);

          /*
           * If the element type is pass-by-reference, we now have a bunch of
!          * Datums that are pointers into the statarray, so we need to keep
!          * that until free_attstatsslot.  Otherwise, all the useful info is in
!          * sslot->values[], so we can free the array object immediately.
           */
          if (!typeForm->typbyval)
!             sslot->values_arr = statarray;
!         else
!             pfree(statarray);

          ReleaseSysCache(typeTuple);
      }

!     if (flags & ATTSTATSSLOT_NUMBERS)
      {
          val = SysCacheGetAttr(STATRELATTINH, statstuple,
                                Anum_pg_statistic_stanumbers1 + i,
                                &isnull);
          if (isnull)
              elog(ERROR, "stanumbers is null");
!
!         /*
!          * Detoast the array if needed, and in any case make a copy that's
!          * under control of this AttStatsSlot.
!          */
!         statarray = DatumGetArrayTypePCopy(val);

          /*
           * We expect the array to be a 1-D float4 array; verify that. We don't
*************** get_attstatsslot(HeapTuple statstuple,
*** 2986,3000 ****
              ARR_HASNULL(statarray) ||
              ARR_ELEMTYPE(statarray) != FLOAT4OID)
              elog(ERROR, "stanumbers is not a 1-D float4 array");
-         *numbers = (float4 *) palloc(narrayelem * sizeof(float4));
-         memcpy(*numbers, ARR_DATA_PTR(statarray), narrayelem * sizeof(float4));
-         *nnumbers = narrayelem;

!         /*
!          * Free statarray if it's a detoasted copy.
!          */
!         if ((Pointer) statarray != DatumGetPointer(val))
!             pfree(statarray);
      }

      return true;
--- 2991,3003 ----
              ARR_HASNULL(statarray) ||
              ARR_ELEMTYPE(statarray) != FLOAT4OID)
              elog(ERROR, "stanumbers is not a 1-D float4 array");

!         /* Give caller a pointer directly into the statarray */
!         sslot->numbers = (float4 *) ARR_DATA_PTR(statarray);
!         sslot->nnumbers = narrayelem;
!
!         /* We'll free the statarray in free_attstatsslot */
!         sslot->numbers_arr = statarray;
      }

      return true;
*************** get_attstatsslot(HeapTuple statstuple,
*** 3003,3030 ****
  /*
   * free_attstatsslot
   *        Free data allocated by get_attstatsslot
-  *
-  * atttype is the type of the individual values in values[].
-  * It need be valid only if values != NULL.
   */
  void
! free_attstatsslot(Oid atttype,
!                   Datum *values, int nvalues,
!                   float4 *numbers, int nnumbers)
  {
!     if (values)
!     {
!         if (!get_typbyval(atttype))
!         {
!             int            i;
!
!             for (i = 0; i < nvalues; i++)
!                 pfree(DatumGetPointer(values[i]));
!         }
!         pfree(values);
!     }
!     if (numbers)
!         pfree(numbers);
  }

  /*                ---------- PG_NAMESPACE CACHE ----------                 */
--- 3006,3024 ----
  /*
   * free_attstatsslot
   *        Free data allocated by get_attstatsslot
   */
  void
! free_attstatsslot(AttStatsSlot *sslot)
  {
!     /* The values[] array was separately palloc'd by deconstruct_array */
!     if (sslot->values)
!         pfree(sslot->values);
!     /* The numbers[] array points into numbers_arr, do not pfree it */
!     /* Free the detoasted array objects, if any */
!     if (sslot->values_arr)
!         pfree(sslot->values_arr);
!     if (sslot->numbers_arr)
!         pfree(sslot->numbers_arr);
  }

  /*                ---------- PG_NAMESPACE CACHE ----------                 */
diff --git a/src/include/utils/lsyscache.h b/src/include/utils/lsyscache.h
index 88629d9..a0d90e7 100644
*** a/src/include/utils/lsyscache.h
--- b/src/include/utils/lsyscache.h
*************** typedef enum IOFuncSelector
*** 35,40 ****
--- 35,62 ----
      IOFunc_send
  } IOFuncSelector;

+ /* Flag bits for get_attstatsslot */
+ #define ATTSTATSSLOT_VALUES        0x01
+ #define ATTSTATSSLOT_NUMBERS    0x02
+
+ /* Result struct for get_attstatsslot */
+ typedef struct AttStatsSlot
+ {
+     /* Always filled: */
+     Oid            staop;            /* Actual staop for the found slot */
+     /* Filled if ATTSTATSSLOT_VALUES is specified: */
+     Oid            valuetype;        /* Actual datatype of the values */
+     Datum       *values;            /* slot's "values" array, or NULL if none */
+     int            nvalues;        /* length of values[], or 0 */
+     /* Filled if ATTSTATSSLOT_NUMBERS is specified: */
+     float4       *numbers;        /* slot's "numbers" array, or NULL if none */
+     int            nnumbers;        /* length of numbers[], or 0 */
+
+     /* Remaining fields are private to get_attstatsslot/free_attstatsslot */
+     void       *values_arr;        /* palloc'd values array, if any */
+     void       *numbers_arr;    /* palloc'd numbers array, if any */
+ } AttStatsSlot;
+
  /* Hook for plugins to get control in get_attavgwidth() */
  typedef int32 (*get_attavgwidth_hook_type) (Oid relid, AttrNumber attnum);
  extern PGDLLIMPORT get_attavgwidth_hook_type get_attavgwidth_hook;
*************** extern Oid    getBaseType(Oid typid);
*** 148,162 ****
  extern Oid    getBaseTypeAndTypmod(Oid typid, int32 *typmod);
  extern int32 get_typavgwidth(Oid typid, int32 typmod);
  extern int32 get_attavgwidth(Oid relid, AttrNumber attnum);
! extern bool get_attstatsslot(HeapTuple statstuple,
!                  Oid atttype, int32 atttypmod,
!                  int reqkind, Oid reqop,
!                  Oid *actualop,
!                  Datum **values, int *nvalues,
!                  float4 **numbers, int *nnumbers);
! extern void free_attstatsslot(Oid atttype,
!                   Datum *values, int nvalues,
!                   float4 *numbers, int nnumbers);
  extern char *get_namespace_name(Oid nspid);
  extern char *get_namespace_name_or_temp(Oid nspid);
  extern Oid    get_range_subtype(Oid rangeOid);
--- 170,178 ----
  extern Oid    getBaseTypeAndTypmod(Oid typid, int32 *typmod);
  extern int32 get_typavgwidth(Oid typid, int32 typmod);
  extern int32 get_attavgwidth(Oid relid, AttrNumber attnum);
! extern bool get_attstatsslot(AttStatsSlot *sslot, HeapTuple statstuple,
!                  int reqkind, Oid reqop, int flags);
! extern void free_attstatsslot(AttStatsSlot *sslot);
  extern char *get_namespace_name(Oid nspid);
  extern char *get_namespace_name_or_temp(Oid nspid);
  extern Oid    get_range_subtype(Oid rangeOid);
diff --git a/src/include/utils/selfuncs.h b/src/include/utils/selfuncs.h
index b7617dd..958bdba 100644
*** a/src/include/utils/selfuncs.h
--- b/src/include/utils/selfuncs.h
*************** typedef struct VariableStatData
*** 72,79 ****
      /* NB: if statsTuple!=NULL, it must be freed when caller is done */
      void        (*freefunc) (HeapTuple tuple);    /* how to free statsTuple */
      Oid            vartype;        /* exposed type of expression */
!     Oid            atttype;        /* type to pass to get_attstatsslot */
!     int32        atttypmod;        /* typmod to pass to get_attstatsslot */
      bool        isunique;        /* matches unique index or DISTINCT clause */
      bool        acl_ok;            /* result of ACL check on table or column */
  } VariableStatData;
--- 72,79 ----
      /* NB: if statsTuple!=NULL, it must be freed when caller is done */
      void        (*freefunc) (HeapTuple tuple);    /* how to free statsTuple */
      Oid            vartype;        /* exposed type of expression */
!     Oid            atttype;        /* actual type (after stripping relabel) */
!     int32        atttypmod;        /* actual typmod (after stripping relabel) */
      bool        isunique;        /* matches unique index or DISTINCT clause */
      bool        acl_ok;            /* result of ACL check on table or column */
  } VariableStatData;

-- 
Sent via pgsql-hackers mailing list (pgsql-hackers@postgresql.org)
To make changes to your subscription:
http://www.postgresql.org/mailpref/pgsql-hackers

pgsql-hackers by date:

Previous
From: Heikki Linnakangas
Date:
Subject: Re: [HACKERS] eval_const_expresisions and ScalarArrayOpExpr
Next
From: Tom Lane
Date:
Subject: Re: [HACKERS] eval_const_expresisions and ScalarArrayOpExpr