40 #include <visp3/core/vpConfig.h>
42 #if defined(VISP_HAVE_POLOLU) && defined(VISP_HAVE_THREADS)
49 #include <visp3/core/vpMath.h>
50 #include <visp3/robot/vpPololu.h>
52 #ifdef ENABLE_VISP_NAMESPACE
56 void usage(
const char **argv,
int error,
const std::string &device,
int baudrate,
int channel,
57 unsigned short pwm_min,
unsigned short pwm_max,
float angle_min,
float angle_max)
59 std::cout <<
"Synopsis" << std::endl
60 <<
" " << argv[0] <<
" [--device <name>] [--baud <rate>] [--channel <number>] [--calibrate]"
61 <<
" [--range-pwm <min max> ] [--verbose, -v] [--help, -h]" << std::endl
63 std::cout <<
"Description" << std::endl
64 <<
" --device <name> Device name." << std::endl
65 <<
" Default: " << device << std::endl
67 <<
" --baud <rate> Serial link baud rate." << std::endl
68 <<
" Default: " << baudrate << std::endl
70 <<
" --channel <number> Channel to dial with." << std::endl
71 <<
" Default: " << channel << std::endl
73 <<
" --range-pwm <min max> Set PWM min and max values." << std::endl
74 <<
" You can use \"--calibrate\" to retrieve min and max pwm values."
75 <<
" Default: " << pwm_min <<
" " << pwm_max << std::endl
77 <<
" --range-angles <min max> Set angle min and max values (deg)." << std::endl
80 <<
" --verbose, -v Enable verbosity." << std::endl
82 <<
" --calibrate Start pwm calibration determining min and max admissible values." << std::endl
83 <<
" Once calibration done you can use \"--range-pwm <min max>\" option to set" << std::endl
84 <<
" the corresponding values" << std::endl
86 <<
" --help, -h Print this helper message." << std::endl
89 std::cout <<
"Error" << std::endl
91 <<
"Unsupported parameter " << argv[error] << std::endl;
95 int main(
int argc,
const char **argv)
98 std::string opt_device =
"COM4";
100 std::string opt_device =
"/dev/ttyACM0";
105 int opt_baudrate = 38400;
106 bool opt_verbose =
false;
107 bool opt_calibrate =
false;
108 unsigned short opt_pwm_min = 4000;
109 unsigned short opt_pwm_max = 8000;
110 float opt_angle_min =
static_cast<float>(
vpMath::rad(-45));
111 float opt_angle_max =
static_cast<float>(
vpMath::rad(45));
112 float opt_positioning_velocity =
static_cast<float>(
vpMath::rad(10));
113 float last_angle = 0;
116 for (
int i = 1; i < argc; i++) {
117 if (std::string(argv[i]) ==
"--device" && i + 1 < argc) {
118 opt_device = std::string(argv[i + 1]);
121 else if (std::string(argv[i]) ==
"--baud" && i + 1 < argc) {
122 opt_baudrate = std::atoi(argv[i + 1]);
125 else if (std::string(argv[i]) ==
"--channel" && i + 1 < argc) {
126 opt_channel = std::atoi(argv[i + 1]);
129 else if (std::string(argv[i]) ==
"--range-pwm" && i + 2 < argc) {
130 opt_pwm_min =
static_cast<unsigned short>(
vpMath::rad(std::atoi(argv[i + 1])));
131 opt_pwm_max =
static_cast<unsigned short>(
vpMath::rad(std::atoi(argv[i + 2])));
134 else if (std::string(argv[i]) ==
"--range-angles" && i + 2 < argc) {
135 opt_angle_min =
static_cast<float>(std::atof(argv[i + 1]));
136 opt_angle_max =
static_cast<float>(std::atof(argv[i + 2]));
139 else if (std::string(argv[i]) ==
"--calibrate") {
140 opt_calibrate =
true;
142 else if (std::string(argv[i]) ==
"--verbose" || std::string(argv[i]) ==
"-v") {
145 else if (std::string(argv[i]) ==
"--help" || std::string(argv[i]) ==
"-h") {
146 usage(argv, 0, opt_device, opt_baudrate, opt_channel, opt_pwm_min, opt_pwm_max, opt_angle_min, opt_angle_max);
150 usage(argv, i, opt_device, opt_baudrate, opt_channel, opt_pwm_min, opt_pwm_max, opt_angle_min, opt_angle_max);
157 vpPololu servo(opt_device, opt_baudrate, opt_channel, opt_verbose);
159 std::cout <<
"Pololu board is " << (servo.connected() ?
"connected" :
"disconnected") << std::endl;
162 std::cout <<
"Proceed to calibration to determine pwm min and max values..." << std::endl;
163 std::cout <<
"WARNING: Calibration will move the servo at channel " << opt_channel <<
"!" << std::endl;
164 std::cout <<
"Press Enter to move to min and max pwm positions..." << std::endl;
167 unsigned short pwm_min, pwm_max;
168 servo.calibrate(pwm_min, pwm_max);
169 std::cout <<
"Servo on channel " << opt_channel <<
" has pwm range [" << pwm_min <<
", " << pwm_max <<
"]" << std::endl;
173 servo.setPwmRange(opt_pwm_min, opt_pwm_max);
174 servo.setAngularRange(opt_angle_min, opt_angle_max);
177 servo.getRangePwm(opt_pwm_min, opt_pwm_max);
178 std::cout <<
"Position range (pwm): " << opt_pwm_min <<
" " << opt_pwm_max << std::endl;
179 servo.getRangeAngles(opt_angle_min, opt_angle_max);
180 std::cout <<
"Position range (deg): " <<
vpMath::deg(opt_angle_min) <<
" " <<
vpMath::deg(opt_angle_max) << std::endl;
183 std::cout <<
"Move to min position (pwm): " << opt_pwm_min <<
" at max velocity" << std::endl;
184 servo.setPwmPosition(opt_pwm_min, 0);
185 std::this_thread::sleep_for(std::chrono::seconds(3));
186 std::cout <<
"Servo reached position (pwm): " << servo.getPwmPosition() << std::endl;
188 std::cout <<
"Move to max position (pwm): " << opt_pwm_max <<
" at max velocity" << std::endl;
189 servo.setPwmPosition(opt_pwm_max, 0);
190 std::this_thread::sleep_for(std::chrono::seconds(3));
191 std::cout <<
"Servo reached position (pwm): " << servo.getPwmPosition() << std::endl;
194 std::cout <<
"Move to min position (deg): " <<
vpMath::deg(opt_angle_min) <<
" at max velocity" << std::endl;
195 servo.setAngularPosition(opt_angle_min, 0);
196 std::this_thread::sleep_for(std::chrono::seconds(3));
197 std::cout <<
"Servo reached position (deg): " <<
vpMath::deg(servo.getAngularPosition()) << std::endl;
199 std::cout <<
"Move to max position (deg): " <<
vpMath::deg(opt_angle_max) <<
" at max velocity" << std::endl;
200 servo.setAngularPosition(opt_angle_max, 0);
201 std::this_thread::sleep_for(std::chrono::seconds(3));
202 std::cout <<
"Servo reached position (deg): " <<
vpMath::deg(servo.getAngularPosition()) << std::endl;
205 std::cout <<
"Move to zero position (deg): " <<
vpMath::deg(0) <<
" at max velocity" << std::endl;
206 servo.setAngularPosition(0, 0);
207 std::this_thread::sleep_for(std::chrono::seconds(3));
208 last_angle = servo.getAngularPosition();
209 std::cout <<
"Servo reached position (deg): " <<
vpMath::deg(last_angle) << std::endl;
212 std::cout <<
"Move to min position (deg): " <<
vpMath::deg(opt_angle_min) <<
" at " <<
vpMath::deg(opt_positioning_velocity) <<
" deg/s" << std::endl;
213 servo.setAngularPosition(opt_angle_min, opt_positioning_velocity);
215 time_s =
static_cast<int>(std::abs((opt_angle_min - last_angle) / opt_positioning_velocity) + 2);
217 std::this_thread::sleep_for(std::chrono::seconds(time_s));
218 last_angle = servo.getAngularPosition();
219 std::cout <<
"Servo reached position (deg): " <<
vpMath::deg(last_angle) << std::endl;
221 std::cout <<
"Move to max position (deg): " <<
vpMath::deg(opt_angle_max) <<
" at " <<
vpMath::deg(opt_positioning_velocity) <<
" deg/s" << std::endl;
222 servo.setAngularPosition(opt_angle_max, opt_positioning_velocity);
224 time_s =
static_cast<int>(std::abs((opt_angle_max - last_angle) / opt_positioning_velocity) + 2);
225 std::this_thread::sleep_for(std::chrono::seconds(time_s));
226 last_angle = servo.getAngularPosition();
227 std::cout <<
"Servo reached position (deg): " <<
vpMath::deg(last_angle) << std::endl;
240 std::cout <<
"ViSP doesn't support Pololu 3rd party library" << std::endl;
error that can be emitted by ViSP classes.
const char * getMessage() const
static double rad(double deg)
static double deg(double rad)
Interface for the Pololu Maestro USB Servo Controllers.