iceoryx_utils/internal/concurrent/lockfree_queue/cyclic_index.hpp🔗
Namespaces🔗
Name |
---|
iox building block to easily create free function for logging in a library context |
iox::concurrent |
Classes🔗
Name | |
---|---|
class | iox::concurrent::CyclicIndex index structure that can contain logical values 0, ..., CycleLength-1 but also stores an internal cycle counter to be used in compare_exchange |
Source code🔗
// Copyright (c) 2019, 2020 by Robert Bosch GmbH, Apex.AI Inc. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//
// SPDX-License-Identifier: Apache-2.0
#ifndef IOX_UTILS_LOCKFREE_QUEUE_CYCLIC_INDEX_HPP
#define IOX_UTILS_LOCKFREE_QUEUE_CYCLIC_INDEX_HPP
#include <limits>
#include <stdint.h>
#include <type_traits>
namespace iox
{
namespace concurrent
{
template <uint64_t CycleLength, typename ValueType = uint64_t>
class CyclicIndex
{
public:
static_assert(std::is_unsigned<ValueType>::value, "ValueType must be an unsigned integral type");
static_assert(CycleLength >= 1U, "CycleLength must be >= 1");
using value_t = ValueType;
static constexpr ValueType MAX_INDEX = CycleLength - 1U;
static constexpr ValueType MAX_VALUE = std::numeric_limits<ValueType>::max();
// assumes MAX_VALUE >= CycleLength, otherwise we could not fit in even one cycle
static constexpr ValueType MAX_CYCLE = MAX_VALUE / CycleLength;
static constexpr ValueType INDEX_AT_MAX_VALUE = MAX_VALUE % CycleLength;
static constexpr ValueType OVERFLOW_START_INDEX = (INDEX_AT_MAX_VALUE + 1U) % CycleLength;
static_assert(CycleLength < MAX_VALUE / 2U, "CycleLength is too large, need at least one bit for cycle");
static_assert(CycleLength > 0, "CycleLength must be > 0");
explicit CyclicIndex(ValueType value = 0U) noexcept;
CyclicIndex(ValueType index, ValueType cycle) noexcept;
CyclicIndex(const CyclicIndex&) = default;
CyclicIndex(CyclicIndex&&) = default;
CyclicIndex& operator=(const CyclicIndex&) = default;
CyclicIndex& operator=(CyclicIndex&&) = default;
ValueType getIndex() const noexcept;
ValueType getCycle() const noexcept;
ValueType getValue() const noexcept;
CyclicIndex operator+(const ValueType value) const noexcept;
CyclicIndex next() const noexcept;
bool isOneCycleBehind(const CyclicIndex& other) const noexcept;
int64_t operator-(const CyclicIndex<CycleLength, ValueType>& rhs) const;
private:
ValueType m_value{0U};
};
} // namespace concurrent
} // namespace iox
#include "cyclic_index.inl"
#endif // IOX_UTILS_LOCKFREE_QUEUE_CYCLIC_INDEX_HPP
Updated on 31 May 2022 at 15:29:15 CEST