

# FPGA Temperature Regulation using PID Control

## University of Connecticut

## ECE 4401, Final Project

David Paquette

December 18, 2015

## Contents

|                                             |           |
|---------------------------------------------|-----------|
| <b>1 Objective</b>                          | <b>2</b>  |
| <b>2 Introduction</b>                       | <b>2</b>  |
| 2.1 High Level Overview . . . . .           | 2         |
| 2.2 VHDL Project Overview . . . . .         | 2         |
| <b>3 Temperature Sensor Module</b>          | <b>4</b>  |
| <b>4 Desired Temperature Control Module</b> | <b>4</b>  |
| <b>5 DC Motor Control Module</b>            | <b>4</b>  |
| <b>6 UART Serial Communication Module</b>   | <b>6</b>  |
| 6.1 UART Library Introduction . . . . .     | 6         |
| 6.2 Using the UART Component . . . . .      | 6         |
| 6.3 Data Encoding . . . . .                 | 7         |
| <b>7 VGA BRAM Display Module</b>            | <b>7</b>  |
| 7.1 VGA Display Overview . . . . .          | 7         |
| 7.2 BRAM Module . . . . .                   | 8         |
| <b>8 Digital PID Control Module</b>         | <b>9</b>  |
| 8.1 Brief PID Control Theory . . . . .      | 9         |
| 8.2 System Identification . . . . .         | 9         |
| 8.3 Digital PID Controller Design . . . . . | 10        |
| <b>9 Implementation Images</b>              | <b>12</b> |
| <b>10 References</b>                        | <b>13</b> |
| <b>11 Appendix</b>                          | <b>13</b> |

# 1 Objective

The objective of this project is to design and build a system to regulate the on board temperature of an FPGA, using PID control in VHDL. The desired temperature must be user selectable during system operation. The desired temperature, current temperature and current fan speed must be displayed on an external display. The current temperature and current fan speed must be collected by an external computer.

## 2 Introduction

### 2.1 High Level Overview

The FPGA development board being used is the Nexys 4 DDR Artix-7. This platform contains a built in temperature sensor and push-button pad, used for reading the onboard FPGA temperature and setting the desired temperature, respectively. To cool the FPGA, a DC computer fan is used, specifically the Asus K52F [4]. To modulate the speed of the fan PWM control is used. Any standard VGA display can be used, the FPGA development board has a standard VGA port built in. The FPGA board can also communicate over UART serial using the built in micro-usb connector, this is used to send fan and temperature data to an external computer. An overview of the components used can be seen in figure 1.



**FIG 1.** High level overview of the system components.

### 2.2 VHDL Project Overview

Figure 2 shows an overview of the various VHDL components used in this project. The main modules that were developed for this project were *SetPointControl*, *PIDController*, *TemperatureSensor*, *PWMInterface*, *Decimal2ASCII* and *Values2Serial*. The remaining components were originally written for use in other labs but were reused here. The only entity not developed in this course was the *UART* component. This library was found online in order to speed up the the development of serial data capture, the original source can be seen in [1]. The three highest level components were instantiated in the top level component *TemperatureControlProject*.



**FIG 2.** High level VHDL component hierarchy.



**FIG 3.** Main component data path (Wishbone signals are shown as dotted lines).

Figure 3 shows the data path of the main components. The *Master* component contains instantiations of *SetPointControl*, *PIDController*, *PWMDriver*, *TemperatureSensor* and *UARTSerial*. I decided against making these specific components communicate over the wishbone bus due to the uncertain data send/retrieval time that is introduced when communicating over a central bus. Because digital PID control must be normalized over a constant sampling rate, introducing unknown wait cycles did not seem like a good design choice. It may seem odd that I have the VGA controller communicating over the Wishbone bus but not the UART serial communication. This is because during development I was using the serial connection to collect data for the use of system identification and, once again, I did not want to introduce any unnecessary uncertainties into the data. Because the VGA display is just for the user, and is not a system critical component, any delays introduced by the bus wouldn't be an issue.

### 3 Temperature Sensor Module

The temperature sensor used was included on the FPGA development board. The sensor is physically located on the FPGA. Xilinx provided documentation on how to read and interpret the data from the sensor [2]. Xilinx also provided a VHDL module, *FPGAMonitor*, for returning the ADC(analog-digital converter) code from the on board temperature sensor. The ADC code then needs to be transformed to a meaningful temperature unit, in this project celsius is used. The equation for transforming the ADC code to a temperature is

$$Temp(^{\circ}C) = \frac{(ADCCode)503.975}{4096} - 273.15$$

Derived from

$$Voltage = 10 \frac{kT}{q} \ln(10)$$

Where  $k$  is Boltzmann's constant,  $T$  is temperature in kelvin and  $q$  is the charge on an electron. The analog voltage is sampled by the 12-bit ADCX to produce an ADC code. This equation is implemented in the *TemperatureSensorModule* component. The ADC code is returned in a 12-bit signal, 8-bits for integer values and 4 for decimal. For the sake of simplicity, I will only be using the 8-bits for integer values, which means the accuracy is in  $1^{\circ}\text{C}$  increments.



**FIG 4.** Block diagram of decoding sensor data.

### 4 Desired Temperature Control Module

To change the desired temperature during operation two push buttons are used. These buttons are built into the FPGA development board. When pressed, the up button (BTNU) increments the desired temperature and the down button (BTND) decrements the desired temperature. To prevent mechanical glitches and make sure only one change is made per button press, a debouncer and one press filter were used. The state machine for the one press filter can be seen in figure 7. A button press event occurs only in the *press* state. The *OnePressFilter* VHDL module contains the button debouncer module. Two instantiations of the *OnePressFilter* module are made in the *Master* module. This module outputs a single 8-bit signal, containing the desired temperature.



**FIG 5.** State machine diagram of the one press filter.

### 5 DC Motor Control Module

To alter the temperature of the FPGA, an off-the-shelf DC computer fan is used [4]. To control the speed of the DC motor the FPGA outputs a PWM signal from the JA PMOD GPIO header. This module takes one

8-bit input signal containing the fan speed, which is limited to an integer with the range of 0 to 100. This value represents the fan speed from 0% to 100%. This value is then converted to a duty cycle percent using the logic shown in figure 6.



**FIG 6.** Duty cycle(%) to PWM signal.

The FPGA PMOD output is a 0V LOW and 3.3V HIGH signal. However, the DC fan being used requires a potential of 5V. The 3.3V to 5V DC PWM motor driver circuit, shown in figure 7, was used to drive the fan. The circuit uses an n-MOSFET with a low threshold voltage. Specifically, the IRLB721 [3] was used due to its low max threshold voltage ( $V_{TH_{max}} = 2.35V$ ). The FPGA was more than adequately able to switch the FET.



**FIG 7.** 3.3V to 5V DC PWM Motor Driver Schematic.

## 6 UART Serial Communication Module

### 6.1 UART Library Introduction

For data acquisition I used a UART serial communication between the FPGA and an external computer. However, because this was not the main focus of this project I decided to find a library to simplify UART data transmission. I found a well documented library with plenty of useful examples here [1]. However, to interface with this library another module needed to be created. The *Values2Serial* module handles interfacing with the UART library. The UART library was contained in a single component and the I/O diagram can be seen in figure 8.



**FIG 8.** UART library component port map diagram (only relevant ports are shown).

Using this library did cause some issues during development. The hardware that the library was developed on used an active high reset scheme, which is opposite from the Nexys 4 DDR, so this caused some minor issues. Another variation was the TX/RX convention used by Xilinx differed from this library. Xilinx uses RX and TX from the perspective of the attached device, in our case this would be the computer. So, the TX port is actually the port where the FPGA receives data and the RX data is the port that the FPGA sends data. This was confusing to work through, but it was well documented in [2].



**FIG 9.** State machine for interfacing with the UART library.

### 6.2 Using the UART Component

The state machine shown in figure 9 describes how the *Values2Serial* component interfaces with the UART library. The *UART* component accepts 8-bit signals, so in order to send a message containing all the required data, a data transmission format was specified. To begin data transfer, the *stb\_i* pin must be asserted and the data must be placed on the 8-bit *data\_in* port. We must then wait for the *ack\_o* to be asserted. When the *ack\_o* is asserted, we deassert the *stb\_i* pin and then wait for 10 ms, then repeat the process for the next 8-bit piece of data.

### 6.3 Data Encoding

To configure the connection on the external computer any standard serial communication software can be used. However, the settings must be configured to the following settings: baud rate=115200, data bits=8, parity=none, stop bits=1. The data being received will be in the following format: XX2CXX0A. Where the XX represents an 8-bit hexadecimal number, and 2C is the hexadecimal representation of the ASCII comma (,) character and 0A is the hexadecimal representation of the ASCII newline character (\n). The first XX contains the current temperature and the second XX contains the current fan speed. Below is an example of the raw serial output.

```
24 2C 17 0A 24 2C 17 0A 23 2C 17 0A 23 2C 17 0A 22 2C 17 0A 22 2C 17 0A
```

To decode this we start reading it left to right. We first see the hexadecimal 62, if we reference our data format from above, we see that this first number is our current temperature in hexadecimal. For it to be human readable we must convert it to decimal, 0x24 → 36. So, the current temperature is 36°C. Moving to the next number we see 2C, again we reference our data format from above and see that this is just ASCII for a comma (.). The next number is hexadecimal 17. Check the format guide and decode to decimal, we can see that this shows the fan speed is 23%. The next number is 0A, which is ASCII for a newline character. This concludes one data reading from the FPGA. Repeat the above procedure to decode the entire message. However, that doesn't seem like very much fun. So, I wrote the below Python script to decode the data for us.

---

```
f = open('dataFromSerial.txt')
file = open('decodedData.txt', 'w')
data = f.readlines().split("0A")
for line in iter(data):
    d = line.split("2C")
    try:
        x = str(int(d[0],16))+ ", " + str(int(d[1],16))
        print(x)
        file.write(x+'\n')
    except ValueError:
        print("there was probably a problem or something")
f.close()
file.close()
```

---

If we pass the above data to this Python function, then we get the following in a much easier to read output format (Current temperature, fan speed).

```
36, 23
36, 23
35, 23
35, 23
34, 23
34, 23
```

## 7 VGA BRAM Display Module

### 7.1 VGA Display Overview

In this project information is first written to the BRAM over the Wishbone bus and is then read from from BRAM over Wishbone by the *VGA* module. The *VGA* module was used in previous labs and the implementation won't be discussed here [5]. The module for writing to BRAM was modified from a previous lab. The basics can be seen in figure 10.



**FIG 10.** State machine for interfacing with the BRAM module over Wishbone.

## 7.2 BRAM Module

We start by initializing the BRAM memory by writing all zeros over Wishbone to the BRAM module. The VGA display resolution we used is 640 by 480, and for a single ASCII character we use a block of 11 by 12 giving us a 80 by 40 grid. We first need to convert our hexadecimal ASCII characters to the 11 by 12 pixel map. Once we wait a single clock cycle for the ASCII2Pixel look up table module to complete, we then iterate through the 11 rows and write each one to the BRAM Wishbone bus. The fan speed, current temperature and desired temperature are being displayed on the display. The display format being used is

$$\begin{aligned} cT &= XX \\ dT &= XX \\ sF &= XX \end{aligned}$$

Where  $cT$  is the current temperature,  $dT$  is the desired temperature,  $sF$  is the fan speed and the  $XX$  is any decimal integer number from 0 to 99. This means there are 15 characters that need to be written to memory. Each time a character is written the current column is incremented, at the end of the value at each key value pair the row is incremented, at the end of writing the value for the fan speed the row and column is reset to its initial value. One issue I ran into was displaying an integer value 0 to 100 on the display, using the given *ASCII2Pixel* module. I decided to only display two digits (0 to 99) and if we ever need to display a 100 (most likely for fan speed) the display will show a MV (max value) instead of a number. The method I used to convert a decimal value to its two digit ASCII equivalent is as follows

$$\begin{aligned} ASCII_{\text{OnesDigit}} &= 0x30 + \text{value mod } 10 \\ ASCII_{\text{TensDigit}} &= 0x30 + \frac{\text{value} - (\text{value mod } 10)}{10} \end{aligned}$$

Where  $\text{value}$  is the integer in range 0 to 100 and  $ASCII_{\text{OnesDigit}}$  is the ASCII value for the ones place and  $ASCII_{\text{TensDigit}}$  is the ASCII value for the tens digit. We bias by 0x30 due to standard ASCII convention.

## 8 Digital PID Control Module

### 8.1 Brief PID Control Theory

A PID controller is a feedback based controller that modulates an actuator based on the error between desired and current values. Very little information about the system is needed for practical use. PID stands for proportional (P) integral (I) and derivative (D). Each part of the controller has a specific function. The magnitude of influence on the controller output due to each component can be individually tuned. Below is equation for the continuous-time PID controller

$$e(t) = r(t) - y(t)$$

$$u(t) = K_p e(t) + K_i \int e(t) dt + K_d \frac{de(t)}{dt}$$

Where  $e(t)$  is a function of error in terms of time, and is computed by subtracting the reference  $r(t)$  by the system output  $y(t)$ . The reference is also known as the set point or desired value. In our case the reference is the desired temperature, the system output is the current temperature of the FPGA and the input signal  $u(t)$  produced by the controller is the fan speed. A block diagram of the system can be seen in figure 11. The  $K_p$ ,  $K_i$ , and  $K_d$  terms are the gains we use to tune each component of the PID controller. Because an FPGA is not a continuous-time system so we need to sample and discretize the above PID equations, these become

$$e[k] = r[k] - y[k]$$

$$u[k + 1] = K_p e[k] + K_i \sum e[k] T_s + K_d \frac{e[k] - e[k - 1]}{T_s}$$

Where  $T_s$  is our sampling rate, I decided on 10 ms so  $T_s = 0.01$  seconds.



**FIG 11.** Block diagram of FPGA temperature regulation system.

### 8.2 System Identification

In order to tune our controller (explained in the following section) a model of our system is a generally useful thing to have. To develop a model we need data that relates the input (fan speed) to the output (FPGA temperature). Using a standard system identification technique, I first set the fan speed to zero. Once the temperature steadied out, I applied a step input of 60% fan speed to the system. During this process I collected the temperature and fan speed over the UART serial connection, saved it as text file and then used the decoding procedure discussed in section 6.3 to obtain usable data. A plot of the open loop unit step response of the system can be seen in figure 12.



**FIG 12.** Open loop step response of the system.

As shown in figure 12, as the fan speed is increased to a constant value the temperature exponentially decays to a steady value. I used MATLAB's System Identification toolbox [6] to estimate the input/output relationship between the fan speed and the temperature. However, I had issues using the raw collected data due to the large 1°C step sizes. So, I first used MATLAB to fit just the output temperature data to a model. The following model was generated

$$m(t) = 3.591e^{-0.0019t} + 30.52e^{5.836 \times 10^{-6} t}$$

Where  $m(t)$  is valid for  $t = 0$  to  $t = 2500$ , or 0 to 2500 milliseconds. This model's confidence bound was 95%. I then used the System Identification toolbox with this model to find the I/O relationship. Because  $m(t)$  is a simple exponentially decaying function, I assumed that this it's a first order system. The computed discrete time state-space equation is

$$\begin{aligned} x[k+1] &= 0.9978x[k] + 0.125u[k] \\ y[k] &= -0.08438x[k] \end{aligned}$$

However, because this is only a single order system, it may be more convenient to represent it using a discrete-time transfer function

$$G(z) = \frac{-0.01055z^{-1}}{1 - 0.9978z^{-1}}$$

### 8.3 Digital PID Controller Design

The PID gains  $K_p$ ,  $K_i$ , and  $K_d$  gains can be tuned analytically or quantitatively. Analytic tuning typically requires a deep understanding of how the system behaves as well as a good intuition of how each gain affects the transient and steady-state system behavior. This method is more iterative, typically more testing and validation has to occur. Because the system dynamics were estimated in the previous section we can use MATLAB to quantitatively compute our gains, designing for a desired transient and steady-state response. Using MATLAB's PID Tuning [7] toolbox with the system defined above, the below gains were computed

$$K_p = -0.303 \quad K_i = -0.123 \quad K_d = 0$$

The first thing that jumps out is that our  $K_d$  term is zero. The  $K_d$  gain heavily influences rapid changes in the system, because our system is represented by a fairly slowly changing first order differential equation, it is not too much of a surprise that MATLAB computed  $K_d$  to equal 0. Something else that stands out is

that the other gains are negative. Intuitively this makes sense and was expected. Our actuator's speed is inversely proportional to our process variable. So, an increase in fan speed decreases the FPGA temperature. This physical behavior explains the negative gains. The below code sample is the VHDL implementation of the discrete-time PID controller shown in section 8.1.

---

```

elsif(samplingRateClock'event and samplingRateClock='1') then
    error := (setpoint - sensorFeedbackValue); --compute new error e[k]
    errorSum := errorSum + error; --continually compute discrete integral
    if(errorSum > 10000) then --integral wind up check
        errorSum := 10000;
    elsif(errorSum < -10000) then
        errorSum := -10000;
    end if;
    errorChange := error - previousError; --compute discrete derivative
    output := (kp*error + ki*errorSum + kd*errorChange)/100; --compute and scale output
    previousError := error; --save error for use in next iteration for discrete derivative
    if(output>100) then --saturate sensor if needed
        output := 100;
    elsif(output<0)then
        output:=0;
    end if;
    controllerOutput<= output;
end if;

```

---

There are a few differences between a pure PID controller and the above VHDL process. The first is the controller output is being divider by 100. This is due to me avoiding the problem of using decimals (again). I scaled the gains by a factor of 100, so that controller can still compute a full range of output and then it is divided by 100 to scale it back to physically realistic fan speeds. The second difference is the first if/else statement. This prevents the integral term from becoming too large when the controller cannot stabilize the output for whatever reason. If the integral term continually summed up or down to either of its bounds, then the behavior of the system would be undefined and it would act erratically. The second set of if/else statements is the final defense against the PID controller attempting to set the fan to a value higher than 100% or less than 0%.



**FIG 13.** Closed loop step response of the system, reacting from a set point change of 34°C to 32°C.

The PID gains performed well when implemented using the above VHDL process. To test the system, I set the desired temperature to 34°C and waited for the system to stabilize. I then changed the desired temperature to 32°C, and collected the data using the serial UART connection. The closed loop system response can be seen in figure 13. There is a small amount of overshoot. After about 20 seconds of settling time, the PID controller's steady state output of 48% seems to adequately minimize the error.

## 9 Implementation Images



**IMG 1.** Photo of the FPGA with the DC fan and motor driver circuit.



**IMG 2.** Photo of the VGA display during operation.

## 10 References

- [1] <http://bytebash.com/2011/10/rs232-uart-vhdl>
- [2] [http://www.xilinx.com/support/documentation/user\\_guides/ug480\\_7Series\\_XADC.pdf](http://www.xilinx.com/support/documentation/user_guides/ug480_7Series_XADC.pdf), page 23
- [3] <https://www.adafruit.com/products/355>
- [4] Special thanks to Andrew Budd for lending me his fan.
- [5] Thanks Dr. Chandy for providing the VGA, BRAM and Char8x1LUT modules.
- [6] <http://www.mathworks.com/products/sysid/>
- [7] <http://www.mathworks.com/discovery/pid-tuning.html>

## 11 Appendix

1. VHDL implementation of main components.

```
-- Company:  
-- Engineer: David Paquette  
--  
-- Create Date: 15:54:03 11/19/15  
-- Design Name:  
-- Module Name:  
-- Project Name:  
-- Target Device:  
-- Tool versions:  
-- Description:  
--  
-- Dependencies:  
--  
-- Revision:  
-- Revision 0.01 - File Created  
-- Additional Comments:  
--  
-----  
library IEEE;  
use IEEE.STD_LOGIC_1164.ALL;  
use IEEE.STD_LOGIC_ARITH.ALL;  
use IEEE.STD_LOGIC_UNSIGNED.ALL;  
  
---- Uncomment the following library declaration if instantiating  
---- any Xilinx primitives in this code.  
--library UNISIM;  
--use UNISIM.VComponents.all;  
  
entity TemperatureControlProject is  
    Port (  
        VGA_R : out std_logic_vector(3 downto 0);  
        VGA_G : out std_logic_vector(3 downto 0);  
        VGA_B : out std_logic_vector(3 downto 0);  
        VGA_HS : out std_logic;  
        VGA_VS : out std_logic;  
            uart_rxd_out : out std_logic;  
            uart_txd_in : in std_logic;  
            btnd : in std_logic;  
            btnu : in std_logic;  
            ja : out std_logic_vector(7 downto 0);  
            CPU_RESETN : in std_logic;  
            CLK100MHZ : in std_logic  
    );  
end TemperatureControlProject;  
  
architecture Behavioral of TemperatureControlProject is  
    -- Wishbone signals  
    signal ACK_I_M:      std_logic_vector(3 downto 0);  
    signal ACK_O_S:      std_logic_vector(3 downto 0);  
    signal ADR_O_M0:     std_logic_vector( 31 downto 0 );  
    signal ADR_O_M1:     std_logic_vector( 31 downto 0 );  
    signal ADR_O_M2:     std_logic_vector( 31 downto 0 );  
    signal ADR_O_M3:     std_logic_vector( 31 downto 0 );  
    signal ADR_I_S:      std_logic_vector( 31 downto 0 );  
    signal CYC_O_M:      std_logic_vector(3 downto 0);
```

```

signal DAT_0_M0:      std_logic_vector( 31 downto 0 );
signal DAT_0_M1:      std_logic_vector( 31 downto 0 );
signal DAT_0_M2:      std_logic_vector( 31 downto 0 );
signal DAT_0_M3:      std_logic_vector( 31 downto 0 );
signal DWR:           std_logic_vector( 31 downto 0 );
signal DAT_0_S0:      std_logic_vector( 31 downto 0 );
signal DAT_0_S1:      std_logic_vector( 31 downto 0 );
signal DAT_0_S2:      std_logic_vector( 31 downto 0 );
signal DAT_0_S3:      std_logic_vector( 31 downto 0 );
signal DRD:           std_logic_vector( 31 downto 0 );
signal IRQ_0_S:        std_logic_vector(3 downto 0);
signal IRQ_I_M:        std_logic;
signal IRQV_I_M:       std_logic_vector(1 downto 0);
signal STB_I_S:        std_logic_vector(3 downto 0);
signal STB_0_M:        std_logic_vector(3 downto 0);
signal WE_0_M:         std_logic_vector(3 downto 0);
signal WE:             std_logic;

signal ascii_data_available : std_logic;
signal ascii_data : std_logic_vector(7 downto 0);

constant buffer_base : std_logic_vector(31 downto 0) := (others => '0');

signal sys_clk, sys_rst : std_logic;
signal clk200 : std_logic;
signal pmod : std_logic_vector(7 downto 0):=(others=>'0');
signal hOut :std_logic :='0';

begin

    sys_rst <= cpu_resetn;
    ja<=pmod;
    pmod(7)<=hOut;

    clock200_inst : entity work.clk200
        port map ( clk_in1 => clk100mhz, clk_out1 => sys_clk, clk_out2 => clk200 );

    wb_intercon : entity work.wb_intercon
        port map ( clk => sys_clk, rst => sys_rst,
                   ack_i_m => ACK_I_M, ack_o_s => ack_o_s,
                   adr_o_m0 => adr_o_m0, adr_o_m1 => adr_o_m1,
                   adr_o_m2 => adr_o_m2, adr_o_m3 => adr_o_m3,
                   dat_o_m0 => dat_o_m0, dat_o_m1 => dat_o_m1,
                   dat_o_m2 => dat_o_m2, dat_o_m3 => dat_o_m3,
                   dat_o_s0 => dat_o_s0, dat_o_s1 => dat_o_s1,
                   dat_o_s2 => dat_o_s2, dat_o_s3 => dat_o_s3,
                   adr_i_s => adr_i_s, drd => drd, dwr => dwr,
                   irq_o_s => irq_o_s, irq_i_m => irq_i_m, irqv_i_m
                   => irqv_i_m,
                   cyc_o_m => cyc_o_m, stb_o_m => stb_o_m, stb_i_s
                   => stb_i_s, we_o_m => we_o_m, we => we );

```

```
temperatureControlMaster : entity work.TemperatureControlMaster
    port map ( clk_i => sys_clk, rst_i => sys_rst,
                adr_o => adr_o_m0, dat_i => drd, dat_o =>
                dat_o_m0,
                ack_i => ack_i_m(0), cyc_o => cyc_o_m(0), stb_o
                => stb_o_m(0),
                we_o => we_o_m(0), tx_in=>uart_txd_in,
                rx_out=>uart_rxd_out,
                incrementSetpointButton=>btnu,
                decrememntSetpointButton=>btnd,
                pwm0ut=>pmod(0));

vga : entity work.wb_vga640x480
    port map ( clk_i => sys_clk, rst_i => sys_rst,
                adr_o => adr_o_m1, dat_i => drd, dat_o =>
                dat_o_m1,
                ack_i => ack_i_m(1), cyc_o => cyc_o_m(1), stb_o
                => stb_o_m(1),
                we_o => we_o_m(1),
                red => vga_r, green => vga_g, blue => vga_b,
                hsync => vga_hs, vsync => vga_vs);

wb_bram : entity work.wb_bram
    port map ( clk_i => sys_clk, rst_i => sys_rst,
                adr_i => adr_i_s, dat_i => dwr, dat_o =>
                dat_o_s0,
                ack_o => ack_o_s(0), stb_i => stb_i_s(0), we_i
                => we );

heater : entity work.GenerateHeat
    port map ( clk=>sys_clk, reset=>sys_rst, output=>h0ut);

cyc_o_m(3) <= '0';
cyc_o_m(2) <= '0';

end Behavioral;
```

```
-- Company:  
-- Engineer: David Paquette  
--  
-- Create Date: 15:59:15 11/19/15  
-- Design Name:  
-- Module Name:  
-- Project Name:  
-- Target Device:  
-- Tool versions:  
-- Description:  
  
-- Dependencies:  
--  
-- Revision:  
-- Revision 0.01 - File Created  
-- Additional Comments:  
--  
-----  
library IEEE;  
use IEEE.STD_LOGIC_1164.ALL;  
use IEEE.STD_LOGIC_ARITH.ALL;  
use IEEE.STD_LOGIC_UNSIGNED.ALL;  
  
---- Uncomment the following library declaration if instantiating  
---- any Xilinx primitives in this code.  
--library UNISIM;  
--use UNISIM.VComponents.all;  
  
entity TemperatureControlMaster is  
    Port ( clk_i : in std_logic;  
           rst_i : in std_logic;  
                  adr_o : out std_logic_vector(31 downto 0);  
           dat_i : in std_logic_vector(31 downto 0);  
           dat_o : out std_logic_vector(31 downto 0);  
           ack_i : in std_logic;  
           cyc_o : out std_logic;  
           stb_o : out std_logic;  
           we_o : out std_logic;  
                  rx_out : out std_logic;  
                  tx_in : in std_logic;  
incrementSetpointButton : in std_logic;  
pwmOut : out std_logic;  
decrememntSetpointButton : in std_logic  
    );  
end TemperatureControlMaster;  
  
architecture Behavioral of TemperatureControlMaster is  
    signal currentTemperature : integer range 0 to 100:=0;  
    signal desiredTemperature : integer range 0 to 100:=32;  
    signal fanSpeedPercent : integer range 0 to 100:=0;  
  
    signal tx, rx, rx_sync, reset, reset_sync,tx_sig, onemsec_clk, pwm_clk :  
        std_logic;  
  
    signal eightBitBuffer : std_logic_vector(7 downto 0):=(others=>'0');
```

```
begin
    tx_sig <= tx_in;

    onemsec_clk_divider : entity work.clock_divider
        generic map ( divisor => 100000 )
        port map (
            clk_in => clk_i,
            reset => rst_i,
            clk_out => onemsec_clk
        );

    pwmFreqClock : entity work.clock_divider
        generic map ( divisor => 2000 )
        port map (
            clk_in => clk_i,
            reset => rst_i,
            clk_out => pwm_clk
        );

    memoryWriter : entity work.MemoryWriter
        port map ( clk_i => clk_i, rst_i => rst_i ,
                   adr_o => adr_o, dat_i => dat_i, dat_o => dat_o,
                   ack_i => ack_i, cyc_o => cyc_o, stb_o => stb_o,
                   we_o => we_o, currentTemperature=> currentTemperature,
                   desiredTemperature=> desiredTemperature,
                   fanSpeedPercent=> fanSpeedPercent
        );
    pidController : entity work.PIDController
        port map( samplingRateClock=>onemsec_clk,
                  reset=>rst_i,
                  setpoint=>desiredTemperature,
                  sensorFeedbackValue=>currentTemperature,
                  controlOutput =>fanSpeedPercent );

    temperatureSetPointControl : entity work.TemperatureSetpointControl
        port map(clk_i=>onemsec_clk,
                  rst_i=>rst_i,
                  incrementButton=>incrementSetpointButton,
                  decrementButton=>decrememntSetpointButton,
                  selectedTemperature=>desiredTemperature);

    temperatureSensor : entity work.TemperatureSensorInterface
        port map ( clk_i=>clk_i,
                  rst_i=>rst_i,
                  temperatureCelcius=>currentTemperature);

    dcFanInterface: entity work.dcFanInterface
        port map(fanSpeed=>fanSpeedPercent,
                  --fanSpeed=>desiredTemperature,
                  pwmPinOut=>pwmOut,
                  clk_i=>pwm_clk);

    serialController : entity work.ValuesToSerial
    port map (
```

```
CLOCK      => clk_i,
RESET      => reset,
RX         => rx,
TX         => tx,
temperatureIn => eightBitBuffer+currentTemperature,
fanSpeedIn  => eightBitBuffer+fanSpeedPercent
--fanSpeedIn => eightBitBuffer+desiredTemperature
);

process (clk_i, rst_i)
begin
    if(rst_i='0') then
        reset <= '1'; -- the nexys4ddr is active low, so invert
                      reset to use with this serial lib
    elsif (clk_i'event and clk_i = '1') then
        reset <='0';
        rx_sync <= tx_sig; -- the perspective of the tx and rx
                           is reversed for the nexys
        rx   <= rx_sync;
        rx_out <= tx;
    end if;
end process;

end Behavioral;
```

```
--  
-- Company:  
-- Engineer: David Paquette  
--  
-- Create Date: 17:27:40 11/19/2015  
-- Design Name:  
-- Module Name: TemperatureSensorInterface - Behavioral  
-- Project Name:  
-- Target Devices:  
-- Tool versions:  
-- Description:  
--  
-- Dependencies:  
--  
-- Revision:  
-- Revision 0.01 - File Created  
-- Additional Comments:  
--  
--  
library IEEE;  
use IEEE.STD_LOGIC_1164.ALL;  
--use IEEE.STD_LOGIC_ARITH.ALL;  
use IEEE.STD_LOGIC_UNSIGNED.ALL;  
USE ieee.numeric_std.ALL;  
use ieee.math_real.all;  
  
-- Uncomment the following library declaration if using  
-- arithmetic functions with Signed or Unsigned values  
--use IEEE.NUMERIC_STD.ALL;  
  
-- Uncomment the following library declaration if instantiating  
-- any Xilinx primitives in this code.  
--library UNISIM;  
--use UNISIM.VComponents.all;  
  
entity TemperatureSensorInterface is  
generic(  
    X_TMP_COL_WIDTH : natural := 50; -- = SZ_TH_WIDTH - width of a  
                                     TEMP column  
    Y_TMP_COL_HEIGHT : natural := 472; -- = SZ_TH_HEIGHT - height of a  
                                      TEMP column  
    X_TMP_H_LOC : natural := 1050; -- X Location of the TEMP Column  
    Y_TMP_V_LOC : natural := 80; -- Y Location of the TEMP Column  
    INPUT_DATA_WIDTH : natural := 12; -- Data width is 13 for the  
                                     ADT7420 Temperature Sensor and  
                                     -- 12 for the XADC temperature  
                                     -- data and the Accelerometer  
                                     -- Temperature Sensor  
    TMP_TYPE : string := "XADC" -- Either "XADC" or "TEMP_ACC"  
);  
Port (clk_i: in std_logic;  
      rst_i : in std_logic;  
      temperatureCelcius : out integer range 0 to 100 );  
end TemperatureSensorInterface;
```

```

architecture Behavioral of TemperatureSensorInterface is
    signal temperature : std_logic_vector(11 downto 0);
    signal reset : std_logic;
    signal clk : std_logic;

    -- Maximum temperature
    constant TEMP_MAX : std_logic_vector (23 downto 0) := X"000500"; -- 80C *
    16

    -- Scale incoming XADC temperature data, according to the XADC datasheet
    constant XADC_TMP_SCALE : std_logic_vector(17 downto 0) := "111110111" &
        "111110011"; --503.975 (18bit)
    constant thirtyTwobuffer : std_logic_vector(30 downto 0):=(others=>'0');
    -- Convert Kelvin to Celsius
    constant XADC_TMP_OFFSET : std_logic_vector(30 downto 0) := thirtyTwobuffer
        +integer(round(273.15)*4096.0);

    -- Synchronize incoming temperature to the clock
    signal temp_sync0, temp_sync : std_logic_vector(temperature'range);

    -- signal storing the scaled XADC temperature data
    signal temp_xad_scaled : std_logic_vector(temp_sync'length
        +XADC_TMP_SCALE'length-1 downto 0); --12bit*18bit=30bit
    -- signal storing the offseted XADC temperature data
    signal temp_xad_offset : std_logic_vector(XADC_TMP_OFFSET'range); --31bit
    -- signal storing XADC temperature data converted to Celsius
    signal temp_xad_celsius : std_logic_vector(temp_xad_offset'length-8-1
        downto 0); --23bit
    -- Signal storing the FPGA temperature limited to between 0C and 80C * 16
    signal temp_xad_capped : std_logic_vector(temp_xad_celsius'high-1 downto
        0); --no sign bit

    signal temp : std_logic_vector(7 downto 0);
begin
    clk<=clk_i;
    reset<=rst_i;

    temperatureCelcius<=to_integer(unsigned(temp));

    Inst_FPGAMonitor: entity work.FPGAMonitor PORT MAP(
        CLK_I          => clk,
        RST_I          => reset,
        TEMP_0          => temperature
    );

process(clk)
begin
    if clk'EVENT and clk = '1' then
        temp_sync0 <= temperature; --synchronize with pzl_clk domain
        temp_sync <= temp_sync0;

        --30b           12b           18b
        temp_xad_scaled <= temp_sync * XADC_TMP_SCALE; -- ADC *

```

```
      503.975 (fixed-point; decimal point at 9b)

      temp_xad_offset <= '0' & temp_xad_scaled(29 downto 9) -
          XADC_TMP_OFFSET; -- ADC * 503.975 - 273.15 * 4096

      temp_xad_celsius <= temp_xad_offset(temp_xad_offset'high
          downto 8); -- (ADC * 503.975 - 273.15) / 256;
          1LSB=0.625C

      if (temp_xad_celsius(temp_xad_celsius'high) = '1') then --if
          negative, cap to 0
          temp_xad_capped <= (others => '0');
      elsif (temp_xad_celsius(temp_xad_celsius'high-1 downto 0) >
          TEMP_MAX) then --if too big, cap to maximum scale /
          0.0625
          temp_xad_capped <= TEMP_MAX(temp_xad_capped'range);
      else
          temp_xad_capped <=
              temp_xad_celsius(temp_xad_celsius'high-1 downto
              0); --get rid of the sign bit
      end if;

      temp<=temp_xad_capped(11 downto 4); -- remove all data under
          0C (decimals)
  end if;
end process;
end Behavioral;
```

```
--  
-- Company:  
-- Engineer: David Paquette  
--  
-- Create Date: 16:50:06 11/19/2015  
-- Design Name:  
-- Module Name: DCFanInterface - Behavioral  
-- Project Name:  
-- Target Devices:  
-- Tool versions:  
-- Description:  
--  
-- Dependencies:  
--  
-- Revision:  
-- Revision 0.01 - File Created  
-- Additional Comments:  
--  
--  
library IEEE;  
use IEEE.STD_LOGIC_1164.ALL;  
use IEEE.STD_LOGIC_UNSIGNED.ALL;  
USE ieee.numeric_std.ALL;  
-- Uncomment the following library declaration if using  
-- arithmetic functions with Signed or Unsigned values  
--use IEEE.NUMERIC_STD.ALL;  
  
-- Uncomment the following library declaration if instantiating  
-- any Xilinx primitives in this code.  
--library UNISIM;  
--use UNISIM.VComponents.all;  
  
entity DCFanInterface is  
    Port(fanSpeed : in integer range 0 to 100;  
          pwmPinOut: out std_logic;  
          clk_i : in std_logic);  
end DCFanInterface;  
  
architecture Behavioral of DCFanInterface is  
    signal cnt : std_logic_vector(6 downto 0);  
    signal t : std_logic:='0';  
begin  
    --pwmPinOut <= '1';  
    --pwmPinOut <= data(0);  
    pwmPinOut <= t;  
  
    process(clk_i)  
    begin  
        if rising_edge(clk_i) then  
            cnt <= cnt + '1';  
        end if;  
    end process;  
  
    process(fanSpeed, cnt)
```

```
begin
    if unsigned(cnt) < fanSpeed then
        t <= '1';
    else
        t <= '0';
    end if;
end process;
end Behavioral;
```

```
-- Company:  
-- Engineer: David Paquette  
--  
-- Create Date: 11/20/15  
-- Design Name:  
-- Module Name:  
-- Project Name:  
-- Target Device:  
-- Tool versions:  
-- Description:  
  
-- Dependencies:  
--  
-- Revision:  
-- Revision 0.01 - File Created  
-- Additional Comments:  
--  
-----  
library IEEE;  
use IEEE.STD_LOGIC_1164.ALL;  
use IEEE.STD_LOGIC_ARITH.ALL;  
use IEEE.STD_LOGIC_UNSIGNED.ALL;  
  
entity ValuesToSerial is  
    port  
    (  
        -- General  
        CLOCK : in std_logic;  
        RESET : in std_logic;  
        RX : in std_logic;  
        TX : out std_logic;  
        temperatureIn : in std_logic_vector(7 downto 0);  
        fanSpeedIn : in std_logic_vector(7 downto 0)  
    );  
end ValuesToSerial;  
  
architecture RTL of ValuesToSerial is  
    constant BAUD_RATE : positive := 115200;  
    constant CLOCK_FREQUENCY : positive := 100000000;  
  
    signal uart_data_in : std_logic_vector(7 downto 0);  
    signal uart_data_out : std_logic_vector(7 downto 0);  
    signal uart_data_in_stb : std_logic;  
    signal uart_data_in_ack : std_logic;  
    signal uart_data_out_stb : std_logic;  
    signal uart_data_out_ack : std_logic;  
  
    signal s_clk: std_logic;  
    type StateType is (start, writeTemperatureData, writeComma,  
                      writeFanSpeedData, writeNewline, waitForOneSecondTick);  
    signal state : StateType := start;  
  
    signal oneSecondCounter :std_logic_vector(31 downto 0):=(others=>'0');  
begin
```

```
s_clk<=CLOCK;

UART_inst1 : entity work.UART
generic map (
    BAUD_RATE          => BAUD_RATE,
    CLOCK_FREQUENCY    => CLOCK_FREQUENCY
)
port map (
    CLOCK              => CLOCK,
    RESET              => RESET,
    DATA_STREAM_IN     => uart_data_in,
    DATA_STREAM_IN_STB => uart_data_in_stb,
    DATA_STREAM_IN_ACK => uart_data_in_ack,
    DATA_STREAM_OUT    => uart_data_out,
    DATA_STREAM_OUT_STB=> uart_data_out_stb,
    DATA_STREAM_OUT_ACK=> uart_data_out_ack,
    TX                 => TX,
    RX                 => RX
);

process(s_clk,RESET)
begin
    if(RESET='1') then
        uart_data_in_stb      <= '0';
        uart_data_out_ack     <= '0';
        uart_data_in          <= (others => '0');
        state<=start;
        oneSecondCounter<=(others=>'0');
    elsif(s_clk'event and s_clk='1') then
        case state is
            when start =>
                uart_data_in_stb <= '0';
                state<=writeTemperatureData;
                oneSecondCounter<=(others=>'0');
            when writeTemperatureData =>
                if(uart_data_in_ack = '1') then
                    state<=writeComma;
                    uart_data_in_stb <= '0';
                else
                    uart_data_in_stb <= '1';
                    uart_data_in<= temperatureIn;
                end if;
            when writeComma =>
                if(uart_data_in_ack = '1') then
                    state<=writeFanSpeedData;
                    uart_data_in_stb <= '0';
                else
                    uart_data_in_stb <= '1';
                    uart_data_in<= x"2c";
                end if;
            when writeFanSpeedData =>
                if(uart_data_in_ack = '1') then
                    state<=write.NewLine;
                    uart_data_in_stb <= '0';
                else

```

```
        uart_data_in_stb <= '1';
        uart_data_in<= fanSpeedIn;
    end if;
when write.NewLine =>
    if(uart_data_in_ack = '1') then
        state<=waitForOneSecondTick;
        uart_data_in_stb <= '0';
    else
        uart_data_in_stb <= '1';
        uart_data_in<= x"0A";
    end if;
when waitForOneSecondTick =>
    uart_data_in_stb<='0';
    oneSecondCounter <= oneSecondCounter + 1;
    --if(oneSecondCounter >= 100000000) then
    if(oneSecondCounter >= 10000000) then
        state<=start;
    end if;
end case;
end if;
end process;

end RTL;
```

```
-- Company:  
-- Engineer: David Paquette  
--  
-- Create Date: 11/19/15  
-- Design Name:  
-- Module Name:  
-- Project Name:  
-- Target Device:  
-- Tool versions:  
-- Description:  
  
-- Dependencies:  
--  
-- Revision:  
-- Revision 0.01 - File Created  
-- Additional Comments:  
--  
-----  
library IEEE;  
use IEEE.STD_LOGIC_1164.ALL;  
use IEEE.STD_LOGIC_ARITH.ALL;  
use IEEE.STD_LOGIC_UNSIGNED.ALL;  
use ieee.numeric_std.all;  
  
---- Uncomment the following library declaration if instantiating  
---- any Xilinx primitives in this code.  
--library UNISIM;  
--use UNISIM.VComponents.all;  
  
entity MemoryWriter is  
    Port ( clk_i : in std_logic;  
           rst_i : in std_logic;  
                  adr_o : out std_logic_vector(31 downto 0);  
                  dat_i : in std_logic_vector(31 downto 0);  
                  dat_o : out std_logic_vector(31 downto 0);  
                  ack_i : in std_logic;  
                  cyc_o : out std_logic;  
                  stb_o : out std_logic;  
                  we_o : out std_logic;  
                  currentTemperature: in integer range 0 to 100;  
                  desiredTemperature : in integer range 0 to 100;  
                  fanSpeedPercent : in integer range 0 to 100  
            );  
end MemoryWriter;  
  
architecture Behavioral of MemoryWriter is  
    signal ascii : std_logic_vector(7 downto 0);  
    signal pixelnum : integer range 0 to 7;  
  
    signal buffer_base : std_logic_vector(31 downto 0):=(others=>'0');  
    signal rst_p : std_logic;  
    signal txtcolor : std_logic_vector(3 downto 0):="1111"; -- white  
    signal bgcolor : std_logic_vector(3 downto 0):="0001"; -- blue  
    signal pixels : std_logic_vector(7 downto 0);
```

```
type StateType is (initialzeMemoryState, writeIntialMemoryState,
                    startState,
                    writePixelToMemory, getPixelData, waitState);
signal state : StateType := initialzeMemoryState;
signal memoryInitializationComplete : std_logic:='0';
signal row, column, line : integer range 0 to 100:=0;
signal cletter : std_logic_vector(7 downto 0):=x"63"; --"c"
signal dletter : std_logic_vector(7 downto 0):=x"64"; --"d"
signal fletter : std_logic_vector(7 downto 0):=x"66"; --"f"
signal Sletter : std_logic_vector(7 downto 0):=x"53"; --"S"
signal Tletter : std_logic_vector(7 downto 0):=x"54"; --"T"
signal equalletter : std_logic_vector(7 downto 0):=x"3d"; --"="
signal twoDigitAscii : std_logic_vector(15 downto 0);
signal MVLetters : std_logic_vector(15 downto 0):=x"4d56"; --MV
signal number : integer range 0 to 100:=0;
begin
    rst_p <= not rst_i;

    -- the lookup table maps the ascii code to the pixels for that particular
    -- character.
    -- The line input determines which of the 12 lines of the character we want.
    -- The
    -- lookup table is implemented with the builtin registered BRAM, so the
    -- output is
    -- available only at the next clock cycle
    lut : entity work.char8x12_lookup_table
        port map( clk => clk_i, reset => rst_p, ascii => ascii, line => line,
                  pixels => pixels );

    -- two digit int to ascii
    process(number)
    begin
        if(number < 100) then
            twoDigitAscii(15 downto 8)<= x"30" +
                std_logic_vector(to_unsigned((number-(number mod 10))/10, 8));
            twoDigitAscii(7 downto 0)<= x"30" +
                std_logic_vector(to_unsigned((number mod 10), 8));
        else
            twoDigitAscii<=MVLettters;
        end if;
    end process;

    process( clk_i, rst_i, state)
        variable textCounter : std_logic_vector(15 downto 0):=(others=>'0');
    begin
        if (rst_i = '0') then
            state <= initialzeMemoryState;
            column <= 0;
            line <= 0;
            row <= 0;
            stb_o<='0';
            cyc_o<='0';
            we_o <='0';
        end if;
        if (state = writeIntialMemoryState) then
            if (textCounter <= "0000000000000000") then
                textCounter <= textCounter + "0000000000000001";
                if (row < 100) then
                    row <= row + 1;
                else
                    row <= 0;
                    column <= column + 1;
                    if (column > 11) then
                        column <= 0;
                        line <= line + 1;
                    end if;
                end if;
                if (line > 11) then
                    line <= 0;
                    column <= 0;
                    row <= 0;
                    state <= startState;
                end if;
            end if;
        end if;
    end process;
```

```
elsif ( clk_i'event and clk_i='1' ) then
    case state is
        when initializeMemoryState=>
            cyc_o<='1';
            stb_o<='0';
            we_o <='0';
            state<= writeInitialMemoryState;
            line <= line + 1;
            if(line = 11) then
                line <= 0;
                column <= column + 1;
                if(column = 79) then
                    column <= 0;
                    row<=row + 1;
                    if(row = 39) then
                        row <= 0;
                        state<=startState;
                        column <= 0;
                        line <= 0;
                        row <= 0;
                    end if;
                end if;
            end if;
        when writeInitialMemoryState=>
            stb_o<='1';
            we_o <='1';
            adr_o <= buffer_base + (row*80*12 + column +
                80*line)*4;
            dat_o <=
                bgcolor&bgcolor&bgcolor&bgcolor&bgcolor&bgco
                lor&bgcolor&bgcolor;
            if(ack_i='1') then
                state<= initializeMemoryState;
            end if;
        when startState=>
            column <= 39;
            line <= 0;
            row <= 19;
            stb_o<='0';
            cyc_o<='0';
            state<=getPixelData;
        when getPixelData=>
            line <= line + 1;
            if(textCounter=0) then
                --print c
                ascii <= cletter;
                number <= currentTemperature; -- queue up
                    current temp for ascii conversion
            elsif(textCounter=1) then
                --print T
                ascii <= Tletter;
            elsif(textCounter=2) then
                --print =
                ascii <= equalletter;
            elsif(textCounter=3) then
                --print first digit of current temp
```

```
        ascii <= twoDigitAscii(15 downto 8);
elsif(textCounter=4) then
--print second digit of current temp
    ascii <= twoDigitAscii(7 downto 0);
elsif(textCounter=5) then
--print d
    ascii <= dletter;
    number <= desiredTemperature; --queue up
        desired temp for ascii conversion
elsif(textCounter=6) then
--print T
    ascii <= Tletter;
elsif(textCounter=7) then
--print =
    ascii <= equalletter;
elsif(textCounter=8) then
--print first digit of desired temp
    ascii <= twoDigitAscii(15 downto 8);
elsif(textCounter=9) then
    ascii <= twoDigitAscii(7 downto 0);
elsif(textCounter=10) then
--print f
    ascii <= fletter;
    number <= fanSpeedPercent; --queue up fan
        speed for ascii conversion
elsif(textCounter=11) then
--print S
    ascii <= Sletter;
elsif(textCounter=12) then
--print =
    ascii <= equalletter;
elsif(textCounter=13) then
--print first digit of fan speed
    ascii <= twoDigitAscii(15 downto 8);
elsif(textCounter=14) then
    ascii <= twoDigitAscii(7 downto 0);
end if;
if(line >= 11) then
    column <= column + 1;
    line <= 0;
    textCounter := textCounter + 1;
    if(textCounter = 0 )then
    elsif(textCounter = 1) then
        column <= column + 1;
    elsif(textCounter = 2) then
        column <= column + 1;
    elsif(textCounter = 3) then
        column <= column + 1;
    elsif(textCounter = 4) then
        column <= column + 1;
    elsif(textCounter = 5) then
        column <= 39;
        row <= row + 1;
    elsif(textCounter = 6) then
        column <= column + 1;
    elsif(textCounter = 7) then
```

```
        column <= column + 1;
elsif(textCounter = 8) then
    column <= column + 1;
elsif(textCounter = 9) then
    column <= column + 1;
elsif(textCounter = 10) then
    row <=row + 1;
    column <= 39;
elsif(textCounter=11) then
    column <= column + 1;
elsif(textCounter=12) then
    column <= column + 1;
elsif(textCounter=13) then
    column <= column + 1;
elsif(textCounter=14) then
    column <= column + 1;
elsif(textCounter > 14) then
    column <= 39;
    row <=19;
    textCounter := (others=>'0');
end if;
else
    state<=writePixelToMemory;
end if;
when writePixelToMemory=>
    stb_o<='1';
    we_o <='1';
    cyc_o<='1';
    adr_o <= buffer_base + (row*80*12 + column +
     80*line)*4;
    if(pixels(0)='1')then dat_o(3 downto 0) <=
        txtcolor;else dat_o(3 downto 0) <=
        bgcolor;end if;
    if(pixels(1)='1')then dat_o(7 downto 4) <=
        txtcolor;else dat_o(7 downto 4) <=
        bgcolor;end if;
    if(pixels(2)='1')then dat_o(11 downto 8) <=
        txtcolor;else dat_o(11 downto 8) <=
        bgcolor;end if;
    if(pixels(3)='1')then dat_o(15 downto 12) <=
        txtcolor;else dat_o(15 downto 12) <=
        bgcolor;end if;
    if(pixels(4)='1')then dat_o(19 downto 16) <=
        txtcolor;else dat_o(19 downto 16) <=
        bgcolor;end if;
    if(pixels(5)='1')then dat_o(23 downto 20) <=
        txtcolor;else dat_o(23 downto 20) <=
        bgcolor;end if;
    if(pixels(6)='1')then dat_o(27 downto 24) <=
        txtcolor;else dat_o(27 downto 24) <=
        bgcolor;end if;
    if(pixels(7)='1')then dat_o(31 downto 28) <=
        txtcolor;else dat_o(31 downto 28) <=
        bgcolor;end if;
    if(ack_i='1') then
        stb_o<='0';
```

```
        cyc_o<='0';
        we_o <='0';
        state<=getPixelData;
    end if;
    when waitState=>

        end case;
    end if;
end process;

end Behavioral;
```

```
--  
-- Company:  
-- Engineer: David Paquette  
--  
-- Create Date: 16:49:31 11/19/2015  
-- Design Name:  
-- Module Name: PIDController - Behavioral  
-- Project Name:  
-- Target Devices:  
-- Tool versions:  
-- Description:  
--  
-- Dependencies:  
--  
-- Revision:  
-- Revision 0.01 - File Created  
-- Additional Comments:  
--  
--  
library IEEE;  
use IEEE.STD_LOGIC_1164.ALL;  
use IEEE.numeric_std.all;  
  
-- Uncomment the following library declaration if using  
-- arithmetic functions with Signed or Unsigned values  
--use IEEE.NUMERIC_STD.ALL;  
  
-- Uncomment the following library declaration if instantiating  
-- any Xilinx primitives in this code.  
--library UNISIM;  
--use UNISIM.VComponents.all;  
  
entity PIDController is  
    Port (        samplingRateClock : in std_logic;  
                reset : in std_logic;  
                setpoint: in integer range 0 to 100;  
                sensorFeedbackValue : in integer range 0 to 100;  
                controlOutput : out integer range 0 to 100  
    );  
end PIDController;  
  
architecture Behavioral of PIDController is  
    signal controllerOutput : integer range 0 to 100:=0;  
    constant kp : integer range -1000 to 0:=-300;  
    constant ki : integer range -1000 to 0:=-2;  
    constant kd : integer range -1000 to 0:=0;  
begin  
    process(samplingRateClock, reset)  
        variable error : integer range -1000 to 1000:=0;  
        variable previousError : integer range -1000 to 1000:=0;  
        variable errorSum: integer range -100000 to 100000:=-0;  
        variable errorChange: integer range -1000 to 1000:=0;  
        variable output: integer range -1000 to 1000:=0;  
    begin
```

```
if(reset='0') then
    error:=0;
    previousError:=0;
    errorSum:=0;
    errorChange:=0;
    output:=0;
    controllerOutput<= 0;
elsif(samplingRateClock'event and samplingRateClock='1') then
    error := (setpoint - sensorFeedbackValue);
    errorSum      := errorSum + error;
    if(errorSum > 10000) then
        errorSum := 10000;
    elsif(errorSum < -10000) then
        errorSum := -10000;
    end if;
    errorChange := error - previousError;
    output := (kp*error + ki*errorSum + kd*errorChange)/100;
    previousError := error;
    if(output>100) then
        output := 100;
    elsif(output<0)then
        output:=0;
    end if;
    controllerOutput<= output;
end if;
end process;

control0utput<=controllerOutput;

end Behavioral;
```