

# THE IMAGINATION UNIVERSITY PROGRAMME

# **RVfpga Lab 7**7-Segment Displays



# 1. Introduction

This lab describes how the RVfpgaEL2 System was extended to work with 7-segment displays and shows how to modify the 7-segment display controller. The Nexys A7 Board has eight 7-segment displays. We first describe how they work (Section 2) and then analyse the high-level specification of the 8-digit 7-segment display controller included in the RVfpgaEL2 System and provide some fundamental exercises (Sections 3 and 4). Finally, we analyse the low-level implementation of this controller, and provide additional exercises where you will modify and experiment with the controller implementation (Sections 5 and 6).

# 2. 7-Segment Displays on the Nexys A7 Board

The Nexys A7 board contains two 4-digit common-anode 7-segment LED displays<sup>1</sup>, configured to behave as a single 8-digit 7-segment display (see Figure 1). Each of the eight digits is composed of seven segments arranged in a "figure 8" pattern (see Figure 2), with an LED for each segment. Each of these segments can be switched on or off, so any one of 128 patterns can be displayed on a digit by illuminating certain LED segments and leaving the others dark; specifically, among these 128 patterns, the decimal digits can be displayed as shown in Figure 2.



Figure 1. 8-digit 7-segment displays on the Nexys A7



Figure 2. Patterns corresponding to decimal digits

<sup>&</sup>lt;sup>1</sup> The information in this section is described in: https://reference.digilentinc.com/reference/programmable-logic/nexys-a7/reference-manual



The LED segments of a single digit are labelled *A-G*, as shown on the right of Figure 3. The anodes of the seven LEDs for a single digit are tied together into one "common anode" circuit node, but the LED cathodes remain separate (see Figure 3). The eight common anode signals, one for each digit (*ANO-ANT*), act as a "digit enable". The cathodes of the same segment on all eight digits are connected into seven signals, *CA-CG* (see Figure 3). (Note that an eighth signal exists for the decimal point, *DP*, but we will not use it in this lab.) For example, the cathode of segment *D* from the eight digits are grouped together into a single circuit node called *CD*. This signal connection scheme creates a multiplexed display, where the cathode signals are common to all digits, but they can only illuminate the segments of the digit whose corresponding anode signal is asserted. All these signals are driven low when active; thus, to illuminate a segment, for example, segment *D* on digit 2, both the anode *AN2* and the cathode *CD* must be driven low.



Figure 3. Connection of the 8-digit 7-segment Display on the Nexys A7

A scanning display controller circuit can be used to show an 8-digit number on the 8-digit 7-segment displays. This circuit drives the cathodes with the pattern of each digit in a repeating continuous succession at an update rate that is faster than the human eye can detect; at the same time the circuit drives the anodes one at a time. Thus, each digit is illuminated just one-eighth of the time, but, because the eye cannot perceive the darkening of a digit before it is illuminated again, the digit appears to be continuously illuminated.

For each of the 8 digits to appear bright and continuously illuminated, all eight digits should be driven once every 1 to 16 ms, and each digit would be illuminated for 1/8 of the refresh cycle (e.g., for a 16ms refresh cycle, each digit is illuminated for 2ms). As explained above, the controller must drive the cathodes of a digit low with the correct pattern while the corresponding anode signal is also driven low. However, since the Nexys A7 uses NPN transistors to drive enough current into the common anode point, the anode enables are inverted. Therefore, both the ANO...7 and the CA...G/DP signals are driven low when active.

To illustrate the process, assume that you want to show 71 on the two right-most digits. The controller circuit would drive ANO, CB, and CC low for the first 2ms, thus showing a 1 in the right-most digit. Then, for the next 2ms, the circuit would drive AN1, CA, CB, and CC low, thus showing a 7 in the next most significant digit. If the process is repeated indefinitely, the human eye will see number 71 in the two right-most digits.



# 3. High-Level Specification of the 8-Digit 7-Segment Display Controller

In this section, we first describe and analyse the high-level specification of the 8-digit 7-segment displays controller used in the RVfpgaEL2 System, and then we provide exercises for using it.

The 8-digit 7-segment display controller used in this course has been custom-designed for the RVfpgaEL2 System. It includes two registers, called <code>Enables\_Reg</code> and <code>Digits\_Reg</code>, that are mapped to addresses 0x80001038 and 0x8000103C respectively (note that these addresses are unused addresses within the address range reserved for the System Controller, which you can view at <a href="https://github.com/chipsalliance/VeeRwolf">https://github.com/chipsalliance/VeeRwolf</a>).

<u>TASK</u>: Locate the declaration of registers <code>Enables\_Reg</code> and <code>Digits\_Reg</code>, as well as the place where they are assigned a value. The 8-digit 7-segment displays is implemented in file:

[RVfpgaEL2NexysA7DDRPath]/src/VeeRwolf/Peripherals/SystemController/veerwolf\_syscon.sv

Enables\_Reg is an 8-bit register where each bit determines if the corresponding digit is ON (0) or OFF (1). Digits\_Reg is a 32-bit register where each 4-bit group represents the hexadecimal value to show in the corresponding digit. For example, to show 71 on the two right-most digits, the programmer would assign the following values to the registers:

- Enables Reg = 0xFC (two right-most digits enabled)

- Digits Reg = 0x00000071 (value = 71)

You can test the program provided at [RVfpgaEL2NexysA7DDRPath]/Labs/Lab07/71 7SegDispl C-Lang

# 4. Fundamental Exercises

**Exercise 1**. Write a RISC-V assembly program and/or a C program that shows the value of the switches on the four right-most digits of the 7-segment displays. Recall that you can run the same program either in hardware on RVfpgaEL2-NexysA7 or in simulation on RVfpgaEL2-ViDBo. For example, Figure 4 shows the program running on RVfpgaEL2-ViDBo.





Figure 4. Running Exercise 2 on RVfpgaEL2-ViDBo

**Exercise 2**. Write a RISC-V assembly program and/or a C program that shows the string "0-1-2-3-4-5-6-7-8" moving from the right to the left of the 8-digit 7-segment displays. That is, 0 should show up on the right-most digit first. Then it should move to the left and 1 should show up on the right-most digit, and so on. Recall that you can run the same program on either RVfpgaEL2-NexysA7 or RVfpgaEL2-ViDBo.

# 5. 8-Digit 7-Segment Display Controller: Low-Level Implementation

Up until this point, we have shown how to use the 8-digit 7-segment displays only. In this section, we describe their low-level implementation and then we provide exercises for modifying the 8-digit 7-segment display controller.

Similar to previous general-purpose I/O (GPIO) labs, we divide the analysis of the 8-digit 7-segment display controller into three phases:

- 1. Connection between the SoC and the I/O device on the board (left shadowed region in Figure 5);
- 2. Integration of the new controller, which is included inside the VeeRwolfX System Controller contained in the SoC (middle shadowed region in Figure 5);
- 3. Connection between the new controller and the VeeR EL2 Core (right shadowed region in Figure 5).





Figure 5. 8-digit 7-segment displays controller analysis in 3 phases

### 1. Connection of the LEDs/Switches to the SoC

The constraints file of the project ([RVfpgaEL2NexysA7DDRPath]/src/rvfpganexys.xdc) defines the connection between the input/output SoC signals and the board. Each I/O device on the Nexys A7 Board FPGA board is connected to a specific FPGA pin. The signal that connects the eight anodes (see Figure 3) is called AN[i] (with i ranging from 0-7), and the signals that connect the cathodes of similar segments on all 4 digits (see Figure 3) are called CA, CB, CC, CD, CE, CF and CG. Figure 6 shows the snippet of the constraints file where these connections are defined.

```
##7 segment display
                                                                                                       [get_ports { CA }]; #IO_L24N_T3_A00_D16_14 Sch=ca
[get_ports { CB }]; #IO_25_14 Sch=cb
[get_ports { CC }]; #IO_25_15 Sch=cc
[get_ports { CD }]; #IO_L17P_T2_A26_15 Sch=cd
[get_ports { CE }]; #IO_L13P_T2_MRCC_14 Sch=ce
[get_ports { CF }]; #IO_L19P_T3_A10_D26_14 Sch=cf
[get_ports { CF }]; #IO_L4P_T0_P04_14 Sch=cf
                                   PACKAGE_PIN T10
                                                                  IOSTANDARD LVCMOS33
set_property -dict
                                   PACKAGE_PIN R10
                                                                  IOSTANDARD LVCMOS33
set_property -dict
                                   PACKAGE_PIN K16
                                                                  IOSTANDARD LVCMOS33
                                   PACKAGE_PIN K13
set_property -dict
                                                                  IOSTANDARD LVCMOS33
                                { PACKAGE_PIN P15
                                                                  IOSTANDARD LVCMOS33
set_property -dict
                                { PACKAGE_PIN T11
{ PACKAGE_PIN L18
set_property -dict
                                                                  IOSTANDARD LVCMOS33
                                                                                                       [get_ports { CF }]; #IO_L19P_T3_A10_D26_14 Sch=cf
[get_ports { CG }]; #IO_L4P_T0_D04_14 Sch=cg
} [get_ports { DP }]; #IO_L19N_T3_A21_VREF_15 Sch=dp
[get_ports { AN[0] }]; #IO_L23P_T3_F0E_B_15 Sch=an[0]
[get_ports { AN[1] }]; #IO_L23N_T3_FWE_B_15 Sch=an[1]
[get_ports { AN[2] }]; #IO_L24P_T3_A01_D17_14 Sch=an[2]
[get_ports { AN[3] }]; #IO_L19P_T3_A22_15 Sch=an[3]
set_property -dict
                                                                  IOSTANDARD LVCMOS33
                                  { PACKAGE PIN H15
                                                                   IOSTANDARD LVCMOS33
#set property -dict
                                                                  IOSTANDARD LVCMOS33 }
set property -dict
                                   PACKAGE PIN J17
set_property -dict
                                   PACKAGE_PIN J18
                                                                  IOSTANDARD LVCMOS33
                                   PACKAGE_PIN T9
set_property -dict
                                                                  IOSTANDARD LVCMOS33
set_property -dict
                                   PACKAGE_PIN J14
                                                                  IOSTANDARD LVCMOS33
                                                                                                                                       }]; #10_L8N_T1_D12_14 Sch=an[4]
}]; #10_L8N_T1_D12_14 Sch=an[4]
}]; #10_L14P_T2_SRCC_14 Sch=an[5]
}]; #10_L23P_T3_35 Sch=an[6]
}]; #10_L23N_T3_A02_D18_14 Sch=an[7]
                                   PACKAGE_PIN P14
                                                                  IOSTANDARD LVCMOS33
set_property -dict
                                                                                                        [get_ports
set_property -dict
                                   PACKAGE_PIN T14
                                                                  IOSTANDARD LVCMOS33
                                                                                                        [get_ports
set_property -dict
                                   PACKAGE_PIN K2
                                                                  IOSTANDARD LVCMOS33
                                                                                                        [get_ports
set_property -dict
                                   PACKAGE PIN U13
                                                                  IOSTANDARD LVCMOS33
                                                                                                        [get_ports
                                                                                                                             AN[7]
```

Figure 6. Connection of the two 4-digit 7-segment displays inputs (file *rvfpganexys.xdc*)

In the top-module of our system (module rvfpganexys, implemented in file [RVfpgaEL2NexysA7DDRPath]/src/rvfpganexys.sv) you can find the 8-digit 7-segment displays input signals connected to the SoC and at the end of that module you can find their connection to the veerwolf core module (see Figure 7).



```
default nettype none
  lule rvfpganexys
‡(parameter bootr
                                               = "boot_main.mem")
              eter bootrom file
   (input wire clk,
input wire rstn,
    input wire rstn,
output wire [12:0] ddram_a,
output wire [2:0] ddram_ba,
output wire ddram_cas_n,
output wire ddram_cs_n,
output wire ddram_cs_n,
output wire [1:0] ddram_dm,
inout wire [1:0] ddram_dqs_n,
inout wire [1:0] ddram_dqs_n,
output wire [1:0] ddram_dqs_n,
output wire ddram_cdk_n,
output wire ddram_cdk_n,
                                    ddram_clk_p,
    output wire
    output wire
                                     ddram_clk_n,
                              ddram_erk_n,
ddram_cke,
ddram_odt,
o_flash_cs_n,
    output wire
    output wire
    output wire
                                o_flash_mosi,
i_flash_miso,
i_uart_rx,
    output wire
    input wire
input wire
    output wire
                                    o_uart_tx,
    inout wire [15:0] i_sw,
    output reg [7:0] AN,
   output reg
                                     o_accei_cs_n,
o_accel_mosi,
    output wire
     output wire
                                     i_accel_miso,
     input wire
     output wire
                                      accel_sclk
```

Figure 7. Connection of the two 4-digit 7-segment displays to the SoC (file: rvfpganexys.sv).

Finally, the two signals are inserted from the **veerwolf\_core** module into the System Controller module (**veerwolf\_syscon**) (see Figure 8), where the 8-digit 7-segment display controller is implemented.



```
veerwolf_syscon
  #(.clk_freq_hz (clk_freq_hz))
syscon
                     (clk),
  .i_rst
                     (wb_rst),
  .gpio_irq
                     (gpio_irq),
  .ptc_irq
                     (ptc_irq),
  .o_timer_irq
                     (timer_irq),
  .o_sw_irq3
                     (sw_irq3),
  .o_sw_irq4 (sw_irq4),
.i_ram_init_done (i_ram_init_done),
  .i_ram_init_error (i_ram_init_error),
  .o_nmi_vec
                    (nmi_vec),
  .o_nmi_int
                     (nmi_int),
                     (wb_m2s_sys_adr[5:0]),
  .i_wb_adr
                     (wb_m2s_sys_dat),
  .i_wb_dat
                     (wb_m2s_sys_sel),
  .i_wb_sel
  .i_wb_we
                     (wb_m2s_sys_we),
                     (wb_m2s_sys_cyc),
  .i_wb_cyc
                     (wb_m2s_sys_stb),
  .i_wb_stb
                     (wb_s2m_sys_dat),
   .o_wb_rdt
   o wh ack
                     (wh s2m svs_ack),
  .AN (AN),
.Digits_Bits (Digits_Bits));
```

Figure 8. Connection of the 8-digit 7-segment displays to the System Controller (file: veerwolf\_core.v).

**TASK:** Follow these signals (*CA-CG* and *AN*) from the constraints file to the System Controller module (where *CA-CG* are merged into array *Digits\_Bits*). You will need to inspect the following files:

[RVfpgaEL2NexysA7DDRPath]/src/rvfpganexys.xdc

[RVfpgaEL2NexysA7DDRPath]/src/rvfpganexys.sv

[RVfpgaEL2NexysA7DDRPath]/src/VeeRwolf/veerwolf\_core.v

[RVfpgaEL2NexysA7DDRPath]/src/VeeRwolf/Peripherals/SystemController/veerwolf syscon.sv

## 2. Integration of the 8-digit 7-segment display controller into the SoC

### In module veerwolf syscon

([RVfpgaEL2NexysA7DDRPath]/src/VeeRwolf/Peripherals/SystemController/veerwolf\_sysco n.sv) the 8-digit 7-segment display controller is instantiated and integrated in the SoC (see Figure 9).

Figure 9. 8-digit 7-segment displays controller instantiation (file: veerwolf\_syscon.sv).



The SevSegDisplays\_Controller module receives, in addition to the clock signal (i\_clk, renamed as clk) and the reset signal (i\_rst, renamed as rst\_n), two input signals (Enables\_Reg and Digits\_Reg), which are the two memory-mapped control registers already described. This module outputs two signals, AN and Digits\_Bits, which are connected to the 7-segment displays on the board. For the example showing 71 on the two right-most digits, the SevSegdisplays\_Controller would assign the following values to signals AN and Digits Bits:

- From 0 to 2 ms: Signal AN[0] is low to enable digit 0 (the right-most digit) to display. Signals Digits\_Bits[5] and Digits\_Bits[4] (that correspond to CB and CC) are also low to display "1" on digit 0 (the right-most digit). All other signals are high.
- From 2 to 4 ms: Signal AN[1] is low to enable digit 1 to display.

  Digits\_Bits[6], Digits\_Bits[5] and Digits\_Bits[4] (that correspond to CA, CB, and CC) are high to display "7" on digit 1. All other signals are high.
- From 4 to 16 ms: AN [2]...AN [7] are high in 2 ms intervals so that they do not display values. The segments are also high for the remaining digits, digits 2-7.

The SevSegDisplays\_Controller module is implemented in file [RVfpgaEL2NexysA7DDRPath]/src/VeeRwolf/Peripherals/SystemController/veerwolf\_syscon.sv. It contains the following subunits:

- Two multiplexers select the value to send to the AN and Digits\_Bits signals every 2 ms. The multiplexer is implemented inside module SevSeqMux.
- For creating the 2 ms period, we use a counter module provided in files counter.sv and delta\_counter.sv, both included in folder [RVfpgaEL2NexysA7DDRPath]/src/OtherSources/pulp-platform.org\_\_common\_cells\_1.20.0/src. The counter is configured to count from 0 to 2<sup>17</sup>, and the 3 most significant bits are used as the select signals for the two multiplexers described above.
- A decoder is implemented in module **SevenSegDecoder**, which outputs the segment values for a given 4-bit hexadecimal value.

<u>TASKS</u>: Analyse the <u>SevSegDisplays\_Controller</u> module in detail. The simulation performed in the next section can help you on this task. You can also extend the simulation with new signals if necessary.

# 3. Connection between the 8-digit 7-segment displays controller and the VeeR EL2 Core

As described in Lab 6, the device controllers are connected to the VeeR EL2 Core using a multiplexer (see Figure 5). Remember that the 7:1 multiplexer (Figure 10) is instantiated in file

[RVfpgaEL2NexysA7DDRPath]/src/VeeRwolf/Interconnect/WishboneInterconnect/wb\_intercon.v. Then, the wb\_intercon module is instantiated in file
[RVfpgaEL2NexysA7DDRPath]/src/VeeRwolf/Interconnect/WishboneInterconnect/wb\_intercon.vh. This latter file is included in the veerwolf\_core module located here:
[RVfpgaEL2NexysA7DDRPath]/src/VeeRwolf/veerwolf\_core.v.

The multiplexer selects which peripheral to read or write, connecting the CPU (wb\_io\_\* signals) with the Wishbone Bus of one peripheral, depending on the address. For example, if



the address generated by the CPU is in the range 0x80001000-0x8000103F, the System Controller is selected, and thus signals wb\_io\_\* will be connected with signals wb\_sys\_\*.

Figure 10. 7:1 multiplexer that selects the peripheral connected with the CPU (file: wb intercon.v).

<u>TASK</u>: Inspect module <u>veerwolf\_syscon</u> in order to understand how addresses are mapped in the System Controller. Focus on registers <code>Enables\_Reg</code> and <code>Digits\_Reg</code> (as we mentioned before, the addresses assigned to these two registers are 0x80001038 and 0x8000103C respectively).

Figure 11. Connection between the 8-digit 7-segment displays and the core (file *veerwolf\_syscon.sv*).

# 6. Advanced Exercises

**Exercise 3.** Modify the controller described in this lab so that the 8-digit 7-segment displays can show any combination of ON/OFF LEDs.

• You do not need an enable register now. Instead, you need eight 7-bit registers.



Call them: Segments\_Digit0 - Segments\_Digit7, one for each of the eight 7-segment displays. In each of these registers, each bit indicates if the corresponding segment is ON (0) or OFF (1). For example, if all the bits of the first register (Segments\_Digit0) are 0, all segments in the right-most digit will be ON, whereas if all the bits of the first register are 1, all segments of the right-most digit will be OFF. Use the segment numbering shown in Figure 12 (as shown for Segments\_Digit0, but it will be the same numbering for all digits).



Figure 12. Segment numbering

- You can map these two new registers to the same addresses that we used before (first remove the two previous registers <code>Enables\_Reg</code> and <code>Digits\_Reg</code>):
  - o Segments\_Digit0 = Address 0x80001038
    o Segments\_Digit1 = Address 0x80001039
  - 0 ...
  - o Segments Digit7 = Address 0x8000103F
- Note that you do not need the 4-7 decoder anymore (module SevenSegDecoder), as the information provided by the program is already decoded.
- The outputs of the controller are the same as before:
  - The 8-bit AN output from the controller connects with ANO ... AN7.
  - The 7-bit  $\texttt{Digits\_Bits}$  output from the controller connects with CA ... CG (DP is left unconnected on the board).

**Exercise 4.** Use the new controller for printing the following on the 8-digit 7-segment displays: "I SAY HI". As usual, implement both RISC-V assembly and C versions of the program.

Recall that you can run the same program in either RVfpgaEL2-NexysA7 or RVfpgaEL2-ViDBo. For example, Figure 13 shows the program running on the modified RVfpgaEL2-ViDBo developed in Exercise 3.





Figure 13. RVfpgaEL2-ViDBo running Exercise 4

<u>IMPORTANT</u>: The virtual board simulates the same behaviour of this device, thus it receives the same inputs from the SoC: signals AN[7:0] and CA-CG. However, some things must be taken into account when using RVfpgaEL2-ViDBo:

- The physical board supports any combination of LEDs in the 7-segment displays. However, the virtual board only supports the hexadecimal digits plus the following characters:

Any other 7-segment display combination generated by the controller will show the digit as off.

- The symbols shown for number 0 and letter 0, number 1 and letter I, and number 5 and letter S, are the same.