2019-02-14 15:27:17 +01:00
|
|
|
|
2019-02-20 09:17:37 +01:00
|
|
|
/*
|
2019-02-20 09:19:00 +01:00
|
|
|
(c) Raphaël Jacquot 2019
|
|
|
|
|
2019-02-20 09:17:37 +01:00
|
|
|
This file is part of hp_saturn.
|
|
|
|
|
|
|
|
hp_saturn is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
any later version.
|
|
|
|
|
|
|
|
hp_saturn is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with Foobar. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
`default_nettype none //
|
|
|
|
|
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
|
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
`define ALU_DEBUG 1'b1
|
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-19 16:17:16 +01:00
|
|
|
i_clk_ph,
|
2019-02-17 12:05:38 +01:00
|
|
|
i_cycle_ctr,
|
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,
|
2019-02-21 16:55:08 +01:00
|
|
|
i_bus_data_ptr,
|
|
|
|
o_bus_data_nibl,
|
2019-02-20 09:17:37 +01:00
|
|
|
o_bus_xfr_cnt,
|
2019-02-21 16:55:08 +01:00
|
|
|
i_bus_nibble_in,
|
|
|
|
o_bus_nibble_out,
|
|
|
|
|
2019-02-17 19:29:39 +01:00
|
|
|
o_bus_pc_read,
|
2019-02-18 07:43:36 +01:00
|
|
|
o_bus_dp_read,
|
2019-02-17 19:29:39 +01:00
|
|
|
o_bus_dp_write,
|
2019-02-16 22:38:44 +01:00
|
|
|
o_bus_load_pc,
|
|
|
|
o_bus_load_dp,
|
2019-02-17 19:29:39 +01:00
|
|
|
o_bus_config,
|
2019-02-21 16:55:08 +01:00
|
|
|
i_bus_done,
|
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,
|
2019-02-17 19:29:39 +01:00
|
|
|
i_ins_config,
|
2019-02-20 09:17:37 +01:00
|
|
|
i_ins_mem_xfr,
|
|
|
|
i_xfr_dir_out,
|
2019-02-17 19:29:39 +01:00
|
|
|
i_ins_unconfig,
|
2019-02-15 07:07:55 +01:00
|
|
|
|
|
|
|
i_mode_dec,
|
|
|
|
i_set_xm,
|
|
|
|
i_set_carry,
|
2019-02-18 17:38:25 +01:00
|
|
|
i_test_carry,
|
2019-02-15 07:07:55 +01:00
|
|
|
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-19 16:17:16 +01:00
|
|
|
input wire [1:0] i_clk_ph;
|
2019-02-17 12:05:38 +01:00
|
|
|
input wire [31:0] i_cycle_ctr;
|
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;
|
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
/*
|
|
|
|
* I/O to the bus controller
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* data to and from the bus controller */
|
2019-02-16 22:38:44 +01:00
|
|
|
output reg [19:0] o_bus_address;
|
2019-02-21 16:55:08 +01:00
|
|
|
input wire [3:0] i_bus_data_ptr;
|
|
|
|
output reg [3:0] o_bus_data_nibl;
|
2019-02-20 09:17:37 +01:00
|
|
|
output reg [3:0] o_bus_xfr_cnt;
|
2019-02-21 16:55:08 +01:00
|
|
|
input wire [3:0] i_bus_nibble_in;
|
|
|
|
output reg [3:0] o_bus_nibble_out;
|
|
|
|
|
|
|
|
/* control lines to the bus controller */
|
2019-02-17 19:29:39 +01:00
|
|
|
output reg [0:0] o_bus_pc_read;
|
2019-02-18 07:43:36 +01:00
|
|
|
output reg [0:0] o_bus_dp_read;
|
2019-02-17 19:29:39 +01:00
|
|
|
output reg [0:0] o_bus_dp_write;
|
2019-02-16 22:38:44 +01:00
|
|
|
output reg [0:0] o_bus_load_pc;
|
|
|
|
output reg [0:0] o_bus_load_dp;
|
2019-02-17 19:29:39 +01:00
|
|
|
output reg [0:0] o_bus_config;
|
2019-02-21 16:55:08 +01:00
|
|
|
input wire [0:0] i_bus_done;
|
2019-02-12 12:43:36 +01:00
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
/*
|
|
|
|
* lines from the decoder
|
|
|
|
*/
|
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;
|
|
|
|
|
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;
|
2019-02-17 19:29:39 +01:00
|
|
|
input wire [0:0] i_ins_config;
|
2019-02-20 09:17:37 +01:00
|
|
|
input wire [0:0] i_ins_mem_xfr;
|
|
|
|
input wire [0:0] i_xfr_dir_out;
|
2019-02-17 19:29:39 +01:00
|
|
|
input wire [0:0] i_ins_unconfig;
|
2019-02-15 07:07:55 +01:00
|
|
|
|
|
|
|
input wire [0:0] i_mode_dec;
|
|
|
|
input wire [0:0] i_set_xm;
|
|
|
|
input wire [0:0] i_set_carry;
|
2019-02-18 17:38:25 +01:00
|
|
|
input wire [0:0] i_test_carry;
|
2019-02-15 07:07:55 +01:00
|
|
|
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-21 16:55:08 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* clock phases definitions
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-02-19 16:17:16 +01:00
|
|
|
wire [1:0] phase;
|
|
|
|
assign phase = i_clk_ph + 3;
|
2019-02-14 14:35:23 +01:00
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
wire [0:0] phase_0;
|
|
|
|
wire [0:0] phase_1;
|
|
|
|
wire [0:0] phase_2;
|
|
|
|
wire [0:0] phase_3;
|
|
|
|
|
|
|
|
assign phase_0 = (phase == 0);
|
|
|
|
assign phase_1 = (phase == 1);
|
|
|
|
assign phase_2 = (phase == 2);
|
|
|
|
assign phase_3 = (phase == 3);
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
wire alu_active;
|
|
|
|
|
|
|
|
assign alu_active = !i_reset && !i_stalled;
|
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* internal registers
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-02-14 14:35:23 +01:00
|
|
|
/* 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
|
|
|
|
|
|
|
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
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
reg [3:0] D0[0:4];
|
|
|
|
reg [3:0] D1[0:4];
|
2019-02-13 22:43:04 +01:00
|
|
|
|
2019-02-17 23:05:33 +01:00
|
|
|
//reg [63:0] A;
|
|
|
|
reg [3:0] A[0:15];
|
|
|
|
reg [3:0] B[0:15];
|
|
|
|
reg [3:0] C[0:15];
|
|
|
|
reg [3:0] D[0:15];
|
|
|
|
|
|
|
|
reg [3:0] R0[0:15];
|
|
|
|
reg [3:0] R1[0:15];
|
|
|
|
reg [3:0] R2[0:15];
|
|
|
|
reg [3:0] R3[0:15];
|
|
|
|
reg [3:0] R4[0:15];
|
2019-02-13 22:43:04 +01:00
|
|
|
|
|
|
|
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
|
|
|
|
end
|
2019-02-12 11:22:55 +01:00
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* ALU debug modes
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
/******************************************************************************
|
|
|
|
*
|
|
|
|
* states decoding
|
|
|
|
*
|
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* ALU : modes of operation
|
|
|
|
*
|
|
|
|
* - classical alu used for calculations
|
|
|
|
* - data transfer to and from memory
|
|
|
|
* - jump calculations
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
// the ALU is in memory transfer mode
|
|
|
|
reg [0:0] f_mode_xfr;
|
2019-02-21 22:44:55 +01:00
|
|
|
reg [0:0] f_mode_load_ptr;
|
|
|
|
reg [0:0] f_mode_ldreg;
|
|
|
|
reg [0:0] f_mode_jmp;
|
|
|
|
reg [0:0] f_mode_alu;
|
2019-02-21 16:55:08 +01:00
|
|
|
|
|
|
|
wire [0:0] mode_set;
|
2019-02-21 22:44:55 +01:00
|
|
|
wire [0:0] stall_modes;
|
|
|
|
wire [0:0] alu_start_ev;
|
2019-02-21 16:55:08 +01:00
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
wire [0:0] start_in_xfr_mode;
|
|
|
|
wire [0:0] start_in_load_ptr_mode;
|
|
|
|
wire [0:0] start_in_ldreg_mode;
|
2019-02-22 08:22:32 +01:00
|
|
|
wire [0:0] start_in_p_mode;
|
2019-02-21 22:44:55 +01:00
|
|
|
wire [0:0] start_in_jmp_mode;
|
|
|
|
wire [0:0] start_in_alu_mode;
|
|
|
|
|
|
|
|
assign mode_set = f_mode_xfr || f_mode_load_ptr || f_mode_ldreg || f_mode_jmp || f_mode_alu;
|
|
|
|
assign stall_modes = f_mode_xfr || f_mode_alu;
|
|
|
|
|
|
|
|
assign alu_start_ev = alu_active && phase_3;
|
|
|
|
assign start_in_xfr_mode = alu_start_ev && i_ins_mem_xfr && !mode_set;
|
|
|
|
assign start_in_load_ptr_mode = alu_start_ev && i_ins_alu_op && op_copy && dest_ptr && src1_IMM && !mode_set;
|
|
|
|
assign start_in_ldreg_mode = alu_start_ev && i_ins_alu_op && op_copy && dest_A_C && src1_IMM && !mode_set;
|
2019-02-22 08:22:32 +01:00
|
|
|
// there is no need to set a flag for that one
|
|
|
|
assign start_in_p_mode = alu_start_ev && i_ins_alu_op && op_1_cycle_p && !mode_set;
|
2019-02-21 22:44:55 +01:00
|
|
|
assign start_in_jmp_mode = alu_start_ev && i_ins_alu_op && op_jump && src1_IMM && !mode_set;
|
|
|
|
// this is the last mode in the queue, called when no others have been recognized
|
|
|
|
assign start_in_alu_mode = alu_start_ev && i_ins_alu_op && !mode_set &&
|
2019-02-22 08:22:32 +01:00
|
|
|
!(start_in_load_ptr_mode || start_in_ldreg_mode ||
|
|
|
|
start_in_p_mode || start_in_jmp_mode);
|
|
|
|
|
|
|
|
|
|
|
|
assign o_alu_stall_dec = alu_initializing || i_stalled ||
|
|
|
|
stall_modes;
|
|
|
|
|
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
/*
|
|
|
|
* wires for all modes
|
|
|
|
*/
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
/* operation */
|
|
|
|
|
|
|
|
wire [0:0] op_copy;
|
2019-02-22 08:22:32 +01:00
|
|
|
wire [0:0] op_inc;
|
|
|
|
wire [0:0] op_dec;
|
2019-02-21 22:44:55 +01:00
|
|
|
wire [0:0] op_jmp_rel_2;
|
|
|
|
wire [0:0] op_jmp_rel_3;
|
|
|
|
wire [0:0] op_jmp_rel_4;
|
|
|
|
wire [0:0] op_jmp_abs_5;
|
2019-02-22 08:22:32 +01:00
|
|
|
|
|
|
|
wire [0:0] op_inc_p;
|
|
|
|
wire [0:0] op_dec_p;
|
|
|
|
wire [0:0] op_set_p;
|
|
|
|
wire [0:0] op_copy_p_to_c;
|
|
|
|
wire [0:0] op_copy_c_to_p;
|
|
|
|
|
|
|
|
wire [0:0] op_1_cycle_p;
|
2019-02-21 22:44:55 +01:00
|
|
|
wire [0:0] op_jump;
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
assign op_copy = (i_alu_op == `ALU_OP_COPY);
|
|
|
|
assign op_inc = (i_alu_op == `ALU_OP_INC);
|
|
|
|
assign op_dec = (i_alu_op == `ALU_OP_DEC);
|
|
|
|
assign op_jmp_rel_2 = (i_alu_op == `ALU_OP_JMP_REL2);
|
|
|
|
assign op_jmp_rel_3 = (i_alu_op == `ALU_OP_JMP_REL3);
|
|
|
|
assign op_jmp_rel_4 = (i_alu_op == `ALU_OP_JMP_REL4);
|
|
|
|
assign op_jmp_abs_5 = (i_alu_op == `ALU_OP_JMP_ABS5);
|
|
|
|
|
|
|
|
assign op_inc_p = op_inc && src1_P && dest_P;
|
|
|
|
assign op_dec_p = op_dec && src1_P && dest_P;
|
|
|
|
assign op_set_p = op_copy && src1_IMM && dest_P;
|
|
|
|
assign op_copy_p_to_c = op_copy && src1_P && dest_C;
|
|
|
|
assign op_copy_c_to_p = op_copy && src1_C && dest_P;
|
|
|
|
|
|
|
|
assign op_1_cycle_p = op_inc_p || op_dec_p || op_set_p || op_copy_p_to_c || op_copy_c_to_p;
|
|
|
|
assign op_jump = op_jmp_rel_2 || op_jmp_rel_3 || op_jmp_rel_4 || op_jmp_abs_5;
|
2019-02-21 22:44:55 +01:00
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
/* source 1 */
|
|
|
|
wire [0:0] src1_A;
|
|
|
|
wire [0:0] src1_C;
|
|
|
|
wire [0:0] src1_DAT0;
|
|
|
|
wire [0:0] src1_DAT1;
|
2019-02-22 08:22:32 +01:00
|
|
|
wire [0:0] src1_P;
|
2019-02-21 22:44:55 +01:00
|
|
|
wire [0:0] src1_IMM;
|
2019-02-21 16:55:08 +01:00
|
|
|
|
|
|
|
assign src1_A = (i_reg_src1 == `ALU_REG_A);
|
|
|
|
assign src1_C = (i_reg_src1 == `ALU_REG_C);
|
|
|
|
assign src1_DAT0 = (i_reg_src1 == `ALU_REG_DAT0);
|
|
|
|
assign src1_DAT1 = (i_reg_src1 == `ALU_REG_DAT1);
|
2019-02-22 08:22:32 +01:00
|
|
|
assign src1_P = (i_reg_src1 == `ALU_REG_P);
|
2019-02-21 22:44:55 +01:00
|
|
|
assign src1_IMM = (i_reg_src1 == `ALU_REG_IMM);
|
2019-02-21 16:55:08 +01:00
|
|
|
|
|
|
|
/* destination */
|
2019-02-21 22:44:55 +01:00
|
|
|
wire [0:0] dest_A;
|
|
|
|
wire [0:0] dest_C;
|
|
|
|
wire [0:0] dest_D0;
|
|
|
|
wire [0:0] dest_D1;
|
2019-02-21 16:55:08 +01:00
|
|
|
wire [0:0] dest_DAT0;
|
|
|
|
wire [0:0] dest_DAT1;
|
2019-02-22 08:22:32 +01:00
|
|
|
wire [0:0] dest_P;
|
2019-02-21 16:55:08 +01:00
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
assign dest_A = (i_reg_dest == `ALU_REG_A);
|
|
|
|
assign dest_C = (i_reg_dest == `ALU_REG_C);
|
|
|
|
assign dest_D0 = (i_reg_dest == `ALU_REG_D0);
|
|
|
|
assign dest_D1 = (i_reg_dest == `ALU_REG_D1);
|
2019-02-21 16:55:08 +01:00
|
|
|
assign dest_DAT0 = (i_reg_dest == `ALU_REG_DAT0);
|
|
|
|
assign dest_DAT1 = (i_reg_dest == `ALU_REG_DAT1);
|
2019-02-22 08:22:32 +01:00
|
|
|
assign dest_P = (i_reg_dest == `ALU_REG_P);
|
2019-02-21 16:55:08 +01:00
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
wire [0:0] dest_A_C;
|
|
|
|
wire [0:0] dest_ptr;
|
|
|
|
|
|
|
|
assign dest_A_C = dest_A || dest_C;
|
|
|
|
assign dest_ptr = dest_D0 || dest_D1;
|
|
|
|
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
/******************************************************************************
|
2019-02-21 22:44:55 +01:00
|
|
|
*
|
2019-02-22 08:22:32 +01:00
|
|
|
* stuff that doesn't need to access registers
|
2019-02-21 22:44:55 +01:00
|
|
|
*
|
2019-02-22 08:22:32 +01:00
|
|
|
*****************************************************************************/
|
|
|
|
|
|
|
|
reg [0:0] just_reset;
|
2019-02-21 22:44:55 +01:00
|
|
|
|
|
|
|
always @(posedge i_clk) begin
|
|
|
|
|
|
|
|
if (i_reset) begin
|
2019-02-22 08:22:32 +01:00
|
|
|
just_reset <= 1;
|
|
|
|
f_mode_xfr <= 0;
|
|
|
|
f_mode_load_ptr <= 0;
|
|
|
|
f_mode_ldreg <= 0;
|
|
|
|
f_mode_jmp <= 0;
|
2019-02-21 22:44:55 +01:00
|
|
|
f_mode_alu <= 0;
|
|
|
|
end
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
if (just_reset && !i_reset) begin
|
|
|
|
just_reset <= 0;
|
|
|
|
$display("ALU_INIT %0d: [%d] CLEARING JUST_RESET", phase, i_cycle_ctr);
|
|
|
|
end
|
|
|
|
|
|
|
|
/* register to memory transfer
|
|
|
|
*/
|
|
|
|
if (start_in_xfr_mode) begin
|
|
|
|
$display("ALU %0d: [%d] memory transfer started (i_ins_decoded %b)", phase, i_cycle_ctr, i_ins_decoded);
|
|
|
|
$display("ALU %0d: [%d] addr_src A %b | C %b | D0 %b | D1 %b | b1 %b | b0 %b | src %2b", phase, i_cycle_ctr,
|
|
|
|
addr_src_A, addr_src_C, addr_src_D0, addr_src_D1, addr_src_xfr_1, addr_src_xfr_0, addr_src);
|
|
|
|
$display("ALU %0d: [%d] stall the decoder",phase, i_cycle_ctr);
|
|
|
|
f_mode_xfr <= 1;
|
|
|
|
end
|
|
|
|
|
|
|
|
if (alu_active && f_mode_xfr && i_bus_done) begin
|
|
|
|
$display("ALU %0d: [%d] resetting variables after data transfer", phase, i_cycle_ctr);
|
|
|
|
f_mode_xfr <= 0;
|
|
|
|
end
|
|
|
|
|
|
|
|
/* load pointer register with value
|
|
|
|
*/
|
|
|
|
if (start_in_load_ptr_mode) begin
|
|
|
|
$display("ALU %0d: [%d] load_ptr mode started (i_ins_decoded %b)", phase, i_cycle_ctr, i_ins_decoded);
|
|
|
|
f_mode_load_ptr <= 1;
|
|
|
|
end
|
|
|
|
|
|
|
|
/* load register immediate with 1-16 nibbles
|
|
|
|
*/
|
|
|
|
if (start_in_ldreg_mode) begin
|
|
|
|
$display("ALU %0d: [%d] load register mode started (loading reg %c with %0d nibbles)",
|
|
|
|
phase, i_cycle_ctr, dest_A?"A":"C", i_field_last - i_field_start + 1);
|
|
|
|
f_mode_ldreg <= 1;
|
|
|
|
end
|
|
|
|
|
|
|
|
if (do_load_register_done) begin
|
|
|
|
$display("ALU %0d: [%d] resetting variables after loading register", phase, i_cycle_ctr);
|
|
|
|
f_mode_ldreg <= 0;
|
|
|
|
end
|
|
|
|
|
|
|
|
/* a jump instruction just appeared !
|
|
|
|
*/
|
|
|
|
if (start_in_jmp_mode) begin
|
|
|
|
$display("ALU %0d: [%d] jmp mode started (i_ins_decoded %b)", phase, i_cycle_ctr, i_ins_decoded);
|
|
|
|
f_mode_jmp <= 1;
|
|
|
|
end
|
|
|
|
|
2019-02-22 12:00:23 +01:00
|
|
|
if (do_apply_jump) begin
|
|
|
|
$display("ALU %0d: [%d] end of jmp mode", phase, i_cycle_ctr);
|
|
|
|
f_mode_jmp <= 0;
|
|
|
|
end
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
/* general ALU mode (when there is no optimization)
|
|
|
|
*/
|
2019-02-21 22:44:55 +01:00
|
|
|
if (start_in_alu_mode) begin
|
|
|
|
$display("ALU %0d: [%d] alu mode started (i_ins_decoded %b)", phase, i_cycle_ctr, i_ins_decoded);
|
|
|
|
$display("ALU %0d: [%d] stall the decoder",phase, i_cycle_ctr);
|
|
|
|
f_mode_alu <= 1;
|
|
|
|
end
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
end
|
2019-02-21 22:44:55 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
|
|
|
|
/* module 1:
|
|
|
|
* handles all alu mode timing
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
always @(posedge i_clk) begin
|
|
|
|
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
/* module 2:
|
|
|
|
* src1 and src2 can only be written here
|
|
|
|
* address can only be written here
|
|
|
|
* registers can only be read here
|
|
|
|
*
|
2019-02-21 16:55:08 +01:00
|
|
|
* wires specific to the XFR mode
|
|
|
|
*
|
|
|
|
* sources of address data used for XFR mode:
|
|
|
|
* - A [ PC=(A) ]
|
|
|
|
* - C [ CONFIG, UNCNFG, PC=(C) ]
|
|
|
|
* - D0 [ DAT0=reg, reg=DAT0 ]
|
|
|
|
* - D1 [ DAT1=reg, reg=DAT1 ]
|
|
|
|
*/
|
|
|
|
reg [3:0] xfr_data[0:15];
|
|
|
|
reg [3:0] data_counter;
|
|
|
|
|
|
|
|
// copy the address into the transfer buffer
|
|
|
|
|
|
|
|
wire [0:0] addr_src_A;
|
|
|
|
wire [0:0] addr_src_C;
|
|
|
|
wire [0:0] addr_src_D0;
|
|
|
|
wire [0:0] addr_src_D1;
|
|
|
|
wire [0:0] addr_src_xfr_0;
|
|
|
|
wire [0:0] addr_src_xfr_1;
|
|
|
|
wire [1:0] addr_src_xfr;
|
|
|
|
wire [1:0] addr_src;
|
2019-02-21 22:44:55 +01:00
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
wire [0:0] copy_done;
|
|
|
|
wire [0:0] copy_address;
|
|
|
|
wire [0:0] start_load_dp;
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
wire mode_xfr;
|
|
|
|
assign mode_xfr = start_in_xfr_mode || f_mode_xfr;
|
|
|
|
|
|
|
|
assign addr_src_A = (!mode_xfr) && src1_A;
|
|
|
|
assign addr_src_C = (!mode_xfr) && src1_C;
|
|
|
|
assign addr_src_D0 = ( mode_xfr) && (src1_DAT0 || dest_DAT0);
|
|
|
|
assign addr_src_D1 = ( mode_xfr) && (src1_DAT1 || dest_DAT1);
|
2019-02-21 16:55:08 +01:00
|
|
|
assign addr_src_xfr_0 = !addr_src_A && !addr_src_C && !addr_src_D0 && addr_src_D1;
|
|
|
|
assign addr_src_xfr_1 = !addr_src_A && !addr_src_C && (addr_src_D0 || addr_src_D1);
|
|
|
|
assign addr_src_xfr = {addr_src_xfr_1, addr_src_xfr_0};
|
2019-02-21 22:44:55 +01:00
|
|
|
assign addr_src = {2{mode_xfr}} & addr_src_xfr;
|
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
assign copy_done = data_counter == 5;
|
2019-02-21 22:44:55 +01:00
|
|
|
assign copy_address = alu_active && mode_xfr && !copy_done && !xfr_init_done;
|
2019-02-21 16:55:08 +01:00
|
|
|
assign start_load_dp = start_in_xfr_mode;
|
|
|
|
|
|
|
|
// now copy the data aligning the first nibble with index 0 of the buffer
|
|
|
|
// copy nibbles 0-4 at the end so as not to clobber the address set previously
|
|
|
|
// while the bus controller is sending it
|
|
|
|
|
|
|
|
reg [0:0] xfr_init_done;
|
|
|
|
reg [0:0] xfr_data_done;
|
|
|
|
wire [0:0] xfr_data_init;
|
|
|
|
wire [3:0] xfr_data_ctr;
|
|
|
|
wire [0:0] xfr_data_copy;
|
|
|
|
wire [0:0] xfr_copy_done;
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
assign xfr_data_init = alu_active && mode_xfr && copy_done && !xfr_init_done && !xfr_data_done && phase_3;
|
2019-02-21 16:55:08 +01:00
|
|
|
assign xfr_data_ctr = data_counter + i_field_start;
|
2019-02-21 22:44:55 +01:00
|
|
|
assign xfr_copy_done = alu_active && xfr_init_done && copy_done && !xfr_data_init && !xfr_data_done;
|
|
|
|
assign xfr_data_copy = alu_active && (xfr_data_init || xfr_init_done && !xfr_data_done && !copy_done && !xfr_copy_done);
|
2019-02-21 16:55:08 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* sources specific pointers
|
|
|
|
*/
|
|
|
|
wire [3:0] src1_ptr;
|
2019-02-21 22:44:55 +01:00
|
|
|
wire [2:0] src1a_ptr;
|
2019-02-21 16:55:08 +01:00
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
assign src1_ptr = ( {4{copy_address}} & data_counter |
|
|
|
|
{4{xfr_data_copy}} & xfr_data_ctr );
|
|
|
|
assign src1a_ptr = src1_ptr[2:0];
|
2019-02-21 16:55:08 +01:00
|
|
|
|
|
|
|
always @(posedge i_clk) begin
|
|
|
|
|
|
|
|
// initializes modes
|
|
|
|
if (i_reset) begin
|
|
|
|
data_counter <= 0;
|
|
|
|
xfr_init_done <= 0;
|
|
|
|
xfr_data_done <= 0;
|
|
|
|
end
|
|
|
|
|
|
|
|
// always update the data out to the controller
|
2019-02-21 22:44:55 +01:00
|
|
|
if (alu_active)
|
|
|
|
o_bus_data_nibl <= xfr_data[i_bus_data_ptr];
|
|
|
|
|
|
|
|
/****************************************************************************
|
|
|
|
*
|
|
|
|
* register to memory transfer.
|
|
|
|
*
|
|
|
|
***************************************************************************/
|
2019-02-21 16:55:08 +01:00
|
|
|
|
|
|
|
if (start_load_dp) begin
|
|
|
|
o_bus_load_dp <= 1;
|
|
|
|
end
|
|
|
|
|
|
|
|
if (copy_address) begin
|
|
|
|
$write("ALU %0d: [%d] xfr_data[%0d] = ", phase, i_cycle_ctr, data_counter);
|
|
|
|
case (addr_src)
|
|
|
|
2'b00: begin
|
|
|
|
$display("A[%0d] %h", src1_ptr, A[src1_ptr]);
|
|
|
|
xfr_data[data_counter] <= A[src1_ptr];
|
|
|
|
end
|
|
|
|
2'b01: begin
|
|
|
|
$display("C[%0d] %h", src1_ptr, C[src1_ptr]);
|
|
|
|
xfr_data[data_counter] <= C[src1_ptr];
|
|
|
|
end
|
|
|
|
2'b10: begin
|
2019-02-21 22:44:55 +01:00
|
|
|
$display("D0[%0d] %h", src1_ptr, D0[src1a_ptr]);
|
|
|
|
xfr_data[data_counter] <= D0[src1a_ptr];
|
2019-02-21 16:55:08 +01:00
|
|
|
end
|
|
|
|
2'b11: begin
|
2019-02-21 22:44:55 +01:00
|
|
|
$display("D1[%0d] %h", src1_ptr, D1[src1a_ptr]);
|
|
|
|
xfr_data[data_counter] <= D1[src1a_ptr];
|
2019-02-21 16:55:08 +01:00
|
|
|
end
|
|
|
|
default: begin end
|
|
|
|
endcase
|
|
|
|
data_counter <= data_counter + 1;
|
|
|
|
end
|
|
|
|
|
|
|
|
// do not need to update the data counter, which is already at 5
|
|
|
|
if (xfr_data_init) begin
|
|
|
|
$display("ALU %0d: [%d] initialize copy data | s %h | l %h | xdc %h",phase, i_cycle_ctr, i_field_start, i_field_last, xfr_data_ctr);
|
|
|
|
xfr_init_done <= 1;
|
|
|
|
end
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
// need to copy actual data
|
|
|
|
// two sources are possible, A and C, a conditional will suffice
|
2019-02-21 16:55:08 +01:00
|
|
|
if (xfr_data_copy) begin
|
2019-02-21 22:44:55 +01:00
|
|
|
$display("ALU %0d: [%d] copy data DAT[%b][%2d] <= %c[%2d] %h",
|
|
|
|
phase, i_cycle_ctr, dest_DAT1, data_counter, src1_A?"A":"C", src1_ptr, (src1_A)?A[src1_ptr]:C[src1_ptr]);
|
|
|
|
xfr_data[data_counter] <= (src1_A)?A[src1_ptr]:C[src1_ptr];
|
|
|
|
data_counter <= data_counter + 1;
|
2019-02-21 16:55:08 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
if (xfr_copy_done) begin
|
|
|
|
$display("ALU %0d: [%d] xfr_copy_done %h %b %b",phase, i_cycle_ctr, data_counter, xfr_init_done, xfr_data_done);
|
|
|
|
xfr_init_done <= 0;
|
|
|
|
xfr_data_done <= 1;
|
|
|
|
o_bus_load_dp <= 0;
|
|
|
|
// right on time to start the actual transfer
|
|
|
|
o_bus_dp_write <= i_xfr_dir_out;
|
|
|
|
o_bus_dp_read <= !i_xfr_dir_out;
|
|
|
|
o_bus_xfr_cnt <= (i_field_last - i_field_start);
|
|
|
|
end
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
/****************************************************************************
|
|
|
|
*
|
2019-02-21 16:55:08 +01:00
|
|
|
* reset all things that were changed
|
2019-02-21 22:44:55 +01:00
|
|
|
*
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
if (alu_active && i_bus_done) begin
|
2019-02-21 16:55:08 +01:00
|
|
|
$display("ALU %0d: [%d] bus controller is done, cleaning all variables used",phase, i_cycle_ctr);
|
|
|
|
/* variables for the XFR mode */
|
|
|
|
data_counter <= 0;
|
|
|
|
xfr_init_done <= 0;
|
|
|
|
xfr_data_done <= 0;
|
|
|
|
|
|
|
|
/* bus controller control lines */
|
|
|
|
o_bus_dp_write <= 0;
|
|
|
|
o_bus_dp_read <= 0;
|
|
|
|
o_bus_xfr_cnt <= 0;
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
/*
|
2019-02-22 08:22:32 +01:00
|
|
|
* moduls 4:
|
|
|
|
* registers can only be written to here
|
2019-02-21 22:44:55 +01:00
|
|
|
*
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
reg [0:0] alu_initializing;
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
reg [3:0] v_dest_counter;
|
|
|
|
reg [3:0] v_max_counter;
|
|
|
|
wire [2:0] v_dest_counter_ptr;
|
2019-02-22 08:22:32 +01:00
|
|
|
wire [1:0] v_dest_counter_hst;
|
2019-02-21 22:44:55 +01:00
|
|
|
assign v_dest_counter_ptr = v_dest_counter[2:0];
|
2019-02-22 08:22:32 +01:00
|
|
|
assign v_dest_counter_hst = v_dest_counter[1:0];
|
2019-02-21 22:44:55 +01:00
|
|
|
|
|
|
|
wire [0:0] do_load_pointer;
|
|
|
|
wire [0:0] do_load_pointer_done;
|
|
|
|
wire [0:0] do_load_register;
|
|
|
|
wire [0:0] do_load_register_done;
|
|
|
|
|
|
|
|
assign do_load_pointer = alu_active && phase_2 && f_mode_load_ptr && (v_dest_counter != 5);
|
|
|
|
assign do_load_pointer_done = alu_active && phase_3 && f_mode_load_ptr && (v_dest_counter == 5);
|
|
|
|
assign do_load_register = alu_active && phase_2 && f_mode_ldreg && (v_dest_counter != v_max_counter);
|
|
|
|
assign do_load_register_done = alu_active && phase_3 && f_mode_ldreg && (v_dest_counter == v_max_counter);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
always @(posedge i_clk) begin
|
|
|
|
|
|
|
|
if (i_reset) begin
|
2019-02-22 08:22:32 +01:00
|
|
|
/* initialization procedure */
|
|
|
|
alu_initializing <= 1;
|
|
|
|
CARRY <= 0;
|
|
|
|
P <= 0;
|
|
|
|
|
|
|
|
/* counters and flags */
|
|
|
|
v_dest_counter <= 0;
|
|
|
|
v_max_counter <= 0;
|
|
|
|
end
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Initialization of all registers
|
|
|
|
* This happens at the same time the first LOAD_PC command goes out
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
if (!i_reset && alu_initializing) begin
|
|
|
|
$display("ALU_INIT %0d: [%d] init %0d", phase, i_cycle_ctr, v_dest_counter);
|
|
|
|
A[v_dest_counter] <= 0;
|
|
|
|
B[v_dest_counter] <= 0;
|
|
|
|
C[v_dest_counter] <= 0;
|
|
|
|
D[v_dest_counter] <= 0;
|
|
|
|
D0[v_dest_counter_ptr] <= 0;
|
|
|
|
D1[v_dest_counter_ptr] <= 0;
|
|
|
|
R0[v_dest_counter] <= 0;
|
|
|
|
R1[v_dest_counter] <= 0;
|
|
|
|
R2[v_dest_counter] <= 0;
|
|
|
|
R3[v_dest_counter] <= 0;
|
|
|
|
R4[v_dest_counter] <= 0;
|
|
|
|
ST[v_dest_counter] <= 0;
|
|
|
|
HST[v_dest_counter_hst] <= 0;
|
|
|
|
alu_initializing <= (v_dest_counter != 15);
|
|
|
|
v_dest_counter <= v_dest_counter + 1;
|
2019-02-21 22:44:55 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* ptr mode
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
if (start_in_load_ptr_mode) begin
|
|
|
|
v_dest_counter <= 0;
|
|
|
|
end
|
|
|
|
|
|
|
|
if (do_load_pointer) begin
|
|
|
|
$display("ALU %0d: [%d] loading pointer D%b[%0d] <= %h", phase, i_cycle_ctr, dest_D1, v_dest_counter, i_bus_nibble_in);
|
|
|
|
case (dest_D1)
|
|
|
|
0: D0[v_dest_counter_ptr] <= i_bus_nibble_in;
|
|
|
|
1: D0[v_dest_counter_ptr] <= i_bus_nibble_in;
|
|
|
|
default: begin end
|
|
|
|
endcase
|
|
|
|
v_dest_counter <= v_dest_counter + 1;
|
|
|
|
end
|
|
|
|
|
|
|
|
if (do_load_pointer_done) begin
|
|
|
|
$display("ALU %0d: [%d] resetting variables after loading pointer", phase, i_cycle_ctr);
|
|
|
|
v_dest_counter <= 0;
|
|
|
|
f_mode_load_ptr <= 0;
|
|
|
|
end
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* load register mode
|
|
|
|
*
|
|
|
|
* LAHEX / LCHEX
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (start_in_ldreg_mode) begin
|
|
|
|
v_dest_counter <= 0;
|
|
|
|
v_max_counter <= i_field_last - i_field_start + 1;
|
|
|
|
end
|
|
|
|
|
|
|
|
if (do_load_register) begin
|
|
|
|
$display("ALU %0d: [%d] loading register %c[%0d] <= %h", phase, i_cycle_ctr, dest_A?"A":"C", v_dest_counter, i_bus_nibble_in);
|
|
|
|
case (dest_C)
|
|
|
|
0: A[v_dest_counter] <= i_bus_nibble_in;
|
|
|
|
1: C[v_dest_counter] <= i_bus_nibble_in;
|
|
|
|
default: begin end
|
|
|
|
endcase
|
|
|
|
v_dest_counter <= v_dest_counter + 1;
|
|
|
|
end
|
2019-02-21 16:55:08 +01:00
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
if (do_load_register_done) begin
|
|
|
|
v_dest_counter <= 0;
|
|
|
|
v_max_counter <= 0;
|
|
|
|
end
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
/*
|
|
|
|
* P-mode
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (start_in_p_mode && op_set_p) begin
|
|
|
|
$display("ALU %0d: [%d] loading P= %h", phase, i_cycle_ctr, i_imm_value);
|
|
|
|
P <= i_imm_value;
|
|
|
|
end
|
|
|
|
|
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* jump mode
|
|
|
|
*
|
|
|
|
*/
|
2019-02-22 08:22:32 +01:00
|
|
|
|
2019-02-21 22:44:55 +01:00
|
|
|
|
|
|
|
end
|
2019-02-21 16:55:08 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
/* module 5:
|
|
|
|
* manages all that is linked with the program counter
|
|
|
|
*/
|
|
|
|
|
|
|
|
// wire [19:0] goyes_off;
|
|
|
|
// wire [19:0] goyes_pc;
|
|
|
|
|
|
|
|
// wire [0:0] is_jmp_rel2;
|
|
|
|
// wire [0:0] is_rtn_rel2;
|
|
|
|
// wire [0:0] jmp_carry_test;
|
|
|
|
// wire [0:0] exec_rtn_rel2;
|
|
|
|
// wire [0:0] set_jmp_rel2;
|
|
|
|
// wire [0:0] exec_jmp_rel2;
|
|
|
|
|
|
|
|
// wire [0:0] set_unc_jmp;
|
|
|
|
// wire [0:0] exec_unc_jmp;
|
|
|
|
// wire [0:0] exec_unc_rtn;
|
|
|
|
// wire [0:0] pop_pc;
|
|
|
|
// // wire [0:0] reload_pc;
|
|
|
|
// wire [0:0] push_pc;
|
|
|
|
|
|
|
|
|
|
|
|
// assign goyes_off = {{12{i_imm_value[3]}}, i_imm_value, jump_off[3:0]};
|
|
|
|
// assign goyes_pc = jump_bse + goyes_off;
|
|
|
|
// // rtnyes is already handled by i_ins_test_go
|
|
|
|
// assign is_rtn_rel2 = (alu_op == `ALU_OP_JMP_REL2) && (goyes_off == 0);
|
|
|
|
// assign is_jmp_rel2 = (alu_op == `ALU_OP_JMP_REL2) && !(goyes_off == 0);
|
|
|
|
// assign jmp_carry_test = (i_test_carry && (CARRY == i_carry_val));
|
|
|
|
// assign exec_rtn_rel2 = is_rtn_rel2 && jmp_carry_test && alu_done;
|
|
|
|
// // assign set_jmp_rel2 = is_jmp_rel2 && jmp_carry_test && alu_finish;
|
|
|
|
// assign exec_jmp_rel2 = is_jmp_rel2 && jmp_carry_test && alu_done;
|
|
|
|
|
|
|
|
|
|
|
|
// // assign set_unc_jmp = is_alu_op_unc_jump && alu_finish;
|
|
|
|
// // assign exec_unc_jmp = is_alu_op_unc_jump && alu_done;
|
|
|
|
// assign exec_unc_rtn = i_pop && i_ins_rtn;
|
|
|
|
|
|
|
|
// assign pop_pc = i_pop && i_ins_rtn &&
|
|
|
|
// ((!i_ins_test_go) ||
|
|
|
|
// (i_ins_test_go && CARRY));
|
|
|
|
|
|
|
|
// assign reload_pc = (exec_unc_jmp || pop_pc || just_reset || exec_jmp_rel2);
|
|
|
|
// assign push_pc = update_pc && i_push && alu_finish;
|
|
|
|
|
|
|
|
|
2019-02-22 12:00:23 +01:00
|
|
|
/* jump values generator */
|
|
|
|
|
|
|
|
wire mode_jmp;
|
|
|
|
assign mode_jmp = start_in_jmp_mode || f_mode_jmp;
|
|
|
|
|
|
|
|
reg [2:0] jump_offset_counter;
|
|
|
|
reg [19:0] jump_base;
|
|
|
|
reg [19:0] jump_offset;
|
|
|
|
|
|
|
|
wire [0:0] jump_counter_0;
|
|
|
|
wire [0:0] jump_counter_1;
|
|
|
|
wire [0:0] jump_counter_2;
|
|
|
|
wire [0:0] jump_counter_3;
|
|
|
|
wire [0:0] jump_counter_4;
|
|
|
|
|
|
|
|
assign jump_counter_0 = (jump_offset_counter == 0);
|
|
|
|
assign jump_counter_1 = (jump_offset_counter == 1);
|
|
|
|
assign jump_counter_2 = (jump_offset_counter == 2);
|
|
|
|
assign jump_counter_3 = (jump_offset_counter == 3);
|
|
|
|
assign jump_counter_4 = (jump_offset_counter == 4);
|
|
|
|
|
|
|
|
wire [19:0] jump_offset_1;
|
|
|
|
wire [19:0] jump_offset_2;
|
|
|
|
wire [19:0] jump_offset_3;
|
|
|
|
wire [19:0] jump_offset_4;
|
|
|
|
wire [19:0] jump_offset_5;
|
|
|
|
|
|
|
|
assign jump_offset_1 = {{16{i_bus_nibble_in[3] && jump_relative}}, i_imm_value};
|
|
|
|
assign jump_offset_2 = {{12{i_bus_nibble_in[3] && jump_relative}}, i_imm_value, jump_offset[ 3:0]};
|
|
|
|
assign jump_offset_3 = {{ 8{i_bus_nibble_in[3] && jump_relative}}, i_imm_value, jump_offset[ 7:0]};
|
|
|
|
assign jump_offset_4 = {{ 4{i_bus_nibble_in[3] && jump_relative}}, i_imm_value, jump_offset[11:0]};
|
|
|
|
assign jump_offset_5 = { i_imm_value, jump_offset[15:0]};
|
|
|
|
|
|
|
|
|
|
|
|
wire [19:0] new_jump_offset;
|
|
|
|
assign new_jump_offset = ({20{jump_counter_0}} & jump_offset_1) |
|
|
|
|
({20{jump_counter_1}} & jump_offset_2) |
|
|
|
|
({20{jump_counter_2}} & jump_offset_3) |
|
|
|
|
({20{jump_counter_3}} & jump_offset_4) |
|
|
|
|
({20{jump_counter_4}} & jump_offset_5);
|
|
|
|
|
|
|
|
wire [0:0] jump_rel_2_done;
|
|
|
|
wire [0:0] jump_rel_3_done;
|
|
|
|
wire [0:0] jump_rel_4_done;
|
|
|
|
wire [0:0] jump_abs_5_done;
|
|
|
|
wire [0:0] jump_done;
|
|
|
|
|
|
|
|
assign jump_rel_2_done = jump_counter_1 && op_jmp_rel_2;
|
|
|
|
assign jump_rel_3_done = jump_counter_2 && op_jmp_rel_3;
|
|
|
|
assign jump_rel_4_done = jump_counter_3 && op_jmp_rel_4;
|
|
|
|
assign jump_abs_5_done = jump_counter_4 && op_jmp_abs_5;
|
|
|
|
assign jump_done = jump_rel_2_done || jump_rel_3_done || jump_rel_4_done || jump_abs_5_done;
|
|
|
|
|
|
|
|
wire [0:0] do_set_jump_base;
|
|
|
|
wire [0:0] do_calc_jump;
|
|
|
|
wire [0:0] do_apply_jump;
|
|
|
|
assign do_set_jump_base = start_in_jmp_mode && !jump_done && jump_counter_0;
|
|
|
|
assign do_calc_jump = mode_jmp && phase_3 && !jump_done;
|
|
|
|
assign do_apply_jump = mode_jmp && phase_3 && jump_done;
|
|
|
|
|
|
|
|
wire [0:0] jump_relative;
|
|
|
|
assign jump_relative = op_jmp_rel_2 || op_jmp_rel_3 || op_jmp_rel_4;
|
|
|
|
|
|
|
|
wire [19:0] jump_pc;
|
|
|
|
assign jump_pc = jump_relative?(jump_base+new_jump_offset):new_jump_offset;
|
|
|
|
|
|
|
|
/* pc update generator */
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
wire [19:0] next_pc;
|
|
|
|
wire [0:0] update_pc;
|
|
|
|
wire [0:0] reload_pc;
|
|
|
|
wire [0:0] pop_pc;
|
|
|
|
wire [0:0] pc_lines_cleanup;
|
|
|
|
|
2019-02-22 12:00:23 +01:00
|
|
|
assign next_pc = (jump_done)?jump_pc:PC + 1;
|
|
|
|
assign update_pc = (!i_reset && just_reset) || alu_active && phase_3 && (!o_alu_stall_dec) /* || exec_unc_jmp || exec_jmp_rel2 */;
|
2019-02-22 08:22:32 +01:00
|
|
|
assign pop_pc = i_pop && i_ins_rtn && ((!i_ins_test_go) || (i_ins_test_go && CARRY));
|
2019-02-22 12:00:23 +01:00
|
|
|
assign reload_pc = (!i_reset && just_reset) || do_apply_jump;
|
2019-02-22 08:22:32 +01:00
|
|
|
|
|
|
|
assign pc_lines_cleanup = alu_active && phase_0;
|
|
|
|
|
|
|
|
always @(posedge i_clk) begin
|
|
|
|
|
|
|
|
/*
|
|
|
|
* initializes default values
|
|
|
|
*/
|
|
|
|
if (i_reset) begin
|
2019-02-22 12:00:23 +01:00
|
|
|
PC <= ~0;
|
|
|
|
o_bus_load_pc <= 0;
|
|
|
|
rstk_ptr <= 0;
|
|
|
|
jump_offset_counter <= 0;
|
|
|
|
jump_base <= 0;
|
|
|
|
jump_offset <= 0;
|
2019-02-22 08:22:32 +01:00
|
|
|
end
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Similarly to the data registers,
|
|
|
|
* initializes the RSTK while the PC is first loaded
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
if (alu_initializing)
|
|
|
|
RSTK[v_dest_counter_ptr] <= 0;
|
|
|
|
|
2019-02-22 12:00:23 +01:00
|
|
|
/**
|
|
|
|
* handles jumps
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
if (do_set_jump_base) begin
|
|
|
|
// $display("ALU_PC %0d: [%d] set jump base %0d | nibble %h | rel %b | base %h | offset %h | jump_pc %h",
|
|
|
|
// phase, i_cycle_ctr, jump_offset_counter, i_imm_value, jump_relative, PC, new_jump_offset, jump_pc);
|
|
|
|
jump_base <= PC;
|
|
|
|
end
|
|
|
|
|
|
|
|
if (do_calc_jump) begin
|
|
|
|
// $display("ALU_PC %0d: [%d] calc jump %0d | nibble %h | rel %b | base %h | offset %h | jump_pc %h",
|
|
|
|
// phase, i_cycle_ctr, jump_offset_counter, i_imm_value, jump_relative, jump_base, new_jump_offset, jump_pc);
|
|
|
|
jump_offset <= new_jump_offset;
|
|
|
|
jump_offset_counter <= jump_offset_counter + 1;
|
|
|
|
end
|
|
|
|
|
|
|
|
if (do_apply_jump) begin
|
|
|
|
// $display("ALU_PC %0d: [%d] apply jump %0d | nibble %h | rel %b | base %h | offset %h | jump_pc %h",
|
|
|
|
// phase, i_cycle_ctr, jump_offset_counter, i_imm_value, jump_relative, jump_base, new_jump_offset, jump_pc);
|
|
|
|
jump_offset_counter <= 0;
|
|
|
|
end
|
|
|
|
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
* Update the PC.
|
|
|
|
* Request the new PC be loaded to the other modules through
|
|
|
|
* the bus if necessary
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
if (update_pc) begin
|
2019-02-22 12:00:23 +01:00
|
|
|
// $display("ALU_PC %0d: [%d] update pc to %h", phase, i_cycle_ctr, next_pc);
|
2019-02-22 08:22:32 +01:00
|
|
|
PC <= /*pop_pc ? RSTK[rstk_ptr - 1] : */next_pc;
|
|
|
|
end
|
|
|
|
|
|
|
|
if (reload_pc) begin
|
2019-02-22 12:00:23 +01:00
|
|
|
// $display("ALU_PC %0d: [%d] $$$$ RELOADING PC to %h $$$$", phase, i_cycle_ctr, next_pc);
|
2019-02-22 08:22:32 +01:00
|
|
|
o_bus_address <= pop_pc ? RSTK[rstk_ptr-1] : next_pc;
|
|
|
|
o_bus_load_pc <= 1;
|
|
|
|
end
|
|
|
|
|
|
|
|
// // $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);
|
|
|
|
// if (pop_pc) begin
|
|
|
|
// $display("POP RSTK[%0d] to PC %5h", rstk_ptr-1, RSTK[rstk_ptr - 1]);
|
|
|
|
// RSTK[rstk_ptr - 1] <= 0;
|
|
|
|
// rstk_ptr <= rstk_ptr - 1;
|
|
|
|
// end
|
|
|
|
|
|
|
|
// 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-21 16:55:08 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
if (pc_lines_cleanup && o_bus_load_pc)
|
|
|
|
o_bus_load_pc <= 0;
|
2019-02-21 16:55:08 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
end
|
2019-02-21 16:55:08 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2019-02-16 22:38:44 +01:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
* 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-21 16:55:08 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// wire do_busclean;
|
|
|
|
// wire do_alu_init;
|
|
|
|
// wire do_alu_prep;
|
|
|
|
// wire do_alu_calc;
|
|
|
|
// wire do_alu_save;
|
|
|
|
// wire do_alu_pc;
|
|
|
|
// wire do_alu_mode;
|
2019-02-13 22:43:04 +01:00
|
|
|
|
2019-02-22 08:22:32 +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 &&
|
|
|
|
// !write_done && !do_exec_p_eq && !o_bus_config;
|
|
|
|
// 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-22 08:22:32 +01:00
|
|
|
// wire do_go_init;
|
|
|
|
// wire do_go_prep;
|
2019-02-16 11:08:34 +01:00
|
|
|
|
2019-02-22 08:22:32 +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-18 11:36:39 +01:00
|
|
|
// now for the fine tuning ;-)
|
|
|
|
|
|
|
|
// save one cycle on P= n!
|
2019-02-22 08:22:32 +01:00
|
|
|
// wire is_alu_op_copy;
|
|
|
|
// wire is_reg_dest_p;
|
|
|
|
// wire is_reg_src1_imm;
|
|
|
|
// wire do_exec_p_eq;
|
2019-02-18 11:36:39 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// assign is_alu_op_copy = (i_alu_op == `ALU_OP_COPY);
|
|
|
|
// assign is_reg_dest_p = (i_reg_dest == `ALU_REG_P);
|
|
|
|
// assign is_reg_src1_imm = (i_reg_src1 == `ALU_REG_IMM);
|
|
|
|
// assign do_exec_p_eq = alu_active && i_en_alu_save && i_ins_alu_op && is_alu_op_copy && is_reg_dest_p && is_reg_src1_imm;
|
2019-02-18 11:36:39 +01:00
|
|
|
|
2019-02-14 08:59:04 +01:00
|
|
|
// the decoder may request the ALU to not stall it
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// wire bus_commands;
|
|
|
|
// assign bus_commands = o_bus_config ;
|
2019-02-19 16:17:16 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// assign o_alu_stall_dec = alu_initializing ||
|
|
|
|
// (alu_run && (!i_alu_no_stall || alu_finish || i_ins_mem_xfr)) ||
|
|
|
|
// i_stalled || bus_commands || stall_modes;
|
2019-02-19 16:17:16 +01:00
|
|
|
|
2019-02-14 08:59:04 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// wire alu_start;
|
|
|
|
// wire alu_finish;
|
|
|
|
// wire [3:0] f_next;
|
2019-02-13 23:18:50 +01:00
|
|
|
|
2019-02-22 08:22:32 +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-22 08:22:32 +01:00
|
|
|
// wire is_alu_op_unc_jump;
|
|
|
|
// assign is_alu_op_unc_jump = ((alu_op == `ALU_OP_JMP_REL3) ||
|
|
|
|
// (alu_op == `ALU_OP_JMP_REL4) ||
|
|
|
|
// (alu_op == `ALU_OP_JMP_ABS5) ||
|
|
|
|
// i_ins_rtn);
|
|
|
|
// 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-17 23:05:33 +01:00
|
|
|
`ifdef SIM
|
|
|
|
reg [4:0] alu_dbg_ctr;
|
|
|
|
`endif
|
|
|
|
|
2019-02-13 23:18:50 +01:00
|
|
|
always @(posedge i_clk) begin
|
2019-02-16 22:38:44 +01:00
|
|
|
|
|
|
|
`ifdef SIM
|
2019-02-17 12:05:38 +01:00
|
|
|
// if (i_stalled && i_en_alu_dump)
|
|
|
|
// $display("ALU STALLED");
|
2019-02-17 12:57:38 +01:00
|
|
|
`endif
|
2019-02-16 22:38:44 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// `ifdef ALU_DEBUG_DBG
|
|
|
|
// $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 );
|
|
|
|
// `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]);
|
2019-02-17 23:05:33 +01:00
|
|
|
|
|
|
|
$write("A: ");
|
|
|
|
for(alu_dbg_ctr=15;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", A[alu_dbg_ctr]);
|
|
|
|
$write(" R0: ");
|
|
|
|
for(alu_dbg_ctr=15;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", R0[alu_dbg_ctr]);
|
|
|
|
$write(" RSTK5: %5h\n", RSTK[5]);
|
|
|
|
|
|
|
|
$write("B: ");
|
|
|
|
for(alu_dbg_ctr=15;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", B[alu_dbg_ctr]);
|
|
|
|
$write(" R1: ");
|
|
|
|
for(alu_dbg_ctr=15;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", R1[alu_dbg_ctr]);
|
|
|
|
$write(" RSTK4: %5h\n", RSTK[4]);
|
|
|
|
|
|
|
|
$write("C: ");
|
|
|
|
for(alu_dbg_ctr=15;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", C[alu_dbg_ctr]);
|
|
|
|
$write(" R2: ");
|
|
|
|
for(alu_dbg_ctr=15;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", R2[alu_dbg_ctr]);
|
|
|
|
$write(" RSTK3: %5h\n", RSTK[3]);
|
|
|
|
|
|
|
|
$write("D: ");
|
|
|
|
for(alu_dbg_ctr=15;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", D[alu_dbg_ctr]);
|
|
|
|
$write(" R3: ");
|
|
|
|
for(alu_dbg_ctr=15;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", R3[alu_dbg_ctr]);
|
|
|
|
$write(" RSTK2: %5h\n", RSTK[2]);
|
|
|
|
|
2019-02-21 16:55:08 +01:00
|
|
|
$write("D0: ");
|
|
|
|
for(alu_dbg_ctr=4;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", D0[alu_dbg_ctr]);
|
|
|
|
$write(" D1: ");
|
|
|
|
for(alu_dbg_ctr=4;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", D1[alu_dbg_ctr]);
|
|
|
|
$write(" R4: ");
|
2019-02-17 23:05:33 +01:00
|
|
|
for(alu_dbg_ctr=15;alu_dbg_ctr!=31;alu_dbg_ctr=alu_dbg_ctr-1)
|
|
|
|
$write("%h", R4[alu_dbg_ctr]);
|
|
|
|
$write(" RSTK1: %5h\n", RSTK[1]);
|
2019-02-17 19:29:39 +01:00
|
|
|
$display(" ADDR: %5h RSTK0: %5h",
|
|
|
|
o_bus_address, RSTK[0]);
|
2019-02-13 22:43:04 +01:00
|
|
|
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-22 08:22:32 +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;
|
|
|
|
|
|
|
|
// always @(posedge i_clk) begin
|
|
|
|
|
|
|
|
// if (i_reset) begin
|
|
|
|
// alu_op <= 0;
|
|
|
|
// reg_dest <= 0;
|
|
|
|
// reg_src1 <= 0;
|
|
|
|
// reg_src2 <= 0;
|
|
|
|
// f_last <= 0;
|
|
|
|
// end
|
|
|
|
|
|
|
|
// // this happens in phase 3, right after the instruction decoder (in phase 2) is finished
|
|
|
|
// if (do_alu_init) begin
|
|
|
|
|
|
|
|
// `ifdef SIM
|
|
|
|
// if (alu_debug)
|
|
|
|
// $display({"ALU_INIT 3: run %b | done %b | stall %b | op %d | s %h | l %h ",
|
|
|
|
// "| ialu %b | dest %d | src1 %d | src2 %d | imm %h"},
|
|
|
|
// 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, i_imm_value);
|
|
|
|
// `endif
|
|
|
|
|
|
|
|
// 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;
|
|
|
|
|
|
|
|
// end
|
|
|
|
// end
|
2019-02-13 23:18:50 +01:00
|
|
|
|
|
|
|
/*
|
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-22 08:22:32 +01:00
|
|
|
// always @(posedge i_clk) begin
|
|
|
|
|
|
|
|
// if (i_reset) begin
|
|
|
|
// alu_run <= 0;
|
|
|
|
// alu_done <= 0;
|
|
|
|
// f_first <= 0;
|
|
|
|
// f_cur <= 0;
|
|
|
|
// end
|
|
|
|
|
|
|
|
// if (alu_initializing)
|
|
|
|
// f_cur <= f_cur + 1;
|
|
|
|
|
|
|
|
// if (do_alu_init) begin
|
|
|
|
// $display("ALU %0d - ------------------------------------------------- DO_ALU_INIT", phase);
|
|
|
|
// alu_run <= 1;
|
|
|
|
// f_first <= i_field_start;
|
|
|
|
// f_cur <= i_field_start;
|
|
|
|
|
|
|
|
// alu_go_test <= is_alu_op_test;
|
|
|
|
// end
|
|
|
|
|
|
|
|
// 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 <= alu_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_cur <= f_next;
|
|
|
|
// end
|
|
|
|
|
|
|
|
// if (do_alu_save && alu_done) begin
|
|
|
|
// alu_run <= 0;
|
|
|
|
// alu_done <= 0;
|
|
|
|
// end
|
|
|
|
|
|
|
|
// end
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// always @(posedge i_clk) begin
|
|
|
|
|
|
|
|
// if (i_reset) begin
|
|
|
|
// p_src1 <= 0;
|
|
|
|
// p_src2 <= 0;
|
|
|
|
// p_carry <= 0;
|
|
|
|
// jump_bse <= 0;
|
|
|
|
// end
|
|
|
|
|
|
|
|
// if (do_alu_prep) begin
|
|
|
|
// if (alu_debug) begin
|
|
|
|
// `ifdef SIM
|
|
|
|
// $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);
|
|
|
|
// `endif
|
|
|
|
// end
|
|
|
|
|
|
|
|
// /*
|
|
|
|
// * source 1
|
|
|
|
// */
|
|
|
|
// case (alu_op)
|
|
|
|
// `ALU_OP_ZERO: begin end // no source required
|
|
|
|
// `ALU_OP_COPY,
|
|
|
|
// `ALU_OP_EXCH,
|
|
|
|
// `ALU_OP_RST_BIT,
|
|
|
|
// `ALU_OP_SET_BIT,
|
|
|
|
// `ALU_OP_2CMPL,
|
|
|
|
// `ALU_OP_DEC,
|
|
|
|
// `ALU_OP_ADD,
|
|
|
|
// `ALU_OP_TEST_EQ,
|
|
|
|
// `ALU_OP_TEST_NEQ,
|
|
|
|
// `ALU_OP_JMP_REL2,
|
|
|
|
// `ALU_OP_JMP_REL3,
|
|
|
|
// `ALU_OP_JMP_REL4,
|
|
|
|
// `ALU_OP_JMP_ABS5,
|
|
|
|
// `ALU_OP_CLR_MASK:
|
|
|
|
// case (reg_src1)
|
|
|
|
// `ALU_REG_A: p_src1 <= A[f_cur];
|
|
|
|
// `ALU_REG_B: p_src1 <= B[f_cur];
|
|
|
|
// `ALU_REG_C: p_src1 <= C[f_cur];
|
|
|
|
// `ALU_REG_D: p_src1 <= D[f_cur];
|
|
|
|
// `ALU_REG_R0: p_src1 <= R0[f_cur];
|
|
|
|
// `ALU_REG_R1: p_src1 <= R1[f_cur];
|
|
|
|
// `ALU_REG_R2: p_src1 <= R2[f_cur];
|
|
|
|
// `ALU_REG_R3: p_src1 <= R3[f_cur];
|
|
|
|
// `ALU_REG_R4: p_src1 <= R4[f_cur];
|
|
|
|
// `ALU_REG_D0: p_src1 <= D0[f_cur[2:0]];
|
|
|
|
// `ALU_REG_D1: p_src1 <= D1[f_cur[2:0]];
|
|
|
|
// `ALU_REG_P: p_src1 <= P;
|
|
|
|
// `ALU_REG_DAT0,
|
|
|
|
// `ALU_REG_DAT1: p_src1 <= i_bus_nibble_in;
|
|
|
|
// `ALU_REG_HST: p_src1 <= HST;
|
|
|
|
// `ALU_REG_IMM: p_src1 <= i_imm_value;
|
|
|
|
// `ALU_REG_ZERO: p_src1 <= 0;
|
|
|
|
// default: $display("#### SRC_1 UNHANDLED REGISTER %0d", reg_src1);
|
|
|
|
// endcase
|
|
|
|
// default: $display("#### SRC_1 UNHANDLED OPERATION %0d", alu_op);
|
|
|
|
// endcase
|
|
|
|
|
|
|
|
|
|
|
|
// /*
|
|
|
|
// * source 2
|
|
|
|
// */
|
|
|
|
// case (alu_op)
|
|
|
|
// `ALU_OP_ZERO,
|
|
|
|
// `ALU_OP_COPY,
|
|
|
|
// `ALU_OP_RST_BIT,
|
|
|
|
// `ALU_OP_SET_BIT,
|
|
|
|
// `ALU_OP_2CMPL,
|
|
|
|
// `ALU_OP_DEC,
|
|
|
|
// `ALU_OP_JMP_REL2,
|
|
|
|
// `ALU_OP_JMP_REL3,
|
|
|
|
// `ALU_OP_JMP_REL4,
|
|
|
|
// `ALU_OP_JMP_ABS5: begin end // no need for a 2nd operand
|
|
|
|
// `ALU_OP_EXCH,
|
|
|
|
// `ALU_OP_ADD,
|
|
|
|
// `ALU_OP_TEST_EQ,
|
|
|
|
// `ALU_OP_TEST_NEQ,
|
|
|
|
// `ALU_OP_CLR_MASK: begin
|
|
|
|
// case (reg_src2)
|
|
|
|
// `ALU_REG_A: p_src2 <= A[f_cur];
|
|
|
|
// `ALU_REG_B: p_src2 <= B[f_cur];
|
|
|
|
// `ALU_REG_C: p_src2 <= C[f_cur];
|
|
|
|
// `ALU_REG_D: p_src2 <= D[f_cur];
|
|
|
|
// `ALU_REG_R0: p_src2 <= R0[f_cur];
|
|
|
|
// `ALU_REG_R1: p_src2 <= R1[f_cur];
|
|
|
|
// `ALU_REG_R2: p_src2 <= R2[f_cur];
|
|
|
|
// `ALU_REG_R3: p_src2 <= R3[f_cur];
|
|
|
|
// `ALU_REG_R4: p_src2 <= R4[f_cur];
|
|
|
|
// `ALU_REG_D0: p_src2 <= D0[f_cur[2:0]];
|
|
|
|
// `ALU_REG_D1: p_src2 <= D1[f_cur[2:0]];
|
|
|
|
// `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);
|
|
|
|
// endcase
|
|
|
|
|
|
|
|
// // setup p_carry
|
|
|
|
// // $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)
|
|
|
|
// `ALU_OP_2CMPL: p_carry <= alu_start?1'b1:c_carry;
|
|
|
|
// `ALU_OP_DEC: p_carry <= alu_start?1'b0:c_carry;
|
|
|
|
// `ALU_OP_ADD: p_carry <= alu_start?1'b0:c_carry;
|
|
|
|
// `ALU_OP_TEST_NEQ: p_carry <= alu_start?1'b0:c_carry;
|
|
|
|
// endcase
|
|
|
|
|
|
|
|
// // prepare jump base
|
|
|
|
// case (alu_op)
|
|
|
|
// `ALU_OP_JMP_REL2,
|
|
|
|
// `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
|
|
|
|
// endcase
|
|
|
|
|
|
|
|
// end
|
|
|
|
// end
|
|
|
|
|
|
|
|
// always @(posedge i_clk) begin
|
2019-02-16 12:17:40 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// if (i_reset) begin
|
|
|
|
// c_res1 <= 0;
|
|
|
|
// c_res2 <= 0;
|
|
|
|
// c_carry <= 0;
|
|
|
|
// is_zero <= 0;
|
|
|
|
// jump_off <= 0;
|
|
|
|
// end
|
|
|
|
|
|
|
|
// if (do_alu_calc) begin
|
|
|
|
// `ifdef SIM
|
|
|
|
// if (alu_debug)
|
|
|
|
// $display("ALU_CALC 2: run %b | done %b | stall %b | op %d | f %h | c %h | l %h | dest %d | psrc1 %h | psrc2 %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);
|
|
|
|
// if (alu_debug_jump)
|
|
|
|
// $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);
|
|
|
|
// `endif
|
|
|
|
|
|
|
|
// if(alu_start)
|
|
|
|
// case (alu_op)
|
|
|
|
// `ALU_OP_JMP_REL2,
|
|
|
|
// `ALU_OP_JMP_REL3,
|
|
|
|
// `ALU_OP_JMP_REL4,
|
|
|
|
// `ALU_OP_JMP_ABS5: jump_off <= { 16'b0, p_src1 };
|
|
|
|
// endcase
|
|
|
|
|
|
|
|
// // main case
|
|
|
|
// case (alu_op)
|
|
|
|
// `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 ;
|
|
|
|
// c_res1 <= ~p_src1 + {3'b000, p_carry};
|
|
|
|
// is_zero <= ((~p_src1 + {3'b000, p_carry}) == 0) && alu_start?1:is_zero;
|
|
|
|
// end
|
|
|
|
// `ALU_OP_DEC:
|
|
|
|
// {c_carry, c_res1} <= p_src1 + 4'b1111 + {4'b0000, p_carry};
|
|
|
|
// `ALU_OP_ADD:
|
|
|
|
// {c_carry, c_res1} <= p_src1 + p_src2 + {4'b0000, p_carry};
|
|
|
|
// `ALU_OP_TEST_NEQ:
|
|
|
|
// c_carry <= !(p_src1 == p_src2) || p_carry;
|
|
|
|
// `ALU_OP_JMP_REL2: begin end // there is no middle part
|
|
|
|
// `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);
|
|
|
|
// endcase
|
|
|
|
|
|
|
|
// if (alu_finish)
|
|
|
|
// case (alu_op)
|
|
|
|
// `ALU_OP_JMP_REL2: jump_off <= { {12{p_src1[3]}}, p_src1, jump_off[3:0] };
|
|
|
|
// `ALU_OP_JMP_REL3: jump_off <= { {8{p_src1[3]}}, p_src1, jump_off[7:0] };
|
|
|
|
// `ALU_OP_JMP_REL4: jump_off <= { {4{p_src1[3]}}, p_src1, jump_off[11:0] };
|
|
|
|
// endcase
|
|
|
|
|
|
|
|
// // $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 );
|
|
|
|
// end
|
|
|
|
|
|
|
|
// if (do_go_init) begin
|
|
|
|
// // $display("GO_INIT 3: imm %h", i_imm_value);
|
|
|
|
// jump_off <= { {16{1'b0}}, i_imm_value};
|
|
|
|
// end
|
|
|
|
// end
|
2019-02-18 17:38:25 +01:00
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Debug for some JUMP condition testing
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-02-22 08:22:32 +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);
|
|
|
|
// `endif
|
|
|
|
// end
|
|
|
|
// end
|
2019-02-16 11:08:34 +01:00
|
|
|
|
2019-02-18 11:36:39 +01:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* Epic shortcut for P= n case
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// if (do_exec_p_eq) begin
|
|
|
|
// P <= i_imm_value;
|
|
|
|
// end
|
2019-02-18 11:36:39 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* normal way for the ALU to save results.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// if (do_alu_save) begin
|
|
|
|
// `ifdef SIM
|
|
|
|
// if (alu_debug) begin
|
|
|
|
// $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);
|
|
|
|
|
|
|
|
// end
|
|
|
|
// `endif
|
|
|
|
|
|
|
|
// case (alu_op)
|
|
|
|
// `ALU_OP_ZERO,
|
|
|
|
// `ALU_OP_COPY,
|
|
|
|
// `ALU_OP_EXCH, // does the first assign
|
|
|
|
// `ALU_OP_2CMPL,
|
|
|
|
// `ALU_OP_DEC,
|
|
|
|
// `ALU_OP_ADD,
|
|
|
|
// `ALU_OP_CLR_MASK:
|
|
|
|
// case (reg_dest)
|
|
|
|
// `ALU_REG_A: A[f_cur] <= c_res1;
|
|
|
|
// `ALU_REG_B: B[f_cur] <= c_res1;
|
|
|
|
// `ALU_REG_C: C[f_cur] <= c_res1;
|
|
|
|
// `ALU_REG_D: D[f_cur] <= c_res1;
|
|
|
|
// `ALU_REG_R0: R0[f_cur] <= c_res1;
|
|
|
|
// `ALU_REG_R1: R1[f_cur] <= c_res1;
|
|
|
|
// `ALU_REG_R2: R2[f_cur] <= c_res1;
|
|
|
|
// `ALU_REG_R3: R3[f_cur] <= c_res1;
|
|
|
|
// `ALU_REG_R4: R4[f_cur] <= c_res1;
|
|
|
|
// `ALU_REG_D0: D0[f_cur[2:0]] <= c_res1;
|
|
|
|
// `ALU_REG_D1: D1[f_cur[2:0]] <= c_res1;
|
|
|
|
// `ALU_REG_ST: ST[f_cur*4+:4] <= c_res1;
|
|
|
|
// `ALU_REG_P: P <= c_res1;
|
|
|
|
// `ALU_REG_DAT0,
|
|
|
|
// `ALU_REG_DAT1: o_bus_nibble_out <= c_res1;
|
|
|
|
// `ALU_REG_HST: HST <= c_res1;
|
|
|
|
// `ALU_REG_ADDR: begin end // done down below where o_bus_addr is accessible
|
|
|
|
// default: $display("#### ALU_SAVE invalid register %0d for op %0d", reg_dest, alu_op);
|
|
|
|
// 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
|
|
|
|
// `ALU_OP_TEST_EQ,
|
|
|
|
// `ALU_OP_TEST_NEQ,
|
|
|
|
// `ALU_OP_JMP_REL2,
|
|
|
|
// `ALU_OP_JMP_REL3,
|
|
|
|
// `ALU_OP_JMP_REL4,
|
|
|
|
// `ALU_OP_JMP_ABS5: begin end // nothing to save, handled by PC management below
|
|
|
|
// default: $display("#### ALU_SAVE UNHANDLED OP %0d", alu_op);
|
|
|
|
// endcase
|
|
|
|
|
|
|
|
// /*
|
|
|
|
// * in case of exch, we need to update src2 to finish the exchange
|
|
|
|
// */
|
|
|
|
// case (alu_op)
|
|
|
|
// `ALU_OP_EXCH: // 2nd assign, with src2
|
|
|
|
// case (reg_src2)
|
|
|
|
// `ALU_REG_A: A[f_cur] <= c_res2;
|
|
|
|
// `ALU_REG_B: B[f_cur] <= c_res2;
|
|
|
|
// `ALU_REG_C: C[f_cur] <= c_res2;
|
|
|
|
// `ALU_REG_D: D[f_cur] <= c_res2;
|
|
|
|
// `ALU_REG_D0: D0[f_cur[2:0]] <= c_res2;
|
|
|
|
// `ALU_REG_D1: D1[f_cur[2:0]] <= c_res2;
|
|
|
|
// `ALU_REG_R0: R0[f_cur] <= c_res2;
|
|
|
|
// `ALU_REG_R1: R1[f_cur] <= c_res2;
|
|
|
|
// `ALU_REG_R2: R2[f_cur] <= c_res2;
|
|
|
|
// `ALU_REG_R3: R3[f_cur] <= c_res2;
|
|
|
|
// `ALU_REG_R4: R4[f_cur] <= 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
|
|
|
|
// end
|
2019-02-13 22:43:04 +01:00
|
|
|
|
2019-02-16 12:17:40 +01:00
|
|
|
/*
|
|
|
|
* update carry
|
|
|
|
*/
|
2019-02-22 08:22:32 +01:00
|
|
|
// if (do_alu_save) begin
|
|
|
|
// case (alu_op)
|
|
|
|
// `ALU_OP_2CMPL: CARRY <= !is_zero;
|
|
|
|
// `ALU_OP_DEC,
|
|
|
|
// `ALU_OP_ADD,
|
|
|
|
// `ALU_OP_TEST_EQ,
|
|
|
|
// `ALU_OP_TEST_NEQ: CARRY <= c_carry;
|
|
|
|
// endcase
|
|
|
|
// end
|
2019-02-15 07:07:55 +01:00
|
|
|
|
|
|
|
// do whatever is requested by the RTN instruction
|
2019-02-22 08:22:32 +01:00
|
|
|
// if (alu_active && i_ins_rtn) begin
|
2019-02-15 07:07:55 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// if (i_set_xm)
|
|
|
|
// HST[`ALU_HST_XM] <= 1;
|
2019-02-14 22:14:52 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// if (i_set_carry)
|
|
|
|
// CARRY <= i_carry_val;
|
2019-02-17 12:57:38 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// end
|
2019-02-17 12:57:38 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// end
|
2019-02-17 12:57:38 +01:00
|
|
|
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// wire [0:0] read_done;
|
|
|
|
// wire [0:0] setup_load_dp_read;
|
|
|
|
// wire [0:0] setup_load_dp_write;
|
2019-02-14 22:14:52 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// assign read_done_t = is_mem_read && do_alu_save && ((f_cur +1) == f_last);
|
|
|
|
// assign read_done = (phase == 3) && i_stalled && is_mem_read && !do_alu_save && (f_cur == f_last);
|
|
|
|
// assign setup_load_dp_read = do_alu_init && is_mem_read && !read_done;
|
|
|
|
// assign setup_load_dp_write = do_alu_init && is_mem_write && !write_done;
|
|
|
|
// assign setup_load_dp = setup_load_dp_read || setup_load_dp_write;
|
|
|
|
// assign no_extra_cycles = (extra_cycles == 0);
|
|
|
|
// assign cycles_to_go = extra_cycles - 1;
|
2019-02-20 09:17:37 +01:00
|
|
|
|
2019-02-17 19:29:39 +01:00
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* config and unconfig
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// wire is_bus_config;
|
|
|
|
// assign is_bus_config = (alu_op == `ALU_OP_COPY) && (reg_dest == `ALU_REG_ADDR);
|
|
|
|
// wire send_config;
|
|
|
|
// assign send_config = alu_active && (phase == 1) && i_ins_alu_op && alu_run && alu_finish;
|
|
|
|
// wire clean_after_config;
|
|
|
|
// assign clean_after_config = alu_active && (phase == 3) && o_bus_config && !alu_run;
|
2019-02-16 22:38:44 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// always @(posedge i_clk) begin
|
|
|
|
// if (i_reset)
|
|
|
|
// o_bus_config <= 0;
|
2019-02-15 07:07:55 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// // $display("send_config %b | is_bus_cfg %b | i_ins_cfg %b", send_config, is_bus_config, i_ins_config);
|
|
|
|
// if (send_config && is_bus_config && i_ins_config) begin
|
|
|
|
// $display("ALU %0d - =========================== ALU start configure mode", phase);
|
|
|
|
// o_bus_config <= 1;
|
|
|
|
// end
|
2019-02-16 12:17:40 +01:00
|
|
|
|
2019-02-14 22:14:52 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// if (clean_after_config) begin
|
|
|
|
// $display("ALU %0d - --------------------------- ALU end configure mode %b", phase, i_stalled);
|
|
|
|
// o_bus_config <= 0;
|
|
|
|
// end
|
2019-02-17 12:05:38 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// end
|
2019-02-12 11:22:55 +01:00
|
|
|
|
2019-02-15 07:07:55 +01:00
|
|
|
/*****************************************************************************
|
|
|
|
*
|
|
|
|
* execute SETHEX and SETDEC
|
|
|
|
*
|
|
|
|
****************************************************************************/
|
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// always @(posedge i_clk) begin
|
|
|
|
// if (i_reset)
|
|
|
|
// DEC <= 0;
|
2019-02-18 17:38:25 +01:00
|
|
|
|
2019-02-22 08:22:32 +01:00
|
|
|
// // changing calculation modes
|
|
|
|
// if (do_alu_mode) begin
|
|
|
|
// $display("SETTING MODE TO %s", i_mode_dec?"DEC":"HEX");
|
|
|
|
// DEC <= i_mode_dec;
|
|
|
|
// end
|
|
|
|
// end
|
2019-02-15 07:07:55 +01:00
|
|
|
|
|
|
|
|
2019-02-12 11:22:55 +01:00
|
|
|
endmodule
|
2019-02-14 15:27:17 +01:00
|
|
|
|
|
|
|
`endif
|