Commit 1f902d49 authored by Tom Lane's avatar Tom Lane

Inline plpgsql's exec_stmt() into exec_stmts().

This saves one level of C function call per plpgsql statement executed,
and permits a tiny additional optimization of not saving and restoring
estate->err_stmt for each statement in a block.  The net effect seems
nearly un-measurable on x86_64, but there's a clear win on aarch64,
amounting to two or three percent in a loop over a few simple plpgsql
statements.

To do this, we have to get rid of the other existing call sites for
exec_stmt().  Replace them with exec_toplevel_block(), which is just
defined to do what exec_stmts() does, but for a single
PLpgSQL_stmt_block statement.  Hard-wiring the expectation of which
statement type applies here allows us to skip the dispatch switch,
making this not much uglier than the previous factorization.

Amit Khandekar, tweaked a bit by me

Discussion: https://postgr.es/m/CAJ3gD9eBNrmUD7WBBLG8ohaZ485H9y+4eihQTgr+K8Lhka3vcQ@mail.gmail.com
parent ecd9e9f0
...@@ -260,12 +260,12 @@ static MemoryContext get_stmt_mcontext(PLpgSQL_execstate *estate); ...@@ -260,12 +260,12 @@ static MemoryContext get_stmt_mcontext(PLpgSQL_execstate *estate);
static void push_stmt_mcontext(PLpgSQL_execstate *estate); static void push_stmt_mcontext(PLpgSQL_execstate *estate);
static void pop_stmt_mcontext(PLpgSQL_execstate *estate); static void pop_stmt_mcontext(PLpgSQL_execstate *estate);
static int exec_toplevel_block(PLpgSQL_execstate *estate,
PLpgSQL_stmt_block *block);
static int exec_stmt_block(PLpgSQL_execstate *estate, static int exec_stmt_block(PLpgSQL_execstate *estate,
PLpgSQL_stmt_block *block); PLpgSQL_stmt_block *block);
static int exec_stmts(PLpgSQL_execstate *estate, static int exec_stmts(PLpgSQL_execstate *estate,
List *stmts); List *stmts);
static int exec_stmt(PLpgSQL_execstate *estate,
PLpgSQL_stmt *stmt);
static int exec_stmt_assign(PLpgSQL_execstate *estate, static int exec_stmt_assign(PLpgSQL_execstate *estate,
PLpgSQL_stmt_assign *stmt); PLpgSQL_stmt_assign *stmt);
static int exec_stmt_perform(PLpgSQL_execstate *estate, static int exec_stmt_perform(PLpgSQL_execstate *estate,
...@@ -599,11 +599,9 @@ plpgsql_exec_function(PLpgSQL_function *func, FunctionCallInfo fcinfo, ...@@ -599,11 +599,9 @@ plpgsql_exec_function(PLpgSQL_function *func, FunctionCallInfo fcinfo,
* Now call the toplevel block of statements * Now call the toplevel block of statements
*/ */
estate.err_text = NULL; estate.err_text = NULL;
estate.err_stmt = (PLpgSQL_stmt *) (func->action); rc = exec_toplevel_block(&estate, func->action);
rc = exec_stmt(&estate, (PLpgSQL_stmt *) func->action);
if (rc != PLPGSQL_RC_RETURN) if (rc != PLPGSQL_RC_RETURN)
{ {
estate.err_stmt = NULL;
estate.err_text = NULL; estate.err_text = NULL;
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_S_R_E_FUNCTION_EXECUTED_NO_RETURN_STATEMENT), (errcode(ERRCODE_S_R_E_FUNCTION_EXECUTED_NO_RETURN_STATEMENT),
...@@ -613,7 +611,6 @@ plpgsql_exec_function(PLpgSQL_function *func, FunctionCallInfo fcinfo, ...@@ -613,7 +611,6 @@ plpgsql_exec_function(PLpgSQL_function *func, FunctionCallInfo fcinfo,
/* /*
* We got a return value - process it * We got a return value - process it
*/ */
estate.err_stmt = NULL;
estate.err_text = gettext_noop("while casting return value to function's return type"); estate.err_text = gettext_noop("while casting return value to function's return type");
fcinfo->isnull = estate.retisnull; fcinfo->isnull = estate.retisnull;
...@@ -1015,18 +1012,15 @@ plpgsql_exec_trigger(PLpgSQL_function *func, ...@@ -1015,18 +1012,15 @@ plpgsql_exec_trigger(PLpgSQL_function *func,
* Now call the toplevel block of statements * Now call the toplevel block of statements
*/ */
estate.err_text = NULL; estate.err_text = NULL;
estate.err_stmt = (PLpgSQL_stmt *) (func->action); rc = exec_toplevel_block(&estate, func->action);
rc = exec_stmt(&estate, (PLpgSQL_stmt *) func->action);
if (rc != PLPGSQL_RC_RETURN) if (rc != PLPGSQL_RC_RETURN)
{ {
estate.err_stmt = NULL;
estate.err_text = NULL; estate.err_text = NULL;
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_S_R_E_FUNCTION_EXECUTED_NO_RETURN_STATEMENT), (errcode(ERRCODE_S_R_E_FUNCTION_EXECUTED_NO_RETURN_STATEMENT),
errmsg("control reached end of trigger procedure without RETURN"))); errmsg("control reached end of trigger procedure without RETURN")));
} }
estate.err_stmt = NULL;
estate.err_text = gettext_noop("during function exit"); estate.err_text = gettext_noop("during function exit");
if (estate.retisset) if (estate.retisset)
...@@ -1176,18 +1170,15 @@ plpgsql_exec_event_trigger(PLpgSQL_function *func, EventTriggerData *trigdata) ...@@ -1176,18 +1170,15 @@ plpgsql_exec_event_trigger(PLpgSQL_function *func, EventTriggerData *trigdata)
* Now call the toplevel block of statements * Now call the toplevel block of statements
*/ */
estate.err_text = NULL; estate.err_text = NULL;
estate.err_stmt = (PLpgSQL_stmt *) (func->action); rc = exec_toplevel_block(&estate, func->action);
rc = exec_stmt(&estate, (PLpgSQL_stmt *) func->action);
if (rc != PLPGSQL_RC_RETURN) if (rc != PLPGSQL_RC_RETURN)
{ {
estate.err_stmt = NULL;
estate.err_text = NULL; estate.err_text = NULL;
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_S_R_E_FUNCTION_EXECUTED_NO_RETURN_STATEMENT), (errcode(ERRCODE_S_R_E_FUNCTION_EXECUTED_NO_RETURN_STATEMENT),
errmsg("control reached end of trigger procedure without RETURN"))); errmsg("control reached end of trigger procedure without RETURN")));
} }
estate.err_stmt = NULL;
estate.err_text = gettext_noop("during function exit"); estate.err_text = gettext_noop("during function exit");
/* /*
...@@ -1584,6 +1575,40 @@ exception_matches_conditions(ErrorData *edata, PLpgSQL_condition *cond) ...@@ -1584,6 +1575,40 @@ exception_matches_conditions(ErrorData *edata, PLpgSQL_condition *cond)
} }
/* ----------
* exec_toplevel_block Execute the toplevel block
*
* This is intentionally equivalent to executing exec_stmts() with a
* list consisting of the one statement. One tiny difference is that
* we do not bother to save the entry value of estate->err_stmt;
* that's assumed to be NULL.
* ----------
*/
static int
exec_toplevel_block(PLpgSQL_execstate *estate, PLpgSQL_stmt_block *block)
{
int rc;
estate->err_stmt = (PLpgSQL_stmt *) block;
/* Let the plugin know that we are about to execute this statement */
if (*plpgsql_plugin_ptr && (*plpgsql_plugin_ptr)->stmt_beg)
((*plpgsql_plugin_ptr)->stmt_beg) (estate, (PLpgSQL_stmt *) block);
CHECK_FOR_INTERRUPTS();
rc = exec_stmt_block(estate, block);
/* Let the plugin know that we have finished executing this statement */
if (*plpgsql_plugin_ptr && (*plpgsql_plugin_ptr)->stmt_end)
((*plpgsql_plugin_ptr)->stmt_end) (estate, (PLpgSQL_stmt *) block);
estate->err_stmt = NULL;
return rc;
}
/* ---------- /* ----------
* exec_stmt_block Execute a block of statements * exec_stmt_block Execute a block of statements
* ---------- * ----------
...@@ -1917,6 +1942,7 @@ exec_stmt_block(PLpgSQL_execstate *estate, PLpgSQL_stmt_block *block) ...@@ -1917,6 +1942,7 @@ exec_stmt_block(PLpgSQL_execstate *estate, PLpgSQL_stmt_block *block)
static int static int
exec_stmts(PLpgSQL_execstate *estate, List *stmts) exec_stmts(PLpgSQL_execstate *estate, List *stmts)
{ {
PLpgSQL_stmt *save_estmt = estate->err_stmt;
ListCell *s; ListCell *s;
if (stmts == NIL) if (stmts == NIL)
...@@ -1933,28 +1959,8 @@ exec_stmts(PLpgSQL_execstate *estate, List *stmts) ...@@ -1933,28 +1959,8 @@ exec_stmts(PLpgSQL_execstate *estate, List *stmts)
foreach(s, stmts) foreach(s, stmts)
{ {
PLpgSQL_stmt *stmt = (PLpgSQL_stmt *) lfirst(s); PLpgSQL_stmt *stmt = (PLpgSQL_stmt *) lfirst(s);
int rc = exec_stmt(estate, stmt); int rc;
if (rc != PLPGSQL_RC_OK)
return rc;
}
return PLPGSQL_RC_OK;
}
/* ----------
* exec_stmt Distribute one statement to the statements
* type specific execution function.
* ----------
*/
static int
exec_stmt(PLpgSQL_execstate *estate, PLpgSQL_stmt *stmt)
{
PLpgSQL_stmt *save_estmt;
int rc = -1;
save_estmt = estate->err_stmt;
estate->err_stmt = stmt; estate->err_stmt = stmt;
/* Let the plugin know that we are about to execute this statement */ /* Let the plugin know that we are about to execute this statement */
...@@ -2078,17 +2084,25 @@ exec_stmt(PLpgSQL_execstate *estate, PLpgSQL_stmt *stmt) ...@@ -2078,17 +2084,25 @@ exec_stmt(PLpgSQL_execstate *estate, PLpgSQL_stmt *stmt)
break; break;
default: default:
/* point err_stmt to parent, since this one seems corrupt */
estate->err_stmt = save_estmt; estate->err_stmt = save_estmt;
elog(ERROR, "unrecognized cmd_type: %d", stmt->cmd_type); elog(ERROR, "unrecognized cmd_type: %d", stmt->cmd_type);
rc = -1; /* keep compiler quiet */
} }
/* Let the plugin know that we have finished executing this statement */ /* Let the plugin know that we have finished executing this statement */
if (*plpgsql_plugin_ptr && (*plpgsql_plugin_ptr)->stmt_end) if (*plpgsql_plugin_ptr && (*plpgsql_plugin_ptr)->stmt_end)
((*plpgsql_plugin_ptr)->stmt_end) (estate, stmt); ((*plpgsql_plugin_ptr)->stmt_end) (estate, stmt);
if (rc != PLPGSQL_RC_OK)
{
estate->err_stmt = save_estmt; estate->err_stmt = save_estmt;
return rc; return rc;
}
} /* end of loop over statements */
estate->err_stmt = save_estmt;
return PLPGSQL_RC_OK;
} }
......
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