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

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