/* Copyright (C) 2018 David Robillard Copyright (C) 2006-2007 Chris Hamilton This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program. If not, see . */ #ifndef CHILBERT_BITVECMASK_HPP #define CHILBERT_BITVECMASK_HPP #include #include namespace chilbert { namespace detail { /** Mask for a bit that can be incremented like an index. * * This enables fast iteration while setting or resetting bits since it is * internally stored as a mask which avoids repeated index math. */ template struct BitVecMask { static constexpr size_t bits_per_rack = sizeof(Rack) * CHAR_BIT; BitVecMask(const size_t index) : rack{index / bits_per_rack} , mask{Rack{1} << (index - rack * bits_per_rack)} { } void operator++() { if ((mask <<= 1U) == 0U) { mask = 1U; ++rack; } } void operator--() { if ((mask >>= 1U) == 0U) { mask = Rack{1} << (bits_per_rack - 1U); --rack; } } bool operator==(const BitVecMask& rhs) const { return rack == rhs.rack && mask == rhs.mask; } bool operator!=(const BitVecMask& rhs) const { return !operator==(rhs); } size_t rack; Rack mask; }; } // namespace detail } // namespace chilbert #endif