/** * @file electronic_throttle_impl.h * * @date Dec 7, 2013 * @author Andrey Belomutskiy, (c) 2012-2020 */ #pragma once // include the "public" ETB interface #include "electronic_throttle.h" #include "sensor.h" #include "efi_pid.h" #include "error_accumulator.h" #include "electronic_throttle_generated.h" #include "exp_average.h" /** * Hard code ETB update speed. * Since this is a safety critical system with no real reason for a user to ever need to change the update rate, * it's locked to 500hz, along with the ADC. * https://en.wikipedia.org/wiki/Nyquist%E2%80%93Shannon_sampling_theorem */ #define ETB_LOOP_FREQUENCY 500 #define DEFAULT_ETB_PWM_FREQUENCY 800 class EtbController : public IEtbController, public electronic_throttle_s { public: bool init(dc_function_e function, DcMotor *motor, pid_s *pidParameters, const ValueProvider3D* pedalMap, bool initializeThrottles) override; void setIdlePosition(percent_t pos) override; void setWastegatePosition(percent_t pos) override; void reset() override; // Update the controller's state: read sensors, send output, etc void update() override; // Called when the configuration may have changed. Controller will // reset if necessary. void onConfigurationChange(pid_s* previousConfiguration); // Print this throttle's status. void showStatus(); // Helpers for individual parts of throttle control expected observePlant() override; expected getSetpoint() override; expected getSetpointEtb(); expected getSetpointWastegate() const; expected getSetpointIdleValve() const; expected getOpenLoop(percent_t target) override; expected getClosedLoop(percent_t setpoint, percent_t observation) override; expected getClosedLoopAutotune(percent_t setpoint, percent_t actualThrottlePosition); void setOutput(expected outputValue) override; void checkOutput(percent_t output); // Used to inspect the internal PID controller's state const pid_state_s& getPidState() const override { return m_pid; }; // Use the throttle to automatically calibrate the relevant throttle position sensor(s). void autoCalibrateTps() override; // Override if this throttle needs special per-throttle adjustment (bank-to-bank trim, for example) virtual percent_t getThrottleTrim(float /*rpm*/, percent_t /*targetPosition*/) const { return 0; } float getCurrentTarget() const override { return etbCurrentTarget; } // Lua throttle adjustment void setLuaAdjustment(percent_t adjustment) override; float getLuaAdjustment() const; float prevOutput = 0; protected: // This is set if an automatic TPS calibration should be run bool m_isAutocal = false; bool hadTpsError = false; bool hadPpsError = false; dc_function_e getFunction() const { return m_function; } DcMotor* getMotor() { return m_motor; } private: dc_function_e m_function = DC_None; SensorType m_positionSensor = SensorType::Invalid; DcMotor *m_motor = nullptr; Pid m_pid; bool m_shouldResetPid = false; // todo: rename to m_targetErrorAccumulator ErrorAccumulator m_errorAccumulator; /** * @return true if OK, false if should be disabled */ bool checkStatus(); bool isEtbMode() const override { return m_function == DC_Throttle1 || m_function == DC_Throttle2; } ExpAverage m_dutyRocAverage; ExpAverage m_dutyAverage; Timer m_jamDetectTimer; // Pedal -> target map const ValueProvider3D* m_pedalProvider = nullptr; float m_idlePosition = 0; // This is set if automatic PID cal should be run bool m_isAutotune = false; // Autotune helpers bool m_lastIsPositive = false; Timer m_autotuneCycleStart; float m_minCycleTps = 0; float m_maxCycleTps = 0; // Autotune measured parameters: gain and ultimate period // These are set to correct order of magnitude starting points // so we converge more quickly on the correct values float m_a = 8; float m_tu = 0.1f; uint8_t m_autotuneCounter = 0; uint8_t m_autotuneCurrentParam = 0; Timer m_luaAdjustmentTimer; }; void etbPidReset(); class EtbController1 : public EtbController { }; class EtbController2 : public EtbController { public: EtbController2(const ValueProvider3D& throttle2TrimTable) : m_throttle2Trim(throttle2TrimTable) { } percent_t getThrottleTrim(float rpm, percent_t /*targetPosition*/) const override; private: const ValueProvider3D& m_throttle2Trim; };