-
Notifications
You must be signed in to change notification settings - Fork 524
/
Copy pathconstant_fold.go
84 lines (79 loc) · 2.26 KB
/
constant_fold.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
// Copyright 2016 PingCAP, Inc.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// See the License for the specific language governing permissions and
// limitations under the License.
package expression
import (
"github.com/pingcap/tidb/util/chunk"
"github.com/pingcap/tidb/util/logutil"
"go.uber.org/zap"
)
// FoldConstant does constant folding optimization on an expression excluding deferred ones.
func FoldConstant(expr Expression) Expression {
return foldConstant(expr)
}
func foldConstant(expr Expression) Expression {
switch x := expr.(type) {
case *ScalarFunction:
if _, ok := unFoldableFunctions[x.FuncName.L]; ok {
return expr
}
args := x.GetArgs()
sc := x.GetCtx().GetSessionVars().StmtCtx
argIsConst := make([]bool, len(args))
hasNullArg := false
allConstArg := true
for i := 0; i < len(args); i++ {
switch x := args[i].(type) {
case *Constant:
argIsConst[i] = true
hasNullArg = hasNullArg || x.Value.IsNull()
default:
allConstArg = false
}
}
if !allConstArg {
if !hasNullArg || !sc.InNullRejectCheck {
return expr
}
constArgs := make([]Expression, len(args))
for i, arg := range args {
if argIsConst[i] {
constArgs[i] = arg
} else {
constArgs[i] = One
}
}
dummyScalarFunc, err := NewFunctionBase(x.GetCtx(), x.FuncName.L, x.GetType(), constArgs...)
if err != nil {
return expr
}
value, err := dummyScalarFunc.Eval(chunk.Row{})
if err != nil {
return expr
}
if value.IsNull() {
return &Constant{Value: value, RetType: x.RetType}
}
if isTrue, err := value.ToBool(sc); err == nil && isTrue == 0 {
return &Constant{Value: value, RetType: x.RetType}
}
return expr
}
value, err := x.Eval(chunk.Row{})
if err != nil {
logutil.BgLogger().Debug("fold expression to constant", zap.String("expression", x.ExplainInfo()), zap.Error(err))
return expr
}
return &Constant{Value: value, RetType: x.RetType}
}
return expr
}