PID Tutorials for Line Following

line_follower_PID.zip (1639Bytes)
PID_Line_Following_Tutorials.pdf (519025Bytes)

UPDATE : (07/10/2014) Added a PDF version of this tutorial and a sample video of a PID based Line Follower.

UPDATE : ( 01/16/2014 ) A few updates on making things more clearer about P, I , D and Kp, Ki, Kd.

UPDATE : ( 01/18/2014 ) A few minor additions and got rid of some technical errors, misplaced values, etc.

Hello LMRians,

In this writeup, I will try to create a tutorial for tuning your robot using PID to follow a line. This tutorial won't go deep down into the details of PID and its applications but will just attempt to show how a robot can be tuned with the PID parameters to follow a line.

In my PID based line follower, to remove any sort of difficulties and complexities, I chose the Pololu QTR-8RC array sensor. In this tutorial as well, I will hover around this specific line sensor and how it's readings can be integrated into your own PID line following code.

The Line Following Sensor : Pololu QTR-8RC Array Sensor

0J615_600.jpg

 

The Pololu QTR-8RC sensor trims the fat when it comes to PID based line following. It not only signals you as to where the lines are located, but in turn, it also outputs the position of the robot while following a track.

 

KEY TERMS IN PID BASED LINE FOLLOWING:

PID stands for Proportional, Integral and Derivative. In this tutorial, I am taking a few peeks at Pololu's documents. As stated there :

·         The proportional value is approximately proportional to your robot’s position with respect to the line. That is, if your robot is precisely centered on the line, we expect a proportional value of exactly 0.


·         The integral value records the history of your robot’s motion: it is a sum of all of the values of the proportional term that were recorded since the robot started running.


       The derivative is the rate of change of the proportional value. 

 

(Note that the proportional term (P) , the integral term (I) and the derivative term (D) are the measures of the errors encountered while the robot follows a line. Kp , Ki and Kd are the proportional , integral and derivative constants that are then multiplied to the errors to adjust the robot's position. Kp, Ki and Kd are the PID parameters.)

In this tutorial, we will talk about just the proportional and derivative terms and in turn, just Kp and Kd. However, results can be accomplished using the Ki term as well. Perfectly tuning the Kp and Kd terms should be just enough though.

 

There are two important sensor readings to look into, when implementing PID. However, these readings are not simply the classic "analogRead(leftFarSensor)" type of readings. These readings are not only the analog readings, but also the positional readings of the robot. ( Let's not get into the technicalities of the reading methods of the Pololu Array Sensor. If you are interested, it is well documented on the Pololu website. The basic thing is : This sensor provides values from 0 to 2500 ranging from maximum reflectance to minimum reflectance, but at the same time, also provides additional readings on the robot's position i.e. how far the robot has stranded off from the line. )


Setpoint Value:

This is our goal. It is what we are hunting for. The setpoint value is the reading that corresponds to the "perfect" placement of sensors on top of the lines. By perfect, I am referring to the moment when the line is exactly at the center of our sensor. I will add on this later. If the robot is perfectly tuned, then it is capable of positioning itself throughout the course with the setpoint value. 

 

Current Value:

The current value is obviously the instantaneous readings of the sensor. For eg : If you are using this array sensor and are making use of 6 sensors, you will receive a positional reading of 2500 if you are spot on, around 0 if you are far too left from the line and around 5000 if you are far too right.

 

Error:

This is the difference of the two values. Our goal is to make the error zero. Then only can the robot smoothly follow the line. Hopefully, this term will get more clear as we move forward.

 

THE PID EQUATION:

Wikipedia provides a number equations that all refer to the applications of PID. They might look complex and hard-to-understand. However, Line following is one of the simple applications of PID, thus, the equations used are not very daunting after all.

 

1) The first task is to calculate the error. This can also be called the proportional term i.e. it is proportional to the robot's position with respect to the line.

     Error = Setpoint Value - Current Value = 2500 - position 

 As you can see, I have made use of 6 sensors. Our goal is to achieve a state of zero-error in the PID equation. Just imagine, as I stated earlier, the sensor gives a positional reading of 2500 when the robot is perfectly placed. Substituting 2500 as the position in the above equation we get,

     Error = 2500 - position = 2500 - 2500 = 0

Clearly, 2500 is the position that zeros out the equation and is thus, our Setpoint Value.

(Note : You may notice, what if the sensor is to the far left? The positional reading would then be 0. Would that result in an error of 2500? Correct. And that error of 2500 is taken into account, and necessary changes are made in the next equation )

 

2) The second task is to determine the adjusted speeds of the motors.

     MotorSpeed = Kp * Error + Kd * ( Error - LastError );

     LastError = Error;

     RightMotorSpeed = RightBaseSpeed + MotorSpeed;

     LeftMotorSpeed = LeftBaseSpeed - MotorSpeed;


Note the equation -------------- MotorSpeed = Kp * Error + Kd * ( Error - LastError );

The proportional constant is multiplied to the proportional term i.e. the error.

The derivative constant is multiplied to the derivative term. ( the derivative term, as the name suggests, measures the rate of change in the proportional term)

Now that we have calculated our error, the margin by which our robot drifts across the track, it is time for us to scrutinize the error and adjust the motor speeds accordingly. Logically speaking, an error of 2500 ( in other words a positional reading of 0 ) means our robot is out to the left, which means that our robot needs to go a bit right, which inturn means, the right motor needs to slow down and the left motor needs to speed up (Differential Drive). THIS IS PID! It is basically just an implementation of the general rules of line following in real-time, in high-definition, and in split-seconds!

The MotorSpeed value is determined from the equation itself. RightBaseSpeed and LeftBaseSpeed are the speedsat which the robot runs at when the error is zero. Any PWM value from 0 -255 should do. ( If you are using an 8-bit PWM microcontroller that is :P ) 

So this is what we have so far :

 

1
2
3
4
5
6
7
  int error = position - 2500;

int motorSpeed = Kp error + Kd (error - lastError);
lastError = error;

int rightMotorSpeed = rightBaseSpeed + motorSpeed;
int leftMotorSpeed = leftBaseSpeed - motorSpeed;

 

( I have become a fan of this beautifier. Thank you Ladvien Sir! )

Troubles you may run into:

As I stated earlier, PID is a super hero and can be a mega-villain as well. It can deceive you with the erroneous responses but at the same time, it’s accurate results will WOW you, if it gets implemented correctly. These are a few troubles I personally ran into while building my line follower.

1) Errors in Sign :

This is the most dangerous of all the errors you may encounter. A small sign error in any part of the PID equation could spell trouble. In this example, I have used the equation error = 2500 - position . This may not work for you. This works for a line follower following a white line on a black background. If you have a black line on a white background, you might need to set it to error = position - 2500 and might also need to make other changes in the signs in the PID equations.

If you are finding trouble to adjust your robot, an another option may include carrying your robot above a certain height, just to determine if the wheels are spinning in the correct directions. For eg : If the robot is at a position of 5000, the robot is to the far right. However, if the wheels are still spinning in such a way that the robot is still tending to move to the right, there is a sign error in your second equation.

 

 RightMotorSpeed = RightBaseSpeed + MotorSpeed; ( + might need to be changed to - )

     LeftMotorSpeed = LeftBaseSpeed - MotorSpeed; ( - might need to be changed to + )

 

2) Troubles with finding the PID parameter values :

I have read many comments on many PID line following projects and the most popular question that stands out is : "Please provide me the Kp, Ki and Kd values!!!"

Really? It’s a great experience, if not the best, in finding them. You falter and falter and falter, and finally, you will have something running at the end. So give it a try yourself!

Whatever you do and however you forge ahead with tuning the PID parameters, I suggest to start small. Maybe a PWM value of around 100 should do for the base motor speeds. Then plug in the Kp and Kd terms. As I mentioned way above, the P term is the Proportional and the D term is the Derivative. The derivative of an error is smaller than the error itslef. Thus, to bring about meaningful corrections, it needs to multiplied by a bigger constant and thus, the Kp term is very small in comparison to the Kd term. You could start just with the Kp term, and then when you have something fine, you can add the Kd term and experiment as you move towards your goal of a smooth line follower.

Thanks for reading this tutorial. It’s not that well managed and systematic, I guess. I may have made a few errors along the way, so constructive suggestions are highly appreciated. I, myself, am still learning to make my programs look more matured. I will try to make some more updates and hopefully add some more ideas, if I learn them in the days to come. Do find the attached code if you are having troubles. ( Still, there are no values for the constants :stuck_out_tongue: . I’ll give you a hint though. Kd is at least 20 times bigger than Kp in my case )

Happy Tuning!
https://www.youtube.com/watch?v=G7PC9B4DBdI

Quick edit please.

Early on, you say:

"In this tutorial, we will talk about just the Kp term and the Kd term, however, results can be accomplished using the Ki term as well. Perfectly tuning the Kp and Kd terms should be just enough though."

Can you maybe place the Kp term with your description of Proportional  and Kd term with Derivative, and show the equation MotorSpeed = Kp * Error + Kd * ( Error - LastError );

 

Otherwise, I found this article very articulate and informative.

Great work!

p.s.  maybe place the associated code as an attachement instead of inline.  The site does not handle inline code well.

Cheers.

 

Thanks for spotting that

Thanks for spotting that unix_guru. I appreciate it. :slight_smile:

I too am still learning more and more about PID, so this tutorial is basically just an idea about how the line following code works.

I guess I made it sound like the Kp term and the proportional term are the same thing! ( they are definitely not! )

I’ll try and make things a bit clearer in that part.

Thanks again

Oh and yeah, I hope we have a ...... thing in the future for inserting code on LMR posts.

Ashim

This is a great

This is a great tutorial…the explanation for the PID parameters is really  nice.It’ll be really useful since I’m building a line  follower now…Thanks for this tutorial!

Also your line follower is really nice. :slight_smile:

Thanks for the compliment

Thanks for the compliment robodude. Great to know it’s going to assist you!

Thanks for the compliment

Sorry - Double post.

code not working

I built this robot but no matter what I do, the robot only goes straight and doesn’t follow the line at all. Here is my code. Any ideas?

#include <QTRSensors.h>

 

#define Kp 20 // experiment to determine this, start by something small that just makes your bot follow the line at a slow speed 

#define Ki 1

#define Kd 400 // experiment to determine this, slowly increase the speeds and adjust this value. ( Note: Kp < Kd)

#define rightMaxSpeed 75 // max speed of the robot

#define leftMaxSpeed 75 // max speed of the robot

#define rightBaseSpeed 50 // this is the speed at which the motors should spin when the robot is perfectly on the line

#define leftBaseSpeed 50  // this is the speed at which the motors should spin when the robot is perfectly on the line

#define NUM_SENSORS 6     // number of sensors used

#define TIMEOUT 2500  // waits for 2500 us for sensor outputs to go low

#define EMITTER_PIN 2     // emitter is controlled by digital pin 2

#define rightMotor1 3

#define rightMotor2 4

#define rightMotorPWM 5

#define leftMotor1 12

#define leftMotor2 13

#define leftMotorPWM 11

#define motorPower 8

 

QTRSensorsRC qtrrc((unsigned char[]) {14, 15, 16, 17, 18, 19} ,NUM_SENSORS, TIMEOUT, EMITTER_PIN); // sensor connected through analog pins A0 - A5 i.e. digital pins 14-19

 

unsigned int sensorValues[NUM_SENSORS];

 

void setup()

{

  pinMode(rightMotor1, OUTPUT);

  pinMode(rightMotor2, OUTPUT);

  pinMode(rightMotorPWM, OUTPUT);

  pinMode(leftMotor1, OUTPUT);

  pinMode(leftMotor2, OUTPUT);

  pinMode(leftMotorPWM, OUTPUT);

  pinMode(motorPower, OUTPUT);

 

  for (int i = 0; i < 100; i++) // calibrate for sometime by sliding the sensors across the line

    qtrrc.calibrate();

    delay(20);  

  delay(2000); // wait for 2s to position the bot before entering the main loop

 

int lastError = 0;

int I = 0;

 

void loop()

{

  unsigned int sensors[NUM_SENSORS];

  int position = qtrrc.readLine(sensors); // get calibrated readings along with the line position, refer to the QTR Sensors Arduino Library for more details on line position.

  int error = position - 2500;

 

  int P = error * Kp;

  int I = (I + error) * Ki;

  int D = (error - lastError) * Kd;

  int motorSpeed = P + I + D;

  lastError = error;

 

  int rightMotorSpeed = rightBaseSpeed + motorSpeed;

  int leftMotorSpeed = leftBaseSpeed - motorSpeed;

 

  if (rightMotorSpeed > rightMaxSpeed) rightMotorSpeed = rightMaxSpeed; // prevent the motor from going beyond max speed

  if (leftMotorSpeed > leftMaxSpeed) leftMotorSpeed = leftMaxSpeed; // prevent the motor from going beyond max speed

  if (rightMotorSpeed < 0) rightMotorSpeed = 0; // keep the motor speed positive

  if (leftMotorSpeed < 0) leftMotorSpeed = 0; // keep the motor speed positive

 

  digitalWrite(motorPower, HIGH); // move forward with appropriate speeds

  digitalWrite(rightMotor1, HIGH);

  digitalWrite(rightMotor2, LOW);

  analogWrite(rightMotorPWM, rightMotorSpeed);

  digitalWrite(motorPower, HIGH);

  digitalWrite(leftMotor1, HIGH);

  digitalWrite(leftMotor2, LOW);

  analogWrite(leftMotorPWM, leftMotorSpeed);

}

Thank you very much for your article.

Thank you very much.It is very useful to lead me to the right way on completing my line following robot.

The Kp term  and the Kd term did well . The equation could work in balance robot with a ultrasonic sensor.Maybe you can try.

Thank you again.

white line follower

i need a white line follower robot i couldnt edit this code for it do you have code for white line follower? hpw can i transllate this code for white line follower robot.

Power the motor …Urgent Question

If we power the motor through ‘motor power pin’ (which is actually a digital pin) then the motor can only absorb either 5V or 0V power(as per HIGH or LOW command)… In this case, can a DC motor run with this very low power? because to run two DC motors, it is supposed to require at least 10V-12V power…

 

Regards,

Navid

What’s with all the complex

What’s with all the complex code? surely this can be simplified…

Left Color sensor         Right Color sensor
                         &nbsp;      /
                           &nbsp;  /
                             /
               Color sensor detection
                  Adjust accordingly

Blue Background

Hi i am supposed to make a line follower that can follow a white line in blue background. I need help with sensors. Can you suggest me some nice sensors for that.
Thanks