Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add antiwindup to PID controller #38

Closed
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions cfg/Parameters.cfg
Original file line number Diff line number Diff line change
Expand Up @@ -45,6 +45,7 @@ def generate(gen):
gen.add( "d" , double_t, 1,"Derivative gain.", 1.0 , -1000 , 1000)
gen.add( "i_clamp_min" , double_t, 1,"Min bounds for the integral windup", -10.0 , -1000 , 0)
gen.add( "i_clamp_max" , double_t, 1,"Max bounds for the integral windup", 10.0 , 0 , 1000)
gen.add( "antiwindup" , bool_t, 1,"Antiwindup.", False)
# PkgName #NodeName #Prefix for generated .h include file, e.g. ParametersConfig.py
exit(gen.generate(PACKAGE, "control_toolbox", "Parameters"))

Expand Down
35 changes: 22 additions & 13 deletions include/control_toolbox/pid.h
Original file line number Diff line number Diff line change
Expand Up @@ -120,20 +120,29 @@ class Pid
struct Gains
{
// Optional constructor for passing in values
Gains(double p, double i, double d, double i_max, double i_min)
Gains(double p, double i, double d, double i_max, double i_min, bool antiwindup)
: p_gain_(p),
i_gain_(i),
d_gain_(d),
i_max_(i_max),
i_min_(i_min)
i_min_(i_min),
antiwindup_(antiwindup)
{}
// Default constructor
Gains() {}
double p_gain_; /**< Proportional gain. */
double i_gain_; /**< Integral gain. */
double d_gain_; /**< Derivative gain. */
double i_max_; /**< Maximum allowable integral term. */
double i_min_; /**< Minimum allowable integral term. */
Gains()
: p_gain_(0.0),
i_gain_(0.0),
d_gain_(0.0),
i_max_(0.0),
i_min_(0.0),
antiwindup_(false)
{}
double p_gain_; /**< Proportional gain. */
double i_gain_; /**< Integral gain. */
double d_gain_; /**< Derivative gain. */
double i_max_; /**< Maximum allowable integral term. */
double i_min_; /**< Minimum allowable integral term. */
bool antiwindup_; /**< Antiwindup. */
};

/*!
Expand All @@ -147,7 +156,7 @@ class Pid
* \param i_max The max integral windup.
* \param i_min The min integral windup.
*/
Pid(double p = 0.0, double i = 0.0, double d = 0.0, double i_max = 0.0, double i_min = -0.0);
Pid(double p = 0.0, double i = 0.0, double d = 0.0, double i_max = 0.0, double i_min = -0.0, bool antiwindup = false);

/**
* \brief Copy constructor required for preventing mutexes from being copied
Expand All @@ -170,7 +179,7 @@ class Pid
* \param i_max The max integral windup.
* \param i_min The min integral windup.
*/
void initPid(double p, double i, double d, double i_max, double i_min);
void initPid(double p, double i, double d, double i_max, double i_min, bool antiwindup);

/*!
* \brief Zeros out Pid values and initialize Pid-gains and integral term limits
Expand All @@ -182,7 +191,7 @@ class Pid
* \param i_max The max integral windup.
* \param i_min The min integral windup.
*/
void initPid(double p, double i, double d, double i_max, double i_min, const ros::NodeHandle& /*node*/);
void initPid(double p, double i, double d, double i_max, double i_min, bool antiwindup, const ros::NodeHandle& /*node*/);

/*!
* \brief Initialize PID with the parameters in a namespace
Expand Down Expand Up @@ -229,7 +238,7 @@ class Pid
* \param i_max The max integral windup.
* \param i_min The min integral windup.
*/
void getGains(double &p, double &i, double &d, double &i_max, double &i_min);
void getGains(double &p, double &i, double &d, double &i_max, double &i_min, bool &antiwindup);

/*!
* \brief Get PID gains for the controller.
Expand All @@ -245,7 +254,7 @@ class Pid
* \param i_max The max integral windup.
* \param i_min The min integral windup.
*/
void setGains(double p, double i, double d, double i_max, double i_min);
void setGains(double p, double i, double d, double i_max, double i_min, bool antiwindup);

/*!
* \brief Set PID gains for the controller.
Expand Down
48 changes: 33 additions & 15 deletions src/pid.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -41,14 +41,16 @@
#include <control_toolbox/pid.h>
#include <tinyxml.h>

#include <boost/algorithm/clamp.hpp>

namespace control_toolbox {

static const std::string DEFAULT_NAMESPACE = "pid"; // \todo better default prefix?

Pid::Pid(double p, double i, double d, double i_max, double i_min)
Pid::Pid(double p, double i, double d, double i_max, double i_min, bool antiwindup)
: dynamic_reconfig_initialized_(false)
{
setGains(p,i,d,i_max,i_min);
setGains(p,i,d,i_max,i_min,antiwindup);

reset();
}
Expand All @@ -67,19 +69,19 @@ Pid::~Pid()
{
}

void Pid::initPid(double p, double i, double d, double i_max, double i_min,
void Pid::initPid(double p, double i, double d, double i_max, double i_min, bool antiwindup,
const ros::NodeHandle& /*node*/)
{
initPid(p, i, d, i_max, i_min);
initPid(p, i, d, i_max, i_min, antiwindup);

// Create node handle for dynamic reconfigure
ros::NodeHandle nh(DEFAULT_NAMESPACE);
initDynamicReconfig(nh);
}

void Pid::initPid(double p, double i, double d, double i_max, double i_min)
void Pid::initPid(double p, double i, double d, double i_max, double i_min, bool antiwindup)
{
setGains(p,i,d,i_max,i_min);
setGains(p,i,d,i_max,i_min, antiwindup);

reset();
}
Expand Down Expand Up @@ -124,6 +126,8 @@ bool Pid::init(const ros::NodeHandle &node, const bool quiet)
gains.i_max_ = std::abs(gains.i_max_); // make sure the value is >= 0
}

nh.param("antiwindup", gains.antiwindup_, false);

setGains(gains);

reset();
Expand All @@ -145,7 +149,8 @@ bool Pid::initXml(TiXmlElement *config)
config->Attribute("i") ? atof(config->Attribute("i")) : 0.0,
config->Attribute("d") ? atof(config->Attribute("d")) : 0.0,
std::abs(i_clamp),
-std::abs(i_clamp)
-std::abs(i_clamp),
config->Attribute("antiwindup") ? atof(config->Attribute("antiwindup")) : false
);

reset();
Expand Down Expand Up @@ -180,7 +185,7 @@ void Pid::reset()
cmd_ = 0.0;
}

void Pid::getGains(double &p, double &i, double &d, double &i_max, double &i_min)
void Pid::getGains(double &p, double &i, double &d, double &i_max, double &i_min, bool &antiwindup)
{
Gains gains = *gains_buffer_.readFromRT();

Expand All @@ -189,16 +194,17 @@ void Pid::getGains(double &p, double &i, double &d, double &i_max, double &i_min
d = gains.d_gain_;
i_max = gains.i_max_;
i_min = gains.i_min_;
antiwindup = gains.antiwindup_;
}

Pid::Gains Pid::getGains()
{
return *gains_buffer_.readFromRT();
}

void Pid::setGains(double p, double i, double d, double i_max, double i_min)
void Pid::setGains(double p, double i, double d, double i_max, double i_min, bool antiwindup)
{
Gains gains(p,i,d,i_max,i_min);
Gains gains(p,i,d,i_max,i_min, antiwindup);

setGains(gains);
}
Expand All @@ -221,7 +227,7 @@ void Pid::updateDynamicReconfig()
control_toolbox::ParametersConfig config;

// Get starting values
getGains(config.p, config.i, config.d, config.i_clamp_max, config.i_clamp_min);
getGains(config.p, config.i, config.d, config.i_clamp_max, config.i_clamp_min, config.antiwindup);

updateDynamicReconfig(config);
}
Expand All @@ -240,6 +246,7 @@ void Pid::updateDynamicReconfig(Gains gains_config)
config.d = gains_config.d_gain_;
config.i_clamp_max = gains_config.i_max_;
config.i_clamp_min = gains_config.i_min_;
config.antiwindup = gains_config.antiwindup_;

updateDynamicReconfig(config);
}
Expand All @@ -261,7 +268,7 @@ void Pid::dynamicReconfigCallback(control_toolbox::ParametersConfig &config, uin
ROS_DEBUG_STREAM_NAMED("pid","Dynamics reconfigure callback recieved.");

// Set the gains
setGains(config.p, config.i, config.d, config.i_clamp_max, config.i_clamp_min);
setGains(config.p, config.i, config.d, config.i_clamp_max, config.i_clamp_min, config.antiwindup);
}

double Pid::computeCommand(double error, ros::Duration dt)
Expand Down Expand Up @@ -299,18 +306,28 @@ double Pid::computeCommand(double error, double error_dot, ros::Duration dt)
if (dt == ros::Duration(0.0) || std::isnan(error) || std::isinf(error) || std::isnan(error_dot) || std::isinf(error_dot))
return 0.0;


// Calculate proportional contribution to command
p_term = gains.p_gain_ * p_error_;

// Calculate the integral of the position error
i_error_ += dt.toSec() * p_error_;

if(gains.antiwindup_)
{
// Prevent i_error_ from climbing higher than permitted by i_max_/i_min_
i_error_ = boost::algorithm::clamp(i_error_,
gains.i_min_ / std::abs(gains.i_gain_),
gains.i_max_ / std::abs(gains.i_gain_));
}

// Calculate integral contribution to command
i_term = gains.i_gain_ * i_error_;

// Limit i_term so that the limit is meaningful in the output
i_term = std::max( gains.i_min_, std::min( i_term, gains.i_max_) );
if(!gains.antiwindup_)
{
// Limit i_term so that the limit is meaningful in the output
i_term = boost::algorithm::clamp(i_term, gains.i_min_, gains.i_max_);
}

// Calculate derivative contribution to command
d_term = gains.d_gain_ * d_error_;
Expand Down Expand Up @@ -356,6 +373,7 @@ void Pid::printValues()
<< " D Gain: " << gains.d_gain_ << "\n"
<< " I_Max: " << gains.i_max_ << "\n"
<< " I_Min: " << gains.i_min_ << "\n"
<< " Antiwindup: " << gains.antiwindup_ << "\n"
<< " P_Error_Last: " << p_error_last_ << "\n"
<< " P_Error: " << p_error_ << "\n"
<< " I_Error: " << i_error_ << "\n"
Expand Down
3 changes: 2 additions & 1 deletion src/pid_gains_setter.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -53,11 +53,12 @@ bool PidGainsSetter::setGains(control_toolbox::SetPidGains::Request &req,
control_toolbox::SetPidGains::Response &resp)
{
for (size_t i = 0; i < pids_.size(); ++i)
pids_[i]->setGains(req.p, req.i, req.d, req.i_clamp, -req.i_clamp);
pids_[i]->setGains(req.p, req.i, req.d, req.i_clamp, -req.i_clamp, req.antiwindup);
node_.setParam("p", req.p);
node_.setParam("i", req.i);
node_.setParam("d", req.d);
node_.setParam("i_clamp", req.i_clamp);
node_.setParam("antiwindup", req.antiwindup);
return true;
}

Expand Down
1 change: 1 addition & 0 deletions srv/SetPidGains.srv
Original file line number Diff line number Diff line change
Expand Up @@ -2,4 +2,5 @@ float64 p
float64 i
float64 d
float64 i_clamp
bool antiwindup
---
Loading