0
0
Fork 0
mirror of https://github.com/matrix-construct/construct synced 2025-01-16 17:46:54 +01:00
construct/include/ircd/vector_view.h

165 lines
4.2 KiB
C
Raw Normal View History

2018-02-03 18:22:01 -08:00
// Matrix Construct
//
// Copyright (C) Matrix Construct Developers, Authors & Contributors
// Copyright (C) 2016-2018 Jason Volk <jason@zemos.net>
//
// Permission to use, copy, modify, and/or distribute this software for any
// purpose with or without fee is hereby granted, provided that the above
// copyright notice and this permission notice is present in all copies. The
// full license for this software is available in the LICENSE file.
2017-11-15 17:21:26 -08:00
#pragma once
#define HAVE_IRCD_VECTOR_VIEW_H
namespace ircd
{
template<class T> struct vector_view;
template<class T> bool empty(const vector_view<T> &) noexcept;
template<class T> size_t size(const vector_view<T> &) noexcept;
template<class T> T *data(const vector_view<T> &) noexcept;
2017-11-15 17:21:26 -08:00
}
/// Template to represent a contiguous vector or array in a generic way.
2017-11-15 17:21:26 -08:00
template<class T>
struct ircd::vector_view
{
using value_type = T;
using pointer = value_type *;
using reference = value_type &;
using difference_type = size_t;
using iterator = value_type *;
using const_iterator = const value_type *;
value_type *_data { nullptr };
value_type *_stop { nullptr };
public:
value_type *data() const noexcept { return _data; }
size_t size() const noexcept { return std::distance(_data, _stop); }
bool empty() const noexcept { return !size(); }
2017-11-15 17:21:26 -08:00
const_iterator begin() const noexcept { return data(); }
const_iterator end() const noexcept { return _stop; }
const_iterator cbegin() noexcept { return data(); }
const_iterator cend() noexcept { return _stop; }
iterator begin() noexcept { return data(); }
iterator end() noexcept { return _stop; }
2017-11-15 17:21:26 -08:00
// Bounds check in debug only.
value_type &operator[](const size_t &pos) const noexcept
2017-11-15 17:21:26 -08:00
{
assert(pos < size());
2017-11-15 17:21:26 -08:00
return *(data() + pos);
}
// Bounds check at runtime.
2017-11-15 17:21:26 -08:00
value_type &at(const size_t &pos) const
{
if(unlikely(pos >= size()))
throw std::out_of_range
{
"vector_view::range_check"
};
2017-11-15 17:21:26 -08:00
return operator[](pos);
}
value_type &back() const
{
return at(size() - 1);
}
value_type &front() const
{
return at(0);
}
2017-11-15 17:21:26 -08:00
vector_view(value_type *const &start, value_type *const &stop)
noexcept
2017-11-15 17:21:26 -08:00
:_data{start}
,_stop{stop}
{}
vector_view(value_type *const &start, const size_t &size)
noexcept
2017-11-15 17:21:26 -08:00
:vector_view(start, start + size)
{}
vector_view(const vector_view<value_type> &start, const size_t &size)
noexcept
:vector_view(start.data(), std::min(start.size(), size))
{}
2017-11-15 17:21:26 -08:00
vector_view(const std::initializer_list<value_type> &list)
noexcept
2017-11-15 17:21:26 -08:00
:vector_view(std::begin(list), std::end(list))
{}
template<class U,
class A>
vector_view(const std::vector<U, A> &v)
noexcept
2017-11-15 17:21:26 -08:00
:vector_view(v.data(), v.size())
{}
template<class U,
class A>
vector_view(std::vector<U, A> &v)
noexcept
2017-11-15 17:21:26 -08:00
:vector_view(v.data(), v.size())
{}
template<size_t SIZE>
vector_view(value_type (&buffer)[SIZE])
noexcept
2017-11-15 17:21:26 -08:00
:vector_view(buffer, SIZE)
{}
template<class U,
size_t SIZE>
vector_view(const std::array<U, SIZE> &array)
noexcept
:vector_view(const_cast<pointer>(array.data()), array.size())
2017-11-15 17:21:26 -08:00
{}
template<size_t SIZE>
vector_view(std::array<value_type, SIZE> &array)
noexcept
2017-11-15 17:21:26 -08:00
:vector_view(array.data(), array.size())
{}
// Required for reasonable implicit const conversion of value_type.
vector_view(const vector_view<typename std::remove_const<value_type>::type> &v)
noexcept
:vector_view(v.data(), v.size())
{}
vector_view() noexcept = default;
vector_view &operator=(const vector_view &) noexcept = default;
2017-11-15 17:21:26 -08:00
};
template<class T>
inline T *
ircd::data(const vector_view<T> &v)
noexcept
{
return v.data();
}
template<class T>
inline size_t
ircd::size(const vector_view<T> &v)
noexcept
{
return v.size();
}
template<class T>
inline bool
ircd::empty(const vector_view<T> &v)
noexcept
{
return v.empty();
}