Files
doris/be/src/exprs/operators.cpp
Mingyu Chen 869fdff2f0 [refactor] add reference path for source file from impala (#9115)
According to the requirements of the APLv2, the referenced code needs to be marked with the path of the source code.
2022-04-20 12:29:57 +08:00

141 lines
7.2 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.
// This file is copied from
// https://github.com/apache/impala/blob/branch-2.9.0/be/src/exprs/operators.cc
// and modified by Doris
#include "exprs/operators.h"
#include "exprs/anyval_util.h"
#include "runtime/datetime_value.h"
#include "runtime/string_value.h"
#include "util/debug_util.h"
namespace doris {
void Operators::init() {}
#define BINARY_OP_FN(NAME, TYPE_NAME, TYPE, OP) \
TYPE Operators::NAME##_##TYPE_NAME##_##TYPE_NAME(FunctionContext* c, const TYPE& v1, \
const TYPE& v2) { \
if (v1.is_null || v2.is_null) return TYPE::null(); \
return TYPE(v1.val OP v2.val); \
}
#define BINARY_OP_CHECK_ZERO_FN(NAME, TYPE_NAME, TYPE, OP) \
TYPE Operators::NAME##_##TYPE_NAME##_##TYPE_NAME(FunctionContext* c, const TYPE& v1, \
const TYPE& v2) { \
if (v1.is_null || v2.is_null || v2.val == 0) return TYPE::null(); \
return TYPE(v1.val OP v2.val); \
}
#define BITNOT_FN(TYPE, TYPE_NAME) \
TYPE Operators::bitnot_##TYPE_NAME(FunctionContext* c, const TYPE& v) { \
if (v.is_null) return TYPE::null(); \
return TYPE(~v.val); \
}
// Return infinity if overflow.
#define FACTORIAL_FN(TYPE) \
BigIntVal Operators::Factorial_##TYPE(FunctionContext* c, const TYPE& v) { \
if (v.is_null) return BigIntVal::null(); \
int64_t fact = ComputeFactorial(v.val); \
if (fact < 0) { \
return BigIntVal::null(); \
} \
return BigIntVal(fact); \
}
#define BINARY_PREDICATE_NUMERIC_FN(NAME, TYPE_NAME, TYPE, OP) \
BooleanVal Operators::NAME##_##TYPE_NAME##_##TYPE_NAME(FunctionContext* c, const TYPE& v1, \
const TYPE& v2) { \
if (v1.is_null || v2.is_null) return BooleanVal::null(); \
return BooleanVal(v1.val OP v2.val); \
}
#define BINARY_PREDICATE_NONNUMERIC_FN(NAME, TYPE_NAME, FUNC_NAME, TYPE, DORIS_TYPE, OP) \
BooleanVal Operators::NAME##_##TYPE_NAME##_##TYPE_NAME(FunctionContext* c, const TYPE& v1, \
const TYPE& v2) { \
if (v1.is_null || v2.is_null) return BooleanVal::null(); \
DORIS_TYPE iv1 = DORIS_TYPE::from_##FUNC_NAME(v1); \
DORIS_TYPE iv2 = DORIS_TYPE::from_##FUNC_NAME(v2); \
return BooleanVal(iv1 OP iv2); \
}
#define BINARY_OP_NUMERIC_TYPES(NAME, OP) \
BINARY_OP_FN(NAME, tiny_int_val, TinyIntVal, OP); \
BINARY_OP_FN(NAME, small_int_val, SmallIntVal, OP); \
BINARY_OP_FN(NAME, int_val, IntVal, OP); \
BINARY_OP_FN(NAME, big_int_val, BigIntVal, OP); \
BINARY_OP_FN(NAME, large_int_val, LargeIntVal, OP); \
BINARY_OP_FN(NAME, float_val, FloatVal, OP); \
BINARY_OP_FN(NAME, double_val, DoubleVal, OP);
#define BINARY_OP_INT_TYPES(NAME, OP) \
BINARY_OP_FN(NAME, tiny_int_val, TinyIntVal, OP); \
BINARY_OP_FN(NAME, small_int_val, SmallIntVal, OP); \
BINARY_OP_FN(NAME, int_val, IntVal, OP); \
BINARY_OP_FN(NAME, big_int_val, BigIntVal, OP); \
BINARY_OP_FN(NAME, large_int_val, LargeIntVal, OP);
#define BINARY_OP_CHECK_ZERO_INT_TYPES(NAME, OP) \
BINARY_OP_CHECK_ZERO_FN(NAME, tiny_int_val, TinyIntVal, OP); \
BINARY_OP_CHECK_ZERO_FN(NAME, small_int_val, SmallIntVal, OP); \
BINARY_OP_CHECK_ZERO_FN(NAME, int_val, IntVal, OP); \
BINARY_OP_CHECK_ZERO_FN(NAME, big_int_val, BigIntVal, OP); \
BINARY_OP_CHECK_ZERO_FN(NAME, large_int_val, LargeIntVal, OP);
#define BINARY_PREDICATE_ALL_TYPES(NAME, OP) \
BINARY_PREDICATE_NUMERIC_FN(NAME, boolean_val, BooleanVal, OP); \
BINARY_PREDICATE_NUMERIC_FN(NAME, tiny_int_val, TinyIntVal, OP); \
BINARY_PREDICATE_NUMERIC_FN(NAME, small_int_val, SmallIntVal, OP); \
BINARY_PREDICATE_NUMERIC_FN(NAME, int_val, IntVal, OP); \
BINARY_PREDICATE_NUMERIC_FN(NAME, big_int_val, BigIntVal, OP); \
BINARY_PREDICATE_NUMERIC_FN(NAME, large_int_val, LargeIntVal, OP); \
BINARY_PREDICATE_NUMERIC_FN(NAME, float_val, FloatVal, OP); \
BINARY_PREDICATE_NUMERIC_FN(NAME, double_val, DoubleVal, OP); \
BINARY_PREDICATE_NONNUMERIC_FN(NAME, string_val, string_val, StringVal, StringValue, OP); \
BINARY_PREDICATE_NONNUMERIC_FN(NAME, datetime_val, datetime_val, DateTimeVal, DateTimeValue, \
OP);
BINARY_OP_NUMERIC_TYPES(add, +);
BINARY_OP_NUMERIC_TYPES(subtract, -);
BINARY_OP_NUMERIC_TYPES(multiply, *);
BINARY_OP_CHECK_ZERO_FN(divide, double_val, DoubleVal, /);
BINARY_OP_CHECK_ZERO_INT_TYPES(int_divide, /);
BINARY_OP_CHECK_ZERO_INT_TYPES(mod, %);
// Bit operator
BINARY_OP_INT_TYPES(bitand, &);
BINARY_OP_INT_TYPES(bitxor, ^);
BINARY_OP_INT_TYPES(bitor, |);
BITNOT_FN(TinyIntVal, tiny_int_val);
BITNOT_FN(SmallIntVal, small_int_val);
BITNOT_FN(IntVal, int_val);
BITNOT_FN(BigIntVal, big_int_val);
BITNOT_FN(LargeIntVal, large_int_val);
BINARY_PREDICATE_ALL_TYPES(eq, ==);
BINARY_PREDICATE_ALL_TYPES(ne, !=);
BINARY_PREDICATE_ALL_TYPES(gt, >);
BINARY_PREDICATE_ALL_TYPES(lt, <);
BINARY_PREDICATE_ALL_TYPES(ge, >=);
BINARY_PREDICATE_ALL_TYPES(le, <=);
} // namespace doris