Use a fresh copy of query_list when making a second plan in GetCachedPlan.
authorTom Lane <tgl@sss.pgh.pa.us>
Mon, 26 Sep 2011 16:44:17 +0000 (12:44 -0400)
committerTom Lane <tgl@sss.pgh.pa.us>
Mon, 26 Sep 2011 16:44:17 +0000 (12:44 -0400)
The code path that tried a generic plan, didn't like it, and then made a
custom plan was mistakenly passing the same copy of the query_list to the
planner both times.  This doesn't work too well for nontrivial queries,
since the planner tends to scribble on its input.  Diagnosis and fix by
Yamamoto Takashi.

src/backend/utils/cache/plancache.c

index cfeb8245b8c7fc12daa22e7c2ddcbd2a249bb40a..56dace0e89c9664897f88da73dc06d61f4965b8e 100644 (file)
@@ -697,7 +697,8 @@ CheckCachedPlan(CachedPlanSource *plansource)
 /*
  * BuildCachedPlan: construct a new CachedPlan from a CachedPlanSource.
  *
- * qlist should be the result value from a previous RevalidateCachedQuery.
+ * qlist should be the result value from a previous RevalidateCachedQuery,
+ * or it can be set to NIL if we need to re-copy the plansource's query_list.
  *
  * To build a generic, parameter-value-independent plan, pass NULL for
  * boundParams.  To build a custom plan, pass the actual parameter values via
@@ -980,6 +981,13 @@ GetCachedPlan(CachedPlanSource *plansource, ParamListInfo boundParams,
             * plan.
             */
            customplan = choose_custom_plan(plansource, boundParams);
+
+           /*
+            * If we choose to plan again, we need to re-copy the query_list,
+            * since the planner probably scribbled on it.  We can force
+            * BuildCachedPlan to do that by passing NIL.
+            */
+           qlist = NIL;
        }
    }