-
Notifications
You must be signed in to change notification settings - Fork 289
Expand file tree
/
Copy pathmain_line_follower.cpp
More file actions
132 lines (107 loc) · 5.47 KB
/
main_line_follower.cpp
File metadata and controls
132 lines (107 loc) · 5.47 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
#include "mbed.h"
// pes board pin map
#include "PESBoardPinMap.h"
// drivers
#include "DebounceIn.h"
#include "DCMotor.h"
#include "LineFollower.h"
#define USE_GEAR_RATIO_78 false // set this to true use gear ratio 78.125, otherwise 100.00 is used
bool do_execute_main_task = false; // this variable will be toggled via the user button (blue button) and
// decides whether to execute the main task or not
bool do_reset_all_once = false; // this variable is used to reset certain variables and objects and
// shows how you can run a code segment only once
// objects for user button (blue button) handling on nucleo board
DebounceIn user_button(BUTTON1); // create DebounceIn to evaluate the user button
void toggle_do_execute_main_fcn(); // custom function which is getting executed when user
// button gets pressed, definition at the end
// main runs as an own thread
int main()
{
// attach button fall function address to user button object
user_button.fall(&toggle_do_execute_main_fcn);
// while loop gets executed every main_task_period_ms milliseconds, this is a
// simple approach to repeatedly execute main
const int main_task_period_ms = 20; // define main task period time in ms e.g. 20 ms, therefore
// the main task will run 50 times per second
Timer main_task_timer; // create Timer object which we use to run the main task
// every main_task_period_ms
// led on nucleo board
DigitalOut user_led(LED1);
// --- adding variables and objects and applying functions starts here ---
// create object to enable power electronics for the dc motors
DigitalOut enable_motors(PB_ENABLE_DCMOTORS);
const float voltage_max = 12.0f; // maximum voltage of battery packs, adjust this to
// 6.0f V if you only use one battery pack
#if USE_GEAR_RATIO_78
// https://www.pololu.com/product/3477/specs
const float gear_ratio = 78.125f;
const float kn = 180.0f / 12.0f;
#else
// https://www.pololu.com/product/3490/specs
const float gear_ratio = 100.00f;
const float kn = 140.0f / 12.0f;
#endif
// motor M1 and M2, do NOT enable motion planner when used with the LineFollower (disabled per default)
DCMotor motor_M1(PB_PWM_M1, PB_ENC_A_M1, PB_ENC_B_M1, gear_ratio, kn, voltage_max);
DCMotor motor_M2(PB_PWM_M2, PB_ENC_A_M2, PB_ENC_B_M2, gear_ratio, kn, voltage_max);
#if USE_GEAR_RATIO_78
const float d_wheel = 0.035f; // wheel diameter in meters
const float b_wheel = 0.1518f; // wheelbase, distance from wheel to wheel in meters
const float bar_dist = 0.118f; // distance from wheel axis to leds on sensor bar / array in meters
#else
const float d_wheel = 0.0372f; // wheel diameter in meters
const float b_wheel = 0.156f; // wheelbase, distance from wheel to wheel in meters
const float bar_dist = 0.114f; // distance from wheel axis to leds on sensor bar / array in meters
#endif
// line follower, tune max. vel rps to your needs
LineFollower lineFollower(PB_9, PB_8, bar_dist, d_wheel, b_wheel, motor_M2.getMaxPhysicalVelocity());
// nonlinear controller gains, tune to your needs
#if USE_GEAR_RATIO_78
const float Kp = 1.0f * 2.0f;
const float Kp_nl = 1.0f * 17.0f;
#else
const float Kp = 1.2f * 2.0f;
const float Kp_nl = 1.2f * 17.0f;
#endif
lineFollower.setRotationalVelocityControllerGains(Kp, Kp_nl);
// start timer
main_task_timer.start();
// this loop will run forever
while (true) {
main_task_timer.reset();
// --- code that runs every cycle at the start goes here ---
if (do_execute_main_task) {
// --- code that runs when the blue button was pressed goes here ---
// visual feedback that the main task is executed, setting this once would actually be enough
enable_motors = 1;
// setpoints for the dc motors in rps
motor_M1.setVelocity(lineFollower.getRightWheelVelocity()); // set a desired speed for speed controlled dc motors M1
motor_M2.setVelocity(lineFollower.getLeftWheelVelocity()); // set a desired speed for speed controlled dc motors M2
} else {
// the following code block gets executed only once
if (do_reset_all_once) {
do_reset_all_once = false;
// --- variables and objects that should be reset go here ---
// reset variables and objects
enable_motors = 0;
}
}
// toggling the user led
user_led = !user_led;
// --- code that runs every cycle at the end goes here ---
// read timer and make the main thread sleep for the remaining time span (non blocking)
int main_task_elapsed_time_ms = duration_cast<milliseconds>(main_task_timer.elapsed_time()).count();
if (main_task_period_ms - main_task_elapsed_time_ms < 0)
printf("Warning: Main task took longer than main_task_period_ms\n");
else
thread_sleep_for(main_task_period_ms - main_task_elapsed_time_ms);
}
}
void toggle_do_execute_main_fcn()
{
// toggle do_execute_main_task if the button was pressed
do_execute_main_task = !do_execute_main_task;
// set do_reset_all_once to true if do_execute_main_task changed from false to true
if (do_execute_main_task)
do_reset_all_once = true;
}