Diligent Engine  v.2.4.g
LockHelper.hpp
Go to the documentation of this file.
1 /*
2  * Copyright 2019-2021 Diligent Graphics LLC
3  * Copyright 2015-2019 Egor Yusov
4  *
5  * Licensed under the Apache License, Version 2.0 (the "License");
6  * you may not use this file except in compliance with the License.
7  * You may obtain a copy of the License at
8  *
9  * http://www.apache.org/licenses/LICENSE-2.0
10  *
11  * Unless required by applicable law or agreed to in writing, software
12  * distributed under the License is distributed on an "AS IS" BASIS,
13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14  * See the License for the specific language governing permissions and
15  * limitations under the License.
16  *
17  * In no event and under no legal theory, whether in tort (including negligence),
18  * contract, or otherwise, unless required by applicable law (such as deliberate
19  * and grossly negligent acts) or agreed to in writing, shall any Contributor be
20  * liable for any damages, including any direct, indirect, special, incidental,
21  * or consequential damages of any character arising as a result of this License or
22  * out of the use or inability to use the software (including but not limited to damages
23  * for loss of goodwill, work stoppage, computer failure or malfunction, or any and
24  * all other commercial damages or losses), even if such Contributor has been advised
25  * of the possibility of such damages.
26  */
27 
28 #pragma once
29 
30 #include "../../Platforms/interface/Atomics.hpp"
31 #include "../../Platforms/Basic/interface/DebugUtilities.hpp"
32 
33 namespace ThreadingTools
34 {
35 
36 class LockFlag
37 {
38 public:
39  enum
40  {
43  };
45  {
46  //m_Flag.store(InitFlag);
47  m_Flag = InitFlag;
48  }
49 
50  operator Atomics::Long() const { return m_Flag; }
51 
52 private:
53  friend class LockHelper;
54  Atomics::AtomicLong m_Flag;
55 };
56 
57 // Spinlock implementation. This kind of lock should be used in scenarios
58 // where simultaneous access is uncommon but possible.
60 {
61 public:
62  LockHelper() noexcept {}
63 
65  {
66  Lock(LockFlag);
67  }
68 
70  m_pLockFlag{std::move(LockHelper.m_pLockFlag)}
71  {
72  LockHelper.m_pLockFlag = nullptr;
73  }
74 
76  {
77  m_pLockFlag = std::move(LockHelper.m_pLockFlag);
78  LockHelper.m_pLockFlag = nullptr;
79  return *this;
80  }
81 
83  {
84  Unlock();
85  }
86 
87  static bool UnsafeTryLock(LockFlag& LockFlag) noexcept
88  {
92  }
93 
94  bool TryLock(LockFlag& LockFlag) noexcept
95  {
97  {
98  m_pLockFlag = &LockFlag;
99  return true;
100  }
101  else
102  return false;
103  }
104 
105  static constexpr const int DefaultSpinCountToYield = 256;
106 
107  static void UnsafeLock(LockFlag& LockFlag, int SpinCountToYield = DefaultSpinCountToYield) noexcept
108  {
109  int SpinCount = 0;
110  while (!UnsafeTryLock(LockFlag))
111  {
112  ++SpinCount;
113  if (SpinCount == SpinCountToYield)
114  {
115  SpinCount = 0;
116  YieldThread();
117  }
118  }
119  }
120 
121  void Lock(LockFlag& LockFlag, int SpinCountToYield = DefaultSpinCountToYield) noexcept
122  {
123  VERIFY(m_pLockFlag == NULL, "Object already locked");
124  // Wait for the flag to become unlocked and lock it
125  int SpinCount = 0;
126  while (!TryLock(LockFlag))
127  {
128  ++SpinCount;
129  if (SpinCount == SpinCountToYield)
130  {
131  SpinCount = 0;
132  YieldThread();
133  }
134  }
135  }
136 
137  static void UnsafeUnlock(LockFlag& LockFlag) noexcept
138  {
140  }
141 
142  void Unlock() noexcept
143  {
144  if (m_pLockFlag)
145  UnsafeUnlock(*m_pLockFlag);
146  m_pLockFlag = NULL;
147  }
148 
149 private:
150  static void YieldThread() noexcept;
151 
152  LockFlag* m_pLockFlag = nullptr;
153 
154  // clang-format off
155  LockHelper (const LockHelper& LockHelper) = delete;
156  LockHelper& operator=(const LockHelper& LockHelper) = delete;
157  // clang-format on
158 };
159 
160 } // namespace ThreadingTools
ThreadingTools::LockHelper::LockHelper
LockHelper(LockFlag &LockFlag) noexcept
Definition: LockHelper.hpp:64
ThreadingTools
Definition: LockHelper.hpp:33
ThreadingTools::LockHelper::DefaultSpinCountToYield
static constexpr const int DefaultSpinCountToYield
Definition: LockHelper.hpp:105
BasicAtomics::Long
long Long
Definition: BasicAtomics.hpp:34
ThreadingTools::LockHelper
Definition: LockHelper.hpp:59
ThreadingTools::LockHelper::Lock
void Lock(LockFlag &LockFlag, int SpinCountToYield=DefaultSpinCountToYield) noexcept
Definition: LockHelper.hpp:121
ThreadingTools::LockFlag::LOCK_FLAG_LOCKED
@ LOCK_FLAG_LOCKED
Definition: LockHelper.hpp:42
ThreadingTools::LockHelper::LockHelper
LockHelper(LockHelper &&LockHelper) noexcept
Definition: LockHelper.hpp:69
ThreadingTools::LockFlag
Definition: LockHelper.hpp:36
ThreadingTools::LockFlag::LockFlag
LockFlag(Atomics::Long InitFlag=LOCK_FLAG_UNLOCKED) noexcept
Definition: LockHelper.hpp:44
ThreadingTools::LockHelper::UnsafeUnlock
static void UnsafeUnlock(LockFlag &LockFlag) noexcept
Definition: LockHelper.hpp:137
ThreadingTools::LockHelper::~LockHelper
~LockHelper()
Definition: LockHelper.hpp:82
ThreadingTools::LockHelper::LockHelper
LockHelper() noexcept
Definition: LockHelper.hpp:62
ThreadingTools::LockHelper::UnsafeLock
static void UnsafeLock(LockFlag &LockFlag, int SpinCountToYield=DefaultSpinCountToYield) noexcept
Definition: LockHelper.hpp:107
BasicAtomics::AtomicLong
std::atomic< Long > AtomicLong
Definition: BasicAtomics.hpp:35
ThreadingTools::LockFlag::LOCK_FLAG_UNLOCKED
@ LOCK_FLAG_UNLOCKED
Definition: LockHelper.hpp:41
BasicAtomics::AtomicCompareExchange
static Type AtomicCompareExchange(std::atomic< Type > &Destination, Type Exchange, Type Comparand)
Definition: BasicAtomics.hpp:58
VERIFY
#define VERIFY(...)
Definition: DebugUtilities.hpp:76
ThreadingTools::LockHelper::operator=
const LockHelper & operator=(LockHelper &&LockHelper) noexcept
Definition: LockHelper.hpp:75
ThreadingTools::LockHelper::TryLock
bool TryLock(LockFlag &LockFlag) noexcept
Definition: LockHelper.hpp:94
ThreadingTools::LockHelper::Unlock
void Unlock() noexcept
Definition: LockHelper.hpp:142
ThreadingTools::LockHelper::UnsafeTryLock
static bool UnsafeTryLock(LockFlag &LockFlag) noexcept
Definition: LockHelper.hpp:87