Imt.Base C++ API V4.1.1.0
Loading...
Searching...
No Matches
WaitFreeRingBuffer.h
Go to the documentation of this file.
1// (c) IMT - Information Management Technology AG, CH-9470 Buchs, www.imt.ch.
2//
3// ActiveParts (AP) and the corresponding Data Flow Framework (DFF) is invented and designed by Jakob Daescher.
4// ANY USE OF THIS CODE CONSTITUTES ACCEPTANCE OF THE TERMS OF THE COPYRIGHT NOTICE.
5// ===================================================================================================
6// COPYRIGHT NOTICE
7// ===================================================================================================
8// Copyright (C) 2005-2075, IMT Information Management Technology AG, 9470 Buchs, Switzerland
9// All rights reserved.
10// This code is proprietary software of IMT Information Management Technology AG (hereinafter: "IMT").
11// Proprietary software is computer software licensed under exclusive legal right of IMT.
12//
13// The licensee is given the irrevocable, perpetual, worldwide, non-exclusive right and license to use,
14// execute and reproduce the software in binary form within the licensed products.
15//
16// Redistribution and use in source forms, without modification, are permitted provided that the following conditions are met:
17// (1) Copying of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
18// (2) Copying of source code is only allowed for regulatory documentation and archiving purposes
19// (3) Redistributions in binary form must reproduce the above copyright notice,
20// this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
21//
22// IMT provide no reassurances that the source code provided does not infringe
23// any patent, copyright, or any other intellectual property rights of third parties.
24// IMT disclaim any liability to any recipient for claims brought against
25// recipient by any third party for infringement of that parties intellectual property rights.
26//
27// THIS SOFTWARE IS PROVIDED BY IMT AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
28// INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
29// IN NO EVENT SHALL IMT OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY,
30// OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCURE-MENT OF SUBSTITUTE GOODS OR SERVICES;
31// LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
32// WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING
33// IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
34// ===================================================================================================
35
36#ifndef IMT_BASE_CORE_UTIL_WAITFREERINGBUFFER_H
37#define IMT_BASE_CORE_UTIL_WAITFREERINGBUFFER_H
38
39// Platform include.
41
42// Standard includes.
43#include <array>
44#include <atomic>
45#include <cstring>
46
47namespace imt {
48namespace base {
49namespace core {
50namespace util {
51
60template<typename ElementType, uint16_t MaxSize>
61class WaitFreeRingBuffer final {
62
63 // At least one element required.
64 static_assert(MaxSize > 0U);
65
66 // Lock-Free assurance, see https://en.cppreference.com/w/cpp/atomic/atomic_is_lock_free
67 static_assert(ATOMIC_SHORT_LOCK_FREE == 2);
68
69public:
70
71 /*
72 * Default constructor.
73 */
74 WaitFreeRingBuffer() = default;
75
76 /*
77 * Destructor.
78 */
79 ~WaitFreeRingBuffer() noexcept = default;
80
88 bool push(ElementType const& value) {
89 uint16_t const inPos {m_inPos.load(std::memory_order_relaxed)};
90 uint16_t const nextPos {next(inPos)};
91 if (nextPos == m_outPos.load(std::memory_order_acquire)) {
92 return false;
93 }
94 m_buf[inPos] = value;
95 m_inPos.store(nextPos, std::memory_order_release);
96 return true;
97 }
98
99 // AXIVION Next Routine AutosarC++19_03-A8.4.4: Use output parameter instead of return value for better usage.
100 // AXIVION Next Routine AutosarC++19_03-A8.4.8: Use output parameter instead of return value for better usage.
101 // AXIVION Next Routine AutosarC++19_03-A8.4.9: Intended that output parameter is replaced, not modified.
109 bool pop(ElementType& value) {
110 uint16_t const outPos {m_outPos.load(std::memory_order_relaxed)};
111 if (outPos == m_inPos.load(std::memory_order_acquire)) {
112 return false;
113 }
114 value = m_buf[outPos];
115 m_outPos.store(next(outPos), std::memory_order_release);
116 return true;
117 }
118
125 uint16_t const inPos {m_inPos.load(std::memory_order_acquire)};
126 uint16_t const outPos {m_outPos.load(std::memory_order_relaxed)};
127 return getSizeUsed(inPos, outPos);
128 }
129
136 uint16_t const inPos {m_inPos.load(std::memory_order_relaxed)};
137 uint16_t const outPos {m_outPos.load(std::memory_order_acquire)};
138 uint16_t const sizeUsed {getSizeUsed(inPos, outPos)};
139 return BUFFER_SIZE - sizeUsed - 1U;
140 }
141
148 return MAX_SIZE;
149 }
150
151private:
152
153 // Constants.
154 static constexpr uint16_t MAX_SIZE {MaxSize};
155 static constexpr uint16_t BUFFER_SIZE {MAX_SIZE + 1U};
156
157 // Deleted constructors/assigners.
158 WaitFreeRingBuffer(WaitFreeRingBuffer const&) = delete;
159 WaitFreeRingBuffer(WaitFreeRingBuffer&& other) = delete;
160 WaitFreeRingBuffer& operator=(WaitFreeRingBuffer const&) = delete;
161 WaitFreeRingBuffer& operator=(WaitFreeRingBuffer&& other) = delete;
162
163 // Helpers.
164 static uint16_t next(uint16_t const current) {
165 return (current + 1U) % BUFFER_SIZE;
166 }
167
168 static uint16_t getSizeUsed(uint16_t const inPos, uint16_t const outPos) {
169 return (inPos >= outPos) ? (inPos - outPos) : (BUFFER_SIZE - (outPos - inPos));
170 }
171
172 // Members.
173 std::atomic<uint16_t> m_inPos {0U};
174 std::atomic<uint16_t> m_outPos {0U};
175 std::array<ElementType, BUFFER_SIZE> m_buf {};
176};
177
178} // namespace util
179} // namespace core
180} // namespace base
181} // namespace imt
182
183#endif // IMT_BASE_CORE_UTIL_WAITFREERINGBUFFER_H
This template class implements a FIFO ringbuffer to transfer data from a producer thread to a consume...
bool push(ElementType const &value)
Add a value to the head of the buffer.
uint16_t getReadAvailable() const
Get number of available values to read.
static uint16_t getMaxSize()
Get template parameter value MaxSize.
uint16_t getWriteAvailable() const
Get number of available values to write.
bool pop(ElementType &value)
Read and remove the value from the tail of the buffer.
This is a application specific file which is used to configure Imt.Base.Core.Math.
unsigned __int16 uint16_t
Definition stdint.h:63