Compare commits

...

7 Commits

Author SHA1 Message Date
EmaMaker e70f3fe1e5 sim: nonlinear simulation
Demonstrates both the analytical solution and the current PID values work
2024-01-13 13:15:52 +01:00
EmaMaker 83f28d04a2 sim: let matlab derive transfer function
and recalibrate pid with that
2024-01-13 13:15:29 +01:00
EmaMaker ca0207932d update .gitignore 2024-01-13 13:14:54 +01:00
EmaMaker 7b7c09714d [UNSURE] open-loop wheel sync 2024-01-12 13:43:05 +01:00
EmaMaker b5798d2eb3 Revert "pid: eyeballed for a fully charged battery"
This reverts commit 23e63c7625.
2024-01-12 13:34:05 +01:00
EmaMaker 9f02ddf349 update README.md 2024-01-05 15:58:03 +01:00
EmaMaker 23e63c7625 pid: eyeballed for a fully charged battery
Made a video showing the result. Apparently the center of mass is not exactly at the center of the
robot and the battery needs to stick out a bit at the front to rebalance it.

Also my desk is a bit tilted so the pitch is heavily affected by the yaw, the video shows it
clearly. Next step: a inner control loop for the yaw.
2024-01-05 14:39:14 +01:00
22 changed files with 39 additions and 10 deletions

3
.gitignore vendored
View File

@ -1,2 +1,5 @@
matlab-sim/slprj/
matlab-sim/*.slxc
matlab-sim-v2/slprj/
matlab-sim-v2/*.slxc
**/*.autosave

View File

@ -4,15 +4,17 @@ My attempt at building a two-wheeled self-balancing robot. The project started a
later evolved as a University project. I'll upload the university report to this repo as soon as it is complete. This repo will also be
supported by a blog post on [emamaker.com](https://emamaker.com)
The physical model of the robot has been derived by calculatin the Lagrangian function. This model
is heavily non-linear and has been linearized around the 0° point (equilibrium).
The physical model of the robot has been derived by calculating the Lagrangian function. This model
is heavily non-linear and has been linearized around the 0° point of equilibrium.
The linearized model has been reconstructed in MatLab Simulink and a discrete-time PID controller
has been calculated by trial and error.
has been calculated by trial and error. This PID Controller was used to dimension the motors (max. torque
in particular) and was later refined with the Ziegler-Nichols method.
The robot has later been built using a Raspberry Pi Pico microcontroller paired with a MPU6050
6-axis IMU, from which the pitch angle is derived with the use of a [Madgwick](https://ahrs.readthedocs.io/en/latest/filters/madgwick.html) filter.
(Initially this was a simple complementary filter, but it was way to noise, check the blog post).
(Initially this was a simple complementary filter, but it was way too subject to noise from the
accelerometer, check the blog post).
The two motors (Hitec HS-322HD servomotors, check blog post for the choice
of motors) are driven by a L298N motor driver. Everything is powered by a 2S 7.4V LiPo battery.

View File

@ -16,10 +16,10 @@ Pc = 0.083
PID Kp = 0.6Kc, Ti = 0.5Pc Td = 0.125Pc
Ki = KpTc/ti
Kd = KpTd/Tc
*/
*/
constexpr double KP = 6;
constexpr double KP = 5.8;
constexpr double KI = 0.2;
constexpr double KD = 0.1;
@ -30,6 +30,12 @@ double input = 0;
double roll{ 0 }, pitch{ 0 }, yaw{ 0 };
constexpr double max_torque_sx = 0.392;
constexpr double max_torque_dx = 0.36;
constexpr double max_torque = max_torque_dx+max_torque_sx;
constexpr double torque_sx_coeff = max_torque_sx/max_torque;
constexpr double torque_dx_coeff = max_torque_dx/max_torque;
void setup(void) {
Serial.begin(9600);
delay(500);
@ -47,7 +53,7 @@ void setup(void) {
}
myController.begin(&input, &output, &setpoint, KP, KI, KD, P_ON_E, FORWARD);
myController.setOutputLimits(-0.72, 0.72); // double of max torque motors can exhert
myController.setOutputLimits(-max_torque, max_torque); // double of max torque motors can exhert
// myController.setWindUpLimits(-0.2 , 0.02);
myController.setSampleTime(1);
myController.start();
@ -71,11 +77,11 @@ void compute(){
myController.compute();
double torque = abs(output) * 0.5;
double torque = abs(output);
double s = sign(output);
double pwm_dx = torque <= 0.005 ? 30*s : torque_to_pwm_dx(torque)*s;
double pwm_sx = torque <= 0.005 ? 30*s : torque_to_pwm_sx(torque)*s;
double pwm_dx = torque <= 0.005 ? 30*s : torque_to_pwm_dx(torque*torque_dx_coeff)*s;
double pwm_sx = torque <= 0.005 ? 30*s : torque_to_pwm_sx(torque*torque_sx_coeff)*s;
move_pwm(MOT_SX, pwm_sx);
move_pwm(MOT_DX, pwm_dx);

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

Binary file not shown.

View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<MF0 version="1.1" packageUris="http://schema.mathworks.com/mf0/SlCache/19700101">
<slcache.FileAttributes type="slcache.FileAttributes" uuid="a895ae6d-62f1-41eb-8898-3e45d1f4e90e">
<checksum>Lb/jesGfhHHIDu90dapcipftUCY9o9NhkvhP/XJ3BQXuHr3IUkBcW/fPUAxXIdZWqCUuUKCUuY5W8wtaxZ2B+w==</checksum>
</slcache.FileAttributes>
</MF0>

View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<MF0 version="1.1" packageUris="http://schema.mathworks.com/mf0/SlCache/19700101">
<slcache.FileAttributes type="slcache.FileAttributes" uuid="81d5dcd0-043c-4202-b2eb-282248cfd3e7">
<checksum>dve/KW2AkD1hUjt0JOZll5FNSnvs4lU9s9KvUWTxKelL6/0n4gjtZ5QDJHRYboTA5D96TchHng97Avmp6nil6w==</checksum>
</slcache.FileAttributes>
</MF0>

View File

@ -0,0 +1,6 @@
<?xml version="1.0" encoding="UTF-8"?>
<MF0 version="1.1" packageUris="http://schema.mathworks.com/mf0/SlCache/19700101">
<slcache.FileAttributes type="slcache.FileAttributes" uuid="78aa8f73-1e99-4ad7-baec-c27c8b664994">
<checksum>z9UdsWR5MLrsZnhLZ6JnGXqVn0khIn4P8lHVs5tHu/tzURqRLyUXB9/aOwDYj3nVU1QjOwCshC7ZzrwihHH9Uw==</checksum>
</slcache.FileAttributes>
</MF0>