}
#endif /* RAW_EXPRESSION_COVERAGE_TEST */
+ /*
+ * Caution: when changing the set of statement types that have non-default
+ * processing here, see also stmt_requires_parse_analysis() and
+ * analyze_requires_snapshot().
+ */
switch (nodeTag(parseTree))
{
/*
}
/*
- * analyze_requires_snapshot
- * Returns true if a snapshot must be set before doing parse analysis
- * on the given raw parse tree.
+ * stmt_requires_parse_analysis
+ * Returns true if parse analysis will do anything non-trivial
+ * with the given raw parse tree.
+ *
+ * Generally, this should return true for any statement type for which
+ * transformStmt() does more than wrap a CMD_UTILITY Query around it.
+ * When it returns false, the caller can assume that there is no situation
+ * in which parse analysis of the raw statement could need to be re-done.
*
- * Classification here should match transformStmt().
+ * Currently, since the rewriter and planner do nothing for CMD_UTILITY
+ * Queries, a false result means that the entire parse analysis/rewrite/plan
+ * pipeline will never need to be re-done. If that ever changes, callers
+ * will likely need adjustment.
*/
bool
-analyze_requires_snapshot(RawStmt *parseTree)
+stmt_requires_parse_analysis(RawStmt *parseTree)
{
bool result;
case T_UpdateStmt:
case T_MergeStmt:
case T_SelectStmt:
+ case T_ReturnStmt:
case T_PLAssignStmt:
result = true;
break;
case T_DeclareCursorStmt:
case T_ExplainStmt:
case T_CreateTableAsStmt:
- /* yes, because we must analyze the contained statement */
+ case T_CallStmt:
result = true;
break;
default:
- /* other utility statements don't have any real parse analysis */
+ /* all other statements just get wrapped in a CMD_UTILITY Query */
result = false;
break;
}
return result;
}
+/*
+ * analyze_requires_snapshot
+ * Returns true if a snapshot must be set before doing parse analysis
+ * on the given raw parse tree.
+ */
+bool
+analyze_requires_snapshot(RawStmt *parseTree)
+{
+ /*
+ * Currently, this should return true in exactly the same cases that
+ * stmt_requires_parse_analysis() does, so we just invoke that function
+ * rather than duplicating it. We keep the two entry points separate for
+ * clarity of callers, since from the callers' standpoint these are
+ * different conditions.
+ *
+ * While there may someday be a statement type for which transformStmt()
+ * does something nontrivial and yet no snapshot is needed for that
+ * processing, it seems likely that making such a choice would be fragile.
+ * If you want to install an exception, document the reasoning for it in a
+ * comment.
+ */
+ return stmt_requires_parse_analysis(parseTree);
+}
+
/*
* transformDeleteStmt -
* transforms a Delete Statement
/*
* We must skip "overhead" operations that involve database access when the
- * cached plan's subject statement is a transaction control command.
- * For the convenience of postgres.c, treat empty statements as control
- * commands too.
+ * cached plan's subject statement is a transaction control command or one
+ * that requires a snapshot not to be set yet (such as SET or LOCK). More
+ * generally, statements that do not require parse analysis/rewrite/plan
+ * activity never need to be revalidated, so we can treat them all like that.
+ * For the convenience of postgres.c, treat empty statements that way too.
*/
-#define IsTransactionStmtPlan(plansource) \
- ((plansource)->raw_parse_tree == NULL || \
- IsA((plansource)->raw_parse_tree->stmt, TransactionStmt))
+#define StmtPlanRequiresRevalidation(plansource) \
+ ((plansource)->raw_parse_tree != NULL && \
+ stmt_requires_parse_analysis((plansource)->raw_parse_tree))
/*
* This is the head of the backend's list of "saved" CachedPlanSources (i.e.,
plansource->query_context = querytree_context;
plansource->query_list = querytree_list;
- if (!plansource->is_oneshot && !IsTransactionStmtPlan(plansource))
+ if (!plansource->is_oneshot && StmtPlanRequiresRevalidation(plansource))
{
/*
* Use the planner machinery to extract dependencies. Data is saved
* in query_context. (We assume that not a lot of extra cruft is
* created by this call.) We can skip this for one-shot plans, and
- * transaction control commands have no such dependencies anyway.
+ * plans not needing revalidation have no such dependencies anyway.
*/
extract_query_dependencies((Node *) querytree_list,
&plansource->relationOids,
/*
* For one-shot plans, we do not support revalidation checking; it's
* assumed the query is parsed, planned, and executed in one transaction,
- * so that no lock re-acquisition is necessary. Also, there is never any
- * need to revalidate plans for transaction control commands (and we
- * mustn't risk any catalog accesses when handling those).
+ * so that no lock re-acquisition is necessary. Also, if the statement
+ * type can't require revalidation, we needn't do anything (and we mustn't
+ * risk catalog accesses when handling, eg, transaction control commands).
*/
- if (plansource->is_oneshot || IsTransactionStmtPlan(plansource))
+ if (plansource->is_oneshot || !StmtPlanRequiresRevalidation(plansource))
{
Assert(plansource->is_valid);
return NIL;
/* Otherwise, never any point in a custom plan if there's no parameters */
if (boundParams == NULL)
return false;
- /* ... nor for transaction control statements */
- if (IsTransactionStmtPlan(plansource))
+ /* ... nor when planning would be a no-op */
+ if (!StmtPlanRequiresRevalidation(plansource))
return false;
/* Let settings force the decision */
if (!plansource->is_valid)
continue;
- /* Never invalidate transaction control commands */
- if (IsTransactionStmtPlan(plansource))
+ /* Never invalidate if parse/plan would be a no-op anyway */
+ if (!StmtPlanRequiresRevalidation(plansource))
continue;
/*
if (!plansource->is_valid)
continue;
- /* Never invalidate transaction control commands */
- if (IsTransactionStmtPlan(plansource))
+ /* Never invalidate if parse/plan would be a no-op anyway */
+ if (!StmtPlanRequiresRevalidation(plansource))
continue;
/*
{
CachedPlanSource *plansource = dlist_container(CachedPlanSource,
node, iter.cur);
- ListCell *lc;
Assert(plansource->magic == CACHEDPLANSOURCE_MAGIC);
* We *must not* mark transaction control statements as invalid,
* particularly not ROLLBACK, because they may need to be executed in
* aborted transactions when we can't revalidate them (cf bug #5269).
+ * In general there's no point in invalidating statements for which a
+ * new parse analysis/rewrite/plan cycle would certainly give the same
+ * results.
*/
- if (IsTransactionStmtPlan(plansource))
+ if (!StmtPlanRequiresRevalidation(plansource))
continue;
- /*
- * In general there is no point in invalidating utility statements
- * since they have no plans anyway. So invalidate it only if it
- * contains at least one non-utility statement, or contains a utility
- * statement that contains a pre-analyzed query (which could have
- * dependencies.)
- */
- foreach(lc, plansource->query_list)
- {
- Query *query = lfirst_node(Query, lc);
-
- if (query->commandType != CMD_UTILITY ||
- UtilityContainsQuery(query->utilityStmt))
- {
- /* non-utility statement, so invalidate */
- plansource->is_valid = false;
- if (plansource->gplan)
- plansource->gplan->is_valid = false;
- /* no need to look further */
- break;
- }
- }
+ plansource->is_valid = false;
+ if (plansource->gplan)
+ plansource->gplan->is_valid = false;
}
/* Likewise invalidate cached expressions */