mirror of
https://github.com/sqlite/sqlite.git
synced 2025-08-08 14:02:16 +03:00
Do not allow virtual table constructors to be called recursively.
FossilOrigin-Name: 0a72726da21581ab16cb3e964bd825b8f2e931e4
This commit is contained in:
@@ -1019,7 +1019,8 @@ static int fts3ContentColumns(
|
|||||||
const char *zTbl, /* Name of content table */
|
const char *zTbl, /* Name of content table */
|
||||||
const char ***pazCol, /* OUT: Malloc'd array of column names */
|
const char ***pazCol, /* OUT: Malloc'd array of column names */
|
||||||
int *pnCol, /* OUT: Size of array *pazCol */
|
int *pnCol, /* OUT: Size of array *pazCol */
|
||||||
int *pnStr /* OUT: Bytes of string content */
|
int *pnStr, /* OUT: Bytes of string content */
|
||||||
|
char **pzErr /* OUT: error message */
|
||||||
){
|
){
|
||||||
int rc = SQLITE_OK; /* Return code */
|
int rc = SQLITE_OK; /* Return code */
|
||||||
char *zSql; /* "SELECT *" statement on zTbl */
|
char *zSql; /* "SELECT *" statement on zTbl */
|
||||||
@@ -1030,6 +1031,9 @@ static int fts3ContentColumns(
|
|||||||
rc = SQLITE_NOMEM;
|
rc = SQLITE_NOMEM;
|
||||||
}else{
|
}else{
|
||||||
rc = sqlite3_prepare(db, zSql, -1, &pStmt, 0);
|
rc = sqlite3_prepare(db, zSql, -1, &pStmt, 0);
|
||||||
|
if( rc!=SQLITE_OK ){
|
||||||
|
*pzErr = sqlite3_mprintf("%s", sqlite3_errmsg(db));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
sqlite3_free(zSql);
|
sqlite3_free(zSql);
|
||||||
|
|
||||||
@@ -1281,7 +1285,7 @@ static int fts3InitVtab(
|
|||||||
if( nCol==0 ){
|
if( nCol==0 ){
|
||||||
sqlite3_free((void*)aCol);
|
sqlite3_free((void*)aCol);
|
||||||
aCol = 0;
|
aCol = 0;
|
||||||
rc = fts3ContentColumns(db, argv[1], zContent, &aCol, &nCol, &nString);
|
rc = fts3ContentColumns(db, argv[1], zContent,&aCol,&nCol,&nString,pzErr);
|
||||||
|
|
||||||
/* If a languageid= option was specified, remove the language id
|
/* If a languageid= option was specified, remove the language id
|
||||||
** column from the aCol[] array. */
|
** column from the aCol[] array. */
|
||||||
|
18
manifest
18
manifest
@@ -1,5 +1,5 @@
|
|||||||
C Fix\sincorrect\scolumn\snames\sin\sUPDATE\sstatements\sgenerated\sby\sthe\ssqldiff\nutility.
|
C Do\snot\sallow\svirtual\stable\sconstructors\sto\sbe\scalled\srecursively.
|
||||||
D 2015-04-09T19:39:54.853
|
D 2015-04-10T07:55:07.186
|
||||||
F Makefile.arm-wince-mingw32ce-gcc d6df77f1f48d690bd73162294bbba7f59507c72f
|
F Makefile.arm-wince-mingw32ce-gcc d6df77f1f48d690bd73162294bbba7f59507c72f
|
||||||
F Makefile.in 5f78b1ab81b64e7c57a75d170832443e66c0880a
|
F Makefile.in 5f78b1ab81b64e7c57a75d170832443e66c0880a
|
||||||
F Makefile.linux-gcc 91d710bdc4998cb015f39edf3cb314ec4f4d7e23
|
F Makefile.linux-gcc 91d710bdc4998cb015f39edf3cb314ec4f4d7e23
|
||||||
@@ -78,7 +78,7 @@ F ext/fts3/README.content fdc666a70d5257a64fee209f97cf89e0e6e32b51
|
|||||||
F ext/fts3/README.syntax a19711dc5458c20734b8e485e75fb1981ec2427a
|
F ext/fts3/README.syntax a19711dc5458c20734b8e485e75fb1981ec2427a
|
||||||
F ext/fts3/README.tokenizers e0a8b81383ea60d0334d274fadf305ea14a8c314
|
F ext/fts3/README.tokenizers e0a8b81383ea60d0334d274fadf305ea14a8c314
|
||||||
F ext/fts3/README.txt 8c18f41574404623b76917b9da66fcb0ab38328d
|
F ext/fts3/README.txt 8c18f41574404623b76917b9da66fcb0ab38328d
|
||||||
F ext/fts3/fts3.c 23bd9d37a777342f5c22a648e9b4b005dde9e58f
|
F ext/fts3/fts3.c 57d863c3bd360e575ecc293570af7c9b0bdd2209
|
||||||
F ext/fts3/fts3.h 3a10a0af180d502cecc50df77b1b22df142817fe
|
F ext/fts3/fts3.h 3a10a0af180d502cecc50df77b1b22df142817fe
|
||||||
F ext/fts3/fts3Int.h 394858c12a17740f7a1f6bd372c4606d4425a8d1
|
F ext/fts3/fts3Int.h 394858c12a17740f7a1f6bd372c4606d4425a8d1
|
||||||
F ext/fts3/fts3_aux.c 5c211e17a64885faeb16b9ba7772f9d5445c2365
|
F ext/fts3/fts3_aux.c 5c211e17a64885faeb16b9ba7772f9d5445c2365
|
||||||
@@ -302,7 +302,7 @@ F src/vdbeblob.c 4f2e8e075d238392df98c5e03a64342465b03f90
|
|||||||
F src/vdbemem.c c0dc81285b7571b0a31c40f17846fe2397ec1cd9
|
F src/vdbemem.c c0dc81285b7571b0a31c40f17846fe2397ec1cd9
|
||||||
F src/vdbesort.c 919717d7599fa31d343ec28bffd0f9e91a4ff5f6
|
F src/vdbesort.c 919717d7599fa31d343ec28bffd0f9e91a4ff5f6
|
||||||
F src/vdbetrace.c 7e4222955e07dd707a2f360c0eb73452be1cb010
|
F src/vdbetrace.c 7e4222955e07dd707a2f360c0eb73452be1cb010
|
||||||
F src/vtab.c 62d49237bd8f3be4863815a39387b0f9897fa5e1
|
F src/vtab.c ff722a886ed61e2e2889ee221b0a4f6dcaabb8e1
|
||||||
F src/vxworks.h c18586c8edc1bddbc15c004fa16aeb1e1342b4fb
|
F src/vxworks.h c18586c8edc1bddbc15c004fa16aeb1e1342b4fb
|
||||||
F src/wal.c 878c8e1a51cb2ec45c395d26b7d5cd9e1a098e4a
|
F src/wal.c 878c8e1a51cb2ec45c395d26b7d5cd9e1a098e4a
|
||||||
F src/wal.h df01efe09c5cb8c8e391ff1715cca294f89668a4
|
F src/wal.h df01efe09c5cb8c8e391ff1715cca294f89668a4
|
||||||
@@ -606,7 +606,7 @@ F test/fts3tok_err.test 52273cd193b9036282f7bacb43da78c6be87418d
|
|||||||
F test/fts3varint.test 752c08ed5d32c5d7dc211b056f4ed68a76b7e36e
|
F test/fts3varint.test 752c08ed5d32c5d7dc211b056f4ed68a76b7e36e
|
||||||
F test/fts4aa.test 10aac8e9d62c7357590acfabe3fad01e9a9ce1cb
|
F test/fts4aa.test 10aac8e9d62c7357590acfabe3fad01e9a9ce1cb
|
||||||
F test/fts4check.test 74d77f6cdb768ac49df5afda575cef14ae3d239a
|
F test/fts4check.test 74d77f6cdb768ac49df5afda575cef14ae3d239a
|
||||||
F test/fts4content.test 2e7252557d6d24afa101d9ba1de710d6140e6d06
|
F test/fts4content.test abb0c77bc3da3df64fec72e00844d2257a90025d
|
||||||
F test/fts4docid.test e33c383cfbdff0284685604d256f347a18fdbf01
|
F test/fts4docid.test e33c383cfbdff0284685604d256f347a18fdbf01
|
||||||
F test/fts4growth.test df10fde9f47cf5c71861e63fd8efcd573c4f7e53
|
F test/fts4growth.test df10fde9f47cf5c71861e63fd8efcd573c4f7e53
|
||||||
F test/fts4growth2.test 2f063be1902a73cd087355837c52fed42ac11a5d
|
F test/fts4growth2.test 2f063be1902a73cd087355837c52fed42ac11a5d
|
||||||
@@ -1250,7 +1250,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 5063f9070afde9374ea0f2bc338fee840d8b3dd4
|
P ee53b46011852e27db23708387fe1e918cc8284c
|
||||||
R b81a13af06affec3f4deb7b3b6092ef8
|
R 7e9242555b174ae9f2c42129917fe80c
|
||||||
U drh
|
U dan
|
||||||
Z 7a24354f8d3816810c976cc3a62d01ee
|
Z a71357da8ab26f740f6d9ebc28389132
|
||||||
|
@@ -1 +1 @@
|
|||||||
ee53b46011852e27db23708387fe1e918cc8284c
|
0a72726da21581ab16cb3e964bd825b8f2e931e4
|
33
src/vtab.c
33
src/vtab.c
@@ -24,6 +24,8 @@
|
|||||||
struct VtabCtx {
|
struct VtabCtx {
|
||||||
VTable *pVTable; /* The virtual table being constructed */
|
VTable *pVTable; /* The virtual table being constructed */
|
||||||
Table *pTab; /* The Table object to which the virtual table belongs */
|
Table *pTab; /* The Table object to which the virtual table belongs */
|
||||||
|
VtabCtx *pPrior; /* Parent context (if any) */
|
||||||
|
int bDeclared; /* True after sqlite3_declare_vtab() is called */
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -487,15 +489,27 @@ static int vtabCallConstructor(
|
|||||||
int (*xConstruct)(sqlite3*,void*,int,const char*const*,sqlite3_vtab**,char**),
|
int (*xConstruct)(sqlite3*,void*,int,const char*const*,sqlite3_vtab**,char**),
|
||||||
char **pzErr
|
char **pzErr
|
||||||
){
|
){
|
||||||
VtabCtx sCtx, *pPriorCtx;
|
VtabCtx sCtx;
|
||||||
VTable *pVTable;
|
VTable *pVTable;
|
||||||
int rc;
|
int rc;
|
||||||
const char *const*azArg = (const char *const*)pTab->azModuleArg;
|
const char *const*azArg = (const char *const*)pTab->azModuleArg;
|
||||||
int nArg = pTab->nModuleArg;
|
int nArg = pTab->nModuleArg;
|
||||||
char *zErr = 0;
|
char *zErr = 0;
|
||||||
char *zModuleName = sqlite3MPrintf(db, "%s", pTab->zName);
|
char *zModuleName;
|
||||||
int iDb;
|
int iDb;
|
||||||
|
VtabCtx *pCtx;
|
||||||
|
|
||||||
|
/* Check that the virtual-table is not already being initialized */
|
||||||
|
for(pCtx=db->pVtabCtx; pCtx; pCtx=pCtx->pPrior){
|
||||||
|
if( pCtx->pTab==pTab ){
|
||||||
|
*pzErr = sqlite3MPrintf(db,
|
||||||
|
"vtable constructor called recursively: %s", pTab->zName
|
||||||
|
);
|
||||||
|
return SQLITE_LOCKED;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
zModuleName = sqlite3MPrintf(db, "%s", pTab->zName);
|
||||||
if( !zModuleName ){
|
if( !zModuleName ){
|
||||||
return SQLITE_NOMEM;
|
return SQLITE_NOMEM;
|
||||||
}
|
}
|
||||||
@@ -516,11 +530,13 @@ static int vtabCallConstructor(
|
|||||||
assert( xConstruct );
|
assert( xConstruct );
|
||||||
sCtx.pTab = pTab;
|
sCtx.pTab = pTab;
|
||||||
sCtx.pVTable = pVTable;
|
sCtx.pVTable = pVTable;
|
||||||
pPriorCtx = db->pVtabCtx;
|
sCtx.pPrior = db->pVtabCtx;
|
||||||
|
sCtx.bDeclared = 0;
|
||||||
db->pVtabCtx = &sCtx;
|
db->pVtabCtx = &sCtx;
|
||||||
rc = xConstruct(db, pMod->pAux, nArg, azArg, &pVTable->pVtab, &zErr);
|
rc = xConstruct(db, pMod->pAux, nArg, azArg, &pVTable->pVtab, &zErr);
|
||||||
db->pVtabCtx = pPriorCtx;
|
db->pVtabCtx = sCtx.pPrior;
|
||||||
if( rc==SQLITE_NOMEM ) db->mallocFailed = 1;
|
if( rc==SQLITE_NOMEM ) db->mallocFailed = 1;
|
||||||
|
assert( sCtx.pTab==pTab );
|
||||||
|
|
||||||
if( SQLITE_OK!=rc ){
|
if( SQLITE_OK!=rc ){
|
||||||
if( zErr==0 ){
|
if( zErr==0 ){
|
||||||
@@ -536,7 +552,7 @@ static int vtabCallConstructor(
|
|||||||
memset(pVTable->pVtab, 0, sizeof(pVTable->pVtab[0]));
|
memset(pVTable->pVtab, 0, sizeof(pVTable->pVtab[0]));
|
||||||
pVTable->pVtab->pModule = pMod->pModule;
|
pVTable->pVtab->pModule = pMod->pModule;
|
||||||
pVTable->nRef = 1;
|
pVTable->nRef = 1;
|
||||||
if( sCtx.pTab ){
|
if( sCtx.bDeclared==0 ){
|
||||||
const char *zFormat = "vtable constructor did not declare schema: %s";
|
const char *zFormat = "vtable constructor did not declare schema: %s";
|
||||||
*pzErr = sqlite3MPrintf(db, zFormat, pTab->zName);
|
*pzErr = sqlite3MPrintf(db, zFormat, pTab->zName);
|
||||||
sqlite3VtabUnlock(pVTable);
|
sqlite3VtabUnlock(pVTable);
|
||||||
@@ -706,8 +722,8 @@ int sqlite3VtabCallCreate(sqlite3 *db, int iDb, const char *zTab, char **pzErr){
|
|||||||
** virtual table module.
|
** virtual table module.
|
||||||
*/
|
*/
|
||||||
int sqlite3_declare_vtab(sqlite3 *db, const char *zCreateTable){
|
int sqlite3_declare_vtab(sqlite3 *db, const char *zCreateTable){
|
||||||
|
VtabCtx *pCtx = db->pVtabCtx;
|
||||||
Parse *pParse;
|
Parse *pParse;
|
||||||
|
|
||||||
int rc = SQLITE_OK;
|
int rc = SQLITE_OK;
|
||||||
Table *pTab;
|
Table *pTab;
|
||||||
char *zErr = 0;
|
char *zErr = 0;
|
||||||
@@ -718,11 +734,12 @@ int sqlite3_declare_vtab(sqlite3 *db, const char *zCreateTable){
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
sqlite3_mutex_enter(db->mutex);
|
sqlite3_mutex_enter(db->mutex);
|
||||||
if( !db->pVtabCtx || !(pTab = db->pVtabCtx->pTab) ){
|
if( !pCtx || pCtx->bDeclared ){
|
||||||
sqlite3Error(db, SQLITE_MISUSE);
|
sqlite3Error(db, SQLITE_MISUSE);
|
||||||
sqlite3_mutex_leave(db->mutex);
|
sqlite3_mutex_leave(db->mutex);
|
||||||
return SQLITE_MISUSE_BKPT;
|
return SQLITE_MISUSE_BKPT;
|
||||||
}
|
}
|
||||||
|
pTab = pCtx->pTab;
|
||||||
assert( (pTab->tabFlags & TF_Virtual)!=0 );
|
assert( (pTab->tabFlags & TF_Virtual)!=0 );
|
||||||
|
|
||||||
pParse = sqlite3StackAllocZero(db, sizeof(*pParse));
|
pParse = sqlite3StackAllocZero(db, sizeof(*pParse));
|
||||||
@@ -745,7 +762,7 @@ int sqlite3_declare_vtab(sqlite3 *db, const char *zCreateTable){
|
|||||||
pParse->pNewTable->nCol = 0;
|
pParse->pNewTable->nCol = 0;
|
||||||
pParse->pNewTable->aCol = 0;
|
pParse->pNewTable->aCol = 0;
|
||||||
}
|
}
|
||||||
db->pVtabCtx->pTab = 0;
|
pCtx->bDeclared = 1;
|
||||||
}else{
|
}else{
|
||||||
sqlite3ErrorWithMsg(db, SQLITE_ERROR, (zErr ? "%s" : 0), zErr);
|
sqlite3ErrorWithMsg(db, SQLITE_ERROR, (zErr ? "%s" : 0), zErr);
|
||||||
sqlite3DbFree(db, zErr);
|
sqlite3DbFree(db, zErr);
|
||||||
|
@@ -48,6 +48,9 @@ ifcapable !fts3 {
|
|||||||
#
|
#
|
||||||
# 9.* - Test using content=xxx where xxx is a virtual table.
|
# 9.* - Test using content=xxx where xxx is a virtual table.
|
||||||
#
|
#
|
||||||
|
# 11.* - Test that circular references (e.g. "t1(content=t1)") are
|
||||||
|
# detected.
|
||||||
|
#
|
||||||
|
|
||||||
do_execsql_test 1.1.1 {
|
do_execsql_test 1.1.1 {
|
||||||
CREATE TABLE t1(a, b, c);
|
CREATE TABLE t1(a, b, c);
|
||||||
@@ -406,7 +409,7 @@ do_execsql_test 5.1.7 {
|
|||||||
#
|
#
|
||||||
do_catchsql_test 6.1.1 {
|
do_catchsql_test 6.1.1 {
|
||||||
CREATE VIRTUAL TABLE ft7 USING fts4(content=t7);
|
CREATE VIRTUAL TABLE ft7 USING fts4(content=t7);
|
||||||
} {1 {vtable constructor failed: ft7}}
|
} {1 {no such table: main.t7}}
|
||||||
|
|
||||||
do_execsql_test 6.2.1 {
|
do_execsql_test 6.2.1 {
|
||||||
CREATE TABLE t7(one, two);
|
CREATE TABLE t7(one, two);
|
||||||
@@ -433,7 +436,7 @@ do_execsql_test 6.2.3 {
|
|||||||
}
|
}
|
||||||
do_catchsql_test 6.2.4 {
|
do_catchsql_test 6.2.4 {
|
||||||
SELECT * FROM ft7;
|
SELECT * FROM ft7;
|
||||||
} {1 {vtable constructor failed: ft7}}
|
} {1 {no such table: main.t7}}
|
||||||
do_execsql_test 6.2.5 {
|
do_execsql_test 6.2.5 {
|
||||||
CREATE TABLE t7(x, y);
|
CREATE TABLE t7(x, y);
|
||||||
INSERT INTO t7 VALUES('A B', 'B A');
|
INSERT INTO t7 VALUES('A B', 'B A');
|
||||||
@@ -622,4 +625,15 @@ do_execsql_test 10.7 {
|
|||||||
{...c d [e] f g...}
|
{...c d [e] f g...}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#-------------------------------------------------------------------------
|
||||||
|
# Test cases 11.*
|
||||||
|
#
|
||||||
|
reset_db
|
||||||
|
|
||||||
|
do_catchsql_test 11.1 {
|
||||||
|
CREATE VIRTUAL TABLE x1 USING fts4(content=x1);
|
||||||
|
} {1 {vtable constructor called recursively: x1}}
|
||||||
|
|
||||||
|
|
||||||
finish_test
|
finish_test
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user