hp-saturn/saturn-alu.v

335 lines
7.7 KiB
Coq
Raw Normal View History

2019-02-12 12:43:36 +01:00
`ifndef _SATURN_ALU
`define _SATURN_ALU
`include "def-alu.v"
`define ALU_DEBUG
module saturn_alu (
i_clk,
i_reset,
2019-02-13 22:43:04 +01:00
i_en_alu_dump,
i_en_alu_prep,
i_en_alu_calc,
2019-02-13 22:43:04 +01:00
i_en_alu_init,
2019-02-12 12:43:36 +01:00
i_en_alu_save,
2019-02-13 22:43:04 +01:00
o_alu_stall_dec,
i_ins_decoded,
2019-02-12 12:43:36 +01:00
i_field_start,
i_field_last,
2019-02-13 22:43:04 +01:00
i_imm_value,
2019-02-12 12:43:36 +01:00
i_alu_op,
i_alu_no_stall,
2019-02-13 22:43:04 +01:00
i_reg_dest,
i_reg_src1,
i_reg_src2,
i_ins_alu_op,
2019-02-12 12:43:36 +01:00
2019-02-13 22:43:04 +01:00
o_reg_p,
o_pc
);
2019-02-12 12:43:36 +01:00
input wire [0:0] i_clk;
input wire [0:0] i_reset;
2019-02-13 22:43:04 +01:00
input wire [0:0] i_en_alu_dump;
2019-02-12 12:43:36 +01:00
input wire [0:0] i_en_alu_prep;
input wire [0:0] i_en_alu_calc;
2019-02-13 22:43:04 +01:00
input wire [0:0] i_en_alu_init;
2019-02-12 12:43:36 +01:00
input wire [0:0] i_en_alu_save;
2019-02-13 23:18:50 +01:00
output wire [0:0] o_alu_stall_dec;
2019-02-13 22:43:04 +01:00
input wire [0:0] i_ins_decoded;
2019-02-12 12:43:36 +01:00
input wire [3:0] i_field_start;
input wire [3:0] i_field_last;
2019-02-13 22:43:04 +01:00
input wire [3:0] i_imm_value;
2019-02-12 12:43:36 +01:00
input wire [4:0] i_alu_op;
input wire [0:0] i_alu_no_stall;
2019-02-13 22:43:04 +01:00
input wire [4:0] i_reg_dest;
input wire [4:0] i_reg_src1;
input wire [4:0] i_reg_src2;
2019-02-12 12:43:36 +01:00
2019-02-13 22:43:04 +01:00
input wire i_ins_alu_op;
2019-02-12 12:43:36 +01:00
2019-02-13 22:43:04 +01:00
output wire [3:0] o_reg_p;
output wire [19:0] o_pc;
2019-02-12 12:43:36 +01:00
assign o_reg_p = P;
assign o_pc = PC;
2019-02-12 12:43:36 +01:00
reg [19:0] PC;
2019-02-13 22:43:04 +01:00
reg [19:0] D0;
reg [19:0] D1;
reg [63:0] A;
reg [63:0] B;
reg [63:0] C;
reg [63:0] D;
reg [63:0] R0;
reg [63:0] R1;
reg [63:0] R2;
reg [63:0] R3;
reg [63:0] R4;
reg [0:0] CARRY;
reg [0:0] DEC;
reg [3:0] P;
reg [3:0] HST;
reg [15:0] ST;
reg [2:0] rstk_ptr;
reg [19:0] RSTK[0:7];
2019-02-12 12:43:36 +01:00
initial begin
2019-02-13 22:43:04 +01:00
// alu internal control bits
alu_run = 0;
2019-02-13 23:18:50 +01:00
alu_done = 0;
// o_alu_stall_dec = 0;
2019-02-13 22:43:04 +01:00
// processor registers
PC = 0;
2019-02-13 22:43:04 +01:00
D0 = 0;
D1 = 0;
A = 0;
B = 0;
C = 0;
D = 0;
R0 = 0;
R1 = 0;
R2 = 0;
R3 = 0;
R4 = 0;
CARRY = 0;
DEC = 0;
P = 0;
HST = 0;
ST = 0;
rstk_ptr = 0;
RSTK[0] = 0;
RSTK[1] = 0;
RSTK[2] = 0;
RSTK[3] = 0;
RSTK[4] = 0;
RSTK[5] = 0;
RSTK[6] = 0;
RSTK[7] = 0;
2019-02-12 12:43:36 +01:00
end
2019-02-13 22:43:04 +01:00
wire do_reg_dump;
wire do_alu_init;
wire do_alu_prep;
wire do_alu_calc;
wire do_alu_save;
wire do_alu_shpc;
wire do_alu_pc;
2019-02-13 22:43:04 +01:00
assign do_reg_dump = (!i_reset) && i_en_alu_dump && i_ins_decoded && !o_alu_stall_dec;
assign do_alu_init = (!i_reset) && i_en_alu_init && i_ins_alu_op && !alu_run;
assign do_alu_prep = (!i_reset) && i_en_alu_prep && alu_run;
assign do_alu_calc = (!i_reset) && i_en_alu_calc && alu_run;
assign do_alu_save = (!i_reset) && i_en_alu_save && alu_run;
assign do_alu_shpc = (!i_reset) && i_en_alu_dump;
assign do_alu_pc = (!i_reset) && i_en_alu_save;
reg alu_run;
reg alu_done;
wire test_finish;
wire [3:0] f_next;
2019-02-13 22:43:04 +01:00
assign test_finish = f_start == f_last;
assign f_next = (f_start + 1) & 4'hF;
2019-02-13 23:18:50 +01:00
// the decoder may request the ALU to not stall it
assign o_alu_stall_dec = alu_run && (!i_alu_no_stall || test_finish);
2019-02-13 23:18:50 +01:00
2019-02-13 22:43:04 +01:00
reg [4:0] alu_op;
reg [4:0] reg_dest;
reg [4:0] reg_src1;
reg [4:0] reg_src2;
reg [3:0] f_start;
reg [3:0] f_last;
reg [3:0] p_src1;
reg [3:0] p_src2;
reg p_carry;
reg [3:0] c_res1;
reg [3:0] c_res2;
reg c_carry;
2019-02-13 23:18:50 +01:00
/*
* dump all registers
* this only reads things...
*
*/
2019-02-13 22:43:04 +01:00
2019-02-13 23:18:50 +01:00
always @(posedge i_clk) begin
2019-02-13 22:43:04 +01:00
if (do_reg_dump) begin
2019-02-13 23:18:50 +01:00
$display("ALU_DUMP 0: run %b | done %b ", alu_run, alu_done);
2019-02-13 22:43:04 +01:00
`ifdef SIM
// display registers
$display("PC: %05h Carry: %b h: %s rp: %h RSTK7: %05h",
PC, CARRY, DEC?"DEC":"HEX", rstk_ptr, RSTK[7]);
$display("P: %h HST: %b ST: %b RSTK6: %5h",
P, HST, ST, RSTK[6]);
$display("A: %h R0: %h RSTK5: %5h", A, R0, RSTK[5]);
$display("B: %h R1: %h RSTK4: %5h", B, R1, RSTK[4]);
$display("C: %h R2: %h RSTK3: %5h", C, R2, RSTK[3]);
$display("D: %h R3: %h RSTK2: %5h", D, R3, RSTK[2]);
$display("D0: %h D1: %h R4: %h RSTK1: %5h",
D0, D1, R4, RSTK[1]);
$display(" RSTK0: %5h",
RSTK[0]);
`endif
end
2019-02-13 23:18:50 +01:00
end
2019-02-13 22:43:04 +01:00
2019-02-13 23:18:50 +01:00
always @(posedge i_clk) begin
2019-02-13 22:43:04 +01:00
// this happens in phase 3, right after the instruction decoder (in phase 2) is finished
if (do_alu_init) begin
$display({"ALU_INIT 3: run %b | done %b | stall %b | op %d | s %h | l %h ",
"| ialu %b | dest %d | src1 %d | src2 %d"},
alu_run, alu_done, o_alu_stall_dec, i_alu_op,i_field_start, i_field_last,
i_ins_alu_op, i_reg_dest, i_reg_src1, i_reg_src2);
2019-02-13 22:43:04 +01:00
alu_op <= i_alu_op;
reg_dest <= i_reg_dest;
reg_src1 <= i_reg_src1;
reg_src2 <= i_reg_src2;
f_start <= i_field_start;
f_last <= i_field_last;
end
2019-02-13 23:18:50 +01:00
end
/*
* handles alu_done
*/
2019-02-13 23:18:50 +01:00
always @(posedge i_clk) begin
if (do_alu_init) alu_run <= 1;
if (do_alu_prep) begin
// $display("ALU_TEST 1: tf %b | nxt %h", test_finish, f_next);
alu_done <= 0;
end
if (do_alu_calc) begin
// $display("ALU_TEST 2: tf %b | nxt %h", test_finish, f_next);
alu_done <= test_finish;
// f_next <= (f_start + 1) & 4'hF;
end
if (do_alu_save) begin
// $display("ALU_TEST 3: tf %b | nxt %h", test_finish, f_next);
f_start <= f_next;
end
2019-02-13 23:18:50 +01:00
if (do_alu_save && alu_done) begin
alu_run <= 0;
alu_done <= 0;
end
end
2019-02-13 22:43:04 +01:00
2019-02-13 23:18:50 +01:00
always @(posedge i_clk) begin
if (do_alu_prep) begin
`ifdef ALU_DEBUG
$display("ALU_PREP 1: run %b | done %b | stall %b | op %d | s %h | l %h",
alu_run, alu_done, o_alu_stall_dec, alu_op, f_start, f_last);
2019-02-13 22:43:04 +01:00
`endif
// setup value for src1
case (alu_op)
`ALU_OP_ZERO: begin end // no source required
`ALU_OP_COPY,
`ALU_OP_JMP_REL3:
2019-02-13 22:43:04 +01:00
case (reg_src1)
`ALU_REG_A: p_src1 <= A [f_start*4+:4];
`ALU_REG_B: p_src1 <= B [f_start*4+:4];
`ALU_REG_C: p_src1 <= C [f_start*4+:4];
`ALU_REG_D: p_src1 <= D [f_start*4+:4];
`ALU_REG_D0: p_src1 <= D0[f_start*4+:4];
`ALU_REG_D1: p_src1 <= D1[f_start*4+:4];
2019-02-13 22:43:04 +01:00
`ALU_REG_P: p_src1 <= P;
`ALU_REG_IMM: p_src1 <= i_imm_value;
endcase
endcase
// setup p_carry
end
2019-02-13 23:18:50 +01:00
end
2019-02-13 22:43:04 +01:00
2019-02-13 23:18:50 +01:00
always @(posedge i_clk) begin
2019-02-13 22:43:04 +01:00
if (do_alu_calc) begin
`ifdef ALU_DEBUG
$display("ALU_CALC 2: run %b | done %b | stall %b | op %d | s %h | l %h | src1 %h | src2 %h | p_carry %b",
alu_run, alu_done, o_alu_stall_dec, alu_op, f_start, f_last, p_src1, p_src2, p_carry);
`endif
2019-02-13 22:43:04 +01:00
case (alu_op)
`ALU_OP_ZERO: c_res1 <= 0;
`ALU_OP_COPY: c_res1 <= p_src1;
endcase
end
2019-02-13 23:18:50 +01:00
end
2019-02-13 22:43:04 +01:00
2019-02-13 23:18:50 +01:00
always @(posedge i_clk) begin
2019-02-13 22:43:04 +01:00
if (do_alu_save) begin
`ifdef ALU_DEBUG
$display({"ALU_SAVE 3: run %b | done %b | stall %b | op %d | s %h | l %h |",
" res1 %h | res2 %h | c_carry %b"},
alu_run, alu_done, o_alu_stall_dec, alu_op,
f_start, f_last, c_res1, c_res2, c_carry);
`endif
2019-02-13 22:43:04 +01:00
case (alu_op)
`ALU_OP_ZERO,
`ALU_OP_COPY:
case (reg_dest)
`ALU_REG_C: C [f_start*4+:4] <= c_res1;
`ALU_REG_D0: D0[f_start*4+:4] <= c_res1;
`ALU_REG_D1: D1[f_start*4+:4] <= c_res1;
`ALU_REG_P: P <= c_res1;
2019-02-13 22:43:04 +01:00
endcase
endcase
2019-02-12 12:43:36 +01:00
end
end
wire [19:0] next_pc;
assign next_pc = PC + 1;
always @(posedge i_clk) begin
if (i_reset)
PC <= ~0;
if (do_alu_shpc) begin
// if (!o_alu_stall_dec)
// $display("ALU_SHPC 0: pc %5h", PC);
if (o_alu_stall_dec)
$display("ALU_SHPC 0: STALL");
end
if (do_alu_pc) begin
// if (!o_alu_stall_dec)
// $display("ALU_PC 3: nx %5h", next_pc);
if (!o_alu_stall_dec)
PC <= next_pc;
end
2019-02-12 12:43:36 +01:00
end
endmodule