



**Oregon State  
University**

OREGON STATE UNIVERSITY  
ECE 271

**Final Design Project**

*Benjamin Geyer*

*Christien Hotchkiss*

*Phi Luu*

November 30<sup>th</sup>, 2018

# Contents

|          |                                           |           |
|----------|-------------------------------------------|-----------|
| <b>1</b> | <b>Introduction</b>                       | <b>2</b>  |
| 1.1      | Project Goals .....                       | 2         |
| <b>2</b> | <b>High-Level Description</b>             | <b>5</b>  |
| 2.1      | Top-Level Hardware Diagram .....          | 5         |
| 2.2      | Top-Level Hardware Diagram .....          | 7         |
| <b>3</b> | <b>Controllers</b>                        | <b>10</b> |
| 3.1      | PS/2.....                                 | 10        |
| 3.2      | RC5 IR.....                               | 10        |
| 3.3      | NES Controller.....                       | 10        |
| <b>4</b> | <b>HDL Modules</b>                        | <b>11</b> |
| 4.1      | PS/2 Modules .....                        | 11        |
| 4.1.1    | PS/2 Decoder .....                        | 11        |
| 4.2      | RC5 IR Modules .....                      | 14        |
| 4.2.1    | RC5 IR Decoder .....                      | 14        |
| 4.2.2    | RC5 IR Convert Pulse.....                 | 15        |
| 4.2.3    | RC5 IR Read Word .....                    | 16        |
| 4.3      | NES Modules .....                         | 17        |
| 4.3.1    | NES Decoder .....                         | 17        |
| 4.4      | General Modules Used in Each Design ..... | 18        |
| 4.4.1    | Clock Frequency Module .....              | 18        |
| 4.4.2    | Box Controller Module .....               | 19        |
| 4.4.3    | XY Counter Module .....                   | 21        |
| 4.4.4    | Drawer Module .....                       | 24        |
| <b>5</b> | <b>Putting It All Together</b>            | <b>26</b> |
| <b>6</b> | <b>Above and Beyond</b>                   | <b>28</b> |
| <b>7</b> | <b>Appendix</b>                           | <b>29</b> |
| 7.1      | SystemVerilog Source Code.....            | 29        |
| 7.1.1    | Top Level .....                           | 29        |
| 7.1.2    | PS/2 Top Level .....                      | 30        |
| 7.1.3    | PS/2 Controller .....                     | 32        |
| 7.1.4    | RC5 IR Top Level .....                    | 35        |
| 7.1.5    | RC5 IR Controller .....                   | 36        |
| 7.1.6    | NES Top Level .....                       | 42        |
| 7.1.7    | NES Controller .....                      | 44        |
| 7.1.8    | Box Controller .....                      | 47        |
| 7.1.9    | Half Clock .....                          | 49        |
| 7.1.10   | XY Counter .....                          | 50        |
| 7.1.11   | Drawer .....                              | 51        |

# 1 Introduction

The purpose of the final project is to fully design and simulate a VGA controller. A VGA, or video graphics array, can be used to display a wide range of resolutions where each pixel is controlled by 3 analog pins corresponding to Red, Green, and Blue. Using system verilog, an extremely powerful hardware description language, we created modules to decode data transmitted through SNES/NES, PS/2, and Infrared, and outputted the data to a VGA controller that displays an image. Our team tested our modules using both ModelSim and the DE10-Lite. We were able to simulate and robustly test our modules using ModelSim. ModelSim creates a simulation that exercises all possible states of the design to ensure that each unique input scenario is handled properly according to the system verilog code. Using this program is also a much more efficient way to debug issues in the design and is far more practical than testing each scenario using a board such as an FPGA. While not specifically required for this project, our team also decided to test our design using the DE10-Lite, which is a FPGA provided to us through Oregon State University. By connecting the DE10-Lite with a monitor using a VGA cable, we were able to determine that our design correctly outputs VGA images, which aligned with our results from simulation. The main goal of this project is to successfully handle different types of inputs that are wired to control the output as a VGA image.

## 1.1 Project Goals

- Effectively decode data transmitted through SNES/NES, PS/2, and Infrared
- Control a VGA image output for each type of input
- VGA image of a box should stop moving and change color when it hits the edge of the display



Figure 1.1.1: NES controller



Figure 1.1.2: PS/2 keyboard



Figure 1.1.3: PS/2 port



Figure 1.1.4: PS/2 port 6-pin mini-DIN connector



Figure 1.1.5: ECE 272 DE10-Lite FPGA courtesy of the EECS department at OSU



Figure 1.1.6: Timing specifications and pixel range of the VGA output

## 2 High-Level Description

### 2.1 Top-Level Hardware Diagram



Figure 2.1.1: Top-level hardware diagram for PS/2 keyboard

**Inputs:** The FPGA reads in data and clock signals from the PS/2 keyboard.

**Outputs:** The FPGA outputs VSync, HSync signals as well as Red, Green, and Blue values to display as a VGA image.

**Description:** The top-level hardware diagram above shows that the DE10-Lite takes in

a Data and Clock signal from the PS/2 Keyboard. Then, it manipulates that data to form five outputs required for a VGA image.



Figure 2.1.2: Top-level hardware diagram for RC5 IR

Inputs: The FPGA reads in a data signal from RC5 IR.

Outputs: The FPGA outputs VSync, HSync signals as well as Red, Green, and Blue values to display as a VGA image.

Description: The top-level hardware diagram above shows that the DE10-Lite takes in a Data signal in the form of RC5 IR. Then, the FPGA manipulates that data to form five outputs required for a VGA image.



Figure 2.1.3: Top-level hardware diagram for NES controller

Inputs: The FPGA reads Latch, Clock, and Data signals from the NES Controller.

Outputs: The FPGA outputs VSync, HSync signals as well as Red, Green, and Blue values to display as a VGA image.

Description: The top-level hardware diagram above shows that the DE10-Lite takes in a Data, Latch, and Clock signal from the NES Controller. Then, the FPGA manipulates that data to form five outputs required for a VGA image.

## 2.2 Top-Level Hardware Diagram



Figure 2.2.1: Top-level HDL diagram for PS/2 keyboard



Figure 2.2.2: Top-level diagram for PS/2 keyboard



Figure 2.2.3: Top-level HDL diagram for RC5 IR



Figure 2.2.4: Top-level diagram for RC5 IR



Figure 2.2.5: Top-level HDL diagram for NES controller



Figure 2.2.6: Top-level diagram for NES controller

## 3 Controllers

### 3.1 PS/2

The PS/2 port is a 6-pin mini-DIN connector used for connecting keyboards or mice to a computer system. It was first designed by IBM in 1987. A mini-DIN connector refers to a family of multi-pin electrical connectors used in a variety of environments. They are 9.5 millimeters in diameter and come in seven patterns ranging from three to nine pins. PS/2 ports have serial, synchronous, and bidirectional communication in which the attached device generates a clock signal while the host controls communication along the clock line. When the clock is LOW, communication from the PS/2 port is prevented. The assignments of each pin on the 6-pin connector are illustrated clearly in Figure 1.1.4, which can be found on page 4.

### 3.2 RC5 IR

RC5 IR refers to a protocol developed by Phillips in the late 1980's as a consumer infrared remote control communication code for consumer electronics. The RC5 uses a bi-phase coding with a frequency of 36 kHz and is comprised of a keypad and a transmitter integrated circuit, which drives an infrared LED. The transmission of data starts with two bits followed by a toggle bit, which changes value upon a each key-press. There are five address bits, which indicate the device that is being controlled. Lastly, the six commands bits represent the data that is to be transmitted from the RC5 to the device.

### 3.3 NES Controller

The NES Controller is an 8-bit controller developed by Nintendo in the 1980's. The device transmits 8-bits of data, which equates to one bit for each of the buttons on the controller: A, B, Select, Start, Up, Down, Left, and Right. The NES controller also contains a latch to store the state of buttons internally as well as a clock. Every 60 Hz, the NES sends a 12 microsecond signal to the latch, indicating that it must store the state of all eight buttons. Following the Latch signal, 8 clock pulses are emitted, one for each button on the controller. If the button is pressed upon the rising edge of the pulse, data is asserted to ground. This means that in the NES controller, the buttons are active low because Data asserts to ground when the button is pressed.

## 4 HDL Modules

## 4.1 PS/2 Modules

#### 4.1.1 PS/2 Decoder



Figure 4.1.1: PS/2 keyboard decoder



Figure 4.1.2: PS/2 module's simulation testing - Keypress W (up)



Figure 4.1.3: PS/2 module's simulation testing - Keypress S (down)



Figure 4.1.4: PS/2 module's simulation testing - Keypress A (left)



Figure 4.1.5: PS/2 module's simulation testing - Keypress D (right)



Figure 4.1.6: PS/2 module's simulation testing - Key release signal (0xF0)

Inputs: Clock and Data signal.

Outputs: Two buses to represent the box's movement in the  $x$  and  $y$  direction.

Description: The PS/2 keyboard decoder module begins with an always flip flop on the negative edge of the clock signal, indicating that the data values are generated only after the clock has gone to low. The module then determines the value of `count`, which continuously increments between 0 and 10. If the value of `count` is 10, then the keyboard inputs, recognized as A, W, S, and D, are translated into variables representing the movement and direction of the box image. If the `count` is between 1 and 8, then data from the PS/2 keyboard is simply stored until `count` returns to 10, in which the values are then outputted to directional values stored in the variables `dir_x` and `dir_y`.

## 4.2 RC5 IR Modules

### 4.2.1 RC5 IR Decoder



Figure 4.2.1: RC5 IR decoder



Figure 4.2.2: RC5 IR decoder decoding up, down, left, right, and neutral direction controls from data word

Inputs: Clock and Data signal.

Outputs: Two buses to represent the box's movement in the  $x$  and  $y$  direction.

Description: The first step in designing the RC5 IR module was slowing the normal FPGA clock frequency of 50 MHz to 36 KHz, the frequency used in RC5 data transmission. RC5 works by using bi-phase encoding, meaning that during each clock period, there is either a 1 or 0, where 1 represents a rising edge and 0 represents a falling edge. Next, the

ConvertPulse and ReadData modules were instantiated. Convert Pulse takes in the 32-bit pulse emitted by the RC5 and treats the entire pulse as HIGH and the absence of a pulse as LOW. Then, when the pulse signal encoding and bi-phase encoding are decoded, a 14-bit result follows. The first 2-bits are start bits and represent when a new data transmission occurs. The toggle is unaffected if no new button is pressed. However, if a new data line is activated, the toggle bit changes. The next 5-bits are address bits to indicate the path to a device that is being transmitted to. Lastly, 6-bits are used to represent the transmitted data. In our module, the encodings used to represent the direction and

#### 4.2.2 RC5 IR Convert Pulse



Figure 4.2.3: IR convert pulse



Figure 4.2.4: Testing conversion of inputs from idle to HIGH to LOW

Description: The module above converts the encoded pulse outputted in bi-phase encoding by RC5 IR into binary form.

### 4.2.3 RC5 IR Read Word



Figure 4.2.5: IR read word



Figure 4.2.6: Readind data word “111111” then resetting

Description: The module above converts encoded binary into data.

## 4.3 NES Modules

### 4.3.1 NES Decoder



Figure 4.3.1: NES controller decoder



Figure 4.3.2: NES module's simulation testing - Moving up keypress

Inputs: Clock and Data signal.

Outputs: Two buses to represent the box's movement in the  $x$  and  $y$  direction. Also outputs new clock signal and latch data.

Description: Initially, the decoder sends a latch output to the controller. Every 60Hz, the decoder asserts the latch for 12 microseconds, which stabilizes each button's state. For this module, a clock slowing mechanism was inserted so that the internal clock of the FPGA, which is naturally 50MHz, can be used. On the falling edge of the latch, the decoder reads the first value, which is A. Six microseconds after, it begins sending 8 clock pulses which have a period of 12 microseconds. On every rising edge of the clock, the decoder cycles to the next button value, for a total of eight button values. The value is then read on the falling edge of that cycle and stored in module variables. Since the only button actions which affect our box image are the up, down, left, and right arrows, those are the only ones that are stored and sent to different modules.

## 4.4 General Modules Used in Each Design

### 4.4.1 Clock Frequency Module



Figure 4.4.1: Half clock module



Figure 4.4.2: Half clock module's simulation testing - Making a half-as-fast clock

Inputs: `oldClock` signal.

Outputs: `newClock` signal representing a clock with half the frequency of `oldClock`.

Description: The clock module plays an essential role in allowing the output to be displayed as VGA. The natural frequency of the DE10-Lite is 50 MHz. However, the pixel clock on the VGA mode timing specifications is 25 MHz. Because of this difference, a half clock module is required to half the clock frequency of the FPGA so that it corresponds with the pixel clock at 25 MHz.

#### 4.4.2 Box Controller Module



Figure 4.4.3: Box controller module



Figure 4.4.4: Box controller module’s simulation testing - Box collides with the top edge of the screen



Figure 4.4.5: Box controller module’s simulation testing - Box collides with the left edge of the screen

Inputs: Vertical Sync. X and Y direction in which the box is moving.

Outputs: Three 4-bit buses representing red, green, and blue. Two 10-bit buses representing the x and y location of the box.

Description: The Box Controller Module controls the action and animations of the box image. The module takes in 3 inputs representing vertical sync,  $X$  direction, and  $Y$  direction.  $X$  direction and  $Y$  specify which direction the box is moving, where 0 represents left or up and 1 represents right or down. The Box Controller checks to ensure that the box will remain in the active pixel range. It then checks to see if a collision with a vertical or horizontal boundary has occurred. If this is true, then the red, green, and blue data lines corresponding to the box are switched, forcing the image to change color. In addition, the box stops moving. For example, if a red box were heading to the right and encounters a collision with the rightmost part of the screen, then the box will change colors to blue, and also stop moving so it does not continue past the edge of the screen. The current state of the box, including its position and color, are outputted to Drawer module.

#### 4.4.3 XY Counter Module



Figure 4.4.6: XY Counter module



Figure 4.4.7: XY counter module's simulation testing - Horizontal front porch



Figure 4.4.8: XY counter module's simulation testing - Horizontal sync pulse



Figure 4.4.9: XY counter module's simulation testing - Horizontal back porch and counter completes one line



Figure 4.4.10: XY counter module's simulation testing - Vertical front porch



Figure 4.4.11: XY counter module's simulation testing - Vertical sync pulse



Figure 4.4.12: XY Counter module's simulation testing - Vertical back porch and counter completes one screen

Inputs: Clock

Outputs: Vertical Sync and Horizontal Sync. Two 10-bit buses representing the current  $x$  and  $y$  pixel location.

Description: The XY Counter module plays a critical role in outputting data as a VGA image. Its purpose is to keep track of horizontal ( $x$ ) and vertical ( $y$ ) values to track the pixel position. The module takes only one input, which is the clock, and outputs values for

HSync, VSync, and  $x$  and  $y$  positions. By looking at Figure 4.4.6, HSync and VSync values are calculated and then outputted directly to the VGA. The  $x$  representing the pixel position is continuously incremented until it reaches 800, which signifies a new frame, during which  $y$  is initialized and incremented by one. At the same time,  $x$  is reset to 0 and incremented again to 800. This process repeats until  $y$  reaches 524, which signifies the end of the screen. At this moment, both  $x$  and  $y$  are reset to 0. The XY Counter module outputs the  $x$  and  $y$  location to the Drawer module.

#### 4.4.4 Drawer Module



Figure 4.4.13: Drawer module

**Inputs:** Three 4-bit buses representing red, green, and blue.  $X$  and  $Y$  counter as well as  $x$  and  $y$  locations of the box.

**Outputs:** Three 4-bit buses representing red, green, and blue.

**Description:** The Drawer module shown above is the last step in displaying the output as a VGA image. This module reads in data corresponding to the  $x$  and  $y$  locations that are outputted from the XY Counter module as well as  $x$  and  $y$  values indicating the size of the box to be displayed. If the  $x$  and  $y$  locations outputted from the counter module are not within the active pixel range specified by the VGA timing specifications, then a black box is drawn, effectively showing nothing. However, if the location of  $x$  and  $y$  are within the active pixel range, then the outputs: red, green, and blue, are assigned values so that a colorful box image appear on the monitor. This box will appear at the specified box  $x$  and  $y$  values,



Figure 4.4.14: Drawer module's simulation testing - Different pixel colors at different coordinates

and will have a size specified by the inputs. If the current pixel is outside the location of the box, black is displayed, otherwise either red, green, or blue are displayed to form the box.

## 5 Putting It All Together



Figure 5.0.1: PS/2 Top Level module's simulation testing - Box starts moving up after receiving a W keypress



Figure 5.0.2: PS/2 Top Level module's simulation testing - Box has moved up 1 pixel. For the remaining 219 pixels the box has left before colliding with the screen, it moves in the same way



Figure 5.0.3: PS/2 Top Level module’s simulation testing - Box changes color when colliding with the screen

Since our group decided to implement all three modes of input, constructing this project was essentially like creating three designs with several commonalities. In each design, the XY Counter, Half Clock, Box Controller, and Drawer modules were all used in transferring data from the FPGA to a VGA output. The one difference between each design was implementation of the decoder module. Each unique form of input needed a separate decoder module that was able to read and transmit data correctly from the controller to the other modules within the FPGA. A top level HDL file was made for all three modes of input and tested thoroughly using ModelSim.

## 6 Above and Beyond

After looking at the rubric, our group decided to take it one step further by accomplishing all of the tasks listed on the rubric. We successfully implemented VGA output using all three

forms of input: PS/2, RC5 IR, and NES. In addition, we attempted to animate the behavior of the outputted image, rather than having a static and plain figure. By adding the box controller module, we included logic that would force the image to change color and direction upon one condition. If the box hit the edge of the active pixel range, whether in the horizontal or vertical direction, then it would change color. In addition to changing color, the box stops moving, creating a collision.

## 7 Appendix

### 7.1 SystemVerilog Source Code

#### 7.1.1 Top Level

---

```

1  module top_level(
2      input logic clk,
3      input logic[1:0] dir_x, dir_y,
4      output logic vs, hs,
5      output logic[3:0] r, g, b
6  );
7
8  logic[9:0] xpos, ypos;
9  logic slowclk;
10
11 logic[3:0] box_r, box_g, box_b;
12 logic[9:0] box_x, box_y, box_size;
13
14 half_clock hc (
15     .oldclk(clk),
16     .newclk(slowclk)
17 );
18
19 XYCounter c (
20     .clk(slowclk),
21     .vs(vs),
22     .hs(hs),
23     .x(xpos),
24     .y(ypos)
25 );
26
27 box_controller box (

```

```

28     .vs(vs),
29     .dir_x(dir_x),
30     .dir_y(dir_y),
31     .x(xpos),
32     .y( ypos),
33     .box_r(box_r),
34     .box_g(box_g),
35     .box_b(box_b),
36     .box_x(box_x),
37     .box_y(box_y),
38     .box_size(box_size)
39 );
40
41 Drawer d (
42     .clk(slowclk),
43     .box_r(box_r),
44     .box_g(box_g),
45     .box_b(box_b),
46     .x(xpos),
47     .y( ypos),
48     .box_x(box_x),
49     .box_y(box_y),
50     .box_size(box_size),
51     .r(r),
52     .g(g),
53     .b(b)
54 );
55
56 endmodule

```

---

### 7.1.2 PS/2 Top Level

---

```

1 module ps2_top_level(
2     input logic clk, psClk, data,
3     output logic vs, hs,
4     output logic[3:0] r, g, b
5 );
6
7 // Track x and y and slow clock

```

```

8   logic[9:0] xpos, ypos;
9   logic slowclk;
10
11 // Get direction input
12 logic[1:0] dir_x, dir_y;
13
14 // Track box info
15 logic[3:0] box_r, box_g, box_b;
16 logic[9:0] box_x, box_y, box_size;
17
18 // Slow clock
19 half_clock hc (
20     .oldclk(clk),
21     .newclk(slowclk)
22 );
23
24 // Generate position and vs/hs signals
25 XYCounter c (
26     .clk(slowclk),
27     .vs(vs),
28     .hs(hs),
29     .x(xpos),
30     .y(ypos)
31 );
32
33 // Get keyboard input for direction
34 ps2 P (
35     .clk(psClk),
36     .data(data),
37     .dir_x(dir_x),
38     .dir_y(dir_y)
39 );
40
41 // Move box
42 box_controller box (
43     .vs(vs),
44     .dir_x(dir_x),
45     .dir_y(dir_y),
46     .box_r(box_r),
47     .box_g(box_g),
48     .box_b(box_b),

```

```

49     .box_x(box_x),
50     .box_y(box_y),
51     .box_size(box_size)
52 );
53
54 // Draw box
55 Drawer d (
56     .clk(slowclk),
57     .box_r(box_r),
58     .box_g(box_g),
59     .box_b(box_b),
60     .x(xpos),
61     .y(ypos),
62     .box_x(box_x),
63     .box_y(box_y),
64     .box_size(box_size),
65     .r(r),
66     .g(g),
67     .b(b)
68 );
69
70 endmodule

```

---

### 7.1.3 PS/2 Controller

---

```

1 module ps2 #(parameter
2     UP = 8'h1D, DOWN = 8'h1B, LEFT = 8'h1C, RIGHT = 8'h23 // PS2 key code of W,
3     ↵ S, A, and D, respectively
4 )((
5     input logic clk, data,
6     output logic[1:0] dir_x, dir_y
7 );
8
9 // Track data input
10 logic[7:0] bits = 8'b0;
11
12 // Track count of bits coming in and number of ones
13 logic[3:0] count = 0;
14 logic[3:0] even_odd = 0;

```

```

14
15 // Hold output values
16 // 0 = up/left
17 // 1 = down/right
18 // 2 = stop
19 logic[1:0] x = 2;
20 logic[1:0] y = 2;
21
22 // Track valid input and whether past input was F0 (indicating key release)
23 logic valid = 0;
24 logic fo = 0;
25
26 always_ff @(negedge clk)
27 begin
28     if (count == 10) // Ending bit
29         begin
30             if (valid)
31                 // Translate directions to movement
32                 begin
33                     // not released && is not moving && to move up -> move up
34                     if (~fo && y == 2 && bits == UP) y <= 0;
35                     // released && is moving up && to move up -> stop
36                     else if (fo && y == 0 && bits == UP) y <= 2;
37                     // not released && is not moving && to move down -> move
38                     // down
39                     else if (~fo && y == 2 && bits == DOWN) y <= 1;
40                     // released && is moving down && to move down -> stop
41                     else if (fo && y == 1 && bits == DOWN) y <= 2;
42
43                     // not released && is not moving && to move left -> move
44                     // left
45                     if (~fo && x == 2 && bits == LEFT) x <= 0;
46                     // released && is moving left && to move left -> stop
47                     else if (fo && x == 0 && bits == LEFT) x <= 2;
48                     // not released && is not moving && to move right -> move
49                     // right
50                     else if (~fo && x == 2 && bits == RIGHT) x <= 1;
51                     // released && is moving right && to move right -> stop
52                     else if (fo && x == 1 && bits == RIGHT) x <= 2;
53
54             // Get code for key release

```

```

52             if (bits == 8'hF0) fo <= 1;
53             else fo <= 0;
54         end
55
56         even_odd <= 0;
57         count <= 0;
58         bits <= 8'd0;
59         valid <= 0;
60     end
61     else if (count == 9) // Check parity bit
62     begin
63         count <= count + 1;
64
65         // Check validity based on number of ones and parity bit (odd
66         // ← total)
67         valid <= even_odd[0] == ~data;
68     end
69     else if (count == 0) // Start bit -- do nothing but advance count
70     begin
71         count <= count + 1;
72     end
73     else
74     begin
75         // Track count and number of odd
76         even_odd <= even_odd + data;
77         count <= count + 1;
78
79         // Load data
80         bits <= (bits >> 1) + (data * 128);
81     end
82 end
83
84 // Set output
85 assign dir_x = x;
86 assign dir_y = y;
87
88 endmodule

```

---

#### 7.1.4 RC5 IR Top Level

---

```
1  module ir_top_level(
2      input logic clk, data,
3      output logic vs, hs,
4      output logic[3:0] r, g, b
5  );
6
7  // Track x and y and slow clock
8  logic[9:0] xpos, ypos;
9  logic slowclk;
10
11 // Get direction input
12 logic[1:0] dir_x, dir_y;
13
14 // Track box info
15 logic[3:0] box_r, box_g, box_b;
16 logic[9:0] box_x, box_y, box_size;
17
18 // Slow clock
19 half_clock hc (
20     .oldclk(clk),
21     .newclk(slowclk)
22 );
23
24 // Generate position and vs/hs signals
25 XYCounter c (
26     .clk(slowclk),
27     .vs(vs),
28     .hs(hs),
29     .x(xpos),
30     .y(ypos)
31 );
32
33 // Get IR input for direction
34 ir I (
35     .clk(clk),
36     .data(data),
37     .dir_x(dir_x),
38     .dir_y(dir_y)
```

```

39 );
40
41 // Move box
42 box_controller box (
43     .vs(vs),
44     .dir_x(dir_x),
45     .dir_y(dir_y),
46     .box_r(box_r),
47     .box_g(box_g),
48     .box_b(box_b),
49     .box_x(box_x),
50     .box_y(box_y),
51     .box_size(box_size)
52 );
53
54 // Draw box
55 Drawer d (
56     .clk(slowclk),
57     .box_r(box_r),
58     .box_g(box_g),
59     .box_b(box_b),
60     .x(xpos),
61     .y( ypos),
62     .box_x(box_x),
63     .box_y(box_y),
64     .box_size(box_size),
65     .r(r),
66     .g(g),
67     .b(b)
68 );
69
70 endmodule

```

---

### 7.1.5 RC5 IR Controller

---

```

1 // IR module for RC5 code
2 module ir #(parameter
3     UP = 2, DOWN = 8, LEFT = 4, RIGHT = 6
4 )(

```

```

5      input logic data, clk,
6      output logic[1:0] dir_x, dir_y
7  );
8
9 // Count for slowing clock and new clock
10 logic[10:0] clkCount = 0;
11 logic newclk = 0;
12
13 // Complete data word and its update status
14 logic[5:0] dataWord = 0;
15 logic newDataWord = 0;
16
17 // Internal x/y movement
18 logic[1:0] x = 2;
19 logic[1:0] y = 2;
20
21 // Used for handling pulses
22 logic pusleVal = 0;
23 logic newPulseVal = 0;
24 logic waitForRepeat = 0;
25
26 // Slow clock from 50MHz to 36KHz as is used for RC5
27 always_ff @(posedge clk)
28     if (clkCount < 1388)
29         clkCount <= clkCount + 1;
30     else
31         begin
32             clkCount <= 0;
33             newclk = ~newclk;
34         end
35
36 // Converts from encoded pulse to binary
37 irConvertPulse cp(
38     .data(data),
39     .clk(newclk),
40     .val(pusleVal),
41     .newval(newPulseVal),
42     .waitForRepeat(waitForRepeat)
43 );
44
45 // Converts from encoded binary to data word

```

```

46    irReadWord rw(
47        .val(pulseVal),
48        .clk(newPulseVal),
49        .waitForRepeat(waitForRepeat),
50        .data(dataWord),
51        .finished(newDataWord)
52    );
53
54 // Translate from data word to movement value
55 always_ff @(posedge newDataWord)
56 begin
57     if (dataWord == UP) y <= 0;
58     else if (dataWord == DOWN) y <= 1;
59     else y <= 2;
60
61     if (dataWord == LEFT) x <= 0;
62     else if (dataWord == RIGHT) x <= 1;
63     else x <= 2;
64 end
65
66 assign dir_x = x;
67 assign dir_y = y;
68
69 endmodule
70
71
72 module irReadWord(
73     input logic val, clk, waitForRepeat,
74     output logic[5:0] data,
75     output logic finished
76 );
77
78 // Stores output and counts bits
79 logic[5:0] out = 0;
80 logic[4:0] count = 0;
81
82 // Tracks previous toggle -- not used in this implementation
83 // but necessary for certain controls so it will be tracked
84 // anyway
85 logic prevToggle = 1;
86

```

```

87  always_ff @(posedge clk, negedge waitForRepeat)
88    begin
89      if (~waitForRepeat) // Reset if time limit for repeating signal is up
90        begin
91          out <= 0;
92          count <= 0;
93          finished <= 1;
94        end
95      else if (count == 13) // Reset count, store final value, and signal
96        ↳ finished transmission
97        begin
98            out <= (out << 1) + val;
99            finished <= 1;
100           count <= 0;
101        end
102      else if (count == 2 && val != prevToggle) // Set toggle and reset value
103        ↳ if different button press
104        begin
105            prevToggle <= val;
106            out <= 0;
107            count <= count + 1;
108            finished <= 0;
109        end
110      else if (count > 7) // If in data word, add to output
111        begin
112            out <= (out << 1) + val;
113            count <= count + 1;
114            finished <= 0;
115        end
116      else
117        begin
118            count <= count + 1;
119            finished <= 0;
120        end
121    assign data = out;
122
123  endmodule
124
125

```

```

126 module irConvertPulse(
127     input logic data, clk,
128     output logic val, newval, waitForRepeat
129 );
130
131 // Tracks two sections with potential to have pulses
132 logic[5:0] dataCounterOne = 0;
133 logic[5:0] dataCounterTwo = 0;
134
135 // Counts time
136 logic[5:0] timeCounter = 0;
137
138 // Counts time from
139 logic[6:0] waitCounter = 78;
140 logic idle = 1;
141 logic out = 0;
142
143 logic tcReset = 0;
144 logic dcReset = 0;
145 logic dcResetSuccess = 0;
146
147 always_ff @(posedge data)
148 begin
149     if (idle) // Reset time counter if start of a transmission
150         begin
151             dataCounterOne <= 0;
152             dataCounterTwo <= 1;
153             tcReset <= 1;
154         end
155     else if (dcReset)
156         begin
157             if (timeCounter > 31 && dataCounterTwo < 32)
158                 begin
159                     dataCounterOne <= 0;
160                     dataCounterTwo <= 1;
161                 end
162             else if (dataCounterOne < 32)
163                 begin
164                     dataCounterOne <= 1;
165                     dataCounterTwo <= 0;
166                 end

```

```

167
168         dcResetSuccess <= 1;
169     end
170 else // Increment data
171 begin
172     if (timeCounter > 31 && dataCounterTwo < 32) dataCounterTwo <=
173         → dataCounterTwo + 1;
174     else if (dataCounterOne < 32) dataCounterOne <= dataCounterOne +
175         → 1;
176
177         tcReset <= 0;
178         dcResetSuccess <= 0;
179     end
180 end
181
182 always_ff @(posedge clk)
183 begin
184     if (tcReset)
185         begin
186             timeCounter <= 33;
187             waitCounter <= 0;
188             idle <= 0;
189         end
190     else if (timeCounter == 63) // Set output and reset
191         begin
192             // Increment counter if still in window for additional
193             if (idle && dataCounterOne == 0 && dataCounterTwo == 0 &&
194                 → waitCounter < 78)
195                 waitCounter <= waitCounter + 1;
196
197             out <= dataCounterOne == 0 && dataCounterTwo == 32 && ~dcReset;
198             newval <= (dataCounterOne == 32 || dataCounterTwo == 32) &&
199                 → ~dcReset;
200             idle <= (dataCounterOne == 0 && dataCounterTwo == 0) || ~dcReset;
201             dcReset <= 1;
202             timeCounter <= 0;
203         end
204     else // Increment data
205         begin
206             timeCounter <= timeCounter + 1;
207             newval <= 0;

```

```

204
205         if (dcResetSuccess)
206             dcReset <= 0;
207     end
208 end
209
210 assign val = out;
211
212 // Do not wait if outside of continued transmission window
213 assign waitForRepeat = waitCounter < 78;
214
215 endmodule

```

---

### 7.1.6 NES Top Level

---

```

1 module nes_top_level(
2     input logic clk, data,
3     output logic vs, hs, latch_out, clk_out,
4     output logic[3:0] r, g, b
5 );
6
7 // Track x and y and slow clock
8 logic[9:0] xpos, ypos;
9 logic slowclk;
10
11 // Get direction input
12 logic[1:0] dir_x, dir_y;
13
14 // Track box info
15 logic[3:0] box_r, box_g, box_b;
16 logic[9:0] box_x, box_y, box_size;
17
18 // Slow clock
19 half_clock hc (
20     .oldclk(clk),
21     .newclk(slowclk)
22 );
23
24 // Generate position and vs/hs signals

```

```

25 XYCounter c (
26     .clk(slowclk),
27     .vs(vs),
28     .hs(hs),
29     .x(xpos),
30     .y( ypos)
31 );
32
33 // Get nes input for direction
34 nes N (
35     .clk(clk),
36     .data(data),
37     .dir_x(dir_x),
38     .dir_y(dir_y),
39     .latch_out(latch_out),
40     .clk_out(clk_out)
41 );
42
43 // Move box
44 box_controller box (
45     .vs(vs),
46     .dir_x(dir_x),
47     .dir_y(dir_y),
48     .box_r(box_r),
49     .box_g(box_g),
50     .box_b(box_b),
51     .box_x(box_x),
52     .box_y(box_y),
53     .box_size(box_size)
54 );
55
56 // Draw box
57 Drawer d (
58     .clk(slowclk),
59     .box_r(box_r),
60     .box_g(box_g),
61     .box_b(box_b),
62     .x(xpos),
63     .y( ypos),
64     .box_x(box_x),
65     .box_y(box_y),

```

```

66     .box_size(box_size),
67     .r(r),
68     .g(g),
69     .b(b)
70 );
71
72 endmodule

```

---

### 7.1.7 NES Controller

---

```

1 module nes #(parameter
2     UP = 5, DOWN = 6, LEFT = 7, RIGHT = 8
3 )(
4     input logic data, clk,
5     output logic latch_out, clk_out,
6     output logic[1:0] dir_x, dir_y
7 );
8
9 // Track status of x and y direction and if a directional button was pressed
10 logic[1:0] x = 2;
11 logic[1:0] y = 2;
12 logic x_btn_press = 0;
13 logic y_btn_press = 0;
14
15 // Track when to poll controller for button presses
16 logic[19:0] pollClkCount = 0;
17 logic pollClk = 0;
18
19 // Track when to pulse clock
20 logic[9:0] pulseClkCount = 0;
21 logic pulseClk = 0;
22
23 // Track info about latch and clock pulse
24 logic latch_inner = 0;
25 logic new_pulse = 0;
26 logic start_pulse_out = 0;
27 logic[3:0] pulseCounter = 0;
28
29 // Convert clk from 50MHz to 60Hz for polling controller

```

```

30 // and to 83kHz to produce clock pulses
31 always_ff @(posedge clk)
32 begin
33 if (pollClkCount == 416666)
34 begin
35 // Start polling controller by resetting pulse data, starting
36 // → latch,
37 // and signaling start of new pulse
38 if (~pollClk)
39 begin
40 latch_inner <= 1;
41 new_pulse <= 1;
42 pulseClkCount <= 0;
43 pulseClk <= 0;
44 pulseCounter <= 0;
45 end
46
47 pollClkCount <= 0;
48 pollClk <= ~pollClk;
49 end
50 else
51 pollClkCount <= pollClkCount + 1;
52
53 if (pulseClkCount == 301)
54 begin
55 if (new_pulse && ~pulseClk) // Only update if controller poll
56 // → has started
57 begin
58 // Track beginning of clock pulse to controller (after
59 // → latch)
60 if (pulseCounter > 0 && latch_inner == 0)
61 start_pulse_out <= 1;
62
63 // Count pulse
64 pulseCounter <= pulseCounter + 1;
65 end
66
67 if (new_pulse && pulseClk) // Only update if controller poll has
68 // → started
69 begin
70 // Reset latch once its 12us have passed

```

```

67     if (pulseCounter > 0 && latch_inner)
68         latch_inner <= 0;
69
70     // Reset pulse data if all buttons have been accounted
71     if (pulseCounter == 9)
72         begin
73             new_pulse <= 0;
74             start_pulse_out <= 0;
75             y_btn_press <= 0;
76             x_btn_press <= 0;
77         end
78     else
79         begin
80             // Check for UP or DOWN buttons being pressed and
81             // track that
82             // a button in y direction has been pressed
83             if (pulseCounter == UP && data == 0)
84                 begin
85                     y <= 0;
86                     y_btn_press <= 1;
87                 end
88             else if (pulseCounter == DOWN && data == 0)
89                 begin
90                     y <= 1;
91                     y_btn_press <= 1;
92                 end
93             else if (pulseCounter >= UP && pulseCounter >=
94                 // DOWN && y_btn_press == 0) // Reset if no y
95                 // button press
96                 y <= 2;
97
98             // Check for RIGHT or LEFT buttons being pressed
99             // and track that
100            // a button in x direction has been pressed
101            if (pulseCounter == LEFT && data == 0)
102                begin
103                    x <= 0;
104                    x_btn_press <= 1;
105                end
106            else if (pulseCounter == RIGHT && data == 0)
107                begin

```

```

104           x <= 1;
105           x_btn_press <= 1;
106       end
107   else if (pulseCounter >= LEFT && pulseCounter >=
108     ↵  RIGHT && y_btn_press == 0) // Reset if no x
109     ↵  button press
110           x <= 2;
111       end
112   end
113
114   pulseClkCount <= 0;
115   pulseClk <= ~pulseClk;
116 end
117
118 // Assign outputs, and only set clock if part of pulse output
119 assign latch_out = latch_inner;
120 assign clk_out = start_pulse_out && pulseClk;
121 assign dir_x = x;
122 assign dir_y = y;
123
124
125 endmodule

```

---

### 7.1.8 Box Controller

---

```

1 module box_controller #(parameter
2   SIZE = 40
3 )(
4   input logic vs,
5   input logic[1:0] dir_x, dir_y,
6   output logic[3:0] box_r, box_g, box_b,
7   output logic[9:0] box_x, box_y, box_size
8 );
9
10 // Track x and y
11 logic[9:0] xpos = 300;
12 logic[9:0] ypos = 220;

```

```

13
14 // Track box color
15 logic[3:0] r = 4'b1111;
16 logic[3:0] g = 4'b0;
17 logic[3:0] b = 4'b0;
18
19 always_ff @(posedge vs)
20 begin
21     if(~(dir_x == 0 && xpos == 0) && ~(dir_x == 1 && xpos == 640 - SIZE)) // 
22         ↵ Check that box will remain in screen bounds
23     begin
24         if((xpos - 1 == 0 && dir_x == 0) || (xpos + 1 == 640 - SIZE &&
25             ↵ dir_x == 1)) // Check for collision
26             begin
27                 // Swap colors if collision occurs
28                 r <= g;
29                 g <= b;
30                 b <= r;
31             end
32
33         // Increment position
34         case(dir_x)
35             0: xpos <= xpos - 1;
36             1: xpos <= xpos + 1;
37         endcase
38     end
39
40     if(~(dir_y == 0 && ypos == 0) && ~(dir_y == 1 && ypos == 480 - SIZE)) // 
41         ↵ Check that box will remain in screen bounds
42     begin
43         if((ypos - 1 == 0 && dir_y == 0) || (ypos + 1 == 480 - SIZE &&
44             ↵ dir_y == 1)) // Check for collision
45             begin
46                 // Swap colors if collision occurs
47                 r <= g;
48                 g <= b;
49                 b <= r;
50             end
51
52         // Increment position
53         case(dir_y)

```

```

50          0: ypos <= ypos - 1;
51          1: ypos <= ypos + 1;
52      endcase
53  end
54
55
56 assign box_x = xpos;
57 assign box_y = ypos;
58 assign box_size = SIZE;
59 assign box_r = r;
60 assign box_g = g;
61 assign box_b = b;
62
63 endmodule

```

---

### 7.1.9 Half Clock

---

```

1 module half_clock(
2     input logic oldclk,
3     output logic newclk
4 );
5
6 logic count = 0;
7
8 always_ff @(posedge oldclk)
9     if (count == 1)
10         begin
11             newclk <= 1;
12             count <= 0;
13         end
14     else
15         begin
16             newclk <= 0;
17             count <= count + 1;
18         end
19
20 endmodule

```

---

### 7.1.10 XY Counter

---

```
1  module XYCounter(
2      input logic clk,
3      output logic vs, hs,
4      output logic[9:0] x, y
5  );
6
7  // Track current x and y
8  logic[9:0] xpos = 0;
9  logic[9:0] ypos = 0;
10
11 always_ff @(posedge clk)
12 begin
13     // Generate HSync and VSync signals when in sync section
14     hs <= ~((xpos >= 640 + 16) && (xpos < 640 + 16 + 96));
15     vs <= ~((ypos >= 480 + 10) && (ypos < 480 + 10 + 2));
16
17     if(xpos == 800 && ypos == 525) // Reset when at end of screen
18         begin
19             ypos <= 0;
20             xpos <= 0;
21         end
22     else if(xpos == 800) // Increment y and reset x at end of line
23         begin
24             ypos <= ypos + 1;
25             xpos <= 0;
26         end
27     else // Otherwise increment x
28         xpos <= xpos + 1;
29     end
30
31     assign x = xpos;
32     assign y = ypos;
33
34 endmodule
```

---

### 7.1.11 Drawer

---

```
1 module Drawer(
2     input logic clk,
3     input logic[3:0] box_r, box_g, box_b,
4     input logic[9:0] x, y, box_x, box_y, box_size,
5     output logic[3:0] r, g, b
6 );
7
8 always_ff @(posedge clk)
9     // Draw black if not in box pixels
10    if (x > 640 || y > 480 || x < box_x || y < box_y || x > box_x + box_size || y
11        > box_y + box_size)
12        begin
13            r <= 4'b0;
14            g <= 4'b0;
15            b <= 4'b0;
16        end
17    else // Draw box
18        begin
19            r <= box_r;
20            g <= box_g;
21            b <= box_b;
22        end
23 endmodule
```

---