/* This file is part of the dynarmic project. * Copyright (c) 2016 MerryMage * This software may be used and distributed according to the terms of the GNU * General Public License version 2 or any later version. */ #pragma once #include #include #include #include "common/assert.h" namespace Dynarmic { namespace Common { /// The size of a type in terms of bits template constexpr size_t BitSize() { return sizeof(T) * CHAR_BIT; } /// Extract bits [begin_bit, end_bit] inclusive from value of type T. template constexpr T Bits(const T value) { static_assert(begin_bit < end_bit, "invalid bit range (position of beginning bit cannot be greater than that of end bit)"); static_assert(begin_bit < BitSize(), "begin_bit must be smaller than size of T"); static_assert(end_bit < BitSize(), "begin_bit must be smaller than size of T"); return (value >> begin_bit) & ((1 << (end_bit - begin_bit + 1)) - 1); } #ifdef _MSC_VER #pragma warning(push) #pragma warning(disable:4554) #endif /// Extracts a single bit at bit_position from value of type T. template constexpr bool Bit(const T value) { static_assert(bit_position < BitSize(), "bit_position must be smaller than size of T"); return ((value >> bit_position) & 1) != 0; } /// Extracts a single bit at bit_position from value of type T. template inline bool Bit(size_t bit_position, const T value) { ASSERT_MSG(bit_position < BitSize(), "bit_position must be smaller than size of T"); return ((value >> bit_position) & 1) != 0; } #ifdef _MSC_VER #pragma warning(pop) #endif /// Sign-extends a value that has bit_count bits to the full bitwidth of type T. template inline T SignExtend(const T value) { static_assert(bit_count <= BitSize(), "bit_count larger than bitsize of T"); constexpr T mask = static_cast(1ULL << bit_count) - 1; const bool signbit = Bit(value); if (signbit) { return value | ~mask; } return value; } /// Sign-extends a value that has bit_count bits to the full bitwidth of type T. template inline T SignExtend(const size_t bit_count, const T value) { ASSERT_MSG(bit_count <= BitSize(), "bit_count larger than bitsize of T"); const T mask = static_cast(1ULL << bit_count) - 1; const bool signbit = Bit(bit_count - 1, value); if (signbit) { return value | ~mask; } return value; } template inline size_t BitCount(Integral value) { return std::bitset()>(value).count(); } } // namespace Common } // namespace Dynarmic