patch 4.0
This commit is contained in:
@ -19,42 +19,40 @@
|
||||
#include "ob_expr_instr.h"
|
||||
#include "ob_expr_util.h"
|
||||
#include "share/object/ob_obj_cast.h"
|
||||
#include "sql/parser/ob_item_type.h"
|
||||
#include "objit/common/ob_item_type.h"
|
||||
#include "sql/session/ob_sql_session_info.h"
|
||||
|
||||
using namespace oceanbase::common;
|
||||
using namespace oceanbase::sql;
|
||||
namespace oceanbase {
|
||||
namespace sql {
|
||||
namespace oceanbase
|
||||
{
|
||||
namespace sql
|
||||
{
|
||||
|
||||
ObExprInstr::ObExprInstr(ObIAllocator& alloc)
|
||||
ObExprInstr::ObExprInstr(ObIAllocator &alloc)
|
||||
: ObLocationExprOperator(alloc, T_FUN_SYS_INSTR, N_INSTR, 2, NOT_ROW_DIMENSION)
|
||||
{
|
||||
need_charset_convert_ = false;
|
||||
}
|
||||
|
||||
ObExprInstr::~ObExprInstr()
|
||||
{}
|
||||
ObExprInstr::~ObExprInstr() {}
|
||||
|
||||
int ObExprInstr::calc_result2(ObObj& result, const ObObj& obj1, const ObObj& obj2, ObExprCtx& expr_ctx) const
|
||||
{
|
||||
return ObLocationExprOperator::calc_result2(result, obj2, obj1, expr_ctx);
|
||||
}
|
||||
|
||||
int ObExprInstr::calc_mysql_instr_expr(const ObExpr& expr, ObEvalCtx& ctx, ObDatum& res_datum)
|
||||
int ObExprInstr::calc_mysql_instr_expr(const ObExpr &expr, ObEvalCtx &ctx, ObDatum &res_datum)
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
if (OB_UNLIKELY(2 != expr.arg_cnt_) || OB_ISNULL(expr.args_) || OB_ISNULL(expr.args_[0]) ||
|
||||
OB_ISNULL(expr.args_[1])) {
|
||||
if (OB_UNLIKELY(2 != expr.arg_cnt_) || OB_ISNULL(expr.args_) ||
|
||||
OB_ISNULL(expr.args_[0]) || OB_ISNULL(expr.args_[1])) {
|
||||
ret = OB_ERR_UNEXPECTED;
|
||||
LOG_WARN("invalid expr", K(ret), K(expr));
|
||||
} else if (OB_FAIL(ObLocationExprOperator::calc_(expr, *expr.args_[1], *expr.args_[0], ctx, res_datum))) {
|
||||
} else if (OB_FAIL(ObLocationExprOperator::calc_(expr, *expr.args_[1], *expr.args_[0],
|
||||
ctx, res_datum))) {
|
||||
LOG_WARN("ObLocationExprOperator::calc_ faied", K(ret));
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprInstr::cg_expr(ObExprCGCtx& op_cg_ctx, const ObRawExpr& raw_expr, ObExpr& rt_expr) const
|
||||
int ObExprInstr::cg_expr(ObExprCGCtx &op_cg_ctx, const ObRawExpr &raw_expr,
|
||||
ObExpr &rt_expr) const
|
||||
{
|
||||
UNUSED(op_cg_ctx);
|
||||
UNUSED(raw_expr);
|
||||
@ -64,18 +62,18 @@ int ObExprInstr::cg_expr(ObExprCGCtx& op_cg_ctx, const ObRawExpr& raw_expr, ObEx
|
||||
|
||||
/***** oracle ******/
|
||||
|
||||
ObExprOracleInstr::ObExprOracleInstr(ObIAllocator& alloc)
|
||||
: ObLocationExprOperator(alloc, T_FUN_SYS_INSTR, N_INSTR, MORE_THAN_ONE, NOT_ROW_DIMENSION)
|
||||
{}
|
||||
ObExprOracleInstr::ObExprOracleInstr(ObIAllocator &alloc)
|
||||
: ObLocationExprOperator(alloc, T_FUN_SYS_INSTR, N_INSTR, MORE_THAN_ONE, NOT_ROW_DIMENSION) {}
|
||||
|
||||
ObExprOracleInstr::~ObExprOracleInstr()
|
||||
{}
|
||||
ObExprOracleInstr::~ObExprOracleInstr() {}
|
||||
|
||||
int ObExprOracleInstr::calc_result_type2(
|
||||
ObExprResType& type, ObExprResType& type1, ObExprResType& type2, common::ObExprTypeCtx& type_ctx) const
|
||||
int ObExprOracleInstr::calc_result_type2(ObExprResType &type,
|
||||
ObExprResType &type1,
|
||||
ObExprResType &type2,
|
||||
common::ObExprTypeCtx &type_ctx) const
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
const ObSQLSessionInfo* session = type_ctx.get_session();
|
||||
const ObSQLSessionInfo *session = type_ctx.get_session();
|
||||
if (OB_ISNULL(session)) {
|
||||
ret = OB_ERR_UNEXPECTED;
|
||||
LOG_WARN("session is NULL", K(ret));
|
||||
@ -89,14 +87,19 @@ int ObExprOracleInstr::calc_result_type2(
|
||||
OZ(params.push_back(&type1));
|
||||
ObExprResType tmp_type;
|
||||
OZ(aggregate_string_type_and_charset_oracle(*session, params, tmp_type));
|
||||
// 如果两个参数的cs type不一致,计算的结果是有问题的
|
||||
// 虽然这里会对参数的cs type进行转换,但是instr返回的结果是以字符为单位的,cs type的转换
|
||||
// 不会影响instr的结果
|
||||
OZ(params.push_back(&type2));
|
||||
OZ(deduce_string_param_calc_type_and_charset(*session, tmp_type, params));
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprOracleInstr::calc_result_typeN(
|
||||
ObExprResType& type, ObExprResType* type_array, int64_t param_num, ObExprTypeCtx& type_ctx) const
|
||||
int ObExprOracleInstr::calc_result_typeN(ObExprResType &type,
|
||||
ObExprResType *type_array,
|
||||
int64_t param_num,
|
||||
ObExprTypeCtx &type_ctx) const
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
if (OB_UNLIKELY(param_num < 2 || param_num > 4)) {
|
||||
@ -108,67 +111,47 @@ int ObExprOracleInstr::calc_result_typeN(
|
||||
} else if (OB_FAIL(calc_result_type2(type, type_array[0], type_array[1], type_ctx))) {
|
||||
LOG_WARN("fail calc result type", K(param_num), K(ret));
|
||||
} else {
|
||||
// 如果用户指定了 position、occurrence,则转换成 Number
|
||||
// 如果没有指定,则在运行时用一个 int 指定为默认值 1
|
||||
if (3 == param_num) {
|
||||
type_array[2].set_calc_type(ObNumberType); // position
|
||||
type_array[2].set_calc_type(ObNumberType); // position
|
||||
} else if (4 == param_num) {
|
||||
type_array[2].set_calc_type(ObNumberType); // position
|
||||
type_array[3].set_calc_type(ObNumberType); // occurrence
|
||||
type_array[2].set_calc_type(ObNumberType); // position
|
||||
type_array[3].set_calc_type(ObNumberType); // occurrence
|
||||
}
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprOracleInstr::calc_resultN(
|
||||
ObObj& result, const common::ObObj* param_array, int64_t param_num, common::ObExprCtx& expr_ctx) const
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
if (OB_ISNULL(expr_ctx.calc_buf_)) {
|
||||
ret = OB_INVALID_ARGUMENT;
|
||||
LOG_WARN("the pointer is null", K(expr_ctx.calc_buf_), K(ret));
|
||||
} else if (OB_UNLIKELY(param_num < 2 || param_num > 4)) {
|
||||
ret = OB_ERR_PARAM_SIZE;
|
||||
LOG_WARN("invalid param count", K(param_num), K(ret));
|
||||
} else if (param_array[0].is_null() || param_array[1].is_null()) {
|
||||
result.set_null();
|
||||
} else if (param_array[0].is_clob() && (0 == param_array[0].get_string().length())) {
|
||||
number::ObNumber num;
|
||||
if (OB_FAIL(num.from((uint64_t)(0), *(expr_ctx.calc_buf_)))) {
|
||||
LOG_WARN("get nmb from int failed", K(ret));
|
||||
} else {
|
||||
result.set_number(num);
|
||||
}
|
||||
} else if (2 == param_num) {
|
||||
ObObj position(static_cast<int64_t>(1));
|
||||
ObObj occurrence(static_cast<int64_t>(1));
|
||||
ret = calc(result, param_array[0], param_array[1], position, occurrence, expr_ctx);
|
||||
} else if (3 == param_num) {
|
||||
ObObj occurrence(static_cast<int64_t>(1));
|
||||
ret = calc(result, param_array[0], param_array[1], param_array[2], occurrence, expr_ctx);
|
||||
} else {
|
||||
ret = calc(result, param_array[0], param_array[1], param_array[2], param_array[3], expr_ctx);
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprOracleInstr::calc(ObObj& result, const ObObj& haystack, const ObObj& needle, const ObObj& position,
|
||||
const ObObj& occurrence, ObExprCtx& expr_ctx)
|
||||
int ObExprOracleInstr::calc(ObObj &result,
|
||||
const ObObj &haystack,
|
||||
const ObObj &needle,
|
||||
const ObObj &position,
|
||||
const ObObj &occurrence,
|
||||
ObExprCtx &expr_ctx)
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
if (OB_ISNULL(expr_ctx.calc_buf_)) {
|
||||
ret = OB_NOT_INIT;
|
||||
LOG_WARN("expr_ctx.calc_buf_ is NULL", K(ret));
|
||||
} else if (OB_UNLIKELY(haystack.is_null() || needle.is_null() || position.is_null() || occurrence.is_null())) {
|
||||
} else if (OB_UNLIKELY(haystack.is_null() ||
|
||||
needle.is_null() ||
|
||||
position.is_null() ||
|
||||
occurrence.is_null())) {
|
||||
result.set_null();
|
||||
} else if (OB_UNLIKELY(!is_type_valid(haystack.get_type()) || !is_type_valid(needle.get_type()) ||
|
||||
!is_type_valid(position.get_type()) || !is_type_valid(occurrence.get_type()))) {
|
||||
} else if (OB_UNLIKELY(!is_type_valid(haystack.get_type())
|
||||
|| !is_type_valid(needle.get_type())
|
||||
|| !is_type_valid(position.get_type())
|
||||
|| !is_type_valid(occurrence.get_type()))) {
|
||||
ret = OB_INVALID_ARGUMENT;
|
||||
LOG_WARN("the param is not castable", K(haystack), K(needle), K(position), K(occurrence), K(ret));
|
||||
LOG_WARN("the param is not castable", K(haystack), K(needle), K(position),
|
||||
K(occurrence), K(ret));
|
||||
} else {
|
||||
int64_t pos = 0;
|
||||
int64_t occ = 0;
|
||||
const ObCollationType& calc_cs_type = (haystack.get_collation_type() == needle.get_collation_type())
|
||||
? haystack.get_collation_type()
|
||||
: CS_TYPE_INVALID;
|
||||
const ObCollationType &calc_cs_type = (haystack.get_collation_type() ==
|
||||
needle.get_collation_type()) ?
|
||||
haystack.get_collation_type() : CS_TYPE_INVALID;
|
||||
if (OB_UNLIKELY(!ObCharset::is_valid_collation(static_cast<int64_t>(calc_cs_type)))) {
|
||||
ret = OB_ERR_UNEXPECTED;
|
||||
LOG_WARN("calc cs_type is invalid", K(ret), K(calc_cs_type));
|
||||
@ -180,6 +163,9 @@ int ObExprOracleInstr::calc(ObObj& result, const ObObj& haystack, const ObObj& n
|
||||
ret = OB_ERR_ARGUMENT_OUT_OF_RANGE;
|
||||
LOG_USER_ERROR(OB_ERR_ARGUMENT_OUT_OF_RANGE, occ);
|
||||
} else if (OB_UNLIKELY(pos == 0)) {
|
||||
// 这里是为了兼容 Oracle 行为,Oracle 对于 pos = 0,总返回 0
|
||||
// 例外:select instr('abd','b',0,0) from dual;
|
||||
// 这种情况下因为 acc=0 ,上面已经报错,不返回 0
|
||||
number::ObNumber num;
|
||||
num.from(static_cast<int64_t>(0), *(expr_ctx.calc_buf_));
|
||||
result.set_number(num);
|
||||
@ -192,8 +178,9 @@ int ObExprOracleInstr::calc(ObObj& result, const ObObj& haystack, const ObObj& n
|
||||
ret = OB_ERR_UNEXPECTED;
|
||||
LOG_WARN("lob locator is null", K(ret));
|
||||
} else {
|
||||
const ObString lob_str(static_cast<int64_t>(haystack.get_lob_locator()->get_payload_length()),
|
||||
haystack.get_lob_locator()->get_payload_ptr());
|
||||
const ObString lob_str(
|
||||
static_cast<int64_t>(haystack.get_lob_locator()->get_payload_length()),
|
||||
haystack.get_lob_locator()->get_payload_ptr());
|
||||
str1 = lob_str;
|
||||
}
|
||||
} else {
|
||||
@ -202,7 +189,8 @@ int ObExprOracleInstr::calc(ObObj& result, const ObObj& haystack, const ObObj& n
|
||||
if (OB_FAIL(ret)) {
|
||||
} else if (pos > 0) {
|
||||
for (int64_t i = 0; i < occ; ++i) {
|
||||
idx = ObCharset::locate(calc_cs_type, str1.ptr(), str1.length(), str2.ptr(), str2.length(), pos);
|
||||
idx = ObCharset::locate(calc_cs_type, str1.ptr(), str1.length(), str2.ptr(),
|
||||
str2.length(), pos);
|
||||
if (idx <= 0) {
|
||||
break;
|
||||
} else {
|
||||
@ -210,7 +198,8 @@ int ObExprOracleInstr::calc(ObObj& result, const ObObj& haystack, const ObObj& n
|
||||
}
|
||||
}
|
||||
} else {
|
||||
if (OB_FAIL(slow_reverse_search(*(expr_ctx.calc_buf_), calc_cs_type, str1, str2, pos, occ, idx))) {
|
||||
if (OB_FAIL(slow_reverse_search(*(expr_ctx.calc_buf_), calc_cs_type, str1,
|
||||
str2, pos, occ, idx))) {
|
||||
LOG_WARN("slow_reverse_search failed", K(ret), K(str1), K(str2), K(occ));
|
||||
}
|
||||
}
|
||||
@ -227,16 +216,31 @@ int ObExprOracleInstr::calc(ObObj& result, const ObObj& haystack, const ObObj& n
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprOracleInstr::slow_reverse_search(ObIAllocator& alloc, const ObCollationType& cs_type, const ObString& str1,
|
||||
const ObString& str2, int64_t neg_start, int64_t occ, uint32_t& idx)
|
||||
int ObExprOracleInstr::slow_reverse_search(ObIAllocator &alloc,
|
||||
const ObCollationType &cs_type,
|
||||
const ObString &str1,
|
||||
const ObString &str2,
|
||||
int64_t neg_start,
|
||||
int64_t occ,
|
||||
uint32_t &idx)
|
||||
{
|
||||
/* TODO: 应该采用更加高效的 kmp 算法。
|
||||
* 但是,同时考虑到要支持各种字符集 (pos表示字符偏移,而不是字节偏移)
|
||||
* 所以必须使用支持字符集的 kmp 算法,有一定工作量
|
||||
*
|
||||
* kmp 算法可以参考 ObExprSubstringIndex 类中的实现,它不需要考虑
|
||||
* 字符集合collation,所以用 byte compare 的算法就足够
|
||||
*
|
||||
* 下面采用的算法是从后到前逐个比较
|
||||
* 这个算法比较低效,复杂度为 O(str2.length() * str1.length())
|
||||
*/
|
||||
int ret = OB_SUCCESS;
|
||||
size_t str1_len_in_char = ObCharset::strlen_char(cs_type, str1.ptr(), str1.length());
|
||||
size_t str2_len_in_char = ObCharset::strlen_char(cs_type, str2.ptr(), str2.length());
|
||||
size_t str1_len_in_char = ObCharset::strlen_char(cs_type, str1.ptr(), str1.length());
|
||||
size_t str2_len_in_char = ObCharset::strlen_char(cs_type, str2.ptr(), str2.length());
|
||||
// return 0 if start_idx_in_char less than 0.
|
||||
int64_t start_idx_in_char = static_cast<int64_t>(str1_len_in_char) + neg_start;
|
||||
ObFixedArray<size_t, ObIAllocator> str1_byte_num(&alloc, str1_len_in_char);
|
||||
ObFixedArray<size_t, ObIAllocator> str1_byte_offset(&alloc, str1_len_in_char + 1);
|
||||
ObFixedArray<size_t, ObIAllocator> str1_byte_num(&alloc, str1_len_in_char);
|
||||
ObFixedArray<size_t, ObIAllocator> str1_byte_offset(&alloc, str1_len_in_char+1);
|
||||
|
||||
if (0 <= neg_start) {
|
||||
ret = OB_INVALID_ARGUMENT;
|
||||
@ -244,49 +248,55 @@ int ObExprOracleInstr::slow_reverse_search(ObIAllocator& alloc, const ObCollatio
|
||||
} else if (0 > start_idx_in_char) {
|
||||
idx = 0;
|
||||
LOG_DEBUG("neg_start is too large, return 0", K(ret), K(idx));
|
||||
} else if (OB_FAIL(ObExprUtil::get_mb_str_info(str1, cs_type, str1_byte_num, str1_byte_offset))) {
|
||||
} else if (OB_FAIL(ObExprUtil::get_mb_str_info(str1, cs_type,
|
||||
str1_byte_num, str1_byte_offset))) {
|
||||
LOG_WARN("get_mb_str_info failed", K(ret), K(str1), K(cs_type), K(str1_len_in_char));
|
||||
} else if (str1_byte_num.count() + 1 != str1_byte_offset.count()) {
|
||||
} else if (str1_byte_num.count()+1 != str1_byte_offset.count()) {
|
||||
ret = OB_ERR_UNEXPECTED;
|
||||
LOG_WARN("size of str1_byte_num and size of str1_byte_offset should be same",
|
||||
K(ret),
|
||||
K(str1_byte_num),
|
||||
K(str1_byte_offset));
|
||||
K(ret), K(str1_byte_num), K(str1_byte_offset));
|
||||
} else if (start_idx_in_char >= str1_byte_num.count()) {
|
||||
ret = OB_ERR_UNEXPECTED;
|
||||
LOG_WARN(
|
||||
"start_idx_in_char should less than size of str1_byte_num", K(ret), K(start_idx_in_char), K(str1_byte_num));
|
||||
LOG_WARN("start_idx_in_char should less than size of str1_byte_num", K(ret),
|
||||
K(start_idx_in_char),
|
||||
K(str1_byte_num));
|
||||
} else {
|
||||
start_idx_in_char = std::min(start_idx_in_char,
|
||||
std::max(0L, static_cast<int64_t>(str1_len_in_char) - static_cast<int64_t>(str2_len_in_char)));
|
||||
for (; start_idx_in_char >= 0; --start_idx_in_char) {
|
||||
// 尽可能定位到最合适的地方
|
||||
start_idx_in_char = std::min(start_idx_in_char, std::max(0L,
|
||||
static_cast<int64_t>(str1_len_in_char) -
|
||||
static_cast<int64_t>(str2_len_in_char)));
|
||||
for ( ; start_idx_in_char >= 0; --start_idx_in_char) {
|
||||
size_t cur_byte_offset = str1_byte_offset[start_idx_in_char];
|
||||
int64_t compare_len = std::min(static_cast<int64_t>(str1.length()) - static_cast<int64_t>(cur_byte_offset),
|
||||
static_cast<int64_t>(str2.length()));
|
||||
int64_t compare_len = std::min(static_cast<int64_t>(str1.length()) -
|
||||
static_cast<int64_t>(cur_byte_offset),
|
||||
static_cast<int64_t>(str2.length()));
|
||||
idx = ObCharset::instr(cs_type,
|
||||
str1.ptr() + cur_byte_offset,
|
||||
compare_len, /* only compare limited bytes */
|
||||
str2.ptr(),
|
||||
str2.length());
|
||||
str1.ptr() + cur_byte_offset,
|
||||
compare_len, /* only compare limited bytes */
|
||||
str2.ptr(),
|
||||
str2.length());
|
||||
if (idx > 0 && (--occ == 0)) {
|
||||
// 调整成相对于 str1 起点的偏移值
|
||||
idx = idx + static_cast<uint32_t>(cur_byte_offset);
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (OB_SUCCESS == ret) {
|
||||
if (0 < occ) {
|
||||
if (0 < occ) { // 匹配次数不足
|
||||
idx = 0;
|
||||
} else {
|
||||
if (0 != idx) {
|
||||
// ObCharset::instr() return idx which starts with 1
|
||||
// ObCharset::instr()返回的idx是以1开始的
|
||||
--idx;
|
||||
typename ObFixedArray<size_t, ObIAllocator>::iterator iter =
|
||||
std::find(str1_byte_offset.begin(), str1_byte_offset.end(), idx);
|
||||
std::find(str1_byte_offset.begin(), str1_byte_offset.end(), idx);
|
||||
if (iter == str1_byte_offset.end()) {
|
||||
// @shaoge 表明遇到了非法的字符
|
||||
ret = OB_ERR_UNEXPECTED;
|
||||
LOG_WARN("cannot find char idx by byte idx", K(ret), K(idx), K(str1_byte_offset), K(str1), K(str2));
|
||||
LOG_WARN("cannot find char idx by byte idx", K(ret), K(idx), K(str1_byte_offset),
|
||||
K(str1), K(str2));
|
||||
} else {
|
||||
// start with 1
|
||||
// 最终结果要以1开始
|
||||
idx = iter - str1_byte_offset.begin() + 1;
|
||||
}
|
||||
}
|
||||
@ -296,19 +306,19 @@ int ObExprOracleInstr::slow_reverse_search(ObIAllocator& alloc, const ObCollatio
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprOracleInstr::calc_oracle_instr_arg(const ObExpr& expr, ObEvalCtx& ctx, bool& is_null, ObDatum*& haystack,
|
||||
ObDatum*& needle, int64_t& pos_int, int64_t& occ_int, ObCollationType& calc_cs_type)
|
||||
// haystack/needle没有直接传递ObString作为参数,因为想保证ObDatum结果的const属性
|
||||
int ObExprOracleInstr::calc_oracle_instr_arg(const ObExpr &expr, ObEvalCtx &ctx,
|
||||
bool &is_null,
|
||||
ObDatum *&haystack, ObDatum *&needle, int64_t &pos_int,
|
||||
int64_t &occ_int, ObCollationType &calc_cs_type)
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
ObDatum* pos = NULL;
|
||||
ObDatum* occ = NULL;
|
||||
ObDatum *pos = NULL;
|
||||
ObDatum *occ = NULL;
|
||||
is_null = false;
|
||||
pos_int = 1;
|
||||
occ_int = 1;
|
||||
if (OB_UNLIKELY(expr.arg_cnt_ < 2 || expr.arg_cnt_ > 4)) {
|
||||
ret = OB_ERR_UNEXPECTED;
|
||||
LOG_WARN("unexpected arg cnt", K(ret), K(expr.arg_cnt_));
|
||||
} else if (OB_FAIL(expr.args_[0]->eval(ctx, haystack))) {
|
||||
if (OB_FAIL(expr.args_[0]->eval(ctx, haystack))) {
|
||||
LOG_WARN("eval arg 0 failed", K(ret));
|
||||
} else if (haystack->is_null()) {
|
||||
is_null = true;
|
||||
@ -359,18 +369,19 @@ int ObExprOracleInstr::calc_oracle_instr_arg(const ObExpr& expr, ObEvalCtx& ctx,
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprOracleInstr::calc_oracle_instr_expr(const ObExpr& expr, ObEvalCtx& ctx, ObDatum& res_datum)
|
||||
int ObExprOracleInstr::calc_oracle_instr_expr(const ObExpr &expr, ObEvalCtx &ctx,
|
||||
ObDatum &res_datum)
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
// instr(haystack, needle, pos, occ);
|
||||
ObDatum* haystack = NULL;
|
||||
ObDatum* needle = NULL;
|
||||
ObDatum *haystack = NULL;
|
||||
ObDatum *needle = NULL;
|
||||
bool is_null = false;
|
||||
int64_t pos_int = 1;
|
||||
int64_t occ_int = 1;
|
||||
ObCollationType calc_cs_type = CS_TYPE_INVALID;
|
||||
if (OB_FAIL(ObExprOracleInstr::calc_oracle_instr_arg(
|
||||
expr, ctx, is_null, haystack, needle, pos_int, occ_int, calc_cs_type))) {
|
||||
if (OB_FAIL(ObExprOracleInstr::calc_oracle_instr_arg(expr, ctx, is_null, haystack,
|
||||
needle, pos_int, occ_int, calc_cs_type))) {
|
||||
LOG_WARN("calc_oracle_instr_arg failed", K(ret));
|
||||
} else if (is_null) {
|
||||
res_datum.set_null();
|
||||
@ -398,12 +409,13 @@ int ObExprOracleInstr::calc_oracle_instr_expr(const ObExpr& expr, ObEvalCtx& ctx
|
||||
res_datum.set_number(res_nmb);
|
||||
}
|
||||
} else {
|
||||
const ObString& str1 = haystack->get_string();
|
||||
const ObString& str2 = needle->get_string();
|
||||
const ObString &str1 = haystack->get_string();
|
||||
const ObString &str2 = needle->get_string();
|
||||
uint32_t idx = 0;
|
||||
if (pos_int > 0) {
|
||||
for (int64_t i = 0; i < occ_int; ++i) {
|
||||
idx = ObCharset::locate(calc_cs_type, str1.ptr(), str1.length(), str2.ptr(), str2.length(), pos_int);
|
||||
idx = ObCharset::locate(calc_cs_type, str1.ptr(), str1.length(),
|
||||
str2.ptr(), str2.length(),pos_int);
|
||||
if (idx <= 0) {
|
||||
break;
|
||||
} else {
|
||||
@ -411,10 +423,12 @@ int ObExprOracleInstr::calc_oracle_instr_expr(const ObExpr& expr, ObEvalCtx& ctx
|
||||
}
|
||||
}
|
||||
} else {
|
||||
ObIAllocator& tmp_alloc = ctx.get_reset_tmp_alloc();
|
||||
if (OB_FAIL(
|
||||
ObExprOracleInstr::slow_reverse_search(tmp_alloc, calc_cs_type, str1, str2, pos_int, occ_int, idx))) {
|
||||
LOG_WARN("slow_reverse_search failed", K(ret), K(calc_cs_type), K(str1), K(str2), K(pos_int), K(occ_int));
|
||||
ObEvalCtx::TempAllocGuard alloc_guard(ctx);
|
||||
ObIAllocator &tmp_alloc = alloc_guard.get_allocator();
|
||||
if (OB_FAIL(ObExprOracleInstr::slow_reverse_search(tmp_alloc, calc_cs_type, str1,
|
||||
str2, pos_int, occ_int, idx))) {
|
||||
LOG_WARN("slow_reverse_search failed", K(ret), K(calc_cs_type), K(str1), K(str2),
|
||||
K(pos_int), K(occ_int));
|
||||
}
|
||||
}
|
||||
if (OB_SUCC(ret)) {
|
||||
@ -431,14 +445,19 @@ int ObExprOracleInstr::calc_oracle_instr_expr(const ObExpr& expr, ObEvalCtx& ctx
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprOracleInstr::cg_expr(ObExprCGCtx& expr_cg_ctx, const ObRawExpr& raw_expr, ObExpr& rt_expr) const
|
||||
int ObExprOracleInstr::cg_expr(ObExprCGCtx &expr_cg_ctx, const ObRawExpr &raw_expr,
|
||||
ObExpr &rt_expr) const
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
UNUSED(expr_cg_ctx);
|
||||
UNUSED(raw_expr);
|
||||
if (OB_UNLIKELY(rt_expr.arg_cnt_ < 2 || rt_expr.arg_cnt_ > 4)) {
|
||||
ret = OB_ERR_UNEXPECTED;
|
||||
LOG_WARN("unexpected arg cnt", K(ret), K(rt_expr.arg_cnt_));
|
||||
}
|
||||
rt_expr.eval_func_ = calc_oracle_instr_expr;
|
||||
return ret;
|
||||
}
|
||||
|
||||
} // namespace sql
|
||||
} // namespace oceanbase
|
||||
} //namespace sql
|
||||
} //namespace oceanbase
|
||||
|
||||
Reference in New Issue
Block a user