mirror of
https://github.com/sqlite/sqlite.git
synced 2025-08-10 01:02:56 +03:00
For all binary operators, try to avoid computing subquery operands if the
other operand is NULL. FossilOrigin-Name: d86eb16283c4b573c506d4faa422d5d9aeb6abc279d8e6a8e2104737162d417f
This commit is contained in:
12
manifest
12
manifest
@@ -1,5 +1,5 @@
|
|||||||
C Slightly\ssmaller\sand\sfaster\sversion\sof\sthe\sprevious\scheck-in.
|
C For\sall\sbinary\soperators,\stry\sto\savoid\scomputing\ssubquery\soperands\sif\sthe\nother\soperand\sis\sNULL.
|
||||||
D 2025-06-30T11:04:55.886
|
D 2025-06-30T12:14:47.305
|
||||||
F .fossil-settings/binary-glob 61195414528fb3ea9693577e1980230d78a1f8b0a54c78cf1b9b24d0a409ed6a x
|
F .fossil-settings/binary-glob 61195414528fb3ea9693577e1980230d78a1f8b0a54c78cf1b9b24d0a409ed6a x
|
||||||
F .fossil-settings/empty-dirs dbb81e8fc0401ac46a1491ab34a7f2c7c0452f2f06b54ebb845d024ca8283ef1
|
F .fossil-settings/empty-dirs dbb81e8fc0401ac46a1491ab34a7f2c7c0452f2f06b54ebb845d024ca8283ef1
|
||||||
F .fossil-settings/ignore-glob 35175cdfcf539b2318cb04a9901442804be81cd677d8b889fcc9149c21f239ea
|
F .fossil-settings/ignore-glob 35175cdfcf539b2318cb04a9901442804be81cd677d8b889fcc9149c21f239ea
|
||||||
@@ -736,7 +736,7 @@ F src/date.c 9db4d604e699a73e10b8e85a44db074a1f04c0591a77e2abfd77703f50dce1e9
|
|||||||
F src/dbpage.c b3e218f8ed74fcbb7fa805df8ca669a3718d397617b3d8a8aac3307dc315c4d6
|
F src/dbpage.c b3e218f8ed74fcbb7fa805df8ca669a3718d397617b3d8a8aac3307dc315c4d6
|
||||||
F src/dbstat.c 73362c0df0f40ad5523a6f5501224959d0976757b511299bf892313e79d14f5c
|
F src/dbstat.c 73362c0df0f40ad5523a6f5501224959d0976757b511299bf892313e79d14f5c
|
||||||
F src/delete.c 03a77ba20e54f0f42ebd8eddf15411ed6bdb06a2c472ac4b6b336521bf7cea42
|
F src/delete.c 03a77ba20e54f0f42ebd8eddf15411ed6bdb06a2c472ac4b6b336521bf7cea42
|
||||||
F src/expr.c bfcf223fec3118214c2cc41c6aa520c716cb0e67a673258cd6fab3d143724c15
|
F src/expr.c d726acf67585c14473e354c3d1123a4290e357bdfd389056a05df15b70f66d49
|
||||||
F src/fault.c 460f3e55994363812d9d60844b2a6de88826e007
|
F src/fault.c 460f3e55994363812d9d60844b2a6de88826e007
|
||||||
F src/fkey.c 928ed2517e8732113d2b9821aa37af639688d752f4ea9ac6e0e393d713eeb76f
|
F src/fkey.c 928ed2517e8732113d2b9821aa37af639688d752f4ea9ac6e0e393d713eeb76f
|
||||||
F src/func.c de47a8295503aa130baae5e6d9868ecf4f7c4dbffa65d83ad1f70bdbac0ee2d6
|
F src/func.c de47a8295503aa130baae5e6d9868ecf4f7c4dbffa65d83ad1f70bdbac0ee2d6
|
||||||
@@ -2208,8 +2208,8 @@ F tool/version-info.c 3b36468a90faf1bbd59c65fd0eb66522d9f941eedd364fabccd7227350
|
|||||||
F tool/warnings-clang.sh bbf6a1e685e534c92ec2bfba5b1745f34fb6f0bc2a362850723a9ee87c1b31a7
|
F tool/warnings-clang.sh bbf6a1e685e534c92ec2bfba5b1745f34fb6f0bc2a362850723a9ee87c1b31a7
|
||||||
F tool/warnings.sh 1ad0169b022b280bcaaf94a7fa231591be96b514230ab5c98fbf15cd7df842dd
|
F tool/warnings.sh 1ad0169b022b280bcaaf94a7fa231591be96b514230ab5c98fbf15cd7df842dd
|
||||||
F tool/win/sqlite.vsix deb315d026cc8400325c5863eef847784a219a2f
|
F tool/win/sqlite.vsix deb315d026cc8400325c5863eef847784a219a2f
|
||||||
P 3c6c71bcea16443b82234414137dfa1b59e2ee8fe5097906c738fc1228fec4e6
|
P f6e6fd02f4ad49c390a2d3c9626d57f9b2fff1f67eb361b30074cc1f5121810e
|
||||||
R 2d8283b273252189c8c03b9b869c5bf9
|
R 886172909745da80cda276284a701cfd
|
||||||
U drh
|
U drh
|
||||||
Z f639db3f82a44bb4f9d7dfb2e1c4da9d
|
Z 78219ecf578f4ce2acfc88c5fa4f82e5
|
||||||
# Remove this line to create a well-formed Fossil manifest.
|
# Remove this line to create a well-formed Fossil manifest.
|
||||||
|
@@ -1 +1 @@
|
|||||||
f6e6fd02f4ad49c390a2d3c9626d57f9b2fff1f67eb361b30074cc1f5121810e
|
d86eb16283c4b573c506d4faa422d5d9aeb6abc279d8e6a8e2104737162d417f
|
||||||
|
251
src/expr.c
251
src/expr.c
@@ -2373,6 +2373,81 @@ Expr *sqlite3ExprSimplifiedAndOr(Expr *pExpr){
|
|||||||
return pExpr;
|
return pExpr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
** Return true if it might be advantageous to compute the right operand
|
||||||
|
** of expression pExpr first, before the left operand.
|
||||||
|
**
|
||||||
|
** Normally the left operand is computed before the right operand. But if
|
||||||
|
** the left operand contains a subquery and the right does not, then it
|
||||||
|
** might be more efficient to compute the right operand first.
|
||||||
|
*/
|
||||||
|
static int exprEvalRhsFirst(Expr *pExpr){
|
||||||
|
if( ExprHasProperty(pExpr->pLeft, EP_Subquery)
|
||||||
|
&& !ExprHasProperty(pExpr->pRight, EP_Subquery)
|
||||||
|
){
|
||||||
|
return 1;
|
||||||
|
}else{
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
** Compute the two operands of a binary operator.
|
||||||
|
**
|
||||||
|
** If either operand contains a subquery, then the code strives to
|
||||||
|
** compute the operand containing the subquery second. If the other
|
||||||
|
** operand evalutes to NULL, then a jump is made. The address of the
|
||||||
|
** IsNull operand that does this jump is returned. The caller can use
|
||||||
|
** this to optimize the computation so as to avoid doing the potentially
|
||||||
|
** expensive subquery.
|
||||||
|
**
|
||||||
|
** If no optimization opportunities exist, return 0.
|
||||||
|
*/
|
||||||
|
static int exprComputeOperands(
|
||||||
|
Parse *pParse, /* Parsing context */
|
||||||
|
Expr *pExpr, /* The comparison expression */
|
||||||
|
int *pR1, /* OUT: Register holding the left operand */
|
||||||
|
int *pR2, /* OUT: Register holding the right operand */
|
||||||
|
int *pFree1, /* OUT: Temp register to free if not zero */
|
||||||
|
int *pFree2 /* OUT: Another temp register to free if not zero */
|
||||||
|
){
|
||||||
|
int addrIsNull;
|
||||||
|
int r1, r2;
|
||||||
|
Vdbe *v = pParse->pVdbe;
|
||||||
|
|
||||||
|
assert( v!=0 );
|
||||||
|
/*
|
||||||
|
** If the left operand contains a (possibly expensive) subquery and the
|
||||||
|
** right operand does not and the right operation might be NULL,
|
||||||
|
** then compute the right operand first and do an IsNull jump if the
|
||||||
|
** right operand evalutes to NULL.
|
||||||
|
*/
|
||||||
|
if( exprEvalRhsFirst(pExpr) && sqlite3ExprCanBeNull(pExpr->pRight) ){
|
||||||
|
r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, pFree2);
|
||||||
|
addrIsNull = sqlite3VdbeAddOp1(v, OP_IsNull, r2); VdbeCoverage(v);
|
||||||
|
}else{
|
||||||
|
addrIsNull = 0;
|
||||||
|
}
|
||||||
|
r1 = sqlite3ExprCodeTemp(pParse, pExpr->pLeft, pFree1);
|
||||||
|
if( addrIsNull==0 ){
|
||||||
|
/*
|
||||||
|
** If the right operand contains a subquery and the left operand does not
|
||||||
|
** and the left operand might be NULL, then check the left operand do
|
||||||
|
** an IsNull check on the left operand before computing the right
|
||||||
|
** operand.
|
||||||
|
*/
|
||||||
|
if( ExprHasProperty(pExpr->pRight, EP_Subquery)
|
||||||
|
&& sqlite3ExprCanBeNull(pExpr->pLeft)
|
||||||
|
){
|
||||||
|
addrIsNull = sqlite3VdbeAddOp1(v, OP_IsNull, r1); VdbeCoverage(v);
|
||||||
|
}
|
||||||
|
r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, pFree2);
|
||||||
|
}
|
||||||
|
*pR1 = r1;
|
||||||
|
*pR2 = r2;
|
||||||
|
return addrIsNull;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
** pExpr is a TK_FUNCTION node. Try to determine whether or not the
|
** pExpr is a TK_FUNCTION node. Try to determine whether or not the
|
||||||
** function is a constant function. A function is constant if all of
|
** function is a constant function. A function is constant if all of
|
||||||
@@ -4961,11 +5036,17 @@ expr_code_doover:
|
|||||||
case TK_NE:
|
case TK_NE:
|
||||||
case TK_EQ: {
|
case TK_EQ: {
|
||||||
Expr *pLeft = pExpr->pLeft;
|
Expr *pLeft = pExpr->pLeft;
|
||||||
|
int addrIsNull = 0;
|
||||||
if( sqlite3ExprIsVector(pLeft) ){
|
if( sqlite3ExprIsVector(pLeft) ){
|
||||||
codeVectorCompare(pParse, pExpr, target, op, p5);
|
codeVectorCompare(pParse, pExpr, target, op, p5);
|
||||||
}else{
|
}else{
|
||||||
r1 = sqlite3ExprCodeTemp(pParse, pLeft, ®Free1);
|
if( ExprHasProperty(pExpr, EP_Subquery) && p5!=SQLITE_NULLEQ ){
|
||||||
r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, ®Free2);
|
addrIsNull = exprComputeOperands(pParse, pExpr,
|
||||||
|
&r1, &r2, ®Free1, ®Free2);
|
||||||
|
}else{
|
||||||
|
r1 = sqlite3ExprCodeTemp(pParse, pExpr->pLeft, ®Free1);
|
||||||
|
r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, ®Free2);
|
||||||
|
}
|
||||||
sqlite3VdbeAddOp2(v, OP_Integer, 1, inReg);
|
sqlite3VdbeAddOp2(v, OP_Integer, 1, inReg);
|
||||||
codeCompare(pParse, pLeft, pExpr->pRight, op, r1, r2,
|
codeCompare(pParse, pLeft, pExpr->pRight, op, r1, r2,
|
||||||
sqlite3VdbeCurrentAddr(v)+2, p5,
|
sqlite3VdbeCurrentAddr(v)+2, p5,
|
||||||
@@ -4980,9 +5061,15 @@ expr_code_doover:
|
|||||||
sqlite3VdbeAddOp2(v, OP_Integer, 0, inReg);
|
sqlite3VdbeAddOp2(v, OP_Integer, 0, inReg);
|
||||||
}else{
|
}else{
|
||||||
sqlite3VdbeAddOp3(v, OP_ZeroOrNull, r1, inReg, r2);
|
sqlite3VdbeAddOp3(v, OP_ZeroOrNull, r1, inReg, r2);
|
||||||
|
if( addrIsNull ){
|
||||||
|
sqlite3VdbeAddOp2(v, OP_Goto, 0, sqlite3VdbeCurrentAddr(v)+2);
|
||||||
|
sqlite3VdbeJumpHere(v, addrIsNull);
|
||||||
|
sqlite3VdbeAddOp2(v, OP_Null, 0, inReg);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
testcase( regFree1==0 );
|
testcase( regFree1==0 );
|
||||||
testcase( regFree2==0 );
|
testcase( regFree2==0 );
|
||||||
|
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -4998,6 +5085,7 @@ expr_code_doover:
|
|||||||
case TK_LSHIFT:
|
case TK_LSHIFT:
|
||||||
case TK_RSHIFT:
|
case TK_RSHIFT:
|
||||||
case TK_CONCAT: {
|
case TK_CONCAT: {
|
||||||
|
int addrIsNull;
|
||||||
assert( TK_AND==OP_And ); testcase( op==TK_AND );
|
assert( TK_AND==OP_And ); testcase( op==TK_AND );
|
||||||
assert( TK_OR==OP_Or ); testcase( op==TK_OR );
|
assert( TK_OR==OP_Or ); testcase( op==TK_OR );
|
||||||
assert( TK_PLUS==OP_Add ); testcase( op==TK_PLUS );
|
assert( TK_PLUS==OP_Add ); testcase( op==TK_PLUS );
|
||||||
@@ -5009,11 +5097,22 @@ expr_code_doover:
|
|||||||
assert( TK_LSHIFT==OP_ShiftLeft ); testcase( op==TK_LSHIFT );
|
assert( TK_LSHIFT==OP_ShiftLeft ); testcase( op==TK_LSHIFT );
|
||||||
assert( TK_RSHIFT==OP_ShiftRight ); testcase( op==TK_RSHIFT );
|
assert( TK_RSHIFT==OP_ShiftRight ); testcase( op==TK_RSHIFT );
|
||||||
assert( TK_CONCAT==OP_Concat ); testcase( op==TK_CONCAT );
|
assert( TK_CONCAT==OP_Concat ); testcase( op==TK_CONCAT );
|
||||||
r1 = sqlite3ExprCodeTemp(pParse, pExpr->pLeft, ®Free1);
|
if( ExprHasProperty(pExpr, EP_Subquery) ){
|
||||||
r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, ®Free2);
|
addrIsNull = exprComputeOperands(pParse, pExpr,
|
||||||
|
&r1, &r2, ®Free1, ®Free2);
|
||||||
|
}else{
|
||||||
|
r1 = sqlite3ExprCodeTemp(pParse, pExpr->pLeft, ®Free1);
|
||||||
|
r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, ®Free2);
|
||||||
|
addrIsNull = 0;
|
||||||
|
}
|
||||||
sqlite3VdbeAddOp3(v, op, r2, r1, target);
|
sqlite3VdbeAddOp3(v, op, r2, r1, target);
|
||||||
testcase( regFree1==0 );
|
testcase( regFree1==0 );
|
||||||
testcase( regFree2==0 );
|
testcase( regFree2==0 );
|
||||||
|
if( addrIsNull ){
|
||||||
|
sqlite3VdbeAddOp2(v, OP_Goto, 0, sqlite3VdbeCurrentAddr(v)+2);
|
||||||
|
sqlite3VdbeJumpHere(v, addrIsNull);
|
||||||
|
sqlite3VdbeAddOp2(v, OP_Null, 0, target);
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case TK_UMINUS: {
|
case TK_UMINUS: {
|
||||||
@@ -5830,67 +5929,6 @@ static void exprCodeBetween(
|
|||||||
testcase( xJump==0 );
|
testcase( xJump==0 );
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
** Compute the two operands of a binary operator.
|
|
||||||
**
|
|
||||||
** If either operand contains a subquery, then the code strives to
|
|
||||||
** compute the operand containing the subquery second. If the other
|
|
||||||
** operand evalutes to NULL, then a jump is made. The address of the
|
|
||||||
** IsNull operand that does this jump is returned. The caller can use
|
|
||||||
** this to optimize the computation so as to avoid doing the potentially
|
|
||||||
** expensive subquery.
|
|
||||||
**
|
|
||||||
** If no optimization opportunities exist, return 0.
|
|
||||||
*/
|
|
||||||
static int exprComputeOperands(
|
|
||||||
Parse *pParse, /* Parsing context */
|
|
||||||
Expr *pExpr, /* The comparison expression */
|
|
||||||
int *pR1, /* OUT: Register holding the left operand */
|
|
||||||
int *pR2, /* OUT: Register holding the right operand */
|
|
||||||
int *pFree1, /* OUT: Temp register to free if not zero */
|
|
||||||
int *pFree2 /* OUT: Another temp register to free if not zero */
|
|
||||||
){
|
|
||||||
int addrIsNull;
|
|
||||||
int r1, r2;
|
|
||||||
Vdbe *v = pParse->pVdbe;
|
|
||||||
|
|
||||||
assert( v!=0 );
|
|
||||||
/*
|
|
||||||
** If the left operand contains a (possibly expensive) subquery and the
|
|
||||||
** right operand does not and the right operation might be NULL,
|
|
||||||
** then compute the right operand first and do an IsNull jump if the
|
|
||||||
** right operand evalutes to NULL.
|
|
||||||
*/
|
|
||||||
if( ExprHasProperty(pExpr->pLeft, EP_Subquery)
|
|
||||||
&& !ExprHasProperty(pExpr->pRight, EP_Subquery)
|
|
||||||
&& sqlite3ExprCanBeNull(pExpr->pRight)
|
|
||||||
){
|
|
||||||
r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, pFree2);
|
|
||||||
addrIsNull = sqlite3VdbeAddOp1(v, OP_IsNull, r2); VdbeCoverage(v);
|
|
||||||
}else{
|
|
||||||
addrIsNull = 0;
|
|
||||||
}
|
|
||||||
r1 = sqlite3ExprCodeTemp(pParse, pExpr->pLeft, pFree1);
|
|
||||||
if( addrIsNull==0 ){
|
|
||||||
/*
|
|
||||||
** If the right operand contains a subquery and the left operand does not
|
|
||||||
** and the left operand might be NULL, then check the left operand do
|
|
||||||
** an IsNull check on the left operand before computing the right
|
|
||||||
** operand.
|
|
||||||
*/
|
|
||||||
if( ExprHasProperty(pExpr->pRight, EP_Subquery)
|
|
||||||
&& sqlite3ExprCanBeNull(pExpr->pLeft)
|
|
||||||
){
|
|
||||||
addrIsNull = sqlite3VdbeAddOp1(v, OP_IsNull, r1); VdbeCoverage(v);
|
|
||||||
}
|
|
||||||
r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, pFree2);
|
|
||||||
}
|
|
||||||
*pR1 = r1;
|
|
||||||
*pR2 = r2;
|
|
||||||
return addrIsNull;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
** Generate code for a boolean expression such that a jump is made
|
** Generate code for a boolean expression such that a jump is made
|
||||||
** to the label "dest" if the expression is true but execution
|
** to the label "dest" if the expression is true but execution
|
||||||
@@ -5923,17 +5961,27 @@ void sqlite3ExprIfTrue(Parse *pParse, Expr *pExpr, int dest, int jumpIfNull){
|
|||||||
Expr *pAlt = sqlite3ExprSimplifiedAndOr(pExpr);
|
Expr *pAlt = sqlite3ExprSimplifiedAndOr(pExpr);
|
||||||
if( pAlt!=pExpr ){
|
if( pAlt!=pExpr ){
|
||||||
sqlite3ExprIfTrue(pParse, pAlt, dest, jumpIfNull);
|
sqlite3ExprIfTrue(pParse, pAlt, dest, jumpIfNull);
|
||||||
}else if( op==TK_AND ){
|
|
||||||
int d2 = sqlite3VdbeMakeLabel(pParse);
|
|
||||||
testcase( jumpIfNull==0 );
|
|
||||||
sqlite3ExprIfFalse(pParse, pExpr->pLeft, d2,
|
|
||||||
jumpIfNull^SQLITE_JUMPIFNULL);
|
|
||||||
sqlite3ExprIfTrue(pParse, pExpr->pRight, dest, jumpIfNull);
|
|
||||||
sqlite3VdbeResolveLabel(v, d2);
|
|
||||||
}else{
|
}else{
|
||||||
testcase( jumpIfNull==0 );
|
Expr *pFirst, *pSecond;
|
||||||
sqlite3ExprIfTrue(pParse, pExpr->pLeft, dest, jumpIfNull);
|
if( exprEvalRhsFirst(pExpr) ){
|
||||||
sqlite3ExprIfTrue(pParse, pExpr->pRight, dest, jumpIfNull);
|
pFirst = pExpr->pRight;
|
||||||
|
pSecond = pExpr->pLeft;
|
||||||
|
}else{
|
||||||
|
pFirst = pExpr->pLeft;
|
||||||
|
pSecond = pExpr->pRight;
|
||||||
|
}
|
||||||
|
if( op==TK_AND ){
|
||||||
|
int d2 = sqlite3VdbeMakeLabel(pParse);
|
||||||
|
testcase( jumpIfNull==0 );
|
||||||
|
sqlite3ExprIfFalse(pParse, pFirst, d2,
|
||||||
|
jumpIfNull^SQLITE_JUMPIFNULL);
|
||||||
|
sqlite3ExprIfTrue(pParse, pSecond, dest, jumpIfNull);
|
||||||
|
sqlite3VdbeResolveLabel(v, d2);
|
||||||
|
}else{
|
||||||
|
testcase( jumpIfNull==0 );
|
||||||
|
sqlite3ExprIfTrue(pParse, pFirst, dest, jumpIfNull);
|
||||||
|
sqlite3ExprIfTrue(pParse, pSecond, dest, jumpIfNull);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -5972,10 +6020,16 @@ void sqlite3ExprIfTrue(Parse *pParse, Expr *pExpr, int dest, int jumpIfNull){
|
|||||||
case TK_GE:
|
case TK_GE:
|
||||||
case TK_NE:
|
case TK_NE:
|
||||||
case TK_EQ: {
|
case TK_EQ: {
|
||||||
|
int addrIsNull;
|
||||||
if( sqlite3ExprIsVector(pExpr->pLeft) ) goto default_expr;
|
if( sqlite3ExprIsVector(pExpr->pLeft) ) goto default_expr;
|
||||||
testcase( jumpIfNull==0 );
|
if( ExprHasProperty(pExpr, EP_Subquery) && jumpIfNull!=SQLITE_NULLEQ ){
|
||||||
r1 = sqlite3ExprCodeTemp(pParse, pExpr->pLeft, ®Free1);
|
addrIsNull = exprComputeOperands(pParse, pExpr,
|
||||||
r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, ®Free2);
|
&r1, &r2, ®Free1, ®Free2);
|
||||||
|
}else{
|
||||||
|
r1 = sqlite3ExprCodeTemp(pParse, pExpr->pLeft, ®Free1);
|
||||||
|
r2 = sqlite3ExprCodeTemp(pParse, pExpr->pRight, ®Free2);
|
||||||
|
addrIsNull = 0;
|
||||||
|
}
|
||||||
codeCompare(pParse, pExpr->pLeft, pExpr->pRight, op,
|
codeCompare(pParse, pExpr->pLeft, pExpr->pRight, op,
|
||||||
r1, r2, dest, jumpIfNull, ExprHasProperty(pExpr,EP_Commuted));
|
r1, r2, dest, jumpIfNull, ExprHasProperty(pExpr,EP_Commuted));
|
||||||
assert(TK_LT==OP_Lt); testcase(op==OP_Lt); VdbeCoverageIf(v,op==OP_Lt);
|
assert(TK_LT==OP_Lt); testcase(op==OP_Lt); VdbeCoverageIf(v,op==OP_Lt);
|
||||||
@@ -5990,6 +6044,13 @@ void sqlite3ExprIfTrue(Parse *pParse, Expr *pExpr, int dest, int jumpIfNull){
|
|||||||
VdbeCoverageIf(v, op==OP_Ne && jumpIfNull!=SQLITE_NULLEQ);
|
VdbeCoverageIf(v, op==OP_Ne && jumpIfNull!=SQLITE_NULLEQ);
|
||||||
testcase( regFree1==0 );
|
testcase( regFree1==0 );
|
||||||
testcase( regFree2==0 );
|
testcase( regFree2==0 );
|
||||||
|
if( addrIsNull ){
|
||||||
|
if( jumpIfNull ){
|
||||||
|
sqlite3VdbeChangeP2(v, addrIsNull, dest);
|
||||||
|
}else{
|
||||||
|
sqlite3VdbeJumpHere(v, addrIsNull);
|
||||||
|
}
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
case TK_ISNULL:
|
case TK_ISNULL:
|
||||||
@@ -6097,17 +6158,27 @@ void sqlite3ExprIfFalse(Parse *pParse, Expr *pExpr, int dest, int jumpIfNull){
|
|||||||
Expr *pAlt = sqlite3ExprSimplifiedAndOr(pExpr);
|
Expr *pAlt = sqlite3ExprSimplifiedAndOr(pExpr);
|
||||||
if( pAlt!=pExpr ){
|
if( pAlt!=pExpr ){
|
||||||
sqlite3ExprIfFalse(pParse, pAlt, dest, jumpIfNull);
|
sqlite3ExprIfFalse(pParse, pAlt, dest, jumpIfNull);
|
||||||
}else if( pExpr->op==TK_AND ){
|
|
||||||
testcase( jumpIfNull==0 );
|
|
||||||
sqlite3ExprIfFalse(pParse, pExpr->pLeft, dest, jumpIfNull);
|
|
||||||
sqlite3ExprIfFalse(pParse, pExpr->pRight, dest, jumpIfNull);
|
|
||||||
}else{
|
}else{
|
||||||
int d2 = sqlite3VdbeMakeLabel(pParse);
|
Expr *pFirst, *pSecond;
|
||||||
testcase( jumpIfNull==0 );
|
if( exprEvalRhsFirst(pExpr) ){
|
||||||
sqlite3ExprIfTrue(pParse, pExpr->pLeft, d2,
|
pFirst = pExpr->pRight;
|
||||||
jumpIfNull^SQLITE_JUMPIFNULL);
|
pSecond = pExpr->pLeft;
|
||||||
sqlite3ExprIfFalse(pParse, pExpr->pRight, dest, jumpIfNull);
|
}else{
|
||||||
sqlite3VdbeResolveLabel(v, d2);
|
pFirst = pExpr->pLeft;
|
||||||
|
pSecond = pExpr->pRight;
|
||||||
|
}
|
||||||
|
if( pExpr->op==TK_AND ){
|
||||||
|
testcase( jumpIfNull==0 );
|
||||||
|
sqlite3ExprIfFalse(pParse, pFirst, dest, jumpIfNull);
|
||||||
|
sqlite3ExprIfFalse(pParse, pSecond, dest, jumpIfNull);
|
||||||
|
}else{
|
||||||
|
int d2 = sqlite3VdbeMakeLabel(pParse);
|
||||||
|
testcase( jumpIfNull==0 );
|
||||||
|
sqlite3ExprIfTrue(pParse, pFirst, d2,
|
||||||
|
jumpIfNull^SQLITE_JUMPIFNULL);
|
||||||
|
sqlite3ExprIfFalse(pParse, pSecond, dest, jumpIfNull);
|
||||||
|
sqlite3VdbeResolveLabel(v, d2);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -6166,11 +6237,11 @@ void sqlite3ExprIfFalse(Parse *pParse, Expr *pExpr, int dest, int jumpIfNull){
|
|||||||
assert(TK_GT==OP_Gt); testcase(op==OP_Gt); VdbeCoverageIf(v,op==OP_Gt);
|
assert(TK_GT==OP_Gt); testcase(op==OP_Gt); VdbeCoverageIf(v,op==OP_Gt);
|
||||||
assert(TK_GE==OP_Ge); testcase(op==OP_Ge); VdbeCoverageIf(v,op==OP_Ge);
|
assert(TK_GE==OP_Ge); testcase(op==OP_Ge); VdbeCoverageIf(v,op==OP_Ge);
|
||||||
assert(TK_EQ==OP_Eq); testcase(op==OP_Eq);
|
assert(TK_EQ==OP_Eq); testcase(op==OP_Eq);
|
||||||
VdbeCoverageIf(v, op==OP_Eq && jumpIfNull==SQLITE_NULLEQ);
|
|
||||||
VdbeCoverageIf(v, op==OP_Eq && jumpIfNull!=SQLITE_NULLEQ);
|
VdbeCoverageIf(v, op==OP_Eq && jumpIfNull!=SQLITE_NULLEQ);
|
||||||
|
VdbeCoverageIf(v, op==OP_Eq && jumpIfNull==SQLITE_NULLEQ);
|
||||||
assert(TK_NE==OP_Ne); testcase(op==OP_Ne);
|
assert(TK_NE==OP_Ne); testcase(op==OP_Ne);
|
||||||
VdbeCoverageIf(v, op==OP_Ne && jumpIfNull==SQLITE_NULLEQ);
|
|
||||||
VdbeCoverageIf(v, op==OP_Ne && jumpIfNull!=SQLITE_NULLEQ);
|
VdbeCoverageIf(v, op==OP_Ne && jumpIfNull!=SQLITE_NULLEQ);
|
||||||
|
VdbeCoverageIf(v, op==OP_Ne && jumpIfNull==SQLITE_NULLEQ);
|
||||||
testcase( regFree1==0 );
|
testcase( regFree1==0 );
|
||||||
testcase( regFree2==0 );
|
testcase( regFree2==0 );
|
||||||
if( addrIsNull ){
|
if( addrIsNull ){
|
||||||
|
Reference in New Issue
Block a user