The Rotary Inverted Pendulum, otherwise known as a Furuta pendulum, is a variation on a classic problem in the area of control system, the pendulum on a cart. Instead of being attached to a moving cart constrained to move linearly in only one dimension, the Furuta pendulum’s “cart” is a rotary object. The Furuta pendulum is an example of a nonlinear system, which also makes it more difficult to maintain upright. It is used to verify the performance of control algorithm techniques, of which can later be used to control more complicated systems, such as an aircraft flight. For an example of a Furuta pendulum, you can see a video of it here.
We decided to make our system primarily out of aluminum instead of 3D-printing all of the parts.
The primary reason for this was that we wanted a very smooth rotation at the point of contact of the pendulum,
allowing it to spin freely. Therefore, we decided to use a ball bearing because rolling friction is
lower than sliding friction. The system would also be more robust if made of metal, and it would be easier to
attach with the shaft of the stepper motor. We purchased aluminum bars and rods from McMaster-Carr.
We used aluminum because it is easier to machine aluminum than it is to machine steel, and our application
wasn’t subject to any high stresses that aluminum could not handle.
The relevant parts were machined using a mill and a lathe. For assembling the pieces together,
various methods were used. The motor shaft connects with the pendulum assembly through a shaft coupler
and two set screws because the two rods are different sizes and need to spin together.
This shaft cwas threaded into the aluminum disc and held in place with two nuts at the other end.
The pendulum has a ball bearing press fitted into it, and is connected to a shaft that is threaded onto the
aluminum disc. If we were to do it over, we wouldn’t thread this piece because it had a tendency to slowly
unscrew itself from the momentum of the pendulum. Instead, we would use a set screw to hold the shaft in place.
We decided to laser-cut a box to hold the breadboard with the motor driver and excess wires.
This allowed us to not have to use any screws or glue because of the tighter tolerances allowed by laser cutting.
We made our box out of both ¼’’ acrylic and maple wood: we chose these materials because they could be easily cut at Cornell RPL.
Originally, we planned on having the motor sit on top of the box to provide enough height for the pendulum to
swing around, but the vibrations from the stepper motor required us to have the motor clamped or fixed in some way.
We later cut out holes on the top of the box so that we could screw on the motor from the inside of the box.
Ideally, long enough screws of the desired width should be used so that the motor sits on top of the box instead
of dangling from the top.
The pendulum, if undisturbed, would hang vertically downward because of the force of gravity.
Therefore to get the pendulum to pointing vertically upward, we would need to apply a control input through
controlling the motor to swing the pendulum up. This analysis was done using MATLAB, but it could have also
been done using the SymPy and python-control libraries on Python.
We began by first deriving the dynamics of the system. We found a paper online
(Benjamin Seth Cazzolato and Zebb Prime 2011, “On the Dynamics
of the Furuta Pendulum”). The paper made a few assumptions to the dynamics that wasn’t acceptable for our system
and we wanted to understand the system better, so we decided to derive it ourselves.
We decided to use the Lagrangian approach towards finding the dynamics, which involves finding the position and
velocity of the pendulum. We assumed there was no damping in the system, and the only force acting on the system is gravity.
We decided to use the Lagrangian approach towards finding the dynamics, which involves finding the position
and velocity of the pendulum, essentially doing the same steps as the paper. We assumed there was no damping
in the system, and the only force acting on the system is gravity and the supplied torque. Our state are the
two angles and angular velocities: one of the spinning motor and one of the pendulum.
We then linearized our system about our desired equilibrium point, corresponding to theta = 0, by taking the
Jacobian of the A and B matrices with respect to the state variables. This linearized form is only accurate
near the equilibrium points; if the pendulum bar deviated too much away from the upright position, the dynamics
would be inaccurate in describing the motion.
The system is highly dependent on the parameters of the system. We weighed each piece of the system,
measured their lengths, estimated their distance from the rotation point to the center of mass, and
calculated the moments of inertia theoretically based on their geometries and weights.
When finding the moment of inertia of each piece of the assembly, we made a couple of simplifications.
One was that we ignored the small differences from the theoretical moment of inertia from the actual
that was the result of the holes drilled into the aluminum pieces. They were very small relative to the
mass of the disc and pendulum that thus negligible. Also, we assumed that the moment of inertia for the
pendulum in the y and z direction were the same. This is because the width and depth of the pendulum are
very similar and much smaller than the length of the pendulum.
Once the dynamics of the furuta pendulum was derived, we could begin controlling it. We began by first looking at the root locus of the Furuta pendulum without any control:
We could see that the system was unstable because there was a pole in the right half plane of the root locus.
This was as expected: the pendulum without any control was not going to be stable.
We also compared this root locus with that of other papers on the Furuta pendulum to act as another
check that our derivation of the dynamics was correct.
Using MATLAB’s sisotool (one of MATLAB’s many god-tier functions), we could look at how the properties
of the system changed as we adjusted the controller. In order to get the root locus to be stable, we
had to add a pole to the right of the rightmost zero: this would allow the right half plane pole to
have a chance of entering the left half plane. Because we had decided to implement the system as a
PID controller in the python code, the zero would occur at s = 0. Afterward, we added two zeros in
the left hand plane to pull the right hand side poles left as the gain on the system increases.
We picked the two zeros with two constraints in mind: we tried to minimize the overshoot because our
refresh rate of our system was not very high, and we also tried to minimize the gain required because
there were limitations on our system on the speed and torque that our motor could provide.
The designed controller can be seen below:
To make our system more realistic, we added a time delay because it doesn’t respond instantaneously:
this was done by adding a Padé approximation to the feedback loop of the system. The root locus of this
system can be seen:
Just like before, sisotool was used to design a controller. The resulting controller is shown:
A block diagram of our system is shown below:
An IMU, or inertial measurement unit, was needed because we have to know the position of the pendulum at all times.
The reason we decided to use an IMU over a rotary encoder was because it was readily available, which is a choice
that we elaborate more on in the ‘Results’ section.
Initially, we tried using Sparkfun MPU6050, a 6 degree
of freedom IMU (3-axis accelerometer, 3-axis gyroscope). However, while using the sensor for a few weeks, we
quickly realized that without an external filter, its accuracy would decrease over time. This would be a big
problem in a controls-related project, as it would be difficult to maintain the pendulum upright if we can’t
read proper angular position data.
Thankfully, we ended up getting our hands on the
Adafruit BNO055,
which is a 9 degree of freedom (3-axis accelerometer, 3-axis gyroscope, 3-axis magnetometer) sensor that has its own
CircuitPython library. This makes the IMU much easier for us to use, as we don’t have to read from defined registers
but rather call existing functions from the library. Additionally, the IMU implements its own sensor fusion algorithms,
which made our life much more simpler. However, we did have to account for the fact that it has a slow sampling rate (100 Hz).
Because we have to know the position of the motor at all times, we were deciding between using a DC motor
with an encoder or a stepper motor. Stepper motors are DC motors that move in discrete steps. They can either
be unipolar, which can reverse current without additional circuitry with its common lead that predetermines its
polarity, or bipolar, which needs current to be reversed to reverse the polarity of the poles of the stator.
Although a DC motor performs better than a stepper motor in terms of torque, it is more expensive; also, we
already had a stepper motor and a driver (Sparkfun Big Easy Driver),
so we ultimately ended up using those components. We need a driver with a stepper motor because the principle of stepper
motors is the switching of its stator poles at logic HIGH and LOW values in a way so that it takes steps.
To do this, we need a way for the Raspberry Pi to develop such logic to control the stepper motor properly,
and that’s where the motor driver comes into play.
The motor that we chose to use has 400 steps per revolution, which means that each step angle is 0.9 degrees.
This fine level of control was desirable for our application. We needed to be able to control the motor at
different rotation rates. but we encountered a lot of vibrations as the shaft was rotating, which was the
result of the delay between inputs being too long. To combat this, we used microstepping so that the motor can
spin at a slower rate.
As mentioned in the ‘Hardware’ section, our sensor outputs data at 100 Hz, or every 10 ms -- this is too slow for us. Doing everything (reading sensor output, feeding into our controller, controlling our motor properly) serially would take way too long, and we wanted to be able to continuously read sensor data regardless of what other tasks we would be doing.
While deciding between threading and multiprocessing, we took into account that we would have to continuously read sensor data and analyze it, so we wanted a module that would allow various functions to share objects within the same memory space. Because only one thread writes to a global list that stores sensor data and we do not have multiple threads writing to the same memory concurrently, we did not have to worry about using locks.
Because the controller updates more frequently than that of the IMU, we needed to ensure that the controller only updates after the IMU has a new reading. Otherwise, the controller would be outputting a much stronger torque than necessary, potentially throwing the system unstable. To do so, we made the controller check if the IMU had already taken in a new input. Additionally, we had to make sure that when there is a new output from the controller, the motor abandons whatever it was doing beforehand as to make sure to minimize delays.
Along with the slow sampling rate, another issue we found with the IMU was the fact that it would output completely off data at random times. We also found that as the IMU moved a lot more, its accuracy would decrease significantly. This is something that we had to account for within our software. Because the gyroscope data especially had a tendency to read unusual data and drift over time, we kept a running average. We also should’ve applied an additional Kalman Filter on the output of the IMU data so that we can have a better estimate of the pendulum position in the times between sensor readings. This would have been especially helpful in allowing us to update the control input more frequently and thus be better at stabilizing the pendulum.
As mentioned briefly above, we ran into some issues with broken parts, which halted our progress significantly. Not realizing we were using broken motor drivers that didn’t limit current to the stepper motor was a major time loss for us, because we spent a lot of time trying to figure out why the shaft would not rotate by the desired angle.
We needed to attach the IMU to the end of the pendulum so that we could measure the angular velocity and position of the end of the pendulum. This meant that we needed to run a wire down the pendulum. To prevent the wire from tangling or affecting the pendulum motion as it spun, we added a lot of slack to the wire. An alternative to this would’ve been instead to use an encoder to calculate the rotation rate of the pendulum; the encoder would be able to sit on the aluminum disc and a long wire would not need to go to the pendulum.
As mentioned briefly above, we ran into some issues with broken parts, which halted our progress significantly. Not realizing we We had originally tried to just 3D print a couple of pieces in our pendulum system, but found that the material wasn’t adequate. Specifically, we had tried to 3D print a shaft coupling to connect the motor and the aluminum disc. It was designed so that there could be screws that would attach and tighten the plastic so that it wouldn’t slip when the motor rotated. However, even though we tightened it, the plastic would creep, and the aluminum would slip if the motor tried to change its speed very quickly. Therefore, we had to machine the shaft coupler and use set screws to hold the two ends in place.
We faced many challenges with the stepper motor, and we think that this component was the primary reason why our pendulum wasn’t able to self balance. One issue we faced was that we didn’t realize that the first motor driver we used (DRV8825) was not current limiting because the potentiometer was broken, which meant that the motor became extremely hot and damaged. Around the third week of this project, the motor did not respond to our inputs, so we decided to get another one. Luckily, we were able to borrow a friend’s stepper motor and driver.
We initially were hesitant about using a stepper motor, but we believed it could work because if we used microstepping with a very short delay time, we could achieve a reasonable speed and precision for our motors. However, it wasn’t until when we implemented our controller that we realized that the system just could not work with a stepper motor. A stepper motor is usually chosen for its precision of steps, not for its torque. Because the control inputs to our system was in torques, we needed the motor to be able to vary its rotation rate very quickly, because torque is equal to the moment of inertia times angular acceleration.
Additionally, balancing an inverted pendulum requires great precision. Therefore, we wanted to characterize our motor by trying to find a relationship between the speed it outputted and the delay used for the motor to take a step. We set up and ran a test in which we varied the delays between quarter stepping and full stepping to see the bound on the angular velocity of the motor.. We measured the angular velocity of the stepper motor by spinning the motor for a known angle and dividing that number by the time it took to reach it. On one end, if the time delay was too small, the motor did not move the desired number of steps; on the other end, if the time delay was too long, the motor would vibrate too much. The results of this are detailed in the following table. Do note that the slowest angular velocity of the full step regime still had a fair bit of vibration, which was not ideal.
The results of the test acted as a lookup table in our code. Given our desired motor torques, we would find the corresponding velocities that would generate the given torque. If the value desired is between two angular velocities of a given stepper step size, we would find a linear interpolation between the two end values. This isn't quite correct, but it should be a good enough approximation. If the value desired is greater than the maximum allowable angular velocity, we would set it to the minimum delay possible in the full-step mode so that the motor could move at that maximum speed. However, in spite of our efforts, there is still some uncertainty in the measurements because we did not have an encoder.
Despite our best efforts, the control of the stepper motor was too abrupt, making control of the system very difficult if not impossible. Although we tried our best to quantify the behavior of the motor to get it to behave as we wanted, the motor wasn’t precise enough when we wanted it to be; the difference between stepper motor states was too sudden. Although at short bursts the motor could move very quickly, it couldn’t sustain the acceleration. Also, there wasn’t a smooth continuous transition between the maximum speed of the motor running at half step and the minimum speed of the motor running at full step, so the motor was very jerky at times. Having a maximum bound on the motor also didn’t help: if the pendulum tipped over too much, it wouldn’t be able to recover. Because of the slow update of the IMU, this was especially problematic. Specifically, if the pendulum swung even 20-30 degrees past the desired equilibrium point, the pendulum would fall over.
We accomplished a lot of tasks for this project. On the hardware and mechanical side, we were able to read accurate angle and angular rate readings for the pendulum and validate those readings through testing. We also, to the best of our abilities, characterized the stepper motor, machined the pendulum assembly, and designed housing to store all of our electrical components. On the software side, we were able to deal with the slow data output rate of the IMU by using multithreading and having that process continuously run in the background. Finally, on the controls side, we modeled our system on Matlab and designed a PID controller that gave us the proper output.
However, our major issue came with the stepper motor, which could not move accurately enough at a desired angular velocity. Additionally, choosing an IMU over an encoder made controlling the pendulum harder because the IMU samples slowly and the data is not as real time as it needs to be. The extra wires coming from the IMU ended up becoming an inconvenience as well and sometimes provided external force that we couldn't account for.
Overall, we definitely learned a lot throughout the entire process, from choosing the right hardware to writing clean and efficient code. We enjoyed the late nights and definitely appreciated a lot of technical advice from the TAs and Prof. Skovira.
Aluminum Bar 3/16'' x 1'' x 2' (8975K586) | $4.48 |
Aluminum Rod 9/16'' diam x 2ft (8974K46) | $5.59 |
Aluminum Rod 3'' diam x 1'' (1610T19) | $8.61 |
Ball bearings | $2.00 |
Big Easy Driver | $19.95 |
Wantai Stepper Motor | $13.99 |
BNO055 | $33.07 |
$87.69 |
We thank Professor Skovira and ECE 5725 teaching assistants, all of whom provided guidance and encouragement when we ran into problems, as well as providing some electronic parts (IMU). We would also like to thank Tanishq Aggarwal, who lent us a motor driver and stepper motor. We would also like to thank Monika Bandi for being super exceptional for our laser cutting needs, and Emma Sung for being a good pal in the machine shop.
The complete Python scripts, along with the derivations & controller designing scripts in Matlab, can be found on our Github. Specifically, the files are as follows:
Hardware, Software, Website
Mechanical Design, Controls, Hardware, Software