PostgreSQL Source Code git master
All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
pg_overexplain.c File Reference
#include "postgres.h"
#include "catalog/pg_class.h"
#include "commands/defrem.h"
#include "commands/explain.h"
#include "commands/explain_format.h"
#include "commands/explain_state.h"
#include "fmgr.h"
#include "parser/parsetree.h"
#include "storage/lock.h"
#include "utils/builtins.h"
#include "utils/lsyscache.h"
Include dependency graph for pg_overexplain.c:

Go to the source code of this file.

Data Structures

struct  overexplain_options
 

Functions

 PG_MODULE_MAGIC_EXT (.name="pg_overexplain",.version=PG_VERSION)
 
static overexplain_optionsoverexplain_ensure_options (ExplainState *es)
 
static void overexplain_debug_handler (ExplainState *es, DefElem *opt, ParseState *pstate)
 
static void overexplain_range_table_handler (ExplainState *es, DefElem *opt, ParseState *pstate)
 
static void overexplain_per_node_hook (PlanState *planstate, List *ancestors, const char *relationship, const char *plan_name, ExplainState *es)
 
static void overexplain_per_plan_hook (PlannedStmt *plannedstmt, IntoClause *into, ExplainState *es, const char *queryString, ParamListInfo params, QueryEnvironment *queryEnv)
 
static void overexplain_debug (PlannedStmt *plannedstmt, ExplainState *es)
 
static void overexplain_range_table (PlannedStmt *plannedstmt, ExplainState *es)
 
static void overexplain_alias (const char *qlabel, Alias *alias, ExplainState *es)
 
static void overexplain_bitmapset (const char *qlabel, Bitmapset *bms, ExplainState *es)
 
static void overexplain_intlist (const char *qlabel, List *intlist, ExplainState *es)
 
void _PG_init (void)
 

Variables

static int es_extension_id
 
static explain_per_node_hook_type prev_explain_per_node_hook
 
static explain_per_plan_hook_type prev_explain_per_plan_hook
 

Function Documentation

◆ _PG_init()

void _PG_init ( void  )

Definition at line 68 of file pg_overexplain.c.

69{
70 /* Get an ID that we can use to cache data in an ExplainState. */
71 es_extension_id = GetExplainExtensionId("pg_overexplain");
72
73 /* Register the new EXPLAIN options implemented by this module. */
77
78 /* Use the per-node and per-plan hooks to make our options do something. */
83}
explain_per_node_hook_type explain_per_node_hook
Definition: explain.c:57
explain_per_plan_hook_type explain_per_plan_hook
Definition: explain.c:56
int GetExplainExtensionId(const char *extension_name)
void RegisterExtensionExplainOption(const char *option_name, ExplainOptionHandler handler)
static void overexplain_per_plan_hook(PlannedStmt *plannedstmt, IntoClause *into, ExplainState *es, const char *queryString, ParamListInfo params, QueryEnvironment *queryEnv)
static explain_per_plan_hook_type prev_explain_per_plan_hook
static explain_per_node_hook_type prev_explain_per_node_hook
static void overexplain_per_node_hook(PlanState *planstate, List *ancestors, const char *relationship, const char *plan_name, ExplainState *es)
static void overexplain_debug_handler(ExplainState *es, DefElem *opt, ParseState *pstate)
static int es_extension_id
static void overexplain_range_table_handler(ExplainState *es, DefElem *opt, ParseState *pstate)

References es_extension_id, explain_per_node_hook, explain_per_plan_hook, GetExplainExtensionId(), overexplain_debug_handler(), overexplain_per_node_hook(), overexplain_per_plan_hook(), overexplain_range_table_handler(), prev_explain_per_node_hook, prev_explain_per_plan_hook, and RegisterExtensionExplainOption().

◆ overexplain_alias()

static void overexplain_alias ( const char *  qlabel,
Alias alias,
ExplainState es 
)
static

Definition at line 682 of file pg_overexplain.c.

683{
685 bool first = true;
686
687 Assert(alias != NULL);
688
691
692 foreach_node(String, cn, alias->colnames)
693 {
694 appendStringInfo(&buf, "%s%s",
695 first ? "" : ", ",
696 quote_identifier(cn->sval));
697 first = false;
698 }
699
701 ExplainPropertyText(qlabel, buf.data, es);
702 pfree(buf.data);
703}
void ExplainPropertyText(const char *qlabel, const char *value, ExplainState *es)
Assert(PointerIsAligned(start, uint64))
void pfree(void *pointer)
Definition: mcxt.c:1524
#define foreach_node(type, var, lst)
Definition: pg_list.h:496
static char * buf
Definition: pg_test_fsync.c:72
const char * quote_identifier(const char *ident)
Definition: ruleutils.c:13019
void appendStringInfo(StringInfo str, const char *fmt,...)
Definition: stringinfo.c:145
void appendStringInfoChar(StringInfo str, char ch)
Definition: stringinfo.c:242
void initStringInfo(StringInfo str)
Definition: stringinfo.c:97
char * aliasname
Definition: primnodes.h:51
List * colnames
Definition: primnodes.h:52
Definition: value.h:64

References Alias::aliasname, appendStringInfo(), appendStringInfoChar(), Assert(), buf, Alias::colnames, ExplainPropertyText(), foreach_node, initStringInfo(), pfree(), and quote_identifier().

Referenced by overexplain_range_table().

◆ overexplain_bitmapset()

static void overexplain_bitmapset ( const char *  qlabel,
Bitmapset bms,
ExplainState es 
)
static

Definition at line 711 of file pg_overexplain.c.

712{
713 int x = -1;
714
716
717 if (bms_is_empty(bms))
718 {
719 ExplainPropertyText(qlabel, "none", es);
720 return;
721 }
722
724 while ((x = bms_next_member(bms, x)) >= 0)
725 appendStringInfo(&buf, " %d", x);
726 Assert(buf.data[0] == ' ');
727 ExplainPropertyText(qlabel, buf.data + 1, es);
728 pfree(buf.data);
729}
int bms_next_member(const Bitmapset *a, int prevbit)
Definition: bitmapset.c:1306
#define bms_is_empty(a)
Definition: bitmapset.h:118
int x
Definition: isn.c:75

References appendStringInfo(), Assert(), bms_is_empty, bms_next_member(), buf, ExplainPropertyText(), initStringInfo(), pfree(), and x.

Referenced by overexplain_debug(), overexplain_per_node_hook(), and overexplain_range_table().

◆ overexplain_debug()

static void overexplain_debug ( PlannedStmt plannedstmt,
ExplainState es 
)
static

Definition at line 285 of file pg_overexplain.c.

286{
287 char *commandType = NULL;
288 StringInfoData flags;
289
290 /* Even in text mode, we want to set this output apart as its own group. */
291 ExplainOpenGroup("PlannedStmt", "PlannedStmt", true, es);
292 if (es->format == EXPLAIN_FORMAT_TEXT)
293 {
295 appendStringInfo(es->str, "PlannedStmt:\n");
296 es->indent++;
297 }
298
299 /* Print the command type. */
300 switch (plannedstmt->commandType)
301 {
302 case CMD_UNKNOWN:
303 commandType = "unknown";
304 break;
305 case CMD_SELECT:
306 commandType = "select";
307 break;
308 case CMD_UPDATE:
309 commandType = "update";
310 break;
311 case CMD_INSERT:
312 commandType = "insert";
313 break;
314 case CMD_DELETE:
315 commandType = "delete";
316 break;
317 case CMD_MERGE:
318 commandType = "merge";
319 break;
320 case CMD_UTILITY:
321 commandType = "utility";
322 break;
323 case CMD_NOTHING:
324 commandType = "nothing";
325 break;
326 }
327 ExplainPropertyText("Command Type", commandType, es);
328
329 /* Print various properties as a comma-separated list of flags. */
330 initStringInfo(&flags);
331 if (plannedstmt->hasReturning)
332 appendStringInfo(&flags, ", hasReturning");
333 if (plannedstmt->hasModifyingCTE)
334 appendStringInfo(&flags, ", hasModifyingCTE");
335 if (plannedstmt->canSetTag)
336 appendStringInfo(&flags, ", canSetTag");
337 if (plannedstmt->transientPlan)
338 appendStringInfo(&flags, ", transientPlan");
339 if (plannedstmt->dependsOnRole)
340 appendStringInfo(&flags, ", dependsOnRole");
341 if (plannedstmt->parallelModeNeeded)
342 appendStringInfo(&flags, ", parallelModeNeeded");
343 if (flags.len == 0)
344 appendStringInfo(&flags, ", none");
345 ExplainPropertyText("Flags", flags.data + 2, es);
346
347 /* Various lists of integers. */
348 overexplain_bitmapset("Subplans Needing Rewind",
349 plannedstmt->rewindPlanIDs, es);
350 overexplain_intlist("Relation OIDs",
351 plannedstmt->relationOids, es);
352 overexplain_intlist("Executor Parameter Types",
353 plannedstmt->paramExecTypes, es);
354
355 /*
356 * Print the statement location. (If desired, we could alternatively print
357 * stmt_location and stmt_len as two separate fields.)
358 */
359 if (plannedstmt->stmt_location == -1)
360 ExplainPropertyText("Parse Location", "Unknown", es);
361 else if (plannedstmt->stmt_len == 0)
362 ExplainPropertyText("Parse Location",
363 psprintf("%d to end", plannedstmt->stmt_location),
364 es);
365 else
366 ExplainPropertyText("Parse Location",
367 psprintf("%d for %d bytes",
368 plannedstmt->stmt_location,
369 plannedstmt->stmt_len),
370 es);
371
372 /* Done with this group. */
373 if (es->format == EXPLAIN_FORMAT_TEXT)
374 es->indent--;
375 ExplainCloseGroup("PlannedStmt", "PlannedStmt", true, es);
376}
void ExplainOpenGroup(const char *objtype, const char *labelname, bool labeled, ExplainState *es)
void ExplainIndentText(ExplainState *es)
void ExplainCloseGroup(const char *objtype, const char *labelname, bool labeled, ExplainState *es)
@ EXPLAIN_FORMAT_TEXT
Definition: explain_state.h:29
@ CMD_MERGE
Definition: nodes.h:275
@ CMD_UTILITY
Definition: nodes.h:276
@ CMD_INSERT
Definition: nodes.h:273
@ CMD_DELETE
Definition: nodes.h:274
@ CMD_UNKNOWN
Definition: nodes.h:270
@ CMD_UPDATE
Definition: nodes.h:272
@ CMD_SELECT
Definition: nodes.h:271
@ CMD_NOTHING
Definition: nodes.h:278
static void overexplain_bitmapset(const char *qlabel, Bitmapset *bms, ExplainState *es)
static void overexplain_intlist(const char *qlabel, List *intlist, ExplainState *es)
char * psprintf(const char *fmt,...)
Definition: psprintf.c:43
StringInfo str
Definition: explain_state.h:46
ExplainFormat format
Definition: explain_state.h:59
bool hasModifyingCTE
Definition: plannodes.h:65
bool canSetTag
Definition: plannodes.h:68
Bitmapset * rewindPlanIDs
Definition: plannodes.h:124
ParseLoc stmt_len
Definition: plannodes.h:145
bool hasReturning
Definition: plannodes.h:62
ParseLoc stmt_location
Definition: plannodes.h:143
bool transientPlan
Definition: plannodes.h:71
List * relationOids
Definition: plannodes.h:130
bool dependsOnRole
Definition: plannodes.h:74
CmdType commandType
Definition: plannodes.h:53
List * paramExecTypes
Definition: plannodes.h:136
bool parallelModeNeeded
Definition: plannodes.h:77

References appendStringInfo(), PlannedStmt::canSetTag, CMD_DELETE, CMD_INSERT, CMD_MERGE, CMD_NOTHING, CMD_SELECT, CMD_UNKNOWN, CMD_UPDATE, CMD_UTILITY, PlannedStmt::commandType, StringInfoData::data, PlannedStmt::dependsOnRole, EXPLAIN_FORMAT_TEXT, ExplainCloseGroup(), ExplainIndentText(), ExplainOpenGroup(), ExplainPropertyText(), ExplainState::format, PlannedStmt::hasModifyingCTE, PlannedStmt::hasReturning, ExplainState::indent, initStringInfo(), StringInfoData::len, overexplain_bitmapset(), overexplain_intlist(), PlannedStmt::parallelModeNeeded, PlannedStmt::paramExecTypes, psprintf(), PlannedStmt::relationOids, PlannedStmt::rewindPlanIDs, PlannedStmt::stmt_len, PlannedStmt::stmt_location, ExplainState::str, and PlannedStmt::transientPlan.

Referenced by overexplain_per_plan_hook().

◆ overexplain_debug_handler()

static void overexplain_debug_handler ( ExplainState es,
DefElem opt,
ParseState pstate 
)
static

Definition at line 109 of file pg_overexplain.c.

110{
112
113 options->debug = defGetBoolean(opt);
114}
bool defGetBoolean(DefElem *def)
Definition: define.c:94
static overexplain_options * overexplain_ensure_options(ExplainState *es)

References defGetBoolean(), and overexplain_ensure_options().

Referenced by _PG_init().

◆ overexplain_ensure_options()

static overexplain_options * overexplain_ensure_options ( ExplainState es)
static

Definition at line 90 of file pg_overexplain.c.

91{
93
95
96 if (options == NULL)
97 {
100 }
101
102 return options;
103}
void * GetExplainExtensionState(ExplainState *es, int extension_id)
void SetExplainExtensionState(ExplainState *es, int extension_id, void *opaque)
void * palloc0(Size size)
Definition: mcxt.c:1347
static char ** options

References es_extension_id, GetExplainExtensionState(), options, palloc0(), and SetExplainExtensionState().

Referenced by overexplain_debug_handler(), and overexplain_range_table_handler().

◆ overexplain_intlist()

static void overexplain_intlist ( const char *  qlabel,
List intlist,
ExplainState es 
)
static

Definition at line 737 of file pg_overexplain.c.

738{
740
742
743 if (list == NIL)
744 {
745 ExplainPropertyText(qlabel, "none", es);
746 return;
747 }
748
749 if (IsA(list, IntList))
750 {
752 appendStringInfo(&buf, " %d", i);
753 }
754 else if (IsA(list, OidList))
755 {
756 foreach_oid(o, list)
757 appendStringInfo(&buf, " %u", o);
758 }
759 else if (IsA(list, XidList))
760 {
762 appendStringInfo(&buf, " %u", x);
763 }
764 else
765 {
766 appendStringInfo(&buf, " not an integer list");
767 Assert(false);
768 }
769
770 if (buf.len > 0)
771 ExplainPropertyText(qlabel, buf.data + 1, es);
772
773 pfree(buf.data);
774}
int i
Definition: isn.c:77
#define IsA(nodeptr, _type_)
Definition: nodes.h:164
#define NIL
Definition: pg_list.h:68
#define foreach_xid(var, lst)
Definition: pg_list.h:472
#define foreach_oid(var, lst)
Definition: pg_list.h:471
#define foreach_int(var, lst)
Definition: pg_list.h:470

References appendStringInfo(), Assert(), buf, ExplainPropertyText(), foreach_int, foreach_oid, foreach_xid, i, initStringInfo(), IsA, sort-test::list, NIL, pfree(), and x.

Referenced by overexplain_debug(), and overexplain_range_table().

◆ overexplain_per_node_hook()

static void overexplain_per_node_hook ( PlanState planstate,
List ancestors,
const char *  relationship,
const char *  plan_name,
ExplainState es 
)
static

Definition at line 134 of file pg_overexplain.c.

137{
139 Plan *plan = planstate->plan;
140
142 (*prev_explain_per_node_hook) (planstate, ancestors, relationship,
143 plan_name, es);
144
146 if (options == NULL)
147 return;
148
149 /*
150 * If the "debug" option was given, display miscellaneous fields from the
151 * "Plan" node that would not otherwise be displayed.
152 */
153 if (options->debug)
154 {
155 /*
156 * Normal EXPLAIN will display "Disabled: true" if the node is
157 * disabled; but that is based on noticing that plan->disabled_nodes
158 * is higher than the sum of its children; here, we display the raw
159 * value, for debugging purposes.
160 */
161 ExplainPropertyInteger("Disabled Nodes", NULL, plan->disabled_nodes,
162 es);
163
164 /*
165 * Normal EXPLAIN will display the parallel_aware flag; here, we show
166 * the parallel_safe flag as well.
167 */
168 ExplainPropertyBool("Parallel Safe", plan->parallel_safe, es);
169
170 /*
171 * The plan node ID isn't normally displayed, since it is only useful
172 * for debugging.
173 */
174 ExplainPropertyInteger("Plan Node ID", NULL, plan->plan_node_id, es);
175
176 /*
177 * It is difficult to explain what extParam and allParam mean in plain
178 * language, so we simply display these fields labelled with the
179 * structure member name. For compactness, the text format omits the
180 * display of this information when the bitmapset is empty.
181 */
182 if (es->format != EXPLAIN_FORMAT_TEXT || !bms_is_empty(plan->extParam))
183 overexplain_bitmapset("extParam", plan->extParam, es);
184 if (es->format != EXPLAIN_FORMAT_TEXT || !bms_is_empty(plan->allParam))
185 overexplain_bitmapset("allParam", plan->allParam, es);
186 }
187
188 /*
189 * If the "range_table" option was specified, display information about
190 * the range table indexes for this node.
191 */
192 if (options->range_table)
193 {
194 switch (nodeTag(plan))
195 {
196 case T_SeqScan:
197 case T_SampleScan:
198 case T_IndexScan:
199 case T_IndexOnlyScan:
200 case T_BitmapHeapScan:
201 case T_TidScan:
202 case T_TidRangeScan:
203 case T_SubqueryScan:
204 case T_FunctionScan:
205 case T_TableFuncScan:
206 case T_ValuesScan:
207 case T_CteScan:
208 case T_NamedTuplestoreScan:
209 case T_WorkTableScan:
210 ExplainPropertyInteger("Scan RTI", NULL,
211 ((Scan *) plan)->scanrelid, es);
212 break;
213 case T_ForeignScan:
214 overexplain_bitmapset("Scan RTIs",
215 ((ForeignScan *) plan)->fs_base_relids,
216 es);
217 break;
218 case T_CustomScan:
219 overexplain_bitmapset("Scan RTIs",
220 ((CustomScan *) plan)->custom_relids,
221 es);
222 break;
223 case T_ModifyTable:
224 ExplainPropertyInteger("Nominal RTI", NULL,
225 ((ModifyTable *) plan)->nominalRelation, es);
226 ExplainPropertyInteger("Exclude Relation RTI", NULL,
227 ((ModifyTable *) plan)->exclRelRTI, es);
228 break;
229 case T_Append:
230 overexplain_bitmapset("Append RTIs",
231 ((Append *) plan)->apprelids,
232 es);
233 break;
234 case T_MergeAppend:
235 overexplain_bitmapset("Append RTIs",
236 ((MergeAppend *) plan)->apprelids,
237 es);
238 break;
239 default:
240 break;
241 }
242 }
243}
void ExplainPropertyInteger(const char *qlabel, const char *unit, int64 value, ExplainState *es)
void ExplainPropertyBool(const char *qlabel, bool value, ExplainState *es)
#define nodeTag(nodeptr)
Definition: nodes.h:139
#define plan(x)
Definition: pg_regress.c:161
Plan * plan
Definition: execnodes.h:1159

References bms_is_empty, es_extension_id, EXPLAIN_FORMAT_TEXT, ExplainPropertyBool(), ExplainPropertyInteger(), ExplainState::format, GetExplainExtensionState(), nodeTag, options, overexplain_bitmapset(), PlanState::plan, plan, and prev_explain_per_node_hook.

Referenced by _PG_init().

◆ overexplain_per_plan_hook()

static void overexplain_per_plan_hook ( PlannedStmt plannedstmt,
IntoClause into,
ExplainState es,
const char *  queryString,
ParamListInfo  params,
QueryEnvironment queryEnv 
)
static

Definition at line 252 of file pg_overexplain.c.

258{
260
262 (*prev_explain_per_plan_hook) (plannedstmt, into, es, queryString,
263 params, queryEnv);
264
266 if (options == NULL)
267 return;
268
269 if (options->debug)
270 overexplain_debug(plannedstmt, es);
271
272 if (options->range_table)
273 overexplain_range_table(plannedstmt, es);
274}
static void overexplain_range_table(PlannedStmt *plannedstmt, ExplainState *es)
static void overexplain_debug(PlannedStmt *plannedstmt, ExplainState *es)

References es_extension_id, GetExplainExtensionState(), options, overexplain_debug(), overexplain_range_table(), and prev_explain_per_plan_hook.

Referenced by _PG_init().

◆ overexplain_range_table()

static void overexplain_range_table ( PlannedStmt plannedstmt,
ExplainState es 
)
static

Definition at line 383 of file pg_overexplain.c.

384{
385 Index rti;
386
387 /* Open group, one entry per RangeTblEntry */
388 ExplainOpenGroup("Range Table", "Range Table", false, es);
389
390 /* Iterate over the range table */
391 for (rti = 1; rti <= list_length(plannedstmt->rtable); ++rti)
392 {
393 RangeTblEntry *rte = rt_fetch(rti, plannedstmt->rtable);
394 char *kind = NULL;
395 char *relkind;
396
397 /* NULL entries are possible; skip them */
398 if (rte == NULL)
399 continue;
400
401 /* Translate rtekind to a string */
402 switch (rte->rtekind)
403 {
404 case RTE_RELATION:
405 kind = "relation";
406 break;
407 case RTE_SUBQUERY:
408 kind = "subquery";
409 break;
410 case RTE_JOIN:
411 kind = "join";
412 break;
413 case RTE_FUNCTION:
414 kind = "function";
415 break;
416 case RTE_TABLEFUNC:
417 kind = "tablefunc";
418 break;
419 case RTE_VALUES:
420 kind = "values";
421 break;
422 case RTE_CTE:
423 kind = "cte";
424 break;
426 kind = "namedtuplestore";
427 break;
428 case RTE_RESULT:
429 kind = "result";
430 break;
431 case RTE_GROUP:
432 kind = "group";
433 break;
434 }
435
436 /* Begin group for this specific RTE */
437 ExplainOpenGroup("Range Table Entry", NULL, true, es);
438
439 /*
440 * In text format, the summary line displays the range table index and
441 * rtekind, plus indications if rte->inh and/or rte->inFromCl are set.
442 * In other formats, we display those as separate properties.
443 */
444 if (es->format == EXPLAIN_FORMAT_TEXT)
445 {
447 appendStringInfo(es->str, "RTI %u (%s%s%s):\n", rti, kind,
448 rte->inh ? ", inherited" : "",
449 rte->inFromCl ? ", in-from-clause" : "");
450 es->indent++;
451 }
452 else
453 {
454 ExplainPropertyUInteger("RTI", NULL, rti, es);
455 ExplainPropertyText("Kind", kind, es);
456 ExplainPropertyBool("Inherited", rte->inh, es);
457 ExplainPropertyBool("In From Clause", rte->inFromCl, es);
458 }
459
460 /* rte->alias is optional; rte->eref is requested */
461 if (rte->alias != NULL)
462 overexplain_alias("Alias", rte->alias, es);
463 overexplain_alias("Eref", rte->eref, es);
464
465 /*
466 * We adhere to the usual EXPLAIN convention that schema names are
467 * displayed only in verbose mode, and we emit nothing if there is no
468 * relation OID.
469 */
470 if (rte->relid != 0)
471 {
472 const char *relname;
473 const char *qualname;
474
476
477 if (es->verbose)
478 {
479 Oid nspoid = get_rel_namespace(rte->relid);
480 char *nspname;
481
482 nspname = get_namespace_name_or_temp(nspoid);
483 qualname = psprintf("%s.%s", quote_identifier(nspname),
484 relname);
485 }
486 else
487 qualname = relname;
488
489 ExplainPropertyText("Relation", qualname, es);
490 }
491
492 /* Translate relkind, if any, to a string */
493 switch (rte->relkind)
494 {
495 case RELKIND_RELATION:
496 relkind = "relation";
497 break;
498 case RELKIND_INDEX:
499 relkind = "index";
500 break;
501 case RELKIND_SEQUENCE:
502 relkind = "sequence";
503 break;
504 case RELKIND_TOASTVALUE:
505 relkind = "toastvalue";
506 break;
507 case RELKIND_VIEW:
508 relkind = "view";
509 break;
510 case RELKIND_MATVIEW:
511 relkind = "matview";
512 break;
513 case RELKIND_COMPOSITE_TYPE:
514 relkind = "composite_type";
515 break;
516 case RELKIND_FOREIGN_TABLE:
517 relkind = "foreign_table";
518 break;
519 case RELKIND_PARTITIONED_TABLE:
520 relkind = "parititioned_table";
521 break;
522 case RELKIND_PARTITIONED_INDEX:
523 relkind = "parititioned_index";
524 break;
525 case '\0':
526 relkind = NULL;
527 break;
528 default:
529 relkind = psprintf("%c", rte->relkind);
530 break;
531 }
532
533 /* If there is a relkind, show it */
534 if (relkind != NULL)
535 ExplainPropertyText("Relation Kind", relkind, es);
536
537 /* If there is a lock mode, show it */
538 if (rte->rellockmode != 0)
539 ExplainPropertyText("Relation Lock Mode",
541 rte->rellockmode), es);
542
543 /*
544 * If there is a perminfoindex, show it. We don't try to display
545 * information from the RTEPermissionInfo node here because they are
546 * just indexes plannedstmt->permInfos which could be separately
547 * dumped if someone wants to add EXPLAIN (PERMISSIONS) or similar.
548 */
549 if (rte->perminfoindex != 0)
550 ExplainPropertyInteger("Permission Info Index", NULL,
551 rte->perminfoindex, es);
552
553 /*
554 * add_rte_to_flat_rtable will clear rte->tablesample and
555 * rte->subquery in the finished plan, so skip those fields.
556 *
557 * However, the security_barrier flag is not shown by the core code,
558 * so let's print it here.
559 */
560 if (es->format != EXPLAIN_FORMAT_TEXT || rte->security_barrier)
561 ExplainPropertyBool("Security Barrier", rte->security_barrier, es);
562
563 /*
564 * If this is a join, print out the fields that are specifically valid
565 * for joins.
566 */
567 if (rte->rtekind == RTE_JOIN)
568 {
569 char *jointype;
570
571 switch (rte->jointype)
572 {
573 case JOIN_INNER:
574 jointype = "Inner";
575 break;
576 case JOIN_LEFT:
577 jointype = "Left";
578 break;
579 case JOIN_FULL:
580 jointype = "Full";
581 break;
582 case JOIN_RIGHT:
583 jointype = "Right";
584 break;
585 case JOIN_SEMI:
586 jointype = "Semi";
587 break;
588 case JOIN_ANTI:
589 jointype = "Anti";
590 break;
591 case JOIN_RIGHT_SEMI:
592 jointype = "Right Semi";
593 break;
594 case JOIN_RIGHT_ANTI:
595 jointype = "Right Anti";
596 break;
597 default:
598 jointype = "???";
599 break;
600 }
601
602 /* Join type */
603 ExplainPropertyText("Join Type", jointype, es);
604
605 /* # of JOIN USING columns */
606 if (es->format != EXPLAIN_FORMAT_TEXT || rte->joinmergedcols != 0)
607 ExplainPropertyInteger("JOIN USING Columns", NULL,
608 rte->joinmergedcols, es);
609
610 /*
611 * add_rte_to_flat_rtable will clear joinaliasvars, joinleftcols,
612 * joinrightcols, and join_using_alias here, so skip those fields.
613 */
614 }
615
616 /*
617 * add_rte_to_flat_rtable will clear functions, tablefunc, and
618 * values_lists, but we can display funcordinality.
619 */
620 if (rte->rtekind == RTE_FUNCTION)
621 ExplainPropertyBool("WITH ORDINALITY", rte->funcordinality, es);
622
623 /*
624 * If this is a CTE, print out CTE-related properties.
625 */
626 if (rte->rtekind == RTE_CTE)
627 {
628 ExplainPropertyText("CTE Name", rte->ctename, es);
629 ExplainPropertyUInteger("CTE Levels Up", NULL, rte->ctelevelsup,
630 es);
631 ExplainPropertyBool("CTE Self-Reference", rte->self_reference, es);
632 }
633
634 /*
635 * add_rte_to_flat_rtable will clear coltypes, coltypemods, and
636 * colcollations, so skip those fields.
637 *
638 * If this is an ephemeral named relation, print out ENR-related
639 * properties.
640 */
641 if (rte->rtekind == RTE_NAMEDTUPLESTORE)
642 {
643 ExplainPropertyText("ENR Name", rte->enrname, es);
644 ExplainPropertyFloat("ENR Tuples", NULL, rte->enrtuples, 0, es);
645 }
646
647 /*
648 * add_rte_to_flat_rtable will clear groupexprs and securityQuals, so
649 * skip that field. We have handled inFromCl above, so the only thing
650 * left to handle here is rte->lateral.
651 */
652 if (es->format != EXPLAIN_FORMAT_TEXT || rte->lateral)
653 ExplainPropertyBool("Lateral", rte->lateral, es);
654
655 /* Done with this RTE */
656 if (es->format == EXPLAIN_FORMAT_TEXT)
657 es->indent--;
658 ExplainCloseGroup("Range Table Entry", NULL, true, es);
659 }
660
661 /* Print PlannedStmt fields that contain RTIs. */
662 if (es->format != EXPLAIN_FORMAT_TEXT ||
663 !bms_is_empty(plannedstmt->unprunableRelids))
664 overexplain_bitmapset("Unprunable RTIs", plannedstmt->unprunableRelids,
665 es);
666 if (es->format != EXPLAIN_FORMAT_TEXT ||
667 plannedstmt->resultRelations != NIL)
668 overexplain_intlist("Result RTIs", plannedstmt->resultRelations, es);
669
670 /* Close group, we're all done */
671 ExplainCloseGroup("Range Table", "Range Table", false, es);
672}
unsigned int Index
Definition: c.h:585
void ExplainPropertyUInteger(const char *qlabel, const char *unit, uint64 value, ExplainState *es)
void ExplainPropertyFloat(const char *qlabel, const char *unit, double value, int ndigits, ExplainState *es)
const char * GetLockmodeName(LOCKMETHODID lockmethodid, LOCKMODE mode)
Definition: lock.c:4220
#define DEFAULT_LOCKMETHOD
Definition: lock.h:126
char * get_rel_name(Oid relid)
Definition: lsyscache.c:2011
Oid get_rel_namespace(Oid relid)
Definition: lsyscache.c:2035
char * get_namespace_name_or_temp(Oid nspid)
Definition: lsyscache.c:3473
@ JOIN_SEMI
Definition: nodes.h:313
@ JOIN_FULL
Definition: nodes.h:301
@ JOIN_INNER
Definition: nodes.h:299
@ JOIN_RIGHT
Definition: nodes.h:302
@ JOIN_RIGHT_SEMI
Definition: nodes.h:315
@ JOIN_LEFT
Definition: nodes.h:300
@ JOIN_RIGHT_ANTI
Definition: nodes.h:316
@ JOIN_ANTI
Definition: nodes.h:314
@ RTE_JOIN
Definition: parsenodes.h:1028
@ RTE_CTE
Definition: parsenodes.h:1032
@ RTE_NAMEDTUPLESTORE
Definition: parsenodes.h:1033
@ RTE_VALUES
Definition: parsenodes.h:1031
@ RTE_SUBQUERY
Definition: parsenodes.h:1027
@ RTE_RESULT
Definition: parsenodes.h:1034
@ RTE_FUNCTION
Definition: parsenodes.h:1029
@ RTE_TABLEFUNC
Definition: parsenodes.h:1030
@ RTE_GROUP
Definition: parsenodes.h:1037
@ RTE_RELATION
Definition: parsenodes.h:1026
#define rt_fetch(rangetable_index, rangetable)
Definition: parsetree.h:31
NameData relname
Definition: pg_class.h:38
static int list_length(const List *l)
Definition: pg_list.h:152
static void overexplain_alias(const char *qlabel, Alias *alias, ExplainState *es)
unsigned int Oid
Definition: postgres_ext.h:30
List * resultRelations
Definition: plannodes.h:106
Bitmapset * unprunableRelids
Definition: plannodes.h:97
List * rtable
Definition: plannodes.h:91
char * ctename
Definition: parsenodes.h:1210
Index ctelevelsup
Definition: parsenodes.h:1212
bool funcordinality
Definition: parsenodes.h:1193
char * enrname
Definition: parsenodes.h:1245
JoinType jointype
Definition: parsenodes.h:1165
RTEKind rtekind
Definition: parsenodes.h:1061

References appendStringInfo(), bms_is_empty, RangeTblEntry::ctelevelsup, RangeTblEntry::ctename, DEFAULT_LOCKMETHOD, RangeTblEntry::enrname, EXPLAIN_FORMAT_TEXT, ExplainCloseGroup(), ExplainIndentText(), ExplainOpenGroup(), ExplainPropertyBool(), ExplainPropertyFloat(), ExplainPropertyInteger(), ExplainPropertyText(), ExplainPropertyUInteger(), ExplainState::format, RangeTblEntry::funcordinality, get_namespace_name_or_temp(), get_rel_name(), get_rel_namespace(), GetLockmodeName(), ExplainState::indent, RangeTblEntry::inh, JOIN_ANTI, JOIN_FULL, JOIN_INNER, JOIN_LEFT, JOIN_RIGHT, JOIN_RIGHT_ANTI, JOIN_RIGHT_SEMI, JOIN_SEMI, RangeTblEntry::jointype, list_length(), NIL, overexplain_alias(), overexplain_bitmapset(), overexplain_intlist(), psprintf(), quote_identifier(), relname, PlannedStmt::resultRelations, rt_fetch, PlannedStmt::rtable, RTE_CTE, RTE_FUNCTION, RTE_GROUP, RTE_JOIN, RTE_NAMEDTUPLESTORE, RTE_RELATION, RTE_RESULT, RTE_SUBQUERY, RTE_TABLEFUNC, RTE_VALUES, RangeTblEntry::rtekind, ExplainState::str, PlannedStmt::unprunableRelids, and ExplainState::verbose.

Referenced by overexplain_per_plan_hook().

◆ overexplain_range_table_handler()

static void overexplain_range_table_handler ( ExplainState es,
DefElem opt,
ParseState pstate 
)
static

Definition at line 120 of file pg_overexplain.c.

122{
124
125 options->range_table = defGetBoolean(opt);
126}

References defGetBoolean(), and overexplain_ensure_options().

Referenced by _PG_init().

◆ PG_MODULE_MAGIC_EXT()

PG_MODULE_MAGIC_EXT ( name = "pg_overexplain",
version = PG_VERSION 
)

Variable Documentation

◆ es_extension_id

int es_extension_id
static

◆ prev_explain_per_node_hook

explain_per_node_hook_type prev_explain_per_node_hook
static

Definition at line 61 of file pg_overexplain.c.

Referenced by _PG_init(), and overexplain_per_node_hook().

◆ prev_explain_per_plan_hook

explain_per_plan_hook_type prev_explain_per_plan_hook
static

Definition at line 62 of file pg_overexplain.c.

Referenced by _PG_init(), and overexplain_per_plan_hook().