Virtual Juggling

Chris Wilkens and David Rush 6.111 December 14, 2005

#### Abstract:

The Virtual Juggling Simulator is designed to allow a user to juggle virtual balls by moving their hands in the air. A camera is focused on a user whose hands are found by using a threshold for the input color. The hand's position, velocity and state (based on past and current velocities) are calculated and passed on to a Ball Manager. The Ball Manager determines each ball's new position, velocity and state determined on by their old states and the current inputs. The Ball Manager then passes on to the Output module if and where to display each of the 31 balls on the next frame and the balls are overlaid onto the raw image passed from the camera and displayed on a monitor. The balls' state includes if it is in play, in a hand, in the right hand or left hand, its direction and what its x and y velocity and position are. The hands can be in a throw or catch state. Button and switch inputs are wired from the labkit to reset system, add to or remove balls from the system and to set the gravity for the system. The functionalities of the system are controlled by the major Video Processor, Ball Manager, Output and Controller modules each with a number of submodules.

# **Table of Contents**

| Table of Contents                 | ii  |
|-----------------------------------|-----|
| List of Figures                   | iii |
| Overview                          |     |
| Module Structure                  |     |
| Camera Input Module               |     |
| Controller                        |     |
| Level to Pulse                    | 5   |
| Video Processor                   | 5   |
| Hand Detector                     | 6   |
| Hand Logic                        | 7   |
| Ball Manager                      |     |
| Block Random Access Memory (BRAM) |     |
| Physics                           |     |
| Display                           |     |
| Output Module                     |     |
| Aggregate Ball Sprite Memory      |     |
| Testing                           |     |
| Conclusion                        |     |
| Appendix: Verilog Source          |     |
|                                   |     |

# List of Figures

| Figure 1: Screenshot of the Simulator       | . 2 |
|---------------------------------------------|-----|
| Figure 2: System Block Diagram              |     |
| Figure 3: Controller Simulation Results     |     |
| Figure 4: Level to Pulse simulation results | . 5 |
| Figure 5: Video Processor Block Diagram     |     |
| Figure 6: Ball Manager Simulation Results   |     |
| Figure 7: Ball Manager Block Diagram        |     |
| Figure 8: Output Block Diagram              | 15  |

### Overview

The Juggling Simulator is designed so that a user can juggle virtual balls on a screen by moving their hands in space. The user stands in front of a camera with their hands exposed and they are found by taking the center of mass of the red pixels within the screen. The video assumes that there is one hand on each side of the screen so it finds the center of mass of the red pixels for each side of the screen and displays a small green square where it finds each hand so that their locations can be seen by the user. The pixels that are considered red are found by setting a threshold value to compare to the incoming pixels. Flesh comes up as red as do most warm colors so it is important to have a dark background and all flesh but the hands covered to reduce noise within the system.

As the hands move around their positions are updated each frame which is 60 times per second. From their current position and past positions their velocity and acceleration are calculated and from that throw and catch signals are generated for the hands based on empirically found threshold values. If the hand has accelerated upwards and decelerated beyond a threshold value it is said to be in a throw state for one frame and will release one ball from the hand at its current position and velocity if is holding a ball. If the hand is not moving up it is said to be in a catch state and will catch a ball if the ball is moving down and is in the proximity of the hand. The number of balls a hand can catch is only limited by the number of balls in the system.

Once a ball has been released from the hand it is governed by the laws of virtual physics. The new ball position is calculated each frame by using its current velocity. Each frame the x velocity remains constant unless the ball collides with a side edge of the screen in which case it bounces off with one half of the impact x velocity. If the ball hits the bottom edge of the screen it rebounds with the same y velocity in the opposite direction. If the ball collides the top edge of the screen the new y velocity is set to one pixel per frame in the downward direction. Each frame the y velocity is updated by subtracting the current gravity. This gives the ball acceleration in the downward direction.

The user can modify the juggling experience by pressing buttons and setting switches on the labkit. If the user presses the enter button the system is set to its default setting with three balls in the system- two in the right hand and one in the left. Balls can be added or removed from the system by pressing the up and down buttons. The minimum number of balls is zero and the maximum is 31. When added the balls are placed alternately in the right and left hands. Switches five through zero set the gravity for the system with the high order switches corresponding to the more significant bits. When switch seven is on all the pixels that are determined to be red are displayed as red on the screen otherwise the video output is simply that of the input. With this feature it is easy to identify sources of noise and more easily see where the hand is in relation to the screen. When on, switch six tints each half the screen based on what state each hand is in. If the half of the screen is tinted blue the hand is in a catch state, if gray it is neither in a catch or throw state and when it is tinted green it is in a throw state.



**Figure 1: Screenshot of the Simulator** 

This shot shows David juggling 32 balls in the simulator. One can clearly identify his hands (red) and the 32 cyan balls. Based on the tinting of the screen, one can see that the left hand is not in a catch state, while the right hand is.

# **Module Structure**

The main module of the simulator is the VIRTUAL\_JUGGLING module. This model receives the VGA signal of the camera image and performs all necessary processing before the signal is sent to the output. To accomplish this, it contains instantiations of the major blocks of the system.

Within this module are many wires that will henceforth be grouped for simplicity. These wires come in three basic flavors: video signals, 32-bit vectors, and narrow logic signals. The video signals are composed of three 8-bit color channels, four single-bit control signals (sync, blank, hsync, vsync), and 11 and 10 bit *hcount* and *vcount* signals. In general, all position and velocity measurements are carried as 32-bit vectors in which the top 16 bits code the 'x' value and the bottom 16 bits code the 'y' value. Each of these 16 bit values is encoded as a signed, two's-complement value with 5 fractional bits. Finally, there are a few miscellaneous signals that represent their own unique codes. The major modules of the system are shown below.





## Camera Input Module

(by 6.111 staff and Chris Wilkens)

The first block in the video pipeline translates the camera's input signal into a VGA signal. It decodes NTSC video from the camera and buffers it to the ZBT memory, from which it is read out and formed into a VGA signal. This implementation uses the ZBT buffering and output provided by the 6.111 staff with a few small modifications. First, on the output end we downsample the image such that a pixel from the camera occupies four pixels of the 1024x768 XGA display. This allows us to fill the entire output frame with video from the camera. One advantageous result of this fact is that we need not read from the ZBT frame buffer as frequently, which enables the second modification. In order to do color comparisons, we modify the existing implementation to buffer 18 bits of information for each pixel. This gives us 8 bits of the Y and Cr values and 2 bits of the Cb (the Cb is not necessary for our color detection.) Since the Video Processor does color detection based on the Y and Cr signals, we keep these signals intact instead of producing reasonable RGB values at the output of this module. Once this

module has generated the proper video signals, they are subsequently sent to the Video Processor.

# Controller

(by David Rush)

The Controller module takes in a number of signals from the labkit and the video processor and outputs signals synchronized to *new\_frame\_sync* that can be used by the ball manager.

The inputs to the Controller are *clk*, *new\_frame\_raw*, *reset\_raw*, *add\_ball\_raw*, *dec\_ball\_raw* and *gravity\_raw*. The clock runs at 65MHZ and comes from the labkit. The one bit *new\_frame\_raw* signal comes from the Video Processing module and signals that a new frame has started. The *reset\_raw* is a one bit inverted (so pressed is high and depressed is low) and debounced button from the labkit as are the *add\_ball\_raw* and *dec\_ball\_raw* signals. The *gravity\_raw* is an eight bit signal from the debounced switches on the labkit.

The outputs of the Controller are *new\_fram\_sync*, *reset\_sync*, *add\_ball\_sync*, *dec\_ball\_sync* and *gravity\_sync*. They are each one bit except for the eight bit *gravity\_sync* signal and are all sent into the Ball Manager to be used by the Physics module and Display module.



Figure 3: Controller Simulation Results

The controller receives signals at different times during the frame and synchronizes all outputs with *new\_frame\_sync* 

Within the Controller each of the output signals are created using the input signals. Within an always block that triggers at each positive edge of the 65 MHZ clock a number of signals are assigned using non blocking assignments. The *new\_frame\_sync* is assigned to *new\_frame\_raw* and *gravity\_sync* is assigned to *gravity\_raw* if *new\_frame\_raw* or it remains *gravity\_sync*. The *reset\_sync* signal is set to one if *new\_frame\_raw* and *reset\_hold* are high otherwise zero. The *reset\_hold* signal is assigned to one if *reset\_raw* is high otherwise it is set to zero if *new\_frame\_raw* is high otherwise it is set to zero if *new\_frame\_raw* is high otherwise it is left *reset\_hold*. The process creates a *reset\_sync* signal that is set to one

for one clock cycle synchronized with *new\_frame\_sync* if the reset button was pressed any time during the preceding frame. The same process was used to create the *add\_ball\_sync* and *dec\_ball\_sync* signals except the \_hold signals were replaced with *add\_ball\_hold* and *dec\_ball\_hold*. Also for the assignment of *add\_ball\_hold* and *dec\_ball\_hold* the analogous *reset\_raw* signal was replaced with an *add\_ball\_pulse* and a *dec\_ball\_pulse* which came from sending the *add\_ball\_raw* and *dec\_ball\_raw* signals into their own Level to Pulse module instances to return the *add\_ball\_pulse* and *dec\_ball\_pulse* signals that are high for only 1 clock cycle regardless of how long the buttons were held down. Registers were created to hold values for each of the signals that were assigned within the always block.

#### Level to Pulse

The level to pulse takes in an input that is high for any number of cycles and outputs a signal that is high for only the first clock edge after the input signal went high.

The inputs are *clk* and *level\_in* and the output is *pulse\_out*. At each positive edge of the clock *level\_hold* is assigned to *level\_in* and pulse out is assigned to the combination of *level\_in* and not *level\_hold*. A register is used to store each *level\_hold* and *pulse\_out*.



**Figure 4: Level to Pulse simulation results** The level to pulse receives a level input and returns a one clock pulse

## Video Processor

(by Chris Wilkens)

The VJ\_VIDEO\_PROCESSOR module processes the VGA signal from the camera modules and determines the location and action of the juggler's hands. It determines the position and velocity information and forwards them to the appropriate modules. When the positions are ready for a given frame, it raises the *new\_frame* signal to notify the ball manager that it can begin its physics calculations.



**Figure 5: Video Processor Block Diagram** This diagram shows the modules of the Video Processor.

As seen in the block diagram, the specific implementation of the video processor is composed of two major parts: the hand detector, which locates the hands in the image, and the logic parser, which translates a series of positions into action signals that can be sent to the ball manager. In addition to providing the proper signals to these modules, the video processor modifies the output by tinting the flesh-colored pixels red. Finally, the *new\_frame* signal is simply the AND of the *logic\_ready* signals from the two hand logic modules.

#### Hand Detector

The hand detector module of the video processor watches the streaming VGA signal to calculate the location of the hands. Abstracted as a black box, this module takes the VGA color signal as input and returns the positions of the hands in the frame when the frame has been completely streamed. For each pixel, it also raises the *hand\_pixel* 

signal whenever the current pixel belongs to the hand (to allow the video processor to change its color.) In order to signal other components of the video processor, a *pos\_ready* pulse is raised for one clock cycle when the hand positions are ready.

For our implementation, we define the position of the left hand to be the center of mass of all flesh-colored pixels in the left half of the screen (or right half for the right hand.) Each coordinate of each hand is calculated by keeping a running sum of the appropriate coordinate values along with a count of the number of elements in the sum. Once the frame has passed, these two numbers in the sums are divided and the result becomes the appropriate coordinate of the hand. This is accomplished with four instances of a weighted average module (one for each coordinate to be calculated.) Due to timing delays through the divide module, it takes 32 cycles after the end of the frame in order to perform the division. Thus, in order to ensure that the process has ample time to complete, we wait many clock cycles until the vertical scan reaches line 770 before raising the *pos\_ready* signal. The output position is registered when *pos\_ready* is raised, and the internal sums are reset on the vertical sync signal.

#### Hand Logic

Two instances of the Hand Logic module sit beside the hand detector in the video processor. They watch the hand positions to determine whether or not the hand is throwing or catching. Abstracted, each module takes the new position and *pos\_ready* signal and produces the hand velocity and logic signals. The hand logic signals consist of two bits, a catch bit and a throw bit (*hand\_logic*[1] is throw, *hand\_logic*[0] is catch.) The catch bit will be high whenever the hand is in a state in which the system deems that the hand could be catching, leaving the details of the catch timing up to the ball manager. In contrast, the throw signal is a pulse held high for only a single frame. When these values are computed, the module raises a *logic\_ready* signal to notify others that the logic has been calculated.

This is, perhaps, the most interesting part of the video processor because it has the most room for modifications. There are an infinite number of ways one could determine a throw or a catch, but we chose a fairly simple implementation. First, we decided that if the hand was moving up, then it wasn't catching a ball. This seemed reasonable, though we realized that it isn't entirely true in real life. That said, we set a threshold (slightly positive to accommodate noise) for which the hand would be considered catching if its velocity was under that value. Another aspect we noted was that the positions tended to fluctuate due to noise. To compensate for this, we maintained a history of the last four velocities and took the average in order to compute the "actual" velocity of the juggler's hand.

The throw signal was more complicated. If we assumed that the juggler was not holding the balls, then a throw would occur by conservation of momentum whenever the hand decelerated. Thus, we decided to locate throws based on the acceleration of the hand. As with the velocity, we buffered the last four accelerations to get an average and then put a threshold on that average. Our final dilemma was that the throws would tend to "bounce," just like a metal button. For this, we effectively debounced the throw signal by specifying that the hand couldn't throw again until it had been out of the acceleration region for six consecutive frames. This effectively ensured that we only signaled one throw per movement from the juggler, and the appropriate output signal was raised for a single frame.

Ultimately, these values required one clock cycle to produce, so the hand logic module introduces a one cycle delay after the *pos\_ready* signal before all the hand information is ready.

#### **Ball Manager**

The Ball Manager takes in signals from the Video Processor and Controller and based on those signals and the balls' current states determines what to do next with the balls and then it sends to the output module if and where to display the balls. The ball manager is composed of three submodules and a dual port 70 bit by 32 address Block RAM (BRAM).

The inputs to the Ball Manager are a 65 MHZ clock from the labkit, a reset, an  $add\_ball$ , a  $dec\_ball$ , and an eight bit gravity signal from the Controller all synchronized to a *new\_frame* signal which also comes from the controller. There are also signals from the video processor which are *left\_throw*, *right\_throw*, *l\_h\_catch*, *r\_h\_catch*, *l\_h\_y*, *l\_h\_x*, *l\_h\_y\_vel*, *l\_h\_x\_vel*, *r\_h\_y*, *r\_h\_x*, *r\_h\_y\_vel*, and *r\_h\_x\_vel*. The clock determines the cycles per second of the system. The one bit reset signal returns the ball manager to its default state. The one bit *add\_ball* and *dec\_ball* signals add and remove balls from the system with a minimum of zero and a maximum of 31 balls. The eight bit gravity signal determines how fast to balls accelerate in the downward direction. The one bit *new\_frame* signals to the system that a new frame is starting and all the balls' states need to be recalculated. The *left\_throw*, *right\_throw*, *l\_h\_catch* and *r\_h\_catch* are each one bit signals indicating that the left and right hands are in a throw or catch state for that frame. The signals *l\_h\_y*, *l\_h\_x*, *l\_h\_y\_vel*, *l\_h\_x\_vel*, *r\_h\_y*, *r\_h\_x*, *r\_h\_y\_vel*, and *r\_h\_x\_vel* are each 16 bit signed values and give the left and right hand's y position, x position, y velocity and x velocity with new values being supplied each frame.

The outputs of the Ball Manager are all sent to the output module and are *display\_enb*, *ball\_pos\_out*, *ball\_number*, and *write\_request*. The one bit *display\_enb* signal indicates that the ball for which information is currently being sent is to be displayed. The *ball\_pos\_out* is a 32 bit signals with the x position in the upper 16 bits and the y position in the lower 16 bits. The *ball\_number* is a 5 bit signal indicates that the ball manager wants the information it is sending to be taken by the output module. The output module can be thought of as a memory with the *write\_request* signal equivalent to the write enable and the other signals the values that are to be stored in the memory.



Figure 6: Ball Manager Simulation Results

The ball manager takes in all its inputs and sends values to the output module at the appropriate times. After a throw between clock cycles 150 and 300 the outputs *write\_request* as well as display enable and a ball position are given at clock number 397 for ball number zero since it was thrown from the right hand and is in the air and thus needs to be displayed by the display module

Within the Ball Manager the Physics and Display modules are instantiated and wired together. The *clock*, *reset*, *add\_ball*, *dec\_ball*, *gravity*, *new\_frame*, *left\_throw*, *right\_throw*, *l\_h\_catch*, *r\_h\_catch*, *l\_h\_y*, *l\_h\_x*, *l\_h\_y\_vel*, *l\_h\_x\_vel*, *r\_h\_y*, *r\_h\_x*, *r\_h\_y\_vel*, and *r\_h\_x\_vel* signals are all wire directly to up as inputs to the Physics module. The 70 bit *ball\_info\_in* from the output of the BRAM is also wired as an input of the Physics module. The outputs of the Physics module are the 70 bit *ball\_info\_out*, five bit *count\_addr* and one bit *wea*, which are wired up to the BRAM, and the one bit *done\_calc* which is sent as an input to the Display module.

The inputs of the Display module are wired up to the 65 MHZ clock. The one bit reset and *new\_frame* are taken directly from the inputs to the Ball Manager and the one bit *done\_calc* is wired from the output of the Physics module and 70 bit *info\_from\_bram* is wired to the b port of the dual port BRAM. The outputs of the Display module are the five bit *bram\_addr\_b* which is wired to the "b" port address of the BRAM, the 32 bit *ball\_pos\_out* which is wire directly to the Ball Manager output as are the one bit *display\_enb*, five bit *ball\_number*, and one bit *write\_request*.

The 70 bit wide by 32 address deep Block RAM (BRAM) is wired within the Ball Manager as well. Both the "a" and "b" clocks are wired to the 65 MHZ clock. The a port address is wired to five bit *count\_addr* from the Physics module and the 70 bit data input is wired to the *ball\_info\_out* (out of the Physics module). The 70 bit "a" output is wired to *ball\_info\_out* and sent to the Physics module. The five bit "b" address is wired to the *bram\_addr\_b* from the Display module and the 70 bit "b" output is wired to the *info\_from\_bram* that is given to the Display module.



**Figure 7: Ball Manager Block Diagram** This diagram shows the smaller modules of the ball manager.

#### **Block Random Access Memory (BRAM)**

The dual port Block Random Access Memory stores the state of each ball in its 70 bit wide by 32 location memory. The "a" port is read/write and is used by the Physics module to retrieve ball data from the BRAM and store the new ball data back into it. The "b" port is read only and is used by the Display module to take the ball data and extract the current position and state and send it onto the output module.

#### Physics

The Physics module is the biggest and most complex part of the ball controller. The calculations for each ball's position, velocity and state are done within this module and results are stored in a dual port 70 bit by 32 location BRAM. The Physics module takes in inputs from the Controller, the Video Processor and the BRAM and it outputs data to the BRAM and a signal indicating the calculations for that frame are done. The Controller inputs (wired in through the Ball Manager) control whether the system should be reset, a new frame should be processed whether a ball should be added or removed from the system and what gravity is. Video Processor inputs (also wired in through the Ball Manager) give the left and right hand's x and y positions and velocities as well as the left and right hand throw and catch signals. The input from the BRAM gives each ball's current state, position and velocity. The outputs determine if, what and where in the BRAM the ball data should be stored as well as indicate if the Physics module is done with its calculations and storage for the frame.

The inputs to the Physics module are a 65 MHZ clock from the labkit, a *reset*, an add\_ball, a dec\_ball, and an eight bit gravity signal from the Controller all synchronized to a *new\_frame* signal which also comes from the controller. There are also signals from the video processor which are *left\_throw*, *right\_throw*, *l\_h\_catch*, *r\_h\_catch*, *l\_h\_y*,  $l_h_x$ ,  $l_h_y_vel$ ,  $l_h_x_vel$ ,  $r_h_y$ ,  $r_h_x$ ,  $r_h_y_vel$ , and  $r_h_x_vel$ . The final input is *ball\_info\_in* from the BRAM. The clock determines the cycles per second of the system. The one bit reset signal returns the Physics module to its default state. The one bit add ball and dec ball signals add and remove balls from the system with a minimum of zero and a maximum of 31 balls. The eight bit gravity signal determines how fast to balls accelerate in the downward direction. The one bit *new\_frame* signals to the system that a new frame is starting and all the balls' states need to be recalculated. The *left\_throw*, *right\_throw*, *l\_h\_catch* and *r\_h\_catch* are each one bit signals indicating that the left and right hands are in a throw or catch state for that frame. The signals l h y, l h x,  $l_h_y_vel$ ,  $l_h_x_vel$ ,  $r_h_y$ ,  $r_h_x$ ,  $r_h_y_vel$ , and  $r_h_x_vel$  are each 16 bit signed values and give the left and right hand's y position, x position, y velocity and x velocity with new values being supplied each frame. The 70 bit *ball\_info\_in* from the BRAM is contains each balls' state, position and velocity and is used to determine the new state, position and velocity in combination with the inputs.

The outputs for the Physics module are *ball\_info\_out, count\_addr, done\_calc* and *wea*. The 70 bit *ball\_info\_out* signal is the new position, velocity and state of each ball and the *count\_addr* is the current ball that is being updated and written back into memory at the address corresponding to it's ball number. The *done\_calc* signal is a one bit signal to indicate to the display module that all the physics for the frame has been calculated. The *wea* is the write enable for the BRAM.

A number of parameters were used to simplify expressions and keep the code from being cluttered with numbers. They are *ball\_w*, *ball\_h*, *hand\_h*, *hand\_w*, *screenw*, *screenh*, *ball\_right*, *ball\_left*, *ball\_down*, *ball\_up*, *read\_state* and *write\_state*. The first six are just the sizes of the balls, hands and screen and can be changed to optimize the system. Bigger hands made it easier to catch. Bigger balls meant they hit walls sooner an different screen sizes would have affected the display. The *ball\_* parameters are the states of the balls when moving around when in play and not in the hands. The *\_states* signal if the system is in a read state (reading from BRAM) or write state (writing to BRAM).

The first thing the system does is parse the incoming 70 bit *ball\_info\_in* and assign *in\_play\_in*, *in\_hand\_in*, *in\_left\_in*, *in\_right\_in*, *y\_dir\_in*, *x\_dir\_in*, *ball\_y\_in*, *ball\_x\_in*, *y\_vel\_in*, and *x\_vel\_in* based on those signals. The first six signals are each one bit and are stored in the high six bits of the incoming signal. The remaining signals

are each 16 bits and are stored in the given order within the *ball\_info\_in*. The Physics module takes these signals as input and creates new values for each *in\_play\_out*, *in\_hand\_out* and so on and appends them together in a 70 bit *ball\_info\_out* that is stored back into the BRAM for each ball.

On the positive edge of the 65 MHZ clock the always block is evaluated. If there is a *new\_frame*, *done\_calc*, *count\_addr*, *wea* and *read\_hold* are set to zero. The read hold signal holds the system in the read state for two cycles so the data is retrieved and assigned to the ball input values before going into the write state. On new\_frame *count divider* is set to three. Count divider is a signal that counts to 2 and each time it reaches two it is set back to zero and the *count\_addr* is incremented (each ball takes three clock cycles in the Physics module and this ensures that each ball gets three cycles). Setting the *count\_divider* to three here means that when the first time it incremented after a reset it will return to the zero count that it needs to start with after its initial increment. Also on the new frame, reset level is set to the reset input, left throw level is set to *left\_throw, right\_throw\_level* is set to *right\_throw, add\_ball\_level* is set to *add\_ball,* and dec\_ball\_level is set to dec\_ball, this ensures that each of those signals values are valid for the entire frame. The *which* hand is also flipped if the *add* ball signal is high. This signal determines which hand each new ball is placed into (alternating each time a ball is added). If it's not a *new\_frame* then if the *count\_addr* signal has reached 31 then the *done\_calc* signal is set high and no more physics is calculated that frame. Else if *done\_calc* is not high the *count\_divider* is incremented or set back to zero if it equals two, the *count addr* is incremented if *count divider* is equal to two or it remains unchanged and the *read\_or\_write* case statement is evaluated. If the system is in the read state the read hold bit is flipped, wea is set to zero and read or write is set to read\_hold. This means that after two cycles in the read state it will enter the write state.

In the write state the state *read\_or\_write* is set back to the *read\_state* and *wea* is set to one since at the rising edge of the next clock there will be new data ready to write into the BRAM.

The rest of the write state within the Physics module is a large if else block that assigns the output values that are put assigned into *ball\_info\_out* and stored back into the BRAM. The overview is that if the system is being reset put balls zero and one in the right hand and ball two in the left and leave the rest out of the system. If the ball is in play then if the *dec\_ball\_level* is high take the ball out of play and set *dec\_ball\_level* back to zero so no more balls get removed. Otherwise if the ball is in a hand if it's in the right hand is in a throw state and a fall hasn't already been thrown (*right\_thrown* signal is low) then have the ball in play, don't have it in a hand, assign it the hand's current position, velocity and direction and raise the *right\_thrown* signal so no more balls are thrown this frame out of the right hand. The analogous case is true for the left hand. Otherwise if the ball is in a hand and no hand is throwing, assign all the outputs to the original inputs since nothing is changing for them this frame.

If the ball is not in a hand but is in play and is not removed leave it in play and check to see if it has been caught. It is caught if the right or left hand is in a catch state, the y direction of the ball is down and the ball is within a given rectangle area above the center of the hand that can be set by the hand width and length. It is placed in the hand that it is caught in if it is caught otherwise the new position velocity and direction of ball are

calculated. In the in play, not removed, not caught state the x and y velocities and positions are calculated separately depending on which direction the ball is moving. If the *x\_dir\_in* is *ball\_right* the balls x postion if it's input position plus it's input velocity. The new x velocity remains the same unless it comes within a distance of two times the velocity of the right wall and then it is halved and the x direction is also switched. If the ball is moving left the new *ball\_x\_out* is the *ball\_x\_in* minus the *x\_vel\_in* and the velocity and direction remain the same unless the ball is within two times the x vel in in which case the x direction is flipped and the velocity is halved. The y direction and velocity are done similarly except if the ball is moving down the balls position is the old position plus the velocity and the direction is the same unless it comes within two times the y velocity of the bottom of the screen in which case the direction changes and the velocity does not. Otherwise the velocity is updated by adding gravity to it each frame. If the ball is moving up the new position is the input position minus the input velocity. The direction remains the same unless the gravity is greater than the current velocity or the ball comes within two times the velocity of the top of the screen. In the later case the velocity is set to one pixel per frame. The velocity in this state is updated by subtracting gravity from the input velocity.

If the ball is not in play then if a ball is added (*add\_ball\_level*) is high the *in\_play\_out* and *in\_hand\_out* bits are set high and the *in\_left\_out* and *in\_right\_out* are set to *which\_hand and ~which\_hand* respectively. The rest of the outputs. The rest of the output values are set to zero as is *add\_ball\_level* so no more balls are added that frame. If the ball is not in play and no ball is added then all the output signals are set to zero.

#### Display

When triggered to do so the Display module reads through the BRAM memory and extracts the balls' positions and logic to send onto the output module along with which ball that data is for so it can know if and where to display each of the balls.

The inputs to the display are the clock, reset, *new\_frame*, *done\_calc* and *info\_from\_bram* signals. The clock is 65 MHZ from the *labkit*. The reset is synchronized to *new\_frame* and both come from the controller. The *done\_calc* signal comes from the Physics module and signals that the Physics has been calculated and stored for this frame and that the Display module can begin extracting the needed data to give to the output module. The *info\_from\_bram* is a 70 bit signal holding all the information about each ball that the Display module parses and sends to the output module.

The outputs are *bram\_addr\_b*, *ball\_pos\_out*, *display\_enb*, *ball\_number* and *write\_request*. The five bit *bram\_addr\_b* is the address the data is going to come from out of the BRAM. The *ball\_pos\_out* is the 32 bit signal with the x position of the current ball stored in the high 16 bits and the y position in the low 16 bits. The *diplay\_enb* is a one bit signal that determines that particular ball is to be displayed. The five bit *ball\_number* is the number corresponding to the ball for which the current position and display logic are being outputted. The *write\_request* is a one bit signal sent to the Output module which indicates that the data coming in is valid and should be stored.

Registers were created for each of the outputs and three internally used signals since all are assigned in a procedural block. The three internally used signals are a one bit *stop\_calc* that indicates that all 31 of the balls that can be used have been processed and the Display module should wait for a *new\_frame* to do anything. The five bit *ball\_number\_hold* is used to store the *bram\_addr\_b* signal before it is passed onto *ball\_number* to pipeline the process since it takes a clock cycle to retrieve the data and another cycle to do the logic on it. A *write\_request\_hold* register is used for the same reason.

Within the procedural block a number of if statement may be evaluated. If reset it high *stop\_calc*, *bram\_addr\_b*, *write\_request* and *ball\_number* are all set to zero since the system is going to be reset and old values in the BRAM don't matter so we'll wait until they've been rewritten during the reset frame before giving the Output module anything to display. The system will wait until it receives a *done\_calc* from the Physics module before doing anything. Else if the reset isn't high if *stop\_calc* is high the system is in a waiting state for new frame. The *stop\_calc* signal is set back to zero if *new\_frame* is high otherwise it takes on its old value. The same happens for *bram\_addr\_b* and *ball\_number*. The *write request* is set to zero. Else if neither reset or stop calc then if *done calc* is high (signaling the Physics module is completed with calculations and writing into memory for the frame) the following happens: *bram\_addr\_b* is incremented, *stop\_calc* is set to one if *ball\_number* equals 31 (which means all 31 balls have been processed and their data sent on) otherwise it stays zero, *write\_request\_hold* is set to 0 if *ball\_number* equals 30 (which means all 31 balls will have been processed and their data sent on...not 32 balls due to pipelining) otherwise one, write\_request is assigned to write request hold, ball pos out is assigned to the appended bits 47-32 and 63-48 from *info\_from\_bram* (where the x and y positions are stored respectively), *display\_enb* is set to zero if *ball\_number* equals 31 otherwise it is set to the logical value of bits 69 and not 68 or 67 from *info\_from\_bram* (in play, in left hand and in right hand bits respectively) and *ball\_number\_hold* is set to *bram\_addr b*.

#### **Output Module**

(by Chris Wilkens)

The fourth major block of the implementation is the output module. This module sits in the path of the outgoing VGA signal and overlays the hands and balls. In our implementation, this is accomplished by instantiating a sprite for each object to be displayed. For the two hands, we instantiate two separate directly within the output module and connect them to the position signals from the video processor. However, the situation is slightly more complicated for the 32 balls. From the output module's perspective, these are implemented using a single "aggregate sprite" that can be queried for a pixel location and returns the color based on the location of all 32 balls in the system. This requires a set of connections resembling a memory, which are received from the ball manager's display module and passed directly through to the aggregate ball sprite module.



This diagram shows the structure of the output module.

#### Aggregate Ball Sprite Memory

The ball sprite memory module acts like a memory for the 32 output ball sprites. From the perspective of the output module, it behaves like a single sprite that can be queried for a pixel value. However, it also has a set of "memory" signals that can be used to configure the output. The ball manager uses these signals to sequentially set the positions of the 32 balls. On a given cycle, the ball manager can supply a write enable, a ball number (similar to a memory address,) the position in which the specified ball is to be displayed (similar to memory data,) and whether or not the specified ball is actually to appear on the screen (more memory data.) This information is stored internally such that when the output module queries for a certain location on the screen, the sprite memory can check each ball simultaneously to determine whether or not it claims the given pixel.

We implemented this module as an array of sprite modules. The connections to these sprite modules are large arrays of registers that hold the position and enable signals for the balls. On cycles when the write enable bit is high, this module stores the position and enable signals in the registers for the appropriate ball number. As a consequence, the position of that sprite is updated, such that when a pixel request comes, the module can simply OR the outputs of each of its 32 sprites, allowing the output module to see it as one sprite.

# Testing

Our testing strategy for the project consisted of three facets: independent testing of the video modules in hardware, independent testing of the ball manager in simulation, and combined testing of the modules in hardware. For the video modules, we incrementally tested them in hardware because simulation, in general, would be prohibitively difficult. The first step was to demonstrate that we could buffer and display the proper video. Because we were originally using an older module provided by the staff, this process took time. We did not reach our final design until the staff released a new module that buffered video to the ZBT instead of to the BRAM. At this point, what was supposed to be a simple set of modifications to the existing verilog were confounded by what appeared to be issues computing the address. Once these were fixed, however, the video was ready to go.

The next step, though it was started concurrently with testing video input, was to develop the hand detector module that located the hands on the screen. For this, we first tested thresholds for detecting flesh colored pixels. By connecting our thresholds to the switches of the labkit and coloring the selected pixels red, we were able to tweak the parameters and determine good limits for the Y and Cr channels of the input. With the critical pixels selected, it remained to compute the appropriate centers of mass. This was implemented and debugged by connecting the positions of the hands to sprites in order to display them in the video. This process was marginally complicated because the calculations required many clock cycles, but seemed simple enough. However, it proved to be our first major but silly snag. For some inexplicable reason, the positions were not changing, and it wasn't clear why. Simulation testing revealed that the hand detector worked according to specification, so it was unclear why things appeared to break on synthesis. After an exorbitantly large amount of debugging, it was discovered that the clock signal being passed to the hand detector was not in fact a valid signal at all and that the synthesizer had missed it. Once that was fixed, the module worked as promised.

With the position information, it was necessary to compute the hand logic values. Our first attempt was to compute the velocity of the hand by taking the difference with the last data point. This, however, proved to be a torturous task, as early attempts to subtract the two positions simply yielded zero velocities. After days of debugging, it was finally determined that the source of the error was the *pos\_ready* signal. This signal was only supposed to be high for one clock cycle per frame in order to notify the hand logic module to calculate the velocities. Instead, the signal was high for many clock cycles, effectively erasing the memory of the logic calculator. Unfortunately, this took days of debugging to discover. Again, once it was discovered, everything ran smoothly. We then proceeded to hook the throw and catch signals up to simple logic for testing purposes. We also modified the video processor to tint the screen whenever a throw or a catch was detected. This was an extremely simple modification that allowed us to clearly visualize and tune the hand logic. Using this method, we iterated on the original, simple logic to produce more complicated throw and catch algorithms.

Concurrent with development of the video processor was development of the output module. The first stage of testing was to simply wire the hand sprites to the positions generated by the video processor. However, the ball sprites remained untested. For this, we created a test jig that cycled through each ball and modified it for each frame. This allowed us to test the sprite "memory" aspect of the module independent of the ball manager. With this testing complete, the video processor and output modules were ready for integration.

While Chris was developing the video input an output, I (David Rush) was developing the ball manager. The biggest thing I learned during the creation of the juggling simulator is that small modules are much better. Building a big system and trying to test it is a disaster because many things are likely to be wrong. Basic functionality should first be established with each small component before more complexity should be added. The basic inputs should be tested first and only once they are working should more complex inputs be added. I made the mistake of writing a huge Physics module and had no way but to enter a combination of inputs into the simulator and check to see that nothing came out. I ended up wiring every signal I had to an output to check that any of them were correct and try to figure out where along the chain there was a problem. I iterated this process dozens of times because there were dozens of mistakes that were made. Velocities were wired to position outputs, mot all the outputs were assigned in the right places, wires were missing or mismatched, some code didn't do what I thought and many times my timing was off by a clock cycle. After many, many iterations of trying to simulate the Physics module and getting a little more correct each time after hours of debugging I learned that small was better and each of the subsequent modules I wrote were much smaller and easier to debug.

For each of the modules within the Ball Manager testing in simulation was the easiest way to initially see if the desired results were had. The level to pulse module was easiest and it can be seen in it's graph that level signals were put in and pulses came out.

The Controller was similar in that it had a limited number of inputs and the expected outputs could easily be checked against the actual outputs. If the outputs weren't all synchronized it wasn't working right.

Within the display module I (David) found in simulation that my ball addresses weren't matching up with the right data and after drawing out the timing diagrams I realized I needed a two clock cycle delay between sending the input address and having the valid data to give to the output module. I pipelined the process by buffering the ball number and write enable and that way each ball only took one clock cycle even if the first ball data didn't come out for 3 cycles.

After testing in simulation I visually tested the Ball Manager since looking at 32 bits worth of position data can't very well tell you if the ball is being displayed at the right position after many seconds since the frame updates 60 times per second. At first it was noticed that all the balls were in the upper left of the screen and moving only a little bit. After checking the position calculations I noticed that I had wired the output position bits to the velocity which was always much smaller. It was later noticed that one ball was always displayed and never moved and that was due to an off by one error in the display module. The balls were also very jittery at one point and it was found to be because one signal wasn't set for an entire frame when it should have been.

After 100 hours of work there were many more errors and bugs that were worked through and in the end though the system worked. This is just a highlight of some of the more interesting errors.

## Conclusion

Overall, this project provided an excellent demonstration of the principles of system design, including modularity and simplicity. In the end, the project worked fairly well. Though it couldn't truly simulate the experience of juggling real balls, it could reliably simulate simple juggling moves. Moreover, when configured properly, it was simple enough that even those who couldn't juggle in real life could juggle in the simulator. The net result was a pleasantly functional system.

One of the most prominent design principles in the project was the supremacy of simplicity in system design. For example, our method of hand detection was hardly revolutionary, and calculating the center of mass of the pixels on each side of the screen was very straightforward. However, despite the lack of complexity, this method of hand detection was extremely reliable when the proper precautions were taken. Another simple yet innovative idea was displaying the hand logic by tinting the screen. The implementation was simple – only one line of verilog code was modified – but it allowed the untrained user to gain a much better understanding of the system's interpretation of his or her movements. It also greatly facilitated debugging, as it provided insight into what the system was "thinking." Yet another innovation of simplicity was the idea to allow the user to fix the release velocities of the balls when they are thrown. While this may significantly detract from the usefulness of the simulator for an experienced user, it greatly enhances playability for novices and allows them to become acclimated to the system before enabling the more complex mode.

While the simulator performs fairly well, there are a number of things that could be changed. If one were to rebuild the system from scratch, the system should be better modularized, particularly the ball manager. Also, as with any similar system, there are an infinite number of ways to tweak the video processing that could potentially improve the project's operation. In addition to enhancements to the existing foundation, there are many small additions that could be made. For instance, there were many corner cases that we largely ignored, such as balls on the floor and balls flying above the ceiling. The graphics used in the simulation were also fairly basic and could certainly use improvement. Overall, however, the project was successful. None of its shortcomings dramatically affected its functionality, and it was an enjoyable project to build and use.

# Appendix: Verilog Source

This appendix contains the verilog source for our modules. The modules described above are in files as follows:

| System Block    | Associated Files     |
|-----------------|----------------------|
| General Modules | Labkit.v             |
|                 | Debounce.v           |
|                 | Display_16hex.v      |
|                 | Virtual_juggling.v   |
| Camera Input    | Ntsc2zbt.v           |
|                 | Video_decoder.v      |
|                 | Video_zbt.v          |
|                 | 6111zbt.v            |
| Controller      | Vj_controller.v      |
|                 | Vj_level_to_pulse.v  |
| Video Processor | Vj_video_processor.v |
|                 | Divide.v             |
|                 | Yuv_to_rgb.v         |
| Ball Manager    | Vj_ball_manager.v    |
|                 | Vj_bm_physics.v      |
|                 | Vj_bm_display.v      |
|                 | Bram70x32.v          |
| Output          | Vj_output.v          |

```
//
// File: zbt_6111_sample.v
// Date: 26-Nov-05
// Author: I. Chuang <ichuang@mit.edu>
// Sample code for the MIT 6.111 labkit demonstrating use of the ZBT
// memories for video display. Video input from the NTSC digitizer is
// displayed within an XGA 1024x768 window. One ZBT memory (ram0) is used
// as the video frame buffer, with 8 bits used per pixel (black & white).
//
// Since the ZBT is read once for every four pixels, this frees up time for
// data to be stored to the ZBT during other pixel times. The NTSC decoder
// runs at 27 MHz, whereas the XGA runs at 65 MHz, so we synchronize
// signals between the two (see ntsc2zbt.v) and let the NTSC data be
// stored to ZBT memory whenever it is available, during cycles when
// pixel reads are not being performed.
//
// We use a very simple ZBT interface, which does not involve any clock
// generation or hiding of the pipelining. See zbt_6111.v for more info.
// switch[7] selects between display of NTSC video and test bars
// switch[6] is used for testing the NTSC decoder
// switch[1] selects between test bar periods; these are stored to ZBT
        during blanking periods
//
// switch[0] selects vertical test bars (hardwired; not stored in ZBT)
```

file:///C//Documents%20and%20Settings/cwilkens/My%20Documents/Colle...1%20Project/Verilog%20Files/Verilog%20Source%20as%20Text/labkit.txt

// 2) Expanded "tv\_in\_ycrcb" to 20 bits.

- // 3) Renamed "tv\_out\_data" to "tv\_out\_i2c\_data" and "tv\_out\_sclk" to
- // "tv\_out\_i2c\_clock".
- // 4) Reversed disp\_data\_in and disp\_data\_out signals, so that "out" is an
- // output of the FPGA, and "in" is an input.
- //
- // CHANGES FOR BOARD REVISION 003
- //
- // 1) Combined flash chip enables into a single signal, flash\_ce\_b.

```
// CHANGES FOR BOARD REVISION 002
```

- //
- // 1) Added SRAM clock feedback path input and output

// Complete change history (including bug fixes)

- // 2) Renamed "mousedata" to "mouse\_data"
- // 3) Renamed some ZBT memory signals. Parity bits are now incorporated into
- // the data bus, and the byte write enables have been combined into the

// 2005-Sep-09: Added missing default assignments to "ac97\_sdata\_out",

// 2005-Jan-23: Reduced flash address bus to 24 bits, to match 128Mb devices

actually populated on the boards. (The boards support up to

// 2004-May-01: Changed "disp\_data\_in" to be an output, and gave it a default
// value. (Previous versions of this file declared this port to

// 2004-Apr-29: Reduced SRAM address busses to 19 bits, to match 18Mb devices

actually populated on the boards. (The boards support up to

"disp\_data\_out", "analyzer[2-3]\_clock" and

256Mb devices, with 25 address lines.)

72Mb devices, with 21 address lines.)

// 2004-Oct-31: Adapted to new revision 004 board.

- // 4-bit ram#\_bwe\_b bus.
- // 4) Removed the "systemace\_clock" net, since the SystemACE clock is now
- // hardwired on the PCB to the oscillator.

"analyzer[2-3]\_data".

be an input.)

// 2004-Apr-29: Change history started

//

//

//

|| ||

//

//

//

//

|| ||

//

//

//

//

file:///Cl/Documents%20and%20Settings/cwilkens/My%20D...rilog%20Files/Verilog%20Source%20as%20Text/labkit.txt (2 of 13)12/13/2005 6:59:23 PM

module labkit(beep, audio\_reset\_b,

ac97\_sdata\_out, ac97\_sdata\_in, ac97\_synch, ac97\_bit\_clock,

vga\_out\_red, vga\_out\_green, vga\_out\_blue, vga\_out\_sync\_b, vga\_out\_blank\_b, vga\_out\_pixel\_clock, vga\_out\_hsync, vga\_out\_vsync,

tv\_out\_ycrcb, tv\_out\_reset\_b, tv\_out\_clock, tv\_out\_i2c\_clock, tv\_out\_i2c\_data, tv\_out\_pal\_ntsc, tv\_out\_hsync\_b, tv\_out\_vsync\_b, tv\_out\_blank\_b, tv\_out\_subcar\_reset,

tv\_in\_ycrcb, tv\_in\_data\_valid, tv\_in\_line\_clock1, tv\_in\_line\_clock2, tv\_in\_aef, tv\_in\_hff, tv\_in\_aff, tv\_in\_i2c\_clock, tv\_in\_i2c\_data, tv\_in\_fifo\_read, tv\_in\_fifo\_clock, tv\_in\_iso, tv\_in\_reset\_b, tv\_in\_clock,

ram0\_data, ram0\_address, ram0\_adv\_ld, ram0\_clk, ram0\_cen\_b, ram0\_ce\_b, ram0\_oe\_b, ram0\_we\_b, ram0\_bwe\_b,

ram1\_data, ram1\_address, ram1\_adv\_ld, ram1\_clk, ram1\_ce\_b, ram1\_ce\_b, ram1\_oe\_b, ram1\_we\_b, ram1\_bwe\_b,

clock\_feedback\_out, clock\_feedback\_in,

flash\_data, flash\_address, flash\_ce\_b, flash\_oe\_b, flash\_we\_b, flash\_reset\_b, flash\_sts, flash\_byte\_b,

rs232\_txd, rs232\_rxd, rs232\_rts, rs232\_cts,

mouse\_clock, mouse\_data, keyboard\_clock, keyboard\_data,

clock\_27mhz, clock1, clock2,

disp\_blank, disp\_data\_out, disp\_clock, disp\_rs, disp\_ce\_b, disp\_reset\_b, disp\_data\_in,

button0, button1, button2, button3, button\_enter, button\_right, button\_left, button\_down, button\_up, file:///Cl/Documents%20and%20Settings/cwilkens/My%20Documents/Colle...l%20Project/Verilog%20Files/Verilog%20Source%20as%20Text/labkit.txt

switch,

led,

user1, user2, user3, user4,

daughtercard,

```
systemace_data, systemace_address, systemace_ce_b,
systemace_we_b, systemace_oe_b, systemace_irq, systemace_mpbrdy,
```

analyzer1\_data, analyzer1\_clock, analyzer2\_data, analyzer2\_clock, analyzer3\_data, analyzer3\_clock, analyzer4\_data, analyzer4\_clock);

```
output beep, audio_reset_b, ac97_synch, ac97_sdata_out; input ac97_bit_clock, ac97_sdata_in;
```

```
output [7:0] vga_out_red, vga_out_green, vga_out_blue;
output vga_out_sync_b, vga_out_blank_b, vga_out_pixel_clock,
vga_out_hsync, vga_out_vsync;
```

```
output [9:0] tv_out_ycrcb;
output tv_out_reset_b, tv_out_clock, tv_out_i2c_clock, tv_out_i2c_data,
    tv_out_pal_ntsc, tv_out_hsync_b, tv_out_vsync_b, tv_out_blank_b,
    tv_out_subcar_reset;
```

```
input [19:0] tv_in_ycrcb;
input tv_in_data_valid, tv_in_line_clock1, tv_in_line_clock2, tv_in_aef,
    tv_in_hff, tv_in_aff;
output tv_in_i2c_clock, tv_in_fifo_read, tv_in_fifo_clock, tv_in_iso,
    tv_in_reset_b, tv_in_clock;
inout tv_in_i2c_data;
```

inout [35:0] ram0\_data; output [18:0] ram0\_address; output ram0\_adv\_ld, ram0\_clk, ram0\_cen\_b, ram0\_ce\_b, ram0\_oe\_b, ram0\_we\_b; output [3:0] ram0\_bwe\_b;

inout [35:0] ram1\_data; output [18:0] ram1\_address; output ram1\_adv\_ld, ram1\_clk, ram1\_ce\_b, ram1\_ce\_b, ram1\_oe\_b, ram1\_we\_b; output [3:0] ram1\_bwe\_b;

input clock\_feedback\_in; output clock\_feedback\_out;

inout [15:0] flash\_data; output [23:0] flash\_address; output flash\_ce\_b, flash\_oe\_b, flash\_we\_b, flash\_reset\_b, flash\_byte\_b; input flash\_sts;

output rs232\_txd, rs232\_rts; input rs232\_rxd, rs232\_cts;

input mouse\_clock, mouse\_data, keyboard\_clock, keyboard\_data;

input clock\_27mhz, clock1, clock2;

output disp\_blank, disp\_clock, disp\_rs, disp\_ce\_b, disp\_reset\_b; input disp\_data\_in; output disp\_data\_out;

input button0, button1, button2, button3, button\_enter, button\_right, button\_left, button\_down, button\_up; input [7:0] switch; output [7:0] led;

inout [31:0] user1, user2, user3, user4;

inout [43:0] daughtercard;

inout [15:0] systemace\_data; output [6:0] systemace\_address; output systemace\_ce\_b, systemace\_we\_b, systemace\_oe\_b; input systemace\_irq, systemace\_mpbrdy;

output [15:0] analyzer1\_data, analyzer2\_data, analyzer3\_data, analyzer4\_data; output analyzer1\_clock, analyzer2\_clock, analyzer3\_clock, analyzer4\_clock;

```
// Audio Input and Output
assign beep= 1'b0;
assign audio_reset_b = 1'b0;
assign ac97_synch = 1'b0;
assign ac97_sdata_out = 1'b0;
/*
```

\*/

```
// ac97_sdata_in is an input
```

```
// Video Output
assign tv_out_ycrcb = 10'h0;
assign tv_out_reset_b = 1'b0;
assign tv_out_clock = 1'b0;
assign tv_out_i2c_clock = 1'b0;
assign tv_out_i2c_data = 1'b0;
assign tv_out_pal_ntsc = 1'b0;
assign tv_out_hsync_b = 1'b1;
assign tv_out_blank_b = 1'b1;
assign tv_out_subcar_reset = 1'b0;
```

```
// Video Input
//assign tv_in_i2c_clock = 1'b0;
assign tv_in_fifo_read = 1'b1;
assign tv_in_fifo_clock = 1'b0;
assign tv_in_iso = 1'b1;
//assign tv_in_reset_b = 1'b0;
assign tv_in_clock = clock_27mhz;//1'b0;
//assign tv_in_i2c_data = 1'bZ;
// tv_in_ycrcb, tv_in_data_valid, tv_in_line_clock1, tv_in_line_clock2,
// tv_in_aef, tv_in_hff, and tv_in_aff are inputs
```

// SRAMs

/\* change lines below to enable ZBT RAM bank0 \*/

/\* assign ram0\_data = 36'hZ; assign ram0\_address = 19'h0; assign ram0\_clk = 1'b0; assign ram0\_we\_b = 1'b1; file:///Cl/Documents%20and%20Settings/cwilkens/My%20Documents/Colle...l%20Project/Verilog%20Files/Verilog%20Source%20as%20Text/labkit.txt

```
assign ram0_cen_b = 1'b0; // clock enable */
```

```
/* enable RAM pins */
```

```
assign ram0_ce_b = 1'b0;
assign ram0_oe_b = 1'b0;
assign ram0_adv_ld = 1'b0;
assign ram0_bwe_b = 4'h0;
```

/\*\*\*\*\*\*\*\*/

```
assign ram1_data = 36'hZ;
assign ram1_address = 19'h0;
assign ram1_adv_ld = 1'b0;
assign ram1_clk = 1'b0;
assign ram1_ce_b = 1'b1;
assign ram1_ce_b = 1'b1;
assign ram1_oe_b = 1'b1;
assign ram1_we_b = 1'b1;
assign ram1_bwe_b = 4'hF;
```

```
assign clock_feedback_out = 1'b0;
// clock_feedback_in is an input
```

```
// Flash ROM
assign flash_data = 16'hZ;
assign flash_address = 24'h0;
assign flash_ce_b = 1'b1;
assign flash_oe_b = 1'b1;
assign flash_we_b = 1'b1;
assign flash_reset_b = 1'b0;
assign flash_byte_b = 1'b1;
// flash_sts is an input
```

```
// RS-232 Interface
assign rs232_txd = 1'b1;
assign rs232_rts = 1'b1;
// rs232_rxd and rs232_cts are inputs
```

# // PS/2 Ports // mouse\_clock, mouse\_data, keyboard\_clock, and keyboard\_data are inputs

```
// LED Displays
/*
  assign disp_blank = 1'b1;
  assign disp_clock = 1'b0;
  assign disp_rs = 1'b0;
  assign disp_ce_b = 1'b1;
  assign disp_reset_b = 1'b0;
  assign disp_data_out = 1'b0;
*/
 // disp_data_in is an input
 // Buttons, Switches, and Individual LEDs
  //lab3 assign led = 8'hFF;
  // button0, button1, button2, button3, button_enter, button_right,
  // button_left, button_down, button_up, and switches are inputs
 // User I/Os
  assign user1 = 32'hZ;
  assign user2 = 32'hZ;
  assign user3 = 32'hZ;
  assign user4 = 32'hZ;
 // Daughtercard Connectors
  assign daughtercard = 44'hZ;
  // SystemACE Microprocessor Port
  assign systemace_data = 16'hZ;
  assign systemace_address = 7'h0;
  assign systemace_ce_b = 1'b1;
  assign systemace_we_b = 1'b1;
  assign systemace_oe_b = 1'b1;
  // systemace_irq and systemace_mpbrdy are inputs
 // Logic Analyzer
  assign analyzer1_data = 16'h0;
  assign analyzer1_clock = 1'b1;
  assign analyzer2_data = 16'h0;
  assign analyzer2_clock = 1'b1;
  assign analyzer3_data = 16'h0;
  assign analyzer3_clock = 1'b1;
  assign analyzer4_data = 16'h0;
  assign analyzer4_clock = 1'b1;
```

#### 

// Demonstration of ZBT RAM as video memory

// use FPGA's digital clock manager to produce a // 65MHz clock (actually 64.8MHz) wire clock\_65mhz\_unbuf,clock\_65mhz; DCM vclk1(.CLKIN(clock\_27mhz),.CLKFX(clock\_65mhz\_unbuf)); // synthesis attribute CLKFX\_DIVIDE of vclk1 is 10 // synthesis attribute CLKFX\_MULTIPLY of vclk1 is 24 // synthesis attribute CLK\_FEEDBACK of vclk1 is NONE // synthesis attribute CLKIN\_PERIOD of vclk1 is 37 BUFG vclk2(.O(clock\_65mhz),.I(clock\_65mhz\_unbuf));

wire  $clk = clock_{65mhz}$ ;

// ENTER button is user reset
wire reset,user\_reset;
debounce db1(power\_on\_reset, clk, ~button\_enter, user\_reset);
assign reset = user\_reset | power\_on\_reset;

#### // DEBOUNCE other signals

wire db\_button\_up, db\_button\_down,db\_button\_right,db\_button\_left; debounce incdb(power\_on\_reset, clock\_65mhz, ~button\_up, db\_button\_up); debounce decdb(power\_on\_reset, clock\_65mhz, ~button\_down, db\_button\_down); debounce butdr(power\_on\_reset, clock\_65mhz, ~button\_right, db\_button\_right); debounce butdl(power\_on\_reset, clock\_65mhz, ~button\_left, db\_button\_left); wire [7:0] db\_switch; file:///Cl/Documents%20and%20Settings/cwilkens/My%20Documents/Colle...l%20Project/Verilog%20Files/Verilog%20Source%20as%20Text/labkit.txt

debounce switchdb[7:0](power\_on\_reset, clk, switch, db\_switch); wire db\_button0, db\_button1, db\_button2, db\_button3; debounce b0db(power\_on\_reset, clock\_65mhz, ~button0, db\_button0); debounce b1db(power\_on\_reset, clock\_65mhz, ~button1, db\_button1); debounce b2db(power\_on\_reset, clock\_65mhz, ~button2, db\_button2); debounce b3db(power\_on\_reset, clock\_65mhz, ~button3, db\_button3);

// display module for debugging

reg [63:0] dispdata; display\_16hex hexdisp1(reset, clock\_65mhz, dispdata,// was clk disp\_blank, disp\_clock, disp\_rs, disp\_ce\_b, disp\_reset\_b, disp\_data\_out);

// generate basic XVGA video signals
wire [10:0] hcount;
wire [9:0] vcount;
wire hsync,vsync,blank;
xvga xvga1(clk,hcount,vcount,hsync,vsync,blank);

// wire up to ZBT ram

wire [35:0] vram\_write\_data; wire [35:0] vram\_read\_data; wire [18:0] vram\_addr; wire vram\_we;

```
// generate pixel value from reading ZBT memory
wire [17:0] vr_pixel;
wire [18:0] vram_addr1;
```

.tv\_in\_i2c\_clock(tv\_in\_i2c\_clock), .tv\_in\_i2c\_data(tv\_in\_i2c\_data));

wire [29:0] ycrcb; // video data (luminance, chrominance)
wire [2:0] fvh; // sync for field, vertical, horizontal
wire dv; // data valid

```
ntsc_decode decode (.clk(tv_in_line_clock1), .reset(reset),
.tv_in_ycrcb(tv_in_ycrcb[19:10]),
.ycrcb(ycrcb), .f(fvh[2]),
.v(fvh[1]), .h(fvh[0]), .data_valid(dv));
```

// code to write NTSC data to video memory

```
// code to write pattern to ZBT memory
reg [31:0] count;
always @(posedge clk) count <= reset ? 0 : count + 1;</pre>
```

```
wire [18:0] vram_addr2 = count[0+18:0];
wire [35:0] vpat = ( db_button0 ? {4{count[3+3:3],4'b0}}
            : {4{count[3+4:4],4'b0}} );
```

// mux selecting read/write to memory based on which write-enable is chosen

```
wire sw_ntsc = ~db_button3;
wire my_we = sw_ntsc ? (hcount[1:0]==2'd2) : blank;
wire [18:0] write_addr = sw_ntsc ? ntsc_addr : vram_addr2;
wire [35:0] write_data = sw_ntsc ? ntsc_data : vpat;
// wire write enable = sw_ntsc ? (my_we & ntsc_we) : my_we;
```

```
// assign vram_addr = write_enable ? write_addr : vram_addr1;
// assign vram_we = write_enable;
```

```
assign vram_addr = my_we ? write_addr : vram_addr1;
```

```
assign vram_we = my_we;
```

```
assign vram_write_data = write_data;
```

file:///Cl/Documents%20and%20Settings/cwilkens/My%20Documents/Colle...l%20Project/Verilog%20Files/Verilog%20Source%20as%20Text/labkit.txt

// select output pixel data

```
reg [17:0] pixel; // pixel[8:4] are the top five bits of Y, pixel[3:0] are the top 4 bits of U wire b,hs,vs;
```

```
delayN dn1(clk,hsync,hs); // delay by 3 cycles to sync with ZBT read
delayN dn2(clk,vsync,vs);
delayN dn3(clk,blank,b);
```

```
always @(posedge clk)
begin
pixel <= db_button0 ? {hcount[8:6],5'b0,hcount[8:6],5'b0,hcount[8:7]} : vr_pixel;
end</pre>
```

```
// VGA Output. In order to meet the setup and hold times of the // AD7125, we send it ~clock_65mhz.
```

```
wire [63:0] hex_debug;
```

```
assign vga_out_pixel_clock = ~clock_65mhz;
```

```
/*assign vga_out_red = pixel;
assign vga_out_green = pixel;
assign vga_out_blue = pixel;
assign vga_out_sync_b = 1'b1; // not used
assign vga_out_pixel_clock = ~clock_65mhz;
assign vga_out_blank_b = ~b;
assign vga_out_hsync = hs;
assign vga_out_vsync = vs;*/
```

```
// debugging
```

```
assign led = ~{vram_addr[18:13],reset,db_button0};
```

always @(posedge clock\_65mhz)// was clk
// dispdata <= {vram\_read\_data,9'b0,vram\_addr};
dispdata <= hex\_debug;//{ntsc\_data,9'b0,ntsc\_addr};</pre>

endmodule

file:///C//Documents%20and%20Settings/cwilkens/My%20Documents/Colleg...ct/Verilog%20Files/Verilog%20Source%20as%20Text/virtual\_juggling.txt

/\*

This module represents the entire VirtualJuggling simulator.

\*/

module virtual\_juggling(clock\_65mhz, reset,

vga\_in\_sync\_b, vga\_in\_blank\_b, y, u, v, vga\_in\_hsync, vga\_in\_vsync, hcount\_in, vcount\_in,

vga\_out\_sync\_b, vga\_out\_blank\_b, vga\_out\_red, vga\_out\_green, vga\_out\_blue, vga\_out\_hsync, vga\_out\_vsync,

ball\_inc,ball\_dec,left\_throw\_button,right\_throw\_button,db\_switch, hex\_display);

input clock\_65mhz; input reset; input vga\_in\_sync\_b, vga\_in\_blank\_b; input [7:0] y, u, v; input vga\_in\_hsync, vga\_in\_vsync; input [10:0] hcount\_in; input [9:0] vcount\_in;

output vga\_out\_sync\_b, vga\_out\_blank\_b; output [7:0] vga\_out\_red, vga\_out\_green, vga\_out\_blue; output vga\_out\_hsync, vga\_out\_vsync;

input ball\_inc,ball\_dec; input left\_throw\_button,right\_throw\_button; input [7:0] db\_switch; output [63:0] hex\_display;

// PASS THROUGH with delay FOR DEBUGGING
/\*reg vga\_out\_sync\_b, vga\_out\_blank\_b;
reg [7:0] vga\_out\_red, vga\_out\_green, vga\_out\_blue;
reg vga\_out\_hsync, vga\_out\_vsync;

always @ (posedge clock\_65mhz) begin vga\_out\_sync\_b <= vga\_in\_sync\_b;</pre>

```
vga_out_blank_b <= vga_in_blank_b;
{vga_out_red,vga_out_green,vga_out_blue} <= {y,u,v};
vga_out_hsync <= vga_in_hsync;
vga_out_vsync <= vga_in_vsync;
end*/
```

// Logic wires
wire new\_frame;
wire [31:0] left\_hand\_pos, right\_hand\_pos;
wire [31:0] left\_hand\_vel, right\_hand\_vel;
wire [1:0] left\_hand\_logic, right\_hand\_logic;

// Intermediate video wires between video processor and output wire vga\_mid\_sync\_b, vga\_mid\_blank\_b; wire [10:0] hcount\_mid; wire [9:0] vcount\_mid; wire [7:0] vga\_mid\_red, vga\_mid\_green, vga\_mid\_blue; wire vga\_mid\_hsync, vga\_mid\_vsync;

// Video processor module

wire [31:0] VP\_DEBUG;

vj\_video\_processor video\_proc(clock\_65mhz,reset,

vga\_in\_sync\_b,vga\_in\_blank\_b,hcount\_in,vcount\_in, vga\_in\_hsync,vga\_in\_vsync, y,u,v,

db\_switch,

vga\_mid\_red,vga\_mid\_green,vga\_mid\_blue, vga\_mid\_sync\_b,vga\_mid\_blank\_b, hcount\_mid,vcount\_mid, vga\_mid\_hsync,vga\_mid\_vsync, new\_frame, left\_hand\_pos,right\_hand\_pos, left\_hand\_vel,right\_hand\_vel, left\_hand\_logic,right\_hand\_logic, VP\_DEBUG,8'b0);

//Signals from ball manager to output module
wire bm\_we,bm\_be;
wire [4:0] bm\_ball\_number;
wire [31:0] bm\_ball\_pos;
wire [7:0] db\_switch; //added for color debugging

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colleg...ct/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/virtual_juggling.txt$ 

\*/

// Output module

vj\_output outputmodule(clock\_65mhz,reset,

vga\_mid\_red,vga\_mid\_green,vga\_mid\_blue, hcount\_mid,vcount\_mid, vga\_mid\_sync\_b,vga\_mid\_blank\_b,vga\_mid\_hsync,vga\_mid\_vsync, left\_hand\_pos,right\_hand\_pos, bm\_we,bm\_ball\_number,bm\_ball\_pos,bm\_be, vga\_out\_red,vga\_out\_green,vga\_out\_blue, vga\_out\_sync\_b,vga\_out\_blank\_b, vga\_out\_hsync,vga\_out\_vsync);

// DEBUG OUTPUT

| <pre>//assign hex_display = {left_hand_pos,right_hand_pos};</pre>                               |
|-------------------------------------------------------------------------------------------------|
| <pre>//assign hex_display = {new_frame,29'b0,left_hand_logic,30'b0,right_hand_logic};</pre>     |
| <pre>//assign hex_display = {left_hand_vel,VP_DEBUG};</pre>                                     |
| <pre>//assign hex_display = {left_hand_vel,right_hand_vel};</pre>                               |
| <pre>//assign hex_display = {left_hand_pos,TEST_JIG_DEBUG};</pre>                               |
| <pre>//assign hex_display = {20'hABCDE,2'b0,bm_we,bm_be,3'b0,bm_ball_number,bm_ball_pos};</pre> |

//Controller modual instantiation

wire new\_frame\_sync; wire reset\_sync; wire add\_ball\_sync; wire dec\_ball\_sync; wire [7:0] gravity\_sync; wire left\_throw\_sync,right\_throw\_sync;

vj\_controller vj\_control\_instan(.clk(clock\_65mhz),.new\_frame\_raw(new\_frame),.reset\_raw(reset),. add\_ball\_raw(ball\_inc),

 $.dec\_ball\_raw(ball\_dec),.left\_throw\_raw(left\_throw\_button),.right\_throw\_raw(right\_throw\_button),$ 

.gravity\_raw(db\_switch),.new\_frame\_sync(new\_frame\_sync), .reset\_sync(reset\_sync),.add\_ball\_sync(add\_ball\_sync),.dec\_ball\_sync(dec\_ball\_sync), .left\_throw\_sync(left\_throw\_sync),.right\_throw\_sync(right\_throw\_sync), .gravity\_sync(gravity\_sync));

```
// wire [31:0] left_hand_pos, right_hand_pos;
```

- // wire [31:0] left\_hand\_vel, right\_hand\_vel;
- // wire [1:0] left\_hand\_logic, right\_hand\_logic;

```
wire [15:0] l_h_x, l_h_y, r_h_x, r_h_y;
// wire [15:0] l_h_x_vel, l_h_y_vel, r_h_x_vel, r_h_y_vel;
/* wire left_throw, l_h_catch, right_throw, r_h_catch;
  assign left_throw =left_hand_logic[1];
  assign l_h_catch =left_hand_logic[0];
  assign right_throw =right_hand_logic[1];
  assign r_h_catch =right_hand_logic[0];
 */
  assign 1 h_x = \text{left} \text{ hand} \text{pos}[31:16];
  assign l_h_y = left_hand_pos[15:0];
  assign r_h_x = right_hand_pos[31:16];
  assign r_h_y = right_hand_pos[15:0];
// assign l_h_x_vel = left_hand_vel[31:16];
// assign l_h_y_vel = left_hand_vel[15:0];
// assign r_h_x_vel = right_hand_vel[31:16];
// assign r_h_y_vel = right_hand_vel[15:0];
```

```
wire signed [16:0] control_left_y_vel;
wire signed [16:0] control_right_y_vel;
wire signed [16:0] control_left_x_vel;
assign control_left_y_vel = (17'b1111111100000000 + {1,left_hand_vel[15:0]});
assign control_right_y_vel = (17'b1111111100000000 + {1,right_hand_vel[15:0]});
assign control_left_x_vel = (17'b11111111100000000 + {1,right_hand_vel[15:0]});
assign control_left_x_vel = (17'b11111111100000001 + {1,right_hand_vel[31:16]});
```

vj\_ball\_manager vj\_bm\_instan(.vclock(clock\_65mhz),.reset(reset\_sync),.new\_frame (new\_frame\_sync),.gravity(gravity\_sync),

.add\_ball(add\_ball\_sync),.dec\_ball(dec\_ball\_sync),

 $.left\_throw(left\_hand\_logic[1]), //1'b1), // \ (left\_throw\_force|left\_hand\_logic[1]), //1'b1), //1'b1), // \ (left\_throw\_force|left\_hand$ 

 $.l\_h\_catch(left\_hand\_logic[0]), // |left\_hand\_logic[0]),$ 

.right\_throw(right\_hand\_logic[1]),//1'b1),// (right\_throw\_force)|right\_hand\_logic[1]),

.r\_h\_catch(right\_hand\_logic[0]),//|right\_hand\_logic[0]),

 $.l_h_y(l_h_y),.l_h_x(l_h_x),.l_h_y_vel(left_hand_vel[15:0]),//)16'b1111111100000000,// control_left_y_vel[16:1]),$ 

file:///C//Documents%20and%20Settings/cwilkens/My%20Documents/Colleg...ct/Verilog%20Files/Verilog%20Source%20as%20Text/virtual\_juggling.txt

.r\_h\_y(r\_h\_y),.r\_h\_x(r\_h\_x),.r\_h\_y\_vel(right\_hand\_vel[15:0]),//),16'b1111111100000000// control\_right\_y\_vel[16:1]),

.r\_h\_x\_vel(control\_right\_x\_vel[16:1]),//right\_hand\_vel[31:16]),16'b1111111110000001

.display\_enb(bm\_be),.ball\_pos\_out(bm\_ball\_pos),.ball\_number(bm\_ball\_number),.write\_request (bm\_we));

assign hex\_display = {bm\_ball\_pos,l\_h\_x,3'b000,new\_frame\_sync,3'b000,add\_ball\_sync,3'b000, bm\_be,3'b000,bm\_we};

/\*

vj\_ball\_manager vj\_bm\_instan(.vclock(clock\_65mhz),.reset(reset\_sync),.new\_frame (new\_frame\_sync),.gravity(gravity\_sync),

.add\_ball(add\_ball\_sync),.dec\_ball(dec\_ball\_sync),

.left\_throw(left\_hand\_logic[1]),//1'b1),// (left\_throw\_force|left\_hand\_logic[1]),

.l\_h\_catch(left\_hand\_logic[0]),//|left\_hand\_logic[0]),

.right\_throw(right\_hand\_logic[1]),//1'b1),// (right\_throw\_force)|right\_hand\_logic[1]),

.r\_h\_catch(right\_hand\_logic[0]),//|right\_hand\_logic[0]),

.l\_h\_y(l\_h\_y),.l\_h\_x(l\_h\_x),.l\_h\_y\_vel(left\_hand\_vel[15:0]),//l16'b1111111100000000),

.l\_h\_x\_vel(left\_hand\_vel[31:16]),//16'b000000001111111),

 $.r_h_y(r_h_y), .r_h_x(r_h_x), .r_h_y_vel(right_hand_vel[15:0]), //16'b1111111100000000),$ 

.r\_h\_x\_vel(right\_hand\_vel[31:16]),//16'b1111111111000000),

.display\_enb(bm\_be),.ball\_pos\_out(bm\_ball\_pos),.ball\_number(bm\_ball\_number),.write\_request (bm\_we));

assign hex\_display = {bm\_ball\_pos,l\_h\_x,3'b000,new\_frame\_sync,3'b000,add\_ball\_sync,3'b000, bm\_be,3'b000,bm\_we}; \*/

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle... Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/vj_video_processor.txt$ 

```
/*
vj_video_processor
```

This is the main video processing module for the juggling simulator. It receives the raw camera image as a VGA stream (with colors in the YUV space instead of RGB) and locates the hands in the frame. It also converts the signal to RGB video so it can be displayed later.

Notes:

1. This module delays the video by one clock cycle.

2. This module displays the camera image in B&W

```
*/
```

```
module vj_video_processor(clock_65mhz,reset,
```

sync\_in,blank\_in,hcount\_in,vcount\_in, hsync\_in,vsync\_in, y,u,v,

db\_switch,

vga\_red,vga\_green,vga\_blue, sync\_out,blank\_out,hcount\_out,vcount\_out, hsync\_out,vsync\_out, new\_frame, left\_pos,right\_pos, left\_vel,right\_vel, left\_logic,right\_logic, DEBUG\_OUT,DEBUG\_SETTINGS);

// VGA signals for incoming video
input clock\_65mhz;
input reset;
input sync\_in,blank\_in;
input [10:0] hcount\_in;
input [9:0] vcount\_in;
input hsync\_in;
input vsync\_in;
// Incoming color signals are 8-bit YUV values, not RGB
input [7:0] y,u,v;
input [7:0] db\_switch;

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle... Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/vj_video_processor.txt/source\%20 as\%20 as\%2$ 

// The module converts the YUV signal to RGB and outputs // it to these ports, 8 bits per channel output [7:0] vga\_red,vga\_green,vga\_blue; // Since the module must delay the signal, the output // video signal is sent to the following ports output sync\_out,blank\_out; output [10:0] hcount\_out; output [9:0] vcount\_out; output hsync\_out, vsync\_out; // Output signals // new\_frame is raised on the clock cycle when the hand // information for the current frame has been calculated. // This signals the beginning of the space between the // frame and vsync. output new\_frame; // Hand position/velocity signals // bits [31:16] are for x, [15:0] are for y // the low order bit of each value (e.g. 16 or 0) // corresponds to 2^-5 (i.e. each value has 5 fractional bits) output [31:0] left\_pos,right\_pos; output [31:0] left\_vel,right\_vel; // output [1:0] left\_logic,right\_logic;

output [31:0] DEBUG\_OUT; input [7:0] DEBUG\_SETTINGS;

```
reg [31:0] DEBUG_OUT;
```

// We delay the video by one clock cycle in order to calculate hand information.
reg [24:0] video\_delay;

reg sync\_out,blank\_out; reg [10:0] hcount\_out; reg [9:0] vcount\_out; reg hsync\_out,vsync\_out;

reg [7:0] vga\_red,vga\_green,vga\_blue;

wire pos\_ready,left\_ready, right\_ready;

assign new\_frame = left\_ready & right\_ready;

// Modules for computing the velocity and logical motions of the hands

vj\_vp\_hand\_logic left\_hand(clock\_65mhz,reset,pos\_ready,left\_pos,left\_vel,left\_logic,left\_ready, DEBUG\_SETTINGS);

vj\_vp\_hand\_logic right\_hand(clock\_65mhz,reset,pos\_ready,right\_pos,right\_vel,right\_logic, right\_ready,DEBUG\_SETTINGS);

wire hand\_pixel; // Receives output from hand detector stating whether or not this pixel belongs to a hand.

wire [7:0] red, green, blue; // Outputs from the YUV to RGB converter.

vj\_vp\_hand\_detector hand\_detector(clock\_65mhz,reset,hcount\_in,vcount\_in,vsync\_in,y,u,v, pos\_ready,left\_pos,right\_pos,hand\_pixel, DEBUG\_SETTINGS);

yuv\_to\_rgb video\_converter(y, u, v, red, green, blue);

reg test = 0;

```
always @ (posedge clock_65mhz) begin
```

{sync\_out,blank\_out,hcount\_out,vcount\_out,hsync\_out,vsync\_out} <=</pre>

```
{sync_in,blank_in,hcount_in,vcount_in,hsync_in,vsync_in};
```

{vga\_red, vga\_green, vga\_blue} <= (hand\_pixel & db\_switch[7]) ? 24'hFF0000 : (~db\_switch [6]) ? {red, green, blue} :

hcount\_out < 11'd512 ? (left\_logic[1] ? {8'h00, green, 8'h00} : left\_logic[0] ? {8'h00, 8'h00, blue} : {red, green, blue}) :

```
right_logic[1] ? {8'h00, green, 8'h00} : right_logic[0] ? {8'h00, 8'h00, blue} : {red, green, blue};
```

```
DEBUG_OUT <= left_logic[1] ? left_vel : DEBUG_OUT;</pre>
```

end

endmodule

```
/*
```

```
Hand detector module
```

```
*/
```

```
input clock_65mhz;
input reset;
input [7:0] y, u, v;
```

input [10:0] hcount; input [9:0] vcount; input vsync;

output pos\_ready; output [31:0] left\_pos, right\_pos; output hand\_pixel;

input [7:0] DEBUG\_COLOR\_TOLERANCE;

parameter RED\_U\_THRESHOLD = 8'b10010000; parameter RED\_Y\_MIN = 8'b00011000; parameter RED\_Y\_MAX = 8'b11111111; parameter V\_BACK\_PORCH\_START = 10'd770;// time to get off the screen and compute the COMs

reg new\_frame; reg stale\_frame; reg pos\_ready;

```
reg [31:0] left_pos, right_pos;
```

reg hand\_pixel;

reg left\_hand\_pixel, right\_hand\_pixel;

wire [15:0] left\_x, left\_y; wire [15:0] right\_x, right\_y;

```
reg average_reset = 0;
```

// We need to hold the hcount and vcount values so
// that the averager uses the correct position
reg [10:0] x\_inc;
reg [9:0] y\_inc;

always @ (posedge clock\_65mhz) begin
 // Intentional use of blocking assignment to effect sequential execution.
 hand\_pixel = ((u > RED\_U\_THRESHOLD) & (RED\_Y\_MIN < y) & (y < RED\_Y\_MAX));
 left\_hand\_pixel <= hand\_pixel & (hcount < 11'd512);
 right\_hand\_pixel <= hand\_pixel & (hcount > 11'd511) & (hcount < 11'd1024);</pre>

 $file: ///C // Documents \% 20 and \% 20 Settings / cwilkens / My \% 20 Documents / Colle... Verilog \% 20 Files / Verilog \% 20 Source \% 20 as \% 20 Text / vj_video_processor.txt / vj_video_processor.$ 

// Hold the hcount and vcount values so that they are ready for the divider on the next cycle.
x\_inc <= hcount;</pre>

y\_inc <= vcount;</pre>

// CODE FOR TESTING COLOR TOLERANCES...

//hand\_pixel = DEBUG\_COLOR\_TOLERANCE[1] ?
//({1'b0,(DEBUG\_COLOR\_TOLERANCE[0] ? v : u)} > {DEBUG\_COLOR\_TOLERANCE

[6:2],4'b0}):

 $//({1'b0,(DEBUG_COLOR_TOLERANCE[0] ? v : u)} < {DEBUG_COLOR_TOLERANCE [6:2],4'b0});$ 

// If we have reached the end of the displayed frame, then grab the COM's

```
if ((vcount == V_BACK_PORCH_START) & ~stale_frame) begin
```

```
pos_ready <= 1;
```

```
left_pos \le {left_x[13:0], 2'b0, left_y[13:0], 2'b0}; // We calculate the COM with 4 fractional bits, but want 5.
```

```
right_pos <= {right_x[13:0],2'b0,right_y[13:0],2'b0};
```

```
stale_frame <= 1;</pre>
```

end

// Make sure that new\_frame is only high for one cycle.

```
else if (pos_ready) begin
```

```
pos_ready <= 0;
end
else if (~vsync) begin
```

```
stale_frame <=0;
```

end

```
// On vsync, reset the COMs.
if (~vsync) average_reset <= 1;
else average_reset <= 0;</pre>
```

end

// The parameters are defined to ignore the porches, in case that is an issue.

weighted\_average left\_x\_com(clock\_65mhz,reset,left\_hand\_pixel,average\_reset,{2'b0,x\_inc,3'b0},
left\_x);

```
//defparam left_x_com.IN_MIN = 16'd0;
```

```
//defparam left_x_com.IN_MAX = 16'd1023;
```

weighted\_average left\_y\_com(clock\_65mhz,reset,left\_hand\_pixel,average\_reset,{3'b0,y\_inc,3'b0},
left\_y);

//defparam left\_y\_com.IN\_MIN = 16'd0; //defparam left\_y\_com.IN\_MAX = 16'd767;

weighted\_average right\_x\_com(clock\_65mhz,reset,right\_hand\_pixel,average\_reset,{2'b0,

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/vj_video_processor.txt$ 

```
x_inc,3'b0},right_x);
//defparam right_x_com.IN_MIN = 16'd0;
//defparam right_x_com.IN_MAX = 16'd1023;
weighted_average right_y_com(clock_65mhz,reset,right_hand_pixel,average_reset,{3'b0,
y_inc,3'b0},right_y);
//defparam right_y_com.IN_MIN = 16'd0;
//defparam right_y_com.IN_MAX = 16'd767;
```

endmodule

output [31:0] vel; output [1:0] logic; output logic\_ready;

```
input [7:0] DEBUG;
```

parameter CATCH\_THRESHOLD = 12'b1111111100; // the y velocity must be greater than this (negative) value

parameter THROW\_THRESHOLD = 12'b00000000110; // the acceleration must be greater than this value

```
parameter THROW_HISTORY_DEPTH = 6;
```

```
// the output velocity is an average
reg signed [15:0] x_vel[3:0];
reg signed [15:0] y_vel[3:0];
reg signed [15:0] y_accel[3:0];
reg throw_logic,catch_logic;
reg [THROW_HISTORY_DEPTH-1:0] throw_history;
reg [31:0] old_pos;
reg logic_ready;
```

//added this
parameter X\_VEL\_CONST = 19'b0; // LINE ADDED
parameter Y\_VEL\_CONST = 19'b111111110000000000; // LINE ADDED

wire signed [18:0] x\_vel\_sum = x\_vel[3]+x\_vel[2]+x\_vel[1]+x\_vel[0]+X\_VEL\_CONST; // LINE CHANGED

wire signed [18:0] y\_vel\_sum = y\_vel[3]+y\_vel[2]+y\_vel[1]+y\_vel[0]+Y\_VEL\_CONST; // LINE CHANGED

wire signed [17:0] y\_accel\_sum = y\_accel[3]+y\_accel[2]+y\_accel[1]+y\_accel[0];

```
assign vel = {x_vel_sum[17:2],y_vel_sum[18:3]};// LINE CHANGED
```

#### /\*

```
wire signed [17:0] x_vel_sum = x_vel[3]+x_vel[2]+x_vel[1]+x_vel[0];
wire signed [17:0] y_vel_sum = y_vel[3]+y_vel[2]+y_vel[1]+y_vel[0];
```

wire signed [17:0] y\_accel\_sum = y\_accel[3]+y\_accel[2]+y\_accel[1]+y\_accel[0];

assign vel = {x\_vel\_sum[17:2],y\_vel\_sum[17:2]};//{x\_vel\_sum[17:2],y\_vel[3]};//

## \*/

assign logic = {throw\_logic, catch\_logic & (!throw\_logic)};

```
always @ (posedge clk) begin
                                                                                      if (reset) begin
                                                                                                                                   \{x_vel[3], x_vel[2], x_vel[1], x_vel[0]\} \le 64'b0;
                                                                                                                                  {y_vel[3],y_vel[2],y_vel[1],y_vel[0]} <= 64'b0;
                                                                                                                                   {y_accel[3],y_accel[2],y_accel[1],y_accel[0]} <= 64'b0;
                                                                                                                                  throw_history <= 0;
                                                                                                                                   old_pos \leq 0;
                                                                                                                                  catch_logic <= 0;
                                                                                                                                  throw_logic <= 0;
                                                                                       end
                                                                                      else if (pos_ready) begin
                                                                                                                                   old_pos <= pos;
                                                                                                                                  \{x_vel[3], x_vel[2], x_vel[1], x_vel[0]\} \le \{x_vel[2], x_vel[1], x_vel[0], pos[31:16] - old_pos[31:16] - ol
[31:16]};
                                                                                                                                  \{y_vel[3], y_vel[2], y_vel[1], y_vel[0]\} \le \{y_vel[2], y_vel[1], y_vel[0], pos[15:0] - old_pos[15:0] - old_po
[15:0]};
```

 $\{y\_accel[3],y\_accel[2],y\_accel[1],y\_accel[0]\} <= \{y\_accel[2],y\_accel[1],y\_accel[0],pos[15:0]-y\_vel[0]\};$ 

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/vj_video_processor.txt$ 

```
catch_logic <= (y_vel_sum[17:6] < 12'h7FF) | (y_vel_sum[17:6] >
CATCH THRESHOLD);
              throw_history[THROW_HISTORY_DEPTH-1:0] <= {throw_history
[THROW_HISTORY_DEPTH-2:0],(y_accel_sum[17:6] < 12'h7FF) & (y_accel_sum[17:6] >
THROW_THRESHOLD) & !catch_logic };//(y_vel[3][15:4] + y_vel[2][15:4])-(y_vel[1][15:4]+y_vel[0]
[15:4]) > {8'hFF,DEBUG[7:4]};
              throw_logic <= throw_history[0] & (!throw_history[THROW_HISTORY_DEPTH-1:1]);
              logic_ready <= 1;
         end
         else if (logic_ready) begin
              logic_ready <= 0;
         end
    end
endmodule
/*
weighted_average
calculates a weighted average of
sequentially supplied values.
*/
module weighted_average(clk,reset,enable,average_reset,inval,average
         /*,sum,count,raw_average,unused_remainder*/);
    input clk;
    input reset;
    input enable;
    input average_reset;
    input [15:0] inval;
    output [15:0] average;
    //output [31:0] sum;
    //output [31:0] count;
    parameter IN_MIN = 16'h0000;
    parameter IN_MAX = 16'hFFFF;
    reg [31:0] sum = 0;
    reg [31:0] count = 0;
    reg [15:0] average;
                                // The top 16 bits of AVERAGE should always be 0
    wire [31:0] raw_average;
    wire [31:0] unused_remainder; // We ignore the remainder output.
```

```
wire unused_rfd;
```

```
always @ (posedge clk) begin
if (reset | average_reset) begin
sum <= 0;
count <= 0;
//average <= raw_average[15:0];
end
else if (enable && (inval > IN_MIN) && (inval < IN_MAX)) begin
sum <= sum + {16'b0,inval};
count <= count + 1;
end
average <= (|sum) ? raw_average[15:0] : 16'b0; // Force the average to be 0 if the sum is 0
end
```

divide average\_divide(sum, count, raw\_average, unused\_remainder, clk, unused\_rfd, 1'b0, 1'b0, 1'b1);

```
module yuv_to_rgb(y,u,v,vga_r,vga_g,vga_b);
```

input [7:0] y; input [7:0] u; input [7:0] v; output [7:0] vga\_r; output [7:0] vga\_g; output [7:0] vga\_b; assign vga\_r = y; assign vga\_g = y; assign vga\_b = y;

module vj\_controller(clk,new\_frame\_raw,reset\_raw,add\_ball\_raw,dec\_ball\_raw,left\_throw\_raw, right\_throw\_raw,gravity\_raw,

new\_frame\_sync,reset\_sync,add\_ball\_sync,dec\_ball\_sync,left\_throw\_sync,right\_throw\_sync, gravity\_sync); input clk; input new\_frame\_raw; input reset\_raw; input add\_ball\_raw; input dec\_ball\_raw; input left\_throw\_raw; input right\_throw\_raw; input [7:0] gravity\_raw; output new\_frame\_sync; output reset\_sync; output add\_ball\_sync; output dec\_ball\_sync; output left\_throw\_sync; output right\_throw\_sync; output [7:0] gravity\_sync; reg [7:0] gravity\_sync = 7'b0; reg reset\_sync=0; reg reset\_hold=0; reg right\_throw\_sync=0; reg right\_throw\_hold=0; reg left\_throw\_sync=0; reg left\_throw\_hold=0; reg new\_frame\_sync=0; reg add\_ball\_hold=0; reg add\_ball\_sync=0; reg dec\_ball\_hold=0; reg dec\_ball\_sync=0; wire dec\_ball\_pulse; wire add\_ball\_pulse; wire right\_throw\_force\_pulse; wire left\_throw\_force\_pulse; // reg right\_throw\_force\_hold=0; //to hold right throw that is a pulse until new frame // reg left\_throw\_force\_hold=0; // reg right\_throw\_force=0; // reg left\_throw\_force=0;

```
vj_level_to_pulse foradd(clk,add_ball_raw,add_ball_pulse);
   vj_level_to_pulse fordec(clk,dec_ball_raw,dec_ball_pulse);
   vj_level_to_pulse forleft(clk,left_throw_raw,left_throw_force_pulse);
   vj_level_to_pulse forright(clk,right_throw_raw,right_throw_force_pulse);
  always @ (posedge clk) begin
   new_frame_sync <= new_frame_raw;</pre>
   gravity_sync <= new_frame_raw ? gravity_raw : gravity_sync;</pre>
   reset_sync <= (new_frame_raw && reset_hold) ? 1 : 0;
   reset_hold <= reset_raw ? 1 : (new_frame_raw ? 0 : reset_hold);
   right_throw_sync <= (new_frame_raw && right_throw_hold) ? 1 : 0;
   right_throw_hold <= right_throw_force_pulse ? 1 : (new_frame_raw ? 0 : right_throw_hold);
   left_throw_sync <= (new_frame_raw && left_throw_hold) ? 1 : 0;
   left throw hold <= left throw force pulse ? 1 : (new frame raw ? 0 : left throw hold);
   add_ball_sync <= (new_frame_raw && add_ball_hold) ? 1 : 0;
   add ball hold <= add ball pulse ? 1 : (new frame raw ? 0 : add ball hold);
   dec_ball_sync <= (new_frame_raw && dec_ball_hold) ? 1 : 0;
   dec_ball_hold <= dec_ball_pulse ? 1 : (new_frame_raw ? 0 : dec_ball_hold);
/*
   right_throw_force <= new_frame_raw ? right_throw_force_hold : right_throw_force;
   right_throw_force_hold <= new_frame_raw ? 0 : (right_throw_force_pulse ? 1 :
right_throw_force_hold);
   left_throw_force <= new_frame_raw ? left_throw_force_hold : left_throw_force;</pre>
   left_throw_force_hold <= new_frame_raw ? 0 : (left_throw_force_pulse ? 1 :
left_throw_force_hold);
*/
 end
```

```
module vj_level_to_pulse(clk,level_in,pulse_out);
input clk;
input level_in;
output pulse_out;
reg level_hold=0;
reg pulse_out=0;
```

```
always @ (posedge clk) begin
  level_hold <= level_in;
  pulse_out <= (level_in && ~level_hold);
end</pre>
```

display\_enb,ball\_pos\_out,ball\_number,write\_request);

input vclock; // 65MHz clock input reset; // 1 to initialize module -- IS COMING in syncronized to new\_frame input new\_frame; //from Controller input [7:0] gravity; // gravity from switches debounced and synchronized to new\_frame input add\_ball; //adds a ball when button 1 is pressed input dec\_ball; //removes a ball when button 0 is pressed input left\_throw; //pulse when left hand is in throwing position input l\_h\_catch; //signal when left hand is in catch position input right\_throw; //when right hand throws input r\_h\_catch; //when right hand catches input [15:0] l\_h\_y; //left hand y position input [15:0] 1\_h\_x; // left hand x position input [15:0] l\_h\_y\_vel; //left hand y velocity input [15:0] l\_h\_x\_vel; // left hand x velocity input [15:0] r\_h\_y; //right hand y position input [15:0] r\_h\_x; // right hand x position input [15:0] r\_h\_y\_vel; //right hand y velocity input [15:0] r\_h\_x\_vel; // right hand x velocity

output display\_enb; //says the ball is to be displayed output [31:0] ball\_pos\_out; output [4:0] ball\_number; output write\_request;

wire display\_enb; wire [31:0] ball\_pos\_out; wire [4:0] ball\_number; wire write\_request; wire [69:0] info\_from\_bram; wire [4:0] bram\_addr\_b; wire wea; wire done\_calc; wire [69:0] ball\_info\_in;// = 70'b0; file:///Cl/Documents%20and%20Settings/cwilkens/My%20Documents/Coll...erilog%20Files/Verilog%20Source%20as%20Text/vj\_bm\_ball\_manager.txt

wire [69:0] ball\_info\_out;// = 70'b0; wire [4:0] count\_addr;// = 5'b0;

vj\_bm\_display instan\_bm\_disp(.vclock(vclock),.reset(reset),.new\_frame(new\_frame),.done\_calc (done\_calc),

.info\_from\_bram(info\_from\_bram),.bram\_addr\_b(bram\_addr\_b), .ball\_pos\_out(ball\_pos\_out),.display\_enb(display\_enb), .ball\_number(ball\_number),.write\_request(write\_request));

bram70x32 instan( .addra(count\_addr), .addrb(bram\_addr\_b), .clka(vclock), .clkb(vclock), .dina(ball\_info\_out), .douta(ball\_info\_in), .doutb(info\_from\_bram), .wea(wea)); // synthesis black\_box

ball\_info\_in,ball\_info\_out,count\_addr,done\_calc,wea);

input vclock; // 65MHz clock

input reset; // 1 to initialize module -- IS COMING in syncronized to new\_frame input new\_frame; //from Controller

input [7:0] gravity; // gravity from switches

input add\_ball; //is 1 for whole frame synced to new\_frame

input dec\_ball; //is 1 for whole frame and synced to new\_frame

input left\_throw; //pulse when left hand is in throwing position

input l\_h\_catch; //signal when left hand is in catch position

input right\_throw; //when right hand throws

input r\_h\_catch; //when right hand catches

input [15:0] l\_h\_y; //left hand y position

input [15:0] l\_h\_x; // left hand x position

input [15:0] l\_h\_y\_vel; //left hand y velocity

input [15:0] l\_h\_x\_vel; // left hand x velocity

input [15:0] r\_h\_y; //right hand y position

input [15:0] r\_h\_x; // right hand x position

input [15:0] r\_h\_y\_vel; //right hand y velocity

input [15:0] r\_h\_x\_vel; // right hand x velocity

input [69:0] ball\_info\_in; //ball data from BRAM

output [69:0] ball\_info\_out; // //ball data to BRAM output [4:0] count\_addr; //address to bram output done\_calc; output wea; //write enable to BRAM

parameter ball\_r =  $\{11'd40,5'b00000\};$ 

parameter ball\_w =  $\{11'd33,5'b00000\};$ parameter ball\_h =  $\{11'd33,5'b00000\};$ 

parameter hand\_ $h = \{11'd50, 5'b00000\};$ 

file:///Cl/Documents%20and%20Settings/cwilkens/My%20Documents/Colle...ject/Verilog%20Files/Verilog%20Source%20as%20Text/vj\_bm\_physics.txt

```
parameter hand_w = \{11'd140,5'b00000\};
```

```
parameter ball_right=0; //to keep track of states within case statement
parameter ball_left=1;
parameter ball_down=0;
parameter ball_up=1;
```

```
parameter screenw = {11'd1023,5'b00000};
parameter screenh = {11'd767,5'b00000};
```

wire in\_play\_in; wire in\_hand\_in; wire in\_left\_in; wire in\_right\_in; wire y\_dir\_in; wire x\_dir\_in; wire [15:0] ball\_y\_in; wire [15:0] ball\_x\_in; wire [15:0] y\_vel\_in; wire [15:0] x\_vel\_in;

```
assign in_play_in = ball_info_in[69];
assign in_hand_in = ball_info_in[68];
assign in_left_in = ball_info_in[67];
assign in_right_in = ball_info_in[66];
assign y_dir_in = ball_info_in[65];
assign x_dir_in = ball_info_in[64];
assign ball_y_in = ball_info_in[63:48];
assign ball_x_in = ball_info_in[47:32];
assign y_vel_in = ball_info_in[31:16];
assign x_vel_in = ball_info_in[15:0];
```

```
reg [15:0] ball_x_out = 16'h0000;
reg [15:0] ball_y_out = 16'd0;
//keep track of x and y velocities
reg [15:0] x_vel_out = 16'd0;
reg [15:0] y_vel_out = 16'd0;
```

reg x\_dir\_out = 0; //controls movement in x dir left or right
reg y\_dir\_out = 0; //controls movement in y dir up or down

wire [15:0] two\_x\_vel\_in;

assign two\_x\_vel\_in = (2 \* x\_vel\_in);

wire [15:0] two\_y\_vel\_in; assign two\_y\_vel\_in = (2 \* y\_vel\_in);

//so i have a comparative value to know if the ball is too close to the edge wire [15:0] far\_right; // assign far\_right = ball\_x\_in + ball\_w; // how far toward right the ball is wire [15:0] far\_bot; assign far\_bot = ball\_y\_in + ball\_h; //how far toward botton the ball is

//ball movement control with everything else around it

parameter read\_state = 0;

parameter write\_state = 1;

reg read\_hold=0; //so the program stays in the read\_state two cycles per round.

reg read\_or\_write=0; // to determine whether we're reading or writing to the BRAM alternating the we signal

reg [6:0]balls= 7'b0; //keeps track of how many balls are currently in the system

reg [4:0]count\_addr= 5'b0; //keeps track of which BRAM address the ball controller is at

reg done\_calc=0; //all the balls have been read and written during a frame when high

reg in\_play\_out=0; //ball is in play when high

reg in\_hand\_out=0; //ball is in a hand when high

reg in\_left\_out=0; //ball in in left hand when high

```
reg in_right_out=0; //ball is in right hand when high
```

```
reg wea=0; // write enable on BRAM
```

reg which\_hand=0; //switches which hand the added ball goes into

reg reset\_level=0; // leveled reset to be high for one entire frame to reset each value

reg add\_ball\_level=0; //high when add ball goes high and stays until a ball is added (add\_ball synced to new\_frame)

reg dec\_ball\_level=0; //high when dec\_ball goes high until ball is deleted (dec\_ball synced to new\_frame)

//reg ball\_added=0; //high when ball has been added and set low every new frame

//reg ball\_deced=0; //high when ball has been removed and set low every new frame

reg [1:0] count\_divider= 2'b0; //so count\_addr increments only once per 3 clock cycles

reg left\_thrown = 0; //signals within one frame that a ball has been thrown from the left hand so no more that frame

reg right\_thrown = 0; //signals within one frame that a ball has been thrown from right hand and no more that frame

reg left\_throw\_level=0;

```
reg right_throw_level=0;
```

//display\_16hex hexInstan(reset, vclock, {ball\_x,x\_vel,two\_x\_vel,neg\_two\_x\_vel}, disp\_blank, disp\_clock, disp\_rs, disp\_ce\_b, disp\_reset\_b, disp\_data\_out);

always @ (posedge vclock) begin

```
// reset_level <= reset ? 1 : (new_frame ? 0 : reset_level); //makes a reset level signal for one frame
// add_ball_level <= add_ball ? 1 : (new_frame ? 0 : add_ball_level); //,makes add ball level for one
frame
// dec_ball_level <= dec_ball ? 1 : (new_frame ? 0 : dec_ball_level);</pre>
// right_thrown <= new_frame ? 0 : right_thrown;</pre>
// left_thrown <= new_frame ? 0 : left_thrown;</pre>
// right_throw_level <= ?</pre>
/* if(~add_ball_level) begin
    add_ball_level <= add_ball ? 1 : add_ball_level;
  end
  if(~dec_ball_level) begin
    dec_ball_level <= dec_ball ? 1 : dec_ball_level;</pre>
  end
*/
  if (new_frame) begin
       done_calc \leq 0;
    count_addr <= 0;
    count_divider <= 3;
    wea <= 0; ///added for debugging
    read_hold \leq = 0;
    reset level <= reset;
    left_throw_level <= left_throw;</pre>
    right_throw_level <= right_throw;
    which_hand <= add_ball ? ~which_hand : which_hand; //switches which hand ball goes into each
time you add one
    add_ball_level <= add_ball;
    dec_ball_level <= dec_ball;
```

```
right_thrown <= 0;
 left_thrown <= 0;
end
   else if (count_addr == 31) begin
     done_calc <= 1;
   end
else if (~done_calc) begin //includes case statement
 count_divider <= (count_divider == 2) ? 0 : (count_divider + 1);
     count_addr <= (count_divider == 2) ? (count_addr + 1) : count_addr;
  case (read_or_write)
   read_state: begin
        read_hold <= ~read_hold;</pre>
        wea \leq 0;
        read_or_write <= read_hold; //sent it to write_state after 2nd cycle in read_state
       end
   write_state: begin
     read_or_write <= read_state; //send it back to read state after 1 cycle in write state
     wea <= 1:
     if (reset || reset_level) begin
       if (count_addr < 2) begin
         in_play_out <= 1;
         in_hand_out <= 1;
         in_left_out <= 0;</pre>
         in_right_out <= 1;
         y_dir_out <= 1'b0;
         x_dir_out <= 1'b0;
         ball_y_out <= 16'b0;
            ball_x_out <= 16'b0;
            y_vel_out <= 16'b0;
            x_vel_out <= 16'b0;
       end
       else if (count_addr == 2)begin
         in_play_out <= 1;
         in_hand_out <= 1;
         in_left_out <= 1;
         in_right_out <= 0;
         y_dir_out <= 1'b0;
            x_dir_out \leq 1'b0;
            ball_y_out <= 16'b0;
            ball_x_out <= 16'b0;
```

```
y_vel_out <= 16'b0;
      x_vel_out <= 16'b0;
  end
  else begin
   in_play_out <= 0;
   in_hand_out <= 0;
   in_left_out <= 0;
   in_right_out <= 0;
   y_dir_out <= 1'b0;
      x_dir_out <= 1'b0;
       ball_y_out <= 16'b0;
      ball_x_out <= 16'b0;
       y_vel_out <= 16'b0;
       x_vel_out <= 16'b0;
  end
end
else if (in_play_in) begin //if it's in play do all this
 if (dec_ball_level) begin
   in_play_out <= 0;
   dec_ball_level <= 0;
   end
```

```
else if (in_hand_in) begin // if it is in a hand do this
```

```
in_play_out <= 1;
in_hand_out <= 0;
in_left_out <= 0;
in_right_out <= 0;</pre>
ball_x_out <= r_h_x;
ball_y_out <= r_h_y;
            x_vel_out \le r_h_x_vel[15]? -r_h_x_vel: r_h_x_vel;
            y_vel_out <= r_h_y_vel[15] ? -r_h_y_vel : r_h_y_vel;
            y_dir_out \leq r_h_y_vel[15] ? ball_up : ball_down;
            x_dir_out <= r_h_x_vel[15] ? ball_left : ball_right;
right_thrown <= 1;
          end
    else if (in_left_in && left_throw_level && ~left_thrown) begin
            in_play_out <= 1;
in_hand_out <= 0;
in_left_out <= 0;
in_right_out <= 0;
```

$$ball_x_out <= l_h_x; \\ball_y_out <= l_h_x, vel[15]? -l_h_x_vel : l_h_x_vel; \\y_vel_out <= l_h_x_vel[15]? -l_h_y_vel : l_h_y_vel; \\y_vel_out <= l_h_x_vel[15]? ball_op: ball_down; \\x_dr_out <= l_h_x_vel[15]? ball_left: ball_right; \\left_thrown <= 1; \\end \\else begin \\in_play_out <= in_play_in; \\in_hand_out <= in_hand_in; \\in_left_out <= in_ileft_in; \\in_right_out <= ball_x_in; \\ball_y_out <= ball_y_in; \\x_vel_out <= x_vel_in; \\y_vel_out <= y_vel_in; \\y_vel_out <= y_vel_in; \\y_vel_out <= y_dir_in; \\x_dir_out <= x_dir_in; \\end \\end \\else begin //if not in hand do all this \\in_play_out <= 1; \\if (l_h_catch &&  l_h(n) && \\(lt_hand_w/2) + ball_x(n) > l_h(n) && \\(lt_hand_w/2) + ball_x(n) - l_h(n) && \\(lt_hand_w/2) + lt_h(lt_h(n) && \\(lt_hand_w/2) + lt_h(n) && \\(lt_hand_w/2) && \\(lt_hand$$

```
in_hand_out <= 1;
                            in_right_out <= 1;</pre>
                            in_left_out \leq 0;
                        end
                  else begin
             //x_vel_out <= x_vel_in;</pre>
             in_left_out <= in_left_in;
             in right out <= in right in;
             in hand out \leq 0;
                                     //needs to be set otherwise if a ball is added to the hand in_hand is
then one
             in_right_out <= 0;
             in_left_out <= 0;
             if (x_dir_in == ball_right) begin
               ball_x_out <= ball_x_in + x_vel_in;
               x_dir_out \le (screenw \le (two_x_vel_in + far_right)) ? ball_left : ball_right;
               x_vel_out \le (screenw < (two_x_vel_in + far_right)) ? (x_vel_in / 2) : x_vel_in;
             end
             else begin
               ball_x_out <= ball_x_in - x_vel_in;
               <u>x_dir_out</u> <= (ball_x_in < (two_x_vel_in/* + ball_w*/)) ? ball_right : ball_left;
               x_vel_out \le (ball_x_in < (two_x_vel_in/* + ball_w*/)) ? (x_vel_in / 2) : x_vel_in;
             end
             if (y_dir_in == ball_down) begin
               ball_y_out <= ball_y_in + y_vel_in;
               y_dir_out <= (screenh < (far_bot + (two_y_vel_in + gravity))) ? ball_up : ball_down;</pre>
               y_vel_out <= (y_vel_in + gravity);</pre>
             end
             else begin
               ball_y_out <= ball_y_in - y_vel_in;</pre>
               y_dir_out <= ((ball_y_in < two_y_vel_in) || (y_vel_in < gravity)) ? ball_down : ball_up;</pre>
               y_vel_out \le (ball_y_in < two_y_vel_in) ? 16'd32 : ((y_vel_in < gravity) ? (gravity -
y_vel_in) : (y_vel_in - gravity));
             end
           end
         end
        end
        else begin //when not in play do this
         if (add_ball_level) begin
           in_play_out <= 1;
           in_hand_out <= 1;
           in_left_out <= which_hand;
           in_right_out <= ~which_hand;
               y_dir_out \le 1'b0;
```

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/vj_bm_physics.txt$ 

```
x_dir_out \leq 1'b0;
           ball_y_out <= 16'b0;
           ball_x_out <= 16'b0;
           y_vel_out <= 16'b0;
           x_vel_out <= 16'b0;
        add_ball_level <= 0;
         end
      else begin
        in_play_out <= 1'b0;
        in_hand_out <= 1'b0;
        in_left_out <= 1'b0;
        in_right_out <= 1'b0;</pre>
           y_dir_out <= 1'b0;
           x_dir_out \le 1'b0;
           ball_y_out <= 16'b0;
           ball_x_out <= 16'b0;
           y_vel_out <= 16'b0;
           x_vel_out <= 16'b0;
      end
    end
  end
endcase
end
```

assign ball\_info\_out = {in\_play\_out,in\_hand\_out,in\_left\_out,in\_right\_out,y\_dir\_out,x\_dir\_out, ball\_y\_out,ball\_x\_out,y\_vel\_out,x\_vel\_out};

endmodule

end

input vclock; // 65MHz clock input reset; // 1 to initialize module -- IS COMING in syncronized to new\_frame input new\_frame; //from Controller input done\_calc; input [69:0] info\_from\_bram;

```
output [4:0] bram_addr_b;
output [31:0] ball_pos_out;
output display_enb;
output [4:0] ball_number;
output write_request;
// output stop_calc;
```

```
reg [4:0]bram_addr_b=0;
reg [31:0] ball_pos_out=0;
reg display_enb=0; //need to set equal to 0
reg [4:0] ball_number=0;
reg write_request=0;
```

reg stop\_calc=0; reg [4:0]ball\_number\_hold=0; reg write\_request\_hold=0;

```
always @ (posedge vclock) begin
if (reset) begin
stop_calc <= 0;
bram_addr_b <= 0;
write_request <= 0;
ball_number <= 0;
end
else if (stop_calc) begin
stop_calc <= new_frame ? 0 : stop_calc;
bram_addr_b <= new_frame ? 0 : bram_addr_b;
write_request <= 0;
ball_number <= new_frame ? 0 : ball_number;
end
else if (done_calc) begin
```

file:///Cl/Documents%20and%20Settings/cwilkens/My%20Documents/Colle...ject/Verilog%20Files/Verilog%20Source%20as%20Text/vj\_bm\_display.txt

bram\_addr\_b <= (bram\_addr\_b + 1); //inc bram\_addr until is reaches the last value then reset stop\_calc <= (ball\_number == 31) ? 1 : stop\_calc; //stop\_calculations when you hit the last bram address

write\_request\_hold <= (ball\_number == 30) ? 0 : 1; //stop at 30 because of delay in pipeline
write\_request <= write\_request\_hold;</pre>

ball\_pos\_out <= {info\_from\_bram[47:32],info\_from\_bram[63:48]}; //x now in high and y in low 16 bits

display\_enb <= (ball\_number == 31) ? 0 : (info\_from\_bram[69] && ~(info\_from\_bram[68] || info\_from\_bram[67]));

ball\_number <= ball\_number\_hold; //to delay ball\_number ball\_number\_hold <= bram\_addr\_b;</pre>

end

end

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...0 Project/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/vj_output.txt/Source\%20 Source\%20 Source\%2$ 

/\* vj\_output

This is the output module for the juggling simulator. It takes a video feed and overlays the hand and ball sprites.

Notes:

This module delays the video feed by TWO clock cycles. The first cycle allows the system to

\*/

module vj\_output(clock\_65mhz,reset, // Video inputs r,g,b, hcount,vcount,

sync\_in,blank\_in,hsync\_in,vsync\_in,

// Sprite position inputs
left\_hand\_pos,right\_hand\_pos,
we,ball\_num,new\_pos,be,

// Video outputs
 vga\_out\_red,vga\_out\_green,vga\_out\_blue,
 vga\_out\_sync\_b,vga\_out\_blank\_b,
 vga\_out\_hsync,vga\_out\_vsync);
input clock\_65mhz;
input reset;

// Incoming XGA video signals
input [7:0] r, g, b;
input [10:0] hcount;
input [9:0] vcount;
input sync\_in, blank\_in, hsync\_in, vsync\_in;

// Incoming hand locations
input [31:0] left\_hand\_pos, right\_hand\_pos;

// Incoming ball-sprite memory signals
input we;
input [4:0] ball\_num;

input [31:0] new\_pos; input be;

// Output XGA signals
output [7:0] vga\_out\_red, vga\_out\_green, vga\_out\_blue;
output vga\_out\_sync\_b, vga\_out\_blank\_b;
output vga\_out\_hsync, vga\_out\_vsync;

```
reg vga_out_sync_b, vga_out_blank_b;
reg vga_out_hsync, vga_out_vsync;
reg vga_out_red, vga_out_green, vga_out_blue;
```

```
wire [23:0] overlay_pixel, left_hand_pixel, right_hand_pixel, ball_pixel;
```

```
vj_output_hand_sprite left_hand_sprite(left_hand_pos[31:21],left_hand_pos[14:5], hcount,vcount,left_hand_pixel);
```

- vj\_output\_hand\_sprite right\_hand\_sprite(right\_hand\_pos[31:21],right\_hand\_pos[14:5], hcount,vcount,right\_hand\_pixel);
- vj\_output\_agregate\_ball\_sprite\_memory ball\_sprite(clock\_65mhz,reset,hcount,vcount, ball\_pixel,we,ball\_num,new\_pos,be);

```
// We OR the output of the left hand, right hand, and ball pixels
// to determine the overlay pixel. A black overlay pixel is not displayed.
assign overlay_pixel = left_hand_pixel | right_hand_pixel | ball_pixel;
```

// Registers for pipelining the output
reg [3:0] vga\_signal\_pipeline;
reg [23:0] vga\_color\_pipeline;

```
always @ (posedge clock_65mhz) begin
```

// Delay the video by TWO clock cycles

// one cycle is for computing output,

// the other is to guarantee good setup/hold times

vga\_signal\_pipeline <= {sync\_in,blank\_in,hsync\_in,vsync\_in};</pre>

```
{vga_out_sync_b,vga_out_blank_b,vga_out_hsync,vga_out_vsync} <= vga_signal_pipeline;
```

```
vga_color_pipeline <= (|overlay_pixel) ? overlay_pixel : {r,g,b};
{vga_out_red,vga_out_green,vga_out_blue} <= vga_color_pipeline;</pre>
```

end

 $file:///C/Documents\% 20 and\% 20 Settings/cwilkens/My\% 20 Documents/Colle...0 Project/Verilog\% 20 Files/Verilog\% 20 Source\% 20 as\% 20 Text/vj_output.txt files///C/Documents\% 20 and\% 20 Settings/cwilkens/My\% 20 Documents/Colle...0 Project/Verilog\% 20 Files/Verilog\% 20 Source\% 20 as\% 20 Text/vj_output.txt files///C/Documents/Colle...0 Project/Verilog\% 20 Files/Verilog\% 20 Source\% 20 as\% 20 Text/vj_output.txt files///C/Documents/Colle...0 Project/Verilog\% 20 as\% 20 Text/vj_output.txt files///C/Documents/Colle...0 Project/Verilog\% 20 as\% 20 Text/vj_output.txt files///C/Documents/Colle...0 Project/Verilog\% 20 at\% 20 a$ 

/\*

This module acts as a single sprite for displaying all the balls on the screen. The output module provides hcount and vcount, while the module returns an output pixel color. As with other sprites, this logic is purely combinational.

This module also acts as a form of memory. By providing synchronous inputs, other modules may modify the location of a given ball on the screen.

\*/

input we; // Write enable signal (syncronous)
input [4:0] ball\_num; // Number of ball to modify (synchronous)
input [31:0] new\_pos; // New position of ball (synchronous)
input be;

//DEBUG
//output [11\*4-1:0] x\_out;
//output [10\*4-1:0] y\_out;
//output [4-1:0] ball\_enable\_out;

```
parameter NUM_BALLS = 32;
```

reg [11\*32-1:0] x = 0; reg [10\*32-1:0] y = 0; reg [32-1:0] ball\_enable = 0; wire [32-1:0] ball\_pixels;

```
//DEBUG
```

//assign x\_out = x[11\*4-1:0];
//assign y\_out = y[10\*4-1:0];
//assign ball\_enable\_out = ball\_enable[4-1:0];

vj\_output\_ball\_sprite ball\_sprites[32-1:0](x,y,ball\_enable,hcount,vcount,ball\_pixels);

assign pixel\_out = (|ball\_pixels) ? 24'h00FFFF : 24'h0;

```
always @ (posedge clk) begin
         if (reset) begin
              x <= 0;
              y <= 0;
              ball_enable \leq 0;
         end
         else if (we) begin
              //x[ball_num+10:ball_num] \le new_pos[26:16];
              //y[ball_num+9:ball_num] <= new_pos[9:0];</pre>
              x[11*ball_num+10] \le new_pos[31];
              x[11*ball_num+9] \le new_pos[30];
              x[11*ball_num+8] \le new_pos[29];
              x[11*ball_num+7] \le new_pos[28];
              x[11*ball num+6] \le new pos[27];
              x[11*ball_num+5] \le new_pos[26];
              x[11*ball num+4] \le new pos[25];
              x[11*ball_num+3] \le new_pos[24];
              x[11*ball_num+2] \le new_pos[23];
              x[11*ball_num+1] <= new_pos[22];
              x[11*ball_num] \le new_pos[21];
              v[10*ball_num+9] \le new_pos[14];
              y[10*ball_num+8] <= new_pos[13];
              y[10*ball_num+7] \le new_pos[12];
              y[10*ball_num+6] <= new_pos[11];</pre>
              y[10*ball_num+5] \le new_pos[10];
              y[10*ball_num+4] <= new_pos[9];</pre>
              y[10*ball_num+3] \le new_pos[8];
              y[10*ball_num+2] \le new_pos[7];
              y[10*ball_num+1] \le new_pos[6];
              y[10*ball_num] \le new_pos[5];
              ball_enable[ball_num] <= be;
         end
    end
endmodule
```

/\*

This module is a sprite for displaying a single hand on the screen.

 $file:///C/Documents\% 20 and\% 20 Settings/cwilkens/My\% 20 Documents/Colle...0 Project/Verilog\% 20 Files/Verilog\% 20 Source\% 20 as\% 20 Text/vj_output.txt files///C/Documents\% 20 and\% 20 Settings/cwilkens/My\% 20 Documents/Colle...0 Project/Verilog\% 20 Files/Verilog\% 20 Source\% 20 as\% 20 Text/vj_output.txt files///C/Documents/Colle...0 Project/Verilog\% 20 Files/Verilog\% 20 Source\% 20 as\% 20 Text/vj_output.txt files///C/Documents/Colle...0 Project/Verilog\% 20 as\% 20 Text/vj_output.txt files///C/Documents/Colle...0 Project/Verilog\% 20 as\% 20 Text/vj_output.txt files///C/Documents/Colle...0 Project/Verilog\% 20 at\% 20 a$ 

Note that this sprite doesn't contain its own color information.

# CODE BASE TAKEN FROM LAB 4 \*/

```
module vj_output_hand_sprite(x,y,hcount,vcount,pixel);
    parameter WIDTH = 16;
    parameter HEIGHT = 16;
    parameter COLOR = 24'h00FF00;
    input [10:0] x;
    input [9:0] y;
    input [10:0] hcount;
    input [9:0] vcount;
    output [23:0] pixel;
    reg [23:0] pixel;
    always @ (x or y or hcount or vcount) begin
         if ((hcount \geq x \&\& hcount < (x+WIDTH)) \&\&
                   (vcount \ge y \&\& vcount < (y+HEIGHT)))
              pixel = COLOR;
         else pixel = 0;
    end
endmodule
```

/\*

This module is a sprite for displaying a single ball on the screen.

## CODE BASE TAKEN FROM LAB 4

\*/

```
module vj_output_ball_sprite(x,y,enable,hcount,vcount,pixel);
    parameter RADIUS = 16;
    input [10:0] x;
    input [9:0] y;
    input enable;
    input [10:0] hcount;
    input [9:0] vcount;
    output pixel;
    reg pixel;
```

// Compute the position deltas before we use them
wire [10:0] delta\_x = x-hcount;
wire [9:0] delta\_y = y-vcount;

```
always @ (enable or x or y or hcount or vcount) begin
         if (enable &&
                   (hcount \geq (x-RADIUS) && hcount \leq (x+RADIUS)) &&
                   (vcount >= (y-RADIUS) && vcount <= (y+RADIUS))) begin
              //pixel = 1;
              // In order to avoid delaying the video signal, we hard-wire the hands
              /*if ((delta_x <= 11'b0000000010) || (delta_x >= 11'b1111111101))
                   pixel = 1;
              else if ((delta_y <= 10'b000000010) || (delta_y >= 10'b111111101))
                   pixel = 1;
              else if ((((delta_x <= 11'b0000000100) || (delta_x >= 11'b1111111011))))
                        && (((delta_y <= 10'b000001000) \parallel (delta_y >= 10'b1111110111))))
                   pixel = 1;
              else if ((((delta_x <= 11'b0000001000) \parallel (delta_x >= 11'b1111110111)))
                        && (((delta_y <= 10'b000000100) \parallel (delta_y >= 10'b1111111011))))
                   pixel = 1;
              else if ((delta_x+delta_y <= 11'b00000001010) || (delta_x+delta_y >= 11'b1111110101))
                   pixel = 1;*/
              if (((delta_x+delta_y <= 2*RADIUS) || (delta_x+delta_y >= 2*RADIUS))&&
                        ((delta_x-delta_y \le 2*RADIUS) \parallel (delta_x-delta_y \ge 2*RADIUS)))
                   pixel = 1;
              else
                   pixel = 0;
         end
         else pixel = 0;
    end
endmodule
```

```
/*
```

This module is used for testing the output module. It displays the hand sprites and all 32 balls on the screen.

## \*/

module vj\_output\_test\_jig(clock\_65mhz,reset,new\_frame,we,ball\_number,new\_pos,be,DEBUG\_DATA, SWITCHES);

```
input clock_65mhz;
input reset;
input new_frame;
```

```
output we;
output [4:0] ball_number;
output [31:0] new_pos;
```

output be;

```
input [4:0] SWITCHES;
output [31:0] DEBUG_DATA;
reg counting = 0;
reg we,be;
reg [4:0] ball_number;
reg [31:0] new_pos;
reg [31:0] inner_counter = 32'b0;
assign DEBUG_DATA = inner_counter;
always @ (posedge clock_65mhz) begin
    if (reset) begin
         inner_counter <= 0;
         counting \leq 0;
         ball_number <= 0;
     end
    if (new_frame) begin
         counting \leq 1;
         ball_number <= 0;
         inner_counter <= inner_counter + 1;
     end
         if (counting) begin
     else
         if (ball_number == 5'b11111) counting <= 0;//stop counting.
         ball_number <= ball_number+1;</pre>
          we <= 1;
         be <= inner_counter[4+ball_number[0]];
         new_pos \le \{(\{1'b0, ball_number[2:0], inner_counter[10:4], 5'b0\}) + 16'd64, \}
                        ({2'b0,ball_number[4:3],inner_counter[10:4],5'b0})+16'd64};
     end
           begin
     else
         we <= 0;
         be <= 0;
     end
end
```

```
endmodule
```

file:///C|/Documents%20 and%20 Settings/cwilkens/My%20 Documents/Colle... l%20 Project/Verilog%20 Files/Verilog%20 Source%20 as%20 Text/divide.txt

| /***                                                                        | ***************************************                                    |  |  |
|-----------------------------------------------------------------------------|----------------------------------------------------------------------------|--|--|
| *                                                                           | This file is owned and controlled by Xilinx and must be used *             |  |  |
|                                                                             | solely for design, simulation, implementation and creation of *            |  |  |
|                                                                             | design files limited to Xilinx devices or technologies. Use *              |  |  |
|                                                                             | with non-Xilinx devices or technologies is expressly prohibited *          |  |  |
|                                                                             | and immediately terminates your license.                                   |  |  |
| *                                                                           | *                                                                          |  |  |
| *                                                                           | XILINX IS PROVIDING THIS DESIGN, CODE, OR INFORMATION "AS IS" *            |  |  |
| *                                                                           | SOLELY FOR USE IN DEVELOPING PROGRAMS AND SOLUTIONS FOR *                  |  |  |
| *                                                                           | XILINX DEVICES. BY PROVIDING THIS DESIGN, CODE, OR INFORMATION *           |  |  |
| *                                                                           | AS ONE POSSIBLE IMPLEMENTATION OF THIS FEATURE, APPLICATION *              |  |  |
| *                                                                           | OR STANDARD, XILINX IS MAKING NO REPRESENTATION THAT THIS *                |  |  |
| *                                                                           | IMPLEMENTATION IS FREE FROM ANY CLAIMS OF INFRINGEMENT, *                  |  |  |
| *                                                                           | AND YOU ARE RESPONSIBLE FOR OBTAINING ANY RIGHTS YOU MAY REQUIRE           |  |  |
| *                                                                           | FOR YOUR IMPLEMENTATION. XILINX EXPRESSLY DISCLAIMS ANY *                  |  |  |
| *                                                                           | WARRANTY WHATSOEVER WITH RESPECT TO THE ADEQUACY OF THE *                  |  |  |
| *                                                                           | IMPLEMENTATION, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OR *           |  |  |
| *                                                                           | REPRESENTATIONS THAT THIS IMPLEMENTATION IS FREE FROM CLAIMS OF *          |  |  |
| *                                                                           | INFRINGEMENT, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS *          |  |  |
| *                                                                           | FOR A PARTICULAR PURPOSE. *                                                |  |  |
| *                                                                           | *                                                                          |  |  |
| *                                                                           | Xilinx products are not intended for use in life support *                 |  |  |
| *                                                                           | appliances, devices, or systems. Use in such applications are *            |  |  |
| *                                                                           | expressly prohibited. *                                                    |  |  |
| *                                                                           | *                                                                          |  |  |
| *                                                                           | (c) Copyright 1995-2004 Xilinx, Inc. *                                     |  |  |
|                                                                             | All rights reserved. *                                                     |  |  |
| ****                                                                        | ***************************************                                    |  |  |
| // The synopsys directives "translate_off/translate_on" specified below are |                                                                            |  |  |
| // su                                                                       | pported by XST, FPGA Compiler II, Mentor Graphics and Synplicity synthesis |  |  |

\*

// tools. Ensure they are correct for your synthesis tool(s).

// You must compile the wrapper file divide.v when simulating

// the core, divide. When compiling the wrapper file, be sure to

// reference the XilinxCoreLib Verilog simulation library. For detailed

// instructions, please refer to the "CORE Generator Guide".

module divide ( dividend, divisor, quot, remd, clk, rfd, aclr, sclr, ce); // synthesis black\_box

input [31:0] dividend;

- input [31:0] divisor;
- output [31 : 0] quot;
- output [31 : 0] remd;
- input clk;

output rfd;

input aclr;

input sclr;

input ce;

// synopsys translate\_off

### SDIVIDER\_V3\_0 #(

- 0, // c\_has\_aclr
- 0, // c\_has\_ce
- 0, // c\_has\_sclr
- 1, // c\_sync\_enable
- 1, // divclk\_sel
- 32, // dividend\_width
- 32, // divisor\_width
- 0, // fractional\_b
- 32, // fractional\_width
- 0) // signed\_b

inst (

.DIVIDEND(dividend), .DIVISOR(divisor), .QUOT(quot), .REMD(remd), .CLK(clk), .RFD(rfd), .ACLR(aclr), .SCLR(sclr),

.CE(ce));

// synopsys translate\_on

file:///C|/Documents%20 and%20 Settings/cwilkens/My%20 Documents/Colle... l%20 Project/Verilog%20 Files/Verilog%20 Source%20 as%20 Text/divide.txt

// FPGA Express black box declaration
// synopsys attribute fpga\_dont\_touch "true"
// synthesis attribute fpga\_dont\_touch of divide is "true"

// XST black box declaration
// box\_type "black\_box"
// synthesis attribute box\_type of divide is "black\_box"

endmodule

file:///C|/Documents%20 and%20 Settings/cwilkens/My%20 Documents/Colle...0 Project/Verilog%20 Files/Verilog%20 Source%20 as%20 Text/bram70 x 32.txt

| /**> | ***************************************                                    |
|------|----------------------------------------------------------------------------|
| *    | This file is owned and controlled by Xilinx and must be used *             |
| *    | solely for design, simulation, implementation and creation of *            |
| *    | design files limited to Xilinx devices or technologies. Use *              |
| *    | with non-Xilinx devices or technologies is expressly prohibited *          |
| *    | and immediately terminates your license.                                   |
| *    | *                                                                          |
| *    | XILINX IS PROVIDING THIS DESIGN, CODE, OR INFORMATION "AS IS" *            |
| *    | SOLELY FOR USE IN DEVELOPING PROGRAMS AND SOLUTIONS FOR *                  |
| *    | XILINX DEVICES. BY PROVIDING THIS DESIGN, CODE, OR INFORMATION *           |
| *    | AS ONE POSSIBLE IMPLEMENTATION OF THIS FEATURE, APPLICATION *              |
| *    | OR STANDARD, XILINX IS MAKING NO REPRESENTATION THAT THIS *                |
| *    | IMPLEMENTATION IS FREE FROM ANY CLAIMS OF INFRINGEMENT, *                  |
| *    | AND YOU ARE RESPONSIBLE FOR OBTAINING ANY RIGHTS YOU MAY REQUIRE           |
| *    | FOR YOUR IMPLEMENTATION. XILINX EXPRESSLY DISCLAIMS ANY                    |
| *    | WARRANTY WHATSOEVER WITH RESPECT TO THE ADEQUACY OF THE *                  |
| *    | IMPLEMENTATION, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OR *           |
| *    | REPRESENTATIONS THAT THIS IMPLEMENTATION IS FREE FROM CLAIMS OF *          |
| *    | INFRINGEMENT, IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS *          |
| *    | FOR A PARTICULAR PURPOSE. *                                                |
| *    | *                                                                          |
| *    | Xilinx products are not intended for use in life support *                 |
| *    | appliances, devices, or systems. Use in such applications are *            |
| *    | expressly prohibited. *                                                    |
| *    | *                                                                          |
| *    | (c) Copyright 1995-2004 Xilinx, Inc. *                                     |
| *    | All rights reserved.                                                       |
| ***  | ***************************************                                    |
| // T | he synopsys directives "translate_off/translate_on" specified below are    |
| 11   | and and her VST EDCA Commiler II. Manton Crambias and Semulicity synthesis |

\*

// supported by XST, FPGA Compiler II, Mentor Graphics and Synplicity synthesis
// tools. Ensure they are correct for your synthesis tool(s).

// You must compile the wrapper file bram70x32.v when simulating
// the core, bram70x32. When compiling the wrapper file, be sure to
// reference the XilinxCoreLib Verilog simulation library. For detailed
// is the time of the #CODE Compiler of the #CODE Compiler
// is the time of t

// instructions, please refer to the "CORE Generator Guide".

module bram70x32 ( addra, addrb, clka, clkb, dina, douta, doutb, wea); // synthesis black\_box

input [4 : 0] addra; input [4 : 0] addrb; input clka; input clkb; input [69 : 0] dina; output [69 : 0] douta; output [69 : 0] doutb; input wea;

// synopsys translate\_off

#### BLKMEMDP\_V6\_1 #(

- 5, // c\_addra\_width
- 5, // c\_addrb\_width
- "0", // c\_default\_data
- 32, // c\_depth\_a
- 32, // c\_depth\_b
- 0, // c\_enable\_rlocs
- 1, // c\_has\_default\_data
- 1, // c\_has\_dina
- 0, // c\_has\_dinb
- 1, // c\_has\_douta
- 1, // c\_has\_doutb
- 0, // c\_has\_ena
- 0, // c\_has\_enb
- 0, // c\_has\_limit\_data\_pitch
- 0, // c\_has\_nda
- 0, // c\_has\_ndb
- 0, // c\_has\_rdya
- 0, // c\_has\_rdyb
- 0, // c\_has\_rfda
- 0, // c\_has\_rfdb
- 0, // c\_has\_sinita
- 0, // c\_has\_sinitb
- 1, // c\_has\_wea
- 0, // c\_has\_web
- 18, // c\_limit\_data\_pitch

"mif\_file\_16\_1", // c\_mem\_init\_file

file:///Cl/Documents%20and%20Settings/cwilkens/My%20Documents/Colle...0Project/Verilog%20Files/Verilog%20Source%20as%20Text/bram70x32.txt

- 0, // c\_pipe\_stages\_a
- 0, // c\_pipe\_stages\_b
- 0, // c\_reg\_inputsa
- 0, // c\_reg\_inputsb
- "0", // c\_sinita\_value
- "0", // c\_sinitb\_value
- 70, // c\_width\_a
- 70, // c\_width\_b
- 0, // c\_write\_modea
- 0, // c\_write\_modeb
- "0", // c\_ybottom\_addr
- 1, // c\_yclka\_is\_rising
- 1, // c\_yclkb\_is\_rising
- 1, // c\_yena\_is\_high
- 1, // c\_yenb\_is\_high
- "hierarchy1", // c\_yhierarchy
- 0, // c\_ymake\_bmm
- "16kx1", // c\_yprimitive\_type
- 1, // c\_ysinita\_is\_high
- 1, // c\_ysinitb\_is\_high
- "1024", // c\_ytop\_addr
- 0, // c\_yuse\_single\_primitive
- 1, // c\_ywea\_is\_high
- 1, // c\_yweb\_is\_high
- 1) // c\_yydisable\_warnings

# inst (

```
.ADDRA(addra),
```

- .ADDRB(addrb),
- .CLKA(clka),
- .CLKB(clkb),
- .DINA(dina),
- .DOUTA(douta),
- .DOUTB(doutb),
- .WEA(wea),
- .DINB(),
- .ENA(),
- .ENB(), .NDA(),
- .NDB(),
- .RFDA(),
- .RFDB(),
- .RDYA(),
- .RDYB(),

## .SINITA(), .SINITB(), .WEB());

// synopsys translate\_on

// FPGA Express black box declaration
// synopsys attribute fpga\_dont\_touch "true"
// synthesis attribute fpga\_dont\_touch of bram70x32 is "true"

// XST black box declaration
// box\_type "black\_box"
// synthesis attribute box\_type of bram70x32 is "black\_box"

endmodule

```
module debounce (reset, clk, noisy, clean);
input reset, clk, noisy;
output clean;
```

```
parameter NDELAY = 650000;
parameter NBITS = 20;
```

```
reg [NBITS-1:0] count;
reg xnew, clean;
```

```
always @(posedge clk)
if (reset) begin xnew <= noisy; clean <= noisy; count <= 0; end
else if (noisy != xnew) begin xnew <= noisy; count <= 0; end
else if (count == NDELAY) clean <= xnew;
else count <= count+1;</pre>
```

endmodule

```
//
// 6.111 FPGA Labkit -- Hex display driver
//
//
// File: display_16hex.v
// Date: 24-Sep-05
//
// Created: April 27, 2004
// Author: Nathan Ickes
//
// This module drives the labkit hex displays and shows the value of
// 8 bytes (16 hex digits) on the displays.
//
// 24-Sep-05 Ike: updated to use new reset-once state machine, remove clear
// 02-Nov-05 Ike: updated to make it completely synchronous
//
// Inputs:
//
        - active high
// reset
  clock_27mhz - the synchronous clock
//
          - 64 bits; each 4 bits gives a hex digit
// data
//
// Outputs:
//
           - display lines used in the 6.111 labkit (rev 003 & 004)
//
   disp_*
//
module display_16hex (reset, clock_27mhz, data_in,
        disp_blank, disp_clock, disp_rs, disp_ce_b,
        disp_reset_b, disp_data_out);
 input reset, clock_27mhz; // clock and reset (active high reset)
 input [63:0] data_in;
                           // 16 hex nibbles to display
 output disp_blank, disp_clock, disp_data_out, disp_rs, disp_ce_b,
     disp_reset_b;
 reg disp_data_out, disp_rs, disp_ce_b, disp_reset_b;
```

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/display_16 hex.txt/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/lises/li$ 

```
//
 // Display Clock
 // Generate a 500kHz clock for driving the displays.
 reg [5:0] count;
 reg [7:0] reset_count;
// reg
            old_clock;
 wire
         dreset;
         clock = (count < 27) ? 0 : 1;
 wire
 always @(posedge clock_27mhz)
  begin
    \operatorname{count} \ll \operatorname{reset} ? 0 : (\operatorname{count} = 53 ? 0 : \operatorname{count} + 1);
    reset_count <= reset ? 100 : ((reset_count==0) ? 0 : reset_count-1);
    old_clock <= clock;</pre>
//
  end
 assign dreset = (reset_count != 0);
 assign disp_clock = ~clock;
 wire clock_tick = ((count=27)?1:0);
// wire clock_tick = clock & ~old_clock;
 //
 // Display State Machine
 //
 rag [7.0] state.
                     // FSM state
```

| // FSM state                              |
|-------------------------------------------|
| // index to current dot being clocked out |
| // control register                       |
| // index of current character             |
| // dots for a single digit                |
| // hex nibble of current character        |
|                                           |
|                                           |

assign disp\_blank = 1'b0; // low <= not blanked

```
always @(posedge clock_27mhz)
if (clock_tick)
```

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/display_16 hex.txt$ 

```
begin
 if (dreset)
   begin
     state \leq 0;
     dot_index \leq 0;
     control <= 32'h7F7F7F7F;
   end
  else
   casex (state)
    8'h00:
     begin
       // Reset displays
       disp_data_out <= 1'b0;
       disp_rs <= 1'b0; // dot register
       disp_ce_b \leq 1'b1;
       disp_reset_b <= 1'b0;
       dot_index \leq 0;
       state <= state+1;
     end
    8'h01:
     begin
       // End reset
       disp_reset_b <= 1'b1;
       state <= state+1;
     end
    8'h02:
     begin
       // Initialize dot register (set all dots to zero)
       disp_ce_b <= 1'b0;
       disp_data_out <= 1'b0; // dot_index[0];
       if (dot_index == 639)
         state <= state+1;
       else
         dot_index <= dot_index+1;</pre>
     end
    8'h03:
     begin
       // Latch dot data
       disp_ce_b <= 1'b1;
                                 // re-purpose to init ctrl reg
       dot_index <= 31;
```

```
state <= state+1;
end</pre>
```

#### 8'h04:

### begin

```
// Setup the control register
disp_rs <= 1'b1; // Select the control register
disp_ce_b <= 1'b0;
disp_data_out <= control[31];
control <= {control[30:0], 1'b0}; // shift left
if (dot_index == 0)
state <= state+1;
else
dot_index <= dot_index-1;
end
```

### 8'h05:

```
begin
// Latch the control register data / dot data
disp_ce_b <= 1'b1;
dot_index <= 39; // init for single char
char_index <= 15; // start with MS char
data <= data_in;
state <= state+1;
end</pre>
```

# 8'h06:

```
begin
 // Load the user's dot data into the dot reg, char by char
 disp_rs \leq 1'b0;
                                // Select the dot register
 disp_ce_b <= 1'b0;
 disp_data_out <= dots[dot_index]; // dot data from msb
 if (dot_index == 0)
   if (char_index == 0)
    state \leq 5;
                             // all done, latch data
   else
    begin
      char_index <= char_index - 1; // goto next char
      data <= data in;
      dot_index <= 39;
    end
 else
   dot_index <= dot_index-1; // else loop thru all dots
```

end

```
endcase // casex(state) end
```

```
always @ (data or char_index)
 case (char index)
  4'h0:
               nibble \leq data[3:0];
               nibble \leq data[7:4];
  4'h1:
  4'h2:
               nibble \leq data[11:8];
  4'h3:
               nibble \leq data[15:12];
  4'h4:
               nibble <= data[19:16];
  4'h5:
               nibble \leq data[23:20];
  4'h6:
               nibble \leq data[27:24];
  4'h7:
               nibble \leq data[31:28];
               nibble <= data[35:32];
  4'h8:
  4'h9:
               nibble \leq data[39:36];
                nibble <= data[43:40];
  4'hA:
  4'hB:
                nibble \leq data[47:44];
  4'hC:
                nibble \leq data[51:48];
  4'hD:
                nibble \leq data[55:52];
  4'hE:
                nibble \leq data[59:56];
  4'hF:
                nibble \leq data[63:60];
```

endcase

```
always @(nibble)
case (nibble)
 4'h0: dots <= 40'b00111110_01010001_01001001_01000101_00111110;
 4'h2: dots <= 40'b01100010_01010001_01001001_01001001_01000110;
 4'h4: dots <= 40'b00011000_00010100_00010010_01111111_00010000;
 4'h5: dots <= 40'b00100111_01000101_01000101_01000101_00111001;
 4'h6: dots <= 40'b00111100_01001010_01001001_01001001_00110000;
 4'h9: dots <= 40'b00000110 01001001 01001001 00101001 00011110;
 4'hA: dots <= 40'b01111110_00001001_00001001_00001001_0111110;
 4'hC: dots <= 40'b00111110_01000001_01000001_01000001_00100010;
 4'hD: dots <= 40'b01111111_01000001_01000001_01000001_00111110;
 4'hE: dots <= 40'b01111111 01001001 01001001 01001001 01000001;
```

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/display_16 hex.txt/display_16 hex.tx$ 

endcase

endmodule

//

// File: ntsc2zbt.v
// Date: 27-Nov-05
// Author: I. Chuang <ichuang@mit.edu>
//
// Example for MIT 6.111 labkit showing how to prepare NTSC data
// (from Javier's decoder) to be loaded into the ZBT RAM for video
// display.
//
// The ZBT memory is 36 bits wide; we only use 32 bits of this, to
// store 4 bytes of black-and-white intensity data from the NTSC
// video input.

#### 

// Prepare data and address values to fill ZBT memory with NTSC data

module ntsc\_to\_zbt(clk, vclk, fvh, dv, din, ntsc\_addr, ntsc\_data, ntsc\_we, sw);

clk; // system clock input vclk; // video clock from camera input input [2:0] fvh; input dv; input [29:0] din; output [18:0] ntsc\_addr; output [35:0] ntsc\_data; output ntsc\_we; // write enable for NTSC data // switch which determines mode (for debugging) input sw;

parameter  $COL_START = 10'd30;$ parameter  $ROW_START = 10'd30;$ 

// here put the luminance data from the ntsc decoder into the ram
// this is for 1024 x 768 XGA display

reg [9:0] col = 0;row = 0; reg [9:0] reg [17:0] vdata = 0;reg vwe; old\_dv; reg old\_frame; // frames are even / odd interlaced reg even\_odd; // decode interlaced frame to this wire reg

```
wire
          frame = fvh[2];
wire
          frame_edge = frame & ~old_frame;
always @ (posedge vclk) //LLC1 is reference
 begin
   old_dv \ll dv;
   vwe <= dv && !fvh[2] & ~old_dv; // if data valid, write it
   old_frame <= frame;
   even_odd = frame_edge ? ~even_odd : even_odd;
   if (!fvh[2])
    begin
      col \leq tvh[0] ? COL_START :
           (!fvh[2] && !fvh[1] && dv && (col < 1024)) ? col + 1 : col;
      row \le fvh[1]? ROW_START :
          (!fvh[2] && fvh[0] && (row < 768)) ? row + 1 : row;
      vdata <= (dv && !fvh[2]) ? {din[29:22],din[19:12],din[9:8]} : vdata;
    end
 end
// synchronize with system clock
reg [9:0] x[1:0],y[1:0];
reg [17:0] data[1:0];
       we[1:0];
reg
       eo[1:0];
reg
always @(posedge clk)
 begin
   {x[1],x[0]} \le {x[0],col};
   \{y[1], y[0]\} \le \{y[0], row\};\
   \{data[1], data[0]\} \le \{data[0], vdata\};\
   \{we[1], we[0]\} \le \{we[0], vwe\};\
   \{eo[1], eo[0]\} \le \{eo[0], even_odd\};\
 end
// edge detection on write enable signal
reg old_we;
wire we_edge = we[1] & \simold_we;
```

always @(posedge clk) old\_we <= we[1];

// shift each set of four bytes into a large register for the ZBT

```
reg [35:0] mydata;
always @(posedge clk)
if (we_edge)
mydata <= { mydata[17:0], data[1] };</pre>
```

// compute address to store data in

wire [18:0] myaddr =  $\{y[1][8:0], eo[1], x[1][9:1]\};$ 

```
// alternate (256x192) image data and address
wire [35:0] mydata2 = {data[1],data[1],data[1],data[1]};
wire [18:0] myaddr2 = {1'b0, y[1][8:0], eo[1], x[1][7:0]};
```

// update the output address and data only when four bytes ready // WRITE EVERY 2!!!

```
reg [18:0] ntsc_addr;
reg [35:0] ntsc_data;
wire ntsc_we = sw ? we_edge : (we_edge & (x[1][0]==1'b0));
always @(posedge clk)
if ( ntsc_we )
begin
ntsc_addr <= sw ? myaddr2 : myaddr; // normal and expanded modes
ntsc_data <= sw ? {4'b0,mydata2} : {4'b0,mydata};
end
```

endmodule // ntsc\_to\_zbt

```
//
// File: video_decoder.v
// Date: 31-Oct-05
// Author: J. Castro (MIT 6.111, fall 2005)
//
// This file contains the ntsc_decode and adv7185init modules
//
// These modules are used to grab input NTSC video data from the RCA
// phono jack on the right hand side of the 6.111 labkit (connect
// the camera to the LOWER jack).
//
```

```
//
```

// NTSC decode - 16-bit CCIR656 decoder

// By Javier Castro

// This module takes a stream of LLC data from the adv7185

// NTSC/PAL video decoder and generates the corresponding pixels,

// that are encoded within the stream, in YCrCb format.

// Make sure that the adv7185 is set to run in 16-bit LLC2 mode.

module ntsc\_decode(clk, reset, tv\_in\_ycrcb, ycrcb, f, v, h, data\_valid);

// clk - line-locked clock (in this case, LLC1 which runs at 27Mhz)

// reset - system reset

// tv\_in\_ycrcb - 10-bit input from chip. should map to pins [19:10]

// ycrcb - 24 bit luminance and chrominance (8 bits each)

 $/\!/\,f$  - field: 1 indicates an even field, 0 an odd field

// v - vertical sync: 1 means vertical sync

// h - horizontal sync: 1 means horizontal sync

input clk; input reset; input [9:0] tv\_in\_ycrcb; // modified for 10 bit input - should be P[19:10] output [29:0] ycrcb; output f; output f; output v; output h; output h; output data\_valid; // output [4:0] state;  $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/video_decoder.txt$ 

| parameter | $SYNC_1 = 0;$      |
|-----------|--------------------|
| parameter | <b>SYNC_2</b> = 1; |
| parameter | <b>SYNC_3</b> = 2; |
| parameter | $SAV_f1_cb0 = 3;$  |
| parameter | $SAV_f1_y0 = 4;$   |
| parameter | SAV_f1_cr1 = 5;    |
| parameter | $SAV_f1_y1 = 6;$   |
| parameter | $EAV_f1 = 7;$      |
| parameter | $SAV_VBI_f1 = 8;$  |
| parameter | $EAV_VBI_f1 = 9;$  |
| parameter | $SAV_f2_cb0 = 10;$ |
| parameter | $SAV_f2_y0 = 11;$  |
| parameter | SAV_f2_cr1 = 12;   |
| parameter | SAV_f2_y1 = 13;    |
| parameter | $EAV_f2 = 14;$     |
| parameter | $SAV_VBI_f2 = 15;$ |
| parameter | $EAV_VBI_f2 = 16;$ |
|           |                    |

// In the start state, the module doesn't know where

// in the sequence of pixels, it is looking.

// Once we determine where to start, the FSM goes through a normal
// sequence of SAV process\_YCrCb EAV... repeat

// The data stream looks as follows

 $/\!/ SAV_FF \mid SAV_00 \mid SAV_00 \mid SAV_XY \mid Cb0 \mid Y0 \mid Cr1 \mid Y1 \mid Cb2 \mid Y2 \mid ... \mid EAV \ sequence$ 

// There are two things we need to do:

// 1. Find the two SAV blocks (stands for Start Active Video perhaps?)

// 2. Decode the subsequent data

```
reg [4:0] current_state = 5'h00;
```

- reg [9:0] y = 10'h000; // luminance
- reg [9:0] cr = 10'h000; // chrominance
- reg [9:0] cb = 10'h000; // more chrominance

assign state = current\_state;

```
always @ (posedge clk)
begin
if (reset)
```

begin

end

else

### begin

// these states don't do much except allow us to know where we are in the stream.

// whenever the synchronization code is seen, go back to the sync\_state before

// transitioning to the new state

case (current\_state)

SYNC\_1: current\_state <= (tv\_in\_ycrcb == 10'h000) ? SYNC\_2 : SYNC\_1; SYNC\_2: current\_state <= (tv\_in\_ycrcb == 10'h000) ? SYNC\_3 : SYNC\_1; SYNC\_3: current\_state <= (tv\_in\_ycrcb == 10'h200) ? SAV\_f1\_cb0 : (tv\_in\_ycrcb == 10'h274) ? EAV\_f1 : (tv\_in\_ycrcb == 10'h2ac) ? SAV\_VBI\_f1 : (tv\_in\_ycrcb == 10'h2d8) ? EAV\_VBI\_f1 : (tv\_in\_ycrcb == 10'h31c) ? SAV\_f2\_cb0 :

 $(tv_in_ycrcb == 10'h368)$ ? EAV\_f2 :

 $(tv_in_ycrcb == 10'h3b0)$ ? SAV\_VBI\_f2 :

(tv\_in\_ycrcb == 10'h3c4) ? EAV\_VBI\_f2 : SYNC\_1;

SAV\_f1\_cb0: current\_state <= (tv\_in\_ycrcb == 10'h3ff) ? SYNC\_1 : SAV\_f1\_y0; SAV\_f1\_y0: current\_state <= (tv\_in\_ycrcb == 10'h3ff) ? SYNC\_1 : SAV\_f1\_cr1; SAV\_f1\_cr1: current\_state <= (tv\_in\_ycrcb == 10'h3ff) ? SYNC\_1 : SAV\_f1\_y1; SAV\_f1\_y1: current\_state <= (tv\_in\_ycrcb == 10'h3ff) ? SYNC\_1 : SAV\_f1\_cb0;

SAV\_f2\_cb0: current\_state <= (tv\_in\_ycrcb == 10'h3ff) ? SYNC\_1 : SAV\_f2\_y0; SAV\_f2\_y0: current\_state <= (tv\_in\_ycrcb == 10'h3ff) ? SYNC\_1 : SAV\_f2\_cr1; SAV\_f2\_cr1: current\_state <= (tv\_in\_ycrcb == 10'h3ff) ? SYNC\_1 : SAV\_f2\_y1; SAV\_f2\_y1: current\_state <= (tv\_in\_ycrcb == 10'h3ff) ? SYNC\_1 : SAV\_f2\_cb0;

// These states are here in the event that we want to cover these signals
// in the future. For now, they just send the state machine back to SYNC\_1
EAV\_f1: current\_state <= SYNC\_1;
SAV\_VBI\_f1: current\_state <= SYNC\_1;
EAV\_VBI\_f1: current\_state <= SYNC\_1;
EAV\_f2: current\_state <= SYNC\_1;
SAV\_VBI\_f2: current\_state <= SYNC\_1;
EAV\_VBI\_f2: current\_state <= SYNC\_1;
EAV\_VBI\_f2: current\_state <= SYNC\_1;</pre>

endcase end end // always @ (posedge clk)  $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 as\%2$ 

// implement our decoding mechanism

```
wire y_enable;
wire cr_enable;
wire cb_enable;
```

```
// if y is coming in, enable the register
// likewise for cr and cb
assign y_enable = (current_state == SAV_f1_y0) ||
        (current_state == SAV_f1_y1) ||
        (current_state == SAV_f2_y0) ||
        (current_state == SAV_f2_y1);
assign cr_enable = (current_state == SAV_f1_cr1) ||
        (current_state == SAV_f2_cr1);
assign cb_enable = (current_state == SAV_f1_cb0) ||
        (current_state == SAV_f2_cb0);
```

```
// f, v, and h only go high when active
assign {v,h} = (current_state == SYNC_3) ? tv_in_ycrcb[7:6] : 2'b00;
```

```
// data is valid when we have all three values: y, cr, cb
assign data_valid = y_enable;
assign ycrcb = {y,cr,cb};
```

reg f = 0;

```
always @ (posedge clk)
begin
    y <= y_enable ? tv_in_ycrcb : y;
    cr <= cr_enable ? tv_in_ycrcb : cr;
    cb <= cb_enable ? tv_in_ycrcb : cb;
    f <= (current_state == SYNC_3) ? tv_in_ycrcb[8] : f;
end</pre>
```

```
endmodule
```

file:///Cl/Documents%20and%20Settings/cwilkens/My%20Documents/Colle...ject/Verilog%20Files/Verilog%20Source%20as%20Text/video\_decoder.txt

### // Author: Nathan Ickes

//

// Register 0

`define INPUT\_SELECT 4'h0 // 0: CVBS on AIN1 (composite video in) // 7: Y on AIN2, C on AIN5 (s-video in) // (These are the only configurations supported by the 6.111 labkit hardware) 4'h0 `define INPUT\_MODE // 0: Autodetect: NTSC or PAL (BGHID), w/o pedestal // 1: Autodetect: NTSC or PAL (BGHID), w/pedestal // 2: Autodetect: NTSC or PAL (N), w/o pedestal // 3: Autodetect: NTSC or PAL (N), w/pedestal // 4: NTSC w/o pedestal // 5: NTSC w/pedestal // 6: NTSC 4.43 w/o pedestal // 7: NTSC 4.43 w/pedestal // 8: PAL BGHID w/o pedestal // 9: PAL N w/pedestal // A: PAL M w/o pedestal // B: PAL M w/pedestal // C: PAL combination N // D: PAL combination N w/pedestal

// E-F: [Not valid]

# `define ADV7185\_REGISTER\_0 {`INPUT\_MODE, `INPUT\_SELECT}

| `define VIDEO_QUALITY 2'l             |  |      |
|---------------------------------------|--|------|
| // 0: Broadcast quality               |  |      |
| // 1: TV quality                      |  |      |
| // 2: VCR quality                     |  |      |
| <pre>// 3: Surveillance quality</pre> |  |      |
| `define SQUARE_PIXEL_IN_MODE          |  | 1'b0 |
| // 0: Normal mode                     |  |      |
| // 1: Square pixel mode               |  |      |
|                                       |  |      |

1'b0 `define DIFFERENTIAL\_INPUT // 0: Single-ended inputs // 1: Differential inputs `define FOUR\_TIMES\_SAMPLING 1'b0 // 0: Standard sampling rate // 1: 4x sampling rate (NTSC only) 1'b0 `define BETACAM // 0: Standard video input // 1: Betacam video input `define AUTOMATIC\_STARTUP\_ENABLE 1'b1 // 0: Change of input triggers reacquire // 1: Change of input does not trigger reacquire

### `define ADV7185\_REGISTER\_1 {`AUTOMATIC\_STARTUP\_ENABLE, 1'b0, `BETACAM, `FOUR\_TIMES\_SAMPLING, `DIFFERENTIAL\_INPUT, `SQUARE\_PIXEL\_IN\_MODE, `VIDEO\_QUALITY}

// Register 2

### `define Y\_PEAKING\_FILTER

3'h4

- // 0: Composite = 4.5dB, s-video = 9.25dB // 1: Composite = 4.5dB, s-video = 9.25dB // 2: Composite = 4.5dB, s-video = 5.75dB // 3: Composite = 1.25dB, s-video = 3.3dB // 4: Composite = 0.0dB, s-video = 0.0dB // 5: Composite = -1.25dB, s-video = -3.0dB // 6: Composite = -1.75dB, s-video = -3.0dB // 6: Composite = -1.75dB, s-video = -8.0dB // 7: Composite = -3.0dB, s-video = -8.0dB // 0: No coring // 1: Truncate if Y < black+8 // 2: Truncate if Y < black+16</pre>
- // 3: Truncate if Y < black+32

### `define ADV7185\_REGISTER\_2 {3'b000, `CORING, `Y\_PEAKING\_FILTER}

`define INTERFACE\_SELECT

2'h0

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 as\%2$ 

// 0: Philips-compatible // 1: Broktree API A-compatible // 2: Broktree API B-compatible // 3: [Not valid] `define OUTPUT\_FORMAT 4'h0 // 0: 10-bit @ LLC, 4:2:2 CCIR656 // 1: 20-bit @ LLC, 4:2:2 CCIR656 // 2: 16-bit @ LLC, 4:2:2 CCIR656 // 3: 8-bit @ LLC, 4:2:2 CCIR656 // 4: 12-bit @ LLC, 4:1:1 // 5-F: [Not valid] // (Note that the 6.111 labkit hardware provides only a 10-bit interface to // the ADV7185.) `define TRISTATE\_OUTPUT\_DRIVERS 1'b0 // 0: Drivers tristated when ~OE is high // 1: Drivers always tristated `define VBI ENABLE 1'b0 // 0: Decode lines during vertical blanking interval

// 1: Decode only active video regions

## `define ADV7185\_REGISTER\_3 {`VBI\_ENABLE, `TRISTATE\_OUTPUT\_DRIVERS, `OUTPUT\_FORMAT, `INTERFACE\_SELECT}

`define OUTPUT\_DATA\_RANGE 1'b0
 // 0: Output values restricted to CCIR-compliant range
 // 1: Use full output range
`define BT656\_TYPE 1'b0
 // 0: BT656-3-compatible
 // 1: BT656-4-compatible

### `define ADV7185\_REGISTER\_4 {`BT656\_TYPE, 3'b000, 3'b110, `OUTPUT\_DATA\_RANGE}

`define GENERAL\_PURPOSE\_OUTPUTS4'b0000`define GPO\_0\_1\_ENABLE1'b0

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 as\%2$ 

// 0: General purpose outputs 0 and 1 tristated
// 1: General purpose outputs 0 and 1 enabled
`define GPO\_2\_3\_ENABLE 1'b0
// 0: General purpose outputs 2 and 3 tristated
// 1: General purpose outputs 2 and 3 enabled
`define BLANK\_CHROMA\_IN\_VBI 1'b1
// 0: Chroma decoded and output during vertical blanking
// 1: Chroma blanked during vertical blanking
`define HLOCK\_ENABLE 1'b0
// 0: GPO 0 is a general purpose output
// 1: GPO 0 shows HLOCK status

# `define ADV7185\_REGISTER\_5 {`HLOCK\_ENABLE, `BLANK\_CHROMA\_IN\_VBI, `GPO\_2\_3\_ENABLE, `GPO\_0\_1\_ENABLE, `GENERAL\_PURPOSE\_OUTPUTS}

// Register 7

`define FIFO\_FLAG\_MARGIN5'h10// Sets the locations where FIFO almost-full and almost-empty flags are set`define FIFO\_RESET'/ 0: Normal operation// 1: Reset FIFO. This bit is automatically cleared`define AUTOMATIC\_FIFO\_RESET'/ 0: No automatic reset// 1: FIFO is autmatically reset at the end of each video field`define FIFO\_FLAG\_SELF\_TIME'/ 0: FIFO flags are synchronized to CLKIN// 1: FIFO flags are synchronized to internal 27MHz clock

# `define ADV7185\_REGISTER\_7 {`FIFO\_FLAG\_SELF\_TIME, `AUTOMATIC\_FIFO\_RESET, `FIFO\_RESET, `FIFO\_FLAG\_MARGIN}

// Register 8

`define INPUT\_CONTRAST\_ADJUST8'h80

# `define ADV7185\_REGISTER\_8 {`INPUT\_CONTRAST\_ADJUST}

 $file: ///C // Documents \% 20 and \% 20 Settings / cwilkens / My \% 20 Documents / Colle... ject / Verilog \% 20 Files / Verilog \% 20 Source \% 20 as \% 20 Text / video_decoder.txt = 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.0000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.000 / 0.$ 

#### // Register 9

# `define INPUT\_SATURATION\_ADJUST8'h8C

# `define ADV7185\_REGISTER\_9 {`INPUT\_SATURATION\_ADJUST}

# `define INPUT\_BRIGHTNESS\_ADJUST8'h00

# `define ADV7185\_REGISTER\_A {`INPUT\_BRIGHTNESS\_ADJUST}

`define INPUT\_HUE\_ADJUST 8'h00

# `define ADV7185\_REGISTER\_B {`INPUT\_HUE\_ADJUST}

`define DEFAULT\_VALUE\_ENABLE 1'b0
// 0: Use programmed Y, Cr, and Cb values
// 1: Use default values
`define DEFAULT\_VALUE\_AUTOMATIC\_ENABLE 1'b0
// 0: Use programmed Y, Cr, and Cb values
// 1: Use default values if lock is lost
`define DEFAULT\_Y\_VALUE 6'h0C
// Default Y value

# `define ADV7185\_REGISTER\_C {`DEFAULT\_Y\_VALUE, `DEFAULT\_VALUE\_AUTOMATIC\_ENABLE, `DEFAULT\_VALUE\_ENABLE}

  $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/video_decoder.txt$ 

`define DEFAULT\_CR\_VALUE4'h8// Most-significant four bits of default Cr value`define DEFAULT\_CB\_VALUE4'h8// Most-significant four bits of default Cb value

### `define ADV7185\_REGISTER\_D {`DEFAULT\_CB\_VALUE, `DEFAULT\_CR\_VALUE}

#### // Register E

| `define TEMPORAL_DECIMATION_ENABLE          | 1'b0 |
|---------------------------------------------|------|
| // 0: Disable                               |      |
| // 1: Enable                                |      |
| `define TEMPORAL_DECIMATION_CONTROL         | 2'h0 |
| // 0: Supress frames, start with even field |      |
| // 1: Supress frames, start with odd field  |      |
| // 2: Supress even fields only              |      |
| // 3: Supress odd fields only               |      |
| `define TEMPORAL_DECIMATION_RATE            | 4'h0 |
| // 0-F: Number of fields/frames to skip     |      |

# `define ADV7185\_REGISTER\_E {1'b0, `TEMPORAL\_DECIMATION\_RATE, `TEMPORAL\_DECIMATION\_CONTROL, `TEMPORAL\_DECIMATION\_ENABLE}

// Register F 2'h0 `define POWER\_SAVE\_CONTROL // 0: Full operation // 1: CVBS only // 2: Digital only // 3: Power save mode `define POWER\_DOWN\_SOURCE\_PRIORITY 1'b0 // 0: Power-down pin has priority // 1: Power-down control bit has priority `define POWER\_DOWN\_REFERENCE 1'b0 // 0: Reference is functional // 1: Reference is powered down `define POWER\_DOWN\_LLC\_GENERATOR 1'b0 // 0: LLC generator is functional // 1: LLC generator is powered down

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 as\%2$ 

1'b0 `define POWER\_DOWN\_CHIP // 0: Chip is functional // 1: Input pads disabled and clocks stopped `define TIMING\_REACQUIRE 1'b0 // 0: Normal operation // 1: Reacquire video signal (bit will automatically reset) `define RESET\_CHIP 1'b0 // 0: Normal operation // 1: Reset digital core and I2C interface (bit will automatically reset) `define ADV7185\_REGISTER\_F {`RESET\_CHIP, `TIMING\_REACQUIRE, `POWER\_DOWN\_CHIP, `POWER\_DOWN\_LLC\_GENERATOR, `POWER\_DOWN\_REFERENCE, `POWER\_DOWN\_SOURCE\_PRIORITY, `POWER\_SAVE\_CONTROL} // Register 33 

1'b1 `define PEAK WHITE UPDATE // 0: Update gain once per line // 1: Update gain once per field 1'b1 `define AVERAGE BIRIGHTNESS LINES // 0: Use lines 33 to 310 // 1: Use lines 33 to 270 3'h0 `define MAXIMUM\_IRE // 0: PAL: 133, NTSC: 122 // 1: PAL: 125, NTSC: 115 // 2: PAL: 120, NTSC: 110 // 3: PAL: 115, NTSC: 105 // 4: PAL: 110, NTSC: 100 // 5: PAL: 105, NTSC: 100 // 6-7: PAL: 100, NTSC: 100 1'b1 `define COLOR\_KILL // 0: Disable color kill // 1: Enable color kill

# `define ADV7185\_REGISTER\_33 {1'b1, `COLOR\_KILL, 1'b1, `MAXIMUM\_IRE, `AVERAGE\_BIRIGHTNESS\_LINES, `PEAK\_WHITE\_UPDATE}

`define ADV7185\_REGISTER\_10 8'h00
`define ADV7185\_REGISTER\_11 8'h00
`define ADV7185\_REGISTER\_12 8'h00
`define ADV7185\_REGISTER\_13 8'h45

`define ADV7185 REGISTER 148'h18 `define ADV7185 REGISTER 158'h60 `define ADV7185 REGISTER 168'h00 `define ADV7185 REGISTER 178'h01 `define ADV7185 REGISTER 188'h00 `define ADV7185\_REGISTER\_19 8'h10 `define ADV7185 REGISTER 1A 8'h10 `define ADV7185\_REGISTER\_1B 8'hF0 `define ADV7185 REGISTER 1C 8'h16 `define ADV7185\_REGISTER\_1D 8'h01 `define ADV7185\_REGISTER\_1E 8'h00 `define ADV7185 REGISTER 1F 8'h3D `define ADV7185\_REGISTER\_20 8'hD0 `define ADV7185 REGISTER 218'h09 `define ADV7185\_REGISTER\_22 8'h8C `define ADV7185\_REGISTER\_23 8'hE2 `define ADV7185 REGISTER 24 8'h1F `define ADV7185 REGISTER 258'h07 `define ADV7185 REGISTER 268'hC2 `define ADV7185\_REGISTER\_27 8'h58 `define ADV7185\_REGISTER\_28 8'h3C `define ADV7185 REGISTER 298'h00 `define ADV7185 REGISTER 2A 8'h00 `define ADV7185 REGISTER 2B 8'hA0 `define ADV7185\_REGISTER\_2C 8'hCE `define ADV7185\_REGISTER\_2D 8'hF0 `define ADV7185 REGISTER 2E 8'h00 `define ADV7185 REGISTER 2F 8'hF0 `define ADV7185 REGISTER 308'h00 `define ADV7185\_REGISTER\_31 8'h70 `define ADV7185\_REGISTER\_32 8'h00 `define ADV7185\_REGISTER\_34 8'h0F `define ADV7185\_REGISTER\_35 8'h01 `define ADV7185 REGISTER 368'h00 `define ADV7185\_REGISTER\_37 8'h00 `define ADV7185\_REGISTER\_38 8'h00 `define ADV7185 REGISTER 398'h00 `define ADV7185 REGISTER 3A 8'h00 `define ADV7185 REGISTER 3B 8'h00

`define ADV7185\_REGISTER\_44 8'h41 `define ADV7185\_REGISTER\_45 8'hBB  $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 Text/video_decoder.txt/Verilog\%20 as\%20 as\%2$ 

`define ADV7185\_REGISTER\_F1 8'hEF `define ADV7185\_REGISTER\_F2 8'h80

input reset;

input clock\_27mhz;

output tv\_in\_reset\_b; // Reset signal to ADV7185 output tv\_in\_i2c\_clock; // I2C clock output to ADV7185 output tv\_in\_i2c\_data; // I2C data line to ADV7185 input source; // 0: composite, 1: s-video

initial begin

\$display("ADV7185 Initialization values:");

| ¢dianlay("  | Register 0: 0x%X", `ADV7185_REGISTER_0);   |
|-------------|--------------------------------------------|
| 1 * `       |                                            |
| \$display(" | Register 1: 0x%X", `ADV7185_REGISTER_1);   |
| \$display(" | Register 2: 0x%X", `ADV7185_REGISTER_2);   |
| \$display(" | Register 3: 0x%X", `ADV7185_REGISTER_3);   |
| \$display(" | Register 4: 0x%X", `ADV7185_REGISTER_4);   |
| \$display(" | Register 5: 0x%X", `ADV7185_REGISTER_5);   |
| \$display(" | Register 7: 0x%X", `ADV7185_REGISTER_7);   |
| \$display(" | Register 8: 0x%X", `ADV7185_REGISTER_8);   |
| \$display(" | Register 9: 0x%X", `ADV7185_REGISTER_9);   |
| \$display(" | Register A: 0x%X", `ADV7185_REGISTER_A);   |
| \$display(" | Register B: 0x%X", `ADV7185_REGISTER_B);   |
| \$display(" | Register C: 0x%X", `ADV7185_REGISTER_C);   |
| \$display(" | Register D: 0x%X", `ADV7185_REGISTER_D);   |
| \$display(" | Register E: 0x%X", `ADV7185_REGISTER_E);   |
| \$display(" | Register F: 0x%X", `ADV7185_REGISTER_F);   |
| \$display(" | Register 33: 0x%X", `ADV7185_REGISTER_33); |
| end         |                                            |
|             |                                            |

//

// Generate a 1MHz for the I2C driver (resulting I2C clock rate is 250kHz)
//

reg [7:0] clk\_div\_count, reset\_count; reg clock\_slow; wire reset\_slow;

initial

```
begin
   clk_div_count <= 8'h00;
   // synthesis attribute init of clk_div_count is "00";
   clock_slow <= 1'b0;
   // synthesis attribute init of clock_slow is "0";
 end
always @(posedge clock_27mhz)
 if (clk_div_count == 26)
  begin
    clock_slow <= ~clock_slow;
    clk_div_count <= 0;
  end
 else
  clk_div_count <= clk_div_count+1;
always @(posedge clock_27mhz)
 if (reset)
  reset_count <= 100;
 else
  reset_count <= (reset_count==0) ? 0 : reset_count-1;</pre>
assign reset_slow = reset_count != 0;
//
// I2C driver
//
reg load;
reg [7:0] data;
wire ack, idle;
i2c i2c(.reset(reset_slow), .clock4x(clock_slow), .data(data), .load(load),
     .ack(ack), .idle(idle), .scl(tv_in_i2c_clock),
     .sda(tv_in_i2c_data));
//
// State machine
//
reg [7:0] state;
reg tv_in_reset_b;
reg old_source;
```

```
always @(posedge clock_slow)
 if (reset_slow)
   begin
     state \leq 0;
     load \leq = 0;
     tv_in_reset_b <= 0;</pre>
     old_source <= 0;
   end
  else
   case (state)
    8'h00:
      begin
        // Assert reset
        load \leq 1'b0;
        tv_in_reset_b <= 1'b0;
        if (!ack)
         state <= state+1;</pre>
      end
    8'h01:
      state <= state+1;</pre>
    8'h02:
      begin
        // Release reset
        tv_in_reset_b <= 1'b1;</pre>
        state <= state+1;</pre>
           end
    8'h03:
      begin
        // Send ADV7185 address
        data <= 8'h8A;
        load <= 1'b1;
        if (ack)
         state <= state+1;
      end
    8'h04:
      begin
        // Send subaddress of first register
        data <= 8'h00;
        if (ack)
         state <= state+1;
      end
    8'h05:
```

```
begin
  // Write to register 0
  data <= `ADV7185_REGISTER_0 | {5'h00, {3{source}}};
  if (ack)
    state <= state+1;
 end
8'h06:
begin
  // Write to register 1
  data <= `ADV7185_REGISTER_1;</pre>
  if (ack)
    state <= state+1;
 end
8'h07:
 begin
  // Write to register 2
  data <= `ADV7185_REGISTER_2;</pre>
  if (ack)
    state <= state+1;
 end
8'h08:
 begin
  // Write to register 3
  data <= `ADV7185_REGISTER_3;
  if (ack)
    state <= state+1;
 end
8'h09:
 begin
  // Write to register 4
  data <= `ADV7185_REGISTER_4;
  if (ack)
    state <= state+1;
 end
8'h0A:
 begin
  // Write to register 5
  data <= `ADV7185_REGISTER_5;
  if (ack)
    state <= state+1;
 end
8'h0B:
 begin
```

```
// Write to register 6
  data <= 8'h00; // Reserved register, write all zeros
  if (ack)
    state <= state+1;
 end
8'h0C:
 begin
  // Write to register 7
   data <= `ADV7185_REGISTER_7;
  if (ack)
    state <= state+1;
 end
8'h0D:
 begin
  // Write to register 8
   data <= `ADV7185_REGISTER_8;
  if (ack)
    state <= state+1;
 end
8'h0E:
 begin
  // Write to register 9
  data <= `ADV7185_REGISTER_9;</pre>
  if (ack)
    state <= state+1;
 end
8'h0F: begin
 // Write to register A
 data <= `ADV7185_REGISTER_A;
if (ack)
 state <= state+1;
end
8'h10:
 begin
  // Write to register B
   data <= `ADV7185_REGISTER_B;
  if (ack)
    state <= state+1;
 end
8'h11:
 begin
  // Write to register C
  data <= `ADV7185_REGISTER_C;</pre>
```

```
if (ack)
    state <= state+1;
 end
8'h12:
begin
  // Write to register D
  data <= `ADV7185_REGISTER_D;
  if (ack)
    state <= state+1;
 end
8'h13:
begin
  // Write to register E
  data <= `ADV7185_REGISTER_E;
  if (ack)
    state <= state+1;
 end
8'h14:
 begin
  // Write to register F
  data <= `ADV7185_REGISTER_F;
  if (ack)
    state <= state+1;
 end
8'h15:
 begin
  // Wait for I2C transmitter to finish
  load \leq 1'b0;
  if (idle)
    state <= state+1;
 end
8'h16:
 begin
  // Write address
  data <= 8'h8A;
  load <= 1'b1;
  if (ack)
    state <= state+1;
 end
8'h17:
 begin
  data <= 8'h33;
  if (ack)
```

```
state <= state+1;
 end
8'h18:
 begin
   data <= `ADV7185_REGISTER_33;
   if (ack)
    state <= state+1;
 end
8'h19:
begin
   load <= 1'b0;
   if (idle)
    state <= state+1;
 end
8'h1A: begin
 data \leq 8'h8A;
 load <= 1'b1;
 if (ack)
   state <= state+1;</pre>
end
8'h1B:
 begin
   data <= 8'h33;
   if (ack)
    state <= state+1;
 end
8'h1C:
 begin
   load \leq 1'b0;
   if (idle)
    state <= state+1;</pre>
 end
8'h1D:
 begin
   load <= 1'b1;
   data <= 8'h8B;
   if (ack)
    state <= state+1;
 end
8'h1E:
begin
   data <= 8'hFF;
```

```
if (ack)
      state <= state+1;
   end
  8'h1F:
   begin
     load \leq 1'b0;
     if (idle)
      state <= state+1;
   end
  8'h20:
   begin
     // Idle
     if (old_source != source) state <= state+1;
     old_source <= source;
   end
  8'h21: begin
   // Send ADV7185 address
   data \leq 8'h8A:
   load \leq 1'b1;
   if (ack) state <= state+1;
 end
 8'h22: begin
   // Send subaddress of register 0
   data <= 8'h00;
   if (ack) state <= state+1;
 end
 8'h23: begin
   // Write to register 0
   data \leq ADV7185_REGISTER_0 | \{5'h00, \{3\{source\}\}\};
   if (ack) state <= state+1;
 end
 8'h24: begin
   // Wait for I2C transmitter to finish
   load <= 1'b0;
   if (idle) state \leq 8'h20;
 end
endcase
```

endmodule

 $/\!/$  i2c module for use with the ADV7185

module i2c (reset, clock4x, data, load, idle, ack, scl, sda);

```
input reset;
input clock4x;
input [7:0] data;
input load;
output ack;
output idle;
output scl;
output sda;
reg [7:0] ldata;
reg ack, idle;
reg scl;
reg sdai;
reg [7:0] state;
assign sda = sdai ? 1'bZ : 1'b0;
always @(posedge clock4x)
 if (reset)
  begin
    state <= 0;
    ack <= 0;
  end
 else
  case (state)
    8'h00: // idle
     begin
       scl <= 1'b1;
       sdai <= 1'b1;
       ack <= 1'b0;
       idle <= 1'b1;
       if (load)
        begin
          ldata <= data;
          ack <= 1'b1;
          state <= state+1;</pre>
        end
     end
    8'h01: // Start
     begin
       ack <= 1'b0;
```

```
idle \leq 1'b0;
   sdai <= 1'b0;
   state <= state+1;
 end
8'h02:
 begin
   scl <= 1'b0;
   state <= state+1;
 end
8'h03: // Send bit 7
 begin
   ack <= 1'b0;
   sdai <= ldata[7];</pre>
   state <= state+1;</pre>
 end
8'h04:
 begin
   scl <= 1'b1;
   state <= state+1;
 end
8'h05:
 begin
   state <= state+1;
 end
8'h06:
 begin
   scl <= 1'b0;
   state <= state+1;
 end
8'h07:
 begin
   sdai <= ldata[6];</pre>
   state <= state+1;
 end
8'h08:
 begin
   scl <= 1'b1;
   state <= state+1;
 end
8'h09:
 begin
   state <= state+1;
 end
```

```
8'h0A:
 begin
   scl <= 1'b0;
   state <= state+1;
 end
8'h0B:
 begin
   sdai <= ldata[5];</pre>
   state <= state+1;</pre>
 end
8'h0C:
 begin
   scl <= 1'b1;
   state <= state+1;</pre>
 end
8'h0D:
 begin
   state <= state+1;
 end
8'h0E:
 begin
   scl <= 1'b0;
   state <= state+1;
 end
8'h0F:
 begin
   sdai \ll 1data[4];
   state <= state+1;
 end
8'h10:
 begin
   scl <= 1'b1;
   state <= state+1;</pre>
 end
8'h11:
 begin
   state <= state+1;
 end
8'h12:
 begin
   scl <= 1'b0;
   state <= state+1;
 end
```

```
8'h13:
 begin
   sdai <= ldata[3];</pre>
   state <= state+1;
 end
8'h14:
 begin
   scl <= 1'b1;
   state <= state+1;</pre>
 end
8'h15:
 begin
   state <= state+1;
 end
8'h16:
 begin
   scl <= 1'b0;
   state <= state+1;</pre>
 end
8'h17:
 begin
   sdai \ll 1data[2];
   state <= state+1;
 end
8'h18:
 begin
   scl <= 1'b1;
   state <= state+1;
 end
8'h19:
 begin
   state <= state+1;</pre>
 end
8'h1A:
 begin
   scl <= 1'b0;
   state <= state+1;
 end
8'h1B:
 begin
   sdai <= ldata[1];</pre>
   state <= state+1;
 end
```

```
8'h1C:
 begin
   scl <= 1'b1;
   state <= state+1;
 end
8'h1D:
 begin
   state <= state+1;
 end
8'h1E:
 begin
   scl <= 1'b0;
   state <= state+1;
 end
8'h1F:
 begin
   sdai \ll 1data[0];
   state <= state+1;
 end
8'h20:
 begin
   scl <= 1'b1;
   state <= state+1;
 end
8'h21:
 begin
   state <= state+1;
 end
8'h22:
 begin
   scl <= 1'b0;
   state <= state+1;</pre>
 end
8'h23: // Acknowledge bit
 begin
   state <= state+1;
 end
8'h24:
 begin
   scl <= 1'b1;
   state <= state+1;
 end
8'h25:
```

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...ject/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/video_decoder.txt$ 

```
begin
    state <= state+1;
  end
 8'h26:
  begin
    scl <= 1'b0;
    if (load)
     begin
       ldata <= data;
       ack <= 1'b1;
       state <= 3;
     end
    else
     state <= state+1;
  end
 8'h27:
  begin
    sdai <= 1'b0;
    state <= state+1;</pre>
  end
 8'h28:
  begin
    scl <= 1'b1;
    state <= state+1;</pre>
  end
 8'h29:
  begin
    sdai <= 1'b1;
    state <= 0;
  end
endcase
```

```
endmodule
```

```
// xvga: Generate XVGA display signals (1024 x 768 @ 60Hz)
module xvga(vclock,hcount,vcount,hsync,vsync,blank);
 input vclock;
 output [10:0] hcount;
 output [9:0] vcount;
 output
           vsync;
 output
           hsync;
            blank;
 output
       hsync,vsync,hblank,vblank,blank;
 reg
 reg [10:0] hcount; // pixel number on current line
 reg [9:0] vcount;
                   // line number
 // horizontal: 1344 pixels total
 // display 1024 pixels per line
         hsyncon, hsyncoff, hreset, hblankon;
  wire
  assign hblankon = (hcount == 1023);
 assign
         hsyncon = (hcount == 1047);
  assign
         hsyncoff = (hcount == 1183);
  assign
         hreset = (hcount == 1343);
 // vertical: 806 lines total
 // display 768 lines
  wire
         vsyncon,vsyncoff,vreset,vblankon;
  assign vblankon = hreset & (vcount == 767);
        vsyncon = hreset \& (vcount == 776);
  assign
         vsyncoff = hreset \& (vcount == 782);
  assign
  assign
         vreset = hreset & (vcount == 805);
 // sync and blanking
         next_hblank,next_vblank;
  wire
 assign next_hblank = hreset ? 0 : hblankon ? 1 : hblank;
  assign next_vblank = vreset ? 0 : vblankon ? 1 : vblank;
 always @(posedge vclock) begin
   hcount \leq hreset ? 0 : hcount + 1;
   hblank <= next hblank;
   hsync <= hsyncon ? 0 : hsyncoff ? 1 : hsync; // active low
   vcount <= hreset ? (vreset ? 0 : vcount + 1) : vcount;
   vblank <= next_vblank;
```

 $file:///C|/Documents\%20 and\%20 Settings/cwilkens/My\%20 Documents/Colle...0 Project/Verilog\%20 Files/Verilog\%20 Source\%20 as\%20 Text/video_zbt.txt$ 

vsync <= vsyncon ? 0 : vsyncoff ? 1 : vsync; // active low

```
blank <= next_vblank | (next_hblank & ~hreset);</pre>
```

```
end
```

endmodule

```
// generate display pixels from reading the ZBT ram
// note that the ZBT ram has 2 cycles of read (and write) latency
//
// We take care of that by latching the data at an appropriate time.
//
```

// Note that the ZBT stores 36 bits per word; we use only 32 bits here, // decoded into four bytes of pixel data.

```
input reset, clk;
input [10:0] hcount;
input [9:0] vcount;
output [17:0] vr_pixel;
output [18:0] vram_addr;
input [35:0] vram_read_data;
```

```
parameter HMID = 9'd367; // The horizontal center of the image in MEMORY
parameter HSTART = HMID-9'd256; // The horizontal counter decrements!!!
parameter VMID = 9'd287; // The vertical center of the image in MEMORY
parameter VSTART = VMID-9'd192;
```

```
wire [18:0] vram_addr = {1'b0,vcount[9:1]+VSTART, ~hcount[10:2]-9'd180};
```

```
wire [1:0] hc4 = hcount[1:0];
reg [17:0] vr_pixel;
reg [35:0] vr_data_latched;
reg [35:0] last_vr_data;
```

```
always @(posedge clk)
last_vr_data <= (hc4==2'd3) ? vr_data_latched : last_vr_data;</pre>
```

```
always @(posedge clk)
vr_data_latched <= (hc4==2'd1) ? vram_read_data : vr_data_latched;</pre>
```

 $file: ///C // Documents \% 20 and \% 20 Settings / cwilkens / My\% 20 Documents / Colle...0 Project / Verilog \% 20 Files / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Text / video_z bt.txt / Verilog \% 20 Source \% 20 as \% 20 Source \% 20 Source \% 20 as \% 20 Source \% 20 as \% 20 Source \% 20 Source$ 

```
always @(*)  // each 36-bit word from RAM is decoded to 4 bytes
case (hc4)
  2'd3: vr_pixel = last_vr_data[17:0];//last_vr_data[8:0];
  2'd2: vr_pixel = last_vr_data[17:0];//last_vr_data[8+9:0+9];
  2'd1: vr_pixel = last_vr_data[35:18];//last_vr_data[8+18:0+18];
  2'd0: vr_pixel = last_vr_data[35:18];//last_vr_data[8+27:0+27];
  endcase
```

endmodule // vram\_display

module delayN(clk,in,out);
input clk;
input in;
output out;

```
parameter NDELAY = 3;
```

reg [NDELAY-1:0] shiftreg; wire out = shiftreg[NDELAY-1];

```
always @(posedge clk)
shiftreg <= {shiftreg[NDELAY-2:0],in};</pre>
```

endmodule // delayN

//
// File: zbt\_6111.v
// Date: 27-Nov-05
// Author: I. Chuang <ichuang@mit.edu>
//
//
// Simple ZDT\_himme for the MIT < 111.http://</pre>

// Simple ZBT driver for the MIT 6.111 labkit, which does not hide the
// pipeline delays of the ZBT from the user. The ZBT memories have
// two cycle latencies on read and write, and also need extra-long data hold
// times around the clock positive edge to work reliably.
//

// Ike's simple ZBT RAM driver for the MIT 6.111 labkit
//
// Data for writes can be presented and clocked in immediately; the actual
// writing to RAM will happen two cycles later.

//

// Read requests are processed immediately, but the read data is not available
// until two cycles after the initial request.

//

// A clock enable signal is provided; it enables the RAM clock when high.

```
module zbt_6111(clk, cen, we, addr, write_data, read_data,
```

ram\_clk, ram\_we\_b, ram\_address, ram\_data, ram\_cen\_b);

| input clk;                                                 | // system clock                              |  |
|------------------------------------------------------------|----------------------------------------------|--|
| input cen;                                                 | // clock enable for gating ZBT cycles        |  |
| input we;                                                  | // write enable (active HIGH)                |  |
| input [18:0] addr; // memory address                       |                                              |  |
| input [35:0] write_data; // data to write                  |                                              |  |
| output [35:0] read_data; // data read from memory          |                                              |  |
| output ram_                                                | _clk; // physical line to ram clock          |  |
| output ram_                                                | _we_b; // physical line to ram we_b          |  |
| output [18:0] ram_address; // physical line to ram address |                                              |  |
| inout [35:0] ram_data; // physical line to ram data        |                                              |  |
| output ram_                                                | _cen_b; // physical line to ram clock enable |  |

```
// clock enable (should be synchronous and one cycle high at a time)
wire ram_cen_b = ~cen;
```

// create delayed ram\_we signal: note the delay is by two cycles!
// ie we present the data to be written two cycles after we is raised

 $file: ///C // Documents \% 20 and \% 20 Settings / cwilkens / My \% 20 Documents / Colle... 20 Project / Verilog \% 20 Files / Verilog \% 20 Source \% 20 as \% 20 Text / zbt_6111.txt$ 

// this means the bus is tri-stated two cycles after we is raised.

```
reg [1:0] we_delay;
```

always @(posedge clk)
we\_delay <= cen ? {we\_delay[0],we} : we\_delay;</pre>

// create two-stage pipeline for write data

```
reg [35:0] write_data_old1;
reg [35:0] write_data_old2;
always @(posedge clk)
if (cen)
{write_data_old2, write_data_old1} <= {write_data_old1, write_data};</pre>
```

// wire to ZBT RAM signals

| assign                | $ram_we_b = \sim we;$                                  |
|-----------------------|--------------------------------------------------------|
| assign                | ram_clk = ~clk; // RAM is not happy with our data hold |
|                       | <pre>// times if its clk edges equal FPGA's</pre>      |
|                       | // so we clock it on the falling edges                 |
|                       | // and thus let data stabilize longer                  |
| assign                | ram_address = addr;                                    |
| assign                | ram_data = we_delay[1] ? write_data_old2 : {36{1'bZ}}; |
| assign                | $read_data = ram_data;$                                |
| endmodule // zbt_6111 |                                                        |