Files
doris/be/src/exprs/decimalv2_operators.cpp
2020-10-20 09:28:57 +08:00

220 lines
7.0 KiB
C++

// Licensed to the Apache Software Foundation (ASF) under one
// or more contributor license agreements. See the NOTICE file
// distributed with this work for additional information
// regarding copyright ownership. The ASF licenses this file
// to you 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, WITHOUT WARRANTIES OR CONDITIONS OF ANY
// KIND, either express or implied. See the License for the
// specific language governing permissions and limitations
// under the License.
#include "exprs/decimalv2_operators.h"
#include <iomanip>
#include <sstream>
#include <math.h>
#include "exprs/anyval_util.h"
#include "exprs/case_expr.h"
#include "exprs/expr.h"
#include "runtime/tuple_row.h"
// #include "util/decimal_util.h"
#include "util/string_parser.hpp"
namespace doris {
void DecimalV2Operators::init() {
}
#define CAST_INT_TO_DECIMAL(from_type) \
DecimalV2Val DecimalV2Operators::cast_to_decimalv2_val( \
FunctionContext* context, const from_type& val) { \
if (val.is_null) return DecimalV2Val::null(); \
DecimalV2Value dv(val.val, 0);\
DecimalV2Val result;\
dv.to_decimal_val(&result);\
return result;\
}
#define CAST_INT_TO_DECIMALS() \
CAST_INT_TO_DECIMAL(TinyIntVal);\
CAST_INT_TO_DECIMAL(SmallIntVal);\
CAST_INT_TO_DECIMAL(IntVal);\
CAST_INT_TO_DECIMAL(BigIntVal);\
CAST_INT_TO_DECIMAL(LargeIntVal);\
CAST_INT_TO_DECIMALS();
DecimalV2Val DecimalV2Operators::cast_to_decimalv2_val(
FunctionContext* context, const FloatVal& val) {
if (val.is_null) {
return DecimalV2Val::null();
}
DecimalV2Value dv;
dv.assign_from_float(val.val);
DecimalV2Val result;
dv.to_decimal_val(&result);
return result;
}
DecimalV2Val DecimalV2Operators::cast_to_decimalv2_val(
FunctionContext* context, const DoubleVal& val) {
if (val.is_null) {
return DecimalV2Val::null();
}
DecimalV2Value dv;
dv.assign_from_double(val.val);
DecimalV2Val result;
dv.to_decimal_val(&result);
return result;
}
DecimalV2Val DecimalV2Operators::cast_to_decimalv2_val(
FunctionContext* context, const DateTimeVal& val) {
if (val.is_null) {
return DecimalV2Val::null();
}
DateTimeValue dt_value = DateTimeValue::from_datetime_val(val);
DecimalV2Value dv(dt_value.to_int64(), 0);
DecimalV2Val result;
dv.to_decimal_val(&result);
return result;
}
DecimalV2Val DecimalV2Operators::cast_to_decimalv2_val(
FunctionContext* context, const StringVal& val) {
if (val.is_null) {
return DecimalV2Val::null();
}
DecimalV2Value dv;
if (dv.parse_from_str((const char*)val.ptr, val.len)) {
return DecimalV2Val::null();
}
DecimalV2Val result;
dv.to_decimal_val(&result);
return result;
}
#define CAST_DECIMAL_TO_INT(to_type, type_name) \
to_type DecimalV2Operators::cast_to_##type_name( \
FunctionContext* context, const DecimalV2Val& val) { \
if (val.is_null) return to_type::null(); \
DecimalV2Value dv = DecimalV2Value::from_decimal_val(val); \
return to_type(dv);\
}
#define CAST_FROM_DECIMAL() \
CAST_DECIMAL_TO_INT(BooleanVal, boolean_val);\
CAST_DECIMAL_TO_INT(TinyIntVal, tiny_int_val);\
CAST_DECIMAL_TO_INT(SmallIntVal, small_int_val);\
CAST_DECIMAL_TO_INT(IntVal, int_val);\
CAST_DECIMAL_TO_INT(BigIntVal, big_int_val);\
CAST_DECIMAL_TO_INT(LargeIntVal, large_int_val);\
CAST_DECIMAL_TO_INT(FloatVal, float_val);\
CAST_DECIMAL_TO_INT(DoubleVal, double_val);
CAST_FROM_DECIMAL();
StringVal DecimalV2Operators::cast_to_string_val(
FunctionContext* ctx, const DecimalV2Val& val) {
if (val.is_null) {
return StringVal::null();
}
const DecimalV2Value& dv = DecimalV2Value::from_decimal_val(val);
return AnyValUtil::from_string_temp(ctx, dv.to_string());
}
DateTimeVal DecimalV2Operators::cast_to_datetime_val(
FunctionContext* context, const DecimalV2Val& val) {
if (val.is_null) {
return DateTimeVal::null();
}
const DecimalV2Value& dv = DecimalV2Value::from_decimal_val(val);
DateTimeValue dt;
if (!dt.from_date_int64(dv)) {
return DateTimeVal::null();
}
DateTimeVal result;
dt.to_datetime_val(&result);
return result;
}
DateTimeVal DecimalV2Operators::cast_to_date_val(
FunctionContext* context, const DecimalV2Val& val) {
if (val.is_null) {
return DateTimeVal::null();
}
// convert from DecimalV2Val to DecimalV2Value for calculation
const DecimalV2Value& dv = DecimalV2Value::from_decimal_val(val);
DateTimeValue dt;
if (!dt.from_date_int64(dv)) {
return DateTimeVal::null();
}
dt.cast_to_date();
DateTimeVal result;
dt.to_datetime_val(&result);
return result;
}
DecimalVal DecimalV2Operators::cast_to_decimal_val(
FunctionContext* context, const DecimalV2Val& val) {
if (val.is_null) return DecimalVal::null();
DecimalV2Value v2(val.val);
DecimalValue dv(v2.int_value(), v2.frac_value());
DecimalVal result;
dv.to_decimal_val(&result);
return result;
}
#define DECIMAL_ARITHMETIC_OP(FN_NAME, OP) \
DecimalV2Val DecimalV2Operators::FN_NAME##_decimalv2_val_decimalv2_val( \
FunctionContext* context, const DecimalV2Val& v1, const DecimalV2Val& v2) { \
if (v1.is_null || v2.is_null) return DecimalV2Val::null(); \
DecimalV2Value iv1 = DecimalV2Value::from_decimal_val(v1); \
DecimalV2Value iv2 = DecimalV2Value::from_decimal_val(v2); \
DecimalV2Value ir = iv1 OP iv2; \
DecimalV2Val result;\
ir.to_decimal_val(&result); \
return result; \
}
#define DECIMAL_ARITHMETIC_OPS() \
DECIMAL_ARITHMETIC_OP(add, +);\
DECIMAL_ARITHMETIC_OP(subtract, -);\
DECIMAL_ARITHMETIC_OP(multiply, *);\
DECIMAL_ARITHMETIC_OP(divide, /);\
DECIMAL_ARITHMETIC_OP(mod, %);\
DECIMAL_ARITHMETIC_OPS();
#define DECIMAL_BINARY_PREDICATE_NONNUMERIC_FN(NAME, OP) \
BooleanVal DecimalV2Operators::NAME##_decimalv2_val_decimalv2_val(\
FunctionContext* c, const DecimalV2Val& v1, const DecimalV2Val& v2) {\
if (v1.is_null || v2.is_null) return BooleanVal::null();\
DecimalV2Value iv1 = DecimalV2Value::from_decimal_val(v1);\
DecimalV2Value iv2 = DecimalV2Value::from_decimal_val(v2);\
return BooleanVal(iv1 OP iv2);\
}
#define BINARY_PREDICATE_NONNUMERIC_FNS() \
DECIMAL_BINARY_PREDICATE_NONNUMERIC_FN(eq, ==); \
DECIMAL_BINARY_PREDICATE_NONNUMERIC_FN(ne, !=); \
DECIMAL_BINARY_PREDICATE_NONNUMERIC_FN(gt, >); \
DECIMAL_BINARY_PREDICATE_NONNUMERIC_FN(lt, <); \
DECIMAL_BINARY_PREDICATE_NONNUMERIC_FN(ge, >=); \
DECIMAL_BINARY_PREDICATE_NONNUMERIC_FN(le, <=);
BINARY_PREDICATE_NONNUMERIC_FNS();
}