Commit 8e166e16 authored by Tom Lane's avatar Tom Lane

Rearrange explain.c's API so callers need not embed sizeof(ExplainState).

The folly of the previous arrangement was just demonstrated: there's no
convenient way to add fields to ExplainState without breaking ABI, even
if callers have no need to touch those fields.  Since we might well need
to do that again someday in back branches, let's change things so that
only explain.c has to have sizeof(ExplainState) compiled into it.  This
costs one extra palloc() per EXPLAIN operation, which is surely pretty
negligible.
parent a5cd70dc
...@@ -294,32 +294,31 @@ explain_ExecutorEnd(QueryDesc *queryDesc) ...@@ -294,32 +294,31 @@ explain_ExecutorEnd(QueryDesc *queryDesc)
msec = queryDesc->totaltime->total * 1000.0; msec = queryDesc->totaltime->total * 1000.0;
if (msec >= auto_explain_log_min_duration) if (msec >= auto_explain_log_min_duration)
{ {
ExplainState es; ExplainState *es = NewExplainState();
ExplainInitState(&es); es->analyze = (queryDesc->instrument_options && auto_explain_log_analyze);
es.analyze = (queryDesc->instrument_options && auto_explain_log_analyze); es->verbose = auto_explain_log_verbose;
es.verbose = auto_explain_log_verbose; es->buffers = (es->analyze && auto_explain_log_buffers);
es.buffers = (es.analyze && auto_explain_log_buffers); es->timing = (es->analyze && auto_explain_log_timing);
es.timing = (es.analyze && auto_explain_log_timing); es->summary = es->analyze;
es.summary = es.analyze; es->format = auto_explain_log_format;
es.format = auto_explain_log_format;
ExplainBeginOutput(es);
ExplainBeginOutput(&es); ExplainQueryText(es, queryDesc);
ExplainQueryText(&es, queryDesc); ExplainPrintPlan(es, queryDesc);
ExplainPrintPlan(&es, queryDesc); if (es->analyze && auto_explain_log_triggers)
if (es.analyze && auto_explain_log_triggers) ExplainPrintTriggers(es, queryDesc);
ExplainPrintTriggers(&es, queryDesc); ExplainEndOutput(es);
ExplainEndOutput(&es);
/* Remove last line break */ /* Remove last line break */
if (es.str->len > 0 && es.str->data[es.str->len - 1] == '\n') if (es->str->len > 0 && es->str->data[es->str->len - 1] == '\n')
es.str->data[--es.str->len] = '\0'; es->str->data[--es->str->len] = '\0';
/* Fix JSON to output an object */ /* Fix JSON to output an object */
if (auto_explain_log_format == EXPLAIN_FORMAT_JSON) if (auto_explain_log_format == EXPLAIN_FORMAT_JSON)
{ {
es.str->data[0] = '{'; es->str->data[0] = '{';
es.str->data[es.str->len - 1] = '}'; es->str->data[es->str->len - 1] = '}';
} }
/* /*
...@@ -330,10 +329,10 @@ explain_ExecutorEnd(QueryDesc *queryDesc) ...@@ -330,10 +329,10 @@ explain_ExecutorEnd(QueryDesc *queryDesc)
*/ */
ereport(LOG, ereport(LOG,
(errmsg("duration: %.3f ms plan:\n%s", (errmsg("duration: %.3f ms plan:\n%s",
msec, es.str->data), msec, es->str->data),
errhidestmt(true))); errhidestmt(true)));
pfree(es.str->data); pfree(es->str->data);
} }
} }
......
...@@ -125,45 +125,42 @@ void ...@@ -125,45 +125,42 @@ void
ExplainQuery(ExplainStmt *stmt, const char *queryString, ExplainQuery(ExplainStmt *stmt, const char *queryString,
ParamListInfo params, DestReceiver *dest) ParamListInfo params, DestReceiver *dest)
{ {
ExplainState es; ExplainState *es = NewExplainState();
TupOutputState *tstate; TupOutputState *tstate;
List *rewritten; List *rewritten;
ListCell *lc; ListCell *lc;
bool timing_set = false; bool timing_set = false;
/* Initialize ExplainState. */
ExplainInitState(&es);
/* Parse options list. */ /* Parse options list. */
foreach(lc, stmt->options) foreach(lc, stmt->options)
{ {
DefElem *opt = (DefElem *) lfirst(lc); DefElem *opt = (DefElem *) lfirst(lc);
if (strcmp(opt->defname, "analyze") == 0) if (strcmp(opt->defname, "analyze") == 0)
es.analyze = defGetBoolean(opt); es->analyze = defGetBoolean(opt);
else if (strcmp(opt->defname, "verbose") == 0) else if (strcmp(opt->defname, "verbose") == 0)
es.verbose = defGetBoolean(opt); es->verbose = defGetBoolean(opt);
else if (strcmp(opt->defname, "costs") == 0) else if (strcmp(opt->defname, "costs") == 0)
es.costs = defGetBoolean(opt); es->costs = defGetBoolean(opt);
else if (strcmp(opt->defname, "buffers") == 0) else if (strcmp(opt->defname, "buffers") == 0)
es.buffers = defGetBoolean(opt); es->buffers = defGetBoolean(opt);
else if (strcmp(opt->defname, "timing") == 0) else if (strcmp(opt->defname, "timing") == 0)
{ {
timing_set = true; timing_set = true;
es.timing = defGetBoolean(opt); es->timing = defGetBoolean(opt);
} }
else if (strcmp(opt->defname, "format") == 0) else if (strcmp(opt->defname, "format") == 0)
{ {
char *p = defGetString(opt); char *p = defGetString(opt);
if (strcmp(p, "text") == 0) if (strcmp(p, "text") == 0)
es.format = EXPLAIN_FORMAT_TEXT; es->format = EXPLAIN_FORMAT_TEXT;
else if (strcmp(p, "xml") == 0) else if (strcmp(p, "xml") == 0)
es.format = EXPLAIN_FORMAT_XML; es->format = EXPLAIN_FORMAT_XML;
else if (strcmp(p, "json") == 0) else if (strcmp(p, "json") == 0)
es.format = EXPLAIN_FORMAT_JSON; es->format = EXPLAIN_FORMAT_JSON;
else if (strcmp(p, "yaml") == 0) else if (strcmp(p, "yaml") == 0)
es.format = EXPLAIN_FORMAT_YAML; es->format = EXPLAIN_FORMAT_YAML;
else else
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
...@@ -177,22 +174,22 @@ ExplainQuery(ExplainStmt *stmt, const char *queryString, ...@@ -177,22 +174,22 @@ ExplainQuery(ExplainStmt *stmt, const char *queryString,
opt->defname))); opt->defname)));
} }
if (es.buffers && !es.analyze) if (es->buffers && !es->analyze)
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("EXPLAIN option BUFFERS requires ANALYZE"))); errmsg("EXPLAIN option BUFFERS requires ANALYZE")));
/* if the timing was not set explicitly, set default value */ /* if the timing was not set explicitly, set default value */
es.timing = (timing_set) ? es.timing : es.analyze; es->timing = (timing_set) ? es->timing : es->analyze;
/* check that timing is used with EXPLAIN ANALYZE */ /* check that timing is used with EXPLAIN ANALYZE */
if (es.timing && !es.analyze) if (es->timing && !es->analyze)
ereport(ERROR, ereport(ERROR,
(errcode(ERRCODE_INVALID_PARAMETER_VALUE), (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
errmsg("EXPLAIN option TIMING requires ANALYZE"))); errmsg("EXPLAIN option TIMING requires ANALYZE")));
/* currently, summary option is not exposed to users; just set it */ /* currently, summary option is not exposed to users; just set it */
es.summary = es.analyze; es->summary = es->analyze;
/* /*
* Parse analysis was done already, but we still have to run the rule * Parse analysis was done already, but we still have to run the rule
...@@ -210,7 +207,7 @@ ExplainQuery(ExplainStmt *stmt, const char *queryString, ...@@ -210,7 +207,7 @@ ExplainQuery(ExplainStmt *stmt, const char *queryString,
rewritten = QueryRewrite((Query *) copyObject(stmt->query)); rewritten = QueryRewrite((Query *) copyObject(stmt->query));
/* emit opening boilerplate */ /* emit opening boilerplate */
ExplainBeginOutput(&es); ExplainBeginOutput(es);
if (rewritten == NIL) if (rewritten == NIL)
{ {
...@@ -218,8 +215,8 @@ ExplainQuery(ExplainStmt *stmt, const char *queryString, ...@@ -218,8 +215,8 @@ ExplainQuery(ExplainStmt *stmt, const char *queryString,
* In the case of an INSTEAD NOTHING, tell at least that. But in * In the case of an INSTEAD NOTHING, tell at least that. But in
* non-text format, the output is delimited, so this isn't necessary. * non-text format, the output is delimited, so this isn't necessary.
*/ */
if (es.format == EXPLAIN_FORMAT_TEXT) if (es->format == EXPLAIN_FORMAT_TEXT)
appendStringInfoString(es.str, "Query rewrites to nothing\n"); appendStringInfoString(es->str, "Query rewrites to nothing\n");
} }
else else
{ {
...@@ -228,41 +225,44 @@ ExplainQuery(ExplainStmt *stmt, const char *queryString, ...@@ -228,41 +225,44 @@ ExplainQuery(ExplainStmt *stmt, const char *queryString,
/* Explain every plan */ /* Explain every plan */
foreach(l, rewritten) foreach(l, rewritten)
{ {
ExplainOneQuery((Query *) lfirst(l), NULL, &es, ExplainOneQuery((Query *) lfirst(l), NULL, es,
queryString, params); queryString, params);
/* Separate plans with an appropriate separator */ /* Separate plans with an appropriate separator */
if (lnext(l) != NULL) if (lnext(l) != NULL)
ExplainSeparatePlans(&es); ExplainSeparatePlans(es);
} }
} }
/* emit closing boilerplate */ /* emit closing boilerplate */
ExplainEndOutput(&es); ExplainEndOutput(es);
Assert(es.indent == 0); Assert(es->indent == 0);
/* output tuples */ /* output tuples */
tstate = begin_tup_output_tupdesc(dest, ExplainResultDesc(stmt)); tstate = begin_tup_output_tupdesc(dest, ExplainResultDesc(stmt));
if (es.format == EXPLAIN_FORMAT_TEXT) if (es->format == EXPLAIN_FORMAT_TEXT)
do_text_output_multiline(tstate, es.str->data); do_text_output_multiline(tstate, es->str->data);
else else
do_text_output_oneline(tstate, es.str->data); do_text_output_oneline(tstate, es->str->data);
end_tup_output(tstate); end_tup_output(tstate);
pfree(es.str->data); pfree(es->str->data);
} }
/* /*
* Initialize ExplainState. * Create a new ExplainState struct initialized with default options.
*/ */
void ExplainState *
ExplainInitState(ExplainState *es) NewExplainState(void)
{ {
/* Set default options. */ ExplainState *es = (ExplainState *) palloc0(sizeof(ExplainState));
memset(es, 0, sizeof(ExplainState));
/* Set default options (most fields can be left as zeroes). */
es->costs = true; es->costs = true;
/* Prepare output buffer. */ /* Prepare output buffer. */
es->str = makeStringInfo(); es->str = makeStringInfo();
return es;
} }
/* /*
......
...@@ -60,7 +60,7 @@ extern PGDLLIMPORT explain_get_index_name_hook_type explain_get_index_name_hook; ...@@ -60,7 +60,7 @@ extern PGDLLIMPORT explain_get_index_name_hook_type explain_get_index_name_hook;
extern void ExplainQuery(ExplainStmt *stmt, const char *queryString, extern void ExplainQuery(ExplainStmt *stmt, const char *queryString,
ParamListInfo params, DestReceiver *dest); ParamListInfo params, DestReceiver *dest);
extern void ExplainInitState(ExplainState *es); extern ExplainState *NewExplainState(void);
extern TupleDesc ExplainResultDesc(ExplainStmt *stmt); extern TupleDesc ExplainResultDesc(ExplainStmt *stmt);
......
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