Ethereum  PoC-8
The C++ Implementation of Ethereum
vector_ref.h
Go to the documentation of this file.
1 #pragma once
2 
3 #include <cstring>
4 #include <cassert>
5 #include <type_traits>
6 #include <vector>
7 #include <string>
8 #include <atomic>
9 
10 #ifdef __INTEL_COMPILER
11 #pragma warning(disable:597) //will not be called for implicit or explicit conversions
12 #endif
13 
14 namespace dev
15 {
16 
20 template <class _T>
22 {
23 public:
24  using value_type = _T;
25  using element_type = _T;
26  using mutable_value_type = typename std::conditional<std::is_const<_T>::value, typename std::remove_const<_T>::type, _T>::type;
27 
28  static_assert(std::is_pod<value_type>::value, "vector_ref can only be used with PODs due to its low-level treatment of data.");
29 
30  vector_ref(): m_data(nullptr), m_count(0) {}
32  vector_ref(_T* _data, size_t _count): m_data(_data), m_count(_count) {}
34  vector_ref(typename std::conditional<std::is_const<_T>::value, std::string const*, std::string*>::type _data): m_data(reinterpret_cast<_T*>(_data->data())), m_count(_data->size() / sizeof(_T)) {}
36  vector_ref(typename std::conditional<std::is_const<_T>::value, std::vector<typename std::remove_const<_T>::type> const*, std::vector<_T>*>::type _data): m_data(_data->data()), m_count(_data->size()) {}
38  vector_ref(typename std::conditional<std::is_const<_T>::value, std::string const&, std::string&>::type _data): m_data(reinterpret_cast<_T*>(_data.data())), m_count(_data.size() / sizeof(_T)) {}
39  explicit operator bool() const { return m_data && m_count; }
40 
41  bool contentsEqual(std::vector<mutable_value_type> const& _c) const { if (!m_data || m_count == 0) return _c.empty(); else return _c.size() == m_count && !memcmp(_c.data(), m_data, m_count * sizeof(_T)); }
42  std::vector<mutable_value_type> toVector() const { return std::vector<mutable_value_type>(m_data, m_data + m_count); }
43  std::vector<unsigned char> toBytes() const { return std::vector<unsigned char>(reinterpret_cast<unsigned char const*>(m_data), reinterpret_cast<unsigned char const*>(m_data) + m_count * sizeof(_T)); }
44  std::string toString() const { return std::string((char const*)m_data, ((char const*)m_data) + m_count * sizeof(_T)); }
45 
46  template <class _T2> explicit operator vector_ref<_T2>() const { assert(m_count * sizeof(_T) / sizeof(_T2) * sizeof(_T2) / sizeof(_T) == m_count); return vector_ref<_T2>(reinterpret_cast<_T2*>(m_data), m_count * sizeof(_T) / sizeof(_T2)); }
47  operator vector_ref<_T const>() const { return vector_ref<_T const>(m_data, m_count); }
48 
49  _T* data() const { return m_data; }
51  size_t count() const { return m_count; }
53  size_t size() const { return m_count; }
54  bool empty() const { return !m_count; }
56  vector_ref<_T> next() const { if (!m_data) return *this; else return vector_ref<_T>(m_data + m_count, m_count); }
60  vector_ref<_T> cropped(size_t _begin, size_t _count) const { if (m_data && _begin <= m_count && _count <= m_count && _begin + _count <= m_count) return vector_ref<_T>(m_data + _begin, _count == ~size_t(0) ? m_count - _begin : _count); else return vector_ref<_T>(); }
62  vector_ref<_T> cropped(size_t _begin) const { if (m_data && _begin <= m_count) return vector_ref<_T>(m_data + _begin, m_count - _begin); else return vector_ref<_T>(); }
63  void retarget(_T* _d, size_t _s) { m_data = _d; m_count = _s; }
64  void retarget(std::vector<_T> const& _t) { m_data = _t.data(); m_count = _t.size(); }
65  template <class T> bool overlapsWith(vector_ref<T> _t) const { void const* f1 = data(); void const* t1 = data() + size(); void const* f2 = _t.data(); void const* t2 = _t.data() + _t.size(); return f1 < t2 && t1 > f2; }
67  void copyTo(vector_ref<typename std::remove_const<_T>::type> _t) const { if (overlapsWith(_t)) memmove(_t.data(), m_data, std::min(_t.size(), m_count) * sizeof(_T)); else memcpy(_t.data(), m_data, std::min(_t.size(), m_count) * sizeof(_T)); }
69  void populate(vector_ref<typename std::remove_const<_T>::type> _t) const { copyTo(_t); memset(_t.data() + m_count, 0, std::max(_t.size(), m_count) - m_count); }
72  void cleanse()
73  {
74  static std::atomic<unsigned char> s_cleanseCounter{0u};
75  uint8_t* p = (uint8_t*)begin();
76  size_t const len = (uint8_t*)end() - p;
77  size_t loop = len;
78  size_t count = s_cleanseCounter;
79  while (loop--)
80  {
81  *(p++) = (uint8_t)count;
82  count += (17 + ((size_t)p & 0xf));
83  }
84  p = (uint8_t*)memchr((uint8_t*)begin(), (uint8_t)count, len);
85  if (p)
86  count += (63 + (size_t)p);
87  s_cleanseCounter = (uint8_t)count;
88  memset((uint8_t*)begin(), 0, len);
89  }
90 
91  _T* begin() { return m_data; }
92  _T* end() { return m_data + m_count; }
93  _T const* begin() const { return m_data; }
94  _T const* end() const { return m_data + m_count; }
95 
96  _T& operator[](size_t _i) { assert(m_data); assert(_i < m_count); return m_data[_i]; }
97  _T const& operator[](size_t _i) const { assert(m_data); assert(_i < m_count); return m_data[_i]; }
98 
99  bool operator==(vector_ref<_T> const& _cmp) const { return m_data == _cmp.m_data && m_count == _cmp.m_count; }
100  bool operator!=(vector_ref<_T> const& _cmp) const { return !operator==(_cmp); }
101 
102  void reset() { m_data = nullptr; m_count = 0; }
103 
104 private:
105  _T* m_data;
106  size_t m_count;
107 };
108 
109 template<class _T> vector_ref<_T const> ref(_T const& _t) { return vector_ref<_T const>(&_t, 1); }
110 template<class _T> vector_ref<_T> ref(_T& _t) { return vector_ref<_T>(&_t, 1); }
111 template<class _T> vector_ref<_T const> ref(std::vector<_T> const& _t) { return vector_ref<_T const>(&_t); }
112 template<class _T> vector_ref<_T> ref(std::vector<_T>& _t) { return vector_ref<_T>(&_t); }
113 
114 }
dev::vector_ref::begin
_T * begin()
Definition: vector_ref.h:91
dev::ref
vector_ref< _T const > ref(_T const &_t)
Definition: vector_ref.h:109
dev::vector_ref::vector_ref
vector_ref(typename std::conditional< std::is_const< _T >::value, std::string const &, std::string & >::type _data)
Creates a new vector_ref pointing to the data part of a string (given as reference).
Definition: vector_ref.h:38
dev::vector_ref
Definition: vector_ref.h:22
dev::vector_ref::toVector
std::vector< mutable_value_type > toVector() const
Definition: vector_ref.h:42
dev::vector_ref::operator[]
_T const & operator[](size_t _i) const
Definition: vector_ref.h:97
dev::vector_ref::copyTo
void copyTo(vector_ref< typename std::remove_const< _T >::type > _t) const
Copies the contents of this vector_ref to the contents of _t, up to the max size of _t.
Definition: vector_ref.h:67
dev::vector_ref< byte const >::value_type
byte const value_type
Definition: vector_ref.h:24
dev::vector_ref::retarget
void retarget(std::vector< _T > const &_t)
Definition: vector_ref.h:64
dev::vector_ref::next
vector_ref< _T > next() const
Definition: vector_ref.h:56
dev::vector_ref::end
_T * end()
Definition: vector_ref.h:92
dev::vector_ref::begin
_T const * begin() const
Definition: vector_ref.h:93
dev::vector_ref::populate
void populate(vector_ref< typename std::remove_const< _T >::type > _t) const
Copies the contents of this vector_ref to the contents of _t, and zeros further trailing elements in ...
Definition: vector_ref.h:69
dev::vector_ref::cropped
vector_ref< _T > cropped(size_t _begin) const
Definition: vector_ref.h:62
dev::vector_ref::cropped
vector_ref< _T > cropped(size_t _begin, size_t _count) const
Definition: vector_ref.h:60
dev::vector_ref::operator[]
_T & operator[](size_t _i)
Definition: vector_ref.h:96
dev::vector_ref::operator!=
bool operator!=(vector_ref< _T > const &_cmp) const
Definition: vector_ref.h:100
dev::vector_ref::toString
std::string toString() const
Definition: vector_ref.h:44
dev::vector_ref::data
_T * data() const
Definition: vector_ref.h:49
dev::vector_ref::size
size_t size() const
Definition: vector_ref.h:53
dev::vector_ref::vector_ref
vector_ref(_T *_data, size_t _count)
Creates a new vector_ref to point to _count elements starting at _data.
Definition: vector_ref.h:32
dev::vector_ref< byte const >::mutable_value_type
typename std::conditional< std::is_const< byte const >::value, typename std::remove_const< byte const >::type, byte const >::type mutable_value_type
Definition: vector_ref.h:26
dev::vector_ref< byte const >::element_type
byte const element_type
Definition: vector_ref.h:25
dev::vector_ref::empty
bool empty() const
Definition: vector_ref.h:54
dev::vector_ref::retarget
void retarget(_T *_d, size_t _s)
Definition: vector_ref.h:63
dev::vector_ref::toBytes
std::vector< unsigned char > toBytes() const
Definition: vector_ref.h:43
dev::vector_ref::overlapsWith
bool overlapsWith(vector_ref< T > _t) const
Definition: vector_ref.h:65
dev
Definition: Address.cpp:21
dev::vector_ref::cleanse
void cleanse()
Definition: vector_ref.h:72
dev::vector_ref::reset
void reset()
Definition: vector_ref.h:102
dev::vector_ref::count
size_t count() const
Definition: vector_ref.h:51
dev::vector_ref::end
_T const * end() const
Definition: vector_ref.h:94
dev::vector_ref::vector_ref
vector_ref(typename std::conditional< std::is_const< _T >::value, std::vector< typename std::remove_const< _T >::type > const *, std::vector< _T > * >::type _data)
Creates a new vector_ref pointing to the data part of a vector (given as pointer).
Definition: vector_ref.h:36
dev::vector_ref::vector_ref
vector_ref()
Definition: vector_ref.h:30
dev::vector_ref::operator==
bool operator==(vector_ref< _T > const &_cmp) const
Definition: vector_ref.h:99
dev::vector_ref::vector_ref
vector_ref(typename std::conditional< std::is_const< _T >::value, std::string const *, std::string * >::type _data)
Creates a new vector_ref pointing to the data part of a string (given as pointer).
Definition: vector_ref.h:34
dev::vector_ref::contentsEqual
bool contentsEqual(std::vector< mutable_value_type > const &_c) const
Definition: vector_ref.h:41