blob: 12345d034171d9e2942c0928bb44d07c26408a4d [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.
#include "util/bit_util.h"
#include <gtest/gtest-message.h>
#include <gtest/gtest-test-part.h>
#include <gtest/gtest.h>
#include <bit>
#include <boost/utility/binary.hpp>
#include <random>
#include "gtest/gtest_pred_impl.h"
#include "runtime/primitive_type.h"
#include "util/simd/bits.h"
#include "vec/columns/column_nullable.h"
#include "vec/common/endian.h"
namespace doris {
TEST(BitUtil, Ceil) {
EXPECT_EQ(BitUtil::ceil(0, 1), 0);
EXPECT_EQ(BitUtil::ceil(1, 1), 1);
EXPECT_EQ(BitUtil::ceil(1, 2), 1);
EXPECT_EQ(BitUtil::ceil(1, 8), 1);
EXPECT_EQ(BitUtil::ceil(7, 8), 1);
EXPECT_EQ(BitUtil::ceil(8, 8), 1);
EXPECT_EQ(BitUtil::ceil(9, 8), 2);
}
TEST(BitUtil, BigEndianToHost) {
uint16_t v16 = 0x1234;
uint32_t v32 = 0x12345678;
uint64_t v64 = 0x123456789abcdef0;
unsigned __int128 v128 = ((__int128)0x123456789abcdef0LL << 64) | 0x123456789abcdef0LL;
wide::UInt256 v256 =
wide::UInt256(0x123456789abcdef0) << 192 | wide::UInt256(0x123456789abcdef0) << 128 |
wide::UInt256(0x123456789abcdef0) << 64 | wide::UInt256(0x123456789abcdef0);
EXPECT_EQ(to_endian<std::endian::big>(v16), 0x3412);
EXPECT_EQ(to_endian<std::endian::big>(v32), 0x78563412);
EXPECT_EQ(to_endian<std::endian::big>(v64), 0xf0debc9a78563412);
EXPECT_EQ(to_endian<std::endian::big>(v128),
((__int128)0xf0debc9a78563412LL << 64) | 0xf0debc9a78563412LL);
EXPECT_EQ(to_endian<std::endian::big>(v256),
wide::UInt256(0xf0debc9a78563412) << 192 | wide::UInt256(0xf0debc9a78563412) << 128 |
wide::UInt256(0xf0debc9a78563412) << 64 | wide::UInt256(0xf0debc9a78563412));
}
void insert_true(vectorized::ColumnNullable* column, size_t num = 1) {
for (int i = 0; i < num; i++) {
assert_cast<vectorized::ColumnUInt8*>(column->get_nested_column_ptr().get())
->insert_value(1);
column->push_false_to_nullmap(1);
}
}
void insert_false(vectorized::ColumnNullable* column, size_t num = 1) {
for (int i = 0; i < num; i++) {
assert_cast<vectorized::ColumnUInt8*>(column->get_nested_column_ptr().get())
->insert_value(0);
column->push_false_to_nullmap(1);
}
}
void insert_null(vectorized::ColumnNullable* column, size_t num = 1) {
for (int i = 0; i < num; i++) {
column->insert_default();
}
}
size_t brute_force_count_zero_num(const uint8_t* __restrict data,
const uint8_t* __restrict null_map, size_t size) {
size_t num = 0;
for (size_t i = 0; i < size; ++i) {
if (data[i] == 0 || null_map[i]) {
num++;
}
}
return num;
}
TEST(BitUtil, CountZero) {
{
auto column = vectorized::ColumnNullable::create(vectorized::ColumnUInt8::create(),
vectorized::ColumnUInt8::create());
insert_false(column.get(), 5);
insert_null(column.get(), 1);
insert_false(column.get(), 8);
insert_null(column.get(), 1);
insert_false(column.get(), 54);
insert_true(column.get(), 1);
insert_false(column.get(), 14);
ASSERT_EQ(
brute_force_count_zero_num(assert_cast<const vectorized::ColumnUInt8*>(
column->get_nested_column_ptr().get())
->get_data()
.data(),
column->get_null_map_data().data(), column->size()),
simd::count_zero_num((int8_t*)assert_cast<const vectorized::ColumnUInt8*>(
column->get_nested_column_ptr().get())
->get_data()
.data(),
column->get_null_map_data().data(), (uint32_t)column->size()));
}
{
auto column = vectorized::ColumnNullable::create(vectorized::ColumnUInt8::create(),
vectorized::ColumnUInt8::create());
std::mt19937 rng(12345);
std::uniform_int_distribution<int> val_dist(0, 1);
std::uniform_int_distribution<int> null_dist(0, 5);
for (int i = 0; i < 10000; ++i) {
if (null_dist(rng) == 0) {
insert_null(column.get(), 1);
} else {
if (val_dist(rng) == 0) {
insert_false(column.get(), 1);
} else {
insert_true(column.get(), 1);
}
}
}
ASSERT_EQ(
brute_force_count_zero_num(assert_cast<const vectorized::ColumnUInt8*>(
column->get_nested_column_ptr().get())
->get_data()
.data(),
column->get_null_map_data().data(), column->size()),
simd::count_zero_num((int8_t*)assert_cast<const vectorized::ColumnUInt8*>(
column->get_nested_column_ptr().get())
->get_data()
.data(),
column->get_null_map_data().data(), (uint32_t)column->size()));
}
}
} // namespace doris