@ -55,7 +55,7 @@ TransactionImpl::TransactionImpl(TransactionDB* txn_db,
void TransactionImpl : : Initialize ( const TransactionOptions & txn_options ) {
void TransactionImpl : : Initialize ( const TransactionOptions & txn_options ) {
txn_id_ = GenTxnID ( ) ;
txn_id_ = GenTxnID ( ) ;
exec_status _ = STARTED ;
txn_state _ = STARTED ;
lock_timeout_ = txn_options . lock_timeout * 1000 ;
lock_timeout_ = txn_options . lock_timeout * 1000 ;
if ( lock_timeout_ < 0 ) {
if ( lock_timeout_ < 0 ) {
@ -84,7 +84,7 @@ TransactionImpl::~TransactionImpl() {
if ( expiration_time_ > 0 ) {
if ( expiration_time_ > 0 ) {
txn_db_impl_ - > RemoveExpirableTransaction ( txn_id_ ) ;
txn_db_impl_ - > RemoveExpirableTransaction ( txn_id_ ) ;
}
}
if ( ! name_ . empty ( ) & & exec_status _ ! = COMMITED ) {
if ( ! name_ . empty ( ) & & txn_state _ ! = COMMITED ) {
txn_db_impl_ - > UnregisterTransaction ( this ) ;
txn_db_impl_ - > UnregisterTransaction ( this ) ;
}
}
}
}
@ -97,7 +97,7 @@ void TransactionImpl::Clear() {
void TransactionImpl : : Reinitialize ( TransactionDB * txn_db ,
void TransactionImpl : : Reinitialize ( TransactionDB * txn_db ,
const WriteOptions & write_options ,
const WriteOptions & write_options ,
const TransactionOptions & txn_options ) {
const TransactionOptions & txn_options ) {
if ( ! name_ . empty ( ) & & exec_status _ ! = COMMITED ) {
if ( ! name_ . empty ( ) & & txn_state _ ! = COMMITED ) {
txn_db_impl_ - > UnregisterTransaction ( this ) ;
txn_db_impl_ - > UnregisterTransaction ( this ) ;
}
}
TransactionBaseImpl : : Reinitialize ( txn_db - > GetRootDB ( ) , write_options ) ;
TransactionBaseImpl : : Reinitialize ( txn_db - > GetRootDB ( ) , write_options ) ;
@ -128,21 +128,21 @@ Status TransactionImpl::CommitBatch(WriteBatch* batch) {
if ( IsExpired ( ) ) {
if ( IsExpired ( ) ) {
s = Status : : Expired ( ) ;
s = Status : : Expired ( ) ;
} else if ( expiration_time_ > 0 ) {
} else if ( expiration_time_ > 0 ) {
ExecutionStatus expected = STARTED ;
TransactionState expected = STARTED ;
can_commit = std : : atomic_compare_exchange_strong ( & exec_status _, & expected ,
can_commit = std : : atomic_compare_exchange_strong ( & txn_state _, & expected ,
AWAITING_COMMIT ) ;
AWAITING_COMMIT ) ;
} else if ( exec_status _ = = STARTED ) {
} else if ( txn_state _ = = STARTED ) {
// lock stealing is not a concern
// lock stealing is not a concern
can_commit = true ;
can_commit = true ;
}
}
if ( can_commit ) {
if ( can_commit ) {
exec_status _. store ( AWAITING_COMMIT ) ;
txn_state _. store ( AWAITING_COMMIT ) ;
s = db_ - > Write ( write_options_ , batch ) ;
s = db_ - > Write ( write_options_ , batch ) ;
if ( s . ok ( ) ) {
if ( s . ok ( ) ) {
exec_status _. store ( COMMITED ) ;
txn_state _. store ( COMMITED ) ;
}
}
} else if ( exec_status _ = = LOCKS_STOLEN ) {
} else if ( txn_state _ = = LOCKS_STOLEN ) {
s = Status : : Expired ( ) ;
s = Status : : Expired ( ) ;
} else {
} else {
s = Status : : InvalidArgument ( " Transaction is not in state for commit. " ) ;
s = Status : : InvalidArgument ( " Transaction is not in state for commit. " ) ;
@ -170,16 +170,16 @@ Status TransactionImpl::Prepare() {
if ( expiration_time_ > 0 ) {
if ( expiration_time_ > 0 ) {
// must concern ourselves with expiraton and/or lock stealing
// must concern ourselves with expiraton and/or lock stealing
// need to compare/exchange bc locks could be stolen under us here
// need to compare/exchange bc locks could be stolen under us here
ExecutionStatus expected = STARTED ;
TransactionState expected = STARTED ;
can_prepare = std : : atomic_compare_exchange_strong ( & exec_status _, & expected ,
can_prepare = std : : atomic_compare_exchange_strong ( & txn_state _, & expected ,
AWAITING_PREPARE ) ;
AWAITING_PREPARE ) ;
} else if ( exec_status _ = = STARTED ) {
} else if ( txn_state _ = = STARTED ) {
// expiration and lock stealing is not possible
// expiration and lock stealing is not possible
can_prepare = true ;
can_prepare = true ;
}
}
if ( can_prepare ) {
if ( can_prepare ) {
exec_status _. store ( AWAITING_PREPARE ) ;
txn_state _. store ( AWAITING_PREPARE ) ;
// transaction can't expire after preparation
// transaction can't expire after preparation
expiration_time_ = 0 ;
expiration_time_ = 0 ;
WriteOptions write_options = write_options_ ;
WriteOptions write_options = write_options_ ;
@ -191,15 +191,15 @@ Status TransactionImpl::Prepare() {
if ( s . ok ( ) ) {
if ( s . ok ( ) ) {
assert ( log_number_ ! = 0 ) ;
assert ( log_number_ ! = 0 ) ;
dbimpl_ - > MarkLogAsContainingPrepSection ( log_number_ ) ;
dbimpl_ - > MarkLogAsContainingPrepSection ( log_number_ ) ;
exec_status _. store ( PREPARED ) ;
txn_state _. store ( PREPARED ) ;
}
}
} else if ( exec_status _ = = LOCKS_STOLEN ) {
} else if ( txn_state _ = = LOCKS_STOLEN ) {
s = Status : : Expired ( ) ;
s = Status : : Expired ( ) ;
} else if ( exec_status _ = = PREPARED ) {
} else if ( txn_state _ = = PREPARED ) {
s = Status : : InvalidArgument ( " Transaction has already been prepared. " ) ;
s = Status : : InvalidArgument ( " Transaction has already been prepared. " ) ;
} else if ( exec_status _ = = COMMITED ) {
} else if ( txn_state _ = = COMMITED ) {
s = Status : : InvalidArgument ( " Transaction has already been committed. " ) ;
s = Status : : InvalidArgument ( " Transaction has already been committed. " ) ;
} else if ( exec_status _ = = ROLLEDBACK ) {
} else if ( txn_state _ = = ROLLEDBACK ) {
s = Status : : InvalidArgument ( " Transaction has already been rolledback. " ) ;
s = Status : : InvalidArgument ( " Transaction has already been rolledback. " ) ;
} else {
} else {
s = Status : : InvalidArgument ( " Transaction is not in state for commit. " ) ;
s = Status : : InvalidArgument ( " Transaction is not in state for commit. " ) ;
@ -223,14 +223,14 @@ Status TransactionImpl::Commit() {
// to change our state out from under us in the even that we expire and have
// to change our state out from under us in the even that we expire and have
// our locks stolen. In this case the only valid state is STARTED because
// our locks stolen. In this case the only valid state is STARTED because
// a state of PREPARED would have a cleared expiration_time_.
// a state of PREPARED would have a cleared expiration_time_.
ExecutionStatus expected = STARTED ;
TransactionState expected = STARTED ;
commit_single = std : : atomic_compare_exchange_strong (
commit_single = std : : atomic_compare_exchange_strong ( & txn_state_ , & expected ,
& exec_status_ , & expected , AWAITING_COMMIT ) ;
AWAITING_COMMIT ) ;
TEST_SYNC_POINT ( " TransactionTest::ExpirableTransactionDataRace:1 " ) ;
TEST_SYNC_POINT ( " TransactionTest::ExpirableTransactionDataRace:1 " ) ;
} else if ( exec_status _ = = PREPARED ) {
} else if ( txn_state _ = = PREPARED ) {
// expiration and lock stealing is not a concern
// expiration and lock stealing is not a concern
commit_prepared = true ;
commit_prepared = true ;
} else if ( exec_status _ = = STARTED ) {
} else if ( txn_state _ = = STARTED ) {
// expiration and lock stealing is not a concern
// expiration and lock stealing is not a concern
commit_single = true ;
commit_single = true ;
}
}
@ -241,16 +241,15 @@ Status TransactionImpl::Commit() {
s = Status : : InvalidArgument (
s = Status : : InvalidArgument (
" Commit-time batch contains values that will not be committed. " ) ;
" Commit-time batch contains values that will not be committed. " ) ;
} else {
} else {
exec_status _. store ( AWAITING_COMMIT ) ;
txn_state _. store ( AWAITING_COMMIT ) ;
s = db_ - > Write ( write_options_ , GetWriteBatch ( ) - > GetWriteBatch ( ) ) ;
s = db_ - > Write ( write_options_ , GetWriteBatch ( ) - > GetWriteBatch ( ) ) ;
Clear ( ) ;
Clear ( ) ;
if ( s . ok ( ) ) {
if ( s . ok ( ) ) {
exec_status _. store ( COMMITED ) ;
txn_state _. store ( COMMITED ) ;
}
}
}
}
} else if ( commit_prepared ) {
} else if ( commit_prepared ) {
exec_status_ . store ( AWAITING_COMMIT ) ;
txn_state_ . store ( AWAITING_COMMIT ) ;
// We take the commit-time batch and append the Commit marker.
// We take the commit-time batch and append the Commit marker.
// The Memtable will ignore the Commit marker in non-recovery mode
// The Memtable will ignore the Commit marker in non-recovery mode
@ -279,12 +278,12 @@ Status TransactionImpl::Commit() {
txn_db_impl_ - > UnregisterTransaction ( this ) ;
txn_db_impl_ - > UnregisterTransaction ( this ) ;
Clear ( ) ;
Clear ( ) ;
exec_status _. store ( COMMITED ) ;
txn_state _. store ( COMMITED ) ;
} else if ( exec_status _ = = LOCKS_STOLEN ) {
} else if ( txn_state _ = = LOCKS_STOLEN ) {
s = Status : : Expired ( ) ;
s = Status : : Expired ( ) ;
} else if ( exec_status _ = = COMMITED ) {
} else if ( txn_state _ = = COMMITED ) {
s = Status : : InvalidArgument ( " Transaction has already been committed. " ) ;
s = Status : : InvalidArgument ( " Transaction has already been committed. " ) ;
} else if ( exec_status _ = = ROLLEDBACK ) {
} else if ( txn_state _ = = ROLLEDBACK ) {
s = Status : : InvalidArgument ( " Transaction has already been rolledback. " ) ;
s = Status : : InvalidArgument ( " Transaction has already been rolledback. " ) ;
} else {
} else {
s = Status : : InvalidArgument ( " Transaction is not in state for commit. " ) ;
s = Status : : InvalidArgument ( " Transaction is not in state for commit. " ) ;
@ -295,22 +294,22 @@ Status TransactionImpl::Commit() {
Status TransactionImpl : : Rollback ( ) {
Status TransactionImpl : : Rollback ( ) {
Status s ;
Status s ;
if ( exec_status _ = = PREPARED ) {
if ( txn_state _ = = PREPARED ) {
WriteBatch rollback_marker ;
WriteBatch rollback_marker ;
WriteBatchInternal : : MarkRollback ( & rollback_marker , name_ ) ;
WriteBatchInternal : : MarkRollback ( & rollback_marker , name_ ) ;
exec_status _. store ( AWAITING_ROLLBACK ) ;
txn_state _. store ( AWAITING_ROLLBACK ) ;
s = db_impl_ - > WriteImpl ( write_options_ , & rollback_marker ) ;
s = db_impl_ - > WriteImpl ( write_options_ , & rollback_marker ) ;
if ( s . ok ( ) ) {
if ( s . ok ( ) ) {
// we do not need to keep our prepared section around
// we do not need to keep our prepared section around
assert ( log_number_ > 0 ) ;
assert ( log_number_ > 0 ) ;
dbimpl_ - > MarkLogAsHavingPrepSectionFlushed ( log_number_ ) ;
dbimpl_ - > MarkLogAsHavingPrepSectionFlushed ( log_number_ ) ;
Clear ( ) ;
Clear ( ) ;
exec_status _. store ( ROLLEDBACK ) ;
txn_state _. store ( ROLLEDBACK ) ;
}
}
} else if ( exec_status _ = = STARTED ) {
} else if ( txn_state _ = = STARTED ) {
// prepare couldn't have taken place
// prepare couldn't have taken place
Clear ( ) ;
Clear ( ) ;
} else if ( exec_status _ = = COMMITED ) {
} else if ( txn_state _ = = COMMITED ) {
s = Status : : InvalidArgument ( " This transaction has already been committed. " ) ;
s = Status : : InvalidArgument ( " This transaction has already been committed. " ) ;
} else {
} else {
s = Status : : InvalidArgument (
s = Status : : InvalidArgument (
@ -321,7 +320,7 @@ Status TransactionImpl::Rollback() {
}
}
Status TransactionImpl : : RollbackToSavePoint ( ) {
Status TransactionImpl : : RollbackToSavePoint ( ) {
if ( exec_status _ ! = STARTED ) {
if ( txn_state _ ! = STARTED ) {
return Status : : InvalidArgument ( " Transaction is beyond state for rollback. " ) ;
return Status : : InvalidArgument ( " Transaction is beyond state for rollback. " ) ;
}
}
@ -522,8 +521,8 @@ Status TransactionImpl::ValidateSnapshot(ColumnFamilyHandle* column_family,
bool TransactionImpl : : TryStealingLocks ( ) {
bool TransactionImpl : : TryStealingLocks ( ) {
assert ( IsExpired ( ) ) ;
assert ( IsExpired ( ) ) ;
ExecutionStatus expected = STARTED ;
TransactionState expected = STARTED ;
return std : : atomic_compare_exchange_strong ( & exec_status _, & expected ,
return std : : atomic_compare_exchange_strong ( & txn_state _, & expected ,
LOCKS_STOLEN ) ;
LOCKS_STOLEN ) ;
}
}
@ -534,7 +533,7 @@ void TransactionImpl::UnlockGetForUpdate(ColumnFamilyHandle* column_family,
Status TransactionImpl : : SetName ( const TransactionName & name ) {
Status TransactionImpl : : SetName ( const TransactionName & name ) {
Status s ;
Status s ;
if ( exec_status _ = = STARTED ) {
if ( txn_state _ = = STARTED ) {
if ( name_ . length ( ) ) {
if ( name_ . length ( ) ) {
s = Status : : InvalidArgument ( " Transaction has already been named. " ) ;
s = Status : : InvalidArgument ( " Transaction has already been named. " ) ;
} else if ( txn_db_impl_ - > GetTransactionByName ( name ) ! = nullptr ) {
} else if ( txn_db_impl_ - > GetTransactionByName ( name ) ! = nullptr ) {