



## UART - Based Counter with Clock Division

### Instructions:

1. Use **Behavioral** modeling for writing VHDL description
2. Write the testbench to perform RTL simulation.
3. Demonstrate the simulations to your TA
4. Perform Pin-Planning, run on Xenon board and demonstrate to your TA.
5. Submit the entire project files in .zip format in moodle.

### Part I - Clock Divider

#### Designing of the Clock Divider

In this experiment you will be doing clock divider. There is a 50 Mhz on board clock. You will divide it to generate 5 MHz clock and 2 Hz clock. Code for testbench will be written by you. You can run the simulation to check the waveform. You can map it in the board also so that you can assign an LED and can check if the LED is blinking.



Figure 1: Waveform

**NOTE:** The above image(1) is for representative purpose only. May not have any similarity with the waveform that will be generated after simulation.

#### Method of this experiment

Suppose we need to generate  $f = 5 \text{ MHz}$  from 50 MHz master clock. For this, we need a counter such that the clock out remains HIGH for 5 Input (master) Clock Cycles and LOW for next 5 Clock Cycles. In order to do this, we set-up a counter that starts from 1 and increments at every positive edge of the Input Clock (master) till the count reaches its maximum value which is 5 in this case.

$$count = 50\text{MHz}/(2 * f) = 5$$

After the count reaches 5, count will be initialized back to 1. And clock output will go LOW till count reaches maximum again.

**Note:** Here we are counting from 1 to maximum count. (Not from 0 to maximum count -1). The value of count may be rounded off to nearest integer.

# VHDL Description and RTL Simulation

- Generate a 50 MHz square wave input by creating your own Test bench, and 5 MHz clock and 2 Hz clock output from your design. Please verify the simulation using your own Test bench.

## Part II - Integrating UART

- Here, you will understand and use a program for communicating between Xenon and a computer using UART. This program will take inputs from the computer's keyboard that can be used in programs running on the FPGA, to perform appropriate operations.



Figure 2: Step 1: Finding serial converter in device manager



Figure 3: Step 2: In properties go to Advanced Options



Figure 4: Step 3: Click Load VCP Option



Figure 5: Step 4: Scan for Hardware Changes



Figure 6: Step 5: You will find Serial Port under PORTS (COM,LPT)

1. For recognizing keyboard inputs on the computer and transmitting to the FPGA, through the serial terminal, you need to use the TeraTerm software, RealTerm software (or any equivalent software). The guidelines for downloading and using the TeraTerm software can be found in the following link: [https://drive.google.com/file/d/10-M1yhnHNjE3\\_Zf5WIJAFQeIj7CPNsPx/view?usp=sharing](https://drive.google.com/file/d/10-M1yhnHNjE3_Zf5WIJAFQeIj7CPNsPx/view?usp=sharing) The download link for RealTerm software can be found in the following link: <https://realterm.i2cchip.com/>

**Note:** Set the baud rate to 9600.

**Before opening RealTerm or TeraTerm, make sure to close UART JTAG.**

- Design a system model with following characteristics:
  - System has a keyboard input, a 50 MHz Clock input and a reset input

- if keyboard input is "1" then implement a Johnson counter using 8 LEDs. The Johnson counter output will be displayed on LEDs such that for each clock (0.5 sec interval, using 2 Hz clock), the next state of the Johnson sequence appears on the LEDs with LED1 as the LSB and LED8 as the MSB. The pattern should loop continuously through all valid states of an 8-bit Johnson counter.

| Time (s)                 | LED8 | ...      | LED1 |
|--------------------------|------|----------|------|
| 0.0                      | 0    | 00000000 | 0    |
| 0.5                      | 0    | 00000001 | 1    |
| 1.0                      | 0    | 00000011 | 1    |
| ... fill in the rest ... |      |          |      |

- if keyboard input is "2", then implement an 8-bit ring counter. The ring counter should begin with only LED1 ON and others OFF. With each clock (0.5 sec interval, using 2 Hz clock), the logic '1' should shift to the next LED in a circular fashion (LED1 → LED2 → ... → LED8 → LED1 ...). This pattern should continue in a loop.

| Time (s)                 | LED8 | ...      | LED1 |
|--------------------------|------|----------|------|
| 0.0                      | 0    | 00000001 | 1    |
| 0.5                      | 0    | 00000010 | 0    |
| 1.0                      | 0    | 00000100 | 0    |
| ... fill in the rest ... |      |          |      |

## UART Code

The code below will be your Top-level entity. Copy paste or download [here](#) and edit as required.

```

1 library ieee;
2 use ieee.std_logic_1164.ALL;
3 use ieee.numeric_std.all;
4
5 entity UART is
6   generic (
7     g_CLKS_PER_BIT : integer := 521    -- For 5 MHz clock and 9600 baud
8   );
9   port (
10     i_Clk      : in std_logic;      -- 50 MHz input clock
11     i_RX_Serial : in std_logic;
12     o_LED      : out std_logic_vector(7 downto 0)  -- NEW: LED output
13   );
14 end UART;
15
16 architecture rtl of UART is
17
18   -- Internal 5 MHz clock from clock divider
19   signal clk_5MHz : std_logic;
20
21   -- UART RX Signals
22   type t_SM_Main is (s_Idle, s_RX_Start_Bit, s_RX_Data_Bits,
23                       s_RX_Stop_Bit, s_Cleanup);
24   signal r_SM_Main : t_SM_Main := s_Idle;
25   signal r_RX_Data_R : std_logic := '0';
26   signal r_RX_Data : std_logic := '0';
27
28   signal r_Clk_Count : integer range 0 to g_CLKS_PER_BIT-1 := 0;
29   signal r_Bit_Index : integer range 0 to 7 := 0;
30   signal r_RX_Byte : std_logic_vector(7 downto 0) := (others => '0');
31   signal r_RX_DV    : std_logic := '0';
32
33   -- LED Counter State
34   signal r_LED      : std_logic_vector(7 downto 0) := "00000001";
35   signal mode       : std_logic_vector(7 downto 0) := (others => '0');
36
37   -- Component declaration for Clock Divider
38   component ClockDiv is
39     port (

```

```

40      clk_in       : in  std_logic;
41      reset        : in  std_logic;
42      clk_out_5MHz : out std_logic;
43      clk_out_2Hz  : out std_logic
44  );
45 end component;
46
47 signal clk_2Hz : std_logic;
48
49 begin
50
51 -- Instantiate the clock divider (divide 50 MHz -> 5 MHz) according to your code
52 u_clk_div : ClockDiv
53   port map (
54     clk_in       => ----
55     reset        => ----
56     clk_out_5MHz => ----
57     clk_out_2Hz  => ----
58  );
59
60 -----
61 -- Double-register the incoming serial data
62 -----
63 p_SAMPLE : process (clk_5MHz)
64 begin
65   if rising_edge(clk_5MHz) then
66     r_RX_Data_R <= i_RX_Serial;
67     r_RX_Data   <= r_RX_Data_R;
68   end if;
69 end process p_SAMPLE;
70
71 -----
72 -- UART RX State Machine using 5 MHz clock
73 -----
74 p_UART_RX : process (clk_5MHz)
75 begin
76   if rising_edge(clk_5MHz) then
77     case r_SM_Main is
78       when s_Idle =>
79         r_RX_DV      <= '0';
80         r_Clk_Count <= 0;
81         r_Bit_Index <= 0;
82
83       if r_RX_Data = '0' then
84         r_SM_Main <= s_RX_Start_Bit;
85       else
86         r_SM_Main <= s_Idle;
87       end if;
88
89       when s_RX_Start_Bit =>
90         if r_Clk_Count = (g_CLKS_PER_BIT-1)/2 then
91           if r_RX_Data = '0' then
92             r_Clk_Count <= 0;
93             r_SM_Main  <= s_RX_Data_Bits;
94           else
95             r_SM_Main  <= s_Idle;
96           end if;
97         else
98           r_Clk_Count <= r_Clk_Count + 1;
99           r_SM_Main  <= s_RX_Start_Bit;
100          end if;
101
102       when s_RX_Data_Bits =>
103         if r_Clk_Count < g_CLKS_PER_BIT-1 then
104           r_Clk_Count <= r_Clk_Count + 1;
105           r_SM_Main  <= s_RX_Data_Bits;

```

```

106      else
107          r_Clk_Count           <= 0;
108          r_RX_Byte(r_Bit_Index) <= r_RX_Data;
109
110      if r_Bit_Index < 7 then
111          r_Bit_Index <= r_Bit_Index + 1;
112          r_SM_Main    <= s_RX_Data_Bits;
113      else
114          r_Bit_Index <= 0;
115          r_SM_Main    <= s_RX_Stop_Bit;
116      end if;
117  end if;
118
119  when s_RX_Stop_Bit =>
120      if r_Clk_Count < g_CLKS_PER_BIT-1 then
121          r_Clk_Count <= r_Clk_Count + 1;
122          r_SM_Main    <= s_RX_Stop_Bit;
123      else
124          r_RX_DV       <= '1';
125          r_Clk_Count <= 0;
126          r_SM_Main    <= s_Cleanup;
127      end if;
128
129  when s_Cleanup =>
130      r_SM_Main <= s_Idle;
131      r_RX_DV   <= '0';
132      mode      <= r_RX_Byte;
133
134
135  when others =>
136      r_SM_Main <= s_Idle;
137
138  end case;
139 end if;
140 end process p_UART_RX;
141
142 -----
143 -- LED Control: Johnson or Ring counter based on mode
144 -----
145 p_LED : process (clk_2Hz)
146 begin
147     if rising_edge(clk_2Hz) then
148         case mode is
149
150             when x"31" =>
151                 -- Write Logic for Johnson Counter
152
153             when x"32" =>
154                 -- Write Logic for Ring Counter
155
156             when others =>
157                 r_LED <= "00000001"; -- reset
158         end case;
159     end if;
160 end process p_LED;
161
162 -- Outputs
163 o_LED      <= r_led;
164
165
166 end rtl;

```

## On Xenon board

- For the generated system output, show it on LED1-LED8 on Xenon board.

- Do Pin-mapping for the clock output to LEDs and 50 MHz input clock, reset to SW8 and input switch as SW1.
- Keep SW8 ON for some time then make it OFF.
- Pin Plan for Uart Rx as shown in the figure below.

| Node Name      | Direction | Location | I/O Bank |
|----------------|-----------|----------|----------|
| in i_Clk       | Input     | PIN_26   | 2        |
| in i_RX_Serial | Input     | PIN_25   | 1B       |

Figure 7: Uart Rx Pin Planning

- Get the LED output verified by your respective TA.
- Plain Uart RX with Testbench as a sanity check (50Mhz Clock, 115200 baud rate) [UART\\_Rx](#), [Testbench](#)

| Clock Source Frequency | FPGA Pin no. |
|------------------------|--------------|
| 1 Hz CLK               | 55           |
| 50 MHz CLK             | 26           |
| Ext CLK                | 27           |
| 10 MHz CLK             | 29           |

Figure 8: Pin-mapping for on-board Clock Sources

| Switch | FPGA Pin no. | LED   | FPGA Pin no. |
|--------|--------------|-------|--------------|
| SW 8   | 47           | LED 8 | 60           |
| SW 7   | 46           | LED 7 | 59           |
| SW 6   | 45           | LED 6 | 58           |
| SW 5   | 44           | LED 5 | 57           |
| SW 4   | 43           | LED 4 | 56           |
| SW 3   | 41           | LED 3 | 54           |
| SW 2   | 39           | LED 2 | 52           |
| SW 1   | 38           | LED 1 | 50           |

Figure 9: Pin-mapping for on-board Switches and LED's