mirror of
https://github.com/pocketpy/pocketpy
synced 2025-10-24 05:20:17 +00:00
1357 lines
41 KiB
C++
1357 lines
41 KiB
C++
/***************************************************************************
|
|
* Copyright (c) Johan Mabille, Sylvain Corlay and Wolf Vollprecht *
|
|
* Copyright (c) QuantStack *
|
|
* *
|
|
* Distributed under the terms of the BSD 3-Clause License. *
|
|
* *
|
|
* The full license is in the file LICENSE, distributed with this software. *
|
|
****************************************************************************/
|
|
|
|
#ifndef XDYNAMIC_BITSET_HPP
|
|
#define XDYNAMIC_BITSET_HPP
|
|
|
|
#include <climits>
|
|
#include <type_traits>
|
|
#include <vector>
|
|
#include <initializer_list>
|
|
#include <iterator>
|
|
#include <memory>
|
|
#include <algorithm>
|
|
|
|
#include "xclosure.hpp"
|
|
#include "xspan.hpp"
|
|
#include "xiterator_base.hpp"
|
|
#include "xtype_traits.hpp"
|
|
|
|
namespace xtl
|
|
{
|
|
template <class B, bool is_const>
|
|
class xbitset_reference;
|
|
|
|
template <class B, bool is_const>
|
|
class xbitset_iterator;
|
|
|
|
/******************
|
|
* xdyamic_bitset *
|
|
******************/
|
|
|
|
template <class B>
|
|
class xdynamic_bitset_base;
|
|
|
|
template <class B, class A>
|
|
class xdynamic_bitset;
|
|
|
|
template <class X>
|
|
class xdynamic_bitset_view;
|
|
|
|
template <class X>
|
|
struct xdynamic_bitset_traits;
|
|
|
|
template <class B, class A>
|
|
struct xdynamic_bitset_traits<xdynamic_bitset<B, A>>
|
|
{
|
|
using storage_type = std::vector<B, A>;
|
|
using block_type = typename storage_type::value_type;
|
|
};
|
|
|
|
template <class X>
|
|
struct xdynamic_bitset_traits<xdynamic_bitset_view<X>>
|
|
{
|
|
using storage_type = xtl::span<X>;
|
|
using block_type = typename storage_type::value_type;
|
|
};
|
|
|
|
template <class X>
|
|
struct container_internals;
|
|
|
|
template <class X>
|
|
struct container_internals<xtl::span<X>>
|
|
{
|
|
using value_type = typename xtl::span<X>::value_type;
|
|
static_assert(xtl::is_scalar<value_type>::value, "");
|
|
using allocator_type = std::allocator<value_type>;
|
|
using size_type = std::size_t;
|
|
using difference_type = typename xtl::span<X>::difference_type;
|
|
};
|
|
|
|
template <class X, class A>
|
|
struct container_internals<std::vector<X, A>>
|
|
{
|
|
using value_type = X;
|
|
static_assert(xtl::is_scalar<value_type>::value, "");
|
|
using allocator_type = A;
|
|
using size_type = typename std::vector<X>::size_type;
|
|
using difference_type = typename std::vector<X>::difference_type;
|
|
};
|
|
|
|
template <class B>
|
|
class xdynamic_bitset_base
|
|
{
|
|
public:
|
|
|
|
using self_type = xdynamic_bitset_base<B>;
|
|
using derived_class = B;
|
|
|
|
using storage_type = typename xdynamic_bitset_traits<B>::storage_type;
|
|
using block_type = typename xdynamic_bitset_traits<B>::block_type;
|
|
using temporary_type = xdynamic_bitset<block_type, std::allocator<block_type>>;
|
|
|
|
using allocator_type = typename container_internals<storage_type>::allocator_type;
|
|
using value_type = bool;
|
|
using reference = xbitset_reference<derived_class, false>;
|
|
using const_reference = xbitset_reference<derived_class, true>;
|
|
|
|
using pointer = typename reference::pointer;
|
|
using const_pointer = typename const_reference::pointer;
|
|
using size_type = typename container_internals<storage_type>::size_type;
|
|
using difference_type = typename storage_type::difference_type;
|
|
using iterator = xbitset_iterator<derived_class, false>;
|
|
using const_iterator = xbitset_iterator<derived_class, true>;
|
|
using reverse_iterator = std::reverse_iterator<iterator>;
|
|
using const_reverse_iterator = std::reverse_iterator<const_iterator>;
|
|
|
|
using const_block_iterator = typename storage_type::const_iterator;
|
|
|
|
bool empty() const noexcept;
|
|
size_type size() const noexcept;
|
|
|
|
void swap(self_type& rhs);
|
|
|
|
reference at(size_type i);
|
|
const_reference at(size_type i) const;
|
|
|
|
reference operator[](size_type i);
|
|
const_reference operator[](size_type i) const;
|
|
|
|
reference front();
|
|
const_reference front() const;
|
|
|
|
reference back();
|
|
const_reference back() const;
|
|
|
|
iterator begin() noexcept;
|
|
iterator end() noexcept;
|
|
|
|
const_iterator begin() const noexcept;
|
|
const_iterator end() const noexcept;
|
|
|
|
const_iterator cbegin() const noexcept;
|
|
const_iterator cend() const noexcept;
|
|
|
|
reverse_iterator rbegin() noexcept;
|
|
reverse_iterator rend() noexcept;
|
|
|
|
const_reverse_iterator rbegin() const noexcept;
|
|
const_reverse_iterator rend() const noexcept;
|
|
|
|
const_reverse_iterator crbegin() const noexcept;
|
|
const_reverse_iterator crend() const noexcept;
|
|
|
|
const_block_iterator block_begin() const noexcept;
|
|
const_block_iterator block_end() const noexcept;
|
|
|
|
template <class R>
|
|
self_type& operator&=(const xdynamic_bitset_base<R>& rhs);
|
|
template <class R>
|
|
self_type& operator|=(const xdynamic_bitset_base<R>& rhs);
|
|
template <class R>
|
|
self_type& operator^=(const xdynamic_bitset_base<R>& rhs);
|
|
|
|
temporary_type operator<<(size_type pos);
|
|
self_type& operator<<=(size_type pos);
|
|
temporary_type operator>>(size_type pos);
|
|
self_type& operator>>=(size_type pos);
|
|
|
|
self_type& set();
|
|
self_type& set(size_type pos, value_type value = true);
|
|
|
|
self_type& reset();
|
|
self_type& reset(size_type pos);
|
|
|
|
self_type& flip();
|
|
self_type& flip(size_type pos);
|
|
|
|
bool all() const noexcept;
|
|
bool any() const noexcept;
|
|
bool none() const noexcept;
|
|
size_type count() const noexcept;
|
|
|
|
size_type block_count() const noexcept;
|
|
block_type* data() noexcept;
|
|
const block_type* data() const noexcept;
|
|
|
|
template <class Y>
|
|
bool operator==(const xdynamic_bitset_base<Y>& rhs) const noexcept;
|
|
template <class Y>
|
|
bool operator!=(const xdynamic_bitset_base<Y>& rhs) const noexcept;
|
|
|
|
derived_class& derived_cast();
|
|
const derived_class& derived_cast() const;
|
|
|
|
protected:
|
|
|
|
xdynamic_bitset_base(const storage_type& buffer, std::size_t size);
|
|
|
|
~xdynamic_bitset_base() = default;
|
|
xdynamic_bitset_base(const xdynamic_bitset_base& rhs) = default;
|
|
xdynamic_bitset_base(xdynamic_bitset_base&& rhs) = default;
|
|
xdynamic_bitset_base& operator=(const xdynamic_bitset_base& rhs) = default;
|
|
xdynamic_bitset_base& operator=(xdynamic_bitset_base&& rhs) = default;
|
|
|
|
size_type m_size;
|
|
storage_type m_buffer;
|
|
|
|
static constexpr std::size_t s_bits_per_block = CHAR_BIT * sizeof(block_type);
|
|
|
|
size_type compute_block_count(size_type bits_count) const noexcept;
|
|
size_type block_index(size_type pos) const noexcept;
|
|
size_type bit_index(size_type pos) const noexcept;
|
|
block_type bit_mask(size_type pos) const noexcept;
|
|
size_type count_extra_bits() const noexcept;
|
|
void zero_unused_bits();
|
|
private:
|
|
|
|
// Make views and buffers friends
|
|
template<typename BB>
|
|
friend class xdynamic_bitset_base;
|
|
};
|
|
|
|
// NOTE this view ZEROS out remaining bits!
|
|
template <class X>
|
|
class xdynamic_bitset_view
|
|
: public xdynamic_bitset_base<xdynamic_bitset_view<X>>
|
|
{
|
|
public:
|
|
|
|
using base_class = xdynamic_bitset_base<xdynamic_bitset_view<X>>;
|
|
using storage_type = typename base_class::storage_type;
|
|
using block_type = typename base_class::block_type;
|
|
|
|
xdynamic_bitset_view(block_type* ptr, std::size_t size);
|
|
|
|
xdynamic_bitset_view() = default;
|
|
~xdynamic_bitset_view() = default;
|
|
xdynamic_bitset_view(const xdynamic_bitset_view& rhs) = default;
|
|
xdynamic_bitset_view(xdynamic_bitset_view&& rhs) = default;
|
|
xdynamic_bitset_view& operator=(const xdynamic_bitset_view& rhs) = default;
|
|
xdynamic_bitset_view& operator=(xdynamic_bitset_view&& rhs) = default;
|
|
|
|
void resize(std::size_t sz);
|
|
};
|
|
|
|
namespace detail_bitset
|
|
{
|
|
template <class T>
|
|
constexpr T integer_ceil(T n, T div)
|
|
{
|
|
return (n + div - T(1)) / div;
|
|
}
|
|
}
|
|
|
|
template <class X>
|
|
inline xdynamic_bitset_view<X>::xdynamic_bitset_view(block_type* ptr, std::size_t size)
|
|
: base_class(storage_type(ptr, detail_bitset::integer_ceil(size, base_class::s_bits_per_block)), size)
|
|
{
|
|
base_class::zero_unused_bits();
|
|
}
|
|
|
|
template <class X>
|
|
inline void xdynamic_bitset_view<X>::resize(std::size_t sz)
|
|
{
|
|
if (sz != this->m_size) {
|
|
#if defined(XTL_NO_EXCEPTIONS)
|
|
std::fprintf(stderr, "cannot resize bitset_view\n");
|
|
std::terminate();
|
|
#else
|
|
throw std::runtime_error("cannot resize bitset_view");
|
|
#endif
|
|
}
|
|
}
|
|
|
|
template <class B, class A = std::allocator<B>>
|
|
class xdynamic_bitset;
|
|
|
|
template <class B>
|
|
auto operator~(const xdynamic_bitset_base<B>& lhs);
|
|
|
|
template <class L, class R>
|
|
auto operator&(const xdynamic_bitset_base<L>& lhs, const xdynamic_bitset_base<R>& rhs);
|
|
|
|
template <class L, class R>
|
|
auto operator|(const xdynamic_bitset_base<L>& lhs, const xdynamic_bitset_base<R>& rhs);
|
|
|
|
template <class L, class R>
|
|
auto operator^(const xdynamic_bitset_base<L>& lhs, const xdynamic_bitset_base<R>& rhs);
|
|
|
|
template <class B>
|
|
void swap(const xdynamic_bitset_base<B>& lhs, const xdynamic_bitset_base<B>& rhs);
|
|
|
|
/*********************
|
|
* xbitset_reference *
|
|
*********************/
|
|
|
|
template <class B, bool is_const>
|
|
class xbitset_reference
|
|
{
|
|
public:
|
|
|
|
using self_type = xbitset_reference<B, is_const>;
|
|
using pointer = std::conditional_t<is_const,
|
|
const xclosure_pointer<const self_type>,
|
|
xclosure_pointer<self_type>>;
|
|
|
|
operator bool() const noexcept;
|
|
|
|
xbitset_reference(const self_type&) = default;
|
|
xbitset_reference(self_type&&) = default;
|
|
|
|
self_type& operator=(const self_type&) noexcept;
|
|
self_type& operator=(self_type&&) noexcept;
|
|
self_type& operator=(bool) noexcept;
|
|
|
|
bool operator~() const noexcept;
|
|
|
|
self_type& operator&=(bool) noexcept;
|
|
self_type& operator|=(bool) noexcept;
|
|
self_type& operator^=(bool) noexcept;
|
|
self_type& flip() noexcept;
|
|
|
|
pointer operator&() noexcept;
|
|
|
|
private:
|
|
|
|
using block_type = typename xdynamic_bitset_traits<B>::block_type;
|
|
using closure_type = std::conditional_t<is_const, const block_type&, block_type&>;
|
|
|
|
xbitset_reference(closure_type block, block_type pos);
|
|
|
|
void assign(bool) noexcept;
|
|
void set() noexcept;
|
|
void reset() noexcept;
|
|
|
|
closure_type m_block;
|
|
const block_type m_mask;
|
|
|
|
template <class BO, bool is_const_other>
|
|
friend class xbitset_reference;
|
|
|
|
friend class xdynamic_bitset_base<B>;
|
|
};
|
|
|
|
/********************
|
|
* xbitset_iterator *
|
|
********************/
|
|
|
|
template <class B, bool is_const>
|
|
class xbitset_iterator : public xrandom_access_iterator_base<xbitset_iterator<B, is_const>,
|
|
typename xdynamic_bitset_base<B>::value_type,
|
|
typename xdynamic_bitset_base<B>::difference_type,
|
|
std::conditional_t<is_const,
|
|
typename xdynamic_bitset_base<B>::const_pointer,
|
|
typename xdynamic_bitset_base<B>::pointer>,
|
|
std::conditional_t<is_const,
|
|
typename xdynamic_bitset_base<B>::const_reference,
|
|
typename xdynamic_bitset_base<B>::reference>>
|
|
{
|
|
public:
|
|
|
|
using self_type = xbitset_iterator<B, is_const>;
|
|
using container_type = xdynamic_bitset_base<B>;
|
|
using value_type = typename container_type::value_type;
|
|
using reference = std::conditional_t<is_const,
|
|
typename container_type::const_reference,
|
|
typename container_type::reference>;
|
|
using pointer = std::conditional_t<is_const,
|
|
typename container_type::const_pointer,
|
|
typename container_type::pointer>;
|
|
using size_type = typename container_type::size_type;
|
|
using difference_type = typename container_type::difference_type;
|
|
using base_type = xrandom_access_iterator_base<self_type, value_type, difference_type, pointer, reference>;
|
|
|
|
using container_reference = std::conditional_t<is_const, const container_type&, container_type&>;
|
|
using container_pointer = std::conditional_t<is_const, const container_type*, container_type*>;
|
|
|
|
xbitset_iterator() noexcept;
|
|
xbitset_iterator(container_reference c, size_type index) noexcept;
|
|
|
|
self_type& operator++();
|
|
self_type& operator--();
|
|
|
|
self_type& operator+=(difference_type n);
|
|
self_type& operator-=(difference_type n);
|
|
|
|
difference_type operator-(const self_type& rhs) const;
|
|
|
|
reference operator*() const;
|
|
pointer operator->() const;
|
|
|
|
bool operator==(const self_type& rhs) const;
|
|
bool operator<(const self_type& rhs) const;
|
|
|
|
private:
|
|
|
|
container_pointer p_container;
|
|
size_type m_index;
|
|
};
|
|
|
|
template <class B, class Allocator>
|
|
class xdynamic_bitset
|
|
: public xdynamic_bitset_base<xdynamic_bitset<B, Allocator>>
|
|
{
|
|
public:
|
|
|
|
using allocator_type = Allocator;
|
|
using storage_type = std::vector<B, Allocator>;
|
|
|
|
using base_type = xdynamic_bitset_base<xdynamic_bitset<B, Allocator>>;
|
|
using self_type = xdynamic_bitset<B, Allocator>;
|
|
using block_type = B;
|
|
|
|
using reference = typename base_type::reference;
|
|
using const_reference = typename base_type::const_reference;
|
|
|
|
using pointer = typename reference::pointer;
|
|
using const_pointer = typename const_reference::pointer;
|
|
using size_type = typename storage_type::size_type;
|
|
using difference_type = typename storage_type::difference_type;
|
|
using iterator = xbitset_iterator<self_type, false>;
|
|
using const_iterator = xbitset_iterator<self_type, true>;
|
|
using reverse_iterator = std::reverse_iterator<iterator>;
|
|
using const_reverse_iterator = std::reverse_iterator<const_iterator>;
|
|
|
|
using base_type::base_type;
|
|
using base_type::begin;
|
|
using base_type::cbegin;
|
|
using base_type::end;
|
|
using base_type::cend;
|
|
using base_type::rbegin;
|
|
using base_type::rend;
|
|
using base_type::size;
|
|
|
|
xdynamic_bitset();
|
|
|
|
explicit xdynamic_bitset(const allocator_type& allocator);
|
|
|
|
xdynamic_bitset(size_type count, bool b, const allocator_type& alloc = allocator_type());
|
|
explicit xdynamic_bitset(size_type count, const allocator_type& alloc = allocator_type());
|
|
xdynamic_bitset(std::initializer_list<bool> init, const allocator_type& alloc = allocator_type());
|
|
|
|
template <class BlockInputIt>
|
|
xdynamic_bitset(BlockInputIt first, BlockInputIt last, const allocator_type& alloc = allocator_type());
|
|
|
|
xdynamic_bitset(const xdynamic_bitset& rhs);
|
|
|
|
// Allow creation from views for e.g. temporary creation
|
|
template <class Y>
|
|
xdynamic_bitset(const xdynamic_bitset_base<Y>& rhs);
|
|
|
|
~xdynamic_bitset() = default;
|
|
xdynamic_bitset(xdynamic_bitset&& rhs) = default;
|
|
xdynamic_bitset& operator=(const xdynamic_bitset& rhs) = default;
|
|
xdynamic_bitset& operator=(xdynamic_bitset&& rhs) = default;
|
|
|
|
void assign(size_type count, bool b);
|
|
template <class BlockInputIt>
|
|
void assign(BlockInputIt first, BlockInputIt last);
|
|
void assign(std::initializer_list<bool> init);
|
|
|
|
size_type max_size() const noexcept;
|
|
void reserve(size_type new_cap);
|
|
size_type capacity() const noexcept;
|
|
|
|
allocator_type get_allocator() const;
|
|
|
|
void resize(size_type size, bool b = false);
|
|
void clear() noexcept;
|
|
void push_back(bool b);
|
|
void pop_back();
|
|
};
|
|
|
|
/**********************************
|
|
* xdynamic_bitset implementation *
|
|
**********************************/
|
|
|
|
template <class B, class A>
|
|
inline xdynamic_bitset<B, A>::xdynamic_bitset()
|
|
: base_type(storage_type(), size_type(0))
|
|
{
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline xdynamic_bitset<B, A>::xdynamic_bitset(const allocator_type& allocator)
|
|
: base_type(storage_type(allocator), size_type(0))
|
|
{
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline xdynamic_bitset<B, A>::xdynamic_bitset(size_type count, bool b, const allocator_type& alloc)
|
|
: base_type(storage_type(this->compute_block_count(count), b ? ~block_type(0) : block_type(0), alloc), count)
|
|
{
|
|
this->zero_unused_bits();
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline xdynamic_bitset<B, A>::xdynamic_bitset(size_type count, const allocator_type& alloc)
|
|
: base_type(storage_type(this->compute_block_count(count), block_type(0), alloc), count)
|
|
{
|
|
}
|
|
|
|
template <class B, class A>
|
|
template <class BlockInputIt>
|
|
inline xdynamic_bitset<B, A>::xdynamic_bitset(BlockInputIt first, BlockInputIt last, const allocator_type& alloc)
|
|
: base_type(storage_type(first, last, alloc), size_type(std::distance(first, last)) * base_type::s_bits_per_block)
|
|
{
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline xdynamic_bitset<B, A>::xdynamic_bitset(std::initializer_list<bool> init, const allocator_type& alloc)
|
|
: xdynamic_bitset(init.size(), alloc)
|
|
{
|
|
std::copy(init.begin(), init.end(), begin());
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline xdynamic_bitset<B, A>::xdynamic_bitset(const xdynamic_bitset& rhs)
|
|
: base_type(storage_type(rhs.block_begin(), rhs.block_end()), rhs.size())
|
|
{
|
|
}
|
|
|
|
template <class B, class A>
|
|
template <class Y>
|
|
inline xdynamic_bitset<B, A>::xdynamic_bitset(const xdynamic_bitset_base<Y>& rhs)
|
|
: base_type(storage_type(rhs.block_begin(), rhs.block_end()), rhs.size())
|
|
{
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline void xdynamic_bitset<B, A>::assign(size_type count, bool b)
|
|
{
|
|
resize(count);
|
|
b ? this->set() : this->reset();
|
|
}
|
|
|
|
template <class B, class A>
|
|
template <class BlockInputIt>
|
|
inline void xdynamic_bitset<B, A>::assign(BlockInputIt first, BlockInputIt last)
|
|
{
|
|
resize(size_type(std::distance(first, last)) * base_type::s_bits_per_block);
|
|
std::copy(first, last, this->m_buffer.begin());
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline void xdynamic_bitset<B, A>::assign(std::initializer_list<bool> init)
|
|
{
|
|
resize(init.size());
|
|
std::copy(init.begin(), init.end(), begin());
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline auto xdynamic_bitset<B, A>::get_allocator() const -> allocator_type
|
|
{
|
|
return base_type::m_buffer.get_allocator();
|
|
}
|
|
|
|
template <class B>
|
|
inline bool xdynamic_bitset_base<B>::empty() const noexcept
|
|
{
|
|
return m_size == 0;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::size() const noexcept -> size_type
|
|
{
|
|
return m_size;
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline auto xdynamic_bitset<B, A>::max_size() const noexcept -> size_type
|
|
{
|
|
return base_type::m_buffer.max_size() * base_type::s_bits_per_block;
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline void xdynamic_bitset<B, A>::reserve(size_type new_cap)
|
|
{
|
|
base_type::m_buffer.reserve(this->compute_block_count(new_cap));
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline auto xdynamic_bitset<B, A>::capacity() const noexcept -> size_type
|
|
{
|
|
return base_type::m_buffer.capacity() * base_type::s_bits_per_block;
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline void xdynamic_bitset<B, A>::resize(size_type asize, bool b)
|
|
{
|
|
size_type old_block_count = base_type::block_count();
|
|
size_type new_block_count = base_type::compute_block_count(asize);
|
|
block_type value = b ? ~block_type(0) : block_type(0);
|
|
|
|
if (new_block_count != old_block_count)
|
|
{
|
|
base_type::m_buffer.resize(new_block_count, value);
|
|
}
|
|
|
|
if (b && asize > base_type::m_size)
|
|
{
|
|
size_type extra_bits = base_type::count_extra_bits();
|
|
if (extra_bits > 0)
|
|
{
|
|
base_type::m_buffer[old_block_count - 1] |= (value << extra_bits);
|
|
}
|
|
}
|
|
|
|
base_type::m_size = asize;
|
|
base_type::zero_unused_bits();
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline void xdynamic_bitset<B, A>::clear() noexcept
|
|
{
|
|
base_type::m_buffer.clear();
|
|
base_type::m_size = size_type(0);
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline void xdynamic_bitset<B, A>::push_back(bool b)
|
|
{
|
|
size_type s = size();
|
|
resize(s + 1);
|
|
this->set(s, b);
|
|
}
|
|
|
|
template <class B, class A>
|
|
inline void xdynamic_bitset<B, A>::pop_back()
|
|
{
|
|
size_type old_block_count = base_type::m_buffer.size();
|
|
size_type new_block_count = base_type::compute_block_count(base_type::m_size - 1);
|
|
|
|
if (new_block_count != old_block_count)
|
|
{
|
|
base_type::m_buffer.pop_back();
|
|
}
|
|
|
|
--base_type::m_size;
|
|
base_type::zero_unused_bits();
|
|
}
|
|
|
|
template <class B>
|
|
inline void xdynamic_bitset_base<B>::swap(self_type& rhs)
|
|
{
|
|
using std::swap;
|
|
swap(m_buffer, rhs.m_buffer);
|
|
swap(m_size, rhs.m_size);
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::at(size_type i) -> reference
|
|
{
|
|
// TODO add real check, remove m_buffer.at ...
|
|
return reference(m_buffer.at(block_index(i)), bit_index(i));
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::at(size_type i) const -> const_reference
|
|
{
|
|
// TODO add real check, remove m_buffer.at ...
|
|
return const_reference(m_buffer.at(block_index(i)), bit_index(i));
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::operator[](size_type i) -> reference
|
|
{
|
|
return reference(m_buffer[block_index(i)], bit_index(i));
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::operator[](size_type i) const -> const_reference
|
|
{
|
|
return const_reference(m_buffer[block_index(i)], bit_index(i));
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::front() -> reference
|
|
{
|
|
return (*this)[0];
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::front() const -> const_reference
|
|
{
|
|
return (*this)[0];
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::back() -> reference
|
|
{
|
|
return (*this)[m_size - 1];
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::back() const -> const_reference
|
|
{
|
|
return (*this)[m_size - 1];
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::begin() noexcept -> iterator
|
|
{
|
|
return iterator(*this, size_type(0));
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::end() noexcept -> iterator
|
|
{
|
|
return iterator(*this, size());
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::begin() const noexcept -> const_iterator
|
|
{
|
|
return cbegin();
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::end() const noexcept -> const_iterator
|
|
{
|
|
return cend();
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::cbegin() const noexcept -> const_iterator
|
|
{
|
|
return const_iterator(*this, size_type(0));
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::cend() const noexcept -> const_iterator
|
|
{
|
|
return const_iterator(*this, size());
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::rbegin() noexcept -> reverse_iterator
|
|
{
|
|
return reverse_iterator(end());
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::rend() noexcept -> reverse_iterator
|
|
{
|
|
return reverse_iterator(begin());
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::rbegin() const noexcept -> const_reverse_iterator
|
|
{
|
|
return crbegin();
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::rend() const noexcept -> const_reverse_iterator
|
|
{
|
|
return crend();
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::crbegin() const noexcept -> const_reverse_iterator
|
|
{
|
|
return const_reverse_iterator(cend());
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::crend() const noexcept -> const_reverse_iterator
|
|
{
|
|
return const_reverse_iterator(cbegin());
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::block_begin() const noexcept -> const_block_iterator
|
|
{
|
|
return m_buffer.begin();
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::block_end() const noexcept -> const_block_iterator
|
|
{
|
|
return m_buffer.end();
|
|
}
|
|
|
|
template <class B>
|
|
template <class R>
|
|
inline auto xdynamic_bitset_base<B>::operator&=(const xdynamic_bitset_base<R>& rhs) -> self_type&
|
|
{
|
|
size_type size = block_count();
|
|
for (size_type i = 0; i < size; ++i)
|
|
{
|
|
m_buffer[i] &= rhs.m_buffer[i];
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
template <class R>
|
|
inline auto xdynamic_bitset_base<B>::operator|=(const xdynamic_bitset_base<R>& rhs) -> self_type&
|
|
{
|
|
size_type size = block_count();
|
|
for (size_type i = 0; i < size; ++i)
|
|
{
|
|
m_buffer[i] |= rhs.m_buffer[i];
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
template <class R>
|
|
inline auto xdynamic_bitset_base<B>::operator^=(const xdynamic_bitset_base<R>& rhs) -> self_type&
|
|
{
|
|
size_type size = block_count();
|
|
for (size_type i = 0; i < size; ++i)
|
|
{
|
|
m_buffer[i] ^= rhs.m_buffer[i];
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::operator<<(size_type pos) -> temporary_type
|
|
{
|
|
temporary_type tmp(this->derived_cast());
|
|
tmp <<= pos;
|
|
return tmp;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::operator<<=(size_type pos) -> self_type&
|
|
{
|
|
if (pos >= m_size)
|
|
{
|
|
return reset();
|
|
}
|
|
|
|
if (pos > 0)
|
|
{
|
|
size_type last = block_count() - 1;
|
|
size_type div = pos / s_bits_per_block;
|
|
size_type r = bit_index(pos);
|
|
block_type* b = &m_buffer[0];
|
|
|
|
if (r != 0)
|
|
{
|
|
size_type rs = s_bits_per_block - r;
|
|
for (size_type i = last - div; i > 0; --i)
|
|
{
|
|
b[i + div] = (b[i] << r) | (b[i - 1] >> rs);
|
|
}
|
|
b[div] = b[0] << r;
|
|
}
|
|
else
|
|
{
|
|
for (size_type i = last - div; i > 0; --i)
|
|
{
|
|
b[i + div] = b[i];
|
|
}
|
|
b[div] = b[0];
|
|
}
|
|
|
|
std::fill_n(m_buffer.begin(), div, block_type(0));
|
|
zero_unused_bits();
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::operator>>(size_type pos) -> temporary_type
|
|
{
|
|
temporary_type tmp(this->derived_cast());
|
|
tmp >>= pos;
|
|
return tmp;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::operator>>=(size_type pos) -> self_type&
|
|
{
|
|
if (pos >= m_size)
|
|
{
|
|
return reset();
|
|
}
|
|
|
|
if (pos > 0)
|
|
{
|
|
size_type last = block_count() - 1;
|
|
size_type div = pos / s_bits_per_block;
|
|
size_type r = bit_index(pos);
|
|
block_type* b = &m_buffer[0];
|
|
|
|
if (r != 0)
|
|
{
|
|
size_type ls = s_bits_per_block - r;
|
|
for (size_type i = div; i < last; ++i)
|
|
{
|
|
b[i - div] = (b[i] >> r) | (b[i + 1] << ls);
|
|
}
|
|
b[last - div] = b[last] >> r;
|
|
}
|
|
else
|
|
{
|
|
for (size_type i = div; i <= last; ++i)
|
|
{
|
|
b[i - div] = b[i];
|
|
}
|
|
}
|
|
|
|
std::fill_n(m_buffer.begin() + static_cast<std::ptrdiff_t>(block_count() - div), div, block_type(0));
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::set() -> self_type&
|
|
{
|
|
std::fill(m_buffer.begin(), m_buffer.end(), ~block_type(0));
|
|
zero_unused_bits();
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::set(size_type pos, value_type value) -> self_type&
|
|
{
|
|
if (value)
|
|
{
|
|
m_buffer[block_index(pos)] |= bit_mask(pos);
|
|
}
|
|
else
|
|
{
|
|
reset(pos);
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::reset() -> self_type&
|
|
{
|
|
std::fill(m_buffer.begin(), m_buffer.end(), block_type(0));
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::reset(size_type pos) -> self_type&
|
|
{
|
|
m_buffer[block_index(pos)] &= ~bit_mask(pos);
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::flip() -> self_type&
|
|
{
|
|
size_type size = block_count();
|
|
for (size_type i = 0; i < size; ++i)
|
|
{
|
|
m_buffer[i] = ~m_buffer[i];
|
|
}
|
|
zero_unused_bits();
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::flip(size_type pos) -> self_type&
|
|
{
|
|
m_buffer[block_index(pos)] ^= bit_mask(pos);
|
|
return *this;
|
|
}
|
|
|
|
template <class B>
|
|
inline bool xdynamic_bitset_base<B>::all() const noexcept
|
|
{
|
|
if (empty())
|
|
return true;
|
|
|
|
size_type extra_bits = count_extra_bits();
|
|
constexpr block_type all_ones = ~block_type(0);
|
|
|
|
size_type size = extra_bits != 0 ? block_count() - 1 : block_count();
|
|
for (size_type i = 0; i < size; ++i)
|
|
{
|
|
if (m_buffer[i] != all_ones)
|
|
return false;
|
|
}
|
|
|
|
if (extra_bits != 0)
|
|
{
|
|
block_type mask = ~(~block_type(0) << extra_bits);
|
|
if (m_buffer.back() != mask)
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
template <class B>
|
|
inline bool xdynamic_bitset_base<B>::any() const noexcept
|
|
{
|
|
size_type size = block_count();
|
|
for (size_type i = 0; i < size; ++i)
|
|
{
|
|
if (m_buffer[i])
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
template <class B>
|
|
inline bool xdynamic_bitset_base<B>::none() const noexcept
|
|
{
|
|
return !any();
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::count() const noexcept -> size_type
|
|
{
|
|
static constexpr unsigned char table[] =
|
|
{
|
|
0, 1, 1, 2, 1, 2, 2, 3, 1, 2, 2, 3, 2, 3, 3, 4, 1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5,
|
|
1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
|
|
1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
|
|
2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
|
|
1, 2, 2, 3, 2, 3, 3, 4, 2, 3, 3, 4, 3, 4, 4, 5, 2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6,
|
|
2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
|
|
2, 3, 3, 4, 3, 4, 4, 5, 3, 4, 4, 5, 4, 5, 5, 6, 3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7,
|
|
3, 4, 4, 5, 4, 5, 5, 6, 4, 5, 5, 6, 5, 6, 6, 7, 4, 5, 5, 6, 5, 6, 6, 7, 5, 6, 6, 7, 6, 7, 7, 8
|
|
};
|
|
size_type res = 0;
|
|
const unsigned char* p = static_cast<const unsigned char*>(static_cast<const void*>(&m_buffer[0]));
|
|
size_type length = m_buffer.size() * sizeof(block_type);
|
|
for (size_type i = 0; i < length; ++i, ++p)
|
|
{
|
|
res += table[*p];
|
|
}
|
|
return res;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::block_count() const noexcept -> size_type
|
|
{
|
|
return m_buffer.size();
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::data() noexcept -> block_type*
|
|
{
|
|
return m_buffer.data();
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::data() const noexcept -> const block_type*
|
|
{
|
|
return m_buffer.data();
|
|
}
|
|
|
|
template <class B>
|
|
template <class Y>
|
|
inline bool xdynamic_bitset_base<B>::operator==(const xdynamic_bitset_base<Y>& rhs) const noexcept
|
|
{
|
|
bool is_equal = m_size == rhs.m_size;
|
|
if (!is_equal) { return false; }
|
|
|
|
// we know that block type of lhs & rhs is the same
|
|
auto n_blocks = block_count();
|
|
|
|
for (std::size_t i = 0; i < n_blocks; ++i)
|
|
{
|
|
if (m_buffer[i] != rhs.m_buffer[i])
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
template <class B>
|
|
template <class Y>
|
|
inline bool xdynamic_bitset_base<B>::operator!=(const xdynamic_bitset_base<Y>& rhs) const noexcept
|
|
{
|
|
return !(*this == rhs);
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::derived_cast() -> derived_class&
|
|
{
|
|
return *(reinterpret_cast<derived_class*>(this));
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::derived_cast() const -> const derived_class&
|
|
{
|
|
return *(reinterpret_cast<const derived_class*>(this));
|
|
}
|
|
|
|
template <class B>
|
|
inline xdynamic_bitset_base<B>::xdynamic_bitset_base(const storage_type& buffer, std::size_t size)
|
|
: m_size(size), m_buffer(buffer)
|
|
{
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::compute_block_count(size_type bits_count) const noexcept -> size_type
|
|
{
|
|
return bits_count / s_bits_per_block
|
|
+ static_cast<size_type>(bits_count % s_bits_per_block != 0);
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::block_index(size_type pos) const noexcept -> size_type
|
|
{
|
|
return pos / s_bits_per_block;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::bit_index(size_type pos) const noexcept -> size_type
|
|
{
|
|
return pos % s_bits_per_block;
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::bit_mask(size_type pos) const noexcept -> block_type
|
|
{
|
|
return block_type(1) << bit_index(pos);
|
|
}
|
|
|
|
template <class B>
|
|
inline auto xdynamic_bitset_base<B>::count_extra_bits() const noexcept -> size_type
|
|
{
|
|
return bit_index(size());
|
|
}
|
|
|
|
template <class B>
|
|
inline void xdynamic_bitset_base<B>::zero_unused_bits()
|
|
{
|
|
size_type extra_bits = count_extra_bits();
|
|
if (extra_bits != 0)
|
|
{
|
|
m_buffer.back() &= ~(~block_type(0) << extra_bits);
|
|
}
|
|
}
|
|
|
|
template <class B>
|
|
inline auto operator~(const xdynamic_bitset_base<B>& lhs)
|
|
{
|
|
using temporary_type = typename xdynamic_bitset_base<B>::temporary_type;
|
|
temporary_type res(lhs.derived_cast());
|
|
res.flip();
|
|
return res;
|
|
}
|
|
|
|
template <class L, class R>
|
|
inline auto operator&(const xdynamic_bitset_base<L>& lhs, const xdynamic_bitset_base<R>& rhs)
|
|
{
|
|
using temporary_type = typename xdynamic_bitset_base<L>::temporary_type;
|
|
temporary_type res(lhs.derived_cast());
|
|
res &= rhs;
|
|
return res;
|
|
}
|
|
|
|
template <class L, class R>
|
|
inline auto operator|(const xdynamic_bitset_base<L>& lhs, const xdynamic_bitset_base<R>& rhs)
|
|
{
|
|
using temporary_type = typename xdynamic_bitset_base<L>::temporary_type;
|
|
temporary_type res(lhs.derived_cast());
|
|
res |= rhs;
|
|
return res;
|
|
}
|
|
|
|
template <class L, class R>
|
|
inline auto operator^(const xdynamic_bitset_base<L>& lhs, const xdynamic_bitset_base<R>& rhs)
|
|
{
|
|
using temporary_type = typename xdynamic_bitset_base<L>::temporary_type;
|
|
temporary_type res(lhs.derived_cast());
|
|
res ^= rhs;
|
|
return res;
|
|
}
|
|
|
|
template <class B>
|
|
inline void swap(const xdynamic_bitset_base<B>& lhs, const xdynamic_bitset_base<B>& rhs)
|
|
{
|
|
return lhs.swap(rhs);
|
|
}
|
|
|
|
/************************************
|
|
* xbitset_reference implementation *
|
|
************************************/
|
|
|
|
template <class B, bool C>
|
|
inline xbitset_reference<B, C>::xbitset_reference(closure_type block, block_type pos)
|
|
: m_block(block), m_mask(block_type(1) << pos)
|
|
{
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline xbitset_reference<B, C>::operator bool() const noexcept
|
|
{
|
|
return (m_block & m_mask) != 0;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_reference<B, C>::operator=(const self_type& rhs) noexcept -> self_type&
|
|
{
|
|
assign(rhs);
|
|
return *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_reference<B, C>::operator=(self_type&& rhs) noexcept -> self_type&
|
|
{
|
|
assign(rhs);
|
|
return *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_reference<B, C>::operator=(bool rhs) noexcept -> self_type&
|
|
{
|
|
assign(rhs);
|
|
return *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline bool xbitset_reference<B, C>::operator~() const noexcept
|
|
{
|
|
return (m_block & m_mask) == 0;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_reference<B, C>::operator&=(bool rhs) noexcept -> self_type&
|
|
{
|
|
if (!rhs)
|
|
{
|
|
reset();
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_reference<B, C>::operator|=(bool rhs) noexcept -> self_type&
|
|
{
|
|
if (rhs)
|
|
{
|
|
set();
|
|
}
|
|
return *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_reference<B, C>::operator^=(bool rhs) noexcept -> self_type&
|
|
{
|
|
return rhs ? flip() : *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_reference<B, C>::flip() noexcept -> self_type&
|
|
{
|
|
m_block ^= m_mask;
|
|
return *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_reference<B, C>::operator&() noexcept -> pointer
|
|
{
|
|
return pointer(*this);
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline void xbitset_reference<B, C>::assign(bool rhs) noexcept
|
|
{
|
|
rhs ? set() : reset();
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline void xbitset_reference<B, C>::set() noexcept
|
|
{
|
|
m_block |= m_mask;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline void xbitset_reference<B, C>::reset() noexcept
|
|
{
|
|
m_block &= ~m_mask;
|
|
}
|
|
|
|
/***********************************
|
|
* xbitset_iterator implementation *
|
|
***********************************/
|
|
|
|
template <class B, bool C>
|
|
inline xbitset_iterator<B, C>::xbitset_iterator() noexcept
|
|
: p_container(nullptr), m_index(0)
|
|
{
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline xbitset_iterator<B, C>::xbitset_iterator(container_reference c, size_type index) noexcept
|
|
: p_container(&c), m_index(index)
|
|
{
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_iterator<B, C>::operator++() -> self_type&
|
|
{
|
|
++m_index;
|
|
return *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_iterator<B, C>::operator--() -> self_type&
|
|
{
|
|
--m_index;
|
|
return *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_iterator<B, C>::operator+=(difference_type n) -> self_type&
|
|
{
|
|
difference_type res = static_cast<difference_type>(m_index) + n;
|
|
m_index = static_cast<size_type>(res);
|
|
return *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_iterator<B, C>::operator-=(difference_type n) -> self_type&
|
|
{
|
|
difference_type res = static_cast<difference_type>(m_index) - n;
|
|
m_index = static_cast<size_type>(res);
|
|
return *this;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_iterator<B, C>::operator-(const self_type& rhs) const -> difference_type
|
|
{
|
|
return difference_type(m_index - rhs.m_index);
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_iterator<B, C>::operator*() const -> reference
|
|
{
|
|
return (*p_container)[m_index];
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline auto xbitset_iterator<B, C>::operator->() const -> pointer
|
|
{
|
|
return &(operator*());
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline bool xbitset_iterator<B, C>::operator==(const self_type& rhs) const
|
|
{
|
|
return p_container == rhs.p_container && m_index == rhs.m_index;
|
|
}
|
|
|
|
template <class B, bool C>
|
|
inline bool xbitset_iterator<B, C>::operator<(const self_type& rhs) const
|
|
{
|
|
return p_container == rhs.p_container && m_index < rhs.m_index;
|
|
}
|
|
}
|
|
|
|
#endif
|