#pragma once namespace nall { //warning: so that BitRange can modify the underlying number directly, it must bind a reference. //as a result, auto value = number.bits() will capture by-reference, rather than by-value. template struct BitRange { static_assert(Precision >= 1 && Precision <= 64); static inline constexpr auto bits() -> uint { return Precision; } using utype = conditional_t>>>; static inline constexpr auto mask() -> utype { return ~0ull >> 64 - bits(); } inline BitRange(utype& source, int lo, int hi) : source(source) { if(lo < 0) lo = Precision + lo; if(hi < 0) hi = Precision + hi; if(lo > hi) swap(lo, hi); this->lo = lo; this->hi = hi; } inline auto& operator=(BitRange& source) { return set(source.get()); } inline operator utype() const { return get(); } inline auto operator++(int) { auto value = get(); set(value + 1); return value; } inline auto operator--(int) { auto value = get(); set(value - 1); return value; } inline auto& operator++() { return set(get() + 1); } inline auto& operator--() { return set(get() - 1); } template inline auto& operator =(const T& value) { return set( value); } template inline auto& operator *=(const T& value) { return set(get() * value); } template inline auto& operator /=(const T& value) { return set(get() / value); } template inline auto& operator %=(const T& value) { return set(get() % value); } template inline auto& operator +=(const T& value) { return set(get() + value); } template inline auto& operator -=(const T& value) { return set(get() - value); } template inline auto& operator<<=(const T& value) { return set(get() << value); } template inline auto& operator>>=(const T& value) { return set(get() >> value); } template inline auto& operator &=(const T& value) { return set(get() & value); } template inline auto& operator ^=(const T& value) { return set(get() ^ value); } template inline auto& operator |=(const T& value) { return set(get() | value); } private: inline auto get() const -> utype { const utype rangeBits = hi - lo + 1; const utype rangeMask = (1ull << rangeBits) - 1 << lo & mask(); return (source & rangeMask) >> lo; } inline auto& set(const utype& value) { const utype rangeBits = hi - lo + 1; const utype rangeMask = (1ull << rangeBits) - 1 << lo & mask(); source = source & ~rangeMask | value << lo & rangeMask; return *this; } utype& source; uint lo; uint hi; }; }