



|                     |                     |              |
|---------------------|---------------------|--------------|
| Name: Nabeeha Aamir | Student ID: na10076 | Section: T-4 |
| Divya Lakhani       | dl10267             |              |
| Areesha Ashfaq      | aa09879             |              |

## Lab 5: Designing FSM Using FPGA Switches and LEDs

### Objective

The objective of this lab is to design, implement, and verify a Finite State Machine (FSM) on an FPGA using input switches, LEDs, and a reset button. Students will learn how to interface user inputs with hardware logic, implement state-based control, and manage synchronous counters using Verilog HDL.

By the end of this lab, students will be able to:

- Design an FSM using a state diagram
- Interface FPGA switches and LEDs with control logic
- Implement a decrement counter controlled by FSM states
- Integrate provided switch, button debouncing, and LED interface modules
- Verify FSM behavior through FPGA synthesis and on-board testing

### System Description

The system behavior is defined as follows:

- The system continuously monitors the FPGA switches for input.
- If the switch input value is **zero**, the system remains in the input-waiting state.
- When the switch input becomes **non-zero**, the FSM:
  - Captures the switch value
  - Displays the value on the LEDs
  - Starts a decrement counter from the captured value down to zero
- While counting down:
  - The LEDs reflect the current counter value
  - Switch inputs are ignored
- When the counter reaches zero:
  - The FSM automatically returns to the input-waiting state
- At any time during counting, pressing the **reset button**:
  - Immediately returns the FSM to the input-waiting state
  - Clears the counter and LED output

**Note:** The switch interface, button debouncing interface, and LED interface are provided and must be used



---

without modification.

## Task 1: FSM Design and State Diagram

### Objective

To design an FSM that controls system behavior based on switch input, counter value, and reset signal.

### Procedure

- (a) Identify the required FSM states, such as:

- Input Waiting State
- Countdown State
- Reset State (if implemented explicitly)

- (b) Draw a complete FSM state diagram showing:

- State transitions
- Conditions for transitions (switch input, counter reaching zero, reset)
- Outputs associated with each state

- (c) Clearly indicate:

- How switch input is latched
- When the counter is enabled or disabled
- How LEDs are driven in each state

### Deliverables

- FSM state diagram with labeled transitions and states





## Task 2: Verilog FSM and Counter Implementation

### Objective

To implement the FSM and decrement counter in Verilog HDL and verify functionality through simulation.

### Procedure

- (a) Implement the FSM in Verilog:
  - Use synchronous state transitions triggered by the system clock
  - Include an asynchronous or synchronous reset as specified
- (b) Implement a decrement counter:
  - Load the counter with the switch input value
  - Decrement the counter on each clock cycle while enabled



- Stop counting when the counter reaches zero
- (c) Integrate the provided modules:

- Switch interface:

```
module switches(
    input clk,
    input rst,
    input [31:0] writeData,
    input writeEnable,
    input readEnable,
    input [29:0] memAddress,
    output reg [31:0] readData = 0, // not to be read
    output reg [15:0] leds
);
    ...
endmodule
```

- Button debouncing interface:

```
module debouncer(
    input clk,
    input pbin,
    output pbout
```

- LED interface:

```
module leds(
    input clk, rst,
    input [15:0] btns,
    input [31:0] writeData, // not to be written
    input writeEnable, // not to be used
    input readEnable,
    input [29:0] memAddress,
    input [15:0] switches,
    output reg [31:0] readData
```

**Clock divider module:**

```
module clock_divider(
    input clk, //input clock
    input rst, //reset btn
    output reg tick //after one sec tick
);

reg [26:0] count; //register of space 27

always @(posedge clk or posedge rst) begin
    if (rst) begin //if reset is enabled
        count <= 0;
        tick <= 0;
    end
    else if (count == 20-1) begin
        count <= 0;
        tick <= 1;
    end
    else begin
        count <= count + 1;
        tick <= 0;
    end
end

endmodule
```

**Debouncer module:**

```
module debouncer(
    input clk,
    input pbin,
    output reg pbout
);

reg [19:0] counter = 0;
reg sync_0 = 0;
reg sync_1 = 0;

always @(posedge clk) begin
    sync_0 <= pbin;
    sync_1 <= sync_0;
end

always @(posedge clk) begin
    if (pbout == sync_1)
        counter <= 0;
    else begin
        counter <= counter + 1;
        if (counter == 1_000_000) begin
            pbout <= sync_1;
            counter <= 0;
        end
    end
end
endmodule
```



## Priority encoder module:

```
module priority_encoder(
    input [15:0] sw,
    output reg [3:0] value,
    output reg valid
);

integer i;

always @(*) begin
    value = 0;
    valid = 0;

    for (i = 15; i >= 0; i = i - 1) begin
        if (sw[i]) begin
            value = i[3:0];
            valid = 1;
        end
    end
end
endmodule
```

## Fsm counter module:

```
module fsm_counter(
    input clk,
    input rst,
    input tick,
    input [3:0] start_value, //starting value to display
    input start_valid,
    output reg [3:0] display_value, //displaying current value on the seven seg display
    output reg busy //if its counting down, we need to know that
);
localparam IDLE = 2'b00; //IDLE STATE 00
localparam WAIT1 = 2'b01; //wait a second before counting down
localparam COUNT = 2'b10;
reg [1:0] state;
reg [3:0] counter;
reg start_valid_d;
always @ (posedge clk or posedge rst) begin
    if (rst)
        start_valid_d <= 0;
    else
        start_valid_d <= start_valid;
end

wire start_pulse = start_valid & ~start_valid_d;

always @ (posedge clk or posedge rst) begin
    if (rst) begin
        state <= IDLE;
        counter <= 0;
```



```
    counter <= 0;
    display_value <= 0;
    busy <= 0;
end
else begin
    case(state)
    IDLE: begin
        busy <= 0;
        if (start_pulse) begin
            counter <= start_value;
            display_value <= start_value;
            busy <= 1;
            state <= WAIT1; //waiting for one full tick
        end
    end
    WAIT1: begin
        busy <= 1;
        if (tick)
            state <= COUNT;//go to the next number after 1 sec
    end
    COUNT: begin
        busy <= 1;
        if (tick) begin
            if (counter > 0) begin
                counter <= counter - 1;
                display_value <= counter - 1;
            end
            else begin
                else begin
                    state <= IDLE;
                    busy <= 0;
                end
            end
        end
    end
endcase
end
endmodule
```

### Switches module:

```
module switches(
    input clk,
    input rst,
    input [31:0] writeData,
    input writeEnable,
    input readEnable,
    input [29:0] memAddress,
    input [15:0] sw, //real life switches
    output reg [31:0] readData,
    output reg [15:0] leds
);

    always @(posedge clk or posedge rst) begin
        if (rst) begin
            readData <= 0;
            leds <= 0;
        end
        else begin
            leds <= sw;
            if (readEnable)
                readData <= {16'b0, sw};
            else
                readData <= 0;
        end
    end
endmodule
```



### LEDs module:

```
1 module leds(
2     input clk,
3     input rst,
4     input [15:0] btns,
5     input [31:0] writeData,
6     input writeEnable,
7     input readEnable,
8     input [29:0] memAddress,
9     input [3:0] display_value, //read the current value from FSM
10    output reg [31:0] readData,
11    output [6:0] seg,
12    output [3:0] an
13 );
14
15     always @ (posedge clk or posedge rst) begin
16         if (rst)
17             readData <= 0;
18         else if (readEnable)
19             readData <= {28'b0, display_value};
20         else
21             readData <= 0;
22     end
23
24     sevenseg_decode ssd( //seven segment decoder initialized
25         .bin(display_value),
26         .seg(seg)
27 );
28
29     //we are using only the first segment
30     assign an = 4'b1110;
31
32 endmodule
```

### Seven segment decoder:

```
1 `timescale 1ns/1ps
2 module sevenseg_decode(
3     input wire [3:0] bin,
4     output reg [6:0] seg
5 );
6
7     always @(*) begin
8         case (bin)
9             4'h0: seg = 7'b1000000;
10            4'h1: seg = 7'b1111001;
11            4'h2: seg = 7'b0100100;
12            4'h3: seg = 7'b0110000;
13            4'h4: seg = 7'b0011001;
14            4'h5: seg = 7'b0010010;
15            4'h6: seg = 7'b0000010;
16            4'h7: seg = 7'b1111100;
17            4'h8: seg = 7'b0000000;
18            4'h9: seg = 7'b0010000;
19            4'hA: seg = 7'b0001000;
20            4'hB: seg = 7'b0000011;
21            4'hC: seg = 7'b1000110;
22            4'hD: seg = 7'b0100001;
23            4'hE: seg = 7'b0000110;
24            4'hF: seg = 7'b0001110;
25            default: seg = 7'b1111111;
26        endcase
27    end
28 endmodule
```



## TOP MODULE:

```
0 module top(
1   input clk,
2   input rst_btn,
3   input [15:0] sw,
4   output [6:0] seg,
5   output [3:0] an
6 );
7
8   wire rst; //using wires (internal)
9   wire tick;
10  wire [3:0] start_value;
11  wire start_valid;
12  wire [3:0] display_value;
13  wire busy;
14  wire [31:0] dummy_read;
15
16  debouncer db( //DEBOUNCER INITIALIZED
17    .clk(clk),
18    .pbin(rst_btn),
19    .pbout(rst)
20 );
21
22  priority_encoder pe(
23    .sw(sw),
24    .value(start_value),
25    .valid(start_valid)
26 );
27
28  clock_divider cd( //1SEC CLOCK DIVIDER
29    .clk(clk),
30    .rst(rst),
31    .tick(tick)
32
33  fsm_counter fsm(
34    .clk(clk),
35    .rst(rst),
36    .tick(tick),
37    .start_value(start_value),
38    .start_valid(start_valid),
39    .display_value(display_value),
40    .busy(busy)
41 );
42
43  leds led_interface(
44    .clk(clk),
45    .rst(rst),
46    .btns(16'b0),
47    .writeData(32'b0),
48    .writeEnable(1'b0),
49    .readEnable(1'b1),
50    .memAddress(30'b0),
51    .display_value(display_value),
52    .readData(dummy_read),
53    .seg(seg),
54    .an(an)
55 );
56
57 endmodule
```



Develop a testbench to:

- Apply different switch input values
- Verify FSM transitions
- Confirm correct countdown behavior
- Verify reset functionality

Test bench :

```
'timescale 1ns/1ps

module tb_top;
    reg clk;
    reg rst;
    reg tick;
    reg [3:0] start_value;
    reg start_valid; // This line is highlighted in yellow
    wire [3:0] display_value;
    wire busy;

//we use the FSM counter to instantiate
fsm_counter uut (
    .clk(clk),
    .rst(rst),
    .tick(tick),
    .start_value(start_value),
    .start_valid(start_valid),
    .display_value(display_value),
    .busy(busy)
);

initial begin //this is where we generate the clock
    clk = 0;
    forever #5 clk = ~clk; //keeping a 10ns period
end

initial begin //generation of 1 sec ticks
initial begin //this is where we generate the clock
    clk = 0;
    forever #5 clk = ~clk; //keeping a 10ns period
end

initial begin //generation of 1 sec ticks
    tick = 0;
    forever begin
        #50 tick = 1; // 1 sec
        #10 tick = 0;
    end
end

initial begin
    //Initial IDLE STATE
    rst = 1; //Reset is pressed to ensure we are in our idle state
    start_valid = 0;
    start_value = 0;
    #20;
    rst = 0;
    //IF SWITCH 8 IS PRESSED !
    #20;
    start_value = 4'd8;
    start_valid = 1; //need to tell if we have pressed the switch or not
    #10;
    start_valid = 0;
    #1000; //full countdown time
    start_value = 4'd5;
```

```

#1000; // initial countdowntime time
start_value = 4'd5;
start_valid = 1;
#10;
start_valid = 0;
#100;
rst = 1; //during the cntdwn reset
#20;
rst = 0;
#200;
$stop;
end

//show values
always @(posedge tick) begin
    if (busy)
        $display("Time %t : Display = %d", $time, display_value);
end
endmodule

```

Case 1 : Switch 8 is pressed



Case 2 : switch 10 is pressed:





### Example Pin Constraint (Switch)

#### Map Switch Input to FPGA Pin

```
set_property PACKAGE_PIN V17 [get_ports {switches[0]}]
set_property IOSTANDARD LVCMOS33 [get_ports {switches[0]}]
```

### Explanation:

- PACKAGE\_PIN V17 specifies the physical FPGA pin
- IOSTANDARD LVCMOS33 defines voltage level (3.3V CMOS)

### Example Pin Constraint (LED)

#### Map LED Output to FPGA Pin

```
set_property PACKAGE_PIN U16 [get_ports {leds[0]}]
set_property IOSTANDARD LVCMOS33 [get_ports {leds[0]}]
```



## Clock Constraint

### Define System Clock (100 MHz)

```
create_clock -period 10.000 -name sys_clk [get_ports clk]
```

**Explanation:** The clock period of 10 ns corresponds to 100 MHz. All logic must complete within this time.

## Understanding Timing Slack

$$\text{Slack} = \text{Required Time} - \text{Actual Path Delay}$$

| Slack Value | Meaning          | Status  |
|-------------|------------------|---------|
| Positive    | Timing met       | PASS    |
| Zero        | Critical timing  | WARNING |
| Negative    | Timing violation | FAIL    |

## Real Timing Report Examples

Example 1 – Good Design (PASS)

```
Slack (MET) : 3.215ns  
Required Time : 10.000ns  
Path Delay : 6.785ns
```

**Interpretation:** Design runs safely at 100 MHz.

Example 2 – Marginal Design (CRITICAL)

```
Slack (MET) : 0.042ns  
Required Time : 10.000ns  
Path Delay : 9.958ns
```

**Interpretation:** Design barely meets timing and may fail under temperature/voltage variation.

Example 3 – Failing Design (VIOLATION)

```
Slack (VIOLATED) : -1.372ns  
Required Time : 10.000ns  
Path Delay : 11.372ns
```

**Interpretation:** Design cannot operate at 100 MHz — must be optimized or clock slowed.



## Procedure

- (a) Add an XDC file to your project
- (b) Assign pin constraints to all switches, LEDs, reset, and clock
- (c) Add a clock constraint
- (d) Run synthesis
- (e) Report your Worst Negative Slack (WNS)

## Deliverables

- XDC file
- Screenshot of Vivado Timing Summary
- WNS value and interpretation (PASS / MARGINAL / FAIL)

```
set_property IOSTANDARD LVCMOS33 [get_ports {an[3]}]
set_property IOSTANDARD LVCMOS33 [get_ports {an[2]}]
set_property IOSTANDARD LVCMOS33 [get_ports {an[1]}]
set_property IOSTANDARD LVCMOS33 [get_ports {an[0]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[15]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[14]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[13]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[12]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[11]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[10]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[9]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[8]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[7]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[6]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[5]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[4]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[3]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[2]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[1]}]
set_property IOSTANDARD LVCMOS33 [get_ports {led[0]}]
set_property IOSTANDARD LVCMOS33 [get_ports {seg[6]}]
set_property IOSTANDARD LVCMOS33 [get_ports {seg[5]}]
set_property IOSTANDARD LVCMOS33 [get_ports {seg[4]}]
set_property IOSTANDARD LVCMOS33 [get_ports {seg[3]}]
set_property IOSTANDARD LVCMOS33 [get_ports {seg[2]}]
set_property IOSTANDARD LVCMOS33 [get_ports {seg[1]}]
set_property IOSTANDARD LVCMOS33 [get_ports {seg[0]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[15]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[14]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[13]}]
```



```
set_property IOSTANDARD LVCMOS33 [get_ports {sw[12]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[11]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[10]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[9]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[8]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[7]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[6]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[5]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[4]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[3]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[2]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[1]}]
set_property IOSTANDARD LVCMOS33 [get_ports {sw[0]}]
set_property IOSTANDARD LVCMOS33 [get_ports clk]
set_property IOSTANDARD LVCMOS33 [get_ports rst_btn]
set_property PACKAGE_PIN U18 [get_ports rst_btn]
set_property PACKAGE_PIN W5 [get_ports clk]
set_property PACKAGE_PIN V17 [get_ports {sw[0]}]
set_property PACKAGE_PIN V16 [get_ports {sw[1]}]
set_property PACKAGE_PIN W16 [get_ports {sw[2]}]
set_property PACKAGE_PIN W17 [get_ports {sw[3]}]
set_property PACKAGE_PIN W15 [get_ports {sw[4]}]
set_property PACKAGE_PIN V15 [get_ports {sw[5]}]
set_property PACKAGE_PIN W14 [get_ports {sw[6]}]
set_property PACKAGE_PIN W13 [get_ports {sw[7]}]
set_property PACKAGE_PIN V2 [get_ports {sw[8]}]
set_property PACKAGE_PIN T3 [get_ports {sw[9]}]
set_property PACKAGE_PIN T2 [get_ports {sw[10]}]
set_property PACKAGE_PIN R3 [get_ports {sw[11]}]
set_property PACKAGE_PIN W2 [get_ports {sw[12]}]
set_property PACKAGE_PIN U1 [get_ports {sw[13]}]
set_property PACKAGE_PIN T1 [get_ports {sw[14]}]
set_property PACKAGE_PIN R2 [get_ports {sw[15]}]
set_property PACKAGE_PIN W7 [get_ports {seg[0]}]
set_property PACKAGE_PIN W6 [get_ports {seg[1]}]
set_property PACKAGE_PIN U8 [get_ports {seg[2]}]
set_property PACKAGE_PIN V8 [get_ports {seg[3]}]
set_property PACKAGE_PIN U5 [get_ports {seg[4]}]
set_property PACKAGE_PIN V5 [get_ports {seg[5]}]
set_property PACKAGE_PIN U7 [get_ports {seg[6]}]
set_property PACKAGE_PIN U2 [get_ports {an[0]}]
set_property PACKAGE_PIN U4 [get_ports {an[1]}]
set_property PACKAGE_PIN V4 [get_ports {an[2]}]
set_property PACKAGE_PIN W4 [get_ports {an[3]}]
set_property PACKAGE_PIN U16 [get_ports {led[0]}]
set_property PACKAGE_PIN E19 [get_ports {led[1]}]
set_property PACKAGE_PIN U19 [get_ports {led[2]}]
set_property PACKAGE_PIN V19 [get_ports {led[3]}]
set_property PACKAGE_PIN W18 [get_ports {led[4]}]
set_property PACKAGE_PIN U15 [get_ports {led[5]}]
```



```
set_property PACKAGE_PIN U14 [get_ports {led[6]}]
set_property PACKAGE_PIN V14 [get_ports {led[7]}]
set_property PACKAGE_PIN V13 [get_ports {led[8]}]
set_property PACKAGE_PIN V3 [get_ports {led[9]}]
set_property PACKAGE_PIN W3 [get_ports {led[10]}]
set_property PACKAGE_PIN U3 [get_ports {led[11]}]
set_property PACKAGE_PIN P3 [get_ports {led[12]}]
set_property PACKAGE_PIN N3 [get_ports {led[13]}]
set_property PACKAGE_PIN P1 [get_ports {led[14]}]
set_property PACKAGE_PIN L1 [get_ports {led[15]}]
```

### Worst negative slack:

#### Design Timing Summary

| Setup                                       | Hold                                    | Pulse Width                                              |
|---------------------------------------------|-----------------------------------------|----------------------------------------------------------|
| Worst Negative Slack (WNS): <b>6.146 ns</b> | Worst Hold Slack (WHS): <b>0.250 ns</b> | Worst Pulse Width Slack (WPWS): <b>4.500 ns</b>          |
| Total Negative Slack (TNS): <b>0.000 ns</b> | Total Hold Slack (THS): <b>0.000 ns</b> | Total Pulse Width Negative Slack (TPWS): <b>0.000 ns</b> |
| Number of Failing Endpoints: <b>0</b>       | Number of Failing Endpoints: <b>0</b>   | Number of Failing Endpoints: <b>0</b>                    |
| Total Number of Endpoints: <b>117</b>       | Total Number of Endpoints: <b>117</b>   | Total Number of Endpoints: <b>60</b>                     |

All user specified timing constraints are met.

The Timing Summary report shows:

- Worst Negative Slack (WNS) = **6.146 ns**
- Total Negative Slack (TNS) = **0.000 ns**
- Number of Failing Endpoints = **0**

Since the WNS is positive and TNS is zero, all timing constraints are satisfied. This indicates that the design meets the required 100 MHz timing specification and operates without setup or hold violations.



## Task 4: FPGA Synthesis and On-Board Verification

### Objective

To synthesize the FSM design on an FPGA and verify functionality using physical switches, LEDs, and reset button.

### Procedure

- (a) Integrate the FSM module into the top-level FPGA design.
- (b) Assign FPGA pins for:
  - Switch inputs
  - LED outputs
  - Reset button
- (c) Synthesize the design and analyze:
  - RTL schematic
  - State transitions
  - Counter implementation
- (d) Program the FPGA and perform on-board testing:
  - Apply zero and non-zero switch values
  - Observe LED countdown behavior
  - Press reset during countdown to verify immediate reset
- (e) Demonstrate correct operation to the lab instructor or RA.

## Deliverables

- RTL schematic screenshots
- Successful FPGA demonstration

## Expected Learning Outcomes

Upon successful completion of this lab, students will be able to:

- Design FSMs for hardware control applications
- Interface user inputs with synchronous digital systems
- Implement counters controlled by FSM logic
- Debug and verify FPGA-based designs using real hardware





