github-actions[bot] commented on code in PR #16438: URL: https://github.com/apache/doris/pull/16438#discussion_r1097037767
########## be/src/olap/types.h: ########## @@ -430,6 +432,332 @@ class ArrayTypeInfo : public TypeInfo { TypeInfoPtr _item_type_info; const size_t _item_size; }; +///====================== MapType Info ==========================/// +class MapTypeInfo : public TypeInfo { +public: + explicit MapTypeInfo(TypeInfoPtr key_type_info, TypeInfoPtr value_type_info) + : _key_type_info(std::move(key_type_info)), + _value_type_info(std::move(value_type_info)) {} + ~MapTypeInfo() override = default; + + inline bool equal(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const MapValue*>(left); + auto r_value = reinterpret_cast<const MapValue*>(right); + return l_value->size() == r_value->size(); + } + + int cmp(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const MapValue*>(left); + auto r_value = reinterpret_cast<const MapValue*>(right); + uint32_t l_size = l_value->size(); + uint32_t r_size = r_value->size(); + if (l_size < r_size) { + return -1; + } else if (l_size > r_size) { + return 1; + } else { + return 0; + } + } + + void shallow_copy(void* dest, const void* src) const override { + auto dest_value = reinterpret_cast<MapValue*>(dest); + auto src_value = reinterpret_cast<const MapValue*>(src); + dest_value->shallow_copy(src_value); + } + + void deep_copy(void* dest, const void* src, MemPool* mem_pool) const override { DCHECK(false); } + + void copy_object(void* dest, const void* src, MemPool* mem_pool) const override { + deep_copy(dest, src, mem_pool); + } + + void direct_copy(void* dest, const void* src) const override { CHECK(false); } + + void direct_copy(uint8_t** base, void* dest, const void* src) const { CHECK(false); } + + void direct_copy_may_cut(void* dest, const void* src) const override { direct_copy(dest, src); } + + Status convert_from(void* dest, const void* src, const TypeInfo* src_type, MemPool* mem_pool, + size_t variable_len = 0) const override { + return Status::Error<ErrorCode::NOT_IMPLEMENTED_ERROR>(); + } + + Status from_string(void* buf, const std::string& scan_key, const int precision = 0, + const int scale = 0) const override { + return Status::Error<ErrorCode::NOT_IMPLEMENTED_ERROR>(); + } + + std::string to_string(const void* src) const override { return "{}"; } + + void set_to_max(void* buf) const override { + DCHECK(false) << "set_to_max of list is not implemented."; + } + + void set_to_min(void* buf) const override { + DCHECK(false) << "set_to_min of list is not implemented."; + } + + uint32_t hash_code(const void* data, uint32_t seed) const override { + auto map_value = reinterpret_cast<const MapValue*>(data); + auto size = map_value->size(); + uint32_t result = HashUtil::hash(&size, sizeof(size), seed); + result = seed * result + _key_type_info->hash_code(map_value->key_data(), seed) + + _value_type_info->hash_code(map_value->value_data(), seed); + return result; + } + + // todo . is here only to need return 16 for two ptr? + const size_t size() const override { return sizeof(MapValue); } + + FieldType type() const override { return OLAP_FIELD_TYPE_MAP; } + + inline const TypeInfo* get_key_type_info() const { return _key_type_info.get(); } + inline const TypeInfo* get_value_type_info() const { return _value_type_info.get(); } + +private: + TypeInfoPtr _key_type_info; + TypeInfoPtr _value_type_info; +}; + +class StructTypeInfo : public TypeInfo { +public: + explicit StructTypeInfo(std::vector<TypeInfoPtr>& type_infos) { + for (TypeInfoPtr& type_info : type_infos) { + _type_infos.push_back(std::move(type_info)); + } + } + ~StructTypeInfo() override = default; + + bool equal(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const StructValue*>(left); + auto r_value = reinterpret_cast<const StructValue*>(right); + if (l_value->size() != r_value->size()) { + return false; + } + uint32_t size = l_value->size(); + + if (!l_value->has_null() && !r_value->has_null()) { + for (size_t i = 0; i < size; ++i) { + if (!_type_infos[i]->equal(l_value->child_value(i), r_value->child_value(i))) { + return false; + } + } + } else { + for (size_t i = 0; i < size; ++i) { + if (l_value->is_null_at(i)) { + if (r_value->is_null_at(i)) { // both are null + continue; + } else { // left is null & right is not null + return false; + } + } else if (r_value->is_null_at(i)) { // left is not null & right is null + return false; + } + if (!_type_infos[i]->equal(l_value->child_value(i), r_value->child_value(i))) { + return false; + } + } + } + return true; + } + + int cmp(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const StructValue*>(left); + auto r_value = reinterpret_cast<const StructValue*>(right); + uint32_t l_size = l_value->size(); + uint32_t r_size = r_value->size(); + size_t cur = 0; + + if (!l_value->has_null() && !r_value->has_null()) { + while (cur < l_size && cur < r_size) { + int result = + _type_infos[cur]->cmp(l_value->child_value(cur), r_value->child_value(cur)); + if (result != 0) { + return result; + } + ++cur; + } + } else { + while (cur < l_size && cur < r_size) { + if (l_value->is_null_at(cur)) { + if (!r_value->is_null_at(cur)) { // left is null & right is not null + return -1; + } + } else if (r_value->is_null_at(cur)) { // left is not null & right is null + return 1; + } else { // both are not null + int result = _type_infos[cur]->cmp(l_value->child_value(cur), + r_value->child_value(cur)); + if (result != 0) { + return result; + } + } + ++cur; + } + } + + if (l_size < r_size) { + return -1; + } else if (l_size > r_size) { + return 1; + } else { + return 0; + } + } + + void shallow_copy(void* dest, const void* src) const override { + auto dest_value = reinterpret_cast<StructValue*>(dest); + auto src_value = reinterpret_cast<const StructValue*>(src); + dest_value->shallow_copy(src_value); + } + + void deep_copy(void* dest, const void* src, MemPool* mem_pool) const override { + auto dest_value = reinterpret_cast<StructValue*>(dest); + auto src_value = reinterpret_cast<const StructValue*>(src); + + if (src_value->size() == 0) { + new (dest_value) StructValue(src_value->size()); + return; + } + + dest_value->set_size(src_value->size()); + dest_value->set_has_null(src_value->has_null()); + + size_t allocate_size = src_value->size() * sizeof(*src_value->values()); + // allocate memory for children value + for (size_t i = 0; i < src_value->size(); ++i) { + if (src_value->is_null_at(i)) continue; Review Comment: warning: statement should be inside braces [readability-braces-around-statements] ```suggestion if (src_value->is_null_at(i)) { continue; } ``` ########## be/src/olap/types.h: ########## @@ -430,6 +432,332 @@ TypeInfoPtr _item_type_info; const size_t _item_size; }; +///====================== MapType Info ==========================/// +class MapTypeInfo : public TypeInfo { +public: + explicit MapTypeInfo(TypeInfoPtr key_type_info, TypeInfoPtr value_type_info) + : _key_type_info(std::move(key_type_info)), + _value_type_info(std::move(value_type_info)) {} + ~MapTypeInfo() override = default; + + inline bool equal(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const MapValue*>(left); + auto r_value = reinterpret_cast<const MapValue*>(right); + return l_value->size() == r_value->size(); + } + + int cmp(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const MapValue*>(left); + auto r_value = reinterpret_cast<const MapValue*>(right); + uint32_t l_size = l_value->size(); + uint32_t r_size = r_value->size(); + if (l_size < r_size) { + return -1; + } else if (l_size > r_size) { + return 1; + } else { + return 0; + } + } + + void shallow_copy(void* dest, const void* src) const override { + auto dest_value = reinterpret_cast<MapValue*>(dest); + auto src_value = reinterpret_cast<const MapValue*>(src); + dest_value->shallow_copy(src_value); + } + + void deep_copy(void* dest, const void* src, MemPool* mem_pool) const override { DCHECK(false); } + + void copy_object(void* dest, const void* src, MemPool* mem_pool) const override { + deep_copy(dest, src, mem_pool); + } + + void direct_copy(void* dest, const void* src) const override { CHECK(false); } + + void direct_copy(uint8_t** base, void* dest, const void* src) const { CHECK(false); } + + void direct_copy_may_cut(void* dest, const void* src) const override { direct_copy(dest, src); } + + Status convert_from(void* dest, const void* src, const TypeInfo* src_type, MemPool* mem_pool, + size_t variable_len = 0) const override { + return Status::Error<ErrorCode::NOT_IMPLEMENTED_ERROR>(); + } + + Status from_string(void* buf, const std::string& scan_key, const int precision = 0, + const int scale = 0) const override { + return Status::Error<ErrorCode::NOT_IMPLEMENTED_ERROR>(); + } + + std::string to_string(const void* src) const override { return "{}"; } + + void set_to_max(void* buf) const override { + DCHECK(false) << "set_to_max of list is not implemented."; + } + + void set_to_min(void* buf) const override { + DCHECK(false) << "set_to_min of list is not implemented."; + } + + uint32_t hash_code(const void* data, uint32_t seed) const override { + auto map_value = reinterpret_cast<const MapValue*>(data); + auto size = map_value->size(); + uint32_t result = HashUtil::hash(&size, sizeof(size), seed); + result = seed * result + _key_type_info->hash_code(map_value->key_data(), seed) + + _value_type_info->hash_code(map_value->value_data(), seed); + return result; + } + + // todo . is here only to need return 16 for two ptr? + const size_t size() const override { return sizeof(MapValue); } + + FieldType type() const override { return OLAP_FIELD_TYPE_MAP; } + + inline const TypeInfo* get_key_type_info() const { return _key_type_info.get(); } + inline const TypeInfo* get_value_type_info() const { return _value_type_info.get(); } + +private: + TypeInfoPtr _key_type_info; + TypeInfoPtr _value_type_info; +}; + +class StructTypeInfo : public TypeInfo { +public: + explicit StructTypeInfo(std::vector<TypeInfoPtr>& type_infos) { + for (TypeInfoPtr& type_info : type_infos) { + _type_infos.push_back(std::move(type_info)); + } + } + ~StructTypeInfo() override = default; + + bool equal(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const StructValue*>(left); + auto r_value = reinterpret_cast<const StructValue*>(right); + if (l_value->size() != r_value->size()) { + return false; + } + uint32_t size = l_value->size(); + + if (!l_value->has_null() && !r_value->has_null()) { + for (size_t i = 0; i < size; ++i) { + if (!_type_infos[i]->equal(l_value->child_value(i), r_value->child_value(i))) { + return false; + } + } + } else { + for (size_t i = 0; i < size; ++i) { + if (l_value->is_null_at(i)) { + if (r_value->is_null_at(i)) { // both are null + continue; + } else { // left is null & right is not null + return false; + } + } else if (r_value->is_null_at(i)) { // left is not null & right is null + return false; + } + if (!_type_infos[i]->equal(l_value->child_value(i), r_value->child_value(i))) { + return false; + } + } + } + return true; + } + + int cmp(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const StructValue*>(left); + auto r_value = reinterpret_cast<const StructValue*>(right); + uint32_t l_size = l_value->size(); + uint32_t r_size = r_value->size(); + size_t cur = 0; + + if (!l_value->has_null() && !r_value->has_null()) { + while (cur < l_size && cur < r_size) { + int result = + _type_infos[cur]->cmp(l_value->child_value(cur), r_value->child_value(cur)); + if (result != 0) { + return result; + } + ++cur; + } + } else { + while (cur < l_size && cur < r_size) { + if (l_value->is_null_at(cur)) { + if (!r_value->is_null_at(cur)) { // left is null & right is not null + return -1; + } + } else if (r_value->is_null_at(cur)) { // left is not null & right is null + return 1; + } else { // both are not null + int result = _type_infos[cur]->cmp(l_value->child_value(cur), + r_value->child_value(cur)); + if (result != 0) { + return result; + } + } + ++cur; + } + } + + if (l_size < r_size) { + return -1; + } else if (l_size > r_size) { + return 1; + } else { + return 0; + } + } + + void shallow_copy(void* dest, const void* src) const override { + auto dest_value = reinterpret_cast<StructValue*>(dest); + auto src_value = reinterpret_cast<const StructValue*>(src); + dest_value->shallow_copy(src_value); + } + + void deep_copy(void* dest, const void* src, MemPool* mem_pool) const override { + auto dest_value = reinterpret_cast<StructValue*>(dest); + auto src_value = reinterpret_cast<const StructValue*>(src); + + if (src_value->size() == 0) { + new (dest_value) StructValue(src_value->size()); + return; + } + + dest_value->set_size(src_value->size()); + dest_value->set_has_null(src_value->has_null()); + + size_t allocate_size = src_value->size() * sizeof(*src_value->values()); + // allocate memory for children value + for (size_t i = 0; i < src_value->size(); ++i) { + if (src_value->is_null_at(i)) continue; + allocate_size += _type_infos[i]->size(); + } + + dest_value->set_values((void**)mem_pool->allocate(allocate_size)); + auto ptr = reinterpret_cast<uint8_t*>(dest_value->mutable_values()); + ptr += dest_value->size() * sizeof(*dest_value->values()); + + for (size_t i = 0; i < src_value->size(); ++i) { + dest_value->set_child_value(nullptr, i); + if (src_value->is_null_at(i)) continue; Review Comment: warning: statement should be inside braces [readability-braces-around-statements] ```suggestion if (src_value->is_null_at(i)) { continue; } ``` ########## be/src/olap/types.h: ########## @@ -430,6 +432,332 @@ TypeInfoPtr _item_type_info; const size_t _item_size; }; +///====================== MapType Info ==========================/// +class MapTypeInfo : public TypeInfo { +public: + explicit MapTypeInfo(TypeInfoPtr key_type_info, TypeInfoPtr value_type_info) + : _key_type_info(std::move(key_type_info)), + _value_type_info(std::move(value_type_info)) {} + ~MapTypeInfo() override = default; + + inline bool equal(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const MapValue*>(left); + auto r_value = reinterpret_cast<const MapValue*>(right); + return l_value->size() == r_value->size(); + } + + int cmp(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const MapValue*>(left); + auto r_value = reinterpret_cast<const MapValue*>(right); + uint32_t l_size = l_value->size(); + uint32_t r_size = r_value->size(); + if (l_size < r_size) { + return -1; + } else if (l_size > r_size) { + return 1; + } else { + return 0; + } + } + + void shallow_copy(void* dest, const void* src) const override { + auto dest_value = reinterpret_cast<MapValue*>(dest); + auto src_value = reinterpret_cast<const MapValue*>(src); + dest_value->shallow_copy(src_value); + } + + void deep_copy(void* dest, const void* src, MemPool* mem_pool) const override { DCHECK(false); } + + void copy_object(void* dest, const void* src, MemPool* mem_pool) const override { + deep_copy(dest, src, mem_pool); + } + + void direct_copy(void* dest, const void* src) const override { CHECK(false); } + + void direct_copy(uint8_t** base, void* dest, const void* src) const { CHECK(false); } + + void direct_copy_may_cut(void* dest, const void* src) const override { direct_copy(dest, src); } + + Status convert_from(void* dest, const void* src, const TypeInfo* src_type, MemPool* mem_pool, + size_t variable_len = 0) const override { + return Status::Error<ErrorCode::NOT_IMPLEMENTED_ERROR>(); + } + + Status from_string(void* buf, const std::string& scan_key, const int precision = 0, + const int scale = 0) const override { + return Status::Error<ErrorCode::NOT_IMPLEMENTED_ERROR>(); + } + + std::string to_string(const void* src) const override { return "{}"; } + + void set_to_max(void* buf) const override { + DCHECK(false) << "set_to_max of list is not implemented."; + } + + void set_to_min(void* buf) const override { + DCHECK(false) << "set_to_min of list is not implemented."; + } + + uint32_t hash_code(const void* data, uint32_t seed) const override { + auto map_value = reinterpret_cast<const MapValue*>(data); + auto size = map_value->size(); + uint32_t result = HashUtil::hash(&size, sizeof(size), seed); + result = seed * result + _key_type_info->hash_code(map_value->key_data(), seed) + + _value_type_info->hash_code(map_value->value_data(), seed); + return result; + } + + // todo . is here only to need return 16 for two ptr? + const size_t size() const override { return sizeof(MapValue); } + + FieldType type() const override { return OLAP_FIELD_TYPE_MAP; } + + inline const TypeInfo* get_key_type_info() const { return _key_type_info.get(); } + inline const TypeInfo* get_value_type_info() const { return _value_type_info.get(); } + +private: + TypeInfoPtr _key_type_info; + TypeInfoPtr _value_type_info; +}; + +class StructTypeInfo : public TypeInfo { +public: + explicit StructTypeInfo(std::vector<TypeInfoPtr>& type_infos) { + for (TypeInfoPtr& type_info : type_infos) { + _type_infos.push_back(std::move(type_info)); + } + } + ~StructTypeInfo() override = default; + + bool equal(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const StructValue*>(left); + auto r_value = reinterpret_cast<const StructValue*>(right); + if (l_value->size() != r_value->size()) { + return false; + } + uint32_t size = l_value->size(); + + if (!l_value->has_null() && !r_value->has_null()) { + for (size_t i = 0; i < size; ++i) { + if (!_type_infos[i]->equal(l_value->child_value(i), r_value->child_value(i))) { + return false; + } + } + } else { + for (size_t i = 0; i < size; ++i) { + if (l_value->is_null_at(i)) { + if (r_value->is_null_at(i)) { // both are null + continue; + } else { // left is null & right is not null + return false; + } + } else if (r_value->is_null_at(i)) { // left is not null & right is null + return false; + } + if (!_type_infos[i]->equal(l_value->child_value(i), r_value->child_value(i))) { + return false; + } + } + } + return true; + } + + int cmp(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const StructValue*>(left); + auto r_value = reinterpret_cast<const StructValue*>(right); + uint32_t l_size = l_value->size(); + uint32_t r_size = r_value->size(); + size_t cur = 0; + + if (!l_value->has_null() && !r_value->has_null()) { + while (cur < l_size && cur < r_size) { + int result = + _type_infos[cur]->cmp(l_value->child_value(cur), r_value->child_value(cur)); + if (result != 0) { + return result; + } + ++cur; + } + } else { + while (cur < l_size && cur < r_size) { + if (l_value->is_null_at(cur)) { + if (!r_value->is_null_at(cur)) { // left is null & right is not null + return -1; + } + } else if (r_value->is_null_at(cur)) { // left is not null & right is null + return 1; + } else { // both are not null + int result = _type_infos[cur]->cmp(l_value->child_value(cur), + r_value->child_value(cur)); + if (result != 0) { + return result; + } + } + ++cur; + } + } + + if (l_size < r_size) { + return -1; + } else if (l_size > r_size) { + return 1; + } else { + return 0; + } + } + + void shallow_copy(void* dest, const void* src) const override { + auto dest_value = reinterpret_cast<StructValue*>(dest); + auto src_value = reinterpret_cast<const StructValue*>(src); + dest_value->shallow_copy(src_value); + } + + void deep_copy(void* dest, const void* src, MemPool* mem_pool) const override { + auto dest_value = reinterpret_cast<StructValue*>(dest); + auto src_value = reinterpret_cast<const StructValue*>(src); + + if (src_value->size() == 0) { + new (dest_value) StructValue(src_value->size()); + return; + } + + dest_value->set_size(src_value->size()); + dest_value->set_has_null(src_value->has_null()); + + size_t allocate_size = src_value->size() * sizeof(*src_value->values()); + // allocate memory for children value + for (size_t i = 0; i < src_value->size(); ++i) { + if (src_value->is_null_at(i)) continue; + allocate_size += _type_infos[i]->size(); + } + + dest_value->set_values((void**)mem_pool->allocate(allocate_size)); + auto ptr = reinterpret_cast<uint8_t*>(dest_value->mutable_values()); + ptr += dest_value->size() * sizeof(*dest_value->values()); + + for (size_t i = 0; i < src_value->size(); ++i) { + dest_value->set_child_value(nullptr, i); + if (src_value->is_null_at(i)) continue; + dest_value->set_child_value(ptr, i); + ptr += _type_infos[i]->size(); + } + + // copy children value + for (size_t i = 0; i < src_value->size(); ++i) { + if (src_value->is_null_at(i)) continue; Review Comment: warning: statement should be inside braces [readability-braces-around-statements] ```suggestion if (src_value->is_null_at(i)) { continue; } ``` ########## be/src/olap/types.h: ########## @@ -430,6 +432,332 @@ TypeInfoPtr _item_type_info; const size_t _item_size; }; +///====================== MapType Info ==========================/// +class MapTypeInfo : public TypeInfo { +public: + explicit MapTypeInfo(TypeInfoPtr key_type_info, TypeInfoPtr value_type_info) + : _key_type_info(std::move(key_type_info)), + _value_type_info(std::move(value_type_info)) {} + ~MapTypeInfo() override = default; + + inline bool equal(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const MapValue*>(left); + auto r_value = reinterpret_cast<const MapValue*>(right); + return l_value->size() == r_value->size(); + } + + int cmp(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const MapValue*>(left); + auto r_value = reinterpret_cast<const MapValue*>(right); + uint32_t l_size = l_value->size(); + uint32_t r_size = r_value->size(); + if (l_size < r_size) { + return -1; + } else if (l_size > r_size) { + return 1; + } else { + return 0; + } + } + + void shallow_copy(void* dest, const void* src) const override { + auto dest_value = reinterpret_cast<MapValue*>(dest); + auto src_value = reinterpret_cast<const MapValue*>(src); + dest_value->shallow_copy(src_value); + } + + void deep_copy(void* dest, const void* src, MemPool* mem_pool) const override { DCHECK(false); } + + void copy_object(void* dest, const void* src, MemPool* mem_pool) const override { + deep_copy(dest, src, mem_pool); + } + + void direct_copy(void* dest, const void* src) const override { CHECK(false); } + + void direct_copy(uint8_t** base, void* dest, const void* src) const { CHECK(false); } + + void direct_copy_may_cut(void* dest, const void* src) const override { direct_copy(dest, src); } + + Status convert_from(void* dest, const void* src, const TypeInfo* src_type, MemPool* mem_pool, + size_t variable_len = 0) const override { + return Status::Error<ErrorCode::NOT_IMPLEMENTED_ERROR>(); + } + + Status from_string(void* buf, const std::string& scan_key, const int precision = 0, + const int scale = 0) const override { + return Status::Error<ErrorCode::NOT_IMPLEMENTED_ERROR>(); + } + + std::string to_string(const void* src) const override { return "{}"; } + + void set_to_max(void* buf) const override { + DCHECK(false) << "set_to_max of list is not implemented."; + } + + void set_to_min(void* buf) const override { + DCHECK(false) << "set_to_min of list is not implemented."; + } + + uint32_t hash_code(const void* data, uint32_t seed) const override { + auto map_value = reinterpret_cast<const MapValue*>(data); + auto size = map_value->size(); + uint32_t result = HashUtil::hash(&size, sizeof(size), seed); + result = seed * result + _key_type_info->hash_code(map_value->key_data(), seed) + + _value_type_info->hash_code(map_value->value_data(), seed); + return result; + } + + // todo . is here only to need return 16 for two ptr? + const size_t size() const override { return sizeof(MapValue); } + + FieldType type() const override { return OLAP_FIELD_TYPE_MAP; } + + inline const TypeInfo* get_key_type_info() const { return _key_type_info.get(); } + inline const TypeInfo* get_value_type_info() const { return _value_type_info.get(); } + +private: + TypeInfoPtr _key_type_info; + TypeInfoPtr _value_type_info; +}; + +class StructTypeInfo : public TypeInfo { +public: + explicit StructTypeInfo(std::vector<TypeInfoPtr>& type_infos) { + for (TypeInfoPtr& type_info : type_infos) { + _type_infos.push_back(std::move(type_info)); + } + } + ~StructTypeInfo() override = default; + + bool equal(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const StructValue*>(left); + auto r_value = reinterpret_cast<const StructValue*>(right); + if (l_value->size() != r_value->size()) { + return false; + } + uint32_t size = l_value->size(); + + if (!l_value->has_null() && !r_value->has_null()) { + for (size_t i = 0; i < size; ++i) { + if (!_type_infos[i]->equal(l_value->child_value(i), r_value->child_value(i))) { + return false; + } + } + } else { + for (size_t i = 0; i < size; ++i) { + if (l_value->is_null_at(i)) { + if (r_value->is_null_at(i)) { // both are null + continue; + } else { // left is null & right is not null + return false; + } + } else if (r_value->is_null_at(i)) { // left is not null & right is null + return false; + } + if (!_type_infos[i]->equal(l_value->child_value(i), r_value->child_value(i))) { + return false; + } + } + } + return true; + } + + int cmp(const void* left, const void* right) const override { + auto l_value = reinterpret_cast<const StructValue*>(left); + auto r_value = reinterpret_cast<const StructValue*>(right); + uint32_t l_size = l_value->size(); + uint32_t r_size = r_value->size(); + size_t cur = 0; + + if (!l_value->has_null() && !r_value->has_null()) { + while (cur < l_size && cur < r_size) { + int result = + _type_infos[cur]->cmp(l_value->child_value(cur), r_value->child_value(cur)); + if (result != 0) { + return result; + } + ++cur; + } + } else { + while (cur < l_size && cur < r_size) { + if (l_value->is_null_at(cur)) { + if (!r_value->is_null_at(cur)) { // left is null & right is not null + return -1; + } + } else if (r_value->is_null_at(cur)) { // left is not null & right is null + return 1; + } else { // both are not null + int result = _type_infos[cur]->cmp(l_value->child_value(cur), + r_value->child_value(cur)); + if (result != 0) { + return result; + } + } + ++cur; + } + } + + if (l_size < r_size) { + return -1; + } else if (l_size > r_size) { + return 1; + } else { + return 0; + } + } + + void shallow_copy(void* dest, const void* src) const override { + auto dest_value = reinterpret_cast<StructValue*>(dest); + auto src_value = reinterpret_cast<const StructValue*>(src); + dest_value->shallow_copy(src_value); + } + + void deep_copy(void* dest, const void* src, MemPool* mem_pool) const override { + auto dest_value = reinterpret_cast<StructValue*>(dest); + auto src_value = reinterpret_cast<const StructValue*>(src); + + if (src_value->size() == 0) { + new (dest_value) StructValue(src_value->size()); + return; + } + + dest_value->set_size(src_value->size()); + dest_value->set_has_null(src_value->has_null()); + + size_t allocate_size = src_value->size() * sizeof(*src_value->values()); + // allocate memory for children value + for (size_t i = 0; i < src_value->size(); ++i) { + if (src_value->is_null_at(i)) continue; + allocate_size += _type_infos[i]->size(); + } + + dest_value->set_values((void**)mem_pool->allocate(allocate_size)); + auto ptr = reinterpret_cast<uint8_t*>(dest_value->mutable_values()); + ptr += dest_value->size() * sizeof(*dest_value->values()); + + for (size_t i = 0; i < src_value->size(); ++i) { + dest_value->set_child_value(nullptr, i); + if (src_value->is_null_at(i)) continue; + dest_value->set_child_value(ptr, i); + ptr += _type_infos[i]->size(); + } + + // copy children value + for (size_t i = 0; i < src_value->size(); ++i) { + if (src_value->is_null_at(i)) continue; + _type_infos[i]->deep_copy(dest_value->mutable_child_value(i), src_value->child_value(i), + mem_pool); + } + } + + void copy_object(void* dest, const void* src, MemPool* mem_pool) const override { + deep_copy(dest, src, mem_pool); + } + + void direct_copy(void* dest, const void* src) const override { + auto dest_value = static_cast<StructValue*>(dest); + auto base = reinterpret_cast<uint8_t*>(dest_value->mutable_values()); + direct_copy(&base, dest, src); + } + + void direct_copy(uint8_t** base, void* dest, const void* src) const { + auto dest_value = static_cast<StructValue*>(dest); + auto src_value = static_cast<const StructValue*>(src); + + dest_value->set_size(src_value->size()); + dest_value->set_has_null(src_value->has_null()); + *base += src_value->size() * sizeof(*src_value->values()); + + for (size_t i = 0; i < src_value->size(); ++i) { + dest_value->set_child_value(nullptr, i); + if (src_value->is_null_at(i)) continue; Review Comment: warning: statement should be inside braces [readability-braces-around-statements] ```suggestion if (src_value->is_null_at(i)) { continue; } ``` -- This is an automated message from the Apache Git Service. To respond to the message, please log on to GitHub and use the URL above to go to the specific comment. To unsubscribe, e-mail: commits-unsubscr...@doris.apache.org For queries about this service, please contact Infrastructure at: us...@infra.apache.org --------------------------------------------------------------------- To unsubscribe, e-mail: commits-unsubscr...@doris.apache.org For additional commands, e-mail: commits-h...@doris.apache.org