forked from github/verilator
79 lines
2.3 KiB
C++
79 lines
2.3 KiB
C++
|
// -*- mode: C++; c-file-style: "cc-mode" -*-
|
||
|
//*************************************************************************
|
||
|
//
|
||
|
// Copyright 2018-2018 by Wilson Snyder. This program is free software; you can
|
||
|
// redistribute it and/or modify it under the terms of either the GNU
|
||
|
// Lesser General Public License Version 3 or the Perl Artistic License.
|
||
|
// Version 2.0.
|
||
|
//
|
||
|
// Verilator 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.
|
||
|
//
|
||
|
//*************************************************************************
|
||
|
|
||
|
#include <atomic>
|
||
|
#include <cstdio>
|
||
|
#include <iostream>
|
||
|
#include <unistd.h>
|
||
|
#include "svdpi.h"
|
||
|
|
||
|
//======================================================================
|
||
|
|
||
|
#if defined(VERILATOR)
|
||
|
# ifdef T_DPI_THREADS_COLLIDE
|
||
|
# include "Vt_dpi_threads_collide__Dpi.h"
|
||
|
# else
|
||
|
# include "Vt_dpi_threads__Dpi.h"
|
||
|
# endif
|
||
|
#elif defined(VCS)
|
||
|
# include "../vc_hdrs.h"
|
||
|
#elif defined(CADENCE)
|
||
|
# define NEED_EXTERNS
|
||
|
#else
|
||
|
# error "Unknown simulator for DPI test"
|
||
|
#endif
|
||
|
|
||
|
#ifdef NEED_EXTERNS
|
||
|
extern "C" {
|
||
|
extern void dpii_sys_task();
|
||
|
extern int dpii_failure();
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
//======================================================================
|
||
|
|
||
|
struct state {
|
||
|
std::atomic<bool> task_is_running;
|
||
|
std::atomic<int> failure;
|
||
|
state() : task_is_running(false)
|
||
|
, failure(false) {}
|
||
|
};
|
||
|
|
||
|
static state st;
|
||
|
|
||
|
void dpii_sys_task() {
|
||
|
bool other_task_running = atomic_exchange(&st.task_is_running, true);
|
||
|
if (other_task_running) {
|
||
|
// Another task is running. This is a collision.
|
||
|
st.failure = 1;
|
||
|
std::cerr << "t_dpi_threads_c.cpp dpii_sys_task() saw threads collide.\n";
|
||
|
} else {
|
||
|
std::cerr << "t_dpi_threads_c.cpp dpii_sys_task() no collision. @" << &st.task_is_running << "\n";
|
||
|
}
|
||
|
|
||
|
// Spend some time in the DPI call, so that if we can have a collision
|
||
|
// we probably will. Technically this is not guaranteed to detect every
|
||
|
// race. However, one second is so much greater than the expected
|
||
|
// runtime of everything else in the test, it really should pick up on
|
||
|
// races just about all of the time.
|
||
|
sleep(1);
|
||
|
|
||
|
atomic_exchange(&st.task_is_running, false);
|
||
|
}
|
||
|
|
||
|
int dpii_failure() {
|
||
|
return st.failure;
|
||
|
}
|