

## Laboratorio Sesión 06: Políticas de escritura

### Objetivo

El objetivo de esta sesión es asentar los conocimientos sobre las diferentes políticas de escritura de las memorias cache. Para hacerlo programaréis un simulador de cache básico que simule lecturas y escrituras.

### Características de la memoria cache



En esta sesión programaremos una memoria cache con las siguientes características:

- Las direcciones son de 32 bits (para simplificar asumiremos que todos los accesos son a bytes)
  - La cache será de mapeo directo
  - Tamaño de la cache: 4 Kbytes  $= 4096 \text{ bytes} \Rightarrow 4096 / 32 = 128 \text{ líneas NC} = 2^7 \Rightarrow 7 \text{ bits para } \#lnea\ NC$
  - Tamaño de la línea de cache: 32 bytes  $= 2^5 \text{ bytes} \Rightarrow 5 \text{ bits para el } \#byte$
  - Política de escritura: Write Through  $= \text{Actualiza HC y MP simultáneamente}$
  - Política de emplazamiento: Write NO Allocate  $= \text{Escritura directamente en MP en caso de fallo de escritura}$
- bits TAG =  $32 - 7 - 5 = 20$  bits

### Toma de contacto con el entorno simulador

El simulador se compone de 3 ficheros: CacheSimWT.o, CacheSim.h y MiSimulador.c. El programa principal y algunos componentes del simulador ya están programados y se encuentran en el fichero CacheSimWT.o. Este fichero se encarga de generar las secuencias de test, de imprimir los resultados de la simulación por pantalla con un formato agradable y de comprobar el correcto funcionamiento de vuestro simulador. Antes de que empiezas a programar el simulador, es interesante hacer algunas pruebas con este entorno. Para comenzar, compilad el simulador (MiSimulador.c no funciona correctamente, pero compila).

```
$> gcc -m32 CacheSimWT.o MiSimulador.c tiempo.c -o sim
```

El programa tiene 3 tests:

- Test 0: Genera la secuencia de 20 referencias de la tabla del trabajo previo
- Test 1: Genera accesos secuenciales a un vector de enteros (1000 referencias)
- Test 2: Genera los accesos de un producto de matrices de 25x25 (62500 referencias)

Para pasar cualquiera de los tests, sólo es necesario poner el nº de test como parámetro del simulador. Por ejemplo, para pasar el test 0 escribiríamos:

```
$> sim 0
Test 0 FAIL :-(
```

Evidentemente el test ha fallado, ya que aún no hemos programado el simulador. En caso de que el simulador falle, nos interesaría ver qué está pasando. Para ello podemos utilizar la opción v (de verbose) en el simulador (la v debe aparecer como primer parámetro):

```
$> sim v 0
```

```
eca130 L -> 1 MP:4212dfa0 1 MC:bffffe7a0 TAG:804a000 byte:40007dcf MISS L:134513233 E:1073774031 -> fd0
eca131 E -> 1 MP:4212dfa0 1 MC:bffffe7a0 TAG:804a000 byte:40007dcf MISS L:134513233 E:1073774031 -> fd0
ec2172 E -> 1 MP:4212dfa0 1 MC:bffffe7a0 TAG:804a000 byte:40007dcf MISS L:134513233 E:1073774031 -> fd0
...
Test 0 FAIL :-(
```

Esta opción nos dará una salida parecida a la anterior. Como podéis ver las columnas corresponden básicamente a la tabla del ejercicio previo. De esta forma, comparando la salida y la tabla podemos ver dónde está el problema. Dado que en los tests 1 y 2 el número de referencias es muy alto, os recomendamos que no los probéis hasta que os funcione perfectamente el test 0. Con la opción v, los tests 1 y 2 se paran tan pronto aparece el primer error para ayudar a su identificación.

## Programación del módulo MiSimulador.c

Para programar vuestro simulador de cache tenéis que programar 3 secciones del fichero `MiSimulador.c`:

1. **Estructuras globales** En esta sección tenéis que declarar las estructuras de datos globales necesarias para mantener el estado de la cache. Es necesario que sean globales, ya que la parte principal del simulador es la rutina `reference` que se ejecuta una vez por referencia y, como ya sabéis, su estado desaparece una vez se ejecuta.
2. **Inicialización de la cache** La rutina `init_cache` se llama antes de pasar cada test para inicializar las estructuras de datos globales necesarias. El objetivo es dejar la cache en un estado inicial correcto (cache vacía).
3. **Simulación de referencias** La simulación de las referencias tenéis que hacerla en la rutina `reference`. Esta rutina se llama una vez por cada referencia a simular. Solo es necesario que generéis el valor correcto de las 11 variables locales que ya tenéis declaradas al inicio de la subrutina y que se corresponden básicamente a las columnas de la tabla del trabajo previo (excepto el booleano `replacement`, que no era necesario en el trabajo previo).

```
void reference (unsigned int address, unsigned int LE)
{
    unsigned int byte;
    unsigned int bloque_m;
    unsigned int linea_mc;
    unsigned int tag;
    unsigned int miss; // booleano que indica si es miss
    unsigned int lec_mp; // booleano que indica si se lee de MP
    unsigned int mida_lec_mp;
    unsigned int esc_mp; // booleano que indica si se escribe en MP
    unsigned int mida_esc_mp;
    unsigned int replacement; // booleano que indica si se reemplaza
                             // una línea válida
    unsigned int tag_out; // TAG de la línea reemplazada
```

En otras palabras, lo que tenéis que hacer es implementar el algoritmo que, de forma intuitiva, habéis hecho servir manualmente para llenar la tabla del estudio previo. Después de vuestro código, la rutina acaba con una llamada a la rutina `test_and_print` para comprobar si los valores de las variables son correctos e imprimirlos por pantalla en caso de tener la opción v activada.

4. **Impresión de resultados** Si tenéis que imprimir algún resultado final, la rutina `final` se ejecuta una vez al finalizar el programa.

## Estudio Previo

1. Rellenad la tabla de la hoja de respuestas indicando, para cada referencia de la secuencia de referencias, la información siguiente (en hexadecimal):
  - el byte de la línea a que se accede (byte)
  - el bloque de memoria (bloque M)
  - la línea de memoria cache donde se mapeará la referencia (línea MC)

- la etiqueta (TAG) que se guardará de esta referencia
- si el acceso es HIT o MISS,
- si hay una lectura de memoria principal, indicad la cantidad de bytes leídos (lec MP),
- si hay una escritura en memoria principal, indicad la cantidad de bytes escritos (esc MP),
- y en caso de que se reemplace una línea válida, el TAG de la línea reemplazada (TAG out).

La memoria cache tiene las siguientes características:

- Las direcciones son de 32 bits (para simplificar asumiremos que todos los accesos son a bytes)
- La cache será de mapeo directo
- Tamaño de la cache: 4 Kbytes
- Tamaño de la línea de cache: 32 bytes
- Política de escritura: Write Through
- Política de emplazamiento: Write NO Allocate

2. Rellenad la tabla de la hoja de respuestas indicando, para cada referencia de la secuencia de referencias, la información siguiente (en hexadecimal):

- el byte de la línea a que se accede (byte)
- el bloque de memoria (bloque M)
- la línea de memoria cache donde se mapeará la referencia (línea MC)
- la etiqueta (TAG) que se guardará de esta referencia
- si el acceso es HIT o MISS,
- si hay una lectura de memoria principal, indicad la cantidad de bytes leídos (lec MP),
- si hay una escritura en memoria principal, indicad la cantidad de bytes escritos (esc MP),
- y en caso de que se reemplace una línea válida, el TAG de la línea reemplazada (TAG out).

La memoria cache tiene las siguientes características:

- Las direcciones son de 32 bits (para simplificar asumiremos que todos los accesos son a bytes)
- La cache será de mapeo directo
- Tamaño de la cache: 4 Kbytes
- Tamaño de la línea de cache: 32 bytes
- Política de escritura: Copy Back
- Política de emplazamiento: Write Allocate

## Trabajo a realizar durante la Práctica

1. Programad una versión del simulador de cache de lectura/escritura con política Write Through + Write NO Allocate y comprobad su correcto funcionamiento. Cuando funcione entregad el fichero MiSimulador.c en el Racó de la asignatura.
2. Implementad también un simulador con el mismo funcionamiento pero con política de escritura Copy Back + Write Allocate en el fichero MiSimulador2.c. Para comprobar su funcionamiento utilizad el fichero CacheSimCB.o. Cuando funcione entregad el fichero MiSimulador2.c en el Racó de la asignatura.

3. Modificad vuestros simuladores de cache para contar cuántos accesos aciertan y fallan en cache en una ejecución del programa. Podéis imprimir el resultado poniendo el código en la rutina `final`. Averiguad qué política de escritura es mejor para los accesos que ejecuta el `test 2`.
4. Recordad entregar los ficheros `MiSimulador.c` y `MiSimulador2.c` en el Racó de la asignatura. Debéis entregar sólo los dos ficheros fuentes, sin comprimir ni cambiarles el nombre, y sólo una versión por pareja de laboratorio (es indistinto que miembro de la pareja entregue).

Nombre: \_\_\_\_\_

Grupo: \_\_\_\_\_

Nombre: \_\_\_\_\_

## Hoja de respuesta al Estudio Previo

- La memoria cache tiene las siguientes características:

- Las direcciones son de 32 bits (para simplificar asumiremos que todos los accesos son a bytes)
- La cache será de mapeo directo
- Tamaño de la cache: 4 Kbytes
- Tamaño de la línea de cache: 32 bytes
- Política de escritura: Write Through
- Política de emplazamiento: Write NO Allocate

Rellenad la siguiente tabla (en hexadecimal):

| @        | L/E | byte | bloque M | línea MC | TAG   | H/M | lec MP | esc MP | TAG out |
|----------|-----|------|----------|----------|-------|-----|--------|--------|---------|
| 10f92150 | L   | 10   | 087C90A  | 0A       | 10F92 | M   | 32     | —      | —       |
| 10f92151 | E   | 11   | 087C90A  | 0A       | 10F92 | H   | —      | 1      | —       |
| 10f8a192 | E   | 12   | 087C50C  | 0C       | 10F8A | M   | —      | 1      | —       |
| 10f92153 | L   | 13   | 087C90A  | 0A       | 10F92 | H   | —      | —      | —       |
| 10f8b195 | L   | 15   | 087C58C  | 0C       | 10F8B | M   | 32     | —      | —       |
| 10f8b195 | L   | 15   | 087C58C  | 0C       | 10F8B | H   | —      | —      | —       |
| 10f93156 | L   | 16   | 087C90A  | 0A       | 10F93 | M   | 32     | —      | 10F92   |
| 10f92157 | E   | 17   | 087C90A  | 0A       | 10F92 | M   | —      | 1      | —       |
| 10f8a198 | L   | 18   | 087C50C  | 0C       | 10F8A | H   | 32     | —      | 10F8B   |
| 10f93159 | E   | 19   | 087C90A  | 0A       | 10F93 | H   | —      | 1      | —       |
| 12f92250 | L   | 10   | 097C902  | 12       | 12F92 | M   | 32     | —      | —       |
| 10f92151 | E   | 11   | 087C90A  | 0A       | 10F92 | H   | —      | 1      | —       |
| 10f8a192 | L   | 12   | 087C50C  | 0C       | 10F8A | H   | —      | —      | —       |
| 12f92253 | E   | 13   | 097C912  | 12       | 12F92 | H   | —      | 1      | —       |
| 10f8b195 | E   | 15   | 087C58C  | 0C       | 10F8B | M   | —      | 1      | —       |
| 10f8b195 | L   | 15   | 087C58C  | 0C       | 10F8B | H   | 32     | —      | 10F8A   |
| 10f93156 | L   | 16   | 087C90C  | 0A       | 10F93 | H   | —      | —      | —       |
| 12f92257 | L   | 17   | 097C912  | 12       | 12F92 | H   | —      | —      | —       |
| 10f8a298 | L   | 18   | 087C514  | 14       | 10F8A | H   | 32     | —      | —       |
| 10f93159 | L   | 19   | 087C90A  | 0A       | 10F93 | H   | —      | —      | —       |

2. La memoria cache tiene las siguientes características:

- Las direcciones son de 32 bits (para simplificar asumiremos que todos los accesos son a bytes)
- La cache será de mapeo directo
- Tamaño de la cache: 4 Kbytes = 4096 bytes  $\Rightarrow 4096/32 = 128 = 2^7$  bytes  $\Rightarrow 7$  bits #línea MC
- Tamaño de la línea de cache: 32 bytes = 2<sup>5</sup> bytes  $\Rightarrow 5$  bits #byte
- Política de escritura: Copy Back = El dato se actualiza solo en la MC
- Política de emplazamiento: Write Allocate = Se trae de MP a MC y luego se escribe

Rellenad la siguiente tabla (en hexadecimal):

| @        | L/E | byte | bloque M | línea MC | TAG   | H/M | lec MP | esc MP | TAG out |
|----------|-----|------|----------|----------|-------|-----|--------|--------|---------|
| 10f92150 | L   | 10   | 087C90A  | 0A       | 10F92 | H   | 32     | -      | -       |
| 10f92151 | E   | 11   | 087C90A  | 0A       | 10F92 | H   | -      | -      | -       |
| 10f8a192 | E   | 12   | 087C50C  | 0C       | 10F8A | H   | 32     | -      | -       |
| 10f92153 | L   | 13   | 087C90A  | 0A       | 10F92 | H   | -      | -      | -       |
| 10f8b195 | L   | 15   | 087C58C  | 0C       | 10F8B | H   | 32     | 32     | 10F8A   |
| 10f8b195 | L   | 15   | 087C58C  | 0C       | 10F8B | H   | -      | -      | -       |
| 10f93156 | L   | 16   | 087C98A  | 0A       | 10F93 | H   | 32     | 32     | 10F92   |
| 10f92157 | E   | 17   | 087C90A  | 0A       | 10F92 | H   | 32     | -      | 10F93   |
| 10f8a198 | L   | 18   | 087C50C  | 0C       | 10F8A | H   | 32     | -      | 10F8B   |
| 10f93159 | E   | 19   | 087C98A  | 0A       | 10F93 | H   | 32     | 32     | 10F92   |
| 12f92250 | L   | 10   | 097C912  | 12       | 12F92 | M   | 32     | -      | -       |
| 10f92151 | E   | 11   | 087C90A  | 0A       | 10F92 | H   | 32     | 32     | 10F93   |
| 10f8a192 | L   | 12   | 087C50C  | 0C       | 10F8A | H   | -      | -      | -       |
| 12f92253 | E   | 13   | 097C912  | 12       | 12F92 | H   | -      | -      | -       |
| 10f8b195 | E   | 15   | 087C58C  | 0C       | 10F8B | M   | 32     | -      | 10F8A   |
| 10f8b195 | L   | 15   | 087C58C  | 0C       | 10F8B | H   | -      | -      | -       |
| 10f93156 | L   | 16   | 087C98C  | 0A       | 10F93 | M   | 32     | 32     | 10F92   |
| 12f92257 | L   | 17   | 097C912  | 12       | 12F92 | H   | -      | -      | -       |
| 10f8a298 | L   | 18   | 087C514  | 14       | 10F8A | H   | 32     | -      | -       |
| 10f93159 | L   | 19   | 087C98A  | 0A       | 10F93 | H   | -      | -      | -       |