Remote Control Vehicle Balance Controller
In this article, I want to present a PID-based multi-purpose balance controller that runs on the Teensy 3.2 board using Arduino libraries. It can control PWM-based actuators like servos or LEDs and electronic speed controllers driving brushless motors. This control is based on an external signal like the output from an RC receiver, and the output of an inertial measurement unit.
That means you can use this controller to stabilise or balance small vehicles like planes, multicopters, VTOLs, hydrofoils, hovercraft, or rockets…
To get the most use out of this article, you should have a basic understanding about PID, standard RC components like ESCs (electronic speed controllers), brushless motors, RX/TX, and some C/C++ programming skills.
Why use PID for balance control?
A PID controller governs a system so that the difference between a setpoint and a measured value is minimized, effectively driving our system to where we want it to be. Note that anything measurable and ‘steerable’ (that is at least somewhat linear) can be fed into a PID controller, like temperature, position, altitude, force… Using an IMU (Inertial Measurement Unit) to give us our measurement, we can generate an output signal that depends on the attitude and angular velocity of our system.
In quadcopters, planes, and many other vehicle types, we have to constantly monitor and manipulate the attitude, thrust, velocity and similar parameters to achieve the desired result. To influence those parameters, there are control surfaces, levers or propellers/jets which inflict forces on our craft in any necessary degree of freedom.
Those propellers and levers generally control rates too - rate of rotation, rate of climb, etc. That makes them ideal for being governed by a PID controller.
Cascaded PID
It is useful to use not only one but two PID controllers because the sensor output from the gyroscope is way faster than the output from the sensor fusion algorithm that fuses gyroscope and accelerometer data and produces an accurate attitude estimation. Gyroscope data drifts only over time, whereas accelerometer data does not drift but is subject to high-frequency noise. We can cascade the controllers: the first loop receives the setpoint interpreted as attitude and the currently most accurate estimation of the attitude and produces a needed angular rate as a result. This rate is fed into another loop as setpoint, and this rate loop receives the faster angular rate readings from the gyroscope directly. If the second loop executes more often, then it will ensure that the overall loop rate is fast and not limited by the time it takes to fuse the IMU data.
The setpoint can either be interpreted as a desired absolute angle or as a desired angular rate. In the first case, the PID controller will attempt to stabilize the craft to this angle, whereas in the second case, it will try to rotate the vehicle at the given rate. This is known as ‘rate mode’ or ‘acro mode’ in some flight controllers.
To achieve ‘rate mode’, the attitude loop simply needs to be ignored. A constant input on the roll stick of a plane in rate mode would result in a continuous roll, or a multicopter yaw stick would control the rate of z-rotation of the copter. In ‘stabilize mode’, a constant stick deviation is mapped to a constant vehicle deviation.
Hardware
I am using a Teensy 3.2 as the brains. It has an ARM Cortex M4 chip that can be overclocked to 120MHz which is more than sufficient for this control task. The board also has plenty of GPIO pins and can tolerate a wide range of supply voltages (such as 5V from a standard RC electronic speed controller) with its internal voltage regulator.
As IMU sensor, I previously used the widely available and familiar MPU6050. This sensor offers measurement of absolute orientation as well as angular rates from the gyroscope and provides data moderately quickly using the I2C protocol. But now I am using Kris Winers EM-SENtral-based ‘Ultimate Fusion Solution’ which has proven to be extremely fast and accurate, if a bit expensive. The sample rate for accelerometers and gyroscopes is set 1kHz for now. The proprietary sensor fusion in the SENtral processor reacts extremely quick to changes in angular position. I chose to let the sensors do their fusion for now because it is initially very simple, but I would like to use one of the newer Invensense sensors (ICM-20608, ICM-40602) and implement my own Kalman/Madgwick filter at some point. The ICM-40602 improves on the previous generation mostly in energy consumption, which is completely irrelevant in this case.
As basic peripheral hardware (actuators), any standard ESCs, servos and TX/RX combo using PWM should work.
Four simple steps
Here are the steps that need to be done to stabilize and control:
- Read input from the RC receiver
- Read sensor values from IMU
- Calculate PID response with those measurements as input
- Write the result to the connected actuators, using predefined weights
Reading input from the receiver
The receiver sends pulses of varying on-time corresponding to the stick
positions to our board. Using interrupts, we can measure the time passing
between a rising edge and a falling edge of one signal, which should always be
roughly within 1000us and 2000us (RC standard). We can simply hook an interrupt
to each input pin, log the system time on a rising flank, and calculate the
duration in microseconds since rising flank when the signal is falling again.
The code for this is in src/PWMReceiver.cpp
. PPM, which is conceptually
similar but needs only one wire, is also implemented in
src/PPMReceiver.cpp
. All receiver interfaces should implement virtual
functions in include/Receiver.h
.
Race Copters, Not Conditions
The duration measurement from our interrupt routines is written each time the interrupt routine executes on a falling edge. That means, we have to be careful when reading those values! I used shared volatile variables which are written to by the interrupts to store the measurements. The main loop copies the data to variables which it can use undisturbed. This way, it is always clear that the interrupt writes while the main loop reads the shared variables. When the variables are read, no interrupts are allowed since they might overwrite the values while reading them, violating the ‘sharing XOR mutability’ principle and thus inviting race conditions.
For a really good description on how to read RC receiver PPM output, have look at this excellent article by Ryan Boland. He explains it better than I will ever be able to - and with oscilloscope screenshots!
Read sensor values from IMU
Reading the measurements and computation results from the ‘Ultimate IMU Solution’ is done in a similar way like in Kris Winers example program, however I adapted large parts to skip software sensor fusion and provide an object-oriented interface to the sensor.
Calculate PID response
This is where the magic happens, but it is actually very simple. The PID algorithm in this discrete form (really the one everyone uses) is short and sweet, using the “poor man’s derivative (-)” and the “poor man’s integral (+)”. Some smoothing is applied to the D term (if enabled), but I am unsure if this is actually beneficial.
I adapted some ideas presented in Brett Beauregard’s series “Improving the beginners PID”
Eventually, PID coefficients will depend on the currently chosen mode (if I get to the implementation).
Write result to connected actuators
The output is basically finished. It only needs to be scaled a bit and then
written out using the various output methods defined in
src/ESCOutput.cpp
, src/ServoOutput.cpp
and
src/FastPWMOutput.cpp
. Eventually, this will need more work: depending
on the current flight mode, output mixer volumes will have to be chosen for
each output channel. Possible parameters for the output channel mixers are
throttle, roll/pitch/yaw attitude PID output, roll/pitch/yaw rate output,
rate/stabilize mode, and even motor type (servo/ESC).
This is basically a Matrix-vector multiplication. Each output is the weighted sum of the RC inputs and the PID controlling values in each axis. Since we have 8 outputs and 8+3 (currently) values that influence them, there are 8 rows and 11 columns in our matrix. The input vector consists of 8 receiver inputs + 3 PID control values. The i-th row of the matrix consists of the weights for each output for the j-th control value.
This is a bit complex, but essentially it allows us to say: “This output reacts to the throttle stick with a factor of 0.8, and to the yaw pid value with a factor of 0.1.” Additionally, we can store a matrix for each flight mode, and transition from one matrix to the next smoothly while transitioning flight modes.
Where is the catch? Well, none of this is implemented… The matrix multiplication part can be done in under a microsecond(!) on the teensy 3.5 using the capabilities of the Cortex M4. That is an interesting challenge and I hope I will get to it.
All of this depends on getting the PID and IMU and Output implementations to work with 16 bit fixed-point numbers, and never floats.
Give me the Code!
The code can be found at https://github.com/barafael/raPID
Further Ideas/Inspiration
I am thinking about allowing for several different flight/operation modes which consist mostly of a set of PID-coefficients and settings for mixers on each actuator output. A mixer would control how much impact each PID output should have on each actuator output.
Transitioning between flight modes could interpolate the values of the two mixers so that modes transition smoothly into each other.
If this transitional flight mode interpolation sounds familiar to you, it might be because of OpenAeroVTOL. It runs (really well) on the Hobbyking KK2.1.5 Multi-Rotor LCD Flight Control Board.
There is active and friendly support on the RCgroups forums, and it is often stated that this board is able to stabilize almost anything small-scale. However, development is confined by the hardware used (who knows for how long HK will be making the kk-board), the development style (code drops on release by the single author every couple months), and the word ‘VTOL’ in the name. It would be nice to really have a general controller not for just one niche, sacrificing some features (Tailsitter option, Heli-style rotor control) for a more general approach (also with more than the 3 flight modes which OAV has). Additionally, OAV does not support a derivative gain (the D in PID), arguing that this form of feedback is unnecessary for vehicles that naturally have a high damping factor due to having a higher mass or large wings. The accuracy of this statement is debatable, and to provide a more general approach a D gain is nice to have (if only for meta-stable systems or niche cases like multicopters).
Another feature of OAV is that it can entirely be configured via the onboard LCD screen (that is a miracle, really). Since version 1.5, you may also use (exclusively) Microsoft Excel to configure it, using a macro-based GUI in .xlsx format (yes, really). I am not at the point to even think about this by a long shot, but it would be nice to have a graphical configuration interface on a computer that can set options on the board over a serial or even wireless connection.