2020-09-02 12:39:50 -07:00
|
|
|
/**
|
|
|
|
* @file debounce.cpp
|
|
|
|
* @brief Generic button debounce class
|
|
|
|
*
|
|
|
|
* @date Aug 31, 2020
|
|
|
|
* @author David Holdeman, (c) 2020
|
|
|
|
*/
|
|
|
|
#include "debounce.h"
|
2020-09-08 08:29:38 -07:00
|
|
|
#include "pin_repository.h"
|
|
|
|
#include "engine_configuration.h"
|
|
|
|
#include "hardware.h"
|
2020-09-02 12:39:50 -07:00
|
|
|
|
2020-09-08 08:29:38 -07:00
|
|
|
ButtonDebounce* ButtonDebounce::s_firstDebounce = nullptr;
|
|
|
|
|
|
|
|
void ButtonDebounce::init (int t, brain_pin_e *pin, pin_input_mode_e *mode) {
|
|
|
|
if (!initialized) {
|
|
|
|
ButtonDebounce *listItem = s_firstDebounce;
|
|
|
|
if (listItem == nullptr) {
|
|
|
|
s_firstDebounce = this;
|
|
|
|
} else {
|
|
|
|
while (listItem->nextDebounce != nullptr) {
|
|
|
|
listItem = listItem->nextDebounce;
|
|
|
|
}
|
|
|
|
listItem->nextDebounce = this;
|
|
|
|
}
|
|
|
|
}
|
2020-09-02 12:39:50 -07:00
|
|
|
threshold = MS2NT(t);
|
|
|
|
timeLast = 0;
|
2020-09-08 08:29:38 -07:00
|
|
|
this->pin = pin;
|
|
|
|
active_pin = *pin;
|
|
|
|
this->mode = mode;
|
|
|
|
active_mode = *mode;
|
2020-09-08 11:27:23 -07:00
|
|
|
#ifndef EFI_UNIT_TEST
|
2020-09-02 12:39:50 -07:00
|
|
|
// getInputMode converts from pin_input_mode_e to iomode_t
|
2020-09-08 08:29:38 -07:00
|
|
|
efiSetPadMode("Button", active_pin, getInputMode(active_mode));
|
2020-09-02 12:39:50 -07:00
|
|
|
#endif
|
2020-09-08 08:29:38 -07:00
|
|
|
initialized = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ButtonDebounce::updateConfigurationList () {
|
|
|
|
ButtonDebounce *listItem = s_firstDebounce;
|
|
|
|
while (listItem != nullptr) {
|
|
|
|
listItem->updateConfiguration();
|
|
|
|
if (listItem->nextDebounce != nullptr) {
|
|
|
|
listItem = listItem->nextDebounce;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ButtonDebounce::updateConfiguration () {
|
|
|
|
#ifndef EFI_ACTIVE_CONFIGURATION_IN_FLASH
|
|
|
|
if (*pin != active_pin || *mode != active_mode) {
|
|
|
|
#else
|
|
|
|
if (*pin != active_pin || *mode != active_mode || (isActiveConfigurationVoid && (*pin != 0 || *mode != 0))) {
|
|
|
|
#endif /* EFI_ACTIVE_CONFIGURATION_IN_FLASH */
|
2020-09-08 11:27:23 -07:00
|
|
|
#ifndef EFI_UNIT_TEST
|
2020-09-08 08:29:38 -07:00
|
|
|
brain_pin_markUnused(active_pin);
|
|
|
|
efiSetPadMode("Button", *pin, getInputMode(*mode));
|
2020-09-08 11:27:23 -07:00
|
|
|
#endif /* EFI_UNIT_TEST */
|
2020-09-08 08:29:38 -07:00
|
|
|
}
|
|
|
|
active_pin = *pin;
|
|
|
|
active_mode = *mode;
|
2020-09-02 12:39:50 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
@returns true if the button is pressed, and will not return true again within the set timeout
|
|
|
|
*/
|
|
|
|
bool ButtonDebounce::readPinEvent() {
|
2020-09-09 14:18:54 -07:00
|
|
|
readValue = false;
|
|
|
|
return readPinState();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ButtonDebounce::readPinState() {
|
2020-09-02 12:39:50 -07:00
|
|
|
if (!pin) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
efitick_t timeNow = getTimeNowNt();
|
|
|
|
// If it's been less than the threshold since we were last called
|
|
|
|
if ((timeNow - timeLast) < threshold) {
|
2020-09-09 14:18:54 -07:00
|
|
|
return readValue;
|
2020-09-02 12:39:50 -07:00
|
|
|
}
|
|
|
|
// readValue is a class variable, so it needs to be reset.
|
|
|
|
// We don't actually need it to be a class variable in this method,
|
|
|
|
// but when a method is implemented to actually get the pin's state,
|
|
|
|
// for example to implement long button presses, it will be needed.
|
|
|
|
readValue = false;
|
2020-09-10 19:16:20 -07:00
|
|
|
#if EFI_PROD_CODE || EFI_UNIT_TEST
|
2020-09-08 08:29:38 -07:00
|
|
|
readValue = efiReadPin(active_pin);
|
2020-09-10 19:16:20 -07:00
|
|
|
#endif
|
|
|
|
#if EFI_PROD_CODE
|
2020-09-02 12:39:50 -07:00
|
|
|
// Invert
|
2020-09-08 08:29:38 -07:00
|
|
|
if (getInputMode(active_mode) == PAL_MODE_INPUT_PULLUP) {
|
2020-09-02 12:39:50 -07:00
|
|
|
readValue = !readValue;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
if (readValue) {
|
|
|
|
timeLast = timeNow;
|
|
|
|
}
|
|
|
|
return readValue;
|
|
|
|
}
|