X-Git-Url: https://git.lttng.org/?a=blobdiff_plain;f=src%2Fcommon%2Fcontainer-wrapper.hpp;h=25385fc43837961ec4301dd9f7254f0b89ccc3b6;hb=HEAD;hp=bccb23ddaeaf05efc27c51985ddd5a67004ceaa9;hpb=f74e820c5e8f81d6416f4557bce8b56ee6be4746;p=lttng-tools.git diff --git a/src/common/container-wrapper.hpp b/src/common/container-wrapper.hpp index bccb23dda..25385fc43 100644 --- a/src/common/container-wrapper.hpp +++ b/src/common/container-wrapper.hpp @@ -8,6 +8,8 @@ #ifndef LTTNG_CONTAINER_WRAPPER_H #define LTTNG_CONTAINER_WRAPPER_H +#include +#include #include #include @@ -58,6 +60,11 @@ class random_access_container_wrapper { return *this; } + ptrdiff_t operator-(const _iterator& other) const + { + return _index - other._index; + } + bool operator==(const _iterator& other) const noexcept { return _index == other._index; @@ -71,7 +78,7 @@ class random_access_container_wrapper { typename std::conditional::value, IteratorElementType, IteratorElementType&>::type - operator*() const noexcept + operator*() const { return _container[_index]; } @@ -95,9 +102,9 @@ public: return iterator(*this); } - iterator end() noexcept + iterator end() { - return iterator(*this, ContainerOperations::size(_container)); + return iterator(*this, size()); } const_iterator begin() const noexcept @@ -105,21 +112,40 @@ public: return const_iterator(*this); } - const_iterator end() const noexcept + const_iterator end() const { - return const_iterator(*this, ContainerOperations::size(_container)); + return const_iterator(*this, size()); } - std::size_t size() const noexcept + std::size_t size() const { return ContainerOperations::size(_container); } + bool empty() const + { + return size() == 0; + } + typename std::conditional::value, ElementType, ElementType&>::type operator[](std::size_t index) { - LTTNG_ASSERT(index < ContainerOperations::size(_container)); - return ContainerOperations::get(_container, index); + /* + * To share code between the const and mutable versions of this operator, 'this' + * is casted to a const reference. A const_cast then ensures that a mutable + * reference (or pointer) is returned. + * + * We typically avoid const_cast, but this is safe: if the user is calling the + * mutable version of this operator, it had a mutable object anyhow. + * + * For more information, see Item 3 of Effective C++. + */ + const auto& const_this = static_cast(*this); + + /* NOLINTNEXTLINE(cppcoreguidelines-pro-type-const-cast) */ + return const_cast::value, + ElementType, + ElementType&>::type>(const_this[index]); } typename std::conditional::value, @@ -127,7 +153,13 @@ public: const ElementType&>::type operator[](std::size_t index) const { - LTTNG_ASSERT(index < ContainerOperations::size(_container)); + if (index >= ContainerOperations::size(_container)) { + throw std::invalid_argument(lttng::format( + "Out of bound access through random_access_container_wrapper: index={}, size={}", + index, + size())); + } + return ContainerOperations::get(_container, index); }