/* 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 3 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 <http://www.gnu.org/licenses/>. */ #define DECLARE_TYPESAFE_INDEX(NAME, TYPE) typedef typesafe_index<TYPE, class TAG_##NAME> NAME /// This template allows for indexing with compile time check and generating /// error for using one index variable for another array that is to be indexed /// with different variable. /// base_type specifies /// /// @param base_type Base integral type /// /// @param tag Tag Name to prevent copy from one type to other /// template <class base_type, class tag> class typesafe_index { public: explicit typesafe_index(base_type i = base_type()) : p(i) {} void operator=(const base_type& val) { p = val; } constexpr base_type get_int() const { return p; } typesafe_index operator++() { return typesafe_index(++p); } typesafe_index operator++(int) { return typesafe_index(p++); } typesafe_index operator%(const base_type& val) { return typesafe_index(p % val); } bool operator<(const base_type& val) const { return (p<val); } bool operator<=(const base_type& val) const { return (p<=val); } bool operator>=(const base_type& val) const { return (p>=val); } bool operator>(const base_type& val) const { return (p>val); } bool operator!=(const base_type& val) const { return (p!=val); } bool operator==(const base_type& val) const { return (p==val); } explicit constexpr operator base_type() const { return p; } typesafe_index operator+(const base_type& val) const { return typesafe_index(p+val); } private: base_type p; }; /// This template associates the base_type array with accessor_type(index). /// So the elements can be accessed using [] only using accessor_type index /// _priv_instance is kept public for use in Parameter declaration. /// template <class base_type, uint32_t num_instances, typename accessor_type> class RestrictIDTypeArray { public: base_type _priv_instance[num_instances]; base_type& operator[](const accessor_type& index) { return _priv_instance[index.get_int()]; } constexpr const base_type& operator[](const accessor_type& index) const { return _priv_instance[index.get_int()]; } };