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

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