

## Game Engine

Eva Nootenboom, 5281288  
Willem van Overbeeke, 5363950  
Dagmar Westenbrink, 5379954

December 10, 2021



## **Abstract**

This report describes a module of an integrated circuit, which is responsible for actualising the rules of a game. The game, called ElecTRON, is a two player game where players move at a constant speed while trying to enclose the other player and avoid being enclosed themselves. This module does the operations that make the game run. This so called game engine consist of a counter, a few registers and a controller. The game engine has been tested by synthesis and simulations. After simulating the synthesized version, it can be concluded that this module works correctly.

# Contents

|          |                                   |           |
|----------|-----------------------------------|-----------|
| <b>1</b> | <b>Introduction</b>               | <b>4</b>  |
| <b>2</b> | <b>Specification</b>              | <b>5</b>  |
| 2.1      | Requirements . . . . .            | 5         |
| 2.2      | Inputs . . . . .                  | 6         |
| 2.3      | Outputs . . . . .                 | 7         |
| <b>3</b> | <b>Design</b>                     | <b>10</b> |
| 3.1      | Busy counter . . . . .            | 10        |
| 3.2      | Register . . . . .                | 11        |
| 3.3      | Controller . . . . .              | 12        |
| <b>4</b> | <b>Results</b>                    | <b>19</b> |
| 4.1      | Busy Counter . . . . .            | 19        |
| 4.2      | Register . . . . .                | 19        |
| 4.3      | Controller . . . . .              | 19        |
| 4.4      | Synthesized Circuit . . . . .     | 20        |
| <b>5</b> | <b>Conclusion</b>                 | <b>21</b> |
| <b>A</b> | <b>Simulations</b>                | <b>22</b> |
| <b>B</b> | <b>Figures</b>                    | <b>40</b> |
| B.1      | Playing field . . . . .           | 40        |
| B.2      | Steps of the simulation . . . . . | 40        |
| <b>C</b> | <b>VHDL code</b>                  | <b>41</b> |
| C.1      | busy_counter . . . . .            | 41        |
| C.2      | register . . . . .                | 43        |
| C.3      | game_engine . . . . .             | 50        |

## 1 Introduction

The game engine is a module of the chip for ElecTRON. ElecTRON is a game, based on the movie tron [1], of two players in a battle to stay alive the longest. The game is played on a grid, consisting of two layers displayed on top of each other. While both player are alive they move at a constant speed throughout the grid. Every grid cell that is visited by a player is made into a wall and is no longer accessible to either player. This leads to a game of rising tension as the playing field gets smaller and smaller while the players try to enclose each other.

From the description above, a few functions for the chip can be distilled. Firstly, the players need to move and the position needs to be changed by the game engine. Secondly, the traveled path needs to be stored in memory, and lastly memory must be read to see if a player collides with a wall. All these operations which make the game run are done by the game engine. Two other modules control the memory and the output to a monitor.

Some other features were proposed but have not yet been implemented. These features include: a sound engine; adding a second layer to the playing field; and giving the players the ability to get a short boost. Although these features are not yet implemented, the code has been written with these features in mind. This is done to make the implementation of these extras easier.

## 2 Specification

In this section all the requirements, what the module should do and the in- and outputs of the module are explained in detail. The black box with all the inputs and outputs can be seen in Figure 1.



Figure 1: Black box Game Engine.

### 2.1 Requirements

The module is required to do the following (all these requirements go for both players):

- Determine which direction the player is moving.
- Determine the next position of the player.
- Send the direction and current position to the graphics engine.
- Determine the shape of the wall of the previous position of the player and send this to the memory module.
- Determine whether or not the player collides with the border.
- Determine whether or not a player has already been at the next position; meaning that the player collides with a wall.
- Determine whether or not the players are colliding with each other.
- Determine who won, when one or both players collided with a border or a wall.
- Determine the state of a player: collided at a border of a cell; collided in the middle of the cell; still playing the game; the player is not yet ready.
- Send the state of the player to the graphics engine.
- Send to the graphics engine if the game has started or not, is going on or else which player has won.
- Determine when the game starts and finishes.

In order to do all of this the game engine needs to store data. In Section 3.2 is it explained which data should be stored and how that is done.

## 2.2 Inputs

The game engine has the following inputs: `clk`, `reset`, `input`, `busy`, `read_memory` and `memory_ready` (see Figure 1). The following sections give details on the different inputs.

### 2.2.1 Clk

`clk` is the main clock signal from the chip. The frequency of the clock is 25 MHz. The FSM only goes to another state when the clock is going from '0' to '1'; on a rising edge. The register also only updates the stored values at a rising edge of the clock.

### 2.2.2 Reset

This is a real button. When this button is pressed the chip is reset. This means that the game will go back to its starting screen, the game engine will go back to the `reset_state` and the register will be made empty so a new game can be played.

### 2.2.3 Input

The `input` is a four bit signal. This input comes from the buttons which are pressed by the players. Both players can press four buttons: up, down, left and right. By using SR-latches among other things this is transferred into a four bit signal. The two most significant bits give the direction of player 1 and the two least significant bits give the direction of player 0. In Table 1 the bits that correspond to each direction can be seen.

Table 1: The meaning of the bits related to the direction of the players.

| bits | direction |
|------|-----------|
| 00   | up        |
| 01   | left      |
| 10   | down      |
| 11   | right     |

### 2.2.4 Busy

`busy` is a signal that is created by the graphics engine. This is a signal slightly below 60 Hz. It is '1' when the graphics engine is using the memory and '0' otherwise. While the graphics engine is using the memory, the game engine cannot. So only when the busy signal is '0' the game engine can use the memory. The signal is '0' for 36000 clock cycles, which is reasoned to be enough for all calculations and reading from and writing to memory.

### 2.2.5 Read\_memory

`read_memory` is an 8 bit signal that is created by the memory module in the chip. The memory stores a value for 32x32 addresses. `read_memory` is the value at one of those addresses, as requested from the game engine. The four most significant bits are used for layer 1 and the four least significant bits are used for layer 0. Due to the fact that at this moment the game is only made for one layer the four most significant bits are always '0'. Of the other four bits, the fourth least significant bit represents

the player and the third least significant bit to the first least significant bit represent the wall shape. For example, 0111 means that player 0 left a trail in the shape  $\sqcup$ . In Table 2 the bits that correspond to each wall shape can be seen.

Table 2: The meaning of the bits related to the wall shape

| bits | wall shape |
|------|------------|
| 000  | empty      |
| 001  | -          |
| 010  |            |
| 110  | Γ          |
| 101  | ⊤          |
| 100  | ⊣          |
| 111  | ⊲          |

### 2.2.6 Memory\_ready

`memory_ready` is an one bit signal that is created by the memory module of the chip. This signal is '0' when the memory module is busy reading a value from the memory component or writing a value to the memory component. Otherwise the value is '1', meaning the memory controller is not busy.

## 2.3 Outputs

The game engine has the following outputs: `state_vga`, `write_enable`, `write_memory`, `address`, `position_0_vga`, `position_1_vga`, `direction_0_vga`, `direction_1_vga`, `player_state_0_vga`, `player_state_1_vga`, `go_to` and `clear_memory` (see Figure 1).The following sections give details on the different outputs.

### 2.3.1 state\_vga

`state_vga` is three bit signal that is send to the graphics engine. This signal contains information about what state the game is in. The following values correspond to the following state in the game:

Table 3: The meaning of the bits related to the game state.

| bits | game state       |
|------|------------------|
| 000  | startscreen      |
| 001  | tie              |
| 010  | player 0 won     |
| 011  | player 1 won     |
| 111  | playing the game |

### 2.3.2 player\_state\_0\_vga and player\_state\_1\_vga

There are several states the player can be in: alive, collided at a border, collided at the center or not ready. The signals `player_state_0_vga` and `player_state_1_vga` consist of two bits and let

the graphics engine know what state the players are in. With this information the graphics engine can display the player correctly. The following values correspond to the following situations in the game.

Table 4: The meaning of the bits related to the situation in the game.

| bits | player state                            |
|------|-----------------------------------------|
| 00   | player collided in the middle of a cell |
| 01   | player collided at the border of a cell |
| 10   | player is not ready to start the game   |
| 11   | player is playing the game              |

### 2.3.3 write\_enable

`write_enable` is a signal that goes to the memory module. This signal has to be '1' when writing a value to the memory. Once the memory is done writing this value onto the right address (see Section 2.3.5) the `write_enable` goes to '0'.

### 2.3.4 write\_memory

`write_memory` is the value that should be written to memory onto the correct address (see Section 2.3.5). It has the same format as `read_memory` (see Section 2.2.5).

### 2.3.5 address

`address` is a 10 bit signal. This signal refers to one address in the memory. Each address corresponds to one cell in the playing field. The field has a grid of 30x30 and the memory has 32x32 addresses, meaning that the two highest addresses for both x and y are not used. The most significant five bits are the position on the y axis, with '00000' being the first row of the field (the highest on the screen). Five least significant bits: position in x-axis, with '00000' the most left position in the field.

### 2.3.6 position\_0\_vga and position\_1\_vga

`position_0_vga` and `position_1_vga` are 11 bit signals. These signals are send to the graphics engine and give information about the current position of the players. The first bit corresponds to the layer the player is on. At this moment it is always '0', because only one layer used. The other ten bits are split in two. See Section 2.3.5 for the meaning of the bits.

### 2.3.7 direction\_0\_vga and direction\_1\_vga

`direction_0_vga` and `direction_1_vga` are signals of two bits each, which send information to the graphics engine about the current direction of the players. The bits have the same meaning as the input signal. The meaning of the bits can be seen in Table 1.

### 2.3.8 go\_to

`go_to` is a one bit signal which is sent to the memory module. This signal needs to be '1' for one clock cycle when something is read from or written to a specific address in memory.

### **2.3.9 clear\_memory**

`clear_memory` is a one bit signal which is sent to the memory module. When this signal is '1' the memory module will clear the memory, meaning that all the values in the memory will be set to "00000000".

### 3 Design

The design of the game engine is divided in three parts: A counter, a register and a controller. The counter counts the amount of times the busy signal goes from high to low. The register functions as memory to store values. Finally, the controller is used to determine, for example, the next position of the player, send information to the graphics engine and send data which needs to be stored to the memory. In this section it is explained in detail how all these parts work.

#### 3.1 Busy counter

The busy counter is a component that counts the number of times the busy signal goes from high to low. The busy signal tells when the graphics engine is creating the output for the monitor. Because the graphics engine reads from the memory when rendering, the game engine cannot read from or write to memory at that same time. Only when busy is low it is possible for the game engine to do computations. Keeping in mind the delays from reading from and writing to memory the game engine is reasoned to still have enough time to complete all computations before busy rises to '1' again.

If calculations are done every time busy is low, the players move every frame update. In that case the movement of the player is too fast. For this problem the busy counter was made. It counts the falling edges of the busy signal and outputs its count to the controller. In the controller the count is compared to a constant, which is set at 16 for now. Once all the different modules are implemented together it will be known whether or not this value is okay, too low or too high. When the count is bigger or equal to the constant the counter will be reset by the controller.

The busy counter is realised as an FSM. The finite state diagram can be seen in Figure 2 and the VHDL code in Appendix 3.1. In the state `busy_high_to_low` the count is increased, thus counting falling edges. This means the count increases one clock cycle after the falling edge of busy. However, given the low frequency of busy, one clock cycle does not matter.



Figure 2: FSM busy counter.

## 3.2 Register

The game engine needs registers to store values, because some values need to be kept the same even though the FSM is going to another state. The register has an enable function, meaning only when the enable signal is high the value should be changed. For instance, when the inputs of player 1 are being sampled, the enable signal `e_next_direction_1` is set to '1' while the data signal `d_next_direction_1` is made equal to the input of player 1. The VHDL code for the register can be seen in Appendix C.2 and the encoding for the different signals can be found in Section 2.2 or Section 2.3.

Because the area on the chip and the number of flip-flops are limited, storing a certain value on the chip must be necessary otherwise it takes up much needed space. The reasons for storing a value in the register is given for each value that needs to be stored in the following paragraphs. The entity in the VHDL code of the register is named `ge_register` (with `ge` being short for game engine).

### 3.2.1 position and next\_position

To keep track of the players during the game, the position needs to be stored because the next position of a player depends on its last position as well as the inputs. This can be achieved by only saving one pair of coordinates, however, both values are needed for a special situation. Namely, when the two players are directly facing each, as seen in Figure 5b. The next cycle both players crash into the wall left by the other player. However, the player should explode at the edge of the cell, meaning this situation has to be checked for separately. In terms of `position` and `next_position`, the position of one player equals the next position of the other player and vice versa. Both `position` and `next_position` are needed in this comparison so both have to be stored in the register.

### 3.2.2 direction

The shape of a wall does not only depend on the direction in which the player leaves a cell, but also on the direction the player entered the cell. The direction the player came from is the input of the previous cycle. This means that the direction needs to be stored for the next cycle to then again determine the wallshape.

### 3.2.3 wallshape

After assigning a value to the wall shape, it needs to be written to memory, this is done in different states of the FSM. Therefore the wallshape needs to be stored in the register.

### 3.2.4 read\_memory

The part of the register for storing the data read from the memory is not yet used, but once the second layer is implemented, storing that data is necessary. This follows from the method of saving the data of the playing field in the memory. The 8 bits per address are divided in two, the most significant four are for layer 1 and the least significant four bits are for layer 0. Writing a wall to layer 1 should not destroy the data for layer 0 and vice versa. Thus the data read from the memory needs to be stored in order only change half of it.

### **3.2.5 next\_direction**

The part in the register labeled `next_direction` is used to store the inputs before starting the computations. This ensures that the direction a player leaves a cell does not change while calculating the next position or the shape of the wall.

### **3.2.6 player\_state**

The information about whether or not the player is alive or how a player died, is stored in the `player state`. Checking if and how a player died is done in multiple states and in some instances the next state of the controller FSM depends on the player state.

## **3.3 Controller**

The calculations and the communication with the memory and register are done by the controller. First, some details about the communication with the memory module are given, thereafter about whether the FSM is a mealy or a moore machine and finally the states of the FSM are listed with a description of their functionality. Some state are doubled, one per player. If so, the number of the player is replaced with a #. The FSM of the controller can be seen in Figure 3.



Figure 3: FSM game engine.



Figure 4: Waveform displaying the communication with the memory module.

### 3.3.1 Communication with the Memory Module

Whenever the game engine wants to communicate with the memory module two states are needed. The first reason is that the memory module needs the `go_to` signal to be high for only one clock cycle. The second reason is that `memory_ready` will be high for one more clock cycle after a task is sent to the memory module. This means that only after a rising clock edge `memory_ready` can be checked to see whether the FSM can go to the next state. Thus, the first state should send the task to the memory module and after one clock cycle the FSM can go to the state in which it checks whether or not the memory module is finished while also sending the task (this is needed by the memory controller).

Figure 4 shows a waveform containing a task for the memory module. Highlighted in red is the first of the two states. In this state `go_to` is set high, meaning the next rising edge the memory goes to a specified address. As can be seen, this state `memory_ready` is still high and thus the second state is needed.

### 3.3.2 Mealy or Moore Machine

The FSM of the controller is neither a moore machine nor a mealy machine. The VHDL code below is a sample of the FSM to illustrate. The first piece is taken from the state `read_memory_player_1` and shows how some of the output signals are dependent on input signals. For instance, the signal `e_player_1_state` changes when `read_memory` changes. The risk of having this mealy implementation is that the critical path becomes to long. Especially with multiple mealy machines together, there could be unexpectedly long delay times. It is reasoned however, that this is not the case in this instance because the outputs only go to the register. Because of this there is no path directly from input to output, there is always a register in between.

The second sample is from the state `want_to_write_1`. In this state the output signals depend directly on the input signals also. However, the input signals come from the register, which only change with the clock. Thus in this case a direct path from the inputs to the outputs of the game

engine is prevented as well.

Whenever a signal going out of the FSM not only depends on the state but also on inputs, it is similar to one of the two cases illustrated above. This ensures that no direct path from input to output is created without the register in between and no timing issues of the mealy implementation arise.

```

1 -- from the state 'read_memory_player_1'
2 if (memory_ready = '1') then
3   if (read_memory = "00000000") then
4     e_player_1_state <= '0';
5     d_player_1_state <= (others => '0');
6   else
7     e_player_1_state <= '1';
8     d_player_1_state <= "00";
9   end if;
10  new_state <= check_collision;
11 else
12  e_player_1_state <= '0';
13  d_player_1_state <= (others => '0');
14  new_state <= read_memory_player_1;
15 end if;
16
17 -- from the state 'want_to_write_1'
18 if (player_1_state = "01") then
19   write_enable      <= '0';
20   write_memory      <= "00000000";
21   address           <= "0000000000";
22   go_to              <= '0';
23   new_state          <= change_data;
24 else
25   write_enable      <= '1';
26   write_memory(7 downto 3) <= "00001";
27   write_memory(2 downto 0) <= wallshape_1;
28   address           <= position_1(9 downto 0);
29   go_to              <= '1';
30   new_state          <= write_memory_player_1;
31 end if;
```

VHDL/mealy\_vs\_moore.vhd

### 3.3.3 Reset

When the reset button is pressed the FSM goes to the `reset_state`. This happens regardless of the state the FSM is in. This is not shown in Figure 3 for readability. In the `reset_state` all signals are set to zero. In order to start the game, the reset button always needs to be pushed beforehand. When `reset` goes low, the FSM goes to the `want_to_load` state.

### 3.3.4 Loading

In the `want_to_load` state and the `loading` state the initial condition are set before starting the game. Some initial values for the register are hard coded, such as position and direction, and some are set to zero. Furthermore the entire memory needs to be set to zero. This is done by the memory module when `clear_memory` becomes '1'. Two states are needed for this as is explained in Section 3.3.1. When the memory is finished and `memory_ready` becomes '1' the FSM goes to the state `get_ready`.

### 3.3.5 Getting Ready

The `get_ready`, `player_0_ready` and `player_1_ready` states are made to give the players a heads-up before the game starts. When a sound engine is added, it could create a sound effect at this point. The FSM stays in this state until one of players enters the direction that the player is facing. For instance, when the player is facing up the person playing has to press the button pointing up. For now this direction is always up, maybe later this can be randomized with the position. The starting positions at this moment can be seen in Figure 23 in the appendix. When the correct button is pressed the color of the player changes to let the person know which of the two players is theirs. The FSM then goes to the state `player_0_ready` or `player_1_ready` depending on which player presses the right button first. Once both players are paying attention and have entered the correct direction the game engine can starts its first game cycle and the FSM goes to `wait_state`.

### 3.3.6 Waiting

In the state `wait_state` the FSM waits until the busy counter (see Section 3.1) reaches 16. By making this value lower or higher the players move faster or slower during the game respectively. After waiting for sixteen falling edges of `busy`, the FSM goes to `busy_reset` where the busy counter is reset. One clock cycle later the FSM is in the `read_inputs` state.

### 3.3.7 Reading the Inputs

In the state `read_inputs` it is determined which direction the player wants to go. The value of the input `input` is stored in the register for `next_direction_0` and `next_direction_1`. This value is stored so that the direction of the player cannot change halfway one cycle in the FSM and thus avoids errors. The FSM then goes to the state `wall_shape`

### 3.3.8 Wall Shape and Next Position

In the `wall_shape` state the wall shape and the next position are determined. The wall shape a player leaves behind is determined using `direction_#` and `next_direction_#`. When a player wants to go in the opposite direction of what is was going it collides against itself. In this case no wall shape is determined and the `player_#_state` is set to "00", meaning the player dies in the middle of a cell.

The next position is calculated based on the direction a player is going and the position it is now. This is done by increasing or decreasing the y or x value of `position_#` with 1. The FSM then goes to the state `check_border`.

### 3.3.9 Collision with Border

The playing field is a grid of 30 by 30 cells (five by five bits). Around this grid is a border. When a player runs into a border, that player loses. To check if a player collides with a border the state `check_border` is made. It checks whether or not a player is going to the border by looking at the position of a player and its direction. If the direction is towards the border and the player is next to the border, the player crashes into the border, meaning the `player_state` changes. Then the FSM goes to the state `want_to_read_0`.

### 3.3.10 Reading from Memory

The states `want_to_read_#` and `read_memory_player_#` are used to read from the memory component. To do this the value of `go_to` should be '1' for one clock cycle. Because of this and what is explained in Section 3.3.1 two states are needed. Furthermore the value of `address` is set to the cell of the next position of the player. With this information from the memory it is checked if a player already has been here and thus if there is a wall there. Next, the FSM goes to the `check_collision` state.

### 3.3.11 Checking for player against player crashes

A few instances could occur in which the players crash against each other, which is checked in the state as `check_collision`. Four cases need to be checked. The first is when both players go to the same cell. In this case the players crash against each other in the middle of the cell. The second is when both players try to go to the cell the other player is on. The players then crash at the border of a cell. The last two cases are if one player tries to enter the cell which the other player is about to leave. Although the other player will leave a wall, this is not yet written to memory, thus making it necessary to check separately. Then the FSM goes to `want_to_write_0`. These four cases are also displayed in Figure 5.

### 3.3.12 Writing to Memory

Writing to memory is done by two states per player: `want_to_write_#` and `write_memory_player_#`. The `want_to_write_#` states are needed, next to the reasons mentioned in Section 3.3.1, because the state of the player must be checked. If a player crashes against the edge of a cell, the player never leaves the cell, so no wall should be written to the memory. After this the FSM goes to `change_data`.

### 3.3.13 Update Data

In the state `change_data` the signals `direction_#` and `position_#` are updated. Meaning that it gets the value of `next_direction_#` and `next_position_#`. However, if a player has crashed into the side of a cell, the position is not changed. By changing these signals the outputs `position_0_vga`, `position_1_vga`, `direction_0_vga` and `direction_1_vga` are also updated. After that the FSM goes to `check_who_won`.



(a) Case 1: Both players try to go to the same cell.

(b) Case 2: Players try to go to each other's cell.



(c) Case 3 and 4: One player tries to go to the cell of the other player.

Figure 5: Different player against player collisions.

### 3.3.14 Check Who Won

In the state `check_who_won` it is checked if one or both players have died. This is done by checking what the value for `player_#_state` is. If both players are still alive the game continues and the FSM goes back to the state `wait_state`. If one or both players have died the FSM goes to `player_0_won`, `player_1_won` or `tie`, depending on who won.

### 3.3.15 Game Over

After it is decided who has won the game, the FSM goes to one of the following states: `player_0_won`, `player_1_won` or `tie`. Depending on who won, the `state_vga` is set to a certain value (see Table 3). The FSM then stays in that state until the reset button is pressed.

## 4 Results

To test if the written VHDL code works correctly, it has been tested. This is done by first synthesizing the code and then simulating that using testbenches. First the register and the busy counter are synthesized and simulated separately after which the design is tested.

### 4.1 Busy Counter

In Figure 6 and 7 the simulation results of the synthesized circuit can be seen. It can be seen that after the `reset` has become '0' the busy counter counts. Furthermore it can be seen that it will start counting again from zero if `reset` goes high and low. From this it can be concluded that the busy counter works correctly.

### 4.2 Register

In Figures 8 and 9 the simulation results of the synthesized register can be seen. When `reset` is high, the outputs are all zeros. The data input signals, the signals starting with `d_`, are set to an arbitrary value by the testbench. These arbitrary values only appear at the output signals, starting with `q_`, when the corresponding enable signal is high. The enable signals are labeled using `e_` at the beginning of the name. In the simulation results this can be seen, meaning that the register works correctly.

### 4.3 Controller

To test the controller a couple of different testbenches are used to test most ways the game can run. For all simulations the synthesized circuit has been used.

For the first simulation (see Figure 10 and 11) one cycle in the FSM is tested where both players do not die. In this simulation the busy signal has a much higher frequency and a different duty cycle than in reality to reduce simulation times. Moreover, for the same reason the signal `memory_ready` is made '1' after less clock cycles than the memory module would do. From this simulation it can be concluded that the game starts up the correct way. Furthermore it can also be concluded that the busy signal counts correctly and that reading and writing from and to the memory works correct.

In all simulations explained below the `wait_state` and `busy_reset` states are skipped and `memory_ready` is always set to '1'. This is because it is already known that it works and it makes the simulation a lot easier and shorter. With this the following simulations are done:

- Both players collided against a border at the same time. One of the players collides against the border on left side of the playing field and the other player collides against the border on the right side of the playing field. How the players move thorough the field can be seen in Figure 24 and the simulation can be seen in Figure 12.
- One player collides against a border and the other player stays alive. In Figure 13 player 0 collides against the border at the top of the playing field and in Figure 14 player 1 collides at the border at the bottom of the playing field.
- One of the player presses the button in the direction opposite of what the player is going and the other player stays alive (see Figure 15 for the simulation results).

- The players collide head-on at the border of a cell. How the players collide can be seen in Figure 5b. (see Figure 16 and 17 for the simulation results)
- The players collide head to head in the middle of a cell. How the players collide can be seen in Figure 5a (see Figure 18 and 19 for the simulation results).
- The players move in a wall made by either himself or the other player. In the simulations in Figure 20, Figure 21 and Figure 22 it can be seen that first player 1 moves into a wall and player 0 stays alive. Then the other way around and lastly both players move into a wall at the same moment. Because the game engine is not connected to the memory module yet, the testbench just creates a wall in a cell even though the players have not been there yet, just to test the code.

All these simulation give the correct output and thus it can be concluded this all works correct.

#### 4.4 Synthesized Circuit

After synthesis the following information is obtained about the circuit

Table 5: Synthesized circuit.

|                               |                             |
|-------------------------------|-----------------------------|
| Number of sequential cells    | 75                          |
| Number of combinatorial cells | 473                         |
| Area of the circuit           | $10\,271.341 \mu\text{m}^2$ |

After synthesis the total area of the complete circuit should not pass  $65\,000 \mu\text{m}^2$ . The game engine module uses around 16 % of this. Depending of how much area the other modules use, the design will fit on the chip.

## **5 Conclusion**

The simulations of both the synthesized version and the non-synthesized version show that the game engine works correctly. The game engine can determine positions, directions, wall shapes, if a player has died and how and it can send the necessary information to the graphics engine and the memory module. Thus the game engine meets all the requirements.

## **References**

- [1] Steven Lisberger, "Tron", 1983, Walt Disney Productions,

## A Simulations

In this section all the simulation results of the synthesized circuit can be seen.



Figure 6: Simulation results from the busy counter (1).  
 This is the simulations result of the busy\_counter. Because of readability it is split into two figures.



Figure 7: Simulation results from the busy counter (2).



Figure 8: Simulation results from the register (1).



Figure 9: Simulation results from the register (2).



Figure 10: Simulation results from the game engine to test the wait\_state and the states with memory (1).



Figure 11: Simulation results from the game engine to test the wait\_state and the states with memory (2).



Figure 12: Simulation results from the game engine to test the left and right border.



Figure 13: Simulation results from the game engine to test the upper and lower border (1).



Figure 14: Simulation results from the game engine to test the upper and lower border (2).



Figure 15: Simulation results from the game engine to test if a player can go back.



Figure 16: Simulation results from the game engine to test the head-on collision at the border of a cell (1).



Figure 17: Simulation results from the game engine to test the head-on collision at the border of a cell (2).



Figure 18: Simulation results from the game engine to test the head-on collision in the middle of a cell (1).



Figure 19: Simulation results from the game engine to test the head-on collision in the middle of a cell (2).



Figure 20: Simulation results from the game engine to test when a player moves into a wall(1).



Figure 21: Simulation results from the game engine to test when a player moves into a wall(2).



Figure 22: Simulation results from the game engine to test when a player moves into a wall(3).

## B Figures

### B.1 Playing field

This is the playing field with the players at their starting positions.



Figure 23: The starting positions of the players

### B.2 Steps of the simulation

These are the steps of player 1 according to the simulation that checks the left and right border. The starting position and the final position of player 0 according to the same simulation.



Figure 24: The steps taken in the simulation for player 1 and the begin and end position of player 0.

## C VHDL code

### C.1 **busy\_counter**

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3
4 entity busy_counter is
5     port(clk           : in  std_logic;
6          global_reset   : in  std_logic;
7          game_engine_reset : in  std_logic;
```

```

8      busy          : in  std_logic;
9      busy_count    : out std_logic_vector(4 downto 0));
10 end busy_counter;
11
12 library IEEE;
13 use IEEE.std_logic_1164.ALL;
14 use IEEE.numeric_std.all;
15
16 architecture behaviour of busy_counter is
17
18   -- implemented as a FSM with the count increasing when busy
19   -- changes from high to low
20   type busy_counter_state_type is (reset, busy_low, busy_high,
21   busy_high_to_low);
22   signal busy_counter_state, next_busy_counter_state:
23   busy_counter_state_type;
24   signal unsigned_busy_count, next_unsigned_busy_count: unsigned
25   (4 downto 0);
26
27 begin
28
29   process (clk)
30   begin
31     -- only count when the reset from the game engine and the
32     -- global reset are '0'
33     if (clk'event and clk = '1') then
34       if (global_reset = '1' or game_engine_reset = '1') then
35         busy_counter_state <= reset;
36         unsigned_busy_count <= (others => '0');
37       else
38         busy_counter_state <= next_busy_counter_state;
39         unsigned_busy_count <= next_unsigned_busy_count;
40       end if;
41     end if;
42   end process;
43
44   process (busy, busy_counter_state)
45   begin
46     case busy_counter_state is
47
48       when reset =>
49         -- the counter is '0'
50         next_busy_counter_state <= busy_low;
51         next_unsigned_busy_count <= (others => '0');
52
53       when busy_low =>
54         -- the busy signal is '0'

```

```

50      -- the counter remains the same
51      next_unsigned_busy_count <= unsigned_busy_count;
52
53      -- when the busy signal goes to '1' the next state is
54      -- 'busy_high'
55      if (busy = '1') then
56          next_busy_counter_state <= busy_high;
57      else
58          next_busy_counter_state <= busy_low;
59      end if;
60
61      when busy_high =>
62          -- the busy signal is '1'
63          -- the counter remains the same
64          next_unsigned_busy_count <= unsigned_busy_count;
65
66          -- when the busy signal goes to '0' the next state is
67          -- 'busy_high_to_low'
68          if (busy = '0') then
69              next_busy_counter_state <= busy_high_to_low;
70          else
71              next_busy_counter_state <= busy_high;
72          end if;
73
74      when busy_high_to_low =>
75          -- the busy signal went from '1' to '0'
76          -- the counter is added by 1
77          next_unsigned_busy_count <= unsigned_busy_count + 1;
78
79          -- the next state is 'busy_low'
80          next_busy_counter_state <= busy_low;
81      end case;
82  end process;
83  busy_count <= std_logic_vector(unsigned_busy_count);
84
85 end behaviour;

```

VHDL/busy\_counter.vhd

## C.2 register

### C.2.1 ge\_register

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3
4 entity ge_register is

```

```

5   port(clk, reset      : in  std_logic;
6   e_position_0  : in  std_logic;
7   e_position_1  : in  std_logic;
8   d_position_0  : in  std_logic_vector(10 downto 0);
9   d_position_1  : in  std_logic_vector(10 downto 0);
10  e_wallshape_0 : in  std_logic;
11  e_wallshape_1 : in  std_logic;
12  d_wallshape_0 : in  std_logic_vector(2 downto 0);
13  d_wallshape_1 : in  std_logic_vector(2 downto 0);
14  e_read_mem_0  : in  std_logic;
15  e_read_mem_1  : in  std_logic;
16  d_read_mem_0  : in  std_logic_vector(7 downto 0);
17  d_read_mem_1  : in  std_logic_vector(7 downto 0);
18  e_next_pos_0  : in  std_logic;
19  e_next_pos_1  : in  std_logic;
20  d_next_pos_0  : in  std_logic_vector(10 downto 0);
21  d_next_pos_1  : in  std_logic_vector(10 downto 0);
22  e_direction_0 : in  std_logic;
23  e_direction_1 : in  std_logic;
24  d_direction_0 : in  std_logic_vector(1 downto 0);
25  d_direction_1 : in  std_logic_vector(1 downto 0);
26  e_next_dir_0  : in  std_logic;
27  e_next_dir_1  : in  std_logic;
28  d_next_dir_0  : in  std_logic_vector(1 downto 0);
29  d_next_dir_1  : in  std_logic_vector(1 downto 0);
30  e_p_state_0   : in  std_logic;
31  e_p_state_1   : in  std_logic;
32  d_p_state_0   : in  std_logic_vector(1 downto 0);
33  d_p_state_1   : in  std_logic_vector(1 downto 0);
34  q_position_0  : out std_logic_vector(10 downto 0);
35  q_position_1  : out std_logic_vector(10 downto 0);
36  q_wallshape_0 : out std_logic_vector(2 downto 0);
37  q_wallshape_1 : out std_logic_vector(2 downto 0);
38  q_read_mem_0  : out std_logic_vector(7 downto 0);
39  q_read_mem_1  : out std_logic_vector(7 downto 0);
40  q_next_pos_0  : out std_logic_vector(10 downto 0);
41  q_next_pos_1  : out std_logic_vector(10 downto 0);
42  q_direction_0 : out std_logic_vector(1 downto 0);
43  q_direction_1 : out std_logic_vector(1 downto 0);
44  q_next_dir_0  : out std_logic_vector(1 downto 0);
45  q_next_dir_1  : out std_logic_vector(1 downto 0);
46  q_p_state_0   : out std_logic_vector(1 downto 0);
47  q_p_state_1   : out std_logic_vector(1 downto 0));
48 end ge_register;

```

VHDL/ge\_register.vhd

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3
4 architecture structural of ge_register is
5   component reg_8
6     port(clk      : in  std_logic;
7           reset    : in  std_logic;
8           e        : in  std_logic;
9           d        : in  std_logic_vector(7 downto 0);
10          q        : out std_logic_vector(7 downto 0));
11 end component;
12
13 component reg_3
14   port(clk      : in  std_logic;
15         reset    : in  std_logic;
16         e        : in  std_logic;
17         d        : in  std_logic_vector(2 downto 0);
18         q        : out std_logic_vector(2 downto 0));
19 end component;
20
21 component reg_2
22   port(clk      : in  std_logic;
23         reset    : in  std_logic;
24         e        : in  std_logic;
25         d        : in  std_logic_vector(1 downto 0);
26         q        : out std_logic_vector(1 downto 0));
27 end component;
28
29 component reg_11
30   port(clk      : in  std_logic;
31         reset    : in  std_logic;
32         e        : in  std_logic;
33         d        : in  std_logic_vector(10 downto 0);
34         q        : out std_logic_vector(10 downto 0));
35 end component;
36
37 begin
38
39 pos0:reg_11 port map      ( clk  => clk,
40                           reset=> reset,
41                           e    => e_position_0,
42                           d    => d_position_0,
43                           q    => q_position_0);
44 pos1:reg_11 port map      ( clk  => clk,
45                           reset=> reset,
46                           e    => e_position_1,
47                           d    => d_position_1,

```

```

48      q      => q_position_1);
49 next_pos0:reg_11 port map      ( clk  => clk,
50      reset=> reset,
51      e     => e_next_pos_0,
52      d     => d_next_pos_0,
53      q     => q_next_pos_0);
54 next_pos1:reg_11 port map      ( clk  => clk,
55      reset=> reset,
56      e     => e_next_pos_1,
57      d     => d_next_pos_1,
58      q     => q_next_pos_1);
59 wall_0:reg_3 port map          ( clk  => clk,
60      reset=> reset,
61      e     => e_wallshape_0,
62      d     => d_wallshape_0,
63      q     => q_wallshape_0);
64 wall_1:reg_3 port map          ( clk  => clk,
65      reset=> reset,
66      e     => e_wallshape_1,
67      d     => d_wallshape_1,
68      q     => q_wallshape_1);
69 r_mem_0:reg_8 port map          ( clk  => clk,
70      reset=> reset,
71      e     => e_read_mem_0,
72      d     => d_read_mem_0,
73      q     => q_read_mem_0);
74 r_mem_1:reg_8 port map          ( clk  => clk,
75      reset=> reset,
76      e     => e_read_mem_1,
77      d     => d_read_mem_1,
78      q     => q_read_mem_1);
79 dir_0:reg_2 port map           ( clk  => clk,
80      reset=> reset,
81      e     => e_direction_0,
82      d     => d_direction_0,
83      q     => q_direction_0);
84 dir_1:reg_2 port map           ( clk  => clk,
85      reset=> reset,
86      e     => e_direction_1,
87      d     => d_direction_1,
88      q     => q_direction_1);
89 n_dir_0:reg_2 port map          ( clk  => clk,
90      reset=> reset,
91      e     => e_next_dir_0,
92      d     => d_next_dir_0,
93      q     => q_next_dir_0);
94 n_dir_1:reg_2 port map          ( clk  => clk,

```

```

95         reset=> reset,
96         e      => e_next_dir_1,
97         d      => d_next_dir_1,
98         q      => q_next_dir_1);
99 p_state_0:reg_2 port map          ( clk  => clk,
100           reset=> reset,
101           e      => e_p_state_0,
102           d      => d_p_state_0,
103           q      => q_p_state_0);
104 p_state_1:reg_2 port map          ( clk  => clk,
105           reset=> reset,
106           e      => e_p_state_1,
107           d      => d_p_state_1,
108           q      => q_p_state_1);
109
110 end structural;

```

VHDL/ge\_register-structural.vhd

### C.2.2 reg\_2

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3
4 entity reg_2 is
5   port(clk  : in  std_logic;
6        reset : in  std_logic;
7        e     : in  std_logic;
8        d     : in  std_logic_vector(1 downto 0);
9        q     : out std_logic_vector(1 downto 0));
10 end reg_2;

```

VHDL/reg\_2.vhd

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3
4 architecture behaviour of reg_2 is
5 begin
6
7   process (clk)
8 begin
9     if (clk'event and clk = '1') then
10       if (reset = '1') then
11         q <= "00";
12       elsif (e = '1') then
13         q <= d;

```

```

14      end if;
15  end if;
16 end process;
17
18 end behaviour;

```

VHDL/reg\_2-behaviour.vhd

### C.2.3 reg\_3

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3
4 entity reg_3 is
5   port(clk    : in  std_logic;
6        reset  : in  std_logic;
7        e      : in  std_logic;
8        d      : in  std_logic_vector(2 downto 0);
9        q      : out std_logic_vector(2 downto 0));
10 end reg_3;

```

VHDL/reg\_3.vhd

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3
4 architecture behaviour of reg_3 is
5 begin
6
7   process (clk)
8 begin
9     if (clk'event and clk = '1') then
10       if (reset = '1') then
11         q <= "000";
12       elsif (e = '1') then
13         q <= d;
14       end if;
15     end if;
16   end process;
17
18 end behaviour;

```

VHDL/reg\_3-behaviour.vhd

### C.2.4 reg\_8

```

1 library IEEE;

```

```

2 use IEEE.std_logic_1164.ALL;
3
4 entity reg_8 is
5   port(clk    : in  std_logic;
6         reset : in  std_logic;
7         e     : in  std_logic;
8         d     : in  std_logic_vector(7 downto 0);
9         q     : out std_logic_vector(7 downto 0));
10 end reg_8;

```

VHDL/reg\_8.vhd

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3
4 architecture behaviour of reg_8 is
5 begin
6
7   process (clk)
8   begin
9     if (clk'event and clk = '1') then
10       if (reset = '1') then
11         q <= "00000000";
12       elsif (e = '1') then
13         q <= d;
14       end if;
15     end if;
16   end process;
17
18 end behaviour;

```

VHDL/reg\_8-behaviour.vhd

### C.2.5 reg\_11

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3
4 entity reg_11 is
5   port(clk    : in  std_logic;
6         reset : in  std_logic;
7         e     : in  std_logic;
8         d     : in  std_logic_vector(10 downto 0);
9         q     : out std_logic_vector(10 downto 0));
10 end reg_11;

```

VHDL/reg\_11.vhd

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3
4 architecture behaviour of reg_11 is
5 begin
6
7 process (clk)
8 begin
9 if (clk'event and clk = '1') then
10 if (reset = '1') then
11     q <= "00000000000";
12 elsif (e = '1') then
13     q <= d;
14 end if;
15 end if;
16 end process;
17
18 end behaviour;

```

VHDL/reg\_11-behaviour.vhd

### C.3 game\_engine

```

1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3 use ieee.numeric_std.all;
4
5 entity game_engine is
6     port(clk           : in  std_logic;
7           reset         : in  std_logic;
8           input          : in  std_logic_vector(3 downto 0);
9           busy          : in  std_logic;
10          read_memory   : in  std_logic_vector(7 downto 0);
11          memory_ready  : in  std_logic;
12          state_vga     : out std_logic_vector(2 downto 0);
13          write_enable   : out std_logic;
14          write_memory   : out std_logic_vector(7 downto 0);
15          address        : out std_logic_vector(9 downto 0);
16          position_0_vga : out std_logic_vector(10 downto 0);
17          position_1_vga : out std_logic_vector(10 downto 0);
18          direction_0_vga: out std_logic_vector(1 downto 0);
19          direction_1_vga: out std_logic_vector(1 downto 0);
20          player_state_0_vga: out std_logic_vector(1 downto 0);
21          player_state_1_vga: out std_logic_vector(1 downto 0);
22          go_to          : out std_logic;
23          clear_memory   : out std_logic);

```

```
24 end game_engine;
```

### VHDL/game\_engine.vhd

```
1 library IEEE;
2 use IEEE.std_logic_1164.ALL;
3 use ieee.numeric_std.all;
4
5 architecture behaviour of game_engine is
6   type game_state is (reset_state, want_to_load, loading_state,
7     get_ready, read_inputs, wall_shape, check_border,
8     want_to_read_0, want_to_read_1, read_memory_player_0,
9     read_memory_player_1, check_collision, check_who_won,
10    wait_state, want_to_write_0, want_to_write_1,
11    write_memory_player_0, write_memory_player_1, change_data,
12    player_0_won, player_1_won, tie, player_0_ready,
13    player_1_ready, busy_reset);
14
15 signal state, new_state: game_state;
16 signal direction_0, direction_1, next_direction_0,
17   next_direction_1 : std_logic_vector(1 downto 0);
18 signal d_direction_0, d_direction_1, d_next_direction_0,
19   d_next_direction_1 : std_logic_vector(1 downto 0);
20 signal position_0, position_1, next_position_0, next_position_1
21   : std_logic_vector (10 downto 0);
22 signal d_position_0, d_position_1, d_next_position_0,
23   d_next_position_1 : std_logic_vector (10 downto 0);
24 signal wallshape_0, wallshape_1 : std_logic_vector (2 downto 0);
25 signal d_wallshape_0, d_wallshape_1 : std_logic_vector (2 downto
26   0);
27 signal read_memory_0, read_memory_1 : std_logic_vector (7 downto
28   0);
29 signal d_read_memory_0, d_read_memory_1 : std_logic_vector (7
30   downto 0);
31 signal player_0_state, player_1_state: std_logic_vector (1
32   downto 0);
33 signal d_player_0_state, d_player_1_state: std_logic_vector (1
34   downto 0);
35 signal e_position_0, e_position_1, e_wallshape_0, e_wallshape_1,
36   e_read_memory_0, e_read_memory_1, e_next_position_0,
37   e_next_position_1, e_direction_0, e_direction_1,
38   e_next_direction_0, e_next_direction_1, e_player_0_state,
39   e_player_1_state: std_logic;
40 signal busy_counter_reset: std_logic;
41 signal unsigned_busy_count: std_logic_vector(4 downto 0);
42
```

```

23 component busy_counter is
24 port(clk : in std_logic;
25     global_reset : in std_logic;
26     game_engine_reset : in std_logic;
27     busy : in std_logic;
28     busy_count : out std_logic_vector(4 downto 0));
29 end component;
30
31
32 component ge_register is
33 port(clk, reset : in std_logic;
34     e_position_0 : in std_logic;
35     e_position_1 : in std_logic;
36     d_position_0 : in std_logic_vector(10 downto 0);
37     d_position_1 : in std_logic_vector(10 downto 0);
38     e_wallshape_0 : in std_logic;
39     e_wallshape_1 : in std_logic;
40     d_wallshape_0 : in std_logic_vector(2 downto 0);
41     d_wallshape_1 : in std_logic_vector(2 downto 0);
42     e_read_mem_0 : in std_logic;
43     e_read_mem_1 : in std_logic;
44     d_read_mem_0 : in std_logic_vector(7 downto 0);
45     d_read_mem_1 : in std_logic_vector(7 downto 0);
46     e_next_pos_0 : in std_logic;
47     e_next_pos_1 : in std_logic;
48     d_next_pos_0 : in std_logic_vector(10 downto 0);
49     d_next_pos_1 : in std_logic_vector(10 downto 0);
50     e_direction_0 : in std_logic;
51     e_direction_1 : in std_logic;
52     d_direction_0 : in std_logic_vector(1 downto 0);
53     d_direction_1 : in std_logic_vector(1 downto 0);
54     e_next_dir_0 : in std_logic;
55     e_next_dir_1 : in std_logic;
56     d_next_dir_0 : in std_logic_vector(1 downto 0);
57     d_next_dir_1 : in std_logic_vector(1 downto 0);
58     e_p_state_0 : in std_logic;
59     e_p_state_1 : in std_logic;
60     d_p_state_0 : in std_logic_vector(1 downto 0);
61     d_p_state_1 : in std_logic_vector(1 downto 0);
62     q_position_0 : out std_logic_vector(10 downto 0);
63     q_position_1 : out std_logic_vector(10 downto 0);
64     q_wallshape_0 : out std_logic_vector(2 downto 0);
65     q_wallshape_1 : out std_logic_vector(2 downto 0);
66     q_read_mem_0 : out std_logic_vector(7 downto 0);
67     q_read_mem_1 : out std_logic_vector(7 downto 0);
68     q_next_pos_0 : out std_logic_vector(10 downto 0);
69     q_next_pos_1 : out std_logic_vector(10 downto 0);

```

```

70      q_direction_0 : out std_logic_vector(1 downto 0);
71      q_direction_1 : out std_logic_vector(1 downto 0);
72      q_next_dir_0  : out std_logic_vector(1 downto 0);
73      q_next_dir_1  : out std_logic_vector(1 downto 0);
74      q_p_state_0   : out std_logic_vector(1 downto 0);
75      q_p_state_1   : out std_logic_vector(1 downto 0));
76  end component;
77
78 begin
79
80 reg: ge_register port map (clk => clk,
81     reset          => reset,
82     e_position_0   => e_position_0,
83     e_position_1   => e_position_1,
84     d_position_0   => d_position_0,
85     d_position_1   => d_position_1,
86     e_wallshape_0  => e_wallshape_0,
87     e_wallshape_1  => e_wallshape_1,
88     d_wallshape_0  => d_wallshape_0,
89     d_wallshape_1  => d_wallshape_1,
90     e_read_mem_0   => e_read_memory_0,
91     e_read_mem_1   => e_read_memory_1,
92     d_read_mem_0   => d_read_memory_0,
93     d_read_mem_1   => d_read_memory_1,
94     e_next_pos_0   => e_next_position_0,
95     e_next_pos_1   => e_next_position_1,
96     d_next_pos_0   => d_next_position_0,
97     d_next_pos_1   => d_next_position_1,
98     e_direction_0  => e_direction_0,
99     e_direction_1  => e_direction_1,
100    d_direction_0  => d_direction_0,
101    d_direction_1  => d_direction_1,
102    e_next_dir_0   => e_next_direction_0,
103    e_next_dir_1   => e_next_direction_1,
104    d_next_dir_0   => d_next_direction_0,
105    d_next_dir_1   => d_next_direction_1,
106    e_p_state_0    => e_player_0_state,
107    e_p_state_1    => e_player_1_state,
108    d_p_state_0    => d_player_0_state,
109    d_p_state_1    => d_player_1_state,
110    q_position_0   => position_0,
111    q_position_1   => position_1,
112    q_wallshape_0  => wallshape_0,
113    q_wallshape_1  => wallshape_1,
114    q_read_mem_0   => read_memory_0,
115    q_read_mem_1   => read_memory_1,
116    q_next_pos_0   => next_position_0,

```

```

117     q_next_pos_1  => next_position_1,
118     q_direction_0 => direction_0,
119     q_direction_1 => direction_1,
120     q_next_dir_0  => next_direction_0,
121     q_next_dir_1  => next_direction_1,
122     q_p_state_0   => player_0_state,
123     q_p_state_1   => player_1_state);
124
125 counter: busy_counter port map (clk => clk,
126                                 global_reset => reset,
127                                 game_engine_reset => busy_counter_reset,
128                                 busy => busy,
129                                 busy_count => unsigned_busy_count);
130
131 -- outputs from the register to the graphics engine
132 position_0_vga  <= position_0;
133 position_1_vga  <= position_1;
134 direction_0_vga <= direction_0;
135 direction_1_vga <= direction_1;
136 player_state_0_vga <= player_0_state;
137 player_state_1_vga <= player_1_state;
138
139
140
141 updates: process (clk)
142 begin
143   if (clk'event and clk = '1') then
144     -- go to the reset state when the reset button is pressed
145     if (reset = '1') then
146       state <= reset_state;
147     -- go to the next state according to the FSM
148     else
149       state <= new_state;
150     end if;
151   end if;
152 end process;
153
154 create_next_state: process (state, new_state, reset, input, busy,
155                           read_memory, memory_ready, clk, unsigned_busy_count,
156                           direction_0, direction_1, next_direction_0, next_direction_1,
157                           position_0, position_1, next_position_0, next_position_1,
158                           wallshape_0, wallshape_1, read_memory_0, read_memory_1,
159                           player_0_state, player_1_state, e_position_0, e_position_1,
160                           e_wallshape_0, e_wallshape_1, e_read_memory_0, e_read_memory_1,
161                           e_next_position_0, e_next_position_1, e_direction_0,
162                           e_direction_1, e_next_direction_0, e_next_direction_1,
163                           e_player_0_state, e_player_1_state )

```

```

155 begin
156   case state is
157     when reset_state =>
158       -- in this state all the values are set to zero to reset
159       -- everything
160       state_vga           <= "000";
161       write_enable         <= '0';
162       write_memory        <= "00000000";
163       address             <= "0000000000";
164       go_to               <= '0';
165       busy_counter_reset  <= '0';
166       clear_memory        <= '0';

167       e_position_0         <= '0';
168       e_position_1         <= '0';
169       e_wallshape_0        <= '0';
170       e_wallshape_1        <= '0';
171       e_read_memory_0      <= '0';
172       e_read_memory_1      <= '0';
173       e_next_position_0    <= '0';
174       e_next_position_1    <= '0';
175       e_direction_0        <= '0';
176       e_direction_1        <= '0';
177       e_next_direction_0   <= '0';
178       e_next_direction_1   <= '0';
179       e_player_0_state     <= '0';
180       e_player_1_state     <= '0';

181       d_position_0         <= (others => '0');
182       d_position_1         <= (others => '0');
183       d_wallshape_0        <= (others => '0');
184       d_wallshape_1        <= (others => '0');
185       d_read_memory_0       <= (others => '0');
186       d_read_memory_1       <= (others => '0');
187       d_next_position_0     <= (others => '0');
188       d_next_position_1     <= (others => '0');
189       d_direction_0         <= (others => '0');
190       d_direction_1         <= (others => '0');
191       d_next_direction_0    <= (others => '0');
192       d_next_direction_1    <= (others => '0');
193       d_player_0_state      <= (others => '0');
194       d_player_1_state      <= (others => '0');

195       -- go to the state 'want_to_load' next
196       new_state <= want_to_load;
197
198       when want_to_load =>

```

```

201      -- let the memory module know that the memory has to be
202      -- cleared
203      state_vga          <= "000";
204      write_enable        <= '0';
205      write_memory        <= "00000000";
206      address            <= "0000000000";
207      busy_counter_reset <= '0';
208      go_to              <= '0';
209      clear_memory       <= '1';

210      e_position_0        <= '0';
211      e_position_1        <= '0';
212      e_wallshape_0        <= '0';
213      e_wallshape_1        <= '0';
214      e_read_memory_0      <= '0';
215      e_read_memory_1      <= '0';
216      e_next_position_0    <= '0';
217      e_next_position_1    <= '0';
218      e_direction_0        <= '0';
219      e_direction_1        <= '0';
220      e_next_direction_0   <= '0';
221      e_next_direction_1   <= '0';
222      e_player_0_state     <= '0';
223      e_player_1_state     <= '0';

224
225      d_position_0        <= (others => '0');
226      d_position_1        <= (others => '0');
227      d_wallshape_0        <= (others => '0');
228      d_wallshape_1        <= (others => '0');
229      d_read_memory_0      <= (others => '0');
230      d_read_memory_1      <= (others => '0');
231      d_next_position_0    <= (others => '0');
232      d_next_position_1    <= (others => '0');
233      d_direction_0        <= (others => '0');
234      d_direction_1        <= (others => '0');
235      d_next_direction_0   <= (others => '0');
236      d_next_direction_1   <= (others => '0');
237      d_player_0_state     <= (others => '0');
238      d_player_1_state     <= (others => '0');

239
240      -- go to the state 'loading_state' next to continue to
241      -- clear the memory
242      -- 'memory_ready' will still be '1', need to wait an extra
243      -- clockcycle
244      new_state <= loading_state;

when loading_state =>

```

```

245      -- continue to let the memory module know to clear the
246      -- memory and check when the memory is done
247      state_vga           <= "000";
248      write_enable        <= '0';
249      write_memory        <= "00000000";
250      address             <= "0000000000";
251      busy_counter_reset   <= '0';
252      go_to               <= '0';
253      clear_memory        <= '1';

254      -- initialize the position, direction and state of the
255      -- players
256      e_position_0         <= '1';
257      e_position_1         <= '1';
258      e_direction_0        <= '1';
259      e_direction_1        <= '1';
260      e_player_0_state    <= '1';
261      e_player_1_state    <= '1';

262      d_position_0         <= "01110111001";
263      d_position_1         <= "01110100100";
264      d_direction_0        <= "00";
265      d_direction_1        <= "00";
266      d_player_0_state    <= "10";
267      d_player_1_state    <= "10";

268      e_wallshape_0        <= '0';
269      e_wallshape_1        <= '0';
270      e_read_memory_0      <= '0';
271      e_read_memory_1      <= '0';
272      e_next_position_0    <= '0';
273      e_next_position_1    <= '0';
274      e_next_direction_0   <= '0';
275      e_next_direction_1   <= '0';

276      d_wallshape_0        <= (others => '0');
277      d_wallshape_1        <= (others => '0');
278      d_read_memory_0       <= (others => '0');
279      d_read_memory_1       <= (others => '0');
280      d_next_position_0    <= (others => '0');
281      d_next_position_1    <= (others => '0');
282      d_next_direction_0   <= (others => '0');
283      d_next_direction_1   <= (others => '0');

284      -- when the memory is finished go to the next state:
285      -- 'get_ready' otherwise stay in this state
286      if (memory_ready = '1') then

```

```

289         new_state <= get_ready;
290     else
291         new_state <= loading_state;
292     end if;
293
294     when get_ready =>
295         -- wait for the player to press the button in the right
296         -- direction: meaning they are ready to play
297         state_vga           <= "000";
298         write_enable        <= '0';
299         write_memory        <= "00000000";
300         address            <= "0000000000";
301         go_to              <= '0';
302         clear_memory       <= '0';
303
303         e_position_0        <= '0';
304         e_position_1        <= '0';
305         e_wallshape_0       <= '0';
306         e_wallshape_1       <= '0';
307         e_read_memory_0    <= '0';
308         e_read_memory_1    <= '0';
309         e_next_position_0   <= '0';
310         e_next_position_1   <= '0';
311         e_direction_0      <= '0';
312         e_direction_1      <= '0';
313         e_next_direction_0 <= '0';
314         e_next_direction_1 <= '0';
315         e_player_0_state   <= '0';
316         e_player_1_state   <= '0';
317
318         d_position_0        <= (others => '0');
319         d_position_1        <= (others => '0');
320         d_wallshape_0       <= (others => '0');
321         d_wallshape_1       <= (others => '0');
322         d_read_memory_0    <= (others => '0');
323         d_read_memory_1    <= (others => '0');
324         d_next_position_0   <= (others => '0');
325         d_next_position_1   <= (others => '0');
326         d_direction_0      <= (others => '0');
327         d_direction_1      <= (others => '0');
328         d_next_direction_0 <= (others => '0');
329         d_next_direction_1 <= (others => '0');
330         d_player_0_state   <= (others => '0');
331         d_player_1_state   <= (others => '0');
332
333         -- when player 0 is ready to play the next state is
334             'player_0_ready'

```

```

334      if (input(1 downto 0) = direction_0) then
335          new_state           <= player_0_ready;
336          -- when player 1 is ready to play the next state is
337          -- 'player_1_ready'
338      elsif (input(3 downto 2) = direction_1) then
339          new_state           <= player_1_ready;
340          -- when neither player is ready the next state is this
341          -- state
342      else
343          new_state           <= get_ready;
344      end if;
345
346      when player_0_ready =>
347          -- player 0 is ready to play
348          state_vga           <= "000";
349          write_enable         <= '0';
350          write_memory        <= "00000000";
351          address              <= "0000000000";
352          go_to                <= '0';
353          busy_counter_reset   <= '0';
354          clear_memory         <= '0';
355
356          -- change the player state of player 0 to let the VGA know
357          -- that player 0 is ready to play
358          e_player_0_state     <= '1';
359          d_player_0_state     <= "11";
360
361          e_position_0          <= '0';
362          e_position_1          <= '0';
363          e_wallshape_0         <= '0';
364          e_wallshape_1         <= '0';
365          e_read_memory_0       <= '0';
366          e_read_memory_1       <= '0';
367          e_next_position_0     <= '0';
368          e_next_position_1     <= '0';
369          e_direction_0          <= '0';
370          e_direction_1          <= '0';
371          e_next_direction_0    <= '0';
372          e_next_direction_1    <= '0';
373          e_player_1_state     <= '0';
374
375          d_position_0          <= (others => '0');
376          d_position_1          <= (others => '0');
377          d_wallshape_0         <= (others => '0');
378          d_wallshape_1         <= (others => '0');
379          d_read_memory_0        <= (others => '0');
380          d_read_memory_1        <= (others => '0');

```

```

378     d_next_position_0      <= (others => '0');
379     d_next_position_1      <= (others => '0');
380     d_direction_0          <= (others => '0');
381     d_direction_1          <= (others => '0');
382     d_next_direction_0     <= (others => '0');
383     d_next_direction_1     <= (others => '0');
384     d_player_1_state       <= (others => '0');

385
386 -- when player 1 is ready go to the state 'wait_state' if
387 -- player 1 is not ready the next state is this state
388 if (input(3 downto 2) = direction_1) then
389     new_state <= wait_state;
390 else
391     new_state <= player_0_ready;
392 end if;

393 when player_1_ready =>
394     -- player 1 is ready to play
395     state_vga            <= "000";
396     write_enable           <= '0';
397     write_memory          <= "00000000";
398     address               <= "0000000000";
399     go_to                 <= '0';
400     busy_counter_reset    <= '0';
401     clear_memory          <= '0';

402
403 -- change the player state of player 1 to let the VGA know
404 -- that player 1 is ready to play
405 e_player_1_state      <= '1';
406 d_player_1_state       <= "11";

407
408     e_position_0          <= '0';
409     e_position_1          <= '0';
410     e_wallshape_0         <= '0';
411     e_wallshape_1         <= '0';
412     e_read_memory_0       <= '0';
413     e_read_memory_1       <= '0';
414     e_next_position_0     <= '0';
415     e_next_position_1     <= '0';
416     e_direction_0         <= '0';
417     e_direction_1         <= '0';
418     e_next_direction_0    <= '0';
419     e_next_direction_1    <= '0';
420     e_player_0_state      <= '0';

421
422     d_position_0          <= (others => '0');
423     d_position_1          <= (others => '0');

```

```

423      d_wallshape_0      <= (others => '0');
424      d_wallshape_1      <= (others => '0');
425      d_read_memory_0    <= (others => '0');
426      d_read_memory_1    <= (others => '0');
427      d_next_position_0  <= (others => '0');
428      d_next_position_1  <= (others => '0');
429      d_direction_0       <= (others => '0');
430      d_direction_1       <= (others => '0');
431      d_next_direction_0 <= (others => '0');
432      d_next_direction_1 <= (others => '0');
433      d_player_0_state   <= (others => '0');

434
435      -- if player 0 is ready the next state is 'wait_state', if
436          -- player 0 is not ready the next state is this state
437      if (input(1 downto 0) = direction_0) then
438          new_state <= wait_state;
439      else
440          new_state <= player_1_ready;
441      end if;

442
443
444      when wait_state =>
445          -- wait for a certain amount of busy signal cycles before
446              -- going on
447          -- this determines how fast a player moves in the
448              -- playingfield
449          state_vga           <= "111";
450          write_enable         <= '0';
451          write_memory        <= "00000000";
452          address             <= "0000000000";
453          go_to               <= '0';
454          busy_counter_reset  <= '0';
455          clear_memory        <= '0';

456          e_position_0        <= '0';
457          e_position_1        <= '0';
458          e_wallshape_0       <= '0';
459          e_wallshape_1       <= '0';
460          e_read_memory_0     <= '0';
461          e_read_memory_1     <= '0';
462          e_next_position_0   <= '0';
463          e_next_position_1   <= '0';
464          e_direction_0       <= '0';
465          e_direction_1       <= '0';
466          e_next_direction_0 <= '0';
467          e_next_direction_1 <= '0';

```

```

467
468
469      d_position_0      <= (others => '0');
470      d_position_1      <= (others => '0');
471      d_wallshape_0     <= (others => '0');
472      d_wallshape_1     <= (others => '0');
473      d_read_memory_0   <= (others => '0');
474      d_read_memory_1   <= (others => '0');
475      d_next_position_0 <= (others => '0');
476      d_next_position_1 <= (others => '0');
477      d_direction_0     <= (others => '0');
478      d_direction_1     <= (others => '0');
479      d_next_direction_0 <= (others => '0');
480      d_next_direction_1 <= (others => '0');

481
482      e_player_0_state  <= '1';
483      d_player_0_state  <= "11";
484      e_player_1_state  <= '1';
485      d_player_1_state  <= "11";

486
487      -- when waited long enough go the next state: read_inputs,
488      -- otherwise keep waiting
489      if (unsigned( unsigned_busy_count) >= 16) then
490          new_state <= busy_reset;
491      else
492          -- new_state <= wait_state;
493          new_state <= busy_reset;
494      end if;

495
496      when busy_reset =>
497          -- reset the counter
498          busy_counter_reset    <= '1';

499
500          state_vga           <= "111";
501          write_enable         <= '0';
502          write_memory         <= "00000000";
503          address              <= "0000000000";
504          go_to                <= '0';
505          clear_memory         <= '0';

506
507          e_position_0         <= '0';
508          e_position_1         <= '0';
509          e_wallshape_0        <= '0';
510          e_wallshape_1        <= '0';
511          e_read_memory_0      <= '0';
512          e_read_memory_1      <= '0';

```

```

513     e_next_position_0      <= '0';
514     e_next_position_1      <= '0';
515     e_direction_0          <= '0';
516     e_direction_1          <= '0';
517     e_next_direction_0     <= '0';
518     e_next_direction_1     <= '0';
519     e_player_0_state       <= '0';
520     e_player_1_state       <= '0';

521
522     d_position_0          <= (others => '0');
523     d_position_1          <= (others => '0');
524     d_wallshape_0          <= (others => '0');
525     d_wallshape_1          <= (others => '0');
526     d_read_memory_0        <= (others => '0');
527     d_read_memory_1        <= (others => '0');
528     d_next_position_0       <= (others => '0');
529     d_next_position_1       <= (others => '0');
530     d_direction_0          <= (others => '0');
531     d_direction_1          <= (others => '0');
532     d_next_direction_0      <= (others => '0');
533     d_next_direction_1      <= (others => '0');
534     d_player_0_state        <= (others => '0');
535     d_player_1_state        <= (others => '0');

536
537     new_state <= read_inputs;

538
539     when read_inputs =>
540         -- read the inputs from the players and remember them
541         state_vga            <= "111";
542         write_enable          <= '0';
543         write_memory          <= "00000000";
544         address               <= "0000000000";
545         go_to                 <= '0';
546         busy_counter_reset    <= '0';
547         clear_memory          <= '0';

548
549         -- remember the values of the input of the players in
550         -- 'next_direction_#player'
551         e_next_direction_0     <= '1';
552         e_next_direction_1     <= '1';

553         d_next_direction_0     <= input(1 downto 0);
554         d_next_direction_1     <= input(3 downto 2);

555
556         e_position_0           <= '0';
557         e_position_1           <= '0';

```

```

558     e_wallshape_0      <= '0';
559     e_wallshape_1      <= '0';
560     e_read_memory_0    <= '0';
561     e_read_memory_1    <= '0';
562     e_next_position_0  <= '0';
563     e_next_position_1  <= '0';
564     e_direction_0      <= '0';
565     e_direction_1      <= '0';
566     e_player_0_state   <= '0';
567     e_player_1_state   <= '0';

568
569     d_position_0        <= (others => '0');
570     d_position_1        <= (others => '0');
571     d_wallshape_0       <= (others => '0');
572     d_wallshape_1       <= (others => '0');
573     d_read_memory_0     <= (others => '0');
574     d_read_memory_1     <= (others => '0');
575     d_next_position_0   <= (others => '0');
576     d_next_position_1   <= (others => '0');
577     d_direction_0        <= (others => '0');
578     d_direction_1        <= (others => '0');
579     d_player_0_state    <= (others => '0');
580     d_player_1_state    <= (others => '0');

581
582 -- go to the state 'wall_shape' next
583 new_state <= wall_shape;

584
585 when wall_shape =>
586     -- determine how the player went over a cell
587     state_vga           <= "111";
588     write_enable         <= '0';
589     write_memory         <= "00000000";
590     address              <= "000000000000";
591     go_to                <= '0';
592     busy_counter_reset   <= '0';
593     clear_memory         <= '0';

594
595     e_position_0        <= '0';
596     e_position_1        <= '0';
597     e_read_memory_0     <= '0';
598     e_read_memory_1     <= '0';
599     e_direction_0        <= '0';
600     e_direction_1        <= '0';
601     e_next_direction_0  <= '0';
602     e_next_direction_1  <= '0';

603
604     d_position_0        <= (others => '0');

```

```

605      d_position_1      <= (others => '0');
606      d_read_memory_0    <= (others => '0');
607      d_read_memory_1    <= (others => '0');
608      d_direction_0      <= (others => '0');
609      d_direction_1      <= (others => '0');
610      d_next_direction_0 <= (others => '0');
611      d_next_direction_1 <= (others => '0');

612
613      -- determine the wall shape on the cell of player 0
614      -- previous: left, next: left or previous: right, next:
615      --     right --> horizontal
616      if ((direction_0= "01") and (next_direction_0  ="01")) or
617          ((direction_0= "11") and (next_direction_0  ="11")) then
618          e_wallshape_0 <= '1';
619          d_wallshape_0 <= "001";
620          e_player_0_state <= '0';
621          d_player_0_state <= (others => '0');
622
623          -- previous: up, next: up or previous: down, next: down
624          --     ---> vertical
625
626      elsif ((direction_0= "00") and (next_direction_0  ="00"))
627          or ((direction_0= "10") and (next_direction_0  ="10"))
628      then
629          e_wallshape_0 <= '1';
630          d_wallshape_0 <= "010";
631          e_player_0_state <= '0';
632          d_player_0_state <= (others => '0');
633
634          -- previous: up, next: right or previous: left, next: down
635          --     ---> corner in the lower right
636
637      elsif ((direction_0= "00") and (next_direction_0  ="11"))
638          or ((direction_0= "01") and (next_direction_0  ="10"))
639      then
640          e_wallshape_0 <= '1';
641          d_wallshape_0 <= "110";
642          e_player_0_state <= '0';
643          d_player_0_state <= (others => '0');
644
645          -- previous: up, next: left or previous: right, next: down
646          --     ---> corner in the lower left
647
648      elsif ((direction_0= "00") and (next_direction_0  ="01"))
649          or ((direction_0= "11") and (next_direction_0  ="10"))
650      then
651          e_wallshape_0 <= '1';
652          d_wallshape_0 <= "101";
653          e_player_0_state <= '0';
654          d_player_0_state <= (others => '0');
655
656          -- previous: right, next: up or previous: down, next: left
657          --     ---> corner in the upper left
658
659      elsif ((direction_0= "11") and (next_direction_0  ="00"))

```

```

        or ((direction_0= "10") and (next_direction_0  ="01"))
        then
            e_wallshape_0 <= '1';
            d_wallshape_0 <= "100";
            e_player_0_state <= '0';
            d_player_0_state <= (others => '0');
            -- previous: down, next: right or previous: left, next: up
            --> corner in the upper right
    elseif ((direction_0= "10") and (next_direction_0 ="11"))
        or ((direction_0= "01") and (next_direction_0  ="00"))
        then
            e_wallshape_0 <= '1';
            d_wallshape_0 <= "111";
            e_player_0_state <= '0';
            d_player_0_state <= (others => '0');
            -- the player went back to where it came from and it
            collided with itself in the middle of a cell
    else
        e_wallshape_0 <= '0';
        d_wallshape_0 <= (others => '0');
        e_player_0_state <= '1';
        d_player_0_state <= "00";
    end if;

-- determine the wall shape on the cell of player 1
-- previous: left, next: left or previous: right, next:
-- right --> horizontal
if ((direction_1= "01") and (next_direction_1 ="01")) or
((direction_1= "11") and (next_direction_1  ="11")) then

    e_wallshape_1 <= '1';
    d_wallshape_1 <= "001";
    e_player_1_state <= '0';
    d_player_1_state <= (others => '0');
    -- previous: up, next: up or previous: down, next: down
    --> vertical
elseif ((direction_1= "00") and (next_direction_1 ="00"))
    or ((direction_1= "10") and (next_direction_1  ="10"))
    then
        e_wallshape_1 <= '1';
        d_wallshape_1 <= "010";
        e_player_1_state <= '0';
        d_player_1_state <= (others => '0');
        -- previous: up, next: right or previous: left, next: down
        --> corner in the lower right
elseif ((direction_1= "00") and (next_direction_1 ="11"))
    or ((direction_1= "01") and (next_direction_1  ="10"))

```

```

        then
673      e_wallshape_1 <= '1';
674      d_wallshape_1 <= "110";
675      e_player_1_state <= '0';
676      d_player_1_state <= (others => '0');
677      -- previous: up, next: left or previous: right, next: down
678      --> corner in the lower left
679      elsif ((direction_1= "00") and (next_direction_1 ="01"))
680          or ((direction_1= "11") and (next_direction_1 ="10"))
681          then
682              e_wallshape_1 <= '1';
683              d_wallshape_1 <= "101";
684              e_player_1_state <= '0';
685              d_player_1_state <= (others => '0');
686              -- previous: right, next: up or previous: down, next: left
687              --> corner in the upper left
688              elsif ((direction_1= "11") and (next_direction_1 ="00"))
689                  or ((direction_1= "10") and (next_direction_1 ="01"))
690                  then
691                      e_wallshape_1 <= '1';
692                      d_wallshape_1 <= "100";
693                      e_player_1_state <= '0';
694                      d_player_1_state <= (others => '0');
695                      -- previous: down, next: right or previous: left, next: up
696                      --> corner in the upper right
697                      elsif ((direction_1= "10") and (next_direction_1 ="11"))
698                          or ((direction_1= "01") and (next_direction_1 ="00"))
699                          then
700                              e_wallshape_1 <= '1';
701                              d_wallshape_1 <= "111";
702                              e_player_1_state <= '0';
703                              d_player_1_state <= (others => '0');
704                              -- the player went back to where it came from and it
705                              -- collided with itself in the middle of a cell
706              else
707                  e_player_1_state <= '1';
708                  d_player_1_state <= "00";
709                  e_wallshape_1 <= '0';
710                  d_wallshape_1 <= (others => '0');
711              end if;

712              -- determine the next position of player 0
713              e_next_position_0 <= '1';
714              if (next_direction_0 = "01") then      -- moves to the
715                  left, x is decreased with 1
716                  d_next_position_0(4 downto 0)  <=
717                      std_logic_vector(to_unsigned(to_integer(unsigned(position_0(4

```

```

        downto 0))) - 1, 5));
d_next_position_0(9 downto 5) <= position_0(9 downto 5);
d_next_position_0(10)      <= position_0(10);
elsif (next_direction_0 = "11") then -- moves to the
    right, x is increased with 1
d_next_position_0(4 downto 0) <=
    std_logic_vector(to_unsigned(to_integer(unsigned(position_0(4
        downto 0)))) + 1, 5));
d_next_position_0(9 downto 5) <= position_0(9 downto 5);
d_next_position_0(10)      <= position_0(10);
elsif (next_direction_0 <= "00") then -- moves up, y is
    decreased with 1
d_next_position_0(4 downto 0) <= position_0(4 downto 0);
d_next_position_0(9 downto 5) <=
    std_logic_vector(to_unsigned(to_integer(unsigned(position_0(9
        downto 5)))) - 1, 5));
d_next_position_0(10)      <= position_0(10);
else                      --moves down, y is increased with 1
    d_next_position_0(4 downto 0) <= position_0(4 downto 0);
    d_next_position_0(9 downto 5) <=
        std_logic_vector(to_unsigned(to_integer(unsigned(position_0(9
            downto 5)))) + 1, 5));
    d_next_position_0(10)      <= position_0(10);
end if;

-- determine the next position of player 1
e_next_position_1 <= '1';
if (next_direction_1 = "01") then -- moves to the
    left, x is decreased with 1
d_next_position_1(4 downto 0) <=
    std_logic_vector(to_unsigned(to_integer(unsigned(position_1(4
        downto 0)))) - 1, 5));
d_next_position_1(9 downto 5) <= position_1(9 downto 5);
d_next_position_1(10)      <= position_1(10);
elsif (next_direction_1 = "11") then -- moves to the
    right, x is increased with 1
d_next_position_1(4 downto 0) <=
    std_logic_vector(to_unsigned(to_integer(unsigned(position_1(4
        downto 0)))) + 1, 5));
d_next_position_1(9 downto 5) <= position_1(9 downto 5);
d_next_position_1(10)      <= position_1(10);
elsif (next_direction_1 <= "00") then -- moves up, y is
    decreased with 1
d_next_position_1(4 downto 0) <= position_1(4 downto 0);
d_next_position_1(9 downto 5) <=
    std_logic_vector(to_unsigned(to_integer(unsigned(position_1(9
        downto 5)))) - 1, 5));

```

```

736      d_next_position_1(10)      <= position_1(10);
737  else                      --moves down, y is increased with 1
738      d_next_position_1(4 downto 0) <= position_1(4 downto 0);
739      d_next_position_1(9 downto 5) <=
740          std_logic_vector(to_unsigned(to_integer(unsigned(position_1(9
741          downto 5))) + 1, 5));
740      d_next_position_1(10)      <= position_1(10);
741  end if;

742
743  -- go the state 'check_border' next
744  new_state <= check_border;

745
746 when check_border =>
747     -- check whether or not a player collided with the border
748     -- of the playing field
749     state_vga           <= "111";
750     write_enable         <= '0';
751     write_memory        <= "00000000";
752     address              <= "000000000000";
753     go_to                <= '0';
754     busy_counter_reset   <= '0';
755     clear_memory         <= '0';

756     e_position_0         <= '0';
757     e_position_1         <= '0';
758     e_wallshape_0        <= '0';
759     e_wallshape_1        <= '0';
760     e_read_memory_0      <= '0';
761     e_read_memory_1      <= '0';
762     e_next_position_0    <= '0';
763     e_next_position_1    <= '0';
764     e_direction_0        <= '0';
765     e_direction_1        <= '0';
766     e_next_direction_0   <= '0';
767     e_next_direction_1   <= '0';

768     d_position_0         <= (others => '0');
769     d_position_1         <= (others => '0');
770     d_wallshape_0        <= (others => '0');
771     d_wallshape_1        <= (others => '0');
772     d_read_memory_0      <= (others => '0');
773     d_read_memory_1      <= (others => '0');
774     d_next_position_0    <= (others => '0');
775     d_next_position_1    <= (others => '0');
776     d_direction_0        <= (others => '0');
777     d_direction_1        <= (others => '0');
778     d_next_direction_0   <= (others => '0');
779

```

```

780     d_next_direction_1      <= (others => '0');

781
782     -- check if player 0 collides with a border
783     if (((position_0(4 downto 0) = "00000") and
784          (next_direction_0 = "01")) or (next_position_0(4 downto
785          0) = "11110")) or (((position_0(9 downto 5) = "00000")
786          and (next_direction_0 = "00")) or (next_position_0(9
787          downto 5) = "11110"))then
788         e_player_0_state <= '1';
789         d_player_0_state <= "01";
790     else
791         e_player_0_state <= '0';
792         d_player_0_state <= (others => '0');
793     end if;

794
795     -- check if player 1 collides with a border
796     if (((position_1(4 downto 0) = "00000") and
797          (next_direction_1 = "01")) or (next_position_1(4 downto
798          0) = "11110")) or (((position_1(9 downto 5) = "00000")
799          and (next_direction_1 = "00")) or (next_position_1(9
800          downto 5) = "11110"))then
801         e_player_1_state <= '1';
802         d_player_1_state <= "01";
803     else
804         e_player_1_state <= '0';
805         d_player_1_state <= (others => '0');
806     end if;

807
808     -- the next state is 'want_to_read_0'
809     new_state <= want_to_read_0;

810
811     when want_to_read_0 =>
812         -- let the memory module know that we want to read
813         -- information from the next position of player 0
814         state_vga           <= "111";
815         write_enable        <= '0';
816         write_memory        <= "00000000";
817         address             <= next_position_0(9 downto 0);
818         go_to               <= '1';
819         busy_counter_reset <= '0';
820         clear_memory        <= '0';

821
822         e_position_0        <= '0';
823         e_position_1        <= '0';
824         e_wallshape_0       <= '0';
825         e_wallshape_1       <= '0';
826         e_read_memory_0    <= '0';

```

```

818      e_read_memory_1      <= '0';
819      e_next_position_0    <= '0';
820      e_next_position_1    <= '0';
821      e_direction_0       <= '0';
822      e_direction_1       <= '0';
823      e_next_direction_0   <= '0';
824      e_next_direction_1   <= '0';
825      e_player_0_state    <= '0';
826      e_player_1_state    <= '0';

827
828      d_position_0         <= (others => '0');
829      d_position_1         <= (others => '0');
830      d_wallshape_0        <= (others => '0');
831      d_wallshape_1        <= (others => '0');
832      d_read_memory_0      <= (others => '0');
833      d_read_memory_1      <= (others => '0');
834      d_next_position_0    <= (others => '0');
835      d_next_position_1    <= (others => '0');
836      d_direction_0        <= (others => '0');
837      d_direction_1        <= (others => '0');
838      d_next_direction_0   <= (others => '0');
839      d_next_direction_1   <= (others => '0');
840      d_player_0_state     <= (others => '0');
841      d_player_1_state     <= (others => '0');

842
843      -- the next state is 'read_memory_player_0'
844      new_state <= read_memory_player_0;
845
846      when read_memory_player_0 =>
847          -- read the data from the address of the next position of
848          -- player 0
849          state_vga           <= "111";
850          write_enable         <= '0';
851          write_memory        <= "00000000";
852          address             <= next_position_0(9 downto 0);
853          go_to               <= '0';
854          busy_counter_reset   <= '0';
855          clear_memory        <= '0';

856          e_position_0        <= '0';
857          e_position_1        <= '0';
858          e_wallshape_0       <= '0';
859          e_wallshape_1       <= '0';
860          e_read_memory_0     <= '0';
861          e_read_memory_1     <= '0';
862          e_next_position_0   <= '0';
863          e_next_position_1   <= '0';

```

```

864     e_direction_0      <= '0';
865     e_direction_1      <= '0';
866     e_next_direction_0      <= '0';
867     e_next_direction_1      <= '0';
868     e_player_1_state      <= '0';
869
870     d_position_0      <= (others => '0');
871     d_position_1      <= (others => '0');
872     d_wallshape_0      <= (others => '0');
873     d_wallshape_1      <= (others => '0');
874     d_read_memory_0      <= (others => '0');
875     d_read_memory_1      <= (others => '0');
876     d_next_position_0      <= (others => '0');
877     d_next_position_1      <= (others => '0');
878     d_direction_0      <= (others => '0');
879     d_direction_1      <= (others => '0');
880     d_next_direction_0      <= (others => '0');
881     d_next_direction_1      <= (others => '0');
882     d_player_1_state      <= (others => '0');
883
884 -- wait till the memory module is done with processing the
   information to go to the next state: 'want_to_read_1'
885 if (memory_ready = '1') then
886   -- when there is already data on the next position of
      player 0, player 0 collides against wall
887   if (read_memory = "00000000") then
888     e_player_0_state <= '0';
889     d_player_0_state <= (others => '0');
890   else
891     e_player_0_state <= '1';
892     d_player_0_state <= "00";
893   end if;
894   new_state <= want_to_read_1;
895 else
896   e_player_0_state <= '0';
897   d_player_0_state <= (others => '0');
898   new_state <= read_memory_player_0;
899 end if;
900
901 when want_to_read_1 =>
902   -- let the memory module know information needs to be read
      from the next position of player 1
903   state_vga      <= "111";
904   write_enable    <= '0';
905   write_memory    <= "00000000";
906   address        <= next_position_1(9 downto 0);
907   go_to          <= '1';

```

```

908     busy_counter_reset      <= '0';
909     clear_memory           <= '0';
910
911     e_position_0            <= '0';
912     e_position_1            <= '0';
913     e_wallshape_0           <= '0';
914     e_wallshape_1           <= '0';
915     e_read_memory_0          <= '0';
916     e_read_memory_1          <= '0';
917     e_next_position_0        <= '0';
918     e_next_position_1        <= '0';
919     e_direction_0            <= '0';
920     e_direction_1            <= '0';
921     e_next_direction_0       <= '0';
922     e_next_direction_1       <= '0';
923     e_player_0_state         <= '0';
924     e_player_1_state         <= '0';
925
926     d_position_0            <= (others => '0');
927     d_position_1            <= (others => '0');
928     d_wallshape_0           <= (others => '0');
929     d_wallshape_1           <= (others => '0');
930     d_read_memory_0          <= (others => '0');
931     d_read_memory_1          <= (others => '0');
932     d_next_position_0        <= (others => '0');
933     d_next_position_1        <= (others => '0');
934     d_direction_0             <= (others => '0');
935     d_direction_1             <= (others => '0');
936     d_next_direction_0       <= (others => '0');
937     d_next_direction_1       <= (others => '0');
938     d_player_0_state          <= (others => '0');
939     d_player_1_state          <= (others => '0');
940
941 -- the next state is 'read_memory_player_1'
942 new_state <= read_memory_player_1;
943
944 when read_memory_player_1 =>
945   -- read the data from the address of the next position of
946   -- player 1
947   state_vga                <= "111";
948   write_enable               <= '0';
949   write_memory               <= "00000000";
950   address                   <= next_position_1(9 downto 0);
951   go_to                     <= '0';
952   busy_counter_reset         <= '0';
953   clear_memory              <= '0';

```

```

954     e_position_0      <= '0';
955     e_position_1      <= '0';
956     e_wallshape_0      <= '0';
957     e_wallshape_1      <= '0';
958     e_read_memory_0      <= '0';
959     e_read_memory_1      <= '1';
960     e_next_position_0      <= '0';
961     e_next_position_1      <= '0';
962     e_direction_0      <= '0';
963     e_direction_1      <= '0';
964     e_next_direction_0      <= '0';
965     e_next_direction_1      <= '0';
966     e_player_0_state      <= '0';

967
968     d_position_0      <= (others => '0');
969     d_position_1      <= (others => '0');
970     d_wallshape_0      <= (others => '0');
971     d_wallshape_1      <= (others => '0');
972     d_read_memory_0      <= (others => '0');
973     d_read_memory_1      <= (others => '0');
974     d_next_position_0      <= (others => '0');
975     d_next_position_1      <= (others => '0');
976     d_direction_0      <= (others => '0');
977     d_direction_1      <= (others => '0');
978     d_next_direction_0      <= (others => '0');
979     d_next_direction_1      <= (others => '0');
980     d_player_0_state      <= (others => '0');

981
982 -- wait till the memory module is done with processing the
   information to go to the next state: 'check_collision'
983 if (memory_ready = '1') then
984   -- when there is already data on the next position of
      player 1, player 1 collides against wall
985   if (read_memory = "00000000") then
986     e_player_1_state <= '0';
987     d_player_1_state <= (others => '0');
988   else
989     e_player_1_state <= '1';
990     d_player_1_state <= "00";
991   end if;
992   new_state <= check_collision;
993 else
994   e_player_1_state <= '0';
995   d_player_1_state <= (others => '0');
996   new_state <= read_memory_player_1;
997 end if;
998
```

```

999      when check_collision =>
1000        -- check whether or not players collide with eachother
1001        state_vga          <= "111";
1002        write_enable        <= '0';
1003        write_memory       <= "000000000";
1004        address            <= "000000000000";
1005        go_to               <= '0';
1006        busy_counter_reset <= '0';
1007        clear_memory       <= '0';

1008
1009        e_position_0         <= '0';
1010        e_position_1         <= '0';
1011        e_wallshape_0        <= '0';
1012        e_wallshape_1        <= '0';
1013        e_read_memory_0      <= '0';
1014        e_read_memory_1      <= '0';
1015        e_next_position_0    <= '0';
1016        e_next_position_1    <= '0';
1017        e_direction_0        <= '0';
1018        e_direction_1        <= '0';
1019        e_next_direction_0   <= '0';
1020        e_next_direction_1   <= '0';

1021
1022        d_position_0         <= (others => '0');
1023        d_position_1         <= (others => '0');
1024        d_wallshape_0        <= (others => '0');
1025        d_wallshape_1        <= (others => '0');
1026        d_read_memory_0      <= (others => '0');
1027        d_read_memory_1      <= (others => '0');
1028        d_next_position_0    <= (others => '0');
1029        d_next_position_1    <= (others => '0');
1030        d_direction_0        <= (others => '0');
1031        d_direction_1        <= (others => '0');
1032        d_next_direction_0   <= (others => '0');
1033        d_next_direction_1   <= (others => '0');

1034
1035        -- when a player collides change its player state
1036        -- accordingly
1037        if (next_position_0 = next_position_1) then -- collide at
1038          -- eachother at middle of square
1039          e_player_0_state <= '1';
1040          e_player_1_state <= '1';
1041          d_player_0_state <= "00";
1042          d_player_1_state <= "00";
1043
1044        elsif (position_0 = next_position_1) and (position_1 =
1045          next_position_0) then -- collide at eachother at border
1046          e_player_0_state <= '1';

```

```

1043         e_player_1_state <= '1';
1044         d_player_0_state <= "01";
1045         d_player_1_state <= "01";
1046     elsif (position_0 = next_position_1) then -- player 1
1047         collides at the wall of player 0 made the previous time
1048         e_player_0_state <= '0';
1049         e_player_1_state <= '1';
1050         d_player_0_state <= (others => '0');
1051         d_player_1_state <= "00";
1052     elsif (position_1 = next_position_0) then -- player 0
1053         collides at the wall of player 1 made the previous time
1054         e_player_0_state <= '1';
1055         e_player_1_state <= '0';
1056         d_player_0_state <= "00";
1057         d_player_1_state <= (others => '0');
1058     else
1059         -- otherwise do not change the
1060         state of the player
1061         e_player_0_state <= '0';
1062         e_player_1_state <= '0';
1063         d_player_0_state <= (others => '0');
1064         d_player_1_state <= (others => '0');
1065     end if;
1066
1067     -- go to 'want_to_write_0' state next
1068     new_state<= want_to_write_0;
1069
1070     when want_to_write_0 =>
1071         -- check if writing is needed for player 0 and let the
1072         memory know
1073         state_vga           <= "111";
1074
1075         busy_counter_reset   <= '0';
1076         clear_memory        <= '0';
1077
1078         e_position_0          <= '0';
1079         e_position_1          <= '0';
1080         e_wallshape_0         <= '0';
1081         e_wallshape_1         <= '0';
1082
1083         e_read_memory_0       <= '0';
1084         e_read_memory_1       <= '0';
1085         e_next_position_0      <= '0';

```

```

1086      e_player_1_state      <= '0';
1087
1088      d_position_0          <= (others => '0');
1089      d_position_1          <= (others => '0');
1090      d_wallshape_0          <= (others => '0');
1091      d_wallshape_1          <= (others => '0');
1092          d_read_memory_0      <= (others => '0');
1093          d_read_memory_1      <= (others => '0');
1094          d_next_position_0     <= (others => '0');
1095          d_next_position_1     <= (others => '0');
1096          d_direction_0          <= (others => '0');
1097          d_direction_1          <= (others => '0');
1098          d_next_direction_0     <= (others => '0');
1099          d_next_direction_1     <= (others => '0');
1100      d_player_0_state        <= (others => '0');
1101      d_player_1_state        <= (others => '0');

1102
1103      -- when player 0 collided on border, there shall be no
1104      -- writing and the next state is 'want_to_write_1'
1105      if (player_0_state = "01") then
1106          write_enable            <= '0';
1107          write_memory            <= "00000000";
1108          address                <= "0000000000";
1109          go_to                  <= '0';
1110          new_state               <= want_to_write_1;
1111      -- if player 0 did not collide the next state will be
1112      -- 'write_memory_player_0'
1113      else
1114          write_enable            <= '1';
1115          write_memory(7 downto 3)  <= "00000";
1116          write_memory(2 downto 0)  <= wallshape_0;
1117          address                <= position_0(9 downto 0);
1118          go_to                  <= '1';
1119          new_state               <= write_memory_player_0;
1120      end if;

1121
1122      when write_memory_player_0 =>
1123          -- send to the memory module the wall shape of player 0 on
1124          -- the address of its position
1125          state_vga                <= "111";
1126          write_enable              <= '1';
1127          write_memory(7 downto 3)  <= "00000";
1128          write_memory(2 downto 0)  <= wallshape_0;
1129          address                <= position_0(9 downto 0);

```

```

1130
1131     e_position_0      <= '0';
1132     e_position_1      <= '0';
1133     e_wallshape_0     <= '0';
1134     e_wallshape_1     <= '0';
1135     e_read_memory_0   <= '0';
1136     e_read_memory_1   <= '0';
1137     e_next_position_0 <= '0';
1138     e_next_position_1 <= '0';
1139     e_direction_0     <= '0';
1140     e_direction_1     <= '0';
1141     e_next_direction_0 <= '0';
1142     e_next_direction_1 <= '0';
1143     e_player_0_state  <= '0';
1144     e_player_1_state  <= '0';

1145
1146     d_position_0      <= (others => '0');
1147     d_position_1      <= (others => '0');
1148     d_wallshape_0     <= (others => '0');
1149     d_wallshape_1     <= (others => '0');
1150     d_read_memory_0   <= (others => '0');
1151     d_read_memory_1   <= (others => '0');
1152     d_next_position_0 <= (others => '0');
1153     d_next_position_1 <= (others => '0');
1154     d_direction_0     <= (others => '0');
1155     d_direction_1     <= (others => '0');
1156     d_next_direction_0 <= (others => '0');
1157     d_next_direction_1 <= (others => '0');
1158     d_player_0_state  <= (others => '0');
1159     d_player_1_state  <= (others => '0');

1160
1161 -- wait until the memory is ready to go to the next state
1162     'want_to_write_1'
1163 if (memory_ready = '1') then
1164     new_state <= want_to_write_1;
1165 else
1166     new_state <= write_memory_player_0;
1167 end if;

1168 when want_to_write_1 =>
1169     -- check if writing is needed for player 0 and let the
1170     -- memory know
1171     state_vga          <= "111";
1172
1173     busy_counter_reset <= '0';
1174     clear_memory       <= '0';

```

```

1175      e_position_0      <= '0';
1176      e_position_1      <= '0';
1177      e_wallshape_0      <= '0';
1178      e_wallshape_1      <= '0';
1179      e_read_memory_0      <= '0';
1180      e_read_memory_1      <= '0';
1181      e_next_position_0      <= '0';
1182      e_next_position_1      <= '0';
1183      e_direction_0      <= '0';
1184      e_direction_1      <= '0';
1185      e_next_direction_0      <= '0';
1186      e_next_direction_1      <= '0';
1187      e_player_0_state      <= '0';
1188      e_player_1_state      <= '0';

1189
1190      d_position_0      <= (others => '0');
1191      d_position_1      <= (others => '0');
1192      d_wallshape_0      <= (others => '0');
1193      d_wallshape_1      <= (others => '0');
1194      d_read_memory_0      <= (others => '0');
1195      d_read_memory_1      <= (others => '0');
1196      d_next_position_0      <= (others => '0');
1197      d_next_position_1      <= (others => '0');
1198      d_direction_0      <= (others => '0');
1199      d_direction_1      <= (others => '0');
1200      d_next_direction_0      <= (others => '0');
1201      d_next_direction_1      <= (others => '0');
1202      d_player_0_state      <= (others => '0');
1203      d_player_1_state      <= (others => '0');

1204
1205      -- when player 1 collided on a border, there shall be no
1206      -- writing and the next state will be 'change_data'
1207      if (player_1_state = "01") then
1208          write_enable      <= '0';
1209          write_memory      <= "00000000";
1210          address          <= "0000000000";
1211          go_to            <= '0';
1212          new_state         <= change_data;
1213          -- if player 1 did not collide, the next state will be
1214          -- 'write_memory_player_1'
1215      else
1216          write_enable      <= '1';
1217          write_memory(7 downto 3)  <= "00001";
1218          write_memory(2 downto 0)  <= wallshape_1;
1219          address          <= position_1(9 downto 0);
1220          go_to            <= '1';
1221          new_state         <= write_memory_player_1;

```

```

1220         end if;
1221
1222     when write_memory_player_1 =>
1223         -- send to the memory module the wall shape of player 1 on
1224         -- the address of its position
1225         state_vga           <= "111";
1226         write_enable        <= '1';
1227         write_memory(7 downto 3) <= "00001";
1228         write_memory(2 downto 0) <= wallshape_1;
1229         address            <= position_1(9 downto 0);
1230         go_to               <= '0';
1231         busy_counter_reset <= '0';
1232         clear_memory       <= '0';
1233
1234         e_position_0        <= '0';
1235         e_position_1        <= '0';
1236         e_wallshape_0       <= '0';
1237         e_wallshape_1       <= '0';
1238         e_read_memory_0     <= '0';
1239         e_read_memory_1     <= '0';
1240         e_next_position_0   <= '0';
1241         e_next_position_1   <= '0';
1242         e_direction_0       <= '0';
1243         e_direction_1       <= '0';
1244         e_next_direction_0  <= '0';
1245         e_next_direction_1  <= '0';
1246         e_player_0_state    <= '0';
1247         e_player_1_state    <= '0';
1248
1249         d_position_0        <= (others => '0');
1250         d_position_1        <= (others => '0');
1251         d_wallshape_0       <= (others => '0');
1252         d_wallshape_1       <= (others => '0');
1253         d_read_memory_0     <= (others => '0');
1254         d_read_memory_1     <= (others => '0');
1255         d_next_position_0   <= (others => '0');
1256         d_next_position_1   <= (others => '0');
1257         d_direction_0       <= (others => '0');
1258         d_direction_1       <= (others => '0');
1259         d_next_direction_0  <= (others => '0');
1260         d_next_direction_1  <= (others => '0');
1261         d_player_0_state    <= (others => '0');
1262         d_player_1_state    <= (others => '0');
1263
1264         -- wait till the memory is finished before going to the
1265         -- next state 'change_data'
1266         if (memory_ready = '1') then

```

```

1265         new_state <= change_data;
1266     else
1267         new_state <= write_memory_player_1;
1268     end if;
1269
1270     when change_data =>
1271         -- change the data that is going to the graphics engine
1272         -- and update data in the register
1273         state_vga           <= "111";
1274         write_enable        <= '0';
1275         write_memory        <= "00000000";
1276         address             <= "0000000000";
1277         go_to               <= '0';
1278         busy_counter_reset <= '0';
1279         clear_memory        <= '0';
1280
1281         e_wallshape_0       <= '0';
1282         e_wallshape_1       <= '0';
1283         e_read_memory_0     <= '0';
1284         e_read_memory_1     <= '0';
1285         e_next_position_0   <= '0';
1286         e_next_position_1   <= '0';
1287         e_next_direction_0  <= '0';
1288         e_next_direction_1  <= '0';
1289         e_player_0_state    <= '0';
1290         e_player_1_state    <= '0';
1291
1292         d_wallshape_0       <= (others => '0');
1293         d_wallshape_1       <= (others => '0');
1294         d_read_memory_0     <= (others => '0');
1295         d_read_memory_1     <= (others => '0');
1296         d_next_position_0   <= (others => '0');
1297         d_next_position_1   <= (others => '0');
1298         d_next_direction_0  <= (others => '0');
1299         d_next_direction_1  <= (others => '0');
1300         d_player_0_state    <= (others => '0');
1301         d_player_1_state    <= (others => '0');
1302
1303         e_direction_0      <= '1';
1304         e_direction_1      <= '1';
1305
1306         d_direction_0      <= next_direction_0;
1307         d_direction_1      <= next_direction_1;
1308
1309         -- if player 0 collides against a border do not change its
1310         -- position, otherwise do
1311         if (player_0_state = "01") then

```

```

1310     e_position_0 <= '0';
1311     d_position_0 <= (others => '0');
1312 else
1313     e_position_0 <= '1';
1314     d_position_0 <= next_position_0;
1315 end if;
1316
1317 -- if player 1 collides against a border do not change its
1318 -- position, otherwise do
1319 if (player_1_state = "01") then
1320     e_position_1 <= '0';
1321     d_position_1 <= (others => '0');
1322 else
1323     e_position_1 <= '1';
1324     d_position_1 <= next_position_1;
1325 end if;
1326
1327 -- check_who_won is the next state
1328 new_state <= check_who_won;
1329
1330 when check_who_won =>
1331     -- check who won based on the player states
1332     state_vga          <= "111";
1333     write_enable        <= '0';
1334     write_memory        <= "00000000";
1335     address            <= "0000000000";
1336     go_to              <= '0';
1337     busy_counter_reset <= '0';
1338     clear_memory       <= '0';
1339
1340     e_position_0        <= '0';
1341     e_position_1        <= '0';
1342     e_wallshape_0       <= '0';
1343     e_wallshape_1       <= '0';
1344     e_read_memory_0    <= '0';
1345     e_read_memory_1    <= '0';
1346     e_next_position_0   <= '0';
1347     e_next_position_1   <= '0';
1348     e_direction_0      <= '0';
1349     e_direction_1      <= '0';
1350     e_next_direction_0 <= '0';
1351     e_next_direction_1 <= '0';
1352     e_player_0_state   <= '0';
1353     e_player_1_state   <= '0';
1354
1355     d_position_0        <= (others => '0');
1356     d_position_1        <= (others => '0');

```

```

1356     d_wallshape_0      <= (others => '0');
1357     d_wallshape_1      <= (others => '0');
1358     d_read_memory_0    <= (others => '0');
1359     d_read_memory_1    <= (others => '0');
1360     d_next_position_0  <= (others => '0');
1361     d_next_position_1  <= (others => '0');
1362     d_direction_0      <= (others => '0');
1363     d_direction_1      <= (others => '0');
1364     d_next_direction_0 <= (others => '0');
1365     d_next_direction_1 <= (others => '0');
1366     d_player_0_state   <= (others => '0');
1367     d_player_1_state   <= (others => '0');

1368
1369 -- if both players are still playing, go back to the
1370   'wait_state'
1370 if ((player_0_state = "11") and (player_1_state = "11"))
1371   then
1371     new_state <= wait_state;
1372 -- if only player 0 is still playing, player 0 won
1373 elsif (player_0_state = "11") then
1374   new_state <= player_0_won;
1375 -- if only player 1 is still playing, player 1 won
1376 elsif (player_1_state = "11") then
1377   new_state <= player_1_won;
1378 -- when both players collided, nobody won and it is a tie
1379 else
1380   new_state <= tie;
1381 end if;

1382
1383 when player_0_won =>
1384   -- player 0 won and tell that to the graphics engine
1385   state_vga          <= "010";
1386   write_enable        <= '0';
1387   write_memory        <= "00000000";
1388   address            <= "0000000000";
1389   go_to              <= '0';
1390   busy_counter_reset <= '0';
1391   clear_memory       <= '0';

1392
1393   e_position_0       <= '0';
1394   e_position_1       <= '0';
1395   e_wallshape_0      <= '0';
1396   e_wallshape_1      <= '0';
1397   e_read_memory_0    <= '0';
1398   e_read_memory_1    <= '0';
1399   e_next_position_0  <= '0';
1400   e_next_position_1  <= '0';

```

```

1401      e_direction_0      <= '0';
1402      e_direction_1      <= '0';
1403      e_next_direction_0      <= '0';
1404      e_next_direction_1      <= '0';
1405      e_player_0_state      <= '0';
1406      e_player_1_state      <= '0';

1407
1408      d_position_0      <= (others => '0');
1409      d_position_1      <= (others => '0');
1410      d_wallshape_0      <= (others => '0');
1411      d_wallshape_1      <= (others => '0');
1412      d_read_memory_0      <= (others => '0');
1413      d_read_memory_1      <= (others => '0');
1414      d_next_position_0      <= (others => '0');
1415      d_next_position_1      <= (others => '0');
1416      d_direction_0      <= (others => '0');
1417      d_direction_1      <= (others => '0');
1418      d_next_direction_0      <= (others => '0');
1419      d_next_direction_1      <= (others => '0');
1420      d_player_0_state      <= (others => '0');
1421      d_player_1_state      <= (others => '0');

1422
1423      -- stay in this state
1424      new_state <= player_0_won;
1425
1426      when player_1_won =>
1427          -- player 1 won and tell that to the graphics engine
1428          state_vga      <= "011";
1429          write_enable      <= '0';
1430          write_memory      <= "00000000";
1431          address      <= "0000000000";
1432          go_to      <= '0';
1433          busy_counter_reset      <= '0';
1434          clear_memory      <= '0';

1435
1436      e_position_0      <= '0';
1437      e_position_1      <= '0';
1438      e_wallshape_0      <= '0';
1439      e_wallshape_1      <= '0';
1440      e_read_memory_0      <= '0';
1441      e_read_memory_1      <= '0';
1442      e_next_position_0      <= '0';
1443      e_next_position_1      <= '0';
1444      e_direction_0      <= '0';
1445      e_direction_1      <= '0';
1446      e_next_direction_0      <= '0';
1447      e_next_direction_1      <= '0';

```

```

1448     e_player_0_state      <= '0';
1449     e_player_1_state      <= '0';

1450
1451     d_position_0          <= (others => '0');
1452     d_position_1          <= (others => '0');
1453     d_wallshape_0         <= (others => '0');
1454     d_wallshape_1         <= (others => '0');
1455     d_read_memory_0        <= (others => '0');
1456     d_read_memory_1        <= (others => '0');
1457     d_next_position_0      <= (others => '0');
1458     d_next_position_1      <= (others => '0');
1459     d_direction_0          <= (others => '0');
1460     d_direction_1          <= (others => '0');
1461     d_next_direction_0     <= (others => '0');
1462     d_next_direction_1     <= (others => '0');
1463     d_player_0_state       <= (others => '0');
1464     d_player_1_state       <= (others => '0');

1465
1466 -- stay in this state
1467 new_state <= player_1_won;
1468
1469 when tie =>
1470   -- both players lost and tell that to the graphics engine
1471   state_vga           <= "001";
1472   write_enable         <= '0';
1473   write_memory         <= "00000000";
1474   address              <= "0000000000";
1475   go_to                <= '0';
1476   busy_counter_reset    <= '0';
1477   clear_memory         <= '0';

1478
1479   e_position_0          <= '0';
1480   e_position_1          <= '0';
1481   e_wallshape_0         <= '0';
1482   e_wallshape_1         <= '0';
1483   e_read_memory_0        <= '0';
1484   e_read_memory_1        <= '0';
1485   e_next_position_0      <= '0';
1486   e_next_position_1      <= '0';
1487   e_direction_0          <= '0';
1488   e_direction_1          <= '0';
1489   e_next_direction_0     <= '0';
1490   e_next_direction_1     <= '0';
1491   e_player_0_state       <= '0';
1492   e_player_1_state       <= '0';

1493
1494   d_position_0          <= (others => '0');

```

```

1495      d_position_1          <= (others => '0');
1496      d_wallshape_0         <= (others => '0');
1497      d_wallshape_1         <= (others => '0');
1498      d_read_memory_0        <= (others => '0');
1499      d_read_memory_1        <= (others => '0');
1500      d_next_position_0      <= (others => '0');
1501      d_next_position_1      <= (others => '0');
1502      d_direction_0          <= (others => '0');
1503      d_direction_1          <= (others => '0');
1504      d_next_direction_0      <= (others => '0');
1505      d_next_direction_1      <= (others => '0');
1506      d_player_0_state       <= (others => '0');
1507      d_player_1_state       <= (others => '0');

1508
1509      -- stay in this state
1510      new_state <= tie;
1511  end case;
1512 end process;
1513 end behaviour;

```

VHDL/game\_engine-behaviour.vhd