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/] [memaccess.cpp] - 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 Memory Access stage.
 */
 
#include "memaccess.h"
 
namespace debugger {
 
MemAccess::MemAccess(sc_module_name name_) 
    : sc_module(name_) {
    SC_METHOD(comb);
    sensitive << i_nrst;
    sensitive << i_mem_req_ready;
    sensitive << i_e_valid;
    sensitive << i_e_pc;
    sensitive << i_e_instr;
    sensitive << i_res_addr;
    sensitive << i_res_data;
    sensitive << i_memop_sign_ext;
    sensitive << i_memop_load;
    sensitive << i_memop_store;
    sensitive << i_memop_size;
    sensitive << i_memop_addr;
    sensitive << i_mem_data_valid;
    sensitive << i_mem_data_addr;
    sensitive << i_mem_data;
    sensitive << r.valid;
    sensitive << r.wdata;
    sensitive << r.wait_req;
    sensitive << r.wait_resp;
 
    SC_METHOD(registers);
    sensitive << i_clk.pos();
};
 
void MemAccess::generateVCD(sc_trace_file *i_vcd, sc_trace_file *o_vcd) {
    if (o_vcd) {
        sc_trace(o_vcd, i_e_valid, "/top/proc0/mem0/i_e_valid");
        sc_trace(o_vcd, i_e_pc, "/top/proc0/mem0/i_e_pc");
        sc_trace(o_vcd, i_e_instr, "/top/proc0/mem0/i_e_instr");
        sc_trace(o_vcd, o_mem_valid, "/top/proc0/mem0/o_mem_valid");
        sc_trace(o_vcd, o_mem_write, "/top/proc0/mem0/o_mem_write");
        sc_trace(o_vcd, i_mem_req_ready, "/top/proc0/mem0/i_mem_req_ready");
        sc_trace(o_vcd, o_mem_sz, "/top/proc0/mem0/o_mem_sz");
        sc_trace(o_vcd, o_mem_addr, "/top/proc0/mem0/o_mem_addr");
        sc_trace(o_vcd, o_mem_data, "/top/proc0/mem0/o_mem_data");
        sc_trace(o_vcd, i_mem_data_valid, "/top/proc0/mem0/i_mem_data_valid");
        sc_trace(o_vcd, i_mem_data_addr, "/top/proc0/mem0/i_mem_data_addr");
        sc_trace(o_vcd, i_mem_data, "/top/proc0/mem0/i_mem_data");
        sc_trace(o_vcd, o_mem_resp_ready, "/top/proc0/mem0/o_mem_resp_ready");
 
        sc_trace(o_vcd, o_valid, "/top/proc0/mem0/o_valid");
        sc_trace(o_vcd, o_pc, "/top/proc0/mem0/o_pc");
        sc_trace(o_vcd, o_instr, "/top/proc0/mem0/o_instr");
        sc_trace(o_vcd, o_wena, "/top/proc0/mem0/o_wena");
        sc_trace(o_vcd, o_waddr, "/top/proc0/mem0/o_waddr");
        sc_trace(o_vcd, o_wdata, "/top/proc0/mem0/o_wdata");
        sc_trace(o_vcd, o_hold, "/top/proc0/mem0/o_hold");
        sc_trace(o_vcd, r.wait_resp, "/top/proc0/mem0/r.wait_resp");
    }
}
 
void MemAccess::comb() {
    bool w_o_mem_valid;
    bool w_o_mem_write;
    sc_uint<2> wb_o_mem_sz;
    sc_uint<BUS_ADDR_WIDTH> wb_o_mem_addr;
    sc_uint<RISCV_ARCH> wb_o_mem_wdata;
    sc_uint<RISCV_ARCH> wb_res_wdata;
    bool w_memop;
    bool w_o_valid;
    bool w_o_wena;
    bool w_o_hold;
    bool w_mem_fire;
 
    v = r;
 
    w_o_mem_valid = 0;
    w_o_mem_write = 0;
    wb_o_mem_sz = 0;
    wb_o_mem_addr = 0;
    wb_o_mem_wdata = 0;
    v.valid = 0;
    w_o_hold = 0;
 
    w_memop = i_memop_load.read() || i_memop_store.read();
 
    if (r.wait_req.read()) {
        if (i_mem_req_ready.read()) {
            v.wait_req = 0;
            v.wait_resp = 1;
        }
        w_o_mem_valid = 1;
        w_o_mem_write = r.wait_req_write;
        wb_o_mem_sz = r.wait_req_sz;
        wb_o_mem_addr = r.wait_req_addr;
        wb_o_mem_wdata = r.wait_req_wdata;
    } else if (i_e_valid.read()) {
        v.valid = !w_memop;
        v.pc = i_e_pc;
        v.instr = i_e_instr;
        v.waddr = i_res_addr;
        v.wdata = i_res_data;
        if (i_res_addr.read() == 0) {
            v.wena = 0;
        } else {
            v.wena = 1;
        }
 
        if (w_memop) {
            w_o_mem_valid = 1;
            w_o_mem_write = i_memop_store;
            wb_o_mem_sz = i_memop_size;
            wb_o_mem_addr = i_memop_addr;
            wb_o_mem_wdata = i_res_data;
            v.sign_ext = i_memop_sign_ext;
            v.size = i_memop_size;
 
            v.wait_resp = i_mem_req_ready;
            v.wait_req = !i_mem_req_ready;
            v.wait_req_write = i_memop_store;
            v.wait_req_sz = i_memop_size;
            v.wait_req_addr = i_memop_addr;
            v.wait_req_wdata = i_res_data;
        } else {
            w_o_mem_valid = 0;
            w_o_mem_write = 0;
            wb_o_mem_sz = 0;
            wb_o_mem_addr = 0;
            wb_o_mem_wdata = 0;
            v.sign_ext = 0;
            v.size = 0;
            v.wait_req_addr = 0;
            v.wait_req = 0;
            v.wait_resp = 0;
        }
    } else if (i_mem_data_valid.read()) {
        v.wait_resp = 0;
    }
 
    w_o_hold = (i_e_valid.read() && w_memop) || r.wait_req.read() 
            || (r.wait_resp.read() && !i_mem_data_valid.read());
 
    w_mem_fire = i_mem_data_valid;
    if (w_mem_fire) {
        if (r.sign_ext.read()) {
            switch (r.size.read()) {
            case MEMOP_1B:
                wb_res_wdata = i_mem_data;
                if (i_mem_data.read()[7]) {
                    wb_res_wdata(63, 8) = ~0;
                }
                break;
            case MEMOP_2B:
                wb_res_wdata = i_mem_data;
                if (i_mem_data.read()[15]) {
                    wb_res_wdata(63, 16) = ~0;
                }
                break;
            case MEMOP_4B:
                wb_res_wdata = i_mem_data;
                if (i_mem_data.read()[31]) {
                    wb_res_wdata(63, 32) = ~0;
                }
                break;
            default:
                wb_res_wdata = i_mem_data;
            }
        } else {
            wb_res_wdata = i_mem_data;
        }
    } else {
        wb_res_wdata = r.wdata;
    }
 
    w_o_valid = r.valid.read() || w_mem_fire;
    w_o_wena = r.wena & w_o_valid;
 
    if (!i_nrst.read()) {
        v.valid = false;
        v.pc = 0;
        v.instr = 0;
        v.waddr = 0;
        v.wdata = 0;
        v.wena = 0;
        v.size = 0;
        v.sign_ext = 0;
        v.wait_req = 0;
        v.wait_req_write = 0;
        v.wait_req_sz = 0;
        v.wait_req_addr = 0;
        v.wait_req_wdata = 0;
        v.wait_resp = 0;
    }
 
    o_mem_resp_ready = 1;
 
    o_mem_valid = w_o_mem_valid;
    o_mem_write = w_o_mem_write;
    o_mem_sz = wb_o_mem_sz;
    o_mem_addr = wb_o_mem_addr;
    o_mem_data = wb_o_mem_wdata;
 
    o_wena = w_o_wena;
    o_waddr = r.waddr;
    o_wdata = wb_res_wdata;
    o_valid = w_o_valid;
    o_pc = r.pc;
    o_instr = r.instr;
    o_hold = w_o_hold;
}
 
void MemAccess::registers() {
    r = v;
}
 
}  // namespace debugger
 
 

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.