-
Notifications
You must be signed in to change notification settings - Fork 145
/
main_ws1.cpp
109 lines (86 loc) · 4.21 KB
/
main_ws1.cpp
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
#include "mbed.h"
// pes board pin map
#include "pm2_drivers/PESBoardPinMap.h"
// drivers
#include "pm2_drivers/DebounceIn.h"
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(USER_BUTTON); // create DebounceIn object to evaluate the user button
// falling and rising edge
void toggle_do_execute_main_fcn(); // custom function which is getting executed when user
// button gets pressed, definition below
// function declaration, definition at the end
float ir_sensor_compensation(float ir_distance_mV);
// main runs as an own thread
int main()
{
// attach button fall function address to user button object, button has a pull-up resistor
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, there for
// 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(USER_LED);
// additional led
// create DigitalOut object to command extra led, you need to add an aditional resistor, e.g. 220...500 Ohm
// a led has an anode (+) and a cathode (-), the cathode needs to be connected to ground via a resistor
DigitalOut led1(PB_9);
// ir distance sensor
float ir_distance_mV = 0.0f; // define a variable to store measurement (in mV)
float ir_distance_cm = 0.0f;
AnalogIn ir_analog_in(PC_2); // create AnalogIn object to read in the infrared distance sensor
// 0...3.3V are mapped to 0...1
// start timer
main_task_timer.start();
// this loop will run forever
while (true) {
main_task_timer.reset();
if (do_execute_main_task) {
// visual feedback that the main task is executed, setting this once would actually be enough
led1 = 1;
// read analog input
ir_distance_mV = 1.0e3f * ir_analog_in.read() * 3.3f;
ir_distance_cm = ir_sensor_compensation(ir_distance_mV);
} else {
// the following code block gets executed only once
if (do_reset_all_once) {
do_reset_all_once = false;
// reset variables and objects
led1 = 0;
ir_distance_mV = 0.0f;
ir_distance_cm = 0.0f;
}
}
// toggling the user led
user_led = !user_led;
// print to the serial terminal
printf("IR distance mV: %f IR distance cm: %f \n", ir_distance_mV, ir_distance_cm);
// read timer and make the main thread sleep for the remaining time span (non blocking)
int main_task_elapsed_time_ms = std::chrono::duration_cast<std::chrono::milliseconds>(main_task_timer.elapsed_time()).count();
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;
}
float ir_sensor_compensation(float ir_distance_mV)
{
// insert values that you got from the MATLAB file
static const float a = 2.574e+04f;
static const float b = -29.37f;
// avoid division by zero by adding a small value to the denominator
if (ir_distance_mV + b == 0.0f)
ir_distance_mV -= 0.001f;
return a / (ir_distance_mV + b);
}