Commit 7568a5a8 authored by Barry Perlman's avatar Barry Perlman Committed by Yoni Fogel

Addresses #1745 Merge fix from 2.0.0 branch into main with command svn merge...

Addresses #1745 Merge fix from 2.0.0 branch into main with command svn merge -c 11736 ../../mysql.branches/2.0.0/tokudb  (inside toku/tokudb)

git-svn-id: file:///svn/toku/tokudb@11756 c7de825b-a66e-492c-adef-691d508d4ae1
parent 7f1d3808
...@@ -724,7 +724,7 @@ static void cachetable_complete_write_pair (CACHETABLE ct, PAIR p, BOOL do_remov ...@@ -724,7 +724,7 @@ static void cachetable_complete_write_pair (CACHETABLE ct, PAIR p, BOOL do_remov
// the pair dirty state is adjusted, and the write is completed. The write_me boolean // the pair dirty state is adjusted, and the write is completed. The write_me boolean
// is true when the pair is dirty and the pair is requested to be written. The keep_me // is true when the pair is dirty and the pair is requested to be written. The keep_me
// boolean is true, so the pair is not yet evicted from the cachetable. // boolean is true, so the pair is not yet evicted from the cachetable.
// Requires: This thread must hold the write lock for the pair.
static void cachetable_write_pair(CACHETABLE ct, PAIR p) { static void cachetable_write_pair(CACHETABLE ct, PAIR p) {
rwlock_read_lock(&ct->pending_lock, ct->mutex); rwlock_read_lock(&ct->pending_lock, ct->mutex);
...@@ -964,11 +964,20 @@ static u_int64_t tdelta(struct timeval *tnew, struct timeval *told) { ...@@ -964,11 +964,20 @@ static u_int64_t tdelta(struct timeval *tnew, struct timeval *told) {
} }
#endif #endif
// On entry: hold the ct lock and the rwlock_write_lock. // for debug
static PAIR write_for_checkpoint_pair = NULL;
// On entry: hold the ct lock
// On exit: the node is written out
// Method: take write lock
// if still pending write out the node
// release the write lock.
static void static void
write_pair_for_checkpoint (CACHETABLE ct, PAIR p) write_pair_for_checkpoint (CACHETABLE ct, PAIR p)
{ {
assert(p->state!=CTPAIR_WRITING); write_for_checkpoint_pair = p;
rwlock_write_lock(&p->rwlock, ct->mutex); // grab an exclusive lock on the pair
assert(p->state!=CTPAIR_WRITING); // if we have the write lock, no one else should be writing out the node
if (p->checkpoint_pending) { if (p->checkpoint_pending) {
// this is essentially a flush_and_maybe_remove except that // this is essentially a flush_and_maybe_remove except that
// we already have p->rwlock and we just do the write in our own thread. // we already have p->rwlock and we just do the write in our own thread.
...@@ -979,8 +988,12 @@ write_pair_for_checkpoint (CACHETABLE ct, PAIR p) ...@@ -979,8 +988,12 @@ write_pair_for_checkpoint (CACHETABLE ct, PAIR p)
assert(ct->size_writing>=0); assert(ct->size_writing>=0);
p->write_me = TRUE; p->write_me = TRUE;
p->remove_me = FALSE; p->remove_me = FALSE;
cachetable_write_pair(ct, p); // unlocks the pair cachetable_write_pair(ct, p); // releases the write lock on the pair
}
else {
rwlock_write_unlock(&p->rwlock); // didn't call cachetable_write_pair so we have to unlock it ourselves.
} }
write_for_checkpoint_pair = NULL;
} }
// for debugging // for debugging
...@@ -1029,9 +1042,7 @@ int toku_cachetable_get_and_pin(CACHEFILE cachefile, CACHEKEY key, u_int32_t ful ...@@ -1029,9 +1042,7 @@ int toku_cachetable_get_and_pin(CACHEFILE cachefile, CACHEKEY key, u_int32_t ful
} }
if (p->checkpoint_pending) { if (p->checkpoint_pending) {
get_and_pin_footprint = 4; get_and_pin_footprint = 4;
rwlock_write_lock(&p->rwlock, ct->mutex); write_pair_for_checkpoint(ct, p);
get_and_pin_footprint = 5;
write_pair_for_checkpoint(ct, p); // releases the pair_write_lock, but not the cachetable lock
} }
// still have the cachetable lock // still have the cachetable lock
// TODO: #1398 kill this hack before it multiplies further // TODO: #1398 kill this hack before it multiplies further
...@@ -1642,16 +1653,7 @@ toku_cachetable_end_checkpoint(CACHETABLE ct, TOKULOGGER logger, char **error_st ...@@ -1642,16 +1653,7 @@ toku_cachetable_end_checkpoint(CACHETABLE ct, TOKULOGGER logger, char **error_st
while ((p = ct->pending_head)!=0) { while ((p = ct->pending_head)!=0) {
ct->pending_head = ct->pending_head->pending_next; ct->pending_head = ct->pending_head->pending_next;
pending_pairs_remove(ct, p); pending_pairs_remove(ct, p);
write_pair_for_checkpoint(ct, p); // if still pending, clear the pending bit and write out the node
rwlock_write_lock(&p->rwlock, ct->mutex); // grab an exclusive lock on the pair
// If it is no longer pending we don't have do do anything
if (p->checkpoint_pending) {
write_pair_for_checkpoint(ct, p); // clears the pending bit, writes it out, and unlocks the write pair,
} else {
// it was previously written, so we just have to release the lock.
rwlock_write_unlock(&p->rwlock); // didn't call cachetable_write_pair so we have to unlock it ourselves.
}
// Don't need to unlock and lock cachetable, because the cachetable was unlocked and locked while the flush callback ran. // Don't need to unlock and lock cachetable, because the cachetable was unlocked and locked while the flush callback ran.
} }
} }
......
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