| #include "HAL/Interrupts.hpp" |
| #include "ChipObject.h" |
| |
| extern void remapDigitalSource(bool analogTrigger, uint32_t &pin, uint8_t &module); |
| |
| struct Interrupt // FIXME: why is this internal? |
| { |
| tInterrupt *anInterrupt; |
| tInterruptManager *manager; |
| }; |
| |
| void* initializeInterrupts(uint32_t interruptIndex, bool watcher, int32_t *status) |
| { |
| Interrupt* anInterrupt = new Interrupt(); |
| // Expects the calling leaf class to allocate an interrupt index. |
| anInterrupt->anInterrupt = tInterrupt::create(interruptIndex, status); |
| anInterrupt->anInterrupt->writeConfig_WaitForAck(false, status); |
| anInterrupt->manager = new tInterruptManager( |
| (1 << interruptIndex) | (1 << (interruptIndex + 8)), watcher, status); |
| return anInterrupt; |
| } |
| |
| void cleanInterrupts(void* interrupt_pointer, int32_t *status) |
| { |
| Interrupt* anInterrupt = (Interrupt*)interrupt_pointer; |
| delete anInterrupt->anInterrupt; |
| delete anInterrupt->manager; |
| anInterrupt->anInterrupt = NULL; |
| anInterrupt->manager = NULL; |
| } |
| |
| /** |
| * In synchronous mode, wait for the defined interrupt to occur. |
| * @param timeout Timeout in seconds |
| * @param ignorePrevious If true, ignore interrupts that happened before |
| * waitForInterrupt was called. |
| * @return The mask of interrupts that fired. |
| */ |
| uint32_t waitForInterrupt(void* interrupt_pointer, double timeout, bool ignorePrevious, int32_t *status) |
| { |
| uint32_t result; |
| Interrupt* anInterrupt = (Interrupt*)interrupt_pointer; |
| |
| result = anInterrupt->manager->watch((int32_t)(timeout * 1e3), ignorePrevious, status); |
| |
| // Don't report a timeout as an error - the return code is enough to tell |
| // that a timeout happened. |
| if(*status == -NiFpga_Status_IrqTimeout) { |
| *status = NiFpga_Status_Success; |
| } |
| |
| return result; |
| } |
| |
| /** |
| * Enable interrupts to occur on this input. |
| * Interrupts are disabled when the RequestInterrupt call is made. This gives time to do the |
| * setup of the other options before starting to field interrupts. |
| */ |
| void enableInterrupts(void* interrupt_pointer, int32_t *status) |
| { |
| Interrupt* anInterrupt = (Interrupt*)interrupt_pointer; |
| anInterrupt->manager->enable(status); |
| } |
| |
| /** |
| * Disable Interrupts without without deallocating structures. |
| */ |
| void disableInterrupts(void* interrupt_pointer, int32_t *status) |
| { |
| Interrupt* anInterrupt = (Interrupt*)interrupt_pointer; |
| anInterrupt->manager->disable(status); |
| } |
| |
| /** |
| * Return the timestamp for the rising interrupt that occurred most recently. |
| * This is in the same time domain as GetClock(). |
| * @return Timestamp in seconds since boot. |
| */ |
| double readRisingTimestamp(void* interrupt_pointer, int32_t *status) |
| { |
| Interrupt* anInterrupt = (Interrupt*)interrupt_pointer; |
| uint32_t timestamp = anInterrupt->anInterrupt->readRisingTimeStamp(status); |
| return timestamp * 1e-6; |
| } |
| |
| /** |
| * Return the timestamp for the falling interrupt that occurred most recently. |
| * This is in the same time domain as GetClock(). |
| * @return Timestamp in seconds since boot. |
| */ |
| double readFallingTimestamp(void* interrupt_pointer, int32_t *status) |
| { |
| Interrupt* anInterrupt = (Interrupt*)interrupt_pointer; |
| uint32_t timestamp = anInterrupt->anInterrupt->readFallingTimeStamp(status); |
| return timestamp * 1e-6; |
| } |
| |
| void requestInterrupts(void* interrupt_pointer, uint8_t routing_module, uint32_t routing_pin, |
| bool routing_analog_trigger, int32_t *status) |
| { |
| Interrupt* anInterrupt = (Interrupt*)interrupt_pointer; |
| anInterrupt->anInterrupt->writeConfig_WaitForAck(false, status); |
| remapDigitalSource(routing_analog_trigger, routing_pin, routing_module); |
| anInterrupt->anInterrupt->writeConfig_Source_AnalogTrigger(routing_analog_trigger, status); |
| anInterrupt->anInterrupt->writeConfig_Source_Channel(routing_pin, status); |
| anInterrupt->anInterrupt->writeConfig_Source_Module(routing_module, status); |
| } |
| |
| void attachInterruptHandler(void* interrupt_pointer, InterruptHandlerFunction handler, void* param, |
| int32_t *status) |
| { |
| Interrupt* anInterrupt = (Interrupt*)interrupt_pointer; |
| anInterrupt->manager->registerHandler(handler, param, status); |
| } |
| |
| void setInterruptUpSourceEdge(void* interrupt_pointer, bool risingEdge, bool fallingEdge, |
| int32_t *status) |
| { |
| Interrupt* anInterrupt = (Interrupt*)interrupt_pointer; |
| anInterrupt->anInterrupt->writeConfig_RisingEdge(risingEdge, status); |
| anInterrupt->anInterrupt->writeConfig_FallingEdge(fallingEdge, status); |
| } |