Enhance the "PRAGMA integrity_check" statement so that it verifies datatype
constraints on non-STRICT tables: (1) Columns with TEXT affinity should not contain numeric values, and (2) columns with NUMERIC affinity should not contain TEXT values that could be converted into numbers. FossilOrigin-Name: aa6e9086196248675b9e33dbfaad9ef67097432535896790f80d2470002978b1
This commit is contained in:
commit
7747ffba47
19
manifest
19
manifest
@ -1,5 +1,5 @@
|
||||
C Improved\sdetection\sof\sdatabase\scorrupting\swhen\smoving\spages\son\san\nautovacuumed\sdatabase\swhen\screating\sa\snew\sbtree.\ndbsqlfuzz\s9a55397eae13cec64aebf1fd35489c3a90bdaac5
|
||||
D 2022-10-10T12:02:53.891
|
||||
C Enhance\sthe\s"PRAGMA\sintegrity_check"\sstatement\sso\sthat\sit\sverifies\sdatatype\nconstraints\son\snon-STRICT\stables:\s\s(1)\sColumns\swith\sTEXT\saffinity\sshould\snot\ncontain\snumeric\svalues,\sand\s(2)\scolumns\swith\sNUMERIC\saffinity\sshould\snot\ncontain\sTEXT\svalues\sthat\scould\sbe\sconverted\sinto\snumbers.
|
||||
D 2022-10-10T23:54:17.674
|
||||
F .fossil-settings/empty-dirs dbb81e8fc0401ac46a1491ab34a7f2c7c0452f2f06b54ebb845d024ca8283ef1
|
||||
F .fossil-settings/ignore-glob 35175cdfcf539b2318cb04a9901442804be81cd677d8b889fcc9149c21f239ea
|
||||
F LICENSE.md df5091916dbb40e6e9686186587125e1b2ff51f022cc334e886c19a0e9982724
|
||||
@ -581,7 +581,7 @@ F src/parse.y 8e67d820030d2655b9942ffe61c1e7e6b96cea2f2f72183533299393907d0564
|
||||
F src/pcache.c f4268f7f73c6a3db12ce22fd25bc68dc42315d19599414ab1207d7cf32f79197
|
||||
F src/pcache.h 4f87acd914cef5016fae3030343540d75f5b85a1877eed1a2a19b9f284248586
|
||||
F src/pcache1.c dee95e3cd2b61e6512dc814c5ab76d5eb36f0bfc9441dbb4260fccc0d12bbddc
|
||||
F src/pragma.c 9bf7d8a2a9ad3bc36df3ec0d61817a44c38a1da527d59c26c203047f906e334a
|
||||
F src/pragma.c 444f596c9e9123d19451ad762917f0f7545e4acecb10f67e09a19f01289cefce
|
||||
F src/pragma.h e690a356c18e98414d2e870ea791c1be1545a714ba623719deb63f7f226d8bb7
|
||||
F src/prepare.c 1b02be0441eda4579471fea097f678effcbb77ef0c39ab3f703c837822bcd674
|
||||
F src/printf.c e99ee9741e79ae3873458146f59644276657340385ade4e76a5f5d1c25793764
|
||||
@ -655,12 +655,12 @@ F src/threads.c 4ae07fa022a3dc7c5beb373cf744a85d3c5c6c3c
|
||||
F src/tokenize.c 1305797eab3542a0896b552c6e7669c972c1468e11e92b370533c1f37a37082b
|
||||
F src/treeview.c 07787f67cd297a6d09d04b8d70c06769c60c9c1d9080378f93929c16f8fd3298
|
||||
F src/trigger.c 4163ada044af89d51caba1cb713a73165347b2ec05fe84a283737c134d61fcd5
|
||||
F src/update.c c52a7991bece0453d22c77c08469512ee2f1391c12503fd347d1c939220c5877
|
||||
F src/update.c 5b0302c47cf31b533d5dff04c497ca1d8b9d89c39727e633fbe7b882fd5ac5aa
|
||||
F src/upsert.c 8789047a8f0a601ea42fa0256d1ba3190c13746b6ba940fe2d25643a7e991937
|
||||
F src/utf.c ee39565f0843775cc2c81135751ddd93eceb91a673ea2c57f61c76f288b041a0
|
||||
F src/util.c 0be191521ff6d2805995f4910f0b6231b42843678b2efdc1abecaf39929a673f
|
||||
F src/vacuum.c bb346170b0b54c6683bba4a5983aea40485597fdf605c87ec8bc2e199fe88cd8
|
||||
F src/vdbe.c a26eef5a6c0ebb97f83ec18bd483e0d17f0c6b079e7650085f2ef6a82fd78d99
|
||||
F src/vdbe.c 387270d575129b0c6ab0d19dad6de5abc1576ce6cc97f30d233ee13775ea4527
|
||||
F src/vdbe.h 64619af62603dc3c4f5ff6ff6d2c8f389abd667a29ce6007ed44bd22b3211cd0
|
||||
F src/vdbeInt.h 17b7461ffcf9ee760d1341731715a419f6b8c763089a7ece25c2e8098d702b3f
|
||||
F src/vdbeapi.c fc3183daf72808b4311b228989120fdbc2dc44972fb0d77d5c453460cc0e5b2c
|
||||
@ -1262,7 +1262,7 @@ F test/memdb1.test 2c4e9cc10d21c6bf4e217d72b7f6b8ba9b2605971bb2c5e6df76018e189f9
|
||||
F test/memjournal.test 70f3a00c7f84ee2978ad14e831231caa1e7f23915a2c54b4f775a021d5740c6c
|
||||
F test/memjournal2.test 6b9083cfaab9a3281ec545c3da2487999e8025fb7501bbae10f713f80c56454c
|
||||
F test/memleak.test 10b9c6c57e19fc68c32941495e9ba1c50123f6e2
|
||||
F test/memsubsys1.test 9e7555a22173b8f1c96c281ce289b338fcba2abe8b157f8798ca195bbf1d347e
|
||||
F test/memsubsys1.test 86b8158752af9188ed5b32a30674a1ef71183e6bc4e6808e815cd658ca9058a6
|
||||
F test/memsubsys2.test 3e4a8d0c05fd3e5fa92017c64666730a520c7e08
|
||||
F test/merge1.test 2de6d6ef8d25402764b1aab49d8f9d7f89208c89a6674e437f76de4c812157b8
|
||||
F test/minmax.test fe638b55d77d2375531a8f549b338eafcd9adfbd2f72df37ed77d9b26ca0a71a
|
||||
@ -2002,8 +2002,9 @@ F vsixtest/vsixtest.tcl 6a9a6ab600c25a91a7acc6293828957a386a8a93
|
||||
F vsixtest/vsixtest.vcxproj.data 2ed517e100c66dc455b492e1a33350c1b20fbcdc
|
||||
F vsixtest/vsixtest.vcxproj.filters 37e51ffedcdb064aad6ff33b6148725226cd608e
|
||||
F vsixtest/vsixtest_TemporaryKey.pfx e5b1b036facdb453873e7084e1cae9102ccc67a0
|
||||
P 1e2796b3741a4c0b966e6c01a7d62dea8de9d1f4bededea3d7ba7004dabc2ffe
|
||||
R 87396513be591e6dc5cb0d1c97d70179
|
||||
P 327965bc71023d9380a9d6805062659108dab4bfcd386b7aba813754f270d33a a140173102febe9ef8064ee9b95bee489db54caba149e577d69e4d75161bf390
|
||||
R 27cc3a1e54ea78f4322ce43e3f2008e7
|
||||
T +closed a140173102febe9ef8064ee9b95bee489db54caba149e577d69e4d75161bf390
|
||||
U drh
|
||||
Z bc3536edefe4a38a9e628abd562bbf72
|
||||
Z e702316a9b12d63d22f572b11832de1b
|
||||
# Remove this line to create a well-formed Fossil manifest.
|
||||
|
@ -1 +1 @@
|
||||
327965bc71023d9380a9d6805062659108dab4bfcd386b7aba813754f270d33a
|
||||
aa6e9086196248675b9e33dbfaad9ef67097432535896790f80d2470002978b1
|
142
src/pragma.c
142
src/pragma.c
@ -1752,8 +1752,9 @@ void sqlite3Pragma(
|
||||
int loopTop;
|
||||
int iDataCur, iIdxCur;
|
||||
int r1 = -1;
|
||||
int bStrict;
|
||||
int bStrict; /* True for a STRICT table */
|
||||
int r2; /* Previous key for WITHOUT ROWID tables */
|
||||
int mxCol; /* Maximum non-virtual column number */
|
||||
|
||||
if( !IsOrdinaryTable(pTab) ) continue;
|
||||
if( pObjTab && pObjTab!=pTab ) continue;
|
||||
@ -1778,11 +1779,24 @@ void sqlite3Pragma(
|
||||
assert( sqlite3NoTempsInRange(pParse,1,7+j) );
|
||||
sqlite3VdbeAddOp2(v, OP_Rewind, iDataCur, 0); VdbeCoverage(v);
|
||||
loopTop = sqlite3VdbeAddOp2(v, OP_AddImm, 7, 1);
|
||||
|
||||
/* Fetch the right-most column from the table. This will cause
|
||||
** the entire record header to be parsed and sanity checked. It
|
||||
** will also prepopulate the cursor column cache that is used
|
||||
** by the OP_IsType code, so it is a required step.
|
||||
*/
|
||||
mxCol = pTab->nCol-1;
|
||||
while( mxCol>=0
|
||||
&& ((pTab->aCol[mxCol].colFlags & COLFLAG_VIRTUAL)!=0
|
||||
|| pTab->iPKey==mxCol) ) mxCol--;
|
||||
if( mxCol>=0 ){
|
||||
sqlite3ExprCodeGetColumnOfTable(v, pTab, iDataCur, mxCol, 3);
|
||||
if( sqlite3VdbeGetLastOp(v)->opcode==OP_Column ){
|
||||
sqlite3VdbeChangeP5(v, OPFLAG_TYPEOFARG);
|
||||
}
|
||||
}
|
||||
|
||||
if( !isQuick ){
|
||||
/* Sanity check on record header decoding */
|
||||
sqlite3VdbeAddOp3(v, OP_Column, iDataCur, pTab->nNVCol-1,3);
|
||||
sqlite3VdbeChangeP5(v, OPFLAG_TYPEOFARG);
|
||||
VdbeComment((v, "(right-most column)"));
|
||||
if( pPk ){
|
||||
/* Verify WITHOUT ROWID keys are in ascending order */
|
||||
int a1;
|
||||
@ -1802,44 +1816,122 @@ void sqlite3Pragma(
|
||||
}
|
||||
}
|
||||
}
|
||||
/* Verify that all NOT NULL columns really are NOT NULL. At the
|
||||
** same time verify the type of the content of STRICT tables */
|
||||
/* Verify datatypes for all columns:
|
||||
**
|
||||
** (1) NOT NULL columns may not contain a NULL
|
||||
** (2) Datatype must be exact for non-ANY columns in STRICT tables
|
||||
** (3) Datatype for TEXT columns in non-STRICT tables must be
|
||||
** NULL, TEXT, or BLOB.
|
||||
** (4) Datatype for numeric columns in non-STRICT tables must not
|
||||
** be a TEXT value that can be losslessly converted to numeric.
|
||||
*/
|
||||
bStrict = (pTab->tabFlags & TF_Strict)!=0;
|
||||
for(j=0; j<pTab->nCol; j++){
|
||||
char *zErr;
|
||||
Column *pCol = pTab->aCol + j;
|
||||
int doError, jmp2;
|
||||
Column *pCol = pTab->aCol + j; /* The column to be checked */
|
||||
int labelError; /* Jump here to report an error */
|
||||
int labelOk; /* Jump here if all looks ok */
|
||||
int p1, p3, p4; /* Operands to the OP_IsType opcode */
|
||||
int doTypeCheck; /* Check datatypes (besides NOT NULL) */
|
||||
|
||||
if( j==pTab->iPKey ) continue;
|
||||
if( pCol->notNull==0 && !bStrict ) continue;
|
||||
doError = bStrict ? sqlite3VdbeMakeLabel(pParse) : 0;
|
||||
sqlite3ExprCodeGetColumnOfTable(v, pTab, iDataCur, j, 3);
|
||||
if( sqlite3VdbeGetLastOp(v)->opcode==OP_Column ){
|
||||
sqlite3VdbeChangeP5(v, OPFLAG_TYPEOFARG);
|
||||
if( bStrict ){
|
||||
doTypeCheck = pCol->eCType>COLTYPE_ANY;
|
||||
}else{
|
||||
doTypeCheck = pCol->affinity>SQLITE_AFF_BLOB;
|
||||
}
|
||||
if( pCol->notNull==0 && !doTypeCheck ) continue;
|
||||
|
||||
/* Compute the operands that will be needed for OP_IsType */
|
||||
p4 = SQLITE_NULL;
|
||||
if( pCol->colFlags & COLFLAG_VIRTUAL ){
|
||||
sqlite3ExprCodeGetColumnOfTable(v, pTab, iDataCur, j, 3);
|
||||
p1 = -1;
|
||||
p3 = 3;
|
||||
}else{
|
||||
if( pCol->iDflt ){
|
||||
sqlite3_value *pDfltValue = 0;
|
||||
sqlite3ValueFromExpr(db, sqlite3ColumnExpr(pTab,pCol), ENC(db),
|
||||
pCol->affinity, &pDfltValue);
|
||||
if( pDfltValue ){
|
||||
p4 = sqlite3_value_type(pDfltValue);
|
||||
sqlite3ValueFree(pDfltValue);
|
||||
}
|
||||
}
|
||||
p1 = iDataCur;
|
||||
if( !HasRowid(pTab) ){
|
||||
testcase( j!=sqlite3TableColumnToStorage(pTab, j) );
|
||||
p3 = sqlite3TableColumnToIndex(sqlite3PrimaryKeyIndex(pTab), j);
|
||||
}else{
|
||||
p3 = sqlite3TableColumnToStorage(pTab,j);
|
||||
testcase( p3!=j);
|
||||
}
|
||||
}
|
||||
|
||||
labelError = sqlite3VdbeMakeLabel(pParse);
|
||||
labelOk = sqlite3VdbeMakeLabel(pParse);
|
||||
if( pCol->notNull ){
|
||||
jmp2 = sqlite3VdbeAddOp1(v, OP_NotNull, 3); VdbeCoverage(v);
|
||||
/* (1) NOT NULL columns may not contain a NULL */
|
||||
int jmp2 = sqlite3VdbeAddOp4Int(v, OP_IsType, p1, labelOk, p3, p4);
|
||||
sqlite3VdbeChangeP5(v, 0x0f);
|
||||
VdbeCoverage(v);
|
||||
zErr = sqlite3MPrintf(db, "NULL value in %s.%s", pTab->zName,
|
||||
pCol->zCnName);
|
||||
sqlite3VdbeAddOp4(v, OP_String8, 0, 3, 0, zErr, P4_DYNAMIC);
|
||||
if( bStrict && pCol->eCType!=COLTYPE_ANY ){
|
||||
sqlite3VdbeGoto(v, doError);
|
||||
if( doTypeCheck ){
|
||||
sqlite3VdbeGoto(v, labelError);
|
||||
sqlite3VdbeJumpHere(v, jmp2);
|
||||
}else{
|
||||
integrityCheckResultRow(v);
|
||||
/* VDBE byte code will fall thru */
|
||||
}
|
||||
sqlite3VdbeJumpHere(v, jmp2);
|
||||
}
|
||||
if( bStrict && pCol->eCType!=COLTYPE_ANY ){
|
||||
jmp2 = sqlite3VdbeAddOp3(v, OP_IsNullOrType, 3, 0,
|
||||
sqlite3StdTypeMap[pCol->eCType-1]);
|
||||
if( bStrict && doTypeCheck ){
|
||||
/* (2) Datatype must be exact for non-ANY columns in STRICT tables*/
|
||||
static unsigned char aStdTypeMask[] = {
|
||||
0x1f, /* ANY */
|
||||
0x18, /* BLOB */
|
||||
0x11, /* INT */
|
||||
0x11, /* INTEGER */
|
||||
0x13, /* REAL */
|
||||
0x14 /* TEXT */
|
||||
};
|
||||
sqlite3VdbeAddOp4Int(v, OP_IsType, p1, labelOk, p3, p4);
|
||||
assert( pCol->eCType>=1 && pCol->eCType<=sizeof(aStdTypeMask) );
|
||||
sqlite3VdbeChangeP5(v, aStdTypeMask[pCol->eCType-1]);
|
||||
VdbeCoverage(v);
|
||||
zErr = sqlite3MPrintf(db, "non-%s value in %s.%s",
|
||||
sqlite3StdType[pCol->eCType-1],
|
||||
pTab->zName, pTab->aCol[j].zCnName);
|
||||
sqlite3VdbeAddOp4(v, OP_String8, 0, 3, 0, zErr, P4_DYNAMIC);
|
||||
sqlite3VdbeResolveLabel(v, doError);
|
||||
integrityCheckResultRow(v);
|
||||
sqlite3VdbeJumpHere(v, jmp2);
|
||||
}else if( !bStrict && pCol->affinity==SQLITE_AFF_TEXT ){
|
||||
/* (3) Datatype for TEXT columns in non-STRICT tables must be
|
||||
** NULL, TEXT, or BLOB. */
|
||||
sqlite3VdbeAddOp4Int(v, OP_IsType, p1, labelOk, p3, p4);
|
||||
sqlite3VdbeChangeP5(v, 0x1c); /* NULL, TEXT, or BLOB */
|
||||
VdbeCoverage(v);
|
||||
zErr = sqlite3MPrintf(db, "NUMERIC value in %s.%s",
|
||||
pTab->zName, pTab->aCol[j].zCnName);
|
||||
sqlite3VdbeAddOp4(v, OP_String8, 0, 3, 0, zErr, P4_DYNAMIC);
|
||||
}else if( !bStrict && pCol->affinity>=SQLITE_AFF_NUMERIC ){
|
||||
/* (4) Datatype for numeric columns in non-STRICT tables must not
|
||||
** be a TEXT value that can be converted to numeric. */
|
||||
sqlite3VdbeAddOp4Int(v, OP_IsType, p1, labelOk, p3, p4);
|
||||
sqlite3VdbeChangeP5(v, 0x1b); /* NULL, INT, FLOAT, or BLOB */
|
||||
VdbeCoverage(v);
|
||||
if( p1>=0 ){
|
||||
sqlite3ExprCodeGetColumnOfTable(v, pTab, iDataCur, j, 3);
|
||||
}
|
||||
sqlite3VdbeAddOp4(v, OP_Affinity, 3, 1, 0, "C", P4_STATIC);
|
||||
sqlite3VdbeAddOp4Int(v, OP_IsType, -1, labelOk, 3, p4);
|
||||
sqlite3VdbeChangeP5(v, 0x1c); /* NULL, TEXT, or BLOB */
|
||||
VdbeCoverage(v);
|
||||
zErr = sqlite3MPrintf(db, "TEXT value in %s.%s",
|
||||
pTab->zName, pTab->aCol[j].zCnName);
|
||||
sqlite3VdbeAddOp4(v, OP_String8, 0, 3, 0, zErr, P4_DYNAMIC);
|
||||
}
|
||||
sqlite3VdbeResolveLabel(v, labelError);
|
||||
integrityCheckResultRow(v);
|
||||
sqlite3VdbeResolveLabel(v, labelOk);
|
||||
}
|
||||
/* Verify CHECK constraints */
|
||||
if( pTab->pCheck && (db->flags & SQLITE_IgnoreChecks)==0 ){
|
||||
|
@ -59,11 +59,14 @@ static void updateVirtualTable(
|
||||
** it has been converted into REAL.
|
||||
*/
|
||||
void sqlite3ColumnDefault(Vdbe *v, Table *pTab, int i, int iReg){
|
||||
Column *pCol;
|
||||
assert( pTab!=0 );
|
||||
if( !IsView(pTab) ){
|
||||
assert( pTab->nCol>i );
|
||||
pCol = &pTab->aCol[i];
|
||||
if( pCol->iDflt ){
|
||||
sqlite3_value *pValue = 0;
|
||||
u8 enc = ENC(sqlite3VdbeDb(v));
|
||||
Column *pCol = &pTab->aCol[i];
|
||||
assert( !IsView(pTab) );
|
||||
VdbeComment((v, "%s.%s", pTab->zName, pCol->zCnName));
|
||||
assert( i<pTab->nCol );
|
||||
sqlite3ValueFromExpr(sqlite3VdbeDb(v),
|
||||
@ -74,7 +77,7 @@ void sqlite3ColumnDefault(Vdbe *v, Table *pTab, int i, int iReg){
|
||||
}
|
||||
}
|
||||
#ifndef SQLITE_OMIT_FLOATING_POINT
|
||||
if( pTab->aCol[i].affinity==SQLITE_AFF_REAL && !IsVirtual(pTab) ){
|
||||
if( pCol->affinity==SQLITE_AFF_REAL && !IsVirtual(pTab) ){
|
||||
sqlite3VdbeAddOp1(v, OP_RealAffinity, iReg);
|
||||
}
|
||||
#endif
|
||||
|
69
src/vdbe.c
69
src/vdbe.c
@ -2588,19 +2588,66 @@ case OP_IsNull: { /* same as TK_ISNULL, jump, in1 */
|
||||
break;
|
||||
}
|
||||
|
||||
/* Opcode: IsNullOrType P1 P2 P3 * *
|
||||
** Synopsis: if typeof(r[P1]) IN (P3,5) goto P2
|
||||
/* Opcode: IsType P1 P2 P3 P4 P5
|
||||
** Synopsis: if typeof(P1.P3) in P5 goto P2
|
||||
**
|
||||
** Jump to P2 if the type of a column in a btree is one of the types specified
|
||||
** by the P5 bitmask.
|
||||
**
|
||||
** P1 is normally a cursor on a btree for which the row decode cache is
|
||||
** valid through at least column P3. In other words, there should have been
|
||||
** a prior OP_Column for column P3 or greater. If the cursor is not valid,
|
||||
** then this opcode might give spurious results.
|
||||
** The the btree row has fewer than P3 columns, then use P4 as the
|
||||
** datatype.
|
||||
**
|
||||
** If P1 is -1, then P3 is a register number and the datatype is taken
|
||||
** from the value in that register.
|
||||
**
|
||||
** P5 is a bitmask of data types. SQLITE_INTEGER is the least significant
|
||||
** (0x01) bit. SQLITE_FLOAT is the 0x02 bit. SQLITE_TEXT is 0x04.
|
||||
** SQLITE_BLOB is 0x08. SQLITE_NULL is 0x10.
|
||||
**
|
||||
** Take the jump to address P2 if and only if the datatype of the
|
||||
** value determined by P1 and P3 corresponds to one of the bits in the
|
||||
** P5 bitmask.
|
||||
**
|
||||
** Jump to P2 if the value in register P1 is NULL or has a datatype P3.
|
||||
** P3 is an integer which should be one of SQLITE_INTEGER, SQLITE_FLOAT,
|
||||
** SQLITE_BLOB, SQLITE_NULL, or SQLITE_TEXT.
|
||||
*/
|
||||
case OP_IsNullOrType: { /* jump, in1 */
|
||||
int doTheJump;
|
||||
pIn1 = &aMem[pOp->p1];
|
||||
doTheJump = (pIn1->flags & MEM_Null)!=0 || sqlite3_value_type(pIn1)==pOp->p3;
|
||||
VdbeBranchTaken( doTheJump, 2);
|
||||
if( doTheJump ) goto jump_to_p2;
|
||||
case OP_IsType: { /* jump */
|
||||
VdbeCursor *pC;
|
||||
u16 typeMask;
|
||||
u32 serialType;
|
||||
|
||||
assert( pOp->p1>=(-1) && pOp->p1<p->nCursor );
|
||||
assert( pOp->p1>=0 || (pOp->p3>=0 && pOp->p3<=(p->nMem+1 - p->nCursor)) );
|
||||
if( pOp->p1>=0 ){
|
||||
pC = p->apCsr[pOp->p1];
|
||||
assert( pC!=0 );
|
||||
assert( pOp->p3>=0 );
|
||||
if( pOp->p3<pC->nHdrParsed ){
|
||||
serialType = pC->aType[pOp->p3];
|
||||
if( serialType==0 ){
|
||||
typeMask = 0x10; /* SQLITE_NULL */
|
||||
}else if( serialType==7 ){
|
||||
typeMask = 0x02; /* SQLITE_FLOAT */
|
||||
}else if( serialType<12 ){
|
||||
typeMask = 0x01; /* SQLITE_INTEGER */
|
||||
}else if( serialType&1 ){
|
||||
typeMask = 0x04; /* SQLITE_TEXT */
|
||||
}else{
|
||||
typeMask = 0x08; /* SQLITE_BLOB */
|
||||
}
|
||||
}else{
|
||||
typeMask = 1 << (pOp->p4.i - 1);
|
||||
}
|
||||
}else{
|
||||
assert( memIsValid(&aMem[pOp->p3]) );
|
||||
typeMask = 1 << (sqlite3_value_type((sqlite3_value*)&aMem[pOp->p3])-1);
|
||||
}
|
||||
VdbeBranchTaken( (typeMask & pOp->p5)!=0, 2);
|
||||
if( typeMask & pOp->p5 ){
|
||||
goto jump_to_p2;
|
||||
}
|
||||
break;
|
||||
}
|
||||
|
||||
|
@ -174,7 +174,7 @@ do_test memsubsys1-4.4 {
|
||||
} 0
|
||||
do_test memsubsys1-4.5 {
|
||||
set maxreq [lindex [sqlite3_status SQLITE_STATUS_MALLOC_SIZE 0] 2]
|
||||
expr {$maxreq<7000}
|
||||
expr {$maxreq<9000}
|
||||
} 1
|
||||
|
||||
db close
|
||||
|
Loading…
Reference in New Issue
Block a user