# ID2203 Tutorial 3 - Shared Memory

Cosmin Arad Tallat M. Shafaat icarad(0)kth.se tallat(0)kth.se

Handout: February 19, 2010 February 26, 2010

## 1 Introduction

The goal of this tutorial is to understand and get accustomed to implementing *atomic register* shared memory abstractions for the synchronous and asynchronous system models.

# 2 Assignment

Using the Kompics framework, you will implement two atomic register abstractions as components: a fail-stop atomic register with multiple writers, described in Algorithm 1 and Algorithm 2 (Read-Impose Write-Consult), and a fail-silent atomic register with multiple writers, described in Algorithm 3 and Algorithm 4 (Read-Impose Write-Consult-Majority).

You will reuse the communication component, the timer component, the delay component, the beb broadcast component, and the pfd component. You have to implement a FailStopAtomicRegisterComponent (RIWC) and a FailSilentAtomicRegisterComponent (RIWCM), together with the events that are accepted and triggered by these components: AtomicRegister-ReadEvent, AtomicRegisterReadReturnEvent, AtomicRegisterWriteEvent, AtomicRegisterWriteReturnEvent and the messages that they exchange: ReadMessage, WriteMessage, AckMessage, and ReadValueMessage.

You will slightly change the application component from the previous assignment, to handle the AtomicRegisterReadReturnEvent and the AtomicRegisterWriteReturnEvent, and an ApplicationEvent raised by the main program and handled by the application component, that contains the string ops. The ops string contains read and write operations that are to be issued by the application component on the atomic register with index 0. Here is an example: W5:R:W6:R:D200:W3:D100:R. This means that the application

component will issue a Write(5) operation on register 0. Upon completion of this operation, it will issue a Read operation on register 0. Upon completion of this operation, it will issue a Write(6) operation on register 0. Upon completion of this operation, it will do nothing (sleep/timeout) for 200 milliseconds and then it will issue a Write(3) operation on register 0. Upon completion of this operation, it will again wait for 100 milliseconds and issue a Read operation on register 0.

You use the same InitEvent to pass to the components the set of all processes ( $\Pi$ ), or the number of processes, respectively. Note that the algorithms assume that each process knows and can communicate with all other processes in  $\Pi$ . This corresponds to a fully connected topology.

Implement the RIWC and RIWCM components and experiment with them as instructed in the following exercises. Describe your experiments in a written report. For each exercise include the topology descriptors used, and explain the behavior that you observe. For some exercises, you are required to start a process later than others. This is done by specifying the delay in the command of a scenario. For instance, given the following scenario:

The system will launch process 1 and 2, while process 3 will start executing after 1000 ms.

You need to read Sections 4.1-4.4 of the textbook, in order to fully understand the ideas behind the two algorithms. These algorithms implement arrays of atomic registers and thus are parameterized by the size of the array, which you have to use as a parameter to the algorithms.

The assignment is due on February 26th. You have to send your source code and written report by email before the next tutorial session. During the tutorial session you will present the assignment on a given topology description. You can work in groups of maximum 2 students. Be prepared to answer questions about your process's system architecture and explain the behavior of the algorithms. Any questions are welcome on the forum.

**Exercise 1** Verify the termination and atomicity properties of RIWC and RIWCM. Use a topology with 3 processes and 3 bidirectional links with

the same latency (say 1000ms) and execute RIWC and RIWCM. Use the following ops: process 1: D30000, process 2: D500:W4:D25000 and process 3: D10000:R. Here you have sequential operations and you should check that the read returns the last value written.

Exercise 2 Use a topology with 3 processes and 3 bidirectional links with the same latency (say 1000ms) and execute RIWCM. Start process 1 with ops: D500:W5:R:D5000:R:D30000 and process 2 with ops:

D500:W6:R:D5000:R:D30000, and don't start process 3 yet. This is a special case of failure called "initialy dead processes". Create your scenario such that when processes 1 and 2 begin to wait for 30 seconds, process 3 starts with ops: D500:R:D500:R:D10000.

What is the value read by process 3? Explain why. Can RIWCM be used in a fail-recovery model?

Exercise 3 Using the topology given below, or a similar one, execute RI-WCM for each possible starting order of the processes: 123, 132, 213, 231, 312, 321, with the following operations for process i: D500:Wi:R:D500:R:D8000. Order of processes means that the 3 processes should be started with a small delay (e.g. 100ms) between any 2 consecutive processes. For instance, for order 123, you can use the following scenario:

Give the values returned by the two read operations in each process and give a linearization of all the register operations (read and write from all processes).

```
link(1, 2, 1000, 0).bidirectional();
link(1, 3, 2000, 0).bidirectional();
link(2, 3, 1750, 0).bidirectional();
}
};
```

Exercise 4 Change the latencies in the previous topology, and possibly the waiting values in ops so that you obtain 3 distinct linearizations so that in each linearization, the write of value i is the last of the 3 write operations.

#### Implements: (N,N)AtomicRegister (nn-areg). Uses:BestEffortBroadcast (beb); PerfectPointToPointLinks (pp2p); PerfectFailureDetector $(\mathcal{P})$ . 1: upon event $\langle Init \rangle$ do correct := $\Pi$ ; 2: i := rank(self);3: 4: for all r do 5: $writeSet[r] := \emptyset;$ reading[r] := false;6: $\operatorname{regid}[r] := 0;$ 7: readval[r] := 0;8: $\mathbf{v}[r] := 0;$ 9: ts[r] := 0;10: mrank[r] := 0;11: end for 12: 13: end event 14: **upon event** $\langle crash \mid p_i \rangle$ **do** $correct := correct \setminus \{p_i\};$ 15: 16: end event

Algorithm 1 Read-Impose Write-Consult (part 1)

17: **upon event**  $\langle nn\text{-}aRegRead \mid r \rangle$  **do**  $\operatorname{regid}[r] := \operatorname{regid}[r] + 1;$ 

reading[r] := true;

 $writeSet[r] := \emptyset;$ 

readval[r] := v[r];

18:

19:

20:

21:

22:

23: end event

```
24: upon event \langle nn\text{-}aRegWrite \mid r, val \rangle do
          \operatorname{reqid}[r] := \operatorname{reqid}[r] + 1;
25:
          writeSet[r] := \emptyset;
26:
          trigger \langle bebBroadcast \mid [WRITE, r, reqid[r], (ts[r]+1, i), val] \rangle;
27:
28: end event
```

**trigger**  $\langle bebBroadcast \mid [WRITE, r, reqid[r], (ts[r], mrank[r]), v[r]] \rangle$ ;

#### Algorithm 2 Read-Impose Write-Consult (part 2) 1: **upon event** $\langle bebDeliver \mid p_j, [WRITE, r, id, (t, j), val] \rangle$ **do** if (t, j) > (ts[r], mrank[r]) then v[r] := val;3: ts[r] := t;4: mrank[r] := j;5: end if 6: **trigger** $\langle pp2pSend \mid p_j, [Ack, r, id] \rangle;$ 7: 8: end event 9: **upon event** $\langle pp2pDeliver \mid p_j, [Ack, r, id] \rangle$ **do** if (id = reqid[r]) then 10: $writeSet[r] := writeSet[r] \cup \{p_j\};$ 11: 12: end if 13: end event 14: **upon exists** r **such that** correct $\subseteq$ writeSet[r] **do** if (reading[r] = true) then 15: reading[r] := false;16: **trigger** $\langle nn\text{-}aRegReadReturn \mid r, readval[r] \rangle$ ; 17: 18: else **trigger** $\langle nn\text{-}aRegWriteReturn \mid r \rangle$ ;

19:

20: 21: **end** 

end if

```
Algorithm 3 Read-Impose Write-Consult-Majority (part 1)
Implements:
         (N,N)AtomicRegister (nn-areg).
Uses:
         BestEffortBroadcast (beb);
         PerfectPointToPointLinks (pp2p).
 1: upon event \langle Init \rangle do
 2:
         i := rank(self);
         for all r do
 3:
              writeSet[r] := \emptyset;
 4:
             readSet[r] := \emptyset;
 5:
 6:
             reading[r] := false;
             \operatorname{reqid}[r] := 0;
 7:
             v[r] := 0;
 8:
             ts[r] := 0;
 9:
             mrank[r] := 0;
10:
         end for
11:
12: end event
13: upon event \langle nn\text{-}aRegRead \mid r \rangle do
14:
         \operatorname{reqid}[r] := \operatorname{reqid}[r] + 1;
15:
         reading[r] := true;
         readSet[r] := \emptyset;
16:
         writeSet[r] := \emptyset;
17:
         trigger \langle bebBroadcast \mid [Read, r, reqid[r]] \rangle;
19: end event
20: upon event \langle nn\text{-}aRegWrite \mid r, val \rangle do
         \operatorname{regid}[r] := \operatorname{regid}[r] + 1;
21:
         writeval[r] := val;
22:
         readSet[r] := \emptyset;
23:
24:
         writeSet[r] := \emptyset;
25:
         trigger \langle bebBroadcast \mid [Read, r, reqid[r]] \rangle;
26: end event
27: upon event \langle bebDeliver \mid p_i, [Read, r, id] \rangle do
         trigger \langle pp2pSend \mid p_j, [ReadVal, r, id, (ts[r], mrank[r]), v[r]] \rangle;
28:
```

29: end event

### Algorithm 4 Read-Impose Write-Consult-Majority (part 2)

```
1: upon event \langle pp2pDeliver \mid p_i, [ReadVal, r, id, (t, rk), val] \rangle do
        if (id = reqid[r]) then
            readSet[r] := readSet[r] \cup \{((t, rk), val)\};
 3:
        end if
 4:
 5: end event
 6: upon exists r such that |\text{readSet}[r]| > N/2 do
 7:
        ((t, rk), v) := highest(readSet[r]);
 8:
        readval[r] := v;
        if (reading[r] = true) then
 9:
            \mathbf{trigger} \ \langle bebBroadcast \mid [Write, r, reqid[r], (t, rk), readval[r]] \rangle;
10:
11:
            trigger \langle bebBroadcast \mid [WRITE, r, reqid[r], (t+1,i), readval[r]] \rangle;
12:
13:
        end if
14: end
15: upon event \langle bebDeliver \mid p_j, [WRITE, r, id, (t, j), val] \rangle do
        if (t, j) > (ts[r], mrank[r]) then
16:
            v[r] := val;
17:
            ts[r] := t;
18:
19:
            mrank[r] := j;
20:
        end if
        trigger \langle pp2pSend \mid p_j, [Ack, r, id] \rangle;
21:
22: end event
23: upon event \langle pp2pDeliver \mid p_i, [Ack, r, id] \rangle do
        if (id = reqid[r]) then
24:
            writeSet[r] := writeSet[r] \cup \{p_i\};
25:
        end if
26:
27: end event
28: upon exists r such that |\text{writeSet}[r]| > N/2 do
        if (reading[r] = true) then
29:
            reading[r] := false;
30:
            trigger \langle nn\text{-}aRegReadReturn \mid r, readval[r] \rangle;
31:
32:
            trigger \langle nn\text{-}aRegWriteReturn \mid r \rangle;
33:
        end if
34:
35: end
```