# Projeto de Processador REDUX-V

# Pietro Comin - GRR20241955 5 de maio de 2025

#### Resumo

Este relatório documenta o projeto do processador REDUX-V, abordando sua arquitetura, conjunto de instruções, componentes principais como a ULA e a memória de controle, além da justificativa para instruções estendidas adicionadas.

## 1 Introdução

Este relatório apresenta o desenvolvimento completo da arquitetura REDUX-V, uma arquitetura de 8 bits do tipo load-store, com organização monociclo e endereçamento por byte. O projeto foi implementado no Logisim Evolution, utilizando memória RAM do tipo dual\_port\_ram, e com clock de dois ticks por ciclo.

Por praticidade e facilidade nos testes, foram adicionadas duas memórias ROM dentro do componente dual\_port\_ram, que contêm o hexadecimal das instruções de ambos os trabalhos desenvolvidos até então na disciplina. Na implementação do código, o fato de haver uma ou duas unidades de memória não é importante, pois somente são armazenados dados na RAM a partir do endereço final das instruções contidas na ROM.

Além das instruções da arquitetura base, foram implementadas três novas instruções utilizando opcodes reservados: jien, muli e brzrue. Estas instruções, que serão melhor detalhadas mais à frente, visam ampliar a flexibilidade do programa de teste, com destaque para saltos condicionais estendidos e multiplicação com imediato.

# 2 Conjunto de Instruções Base

A arquitetura REDUX-V suporta um conjunto enxuto de instruções, como:

- Operações lógicas: and, or, not, xor
- Operações aritméticas: add, sub, addi, slr, srr
- Controle de fluxo: ji (jump immediate), brzr (branch if zero)
- Manipulação de registradores e memória: load, store

# 3 Datapath do Processador



Figura 1: Datapath do processador REDUX-V

O datapath define o caminho de dados percorrido durante a execução de uma instrução. Ele inclui componentes como o banco de registradores, a ULA, multiplexadores, unidade de controle, extensor de sinal, componente de variação nos saltos e memória de dados.

# 4 Projeto da ULA



Figura 2: Diagrama interno da ULA

A ULA é responsável por executar as operações lógicas, aritméticas e de deslocamento da arquitetura REDUX-V. Ela recebe dois operandos e um código de operação (func), realizando o processamento e enviando o resultado ao banco de registradores ou à memória.

### 4.1 Operações Suportadas

• Operações lógicas: not, and, or, xor

• Operações aritméticas: add, sub, addi, muli

• Deslocamentos: slr, srr

• Comparações: implícitas no controle de fluxo (brzr)

#### 4.2 Controle da ULA

O código de operação que define o comportamento da ULA é fornecido pela memória de controle, a partir do opcode decodificado da instrução atual.

### 5 Memória de Controle

A memória de controle foi projetada para fornecer os sinais necessários à execução de cada instrução, com base no opcode correspondente. Ela determina os sinais de controle para os multiplexadores, leitura e escrita de registradores, operação da ULA e controle de fluxo (como atualização do PC).



Figura 3: Diagrama da memória de controle do processador REDUX-V

# 6 Justificativa das Instruções Adicionais

As três instruções estendidas adicionadas à arquitetura (jien, muli e brzrue) visam ampliar a capacidade da linguagem de máquina e tornar os programas mais enxutos e expressivos.

• **jien** (Jump Immediate Extended Negative): permite saltos para trás mais longos do que o ji. Ideal para loops e estruturas de repetição. É uma instrução do tipo I que recebe um imediato, multiplica-o por dois, soma 9 e retorna o resultado negativo em B+1. A soma com 9 acontece pois o range de um imediato de 4 bits é [-8, 7], então a instrução é otimizada.

Adendo: A instrução **jien** foi primeiramente implementada com a ideia de permitir um salto positivo ou negativo, dependendo do conteúdo de **r[0]**. Se seu conteúdo fosse igual a 0, o salto seria positivo; caso contrário, negativo. Contudo, para melhor usabilidade no código de teste, a instrução foi modificada para sempre realizar um salto negativo. O circuito anterior foi mantido, mas encontra-se inoperante.

- muli (Multiplication Immediate): substitui múltiplas instruções addi por uma única operação de multiplicação com imediato. Como são reservados somente 4 bits de imediato, é difícil atingir valores grandes usando poucas instruções e com mais precisão que usar um shift. Por isso muli foi implementado.
- brzrue (Branch on Zero Unsigned Extended): permite saltos condicionais longos para frente, o que facilita implementações mais complexas com bifurcações. Também uma instrução do tipo I, recebe um imediato, multiplica-o por dois e soma 8.



Figura 4: Componente de variação utilizado no cálculo de endereço das instruções jien e brzrue

### 6.1 Exemplo de Código com Instruções Estendidas

Essas instruções contribuem para a legibilidade e eficiência do código Assembly por aumentar o range dos saltos e tirar a necessidade de cálculo manual do endereço de destino do PC.

# 7 Assembly

#### 7.1 Trabalho 1

```
1 sub r0, r0
2 sub r1, r1
3 sub r2, r2
4 sub r3, r3
6; os vetores comecarao no endereco de memoria 110
8; inicia valores
9 addi 1
10 \text{ sub } r3, r0; r3 = -1
11 addi 7
                  ; r0 = 10
12 addi 2
13 sub r1, r0
                  ; r1 = -10
                  ; r0 = 20
14 add r0, r0
                  ; r0 = 40
15 add r0, r0
                  ; r0 = 45
16 addi 5
                  ; r0 = 50
17 addi 5
               ; r2 = 50
18 add r2, r0
                  ; r0 = 60
19 sub r0, r1
20 st r0, r3
                  ; 60 armazenado em Oxff
                  ; r3 = -2
21 add r3, r3
22 add r2, r2
                  ; r2 = 100
23 sub r2, r1
                  ; r2 = 110
24 st r2, r3
                  ; 110 armazenado em Oxfe
25 sub r0, r0
26 sub r2, r2
27 sub r3, r3
30 ; calcula o fim_loop (PC = 23)
31 addi 1
32 sub r3, r3
                ; end = -1
33 sub r3, r0
34 ld r3, r3
                 ; end = fim_loop
35 sub r0, r0
37 loop:
38 brzr r1, r3
39; ajusta i
40 addi 7
                  ; r0 = 7
41 addi 3
                  ; r0 = 10
42 add r1, r0
                  ; i += 10
43; guarda 1
44 sub r0, r0
                  ; r0 = 0
                   ; r0 = 2
45 addi 2
46 sub r3, r3
47 sub r3, r0
                  ; end = -2
48 ld r3, r3
                 ; end = 110
49 add r3, r1
                  ; end = 110 + deslocamento (i)
                  ; r0 = 0
50 sub r0, r0
51 st r2, r3
                  ; M[end] = x
52; guarda 2
^{53} addi 1 ; r0 = 1 ^{54} add r2, r0 ; x += 1
```

```
55 addi 7
                 ; r0 = 8
                 ; r0 = 10
                  ; end = 110 + deslocamento + 10
57 add r3, r0
             ; M[end] = x
58 st r2, r3
59; ajusta parametros para reiniciar loop
60 sub r0, r0
61 addi 7
62 addi 2
                  ; r0 = 9
63 sub r1, r0
                 ; i -= 9
64 addi 1
                  ; r0 = 10
                  ; r0 = 20
65 add r0, r0
                  ; r0 = 23
66 addi 3
67 sub r3, r3
                  ; end = 0
                  ; end = 23
68 add r3, r0
69 sub r0, r0
                  ; r0 = 0
70 addi 1
                  ; r0 = 1
71 add r2, r0
                 ; x += 1
72 sub r0, r0
                  ; r0 = 0
73 brzr r0, r3
                  ; forca o salto para PC = 23
75 fim_loop:
76
77
79 ; soma dos vetores
                 ; PC = 60
81 sub r2, r2
82 sub r3, r3
83 addi 7
                ; r0 = 10
; i = -10
84 addi 3
85 sub r1, r0
86 sub r0, r0
88; calcula r3 (PC = 66)
89 sub r3, r3
                  ; r0 = 2
90 addi 2
91 sub r3, r0
                  ; r3 = -2
                  ; r3 = 110
92 ld r3, r3
                  ; r0 = 4
93 addi 2
94 sub r3, r0
                 ; r3 = 106
95 sub r0, r0
97 loop_2:
98 brzr r1, r3
99 addi 7
                ; r0 = 10
; i += 10
100 addi 3
101 add r1, r0
102 ;
103 sub r0, r0
104 sub r3, r3
                  ; r0 = 2
105 addi 2
106 sub r3, r0
                  ; r3 = -2
                  ; r3 = 110
107 ld r3, r3
                  ; r3 += deslocamento (i)
108 add r3, r1
109 ld r2, r3
                  ; r2 = M[end]
110 addi 7
                  ; r0 = 9
                  ; r0 = 10
111 addi 1
112 add r3, r0
                 ; r3 += 10
```

```
; r0 = M[r3]
; soma dos vetores
113 ld r0, r3
114 add r2, r0
115 sub r0, r0
116 addi 7
117 addi 3
                    ; r0 = 10
118 add r3, r0
                    ; r3 += 10
                   ; M[end] = r2
119 st r2, r3
120 sub r0, r0
                   ; r0 = 0
121 ;
122 addi 7
                   ; r0 = 9
123 addi 2
                    ; i -= 9
124 sub r1, r0
                    ; r0 = 16
125 addi 7
126 add r0, r0
                   ; r0 = 32
                   ; r0 = 64
127 add r0, r0
                   ; r0 = 66
128 addi 2
129 sub r3, r3
130 add r3, r0
                 ; r3 = 66
131 sub r0, r0
132 brzr r0, r3
133
134
135 fim_loop_2:
                    ; PC = 106
136 addi 3
                   ; r3 = 109
137 add r3, r0
138 sub r0, r0
                    ; halt (PC = PC + 0)
139 ji 0
```

#### 7.2 Trabalho 2

```
2 sub r0, r0
3 sub r1, r1
4 sub r2, r2
5 sub r3, r3
_{7}; os vetores comecarao no endereco de memoria 72
9; inicia valores
10 addi 7
                 ; r0 = 10
11 addi 3
12 sub r1, r0
                  ; r1 = -10
13 muli 7
                  ; r0 = 70
                  ; r0 = 72
14 addi 2
15 add r3, r0
                  ; r3 = 72
16 sub r2, r2
17 sub r0, r0
                 ; r0 = i
18 add r0, r1
19
20 loop:
21 brzruie 8
                 ; 8 * 2 + 8
22 ; ajusta i
23 sub r0, r0
24 addi 7
                  ; r0 = 7
                  ; r0 = 10
25 addi 3
                  ; i += 10
26 add r1, r0
27; guarda 1
```

```
; r0 = 0
28 sub r0, r0
                 ; end = base + deslocamento (i)
29 add r3, r1
30 st r2, r3
                 ; M[end] = x
31; guarda 2
32 addi 1
                  ; r0 = 1
33 add r2, r0
                 ; x += 1
34 addi 7
                  ; r0 = 8
35 addi 2
                 ; r0 = 10
36 add r3, r0
                 ; end = base + deslocamento + 10
37 st r2, r3
                 ; M[end] = x
                 ; r3 = base + 10
38 sub r3, r1
39 sub r3, r0
                 ; r3 = base
40 ; ajusta parametros para reiniciar loop
41 addi -1 ; r0 = 9
              ; i -= 9
42 sub r1, r0
                 ; r0 = 0
43 sub r0, r0
                 ; r0 = 1
44 addi 1
45 add r2, r0
                 ; x += 1
                 ; r0 = 0
46 sub r0, r0
47 add r0, r1
                 ; se i = 0 saira do loop
                  ; -(8 * 2 + 7)
48 jie 8
49
50 fim_loop:
51
52
54 ; soma dos vetores
56 sub r2, r2
57 addi 7
                ; r0 = 10
58 addi 3
59 sub r1, r0
                 ; i = -10
60 sub r0, r0
61 add r0, r1
                 ; r0 = i
62 or r0, r0
                  ; jump pula para ca
63
64 loop_2:
                 ; PC = 44
65 brzruie 9
                 ; r0 = 0
66 sub r0, r0
67 addi 7
68 addi 3
                 ; r0 = 10
                  ; i += 10
69 add r1, r0
70 ;
71 sub r0, r0
72 add r3, r1
                 ; r3 += deslocamento (i)
                 ; r2 = M[end]
73 ld r2, r3
                 ; r0 = 7
74 addi 7
                 ; r0 = 10
75 addi 3
76 add r3, r0
                 ; r3 = base + 10
                 ; r0 = M[end]
77 ld r0, r3
78 add r2, r0
                  ; soma dos vetores
79 sub r0, r0
80 addi 7
                  ; r0 = 10
81 addi 3
                 ; r3 = base + 10
82 add r3, r0
                 ; M[end] = r2
83 st r2, r3
84 sub r3, r1
85 sub r3, r0
```

```
86 sub r3, r0
              ; r3 = base
87 ;
88 addi -1
                   ; r0 = 9
89 sub r1, r0
                   ; i -= 9
90 sub r0, r0
91 add r0, r1
                   ; r0 = i
                   ; PC = 68
92 jien 8
93
95 fim_loop_2:
96 or r0, r0
97 ji 0
                   ; halt (PC = PC + 0)
```

## 8 Conclusão

O projeto do processador REDUX-V evidenciou os princípios fundamentais da arquitetura de computadores, incluindo o funcionamento de uma ULA, memória de controle e fluxo de dados via datapath. As instruções adicionais propostas aumentaram a flexibilidade da linguagem e a expressividade dos programas escritos para essa arquitetura.