| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
| #include "fts3Int.h" |
| #if !defined(SQLITE_CORE) || defined(SQLITE_ENABLE_FTS3) |
|
|
| #include <string.h> |
| #include <assert.h> |
| #include <stdlib.h> |
| #include <stdio.h> |
|
|
| #define FTS_MAX_APPENDABLE_HEIGHT 16 |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| #define FTS3_NODE_PADDING (FTS3_VARINT_MAX*2) |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| #ifdef SQLITE_TEST |
| int test_fts3_node_chunksize = (4*1024); |
| int test_fts3_node_chunk_threshold = (4*1024)*4; |
| # define FTS3_NODE_CHUNKSIZE test_fts3_node_chunksize |
| # define FTS3_NODE_CHUNK_THRESHOLD test_fts3_node_chunk_threshold |
| #else |
| # define FTS3_NODE_CHUNKSIZE (4*1024) |
| # define FTS3_NODE_CHUNK_THRESHOLD (FTS3_NODE_CHUNKSIZE*4) |
| #endif |
|
|
| |
| |
| |
| |
| #define FTS_STAT_DOCTOTAL 0 |
| #define FTS_STAT_INCRMERGEHINT 1 |
| #define FTS_STAT_AUTOINCRMERGE 2 |
|
|
| |
| |
| |
| |
| |
| |
| #ifdef FTS3_LOG_MERGES |
| static void fts3LogMerge(int nMerge, sqlite3_int64 iAbsLevel){ |
| sqlite3_log(SQLITE_OK, "%d-way merge from level %d", nMerge, (int)iAbsLevel); |
| } |
| #else |
| #define fts3LogMerge(x, y) |
| #endif |
|
|
|
|
| typedef struct PendingList PendingList; |
| typedef struct SegmentNode SegmentNode; |
| typedef struct SegmentWriter SegmentWriter; |
|
|
| |
| |
| |
| |
| struct PendingList { |
| sqlite3_int64 nData; |
| char *aData; |
| sqlite3_int64 nSpace; |
| sqlite3_int64 iLastDocid; |
| sqlite3_int64 iLastCol; |
| sqlite3_int64 iLastPos; |
| }; |
|
|
|
|
| |
| |
| |
| struct Fts3DeferredToken { |
| Fts3PhraseToken *pToken; |
| int iCol; |
| Fts3DeferredToken *pNext; |
| PendingList *pList; |
| }; |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| struct Fts3SegReader { |
| int iIdx; |
| u8 bLookup; |
| u8 rootOnly; |
|
|
| sqlite3_int64 iStartBlock; |
| sqlite3_int64 iLeafEndBlock; |
| sqlite3_int64 iEndBlock; |
| sqlite3_int64 iCurrentBlock; |
|
|
| char *aNode; |
| int nNode; |
| int nPopulate; |
| sqlite3_blob *pBlob; |
|
|
| Fts3HashElem **ppNextElem; |
|
|
| |
| |
| |
| |
| |
| int nTerm; |
| char *zTerm; |
| int nTermAlloc; |
| char *aDoclist; |
| int nDoclist; |
|
|
| |
| |
| |
| char *pOffsetList; |
| int nOffsetList; |
| sqlite3_int64 iDocid; |
| }; |
|
|
| #define fts3SegReaderIsPending(p) ((p)->ppNextElem!=0) |
| #define fts3SegReaderIsRootOnly(p) ((p)->rootOnly!=0) |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| struct SegmentWriter { |
| SegmentNode *pTree; |
| sqlite3_int64 iFirst; |
| sqlite3_int64 iFree; |
| char *zTerm; |
| int nTerm; |
| int nMalloc; |
| char *zMalloc; |
| int nSize; |
| int nData; |
| char *aData; |
| i64 nLeafData; |
| }; |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| struct SegmentNode { |
| SegmentNode *pParent; |
| SegmentNode *pRight; |
| SegmentNode *pLeftmost; |
| int nEntry; |
| char *zTerm; |
| int nTerm; |
| int nMalloc; |
| char *zMalloc; |
| int nData; |
| char *aData; |
| }; |
|
|
| |
| |
| |
| #define SQL_DELETE_CONTENT 0 |
| #define SQL_IS_EMPTY 1 |
| #define SQL_DELETE_ALL_CONTENT 2 |
| #define SQL_DELETE_ALL_SEGMENTS 3 |
| #define SQL_DELETE_ALL_SEGDIR 4 |
| #define SQL_DELETE_ALL_DOCSIZE 5 |
| #define SQL_DELETE_ALL_STAT 6 |
| #define SQL_SELECT_CONTENT_BY_ROWID 7 |
| #define SQL_NEXT_SEGMENT_INDEX 8 |
| #define SQL_INSERT_SEGMENTS 9 |
| #define SQL_NEXT_SEGMENTS_ID 10 |
| #define SQL_INSERT_SEGDIR 11 |
| #define SQL_SELECT_LEVEL 12 |
| #define SQL_SELECT_LEVEL_RANGE 13 |
| #define SQL_SELECT_LEVEL_COUNT 14 |
| #define SQL_SELECT_SEGDIR_MAX_LEVEL 15 |
| #define SQL_DELETE_SEGDIR_LEVEL 16 |
| #define SQL_DELETE_SEGMENTS_RANGE 17 |
| #define SQL_CONTENT_INSERT 18 |
| #define SQL_DELETE_DOCSIZE 19 |
| #define SQL_REPLACE_DOCSIZE 20 |
| #define SQL_SELECT_DOCSIZE 21 |
| #define SQL_SELECT_STAT 22 |
| #define SQL_REPLACE_STAT 23 |
|
|
| #define SQL_SELECT_ALL_PREFIX_LEVEL 24 |
| #define SQL_DELETE_ALL_TERMS_SEGDIR 25 |
| #define SQL_DELETE_SEGDIR_RANGE 26 |
| #define SQL_SELECT_ALL_LANGID 27 |
| #define SQL_FIND_MERGE_LEVEL 28 |
| #define SQL_MAX_LEAF_NODE_ESTIMATE 29 |
| #define SQL_DELETE_SEGDIR_ENTRY 30 |
| #define SQL_SHIFT_SEGDIR_ENTRY 31 |
| #define SQL_SELECT_SEGDIR 32 |
| #define SQL_CHOMP_SEGDIR 33 |
| #define SQL_SEGMENT_IS_APPENDABLE 34 |
| #define SQL_SELECT_INDEXES 35 |
| #define SQL_SELECT_MXLEVEL 36 |
|
|
| #define SQL_SELECT_LEVEL_RANGE2 37 |
| #define SQL_UPDATE_LEVEL_IDX 38 |
| #define SQL_UPDATE_LEVEL 39 |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3SqlStmt( |
| Fts3Table *p, |
| int eStmt, |
| sqlite3_stmt **pp, |
| sqlite3_value **apVal |
| ){ |
| const char *azSql[] = { |
| "DELETE FROM %Q.'%q_content' WHERE rowid = ?", |
| "SELECT NOT EXISTS(SELECT docid FROM %Q.'%q_content' WHERE rowid!=?)", |
| "DELETE FROM %Q.'%q_content'", |
| "DELETE FROM %Q.'%q_segments'", |
| "DELETE FROM %Q.'%q_segdir'", |
| "DELETE FROM %Q.'%q_docsize'", |
| "DELETE FROM %Q.'%q_stat'", |
| "SELECT %s WHERE rowid=?", |
| "SELECT (SELECT max(idx) FROM %Q.'%q_segdir' WHERE level = ?) + 1", |
| "REPLACE INTO %Q.'%q_segments'(blockid, block) VALUES(?, ?)", |
| "SELECT coalesce((SELECT max(blockid) FROM %Q.'%q_segments') + 1, 1)", |
| "REPLACE INTO %Q.'%q_segdir' VALUES(?,?,?,?,?,?)", |
|
|
| |
| "SELECT idx, start_block, leaves_end_block, end_block, root " |
| "FROM %Q.'%q_segdir' WHERE level = ? ORDER BY idx ASC", |
| "SELECT idx, start_block, leaves_end_block, end_block, root " |
| "FROM %Q.'%q_segdir' WHERE level BETWEEN ? AND ?" |
| "ORDER BY level DESC, idx ASC", |
|
|
| "SELECT count(*) FROM %Q.'%q_segdir' WHERE level = ?", |
| "SELECT max(level) FROM %Q.'%q_segdir' WHERE level BETWEEN ? AND ?", |
|
|
| "DELETE FROM %Q.'%q_segdir' WHERE level = ?", |
| "DELETE FROM %Q.'%q_segments' WHERE blockid BETWEEN ? AND ?", |
| "INSERT INTO %Q.'%q_content' VALUES(%s)", |
| "DELETE FROM %Q.'%q_docsize' WHERE docid = ?", |
| "REPLACE INTO %Q.'%q_docsize' VALUES(?,?)", |
| "SELECT size FROM %Q.'%q_docsize' WHERE docid=?", |
| "SELECT value FROM %Q.'%q_stat' WHERE id=?", |
| "REPLACE INTO %Q.'%q_stat' VALUES(?,?)", |
| "", |
| "", |
|
|
| "DELETE FROM %Q.'%q_segdir' WHERE level BETWEEN ? AND ?", |
| "SELECT ? UNION SELECT level / (1024 * ?) FROM %Q.'%q_segdir'", |
|
|
| |
| |
| |
| |
| |
| "SELECT level, count(*) AS cnt FROM %Q.'%q_segdir' " |
| " GROUP BY level HAVING cnt>=?" |
| " ORDER BY (level %% 1024) ASC, 2 DESC LIMIT 1", |
|
|
| |
| |
| |
| "SELECT 2 * total(1 + leaves_end_block - start_block) " |
| " FROM (SELECT * FROM %Q.'%q_segdir' " |
| " WHERE level = ? ORDER BY idx ASC LIMIT ?" |
| " )", |
|
|
| |
| |
| "DELETE FROM %Q.'%q_segdir' WHERE level = ? AND idx = ?", |
|
|
| |
| |
| |
| "UPDATE %Q.'%q_segdir' SET idx = ? WHERE level=? AND idx=?", |
|
|
| |
| |
| |
| "SELECT idx, start_block, leaves_end_block, end_block, root " |
| "FROM %Q.'%q_segdir' WHERE level = ? AND idx = ?", |
|
|
| |
| |
| |
| "UPDATE %Q.'%q_segdir' SET start_block = ?, root = ?" |
| "WHERE level = ? AND idx = ?", |
|
|
| |
| |
| |
| "SELECT 1 FROM %Q.'%q_segments' WHERE blockid=? AND block IS NULL", |
|
|
| |
| |
| "SELECT idx FROM %Q.'%q_segdir' WHERE level=? ORDER BY 1 ASC", |
|
|
| |
| |
| "SELECT max( level %% 1024 ) FROM %Q.'%q_segdir'", |
|
|
| |
| "SELECT level, idx, end_block " |
| "FROM %Q.'%q_segdir' WHERE level BETWEEN ? AND ? " |
| "ORDER BY level DESC, idx ASC", |
|
|
| |
| "UPDATE OR FAIL %Q.'%q_segdir' SET level=-1,idx=? " |
| "WHERE level=? AND idx=?", |
| "UPDATE OR FAIL %Q.'%q_segdir' SET level=? WHERE level=-1" |
|
|
| }; |
| int rc = SQLITE_OK; |
| sqlite3_stmt *pStmt; |
|
|
| assert( SizeofArray(azSql)==SizeofArray(p->aStmt) ); |
| assert( eStmt<SizeofArray(azSql) && eStmt>=0 ); |
| |
| pStmt = p->aStmt[eStmt]; |
| if( !pStmt ){ |
| int f = SQLITE_PREPARE_PERSISTENT|SQLITE_PREPARE_NO_VTAB; |
| char *zSql; |
| if( eStmt==SQL_CONTENT_INSERT ){ |
| zSql = sqlite3_mprintf(azSql[eStmt], p->zDb, p->zName, p->zWriteExprlist); |
| }else if( eStmt==SQL_SELECT_CONTENT_BY_ROWID ){ |
| f &= ~SQLITE_PREPARE_NO_VTAB; |
| zSql = sqlite3_mprintf(azSql[eStmt], p->zReadExprlist); |
| }else{ |
| zSql = sqlite3_mprintf(azSql[eStmt], p->zDb, p->zName); |
| } |
| if( !zSql ){ |
| rc = SQLITE_NOMEM; |
| }else{ |
| rc = sqlite3_prepare_v3(p->db, zSql, -1, f, &pStmt, NULL); |
| sqlite3_free(zSql); |
| assert( rc==SQLITE_OK || pStmt==0 ); |
| p->aStmt[eStmt] = pStmt; |
| } |
| } |
| if( apVal ){ |
| int i; |
| int nParam = sqlite3_bind_parameter_count(pStmt); |
| for(i=0; rc==SQLITE_OK && i<nParam; i++){ |
| rc = sqlite3_bind_value(pStmt, i+1, apVal[i]); |
| } |
| } |
| *pp = pStmt; |
| return rc; |
| } |
|
|
|
|
| static int fts3SelectDocsize( |
| Fts3Table *pTab, |
| sqlite3_int64 iDocid, |
| sqlite3_stmt **ppStmt |
| ){ |
| sqlite3_stmt *pStmt = 0; |
| int rc; |
|
|
| rc = fts3SqlStmt(pTab, SQL_SELECT_DOCSIZE, &pStmt, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pStmt, 1, iDocid); |
| rc = sqlite3_step(pStmt); |
| if( rc!=SQLITE_ROW || sqlite3_column_type(pStmt, 0)!=SQLITE_BLOB ){ |
| rc = sqlite3_reset(pStmt); |
| if( rc==SQLITE_OK ) rc = FTS_CORRUPT_VTAB; |
| pStmt = 0; |
| }else{ |
| rc = SQLITE_OK; |
| } |
| } |
|
|
| *ppStmt = pStmt; |
| return rc; |
| } |
|
|
| int sqlite3Fts3SelectDoctotal( |
| Fts3Table *pTab, |
| sqlite3_stmt **ppStmt |
| ){ |
| sqlite3_stmt *pStmt = 0; |
| int rc; |
| rc = fts3SqlStmt(pTab, SQL_SELECT_STAT, &pStmt, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int(pStmt, 1, FTS_STAT_DOCTOTAL); |
| if( sqlite3_step(pStmt)!=SQLITE_ROW |
| || sqlite3_column_type(pStmt, 0)!=SQLITE_BLOB |
| ){ |
| rc = sqlite3_reset(pStmt); |
| if( rc==SQLITE_OK ) rc = FTS_CORRUPT_VTAB; |
| pStmt = 0; |
| } |
| } |
| *ppStmt = pStmt; |
| return rc; |
| } |
|
|
| int sqlite3Fts3SelectDocsize( |
| Fts3Table *pTab, |
| sqlite3_int64 iDocid, |
| sqlite3_stmt **ppStmt |
| ){ |
| return fts3SelectDocsize(pTab, iDocid, ppStmt); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static void fts3SqlExec( |
| int *pRC, |
| Fts3Table *p, |
| int eStmt, |
| sqlite3_value **apVal |
| ){ |
| sqlite3_stmt *pStmt; |
| int rc; |
| if( *pRC ) return; |
| rc = fts3SqlStmt(p, eStmt, &pStmt, apVal); |
| if( rc==SQLITE_OK ){ |
| sqlite3_step(pStmt); |
| rc = sqlite3_reset(pStmt); |
| } |
| *pRC = rc; |
| } |
|
|
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3Writelock(Fts3Table *p){ |
| int rc = SQLITE_OK; |
| |
| if( p->nPendingData==0 ){ |
| sqlite3_stmt *pStmt; |
| rc = fts3SqlStmt(p, SQL_DELETE_SEGDIR_LEVEL, &pStmt, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_null(pStmt, 1); |
| sqlite3_step(pStmt); |
| rc = sqlite3_reset(pStmt); |
| } |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static sqlite3_int64 getAbsoluteLevel( |
| Fts3Table *p, |
| int iLangid, |
| int iIndex, |
| int iLevel |
| ){ |
| sqlite3_int64 iBase; |
| assert_fts3_nc( iLangid>=0 ); |
| assert( p->nIndex>0 ); |
| assert( iIndex>=0 && iIndex<p->nIndex ); |
|
|
| iBase = ((sqlite3_int64)iLangid * p->nIndex + iIndex) * FTS3_SEGDIR_MAXLEVEL; |
| return iBase + iLevel; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| int sqlite3Fts3AllSegdirs( |
| Fts3Table *p, |
| int iLangid, |
| int iIndex, |
| int iLevel, |
| sqlite3_stmt **ppStmt |
| ){ |
| int rc; |
| sqlite3_stmt *pStmt = 0; |
|
|
| assert( iLevel==FTS3_SEGCURSOR_ALL || iLevel>=0 ); |
| assert( iLevel<FTS3_SEGDIR_MAXLEVEL ); |
| assert( iIndex>=0 && iIndex<p->nIndex ); |
|
|
| if( iLevel<0 ){ |
| |
| rc = fts3SqlStmt(p, SQL_SELECT_LEVEL_RANGE, &pStmt, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pStmt, 1, getAbsoluteLevel(p, iLangid, iIndex, 0)); |
| sqlite3_bind_int64(pStmt, 2, |
| getAbsoluteLevel(p, iLangid, iIndex, FTS3_SEGDIR_MAXLEVEL-1) |
| ); |
| } |
| }else{ |
| |
| rc = fts3SqlStmt(p, SQL_SELECT_LEVEL, &pStmt, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pStmt, 1, getAbsoluteLevel(p, iLangid, iIndex,iLevel)); |
| } |
| } |
| *ppStmt = pStmt; |
| return rc; |
| } |
|
|
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3PendingListAppendVarint( |
| PendingList **pp, |
| sqlite3_int64 i |
| ){ |
| PendingList *p = *pp; |
|
|
| |
| if( !p ){ |
| p = sqlite3_malloc64(sizeof(*p) + 100); |
| if( !p ){ |
| return SQLITE_NOMEM; |
| } |
| p->nSpace = 100; |
| p->aData = (char *)&p[1]; |
| p->nData = 0; |
| } |
| else if( p->nData+FTS3_VARINT_MAX+1>p->nSpace ){ |
| i64 nNew = p->nSpace * 2; |
| p = sqlite3_realloc64(p, sizeof(*p) + nNew); |
| if( !p ){ |
| sqlite3_free(*pp); |
| *pp = 0; |
| return SQLITE_NOMEM; |
| } |
| p->nSpace = (int)nNew; |
| p->aData = (char *)&p[1]; |
| } |
|
|
| |
| p->nData += sqlite3Fts3PutVarint(&p->aData[p->nData], i); |
| p->aData[p->nData] = '\0'; |
| *pp = p; |
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3PendingListAppend( |
| PendingList **pp, |
| sqlite3_int64 iDocid, |
| sqlite3_int64 iCol, |
| sqlite3_int64 iPos, |
| int *pRc |
| ){ |
| PendingList *p = *pp; |
| int rc = SQLITE_OK; |
|
|
| assert( !p || p->iLastDocid<=iDocid ); |
|
|
| if( !p || p->iLastDocid!=iDocid ){ |
| u64 iDelta = (u64)iDocid - (u64)(p ? p->iLastDocid : 0); |
| if( p ){ |
| assert( p->nData<p->nSpace ); |
| assert( p->aData[p->nData]==0 ); |
| p->nData++; |
| } |
| if( SQLITE_OK!=(rc = fts3PendingListAppendVarint(&p, iDelta)) ){ |
| goto pendinglistappend_out; |
| } |
| p->iLastCol = -1; |
| p->iLastPos = 0; |
| p->iLastDocid = iDocid; |
| } |
| if( iCol>0 && p->iLastCol!=iCol ){ |
| if( SQLITE_OK!=(rc = fts3PendingListAppendVarint(&p, 1)) |
| || SQLITE_OK!=(rc = fts3PendingListAppendVarint(&p, iCol)) |
| ){ |
| goto pendinglistappend_out; |
| } |
| p->iLastCol = iCol; |
| p->iLastPos = 0; |
| } |
| if( iCol>=0 ){ |
| assert( iPos>p->iLastPos || (iPos==0 && p->iLastPos==0) ); |
| rc = fts3PendingListAppendVarint(&p, 2+iPos-p->iLastPos); |
| if( rc==SQLITE_OK ){ |
| p->iLastPos = iPos; |
| } |
| } |
|
|
| pendinglistappend_out: |
| *pRc = rc; |
| if( p!=*pp ){ |
| *pp = p; |
| return 1; |
| } |
| return 0; |
| } |
|
|
| |
| |
| |
| static void fts3PendingListDelete(PendingList *pList){ |
| sqlite3_free(pList); |
| } |
|
|
| |
| |
| |
| static int fts3PendingTermsAddOne( |
| Fts3Table *p, |
| int iCol, |
| int iPos, |
| Fts3Hash *pHash, |
| const char *zToken, |
| int nToken |
| ){ |
| PendingList *pList; |
| int rc = SQLITE_OK; |
|
|
| pList = (PendingList *)fts3HashFind(pHash, zToken, nToken); |
| if( pList ){ |
| assert( (i64)pList->nData+(i64)nToken+(i64)sizeof(Fts3HashElem) |
| <= (i64)p->nPendingData ); |
| p->nPendingData -= (int)(pList->nData + nToken + sizeof(Fts3HashElem)); |
| } |
| if( fts3PendingListAppend(&pList, p->iPrevDocid, iCol, iPos, &rc) ){ |
| if( pList==fts3HashInsert(pHash, zToken, nToken, pList) ){ |
| |
| |
| |
| assert( 0==fts3HashFind(pHash, zToken, nToken) ); |
| sqlite3_free(pList); |
| rc = SQLITE_NOMEM; |
| } |
| } |
| if( rc==SQLITE_OK ){ |
| assert( (i64)p->nPendingData + pList->nData + nToken |
| + sizeof(Fts3HashElem) <= 0x3fffffff ); |
| p->nPendingData += (int)(pList->nData + nToken + sizeof(Fts3HashElem)); |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| static int fts3PendingTermsAdd( |
| Fts3Table *p, |
| int iLangid, |
| const char *zText, |
| int iCol, |
| u32 *pnWord |
| ){ |
| int rc; |
| int iStart = 0; |
| int iEnd = 0; |
| int iPos = 0; |
| int nWord = 0; |
|
|
| char const *zToken; |
| int nToken = 0; |
|
|
| sqlite3_tokenizer *pTokenizer = p->pTokenizer; |
| sqlite3_tokenizer_module const *pModule = pTokenizer->pModule; |
| sqlite3_tokenizer_cursor *pCsr; |
| int (*xNext)(sqlite3_tokenizer_cursor *pCursor, |
| const char**,int*,int*,int*,int*); |
|
|
| assert( pTokenizer && pModule ); |
|
|
| |
| |
| |
| if( zText==0 ){ |
| *pnWord = 0; |
| return SQLITE_OK; |
| } |
|
|
| rc = sqlite3Fts3OpenTokenizer(pTokenizer, iLangid, zText, -1, &pCsr); |
| if( rc!=SQLITE_OK ){ |
| return rc; |
| } |
|
|
| xNext = pModule->xNext; |
| while( SQLITE_OK==rc |
| && SQLITE_OK==(rc = xNext(pCsr, &zToken, &nToken, &iStart, &iEnd, &iPos)) |
| ){ |
| int i; |
| if( iPos>=nWord ) nWord = iPos+1; |
|
|
| |
| |
| |
| if( iPos<0 || !zToken || nToken<=0 ){ |
| rc = SQLITE_ERROR; |
| break; |
| } |
|
|
| |
| rc = fts3PendingTermsAddOne( |
| p, iCol, iPos, &p->aIndex[0].hPending, zToken, nToken |
| ); |
| |
| |
| |
| for(i=1; rc==SQLITE_OK && i<p->nIndex; i++){ |
| struct Fts3Index *pIndex = &p->aIndex[i]; |
| if( nToken<pIndex->nPrefix ) continue; |
| rc = fts3PendingTermsAddOne( |
| p, iCol, iPos, &pIndex->hPending, zToken, pIndex->nPrefix |
| ); |
| } |
| } |
|
|
| pModule->xClose(pCsr); |
| *pnWord += nWord; |
| return (rc==SQLITE_DONE ? SQLITE_OK : rc); |
| } |
|
|
| |
| |
| |
| |
| |
| static int fts3PendingTermsDocid( |
| Fts3Table *p, |
| int bDelete, |
| int iLangid, |
| sqlite_int64 iDocid |
| ){ |
| assert( iLangid>=0 ); |
| assert( bDelete==1 || bDelete==0 ); |
|
|
| |
| |
| |
| |
| |
| |
| if( iDocid<p->iPrevDocid |
| || (iDocid==p->iPrevDocid && p->bPrevDelete==0) |
| || p->iPrevLangid!=iLangid |
| || p->nPendingData>p->nMaxPendingData |
| ){ |
| int rc = sqlite3Fts3PendingTermsFlush(p); |
| if( rc!=SQLITE_OK ) return rc; |
| } |
| p->iPrevDocid = iDocid; |
| p->iPrevLangid = iLangid; |
| p->bPrevDelete = bDelete; |
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| void sqlite3Fts3PendingTermsClear(Fts3Table *p){ |
| int i; |
| for(i=0; i<p->nIndex; i++){ |
| Fts3HashElem *pElem; |
| Fts3Hash *pHash = &p->aIndex[i].hPending; |
| for(pElem=fts3HashFirst(pHash); pElem; pElem=fts3HashNext(pElem)){ |
| PendingList *pList = (PendingList *)fts3HashData(pElem); |
| fts3PendingListDelete(pList); |
| } |
| fts3HashClear(pHash); |
| } |
| p->nPendingData = 0; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3InsertTerms( |
| Fts3Table *p, |
| int iLangid, |
| sqlite3_value **apVal, |
| u32 *aSz |
| ){ |
| int i; |
| for(i=2; i<p->nColumn+2; i++){ |
| int iCol = i-2; |
| if( p->abNotindexed[iCol]==0 ){ |
| const char *zText = (const char *)sqlite3_value_text(apVal[i]); |
| int rc = fts3PendingTermsAdd(p, iLangid, zText, iCol, &aSz[iCol]); |
| if( rc!=SQLITE_OK ){ |
| return rc; |
| } |
| aSz[p->nColumn] += sqlite3_value_bytes(apVal[i]); |
| } |
| } |
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3InsertData( |
| Fts3Table *p, |
| sqlite3_value **apVal, |
| sqlite3_int64 *piDocid |
| ){ |
| int rc; |
| sqlite3_stmt *pContentInsert; |
|
|
| if( p->zContentTbl ){ |
| sqlite3_value *pRowid = apVal[p->nColumn+3]; |
| if( sqlite3_value_type(pRowid)==SQLITE_NULL ){ |
| pRowid = apVal[1]; |
| } |
| if( sqlite3_value_type(pRowid)!=SQLITE_INTEGER ){ |
| return SQLITE_CONSTRAINT; |
| } |
| *piDocid = sqlite3_value_int64(pRowid); |
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| rc = fts3SqlStmt(p, SQL_CONTENT_INSERT, &pContentInsert, &apVal[1]); |
| if( rc==SQLITE_OK && p->zLanguageid ){ |
| rc = sqlite3_bind_int( |
| pContentInsert, p->nColumn+2, |
| sqlite3_value_int(apVal[p->nColumn+4]) |
| ); |
| } |
| if( rc!=SQLITE_OK ) return rc; |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| if( SQLITE_NULL!=sqlite3_value_type(apVal[3+p->nColumn]) ){ |
| if( SQLITE_NULL==sqlite3_value_type(apVal[0]) |
| && SQLITE_NULL!=sqlite3_value_type(apVal[1]) |
| ){ |
| |
| return SQLITE_ERROR; |
| } |
| rc = sqlite3_bind_value(pContentInsert, 1, apVal[3+p->nColumn]); |
| if( rc!=SQLITE_OK ) return rc; |
| } |
|
|
| |
| |
| |
| sqlite3_step(pContentInsert); |
| rc = sqlite3_reset(pContentInsert); |
|
|
| *piDocid = sqlite3_last_insert_rowid(p->db); |
| return rc; |
| } |
|
|
|
|
|
|
| |
| |
| |
| |
| static int fts3DeleteAll(Fts3Table *p, int bContent){ |
| int rc = SQLITE_OK; |
|
|
| |
| sqlite3Fts3PendingTermsClear(p); |
|
|
| |
| |
| assert( p->zContentTbl==0 || bContent==0 ); |
| if( bContent ) fts3SqlExec(&rc, p, SQL_DELETE_ALL_CONTENT, 0); |
| fts3SqlExec(&rc, p, SQL_DELETE_ALL_SEGMENTS, 0); |
| fts3SqlExec(&rc, p, SQL_DELETE_ALL_SEGDIR, 0); |
| if( p->bHasDocsize ){ |
| fts3SqlExec(&rc, p, SQL_DELETE_ALL_DOCSIZE, 0); |
| } |
| if( p->bHasStat ){ |
| fts3SqlExec(&rc, p, SQL_DELETE_ALL_STAT, 0); |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| static int langidFromSelect(Fts3Table *p, sqlite3_stmt *pSelect){ |
| int iLangid = 0; |
| if( p->zLanguageid ) iLangid = sqlite3_column_int(pSelect, p->nColumn+1); |
| return iLangid; |
| } |
|
|
| |
| |
| |
| |
| |
| static void fts3DeleteTerms( |
| int *pRC, |
| Fts3Table *p, |
| sqlite3_value *pRowid, |
| u32 *aSz, |
| int *pbFound |
| ){ |
| int rc; |
| sqlite3_stmt *pSelect; |
|
|
| assert( *pbFound==0 ); |
| if( *pRC ) return; |
| rc = fts3SqlStmt(p, SQL_SELECT_CONTENT_BY_ROWID, &pSelect, &pRowid); |
| if( rc==SQLITE_OK ){ |
| if( SQLITE_ROW==sqlite3_step(pSelect) ){ |
| int i; |
| int iLangid = langidFromSelect(p, pSelect); |
| i64 iDocid = sqlite3_column_int64(pSelect, 0); |
| rc = fts3PendingTermsDocid(p, 1, iLangid, iDocid); |
| for(i=1; rc==SQLITE_OK && i<=p->nColumn; i++){ |
| int iCol = i-1; |
| if( p->abNotindexed[iCol]==0 ){ |
| const char *zText = (const char *)sqlite3_column_text(pSelect, i); |
| rc = fts3PendingTermsAdd(p, iLangid, zText, -1, &aSz[iCol]); |
| aSz[p->nColumn] += sqlite3_column_bytes(pSelect, i); |
| } |
| } |
| if( rc!=SQLITE_OK ){ |
| sqlite3_reset(pSelect); |
| *pRC = rc; |
| return; |
| } |
| *pbFound = 1; |
| } |
| rc = sqlite3_reset(pSelect); |
| }else{ |
| sqlite3_reset(pSelect); |
| } |
| *pRC = rc; |
| } |
|
|
| |
| |
| |
| |
| static int fts3SegmentMerge(Fts3Table *, int, int, int); |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3AllocateSegdirIdx( |
| Fts3Table *p, |
| int iLangid, |
| int iIndex, |
| int iLevel, |
| int *piIdx |
| ){ |
| int rc; |
| sqlite3_stmt *pNextIdx; |
| int iNext = 0; |
|
|
| assert( iLangid>=0 ); |
| assert( p->nIndex>=1 ); |
|
|
| |
| rc = fts3SqlStmt(p, SQL_NEXT_SEGMENT_INDEX, &pNextIdx, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64( |
| pNextIdx, 1, getAbsoluteLevel(p, iLangid, iIndex, iLevel) |
| ); |
| if( SQLITE_ROW==sqlite3_step(pNextIdx) ){ |
| iNext = sqlite3_column_int(pNextIdx, 0); |
| } |
| rc = sqlite3_reset(pNextIdx); |
| } |
|
|
| if( rc==SQLITE_OK ){ |
| |
| |
| |
| |
| |
| if( iNext>=MergeCount(p) ){ |
| fts3LogMerge(16, getAbsoluteLevel(p, iLangid, iIndex, iLevel)); |
| rc = fts3SegmentMerge(p, iLangid, iIndex, iLevel); |
| *piIdx = 0; |
| }else{ |
| *piIdx = iNext; |
| } |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| int sqlite3Fts3ReadBlock( |
| Fts3Table *p, |
| sqlite3_int64 iBlockid, |
| char **paBlob, |
| int *pnBlob, |
| int *pnLoad |
| ){ |
| int rc; |
|
|
| |
| assert( pnBlob ); |
|
|
| if( p->pSegments ){ |
| rc = sqlite3_blob_reopen(p->pSegments, iBlockid); |
| }else{ |
| if( 0==p->zSegmentsTbl ){ |
| p->zSegmentsTbl = sqlite3_mprintf("%s_segments", p->zName); |
| if( 0==p->zSegmentsTbl ) return SQLITE_NOMEM; |
| } |
| rc = sqlite3_blob_open( |
| p->db, p->zDb, p->zSegmentsTbl, "block", iBlockid, 0, &p->pSegments |
| ); |
| } |
|
|
| if( rc==SQLITE_OK ){ |
| int nByte = sqlite3_blob_bytes(p->pSegments); |
| *pnBlob = nByte; |
| if( paBlob ){ |
| char *aByte = sqlite3_malloc64((i64)nByte + FTS3_NODE_PADDING); |
| if( !aByte ){ |
| rc = SQLITE_NOMEM; |
| }else{ |
| if( pnLoad && nByte>(FTS3_NODE_CHUNK_THRESHOLD) ){ |
| nByte = FTS3_NODE_CHUNKSIZE; |
| *pnLoad = nByte; |
| } |
| rc = sqlite3_blob_read(p->pSegments, aByte, nByte, 0); |
| memset(&aByte[nByte], 0, FTS3_NODE_PADDING); |
| if( rc!=SQLITE_OK ){ |
| sqlite3_free(aByte); |
| aByte = 0; |
| } |
| } |
| *paBlob = aByte; |
| } |
| }else if( rc==SQLITE_ERROR ){ |
| rc = FTS_CORRUPT_VTAB; |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| void sqlite3Fts3SegmentsClose(Fts3Table *p){ |
| sqlite3_blob_close(p->pSegments); |
| p->pSegments = 0; |
| } |
| |
| static int fts3SegReaderIncrRead(Fts3SegReader *pReader){ |
| int nRead; |
| int rc; |
|
|
| nRead = MIN(pReader->nNode - pReader->nPopulate, FTS3_NODE_CHUNKSIZE); |
| rc = sqlite3_blob_read( |
| pReader->pBlob, |
| &pReader->aNode[pReader->nPopulate], |
| nRead, |
| pReader->nPopulate |
| ); |
|
|
| if( rc==SQLITE_OK ){ |
| pReader->nPopulate += nRead; |
| memset(&pReader->aNode[pReader->nPopulate], 0, FTS3_NODE_PADDING); |
| if( pReader->nPopulate==pReader->nNode ){ |
| sqlite3_blob_close(pReader->pBlob); |
| pReader->pBlob = 0; |
| pReader->nPopulate = 0; |
| } |
| } |
| return rc; |
| } |
|
|
| static int fts3SegReaderRequire(Fts3SegReader *pReader, char *pFrom, int nByte){ |
| int rc = SQLITE_OK; |
| assert( !pReader->pBlob |
| || (pFrom>=pReader->aNode && pFrom<&pReader->aNode[pReader->nNode]) |
| ); |
| while( pReader->pBlob && rc==SQLITE_OK |
| && (pFrom - pReader->aNode + nByte)>pReader->nPopulate |
| ){ |
| rc = fts3SegReaderIncrRead(pReader); |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| static void fts3SegReaderSetEof(Fts3SegReader *pSeg){ |
| if( !fts3SegReaderIsRootOnly(pSeg) ){ |
| sqlite3_free(pSeg->aNode); |
| sqlite3_blob_close(pSeg->pBlob); |
| pSeg->pBlob = 0; |
| } |
| pSeg->aNode = 0; |
| } |
|
|
| |
| |
| |
| |
| |
| static int fts3SegReaderNext( |
| Fts3Table *p, |
| Fts3SegReader *pReader, |
| int bIncr |
| ){ |
| int rc; |
| char *pNext; |
| int nPrefix; |
| int nSuffix; |
|
|
| if( !pReader->aDoclist ){ |
| pNext = pReader->aNode; |
| }else{ |
| pNext = &pReader->aDoclist[pReader->nDoclist]; |
| } |
|
|
| if( !pNext || pNext>=&pReader->aNode[pReader->nNode] ){ |
|
|
| if( fts3SegReaderIsPending(pReader) ){ |
| Fts3HashElem *pElem = *(pReader->ppNextElem); |
| sqlite3_free(pReader->aNode); |
| pReader->aNode = 0; |
| if( pElem ){ |
| char *aCopy; |
| PendingList *pList = (PendingList *)fts3HashData(pElem); |
| int nCopy = pList->nData+1; |
|
|
| int nTerm = fts3HashKeysize(pElem); |
| if( (nTerm+1)>pReader->nTermAlloc ){ |
| sqlite3_free(pReader->zTerm); |
| pReader->zTerm = (char*)sqlite3_malloc64(((i64)nTerm+1)*2); |
| if( !pReader->zTerm ) return SQLITE_NOMEM; |
| pReader->nTermAlloc = (nTerm+1)*2; |
| } |
| memcpy(pReader->zTerm, fts3HashKey(pElem), nTerm); |
| pReader->zTerm[nTerm] = '\0'; |
| pReader->nTerm = nTerm; |
|
|
| aCopy = (char*)sqlite3_malloc64(nCopy); |
| if( !aCopy ) return SQLITE_NOMEM; |
| memcpy(aCopy, pList->aData, nCopy); |
| pReader->nNode = pReader->nDoclist = nCopy; |
| pReader->aNode = pReader->aDoclist = aCopy; |
| pReader->ppNextElem++; |
| assert( pReader->aNode ); |
| } |
| return SQLITE_OK; |
| } |
|
|
| fts3SegReaderSetEof(pReader); |
|
|
| |
| |
| #ifdef CORRUPT_DB |
| assert( pReader->iCurrentBlock<=pReader->iLeafEndBlock || CORRUPT_DB ); |
| #endif |
| if( pReader->iCurrentBlock>=pReader->iLeafEndBlock ){ |
| return SQLITE_OK; |
| } |
|
|
| rc = sqlite3Fts3ReadBlock( |
| p, ++pReader->iCurrentBlock, &pReader->aNode, &pReader->nNode, |
| (bIncr ? &pReader->nPopulate : 0) |
| ); |
| if( rc!=SQLITE_OK ) return rc; |
| assert( pReader->pBlob==0 ); |
| if( bIncr && pReader->nPopulate<pReader->nNode ){ |
| pReader->pBlob = p->pSegments; |
| p->pSegments = 0; |
| } |
| pNext = pReader->aNode; |
| } |
|
|
| assert( !fts3SegReaderIsPending(pReader) ); |
|
|
| rc = fts3SegReaderRequire(pReader, pNext, FTS3_VARINT_MAX*2); |
| if( rc!=SQLITE_OK ) return rc; |
| |
| |
| |
| pNext += fts3GetVarint32(pNext, &nPrefix); |
| pNext += fts3GetVarint32(pNext, &nSuffix); |
| if( nSuffix<=0 |
| || (&pReader->aNode[pReader->nNode] - pNext)<nSuffix |
| || nPrefix>pReader->nTerm |
| ){ |
| return FTS_CORRUPT_VTAB; |
| } |
|
|
| |
| |
| |
| if( (i64)nPrefix+nSuffix>(i64)pReader->nTermAlloc ){ |
| i64 nNew = ((i64)nPrefix+nSuffix)*2; |
| char *zNew = sqlite3_realloc64(pReader->zTerm, nNew); |
| if( !zNew ){ |
| return SQLITE_NOMEM; |
| } |
| pReader->zTerm = zNew; |
| pReader->nTermAlloc = nNew; |
| } |
|
|
| rc = fts3SegReaderRequire(pReader, pNext, nSuffix+FTS3_VARINT_MAX); |
| if( rc!=SQLITE_OK ) return rc; |
|
|
| memcpy(&pReader->zTerm[nPrefix], pNext, nSuffix); |
| pReader->nTerm = nPrefix+nSuffix; |
| pNext += nSuffix; |
| pNext += fts3GetVarint32(pNext, &pReader->nDoclist); |
| pReader->aDoclist = pNext; |
| pReader->pOffsetList = 0; |
|
|
| |
| |
| |
| |
| if( pReader->nDoclist > pReader->nNode-(pReader->aDoclist-pReader->aNode) |
| || (pReader->nPopulate==0 && pReader->aDoclist[pReader->nDoclist-1]) |
| || pReader->nDoclist==0 |
| ){ |
| return FTS_CORRUPT_VTAB; |
| } |
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| |
| static int fts3SegReaderFirstDocid(Fts3Table *pTab, Fts3SegReader *pReader){ |
| int rc = SQLITE_OK; |
| assert( pReader->aDoclist ); |
| assert( !pReader->pOffsetList ); |
| if( pTab->bDescIdx && fts3SegReaderIsPending(pReader) ){ |
| u8 bEof = 0; |
| pReader->iDocid = 0; |
| pReader->nOffsetList = 0; |
| sqlite3Fts3DoclistPrev(0, |
| pReader->aDoclist, pReader->nDoclist, &pReader->pOffsetList, |
| &pReader->iDocid, &pReader->nOffsetList, &bEof |
| ); |
| }else{ |
| rc = fts3SegReaderRequire(pReader, pReader->aDoclist, FTS3_VARINT_MAX); |
| if( rc==SQLITE_OK ){ |
| int n = sqlite3Fts3GetVarint(pReader->aDoclist, &pReader->iDocid); |
| pReader->pOffsetList = &pReader->aDoclist[n]; |
| } |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3SegReaderNextDocid( |
| Fts3Table *pTab, |
| Fts3SegReader *pReader, |
| char **ppOffsetList, |
| int *pnOffsetList |
| ){ |
| int rc = SQLITE_OK; |
| char *p = pReader->pOffsetList; |
| char c = 0; |
|
|
| assert( p ); |
|
|
| if( pTab->bDescIdx && fts3SegReaderIsPending(pReader) ){ |
| |
| |
| |
| u8 bEof = 0; |
| if( ppOffsetList ){ |
| *ppOffsetList = pReader->pOffsetList; |
| *pnOffsetList = pReader->nOffsetList - 1; |
| } |
| sqlite3Fts3DoclistPrev(0, |
| pReader->aDoclist, pReader->nDoclist, &p, &pReader->iDocid, |
| &pReader->nOffsetList, &bEof |
| ); |
| if( bEof ){ |
| pReader->pOffsetList = 0; |
| }else{ |
| pReader->pOffsetList = p; |
| } |
| }else{ |
| char *pEnd = &pReader->aDoclist[pReader->nDoclist]; |
|
|
| |
| |
| |
| while( 1 ){ |
| |
| |
| |
| |
| |
| |
| |
| while( *p | c ) c = *p++ & 0x80; |
| assert( *p==0 ); |
| |
| if( pReader->pBlob==0 || p<&pReader->aNode[pReader->nPopulate] ) break; |
| rc = fts3SegReaderIncrRead(pReader); |
| if( rc!=SQLITE_OK ) return rc; |
| } |
| p++; |
| |
| |
| |
| |
| if( ppOffsetList ){ |
| *ppOffsetList = pReader->pOffsetList; |
| *pnOffsetList = (int)(p - pReader->pOffsetList - 1); |
| } |
|
|
| |
| while( p<pEnd && *p==0 ) p++; |
| |
| |
| |
| |
| |
| |
| if( p>=pEnd ){ |
| pReader->pOffsetList = 0; |
| }else{ |
| rc = fts3SegReaderRequire(pReader, p, FTS3_VARINT_MAX); |
| if( rc==SQLITE_OK ){ |
| u64 iDelta; |
| pReader->pOffsetList = p + sqlite3Fts3GetVarintU(p, &iDelta); |
| if( pTab->bDescIdx ){ |
| pReader->iDocid = (i64)((u64)pReader->iDocid - iDelta); |
| }else{ |
| pReader->iDocid = (i64)((u64)pReader->iDocid + iDelta); |
| } |
| } |
| } |
| } |
|
|
| return rc; |
| } |
|
|
|
|
| int sqlite3Fts3MsrOvfl( |
| Fts3Cursor *pCsr, |
| Fts3MultiSegReader *pMsr, |
| int *pnOvfl |
| ){ |
| Fts3Table *p = (Fts3Table*)pCsr->base.pVtab; |
| int nOvfl = 0; |
| int ii; |
| int rc = SQLITE_OK; |
| int pgsz = p->nPgsz; |
|
|
| assert( p->bFts4 ); |
| assert( pgsz>0 ); |
|
|
| for(ii=0; rc==SQLITE_OK && ii<pMsr->nSegment; ii++){ |
| Fts3SegReader *pReader = pMsr->apSegment[ii]; |
| if( !fts3SegReaderIsPending(pReader) |
| && !fts3SegReaderIsRootOnly(pReader) |
| ){ |
| sqlite3_int64 jj; |
| for(jj=pReader->iStartBlock; jj<=pReader->iLeafEndBlock; jj++){ |
| int nBlob; |
| rc = sqlite3Fts3ReadBlock(p, jj, 0, &nBlob, 0); |
| if( rc!=SQLITE_OK ) break; |
| if( (nBlob+35)>pgsz ){ |
| nOvfl += (nBlob + 34)/pgsz; |
| } |
| } |
| } |
| } |
| *pnOvfl = nOvfl; |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| void sqlite3Fts3SegReaderFree(Fts3SegReader *pReader){ |
| if( pReader ){ |
| sqlite3_free(pReader->zTerm); |
| if( !fts3SegReaderIsRootOnly(pReader) ){ |
| sqlite3_free(pReader->aNode); |
| } |
| sqlite3_blob_close(pReader->pBlob); |
| } |
| sqlite3_free(pReader); |
| } |
|
|
| |
| |
| |
| int sqlite3Fts3SegReaderNew( |
| int iAge, |
| int bLookup, |
| sqlite3_int64 iStartLeaf, |
| sqlite3_int64 iEndLeaf, |
| sqlite3_int64 iEndBlock, |
| const char *zRoot, |
| int nRoot, |
| Fts3SegReader **ppReader |
| ){ |
| Fts3SegReader *pReader; |
| int nExtra = 0; |
|
|
| assert( zRoot!=0 || nRoot==0 ); |
| #ifdef CORRUPT_DB |
| assert( zRoot!=0 || CORRUPT_DB ); |
| #endif |
|
|
| if( iStartLeaf==0 ){ |
| if( iEndLeaf!=0 ) return FTS_CORRUPT_VTAB; |
| nExtra = nRoot + FTS3_NODE_PADDING; |
| } |
|
|
| pReader = (Fts3SegReader *)sqlite3_malloc64(sizeof(Fts3SegReader) + nExtra); |
| if( !pReader ){ |
| return SQLITE_NOMEM; |
| } |
| memset(pReader, 0, sizeof(Fts3SegReader)); |
| pReader->iIdx = iAge; |
| pReader->bLookup = bLookup!=0; |
| pReader->iStartBlock = iStartLeaf; |
| pReader->iLeafEndBlock = iEndLeaf; |
| pReader->iEndBlock = iEndBlock; |
|
|
| if( nExtra ){ |
| |
| pReader->aNode = (char *)&pReader[1]; |
| pReader->rootOnly = 1; |
| pReader->nNode = nRoot; |
| if( nRoot ) memcpy(pReader->aNode, zRoot, nRoot); |
| memset(&pReader->aNode[nRoot], 0, FTS3_NODE_PADDING); |
| }else{ |
| pReader->iCurrentBlock = iStartLeaf-1; |
| } |
| *ppReader = pReader; |
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| |
| |
| static int SQLITE_CDECL fts3CompareElemByTerm( |
| const void *lhs, |
| const void *rhs |
| ){ |
| char *z1 = fts3HashKey(*(Fts3HashElem **)lhs); |
| char *z2 = fts3HashKey(*(Fts3HashElem **)rhs); |
| int n1 = fts3HashKeysize(*(Fts3HashElem **)lhs); |
| int n2 = fts3HashKeysize(*(Fts3HashElem **)rhs); |
|
|
| int n = (n1<n2 ? n1 : n2); |
| int c = memcmp(z1, z2, n); |
| if( c==0 ){ |
| c = n1 - n2; |
| } |
| return c; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| int sqlite3Fts3SegReaderPending( |
| Fts3Table *p, |
| int iIndex, |
| const char *zTerm, |
| int nTerm, |
| int bPrefix, |
| Fts3SegReader **ppReader |
| ){ |
| Fts3SegReader *pReader = 0; |
| Fts3HashElem *pE; |
| Fts3HashElem **aElem = 0; |
| int nElem = 0; |
| int rc = SQLITE_OK; |
| Fts3Hash *pHash; |
|
|
| pHash = &p->aIndex[iIndex].hPending; |
| if( bPrefix ){ |
| int nAlloc = 0; |
|
|
| for(pE=fts3HashFirst(pHash); pE; pE=fts3HashNext(pE)){ |
| char *zKey = (char *)fts3HashKey(pE); |
| int nKey = fts3HashKeysize(pE); |
| if( nTerm==0 || (nKey>=nTerm && 0==memcmp(zKey, zTerm, nTerm)) ){ |
| if( nElem==nAlloc ){ |
| Fts3HashElem **aElem2; |
| nAlloc += 16; |
| aElem2 = (Fts3HashElem **)sqlite3_realloc64( |
| aElem, nAlloc*sizeof(Fts3HashElem *) |
| ); |
| if( !aElem2 ){ |
| rc = SQLITE_NOMEM; |
| nElem = 0; |
| break; |
| } |
| aElem = aElem2; |
| } |
|
|
| aElem[nElem++] = pE; |
| } |
| } |
|
|
| |
| |
| |
| |
| if( nElem>1 ){ |
| qsort(aElem, nElem, sizeof(Fts3HashElem *), fts3CompareElemByTerm); |
| } |
|
|
| }else{ |
| |
| |
| |
| |
| |
| |
| |
| pE = fts3HashFindElem(pHash, zTerm, nTerm); |
| if( pE ){ |
| aElem = &pE; |
| nElem = 1; |
| } |
| } |
|
|
| if( nElem>0 ){ |
| sqlite3_int64 nByte; |
| nByte = sizeof(Fts3SegReader) + (nElem+1)*sizeof(Fts3HashElem *); |
| pReader = (Fts3SegReader *)sqlite3_malloc64(nByte); |
| if( !pReader ){ |
| rc = SQLITE_NOMEM; |
| }else{ |
| memset(pReader, 0, nByte); |
| pReader->iIdx = 0x7FFFFFFF; |
| pReader->ppNextElem = (Fts3HashElem **)&pReader[1]; |
| memcpy(pReader->ppNextElem, aElem, nElem*sizeof(Fts3HashElem *)); |
| } |
| } |
|
|
| if( bPrefix ){ |
| sqlite3_free(aElem); |
| } |
| *ppReader = pReader; |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3SegReaderCmp(Fts3SegReader *pLhs, Fts3SegReader *pRhs){ |
| int rc; |
| if( pLhs->aNode && pRhs->aNode ){ |
| int rc2 = pLhs->nTerm - pRhs->nTerm; |
| if( rc2<0 ){ |
| rc = memcmp(pLhs->zTerm, pRhs->zTerm, pLhs->nTerm); |
| }else{ |
| rc = memcmp(pLhs->zTerm, pRhs->zTerm, pRhs->nTerm); |
| } |
| if( rc==0 ){ |
| rc = rc2; |
| } |
| }else{ |
| rc = (pLhs->aNode==0) - (pRhs->aNode==0); |
| } |
| if( rc==0 ){ |
| rc = pRhs->iIdx - pLhs->iIdx; |
| } |
| assert_fts3_nc( rc!=0 ); |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3SegReaderDoclistCmp(Fts3SegReader *pLhs, Fts3SegReader *pRhs){ |
| int rc = (pLhs->pOffsetList==0)-(pRhs->pOffsetList==0); |
| if( rc==0 ){ |
| if( pLhs->iDocid==pRhs->iDocid ){ |
| rc = pRhs->iIdx - pLhs->iIdx; |
| }else{ |
| rc = (pLhs->iDocid > pRhs->iDocid) ? 1 : -1; |
| } |
| } |
| assert( pLhs->aNode && pRhs->aNode ); |
| return rc; |
| } |
| static int fts3SegReaderDoclistCmpRev(Fts3SegReader *pLhs, Fts3SegReader *pRhs){ |
| int rc = (pLhs->pOffsetList==0)-(pRhs->pOffsetList==0); |
| if( rc==0 ){ |
| if( pLhs->iDocid==pRhs->iDocid ){ |
| rc = pRhs->iIdx - pLhs->iIdx; |
| }else{ |
| rc = (pLhs->iDocid < pRhs->iDocid) ? 1 : -1; |
| } |
| } |
| assert( pLhs->aNode && pRhs->aNode ); |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3SegReaderTermCmp( |
| Fts3SegReader *pSeg, |
| const char *zTerm, |
| int nTerm |
| ){ |
| int res = 0; |
| if( pSeg->aNode ){ |
| if( pSeg->nTerm>nTerm ){ |
| res = memcmp(pSeg->zTerm, zTerm, nTerm); |
| }else{ |
| res = memcmp(pSeg->zTerm, zTerm, pSeg->nTerm); |
| } |
| if( res==0 ){ |
| res = pSeg->nTerm-nTerm; |
| } |
| } |
| return res; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| static void fts3SegReaderSort( |
| Fts3SegReader **apSegment, |
| int nSegment, |
| int nSuspect, |
| int (*xCmp)(Fts3SegReader *, Fts3SegReader *) |
| ){ |
| int i; |
|
|
| assert( nSuspect<=nSegment ); |
|
|
| if( nSuspect==nSegment ) nSuspect--; |
| for(i=nSuspect-1; i>=0; i--){ |
| int j; |
| for(j=i; j<(nSegment-1); j++){ |
| Fts3SegReader *pTmp; |
| if( xCmp(apSegment[j], apSegment[j+1])<0 ) break; |
| pTmp = apSegment[j+1]; |
| apSegment[j+1] = apSegment[j]; |
| apSegment[j] = pTmp; |
| } |
| } |
|
|
| #ifndef NDEBUG |
| |
| for(i=0; i<(nSuspect-1); i++){ |
| assert( xCmp(apSegment[i], apSegment[i+1])<0 ); |
| } |
| #endif |
| } |
|
|
| |
| |
| |
| static int fts3WriteSegment( |
| Fts3Table *p, |
| sqlite3_int64 iBlock, |
| char *z, |
| int n |
| ){ |
| sqlite3_stmt *pStmt; |
| int rc = fts3SqlStmt(p, SQL_INSERT_SEGMENTS, &pStmt, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pStmt, 1, iBlock); |
| sqlite3_bind_blob(pStmt, 2, z, n, SQLITE_STATIC); |
| sqlite3_step(pStmt); |
| rc = sqlite3_reset(pStmt); |
| sqlite3_bind_null(pStmt, 2); |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| int sqlite3Fts3MaxLevel(Fts3Table *p, int *pnMax){ |
| int rc; |
| int mxLevel = 0; |
| sqlite3_stmt *pStmt = 0; |
|
|
| rc = fts3SqlStmt(p, SQL_SELECT_MXLEVEL, &pStmt, 0); |
| if( rc==SQLITE_OK ){ |
| if( SQLITE_ROW==sqlite3_step(pStmt) ){ |
| mxLevel = sqlite3_column_int(pStmt, 0); |
| } |
| rc = sqlite3_reset(pStmt); |
| } |
| *pnMax = mxLevel; |
| return rc; |
| } |
|
|
| |
| |
| |
| static int fts3WriteSegdir( |
| Fts3Table *p, |
| sqlite3_int64 iLevel, |
| int iIdx, |
| sqlite3_int64 iStartBlock, |
| sqlite3_int64 iLeafEndBlock, |
| sqlite3_int64 iEndBlock, |
| sqlite3_int64 nLeafData, |
| char *zRoot, |
| int nRoot |
| ){ |
| sqlite3_stmt *pStmt; |
| int rc = fts3SqlStmt(p, SQL_INSERT_SEGDIR, &pStmt, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pStmt, 1, iLevel); |
| sqlite3_bind_int(pStmt, 2, iIdx); |
| sqlite3_bind_int64(pStmt, 3, iStartBlock); |
| sqlite3_bind_int64(pStmt, 4, iLeafEndBlock); |
| if( nLeafData==0 ){ |
| sqlite3_bind_int64(pStmt, 5, iEndBlock); |
| }else{ |
| char *zEnd = sqlite3_mprintf("%lld %lld", iEndBlock, nLeafData); |
| if( !zEnd ) return SQLITE_NOMEM; |
| sqlite3_bind_text(pStmt, 5, zEnd, -1, sqlite3_free); |
| } |
| sqlite3_bind_blob(pStmt, 6, zRoot, nRoot, SQLITE_STATIC); |
| sqlite3_step(pStmt); |
| rc = sqlite3_reset(pStmt); |
| sqlite3_bind_null(pStmt, 6); |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3PrefixCompress( |
| const char *zPrev, |
| int nPrev, |
| const char *zNext, |
| int nNext |
| ){ |
| int n; |
| for(n=0; n<nPrev && n<nNext && zPrev[n]==zNext[n]; n++); |
| assert_fts3_nc( n<nNext ); |
| return n; |
| } |
|
|
| |
| |
| |
| |
| static int fts3NodeAddTerm( |
| Fts3Table *p, |
| SegmentNode **ppTree, |
| int isCopyTerm, |
| const char *zTerm, |
| int nTerm |
| ){ |
| SegmentNode *pTree = *ppTree; |
| int rc; |
| SegmentNode *pNew; |
|
|
| |
| |
| |
| if( pTree ){ |
| int nData = pTree->nData; |
| int nReq = nData; |
| int nPrefix; |
| int nSuffix; |
|
|
| nPrefix = fts3PrefixCompress(pTree->zTerm, pTree->nTerm, zTerm, nTerm); |
| nSuffix = nTerm-nPrefix; |
|
|
| |
| |
| |
| if( nSuffix<=0 ) return FTS_CORRUPT_VTAB; |
|
|
| nReq += sqlite3Fts3VarintLen(nPrefix)+sqlite3Fts3VarintLen(nSuffix)+nSuffix; |
| if( nReq<=p->nNodeSize || !pTree->zTerm ){ |
|
|
| if( nReq>p->nNodeSize ){ |
| |
| |
| |
| |
| |
| |
| |
| assert( pTree->aData==(char *)&pTree[1] ); |
| pTree->aData = (char *)sqlite3_malloc64(nReq); |
| if( !pTree->aData ){ |
| return SQLITE_NOMEM; |
| } |
| } |
|
|
| if( pTree->zTerm ){ |
| |
| nData += sqlite3Fts3PutVarint(&pTree->aData[nData], nPrefix); |
| } |
|
|
| nData += sqlite3Fts3PutVarint(&pTree->aData[nData], nSuffix); |
| memcpy(&pTree->aData[nData], &zTerm[nPrefix], nSuffix); |
| pTree->nData = nData + nSuffix; |
| pTree->nEntry++; |
|
|
| if( isCopyTerm ){ |
| if( pTree->nMalloc<nTerm ){ |
| char *zNew = sqlite3_realloc64(pTree->zMalloc, (i64)nTerm*2); |
| if( !zNew ){ |
| return SQLITE_NOMEM; |
| } |
| pTree->nMalloc = nTerm*2; |
| pTree->zMalloc = zNew; |
| } |
| pTree->zTerm = pTree->zMalloc; |
| memcpy(pTree->zTerm, zTerm, nTerm); |
| pTree->nTerm = nTerm; |
| }else{ |
| pTree->zTerm = (char *)zTerm; |
| pTree->nTerm = nTerm; |
| } |
| return SQLITE_OK; |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| pNew = (SegmentNode *)sqlite3_malloc64(sizeof(SegmentNode) + p->nNodeSize); |
| if( !pNew ){ |
| return SQLITE_NOMEM; |
| } |
| memset(pNew, 0, sizeof(SegmentNode)); |
| pNew->nData = 1 + FTS3_VARINT_MAX; |
| pNew->aData = (char *)&pNew[1]; |
|
|
| if( pTree ){ |
| SegmentNode *pParent = pTree->pParent; |
| rc = fts3NodeAddTerm(p, &pParent, isCopyTerm, zTerm, nTerm); |
| if( pTree->pParent==0 ){ |
| pTree->pParent = pParent; |
| } |
| pTree->pRight = pNew; |
| pNew->pLeftmost = pTree->pLeftmost; |
| pNew->pParent = pParent; |
| pNew->zMalloc = pTree->zMalloc; |
| pNew->nMalloc = pTree->nMalloc; |
| pTree->zMalloc = 0; |
| }else{ |
| pNew->pLeftmost = pNew; |
| rc = fts3NodeAddTerm(p, &pNew, isCopyTerm, zTerm, nTerm); |
| } |
|
|
| *ppTree = pNew; |
| return rc; |
| } |
|
|
| |
| |
| |
| static int fts3TreeFinishNode( |
| SegmentNode *pTree, |
| int iHeight, |
| sqlite3_int64 iLeftChild |
| ){ |
| int nStart; |
| assert( iHeight>=1 && iHeight<128 ); |
| nStart = FTS3_VARINT_MAX - sqlite3Fts3VarintLen(iLeftChild); |
| pTree->aData[nStart] = (char)iHeight; |
| sqlite3Fts3PutVarint(&pTree->aData[nStart+1], iLeftChild); |
| return nStart; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3NodeWrite( |
| Fts3Table *p, |
| SegmentNode *pTree, |
| int iHeight, |
| sqlite3_int64 iLeaf, |
| sqlite3_int64 iFree, |
| sqlite3_int64 *piLast, |
| char **paRoot, |
| int *pnRoot |
| ){ |
| int rc = SQLITE_OK; |
|
|
| if( !pTree->pParent ){ |
| |
| int nStart = fts3TreeFinishNode(pTree, iHeight, iLeaf); |
| *piLast = iFree-1; |
| *pnRoot = pTree->nData - nStart; |
| *paRoot = &pTree->aData[nStart]; |
| }else{ |
| SegmentNode *pIter; |
| sqlite3_int64 iNextFree = iFree; |
| sqlite3_int64 iNextLeaf = iLeaf; |
| for(pIter=pTree->pLeftmost; pIter && rc==SQLITE_OK; pIter=pIter->pRight){ |
| int nStart = fts3TreeFinishNode(pIter, iHeight, iNextLeaf); |
| int nWrite = pIter->nData - nStart; |
| |
| rc = fts3WriteSegment(p, iNextFree, &pIter->aData[nStart], nWrite); |
| iNextFree++; |
| iNextLeaf += (pIter->nEntry+1); |
| } |
| if( rc==SQLITE_OK ){ |
| assert( iNextLeaf==iFree ); |
| rc = fts3NodeWrite( |
| p, pTree->pParent, iHeight+1, iFree, iNextFree, piLast, paRoot, pnRoot |
| ); |
| } |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| static void fts3NodeFree(SegmentNode *pTree){ |
| if( pTree ){ |
| SegmentNode *p = pTree->pLeftmost; |
| fts3NodeFree(p->pParent); |
| while( p ){ |
| SegmentNode *pRight = p->pRight; |
| if( p->aData!=(char *)&p[1] ){ |
| sqlite3_free(p->aData); |
| } |
| assert( pRight==0 || p->zMalloc==0 ); |
| sqlite3_free(p->zMalloc); |
| sqlite3_free(p); |
| p = pRight; |
| } |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3SegWriterAdd( |
| Fts3Table *p, |
| SegmentWriter **ppWriter, |
| int isCopyTerm, |
| const char *zTerm, |
| int nTerm, |
| const char *aDoclist, |
| int nDoclist |
| ){ |
| int nPrefix; |
| int nSuffix; |
| i64 nReq; |
| int nData; |
| SegmentWriter *pWriter = *ppWriter; |
|
|
| if( !pWriter ){ |
| int rc; |
| sqlite3_stmt *pStmt; |
|
|
| |
| pWriter = (SegmentWriter *)sqlite3_malloc64(sizeof(SegmentWriter)); |
| if( !pWriter ) return SQLITE_NOMEM; |
| memset(pWriter, 0, sizeof(SegmentWriter)); |
| *ppWriter = pWriter; |
|
|
| |
| pWriter->aData = (char *)sqlite3_malloc64(p->nNodeSize); |
| if( !pWriter->aData ) return SQLITE_NOMEM; |
| pWriter->nSize = p->nNodeSize; |
|
|
| |
| rc = fts3SqlStmt(p, SQL_NEXT_SEGMENTS_ID, &pStmt, 0); |
| if( rc!=SQLITE_OK ) return rc; |
| if( SQLITE_ROW==sqlite3_step(pStmt) ){ |
| pWriter->iFree = sqlite3_column_int64(pStmt, 0); |
| pWriter->iFirst = pWriter->iFree; |
| } |
| rc = sqlite3_reset(pStmt); |
| if( rc!=SQLITE_OK ) return rc; |
| } |
| nData = pWriter->nData; |
|
|
| nPrefix = fts3PrefixCompress(pWriter->zTerm, pWriter->nTerm, zTerm, nTerm); |
| nSuffix = nTerm-nPrefix; |
|
|
| |
| |
| |
| if( nSuffix<=0 ) return FTS_CORRUPT_VTAB; |
|
|
| |
| nReq = sqlite3Fts3VarintLen(nPrefix) + |
| sqlite3Fts3VarintLen(nSuffix) + |
| nSuffix + |
| sqlite3Fts3VarintLen(nDoclist) + |
| nDoclist; |
|
|
| if( nData>0 && nData+nReq>p->nNodeSize ){ |
| int rc; |
|
|
| |
| if( pWriter->iFree==LARGEST_INT64 ) return FTS_CORRUPT_VTAB; |
| rc = fts3WriteSegment(p, pWriter->iFree++, pWriter->aData, nData); |
| if( rc!=SQLITE_OK ) return rc; |
| p->nLeafAdd++; |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| assert( nPrefix<nTerm ); |
| rc = fts3NodeAddTerm(p, &pWriter->pTree, isCopyTerm, zTerm, nPrefix+1); |
| if( rc!=SQLITE_OK ) return rc; |
|
|
| nData = 0; |
| pWriter->nTerm = 0; |
|
|
| nPrefix = 0; |
| nSuffix = nTerm; |
| nReq = 1 + |
| sqlite3Fts3VarintLen(nTerm) + |
| nTerm + |
| sqlite3Fts3VarintLen(nDoclist) + |
| nDoclist; |
| } |
|
|
| |
| pWriter->nLeafData += nReq; |
|
|
| |
| |
| |
| if( nReq>pWriter->nSize ){ |
| char *aNew = sqlite3_realloc64(pWriter->aData, nReq); |
| if( !aNew ) return SQLITE_NOMEM; |
| pWriter->aData = aNew; |
| pWriter->nSize = nReq; |
| } |
| assert( nData+nReq<=pWriter->nSize ); |
|
|
| |
| nData += sqlite3Fts3PutVarint(&pWriter->aData[nData], nPrefix); |
| nData += sqlite3Fts3PutVarint(&pWriter->aData[nData], nSuffix); |
| assert( nSuffix>0 ); |
| memcpy(&pWriter->aData[nData], &zTerm[nPrefix], nSuffix); |
| nData += nSuffix; |
| nData += sqlite3Fts3PutVarint(&pWriter->aData[nData], nDoclist); |
| assert( nDoclist>0 ); |
| memcpy(&pWriter->aData[nData], aDoclist, nDoclist); |
| pWriter->nData = nData + nDoclist; |
|
|
| |
| |
| |
| |
| |
| if( isCopyTerm ){ |
| if( nTerm>pWriter->nMalloc ){ |
| char *zNew = sqlite3_realloc64(pWriter->zMalloc, (i64)nTerm*2); |
| if( !zNew ){ |
| return SQLITE_NOMEM; |
| } |
| pWriter->nMalloc = nTerm*2; |
| pWriter->zMalloc = zNew; |
| pWriter->zTerm = zNew; |
| } |
| assert( pWriter->zTerm==pWriter->zMalloc ); |
| assert( nTerm>0 ); |
| memcpy(pWriter->zTerm, zTerm, nTerm); |
| }else{ |
| pWriter->zTerm = (char *)zTerm; |
| } |
| pWriter->nTerm = nTerm; |
|
|
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| static int fts3SegWriterFlush( |
| Fts3Table *p, |
| SegmentWriter *pWriter, |
| sqlite3_int64 iLevel, |
| int iIdx |
| ){ |
| int rc; |
| if( pWriter->pTree ){ |
| sqlite3_int64 iLast = 0; |
| sqlite3_int64 iLastLeaf; |
| char *zRoot = NULL; |
| int nRoot = 0; |
|
|
| iLastLeaf = pWriter->iFree; |
| rc = fts3WriteSegment(p, pWriter->iFree++, pWriter->aData, pWriter->nData); |
| if( rc==SQLITE_OK ){ |
| rc = fts3NodeWrite(p, pWriter->pTree, 1, |
| pWriter->iFirst, pWriter->iFree, &iLast, &zRoot, &nRoot); |
| } |
| if( rc==SQLITE_OK ){ |
| rc = fts3WriteSegdir(p, iLevel, iIdx, |
| pWriter->iFirst, iLastLeaf, iLast, pWriter->nLeafData, zRoot, nRoot); |
| } |
| }else{ |
| |
| rc = fts3WriteSegdir(p, iLevel, iIdx, |
| 0, 0, 0, pWriter->nLeafData, pWriter->aData, pWriter->nData); |
| } |
| p->nLeafAdd++; |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| static void fts3SegWriterFree(SegmentWriter *pWriter){ |
| if( pWriter ){ |
| sqlite3_free(pWriter->aData); |
| sqlite3_free(pWriter->zMalloc); |
| fts3NodeFree(pWriter->pTree); |
| sqlite3_free(pWriter); |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3IsEmpty(Fts3Table *p, sqlite3_value *pRowid, int *pisEmpty){ |
| sqlite3_stmt *pStmt; |
| int rc; |
| if( p->zContentTbl ){ |
| |
| *pisEmpty = 0; |
| rc = SQLITE_OK; |
| }else{ |
| rc = fts3SqlStmt(p, SQL_IS_EMPTY, &pStmt, &pRowid); |
| if( rc==SQLITE_OK ){ |
| if( SQLITE_ROW==sqlite3_step(pStmt) ){ |
| *pisEmpty = sqlite3_column_int(pStmt, 0); |
| } |
| rc = sqlite3_reset(pStmt); |
| } |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3SegmentMaxLevel( |
| Fts3Table *p, |
| int iLangid, |
| int iIndex, |
| sqlite3_int64 *pnMax |
| ){ |
| sqlite3_stmt *pStmt; |
| int rc; |
| assert( iIndex>=0 && iIndex<p->nIndex ); |
|
|
| |
| |
| |
| |
| |
| |
| rc = fts3SqlStmt(p, SQL_SELECT_SEGDIR_MAX_LEVEL, &pStmt, 0); |
| if( rc!=SQLITE_OK ) return rc; |
| sqlite3_bind_int64(pStmt, 1, getAbsoluteLevel(p, iLangid, iIndex, 0)); |
| sqlite3_bind_int64(pStmt, 2, |
| getAbsoluteLevel(p, iLangid, iIndex, FTS3_SEGDIR_MAXLEVEL-1) |
| ); |
| if( SQLITE_ROW==sqlite3_step(pStmt) ){ |
| *pnMax = sqlite3_column_int64(pStmt, 0); |
| } |
| return sqlite3_reset(pStmt); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3SegmentIsMaxLevel(Fts3Table *p, i64 iAbsLevel, int *pbMax){ |
|
|
| |
| |
| |
| |
| |
| |
| sqlite3_stmt *pStmt; |
| int rc = fts3SqlStmt(p, SQL_SELECT_SEGDIR_MAX_LEVEL, &pStmt, 0); |
| if( rc!=SQLITE_OK ) return rc; |
| sqlite3_bind_int64(pStmt, 1, iAbsLevel+1); |
| sqlite3_bind_int64(pStmt, 2, |
| (((u64)iAbsLevel/FTS3_SEGDIR_MAXLEVEL)+1) * FTS3_SEGDIR_MAXLEVEL |
| ); |
|
|
| *pbMax = 0; |
| if( SQLITE_ROW==sqlite3_step(pStmt) ){ |
| *pbMax = sqlite3_column_type(pStmt, 0)==SQLITE_NULL; |
| } |
| return sqlite3_reset(pStmt); |
| } |
|
|
| |
| |
| |
| |
| |
| static int fts3DeleteSegment( |
| Fts3Table *p, |
| Fts3SegReader *pSeg |
| ){ |
| int rc = SQLITE_OK; |
| if( pSeg->iStartBlock ){ |
| sqlite3_stmt *pDelete; |
| rc = fts3SqlStmt(p, SQL_DELETE_SEGMENTS_RANGE, &pDelete, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pDelete, 1, pSeg->iStartBlock); |
| sqlite3_bind_int64(pDelete, 2, pSeg->iEndBlock); |
| sqlite3_step(pDelete); |
| rc = sqlite3_reset(pDelete); |
| } |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3DeleteSegdir( |
| Fts3Table *p, |
| int iLangid, |
| int iIndex, |
| int iLevel, |
| Fts3SegReader **apSegment, |
| int nReader |
| ){ |
| int rc = SQLITE_OK; |
| int i; |
| sqlite3_stmt *pDelete = 0; |
|
|
| for(i=0; rc==SQLITE_OK && i<nReader; i++){ |
| rc = fts3DeleteSegment(p, apSegment[i]); |
| } |
| if( rc!=SQLITE_OK ){ |
| return rc; |
| } |
|
|
| assert( iLevel>=0 || iLevel==FTS3_SEGCURSOR_ALL ); |
| if( iLevel==FTS3_SEGCURSOR_ALL ){ |
| rc = fts3SqlStmt(p, SQL_DELETE_SEGDIR_RANGE, &pDelete, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pDelete, 1, getAbsoluteLevel(p, iLangid, iIndex, 0)); |
| sqlite3_bind_int64(pDelete, 2, |
| getAbsoluteLevel(p, iLangid, iIndex, FTS3_SEGDIR_MAXLEVEL-1) |
| ); |
| } |
| }else{ |
| rc = fts3SqlStmt(p, SQL_DELETE_SEGDIR_LEVEL, &pDelete, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64( |
| pDelete, 1, getAbsoluteLevel(p, iLangid, iIndex, iLevel) |
| ); |
| } |
| } |
|
|
| if( rc==SQLITE_OK ){ |
| sqlite3_step(pDelete); |
| rc = sqlite3_reset(pDelete); |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static void fts3ColumnFilter( |
| int iCol, |
| int bZero, |
| char **ppList, |
| int *pnList |
| ){ |
| char *pList = *ppList; |
| int nList = *pnList; |
| char *pEnd = &pList[nList]; |
| int iCurrent = 0; |
| char *p = pList; |
|
|
| assert( iCol>=0 ); |
| while( 1 ){ |
| char c = 0; |
| while( p<pEnd && (c | *p)&0xFE ) c = *p++ & 0x80; |
| |
| if( iCol==iCurrent ){ |
| nList = (int)(p - pList); |
| break; |
| } |
|
|
| nList -= (int)(p - pList); |
| pList = p; |
| if( nList<=0 ){ |
| break; |
| } |
| p = &pList[1]; |
| p += fts3GetVarint32(p, &iCurrent); |
| } |
|
|
| if( bZero && (pEnd - &pList[nList])>0){ |
| memset(&pList[nList], 0, pEnd - &pList[nList]); |
| } |
| *ppList = pList; |
| *pnList = nList; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| static int fts3MsrBufferData( |
| Fts3MultiSegReader *pMsr, |
| char *pList, |
| i64 nList |
| ){ |
| if( (nList+FTS3_NODE_PADDING)>pMsr->nBuffer ){ |
| char *pNew; |
| int nNew = nList*2 + FTS3_NODE_PADDING; |
| pNew = (char *)sqlite3_realloc64(pMsr->aBuffer, nNew); |
| if( !pNew ) return SQLITE_NOMEM; |
| pMsr->aBuffer = pNew; |
| pMsr->nBuffer = nNew; |
| } |
|
|
| assert( nList>0 ); |
| memcpy(pMsr->aBuffer, pList, nList); |
| memset(&pMsr->aBuffer[nList], 0, FTS3_NODE_PADDING); |
| return SQLITE_OK; |
| } |
|
|
| int sqlite3Fts3MsrIncrNext( |
| Fts3Table *p, |
| Fts3MultiSegReader *pMsr, |
| sqlite3_int64 *piDocid, |
| char **paPoslist, |
| int *pnPoslist |
| ){ |
| int nMerge = pMsr->nAdvance; |
| Fts3SegReader **apSegment = pMsr->apSegment; |
| int (*xCmp)(Fts3SegReader *, Fts3SegReader *) = ( |
| p->bDescIdx ? fts3SegReaderDoclistCmpRev : fts3SegReaderDoclistCmp |
| ); |
|
|
| if( nMerge==0 ){ |
| *paPoslist = 0; |
| return SQLITE_OK; |
| } |
|
|
| while( 1 ){ |
| Fts3SegReader *pSeg; |
| pSeg = pMsr->apSegment[0]; |
|
|
| if( pSeg->pOffsetList==0 ){ |
| *paPoslist = 0; |
| break; |
| }else{ |
| int rc; |
| char *pList; |
| int nList; |
| int j; |
| sqlite3_int64 iDocid = apSegment[0]->iDocid; |
|
|
| rc = fts3SegReaderNextDocid(p, apSegment[0], &pList, &nList); |
| j = 1; |
| while( rc==SQLITE_OK |
| && j<nMerge |
| && apSegment[j]->pOffsetList |
| && apSegment[j]->iDocid==iDocid |
| ){ |
| rc = fts3SegReaderNextDocid(p, apSegment[j], 0, 0); |
| j++; |
| } |
| if( rc!=SQLITE_OK ) return rc; |
| fts3SegReaderSort(pMsr->apSegment, nMerge, j, xCmp); |
|
|
| if( nList>0 && fts3SegReaderIsPending(apSegment[0]) ){ |
| rc = fts3MsrBufferData(pMsr, pList, (i64)nList+1); |
| if( rc!=SQLITE_OK ) return rc; |
| assert( (pMsr->aBuffer[nList] & 0xFE)==0x00 ); |
| pList = pMsr->aBuffer; |
| } |
|
|
| if( pMsr->iColFilter>=0 ){ |
| fts3ColumnFilter(pMsr->iColFilter, 1, &pList, &nList); |
| } |
|
|
| if( nList>0 ){ |
| *paPoslist = pList; |
| *piDocid = iDocid; |
| *pnPoslist = nList; |
| break; |
| } |
| } |
| } |
|
|
| return SQLITE_OK; |
| } |
|
|
| static int fts3SegReaderStart( |
| Fts3Table *p, |
| Fts3MultiSegReader *pCsr, |
| const char *zTerm, |
| int nTerm |
| ){ |
| int i; |
| int nSeg = pCsr->nSegment; |
|
|
| |
| |
| |
| |
| |
| |
| for(i=0; pCsr->bRestart==0 && i<pCsr->nSegment; i++){ |
| int res = 0; |
| Fts3SegReader *pSeg = pCsr->apSegment[i]; |
| do { |
| int rc = fts3SegReaderNext(p, pSeg, 0); |
| if( rc!=SQLITE_OK ) return rc; |
| }while( zTerm && (res = fts3SegReaderTermCmp(pSeg, zTerm, nTerm))<0 ); |
|
|
| if( pSeg->bLookup && res!=0 ){ |
| fts3SegReaderSetEof(pSeg); |
| } |
| } |
| fts3SegReaderSort(pCsr->apSegment, nSeg, nSeg, fts3SegReaderCmp); |
|
|
| return SQLITE_OK; |
| } |
|
|
| int sqlite3Fts3SegReaderStart( |
| Fts3Table *p, |
| Fts3MultiSegReader *pCsr, |
| Fts3SegFilter *pFilter |
| ){ |
| pCsr->pFilter = pFilter; |
| return fts3SegReaderStart(p, pCsr, pFilter->zTerm, pFilter->nTerm); |
| } |
|
|
| int sqlite3Fts3MsrIncrStart( |
| Fts3Table *p, |
| Fts3MultiSegReader *pCsr, |
| int iCol, |
| const char *zTerm, |
| int nTerm |
| ){ |
| int i; |
| int rc; |
| int nSegment = pCsr->nSegment; |
| int (*xCmp)(Fts3SegReader *, Fts3SegReader *) = ( |
| p->bDescIdx ? fts3SegReaderDoclistCmpRev : fts3SegReaderDoclistCmp |
| ); |
|
|
| assert( pCsr->pFilter==0 ); |
| assert( zTerm && nTerm>0 ); |
|
|
| |
| rc = fts3SegReaderStart(p, pCsr, zTerm, nTerm); |
| if( rc!=SQLITE_OK ) return rc; |
|
|
| |
| for(i=0; i<nSegment; i++){ |
| Fts3SegReader *pSeg = pCsr->apSegment[i]; |
| if( !pSeg->aNode || fts3SegReaderTermCmp(pSeg, zTerm, nTerm) ){ |
| break; |
| } |
| } |
| pCsr->nAdvance = i; |
|
|
| |
| for(i=0; i<pCsr->nAdvance; i++){ |
| rc = fts3SegReaderFirstDocid(p, pCsr->apSegment[i]); |
| if( rc!=SQLITE_OK ) return rc; |
| } |
| fts3SegReaderSort(pCsr->apSegment, i, i, xCmp); |
|
|
| assert( iCol<0 || iCol<p->nColumn ); |
| pCsr->iColFilter = iCol; |
|
|
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| int sqlite3Fts3MsrIncrRestart(Fts3MultiSegReader *pCsr){ |
| int i; |
|
|
| assert( pCsr->zTerm==0 ); |
| assert( pCsr->nTerm==0 ); |
| assert( pCsr->aDoclist==0 ); |
| assert( pCsr->nDoclist==0 ); |
|
|
| pCsr->nAdvance = 0; |
| pCsr->bRestart = 1; |
| for(i=0; i<pCsr->nSegment; i++){ |
| pCsr->apSegment[i]->pOffsetList = 0; |
| pCsr->apSegment[i]->nOffsetList = 0; |
| pCsr->apSegment[i]->iDocid = 0; |
| } |
|
|
| return SQLITE_OK; |
| } |
|
|
| static int fts3GrowSegReaderBuffer(Fts3MultiSegReader *pCsr, i64 nReq){ |
| if( nReq>pCsr->nBuffer ){ |
| char *aNew; |
| pCsr->nBuffer = nReq*2; |
| aNew = sqlite3_realloc64(pCsr->aBuffer, pCsr->nBuffer); |
| if( !aNew ){ |
| return SQLITE_NOMEM; |
| } |
| pCsr->aBuffer = aNew; |
| } |
| return SQLITE_OK; |
| } |
|
|
|
|
| int sqlite3Fts3SegReaderStep( |
| Fts3Table *p, |
| Fts3MultiSegReader *pCsr |
| ){ |
| int rc = SQLITE_OK; |
|
|
| int isIgnoreEmpty = (pCsr->pFilter->flags & FTS3_SEGMENT_IGNORE_EMPTY); |
| int isRequirePos = (pCsr->pFilter->flags & FTS3_SEGMENT_REQUIRE_POS); |
| int isColFilter = (pCsr->pFilter->flags & FTS3_SEGMENT_COLUMN_FILTER); |
| int isPrefix = (pCsr->pFilter->flags & FTS3_SEGMENT_PREFIX); |
| int isScan = (pCsr->pFilter->flags & FTS3_SEGMENT_SCAN); |
| int isFirst = (pCsr->pFilter->flags & FTS3_SEGMENT_FIRST); |
|
|
| Fts3SegReader **apSegment = pCsr->apSegment; |
| int nSegment = pCsr->nSegment; |
| Fts3SegFilter *pFilter = pCsr->pFilter; |
| int (*xCmp)(Fts3SegReader *, Fts3SegReader *) = ( |
| p->bDescIdx ? fts3SegReaderDoclistCmpRev : fts3SegReaderDoclistCmp |
| ); |
|
|
| if( pCsr->nSegment==0 ) return SQLITE_OK; |
|
|
| do { |
| int nMerge; |
| int i; |
| |
| |
| |
| |
| for(i=0; i<pCsr->nAdvance; i++){ |
| Fts3SegReader *pSeg = apSegment[i]; |
| if( pSeg->bLookup ){ |
| fts3SegReaderSetEof(pSeg); |
| }else{ |
| rc = fts3SegReaderNext(p, pSeg, 0); |
| } |
| if( rc!=SQLITE_OK ) return rc; |
| } |
| fts3SegReaderSort(apSegment, nSegment, pCsr->nAdvance, fts3SegReaderCmp); |
| pCsr->nAdvance = 0; |
|
|
| |
| assert( rc==SQLITE_OK ); |
| if( apSegment[0]->aNode==0 ) break; |
|
|
| pCsr->nTerm = apSegment[0]->nTerm; |
| pCsr->zTerm = apSegment[0]->zTerm; |
|
|
| |
| |
| |
| |
| |
| |
| |
| if( pFilter->zTerm && !isScan ){ |
| if( pCsr->nTerm<pFilter->nTerm |
| || (!isPrefix && pCsr->nTerm>pFilter->nTerm) |
| || memcmp(pCsr->zTerm, pFilter->zTerm, pFilter->nTerm) |
| ){ |
| break; |
| } |
| } |
|
|
| nMerge = 1; |
| while( nMerge<nSegment |
| && apSegment[nMerge]->aNode |
| && apSegment[nMerge]->nTerm==pCsr->nTerm |
| && 0==memcmp(pCsr->zTerm, apSegment[nMerge]->zTerm, pCsr->nTerm) |
| ){ |
| nMerge++; |
| } |
|
|
| assert( isIgnoreEmpty || (isRequirePos && !isColFilter) ); |
| if( nMerge==1 |
| && !isIgnoreEmpty |
| && !isFirst |
| && (p->bDescIdx==0 || fts3SegReaderIsPending(apSegment[0])==0) |
| ){ |
| pCsr->nDoclist = apSegment[0]->nDoclist; |
| if( fts3SegReaderIsPending(apSegment[0]) ){ |
| rc = fts3MsrBufferData(pCsr, apSegment[0]->aDoclist, |
| (i64)pCsr->nDoclist); |
| pCsr->aDoclist = pCsr->aBuffer; |
| }else{ |
| pCsr->aDoclist = apSegment[0]->aDoclist; |
| } |
| if( rc==SQLITE_OK ) rc = SQLITE_ROW; |
| }else{ |
| int nDoclist = 0; |
| sqlite3_int64 iPrev = 0; |
|
|
| |
| |
| |
| |
| for(i=0; i<nMerge; i++){ |
| fts3SegReaderFirstDocid(p, apSegment[i]); |
| } |
| fts3SegReaderSort(apSegment, nMerge, nMerge, xCmp); |
| while( apSegment[0]->pOffsetList ){ |
| int j; |
| char *pList = 0; |
| int nList = 0; |
| int nByte; |
| sqlite3_int64 iDocid = apSegment[0]->iDocid; |
| fts3SegReaderNextDocid(p, apSegment[0], &pList, &nList); |
| j = 1; |
| while( j<nMerge |
| && apSegment[j]->pOffsetList |
| && apSegment[j]->iDocid==iDocid |
| ){ |
| fts3SegReaderNextDocid(p, apSegment[j], 0, 0); |
| j++; |
| } |
|
|
| if( isColFilter ){ |
| fts3ColumnFilter(pFilter->iCol, 0, &pList, &nList); |
| } |
|
|
| if( !isIgnoreEmpty || nList>0 ){ |
|
|
| |
| |
| sqlite3_int64 iDelta; |
| if( p->bDescIdx && nDoclist>0 ){ |
| if( iPrev<=iDocid ) return FTS_CORRUPT_VTAB; |
| iDelta = (i64)((u64)iPrev - (u64)iDocid); |
| }else{ |
| if( nDoclist>0 && iPrev>=iDocid ) return FTS_CORRUPT_VTAB; |
| iDelta = (i64)((u64)iDocid - (u64)iPrev); |
| } |
|
|
| nByte = sqlite3Fts3VarintLen(iDelta) + (isRequirePos?nList+1:0); |
|
|
| rc = fts3GrowSegReaderBuffer(pCsr, |
| (i64)nByte+nDoclist+FTS3_NODE_PADDING); |
| if( rc ) return rc; |
|
|
| if( isFirst ){ |
| char *a = &pCsr->aBuffer[nDoclist]; |
| int nWrite; |
| |
| nWrite = sqlite3Fts3FirstFilter(iDelta, pList, nList, a); |
| if( nWrite ){ |
| iPrev = iDocid; |
| nDoclist += nWrite; |
| } |
| }else{ |
| nDoclist += sqlite3Fts3PutVarint(&pCsr->aBuffer[nDoclist], iDelta); |
| iPrev = iDocid; |
| if( isRequirePos ){ |
| memcpy(&pCsr->aBuffer[nDoclist], pList, nList); |
| nDoclist += nList; |
| pCsr->aBuffer[nDoclist++] = '\0'; |
| } |
| } |
| } |
|
|
| fts3SegReaderSort(apSegment, nMerge, j, xCmp); |
| } |
| if( nDoclist>0 ){ |
| rc = fts3GrowSegReaderBuffer(pCsr, (i64)nDoclist+FTS3_NODE_PADDING); |
| if( rc ) return rc; |
| memset(&pCsr->aBuffer[nDoclist], 0, FTS3_NODE_PADDING); |
| pCsr->aDoclist = pCsr->aBuffer; |
| pCsr->nDoclist = nDoclist; |
| rc = SQLITE_ROW; |
| } |
| } |
| pCsr->nAdvance = nMerge; |
| }while( rc==SQLITE_OK ); |
|
|
| return rc; |
| } |
|
|
|
|
| void sqlite3Fts3SegReaderFinish( |
| Fts3MultiSegReader *pCsr |
| ){ |
| if( pCsr ){ |
| int i; |
| for(i=0; i<pCsr->nSegment; i++){ |
| sqlite3Fts3SegReaderFree(pCsr->apSegment[i]); |
| } |
| sqlite3_free(pCsr->apSegment); |
| sqlite3_free(pCsr->aBuffer); |
|
|
| pCsr->nSegment = 0; |
| pCsr->apSegment = 0; |
| pCsr->aBuffer = 0; |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static void fts3ReadEndBlockField( |
| sqlite3_stmt *pStmt, |
| int iCol, |
| i64 *piEndBlock, |
| i64 *pnByte |
| ){ |
| const unsigned char *zText = sqlite3_column_text(pStmt, iCol); |
| if( zText ){ |
| int i; |
| int iMul = 1; |
| u64 iVal = 0; |
| for(i=0; zText[i]>='0' && zText[i]<='9'; i++){ |
| iVal = iVal*10 + (zText[i] - '0'); |
| } |
| *piEndBlock = (i64)iVal; |
| while( zText[i]==' ' ) i++; |
| iVal = 0; |
| if( zText[i]=='-' ){ |
| i++; |
| iMul = -1; |
| } |
| for(; zText[i]>='0' && zText[i]<='9'; i++){ |
| iVal = iVal*10 + (zText[i] - '0'); |
| } |
| *pnByte = ((i64)iVal * (i64)iMul); |
| } |
| } |
|
|
|
|
| |
| |
| |
| |
| static int fts3PromoteSegments( |
| Fts3Table *p, |
| sqlite3_int64 iAbsLevel, |
| sqlite3_int64 nByte |
| ){ |
| int rc = SQLITE_OK; |
| sqlite3_stmt *pRange; |
|
|
| rc = fts3SqlStmt(p, SQL_SELECT_LEVEL_RANGE2, &pRange, 0); |
|
|
| if( rc==SQLITE_OK ){ |
| int bOk = 0; |
| i64 iLast = (iAbsLevel/FTS3_SEGDIR_MAXLEVEL + 1) * FTS3_SEGDIR_MAXLEVEL - 1; |
| i64 nLimit = (nByte*3)/2; |
|
|
| |
| |
| |
| |
| |
| sqlite3_bind_int64(pRange, 1, iAbsLevel+1); |
| sqlite3_bind_int64(pRange, 2, iLast); |
| while( SQLITE_ROW==sqlite3_step(pRange) ){ |
| i64 nSize = 0, dummy; |
| fts3ReadEndBlockField(pRange, 2, &dummy, &nSize); |
| if( nSize<=0 || nSize>nLimit ){ |
| |
| |
| |
| |
| |
| bOk = 0; |
| break; |
| } |
| bOk = 1; |
| } |
| rc = sqlite3_reset(pRange); |
|
|
| if( bOk ){ |
| int iIdx = 0; |
| sqlite3_stmt *pUpdate1 = 0; |
| sqlite3_stmt *pUpdate2 = 0; |
|
|
| if( rc==SQLITE_OK ){ |
| rc = fts3SqlStmt(p, SQL_UPDATE_LEVEL_IDX, &pUpdate1, 0); |
| } |
| if( rc==SQLITE_OK ){ |
| rc = fts3SqlStmt(p, SQL_UPDATE_LEVEL, &pUpdate2, 0); |
| } |
|
|
| if( rc==SQLITE_OK ){ |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| sqlite3_bind_int64(pRange, 1, iAbsLevel); |
| while( SQLITE_ROW==sqlite3_step(pRange) ){ |
| sqlite3_bind_int(pUpdate1, 1, iIdx++); |
| sqlite3_bind_int(pUpdate1, 2, sqlite3_column_int(pRange, 0)); |
| sqlite3_bind_int(pUpdate1, 3, sqlite3_column_int(pRange, 1)); |
| sqlite3_step(pUpdate1); |
| rc = sqlite3_reset(pUpdate1); |
| if( rc!=SQLITE_OK ){ |
| sqlite3_reset(pRange); |
| break; |
| } |
| } |
| } |
| if( rc==SQLITE_OK ){ |
| rc = sqlite3_reset(pRange); |
| } |
|
|
| |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pUpdate2, 1, iAbsLevel); |
| sqlite3_step(pUpdate2); |
| rc = sqlite3_reset(pUpdate2); |
| } |
| } |
| } |
|
|
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3SegmentMerge( |
| Fts3Table *p, |
| int iLangid, |
| int iIndex, |
| int iLevel |
| ){ |
| int rc; |
| int iIdx = 0; |
| sqlite3_int64 iNewLevel = 0; |
| SegmentWriter *pWriter = 0; |
| Fts3SegFilter filter; |
| Fts3MultiSegReader csr; |
| int bIgnoreEmpty = 0; |
| i64 iMaxLevel = 0; |
|
|
| assert( iLevel==FTS3_SEGCURSOR_ALL |
| || iLevel==FTS3_SEGCURSOR_PENDING |
| || iLevel>=0 |
| ); |
| assert( iLevel<FTS3_SEGDIR_MAXLEVEL ); |
| assert( iIndex>=0 && iIndex<p->nIndex ); |
|
|
| rc = sqlite3Fts3SegReaderCursor(p, iLangid, iIndex, iLevel, 0, 0, 1, 0, &csr); |
| if( rc!=SQLITE_OK || csr.nSegment==0 ) goto finished; |
|
|
| if( iLevel!=FTS3_SEGCURSOR_PENDING ){ |
| rc = fts3SegmentMaxLevel(p, iLangid, iIndex, &iMaxLevel); |
| if( rc!=SQLITE_OK ) goto finished; |
| } |
|
|
| if( iLevel==FTS3_SEGCURSOR_ALL ){ |
| |
| |
| |
| |
| if( csr.nSegment==1 && 0==fts3SegReaderIsPending(csr.apSegment[0]) ){ |
| rc = SQLITE_DONE; |
| goto finished; |
| } |
| iNewLevel = iMaxLevel; |
| bIgnoreEmpty = 1; |
|
|
| }else{ |
| |
| |
| |
| |
| assert( FTS3_SEGCURSOR_PENDING==-1 ); |
| iNewLevel = getAbsoluteLevel(p, iLangid, iIndex, iLevel+1); |
| rc = fts3AllocateSegdirIdx(p, iLangid, iIndex, iLevel+1, &iIdx); |
| bIgnoreEmpty = (iLevel!=FTS3_SEGCURSOR_PENDING) && (iNewLevel>iMaxLevel); |
| } |
| if( rc!=SQLITE_OK ) goto finished; |
|
|
| assert( csr.nSegment>0 ); |
| assert_fts3_nc( iNewLevel>=getAbsoluteLevel(p, iLangid, iIndex, 0) ); |
| assert_fts3_nc( |
| iNewLevel<getAbsoluteLevel(p, iLangid, iIndex,FTS3_SEGDIR_MAXLEVEL) |
| ); |
|
|
| memset(&filter, 0, sizeof(Fts3SegFilter)); |
| filter.flags = FTS3_SEGMENT_REQUIRE_POS; |
| filter.flags |= (bIgnoreEmpty ? FTS3_SEGMENT_IGNORE_EMPTY : 0); |
|
|
| rc = sqlite3Fts3SegReaderStart(p, &csr, &filter); |
| while( SQLITE_OK==rc ){ |
| rc = sqlite3Fts3SegReaderStep(p, &csr); |
| if( rc!=SQLITE_ROW ) break; |
| rc = fts3SegWriterAdd(p, &pWriter, 1, |
| csr.zTerm, csr.nTerm, csr.aDoclist, csr.nDoclist); |
| } |
| if( rc!=SQLITE_OK ) goto finished; |
| assert_fts3_nc( pWriter || bIgnoreEmpty ); |
|
|
| if( iLevel!=FTS3_SEGCURSOR_PENDING ){ |
| rc = fts3DeleteSegdir( |
| p, iLangid, iIndex, iLevel, csr.apSegment, csr.nSegment |
| ); |
| if( rc!=SQLITE_OK ) goto finished; |
| } |
| if( pWriter ){ |
| rc = fts3SegWriterFlush(p, pWriter, iNewLevel, iIdx); |
| if( rc==SQLITE_OK ){ |
| if( iLevel==FTS3_SEGCURSOR_PENDING || iNewLevel<iMaxLevel ){ |
| rc = fts3PromoteSegments(p, iNewLevel, pWriter->nLeafData); |
| } |
| } |
| } |
|
|
| finished: |
| fts3SegWriterFree(pWriter); |
| sqlite3Fts3SegReaderFinish(&csr); |
| return rc; |
| } |
|
|
|
|
| |
| |
| |
| int sqlite3Fts3PendingTermsFlush(Fts3Table *p){ |
| int rc = SQLITE_OK; |
| int i; |
| |
| for(i=0; rc==SQLITE_OK && i<p->nIndex; i++){ |
| rc = fts3SegmentMerge(p, p->iPrevLangid, i, FTS3_SEGCURSOR_PENDING); |
| if( rc==SQLITE_DONE ) rc = SQLITE_OK; |
| } |
|
|
| |
| |
| |
| if( rc==SQLITE_OK && p->bHasStat |
| && p->nAutoincrmerge==0xff && p->nLeafAdd>0 |
| ){ |
| sqlite3_stmt *pStmt = 0; |
| rc = fts3SqlStmt(p, SQL_SELECT_STAT, &pStmt, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int(pStmt, 1, FTS_STAT_AUTOINCRMERGE); |
| rc = sqlite3_step(pStmt); |
| if( rc==SQLITE_ROW ){ |
| p->nAutoincrmerge = sqlite3_column_int(pStmt, 0); |
| if( p->nAutoincrmerge==1 ) p->nAutoincrmerge = 8; |
| }else if( rc==SQLITE_DONE ){ |
| p->nAutoincrmerge = 0; |
| } |
| rc = sqlite3_reset(pStmt); |
| } |
| } |
|
|
| if( rc==SQLITE_OK ){ |
| sqlite3Fts3PendingTermsClear(p); |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| static void fts3EncodeIntArray( |
| int N, |
| u32 *a, |
| char *zBuf, |
| int *pNBuf |
| ){ |
| int i, j; |
| for(i=j=0; i<N; i++){ |
| j += sqlite3Fts3PutVarint(&zBuf[j], (sqlite3_int64)a[i]); |
| } |
| *pNBuf = j; |
| } |
|
|
| |
| |
| |
| static void fts3DecodeIntArray( |
| int N, |
| u32 *a, |
| const char *zBuf, |
| int nBuf |
| ){ |
| int i = 0; |
| if( nBuf && (zBuf[nBuf-1]&0x80)==0 ){ |
| int j; |
| for(i=j=0; i<N && j<nBuf; i++){ |
| sqlite3_int64 x; |
| j += sqlite3Fts3GetVarint(&zBuf[j], &x); |
| a[i] = (u32)(x & 0xffffffff); |
| } |
| } |
| while( i<N ) a[i++] = 0; |
| } |
|
|
| |
| |
| |
| |
| |
| static void fts3InsertDocsize( |
| int *pRC, |
| Fts3Table *p, |
| u32 *aSz |
| ){ |
| char *pBlob; |
| int nBlob; |
| sqlite3_stmt *pStmt; |
| int rc; |
|
|
| if( *pRC ) return; |
| pBlob = sqlite3_malloc64( 10*(sqlite3_int64)p->nColumn ); |
| if( pBlob==0 ){ |
| *pRC = SQLITE_NOMEM; |
| return; |
| } |
| fts3EncodeIntArray(p->nColumn, aSz, pBlob, &nBlob); |
| rc = fts3SqlStmt(p, SQL_REPLACE_DOCSIZE, &pStmt, 0); |
| if( rc ){ |
| sqlite3_free(pBlob); |
| *pRC = rc; |
| return; |
| } |
| sqlite3_bind_int64(pStmt, 1, p->iPrevDocid); |
| sqlite3_bind_blob(pStmt, 2, pBlob, nBlob, sqlite3_free); |
| sqlite3_step(pStmt); |
| *pRC = sqlite3_reset(pStmt); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static void fts3UpdateDocTotals( |
| int *pRC, |
| Fts3Table *p, |
| u32 *aSzIns, |
| u32 *aSzDel, |
| int nChng |
| ){ |
| char *pBlob; |
| int nBlob; |
| u32 *a; |
| sqlite3_stmt *pStmt; |
| int i; |
| int rc; |
|
|
| const int nStat = p->nColumn+2; |
|
|
| if( *pRC ) return; |
| a = sqlite3_malloc64( (sizeof(u32)+10)*(sqlite3_int64)nStat ); |
| if( a==0 ){ |
| *pRC = SQLITE_NOMEM; |
| return; |
| } |
| pBlob = (char*)&a[nStat]; |
| rc = fts3SqlStmt(p, SQL_SELECT_STAT, &pStmt, 0); |
| if( rc ){ |
| sqlite3_free(a); |
| *pRC = rc; |
| return; |
| } |
| sqlite3_bind_int(pStmt, 1, FTS_STAT_DOCTOTAL); |
| if( sqlite3_step(pStmt)==SQLITE_ROW ){ |
| fts3DecodeIntArray(nStat, a, |
| sqlite3_column_blob(pStmt, 0), |
| sqlite3_column_bytes(pStmt, 0)); |
| }else{ |
| memset(a, 0, sizeof(u32)*(nStat) ); |
| } |
| rc = sqlite3_reset(pStmt); |
| if( rc!=SQLITE_OK ){ |
| sqlite3_free(a); |
| *pRC = rc; |
| return; |
| } |
| if( nChng<0 && a[0]<(u32)(-nChng) ){ |
| a[0] = 0; |
| }else{ |
| a[0] += nChng; |
| } |
| for(i=0; i<p->nColumn+1; i++){ |
| u32 x = a[i+1]; |
| if( x+aSzIns[i] < aSzDel[i] ){ |
| x = 0; |
| }else{ |
| x = x + aSzIns[i] - aSzDel[i]; |
| } |
| a[i+1] = x; |
| } |
| fts3EncodeIntArray(nStat, a, pBlob, &nBlob); |
| rc = fts3SqlStmt(p, SQL_REPLACE_STAT, &pStmt, 0); |
| if( rc ){ |
| sqlite3_free(a); |
| *pRC = rc; |
| return; |
| } |
| sqlite3_bind_int(pStmt, 1, FTS_STAT_DOCTOTAL); |
| sqlite3_bind_blob(pStmt, 2, pBlob, nBlob, SQLITE_STATIC); |
| sqlite3_step(pStmt); |
| *pRC = sqlite3_reset(pStmt); |
| sqlite3_bind_null(pStmt, 2); |
| sqlite3_free(a); |
| } |
|
|
| |
| |
| |
| |
| static int fts3DoOptimize(Fts3Table *p, int bReturnDone){ |
| int bSeenDone = 0; |
| int rc; |
| sqlite3_stmt *pAllLangid = 0; |
|
|
| rc = sqlite3Fts3PendingTermsFlush(p); |
| if( rc==SQLITE_OK ){ |
| rc = fts3SqlStmt(p, SQL_SELECT_ALL_LANGID, &pAllLangid, 0); |
| } |
| if( rc==SQLITE_OK ){ |
| int rc2; |
| sqlite3_bind_int(pAllLangid, 1, p->iPrevLangid); |
| sqlite3_bind_int(pAllLangid, 2, p->nIndex); |
| while( sqlite3_step(pAllLangid)==SQLITE_ROW ){ |
| int i; |
| int iLangid = sqlite3_column_int(pAllLangid, 0); |
| for(i=0; rc==SQLITE_OK && i<p->nIndex; i++){ |
| rc = fts3SegmentMerge(p, iLangid, i, FTS3_SEGCURSOR_ALL); |
| if( rc==SQLITE_DONE ){ |
| bSeenDone = 1; |
| rc = SQLITE_OK; |
| } |
| } |
| } |
| rc2 = sqlite3_reset(pAllLangid); |
| if( rc==SQLITE_OK ) rc = rc2; |
| } |
|
|
| sqlite3Fts3SegmentsClose(p); |
|
|
| return (rc==SQLITE_OK && bReturnDone && bSeenDone) ? SQLITE_DONE : rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3DoRebuild(Fts3Table *p){ |
| int rc; |
|
|
| rc = fts3DeleteAll(p, 0); |
| if( rc==SQLITE_OK ){ |
| u32 *aSz = 0; |
| u32 *aSzIns = 0; |
| u32 *aSzDel = 0; |
| sqlite3_stmt *pStmt = 0; |
| int nEntry = 0; |
|
|
| |
| char *zSql = sqlite3_mprintf("SELECT %s" , p->zReadExprlist); |
| if( !zSql ){ |
| rc = SQLITE_NOMEM; |
| }else{ |
| rc = sqlite3_prepare_v2(p->db, zSql, -1, &pStmt, 0); |
| sqlite3_free(zSql); |
| } |
|
|
| if( rc==SQLITE_OK ){ |
| sqlite3_int64 nByte = sizeof(u32) * ((sqlite3_int64)p->nColumn+1)*3; |
| aSz = (u32 *)sqlite3_malloc64(nByte); |
| if( aSz==0 ){ |
| rc = SQLITE_NOMEM; |
| }else{ |
| memset(aSz, 0, nByte); |
| aSzIns = &aSz[p->nColumn+1]; |
| aSzDel = &aSzIns[p->nColumn+1]; |
| } |
| } |
|
|
| while( rc==SQLITE_OK && SQLITE_ROW==sqlite3_step(pStmt) ){ |
| int iCol; |
| int iLangid = langidFromSelect(p, pStmt); |
| rc = fts3PendingTermsDocid(p, 0, iLangid, sqlite3_column_int64(pStmt, 0)); |
| memset(aSz, 0, sizeof(aSz[0]) * (p->nColumn+1)); |
| for(iCol=0; rc==SQLITE_OK && iCol<p->nColumn; iCol++){ |
| if( p->abNotindexed[iCol]==0 ){ |
| const char *z = (const char *) sqlite3_column_text(pStmt, iCol+1); |
| rc = fts3PendingTermsAdd(p, iLangid, z, iCol, &aSz[iCol]); |
| aSz[p->nColumn] += sqlite3_column_bytes(pStmt, iCol+1); |
| } |
| } |
| if( p->bHasDocsize ){ |
| fts3InsertDocsize(&rc, p, aSz); |
| } |
| if( rc!=SQLITE_OK ){ |
| sqlite3_finalize(pStmt); |
| pStmt = 0; |
| }else{ |
| nEntry++; |
| for(iCol=0; iCol<=p->nColumn; iCol++){ |
| aSzIns[iCol] += aSz[iCol]; |
| } |
| } |
| } |
| if( p->bFts4 ){ |
| fts3UpdateDocTotals(&rc, p, aSzIns, aSzDel, nEntry); |
| } |
| sqlite3_free(aSz); |
|
|
| if( pStmt ){ |
| int rc2 = sqlite3_finalize(pStmt); |
| if( rc==SQLITE_OK ){ |
| rc = rc2; |
| } |
| } |
| } |
|
|
| return rc; |
| } |
|
|
|
|
| |
| |
| |
| |
| |
| |
| static int fts3IncrmergeCsr( |
| Fts3Table *p, |
| sqlite3_int64 iAbsLevel, |
| int nSeg, |
| Fts3MultiSegReader *pCsr |
| ){ |
| int rc; |
| sqlite3_stmt *pStmt = 0; |
| sqlite3_int64 nByte; |
|
|
| |
| memset(pCsr, 0, sizeof(*pCsr)); |
| nByte = sizeof(Fts3SegReader *) * nSeg; |
| pCsr->apSegment = (Fts3SegReader **)sqlite3_malloc64(nByte); |
|
|
| if( pCsr->apSegment==0 ){ |
| rc = SQLITE_NOMEM; |
| }else{ |
| memset(pCsr->apSegment, 0, nByte); |
| rc = fts3SqlStmt(p, SQL_SELECT_LEVEL, &pStmt, 0); |
| } |
| if( rc==SQLITE_OK ){ |
| int i; |
| int rc2; |
| sqlite3_bind_int64(pStmt, 1, iAbsLevel); |
| assert( pCsr->nSegment==0 ); |
| for(i=0; rc==SQLITE_OK && sqlite3_step(pStmt)==SQLITE_ROW && i<nSeg; i++){ |
| rc = sqlite3Fts3SegReaderNew(i, 0, |
| sqlite3_column_int64(pStmt, 1), |
| sqlite3_column_int64(pStmt, 2), |
| sqlite3_column_int64(pStmt, 3), |
| sqlite3_column_blob(pStmt, 4), |
| sqlite3_column_bytes(pStmt, 4), |
| &pCsr->apSegment[i] |
| ); |
| pCsr->nSegment++; |
| } |
| rc2 = sqlite3_reset(pStmt); |
| if( rc==SQLITE_OK ) rc = rc2; |
| } |
|
|
| return rc; |
| } |
|
|
| typedef struct IncrmergeWriter IncrmergeWriter; |
| typedef struct NodeWriter NodeWriter; |
| typedef struct Blob Blob; |
| typedef struct NodeReader NodeReader; |
|
|
| |
| |
| |
| |
| |
| |
| struct Blob { |
| char *a; |
| int n; |
| int nAlloc; |
| }; |
|
|
| |
| |
| |
| |
| struct NodeWriter { |
| sqlite3_int64 iBlock; |
| Blob key; |
| Blob block; |
| }; |
|
|
| |
| |
| |
| |
| struct IncrmergeWriter { |
| i64 nLeafEst; |
| i64 nWork; |
| sqlite3_int64 iAbsLevel; |
| int iIdx; |
| sqlite3_int64 iStart; |
| sqlite3_int64 iEnd; |
| sqlite3_int64 nLeafData; |
| u8 bNoLeafData; |
| NodeWriter aNodeWriter[FTS_MAX_APPENDABLE_HEIGHT]; |
| }; |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| struct NodeReader { |
| const char *aNode; |
| int nNode; |
| int iOff; |
|
|
| |
| sqlite3_int64 iChild; |
| Blob term; |
| const char *aDoclist; |
| int nDoclist; |
| }; |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static void blobGrowBuffer(Blob *pBlob, int nMin, int *pRc){ |
| if( *pRc==SQLITE_OK && nMin>pBlob->nAlloc ){ |
| int nAlloc = nMin; |
| char *a = (char *)sqlite3_realloc64(pBlob->a, nAlloc); |
| if( a ){ |
| pBlob->nAlloc = nAlloc; |
| pBlob->a = a; |
| }else{ |
| *pRc = SQLITE_NOMEM; |
| } |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int nodeReaderNext(NodeReader *p){ |
| int bFirst = (p->term.n==0); |
| int nPrefix = 0; |
| int nSuffix = 0; |
| int rc = SQLITE_OK; |
|
|
| assert( p->aNode ); |
| if( p->iChild && bFirst==0 ) p->iChild++; |
| if( p->iOff>=p->nNode ){ |
| |
| p->aNode = 0; |
| }else{ |
| if( bFirst==0 ){ |
| p->iOff += fts3GetVarint32(&p->aNode[p->iOff], &nPrefix); |
| } |
| p->iOff += fts3GetVarint32(&p->aNode[p->iOff], &nSuffix); |
|
|
| if( nPrefix>p->term.n || nSuffix>p->nNode-p->iOff || nSuffix==0 ){ |
| return FTS_CORRUPT_VTAB; |
| } |
| blobGrowBuffer(&p->term, nPrefix+nSuffix, &rc); |
| if( rc==SQLITE_OK && ALWAYS(p->term.a!=0) ){ |
| memcpy(&p->term.a[nPrefix], &p->aNode[p->iOff], nSuffix); |
| p->term.n = nPrefix+nSuffix; |
| p->iOff += nSuffix; |
| if( p->iChild==0 ){ |
| p->iOff += fts3GetVarint32(&p->aNode[p->iOff], &p->nDoclist); |
| if( (p->nNode-p->iOff)<p->nDoclist ){ |
| return FTS_CORRUPT_VTAB; |
| } |
| p->aDoclist = &p->aNode[p->iOff]; |
| p->iOff += p->nDoclist; |
| } |
| } |
| } |
|
|
| assert_fts3_nc( p->iOff<=p->nNode ); |
| return rc; |
| } |
|
|
| |
| |
| |
| static void nodeReaderRelease(NodeReader *p){ |
| sqlite3_free(p->term.a); |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| static int nodeReaderInit(NodeReader *p, const char *aNode, int nNode){ |
| memset(p, 0, sizeof(NodeReader)); |
| p->aNode = aNode; |
| p->nNode = nNode; |
|
|
| |
| if( aNode && aNode[0] ){ |
| |
| p->iOff = 1 + sqlite3Fts3GetVarint(&p->aNode[1], &p->iChild); |
| }else{ |
| p->iOff = 1; |
| } |
|
|
| return aNode ? nodeReaderNext(p) : SQLITE_OK; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3IncrmergePush( |
| Fts3Table *p, |
| IncrmergeWriter *pWriter, |
| const char *zTerm, |
| int nTerm |
| ){ |
| sqlite3_int64 iPtr = pWriter->aNodeWriter[0].iBlock; |
| int iLayer; |
|
|
| assert( nTerm>0 ); |
| for(iLayer=1; ALWAYS(iLayer<FTS_MAX_APPENDABLE_HEIGHT); iLayer++){ |
| sqlite3_int64 iNextPtr = 0; |
| NodeWriter *pNode = &pWriter->aNodeWriter[iLayer]; |
| int rc = SQLITE_OK; |
| int nPrefix; |
| int nSuffix; |
| int nSpace; |
|
|
| |
| |
| |
| |
| nPrefix = fts3PrefixCompress(pNode->key.a, pNode->key.n, zTerm, nTerm); |
| nSuffix = nTerm - nPrefix; |
| if(nSuffix<=0 ) return FTS_CORRUPT_VTAB; |
| nSpace = sqlite3Fts3VarintLen(nPrefix); |
| nSpace += sqlite3Fts3VarintLen(nSuffix) + nSuffix; |
|
|
| if( pNode->key.n==0 || (pNode->block.n + nSpace)<=p->nNodeSize ){ |
| |
| |
| |
|
|
| Blob *pBlk = &pNode->block; |
| if( pBlk->n==0 ){ |
| blobGrowBuffer(pBlk, p->nNodeSize, &rc); |
| if( rc==SQLITE_OK ){ |
| pBlk->a[0] = (char)iLayer; |
| pBlk->n = 1 + sqlite3Fts3PutVarint(&pBlk->a[1], iPtr); |
| } |
| } |
| blobGrowBuffer(pBlk, pBlk->n + nSpace, &rc); |
| blobGrowBuffer(&pNode->key, nTerm, &rc); |
|
|
| if( rc==SQLITE_OK ){ |
| if( pNode->key.n ){ |
| pBlk->n += sqlite3Fts3PutVarint(&pBlk->a[pBlk->n], nPrefix); |
| } |
| pBlk->n += sqlite3Fts3PutVarint(&pBlk->a[pBlk->n], nSuffix); |
| assert( nPrefix+nSuffix<=nTerm ); |
| assert( nPrefix>=0 ); |
| memcpy(&pBlk->a[pBlk->n], &zTerm[nPrefix], nSuffix); |
| pBlk->n += nSuffix; |
|
|
| memcpy(pNode->key.a, zTerm, nTerm); |
| pNode->key.n = nTerm; |
| } |
| }else{ |
| |
| |
| |
| rc = fts3WriteSegment(p, pNode->iBlock, pNode->block.a, pNode->block.n); |
|
|
| assert( pNode->block.nAlloc>=p->nNodeSize ); |
| pNode->block.a[0] = (char)iLayer; |
| pNode->block.n = 1 + sqlite3Fts3PutVarint(&pNode->block.a[1], iPtr+1); |
|
|
| iNextPtr = pNode->iBlock; |
| pNode->iBlock++; |
| pNode->key.n = 0; |
| } |
|
|
| if( rc!=SQLITE_OK || iNextPtr==0 ) return rc; |
| iPtr = iNextPtr; |
| } |
|
|
| assert( 0 ); |
| return 0; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3AppendToNode( |
| Blob *pNode, |
| Blob *pPrev, |
| const char *zTerm, |
| int nTerm, |
| const char *aDoclist, |
| int nDoclist |
| ){ |
| int rc = SQLITE_OK; |
| int bFirst = (pPrev->n==0); |
| int nPrefix; |
| int nSuffix; |
|
|
| |
| |
| assert( pNode->n>0 ); |
| assert_fts3_nc( (pNode->a[0]=='\0')==(aDoclist!=0) ); |
|
|
| blobGrowBuffer(pPrev, nTerm, &rc); |
| if( rc!=SQLITE_OK ) return rc; |
| assert( pPrev!=0 ); |
| assert( pPrev->a!=0 ); |
|
|
| nPrefix = fts3PrefixCompress(pPrev->a, pPrev->n, zTerm, nTerm); |
| nSuffix = nTerm - nPrefix; |
| if( nSuffix<=0 ) return FTS_CORRUPT_VTAB; |
| memcpy(pPrev->a, zTerm, nTerm); |
| pPrev->n = nTerm; |
|
|
| if( bFirst==0 ){ |
| pNode->n += sqlite3Fts3PutVarint(&pNode->a[pNode->n], nPrefix); |
| } |
| pNode->n += sqlite3Fts3PutVarint(&pNode->a[pNode->n], nSuffix); |
| memcpy(&pNode->a[pNode->n], &zTerm[nPrefix], nSuffix); |
| pNode->n += nSuffix; |
|
|
| if( aDoclist ){ |
| pNode->n += sqlite3Fts3PutVarint(&pNode->a[pNode->n], nDoclist); |
| memcpy(&pNode->a[pNode->n], aDoclist, nDoclist); |
| pNode->n += nDoclist; |
| } |
|
|
| assert( pNode->n<=pNode->nAlloc ); |
|
|
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| static int fts3IncrmergeAppend( |
| Fts3Table *p, |
| IncrmergeWriter *pWriter, |
| Fts3MultiSegReader *pCsr |
| ){ |
| const char *zTerm = pCsr->zTerm; |
| int nTerm = pCsr->nTerm; |
| const char *aDoclist = pCsr->aDoclist; |
| int nDoclist = pCsr->nDoclist; |
| int rc = SQLITE_OK; |
| int nSpace; |
| int nPrefix; |
| int nSuffix; |
| NodeWriter *pLeaf; |
|
|
| pLeaf = &pWriter->aNodeWriter[0]; |
| nPrefix = fts3PrefixCompress(pLeaf->key.a, pLeaf->key.n, zTerm, nTerm); |
| nSuffix = nTerm - nPrefix; |
| if(nSuffix<=0 ) return FTS_CORRUPT_VTAB; |
|
|
| nSpace = sqlite3Fts3VarintLen(nPrefix); |
| nSpace += sqlite3Fts3VarintLen(nSuffix) + nSuffix; |
| nSpace += sqlite3Fts3VarintLen(nDoclist) + nDoclist; |
|
|
| |
| |
| |
| |
| if( pLeaf->block.n>0 |
| && (pLeaf->block.n + nSpace)>p->nNodeSize |
| && pLeaf->iBlock < (pWriter->iStart + pWriter->nLeafEst) |
| ){ |
| rc = fts3WriteSegment(p, pLeaf->iBlock, pLeaf->block.a, pLeaf->block.n); |
| pWriter->nWork++; |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| if( rc==SQLITE_OK ){ |
| rc = fts3IncrmergePush(p, pWriter, zTerm, nPrefix+1); |
| } |
|
|
| |
| pLeaf->iBlock++; |
| pLeaf->key.n = 0; |
| pLeaf->block.n = 0; |
|
|
| nSuffix = nTerm; |
| nSpace = 1; |
| nSpace += sqlite3Fts3VarintLen(nSuffix) + nSuffix; |
| nSpace += sqlite3Fts3VarintLen(nDoclist) + nDoclist; |
| } |
|
|
| pWriter->nLeafData += nSpace; |
| blobGrowBuffer(&pLeaf->block, pLeaf->block.n + nSpace, &rc); |
| if( rc==SQLITE_OK ){ |
| if( pLeaf->block.n==0 ){ |
| pLeaf->block.n = 1; |
| pLeaf->block.a[0] = '\0'; |
| } |
| rc = fts3AppendToNode( |
| &pLeaf->block, &pLeaf->key, zTerm, nTerm, aDoclist, nDoclist |
| ); |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static void fts3IncrmergeRelease( |
| Fts3Table *p, |
| IncrmergeWriter *pWriter, |
| int *pRc |
| ){ |
| int i; |
| int iRoot; |
| NodeWriter *pRoot; |
| int rc = *pRc; |
|
|
| |
| |
| |
| |
| for(iRoot=FTS_MAX_APPENDABLE_HEIGHT-1; iRoot>=0; iRoot--){ |
| NodeWriter *pNode = &pWriter->aNodeWriter[iRoot]; |
| if( pNode->block.n>0 ) break; |
| assert( *pRc || pNode->block.nAlloc==0 ); |
| assert( *pRc || pNode->key.nAlloc==0 ); |
| sqlite3_free(pNode->block.a); |
| sqlite3_free(pNode->key.a); |
| } |
|
|
| |
| if( iRoot<0 ) return; |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| if( iRoot==0 ){ |
| Blob *pBlock = &pWriter->aNodeWriter[1].block; |
| blobGrowBuffer(pBlock, 1 + FTS3_VARINT_MAX, &rc); |
| if( rc==SQLITE_OK ){ |
| pBlock->a[0] = 0x01; |
| pBlock->n = 1 + sqlite3Fts3PutVarint( |
| &pBlock->a[1], pWriter->aNodeWriter[0].iBlock |
| ); |
| } |
| iRoot = 1; |
| } |
| pRoot = &pWriter->aNodeWriter[iRoot]; |
|
|
| |
| for(i=0; i<iRoot; i++){ |
| NodeWriter *pNode = &pWriter->aNodeWriter[i]; |
| if( pNode->block.n>0 && rc==SQLITE_OK ){ |
| rc = fts3WriteSegment(p, pNode->iBlock, pNode->block.a, pNode->block.n); |
| } |
| sqlite3_free(pNode->block.a); |
| sqlite3_free(pNode->key.a); |
| } |
|
|
| |
| if( rc==SQLITE_OK ){ |
| rc = fts3WriteSegdir(p, |
| pWriter->iAbsLevel+1, |
| pWriter->iIdx, |
| pWriter->iStart, |
| pWriter->aNodeWriter[0].iBlock, |
| pWriter->iEnd, |
| (pWriter->bNoLeafData==0 ? pWriter->nLeafData : 0), |
| pRoot->block.a, pRoot->block.n |
| ); |
| } |
| sqlite3_free(pRoot->block.a); |
| sqlite3_free(pRoot->key.a); |
|
|
| *pRc = rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3TermCmp( |
| const char *zLhs, int nLhs, |
| const char *zRhs, int nRhs |
| ){ |
| int nCmp = MIN(nLhs, nRhs); |
| int res; |
|
|
| if( nCmp && ALWAYS(zLhs) && ALWAYS(zRhs) ){ |
| res = memcmp(zLhs, zRhs, nCmp); |
| }else{ |
| res = 0; |
| } |
| if( res==0 ) res = nLhs - nRhs; |
|
|
| return res; |
| } |
|
|
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3IsAppendable(Fts3Table *p, sqlite3_int64 iEnd, int *pbRes){ |
| int bRes = 0; |
| sqlite3_stmt *pCheck = 0; |
| int rc; |
|
|
| rc = fts3SqlStmt(p, SQL_SEGMENT_IS_APPENDABLE, &pCheck, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pCheck, 1, iEnd); |
| if( SQLITE_ROW==sqlite3_step(pCheck) ) bRes = 1; |
| rc = sqlite3_reset(pCheck); |
| } |
| |
| *pbRes = bRes; |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3IncrmergeLoad( |
| Fts3Table *p, |
| sqlite3_int64 iAbsLevel, |
| int iIdx, |
| const char *zKey, |
| int nKey, |
| IncrmergeWriter *pWriter |
| ){ |
| int rc; |
| sqlite3_stmt *pSelect = 0; |
|
|
| rc = fts3SqlStmt(p, SQL_SELECT_SEGDIR, &pSelect, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_int64 iStart = 0; |
| sqlite3_int64 iLeafEnd = 0; |
| sqlite3_int64 iEnd = 0; |
| const char *aRoot = 0; |
| int nRoot = 0; |
| int rc2; |
| int bAppendable = 0; |
|
|
| |
| sqlite3_bind_int64(pSelect, 1, iAbsLevel+1); |
| sqlite3_bind_int(pSelect, 2, iIdx); |
| if( sqlite3_step(pSelect)==SQLITE_ROW ){ |
| iStart = sqlite3_column_int64(pSelect, 1); |
| iLeafEnd = sqlite3_column_int64(pSelect, 2); |
| fts3ReadEndBlockField(pSelect, 3, &iEnd, &pWriter->nLeafData); |
| if( pWriter->nLeafData<0 ){ |
| pWriter->nLeafData = pWriter->nLeafData * -1; |
| } |
| pWriter->bNoLeafData = (pWriter->nLeafData==0); |
| nRoot = sqlite3_column_bytes(pSelect, 4); |
| aRoot = sqlite3_column_blob(pSelect, 4); |
| if( aRoot==0 ){ |
| sqlite3_reset(pSelect); |
| return nRoot ? SQLITE_NOMEM : FTS_CORRUPT_VTAB; |
| } |
| }else{ |
| return sqlite3_reset(pSelect); |
| } |
|
|
| |
| rc = fts3IsAppendable(p, iEnd, &bAppendable); |
|
|
| |
| if( rc==SQLITE_OK && bAppendable ){ |
| char *aLeaf = 0; |
| int nLeaf = 0; |
|
|
| rc = sqlite3Fts3ReadBlock(p, iLeafEnd, &aLeaf, &nLeaf, 0); |
| if( rc==SQLITE_OK ){ |
| NodeReader reader; |
| for(rc = nodeReaderInit(&reader, aLeaf, nLeaf); |
| rc==SQLITE_OK && reader.aNode; |
| rc = nodeReaderNext(&reader) |
| ){ |
| assert( reader.aNode ); |
| } |
| if( fts3TermCmp(zKey, nKey, reader.term.a, reader.term.n)<=0 ){ |
| bAppendable = 0; |
| } |
| nodeReaderRelease(&reader); |
| } |
| sqlite3_free(aLeaf); |
| } |
|
|
| if( rc==SQLITE_OK && bAppendable ){ |
| |
| |
| int i; |
| int nHeight = (int)aRoot[0]; |
| NodeWriter *pNode; |
| if( nHeight<1 || nHeight>=FTS_MAX_APPENDABLE_HEIGHT ){ |
| sqlite3_reset(pSelect); |
| return FTS_CORRUPT_VTAB; |
| } |
|
|
| pWriter->nLeafEst = (int)((iEnd - iStart) + 1)/FTS_MAX_APPENDABLE_HEIGHT; |
| pWriter->iStart = iStart; |
| pWriter->iEnd = iEnd; |
| pWriter->iAbsLevel = iAbsLevel; |
| pWriter->iIdx = iIdx; |
|
|
| for(i=nHeight+1; i<FTS_MAX_APPENDABLE_HEIGHT; i++){ |
| pWriter->aNodeWriter[i].iBlock = pWriter->iStart + i*pWriter->nLeafEst; |
| } |
|
|
| pNode = &pWriter->aNodeWriter[nHeight]; |
| pNode->iBlock = pWriter->iStart + pWriter->nLeafEst*nHeight; |
| blobGrowBuffer(&pNode->block, |
| MAX(nRoot, p->nNodeSize)+FTS3_NODE_PADDING, &rc |
| ); |
| if( rc==SQLITE_OK ){ |
| memcpy(pNode->block.a, aRoot, nRoot); |
| pNode->block.n = nRoot; |
| memset(&pNode->block.a[nRoot], 0, FTS3_NODE_PADDING); |
| } |
|
|
| for(i=nHeight; i>=0 && rc==SQLITE_OK; i--){ |
| NodeReader reader; |
| memset(&reader, 0, sizeof(reader)); |
| pNode = &pWriter->aNodeWriter[i]; |
|
|
| if( pNode->block.a){ |
| rc = nodeReaderInit(&reader, pNode->block.a, pNode->block.n); |
| while( reader.aNode && rc==SQLITE_OK ) rc = nodeReaderNext(&reader); |
| blobGrowBuffer(&pNode->key, reader.term.n, &rc); |
| if( rc==SQLITE_OK ){ |
| assert_fts3_nc( reader.term.n>0 || reader.aNode==0 ); |
| if( reader.term.n>0 ){ |
| memcpy(pNode->key.a, reader.term.a, reader.term.n); |
| } |
| pNode->key.n = reader.term.n; |
| if( i>0 ){ |
| char *aBlock = 0; |
| int nBlock = 0; |
| pNode = &pWriter->aNodeWriter[i-1]; |
| pNode->iBlock = reader.iChild; |
| rc = sqlite3Fts3ReadBlock(p, reader.iChild, &aBlock, &nBlock,0); |
| blobGrowBuffer(&pNode->block, |
| MAX(nBlock, p->nNodeSize)+FTS3_NODE_PADDING, &rc |
| ); |
| if( rc==SQLITE_OK ){ |
| memcpy(pNode->block.a, aBlock, nBlock); |
| pNode->block.n = nBlock; |
| memset(&pNode->block.a[nBlock], 0, FTS3_NODE_PADDING); |
| } |
| sqlite3_free(aBlock); |
| } |
| } |
| } |
| nodeReaderRelease(&reader); |
| } |
| } |
|
|
| rc2 = sqlite3_reset(pSelect); |
| if( rc==SQLITE_OK ) rc = rc2; |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3IncrmergeOutputIdx( |
| Fts3Table *p, |
| sqlite3_int64 iAbsLevel, |
| int *piIdx |
| ){ |
| int rc; |
| sqlite3_stmt *pOutputIdx = 0; |
|
|
| rc = fts3SqlStmt(p, SQL_NEXT_SEGMENT_INDEX, &pOutputIdx, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pOutputIdx, 1, iAbsLevel+1); |
| sqlite3_step(pOutputIdx); |
| *piIdx = sqlite3_column_int(pOutputIdx, 0); |
| rc = sqlite3_reset(pOutputIdx); |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3IncrmergeWriter( |
| Fts3Table *p, |
| sqlite3_int64 iAbsLevel, |
| int iIdx, |
| Fts3MultiSegReader *pCsr, |
| IncrmergeWriter *pWriter |
| ){ |
| int rc; |
| int i; |
| i64 nLeafEst = 0; |
| sqlite3_stmt *pLeafEst = 0; |
| sqlite3_stmt *pFirstBlock = 0; |
|
|
| |
| rc = fts3SqlStmt(p, SQL_MAX_LEAF_NODE_ESTIMATE, &pLeafEst, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pLeafEst, 1, iAbsLevel); |
| sqlite3_bind_int64(pLeafEst, 2, pCsr->nSegment); |
| if( SQLITE_ROW==sqlite3_step(pLeafEst) ){ |
| nLeafEst = sqlite3_column_int64(pLeafEst, 0); |
| } |
| rc = sqlite3_reset(pLeafEst); |
| } |
| if( rc!=SQLITE_OK ) return rc; |
|
|
| |
| rc = fts3SqlStmt(p, SQL_NEXT_SEGMENTS_ID, &pFirstBlock, 0); |
| if( rc==SQLITE_OK ){ |
| if( SQLITE_ROW==sqlite3_step(pFirstBlock) ){ |
| pWriter->iStart = sqlite3_column_int64(pFirstBlock, 0); |
| pWriter->iEnd = pWriter->iStart - 1; |
| pWriter->iEnd += nLeafEst * FTS_MAX_APPENDABLE_HEIGHT; |
| } |
| rc = sqlite3_reset(pFirstBlock); |
| } |
| if( rc!=SQLITE_OK ) return rc; |
|
|
| |
| |
| |
| rc = fts3WriteSegment(p, pWriter->iEnd, 0, 0); |
| if( rc!=SQLITE_OK ) return rc; |
|
|
| pWriter->iAbsLevel = iAbsLevel; |
| pWriter->nLeafEst = nLeafEst; |
| pWriter->iIdx = iIdx; |
|
|
| |
| for(i=0; i<FTS_MAX_APPENDABLE_HEIGHT; i++){ |
| pWriter->aNodeWriter[i].iBlock = pWriter->iStart + i*pWriter->nLeafEst; |
| } |
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3RemoveSegdirEntry( |
| Fts3Table *p, |
| sqlite3_int64 iAbsLevel, |
| int iIdx |
| ){ |
| int rc; |
| sqlite3_stmt *pDelete = 0; |
|
|
| rc = fts3SqlStmt(p, SQL_DELETE_SEGDIR_ENTRY, &pDelete, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pDelete, 1, iAbsLevel); |
| sqlite3_bind_int(pDelete, 2, iIdx); |
| sqlite3_step(pDelete); |
| rc = sqlite3_reset(pDelete); |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| static int fts3RepackSegdirLevel( |
| Fts3Table *p, |
| sqlite3_int64 iAbsLevel |
| ){ |
| int rc; |
| int *aIdx = 0; |
| int nIdx = 0; |
| int nAlloc = 0; |
| int i; |
| sqlite3_stmt *pSelect = 0; |
| sqlite3_stmt *pUpdate = 0; |
|
|
| rc = fts3SqlStmt(p, SQL_SELECT_INDEXES, &pSelect, 0); |
| if( rc==SQLITE_OK ){ |
| int rc2; |
| sqlite3_bind_int64(pSelect, 1, iAbsLevel); |
| while( SQLITE_ROW==sqlite3_step(pSelect) ){ |
| if( nIdx>=nAlloc ){ |
| int *aNew; |
| nAlloc += 16; |
| aNew = sqlite3_realloc64(aIdx, nAlloc*sizeof(int)); |
| if( !aNew ){ |
| rc = SQLITE_NOMEM; |
| break; |
| } |
| aIdx = aNew; |
| } |
| aIdx[nIdx++] = sqlite3_column_int(pSelect, 0); |
| } |
| rc2 = sqlite3_reset(pSelect); |
| if( rc==SQLITE_OK ) rc = rc2; |
| } |
|
|
| if( rc==SQLITE_OK ){ |
| rc = fts3SqlStmt(p, SQL_SHIFT_SEGDIR_ENTRY, &pUpdate, 0); |
| } |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pUpdate, 2, iAbsLevel); |
| } |
|
|
| assert( p->bIgnoreSavepoint==0 ); |
| p->bIgnoreSavepoint = 1; |
| for(i=0; rc==SQLITE_OK && i<nIdx; i++){ |
| if( aIdx[i]!=i ){ |
| sqlite3_bind_int(pUpdate, 3, aIdx[i]); |
| sqlite3_bind_int(pUpdate, 1, i); |
| sqlite3_step(pUpdate); |
| rc = sqlite3_reset(pUpdate); |
| } |
| } |
| p->bIgnoreSavepoint = 0; |
|
|
| sqlite3_free(aIdx); |
| return rc; |
| } |
|
|
| static void fts3StartNode(Blob *pNode, int iHeight, sqlite3_int64 iChild){ |
| pNode->a[0] = (char)iHeight; |
| if( iChild ){ |
| assert( pNode->nAlloc>=1+sqlite3Fts3VarintLen(iChild) ); |
| pNode->n = 1 + sqlite3Fts3PutVarint(&pNode->a[1], iChild); |
| }else{ |
| assert( pNode->nAlloc>=1 ); |
| pNode->n = 1; |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3TruncateNode( |
| const char *aNode, |
| int nNode, |
| Blob *pNew, |
| const char *zTerm, |
| int nTerm, |
| sqlite3_int64 *piBlock |
| ){ |
| NodeReader reader; |
| Blob prev = {0, 0, 0}; |
| int rc = SQLITE_OK; |
| int bLeaf; |
|
|
| if( nNode<1 ) return FTS_CORRUPT_VTAB; |
| bLeaf = aNode[0]=='\0'; |
|
|
| |
| blobGrowBuffer(pNew, nNode, &rc); |
| if( rc!=SQLITE_OK ) return rc; |
| pNew->n = 0; |
|
|
| |
| for(rc = nodeReaderInit(&reader, aNode, nNode); |
| rc==SQLITE_OK && reader.aNode; |
| rc = nodeReaderNext(&reader) |
| ){ |
| if( pNew->n==0 ){ |
| int res = fts3TermCmp(reader.term.a, reader.term.n, zTerm, nTerm); |
| if( res<0 || (bLeaf==0 && res==0) ) continue; |
| fts3StartNode(pNew, (int)aNode[0], reader.iChild); |
| *piBlock = reader.iChild; |
| } |
| rc = fts3AppendToNode( |
| pNew, &prev, reader.term.a, reader.term.n, |
| reader.aDoclist, reader.nDoclist |
| ); |
| if( rc!=SQLITE_OK ) break; |
| } |
| if( pNew->n==0 ){ |
| fts3StartNode(pNew, (int)aNode[0], reader.iChild); |
| *piBlock = reader.iChild; |
| } |
| assert( pNew->n<=pNew->nAlloc ); |
|
|
| nodeReaderRelease(&reader); |
| sqlite3_free(prev.a); |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3TruncateSegment( |
| Fts3Table *p, |
| sqlite3_int64 iAbsLevel, |
| int iIdx, |
| const char *zTerm, |
| int nTerm |
| ){ |
| int rc = SQLITE_OK; |
| Blob root = {0,0,0}; |
| Blob block = {0,0,0}; |
| sqlite3_int64 iBlock = 0; |
| sqlite3_int64 iNewStart = 0; |
| sqlite3_int64 iOldStart = 0; |
| sqlite3_stmt *pFetch = 0; |
|
|
| rc = fts3SqlStmt(p, SQL_SELECT_SEGDIR, &pFetch, 0); |
| if( rc==SQLITE_OK ){ |
| int rc2; |
| sqlite3_bind_int64(pFetch, 1, iAbsLevel); |
| sqlite3_bind_int(pFetch, 2, iIdx); |
| if( SQLITE_ROW==sqlite3_step(pFetch) ){ |
| const char *aRoot = sqlite3_column_blob(pFetch, 4); |
| int nRoot = sqlite3_column_bytes(pFetch, 4); |
| iOldStart = sqlite3_column_int64(pFetch, 1); |
| rc = fts3TruncateNode(aRoot, nRoot, &root, zTerm, nTerm, &iBlock); |
| } |
| rc2 = sqlite3_reset(pFetch); |
| if( rc==SQLITE_OK ) rc = rc2; |
| } |
|
|
| while( rc==SQLITE_OK && iBlock ){ |
| char *aBlock = 0; |
| int nBlock = 0; |
| iNewStart = iBlock; |
|
|
| rc = sqlite3Fts3ReadBlock(p, iBlock, &aBlock, &nBlock, 0); |
| if( rc==SQLITE_OK ){ |
| rc = fts3TruncateNode(aBlock, nBlock, &block, zTerm, nTerm, &iBlock); |
| } |
| if( rc==SQLITE_OK ){ |
| rc = fts3WriteSegment(p, iNewStart, block.a, block.n); |
| } |
| sqlite3_free(aBlock); |
| } |
|
|
| |
| if( rc==SQLITE_OK && iNewStart ){ |
| sqlite3_stmt *pDel = 0; |
| rc = fts3SqlStmt(p, SQL_DELETE_SEGMENTS_RANGE, &pDel, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pDel, 1, iOldStart); |
| sqlite3_bind_int64(pDel, 2, iNewStart-1); |
| sqlite3_step(pDel); |
| rc = sqlite3_reset(pDel); |
| } |
| } |
|
|
| if( rc==SQLITE_OK ){ |
| sqlite3_stmt *pChomp = 0; |
| rc = fts3SqlStmt(p, SQL_CHOMP_SEGDIR, &pChomp, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int64(pChomp, 1, iNewStart); |
| sqlite3_bind_blob(pChomp, 2, root.a, root.n, SQLITE_STATIC); |
| sqlite3_bind_int64(pChomp, 3, iAbsLevel); |
| sqlite3_bind_int(pChomp, 4, iIdx); |
| sqlite3_step(pChomp); |
| rc = sqlite3_reset(pChomp); |
| sqlite3_bind_null(pChomp, 2); |
| } |
| } |
|
|
| sqlite3_free(root.a); |
| sqlite3_free(block.a); |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3IncrmergeChomp( |
| Fts3Table *p, |
| sqlite3_int64 iAbsLevel, |
| Fts3MultiSegReader *pCsr, |
| int *pnRem |
| ){ |
| int i; |
| int nRem = 0; |
| int rc = SQLITE_OK; |
|
|
| for(i=pCsr->nSegment-1; i>=0 && rc==SQLITE_OK; i--){ |
| Fts3SegReader *pSeg = 0; |
| int j; |
|
|
| |
| |
| for(j=0; ALWAYS(j<pCsr->nSegment); j++){ |
| pSeg = pCsr->apSegment[j]; |
| if( pSeg->iIdx==i ) break; |
| } |
| assert( j<pCsr->nSegment && pSeg->iIdx==i ); |
|
|
| if( pSeg->aNode==0 ){ |
| |
| rc = fts3DeleteSegment(p, pSeg); |
| if( rc==SQLITE_OK ){ |
| rc = fts3RemoveSegdirEntry(p, iAbsLevel, pSeg->iIdx); |
| } |
| *pnRem = 0; |
| }else{ |
| |
| |
| |
| const char *zTerm = pSeg->zTerm; |
| int nTerm = pSeg->nTerm; |
| rc = fts3TruncateSegment(p, iAbsLevel, pSeg->iIdx, zTerm, nTerm); |
| nRem++; |
| } |
| } |
|
|
| if( rc==SQLITE_OK && nRem!=pCsr->nSegment ){ |
| rc = fts3RepackSegdirLevel(p, iAbsLevel); |
| } |
|
|
| *pnRem = nRem; |
| return rc; |
| } |
|
|
| |
| |
| |
| static int fts3IncrmergeHintStore(Fts3Table *p, Blob *pHint){ |
| sqlite3_stmt *pReplace = 0; |
| int rc; |
|
|
| rc = fts3SqlStmt(p, SQL_REPLACE_STAT, &pReplace, 0); |
| if( rc==SQLITE_OK ){ |
| sqlite3_bind_int(pReplace, 1, FTS_STAT_INCRMERGEHINT); |
| sqlite3_bind_blob(pReplace, 2, pHint->a, pHint->n, SQLITE_STATIC); |
| sqlite3_step(pReplace); |
| rc = sqlite3_reset(pReplace); |
| sqlite3_bind_null(pReplace, 2); |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3IncrmergeHintLoad(Fts3Table *p, Blob *pHint){ |
| sqlite3_stmt *pSelect = 0; |
| int rc; |
|
|
| pHint->n = 0; |
| rc = fts3SqlStmt(p, SQL_SELECT_STAT, &pSelect, 0); |
| if( rc==SQLITE_OK ){ |
| int rc2; |
| sqlite3_bind_int(pSelect, 1, FTS_STAT_INCRMERGEHINT); |
| if( SQLITE_ROW==sqlite3_step(pSelect) ){ |
| const char *aHint = sqlite3_column_blob(pSelect, 0); |
| int nHint = sqlite3_column_bytes(pSelect, 0); |
| if( aHint ){ |
| blobGrowBuffer(pHint, nHint, &rc); |
| if( rc==SQLITE_OK ){ |
| if( ALWAYS(pHint->a!=0) ) memcpy(pHint->a, aHint, nHint); |
| pHint->n = nHint; |
| } |
| } |
| } |
| rc2 = sqlite3_reset(pSelect); |
| if( rc==SQLITE_OK ) rc = rc2; |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static void fts3IncrmergeHintPush( |
| Blob *pHint, |
| i64 iAbsLevel, |
| int nInput, |
| int *pRc |
| ){ |
| blobGrowBuffer(pHint, pHint->n + 2*FTS3_VARINT_MAX, pRc); |
| if( *pRc==SQLITE_OK ){ |
| pHint->n += sqlite3Fts3PutVarint(&pHint->a[pHint->n], iAbsLevel); |
| pHint->n += sqlite3Fts3PutVarint(&pHint->a[pHint->n], (i64)nInput); |
| } |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3IncrmergeHintPop(Blob *pHint, i64 *piAbsLevel, int *pnInput){ |
| const int nHint = pHint->n; |
| int i; |
|
|
| i = pHint->n-1; |
| if( (pHint->a[i] & 0x80) ) return FTS_CORRUPT_VTAB; |
| while( i>0 && (pHint->a[i-1] & 0x80) ) i--; |
| if( i==0 ) return FTS_CORRUPT_VTAB; |
| i--; |
| while( i>0 && (pHint->a[i-1] & 0x80) ) i--; |
|
|
| pHint->n = i; |
| i += sqlite3Fts3GetVarint(&pHint->a[i], piAbsLevel); |
| i += fts3GetVarint32(&pHint->a[i], pnInput); |
| assert( i<=nHint ); |
| if( i!=nHint ) return FTS_CORRUPT_VTAB; |
|
|
| return SQLITE_OK; |
| } |
|
|
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| int sqlite3Fts3Incrmerge(Fts3Table *p, int nMerge, int nMin){ |
| int rc; |
| int nRem = nMerge; |
| Fts3MultiSegReader *pCsr; |
| Fts3SegFilter *pFilter; |
| IncrmergeWriter *pWriter; |
| int nSeg = 0; |
| sqlite3_int64 iAbsLevel = 0; |
| Blob hint = {0, 0, 0}; |
| int bDirtyHint = 0; |
|
|
| |
| const int nAlloc = sizeof(*pCsr) + sizeof(*pFilter) + sizeof(*pWriter); |
| pWriter = (IncrmergeWriter *)sqlite3_malloc64(nAlloc); |
| if( !pWriter ) return SQLITE_NOMEM; |
| pFilter = (Fts3SegFilter *)&pWriter[1]; |
| pCsr = (Fts3MultiSegReader *)&pFilter[1]; |
|
|
| rc = fts3IncrmergeHintLoad(p, &hint); |
| while( rc==SQLITE_OK && nRem>0 ){ |
| const i64 nMod = FTS3_SEGDIR_MAXLEVEL * p->nIndex; |
| sqlite3_stmt *pFindLevel = 0; |
| int bUseHint = 0; |
| int iIdx = 0; |
|
|
| |
| |
| |
| |
| |
| |
| rc = fts3SqlStmt(p, SQL_FIND_MERGE_LEVEL, &pFindLevel, 0); |
| sqlite3_bind_int(pFindLevel, 1, MAX(2, nMin)); |
| if( sqlite3_step(pFindLevel)==SQLITE_ROW ){ |
| iAbsLevel = sqlite3_column_int64(pFindLevel, 0); |
| nSeg = sqlite3_column_int(pFindLevel, 1); |
| assert( nSeg>=2 ); |
| }else{ |
| nSeg = -1; |
| } |
| rc = sqlite3_reset(pFindLevel); |
|
|
| |
| |
| |
| |
| |
| if( rc==SQLITE_OK && hint.n ){ |
| int nHint = hint.n; |
| sqlite3_int64 iHintAbsLevel = 0; |
| int nHintSeg = 0; |
|
|
| rc = fts3IncrmergeHintPop(&hint, &iHintAbsLevel, &nHintSeg); |
| if( nSeg<0 || (iAbsLevel % nMod) >= (iHintAbsLevel % nMod) ){ |
| |
| |
| |
| |
| |
| |
| iAbsLevel = iHintAbsLevel; |
| nSeg = MIN(MAX(nMin,nSeg), nHintSeg); |
| bUseHint = 1; |
| bDirtyHint = 1; |
| }else{ |
| |
| |
| hint.n = nHint; |
| } |
| } |
|
|
| |
| |
| |
| if( nSeg<=0 ) break; |
|
|
| assert( nMod<=0x7FFFFFFF ); |
| if( iAbsLevel<0 || iAbsLevel>(nMod<<32) ){ |
| rc = FTS_CORRUPT_VTAB; |
| break; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| memset(pWriter, 0, nAlloc); |
| pFilter->flags = FTS3_SEGMENT_REQUIRE_POS; |
|
|
| if( rc==SQLITE_OK ){ |
| rc = fts3IncrmergeOutputIdx(p, iAbsLevel, &iIdx); |
| assert( bUseHint==1 || bUseHint==0 ); |
| if( iIdx==0 || (bUseHint && iIdx==1) ){ |
| int bIgnore = 0; |
| rc = fts3SegmentIsMaxLevel(p, iAbsLevel+1, &bIgnore); |
| if( bIgnore ){ |
| pFilter->flags |= FTS3_SEGMENT_IGNORE_EMPTY; |
| } |
| } |
| } |
|
|
| if( rc==SQLITE_OK ){ |
| rc = fts3IncrmergeCsr(p, iAbsLevel, nSeg, pCsr); |
| } |
| if( SQLITE_OK==rc && pCsr->nSegment==nSeg |
| && SQLITE_OK==(rc = sqlite3Fts3SegReaderStart(p, pCsr, pFilter)) |
| ){ |
| int bEmpty = 0; |
| rc = sqlite3Fts3SegReaderStep(p, pCsr); |
| if( rc==SQLITE_OK ){ |
| bEmpty = 1; |
| }else if( rc!=SQLITE_ROW ){ |
| sqlite3Fts3SegReaderFinish(pCsr); |
| break; |
| } |
| if( bUseHint && iIdx>0 ){ |
| const char *zKey = pCsr->zTerm; |
| int nKey = pCsr->nTerm; |
| rc = fts3IncrmergeLoad(p, iAbsLevel, iIdx-1, zKey, nKey, pWriter); |
| }else{ |
| rc = fts3IncrmergeWriter(p, iAbsLevel, iIdx, pCsr, pWriter); |
| } |
|
|
| if( rc==SQLITE_OK && pWriter->nLeafEst ){ |
| fts3LogMerge(nSeg, iAbsLevel); |
| if( bEmpty==0 ){ |
| do { |
| rc = fts3IncrmergeAppend(p, pWriter, pCsr); |
| if( rc==SQLITE_OK ) rc = sqlite3Fts3SegReaderStep(p, pCsr); |
| if( pWriter->nWork>=nRem && rc==SQLITE_ROW ) rc = SQLITE_OK; |
| }while( rc==SQLITE_ROW ); |
| } |
|
|
| |
| if( rc==SQLITE_OK ){ |
| nRem -= (1 + pWriter->nWork); |
| rc = fts3IncrmergeChomp(p, iAbsLevel, pCsr, &nSeg); |
| if( nSeg!=0 ){ |
| bDirtyHint = 1; |
| fts3IncrmergeHintPush(&hint, iAbsLevel, nSeg, &rc); |
| } |
| } |
| } |
|
|
| if( nSeg!=0 ){ |
| pWriter->nLeafData = pWriter->nLeafData * -1; |
| } |
| fts3IncrmergeRelease(p, pWriter, &rc); |
| if( nSeg==0 && pWriter->bNoLeafData==0 ){ |
| fts3PromoteSegments(p, iAbsLevel+1, pWriter->nLeafData); |
| } |
| } |
|
|
| sqlite3Fts3SegReaderFinish(pCsr); |
| } |
|
|
| |
| if( bDirtyHint && rc==SQLITE_OK ){ |
| rc = fts3IncrmergeHintStore(p, &hint); |
| } |
|
|
| sqlite3_free(pWriter); |
| sqlite3_free(hint.a); |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3Getint(const char **pz){ |
| const char *z = *pz; |
| int i = 0; |
| while( (*z)>='0' && (*z)<='9' && i<214748363 ) i = 10*i + *(z++) - '0'; |
| *pz = z; |
| return i; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3DoIncrmerge( |
| Fts3Table *p, |
| const char *zParam |
| ){ |
| int rc; |
| int nMin = (MergeCount(p) / 2); |
| int nMerge = 0; |
| const char *z = zParam; |
|
|
| |
| nMerge = fts3Getint(&z); |
|
|
| |
| |
| if( z[0]==',' && z[1]!='\0' ){ |
| z++; |
| nMin = fts3Getint(&z); |
| } |
|
|
| if( z[0]!='\0' || nMin<2 ){ |
| rc = SQLITE_ERROR; |
| }else{ |
| rc = SQLITE_OK; |
| if( !p->bHasStat ){ |
| assert( p->bFts4==0 ); |
| sqlite3Fts3CreateStatTable(&rc, p); |
| } |
| if( rc==SQLITE_OK ){ |
| rc = sqlite3Fts3Incrmerge(p, nMerge, nMin); |
| } |
| sqlite3Fts3SegmentsClose(p); |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3DoAutoincrmerge( |
| Fts3Table *p, |
| const char *zParam |
| ){ |
| int rc = SQLITE_OK; |
| sqlite3_stmt *pStmt = 0; |
| p->nAutoincrmerge = fts3Getint(&zParam); |
| if( p->nAutoincrmerge==1 || p->nAutoincrmerge>MergeCount(p) ){ |
| p->nAutoincrmerge = 8; |
| } |
| if( !p->bHasStat ){ |
| assert( p->bFts4==0 ); |
| sqlite3Fts3CreateStatTable(&rc, p); |
| if( rc ) return rc; |
| } |
| rc = fts3SqlStmt(p, SQL_REPLACE_STAT, &pStmt, 0); |
| if( rc ) return rc; |
| sqlite3_bind_int(pStmt, 1, FTS_STAT_AUTOINCRMERGE); |
| sqlite3_bind_int(pStmt, 2, p->nAutoincrmerge); |
| sqlite3_step(pStmt); |
| rc = sqlite3_reset(pStmt); |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| static u64 fts3ChecksumEntry( |
| const char *zTerm, |
| int nTerm, |
| int iLangid, |
| int iIndex, |
| i64 iDocid, |
| int iCol, |
| int iPos |
| ){ |
| int i; |
| u64 ret = (u64)iDocid; |
|
|
| ret += (ret<<3) + iLangid; |
| ret += (ret<<3) + iIndex; |
| ret += (ret<<3) + iCol; |
| ret += (ret<<3) + iPos; |
| for(i=0; i<nTerm; i++) ret += (ret<<3) + zTerm[i]; |
|
|
| return ret; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static u64 fts3ChecksumIndex( |
| Fts3Table *p, |
| int iLangid, |
| int iIndex, |
| int *pRc |
| ){ |
| Fts3SegFilter filter; |
| Fts3MultiSegReader csr; |
| int rc; |
| u64 cksum = 0; |
|
|
| if( *pRc ) return 0; |
|
|
| memset(&filter, 0, sizeof(filter)); |
| memset(&csr, 0, sizeof(csr)); |
| filter.flags = FTS3_SEGMENT_REQUIRE_POS|FTS3_SEGMENT_IGNORE_EMPTY; |
| filter.flags |= FTS3_SEGMENT_SCAN; |
|
|
| rc = sqlite3Fts3SegReaderCursor( |
| p, iLangid, iIndex, FTS3_SEGCURSOR_ALL, 0, 0, 0, 1,&csr |
| ); |
| if( rc==SQLITE_OK ){ |
| rc = sqlite3Fts3SegReaderStart(p, &csr, &filter); |
| } |
|
|
| if( rc==SQLITE_OK ){ |
| while( SQLITE_ROW==(rc = sqlite3Fts3SegReaderStep(p, &csr)) ){ |
| char *pCsr = csr.aDoclist; |
| char *pEnd = &pCsr[csr.nDoclist]; |
|
|
| i64 iDocid = 0; |
| i64 iCol = 0; |
| u64 iPos = 0; |
|
|
| pCsr += sqlite3Fts3GetVarint(pCsr, &iDocid); |
| while( pCsr<pEnd ){ |
| u64 iVal = 0; |
| pCsr += sqlite3Fts3GetVarintU(pCsr, &iVal); |
| if( pCsr<pEnd ){ |
| if( iVal==0 || iVal==1 ){ |
| iCol = 0; |
| iPos = 0; |
| if( iVal ){ |
| pCsr += sqlite3Fts3GetVarint(pCsr, &iCol); |
| }else{ |
| pCsr += sqlite3Fts3GetVarintU(pCsr, &iVal); |
| if( p->bDescIdx ){ |
| iDocid = (i64)((u64)iDocid - iVal); |
| }else{ |
| iDocid = (i64)((u64)iDocid + iVal); |
| } |
| } |
| }else{ |
| iPos += (iVal - 2); |
| cksum = cksum ^ fts3ChecksumEntry( |
| csr.zTerm, csr.nTerm, iLangid, iIndex, iDocid, |
| (int)iCol, (int)iPos |
| ); |
| } |
| } |
| } |
| } |
| } |
| sqlite3Fts3SegReaderFinish(&csr); |
|
|
| *pRc = rc; |
| return cksum; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| int sqlite3Fts3IntegrityCheck(Fts3Table *p, int *pbOk){ |
| int rc = SQLITE_OK; |
| u64 cksum1 = 0; |
| u64 cksum2 = 0; |
| sqlite3_stmt *pAllLangid = 0; |
|
|
| |
| rc = fts3SqlStmt(p, SQL_SELECT_ALL_LANGID, &pAllLangid, 0); |
| if( rc==SQLITE_OK ){ |
| int rc2; |
| sqlite3_bind_int(pAllLangid, 1, p->iPrevLangid); |
| sqlite3_bind_int(pAllLangid, 2, p->nIndex); |
| while( rc==SQLITE_OK && sqlite3_step(pAllLangid)==SQLITE_ROW ){ |
| int iLangid = sqlite3_column_int(pAllLangid, 0); |
| int i; |
| for(i=0; i<p->nIndex; i++){ |
| cksum1 = cksum1 ^ fts3ChecksumIndex(p, iLangid, i, &rc); |
| } |
| } |
| rc2 = sqlite3_reset(pAllLangid); |
| if( rc==SQLITE_OK ) rc = rc2; |
| } |
|
|
| |
| if( rc==SQLITE_OK ){ |
| sqlite3_tokenizer_module const *pModule = p->pTokenizer->pModule; |
| sqlite3_stmt *pStmt = 0; |
| char *zSql; |
| |
| zSql = sqlite3_mprintf("SELECT %s" , p->zReadExprlist); |
| if( !zSql ){ |
| rc = SQLITE_NOMEM; |
| }else{ |
| rc = sqlite3_prepare_v2(p->db, zSql, -1, &pStmt, 0); |
| sqlite3_free(zSql); |
| } |
|
|
| while( rc==SQLITE_OK && SQLITE_ROW==sqlite3_step(pStmt) ){ |
| i64 iDocid = sqlite3_column_int64(pStmt, 0); |
| int iLang = langidFromSelect(p, pStmt); |
| int iCol; |
|
|
| for(iCol=0; rc==SQLITE_OK && iCol<p->nColumn; iCol++){ |
| if( p->abNotindexed[iCol]==0 ){ |
| const char *zText = (const char *)sqlite3_column_text(pStmt, iCol+1); |
| sqlite3_tokenizer_cursor *pT = 0; |
|
|
| rc = sqlite3Fts3OpenTokenizer(p->pTokenizer, iLang, zText, -1, &pT); |
| while( rc==SQLITE_OK ){ |
| char const *zToken; |
| int nToken = 0; |
| int iDum1 = 0, iDum2 = 0; |
| int iPos = 0; |
|
|
| rc = pModule->xNext(pT, &zToken, &nToken, &iDum1, &iDum2, &iPos); |
| if( rc==SQLITE_OK ){ |
| int i; |
| cksum2 = cksum2 ^ fts3ChecksumEntry( |
| zToken, nToken, iLang, 0, iDocid, iCol, iPos |
| ); |
| for(i=1; i<p->nIndex; i++){ |
| if( p->aIndex[i].nPrefix<=nToken ){ |
| cksum2 = cksum2 ^ fts3ChecksumEntry( |
| zToken, p->aIndex[i].nPrefix, iLang, i, iDocid, iCol, iPos |
| ); |
| } |
| } |
| } |
| } |
| if( pT ) pModule->xClose(pT); |
| if( rc==SQLITE_DONE ) rc = SQLITE_OK; |
| } |
| } |
| } |
|
|
| sqlite3_finalize(pStmt); |
| } |
|
|
| if( rc==SQLITE_CORRUPT_VTAB ){ |
| rc = SQLITE_OK; |
| *pbOk = 0; |
| }else{ |
| *pbOk = (rc==SQLITE_OK && cksum1==cksum2); |
| } |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3DoIntegrityCheck( |
| Fts3Table *p |
| ){ |
| int rc; |
| int bOk = 0; |
| rc = sqlite3Fts3IntegrityCheck(p, &bOk); |
| if( rc==SQLITE_OK && bOk==0 ) rc = FTS_CORRUPT_VTAB; |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| static int fts3SpecialInsert(Fts3Table *p, sqlite3_value *pVal){ |
| int rc = SQLITE_ERROR; |
| const char *zVal = (const char *)sqlite3_value_text(pVal); |
| int nVal = sqlite3_value_bytes(pVal); |
|
|
| if( !zVal ){ |
| return SQLITE_NOMEM; |
| }else if( nVal==8 && 0==sqlite3_strnicmp(zVal, "optimize", 8) ){ |
| rc = fts3DoOptimize(p, 0); |
| }else if( nVal==7 && 0==sqlite3_strnicmp(zVal, "rebuild", 7) ){ |
| rc = fts3DoRebuild(p); |
| }else if( nVal==15 && 0==sqlite3_strnicmp(zVal, "integrity-check", 15) ){ |
| rc = fts3DoIntegrityCheck(p); |
| }else if( nVal>6 && 0==sqlite3_strnicmp(zVal, "merge=", 6) ){ |
| rc = fts3DoIncrmerge(p, &zVal[6]); |
| }else if( nVal>10 && 0==sqlite3_strnicmp(zVal, "automerge=", 10) ){ |
| rc = fts3DoAutoincrmerge(p, &zVal[10]); |
| }else if( nVal==5 && 0==sqlite3_strnicmp(zVal, "flush", 5) ){ |
| rc = sqlite3Fts3PendingTermsFlush(p); |
| } |
| #if defined(SQLITE_DEBUG) || defined(SQLITE_TEST) |
| else{ |
| int v; |
| if( nVal>9 && 0==sqlite3_strnicmp(zVal, "nodesize=", 9) ){ |
| v = atoi(&zVal[9]); |
| if( v>=24 && v<=p->nPgsz-35 ) p->nNodeSize = v; |
| rc = SQLITE_OK; |
| }else if( nVal>11 && 0==sqlite3_strnicmp(zVal, "maxpending=", 11) ){ |
| v = atoi(&zVal[11]); |
| if( v>=64 && v<=FTS3_MAX_PENDING_DATA ) p->nMaxPendingData = v; |
| rc = SQLITE_OK; |
| }else if( nVal>21 && 0==sqlite3_strnicmp(zVal,"test-no-incr-doclist=",21) ){ |
| p->bNoIncrDoclist = atoi(&zVal[21]); |
| rc = SQLITE_OK; |
| }else if( nVal>11 && 0==sqlite3_strnicmp(zVal,"mergecount=",11) ){ |
| v = atoi(&zVal[11]); |
| if( v>=4 && v<=FTS3_MERGE_COUNT && (v&1)==0 ) p->nMergeCount = v; |
| rc = SQLITE_OK; |
| } |
| } |
| #endif |
| return rc; |
| } |
|
|
| #ifndef SQLITE_DISABLE_FTS4_DEFERRED |
| |
| |
| |
| |
| void sqlite3Fts3FreeDeferredDoclists(Fts3Cursor *pCsr){ |
| Fts3DeferredToken *pDef; |
| for(pDef=pCsr->pDeferred; pDef; pDef=pDef->pNext){ |
| fts3PendingListDelete(pDef->pList); |
| pDef->pList = 0; |
| } |
| } |
|
|
| |
| |
| |
| |
| void sqlite3Fts3FreeDeferredTokens(Fts3Cursor *pCsr){ |
| Fts3DeferredToken *pDef; |
| Fts3DeferredToken *pNext; |
| for(pDef=pCsr->pDeferred; pDef; pDef=pNext){ |
| pNext = pDef->pNext; |
| fts3PendingListDelete(pDef->pList); |
| sqlite3_free(pDef); |
| } |
| pCsr->pDeferred = 0; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| int sqlite3Fts3CacheDeferredDoclists(Fts3Cursor *pCsr){ |
| int rc = SQLITE_OK; |
| if( pCsr->pDeferred ){ |
| int i; |
| sqlite3_int64 iDocid; |
| Fts3DeferredToken *pDef; |
| |
| Fts3Table *p = (Fts3Table *)pCsr->base.pVtab; |
| sqlite3_tokenizer *pT = p->pTokenizer; |
| sqlite3_tokenizer_module const *pModule = pT->pModule; |
| |
| assert( pCsr->isRequireSeek==0 ); |
| iDocid = sqlite3_column_int64(pCsr->pStmt, 0); |
| |
| for(i=0; i<p->nColumn && rc==SQLITE_OK; i++){ |
| if( p->abNotindexed[i]==0 ){ |
| const char *zText = (const char *)sqlite3_column_text(pCsr->pStmt, i+1); |
| sqlite3_tokenizer_cursor *pTC = 0; |
|
|
| rc = sqlite3Fts3OpenTokenizer(pT, pCsr->iLangid, zText, -1, &pTC); |
| while( rc==SQLITE_OK ){ |
| char const *zToken; |
| int nToken = 0; |
| int iDum1 = 0, iDum2 = 0; |
| int iPos = 0; |
|
|
| rc = pModule->xNext(pTC, &zToken, &nToken, &iDum1, &iDum2, &iPos); |
| for(pDef=pCsr->pDeferred; pDef && rc==SQLITE_OK; pDef=pDef->pNext){ |
| Fts3PhraseToken *pPT = pDef->pToken; |
| if( (pDef->iCol>=p->nColumn || pDef->iCol==i) |
| && (pPT->bFirst==0 || iPos==0) |
| && (pPT->n==nToken || (pPT->isPrefix && pPT->n<nToken)) |
| && (0==memcmp(zToken, pPT->z, pPT->n)) |
| ){ |
| fts3PendingListAppend(&pDef->pList, iDocid, i, iPos, &rc); |
| } |
| } |
| } |
| if( pTC ) pModule->xClose(pTC); |
| if( rc==SQLITE_DONE ) rc = SQLITE_OK; |
| } |
| } |
|
|
| for(pDef=pCsr->pDeferred; pDef && rc==SQLITE_OK; pDef=pDef->pNext){ |
| if( pDef->pList ){ |
| rc = fts3PendingListAppendVarint(&pDef->pList, 0); |
| } |
| } |
| } |
|
|
| return rc; |
| } |
|
|
| int sqlite3Fts3DeferredTokenList( |
| Fts3DeferredToken *p, |
| char **ppData, |
| int *pnData |
| ){ |
| char *pRet; |
| int nSkip; |
| sqlite3_int64 dummy; |
|
|
| *ppData = 0; |
| *pnData = 0; |
|
|
| if( p->pList==0 ){ |
| return SQLITE_OK; |
| } |
|
|
| pRet = (char *)sqlite3_malloc64(p->pList->nData); |
| if( !pRet ) return SQLITE_NOMEM; |
|
|
| nSkip = sqlite3Fts3GetVarint(p->pList->aData, &dummy); |
| *pnData = p->pList->nData - nSkip; |
| *ppData = pRet; |
| |
| memcpy(pRet, &p->pList->aData[nSkip], *pnData); |
| return SQLITE_OK; |
| } |
|
|
| |
| |
| |
| int sqlite3Fts3DeferToken( |
| Fts3Cursor *pCsr, |
| Fts3PhraseToken *pToken, |
| int iCol |
| ){ |
| Fts3DeferredToken *pDeferred; |
| pDeferred = sqlite3_malloc64(sizeof(*pDeferred)); |
| if( !pDeferred ){ |
| return SQLITE_NOMEM; |
| } |
| memset(pDeferred, 0, sizeof(*pDeferred)); |
| pDeferred->pToken = pToken; |
| pDeferred->pNext = pCsr->pDeferred; |
| pDeferred->iCol = iCol; |
| pCsr->pDeferred = pDeferred; |
|
|
| assert( pToken->pDeferred==0 ); |
| pToken->pDeferred = pDeferred; |
|
|
| return SQLITE_OK; |
| } |
| #endif |
|
|
| |
| |
| |
| |
| |
| static int fts3DeleteByRowid( |
| Fts3Table *p, |
| sqlite3_value *pRowid, |
| int *pnChng, |
| u32 *aSzDel |
| ){ |
| int rc = SQLITE_OK; |
| int bFound = 0; |
|
|
| fts3DeleteTerms(&rc, p, pRowid, aSzDel, &bFound); |
| if( bFound && rc==SQLITE_OK ){ |
| int isEmpty = 0; |
| rc = fts3IsEmpty(p, pRowid, &isEmpty); |
| if( rc==SQLITE_OK ){ |
| if( isEmpty ){ |
| |
| |
| |
| rc = fts3DeleteAll(p, 1); |
| *pnChng = 0; |
| memset(aSzDel, 0, sizeof(u32) * (p->nColumn+1) * 2); |
| }else{ |
| *pnChng = *pnChng - 1; |
| if( p->zContentTbl==0 ){ |
| fts3SqlExec(&rc, p, SQL_DELETE_CONTENT, &pRowid); |
| } |
| if( p->bHasDocsize ){ |
| fts3SqlExec(&rc, p, SQL_DELETE_DOCSIZE, &pRowid); |
| } |
| } |
| } |
| } |
|
|
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| int sqlite3Fts3UpdateMethod( |
| sqlite3_vtab *pVtab, |
| int nArg, |
| sqlite3_value **apVal, |
| sqlite_int64 *pRowid |
| ){ |
| Fts3Table *p = (Fts3Table *)pVtab; |
| int rc = SQLITE_OK; |
| u32 *aSzIns = 0; |
| u32 *aSzDel = 0; |
| int nChng = 0; |
| int bInsertDone = 0; |
|
|
| |
| |
| assert( p->bHasStat==0 || p->bHasStat==1 ); |
|
|
| assert( p->pSegments==0 ); |
| assert( |
| nArg==1 |
| || nArg==(2 + p->nColumn + 3) |
| ); |
|
|
| |
| |
| |
| |
| if( nArg>1 |
| && sqlite3_value_type(apVal[0])==SQLITE_NULL |
| && sqlite3_value_type(apVal[p->nColumn+2])!=SQLITE_NULL |
| ){ |
| rc = fts3SpecialInsert(p, apVal[p->nColumn+2]); |
| goto update_out; |
| } |
|
|
| if( nArg>1 && sqlite3_value_int(apVal[2 + p->nColumn + 2])<0 ){ |
| rc = SQLITE_CONSTRAINT; |
| goto update_out; |
| } |
|
|
| |
| aSzDel = sqlite3_malloc64(sizeof(aSzDel[0])*((sqlite3_int64)p->nColumn+1)*2); |
| if( aSzDel==0 ){ |
| rc = SQLITE_NOMEM; |
| goto update_out; |
| } |
| aSzIns = &aSzDel[p->nColumn+1]; |
| memset(aSzDel, 0, sizeof(aSzDel[0])*(p->nColumn+1)*2); |
|
|
| rc = fts3Writelock(p); |
| if( rc!=SQLITE_OK ) goto update_out; |
|
|
| |
| |
| |
| |
| |
| |
| |
| |
| |
| if( nArg>1 && p->zContentTbl==0 ){ |
| |
| sqlite3_value *pNewRowid = apVal[3+p->nColumn]; |
| if( sqlite3_value_type(pNewRowid)==SQLITE_NULL ){ |
| pNewRowid = apVal[1]; |
| } |
|
|
| if( sqlite3_value_type(pNewRowid)!=SQLITE_NULL && ( |
| sqlite3_value_type(apVal[0])==SQLITE_NULL |
| || sqlite3_value_int64(apVal[0])!=sqlite3_value_int64(pNewRowid) |
| )){ |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| if( sqlite3_vtab_on_conflict(p->db)==SQLITE_REPLACE ){ |
| rc = fts3DeleteByRowid(p, pNewRowid, &nChng, aSzDel); |
| }else{ |
| rc = fts3InsertData(p, apVal, pRowid); |
| bInsertDone = 1; |
| } |
| } |
| } |
| if( rc!=SQLITE_OK ){ |
| goto update_out; |
| } |
|
|
| |
| if( sqlite3_value_type(apVal[0])!=SQLITE_NULL ){ |
| assert( sqlite3_value_type(apVal[0])==SQLITE_INTEGER ); |
| rc = fts3DeleteByRowid(p, apVal[0], &nChng, aSzDel); |
| } |
| |
| |
| if( nArg>1 && rc==SQLITE_OK ){ |
| int iLangid = sqlite3_value_int(apVal[2 + p->nColumn + 2]); |
| if( bInsertDone==0 ){ |
| rc = fts3InsertData(p, apVal, pRowid); |
| if( rc==SQLITE_CONSTRAINT && p->zContentTbl==0 ){ |
| rc = FTS_CORRUPT_VTAB; |
| } |
| } |
| if( rc==SQLITE_OK ){ |
| rc = fts3PendingTermsDocid(p, 0, iLangid, *pRowid); |
| } |
| if( rc==SQLITE_OK ){ |
| assert( p->iPrevDocid==*pRowid ); |
| rc = fts3InsertTerms(p, iLangid, apVal, aSzIns); |
| } |
| if( p->bHasDocsize ){ |
| fts3InsertDocsize(&rc, p, aSzIns); |
| } |
| nChng++; |
| } |
|
|
| if( p->bFts4 ){ |
| fts3UpdateDocTotals(&rc, p, aSzIns, aSzDel, nChng); |
| } |
|
|
| update_out: |
| sqlite3_free(aSzDel); |
| sqlite3Fts3SegmentsClose(p); |
| return rc; |
| } |
|
|
| |
| |
| |
| |
| |
| int sqlite3Fts3Optimize(Fts3Table *p){ |
| int rc; |
| rc = sqlite3_exec(p->db, "SAVEPOINT fts3", 0, 0, 0); |
| if( rc==SQLITE_OK ){ |
| rc = fts3DoOptimize(p, 1); |
| if( rc==SQLITE_OK || rc==SQLITE_DONE ){ |
| int rc2 = sqlite3_exec(p->db, "RELEASE fts3", 0, 0, 0); |
| if( rc2!=SQLITE_OK ) rc = rc2; |
| }else{ |
| sqlite3_exec(p->db, "ROLLBACK TO fts3", 0, 0, 0); |
| sqlite3_exec(p->db, "RELEASE fts3", 0, 0, 0); |
| } |
| } |
| sqlite3Fts3SegmentsClose(p); |
| return rc; |
| } |
|
|
| #endif |
|
|