Blame SOURCES/sqlite-3.26-CVE-2019-13752.patch

ec7966
From 92b243715eea17997ed9707540757d0667ad9eb2 Mon Sep 17 00:00:00 2001
ec7966
From: Ondrej Dubaj <odubaj@redhat.com>
ec7966
Date: Thu, 2 Jan 2020 09:54:41 +0100
ec7966
Subject: [PATCH] Improved detection of corrupt shadow tables in FTS3. Enable
ec7966
 the debugging special-inserts for FTS3 for both SQLITE_DEBUG and SQLITE_TEST.
ec7966
ec7966
Resolves: CVE-2019-13752
ec7966
---
ec7966
 ext/fts3/fts3.c       |  2 +-
ec7966
 ext/fts3/fts3Int.h    |  2 +-
ec7966
 ext/fts3/fts3_write.c | 42 +++++++++++++++++++++++++++---------------
ec7966
 3 files changed, 29 insertions(+), 17 deletions(-)
ec7966
ec7966
diff --git a/ext/fts3/fts3.c b/ext/fts3/fts3.c
ec7966
index f6fb931..6d6bd46 100644
ec7966
--- a/ext/fts3/fts3.c
ec7966
+++ b/ext/fts3/fts3.c
ec7966
@@ -4304,7 +4304,7 @@ static int fts3EvalPhraseStart(Fts3Cursor *pCsr, int bOptOk, Fts3Phrase *p){
ec7966
   int bIncrOk = (bOptOk 
ec7966
    && pCsr->bDesc==pTab->bDescIdx 
ec7966
    && p->nToken<=MAX_INCR_PHRASE_TOKENS && p->nToken>0
ec7966
-#ifdef SQLITE_TEST
ec7966
+#if defined(SQLITE_DEBUG) || defined(SQLITE_TEST)
ec7966
    && pTab->bNoIncrDoclist==0
ec7966
 #endif
ec7966
   );
ec7966
diff --git a/ext/fts3/fts3Int.h b/ext/fts3/fts3Int.h
ec7966
index 077bad7..6f5a7a0 100644
ec7966
--- a/ext/fts3/fts3Int.h
ec7966
+++ b/ext/fts3/fts3Int.h
ec7966
@@ -283,7 +283,7 @@ struct Fts3Table {
ec7966
   int mxSavepoint;       /* Largest valid xSavepoint integer */
ec7966
 #endif
ec7966
 
ec7966
-#ifdef SQLITE_TEST
ec7966
+#if defined(SQLITE_DEBUG) || defined(SQLITE_TEST)
ec7966
   /* True to disable the incremental doclist optimization. This is controled
ec7966
   ** by special insert command 'test-no-incr-doclist'.  */
ec7966
   int bNoIncrDoclist;
ec7966
diff --git a/ext/fts3/fts3_write.c b/ext/fts3/fts3_write.c
ec7966
index 8fc6589..ee668aa 100644
ec7966
--- a/ext/fts3/fts3_write.c
ec7966
+++ b/ext/fts3/fts3_write.c
ec7966
@@ -23,7 +23,7 @@
ec7966
 #include <string.h>
ec7966
 #include <assert.h>
ec7966
 #include <stdlib.h>
ec7966
-
ec7966
+#include <stdio.h>
ec7966
 
ec7966
 #define FTS_MAX_APPENDABLE_HEIGHT 16
ec7966
 
ec7966
@@ -2021,6 +2021,11 @@ static int fts3NodeAddTerm(
ec7966
     nPrefix = fts3PrefixCompress(pTree->zTerm, pTree->nTerm, zTerm, nTerm);
ec7966
     nSuffix = nTerm-nPrefix;
ec7966
 
ec7966
+    /* If nSuffix is zero or less, then zTerm/nTerm must be a prefix of 
ec7966
+    ** pWriter->zTerm/pWriter->nTerm. i.e. must be equal to or less than when
ec7966
+    ** compared with BINARY collation. This indicates corruption.  */
ec7966
+    if( nSuffix<=0 ) return FTS_CORRUPT_VTAB;
ec7966
+
ec7966
     nReq += sqlite3Fts3VarintLen(nPrefix)+sqlite3Fts3VarintLen(nSuffix)+nSuffix;
ec7966
     if( nReq<=p->nNodeSize || !pTree->zTerm ){
ec7966
 
ec7966
@@ -2309,9 +2314,11 @@ static int fts3SegWriterAdd(
ec7966
   /* Append the prefix-compressed term and doclist to the buffer. */
ec7966
   nData += sqlite3Fts3PutVarint(&pWriter->aData[nData], nPrefix);
ec7966
   nData += sqlite3Fts3PutVarint(&pWriter->aData[nData], nSuffix);
ec7966
+  assert( nSuffix>0 );
ec7966
   memcpy(&pWriter->aData[nData], &zTerm[nPrefix], nSuffix);
ec7966
   nData += nSuffix;
ec7966
   nData += sqlite3Fts3PutVarint(&pWriter->aData[nData], nDoclist);
ec7966
+  assert( nDoclist>0 );
ec7966
   memcpy(&pWriter->aData[nData], aDoclist, nDoclist);
ec7966
   pWriter->nData = nData + nDoclist;
ec7966
 
ec7966
@@ -2331,6 +2338,7 @@ static int fts3SegWriterAdd(
ec7966
       pWriter->zTerm = zNew;
ec7966
     }
ec7966
     assert( pWriter->zTerm==pWriter->zMalloc );
ec7966
+    assert( nTerm>0 );
ec7966
     memcpy(pWriter->zTerm, zTerm, nTerm);
ec7966
   }else{
ec7966
     pWriter->zTerm = (char *)zTerm;
ec7966
@@ -2639,6 +2647,7 @@ static int fts3MsrBufferData(
ec7966
     pMsr->aBuffer = pNew;
ec7966
   }
ec7966
 
ec7966
+  assert( nList>0 );
ec7966
   memcpy(pMsr->aBuffer, pList, nList);
ec7966
   return SQLITE_OK;
ec7966
 }
ec7966
@@ -3821,6 +3830,7 @@ static int fts3IncrmergePush(
ec7966
     ** be added to.  */
ec7966
     nPrefix = fts3PrefixCompress(pNode->key.a, pNode->key.n, zTerm, nTerm);
ec7966
     nSuffix = nTerm - nPrefix;
ec7966
+    if( NEVER(nSuffix<=0) ) return FTS_CORRUPT_VTAB;
ec7966
     nSpace  = sqlite3Fts3VarintLen(nPrefix);
ec7966
     nSpace += sqlite3Fts3VarintLen(nSuffix) + nSuffix;
ec7966
 
ec7966
@@ -5300,7 +5310,7 @@ static int fts3DoIntegrityCheck(
ec7966
 ** meaningful value to insert is the text 'optimize'.
ec7966
 */
ec7966
 static int fts3SpecialInsert(Fts3Table *p, sqlite3_value *pVal){
ec7966
-  int rc;                         /* Return Code */
ec7966
+  int rc = SQLITE_ERROR;           /* Return Code */
ec7966
   const char *zVal = (const char *)sqlite3_value_text(pVal);
ec7966
   int nVal = sqlite3_value_bytes(pVal);
ec7966
 
ec7966
@@ -5316,21 +5326,23 @@ static int fts3SpecialInsert(Fts3Table *p, sqlite3_value *pVal){
ec7966
     rc = fts3DoIncrmerge(p, &zVal[6]);
ec7966
   }else if( nVal>10 && 0==sqlite3_strnicmp(zVal, "automerge=", 10) ){
ec7966
     rc = fts3DoAutoincrmerge(p, &zVal[10]);
ec7966
-#ifdef SQLITE_TEST
ec7966
-  }else if( nVal>9 && 0==sqlite3_strnicmp(zVal, "nodesize=", 9) ){
ec7966
-    p->nNodeSize = atoi(&zVal[9]);
ec7966
-    rc = SQLITE_OK;
ec7966
-  }else if( nVal>11 && 0==sqlite3_strnicmp(zVal, "maxpending=", 9) ){
ec7966
-    p->nMaxPendingData = atoi(&zVal[11]);
ec7966
-    rc = SQLITE_OK;
ec7966
-  }else if( nVal>21 && 0==sqlite3_strnicmp(zVal, "test-no-incr-doclist=", 21) ){
ec7966
-    p->bNoIncrDoclist = atoi(&zVal[21]);
ec7966
-    rc = SQLITE_OK;
ec7966
-#endif
ec7966
+#if defined(SQLITE_DEBUG) || defined(SQLITE_TEST)
ec7966
   }else{
ec7966
-    rc = SQLITE_ERROR;
ec7966
+        int v;
ec7966
+    if( nVal>9 && 0==sqlite3_strnicmp(zVal, "nodesize=", 9) ){
ec7966
+      v = atoi(&zVal[9]);
ec7966
+      if( v>=24 && v<=p->nPgsz-35 ) p->nNodeSize = v;
ec7966
+      rc = SQLITE_OK;
ec7966
+    }else if( nVal>11 && 0==sqlite3_strnicmp(zVal, "maxpending=", 9) ){
ec7966
+      v = atoi(&zVal[11]);
ec7966
+      if( v>=64 && v<=FTS3_MAX_PENDING_DATA ) p->nMaxPendingData = v;
ec7966
+      rc = SQLITE_OK;
ec7966
+    }else if( nVal>21 && 0==sqlite3_strnicmp(zVal,"test-no-incr-doclist=",21) ){
ec7966
+      p->bNoIncrDoclist = atoi(&zVal[21]);
ec7966
+      rc = SQLITE_OK;
ec7966
+    }
ec7966
+#endif
ec7966
   }
ec7966
-
ec7966
   return rc;
ec7966
 }
ec7966
 
ec7966
-- 
ec7966
2.19.1
ec7966