2019-02-14 15:27:17 +01:00
|
|
|
|
|
2019-02-12 12:43:36 +01:00
|
|
|
|
`ifndef _SATURN_ALU
|
|
|
|
|
`define _SATURN_ALU
|
|
|
|
|
|
|
|
|
|
`include "def-alu.v"
|
|
|
|
|
|
2019-02-14 14:35:23 +01:00
|
|
|
|
`ifdef SIM
|
|
|
|
|
// `define ALU_DEBUG_DBG
|
|
|
|
|
`endif
|
|
|
|
|
|
|
|
|
|
`define ALU_DEBUG 1'b0
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`define ALU_DEBUG_DUMP 1'b1
|
2019-02-16 12:17:40 +01:00
|
|
|
|
`define ALU_DEBUG_JUMP 1'b0
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`define ALU_DEBUG_PC 1'b0
|
2019-02-14 08:59:04 +01:00
|
|
|
|
|
2019-02-12 11:22:55 +01:00
|
|
|
|
module saturn_alu (
|
|
|
|
|
i_clk,
|
|
|
|
|
i_reset,
|
2019-02-13 22:43:04 +01:00
|
|
|
|
i_en_alu_dump,
|
2019-02-14 22:14:52 +01:00
|
|
|
|
i_en_alu_prep,
|
|
|
|
|
i_en_alu_calc,
|
2019-02-13 22:43:04 +01:00
|
|
|
|
i_en_alu_init,
|
2019-02-14 22:14:52 +01:00
|
|
|
|
i_en_alu_save,
|
2019-02-16 22:38:44 +01:00
|
|
|
|
i_stalled,
|
|
|
|
|
|
|
|
|
|
o_bus_address,
|
|
|
|
|
o_bus_load_pc,
|
|
|
|
|
o_bus_load_dp,
|
|
|
|
|
o_bus_nibble_out,
|
2019-02-12 12:43:36 +01:00
|
|
|
|
|
2019-02-14 14:35:23 +01:00
|
|
|
|
i_push,
|
|
|
|
|
i_pop,
|
|
|
|
|
i_alu_debug,
|
|
|
|
|
|
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,
|
2019-02-14 08:59:04 +01:00
|
|
|
|
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-16 11:08:34 +01:00
|
|
|
|
i_ins_test_go,
|
2019-02-15 07:07:55 +01:00
|
|
|
|
i_ins_set_mode,
|
|
|
|
|
i_ins_rtn,
|
|
|
|
|
|
|
|
|
|
i_mode_dec,
|
|
|
|
|
i_set_xm,
|
|
|
|
|
i_set_carry,
|
|
|
|
|
i_carry_val,
|
2019-02-12 12:43:36 +01:00
|
|
|
|
|
2019-02-13 22:43:04 +01:00
|
|
|
|
o_reg_p,
|
2019-02-14 08:59:04 +01:00
|
|
|
|
o_pc
|
2019-02-12 11:22:55 +01:00
|
|
|
|
);
|
|
|
|
|
|
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-16 22:38:44 +01:00
|
|
|
|
input wire [0:0] i_stalled;
|
|
|
|
|
|
|
|
|
|
output reg [19:0] o_bus_address;
|
|
|
|
|
output reg [0:0] o_bus_load_pc;
|
|
|
|
|
output reg [0:0] o_bus_load_dp;
|
|
|
|
|
output reg [3:0] o_bus_nibble_out;
|
2019-02-12 12:43:36 +01:00
|
|
|
|
|
2019-02-14 14:35:23 +01:00
|
|
|
|
input wire [0:0] i_push;
|
|
|
|
|
input wire [0:0] i_pop;
|
|
|
|
|
input wire [0:0] i_alu_debug;
|
|
|
|
|
|
|
|
|
|
wire alu_debug;
|
|
|
|
|
wire alu_debug_dump;
|
|
|
|
|
wire alu_debug_jump;
|
|
|
|
|
wire alu_debug_pc;
|
|
|
|
|
assign alu_debug = `ALU_DEBUG || i_alu_debug;
|
|
|
|
|
assign alu_debug_dump = `ALU_DEBUG_DUMP || i_alu_debug;
|
|
|
|
|
assign alu_debug_jump = `ALU_DEBUG_JUMP || i_alu_debug;
|
|
|
|
|
assign alu_debug_pc = `ALU_DEBUG_PC || i_alu_debug;
|
|
|
|
|
|
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;
|
2019-02-14 08:59:04 +01:00
|
|
|
|
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-15 07:07:55 +01:00
|
|
|
|
input wire [0:0] i_ins_alu_op;
|
2019-02-16 11:08:34 +01:00
|
|
|
|
input wire [0:0] i_ins_test_go;
|
2019-02-15 07:07:55 +01:00
|
|
|
|
input wire [0:0] i_ins_set_mode;
|
|
|
|
|
input wire [0:0] i_ins_rtn;
|
|
|
|
|
|
|
|
|
|
input wire [0:0] i_mode_dec;
|
|
|
|
|
input wire [0:0] i_set_xm;
|
|
|
|
|
input wire [0:0] i_set_carry;
|
|
|
|
|
input wire [0:0] i_carry_val;
|
2019-02-12 12:43:36 +01:00
|
|
|
|
|
2019-02-13 22:43:04 +01:00
|
|
|
|
output wire [3:0] o_reg_p;
|
2019-02-14 08:59:04 +01:00
|
|
|
|
output wire [19:0] o_pc;
|
2019-02-12 12:43:36 +01:00
|
|
|
|
|
|
|
|
|
assign o_reg_p = P;
|
2019-02-14 08:59:04 +01:00
|
|
|
|
assign o_pc = PC;
|
2019-02-12 12:43:36 +01:00
|
|
|
|
|
2019-02-14 14:35:23 +01:00
|
|
|
|
/* internal registers */
|
|
|
|
|
|
|
|
|
|
/* copy of arguments */
|
|
|
|
|
reg [4:0] alu_op;
|
|
|
|
|
reg [4:0] reg_dest;
|
|
|
|
|
reg [4:0] reg_src1;
|
|
|
|
|
reg [4:0] reg_src2;
|
2019-02-15 16:58:38 +01:00
|
|
|
|
reg [3:0] f_first;
|
|
|
|
|
reg [3:0] f_cur;
|
2019-02-14 14:35:23 +01:00
|
|
|
|
reg [3:0] f_last;
|
|
|
|
|
|
|
|
|
|
/* internal pointers */
|
|
|
|
|
|
|
|
|
|
reg [3:0] p_src1;
|
|
|
|
|
reg [3:0] p_src2;
|
2019-02-14 22:14:52 +01:00
|
|
|
|
reg [0:0] p_carry;
|
2019-02-14 14:35:23 +01:00
|
|
|
|
reg [3:0] c_res1;
|
|
|
|
|
reg [3:0] c_res2;
|
2019-02-14 22:14:52 +01:00
|
|
|
|
reg [0:0] c_carry;
|
|
|
|
|
reg [0:0] is_zero;
|
2019-02-14 14:35:23 +01:00
|
|
|
|
|
|
|
|
|
/* alu status */
|
|
|
|
|
|
|
|
|
|
reg alu_run;
|
|
|
|
|
reg alu_done;
|
2019-02-16 07:35:06 +01:00
|
|
|
|
reg alu_go_test;
|
2019-02-14 14:35:23 +01:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* next PC in case of jump
|
|
|
|
|
*/
|
|
|
|
|
reg [19:0] jump_bse;
|
|
|
|
|
reg [19:0] jump_off;
|
|
|
|
|
wire [19:0] jump_pc;
|
|
|
|
|
assign jump_pc = (alu_op == `ALU_OP_JMP_ABS5)?jump_off:(jump_bse + jump_off);
|
|
|
|
|
|
|
|
|
|
reg [2:0] rstk_ptr;
|
|
|
|
|
|
|
|
|
|
/* public registers */
|
|
|
|
|
|
2019-02-14 08:59:04 +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 [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
|
2019-02-14 14:35:23 +01:00
|
|
|
|
alu_op = 0;
|
|
|
|
|
reg_dest = 0;
|
|
|
|
|
reg_src1 = 0;
|
|
|
|
|
reg_src2 = 0;
|
2019-02-15 16:58:38 +01:00
|
|
|
|
f_first = 0;
|
|
|
|
|
f_cur = 0;
|
2019-02-14 14:35:23 +01:00
|
|
|
|
f_last = 0;
|
|
|
|
|
|
2019-02-13 22:43:04 +01:00
|
|
|
|
alu_run = 0;
|
2019-02-13 23:18:50 +01:00
|
|
|
|
alu_done = 0;
|
2019-02-14 22:14:52 +01:00
|
|
|
|
|
|
|
|
|
p_src1 = 0;
|
|
|
|
|
p_src2 = 0;
|
|
|
|
|
p_carry = 0;
|
|
|
|
|
c_res1 = 0;
|
|
|
|
|
c_res2 = 0;
|
|
|
|
|
c_carry = 0;
|
|
|
|
|
is_zero = 0;
|
2019-02-13 23:18:50 +01:00
|
|
|
|
// o_alu_stall_dec = 0;
|
2019-02-13 22:43:04 +01:00
|
|
|
|
// processor registers
|
2019-02-14 08:59:04 +01:00
|
|
|
|
PC = 0;
|
|
|
|
|
|
2019-02-16 11:08:34 +01:00
|
|
|
|
// D0 = 0;
|
|
|
|
|
// D1 = 0;
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-16 11:08:34 +01:00
|
|
|
|
// A = 0;
|
|
|
|
|
// B = 0;
|
|
|
|
|
// C = 0;
|
|
|
|
|
// D = 0;
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-16 11:08:34 +01:00
|
|
|
|
// R0 = 0;
|
|
|
|
|
// R1 = 0;
|
|
|
|
|
// R2 = 0;
|
|
|
|
|
// R3 = 0;
|
|
|
|
|
// R4 = 0;
|
|
|
|
|
|
|
|
|
|
// CARRY = 0;
|
|
|
|
|
// DEC = 0;
|
|
|
|
|
// P = 0;
|
|
|
|
|
// HST = 0;
|
|
|
|
|
// ST = 0;
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
|
|
|
|
rstk_ptr = 0;
|
2019-02-16 11:08:34 +01:00
|
|
|
|
// 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-12 11:22:55 +01:00
|
|
|
|
|
2019-02-16 22:38:44 +01:00
|
|
|
|
/*
|
|
|
|
|
* can the alu function ?
|
|
|
|
|
*/
|
|
|
|
|
wire alu_active;
|
|
|
|
|
|
|
|
|
|
assign alu_active = !i_reset && !i_stalled;
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* simulation only states, when alu is active
|
|
|
|
|
*/
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`ifdef SIM
|
2019-02-13 22:43:04 +01:00
|
|
|
|
wire do_reg_dump;
|
2019-02-14 15:27:17 +01:00
|
|
|
|
wire do_alu_shpc;
|
2019-02-17 08:35:26 +01:00
|
|
|
|
assign do_reg_dump = alu_active && i_en_alu_dump && !o_bus_load_pc &&
|
|
|
|
|
i_ins_decoded && !o_alu_stall_dec;
|
2019-02-16 22:38:44 +01:00
|
|
|
|
assign do_alu_shpc = alu_active && i_en_alu_dump;
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`endif
|
|
|
|
|
|
2019-02-16 22:38:44 +01:00
|
|
|
|
wire do_busclean;
|
2019-02-13 22:43:04 +01:00
|
|
|
|
wire do_alu_init;
|
|
|
|
|
wire do_alu_prep;
|
|
|
|
|
wire do_alu_calc;
|
|
|
|
|
wire do_alu_save;
|
2019-02-14 08:59:04 +01:00
|
|
|
|
wire do_alu_pc;
|
2019-02-15 07:07:55 +01:00
|
|
|
|
wire do_alu_mode;
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-16 22:38:44 +01:00
|
|
|
|
assign do_busclean = alu_active && i_en_alu_dump;
|
|
|
|
|
assign do_alu_init = alu_active && i_en_alu_init && i_ins_alu_op && !alu_run;
|
|
|
|
|
assign do_alu_prep = alu_active && i_en_alu_prep && alu_run;
|
|
|
|
|
assign do_alu_calc = alu_active && i_en_alu_calc && alu_run;
|
|
|
|
|
assign do_alu_save = alu_active && i_en_alu_save && alu_run;
|
|
|
|
|
assign do_alu_pc = alu_active && i_en_alu_save;
|
|
|
|
|
assign do_alu_mode = alu_active && i_en_alu_save && i_ins_set_mode;
|
2019-02-14 08:59:04 +01:00
|
|
|
|
|
2019-02-16 11:08:34 +01:00
|
|
|
|
wire do_go_init;
|
|
|
|
|
wire do_go_prep;
|
|
|
|
|
wire do_go_calc;
|
|
|
|
|
|
2019-02-16 22:38:44 +01:00
|
|
|
|
assign do_go_init = alu_active && i_en_alu_save && i_ins_test_go;
|
|
|
|
|
assign do_go_prep = alu_active && i_en_alu_prep && i_ins_test_go;
|
2019-02-16 11:08:34 +01:00
|
|
|
|
|
2019-02-14 08:59:04 +01:00
|
|
|
|
// the decoder may request the ALU to not stall it
|
|
|
|
|
|
2019-02-16 07:35:06 +01:00
|
|
|
|
assign o_alu_stall_dec = alu_run && (!i_alu_no_stall || alu_finish || alu_go_test);
|
2019-02-14 08:59:04 +01:00
|
|
|
|
|
2019-02-14 14:35:23 +01:00
|
|
|
|
wire alu_start;
|
|
|
|
|
wire alu_finish;
|
|
|
|
|
wire [3:0] f_next;
|
2019-02-13 23:18:50 +01:00
|
|
|
|
|
2019-02-15 16:58:38 +01:00
|
|
|
|
assign alu_start = f_cur == f_first;
|
|
|
|
|
assign alu_finish = f_cur == f_last;
|
|
|
|
|
assign f_next = (f_cur + 1) & 4'hF;
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-14 14:35:23 +01:00
|
|
|
|
/*
|
|
|
|
|
* test things on alu_op
|
|
|
|
|
*/
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-14 14:35:23 +01:00
|
|
|
|
wire is_alu_op_jump;
|
|
|
|
|
assign is_alu_op_jump = ((alu_op == `ALU_OP_JMP_REL3) ||
|
|
|
|
|
(alu_op == `ALU_OP_JMP_REL4) ||
|
2019-02-15 09:00:44 +01:00
|
|
|
|
(alu_op == `ALU_OP_JMP_ABS5) ||
|
|
|
|
|
i_ins_rtn);
|
2019-02-16 07:35:06 +01:00
|
|
|
|
wire is_alu_op_test;
|
|
|
|
|
assign is_alu_op_test = ((alu_op == `ALU_OP_TEST_EQ) ||
|
|
|
|
|
(alu_op == `ALU_OP_TEST_NEQ));
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-15 07:07:55 +01:00
|
|
|
|
/*****************************************************************************
|
2019-02-13 23:18:50 +01:00
|
|
|
|
*
|
2019-02-15 07:07:55 +01:00
|
|
|
|
* Dump all registers at the end of each instruction's execution cycle
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-13 23:18:50 +01:00
|
|
|
|
always @(posedge i_clk) begin
|
2019-02-16 22:38:44 +01:00
|
|
|
|
|
|
|
|
|
`ifdef SIM
|
|
|
|
|
if (i_stalled && i_en_alu_dump)
|
|
|
|
|
$display("ALU STALLED");
|
|
|
|
|
`endif
|
|
|
|
|
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`ifdef ALU_DEBUG_DBG
|
2019-02-14 14:35:23 +01:00
|
|
|
|
$display("iad %b | AD %b | ad %b | ADD %b | add %b | ADJ %b | adj %b | ADP %b | adp %b",
|
|
|
|
|
i_alu_debug,
|
|
|
|
|
`ALU_DEBUG, i_alu_debug,
|
|
|
|
|
`ALU_DEBUG_DUMP, alu_debug_dump,
|
|
|
|
|
`ALU_DEBUG_JUMP, alu_debug_jump,
|
|
|
|
|
`ALU_DEBUG_PC, alu_debug_pc );
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`endif
|
2019-02-14 14:35:23 +01:00
|
|
|
|
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`ifdef SIM
|
2019-02-14 14:35:23 +01:00
|
|
|
|
if (do_reg_dump && alu_debug_dump) begin
|
2019-02-14 15:27:17 +01:00
|
|
|
|
|
2019-02-14 14:35:23 +01:00
|
|
|
|
$display("ALU_DUMP 0: run %b | done %b", alu_run, alu_done);
|
2019-02-13 22:43:04 +01:00
|
|
|
|
// 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]);
|
|
|
|
|
end
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`endif
|
2019-02-13 23:18:50 +01:00
|
|
|
|
end
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-15 07:07:55 +01:00
|
|
|
|
/*****************************************************************************
|
|
|
|
|
*
|
|
|
|
|
* Initialize the ALU, to prepare it to execute the instruction
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2019-02-15 09:00:44 +01:00
|
|
|
|
wire [0:0] is_mem_read;
|
|
|
|
|
wire [0:0] is_mem_write;
|
|
|
|
|
wire [0:0] is_mem_xfer;
|
|
|
|
|
wire [4:0] mem_reg;
|
|
|
|
|
assign is_mem_read = (i_reg_src1 == `ALU_REG_DAT0) || (i_reg_src1 == `ALU_REG_DAT1);
|
|
|
|
|
assign is_mem_write = (i_reg_dest == `ALU_REG_DAT0) || (i_reg_dest == `ALU_REG_DAT1);
|
|
|
|
|
assign is_mem_xfer = is_mem_read || is_mem_write;
|
|
|
|
|
assign mem_reg = is_mem_read?i_reg_src1:i_reg_dest;
|
|
|
|
|
|
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
|
2019-02-14 14:35:23 +01:00
|
|
|
|
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`ifdef SIM
|
2019-02-14 14:35:23 +01:00
|
|
|
|
if (alu_debug)
|
|
|
|
|
$display({"ALU_INIT 3: run %b | done %b | stall %b | op %d | s %h | l %h ",
|
2019-02-14 22:54:54 +01:00
|
|
|
|
"| ialu %b | dest %d | src1 %d | src2 %d | imm %h"},
|
2019-02-14 14:35:23 +01:00
|
|
|
|
alu_run, alu_done, o_alu_stall_dec, i_alu_op,i_field_start, i_field_last,
|
2019-02-14 22:54:54 +01:00
|
|
|
|
i_ins_alu_op, i_reg_dest, i_reg_src1, i_reg_src2, i_imm_value);
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`endif
|
2019-02-14 14:35:23 +01:00
|
|
|
|
|
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_last <= i_field_last;
|
2019-02-15 09:00:44 +01:00
|
|
|
|
|
|
|
|
|
if (is_mem_xfer) begin
|
|
|
|
|
`ifdef SIM
|
|
|
|
|
$display("ALU_XFER 3: read %b | write %b | mem_reg DAT%b",
|
|
|
|
|
is_mem_read, is_mem_write, mem_reg[0]);
|
|
|
|
|
$display(".------------------------------------.");
|
|
|
|
|
$display("| SHOULD TELL THE BUS CONTROLLER TO |");
|
|
|
|
|
$display("| LOAD D0 OR D1 INTO MODULES' DP REG |");
|
|
|
|
|
$display("`------------------------------------´");
|
|
|
|
|
`endif
|
|
|
|
|
// DO SOMETHING TO GET THE BUS CONTROLLER TO SEND OUT
|
|
|
|
|
// THE CONTENTS OF D0 OR D1 AS THE DP
|
|
|
|
|
end
|
2019-02-13 22:43:04 +01:00
|
|
|
|
end
|
2019-02-13 23:18:50 +01:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
/*
|
2019-02-15 16:58:38 +01:00
|
|
|
|
* handles f_start, alu_run and alu_done
|
2019-02-13 23:18:50 +01:00
|
|
|
|
*/
|
2019-02-14 08:59:04 +01:00
|
|
|
|
|
2019-02-13 23:18:50 +01:00
|
|
|
|
always @(posedge i_clk) begin
|
2019-02-15 16:58:38 +01:00
|
|
|
|
|
|
|
|
|
if (do_alu_init) begin
|
|
|
|
|
alu_run <= 1;
|
|
|
|
|
f_first <= i_field_start;
|
|
|
|
|
f_cur <= i_field_start;
|
2019-02-16 07:35:06 +01:00
|
|
|
|
|
|
|
|
|
alu_go_test <= is_alu_op_test;
|
2019-02-15 16:58:38 +01:00
|
|
|
|
end
|
|
|
|
|
|
2019-02-14 08:59:04 +01:00
|
|
|
|
if (do_alu_prep) begin
|
|
|
|
|
// $display("ALU_TEST 1: tf %b | nxt %h", test_finish, f_next);
|
|
|
|
|
alu_done <= 0;
|
|
|
|
|
end
|
2019-02-15 16:58:38 +01:00
|
|
|
|
|
2019-02-14 08:59:04 +01:00
|
|
|
|
if (do_alu_calc) begin
|
|
|
|
|
// $display("ALU_TEST 2: tf %b | nxt %h", test_finish, f_next);
|
2019-02-14 14:35:23 +01:00
|
|
|
|
alu_done <= alu_finish;
|
2019-02-14 08:59:04 +01:00
|
|
|
|
// f_next <= (f_start + 1) & 4'hF;
|
|
|
|
|
end
|
2019-02-15 16:58:38 +01:00
|
|
|
|
|
2019-02-14 08:59:04 +01:00
|
|
|
|
if (do_alu_save) begin
|
|
|
|
|
// $display("ALU_TEST 3: tf %b | nxt %h", test_finish, f_next);
|
2019-02-15 16:58:38 +01:00
|
|
|
|
f_cur <= f_next;
|
2019-02-14 08:59:04 +01:00
|
|
|
|
end
|
2019-02-15 16:58:38 +01:00
|
|
|
|
|
2019-02-13 23:18:50 +01:00
|
|
|
|
if (do_alu_save && alu_done) begin
|
|
|
|
|
alu_run <= 0;
|
|
|
|
|
alu_done <= 0;
|
|
|
|
|
end
|
2019-02-16 07:35:06 +01:00
|
|
|
|
|
|
|
|
|
|
2019-02-16 11:08:34 +01:00
|
|
|
|
// if (do_alu_save && alu_done)
|
|
|
|
|
// case (alu_op)
|
|
|
|
|
// `ALU_OP_TEST_EQ,
|
|
|
|
|
// `ALU_OP_TEST_NEQ:
|
|
|
|
|
// begin
|
|
|
|
|
// $display("#### UNBLOCK THE DECODER");
|
|
|
|
|
// alu_go_test <= 1;
|
|
|
|
|
// end
|
|
|
|
|
// endcase
|
2019-02-16 07:35:06 +01:00
|
|
|
|
|
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-14 08:59:04 +01:00
|
|
|
|
if (do_alu_prep) begin
|
2019-02-14 14:35:23 +01:00
|
|
|
|
if (alu_debug) begin
|
|
|
|
|
`ifdef SIM
|
2019-02-16 11:08:34 +01:00
|
|
|
|
$display("ALU_PREP 1: run %b | done %b | stall %b | op %d | f %h | c %h | l %h | imm %h",
|
|
|
|
|
alu_run, alu_done, o_alu_stall_dec, alu_op, f_first, f_cur, f_last, i_imm_value);
|
2019-02-14 14:35:23 +01:00
|
|
|
|
`endif
|
|
|
|
|
end
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-16 12:17:40 +01:00
|
|
|
|
/*
|
|
|
|
|
* source 1
|
|
|
|
|
*/
|
2019-02-13 22:43:04 +01:00
|
|
|
|
case (alu_op)
|
|
|
|
|
`ALU_OP_ZERO: begin end // no source required
|
2019-02-14 08:59:04 +01:00
|
|
|
|
`ALU_OP_COPY,
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`ALU_OP_EXCH,
|
2019-02-14 14:35:23 +01:00
|
|
|
|
`ALU_OP_RST_BIT,
|
|
|
|
|
`ALU_OP_SET_BIT,
|
2019-02-14 22:14:52 +01:00
|
|
|
|
`ALU_OP_2CMPL,
|
2019-02-16 07:35:06 +01:00
|
|
|
|
`ALU_OP_ADD,
|
2019-02-16 12:17:40 +01:00
|
|
|
|
`ALU_OP_TEST_EQ,
|
|
|
|
|
`ALU_OP_TEST_NEQ,
|
2019-02-14 14:35:23 +01:00
|
|
|
|
`ALU_OP_JMP_REL3,
|
|
|
|
|
`ALU_OP_JMP_REL4,
|
2019-02-14 22:54:54 +01:00
|
|
|
|
`ALU_OP_JMP_ABS5,
|
|
|
|
|
`ALU_OP_CLR_MASK:
|
2019-02-13 22:43:04 +01:00
|
|
|
|
case (reg_src1)
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`ALU_REG_A: p_src1 <= A [f_cur*4+:4];
|
|
|
|
|
`ALU_REG_B: p_src1 <= B [f_cur*4+:4];
|
|
|
|
|
`ALU_REG_C: p_src1 <= C [f_cur*4+:4];
|
|
|
|
|
`ALU_REG_D: p_src1 <= D [f_cur*4+:4];
|
|
|
|
|
`ALU_REG_D0: p_src1 <= D0[f_cur*4+:4];
|
|
|
|
|
`ALU_REG_D1: p_src1 <= D1[f_cur*4+:4];
|
2019-02-15 09:00:44 +01:00
|
|
|
|
`ALU_REG_P: p_src1 <= P;
|
|
|
|
|
`ALU_REG_HST: p_src1 <= HST;
|
|
|
|
|
`ALU_REG_IMM: p_src1 <= i_imm_value;
|
|
|
|
|
`ALU_REG_ZERO: p_src1 <= 0;
|
2019-02-14 22:54:54 +01:00
|
|
|
|
default: $display("#### SRC_1 UNHANDLED REGISTER %0d", reg_src1);
|
2019-02-13 22:43:04 +01:00
|
|
|
|
endcase
|
2019-02-14 22:54:54 +01:00
|
|
|
|
default: $display("#### SRC_1 UNHANDLED OPERATION %0d", alu_op);
|
|
|
|
|
endcase
|
|
|
|
|
|
2019-02-16 12:17:40 +01:00
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
|
* source 2
|
|
|
|
|
*/
|
2019-02-14 22:54:54 +01:00
|
|
|
|
case (alu_op)
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`ALU_OP_ZERO,
|
|
|
|
|
`ALU_OP_COPY,
|
|
|
|
|
`ALU_OP_RST_BIT,
|
|
|
|
|
`ALU_OP_SET_BIT,
|
|
|
|
|
`ALU_OP_2CMPL,
|
|
|
|
|
`ALU_OP_JMP_REL3,
|
|
|
|
|
`ALU_OP_JMP_REL4,
|
|
|
|
|
`ALU_OP_JMP_ABS5: begin end // no need for a 2nd operand
|
|
|
|
|
`ALU_OP_EXCH,
|
2019-02-16 07:35:06 +01:00
|
|
|
|
`ALU_OP_ADD,
|
2019-02-16 12:17:40 +01:00
|
|
|
|
`ALU_OP_TEST_EQ,
|
|
|
|
|
`ALU_OP_TEST_NEQ,
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`ALU_OP_CLR_MASK: begin
|
|
|
|
|
case (reg_src2)
|
|
|
|
|
`ALU_REG_A: p_src2 <= A [f_cur*4+:4];
|
|
|
|
|
`ALU_REG_B: p_src2 <= B [f_cur*4+:4];
|
|
|
|
|
`ALU_REG_C: p_src2 <= C [f_cur*4+:4];
|
|
|
|
|
`ALU_REG_D: p_src2 <= D [f_cur*4+:4];
|
|
|
|
|
`ALU_REG_D0: p_src2 <= D0[f_cur*4+:4];
|
|
|
|
|
`ALU_REG_D1: p_src2 <= D1[f_cur*4+:4];
|
|
|
|
|
`ALU_REG_P: p_src2 <= P;
|
|
|
|
|
`ALU_REG_HST: p_src2 <= HST;
|
|
|
|
|
`ALU_REG_IMM: p_src2 <= i_imm_value;
|
|
|
|
|
`ALU_REG_ZERO: p_src2 <= 0;
|
|
|
|
|
default: $display("#### SRC_2 UNHANDLED REGISTER %0d", reg_src2);
|
|
|
|
|
endcase
|
|
|
|
|
end
|
|
|
|
|
default: $display("#### SRC_2 UNHANDLED OPERATION %0d", alu_op);
|
2019-02-13 22:43:04 +01:00
|
|
|
|
endcase
|
|
|
|
|
|
|
|
|
|
// setup p_carry
|
2019-02-14 22:14:52 +01:00
|
|
|
|
// $display("fs %h | fs=0 %b | cc %b | npc %b", f_start, (f_start == 0), c_carry, (f_start == 0)?1'b1:c_carry);
|
|
|
|
|
case (alu_op)
|
2019-02-16 07:35:06 +01:00
|
|
|
|
`ALU_OP_2CMPL: p_carry <= alu_start?1'b1:c_carry;
|
|
|
|
|
`ALU_OP_ADD: p_carry <= alu_start?0:c_carry;
|
2019-02-16 12:17:40 +01:00
|
|
|
|
`ALU_OP_TEST_NEQ: p_carry <= alu_start?0:c_carry;
|
2019-02-15 16:58:38 +01:00
|
|
|
|
endcase
|
|
|
|
|
|
|
|
|
|
// prepare jump base
|
|
|
|
|
case (alu_op)
|
|
|
|
|
`ALU_OP_JMP_REL3,
|
|
|
|
|
`ALU_OP_JMP_REL4:
|
|
|
|
|
begin
|
|
|
|
|
// the address of the first digit of the offset
|
|
|
|
|
if (!i_push && alu_start)
|
|
|
|
|
jump_bse <= PC - 1;
|
|
|
|
|
// doc says address of the next instruction, but appears to be off by 1
|
|
|
|
|
if (i_push)
|
|
|
|
|
jump_bse <= PC;
|
|
|
|
|
end
|
2019-02-14 22:14:52 +01:00
|
|
|
|
endcase
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
|
|
|
|
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-16 12:17:40 +01:00
|
|
|
|
|
|
|
|
|
if (i_reset) begin
|
|
|
|
|
c_carry <= 0;
|
|
|
|
|
end
|
|
|
|
|
|
2019-02-13 22:43:04 +01:00
|
|
|
|
if (do_alu_calc) begin
|
2019-02-14 14:35:23 +01:00
|
|
|
|
`ifdef SIM
|
|
|
|
|
if (alu_debug)
|
2019-02-15 16:58:38 +01:00
|
|
|
|
$display("ALU_CALC 2: run %b | done %b | stall %b | op %d | f %h | c %h | l %h | dest %d | src1 %h | src2 %h | p_carry %b",
|
|
|
|
|
alu_run, alu_done, o_alu_stall_dec, alu_op, f_first, f_cur, f_last, reg_dest, p_src1, p_src2, p_carry);
|
2019-02-14 14:35:23 +01:00
|
|
|
|
if (alu_debug_jump)
|
2019-02-15 16:58:38 +01:00
|
|
|
|
$display("ALU_JUMP 2: run %b | done %b | stall %b | op %d | f %h | c %h | l %h | jbs %5h | jof %5h | jpc %5h | fin %b",
|
|
|
|
|
alu_run, alu_done, o_alu_stall_dec, alu_op, f_first, f_cur, f_last, jump_bse, jump_off, jump_pc, alu_finish);
|
2019-02-14 08:59:04 +01:00
|
|
|
|
`endif
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
|
|
|
|
case (alu_op)
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`ALU_OP_JMP_REL3,
|
|
|
|
|
`ALU_OP_JMP_REL4,
|
|
|
|
|
`ALU_OP_JMP_ABS5:
|
|
|
|
|
if (alu_start)
|
|
|
|
|
jump_off <= { 16'b0, p_src1 };
|
2019-02-14 14:35:23 +01:00
|
|
|
|
endcase
|
|
|
|
|
|
2019-02-15 16:58:38 +01:00
|
|
|
|
// main case
|
2019-02-14 14:35:23 +01:00
|
|
|
|
case (alu_op)
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`ALU_OP_ZERO: c_res1 <= 0;
|
|
|
|
|
`ALU_OP_EXCH:
|
|
|
|
|
begin
|
|
|
|
|
c_res1 <= p_src2;
|
|
|
|
|
c_res2 <= p_src1;
|
|
|
|
|
end
|
|
|
|
|
`ALU_OP_COPY,
|
|
|
|
|
`ALU_OP_RST_BIT,
|
|
|
|
|
`ALU_OP_SET_BIT: c_res1 <= p_src1;
|
|
|
|
|
`ALU_OP_2CMPL:
|
|
|
|
|
begin
|
|
|
|
|
c_carry <= (~p_src1 == 4'hf) && p_carry ;
|
2019-02-16 22:38:44 +01:00
|
|
|
|
c_res1 <= ~p_src1 + {3'b000, p_carry};
|
|
|
|
|
is_zero <= ((~p_src1 + {3'b000, p_carry}) == 0) && alu_start?1:is_zero;
|
2019-02-15 16:58:38 +01:00
|
|
|
|
end
|
2019-02-16 12:17:40 +01:00
|
|
|
|
`ALU_OP_ADD:
|
2019-02-16 22:38:44 +01:00
|
|
|
|
{c_carry, c_res1} <= p_src1 + p_src2 + {4'b0000, p_carry};
|
2019-02-16 12:17:40 +01:00
|
|
|
|
`ALU_OP_TEST_NEQ:
|
|
|
|
|
c_carry <= !(p_src1 == p_src2) || p_carry;
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`ALU_OP_JMP_REL3,
|
|
|
|
|
`ALU_OP_JMP_REL4,
|
|
|
|
|
`ALU_OP_JMP_ABS5: jump_off[f_cur*4+:4] <= p_src1;
|
|
|
|
|
`ALU_OP_CLR_MASK: c_res1 <= p_src1 & ~p_src2;
|
|
|
|
|
default: $display("#### CALC 2 UNHANDLED OPERATION %0d", alu_op);
|
2019-02-14 14:35:23 +01:00
|
|
|
|
endcase
|
|
|
|
|
|
|
|
|
|
case (alu_op)
|
|
|
|
|
`ALU_OP_JMP_REL3: if (alu_finish)
|
2019-02-15 16:58:38 +01:00
|
|
|
|
jump_off <= { {8{p_src1[3]}}, p_src1, jump_off[7:0] };
|
2019-02-14 14:35:23 +01:00
|
|
|
|
`ALU_OP_JMP_REL4: if (alu_finish)
|
2019-02-15 16:58:38 +01:00
|
|
|
|
jump_off <= { {4{p_src1[3]}}, p_src1, jump_off[11:0] };
|
2019-02-13 22:43:04 +01:00
|
|
|
|
endcase
|
2019-02-14 22:14:52 +01:00
|
|
|
|
|
|
|
|
|
// $display("-------C- SRC1 %b %h | ~SRC1 %b %h | PC %b | RES1 %b %h | CC %b",
|
|
|
|
|
// p_src1, p_src1, ~p_src1, ~p_src1, p_carry,
|
|
|
|
|
// (~p_src1) + p_carry, (~p_src1) + p_carry,
|
|
|
|
|
// (~p_src1) == 4'hf );
|
2019-02-13 22:43:04 +01:00
|
|
|
|
end
|
2019-02-16 11:08:34 +01:00
|
|
|
|
|
|
|
|
|
if (do_go_init) begin
|
2019-02-16 12:17:40 +01:00
|
|
|
|
// $display("GO_INIT 3: imm %h", i_imm_value);
|
2019-02-16 11:08:34 +01:00
|
|
|
|
jump_off <= { {16{1'b0}}, i_imm_value};
|
|
|
|
|
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-15 07:07:55 +01:00
|
|
|
|
|
2019-02-16 11:08:34 +01:00
|
|
|
|
if (do_alu_save || do_go_prep) begin
|
|
|
|
|
if (alu_debug_jump) begin
|
|
|
|
|
`ifdef SIM
|
|
|
|
|
$display({"ALU_JUMP 3: run %b | done %b | stall %b | op %d | f %h | ",
|
|
|
|
|
"c %h | l %h | bse %5h | jof %5h | jpc %5h | fin %b"},
|
|
|
|
|
alu_run, alu_done, o_alu_stall_dec, alu_op, f_first, f_cur,
|
|
|
|
|
f_last, jump_bse, jump_off, jump_pc, alu_finish);
|
2019-02-14 08:59:04 +01:00
|
|
|
|
`endif
|
2019-02-16 11:08:34 +01:00
|
|
|
|
end
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
if (do_alu_save) begin
|
|
|
|
|
`ifdef SIM
|
|
|
|
|
if (alu_debug) begin
|
2019-02-16 12:17:40 +01:00
|
|
|
|
$display({"ALU_SAVE 3: run %b | done %b | stall %b | op %d | f %h | c %h | l %h |",
|
|
|
|
|
" dest %d | cres1 %h | cres2 %h | psrc1 %h | psrc2 %h | c_carry %b"},
|
|
|
|
|
alu_run, alu_done, o_alu_stall_dec, alu_op,
|
|
|
|
|
f_first, f_cur, f_last, reg_dest, c_res1, c_res2, p_src1, p_src2, c_carry);
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-14 22:14:52 +01:00
|
|
|
|
// $display("-------S- SRC1 %b %h | ~SRC1 %b %h | PC %b | RES1 %b %h | CC %b",
|
|
|
|
|
// p_src1, p_src1, ~p_src1, ~p_src1, p_carry,
|
|
|
|
|
// (~p_src1) + p_carry, (~p_src1) + p_carry,
|
|
|
|
|
// (~p_src1) == 4'hf );
|
2019-02-16 11:08:34 +01:00
|
|
|
|
end
|
|
|
|
|
`endif
|
2019-02-14 22:14:52 +01:00
|
|
|
|
|
2019-02-13 22:43:04 +01:00
|
|
|
|
case (alu_op)
|
2019-02-14 22:54:54 +01:00
|
|
|
|
`ALU_OP_ZERO,
|
|
|
|
|
`ALU_OP_COPY,
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`ALU_OP_EXCH, // does the first assign
|
2019-02-14 22:54:54 +01:00
|
|
|
|
`ALU_OP_2CMPL,
|
2019-02-16 07:35:06 +01:00
|
|
|
|
`ALU_OP_ADD,
|
2019-02-14 22:54:54 +01:00
|
|
|
|
`ALU_OP_CLR_MASK:
|
|
|
|
|
case (reg_dest)
|
2019-02-15 16:58:38 +01:00
|
|
|
|
`ALU_REG_A: A[f_cur*4+:4] <= c_res1;
|
|
|
|
|
`ALU_REG_B: B[f_cur*4+:4] <= c_res1;
|
|
|
|
|
`ALU_REG_C: C[f_cur*4+:4] <= c_res1;
|
|
|
|
|
`ALU_REG_D: D[f_cur*4+:4] <= c_res1;
|
|
|
|
|
`ALU_REG_D0: D0[f_cur*4+:4] <= c_res1;
|
|
|
|
|
`ALU_REG_D1: D1[f_cur*4+:4] <= c_res1;
|
|
|
|
|
`ALU_REG_ST: ST[f_cur*4+:4] <= c_res1;
|
|
|
|
|
`ALU_REG_P: P <= c_res1;
|
|
|
|
|
`ALU_REG_HST: HST <= c_res1;
|
2019-02-14 22:54:54 +01:00
|
|
|
|
endcase
|
|
|
|
|
`ALU_OP_RST_BIT,
|
|
|
|
|
`ALU_OP_SET_BIT:
|
|
|
|
|
case (reg_dest)
|
|
|
|
|
`ALU_REG_ST: ST[c_res1] <= alu_op==`ALU_OP_SET_BIT?1:0;
|
|
|
|
|
default: $display("#### ALU_SAVE invalid register %0d for op %0d", reg_dest, alu_op);
|
|
|
|
|
endcase
|
2019-02-16 12:17:40 +01:00
|
|
|
|
`ALU_OP_TEST_EQ,
|
|
|
|
|
`ALU_OP_TEST_NEQ,
|
2019-02-15 07:07:55 +01:00
|
|
|
|
`ALU_OP_JMP_REL3,
|
|
|
|
|
`ALU_OP_JMP_REL4,
|
|
|
|
|
`ALU_OP_JMP_ABS5: begin end // nothing to save, handled by PC management below
|
2019-02-14 22:54:54 +01:00
|
|
|
|
default: $display("#### ALU_SAVE UNHANDLED OP %0d", alu_op);
|
2019-02-13 22:43:04 +01:00
|
|
|
|
endcase
|
2019-02-15 16:58:38 +01:00
|
|
|
|
|
2019-02-16 12:17:40 +01:00
|
|
|
|
/*
|
|
|
|
|
* in case of exch, we need to update src2 to finish the exchange
|
|
|
|
|
*/
|
2019-02-15 16:58:38 +01:00
|
|
|
|
case (alu_op)
|
|
|
|
|
`ALU_OP_EXCH: // 2nd assign, with src2
|
|
|
|
|
case (reg_src2)
|
|
|
|
|
`ALU_REG_A: A[f_cur*4+:4] <= c_res2;
|
|
|
|
|
`ALU_REG_B: B[f_cur*4+:4] <= c_res2;
|
|
|
|
|
`ALU_REG_C: C[f_cur*4+:4] <= c_res2;
|
|
|
|
|
`ALU_REG_D: D[f_cur*4+:4] <= c_res2;
|
|
|
|
|
// `ALU_REG_D0: D0[f_start*4+:4] <= c_res2;
|
|
|
|
|
// `ALU_REG_D1: D1[f_start*4+:4] <= c_res2;
|
|
|
|
|
// `ALU_REG_ST: ST[f_start*4+:4] <= c_res2;
|
|
|
|
|
// `ALU_REG_P: P <= c_res2;
|
|
|
|
|
// `ALU_REG_HST: HST <= c_res2;
|
|
|
|
|
endcase
|
|
|
|
|
endcase
|
2019-02-15 07:07:55 +01:00
|
|
|
|
end
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
2019-02-16 12:17:40 +01:00
|
|
|
|
/*
|
|
|
|
|
* update carry
|
|
|
|
|
*/
|
2019-02-15 07:07:55 +01:00
|
|
|
|
if (do_alu_save) begin
|
2019-02-14 22:14:52 +01:00
|
|
|
|
case (alu_op)
|
|
|
|
|
`ALU_OP_2CMPL: CARRY <= !is_zero;
|
2019-02-16 12:17:40 +01:00
|
|
|
|
`ALU_OP_TEST_EQ,
|
|
|
|
|
`ALU_OP_TEST_NEQ: CARRY <= c_carry;
|
2019-02-14 22:14:52 +01:00
|
|
|
|
endcase
|
2019-02-15 07:07:55 +01:00
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
// do whatever is requested by the RTN instruction
|
2019-02-16 22:38:44 +01:00
|
|
|
|
if (alu_active && i_ins_rtn) begin
|
2019-02-15 07:07:55 +01:00
|
|
|
|
|
|
|
|
|
if (i_set_xm)
|
|
|
|
|
HST[`ALU_HST_XM] <= 1;
|
|
|
|
|
|
|
|
|
|
if (i_set_carry)
|
|
|
|
|
CARRY <= i_carry_val;
|
2019-02-14 22:14:52 +01:00
|
|
|
|
|
2019-02-12 12:43:36 +01:00
|
|
|
|
end
|
2019-02-15 07:07:55 +01:00
|
|
|
|
|
2019-02-14 08:59:04 +01:00
|
|
|
|
end
|
|
|
|
|
|
2019-02-15 07:07:55 +01:00
|
|
|
|
/*****************************************************************************
|
|
|
|
|
*
|
|
|
|
|
* Handles all changes to PC
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
2019-02-16 22:38:44 +01:00
|
|
|
|
reg [0:0] just_reset;
|
2019-02-14 08:59:04 +01:00
|
|
|
|
wire [19:0] next_pc;
|
2019-02-16 11:08:34 +01:00
|
|
|
|
wire [19:0] goyes_off;
|
|
|
|
|
wire [19:0] goyes_pc;
|
2019-02-14 22:14:52 +01:00
|
|
|
|
wire [0:0] update_pc;
|
2019-02-16 12:17:40 +01:00
|
|
|
|
wire [0:0] uncond_jmp;
|
2019-02-16 11:08:34 +01:00
|
|
|
|
wire [0:0] pop_pc;
|
2019-02-16 12:17:40 +01:00
|
|
|
|
wire [0:0] reload_pc;
|
2019-02-14 22:14:52 +01:00
|
|
|
|
wire [0:0] push_pc;
|
|
|
|
|
|
|
|
|
|
assign next_pc = (is_alu_op_jump && alu_finish)?jump_pc:PC + 1;
|
2019-02-16 11:08:34 +01:00
|
|
|
|
assign goyes_off = {{12{i_imm_value[3]}}, i_imm_value, jump_off[3:0]};
|
|
|
|
|
assign goyes_pc = jump_bse + goyes_off;
|
|
|
|
|
|
2019-02-16 22:38:44 +01:00
|
|
|
|
assign update_pc = !o_alu_stall_dec || is_alu_op_jump || just_reset;
|
2019-02-16 12:17:40 +01:00
|
|
|
|
assign uncond_jmp = is_alu_op_jump && alu_done;
|
|
|
|
|
assign pop_pc = i_pop && i_ins_rtn &&
|
|
|
|
|
((!i_ins_test_go) ||
|
|
|
|
|
(i_ins_test_go && c_carry));
|
2019-02-16 22:38:44 +01:00
|
|
|
|
assign reload_pc = uncond_jmp || pop_pc || just_reset;
|
2019-02-16 12:17:40 +01:00
|
|
|
|
assign push_pc = update_pc && i_push && alu_finish;
|
2019-02-14 22:14:52 +01:00
|
|
|
|
|
2019-02-14 08:59:04 +01:00
|
|
|
|
always @(posedge i_clk) begin
|
2019-02-16 22:38:44 +01:00
|
|
|
|
if (i_reset) begin
|
|
|
|
|
PC <= ~0;
|
|
|
|
|
just_reset <= 1;
|
|
|
|
|
end
|
|
|
|
|
|
2019-02-14 08:59:04 +01:00
|
|
|
|
|
2019-02-15 07:07:55 +01:00
|
|
|
|
/*
|
|
|
|
|
* some debug information
|
|
|
|
|
*/
|
|
|
|
|
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`ifdef SIM
|
2019-02-14 14:35:23 +01:00
|
|
|
|
if (do_alu_shpc && alu_debug_pc) begin
|
|
|
|
|
if (!o_alu_stall_dec)
|
|
|
|
|
$display("ALU_SHPC 0: pc %5h", PC);
|
2019-02-14 08:59:04 +01:00
|
|
|
|
if (o_alu_stall_dec)
|
|
|
|
|
$display("ALU_SHPC 0: STALL");
|
|
|
|
|
end
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`endif
|
2019-02-14 08:59:04 +01:00
|
|
|
|
|
2019-02-14 14:35:23 +01:00
|
|
|
|
/*
|
2019-02-15 07:07:55 +01:00
|
|
|
|
* updates the PC
|
2019-02-14 14:35:23 +01:00
|
|
|
|
*/
|
2019-02-14 08:59:04 +01:00
|
|
|
|
if (do_alu_pc) begin
|
2019-02-16 22:38:44 +01:00
|
|
|
|
// $display("DO ALU PC");
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`ifdef SIM
|
2019-02-14 14:35:23 +01:00
|
|
|
|
if (alu_debug_pc)
|
2019-02-16 11:08:34 +01:00
|
|
|
|
$display({"ALU_PC 3: !stl %b | nx %5h | done %b | fin %b | ",
|
|
|
|
|
"jmp %b | ins_rtn %b | push %b | ",
|
|
|
|
|
"imm %h | j_bs %h | go_off %h | go_pc %h"},
|
|
|
|
|
!o_alu_stall_dec, next_pc, alu_done, alu_finish,
|
|
|
|
|
is_alu_op_jump, i_ins_rtn, i_push,
|
|
|
|
|
i_imm_value, jump_bse, goyes_off, goyes_pc);
|
2019-02-16 12:17:40 +01:00
|
|
|
|
if (reload_pc) begin
|
2019-02-15 09:00:44 +01:00
|
|
|
|
$display(".---------------------------------.");
|
|
|
|
|
$display("| SHOULD TELL THE BUS CONTROLLER |");
|
|
|
|
|
$display("| TO LOAD PC INTO MODULES' PC REG |");
|
|
|
|
|
$display("`---------------------------------´");
|
|
|
|
|
end
|
2019-02-14 15:27:17 +01:00
|
|
|
|
`endif
|
2019-02-16 22:38:44 +01:00
|
|
|
|
|
|
|
|
|
// if we just came out of reset, simulate a long jump to 0
|
|
|
|
|
if (just_reset) begin
|
|
|
|
|
$display("ALU_RSET 3: simulating GOVLNG to PC");
|
|
|
|
|
// cancel this signal for good once reset is gone
|
|
|
|
|
just_reset <= 0;
|
|
|
|
|
end
|
|
|
|
|
|
2019-02-15 09:00:44 +01:00
|
|
|
|
// this may do wierd things with C=RSTK...
|
2019-02-14 22:14:52 +01:00
|
|
|
|
if (update_pc) begin
|
2019-02-16 11:08:34 +01:00
|
|
|
|
PC <= pop_pc ? RSTK[rstk_ptr-1] : next_pc;
|
2019-02-15 07:07:55 +01:00
|
|
|
|
end
|
|
|
|
|
|
2019-02-16 22:38:44 +01:00
|
|
|
|
if (reload_pc) begin
|
|
|
|
|
$display("ALU_PC 3: $$$$ RELOADING PC $$$$");
|
|
|
|
|
o_bus_address <= pop_pc ? RSTK[rstk_ptr-1] : next_pc;
|
|
|
|
|
o_bus_load_pc <= 1;
|
|
|
|
|
end
|
2019-02-16 12:17:40 +01:00
|
|
|
|
|
|
|
|
|
// $display("pop %b && rtn %b && ((!go %b) || (go %b && c %b))",
|
|
|
|
|
// i_pop, i_ins_rtn, !i_ins_test_go, i_ins_test_go, c_carry);
|
2019-02-16 11:08:34 +01:00
|
|
|
|
if (pop_pc) begin
|
|
|
|
|
$display("POP RSTK[%0d] to PC %5h", rstk_ptr-1, RSTK[rstk_ptr - 1]);
|
2019-02-15 07:07:55 +01:00
|
|
|
|
RSTK[rstk_ptr - 1] <= 0;
|
|
|
|
|
rstk_ptr <= rstk_ptr - 1;
|
2019-02-14 14:35:23 +01:00
|
|
|
|
end
|
2019-02-14 22:14:52 +01:00
|
|
|
|
|
|
|
|
|
if (push_pc) begin
|
|
|
|
|
$display("PUSH PC %5h to RSTK[%0d]", PC, rstk_ptr);
|
|
|
|
|
RSTK[rstk_ptr] <= PC;
|
|
|
|
|
rstk_ptr <= rstk_ptr + 1;
|
|
|
|
|
end
|
2019-02-14 08:59:04 +01:00
|
|
|
|
end
|
2019-02-16 22:38:44 +01:00
|
|
|
|
|
|
|
|
|
// deactivate o_load_pc
|
|
|
|
|
if (do_busclean && o_bus_load_pc)
|
|
|
|
|
o_bus_load_pc <= 0;
|
|
|
|
|
|
2019-02-12 12:43:36 +01:00
|
|
|
|
end
|
2019-02-12 11:22:55 +01:00
|
|
|
|
|
2019-02-15 07:07:55 +01:00
|
|
|
|
/*****************************************************************************
|
|
|
|
|
*
|
|
|
|
|
* execute SETHEX and SETDEC
|
|
|
|
|
*
|
|
|
|
|
****************************************************************************/
|
|
|
|
|
|
|
|
|
|
always @(posedge i_clk)
|
|
|
|
|
// changing calculation modes
|
2019-02-15 09:00:44 +01:00
|
|
|
|
if (do_alu_mode) begin
|
2019-02-15 07:07:55 +01:00
|
|
|
|
$display("SETTING MODE TO %s", i_mode_dec?"DEC":"HEX");
|
|
|
|
|
DEC <= i_mode_dec;
|
|
|
|
|
end
|
|
|
|
|
|
|
|
|
|
|
2019-02-12 11:22:55 +01:00
|
|
|
|
endmodule
|
2019-02-14 15:27:17 +01:00
|
|
|
|
|
|
|
|
|
`endif
|