custom-board-bundle-sample-.../firmware/hw_layer/digital_input_hw.cpp

288 lines
7.7 KiB
C++
Raw Normal View History

2015-07-10 06:01:56 -07:00
/*
* @file digital_input_hw.cpp
* @brief Helper methods related to Input Capture Unit (ICU)
*
2018-12-09 11:47:24 -08:00
* There are some ChibiOS limitation or STM32 limitations or limitations of my brain
*
* See http://www.chibios.com/forum/viewtopic.php?t=1461
* "PWM input requires a whole timer on the STM32.
* You could use channel 1 and channel 2 of the same timer but not simultaneously.
* Giovanni"
*
* See http://www.chibios.com/forum/viewtopic.php?f=2&t=247&hilit=icu+channel&start=50
* "It is not possible, the TIM timers support one ICU channel at time.
* Giovanni"
*
* See https://stackoverflow.com/questions/43440599/stm32-multi-channel-input-capture-overcapturing-on-all-channels-interrupts-not
* where they seem to be capturing something on multiple channels maybe not PWM mode of ICU is the key difference?
*
2018-12-09 13:46:53 -08:00
* rus084 is reminding that EXTI could be enough for our needs
* See joystick.cpp
2018-12-12 20:36:36 -08:00
* See trigger_input.cpp
2018-12-09 11:47:24 -08:00
*
2015-07-10 06:01:56 -07:00
* @date Jun 23, 2013
2018-01-20 17:55:31 -08:00
* @author Andrey Belomutskiy, (c) 2012-2018
2015-07-10 06:01:56 -07:00
*/
#include "digital_input_hw.h"
#include "fl_stack.h"
2018-12-15 22:13:44 -08:00
#if EFI_ICU_INPUTS || defined(__DOXYGEN__)
2015-07-10 06:01:56 -07:00
2019-01-03 21:16:08 -08:00
#include "mpu_util.h"
2015-07-10 06:01:56 -07:00
#include "eficonsole.h"
#include "pin_repository.h"
static void icuWidthCallback(ICUDriver *driver);
static void icuPeriordCallBack(ICUDriver *driver);
/*
* 30ms seems like width maximum, at 16bit precision that means
* CORE_CLOCK / 33.33333 = TICKS * 65536
* 168000000 / 33.333333 / 65536 = 76.90
*/
static ICUConfig wave_icucfg = { ICU_INPUT_ACTIVE_LOW, CORE_CLOCK / 100, icuWidthCallback, icuPeriordCallBack, 0,
ICU_CHANNEL_1, 0 };
static ArrayList<digital_input_s, 8> registeredIcus;
//Nullable
2015-07-10 06:01:56 -07:00
static digital_input_s * finddigital_input_s(ICUDriver *driver) {
for (int i = 0; i < registeredIcus.size; i++) {
if (registeredIcus.elements[i].driver == driver) {
return &registeredIcus.elements[i];
}
}
firmwareError(CUSTOM_ERR_ICU, "reader not found");
2015-07-10 06:01:56 -07:00
return (digital_input_s *) NULL;
}
static void icuWidthCallback(ICUDriver *driver) {
/*
* see comment in icuPeriordCallBack
int rowWidth = icuGetWidth(driver);
*/
digital_input_s * hw = finddigital_input_s(driver);
hw->widthListeners.invokeJustArgCallbacks();
}
static void icuPeriordCallBack(ICUDriver *driver) {
/*
* we do not use timer period at all - we just need the event. For all time characteristics,
* we use system time
* int period = icuGetPeriod(driver);
*/
digital_input_s * hw = finddigital_input_s(driver);
hw->periodListeners.invokeJustArgCallbacks();
}
static uint32_t getAlternateFunctions(ICUDriver *driver) {
if (driver == NULL) {
firmwareError(CUSTOM_ERR_ICU_AF, "getAlternateFunctions(NULL)");
2015-07-10 06:01:56 -07:00
return 0xffffffff;
}
#if STM32_ICU_USE_TIM1
if (driver == &ICUD1) {
return GPIO_AF_TIM1;
}
#endif
#if STM32_ICU_USE_TIM2
if (driver == &ICUD2) {
return GPIO_AF_TIM2;
}
#endif
#if STM32_ICU_USE_TIM3
if (driver == &ICUD3) {
return GPIO_AF_TIM3;
}
#endif
#if STM32_ICU_USE_TIM4
if (driver == &ICUD4) {
return GPIO_AF_TIM4;
}
#endif
#if STM32_ICU_USE_TIM9
if (driver == &ICUD9) {
return GPIO_AF_TIM9;
}
#endif
firmwareError(CUSTOM_ERR_ICU_DRIVER, "No such driver");
2015-07-10 06:01:56 -07:00
return 0xffffffff;
}
icuchannel_t getInputCaptureChannel(brain_pin_e hwPin) {
switch (hwPin) {
case GPIOA_2: // TIM9
2017-08-27 18:33:24 -07:00
case GPIOA_5: // TIM2 stm32f4discovery/Frankenso default
case GPIOA_6: // TIM3
case GPIOA_8: // TIM1
case GPIOA_15: // TIM2
2017-08-27 18:33:24 -07:00
case GPIOC_6: // TIM3 or TIM8 stm32f4discovery/Frankenso default
case GPIOE_5: // TIM9
case GPIOE_9: // TIM1
2015-07-10 06:01:56 -07:00
return ICU_CHANNEL_1;
case GPIOA_1: // TIM2
case GPIOA_3: // TIM9
2017-08-27 18:33:24 -07:00
case GPIOA_7: // TIM3
case GPIOA_9: // TIM1
case GPIOB_3: // TIM2
case GPIOB_5: // TIM2
2017-08-27 18:33:24 -07:00
case GPIOC_7: // TIM3 or TIM8
case GPIOE_6: // TIM9
case GPIOE_11: // TIM1
2015-07-10 06:01:56 -07:00
return ICU_CHANNEL_2;
default:
firmwareError(CUSTOM_ERR_ICU_PIN, "Unexpected hw pin in getInputCaptureChannel %s", hwPortname(hwPin));
2015-07-10 06:01:56 -07:00
return ICU_CHANNEL_1;
}
}
2016-02-29 19:02:59 -08:00
/**
2016-03-01 07:01:30 -08:00
* as of Feb 2016, TIM1, TIM2, TIM3 and TIM9 are used for input capture
2016-02-29 19:02:59 -08:00
* (that's the kind of event you need for shaft position sensor)
* ChibiOS limitation is that only channels #1 and #2 could be used for input capture
*
* TODO: migrate slow ADC to software timer so that TIM8 is also available for input capture
2018-12-09 11:32:50 -08:00
* todo: https://github.com/rusefi/rusefi/issues/630 ?
* @return NULL if pin could not be used for ICU
2016-02-29 19:02:59 -08:00
*/
//Nullable
2017-02-06 16:03:19 -08:00
ICUDriver * getInputCaptureDriver(const char *msg, brain_pin_e hwPin) {
2017-02-06 17:02:18 -08:00
if (hwPin == GPIO_UNASSIGNED || hwPin == GPIO_INVALID) {
return NULL;
}
2015-07-10 06:01:56 -07:00
#if STM32_ICU_USE_TIM1
2015-10-12 16:01:27 -07:00
if (hwPin == GPIOA_8 ||
hwPin == GPIOA_9 ||
2015-10-12 16:01:27 -07:00
hwPin == GPIOE_9 ||
hwPin == GPIOE_11) {
2015-07-10 06:01:56 -07:00
return &ICUD1;
}
#endif
#if STM32_ICU_USE_TIM2
2016-02-29 19:02:59 -08:00
if (hwPin == GPIOA_1 ||
hwPin == GPIOA_5 ||
hwPin == GPIOA_15 ||
2016-02-29 19:02:59 -08:00
hwPin == GPIOB_3) {
2015-07-10 06:01:56 -07:00
return &ICUD2;
}
#endif
#if STM32_ICU_USE_TIM3
2017-02-06 16:03:19 -08:00
if (hwPin == GPIOA_6 ||
hwPin == GPIOA_7 ||
hwPin == GPIOB_4 ||
hwPin == GPIOB_5 ||
2016-02-29 19:02:59 -08:00
hwPin == GPIOC_6 ||
hwPin == GPIOC_7) {
2015-07-10 06:01:56 -07:00
return &ICUD3;
}
#endif
2016-02-29 19:02:59 -08:00
#if STM32_ICU_USE_TIM8
if (hwPin == GPIOC_6 ||
hwPin == GPIOC_7) {
return &ICUD8;
2016-02-29 19:02:59 -08:00
}
#endif
2015-07-10 06:01:56 -07:00
#if STM32_ICU_USE_TIM9
2015-10-12 16:01:27 -07:00
if (hwPin == GPIOA_2 ||
hwPin == GPIOA_3 ||
2017-02-06 17:02:18 -08:00
hwPin == GPIOE_5 ||
hwPin == GPIOE_6) {
2015-07-10 06:01:56 -07:00
return &ICUD9;
}
#endif
2017-02-06 16:03:19 -08:00
firmwareError(CUSTOM_ERR_NOT_INPUT_PIN, "%s: Not input pin %s", msg, hwPortname(hwPin));
2015-07-10 06:01:56 -07:00
return (ICUDriver *) NULL;
}
void turnOnCapturePin(const char *msg, brain_pin_e brainPin) {
2017-02-06 16:03:19 -08:00
ICUDriver *driver = getInputCaptureDriver(msg, brainPin);
2015-07-10 06:01:56 -07:00
if (driver != NULL) {
iomode_t mode = (iomode_t) PAL_MODE_ALTERNATE(getAlternateFunctions(driver));
2017-05-15 05:40:54 -07:00
efiSetPadMode(msg, brainPin, mode);
2015-07-10 06:01:56 -07:00
}
}
2018-01-01 08:27:15 -08:00
/**
* takes next digital_input_s from the registeredIcus pool
*/
digital_input_s * addWaveAnalyzerDriver(const char *msg, brain_pin_e brainPin) {
2017-02-06 16:03:19 -08:00
ICUDriver *driver = getInputCaptureDriver(msg, brainPin);
if (driver == NULL) {
2019-02-06 15:41:48 -08:00
warning(CUSTOM_ERR_INVALID_INPUT_ICU_PIN, "w_not input pin");
return NULL;
}
2015-07-10 06:01:56 -07:00
digital_input_s *hw = registeredIcus.add();
2017-04-05 16:48:40 -07:00
hw->widthListeners.clear();
hw->periodListeners.clear();
2017-05-05 18:05:59 -07:00
hw->started = false;
2017-02-06 17:02:18 -08:00
hw->brainPin = brainPin;
2015-07-10 06:01:56 -07:00
hw->driver = driver;
turnOnCapturePin(msg, brainPin);
return hw;
}
2018-01-01 08:27:15 -08:00
/**
* turns pin off and returns digital_input_s back into registeredIcus pool
*/
void removeWaveAnalyzerDriver(const char *msg, brain_pin_e brainPin) {
2017-05-05 18:05:59 -07:00
if (brainPin == GPIO_UNASSIGNED) {
return;
}
unmarkPin(brainPin);
ICUDriver *driver = getInputCaptureDriver(msg, brainPin);
if (driver == NULL) {
return;
}
int regSize = registeredIcus.size;
for (int i = 0; i < regSize; i++) {
if (registeredIcus.elements[i].driver == driver) {
// removing from driver from the list of used drivers
memcpy(&registeredIcus.elements[i], &registeredIcus.elements[regSize - 1],
sizeof(digital_input_s));
registeredIcus.size--;
2017-05-05 18:50:22 -07:00
icuDisableNotificationsI(driver);
2017-05-07 16:35:04 -07:00
icuStopCapture(driver);
2017-05-05 18:50:22 -07:00
icuStop(driver);
2017-05-05 18:05:59 -07:00
return;
}
}
}
2019-04-07 15:25:46 -07:00
void startInputDriver(const char *msg, /*nullable*/digital_input_s *hw, bool isActiveHigh) {
if (hw == NULL) {
// we can get NULL driver if user somehow has invalid pin in his configuration
2019-02-06 15:41:48 -08:00
warning(CUSTOM_ERR_INVALID_INPUT_ICU_PIN, "s_not input pin");
return;
}
2015-07-10 06:01:56 -07:00
hw->isActiveHigh = isActiveHigh;
if (hw->isActiveHigh) {
wave_icucfg.mode = ICU_INPUT_ACTIVE_HIGH;
} else {
wave_icucfg.mode = ICU_INPUT_ACTIVE_LOW;
}
ICUDriver *driver = hw->driver;
if (driver != NULL) {
if (hw->started) {
2017-03-21 11:58:14 -07:00
icuDisableNotificationsI(driver);
2017-05-07 16:35:04 -07:00
icuStopCapture(driver);
2015-07-10 06:01:56 -07:00
icuStop(driver);
}
2017-02-06 17:02:18 -08:00
wave_icucfg.channel = getInputCaptureChannel(hw->brainPin);
2019-04-07 15:25:46 -07:00
efiIcuStart(msg, driver, &wave_icucfg);
2018-07-25 20:03:04 -07:00
efiAssertVoid(CUSTOM_ERR_6672, driver != NULL, "di: driver is NULL");
efiAssertVoid(CUSTOM_ERR_6673, driver->state == ICU_READY, "di: driver not ready");
icuStartCapture(driver); // this would change state from READY to WAITING
2017-03-21 11:58:14 -07:00
icuEnableNotifications(driver);
2015-07-10 06:01:56 -07:00
}
hw->started = true;
}
2018-12-15 22:13:44 -08:00
#endif /* EFI_ICU_INPUTS */