2015-06-23 00:08:42 -07:00
|
|
|
//
|
|
|
|
|
// Copyright (c) 2013 Juan Palacios juan.palacios.puyana@gmail.com
|
|
|
|
|
// Subject to the BSD 2-Clause License
|
|
|
|
|
// - see < http://opensource.org/licenses/BSD-2-Clause>
|
|
|
|
|
//
|
|
|
|
|
|
2016-07-27 00:39:38 -07:00
|
|
|
#ifndef WPIUTIL_SUPPORT_CONCURRENT_QUEUE_H_
|
|
|
|
|
#define WPIUTIL_SUPPORT_CONCURRENT_QUEUE_H_
|
2015-06-23 00:08:42 -07:00
|
|
|
|
|
|
|
|
#include <queue>
|
|
|
|
|
#include <thread>
|
|
|
|
|
#include <mutex>
|
|
|
|
|
#include <condition_variable>
|
|
|
|
|
|
2016-07-27 00:39:38 -07:00
|
|
|
namespace wpi {
|
|
|
|
|
|
2015-06-23 00:08:42 -07:00
|
|
|
template <typename T>
|
2015-06-25 22:57:43 -07:00
|
|
|
class ConcurrentQueue {
|
|
|
|
|
public:
|
2015-07-13 22:56:58 -07:00
|
|
|
bool empty() const {
|
|
|
|
|
std::unique_lock<std::mutex> mlock(mutex_);
|
|
|
|
|
return queue_.empty();
|
|
|
|
|
}
|
|
|
|
|
|
2015-07-18 01:32:21 -07:00
|
|
|
typename std::queue<T>::size_type size() const {
|
|
|
|
|
std::unique_lock<std::mutex> mlock(mutex_);
|
|
|
|
|
return queue_.size();
|
|
|
|
|
}
|
|
|
|
|
|
2015-06-25 22:57:43 -07:00
|
|
|
T pop() {
|
|
|
|
|
std::unique_lock<std::mutex> mlock(mutex_);
|
|
|
|
|
while (queue_.empty()) {
|
|
|
|
|
cond_.wait(mlock);
|
2015-06-23 00:08:42 -07:00
|
|
|
}
|
2015-06-25 22:57:43 -07:00
|
|
|
auto item = std::move(queue_.front());
|
|
|
|
|
queue_.pop();
|
|
|
|
|
return item;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void pop(T& item) {
|
|
|
|
|
std::unique_lock<std::mutex> mlock(mutex_);
|
|
|
|
|
while (queue_.empty()) {
|
|
|
|
|
cond_.wait(mlock);
|
2015-06-23 00:08:42 -07:00
|
|
|
}
|
2015-06-25 22:57:43 -07:00
|
|
|
item = queue_.front();
|
|
|
|
|
queue_.pop();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void push(const T& item) {
|
|
|
|
|
std::unique_lock<std::mutex> mlock(mutex_);
|
|
|
|
|
queue_.push(item);
|
|
|
|
|
mlock.unlock();
|
|
|
|
|
cond_.notify_one();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void push(T&& item) {
|
|
|
|
|
std::unique_lock<std::mutex> mlock(mutex_);
|
2015-07-21 22:43:02 -07:00
|
|
|
queue_.push(std::forward<T>(item));
|
|
|
|
|
mlock.unlock();
|
|
|
|
|
cond_.notify_one();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename... Args>
|
|
|
|
|
void emplace(Args&&... args) {
|
|
|
|
|
std::unique_lock<std::mutex> mlock(mutex_);
|
|
|
|
|
queue_.emplace(std::forward<Args>(args)...);
|
2015-06-25 22:57:43 -07:00
|
|
|
mlock.unlock();
|
|
|
|
|
cond_.notify_one();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ConcurrentQueue() = default;
|
|
|
|
|
ConcurrentQueue(const ConcurrentQueue&) = delete;
|
|
|
|
|
ConcurrentQueue& operator=(const ConcurrentQueue&) = delete;
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
std::queue<T> queue_;
|
2015-07-19 16:36:08 -07:00
|
|
|
mutable std::mutex mutex_;
|
2015-06-25 22:57:43 -07:00
|
|
|
std::condition_variable cond_;
|
2015-06-23 00:08:42 -07:00
|
|
|
};
|
|
|
|
|
|
2016-07-27 00:39:38 -07:00
|
|
|
} // namespace wpi
|
|
|
|
|
|
|
|
|
|
#endif // WPIUTIL_SUPPORT_CONCURRENT_QUEUE_H_
|