# Laboratorio N° 1

 $Dise\~no\ Digital\ Avanzado$ 

Mathias Sebastian Garcia

# Índice

|    | Código RTL           .1. Contador           .2. Shift register | 4  |
|----|----------------------------------------------------------------|----|
|    | .3. Top                                                        | ļ  |
| 2. | Testbench                                                      | 7  |
|    | .1. Código                                                     | -  |
|    | .2. Capturas de simulación                                     | Ç  |
| 3. | mplementación en FPGA                                          | .( |
|    | mplementación en FPGA<br>.1. Wrapper VIO                       | L( |
|    | .2. Wrapper ILA                                                | Ĺ  |
|    | .3. Wrapper completo                                           |    |
|    | 4. Capturas                                                    | 14 |

# 1. Código RTL

Link de github a los archivos de RTL.

#### 1.1. Contador

```
module counter
   // * Parameters
    // * -----
                                     NB_COUNTER = 32
   parameter
                                          NB_SW
                                                       = 3
    parameter
                                          ADD_PIPE_OUT = 0
    parameter
    // * Outputs
    // * ------
    output logic
                                         o_count_reached
    // * Inputs
// * -----
    input logic [ NB_SW - 1 : 0 ] i_sw
    // * Clock and reset
    // * -----
                                  i_reset
i_clock
    input logic
   input logic
    // * -----
    // * Localparams
    // * -----
                                          RO = 2**(NB_COUNTER-10)-1;
R1 = 2**(NB_COUNTER-11)-1;
R2 = 2**(NB_COUNTER-12)-1;
    localparam
    localparam
    localparam
                                                       = 2**(NB_COUNTER-13)-1 ;
    localparam
    // * Internal logics
             [ NB_COUNTER - 1 : 0 ] count
[ NB_COUNTER - 1 : 0 ] next_count
[ NB_COUNTER - 1 : 0 ] limit_count
    logic
    logic
    logic
                                          limit_reached
    logic
    logic
                                          count_enable
                                          piped_limit_reached
    logic
    // * Max count decode
    // * -----
    always_comb
    {\tt begin} \; : \; {\tt proc\_limit\_count\_decode}
        unique case (i_sw[2:1]) inside
2'b00: limit_count = RO[NB_COUNTER-1:0];
            2'b01: limit_count = R1[NB_COUNTER-1:0];
2'b10: limit_count = R2[NB_COUNTER-1:0];
            2'b11: limit_count = R3[NB_COUNTER-1:0];
        endcase
    end
    // * Count process
   assign next_count = count + {{NB_COUNTER-1{1'b0}}, 1'b1} ; assign count_enable = i_sw[0] ;
    assign limit_reached = (count >= limit_count) && count_enable;
    always_ff @(posedge i_clock)
    begin : proc_count
        if (i_reset || limit_reached)
           count <= '0;
        else if (count_enable)
           count <= next_count;</pre>
    end
    // * Piping output
```

```
// *
generate
    if (ADD_PIPE_OUT)
    begin : gen_out_piped
        always_ff @(posedge i_clock)
        begin : proc_pipe_out
            piped_limit_reached <= limit_reached;
        end
    end else
    begin : gen_out_unpiped
        assign piped_limit_reached = limit_reached;
    end
endgenerate

// *
// * Output assignment
// *
// * output assignment
// *
endmodule</pre>
```

## 1.2. Shift register

```
module shiftreg
#(
   // * Parameters
   // * -----
                            NB_DATA = 3
   parameter
   // * Outputs
   // * -----
   output logic [ NB_DATA - 1 : 0 ] o_led ,
   // * Inputs
   // * -----
   input logic
                      i_valid ,
   // * Clock and reset
   // * -----
                 i_reset
i_clock
   input logic
   input logic
);
   // * Internal logics // * -----
   logic     [ NB_DATA - 1 : 0 ] shift_register ;
   // * -----
   // * Shift register
   // * ---
   always_ff @(posedge i_clock)
   begin : proc_data_move
      if (i_reset)
        shift_register <= {{NB_DATA-1{1'b0}}, 1'b1};</pre>
      else if (i_valid)
        shift_register <= {shift_register[NB_DATA-2:0], shift_register[NB_DATA-1]};</pre>
   end
   // * Output assignment
   assign o_led = shift_register;
endmodule
```

#### 1.3. Top

```
module shiftleds
  // * Parameters
  // * -----
                 N_LEDS = 4
  parameter
                            NB_COUNTER = 32
  parameter
                            NB_SW = 4
  parameter
  // * Outputs
   // * -----
  output logic [ N_LEDS - 1 : 0 ] o_led output logic [ N_LEDS - 1 : 0 ] o_led_b output logic [ N_LEDS - 1 : 0 ] o_led_g
  // * Inputs
// * ------
  input logic [ NB_SW - 1 : 0 ] i_sw ,
   // * -----
   // * Clock and reset
   // * -----
              i_reset
i_clock
  input logic
  input logic
   // * -----
   // * Internal logics
  // * -----
  //* Counter instantiation
   // * ------
  counter
     .NB_COUNTER ( NB_COUNTER .NB_SW ( NB_SW - 1
  u_counter
     .o_count_reached ( counter_count_reached ) ,
     .i_sw (i_sw [NB_SW-2:0]),
.i_reset (~i_reset ),
.i_clock (i_clock )
              ( i_clock
     .i_clock
   // * Shift register instantiation
  // * ---
  shiftreg
  #(
     .NB_DATA ( N_LEDS )
  u_shiftreg
      .o_led
               ( shiftreg_led
     .i_valid
     .i_reset
     .i_clock
  // * Output assignment
   // * -----
  assign o_led = shiftreg_led ;
assign o_led_b = (i_sw[3]) ? shiftreg_led : '0 ;
assign o_led_g = (i_sw[3]) ? '0 : shiftreg_led;
   // * Assertions
   // * -----
   ifdef ENABLE_SVA
     logic enable_sva = 1'b1;
     assert property (
       @(posedge i_clock)
```

```
disable iff (!enable_sva)
            counter_count_reached |=> !counter_count_reached
       ) else $error("Valid signal for shift register lasted more than one clock");
        assert property (
            @(posedge i_clock)
disable iff (!enable_sva)
            $rose(counter_count_reached) |=> (u_counter.count == 0)
        ) else $error("Counter limit reached but count register didnt go back to zero. Count value: %0d", u_counter.count);
        assert property (
            @(posedge i_clock)
            disable iff (!enable_sva)
            (u_counter.count == u_counter.limit_count) |-> $rose(counter_count_reached)
        ) else $error("Limit count reached but count reached never rose");
        assert property (
            @(posedge i_clock)
            disable iff (!enable_sva)
            $fell(i_reset) |=> (u_counter.count == 0)
        ) else $error("Reset negedge happened but counter didnt set to default value. Count value: %Od", u_counter.count);
        assert property (
            @(posedge i_clock)
            disable iff (!enable_sva)
            fell(i\_reset) \mid \Rightarrow (u\_shiftreg.shift\_register == \{\{N\_LEDS-1\{1'b0\}\}, \ 1'b1\})
        ) else $error("Reset negedge happened but shift register didnt set to default value. Value found: %0b", u_shiftreg.shift_register);
        assert property (
            @(posedge i_clock)
            disable iff (!enable_sva)
            !i_sw[0] |=> $stable(u_counter.count)
       ) else $error("Count is not stable during disable");
    `endif
endmodule
```

### 2. Testbench

Link de github al archivo de testbench.

#### 2.1. Código

```
`timescale 1ns/100ps
define N_LEDS
`define NB SEL
define NB_COUNTER
define NB_SW
`define CLOCK_PERIOD_NS 10
module tb_shiftleds();
   // * Localparams
   // * -----
                   N_LEDS = `N_LEDS
NB_SEL = `NB_SEL
   localparam
   localparam
                             NB_COUNTER = `NB_COUNTER;
NB_SW = `NB_SW ;
   localparam
   localparam
   // * Declaration
   // * -----
   wire [ N_LEDS
   reg [ NB_SW
                              i_reset
   reg
   reg
                               i_clock
   wire [ NB_COUNTER - 1 : 0 ] active_count_limit
   wire [ NB_COUNTER - 1 : 0 ] counter_limit [4]
   reg [ NB_COUNTER - 1 : 0 ] counter_record
   // * Observers assigns
   // * -----
   assign counter_limit[0] = u_shiftleds.u_counter.R0;
   assign counter_limit[1] = u_shiftleds.u_counter.R1;
   assign counter_limit[2] = u_shiftleds.u_counter.R2;
   assign counter_limit[3] = u_shiftleds.u_counter.R3;
   assign active_count_limit = (i_sw[2:1] == 2'b00 ) ? counter_limit[0] :
                               (i_sw[2:1] == 2'b01 ) ? counter_limit[1] :
                               (i_sw[2:1] == 2'b10 ) ? counter_limit[2] :
                                                      counter_limit[3] ;
   // * Clock generation
   // * -----
   initial
   {\tt begin} \; : \; {\tt proc\_clock\_initializacion}
      i_clock = '0;
   end
   always #(`CLOCK_PERIOD_NS/2) i_clock = ~i_clock;
   // * Tasks
   // * -----
   task automatic wait_for_n_clocks(int n_clocks = 1, bit safe_check = 1'b1);
       repeat(n_clocks) @(posedge i_clock);
       if (safe_check) #1ns;
   {\tt endtask} \; : \; {\tt wait\_for\_n\_clocks}
   task automatic run_reset(int clock_duration = 1);
       \label{locks} \verb|wait_for_n_clocks(.n_clocks(1), .safe_check(0)); \\
       force i_reset = 1'b0;
       wait_for_n_clocks(.n_clocks(clock_duration), .safe_check(0));
       force i_reset = 1'b1;
   endtask : run_reset
   task automatic check_led_change(int n_changes = N_LEDS);
       logic [N_LEDS-1:0] led_record;
       repeat (n_changes) begin : proc_check_repeatedly
           led_record = u_shiftleds.o_led;
           wait_for_n_clocks(active_count_limit);
```

```
assert(u_shiftleds.counter_count_reached == 1'b1);
       wait_for_n_clocks(1);
       assert(o_led == {led_record[N_LEDS-2:0], led_record[N_LEDS-1]});
       if (i_sw[3]) begin
           assert (o_led_b == o_led);
           assert (o_led_g == '0 );
       end else begin
          assert (o_led_b == '0 );
           assert (o_led_g == o_led);
       end
       $display("Led value: %4b - Led recorded: %4b - Expected value: %4b", o_led, led_record, {led_record[N_LEDS-2:0], led_record[N_LEDS-1]]}
   end
endtask : check_led_change
// * Assertion handling
// * -----
initial
begin : proc_bypass_sva_first_clock
   $assertoff(0,u_shiftleds);
   wait_for_n_clocks(.n_clocks(2), .safe_check(0));
   $asserton(0,u_shiftleds);
end
// * -----
// * Stimulus generation
// * -----
initial
{\tt begin} \; : \; {\tt proc\_stimulus}
   i_sw = '0;
i_reset = 1'b0;
   run_reset(5);
   wait_for_n_clocks(20);
   i_sw[0] = 1'b1;
   check_led_change(8);
   wait_for_n_clocks(4);
   i_sw[2:1] = 2'b01;
   run_reset();
   check_led_change(8);
   i_sw[3] = 1'b1;
   i_sw[2:1] = 2'b10;
   run_reset();
   check_led_change(8);
   i_sw[2:1] = 2'b11;
   run_reset();
   check_led_change(8);
   wait_for_n_clocks(.n_clocks(100), .safe_check(0));
   i_sw[0] = 1'b0;
   wait_for_n_clocks(.n_clocks(2000), .safe_check(0));
   counter_record = u_shiftleds.u_counter.count;
   i_sw[0] = 1'b1;
   wait_for_n_clocks(1);
   assert(u_shiftleds.u_counter.count == counter_record+1);
   $finish:
end
// * DUT instantiation
// * -----
shiftleds
#(
   .N_LEDS ( N_LEDS ) , .NB_COUNTER ( NB_COUNTER ) ,
   .NB_SW
               ( NB_SW
u\_shiftleds
               (o_led
   .o_led
   .o_led_b
               ( o_led_b
                          ),
                           ) ,
   .o_led_g
               ( o_led_g
                           ),
   .i_sw
               (i_sw
               (i_reset ),
   .i_reset
   .i_clock
               ( i_clock
```

endmodule

# 2.2. Capturas de simulación



Figura 1: Operación normal



Figura 2: sw[0]=0 mantiene la cuenta y los leds sin alterar



Figura 3: Volver a setear sw[0]=1 retoma la cuenta desde el valor previo

# 3. Implementación en FPGA

Link de github a los archivos de los wrappers.

#### 3.1. Wrapper VIO

```
//Copyright 1986-2022 Xilinx, Inc. All Rights Reserved.
//Copyright 2022-2023 Advanced Micro Devices, Inc. All Rights Reserved.
//Tool Version: Vivado v.2023.1 (lin64) Build 3865809 Sun May 7 15:04:56 MDT 2023
                n. vivado v.2023.1 (viva) butta 360809 sun nay 7.

: Sat Sep 28 23:19:27 2024

: dsktp running 64-bit Fedora Linux 40 (KDE Plasma)

: generate_target vio_wrapper.bd
//Date
//Host
//Command
//Design
                 <li: vio_wrapper</li>: IP block netlist
//Purpose
`timescale 1 ps / 1 ps
module vio_wrapper
    (clk_0,
     probe_in0_0,
     probe_in1_0,
     probe_in2_0,
     probe_out0_0,
     probe_out1_0,
     probe_out2_0);
  input clk_0;
  input [3:0]probe_in0_0;
  input [3:0]probe_in1_0;
  input [3:0]probe_in2_0;
  output [0:0]probe_out0_0;
  output [0:0]probe_out1_0;
  output [3:0]probe_out2_0;
  wire clk_0;
  wire [3:0]probe_in0_0;
  wire [3:0]probe_in1_0;
  wire [3:0]probe_in2_0;
  wire [0:0]probe_out0_0;
  wire [0:0]probe_out1_0;
  wire [3:0]probe_out2_0;
  vio vio_i
         (.clk_0(clk_0),
          .probe_in0_0(probe_in0_0),
          .probe_in1_0(probe_in1_0),
          .probe_in2_0(probe_in2_0),
           .probe_out0_0(probe_out0_0),
          .probe_outt_0(probe_outt_0),
.probe_out1_0(probe_out1_0),
.probe_out2_0(probe_out2_0));
endmodule
```

### 3.2. Wrapper ILA

```
//Copyright 1986-2022 Xilinx, Inc. All Rights Reserved.
//Copyright 2022-2023 Advanced Micro Devices, Inc. All Rights Reserved.
//Tool Version: Vivado v.2023.1 (lin64) Build 3865809 Sun May 7 15:04:56 MDT 2023
                : Sat Sep 28 23:21:04 2024
: dsktp running 64-bit Fedora Linux 40 (KDE Plasma)
: generate_target ila_wrapper.bd
: ila_wrapper
: IP block netlist
//Date
//Host
//Command
//Design
//Purpose
`timescale 1 ps / 1 ps
module ila_wrapper (clk_0, probe0_0, probe1_0,
     probe2_0);
  input clk_0;
input [3:0]probe0_0;
input [3:0]probe1_0;
  input [3:0]probe2_0;
  wire clk_0;
wire [3:0]probe0_0;
wire [3:0]probe1_0;
wire [3:0]probe2_0;
   ila ila_i
          (.clk_0(clk_0),
           .probe0_0(probe0_0),
           .probe1_0(probe1_0),
           .probe2_0(probe2_0));
endmodule
```

#### 3.3. Wrapper completo

```
module shiftleds_wrapper (
   // * Outputs
   // * -----
   output logic [4-1:0] o_led ,
   output logic [4-1:0] o_led_b
   output logic [4-1:0] o_led_g
   // * Inputs
   // * ------
   input logic [4-1:0] i_sw ,
   // * Clock and reset
   // * -----
               i_reset_n , i_clock
   input logic
   input logic
   // * Internal logics
   // * -----
  // * -----
   // * VIO instantion
   // * -----
   vio_wrapper
   u_vio_wrapper
       .clk_0 (i_clock
       .clk_0 ( i_clock ) , .probe_in0_0 ( shiftleds_led ) ,
       .probe_in1_0 ( shiftleds_led_b ) ,
       .probe_in2_0 ( shiftleds_led_g ) ,
       .probe_out0_0 ( vio_enable ) ,
.probe_out1_0 ( vio_reset ) ,
.probe_out2_0 ( vio_sw )
   // * ILA instantion
   ila_wrapper
   u_ila_wrapper
                 ( i_clock ) ,
( shiftleds_led ) ,
( shiftleds_led_b ) ,
( shiftleds_led_g )
       .clk_0
       .probe0_0
       .probe1_0
      .probe2_0
   );
   // * Muxing shiftleds input
   // * -----
   assign sw_to_device = (vio_enable) ? vio_sw : i_sw
   assign reset_to_device = (vio_enable) ? ~vio_reset : i_reset_n ;
   // * Main device
   // * -----
   shiftleds
   u_shiftleds
                 ( shiftleds_led ) ,
( shiftleds_led_b ) ,
( shiftleds_led_g ) ,
( sw_to_device ) ,
( reset_to_device ) ,
( i_clock )
      .o_led
      .o_led_b
      .o_led_g
      .i_sw
       .i_reset
       .i_clock
   );
   // * -----
```

```
// * Output assign
// *
assign o_led = shiftleds_led ;
assign o_led_b = shiftleds_led_b ;
assign o_led_g = shiftleds_led_g ;
endmodule
```

## 3.4. Capturas



(a) VIO



Figura 4: Led verde seleccionado



(a) VIO



Figura 5: Led azul seleccionado



(a) VIO



(b) FPGA - Led 4 prendido



(c) ILA - Sampling constante en  $50\,\%$ 

Figura 6: Led quieto



Figura 7: FPGA en reset - Led0 prendido