blob: 9fc1623151801e51304ee37f573c7794a2597da8 [file] [log] [blame]
// 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/ClickHouse/ClickHouse/blob/master/src/Functions/array/arrayReverse.cpp
// and modified by Doris
#pragma once
#include "vec/columns/column_array.h"
#include "vec/columns/column_const.h"
#include "vec/data_types/data_type_array.h"
#include "vec/data_types/data_type_number.h"
#include "vec/functions/array/function_array_utils.h"
namespace doris::vectorized {
struct ArrayReverseImpl {
static Status _execute(Block& block, const ColumnNumbers& arguments, uint32_t result,
size_t input_rows_count) {
ColumnPtr src_column =
block.get_by_position(arguments[0]).column->convert_to_full_column_if_const();
ColumnArrayExecutionData src;
if (!extract_column_array_info(*src_column, src)) {
return Status::RuntimeError(
fmt::format("execute failed, unsupported types for function {}({})", "reverse",
block.get_by_position(arguments[0]).type->get_name()));
}
bool is_nullable = src.nested_nullmap_data ? true : false;
ColumnArrayMutableData dst = create_mutable_data(src.nested_col.get(), is_nullable);
dst.offsets_ptr->reserve(input_rows_count);
auto res_val = _execute_internal(*src.nested_col, *src.offsets_ptr, *dst.nested_col,
*dst.offsets_ptr, src.nested_nullmap_data,
dst.nested_nullmap_data);
if (!res_val) {
return Status::RuntimeError(
fmt::format("execute failed or unsupported types for function {}({})",
"reverse", block.get_by_position(arguments[0]).type->get_name()));
}
ColumnPtr res_column = assemble_column_array(dst);
block.replace_by_position(result, std::move(res_column));
return Status::OK();
}
static bool _execute_internal(const IColumn& src_column,
const ColumnArray::Offsets64& src_offsets, IColumn& dest_column,
ColumnArray::Offsets64& dest_offsets, const UInt8* src_null_map,
ColumnUInt8::Container* dest_null_map) {
size_t prev_src_offset = 0;
for (auto curr_src_offset : src_offsets) {
size_t array_size = curr_src_offset - prev_src_offset;
for (size_t j = 0; j < array_size; ++j) {
size_t j_reverse = curr_src_offset - j - 1;
if (src_null_map && src_null_map[j_reverse]) {
DCHECK(dest_null_map != nullptr);
// Note: here we need to insert default value
dest_column.insert_default();
(*dest_null_map).push_back(true);
continue;
}
dest_column.insert_from(src_column, j_reverse);
if (dest_null_map) {
(*dest_null_map).push_back(false);
}
}
dest_offsets.push_back(curr_src_offset);
prev_src_offset = curr_src_offset;
}
return true;
}
};
} // namespace doris::vectorized