2018-07-23 00:54:28 +00:00
|
|
|
// -*- 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
|
2020-03-21 15:24:24 +00:00
|
|
|
// Lesser General Public License Version 3 or the Perl Artistic License
|
2018-07-23 00:54:28 +00:00
|
|
|
// Version 2.0.
|
2020-03-21 15:24:24 +00:00
|
|
|
// SPDX-License-Identifier: LGPL-3.0-only OR Artistic-2.0
|
2018-07-23 00:54:28 +00:00
|
|
|
//
|
|
|
|
//*************************************************************************
|
|
|
|
|
|
|
|
#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" {
|
2020-10-28 00:03:17 +00:00
|
|
|
extern void dpii_sys_task();
|
|
|
|
extern int dpii_failure();
|
2018-07-23 00:54:28 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//======================================================================
|
|
|
|
|
|
|
|
struct state {
|
|
|
|
std::atomic<bool> task_is_running;
|
|
|
|
std::atomic<int> failure;
|
2019-11-10 01:35:12 +00:00
|
|
|
state()
|
|
|
|
: task_is_running(false)
|
|
|
|
, failure(false) {}
|
2018-07-23 00:54:28 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
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 {
|
2020-10-28 00:03:17 +00:00
|
|
|
std::cerr << "t_dpi_threads_c.cpp dpii_sys_task() no collision. @" << &st.task_is_running
|
|
|
|
<< "\n";
|
2018-07-23 00:54:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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);
|
|
|
|
}
|
|
|
|
|
2019-11-10 01:35:12 +00:00
|
|
|
int dpii_failure() { return st.failure; }
|