

# Proyecto Optativo 1:



**Gestión de riesgos en MIPS segmentado  
con Cache de instrucciones**

Miguel Allué Barón - 593599

Cristian Simón Moreno - 611487

## Paso 1: Señales de control a la UC

| Operación         | Codificación |
|-------------------|--------------|
| <b>NOP</b>        | 000000       |
| <b>ARITMETICA</b> | 000001       |
| <b>LOAD</b>       | 000010       |
| <b>STORE</b>      | 000011       |
| <b>BEQ</b>        | 000100       |

| Op_code     | Branch | RegDst | ALUSrc | MemWrite | MemRead | MemtoReg | RegWrite |
|-------------|--------|--------|--------|----------|---------|----------|----------|
| <b>NOP</b>  | 0      | 0      | 0      | 0        | 0       | 0        | 0        |
| <b>ARIT</b> | 0      | 1      | 0      | 0        | 0       | 0        | 1        |
| <b>LW</b>   | 0      | 0      | 1      | 0        | 1       | 1        | 1        |
| <b>SW</b>   | 0      | 0      | 1      | 1        | 0       | 0        | 0        |
| <b>BEQ</b>  | 1      | 0      | 0      | 0        | 0       | 0        | 0        |

## Paso 2: Conexiones de la MC

Estructura de la dirección (ADDR):



- **dir\_MI:** Lo usamos para acceder a la memoria principal de 128 palabras.  
 $dir\_MI \leq ADDR (8 \text{ downto } 4) \& palabra\_UC$   
 palabra\_UC -> nos permite traernos 4 bloques (00, 01, 10, 11)
- **dir\_palabra:** Lo usamos para elegir la instrucción solicitada de un determinado bloque.  
 $dir\_palabra \leq ADDR (3 \text{ downto } 2)$
- **dir\_cjto:** Lo usamos para elegir el conjunto al que se accede en la cache de instrucciones.  
 $dir\_cjto \leq ADDR (5 \text{ downto } 4)$
- **TAG:**  $ADDR (31 \text{ downto } 6)$

## Unidad de control de la MC



## Paso 3: Detención del segmentado

Tenemos 3 tipos de riesgos que debemos tratar:

### 1) Por riesgos de datos:

Para detectarlas deberemos comparar los operandos de la instrucción actual con los destinos de las instrucciones que tendremos en las etapas EX y MEM, es decir:

#### En la etapa EX:

- IR\_ID(25 downto 21) = RW\_EX
- IR\_ID(20 downto 16) = RW\_EX

#### En la etapa MEM:

- IR\_ID(25 downto 21) = RW\_MEM
- IR\_ID(20 downto 16) = RW\_MEM

Además de esta comparación deberemos comprobar el **tipo de instrucción** que tendremos delante, para evitar hacer paradas innecesarias. Para comprobar el tipo de

instrucción que tendremos en las etapas EX y MEM usaremos las señales **RegWrite** (en EX y MEM), **RegDst\_EX**, **MemtoReg\_MEM** y **RegWrite\_MEM**.

Para realizar la gestión de estos riesgos lo primero que hemos hecho ha sido hacer un análisis de que instrucciones pueden presentar riesgos, y serán aquellas en las que se escriba en el banco de registros, ya que si la siguiente instrucción va a leer el destino de la operación anterior y aun no lo hemos escrito, obtendremos un valor erróneo.

| F | D | EX | MEM | WB |    |     |    |
|---|---|----|-----|----|----|-----|----|
|   | F | D  | D   | D  | EX | MEM | WB |

Deberemos esperar a que se escriba en el banco de registros para poder leer el dato en la siguiente instrucción.

Por lo tanto tendremos riesgos si en la instrucción anterior detectamos un **LOAD** o una **ARITMETICA**. El código que hemos realizado para la detección y gestión de riesgos de datos es el siguiente:

```
|DetectarRiesgos: process (IR_ID, RegWrite_EX, RegWrite_MEM, MemtoReg_MEM, RW_EX, RW_MEM, riesgo_rs_d1, riesgo_rs_d2, riesgo_rt_d1, riesgo_rt_d2)
begin
    riesgo_rs_d1 <= '0'; --riesgo de datos en rs con la instrucción que hay en EX.
    riesgo_rs_d2 <= '0'; --riesgo de datos en rs con la instrucción que hay en Mem
    riesgo_rt_d1 <= '0'; --riesgo de datos en rt con la instrucción que hay en EX
    riesgo_rt_d2 <= '0'; --riesgo de datos en rt con la instrucción que hay en Mem
    -- STORE, ARIT Y BEQ -> Riesgos con LOAD Y ARIT
    if (IR_ID(31 downto 26) = "000011" OR IR_ID(31 downto 26)= "000001" OR IR_ID(31 downto 26)= "000100") then
        --EX / Comprobamos si es un un load o aritmetica la de EX
        if(RegWrite_EX='1') then
            if(IR_ID(25 downto 21) = RW_EX) then
                riesgo_rs_d1 <= '1';
            end if;
            if(IR_ID(20 downto 16) = RW_EX) then
                riesgo_rt_d1 <= '1';
            end if;
        end if;
        --MEM / Comprobamos si es un un load o aritmetica la de MEM
        if(RegWrite_MEM='1') then
            if(IR_ID(25 downto 21) = RW_MEM) then
                riesgo_rs_d2 <= '1';
            end if;
            if(IR_ID(20 downto 16) = RW_MEM) then
                riesgo_rt_d2 <= '1';
            end if;
        end if;
    end if;
```

```

-- LOAD - Riesgos con ARIT
elsif (IR_ID(31 downto 26)= "000010") then
    --EX / Comprobamos si es una aritmética la de EX
    if(regDst_EX='1') then
        if(IR_ID(20 downto 16) = RW_EX) then
            riesgo_rt_d1 <= '1';
        end if;
    end if;
    --MEM / Comprobamos si es una aritmética la de MEM
    if(MemtoReg_MEM='0' AND RegWrite_MEM='1') then
        if(IR_ID(20 downto 16) = RW_MEM) then
            riesgo_rt_d2 <= '1';
        end if;
    end if;
end if;

-- COMPROBAMOS SI HA HABIDO ALGUN RIESGO
if (riesgo_rs_d1 = '1' OR riesgo_rs_d2 = '1' OR riesgo_rt_d1 = '1' OR riesgo_rt_d2 = '1') then
    avanzar_ID <= '0';
end if;
if (riesgo_rs_d1 = '0' AND riesgo_rs_d2 = '0' AND riesgo_rt_d1 = '0' AND riesgo_rt_d2 = '0') then
    avanzar_ID <= '1';
end if;
end process;

```

Primero pondremos las señales (**riesgo\_rs\_d1**, **riesgo\_rs\_d2**, **riesgo\_rt\_d1** y **riesgo\_rt\_d2**) que nos servirán para indicar si hay riesgos a 0, después los hemos dividido en 2 partes:

- Si detectamos que la operación actual es un **Store**, **Aritmética** o un **Beq**, comprobaremos que si en la etapa EX o MEM hay un Load o una Aritmética y si la hay, y los operandos y los destinos son iguales pondremos la señal correspondiente de riesgo a 1.
- Si detectamos que la operación actual es un **Load**, comprobaremos si en la etapa EX o MEM hay una Aritmética y si la hay, y los operandos y los destinos son iguales pondremos la señal correspondiente de riesgo a 1.

Luego comprobaremos si hemos detectado algún riesgo, si hemos detectado alguno pondremos **avanzar\_ID a 0**, por lo que detendremos la etapa Decode, y también deberemos detener la etapa Fetch, ya que si no la instrucción guardada en Decode la perderemos.

## Un ejemplo de riesgo de control:

Tenemos las siguientes instrucciones (Operación - hexadecimal - binario - explicación):

|                               |                           |                   |
|-------------------------------|---------------------------|-------------------|
| 0 - LDI R0, [#0] -> 0800 0000 | 000010 00000 00000 000... | R0 <- 2           |
| 1 - LDI R1, [#4] -> 0801 0004 | 000010 00000 00001 000... | R1 <- 1           |
| 2 - ADD R0, R1 -> 0420 0000   | 000001 00001 00000 000... | R0 <- R0+R1=2+1=3 |

Por lo tanto al llegar a la instrucción 2 (ADD R0, R1 ) podemos tener riesgos en 2 etapas:

1. Cuando la instrucción 0 (LDI R0, [#0] ) este en la etapa EX y la instrucción 1 (LDI R1, [#4] ) este en la etapa MEM.
  - a. Con la instrucción 0 -> rt (20..16) de 0 = rt (20..16) de 2 -> activamos **riesgo\_rt\_d2**
  - b. Con la instrucción 1 -> rt (20..16) de 1 = rs (25..20) de 2 -> activamos **riesgo\_rs\_d1**
2. Cuando la instrucción 1 este en la etapa MEM.
  - a. Con la instrucción 1 -> rt (20..16) de 1 = rs (25..20) de 2 -> activamos **riesgo\_rs\_d2**

Lo podemos ver mejor en el cronograma:



Como vemos cargamos los 2 loads, y se detectan los riesgos del add, primero cuando uno está en la etapa EX y el otro en la MEM, y en el siguiente ciclo cuando hay solo un load en la etapa MEM.

## 2) Por riesgos de control y fallos en la MC:

La realizaremos en la etapa Fetch, con el siguiente código:

```
-- SI SALTO O M_I LISTA -> LOAD_PC = 1
actualizarPC: process (Branch, MI_ready, avanzar_ID)
begin
    if ((Branch = '1' OR MI_READY = '1') AND avanzar_ID = '1') then
        load_PC <= '1';
        avanzar_F <= '1';
    else
        load_PC <= '0';
        avanzar_F <= '0';
    end if;
end process;
```

Con esto lo que haremos será si tenemos salto (**Branch = 1**), o la memoria de instrucciones tiene la instrucción lista (**MI\_READY = 1**) y en Decode avanzamos (**avanzar\_ID = 1**), cargaremos PC y avanzaremos en Fetch. En caso contrario pararemos, si no hay que saltar y la memoria no está preparada paramos PC, ya que si no hemos cargado PC, no debemos avanzar a PC+4

## Caso de prueba:

### Instrucciones:

|                                 |                             |
|---------------------------------|-----------------------------|
| 0 - LDI R0, [#0] -> 0800 0000   | $R0 \leftarrow 2$           |
| 1 - LDI R1, [#4] -> 0801 0004   | $R1 \leftarrow 1$           |
| 2 - ADD R0, R1 -> 0420 0000     | $R0 \leftarrow R0+R1=2+1=3$ |
| 3 - ADD R0, R0 -> 0400 0000     | $R0 \leftarrow R0+R0=3+3=6$ |
| 4 - LDI R1, [#8] -> 0801 0008   | $R1 \leftarrow 3$           |
| 5 - ADD R0, R1 -> 0420 0000     | $R0 \leftarrow R0+R1=6+3=9$ |
| 6 - STI R0, [#12] -> 0C00 000C  | $\#12 \leftarrow 9$         |
| 7 - BEQ R0, R0, #0 -> 1000 FFF8 | Salto a la #0               |



### Contenido de memoria:

#0 -> 2

#4 -> 1

#8 -> 3

### Riesgos:

Este código nos presenta multitud de dependencias:

- Los 2 loads iniciales (Instrucciones 0 y 1) con la instrucción 2 (No podemos hacer la operación hasta que no tengamos los 2 registros con los valores correctos y listos para ser leídos).
- La instrucción 2 con la 3 (2 ADD que comparten operandos, por lo que hasta que no escribamos en el banco el resultado de la primera operación no podremos usarlo en la segunda).
- La instrucción 4 con la 5 (No podemos hacer la operación hasta que no tengamos los 2 registros con los valores correctos y listos para ser leídos).

- La 5 con la 6 (Debemos esperar a que el dato este escrito en el banco antes de cogerlo y almacenarlo en memoria).

Con este caso de prueba comprobaremos casi todos los posibles riesgos existentes. Por riesgo de datos, por ejemplo tendremos el de los load con las aritméticas:



Podemos ver cómo tras ejecutar los 2 loads iniciales del código (0800 0000 y 0801 0000), se detectan riesgos para la ejecución de la aritmética (0420 0000) por lo que se detiene la etapa Decode mandando NOP, deteniendo también la etapa Fetch. Una vez desaparecidos los riesgos se hace la aritmética.



Si no hay que saltar o la memoria no está preparada paramos PC, ya que si no hemos cargado PC, no debemos avanzar a PC+4. Mientras no esté lista la memoria mandaremos NOPs.

|                                   |               |               |               |               |                 |                 |                 |
|-----------------------------------|---------------|---------------|---------------|---------------|-----------------|-----------------|-----------------|
| /mips_test/MIPS/dk                | 1             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/ready             | 0             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/load_PC           | 0             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/Branch            | 0             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/PC_in             | 00000014      | 00000000      | 00000001      | 00000008      | 0000000C        | 00000010        |                 |
| /mips_test/MIPS/PC_out            | 00000010      | 00000000      | 00000000      | 00000004      | 00000008        | 0000000C        |                 |
| /mips_test/MIPS/Op_code_ID        | 000000        | 000011        | 000100        | 000000        | 000010          | 000000          |                 |
| /mips_test/MIPS/PC_out            | 00000020      | 00000000      | 08000000      | 08010004      | 04200000        | 04000000        |                 |
| /mips_test/MIPS/BusA_EX           | 01000000      | 09000000      |               |               |                 |                 |                 |
| /mips_test/MIPS/Mux_out           | 02000000      | 0000000C      | 09000000      |               | 00000000        | 00000004        |                 |
| /mips_test/MIPS/ALU_out_EX        | 03000000      | 0900000C      | 12000000      |               | 09000000        | 09000004        |                 |
| /mips_test/MIPS/MemRead_MEM       | 0             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/MemWrite_MEM      | 0             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/Mem_D/RAM         | (02000000...) | (02000000...) | (01000000...) | (03000000...) | (FFFFFFFFFF...) | (FFFFFFFFFF...) | (FFFFFFFFFF...) |
| /mips_test/MIPS/ReqWrite_WB       | 0             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/Register_bank/... | (02000000...) | (05000000...) | (03000000...) | (00000000...) | (09000000...)   | (03000000...)   | (09000000...)   |
| /mips_test/MIPS/hesgo_rs_d1       | 1             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/hesgo_rs_d2       | 0             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/hesgo_rt_d1       | 1             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/hesgo_rt_d2       | 0             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/avanzar_F         | 0             |               |               |               |                 |                 |                 |
| /mips_test/MIPS/avanzar_ID        | 0             |               |               |               |                 |                 |                 |

Si hay que saltar actualizamos el PC, en este caso la memoria casi no tarda en tener listos los datos porque ya están en la cache.