-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcqueue.h
executable file
·144 lines (130 loc) · 2.67 KB
/
cqueue.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
#pragma once
#include <stdint.h>
#include <atomic>
/*
* size of queue, n: 4
* 00
* 01
* 10
* 11
* mask: n(100) -1 = 11
*
* pos mask val
* 00 11 00
* 01 11 01
* 10 11 10
* 11 11 11
*
* push -> from head
* pop -> from tail
*
* 3
* 2
* 1
* 0 <-- head=0, tail=0 (empty)
*
* push[1]
* 3
* 2
* 1 <-- head=1
* 0 <-- tail=0
*
* pop[0]
* 3
* 2
* 1 <--head=1, tail=1 (empty)
* 0
*
* push[1]
* 3
* 2 <-- head=2
* 1 <-- tail=1
* 0
*
* push[2]
* 3 <-- head=3
* 2
* 1 <-- tail=1
* 0
*
* push[3]
* 3
* 2
* 1 <-- tail=1
* 0 <-- head(4)=0
*
* push[4]
* 3
* 2
* 1 <-- head(5)=1 tail=1 (full)
* 0
*
* pop[3]
* 3
* 2 <-- tail=2
* 1 <-- head(5)=1
* 0
*
* pop[2]
* 3 <-- tail=3
* 2
* 1 <-- head(5)=1
* 0
*
* pop[1]
* 3
* 2
* 1 <-- head(5)=1
* 0 <-- tail(4)=0
*
* pop[0]
* 3
* 2
* 1 <-- head(5)=1 tail(5)=1 (empty)
* 0
*
* empty head = tail
* full head&mask = tail&mask or head = tail+#elements
* not full tail < head or head < tail + #elements
*
*
* tail + qsize-1 > head -> not full
* tail < head -> not full
* tail == head -> empty
*
* push - wait if full
*
*
*/
namespace collections
{
const int64_t lCachLine = 64;
// note: to push a class, this will need to be modified to do placement new
// allocates a ringbuffer of n*size of data and pushes data onto queue and copies data off
///
/// \brief The CQueueLockFreeSingle class
/// \note limitation, 64bit overflow
class CQueueLockFreeSingle final
{
private:
std::atomic<uint64_t> m_ulQueueSize;
uint8_t m_padding_0[lCachLine-sizeof(int64_t)];
std::atomic<uint64_t> m_ulHead = { 0 };
uint8_t m_padding_1[lCachLine-sizeof(int64_t)];
std::atomic<uint64_t> m_ulTail = { 0 };
uint8_t m_padding_2[lCachLine-sizeof(int64_t)];
const uint64_t m_ulDataSize;
const uint64_t m_ulMask;
void* m_pQueue;
// diagnotics
int m_nPageSize = 0;
public:
CQueueLockFreeSingle( uint64_t a_ulQueueItemCount, uint64_t a_ulDataSize );
CQueueLockFreeSingle( const CQueueLockFreeSingle& ) = delete;
~CQueueLockFreeSingle();
CQueueLockFreeSingle& operator= ( const CQueueLockFreeSingle& other ) = delete;
bool operator==( const CQueueLockFreeSingle& other ) = delete;
bool pop( void* a_pData );
bool push( void* a_pData );
};
}