OpenCores
URL https://opencores.org/ocsvn/riscv_vhdl/riscv_vhdl/trunk

Subversion Repositories riscv_vhdl

[/] [riscv_vhdl/] [trunk/] [debugger/] [src/] [cpu_sysc_plugin/] [riverlib/] [core/] [execute.h] - Rev 2

Go to most recent revision | Compare with Previous | Blame | View Log

/**
 * @file
 * @copyright  Copyright 2016 GNSS Sensor Ltd. All right reserved.
 * @author     Sergey Khabarov - sergeykhbr@gmail.com
 * @brief      CPU Instruction Execution stage.
 */
 
#ifndef __DEBUGGER_RIVERLIB_EXECUTE_H__
#define __DEBUGGER_RIVERLIB_EXECUTE_H__
 
#include <systemc.h>
#include "../river_cfg.h"
#include "arith/int_div.h"
#include "arith/int_mul.h"
#include "arith/shift.h"
 
namespace debugger {
 
SC_MODULE(InstrExecute) {
    sc_in<bool> i_clk;
    sc_in<bool> i_nrst;                         // Reset active LOW
    sc_in<bool> i_pipeline_hold;                // Hold execution by any reason
    sc_in<bool> i_d_valid;                      // Decoded instruction is valid
    sc_in<sc_uint<BUS_ADDR_WIDTH>> i_d_pc;      // Instruction pointer on decoded instruction
    sc_in<sc_uint<32>> i_d_instr;               // Decoded instruction value
    sc_in<bool> i_wb_done;                      // write back done (Used to clear hazardness)
    sc_in<bool> i_memop_store;                  // Store to memory operation
    sc_in<bool> i_memop_load;                   // Load from memoru operation
    sc_in<bool> i_memop_sign_ext;               // Load memory value with sign extending
    sc_in<sc_uint<2>> i_memop_size;             // Memory transaction size
    sc_in<bool> i_unsigned_op;                  // Unsigned operands
    sc_in<bool> i_rv32;                         // 32-bits instruction
    sc_in<sc_bv<ISA_Total>> i_isa_type;         // Type of the instruction's structure (ISA spec.)
    sc_in<sc_bv<Instr_Total>> i_ivec;           // One pulse per supported instruction.
    sc_in<bool> i_ie;                           // Interrupt enable bit
    sc_in<sc_uint<BUS_ADDR_WIDTH>> i_mtvec;     // Interrupt descriptor table
    sc_in<sc_uint<2>> i_mode;                   // Current processor mode
    sc_in<bool> i_break_mode;                   // Behaviour on EBREAK instruction: 0 = halt; 1 = generate trap
    sc_in<bool> i_unsup_exception;              // Unsupported instruction exception
    sc_in<bool> i_ext_irq;                      // External interrupt from PLIC (todo: timer & software interrupts)
    sc_in<bool> i_dport_npc_write;              // Write npc value from debug port
    sc_in<sc_uint<BUS_ADDR_WIDTH>> i_dport_npc; // Debug port npc value to write
 
    sc_out<sc_uint<5>> o_radr1;                 // Integer register index 1
    sc_in<sc_uint<RISCV_ARCH>> i_rdata1;        // Integer register value 1
    sc_out<sc_uint<5>> o_radr2;                 // Integer register index 2
    sc_in<sc_uint<RISCV_ARCH>> i_rdata2;        // Integer register value 2
    sc_out<sc_uint<5>> o_res_addr;              // Address to store result of the instruction (0=do not store)
    sc_out<sc_uint<RISCV_ARCH>> o_res_data;     // Value to store
    sc_out<bool> o_pipeline_hold;               // Hold pipeline while 'writeback' not done or multi-clock instruction.
    sc_out<bool> o_xret;                        // XRET instruction: MRET, URET or other.
    sc_out<sc_uint<12>> o_csr_addr;             // CSR address. 0 if not a CSR instruction with xret signals mode switching
    sc_out<bool> o_csr_wena;                    // Write new CSR value
    sc_in<sc_uint<RISCV_ARCH>> i_csr_rdata;     // CSR current value
    sc_out<sc_uint<RISCV_ARCH>> o_csr_wdata;    // CSR new value
    sc_out<bool> o_trap_ena;                    // Trap occurs  pulse
    sc_out<sc_uint<5>> o_trap_code;             // bit[4] : 1=interrupt; 0=exception; bits[3:0]=code
    sc_out<sc_uint<BUS_ADDR_WIDTH>> o_trap_pc;  // trap on pc
 
    sc_out<bool> o_memop_sign_ext;              // Load data with sign extending
    sc_out<bool> o_memop_load;                  // Load data instruction
    sc_out<bool> o_memop_store;                 // Store data instruction
    sc_out<sc_uint<2>> o_memop_size;            // 0=1bytes; 1=2bytes; 2=4bytes; 3=8bytes
    sc_out<sc_uint<BUS_ADDR_WIDTH>> o_memop_addr;// Memory access address
 
    sc_out<bool> o_valid;                       // Output is valid
    sc_out<sc_uint<BUS_ADDR_WIDTH>> o_pc;       // Valid instruction pointer
    sc_out<sc_uint<BUS_ADDR_WIDTH>> o_npc;      // Next instruction pointer. Next decoded pc must match to this value or will be ignored.
    sc_out<sc_uint<32>> o_instr;                // Valid instruction value
    sc_out<bool> o_breakpoint;                  // ebreak instruction
    sc_out<bool> o_call;                        // CALL pseudo instruction detected
    sc_out<bool> o_ret;                         // RET pseudoinstruction detected
 
    void comb();
    void registers();
 
    SC_HAS_PROCESS(InstrExecute);
 
    InstrExecute(sc_module_name name_);
    virtual ~InstrExecute();
 
    void generateVCD(sc_trace_file *i_vcd, sc_trace_file *o_vcd);
 
private:
    enum EMultiCycleInstruction {
        Multi_MUL,
        Multi_DIV,
        Multi_Total
    };
 
    struct multi_arith_type {
        sc_signal<sc_uint<RISCV_ARCH>> arr[Multi_Total];
    };
 
    struct RegistersType {
        sc_signal<bool> d_valid;                        // Valid decoded instruction latch
        sc_signal<sc_uint<BUS_ADDR_WIDTH>> pc;
        sc_signal<sc_uint<BUS_ADDR_WIDTH>> npc;
        sc_signal<sc_uint<32>> instr;
        sc_uint<5> res_addr;
        sc_signal<sc_uint<RISCV_ARCH>> res_val;
        sc_signal<bool> memop_load;
        sc_signal<bool> memop_store;
        bool memop_sign_ext;
        sc_uint<2> memop_size;
        sc_signal<sc_uint<BUS_ADDR_WIDTH>> memop_addr;
 
        sc_signal<sc_uint<5>> multi_res_addr;           // latched output reg. address while multi-cycle instruction
        sc_signal<sc_uint<BUS_ADDR_WIDTH>> multi_pc;    // latched pc-value while multi-cycle instruction
        sc_signal<sc_uint<BUS_ADDR_WIDTH>> multi_npc;   // latched npc-value while multi-cycle instruction
        sc_signal<sc_uint<32>> multi_instr;             // Multi-cycle instruction is under processing
        sc_signal<bool> multi_ena[Multi_Total];         // Enable pulse for Operation that takes more than 1 clock
        sc_signal<bool> multi_rv32;                     // Long operation with 32-bits operands
        sc_signal<bool> multi_unsigned;                 // Long operation with unsiged operands
        sc_signal<bool> multi_residual_high;            // Flag for Divider module: 0=divsion output; 1=residual output
                                                        // Flag for multiplier: 0=usual; 1=get high bits
        sc_signal<bool> multiclock_ena;
        sc_signal<sc_uint<RISCV_ARCH>> multi_a1;        // Multi-cycle operand 1
        sc_signal<sc_uint<RISCV_ARCH>> multi_a2;        // Multi-cycle operand 2
 
        sc_signal<sc_uint<5>> hazard_addr0;             // Updated register address on previous step
        sc_signal<sc_uint<5>> hazard_addr1;             // Updated register address on pre-previous step
        sc_signal<sc_uint<2>> hazard_depth;             // Number of modificated registers that wasn't done yet
 
        sc_signal<bool> ext_irq_pulser;                 // Form 1 clock pulse from strob
        sc_signal<bool> trap_ena;                       // Trap occur, switch mode
        sc_signal<bool> breakpoint;
        sc_uint<5> trap_code_waiting;                   // To avoid multi-cycle instruction collision
        sc_signal<sc_uint<5>> trap_code;                // bit[4] : 1 = interrupt; 0 = exception
                                                        // bit[3:0] : trap code
        sc_signal<sc_uint<BUS_ADDR_WIDTH>> trap_pc;     // pc that caused a trap 
        sc_signal<bool> call;
        sc_signal<bool> ret;
    } v, r;
    sc_signal<bool> w_hazard_detected;
    multi_arith_type wb_arith_res;
    sc_signal<bool> w_arith_valid[Multi_Total];
    sc_signal<bool> w_arith_busy[Multi_Total];
    bool w_interrupt;
    bool w_exception;
    bool w_exception_store;
    bool w_exception_load;
    bool w_exception_xret;
    sc_uint<5> wb_exception_code;
 
    sc_signal<sc_uint<RISCV_ARCH>> wb_shifter_a1;      // Shifters operand 1
    sc_signal<sc_uint<6>> wb_shifter_a2;               // Shifters operand 2
    sc_signal<sc_uint<RISCV_ARCH>> wb_sll;
    sc_signal<sc_uint<RISCV_ARCH>> wb_sllw;
    sc_signal<sc_uint<RISCV_ARCH>> wb_srl;
    sc_signal<sc_uint<RISCV_ARCH>> wb_srlw;
    sc_signal<sc_uint<RISCV_ARCH>> wb_sra;
    sc_signal<sc_uint<RISCV_ARCH>> wb_sraw;
 
    IntMul *mul0;
    IntDiv *div0;
    Shifter *sh0;
};
 
 
}  // namespace debugger
 
#endif  // __DEBUGGER_RIVERLIB_EXECUTE_H__
 

Go to most recent revision | Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

© copyright 1999-2024 OpenCores.org, equivalent to Oliscience, all rights reserved. OpenCores®, registered trademark.