blob: 9bcbd63f0a0f88151a0ffaa5ef8505dd5cc4dcaf [file] [log] [blame]
/*
* Copyright (C) 2004-2021 Savoir-faire Linux Inc.
*
* Author: Emmanuel Lepage <emmanuel.lepage@savoirfairelinux.com>
*
* 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, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#ifndef ENUM_CLASS_UTILS_H
#define ENUM_CLASS_UTILS_H
#include <map>
#include "logger.h"
#include <type_traits>
#include <vector>
#include <cassert>
namespace jami {
/**
* This function adds a safe way to get an enum class size
* @note it cannot be unsigned to avoid some compiler warnings
*/
template<typename A>
constexpr inline int
enum_class_size()
{
return size_t(A::COUNT__);
}
/**
* This generic class represents a multidimensional enum class array.
* It safely converts them to integers. Each enum class needs a "COUNT__" item
* at the end."
*
* This struct enforces:
* * That the rows are indexed using enum_classes
* * That the size of the matrix matches the enum_class size
* * That the operators are within the matrix boundary
*/
template<class Row, typename Value, typename A = Value>
struct Matrix1D
{
Matrix1D(std::initializer_list<std::initializer_list<Value>> s);
// Row is a built-in type ("int" by default)
Value operator[](Row v);
const Value operator[](Row v) const;
/**
* An Iterator for enum classes
*/
class EnumClassIter
{
public:
EnumClassIter(const Matrix1D<Row, Value, A>* p_vec, int pos)
: pos_(pos)
, p_vec_(p_vec)
{}
bool operator!=(const EnumClassIter& other) const;
Row operator*() const;
const EnumClassIter& operator++();
private:
int pos_;
const Matrix1D<Row, Value, A>* p_vec_;
};
// Iterators
EnumClassIter begin();
EnumClassIter end();
// Only use for single reverse mappable arrays, will ASSERT otherwise
Row fromValue(const Value& value) const;
static void setReverseMapping(Matrix1D<Row, const char*> names);
private:
const std::vector<Value> data_;
static std::map<A, Row> reverseMapping_;
};
/**
* A matrix with no value
*
* This is useful to use enum class in C++11 foreach loops
*
* @usage
* for (const MyEnum& value : Matrix0D<MyEnum>()) {
* std::cout << "Name: " << MyEnumNames[value] << std::endl;
* }
*/
template<class EnumClass>
struct Matrix0D
{
/**
* An Iterator for enum classes
*/
class EnumClassIter
{
public:
EnumClassIter(const Matrix0D<EnumClass>* p_vec, int pos)
: pos_(pos)
, p_vec_(p_vec)
{}
bool operator!=(const EnumClassIter& other) const;
EnumClass operator*() const;
const EnumClassIter& operator++();
private:
int pos_;
const Matrix0D<EnumClass>* p_vec_;
};
Matrix0D();
// Iterators
EnumClassIter begin();
EnumClassIter end();
};
/**
* A helper to type to match serializable string to enum elements
*/
template<class Row>
using EnumClassNames = Matrix1D<Row, const char*>;
/**
* Create a matrix type with 2 enum class dimensions M[I,J] = V
* ^ ^ ^
* | | |
* Rows <--- | |
* Columns <----- |
* Value <----------
*/
template<class Row, class Column, typename Value>
using Matrix2D = Matrix1D<Row, Matrix1D<Column, Value>>;
/**
* Create an array of callbacks.
*
* This type hides all the C++ syntax requirements
*/
template<class Row, class Class, typename Result = void, typename... Args>
using CallbackMatrix1D = Matrix1D<Row, Result (Class::*)(Args... args)>;
/**
* Create a method callback matrix.
*
* This type hides all the C++ syntax requirements
*/
template<class Row, class Column, class Class, typename Result = void, typename... Args>
using CallbackMatrix2D = Matrix2D<Row, Column, void (Class::*)(Args... args)>;
/*
* IMPLEMENTATION
*
*/
template<class Row, typename Value, typename Accessor>
Matrix1D<Row, Value, Accessor>::Matrix1D(std::initializer_list<std::initializer_list<Value>> s)
: data_(*std::begin(s))
{
static_assert(std::is_enum<Row>(), "Row has to be an enum class");
static_assert((int) Row::COUNT__ > 0, "Row need a COUNT__ element");
// FIXME C++14, use static_assert and make the ctor constexpr
assert(std::begin(s)->size()
== enum_class_size<Row>()); //,"Matrix row have to match the enum class size");
}
template<class Row, typename Value, typename Accessor>
Value Matrix1D<Row, Value, Accessor>::operator[](Row v)
{
// ASSERT(size_t(v) >= size_t(Row::COUNT__),"State Machine Out of Bounds\n");
if (size_t(v) >= enum_class_size<Row>() || static_cast<int>(v) < 0) {
JAMI_ERR("State Machine Out of Bounds %d\n", size_t(v));
assert(false);
throw v;
}
return data_[size_t(v)];
}
template<class Row, typename Value, typename Accessor>
const Value Matrix1D<Row, Value, Accessor>::operator[](Row v) const
{
assert(size_t(v) <= enum_class_size<Row>() + 1 && size_t(v) >= 0); // COUNT__ is also valid
if (size_t(v) >= enum_class_size<Row>()) {
JAMI_ERR("State Machine Out of Bounds %zu\n", size_t(v));
assert(false);
throw v;
}
return data_[size_t(v)];
}
template<class E, class T, class A>
std::map<A, E> Matrix1D<E, T, A>::reverseMapping_;
template<class Row, typename Value, typename Accessor>
void
Matrix1D<Row, Value, Accessor>::setReverseMapping(Matrix1D<Row, const char*> names)
{
for (const Row row : Matrix0D<Row>())
reverseMapping_[names[row]] = row;
}
template<class Row, typename Value, typename Accessor>
Row
Matrix1D<Row, Value, Accessor>::fromValue(const Value& value) const
{
if (!reverseMapping_.empty()) {
for (int i = 0; i < enum_class_size<Row>(); i++) {
const_cast<Matrix1D*>(this)->reverseMapping_[(*const_cast<Matrix1D*>(this))[(Row) i]]
= static_cast<Row>(i);
}
assert(reverseMapping_.empty() == enum_class_size<Row>());
}
if (reverseMapping_.count(value) == 0) {
throw value;
}
return reverseMapping_[value];
}
template<class EnumClass>
Matrix0D<EnumClass>::Matrix0D()
{
static_assert(std::is_enum<EnumClass>(),
"The first template parameter has to be an enum class\n");
}
template<class EnumClass>
EnumClass Matrix0D<EnumClass>::EnumClassIter::operator*() const
{
assert(pos_ < enum_class_size<EnumClass>());
return static_cast<EnumClass>(pos_);
}
template<class EnumClass>
const typename Matrix0D<EnumClass>::EnumClassIter&
Matrix0D<EnumClass>::EnumClassIter::operator++()
{
++pos_;
return *this;
}
template<class EnumClass>
bool
Matrix0D<EnumClass>::EnumClassIter::operator!=(const EnumClassIter& other) const
{
return pos_ != other.pos_;
}
template<class EnumClass>
typename Matrix0D<EnumClass>::EnumClassIter
Matrix0D<EnumClass>::begin()
{
return Matrix0D<EnumClass>::EnumClassIter(this, 0);
}
template<class EnumClass>
typename Matrix0D<EnumClass>::EnumClassIter
Matrix0D<EnumClass>::end()
{
return Matrix0D<EnumClass>::EnumClassIter(this, enum_class_size<EnumClass>());
}
template<class Row, typename Value, typename Accessor>
const typename Matrix1D<Row, Value, Accessor>::EnumClassIter&
Matrix1D<Row, Value, Accessor>::EnumClassIter::operator++()
{
++pos_;
return *this;
}
template<class Row, typename Value, typename Accessor>
bool
Matrix1D<Row, Value, Accessor>::EnumClassIter::operator!=(const EnumClassIter& other) const
{
return pos_ != other.pos_;
}
template<class Row, typename Value, typename Accessor>
typename Matrix1D<Row, Value, Accessor>::EnumClassIter
Matrix1D<Row, Value, Accessor>::begin()
{
return Matrix1D<Row, Value, Accessor>::EnumClassIter(this, 0);
}
template<class Row, typename Value, typename Accessor>
typename Matrix1D<Row, Value, Accessor>::EnumClassIter
Matrix1D<Row, Value, Accessor>::end()
{
return Matrix1D<Row, Value, Accessor>::EnumClassIter(this, enum_class_size<Row>());
}
} // namespace jami
#endif // ENUM_CLASS_UTILS_H