// Copyright 2019 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 ( "fmt" "math" "github.com/pingcap/errors" "github.com/pingcap/parser/mysql" "github.com/pingcap/parser/terror" "github.com/pingcap/tidb/types" "github.com/pingcap/tidb/util/chunk" ) func (b *builtinArithmeticMultiplyRealSig) vectorized() bool { return true } func (b *builtinArithmeticMultiplyRealSig) vecEvalReal(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalReal(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETReal, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalReal(b.ctx, input, buf); err != nil { return err } result.MergeNulls(buf) x := result.Float64s() y := buf.Float64s() for i := 0; i < n; i++ { if result.IsNull(i) { continue } x[i] = x[i] * y[i] if math.IsInf(x[i], 0) { return types.ErrOverflow.GenWithStackByArgs("DOUBLE", fmt.Sprintf("(%s * %s)", b.args[0].String(), b.args[1].String())) } } return nil } func (b *builtinArithmeticDivideDecimalSig) vectorized() bool { return true } func (b *builtinArithmeticDivideDecimalSig) vecEvalDecimal(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalDecimal(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETDecimal, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalDecimal(b.ctx, input, buf); err != nil { return err } result.MergeNulls(buf) x := result.Decimals() y := buf.Decimals() var to types.MyDecimal var frac int for i := 0; i < n; i++ { if result.IsNull(i) { continue } err = types.DecimalDiv(&x[i], &y[i], &to, types.DivFracIncr) if err == types.ErrDivByZero { if err = handleDivisionByZeroError(b.ctx); err != nil { return err } result.SetNull(i, true) continue } else if err == nil { _, frac = to.PrecisionAndFrac() if frac < b.baseBuiltinFunc.tp.Decimal { if err = to.Round(&to, b.baseBuiltinFunc.tp.Decimal, types.ModeHalfEven); err != nil { return err } } } else { return err } x[i] = to } return nil } func (b *builtinArithmeticModIntSig) vectorized() bool { return false } func (b *builtinArithmeticModIntSig) vecEvalInt(input *chunk.Chunk, result *chunk.Column) error { return errors.Errorf("not implemented") } func (b *builtinArithmeticMinusRealSig) vectorized() bool { return true } func (b *builtinArithmeticMinusRealSig) vecEvalReal(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalReal(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETReal, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalReal(b.ctx, input, buf); err != nil { return err } result.MergeNulls(buf) x := result.Float64s() y := buf.Float64s() for i := 0; i < n; i++ { if result.IsNull(i) { continue } if (x[i] > 0 && -y[i] > math.MaxFloat64-x[i]) || (x[i] < 0 && -y[i] < -math.MaxFloat64-x[i]) { return types.ErrOverflow.GenWithStackByArgs("DOUBLE", fmt.Sprintf("(%s - %s)", b.args[0].String(), b.args[1].String())) } x[i] = x[i] - y[i] } return nil } func (b *builtinArithmeticMinusDecimalSig) vectorized() bool { return true } func (b *builtinArithmeticMinusDecimalSig) vecEvalDecimal(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalDecimal(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETDecimal, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalDecimal(b.ctx, input, buf); err != nil { return err } result.MergeNulls(buf) x := result.Decimals() y := buf.Decimals() var to types.MyDecimal for i := 0; i < n; i++ { if result.IsNull(i) { continue } if err = types.DecimalSub(&x[i], &y[i], &to); err != nil { return err } x[i] = to } return nil } func (b *builtinArithmeticMinusIntSig) vectorized() bool { return false } func (b *builtinArithmeticMinusIntSig) vecEvalInt(input *chunk.Chunk, result *chunk.Column) error { return errors.Errorf("not implemented") } func (b *builtinArithmeticModRealSig) vectorized() bool { return true } func (b *builtinArithmeticModRealSig) vecEvalReal(input *chunk.Chunk, result *chunk.Column) error { n := input.NumRows() buf, err := b.bufAllocator.get(types.ETReal, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalReal(b.ctx, input, buf); err != nil { return err } if err := b.args[0].VecEvalReal(b.ctx, input, result); err != nil { return err } x := result.Float64s() y := buf.Float64s() for i := 0; i < n; i++ { if buf.IsNull(i) { result.SetNull(i, true) continue } if y[i] == 0 { if err := handleDivisionByZeroError(b.ctx); err != nil { return err } result.SetNull(i, true) continue } if result.IsNull(i) { continue } x[i] = math.Mod(x[i], y[i]) } return nil } func (b *builtinArithmeticModDecimalSig) vectorized() bool { return true } func (b *builtinArithmeticModDecimalSig) vecEvalDecimal(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalDecimal(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETDecimal, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalDecimal(b.ctx, input, buf); err != nil { return err } result.MergeNulls(buf) x := result.Decimals() y := buf.Decimals() var to types.MyDecimal for i := 0; i < n; i++ { if result.IsNull(i) { continue } err = types.DecimalMod(&x[i], &y[i], &to) if err == types.ErrDivByZero { if err := handleDivisionByZeroError(b.ctx); err != nil { return err } result.SetNull(i, true) continue } else if err != nil { return err } x[i] = to } return nil } func (b *builtinArithmeticPlusRealSig) vectorized() bool { return true } func (b *builtinArithmeticPlusRealSig) vecEvalReal(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalReal(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETReal, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalReal(b.ctx, input, buf); err != nil { return err } result.MergeNulls(buf) x := result.Float64s() y := buf.Float64s() for i := 0; i < n; i++ { if result.IsNull(i) { continue } if (x[i] > 0 && y[i] > math.MaxFloat64-x[i]) || (x[i] < 0 && y[i] < -math.MaxFloat64-x[i]) { return types.ErrOverflow.GenWithStackByArgs("DOUBLE", fmt.Sprintf("(%s + %s)", b.args[0].String(), b.args[1].String())) } x[i] = x[i] + y[i] } return nil } func (b *builtinArithmeticMultiplyDecimalSig) vectorized() bool { return true } func (b *builtinArithmeticMultiplyDecimalSig) vecEvalDecimal(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalDecimal(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETDecimal, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalDecimal(b.ctx, input, buf); err != nil { return err } result.MergeNulls(buf) x := result.Decimals() y := buf.Decimals() var to types.MyDecimal for i := 0; i < n; i++ { if result.IsNull(i) { continue } err = types.DecimalMul(&x[i], &y[i], &to) if err != nil && !terror.ErrorEqual(err, types.ErrTruncated) { return err } x[i] = to } return nil } func (b *builtinArithmeticIntDivideDecimalSig) vectorized() bool { return false } func (b *builtinArithmeticIntDivideDecimalSig) vecEvalInt(input *chunk.Chunk, result *chunk.Column) error { return errors.Errorf("not implemented") } func (b *builtinArithmeticMultiplyIntSig) vectorized() bool { return true } func (b *builtinArithmeticMultiplyIntSig) vecEvalInt(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalInt(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETInt, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalInt(b.ctx, input, buf); err != nil { return err } x := result.Int64s() y := buf.Int64s() result.MergeNulls(buf) var tmp int64 for i := 0; i < n; i++ { if result.IsNull(i) { continue } tmp = x[i] * y[i] if x[i] != 0 && tmp/x[i] != y[i] { result.SetNull(i, true) return types.ErrOverflow.GenWithStackByArgs("BIGINT", fmt.Sprintf("(%s * %s)", b.args[0].String(), b.args[1].String())) } x[i] = tmp } return nil } func (b *builtinArithmeticDivideRealSig) vectorized() bool { return true } func (b *builtinArithmeticDivideRealSig) vecEvalReal(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalReal(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETReal, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalReal(b.ctx, input, buf); err != nil { return err } result.MergeNulls(buf) x := result.Float64s() y := buf.Float64s() for i := 0; i < n; i++ { if result.IsNull(i) { continue } if y[i] == 0 { if err := handleDivisionByZeroError(b.ctx); err != nil { return err } result.SetNull(i, true) continue } x[i] = x[i] / y[i] if math.IsInf(x[i], 0) { return types.ErrOverflow.GenWithStackByArgs("DOUBLE", fmt.Sprintf("(%s / %s)", b.args[0].String(), b.args[1].String())) } } return nil } func (b *builtinArithmeticIntDivideIntSig) vectorized() bool { return false } func (b *builtinArithmeticIntDivideIntSig) vecEvalInt(input *chunk.Chunk, result *chunk.Column) error { return errors.Errorf("not implemented") } func (b *builtinArithmeticPlusIntSig) vectorized() bool { return true } func (b *builtinArithmeticPlusIntSig) vecEvalInt(input *chunk.Chunk, result *chunk.Column) error { n := input.NumRows() lh, err := b.bufAllocator.get(types.ETInt, n) if err != nil { return err } defer b.bufAllocator.put(lh) if err := b.args[0].VecEvalInt(b.ctx, input, lh); err != nil { return err } // reuse result as rh to avoid buf allocate if err := b.args[1].VecEvalInt(b.ctx, input, result); err != nil { return err } result.MergeNulls(lh) rh := result lhi64s := lh.Int64s() rhi64s := rh.Int64s() resulti64s := result.Int64s() isLHSUnsigned := mysql.HasUnsignedFlag(b.args[0].GetType().Flag) isRHSUnsigned := mysql.HasUnsignedFlag(b.args[1].GetType().Flag) switch { case isLHSUnsigned && isRHSUnsigned: err = b.plusUU(result, lhi64s, rhi64s, resulti64s) case isLHSUnsigned && !isRHSUnsigned: err = b.plusUS(result, lhi64s, rhi64s, resulti64s) case !isLHSUnsigned && isRHSUnsigned: err = b.plusSU(result, lhi64s, rhi64s, resulti64s) case !isLHSUnsigned && !isRHSUnsigned: err = b.plusSS(result, lhi64s, rhi64s, resulti64s) } return err } func (b *builtinArithmeticPlusIntSig) plusUU(result *chunk.Column, lhi64s, rhi64s, resulti64s []int64) error { for i := 0; i < len(lhi64s); i++ { if result.IsNull(i) { continue } lh, rh := lhi64s[i], rhi64s[i] if uint64(lh) > math.MaxUint64-uint64(rh) { return types.ErrOverflow.GenWithStackByArgs("BIGINT UNSIGNED", fmt.Sprintf("(%s + %s)", b.args[0].String(), b.args[1].String())) } resulti64s[i] = lh + rh } return nil } func (b *builtinArithmeticPlusIntSig) plusUS(result *chunk.Column, lhi64s, rhi64s, resulti64s []int64) error { for i := 0; i < len(lhi64s); i++ { if result.IsNull(i) { continue } lh, rh := lhi64s[i], rhi64s[i] if rh < 0 && uint64(-rh) > uint64(lh) { return types.ErrOverflow.GenWithStackByArgs("BIGINT UNSIGNED", fmt.Sprintf("(%s + %s)", b.args[0].String(), b.args[1].String())) } if rh > 0 && uint64(lh) > math.MaxUint64-uint64(lh) { return types.ErrOverflow.GenWithStackByArgs("BIGINT UNSIGNED", fmt.Sprintf("(%s + %s)", b.args[0].String(), b.args[1].String())) } resulti64s[i] = lh + rh } return nil } func (b *builtinArithmeticPlusIntSig) plusSU(result *chunk.Column, lhi64s, rhi64s, resulti64s []int64) error { for i := 0; i < len(lhi64s); i++ { if result.IsNull(i) { continue } lh, rh := lhi64s[i], rhi64s[i] if lh < 0 && uint64(-lh) > uint64(rh) { return types.ErrOverflow.GenWithStackByArgs("BIGINT UNSIGNED", fmt.Sprintf("(%s + %s)", b.args[0].String(), b.args[1].String())) } if lh > 0 && uint64(rh) > math.MaxUint64-uint64(lh) { return types.ErrOverflow.GenWithStackByArgs("BIGINT UNSIGNED", fmt.Sprintf("(%s + %s)", b.args[0].String(), b.args[1].String())) } resulti64s[i] = lh + rh } return nil } func (b *builtinArithmeticPlusIntSig) plusSS(result *chunk.Column, lhi64s, rhi64s, resulti64s []int64) error { for i := 0; i < len(lhi64s); i++ { if result.IsNull(i) { continue } lh, rh := lhi64s[i], rhi64s[i] if (lh > 0 && rh > math.MaxInt64-lh) || (lh < 0 && rh < math.MinInt64-lh) { return types.ErrOverflow.GenWithStackByArgs("BIGINT", fmt.Sprintf("(%s + %s)", b.args[0].String(), b.args[1].String())) } resulti64s[i] = lh + rh } return nil } func (b *builtinArithmeticPlusDecimalSig) vectorized() bool { return true } func (b *builtinArithmeticPlusDecimalSig) vecEvalDecimal(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalDecimal(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETDecimal, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalDecimal(b.ctx, input, buf); err != nil { return err } result.MergeNulls(buf) x := result.Decimals() y := buf.Decimals() to := new(types.MyDecimal) for i := 0; i < n; i++ { if result.IsNull(i) { continue } if err = types.DecimalAdd(&x[i], &y[i], to); err != nil { return err } x[i] = *to } return nil } func (b *builtinArithmeticMultiplyIntUnsignedSig) vectorized() bool { return true } func (b *builtinArithmeticMultiplyIntUnsignedSig) vecEvalInt(input *chunk.Chunk, result *chunk.Column) error { if err := b.args[0].VecEvalInt(b.ctx, input, result); err != nil { return err } n := input.NumRows() buf, err := b.bufAllocator.get(types.ETInt, n) if err != nil { return err } defer b.bufAllocator.put(buf) if err := b.args[1].VecEvalInt(b.ctx, input, buf); err != nil { return err } x := result.Uint64s() y := buf.Uint64s() result.MergeNulls(buf) var res uint64 for i := 0; i < n; i++ { if result.IsNull(i) { continue } res = x[i] * y[i] if x[i] != 0 && res/x[i] != y[i] { return types.ErrOverflow.GenWithStackByArgs("BIGINT UNSIGNED", fmt.Sprintf("(%s * %s)", b.args[0].String(), b.args[1].String())) } x[i] = res } return nil }