Universidade Federal De Campina Grande Departamento De Engenharia Elétrica Laboratório De Arquitetura De Sistemas Digitais – LASD Prof. Rafael B. C. Lima



| Aluno:     |       |
|------------|-------|
| Matrícula: | Data: |

**Sprint 6** – Memórias – CPU MIPS

**Descrição geral do problema**: Utilizando a ferramenta *megawizard plugin-manager*, do Quartus II, implemente uma nova memória ROM de instruções, assim como uma memória RAM de dados. Isso possibilitará o uso das instruções LW e SW.

### Requisitos mínimos:

Abra o projeto da Sprint5 e edite-o para incluir as funcionalidades dessa sprint. **Obs: "File > Open Project"** e NÃO "File > Open".

1. Criação de uma nova memória ROM de instruções, para substituir a memória implementada como um decodificador, na sprint anterior.

## Criação do arquivo de inicialização da memória ROM

- File/New/Memory Initialization File
- Number of words: 256
- Word size: **32**
- Salvar o arquivo com o seguinte nome: RomInstMem\_init.mif
- View/Cells Per Row: 1
- View/Memory Radix: Binary
- Copie e cole os binários das instruções a serem executadas no programa de teste da Tabela 1

### Criação da memória ROM de instruções

- tools/megawizard plugin-manager
- Create a new custom megafunction variation
- Selecione a opção de memória ROM de uma porta (ROM: 1-PORT)
- Selecione sua pasta pessoal e nomeie o arquivo como *RomInstMem*
- Largura da palavra (32 bits) e a quantidade de palavras da memória (256 palavras)
- Desmarque a opção 'q' output port
- Inicialize a memória com o arquivo RomInstMem\_init.mif criado anteriormente
- Finish



Perceba que foi criado um arquivo *RomInstMem.qip* na árvore de arquivos do seu projeto. Clique na seta ">" e visualize o arquivo *RomInstMem.v* 

| Endereço | Assembly          | Código de máquina               |
|----------|-------------------|---------------------------------|
| Endereço | Assembly          | Coulgo de maquina               |
| 00       | ADDi \$1, \$0, AB | 001000000000001000000010101011  |
| 01       | SW \$1, A(\$0)    | 1010110000000010000000000001010 |
| 02       | LW \$2, A(\$0)    | 1000110000000100000000000001010 |
| 03       | SW \$2, B(\$0)    | 1010110000000100000000000001011 |
| 04       | LW \$3, B(\$0)    | 100011000000011000000000001011  |
| 05       | SW \$3, C(\$0)    | 101011000000011000000000001100  |
| 06       | LW \$4, C(\$0)    | 100011000000100000000000001100  |

2. Criação de uma nova memória RAM de dados. A partir de agora, será possível armazenar e carregar o conteúdo de registradores, através das instruções SW e LW.

clk

Data

Mem.

Address

Data

wren

q

### Criação da memória RAM de dados

- tools/megawizard plugin-manager
- Create a new custom megafunction variation
- Selecione a opção de memória RAM de uma porta (RAM: 1-PORT)
- Selecione sua pasta pessoal e nomeie o arquivo como RamDataMem
- Largura da palavra (8 bits) e a quantidade de palavras da memória (256 palavras)
- Desmarque a opção 'q' output port
- Finish



- 3. A fim de completar a segunda versão da CPU v0.2, todos os módulos desenvolvidos até agora devem ser **instanciados** e **conectados** conforme o circuito da Figura 1.
  - Substitua a memória de instruções implementada na sprint anterior pela memória ROM criada agora (módulo RomInstMem.v)
  - Instancie o MuxDDest
  - Instancie a memória RAM de dados (módulo RamDataMem.v)
  - Os fios devem ser criados com uma nomenclatura lógica para facilitar o entendimento geral do circuito e facilitar o debug.
  - Essa CPU v0.2 será capaz de rodar as instruções ADD, SUB, AND, OR, SLT, ADDi, LW e SW.



Figura 1 - CPU V0.2 e memórias

| Nome       | Tamanho |
|------------|---------|
| w_MemtoReg | 1 bit   |
| w_MemWrite | 1 bit   |
| w_wd3      | 8 bit   |
| w_RData    | 8 bit   |

Tabela 2 – novos fios, utilizados na montagem

ALERTA: A CPU de ciclo único necessita de memórias puramente combinacionais. Como a ferramenta do Quartus II somente cria memórias com no mínimo um registro na entrada, o comportamento combinacional será emulado conectando um clock muito mais rápido nessas memórias.

O sinal de 50 MHz, que é disponibilizado na porta de entrada CLOCK\_50 do *Mod\_teste*, deverá ser ligado às portas de entrada *"clock"* de ambas as memórias. Já o clock principal da CPU (PC e Register File), deverá ser de 1Hz, gerado a partir de um divisor de frequência. Os pontos de clock estão resumidos na Tabela 3.

| Módulo             | Frequência do clock |
|--------------------|---------------------|
| PC                 | 1 Hz                |
| Register File      | 1 Hz                |
| Instruction Memory | 50 MHz              |
| Data Memory        | 50 MHz              |

Tabela 3 – Frequência dos clocks

Relembrando da tabela do decodificador de instruções:

|       | ENTR   | ADAS   | SAÍDAS   |        |        |            |        |          |          |      |
|-------|--------|--------|----------|--------|--------|------------|--------|----------|----------|------|
| Instr | OP     | Funct  | RegWrite | RegDst | ULASrc | ULAControl | Branch | MemWrite | MemtoReg | Jump |
| ADD   | 000000 | 100000 | 1        | 1      | 0      | 010        | 0      | 0        | 0        | 0    |
| SUB   | 000000 | 100010 | 1        | 1      | 0      | 110        | 0      | 0        | 0        | 0    |
| AND   | 000000 | 100100 | 1        | 1      | 0      | 000        | 0      | 0        | 0        | 0    |
| OR    | 000000 | 100101 | 1        | 1      | 0      | 001        | 0      | 0        | 0        | 0    |
| NOR   | 000000 | 100111 | 1        | 1      | 0      | 011        | 0      | 0        | 0        | 0    |
| SLT   | 000000 | 101010 | 1        | 1      | 0      | 111        | 0      | 0        | 0        | 0    |
| LW    | 100011 | XXXXXX | 1        | 0      | 1      | 010        | 0      | 0        | 1        | 0    |
| SW    | 101011 | XXXXXX | 0        | Х      | 1      | 010        | 0      | 1        | х        | 0    |
| BEQ   | 000100 | XXXXXX | 0        | Х      | 0      | 110        | 1      | 0        | х        | 0    |
| ADDi  | 001000 | XXXXXX | 1        | 0      | 1      | 010        | 0      | 0        | 0        | 0    |
| J     | 000010 | XXXXXX | 0        | Х      | Х      | XXX        | Х      | 0        | х        | 1    |

Tabela 4 – Tabela do decodificador da Unidade de Controle

Relembrando o conjunto de instruções suportadas pela CPU

| Instrução         | Descrição            | Algoritmo                      |
|-------------------|----------------------|--------------------------------|
| ADD \$X, \$Y, \$Z | Adicionar            | \$X = \$Y + \$Z                |
| SUB \$X, \$Y, \$Z | Subtrair             | \$X = \$Y - \$Z                |
| AND \$X, \$Y, \$Z | AND Bit a bit        | \$X = \$Y & \$Z                |
| OR \$X, \$Y, \$Z  | OR Bit a bit         | \$X = \$Y   \$Z                |
| NOR \$X, \$Y, \$Z | NOR Bit a bit        | \$X = ~(\$Y   \$Z)             |
| SLT \$X, \$Y, \$Z | Menor que            | \$X = 1 se \$Y < \$Z e 0 c.c.  |
| LW \$X, i(\$Y)    | Carregar da memória  | \$X <= Cont. do end. (\$Y+ i)  |
| SW \$X, i(\$Y)    | Armazenar na memória | End. (\$Y+ i) <= \$X           |
| BEQ \$X, \$Y, i   | Desviar se igual     | Se \$X == \$Y, PC = PC + 1 + i |
| ADDi \$X, \$Y, i  | Adicionar Imediato   | \$X = \$Y + i                  |
| Ji                | Desvio incondicional | PC = i                         |

Tabela 5 – Conjunto de instruções MIPS suportadas pela CPU do LASD



Figura 5 - Placa

4. Rode o programa da Tabela 1 e diga qual o conteúdo dos registradores e da memória de dados, ao finalizálo:

#### Registradores:

| Registrador | \$0 | \$1 | \$2 | \$3 | \$4 | \$5 | \$6 | \$7 |
|-------------|-----|-----|-----|-----|-----|-----|-----|-----|
| Dado        |     |     |     |     |     |     |     |     |

### Memória de dados:

| Endereço | 0x00 | 0x01 | 0x02 | 0x03 | <br>0x0A | 0x0B | 0x0C | 0x0D |
|----------|------|------|------|------|----------|------|------|------|
| Dado     |      |      |      |      |          |      |      |      |

# Desafio (Valendo +0,2 na média geral)

- Crie uma lookup table (LUT), na memória de dados, que retorne o valor decodificado em 7-segmentos, correspondente ao endereço da sua respectiva posição.
- Perceba que essa LUT é uma maneira de implementar por software, o decodificador de hexa-7seg que você criou, em hardware, na sprint 2.
- Para testar seu código, conecte o display HEXO, na saída de debug do registrador \$7.
- Entreva um programa em assembly para printar, em HEXO, os números de 0-9, em sequência.
- OBS: Não utilize o decodificador hexa-7seg. Resolva o problema por software.