rusefi/firmware/hw_layer/digital_input/digital_input_exti.cpp

214 lines
4.3 KiB
C++

/*
* digital_input_exti.cpp
*
* Created on: Dec 18, 2018
* @author Andrey Belomutskiy, (c) 2012-2021
*/
#include "pch.h"
#if HAL_USE_PAL && EFI_PROD_CODE
#include "digital_input_exti.h"
/**
* EXTI is a funny thing: you can only use same pin on one port. For example, you can use
* PA0 PB5 PE2 PD7
* but you cannot use
* PA0 PB0 PE2 PD7
* because pin '0' would be used on two different ports
*/
#ifdef STM32_I2C_I2C1_IRQ_PRIORITY
void efiExtiInit() {
nvicEnableVector(I2C1_EV_IRQn, STM32_I2C_I2C1_IRQ_PRIORITY);
}
struct ExtiChannel
{
ExtiCallback Callback = nullptr;
void* CallbackData;
efitick_t Timestamp = 0;
const char* Name = nullptr;
};
static ExtiChannel channels[16];
// EXT is not able to give you the front direction but you could read the pin in the callback.
void efiExtiEnablePin(const char *msg, brain_pin_e brainPin, uint32_t mode, ExtiCallback cb, void *cb_data) {
/* paranoid check, in case of GPIO_UNASSIGNED getHwPort will return NULL
* and we will fail on next check */
if (!isBrainPinValid(brainPin)) {
return;
}
ioportid_t port = getHwPort(msg, brainPin);
if (port == NULL) {
return;
}
bool wasUsed = brain_pin_markUsed(brainPin, msg);
if (wasUsed) {
// error condition we shall bail
return;
}
int index = getHwPin(msg, brainPin);
auto& channel = channels[index];
/* is this index already used? */
if (channel.Callback) {
firmwareError(CUSTOM_ERR_PIN_ALREADY_USED_2, "%s: pin %s/index %d: exti index already used by %s",
msg,
hwPortname(brainPin),
index,
channel.Name);
return;
}
channel.Name = msg;
channel.Callback = cb;
channel.CallbackData = cb_data;
channel.Timestamp = 0;
ioline_t line = PAL_LINE(port, index);
palEnableLineEvent(line, mode);
}
void efiExtiDisablePin(brain_pin_e brainPin)
{
/* paranoid check, in case of GPIO_UNASSIGNED getHwPort will return NULL
* and we will fail on next check */
if (!isBrainPinValid(brainPin))
return;
ioportid_t port = getHwPort("exti", brainPin);
if (port == NULL)
return;
brain_pin_markUnused(brainPin);
int index = getHwPin("exti", brainPin);
auto& channel = channels[index];
/* is this index was used? */
if (!channel.Callback) {
return;
}
ioline_t line = PAL_LINE(port, index);
palDisableLineEvent(line);
/* mark unused */
channel.Timestamp = 0;
channel.Name = nullptr;
channel.Callback = nullptr;
channel.CallbackData = nullptr;
}
digital_input_s* startDigitalCaptureExti(const char *msg, brain_pin_e brainPin) {
return nullptr;
}
#if ! EFI_ICU_INPUTS
digital_input_s* startDigitalCapture(const char *msg, brain_pin_e brainPin) {
return startDigitalCaptureExti(msg, brainPin);
}
#endif // EFI_ICU_INPUTS
static inline void triggerInterrupt() {
// Manually fire the I2C1_EV interrupt, it will be queued after this interrupt returns
NVIC->STIR = I2C1_EV_IRQn;
}
CH_IRQ_HANDLER(STM32_I2C1_EVENT_HANDLER) {
OSAL_IRQ_PROLOGUE();
for (size_t i = 0; i < 16; i++) {
auto& channel = channels[i];
// get the timestamp out under lock
// todo: lock freeeeee!
__disable_irq();
auto timestamp = channel.Timestamp;
channel.Timestamp = 0;
__enable_irq();
if (timestamp != 0) {
channel.Callback(channel.CallbackData, timestamp);
}
}
OSAL_IRQ_EPILOGUE();
}
void handleExtiIsr(uint8_t index) {
// No need to lock anything, we're already the highest-pri interrupt!
uint32_t pr;
extiGetAndClearGroup1(1U << index, pr);
if (pr & (1 << index)) {
auto& timestamp = channels[index].Timestamp;
if (timestamp == 0) {
timestamp = getTimeNowNt();
}
triggerInterrupt();
}
}
CH_FAST_IRQ_HANDLER(Vector58) {
handleExtiIsr(0);
}
CH_FAST_IRQ_HANDLER(Vector5C) {
handleExtiIsr(1);
}
CH_FAST_IRQ_HANDLER(Vector60) {
handleExtiIsr(2);
}
CH_FAST_IRQ_HANDLER(Vector64) {
handleExtiIsr(3);
}
CH_FAST_IRQ_HANDLER(Vector68) {
handleExtiIsr(4);
}
CH_FAST_IRQ_HANDLER(Vector9C) {
handleExtiIsr(5);
handleExtiIsr(6);
handleExtiIsr(7);
handleExtiIsr(8);
handleExtiIsr(9);
}
CH_FAST_IRQ_HANDLER(VectorE0) {
handleExtiIsr(10);
handleExtiIsr(11);
handleExtiIsr(12);
handleExtiIsr(13);
handleExtiIsr(14);
handleExtiIsr(15);
}
#else // not STM32
// TODO: non-stm32 exti
void efiExtiInit() {
firmwareError(OBD_PCM_Processor_Fault, "exti not supported");
}
void efiExtiEnablePin(const char *, brain_pin_e, uint32_t, ExtiCallback, void *) { }
void efiExtiDisablePin(brain_pin_e) { }
#endif
#endif /* HAL_USE_PAL && EFI_PROD_CODE */