1
0
mirror of https://github.com/sqlite/sqlite.git synced 2025-08-07 02:42:48 +03:00

Merge recent trunk changes into the threads branch.

FossilOrigin-Name: 163c247bd8280ab14fe577329c631c8bd884707f
This commit is contained in:
drh
2014-07-28 15:01:37 +00:00
10 changed files with 239 additions and 116 deletions

View File

@@ -1,5 +1,5 @@
C Improvements\sto\scomments\sin\sthe\smulti-threaded\ssorter.\s\sAlso\sinclude\sa\nfunction\sname\schange\sfor\sclarity.\s\sAnd\sadd\sa\stest\sto\shelp\sshow\sthat\sthe\nMergeEngine\sobject\sis\sonly\sused\sby\sa\ssingle\sthread. C Merge\srecent\strunk\schanges\sinto\sthe\sthreads\sbranch.
D 2014-07-28T14:54:50.442 D 2014-07-28T15:01:37.313
F Makefile.arm-wince-mingw32ce-gcc d6df77f1f48d690bd73162294bbba7f59507c72f F Makefile.arm-wince-mingw32ce-gcc d6df77f1f48d690bd73162294bbba7f59507c72f
F Makefile.in cf57f673d77606ab0f2d9627ca52a9ba1464146a F Makefile.in cf57f673d77606ab0f2d9627ca52a9ba1464146a
F Makefile.linux-gcc 91d710bdc4998cb015f39edf3cb314ec4f4d7e23 F Makefile.linux-gcc 91d710bdc4998cb015f39edf3cb314ec4f4d7e23
@@ -161,13 +161,13 @@ F sqlite.pc.in 42b7bf0d02e08b9e77734a47798d1a55a9e0716b
F sqlite3.1 3d8b83c91651f53472ca17599dae3457b8b89494 F sqlite3.1 3d8b83c91651f53472ca17599dae3457b8b89494
F sqlite3.pc.in 48fed132e7cb71ab676105d2a4dc77127d8c1f3a F sqlite3.pc.in 48fed132e7cb71ab676105d2a4dc77127d8c1f3a
F src/alter.c b00900877f766f116f9e16116f1ccacdc21d82f1 F src/alter.c b00900877f766f116f9e16116f1ccacdc21d82f1
F src/analyze.c 1c9831015e8c575796a97692d1493ba720d16f27 F src/analyze.c de34a73b86db9dc3a16beef12cc5573c50223956
F src/attach.c 3801129015ef59d76bf23c95ef9b0069d18a0c52 F src/attach.c 3801129015ef59d76bf23c95ef9b0069d18a0c52
F src/auth.c 523da7fb4979469955d822ff9298352d6b31de34 F src/auth.c 523da7fb4979469955d822ff9298352d6b31de34
F src/backup.c a729e63cf5cd1829507cb7b8e89f99b95141bb53 F src/backup.c a729e63cf5cd1829507cb7b8e89f99b95141bb53
F src/bitvec.c 19a4ba637bd85f8f63fc8c9bae5ade9fb05ec1cb F src/bitvec.c 19a4ba637bd85f8f63fc8c9bae5ade9fb05ec1cb
F src/btmutex.c 976f45a12e37293e32cae0281b15a21d48a8aaa7 F src/btmutex.c 976f45a12e37293e32cae0281b15a21d48a8aaa7
F src/btree.c 32b552d5388fa27adf99b9d7e09f4746a08da4e7 F src/btree.c dff8e7789730e835b66d81f9eb68de2352845012
F src/btree.h 4245a349bfe09611d7ff887dbc3a80cee8b7955a F src/btree.h 4245a349bfe09611d7ff887dbc3a80cee8b7955a
F src/btreeInt.h cf180d86b2e9e418f638d65baa425c4c69c0e0e3 F src/btreeInt.h cf180d86b2e9e418f638d65baa425c4c69c0e0e3
F src/build.c af833cdcba64a465eeba5f2d0c9fc012a9f6c31d F src/build.c af833cdcba64a465eeba5f2d0c9fc012a9f6c31d
@@ -216,7 +216,7 @@ F src/parse.y 22d6a074e5f5a7258947a1dc55a9bf946b765dd0
F src/pcache.c d8eafac28290d4bb80332005435db44991d07fc2 F src/pcache.c d8eafac28290d4bb80332005435db44991d07fc2
F src/pcache.h a5e4f5d9f5d592051d91212c5949517971ae6222 F src/pcache.h a5e4f5d9f5d592051d91212c5949517971ae6222
F src/pcache1.c 102e6f5a2fbc646154463eb856d1fd716867b64c F src/pcache1.c 102e6f5a2fbc646154463eb856d1fd716867b64c
F src/pragma.c e17c5ea1cb9eb9d93c41bbb7c3a17e747d5e0335 F src/pragma.c 30f3b2ac09fef58320375d78e0e18b976198fc69
F src/prepare.c 677521ab7132615a8a26107a1d1c3132f44ae337 F src/prepare.c 677521ab7132615a8a26107a1d1c3132f44ae337
F src/printf.c af06f66927919730f03479fed6ae9854f73419f4 F src/printf.c af06f66927919730f03479fed6ae9854f73419f4
F src/random.c d10c1f85b6709ca97278428fd5db5bbb9c74eece F src/random.c d10c1f85b6709ca97278428fd5db5bbb9c74eece
@@ -227,7 +227,7 @@ F src/shell.c 05e9e7f667a6340643b647c4be0db15dd7627d92
F src/sqlite.h.in a30af69fcbc8fab8b4a00032f9f1d24ba2f01c2c F src/sqlite.h.in a30af69fcbc8fab8b4a00032f9f1d24ba2f01c2c
F src/sqlite3.rc 11094cc6a157a028b301a9f06b3d03089ea37c3e F src/sqlite3.rc 11094cc6a157a028b301a9f06b3d03089ea37c3e
F src/sqlite3ext.h 886f5a34de171002ad46fae8c36a7d8051c190fc F src/sqlite3ext.h 886f5a34de171002ad46fae8c36a7d8051c190fc
F src/sqliteInt.h 7d3b81b03c714486968a04a2583bc14e5cf14fcf F src/sqliteInt.h f2b28ce01099fdeb75a222aeddeece7384c9d56a
F src/sqliteLimit.h 164b0e6749d31e0daa1a4589a169d31c0dec7b3d F src/sqliteLimit.h 164b0e6749d31e0daa1a4589a169d31c0dec7b3d
F src/status.c 7ac05a5c7017d0b9f0b4bcd701228b784f987158 F src/status.c 7ac05a5c7017d0b9f0b4bcd701228b784f987158
F src/table.c 2cd62736f845d82200acfa1287e33feb3c15d62e F src/table.c 2cd62736f845d82200acfa1287e33feb3c15d62e
@@ -284,11 +284,11 @@ F src/update.c 01564b3c430f6c7b0a35afaf7aba7987206fa3a5
F src/utf.c a0314e637768a030e6e84a957d0c4f6ba910cc05 F src/utf.c a0314e637768a030e6e84a957d0c4f6ba910cc05
F src/util.c 3076bdd51cdbf60a6e2e57fada745be37133c73e F src/util.c 3076bdd51cdbf60a6e2e57fada745be37133c73e
F src/vacuum.c 3728d74919d4fb1356f9e9a13e27773db60b7179 F src/vacuum.c 3728d74919d4fb1356f9e9a13e27773db60b7179
F src/vdbe.c 7ba7cc297e30d46901038d3d304fbf9914811c86 F src/vdbe.c c28f377d29cfa4db713581fc3a134642e38d9fcf
F src/vdbe.h c63fad052c9e7388d551e556e119c0bcf6bebdf8 F src/vdbe.h c63fad052c9e7388d551e556e119c0bcf6bebdf8
F src/vdbeInt.h 5e925f50d1a52c392304b73f13c23bf3715fc138 F src/vdbeInt.h 8870adf012235708f125f8cd1c988f487dc3eb6f
F src/vdbeapi.c 24e40422382beb774daab11fe9fe9d37e8a04949 F src/vdbeapi.c 24e40422382beb774daab11fe9fe9d37e8a04949
F src/vdbeaux.c 68ef480fa75b27d5860fb96ca4f5a9af98ba102f F src/vdbeaux.c 3f1d2baa4a8cbdad33cb255a5f4fd1af7a414683
F src/vdbeblob.c 9205ce9d3b064d9600f8418a897fc88b5687d9ac F src/vdbeblob.c 9205ce9d3b064d9600f8418a897fc88b5687d9ac
F src/vdbemem.c d90a1e8acf8b63dc9d14cbbea12bfec6cec31394 F src/vdbemem.c d90a1e8acf8b63dc9d14cbbea12bfec6cec31394
F src/vdbesort.c ef998096c8b2a1a85fbd730183a9b62f652e1af3 F src/vdbesort.c ef998096c8b2a1a85fbd730183a9b62f652e1af3
@@ -297,7 +297,7 @@ F src/vtab.c 21b932841e51ebd7d075e2d0ad1415dce8d2d5fd
F src/wal.c 264df50a1b33124130b23180ded2e2c5663c652a F src/wal.c 264df50a1b33124130b23180ded2e2c5663c652a
F src/wal.h df01efe09c5cb8c8e391ff1715cca294f89668a4 F src/wal.h df01efe09c5cb8c8e391ff1715cca294f89668a4
F src/walker.c 11edb74d587bc87b33ca96a5173e3ec1b8389e45 F src/walker.c 11edb74d587bc87b33ca96a5173e3ec1b8389e45
F src/where.c 9454af8e5e18d6e91e5169eadfb61878e2cb42b1 F src/where.c 4dfcd80380d154be434c4b51e890e17ce9754b3e
F src/whereInt.h 929c1349b5355fd44f22cee5c14d72b3329c58a6 F src/whereInt.h 929c1349b5355fd44f22cee5c14d72b3329c58a6
F test/8_3_names.test ebbb5cd36741350040fd28b432ceadf495be25b2 F test/8_3_names.test ebbb5cd36741350040fd28b432ceadf495be25b2
F test/aggerror.test a867e273ef9e3d7919f03ef4f0e8c0d2767944f2 F test/aggerror.test a867e273ef9e3d7919f03ef4f0e8c0d2767944f2
@@ -1189,7 +1189,7 @@ F tool/vdbe_profile.tcl 67746953071a9f8f2f668b73fe899074e2c6d8c1
F tool/warnings-clang.sh f6aa929dc20ef1f856af04a730772f59283631d4 F tool/warnings-clang.sh f6aa929dc20ef1f856af04a730772f59283631d4
F tool/warnings.sh 0abfd78ceb09b7f7c27c688c8e3fe93268a13b32 F tool/warnings.sh 0abfd78ceb09b7f7c27c688c8e3fe93268a13b32
F tool/win/sqlite.vsix deb315d026cc8400325c5863eef847784a219a2f F tool/win/sqlite.vsix deb315d026cc8400325c5863eef847784a219a2f
P 770685892c8f09b9cddb2fbb2877cfb291e19425 P 9af50a878f67c1c2a4f1520160cc989650d7196a 5350229b52b18a4961858a30538c5c75e5bd3048
R d6f4c6811acb8e4555290f0ceab890b8 R c247ffbe0be6e5ba20860955ecb933f2
U drh U drh
Z 8438d80edf2a72a2fe3b270fb06c810d Z eec00caa889efc5f1f3b6e693b6885aa

View File

@@ -1 +1 @@
9af50a878f67c1c2a4f1520160cc989650d7196a 163c247bd8280ab14fe577329c631c8bd884707f

View File

@@ -371,15 +371,20 @@ static void stat4Destructor(void *pOld){
/* /*
** Implementation of the stat_init(N,K,C) SQL function. The three parameters ** Implementation of the stat_init(N,K,C) SQL function. The three parameters
** are: ** are:
** N: The number of columns in the index including the rowid/pk ** N: The number of columns in the index including the rowid/pk (note 1)
** K: The number of columns in the index excluding the rowid/pk ** K: The number of columns in the index excluding the rowid/pk.
** C: The number of rows in the index ** C: The number of rows in the index (note 2)
** **
** C is only used for STAT3 and STAT4. ** Note 1: In the special case of the covering index that implements a
** WITHOUT ROWID table, N is the number of PRIMARY KEY columns, not the
** total number of columns in the table.
** **
** For ordinary rowid tables, N==K+1. But for WITHOUT ROWID tables, ** Note 2: C is only used for STAT3 and STAT4.
** N=K+P where P is the number of columns in the primary key. For the **
** covering index that implements the original WITHOUT ROWID table, N==K. ** For indexes on ordinary rowid tables, N==K+1. But for indexes on
** WITHOUT ROWID tables, N=K+P where P is the number of columns in the
** PRIMARY KEY of the table. The covering index that implements the
** original WITHOUT ROWID table as N==K as a special case.
** **
** This routine allocates the Stat4Accum object in heap memory. The return ** This routine allocates the Stat4Accum object in heap memory. The return
** value is a pointer to the the Stat4Accum object encoded as a blob (i.e. ** value is a pointer to the the Stat4Accum object encoded as a blob (i.e.
@@ -689,7 +694,10 @@ static void samplePushPrevious(Stat4Accum *p, int iChng){
** R Rowid for the current row. Might be a key record for ** R Rowid for the current row. Might be a key record for
** WITHOUT ROWID tables. ** WITHOUT ROWID tables.
** **
** The SQL function always returns NULL. ** This SQL function always returns NULL. It's purpose it to accumulate
** statistical data and/or samples in the Stat4Accum object about the
** index being analyzed. The stat_get() SQL function will later be used to
** extract relevant information for constructing the sqlite_statN tables.
** **
** The R parameter is only used for STAT3 and STAT4 ** The R parameter is only used for STAT3 and STAT4
*/ */
@@ -783,7 +791,10 @@ static const FuncDef statPushFuncdef = {
/* /*
** Implementation of the stat_get(P,J) SQL function. This routine is ** Implementation of the stat_get(P,J) SQL function. This routine is
** used to query the results. Content is returned for parameter J ** used to query statistical information that has been gathered into
** the Stat4Accum object by prior calls to stat_push(). The P parameter
** is a BLOB which is decoded into a pointer to the Stat4Accum objects.
** The content to returned is determined by the parameter J
** which is one of the STAT_GET_xxxx values defined above. ** which is one of the STAT_GET_xxxx values defined above.
** **
** If neither STAT3 nor STAT4 are enabled, then J is always ** If neither STAT3 nor STAT4 are enabled, then J is always
@@ -1002,24 +1013,23 @@ static void analyzeOneTable(
sqlite3VdbeAddOp4(v, OP_String8, 0, regTabname, 0, pTab->zName, 0); sqlite3VdbeAddOp4(v, OP_String8, 0, regTabname, 0, pTab->zName, 0);
for(pIdx=pTab->pIndex; pIdx; pIdx=pIdx->pNext){ for(pIdx=pTab->pIndex; pIdx; pIdx=pIdx->pNext){
int nCol; /* Number of columns indexed by pIdx */ int nCol; /* Number of columns in pIdx. "N" */
int *aGotoChng; /* Array of jump instruction addresses */
int addrRewind; /* Address of "OP_Rewind iIdxCur" */ int addrRewind; /* Address of "OP_Rewind iIdxCur" */
int addrGotoChng0; /* Address of "Goto addr_chng_0" */
int addrNextRow; /* Address of "next_row:" */ int addrNextRow; /* Address of "next_row:" */
const char *zIdxName; /* Name of the index */ const char *zIdxName; /* Name of the index */
int nColTest; /* Number of columns to test for changes */
if( pOnlyIdx && pOnlyIdx!=pIdx ) continue; if( pOnlyIdx && pOnlyIdx!=pIdx ) continue;
if( pIdx->pPartIdxWhere==0 ) needTableCnt = 0; if( pIdx->pPartIdxWhere==0 ) needTableCnt = 0;
if( !HasRowid(pTab) && IsPrimaryKeyIndex(pIdx) ){ if( !HasRowid(pTab) && IsPrimaryKeyIndex(pIdx) ){
nCol = pIdx->nKeyCol; nCol = pIdx->nKeyCol;
zIdxName = pTab->zName; zIdxName = pTab->zName;
nColTest = nCol - 1;
}else{ }else{
nCol = pIdx->nColumn; nCol = pIdx->nColumn;
zIdxName = pIdx->zName; zIdxName = pIdx->zName;
nColTest = pIdx->uniqNotNull ? pIdx->nKeyCol-1 : nCol-1;
} }
aGotoChng = sqlite3DbMallocRaw(db, sizeof(int)*(nCol+1));
if( aGotoChng==0 ) continue;
/* Populate the register containing the index name. */ /* Populate the register containing the index name. */
sqlite3VdbeAddOp4(v, OP_String8, 0, regIdxname, 0, zIdxName, 0); sqlite3VdbeAddOp4(v, OP_String8, 0, regIdxname, 0, zIdxName, 0);
@@ -1048,7 +1058,7 @@ static void analyzeOneTable(
** regPrev(1) = idx(1) ** regPrev(1) = idx(1)
** ... ** ...
** **
** chng_addr_N: ** endDistinctTest:
** regRowid = idx(rowid) ** regRowid = idx(rowid)
** stat_push(P, regChng, regRowid) ** stat_push(P, regChng, regRowid)
** Next csr ** Next csr
@@ -1061,7 +1071,7 @@ static void analyzeOneTable(
** the regPrev array and a trailing rowid (the rowid slot is required ** the regPrev array and a trailing rowid (the rowid slot is required
** when building a record to insert into the sample column of ** when building a record to insert into the sample column of
** the sqlite_stat4 table. */ ** the sqlite_stat4 table. */
pParse->nMem = MAX(pParse->nMem, regPrev+nCol); pParse->nMem = MAX(pParse->nMem, regPrev+nColTest);
/* Open a read-only cursor on the index being analyzed. */ /* Open a read-only cursor on the index being analyzed. */
assert( iDb==sqlite3SchemaToIndex(db, pIdx->pSchema) ); assert( iDb==sqlite3SchemaToIndex(db, pIdx->pSchema) );
@@ -1071,10 +1081,13 @@ static void analyzeOneTable(
/* Invoke the stat_init() function. The arguments are: /* Invoke the stat_init() function. The arguments are:
** **
** (1) the number of columns in the index including the rowid, ** (1) the number of columns in the index including the rowid
** (2) the number of rows in the index, ** (or for a WITHOUT ROWID table, the number of PK columns),
** (2) the number of columns in the key without the rowid/pk
** (3) the number of rows in the index,
** **
** The second argument is only used for STAT3 and STAT4 **
** The third argument is only used for STAT3 and STAT4
*/ */
#ifdef SQLITE_ENABLE_STAT3_OR_STAT4 #ifdef SQLITE_ENABLE_STAT3_OR_STAT4
sqlite3VdbeAddOp2(v, OP_Count, iIdxCur, regStat4+3); sqlite3VdbeAddOp2(v, OP_Count, iIdxCur, regStat4+3);
@@ -1096,42 +1109,60 @@ static void analyzeOneTable(
addrRewind = sqlite3VdbeAddOp1(v, OP_Rewind, iIdxCur); addrRewind = sqlite3VdbeAddOp1(v, OP_Rewind, iIdxCur);
VdbeCoverage(v); VdbeCoverage(v);
sqlite3VdbeAddOp2(v, OP_Integer, 0, regChng); sqlite3VdbeAddOp2(v, OP_Integer, 0, regChng);
addrGotoChng0 = sqlite3VdbeAddOp0(v, OP_Goto);
/*
** next_row:
** regChng = 0
** if( idx(0) != regPrev(0) ) goto chng_addr_0
** regChng = 1
** if( idx(1) != regPrev(1) ) goto chng_addr_1
** ...
** regChng = N
** goto chng_addr_N
*/
addrNextRow = sqlite3VdbeCurrentAddr(v); addrNextRow = sqlite3VdbeCurrentAddr(v);
for(i=0; i<nCol-1; i++){
char *pColl = (char*)sqlite3LocateCollSeq(pParse, pIdx->azColl[i]);
sqlite3VdbeAddOp2(v, OP_Integer, i, regChng);
sqlite3VdbeAddOp3(v, OP_Column, iIdxCur, i, regTemp);
aGotoChng[i] =
sqlite3VdbeAddOp4(v, OP_Ne, regTemp, 0, regPrev+i, pColl, P4_COLLSEQ);
sqlite3VdbeChangeP5(v, SQLITE_NULLEQ);
VdbeCoverage(v);
}
sqlite3VdbeAddOp2(v, OP_Integer, nCol-1, regChng);
aGotoChng[nCol] = sqlite3VdbeAddOp0(v, OP_Goto);
/* if( nColTest>0 ){
** chng_addr_0: int endDistinctTest = sqlite3VdbeMakeLabel(v);
** regPrev(0) = idx(0) int *aGotoChng; /* Array of jump instruction addresses */
** chng_addr_1: aGotoChng = sqlite3DbMallocRaw(db, sizeof(int)*nColTest);
** regPrev(1) = idx(1) if( aGotoChng==0 ) continue;
** ...
*/ /*
sqlite3VdbeJumpHere(v, addrGotoChng0); ** next_row:
for(i=0; i<nCol-1; i++){ ** regChng = 0
sqlite3VdbeJumpHere(v, aGotoChng[i]); ** if( idx(0) != regPrev(0) ) goto chng_addr_0
sqlite3VdbeAddOp3(v, OP_Column, iIdxCur, i, regPrev+i); ** regChng = 1
** if( idx(1) != regPrev(1) ) goto chng_addr_1
** ...
** regChng = N
** goto endDistinctTest
*/
sqlite3VdbeAddOp0(v, OP_Goto);
addrNextRow = sqlite3VdbeCurrentAddr(v);
if( nColTest==1 && pIdx->nKeyCol==1 && pIdx->onError!=OE_None ){
/* For a single-column UNIQUE index, once we have found a non-NULL
** row, we know that all the rest will be distinct, so skip
** subsequent distinctness tests. */
sqlite3VdbeAddOp2(v, OP_NotNull, regPrev, endDistinctTest);
VdbeCoverage(v);
}
for(i=0; i<nColTest; i++){
char *pColl = (char*)sqlite3LocateCollSeq(pParse, pIdx->azColl[i]);
sqlite3VdbeAddOp2(v, OP_Integer, i, regChng);
sqlite3VdbeAddOp3(v, OP_Column, iIdxCur, i, regTemp);
aGotoChng[i] =
sqlite3VdbeAddOp4(v, OP_Ne, regTemp, 0, regPrev+i, pColl, P4_COLLSEQ);
sqlite3VdbeChangeP5(v, SQLITE_NULLEQ);
VdbeCoverage(v);
}
sqlite3VdbeAddOp2(v, OP_Integer, nColTest, regChng);
sqlite3VdbeAddOp2(v, OP_Goto, 0, endDistinctTest);
/*
** chng_addr_0:
** regPrev(0) = idx(0)
** chng_addr_1:
** regPrev(1) = idx(1)
** ...
*/
sqlite3VdbeJumpHere(v, addrNextRow-1);
for(i=0; i<nColTest; i++){
sqlite3VdbeJumpHere(v, aGotoChng[i]);
sqlite3VdbeAddOp3(v, OP_Column, iIdxCur, i, regPrev+i);
}
sqlite3VdbeResolveLabel(v, endDistinctTest);
sqlite3DbFree(db, aGotoChng);
} }
/* /*
@@ -1141,7 +1172,6 @@ static void analyzeOneTable(
** Next csr ** Next csr
** if !eof(csr) goto next_row; ** if !eof(csr) goto next_row;
*/ */
sqlite3VdbeJumpHere(v, aGotoChng[nCol]);
#ifdef SQLITE_ENABLE_STAT3_OR_STAT4 #ifdef SQLITE_ENABLE_STAT3_OR_STAT4
assert( regRowid==(regStat4+2) ); assert( regRowid==(regStat4+2) );
if( HasRowid(pTab) ){ if( HasRowid(pTab) ){
@@ -1219,7 +1249,6 @@ static void analyzeOneTable(
/* End of analysis */ /* End of analysis */
sqlite3VdbeJumpHere(v, addrRewind); sqlite3VdbeJumpHere(v, addrRewind);
sqlite3DbFree(db, aGotoChng);
} }

View File

@@ -1632,7 +1632,7 @@ static Pgno btreePagecount(BtShared *pBt){
u32 sqlite3BtreeLastPage(Btree *p){ u32 sqlite3BtreeLastPage(Btree *p){
assert( sqlite3BtreeHoldsMutex(p) ); assert( sqlite3BtreeHoldsMutex(p) );
assert( ((p->pBt->nPage)&0x8000000)==0 ); assert( ((p->pBt->nPage)&0x8000000)==0 );
return (int)btreePagecount(p->pBt); return btreePagecount(p->pBt);
} }
/* /*

View File

@@ -480,7 +480,7 @@ static const struct sPragmaNames {
** to support legacy SQL code. The safety level used to be boolean ** to support legacy SQL code. The safety level used to be boolean
** and older scripts may have used numbers 0 for OFF and 1 for ON. ** and older scripts may have used numbers 0 for OFF and 1 for ON.
*/ */
static u8 getSafetyLevel(const char *z, int omitFull, int dflt){ static u8 getSafetyLevel(const char *z, int omitFull, u8 dflt){
/* 123456789 123456789 */ /* 123456789 123456789 */
static const char zText[] = "onoffalseyestruefull"; static const char zText[] = "onoffalseyestruefull";
static const u8 iOffset[] = {0, 1, 2, 4, 9, 12, 16}; static const u8 iOffset[] = {0, 1, 2, 4, 9, 12, 16};
@@ -502,7 +502,7 @@ static u8 getSafetyLevel(const char *z, int omitFull, int dflt){
/* /*
** Interpret the given string as a boolean value. ** Interpret the given string as a boolean value.
*/ */
u8 sqlite3GetBoolean(const char *z, int dflt){ u8 sqlite3GetBoolean(const char *z, u8 dflt){
return getSafetyLevel(z,1,dflt)!=0; return getSafetyLevel(z,1,dflt)!=0;
} }

View File

@@ -3387,7 +3387,7 @@ void sqlite3FileSuffix3(const char*, char*);
#else #else
# define sqlite3FileSuffix3(X,Y) # define sqlite3FileSuffix3(X,Y)
#endif #endif
u8 sqlite3GetBoolean(const char *z,int); u8 sqlite3GetBoolean(const char *z,u8);
const void *sqlite3ValueText(sqlite3_value*, u8); const void *sqlite3ValueText(sqlite3_value*, u8);
int sqlite3ValueBytes(sqlite3_value*, u8); int sqlite3ValueBytes(sqlite3_value*, u8);

View File

@@ -224,21 +224,21 @@ static VdbeCursor *allocateCursor(
** look like a number, leave it alone. ** look like a number, leave it alone.
*/ */
static void applyNumericAffinity(Mem *pRec){ static void applyNumericAffinity(Mem *pRec){
if( (pRec->flags & (MEM_Real|MEM_Int))==0 ){ double rValue;
double rValue; i64 iValue;
i64 iValue; u8 enc = pRec->enc;
u8 enc = pRec->enc; if( (pRec->flags&MEM_Str)==0 ) return;
if( (pRec->flags&MEM_Str)==0 ) return; if( sqlite3AtoF(pRec->z, &rValue, pRec->n, enc)==0 ) return;
if( sqlite3AtoF(pRec->z, &rValue, pRec->n, enc)==0 ) return; if( 0==sqlite3Atoi64(pRec->z, &iValue, pRec->n, enc) ){
if( 0==sqlite3Atoi64(pRec->z, &iValue, pRec->n, enc) ){ pRec->u.i = iValue;
pRec->u.i = iValue; pRec->flags |= MEM_Int;
pRec->flags |= MEM_Int; }else{
}else{ pRec->r = rValue;
pRec->r = rValue; pRec->flags |= MEM_Real;
pRec->flags |= MEM_Real;
}
} }
} }
#define ApplyNumericAffinity(X) \
if(((X)->flags&(MEM_Real|MEM_Int))==0){applyNumericAffinity(X);}
/* /*
** Processing is determine by the affinity parameter: ** Processing is determine by the affinity parameter:
@@ -275,7 +275,7 @@ static void applyAffinity(
}else if( affinity!=SQLITE_AFF_NONE ){ }else if( affinity!=SQLITE_AFF_NONE ){
assert( affinity==SQLITE_AFF_INTEGER || affinity==SQLITE_AFF_REAL assert( affinity==SQLITE_AFF_INTEGER || affinity==SQLITE_AFF_REAL
|| affinity==SQLITE_AFF_NUMERIC ); || affinity==SQLITE_AFF_NUMERIC );
applyNumericAffinity(pRec); ApplyNumericAffinity(pRec);
if( pRec->flags & MEM_Real ){ if( pRec->flags & MEM_Real ){
sqlite3VdbeIntegerAffinity(pRec); sqlite3VdbeIntegerAffinity(pRec);
} }
@@ -767,12 +767,14 @@ case OP_Return: { /* in1 */
/* Opcode: InitCoroutine P1 P2 P3 * * /* Opcode: InitCoroutine P1 P2 P3 * *
** **
** Set up register P1 so that it will OP_Yield to the co-routine ** Set up register P1 so that it will Yield to the coroutine
** located at address P3. ** located at address P3.
** **
** If P2!=0 then the co-routine implementation immediately follows ** If P2!=0 then the coroutine implementation immediately follows
** this opcode. So jump over the co-routine implementation to ** this opcode. So jump over the coroutine implementation to
** address P2. ** address P2.
**
** See also: EndCoroutine
*/ */
case OP_InitCoroutine: { /* jump */ case OP_InitCoroutine: { /* jump */
assert( pOp->p1>0 && pOp->p1<=(p->nMem-p->nCursor) ); assert( pOp->p1>0 && pOp->p1<=(p->nMem-p->nCursor) );
@@ -788,9 +790,11 @@ case OP_InitCoroutine: { /* jump */
/* Opcode: EndCoroutine P1 * * * * /* Opcode: EndCoroutine P1 * * * *
** **
** The instruction at the address in register P1 is an OP_Yield. ** The instruction at the address in register P1 is an Yield.
** Jump to the P2 parameter of that OP_Yield. ** Jump to the P2 parameter of that Yield.
** After the jump, register P1 becomes undefined. ** After the jump, register P1 becomes undefined.
**
** See also: InitCoroutine
*/ */
case OP_EndCoroutine: { /* in1 */ case OP_EndCoroutine: { /* in1 */
VdbeOp *pCaller; VdbeOp *pCaller;
@@ -807,11 +811,16 @@ case OP_EndCoroutine: { /* in1 */
/* Opcode: Yield P1 P2 * * * /* Opcode: Yield P1 P2 * * *
** **
** Swap the program counter with the value in register P1. ** Swap the program counter with the value in register P1. This
** has the effect of yielding to a coroutine.
** **
** If the co-routine ends with OP_Yield or OP_Return then continue ** If the coroutine that is launched by this instruction ends with
** to the next instruction. But if the co-routine ends with ** Yield or Return then continue to the next instruction. But if
** OP_EndCoroutine, jump immediately to P2. ** the coroutine launched by this instruction ends with
** EndCoroutine, then jump to P2 rather than continuing with the
** next instruction.
**
** See also: InitCoroutine
*/ */
case OP_Yield: { /* in1, jump */ case OP_Yield: { /* in1, jump */
int pcDest; int pcDest;
@@ -2196,10 +2205,14 @@ case OP_BitNot: { /* same as TK_BITNOT, in1, out2 */
/* Opcode: Once P1 P2 * * * /* Opcode: Once P1 P2 * * *
** **
** Check if OP_Once flag P1 is set. If so, jump to instruction P2. Otherwise, ** Check the "once" flag number P1. If it is set, jump to instruction P2.
** set the flag and fall through to the next instruction. In other words, ** Otherwise, set the flag and fall through to the next instruction.
** this opcode causes all following opcodes up through P2 (but not including ** In other words, this opcode causes all following opcodes up through P2
** P2) to run just once and to be skipped on subsequent times through the loop. ** (but not including P2) to run just once and to be skipped on subsequent
** times through the loop.
**
** All "once" flags are initially cleared whenever a prepared statement
** first begins to run.
*/ */
case OP_Once: { /* jump */ case OP_Once: { /* jump */
assert( pOp->p1<p->nOnceFlag ); assert( pOp->p1<p->nOnceFlag );
@@ -3503,7 +3516,7 @@ case OP_Close: {
break; break;
} }
/* Opcode: SeekGe P1 P2 P3 P4 * /* Opcode: SeekGE P1 P2 P3 P4 *
** Synopsis: key=r[P3@P4] ** Synopsis: key=r[P3@P4]
** **
** If cursor P1 refers to an SQL table (B-Tree that uses integer keys), ** If cursor P1 refers to an SQL table (B-Tree that uses integer keys),
@@ -3515,9 +3528,13 @@ case OP_Close: {
** is greater than or equal to the key value. If there are no records ** is greater than or equal to the key value. If there are no records
** greater than or equal to the key and P2 is not zero, then jump to P2. ** greater than or equal to the key and P2 is not zero, then jump to P2.
** **
** This opcode leaves the cursor configured to move in forward order,
** from the begining toward the end. In other words, the cursor is
** configured to use Next, not Prev.
**
** See also: Found, NotFound, SeekLt, SeekGt, SeekLe ** See also: Found, NotFound, SeekLt, SeekGt, SeekLe
*/ */
/* Opcode: SeekGt P1 P2 P3 P4 * /* Opcode: SeekGT P1 P2 P3 P4 *
** Synopsis: key=r[P3@P4] ** Synopsis: key=r[P3@P4]
** **
** If cursor P1 refers to an SQL table (B-Tree that uses integer keys), ** If cursor P1 refers to an SQL table (B-Tree that uses integer keys),
@@ -3529,9 +3546,13 @@ case OP_Close: {
** is greater than the key value. If there are no records greater than ** is greater than the key value. If there are no records greater than
** the key and P2 is not zero, then jump to P2. ** the key and P2 is not zero, then jump to P2.
** **
** This opcode leaves the cursor configured to move in forward order,
** from the begining toward the end. In other words, the cursor is
** configured to use Next, not Prev.
**
** See also: Found, NotFound, SeekLt, SeekGe, SeekLe ** See also: Found, NotFound, SeekLt, SeekGe, SeekLe
*/ */
/* Opcode: SeekLt P1 P2 P3 P4 * /* Opcode: SeekLT P1 P2 P3 P4 *
** Synopsis: key=r[P3@P4] ** Synopsis: key=r[P3@P4]
** **
** If cursor P1 refers to an SQL table (B-Tree that uses integer keys), ** If cursor P1 refers to an SQL table (B-Tree that uses integer keys),
@@ -3543,9 +3564,13 @@ case OP_Close: {
** is less than the key value. If there are no records less than ** is less than the key value. If there are no records less than
** the key and P2 is not zero, then jump to P2. ** the key and P2 is not zero, then jump to P2.
** **
** This opcode leaves the cursor configured to move in reverse order,
** from the end toward the beginning. In other words, the cursor is
** configured to use Prev, not Next.
**
** See also: Found, NotFound, SeekGt, SeekGe, SeekLe ** See also: Found, NotFound, SeekGt, SeekGe, SeekLe
*/ */
/* Opcode: SeekLe P1 P2 P3 P4 * /* Opcode: SeekLE P1 P2 P3 P4 *
** Synopsis: key=r[P3@P4] ** Synopsis: key=r[P3@P4]
** **
** If cursor P1 refers to an SQL table (B-Tree that uses integer keys), ** If cursor P1 refers to an SQL table (B-Tree that uses integer keys),
@@ -3557,6 +3582,10 @@ case OP_Close: {
** is less than or equal to the key value. If there are no records ** is less than or equal to the key value. If there are no records
** less than or equal to the key and P2 is not zero, then jump to P2. ** less than or equal to the key and P2 is not zero, then jump to P2.
** **
** This opcode leaves the cursor configured to move in reverse order,
** from the end toward the beginning. In other words, the cursor is
** configured to use Prev, not Next.
**
** See also: Found, NotFound, SeekGt, SeekGe, SeekLt ** See also: Found, NotFound, SeekGt, SeekGe, SeekLt
*/ */
case OP_SeekLT: /* jump, in3 */ case OP_SeekLT: /* jump, in3 */
@@ -3582,12 +3611,15 @@ case OP_SeekGT: { /* jump, in3 */
assert( pC->pCursor!=0 ); assert( pC->pCursor!=0 );
oc = pOp->opcode; oc = pOp->opcode;
pC->nullRow = 0; pC->nullRow = 0;
#ifdef SQLITE_DEBUG
pC->seekOp = pOp->opcode;
#endif
if( pC->isTable ){ if( pC->isTable ){
/* The input value in P3 might be of any type: integer, real, string, /* The input value in P3 might be of any type: integer, real, string,
** blob, or NULL. But it needs to be an integer before we can do ** blob, or NULL. But it needs to be an integer before we can do
** the seek, so covert it. */ ** the seek, so covert it. */
pIn3 = &aMem[pOp->p3]; pIn3 = &aMem[pOp->p3];
applyNumericAffinity(pIn3); ApplyNumericAffinity(pIn3);
iKey = sqlite3VdbeIntValue(pIn3); iKey = sqlite3VdbeIntValue(pIn3);
pC->rowidIsValid = 0; pC->rowidIsValid = 0;
@@ -3737,6 +3769,10 @@ case OP_Seek: { /* in2 */
** is a prefix of any entry in P1 then a jump is made to P2 and ** is a prefix of any entry in P1 then a jump is made to P2 and
** P1 is left pointing at the matching entry. ** P1 is left pointing at the matching entry.
** **
** This operation leaves the cursor in a state where it cannot be
** advanced in either direction. In other words, the Next and Prev
** opcodes do not work after this operation.
**
** See also: NotFound, NoConflict, NotExists. SeekGe ** See also: NotFound, NoConflict, NotExists. SeekGe
*/ */
/* Opcode: NotFound P1 P2 P3 P4 * /* Opcode: NotFound P1 P2 P3 P4 *
@@ -3752,6 +3788,10 @@ case OP_Seek: { /* in2 */
** falls through to the next instruction and P1 is left pointing at the ** falls through to the next instruction and P1 is left pointing at the
** matching entry. ** matching entry.
** **
** This operation leaves the cursor in a state where it cannot be
** advanced in either direction. In other words, the Next and Prev
** opcodes do not work after this operation.
**
** See also: Found, NotExists, NoConflict ** See also: Found, NotExists, NoConflict
*/ */
/* Opcode: NoConflict P1 P2 P3 P4 * /* Opcode: NoConflict P1 P2 P3 P4 *
@@ -3771,6 +3811,10 @@ case OP_Seek: { /* in2 */
** This opcode is similar to OP_NotFound with the exceptions that the ** This opcode is similar to OP_NotFound with the exceptions that the
** branch is always taken if any part of the search key input is NULL. ** branch is always taken if any part of the search key input is NULL.
** **
** This operation leaves the cursor in a state where it cannot be
** advanced in either direction. In other words, the Next and Prev
** opcodes do not work after this operation.
**
** See also: NotFound, Found, NotExists ** See also: NotFound, Found, NotExists
*/ */
case OP_NoConflict: /* jump, in3 */ case OP_NoConflict: /* jump, in3 */
@@ -3793,6 +3837,9 @@ case OP_Found: { /* jump, in3 */
assert( pOp->p4type==P4_INT32 ); assert( pOp->p4type==P4_INT32 );
pC = p->apCsr[pOp->p1]; pC = p->apCsr[pOp->p1];
assert( pC!=0 ); assert( pC!=0 );
#ifdef SQLITE_DEBUG
pC->seekOp = 0;
#endif
pIn3 = &aMem[pOp->p3]; pIn3 = &aMem[pOp->p3];
assert( pC->pCursor!=0 ); assert( pC->pCursor!=0 );
assert( pC->isTable==0 ); assert( pC->isTable==0 );
@@ -3864,6 +3911,10 @@ case OP_Found: { /* jump, in3 */
** The OP_NotFound opcode performs the same operation on index btrees ** The OP_NotFound opcode performs the same operation on index btrees
** (with arbitrary multi-value keys). ** (with arbitrary multi-value keys).
** **
** This opcode leaves the cursor in a state where it cannot be advanced
** in either direction. In other words, the Next and Prev opcodes will
** not work following this opcode.
**
** See also: Found, NotFound, NoConflict ** See also: Found, NotFound, NoConflict
*/ */
case OP_NotExists: { /* jump, in3 */ case OP_NotExists: { /* jump, in3 */
@@ -3877,6 +3928,9 @@ case OP_NotExists: { /* jump, in3 */
assert( pOp->p1>=0 && pOp->p1<p->nCursor ); assert( pOp->p1>=0 && pOp->p1<p->nCursor );
pC = p->apCsr[pOp->p1]; pC = p->apCsr[pOp->p1];
assert( pC!=0 ); assert( pC!=0 );
#ifdef SQLITE_DEBUG
pC->seekOp = 0;
#endif
assert( pC->isTable ); assert( pC->isTable );
assert( pC->pseudoTableReg==0 ); assert( pC->pseudoTableReg==0 );
pCrsr = pC->pCursor; pCrsr = pC->pCursor;
@@ -4444,11 +4498,15 @@ case OP_NullRow: {
/* Opcode: Last P1 P2 * * * /* Opcode: Last P1 P2 * * *
** **
** The next use of the Rowid or Column or Next instruction for P1 ** The next use of the Rowid or Column or Prev instruction for P1
** will refer to the last entry in the database table or index. ** will refer to the last entry in the database table or index.
** If the table or index is empty and P2>0, then jump immediately to P2. ** If the table or index is empty and P2>0, then jump immediately to P2.
** If P2 is 0 or if the table or index is not empty, fall through ** If P2 is 0 or if the table or index is not empty, fall through
** to the following instruction. ** to the following instruction.
**
** This opcode leaves the cursor configured to move in reverse order,
** from the end toward the beginning. In other words, the cursor is
** configured to use Prev, not Next.
*/ */
case OP_Last: { /* jump */ case OP_Last: { /* jump */
VdbeCursor *pC; VdbeCursor *pC;
@@ -4466,6 +4524,9 @@ case OP_Last: { /* jump */
pC->deferredMoveto = 0; pC->deferredMoveto = 0;
pC->rowidIsValid = 0; pC->rowidIsValid = 0;
pC->cacheStatus = CACHE_STALE; pC->cacheStatus = CACHE_STALE;
#ifdef SQLITE_DEBUG
pC->seekOp = OP_Last;
#endif
if( pOp->p2>0 ){ if( pOp->p2>0 ){
VdbeBranchTaken(res!=0,2); VdbeBranchTaken(res!=0,2);
if( res ) pc = pOp->p2 - 1; if( res ) pc = pOp->p2 - 1;
@@ -4502,6 +4563,10 @@ case OP_Sort: { /* jump */
** If the table or index is empty and P2>0, then jump immediately to P2. ** If the table or index is empty and P2>0, then jump immediately to P2.
** If P2 is 0 or if the table or index is not empty, fall through ** If P2 is 0 or if the table or index is not empty, fall through
** to the following instruction. ** to the following instruction.
**
** This opcode leaves the cursor configured to move in forward order,
** from the begining toward the end. In other words, the cursor is
** configured to use Next, not Prev.
*/ */
case OP_Rewind: { /* jump */ case OP_Rewind: { /* jump */
VdbeCursor *pC; VdbeCursor *pC;
@@ -4513,6 +4578,9 @@ case OP_Rewind: { /* jump */
assert( pC!=0 ); assert( pC!=0 );
assert( isSorter(pC)==(pOp->opcode==OP_SorterSort) ); assert( isSorter(pC)==(pOp->opcode==OP_SorterSort) );
res = 1; res = 1;
#ifdef SQLITE_DEBUG
pC->seekOp = OP_Rewind;
#endif
if( isSorter(pC) ){ if( isSorter(pC) ){
rc = sqlite3VdbeSorterRewind(pC, &res); rc = sqlite3VdbeSorterRewind(pC, &res);
}else{ }else{
@@ -4539,6 +4607,10 @@ case OP_Rewind: { /* jump */
** to the following instruction. But if the cursor advance was successful, ** to the following instruction. But if the cursor advance was successful,
** jump immediately to P2. ** jump immediately to P2.
** **
** The Next opcode is only valid following an SeekGT, SeekGE, or
** OP_Rewind opcode used to position the cursor. Next is not allowed
** to follow SeekLT, SeekLE, or OP_Last.
**
** The P1 cursor must be for a real table, not a pseudo-table. P1 must have ** The P1 cursor must be for a real table, not a pseudo-table. P1 must have
** been opened prior to this opcode or the program will segfault. ** been opened prior to this opcode or the program will segfault.
** **
@@ -4557,7 +4629,7 @@ case OP_Rewind: { /* jump */
*/ */
/* Opcode: NextIfOpen P1 P2 P3 P4 P5 /* Opcode: NextIfOpen P1 P2 P3 P4 P5
** **
** This opcode works just like OP_Next except that if cursor P1 is not ** This opcode works just like Next except that if cursor P1 is not
** open it behaves a no-op. ** open it behaves a no-op.
*/ */
/* Opcode: Prev P1 P2 P3 P4 P5 /* Opcode: Prev P1 P2 P3 P4 P5
@@ -4567,6 +4639,11 @@ case OP_Rewind: { /* jump */
** to the following instruction. But if the cursor backup was successful, ** to the following instruction. But if the cursor backup was successful,
** jump immediately to P2. ** jump immediately to P2.
** **
**
** The Prev opcode is only valid following an SeekLT, SeekLE, or
** OP_Last opcode used to position the cursor. Prev is not allowed
** to follow SeekGT, SeekGE, or OP_Rewind.
**
** The P1 cursor must be for a real table, not a pseudo-table. If P1 is ** The P1 cursor must be for a real table, not a pseudo-table. If P1 is
** not open then the behavior is undefined. ** not open then the behavior is undefined.
** **
@@ -4583,7 +4660,7 @@ case OP_Rewind: { /* jump */
*/ */
/* Opcode: PrevIfOpen P1 P2 P3 P4 P5 /* Opcode: PrevIfOpen P1 P2 P3 P4 P5
** **
** This opcode works just like OP_Prev except that if cursor P1 is not ** This opcode works just like Prev except that if cursor P1 is not
** open it behaves a no-op. ** open it behaves a no-op.
*/ */
case OP_SorterNext: { /* jump */ case OP_SorterNext: { /* jump */
@@ -4614,6 +4691,16 @@ case OP_Next: /* jump */
assert( pOp->opcode!=OP_Prev || pOp->p4.xAdvance==sqlite3BtreePrevious ); assert( pOp->opcode!=OP_Prev || pOp->p4.xAdvance==sqlite3BtreePrevious );
assert( pOp->opcode!=OP_NextIfOpen || pOp->p4.xAdvance==sqlite3BtreeNext ); assert( pOp->opcode!=OP_NextIfOpen || pOp->p4.xAdvance==sqlite3BtreeNext );
assert( pOp->opcode!=OP_PrevIfOpen || pOp->p4.xAdvance==sqlite3BtreePrevious); assert( pOp->opcode!=OP_PrevIfOpen || pOp->p4.xAdvance==sqlite3BtreePrevious);
/* The Next opcode is only used after SeekGT, SeekGE, and Rewind.
** The Prev opcode is only used after SeekLT, SeekLE, and Last. */
assert( pOp->opcode!=OP_Next || pOp->opcode!=OP_NextIfOpen
|| pC->seekOp==OP_SeekGT || pC->seekOp==OP_SeekGE
|| pC->seekOp==OP_Rewind );
assert( pOp->opcode!=OP_Prev || pOp->opcode!=OP_PrevIfOpen
|| pC->seekOp==OP_SeekLT || pC->seekOp==OP_SeekLE
|| pC->seekOp==OP_Last );
rc = pOp->p4.xAdvance(pC->pCursor, &res); rc = pOp->p4.xAdvance(pC->pCursor, &res);
next_tail: next_tail:
pC->cacheStatus = CACHE_STALE; pC->cacheStatus = CACHE_STALE;
@@ -5094,7 +5181,8 @@ case OP_LoadAnalysis: {
** **
** Remove the internal (in-memory) data structures that describe ** Remove the internal (in-memory) data structures that describe
** the table named P4 in database P1. This is called after a table ** the table named P4 in database P1. This is called after a table
** is dropped in order to keep the internal representation of the ** is dropped from disk (using the Destroy opcode) in order to keep
** the internal representation of the
** schema consistent with what is on disk. ** schema consistent with what is on disk.
*/ */
case OP_DropTable: { case OP_DropTable: {
@@ -5106,7 +5194,8 @@ case OP_DropTable: {
** **
** Remove the internal (in-memory) data structures that describe ** Remove the internal (in-memory) data structures that describe
** the index named P4 in database P1. This is called after an index ** the index named P4 in database P1. This is called after an index
** is dropped in order to keep the internal representation of the ** is dropped from disk (using the Destroy opcode)
** in order to keep the internal representation of the
** schema consistent with what is on disk. ** schema consistent with what is on disk.
*/ */
case OP_DropIndex: { case OP_DropIndex: {
@@ -5118,7 +5207,8 @@ case OP_DropIndex: {
** **
** Remove the internal (in-memory) data structures that describe ** Remove the internal (in-memory) data structures that describe
** the trigger named P4 in database P1. This is called after a trigger ** the trigger named P4 in database P1. This is called after a trigger
** is dropped in order to keep the internal representation of the ** is dropped from disk (using the Destroy opcode) in order to keep
** the internal representation of the
** schema consistent with what is on disk. ** schema consistent with what is on disk.
*/ */
case OP_DropTrigger: { case OP_DropTrigger: {

View File

@@ -68,6 +68,9 @@ struct VdbeCursor {
int pseudoTableReg; /* Register holding pseudotable content. */ int pseudoTableReg; /* Register holding pseudotable content. */
i16 nField; /* Number of fields in the header */ i16 nField; /* Number of fields in the header */
u16 nHdrParsed; /* Number of header fields parsed so far */ u16 nHdrParsed; /* Number of header fields parsed so far */
#ifdef SQLITE_DEBUG
u8 seekOp; /* Most recent seek operation on this cursor */
#endif
i8 iDb; /* Index of cursor database in db->aDb[] (or -1) */ i8 iDb; /* Index of cursor database in db->aDb[] (or -1) */
u8 nullRow; /* True if pointing to a row with no data */ u8 nullRow; /* True if pointing to a row with no data */
u8 rowidIsValid; /* True if lastRowid is valid */ u8 rowidIsValid; /* True if lastRowid is valid */

View File

@@ -2786,7 +2786,7 @@ int sqlite3VdbeCursorMoveto(VdbeCursor *p){
*/ */
u32 sqlite3VdbeSerialType(Mem *pMem, int file_format){ u32 sqlite3VdbeSerialType(Mem *pMem, int file_format){
int flags = pMem->flags; int flags = pMem->flags;
int n; u32 n;
if( flags&MEM_Null ){ if( flags&MEM_Null ){
return 0; return 0;
@@ -2816,11 +2816,11 @@ u32 sqlite3VdbeSerialType(Mem *pMem, int file_format){
return 7; return 7;
} }
assert( pMem->db->mallocFailed || flags&(MEM_Str|MEM_Blob) ); assert( pMem->db->mallocFailed || flags&(MEM_Str|MEM_Blob) );
n = pMem->n; assert( pMem->n>=0 );
n = (u32)pMem->n;
if( flags & MEM_Zero ){ if( flags & MEM_Zero ){
n += pMem->u.nZero; n += pMem->u.nZero;
} }
assert( n>=0 );
return ((n*2) + 12 + ((flags&MEM_Str)!=0)); return ((n*2) + 12 + ((flags&MEM_Str)!=0));
} }

View File

@@ -2047,7 +2047,8 @@ static int whereRangeSkipScanEst(
int nLower = -1; int nLower = -1;
int nUpper = p->nSample+1; int nUpper = p->nSample+1;
int rc = SQLITE_OK; int rc = SQLITE_OK;
u8 aff = p->pTable->aCol[ p->aiColumn[nEq] ].affinity; int iCol = p->aiColumn[nEq];
u8 aff = iCol>=0 ? p->pTable->aCol[iCol].affinity : SQLITE_AFF_INTEGER;
CollSeq *pColl; CollSeq *pColl;
sqlite3_value *p1 = 0; /* Value extracted from pLower */ sqlite3_value *p1 = 0; /* Value extracted from pLower */