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

Subversion Repositories ulpi_wrapper

[/] [ulpi_wrapper/] [trunk/] [testbench/] [sc_vpi_module.h] - Rev 3

Compare with Previous | Blame | View Log

#ifndef __SC_VPI_MODULE_H__
#define __SC_VPI_MODULE_H__
 
#include <systemc.h>
#include <assert.h>
#include <vpi_user.h>
 
static int sc_vpi_module_value_change(p_cb_data cb_data);
 
#define sc_vpi_module_read_output_int(obj, name)  \
{                                   \
    s_vpi_value  value_s;           \
    s_vpi_time   vpi_time_s;        \
                                    \
    value_s.format = vpiIntVal;     \
                                    \
    vpi_time_s.type = vpiSimTime;   \
    vpi_time_s.high = 0;            \
    vpi_time_s.low  = 0;            \
                                    \
    std::string path = m_hdl_name;  \
    path = path + "." + name;       \
    vpiHandle handle = vpi_handle_by_name(path.c_str(), NULL); \
    assert(handle != NULL);           \
                                      \
    vpi_get_value(handle, &value_s);  \
    obj.write(value_s.value.integer); \
}
 
#define sc_vpi_module_write_input_int(obj, name)  \
{                                   \
    s_vpi_value  value_s;           \
    s_vpi_time   vpi_time_s;        \
                                    \
    value_s.format = vpiIntVal;     \
                                    \
    vpi_time_s.type = vpiSimTime;   \
    vpi_time_s.high = 0;            \
    vpi_time_s.low  = 0;            \
                                    \
    std::string path = m_hdl_name;  \
    path = path + "." + name;       \
    vpiHandle handle = vpi_handle_by_name(path.c_str(), NULL); \
    assert(handle != NULL);           \
                                      \
    value_s.value.integer = obj.read();  \
    vpi_put_value(handle, &value_s, &vpi_time_s, vpiInertialDelay); \
}
 
class sc_vpi_module
{
public:   
    std::string      m_hdl_name;
    uint64_t         m_last_time;
    sc_signal<bool>  m_stop;
 
    sc_vpi_module(sc_module_name name) : m_hdl_name((std::string)name)
    {
        m_last_time = 0;
        m_stop.write(false);
    }
 
    // Simulation control    
    void stopSimulation()  { m_stop.write(true); }
    bool isStopped()       { return m_stop.read(); }
 
    virtual void read_outputs(void) { }
    virtual void write_inputs(void) { }
 
    bool register_signal(const char *name)
    {
        static s_vpi_value value_s;
        static s_vpi_time  vpi_time;
        s_cb_data          cb_data_s;
 
        vpi_time.high = 0;
        vpi_time.low  = 0;
        vpi_time.type = vpiSimTime;
 
        // For each I/O
        std::string path = m_hdl_name;
        path = path + "." + name;
        vpiHandle handle = vpi_handle_by_name(path.c_str(), NULL);
        if (!handle)
            return false;
 
        // Attach value change callbacks for outputs
        cb_data_s.user_data = (PLI_BYTE8*)this;
        cb_data_s.reason    = cbValueChange;
        cb_data_s.cb_rtn    = sc_vpi_module_value_change;
        cb_data_s.time      = &vpi_time;
        cb_data_s.value     = &value_s;
 
        value_s.format      = vpiIntVal;
 
        cb_data_s.obj  = handle;
        vpi_register_cb(&cb_data_s);
 
        return true;
    }
 
    int value_change(void)
    {
        s_vpi_time vpi_time_s;
 
        vpi_time_s.type = vpiSimTime;
        vpi_time_s.high = 0;
        vpi_time_s.low  = 0;
 
        // Outputs
        read_outputs();
 
        // Get current time
        uint64_t time_value = 0;
        s_vpi_time time_now;
        time_now.type = vpiSimTime;
        vpi_get_time (0, &time_now);
 
        time_value = time_now.high;
        time_value <<= 32;
        time_value |= time_now.low;
 
        // Update systemC TB
        if(sc_pending_activity())
            sc_start((int)(time_value-m_last_time),SC_NS);
 
        m_last_time = time_value;
 
        // Inputs
        write_inputs();
 
        if (isStopped())
            vpi_sim_control(vpiFinish, 0);
 
        return 0;
    }
};
 
static int sc_vpi_module_value_change(p_cb_data cb_data)
{
    sc_vpi_module *p = (sc_vpi_module*)cb_data->user_data;
    return p->value_change();
}
 
#endif
 

Compare with Previous | Blame | View Log

powered by: WebSVN 2.1.0

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