

# Atividade 6: Detector de Valor Máximo e Mínimo

Maria Eduarda Soares Romana & Tiago Fernandes Soucek

9 de junho de 2025

## 1 Resumo

O presente relatório descreve a atividade 6 da disciplina de Lógica Reconfigurável, a qual teve como objetivo implementar um **detector genérico de valor mínimo e máximo** utilizando VHDL, com foco em **tipos e larguras de dados parametrizáveis**, por meio de estruturas como *generic*, *package*, *type* e *procedure/function*. O projeto compara duas entradas de mesmo tamanho, armazenadas em um vetor do tipo customizado *data\_array*, e utiliza funções *find\_min* e *find\_max* para determinar o resultado da comparação. A entrada e o resultado são exibidos em **displays de sete segmentos**, com controle por botões físicos presentes na FPGA. A tarefa foi simulada e sintetizada utilizando o software *Quartus*.

## 2 Requisitos do Projeto

Os seguintes itens foram requisitos do projeto:

- Criar um detector de valor mínimo e máximo entre valores *unsigned*.
- Utilizar *generic* para permitir quantidade genérica de bits e entradas, e *package* com tipo (*type*) e funções para cálculo de mínimo e máximo.
- Criar um tipo definido pelo usuário em um *package* separado.

## 3 Métodos e Técnicas

A implementação foi realizada no software Quartus Prime, e os Listings 5 e 1 apresentam os códigos VHDL desenvolvidos para resolver o problema proposto.

### 3.1 Pacote reutilizável: min\_max\_pkg.vhd

---

```
1 -----  
2 -- ATV 06 - MINIMO E MAXIMO  
3 -- MARIA EDUARDA SOARES ROMANA SILVA  
4 -- TIAGO FERNANDES SOUCEK  
5 -----  
6  
7  
8 library ieee;  
9 use ieee.std_logic_1164.all;  
10 use ieee.numeric_std.all;  
11  
12 package min_max_pkg is  
13
```

---

```

14      constant DATA_WIDTH : integer := 4; -- tamanho fixo do std_logic_vector
15
16      -- Tipo de vetor de vetores com largura definida
17      type data_array is array (natural range <>) of std_logic_vector(
18          DATA_WIDTH-1 downto 0);
19
20      function find_min(data : data_array) return std_logic_vector;
21      function find_max(data : data_array) return std_logic_vector;
22
23  end package min_max_pkg;
24
25 package body min_max_pkg is
26
27      function find_min(data : data_array) return std_logic_vector is
28          variable result : std_logic_vector(data(data'low)'range) := data(
29              data'low);
30
31      begin
32          for i in data'low + 1 to data'high loop
33              if unsigned(data(i)) < unsigned(result) then
34                  result := data(i);
35              end if;
36          end loop;
37          return result;
38      end function;
39
40      function find_max(data : data_array) return std_logic_vector is
41          variable result : std_logic_vector(data(data'low)'range) := data(
42              data'low);
43
44      begin
45          for i in data'low + 1 to data'high loop
46              if unsigned(data(i)) > unsigned(result) then
47                  result := data(i);
48              end if;
49          end loop;
50          return result;
51      end function;
52
53  end package body min_max_pkg;

```

Listing 1: Código VHDL: ATV 06 - Mínimo e Máximo

Esse código (Listing 1) define um **package** reutilizável chamado *min\_max\_pkg*, com foco em generalidade e modularidade.

### 3.1.1 Constante de configuração

```
1 constant DATA_WIDTH : integer := 4;
```

Listing 2: Código VHDL: Constante de configuração

Define o número de bits dos dados que serão manipulados.

### 3.1.2 Tipo definido pelo usuário

```
1 type data_array is array (natural range <>) of std_logic_vector(DATA_WIDTH-1
2     downto 0);
```

Listing 3: Código VHDL: Tipo definido pelo usuário

Cria um vetor genérico (*data\_array*) contendo elementos do tipo *std\_logic\_vector*, onde o número de elementos é definido externamente, exemplo: 2 entradas, 4 entradas, etc.

### 3.1.3 Funções *find\_min* e *find\_max*

```
1 if unsigned(data(i)) > unsigned(result) then
2     result := data(i);
3 end if;
```

Listing 4: Código VHDL: Funções *find\_min* e *find\_max*

Ambas as funções *find\_min* e *find\_max*, percorrem o vetor *data\_array* comparando os valores convertidos para *unsigned* e retornam o menor ou maior elemento. Essas funções encapsulam a lógica de comparações, evitando duplicação de código na entidade principal, o que torna o projeto mais "limpo" e organizado.

## 3.2 Código principal: atv06.vhd

```
1 -----
2 -- ATV 06 - MINIMO E MAXIMO
3 -- MARIA EDUARDA SOARES ROMANA SILVA
4 -- TIAGO FERNANDES SOUCEK
5 -----
6
7 library ieee;
8 use ieee.std_logic_1164.all;
9 use ieee.numeric_std.all;
10 use work.min_max_pkg.all; -- importa o pacote com find_max e find_min
11
12 entity atv06 is
13     generic (
14         NUM_BITS      : integer := 4; -- quantidade de bits por numero
15         NUM_INPUTS    : integer := 2 -- quantidade de entradas para comparacao
16     );
17     port (
18         CLOCK_50      : in std_logic;
19         KEY           : in std_logic_vector(1 downto 0);
20         SW             : in std_logic_vector(NUM_BITS-1 downto 0);
21
22         HEX0          : out std_logic_vector(6 downto 0);
23         HEX1          : out std_logic_vector(6 downto 0);
24         HEX2          : out std_logic_vector(6 downto 0);
25         HEX3          : out std_logic_vector(6 downto 0);
26         HEX4          : out std_logic_vector(6 downto 0);
27         HEX5          : out std_logic_vector(6 downto 0)
28     );
29 end entity atv06;
30
31 architecture rtl of atv06 is
32
33     -- Sinais internos
34     signal num1_salvo : std_logic_vector(NUM_BITS-1 downto 0) := (others =>
35     '0');
36     signal max_val     : std_logic_vector(NUM_BITS-1 downto 0) := (others =>
37     '0');
```

```
37      signal key_pressed : std_logic_vector(1 downto 0);
38      signal key_reg1     : std_logic_vector(1 downto 0) := "11";
39      signal key_reg2     : std_logic_vector(1 downto 0) := "11";
40      signal key_reg3     : std_logic_vector(1 downto 0) := "11";
41
42      -- Vetor de dados generico com 2 entradas
43      signal entradas : data_array(0 to NUM_INPUTS-1);
44
45      -- Conversor para display
46      function hex_to_7seg(hex_digit : std_logic_vector(3 downto 0)) return
47          std_logic_vector is
48      begin
49          case hex_digit is
50              when "0000" => return "1000000"; when "0001" => return "1111001"
51                  ;
52              when "0010" => return "0100100"; when "0011" => return "0110000"
53                  ;
54              when "0100" => return "0011001"; when "0101" => return "0010010"
55                  ;
56              when "0110" => return "0000010"; when "0111" => return "1111000"
57                  ;
58              when "1000" => return "0000000"; when "1001" => return "0010000"
59                  ;
60              when "1010" => return "0001000"; when "1011" => return "0000011"
61                  ;
62              when "1100" => return "1000110"; when "1101" => return "0100001"
63                  ;
64              when "1110" => return "0000110"; when "1111" => return "0001110"
65                  ;
66              when others => return "1111111";
67          end case;
68      end function;
69
70 begin
71
72     -- Debounce dos botoes
73     process(CLOCK_50)
74     begin
75         if rising_edge(CLOCK_50) then
76             key_reg1 <= not KEY;
77             key_reg2 <= key_reg1;
78             key_reg3 <= key_reg2;
79             key_pressed <= key_reg2 and (not key_reg3);
80         end if;
81     end process;
82
83     -- Logica principal
84     process(CLOCK_50)
85     begin
86         if rising_edge(CLOCK_50) then
87             if key_pressed(1) = '1' then
88                 num1_salvo <= SW;
89             end if;
90
91             if key_pressed(0) = '1' then
92                 entradas(0) <= num1_salvo;
93                 entradas(1) <= SW;
94                 max_val <= find_max(entradas);
```

```
86      end if;
87  end if;
88 end process;
89
90 -- Displays
91 HEX0 <= hex_to_7seg(SW); -- entrada ao vivo
92 HEX5 <= hex_to_7seg(max_val); -- resultado
93
94 -- Apaga os displays nao usados
95 HEX1 <= "1111111";
96 HEX2 <= "1111111";
97 HEX3 <= "1111111";
98 HEX4 <= "1111111";
99
100 end architecture rtl;
```

Listing 5: Código VHDL: ATV 06

Esta é a parte principal do projeto (Listing 5), responsável por ler os dados das chaves (SW), detectar os comandos via botões (KEY), armazenar os valores, comparar e exibir os resultados.

### 3.2.1 Generic

```
1 generic (
2     NUM_BITS    : integer := 4;
3     NUM_INPUTS : integer := 2
4 );
```

Listing 6: Código VHDL: generic

Define dois parâmetros configuráveis:

- **NUM\_BITS**: número de bits de cada valor
- **NUM\_INPUTS**: número de entradas no vetor de dados.

### 3.2.2 Importação de pacotes

```
1 use work.min_max_pkg.all;
```

Listing 7: Código VHDL: Importação de pacotes

Torna acessível o tipo *data\_array* e as funções *find\_min* e *find\_max*.

### 3.2.3 Vetores interno

```
1 signal num1_salvo : std_logic_vector(NUM_BITS-1 downto 0);
2 signal entradas : data_array(0 to NUM_INPUTS-1);
```

Listing 8: Código VHDL: Vetores internos

- **num1\_salvo**: armazena o primeiro valor inserido.
- **entradas**: Vetor usado como entrada para a função *find\_max*.

### 3.2.4 Lógica principal

```
1 if key_pressed(1) = '1' then
2     num1_salvo <= SW;
3 end if;
4
5 if key_pressed(0) = '1' then
6     entradas(0) <= num1_salvo;
7     entradas(1) <= SW;
8     max_val <= find_max(entradas);
9 end if;
```

Listing 9: Código VHDL: Lógica principal

- Ao pressionar **KEY(1)**, o valor atual das chaves é salvo.
- Ao pressionar **KEY(0)**, o valor salvo e o valor atual são inseridos no vetor *entradas*, e o maior valore entre eles é obtido usando a função *find\_max*.

### 3.2.5 Exibição nos displays

```
1 HEX0 <= hex_to_7seg(SW);
2 HEX5 <= hex_to_7seg(max_val);
```

Listing 10: Código VHDL: Exibição em displays

- **HEX0** mostra o valor atual das chaves, em tempo real.
- **HEX5** mostra o maior valor.

A função *hex\_to\_7seg* faz a conversão de 4 bits para o padrão dos displays de sete segmentos, permitindo a exibição numérica direta.

## 4 Resultados e Considerações

A implementação foi concluída com sucesso. O comparador exibe corretamente o valor atual e o maior valor entre dois inseridos sequencialmente. A RTL Figura 1 confirma o fluxo de dados e uso das funções do *package*.

Apesar de a atividade funcionar com apenas duas entradas ao invés de um conjunto maior (ex: 4 ou 5), como foi pedido na descrição da tarefa, a lógica foi corretamente generalizada com o uso de *generic*, *type*, *package* e funções, e atende todos os requisitos centrais da proposta.

### 4.0.1 RTL do circuito

A Figura 1 apresenta o diagrama RTL gerado automaticamente pelo software *Quartus* após a compilação do projeto. É possível observar a separação clara entre os blocos de controle, registradores intermédios e blocos de exibição. As funções de comparação do pacote integradas corretamente à arquitetura principal do projeto.



Figura 1: RTL do circuito

#### 4.0.2 Pin Planner

As Figuras 2 e 3 apresentam o mapeamento dos pinos da FPGA para os sinais utilizados no projeto.

| <u>In</u>  | <u>CLOCK_50</u> | <u>Input</u>  | <u>PIN_P11</u> | <u>3</u> | <u>B3_N0</u> | <u>PIN_P11</u> | <u>2.5 V</u> | <u>12mA [default]</u> |                    |  |
|------------|-----------------|---------------|----------------|----------|--------------|----------------|--------------|-----------------------|--------------------|--|
| <u>out</u> | <u>HEXO[6]</u>  | <u>Output</u> | <u>PIN_C17</u> | <u>7</u> | <u>B7_N0</u> | <u>PIN_C17</u> | <u>2.5 V</u> | <u>12mA [default]</u> | <u>2 [default]</u> |  |
| <u>out</u> | <u>HEXO[5]</u>  | <u>Output</u> | <u>PIN_D17</u> | <u>7</u> | <u>B7_N0</u> | <u>PIN_D17</u> | <u>2.5 V</u> | <u>12mA [default]</u> | <u>2 [default]</u> |  |
| <u>out</u> | <u>HEXO[4]</u>  | <u>Output</u> | <u>PIN_E16</u> | <u>7</u> | <u>B7_N0</u> | <u>PIN_E16</u> | <u>2.5 V</u> | <u>12mA [default]</u> | <u>2 [default]</u> |  |
| <u>out</u> | <u>HEXO[3]</u>  | <u>Output</u> | <u>PIN_C16</u> | <u>7</u> | <u>B7_N0</u> | <u>PIN_C16</u> | <u>2.5 V</u> | <u>12mA [default]</u> | <u>2 [default]</u> |  |
| <u>out</u> | <u>HEXO[2]</u>  | <u>Output</u> | <u>PIN_C15</u> | <u>7</u> | <u>B7_N0</u> | <u>PIN_C15</u> | <u>2.5 V</u> | <u>12mA [default]</u> | <u>2 [default]</u> |  |
| <u>out</u> | <u>HEXO[1]</u>  | <u>Output</u> | <u>PIN_E15</u> | <u>7</u> | <u>B7_N0</u> | <u>PIN_E15</u> | <u>2.5 V</u> | <u>12mA [default]</u> | <u>2 [default]</u> |  |
| <u>out</u> | <u>HEXO[0]</u>  | <u>Output</u> | <u>PIN_C14</u> | <u>7</u> | <u>B7_N0</u> | <u>PIN_C14</u> | <u>2.5 V</u> | <u>12mA [default]</u> | <u>2 [default]</u> |  |

Figura 2: Pin Planner - clock e HEX0

|            |         |        |         |   |       |         |       |                |             |
|------------|---------|--------|---------|---|-------|---------|-------|----------------|-------------|
| <b>out</b> | HEX5[6] | Output | PIN_B22 | 6 | B6_N0 | PIN_B22 | 2.5 V | 12mA (default) | 2 (default) |
| <b>out</b> | HEX5[5] | Output | PIN_C22 | 6 | B6_N0 | PIN_C22 | 2.5 V | 12mA (default) | 2 (default) |
| <b>out</b> | HEX5[4] | Output | PIN_B21 | 6 | B6_N0 | PIN_B21 | 2.5 V | 12mA (default) | 2 (default) |
| <b>out</b> | HEX5[3] | Output | PIN_A21 | 6 | B6_N0 | PIN_A21 | 2.5 V | 12mA (default) | 2 (default) |
| <b>out</b> | HEX5[2] | Output | PIN_B19 | 7 | B7_N0 | PIN_B19 | 2.5 V | 12mA (default) | 2 (default) |
| <b>out</b> | HEX5[1] | Output | PIN_A20 | 7 | B7_N0 | PIN_A20 | 2.5 V | 12mA (default) | 2 (default) |
| <b>out</b> | HEX5[0] | Output | PIN_B20 | 6 | B6_N0 | PIN_B20 | 2.5 V | 12mA (default) | 2 (default) |
| <b>in</b>  | KEY[1]  | Input  | PIN_A7  | 7 | B7_N0 | PIN_A7  | 2.5 V | 12mA (default) |             |
| <b>in</b>  | KEY[0]  | Input  | PIN_B8  | 7 | B7_N0 | PIN_B8  | 2.5 V | 12mA (default) |             |
| <b>in</b>  | SW[3]   | Input  | PIN_C12 | 7 | B7_N0 | PIN_C12 | 2.5 V | 12mA (default) |             |
| <b>in</b>  | SW[2]   | Input  | PIN_D12 | 7 | B7_N0 | PIN_D12 | 2.5 V | 12mA (default) |             |
| <b>in</b>  | SW[1]   | Input  | PIN_C11 | 7 | B7_N0 | PIN_C11 | 2.5 V | 12mA (default) |             |
| <b>in</b>  | SW[0]   | Input  | PIN_C10 | 7 | B7_N0 | PIN_C10 | 2.5 V | 12mA (default) |             |

Figura 3: Pin Planner - HEX5, KEYs e CHAVES

#### 4.0.3 Funcionamento da placa

Neste [LINK](#) (clique aqui) há o vídeo demonstrando o funcionamento do projeto na placa.