Extension shipped with Pg variables
view in debug view container.
It shows up while debugging and allows investigation of types
with special processing of some.
While debug session there is Pg variables
view available in debug view container.
It contains all variables that available in default variables
view, but
stored variables are processed by extension.
Description of processing methods are below.
This view refreshed after each step, but you can force refresh it using refresh button in right upper corner.
Each pointer variable are checked: if NULL
- no processing performed
and it can not be expanded (expand button not shown).
But there are can be false positive - when there are garbage (i.e. for
not initialized stack allocated) with value like 0x4
. Extension
treat it like normal and tries to expand. No errors observed yet,
but be careful.
Main feature of this extension is investigation of variables in PostgreSQL source code.
It involves several cases (kind of types):
- Node type
- Special member
- Simple type (or member)
This kind of type is plain old type without extra meaning:
- Built-in types
- Pointers to structs or built-in types
- Simple structs
These processed in usual way (as in default variables view).
Examples:
int
float
_Bool
struct NameData
typedef NameData *Name
Node type - is a type that may have NodeTag
first field. I.e. List
, PlannerInfo
etc...
Info about these type is taken from files, that may have this enum: src/include/nodes/node.h
, src/include/nodes/nodetags.h
.
They are parsed on extension activation.
This is configuration parameter and can be modified, but default
values seems to be ok.
Default processing done in this way:
- Encounter variable with Node type in it
- Get real NodeTag - cast to
Node *
and gettype
field - Cast variable to extracted type if:
- Extracted
NodeTag
is valid - Extracted
NodeTag
differs from declared type
- Extracted
After getting real NodeTag, variable is casted to corresponding type and all members shown for real type (not declared).
Example:
- In some
walker
function argument isNode *
, but inPg variables
view it will be casted to real type. Path *
variables will be shown according real subtype, i.e.AggPath *
If real NodeTag differs from corresponding from declared type, real NodeTag is shown right after type in square brackets.
Example:
First condition in step 3 to check if there is garbage
Also, there are some Special Node types:
-
List - show
elements
members according to tagList
IntList
OidList
XidList
Additionally, you can specify custom pointer type in
List
. See customList
types section in configuration file documentation. -
Bitmapset - show elements of set in pseudo-member
$elements$
WARNING: to obtain elements of set we use debugger evaluation of
bms_next_member
function. If there is breakpoint in - we will fall into infinite recursion and crash.To prevent this, we do not introspect set if there are any breakpoints in
bitmapset.c
source file (even if breakpoint set not inbms_next_member
function). Currently, I did not find any way to check this function directly.
Examples:
NOTE: if there were garbage, extension can mistake and cast variable to wrong NodeTag. For example,
List
can be casted toPathTarget
. It is up to you to detect such errors for now.
Relids
- is a typedef for Bitmapset *
, but it does not have
corresponding NodeTag.
There is support for such cases - aliases.
In configuration file (start from 2 version) there is top level
alises
field - array of type aliases info.
Example:
You create extension and have typedef for PlannerInfo
-
typedef PlannerInfo *Sample
. By default, extension will
not detect this, because there is not T_Sample
tag.
You should add this to config file:
{
"version": 2,
"alises": [
{
"alias": "Sample",
"type": "PlannerInfo *"
}
]
}
NOTE: this only works if
type
is Node type
Special member - is a member of type that must be handled in special way (with custom logic).
All special members info stored in "specialMembers"
field in config file.
For now, there is only 1 kind of special member - array special member (asm).
ASM - is a member of some struct that stored pointer to array of some type
with it's length stored in another member.
It stored in "array"
field in config file.
Example:
You have this struct:
struct Sample
{
struct PlannerInfo **planners;
int planners_count;
}
planners
- is an array with length of planners_count
. To add support for
this you should update config file like this:
{
"version": 2,
"specialMembers": {
"array": [
{
"typeName": "Sample",
"memberName": "planners",
"lengthExpression": "planners_count"
}
]
}
}
There are about 36 supported ASM. For example, simple_rel_array
for PlannerInfo
For more info about configuration check documentation