156 lines
5.8 KiB
C++
156 lines
5.8 KiB
C++
/**
|
|
* Copyright (c) 2021 OceanBase
|
|
* OceanBase CE is licensed under Mulan PubL v2.
|
|
* You can use this software according to the terms and conditions of the Mulan PubL v2.
|
|
* You may obtain a copy of Mulan PubL v2 at:
|
|
* http://license.coscl.org.cn/MulanPubL-2.0
|
|
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
|
|
* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
|
|
* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
|
|
* See the Mulan PubL v2 for more details.
|
|
*/
|
|
|
|
#define USING_LOG_PREFIX SQL_ENG
|
|
|
|
#include "sql/engine/expr/ob_expr_greatest.h"
|
|
#include "sql/engine/expr/ob_expr_operator.h"
|
|
#include "sql/engine/expr/ob_expr_less_than.h"
|
|
//#include "sql/engine/expr/ob_expr_promotion_util.h"
|
|
#include "sql/engine/expr/ob_expr_result_type_util.h"
|
|
#include "sql/session/ob_sql_session_info.h"
|
|
#include "lib/oblog/ob_log.h"
|
|
#include "share/object/ob_obj_cast.h"
|
|
|
|
namespace oceanbase {
|
|
using namespace common;
|
|
namespace sql {
|
|
|
|
ObExprBaseGreatest::ObExprBaseGreatest(ObIAllocator& alloc, int32_t param_num,
|
|
ObExprOperatorType type /* T_FUN_SYS_GREATEST */, const char* name /* N_GREATEST */)
|
|
: ObExprBaseLeastGreatest(alloc, type, name, param_num)
|
|
{}
|
|
|
|
ObExprBaseGreatest::~ObExprBaseGreatest()
|
|
{}
|
|
|
|
int ObExprBaseGreatest::calc_result_typeN(
|
|
ObExprResType& type, ObExprResType* types_stack, int64_t param_num, common::ObExprTypeCtx& type_ctx) const
|
|
{
|
|
UNUSED(type);
|
|
UNUSED(types_stack);
|
|
UNUSED(param_num);
|
|
UNUSED(type_ctx);
|
|
|
|
return OB_SUCCESS;
|
|
}
|
|
|
|
int ObExprBaseGreatest::calc_resultN(
|
|
ObObj& result, const ObObj* objs_stack, int64_t param_num, ObExprCtx& expr_ctx) const
|
|
{
|
|
return ObMinMaxExprOperator::calc_(result, objs_stack, param_num, result_type_, expr_ctx, CO_GT, need_cast_);
|
|
}
|
|
|
|
int ObExprBaseGreatest::calc(
|
|
ObObj& result, const ObObj* objs_stack, int64_t param_num, const ObExprResType& result_type, ObExprCtx& expr_ctx)
|
|
{
|
|
return ObMinMaxExprOperator::calc_(result, objs_stack, param_num, result_type, expr_ctx, CO_GT, true);
|
|
}
|
|
|
|
ObExprGreatestMySQL::ObExprGreatestMySQL(ObIAllocator& alloc) : ObExprBaseGreatest(alloc, MORE_THAN_ONE)
|
|
{}
|
|
|
|
ObExprGreatestMySQL::~ObExprGreatestMySQL()
|
|
{}
|
|
|
|
/* Greatest behavior in MySQL:
|
|
*. cached_field_type is derived using standard logic agg_field_type() as the result column type.
|
|
*. The calculation process of collation: If the parameter contains a numeric type,
|
|
it is binary, otherwise it is calculated by the rules
|
|
*. The calculation process is to find an intermediate result according to get_cmp_type(),
|
|
and then all values are transferred to the intermediate result for comparison operations
|
|
In other words, the comparison process has nothing to do with cached_field_type. The logic of get_calc_type() is:
|
|
only when the parameters are STRING, The return type is STRING, otherwise the return type is numeric.
|
|
When returning a numeric type, all parameters are converted to numeric types and then compared.
|
|
*/
|
|
int ObExprGreatestMySQL::calc_result_typeN(
|
|
ObExprResType& type, ObExprResType* types, int64_t param_num, ObExprTypeCtx& type_ctx) const
|
|
{
|
|
return calc_result_typeN_mysql(type, types, param_num, type_ctx);
|
|
}
|
|
|
|
ObExprGreatestMySQLInner::ObExprGreatestMySQLInner(ObIAllocator& alloc)
|
|
: ObExprBaseGreatest(alloc, MORE_THAN_ONE, T_FUN_SYS_GREATEST_INNER, "")
|
|
{}
|
|
|
|
ObExprGreatestMySQLInner::~ObExprGreatestMySQLInner()
|
|
{}
|
|
|
|
int ObExprGreatestMySQLInner::calc_result_typeN(
|
|
ObExprResType& type, ObExprResType* types, int64_t param_num, ObExprTypeCtx& type_ctx) const
|
|
{
|
|
return calc_result_typeN_mysql(type, types, param_num, type_ctx);
|
|
}
|
|
|
|
ObExprOracleGreatest::ObExprOracleGreatest(ObIAllocator& alloc) : ObExprBaseGreatest(alloc, MORE_THAN_ZERO)
|
|
{}
|
|
|
|
ObExprOracleGreatest::~ObExprOracleGreatest()
|
|
{}
|
|
|
|
/** Oracle greatest behavior:
|
|
* 1. oracle allow greatest with only 1 param. MySQL does not allow 1
|
|
* 2. oracle return type is same with first param. MySQL will check all param and calc one type to
|
|
* return
|
|
*/
|
|
int ObExprOracleGreatest::calc_result_typeN(
|
|
ObExprResType& type, ObExprResType* types, int64_t param_num, ObExprTypeCtx& type_ctx) const
|
|
{
|
|
return calc_result_typeN_oracle(type, types, param_num, type_ctx);
|
|
}
|
|
|
|
int ObExprBaseGreatest::cg_expr(ObExprCGCtx& op_cg_ctx, const ObRawExpr& raw_expr, ObExpr& rt_expr) const
|
|
{
|
|
UNUSED(op_cg_ctx);
|
|
UNUSED(raw_expr);
|
|
int ret = OB_SUCCESS;
|
|
bool all_same_type = true;
|
|
const uint32_t param_num = rt_expr.arg_cnt_;
|
|
if (OB_UNLIKELY(is_oracle_mode() && param_num < 1) || OB_UNLIKELY(!is_oracle_mode() && param_num < 2) ||
|
|
OB_ISNULL(rt_expr.args_)) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("args_ is null or too few arguments", K(ret), K(rt_expr.args_), K(param_num));
|
|
} else {
|
|
if (!is_oracle_mode()) {
|
|
const uint32_t real_param_num = param_num / 3;
|
|
for (int64_t i = 0; OB_SUCC(ret) && i < real_param_num; i++) {
|
|
if (OB_FAIL(ObStaticEngineExprCG::replace_var_rt_expr(
|
|
rt_expr.args_[i], rt_expr.args_[i + real_param_num], &rt_expr, i + real_param_num))) {
|
|
LOG_WARN("replace var rt expr failed", K(ret));
|
|
} else if (OB_FAIL(ObStaticEngineExprCG::replace_var_rt_expr(
|
|
rt_expr.args_[i], rt_expr.args_[i + 2 * real_param_num], &rt_expr, i + 2 * real_param_num))) {
|
|
LOG_WARN("replace var rt expr failed", K(ret));
|
|
}
|
|
}
|
|
}
|
|
for (int i = 0; OB_SUCC(ret) && i < param_num; ++i) {
|
|
if (OB_ISNULL(rt_expr.args_[i])) {
|
|
ret = OB_ERR_UNEXPECTED;
|
|
LOG_WARN("child of expr is null", K(ret), K(i));
|
|
}
|
|
}
|
|
if (OB_SUCC(ret)) {
|
|
rt_expr.eval_func_ = ObExprBaseGreatest::calc_greatest;
|
|
}
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
// same type params
|
|
int ObExprBaseGreatest::calc_greatest(const ObExpr& expr, ObEvalCtx& ctx, ObDatum& expr_datum)
|
|
{
|
|
return ObExprBaseLeastGreatest::calc(expr, ctx, expr_datum, false);
|
|
}
|
|
|
|
} // namespace sql
|
|
} // namespace oceanbase
|