/* -*- mode: C++; c-basic-offset: 4; indent-tabs-mode: nil -*- */ // vim: ft=cpp:expandtab:ts=8:sw=4:softtabstop=4: #ident "$Id$" #ident "Copyright (c) 2007-2012 Tokutek Inc. All rights reserved." #ident "The technology is licensed by the Massachusetts Institute of Technology, Rutgers State University of New Jersey, and the Research Foundation of State University of New York at Stony Brook under United States of America Serial No. 11/760379 and to the patents and/or patent applications resulting from it." /* How expensive is * - Obtaining a read-only lock for the first obtainer. * - Obtaining it for the second one? * - The third one? */ #include #include #include #include #include #include float tdiff (struct timeval *start, struct timeval *end) { return 1e6*(end->tv_sec-start->tv_sec) +(end->tv_usec - start->tv_usec); } /* My own rwlock implementation. */ struct brwl { int mutex; int state; // 0 for unlocked, -1 for a writer, otherwise many readers }; static inline int xchg(volatile int *ptr, int x) { __asm__("xchgl %0,%1" :"=r" (x) :"m" (*(ptr)), "0" (x) :"memory"); return x; } static inline void sfence (void) { asm volatile ("sfence":::"memory"); } static inline void brwl_rlock_fence (struct brwl *l) { while (xchg(&l->mutex, 1)) ; l->state++; sfence(); l->mutex=0; } static inline void brwl_rlock_xchg (struct brwl *l) { while (xchg(&l->mutex, 1)) ; l->state++; xchg(&l->mutex, 0); } // Something wrong with the compiler for longs static inline long fetch_and_add (volatile long *p, long incr) { long result = incr; __asm__ __volatile__ ("lock; xaddl %0, %1" : "+r" (result), "+m" (*p) : : "memory"); return result; } static inline int fetch_and_add_i (volatile int *p, int incr) { int result = incr; __asm__ __volatile__ ("lock; xadd %0, %1" : "+r" (result), "+m" (*p) : : "memory"); return result; } static inline int gcc_fetch_and_add_i (volatile int *p, int incr) { return toku_sync_fetch_and_add(p, incr); } static inline long gcc_fetch_and_add_l (volatile long *p, long incr) { return toku_sync_fetch_and_add(p, incr); } // Something wrong with the compiler for longs /* Returns nonzero if the comparison succeeded. */ static inline long compare_and_swap_full(volatile long *addr, long old, long new_val) { char result; __asm__ __volatile__("lock; cmpxchgl %2, %0; setz %1" : "+m"(*(addr)), "=q"(result) : "r" (new_val), "a"(old) : "memory"); return (int) result; } /* Returns nonzero if the comparison succeeded. */ // Atomically compare *addr to old_val, and replace *addr by new_val // if the first comparison succeeds. Returns nonzero if the comparison // succeeded and *addr was updated. static inline int compare_and_swap_full_i(volatile int *addr, int old, int new_val) { char result; __asm__ __volatile__("lock; cmpxchg %2, %0; setz %1" : "+m"(*(addr)), "=q"(result) : "r" (new_val), "a"(old) : "memory"); return (int) result; } enum {K=100000}; pthread_rwlock_t rwlocks[K]; struct brwl blocks[K]; pthread_mutex_t mlocks[K]; long lvals[K]; int ivals[K]; #define TIME(s, i, init, body) ({ \ int j_tmp; \ printf("%-24s", s); \ for (j_tmp=0; j_tmp<3; j_tmp++) { \ struct timeval start,end; \ int i; \ for (i=0; i