Index: third_party/sqlite/src/src/analyze.c |
diff --git a/third_party/sqlite/src/src/analyze.c b/third_party/sqlite/src/src/analyze.c |
index 01b4ad44b9f579887c6c9e0f40564dcd24cc7e62..17c1de83a98bdec3bb98c917a5bbdec2927bc0fc 100644 |
--- a/third_party/sqlite/src/src/analyze.c |
+++ b/third_party/sqlite/src/src/analyze.c |
@@ -10,8 +10,6 @@ |
** |
************************************************************************* |
** This file contains code associated with the ANALYZE command. |
-** |
-** @(#) $Id: analyze.c,v 1.52 2009/04/16 17:45:48 drh Exp $ |
*/ |
#ifndef SQLITE_OMIT_ANALYZE |
#include "sqliteInt.h" |
@@ -36,9 +34,10 @@ static void openStatTable( |
Parse *pParse, /* Parsing context */ |
int iDb, /* The database we are looking in */ |
int iStatCur, /* Open the sqlite_stat1 table on this cursor */ |
- const char *zWhere /* Delete entries associated with this table */ |
+ const char *zWhere, /* Delete entries for this table or index */ |
+ const char *zWhereType /* Either "tbl" or "idx" */ |
){ |
- static struct { |
+ static const struct { |
const char *zName; |
const char *zCols; |
} aTable[] = { |
@@ -81,7 +80,7 @@ static void openStatTable( |
sqlite3TableLock(pParse, iDb, aRoot[i], 1, zTab); |
if( zWhere ){ |
sqlite3NestedParse(pParse, |
- "DELETE FROM %Q.%s WHERE tbl=%Q", pDb->zName, zTab, zWhere |
+ "DELETE FROM %Q.%s WHERE %s=%Q", pDb->zName, zTab, zWhereType, zWhere |
); |
}else{ |
/* The sqlite_stat[12] table already exists. Delete all rows. */ |
@@ -105,6 +104,7 @@ static void openStatTable( |
static void analyzeOneTable( |
Parse *pParse, /* Parser context */ |
Table *pTab, /* Table whose indices are to be analyzed */ |
+ Index *pOnlyIdx, /* If not NULL, only analyze this one index */ |
int iStatCur, /* Index of VdbeCursor that writes the sqlite_stat1 table */ |
int iMem /* Available memory locations begin here */ |
){ |
@@ -115,7 +115,7 @@ static void analyzeOneTable( |
int i; /* Loop counter */ |
int topOfLoop; /* The top of the loop */ |
int endOfLoop; /* The end of the loop */ |
- int addr; /* The address of an instruction */ |
+ int jZeroRows = -1; /* Jump from here if number of rows is zero */ |
int iDb; /* Index of database containing pTab */ |
int regTabname = iMem++; /* Register containing table name */ |
int regIdxname = iMem++; /* Register containing index name */ |
@@ -126,6 +126,7 @@ static void analyzeOneTable( |
int regRowid = iMem++; /* Rowid for the inserted record */ |
#ifdef SQLITE_ENABLE_STAT2 |
+ int addr = 0; /* Instruction address */ |
int regTemp2 = iMem++; /* Temporary use register */ |
int regSamplerecno = iMem++; /* Index of next sample to record */ |
int regRecno = iMem++; /* Current sample index */ |
@@ -134,13 +135,21 @@ static void analyzeOneTable( |
#endif |
v = sqlite3GetVdbe(pParse); |
- if( v==0 || NEVER(pTab==0) || pTab->pIndex==0 ){ |
- /* Do no analysis for tables that have no indices */ |
+ if( v==0 || NEVER(pTab==0) ){ |
+ return; |
+ } |
+ if( pTab->tnum==0 ){ |
+ /* Do not gather statistics on views or virtual tables */ |
+ return; |
+ } |
+ if( memcmp(pTab->zName, "sqlite_", 7)==0 ){ |
+ /* Do not gather statistics on system tables */ |
return; |
} |
assert( sqlite3BtreeHoldsAllMutexes(db) ); |
iDb = sqlite3SchemaToIndex(db, pTab->pSchema); |
assert( iDb>=0 ); |
+ assert( sqlite3SchemaMutexHeld(db, iDb, 0) ); |
#ifndef SQLITE_OMIT_AUTHORIZATION |
if( sqlite3AuthCheck(pParse, SQLITE_ANALYZE, pTab->zName, 0, |
db->aDb[iDb].zName ) ){ |
@@ -152,10 +161,14 @@ static void analyzeOneTable( |
sqlite3TableLock(pParse, iDb, pTab->tnum, 0, pTab->zName); |
iIdxCur = pParse->nTab++; |
+ sqlite3VdbeAddOp4(v, OP_String8, 0, regTabname, 0, pTab->zName, 0); |
for(pIdx=pTab->pIndex; pIdx; pIdx=pIdx->pNext){ |
- int nCol = pIdx->nColumn; |
- KeyInfo *pKey = sqlite3IndexKeyinfo(pParse, pIdx); |
+ int nCol; |
+ KeyInfo *pKey; |
+ if( pOnlyIdx && pOnlyIdx!=pIdx ) continue; |
+ nCol = pIdx->nColumn; |
+ pKey = sqlite3IndexKeyinfo(pParse, pIdx); |
if( iMem+1+(nCol*2)>pParse->nMem ){ |
pParse->nMem = iMem+1+(nCol*2); |
} |
@@ -166,10 +179,7 @@ static void analyzeOneTable( |
(char *)pKey, P4_KEYINFO_HANDOFF); |
VdbeComment((v, "%s", pIdx->zName)); |
- /* Populate the registers containing the table and index names. */ |
- if( pTab->pIndex==pIdx ){ |
- sqlite3VdbeAddOp4(v, OP_String8, 0, regTabname, 0, pTab->zName, 0); |
- } |
+ /* Populate the register containing the index name. */ |
sqlite3VdbeAddOp4(v, OP_String8, 0, regIdxname, 0, pIdx->zName, 0); |
#ifdef SQLITE_ENABLE_STAT2 |
@@ -229,9 +239,10 @@ static void analyzeOneTable( |
sqlite3VdbeAddOp2(v, OP_AddImm, iMem, 1); |
for(i=0; i<nCol; i++){ |
+ CollSeq *pColl; |
sqlite3VdbeAddOp3(v, OP_Column, iIdxCur, i, regCol); |
-#ifdef SQLITE_ENABLE_STAT2 |
if( i==0 ){ |
+#ifdef SQLITE_ENABLE_STAT2 |
/* Check if the record that cursor iIdxCur points to contains a |
** value that should be stored in the sqlite_stat2 table. If so, |
** store it. */ |
@@ -260,12 +271,17 @@ static void analyzeOneTable( |
sqlite3VdbeJumpHere(v, ne); |
sqlite3VdbeAddOp2(v, OP_AddImm, regRecno, 1); |
- } |
#endif |
- sqlite3VdbeAddOp3(v, OP_Ne, regCol, 0, iMem+nCol+i+1); |
- /**** TODO: add collating sequence *****/ |
- sqlite3VdbeChangeP5(v, SQLITE_JUMPIFNULL); |
+ /* Always record the very first row */ |
+ sqlite3VdbeAddOp1(v, OP_IfNot, iMem+1); |
+ } |
+ assert( pIdx->azColl!=0 ); |
+ assert( pIdx->azColl[i]!=0 ); |
+ pColl = sqlite3LocateCollSeq(pParse, pIdx->azColl[i]); |
+ sqlite3VdbeAddOp4(v, OP_Ne, regCol, 0, iMem+nCol+i+1, |
+ (char*)pColl, P4_COLLSEQ); |
+ sqlite3VdbeChangeP5(v, SQLITE_NULLEQ); |
} |
if( db->mallocFailed ){ |
/* If a malloc failure has occurred, then the result of the expression |
@@ -276,7 +292,11 @@ static void analyzeOneTable( |
} |
sqlite3VdbeAddOp2(v, OP_Goto, 0, endOfLoop); |
for(i=0; i<nCol; i++){ |
- sqlite3VdbeJumpHere(v, sqlite3VdbeCurrentAddr(v)-(nCol*2)); |
+ int addr2 = sqlite3VdbeCurrentAddr(v) - (nCol*2); |
+ if( i==0 ){ |
+ sqlite3VdbeJumpHere(v, addr2-1); /* Set jump dest for the OP_IfNot */ |
+ } |
+ sqlite3VdbeJumpHere(v, addr2); /* Set jump dest for the OP_Ne */ |
sqlite3VdbeAddOp2(v, OP_AddImm, iMem+i+1, 1); |
sqlite3VdbeAddOp3(v, OP_Column, iIdxCur, i, iMem+nCol+i+1); |
} |
@@ -304,8 +324,10 @@ static void analyzeOneTable( |
** If K>0 then it is always the case the D>0 so division by zero |
** is never possible. |
*/ |
- addr = sqlite3VdbeAddOp1(v, OP_IfNot, iMem); |
sqlite3VdbeAddOp2(v, OP_SCopy, iMem, regSampleno); |
+ if( jZeroRows<0 ){ |
+ jZeroRows = sqlite3VdbeAddOp1(v, OP_IfNot, iMem); |
+ } |
for(i=0; i<nCol; i++){ |
sqlite3VdbeAddOp4(v, OP_String8, 0, regTemp, 0, " ", 0); |
sqlite3VdbeAddOp3(v, OP_Concat, regTemp, regSampleno, regSampleno); |
@@ -319,13 +341,33 @@ static void analyzeOneTable( |
sqlite3VdbeAddOp2(v, OP_NewRowid, iStatCur, regRowid); |
sqlite3VdbeAddOp3(v, OP_Insert, iStatCur, regRec, regRowid); |
sqlite3VdbeChangeP5(v, OPFLAG_APPEND); |
- sqlite3VdbeJumpHere(v, addr); |
} |
+ |
+ /* If the table has no indices, create a single sqlite_stat1 entry |
+ ** containing NULL as the index name and the row count as the content. |
+ */ |
+ if( pTab->pIndex==0 ){ |
+ sqlite3VdbeAddOp3(v, OP_OpenRead, iIdxCur, pTab->tnum, iDb); |
+ VdbeComment((v, "%s", pTab->zName)); |
+ sqlite3VdbeAddOp2(v, OP_Count, iIdxCur, regSampleno); |
+ sqlite3VdbeAddOp1(v, OP_Close, iIdxCur); |
+ jZeroRows = sqlite3VdbeAddOp1(v, OP_IfNot, regSampleno); |
+ }else{ |
+ sqlite3VdbeJumpHere(v, jZeroRows); |
+ jZeroRows = sqlite3VdbeAddOp0(v, OP_Goto); |
+ } |
+ sqlite3VdbeAddOp2(v, OP_Null, 0, regIdxname); |
+ sqlite3VdbeAddOp4(v, OP_MakeRecord, regTabname, 3, regRec, "aaa", 0); |
+ sqlite3VdbeAddOp2(v, OP_NewRowid, iStatCur, regRowid); |
+ sqlite3VdbeAddOp3(v, OP_Insert, iStatCur, regRec, regRowid); |
+ sqlite3VdbeChangeP5(v, OPFLAG_APPEND); |
+ if( pParse->nMem<regRec ) pParse->nMem = regRec; |
+ sqlite3VdbeJumpHere(v, jZeroRows); |
} |
/* |
** Generate code that will cause the most recent index analysis to |
-** be laoded into internal hash tables where is can be used. |
+** be loaded into internal hash tables where is can be used. |
*/ |
static void loadAnalysis(Parse *pParse, int iDb){ |
Vdbe *v = sqlite3GetVdbe(pParse); |
@@ -347,20 +389,22 @@ static void analyzeDatabase(Parse *pParse, int iDb){ |
sqlite3BeginWriteOperation(pParse, 0, iDb); |
iStatCur = pParse->nTab; |
pParse->nTab += 2; |
- openStatTable(pParse, iDb, iStatCur, 0); |
+ openStatTable(pParse, iDb, iStatCur, 0, 0); |
iMem = pParse->nMem+1; |
+ assert( sqlite3SchemaMutexHeld(db, iDb, 0) ); |
for(k=sqliteHashFirst(&pSchema->tblHash); k; k=sqliteHashNext(k)){ |
Table *pTab = (Table*)sqliteHashData(k); |
- analyzeOneTable(pParse, pTab, iStatCur, iMem); |
+ analyzeOneTable(pParse, pTab, 0, iStatCur, iMem); |
} |
loadAnalysis(pParse, iDb); |
} |
/* |
** Generate code that will do an analysis of a single table in |
-** a database. |
+** a database. If pOnlyIdx is not NULL then it is a single index |
+** in pTab that should be analyzed. |
*/ |
-static void analyzeTable(Parse *pParse, Table *pTab){ |
+static void analyzeTable(Parse *pParse, Table *pTab, Index *pOnlyIdx){ |
int iDb; |
int iStatCur; |
@@ -370,8 +414,12 @@ static void analyzeTable(Parse *pParse, Table *pTab){ |
sqlite3BeginWriteOperation(pParse, 0, iDb); |
iStatCur = pParse->nTab; |
pParse->nTab += 2; |
- openStatTable(pParse, iDb, iStatCur, pTab->zName); |
- analyzeOneTable(pParse, pTab, iStatCur, pParse->nMem+1); |
+ if( pOnlyIdx ){ |
+ openStatTable(pParse, iDb, iStatCur, pOnlyIdx->zName, "idx"); |
+ }else{ |
+ openStatTable(pParse, iDb, iStatCur, pTab->zName, "tbl"); |
+ } |
+ analyzeOneTable(pParse, pTab, pOnlyIdx, iStatCur, pParse->nMem+1); |
loadAnalysis(pParse, iDb); |
} |
@@ -393,6 +441,7 @@ void sqlite3Analyze(Parse *pParse, Token *pName1, Token *pName2){ |
int i; |
char *z, *zDb; |
Table *pTab; |
+ Index *pIdx; |
Token *pTableName; |
/* Read the database schema. If an error occurs, leave an error message |
@@ -417,11 +466,12 @@ void sqlite3Analyze(Parse *pParse, Token *pName1, Token *pName2){ |
}else{ |
z = sqlite3NameFromToken(db, pName1); |
if( z ){ |
- pTab = sqlite3LocateTable(pParse, 0, z, 0); |
- sqlite3DbFree(db, z); |
- if( pTab ){ |
- analyzeTable(pParse, pTab); |
+ if( (pIdx = sqlite3FindIndex(db, z, 0))!=0 ){ |
+ analyzeTable(pParse, pIdx->pTable, pIdx); |
+ }else if( (pTab = sqlite3LocateTable(pParse, 0, z, 0))!=0 ){ |
+ analyzeTable(pParse, pTab, 0); |
} |
+ sqlite3DbFree(db, z); |
} |
} |
}else{ |
@@ -431,11 +481,12 @@ void sqlite3Analyze(Parse *pParse, Token *pName1, Token *pName2){ |
zDb = db->aDb[iDb].zName; |
z = sqlite3NameFromToken(db, pTableName); |
if( z ){ |
- pTab = sqlite3LocateTable(pParse, 0, z, zDb); |
- sqlite3DbFree(db, z); |
- if( pTab ){ |
- analyzeTable(pParse, pTab); |
+ if( (pIdx = sqlite3FindIndex(db, z, zDb))!=0 ){ |
+ analyzeTable(pParse, pIdx->pTable, pIdx); |
+ }else if( (pTab = sqlite3LocateTable(pParse, 0, z, zDb))!=0 ){ |
+ analyzeTable(pParse, pTab, 0); |
} |
+ sqlite3DbFree(db, z); |
} |
} |
} |
@@ -455,35 +506,52 @@ struct analysisInfo { |
** This callback is invoked once for each index when reading the |
** sqlite_stat1 table. |
** |
-** argv[0] = name of the index |
-** argv[1] = results of analysis - on integer for each column |
+** argv[0] = name of the table |
+** argv[1] = name of the index (might be NULL) |
+** argv[2] = results of analysis - on integer for each column |
+** |
+** Entries for which argv[1]==NULL simply record the number of rows in |
+** the table. |
*/ |
static int analysisLoader(void *pData, int argc, char **argv, char **NotUsed){ |
analysisInfo *pInfo = (analysisInfo*)pData; |
Index *pIndex; |
- int i, c; |
+ Table *pTable; |
+ int i, c, n; |
unsigned int v; |
const char *z; |
- assert( argc==2 ); |
+ assert( argc==3 ); |
UNUSED_PARAMETER2(NotUsed, argc); |
- if( argv==0 || argv[0]==0 || argv[1]==0 ){ |
+ if( argv==0 || argv[0]==0 || argv[2]==0 ){ |
return 0; |
} |
- pIndex = sqlite3FindIndex(pInfo->db, argv[0], pInfo->zDatabase); |
- if( pIndex==0 ){ |
+ pTable = sqlite3FindTable(pInfo->db, argv[0], pInfo->zDatabase); |
+ if( pTable==0 ){ |
return 0; |
} |
- z = argv[1]; |
- for(i=0; *z && i<=pIndex->nColumn; i++){ |
+ if( argv[1] ){ |
+ pIndex = sqlite3FindIndex(pInfo->db, argv[1], pInfo->zDatabase); |
+ }else{ |
+ pIndex = 0; |
+ } |
+ n = pIndex ? pIndex->nColumn : 0; |
+ z = argv[2]; |
+ for(i=0; *z && i<=n; i++){ |
v = 0; |
while( (c=z[0])>='0' && c<='9' ){ |
v = v*10 + c - '0'; |
z++; |
} |
+ if( i==0 ) pTable->nRowEst = v; |
+ if( pIndex==0 ) break; |
pIndex->aiRowEst[i] = v; |
if( *z==' ' ) z++; |
+ if( memcmp(z, "unordered", 10)==0 ){ |
+ pIndex->bUnordered = 1; |
+ break; |
+ } |
} |
return 0; |
} |
@@ -492,21 +560,20 @@ static int analysisLoader(void *pData, int argc, char **argv, char **NotUsed){ |
** If the Index.aSample variable is not NULL, delete the aSample[] array |
** and its contents. |
*/ |
-void sqlite3DeleteIndexSamples(Index *pIdx){ |
+void sqlite3DeleteIndexSamples(sqlite3 *db, Index *pIdx){ |
#ifdef SQLITE_ENABLE_STAT2 |
if( pIdx->aSample ){ |
int j; |
- sqlite3 *dbMem = pIdx->pTable->dbMem; |
for(j=0; j<SQLITE_INDEX_SAMPLES; j++){ |
IndexSample *p = &pIdx->aSample[j]; |
if( p->eType==SQLITE_TEXT || p->eType==SQLITE_BLOB ){ |
- sqlite3DbFree(pIdx->pTable->dbMem, p->u.z); |
+ sqlite3DbFree(db, p->u.z); |
} |
} |
- sqlite3DbFree(dbMem, pIdx->aSample); |
- pIdx->aSample = 0; |
+ sqlite3DbFree(db, pIdx->aSample); |
} |
#else |
+ UNUSED_PARAMETER(db); |
UNUSED_PARAMETER(pIdx); |
#endif |
} |
@@ -539,13 +606,14 @@ int sqlite3AnalysisLoad(sqlite3 *db, int iDb){ |
assert( iDb>=0 && iDb<db->nDb ); |
assert( db->aDb[iDb].pBt!=0 ); |
- assert( sqlite3BtreeHoldsMutex(db->aDb[iDb].pBt) ); |
/* Clear any prior statistics */ |
+ assert( sqlite3SchemaMutexHeld(db, iDb, 0) ); |
for(i=sqliteHashFirst(&db->aDb[iDb].pSchema->idxHash);i;i=sqliteHashNext(i)){ |
Index *pIdx = sqliteHashData(i); |
sqlite3DefaultRowEst(pIdx); |
- sqlite3DeleteIndexSamples(pIdx); |
+ sqlite3DeleteIndexSamples(db, pIdx); |
+ pIdx->aSample = 0; |
} |
/* Check to make sure the sqlite_stat1 table exists */ |
@@ -557,13 +625,11 @@ int sqlite3AnalysisLoad(sqlite3 *db, int iDb){ |
/* Load new statistics out of the sqlite_stat1 table */ |
zSql = sqlite3MPrintf(db, |
- "SELECT idx, stat FROM %Q.sqlite_stat1", sInfo.zDatabase); |
+ "SELECT tbl, idx, stat FROM %Q.sqlite_stat1", sInfo.zDatabase); |
if( zSql==0 ){ |
rc = SQLITE_NOMEM; |
}else{ |
- (void)sqlite3SafetyOff(db); |
rc = sqlite3_exec(db, zSql, analysisLoader, &sInfo, 0); |
- (void)sqlite3SafetyOn(db); |
sqlite3DbFree(db, zSql); |
} |
@@ -581,31 +647,30 @@ int sqlite3AnalysisLoad(sqlite3 *db, int iDb){ |
if( !zSql ){ |
rc = SQLITE_NOMEM; |
}else{ |
- (void)sqlite3SafetyOff(db); |
rc = sqlite3_prepare(db, zSql, -1, &pStmt, 0); |
- (void)sqlite3SafetyOn(db); |
sqlite3DbFree(db, zSql); |
} |
if( rc==SQLITE_OK ){ |
- (void)sqlite3SafetyOff(db); |
while( sqlite3_step(pStmt)==SQLITE_ROW ){ |
- char *zIndex = (char *)sqlite3_column_text(pStmt, 0); |
- Index *pIdx = sqlite3FindIndex(db, zIndex, sInfo.zDatabase); |
+ char *zIndex; /* Index name */ |
+ Index *pIdx; /* Pointer to the index object */ |
+ |
+ zIndex = (char *)sqlite3_column_text(pStmt, 0); |
+ pIdx = zIndex ? sqlite3FindIndex(db, zIndex, sInfo.zDatabase) : 0; |
if( pIdx ){ |
int iSample = sqlite3_column_int(pStmt, 1); |
- sqlite3 *dbMem = pIdx->pTable->dbMem; |
- assert( dbMem==db || dbMem==0 ); |
if( iSample<SQLITE_INDEX_SAMPLES && iSample>=0 ){ |
int eType = sqlite3_column_type(pStmt, 2); |
if( pIdx->aSample==0 ){ |
static const int sz = sizeof(IndexSample)*SQLITE_INDEX_SAMPLES; |
- pIdx->aSample = (IndexSample *)sqlite3DbMallocZero(dbMem, sz); |
+ pIdx->aSample = (IndexSample *)sqlite3DbMallocRaw(0, sz); |
if( pIdx->aSample==0 ){ |
db->mallocFailed = 1; |
break; |
} |
+ memset(pIdx->aSample, 0, sz); |
} |
assert( pIdx->aSample ); |
@@ -625,12 +690,14 @@ int sqlite3AnalysisLoad(sqlite3 *db, int iDb){ |
n = 24; |
} |
pSample->nByte = (u8)n; |
- pSample->u.z = sqlite3DbMallocRaw(dbMem, n); |
- if( pSample->u.z ){ |
- memcpy(pSample->u.z, z, n); |
+ if( n < 1){ |
+ pSample->u.z = 0; |
}else{ |
- db->mallocFailed = 1; |
- break; |
+ pSample->u.z = sqlite3DbStrNDup(0, z, n); |
+ if( pSample->u.z==0 ){ |
+ db->mallocFailed = 1; |
+ break; |
+ } |
} |
} |
} |
@@ -638,7 +705,6 @@ int sqlite3AnalysisLoad(sqlite3 *db, int iDb){ |
} |
} |
rc = sqlite3_finalize(pStmt); |
- (void)sqlite3SafetyOn(db); |
} |
} |
#endif |