Quadcopter Dynamics, Simulation, and Control
Quadcopter Dynamics, Simulation, and Control
Quadcopter Dynamics, Simulation, and Control
Introduction
A helicopter is a ying vehicle which uses rapidly spinning rotors to push air downwards,
thus creating a thrust force keeping the helicopter aloft. Conventional helicopters have two
rotors. These can be arranged as two coplanar rotors both providing upwards thrust, but
spinning in opposite directions (in order to balance the torques exerted upon the body of the
helicopter). The two rotors can also be arranged with one main rotor providing thrust and a
smaller side rotor oriented laterally and counteracting the torque produced by the main rotor.
However, these congurations require complicated machinery to control the direction of mo-
tion; a swashplate is used to change the angle of attack on the main rotors. In order to produce
a torque the angle of attack is modulated by the location of each rotor in each stroke, such that
more thrust is produced on one side of the rotor plane than the other. The complicated design
of the rotor and swashplate mechanism presents some problems, increasing construction costs
and design complexity.
A quadrotor helicopter (quadcopter) is a helicopter which has four equally spaced ro-
tors, usually arranged at the corners of a square body. With four independent rotors, the need
for a swashplate mechanism is alleviated. The swashplate mechanism was needed to allow
the helicopter to utilize more degrees of freedom, but the same level of control can be obtained
by adding two more rotors.
The development of quadcopters has stalled until very recently, because controlling
four independent rotors has proven to be incredibly difcult and impossible without elec-
tronic assistance. The decreasing cost of modern microprocessors has made electronic and
even completely autonomous control of quadcopters feasible for commercial, military, and
even hobbyist purposes.
Quadcopter control is a fundamentally difcult and interesting problem. With six de-
grees of freedom (three translational and three rotational) and only four independent inputs
(rotor speeds), quadcopters are severely underactuated. In order to achieve six degrees of
freedom, rotational and translational motion are coupled. The resulting dynamics are highly
nonlinear, especially after accounting for the complicated aerodynamic effects. Finally, unlike
ground vehicles, helicopters have very little friction to prevent their motion, so they must pro-
vide their own damping in order to stop moving and remain stable. Together, these factors
create a very interesting control problem. We will present a very simplied model of quad-
copter dynamics and design controllers for our dynamics to follow a designated trajectory. We
will then test our controllers with a numerical simulation of a quadcopter in ight.
1
Quadcopter Dynamics
We will start deriving quadcopter dynamics by introducing the two frames in which will op-
erate. The inertial frame is dened by the ground, with gravity pointing in the negative z
direction. The body frame is dened by the orientation of the quadcopter, with the rotor axes
pointing in the positive z direction and the arms pointing in the x and y directions.
Quadcopter Body Frame and Inertial Frame
Kinematics
Before delving into the physics of quadcopter motion, let us formalize the kinematics in the
body and inertial frames. We dene the position and velocity of the quadcopter in the inertial
frame as x = (x, y, z)
T
and x = ( x, y, z)
T
, respectively. Similarly, we dene the roll, pitch, and
yaw angles in the body frame as = (, , )
T
, with corresponding angular velocities equal
to
= (
,
,
)
T
. However, note that the angular velocity vector =
. The angular velocity
is a vector pointing along the axis of rotation, while
is just the time derivative of yaw, pitch,
and roll. In order to convert these angular velocities into the angular velocity vector, we can
use the following relation:
=
_
_
1 0 s
0 c
0 s
_
_
+ c
_
_
For a given vector v in the body frame, the corresponding vector is given by Rv in the inertial
frame.
Physics
In order to properly model the dynamics of the system, we need an understanding of the
physical properties that govern it. We will begin with a description of the motors being used
for our quadcopter, and then use energy considerations to derive the forces and thrusts that
these motors produce on the entire quadcopter. All motors on the quadcopter are identical, so
we can analyze a single one without loss of generality. Note that adjacent propellers, however,
are oriented opposite each other; if a propeller is spinning clockwise, then the two adjacent
ones will be spinning counter-clockwise, so that torques are balanced if all propellers are
spinning at the same rate.
2
Motors
Brushless motors are used for all quadcopter applications. For our electric motors, the torque
produced is given by
= K
t
(I I
0
)
where is the motor torque, I is the input current, I
0
is the current when there is no load on
the motor, and K
t
is the torque proportionality constant. The voltage across the motor is the
sum of the back-EMF and some resistive loss:
V = IR
m
+ K
v
K
t
Further simplifying our model, we assume that K
t
I
0
. This is not altogether unreason-
able, since I
0
is the current when there is no load, and is thus rather small. In practice, this
approximation holds well enough. Thus, we obtain our nal, simplied equation for power:
P
K
v
K
t
.
Forces
The power is used to keep the quadcopter aloft. By conservation of energy, we know that
the energy the motor expends in a given time period is equal to the force generated on the
propeller times the distance that the air it displaces moves (P d t = F d x). Equivalently, the
power is equal to the thrust times the air velocity (P = F
d x
d t
).
P = Tv
h
We assume vehicle speeds are low, so v
h
is the air velocity when hovering. We also assume
that the free stream velocity, v
T
2A
where is the density of the surrounding air and A is the area swept out by the rotor. Using
our simplied equation for power, we can then write
P =
K
v
K
t
=
K
v
K
K
t
T =
T
3
2
_
2A
.
Note that in the general case, =r
_
2A
K
t
_
2
= k
2
3
where k is some appropriately dimensioned constant. Summing over all the motors, we nd
that the total thrust on the quadcopter (in the body frame) is given by
T
B
=
4
i=1
T
i
= k
_
_
0
0
i
2
_
_
.
In addition to the thrust force, we will model friction as a force proportional to the
linear velocity in each direction. This is a highly simplied view of uid friction, but will be
sufcient for our modeling and simulation. Our global drag forces will be modeled by an
additional force term
F
D
=
_
_
k
d
x
k
d
y
k
d
z
_
_
If additional precision is desired, the constant k
d
can be separated into three separate friction
constants, one for each direction of motion. If we were to do this, we would want to model
friction in the body frame rather than the inertial frame.
Torques
Now that we have computed the forces on the quadcopter, we would also like to compute the
torques. Each rotor contributes some torque about the body z axis. This torque is the torque
required to keep the propeller spinning and providing thrust; it creates the instantaneous
angular acceleration and overcomes the frictional drag forces. The drag equation from uid
dynamics gives us the frictional force:
F
D
=
1
2
C
D
Av
2
.
where is the surrounding uid density, A is the reference area (propeller cross-section, not
area swept out by the propeller), and C
D
is a dimensionless constant. This, while only accurate
in some in some cases, is good enough for our purposes. This implies that the torque due to
drag is given by
D
=
1
2
RC
D
Av
2
=
1
2
RC
D
A(R)
2
= b
2
where is the angular velocity of the propeller, R is the radius of the propeller, and b is some
appropriately dimensioned constant. Note that weve assumed that all the force is applied at
the tip of the propeller, which is certainly inaccurate; however, the only result that matters for
our purposes is that the drag torque is proportional to the square of the angular velocity. We
can then write the complete torque about the z axis for the ith motor:
z
= b
2
+ I
M
where I
M
is the moment of inertia about the motor z axis, is the angular acceleration of
the propeller, and b is our drag coefcient. Note that in steady state ight (i.e. not takeoff
or landing), 0, since most of the time the propellers will be maintaining a constant (or
almost constant) thrust and wont be accelerating. Thus, we ignore this term, simplifying the
entire expression to
z
= (1)
i+1
b
i
2
.
where the (1)
i+1
term is positive for the ith propeller if the propeller is spinning clockwise
and negative if it is spinning counterclockwise. The total torque about the z axis is given by
the sum of all the torques from each propeller:
= b
_
1
2
2
2
+
3
2
4
2
_
The roll and pitch torques are derived from standard mechanics. We can arbitrarily choose the
i = 1 and i = 3 motors to be on the roll axis, so
r T = L(k
1
2
k
3
2
) = Lk(
1
2
3
2
)
4
Correspondingly, the pitch torque is given by a similar expression
= Lk(
2
2
4
2
)
where L is the distance from the center of the quadcopter to any of the propellers. All together,
we nd that the torques in the body frame are
B
=
_
_
Lk(
1
2
3
2
)
Lk(
2
2
4
2
)
b
_
1
2
2
2
+
3
2
4
2
_
_
_
The model weve derived so far is highly simplied. We ignore a multitude of advanced
effects that contribute to the highly nonlinear dynamics of a quadcopter. We ignore rotational
drag forces (our rotational velocities are relatively low), blade apping (deformation of pro-
peller blades due to high velocities and exible materials), surrounding uid velocities (wind),
etc. With that said, we now have all the parts necessary to write out the dynamics of our quad-
copter.
Equations of Motion
In the inertial frame, the acceleration of the quadcopter is due to thrust, gravity, and linear
friction. We can obtain the thrust vector in the inertial frame by using our rotation matrix R to
map the thrust vector from the body frame to the inertial frame. Thus, the linear motion can
be summarized as
m x =
_
_
0
0
mg
_
_
+ RT
B
+ F
D
where x is the position of the quadcopter, g is the acceleration due to gravity, F
D
is the drag
force, and T
B
is the thrust vector in the body frame.
While it is convenient to have the linear equations of motion in the inertial frame, the
rotational equations of motion are useful to us in the body frame, so that we can express
rotations about the center of the quadcopter instead of about our inertial center. We derive the
rotational equations of motion from Eulers equations for rigid body dynamics. Expressed in
vector form, Eulers equations are written as
I + (I) =
where is the angular velocity vector, I is the inertia matrix, and is a vector of external
torques. We can rewrite this as
=
_
_
x
y
z
_
_
= I
1
( (I)) .
We can model our quadcopter as two thin uniform rods crossed at the origin with a point mass
(motor) at the end of each. With this in mind, its clear that the symmetries result in a diagonal
inertia matrix of the form
I =
_
_
I
xx
0 0
0 I
yy
0
0 0 I
zz
_
_
.
Therefore, we obtain our nal result for the body frame rotational equations of motion
=
_
_
I
xx
1
I
yy
1
I
zz
1
_
_
_
I
yy
I
zz
I
xx
y
z
I
zz
I
xx
I
yy
x
z
I
xx
I
yy
I
zz
x
y
_
_
5
Simulation
Now that we have complete equations of motion describing the dynamics of the system, we
can create a simulation environment in which to test and viewthe results of various inputs and
controllers. Although more advanced methods are available, we can quickly write a simulator
which utilizes Eulers method for solving differential equations to evolve the system state. In
MATLAB, this simulator would be written as follows.
1 % Simulation times, in seconds.
2 start_time = 0;
3 end_time = 10;
4 dt = 0.005;
5 times = start_time:dt:end_time;
6
7 % Number of points in the simulation.
8 N = numel(times);
9
10 % Initial simulation state.
11 x = [0; 0; 10];
12 xdot = zeros(3, 1);
13 theta = zeros(3, 1);
14
15 % Simulate some disturbance in the angular velocity.
16 % The magnitude of the deviation is in radians / second.
17 deviation = 100;
18 thetadot = deg2rad(2
*
deviation
*
rand(3, 1) - deviation);
19
20 % Step through the simulation, updating the state.
21 for t = times
22 % Take input from our controller.
23 i = input(t);
24
25 omega = thetadot2omega(thetadot, theta);
26
27 % Compute linear and angular accelerations.
28 a = acceleration(i, theta, xdot, m, g, k, kd);
29 omegadot = angular_acceleration(i, omega, I, L, b, k);
30
31 omega = omega + dt
*
omegadot;
32 thetadot = omega2thetadot(omega, theta);
33 theta = theta + dt
*
thetadot;
34 xdot = xdot + dt
*
a;
35 x = x + dt
*
xdot;
36 end
We would then need functions to compute all of the physical forces and torques.
1 % Compute thrust given current inputs and thrust coefficient.
2 function T = thrust(inputs, k)
3 % Inputs are values for
i
2
4 T = [0; 0; k
*
sum(inputs)];
5 end
6
7 % Compute torques, given current inputs, length, drag coefficient, and thrust coefficient.
8 function tau = torques(inputs, L, b, k)
9 % Inputs are values for
i
2
10 tau = [
11 L
*
k
*
(inputs(1) - inputs(3))
6
12 L
*
k
*
(inputs(2) - inputs(4))
13 b
*
(inputs(1) - inputs(2) + inputs(3) - inputs(4))
14 ];
15 end
16
17 function a = acceleration(inputs, angles, xdot, m, g, k, kd)
18 gravity = [0; 0; -g];
19 R = rotation(angles);
20 T = R
*
thrust(inputs, k);
21 Fd = -kd
*
xdot;
22 a = gravity + 1 / m
*
T + Fd;
23 end
24
25 function omegadot = angular_acceleration(inputs, omega, I, L, b, k)
26 tau = torques(inputs, L, b, k);
27 omegaddot = inv(I)
*
(tau - cross(omega, I
*
omega));
28 end
We would also need values for all of our physical constants, a function to compute the rotation
matrix R, and functions to convert from an angular velocity vector to the derivatives of roll,
pitch, and yaw and vice-versa. These are not shown. We can then draw the quadcopter in a
three-dimensional visualization which is updated as the simulation is running.
Quadcopter Simulation. Bars above each propeller represent, roughly, relative thrust
magnitudes.
Control
The purpose of deriving a mathematical model of a quadcopter is to assist in developing con-
trollers for physical quadcopters. The inputs to our system consist of the angular velocities
of each rotor, since all we can control is the voltages across the motors. Note that in our sim-
plied model, we only use the square of the angular velocities,
i
2
, and never the angular
velocity itself,
i
. For notational simplicity, let us introduce the inputs
i
=
i
2
. Since we
can set
i
, we can clearly set
i
as well. With this, we can write our system as a rst order
differential equation in state space. Let x
1
be the position in space of the quadcopter, x
2
be the
quadcopter linear velocity, x
3
be the roll, pitch, and yaw angles, and x
4
be the angular velocity
vector. (Note that all of these are 3-vectors.) With these being our state, we can write the state
7
space equations for the evolution of our state.
x
1
= x
2
x
2
=
_
_
0
0
g
_
_
+
1
m
RT
B
+
1
m
F
D
x
3
=
_
_
1 0 s
0 c
0 s
_
_
1
x
4
x
4
=
_
_
I
xx
1
I
yy
1
I
zz
1
_
_
_
I
yy
I
zz
I
xx
y
z
I
zz
I
xx
I
yy
x
z
I
xx
I
yy
I
zz
x
y
_
_
Note that our inputs are not used in these equations directly. However, as we will see, we can
choose values for and T, and then solve for values of
i
.
PD Control
In order to control the quadcopter, we will use a PD control, with a component proportional
to the error between our desired trajectory and the observed trajectory, and a component pro-
portional to the derivative of the error. Our quadcopter will only have a gyro, so we will only
be able to use the angle derivatives
,
, and
in our controller; these measured values will
give us the derivative of our error, and their integral will provide us with the actual error. We
would like to stabilize the helicopter in a horizontal position, so our desired velocities and
angles will all be zero. Torques are related to our angular velocities by = I
, so we would
like to set the torques proportional to the output of our controller, with = Iu(t). Thus,
_
_
_
_
=
_
_
I
xx
_
K
d
+ K
p
_
T
0
d t
_
I
yy
_
K
d
+ K
p
_
T
0
d t
_
I
zz
_
K
d
+ K
p
_
T
0
d t
_
_
_
We have previously derived the relationship between torque and our inputs, so we know that
B
=
_
_
Lk(
1
3
)
Lk(
2
4
)
b (
1
2
+
3
4
)
_
_
=
_
_
I
xx
_
K
d
+ K
p
_
T
0
d t
_
I
yy
_
K
d
+ K
p
_
T
0
d t
_
I
zz
_
K
d
+ K
p
_
T
0
d t
_
_
_
This gives us a set of three equations with four unknowns. We can constrain this by enforcing
the constraint that our inputs must keep the quadcopter aloft:
T = mg.
Note that this equation ignores the fact that the thrust will not be pointed directly up. This
will limit the applicability of our controller, but should not cause major problems for small
deviations from stability. If we had a way of determining the current angle accurately, we
could compensate. If our gyro is precise enough, we can integrate the values obtained from
the gyro to get the angles and . In this case, we can calculate the thrust necessary to keep
the quadcopter aloft by projecting the thrust mg onto the inertial z axis. We nd that
T
proj
= mg cos cos
Therefore, with a precise angle measurement, we can instead enforce the requirement that the
thrust be equal to
T =
mg
cos cos
8
in which case the component of the thrust pointing along the positive z axis will be equal to
mg. We know that the thrust is proportional to a weighted sum of the inputs:
T =
mg
cos cos
= k
i
=
i
=
mg
k cos cos
With this extra constraint, we have a set of four linear equations with four unknowns
i
. We
can then solve for each
i
, and obtain the following input values:
1
=
mg
4k cos cos
2be
I
xx
+ e
I
zz
kL
4bkL
2
=
mg
4k cos cos
+
e
I
zz
4b
e
I
yy
2kL
3
=
mg
4k cos cos
2be
I
xx
+ e
I
zz
kL
4bkL
4
=
mg
4k cos cos
+
e
I
zz
4b
+
e
I
yy
2kL
This is a complete specication for our PD controller. We can simulate this controller
using our simulation environment. The controller drives the angular velocities and angles to
zero.
Left: Angular velocities. Right: angular displacements. , , are coded as red, green, and
blue.
However, note that the angles are not completely driven to zero. The average steady state
error (error after 10 seconds of simulation) is approximately 0.3
= K
d
+ K
p
_
T
0
d t + K
i
_
T
0
_
T
0
d td t
e
= K
d
+ K
p
_
T
0
d t + K
i
_
T
0
_
T
0
d td t
e
= K
d
+ K
p
_
T
0
d t + K
i
_
T
0
_
T
0
d td t
However, PID controls come with their own shortcomings. One problem that commonly oc-
curs with a PID control is known as integral windup.
11
In some cases, integral wind-up can cause lengthy oscillations instead of settling. In other
cases, wind-up may actually cause the system to become unstable, instead of taking longer to
reach a steady state.
If there is a large disturbance in the process variable, this large disturbance is integrated over
time, becoming a still larger control signal (due to the integral term). However, even once the
system stabilizes, the integral is still large, thus causing the controller to overshoot its target.
It may then begin a series of dieing down oscillations, become unstable, or simply take an
incredibly long time to reach a steady state. In order to avoid this, we disable the integral
function until we reach something close to the steady state. Once we are in a controllable
region near the desired steady state, we turn on the integral function, which pushes the system
towards a low steady-state error.
12
With a properly implemented PID, we achieve an error of approximately 0.06
after 10
seconds.
We have implemented this PID control for use in simulation, in the same way as with the
PD controller shown earlier. Note that there is an additional parameter to tune in a PID. The
disturbances used for all the test cases are identical, shown to compare the controllers.
1 % Compute system inputs and updated state.
2 % Note that input = [
1
, . . .,
4
]
3 function [input, state] = pid_controller(state, thetadot)
4 % Controller gains, tuned by hand and intuition.
5 Kd = 4;
6 Kp = 3;
7 Ki = 5.5;
8
9 % Initialize the integral if necessary.
10 if isfield(state, integral)
11 params.integral = zeros(3, 1);
12 params.integral2 = zeros(3, 1);
13 end
14
15 % Prevent wind-up
16 if max(abs(params.integral2)) > 0.01
17 params.integral2(:) = 0;
18 end
19
20 % Compute total thrust
21 total = state.m
*
state.g / state.k / (cos(state.integral(1))
*
cos(state.integral(2)));
22
23 % Compute errors
13
24 e = Kd
*
thetadot + Kp
*
params.integral - Ki
*
params.integral2;
25
26 % Solve for the inputs,
i
27 input = error2inputs(params, accels, total);
28
29 % Update the state
30 params.integral = params.integral + params.dt .
*
thetadot;
31 params.integral2 = params.integral2 + params.dt .
*
params.integral;
32 end
14
Automatic PID Tuning
Although PID control has the potential to perform very well, it turns out that the quality of the
controller is highly dependent on the gain parameters. Tuning the parameters by hand may
be quite difcult, as the ratios of the parameters is as important as the magnitudes of the pa-
rameters themselves; often, tuning parameters requires detailed knowledge of the system and
an understanding of the conditions in which the PID control will be used. The parameters we
chose previously were tuned by hand for good performance, simply by running simulations
with many possibly disturbances and parameter values, and choosing something that worked
reasonably well. This method is clearly suboptimal, not only because it can be very difcult
and labor-intensive (and sometimes more or less impossible) but also because the resulting
gains are not in any way guaranteed to be optimal or even close to optimal.
Ideally, we would be able to use an algorithm to analyze a system and output the op-
timal PID gains, for some reasonable denition of optimal. This problem has been studied
in depth, and many methods have been proposed. Many of these methods require detailed
knowledge of the system being modeled, and some rely on properties of the system, such as
stability or linearity. The method we will use for choosing our PID parameters is a method
known as extremum seeking.
Extremum seeking works exactly as the name implies. We dene the optimal set of
parameters as some vector
= (K
p
, K
i
, K
d
) which minimizes some cost function J(
). In our
case, we would like to dene a cost function that penalizes high error and error over extended
durations of time. One candidate cost function is given by
J(
) =
1
t
f
t
o
_
t
f
t
0
e(t,
)
2
d t
where e(t,
) is the error in following some reference trajectory with some initial disturbance
using a set of parameters
. Suppose we were able to somehow compute the gradient of
this cost function, J(
(k +1) =
(k) J(
)
where
(k) is the parameter vector after k iterations and is some step size which dictates how
much we adjust our parameter vector at each step of the iteration. As k , the cost function
J(
). By denition,
J(
) =
_
K
p
J(
),
K
i
J(
),
K
d
J(
)
_
.
We know how to compute J(
K
J(
)
J(
+ u
K
) J(
u
K
)
2
where u
K
is the unit vector in the K direction. As 0, this approximation becomes better.
Using this approximation, we can minimize our cost function and achieve locally optimal PID
parameters. We can start with randomly initialized positive weights, disturb the system in
some set manner, evaluate J(