

# ECE 271 Group 2 Design Project

Taylor Brennan, Marcus Wheeler, Tyler Walker

June 4th, 2021

## Contents

|          |                                         |           |
|----------|-----------------------------------------|-----------|
| <b>1</b> | <b>Project Description</b>              | <b>3</b>  |
| 1.1      | Project Deliverables . . . . .          | 3         |
| <b>2</b> | <b>IR TV Remote Description</b>         | <b>4</b>  |
| 2.1      | Timing and Power . . . . .              | 4         |
| <b>3</b> | <b>High Level Description</b>           | <b>5</b>  |
| 3.0.1    | Top Level Simulation . . . . .          | 5         |
| 3.1      | IR Receiver . . . . .                   | 5         |
| 3.1.1    | IR Receiver Simulation . . . . .        | 6         |
| 3.2      | IR Decoder . . . . .                    | 7         |
| 3.2.1    | First Multiplexer . . . . .             | 7         |
| 3.2.2    | Second Multiplexer . . . . .            | 8         |
| 3.2.3    | Comparator . . . . .                    | 9         |
| 3.2.4    | Shift Register . . . . .                | 9         |
| 3.2.5    | Address Swapper . . . . .               | 10        |
| 3.2.6    | Tristate . . . . .                      | 11        |
| 3.2.7    | IR Decoder Simulation . . . . .         | 11        |
| 3.3      | VGA Display Output . . . . .            | 12        |
| 3.3.1    | Top Level Diagram . . . . .             | 12        |
| 3.3.2    | Counter . . . . .                       | 12        |
| 3.3.3    | Sync Count . . . . .                    | 12        |
| 3.3.4    | Color Mux . . . . .                     | 14        |
| 3.4      | Sprite VGA . . . . .                    | 15        |
| 3.4.1    | Top Level Diagram . . . . .             | 15        |
| 3.4.2    | Address Modifier . . . . .              | 16        |
| 3.4.3    | Sprite Comparator . . . . .             | 16        |
| 3.4.4    | Address Converter . . . . .             | 17        |
| 3.4.5    | Sprite ROM . . . . .                    | 17        |
| 3.4.6    | Display Mux . . . . .                   | 18        |
| 3.4.7    | VGA Display . . . . .                   | 18        |
| 3.4.8    | VGA Display Output Simulation . . . . . | 18        |
| <b>A</b> | <b>SystemVerilog Files</b>              | <b>19</b> |
| A.1      | IR Receiver . . . . .                   | 19        |
| A.1.1    | Comparator . . . . .                    | 19        |
| A.1.2    | Clock Counter . . . . .                 | 19        |
| A.1.3    | Enable Counter . . . . .                | 19        |
| A.1.4    | Multiplexer . . . . .                   | 19        |
| A.1.5    | Shift Register . . . . .                | 19        |
| A.1.6    | Tristate . . . . .                      | 19        |
| A.1.7    | Synchronizer . . . . .                  | 20        |
| A.2      | IR Decoder . . . . .                    | 20        |
| A.2.1    | Comparator . . . . .                    | 20        |
| A.2.2    | First Multiplexer . . . . .             | 20        |
| A.2.3    | Second Multiplexer . . . . .            | 20        |

|          |                              |           |
|----------|------------------------------|-----------|
| A.2.4    | Shift Register . . . . .     | 20        |
| A.2.5    | Address Swapper . . . . .    | 20        |
| A.2.6    | Tristate . . . . .           | 21        |
| A.3      | VGA Display Output . . . . . | 21        |
| A.3.1    | Counter . . . . .            | 21        |
| A.3.2    | Sync Count . . . . .         | 21        |
| A.3.3    | Color Mux . . . . .          | 22        |
| A.4      | Sprite VGA . . . . .         | 22        |
| A.4.1    | Address Modifier . . . . .   | 22        |
| A.4.2    | Sprite Comparator . . . . .  | 22        |
| A.4.3    | Address Converter . . . . .  | 22        |
| A.4.4    | Display Mux . . . . .        | 22        |
| <b>B</b> | <b>Do files</b>              | <b>23</b> |
| B.0.1    | RC-5 Encoder . . . . .       | 23        |
| B.0.2    | IR Decoder . . . . .         | 23        |

# 1 Project Description

The goal of this project is to design logic hardware to read an infrared (IR) signal from a TV remote and move a sprite on a VGA display. The TV remote follows the RC-5 IR transmission protocol, which sends a series of time based pulses that correspond with a logic 1 or 0. By using an IR receiver module, decoders to interpret the signals, and registers to convert the signals into sprite movement, the pulses can be used to change where a sprite appears on a VGA display.



Figure 1: RC-5 IR TV Remote (Aliradar ©2021)

In this project, pressing buttons 2, 4, 6, and 8 shown in Figure 1 will move the sprite up, down, left, and right, respectively. The other buttons will not have an effect on the sprite display. By implementing this ability, the sprite movement can be used for other projects, like moving a character in a game or adjusting where a picture shows up on a screen.

## 1.1 Project Deliverables

This project will be able to:

1. Read a pulse signal from an IR receiver and decode it
2. Use the information from the signal to determine which button was pressed
3. Move a sprite in the direction the button corresponds to, if the button corresponds to one

## 2 IR TV Remote Description

The RC-5 IR Transmission Protocol was developed by the manufacturer Phillips and is often used for projects, due to the TV remote controls being cheap and compatible with many device types [1]. RC-5 uses bi-phase coding, where a logic “0” is represented by a half-bit pulse burst and a half-bit amount of space, and a logic “1” is represented by the opposite. Figure 2 below shows the bi-phase pulses.



Figure 2: Pulses for Logic "0" and "1" (SB-Projects ©2001)

There are 14 bits in a complete RC-5 signal. The first two bits are start pulses, which always have the value of “1”. It is important to note that the IR receiver won’t recognize there’s a signal being sent until halfway through the first bit. The next bit is the toggle, which inverts every time a key is released. The next five bits are the address, which starts with the most significant bit. The last six bits are the command, which also start with the most significant bit. This data will be repeated as long as the key is pressed. Figure 3 below shows the complete signal for command 35 going to address 05.



Figure 3: Signal for command 35 (SB-Projects ©2001)

### 2.1 Timing and Power

Each bit has a constant time of 1.778ms, which means there are  $889\mu s$  per half bit. The entire signal takes about 25ms to complete, and there are 64 bits passed in a carrier frequency of 36kHz. Figure 4 shows the timing for the same signal described above.



Figure 4: Timing for IR Signal (Altium ©2021)

In terms of power, the pulse-to-pulse ratio of the carrier frequency is 1/3 to 1/4, which reduces power consumption [2]. One of the largest consumers of power in a circuit is the IR transmitter, so reducing the power consumption is crucial [3].



Figure 5: Top level Diagram

| Description     | FPGA Pin |
|-----------------|----------|
| CLK (50M Hz)    | P11      |
| CLK (10M Hz)    | N5       |
| Reset           | B8       |
| IR Senor Signal | AB5      |

### 3 High Level Description

#### 3.0.1 Top Level Simulation

From here, each functional unit of this design will be broken down into its components, and will have a description for what each piece does. If an individual block is more than one functional unit, it will only be described once.

#### 3.1 IR Receiver

Here is the schematic diagram for the IR Receiver:



Figure 6: IR Receiver Schematic Diagram

### 3.1.1 IR Receiver Simulation

Below is the simulation for the IR Receiver.



Figure 7: IR Receiver Simulation

Inputs: This reads a IR signal.

Outputs: This outputs a 2-bit binary number that relates to the input.

Description: The input for the IR comes from the from the Ardinuo circuit pin. It counts how long a value has been set to high or low, and stores a value into a register to be outputted when the count of values is 2.

## 3.2 IR Decoder

Here is a schematic diagram for the IR Decoder:



Figure 8: IR Decoder Schematic Diagram

### 3.2.1 First Multiplexer



Figure 9: IR Multiplexer 1 Block Diagram



Figure 10: IR Multiplexer 1 simulation

Inputs: This reads a 2-bit number.

Outputs: This outputs a 2-bit binary number that relates to the input.

Description: The input currentBit comes from the IR Receiver output irValue, and the only inputs that are accepted are a 01 (which passes a HIGH value) or a 10 (which passes a LOW value). All other inputs will pass a value of 00, which will do nothing. If passing a HIGH value, we pass 11, and if LOW, we pass 10.

### 3.2.2 Second Multiplexer



Figure 11: Second Multiplexer Block Diagram



Figure 12: Second Multiplexer Simulation

Inputs: currentValue, clk

Outputs: actualValue, validCount

Description: This module depends on the inputs from our first multiplexer - if I get a 11, I know that I have a 1 - a 10 then a 0 - so I pass that out in actualValue. Additionally, if I get a valid input, I increment my count.

### 3.2.3 Comparator



Figure 13: Comparator Block Diagram



Figure 14: Comparator simulation

Inputs: count

Outputs: countEqualToM

Description: Basic comparator - if my count is equal to my parameter, send a pulse. This pulse lets my values through my tristate and resets my counter. Additionally, because I know that any actual inputs will give me 14 bits, but I'm going to get them in clock cycles of two, so when I get to my 28th increment I can reset my counter and enable my tristate to let my values through because I know for sure my shift register is full.

### 3.2.4 Shift Register



Figure 15: Shift Register Block Diagram



Figure 16: Shift Register simulation

Inputs: clk, reset, sin

Outputs: q, sout

Description: This is a shift register - at every cycle it accepts a new value in and increases all other values' positions in line. For our clock we're using the valid input count from our second multiplexer - that way when we know we got a valid input, we can store it in our shift register and not store noise in the meantime.

### 3.2.5 Address Swapper



Figure 17: Address Swapper Block Diagram



Figure 18: AddressSwapper simulation

Inputs: unadulteratedInput

Outputs: command, address, startValues

Description: This is an ugly (but guaranteed correct) way of filtering out the values that we need from our shift register. After we've gotten all 14 bits that we need and the tristate passes them through, the address swapper tells us what bits are what and passes them out of the module.

### 3.2.6 Tristate



Figure 19: Tristate Block Diagram



Figure 20: IR Tristate simulation

Inputs: a, enable

Outputs: y

Description: Just a filter for our Shift Register. It makes it so that we don't constantly send nonsense - only when we know that our register is filled with values.

### 3.2.7 IR Decoder Simulation

Below is the simulation for the IR Decoder.



Figure 21: IR Decoder Simulation

### 3.3 VGA Display Output

#### 3.3.1 Top Level Diagram



Figure 22: VGA Block Diagram

#### 3.3.2 Counter

Inputs: Clock, en, reset, d

Outputs: q

Description: A basic counter - we're using it as a clock divider here in order to get to 25 mHz.

#### 3.3.3 Sync Count



Figure 25: Sync Count Block Diagram



Figure 23: VGA simulation



Figure 24: VGACounter simulation

Inputs: Clock, en, reset

Outputs: horizontalPulseResult (hsync), verticalPulseResult(vsync), startDisplayTiming, endDisplayTiming, pixelLocation, rowLocation

Description: The idea here is to increment our current pixel until we hit the max length of our row - then we reset our current pixel and increment our current row by one. If our horizontal value is within our hsync bound, it's then true, and the same is true with our vertical value. In addition, our display timings tell us when we are actually "in the screen" and able to write pixels to the screen.



Figure 26: Sync Count simulation

### 3.3.4 Color Mux



Figure 27: Color Multiplexer Block Diagram



Figure 28: VGAColorMux simulation

Inputs: Color, Ground, inSync

Outputs: Decision

Description: This is a basic mux - the only notable thing about it is that our inSync value is provided to us from Sync Count and an and gate. If both Vsync and Hsync are high, then our color is let through. Otherwise, we get routed to ground.

## 3.4 Sprite VGA

### 3.4.1 Top Level Diagram



Figure 29: Sprite with movement and VGA



Figure 30: Simulation of a correct input and the values we get from it

Inputs: Signal, Clock, Reset, backgroundColor

Outputs: Hsync, Vsync, VGA RED, VGA BLUE, VGA GREEN, Address, Start

Description: This module takes in two bits representing the value that we got from IR receiver. It then passes it through our IR Decoder in order to find what our command, address, and start values are. Currently start values and address are disregarded, but they're saved and we could do verification with them if wanted. Once the command is received, it goes through the modifierAmount block which determines how much to move our image bounds by. We then pass the modified amount to the comparators that determine whether the sprite is being drawn or not. By shifting the bounds of where the comparators say to draw, we can give the sprite the appearance of movement. When enabled, the displayMux allows the spriteColors stored in our ROM through - otherwise it passes the background colors, those colors are then passed to the VGA for it to determine whether we're in the display bounds or not.

### 3.4.2 Address Modifier



Figure 31: Address Modifier Simulation

Inputs: Command, Clock, Reset, en(able)

Outputs: horizontalAmount, verticalAmount

Description: Address Modifier takes in a 6 bit command received from the decoder and matches it against specific values for up, down, left, and right. The module restricts movement outside of the bounds of our display by checking whether the new location is in bounds before assigning it - if we get a nonsense command or something that moves us out of bounds, we simply ignore it. The output increments and decrements in tens, and won't go above 640x480 or below 0x0 (the start value).

### 3.4.3 Sprite Comparator

Inputs: A(our coordinate), Modifier

Outputs: UpperBound, LowerBound

Description: The Sprite Comparator is simple - it just keeps track of both lower and upper bounds so that we can increment or decrement either. The modifier has already been checked by the previous module to make sure we can't move out of bounds, so we're able to just assign our outputs and not worry about error checking in this module.



Figure 32: Comparator Simulation

### 3.4.4 Address Converter



Figure 33: Address Converter Simulation

Inputs: horizontalInput, verticalInput

Outputs: address

Description: This is a simple module - we take the vertical input, multiply it by the width of the row, and add our horizontal input. This process gives us the address of any given x and y combination.

### 3.4.5 Sprite ROM

Inputs: Address, Clock

Outputs: spriteColors

Description: This used Quartus's built in IP stuff to make a ROM our size. The basic idea is that we now take in the address and read whatever is at that line out into our spriteColors.



Figure 34: Color Mux Simulation

### 3.4.6 Display Mux

Inputs: spriteColors, background, enable

Outputs: Decision

Description: This module takes in both the color options that could be displayed and, depending on the enable, only allows one to break through. The enable is based on a 4-wide and gate with all of the bounds from our comparators. If we're inside of both our horizontal and vertical bounds, then the spriteColors pass through - otherwise the background does.

### 3.4.7 VGA Display

Inputs: reset, clock, pixelInputNoSwitches

Outputs: Hsync, Vsync, VGA R, VGA G, VGA B, horizontalCoordinate, verticalCoordinate

Description: Functions as described earlier - This takes in 25 mHz clock pulses and counts pixels. The Hsync and Vsync tell the television whether it's within the necessary bounds or not, and the VGA series tell the television what colors to write. The coordinates are also how far into the actual display we are - not just from the start of the clock cycles. It only starts incrementing when display is valid, and that's how we can start displaying our sprite at (0,0) to begin with. For a simulation, see Figure 14.

### 3.4.8 VGA Display Output Simulation

# A SystemVerilog Files

## A.1 IR Receiver

### A.1.1 Comparator

```
1 //IR Receiver Comparator
2 module comparator2 #(parameter N = 10, M = 784, O = 143)
3                                     (input logic [N-1:0] a,
4                                      output logic altb, agtc);
5
6     reg [N-1:0] b = M;
7     reg [N-1:0] c = O;
8     assign altb = (a < b);
9     assign agtc = (a > c);
10 endmodule
```

### A.1.2 Clock Counter

```
1 //Counter
2 module counter1 #(parameter N = 10) //25 because  $2^{10} = 10$ 
3                                     (input logic clk,
4                                      input logic reset,
5                                      input logic [N-1:0] d, //stores data
6                                      output logic [N-1:0] q);
7
8     always_ff@(posedge clk, posedge reset) //on the rising edge of the clock and reset
9         if (reset) q <= 0; //reset to zero
10        else q <= d + 1; //iterate up by one
11 endmodule
```

### A.1.3 Enable Counter

```
1 //IR Receiver Counter
2 module NCounter_enable #(parameter N = 8)
3                                     (input logic clk, input logic reset,
4                                      input logic en,
5                                      output logic [N-1:0] q);
6
7     always_ff @(posedge clk, posedge reset)
8         if(reset)
9             q <= 0;
10        else if(en)
11            q <= (q + 1);
12        else
13            q <= 4'b0;
14 endmodule
```

### A.1.4 Multiplexer

```
1 //IR Receiver Mux
2 module ir_mux3 (input logic a, b,
3                  input logic clk, res,
4                  output logic actual_value, register_clk,
5                  output logic [3:0] valid_count);
6
7     always_ff @(posedge clk or posedge res)
8         if(res)
9             valid_count <= 0;
10        else if(a)
11            begin
12                actual_value = 1;
13                valid_count <= valid_count + 1;
14                register_clk = 1;
15            end
16        else if(b)
17            begin
18                actual_value = 0;
19                valid_count <= valid_count + 1;
20                register_clk = 1;
21            end
22        else register_clk = 0;
23 endmodule
```

### A.1.5 Shift Register

```
1 //IR Receiver Shift Register
2 module Nshift_register #(parameter N = 8)
3                                     (input logic clk,
4                                      input logic reset, sin,
5                                      output logic [N-1:0] q,
6                                      output logic sout);
7
8     always_ff @(posedge clk, posedge reset)
9         if(reset)
10            q <= 0;
11        else
12            q <= {q[N-2:0], sin};
13     assign sout = q[N-1];
14 endmodule
```

### A.1.6 Tristate

```
1 //IR Receiver Tristate
2 module irv_tristate #(parameter N = 8)
3                                     (input logic [N-1:0] a,
4                                      input logic enable,
5                                      output logic [N-1:0] y);
6
7     reg b = (N-1);
8
9     always_comb
```

```

10         if(enable)
11             y = a;
12         else
13             y = 4'b0;
14 endmodule

```

### A.1.7 Synchronizer

```

1 //IR Receiver Synchronizer
2 module sync (input logic clk,
3                 input logic d,
4                 output logic q);
5
6     logic n1;
7
8     always_ff@(posedge clk) //flip flop at rise of clock
9         begin
10            n1 <= d;
11            q <= n1;
12        end
13 endmodule

```

## A.2 IR Decoder

### A.2.1 Comparator

```

1 //IR Decoder Comparator
2 module irComparator #(parameter N = 4, M = 14)
3                                         (input logic [N-1:0] count,
4                                         output logic countEqualToM);
5
6     assign countEqualToM = (count == M);
7 endmodule

```

### A.2.2 First Multiplexer

```

1 //IR Decoder Mux
2 module irMux(input logic [1:0] currentBit,
3                 output logic [1:0] value);
4
5     //input 2 bit value. All values ignored except 01 high and 10 low
6     always_comb
7         case(currentBit)
8             1:                               value = 11;
9             2:                               value = 10;
10            default value = 00;
11        endcase
12 endmodule

```

### A.2.3 Second Multiplexer

```

1 //IR Decoder Second Mux
2 module irSecondMux(input logic [1:0] currentValue,
3                           input logic clk,
4                           output logic actualValue,
5                           output logic [4:0] validCount);
6
7     always_ff @(posedge clk)
8         case(currentValue)
9             3:
10                begin
11                    actualValue = 1;
12                    validCount <= validCount + 1;
13                end
14
15                2:
16                    begin
17                        actualValue = 1;
18                        validCount <= validCount + 1;
19                    end
20
21                0: validCount <= 0;
22        endcase
23 endmodule

```

### A.2.4 Shift Register

```

1 //IR Decoder Shift Register
2 module shiftRegister #(parameter N = 14)
3                                         (input logic clk,
4                                         input logic reset,
5                                         input logic sin,
6                                         output logic [N-1:0] q,
7                                         output logic sout);
8
9     always_ff @(posedge clk, posedge reset)
10        if(reset) //if reset = 1
11            q <= 0;
12        else //if reset = 0
13            q <= {q[N-2:0], sin};
14
15        assign sout = q[N-1];
16 endmodule

```

### A.2.5 Address Swapper

```

1 module addressSwapper(input [13:0] unadulteratedInput,
2                         output [5:0] command,
3                                         output [4:0] address,
4                                         output [2:0] startValues);
5
6     assign command[5] = unadulteratedInput[5];
6     assign command[4] = unadulteratedInput[4];

```

```

7 assign command[3] = unadulteratedInput[3];
8 assign command[2] = unadulteratedInput[2];
9 assign command[1] = unadulteratedInput[1];
10 assign command[0] = unadulteratedInput[0];
11
12 assign address[4] = unadulteratedInput[10];
13 assign address[3] = unadulteratedInput[9];
14 assign address[2] = unadulteratedInput[8];
15 assign address[1] = unadulteratedInput[7];
16 assign address[0] = unadulteratedInput[6];
17
18 assign startValues[0] = unadulteratedInput[11];
19 assign startValues[1] = unadulteratedInput[12];
20 assign startValues[2] = unadulteratedInput[13];
21
22
23
24 endmodule
25 //input 110 11111 100011
26 //I get my values one at a time ->
27 //1
28 //-->1
29 //11
30 //-->0
31 //011
32 //-->1
33 //1011
34 //backwards
35 //so, input[0] = shiftRegister[13]

```

## A.2.6 Tristate

```

1 //IR Decoder Tristate
2 module tristate #(parameter N = 8)
3   (input logic [N-1:0] a,
4    input logic enable,
5    output logic [N-1:0] y);
6
7   //if enabled, input goes through to output. Otherwise, 0
8   assign y = enable ? a : 0;
9 endmodule

```

## A.3 VGA Display Output

### A.3.1 Counter

```

1 module counter #(parameter N = 6)
2   (input logic clk,
3    input logic reset,
4    input logic en,
5    input logic [N-1:0] d,
6    output logic [N-1:0] q);
7   always_ff @(posedge clk, posedge reset)
8     if (reset) q <= 0;
9     else if (en) q <= d + 1;
10 endmodule

```

### A.3.2 Sync Count

```

1 module syncCount #(parameter horizontalLow = 95, horizontalEnd = 799, verticalLow = 1, verticalEnd =
2   524,
3   horizontalTimingStart = 143, horizontalTimingEnd = 784,
4   verticalTimingStart = 34,
5   verticalTimingEnd = 514)
6   (input logic clk,
7    input logic en,
8    input logic reset,
9    output logic horizontalPulseResult,
10   output logic verticalPulseResult,
11   output logic startDisplayTiming,
12   output logic endDisplayTiming,
13   output logic [9:0] pixelLocation,
14   output logic [9:0] rowLocation);
15 logic [9:0] currentPixel = 10'b0000000000;
16 logic [9:0] currentRow = 10'b0000000000;
17 always_ff @(posedge clk)
18 begin
19   if (reset)
20     begin
21       pixelLocation <= 10'b0000000000;
22       rowLocation <= 10'b0000000000;
23       currentPixel <= 10'b0000000000;
24       currentRow <= 10'b0000000000;
25     end
26   else if (en)
27     begin
28       //if I'm below my max, add one
29       if (currentPixel < horizontalEnd)
30         begin
31           currentPixel <= currentPixel + 1;
32         end
33       //if my horizontal pixel is at the max
34       else if (currentPixel >= horizontalEnd)
35         begin
36           //and if my vertical row is at the max, reset my row
37           if (currentRow >= verticalEnd)
38             begin
39               currentRow <= 10'b0000000000;
40             end
41           //otherwise, add one to your row.
42           else if (currentRow < verticalEnd)
43             begin
44               currentRow <= currentRow + 1;
45             end
46           //and then reset your horizontal pixel no matter what.
47           currentPixel <= 10'b0000000000;
48         end

```

```

49         pixelLocation <= currentPixel ;
50         rowLocation <= currentRow ;
51     end
52
53     assign horizontalPulseResult = !(currentPixel <= horizontalLow) ;
54     assign verticalPulseResult = !(currentRow <= verticalLow) ;
55     assign startDisplayTiming = ((currentPixel >= horizontalTimingStart) && (currentPixel <
56                                 horizontalTimingEnd)) ;
57     assign endDisplayTiming = (currentRow >= verticalTimingStart) && (currentRow < verticalTimingEnd) ;
58 endmodule

```

### A.3.3 Color Mux

```

1 module colorMux(input logic [3:0] color ,
2                   input logic [3:0] ground ,
3                               input logic inSync ,
4                               output logic [3:0] decision );
5   assign decision = inSync ? color : ground ;
6 endmodule

```

## A.4 Sprite VGA

### A.4.1 Address Modifier

```

1 //Take in my command and determine what I'm adding to the comparators .
2 //((width * row) + height) = address
3 //((address - height)/width = row
4 //height = address - width * row;
5 module modifierAmount #(parameter horizontalEnd = 640, verticalEnd = 500)
6   (input logic [5:0] command,
7    input logic clock ,
8                               input logic reset ,
9                               input logic en ,
10                              output logic [9:0] horizontalAmount ,
11                              output logic [9:0] verticalAmount );
12
13 always_ff @(posedge clock , posedge reset )
14   begin
15     verticalAmount <= 6'b000000 ;
16     horizontalAmount <= 6'b000000 ;
17   end
18   else if (en)
19     case (command)
20       /Down
21       6'b110011 :
22       if (verticalAmount < (verticalEnd - 'd10))
23         verticalAmount <= (verticalAmount + 'd10) ;
24       /Up
25       6'b001100 :
26       if (verticalAmount >= 10)
27         verticalAmount <= (verticalAmount - 'd10) ;
28       /Left
29       6'b111000 :
30       if (horizontalAmount >= 10)
31         horizontalAmount <= (horizontalAmount - 'd10) ;
32       /Right
33       6'b000111 :
34       if (horizontalAmount < (horizontalEnd - 'd10))
35         horizontalAmount <= (horizontalAmount + 'd10) ;
36     endcase
37   end
38 endmodule

```

### A.4.2 Sprite Comparator

```

1 module comparator #(parameter N = 10, M = 256)
2   (input logic [N-1:0] a ,
3                               input logic [N-1:0] modifier ,
4                               output logic upperBound ,
5                               output logic lowerBound );
6   assign lowerBound = (a > (0 + modifier)) ;
7   assign upperBound = ((a + modifier) < M) ;
8 endmodule

```

### A.4.3 Address Converter

```

1 module addressConverter #(parameter width = 256)
2   (input logic [9:0] horizontalInput ,
3                               input logic [9:0] verticalInput ,
4                               output logic [15:0] address );
5   assign address = ((verticalInput * width) + horizontalInput) ;
6 endmodule

```

### A.4.4 Display Mux

```

1 module displayMux #(parameter N = 12)
2   (input logic [N-1:0] spriteColors ,
3                               input logic [N-1:0] background ,
4                               input logic enable ,
5                               output logic [N-1:0] decision );
6
7   assign decision = (enable ? spriteColors : background) ;
8 endmodule

```

## B Do files

### B.0.1 RC-5 Encoder

```
1 vsim -gui rc5_irdecoder
2 add wave *
3 force -freeze sim:/rc5_irdecoder/Clock10_a 1 0, 0 {50000 ps} -r 100000 ps
4 force Reset_a 0 0, 1 10
5 run 20 ms
6 force input_ir 0 0, 1 .889 ms
7 run 1.778 ms
8 force input_ir 1 0, 0 .889 ms
9 run 1.778 ms
10 force input_ir 0 0, 1 .889 ms
11 run 1.778 ms
12 force input_ir 1 0, 0 .889 ms
13 run 12 ms
```

### B.0.2 IR Decoder

```
1 add wave *
2 force -freeze sim:/spriteVGA/Reset 1 0
3 run 10
4 force -freeze sim:/spriteVGA/Reset 0 0
5 force -freeze sim:/spriteVGA/backgroundColor 000000000000 0
6 force -freeze sim:/spriteVGA/spriteColorWire 111111111111 0
7 force -freeze sim:/spriteVGA/Clock 1 0, 0 {50 ps} -r 100
8 force -freeze sim:/spriteVGA/signal 01'0
9 run 400
10 force -freeze sim:/spriteVGA/signal 10 0
11 run 600
12 force -freeze sim:/spriteVGA/signal 01 0
13 run 600
14 force -freeze sim:/spriteVGA/signal 10 0
15 run 600
16 force -freeze sim:/spriteVGA/signal 01 0
17 run 600
```

## References

- [1] S. Bergmans, “Phillips RC-5 Protocol,” published 1, Aug. 2020. [Online]. Available: <https://www.sbprojects.net/knowledge/ir/rc5.php>
- [2] Altium, “RC-5 Transmission Protocol,” published 13 Sept. 2017. [Online]. Available: <https://techdocs.altium.com/display/FPGA/Philips+RC5+Infrared+Transmission+Protocol>
- [3] Military Analysis Network, “Basic Radar Systems,” accessed 3, June 2021. [Online]. Available: <https://fas.org/man/dod-101/navy/docs/es310/radarsys/radarsys.htm>