Update contrib.
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 *************************************************************************
12 ** Code for testing the btree.c module in SQLite. This code
13 ** is not included in the SQLite library. It is used for automated
14 ** testing of the SQLite library.
16 ** $Id: test3.c,v 1.101 2008/08/13 19:11:48 drh Exp $
18 #include "sqliteInt.h"
25 ** Interpret an SQLite error number
27 static char *errorName(int rc){
30 case SQLITE_OK: zName = "SQLITE_OK"; break;
31 case SQLITE_ERROR: zName = "SQLITE_ERROR"; break;
32 case SQLITE_PERM: zName = "SQLITE_PERM"; break;
33 case SQLITE_ABORT: zName = "SQLITE_ABORT"; break;
34 case SQLITE_BUSY: zName = "SQLITE_BUSY"; break;
35 case SQLITE_NOMEM: zName = "SQLITE_NOMEM"; break;
36 case SQLITE_READONLY: zName = "SQLITE_READONLY"; break;
37 case SQLITE_INTERRUPT: zName = "SQLITE_INTERRUPT"; break;
38 case SQLITE_IOERR: zName = "SQLITE_IOERR"; break;
39 case SQLITE_CORRUPT: zName = "SQLITE_CORRUPT"; break;
40 case SQLITE_FULL: zName = "SQLITE_FULL"; break;
41 case SQLITE_CANTOPEN: zName = "SQLITE_CANTOPEN"; break;
42 case SQLITE_PROTOCOL: zName = "SQLITE_PROTOCOL"; break;
43 case SQLITE_EMPTY: zName = "SQLITE_EMPTY"; break;
44 case SQLITE_LOCKED: zName = "SQLITE_LOCKED"; break;
45 default: zName = "SQLITE_Unknown"; break;
51 ** A bogus sqlite3 connection structure for use in the btree
55 static int nRefSqlite3 = 0;
58 ** Usage: btree_open FILENAME NCACHE FLAGS
60 ** Open a new database
62 static int btree_open(
64 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
65 int argc, /* Number of arguments */
66 const char **argv /* Text of each argument */
69 int rc, nCache, flags;
72 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
73 " FILENAME NCACHE FLAGS\"", 0);
76 if( Tcl_GetInt(interp, argv[2], &nCache) ) return TCL_ERROR;
77 if( Tcl_GetInt(interp, argv[3], &flags) ) return TCL_ERROR;
80 sDb.pVfs = sqlite3_vfs_find(0);
81 sDb.mutex = sqlite3MutexAlloc(SQLITE_MUTEX_RECURSIVE);
82 sqlite3_mutex_enter(sDb.mutex);
84 rc = sqlite3BtreeOpen(argv[1], &sDb, &pBt, flags,
85 SQLITE_OPEN_READWRITE | SQLITE_OPEN_CREATE | SQLITE_OPEN_MAIN_DB);
87 Tcl_AppendResult(interp, errorName(rc), 0);
90 sqlite3BtreeSetCacheSize(pBt, nCache);
91 sqlite3_snprintf(sizeof(zBuf), zBuf,"%p", pBt);
92 Tcl_AppendResult(interp, zBuf, 0);
97 ** Usage: btree_close ID
99 ** Close the given database.
101 static int btree_close(
103 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
104 int argc, /* Number of arguments */
105 const char **argv /* Text of each argument */
110 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
114 pBt = sqlite3TestTextToPtr(argv[1]);
115 rc = sqlite3BtreeClose(pBt);
117 Tcl_AppendResult(interp, errorName(rc), 0);
121 if( nRefSqlite3==0 ){
122 sqlite3_mutex_leave(sDb.mutex);
123 sqlite3_mutex_free(sDb.mutex);
132 ** Usage: btree_begin_transaction ID
134 ** Start a new transaction
136 static int btree_begin_transaction(
138 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
139 int argc, /* Number of arguments */
140 const char **argv /* Text of each argument */
145 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
149 pBt = sqlite3TestTextToPtr(argv[1]);
150 sqlite3BtreeEnter(pBt);
151 rc = sqlite3BtreeBeginTrans(pBt, 1);
152 sqlite3BtreeLeave(pBt);
154 Tcl_AppendResult(interp, errorName(rc), 0);
161 ** Usage: btree_rollback ID
165 static int btree_rollback(
167 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
168 int argc, /* Number of arguments */
169 const char **argv /* Text of each argument */
174 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
178 pBt = sqlite3TestTextToPtr(argv[1]);
179 sqlite3BtreeEnter(pBt);
180 rc = sqlite3BtreeRollback(pBt);
181 sqlite3BtreeLeave(pBt);
183 Tcl_AppendResult(interp, errorName(rc), 0);
190 ** Usage: btree_commit ID
192 ** Commit all changes
194 static int btree_commit(
196 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
197 int argc, /* Number of arguments */
198 const char **argv /* Text of each argument */
203 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
207 pBt = sqlite3TestTextToPtr(argv[1]);
208 sqlite3BtreeEnter(pBt);
209 rc = sqlite3BtreeCommit(pBt);
210 sqlite3BtreeLeave(pBt);
212 Tcl_AppendResult(interp, errorName(rc), 0);
219 ** Usage: btree_begin_statement ID
221 ** Start a new statement transaction
223 static int btree_begin_statement(
225 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
226 int argc, /* Number of arguments */
227 const char **argv /* Text of each argument */
232 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
236 pBt = sqlite3TestTextToPtr(argv[1]);
237 sqlite3BtreeEnter(pBt);
238 rc = sqlite3BtreeBeginStmt(pBt);
239 sqlite3BtreeLeave(pBt);
241 Tcl_AppendResult(interp, errorName(rc), 0);
248 ** Usage: btree_rollback_statement ID
252 static int btree_rollback_statement(
254 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
255 int argc, /* Number of arguments */
256 const char **argv /* Text of each argument */
261 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
265 pBt = sqlite3TestTextToPtr(argv[1]);
266 sqlite3BtreeEnter(pBt);
267 rc = sqlite3BtreeRollbackStmt(pBt);
268 sqlite3BtreeLeave(pBt);
270 Tcl_AppendResult(interp, errorName(rc), 0);
277 ** Usage: btree_commit_statement ID
279 ** Commit all changes
281 static int btree_commit_statement(
283 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
284 int argc, /* Number of arguments */
285 const char **argv /* Text of each argument */
290 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
294 pBt = sqlite3TestTextToPtr(argv[1]);
295 sqlite3BtreeEnter(pBt);
296 rc = sqlite3BtreeCommitStmt(pBt);
297 sqlite3BtreeLeave(pBt);
299 Tcl_AppendResult(interp, errorName(rc), 0);
306 ** Usage: btree_create_table ID FLAGS
308 ** Create a new table in the database
310 static int btree_create_table(
312 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
313 int argc, /* Number of arguments */
314 const char **argv /* Text of each argument */
317 int rc, iTable, flags;
320 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
324 pBt = sqlite3TestTextToPtr(argv[1]);
325 if( Tcl_GetInt(interp, argv[2], &flags) ) return TCL_ERROR;
326 sqlite3BtreeEnter(pBt);
327 rc = sqlite3BtreeCreateTable(pBt, &iTable, flags);
328 sqlite3BtreeLeave(pBt);
330 Tcl_AppendResult(interp, errorName(rc), 0);
333 sqlite3_snprintf(sizeof(zBuf), zBuf, "%d", iTable);
334 Tcl_AppendResult(interp, zBuf, 0);
339 ** Usage: btree_drop_table ID TABLENUM
341 ** Delete an entire table from the database
343 static int btree_drop_table(
345 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
346 int argc, /* Number of arguments */
347 const char **argv /* Text of each argument */
354 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
355 " ID TABLENUM\"", 0);
358 pBt = sqlite3TestTextToPtr(argv[1]);
359 if( Tcl_GetInt(interp, argv[2], &iTable) ) return TCL_ERROR;
360 sqlite3BtreeEnter(pBt);
361 rc = sqlite3BtreeDropTable(pBt, iTable, ¬Used1);
362 sqlite3BtreeLeave(pBt);
364 Tcl_AppendResult(interp, errorName(rc), 0);
371 ** Usage: btree_clear_table ID TABLENUM
373 ** Remove all entries from the given table but keep the table around.
375 static int btree_clear_table(
377 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
378 int argc, /* Number of arguments */
379 const char **argv /* Text of each argument */
385 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
386 " ID TABLENUM\"", 0);
389 pBt = sqlite3TestTextToPtr(argv[1]);
390 if( Tcl_GetInt(interp, argv[2], &iTable) ) return TCL_ERROR;
391 sqlite3BtreeEnter(pBt);
392 rc = sqlite3BtreeClearTable(pBt, iTable);
393 sqlite3BtreeLeave(pBt);
395 Tcl_AppendResult(interp, errorName(rc), 0);
402 ** Usage: btree_get_meta ID
406 static int btree_get_meta(
408 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
409 int argc, /* Number of arguments */
410 const char **argv /* Text of each argument */
416 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
420 pBt = sqlite3TestTextToPtr(argv[1]);
421 for(i=0; i<SQLITE_N_BTREE_META; i++){
424 sqlite3BtreeEnter(pBt);
425 rc = sqlite3BtreeGetMeta(pBt, i, &v);
426 sqlite3BtreeLeave(pBt);
428 Tcl_AppendResult(interp, errorName(rc), 0);
431 sqlite3_snprintf(sizeof(zBuf), zBuf,"%d",v);
432 Tcl_AppendElement(interp, zBuf);
438 ** Usage: btree_update_meta ID METADATA...
442 static int btree_update_meta(
444 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
445 int argc, /* Number of arguments */
446 const char **argv /* Text of each argument */
451 int aMeta[SQLITE_N_BTREE_META];
453 if( argc!=2+SQLITE_N_BTREE_META ){
455 sqlite3_snprintf(sizeof(zBuf), zBuf,"%d",SQLITE_N_BTREE_META);
456 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
457 " ID METADATA...\" (METADATA is ", zBuf, " integers)", 0);
460 pBt = sqlite3TestTextToPtr(argv[1]);
461 for(i=1; i<SQLITE_N_BTREE_META; i++){
462 if( Tcl_GetInt(interp, argv[i+2], &aMeta[i]) ) return TCL_ERROR;
464 for(i=1; i<SQLITE_N_BTREE_META; i++){
465 sqlite3BtreeEnter(pBt);
466 rc = sqlite3BtreeUpdateMeta(pBt, i, aMeta[i]);
467 sqlite3BtreeLeave(pBt);
469 Tcl_AppendResult(interp, errorName(rc), 0);
477 ** Usage: btree_pager_stats ID
479 ** Returns pager statistics
481 static int btree_pager_stats(
483 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
484 int argc, /* Number of arguments */
485 const char **argv /* Text of each argument */
492 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
496 pBt = sqlite3TestTextToPtr(argv[1]);
498 /* Normally in this file, with a b-tree handle opened using the
499 ** [btree_open] command it is safe to call sqlite3BtreeEnter() directly.
500 ** But this function is sometimes called with a btree handle obtained
501 ** from an open SQLite connection (using [btree_from_db]). In this case
502 ** we need to obtain the mutex for the controlling SQLite handle before
503 ** it is safe to call sqlite3BtreeEnter().
505 sqlite3_mutex_enter(pBt->db->mutex);
507 sqlite3BtreeEnter(pBt);
508 a = sqlite3PagerStats(sqlite3BtreePager(pBt));
510 static char *zName[] = {
511 "ref", "page", "max", "size", "state", "err",
512 "hit", "miss", "ovfl", "read", "write"
515 Tcl_AppendElement(interp, zName[i]);
516 sqlite3_snprintf(sizeof(zBuf), zBuf,"%d",a[i]);
517 Tcl_AppendElement(interp, zBuf);
519 sqlite3BtreeLeave(pBt);
521 /* Release the mutex on the SQLite handle that controls this b-tree */
522 sqlite3_mutex_leave(pBt->db->mutex);
527 ** Usage: btree_integrity_check ID ROOT ...
529 ** Look through every page of the given BTree file to verify correct
530 ** formatting and linkage. Return a line of text for each problem found.
531 ** Return an empty string if everything worked.
533 static int btree_integrity_check(
535 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
536 int argc, /* Number of arguments */
537 const char **argv /* Text of each argument */
547 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
548 " ID ROOT ...\"", 0);
551 pBt = sqlite3TestTextToPtr(argv[1]);
553 aRoot = (int*)sqlite3_malloc( sizeof(int)*(argc-2) );
554 for(i=0; i<argc-2; i++){
555 if( Tcl_GetInt(interp, argv[i+2], &aRoot[i]) ) return TCL_ERROR;
557 #ifndef SQLITE_OMIT_INTEGRITY_CHECK
558 sqlite3BtreeEnter(pBt);
559 zResult = sqlite3BtreeIntegrityCheck(pBt, aRoot, nRoot, 10000, &nErr);
560 sqlite3BtreeLeave(pBt);
564 sqlite3_free((void*)aRoot);
566 Tcl_AppendResult(interp, zResult, 0);
567 sqlite3_free(zResult);
573 ** Usage: btree_cursor_list ID
575 ** Print information about all cursors to standard output for debugging.
577 static int btree_cursor_list(
579 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
580 int argc, /* Number of arguments */
581 const char **argv /* Text of each argument */
586 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
590 pBt = sqlite3TestTextToPtr(argv[1]);
591 sqlite3BtreeEnter(pBt);
592 sqlite3BtreeCursorList(pBt);
593 sqlite3BtreeLeave(pBt);
598 ** Usage: btree_cursor ID TABLENUM WRITEABLE
600 ** Create a new cursor. Return the ID for the cursor.
602 static int btree_cursor(
604 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
605 int argc, /* Number of arguments */
606 const char **argv /* Text of each argument */
616 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
617 " ID TABLENUM WRITEABLE\"", 0);
620 pBt = sqlite3TestTextToPtr(argv[1]);
621 if( Tcl_GetInt(interp, argv[2], &iTable) ) return TCL_ERROR;
622 if( Tcl_GetBoolean(interp, argv[3], &wrFlag) ) return TCL_ERROR;
623 pCur = (BtCursor *)ckalloc(sqlite3BtreeCursorSize());
624 memset(pCur, 0, sqlite3BtreeCursorSize());
625 sqlite3BtreeEnter(pBt);
626 rc = sqlite3BtreeCursor(pBt, iTable, wrFlag, 0, pCur);
627 sqlite3BtreeLeave(pBt);
629 ckfree((char *)pCur);
630 Tcl_AppendResult(interp, errorName(rc), 0);
633 sqlite3_snprintf(sizeof(zBuf), zBuf,"%p", pCur);
634 Tcl_AppendResult(interp, zBuf, 0);
639 ** Usage: btree_close_cursor ID
641 ** Close a cursor opened using btree_cursor.
643 static int btree_close_cursor(
645 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
646 int argc, /* Number of arguments */
647 const char **argv /* Text of each argument */
654 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
658 pCur = sqlite3TestTextToPtr(argv[1]);
660 sqlite3BtreeEnter(pBt);
661 rc = sqlite3BtreeCloseCursor(pCur);
662 sqlite3BtreeLeave(pBt);
663 ckfree((char *)pCur);
665 Tcl_AppendResult(interp, errorName(rc), 0);
672 ** Usage: btree_move_to ID KEY
674 ** Move the cursor to the entry with the given key.
676 static int btree_move_to(
678 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
679 int argc, /* Number of arguments */
680 const char **argv /* Text of each argument */
688 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
692 pCur = sqlite3TestTextToPtr(argv[1]);
693 sqlite3BtreeEnter(pCur->pBtree);
694 if( sqlite3BtreeFlags(pCur) & BTREE_INTKEY ){
696 if( Tcl_GetInt(interp, argv[2], &iKey) ){
697 sqlite3BtreeLeave(pCur->pBtree);
700 rc = sqlite3BtreeMovetoUnpacked(pCur, 0, iKey, 0, &res);
702 rc = sqlite3BtreeMoveto(pCur, argv[2], strlen(argv[2]), 0, &res);
704 sqlite3BtreeLeave(pCur->pBtree);
706 Tcl_AppendResult(interp, errorName(rc), 0);
709 if( res<0 ) res = -1;
711 sqlite3_snprintf(sizeof(zBuf), zBuf,"%d",res);
712 Tcl_AppendResult(interp, zBuf, 0);
717 ** Usage: btree_delete ID
719 ** Delete the entry that the cursor is pointing to
721 static int btree_delete(
723 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
724 int argc, /* Number of arguments */
725 const char **argv /* Text of each argument */
731 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
735 pCur = sqlite3TestTextToPtr(argv[1]);
736 sqlite3BtreeEnter(pCur->pBtree);
737 rc = sqlite3BtreeDelete(pCur);
738 sqlite3BtreeLeave(pCur->pBtree);
740 Tcl_AppendResult(interp, errorName(rc), 0);
747 ** Usage: btree_insert ID KEY DATA ?NZERO?
749 ** Create a new entry with the given key and data. If an entry already
750 ** exists with the same key the old entry is overwritten.
752 static int btree_insert(
756 Tcl_Obj *CONST objv[]
762 if( objc!=4 && objc!=5 ){
763 Tcl_WrongNumArgs(interp, 1, objv, "ID KEY DATA ?NZERO?");
766 pCur = sqlite3TestTextToPtr(Tcl_GetString(objv[1]));
768 if( Tcl_GetIntFromObj(interp, objv[4], &nZero) ) return TCL_ERROR;
772 sqlite3BtreeEnter(pCur->pBtree);
773 if( sqlite3BtreeFlags(pCur) & BTREE_INTKEY ){
777 if( Tcl_GetWideIntFromObj(interp, objv[2], &iKey) ){
778 sqlite3BtreeLeave(pCur->pBtree);
781 pBuf = Tcl_GetByteArrayFromObj(objv[3], &len);
782 rc = sqlite3BtreeInsert(pCur, 0, iKey, pBuf, len, nZero, 0);
786 unsigned char *pKBuf;
787 unsigned char *pDBuf;
788 pKBuf = Tcl_GetByteArrayFromObj(objv[2], &keylen);
789 pDBuf = Tcl_GetByteArrayFromObj(objv[3], &dlen);
790 rc = sqlite3BtreeInsert(pCur, pKBuf, keylen, pDBuf, dlen, nZero, 0);
792 sqlite3BtreeLeave(pCur->pBtree);
794 Tcl_AppendResult(interp, errorName(rc), 0);
801 ** Usage: btree_next ID
803 ** Move the cursor to the next entry in the table. Return 0 on success
804 ** or 1 if the cursor was already on the last entry in the table or if
805 ** the table is empty.
807 static int btree_next(
809 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
810 int argc, /* Number of arguments */
811 const char **argv /* Text of each argument */
819 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
823 pCur = sqlite3TestTextToPtr(argv[1]);
824 sqlite3BtreeEnter(pCur->pBtree);
825 rc = sqlite3BtreeNext(pCur, &res);
826 sqlite3BtreeLeave(pCur->pBtree);
828 Tcl_AppendResult(interp, errorName(rc), 0);
831 sqlite3_snprintf(sizeof(zBuf),zBuf,"%d",res);
832 Tcl_AppendResult(interp, zBuf, 0);
837 ** Usage: btree_prev ID
839 ** Move the cursor to the previous entry in the table. Return 0 on
840 ** success and 1 if the cursor was already on the first entry in
841 ** the table or if the table was empty.
843 static int btree_prev(
845 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
846 int argc, /* Number of arguments */
847 const char **argv /* Text of each argument */
855 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
859 pCur = sqlite3TestTextToPtr(argv[1]);
860 sqlite3BtreeEnter(pCur->pBtree);
861 rc = sqlite3BtreePrevious(pCur, &res);
862 sqlite3BtreeLeave(pCur->pBtree);
864 Tcl_AppendResult(interp, errorName(rc), 0);
867 sqlite3_snprintf(sizeof(zBuf),zBuf,"%d",res);
868 Tcl_AppendResult(interp, zBuf, 0);
873 ** Usage: btree_first ID
875 ** Move the cursor to the first entry in the table. Return 0 if the
876 ** cursor was left point to something and 1 if the table is empty.
878 static int btree_first(
880 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
881 int argc, /* Number of arguments */
882 const char **argv /* Text of each argument */
890 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
894 pCur = sqlite3TestTextToPtr(argv[1]);
895 sqlite3BtreeEnter(pCur->pBtree);
896 rc = sqlite3BtreeFirst(pCur, &res);
897 sqlite3BtreeLeave(pCur->pBtree);
899 Tcl_AppendResult(interp, errorName(rc), 0);
902 sqlite3_snprintf(sizeof(zBuf),zBuf,"%d",res);
903 Tcl_AppendResult(interp, zBuf, 0);
908 ** Usage: btree_last ID
910 ** Move the cursor to the last entry in the table. Return 0 if the
911 ** cursor was left point to something and 1 if the table is empty.
913 static int btree_last(
915 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
916 int argc, /* Number of arguments */
917 const char **argv /* Text of each argument */
925 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
929 pCur = sqlite3TestTextToPtr(argv[1]);
930 sqlite3BtreeEnter(pCur->pBtree);
931 rc = sqlite3BtreeLast(pCur, &res);
932 sqlite3BtreeLeave(pCur->pBtree);
934 Tcl_AppendResult(interp, errorName(rc), 0);
937 sqlite3_snprintf(sizeof(zBuf),zBuf,"%d",res);
938 Tcl_AppendResult(interp, zBuf, 0);
943 ** Usage: btree_eof ID
945 ** Return TRUE if the given cursor is not pointing at a valid entry.
946 ** Return FALSE if the cursor does point to a valid entry.
948 static int btree_eof(
950 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
951 int argc, /* Number of arguments */
952 const char **argv /* Text of each argument */
959 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
963 pCur = sqlite3TestTextToPtr(argv[1]);
964 sqlite3BtreeEnter(pCur->pBtree);
965 rc = sqlite3BtreeEof(pCur);
966 sqlite3BtreeLeave(pCur->pBtree);
967 sqlite3_snprintf(sizeof(zBuf),zBuf, "%d", rc);
968 Tcl_AppendResult(interp, zBuf, 0);
973 ** Usage: btree_keysize ID
975 ** Return the number of bytes of key. For an INTKEY table, this
976 ** returns the key itself.
978 static int btree_keysize(
980 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
981 int argc, /* Number of arguments */
982 const char **argv /* Text of each argument */
989 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
993 pCur = sqlite3TestTextToPtr(argv[1]);
994 sqlite3BtreeEnter(pCur->pBtree);
995 sqlite3BtreeKeySize(pCur, (i64*)&n);
996 sqlite3BtreeLeave(pCur->pBtree);
997 sqlite3_snprintf(sizeof(zBuf),zBuf, "%llu", n);
998 Tcl_AppendResult(interp, zBuf, 0);
1003 ** Usage: btree_key ID
1005 ** Return the key for the entry at which the cursor is pointing.
1007 static int btree_key(
1009 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1010 int argc, /* Number of arguments */
1011 const char **argv /* Text of each argument */
1019 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1023 pCur = sqlite3TestTextToPtr(argv[1]);
1024 sqlite3BtreeEnter(pCur->pBtree);
1025 sqlite3BtreeKeySize(pCur, (i64*)&n);
1026 if( sqlite3BtreeFlags(pCur) & BTREE_INTKEY ){
1028 sqlite3_snprintf(sizeof(zBuf2),zBuf2, "%llu", n);
1029 Tcl_AppendResult(interp, zBuf2, 0);
1031 zBuf = sqlite3_malloc( n+1 );
1032 rc = sqlite3BtreeKey(pCur, 0, n, zBuf);
1034 sqlite3BtreeLeave(pCur->pBtree);
1035 Tcl_AppendResult(interp, errorName(rc), 0);
1039 Tcl_AppendResult(interp, zBuf, 0);
1042 sqlite3BtreeLeave(pCur->pBtree);
1047 ** Usage: btree_data ID ?N?
1049 ** Return the data for the entry at which the cursor is pointing.
1051 static int btree_data(
1053 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1054 int argc, /* Number of arguments */
1055 const char **argv /* Text of each argument */
1062 if( argc!=2 && argc!=3 ){
1063 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1067 pCur = sqlite3TestTextToPtr(argv[1]);
1068 sqlite3BtreeEnter(pCur->pBtree);
1070 sqlite3BtreeDataSize(pCur, &n);
1074 zBuf = sqlite3_malloc( n+1 );
1075 rc = sqlite3BtreeData(pCur, 0, n, zBuf);
1076 sqlite3BtreeLeave(pCur->pBtree);
1078 Tcl_AppendResult(interp, errorName(rc), 0);
1083 Tcl_AppendResult(interp, zBuf, 0);
1089 ** Usage: btree_fetch_key ID AMT
1091 ** Use the sqlite3BtreeKeyFetch() routine to get AMT bytes of the key.
1092 ** If sqlite3BtreeKeyFetch() fails, return an empty string.
1094 static int btree_fetch_key(
1096 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1097 int argc, /* Number of arguments */
1098 const char **argv /* Text of each argument */
1108 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1112 pCur = sqlite3TestTextToPtr(argv[1]);
1113 if( Tcl_GetInt(interp, argv[2], &n) ) return TCL_ERROR;
1114 sqlite3BtreeEnter(pCur->pBtree);
1115 sqlite3BtreeKeySize(pCur, (i64*)&nKey);
1116 zBuf = sqlite3BtreeKeyFetch(pCur, &amt);
1117 if( zBuf && amt>=n ){
1118 assert( nKey<sizeof(zStatic) );
1120 memcpy(zStatic, zBuf, (int)nKey);
1122 Tcl_AppendResult(interp, zStatic, 0);
1124 sqlite3BtreeLeave(pCur->pBtree);
1129 ** Usage: btree_fetch_data ID AMT
1131 ** Use the sqlite3BtreeDataFetch() routine to get AMT bytes of the key.
1132 ** If sqlite3BtreeDataFetch() fails, return an empty string.
1134 static int btree_fetch_data(
1136 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1137 int argc, /* Number of arguments */
1138 const char **argv /* Text of each argument */
1148 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1152 pCur = sqlite3TestTextToPtr(argv[1]);
1153 if( Tcl_GetInt(interp, argv[2], &n) ) return TCL_ERROR;
1154 sqlite3BtreeEnter(pCur->pBtree);
1155 sqlite3BtreeDataSize(pCur, &nData);
1156 zBuf = sqlite3BtreeDataFetch(pCur, &amt);
1157 if( zBuf && amt>=n ){
1158 assert( nData<sizeof(zStatic) );
1159 if( n>0 ) nData = n;
1160 memcpy(zStatic, zBuf, (int)nData);
1162 Tcl_AppendResult(interp, zStatic, 0);
1164 sqlite3BtreeLeave(pCur->pBtree);
1169 ** Usage: btree_payload_size ID
1171 ** Return the number of bytes of payload
1173 static int btree_payload_size(
1175 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1176 int argc, /* Number of arguments */
1177 const char **argv /* Text of each argument */
1185 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1189 pCur = sqlite3TestTextToPtr(argv[1]);
1190 sqlite3BtreeEnter(pCur->pBtree);
1191 if( sqlite3BtreeFlags(pCur) & BTREE_INTKEY ){
1194 sqlite3BtreeKeySize(pCur, (i64*)&n1);
1196 sqlite3BtreeDataSize(pCur, (u32*)&n2);
1197 sqlite3BtreeLeave(pCur->pBtree);
1198 sqlite3_snprintf(sizeof(zBuf),zBuf, "%d", (int)(n1+n2));
1199 Tcl_AppendResult(interp, zBuf, 0);
1204 ** Usage: btree_cursor_info ID ?UP-CNT?
1206 ** Return integers containing information about the entry the
1207 ** cursor is pointing to:
1209 ** aResult[0] = The page number
1210 ** aResult[1] = The entry number
1211 ** aResult[2] = Total number of entries on this page
1212 ** aResult[3] = Cell size (local payload + header)
1213 ** aResult[4] = Number of free bytes on this page
1214 ** aResult[5] = Number of free blocks on the page
1215 ** aResult[6] = Total payload size (local + overflow)
1216 ** aResult[7] = Header size in bytes
1217 ** aResult[8] = Local payload size
1218 ** aResult[9] = Parent page number
1219 ** aResult[10]= Page number of the first overflow page
1221 static int btree_cursor_info(
1223 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1224 int argc, /* Number of arguments */
1225 const char **argv /* Text of each argument */
1234 if( argc!=2 && argc!=3 ){
1235 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1236 " ID ?UP-CNT?\"", 0);
1239 pCur = sqlite3TestTextToPtr(argv[1]);
1241 if( Tcl_GetInt(interp, argv[2], &up) ) return TCL_ERROR;
1245 sqlite3BtreeEnter(pCur->pBtree);
1246 rc = sqlite3BtreeCursorInfo(pCur, aResult, up);
1248 Tcl_AppendResult(interp, errorName(rc), 0);
1249 sqlite3BtreeLeave(pCur->pBtree);
1253 for(i=0; i<sizeof(aResult)/sizeof(aResult[0]); i++){
1254 sqlite3_snprintf(40,&zBuf[j]," %d", aResult[i]);
1255 j += strlen(&zBuf[j]);
1257 sqlite3BtreeLeave(pCur->pBtree);
1258 Tcl_AppendResult(interp, &zBuf[1], 0);
1263 ** Copied from btree.c:
1265 static u32 t4Get4byte(unsigned char *p){
1266 return (p[0]<<24) | (p[1]<<16) | (p[2]<<8) | p[3];
1270 ** btree_ovfl_info BTREE CURSOR
1272 ** Given a cursor, return the sequence of pages number that form the
1273 ** overflow pages for the data of the entry that the cursor is point
1276 static int btree_ovfl_info(
1278 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1279 int argc, /* Number of arguments */
1280 const char **argv /* Text of each argument */
1295 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1296 " BTREE CURSOR", 0);
1299 pBt = sqlite3TestTextToPtr(argv[1]);
1300 pCur = sqlite3TestTextToPtr(argv[2]);
1301 if( (*(void**)pCur) != (void*)pBt ){
1302 Tcl_AppendResult(interp, "Cursor ", argv[2], " does not belong to btree ",
1306 sqlite3BtreeEnter(pBt);
1307 pPager = sqlite3BtreePager(pBt);
1308 rc = sqlite3BtreeCursorInfo(pCur, aResult, 0);
1310 Tcl_AppendResult(interp, errorName(rc), 0);
1311 sqlite3BtreeLeave(pBt);
1314 dataSize = pBt->pBt->usableSize;
1315 Tcl_DStringInit(&str);
1316 n = aResult[6] - aResult[8];
1317 n = (n + dataSize - 1)/dataSize;
1318 pgno = (u32)aResult[10];
1319 while( pgno && n-- ){
1321 sprintf(zElem, "%d", pgno);
1322 Tcl_DStringAppendElement(&str, zElem);
1323 if( sqlite3PagerGet(pPager, pgno, &pDbPage)!=SQLITE_OK ){
1324 Tcl_DStringFree(&str);
1325 Tcl_AppendResult(interp, "unable to get page ", zElem, 0);
1326 sqlite3BtreeLeave(pBt);
1329 pPage = sqlite3PagerGetData(pDbPage);
1330 pgno = t4Get4byte((unsigned char*)pPage);
1331 sqlite3PagerUnref(pDbPage);
1333 sqlite3BtreeLeave(pBt);
1334 Tcl_DStringResult(interp, &str);
1339 ** The command is provided for the purpose of setting breakpoints.
1340 ** in regression test scripts.
1342 ** By setting a GDB breakpoint on this procedure and executing the
1343 ** btree_breakpoint command in a test script, we can stop GDB at
1344 ** the point in the script where the btree_breakpoint command is
1345 ** inserted. This is useful for debugging.
1347 static int btree_breakpoint(
1349 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1350 int argc, /* Number of arguments */
1351 const char **argv /* Text of each argument */
1357 ** usage: varint_test START MULTIPLIER COUNT INCREMENT
1359 ** This command tests the putVarint() and getVarint()
1360 ** routines, both for accuracy and for speed.
1362 ** An integer is written using putVarint() and read back with
1363 ** getVarint() and varified to be unchanged. This repeats COUNT
1364 ** times. The first integer is START*MULTIPLIER. Each iteration
1365 ** increases the integer by INCREMENT.
1367 ** This command returns nothing if it works. It returns an error message
1368 ** if something goes wrong.
1370 static int btree_varint_test(
1372 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1373 int argc, /* Number of arguments */
1374 const char **argv /* Text of each argument */
1376 u32 start, mult, count, incr;
1379 unsigned char zBuf[100];
1381 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1382 " START MULTIPLIER COUNT INCREMENT\"", 0);
1385 if( Tcl_GetInt(interp, argv[1], (int*)&start) ) return TCL_ERROR;
1386 if( Tcl_GetInt(interp, argv[2], (int*)&mult) ) return TCL_ERROR;
1387 if( Tcl_GetInt(interp, argv[3], (int*)&count) ) return TCL_ERROR;
1388 if( Tcl_GetInt(interp, argv[4], (int*)&incr) ) return TCL_ERROR;
1391 for(i=0; i<count; i++){
1393 n1 = putVarint(zBuf, in);
1395 sprintf(zErr, "putVarint returned %d - should be between 1 and 9", n1);
1396 Tcl_AppendResult(interp, zErr, 0);
1399 n2 = getVarint(zBuf, &out);
1401 sprintf(zErr, "putVarint returned %d and getVarint returned %d", n1, n2);
1402 Tcl_AppendResult(interp, zErr, 0);
1406 sprintf(zErr, "Wrote 0x%016llx and got back 0x%016llx", in, out);
1407 Tcl_AppendResult(interp, zErr, 0);
1410 if( (in & 0xffffffff)==in ){
1412 n2 = getVarint32(zBuf, out32);
1415 sprintf(zErr, "putVarint returned %d and GetVarint32 returned %d",
1417 Tcl_AppendResult(interp, zErr, 0);
1421 sprintf(zErr, "Wrote 0x%016llx and got back 0x%016llx from GetVarint32",
1423 Tcl_AppendResult(interp, zErr, 0);
1428 /* In order to get realistic timings, run getVarint 19 more times.
1429 ** This is because getVarint is called about 20 times more often
1432 for(j=0; j<19; j++){
1433 getVarint(zBuf, &out);
1441 ** usage: btree_from_db DB-HANDLE
1443 ** This command returns the btree handle for the main database associated
1444 ** with the database-handle passed as the argument. Example usage:
1446 ** sqlite3 db test.db
1447 ** set bt [btree_from_db db]
1449 static int btree_from_db(
1451 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1452 int argc, /* Number of arguments */
1453 const char **argv /* Text of each argument */
1461 if( argc!=2 && argc!=3 ){
1462 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1463 " DB-HANDLE ?N?\"", 0);
1467 if( 1!=Tcl_GetCommandInfo(interp, argv[1], &info) ){
1468 Tcl_AppendResult(interp, "No such db-handle: \"", argv[1], "\"", 0);
1472 iDb = atoi(argv[2]);
1475 db = *((sqlite3 **)info.objClientData);
1478 pBt = db->aDb[iDb].pBt;
1479 sqlite3_snprintf(sizeof(zBuf), zBuf, "%p", pBt);
1480 Tcl_SetResult(interp, zBuf, TCL_VOLATILE);
1486 ** usage: btree_set_cache_size ID NCACHE
1488 ** Set the size of the cache used by btree $ID.
1490 static int btree_set_cache_size(
1492 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1493 int argc, /* Number of arguments */
1494 const char **argv /* Text of each argument */
1500 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1504 pBt = sqlite3TestTextToPtr(argv[1]);
1505 if( Tcl_GetInt(interp, argv[2], &nCache) ) return TCL_ERROR;
1507 sqlite3_mutex_enter(pBt->db->mutex);
1508 sqlite3BtreeEnter(pBt);
1509 sqlite3BtreeSetCacheSize(pBt, nCache);
1510 sqlite3BtreeLeave(pBt);
1511 sqlite3_mutex_leave(pBt->db->mutex);
1517 ** Usage: btree_ismemdb ID
1519 ** Return true if the B-Tree is in-memory.
1521 static int btree_ismemdb(
1523 Tcl_Interp *interp, /* The TCL interpreter that invoked this command */
1524 int argc, /* Number of arguments */
1525 const char **argv /* Text of each argument */
1531 Tcl_AppendResult(interp, "wrong # args: should be \"", argv[0],
1535 pBt = sqlite3TestTextToPtr(argv[1]);
1536 sqlite3_mutex_enter(pBt->db->mutex);
1537 sqlite3BtreeEnter(pBt);
1538 res = sqlite3PagerIsMemdb(sqlite3BtreePager(pBt));
1539 sqlite3BtreeLeave(pBt);
1540 sqlite3_mutex_leave(pBt->db->mutex);
1541 Tcl_SetObjResult(interp, Tcl_NewBooleanObj(res));
1547 ** Register commands with the TCL interpreter.
1549 int Sqlitetest3_Init(Tcl_Interp *interp){
1554 { "btree_open", (Tcl_CmdProc*)btree_open },
1555 { "btree_close", (Tcl_CmdProc*)btree_close },
1556 { "btree_begin_transaction", (Tcl_CmdProc*)btree_begin_transaction },
1557 { "btree_commit", (Tcl_CmdProc*)btree_commit },
1558 { "btree_rollback", (Tcl_CmdProc*)btree_rollback },
1559 { "btree_create_table", (Tcl_CmdProc*)btree_create_table },
1560 { "btree_drop_table", (Tcl_CmdProc*)btree_drop_table },
1561 { "btree_clear_table", (Tcl_CmdProc*)btree_clear_table },
1562 { "btree_get_meta", (Tcl_CmdProc*)btree_get_meta },
1563 { "btree_update_meta", (Tcl_CmdProc*)btree_update_meta },
1564 { "btree_pager_stats", (Tcl_CmdProc*)btree_pager_stats },
1565 { "btree_cursor", (Tcl_CmdProc*)btree_cursor },
1566 { "btree_close_cursor", (Tcl_CmdProc*)btree_close_cursor },
1567 { "btree_move_to", (Tcl_CmdProc*)btree_move_to },
1568 { "btree_delete", (Tcl_CmdProc*)btree_delete },
1569 { "btree_next", (Tcl_CmdProc*)btree_next },
1570 { "btree_prev", (Tcl_CmdProc*)btree_prev },
1571 { "btree_eof", (Tcl_CmdProc*)btree_eof },
1572 { "btree_keysize", (Tcl_CmdProc*)btree_keysize },
1573 { "btree_key", (Tcl_CmdProc*)btree_key },
1574 { "btree_data", (Tcl_CmdProc*)btree_data },
1575 { "btree_fetch_key", (Tcl_CmdProc*)btree_fetch_key },
1576 { "btree_fetch_data", (Tcl_CmdProc*)btree_fetch_data },
1577 { "btree_payload_size", (Tcl_CmdProc*)btree_payload_size },
1578 { "btree_first", (Tcl_CmdProc*)btree_first },
1579 { "btree_last", (Tcl_CmdProc*)btree_last },
1580 { "btree_integrity_check", (Tcl_CmdProc*)btree_integrity_check },
1581 { "btree_breakpoint", (Tcl_CmdProc*)btree_breakpoint },
1582 { "btree_varint_test", (Tcl_CmdProc*)btree_varint_test },
1583 { "btree_begin_statement", (Tcl_CmdProc*)btree_begin_statement },
1584 { "btree_commit_statement", (Tcl_CmdProc*)btree_commit_statement },
1585 { "btree_rollback_statement", (Tcl_CmdProc*)btree_rollback_statement },
1586 { "btree_from_db", (Tcl_CmdProc*)btree_from_db },
1587 { "btree_set_cache_size", (Tcl_CmdProc*)btree_set_cache_size },
1588 { "btree_cursor_info", (Tcl_CmdProc*)btree_cursor_info },
1589 { "btree_ovfl_info", (Tcl_CmdProc*)btree_ovfl_info },
1590 { "btree_cursor_list", (Tcl_CmdProc*)btree_cursor_list },
1591 { "btree_ismemdb", (Tcl_CmdProc*)btree_ismemdb },
1595 for(i=0; i<sizeof(aCmd)/sizeof(aCmd[0]); i++){
1596 Tcl_CreateCommand(interp, aCmd[i].zName, aCmd[i].xProc, 0, 0);
1599 /* The btree_insert command is implemented using the tcl 'object'
1600 ** interface, not the string interface like the other commands in this
1601 ** file. This is so binary data can be inserted into btree tables.
1603 Tcl_CreateObjCommand(interp, "btree_insert", btree_insert, 0, 0);