From e61be81cc8aef4f45fdbd4a0c0290629dffa9275 Mon Sep 17 00:00:00 2001 From: Javier Balloffet Date: Wed, 27 Dec 2023 22:40:12 +0100 Subject: [PATCH] Improve app loop Signed-off-by: Javier Balloffet --- andino_firmware/src/app.cpp | 55 +++++++++++++++++++------------------ andino_firmware/src/app.h | 12 ++++++++ 2 files changed, 41 insertions(+), 26 deletions(-) diff --git a/andino_firmware/src/app.cpp b/andino_firmware/src/app.cpp index faf41ef7..1822b870 100644 --- a/andino_firmware/src/app.cpp +++ b/andino_firmware/src/app.cpp @@ -72,13 +72,6 @@ #include "motor.h" #include "pid.h" -// TODO(jballoffet): Move this variables to a different module. - -/* Track the next time we make a PID calculation */ -unsigned long nextPID = andino::Constants::kPidPeriod; - -long lastMotorCommand = andino::Constants::kAutoStopWindow; - // A pair of varibles to help parse serial commands int arg = 0; int index = 0; @@ -108,6 +101,10 @@ PID App::left_pid_controller_(Constants::kPidKp, Constants::kPidKd, Constants::k PID App::right_pid_controller_(Constants::kPidKp, Constants::kPidKd, Constants::kPidKi, Constants::kPidKo, -Constants::kPwmMax, Constants::kPwmMax); +unsigned long App::last_pid_computation_{0}; + +unsigned long App::last_set_motors_speed_cmd_{0}; + void App::setup() { // Required by Arduino libraries to work. init(); @@ -165,24 +162,16 @@ void App::loop() { } } - // Run a PID calculation at the appropriate intervals - if (millis() > nextPID) { - int left_motor_speed = 0; - int right_motor_speed = 0; - left_pid_controller_.compute(left_encoder_.read(), left_motor_speed); - right_pid_controller_.compute(right_encoder_.read(), right_motor_speed); - left_motor_.set_speed(left_motor_speed); - right_motor_.set_speed(right_motor_speed); - nextPID += Constants::kPidPeriod; + // Compute PID output at the configured rate. + if ((millis() - last_pid_computation_) > Constants::kPidPeriod) { + last_pid_computation_ = millis(); + adjust_motors_speed(); } - // Check to see if we have exceeded the auto-stop interval - if ((millis() - lastMotorCommand) > Constants::kAutoStopWindow) { - lastMotorCommand = millis(); - left_motor_.set_speed(0); - right_motor_.set_speed(0); - left_pid_controller_.enable(false); - right_pid_controller_.enable(false); + // Stop the motors if auto-stop interval has been reached. + if ((millis() - last_set_motors_speed_cmd_) > Constants::kAutoStopWindow) { + last_set_motors_speed_cmd_ = millis(); + stop_motors(); } // Required by Arduino libraries to work. @@ -228,6 +217,22 @@ void App::run_command() { } } +void App::adjust_motors_speed() { + int left_motor_speed = 0; + int right_motor_speed = 0; + left_pid_controller_.compute(left_encoder_.read(), left_motor_speed); + right_pid_controller_.compute(right_encoder_.read(), right_motor_speed); + left_motor_.set_speed(left_motor_speed); + right_motor_.set_speed(right_motor_speed); +} + +void App::stop_motors() { + left_motor_.set_speed(0); + right_motor_.set_speed(0); + left_pid_controller_.enable(false); + right_pid_controller_.enable(false); +} + void App::cmd_unknown(const char*, const char*) { Serial.println("Unknown command."); } void App::cmd_read_analog_gpio(const char* arg1, const char*) { @@ -259,7 +264,7 @@ void App::cmd_set_motors_speed(const char* arg1, const char* arg2) { const int right_motor_speed = atoi(arg2); // Reset the auto stop timer. - lastMotorCommand = millis(); + last_set_motors_speed_cmd_ = millis(); if (left_motor_speed == 0 && right_motor_speed == 0) { left_motor_.set_speed(0); right_motor_.set_speed(0); @@ -283,8 +288,6 @@ void App::cmd_set_motors_pwm(const char* arg1, const char* arg2) { const int left_motor_pwm = atoi(arg1); const int right_motor_pwm = atoi(arg2); - // Reset the auto stop timer. - lastMotorCommand = millis(); left_pid_controller_.reset(left_encoder_.read()); right_pid_controller_.reset(right_encoder_.read()); // Sneaky way to temporarily disable the PID. diff --git a/andino_firmware/src/app.h b/andino_firmware/src/app.h index b0745f6e..c879f2e8 100644 --- a/andino_firmware/src/app.h +++ b/andino_firmware/src/app.h @@ -56,6 +56,12 @@ class App { // TODO(jballoffet): Move this method to a different module. static void run_command(); + /// Computes the PID output and updates the motors speed accordingly. + static void adjust_motors_speed(); + + /// Stops the motors and disables the PID. + static void stop_motors(); + /// Callback method for an unknown command (default). // TODO(jballoffet): Parse arguments within callback method. static void cmd_unknown(const char* arg1, const char* arg2); @@ -99,6 +105,12 @@ class App { /// PID controllers (one per wheel). static PID left_pid_controller_; static PID right_pid_controller_; + + /// Tracks the last time the PID computation was made. + static unsigned long last_pid_computation_; + + /// Tracks the last time a `Commands::kSetMotorsSpeed` command was received. + static unsigned long last_set_motors_speed_cmd_; }; } // namespace andino