/** * @file pid.h * * @date Sep 16, 2014 * @author Andrey Belomutskiy, (c) 2012-2020 */ #pragma once #include "engine_state_generated.h" #include "pid_state_generated.h" #if EFI_PROD_CODE || EFI_SIMULATOR #include "tunerstudio_configuration.h" #endif // See PidCic below #define PID_AVG_BUF_SIZE_SHIFT 5 #define PID_AVG_BUF_SIZE (1<periodMs)) #define MS2SEC(x) (x * 0.001) struct pid_s; class Logging; class Pid : public pid_state_s { public: Pid(); explicit Pid(pid_s *parameters); void initPidClass(pid_s *parameters); bool isSame(const pid_s *parameters) const; /** * This version of the method takes dTime from pid_s * * @param Controller input / process output * @returns Output from the PID controller / the input to the process */ float getOutput(float target, float input); virtual float getOutput(float target, float input, float dTime); // doesn't limit the result (used in incremental CIC PID, see below) float getUnclampedOutput(float target, float input, float dTime); void updateFactors(float pFactor, float iFactor, float dFactor); virtual void reset(void); float getP(void) const; float getI(void) const; float getD(void) const; virtual float getOffset(void) const; virtual float getMinValue(void) const; float getIntegration(void) const; float getPrevError(void) const; void setErrorAmplification(float coef); #if EFI_TUNER_STUDIO void postState(TunerStudioOutputChannels *tsOutputChannels) const; void postState(TunerStudioOutputChannels *tsOutputChannels, int pMult) const; #endif /* EFI_TUNER_STUDIO */ void showPidStatus(Logging *logging, const char*msg) const; void sleep(); int resetCounter; // todo: move this to pid_s one day float iTermMin = -1000000.0; float iTermMax = 1000000.0; protected: pid_s *parameters; private: virtual void updateITerm(float value); }; /** * A PID implementation with a modified cascaded integrator-comb (CIC) filtering. * Used for incremental auto-IAC control. See autoIdle() in idle_thread.cpp * See pid_cic.md. * * https://rusefi.com/forum/viewtopic.php?f=9&t=1315 */ class PidCic : public Pid { public: PidCic(); explicit PidCic(pid_s *pid); void reset(void) override; using Pid::getOutput; float getOutput(float target, float input, float dTime) override; private: // Circular running-average buffer for I-term, used by CIC-like filter float iTermBuf[PID_AVG_BUF_SIZE]; // Needed by averaging (smoothing) of iTerm sums float iTermInvNum; // Total PID iterations (>240 days max. for 10ms update period) int totalItermCnt; private: void updateITerm(float value) override; }; /** * A PID with derivative filtering (backward differences) and integrator anti-windup. * See: Wittenmark B., Astrom K., Arzen K. IFAC Professional Brief. Computer Control: An Overview. * Two additional parameters used: derivativeFilterLoss and antiwindupFreq * (If both are 0, then this controller is identical to PidParallelController) */ class PidIndustrial : public Pid { public: PidIndustrial(); explicit PidIndustrial(pid_s *pid); using Pid::getOutput; float getOutput(float target, float input, float dTime) override; public: // todo: move this to pid_s one day float antiwindupFreq = 0.0f; // = 1/ResetTime float derivativeFilterLoss = 0.0f; // = 1/Gain private: float limitOutput(float v) const; };