Skip to content

The C++ simulator and portion of the controls project.

Notifications You must be signed in to change notification settings

sunsided/FCND-Controls-CPP

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

3D Quadrotor Control

In this project, PID control for a simulated drone is implemented. Note that the controller code that was developed here is also used in the follow-up Quadrotor Sensor Fusion and Pose Estimation project. 🚁

Development environment

CMake on Linux

  1. Create a new directory for the build files:
cd FCND-Controls-CPP
mkdir build
  1. Navigate to the build directory and run cmake and then compile and build the code:
cd build
cmake ..
make
  1. You should now be able to run the simulator with ./CPPSim and you should see a single quadcopter, falling down.

Simulator Walkthrough

Now that you have all the code on your computer and the simulator running, let's walk through some of the elements of the code and the simulator itself.

The Code

For the project, the majority of your code will be written in src/QuadControl.cpp. This file contains all of the code for the controller that you will be developing.

All the configuration files for your controller and the vehicle are in the config directory. For example, for all your control gains and other desired tuning parameters, there is a config file called QuadControlParams.txt set up for you. An import note is that while the simulator is running, you can edit this file in real time and see the affects your changes have on the quad!

The syntax of the config files is as follows:

  • [Quad] begins a parameter namespace. Any variable written afterwards becomes Quad.<variablename> in the source code.
  • If not in a namespace, you can also write Quad.<variablename> directly.
  • [Quad1 : Quad] means that the Quad1 namespace is created with a copy of all the variables of Quad. You can then overwrite those variables by specifying new values (e.g. Quad1.Mass to override the copied Quad.Mass). This is convenient for having default values.

You will also be using the simulator to fly some difference trajectories to test out the performance of your C++ implementation of your controller. These trajectories, along with supporting code, are found in the traj directory of the repo.

The Simulator

In the simulator window itself, you can right click the window to select between a set of different scenarios that are designed to test the different parts of your controller.

The simulation (including visualization) is implemented in a single thread. This is so that you can safely breakpoint code at any point and debug, without affecting any part of the simulation.

Due to deterministic timing and careful control over how the pseudo-random number generators are initialized and used, the simulation should be exactly repeatable. This means that any simulation with the same configuration should be exactly identical when run repeatedly or on different machines.

Vehicles are created and graphs are reset whenever a scenario is loaded. When a scenario is reset (due to an end condition such as time or user pressing the ‘R’ key), the config files are all re-read and state of the simulation/vehicles/graphs is reset -- however the number/name of vehicles and displayed graphs are left untouched.

When the simulation is running, you can use the arrow keys on your keyboard to impact forces on your drone to see how your controller reacts to outside forces being applied.

Keyboard / Mouse Controls

There are a handful of keyboard / mouse commands to help with the simulator itself, including applying external forces on your drone to see how your controllers reacts!

  • Left drag - rotate
  • X + left drag - pan
  • Z + left drag - zoom
  • arrow keys, W and S - apply external force
  • C - clear all graphs
  • R - reset simulation
  • Space - pause simulation

Testing it Out

When you run the simulator (or start scenario 1_Intro), the quad is falling straight down. This is due to the fact that the thrusts are simply being set to:

QuadControlParams.Mass * 9.81 / 4

Therefore, if the mass doesn't match the actual mass of the quad, it'll fall down. Take a moment to tune the Mass parameter in QuadControlParams.txt to make the vehicle more or less stay in the same spot.

With the proper mass, your simulation looks a little like this:

The Tasks

For this project, you will be building a controller in C++. You will be implementing and tuning this controller in several steps.

You may find it helpful to consult the Python controller code (source as a reference when you build out this controller in C++.

Notes on Parameter Tuning

  1. Comparison to Python: Note that the vehicle you'll be controlling in this portion of the project has different parameters than the vehicle that's controlled by the Python code linked to above. The tuning parameters that work for the Python controller will not work for this controller
  2. Parameter Ranges: You can find the vehicle's control parameters in a file called QuadControlParams.txt. The default values for these parameters are all too small by a factor of somewhere between about 2X and 4X. So if a parameter has a starting value of 12, it will likely have a value somewhere between 24 and 48 once it's properly tuned.
  3. Parameter Ratios: In this one-page document (source) you can find a derivation of the ratio of velocity proportional gain to position proportional gain for a critically damped double integrator system. The ratio of kpV / kpP should be 4.

Dronecode provides some guidelines on PID tuning at

Body rate and roll/pitch control (scenario 2)

First, body rate and roll/pitch control was implemented; for this, Scenario 2 of the simulation was used. In this scenario, the quad is created above the origin using a small initial rotation speed about its roll axis.

  1. Body rate control

First, the controller needs to stabilize the rotational motion and bring the vehicle back to level attitude. For this, the GenerateMotorCommands() and BodyRateControl() methods were implemented first; the kpPQR controller gain was tuned accordingly to stabilize the rotational motion.

As a result, the roll rotation around the drone's x axis (omega.x) is controlled to 0 in about 0.065 seconds. Since the roll angle itself is not controlled yet, the drone will still stay at an angle and thus will slightly fly sideways. The controller does overshoot about 10 degrees due to motor dynamics.

  • kpPQR = 100, 100, 5
  1. Roll / pitch control

Ignoring yaw control for now, the RollPitchControl method was implemented and the bank controller gain kpBank was tuned accordingly by iteratively attempting to get a fast set time, then backing off to reduce overshoot. After doing so, the drone's roll angle gets corrected close to 0 in about 0.065 seconds; however, the body rate controller now takes slightly longer (0.070 seconds) to control the roll rate to 0.

  • kpPQR = 78, 78, 5
  • kpBank = 3

As a result, the drone's reaction did resemble a path somewhat like this:

... except that the drone actually drops to the ground.

Note that the animation was provided by the starter code and does not resemble the actual results. However, since the GIFs are already in the repo and quite heavy in size, I didn't bother creating a new animation that would bloat the repo up even more. For the time being, just take my word for it. 😁

Position/velocity and yaw angle control (scenario 3)

Next, the altitude, position and yaw control was implemented. This corresponds to Scenario 3 in the simulation. In there, two identical quads are created at some offset from their target points. In addition, one of the drones is initialized with a yaw of 45 degrees.

One of the implications here is that we need to convert information in global frame (such as the altitude) into thrust in the body frame. The following rotation relation came in handy:

  • The AltitudeControl() method was implemented first using a PD control scheme (with feedforward).
  • The LateralPositionControl() method was implemented next, again using a PD control scheme (with feedforward).

The controller parameters had to be re-adjusted and now are

  • kpPQR = 85, 85, 5
  • kpBank = 20
  • kpVelXY = 9.3
  • kpPosXY = 50

In this configuration, the goal position is reached after 0.360 seconds. Without yaw control, one of the two drones does have a bad attitude.

To mitigate, yaw control was added next (YawControl()) and the according controller gains set to

  • kpYaw = 3
  • kpPQR = 85, 85, 10

The drones now both reach their goals and have zero yaw. The following note of the original README.md is kept here for future reference:

Tune position control for settling time. Don’t try to tune yaw control too tightly, as yaw control requires a lot of control authority from a quadcopter and can really affect other degrees of freedom. This is why you often see quadcopters with tilted motors: better yaw authority!

Another one regarding the controller gain magnitudes:

Hint: For a second order system, such as the one for this quadcopter, the velocity gain (kpVelXY and kpVelZ) should be at least ~3-4 times greater than the respective position gain (kpPosXY and kpPosZ).

Interestingly, setting the velocity gain higher than the position gain led to chaos for me.

Non-idealities and robustness (scenario 4)

In Scenario 4 we can explore some of the non-idealities and properties issues of a controller. This is a configuration with 3 quads that are all are trying to move one meter forward. However, this time, these quads are all a bit different:

  • The green quad has its center of mass shifted back
  • The orange vehicle is an ideal quad
  • The red vehicle is heavier than usual.

When running the controllers in the configuration from the previous scenario, not everything works as expected (as, funnily enough, can be expected). While the red drone immediately decides to say hi to the floor, both the orange and green drones fly to about one meter above the reference point.

The original README.md stated

tip: relax the controller

and after tuning the controller gains once more to

  • kpPosXY = 30 (from 50)
  • kpPosZ = 20 (from 1)
  • kpVelXY = 9 (from 9.3)
  • kpVelZ = 9 (from 4)

all but the red drone reached the goal position. The green drone overshoots, but quickly corrects its mistake.

Since the red drone has a higher mass than anticipated, the altitude controller exhibits a systematic bias that cannot be governed by the proportional part (nor by the differential part, for that matter).

Adding an integrator term to build a feed-forward PID controller for the attitude solved this problem, but not before changing the controller once more:

  • kpPosXY = 29 (from 30)
  • kpPosZ = 40 (from 20)
  • KiPosZ = 40 (from 20)

Tracking trajectories

With all parts assembled, the performance of the controller(s) can be tested on simulation Scenario 5, where two drones are required to fly a predefined trajectory:

  • the orange one is following traj/FigureEight.txt
  • the other one is following traj/FigureEightFF.txt - for now this is the same trajectory.

Out of the box, the drones perform pretty creatively:

This reminded me that the controller gains weren't tuned in quite a while, so off we pop:

  • kpPosXY = 35 (from 29)
  • kpPosZ = 25 (from 40)
  • KiPosZ = 42 (from 45)
  • plVelXY = 12 (from 13)
  • kpVelZ = 15 (from 10)
  • kpBank = 13 (from 20)

After that, the yellow drone followed the path reasonably well (within bounds, that is) whereas the red drone constantly missed the goal positions by overshooting. The reason should be that the first drone does not receive any feedforward information (specifically, the target velocities are unknown), whereas the second drone does.

Extra Challenge 1 (Optional)

The Python script config/traj/MakePeriodicTrajectory.py was updated to provide velocities in addition to positional data by taking a location derivative. A new FigureEightFF.txt was created (it was basically the same as the originally provided one though).

As mentioned above, providing velocity information allows them to be used as feed-forward terms, which in turn (drastically) improves the controller performance.

This is, in essence, how it looks.

Extra Challenge 2 (Optional)

For flying a trajectory, is there a way to provide even more information for even better tracking? How about trying to fly this trajectory as quickly as possible (but within following threshold)!

So far, the only feedforward information available in the trajectory files are velocities. The implemented controllers are designed to make use of acceleration information as well; the acceleration is taken as the second derivate of the positions.

Looking at the trajectory generation code in Trajectory.cpp, we find that the trajectory files only list

  • time,
  • x, y and z position,
  • x, y and z velocities,
  • yaw, pitch and roll, as well as
  • omega.x, omega.y and omega.z (i.e., rotational velocities).

In theory, maximum thrust should be achievable when the drone's up axis points along the trajectory to fly. We'd need to take care of not falling to the ground, but the altitude controller should be able to fix that.

Evaluation

To assist with tuning of your controller, the simulator contains real time performance evaluation. We have defined a set of performance metrics for each of the scenarios that your controllers must meet for a successful submission.

There are two ways to view the output of the evaluation:

  • in the command line, at the end of each simulation loop, a PASS or a FAIL for each metric being evaluated in that simulation
  • on the plots, once your quad meets the metrics, you will see a green box appear on the plot notifying you of a PASS

Performance Metrics

The specific performance metrics are as follows:

  • scenario 2

    • roll should less than 0.025 radian of nominal for 0.75 seconds (3/4 of the duration of the loop)
    • roll rate should less than 2.5 radian/sec for 0.75 seconds
  • scenario 3

    • X position of both drones should be within 0.1 meters of the target for at least 1.25 seconds
    • Quad2 yaw should be within 0.1 of the target for at least 1 second
  • scenario 4

    • position error for all 3 quads should be less than 0.1 meters for at least 1.5 seconds
  • scenario 5

    • position error of the quad should be less than 0.25 meters for at least 3 seconds

Authors

Thanks to Fotokite for the initial development of the project code and simulator.

Releases

No releases published

Packages

No packages published

Languages

  • C 86.0%
  • C++ 12.2%
  • Jupyter Notebook 1.3%
  • Python 0.3%
  • TeX 0.2%
  • Objective-C 0.0%