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:
26
manifest
26
manifest
@@ -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
|
||||||
|
@@ -1 +1 @@
|
|||||||
9af50a878f67c1c2a4f1520160cc989650d7196a
|
163c247bd8280ab14fe577329c631c8bd884707f
|
139
src/analyze.c
139
src/analyze.c
@@ -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);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -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);
|
||||||
|
166
src/vdbe.c
166
src/vdbe.c
@@ -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: {
|
||||||
|
@@ -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 */
|
||||||
|
@@ -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));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -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 */
|
||||||
|
Reference in New Issue
Block a user