

# ECGR 3183: Computer Organization - Single Cycle Processor

Mikey Neal

May 5, 2021

## Contents

|                                                     |           |
|-----------------------------------------------------|-----------|
| <b>CPU</b>                                          | <b>3</b>  |
| VHDL Code . . . . .                                 | 3         |
| B-Type Instructions . . . . .                       | 7         |
| B . . . . .                                         | 7         |
| CB-Type Instructions . . . . .                      | 8         |
| CBZ . . . . .                                       | 8         |
| D-Type Instructions . . . . .                       | 8         |
| LDUR . . . . .                                      | 8         |
| STUR . . . . .                                      | 9         |
| R-Type Instructions . . . . .                       | 9         |
| ADD . . . . .                                       | 9         |
| AND . . . . .                                       | 10        |
| ORR . . . . .                                       | 10        |
| SUB . . . . .                                       | 11        |
| Full Output . . . . .                               | 11        |
| <b>CPU Units from the Control Units Project</b>     | <b>12</b> |
| Main Control Unit . . . . .                         | 12        |
| VHDL Code . . . . .                                 | 12        |
| Explanation . . . . .                               | 13        |
| Output . . . . .                                    | 13        |
| ALU Control Unit . . . . .                          | 14        |
| VHDL Code . . . . .                                 | 14        |
| Explanation . . . . .                               | 14        |
| Output . . . . .                                    | 14        |
| <b>CPU Units from the Datapath Elements Project</b> | <b>15</b> |
| Program Counter . . . . .                           | 15        |
| VHDL Code . . . . .                                 | 15        |
| Explanation . . . . .                               | 15        |
| Instruction Memory . . . . .                        | 16        |
| VHDL Code . . . . .                                 | 16        |
| Explanation . . . . .                               | 17        |
| Output . . . . .                                    | 18        |
| Register File . . . . .                             | 19        |
| VHDL Code . . . . .                                 | 19        |
| Explanation . . . . .                               | 19        |
| Output . . . . .                                    | 19        |
| ALU . . . . .                                       | 20        |
| VHDL Code . . . . .                                 | 20        |
| Explanation . . . . .                               | 21        |
| Output . . . . .                                    | 21        |
| Data Memory . . . . .                               | 22        |
| VHDL Code . . . . .                                 | 22        |
| Explanation . . . . .                               | 23        |
| Output . . . . .                                    | 24        |
| Sign Extend . . . . .                               | 25        |
| VHDL Code . . . . .                                 | 25        |
| Explanation . . . . .                               | 25        |
| Output . . . . .                                    | 25        |

# CPU

## VHDL Code

```
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
use IEEE.NUMERIC_STD.ALL;

entity CPU is
    Port ( Clk : in STD_LOGIC);
end CPU;

architecture Behavioral of CPU is
    component INSTRUCTION_RAM is
        port(    write_enable : in std_logic;
                 read_enable : in std_logic;
                     address : in std_logic_vector(63 downto 0);
                     data_in : in std_logic_vector(31 downto 0);
                     data_out : out std_logic_vector(31 downto 0));
    end component;
    component ALU_Control is
        port(    ALU_OP : in std_logic_vector(1 downto 0);
                 UPPER_11_OPCODE : in std_logic_vector(10 downto 0);
                 ALU_Code : out std_logic_vector(3 downto 0));
    end component;
    component ALUv2 is
        Port ( A_in : in STD_LOGIC_VECTOR (63 downto 0);
                B_in : in STD_LOGIC_VECTOR (63 downto 0);
                ALU_Out : out STD_LOGIC_VECTOR (63 downto 0);
                ALU_Flags : out STD_LOGIC_VECTOR (3 downto 0);
                OPCODE : in STD_LOGIC_VECTOR (3 downto 0));
    end component;
    component Data_RAM is
        port(    write_enable : in std_logic;
                 read_enable : in std_logic;
                     address : in std_logic_vector(63 downto 0);
                     data_in : in std_logic_vector(63 downto 0);
                     data_out : out std_logic_vector(63 downto 0));
    end component;
    component REGISTER_FILE
        port(    write_enable : in std_logic;
                 address_in : in std_logic_vector(4 downto 0);
                 address_out_1 : in std_logic_vector(4 downto 0);
                 address_out_2 : in std_logic_vector(4 downto 0);
                 data_in : in std_logic_vector(63 downto 0);
                 data_out_1 : out std_logic_vector(63 downto 0);
                 data_out_2 : out std_logic_vector(63 downto 0));
    end component;
    component Program_Counter is
        Port ( Clock : in STD_LOGIC;
                 PC_INPUT : in STD_LOGIC_VECTOR (63 downto 0);
                 PC_OUTPUT : out STD_LOGIC_VECTOR (63 downto 0));
    end component;
```

```
component Main_Control is
    port(  UPPER_11_OPCODE : in std_logic_vector(10 downto 0);
           Reg2Loc : out std_logic;
           ALUSrc : out std_logic;
           MemtoReg : out std_logic;
           RegWrite : out std_logic;
           MemRead : out std_logic;
           MemWrite : out std_logic;
           Branch : out std_logic;
           UncondBranch : out std_logic;
           ALU_OP : out std_logic_vector(1 downto 0));
end component;
component Sign_Extend is
    Port ( INST_In : in STD_LOGIC_VECTOR (31 downto 0);
           IMM_Out : out STD_LOGIC_VECTOR (63 downto 0));
end component;
signal Instruction_Memory_Write_Enable : std_logic;
signal Instruction_Memory_Read_Enable : std_logic;
signal Instruction_Memory_Address : std_logic_vector(63 downto 0);
signal Instruction_Memory_Data_Input : std_logic_vector(31 downto 0);
signal Instruction_Memory_Data_Output : std_logic_vector(31 downto 0);
signal ALU_Control_OPCODE_Input : std_logic_vector(10 downto 0);
signal ALU_Control_ALU_OPCODE : std_logic_vector(3 downto 0);
signal ALU_Control_ALU_ID : std_logic_vector(1 downto 0);
signal ALU_Input_A : STD_LOGIC_VECTOR (63 downto 0);
signal ALU_Input_B : STD_LOGIC_VECTOR (63 downto 0);
signal ALU_Output : STD_LOGIC_VECTOR (63 downto 0);
signal ALU_Flags : STD_LOGIC_VECTOR (3 downto 0);
signal ALU_OPCODE_Input : STD_LOGIC_VECTOR (3 downto 0);
signal Data_Memory_Write_Enable : std_logic;
signal Data_Memory_Read_Enable : std_logic;
signal Data_Memory_Address : std_logic_vector(63 downto 0);
signal Data_Memory_Data_In : std_logic_vector(63 downto 0);
signal Data_Memory_Data_Out : std_logic_vector(63 downto 0);
signal Register_File_Write_Enable : std_logic;
signal Register_File_Output_1_Select : std_logic_vector(4 downto 0);
signal Register_File_Output_2_Select : std_logic_vector(4 downto 0);
signal Register_File_Input_Select : std_logic_vector(4 downto 0);
signal Register_File_Output_1 : std_logic_vector(63 downto 0);
signal Register_File_Output_2 : std_logic_vector(63 downto 0);
signal Register_File_Input : std_logic_vector(63 downto 0);
signal Program_Counter_Clock : STD_LOGIC;
signal Program_Counter_Input : STD_LOGIC_VECTOR(63 downto 0);
signal Program_Counter_Output : STD_LOGIC_VECTOR(63 downto 0);
signal Main_Control_OPCODE_Input : std_logic_vector(10 downto 0);
signal Reg2Loc : std_logic;
signal ALUSrc : std_logic;
signal MemtoReg : std_logic;
signal RegWrite : std_logic;
signal MemRead : std_logic;
signal MemWrite : std_logic;
signal Branch : std_logic;
signal UncondBranch : std_logic;
```

```

signal Main_Control_ALU_ID : std_logic_vector(1 downto 0);
signal Sign_Extend_OPCODE_Input : STD_LOGIC_VECTOR (31 downto 0);
signal Sign_Extend_Immediate_Output : STD_LOGIC_VECTOR (63 downto 0);
begin
  uut : Data_RAM port map(
    write_enable      => Data_Memory_Write_Enable,
    read_enable       => Data_Memory_Read_Enable,
    address          => Data_Memory_Address,
    data_in           => Data_Memory_Data_In,
    data_out          => Data_Memory_Data_Out);
  uu2 : INSTRUCTION_RAM port map(
    write_enable      => Instruction_Memory_Write_Enable,
    read_enable       => Instruction_Memory_Read_Enable,
    address          => Instruction_Memory_Address,
    data_in           => Instruction_Memory_Data_Input,
    data_out          => Instruction_Memory_Data_Output);
  uu3 : ALU_Control port map(
    UPPER_11_OPCODE  => ALU_Control_OPCODE_Input,
    ALU_Code          => ALU_Control_ALU_OPCODE,
    ALU_OP            => ALU_Control_ALU_ID);
  uu4 : ALUV2 Port Map (
    A_in              => ALU_Input_A,
    B_in              => ALU_Input_B,
    ALU_Out           => ALU_Output,
    ALU_Flags         => ALU_Flags,
    OPCODE            => ALU_OPCODE_Input);
  uu5 : Main_Control port map(
    UPPER_11_OPCODE  => Main_Control_OPCODE_Input,
    Reg2Loc           => Reg2Loc,
    ALUSrc            => ALUSrc,
    MemtoReg          => MemtoReg,
    RegWrite          => RegWrite,
    MemRead           => MemRead,
    MemWrite          => MemWrite,
    Branch            => Branch,
    UncondBranch     => UncondBranch,
    ALU_OP            => Main_Control_ALU_ID);
  uu6 : REGISTER_FILE port map(
    data_out_1        => Register_File_Output_1,
    data_out_2        => Register_File_Output_2,
    data_in           => Register_File_Input,
    write_enable      => Register_File_Write_Enable,
    address_out_1     => Register_File_Output_1_Select,
    address_out_2     => Register_File_Output_2_Select,
    address_in        => Register_File_Input_Select);
  uu7: Sign_Extend Port Map (
    INST_In           => Sign_Extend_OPCODE_Input,
    IMM_Out           => Sign_Extend_Immediate_Output);
  uu8 : Program_Counter port map(
    Clock             => Program_Counter_Clock,
    PC_INPUT          => Program_Counter_Input,
    PC_OUTPUT         => Program_Counter_Output);

```

```

Instruction_Memory_Read_Enable  <= '1';
Instruction_Memory_Write_Enable <= '0';
Instruction_Memory_Data_Input    <= std_logic_vector(TO_UNSIGNED(0,32));

process
    variable First_Run : std_logic := '1';
begin
    if (First_Run = '1') then
        Program_Counter_Input <= std_logic_vector(TO_UNSIGNED(0,64));
    elsif (((ALU_Flags(2) = '1') AND (Branch = '1'))
        or (UncondBranch = '1')) then
        Program_Counter_Input(63 downto 2) <=
            std_logic_vector(unsigned(Program_Counter_Output(63 downto 2)) +
            unsigned(Sign_Extend_Immediate_Output(61 downto 0)));
        Program_Counter_Input( 1 downto 0) <=
            Program_Counter_Output( 1 downto 0);
    else
        Program_Counter_Input(63 downto 0) <=
            std_logic_vector(unsigned(Program_Counter_Output) + 4);
    end if;
    Program_Counter_Clock <= clk;
    wait for 1ps;

    Instruction_Memory_Address <= Program_Counter_Output;
    wait for 1ps;

    if (Program_Counter_Output = std_logic_vector(TO_UNSIGNED(0,64))) then
        First_Run := '0';
    end if;

    Main_Control_OPCODE_Input <= Instruction_Memory_Data_Output(31 downto 21);
    Sign_Extend_OPCODE_Input <= Instruction_Memory_Data_Output(31 downto 0);
    ALU_Control_OPCODE_Input <= Instruction_Memory_Data_Output(31 downto 21);
    wait for 1ps;

    ALU_Control_ALU_ID          <= Main_Control_ALU_ID;
    Register_File_Write_Enable   <= RegWrite;
    Register_File_Input_Select   <= Instruction_Memory_Data_Output(4 downto 0);
    Register_File_Output_1_Select <= Instruction_Memory_Data_Output(9 downto 5);

    if (Reg2Loc = '1') then
        Register_File_Output_2_Select <=
            Instruction_Memory_Data_Output( 4 downto 0);
    else
        Register_File_Output_2_Select <=
            Instruction_Memory_Data_Output(20 downto 16);
    end if;
    wait for 1ps;

    ALU_OPCODE_Input <= ALU_Control_ALU_OPCODE;
    ALU_Input_A      <= Register_File_Output_1;
    if (ALUSrc = '1') then
        ALU_Input_B  <= Sign_Extend_Immediate_Output;

```

```

        else
            ALU_Input_B  <= Register_File_Output_2;
        end if;
        wait for 1ps;

        Data_Memory_Data_In      <= Register_File_Output_2;
        Data_Memory_Address       <= ALU_Output;
        Data_Memory_Write_Enable  <= MemWrite;
        Data_Memory_Read_Enable   <= MemRead;
        wait for 1ps;

        if (MemtoReg = '1') then
            Register_File_Input <= Data_Memory_Data_Out;
        else
            Register_File_Input <= ALU_Output;
        end if;
        wait for 1ps;
    end process;
end Behavioral;

```

## B-Type Instructions

B

Using the Program Counter, the Sign Extend Immediate Output, and the Unconditional Branch Flag, the “B” instruction is working correctly.



Figure 1: B #2

## CB-Type Instructions

### CBZ

Using the Program Counter, the Sign Extend Immediate Output, the ALU Flags (index 2 is the zero flag), and the Branch Flag, the “CBZ” instruction is working correctly.



Figure 2: CBZ X1, #2

## D-Type Instructions

### LDUR

Using the Data Memory, the Sign Extend Immediate Output, and the Main Control Flags, the “LDUR” instruction is working correctly.



Figure 3: LDUR X2, [X10, #0]

## STUR

Using the Data Memory, the Sign Extend Immediate Output, and the Main Control Flags, the “STUR” instruction is working correctly.



Figure 4: STUR X4, [X7, #8]

## R-Type Instructions

### ADD

Using the ALU and the Registers, the “ADD” instruction is working correctly.



Figure 5: ADD X5, X3, X2

AND

Using the ALU and the Registers, the “AND” instruction is working correctly.



**Figure 6:** AND X7, X2, X3

ORR

Using the ALU and the Registers, the “ORR” instruction is working correctly.



**Figure 7: ORR X6, X2, X3**

## SUB

Using the ALU and the Registers, the “SUB” instruction is working correctly.



Figure 8: SUB X4, X3, X2

## Full Output



Figure 9: Graph of the Full Output Using the Program in Instruction Memory

## CPU Units from the Control Units Project

### Main Control Unit

#### VHDL Code

```

library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
use IEEE.NUMERIC_STD.ALL;

entity Main_Control is
    port(   UPPER_11_OPCODE : in std_logic_vector(10 downto 0);
            Reg2Loc : out std_logic;
            ALUSrc : out std_logic;
            MemtoReg : out std_logic;
            RegWrite : out std_logic;
            MemRead : out std_logic;
            MemWrite : out std_logic;
            Branch : out std_logic;
            UncondBranch : out std_logic;
            ALU_OP : out std_logic_vector(1 downto 0));
end Main_Control;

architecture Behavioral of Main_Control is
begin
    process (UPPER_11_OPCODE)
    begin
        if (UPPER_11_OPCODE(10 downto 5) = "000101")
            or (UPPER_11_OPCODE(10 downto 4) = "1011010") then -- CBNZ, CBZ, or B
            Reg2Loc      <= '1';
            ALUSrc       <= '0';
            --MemtoReg    <= '1';
            RegWrite     <= '0';
            MemRead      <= '0';
            MemWrite     <= '0';
            Branch       <= '1';
            UncondBranch <= NOT(UPPER_11_OPCODE(10));
            ALU_OP        <= "01";
        elsif (UPPER_11_OPCODE(10 downto 0) = "11111000000") then -- STUR
            Reg2Loc      <= '1';
            ALUSrc       <= '1';
            --MemtoReg    <= '1';
            RegWrite     <= '0';
            MemRead      <= '0';
            MemWrite     <= '1';
            Branch       <= '0';
            UncondBranch <= '0';
            ALU_OP        <= "00";
        elsif (UPPER_11_OPCODE(10 downto 0) = "11111000010") then -- LDUR
            --Reg2Loc     <= '1';
            ALUSrc       <= '1';
            MemtoReg     <= '1';
            RegWrite     <= '1';
            MemRead      <= '1';
            MemWrite     <= '0';
        end if;
    end process;
end Behavioral;

```

```

        Branch      <= '0';
        UncondBranch <= '0';
        ALU_OP       <= "00";
    else
        Reg2Loc      <= '0';
        ALUSrc       <= '0';
        MemtoReg     <= '0';
        RegWrite     <= '1';
        MemRead      <= '0';
        MemWrite     <= '0';
        Branch       <= '0';
        UncondBranch <= '0';
        ALU_OP       <= "10";
    end if;
end process;
end Behavioral;

```

## Explanation

The Main Control Unit takes in the upper 11 bits of an instruction (denoted as “UPPER\_11\_OPCODE”) and outputs control lines for the rest of the CPU (denoted as “Reg2Loc”, “ALUSrc”, “MemtoReg”, “RegWrite”, “MemRead”, “MemWrite”, “Branch”, “UncondBranch”, and “ALU\_OP”).

## Output



Figure 10: Graph of the output for the Main Control Unit using the Instruction Memory

## ALU Control Unit

### VHDL Code

```

library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
use IEEE.NUMERIC_STD.ALL;

entity ALU_Control is
    port(   ALU_OP : in std_logic_vector(1 downto 0);
            UPPER_11_OPCODE : in std_logic_vector(10 downto 0);
            ALU_Code : out std_logic_vector(3 downto 0));
end ALU_Control;

architecture Behavioral of ALU_Control is
begin
    process (UPPER_11_OPCODE, ALU_OP)
    begin
        if      (UPPER_11_OPCODE(10 downto 0) = "10001010000") then
            ALU_Code <= "0000"; -- AND
        elsif (UPPER_11_OPCODE(10 downto 0) = "10101010000") then
            ALU_Code <= "0001"; -- ORR
        elsif (UPPER_11_OPCODE(10 downto 0) = "11111000000")
            or (UPPER_11_OPCODE(10 downto 0) = "11111000010")
            or (UPPER_11_OPCODE(10 downto 0) = "10001011000") then
            ALU_Code <= "0010"; -- STUR, LDUR, or ADD
        elsif (UPPER_11_OPCODE(10 downto 0) = "11001011000") then
            ALU_Code <= "0110"; -- SUB
        elsif (UPPER_11_OPCODE(10 downto 5) = "000101")
            or (UPPER_11_OPCODE(10 downto 4) = "1011010") then
            ALU_Code <= "0111"; -- B, CBZ, or CBNZ
        end if;
    end process;
end Behavioral;

```

### Explanation

The ALU Control Unit takes in the upper 11 bits of an instruction (denoted as “UPPER\_11\_OPCODE”) and outputs the ALU type Operation Code (denoted as “ALU\_Code”). It also takes in the “ALU\_OP” from the main control, but it is unused as the “UPPER\_11\_OPCODE” can more easily be used to determine the instruction being used.

### Output



Figure 11: Graph of the output for the ALU Control Unit using the Instruction Memory

## CPU Units from the Datapath Elements Project

### Program Counter

#### VHDL Code

```
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
use IEEE.NUMERIC_STD.ALL;

entity Program_Counter is
    Port ( Clock : in STD_LOGIC;
            PC_INPUT : in STD_LOGIC_VECTOR (63 downto 0);
            PC_OUTPUT : out STD_LOGIC_VECTOR (63 downto 0));
end Program_Counter;

architecture Behavioral of Program_Counter is

begin
    process (Clock)
    begin
        if rising_edge(Clock) then
            PC_OUTPUT <= PC_INPUT;
        end if;
    end process;
end Behavioral;
```

#### Explanation

On the rising edge of the clock (denoted as “Clock”), the output (denoted as “PC\_OUTPUT”) will be set equal to the current input (denoted as “PC\_INPUT”).

## Instruction Memory

### VHDL Code

```
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
use IEEE.NUMERIC_STD.ALL;

entity INSTUCTION_RAM is
    port(write_enable, read_enable : in std_logic;
         address : in std_logic_vector(63 downto 0);
         data_in : in std_logic_vector(31 downto 0);
         data_out : out std_logic_vector(31 downto 0));
end INSTUCTION_RAM;

architecture Behavioral of INSTUCTION_RAM is

type memory is array(0 to 255) of std_logic_vector(7 downto 0);
signal ram : memory := (0 => "01000010",
                        1 => "00000001",
                        2 => "01000000",
                        3 => "11111000",
                        4 => "01000011",
                        5 => "10000001",
                        6 => "01000000",
                        7 => "11111000",
                        8 => "01100100",
                        9 => "00000000",
                        10 => "00000010",
                        11 => "11001011",
                        12 => "01100101",
                        13 => "00000000",
                        14 => "00000010",
                        15 => "10001011",
                        16 => "01000001",
                        17 => "00000000",
                        18 => "00000000",
                        19 => "10110100",
                        20 => "01000000",
                        21 => "00000000",
                        22 => "00000000",
                        23 => "10110100",
                        24 => "01000010",
                        25 => "00000001",
                        26 => "01000000",
                        27 => "11111000",
                        28 => "01000110",
                        29 => "00000000",
                        30 => "00000011",
                        31 => "10101010",
                        32 => "01000111",
                        33 => "00000000",
                        34 => "00000011",
```

```

35 => "10001010",
36 => "11100100",
37 => "10000000",
38 => "00000000",
39 => "11111000",
40 => "00000010",
41 => "00000000",
42 => "00000000",
43 => "00010100",
44 => "01000011",
45 => "10000001",
46 => "01000000",
47 => "11111000",
48 => "00001000",
49 => "00000000",
50 => "00000001",
51 => "10001011",
56 => "11110010",
57 => "11111111",
58 => "11111111",
59 => "00010111",
others => "00000000");

begin
    dw : process(write_enable, address, data_in) is
        begin
            if (write_enable = '1') and (unsigned(address) < 253) then
                ram(to_integer(unsigned(address)) + 0) <= data_in( 7 downto 0);
                ram(to_integer(unsigned(address)) + 1) <= data_in(15 downto 8);
                ram(to_integer(unsigned(address)) + 2) <= data_in(23 downto 16);
                ram(to_integer(unsigned(address)) + 3) <= data_in(31 downto 24);
            end if;
        end process dw;
        dr : process(read_enable, address) is
            begin
                if (read_enable = '1') and (unsigned(address) < 253) then
                    data_out( 7 downto 0) <= ram(to_integer(unsigned(address)) + 0);
                    data_out(15 downto 8) <= ram(to_integer(unsigned(address)) + 1);
                    data_out(23 downto 16) <= ram(to_integer(unsigned(address)) + 2);
                    data_out(31 downto 24) <= ram(to_integer(unsigned(address)) + 3);
                end if;
            end process dr;
    end Behavioral;

```

## Explanation

When the write enable (denoted as “write\_enable”) is true, it will take the current input (denoted as “data\_in”) and set the memory (in variable “ram”) at the address (denoted as “address”) to that value. When the read enable (denoted as “read\_enable”) is true, it will take the memory (in variable “ram”) at the address (denoted as “address”) and output that value (through the signal “data\_out”).

## Output



Figure 12: Graph of the output for the Instruction Memory

## Register File

### VHDL Code

```

library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
use IEEE.NUMERIC_STD.ALL;

entity REGISTER_FILE is
    port(write_enable : in std_logic;
         address_in, address_out_1, address_out_2 : in std_logic_vector(4 downto 0);
         data_in : in std_logic_vector(63 downto 0);
         data_out_1, data_out_2 : out std_logic_vector(63 downto 0));
end REGISTER_FILE;

architecture Behavioral of REGISTER_FILE is

type memory is array(0 to 31) of std_logic_vector(63 downto 0);
signal ram : memory := (
    others => "0000000000000000000000000000000000000000000000000000000000000000");
begin
    process(write_enable, address_in, data_in, address_out_1, address_out_2) is
        begin
            if write_enable = '1' then
                if NOT(address_in = "11111") then
                    ram(to_integer(unsigned(address_in))) <= data_in;
                end if;
            end if;
            data_out_1 <= ram(to_integer(unsigned(address_out_1)));
            data_out_2 <= ram(to_integer(unsigned(address_out_2)));
        end process;
    end Behavioral;

```

### Explanation

When the write enable (denoted as “write\_enable”) is true, it will take the current input (denoted as “data\_in”) and set the memory (in variable “ram”) at the input address (denoted as “address\_in”) to that value, unless the address is XZR, then it will do nothing. It will take the memory (in variable “ram”) at the output addresses (denoted as “address\_out\_1” and “address\_out\_2”) and output those values (through the signal “data\_out\_1” and “data\_out\_2” respective to the address\_out numbers).

### Output



Figure 13: Graph of the output for the Register File

## ALU

### VHDL Code

```

library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
use IEEE.NUMERIC_STD.ALL;

entity ALUv2 is
    Port ( A_in : in STD_LOGIC_VECTOR (63 downto 0);
           B_in : in STD_LOGIC_VECTOR (63 downto 0);
           ALU_Out : out STD_LOGIC_VECTOR (63 downto 0);
           ALU_Flags : out STD_LOGIC_VECTOR (3 downto 0);
           OPCODE : in STD_LOGIC_VECTOR (3 downto 0));
end ALUv2;
architecture Behavioral of ALUv2 is

begin
    Process (OPCODE, A_in, B_in)
        variable Temp : STD_LOGIC_VECTOR (63 downto 0);
        variable Temp_ALU_OUT : STD_LOGIC_VECTOR (63 downto 0);
        variable Temp_ALU_FLAGS : STD_LOGIC_VECTOR (3 downto 0);
        variable Temp2 : STD_LOGIC;
    begin
        ALU_Flags(0) <= '0';
        ALU_Flags(1) <= '0';
        if (OPCODE = "0010") or (OPCODE = "0110") then
            if (OPCODE = "0110") then
                Temp_ALU_FLAGS(1) := '1';
                Temp2 := '1';
                Temp := NOT(B_in);
            else
                Temp_ALU_FLAGS(1) := '0';
                Temp2 := '0';
                Temp := (B_in);
            end if;
            for i in 0 to 63 loop
                Temp_ALU_FLAGS(0) := Temp_ALU_FLAGS(1);
                Temp_ALU_OUT(i) := (Temp(i) xor A_in(i)) xor Temp_ALU_FLAGS(1);
                Temp_ALU_FLAGS(1) := (Temp(i) and A_in(i)) or
                    (Temp_ALU_FLAGS(1) and Temp(i)) or
                    (Temp_ALU_FLAGS(1) and A_in(i));
            end loop;
            ALU_Flags(0) <= Temp_ALU_FLAGS(0) xor Temp_ALU_FLAGS(1);
            ALU_Flags(1) <= Temp2 xor Temp_ALU_FLAGS(1);
        elsif(OPCODE = "0000") then
            Temp_ALU_OUT := B_in and A_in;
        elsif(OPCODE = "0001") then
            Temp_ALU_OUT := B_in or A_in;
        elsif(OPCODE = "0111") then
            Temp_ALU_OUT := B_in;
        elsif(OPCODE = "1100") then
    end process;
end Behavioral;

```

## Explanation

The ALU takes two inputs, “A\_in” and “B\_in”, and acts on them based on the operation code (denoted as “OPCODE”), these codes were given in the assignment. The ALU outputs several flags in the “ALU\_Flags” array, the array is formatted {Negative Flag, Zero Flag, Carry Flag, Overflow Flag}. The ALU Output is through “ALU\_Out”.

## Output



Figure 14: Graph of the output for the ALU

## Data Memory

### VHDL Code

```
library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
use IEEE.NUMERIC_STD.ALL;

entity Data_RAM is
    port(write_enable, read_enable : in std_logic;
         address, data_in : in std_logic_vector(63 downto 0);
         data_out : out std_logic_vector(63 downto 0));
end Data_RAM;

architecture Behavioral of Data_RAM is
begin
    type memory is array(0 to 255) of std_logic_vector(7 downto 0);
    signal ram : memory := (0 => "01000010",
                            1 => "00000001",
                            2 => "01000000",
                            3 => "11111000",
                            4 => "01000011",
                            5 => "10000001",
                            6 => "01000000",
                            7 => "11111000",
                            8 => "01100100",
                            9 => "00000000",
                            10 => "00000010",
                            11 => "11001011",
                            12 => "01100101",
                            13 => "00000000",
                            14 => "00000010",
                            15 => "10001011",
                            16 => "01000001",
                            17 => "00000000",
                            18 => "00000000",
                            19 => "10110100",
                            20 => "01000000",
                            21 => "00000000",
                            22 => "00000000",
                            23 => "10110100",
                            24 => "01000010",
                            25 => "00000001",
                            26 => "01000000",
                            27 => "11111000",
                            28 => "01000110",
                            29 => "00000000",
                            30 => "00000011",
                            31 => "10101010",
                            32 => "01000111",
                            33 => "00000000",
                            34 => "00000011",
                            35 => "10001010",
                            36 => "11100100",
```

```

37 => "10000000",
38 => "00000000",
39 => "11111000",
40 => "00000010",
41 => "00000000",
42 => "00000000",
43 => "00010100",
44 => "01000011",
45 => "10000001",
46 => "01000000",
47 => "11111000",
48 => "00001000",
49 => "00000000",
50 => "00000001",
51 => "10001011",
others => "00000000");

begin
dw : process(write_enable, address, data_in) is
begin
    if (write_enable = '1') and (unsigned(address) < 249) then
        ram(to_integer(unsigned(address)) + 0) <= data_in( 7 downto 0);
        ram(to_integer(unsigned(address)) + 1) <= data_in(15 downto 8);
        ram(to_integer(unsigned(address)) + 2) <= data_in(23 downto 16);
        ram(to_integer(unsigned(address)) + 3) <= data_in(31 downto 24);
        ram(to_integer(unsigned(address)) + 4) <= data_in(39 downto 32);
        ram(to_integer(unsigned(address)) + 5) <= data_in(47 downto 40);
        ram(to_integer(unsigned(address)) + 6) <= data_in(55 downto 48);
        ram(to_integer(unsigned(address)) + 7) <= data_in(63 downto 56);
    end if;
end process dw;
dr : process(read_enable, address) is
begin
    if (read_enable = '1') and (unsigned(address) < 249) then
        data_out( 7 downto 0) <= ram(to_integer(unsigned(address)) + 0);
        data_out(15 downto 8) <= ram(to_integer(unsigned(address)) + 1);
        data_out(23 downto 16) <= ram(to_integer(unsigned(address)) + 2);
        data_out(31 downto 24) <= ram(to_integer(unsigned(address)) + 3);
        data_out(39 downto 32) <= ram(to_integer(unsigned(address)) + 4);
        data_out(47 downto 40) <= ram(to_integer(unsigned(address)) + 5);
        data_out(55 downto 48) <= ram(to_integer(unsigned(address)) + 6);
        data_out(63 downto 56) <= ram(to_integer(unsigned(address)) + 7);
    end if;
end process dr;
end Behavioral;

```

## Explanation

When the write enable (denoted as “write\_enable”) is true, it will take the current input (denoted as “data\_in”) and set the memory (in variable “ram”) at the address (denoted as “address”) to that value. When the read enable (denoted as “read\_enable”) is true, it will take the memory (in variable “ram”) at the address (denoted as “address”) and output that value (through the signal “data\_out”).

## Output



Figure 15: Graph of the output for the Data Memory

## Sign Extend

### VHDL Code

```

library IEEE;
use IEEE.STD_LOGIC_1164.ALL;
use IEEE.NUMERIC_STD.ALL;

entity Sign_Extend is
    Port ( INST_In : in STD_LOGIC_VECTOR (31 downto 0);
            IMM_Out : out STD_LOGIC_VECTOR (63 downto 0));
end Sign_Extend;

architecture Behavioral of Sign_Extend is
begin
    process(INST_In)
    begin
        if (INST_In(31 downto 26) = "000101") then -- B
            IMM_Out(31 downto 0) <= INST_In;
            for i in 26 to 63 loop
                IMM_Out(i) <= INST_In(25);
            end loop;
        elsif (INST_In(31 downto 25) = "1011010") then -- CBZ and CBNZ
            IMM_Out(18 downto 0) <= INST_In(23 downto 5);
            for i in 19 to 63 loop
                IMM_Out(i) <= INST_In(23);
            end loop;
        elsif (INST_In(31 downto 21) = "11111000000") or
              (INST_In(31 downto 21) = "11111000010") then -- LOAD and STORE
            IMM_Out(8 downto 0) <= INST_In(20 downto 12);
            for i in 9 to 63 loop
                IMM_Out(i) <= INST_In(20);
            end loop;
        end if;
    end process;
end Behavioral;

```

### Explanation

This component takes an instruction in (denoted as “INST\_In”), and if it is a “B”, “CBZ”, “CBNZ”, “LDUR”, or “STUR” operation, it will extract the Immediate value, sign extend, and output to “IMM\_Out”.

### Output



Figure 16: Graph of the output for the Sign Extend