2013-12-15 18:30:16 -05:00
|
|
|
/*----------------------------------------------------------------------------*/
|
2015-06-25 15:07:55 -04:00
|
|
|
/* Copyright (c) FIRST 2008. All Rights Reserved.
|
|
|
|
|
*/
|
2013-12-15 18:30:16 -05:00
|
|
|
/* Open Source Software - may be modified and shared by FRC teams. The code */
|
|
|
|
|
/* must be accompanied by the FIRST BSD license file in $(WIND_BASE)/WPILib. */
|
|
|
|
|
/*----------------------------------------------------------------------------*/
|
|
|
|
|
|
|
|
|
|
#include "InterruptableSensorBase.h"
|
|
|
|
|
#include "Utility.h"
|
2014-10-08 14:52:24 -04:00
|
|
|
#include "WPIErrors.h"
|
2013-12-15 18:30:16 -05:00
|
|
|
|
2014-08-06 09:46:50 -04:00
|
|
|
Resource *InterruptableSensorBase::m_interrupts = NULL;
|
|
|
|
|
|
2015-06-25 15:07:55 -04:00
|
|
|
InterruptableSensorBase::InterruptableSensorBase() {
|
|
|
|
|
m_interrupt = NULL;
|
|
|
|
|
Resource::CreateResourceObject(&m_interrupts, interrupt_kNumSystems);
|
2013-12-15 18:30:16 -05:00
|
|
|
}
|
|
|
|
|
|
2015-06-25 15:07:55 -04:00
|
|
|
InterruptableSensorBase::~InterruptableSensorBase() {}
|
2013-12-15 18:30:16 -05:00
|
|
|
|
2014-10-29 15:23:55 -04:00
|
|
|
/**
|
|
|
|
|
* Request one of the 8 interrupts asynchronously on this digital input.
|
2015-06-25 15:07:55 -04:00
|
|
|
* Request interrupts in asynchronous mode where the user's interrupt handler
|
|
|
|
|
* will be
|
|
|
|
|
* called when the interrupt fires. Users that want control over the thread
|
|
|
|
|
* priority
|
2014-10-29 15:23:55 -04:00
|
|
|
* should use the synchronous method with their own spawned thread.
|
|
|
|
|
* The default is interrupt on rising edges only.
|
|
|
|
|
*/
|
2015-06-25 15:07:55 -04:00
|
|
|
void InterruptableSensorBase::RequestInterrupts(
|
|
|
|
|
InterruptHandlerFunction handler, void *param) {
|
|
|
|
|
if (StatusIsFatal()) return;
|
|
|
|
|
uint32_t index = m_interrupts->Allocate("Async Interrupt");
|
|
|
|
|
if (index == ~0ul) {
|
|
|
|
|
CloneError(m_interrupts);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
m_interruptIndex = index;
|
|
|
|
|
|
|
|
|
|
// Creates a manager too
|
|
|
|
|
AllocateInterrupts(false);
|
|
|
|
|
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
requestInterrupts(m_interrupt, GetModuleForRouting(), GetChannelForRouting(),
|
|
|
|
|
GetAnalogTriggerForRouting(), &status);
|
|
|
|
|
SetUpSourceEdge(true, false);
|
|
|
|
|
attachInterruptHandler(m_interrupt, handler, param, &status);
|
|
|
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
2014-10-08 14:52:24 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2014-10-29 15:23:55 -04:00
|
|
|
* Request one of the 8 interrupts synchronously on this digital input.
|
2015-06-25 15:07:55 -04:00
|
|
|
* Request interrupts in synchronous mode where the user program will have to
|
|
|
|
|
* explicitly
|
2014-10-29 15:23:55 -04:00
|
|
|
* wait for the interrupt to occur using WaitForInterrupt.
|
2014-10-08 14:52:24 -04:00
|
|
|
* The default is interrupt on rising edges only.
|
|
|
|
|
*/
|
2015-06-25 15:07:55 -04:00
|
|
|
void InterruptableSensorBase::RequestInterrupts() {
|
|
|
|
|
if (StatusIsFatal()) return;
|
|
|
|
|
uint32_t index = m_interrupts->Allocate("Sync Interrupt");
|
|
|
|
|
if (index == ~0ul) {
|
|
|
|
|
CloneError(m_interrupts);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
m_interruptIndex = index;
|
|
|
|
|
|
|
|
|
|
AllocateInterrupts(true);
|
|
|
|
|
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
requestInterrupts(m_interrupt, GetModuleForRouting(), GetChannelForRouting(),
|
|
|
|
|
GetAnalogTriggerForRouting(), &status);
|
|
|
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
|
|
|
SetUpSourceEdge(true, false);
|
2014-10-08 14:52:24 -04:00
|
|
|
}
|
|
|
|
|
|
2015-06-25 15:07:55 -04:00
|
|
|
void InterruptableSensorBase::AllocateInterrupts(bool watcher) {
|
|
|
|
|
wpi_assert(m_interrupt == NULL);
|
|
|
|
|
// Expects the calling leaf class to allocate an interrupt index.
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
m_interrupt = initializeInterrupts(m_interruptIndex, watcher, &status);
|
|
|
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
2013-12-15 18:30:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Cancel interrupts on this device.
|
|
|
|
|
* This deallocates all the chipobject structures and disables any interrupts.
|
|
|
|
|
*/
|
2015-06-25 15:07:55 -04:00
|
|
|
void InterruptableSensorBase::CancelInterrupts() {
|
|
|
|
|
if (StatusIsFatal()) return;
|
|
|
|
|
wpi_assert(m_interrupt != NULL);
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
cleanInterrupts(m_interrupt, &status);
|
|
|
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
|
|
|
m_interrupt = NULL;
|
|
|
|
|
m_interrupts->Free(m_interruptIndex);
|
2013-12-15 18:30:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2015-06-25 15:07:55 -04:00
|
|
|
* In synchronous mode, wait for the defined interrupt to occur. You should
|
|
|
|
|
* <b>NOT</b> attempt to read the
|
|
|
|
|
* sensor from another thread while waiting for an interrupt. This is not
|
|
|
|
|
* threadsafe, and can cause
|
2014-12-24 14:19:01 -05:00
|
|
|
* memory corruption
|
2013-12-15 18:30:16 -05:00
|
|
|
* @param timeout Timeout in seconds
|
2014-10-05 17:17:59 -04:00
|
|
|
* @param ignorePrevious If true, ignore interrupts that happened before
|
|
|
|
|
* WaitForInterrupt was called.
|
|
|
|
|
* @return What interrupts fired
|
2013-12-15 18:30:16 -05:00
|
|
|
*/
|
2015-06-25 15:07:55 -04:00
|
|
|
InterruptableSensorBase::WaitResult InterruptableSensorBase::WaitForInterrupt(
|
|
|
|
|
float timeout, bool ignorePrevious) {
|
|
|
|
|
if (StatusIsFatal()) return InterruptableSensorBase::kTimeout;
|
|
|
|
|
wpi_assert(m_interrupt != NULL);
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
uint32_t result;
|
2014-10-05 17:17:59 -04:00
|
|
|
|
2015-06-25 15:07:55 -04:00
|
|
|
result = waitForInterrupt(m_interrupt, timeout, ignorePrevious, &status);
|
|
|
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
2014-10-05 17:17:59 -04:00
|
|
|
|
2015-06-25 15:07:55 -04:00
|
|
|
return static_cast<WaitResult>(result);
|
2013-12-15 18:30:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Enable interrupts to occur on this input.
|
2015-06-25 15:07:55 -04:00
|
|
|
* Interrupts are disabled when the RequestInterrupt call is made. This gives
|
|
|
|
|
* time to do the
|
2013-12-15 18:30:16 -05:00
|
|
|
* setup of the other options before starting to field interrupts.
|
|
|
|
|
*/
|
2015-06-25 15:07:55 -04:00
|
|
|
void InterruptableSensorBase::EnableInterrupts() {
|
|
|
|
|
if (StatusIsFatal()) return;
|
|
|
|
|
wpi_assert(m_interrupt != NULL);
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
enableInterrupts(m_interrupt, &status);
|
|
|
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
2013-12-15 18:30:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Disable Interrupts without without deallocating structures.
|
|
|
|
|
*/
|
2015-06-25 15:07:55 -04:00
|
|
|
void InterruptableSensorBase::DisableInterrupts() {
|
|
|
|
|
if (StatusIsFatal()) return;
|
|
|
|
|
wpi_assert(m_interrupt != NULL);
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
disableInterrupts(m_interrupt, &status);
|
|
|
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
2013-12-15 18:30:16 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2014-10-05 17:17:59 -04:00
|
|
|
* Return the timestamp for the rising interrupt that occurred most recently.
|
2013-12-15 18:30:16 -05:00
|
|
|
* This is in the same time domain as GetClock().
|
2014-10-05 17:17:59 -04:00
|
|
|
* The rising-edge interrupt should be enabled with
|
|
|
|
|
* {@link #DigitalInput.SetUpSourceEdge}
|
2013-12-15 18:30:16 -05:00
|
|
|
* @return Timestamp in seconds since boot.
|
|
|
|
|
*/
|
2015-06-25 15:07:55 -04:00
|
|
|
double InterruptableSensorBase::ReadRisingTimestamp() {
|
|
|
|
|
if (StatusIsFatal()) return 0.0;
|
|
|
|
|
wpi_assert(m_interrupt != NULL);
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
double timestamp = readRisingTimestamp(m_interrupt, &status);
|
|
|
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
|
|
|
return timestamp;
|
2014-10-05 17:17:59 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Return the timestamp for the falling interrupt that occurred most recently.
|
|
|
|
|
* This is in the same time domain as GetClock().
|
|
|
|
|
* The falling-edge interrupt should be enabled with
|
|
|
|
|
* {@link #DigitalInput.SetUpSourceEdge}
|
|
|
|
|
* @return Timestamp in seconds since boot.
|
|
|
|
|
*/
|
2015-06-25 15:07:55 -04:00
|
|
|
double InterruptableSensorBase::ReadFallingTimestamp() {
|
|
|
|
|
if (StatusIsFatal()) return 0.0;
|
|
|
|
|
wpi_assert(m_interrupt != NULL);
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
double timestamp = readFallingTimestamp(m_interrupt, &status);
|
|
|
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
|
|
|
return timestamp;
|
2013-12-15 18:30:16 -05:00
|
|
|
}
|
2014-10-08 14:52:24 -04:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Set which edge to trigger interrupts on
|
|
|
|
|
*
|
|
|
|
|
* @param risingEdge
|
|
|
|
|
* true to interrupt on rising edge
|
|
|
|
|
* @param fallingEdge
|
|
|
|
|
* true to interrupt on falling edge
|
|
|
|
|
*/
|
2015-06-25 15:07:55 -04:00
|
|
|
void InterruptableSensorBase::SetUpSourceEdge(bool risingEdge,
|
|
|
|
|
bool fallingEdge) {
|
|
|
|
|
if (StatusIsFatal()) return;
|
|
|
|
|
if (m_interrupt == NULL) {
|
|
|
|
|
wpi_setWPIErrorWithContext(
|
|
|
|
|
NullParameter,
|
|
|
|
|
"You must call RequestInterrupts before SetUpSourceEdge");
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
if (m_interrupt != NULL) {
|
|
|
|
|
int32_t status = 0;
|
|
|
|
|
setInterruptUpSourceEdge(m_interrupt, risingEdge, fallingEdge, &status);
|
|
|
|
|
wpi_setErrorWithContext(status, getHALErrorMessage(status));
|
|
|
|
|
}
|
2014-10-08 14:52:24 -04:00
|
|
|
}
|