## Estructura de Computadores

# Unidad 1: Diseño digital y VHDL

Escuela Politécnica Superior - UAM

### Índice

- Introducción
- Lógica combinacional
- Circuitos combinacionales
- Lógica secuencial
- Modelado estructural
- Bancos de prueba (testbenches)

#### Introducción

- Hardware description language (HDL): permite diseñar la funcionalidad de un circuito digital sin poner funciones lógicas ni puertas. Diversas herramientas informáticas producen o <u>sintetizan</u> el circuito concreto que realiza dicha funcionalidad.
- Los circuitos comerciales se diseñan con HDLs
- Los dos HDLs principales son:
  - VHDL
    - Desarrollado en 1981 por el Dpto. de Defensa de EEUU
    - Se convirtió en estándar IEEE (1076) en 1987

#### Verilog

- Desarrollado en 1984 por Gateway Design Automation
- Se convirtió en estándar IEEE (1364) en 1995

# Simulación y síntesis

#### Simulación

- Se aplican ciertos valores a las entradas
- Se comprueba si las salidas son correctas
- Mucho tiempo/dinero ahorrado por depurar en simulación y no en hardware

#### Síntesis

 Transforma el código HDL en un circuito (netlist) describiendo el hardware (una lista de puertas y cables conectándolas)

#### **IMPORTANTE**:

- ✓ No todos los diseños que se simulan son sintetizables.
- ✓ Al describir circuitos en un HDL, es vital pensar en el hardware que se debería generar (no es programar en C).

# Entidad - Arquitectura

 La entidad se utiliza para hacer una descripción "caja negra" del diseño, sólo se detalla su interfaz (los puertos de entrada y salida "ports")

Equivalente en C

```
float CalculaMedia(float a, float b) {
float c;
c = (a+b) / 2;
return c;
```

Los contenidos del circuito se modelan dentro de la arquitectura



```
float CalculaMedia(float a, float b) {
    float c;
    c = (a+b) / 2;
    return c;
}
```

Una entidad puede tener varias arquitecturas

#### PORTS: La conexión con el exterior



# Ejemplo de código VHDL

```
library IEEE;
                                  -- similar a declaración de .h
  use IEEE.std_logic_1164.all; -- para usar std_logic
   entity inversor is
                                              Inversor
ENTIDAD
     port (a : in std_logic;
            y : out std_logic);
                                        a
   end inversor;
ARQUITECTURA
   architecture comportamental of inversor is
   begin
     y <= not a;
                                   -- asignación con flecha
   end comportamental;
```

# Síntesis de código VHDL

#### VHDL:

```
architecture comportamental of ejemplo is
begin
    y <= (not a and not b and not c) or (a and not b and not c) or
    (a and not b and c);
end comportamental;</pre>
```

#### Síntesis:



#### Sintaxis VHDL

- No distingue mayúsculas de minúsculas (case insensitive)
  - Ejemplo: reset y Reset son la misma señal.
  - Recomendación: poner siempre las mismas mayúsculas para facilitar la lectura y las búsquedas.
- Los nombres no pueden empezar por números
  - Ejemplo: 2mux no es un nombre válido.
- Se ignoran los espacios, tabuladores, retornos de carro
- Comentarios:
  - -- Desde un guión doble hasta el final de la línea.

# Sintaxis en VHDL: Identificadores

#### **IDENTIFICADORES**

Nombres o etiquetas que se usan para referirse a: Constantes, Señales, Procesos, Entidades, etc.

Longitud (Número de Caracteres): Sin restricciones

Palabras reservadas por VHDL no pueden ser identificadores

En VHDL, un identificador en mayúsculas es igual que en minúsculas

Están formados por números, letras (mayúsculas o minúsculas) y guión bajo "\_" con las reglas especificadas en la tabla siguiente.

| Reglas para especificar un identificador                         | Incorrecto | Correcto |
|------------------------------------------------------------------|------------|----------|
| Primer carácter debe ser siempre una letra mayúscula o minúscula | 4Suma      | Suma4    |
| Segundo carácter no puede ser un guión bajo ( _ )                | S_4bits    | S4_bits  |
| Dos guiones bajos consecutivos no son permitidos                 | Resta4     | Resta_4_ |
| Un identificador no puede utilizar símbolos especiales           | Clear#8    | Clear_8  |

### Índice

- Introducción
- Lógica combinacional
- Circuitos combinacionales
- Lógica secuencial
- Modelado estructural
- Bancos de prueba (testbenches)

# Tipo std\_logic

- Los valores '0' y '1' del tipo bit se quedan cortos para modelar todos los estados de una señal digital en la realidad
- El paquete IEEE.std\_logic\_1164 define el tipo **std\_logic**, que representa todos los posibles estados de una señal real:
  - U No inicializado, valor por defecto
  - X Desconocido fuerte, indica cortocircuito
  - O Salida de una puerta con nivel lógico bajo
  - 1 Salida de una puerta con nivel lógico alto
  - Z Alta Impedancia
  - W Desconocido débil, terminación de bus
  - L 0 débil, resistencia de pull-down
  - H 1 débil, resistencia de pull-up
  - No importa, usado como comodín para síntesis

El tipo **std\_logic\_vector** define un *array* de bits de tipo **std\_logic**.

# Operadores bitwise

```
entity puertas is
port (a, b: in std_logic_vector(3 downto 0);
   y1, y2, y3, y4, y5: out std_logic_vector(3 downto 0));
end puertas;
architecture comport of puertas is
                                                             [3:0] [3:0] y3[3:0]
begin
                                                        y3[3:0]
-- Pueden actuar sobre bits
-- o sobre buses
  y1 <= a and b; -- AND
  y2 \ll a \text{ or } b; -- OR
                                               y1[3:0]
                                                        y4[3:0]
                                                               [3:0]
y1[3:0]
  y3 \le a xor b; -- XOR
   y4 <= a nand b; -- NAND
                                                               [3:0]
y5[3:0]
   y5 <= a nor b; -- NOR
                                               y2[3:0]
                                                        y5[3:0]
                                                               [3:0]
y2[3:0]
end comport;
```

# Asignación condicional: when ... else

```
entity mux2a1_4bits is
port(d0,d1 : in std_logic_vector(3 downto 0);
       s : in std_logic;
       y : out std_logic_vector(3 downto 0));
end mux2al 4bits;
architecture comport of mux2a1_4bits is
begin
  y \ll d0 when s = `0' else d1;
end comport;
                                     [3:0]
                     d0[3:0]
                                               [3:0]
                                                     y[3:0]
                                     [3:0]
                     d1[3:0]
```

# Asignación condicional: with ... select

```
entity deco2a4 is
port(a : in std logic vector(1 downto 0);
      y : out std logic vector(3 downto 0));
end deco2a4;
architecture comport of deco2a4 is
begin
 with a select
     y \le "0001" when "00",
           "0010" when "01",
           "0100" when "10",
           "1000" when others; -- último caso, siempre
                                  -- conviene incluirlo
end comport;
```

#### Señales internas

```
architecture comport of fulladder is
  signal p, g : std_logic; -- La declaración de señales internas
                               -- se pone entre architecture y begin
begin
  p <= a xor b;
  g <= a and b;
  s <= p xor cin;
  cout <= g or (p and cin);</pre>
end comport;
                                                               cout
                                                     cout
                                         un1 cout
```

#### Concurrencia

```
architecture comport of fulladder is
  signal p, g : std_logic;
begin
  p <= a xor b; -- No importa el orden de las
                        -- líneas, todo el hardware
  q <= a and b;
                         -- "existe" a la vez
  s <= p xor cin;
  cout <= g or (p and cin);</pre>
end comport;
architecture comport of fulladder is
  signal p, q : std logic;
begin
  cout <= g or (p and cin); -- Es el mismo circuito, no</pre>
  s <= p xor cin; -- importa que p se asigne en
  p <= a xor b;
                           -- una línea posterior porque
                           -- el hardware es concurrente
  q <= a and b;
end comport;
```

# Precedencia de operadores

- Orden en que se resuelven los operadores si no hay paréntesis
- Recomendación: usar paréntesis

#### **Primero**

| Operador  | Operación       |  |  |
|-----------|-----------------|--|--|
| not       | NOT             |  |  |
| * / %     | mult div módulo |  |  |
| + -       | suma resta      |  |  |
| < <= > >= | comparar        |  |  |
| = /=      | igual distinto  |  |  |
| and nand  | AND NAND        |  |  |
| xor xnor  | XOR XNOR        |  |  |
| or nor    | OR NOR          |  |  |

# Último

# Formatos de números y bits

| Formato | Nº bits | Base        | Memoria     |
|---------|---------|-------------|-------------|
| 11'     | 1       | Binario     | 1           |
| "101"   | >1      | Binario     | 101         |
| X"AF"   | >1      | Hexadecimal | 10101111    |
| 1       | Depende | Decimal     | 000001      |
| -2      | Depende | Decimal     | 111110 (C2) |
| 1.5     | Depende | Decimal     | IEEE-754    |

Los valores de las señales se asignan con una "flecha":

```
a_bit <= '1';
a_bus <= "101";
a_int <= 1;
```

```
signal a : std_logic_vector(3 downto 0);
signal b : std_logic_vector(0 to 3);
...
a <= "0101";
b <= "0101";

-- es equivalente a:
a(3) <= '0'; a(2) <= '1'; a(1) <= '0'; a(0) <= '1';
b(3) <= '1'; b(2) <= '0'; b(1) <= '1'; b(0) <= '0';</pre>
```

|   | Posición | 3 | 2 | 1 | 0 |
|---|----------|---|---|---|---|
| a | Valor    | 0 | 1 | 0 | 1 |

| h | Posición | 0 | 1 | 2 | 3 |
|---|----------|---|---|---|---|
| U | Valor    | 0 | 1 | 0 | 1 |

signal y : std\_logic\_vector(7 downto 0);

```
signal a, b : std_logic_vector(3 downto 0);
. . .
-- el operador & en VHDL significa concatenar
y \le a \& b(3) \& '0' \& b(2 downto 1);
-- es equivalente a:
y(7) \le a(3); y(6) \le a(2); y(5) \le a(1); y(4) \le a(0);
y(3) \le b(3); y(2) \le 0'; y(1) \le b(2); y(0) \le b(1);
                                                       b
        a
                                        2
                                             1
                                                 0
                       1
                           0
        У
                       5
```

```
signal y : std_logic_vector(7 downto 0);
signal a, b : std_logic_vector(3 downto 0);
-- Se pueden hacer todo tipo de agrupaciones parciales
-- siempre que coincidan el número de bits
y(7 \text{ downto } 4) \le a;
y(3) \le b(3);
v(2) <= `0';
y(1 \text{ downto } 0) \le b(2 \text{ downto } 1);
-- es el mismo resultado para la "señal y" que en la
 - transparencia anterior
          y <= a & b(3) & '0' & b(2 downto 1); _
```

```
signal y : std logic vector(7 downto 0);
signal a, b : std logic vector(3 downto 0);
. . .
-- en VHDL se permite hacer asignaciones por índice en
-- lugar de orden (izquierda a derecha) bit a bit (aggregates)
  y \le (3 \text{ downto } 2 \Rightarrow a(2), 4 \Rightarrow b(1), 5 \Rightarrow 1',
       others => '0'); -- others es "resto de bits"
-- es equivalente a:
  y(7) \le 0'; y(6) \le 0'; y(5) \le 1'; y(4) \le b(1);
  y(3) \le a(2); y(2) \le a(2); y(1) \le 0'; y(0) \le 0';
  muy utilizado para la inicialización de buses:
```

y <= (others => '0'); -- poner todos los bits a '0'

1.4, 1.10

# Z: alta impedancia

#### VHDL:

```
signal y, a: std_logic_vector(3 downto 0);
signal en : std_logic;
...
y <= (others => `Z') when en = `0' else a;
```

#### Síntesis:



# Índice

- Introducción
- Lógica combinacional
- Circuitos combinacionales
- Lógica secuencial
- Modelado estructural
- Bancos de prueba (testbenches)

#### Circuitos combinacionales

- La mayoría de los circuitos combinacionales se pueden definir con asignaciones concurrentes (vistas hasta ahora)
- También hay código secuencial (se ejecuta línea tras línea, como en el software, sin concurrencia, así que el orden del código sí es importante) pero se encapsula en "procesos" process (aparte de funciones y procedimientos que no vemos)
- Suele ser más intuitivo escribir en procesos, pero da lugar a más errores, como la lista de sensibilidad
- Dentro de procesos (y sólo dentro de ellos) se pueden utilizar if, case, for y while

#### **Process**

# Estructura del "process": architecture nombreArq of nombreEnt is -- parte declarativa, señales internas de la arquitectura Begin -- Los procesos, uno o varios, aparecen entre begin y end de -- architecture [ETIQ:] process(lista sensibilidad) -- la etiqueta es opcional -- parte declarativa, variables pero no señales, de uso -- interno en el proceso begin -- código;

end nombreArq;

end process [ETIQ];

Cada vez que cambia alguna señal de la lista sensibilidad, se ejecuta secuencialmente código

```
if condicion_1 then
   -- sec instr 1
[elsif condicion_2 then
   -- sec instr 2]
[elsif condicion_3 then
   -- sec instr 3]
[else
   -- instr por defecto]
end if;
```

Similar a la asignación condicional when...else

#### **Ejemplo**

```
CTRL: process(nivel)
begin
  if nivel > 60 then
    a <= "11";
  elsif nivel > 40 then
    a <= "10";
  elsif nivel > 20 then
    a <= "01";
  elsif nivel > 15 then
    a <= "00";
  end if;
end process CTRL;</pre>
```

Cuidado: si es lógica combinacional SIEMPRE tiene que haber "else"

#### case

```
case expresion is
  when caso_1 =>
    -- sec instr 1
  when caso_2 =>
    -- sec instr 2
  when others =>
    -- instr por defecto
end case;
```

Similar a la asignación condicional with...select

#### **Ejemplo**

```
MUX: process(sel, a, b, c, d)
begin
  case sel is
   when "11" => y <= d;
   when "10" => y <= c;
   when "01" => y <= b;
   when others => y <= a;
  end case;
end process MUX;</pre>
```

Cuidado: siempre tienen que estar todos los caminos. Para no olvidarse ninguno, que el último sea "when others"

1.3, 1.2, 1.11

#### for

```
[ETIQ:] for indice in rango loop
  -- sec instr
end loop [ETIQ];
```

#### **Ejemplo**

```
AND8: process(a, b)

begin

for i in 0 to 7 loop

y(i) <= a(i) and b(i);

end loop;

end process AND8;

AND8: process(a, b)

begin

for i in 7 downto 0 loop

y(i) <= a(i) and b(i);

end loop;

end process AND8;
```

```
-- Hubiera sido más fácil porque
y <= a and b; -- and es operador bitwise
-- además no sería necesario un proceso
```

# while

```
[ETIQ:] while condicion loop
  -- sec instr
end loop [ETIQ];
```

#### **Ejemplo**

```
process(...)
begin
    while a = '1' loop
    ...
    ...
    end loop;
end process;
```

# ¿Concurrente o secuencial?

• El hardware es concurrente. ¿Qué sentido tiene un código secuencial?

```
"
a <= '0';
...
a <= '1';
...
a=¿? a=X</pre>
```

# Actualización de señales en procesos

• El tiempo "se detiene" cuando se ejecuta un proceso. Las señales que se escriben reciben un nuevo valor después de <u>acabar el proceso</u> o, si lo hay, después de un wait.

| Señal | Antes del proceso | Después<br>del proceso |  |
|-------|-------------------|------------------------|--|
| а     | <b>'1'</b>        | <b>'0'</b>             |  |
| b     | ?                 | <b>'</b> 0'            |  |

# Actualización de señales en procesos

• El tiempo "**se detiene**" cuando se ejecuta un proceso. Las señales que se escriben reciben un nuevo valor después de <u>acabar el proceso</u> o, si lo hay, después de un wait.

| Señal | Antes del proceso | Antes 1 <sup>er</sup> wait | Después 1 <sup>er</sup> wait | Antes 2º wait | Después 2º wait |
|-------|-------------------|----------------------------|------------------------------|---------------|-----------------|
| clk   | ?                 | ?                          | <b>'</b> 0'                  | <b>'</b> 0'   | <b>'1'</b>      |

El tiempo "si transcurre" durante la ejecución de un wait



Nota: Un wait no tiene sentido en hardware. Sólo se puede simular.

#### Sentencias condicionales y bucles: Resumen

#### Fuera de proceso

- $\checkmark$  When else .....else ;
- ✓ With select .....when others ;

#### Dentro de proceso

- ✓ If \*\* then -; elsif -; else -; end if;
- ✓ Case \*\* is when –; ....when others –; end case;
- ✓ For \*\* in \*\* downto/to \*\* loop ; end loop ;
- ✓ While \*\* loop –; end loop;

# Lógica combinacional: ¿Dentro o fuera de proceso?

- Excepto en el uso de bucles, cualquier código combinacional escrito en proceso puede escribirse fuera de proceso (sentencias concurrentes) y viceversa.
- Si se utilizan procesos, se deben tener en cuenta las listas de sensibilidad. ¿Riesgo innecesario?
- Lista de sensibilidad en procesos combinacionales
  - Señales que se leen:
    - Elementos que se comparan (if, case)
    - Elementos que se leen en asignaciones

#### Listas de sensibilidad: resumen

sensibilidad

```
Falta a y b en la lista de
  process(d)
  begin
                         sensibilidad
        if a<b then
            z \leq di
       else
            z \le not di
                                 process(a, b, z)
       end if:
                                 begin
  end process;
                                       if a<b then
                                           z \le b and a_i
                                       else
Sobra z de la lista de
                                           z \le a \text{ or } b_i
                                       end if;
```

end process;

#### Índice

- Introducción
- Lógica combinacional
- Circuitos combinacionales
- Lógica secuencial
- Modelado estructural
- Bancos de prueba (testbenches)

# Lógica secuencial

- En VHDL los flip-flops (o señales registradas) se describen siempre igual:
  - > Con un process en el que aparecen el reloj y el reset asíncrono (si lo hay) en la lista de sensibilidad
- Otras descripciones pueden conseguir simulaciones equivalentes, pero no generan el mismo hardware

### Flip-Flop tipo D

```
entity flop is
port(Clk : in std_logic;
     D : in std_logic_vector(3 downto 0);
     Q : out std_logic_vector(3 downto 0));
end flop;
architecture sintetizable of flop is
begin
REG: process(Clk)
  begin
    if Clk = '1' and Clk'event then --if rising_edge(Clk) then
      O <= D;
    end if:
  end process;
end sintetizable;
                      clk
                                 [3:0]
                                                    [3:0]
                                      D[3:0]
                                                           q[3:0]
                   d[3:0]
                                              Q[3:0]
                                          q[3:0]
                                                                  40
```

### Flip-Flop tipo D con Reset asíncrono

```
-- Se añade la entrada Reset en la entidad
architecture sintetizable of flop is
begin
  process(Reset, Clk)
    begin
      if Reset = '1' then
        Q <= (others => '0');
      elsif Clk = '1' and Clk'event then
        O <= D;
      end if;
  end process;
end sintetizable;
              clk
                           [3:0]
                    [3:0]
                                                D[3:0] Q[3:0]
           d[3:0]
                                      R
            reset
                                    q[3:0]
                                                               41
```

### Flip-Flop tipo D con Reset síncrono

```
architecture sintetizable of flop is
begin
  process(Clk) -- El reset síncrono no va en la lista sensib.
  begin
    if Clk = '1' and Clk'event then
      if Reset = '1' then
       O <= (others => '0');
      else
       O \leq Di
      end if;
    end if:
  end process;
end sintetizable;
            clk
                                                <u>[3:0]</u>
                          [3:01
                   [3:0]
                                D[3:0] Q[3:0]
          d[3:0]
           reset
                                R
                                   q[3:0]
```

### Flip-Flop tipo D con Enable

```
-- Se añade la entrada En
architecture sintetizable of flop is
begin
  process(Reset, Clk)
  begin
    if Reset = '1' then
       0 <= (others => '0');
    elsif Clk = '1' and Clk'event then
       if En = 1' then
         Q <= D;
       end if:
    end if:
  end process;
                         clk
                                        [3:0]
end sintetizable;
                                 [3:0]
                                                          [3<u>:</u>0] [3:0]
                       d[3:0]
                                            D[3:0]
                                                                   q[3:0]
                                                   Q[3:0]
                         en
                                                  R
                       reset
 1.12, 1.15
                                                q[3:0]
```

# Latch (cerrojo)

```
architecture sintetizable of latch is
begin
  process(CE, D)
  begin
    if CE = '1' then -- sin flanco, por nivel
      Q <= D;
    end if;
  end process;
end sintetizable;
                                lat
                            D[3:0]
            d[3:0]
                                   Q[3:0]
                               q[3:0]
```

#### Latches indeseados

```
process(a)
begin

    if a = "00" then
        b <= "11";
    elsif a = "01" then
        b <= "10";
    elsif a = "10" then
        b <= "00";
    end if;
end process;</pre>
```

¿Cuánto vale b cuando a="11"? b mantiene el valor anterior => Latch

**Atención**: se generan latches no deseados cuando no se asigna valor por todos los caminos. En este curso no se utilizan latches, pero un código puede generarlos sin querer. Si el hardware sintetizado en prácticas incluye latches, <u>se considerará como un error</u>, sobre todo si se quiere diseñar un combinacional.

#### Listas de sensibilidad: resumen

#### Procesos combinacionales

- Señales que se leen:
  - Elementos que se comparan (if, case)
  - Elementos que se leen en asignaciones

#### o Procesos síncronos

- Reset síncrono
  - CLK
- Reset asíncrono
  - CLK, Reset

#### Índice

- Introducción
- Lógica combinacional
- Circuitos combinacionales
- Lógica secuencial
- Modelado estructural
- Bancos de prueba (testbenches)

#### Modelado estructural

- Los componentes básicos se utilizan como elementos de otros más grandes.
- Es fundamental para la reutilización de código.
- Permite mezclar componentes creados con distintos métodos de diseño:
  - Esquemáticos
  - VHDL, Verilog
- Genera diseños más legibles y más portables.
- Necesario para estrategias de diseño top-bottom o bottom-up.



### Cómo instanciar un componente

```
entity top is
port
( ... );
end top;
```

```
architecture jerarquica of top is
  signal s1, s2 : std_logic;
  component a
 port
  (entrada: in std_logic;
    salida: out std_logic);
  end component;
begin
  u1: a
 port map
  (entrada => s1,
   salida => s2);
end jerarquica;
```

# Ejemplo de diseño jerárquico: componente inferior

```
LIBRARY ieee;
USE ieee.std_logic_1164.ALL;
ENTITY miand2 IS PORT (
  x, y: IN std_logic;
  z: OUT std_logic);
END miand2;
ARCHITECTURE archand2 OF miand2 IS
BEGIN
    z \le x AND y;
END archand2;
```

Descripción del componente de nivel inferior:



"Puerta AND 2 entradas"

# Ejemplo de diseño jerárquico: top-level

```
LIBRARY ieee;
USE ieee.std_logic_1164.ALL;
                                          Componente de nivel superior (top):
ENTITY miand4 IS PORT (
                                                 "Puerta AND 4 entradas"
    a, b, c, d: IN std_logic;
         z: OUT std logic);
END miand4;
ARCHITECTURE archmiand4 OF miand4 IS
                                       Declaración del componente
COMPONENT miand2 PORT (
  x, y: IN std_logic;
  z: OUT std_logic);
END COMPONENT;
SIGNAL s1, s2: std_logic;
                                       Instanciación del componente. Asociación por nombre
BEGIN
 al: miand2 PORT MAP (x=>a, y=>b, z=>s1)
 a2: miand2 PORT MAP (z=>s2, y=>c, x=>d); -- Se puede variar el orden
 a3: miand2 PORT MAP (x=>s1, y=>s2, z=>z);
-- También podría haber código aparte de las instancias
END archmiand4;
```

1.20

#### Índice

- Introducción
- Lógica combinacional
- Circuitos combinacionales
- Lógica secuencial
- Modelado estructural
- Bancos de prueba (testbenches)

### Bancos de prueba (test-benches)

- Código HDL escrito para comprobar que un módulo HDL funciona: el device under test (dut), o unit under test (uut)
- No sintetizable
- Tipos de *testbenches*:
  - Simple
  - Con auto-comprobación

#### Cómo hacer un testbench

- 1. Instanciar el diseño que vamos a verificar
  - El testbench será el nuevo top-level
  - Será una entidad sin ports
- 2. Escribir el código que:
  - Genera los estímulos
  - Observa los resultados
  - Informa al usuario



# Ejemplo

Vamos a comprobar que funciona:

$$y = a \cdot b$$

#### **VHDL**

### Testbench (parte 1, instanciación)

```
entity testbench1 is -- no hay entradas ni salidas (ports)
end;
architecture test of testbench1 is
  component MiAnd -- declaración del uut
  port (a, b: in std_logic;
      y: out std_logic);
  end component;
  -- Señales para conectar todos los puertos del uut
  signal a, b, y: std_logic; -- Pueden ser nombres distintos
begin
  uut: MiAnd port map(
      a => a,
      b \Rightarrow b
      y => y);
```

# Testbench (parte 2, gen. estímulos)

```
-- continuación código anterior
-- Los estímulos se generan en un process
process -- No hay lista de sensibilidad porque hay wait
begin
  a \le 0'; b \le 0';
  wait for 10 ns;
  a <= `0'; b <= `1';
  wait for 10 ns;
  a \le 1'; b \le 0';
  wait for 10 ns;
  a \le 1'; b \le 1';
  wait for 10 ns;
  wait; -- "cuelga" este proceso, si no vuelve a empezar
end process;
end; -- end del architecture
```

El simulador debe analizar el valor de la señal 'y' en cada caso y comprobar el correcto funcionamiento

### Comprobación automática

Para comprobar resultados se utiliza la sentencia assert

```
assert condicion report "Texto" severity nivel;
```

Verifica que condicion se cumple. Si no, saca Texto por el log del simulador y genera una excepción del nivel que se haya especificado.

Dependiendo del nivel (note, warning, <u>error</u>, failure), el simulador parará o no (configurable por usuario).

### Ejemplo auto-comprobación MiAnd

```
process -- No hay lista de sensibilidad porque hay wait
begin
  a <= `0'; b <= `0';
  wait for 10 ns;
  assert y = '0' report "Falla 00" severity error;
  a \le 0'; b \le 1';
  wait for 10 ns;
  assert y = '0' report "Falla 01" severity error;
  a <= 1'; b <= 0';
  wait for 10 ns;
  assert y = '0' report "Falla 10" severity error;
  a <= '1'; b <= '1';
  wait for 10 ns;
  assert y = '1' report "Falla 11" severity error;
  wait; -- "cuelga" este proceso, si no vuelve a empezar
end process;
```

#### Testbenches de circ. secuenciales

El reloj se genera normalmente en un proceso aparte:

```
process
begin
  Clk <= '0';
  wait for CICLO/2; -- CICLO es una constante
  Clk <= '1';
  wait for CICLO/2;
end process; -- al no haber wait final es cíclico</pre>
```

Sería mejor generarlo sólo mientras una señal auxiliar se mantenga activa, para poder acabar la simulación, ver ejemplo de código while.

### Ejemplo para un contador

```
process -- No hay lista de sensibilidad porque hay wait
begin
  Reset <= '1'; -- Activar siempre el reset al comienzo
  wait for CICLO;
  Reset <= '0'; -- Se desactiva para que pueda avanzar
  for i in 0 to 255 loop
   assert conv_integer(Cuenta) = i
      report "Falla en " & integer'image(i)
         severity error; -- Cuenta es std_logic_vector
   wait for CICLO; -- El reloj se genera en paralelo con
                         -- otro proceso, ver pág. anterior
  end loop;
                        -- Cierra el bucle for
end process;
```

#### Listas de sensibilidad: testbenches

- Los procesos de los testbenches pueden no tener listas de sensibilidad.
- Para ello es necesario que haya alguna sentencia wait.
- Si no la hubiera, el simulador se bloquearía, ya que no podría avanzar el tiempo de simulación.

#### Constantes

- Como en cualquier otro lenguaje, en VHDL se pueden utilizar constantes.
- Se declaran también en la parte declarativa, entre architecture y begin, y se deben inicializar:

```
architecture ejemplo of prueba is
   constant C1 : std_logic_vector(3 downto 0) := "0101";
   constant C2 : integer := 5;
   constant CICLO : time := 10 ns;
begin
```

Las constantes pueden ser de cualquier tipo.

### Conversiones de tipos

1.25, 1.18

VHDL es fuertemente tipado, no hay cast automático:

```
architecture ejemplo of prueba is
    signal s1 : std_logic_vector(3 downto 0);
    signal s2 : integer;
begin
    s1 <= s2; -- Incorrecto en VHDL
    s2 <= s1; -- También incorrecto</pre>
```

 Se usan las funciones conv\_std\_logic\_vector y conv\_integer:

```
s1 <= conv_std_logic_vector(s2, 4);
-- Función declarada en std_logic_arith
-- Segundo parámetro: número de bits
s2 <= conv_integer(s1);
-- Declarada en std_logic_signed o std_logic_unsigned
-- No hay segundo parámetro</pre>
```

64

#### Estructura de Computadores

### Unidad 1: Diseño digital y VHDL

Escuela Politécnica Superior - UAM

#### Problemas U1

**1.9.-** Escriba el código VHDL para generar un multiplexor cuya salida es la señal z, y cuyas entradas de datos son a0, a1, a2 y a3, todas de tipo std\_logic\_vector(31 downto 0). La señal de control es sel, de tipo std\_logic\_vector(1 downto 0). No hace falta que incluya la entity ni la architecture, sólo el código para la funcionalidad del MUX.

T

- **1.4.-** Siendo *fuente* un *std\_logic\_vector(3 downto 0)* y *dest* un *std\_logic\_vector(6 downto 0)*, escriba el código para que *dest* sea igual a *fuente* multiplicado por 8 sin utilizar la multiplicación ni la suma.
- **1.10.-** Dada la señal *dato4*, de tipo *std\_logic\_vector(3 downto 0)*, escriba el código VHDL para que *salida8*, de tipo *std\_logic\_vector(7 downto 0)*, sea *dato4* extendida en signo.

T

#### Problemas U1

**1.3.-** Complete la lista de sensibilidad del proceso adjunto.

```
case expresion is
  when caso_1 =>
    -- sec instr 1
  when caso_2 =>
    -- sec instr 2
  when others =>
    -- instr por defecto
end case;
```

**1.2.-** Escriba el código equivalente al proceso adjunto, utilizando una sentencia concurrente de tipo *when – else*. La lista de sensibilidad no se indica, pero suponga que es la correcta.

```
1.11.- Escriba el código equivalente al indicado con una sentencia de tipo case. Añada un proceso y su lista de sensibilidad si fuera necesario.
```

```
process(...)
begin
  if a = b then
    c <= d xor e;
  elsif f = '1' then
    g <= h;
  else
    i <= i + 4;
  end if;
end process;</pre>
```

```
process( ... )
begin
  if a < b then
    s <= b;
  elsif a < c then
    s <= c;
  else
    s <= a;
  end if;
end process;</pre>
```

```
with s select
   z <= a when "00",
        b when "01",
        c when others;</pre>
```

#### 1.17. Multiplexor 4 a 1 con entradas de 8 bits con 4 arquitecturas diferentes:

```
architecture ArchWhen of Mux4a1 is -- a) Arquitectura basada en la asignación condicional WHEN ELSE

begin
    Y <= D0 when Sel = "00" else
    D1 when Sel = "01" else
    D2 when Sel = "10" else
    D3; -- Sel = "11"

end ArchWhen;
```

```
architecture Archlf of Mux4a1 is -- b) Arquitectura basada en la sentencia IF

begin
-- Por ser combinacional, en la lista de sensibilidad hay que poner todas las señales que intervienen process (Sel, D0, D1, D2, D3)
begin
if Sel = "00" then Y <= D0;
elsif Sel = "01" then Y <= D1;
elsif Sel = "10" then Y <= D2;
else Y <= D3;
end if;
end process;
end Archlf;
```

#### 1.17. Multiplexor 4 a 1 con entradas de 8 bits con 4 arquitecturas diferentes:

library IEEE;

end ArchWith;

```
use IEEE.std logic 1164.all;
entity Mux4a1 is
    port (
            D0, D1, D2, D3: in std logic vector(7 downto 0);
            Sel: in std logic vector(1 downto 0);
            Y: out std logic vector(7 downto 0)
end Mux4a1;
architecture ArchWith of Mux4a1 is -- c) Arquitectura basada en asignación concurrente WITH-SELECT
begin
    with Sel select
         Y <=
                     D0 when "00",
                     D1 when "01",
                     D2 when "10",
                     D3 when others;
                                       -- equivale a "11", con others para no olvidarnos casos
```

```
architecture ArchCase of Mux4a1 is -- d) Arquitectura basada en la sentencia CASE

begin

process (Sel, D0, D1, D2, D3)
begin

case Sel is

when "00" => Y <= D0;
when "01" => Y <= D1;
when "10" => Y <= D2;
when "10" => Y <= D3;
-- Se cierra con others
end case;
end process;
end ArchCase;
```

#### Problemas U1

**1.12.-** Siendo *clk, reset, d* y *q* señales de tipo *std\_logic*, escriba el código para que *q* sea un biestable de tipo D activo por flanco de bajada con una señal de reset asíncrono activa a nivel alto.

**1.15.-** Utilizando VHDL, diseñe un proceso que, incluyendo la lista de sensibilidad oportuna, modele un flip-flop tipo D con entrada síncrona de Reset activo en alto y reloj activo por flanco de subida. Se adjunta un esquema de la entidad a modelar.



T

#### 1.19. Diseñar un contador ascendente de 8 bits con reset asíncrono y carga en paralelo

```
library IEEE;
use IEEE.std_logic_1164.all;
use IEEE.std_logic_arith.all;
use IEEE.std_logic_unsigned.all;

entity Cont8Load is
   port (
        Clk , Reset, Load : in std_logic;
        Data : in std_logic_vector(7 downto 0);
        Q : out std_logic_vector(7 downto 0)
        );
end Cont8Load;
```

### 1.20. Contador ascendente de 8 bits con reset asinc. y carga en paralelo que se elige entre cuatro posibles. Utilice los módulos diseñados anteriormente (multiplexor y contador)

```
architecture Practica of Cont8Mux is
                                                    -- Declaración del componente mux
          component Mux4a1
          port (
               Sel: in std logic vector(1 downto 0);
               D0, D1, D2, D3: in std logic vector(7 downto 0);
               Y: out std logic vector(7 downto 0)
          end component;
          component Cont8Load
                                                    -- Declaración del componente contador
          port (
               Clk, Reset, Load: IN std logic;
               Data: IN std logic vector(7 downto 0);
               Q: OUT std logic vector(7 downto 0)
          end component;
          signal Data : std logic vector(7 downto 0); -- Declaración de señales internas, el dato que sale del mux y va al
                                                     -- contador no es un puerto
```

### 1.20. Contador ascendente de 8 bits con reset asinc. y carga en paralelo que se elige entre cuatro posibles. Utilice los módulos diseñados anteriormente (multiplexor y contador)

```
Begin
                         -- Comienzo de la arquitectura Practica de Cont8Mux is
            ElMux: Mux4a1 port map(
                                                  -- Instanciación (mapeo) componente Mux4a1
                         D0 => D0.
                         D1 => D1,
                         D2 => D2,
                         D3 => D3,
                         Sel => Sel,
                         Y => Data
            );
            ElContador: Cont8Load port map(
                                                   -- Instanciación (mapeo) componente Cont8Load
                         Clk => Clk,
                         Reset => Reset,
                         Load => Load,
                         Data => Data,
                         Q => Q
            );
end Practica;
```

#### Problemas U1

- **1.6.-** Escriba una sentencia de tipo *assert* para comprobar en un test-bench que la señal s está a '1' o si no indicar "La señal s no vale 1" y parar la simulación.
- 1.25.- Dibuje el circuito que sería sintetizado con el siguiente código VHDL. Todas las conexiones deben estar etiquetadas en el esquemático. Se sabe que A es una señal de tipo std\_logic\_vector(7 downto 0) Además, todos los componentes deben ser identificados correctamente, bien mediante un símbolo estándar o bien explicando su funcionamiento en texto auxiliar.

**1.18.-** Diseñe un *testbench* para un multiplexor 4 a 1 con entradas de 8 bits del ejercicio 1.18.

#### 1.18. Diseñar un TestBench para el Multiplexor 4 a 1 diseñado:

```
library IEEE;
use IEEE.std logic 1164.all;
entity Mux4a1Tb is
end Mux4a1Tb;
                                                        entity Mux4a1 is
                                                             port ( D0, D1, D2, D3 : in std_logic_vector(7 downto 0);
                                                                   Sel: in std logic vector(1 downto 0);
                                                                   Y: out std logic vector(7 downto 0));
                                                        end Mux4a1;
architecture Test of Mux4a1Tb is
             component Mux4a1
                  port (D0, D1, D2, D3: in std logic vector(7 downto 0);
                       Sel: in std logic vector(1 downto 0);
                       Y: out std logic vector(7 downto 0));
             end component;
             signal sD0, sD1, sD2, sD3, sY: std logic vector(7 downto 0); -- Declaración de señales
             signal sSel: std logic vector(1 downto 0);
             constant CICLO: time := 10 ns;
                                                    -- Constante de tiempo
begin
             uut: Mux4a1 port map (
                                                    -- Se instancia el módulo a probar (Unit Under Test)
                          D0 \Rightarrow sD0.
                          D1 => sD1,
                          D2 => sD2
                          D3 => sD3.
                          Sel => sSel.
                          Y => sY
             );
```

#### 1.18. Diseñar un TestBench para el Multiplexor 4 a 1 diseñado:

```
-- Continuación de la architecture Test of Mux4a1 is
          ProcPrinc: process
                                        -- Generación de estímulos y comprobación de resultados
          begin
                 sD0 <= (others => '0');
                 sD1 <= "00000001";
                 sD2 <= X"02":
                 sD3 <= (1 downto 0 => '1', others => '0');
                 sSel <= "00":
                 wait for CICLO;
                 assert sY = sD0 report "Falla en el caso 00" severity failure;
                                        -- Las señales que no cambien no hace falta volver a ponerlas
                                        -- porque por defecto conservan su valor
                 sSel <= "01";
                 wait for CICLO:
                 assert sY = sD1 report "Falla en el caso 01" severity failure;
                 sSel <= "10":
                 wait for CICLO;
                 assert sY = sD2 report "Falla en el caso 10" severity failure;
                 sSel <= "11";
                 wait for CICLO:
                 assert sY = sD3 report "Falla en el caso 11" severity failure;
                 assert false report "Si ha llegado aquí sin fallos previos, funciona bien" severity note;
                 wait;
                                        -- Mata este proceso, y al ser el único finaliza la simulación
          end process;
end Test;
```