/*
* Copyright (c) 2021-2023 Huawei Device Co., Ltd.
* 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.
*/
/**
* @file safe_queue.h
*
* @brief Provides interfaces for thread-safe queue operations in c_utils.
*
* The file contains the thread-safe abstract class, the SafeQueue
* and SafeStack that override the virtual methods of the abstract class.
*/
#ifndef UTILS_BASE_SAFE_QUEUE_H
#define UTILS_BASE_SAFE_QUEUE_H
#include
#include
namespace OHOS {
/**
* @brief Provides an abstract class for thread-safe queues.
*
* It encapsulates std::lock_guard locks on the basis of std::deque to
* make the interfaces of the queue thread-safe.
*/
template
class SafeQueueInner {
public:
SafeQueueInner() {}
virtual ~SafeQueueInner()
{
if (!deque_.empty()) {
deque_.clear();
}
}
void Erase(const T& object)
{
std::lock_guard lock(mutex_);
for (auto iter = deque_.begin(); iter != deque_.end(); iter++) {
if (*iter == object) {
deque_.erase(iter);
break;
}
}
}
bool Empty()
{
std::lock_guard lock(mutex_);
return deque_.empty();
}
void Push(const T& pt)
{
std::lock_guard lock(mutex_);
return DoPush(pt);
}
void Clear()
{
std::lock_guard lock(mutex_);
if (!deque_.empty()) {
deque_.clear();
}
return;
}
int Size()
{
std::lock_guard lock(mutex_);
return deque_.size();
}
bool Pop(T& pt)
{
std::lock_guard lock(mutex_);
return DoPop(pt);
}
protected:
virtual void DoPush(const T& pt) = 0;
virtual bool DoPop(T& pt) = 0;
std::deque deque_;
std::mutex mutex_;
};
/**
* @brief Provides thread-safe queue operations.
*
* It overrides the DoPush and DoPop methods of abstract classes
* to implement the push and pop functionality of SafeQueue.
*/
template
class SafeQueue : public SafeQueueInner {
protected:
using SafeQueueInner::deque_;
using SafeQueueInner::mutex_;
void DoPush(const T& pt) override
{
deque_.push_back(pt);
}
/**
* @brief Encapsulates the pop_front() method
* to implement the pop function of queues.
*/
bool DoPop(T& pt) override
{
if (deque_.size() > 0) {
pt = deque_.front();
deque_.pop_front();
return true;
}
return false;
}
};
/**
* @brief Provides thread-safe stack operations.
*
* It overrides the DoPush and DoPop methods of abstract classes
* to implement the push and pop functionality of SafeStack.
*/
template
class SafeStack : public SafeQueueInner {
protected:
using SafeQueueInner::deque_;
using SafeQueueInner::mutex_;
void DoPush(const T& pt) override
{
deque_.push_back(pt);
}
/**
* @brief Encapsulates the pop_back() method
* to implement the pop function of stack.
*/
bool DoPop(T& pt) override
{
if (deque_.size() > 0) {
pt = deque_.back();
deque_.pop_back();
return true;
}
return false;
}
};
} // namespace OHOS
#endif