14 #ifndef ATOMIC_QUEUE_H_
15 #define ATOMIC_QUEUE_H_
23 template <
typename T,
unsigned int SIZE,
typename const_ref = T>
29 memset(m_ptrs, 0, SIZE *
sizeof(T));
34 throw nospace_error();
45 while((T)*first == 0) {
47 if(first == &m_ptrs[SIZE]) {
58 unsigned int size()
const {
69 unsigned int x = m_count;
75 if(m_count.compare_set_strong(x, x + 1)) {
83 while((T)*last != 0) {
85 if(last == &m_ptrs[SIZE]) {
90 if(m_pLast.compare_set_strong(last_org, last)) {
92 if(last->compare_set_strong((T)0, t)) {
105 if(first->compare_set_strong((T)item, (T)0)) {
119 const_ref t = *first;
121 if(m_pFirst.compare_set_strong(first_org, first))
126 if(first == &m_ptrs[SIZE]) {
140 T obj = first->exchange((T)0);
148 if(first == &m_ptrs[SIZE]) {
163 template <
typename T,
unsigned int SIZE>
167 template <
typename T,
unsigned int SIZE>
177 void push(
const T&t) {
182 catch (nospace_error &e) {
189 delete m_queue.front();
194 return *m_queue.front();
198 return m_queue.empty();
200 unsigned int size()
const {
201 return m_queue.size();
208 template <
typename T,
unsigned int SIZE>
212 typedef uint_cas_max key;
215 static_assert(SIZE < (1uLL << (
sizeof(key) * 8 - 8)),
"Size mismatch.");
216 for(
unsigned int i = 0; i < SIZE; i++) {
217 m_reservoir.push(key_index_serial(i, 0));
222 assert(m_reservoir.size() == SIZE);
225 void push(
const T&t) {
226 key pack = m_reservoir.atomicPopAny();
228 throw nospace_error();
229 int idx = key2index(pack);
231 int serial = key2serial(pack) + 1;
232 pack = key_index_serial(idx, serial);
236 catch (nospace_error&e) {
238 m_reservoir.push(pack);
240 catch (nospace_error&) {
248 key pack = m_queue.
front();
251 m_reservoir.push(pack);
253 catch (nospace_error&) {
259 return m_array[key2index(m_queue.
front())];
263 return m_queue.
empty();
265 unsigned int size()
const {
266 return m_queue.size();
275 m_reservoir.push(item);
277 catch (nospace_error&) {
288 *val = m_array[key2index(pack)];
292 int key2index(key i) {
return (
unsigned int)i / 0x100;}
293 int key2serial(key i) {
return ((
unsigned int)i % 0x100) - 1;}
294 key key_index_serial(
int index,
int serial) {
return index * 0x100 + (serial % 0xff) + 1;}