]> git.ipfire.org Git - thirdparty/sqlite.git/commitdiff
Fixes to short-circuit AND/OR evaluation.
authordrh <>
Wed, 24 Sep 2025 14:43:42 +0000 (14:43 +0000)
committerdrh <>
Wed, 24 Sep 2025 14:43:42 +0000 (14:43 +0000)
FossilOrigin-Name: bc4cadbd60b97d000d2e8be843acab9d82ddf10a8125fd9cc0ae101bb9850091

manifest
manifest.uuid
src/expr.c

index 96072f415571d6f4d28f3aa42f766ace56e366ca..e0c48d566979eadaca716501e1808524d52de539 100644 (file)
--- a/manifest
+++ b/manifest
@@ -1,5 +1,5 @@
-C Factor\sout\sthe\scode\sthat\shandles\sAND\sand\sOR\sshort-circuiting\sinto\sa\nseparate\ssubroutine,\sfor\sperformance\sand\slegibility.
-D 2025-09-24T13:10:55.288
+C Fixes\sto\sshort-circuit\sAND/OR\sevaluation.
+D 2025-09-24T14:43:42.310
 F .fossil-settings/binary-glob 61195414528fb3ea9693577e1980230d78a1f8b0a54c78cf1b9b24d0a409ed6a x
 F .fossil-settings/empty-dirs dbb81e8fc0401ac46a1491ab34a7f2c7c0452f2f06b54ebb845d024ca8283ef1
 F .fossil-settings/ignore-glob 35175cdfcf539b2318cb04a9901442804be81cd677d8b889fcc9149c21f239ea
@@ -693,7 +693,7 @@ F src/date.c b6f92001f4b1f73f21774927488661d28f4dac9cd9701ed96486d96b44f5b058
 F src/dbpage.c 081c59d84f187aa0eb48d98faf9578a00bde360f68438d646a86b618653d2479
 F src/dbstat.c 73362c0df0f40ad5523a6f5501224959d0976757b511299bf892313e79d14f5c
 F src/delete.c 03a77ba20e54f0f42ebd8eddf15411ed6bdb06a2c472ac4b6b336521bf7cea42
-F src/expr.c 941ab6b635a7c6c30572c0223065e03ba6fbf656aa886f5b821ccf962ce95bc0
+F src/expr.c 0746e286897020cf36ccb09713f43151a5f9ce8f208f0ae3b32fe42908f096c2
 F src/fault.c 460f3e55994363812d9d60844b2a6de88826e007
 F src/fkey.c 928ed2517e8732113d2b9821aa37af639688d752f4ea9ac6e0e393d713eeb76f
 F src/func.c de47a8295503aa130baae5e6d9868ecf4f7c4dbffa65d83ad1f70bdbac0ee2d6
@@ -2175,8 +2175,8 @@ F tool/version-info.c 3b36468a90faf1bbd59c65fd0eb66522d9f941eedd364fabccd7227350
 F tool/warnings-clang.sh bbf6a1e685e534c92ec2bfba5b1745f34fb6f0bc2a362850723a9ee87c1b31a7
 F tool/warnings.sh 1ad0169b022b280bcaaf94a7fa231591be96b514230ab5c98fbf15cd7df842dd
 F tool/win/sqlite.vsix deb315d026cc8400325c5863eef847784a219a2f
-P 0d4447aa61848deecf9170afce031f7b353b91d58474f3f094ce34e46be7f99e
-R 1772208b5f6079b8843463f58173d5e0
+P cab84a65c3825c59f3a7a750f4983563ce7ad007125cbcf35c187326289fea89
+R 249976bcbb8ead84fd46ece0ea605849
 U drh
-Z 0a260950732cda975b0bb04f700f181c
+Z ac1897a73ee8a49c86d6ce76056d5c1b
 # Remove this line to create a well-formed Fossil manifest.
index c8d630d5ed9035c1db1eac79c1dd594ec6eedf33..53d156157c317a11b589b8c4eb8c1d58e4bc6abf 100644 (file)
@@ -1 +1 @@
-cab84a65c3825c59f3a7a750f4983563ce7ad007125cbcf35c187326289fea89
+bc4cadbd60b97d000d2e8be843acab9d82ddf10a8125fd9cc0ae101bb9850091
index 61b01c11123ac4304dee0c5688c8c3738e0f6c97..c464fd851d02b07d535ca7b0f9f09c6a4eca4421 100644 (file)
@@ -4811,21 +4811,26 @@ static int exprPartidxExprLookup(Parse *pParse, Expr *pExpr, int iTarget){
 }
 
 /*
-** This is the code to implement sqlite3ExprCodeTarget() for the
-** case of an AND or OR operator.  It is factored out for performance
-** and legibility.
+** Generate code that evaluates an AND or OR operator leaving a
+** boolean result in a register.  pExpr is the AND/OR expression.
+** Store the result in the "target" register.  Use short-circuit
+** evaluation to avoid computing both operands, if possible.
+**
+** The code generated might require the use of a temporary register.
+** If it does, then write the number of that temporary register
+** into *pTmpReg.  If not, leave *pTmpReg unchanged.
 */
 static SQLITE_NOINLINE int exprCodeTargetAndOr(
   Parse *pParse,     /* Parsing context */
   Expr *pExpr,       /* AND or OR expression to be coded */
   int target,        /* Put result in this register, guaranteed */
-  int *pTmpReg       /* Number of a temporary register */
+  int *pTmpReg       /* Write a temporary register here */
 ){
   int op;            /* The opcode.  TK_AND or TK_OR */
   int skipOp;        /* Opcode for the branch that skips one operand */
   int addrSkip;      /* Branch instruction that skips one of the operands */
-  int regSS = 0;
-  int r1, r2;        /* Registers holding left and right operands */
+  int regSS = 0;     /* Register holding computed operand when other omitted */
+  int r1, r2;        /* Registers for left and right operands, respectively */
   Expr *pAlt;        /* Alternative, simplified expression */
   Vdbe *v;           /* statement being coded */
 
@@ -4842,20 +4847,26 @@ static SQLITE_NOINLINE int exprCodeTargetAndOr(
   v = pParse->pVdbe;
   skipOp = op==TK_AND ? OP_IfNot : OP_If;
   if( exprEvalRhsFirst(pExpr) ){
+    /* Compute the right operand first.  Skip the computation of the left
+    ** operand if the right operand fully determines the result */
     r2 = regSS = sqlite3ExprCodeTarget(pParse, pExpr->pRight, target);
     addrSkip = sqlite3VdbeAddOp1(v, skipOp, r2);
     VdbeComment((v, "skip left operand"));
     VdbeCoverage(v); 
+    r1 = sqlite3ExprCodeTemp(pParse, pExpr->pLeft, pTmpReg);
   }else{
-    r2 = 0;
-    addrSkip = 0;
-  }
-  r1 = regSS = sqlite3ExprCodeTarget(pParse, pExpr->pLeft, target);
-  if( addrSkip==0 ){
+    /* Compute the left operand first */
+    r1 = sqlite3ExprCodeTarget(pParse, pExpr->pLeft, target);
     if( ExprHasProperty(pExpr->pRight, EP_Subquery) ){
+      /* Skip over the computation of the right operand if the right
+      ** operand is a subquery and the left operand completely determines
+      ** the result */
+      regSS = r1;
       addrSkip = sqlite3VdbeAddOp1(v, skipOp, r1);
       VdbeComment((v, "skip right operand"));
       VdbeCoverage(v);
+    }else{
+      addrSkip = regSS = 0;
     }
     r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, pTmpReg);
   }