

# CSE 100/L Logic Design

## Fall 2025

### Lab Assignment 6

#### **Pre-lab deadline:**

- There is no pre-lab assignment for this lab.

#### **Code Submission deadlines:**

- 100% Before the end of your section Tuesday, November 25th.
- 95% after the end of your section Tuesday, November 25th.
- 80% after 11:59PM Tuesday, November 25th.
- 60% after 11:59PM Sunday, November 30th.
- 40% after 11:59PM Monday, December 1st.
- 20% after 11:59PM Tuesday, December 2nd.
- 0% after 11:59PM Wednesday, December 3rd.

#### **Demo deadline:**

- Code may be demonstrated until the end of the next lab assignment. You must submit before you demo, and the code you submit must be the code you demo. You may be required to download the code from Canvas to demo.

#### **Write-Up deadline:**

- Wednesday, December 3rd by 11:59PM. No late exceptions.

#### **BTTF Token Opportunities:**

There are two opportunities to earn a BTTF token, described at the end of the document. You must **demo** the milestone by the deadline shown.

- Milestone 1: Demo the first Milestone, before the end of the day on Thursday, November 13th.
- Milestone 2: Demo the second Milestone, before the end of the day on Thursday, November 20th.

#### **Extra Credit Opportunities:**

There are two extra credit opportunities. They are described at the end of this document.

#### **Academic Integrity Policy:**

The Academic Integrity Policy is stated in the syllabus on Canvas. Students are responsible for reading the syllabus. Not reading the syllabus is not an excuse for academic misconduct.

- Work submitted in CSE 100 must be yours alone. Sharing/sending code, submitting code written by others, and submitting (any) code from generative AI **are all violations of academic integrity**.
- Working together on a whiteboard, on paper, debugging waveforms, explaining concepts and/or examples, is encouraged, **as long as no solution code is shared**.

**Verilog Operator and Procedural Block Constraints:**

In this lab you can use any of the following combinational operators:

- All combinational logic must be implemented using only `assign` statements.
- Bit-wise Operators: `&`, `|`, `~`, `^`, and `~^`
- Concatenation and Replication: `{}` and `{|}`
- **Logical (New!)** `&&`, `||`, and `!`
- **Relational (New!)**: `<`, `>`, `>=`, `<=`, `==`, and `!=` (Caution! All operators yield a 1-bit result.)
- **Arithmetic (New!)**: `+` and `-` (Caution! Two N-bit values yield a N+1-bit result.)

Your designs must be **synchronous with the system clock** specified in the lab. This means:

- use only positive edge-triggered flip-flops (FDRE),
- not use asynchronous clears or pre-sets of any sequential elements,
- connect only **the system clock** as input to the clock pins of any sequential components,
- not connect **the system clock** as the input to any other logic.
- you may **not** use any procedural blocks in your design (i.e. `always@`, `always_ff@`, `always_comb`).

You may only use `assign` statements and FDREs in your design. Use of disallowed operators or procedural blocks will be considered behavioral and will result in a **score of 0!**

## Overview

---

You will design and implement a version of the game "Subway Slugging" using the BASYS3 board and the VGA monitor connected to it. In this game, a slug must avoid trains traveling down tracks by jumping between the three tracks or hovering above the middle track. A collision with a train ends the game and the number of trains the slug has avoided is the score. The slug can only hover for a limited time. Once the slug's energy level, displayed on the left, reaches 0 the slug will drop.

Initially, the slug and the energy level bar are displayed, but no trains are present. The score, initially 0, is displayed on AN1 and AN0. Pressing `btnC` starts the game. No other pushbutton has an effect before the game starts (except the global reset, `btnD`, of course). When the game starts the trains will start traveling down the tracks and pressing `btnL`/`btnR` slides the slug one track to the left/right. If the energy level is not 0 and the slug is in the middle, then the slug will hover while `btnU` is pressed. But hovering will use up energy. The slug will drop as soon as `btnU` is released or the energy level is down to 0. The slug should change color and flash while it is hovering. While the slug is not hovering the energy level increases. Each time a train passes the slug, a point is scored. A crash occurs if the slug is not hovering and overlaps a train. A crash ends the game, and when the game is over, the slug and all trains stop moving, and the slug flashes. Only the global reset, `btnD`, will have an effect when the game is over. Switch `sw[3]` will be a cheat switch that makes the slug immortal: it can go through trains without crashing.

There is a demo bitstream, `example.lab6.bit`, available in Canvas. This provides the baseline for what 100% functionality is for this lab.

### VGA controller

Read the section on the **VGA Port** in the [BASYS3 Board Reference Manual](#). **Do not use the circuit design shown there: it is asynchronous.**

To control the monitor you must generate two control signals, `Hsync` and `Vsync`, as well as the 12 RGB data signals (`vgaRed[3:0]`, `vgaBlue[3:0]`, and `vgaGreen[3:0]`) for each of the screen's 640 x 480 pixels. The value of these 12 signals are sent one at a time for each pixel, row by row from left to right and top to bottom using one cycle of the 25MHz clock (provided to you) for each pixel.

There is also some time between rows and between frames (after all 480 rows) which allows the cathode ray to be re-positioned for the next row or frame. The `Hsync` and `Vsync` signals are used by the monitor to "synchronize" the start of each row and frame; they are low at fixed times between rows and frames. (*Yes the monitors we will use are lcd displays, not cathode ray tubes. But the protocol used to communicate with these monitors is a standard that lives on.*)

One way to think of this is to imagine that you have an 800 x 525 grid of pixels as shown below (instead of the 640 x 480 pixels which correspond to the area you see on the monitor). This grid is traversed starting at the top left, location row 0, column 0. Each row is traversed from left to right followed by the row immediately below it and so on. The region of dimension 640 x 480 at the top left is the **Active Region**: the pixels in this region corresponds to pixels on the screen. The pixels outside this region correspond to time where the cathode ray would be off the screen. So the L-shaped region outside the active region is not part of the screen but represents the time needed between rows and frames in terms of pixels.



Figure 1: 800x525 VGA pixel grid.

The value of the RGB data signals determine the color displayed for pixels in the Active Region, with one cycle of the 25MHz clock corresponding to a pixel. For a pixel outside the Active Region the 12 RGB data signals should be low. The horizontal synchronization signal (**Hsync**) should be low **exactly** for the 96 pixels in each row starting with the 656th pixel and high for the rest. The vertical synchronization signal (**Vsync**) should be low **exactly** for all of the pixels in the 490th and 491st rows and high for all pixels in all other rows. So Hsync and Vsync are low in only the regions shaded pink and blue below, respectively. The frame is continuously transmitted to the monitor to refresh the image. Transmitting one frame takes  $800 \times 525 \times 40\text{ns} = 16,800,000\text{ns} = 16.8\text{ms}$ , so the monitor is being refreshed roughly 60 times per second: at 60Hz.

## Game Requirements

Your implementation of Subway Slugging must meet each of the following requirements.

1. The playing area is the 640 x 480 screen.
2. The border along all 4 edges of the screen is white and 8 pixels wide.
3. The green bar indicating the slug's energy level is near the left border. It is 20 pixels wide and 192 pixels long at the maximum energy level.
4. There are three vertical tracks that are each 60 pixels wide. They are not visible unless you choose to display them. But the trains that will descend along these tracks will be visible.
5. The tracks should be positioned in the right 2/3's of the screen with 10 pixels between adjacent tracks.
6. The slug is a 16 by 16 pixel yellow square (or a shape that fits in this square and touches all 4 sides of the square). The top edge of the slug is in row 360 and the slug is initially in the middle of the middle track.
7. The slug cannot move or change until the game starts (**btnC** is pressed).
8. Pressing **btnC** will start the game, but once the game is started **btnC** has no further effect.
9. During the game, the slug is either at rest, centered on one of the three tracks, or transitioning between two adjacent tracks.
10. Pressing **btnR** starts the transition to the adjacent track on the right (unless the slug is on the rightmost track).
11. Pressing **btnL** starts the transition to the adjacent track on the left (unless the slug is on the leftmost track).
12. The slug will move horizontally at 2 pixels per frame while it is transitioning.

13. Once a transition has started it will continue unless the game ends (a crash occurs).
14. Pressing **btnL** or **btnR** during a transition or while the slug is hovering has no effect.
15. Holding **btnU** down while the slug is centered on the middle track and its energy level is not 0 will cause the slug to hover.
16. The slug changes color while it is hovering.
17. When **btnU** is released or the slug's energy level reaches 0, the slug stops hovering.
18. While the slug is hovering its energy decreases level by 1 every frame down to 0.
19. While the slug is not hovering its energy level increases by 1 every frame up to a maximum of 192.
20. Each track will have two trains.
21. Pressing **btnC** opens the three tracks in the following order: the left track is first, followed by the right track 2 seconds later, and finally the middle track 6 seconds after the right track.
22. When a track is opened, its first train with start.
23. When a train starts, it will pick a random length, wait a random amount of time, and then descend at one pixel per frame until it is completely off the screen.
24. Trains are 60 pixels wide.
25. When the bottom of either train reaches row 400 (440 for the middle track) the other train will start in the same manner: picking a random length and waiting a random time before descending.
26. The two trains continually move down their track, restarting each other until the game ends.
27. When the top of the train reaches the row below the slug, a point is scored.
28. The length of the train is  $60 + B$  pixels where  $B$  is a randomly selected number between 0 and 63.
29. The random amount of wait time before descending is  $T$  frames where  $T$  is a randomly selected number between 0 and 127.
30. A crash occurs whenever the slug is not hovering and it overlaps any train by at least 1 pixel.
31. If a crash occurs, the game is over: the trains stop moving, the slug stops moving and flashes, and only **btnD** can have an effect.
32. When the game is over, pressing **btnD** will start a new game.

## Module Overview



Figure 2: Block diagram of the Lab 6 design.

- A new module generates the V/H coordinates (i.e. pixel address) of the current pixel (but likely will be composed of pre-existing modules)
- A new module is needed to generate the V/Hsync signals.
- A new module is needed to produce the 16-bit RGB pixel data from the user input, the pushbuttons/switches, and the pixel address and manage gameplay. This may contain multiple modules itself, and may have similar components to Lab 4/5. You can reuse code from prior labs, but it must be your code.
- The module **labVGA\_clks** is provided.
- The RingCounter/Selector/hex7seg logic may be reused from prior labs, but it must be your own code.
  - While only two 7-segment displays are required, you may find it useful to allow other information from your design to be displayed on the remaining two.

## Recommended Design Procedure

1. Start now. You will not be able to complete this lab in just a few days. It will take several days to debug and you will need to download and see your design's output on the monitor to debug it.
2. Download this Verilog file and save it as labVGA\_clks.v. In the Vivado Project Manager, add it to your project. Make sure you select the option to copy it into your project.
3. Add an instance of the module **labVGA\_clks** to your top level as follows:

```
labVGA_clks not_so_slow(
    .clkin(clkin),
    .greset(btnR),
    .clk(clk),
    .digsel(digsel));
```

Note that btnD will be used for the global reset, since btnR has another purpose in this lab. The signal clk is your system clock. The signal digsel should be used to advance the Ring Counter for the 7-segment displays; it should not be used as a clock!!!

When you simulate with labVGA\_clks, the clk signal will have the same timing as on the board: it is a 25MHz clock. Only the signal digsel will be different.

4. Map out your components (figure out their inputs/outputs and what they do) but don't implement them yet.
5. Implement your Hsync/Vsync signals and your pixel address counters, you should simulate your design using **test\_lab6\_syncs.v**, found in Canvas. Follow the instructions in the comments at the top. This testbench will check that your Hsync/Vsync signals are correct and that your RGB outputs are low outside the active region. You should do this before adding too much other logic to your design since it will take a while just to simulate one frame even for a simple design in which the screen is just one solid color.
6. **Incremental design is highly recommended.** This means implementing and testing your design in phases. We recommend the following:
  - (a) Implement a VGA controller that outputs the Hsync and Vsync signals, and provides the pixel address and indicator for the active region while displaying a single color. (You should test your design using the testbench mentioned in the previous step at this point rather than continuing.)

- (b) Add the borders to the display.
  - (c) Display the player.
  - (d) Display the energy level bar.
  - (e) Make the energy level bar decrease/increase based on whether `btnU` is pressed.
  - (f) Next you can make the slug respond to `btnL`, `btnR`, and `btnU`. You will need a state machine to make it transition, and hover only when it is on the middle track. You'll need to coordinate the hovering and the energy level bar. (**Getting to this point is sufficient to earn Milestone 1 BTTF**).
  - (g) Next you should work on one track. Please make a separate module for a track. You can start by testing just one and when it is working perfectly, then and only then you should instantiate the other 2 (with different inputs). But first get one track working perfectly before even thinking about other two. This is the most complicated component. (**Getting to this point is sufficient to earn Milestone 2 BTTF**).
  - (h) The last piece is to instantiate the other two tracks, and provide a top level state machine to start the tracks, the slug, and end the game when there is a crash. You'll need a timer for the staggered track opening and to make things flash.
7. To move objects at one pixel per frame and count once per frame you will need a signal frame that is high for one clock cycle once per frame. A simple way to generate this signal is to edge detect the `Vsync` signal since it has a single low pulse in each frame. Another way is to make a signal that is high at one specific pixel address. Which ever way you choose, it's important that this signal not be high in the active region to avoid updating the position of an object while it is being displayed.
  8. There is no `qsec` signal provided for this lab. Do not try to use the `qsec_clks` module. Instead the `frame` signal mentioned above, which is high for one cycle per frame, can be used as the up (or down) input of a counter.
  9. Since you can only simulate a few frames with the simulator, you will need feedback from the monitor to understand what is occurring. If you are not seeing anything on the monitor or the image is not stable, then you will want to check the timing of your `Hsync`, `Vsync` and `R/G/B` signals. (See step 5 above.)
  10. If you need to simulate your design to see how your objects interact you will want to generate your own `frame` signal in the test bench rather than waiting 420,000 clocks for it to go high. You may find it useful to display certain signals on the leds, or values on the 7 segment display. You can use the switches to force certain conditions.
  11. At some point you might suspect that either your BASYS3 board or the monitor is not functioning properly. To convince yourself that this is not the case, you can download the bitfile `testvgaB3.bit`, located in Canvas, to the board. This will also allow you to check that the VGA, buttons, switches, leds and 7-segment displays are working.
  12. Implement your components and state machine. You may use any component that you have built for your previous CSE100 labs. As usual, you may only use the `assign` statement. No conditionals (i.e. if-else, ternary, and case statements). **But in this lab you may use arithmetic and comparisons within assign statements.** Be careful. Ensure the bit-widths of all operands are identical. Hard to detect bugs can emerge if precision and sign issues arrise.
  13. **Demonstrate your working implementation of Subway Slugging to a TA.** The TA will confirm that
    - Your `Hsync/Vsync` signals are validated by the testbench.
    - All requirements are met (see the Game Requirements section above).

- Extra credit: if you implemented any extra credit, please inform the TA when demonstrating your design.
14. Follow the instructions for obtaining Timing Report, found in Canvas. Open the “Timing Summary Report” and use the information there to calculate the maximum frequency at which your design can run. Submit the timing report as part of your write-up.
  15. Remember to archive your project. Files left on the PC are not protected. Should it become necessary to re-image a PC, its disks will be wiped clean.
  16. **Important** Please remember to turn off the power to the BASYS3 board when you are done

## Rubric, Tokens, and Extra Credit

---

The late policy is applied on top of the rubric shown below.

- 100% - Fully Working Lab 6, exactly following the specification provided in the lab document.
- 90% - Fully working Lab 6, with some small error at the discretion of the TA.
- 80% - Functionality below, with all three trains appearing, trains are random sizes, but appear in a repeating order (not randomly).
- 70% - Functionality below, with player/train collision fully implemented and jumping implemented, only one train appears, trains do not appear randomly, and are a static size.
- 60% - Player can move, one train will appear and move, player will score when train passes, player/train collision with some error, no jumping implementation, only one train appears, trains do not appear randomly, and are a static size.

## BTTF Tokens

Reaching the following milestones on time will allow you to earn BTTF tokens. You must demo the milestone by the deadline to earn the token.

### Milestones / BTTF Tokens

1. Milestone 1 (1 token): Demonstrate your in-progress Lab 6 project
  - Running the testbench and verifying `Hsync/Vsync` for 34ms with 0 errors reported on `SERRORS` and `RGBERRORS`, and...
  - On a VGA monitor in the lab, show your FPGA design displaying the borders and the player moving.
2. Milestone 2 (1 token): Demonstrate your in-progress Lab 6 project
  - On a VGA monitor in the lab, show your FPGA design displaying the borders, player, and a moving train. The player must properly collide with the train as well as be able to avoid it.

## Extra Credit

You may complete one extra credit opportunity for a maximum of 30 extra credit points. Design extra credit **can** be used to increase the Canvas lab group score above 100%.

1. (10 point) In addition to satisfying all lab requirements described in the lab manual, add at least two of the following graphical improvements to the game: The player looks like a slug, train tracks are visible for each train, and/or make the slug **SLOWLY** flash different colors when hovering.
2. (20 points) In addition to satisfying all lab requirements described in the lab manual, add logic to your design that provides a set of lives for the player, loadable with **btnL**. They lose a life when colliding with a train. Once all lives are lost, the game enters the loss state until the game is reset with **btnD**. You may decide if the game continues when a life is lost or if the game pauses when a life is lost, then pressing **btnC** resumes game.
3. (30 points) Implement both of the above design extra credit opportunities.

## Write-Up

---

See write up tutorial for instructions on what to include in the written report and how to submit electronically. Your write-up for Lab 6 should also include:

- The major components of your design and their function. (Include a state diagram for any state machines.)
- A description of how these components interact.
- Describe each component's function and how you implemented it (if you did) or how you use it.
- The maximum clock frequency (MHz) at which your design will operate based on the Timing Report.

The following supplementary material for Lab 6 should be submitted electronically as appendices in your one PDF file:

1. One screenshot of simulation in which the Vsync (vertical synch signal) goes low. The vertical counter should be displayed so that it is clear the Vsync signal is low in the correct rows.
2. The Timing Summary Report (just the Design Timing and Clock Summary pages).
3. Scanned notebook pages for this lab.