2019-08-16 20:56:07 +01:00
|
|
|
#ifndef ARRAY_SLICE_H
|
|
|
|
#define ARRAY_SLICE_H
|
|
|
|
|
2020-01-15 21:12:10 +00:00
|
|
|
#include "fixed_array.h"
|
2019-08-16 20:56:07 +01:00
|
|
|
#include <core/error_macros.h>
|
2020-01-19 18:13:00 +00:00
|
|
|
#include <vector>
|
2019-08-16 20:56:07 +01:00
|
|
|
|
2020-01-17 20:43:28 +00:00
|
|
|
// View into an array, referencing a pointer and a size.
|
2019-08-16 20:56:07 +01:00
|
|
|
template <typename T>
|
|
|
|
class ArraySlice {
|
|
|
|
public:
|
2020-01-25 23:26:55 +00:00
|
|
|
inline ArraySlice() :
|
|
|
|
_ptr(nullptr),
|
|
|
|
_size(0) {
|
|
|
|
}
|
|
|
|
|
2020-01-19 18:13:00 +00:00
|
|
|
// TODO Get rid of unsafe constructor, use specialized ones
|
2020-01-15 21:12:10 +00:00
|
|
|
inline ArraySlice(T *p_ptr, size_t p_begin, size_t p_end) {
|
2019-08-16 20:56:07 +01:00
|
|
|
CRASH_COND(p_end <= p_begin);
|
|
|
|
_ptr = p_ptr + p_begin;
|
|
|
|
_size = p_end - p_begin;
|
|
|
|
}
|
|
|
|
|
2021-01-07 22:19:06 +00:00
|
|
|
inline ArraySlice(T *p_ptr, size_t p_size) :
|
|
|
|
_ptr(p_ptr), _size(p_size) {}
|
|
|
|
|
|
|
|
inline ArraySlice(ArraySlice<T> &p_other, size_t p_begin, size_t p_end) {
|
|
|
|
CRASH_COND(p_end <= p_begin);
|
|
|
|
CRASH_COND(p_begin >= vec.size());
|
|
|
|
CRASH_COND(p_end > vec.size()); // `>` because p_end is typically `p_begin + size`
|
|
|
|
_ptr = p_other._ptr + p_begin;
|
|
|
|
_size = p_end - p_begin;
|
|
|
|
}
|
|
|
|
|
2020-01-19 18:13:00 +00:00
|
|
|
inline ArraySlice(std::vector<T> &vec, size_t p_begin, size_t p_end) {
|
|
|
|
CRASH_COND(p_end <= p_begin);
|
|
|
|
CRASH_COND(p_begin >= vec.size());
|
|
|
|
CRASH_COND(p_end > vec.size()); // `>` because p_end is typically `p_begin + size`
|
|
|
|
_ptr = &vec[p_begin];
|
|
|
|
_size = p_end - p_begin;
|
|
|
|
}
|
|
|
|
|
2020-01-15 21:12:10 +00:00
|
|
|
template <unsigned int N>
|
|
|
|
inline ArraySlice(FixedArray<T, N> &a) {
|
|
|
|
_ptr = a.data();
|
|
|
|
_size = a.size();
|
|
|
|
}
|
|
|
|
|
2021-01-07 22:19:06 +00:00
|
|
|
inline ArraySlice<T> sub(size_t from, size_t len) const {
|
|
|
|
CRASH_COND(from + len >= _size);
|
|
|
|
return ArraySlice<T>(_ptr + from, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline ArraySlice<T> sub(size_t from) const {
|
|
|
|
CRASH_COND(from >= _size);
|
|
|
|
return ArraySlice<T>(_ptr + from, _size - from);
|
|
|
|
}
|
|
|
|
|
|
|
|
// const ArraySlice<T> sub_const(size_t from, size_t len) const {
|
|
|
|
// CRASH_COND(from + len >= _size);
|
|
|
|
// return ArraySlice{ _ptr + from, len };
|
|
|
|
// }
|
|
|
|
|
|
|
|
// const ArraySlice<T> sub_const(size_t from) const {
|
|
|
|
// CRASH_COND(from >= _size);
|
|
|
|
// return ArraySlice{ _ptr + from, _size - from };
|
|
|
|
// }
|
|
|
|
|
2020-01-25 23:26:55 +00:00
|
|
|
template <typename U>
|
|
|
|
ArraySlice<U> reinterpret_cast_to() const {
|
|
|
|
const size_t size_in_bytes = _size * sizeof(T);
|
2020-02-11 21:20:05 +00:00
|
|
|
#ifdef DEBUG_ENABLED
|
2020-01-25 23:26:55 +00:00
|
|
|
CRASH_COND(size_in_bytes % sizeof(U) != 0);
|
|
|
|
#endif
|
2020-12-18 20:46:54 +00:00
|
|
|
return ArraySlice<U>(reinterpret_cast<U *>(_ptr), 0, size_in_bytes / sizeof(U));
|
2020-01-25 23:26:55 +00:00
|
|
|
}
|
|
|
|
|
2019-08-16 20:56:07 +01:00
|
|
|
inline T &operator[](size_t i) {
|
2020-02-11 21:20:05 +00:00
|
|
|
#ifdef DEBUG_ENABLED
|
2020-02-07 04:52:06 +01:00
|
|
|
CRASH_COND(i >= _size);
|
2019-08-16 20:56:07 +01:00
|
|
|
#endif
|
|
|
|
return _ptr[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
inline const T &operator[](size_t i) const {
|
2020-02-11 21:20:05 +00:00
|
|
|
#ifdef DEBUG_ENABLED
|
2020-02-07 04:52:06 +01:00
|
|
|
CRASH_COND(i >= _size);
|
2019-08-16 20:56:07 +01:00
|
|
|
#endif
|
|
|
|
return _ptr[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
inline size_t size() const {
|
|
|
|
return _size;
|
|
|
|
}
|
|
|
|
|
2020-01-15 21:12:10 +00:00
|
|
|
inline T *data() {
|
|
|
|
return _ptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline const T *data() const {
|
|
|
|
return _ptr;
|
|
|
|
}
|
|
|
|
|
2019-08-16 20:56:07 +01:00
|
|
|
private:
|
|
|
|
T *_ptr;
|
|
|
|
size_t _size;
|
|
|
|
};
|
|
|
|
|
|
|
|
#endif // ARRAY_SLICE_H
|