4 ** The author disclaims copyright to this source code. In place of
5 ** a legal notice, here is a blessing:
7 ** May you do good and not evil.
8 ** May you find forgiveness for yourself and forgive others.
9 ** May you share freely, never taking more than you give.
11 *************************************************************************
13 ** This file contains C code to implement the TreeView debugging routines.
14 ** These routines print a parse tree to standard output for debugging and
17 ** The interfaces in this file is only available when compiling
20 #include "sqliteInt.h"
24 ** Add a new subitem to the tree. The moreToFollow flag indicates that this
25 ** is not the last item in the tree.
27 static void sqlite3TreeViewPush(TreeView
**pp
, u8 moreToFollow
){
30 *pp
= p
= sqlite3_malloc64( sizeof(*p
) );
32 memset(p
, 0, sizeof(*p
));
36 assert( moreToFollow
==0 || moreToFollow
==1 );
37 if( p
->iLevel
<(int)sizeof(p
->bLine
) ) p
->bLine
[p
->iLevel
] = moreToFollow
;
41 ** Finished with one layer of the tree
43 static void sqlite3TreeViewPop(TreeView
**pp
){
54 ** Generate a single line of output for the tree, with a prefix that contains
55 ** all the appropriate tree lines
57 void sqlite3TreeViewLine(TreeView
*p
, const char *zFormat
, ...){
62 sqlite3StrAccumInit(&acc
, 0, zBuf
, sizeof(zBuf
), 0);
64 for(i
=0; i
<p
->iLevel
&& i
<(int)sizeof(p
->bLine
)-1; i
++){
65 sqlite3_str_append(&acc
, p
->bLine
[i
] ? "| " : " ", 4);
67 sqlite3_str_append(&acc
, p
->bLine
[i
] ? "|-- " : "'-- ", 4);
70 va_start(ap
, zFormat
);
71 sqlite3_str_vappendf(&acc
, zFormat
, ap
);
73 assert( acc
.nChar
>0 || acc
.accError
);
74 sqlite3_str_append(&acc
, "\n", 1);
76 sqlite3StrAccumFinish(&acc
);
77 fprintf(stdout
,"%s", zBuf
);
82 ** Shorthand for starting a new tree item that consists of a single label
84 static void sqlite3TreeViewItem(TreeView
*p
, const char *zLabel
,u8 moreFollows
){
85 sqlite3TreeViewPush(&p
, moreFollows
);
86 sqlite3TreeViewLine(p
, "%s", zLabel
);
90 ** Show a list of Column objects in tree format.
92 void sqlite3TreeViewColumnList(
99 sqlite3TreeViewPush(&pView
, moreToFollow
);
100 sqlite3TreeViewLine(pView
, "COLUMNS");
101 for(i
=0; i
<nCol
; i
++){
102 u16 flg
= aCol
[i
].colFlags
;
103 int colMoreToFollow
= i
<(nCol
- 1);
104 sqlite3TreeViewPush(&pView
, colMoreToFollow
);
105 sqlite3TreeViewLine(pView
, 0);
106 printf(" %s", aCol
[i
].zCnName
);
107 switch( aCol
[i
].eCType
){
108 case COLTYPE_ANY
: printf(" ANY"); break;
109 case COLTYPE_BLOB
: printf(" BLOB"); break;
110 case COLTYPE_INT
: printf(" INT"); break;
111 case COLTYPE_INTEGER
: printf(" INTEGER"); break;
112 case COLTYPE_REAL
: printf(" REAL"); break;
113 case COLTYPE_TEXT
: printf(" TEXT"); break;
114 case COLTYPE_CUSTOM
: {
115 if( flg
& COLFLAG_HASTYPE
){
116 const char *z
= aCol
[i
].zCnName
;
123 if( flg
& COLFLAG_PRIMKEY
) printf(" PRIMARY KEY");
124 if( flg
& COLFLAG_HIDDEN
) printf(" HIDDEN");
125 #ifdef COLFLAG_NOEXPAND
126 if( flg
& COLFLAG_NOEXPAND
) printf(" NO-EXPAND");
128 if( flg
) printf(" flags=%04x", flg
);
131 sqlite3TreeViewPop(&pView
);
133 sqlite3TreeViewPop(&pView
);
137 ** Generate a human-readable description of a WITH clause.
139 void sqlite3TreeViewWith(TreeView
*pView
, const With
*pWith
, u8 moreToFollow
){
141 if( pWith
==0 ) return;
142 if( pWith
->nCte
==0 ) return;
144 sqlite3TreeViewLine(pView
, "WITH (0x%p, pOuter=0x%p)",pWith
,pWith
->pOuter
);
146 sqlite3TreeViewLine(pView
, "WITH (0x%p)", pWith
);
149 sqlite3TreeViewPush(&pView
, moreToFollow
);
150 for(i
=0; i
<pWith
->nCte
; i
++){
153 const struct Cte
*pCte
= &pWith
->a
[i
];
154 sqlite3StrAccumInit(&x
, 0, zLine
, sizeof(zLine
), 0);
155 sqlite3_str_appendf(&x
, "%s", pCte
->zName
);
156 if( pCte
->pCols
&& pCte
->pCols
->nExpr
>0 ){
159 for(j
=0; j
<pCte
->pCols
->nExpr
; j
++){
160 sqlite3_str_appendf(&x
, "%c%s", cSep
, pCte
->pCols
->a
[j
].zEName
);
163 sqlite3_str_appendf(&x
, ")");
165 if( pCte
->eM10d
!=M10d_Any
){
166 sqlite3_str_appendf(&x
, " %sMATERIALIZED",
167 pCte
->eM10d
==M10d_No
? "NOT " : "");
170 sqlite3_str_appendf(&x
, " (pUse=0x%p, nUse=%d)", pCte
->pUse
,
173 sqlite3StrAccumFinish(&x
);
174 sqlite3TreeViewItem(pView
, zLine
, i
<pWith
->nCte
-1);
175 sqlite3TreeViewSelect(pView
, pCte
->pSelect
, 0);
176 sqlite3TreeViewPop(&pView
);
178 sqlite3TreeViewPop(&pView
);
183 ** Generate a human-readable description of a SrcList object.
185 void sqlite3TreeViewSrcList(TreeView
*pView
, const SrcList
*pSrc
){
187 if( pSrc
==0 ) return;
188 for(i
=0; i
<pSrc
->nSrc
; i
++){
189 const SrcItem
*pItem
= &pSrc
->a
[i
];
193 sqlite3StrAccumInit(&x
, 0, zLine
, sizeof(zLine
), 0);
194 x
.printfFlags
|= SQLITE_PRINTF_INTERNAL
;
195 sqlite3_str_appendf(&x
, "{%d:*} %!S", pItem
->iCursor
, pItem
);
197 sqlite3_str_appendf(&x
, " tab=%Q nCol=%d ptr=%p used=%llx",
198 pItem
->pTab
->zName
, pItem
->pTab
->nCol
, pItem
->pTab
, pItem
->colUsed
);
200 if( (pItem
->fg
.jointype
& (JT_LEFT
|JT_RIGHT
))==(JT_LEFT
|JT_RIGHT
) ){
201 sqlite3_str_appendf(&x
, " FULL-OUTER-JOIN");
202 }else if( pItem
->fg
.jointype
& JT_LEFT
){
203 sqlite3_str_appendf(&x
, " LEFT-JOIN");
204 }else if( pItem
->fg
.jointype
& JT_RIGHT
){
205 sqlite3_str_appendf(&x
, " RIGHT-JOIN");
206 }else if( pItem
->fg
.jointype
& JT_CROSS
){
207 sqlite3_str_appendf(&x
, " CROSS-JOIN");
209 if( pItem
->fg
.jointype
& JT_LTORJ
){
210 sqlite3_str_appendf(&x
, " LTORJ");
212 if( pItem
->fg
.fromDDL
){
213 sqlite3_str_appendf(&x
, " DDL");
215 if( pItem
->fg
.isCte
){
216 sqlite3_str_appendf(&x
, " CteUse=0x%p", pItem
->u2
.pCteUse
);
218 if( pItem
->fg
.isOn
|| (pItem
->fg
.isUsing
==0 && pItem
->u3
.pOn
!=0) ){
219 sqlite3_str_appendf(&x
, " ON");
221 if( pItem
->fg
.isTabFunc
) sqlite3_str_appendf(&x
, " isTabFunc");
222 if( pItem
->fg
.isCorrelated
) sqlite3_str_appendf(&x
, " isCorrelated");
223 if( pItem
->fg
.isMaterialized
) sqlite3_str_appendf(&x
, " isMaterialized");
224 if( pItem
->fg
.viaCoroutine
) sqlite3_str_appendf(&x
, " viaCoroutine");
225 if( pItem
->fg
.notCte
) sqlite3_str_appendf(&x
, " notCte");
226 if( pItem
->fg
.isNestedFrom
) sqlite3_str_appendf(&x
, " isNestedFrom");
228 sqlite3StrAccumFinish(&x
);
229 sqlite3TreeViewItem(pView
, zLine
, i
<pSrc
->nSrc
-1);
231 if( pItem
->pSelect
) n
++;
232 if( pItem
->fg
.isTabFunc
) n
++;
233 if( pItem
->fg
.isUsing
) n
++;
234 if( pItem
->fg
.isUsing
){
235 sqlite3TreeViewIdList(pView
, pItem
->u3
.pUsing
, (--n
)>0, "USING");
237 if( pItem
->pSelect
){
239 Table
*pTab
= pItem
->pTab
;
240 sqlite3TreeViewColumnList(pView
, pTab
->aCol
, pTab
->nCol
, 1);
242 assert( (int)pItem
->fg
.isNestedFrom
== IsNestedFrom(pItem
->pSelect
) );
243 sqlite3TreeViewSelect(pView
, pItem
->pSelect
, (--n
)>0);
245 if( pItem
->fg
.isTabFunc
){
246 sqlite3TreeViewExprList(pView
, pItem
->u1
.pFuncArg
, 0, "func-args:");
248 sqlite3TreeViewPop(&pView
);
253 ** Generate a human-readable description of a Select object.
255 void sqlite3TreeViewSelect(TreeView
*pView
, const Select
*p
, u8 moreToFollow
){
259 sqlite3TreeViewLine(pView
, "nil-SELECT");
262 sqlite3TreeViewPush(&pView
, moreToFollow
);
264 sqlite3TreeViewWith(pView
, p
->pWith
, 1);
266 sqlite3TreeViewPush(&pView
, 1);
269 if( p
->selFlags
& SF_WhereBegin
){
270 sqlite3TreeViewLine(pView
, "sqlite3WhereBegin()");
272 sqlite3TreeViewLine(pView
,
273 "SELECT%s%s (%u/%p) selFlags=0x%x nSelectRow=%d",
274 ((p
->selFlags
& SF_Distinct
) ? " DISTINCT" : ""),
275 ((p
->selFlags
& SF_Aggregate
) ? " agg_flag" : ""),
276 p
->selId
, p
, p
->selFlags
,
280 if( cnt
++ ) sqlite3TreeViewPop(&pView
);
285 if( p
->pSrc
&& p
->pSrc
->nSrc
) n
++;
287 if( p
->pGroupBy
) n
++;
288 if( p
->pHaving
) n
++;
289 if( p
->pOrderBy
) n
++;
291 #ifndef SQLITE_OMIT_WINDOWFUNC
293 if( p
->pWinDefn
) n
++;
297 sqlite3TreeViewExprList(pView
, p
->pEList
, n
>0, "result-set");
300 #ifndef SQLITE_OMIT_WINDOWFUNC
303 sqlite3TreeViewPush(&pView
, (n
--)>0);
304 sqlite3TreeViewLine(pView
, "window-functions");
305 for(pX
=p
->pWin
; pX
; pX
=pX
->pNextWin
){
306 sqlite3TreeViewWinFunc(pView
, pX
, pX
->pNextWin
!=0);
308 sqlite3TreeViewPop(&pView
);
311 if( p
->pSrc
&& p
->pSrc
->nSrc
){
312 sqlite3TreeViewPush(&pView
, (n
--)>0);
313 sqlite3TreeViewLine(pView
, "FROM");
314 sqlite3TreeViewSrcList(pView
, p
->pSrc
);
315 sqlite3TreeViewPop(&pView
);
318 sqlite3TreeViewItem(pView
, "WHERE", (n
--)>0);
319 sqlite3TreeViewExpr(pView
, p
->pWhere
, 0);
320 sqlite3TreeViewPop(&pView
);
323 sqlite3TreeViewExprList(pView
, p
->pGroupBy
, (n
--)>0, "GROUPBY");
326 sqlite3TreeViewItem(pView
, "HAVING", (n
--)>0);
327 sqlite3TreeViewExpr(pView
, p
->pHaving
, 0);
328 sqlite3TreeViewPop(&pView
);
330 #ifndef SQLITE_OMIT_WINDOWFUNC
333 sqlite3TreeViewItem(pView
, "WINDOW", (n
--)>0);
334 for(pX
=p
->pWinDefn
; pX
; pX
=pX
->pNextWin
){
335 sqlite3TreeViewWindow(pView
, pX
, pX
->pNextWin
!=0);
337 sqlite3TreeViewPop(&pView
);
341 sqlite3TreeViewExprList(pView
, p
->pOrderBy
, (n
--)>0, "ORDERBY");
344 sqlite3TreeViewItem(pView
, "LIMIT", (n
--)>0);
345 sqlite3TreeViewExpr(pView
, p
->pLimit
->pLeft
, p
->pLimit
->pRight
!=0);
346 if( p
->pLimit
->pRight
){
347 sqlite3TreeViewItem(pView
, "OFFSET", (n
--)>0);
348 sqlite3TreeViewExpr(pView
, p
->pLimit
->pRight
, 0);
349 sqlite3TreeViewPop(&pView
);
351 sqlite3TreeViewPop(&pView
);
354 const char *zOp
= "UNION";
356 case TK_ALL
: zOp
= "UNION ALL"; break;
357 case TK_INTERSECT
: zOp
= "INTERSECT"; break;
358 case TK_EXCEPT
: zOp
= "EXCEPT"; break;
360 sqlite3TreeViewItem(pView
, zOp
, 1);
364 sqlite3TreeViewPop(&pView
);
367 #ifndef SQLITE_OMIT_WINDOWFUNC
369 ** Generate a description of starting or stopping bounds
371 void sqlite3TreeViewBound(
372 TreeView
*pView
, /* View context */
373 u8 eBound
, /* UNBOUNDED, CURRENT, PRECEDING, FOLLOWING */
374 Expr
*pExpr
, /* Value for PRECEDING or FOLLOWING */
375 u8 moreToFollow
/* True if more to follow */
379 sqlite3TreeViewItem(pView
, "UNBOUNDED", moreToFollow
);
380 sqlite3TreeViewPop(&pView
);
384 sqlite3TreeViewItem(pView
, "CURRENT", moreToFollow
);
385 sqlite3TreeViewPop(&pView
);
389 sqlite3TreeViewItem(pView
, "PRECEDING", moreToFollow
);
390 sqlite3TreeViewExpr(pView
, pExpr
, 0);
391 sqlite3TreeViewPop(&pView
);
395 sqlite3TreeViewItem(pView
, "FOLLOWING", moreToFollow
);
396 sqlite3TreeViewExpr(pView
, pExpr
, 0);
397 sqlite3TreeViewPop(&pView
);
402 #endif /* SQLITE_OMIT_WINDOWFUNC */
404 #ifndef SQLITE_OMIT_WINDOWFUNC
406 ** Generate a human-readable explanation for a Window object
408 void sqlite3TreeViewWindow(TreeView
*pView
, const Window
*pWin
, u8 more
){
410 if( pWin
==0 ) return;
412 sqlite3TreeViewItem(pView
, "FILTER", 1);
413 sqlite3TreeViewExpr(pView
, pWin
->pFilter
, 0);
414 sqlite3TreeViewPop(&pView
);
416 sqlite3TreeViewPush(&pView
, more
);
418 sqlite3TreeViewLine(pView
, "OVER %s (%p)", pWin
->zName
, pWin
);
420 sqlite3TreeViewLine(pView
, "OVER (%p)", pWin
);
422 if( pWin
->zBase
) nElement
++;
423 if( pWin
->pOrderBy
) nElement
++;
424 if( pWin
->eFrmType
) nElement
++;
425 if( pWin
->eExclude
) nElement
++;
427 sqlite3TreeViewPush(&pView
, (--nElement
)>0);
428 sqlite3TreeViewLine(pView
, "window: %s", pWin
->zBase
);
429 sqlite3TreeViewPop(&pView
);
431 if( pWin
->pPartition
){
432 sqlite3TreeViewExprList(pView
, pWin
->pPartition
, nElement
>0,"PARTITION-BY");
434 if( pWin
->pOrderBy
){
435 sqlite3TreeViewExprList(pView
, pWin
->pOrderBy
, (--nElement
)>0, "ORDER-BY");
437 if( pWin
->eFrmType
){
439 const char *zFrmType
= "ROWS";
440 if( pWin
->eFrmType
==TK_RANGE
) zFrmType
= "RANGE";
441 if( pWin
->eFrmType
==TK_GROUPS
) zFrmType
= "GROUPS";
442 sqlite3_snprintf(sizeof(zBuf
),zBuf
,"%s%s",zFrmType
,
443 pWin
->bImplicitFrame
? " (implied)" : "");
444 sqlite3TreeViewItem(pView
, zBuf
, (--nElement
)>0);
445 sqlite3TreeViewBound(pView
, pWin
->eStart
, pWin
->pStart
, 1);
446 sqlite3TreeViewBound(pView
, pWin
->eEnd
, pWin
->pEnd
, 0);
447 sqlite3TreeViewPop(&pView
);
449 if( pWin
->eExclude
){
451 const char *zExclude
;
452 switch( pWin
->eExclude
){
453 case TK_NO
: zExclude
= "NO OTHERS"; break;
454 case TK_CURRENT
: zExclude
= "CURRENT ROW"; break;
455 case TK_GROUP
: zExclude
= "GROUP"; break;
456 case TK_TIES
: zExclude
= "TIES"; break;
458 sqlite3_snprintf(sizeof(zBuf
),zBuf
,"invalid(%d)", pWin
->eExclude
);
462 sqlite3TreeViewPush(&pView
, 0);
463 sqlite3TreeViewLine(pView
, "EXCLUDE %s", zExclude
);
464 sqlite3TreeViewPop(&pView
);
466 sqlite3TreeViewPop(&pView
);
468 #endif /* SQLITE_OMIT_WINDOWFUNC */
470 #ifndef SQLITE_OMIT_WINDOWFUNC
472 ** Generate a human-readable explanation for a Window Function object
474 void sqlite3TreeViewWinFunc(TreeView
*pView
, const Window
*pWin
, u8 more
){
475 if( pWin
==0 ) return;
476 sqlite3TreeViewPush(&pView
, more
);
477 sqlite3TreeViewLine(pView
, "WINFUNC %s(%d)",
478 pWin
->pWFunc
->zName
, pWin
->pWFunc
->nArg
);
479 sqlite3TreeViewWindow(pView
, pWin
, 0);
480 sqlite3TreeViewPop(&pView
);
482 #endif /* SQLITE_OMIT_WINDOWFUNC */
485 ** Generate a human-readable explanation of an expression tree.
487 void sqlite3TreeViewExpr(TreeView
*pView
, const Expr
*pExpr
, u8 moreToFollow
){
488 const char *zBinOp
= 0; /* Binary operator */
489 const char *zUniOp
= 0; /* Unary operator */
491 sqlite3TreeViewPush(&pView
, moreToFollow
);
493 sqlite3TreeViewLine(pView
, "nil");
494 sqlite3TreeViewPop(&pView
);
497 if( pExpr
->flags
|| pExpr
->affExpr
|| pExpr
->vvaFlags
|| pExpr
->pAggInfo
){
499 sqlite3StrAccumInit(&x
, 0, zFlgs
, sizeof(zFlgs
), 0);
500 sqlite3_str_appendf(&x
, " fg.af=%x.%c",
501 pExpr
->flags
, pExpr
->affExpr
? pExpr
->affExpr
: 'n');
502 if( ExprHasProperty(pExpr
, EP_OuterON
) ){
503 sqlite3_str_appendf(&x
, " outer.iJoin=%d", pExpr
->w
.iJoin
);
505 if( ExprHasProperty(pExpr
, EP_InnerON
) ){
506 sqlite3_str_appendf(&x
, " inner.iJoin=%d", pExpr
->w
.iJoin
);
508 if( ExprHasProperty(pExpr
, EP_FromDDL
) ){
509 sqlite3_str_appendf(&x
, " DDL");
511 if( ExprHasVVAProperty(pExpr
, EP_Immutable
) ){
512 sqlite3_str_appendf(&x
, " IMMUTABLE");
514 if( pExpr
->pAggInfo
!=0 ){
515 sqlite3_str_appendf(&x
, " agg-column[%d]", pExpr
->iAgg
);
517 sqlite3StrAccumFinish(&x
);
522 case TK_AGG_COLUMN
: {
523 sqlite3TreeViewLine(pView
, "AGG{%d:%d}%s",
524 pExpr
->iTable
, pExpr
->iColumn
, zFlgs
);
528 if( pExpr
->iTable
<0 ){
529 /* This only happens when coding check constraints */
532 sqlite3_snprintf(sizeof(zOp2
),zOp2
," op2=0x%02x",pExpr
->op2
);
536 sqlite3TreeViewLine(pView
, "COLUMN(%d)%s%s",
537 pExpr
->iColumn
, zFlgs
, zOp2
);
539 assert( ExprUseYTab(pExpr
) );
540 sqlite3TreeViewLine(pView
, "{%d:%d} pTab=%p%s",
541 pExpr
->iTable
, pExpr
->iColumn
,
542 pExpr
->y
.pTab
, zFlgs
);
544 if( ExprHasProperty(pExpr
, EP_FixedCol
) ){
545 sqlite3TreeViewExpr(pView
, pExpr
->pLeft
, 0);
550 if( pExpr
->flags
& EP_IntValue
){
551 sqlite3TreeViewLine(pView
, "%d", pExpr
->u
.iValue
);
553 sqlite3TreeViewLine(pView
, "%s", pExpr
->u
.zToken
);
557 #ifndef SQLITE_OMIT_FLOATING_POINT
559 assert( !ExprHasProperty(pExpr
, EP_IntValue
) );
560 sqlite3TreeViewLine(pView
,"%s", pExpr
->u
.zToken
);
565 assert( !ExprHasProperty(pExpr
, EP_IntValue
) );
566 sqlite3TreeViewLine(pView
,"%Q", pExpr
->u
.zToken
);
570 sqlite3TreeViewLine(pView
,"NULL");
574 sqlite3TreeViewLine(pView
,"%s%s",
575 sqlite3ExprTruthValue(pExpr
) ? "TRUE" : "FALSE", zFlgs
);
578 #ifndef SQLITE_OMIT_BLOB_LITERAL
580 assert( !ExprHasProperty(pExpr
, EP_IntValue
) );
581 sqlite3TreeViewLine(pView
,"%s", pExpr
->u
.zToken
);
586 assert( !ExprHasProperty(pExpr
, EP_IntValue
) );
587 sqlite3TreeViewLine(pView
,"VARIABLE(%s,%d)",
588 pExpr
->u
.zToken
, pExpr
->iColumn
);
592 sqlite3TreeViewLine(pView
,"REGISTER(%d)", pExpr
->iTable
);
596 assert( !ExprHasProperty(pExpr
, EP_IntValue
) );
597 sqlite3TreeViewLine(pView
,"ID \"%w\"", pExpr
->u
.zToken
);
600 #ifndef SQLITE_OMIT_CAST
602 /* Expressions of the form: CAST(pLeft AS token) */
603 assert( !ExprHasProperty(pExpr
, EP_IntValue
) );
604 sqlite3TreeViewLine(pView
,"CAST %Q", pExpr
->u
.zToken
);
605 sqlite3TreeViewExpr(pView
, pExpr
->pLeft
, 0);
608 #endif /* SQLITE_OMIT_CAST */
609 case TK_LT
: zBinOp
= "LT"; break;
610 case TK_LE
: zBinOp
= "LE"; break;
611 case TK_GT
: zBinOp
= "GT"; break;
612 case TK_GE
: zBinOp
= "GE"; break;
613 case TK_NE
: zBinOp
= "NE"; break;
614 case TK_EQ
: zBinOp
= "EQ"; break;
615 case TK_IS
: zBinOp
= "IS"; break;
616 case TK_ISNOT
: zBinOp
= "ISNOT"; break;
617 case TK_AND
: zBinOp
= "AND"; break;
618 case TK_OR
: zBinOp
= "OR"; break;
619 case TK_PLUS
: zBinOp
= "ADD"; break;
620 case TK_STAR
: zBinOp
= "MUL"; break;
621 case TK_MINUS
: zBinOp
= "SUB"; break;
622 case TK_REM
: zBinOp
= "REM"; break;
623 case TK_BITAND
: zBinOp
= "BITAND"; break;
624 case TK_BITOR
: zBinOp
= "BITOR"; break;
625 case TK_SLASH
: zBinOp
= "DIV"; break;
626 case TK_LSHIFT
: zBinOp
= "LSHIFT"; break;
627 case TK_RSHIFT
: zBinOp
= "RSHIFT"; break;
628 case TK_CONCAT
: zBinOp
= "CONCAT"; break;
629 case TK_DOT
: zBinOp
= "DOT"; break;
630 case TK_LIMIT
: zBinOp
= "LIMIT"; break;
632 case TK_UMINUS
: zUniOp
= "UMINUS"; break;
633 case TK_UPLUS
: zUniOp
= "UPLUS"; break;
634 case TK_BITNOT
: zUniOp
= "BITNOT"; break;
635 case TK_NOT
: zUniOp
= "NOT"; break;
636 case TK_ISNULL
: zUniOp
= "ISNULL"; break;
637 case TK_NOTNULL
: zUniOp
= "NOTNULL"; break;
641 const char *azOp
[] = {
642 "IS-FALSE", "IS-TRUE", "IS-NOT-FALSE", "IS-NOT-TRUE"
644 assert( pExpr
->op2
==TK_IS
|| pExpr
->op2
==TK_ISNOT
);
645 assert( pExpr
->pRight
);
646 assert( sqlite3ExprSkipCollateAndLikely(pExpr
->pRight
)->op
648 x
= (pExpr
->op2
==TK_ISNOT
)*2 + sqlite3ExprTruthValue(pExpr
->pRight
);
654 assert( !ExprHasProperty(pExpr
, EP_IntValue
) );
655 sqlite3TreeViewLine(pView
, "SPAN %Q", pExpr
->u
.zToken
);
656 sqlite3TreeViewExpr(pView
, pExpr
->pLeft
, 0);
661 /* COLLATE operators without the EP_Collate flag are intended to
662 ** emulate collation associated with a table column. These show
663 ** up in the treeview output as "SOFT-COLLATE". Explicit COLLATE
664 ** operators that appear in the original SQL always have the
665 ** EP_Collate bit set and appear in treeview output as just "COLLATE" */
666 assert( !ExprHasProperty(pExpr
, EP_IntValue
) );
667 sqlite3TreeViewLine(pView
, "%sCOLLATE %Q%s",
668 !ExprHasProperty(pExpr
, EP_Collate
) ? "SOFT-" : "",
669 pExpr
->u
.zToken
, zFlgs
);
670 sqlite3TreeViewExpr(pView
, pExpr
->pLeft
, 0);
674 case TK_AGG_FUNCTION
:
676 ExprList
*pFarg
; /* List of function arguments */
678 if( ExprHasProperty(pExpr
, EP_TokenOnly
) ){
682 assert( ExprUseXList(pExpr
) );
683 pFarg
= pExpr
->x
.pList
;
684 #ifndef SQLITE_OMIT_WINDOWFUNC
685 pWin
= ExprHasProperty(pExpr
, EP_WinFunc
) ? pExpr
->y
.pWin
: 0;
690 assert( !ExprHasProperty(pExpr
, EP_IntValue
) );
691 if( pExpr
->op
==TK_AGG_FUNCTION
){
692 sqlite3TreeViewLine(pView
, "AGG_FUNCTION%d %Q%s agg=%d[%d]/%p",
693 pExpr
->op2
, pExpr
->u
.zToken
, zFlgs
,
694 pExpr
->pAggInfo
? pExpr
->pAggInfo
->selId
: 0,
695 pExpr
->iAgg
, pExpr
->pAggInfo
);
696 }else if( pExpr
->op2
!=0 ){
699 sqlite3_snprintf(sizeof(zBuf
),zBuf
,"0x%02x",pExpr
->op2
);
701 if( pExpr
->op2
==NC_IsCheck
) zOp2
= "NC_IsCheck";
702 if( pExpr
->op2
==NC_IdxExpr
) zOp2
= "NC_IdxExpr";
703 if( pExpr
->op2
==NC_PartIdx
) zOp2
= "NC_PartIdx";
704 if( pExpr
->op2
==NC_GenCol
) zOp2
= "NC_GenCol";
705 sqlite3TreeViewLine(pView
, "FUNCTION %Q%s op2=%s",
706 pExpr
->u
.zToken
, zFlgs
, zOp2
);
708 sqlite3TreeViewLine(pView
, "FUNCTION %Q%s", pExpr
->u
.zToken
, zFlgs
);
711 sqlite3TreeViewExprList(pView
, pFarg
, pWin
!=0, 0);
713 #ifndef SQLITE_OMIT_WINDOWFUNC
715 sqlite3TreeViewWindow(pView
, pWin
, 0);
720 #ifndef SQLITE_OMIT_SUBQUERY
722 assert( ExprUseXSelect(pExpr
) );
723 sqlite3TreeViewLine(pView
, "EXISTS-expr flags=0x%x", pExpr
->flags
);
724 sqlite3TreeViewSelect(pView
, pExpr
->x
.pSelect
, 0);
728 assert( ExprUseXSelect(pExpr
) );
729 sqlite3TreeViewLine(pView
, "subquery-expr flags=0x%x", pExpr
->flags
);
730 sqlite3TreeViewSelect(pView
, pExpr
->x
.pSelect
, 0);
734 sqlite3_str
*pStr
= sqlite3_str_new(0);
736 sqlite3_str_appendf(pStr
, "IN flags=0x%x", pExpr
->flags
);
737 if( pExpr
->iTable
) sqlite3_str_appendf(pStr
, " iTable=%d",pExpr
->iTable
);
738 if( ExprHasProperty(pExpr
, EP_Subrtn
) ){
739 sqlite3_str_appendf(pStr
, " subrtn(%d,%d)",
740 pExpr
->y
.sub
.regReturn
, pExpr
->y
.sub
.iAddr
);
742 z
= sqlite3_str_finish(pStr
);
743 sqlite3TreeViewLine(pView
, z
);
745 sqlite3TreeViewExpr(pView
, pExpr
->pLeft
, 1);
746 if( ExprUseXSelect(pExpr
) ){
747 sqlite3TreeViewSelect(pView
, pExpr
->x
.pSelect
, 0);
749 sqlite3TreeViewExprList(pView
, pExpr
->x
.pList
, 0, 0);
753 #endif /* SQLITE_OMIT_SUBQUERY */
758 ** This is equivalent to
762 ** X is stored in pExpr->pLeft.
763 ** Y is stored in pExpr->pList->a[0].pExpr.
764 ** Z is stored in pExpr->pList->a[1].pExpr.
767 const Expr
*pX
, *pY
, *pZ
;
769 assert( ExprUseXList(pExpr
) );
770 assert( pExpr
->x
.pList
->nExpr
==2 );
771 pY
= pExpr
->x
.pList
->a
[0].pExpr
;
772 pZ
= pExpr
->x
.pList
->a
[1].pExpr
;
773 sqlite3TreeViewLine(pView
, "BETWEEN");
774 sqlite3TreeViewExpr(pView
, pX
, 1);
775 sqlite3TreeViewExpr(pView
, pY
, 1);
776 sqlite3TreeViewExpr(pView
, pZ
, 0);
780 /* If the opcode is TK_TRIGGER, then the expression is a reference
781 ** to a column in the new.* or old.* pseudo-tables available to
782 ** trigger programs. In this case Expr.iTable is set to 1 for the
783 ** new.* pseudo-table, or 0 for the old.* pseudo-table. Expr.iColumn
784 ** is set to the column of the pseudo-table to read, or to -1 to
785 ** read the rowid field.
787 sqlite3TreeViewLine(pView
, "%s(%d)",
788 pExpr
->iTable
? "NEW" : "OLD", pExpr
->iColumn
);
792 sqlite3TreeViewLine(pView
, "CASE");
793 sqlite3TreeViewExpr(pView
, pExpr
->pLeft
, 1);
794 assert( ExprUseXList(pExpr
) );
795 sqlite3TreeViewExprList(pView
, pExpr
->x
.pList
, 0, 0);
798 #ifndef SQLITE_OMIT_TRIGGER
800 const char *zType
= "unk";
801 switch( pExpr
->affExpr
){
802 case OE_Rollback
: zType
= "rollback"; break;
803 case OE_Abort
: zType
= "abort"; break;
804 case OE_Fail
: zType
= "fail"; break;
805 case OE_Ignore
: zType
= "ignore"; break;
807 assert( !ExprHasProperty(pExpr
, EP_IntValue
) );
808 sqlite3TreeViewLine(pView
, "RAISE %s(%Q)", zType
, pExpr
->u
.zToken
);
813 sqlite3TreeViewLine(pView
, "MATCH {%d:%d}%s",
814 pExpr
->iTable
, pExpr
->iColumn
, zFlgs
);
815 sqlite3TreeViewExpr(pView
, pExpr
->pRight
, 0);
819 char *z
= sqlite3_mprintf("VECTOR%s",zFlgs
);
820 assert( ExprUseXList(pExpr
) );
821 sqlite3TreeViewBareExprList(pView
, pExpr
->x
.pList
, z
);
825 case TK_SELECT_COLUMN
: {
826 sqlite3TreeViewLine(pView
, "SELECT-COLUMN %d of [0..%d]%s",
827 pExpr
->iColumn
, pExpr
->iTable
-1,
828 pExpr
->pRight
==pExpr
->pLeft
? " (SELECT-owner)" : "");
829 assert( ExprUseXSelect(pExpr
->pLeft
) );
830 sqlite3TreeViewSelect(pView
, pExpr
->pLeft
->x
.pSelect
, 0);
833 case TK_IF_NULL_ROW
: {
834 sqlite3TreeViewLine(pView
, "IF-NULL-ROW %d", pExpr
->iTable
);
835 sqlite3TreeViewExpr(pView
, pExpr
->pLeft
, 0);
840 sqlite3TreeViewLine(pView
, "ERROR");
843 sqlite3TreeViewExpr(pView
, &tmp
, 0);
847 if( pExpr
->iColumn
<=0 ){
848 sqlite3TreeViewLine(pView
, "First FROM table rowid");
850 sqlite3TreeViewLine(pView
, "First FROM table column %d",
856 sqlite3TreeViewLine(pView
, "op=%d", pExpr
->op
);
861 sqlite3TreeViewLine(pView
, "%s%s", zBinOp
, zFlgs
);
862 sqlite3TreeViewExpr(pView
, pExpr
->pLeft
, 1);
863 sqlite3TreeViewExpr(pView
, pExpr
->pRight
, 0);
865 sqlite3TreeViewLine(pView
, "%s%s", zUniOp
, zFlgs
);
866 sqlite3TreeViewExpr(pView
, pExpr
->pLeft
, 0);
868 sqlite3TreeViewPop(&pView
);
873 ** Generate a human-readable explanation of an expression list.
875 void sqlite3TreeViewBareExprList(
877 const ExprList
*pList
,
880 if( zLabel
==0 || zLabel
[0]==0 ) zLabel
= "LIST";
882 sqlite3TreeViewLine(pView
, "%s (empty)", zLabel
);
885 sqlite3TreeViewLine(pView
, "%s", zLabel
);
886 for(i
=0; i
<pList
->nExpr
; i
++){
887 int j
= pList
->a
[i
].u
.x
.iOrderByCol
;
888 char *zName
= pList
->a
[i
].zEName
;
889 int moreToFollow
= i
<pList
->nExpr
- 1;
891 sqlite3TreeViewPush(&pView
, moreToFollow
);
893 sqlite3TreeViewLine(pView
, 0);
895 switch( pList
->a
[i
].fg
.eEName
){
897 fprintf(stdout
, "AS %s ", zName
);
900 fprintf(stdout
, "TABLE-ALIAS-NAME(\"%s\") ", zName
);
901 if( pList
->a
[i
].fg
.bUsed
) fprintf(stdout
, "(used) ");
902 if( pList
->a
[i
].fg
.bUsingTerm
) fprintf(stdout
, "(USING-term) ");
903 if( pList
->a
[i
].fg
.bNoExpand
) fprintf(stdout
, "(NoExpand) ");
906 fprintf(stdout
, "SPAN(\"%s\") ", zName
);
911 fprintf(stdout
, "iOrderByCol=%d", j
);
913 fprintf(stdout
, "\n");
916 sqlite3TreeViewExpr(pView
, pList
->a
[i
].pExpr
, moreToFollow
);
918 sqlite3TreeViewPop(&pView
);
923 void sqlite3TreeViewExprList(
925 const ExprList
*pList
,
929 sqlite3TreeViewPush(&pView
, moreToFollow
);
930 sqlite3TreeViewBareExprList(pView
, pList
, zLabel
);
931 sqlite3TreeViewPop(&pView
);
935 ** Generate a human-readable explanation of an id-list.
937 void sqlite3TreeViewBareIdList(
942 if( zLabel
==0 || zLabel
[0]==0 ) zLabel
= "LIST";
944 sqlite3TreeViewLine(pView
, "%s (empty)", zLabel
);
947 sqlite3TreeViewLine(pView
, "%s", zLabel
);
948 for(i
=0; i
<pList
->nId
; i
++){
949 char *zName
= pList
->a
[i
].zName
;
950 int moreToFollow
= i
<pList
->nId
- 1;
951 if( zName
==0 ) zName
= "(null)";
952 sqlite3TreeViewPush(&pView
, moreToFollow
);
953 sqlite3TreeViewLine(pView
, 0);
954 if( pList
->eU4
==EU4_NONE
){
955 fprintf(stdout
, "%s\n", zName
);
956 }else if( pList
->eU4
==EU4_IDX
){
957 fprintf(stdout
, "%s (%d)\n", zName
, pList
->a
[i
].u4
.idx
);
959 assert( pList
->eU4
==EU4_EXPR
);
960 if( pList
->a
[i
].u4
.pExpr
==0 ){
961 fprintf(stdout
, "%s (pExpr=NULL)\n", zName
);
963 fprintf(stdout
, "%s\n", zName
);
964 sqlite3TreeViewPush(&pView
, i
<pList
->nId
-1);
965 sqlite3TreeViewExpr(pView
, pList
->a
[i
].u4
.pExpr
, 0);
966 sqlite3TreeViewPop(&pView
);
969 sqlite3TreeViewPop(&pView
);
973 void sqlite3TreeViewIdList(
979 sqlite3TreeViewPush(&pView
, moreToFollow
);
980 sqlite3TreeViewBareIdList(pView
, pList
, zLabel
);
981 sqlite3TreeViewPop(&pView
);
985 ** Generate a human-readable explanation of a list of Upsert objects
987 void sqlite3TreeViewUpsert(
989 const Upsert
*pUpsert
,
992 if( pUpsert
==0 ) return;
993 sqlite3TreeViewPush(&pView
, moreToFollow
);
996 sqlite3TreeViewPush(&pView
, pUpsert
->pNextUpsert
!=0 || moreToFollow
);
997 sqlite3TreeViewLine(pView
, "ON CONFLICT DO %s",
998 pUpsert
->isDoUpdate
? "UPDATE" : "NOTHING");
999 n
= (pUpsert
->pUpsertSet
!=0) + (pUpsert
->pUpsertWhere
!=0);
1000 sqlite3TreeViewExprList(pView
, pUpsert
->pUpsertTarget
, (n
--)>0, "TARGET");
1001 sqlite3TreeViewExprList(pView
, pUpsert
->pUpsertSet
, (n
--)>0, "SET");
1002 if( pUpsert
->pUpsertWhere
){
1003 sqlite3TreeViewItem(pView
, "WHERE", (n
--)>0);
1004 sqlite3TreeViewExpr(pView
, pUpsert
->pUpsertWhere
, 0);
1005 sqlite3TreeViewPop(&pView
);
1007 sqlite3TreeViewPop(&pView
);
1008 pUpsert
= pUpsert
->pNextUpsert
;
1010 sqlite3TreeViewPop(&pView
);
1013 #if TREETRACE_ENABLED
1015 ** Generate a human-readable diagram of the data structure that go
1016 ** into generating an DELETE statement.
1018 void sqlite3TreeViewDelete(
1020 const SrcList
*pTabList
,
1022 const ExprList
*pOrderBy
,
1024 const Trigger
*pTrigger
1027 TreeView
*pView
= 0;
1028 sqlite3TreeViewPush(&pView
, 0);
1029 sqlite3TreeViewLine(pView
, "DELETE");
1037 sqlite3TreeViewPush(&pView
, (--n
)>0);
1038 sqlite3TreeViewWith(pView
, pWith
, 0);
1039 sqlite3TreeViewPop(&pView
);
1042 sqlite3TreeViewPush(&pView
, (--n
)>0);
1043 sqlite3TreeViewLine(pView
, "FROM");
1044 sqlite3TreeViewSrcList(pView
, pTabList
);
1045 sqlite3TreeViewPop(&pView
);
1048 sqlite3TreeViewPush(&pView
, (--n
)>0);
1049 sqlite3TreeViewLine(pView
, "WHERE");
1050 sqlite3TreeViewExpr(pView
, pWhere
, 0);
1051 sqlite3TreeViewPop(&pView
);
1054 sqlite3TreeViewExprList(pView
, pOrderBy
, (--n
)>0, "ORDER-BY");
1057 sqlite3TreeViewPush(&pView
, (--n
)>0);
1058 sqlite3TreeViewLine(pView
, "LIMIT");
1059 sqlite3TreeViewExpr(pView
, pLimit
, 0);
1060 sqlite3TreeViewPop(&pView
);
1063 sqlite3TreeViewTrigger(pView
, pTrigger
, (--n
)>0, 1);
1065 sqlite3TreeViewPop(&pView
);
1067 #endif /* TREETRACE_ENABLED */
1069 #if TREETRACE_ENABLED
1071 ** Generate a human-readable diagram of the data structure that go
1072 ** into generating an INSERT statement.
1074 void sqlite3TreeViewInsert(
1076 const SrcList
*pTabList
,
1077 const IdList
*pColumnList
,
1078 const Select
*pSelect
,
1079 const ExprList
*pExprList
,
1081 const Upsert
*pUpsert
,
1082 const Trigger
*pTrigger
1084 TreeView
*pView
= 0;
1086 const char *zLabel
= "INSERT";
1088 case OE_Replace
: zLabel
= "REPLACE"; break;
1089 case OE_Ignore
: zLabel
= "INSERT OR IGNORE"; break;
1090 case OE_Rollback
: zLabel
= "INSERT OR ROLLBACK"; break;
1091 case OE_Abort
: zLabel
= "INSERT OR ABORT"; break;
1092 case OE_Fail
: zLabel
= "INSERT OR FAIL"; break;
1094 sqlite3TreeViewPush(&pView
, 0);
1095 sqlite3TreeViewLine(pView
, zLabel
);
1098 if( pColumnList
) n
++;
1100 if( pExprList
) n
++;
1104 sqlite3TreeViewPush(&pView
, (--n
)>0);
1105 sqlite3TreeViewWith(pView
, pWith
, 0);
1106 sqlite3TreeViewPop(&pView
);
1109 sqlite3TreeViewPush(&pView
, (--n
)>0);
1110 sqlite3TreeViewLine(pView
, "INTO");
1111 sqlite3TreeViewSrcList(pView
, pTabList
);
1112 sqlite3TreeViewPop(&pView
);
1115 sqlite3TreeViewIdList(pView
, pColumnList
, (--n
)>0, "COLUMNS");
1118 sqlite3TreeViewPush(&pView
, (--n
)>0);
1119 sqlite3TreeViewLine(pView
, "DATA-SOURCE");
1120 sqlite3TreeViewSelect(pView
, pSelect
, 0);
1121 sqlite3TreeViewPop(&pView
);
1124 sqlite3TreeViewExprList(pView
, pExprList
, (--n
)>0, "VALUES");
1127 sqlite3TreeViewPush(&pView
, (--n
)>0);
1128 sqlite3TreeViewLine(pView
, "UPSERT");
1129 sqlite3TreeViewUpsert(pView
, pUpsert
, 0);
1130 sqlite3TreeViewPop(&pView
);
1133 sqlite3TreeViewTrigger(pView
, pTrigger
, (--n
)>0, 1);
1135 sqlite3TreeViewPop(&pView
);
1137 #endif /* TREETRACE_ENABLED */
1139 #if TREETRACE_ENABLED
1141 ** Generate a human-readable diagram of the data structure that go
1142 ** into generating an UPDATE statement.
1144 void sqlite3TreeViewUpdate(
1146 const SrcList
*pTabList
,
1147 const ExprList
*pChanges
,
1150 const ExprList
*pOrderBy
,
1152 const Upsert
*pUpsert
,
1153 const Trigger
*pTrigger
1156 TreeView
*pView
= 0;
1157 const char *zLabel
= "UPDATE";
1159 case OE_Replace
: zLabel
= "UPDATE OR REPLACE"; break;
1160 case OE_Ignore
: zLabel
= "UPDATE OR IGNORE"; break;
1161 case OE_Rollback
: zLabel
= "UPDATE OR ROLLBACK"; break;
1162 case OE_Abort
: zLabel
= "UPDATE OR ABORT"; break;
1163 case OE_Fail
: zLabel
= "UPDATE OR FAIL"; break;
1165 sqlite3TreeViewPush(&pView
, 0);
1166 sqlite3TreeViewLine(pView
, zLabel
);
1176 sqlite3TreeViewPush(&pView
, (--n
)>0);
1177 sqlite3TreeViewWith(pView
, pWith
, 0);
1178 sqlite3TreeViewPop(&pView
);
1181 sqlite3TreeViewPush(&pView
, (--n
)>0);
1182 sqlite3TreeViewLine(pView
, "FROM");
1183 sqlite3TreeViewSrcList(pView
, pTabList
);
1184 sqlite3TreeViewPop(&pView
);
1187 sqlite3TreeViewExprList(pView
, pChanges
, (--n
)>0, "SET");
1190 sqlite3TreeViewPush(&pView
, (--n
)>0);
1191 sqlite3TreeViewLine(pView
, "WHERE");
1192 sqlite3TreeViewExpr(pView
, pWhere
, 0);
1193 sqlite3TreeViewPop(&pView
);
1196 sqlite3TreeViewExprList(pView
, pOrderBy
, (--n
)>0, "ORDER-BY");
1199 sqlite3TreeViewPush(&pView
, (--n
)>0);
1200 sqlite3TreeViewLine(pView
, "LIMIT");
1201 sqlite3TreeViewExpr(pView
, pLimit
, 0);
1202 sqlite3TreeViewPop(&pView
);
1205 sqlite3TreeViewPush(&pView
, (--n
)>0);
1206 sqlite3TreeViewLine(pView
, "UPSERT");
1207 sqlite3TreeViewUpsert(pView
, pUpsert
, 0);
1208 sqlite3TreeViewPop(&pView
);
1211 sqlite3TreeViewTrigger(pView
, pTrigger
, (--n
)>0, 1);
1213 sqlite3TreeViewPop(&pView
);
1215 #endif /* TREETRACE_ENABLED */
1217 #ifndef SQLITE_OMIT_TRIGGER
1219 ** Show a human-readable graph of a TriggerStep
1221 void sqlite3TreeViewTriggerStep(
1223 const TriggerStep
*pStep
,
1228 if( pStep
==0 ) return;
1229 sqlite3TreeViewPush(&pView
,
1230 moreToFollow
|| (showFullList
&& pStep
->pNext
!=0));
1232 if( cnt
++ && pStep
->pNext
==0 ){
1233 sqlite3TreeViewPop(&pView
);
1234 sqlite3TreeViewPush(&pView
, 0);
1236 sqlite3TreeViewLine(pView
, "%s", pStep
->zSpan
? pStep
->zSpan
: "RETURNING");
1237 }while( showFullList
&& (pStep
= pStep
->pNext
)!=0 );
1238 sqlite3TreeViewPop(&pView
);
1242 ** Show a human-readable graph of a Trigger
1244 void sqlite3TreeViewTrigger(
1246 const Trigger
*pTrigger
,
1251 if( pTrigger
==0 ) return;
1252 sqlite3TreeViewPush(&pView
,
1253 moreToFollow
|| (showFullList
&& pTrigger
->pNext
!=0));
1255 if( cnt
++ && pTrigger
->pNext
==0 ){
1256 sqlite3TreeViewPop(&pView
);
1257 sqlite3TreeViewPush(&pView
, 0);
1259 sqlite3TreeViewLine(pView
, "TRIGGER %s", pTrigger
->zName
);
1260 sqlite3TreeViewPush(&pView
, 0);
1261 sqlite3TreeViewTriggerStep(pView
, pTrigger
->step_list
, 0, 1);
1262 sqlite3TreeViewPop(&pView
);
1263 }while( showFullList
&& (pTrigger
= pTrigger
->pNext
)!=0 );
1264 sqlite3TreeViewPop(&pView
);
1266 #endif /* SQLITE_OMIT_TRIGGER */
1270 ** These simplified versions of the tree-view routines omit unnecessary
1271 ** parameters. These variants are intended to be used from a symbolic
1272 ** debugger, such as "gdb", during interactive debugging sessions.
1274 ** This routines are given external linkage so that they will always be
1275 ** accessible to the debugging, and to avoid warnings about unused
1276 ** functions. But these routines only exist in debugging builds, so they
1277 ** do not contaminate the interface.
1279 void sqlite3ShowExpr(const Expr
*p
){ sqlite3TreeViewExpr(0,p
,0); }
1280 void sqlite3ShowExprList(const ExprList
*p
){ sqlite3TreeViewExprList(0,p
,0,0);}
1281 void sqlite3ShowIdList(const IdList
*p
){ sqlite3TreeViewIdList(0,p
,0,0); }
1282 void sqlite3ShowSrcList(const SrcList
*p
){ sqlite3TreeViewSrcList(0,p
); }
1283 void sqlite3ShowSelect(const Select
*p
){ sqlite3TreeViewSelect(0,p
,0); }
1284 void sqlite3ShowWith(const With
*p
){ sqlite3TreeViewWith(0,p
,0); }
1285 void sqlite3ShowUpsert(const Upsert
*p
){ sqlite3TreeViewUpsert(0,p
,0); }
1286 #ifndef SQLITE_OMIT_TRIGGER
1287 void sqlite3ShowTriggerStep(const TriggerStep
*p
){
1288 sqlite3TreeViewTriggerStep(0,p
,0,0);
1290 void sqlite3ShowTriggerStepList(const TriggerStep
*p
){
1291 sqlite3TreeViewTriggerStep(0,p
,0,1);
1293 void sqlite3ShowTrigger(const Trigger
*p
){ sqlite3TreeViewTrigger(0,p
,0,0); }
1294 void sqlite3ShowTriggerList(const Trigger
*p
){ sqlite3TreeViewTrigger(0,p
,0,1);}
1296 #ifndef SQLITE_OMIT_WINDOWFUNC
1297 void sqlite3ShowWindow(const Window
*p
){ sqlite3TreeViewWindow(0,p
,0); }
1298 void sqlite3ShowWinFunc(const Window
*p
){ sqlite3TreeViewWinFunc(0,p
,0); }
1301 #endif /* SQLITE_DEBUG */