2022-09-07 16:36:16 -07:00
|
|
|
/*
|
|
|
|
* @file efi_output.h
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2023-08-22 11:37:17 -07:00
|
|
|
#pragma once
|
|
|
|
|
2022-09-07 16:36:16 -07:00
|
|
|
#include "io_pins.h"
|
|
|
|
#include "smart_gpio.h"
|
2023-10-10 18:36:33 -07:00
|
|
|
#include <rusefi/timer.h>
|
2022-09-07 16:36:16 -07:00
|
|
|
|
2023-08-22 11:37:17 -07:00
|
|
|
// This class acts as a boolean, but has a switch counter inside
|
|
|
|
class SwitchedState {
|
|
|
|
public:
|
2023-11-01 07:09:04 -07:00
|
|
|
SwitchedState(int8_t *p_state) {
|
|
|
|
state = p_state;
|
2023-08-20 14:08:33 -07:00
|
|
|
}
|
|
|
|
|
2023-08-22 11:37:17 -07:00
|
|
|
// returns true if the state has been changed
|
|
|
|
bool update(bool newState);
|
|
|
|
uint16_t getCounter();
|
|
|
|
|
|
|
|
operator bool() const {
|
|
|
|
return (bool)*state;
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
int8_t *state;
|
|
|
|
uint16_t counter = 0;
|
2023-08-20 14:08:33 -07:00
|
|
|
};
|
|
|
|
|
2024-01-06 14:11:25 -08:00
|
|
|
class SimpleSwitchedState {
|
|
|
|
public:
|
|
|
|
SimpleSwitchedState() : state(&value) {
|
|
|
|
}
|
|
|
|
int8_t value = 0;
|
|
|
|
SwitchedState state;
|
|
|
|
};
|
|
|
|
|
2022-09-07 16:36:16 -07:00
|
|
|
// Used if you want a function to be virtual only for unit testing purposes
|
|
|
|
#if EFI_UNIT_TEST
|
|
|
|
#define TEST_VIRTUAL virtual
|
|
|
|
#else
|
|
|
|
#define TEST_VIRTUAL
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @brief Single output pin reference and state
|
|
|
|
*/
|
|
|
|
class OutputPin {
|
|
|
|
public:
|
2023-02-25 00:31:12 -08:00
|
|
|
// initializes pin & registers it in pin repository
|
|
|
|
void initPin(const char *msg, brain_pin_e brainPin, pin_output_mode_e outputMode, bool forceInitWithFatalError = false);
|
|
|
|
|
|
|
|
// same as above, with OM_DEFAULT mode
|
2022-09-07 16:36:16 -07:00
|
|
|
void initPin(const char *msg, brain_pin_e brainPin);
|
|
|
|
|
2023-02-25 00:31:12 -08:00
|
|
|
// dissociates pin from this output and un-registers it in pin repository
|
2022-09-07 16:36:16 -07:00
|
|
|
void deInit();
|
|
|
|
|
2023-11-01 14:54:57 -07:00
|
|
|
bool isInitialized() const;
|
2022-09-07 16:36:16 -07:00
|
|
|
|
|
|
|
bool getAndSet(int logicValue);
|
2023-08-28 21:24:46 -07:00
|
|
|
void setValue(const char *msg, int logicValue, bool isForce = false);
|
|
|
|
TEST_VIRTUAL void setValue(int logicValue, bool isForce = false);
|
2022-09-07 16:36:16 -07:00
|
|
|
void toggle();
|
|
|
|
bool getLogicValue() const;
|
|
|
|
|
|
|
|
brain_pin_diag_e getDiag() const;
|
|
|
|
|
|
|
|
#if EFI_GPIO_HARDWARE
|
2024-03-01 16:12:19 -08:00
|
|
|
ioportid_t m_port = 0;
|
|
|
|
uint8_t m_pin = 0;
|
2022-09-07 16:36:16 -07:00
|
|
|
#endif /* EFI_GPIO_HARDWARE */
|
|
|
|
|
2023-09-10 21:03:10 -07:00
|
|
|
#if EFI_UNIT_TEST || EFI_SIMULATOR
|
|
|
|
int pinToggleCounter = 0;
|
2023-10-10 18:36:33 -07:00
|
|
|
#endif
|
2023-10-10 18:29:27 -07:00
|
|
|
|
2023-10-10 18:36:33 -07:00
|
|
|
#if EFI_SIMULATOR
|
|
|
|
Timer pinToggleTimer;
|
|
|
|
uint32_t durationsInStateMs[2];
|
2023-10-16 06:37:22 -07:00
|
|
|
|
|
|
|
void resetToggleStats();
|
2022-09-07 16:36:16 -07:00
|
|
|
#endif
|
|
|
|
|
|
|
|
brain_pin_e brainPin = Gpio::Unassigned;
|
|
|
|
|
|
|
|
#if (EFI_GPIO_HARDWARE && (BOARD_EXT_GPIOCHIPS > 0))
|
|
|
|
/* used for external pins */
|
|
|
|
bool ext = false;
|
|
|
|
#endif /* EFI_GPIO_HARDWARE */
|
|
|
|
|
|
|
|
int8_t currentLogicValue = INITIAL_PIN_STATE;
|
|
|
|
/**
|
|
|
|
* we track current pin status so that we do not touch the actual hardware if we want to write new pin bit
|
|
|
|
* which is same as current pin value. This maybe helps in case of status leds, but maybe it's a total over-engineering
|
|
|
|
*/
|
|
|
|
private:
|
|
|
|
// todo: inline this method?
|
2023-02-25 00:31:12 -08:00
|
|
|
void setDefaultPinState(pin_output_mode_e mode);
|
2022-09-07 16:36:16 -07:00
|
|
|
void setOnchipValue(int electricalValue);
|
|
|
|
|
|
|
|
// 4 byte pointer is a bit of a memory waste here
|
2023-02-25 00:31:12 -08:00
|
|
|
pin_output_mode_e mode = OM_DEFAULT;
|
2022-09-07 16:36:16 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* OutputPin which is reported on Engine Sniffer
|
|
|
|
*/
|
|
|
|
class NamedOutputPin : public virtual OutputPin {
|
|
|
|
public:
|
|
|
|
NamedOutputPin();
|
|
|
|
explicit NamedOutputPin(const char *name);
|
2023-06-15 08:46:28 -07:00
|
|
|
virtual void setHigh(const char *msg);
|
|
|
|
virtual void setLow(const char *msg);
|
2022-09-07 16:36:16 -07:00
|
|
|
virtual void setHigh();
|
|
|
|
virtual void setLow();
|
|
|
|
const char *getName() const;
|
2023-11-01 14:54:57 -07:00
|
|
|
void setName(const char*);
|
2022-09-07 16:36:16 -07:00
|
|
|
const char *getShortName() const;
|
|
|
|
/**
|
|
|
|
* @return true if pin was stopped
|
|
|
|
*/
|
|
|
|
bool stop();
|
|
|
|
/**
|
|
|
|
* rusEfi Engine Sniffer protocol uses these short names to reduce bytes usage
|
|
|
|
*/
|
|
|
|
const char *shortName = nullptr;
|
2023-11-01 14:54:57 -07:00
|
|
|
|
|
|
|
private:
|
|
|
|
// todo: char pointer is a bit of a memory waste here, we can reduce RAM usage by software-based getName() method
|
|
|
|
const char *name = nullptr;
|
2022-09-07 16:36:16 -07:00
|
|
|
};
|