178 lines
7.3 KiB
Verilog
178 lines
7.3 KiB
Verilog
`timescale 1ns / 1ps
|
|
`include "tb_tools.vh"
|
|
|
|
module tb_risc_v_cpu ();
|
|
reg clk;
|
|
reg reset;
|
|
integer i;
|
|
wire [31:0] out;
|
|
|
|
/* File management variable */
|
|
integer bin_file_inputs;
|
|
integer code_file_inputs;
|
|
reg [8:0] read_instruction_1;
|
|
reg [8:0] read_instruction_2;
|
|
reg [8:0] read_instruction_3;
|
|
reg [8:0] read_instruction_4;
|
|
|
|
reg [113:0] test [0:100];
|
|
integer instruction_addr;
|
|
reg [5:0] reg_number;
|
|
reg [31:0] reg_test_value;
|
|
integer curent_addr;
|
|
integer res;
|
|
reg [8:0] dump;
|
|
reg [50*8:1] message;
|
|
integer size;
|
|
|
|
risc_v_cpu risc_v_cpu (
|
|
.clock(clk),
|
|
.reset(reset),
|
|
.out(out)
|
|
);
|
|
|
|
initial begin
|
|
/* Reset */
|
|
reset = 1'b1;
|
|
#10
|
|
reset = 1'b0;
|
|
|
|
clk = 1'b0;
|
|
|
|
/* Loading Test From File */
|
|
|
|
/* Loading Binary File */
|
|
bin_file_inputs = $fopen("./test.bin", "r");
|
|
if (bin_file_inputs == 0) begin
|
|
$display("bin file handle was NULL");
|
|
$finish;
|
|
end
|
|
|
|
i = 0;
|
|
while (!$feof(bin_file_inputs))
|
|
begin
|
|
read_instruction_1 = $fgetc(bin_file_inputs);
|
|
read_instruction_2 = $fgetc(bin_file_inputs);
|
|
read_instruction_3 = $fgetc(bin_file_inputs);
|
|
read_instruction_4 = $fgetc(bin_file_inputs);
|
|
|
|
if (
|
|
read_instruction_1[8] != 1'b1 &&
|
|
read_instruction_2[8] != 1'b1 &&
|
|
read_instruction_3[8] != 1'b1 &&
|
|
read_instruction_4[8] != 1'b1
|
|
) begin
|
|
risc_v_cpu.uut_instruction.memory[i] = read_instruction_1[7:0];
|
|
risc_v_cpu.uut_instruction.memory[i+1] = read_instruction_2[7:0];
|
|
risc_v_cpu.uut_instruction.memory[i+2] = read_instruction_3[7:0];
|
|
risc_v_cpu.uut_instruction.memory[i+3] = read_instruction_4[7:0];
|
|
i = i + 4;
|
|
end
|
|
end
|
|
|
|
$fclose(bin_file_inputs);
|
|
|
|
/* Extract Value to Test From File */
|
|
code_file_inputs = $fopen("./runtime_test.tmp", "r");
|
|
if (code_file_inputs == 0) begin
|
|
$display("source code file handle was NULL");
|
|
$finish;
|
|
end
|
|
|
|
i = 0;
|
|
for (i = 0; i < 100; i = i + 1) begin // Fill test data structure of 1,
|
|
test[i] = {114{1'b1}}; // to represent the empty state
|
|
end
|
|
|
|
while (!$feof(code_file_inputs))
|
|
begin
|
|
res = $fscanf(code_file_inputs, "%d:%d=%d\n", instruction_addr, reg_number, reg_test_value);
|
|
if (res != 3) begin // If fscanf failed, the test file structure is wrong, then exit
|
|
$display("Parsing test file failed");
|
|
$finish;
|
|
end
|
|
|
|
instruction_addr = instruction_addr / 4;
|
|
|
|
if (test[instruction_addr][5:0] == 6'b111111) begin
|
|
test[instruction_addr][5:0] = reg_number;
|
|
test[instruction_addr][37:6] = reg_test_value;
|
|
end else if (test[instruction_addr][43:38] == 6'b111111) begin
|
|
$display ("1");
|
|
test[instruction_addr][43:38] = reg_number;
|
|
test[instruction_addr][75:44] = reg_test_value;
|
|
end else if (test[instruction_addr][81:76] == 6'b111111) begin
|
|
test[instruction_addr][81:76] = reg_number;
|
|
test[instruction_addr][113:83] = reg_test_value;
|
|
end
|
|
end
|
|
|
|
$fclose(code_file_inputs);
|
|
|
|
/* Run The Program */
|
|
|
|
for (i = 0; i < 100; i = i + 1) begin
|
|
if (test[risc_v_cpu.program_counter.pc_addr / 4][5:0] != 6'b111111) begin
|
|
curent_addr = risc_v_cpu.program_counter.pc_addr / 4;
|
|
`next_cycle
|
|
if (test[curent_addr][5:0] != 6'b111111) begin
|
|
if (test[curent_addr][5:0] < 6'b100000) begin
|
|
`assert_no_wait_reg("RUNTIME", curent_addr, test[curent_addr][5:0], test[curent_addr][37:6], risc_v_cpu.registers_bank.registers[test[curent_addr][4:0]])
|
|
end else if (test[curent_addr][5:0] == 6'b100000) begin
|
|
`assert_no_wait_pc("RUNTIME", curent_addr, test[curent_addr][37:6], risc_v_cpu.program_counter.pc_addr)
|
|
end else if (test[curent_addr][5:0] > 6'b100000) begin
|
|
`assert_no_wait_mem("RUNTIME", curent_addr, test[curent_addr][5:0], test[curent_addr][37:6], risc_v_cpu.memory.memory[test[curent_addr][5:0]])
|
|
end
|
|
end
|
|
if (test[curent_addr][43:38] != 6'b111111) begin
|
|
if (test[curent_addr][43:38] < 6'b100000) begin
|
|
`assert_no_wait_reg("RUNTIME", curent_addr, test[curent_addr][43:38], test[curent_addr][75:44], risc_v_cpu.registers_bank.registers[test[curent_addr][42:38]])
|
|
end else if (test[curent_addr][43:38] == 6'b100000) begin
|
|
`assert_no_wait_pc("RUNTIME", curent_addr, test[curent_addr][75:44], risc_v_cpu.program_counter.pc_addr)
|
|
end else if (test[curent_addr][43:38] > 6'b100000) begin
|
|
`assert_no_wait_mem("RUNTIME", curent_addr, test[curent_addr][43:38], test[curent_addr][75:44], risc_v_cpu.memory.memory[test[curent_addr][43:38]])
|
|
end
|
|
end
|
|
if (test[curent_addr][81:76] != 6'b111111) begin
|
|
if (test[curent_addr][81:76] < 6'b100000) begin
|
|
`assert_no_wait_reg("RUNTIME", curent_addr, test[curent_addr][81:76], test[curent_addr][81:76], risc_v_cpu.registers_bank.registers[test[curent_addr][80:76]])
|
|
end else if (test[curent_addr][81:76] == 6'b100000) begin
|
|
`assert_no_wait_pc("RUNTIME", curent_addr, test[curent_addr][113:83], risc_v_cpu.program_counter.pc_addr)
|
|
end else if (test[curent_addr][81:76] > 6'b100000) begin
|
|
`assert_no_wait_mem("RUNTIME", curent_addr, test[curent_addr][81:76], test[curent_addr][81:76], risc_v_cpu.memory.memory[test[curent_addr][81:76]])
|
|
end
|
|
end
|
|
end
|
|
else
|
|
`next_cycle
|
|
end
|
|
|
|
/* Test State After Execution */
|
|
code_file_inputs = $fopen("./final_test.tmp", "r");
|
|
if (code_file_inputs == 0) begin
|
|
$display("source code file handle was NULL");
|
|
$finish;
|
|
end
|
|
|
|
while (!$feof(code_file_inputs))
|
|
begin
|
|
res = $fscanf(code_file_inputs, "%d=%d\n", reg_number, reg_test_value);
|
|
if (res != 2) begin // If fscanf failed, the test file structure is wrong, then exit
|
|
$display("Parsing test file failed");
|
|
$finish;
|
|
end
|
|
|
|
if (reg_number < 6'b100000) begin
|
|
`assert_no_wait_reg("FINAL", 1'bx, reg_number, reg_test_value, risc_v_cpu.registers_bank.registers[reg_number[4:0]])
|
|
end else if (reg_number == 6'b100000) begin
|
|
`assert_no_wait_pc("FINAL", 1'bx, reg_test_value, risc_v_cpu.program_counter.pc_addr)
|
|
end else if (reg_number > 6'b100000) begin
|
|
`assert_no_wait_mem("FINAL", 1'bx, reg_number, reg_test_value, risc_v_cpu.memory.memory[test[curent_addr][5:0]])
|
|
end
|
|
end
|
|
|
|
`end_message
|
|
end
|
|
|
|
endmodule : tb_risc_v_cpu
|