Commit da6a707c authored by Tom Lane's avatar Tom Lane

Fix pg_plan_queries() to restore the previous setting of ActiveSnapshot

(probably NULL) before exiting.  Up to now it's just left the variable as it
set it, which means that after we're done processing the current client
message, ActiveSnapshot is probably pointing at garbage (because this function
is typically run in MessageContext which will get reset).  There doesn't seem
to have been any code path in which that mattered before 8.3, but now the
plancache module might try to use the stale value if the next client message
is a Bind for a prepared statement that is in need of replanning.  Per report
from Alex Hunsaker.
parent 0d540b09
...@@ -8,7 +8,7 @@ ...@@ -8,7 +8,7 @@
* *
* *
* IDENTIFICATION * IDENTIFICATION
* $PostgreSQL: pgsql/src/backend/tcop/postgres.c,v 1.544 2008/03/10 12:55:13 mha Exp $ * $PostgreSQL: pgsql/src/backend/tcop/postgres.c,v 1.545 2008/03/12 23:58:27 tgl Exp $
* *
* NOTES * NOTES
* this is the "main" module of the postgres backend and * this is the "main" module of the postgres backend and
...@@ -730,31 +730,49 @@ List * ...@@ -730,31 +730,49 @@ List *
pg_plan_queries(List *querytrees, int cursorOptions, ParamListInfo boundParams, pg_plan_queries(List *querytrees, int cursorOptions, ParamListInfo boundParams,
bool needSnapshot) bool needSnapshot)
{ {
List *stmt_list = NIL; List * volatile stmt_list = NIL;
ListCell *query_list; Snapshot saveActiveSnapshot = ActiveSnapshot;
foreach(query_list, querytrees) /* PG_TRY to ensure previous ActiveSnapshot is restored on error */
PG_TRY();
{ {
Query *query = (Query *) lfirst(query_list); Snapshot mySnapshot = NULL;
Node *stmt; ListCell *query_list;
if (query->commandType == CMD_UTILITY) foreach(query_list, querytrees)
{ {
/* Utility commands have no plans. */ Query *query = (Query *) lfirst(query_list);
stmt = query->utilityStmt; Node *stmt;
}
else if (query->commandType == CMD_UTILITY)
{ {
if (needSnapshot) /* Utility commands have no plans. */
stmt = query->utilityStmt;
}
else
{ {
ActiveSnapshot = CopySnapshot(GetTransactionSnapshot()); if (needSnapshot && mySnapshot == NULL)
needSnapshot = false; {
mySnapshot = CopySnapshot(GetTransactionSnapshot());
ActiveSnapshot = mySnapshot;
}
stmt = (Node *) pg_plan_query(query, cursorOptions,
boundParams);
} }
stmt = (Node *) pg_plan_query(query, cursorOptions, boundParams);
stmt_list = lappend(stmt_list, stmt);
} }
stmt_list = lappend(stmt_list, stmt); if (mySnapshot)
FreeSnapshot(mySnapshot);
}
PG_CATCH();
{
ActiveSnapshot = saveActiveSnapshot;
PG_RE_THROW();
} }
PG_END_TRY();
ActiveSnapshot = saveActiveSnapshot;
return stmt_list; return stmt_list;
} }
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment