

# Hardware Synthesis Laboratory I

## Final Project Report

FPGA Image Displayer via VGA  
Using Verilog on Basys 3



Group 23:  
**Stewie Griffin**

|                        |            |
|------------------------|------------|
| Pon Kittinaraporn      | 6630237121 |
| Pakornkiat Opaprakasit | 6630180021 |
| Chayutphong Soisri     | 6630054621 |
| Thitiwut Ariyaprayoon  | 6631326521 |

# Contents

|          |                                     |          |
|----------|-------------------------------------|----------|
| <b>1</b> | <b>Overall Design Block Diagram</b> | <b>1</b> |
| <b>2</b> | <b>Design Decision</b>              | <b>2</b> |
| 2.1      | Module Architecture . . . . .       | 2        |
| 2.2      | Communication Protocols . . . . .   | 2        |
| 2.3      | Clock and Data Rate . . . . .       | 2        |
| 2.4      | Resource Utilization . . . . .      | 2        |
| <b>3</b> | <b>Implementation Detail</b>        | <b>3</b> |
| 3.1      | VGA Controller . . . . .            | 3        |
| 3.2      | Image Pixel Controller . . . . .    | 4        |
| 3.3      | Image Generator . . . . .           | 5        |
| 3.4      | Constraints . . . . .               | 6        |
| <b>4</b> | <b>Challenges Faced</b>             | <b>7</b> |

# Overall Design Block Diagram



**Figure 1:** Overall system block diagram

Figure 1 shows the overall system block diagram.

# Design Decision

## 2.1 Module Architecture

The system is divided into three modules: `image_generator`, `image_pixel_controller`, and `vga_controller`. This modular approach improves clarity, debugging, and scalability. Each module has a defined role—generation, coordination, and display—allowing isolated testing and easier future upgrades.

## 2.2 Communication Protocols

We used native signal interfacing instead of protocols like AXI, as all modules share the same clock and communicate directly. This reduces latency and resource overhead, which suits our tightly-coupled design.

## 2.3 Clock and Data Rate

A shared clock domain was chosen to avoid synchronization issues. The VGA controller operates at 25 MHz to match  $640 \times 480 @60\text{Hz}$  standards. Other modules operate within this timing without critical delays.

## 2.4 Resource Utilization

Using Single Port ROM for storing image data, but logic is efficiently distributed. The current design minimizes LUT and FF usage, focusing on simplicity.

# Implementation Detail

## 3.1 VGA Controller

```
1 module vga_controller(
2     input wire clk,
3     input wire reset,
4     output wire hsync,
5     output wire vsync,
6     output wire video_on,
7     output wire [9:0] curr_x,
8     output wire [9:0] curr_y
9 );
10
11    parameter HD = 640;
12    parameter HF = 16;
13    parameter HS = 96;
14    parameter HB = 48;
15    parameter HMAX = HD + HF + HS + HB - 1;
16    parameter VD = 480;
17    parameter VF = 10;
18    parameter VS = 2;
19    parameter VB = 33;
20    parameter VMAX = VD + VF + VS + VB -1;
21
22    reg [9:0] h_count = 0;
23    reg [9:0] v_count = 0;
24
25    always @(posedge clk or posedge reset) begin
26        if (reset) begin
27            h_count <= 0;
28            v_count <= 0;
29        end else begin
30            if (h_count == HMAX) begin
31                h_count <= 0;
32                if (v_count == VMAX) v_count <= 0;
33                else v_count <= v_count + 1;
34            end
35            else h_count <= h_count + 1;
36        end
37    end
38
39    assign hsync = (h_count >= (HD + HF) &&
40                  h_count < (HD + HF + HS)) ? 0 : 1;
41    assign vsync = (v_count >= (VD + VF) &&
42                  v_count < (VD + VF + VS)) ? 0 : 1;
43    assign video_on = (h_count < HD && v_count < VD);
44    assign curr_x = h_count;
45    assign curr_y = v_count;
46
47 endmodule
```

Figure 1: *vga\_controller.v*

**Module:** vga\_controller

**Description:** Generates timing signals for VGA display at 60 Hz using a 25 MHz clock.

### 3.2 Image Pixel Controller

```
1 module image_pixel_controller (
2     input sw,
3     input wire clk,
4     input wire reset,
5     input wire [9:0] curr_x,
6     input wire [9:0] curr_y,
7     input wire video_on,
8     output wire [7:0] pixel_rgb
9 );
10
11     wire [7:0] rom_data;
12     reg [17:0] rom_addr;
13
14     blk_mem_gen_0 (
15         .addra(rom_addr),
16         .clka(clk),
17         .douta(rom_data),
18         .rsta(reset)
19 );
20
21     always @(*) begin
22         if (video_on && curr_x < 640 && curr_y < 480) begin
23             // Scale 640x480 down to 320x240
24             rom_addr = (curr_y >> 1) * 320 + (curr_x >> 1);
25         end else begin
26             rom_addr = 0;
27         end
28         rom_addr = (sw) ? rom_addr + 76800: rom_addr;
29     end
30
31     assign pixel_rgb = rom_data;
32
33 endmodule
```

Figure 2: *image\_pixel\_controller.v*

**Module:** image\_pixel\_controller

**Description:** Scale image to 320 x 420 and switch image by jumping to another address

### 3.3 Image Generator

```
1 module image_generator (
2     input [1:0] sw,
3     input clk,
4     input reset,
5     output wire [11:0] pixel_rgb,
6     output wire hsync,
7     output wire vsync
8 );
9
10    wire clk_25MHz;
11
12    clk_25MHz (
13        .clk_in1(clk),
14        .reset(reset),
15        .clk_out1(clk_25MHz)
16    );
17
18    wire video_on;
19    wire [9:0] curr_x;
20    wire [9:0] curr_y;
21
22    vga_controller (
23        .clk(clk_25MHz),
24        .reset(reset),
25        .hsync(hsync),
26        .vsync(vsync),
27        .video_on(video_on),
28        .curr_x(curr_x),
29        .curr_y(curr_y)
30    );
31
32    wire [11:0] w_pixel_rgb;
33
34    image_pixel_controller (
35        .sw(sw[1]),
36        .clk(clk_25MHz),
37        .reset(reset),
38        .curr_x(curr_x),
39        .curr_y(curr_y),
40        .video_on(video_on),
41        .pixel_rgb(w_pixel_rgb)
42    );
43
44    assign pixel_rgb[11:9] = (sw[0]) ? 4097 - w_pixel_rgb[7:5] : w_pixel_rgb[7:5];
45    assign pixel_rgb[7:5] = (sw[0]) ? 4097 - w_pixel_rgb[4:2] : w_pixel_rgb[4:2];
46    assign pixel_rgb[3:2] = (sw[0]) ? 4097 - w_pixel_rgb[1:0] : w_pixel_rgb[1:0];
47
48 endmodule
```

Figure 3: *image\_generator.v*

**Module:** image\_generator

**Description:** This is the top module use to connect the wires and all the modules together. It contains Clock Wizrad module, vga\_controller module and image\_pixel\_controller module.

### 3.4 Constraints

```
1 set_property PACKAGE_PIN W5      [get_ports clk]
2 set_property IOSTANDARD LVCMOS33 [get_ports clk]
3     create_clock -add -name sys_clk_pin -period 10.00 -waveform {0 5} [get_ports
4         clk]
5
6 set_property PACKAGE_PIN V17      [get_ports {sw[0]}]
7 set_property IOSTANDARD LVCMOS33 [get_ports {sw[0]}]
8 set_property PACKAGE_PIN V16      [get_ports {sw[1]}]
9 set_property IOSTANDARD LVCMOS33 [get_ports {sw[1]}]
10
11 set_property PACKAGE_PIN G19     [get_ports {pixel_rgb[0]}]
12 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[0]}]
13 set_property PACKAGE_PIN H19     [get_ports {pixel_rgb[1]}]
14 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[1]}]
15 set_property PACKAGE_PIN J19     [get_ports {pixel_rgb[2]}]
16 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[2]}]
17 set_property PACKAGE_PIN N19     [get_ports {pixel_rgb[3]}]
18 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[3]}]
19 set_property PACKAGE_PIN J17     [get_ports {pixel_rgb[4]}]
20 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[4]}]
21 set_property PACKAGE_PIN H17     [get_ports {pixel_rgb[5]}]
22 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[5]}]
23 set_property PACKAGE_PIN G17     [get_ports {pixel_rgb[6]}]
24 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[6]}]
25 set_property PACKAGE_PIN D17     [get_ports {pixel_rgb[7]}]
26 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[7]}]
27 set_property PACKAGE_PIN N18     [get_ports {pixel_rgb[8]}]
28 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[8]}]
29 set_property PACKAGE_PIN L18     [get_ports {pixel_rgb[9]}]
30 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[9]}]
31 set_property PACKAGE_PIN K18     [get_ports {pixel_rgb[10]}]
32 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[10]}]
33 set_property PACKAGE_PIN J18     [get_ports {pixel_rgb[11]}]
34 set_property IOSTANDARD LVCMOS33 [get_ports {pixel_rgb[11]}]
35 set_property PACKAGE_PIN P19     [get_ports hsync]
36 set_property IOSTANDARD LVCMOS33 [get_ports hsync]
37 set_property PACKAGE_PIN R19     [get_ports vsync]
38 set_property IOSTANDARD LVCMOS33 [get_ports vsync]
39 set_property PACKAGE_PIN U18     [get_ports reset]
40 set_property IOSTANDARD LVCMOS33 [get_ports reset]
```

Figure 4: *image\_constr.xdc*

**Module:** *image\_constr.xdc* (not a module)

**Description:** Mapping the logical design (Verilog Code) to Physical Hardware Pins on the Basys 3 Board

# Challenges Faced

- Reading from and writing data to the SD card
- Limited learning resources
- Displaying images on the screen from the SD card
- Understanding communication protocols (e.g., SPI)
- Debugging with macOS machines
- Lack of proper documentation or example code
- Time-consuming trial and error process
- Lengthy synthesis and implementation process in Vivado

The main challenges of this project stemmed from our lack of experience, documentation, and code examples for working with VGA and SD card interfaces. Since we had no prior experience with reading from or writing to an SD card, working with it proved to be extremely challenging for us—as it likely is for many others as well.

Additionally, a significant portion of the project time was spent waiting for Vivado to complete tasks such as compilation, synthesis, or bitstream generation, all of which heavily depend on the performance of the machine used. Apart from software debugging, hardware debugging on macOS was also difficult. Since we used Docker to run Vivado on macOS, the built-in probe tool could not directly connect to the FPGA board via USB, adding another layer of complexity to the process.