aboutsummaryrefslogtreecommitdiff
path: root/Src/nu/LockFreeItem.h
diff options
context:
space:
mode:
authorJean-Francois Mauguit <jfmauguit@mac.com>2024-09-24 09:03:25 -0400
committerGitHub <noreply@github.com>2024-09-24 09:03:25 -0400
commitbab614c421ed7ae329d26bf028c4a3b1d2450f5a (patch)
tree12f17f78986871dd2cfb0a56e5e93b545c1ae0d0 /Src/nu/LockFreeItem.h
parent4bde6044fddf053f31795b9eaccdd2a5a527d21f (diff)
parent20d28e80a5c861a9d5f449ea911ab75b4f37ad0d (diff)
downloadwinamp-bab614c421ed7ae329d26bf028c4a3b1d2450f5a.tar.gz
Merge pull request #5 from WinampDesktop/community
Merge to main
Diffstat (limited to 'Src/nu/LockFreeItem.h')
-rw-r--r--Src/nu/LockFreeItem.h47
1 files changed, 47 insertions, 0 deletions
diff --git a/Src/nu/LockFreeItem.h b/Src/nu/LockFreeItem.h
new file mode 100644
index 00000000..b3cc0649
--- /dev/null
+++ b/Src/nu/LockFreeItem.h
@@ -0,0 +1,47 @@
+#pragma once
+#include <windows.h>
+
+/* This data structure holds one item, and returns you the old item when you replace it
+ it's sort of a "stack of 1" */
+
+template <class value_t>
+class LockFreeItem
+{
+public:
+ typedef value_t *ptr_t;
+ LockFreeItem() { item = 0; }
+
+ value_t *GetItem()
+ {
+ for(;;) // keep trying until we get this right
+ {
+ volatile ptr_t ret;
+ InterlockedExchangePointer(&ret, item);
+ if (InterlockedCompareExchangePointer((volatile PVOID *)&item, 0, ret) == ret)
+ return ret;
+ }
+ }
+
+ // returns the previous value
+ value_t *SetItem(value_t *new_item)
+ {
+ for(;;) // keep trying until we get this right
+ {
+ volatile ptr_t ret;
+ InterlockedExchangePointer(&ret, item);
+ if (InterlockedCompareExchangePointer((volatile PVOID *)&item, new_item, ret) == ret)
+ return ret;
+ }
+ }
+
+ // if there's already a value, returns what you passed in
+ value_t *SetItemIfZero(value_t *new_item)
+ {
+ if (InterlockedCompareExchangePointer((volatile PVOID *)&item, new_item, 0) == 0)
+ return 0;
+ else
+ return new_item;
+ }
+
+ volatile ptr_t item;
+}; \ No newline at end of file