All Downloads are FREE. Search and download functionalities are using the official Maven repository.

inalhdl.spinalhdl-sim_2.11.1.11.0.source-code.VpiPlugin.cpp Maven / Gradle / Ivy

There is a newer version: 1.12.0
Show newest version
#ifndef SHMEM_FILENAME
#define SHMEM_FILENAME "./shmem_name"
#endif

#ifndef INITIAL_SEED
#define INITIAL_SEED 0x5EED5EED
#endif

#if defined(GHDL_PLUGIN)
    #define BUGGY_RW_SYNCH
#endif

#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include"SharedStruct.hpp"

using namespace std;

mt19937 global_mt_rand(INITIAL_SEED);
uniform_int_distribution<> bin_dis(0, 1);
managed_shared_memory segment;
SharedStruct* shared_struct;
stringstream ss;
string val_str;

PLI_INT32 start_cb(p_cb_data);
PLI_INT32 end_cb(p_cb_data);
PLI_INT32 rw_cb(p_cb_data);
PLI_INT32 ro_cb(p_cb_data);
PLI_INT32 delay_rw_cb(p_cb_data);
PLI_INT32 delay_ro_cb(p_cb_data);

#ifdef VCS_PLUGIN
extern "C" void entry_point_cb();
#endif

void set_error(string& error_string){
    vpi_printf("%s\n", error_string.c_str());
    shared_struct->data.resize(error_string.size());
    std::copy(error_string.begin(),
            error_string.end(),
            shared_struct->data.begin());
    shared_struct->data.push_back('\0');
    shared_struct->proc_status.store(ProcStatus::error);
    vpi_control(vpiFinish, -1);
}

bool check_error(){
    s_vpi_error_info err;
    if (vpi_chk_error(&err)) {
        if(err.level == vpiError) {
            string error_string("VPI error : ");
            error_string += err.message;
            set_error(error_string);
            return true;
        } else {
            vpi_printf("VPI message : %s\n", err.message);
        }
    }

    return false;
}

bool register_cb(PLI_INT32(*f)(p_cb_data),
        PLI_INT32 reason,
        int64_t cycles){
    s_cb_data cbData;
    memset(&cbData, 0, sizeof(cbData));
    s_vpi_time simuTime;
    if (cycles < 0){
        cbData.time = NULL;
    } else {
        cbData.time = &simuTime;
        simuTime.type = vpiSimTime;
        simuTime.high = (PLI_INT32) (cycles >> 32);
        simuTime.low = (PLI_INT32) (cycles & 0xFFFFFFFF);
    }

    cbData.reason = reason;
    cbData.cb_rtn = f;
    cbData.user_data = 0;
    cbData.value = 0;

    vpi_register_cb(&cbData);

    return check_error();
}

void entry_point_cb() {
    register_cb(start_cb, cbStartOfSimulation, -1);
    if(check_error()) return;
    register_cb(end_cb, cbEndOfSimulation, -1);
    if(check_error()) return;
    #ifdef BUGGY_RW_SYNCH
    register_cb(delay_ro_cb, cbAfterDelay, 0);
    #else
    register_cb(delay_rw_cb, cbAfterDelay, 0);
    #endif
    if(check_error()) return;
}

bool print_net_in_module(vpiHandle module_handle, stringstream &msg_ss){
    char* module_name = vpi_get_str(vpiName, module_handle);
    vpi_printf(" Signals of %s\n", module_name);
    msg_ss << " Signals of " << module_name << endl;

    vpiHandle net_iterator = vpi_iterate(vpiNet,module_handle);
    if(check_error()) return true;
    if(net_iterator){
        while(vpiHandle net_handle = vpi_scan(net_iterator)){
            if(check_error()) return true;
            string net_full_name = vpi_get_str(vpiFullName, net_handle);
            if(check_error()) return true;

            vpi_printf("    %s\n", net_full_name.c_str());
            msg_ss << "    " << net_full_name.c_str() << endl;

            vpiHandle net = vpi_handle_by_name(const_cast(net_full_name.c_str()),
                    (vpiHandle) NULL);

            if(check_error()) return true;

            string net_full_name2 = vpi_get_str(vpiFullName,net);
            if(check_error()) return true;
            if (net_full_name.compare(net_full_name2)){
                assert(0);
            }

            vpi_free_object(net_handle);
        }
    } else {
        vpi_printf("   No handles.\n");
        msg_ss << "   No handles." << endl;
    }

    vpi_printf("\n");
    msg_ss << endl;
    return false;
}

bool print_signals_cmd(){
    vpiHandle mod_iterator;
    vpiHandle mod_handle, child_mod_handle;
    std::vector mod_handles;
    stringstream msg_ss;

    mod_iterator = vpi_iterate(vpiModule,NULL);
    if(check_error()) return true;
    if(!mod_iterator){ return false; }
    mod_handle = vpi_scan(mod_iterator);
    if(check_error()) return true;
    while(mod_handle) {
        mod_handles.push_back(mod_handle);
        mod_handle = vpi_scan(mod_iterator);
        if(check_error()) return true;
    }

    while(mod_handles.size() != 0) {
        mod_handle = mod_handles.back();
        mod_handles.pop_back();
        if(print_net_in_module(mod_handle, msg_ss)) return true;
        mod_iterator = vpi_iterate(vpiModule, mod_handle);
        if(check_error()) return true;
        if(mod_iterator){ 
            child_mod_handle = vpi_scan(mod_iterator);
            if(check_error()) return true;
            while(child_mod_handle) {
                mod_handles.push_back(child_mod_handle);
                child_mod_handle = vpi_scan(mod_iterator);
                if(check_error()) return true;
            }
        }
        vpi_free_object(mod_handle);
        if(check_error()) return true;
    }
    
    const string msg_str(msg_ss.str());
    shared_struct->data.resize(msg_str.size());
    std::copy(msg_str.begin(), msg_str.end(), shared_struct->data.begin());
    shared_struct->data.push_back('\0');
    return false;
}

bool randomize_in_module(vpiHandle module_handle, mt19937& mt_rand){
    vpiHandle net_iterator = vpi_iterate(vpiNet, module_handle);
    if(check_error()) return true;
    if(net_iterator){
        while(vpiHandle net_handle = vpi_scan(net_iterator)) {
            if(check_error()) return true;
            
            s_vpi_value value_struct;
            bool invalid = false;

            value_struct.format = vpiBinStrVal;
            vpi_get_value(net_handle, &value_struct);
            if(check_error()) return true;
            val_str = (const char*) value_struct.value.str;
            for(char& c : val_str) {
                if ((c != '0') && (c != '1')) { 
                    c = bin_dis(mt_rand) == 0 ? '0' : '1';
                    invalid = true;
                }       
            }

            if(invalid){
                value_struct.value.str = (PLI_BYTE8*)val_str.c_str();
                vpi_put_value(net_handle, 
                          &value_struct, 
                          NULL, 
                          vpiNoDelay);
                if(check_error()) return true;
            }

            vpi_free_object(net_handle);
        }
    } 

    return false;
}

bool set_seed_cmd(){
    global_mt_rand.seed(shared_struct->seed.load());
    return false;
}

bool randomize_cmd(){
    vpiHandle mod_iterator;
    vpiHandle mod_handle, child_mod_handle;
    std::vector mod_handles;
    mt19937 mt_rand(shared_struct->seed.load());

    mod_iterator = vpi_iterate(vpiModule,NULL);
    if(check_error()) return true;
    if(!mod_iterator){ return false; }
    mod_handle = vpi_scan(mod_iterator);
    if(check_error()) return true;
    while(mod_handle) {
        mod_handles.push_back(mod_handle);
        mod_handle = vpi_scan(mod_iterator);
        if(check_error()) return true;
    }

    while(mod_handles.size() != 0) {
        mod_handle = mod_handles.back();
        mod_handles.pop_back();
        if(randomize_in_module(mod_handle, mt_rand)) return true;
        mod_iterator = vpi_iterate(vpiModule, mod_handle);
        if(check_error()) return true;
        if(mod_iterator){ 
            child_mod_handle = vpi_scan(mod_iterator);
            if(check_error()) return true;
            while(child_mod_handle) {
                mod_handles.push_back(child_mod_handle);
                child_mod_handle = vpi_scan(mod_iterator);
                if(check_error()) return true;
            }
        }
        vpi_free_object(mod_handle);
        if(check_error()) return true;
    }

    return false;
}



bool get_signal_handle_cmd(){
    int64_t handle = (int64_t)vpi_handle_by_name((PLI_BYTE8*) shared_struct->data.data(), 
            NULL);
    if(check_error()) return true; 
    if(!handle) {
        string error_string("vpi_handle_by_name failed with argument: ");
        error_string += (const char*) shared_struct->data.data();
        set_error(error_string);
        return true;
    }

    shared_struct->handle.store(handle);
    return false;
}


void sanitize_byte_str(char* byte_str){
    for(size_t i = 0; i< 8; i++) {
        if ((byte_str[i] != '0') && (byte_str[i] != '1')) { 
            byte_str[i] = bin_dis(global_mt_rand) == 0 ? '0' : '1';
        }
    } 
}

bool read_cmd_raw(vpiHandle handle){
    s_vpi_value value_struct;
    value_struct.format = vpiBinStrVal;
    shared_struct->data.clear();
    vpi_get_value(handle, &value_struct);
    if(check_error()) return true;
    size_t valueStrLen = strlen(value_struct.value.str);

    for(size_t i = 0; i < valueStrLen; i++){
        char c = value_struct.value.str[i];
        if ((c != '0') && (c != '1')) {
            vpi_printf("Warning, value '%c' at position %d is neither '0' or '1'. value set to '0'.\n", c, i);
        }
    }

    size_t valueByteLen = valueStrLen/8;
    size_t bitShift = valueStrLen%8;

    if(bitShift != 0) {
        char accum_string[9] = "00000000";
        accum_string[8] = '\0';
        uint8_t accum = 0;
        memcpy(accum_string+(8-bitShift),
                value_struct.value.str,
                bitShift);

        sanitize_byte_str(accum_string);
        accum = stoul(string(accum_string),
                nullptr,
                2);
        shared_struct->data.push_back(accum);
    }

    for(size_t i = 0; idata.push_back(accum);
    }

    return false;
}

bool read_cmd(){
    return read_cmd_raw((vpiHandle)shared_struct->handle.load());
}

bool read_mem_cmd(){
    vpiHandle handle = vpi_handle_by_index((vpiHandle)shared_struct->handle.load(),
                                           (PLI_INT32)shared_struct->index.load()); 

    if(check_error()) return true;
    if(read_cmd_raw(handle)) return true;
    vpi_free_object(handle);
    return check_error();
}

bool write_cmd_raw(vpiHandle handle){
    s_vpi_value value_struct;
    ss.str(std::string());
    ss << setw(8);
    ss << setfill('0');

    for(uint8_t& el: shared_struct->data) ss << bitset<8>(el);

    value_struct.format = vpiBinStrVal;
    val_str = ss.str();
    value_struct.value.str = (PLI_BYTE8*)val_str.c_str();
    vpi_put_value(handle, 
                  &value_struct, 
                  NULL, 
                  vpiNoDelay);

    return check_error();
}


bool write_cmd(){
    return write_cmd_raw((vpiHandle)shared_struct->handle.load());
}

bool write_mem_cmd(){
    vpiHandle handle = vpi_handle_by_index((vpiHandle)shared_struct->handle.load(),
                                           (PLI_INT32)shared_struct->index.load()); 

    if(check_error()) return true;
    if(write_cmd_raw(handle)) return true;
    vpi_free_object(handle);
    return check_error();
}

bool sleep_cmd(){
    #ifdef GHDL_PLUGIN
    register_cb(delay_ro_cb, cbAfterDelay, shared_struct->sleep_cycles);
    #else
    register_cb(delay_rw_cb, cbAfterDelay, shared_struct->sleep_cycles);
    #endif

    check_error();
    return true;
}

bool close_cmd(){
    vpi_control(vpiFinish, 0);
    if(!check_error()) shared_struct->proc_status.store(ProcStatus::closed);
    return true;
}

PLI_INT32 start_cb(p_cb_data){
    ifstream shmem_file(SHMEM_FILENAME);
    string shmem_name;
    getline(shmem_file, shmem_name);
    vpi_printf("Shared memory key : %s\n", shmem_name.c_str());
    segment = managed_shared_memory(open_only, shmem_name.c_str());
    auto ret_struct = segment.find("SharedStruct");
    shared_struct = ret_struct.first;
    shared_struct->time_precision = vpi_get(vpiTimePrecision, 0);
    vpi_printf("Start of simulation\n");
    return 0;
}

PLI_INT32 end_cb(p_cb_data){
    ProcStatus status = shared_struct->proc_status.load();
    if((status != ProcStatus::closed) && (status != ProcStatus::closed)) {
        string error_str("Unexpected termination of the simulation");
        set_error(error_str);
    }
    vpi_printf("End of simulation\n");
    return 0;
}

PLI_INT32 rw_cb(p_cb_data){
    bool run_simulation;
    do {
        run_simulation = false;
        shared_struct->proc_status.store(ProcStatus::ready);
        ProcStatus proc_status = shared_struct->check_not_ready();

        switch(proc_status){
            case ProcStatus::print_signals : run_simulation = print_signals_cmd(); break;
            case ProcStatus::get_signal_handle : run_simulation = get_signal_handle_cmd(); break;
            case ProcStatus::read : run_simulation = read_cmd(); break;
            case ProcStatus::read_mem : run_simulation = read_mem_cmd(); break;
            case ProcStatus::write : run_simulation = write_cmd(); break;
            case ProcStatus::write_mem : run_simulation = write_mem_cmd(); break;
            case ProcStatus::sleep : run_simulation = sleep_cmd(); break;
            case ProcStatus::set_seed : run_simulation = set_seed_cmd(); break;
            case ProcStatus::randomize : run_simulation = randomize_cmd(); break;
            case ProcStatus::close : run_simulation = close_cmd(); break;
            default : {
                          run_simulation = true; 
                          string error_string("Invalid state ");
                          error_string += to_string(static_cast(proc_status));
                          error_string += " detected";
                          set_error(error_string);
                          break;  
                      }
        }
    } while(!run_simulation);

    return 0;
}

PLI_INT32 ro_cb(p_cb_data){
    register_cb(delay_rw_cb, cbAfterDelay, 0);
    check_error();
    return 0;
}

PLI_INT32 delay_rw_cb(p_cb_data){
    register_cb(rw_cb, cbReadWriteSynch, 0);
    check_error();
    return 0;
}

PLI_INT32 delay_ro_cb(p_cb_data){
    register_cb(ro_cb, cbReadOnlySynch, 0);
    check_error();
    return 0;
}

#ifndef VCS_PLUGIN
extern "C" {
    void (*vlog_startup_routines[]) () = {
        entry_point_cb,
        0
    };
}
#endif




© 2015 - 2025 Weber Informatics LLC | Privacy Policy