// Copyright 2013 The ql Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSES/QL-LICENSE file. // Copyright 2015 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" "hash/crc32" "math" "math/rand" "strconv" "strings" "time" "github.com/cznic/mathutil" "github.com/hanchuanchuan/goInception/mysql" "github.com/hanchuanchuan/goInception/sessionctx" "github.com/hanchuanchuan/goInception/types" "github.com/hanchuanchuan/goInception/util/chunk" "github.com/pingcap/errors" "github.com/pingcap/tipb/go-tipb" ) var ( _ functionClass = &absFunctionClass{} _ functionClass = &roundFunctionClass{} _ functionClass = &ceilFunctionClass{} _ functionClass = &floorFunctionClass{} _ functionClass = &logFunctionClass{} _ functionClass = &log2FunctionClass{} _ functionClass = &log10FunctionClass{} _ functionClass = &randFunctionClass{} _ functionClass = &powFunctionClass{} _ functionClass = &convFunctionClass{} _ functionClass = &crc32FunctionClass{} _ functionClass = &signFunctionClass{} _ functionClass = &sqrtFunctionClass{} _ functionClass = &acosFunctionClass{} _ functionClass = &asinFunctionClass{} _ functionClass = &atanFunctionClass{} _ functionClass = &cosFunctionClass{} _ functionClass = &cotFunctionClass{} _ functionClass = °reesFunctionClass{} _ functionClass = &expFunctionClass{} _ functionClass = &piFunctionClass{} _ functionClass = &radiansFunctionClass{} _ functionClass = &sinFunctionClass{} _ functionClass = &tanFunctionClass{} _ functionClass = &truncateFunctionClass{} ) var ( _ builtinFunc = &builtinAbsRealSig{} _ builtinFunc = &builtinAbsIntSig{} _ builtinFunc = &builtinAbsUIntSig{} _ builtinFunc = &builtinAbsDecSig{} _ builtinFunc = &builtinRoundRealSig{} _ builtinFunc = &builtinRoundIntSig{} _ builtinFunc = &builtinRoundDecSig{} _ builtinFunc = &builtinRoundWithFracRealSig{} _ builtinFunc = &builtinRoundWithFracIntSig{} _ builtinFunc = &builtinRoundWithFracDecSig{} _ builtinFunc = &builtinCeilRealSig{} _ builtinFunc = &builtinCeilIntToDecSig{} _ builtinFunc = &builtinCeilIntToIntSig{} _ builtinFunc = &builtinCeilDecToIntSig{} _ builtinFunc = &builtinCeilDecToDecSig{} _ builtinFunc = &builtinFloorRealSig{} _ builtinFunc = &builtinFloorIntToDecSig{} _ builtinFunc = &builtinFloorIntToIntSig{} _ builtinFunc = &builtinFloorDecToIntSig{} _ builtinFunc = &builtinFloorDecToDecSig{} _ builtinFunc = &builtinLog1ArgSig{} _ builtinFunc = &builtinLog2ArgsSig{} _ builtinFunc = &builtinLog2Sig{} _ builtinFunc = &builtinLog10Sig{} _ builtinFunc = &builtinRandSig{} _ builtinFunc = &builtinRandWithSeedSig{} _ builtinFunc = &builtinPowSig{} _ builtinFunc = &builtinConvSig{} _ builtinFunc = &builtinCRC32Sig{} _ builtinFunc = &builtinSignSig{} _ builtinFunc = &builtinSqrtSig{} _ builtinFunc = &builtinAcosSig{} _ builtinFunc = &builtinAsinSig{} _ builtinFunc = &builtinAtan1ArgSig{} _ builtinFunc = &builtinAtan2ArgsSig{} _ builtinFunc = &builtinCosSig{} _ builtinFunc = &builtinCotSig{} _ builtinFunc = &builtinDegreesSig{} _ builtinFunc = &builtinExpSig{} _ builtinFunc = &builtinPISig{} _ builtinFunc = &builtinRadiansSig{} _ builtinFunc = &builtinSinSig{} _ builtinFunc = &builtinTanSig{} _ builtinFunc = &builtinTruncateIntSig{} _ builtinFunc = &builtinTruncateRealSig{} _ builtinFunc = &builtinTruncateDecimalSig{} ) type absFunctionClass struct { baseFunctionClass } func (c *absFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(c.verifyArgs(args)) } argFieldTp := args[0].GetType() argTp := argFieldTp.EvalType() if argTp != types.ETInt && argTp != types.ETDecimal { argTp = types.ETReal } bf := newBaseBuiltinFuncWithTp(ctx, args, argTp, argTp) if mysql.HasUnsignedFlag(argFieldTp.Flag) { bf.tp.Flag |= mysql.UnsignedFlag } if argTp == types.ETReal { bf.tp.Flen, bf.tp.Decimal = mysql.GetDefaultFieldLengthAndDecimal(mysql.TypeDouble) } else { bf.tp.Flen = argFieldTp.Flen bf.tp.Decimal = argFieldTp.Decimal } var sig builtinFunc switch argTp { case types.ETInt: if mysql.HasUnsignedFlag(argFieldTp.Flag) { sig = &builtinAbsUIntSig{bf} sig.setPbCode(tipb.ScalarFuncSig_AbsInt) } else { sig = &builtinAbsIntSig{bf} sig.setPbCode(tipb.ScalarFuncSig_AbsUInt) } case types.ETDecimal: sig = &builtinAbsDecSig{bf} sig.setPbCode(tipb.ScalarFuncSig_AbsDecimal) case types.ETReal: sig = &builtinAbsRealSig{bf} sig.setPbCode(tipb.ScalarFuncSig_AbsReal) default: panic("unexpected argTp") } return sig, nil } type builtinAbsRealSig struct { baseBuiltinFunc } func (b *builtinAbsRealSig) Clone() builtinFunc { newSig := &builtinAbsRealSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals ABS(value). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_abs func (b *builtinAbsRealSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return math.Abs(val), false, nil } type builtinAbsIntSig struct { baseBuiltinFunc } func (b *builtinAbsIntSig) Clone() builtinFunc { newSig := &builtinAbsIntSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals ABS(value). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_abs func (b *builtinAbsIntSig) evalInt(row chunk.Row) (int64, bool, error) { val, isNull, err := b.args[0].EvalInt(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } if val >= 0 { return val, false, nil } if val == math.MinInt64 { return 0, false, types.ErrOverflow.GenWithStackByArgs("BIGINT", fmt.Sprintf("abs(%d)", val)) } return -val, false, nil } type builtinAbsUIntSig struct { baseBuiltinFunc } func (b *builtinAbsUIntSig) Clone() builtinFunc { newSig := &builtinAbsUIntSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals ABS(value). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_abs func (b *builtinAbsUIntSig) evalInt(row chunk.Row) (int64, bool, error) { return b.args[0].EvalInt(b.ctx, row) } type builtinAbsDecSig struct { baseBuiltinFunc } func (b *builtinAbsDecSig) Clone() builtinFunc { newSig := &builtinAbsDecSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalDecimal evals ABS(value). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_abs func (b *builtinAbsDecSig) evalDecimal(row chunk.Row) (*types.MyDecimal, bool, error) { val, isNull, err := b.args[0].EvalDecimal(b.ctx, row) if isNull || err != nil { return nil, isNull, errors.Trace(err) } to := new(types.MyDecimal) if !val.IsNegative() { *to = *val } else { if err = types.DecimalSub(new(types.MyDecimal), val, to); err != nil { return nil, true, errors.Trace(err) } } return to, false, nil } func (c *roundFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(c.verifyArgs(args)) } argTp := args[0].GetType().EvalType() if argTp != types.ETInt && argTp != types.ETDecimal { argTp = types.ETReal } argTps := []types.EvalType{argTp} if len(args) > 1 { argTps = append(argTps, types.ETInt) } bf := newBaseBuiltinFuncWithTp(ctx, args, argTp, argTps...) argFieldTp := args[0].GetType() if mysql.HasUnsignedFlag(argFieldTp.Flag) { bf.tp.Flag |= mysql.UnsignedFlag } bf.tp.Flen = argFieldTp.Flen bf.tp.Decimal = calculateDecimal4RoundAndTruncate(ctx, args, argTp) var sig builtinFunc if len(args) > 1 { switch argTp { case types.ETInt: sig = &builtinRoundWithFracIntSig{bf} case types.ETDecimal: sig = &builtinRoundWithFracDecSig{bf} case types.ETReal: sig = &builtinRoundWithFracRealSig{bf} default: panic("unexpected argTp") } } else { switch argTp { case types.ETInt: sig = &builtinRoundIntSig{bf} case types.ETDecimal: sig = &builtinRoundDecSig{bf} case types.ETReal: sig = &builtinRoundRealSig{bf} default: panic("unexpected argTp") } } return sig, nil } // calculateDecimal4RoundAndTruncate calculates tp.decimals of round/truncate func. func calculateDecimal4RoundAndTruncate(ctx sessionctx.Context, args []Expression, retType types.EvalType) int { if retType == types.ETInt || len(args) <= 1 { return 0 } secondConst, secondIsConst := args[1].(*Constant) if !secondIsConst { return args[0].GetType().Decimal } argDec, isNull, err := secondConst.EvalInt(ctx, chunk.Row{}) if err != nil || isNull || argDec < 0 { return 0 } if argDec > mysql.MaxDecimalScale { return mysql.MaxDecimalScale } return int(argDec) } type builtinRoundRealSig struct { baseBuiltinFunc } func (b *builtinRoundRealSig) Clone() builtinFunc { newSig := &builtinRoundRealSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals ROUND(value). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_round func (b *builtinRoundRealSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return types.Round(val, 0), false, nil } type builtinRoundIntSig struct { baseBuiltinFunc } func (b *builtinRoundIntSig) Clone() builtinFunc { newSig := &builtinRoundIntSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals ROUND(value). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_round func (b *builtinRoundIntSig) evalInt(row chunk.Row) (int64, bool, error) { return b.args[0].EvalInt(b.ctx, row) } type builtinRoundDecSig struct { baseBuiltinFunc } func (b *builtinRoundDecSig) Clone() builtinFunc { newSig := &builtinRoundDecSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalDecimal evals ROUND(value). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_round func (b *builtinRoundDecSig) evalDecimal(row chunk.Row) (*types.MyDecimal, bool, error) { val, isNull, err := b.args[0].EvalDecimal(b.ctx, row) if isNull || err != nil { return nil, isNull, errors.Trace(err) } to := new(types.MyDecimal) if err = val.Round(to, 0, types.ModeHalfEven); err != nil { return nil, true, errors.Trace(err) } return to, false, nil } type builtinRoundWithFracRealSig struct { baseBuiltinFunc } func (b *builtinRoundWithFracRealSig) Clone() builtinFunc { newSig := &builtinRoundWithFracRealSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals ROUND(value, frac). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_round func (b *builtinRoundWithFracRealSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } frac, isNull, err := b.args[1].EvalInt(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return types.Round(val, int(frac)), false, nil } type builtinRoundWithFracIntSig struct { baseBuiltinFunc } func (b *builtinRoundWithFracIntSig) Clone() builtinFunc { newSig := &builtinRoundWithFracIntSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals ROUND(value, frac). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_round func (b *builtinRoundWithFracIntSig) evalInt(row chunk.Row) (int64, bool, error) { val, isNull, err := b.args[0].EvalInt(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } frac, isNull, err := b.args[1].EvalInt(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return int64(types.Round(float64(val), int(frac))), false, nil } type builtinRoundWithFracDecSig struct { baseBuiltinFunc } func (b *builtinRoundWithFracDecSig) Clone() builtinFunc { newSig := &builtinRoundWithFracDecSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalDecimal evals ROUND(value, frac). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_round func (b *builtinRoundWithFracDecSig) evalDecimal(row chunk.Row) (*types.MyDecimal, bool, error) { val, isNull, err := b.args[0].EvalDecimal(b.ctx, row) if isNull || err != nil { return nil, isNull, errors.Trace(err) } frac, isNull, err := b.args[1].EvalInt(b.ctx, row) if isNull || err != nil { return nil, isNull, errors.Trace(err) } to := new(types.MyDecimal) if err = val.Round(to, mathutil.Min(int(frac), b.tp.Decimal), types.ModeHalfEven); err != nil { return nil, true, errors.Trace(err) } return to, false, nil } type ceilFunctionClass struct { baseFunctionClass } func (c *ceilFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (sig builtinFunc, err error) { if err = c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } retTp, argTp := getEvalTp4FloorAndCeil(args[0]) bf := newBaseBuiltinFuncWithTp(ctx, args, retTp, argTp) setFlag4FloorAndCeil(bf.tp, args[0]) argFieldTp := args[0].GetType() bf.tp.Flen, bf.tp.Decimal = argFieldTp.Flen, 0 switch argTp { case types.ETInt: if retTp == types.ETInt { sig = &builtinCeilIntToIntSig{bf} sig.setPbCode(tipb.ScalarFuncSig_CeilIntToInt) } else { sig = &builtinCeilIntToDecSig{bf} sig.setPbCode(tipb.ScalarFuncSig_CeilIntToDec) } case types.ETDecimal: if retTp == types.ETInt { sig = &builtinCeilDecToIntSig{bf} sig.setPbCode(tipb.ScalarFuncSig_CeilDecToInt) } else { sig = &builtinCeilDecToDecSig{bf} sig.setPbCode(tipb.ScalarFuncSig_CeilDecToDec) } default: sig = &builtinCeilRealSig{bf} sig.setPbCode(tipb.ScalarFuncSig_CeilReal) } return sig, nil } type builtinCeilRealSig struct { baseBuiltinFunc } func (b *builtinCeilRealSig) Clone() builtinFunc { newSig := &builtinCeilRealSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinCeilRealSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_ceil func (b *builtinCeilRealSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return math.Ceil(val), false, nil } type builtinCeilIntToIntSig struct { baseBuiltinFunc } func (b *builtinCeilIntToIntSig) Clone() builtinFunc { newSig := &builtinCeilIntToIntSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals a builtinCeilIntToIntSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_ceil func (b *builtinCeilIntToIntSig) evalInt(row chunk.Row) (int64, bool, error) { return b.args[0].EvalInt(b.ctx, row) } type builtinCeilIntToDecSig struct { baseBuiltinFunc } func (b *builtinCeilIntToDecSig) Clone() builtinFunc { newSig := &builtinCeilIntToDecSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalDecimal evals a builtinCeilIntToDecSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_Ceil func (b *builtinCeilIntToDecSig) evalDecimal(row chunk.Row) (*types.MyDecimal, bool, error) { val, isNull, err := b.args[0].EvalInt(b.ctx, row) if isNull || err != nil { return nil, true, errors.Trace(err) } if mysql.HasUnsignedFlag(b.args[0].GetType().Flag) || val >= 0 { return types.NewDecFromUint(uint64(val)), false, nil } return types.NewDecFromInt(val), false, nil } type builtinCeilDecToIntSig struct { baseBuiltinFunc } func (b *builtinCeilDecToIntSig) Clone() builtinFunc { newSig := &builtinCeilDecToIntSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals a builtinCeilDecToIntSig. // Ceil receives func (b *builtinCeilDecToIntSig) evalInt(row chunk.Row) (int64, bool, error) { val, isNull, err := b.args[0].EvalDecimal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } // err here will only be ErrOverFlow(will never happen) or ErrTruncate(can be ignored). res, err := val.ToInt() if err == types.ErrTruncated { err = nil if !val.IsNegative() { res = res + 1 } } return res, false, errors.Trace(err) } type builtinCeilDecToDecSig struct { baseBuiltinFunc } func (b *builtinCeilDecToDecSig) Clone() builtinFunc { newSig := &builtinCeilDecToDecSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalDecimal evals a builtinCeilDecToDecSig. func (b *builtinCeilDecToDecSig) evalDecimal(row chunk.Row) (*types.MyDecimal, bool, error) { val, isNull, err := b.args[0].EvalDecimal(b.ctx, row) if isNull || err != nil { return nil, isNull, errors.Trace(err) } res := new(types.MyDecimal) if val.IsNegative() { err = val.Round(res, 0, types.ModeTruncate) return res, err != nil, errors.Trace(err) } err = val.Round(res, 0, types.ModeTruncate) if err != nil || res.Compare(val) == 0 { return res, err != nil, errors.Trace(err) } err = types.DecimalAdd(res, types.NewDecFromInt(1), res) return res, err != nil, errors.Trace(err) } type floorFunctionClass struct { baseFunctionClass } // getEvalTp4FloorAndCeil gets the types.EvalType of FLOOR and CEIL. func getEvalTp4FloorAndCeil(arg Expression) (retTp, argTp types.EvalType) { fieldTp := arg.GetType() retTp, argTp = types.ETInt, fieldTp.EvalType() switch argTp { case types.ETInt: if fieldTp.Tp == mysql.TypeLonglong { retTp = types.ETDecimal } case types.ETDecimal: if fieldTp.Flen-fieldTp.Decimal > mysql.MaxIntWidth-2 { // len(math.MaxInt64) - 1 retTp = types.ETDecimal } default: retTp, argTp = types.ETReal, types.ETReal } return retTp, argTp } // setFlag4FloorAndCeil sets return flag of FLOOR and CEIL. func setFlag4FloorAndCeil(tp *types.FieldType, arg Expression) { fieldTp := arg.GetType() if (fieldTp.Tp == mysql.TypeLong || fieldTp.Tp == mysql.TypeNewDecimal) && mysql.HasUnsignedFlag(fieldTp.Flag) { tp.Flag |= mysql.UnsignedFlag } // TODO: when argument type is timestamp, add not null flag. } func (c *floorFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (sig builtinFunc, err error) { if err = c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } retTp, argTp := getEvalTp4FloorAndCeil(args[0]) bf := newBaseBuiltinFuncWithTp(ctx, args, retTp, argTp) setFlag4FloorAndCeil(bf.tp, args[0]) bf.tp.Flen, bf.tp.Decimal = args[0].GetType().Flen, 0 switch argTp { case types.ETInt: if retTp == types.ETInt { sig = &builtinFloorIntToIntSig{bf} sig.setPbCode(tipb.ScalarFuncSig_FloorIntToInt) } else { sig = &builtinFloorIntToDecSig{bf} sig.setPbCode(tipb.ScalarFuncSig_FloorIntToDec) } case types.ETDecimal: if retTp == types.ETInt { sig = &builtinFloorDecToIntSig{bf} sig.setPbCode(tipb.ScalarFuncSig_FloorDecToInt) } else { sig = &builtinFloorDecToDecSig{bf} sig.setPbCode(tipb.ScalarFuncSig_FloorDecToDec) } default: sig = &builtinFloorRealSig{bf} sig.setPbCode(tipb.ScalarFuncSig_FloorReal) } return sig, nil } type builtinFloorRealSig struct { baseBuiltinFunc } func (b *builtinFloorRealSig) Clone() builtinFunc { newSig := &builtinFloorRealSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinFloorRealSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_floor func (b *builtinFloorRealSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return math.Floor(val), false, nil } type builtinFloorIntToIntSig struct { baseBuiltinFunc } func (b *builtinFloorIntToIntSig) Clone() builtinFunc { newSig := &builtinFloorIntToIntSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals a builtinFloorIntToIntSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_floor func (b *builtinFloorIntToIntSig) evalInt(row chunk.Row) (int64, bool, error) { return b.args[0].EvalInt(b.ctx, row) } type builtinFloorIntToDecSig struct { baseBuiltinFunc } func (b *builtinFloorIntToDecSig) Clone() builtinFunc { newSig := &builtinFloorIntToDecSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalDecimal evals a builtinFloorIntToDecSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_floor func (b *builtinFloorIntToDecSig) evalDecimal(row chunk.Row) (*types.MyDecimal, bool, error) { val, isNull, err := b.args[0].EvalInt(b.ctx, row) if isNull || err != nil { return nil, true, errors.Trace(err) } if mysql.HasUnsignedFlag(b.args[0].GetType().Flag) || val >= 0 { return types.NewDecFromUint(uint64(val)), false, nil } return types.NewDecFromInt(val), false, nil } type builtinFloorDecToIntSig struct { baseBuiltinFunc } func (b *builtinFloorDecToIntSig) Clone() builtinFunc { newSig := &builtinFloorDecToIntSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals a builtinFloorDecToIntSig. // floor receives func (b *builtinFloorDecToIntSig) evalInt(row chunk.Row) (int64, bool, error) { val, isNull, err := b.args[0].EvalDecimal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } // err here will only be ErrOverFlow(will never happen) or ErrTruncate(can be ignored). res, err := val.ToInt() if err == types.ErrTruncated { err = nil if val.IsNegative() { res-- } } return res, false, errors.Trace(err) } type builtinFloorDecToDecSig struct { baseBuiltinFunc } func (b *builtinFloorDecToDecSig) Clone() builtinFunc { newSig := &builtinFloorDecToDecSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalDecimal evals a builtinFloorDecToDecSig. func (b *builtinFloorDecToDecSig) evalDecimal(row chunk.Row) (*types.MyDecimal, bool, error) { val, isNull, err := b.args[0].EvalDecimal(b.ctx, row) if isNull || err != nil { return nil, true, errors.Trace(err) } res := new(types.MyDecimal) if !val.IsNegative() { err = val.Round(res, 0, types.ModeTruncate) return res, err != nil, errors.Trace(err) } err = val.Round(res, 0, types.ModeTruncate) if err != nil || res.Compare(val) == 0 { return res, err != nil, errors.Trace(err) } err = types.DecimalSub(res, types.NewDecFromInt(1), res) return res, err != nil, errors.Trace(err) } type logFunctionClass struct { baseFunctionClass } func (c *logFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } var ( sig builtinFunc bf baseBuiltinFunc argsLen = len(args) ) if argsLen == 1 { bf = newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) } else { bf = newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal, types.ETReal) } if argsLen == 1 { sig = &builtinLog1ArgSig{bf} } else { sig = &builtinLog2ArgsSig{bf} } return sig, nil } type builtinLog1ArgSig struct { baseBuiltinFunc } func (b *builtinLog1ArgSig) Clone() builtinFunc { newSig := &builtinLog1ArgSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinLog1ArgSig, corresponding to log(x). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_log func (b *builtinLog1ArgSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } if val <= 0 { return 0, true, nil } return math.Log(val), false, nil } type builtinLog2ArgsSig struct { baseBuiltinFunc } func (b *builtinLog2ArgsSig) Clone() builtinFunc { newSig := &builtinLog2ArgsSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinLog2ArgsSig, corresponding to log(b, x). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_log func (b *builtinLog2ArgsSig) evalReal(row chunk.Row) (float64, bool, error) { val1, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } val2, isNull, err := b.args[1].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } if val1 <= 0 || val1 == 1 || val2 <= 0 { return 0, true, nil } return math.Log(val2) / math.Log(val1), false, nil } type log2FunctionClass struct { baseFunctionClass } func (c *log2FunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinLog2Sig{bf} return sig, nil } type builtinLog2Sig struct { baseBuiltinFunc } func (b *builtinLog2Sig) Clone() builtinFunc { newSig := &builtinLog2Sig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinLog2Sig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_log2 func (b *builtinLog2Sig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } if val <= 0 { return 0, true, nil } return math.Log2(val), false, nil } type log10FunctionClass struct { baseFunctionClass } func (c *log10FunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinLog10Sig{bf} return sig, nil } type builtinLog10Sig struct { baseBuiltinFunc } func (b *builtinLog10Sig) Clone() builtinFunc { newSig := &builtinLog10Sig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinLog10Sig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_log10 func (b *builtinLog10Sig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } if val <= 0 { return 0, true, nil } return math.Log10(val), false, nil } type randFunctionClass struct { baseFunctionClass } func (c *randFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } var sig builtinFunc var argTps []types.EvalType if len(args) > 0 { argTps = []types.EvalType{types.ETInt} } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, argTps...) bt := bf if len(args) == 0 { sig = &builtinRandSig{bt, nil} } else { sig = &builtinRandWithSeedSig{bt, nil} } return sig, nil } type builtinRandSig struct { baseBuiltinFunc randGen *rand.Rand } func (b *builtinRandSig) Clone() builtinFunc { newSig := &builtinRandSig{randGen: b.randGen} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals RAND(). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_rand func (b *builtinRandSig) evalReal(row chunk.Row) (float64, bool, error) { if b.randGen == nil { b.randGen = rand.New(rand.NewSource(time.Now().UnixNano())) } return b.randGen.Float64(), false, nil } type builtinRandWithSeedSig struct { baseBuiltinFunc randGen *rand.Rand } func (b *builtinRandWithSeedSig) Clone() builtinFunc { newSig := &builtinRandWithSeedSig{randGen: b.randGen} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals RAND(N). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_rand func (b *builtinRandWithSeedSig) evalReal(row chunk.Row) (float64, bool, error) { seed, isNull, err := b.args[0].EvalInt(b.ctx, row) if err != nil { return 0, true, errors.Trace(err) } if b.randGen == nil { if isNull { // When seed is NULL, it is equal to RAND(). b.randGen = rand.New(rand.NewSource(time.Now().UnixNano())) } else { b.randGen = rand.New(rand.NewSource(seed)) } } return b.randGen.Float64(), false, nil } type powFunctionClass struct { baseFunctionClass } func (c *powFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal, types.ETReal) sig := &builtinPowSig{bf} return sig, nil } type builtinPowSig struct { baseBuiltinFunc } func (b *builtinPowSig) Clone() builtinFunc { newSig := &builtinPowSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals POW(x, y). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_pow func (b *builtinPowSig) evalReal(row chunk.Row) (float64, bool, error) { x, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } y, isNull, err := b.args[1].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } power := math.Pow(x, y) if math.IsInf(power, -1) || math.IsInf(power, 1) || math.IsNaN(power) { return 0, false, types.ErrOverflow.GenWithStackByArgs("DOUBLE", fmt.Sprintf("pow(%s, %s)", strconv.FormatFloat(x, 'f', -1, 64), strconv.FormatFloat(y, 'f', -1, 64))) } return power, false, nil } type roundFunctionClass struct { baseFunctionClass } type convFunctionClass struct { baseFunctionClass } func (c *convFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETString, types.ETString, types.ETInt, types.ETInt) bf.tp.Flen = 64 sig := &builtinConvSig{bf} return sig, nil } type builtinConvSig struct { baseBuiltinFunc } func (b *builtinConvSig) Clone() builtinFunc { newSig := &builtinConvSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalString evals CONV(N,from_base,to_base). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_conv. func (b *builtinConvSig) evalString(row chunk.Row) (res string, isNull bool, err error) { n, isNull, err := b.args[0].EvalString(b.ctx, row) if isNull || err != nil { return res, isNull, errors.Trace(err) } fromBase, isNull, err := b.args[1].EvalInt(b.ctx, row) if isNull || err != nil { return res, isNull, errors.Trace(err) } toBase, isNull, err := b.args[2].EvalInt(b.ctx, row) if isNull || err != nil { return res, isNull, errors.Trace(err) } var ( signed bool negative bool ignoreSign bool ) if fromBase < 0 { fromBase = -fromBase signed = true } if toBase < 0 { toBase = -toBase ignoreSign = true } if fromBase > 36 || fromBase < 2 || toBase > 36 || toBase < 2 { return res, true, nil } n = getValidPrefix(strings.TrimSpace(n), fromBase) if len(n) == 0 { return "0", false, nil } if n[0] == '-' { negative = true n = n[1:] } val, err := strconv.ParseUint(n, int(fromBase), 64) if err != nil { return res, false, types.ErrOverflow.GenWithStackByArgs("BIGINT UNSINGED", n) } if signed { if negative && val > -math.MinInt64 { val = -math.MinInt64 } if !negative && val > math.MaxInt64 { val = math.MaxInt64 } } if negative { val = -val } if int64(val) < 0 { negative = true } else { negative = false } if ignoreSign && negative { val = 0 - val } s := strconv.FormatUint(val, int(toBase)) if negative && ignoreSign { s = "-" + s } res = strings.ToUpper(s) return res, false, nil } type crc32FunctionClass struct { baseFunctionClass } func (c *crc32FunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETInt, types.ETString) bf.tp.Flen = 10 bf.tp.Flag |= mysql.UnsignedFlag sig := &builtinCRC32Sig{bf} return sig, nil } type builtinCRC32Sig struct { baseBuiltinFunc } func (b *builtinCRC32Sig) Clone() builtinFunc { newSig := &builtinCRC32Sig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals a CRC32(expr). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_crc32 func (b *builtinCRC32Sig) evalInt(row chunk.Row) (int64, bool, error) { x, isNull, err := b.args[0].EvalString(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } r := crc32.ChecksumIEEE([]byte(x)) return int64(r), false, nil } type signFunctionClass struct { baseFunctionClass } func (c *signFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETInt, types.ETReal) sig := &builtinSignSig{bf} return sig, nil } type builtinSignSig struct { baseBuiltinFunc } func (b *builtinSignSig) Clone() builtinFunc { newSig := &builtinSignSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals SIGN(v). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_sign func (b *builtinSignSig) evalInt(row chunk.Row) (int64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } if val > 0 { return 1, false, nil } else if val == 0 { return 0, false, nil } else { return -1, false, nil } } type sqrtFunctionClass struct { baseFunctionClass } func (c *sqrtFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinSqrtSig{bf} return sig, nil } type builtinSqrtSig struct { baseBuiltinFunc } func (b *builtinSqrtSig) Clone() builtinFunc { newSig := &builtinSqrtSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a SQRT(x). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_sqrt func (b *builtinSqrtSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } if val < 0 { return 0, true, nil } return math.Sqrt(val), false, nil } type acosFunctionClass struct { baseFunctionClass } func (c *acosFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinAcosSig{bf} return sig, nil } type builtinAcosSig struct { baseBuiltinFunc } func (b *builtinAcosSig) Clone() builtinFunc { newSig := &builtinAcosSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinAcosSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_acos func (b *builtinAcosSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } if val < -1 || val > 1 { return 0, true, nil } return math.Acos(val), false, nil } type asinFunctionClass struct { baseFunctionClass } func (c *asinFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinAsinSig{bf} return sig, nil } type builtinAsinSig struct { baseBuiltinFunc } func (b *builtinAsinSig) Clone() builtinFunc { newSig := &builtinAsinSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinAsinSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_asin func (b *builtinAsinSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } if val < -1 || val > 1 { return 0, true, nil } return math.Asin(val), false, nil } type atanFunctionClass struct { baseFunctionClass } func (c *atanFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } var ( sig builtinFunc bf baseBuiltinFunc argsLen = len(args) ) if argsLen == 1 { bf = newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) } else { bf = newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal, types.ETReal) } if argsLen == 1 { sig = &builtinAtan1ArgSig{bf} } else { sig = &builtinAtan2ArgsSig{bf} } return sig, nil } type builtinAtan1ArgSig struct { baseBuiltinFunc } func (b *builtinAtan1ArgSig) Clone() builtinFunc { newSig := &builtinAtan1ArgSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinAtan1ArgSig, corresponding to atan(x). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_atan func (b *builtinAtan1ArgSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return math.Atan(val), false, nil } type builtinAtan2ArgsSig struct { baseBuiltinFunc } func (b *builtinAtan2ArgsSig) Clone() builtinFunc { newSig := &builtinAtan2ArgsSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinAtan1ArgSig, corresponding to atan(y, x). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_atan func (b *builtinAtan2ArgsSig) evalReal(row chunk.Row) (float64, bool, error) { val1, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } val2, isNull, err := b.args[1].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return math.Atan2(val1, val2), false, nil } type cosFunctionClass struct { baseFunctionClass } func (c *cosFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinCosSig{bf} return sig, nil } type builtinCosSig struct { baseBuiltinFunc } func (b *builtinCosSig) Clone() builtinFunc { newSig := &builtinCosSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinCosSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_cos func (b *builtinCosSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return math.Cos(val), false, nil } type cotFunctionClass struct { baseFunctionClass } func (c *cotFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinCotSig{bf} return sig, nil } type builtinCotSig struct { baseBuiltinFunc } func (b *builtinCotSig) Clone() builtinFunc { newSig := &builtinCotSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinCotSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_cot func (b *builtinCotSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } tan := math.Tan(val) if tan != 0 { cot := 1 / tan if !math.IsInf(cot, 0) && !math.IsNaN(cot) { return cot, false, nil } } return 0, false, types.ErrOverflow.GenWithStackByArgs("DOUBLE", fmt.Sprintf("cot(%s)", strconv.FormatFloat(val, 'f', -1, 64))) } type degreesFunctionClass struct { baseFunctionClass } func (c *degreesFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinDegreesSig{bf} return sig, nil } type builtinDegreesSig struct { baseBuiltinFunc } func (b *builtinDegreesSig) Clone() builtinFunc { newSig := &builtinDegreesSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinDegreesSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_degrees func (b *builtinDegreesSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } res := val * 180 / math.Pi return res, false, nil } type expFunctionClass struct { baseFunctionClass } func (c *expFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinExpSig{bf} return sig, nil } type builtinExpSig struct { baseBuiltinFunc } func (b *builtinExpSig) Clone() builtinFunc { newSig := &builtinExpSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinExpSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_exp func (b *builtinExpSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } exp := math.Exp(val) if math.IsInf(exp, 0) || math.IsNaN(exp) { s := fmt.Sprintf("exp(%s)", strconv.FormatFloat(val, 'f', -1, 64)) return 0, false, types.ErrOverflow.GenWithStackByArgs("DOUBLE", s) } return exp, false, nil } type piFunctionClass struct { baseFunctionClass } func (c *piFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } var ( bf baseBuiltinFunc sig builtinFunc ) bf = newBaseBuiltinFuncWithTp(ctx, args, types.ETReal) bf.tp.Decimal = 6 bf.tp.Flen = 8 sig = &builtinPISig{bf} return sig, nil } type builtinPISig struct { baseBuiltinFunc } func (b *builtinPISig) Clone() builtinFunc { newSig := &builtinPISig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinPISig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_pi func (b *builtinPISig) evalReal(_ chunk.Row) (float64, bool, error) { return float64(math.Pi), false, nil } type radiansFunctionClass struct { baseFunctionClass } func (c *radiansFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinRadiansSig{bf} return sig, nil } type builtinRadiansSig struct { baseBuiltinFunc } func (b *builtinRadiansSig) Clone() builtinFunc { newSig := &builtinRadiansSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals RADIANS(X). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_radians func (b *builtinRadiansSig) evalReal(row chunk.Row) (float64, bool, error) { x, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return x * math.Pi / 180, false, nil } type sinFunctionClass struct { baseFunctionClass } func (c *sinFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinSinSig{bf} return sig, nil } type builtinSinSig struct { baseBuiltinFunc } func (b *builtinSinSig) Clone() builtinFunc { newSig := &builtinSinSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinSinSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_sin func (b *builtinSinSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return math.Sin(val), false, nil } type tanFunctionClass struct { baseFunctionClass } func (c *tanFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } bf := newBaseBuiltinFuncWithTp(ctx, args, types.ETReal, types.ETReal) sig := &builtinTanSig{bf} return sig, nil } type builtinTanSig struct { baseBuiltinFunc } func (b *builtinTanSig) Clone() builtinFunc { newSig := &builtinTanSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a builtinTanSig. // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_tan func (b *builtinTanSig) evalReal(row chunk.Row) (float64, bool, error) { val, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return math.Tan(val), false, nil } type truncateFunctionClass struct { baseFunctionClass } func (c *truncateFunctionClass) getFunction(ctx sessionctx.Context, args []Expression) (builtinFunc, error) { if err := c.verifyArgs(args); err != nil { return nil, errors.Trace(err) } argTp := args[0].GetType().EvalType() if argTp == types.ETTimestamp || argTp == types.ETDatetime || argTp == types.ETDuration || argTp == types.ETString { argTp = types.ETReal } bf := newBaseBuiltinFuncWithTp(ctx, args, argTp, argTp, types.ETInt) bf.tp.Decimal = calculateDecimal4RoundAndTruncate(ctx, args, argTp) bf.tp.Flen = args[0].GetType().Flen - args[0].GetType().Decimal + bf.tp.Decimal bf.tp.Flag |= args[0].GetType().Flag var sig builtinFunc switch argTp { case types.ETInt: sig = &builtinTruncateIntSig{bf} case types.ETReal: sig = &builtinTruncateRealSig{bf} case types.ETDecimal: sig = &builtinTruncateDecimalSig{bf} } return sig, nil } type builtinTruncateDecimalSig struct { baseBuiltinFunc } func (b *builtinTruncateDecimalSig) Clone() builtinFunc { newSig := &builtinTruncateDecimalSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalDecimal evals a TRUNCATE(X,D). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_truncate func (b *builtinTruncateDecimalSig) evalDecimal(row chunk.Row) (*types.MyDecimal, bool, error) { x, isNull, err := b.args[0].EvalDecimal(b.ctx, row) if isNull || err != nil { return nil, isNull, errors.Trace(err) } d, isNull, err := b.args[1].EvalInt(b.ctx, row) if isNull || err != nil { return nil, isNull, errors.Trace(err) } result := new(types.MyDecimal) if err := x.Round(result, mathutil.Min(int(d), b.getRetTp().Decimal), types.ModeTruncate); err != nil { return nil, true, errors.Trace(err) } return result, false, nil } type builtinTruncateRealSig struct { baseBuiltinFunc } func (b *builtinTruncateRealSig) Clone() builtinFunc { newSig := &builtinTruncateRealSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalReal evals a TRUNCATE(X,D). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_truncate func (b *builtinTruncateRealSig) evalReal(row chunk.Row) (float64, bool, error) { x, isNull, err := b.args[0].EvalReal(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } d, isNull, err := b.args[1].EvalInt(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } return types.Truncate(x, int(d)), false, nil } type builtinTruncateIntSig struct { baseBuiltinFunc } func (b *builtinTruncateIntSig) Clone() builtinFunc { newSig := &builtinTruncateIntSig{} newSig.cloneFrom(&b.baseBuiltinFunc) return newSig } // evalInt evals a TRUNCATE(X,D). // See https://dev.mysql.com/doc/refman/5.7/en/mathematical-functions.html#function_truncate func (b *builtinTruncateIntSig) evalInt(row chunk.Row) (int64, bool, error) { x, isNull, err := b.args[0].EvalInt(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } d, isNull, err := b.args[1].EvalInt(b.ctx, row) if isNull || err != nil { return 0, isNull, errors.Trace(err) } floatX := float64(x) return int64(types.Truncate(floatX, int(d))), false, nil }