

**ECE385**

Spring 2020

Experiment #8

**SOC with USB and VGA Interface in  
SystemVerilog**

Yuantao Lu, Yizhen Lu

Section ABD

TA: Nicholas Cebry, Wenjie Pan

## **Introduction**

In this lab, we implemented SOC with USB and VGA interface. We made use of the NIOS-II microprocessor we made in lab7 and added several PIO modules. We then implemented the interface between NIOS-II and EZ-OTG microprocessor by making a tristate buffer like in lab6. After that, we took a look at the software driver for USB and implemented the I/O interface between the two microprocessors and how to take control of the usb from NIOS-II microprocessor by implementing `usb_read`/`usb_write` functions. Finally, we connected the vga modules and modified `ball.sv` to make the ball bounce on the screen. As a result, we would be able to control the ball moving on the VGA screen in the four directions by pressing the keys on the USB keyboard that is connected via the FPGA board.

## **Written Description of Lab 8 System**

### *Written Description of the entire Lab 8 System*

Lab8 is mainly composed of 4 parts: the on chip NIOS-II and on chip EZ-OTG microprocessors, the USB keyboard that is connected to EZ-OTG(on chip CY7C67200), the VGA that is connected to the NIOS-II. What we mainly did is just create interfaces among those four components. The NIOS-II microprocessor is implemented through the platform designer that is inherited from lab7. The EZ-OTG interface is implemented in software and defined to be on the chip CY7C67200. The driver for the USB is in provided code and the VGA is controlled via modules on the top level of lab8. For interaction between NIOS-II and chip CY7C67200, we implemented the tristate buffer in the module `hpi_io_intf.sv`, which controls the data flow between the NIOS-II and chip CY7C67200. Data is written/read to/from the chip CY7C67200 through this module. The software I/O functions in `io_handler.c` controls the signals in `hpi_io_intf.sv` module by using the PIO modules' port address in NIO-II just like the way in lab7. To read data from the chip, we need to set the address signal(the register we want to read from) in the function and set several control bits like `otg_cs`. Then the data will be sent back at `otg_hpi_data` through the tri-state buffer

we implemented. To write the data to the chip, we need to set both the otg\_address(the register on chip to write to) and otg\_data and several control bits. The data also flowed through the tristate buffer. For VGA, the driver is directly implemented in the top level lab8.sv. There are four modules that are related to it: the vga\_clk, VGA\_controller, ball, color\_mapper. We need the vga\_clk that is different from the main clock because we wanted to generate a 25Mhz clock from PLL to make the VGA component work. The VGA\_controller is used to locate the X/Y coordinate we would draw concurrently. The ball will actually control the ball moving on the screen. The keycode signal is added to this module to let the ball respond to the key pressed on the keyboard. Finally, the color\_mapper maps the background screen to a color and the ball to another color that is controlled by signal VGA\_R, VGA\_G, VGA\_B.

#### *Written Description of the USB protocol*

IO\_read: this function serves as the connection between NIOS-II microprocessor and chip CY7C67200. It mainly plays with PIO ports on the NIOS-II that take control of the hpi\_io\_intf.sv module. By setting the address, the cs, the r signals that each correspond to a PIO module on NIOS-II, the driver on the chip will send back the data corresponding to the address back to NIOS-II. The data exchange is done through hpi\_io\_intf.sv module that serves as the tristate buffer.

Steps in c code:

- Set the \*otg\_hpi\_address we wanted to read from;
- Set \*otg\_hpi\_cs control signal to 0(active);
- Set \*otg\_hpi\_r control signal to 0(active);
- Get the data from chip by reading from \*otg\_hpi\_data;
- Set \*otg\_hpi\_r control signal to 1(inactive);
- Set \*otg\_hpi\_cs control signal to 1(inactive);

Note the order of setting those signals is following the hpi communication in AN6010 data sheet.

IO\_write: this function serves as the connection between NIOS-II microprocessor and chip CY7C67200. It mainly plays with PIO ports on the NIOS-II that take

control of the hpi\_io\_intf.sv module. By setting the address we want to write to, the data we want to write, the cs control bit and the w control signal on the PIO ports on NIOS-II, we are able to write data to a certain register on the chip. The data flow is done through hpi\_io\_intf.sv module that serves as the tristate buffer.

Steps in c code:

Set the \*otg\_hpi\_address we wanted to write to;  
Set the \*oth\_hpi\_data we wanted to write;  
Set \*otg\_hpi\_cs control signal to 0(active);  
Set \*otg\_hpi\_w control signal to 0(active);  
Set \*otg\_hpi\_w control signal to 1(inactive);  
Set \*otg\_hpi\_cs control signal to 1(inactive);

Note the order of setting those signals is following the hpi communication in AN6010 data sheet.

USB\_write:

This function writes data to the internal registers of the CY7C67200 USB controller. It first takes the Address of the register as the data and HPI\_ADDR as the address and calls IO\_write, i.e. writing Address to HPI\_ADDR register. It then takes Data we wanted to write to register as the data and HPI\_DATA as the address. Note HPI\_ADDR and HPI\_DATA are registers on the chip. The reason we wanted to write both data and address into registers on the chip is because we do not have direct access to the address on the chip but could only modify registers. Having the above two registers set, EZ-OTG will then store the data in the DATA register into the address in the ADDR register.

USB\_read:

This function reads data from the internal registers of the CY7C67200 USB controller. It first takes the Address of the register as the data and HPI\_ADDR as the address and calls IO\_write, i.e. writing Address to HPI\_ADDR register. EZ-OTG will then fetch the data in the address specified by the HPI\_ADDR

register and load it into the HPI\_DATA register. The next step is to read the data from the HPI\_DATA register by calling IO\_read.

## Block Diagram



*Figure 1: Lab8 top level view*

## Module Descriptions

## Module: *lab8.sv*

Input: *CLOCK\_50*, [3:0] *KEY*, [15:0] *OTG\_DATA*, *OTG\_INT*, [31:0] *DRAM\_DQ*.  
Output: [6:0] *HEX0*, [6:0] *HEX1*, [7:0] *VGA\_R*, [7:0] *VGA\_G*, [7:0] *VGA\_B*,  
*VGA\_CLK*, *VGA\_SYNC\_N*, *VGA\_BLANK\_N*, *VGA\_VS*, *VGA\_HS*, [15:0]  
*OTG\_DATA*, [1:0] *OTG\_ADDR*, *OTG\_CS\_N*, *OTG\_RD\_N*, *OTG\_WR\_N*,  
*OTG\_RST\_N*, [12:0] *DRAM\_ADDR*, [31:0] *DRAM\_DQ*, [1:0] *DRAM\_BA*, [3:0]  
*DRAM\_DQM*, *DRAM\_RAS\_N*, *DRAM\_CAS\_N*, *DRAM\_CKE*, *DRAM\_WE\_N*,  
*DRAM\_CS\_N*, *DRAM\_CLK*.

Description: This is the top level of this lab. It consists of all the hardware necessary for this lab including the module for NIOS-II with the DRAM, the modules for VGA, the module for the interface between NIOS-II and the EZ-OTG.

Purpose: This module is the top level for this lab. It connects all necessary modules together and serves as the top level interface for the user that makes all the inputs and outputs clear.

Module: *hpi\_io\_intf.sv*

Input: *Clk, Reset, [1:0] from\_sw\_address, [15:0] from\_sw\_data\_out, from\_sw\_r, from\_sw\_w, from\_sw\_cs, rom\_sw\_reset, [15:0] OTG\_DATA.*

Output: *[15:0] from\_sw\_data\_in, [15:0] OTG\_DATA, [15:0] OTG\_ADDR, OTG\_RD\_N, OTG\_WR\_N, OTG\_CS\_N, OTG\_RST\_N.*

Description: This module takes the input from the NIOS-II that is generally the port for the PIOs and outputs signals that connect the EZ-OTG microprocessor.

Purpose: This module serves as the tristate buffer that is the I/O interface between NIOS-II microprocessor and EZ-OTG microprocessor. Hardware data exchange between those two processors is done in the way specified by this module. The software is able to control the PIO ports on the NIOS-II in order to make changes to those output signals that are connected to EZ-OTG.

Module: *ball.sv*

Input: *Clk, Reset, frame\_clk, [7:0] keycode, [9:0] Draw\_X, [9:0] Draw\_Y.*

Output: *is\_ball.*

Description: This module is the interface for the moving ball in the VGA. When VGA is rendering the screen pixel by pixel, it determines whether the current pixel defined by Draw\_X and Draw\_Y is a ball. It introduced the frame\_clk that determines whether the frame is changed.

Purpose: This module is the control of the moving ball on the screen. It tells VGA at each frame whether to render a pixel to be the ball or not. When the ball hits the edge of the screen or when a specific keycode is pressed, the ball needs to respond in a way that makes it look like the ball is bouncing and is controlled by the key.

Module: *VGA\_controller.sv*

Input: *Clk, Reset, VGA\_CLK.*

Output: *VGA\_HS, VGA\_VS, VGA\_BLANK\_N, VGA\_SYNC\_N, [9:0] Draw\_X, [9:0] Draw\_Y.*

Description: This module takes necessary input VGA signals and outputs the x, y index of rendering concurrently. The VGA\_HS is the horizontal pulse that becomes low when a row is rendered and VGA\_VS is the vertical pulse that becomes low when a specific row corresponds to an index y is rendered. We can use this signal to determine whether we need to change a frame.

Purpose: This module is mainly the driver for the VGA that updates the necessary VGA signals at each VGA\_CLK raising edge.

Module: *Color\_Mapper.sv*

Input: is\_ball, [9:0] Draw\_X, [9:0] Draw\_Y.

Output: [7:0] VGA\_R, [7:0] VGA\_G, [7:0] VGA\_B.

Description: Given whether the current pixel is ball, give current pixel a RGB color corresponding to the VGA.

Purpose: The module give different RGB colors to the ball and the background.

Module: vga\_clk.v:

Input: inclk0.

Output: c0.

Description: Generates a VGA clock via PLL that runs the VGA controller.

Purpose: We need to update the screen at around 60Hz, so we do not use the general clock but this VGA clock generated from PLL with 25MHz that has 800 \* 525 clock edges per frame. Since 800 \* 525 \* 60Hz is around 25MHz.

Module: HexDriver.sv:

Input: [3:0] In0

Output: [6:0] Out0

Description: This module converts 4-bit input value into 7-bit value in LED format.

Purpose: In order to show the pressed keycode on the FPGA board, we use this module to convert the keycode value into LED form.

## QSYS blocks:

| ... Connections                     | Name             | Description                    | Export                                                                                                                          | Clock                                                                                                                                                                                                                        | Base                              | End         | ... Tags |
|-------------------------------------|------------------|--------------------------------|---------------------------------------------------------------------------------------------------------------------------------|------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-----------------------------------|-------------|----------|
| <input checked="" type="checkbox"/> | clk_0            | Clock Source                   | clk<br>reset                                                                                                                    | <i>Double-click to clk_0</i>                                                                                                                                                                                                 |                                   |             |          |
| <input checked="" type="checkbox"/> | nios2_gen2_0     | Nios II Processor              | clk<br>reset<br>data_master<br>instruction_master<br>irq<br>debug_reset_request<br>debug_mem_slave<br>custom_instruction_master | <i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to [clk]</i> | IRQ 0                             | IRQ 31      |          |
| <input checked="" type="checkbox"/> | onchip_memory2_0 | On-Chip Memory (RAM or ROM...) | clk1<br>s1<br>reset1                                                                                                            | <i>Double-click to [clk1]</i><br><i>Double-click to [clk1]</i>                                                                                                                                                               | * 0x0                             | 0x0000_000f |          |
| <input checked="" type="checkbox"/> | sdram            | SDRAM Controller Intel FPGA... | clk<br>reset<br>s1<br>wire                                                                                                      | <i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to sdram_wire</i>                                                                                                                            | <b>sdram_p1...</b><br>* 1000_0000 | 0x17ff_ffff |          |
| <input checked="" type="checkbox"/> | sdram_pll        | ALTPLL Intel FPGA IP           | inclk_interface<br>inclk_interface_reset<br>pll_slave<br>c0<br>c1                                                               | <i>Double-click to [inclk_in...</i><br><i>Double-click to [inclk_in...</i><br><i>Double-click to sdram_pll_c0</i><br><i>Double-click to sdram_pll_c1</i>                                                                     | clk_0<br>* 0x90                   | 0x0000_009f |          |
| <input checked="" type="checkbox"/> | sysid_qsys_0     | System ID Peripheral Intel...  | clk<br>reset<br>control_slave                                                                                                   | <i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to control_slave</i>                                                                                                                         | clk_0<br>* 0xb0                   | 0x0000_00b7 |          |
| <input checked="" type="checkbox"/> | jtag_uart_0      | JTAG UART Intel FPGA IP        | clk<br>reset<br>avalon_jtag_slave<br>irq                                                                                        | <i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to [clk]</i>                                                                                                 | clk_0<br>* 0xa8                   | 0x0000_00af |          |
| <input checked="" type="checkbox"/> | keycode          | PIO (Parallel I/O) Intel F...  | clk                                                                                                                             | <i>Double-click to clk_0</i>                                                                                                                                                                                                 |                                   |             |          |
| <input checked="" type="checkbox"/> | otg_hpi_address  | PIO (Parallel I/O) Intel F...  | reset<br>s1<br>external_connection                                                                                              | <i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to otg_hpi_address</i>                                                                                                                       | * 0x30                            | 0x0000_003f |          |
| <input checked="" type="checkbox"/> | otg_hpi_data     | PIO (Parallel I/O) Intel F...  | clk<br>reset<br>s1<br>external_connection                                                                                       | <i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to otg_hpi_data</i>                                                                                                                          | clk_0<br>* 0x20                   | 0x0000_002f |          |
| <input checked="" type="checkbox"/> | otg_hpi_r        | PIO (Parallel I/O) Intel F...  | clk<br>reset<br>s1<br>external_connection                                                                                       | <i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to otg_hpi_r</i>                                                                                                                             | clk_0<br>* 0x40                   | 0x0000_004f |          |
| <input checked="" type="checkbox"/> | otg_hpi_w        | PIO (Parallel I/O) Intel F...  | clk<br>reset<br>s1<br>external_connection                                                                                       | <i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to otg_hpi_w</i>                                                                                                                             | clk_0<br>* 0x50                   | 0x0000_005f |          |
| <input checked="" type="checkbox"/> | otg_hpi_cs       | PIO (Parallel I/O) Intel F...  | clk<br>reset<br>s1<br>external_connection                                                                                       | <i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to otg_hpi_cs</i>                                                                                                                            | clk_0<br>* 0x60                   | 0x0000_006f |          |
| <input checked="" type="checkbox"/> | otg_hpi_reset    | PIO (Parallel I/O) Intel F...  | clk<br>reset<br>s1<br>external_connection                                                                                       | <i>Double-click to [clk]</i><br><i>Double-click to [clk]</i><br><i>Double-click to otg_hpi_reset</i>                                                                                                                         | clk_0<br>* 0x70                   | 0x0000_007f |          |
|                                     |                  |                                |                                                                                                                                 |                                                                                                                                                                                                                              |                                   |             |          |

*Figure 2: Qsys blocks in lab8.*

*Functions of each block in the System-level block diagram.*

*clk\_0:*

This block generates the main clock signal for all other modules.

*nois2\_gen2\_0:*

This block represents the NIOS-II economic version. It is the microprocessor we used in this lab, which performs the computing and processing job in the hardware.

*onchip\_memory2\_0:*

It is the memory corresponding to the NIOS-II for loading and storing data that requires high-level performance.

*sdram:*

This block represents the sdram module. It makes a connection with the processor, so the system can use the memory specified by the sdram.

*sdram\_pll:*

This block represents the PLL for the system. It controls a clock with a phase shift of 3ns prior to the main clock. In this way, the sdram will start processing before the clock edge, and the date would be faster processed after the main clock edge.

*sysid\_qsys\_0:*

This block represents the system id checker. The purpose of this block is to check the configuration of the hardware part and the software part to make sure the two matches.

*key\_code:*

This block represents the parallel I/O port for switch input for FPGA. The keycode signal is connected at the top level. Software could use the address of the port of this block to read the keycode from the keyboard.

*otg\_hpi\_address:*

This block represents the parallel I/O port for switch input for FPGA. This port serves as the interface on NIOS-II the EZ-OTG address. It is connected to the hpi\_io\_intf module at the top level.

*otg\_hpi\_data:*

This block represents the parallel I/O port for switch input for FPGA. This port serves as the interface on NIOS-II the EZ-OTG data. It is connected to the hpi\_io\_intf module at the top level.

*otg\_hpi\_r:*

This block represents the parallel I/O port for switch input for FPGA. This port serves as the interface on NIOS-II the EZ-OTG read. It is connected to the hpi\_io\_intf module at the top level.

*otg\_hpi\_w:*

This block represents the parallel I/O port for switch input for FPGA. This port serves as the interface on NIOS-II the EZ-OTG write. It is connected to the hpi\_io\_intf module at the top level.

*otg\_hpi\_reset:*

This block represents the parallel I/O port for switch input for FPGA. This port serves as the interface on NIOS-II the EZ-OTG reset. It is connected to the hpi\_io\_intf module at the top level.

*otg\_hpi\_cs:*

This block represents the parallel I/O port for switch input for FPGA. This port serves as the interface on NIOS-II the EZ-OTG chip-select. It is connected to the hpi\_io\_intf module at the top level.

## **Answer to Hidden Questions**

1. What are the advantages and/or disadvantages of using a USB interface over PS/2 interface to connect to the keyboard? List any two.

Advantages of USB:

1. USB is a bus while PS/2 is a port. So USB is able to connect many devices while PS/2 can only connect one.
2. USB with a driver like in this lab can support hot plug since it is polling while PS/2 can't since it uses interrupt.

Disadvantages of USB:

1. USB uses polling while PS/2 uses interrupt. Interrupt is more efficient than polling because it does not need to keep asking if a keystroke happened but interrupt a task and take a proficiency. So USB is slower.
  2. When a USB bus is being used by other devices, the keystroke could be delayed. For PS/2 it is impossible since interrupts from a port always take the high priority in the operating system.
2. Notice that *Ball\_Y\_Pos* is updated using *Ball\_Y\_Motion*. Will the new value of *Ball\_Y\_Motion* be used when *Ball\_Y\_Pos* is updated, or the old?

What is the difference between writing

"*Ball\_Y\_Pos\_in* = *Ball\_Y\_Pos* + *Ball\_Y\_Motion*;" and  
"*Ball\_Y\_Pos\_in* = *Ball\_Y\_Pos* + *Ball\_Y\_Motion\_in*;"?

How will this impact behavior of the ball during a bounce, and how might that interact with a response to a keypress?

The old *Ball\_Y\_Motion* was used to update the *Ball\_Y\_position*.

The old *Ball\_Y\_Motion* is used so the ball will respond to both the key-press and the boundary hitting with a one frame delay. As a result, when the ball hit the boundary, it would not stop and change immediately but get "into" the screen for one frame and start to bounce back.

If the new *Ball\_Y\_Motion\_in* was used to make the update, the ball will respond to the key press and the boundary hitting immediately. It means if we hold the key

contrary to the boundary while the ball is at the boundary, the frame will stay almost still.

### Answer to Post Lab Questions

1)

|                   |            |
|-------------------|------------|
| LUT               | 2660       |
| DSP               | 0          |
| Memory(BRAM)      | 55296 bits |
| Flip-Flop         | 2235       |
| Frequency         | 77.8MHz    |
| Static Power      | 105.28mW   |
| Dynamic Power     | 27.66mW    |
| I/O Thermal Power | 75.37mW    |
| Total Power       | 208.31mW   |

2)

- What is the difference between VGA\_CLK and Clk?

Clk is the clock for the system which has 50MHz while VGA\_CLK is used to update the screen controlled by VGA that has 25MHz. VGA\_CLK is used to run the VGA\_controller that updates each frame at about 60Hz. The FPGA clock is the Clk.

- In the file io\_handler.h, why is it that the otg\_hpi\_data is defined as an integer pointer while the otg\_hpi\_r is defined as a char pointer?

An int pointer is 32 bits while a char pointer is 8 bits. Our data is actually 32-bit wide so we need to use a 32-bit pointer. The otg\_hpi\_r is either 0 or 1

so we can assign as little memory as possible. Since the smallest possible pointer in c is a char pointer, we use a char pointer instead.

## **Conclusion**

In this lab, we implemented the interface among the NIOS-II, the EZ-OTG, the USB and the VGA. It seemed to be hard at first but after reading the instructions and the code, it became clear. We actually had experience with a hardware driver and implemented data exchange between different hardwares. We also learned about VGA and how it renders the screen. It is important that we learn the interface of the USB keyboard and how we can control the object on the VGA screen using the keyboard. We could further develop from this lab the way to implement simple games on the chips like those arcade games.