Change hal sim to use spinlocks (#1291)

This makes callback registration completely thread safe.

This patch also uses templates and macros to dramatically reduce the amount of
manual boilerplate.
This commit is contained in:
Peter Johnson
2018-09-03 16:08:07 -07:00
committed by GitHub
parent 67b1c85315
commit c0ff6198b3
65 changed files with 1305 additions and 7639 deletions

View File

@@ -7,7 +7,6 @@
#include "../PortsInternal.h"
#include "DigitalPWMDataInternal.h"
#include "mockdata/NotifyCallbackHelpers.h"
using namespace hal;
@@ -22,120 +21,9 @@ void InitializeDigitalPWMData() {
DigitalPWMData* hal::SimDigitalPWMData;
void DigitalPWMData::ResetData() {
m_initialized = false;
m_initializedCallbacks = nullptr;
m_dutyCycle = false;
m_dutyCycleCallbacks = nullptr;
m_pin = 0;
m_pinCallbacks = nullptr;
}
int32_t DigitalPWMData::RegisterInitializedCallback(HAL_NotifyCallback callback,
void* param,
HAL_Bool initialNotify) {
// Must return -1 on a null callback for error handling
if (callback == nullptr) return -1;
int32_t newUid = 0;
{
std::lock_guard<wpi::mutex> lock(m_registerMutex);
m_initializedCallbacks = RegisterCallback(
m_initializedCallbacks, "Initialized", callback, param, &newUid);
}
if (initialNotify) {
// We know that the callback is not null because of earlier null check
HAL_Value value = MakeBoolean(GetInitialized());
callback("Initialized", param, &value);
}
return newUid;
}
void DigitalPWMData::CancelInitializedCallback(int32_t uid) {
m_initializedCallbacks = CancelCallback(m_initializedCallbacks, uid);
}
void DigitalPWMData::InvokeInitializedCallback(HAL_Value value) {
InvokeCallback(m_initializedCallbacks, "Initialized", &value);
}
HAL_Bool DigitalPWMData::GetInitialized() { return m_initialized; }
void DigitalPWMData::SetInitialized(HAL_Bool initialized) {
HAL_Bool oldValue = m_initialized.exchange(initialized);
if (oldValue != initialized) {
InvokeInitializedCallback(MakeBoolean(initialized));
}
}
int32_t DigitalPWMData::RegisterDutyCycleCallback(HAL_NotifyCallback callback,
void* param,
HAL_Bool initialNotify) {
// Must return -1 on a null callback for error handling
if (callback == nullptr) return -1;
int32_t newUid = 0;
{
std::lock_guard<wpi::mutex> lock(m_registerMutex);
m_dutyCycleCallbacks = RegisterCallback(m_dutyCycleCallbacks, "DutyCycle",
callback, param, &newUid);
}
if (initialNotify) {
// We know that the callback is not null because of earlier null check
HAL_Value value = MakeDouble(GetDutyCycle());
callback("DutyCycle", param, &value);
}
return newUid;
}
void DigitalPWMData::CancelDutyCycleCallback(int32_t uid) {
m_dutyCycleCallbacks = CancelCallback(m_dutyCycleCallbacks, uid);
}
void DigitalPWMData::InvokeDutyCycleCallback(HAL_Value value) {
InvokeCallback(m_dutyCycleCallbacks, "DutyCycle", &value);
}
double DigitalPWMData::GetDutyCycle() { return m_dutyCycle; }
void DigitalPWMData::SetDutyCycle(double dutyCycle) {
double oldValue = m_dutyCycle.exchange(dutyCycle);
if (oldValue != dutyCycle) {
InvokeDutyCycleCallback(MakeDouble(dutyCycle));
}
}
int32_t DigitalPWMData::RegisterPinCallback(HAL_NotifyCallback callback,
void* param,
HAL_Bool initialNotify) {
// Must return -1 on a null callback for error handling
if (callback == nullptr) return -1;
int32_t newUid = 0;
{
std::lock_guard<wpi::mutex> lock(m_registerMutex);
m_pinCallbacks =
RegisterCallback(m_pinCallbacks, "Pin", callback, param, &newUid);
}
if (initialNotify) {
// We know that the callback is not null because of earlier null check
HAL_Value value = MakeInt(GetPin());
callback("Pin", param, &value);
}
return newUid;
}
void DigitalPWMData::CancelPinCallback(int32_t uid) {
m_pinCallbacks = CancelCallback(m_pinCallbacks, uid);
}
void DigitalPWMData::InvokePinCallback(HAL_Value value) {
InvokeCallback(m_pinCallbacks, "Pin", &value);
}
int32_t DigitalPWMData::GetPin() { return m_pin; }
void DigitalPWMData::SetPin(int32_t pin) {
int32_t oldValue = m_pin.exchange(pin);
if (oldValue != pin) {
InvokePinCallback(MakeInt(pin));
}
initialized.Reset(false);
dutyCycle.Reset(0.0);
pin.Reset(0);
}
extern "C" {
@@ -143,73 +31,23 @@ void HALSIM_ResetDigitalPWMData(int32_t index) {
SimDigitalPWMData[index].ResetData();
}
int32_t HALSIM_RegisterDigitalPWMInitializedCallback(
int32_t index, HAL_NotifyCallback callback, void* param,
HAL_Bool initialNotify) {
return SimDigitalPWMData[index].RegisterInitializedCallback(callback, param,
initialNotify);
}
#define DEFINE_CAPI(TYPE, CAPINAME, LOWERNAME) \
HAL_SIMDATAVALUE_DEFINE_CAPI(TYPE, HALSIM, DigitalPWM##CAPINAME, \
SimDigitalPWMData, LOWERNAME)
void HALSIM_CancelDigitalPWMInitializedCallback(int32_t index, int32_t uid) {
SimDigitalPWMData[index].CancelInitializedCallback(uid);
}
DEFINE_CAPI(HAL_Bool, Initialized, initialized)
DEFINE_CAPI(double, DutyCycle, dutyCycle)
DEFINE_CAPI(int32_t, Pin, pin)
HAL_Bool HALSIM_GetDigitalPWMInitialized(int32_t index) {
return SimDigitalPWMData[index].GetInitialized();
}
void HALSIM_SetDigitalPWMInitialized(int32_t index, HAL_Bool initialized) {
SimDigitalPWMData[index].SetInitialized(initialized);
}
int32_t HALSIM_RegisterDigitalPWMDutyCycleCallback(int32_t index,
HAL_NotifyCallback callback,
void* param,
HAL_Bool initialNotify) {
return SimDigitalPWMData[index].RegisterDutyCycleCallback(callback, param,
initialNotify);
}
void HALSIM_CancelDigitalPWMDutyCycleCallback(int32_t index, int32_t uid) {
SimDigitalPWMData[index].CancelDutyCycleCallback(uid);
}
double HALSIM_GetDigitalPWMDutyCycle(int32_t index) {
return SimDigitalPWMData[index].GetDutyCycle();
}
void HALSIM_SetDigitalPWMDutyCycle(int32_t index, double dutyCycle) {
SimDigitalPWMData[index].SetDutyCycle(dutyCycle);
}
int32_t HALSIM_RegisterDigitalPWMPinCallback(int32_t index,
HAL_NotifyCallback callback,
void* param,
HAL_Bool initialNotify) {
return SimDigitalPWMData[index].RegisterPinCallback(callback, param,
initialNotify);
}
void HALSIM_CancelDigitalPWMPinCallback(int32_t index, int32_t uid) {
SimDigitalPWMData[index].CancelPinCallback(uid);
}
int32_t HALSIM_GetDigitalPWMPin(int32_t index) {
return SimDigitalPWMData[index].GetPin();
}
void HALSIM_SetDigitalPWMPin(int32_t index, int32_t pin) {
SimDigitalPWMData[index].SetPin(pin);
}
#define REGISTER(NAME) \
SimDigitalPWMData[index].NAME.RegisterCallback(callback, param, initialNotify)
void HALSIM_RegisterDigitalPWMAllCallbacks(int32_t index,
HAL_NotifyCallback callback,
void* param,
HAL_Bool initialNotify) {
SimDigitalPWMData[index].RegisterInitializedCallback(callback, param,
initialNotify);
SimDigitalPWMData[index].RegisterDutyCycleCallback(callback, param,
initialNotify);
SimDigitalPWMData[index].RegisterPinCallback(callback, param, initialNotify);
REGISTER(initialized);
REGISTER(dutyCycle);
REGISTER(pin);
}
} // extern "C"