Skip to content

Instantly share code, notes, and snippets.

@13abylon
Created January 22, 2015 12:40
Show Gist options
  • Star 21 You must be signed in to star a gist
  • Fork 9 You must be signed in to fork a gist
  • Save 13abylon/523685ef6af5bde24dc3 to your computer and use it in GitHub Desktop.
Save 13abylon/523685ef6af5bde24dc3 to your computer and use it in GitHub Desktop.
// g++ test.cpp --std=c++11 -lpthread -O2
//#ifdef WIN32 <- stdafx breaks this ifdef...
//#include "stdafx.h"
//#endif
#include <iostream>
#include <atomic>
#include <thread>
#include <vector>
#include <mutex>
#include <time.h>
#include <condition_variable>
#ifndef WIN32
#include <sys/time.h>
#else
#include <WinSock2.h>
int gettimeofday(struct timeval* tv, void* tz) {
union {
int64_t ns100; // since 1.1.1601 in 100ns units
FILETIME ft;
} now;
GetSystemTimeAsFileTime(&now.ft);
tv->tv_usec = (long)((now.ns100 / 10LL) % 1000000LL);
tv->tv_sec = (long)((now.ns100 - 116444736000000000LL) / 10000000LL);
return 0;
}
#endif
////////////////////////////////////////////////////////////////////////////////
/// @brief seconds with microsecond resolution
////////////////////////////////////////////////////////////////////////////////
double TRI_microtime () {
struct timeval t;
gettimeofday(&t, 0);
return (t.tv_sec) + (t.tv_usec / 1000000.0);
}
std::mutex theMutex;
#ifdef WIN32
SRWLOCK srwlock;
#else
pthread_rwlock_t rwlock = PTHREAD_RWLOCK_INITIALIZER;
#endif
std::atomic<uint64_t> atomic_uint;
uint64_t flat_uint = 1;
long count;
std::mutex m;
std::condition_variable halt;
bool ready = false;
std::vector<std::thread> threads;
void start_wait(void) {
std::unique_lock<std::mutex> lk(m);
halt.wait(lk, []{return ready;});
}
void threadfunc_unlocked()
{
start_wait();
// Wait until main() sends data
long i;
for (i = 1; i < count; i++) {
flat_uint++;
}
}
void threadfunc_unlocked_read()
{
start_wait();
// Wait until main() sends data
long i,j;
uint64_t current;
j = 0;
for (i = 1; i <= count; i++) {
current = flat_uint;
if (current > 0) {
j++;
}
else {
std::cout << "WTF?";
}
}
if (j != count) {
std::cout << "Not matched count j=" << j << "count=" << count <<"\n";
}
}
void threadfunc_mutex()
{
start_wait();
long i;
for (i = 1; i <= count; i++) {
theMutex.lock();
flat_uint++;
theMutex.unlock();
}
}
void threadfunc_mutex_read()
{
start_wait();
uint64_t current;
long i, j;
j = 0;
for (i = 1; i <= count; i++) {
theMutex.lock();
current = flat_uint;
theMutex.unlock();
if (current > 0) {
j++;
}
else {
std::cout << "WTF?";
}
}
if (j != count) {
std::cout << "Not matched count j=" << j << "count=" << count <<"\n";
}
}
void threadfunc_rwlock()
{
start_wait();
long i;
for (i= 0; i <= count; i++) {
#ifdef WIN32
AcquireSRWLockExclusive(&srwlock);
flat_uint++;
ReleaseSRWLockExclusive(&srwlock);
#else
pthread_rwlock_wrlock(&rwlock);
flat_uint++;
pthread_rwlock_unlock(&rwlock);
#endif
}
}
void threadfunc_rwlock_read_wr()
{
start_wait();
long i, j;
uint64_t current;
j = 0;
for (i = 1; i <= count; i++) {
#ifdef WIN32
AcquireSRWLockExclusive(&srwlock);
current = flat_uint;
ReleaseSRWLockExclusive(&srwlock);
#else
pthread_rwlock_wrlock(&rwlock);
current = flat_uint;
pthread_rwlock_unlock(&rwlock);
#endif
if (current > 0) {
j++;
}
else {
std::cout << "WTF?";
}
}
if (j != count) {
std::cout << "Not matched count j=" << j << "count=" << count <<"\n";
}
}
void threadfunc_rwlock_read_rd()
{
start_wait();
long i, j;
uint64_t current;
j = 0;
for (i = 1; i <= count; i++) {
#ifdef WIN32
AcquireSRWLockShared(&srwlock);
current = flat_uint;
ReleaseSRWLockShared(&srwlock);
#else
pthread_rwlock_rdlock(&rwlock);
current = flat_uint;
pthread_rwlock_unlock(&rwlock);
#endif
if (current > 0) {
j++;
}
else {
std::cout << "WTF?";
}
}
if (j != count) {
std::cout << "Not matched count j=" << j << "count=" << count <<"\n";
}
}
void threadfunc_atomic()
{
start_wait();
// Wait until main() sends data
long i;
for (i = 1; i <= count; i++) {
atomic_uint++;
}
}
void threadfunc_atomic_read()
{
start_wait();
// Wait until main() sends data
long i,j;
uint64_t current;
j = 0;
for (i = 1; i <= count; i++) {
current = atomic_uint;
if (current > 0) {
j++;
}
else {
std::cout << "WTF?";
}
}
if (j != count) {
std::cout << "Not matched count j=" << j << "count=" << count <<"\n";
}
}
void threadfunc_atomic_set_release()
{
start_wait();
// Wait until main() sends data
long i;
for (i = 1; i <= count; i++) {
atomic_uint.store(i, std::memory_order_release);
}
}
void threadfunc_atomic_set_cst()
{
start_wait();
// Wait until main() sends data
long i;
for (i = 1; i <= count; i++) {
atomic_uint.store(i, std::memory_order_seq_cst);
}
}
void threadfunc_atomic_set_relaxed()
{
start_wait();
// Wait until main() sends data
long i;
for (i = 1; i <= count; i++) {
atomic_uint.store(i, std::memory_order_relaxed);
}
}
void threadfunc_atomic_read_consume()
{
start_wait();
// Wait until main() sends data
long i,j;
uint64_t current;
j = 0;
for (i = 1; i <= count; i++) {
current = atomic_uint.load(std::memory_order_consume);
if (current > 0) {
j++;
}
else {
std::cout << "WTF?" << errno << " " << i << "\n";
}
}
if (j != count) {
std::cout << "Not matched count j=" << j << "count=" << count <<"\n";
}
}
void threadfunc_atomic_read_acquire()
{
start_wait();
// Wait until main() sends data
long i,j;
uint64_t current;
j = 0;
for (i = 1; i <= count; i++) {
current = atomic_uint.load(std::memory_order_acquire);
if (current > 0) {
j++;
}
else {
std::cout << "WTF?";
}
}
if (j != count) {
std::cout << "Not matched count j=" << j << "count=" << count <<"\n";
}
}
void threadfunc_atomic_weak()
{
start_wait();
// Wait until main() sends data
long i;
uint64_t next;
for (i = 1; i <= count; i++) {
do {
next = atomic_uint.load(std::memory_order_relaxed);
}
while (!atomic_uint.compare_exchange_weak(next,
next + 1,
std::memory_order_release,
std::memory_order_relaxed));
}
}
typedef void(*ThreadFunc)(void);
typedef struct __jobs {
ThreadFunc Write;
const char* WriteStr;
ThreadFunc Read;
const char *ReadStr;
}jobs;
#define FN(NAME) threadfunc_##NAME
#define JOB(READFUNC, WRITEFUNC) {FN(READFUNC), #READFUNC, FN(WRITEFUNC), #WRITEFUNC}
int main(int argc, char** argv)
{
if (argc < 5)
return -1;
int numthreads = atoi(argv[1]);
count = atol(argv[2]);
int factor = atoi(argv[3]);
int type = atoi(argv[4]);
int readerCount = 0;
int writerCount = 0;
double ticStart;
jobs Jobs [] = {
JOB(unlocked, unlocked_read),
JOB(mutex, mutex_read),
JOB(rwlock, rwlock_read_wr),
JOB(rwlock, rwlock_read_rd),
JOB(atomic, atomic_read),
JOB(atomic_set_release, atomic_read_consume),
JOB(atomic_set_release, atomic_read_acquire),
JOB(atomic_set_cst, atomic_read_consume),
JOB(atomic_set_cst, atomic_read_acquire),
JOB(atomic_set_relaxed, atomic_read_consume),
JOB(atomic_set_relaxed, atomic_read_acquire),
JOB(atomic_weak, atomic_read_consume),
JOB(atomic_weak, atomic_read_acquire)
};
#ifdef WIN32
InitializeSRWLock(&srwlock);
#endif
atomic_uint = 1;
if (type >= sizeof(Jobs)/sizeof(jobs)) {
std::cout << "type not supported.\n";
exit(1);
}
for (int i = 0; i < numthreads; i++) {
if (i % factor == 0) {
writerCount ++;
threads.push_back(std::thread(Jobs[type].Write));
}
else {
readerCount ++;
threads.push_back(std::thread(Jobs[type].Read));
}
}
ready = true;
ticStart = TRI_microtime();
halt.notify_all();
for (int i = 0; i < numthreads; i++) {
threads[i].join();
}
std::cout
<< "Threads: "
<< numthreads
<<" count: "
<< count
<< " Readers: "
<< readerCount
<< " Writers: "
<< writerCount
<< " atomic_uint is now: "
<< atomic_uint
<< " flat_uint is now: "
<< flat_uint
<< " Test: "
<< Jobs[type].ReadStr
<< " + "
<< Jobs[type].WriteStr
<< " test Duration: "
<< (float) (TRI_microtime() - ticStart)
<< "\n";
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment