acl/lib_acl_cpp/include/acl_cpp/stdlib/atomic.hpp

193 lines
2.8 KiB
C++
Raw Normal View History

#pragma once
#include "../acl_cpp_define.hpp"
struct ACL_ATOMIC;
extern "C" {
extern ACL_ATOMIC *acl_atomic_new(void);
extern void acl_atomic_free(ACL_ATOMIC *self);
extern void acl_atomic_set(ACL_ATOMIC *self, void *value);
extern void *acl_atomic_cas(ACL_ATOMIC *self, void *cmp, void *value);
extern void *acl_atomic_xchg(ACL_ATOMIC *self, void *value);
}
namespace acl
{
template<typename T>
class atomic
{
public:
atomic(T* t)
{
atomic_ = acl_atomic_new();
acl_atomic_set(atomic_, t);
}
virtual ~atomic(void)
{
acl_atomic_free(atomic_);
}
T* cas(T* cmp, T* val)
{
return (T*) acl_atomic_cas(atomic_, cmp, val);
}
T* xchg(T* val)
{
return (T*) acl_atomic_xchg(atomic_, val);
}
protected:
ACL_ATOMIC* atomic_;
};
class atomic_long : public atomic<long long>
{
public:
atomic_long(long long n = 0);
~atomic_long(void) {}
void set(long long n);
long long fetch_add(long long n);
long long add_fetch(long long n);
operator long long() const
{
return n_;
}
long long value(void)
{
return n_;
}
void operator=(long long n)
{
set(n);
}
long long operator++()
{
return add_fetch(1);
}
long long operator++(int)
{
return fetch_add(1);
}
long long operator--()
{
return add_fetch(-1);
}
long long operator--(int)
{
return fetch_add(-1);
}
long long operator+=(long long n)
{
return add_fetch(n);
}
long long operator+=(int n)
{
return add_fetch(n);
}
long long operator-=(long long n)
{
return add_fetch(-n);
}
long long operator-=(int n)
{
return add_fetch(-n);
}
private:
long long n_;
};
2017-08-31 17:52:18 +08:00
#include "thread.hpp"
class atomic_long_test
{
2017-08-31 17:52:18 +08:00
private:
atomic_long count_;
public:
atomic_long_test(void) {}
~atomic_long_test(void) {}
void run(void)
{
2017-08-31 17:52:18 +08:00
long long n = count_++;
printf(">>n=%lld\r\n", n);
2017-08-31 17:52:18 +08:00
n = count_;
printf(">>n=%lld\r\n", n);
2017-08-31 17:52:18 +08:00
n = ++count_;
printf(">>n=%lld\r\n", n);
2017-08-31 17:52:18 +08:00
n = --count_;
printf(">>n=%lld\r\n", n);
2017-08-31 17:52:18 +08:00
n = count_--;
printf(">>n=%lld\r\n", n);
2017-08-31 17:52:18 +08:00
n = count_;
printf(">>n=%lld\r\n", n);
2017-08-31 17:52:18 +08:00
count_ -= 1;
n = count_;
printf(">>n=%lld\r\n", n);
2017-08-31 17:52:18 +08:00
printf(">>count > 1 ? %s\r\n", count_ >= 1 ? "yes" : "no");
printf(">>1 > count ? %s\r\n", 1 > count_ ? "yes" : "no");
int i = 1;
2017-08-31 17:52:18 +08:00
count_ = i;
n = count_;
printf(">>n=%lld\r\n", n);
}
2017-08-31 17:52:18 +08:00
static void test(void)
{
class mythread : public thread
{
public:
mythread(atomic_long_test& alt) : alt_(alt) {}
~mythread(void) {}
protected:
void* run(void)
{
for (size_t i = 0; i < 100; i++)
alt_.run();
return NULL;
}
private:
atomic_long_test& alt_;
};
atomic_long_test alt;
mythread thr1(alt), thr2(alt), thr3(alt);
thr1.set_detachable(false);
thr2.set_detachable(false);
thr3.set_detachable(false);
thr1.start();
thr2.start();
thr3.start();
thr1.wait();
thr2.wait();
thr3.wait();
}
};
} // namespace acl