Commit fc1c339a authored by unknown's avatar unknown

lock manager passed unit tests


storage/maria/trnman.c:
  comments
include/my_dbug.h:
  make DBUG_ASSERT always a statement
storage/maria/lockman.h:
  comments
include/lf.h:
  lf_pinbox - don't use a fixed-size purgatory.
mysys/lf_alloc-pin.c:
  lf_pinbox - don't use a fixed-size purgatory.
mysys/lf_hash.c:
  lf_pinbox - don't use a fixed-size purgatory.
storage/maria/lockman.c:
  removed IGNORE_ME/UPGDARED matching - it was wrong in the first place.
  updated for "lf_pinbox - don't use a fixed-size purgatory"
storage/maria/unittest/lockman-t.c:
  IGNORE_ME/UPGRADED pair counting bugtest.
  more tests
unittest/mysys/my_atomic-t.c:
  lf_pinbox - don't use a fixed-size purgatory.
parent a1409a06
...@@ -96,20 +96,21 @@ typedef void lf_pinbox_free_func(void *, void *); ...@@ -96,20 +96,21 @@ typedef void lf_pinbox_free_func(void *, void *);
typedef struct { typedef struct {
LF_DYNARRAY pinstack; LF_DYNARRAY pinstack;
lf_pinbox_free_func *free_func; lf_pinbox_free_func *free_func;
void * free_func_arg; void *free_func_arg;
uint free_ptr_offset;
uint32 volatile pinstack_top_ver; /* this is a versioned pointer */ uint32 volatile pinstack_top_ver; /* this is a versioned pointer */
uint32 volatile pins_in_stack; /* number of elements in array */ uint32 volatile pins_in_stack; /* number of elements in array */
} LF_PINBOX; } LF_PINBOX;
/* we want sizeof(LF_PINS) to be close to 128 to avoid false sharing */ /* we want sizeof(LF_PINS) to be 128 to avoid false sharing */
typedef struct { typedef struct {
void * volatile pin[LF_PINBOX_PINS]; void * volatile pin[LF_PINBOX_PINS];
void * purgatory[LF_PURGATORY_SIZE];
LF_PINBOX *pinbox; LF_PINBOX *pinbox;
void *purgatory;
uint32 purgatory_count; uint32 purgatory_count;
uint32 volatile link; uint32 volatile link;
char pad[128-sizeof(uint32)*2 char pad[128-sizeof(uint32)*2
-sizeof(void *)*(LF_PINBOX_PINS+LF_PURGATORY_SIZE+1)]; -sizeof(void *)*(LF_PINBOX_PINS+2)];
} LF_PINS; } LF_PINS;
#define lf_rwlock_by_pins(PINS) \ #define lf_rwlock_by_pins(PINS) \
...@@ -147,8 +148,8 @@ typedef struct { ...@@ -147,8 +148,8 @@ typedef struct {
#define _lf_assert_pin(PINS, PIN) assert((PINS)->pin[PIN] != 0) #define _lf_assert_pin(PINS, PIN) assert((PINS)->pin[PIN] != 0)
#define _lf_assert_unpin(PINS, PIN) assert((PINS)->pin[PIN]==0) #define _lf_assert_unpin(PINS, PIN) assert((PINS)->pin[PIN]==0)
void lf_pinbox_init(LF_PINBOX *pinbox, lf_pinbox_free_func *free_func, void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset,
void * free_func_arg); lf_pinbox_free_func *free_func, void * free_func_arg);
void lf_pinbox_destroy(LF_PINBOX *pinbox); void lf_pinbox_destroy(LF_PINBOX *pinbox);
lock_wrap(lf_pinbox_get_pins, LF_PINS *, lock_wrap(lf_pinbox_get_pins, LF_PINS *,
...@@ -181,7 +182,7 @@ typedef struct st_lf_allocator { ...@@ -181,7 +182,7 @@ typedef struct st_lf_allocator {
uint32 volatile mallocs; uint32 volatile mallocs;
} LF_ALLOCATOR; } LF_ALLOCATOR;
void lf_alloc_init(LF_ALLOCATOR *allocator, uint size); void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset);
void lf_alloc_destroy(LF_ALLOCATOR *allocator); void lf_alloc_destroy(LF_ALLOCATOR *allocator);
uint lf_alloc_in_pool(LF_ALLOCATOR *allocator); uint lf_alloc_in_pool(LF_ALLOCATOR *allocator);
#define _lf_alloc_free(PINS, PTR) _lf_pinbox_free((PINS), (PTR)) #define _lf_alloc_free(PINS, PTR) _lf_pinbox_free((PINS), (PTR))
......
...@@ -100,7 +100,7 @@ extern FILE *_db_fp_(void); ...@@ -100,7 +100,7 @@ extern FILE *_db_fp_(void);
#define DBUG_LONGJMP(a1) longjmp(a1) #define DBUG_LONGJMP(a1) longjmp(a1)
#define DBUG_DUMP(keyword,a1,a2) #define DBUG_DUMP(keyword,a1,a2)
#define DBUG_END() #define DBUG_END()
#define DBUG_ASSERT(A) #define DBUG_ASSERT(A) do { } while(0)
#define DBUG_LOCK_FILE #define DBUG_LOCK_FILE
#define DBUG_FILE (stderr) #define DBUG_FILE (stderr)
#define DBUG_UNLOCK_FILE #define DBUG_UNLOCK_FILE
......
...@@ -15,21 +15,7 @@ ...@@ -15,21 +15,7 @@
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
/* /*
concurrent allocator based on pinning addresses wait-free concurrent allocator based on pinning addresses
strictly speaking it's not lock-free, as it can be blocked
if a thread's purgatory is full and all addresses from there
are pinned.
But until the above happens, it's wait-free.
It can be made strictly wait-free by increasing purgatory size.
If it's larger than pins_in_stack*LF_PINBOX_PINS, then apocalyptical
condition above will never happen. But than the memory requirements
will be O(pins_in_stack^2).
Note, that for large purgatory sizes it makes sense to remove
purgatory array, and link objects in a list using embedded pointer.
TODO test with more than 256 threads TODO test with more than 256 threads
TODO test w/o alloca TODO test w/o alloca
...@@ -43,15 +29,17 @@ ...@@ -43,15 +29,17 @@
static void _lf_pinbox_real_free(LF_PINS *pins); static void _lf_pinbox_real_free(LF_PINS *pins);
void lf_pinbox_init(LF_PINBOX *pinbox, lf_pinbox_free_func *free_func, void lf_pinbox_init(LF_PINBOX *pinbox, uint free_ptr_offset,
void *free_func_arg) lf_pinbox_free_func *free_func,void *free_func_arg)
{ {
DBUG_ASSERT(sizeof(LF_PINS) == 128); DBUG_ASSERT(sizeof(LF_PINS) == 128);
DBUG_ASSERT(free_ptr_offset % sizeof(void *) == 0);
lf_dynarray_init(&pinbox->pinstack, sizeof(LF_PINS)); lf_dynarray_init(&pinbox->pinstack, sizeof(LF_PINS));
pinbox->pinstack_top_ver=0; pinbox->pinstack_top_ver= 0;
pinbox->pins_in_stack=0; pinbox->pins_in_stack= 0;
pinbox->free_func=free_func; pinbox->free_ptr_offset= free_ptr_offset;
pinbox->free_func_arg=free_func_arg; pinbox->free_func= free_func;
pinbox->free_func_arg= free_func_arg;
} }
void lf_pinbox_destroy(LF_PINBOX *pinbox) void lf_pinbox_destroy(LF_PINBOX *pinbox)
...@@ -64,58 +52,64 @@ LF_PINS *_lf_pinbox_get_pins(LF_PINBOX *pinbox) ...@@ -64,58 +52,64 @@ LF_PINS *_lf_pinbox_get_pins(LF_PINBOX *pinbox)
uint32 pins, next, top_ver; uint32 pins, next, top_ver;
LF_PINS *el; LF_PINS *el;
top_ver=pinbox->pinstack_top_ver; top_ver= pinbox->pinstack_top_ver;
do do
{ {
if (!(pins=top_ver % LF_PINBOX_MAX_PINS)) if (!(pins= top_ver % LF_PINBOX_MAX_PINS))
{ {
pins=my_atomic_add32(&pinbox->pins_in_stack, 1)+1; pins= my_atomic_add32(&pinbox->pins_in_stack, 1)+1;
el=(LF_PINS *)_lf_dynarray_lvalue(&pinbox->pinstack, pins); el= (LF_PINS *)_lf_dynarray_lvalue(&pinbox->pinstack, pins);
break; break;
} }
el=(LF_PINS *)_lf_dynarray_value(&pinbox->pinstack, pins); el= (LF_PINS *)_lf_dynarray_value(&pinbox->pinstack, pins);
next=el->link; next= el->link;
} while (!my_atomic_cas32(&pinbox->pinstack_top_ver, &top_ver, } while (!my_atomic_cas32(&pinbox->pinstack_top_ver, &top_ver,
top_ver-pins+next+LF_PINBOX_MAX_PINS)); top_ver-pins+next+LF_PINBOX_MAX_PINS));
el->link=pins; el->link= pins;
el->purgatory_count=0; el->purgatory_count= 0;
el->pinbox=pinbox; el->pinbox= pinbox;
return el; return el;
} }
void _lf_pinbox_put_pins(LF_PINS *pins) void _lf_pinbox_put_pins(LF_PINS *pins)
{ {
LF_PINBOX *pinbox=pins->pinbox; LF_PINBOX *pinbox= pins->pinbox;
uint32 top_ver, nr; uint32 top_ver, nr;
nr=pins->link; nr= pins->link;
#ifdef MY_LF_EXTRA_DEBUG #ifdef MY_LF_EXTRA_DEBUG
{ {
int i; int i;
for (i=0; i < LF_PINBOX_PINS; i++) for (i= 0; i < LF_PINBOX_PINS; i++)
assert(pins->pin[i] == 0); assert(pins->pin[i] == 0);
} }
#endif #endif
/*
Note - this will deadlock if other threads will wait for
the caller to do something after _lf_pinbox_put_pins(),
and they would have pinned addresses that the caller wants to free.
Thus: only free pins when all work is done and nobody can wait for you!!!
*/
while (pins->purgatory_count) while (pins->purgatory_count)
{ {
_lf_pinbox_real_free(pins); _lf_pinbox_real_free(pins);
if (pins->purgatory_count && my_getncpus() == 1) if (pins->purgatory_count)
{ {
my_atomic_rwlock_wrunlock(&pins->pinbox->pinstack.lock); my_atomic_rwlock_wrunlock(&pins->pinbox->pinstack.lock);
pthread_yield(); pthread_yield();
my_atomic_rwlock_wrlock(&pins->pinbox->pinstack.lock); my_atomic_rwlock_wrlock(&pins->pinbox->pinstack.lock);
} }
} }
top_ver=pinbox->pinstack_top_ver; top_ver= pinbox->pinstack_top_ver;
if (nr == pinbox->pins_in_stack) if (nr == pinbox->pins_in_stack)
{ {
int32 tmp=nr; int32 tmp= nr;
if (my_atomic_cas32(&pinbox->pins_in_stack, &tmp, tmp-1)) if (my_atomic_cas32(&pinbox->pins_in_stack, &tmp, tmp-1))
goto ret; goto ret;
} }
do do
{ {
pins->link=top_ver % LF_PINBOX_MAX_PINS; pins->link= top_ver % LF_PINBOX_MAX_PINS;
} while (!my_atomic_cas32(&pinbox->pinstack_top_ver, &top_ver, } while (!my_atomic_cas32(&pinbox->pinstack_top_ver, &top_ver,
top_ver-pins->link+nr+LF_PINBOX_MAX_PINS)); top_ver-pins->link+nr+LF_PINBOX_MAX_PINS));
ret: ret:
...@@ -127,19 +121,20 @@ static int ptr_cmp(void **a, void **b) ...@@ -127,19 +121,20 @@ static int ptr_cmp(void **a, void **b)
return *a < *b ? -1 : *a == *b ? 0 : 1; return *a < *b ? -1 : *a == *b ? 0 : 1;
} }
#define add_to_purgatory(PINS, ADDR) \
do \
{ \
*(void **)((char *)(ADDR)+(PINS)->pinbox->free_ptr_offset)= \
(PINS)->purgatory; \
(PINS)->purgatory= (ADDR); \
(PINS)->purgatory_count++; \
} while (0)
void _lf_pinbox_free(LF_PINS *pins, void *addr) void _lf_pinbox_free(LF_PINS *pins, void *addr)
{ {
while (pins->purgatory_count == LF_PURGATORY_SIZE) if (pins->purgatory_count % LF_PURGATORY_SIZE)
{
_lf_pinbox_real_free(pins); _lf_pinbox_real_free(pins);
if (pins->purgatory_count == LF_PURGATORY_SIZE && my_getncpus() == 1) add_to_purgatory(pins, addr);
{
my_atomic_rwlock_wrunlock(&pins->pinbox->pinstack.lock);
pthread_yield();
my_atomic_rwlock_wrlock(&pins->pinbox->pinstack.lock);
}
}
pins->purgatory[pins->purgatory_count++]=addr;
} }
struct st_harvester { struct st_harvester {
...@@ -178,11 +173,11 @@ static int match_pins(LF_PINS *el, void *addr) ...@@ -178,11 +173,11 @@ static int match_pins(LF_PINS *el, void *addr)
static void _lf_pinbox_real_free(LF_PINS *pins) static void _lf_pinbox_real_free(LF_PINS *pins)
{ {
int npins; int npins;
void **addr=0; void *list;
void **start, **cur, **end=pins->purgatory+pins->purgatory_count; void **addr;
LF_PINBOX *pinbox=pins->pinbox; LF_PINBOX *pinbox= pins->pinbox;
npins=pinbox->pins_in_stack+1; npins= pinbox->pins_in_stack+1;
#ifdef HAVE_ALLOCA #ifdef HAVE_ALLOCA
/* create a sorted list of pinned addresses, to speed up searches */ /* create a sorted list of pinned addresses, to speed up searches */
...@@ -190,64 +185,64 @@ static void _lf_pinbox_real_free(LF_PINS *pins) ...@@ -190,64 +185,64 @@ static void _lf_pinbox_real_free(LF_PINS *pins)
{ {
struct st_harvester hv; struct st_harvester hv;
addr= (void **) alloca(sizeof(void *)*LF_PINBOX_PINS*npins); addr= (void **) alloca(sizeof(void *)*LF_PINBOX_PINS*npins);
hv.granary=addr; hv.granary= addr;
hv.npins=npins; hv.npins= npins;
_lf_dynarray_iterate(&pinbox->pinstack, _lf_dynarray_iterate(&pinbox->pinstack,
(lf_dynarray_func)harvest_pins, &hv); (lf_dynarray_func)harvest_pins, &hv);
npins=hv.granary-addr; npins= hv.granary-addr;
if (npins) if (npins)
qsort(addr, npins, sizeof(void *), (qsort_cmp)ptr_cmp); qsort(addr, npins, sizeof(void *), (qsort_cmp)ptr_cmp);
} }
else
#endif #endif
addr= 0;
start= cur= pins->purgatory; list= pins->purgatory;
end= start+pins->purgatory_count; pins->purgatory= 0;
for (; cur < end; cur++) pins->purgatory_count= 0;
while (list)
{ {
void *cur= list;
list= *(void **)((char *)cur+pinbox->free_ptr_offset);
if (npins) if (npins)
{ {
if (addr) if (addr)
{ {
void **a,**b,**c; void **a,**b,**c;
for (a=addr, b=addr+npins-1, c=a+(b-a)/2; b-a>1; c=a+(b-a)/2) for (a= addr, b= addr+npins-1, c= a+(b-a)/2; b-a>1; c= a+(b-a)/2)
if (*cur == *c) if (cur == *c)
a=b=c; a= b= c;
else if (*cur > *c) else if (cur > *c)
a=c; a= c;
else else
b=c; b= c;
if (*cur == *a || *cur == *b) if (cur == *a || cur == *b)
goto found; goto found;
} }
else else
{ {
if (_lf_dynarray_iterate(&pinbox->pinstack, if (_lf_dynarray_iterate(&pinbox->pinstack,
(lf_dynarray_func)match_pins, *cur)) (lf_dynarray_func)match_pins, cur))
goto found; goto found;
} }
} }
/* not pinned - freeing */ /* not pinned - freeing */
pinbox->free_func(*cur, pinbox->free_func_arg); pinbox->free_func(cur, pinbox->free_func_arg);
continue; continue;
found: found:
/* pinned - keeping */ /* pinned - keeping */
*start++=*cur; add_to_purgatory(pins, cur);
} }
pins->purgatory_count=start-pins->purgatory;
#ifdef MY_LF_EXTRA_DEBUG
while (start < pins->purgatory + LF_PURGATORY_SIZE)
*start++=0;
#endif
} }
static void alloc_free(void *node, LF_ALLOCATOR *allocator) static void alloc_free(void *node, LF_ALLOCATOR *allocator)
{ {
void *tmp; void *tmp;
tmp=allocator->top; tmp= allocator->top;
do do
{ {
(*(void **)node)=tmp; (*(void **)node)= tmp;
} while (!my_atomic_casptr((void **)&allocator->top, (void **)&tmp, node) && } while (!my_atomic_casptr((void **)&allocator->top, (void **)&tmp, node) &&
LF_BACKOFF); LF_BACKOFF);
} }
...@@ -255,18 +250,18 @@ static void alloc_free(void *node, LF_ALLOCATOR *allocator) ...@@ -255,18 +250,18 @@ static void alloc_free(void *node, LF_ALLOCATOR *allocator)
LF_REQUIRE_PINS(1); LF_REQUIRE_PINS(1);
void *_lf_alloc_new(LF_PINS *pins) void *_lf_alloc_new(LF_PINS *pins)
{ {
LF_ALLOCATOR *allocator=(LF_ALLOCATOR *)(pins->pinbox->free_func_arg); LF_ALLOCATOR *allocator= (LF_ALLOCATOR *)(pins->pinbox->free_func_arg);
void *node; void *node;
for (;;) for (;;)
{ {
do do
{ {
node=allocator->top; node= allocator->top;
_lf_pin(pins, 0, node); _lf_pin(pins, 0, node);
} while (node !=allocator->top && LF_BACKOFF); } while (node != allocator->top && LF_BACKOFF);
if (!node) if (!node)
{ {
if (!(node=my_malloc(allocator->element_size, MYF(MY_WME|MY_ZEROFILL)))) if (!(node= my_malloc(allocator->element_size, MYF(MY_WME|MY_ZEROFILL))))
goto ret; goto ret;
#ifdef MY_LF_EXTRA_DEBUG #ifdef MY_LF_EXTRA_DEBUG
my_atomic_add32(&allocator->mallocs, 1); my_atomic_add32(&allocator->mallocs, 1);
...@@ -282,27 +277,27 @@ void *_lf_alloc_new(LF_PINS *pins) ...@@ -282,27 +277,27 @@ void *_lf_alloc_new(LF_PINS *pins)
return node; return node;
} }
void lf_alloc_init(LF_ALLOCATOR *allocator, uint size) void lf_alloc_init(LF_ALLOCATOR *allocator, uint size, uint free_ptr_offset)
{ {
lf_pinbox_init(&allocator->pinbox, lf_pinbox_init(&allocator->pinbox, free_ptr_offset,
(lf_pinbox_free_func *)alloc_free, allocator); (lf_pinbox_free_func *)alloc_free, allocator);
allocator->top=0; allocator->top= 0;
allocator->mallocs=0; allocator->mallocs= 0;
allocator->element_size=size; allocator->element_size= size;
DBUG_ASSERT(size >= (int)sizeof(void *)); DBUG_ASSERT(size >= (int)sizeof(void *));
} }
void lf_alloc_destroy(LF_ALLOCATOR *allocator) void lf_alloc_destroy(LF_ALLOCATOR *allocator)
{ {
void *el=allocator->top; void *el= allocator->top;
while (el) while (el)
{ {
void *tmp=*(void **)el; void *tmp= *(void **)el;
my_free(el, MYF(0)); my_free(el, MYF(0));
el=tmp; el= tmp;
} }
lf_pinbox_destroy(&allocator->pinbox); lf_pinbox_destroy(&allocator->pinbox);
allocator->top=0; allocator->top= 0;
} }
/* /*
...@@ -313,7 +308,8 @@ uint lf_alloc_in_pool(LF_ALLOCATOR *allocator) ...@@ -313,7 +308,8 @@ uint lf_alloc_in_pool(LF_ALLOCATOR *allocator)
{ {
uint i; uint i;
void *node; void *node;
for (node=allocator->top, i=0; node; node=*(void **)node, i++) /* no op */; for (node= allocator->top, i= 0; node; node= *(void **)node, i++)
/* no op */;
return i; return i;
} }
...@@ -20,6 +20,7 @@ ...@@ -20,6 +20,7 @@
TODO TODO
try to get rid of dummy nodes ? try to get rid of dummy nodes ?
for non-unique hash, count only _distinct_ values for non-unique hash, count only _distinct_ values
(but how to do it in lf_hash_delete ?)
*/ */
#include <my_global.h> #include <my_global.h>
#include <my_sys.h> #include <my_sys.h>
...@@ -222,7 +223,8 @@ void lf_hash_init(LF_HASH *hash, uint element_size, uint flags, ...@@ -222,7 +223,8 @@ void lf_hash_init(LF_HASH *hash, uint element_size, uint flags,
uint key_offset, uint key_length, hash_get_key get_key, uint key_offset, uint key_length, hash_get_key get_key,
CHARSET_INFO *charset) CHARSET_INFO *charset)
{ {
lf_alloc_init(&hash->alloc,sizeof(LF_SLIST)+element_size); lf_alloc_init(&hash->alloc, sizeof(LF_SLIST)+element_size,
offsetof(LF_SLIST, key));
lf_dynarray_init(&hash->array, sizeof(LF_SLIST **)); lf_dynarray_init(&hash->array, sizeof(LF_SLIST **));
hash->size=1; hash->size=1;
hash->count=0; hash->count=0;
......
This diff is collapsed.
...@@ -18,7 +18,10 @@ ...@@ -18,7 +18,10 @@
#define _lockman_h #define _lockman_h
/* /*
N - "no lock", not a lock, used sometimes to simplify the code Lock levels:
^^^^^^^^^^^
N - "no lock", not a lock, used sometimes internally to simplify the code
S - Shared S - Shared
X - eXclusive X - eXclusive
IS - Intention Shared IS - Intention Shared
...@@ -35,8 +38,8 @@ struct lockman_lock; ...@@ -35,8 +38,8 @@ struct lockman_lock;
typedef struct st_lock_owner LOCK_OWNER; typedef struct st_lock_owner LOCK_OWNER;
struct st_lock_owner { struct st_lock_owner {
LF_PINS *pins; LF_PINS *pins; /* must be allocated from lockman's pinbox */
struct lockman_lock *all_locks; struct lockman_lock *all_locks; /* a LIFO */
LOCK_OWNER *waiting_for; LOCK_OWNER *waiting_for;
pthread_cond_t *cond; /* transactions waiting for this, wait on 'cond' */ pthread_cond_t *cond; /* transactions waiting for this, wait on 'cond' */
pthread_mutex_t *mutex; /* mutex is required to use 'cond' */ pthread_mutex_t *mutex; /* mutex is required to use 'cond' */
......
...@@ -128,6 +128,11 @@ static void set_short_trid(TRN *trn) ...@@ -128,6 +128,11 @@ static void set_short_trid(TRN *trn)
trn->locks.loid= i; trn->locks.loid= i;
} }
/*
DESCRIPTION
start a new transaction, allocate and initialize transaction object
mutex and cond will be used for lock waits
*/
TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond)
{ {
TRN *trn; TRN *trn;
...@@ -148,6 +153,7 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond) ...@@ -148,6 +153,7 @@ TRN *trnman_new_trn(pthread_mutex_t *mutex, pthread_cond_t *cond)
trnman_active_transactions++; trnman_active_transactions++;
trn= pool; trn= pool;
/* popping an element from a stack */
my_atomic_rwlock_wrlock(&LOCK_pool); my_atomic_rwlock_wrlock(&LOCK_pool);
while (trn && !my_atomic_casptr((void **)&pool, (void **)&trn, while (trn && !my_atomic_casptr((void **)&pool, (void **)&trn,
(void *)trn->next)) (void *)trn->next))
...@@ -213,9 +219,12 @@ void trnman_end_trn(TRN *trn, my_bool commit) ...@@ -213,9 +219,12 @@ void trnman_end_trn(TRN *trn, my_bool commit)
LF_PINS *pins= trn->pins; LF_PINS *pins= trn->pins;
pthread_mutex_lock(&LOCK_trn_list); pthread_mutex_lock(&LOCK_trn_list);
/* remove from active list */
trn->next->prev= trn->prev; trn->next->prev= trn->prev;
trn->prev->next= trn->next; trn->prev->next= trn->next;
/* if this transaction was the oldest - clean up committed list */
if (trn->prev == &active_list_min) if (trn->prev == &active_list_min)
{ {
TRN *t; TRN *t;
...@@ -232,6 +241,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) ...@@ -232,6 +241,7 @@ void trnman_end_trn(TRN *trn, my_bool commit)
} }
} }
/* add transaction to the committed list (for read-from relations) */
if (commit && active_list_min.next != &active_list_max) if (commit && active_list_min.next != &active_list_max)
{ {
trn->commit_trid= global_trid_generator; trn->commit_trid= global_trid_generator;
...@@ -243,7 +253,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) ...@@ -243,7 +253,7 @@ void trnman_end_trn(TRN *trn, my_bool commit)
res= lf_hash_insert(&trid_to_trn, pins, &trn); res= lf_hash_insert(&trid_to_trn, pins, &trn);
DBUG_ASSERT(res == 0); DBUG_ASSERT(res == 0);
} }
else else /* or free it right away */
{ {
trn->next= free_me; trn->next= free_me;
free_me= trn; free_me= trn;
...@@ -251,6 +261,7 @@ void trnman_end_trn(TRN *trn, my_bool commit) ...@@ -251,6 +261,7 @@ void trnman_end_trn(TRN *trn, my_bool commit)
trnman_active_transactions--; trnman_active_transactions--;
pthread_mutex_unlock(&LOCK_trn_list); pthread_mutex_unlock(&LOCK_trn_list);
/* the rest is done outside of a critical section */
lockman_release_locks(&maria_lockman, &trn->locks); lockman_release_locks(&maria_lockman, &trn->locks);
trn->locks.mutex= 0; trn->locks.mutex= 0;
trn->locks.cond= 0; trn->locks.cond= 0;
...@@ -258,7 +269,6 @@ void trnman_end_trn(TRN *trn, my_bool commit) ...@@ -258,7 +269,6 @@ void trnman_end_trn(TRN *trn, my_bool commit)
my_atomic_storeptr((void **)&short_trid_to_trn[trn->locks.loid], 0); my_atomic_storeptr((void **)&short_trid_to_trn[trn->locks.loid], 0);
my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn); my_atomic_rwlock_rdunlock(&LOCK_short_trid_to_trn);
while (free_me) // XXX send them to the purge thread while (free_me) // XXX send them to the purge thread
{ {
int res; int res;
...@@ -288,7 +298,7 @@ void trnman_free_trn(TRN *trn) ...@@ -288,7 +298,7 @@ void trnman_free_trn(TRN *trn)
do do
{ {
/* /*
without volatile cast gcc-3.4.4 moved the assignment without this volatile cast gcc-3.4.4 moved the assignment
down after the loop at -O2 down after the loop at -O2
*/ */
*(TRN * volatile *)&(trn->next)= tmp; *(TRN * volatile *)&(trn->next)= tmp;
...@@ -317,13 +327,13 @@ my_bool trnman_can_read_from(TRN *trn, TrID trid) ...@@ -317,13 +327,13 @@ my_bool trnman_can_read_from(TRN *trn, TrID trid)
LF_REQUIRE_PINS(3); LF_REQUIRE_PINS(3);
if (trid < trn->min_read_from) if (trid < trn->min_read_from)
return TRUE; return TRUE; /* can read */
if (trid > trn->trid) if (trid > trn->trid)
return FALSE; return FALSE; /* cannot read */
found= lf_hash_search(&trid_to_trn, trn->pins, &trid, sizeof(trid)); found= lf_hash_search(&trid_to_trn, trn->pins, &trid, sizeof(trid));
if (!found) if (!found)
return FALSE; /* not in the hash of committed transactions = cannot read*/ return FALSE; /* not in the hash of committed transactions = cannot read */
can= (*found)->commit_trid < trn->trid; can= (*found)->commit_trid < trn->trid;
lf_unpin(trn->pins, 2); lf_unpin(trn->pins, 2);
......
...@@ -14,7 +14,7 @@ ...@@ -14,7 +14,7 @@
along with this program; if not, write to the Free Software along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */
//#define EXTRA_VERBOSE #undef EXTRA_VERBOSE
#include <tap.h> #include <tap.h>
...@@ -24,7 +24,7 @@ ...@@ -24,7 +24,7 @@
#include <lf.h> #include <lf.h>
#include "../lockman.h" #include "../lockman.h"
#define Nlos 10 #define Nlos 100
LOCK_OWNER loarray[Nlos]; LOCK_OWNER loarray[Nlos];
pthread_mutex_t mutexes[Nlos]; pthread_mutex_t mutexes[Nlos];
pthread_cond_t conds[Nlos]; pthread_cond_t conds[Nlos];
...@@ -51,8 +51,7 @@ LOCK_OWNER *loid2lo(uint16 loid) ...@@ -51,8 +51,7 @@ LOCK_OWNER *loid2lo(uint16 loid)
#define lock_ok_a(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK) #define lock_ok_a(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK)
#define lock_ok_i(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE) #define lock_ok_i(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE)
#define lock_ok_l(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE) #define lock_ok_l(O,R,L) test_lock(O,R,L,"",GOT_THE_LOCK_NEED_TO_INSTANT_LOCK_A_SUBRESOURCE)
#define lock_conflict(O,R,L) test_lock(O,R,L,"cannot ",DIDNT_GET_THE_LOCK); \ #define lock_conflict(O,R,L) test_lock(O,R,L,"cannot ",DIDNT_GET_THE_LOCK);
unlock_all(O)
void test_lockman_simple() void test_lockman_simple()
{ {
...@@ -64,7 +63,8 @@ void test_lockman_simple() ...@@ -64,7 +63,8 @@ void test_lockman_simple()
lock_ok_a(1, 1, X); lock_ok_a(1, 1, X);
lock_ok_i(2, 2, IX); lock_ok_i(2, 2, IX);
/* failures */ /* failures */
lock_conflict(2,1,X); /* this removes all locks of lo2 */ lock_conflict(2,1,X);
unlock_all(2);
lock_ok_a(1,2,S); lock_ok_a(1,2,S);
lock_ok_a(1,2,IS); lock_ok_a(1,2,IS);
lock_ok_a(1,2,LS); lock_ok_a(1,2,LS);
...@@ -72,8 +72,36 @@ void test_lockman_simple() ...@@ -72,8 +72,36 @@ void test_lockman_simple()
lock_ok_a(2,3,LS); lock_ok_a(2,3,LS);
lock_ok_i(1,3,IX); lock_ok_i(1,3,IX);
lock_ok_l(2,3,IS); lock_ok_l(2,3,IS);
lockman_release_locks(&lockman, loid2lo(1)); unlock_all(1);
lockman_release_locks(&lockman, loid2lo(2)); unlock_all(2);
lock_ok_i(1,1,IX);
lock_conflict(2,1,S);
lock_ok_a(1,1,LS);
unlock_all(1);
unlock_all(2);
lock_ok_i(1,1,IX);
lock_ok_a(2,1,LS);
lock_ok_a(1,1,LS);
lock_ok_i(1,1,IX);
lock_ok_i(3,1,IS);
unlock_all(1);
unlock_all(2);
unlock_all(3);
lock_ok_i(1,4,IS);
lock_ok_i(2,4,IS);
lock_ok_i(3,4,IS);
lock_ok_a(3,4,LS);
lock_ok_i(4,4,IS);
lock_conflict(4,4,IX);
lock_conflict(2,4,IX);
lock_ok_a(1,4,LS);
unlock_all(1);
unlock_all(2);
unlock_all(3);
unlock_all(4);
} }
...@@ -82,11 +110,13 @@ pthread_mutex_t rt_mutex; ...@@ -82,11 +110,13 @@ pthread_mutex_t rt_mutex;
pthread_cond_t rt_cond; pthread_cond_t rt_cond;
int rt_num_threads; int rt_num_threads;
int litmus; int litmus;
int thread_number= 0, timeouts=0;
void run_test(const char *test, pthread_handler handler, int n, int m) void run_test(const char *test, pthread_handler handler, int n, int m)
{ {
pthread_t t; pthread_t t;
ulonglong now= my_getsystime(); ulonglong now= my_getsystime();
thread_number= timeouts= 0;
litmus= 0; litmus= 0;
diag("Testing %s with %d threads, %d iterations... ", test, n, m); diag("Testing %s with %d threads, %d iterations... ", test, n, m);
...@@ -100,13 +130,12 @@ void run_test(const char *test, pthread_handler handler, int n, int m) ...@@ -100,13 +130,12 @@ void run_test(const char *test, pthread_handler handler, int n, int m)
ok(litmus == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus); ok(litmus == 0, "tested %s in %g secs (%d)", test, ((double)now)/1e7, litmus);
} }
int thread_number= 0, timeouts=0; int Nrows= 100;
#define Nrows 1000 int Ntables= 10;
#define Ntables 10 int table_lock_ratio= 10;
#define TABLE_LOCK_RATIO 10
enum lock_type lock_array[6]={S,X,LS,LX,IS,IX}; enum lock_type lock_array[6]={S,X,LS,LX,IS,IX};
char *lock2str[6]={"S","X","LS","LX","IS","IX"}; char *lock2str[6]={"S","X","LS","LX","IS","IX"};
char *res2str[6]={ char *res2str[4]={
"DIDN'T GET THE LOCK", "DIDN'T GET THE LOCK",
"GOT THE LOCK", "GOT THE LOCK",
"GOT THE LOCK NEED TO LOCK A SUBRESOURCE", "GOT THE LOCK NEED TO LOCK A SUBRESOURCE",
...@@ -128,10 +157,11 @@ pthread_handler_t test_lockman(void *arg) ...@@ -128,10 +157,11 @@ pthread_handler_t test_lockman(void *arg)
row= x % Nrows + Ntables; row= x % Nrows + Ntables;
table= row % Ntables; table= row % Ntables;
locklevel= (x/Nrows) & 3; locklevel= (x/Nrows) & 3;
if ((x/Nrows/4) % TABLE_LOCK_RATIO == 0) if (table_lock_ratio && (x/Nrows/4) % table_lock_ratio == 0)
{ /* table lock */ { /* table lock */
res= lockman_getlock(&lockman, lo, table, lock_array[locklevel]); res= lockman_getlock(&lockman, lo, table, lock_array[locklevel]);
DIAG(("loid=%2d, table %d lock %s, res=%s", loid, table, lock2str[locklevel], res2str[res])); DIAG(("loid=%2d, table %d lock %s, res=%s", loid, table,
lock2str[locklevel], res2str[res]));
if (res == DIDNT_GET_THE_LOCK) if (res == DIDNT_GET_THE_LOCK)
{ {
lockman_release_locks(&lockman, lo); lockman_release_locks(&lockman, lo);
...@@ -145,7 +175,8 @@ pthread_handler_t test_lockman(void *arg) ...@@ -145,7 +175,8 @@ pthread_handler_t test_lockman(void *arg)
{ /* row lock */ { /* row lock */
locklevel&= 1; locklevel&= 1;
res= lockman_getlock(&lockman, lo, table, lock_array[locklevel + 4]); res= lockman_getlock(&lockman, lo, table, lock_array[locklevel + 4]);
DIAG(("loid=%2d, row %d lock %s, res=%s", loid, row, lock2str[locklevel+4], res2str[res])); DIAG(("loid=%2d, row %d lock %s, res=%s", loid, row,
lock2str[locklevel+4], res2str[res]));
switch (res) switch (res)
{ {
case DIDNT_GET_THE_LOCK: case DIDNT_GET_THE_LOCK:
...@@ -159,7 +190,8 @@ pthread_handler_t test_lockman(void *arg) ...@@ -159,7 +190,8 @@ pthread_handler_t test_lockman(void *arg)
/* not implemented, so take a regular lock */ /* not implemented, so take a regular lock */
case GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE: case GOT_THE_LOCK_NEED_TO_LOCK_A_SUBRESOURCE:
res= lockman_getlock(&lockman, lo, row, lock_array[locklevel]); res= lockman_getlock(&lockman, lo, row, lock_array[locklevel]);
DIAG(("loid=%2d, ROW %d lock %s, res=%s", loid, row, lock2str[locklevel], res2str[res])); DIAG(("loid=%2d, ROW %d lock %s, res=%s", loid, row,
lock2str[locklevel], res2str[res]));
if (res == DIDNT_GET_THE_LOCK) if (res == DIDNT_GET_THE_LOCK)
{ {
lockman_release_locks(&lockman, lo); lockman_release_locks(&lockman, lo);
...@@ -196,7 +228,7 @@ int main() ...@@ -196,7 +228,7 @@ int main()
my_init(); my_init();
plan(14); plan(31);
if (my_atomic_initialize()) if (my_atomic_initialize())
return exit_status(); return exit_status();
...@@ -222,11 +254,21 @@ int main() ...@@ -222,11 +254,21 @@ int main()
test_lockman_simple(); test_lockman_simple();
#define CYCLES 100 #define CYCLES 1000
#define THREADS Nlos /* don't change this line */ #define THREADS Nlos /* don't change this line */
/* mixed load, stress-test with random locks */
Nrows= 100;
Ntables= 10;
table_lock_ratio= 10;
run_test("lockman", test_lockman, THREADS,CYCLES); run_test("lockman", test_lockman, THREADS,CYCLES);
/* "real-life" simulation - many rows, no table locks */
Nrows= 1000000;
Ntables= 10;
table_lock_ratio= 0;
run_test("lockman", test_lockman, THREADS,10000);
for (i= 0; i < Nlos; i++) for (i= 0; i < Nlos; i++)
{ {
lockman_release_locks(&lockman, &loarray[i]); lockman_release_locks(&lockman, &loarray[i]);
...@@ -235,7 +277,12 @@ int main() ...@@ -235,7 +277,12 @@ int main()
lf_pinbox_put_pins(loarray[i].pins); lf_pinbox_put_pins(loarray[i].pins);
} }
{
ulonglong now= my_getsystime();
lockman_destroy(&lockman); lockman_destroy(&lockman);
now= my_getsystime()-now;
diag("lockman_destroy: %g", ((double)now)/1e7);
}
pthread_mutex_destroy(&rt_mutex); pthread_mutex_destroy(&rt_mutex);
pthread_cond_destroy(&rt_cond); pthread_cond_destroy(&rt_cond);
......
...@@ -154,7 +154,7 @@ pthread_handler_t test_lf_pinbox(void *arg) ...@@ -154,7 +154,7 @@ pthread_handler_t test_lf_pinbox(void *arg)
typedef union { typedef union {
int32 data; int32 data;
void *not_used; /* to guarantee sizeof(TLA) >= sizeof(void *) */ void *not_used;
} TLA; } TLA;
pthread_handler_t test_lf_alloc(void *arg) pthread_handler_t test_lf_alloc(void *arg)
...@@ -294,7 +294,7 @@ int main() ...@@ -294,7 +294,7 @@ int main()
pthread_mutex_init(&mutex, 0); pthread_mutex_init(&mutex, 0);
pthread_cond_init(&cond, 0); pthread_cond_init(&cond, 0);
my_atomic_rwlock_init(&rwl); my_atomic_rwlock_init(&rwl);
lf_alloc_init(&lf_allocator, sizeof(TLA)); lf_alloc_init(&lf_allocator, sizeof(TLA), offsetof(TLA, not_used));
lf_hash_init(&lf_hash, sizeof(int), LF_HASH_UNIQUE, 0, sizeof(int), 0, lf_hash_init(&lf_hash, sizeof(int), LF_HASH_UNIQUE, 0, sizeof(int), 0,
&my_charset_bin); &my_charset_bin);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment