sl@0
|
1 |
/*
|
sl@0
|
2 |
** 2007 May 05
|
sl@0
|
3 |
**
|
sl@0
|
4 |
** The author disclaims copyright to this source code. In place of
|
sl@0
|
5 |
** a legal notice, here is a blessing:
|
sl@0
|
6 |
**
|
sl@0
|
7 |
** May you do good and not evil.
|
sl@0
|
8 |
** May you find forgiveness for yourself and forgive others.
|
sl@0
|
9 |
** May you share freely, never taking more than you give.
|
sl@0
|
10 |
**
|
sl@0
|
11 |
*************************************************************************
|
sl@0
|
12 |
** Code for testing the btree.c module in SQLite. This code
|
sl@0
|
13 |
** is not included in the SQLite library. It is used for automated
|
sl@0
|
14 |
** testing of the SQLite library.
|
sl@0
|
15 |
**
|
sl@0
|
16 |
** $Id: test_btree.c,v 1.8 2008/09/29 11:49:48 danielk1977 Exp $
|
sl@0
|
17 |
*/
|
sl@0
|
18 |
#include "btreeInt.h"
|
sl@0
|
19 |
#include "tcl.h"
|
sl@0
|
20 |
|
sl@0
|
21 |
/*
|
sl@0
|
22 |
** Usage: sqlite3_shared_cache_report
|
sl@0
|
23 |
**
|
sl@0
|
24 |
** Return a list of file that are shared and the number of
|
sl@0
|
25 |
** references to each file.
|
sl@0
|
26 |
*/
|
sl@0
|
27 |
int sqlite3BtreeSharedCacheReport(
|
sl@0
|
28 |
void * clientData,
|
sl@0
|
29 |
Tcl_Interp *interp,
|
sl@0
|
30 |
int objc,
|
sl@0
|
31 |
Tcl_Obj *CONST objv[]
|
sl@0
|
32 |
){
|
sl@0
|
33 |
#ifndef SQLITE_OMIT_SHARED_CACHE
|
sl@0
|
34 |
extern BtShared *sqlite3SharedCacheList;
|
sl@0
|
35 |
BtShared *pBt;
|
sl@0
|
36 |
Tcl_Obj *pRet = Tcl_NewObj();
|
sl@0
|
37 |
for(pBt=GLOBAL(BtShared*,sqlite3SharedCacheList); pBt; pBt=pBt->pNext){
|
sl@0
|
38 |
const char *zFile = sqlite3PagerFilename(pBt->pPager);
|
sl@0
|
39 |
Tcl_ListObjAppendElement(interp, pRet, Tcl_NewStringObj(zFile, -1));
|
sl@0
|
40 |
Tcl_ListObjAppendElement(interp, pRet, Tcl_NewIntObj(pBt->nRef));
|
sl@0
|
41 |
}
|
sl@0
|
42 |
Tcl_SetObjResult(interp, pRet);
|
sl@0
|
43 |
#endif
|
sl@0
|
44 |
return TCL_OK;
|
sl@0
|
45 |
}
|
sl@0
|
46 |
|
sl@0
|
47 |
/*
|
sl@0
|
48 |
** Print debugging information about all cursors to standard output.
|
sl@0
|
49 |
*/
|
sl@0
|
50 |
void sqlite3BtreeCursorList(Btree *p){
|
sl@0
|
51 |
#ifdef SQLITE_DEBUG
|
sl@0
|
52 |
BtCursor *pCur;
|
sl@0
|
53 |
BtShared *pBt = p->pBt;
|
sl@0
|
54 |
for(pCur=pBt->pCursor; pCur; pCur=pCur->pNext){
|
sl@0
|
55 |
MemPage *pPage = pCur->apPage[pCur->iPage];
|
sl@0
|
56 |
char *zMode = pCur->wrFlag ? "rw" : "ro";
|
sl@0
|
57 |
sqlite3DebugPrintf("CURSOR %p rooted at %4d(%s) currently at %d.%d%s\n",
|
sl@0
|
58 |
pCur, pCur->pgnoRoot, zMode,
|
sl@0
|
59 |
pPage ? pPage->pgno : 0, pCur->aiIdx[pCur->iPage],
|
sl@0
|
60 |
(pCur->eState==CURSOR_VALID) ? "" : " eof"
|
sl@0
|
61 |
);
|
sl@0
|
62 |
}
|
sl@0
|
63 |
#endif
|
sl@0
|
64 |
}
|
sl@0
|
65 |
|
sl@0
|
66 |
|
sl@0
|
67 |
/*
|
sl@0
|
68 |
** Fill aResult[] with information about the entry and page that the
|
sl@0
|
69 |
** cursor is pointing to.
|
sl@0
|
70 |
**
|
sl@0
|
71 |
** aResult[0] = The page number
|
sl@0
|
72 |
** aResult[1] = The entry number
|
sl@0
|
73 |
** aResult[2] = Total number of entries on this page
|
sl@0
|
74 |
** aResult[3] = Cell size (local payload + header)
|
sl@0
|
75 |
** aResult[4] = Number of free bytes on this page
|
sl@0
|
76 |
** aResult[5] = Number of free blocks on the page
|
sl@0
|
77 |
** aResult[6] = Total payload size (local + overflow)
|
sl@0
|
78 |
** aResult[7] = Header size in bytes
|
sl@0
|
79 |
** aResult[8] = Local payload size
|
sl@0
|
80 |
** aResult[9] = Parent page number
|
sl@0
|
81 |
** aResult[10]= Page number of the first overflow page
|
sl@0
|
82 |
**
|
sl@0
|
83 |
** This routine is used for testing and debugging only.
|
sl@0
|
84 |
*/
|
sl@0
|
85 |
int sqlite3BtreeCursorInfo(BtCursor *pCur, int *aResult, int upCnt){
|
sl@0
|
86 |
#if 0
|
sl@0
|
87 |
int cnt, idx;
|
sl@0
|
88 |
MemPage *pPage = pCur->apPage[pCur->iPage];
|
sl@0
|
89 |
BtCursor tmpCur;
|
sl@0
|
90 |
int rc;
|
sl@0
|
91 |
|
sl@0
|
92 |
if( pCur->eState==CURSOR_REQUIRESEEK ){
|
sl@0
|
93 |
rc = sqlite3BtreeRestoreCursorPosition(pCur);
|
sl@0
|
94 |
if( rc!=SQLITE_OK ){
|
sl@0
|
95 |
return rc;
|
sl@0
|
96 |
}
|
sl@0
|
97 |
}
|
sl@0
|
98 |
|
sl@0
|
99 |
assert( pPage->isInit );
|
sl@0
|
100 |
sqlite3BtreeGetTempCursor(pCur, &tmpCur);
|
sl@0
|
101 |
while( upCnt-- ){
|
sl@0
|
102 |
sqlite3BtreeMoveToParent(&tmpCur);
|
sl@0
|
103 |
}
|
sl@0
|
104 |
pPage = tmpCur.pPage;
|
sl@0
|
105 |
aResult[0] = sqlite3PagerPagenumber(pPage->pDbPage);
|
sl@0
|
106 |
assert( aResult[0]==pPage->pgno );
|
sl@0
|
107 |
aResult[1] = tmpCur.idx;
|
sl@0
|
108 |
aResult[2] = pPage->nCell;
|
sl@0
|
109 |
if( tmpCur.idx>=0 && tmpCur.idx<pPage->nCell ){
|
sl@0
|
110 |
sqlite3BtreeParseCell(tmpCur.pPage, tmpCur.idx, &tmpCur.info);
|
sl@0
|
111 |
aResult[3] = tmpCur.info.nSize;
|
sl@0
|
112 |
aResult[6] = tmpCur.info.nData;
|
sl@0
|
113 |
aResult[7] = tmpCur.info.nHeader;
|
sl@0
|
114 |
aResult[8] = tmpCur.info.nLocal;
|
sl@0
|
115 |
}else{
|
sl@0
|
116 |
aResult[3] = 0;
|
sl@0
|
117 |
aResult[6] = 0;
|
sl@0
|
118 |
aResult[7] = 0;
|
sl@0
|
119 |
aResult[8] = 0;
|
sl@0
|
120 |
}
|
sl@0
|
121 |
aResult[4] = pPage->nFree;
|
sl@0
|
122 |
cnt = 0;
|
sl@0
|
123 |
idx = get2byte(&pPage->aData[pPage->hdrOffset+1]);
|
sl@0
|
124 |
while( idx>0 && idx<pPage->pBt->usableSize ){
|
sl@0
|
125 |
cnt++;
|
sl@0
|
126 |
idx = get2byte(&pPage->aData[idx]);
|
sl@0
|
127 |
}
|
sl@0
|
128 |
aResult[5] = cnt;
|
sl@0
|
129 |
if( pPage->pParent==0 || sqlite3BtreeIsRootPage(pPage) ){
|
sl@0
|
130 |
aResult[9] = 0;
|
sl@0
|
131 |
}else{
|
sl@0
|
132 |
aResult[9] = pPage->pParent->pgno;
|
sl@0
|
133 |
}
|
sl@0
|
134 |
if( tmpCur.info.iOverflow ){
|
sl@0
|
135 |
aResult[10] = get4byte(&tmpCur.info.pCell[tmpCur.info.iOverflow]);
|
sl@0
|
136 |
}else{
|
sl@0
|
137 |
aResult[10] = 0;
|
sl@0
|
138 |
}
|
sl@0
|
139 |
sqlite3BtreeReleaseTempCursor(&tmpCur);
|
sl@0
|
140 |
#endif
|
sl@0
|
141 |
return SQLITE_OK;
|
sl@0
|
142 |
}
|