

**ECE 385**

Spring 22  
Experiment #7

**VGA Text Mode Controller with Avalon-MM  
Interface**

Siddarth Iyer, Geitanksha Tandon  
TA: Ruihao Yao (RY)

---

## **1. Introduction:**

---

*a. Briefly summarize the operation of the VGA interface, what are we trying to accomplish with this design?*

In this experiment, we design a simple text mode graphics controller interfaced with the Avalon Memory-Mapped bus to display one of 128 glyphs. It supports 80 columns and 30 rows with the capability of displaying a total of 2400 characters. Data regarding the characters to be printed is stored in the VRAM (Registers in lab 1, On-chip memory in lab 2) and accessed via the Avalon bus to write onto the VGA display. The graphic controller supported monochrome display in week 1 with the ability to set a singular foreground and background color, with capabilities extended in week 2 to display up to 16 colors via color palettes. The glyphs are stored as a bitmap with each glyph consisting of 8x16 pixels.

---

*b. You should address how the design you created builds on top of the basic one provided for Lab 6.2.*

In lab 6.1 we had created a NIOS-II based system on the MAX10 device where the NIOS would be the system controller, and then the peripherals in the FPGA would handle higher-processor operations. In lab 6.2, we altered the system to integrate a USB host-controller interface (MAX2321E) that would communicate with our DE10-Lite board with the SPI protocol. In 6.2, we also used the DE10-Lite board's USB port to connect it to our VGA Monitor.

In lab 7, we had to support a text-mode graphical controller IP core on our platform designer. This needed to be connected to our Avalon bus (memory mapped) to support the 80x30 character mode on our VGA Monitor. We just built the IP and understood how to construct it. We needed to determine which glyphs needed to be drawn and how to draw it.

---

## **2. Written Description of Lab 7 System**

*a. Week 1 (Monochrome Text Display)*

i. Written Description of the entire Lab 7 system

This is answered in the entire lab report - In the first week, we had to create a monochrome text display to track the glyphs via Font\_rom and then print out sentences with these glyphs in monochrome colors. We created an IP module that would handle the text-mode graphics controller and support an 80 character mode.

---

*ii. Describe at a high level your VGA Text Mode controller IP*

The VGA Text Mode controller is connected to the Avalon memory-mapped bus and supports 80 columns and 30 rows. This corresponds to a total of 2400 characters. The data for each of these 2400 cells was stored in 600 32-bit registers with each register storing data for 4 cells. The registers are read from and written to via the Avalon bus. Each cell had a corresponding 8 bits of data, with 7 bits of ‘code’ corresponding to the ‘Glyph code from IBM Codepage 437’, and a singular invert bit. The 601st register stored 4 bit RGB values for the foreground and background color. Based on the current DrawX and DrawY from the VGA Controller module, the IP calculates which cell is currently being written, gets the 7 bit Glyph code, accesses the fontdata and determines which row and column of font data is currently being printed. Based on the fontdata and with the use of invert logic, the RGB value for the pixel is set as either the foreground or background.

| <b>Bit</b> | <b>31</b> | <b>30-24</b> | <b>23</b> | <b>22-16</b> | <b>15</b> | <b>14-8</b> | <b>7</b> | <b>6-0</b> |
|------------|-----------|--------------|-----------|--------------|-----------|-------------|----------|------------|
| Function   | IV3       | CODE3        | IV2       | CODE2        | IV1       | CODE1       | IV0      | CODE0      |

IVn = Inverse bit N

CODEn = Glyph code from IBM Codepage 437

| <b>Bit</b> | <b>31-25</b> | <b>24-21</b> | <b>20-17</b> | <b>16-13</b> | <b>12-9</b> | <b>8-5</b> | <b>4-1</b> | <b>0</b> |
|------------|--------------|--------------|--------------|--------------|-------------|------------|------------|----------|
| Function   | UNUSED       | FGD_R        | FGD_G        | FGD_B        | BKG_R       | BKG_G      | BKG_B      | UNUSED   |

---

*iii. Describe the logic used to read and write your VGA registers*

The VGA registers are read and written via the Avalon bus. The Avalon bus is implemented using platform designer. The avl interface module consisted of various signals: AVL\_READ, AVL\_WRITE, AVL\_CS, AVL\_BYTE\_EN, AVL\_ADDR, AVL\_WRITEDATA, AVL\_READDATA. It also consists of a 50 MHz clock. Therefore, the logic for the reading and writing is placed within an always\_FF clock and triggered on the positive edge of the clock. The signal AVL\_CS is a 1 bit chip select signal used to indicate that the memory is selected. Therefore the registers can only be read from or written to if AVL\_CS is high. The AVL\_READ and AVL\_WRITE are used to indicate whether data is being read from or written to the registers, AVL\_ADDR is a 10-bit address used to reference one of 601 VGA registers, and AVL\_READDATA and AVL\_WRITEDATA are 32 bit values used to store the data read from or to be written to the registers. If AVL\_READ is high, the data from the VGA register is read based on the address specified by AVL\_ADDR and stored in AVL\_READDATA. AVL\_BYTE\_EN is a 4 bit value specifying which bytes of the 32-bit word are to be updated. If AVL\_WRITE is high, based on the BYTE\_EN data in AVL\_WRITEDATA is written into the VGA register specified by AVL\_ADDR.

```

always_ff @(posedge CLK) begin
    if (AVL_CS) begin
        if (AVL_READ)
            begin
                AVL_READDATA <= LOCAL_REG[AVL_ADDR];
            end
        else if(AVL_WRITE)
            begin
                case (AVL_BYTE_EN)
                    4'b1111: LOCAL_REG[AVL_ADDR]<=AVL_WRITEDATA[31:16];
                    4'b1100: LOCAL_REG[AVL_ADDR][31:16]<=AVL_WRITEDATA[31:16];
                    4'b0011: LOCAL_REG[AVL_ADDR][15:0]<=AVL_WRITEDATA[15:0];
                    4'b1000: LOCAL_REG[AVL_ADDR][31:24]<=AVL_WRITEDATA[31:24];
                    4'b0100: LOCAL_REG[AVL_ADDR][23:16]<=AVL_WRITEDATA[23:16];
                    4'b0010: LOCAL_REG[AVL_ADDR][15:8]<=AVL_WRITEDATA[15:8];
                    4'b0001: LOCAL_REG[AVL_ADDR][7:0]<=AVL_WRITEDATA[7:0];
                endcase
            end
    end
end

```

| byteenable[3:0] | Write Action               |
|-----------------|----------------------------|
| <b>1111</b>     | Write full 32-bits.        |
| <b>1100</b>     | Write the two upper bytes. |
| <b>0011</b>     | Write the two lower bytes. |
| <b>1000</b>     | Write byte 3 only.         |
| <b>0100</b>     | Write byte 2 only.         |
| <b>0010</b>     | Write byte 1 only.         |
| <b>0001</b>     | Write byte 0 only.         |

| Name              | Direction | Width     | Description                                                           |
|-------------------|-----------|-----------|-----------------------------------------------------------------------|
| <b>read</b>       | Input     | <b>1</b>  | High when a read operation is to be performed.                        |
| <b>write</b>      | Input     | <b>1</b>  | High when a write operation is to be performed.                       |
| <b>readdata</b>   | Output    | <b>32</b> | 32-bit data to be read.                                               |
| <b>writedata</b>  | Input     | <b>32</b> | 32-bit data to be written.                                            |
| <b>address</b>    | Input     | <b>10</b> | Address of the read or write operation.                               |
| <b>byteenable</b> | Input     | <b>4</b>  | 4-bit active high signal to identify which byte(s) are being written. |
| <b>chipselect</b> | Input     | <b>1</b>  | High during a read or write operation.                                |

iv. *Describe the algorithm used to draw the text characters from the VRAM and font ROM (specifically, describe the equations required to generate the correct addresses to index into the VRAM as well as the font ROM).*

The IP must determine the correct data to be printed based on the current DrawX and DrawY of the VGA. First, the cell from the 80\*30 grid currently being operated on must be determined. DrawX and DrawY which corresponds to the coordinates in the 640\*480 grid are first divided by 8 and 16 to obtain the coordinates in the 80\*30 grid. This is done by accessing bits DrawX[9:3] and DrawY[9:4]. Since the data is stored sequentially in registers it is indexed in a row major order. To calculate the byte address we first multiply the column coordinate by 80 and add the row coordinate to this (DrawX[9:3]+DrawY[9:4]\*80). Since each register corresponds to 4 bytes, we divide the byte address by 4 to get the final VRAM Address. Therefore the VRAM address corresponds to (DrawX[9:3]+DrawY[9:4]\*80)[11:2]. To determine which of the 4 bytes we are dealing with within the 32-bit register, we use the Byte address modded with 4 or (DrawX[9:3]+DrawY[9:4]\*80)[1:0]. Hence, using the calculated address, we receive 32 bits of

data from the register, and then use the 2 bit number to finally get the 8 bits of data for that specific cell. From the 8-bit number we use the bottom 7 bits which is the glyph code. The font rom consists of 128 sequentially stored characters, with each character taking up 16 rows. Within the 16 rows, to determine which row we are currently operating on we make use of DrawY%16 or the bottom 4 bits of DrawY. Therefore, to access the correct row within Font\_ROM we multiply the 7-bit code with 16 and then add the bottom 4 bits of DrawY. This returns an 8-bit value. To determine which of the 8 bits we are currently working with we use DrawX%8 or the bottom 3 bits of DrawX to finally receive a singular bit of 1 or 0.

```

always_comb begin
    ac = DrawX[9:3] + (80*DrawY[9:4]);
    c4 = LOCAL_REG[ac[11:2]]; //reads from register based on address specified.

    case (ac[1:0]) //determines which byte from the 32-bit word
        2'b00: charac = c4[7:0];
        2'b01: charac = c4[15:8];
        2'b10: charac = c4[23:16];
        2'b11: charac = c4[31:24];
    endcase

    font_addr = (charac[6:0]*16 + DrawY[3:0]);
    bitval = font_data[3'b111 - DrawX[2:0]]; //Pixel bit
    invert = charac[7]; // invert bit

end

```

---

v. *Describe your implementation of the inverse color bit, as well as the implementation of the control register.*

The control register stored 4-bit RGB values corresponding to the foreground and background colors. The invert bit was used to invert the color of the pixel. As per normal, if the pixel bit was a 0, the pixel color would be set to the background, and if it was 1, the pixel color would be set to the foreground. However, if the invert bit is a 1, it sets the color to foreground when the pixel bit is a 0, and background when the pixel bit is a 1. We take care of this logic by XORing the pixel bit with the invert bit. If the XOR yields a 1(invert = 0, pixel = 1 or invert = 1, pixel = 0), the RGB value is set to that of the foreground, and set to background otherwise. The foreground colors are set by accessing bits 24:21(red), 20:17(green), 16:13(blue) of register 600(Control register). Similarly the background color is determined by bits 12:9(red), 8:5(green), 4:1(blue)

| <b>Bit</b> | <b>31–25</b> | <b>24–21</b> | <b>20–17</b> | <b>16–13</b> | <b>12–9</b> | <b>8–5</b> | <b>4–1</b> | <b>0</b> |
|------------|--------------|--------------|--------------|--------------|-------------|------------|------------|----------|
| Function   | UNUSED       | FGD_R        | FGD_G        | FGD_B        | BKG_R       | BKG_G      | BKG_B      | UNUSED   |

```

always_ff @(posedge pixel_clk) begin
    if(!blank) begin
        red = 4'h0;
        green = 4'h0;
        blue = 4'h0;
    end

    else if (bitval^invert) begin
        red = LOCAL_REG[600][24:21];
        green = LOCAL_REG[600][20:17];
        blue = LOCAL_REG[600][16:13];
    end
    else begin
        red = LOCAL_REG[600][12:9];
        green = LOCAL_REG[600][8:5];
        blue = LOCAL_REG[600][4:1];
    end
end

```

---

### b. Week 2 (Color Text Display)

i. Describe the hardware changes you had to make to support the use of multi-color text. At the minimum you must describe:

*1. Modification of register-based VRAM to on-chip memory-based VRAM. How did your design share the limited on-chip memory ports?*

In week 1, data for each cell was stored within 600 registers. Each register stored data for 4 cells. However, data per cell for week 2, was expanded to 16 bits. This meant we would've required 1200 32-bit registers, which would've exhausted the FPGA logic units. For this reason, we use on-chip memory, with a true dual port implementation. Register implementation had multiple input and output ports making it easy to read and write, but onchip memory only had 4 read and write ports. Port A is used to manipulate AVL data while port B is used to access VRAM data. The addressability increases from 10 bits to 12 bits to account for the Palette and the larger VRAM. Bit 11 of the address is used to determine whether the VRAM (if 0) or Palette (if 1) is being accessed. The palette is implemented as 8 32-bit registers, and is read from and written to using logic similar to that used to read and write VGA registers from week 1. A mux is created with select bit as AVL\_ADDR[11] to set the AVL\_READDATA as either palletted data or data from VRAM.

*2. Corresponding modifications to the Platform Designer IP (e.g. Part Editor).*

The only modification in the platform designer was to change the addressability (size of address port of IP) from 10 bits to 12 bits to accommodate the additional VRAM and palette. In week 1, the VRAM consisted of 600 32-bit registers and an additional control register. Therefore, 10 bits was enough to address these registers. For week 2, since each cell now had 16-bits of data, only 2 cells could be stored per 32-bits. Therefore 2400 cells would require 1200 words. This would require 11 bits of addressability. However to incorporate the palette implementation we make use of 12 bits. After the VRAM is defined, unused space is left in the middle and the palette data is stored in memory such that the 11th bit is 1. This makes implementing the palette

much easier. Additionally, there is a delay in writing to memory since OCM is synchronous, which was not the case with registers.

| Word Address Range    | Byte Address Range               | Description                                                                                                                   |
|-----------------------|----------------------------------|-------------------------------------------------------------------------------------------------------------------------------|
| <b>0x000 - 0x4AF</b>  | <b>0x0000 0000 - 0x0000 12BF</b> | VRAM – 2 bytes per character, 2 characters per word. 80 column x 30 row. Data format is in raster order (one line at a time). |
| <b>0x4B0 - 0x7FF</b>  | <b>0x0000 12C0 - 0x0000 1FFF</b> | Unused but reserved by Platform Designer                                                                                      |
| <b>0x800 - 0x807</b>  | <b>0x0000 2000 - 0x0000 201F</b> | Palette- 8 words of 2 colors each, for 16-color palette                                                                       |
| <b>0x808 - 0xFFFF</b> | <b>0x0000 2020 - 0x0000 3FFF</b> | Unused but reserved by Platform Designer                                                                                      |

*3. Modified sprite drawing algorithm with the updated indexing equations from on-screen pixels to VRAM.*

Similar to week 1, the data to be printed is determined based on the current DrawX and DrawY of the VGA. Just like week 1, the byte address is first calculated as  $(\text{DrawX}[9:3] + \text{DrawY}[9:4]*80)$ . However, now each VRAM address corresponds to 2 cells instead of 4 and the byte address is divided by 2 instead. The new VRAM Address is therefore  $(\text{DrawX}[9:3] + \text{DrawY}[9:4]*80)[11:1]$ . This address is inputted to port b of memory. The corresponding 32-bits of data is read from the output of port b of memory. Modding the byte address with 2 determines whether the upper 16 bits or lower 16 bits are being dealt with. Therefore the lowest bit of the byte address  $(\text{DrawX}[9:3] + \text{DrawY}[9:4]*80)[0]$  is used to select between the upper 16 and lower 16 bits. Of these 16 bits, [14:8] represents the 7-bit glyph code, and bit 15 represents the invert bit. Accessing Font\_ROM is done similar to week 1 using DrawX, DrawY and the 7-bit glyph code.

*4. Additional modifications necessary to support multicolored text.*

The first step was to read and write data from and to the 8 32-bit palette registers. This logic was similar to that used to read and write VGA registers from week 1. The lowest 3 bits of AVL\_ADDR determined which of the 8 Palettes were being written to or read from. Additionally the palettes would only be read to or written from when AVL\_ADDR[11] was high.

```

always_ff @(posedge CLK) begin
    if (AVL_CS) begin
        if(AVL_WRITE && AVL_ADDR[11])
            begin
                case (AVL_BYTE_EN)
                    4'b1111: Palette[AVL_ADDR[2:0]]<=AVL_WRITEDATA;
                    4'b1100: Palette[AVL_ADDR[2:0]][31:16]<=AVL_WRITEDATA[31:16];
                    4'b0011: Palette[AVL_ADDR[2:0]][15:0]<=AVL_WRITEDATA[15:0];
                    4'b1000: Palette[AVL_ADDR[2:0]][31:24]<=AVL_WRITEDATA[31:24];
                    4'b0100: Palette[AVL_ADDR[2:0]][23:16]<=AVL_WRITEDATA[23:16];
                    4'b0010: Palette[AVL_ADDR[2:0]][15:8]<=AVL_WRITEDATA[15:8];
                    4'b0001: Palette[AVL_ADDR[2:0]][7:0]<=AVL_WRITEDATA[7:0];
                endcase
            end
        else if (AVL_READ && AVL_ADDR[11])
            begin
                palettesdata <= Palette[AVL_ADDR[2:0]];
            end
    end
end

```

The next step was to set the pixel RGB value, based on the color index, and whether the color is to be set to foreground or background. Each index corresponded to a 4-bit value. The top three bits determined which of the 8 palettes were being used. Since each palette contained two colors, the LSB of the index is used to determine the color. For the foreground, the palette index was determined by bits 7:5 of the 16 bits, with bit 4 determining which of the 2 colors within the palette is to be set. Similarly, for the background, the palette index is determined by bits 3:1, with bit 0 determining which of the 2 colors within the palette.

```

always_ff @(posedge pixel_clk) begin
    if(!blank) begin
        red = 4'h0;
        green = 4'h0;
        blue = 4'h0;
    end
    else if (bitval^invert) begin
        if(charac[4]) begin
            red = Palette[charac[7:5]][24:21];
            green = Palette[charac[7:5]][20:17];
            blue = Palette[charac[7:5]][16:13];
        end
        else begin
            red = Palette[charac[7:5]][12:9];
            green = Palette[charac[7:5]][8:5];
            blue = Palette[charac[7:5]][4:1];
        end
    end
    else begin
        if(charac[0]) begin
            red = Palette[charac[3:1]][24:21];
            green = Palette[charac[3:1]][20:17];
            blue = Palette[charac[3:1]][16:13];
        end
        else begin
            red = Palette[charac[3:1]][12:9];
            green = Palette[charac[3:1]][8:5];
            blue = Palette[charac[3:1]][4:1];
        end
    end
end

```

| <b>Bit</b> | <b>31</b> | <b>30-24</b> | <b>23-20</b> | <b>19-16</b> | <b>15</b> | <b>14-8</b> | <b>7-4</b> | <b>3-0</b> |
|------------|-----------|--------------|--------------|--------------|-----------|-------------|------------|------------|
| Function   | IV1       | CODE1        | FGD_IDX1     | BKG_IDX1     | IV0       | CODE0       | FGD_IDX0   | BKG_IDX0   |

| <b>Address</b> | <b>31-25</b> | <b>24-21</b> | <b>20-17</b> | <b>16-13</b> | <b>12-9</b> | <b>8-5</b> | <b>4-1</b> | <b>0</b> |
|----------------|--------------|--------------|--------------|--------------|-------------|------------|------------|----------|
| 0x800          | UNUSED       | C1_R         | C1_G         | C1_B         | C0_R        | C0_G       | C0_B       | UNUSED   |
| 0x801          | UNUSED       | C3_R         | C3_G_        | C3_B         | C2_R        | C2_G       | C2_B       | UNUSED   |
| ...            | ...          | ...          | ...          | ...          | ...         | ...        | ...        | ...      |
| 0x807          | UNUSED       | C15_R        | C15_G        | C15_B        | C14_R       | C14_G      | C14_B      | UNUSED   |

##### 5. Additional hardware/code to draw palettes colors

To draw palette colors, on the hardware side, we created 8 32-bit registers to store the palette data. These are written to, read from and implemented as explained above.

Additionally, we use c code to implement the palette functionality.

```
void textVGAColorScreenSaver()
{
    char color_string[80];
    int fg, bg, x, y;
    textVGAColorClr();
    for (int i = 0; i < 16; i++)
    {
        setColorPalette (i, colors[i].red, colors[i].green, colors[i].blue);
    }
    while (1)
    {
        fg = rand() % 16;
        bg = rand() % 16;
        while (fg == bg)
        {
            fg = rand() % 16;
            bg = rand() % 16;
        }
        sprintf(color_string, "Drawing %s text with %s background", colors[fg].name, colors[bg].name);
        x = rand() % (80-strlen(color_string));
        y = rand() % 30;
        textVGADrawColorText (color_string, x, y, bg, fg);
        usleep (100000);
    }
}
```

This function is the main function to be called. It first sets the color palette based off a predetermined list, and then enters an infinite loop, selecting a random position on the screen and setting the foreground and background to two different colors.

```

void setColorPalette (alt_u8 color, alt_u8 red, alt_u8 green, alt_u8 blue)
{
    int uppermask = color%2;
    alt_32 updatechar = vga_ctrl->palette[color/2]; //reads the 32 bit val
    if (uppermask) {
        updatechar &= 0xFE001FFF;
        updatechar |= (red << 21) | (green << 17) | (blue << 13);
    }
    else {
        updatechar &= 0xFFFFE001;
        updatechar |= (red << 9) | (green << 5) | (blue << 1);
    }
    vga_ctrl->palette[color/2] = updatechar;
}

```

This function is used to update the color palette by writing the 32 bit values in the corresponding memory location. Since there are 16 colors, the palette is determined and accessed by color/2. To determine, which of the 2 colors within the palette is being set, we use color%2. The 12 rgb bits are first set to 0 using bitmasking and then the new rgb values are written into memory by bit shifting and ORing.

```

void textVGADrawColorText(char* str, int x, int y, alt_u8 background, alt_u8 foreground)
{
    int i = 0;
    while (str[i]!=0)
    {
        vga_ctrl->VRAM[(y*COLUMNS + x + i) * 2] = foreground << 4 | background;
        vga_ctrl->VRAM[(y*COLUMNS + x + i) * 2 + 1] = str[i];
        i++;
    }
}

```

This function writes the character to be printed for each corresponding x and y coordinate

---

### 3. Block Diagram

- a. This diagram should represent the placement of all your modules in the top level. Please only include the top-level diagram and not the RTL view of every module.
- b. Note that depending on your layout of the registers inside your main module, the Quartus view may be illegible, in which case you should draw a block diagram using software. You may start from the provided materials (e.g. in IAMM), but you should fill in the specific signals between the modules and the inside subcomponents within each module.
- c. You should have block diagrams for both the Week 1 and Week 2 portions, a good setup is to show the common components (e.g. the SoC setup) first and then show diagrams for both the Week 1 and Week 2 VGA controller component.
- d. If your design has a state machine, you should include a State Diagram as well.



Figure: Diagram for the top level of both files: 7.1, 7.2.

The 2 weeks had the diagrams being very similar. The main difference was that the 601 registers in the week 1 translated into using VRAM and 8 palette registers instead. The main changes in the output on the VGA Monitor is due to the functions that we wrote in C code.

Week 1:



Week 2:



## 4. Module Descriptions

Module: vga\_text\_avl\_interface.sv

Inputs: CLK,RESET,AVL\_READ,AVL\_WRITE,AVL\_CS,[3:0] AVL\_BYTE\_EN,[11:0] AVL\_ADDR,[31:0] AVL\_WRITEDATA

Outputs: hs, vs, [31:0] AVL\_READDATA,[3:0] red, green, blue

Description: This module instantiates the VRAM, vga\_controller, font\_rom, creates the palette, and handles the reading and writing of the palette. Using DrawX and DrawY it accesses the VRAM to determine the data to be printed, and accordingly sets the pixel color

Purpose: This module handles the read/write operation from memory to determine the color to be set on the display.

---

Module: VGA\_controller.sv

Inputs: Clk, Reset

Outputs: hs, vs, pixel\_clk, blank, sync, [9:0] DrawX, DrawY

Description: This module generates the VGA signals by creating a 25MHz clock from the 50MHz clock given to using the always\_ff block. It helps keep track of the coordinates of the pixel (X, Y coordinates) and describes the horizontal and vertical sync signals. It handles the blanking logic which helps set the value of the black color on the VGA.

Purpose: This module simulates the VGA to determine the current coordinate to be drawn and handles the vertical, horizontal sync and blank to draw pixels.

---

Module: font\_rom.sv

Inputs: [10:0] addr

Outputs: [7:0] data

Description: This module defines the 128 glyphs stored in registers with each glyph represented as 16\*8 bits.

Purpose: This module when given an address, outputs the 8 bits of data corresponding to a row in the font\_data defined by that address

---

Module: ram.v

Inputs:[11:0] address\_a, [11:0] address\_b, [3:0] byteena\_a, clock, [31:0] data\_a, [31:0] data\_b, wren\_a, wren\_b

Outputs: [31:0] q\_a, [31:0] q\_b

Description: This module is wizard-generated which creates a true dual port on chip memory element.

Purpose: This module is used to create the on-chip memory to store the VRAM data.

---

Module: Lab7.sv

Inputs: MAX10\_CLK1\_50, [ 1: 0] KEY, [ 9: 0] SW, [ 9: 0] LEDR, [ 7: 0] HEX0, HEX1,HEX2,HEX3,HEX4,HEX5, DRAM\_CLK, DRAM\_CKE, DRAM\_ADDR,[ 1: 0] DRAM\_BA, [15: 0] DRAM\_DQ

Outputs: DRAM\_LDQM, DRAM\_UDQM, DRAM\_CS\_N, DRAM\_WE\_N, DRAM\_CAS\_N, DRAM\_RAS\_N, VGA\_HS,VGA\_VS, VGA\_R, VGA\_G, VGA\_B

Inout: [15: 0] ARDUINO\_IO, ARDUINO\_RESET\_N

Description: This module is the top-level entity for lab 7, in which the according HEX and Switch connections are made and the lab7\_soc is instantiated.

---

## Module: lab72\_soc.v

Only used components have been listed. Additional components are from lab 6.2

| Use                                 | Connections                                                                         | Name                    | Description                           | Export                                                                                                                                                                          | Clock               | Base               | End                | IRQ |
|-------------------------------------|-------------------------------------------------------------------------------------|-------------------------|---------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------------------|--------------------|--------------------|-----|
| <input checked="" type="checkbox"/> |    | <b>clk_0</b>            | Clock Source                          | <b>clk</b><br><b>reset</b>                                                                                                                                                      | <b>exported</b>     |                    |                    |     |
| <input checked="" type="checkbox"/> |    | <b>nios2_gen2_0</b>     | Nios II Processor                     | <i>Double-click to export</i><br>[clk]<br>[reset]<br>[clk]<br>[clk]<br>[clk]<br>[clk]<br>[clk]<br>[clk]<br>[clk]<br>[clk]<br>[clk]<br>[clk]<br>[clk]<br>[clk]<br>[clk]<br>[clk] | <b>clk_0</b>        |                    |                    |     |
| <input checked="" type="checkbox"/> |    | <b>onchip_memory2_0</b> | On-Chip Memory (RAM or ROM) Intel ... | <i>Double-click to export</i><br>[clk1]<br>[clk1]                                                                                                                               | <b>clk_0</b>        | <b>IRQ 0</b>       | <b>IRQ 31</b>      |     |
| <input checked="" type="checkbox"/> |    | <b>sram</b>             | SDRAM Controller Intel FPGA IP        | <i>Double-click to export</i><br>[clk]<br>[reset]<br>[s1]<br>[reset1]                                                                                                           | <b>sram_pll_...</b> | <b>0x0800_6800</b> | <b>0x0800_6fff</b> |     |
| <input checked="" type="checkbox"/> |    | <b>sram_pll</b>         | ALTPPLL Intel FPGA IP                 | <i>Double-click to export</i><br>[ndlk_interface]<br>[ndlk_interface_reset]<br>[ndlk_slave]<br>[c0]<br>[c1]                                                                     | <b>clk_0</b>        | <b>0x0400_0000</b> | <b>0x07ff_ffff</b> |     |
| <input checked="" type="checkbox"/> |   | <b>sysid_qsys_0</b>     | System ID Peripheral Intel FPGA IP    | <i>Double-click to export</i><br>[clk]<br>[reset]<br>[control_slave]                                                                                                            | <b>clk_0</b>        |                    |                    |     |
| <input checked="" type="checkbox"/> |  | <b>key_rst</b>          | PIO (Parallel I/O) Intel FPGA IP      | <i>Double-click to export</i><br>[clk]<br>[reset]<br>[s1]                                                                                                                       | <b>clk_0</b>        | <b>0x0800_71f0</b> | <b>0x0800_71f7</b> |     |
| <input checked="" type="checkbox"/> |  | <b>key_acc</b>          | PIO (Parallel I/O) Intel FPGA IP      | <i>Double-click to export</i><br>[clk]<br>[reset]<br>[s1]                                                                                                                       | <b>clk_0</b>        | <b>0x0800_71c0</b> | <b>0x0800_71cf</b> |     |
|                                     |                                                                                     |                         | Conduit                               | <i>Double-click to export</i><br>[external_connection]                                                                                                                          | <b>key_rst_wire</b> |                    |                    |     |
|                                     |                                                                                     |                         | Conduit                               | <i>Double-click to export</i><br>[external_connection]                                                                                                                          | <b>key_acc_wire</b> | <b>0x0800_71b0</b> | <b>0x0800_71bf</b> |     |

|                                     |                                                                                     |                                 |                                                                                                        |                                                                                                                                                     |                                                  |                                                 |             |                                                                                       |
|-------------------------------------|-------------------------------------------------------------------------------------|---------------------------------|--------------------------------------------------------------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------|--------------------------------------------------|-------------------------------------------------|-------------|---------------------------------------------------------------------------------------|
| <input checked="" type="checkbox"/> |   | <b>key_acc</b>                  | PIO (Parallel I/O) Intel FPGA IP<br>Clock Input<br>Reset Input<br>s1<br>external_connection            | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><b>key_acc_wire</b>                              | <b>clk_0</b><br>[clk]<br>[clk]                   | <input checked="" type="checkbox"/> 0x0800_71b0 | 0x0800_71bf |   |
| <input checked="" type="checkbox"/> |  | <b>jtag_uart_0</b>              | JTAG UART Intel FPGA IP<br>Clock Input<br>Reset Input<br>avalon_jtag_slave<br>irq                      | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i>                    | <b>clk_0</b><br>[clk]<br>[clk]<br>[clk]          | <input checked="" type="checkbox"/> 0x0800_71f8 | 0x0800_71ff |  |
| <input checked="" type="checkbox"/> |   | <b>keycode</b>                  | PIO (Parallel I/O) Intel FPGA IP<br>Clock Input<br>Reset Input<br>s1<br>external_connection            | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><b>keycode</b>                                   | <b>clk_0</b><br>[clk]<br>[clk]                   | <input checked="" type="checkbox"/> 0x0800_71a0 | 0x0800_71af |   |
| <input checked="" type="checkbox"/> |  | <b>usb_irq</b>                  | PIO (Parallel I/O) Intel FPGA IP<br>Clock Input<br>Reset Input<br>s1<br>external_connection            | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><b>usb_irq</b>                                   | <b>clk_0</b><br>[clk]<br>[clk]                   | <input checked="" type="checkbox"/> 0x0800_7190 | 0x0800_719f |  |
| <input checked="" type="checkbox"/> |   | <b>usb_gpx</b>                  | PIO (Parallel I/O) Intel FPGA IP<br>Clock Input<br>Reset Input<br>s1<br>external_connection            | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><b>usb_gpx</b>                                   | <b>clk_0</b><br>[clk]<br>[clk]                   | <input checked="" type="checkbox"/> 0x0800_7180 | 0x0800_718f |   |
| <input checked="" type="checkbox"/> |  | <b>usb_RST</b>                  | PIO (Parallel I/O) Intel FPGA IP<br>Clock Input<br>Reset Input<br>s1<br>external_connection            | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><b>usb_RST</b>                                   | <b>clk_0</b><br>[clk]<br>[clk]                   | <input checked="" type="checkbox"/> 0x0800_7170 | 0x0800_717f |  |
| <input checked="" type="checkbox"/> |   | <b>hex_digits_pio</b>           | PIO (Parallel I/O) Intel FPGA IP<br>Clock Input<br>Reset Input<br>s1<br>external_connection            | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><b>hex_digits</b>                                | <b>clk_0</b><br>[clk]<br>[clk]                   | <input checked="" type="checkbox"/> 0x0800_7160 | 0x0800_716f |   |
| <input checked="" type="checkbox"/> |  | <b>leds_pio</b>                 | PIO (Parallel I/O) Intel FPGA IP<br>Clock Input<br>Reset Input<br>s1<br>external_connection            | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><b>leds</b>                                      | <b>clk_0</b><br>[clk]<br>[clk]                   | <input checked="" type="checkbox"/> 0x0800_7150 | 0x0800_715f |  |
| <input checked="" type="checkbox"/> |   | <b>key</b>                      | PIO (Parallel I/O) Intel FPGA IP<br>Clock Input                                                        | <i>Double-click to export</i>                                                                                                                       | <b>clk_0</b>                                     |                                                 |             |                                                                                       |
| <input checked="" type="checkbox"/> |  | <b>key</b>                      | PIO (Parallel I/O) Intel FPGA IP<br>Clock Input<br>Reset Input<br>s1<br>external_connection            | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><b>key_external_connection</b>                   | <b>clk_0</b><br>[clk]<br>[clk]                   | <input checked="" type="checkbox"/> 0x0800_7140 | 0x0800_714f |  |
| <input checked="" type="checkbox"/> |   | <b>timer_0</b>                  | Interval Timer Intel FPGA IP<br>Clock Input<br>Reset Input<br>s1<br>irq                                | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i>                    | <b>clk_0</b><br>[clk]<br>[clk]<br>[clk]          | <input checked="" type="checkbox"/> 0x0800_7040 | 0x0800_707f |   |
| <input checked="" type="checkbox"/> |  | <b>spi_0</b>                    | SPI (3 Wire Serial) Intel FPGA IP<br>Clock Input<br>Reset Input<br>spi_control_port<br>irq<br>external | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><b>spi0</b>     | <b>clk_0</b><br>[clk]<br>[clk]<br>[clk]<br>[clk] | <input checked="" type="checkbox"/> 0x0800_70a0 | 0x0800_70bf |  |
| <input checked="" type="checkbox"/> |   | <b>VGA_text_mode_controller</b> | VGA Text Mode Controller<br>CLK<br>avl_mm_slave<br>VGA_port<br>RESET                                   | <i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><i>Double-click to export</i><br><b>vga_port</b> | <b>clk_0</b><br>[CLK]<br>[CLK]<br>[CLK]<br>[CLK] | <input checked="" type="checkbox"/> 0x0800_0000 | 0x0800_3fff |                                                                                       |

| Component                   | Name                       | Description                                                                                                                                                                                                                                            |
|-----------------------------|----------------------------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Nios-II Processor           | nios2_gen2_0               | This is our 32-bit CPU - the primary controller that is controlled by C, a high-level programming language. It is used to control the peripherals that primarily handle data and do not require fast processing times, and only need to transmit data. |
| SDRAM                       | sdram                      | The SDRAM (Synchronous Dynamic RAM) is used because of the limited availability of On-Chip memory.                                                                                                                                                     |
| Clock                       | clk_0                      | This module acts as the 50MHz clock from the FPGA which outputs the signal clk which can be used by other modules                                                                                                                                      |
| SDRAM PLL                   | sdram_pll                  | The SDRAM is not on the FPGA and so needs a synchronous clock which is used in this module.                                                                                                                                                            |
| Timer                       | Timer_0                    | This is used for the NIOS II to keep track of elapsed time, by sending a signal with frequency 1000 Hz                                                                                                                                                 |
| SPI (W2)                    | spi_0                      | The SPI (System Peripheral Interface) lets us interact with USB Peripherals like the Keyboard, and mouse if necessary. It handles transfer of data between the NIOSII and the slave peripheral via MISO and MOSI.                                      |
| JTAG UART (W2)              | jtag_uart                  | This allows character movement between the Computer and the FPGA, enabling transferring of text which avoids the slowing down of the CPU for tasks that do not require high processing powers.                                                         |
| VGA Text Mode Controller IP | VGA_text_mode_controller_0 | This IP is used to implement the Avalon Memory Mapped bus and control the VGA via hs, vs, RGB values                                                                                                                                                   |

---

## 5. Design Resources and Statistics

### *Lab 7.1*

|               |           |
|---------------|-----------|
| LUT           | 35703     |
| DSP           | 0         |
| Memory (BRAM) | 11264     |
| Flip-Flop     | 21698     |
| Frequency     | 67.21 MHz |
| Static Power  | 97.24 mW  |
| Dynamic Power | 249.24 mW |
| Total Power   | 368.74 mW |

---

### *Lab 7.2*

|               |           |
|---------------|-----------|
| LUT           | 6034      |
| DSP           | 0         |
| Memory (BRAM) | 49664     |
| Flip-Flop     | 2870      |
| Frequency     | 73.1 MHz  |
| Static Power  | 96.6 mW   |
| Dynamic Power | 79.95 mW  |
| Total Power   | 196.32 mW |

---

In week 1 we implemented the VRAM using registers, while in week 2 we used OCM. This reflects in the statistics. Comparing the LUTs and flip-flops used, it is evident that week 1 required significantly more elements. This makes sense since there were 601 32-bit registers, and each bit is stored in a flip-flop. The BRAM for week 2 is significantly higher than week 1, since week 2 made use of on-chip memory and also required 1200 32-bit storage elements.

Since, week 2 requires significantly less LUTs and Flip-flops, the total power is also significantly lower. Based on statistics, it appears week 2 implementation is a lot more efficient. The only trade off is lack of input output ports from each of the registers, and additional wait states to write data into the synchronous onchip memory.

---

## **6. Conclusion**

In conclusion, we have successfully created a simple text mode graphics controller interfaced with the Avalon Memory-Mapped bus to display glyphs. This lab has solidified our understanding with working with the NIOS II SoC. It gives us a starting place for future graphics rendering required for the final project. We have also implemented the Avalon Memory mapped Bus which is useful for data access and manipulation. This along with lab 6 gives us a detailed understanding to use peripherals such as keyboard, VGA monitor.

---