gyverpid customized
This commit is contained in:
parent
4c650ecc48
commit
ef791654b4
@ -68,8 +68,9 @@ Profile profiles[] = {
|
|||||||
{"Gradual", 10, {{46, 120}, {90, 30}}, 2},
|
{"Gradual", 10, {{46, 120}, {90, 30}}, 2},
|
||||||
{"Long gradual", 15, {{46, 120}, {80, 60}, {90, 15}}, 3},
|
{"Long gradual", 15, {{46, 120}, {80, 60}, {90, 15}}, 3},
|
||||||
{"Wheat", 1, {{46, 120}, {53, 40}, {65, 40}, {72, 40}, {85, 40}}, 5},
|
{"Wheat", 1, {{46, 120}, {53, 40}, {65, 40}, {72, 40}, {85, 40}}, 5},
|
||||||
{"Amazake", 1, {{51, 480}}, 1},
|
// {"Amazake", 1, {{51, 480}}, 1},
|
||||||
{"SV", 0, {{46, 45}}, 1},
|
{"37, 3 days", 1, {{37, 1440}, {37, 1440}, {37, 1440}}, 3},
|
||||||
|
// {"SV", 0, {{46, 45}}, 1},
|
||||||
// {"Yoghurt maker", 0, {{40, 300}, {40, 300}, {30, 300}}, 3},
|
// {"Yoghurt maker", 0, {{40, 300}, {40, 300}, {30, 300}}, 3},
|
||||||
// {"TEST", 0, {{46, 1}}, 1},
|
// {"TEST", 0, {{46, 1}}, 1},
|
||||||
};
|
};
|
||||||
@ -88,9 +89,10 @@ bool inSelectionMode = true;
|
|||||||
// double Kp = 0.5, Ki = 1, Kd = 0.05;
|
// double Kp = 0.5, Ki = 1, Kd = 0.05;
|
||||||
// PID myPID(&Input, &Output, &Setpoint, Kp, Ki, Kd, DIRECT);
|
// PID myPID(&Input, &Output, &Setpoint, Kp, Ki, Kd, DIRECT);
|
||||||
|
|
||||||
#include "GyverPID.h"
|
// #include "GyverPID.h"
|
||||||
|
#include "pid.h"
|
||||||
double Setpoint, Input, Output;
|
double Setpoint, Input, Output;
|
||||||
GyverPID regulator(50, 1, 5, 1000);
|
GyverPID regulator(50, 1, 20, 1000);
|
||||||
|
|
||||||
bool temperatureSensorError = false;
|
bool temperatureSensorError = false;
|
||||||
|
|
||||||
|
|||||||
162
pid.h
Normal file
162
pid.h
Normal file
@ -0,0 +1,162 @@
|
|||||||
|
/*
|
||||||
|
GyverPID - библиотека PID регулятора для Arduino
|
||||||
|
Документация: https://alexgyver.ru/gyverpid/
|
||||||
|
GitHub: https://github.com/GyverLibs/GyverPID
|
||||||
|
Возможности:
|
||||||
|
- Время одного расчёта около 70 мкс
|
||||||
|
- Режим работы по величине или по её изменению (для интегрирующих процессов)
|
||||||
|
- Возвращает результат по встроенному таймеру или в ручном режиме
|
||||||
|
- Встроенные калибровщики коэффициентов
|
||||||
|
- Режим работы по ошибке и по ошибке измерения
|
||||||
|
- Встроенные оптимизаторы интегральной суммы
|
||||||
|
|
||||||
|
AlexGyver, alex@alexgyver.ru
|
||||||
|
https://alexgyver.ru/
|
||||||
|
MIT License
|
||||||
|
|
||||||
|
Версии:
|
||||||
|
v1.1 - убраны дефайны
|
||||||
|
v1.2 - возвращены дефайны
|
||||||
|
v1.3 - вычисления ускорены, библиотека облегчена
|
||||||
|
v2.0 - логика работы чуть переосмыслена, код улучшен, упрощён и облегчён
|
||||||
|
v2.1 - integral вынесен в public
|
||||||
|
v2.2 - оптимизация вычислений
|
||||||
|
v2.3 - добавлен режим PID_INTEGRAL_WINDOW
|
||||||
|
v2.4 - реализация внесена в класс
|
||||||
|
v3.0
|
||||||
|
- Добавлен режим оптимизации интегральной составляющей (см. доку)
|
||||||
|
- Добавлены автоматические калибровщики коэффициентов (см. примеры и доку)
|
||||||
|
v3.1 - исправлен режиме ON_RATE, добавлено автоограничение инт. суммы
|
||||||
|
v3.2 - чуть оптимизации, добавлена getResultNow
|
||||||
|
v3.3 - в тюнерах можно передать другой обработчик класса Stream для отладки
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef _GyverPID_h
|
||||||
|
#define _GyverPID_h
|
||||||
|
#include <Arduino.h>
|
||||||
|
|
||||||
|
#if defined(PID_INTEGER) // расчёты с целыми числами
|
||||||
|
typedef int16_t pidtype;
|
||||||
|
#else // расчёты с float числами
|
||||||
|
typedef float pidtype;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#define NORMAL 0
|
||||||
|
#define REVERSE 1
|
||||||
|
#define ON_ERROR 0
|
||||||
|
#define ON_RATE 1
|
||||||
|
|
||||||
|
class GyverPID {
|
||||||
|
public:
|
||||||
|
float lastP = 0;
|
||||||
|
float lastI = 0;
|
||||||
|
float lastD = 0;
|
||||||
|
|
||||||
|
// ==== pidtype это float или int, в зависимости от выбранного (см. пример integer_calc) ====
|
||||||
|
GyverPID() {}
|
||||||
|
|
||||||
|
// kp, ki, kd, dt
|
||||||
|
GyverPID(float new_kp, float new_ki, float new_kd, uint32_t new_dt = 100) {
|
||||||
|
setDt(new_dt);
|
||||||
|
Kp = new_kp;
|
||||||
|
Ki = new_ki;
|
||||||
|
Kd = new_kd;
|
||||||
|
}
|
||||||
|
|
||||||
|
// направление регулирования: NORMAL (0) или REVERSE (1)
|
||||||
|
void setDirection(bool direction) {
|
||||||
|
_direction = direction;
|
||||||
|
}
|
||||||
|
|
||||||
|
// режим: работа по входной ошибке ON_ERROR (0) или по изменению ON_RATE (1)
|
||||||
|
void setMode(bool mode) {
|
||||||
|
_mode = mode;
|
||||||
|
}
|
||||||
|
|
||||||
|
// лимит выходной величины (например для ШИМ ставим 0-255)
|
||||||
|
void setLimits(int16_t min_output, int16_t max_output) {
|
||||||
|
_minOut = min_output;
|
||||||
|
_maxOut = max_output;
|
||||||
|
}
|
||||||
|
|
||||||
|
// установка времени дискретизации (для getResultTimer)
|
||||||
|
void setDt(uint32_t new_dt) {
|
||||||
|
_dt_s = new_dt / 1000.0f;
|
||||||
|
_dt = new_dt;
|
||||||
|
}
|
||||||
|
|
||||||
|
pidtype setpoint = 0; // заданная величина, которую должен поддерживать регулятор
|
||||||
|
pidtype input = 0; // сигнал с датчика (например температура, которую мы регулируем)
|
||||||
|
pidtype output = 0; // выход с регулятора на управляющее устройство (например величина ШИМ или угол поворота серво)
|
||||||
|
float Kp = 0.0; // коэффициент P
|
||||||
|
float Ki = 0.0; // коэффициент I
|
||||||
|
float Kd = 0.0; // коэффициент D
|
||||||
|
float integral = 0.0; // интегральная сумма
|
||||||
|
|
||||||
|
// возвращает новое значение при вызове (если используем свой таймер с периодом dt!)
|
||||||
|
pidtype getResult() {
|
||||||
|
pidtype error = setpoint - input; // ошибка регулирования
|
||||||
|
pidtype delta_input = prevInput - input; // изменение входного сигнала за dt
|
||||||
|
prevInput = input; // запомнили предыдущее
|
||||||
|
if (_direction) { // смена направления
|
||||||
|
error = -error;
|
||||||
|
delta_input = -delta_input;
|
||||||
|
}
|
||||||
|
lastP = _mode ? 0 : (error * Kp); // пропорциональая составляющая
|
||||||
|
output = lastP;
|
||||||
|
lastD = delta_input * Kd / _dt_s; // дифференциальная составляющая
|
||||||
|
output += lastD;
|
||||||
|
|
||||||
|
#if (PID_INTEGRAL_WINDOW > 0)
|
||||||
|
// ЭКСПЕРИМЕНТАЛЬНЫЙ РЕЖИМ ИНТЕГРАЛЬНОГО ОКНА
|
||||||
|
if (++t >= PID_INTEGRAL_WINDOW) t = 0; // перемотка t
|
||||||
|
integral -= errors[t]; // вычитаем старое
|
||||||
|
errors[t] = error * Ki * _dt_s; // запоминаем в массив
|
||||||
|
integral += errors[t]; // прибавляем новое
|
||||||
|
#else
|
||||||
|
integral += error * Ki * _dt_s; // обычное суммирование инт. суммы
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifdef PID_OPTIMIZED_I
|
||||||
|
// ЭКСПЕРИМЕНТАЛЬНЫЙ РЕЖИМ ОГРАНИЧЕНИЯ ИНТЕГРАЛЬНОЙ СУММЫ
|
||||||
|
output = constrain(output, _minOut, _maxOut);
|
||||||
|
if (Ki != 0) integral = constrain(integral, (_minOut - output) / (Ki * _dt_s), (_maxOut - output) / (Ki * _dt_s));
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if (_mode) integral += delta_input * Kp; // режим пропорционально скорости
|
||||||
|
integral = constrain(integral, _minOut, _maxOut); // ограничиваем инт. сумму
|
||||||
|
lastI = integral;
|
||||||
|
output += integral; // интегральная составляющая
|
||||||
|
output = constrain(output, _minOut, _maxOut); // ограничиваем выход
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
// возвращает новое значение не ранее, чем через dt миллисекунд (встроенный таймер с периодом dt)
|
||||||
|
pidtype getResultTimer() {
|
||||||
|
if (millis() - pidTimer >= _dt) {
|
||||||
|
pidTimer = millis();
|
||||||
|
getResult();
|
||||||
|
}
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
// посчитает выход по реальному прошедшему времени между вызовами функции
|
||||||
|
pidtype getResultNow() {
|
||||||
|
setDt(millis() - pidTimer);
|
||||||
|
pidTimer = millis();
|
||||||
|
return getResult();
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
uint32_t _dt = 100; // время итерации в мс
|
||||||
|
float _dt_s = 0.1; // время итерации в с
|
||||||
|
bool _mode = 0, _direction = 0;
|
||||||
|
int16_t _minOut = 0, _maxOut = 255;
|
||||||
|
pidtype prevInput = 0;
|
||||||
|
uint32_t pidTimer = 0;
|
||||||
|
#if (PID_INTEGRAL_WINDOW > 0)
|
||||||
|
pidtype errors[PID_INTEGRAL_WINDOW];
|
||||||
|
uint16_t t = 0;
|
||||||
|
#endif
|
||||||
|
};
|
||||||
|
#endif
|
||||||
21
t_heater.ino
21
t_heater.ino
@ -22,11 +22,10 @@ void HandlePwmHeaterDisplay() {
|
|||||||
|
|
||||||
byte graphHeight = 10;
|
byte graphHeight = 10;
|
||||||
byte currentAmount = Output==0 ? 0 : (Output-1)/99.0 * (graphHeight) + 1;
|
byte currentAmount = Output==0 ? 0 : (Output-1)/99.0 * (graphHeight) + 1;
|
||||||
Serial.println(currentAmount);
|
|
||||||
|
|
||||||
graphStates[currentGraphItemNumber] = currentAmount;
|
graphStates[currentGraphItemNumber] = currentAmount;
|
||||||
|
|
||||||
byte rightMargin = 128;
|
byte rightMargin = 127;
|
||||||
byte topMargin = 63-graphHeight;
|
byte topMargin = 63-graphHeight;
|
||||||
byte itemsDisplayed = 0;
|
byte itemsDisplayed = 0;
|
||||||
|
|
||||||
@ -44,11 +43,27 @@ void HandlePwmHeaterDisplay() {
|
|||||||
itemsDisplayed++;
|
itemsDisplayed++;
|
||||||
}
|
}
|
||||||
|
|
||||||
oled.setCursor(126-graphItems - 2*8, 7);
|
byte leftPosition = 126-graphItems - 3*6;
|
||||||
|
oled.setCursor(leftPosition, 7);
|
||||||
sprintf(timeBuffer, "%3d", (int)(Output));
|
sprintf(timeBuffer, "%3d", (int)(Output));
|
||||||
oled.invertText(true);
|
oled.invertText(true);
|
||||||
oled.print(timeBuffer);
|
oled.print(timeBuffer);
|
||||||
oled.invertText(false);
|
oled.invertText(false);
|
||||||
|
|
||||||
|
leftPosition -= 6*3;
|
||||||
|
oled.setCursor(leftPosition, 7);
|
||||||
|
sprintf(timeBuffer, "%3d", (int)(regulator.lastD));
|
||||||
|
oled.print(timeBuffer);
|
||||||
|
|
||||||
|
leftPosition -= 6*3;
|
||||||
|
oled.setCursor(leftPosition, 7);
|
||||||
|
sprintf(timeBuffer, "%3d", (int)(regulator.lastI));
|
||||||
|
oled.print(timeBuffer);
|
||||||
|
|
||||||
|
leftPosition -= 6*3;
|
||||||
|
oled.setCursor(leftPosition, 7);
|
||||||
|
sprintf(timeBuffer, "%3d", (int)(regulator.lastP));
|
||||||
|
oled.print(timeBuffer);
|
||||||
}
|
}
|
||||||
|
|
||||||
// LP_TIMER(1000, HandleHeater);
|
// LP_TIMER(1000, HandleHeater);
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user