#### Lecture 26: Multiprocessors

- Today's topics:
  - Directory-based coherence
  - Synchronization
  - Consistency
  - Shared memory vs message-passing

#### Cache Coherence Protocols

- Directory-based: A single location (directory) keeps track of the sharing status of a block of memory
- Snooping: Every cache block is accompanied by the sharing status of that block – all cache controllers monitor the shared bus so they can update the sharing status of the block, if necessary
- Write-invalidate: a processor gains exclusive access of a block before writing by invalidating all other copies
- Write-update: when a processor writes, it updates other shared copies of that block

### Coherence in Distributed Memory Multiprocs

- Distributed memory systems are typically larger → bus-based snooping may not work well
- Option 1: software-based mechanisms message-passing systems or software-controlled cache coherence
- Option 2: hardware-based mechanisms directory-based cache coherence

#### Distributed Memory Multiprocessors



#### **Directory-Based Cache Coherence**

- The physical memory is distributed among all processors
- The directory is also distributed along with the corresponding memory
- The physical address is enough to determine the location of memory
- The (many) processing nodes are connected with a scalable interconnect (not a bus) – hence, messages are no longer broadcast, but routed from sender to receiver – since the processing nodes can no longer snoop, the directory keeps track of sharing state

#### Cache Block States

 What are the different states a block of memory can have within the directory?

- Note that we need information for each cache so that invalidate messages can be sent
- The directory now serves as the arbitrator: if multiple write attempts happen simultaneously, the directory determines the ordering

### **Directory-Based Example**



# Example

| Request  | Cache<br>Hit/Miss | Messages                                                                                                  | Dir<br>State | State in C1 | State in C2 | State in C3 | State in C4 |
|----------|-------------------|-----------------------------------------------------------------------------------------------------------|--------------|-------------|-------------|-------------|-------------|
|          |                   |                                                                                                           |              | Inv         | Inv         | Inv         | Inv         |
| P1: Rd X | Miss              | Rd-req to Dir. Dir responds.                                                                              | X: S: 1      | S           | Inv         | Inv         | Inv         |
| P2: Rd X | Miss              | Rd-req to Dir. Dir responds.                                                                              | X: S: 1, 2   | S           | S           | Inv         | Inv         |
| P2: Wr X | Perms<br>Miss     | Upgr-req to Dir. Dir sends INV to P1. P1 sends ACK to Dir. Dir grants perms to P2.                        | X: M: 2      | Inv         | M           | Inv         | Inv         |
| P3: Wr X | Write<br>Miss     | Wr-req to Dir. Dir fwds<br>request to P2. P2 sends<br>data to Dir. Dir sends data<br>to P3.               | X: M: 3      | Inv         | Inv         | M           | Inv         |
| P3: Rd X | Read Hit          | -                                                                                                         | -            | Inv         | Inv         | M           | Inv         |
| P4: Rd X | Read<br>Miss      | Rd-req to Dir. Dir fwds<br>request to P3. P3 sends<br>data to Dir. Memory wrtbk.<br>Dir sends data to P4. | X: S: 3, 4   | Inv         | Inv         | S           | S           |

### **Directory Actions**

- If block is in uncached state:
  - Read miss: send data, make block shared
  - Write miss: send data, make block exclusive
- If block is in shared state:
  - > Read miss: send data, add node to sharers list
  - Write miss: send data, invalidate sharers, make excl
- If block is in exclusive state:
  - Read miss: ask owner for data, write to memory, send data, make shared, add node to sharers list
  - Data write back: write to memory, make uncached
  - Write miss: ask owner for data, write to memory, send data, update identity of new owner, remain exclusive

### Constructing Locks

- Applications have phases (consisting of many instructions) that must be executed atomically, without other parallel processes modifying the data
- A lock surrounding the data/code ensures that only one program can be in a critical section at a time
- The hardware must provide some basic primitives that allow us to construct locks with different properties



### Synchronization

- The simplest hardware primitive that greatly facilitates synchronization implementations (locks, barriers, etc.) is an atomic read-modify-write
- Atomic exchange: swap contents of register and memory
- Special case of atomic exchange: test & set: transfer memory location into register and write 1 into memory (if memory has 0, lock is free)
- lock: t&s register, location bnz register, lock

CS

st location, #0

When multiple parallel threads execute this code, only one will be able to enter CS

### Coherence Vs. Consistency

- Recall that coherence guarantees (i) write propagation (a write will eventually be seen by other processors), and (ii) write serialization (all processors see writes to the same location in the same order)
- The consistency model defines the ordering of writes and reads to different memory locations – the hardware guarantees a certain consistency model and the programmer attempts to write correct programs with those assumptions

#### Consistency Example

Consider a multiprocessor with bus-based snooping cache coherence

```
Initially A = B = 0
P1
P2
A \leftarrow 1
B \leftarrow 1
...
if (B == 0) if (A == 0)
Crit.Section
Crit.Section
```

### Consistency Example

Consider a multiprocessor with bus-based snooping cache coherence

```
Initially A = B = 0
P1
P2
A \leftarrow 1
B \leftarrow 1
...
if (B == 0) if (A == 0)
Crit.Section
Crit.Section
```

The programmer expected the above code to implement a lock – because of ooo, both processors can enter the critical section

### Sequential Consistency

- A multiprocessor is sequentially consistent if the result of the execution is achieveable by maintaining program order within a processor and interleaving accesses by different processors in an arbitrary fashion
- The multiprocessor in the previous example is not sequentially consistent
- Can implement sequential consistency by requiring the following: program order, write serialization, everyone has seen an update before a value is read – very intuitive for the programmer, but extremely slow

### Shared-Memory Vs. Message-Passing

#### **Shared-memory:**

- Well-understood programming model
- Communication is implicit and hardware handles protection
- Hardware-controlled caching

#### Message-passing:

- No cache coherence → simpler hardware
- Explicit communication → easier for the programmer to restructure code
- Software-controlled caching
- Sender can initiate data transfer

#### Ocean Kernel

```
Procedure Solve(A)
begin
 diff = done = 0;
 while (!done) do
    diff = 0;
    for i \leftarrow 1 to n do
      for j \leftarrow 1 to n do
        temp = A[i,j];
        A[i,j] \leftarrow 0.2 * (A[i,j] + neighbors);
        diff += abs(A[i,j] - temp);
      end for
    end for
    if (diff < TOL) then done = 1;
 end while
end procedure
```



#### Shared Address Space Model

```
int n, nprocs;
float **A, diff;
LOCKDEC(diff_lock);
BARDEC(bar1);
main()
begin
  read(n); read(nprocs);
 A \leftarrow G_MALLOC();
  initialize (A);
  CREATE (nprocs, Solve, A);
  WAIT FOR END (nprocs);
end main
```

```
procedure Solve(A)
  int i, j, pid, done=0;
  float temp, mydiff=0;
  int mymin = 1 + (pid * n/procs);
  int mymax = mymin + n/nprocs -1;
  while (!done) do
    mydiff = diff = 0;
    BARRIER(bar1,nprocs);
    for i ← mymin to mymax
      for i \leftarrow 1 to n do
      endfor
    endfor
    LOCK(diff_lock);
    diff += mydiff;
    UNLOCK(diff_lock);
    BARRIER (bar1, nprocs);
    if (diff < TOL) then done = 1;
    BARRIER (bar1, nprocs);
  endwhile
```

#### Message Passing Model

```
main()
                                                   for i \leftarrow 1 to nn do
 read(n); read(nprocs);
                                                     for j \leftarrow 1 to n do
  CREATE (nprocs-1, Solve);
  Solve();
                                                     endfor
  WAIT FOR END (nprocs-1);
                                                   endfor
                                                   if (pid != 0)
procedure Solve()
                                                    SEND(mydiff, 1, 0, DIFF);
  int i, j, pid, nn = n/nprocs, done=0;
                                                    RECEIVE(done, 1, 0, DONE);
 float temp, tempdiff, mydiff = 0;
                                                   else
  myA \leftarrow malloc(...)
                                                    for i \leftarrow 1 to nprocs-1 do
  initialize(myA);
                                                      RECEIVE(tempdiff, 1, *, DIFF);
  while (!done) do
                                                      mydiff += tempdiff;
    mydiff = 0;
                                                    endfor
    if (pid!=0)
                                                    if (mydiff < TOL) done = 1;
      SEND(&myA[1,0], n, pid-1, ROW);
                                                    for i ← 1 to nprocs-1 do
    if (pid != nprocs-1)
                                                      SEND(done, 1, I, DONE);
      SEND(\&myA[nn,0], n, pid+1, ROW);
                                                    endfor
    if (pid != 0)
                                                   endif
      RECEIVE(&myA[0,0], n, pid-1, ROW);
                                                 endwhile
    if (pid != nprocs-1)
      RECEIVE(\&myA[nn+1,0], n, pid+1, ROW);
                                                                                 19
```

## Title

Bullet