diff options
Diffstat (limited to 'xbmc/threads/Condition.h')
-rw-r--r-- | xbmc/threads/Condition.h | 108 |
1 files changed, 108 insertions, 0 deletions
diff --git a/xbmc/threads/Condition.h b/xbmc/threads/Condition.h new file mode 100644 index 0000000..3738352 --- /dev/null +++ b/xbmc/threads/Condition.h @@ -0,0 +1,108 @@ +/* + * Copyright (C) 2005-2018 Team Kodi + * This file is part of Kodi - https://kodi.tv + * + * SPDX-License-Identifier: GPL-2.0-or-later + * See LICENSES/README.md for more information. + */ + +#pragma once + +#include "threads/CriticalSection.h" + +#include <chrono> +#include <condition_variable> +#include <functional> +#include <mutex> +#include <utility> + +namespace XbmcThreads +{ + + /** + * This is a thin wrapper around std::condition_variable_any. It is subject + * to "spurious returns" + */ + class ConditionVariable + { + private: + std::condition_variable_any cond; + ConditionVariable(const ConditionVariable&) = delete; + ConditionVariable& operator=(const ConditionVariable&) = delete; + + public: + ConditionVariable() = default; + + inline void wait(CCriticalSection& lock, std::function<bool()> predicate) + { + int count = lock.count; + lock.count = 0; + cond.wait(lock.get_underlying(), std::move(predicate)); + lock.count = count; + } + + inline void wait(CCriticalSection& lock) + { + int count = lock.count; + lock.count = 0; + cond.wait(lock.get_underlying()); + lock.count = count; + } + + template<typename Rep, typename Period> + inline bool wait(CCriticalSection& lock, + std::chrono::duration<Rep, Period> duration, + std::function<bool()> predicate) + { + int count = lock.count; + lock.count = 0; + bool ret = cond.wait_for(lock.get_underlying(), duration, predicate); + lock.count = count; + return ret; + } + + template<typename Rep, typename Period> + inline bool wait(CCriticalSection& lock, std::chrono::duration<Rep, Period> duration) + { + int count = lock.count; + lock.count = 0; + std::cv_status res = cond.wait_for(lock.get_underlying(), duration); + lock.count = count; + return res == std::cv_status::no_timeout; + } + + inline void wait(std::unique_lock<CCriticalSection>& lock, std::function<bool()> predicate) + { + cond.wait(*lock.mutex(), std::move(predicate)); + } + + inline void wait(std::unique_lock<CCriticalSection>& lock) { wait(*lock.mutex()); } + + template<typename Rep, typename Period> + inline bool wait(std::unique_lock<CCriticalSection>& lock, + std::chrono::duration<Rep, Period> duration, + std::function<bool()> predicate) + { + return wait(*lock.mutex(), duration, predicate); + } + + template<typename Rep, typename Period> + inline bool wait(std::unique_lock<CCriticalSection>& lock, + std::chrono::duration<Rep, Period> duration) + { + return wait(*lock.mutex(), duration); + } + + inline void notifyAll() + { + cond.notify_all(); + } + + inline void notify() + { + cond.notify_one(); + } + }; + +} + |