Commit 62dc15e0 authored by Vincenzo Liberatore's avatar Vincenzo Liberatore

#Addresses 293

Write lock implementation.

git-svn-id: file:///svn/tokudb@1996 c7de825b-a66e-492c-adef-691d508d4ae1
parent f440e237
...@@ -543,33 +543,7 @@ static int __toku_consolidate(toku_lock_tree* tree, ...@@ -543,33 +543,7 @@ static int __toku_consolidate(toku_lock_tree* tree,
assert(tree && to_insert && txn); assert(tree && to_insert && txn);
toku_range_tree* mainread = tree->mainread; toku_range_tree* mainread = tree->mainread;
assert(mainread); assert(mainread);
/** This is so important that it should go into doxygen at some point,
either here or in the .h file
Memory ownership:
- tree->buf is an array of toku_range's, which the lt owns
The contents of tree->buf are volatile (this is a buffer space
that we pass around to various functions, and every time we
invoke a new function, its previous contents may become
meaningless)
- tree->buf[i].left, .right are toku_points (ultimately a struct),
also owned by lt. We gave a pointer only to this memory to the
range tree earlier when we inserted a range, but the range tree
does not own it!
- tree->buf[i].{left,right}.{key_payload,data_payload} is owned by
the lt, we made copies from the DB at some point
- to_insert we own (it's static)
- to_insert.left, .right are toku_point's, and we own them.
If we have consolidated, we own them because we had allocated
them earlier, but
if we have not consolidated we need to gain ownership now:
we will gain ownership by copying all payloads and
allocating the points.
-to_insert.{left,right}.{key_payload, data_payload} are owned by lt,
we made copies from the DB at consolidation time
*/
/* Find the self read tree */ /* Find the self read tree */
r = __toku_lt_selfread(tree, txn, &selfread); r = __toku_lt_selfread(tree, txn, &selfread);
if (r!=0) return r; if (r!=0) return r;
...@@ -835,6 +809,9 @@ int toku_lt_acquire_write_lock(toku_lock_tree* tree, DB_TXN* txn, ...@@ -835,6 +809,9 @@ int toku_lt_acquire_write_lock(toku_lock_tree* tree, DB_TXN* txn,
if (tree->duplicates && !data) return EINVAL; if (tree->duplicates && !data) return EINVAL;
if (tree->duplicates && key != data && if (tree->duplicates && key != data &&
__toku_lt_is_infinite(key)) return EINVAL; __toku_lt_is_infinite(key)) return EINVAL;
/* Verify that NULL keys have payload and size that are mutually
consistent */
__toku_lt_verify_null_key(key); __toku_lt_verify_null_key(key);
__toku_lt_verify_null_key(data); __toku_lt_verify_null_key(data);
...@@ -847,7 +824,7 @@ int toku_lt_acquire_write_lock(toku_lock_tree* tree, DB_TXN* txn, ...@@ -847,7 +824,7 @@ int toku_lt_acquire_write_lock(toku_lock_tree* tree, DB_TXN* txn,
__toku_init_point(&left, tree, key, data); __toku_init_point(&left, tree, key, data);
__toku_init_point(&right, tree, key, data); __toku_init_point(&right, tree, key, data);
__toku_init_query(&query, &left, &right); __toku_init_query(&query, &left, &right);
/* if 'K' is dominated by selfwrite('txn') then return success. */ /* if 'K' is dominated by selfwrite('txn') then return success. */
r = __toku_lt_dominated(tree, &query, r = __toku_lt_dominated(tree, &query,
......
...@@ -10,10 +10,37 @@ ...@@ -10,10 +10,37 @@
Lock trees are toku-struct's for granting long-lived locks to transactions. Lock trees are toku-struct's for granting long-lived locks to transactions.
See more details on the design document. See more details on the design document.
This is so important that it should go into doxygen at some point,
either here or in the .h file
Memory ownership:
- tree->buf is an array of toku_range's, which the lt owns
The contents of tree->buf are volatile (this is a buffer space
that we pass around to various functions, and every time we
invoke a new function, its previous contents may become
meaningless)
- tree->buf[i].left, .right are toku_points (ultimately a struct),
also owned by lt. We gave a pointer only to this memory to the
range tree earlier when we inserted a range, but the range tree
does not own it!
- tree->buf[i].{left,right}.{key_payload,data_payload} is owned by
the lt, we made copies from the DB at some point
- to_insert we own (it's static)
- to_insert.left, .right are toku_point's, and we own them.
If we have consolidated, we own them because we had allocated
them earlier, but
if we have not consolidated we need to gain ownership now:
we will gain ownership by copying all payloads and
allocating the points.
- to_insert.{left,right}.{key_payload, data_payload} are owned by lt,
we made copies from the DB at consolidation time
TODO: If the various range trees are inconsistent with TODO: If the various range trees are inconsistent with
each other, due to some system error like failed malloc, each other, due to some system error like failed malloc,
we defer to the db panic handler we defer to the db panic handler
TODO: Pass in another parameter to do this. TODO: Pass in another parameter to do this.
*/ */
#include <assert.h> #include <assert.h>
...@@ -144,41 +171,38 @@ int toku_lt_create(toku_lock_tree** ptree, DB* db, BOOL duplicates, ...@@ -144,41 +171,38 @@ int toku_lt_create(toku_lock_tree** ptree, DB* db, BOOL duplicates,
*/ */
int toku_lt_close(toku_lock_tree* tree); int toku_lt_close(toku_lock_tree* tree);
/* /**
* Acquires a read lock on a single key (or key/data). Acquires a read lock on a single key (or key/data).
* Params:
* tree: The lock tree for the db. \param tree: The lock tree for the db.
* txn: The TOKU Transaction this lock is for. \param txn: The TOKU Transaction this lock is for.
* key: The key this lock is for. \param key: The key this lock is for.
* data: The data this lock is for. \param data: The data this lock is for.
* Returns:
* 0: Success. \return
* DB_LOCK_NOTGRANTED: If there is a conflict in getting the lock. - 0: Success.
* This can only happen if some other transaction has - DB_LOCK_NOTGRANTED: If there is a conflict in getting the lock.
* a write lock that overlaps this point. This can only happen if some other transaction has
* EINVAL: If (tree == NULL || txn == NULL || key == NULL) or a write lock that overlaps this point.
* (tree->db is dupsort && data == NULL) or - ENOMEM: If adding the lock would exceed the maximum
* (tree->db is nodup && data != NULL) or memory allowed for payloads.
* (tree->db is dupsort && key != data &&
* (key == toku_lt_infinity || The following is asserted:
* key == toku_lt_neg_infinity)) (tree == NULL || txn == NULL || key == NULL) or
* ENOMEM: If adding the lock would exceed the maximum (tree->db is dupsort && data == NULL) or
* memory allowed for payloads. (tree->db is nodup && data != NULL) or
* Asserts: (tree->db is dupsort && key != data &&
* The EINVAL cases described will use assert to abort instead of returning errors. (key == toku_lt_infinity ||
* If this library is ever exported to users, we will use error datas instead. (toku_lock_tree* tree, DB_TXN* txn, const DBT* key, const DBT* data);
* Memory: If this library is ever exported to users, we will use EINVAL instead.
* It is safe to free keys and datas after this call.
* If the lock tree needs to hold onto the key or data, it will make copies In BDB, txn can actually be NULL (mixed operations with transactions and
* to its local memory. no transactions). This can cause conflicts, nobody was unable (so far)
* *** Note that txn == NULL is not supported at this time. to verify that MySQL does or does not use this.
*/ */
int toku_lt_acquire_read_lock(toku_lock_tree* tree, DB_TXN* txn, int toku_lt_acquire_read_lock(toku_lock_tree* tree, DB_TXN* txn,
const DBT* key, const DBT* data); const DBT* key, const DBT* data);
//In BDB, txn can actually be NULL (mixed operations with transactions and no transactions).
//This can cause conflicts, I was unable (so far) to verify that MySQL does or does not use
//this.
/* /*
* Acquires a read lock on a key range (or key/data range). (Closed range). * Acquires a read lock on a key range (or key/data range). (Closed range).
* Params: * Params:
......
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