032533
From 5f4ce30babee8085fc36680c6103d9a06be49ef7 Mon Sep 17 00:00:00 2001
032533
From: Ondrej Dubaj <odubaj@redhat.com>
032533
Date: Thu, 2 Jan 2020 11:58:39 +0100
032533
Subject: [PATCH] More improvements to shadow table corruption detection in
032533
 FTS3.
032533
032533
---
032533
 ext/fts3/fts3.c       |  4 ++++
032533
 ext/fts3/fts3Int.h    | 10 ++++++++++
032533
 ext/fts3/fts3_write.c | 14 +++++++++++---
032533
 3 files changed, 25 insertions(+), 3 deletions(-)
032533
032533
diff --git a/ext/fts3/fts3.c b/ext/fts3/fts3.c
032533
index 6d6bd46..84fc8a5 100644
032533
--- a/ext/fts3/fts3.c
032533
+++ b/ext/fts3/fts3.c
032533
@@ -1460,6 +1460,10 @@ static int fts3InitVtab(
032533
   fts3DatabasePageSize(&rc, p);
032533
   p->nNodeSize = p->nPgsz-35;
032533
 
032533
+#if defined(SQLITE_DEBUG)||defined(SQLITE_TEST)
032533
+  p->nMergeCount = FTS3_MERGE_COUNT;
032533
+#endif
032533
+
032533
   /* Declare the table schema to SQLite. */
032533
   fts3DeclareVtab(&rc, p);
032533
 
032533
diff --git a/ext/fts3/fts3Int.h b/ext/fts3/fts3Int.h
032533
index 6f5a7a0..0d1b491 100644
032533
--- a/ext/fts3/fts3Int.h
032533
+++ b/ext/fts3/fts3Int.h
032533
@@ -287,9 +287,19 @@ struct Fts3Table {
032533
   /* True to disable the incremental doclist optimization. This is controled
032533
   ** by special insert command 'test-no-incr-doclist'.  */
032533
   int bNoIncrDoclist;
032533
+
032533
+  /* Number of segments in a level */
032533
+  int nMergeCount;
032533
 #endif
032533
 };
032533
 
032533
+/* Macro to find the number of segments to merge */
032533
+#if defined(SQLITE_DEBUG) || defined(SQLITE_TEST)
032533
+# define MergeCount(P) ((P)->nMergeCount)
032533
+#else
032533
+# define MergeCount(P) FTS3_MERGE_COUNT
032533
+#endif
032533
+
032533
 /*
032533
 ** When the core wants to read from the virtual table, it creates a
032533
 ** virtual table cursor (an instance of the following structure) using
032533
diff --git a/ext/fts3/fts3_write.c b/ext/fts3/fts3_write.c
032533
index 8624329..d57d265 100644
032533
--- a/ext/fts3/fts3_write.c
032533
+++ b/ext/fts3/fts3_write.c
032533
@@ -1152,7 +1152,7 @@ static int fts3AllocateSegdirIdx(
032533
     ** segment and allocate (newly freed) index 0 at level iLevel. Otherwise,
032533
     ** if iNext is less than FTS3_MERGE_COUNT, allocate index iNext.
032533
     */
032533
-    if( iNext>=FTS3_MERGE_COUNT ){
032533
+    if( iNext>=MergeCount(p) ){
032533
       fts3LogMerge(16, getAbsoluteLevel(p, iLangid, iIndex, iLevel));
032533
       rc = fts3SegmentMerge(p, iLangid, iIndex, iLevel);
032533
       *piIdx = 0;
032533
@@ -4259,6 +4259,10 @@ static int fts3IncrmergeLoad(
032533
       int i;
032533
       int nHeight = (int)aRoot[0];
032533
       NodeWriter *pNode;
032533
+      if( nHeight<1 || nHeight>FTS_MAX_APPENDABLE_HEIGHT ){
032533
+        sqlite3_reset(pSelect);
032533
+        return FTS_CORRUPT_VTAB;
032533
+      }
032533
 
032533
       pWriter->nLeafEst = (int)((iEnd - iStart) + 1)/FTS_MAX_APPENDABLE_HEIGHT;
032533
       pWriter->iStart = iStart;
032533
@@ -5007,7 +5011,7 @@ static int fts3DoIncrmerge(
032533
   const char *zParam              /* Nul-terminated string containing "A,B" */
032533
 ){
032533
   int rc;
032533
-  int nMin = (FTS3_MERGE_COUNT / 2);
032533
+  int nMin = (MergeCount(p) / 2);
032533
   int nMerge = 0;
032533
   const char *z = zParam;
032533
 
032533
@@ -5052,7 +5056,7 @@ static int fts3DoAutoincrmerge(
032533
   int rc = SQLITE_OK;
032533
   sqlite3_stmt *pStmt = 0;
032533
   p->nAutoincrmerge = fts3Getint(&zParam);
032533
-  if( p->nAutoincrmerge==1 || p->nAutoincrmerge>FTS3_MERGE_COUNT ){
032533
+  if( p->nAutoincrmerge==1 || p->nAutoincrmerge>MergeCount(p) ){
032533
     p->nAutoincrmerge = 8;
032533
   }
032533
   if( !p->bHasStat ){
032533
@@ -5340,6 +5344,10 @@ static int fts3SpecialInsert(Fts3Table *p, sqlite3_value *pVal){
032533
     }else if( nVal>21 && 0==sqlite3_strnicmp(zVal,"test-no-incr-doclist=",21) ){
032533
       p->bNoIncrDoclist = atoi(&zVal[21]);
032533
       rc = SQLITE_OK;
032533
+    }else if( nVal>11 && 0==sqlite3_strnicmp(zVal,"mergecount=",11) ){
032533
+      v = atoi(&zVal[11]);
032533
+      if( v>=4 && v<=FTS3_MERGE_COUNT && (v&1)==0 ) p->nMergeCount = v;
032533
+      rc = SQLITE_OK;
032533
     }
032533
 #endif
032533
   }
032533
-- 
032533
2.19.1
032533