830 likes | 1.03k Views
CSC 536 Lecture 4. Outline. Distributed transactions STM (Software Transactional Memory) ScalaSTM Consistency Defining consistency models Data centric, Client centric Implementing consistency Replica management, Consistency Protocols. Distributed Transactions. Distributed transactions.
E N D
Outline • Distributed transactions • STM (Software Transactional Memory) • ScalaSTM • Consistency • Defining consistency models • Data centric, Client centric • Implementing consistency • Replica management, Consistency Protocols
Distributed transactions • Transactions, like mutual exclusion, protect shared data against simultaneous access by several concurrent processes. • Transactions allow a process to access and modify multiple data items as a single atomic transaction. • If the process backs out halfway during the transaction, everything is restored to the point just before the transaction started.
Distributed transactions: example 1 • A customer dials into her bank web account and does the following: • Withdraws amount x from account 1. • Deposits amount x to account 2. • If telephone connection is broken after the first step but before the second, what happens? • Either both or neither should be completed. • Requires special primitives provided by the DS.
The Transaction Model Primitive Description BEGIN_TRANSACTION Make the start of a transaction END_TRANSACTION Terminate the transaction and try to commit ABORT_TRANSACTION Kill the transaction and restore the old values READ Read data from a file, a table, or otherwise WRITE Write data to a file, a table, or otherwise • Examples of primitives for transactions
Distributed transactions: example 2 BEGIN_TRANSACTION reserve WP -> JFK; reserve JFK -> Nairobi; reserve Nairobi -> Malindi;END_TRANSACTION (a) BEGIN_TRANSACTION reserve WP -> JFK; reserve JFK -> Nairobi; reserve Nairobi -> Malindi full =>ABORT_TRANSACTION (b) • Transaction to reserve three flights commits • Transaction aborts when third flight is unavailable
ACID • Transactions are • Atomic: to the outside world, the transaction happens indivisibly. • Consistent: the transaction does not violate system invariants. • Isolated (or serializable): concurrent transactions do not interfere with each other. • Durable: once a transaction commits, the changes are permanent.
Flat, nested and distributed transactions • A nested transaction • A distributed transaction
Implementation of distributed transactions • For simplicity, we consider transactions on a file system. • Note that if each process executing a transaction just updates the file in place, transactions will not be atomic, and changes will not vanish if the transaction aborts. • Other methods required.
Atomicity • If each process executing a transaction just updates the file in place, transactions will not be atomic, and changes will vanish if the transaction aborts.
Solution 1: Private Workspace • The file index and disk blocks for a three-block file • The situation after a transaction has modified block 0 and appended block 3 • After committing
Solution 2: Writeahead Log x = 0; y = 0; BEGIN_TRANSACTION; x = x + 1; y = y + 2 x = y * y; END_TRANSACTION; (a) Log [x = 0 / 1] (b) Log [x = 0 / 1] [y = 0/2] (c) Log [x = 0 / 1] [y = 0/2] [x = 0/4] (d) • (a) A transaction • (b) – (d) The log before each statement is executed
Concurrency control (1) • We just learned how to achieve atomicity; we will learn about durability when discussing fault tolerance • Need to handle consistency and isolation • Concurrency control allows several transactions to be executed simultaneously, while making sure that the data is left in a consistent state • This is done by scheduling operations on data in an order whereby the final result is the same as if all transactions had run sequentially
Concurrency control (2) • General organization of managers for handling transactions
Concurrency control (3) • General organization of managers for handling distributed transactions.
Serializability • The main issue in concurrency control is the scheduling of conflicting operations (operating on same data item and one of which is a write operation) • Read/Write operations can be synchronized using: • Mutual exclusion mechanisms, or • Scheduling using timestamps • Pessimistic/optimistic concurrency control
The lost update problem Transaction T : Transaction U : balance = b.getBalance(); balance = b.getBalance(); b.setBalance(balance*1.1); b.setBalance(balance*1.1); a.withdraw(balance/10) c.withdraw(balance/10) balance = b.getBalance(); $200 balance = b.getBalance(); $200 b.setBalance(balance*1.1); $220 b.setBalance(balance*1.1); $220 a.withdraw(balance/10) $80 c.withdraw(balance/10) $280 Accounts a, b, and c start with $100, $200, and $300, respectively
The inconsistent retrievals problem : Transaction V : Transaction W a.withdraw(100) aBranch.branchTotal() b.deposit(100) a.withdraw(100); $100 total = a.getBalance() $100 total = total+b.getBalance() $300 total = total+c.getBalance() b.deposit(100) $300 Accounts a and b start with $200 each.
A serialized interleaving of T and U Transaction T : Transaction U : balance = b.getBalance() balance = b.getBalance() b.setBalance(balance*1.1) b.setBalance(balance*1.1) a.withdraw(balance/10) c.withdraw(balance/10) balance = b.getBalance() $200 b.setBalance(balance*1.1) $220 balance = b.getBalance() $220 b.setBalance(balance*1.1) $242 a.withdraw(balance/10) $80 c.withdraw(balance/10) $278
A serialized interleaving of V and W Transaction V : Transaction W : a.withdraw(100); aBranch.branchTotal() b.deposit(100) $100 a.withdraw(100); $300 b.deposit(100) $100 total = a.getBalance() $400 total = total+b.getBalance() total = total+c.getBalance() ...
Read and write operation conflict rules Operations of different Conflict Reason transactions read read No Because the effect of a pair of read operations does not depend on the order in which they are executed read write Yes Because the effect of a read and a write operation depends on the order of their execution write write Yes Because the effect of a pair of write operations depends on the order of their execution
Serializability • Two transactions are serialized • if and only if • All pairs of conflicting operations of the two transactions are executed in the same order at all objects they both access.
A non-serialized interleaving of operations of transactions T and U Transaction T : Transaction U : x = read(i) write(i, 10) y = read(j) write(j, 30) write(j, 20) z = read (i)
Recoverability of aborts • Aborted transactions must be prevented from affecting other concurrent transactions • Dirty reads • Cascading aborts • Premature writes
A dirty read when transaction T aborts Transaction T : Transaction U : a.getBalance() a.getBalance() a.setBalance(balance + 10) a.setBalance(balance + 20) balance = a.getBalance() $100 a.setBalance(balance + 10) $110 balance = a.getBalance() $110 a.setBalance(balance + 20) $130 commit transaction abort transaction
Cascading aborts • Suppose: • U delays committing until concurrent transaction T decides whether to commit or abort • Transaction V has seen the effects due to transaction U • T decides to abort
Cascading aborts • Suppose: • U delays committing until concurrent transaction T decides whether to commit or abort • Transaction V has seen the effects due to transaction U • T decides to abort • V and U must abort
Overwriting uncommitted values Transaction T : Transaction U : a.setBalance(105) a.setBalance(110) $100 a.setBalance(105) $105 a.setBalance(110) $110
Transactions T and U with locks Transaction T : Transaction U : balance = b.getBalance() balance = b.getBalance() b.setBalance(bal*1.1) b.setBalance(bal*1.1) a.withdraw(bal/10) c.withdraw(bal/10) Operations Locks Operations Locks openTransaction bal = b.getBalance() lock B openTransaction b.setBalance(bal*1.1) bal = b.getBalance() waits for T ’s A a.withdraw(bal/10) lock lock on B closeTransaction unlock A , B lock B b.setBalance(bal*1.1) C c.withdraw(bal/10) lock closeTransaction unlock B , C
Two-phase locking (2) • Idea: the scheduler grants locks in a way that creates only serializable schedules. • In 2-phase-locking, the transaction acquires all the locks it needs in the first phase, and then releases them in the second. This will insure a serializableschedule. • Dirty reads, cascading aborts, premature writes are still possible
Two-phase locking (2) • Idea: the scheduler grants locks in a way that creates only serializable schedules. • In 2-phase-locking, the transaction acquires all the locks it needs in the first phase, and then releases them in the second. This will insure a serializableschedule. • Dirty reads, cascading aborts, premature writes are still possible • Under strict 2-phase locking, a transaction that needs to read or write an object must be delayed until other transactions that wrote the same object have committed or aborted • Locks are held until transaction commits or aborts • Example: CORBA Concurrency Control Service
Two-phase locking in a distributed system • The data is assumed to be distributed across multiple machines • Centralized 2PL: central scheduler grants locks • Primary 2PL: local scheduler is coordinator for local data • Distributed 2PL: (data may be replicated) • the local schedulers use a distributed mutual exclusion algorithm to obtain a lock • The local scheduler forwards Read/Write operations to data managers holding the replicas
Two-phase locking issues • Exclusive locks reduce concurrency more than necessary. It is sometimes preferable to allow concurrent transactions to read an object; two types of locks may be needed (read locks and write locks) • Deadlocks are possible. • Solution 1: acquire all locks in the same order. • Solution 2: use a graph to detect potential deadlocks.
Deadlock with write locks Transaction T Transaction U Operations Locks Operations Locks write lock A a.deposit(100); write lock B b.deposit(200) b.withdraw(100) waits for U ’s a.withdraw(200); waits for T ’s lock on B lock on A
The wait-for graph Held by Waits for A T U U T B Waits for Held by
Deadlock prevention with timeouts Transaction T Transaction U Operations Locks Operations Locks A write lock a.deposit(100); B write lock b.deposit(200) b.withdraw(100) a.withdraw(200); U waits for T’s waits for ’s lock on B lock on A (timeout elapses) T’s lock on A becomes vulnerable, unlock A , abort T a.withdraw(200); write locks A unlock A B ,
Disadvantages of locking • High overhead • Deadlocks • Locks cannot be released until the end of the transaction, which reduces concurrency • In most applications, the likelihood of two clients accessing the same object is low
Pessimistic timestamp concurrency control • A transaction’s request to write an object is valid only if that object was last read and written by an earlier transaction • A transaction’s request to read an object is valid only if that object was last written by an earlier transaction • Advantage: Non-blocking and deadlock-free • Disadvantage: Transactions may need to abort and restart
Operation conflicts for timestamp ordering Rule Tc Ti 1. write read Tc must not write an object that has been read by any Ti where Ti > Tc this requires that Tc ≥ the maximum read timestamp of the object. Ti > Tc 2. write write Tc must not write an object that has been written by any Ti where this requires that Tc > write timestamp of the committed object. Ti > Tc 3. read write Tc must not read an object that has been written by any Ti where this requires that Tc > write timestamp of the committed object.
Pessimistic Timestamp Ordering • Concurrency control using timestamps.
Optimistic timestamp ordering • Idea: just go ahead and do the operations without paying attention to what concurrent transactions are doing: • Keep track of when each data item has been read and written. • Before committing, check whether any item has been changed since the transaction started. If so, abort. If not, commit. • Advantage: deadlock free and fast. • Disadvatange: it can fail and transactions must be run again. • Example:ScalaSTM
Software Transactional Memory (STM) • Software transactional memory is a mediator that sits between a critical section of your code (the atomic block) and the program’s heap. • The STM intervenes during reads and writes in the atomic block, allowing it to check and/or avoid interference other threads.
Software Transactional Memory (STM) • STM uses optimistic concurrency control to coordinate thread-safe access to shared data structures • replaces the traditional approach of using locks • Assumes that atomic blocks will run concurrently without conflict • If reads and writes by multiple threads have gotten interleaved incorrectly then all of the writes of the atomic block are rolled back and the entire block is retried • If reads and writes are not interleaved, then it is as if they were done atomically and the atomic block can be committed • Other threads or actors can only see committed changes Keeps old versions of data so that you can back up
ScalaSTM • ScalaSTM is an implementation of STM for Scala • It manages only memory locations encapsulated in instances of mutable class Ref[A] • A is an immutable type • Ref-s ensure that fewer memory locations need to be managed • Changes to Ref-s values make use of Scala’s efficient immutable data structures • Allows atomic blocks to be expressed directly in Scala • No synchronized, no deadlocks or race conditions, and good scalability • Includes concurrent sets and maps and an easier and safer replacement for wait and notifyAll
ScalaSTM first example val (x, y) = (Ref(10), Ref(0)) def sum = atomic { implicit txn => val a = x() valb = y() a + b } def transfer(n: Int) { atomic { implicit txn => x() -= n y() += n } } • Use a Ref for each shared variable to get STM involved • Use atomic for each critical section • atomic is a function with implicit parameter of type InTxn
ScalaSTM first example // sum // transfer(2) atomic atomic | begin txn attempt | begin txn attempt | | read x -> 10 | | read x -> 10 | | : | | write x <- 8 | | | | read y -> 0 | | : | | write y <- 2 | | | commit | | read y -> x read is invalid +-> () | roll back | begin txn attempt | | read x -> 8 | | read y -> 2 | commit +-> 10 • When sum tries to read y, STM detects that the value previously read from x is no longer correct • On the second attempt sum succeeds
ScalaSTM example: ConcurrentIntList import scala.concurrent.stm._ class ConcurrentIntList { private class Node(valelem: Int, prev0: Node, next0: Node) { valisHeader = prev0 == null valprev = Ref(if (isHeader) this else prev0) val next = Ref(if (isHeader) this else next0) } private val header = new Node(-1, null, null) • In shared, mutable linked list, need thread-safety for each node’s prev and next references • Use a Ref for each reference to get STM involved • Ref is a single mutable cell
ScalaSTM example: ConcurrentIntList def addLast(elem: Int) { atomic { implicit txn => valp = header.prev() valnewNode = new Node(elem, p, header) p.next() = newNode header.prev() = newNode } } • Appending a new node involves reads/writes of several references that should be done atomically • If x is a Ref, x() gets the value stored in x, and x() = val sets it to val • Ref-s can only be read and written inside an atomic block