2020-04-16 13:14:08 -04:00
// Copyright (c) 2015-2020 The Bitcoin Core developers
2015-01-05 17:40:24 -03:00
// Distributed under the MIT software license, see the accompanying
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2017-08-25 20:56:53 -03:00
# ifndef BITCOIN_PREVECTOR_H
# define BITCOIN_PREVECTOR_H
2015-10-29 03:11:24 -03:00
2017-02-25 03:06:25 -03:00
# include <assert.h>
2022-09-23 06:48:47 -03:00
# include <cstdlib>
2015-10-29 03:11:24 -03:00
# include <stdint.h>
# include <string.h>
2018-11-04 00:15:05 -03:00
# include <algorithm>
2018-02-27 02:39:22 -03:00
# include <cstddef>
2017-01-10 12:59:48 -03:00
# include <type_traits>
2019-11-29 16:18:24 -03:00
# include <utility>
2015-10-29 03:11:24 -03:00
/** Implements a drop-in replacement for std::vector<T> which stores up to N
* elements directly ( without heap allocation ) . The types Size and Diff are
* used to store element counts , and can be any unsigned + signed type .
*
* Storage layout is either :
* - Direct allocation :
* - Size _size : the number of used elements ( between 0 and N )
* - T direct [ N ] : an array of N elements of type T
* ( only the first _size are initialized ) .
* - Indirect allocation :
* - Size _size : the number of used elements plus N + 1
* - Size capacity : the number of allocated elements
* - T * indirect : a pointer to an array of capacity elements of type T
* ( only the first _size are initialized ) .
*
* The data type T must be movable by memmove / realloc ( ) . Once we switch to C + + ,
* move constructors can be used instead .
*/
template < unsigned int N , typename T , typename Size = uint32_t , typename Diff = int32_t >
class prevector {
2022-04-24 06:55:57 -04:00
static_assert ( std : : is_trivially_copyable_v < T > ) ;
2015-10-29 03:11:24 -03:00
public :
typedef Size size_type ;
typedef Diff difference_type ;
typedef T value_type ;
typedef value_type & reference ;
typedef const value_type & const_reference ;
typedef value_type * pointer ;
typedef const value_type * const_pointer ;
class iterator {
T * ptr ;
public :
typedef Diff difference_type ;
typedef T value_type ;
typedef T * pointer ;
typedef T & reference ;
typedef std : : random_access_iterator_tag iterator_category ;
iterator ( T * ptr_ ) : ptr ( ptr_ ) { }
T & operator * ( ) const { return * ptr ; }
T * operator - > ( ) const { return ptr ; }
T & operator [ ] ( size_type pos ) { return ptr [ pos ] ; }
const T & operator [ ] ( size_type pos ) const { return ptr [ pos ] ; }
iterator & operator + + ( ) { ptr + + ; return * this ; }
iterator & operator - - ( ) { ptr - - ; return * this ; }
iterator operator + + ( int ) { iterator copy ( * this ) ; + + ( * this ) ; return copy ; }
iterator operator - - ( int ) { iterator copy ( * this ) ; - - ( * this ) ; return copy ; }
difference_type friend operator - ( iterator a , iterator b ) { return ( & ( * a ) - & ( * b ) ) ; }
iterator operator + ( size_type n ) { return iterator ( ptr + n ) ; }
iterator & operator + = ( size_type n ) { ptr + = n ; return * this ; }
iterator operator - ( size_type n ) { return iterator ( ptr - n ) ; }
iterator & operator - = ( size_type n ) { ptr - = n ; return * this ; }
bool operator = = ( iterator x ) const { return ptr = = x . ptr ; }
bool operator ! = ( iterator x ) const { return ptr ! = x . ptr ; }
bool operator > = ( iterator x ) const { return ptr > = x . ptr ; }
bool operator < = ( iterator x ) const { return ptr < = x . ptr ; }
bool operator > ( iterator x ) const { return ptr > x . ptr ; }
bool operator < ( iterator x ) const { return ptr < x . ptr ; }
} ;
class reverse_iterator {
T * ptr ;
public :
typedef Diff difference_type ;
typedef T value_type ;
typedef T * pointer ;
typedef T & reference ;
typedef std : : bidirectional_iterator_tag iterator_category ;
reverse_iterator ( T * ptr_ ) : ptr ( ptr_ ) { }
T & operator * ( ) { return * ptr ; }
const T & operator * ( ) const { return * ptr ; }
T * operator - > ( ) { return ptr ; }
const T * operator - > ( ) const { return ptr ; }
reverse_iterator & operator - - ( ) { ptr + + ; return * this ; }
reverse_iterator & operator + + ( ) { ptr - - ; return * this ; }
reverse_iterator operator + + ( int ) { reverse_iterator copy ( * this ) ; + + ( * this ) ; return copy ; }
reverse_iterator operator - - ( int ) { reverse_iterator copy ( * this ) ; - - ( * this ) ; return copy ; }
bool operator = = ( reverse_iterator x ) const { return ptr = = x . ptr ; }
bool operator ! = ( reverse_iterator x ) const { return ptr ! = x . ptr ; }
} ;
class const_iterator {
const T * ptr ;
public :
typedef Diff difference_type ;
typedef const T value_type ;
typedef const T * pointer ;
typedef const T & reference ;
typedef std : : random_access_iterator_tag iterator_category ;
const_iterator ( const T * ptr_ ) : ptr ( ptr_ ) { }
const_iterator ( iterator x ) : ptr ( & ( * x ) ) { }
const T & operator * ( ) const { return * ptr ; }
const T * operator - > ( ) const { return ptr ; }
const T & operator [ ] ( size_type pos ) const { return ptr [ pos ] ; }
const_iterator & operator + + ( ) { ptr + + ; return * this ; }
const_iterator & operator - - ( ) { ptr - - ; return * this ; }
const_iterator operator + + ( int ) { const_iterator copy ( * this ) ; + + ( * this ) ; return copy ; }
const_iterator operator - - ( int ) { const_iterator copy ( * this ) ; - - ( * this ) ; return copy ; }
difference_type friend operator - ( const_iterator a , const_iterator b ) { return ( & ( * a ) - & ( * b ) ) ; }
const_iterator operator + ( size_type n ) { return const_iterator ( ptr + n ) ; }
const_iterator & operator + = ( size_type n ) { ptr + = n ; return * this ; }
const_iterator operator - ( size_type n ) { return const_iterator ( ptr - n ) ; }
const_iterator & operator - = ( size_type n ) { ptr - = n ; return * this ; }
bool operator = = ( const_iterator x ) const { return ptr = = x . ptr ; }
bool operator ! = ( const_iterator x ) const { return ptr ! = x . ptr ; }
bool operator > = ( const_iterator x ) const { return ptr > = x . ptr ; }
bool operator < = ( const_iterator x ) const { return ptr < = x . ptr ; }
bool operator > ( const_iterator x ) const { return ptr > x . ptr ; }
bool operator < ( const_iterator x ) const { return ptr < x . ptr ; }
} ;
class const_reverse_iterator {
const T * ptr ;
public :
typedef Diff difference_type ;
typedef const T value_type ;
typedef const T * pointer ;
typedef const T & reference ;
typedef std : : bidirectional_iterator_tag iterator_category ;
2017-06-18 19:45:23 -04:00
const_reverse_iterator ( const T * ptr_ ) : ptr ( ptr_ ) { }
2015-10-29 03:11:24 -03:00
const_reverse_iterator ( reverse_iterator x ) : ptr ( & ( * x ) ) { }
const T & operator * ( ) const { return * ptr ; }
const T * operator - > ( ) const { return ptr ; }
const_reverse_iterator & operator - - ( ) { ptr + + ; return * this ; }
const_reverse_iterator & operator + + ( ) { ptr - - ; return * this ; }
const_reverse_iterator operator + + ( int ) { const_reverse_iterator copy ( * this ) ; + + ( * this ) ; return copy ; }
const_reverse_iterator operator - - ( int ) { const_reverse_iterator copy ( * this ) ; - - ( * this ) ; return copy ; }
bool operator = = ( const_reverse_iterator x ) const { return ptr = = x . ptr ; }
bool operator ! = ( const_reverse_iterator x ) const { return ptr ! = x . ptr ; }
} ;
private :
2019-12-10 04:05:21 -03:00
# pragma pack(push, 1)
2015-12-01 11:20:35 -03:00
union direct_or_indirect {
2015-10-29 03:11:24 -03:00
char direct [ sizeof ( T ) * N ] ;
struct {
char * indirect ;
2019-12-10 04:05:21 -03:00
size_type capacity ;
2018-12-31 07:50:06 -03:00
} indirect_contents ;
2019-12-10 04:05:21 -03:00
} ;
# pragma pack(pop)
alignas ( char * ) direct_or_indirect _union = { } ;
size_type _size = 0 ;
static_assert ( alignof ( char * ) % alignof ( size_type ) = = 0 & & sizeof ( char * ) % alignof ( size_type ) = = 0 , " size_type cannot have more restrictive alignment requirement than pointer " ) ;
static_assert ( alignof ( char * ) % alignof ( T ) = = 0 , " value_type T cannot have more restrictive alignment requirement than pointer " ) ;
2015-10-29 03:11:24 -03:00
T * direct_ptr ( difference_type pos ) { return reinterpret_cast < T * > ( _union . direct ) + pos ; }
const T * direct_ptr ( difference_type pos ) const { return reinterpret_cast < const T * > ( _union . direct ) + pos ; }
2018-12-31 07:50:06 -03:00
T * indirect_ptr ( difference_type pos ) { return reinterpret_cast < T * > ( _union . indirect_contents . indirect ) + pos ; }
const T * indirect_ptr ( difference_type pos ) const { return reinterpret_cast < const T * > ( _union . indirect_contents . indirect ) + pos ; }
2015-10-29 03:11:24 -03:00
bool is_direct ( ) const { return _size < = N ; }
void change_capacity ( size_type new_capacity ) {
if ( new_capacity < = N ) {
if ( ! is_direct ( ) ) {
T * indirect = indirect_ptr ( 0 ) ;
T * src = indirect ;
T * dst = direct_ptr ( 0 ) ;
memcpy ( dst , src , size ( ) * sizeof ( T ) ) ;
free ( indirect ) ;
_size - = N + 1 ;
}
} else {
if ( ! is_direct ( ) ) {
2017-02-25 03:06:25 -03:00
/* FIXME: Because malloc/realloc here won't call new_handler if allocation fails, assert
success . These should instead use an allocator or new / delete so that handlers
are called as necessary , but performance would be slightly degraded by doing so . */
2018-12-31 07:50:06 -03:00
_union . indirect_contents . indirect = static_cast < char * > ( realloc ( _union . indirect_contents . indirect , ( ( size_t ) sizeof ( T ) ) * new_capacity ) ) ;
assert ( _union . indirect_contents . indirect ) ;
_union . indirect_contents . capacity = new_capacity ;
2015-10-29 03:11:24 -03:00
} else {
char * new_indirect = static_cast < char * > ( malloc ( ( ( size_t ) sizeof ( T ) ) * new_capacity ) ) ;
2017-02-25 03:06:25 -03:00
assert ( new_indirect ) ;
2015-10-29 03:11:24 -03:00
T * src = direct_ptr ( 0 ) ;
T * dst = reinterpret_cast < T * > ( new_indirect ) ;
memcpy ( dst , src , size ( ) * sizeof ( T ) ) ;
2018-12-31 07:50:06 -03:00
_union . indirect_contents . indirect = new_indirect ;
_union . indirect_contents . capacity = new_capacity ;
2015-10-29 03:11:24 -03:00
_size + = N + 1 ;
}
}
}
T * item_ptr ( difference_type pos ) { return is_direct ( ) ? direct_ptr ( pos ) : indirect_ptr ( pos ) ; }
const T * item_ptr ( difference_type pos ) const { return is_direct ( ) ? direct_ptr ( pos ) : indirect_ptr ( pos ) ; }
2018-11-13 06:15:27 -03:00
void fill ( T * dst , ptrdiff_t count , const T & value = T { } ) {
2018-11-04 00:15:05 -03:00
std : : fill_n ( dst , count , value ) ;
2017-12-22 07:04:30 -03:00
}
template < typename InputIterator >
void fill ( T * dst , InputIterator first , InputIterator last ) {
while ( first ! = last ) {
new ( static_cast < void * > ( dst ) ) T ( * first ) ;
+ + dst ;
+ + first ;
}
}
2015-10-29 03:11:24 -03:00
public :
void assign ( size_type n , const T & val ) {
clear ( ) ;
if ( capacity ( ) < n ) {
change_capacity ( n ) ;
}
2017-12-22 07:04:30 -03:00
_size + = n ;
fill ( item_ptr ( 0 ) , n , val ) ;
2015-10-29 03:11:24 -03:00
}
template < typename InputIterator >
void assign ( InputIterator first , InputIterator last ) {
size_type n = last - first ;
clear ( ) ;
if ( capacity ( ) < n ) {
change_capacity ( n ) ;
}
2017-12-22 07:04:30 -03:00
_size + = n ;
fill ( item_ptr ( 0 ) , first , last ) ;
2015-10-29 03:11:24 -03:00
}
2018-09-19 03:02:12 -03:00
prevector ( ) { }
2015-10-29 03:11:24 -03:00
2018-09-19 03:02:12 -03:00
explicit prevector ( size_type n ) {
2015-10-29 03:11:24 -03:00
resize ( n ) ;
}
2018-09-19 03:02:12 -03:00
explicit prevector ( size_type n , const T & val ) {
2015-10-29 03:11:24 -03:00
change_capacity ( n ) ;
2017-12-22 07:04:30 -03:00
_size + = n ;
fill ( item_ptr ( 0 ) , n , val ) ;
2015-10-29 03:11:24 -03:00
}
template < typename InputIterator >
2018-09-19 03:02:12 -03:00
prevector ( InputIterator first , InputIterator last ) {
2015-10-29 03:11:24 -03:00
size_type n = last - first ;
change_capacity ( n ) ;
2017-12-22 07:04:30 -03:00
_size + = n ;
fill ( item_ptr ( 0 ) , first , last ) ;
2015-10-29 03:11:24 -03:00
}
2018-09-19 03:02:12 -03:00
prevector ( const prevector < N , T , Size , Diff > & other ) {
2017-12-22 07:04:30 -03:00
size_type n = other . size ( ) ;
change_capacity ( n ) ;
_size + = n ;
fill ( item_ptr ( 0 ) , other . begin ( ) , other . end ( ) ) ;
2015-10-29 03:11:24 -03:00
}
2018-09-19 03:02:12 -03:00
prevector ( prevector < N , T , Size , Diff > & & other ) {
2016-12-14 00:36:46 -03:00
swap ( other ) ;
}
2015-10-29 03:11:24 -03:00
prevector & operator = ( const prevector < N , T , Size , Diff > & other ) {
if ( & other = = this ) {
return * this ;
}
2017-12-22 07:04:30 -03:00
assign ( other . begin ( ) , other . end ( ) ) ;
2015-10-29 03:11:24 -03:00
return * this ;
}
2016-12-14 00:36:46 -03:00
prevector & operator = ( prevector < N , T , Size , Diff > & & other ) {
swap ( other ) ;
return * this ;
}
2015-10-29 03:11:24 -03:00
size_type size ( ) const {
return is_direct ( ) ? _size : _size - N - 1 ;
}
bool empty ( ) const {
return size ( ) = = 0 ;
}
iterator begin ( ) { return iterator ( item_ptr ( 0 ) ) ; }
const_iterator begin ( ) const { return const_iterator ( item_ptr ( 0 ) ) ; }
iterator end ( ) { return iterator ( item_ptr ( size ( ) ) ) ; }
const_iterator end ( ) const { return const_iterator ( item_ptr ( size ( ) ) ) ; }
reverse_iterator rbegin ( ) { return reverse_iterator ( item_ptr ( size ( ) - 1 ) ) ; }
const_reverse_iterator rbegin ( ) const { return const_reverse_iterator ( item_ptr ( size ( ) - 1 ) ) ; }
reverse_iterator rend ( ) { return reverse_iterator ( item_ptr ( - 1 ) ) ; }
const_reverse_iterator rend ( ) const { return const_reverse_iterator ( item_ptr ( - 1 ) ) ; }
size_t capacity ( ) const {
if ( is_direct ( ) ) {
return N ;
} else {
2018-12-31 07:50:06 -03:00
return _union . indirect_contents . capacity ;
2015-10-29 03:11:24 -03:00
}
}
T & operator [ ] ( size_type pos ) {
return * item_ptr ( pos ) ;
}
const T & operator [ ] ( size_type pos ) const {
return * item_ptr ( pos ) ;
}
void resize ( size_type new_size ) {
2017-12-22 07:04:30 -03:00
size_type cur_size = size ( ) ;
2018-02-27 02:39:22 -03:00
if ( cur_size = = new_size ) {
return ;
}
2017-12-22 07:04:30 -03:00
if ( cur_size > new_size ) {
2016-04-13 14:09:16 -03:00
erase ( item_ptr ( new_size ) , end ( ) ) ;
2018-02-27 02:39:22 -03:00
return ;
2015-10-29 03:11:24 -03:00
}
if ( new_size > capacity ( ) ) {
change_capacity ( new_size ) ;
}
2018-02-27 02:39:22 -03:00
ptrdiff_t increase = new_size - cur_size ;
fill ( item_ptr ( cur_size ) , increase ) ;
_size + = increase ;
2015-10-29 03:11:24 -03:00
}
void reserve ( size_type new_capacity ) {
if ( new_capacity > capacity ( ) ) {
change_capacity ( new_capacity ) ;
}
}
void shrink_to_fit ( ) {
change_capacity ( size ( ) ) ;
}
void clear ( ) {
resize ( 0 ) ;
}
iterator insert ( iterator pos , const T & value ) {
size_type p = pos - begin ( ) ;
size_type new_size = size ( ) + 1 ;
if ( capacity ( ) < new_size ) {
change_capacity ( new_size + ( new_size > > 1 ) ) ;
}
2017-12-22 07:04:30 -03:00
T * ptr = item_ptr ( p ) ;
memmove ( ptr + 1 , ptr , ( size ( ) - p ) * sizeof ( T ) ) ;
2015-10-29 03:11:24 -03:00
_size + + ;
2017-12-22 07:04:30 -03:00
new ( static_cast < void * > ( ptr ) ) T ( value ) ;
return iterator ( ptr ) ;
2015-10-29 03:11:24 -03:00
}
void insert ( iterator pos , size_type count , const T & value ) {
size_type p = pos - begin ( ) ;
size_type new_size = size ( ) + count ;
if ( capacity ( ) < new_size ) {
change_capacity ( new_size + ( new_size > > 1 ) ) ;
}
2017-12-22 07:04:30 -03:00
T * ptr = item_ptr ( p ) ;
memmove ( ptr + count , ptr , ( size ( ) - p ) * sizeof ( T ) ) ;
2015-10-29 03:11:24 -03:00
_size + = count ;
2017-12-22 07:04:30 -03:00
fill ( item_ptr ( p ) , count , value ) ;
2015-10-29 03:11:24 -03:00
}
template < typename InputIterator >
void insert ( iterator pos , InputIterator first , InputIterator last ) {
size_type p = pos - begin ( ) ;
difference_type count = last - first ;
size_type new_size = size ( ) + count ;
if ( capacity ( ) < new_size ) {
change_capacity ( new_size + ( new_size > > 1 ) ) ;
}
2017-12-22 07:04:30 -03:00
T * ptr = item_ptr ( p ) ;
memmove ( ptr + count , ptr , ( size ( ) - p ) * sizeof ( T ) ) ;
2015-10-29 03:11:24 -03:00
_size + = count ;
2017-12-22 07:04:30 -03:00
fill ( ptr , first , last ) ;
2015-10-29 03:11:24 -03:00
}
2018-02-01 07:34:50 -03:00
inline void resize_uninitialized ( size_type new_size ) {
// resize_uninitialized changes the size of the prevector but does not initialize it.
// If size < new_size, the added elements must be initialized explicitly.
if ( capacity ( ) < new_size ) {
change_capacity ( new_size ) ;
_size + = new_size - size ( ) ;
return ;
}
if ( new_size < size ( ) ) {
erase ( item_ptr ( new_size ) , end ( ) ) ;
} else {
_size + = new_size - size ( ) ;
}
}
2015-10-29 03:11:24 -03:00
iterator erase ( iterator pos ) {
2016-04-13 14:09:16 -03:00
return erase ( pos , pos + 1 ) ;
2015-10-29 03:11:24 -03:00
}
iterator erase ( iterator first , iterator last ) {
2017-06-05 15:17:40 -04:00
// Erase is not allowed to the change the object's capacity. That means
// that when starting with an indirectly allocated prevector with
// size and capacity > N, the result may be a still indirectly allocated
// prevector with size <= N and capacity > N. A shrink_to_fit() call is
// necessary to switch to the (more efficient) directly allocated
// representation (with capacity N and size <= N).
2015-10-29 03:11:24 -03:00
iterator p = first ;
char * endp = ( char * ) & ( * end ( ) ) ;
2022-04-24 06:55:57 -04:00
_size - = last - p ;
2015-10-29 03:11:24 -03:00
memmove ( & ( * first ) , & ( * last ) , endp - ( ( char * ) ( & ( * last ) ) ) ) ;
return first ;
}
2020-02-16 00:09:09 -03:00
template < typename . . . Args >
void emplace_back ( Args & & . . . args ) {
2015-10-29 03:11:24 -03:00
size_type new_size = size ( ) + 1 ;
if ( capacity ( ) < new_size ) {
change_capacity ( new_size + ( new_size > > 1 ) ) ;
}
2020-02-16 00:09:09 -03:00
new ( item_ptr ( size ( ) ) ) T ( std : : forward < Args > ( args ) . . . ) ;
2015-10-29 03:11:24 -03:00
_size + + ;
}
2020-02-16 00:09:09 -03:00
void push_back ( const T & value ) {
emplace_back ( value ) ;
}
2015-10-29 03:11:24 -03:00
void pop_back ( ) {
2016-04-13 14:09:16 -03:00
erase ( end ( ) - 1 , end ( ) ) ;
2015-10-29 03:11:24 -03:00
}
T & front ( ) {
return * item_ptr ( 0 ) ;
}
const T & front ( ) const {
return * item_ptr ( 0 ) ;
}
T & back ( ) {
return * item_ptr ( size ( ) - 1 ) ;
}
const T & back ( ) const {
return * item_ptr ( size ( ) - 1 ) ;
}
2022-04-28 14:20:30 -04:00
void swap ( prevector < N , T , Size , Diff > & other ) noexcept
{
2016-04-14 13:26:32 -03:00
std : : swap ( _union , other . _union ) ;
2015-10-29 03:11:24 -03:00
std : : swap ( _size , other . _size ) ;
}
~ prevector ( ) {
if ( ! is_direct ( ) ) {
2018-12-31 07:50:06 -03:00
free ( _union . indirect_contents . indirect ) ;
_union . indirect_contents . indirect = nullptr ;
2015-10-29 03:11:24 -03:00
}
}
bool operator = = ( const prevector < N , T , Size , Diff > & other ) const {
if ( other . size ( ) ! = size ( ) ) {
return false ;
}
const_iterator b1 = begin ( ) ;
const_iterator b2 = other . begin ( ) ;
const_iterator e1 = end ( ) ;
while ( b1 ! = e1 ) {
if ( ( * b1 ) ! = ( * b2 ) ) {
return false ;
}
+ + b1 ;
+ + b2 ;
}
return true ;
}
bool operator ! = ( const prevector < N , T , Size , Diff > & other ) const {
return ! ( * this = = other ) ;
}
bool operator < ( const prevector < N , T , Size , Diff > & other ) const {
if ( size ( ) < other . size ( ) ) {
return true ;
}
if ( size ( ) > other . size ( ) ) {
return false ;
}
const_iterator b1 = begin ( ) ;
const_iterator b2 = other . begin ( ) ;
const_iterator e1 = end ( ) ;
while ( b1 ! = e1 ) {
if ( ( * b1 ) < ( * b2 ) ) {
return true ;
}
if ( ( * b2 ) < ( * b1 ) ) {
return false ;
}
+ + b1 ;
+ + b2 ;
}
return false ;
}
size_t allocated_memory ( ) const {
if ( is_direct ( ) ) {
return 0 ;
} else {
2018-12-31 07:50:06 -03:00
return ( ( size_t ) ( sizeof ( T ) ) ) * _union . indirect_contents . capacity ;
2015-10-29 03:11:24 -03:00
}
}
2016-09-30 12:19:51 -03:00
2016-09-30 12:21:12 -03:00
value_type * data ( ) {
2016-09-30 12:19:51 -03:00
return item_ptr ( 0 ) ;
}
const value_type * data ( ) const {
return item_ptr ( 0 ) ;
}
2015-10-29 03:11:24 -03:00
} ;
2017-08-25 20:56:53 -03:00
# endif // BITCOIN_PREVECTOR_H