123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139 |
- #pragma once
- #include <nall/iterator.hpp>
- #include <nall/range.hpp>
- #include <nall/traits.hpp>
- namespace nall {
- template<typename T> struct array_view {
- using type = array_view;
- inline array_view() {
- _data = nullptr;
- _size = 0;
- }
- inline array_view(nullptr_t) {
- _data = nullptr;
- _size = 0;
- }
- inline array_view(const void* data, uint64_t size) {
- _data = (const T*)data;
- _size = (int)size;
- }
- inline explicit operator bool() const { return _data && _size > 0; }
- inline operator const T*() const {
- #ifdef DEBUG
- struct out_of_bounds {};
- if(_size < 0) throw out_of_bounds{};
- #endif
- return _data;
- }
- inline auto operator++() -> type& { _data++; _size--; return *this; }
- inline auto operator--() -> type& { _data--; _size++; return *this; }
- inline auto operator++(int) -> type { auto copy = *this; ++(*this); return copy; }
- inline auto operator--(int) -> type { auto copy = *this; --(*this); return copy; }
- inline auto operator-=(int distance) -> type& { _data -= distance; _size += distance; return *this; }
- inline auto operator+=(int distance) -> type& { _data += distance; _size -= distance; return *this; }
- inline auto operator[](uint index) const -> const T& {
- #ifdef DEBUG
- struct out_of_bounds {};
- if(index >= _size) throw out_of_bounds{};
- #endif
- return _data[index];
- }
- inline auto operator()(uint index, const T& fallback = {}) const -> T {
- if(index >= _size) return fallback;
- return _data[index];
- }
- template<typename U = T> inline auto data() const -> const U* { return (const U*)_data; }
- template<typename U = T> inline auto size() const -> uint64_t { return _size * sizeof(T) / sizeof(U); }
- inline auto begin() const -> iterator_const<T> { return {_data, (uint)0}; }
- inline auto end() const -> iterator_const<T> { return {_data, (uint)_size}; }
- inline auto rbegin() const -> reverse_iterator_const<T> { return {_data, (uint)_size - 1}; }
- inline auto rend() const -> reverse_iterator_const<T> { return {_data, (uint)-1}; }
- auto read() -> T {
- auto value = operator[](0);
- _data++;
- _size--;
- return value;
- }
- auto view(uint offset, uint length) const -> type {
- #ifdef DEBUG
- struct out_of_bounds {};
- if(offset + length >= _size) throw out_of_bounds{};
- #endif
- return {_data + offset, length};
- }
- //array_view<uint8_t> specializations
- template<typename U> auto readl(U& value, uint size) -> U;
- template<typename U> auto readm(U& value, uint size) -> U;
- template<typename U> auto readvn(U& value, uint size) -> U;
- template<typename U> auto readvi(U& value, uint size) -> U;
- template<typename U> auto readl(U& value, uint offset, uint size) -> U { return view(offset, size).readl(value, size); }
- template<typename U = uint64_t> auto readl(uint size) -> U { U value; return readl(value, size); }
- template<typename U = uint64_t> auto readm(uint size) -> U { U value; return readm(value, size); }
- template<typename U = uint64_t> auto readvn(uint size) -> U { U value; return readvn(value, size); }
- template<typename U = int64_t> auto readvi(uint size) -> U { U value; return readvi(value, size); }
- template<typename U = uint64_t> auto readl(uint offset, uint size) -> U { U value; return readl(value, offset, size); }
- protected:
- const T* _data;
- int _size;
- };
- //array_view<uint8_t>
- template<> template<typename U> inline auto array_view<uint8_t>::readl(U& value, uint size) -> U {
- value = 0;
- for(uint byte : range(size)) value |= (U)read() << byte * 8;
- return value;
- }
- template<> template<typename U> inline auto array_view<uint8_t>::readm(U& value, uint size) -> U {
- value = 0;
- for(uint byte : reverse(range(size))) value |= (U)read() << byte * 8;
- return value;
- }
- template<> template<typename U> inline auto array_view<uint8_t>::readvn(U& value, uint size) -> U {
- value = 0;
- uint shift = 1;
- while(true) {
- auto byte = read();
- value += (byte & 0x7f) * shift;
- if(byte & 0x80) break;
- shift <<= 7;
- value += shift;
- }
- return value;
- }
- template<> template<typename U> inline auto array_view<uint8_t>::readvi(U& value, uint size) -> U {
- value = readvn<U>();
- bool negate = value & 1;
- value >>= 1;
- if(negate) value = ~value;
- return value;
- }
- }
|