2015-10-28 03:21:28 +08:00
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <atomic>
|
|
|
|
#include <memory>
|
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
namespace lockfree {
|
2015-10-28 03:21:28 +08:00
|
|
|
|
|
|
|
template <class T, size_t N>
|
2017-02-18 18:54:37 +08:00
|
|
|
class BoundedSpscQueue {
|
|
|
|
public:
|
|
|
|
static constexpr size_t size = N;
|
|
|
|
|
|
|
|
BoundedSpscQueue() = default;
|
|
|
|
|
|
|
|
BoundedSpscQueue(const BoundedSpscQueue&) = delete;
|
|
|
|
BoundedSpscQueue(BoundedSpscQueue&&) = delete;
|
|
|
|
|
|
|
|
BoundedSpscQueue& operator=(const BoundedSpscQueue&) = delete;
|
|
|
|
|
|
|
|
bool push(const T& item) {
|
|
|
|
// load the current tail
|
|
|
|
// [] [] [1] [2] [3] [4] [5] [$] []
|
|
|
|
// H T
|
|
|
|
auto t = tail.load(std::memory_order_relaxed);
|
|
|
|
|
|
|
|
// what will next tail be after we push
|
|
|
|
// [] [] [1] [2] [3] [4] [5] [$] [ ]
|
|
|
|
// H T T'
|
|
|
|
auto next = increment(t);
|
|
|
|
|
|
|
|
// check if queue is full and do nothing if it is
|
|
|
|
// [3] [4] [5] [6] [7] [8] [$] [ 1 ] [2]
|
|
|
|
// T T'H
|
|
|
|
if (next == head.load(std::memory_order_acquire)) return false;
|
|
|
|
|
|
|
|
// insert the item into the empty spot
|
|
|
|
// [] [] [1] [2] [3] [4] [5] [ ] []
|
|
|
|
// H T T'
|
|
|
|
items[t] = item;
|
|
|
|
|
|
|
|
// release the tail to the consumer (serialization point)
|
|
|
|
// [] [] [1] [2] [3] [4] [5] [ $ ] []
|
|
|
|
// H T T'
|
|
|
|
tail.store(next, std::memory_order_release);
|
2015-10-28 03:21:28 +08:00
|
|
|
|
2017-02-18 18:54:37 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool pop(T& item) {
|
|
|
|
// [] [] [1] [2] [3] [4] [5] [$] []
|
|
|
|
// H T
|
|
|
|
auto h = head.load(std::memory_order_relaxed);
|
|
|
|
|
|
|
|
// [] [] [] [] [ $ ] [] [] [] []
|
|
|
|
// H T
|
|
|
|
if (h == tail.load(std::memory_order_acquire)) return false;
|
|
|
|
|
|
|
|
// move an item from the queue
|
|
|
|
item = std::move(items[h]);
|
|
|
|
|
|
|
|
// serialization point wrt producer
|
|
|
|
// [] [] [] [2] [3] [4] [5] [$] []
|
|
|
|
// H T
|
|
|
|
head.store(increment(h), std::memory_order_release);
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static constexpr size_t capacity = N + 1;
|
|
|
|
|
|
|
|
std::array<T, capacity> items;
|
|
|
|
std::atomic<size_t> head{0}, tail{0};
|
|
|
|
|
|
|
|
size_t increment(size_t idx) const { return (idx + 1) % capacity; }
|
|
|
|
};
|
2015-10-28 03:21:28 +08:00
|
|
|
}
|