This is an old revision of the document!
In this two-week lab, you will use your ball and line modules to make a Pong game.
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 immediately redrawing them and repeating the process. Erasing is done by drawing the objects black.
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.initGame
will reset the appropriate registers (ball location, ball direction, paddle locations, score, etc.)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, or move the objects after drawing them. The invertErasing
signal output by the state machine will cause this flip-flop to invert its value. Given your system clock, how many cycles do you need to wait for, in order to implement a 0.01 second delay?
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.
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:
// 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:
always_ff
block to manage the ball location. Create two registers, BallX
and BallY
that are reset to the center of the screen when initGame
is true. This is sufficient for this exercise.always_ff
block to manage the paddle locations. Create two registers, LPaddleY
and RPaddleY
that position the paddles halfway down the screen when initGame
is true. This is sufficient for this exercise.always_ff
block to manage the player scores. For this exercise, just assign the P1score
and P2score
registers to 0 when initGame
is true. 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.vga_color
to values of your choice in each state.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.
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:
timerRst
signal, increment each cycle, and output a timerDone
signal when it reaches a value of your choosing.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.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.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.
In this exercise you will implement the logic to move the paddles when the user presses the buttons.
Update the following components:
moveAndScore
is high), move the paddles appropriately based on the button inputs (LPaddleUp
, RPaddleUp
, etc.) 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 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. The logic to do this is a bit tricky, so think about it carefully. You don't want the paddles to be able to move off of the screen. Exercise #3 Pass-Off: Demonstrate to the TAs that the buttons move the paddles up and down, and that the ball collides and bounces off the paddles consistently.
In this exercise you will complete the game by adding score keeping. The player scores are already shown on the seven segment displays, you just need to update the registers you created earlier when a player scores a point.
Add logic that looks at looks at the ball location in the move state (moveAndScore
is high), and update the player scores appropriately.
Exercise #4 Pass-Off: Demonstrate your completed game to the TA.
No personal exploration is required for this lab, but feel free to have fun making whatever changes you would like to the game. For example, you might shrink the player paddles when they score a point to make it more challenging, or speed up the gameplay.
Submit your SystemVerilog modules using the code submission on Learning Suite. (Make sure your SystemVerilog conforms to the lab SystemVerilog coding standards).