

# Lezione 21 Z64 7 CU 2

giovedì 16 novembre 2023 17:34

Andiamo a vedere ora in dettaglio le micro-istruzioni : **iniziamo dal fetch** (micro-programma anche esso) :

MAR  $\leftarrow$  RIP  
 MDR  $\leftarrow$  (MAR); RIP  $\leftarrow$  RIP + 8  
 IR  $\leftarrow$  MDR

In dettaglio :



In questa prima micro-istruzione dobbiamo spostare RIP in MAR : quindi i segnali che si attivano sono i seguenti : Brip impostato ad 1 per abilitare il buffer-tri-state che permette la scrittura dei dati sull'internal bus data . Questi dati viaggiano fino al registro MAR , il quale ha il segnale di Wmar settato ad 1 per abilitare la scrittura dentro esso.

Seconda micro-istruzione:



Qui chiediamo alla memoria di darci il dato ed in parallelo incrementiamo RIP. Notiamo che in questa esecuzione non ci basta un solo colpo di clock , visto che la memoria è più lenta : **RAM impiega 3 colpi di clock per eseguire**. Per mandare indirizzo su address bus devo abilitare il buffer-

tri-state (Bab =1), mentre nel contempo e vista l'indipendenza / esecuzione parallela istruzioni, incremento anche RIP ( IncRip=1).

Vediamo il secondo step :



In questo step , devo mantenere il dato costante su address bus, quindi tengo abilitato ancora il buffer-tri-state di MAR , ed inoltre visto che devo scrivere sul control bus, abilito il segnale di scrittura su di esso (Mrd=1).

Ultimo colpo di clock : bufferizzo all'interno del processore i dati che stanno arrivando dal data bus



Per effettuare la scrittura finale su MDR , si mantiene il buffer-tri-state di MAR attivo ( Bab=1), cosi i dato escono su address bus, si mantiene il control bus in scrittura (Mrd=1), si abilita il buffer-tri-state di MDR per fare entrare i dati dal data bus (Bdbin=1) e si abilita il segnale di scrittura su MDR (Wmdr=1)

Mentre nell'ultima fase della micro-istruzione andiamo a copiare MDR( codifica dell'istruzione) nell'IR :



#### Operazione in corso:

$IR \leftarrow MDR$ :

$$W_{IR} = 1, B_{MDROUT} = 1$$

Visto che devo copiare il contenuto di MDR in IR , abilito il buffer-tri-state per permettere scrittura sull'internal bus data (Bmdrout=1), poi i dati viaggiano su internal bus data, fino ad arrivare all'IR : serve il segnale di scrittura abilitato per fare ciò : Wir=1. **Quindi per fare la fase di fetch servono necessariamente 4 cicli di clock.**

Vediamo altri esempi :

`movq %rax, %rcx:`

- $MAR \leftarrow RIP$
  - $MDR \leftarrow (MAR); RIP \leftarrow RIP + 8$
  - 1. •  $IR \leftarrow MDR$
  - $TEMP2 \leftarrow RAX$
  - $RCX \leftarrow TEMP2$
- a. Da notare che per qualunque micro-istruzione esegue la fase di fetch .
  - b. Vediamo ora la micro-istruzione  $TEMP2 \leftarrow RAX$



- i. Notiamo che usiamo TEMP2 (registro tampone) , non passo per la ALU, ma passo per lo shifter , il quale in questo caso fa lo shift di 0 posizioni : il dato passa così come è ed inoltre perché posso usare 1 solo registro alla volta. Per attivare la lettura da RAX devo abilitare  $Rm = 1$  (da uc) , il quale permette di selezionare un qualunque registro, quindi per selezionare solo quel registro devo prendere gli

input dei decoder : codice del registro e taglia del registro virtuale . Queste informazioni vengono prese dall'IR : codice sorgente e taglia sorgente : piloto i mux : Smux=1 (taglia) e Regmux =0 (sorgente), attivo buffer-tri-state di RAX (permette il viaggio dei dati su internal data bus) e raggiungo TEMP2. Infine per attivare la scrittura su TEMP2 , devo abilitare il segnale di write (Wt2=1).

c. Vediamo ora RCX<-TEMP2



- ii. Per abilitare la scrittura di temp2 sull'internal data bus, devo passare per lo shifter ed abilitare il corrispondente buffer-tri-state (Sopcode=000000, Bs=1), quindi i dati arrivano dentro RCX , il quale ha impostato ad 1 il segnale di scrittura (Wm=1). Per quanto riguarda invece i mux : seleziono per quello sorgente il secondo ingresso (Smux=2), mentre per quello della dimensione prendo la seconda linea (Regmax= 1 : registro destinazione).

`movq %rax, 0xaaaa(%rcx, %rcx, 8):`

- $MAR \leftarrow RIP$
- $MDR \leftarrow (MAR)$ ;  $RIP \leftarrow RIP + 8$
- $IR \leftarrow MDR$
- $TEMP2 \leftarrow RCX$
- $TEMP1 \leftarrow \text{Shifter Out[SHL, 000011]}$

- 2.
- $MAR \leftarrow ALU \text{ OUT[ADD]}$
  - $TEMP1 \leftarrow IR[0:31]$
  - $TEMP2 \leftarrow MAR$
  - $MAR \leftarrow ALU \text{ OUT[ADD]}$
  - $MDR \leftarrow RAX$
  - $(MAR) \leftarrow MDR$

- Modalità di indirizzamento completa (base , indice , scala e spiazzamento)
- Partiamo da  $TEMP2 \leftarrow RCX$



- ii. Discorso analogo a prima : riuso componenti, ma con diversa configurazione.
- c.  $\text{TEMP1} \leftarrow \text{shifter out } [\text{shl}, 000011]$  : 000011 rappresenta il codice dello shifter (3) il che equivale a shiftare di 8 :



- ii. Dopo aver shiftato di 3 posizioni , abilito il buffer-tri-state per mandare i dati su internal data bus, e questi dati li scrivo su TEMP1 , abilitando il segnale opportuno ( $W_{\text{temp1}}=1$ ) .
- d.  $\text{TEMP2} \leftarrow \text{RAX}$ :



- ii. Setto i segnali nel giusto ordine : Smux=2 (taglia della destinazione), Regmux=1 (tipo del registro destinazione) , poi abilito il segnale di scrittura nei registri (Rm=1) ed infine abilito la scrittura su TEMP2 (Wt2=1)

e.  $\text{MAR} \leftarrow \text{ALU\_OUT[ADD]}$ :



- ii. Qui passo per ALU: effettua la somma (Aopcode=0000) , abilito il buffer-tri-state per scrivere sull'internal data bus (Ba=1) ed infine scrivo su MAR , abilitando anche qui il segnale (Wmar=1).

f.  $\text{TEMP1} <- \text{IR}[0:31]$  : cosi calcolo lo spiazzamento



- ii. Prendo i 32 bit dello spiazzamento da IR, abilito il suo buffer-tri-state ( $B_{\text{short}}=1$ ), così i dato viaggiano sull'internal bus data ed infine li scrivo in TEMP1, abilitando il segnale di scrittura ( $W_{T1}=1$ ).

g. TEMP2<-MAR:



- ii. Qui abilito il buffer-tri-state di MAR per scrivere su interna bus data ( $B_{\text{marout}}=1$ ), così i dati viaggiano sull'internal bus data, arrivando a TEMP2 e vi vengono scritti in quanto il segnale di scrittura è abilitato ( $W_{T2}=1$ ).

h. MAR  $\leftarrow$  ALU\_OUT[ADD]:



- ii. Sposto base\*indice+ spiazzamento in MAR: faccio fare all'alu la somma (Aopcode=0000) , abilito il buffer-tri-state per scrivere su internal bus data (Ba=1) , facendo viaggiare i dati fino a MAR ed abilitando il segnale di scrittura (Wmar=1).

i. MDR<-RAX:



- ii. Cominciamo dai mux: Smux=1 in quanto devo lavorare con il registro sorgente (RAX) , RegMUX=0 (registro base) , poi abilito la lettura dai registri (Rm=1), abilito il buffer-tri-state per mandare i dati sull'internal data bus , poi li mando a MDR abilitando il buffer-tri-state in entrata (Bmdrin=1), ed infine scrivo il valore abilitando il segnale di scrittura (Wmdr=1).

j. (MAR)<-MDR:

i. Primo step



- 2) Abilito il buffer-tri-state di MAR per scrivere i dati sull'address bus (Bab=1)
- ii. Secondo step :



- 2) Per seconda micro-istruzione , abilito il control bus, ed in particolare la scrittura su se stesso : (Bab=1 e MWR=1)
- iii. Terzo step:



- 2) Abilito il buffer-tri-state di MDR ( $B_{MDROUT}=1$ ) così i dati viaggiano nel data bus.

`movl $0xaaaa, %eax:`

- $MAR \leftarrow RIP$
- $MDR \leftarrow (MAR); RIP \leftarrow RIP + 8$
- $IR \leftarrow MDR$
- $EAX \leftarrow IR[0:31]$



- b. Non c'è spiazzamento , e copia i 32 bit meno significativi di IR in RAX : seleziono  $S_{MUX}=2$  ,  $REG_{MUX}=1$  , abilito il segnale per scrivere sui registri ( $W_m=1$ ) e abilito anche il buffer-tri-state di IR ( $B_{short}=1$ ).

`movq $0xaaaa, %rax:`

- $MAR \leftarrow RIP$
- $MDR \leftarrow (MAR); RIP \leftarrow RIP + 8$
- $IR \leftarrow MDR$
- $MAR \leftarrow RIP$
- $MDR \leftarrow (MAR); RIP \leftarrow RIP + 8$
- $RAX \leftarrow MDR$

- a. Qui stessa costante di prima , ma a 64 bit, quindi abbiamo bisogno di 128 bit .
- b. In questo caso IR punta nel mezzo dell'istruzione !!
- c. Devo per forza trasferire alla memoria : devo sistemare anche il IR .

Quindi in base al tipo di operandi che si hanno , l'unità di controllo ha interazioni diverse (modalità di indirizzamento) : si cerca di ridurre il peso delle micro-operazioni , magari usando delle micro-

funzioni/micro-programma ( salvato nella ROM ) :

```

1 if D == 1 and Bp == 0 and Ip == 0
2   MAR ← IR[0:31]
3 else if D == 0 and Bp == 1 and Ip == 0
4   MAR ← B
5 else if Ip == 1
6   TEMP2 ← I
7   MAR ← SHIFTER_OUT[SHL, T]
8   TEMP1 ← MAR
9 if D == 1
10  TEMP2 ← IR[0:31]
11  MAR ← ALU_OUT[ADD]
12  TEMP1 ← MAR
13 endif
14 if Bp == 1
15  TEMP2 ← B
16  MAR ← ALU_OUT[ADD]
17 endif
18 endif

```

I microprogrammi vanno a finire nelle pagine vuote della ROM ( ve ne sono 8 "Libere"). Quindi di conseguenza devo modificare la mia macchina a stati , che diventa :



Mentre il circuito diventa così :



Vediamo ora altri esempi :

addw %ax, %cx:

- MAR  $\leftarrow$  RIP
  - MDR  $\leftarrow$  (MAR); RIP  $\leftarrow$  RIP + 8
  - IR  $\leftarrow$  MDR
  - TEMP1  $\leftarrow$  AX
  - TEMP2  $\leftarrow$  CX
  - CX  $\leftarrow$  ALU OUT[ADD]
- 1.



- b. Cominciamo con il configurare il mux delle sorgenti a 1 (Sreg=1), quello delle taglie a 0 (Regmux=0), abilitiamo il segnale di lettura del banco registri (Rm=1), abilitiamo il buffer tri-state del registro, così i dati viaggiano sull'internal bus data, arrivando a TEMP1 ed abilitiamo il segnale di scrittura (Wtemp1=1).





- d. visto che in questa micro-istruzione dobbiamo lavorare con un registro , settiamo i mux nei modi corretti : il primo (tipo) lo setto a dest( $S_{\text{MUX}}=1$ ), quello della taglia (secondo) lo setto ad 1 ( $\text{Regmux}=1$ )>registro base di dest , abilito il buffer-tri-state e lo mando a TEMP2, abilitando il segnale di scrittura su TEMP2( $W_{\text{temp2}}=1$ )



- f. Per quanto riguarda il settaggio dei mux è lo stesso di prima ; concentriamoci ora sull'ALU: gli si passa opcode 0000 , si abilita il buffer-tri-state che permette di scrivere sull'internal bus data (Ba=1), abilito il buffer-tri-state per scrivere nel registro flags(Bflagsa=1) ed abilito il segnale di scrittura su Rflags (Wrflags=1); infine per scrivere nel registro devo abilitare il segnale di scrittura nei registri (Wreg=1).

andw %ax, %cx:

- $\text{MAR} \leftarrow \text{RIP}$
  - $\text{MDR} \leftarrow (\text{MAR})$ ;  $\text{RIP} \leftarrow \text{RIP} + 8$
  - 2. •  $\text{IR} \leftarrow \text{MDR}$
  - $\text{TEMP1} \leftarrow \text{AX}$
  - $\text{TEMP2} \leftarrow \text{CX}$
  - $\text{CX} \leftarrow \text{ALU OUT[AND]}$
- a. Simile a quello di sopra , ma cambia l'opcode della ALU : da 0000 a 0110 :



jz displacement:

- MAR  $\leftarrow$  RIP
- MDR  $\leftarrow$  (MAR); RIP  $\leftarrow$  RIP + 8
- IR  $\leftarrow$  MDR
- 3. • IF FLAGS[ZF] == 1 THEN
  - TEMP1  $\leftarrow$  RIP
  - TEMP2  $\leftarrow$  IR[0:31]
  - RIP  $\leftarrow$  ALU OUT[ADD]
  - ENDIF



- b. Abilito il buffer-tri-state di RIP per scrivere sull'internal bus data, poi i dati arrivano a TEMP1 ed abilito la scrittura s questo registro : (Wtemp1=1)



- d. Prendo i 32 bit dello spiazzamento da IR[0:31], abilito il buffer-tri-state per scrivere sull'internal data bus , e lo scrivo su TEMP2 ed abilito il segnale di scrittura (Wtemp2=1).



- f. Infine alla ALU dando il codice della somma (Aluop=0000), abilito il buffer-tri-state(Ba=1) per scrivere sull'internal bus data , lo vado a scrivere nel RIP ed abilito la scrittura nel RIP(Wrip=1).

**Notiamo che si deve essere sicuri di aggiornare il registro flags se e solo se si hanno istruzioni aritmetiche.**