#include #include #include namespace DB { namespace ErrorCodes { extern const int LOGICAL_ERROR; extern const int BAD_ARGUMENTS; } namespace { /// Working with UInt8: last bit = can be true, previous = can be false (Like src/Storages/MergeTree/BoolMask.h). /// This function provides "NOT" operation for BoolMasks by swapping last two bits ("can be true" <-> "can be false"). template struct BitSwapLastTwoImpl { using ResultType = UInt8; static constexpr const bool allow_fixed_string = false; static const constexpr bool allow_string_integer = false; static inline ResultType NO_SANITIZE_UNDEFINED apply([[maybe_unused]] A a) { if constexpr (!std::is_same_v) // Should be a logical error, but this function is callable from SQL. // Need to investigate this. throw DB::Exception(ErrorCodes::BAD_ARGUMENTS, "It's a bug! Only UInt8 type is supported by __bitSwapLastTwo."); auto little_bits = littleBits(a); return static_cast(((little_bits & 1) << 1) | ((little_bits >> 1) & 1)); } #if USE_EMBEDDED_COMPILER static constexpr bool compilable = true; static inline llvm::Value * compile(llvm::IRBuilder<> & b, llvm::Value * arg, bool) { if (!arg->getType()->isIntegerTy()) throw Exception(ErrorCodes::LOGICAL_ERROR, "__bitSwapLastTwo expected an integral type"); return b.CreateOr( b.CreateShl(b.CreateAnd(arg, 1), 1), b.CreateAnd(b.CreateLShr(arg, 1), 1) ); } #endif }; struct NameBitSwapLastTwo { static constexpr auto name = "__bitSwapLastTwo"; }; using FunctionBitSwapLastTwo = FunctionUnaryArithmetic; } template <> struct FunctionUnaryArithmeticMonotonicity { static bool has() { return false; } static IFunction::Monotonicity get(const Field &, const Field &) { return {}; } }; REGISTER_FUNCTION(BitSwapLastTwo) { factory.registerFunction(); } }