Files
doris/be/src/exprs/cast_functions.cpp
ZHAO Chun a99a49a444 Add bitamp_to_string function (#2731)
This CL changes:

1. add function bitmap_to_string and bitmap_from_string, which will
 convert a bitmap to/from string which contains all bit in bitmap
2. add function murmur_hash3_32, which will compute murmur hash for
input strings
3. make the function cast float to string the same with user result
logic
2020-01-13 12:31:37 +08:00

374 lines
13 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/cast_functions.h"
#include <cmath>
#include "exprs/anyval_util.h"
#include "runtime/datetime_value.h"
#include "runtime/string_value.h"
#include "util/string_parser.hpp"
#include "string_functions.h"
#include "util/mysql_global.h"
namespace doris {
void CastFunctions::init() {
}
// The maximum number of characters need to represent a floating-point number (float or
// double) as a string. 24 = 17 (maximum significant digits) + 1 (decimal point) + 1 ('E')
// + 3 (exponent digits) + 2 (negative signs) (see http://stackoverflow.com/a/1701085)
const int MAX_FLOAT_CHARS = 24;
#define CAST_FUNCTION(from_type, to_type, type_name) \
to_type CastFunctions::cast_to_##type_name(FunctionContext* ctx, const from_type& val) { \
if (val.is_null) return to_type::null(); \
return to_type(val.val); \
}
CAST_FUNCTION(TinyIntVal, BooleanVal, boolean_val)
CAST_FUNCTION(SmallIntVal, BooleanVal, boolean_val)
CAST_FUNCTION(IntVal, BooleanVal, boolean_val)
CAST_FUNCTION(BigIntVal, BooleanVal, boolean_val)
CAST_FUNCTION(LargeIntVal, BooleanVal, boolean_val)
CAST_FUNCTION(FloatVal, BooleanVal, boolean_val)
CAST_FUNCTION(DoubleVal, BooleanVal, boolean_val)
CAST_FUNCTION(BooleanVal, TinyIntVal, tiny_int_val)
CAST_FUNCTION(SmallIntVal, TinyIntVal, tiny_int_val)
CAST_FUNCTION(IntVal, TinyIntVal, tiny_int_val)
CAST_FUNCTION(BigIntVal, TinyIntVal, tiny_int_val)
CAST_FUNCTION(LargeIntVal, TinyIntVal, tiny_int_val)
CAST_FUNCTION(FloatVal, TinyIntVal, tiny_int_val)
CAST_FUNCTION(DoubleVal, TinyIntVal, tiny_int_val)
CAST_FUNCTION(BooleanVal, SmallIntVal, small_int_val)
CAST_FUNCTION(TinyIntVal, SmallIntVal, small_int_val)
CAST_FUNCTION(IntVal, SmallIntVal, small_int_val)
CAST_FUNCTION(BigIntVal, SmallIntVal, small_int_val)
CAST_FUNCTION(LargeIntVal, SmallIntVal, small_int_val)
CAST_FUNCTION(FloatVal, SmallIntVal, small_int_val)
CAST_FUNCTION(DoubleVal, SmallIntVal, small_int_val)
CAST_FUNCTION(BooleanVal, IntVal, int_val)
CAST_FUNCTION(TinyIntVal, IntVal, int_val)
CAST_FUNCTION(SmallIntVal, IntVal, int_val)
CAST_FUNCTION(BigIntVal, IntVal, int_val)
CAST_FUNCTION(LargeIntVal, IntVal, int_val)
CAST_FUNCTION(FloatVal, IntVal, int_val)
CAST_FUNCTION(DoubleVal, IntVal, int_val)
CAST_FUNCTION(BooleanVal, BigIntVal, big_int_val)
CAST_FUNCTION(TinyIntVal, BigIntVal, big_int_val)
CAST_FUNCTION(SmallIntVal, BigIntVal, big_int_val)
CAST_FUNCTION(IntVal, BigIntVal, big_int_val)
CAST_FUNCTION(LargeIntVal, BigIntVal, big_int_val)
CAST_FUNCTION(FloatVal, BigIntVal, big_int_val)
CAST_FUNCTION(DoubleVal, BigIntVal, big_int_val)
CAST_FUNCTION(BooleanVal, LargeIntVal, large_int_val)
CAST_FUNCTION(TinyIntVal, LargeIntVal, large_int_val)
CAST_FUNCTION(SmallIntVal, LargeIntVal, large_int_val)
CAST_FUNCTION(IntVal, LargeIntVal, large_int_val)
CAST_FUNCTION(BigIntVal, LargeIntVal, large_int_val)
CAST_FUNCTION(FloatVal, LargeIntVal, large_int_val)
CAST_FUNCTION(DoubleVal, LargeIntVal, large_int_val)
CAST_FUNCTION(BooleanVal, FloatVal, float_val)
CAST_FUNCTION(TinyIntVal, FloatVal, float_val)
CAST_FUNCTION(SmallIntVal, FloatVal, float_val)
CAST_FUNCTION(IntVal, FloatVal, float_val)
CAST_FUNCTION(BigIntVal, FloatVal, float_val)
CAST_FUNCTION(LargeIntVal, FloatVal, float_val)
CAST_FUNCTION(DoubleVal, FloatVal, float_val)
CAST_FUNCTION(BooleanVal, DoubleVal, double_val)
CAST_FUNCTION(TinyIntVal, DoubleVal, double_val)
CAST_FUNCTION(SmallIntVal, DoubleVal, double_val)
CAST_FUNCTION(IntVal, DoubleVal, double_val)
CAST_FUNCTION(BigIntVal, DoubleVal, double_val)
CAST_FUNCTION(LargeIntVal, DoubleVal, double_val)
CAST_FUNCTION(FloatVal, DoubleVal, double_val)
#define CAST_FROM_STRING(num_type, type_name, native_type, string_parser_fn) \
num_type CastFunctions::cast_to_##type_name(FunctionContext* ctx, const StringVal& val) { \
if (val.is_null) return num_type::null(); \
StringParser::ParseResult result; \
num_type ret; \
ret.val = StringParser::string_parser_fn<native_type>( \
reinterpret_cast<char*>(val.ptr), val.len, &result); \
if (UNLIKELY(result != StringParser::PARSE_SUCCESS || std::isnan(ret.val) || std::isinf(ret.val))) { \
return num_type::null(); \
} \
return ret; \
}
#define CAST_FROM_STRINGS() \
CAST_FROM_STRING(TinyIntVal, tiny_int_val, int8_t, string_to_int);\
CAST_FROM_STRING(SmallIntVal, small_int_val, int16_t, string_to_int);\
CAST_FROM_STRING(IntVal, int_val, int32_t, string_to_int);\
CAST_FROM_STRING(BigIntVal, big_int_val, int64_t, string_to_int);\
CAST_FROM_STRING(LargeIntVal, large_int_val, __int128, string_to_int);\
CAST_FROM_STRING(FloatVal, float_val, float, string_to_float);\
CAST_FROM_STRING(DoubleVal, double_val, double, string_to_float);
CAST_FROM_STRINGS();
// Special-case tinyint because boost thinks it's a char and handles it differently.
// e.g. '0' is written as an empty string.
StringVal CastFunctions::cast_to_string_val(FunctionContext* ctx, const TinyIntVal& val) {
if (val.is_null) {
return StringVal::null();
}
int64_t tmp_val = val.val;
return AnyValUtil::from_string_temp(ctx, std::to_string(tmp_val));
}
#define CAST_TO_STRING(num_type) \
StringVal CastFunctions::cast_to_string_val(FunctionContext* ctx, const num_type& val) { \
if (val.is_null) return StringVal::null(); \
return AnyValUtil::from_string_temp(ctx, std::to_string(val.val)); \
}
CAST_TO_STRING(BooleanVal);
CAST_TO_STRING(SmallIntVal);
CAST_TO_STRING(IntVal);
CAST_TO_STRING(BigIntVal);
StringVal CastFunctions::cast_to_string_val(FunctionContext* ctx, const LargeIntVal& val) {
if (val.is_null) {
return StringVal::null();
}
char buf[64];
int len = 64;
char* d = LargeIntValue::to_string(val.val, buf, &len);
return AnyValUtil::from_buffer_temp(ctx, d, len);
}
template<typename T>
int float_to_string(T value, char* buf);
template<>
int float_to_string<float>(float value, char* buf) {
return FloatToBuffer(value, MAX_FLOAT_STR_LENGTH + 2, buf);
}
template<>
int float_to_string<double>(double value, char* buf) {
return DoubleToBuffer(value, MAX_DOUBLE_STR_LENGTH + 2, buf);
}
#define CAST_FLOAT_TO_STRING(float_type, format) \
StringVal CastFunctions::cast_to_string_val(FunctionContext* ctx, const float_type& val) { \
if (val.is_null) return StringVal::null(); \
/* val.val could be -nan, return "nan" instead */ \
if (std::isnan(val.val)) return StringVal("nan"); \
/* Add 1 to MAX_FLOAT_CHARS since snprintf adds a trailing '\0' */ \
StringVal sv(ctx, MAX_DOUBLE_STR_LENGTH + 2); \
if (UNLIKELY(sv.is_null)) { \
return sv; \
} \
const FunctionContext::TypeDesc& returnType = ctx->get_return_type(); \
if (returnType.len > 0) { \
sv.len = snprintf(reinterpret_cast<char*>(sv.ptr), sv.len, format, val.val); \
DCHECK_GT(sv.len, 0); \
DCHECK_LE(sv.len, MAX_FLOAT_CHARS); \
AnyValUtil::TruncateIfNecessary(returnType, &sv); \
} else if (returnType.len == -1) { \
char buf[MAX_DOUBLE_STR_LENGTH + 2]; \
sv.len = float_to_string(val.val, buf); \
memcpy(sv.ptr, buf, sv.len); \
} else { \
DCHECK(false); \
} \
return sv; \
}
// Floats have up to 9 significant digits, doubles up to 17
// (see http://en.wikipedia.org/wiki/Single-precision_floating-point_format
// and http://en.wikipedia.org/wiki/Double-precision_floating-point_format)
CAST_FLOAT_TO_STRING(FloatVal, "%.9g");
CAST_FLOAT_TO_STRING(DoubleVal, "%.17g");
StringVal CastFunctions::cast_to_string_val(FunctionContext* ctx, const DateTimeVal& val) {
if (val.is_null) {
return StringVal::null();
}
DateTimeValue tv = DateTimeValue::from_datetime_val(val);
StringVal sv = StringVal::create_temp_string_val(ctx, 64);
sv.len = tv.to_string((char*)sv.ptr) - (char*)sv.ptr - 1;
return sv;
}
StringVal CastFunctions::cast_to_string_val(FunctionContext* ctx, const StringVal& val) {
if (val.is_null) return StringVal::null();
StringVal sv;
sv.ptr = val.ptr;
sv.len = val.len;
const FunctionContext::TypeDesc& result_type = ctx->get_return_type();
if (result_type.len > 0) {
AnyValUtil::TruncateIfNecessary(result_type, &sv);
}
return sv;
}
#if 0
StringVal CastFunctions::CastToChar(FunctionContext* ctx, const StringVal& val) {
if (val.is_null) return StringVal::null();
const FunctionContext::TypeDesc& type = ctx->GetReturnType();
DCHECK(type.type == FunctionContext::TYPE_FIXED_BUFFER);
DCHECK_GE(type.len, 1);
char* cptr;
if (type.len > val.len) {
cptr = reinterpret_cast<char*>(ctx->impl()->AllocateLocal(type.len));
memcpy(cptr, val.ptr, min(type.len, val.len));
StringValue::PadWithSpaces(cptr, type.len, val.len);
} else {
cptr = reinterpret_cast<char*>(val.ptr);
}
StringVal sv;
sv.ptr = reinterpret_cast<uint8_t*>(cptr);
sv.len = type.len;
return sv;
}
#endif
#define CAST_FROM_DATETIME(to_type, type_name) \
to_type CastFunctions::cast_to_##type_name( \
FunctionContext* ctx, const DateTimeVal& val) { \
if (val.is_null) return to_type::null(); \
DateTimeValue tv = DateTimeValue::from_datetime_val(val); \
return to_type(tv.to_int64()); \
}
CAST_FROM_DATETIME(BooleanVal, boolean_val);
CAST_FROM_DATETIME(TinyIntVal, tiny_int_val);
CAST_FROM_DATETIME(SmallIntVal, small_int_val);
CAST_FROM_DATETIME(IntVal, int_val);
CAST_FROM_DATETIME(BigIntVal, big_int_val);
CAST_FROM_DATETIME(LargeIntVal, large_int_val);
CAST_FROM_DATETIME(FloatVal, float_val);
CAST_FROM_DATETIME(DoubleVal, double_val);
#define CAST_TO_DATETIME(from_type) \
DateTimeVal CastFunctions::cast_to_datetime_val(FunctionContext* ctx, \
const from_type& val) { \
if (val.is_null) return DateTimeVal::null(); \
DateTimeValue date_value; \
if (!date_value.from_date_int64(val.val)) return DateTimeVal::null(); \
date_value.to_datetime(); \
DateTimeVal result; \
date_value.to_datetime_val(&result); \
return result; \
}
#define CAST_TO_DATETIMES() \
CAST_TO_DATETIME(TinyIntVal);\
CAST_TO_DATETIME(SmallIntVal);\
CAST_TO_DATETIME(IntVal);\
CAST_TO_DATETIME(BigIntVal);\
CAST_TO_DATETIME(FloatVal);\
CAST_TO_DATETIME(DoubleVal);
CAST_TO_DATETIMES();
#define CAST_TO_DATE(from_type) \
DateTimeVal CastFunctions::cast_to_date_val(FunctionContext* ctx, \
const from_type& val) { \
if (val.is_null) return DateTimeVal::null(); \
DateTimeValue date_value; \
if (!date_value.from_date_int64(val.val)) return DateTimeVal::null(); \
date_value.cast_to_date(); \
DateTimeVal result; \
date_value.to_datetime_val(&result); \
return result; \
}
#define CAST_TO_DATES() \
CAST_TO_DATE(TinyIntVal);\
CAST_TO_DATE(SmallIntVal);\
CAST_TO_DATE(IntVal);\
CAST_TO_DATE(BigIntVal);\
CAST_TO_DATE(FloatVal);\
CAST_TO_DATE(DoubleVal);
CAST_TO_DATES();
DateTimeVal CastFunctions::cast_to_datetime_val(
FunctionContext* ctx, const DateTimeVal& val) {
if (val.is_null) {
return DateTimeVal::null();
}
DateTimeValue date_value = DateTimeValue::from_datetime_val(val);
date_value.to_datetime();
// Return null if 'val' did not parse
DateTimeVal result;
date_value.to_datetime_val(&result);
return result;
}
DateTimeVal CastFunctions::cast_to_datetime_val(
FunctionContext* ctx, const StringVal& val) {
if (val.is_null) {
return DateTimeVal::null();
}
DateTimeValue date_value;
if (!date_value.from_date_str((char*)val.ptr, val.len)) {
return DateTimeVal::null();
}
date_value.to_datetime();
// Return null if 'val' did not parse
DateTimeVal result;
date_value.to_datetime_val(&result);
return result;
}
DateTimeVal CastFunctions::cast_to_date_val(
FunctionContext* ctx, const DateTimeVal& val) {
if (val.is_null) {
return DateTimeVal::null();
}
DateTimeValue date_value = DateTimeValue::from_datetime_val(val);
date_value.cast_to_date();
// Return null if 'val' did not parse
DateTimeVal result;
date_value.to_datetime_val(&result);
return result;
}
DateTimeVal CastFunctions::cast_to_date_val(
FunctionContext* ctx, const StringVal& val) {
if (val.is_null) {
return DateTimeVal::null();
}
DateTimeValue date_value;
if (!date_value.from_date_str((char*)val.ptr, val.len)) {
return DateTimeVal::null();
}
date_value.cast_to_date();
// Return null if 'val' did not parse
DateTimeVal result;
date_value.to_datetime_val(&result);
return result;
}
}