This is an old revision of the document!


Pong: Part 2

In this two-week lab, you will use your ball and line modules to make a Pong game.

  • Exercise 1: You will use your modules from Part 1 to draw a ball and two paddles on the screen.
  • Exercise 2: Make the ball bounce around the screen. (This exercise is the largest part of the lab)
  • Exercise 3: Make the paddles move when the appropriate buttons are pressed.
  • Exercise 4: Make the ball bounce off of the paddles, and change the ball color when a player misses the ball.

Learning Outcomes

  • Learn how to implement more complex digital systems that combine several modules using state machines, sequential and combination circuitry.

Preliminary

This video shows an implementation of the Pong game for this lab. Your implementation does not need to look exactly like this. Feel free to change colors, object sizes, paddle positions, etc. Note that the video shows a wide-screen monitor, so there are “black bars” on the left and right that aren't part of the drawable space.

As you can see in the video above, the Pong game consists of drawing and moving around three objects: the ball, the left paddle and the right paddle. The animation is done by continuously drawing the objects, waiting some time, erasing them, moving them, and then repeating. Erasing is done by drawing the objects black.

State Machine

The diagram below provides the state machine that will control most of the Pong game. Rather than have separate states for drawing and erasing, the state machine is designed to use the same three states to draw the objects. Thus, the state machine will continuously repeat the state ordering (BALL→PADDLE_L→PADDLE_R→WAIT_TIMER→) to draw the objects and wait for some time, then (BALL→PADDLE_L→PADDLE_R→MOVE→) to erase the objects, and move them before repeating the same sequence again.

Some explanation of the state machine:

  • ballStart and lineStart will trigger the drawing of the ball and line objects, and are connected to the start inputs of the modules you created last lab. ballDone and lineDone are connected to the done ports of these modules.
  • initBall and initPaddles will instruct the ball and paddles locations to be set to where they should be located at the start of a game.
  • Although not shown in the state diagram, a 1-bit flip-fop named erasing will control whether the BALL, PADDLE_L and PADDLE_R states draw the objects in color or black. It is also used to determine whether the state machine should wait for move the objects after drawing them. The switchErase signal output by the state machine will cause this flip-flop to invert its value.
  • You are free to choose how long to wait in the WAIT_TIMER state. Somewhere between 0.1s and 0.001s is a good choice, depending on how fast you want the gameplay.

Given your system clock, how many cycles do you need to wait for, in order to implement a 0.01 second delay?

Module Design

Within the Pong module, the state machine is connected to a number of other components that implement the game loop timer, manage the ball location and direction, paddle locations, and score. This is shown below. These blocks are not submodules, but rather represent different sequential and combinational blocks that will make up your Pong module.

Exercises

Exercise #1 - Drawing the objects

In this exercise, you will implement just enough of the above system to draw the ball and two paddles on the screen, without any movement or user interaction.

Review the state machine and system diagrams above, and make sure you understand how the Pong game will work. In this exercise you will implement a limited set of features: simply drawing the ball and two paddles.

What to do in this exercise:

1. Create a new Vivado project, add your BallDrawer, VLineDrawer, the BitmapToVga and clk_generator modules, and the top-level module provided here: top_pong.sv

2. Inspect the top-level module, and note how the modules are connected. The module instantiates the BitmapToVga module and the glk_generator module needed to draw graphics on the VGA monitor. It also instantiates a Pong module, which controls the inputs to the BitmapToVga module. The four buttons are provided as inputs to the Pong module to control the paddle movements. Note: You shouldn't change the top-level file.

3. Add a constraints file to your project that is configured appropriately.

4. Create a Pong module and add it to your project. The Pong module is where you will add all of your code for this lab. It instantiates a copy of your BallDrawer and VLineDrawer, and will include the state machine and other module components shown in the diagram above.

Your Pong module contains the ports described here:

Module Name = Pong
Port Name Direction Width Description
clk Input 1 100 MHz Clock
reset Input 1 Active-high reset
LPaddleUp Input 1 When high, move left paddle up
LPaddleDown Input 1 When high, move left paddle down
RPaddleUp Input 1 When high, move right paddle up
RPaddleDown Input 1 When high, move right paddle down
vga_x Output 9 BitmapToVga X-Coordinate
vga_y Output 8 BitmapToVga Y-Coordinate
vga_color Output 3 BitmapToVga Color
vga_wr_en Output 1 BitmapToVga Write Enable
P1score Output 8 Player 1 score (player 1 controls the left paddle)
P2score Output 8 Player 2 score

Here is a file to get started:

Pong.sv
// Add your header here
 
`default_nettype none
 
module Pong (
    input wire logic        clk,
    input wire logic        reset,
 
    input wire logic        LPaddleUp,
    input wire logic        LPaddleDown,
    input wire logic        RPaddleUp,
    input wire logic        RPaddleDown,
 
    output logic    [8:0]   vga_x,
    output logic    [7:0]   vga_y,
    output logic    [2:0]   vga_color,
    output logic            vga_wr_en,
 
    output logic    [7:0]   P1score;
    output logic    [7:0]   P2score;
);
 
 
/////// Create a state machine here //////
 
 
////////////////// Ball Drawing /////////////////////
BallDrawer BallDrawer_inst(
    .clk(clk),
    .reset(reset),
    .start(ballStart),
    .done(ballDone),
    .x_in(ballX),
    .y_in(ballY),
    .x_out(ballDrawX),
    .y_out(ballDrawY),
    .draw(ballDrawEn)
);
 
////////////////// Line Drawing /////////////////////
VLineDrawer VLineDrawer_inst(
    .clk(clk),
    .reset(reset),
    .start(lineStart),
    .done(lineDone),
    .x_in(lineX),
    .y_in(lineY),
    .x_out(lineDrawX),
    .y_out(lineDrawY),
    .draw(lineDrawEn),
    .height(/* Todo */)
);
 
endmodule

5. Create the above state machine with just the states needed to start the game and draw the objects: INIT, BALL, PADDLE_L and PADDLE_R. For this exercise you might want to create a DONE state that the state machine stays in once it is down drawing the right paddle.

6. Implement the other components as follows:

  • Game Loop Timer: Not needed in this exercise.
  • Ball Location: You should create a new always_ff block to manage the ball location. Create two registers, BallX and BallY that are reset to the center of the screen when initBall is true. This is sufficient for this exercise.
  • Ball Direction: Not needed in this exercise. * Paddle Locations: You should create a new always_ff block to manage the paddle locations. Create two registers, LPaddleY and RPaddleY that position the paddles halfway down the screen when initPaddles is true. This is sufficient for this exercise. * Player Scores: You should create a new always_ff block to manage the player scores. For this exercise, just assign the P1score and P2score registers to 0. * Erase Switch: Not needed in this exercise. 7. Update your state machine to also output the other outputs shown in the diagram: * vga_x, vga_y, and vga_wr_en should be connected to either ballDrawX, ballDrawY, and ballDrawEn or the corresponding signals from the VLineDrawer, depending on the state. * Assign vga_color to values of your choice in each state. * Assign lineX to appropriate values of your choice in the two states that draw the paddles. Don't place the paddles right at the edge of the screen (x=0 or x=319), as sometimes the first/last few columns/rows of pixels may be cut off. You may notice that the Pong module includes one instantiation of the BallDrawer and one instantiation of the VLineDrawer. You SHOULD NOT add any more instantiations of these module. Even through you are drawing multiple paddles, drawing and erasing balls multiples times, only one instance of each module will be used. 8. Generate a bitstream and program your board. Simulate your design to debug. Exercise #1 Pass-off: Show the TA your display that draws the paddles and ball. —- ==== Exercise #2 - Bouncing Ball ==== In this exercise you will modify your Pong module to make the ball bounce around the screen. You don't need to interact with the paddles or keep score yet. Suggested approach: 1. Add the remaining two states of the state machine. 2. Update the following components: * Game Loop Timer: Implement this as a counter, similar to how you have done in previous labs. It should be reset with the timerRst signal, increment each cycle, and output a timerDone signal when it reaches a value of your choosing. * Ball Location: Add more logic to your ball location component. When in the move state (moveAndScore is high), update the ball location according to its direction. Consider the values of ballMovingRight and ballMovingDown and either add 1 or subtract 1 from ballX and ballY accordingly. * Ball Direction: You should create a new always_ff block to manage the ball directions (ballMovingRight and ballMovingDown registers). On initGame, you should reset these registers to values of your choice. Based on the ball location (ballX and ballY) you should update these registers appropriately. * Paddle Locations: Nothing to update in this exercise. * Player Scores: Nothing to update in this exercise. * Erase Switch: Create a flip-flop, erasing that is reset to 0 on initGame and flips its value if switchErase is high. Use simulation to debug your design. You might consider making the timer delay small to make it faster to simulate your design. Exercise #2 Pass-Off: Show the TA your display with the ball bouncing around the screen. You don't need to worry about paddle collisions yet. —- ==== Exercise #3 - Moving the Paddles and Collisions ==== In this exercise you will implement the logic to move the paddles when the user presses the buttons. === Paddle Locations === Add two new registers that maintain the current paddle y-locations, and modify your Pong state machine to draw the paddles at the location stored in these registers. === Moving the Paddles === In the same state that you move the ball location, add additional logic to also move the paddles, if the appropriate buttons are being pressed. Because we have a wait state in our game loop, you don't need to worry about any button debouncing, it is sufficient to just check the button values, and modify the paddle y-locations as necessary. You may want to move the paddles by 2 or 3 pixels, so that the paddles can move faster than the ball. Build and run your design on the board. Make sure it works before moving on. === Paddle Erasing === Modify your Pong state machine to add new states that will cause the paddles to be erased before they are moved. Exercise #3 Pass-Off: Demonstrate to the TAs that the buttons move the paddles up and down. You don't need to worry about collisions yet. —- ==== Exercises #4 - Collisions ==== In this exercise you will complete the game by adding ball collisions with the paddles. * Update the logic that determines the ball x-direction to take into account the locations of the paddles. If the ball hits a paddle, the register that stores the x-direction should be updated accordingly. Exercise #4 Pass-Off:** Demonstrate your completed game to the TA.

Personal Exploration

Modify your game so that it keeps track of when a player “scores”. You can do this however you like. In the video demo above, the ball changes color when it hits the side walls. If you want to go with this approach:

  • Add a register that maintains the ball color.
  • When a player misses the ball (and the ball bounces off of the side walls), alternate the ball color.

You are welcome to choose a different method instead. You could:

  • Keep score and use the 7-segment controller to display the score on the 7-segment displays.
  • Keep score and display the score in binary on the LEDs.
  • Shrink the player paddle when they score a point.
  • Use the CharDrawer to display a score on the screen.

Final Submission

Submit your SystemVerilog modules using the code submission on Learning Suite. (Make sure your SystemVerilog conforms to the lab SystemVerilog coding standards).