39 #include <visp3/core/vpConfig.h>
41 #if defined(VISP_HAVE_POLOLU) && defined(VISP_HAVE_THREADS)
48 #include <visp3/core/vpMath.h>
49 #include <visp3/robot/vpPololu.h>
51 #ifdef ENABLE_VISP_NAMESPACE
55 void usage(
const char **argv,
int error,
const std::string &device,
int baudrate,
int channel,
56 unsigned short pwm_min,
unsigned short pwm_max,
float angle_min,
float angle_max)
58 std::cout <<
"Synopsis" << std::endl
59 <<
" " << argv[0] <<
" [--device <name>] [--channel <number>] [--calibrate] [--range-pwm <min max> ] [--verbose, -v] [--help, -h]" << std::endl
61 std::cout <<
"Description" << std::endl
62 <<
" --device <name> Device name." << std::endl
63 <<
" Default: " << device << std::endl
65 <<
" --baud <rate> Serial link baud rate." << std::endl
66 <<
" Default: " << baudrate << std::endl
68 <<
" --channel <number> Channel to dial with." << std::endl
69 <<
" Default: " << channel << std::endl
71 <<
" --range-pwm <min max> Set PWM min and max values." << std::endl
72 <<
" You can use \"--calibrate\" to retrieve min and max pwm values."
73 <<
" Default: " << pwm_min <<
" " << pwm_max << std::endl
75 <<
" --range-angles <min max> Set angle min and max values (deg)." << std::endl
78 <<
" --verbose, -v Enable verbosity." << std::endl
80 <<
" --calibrate Start pwm calibration determining min and max admissible values." << std::endl
81 <<
" Once calibration done you can use \"--range-pwm <min max>\" option to set" << std::endl
82 <<
" the corresponding values" << std::endl
84 <<
" --help, -h Print this helper message." << std::endl
87 std::cout <<
"Error" << std::endl
89 <<
"Unsupported parameter " << argv[error] << std::endl;
93 int main(
int argc,
const char **argv)
96 std::string opt_device =
"COM4";
98 std::string opt_device =
"/dev/ttyACM0";
103 int opt_baudrate = 38400;
104 bool opt_verbose =
false;
105 bool opt_calibrate =
false;
106 unsigned short opt_pwm_min = 4000;
107 unsigned short opt_pwm_max = 8000;
108 float opt_angle_min =
static_cast<float>(
vpMath::rad(-45));
109 float opt_angle_max =
static_cast<float>(
vpMath::rad(45));
110 float opt_velocity =
static_cast<float>(
vpMath::rad(5));
111 float last_angle = 0;
113 for (
int i = 1; i < argc; i++) {
114 if (std::string(argv[i]) ==
"--device" && i + 1 < argc) {
115 opt_device = std::string(argv[i + 1]);
118 else if (std::string(argv[i]) ==
"--baud" && i + 1 < argc) {
119 opt_baudrate = std::atoi(argv[i + 1]);
122 else if (std::string(argv[i]) ==
"--channel" && i + 1 < argc) {
123 opt_channel = std::atoi(argv[i + 1]);
126 else if (std::string(argv[i]) ==
"--range-pwm" && i + 2 < argc) {
127 opt_pwm_min =
static_cast<unsigned short>(
vpMath::rad(std::atoi(argv[i + 1])));
128 opt_pwm_max =
static_cast<unsigned short>(
vpMath::rad(std::atoi(argv[i + 2])));
131 else if (std::string(argv[i]) ==
"--range-angles" && i + 2 < argc) {
132 opt_angle_min =
static_cast<float>(std::atof(argv[i + 1]));
133 opt_angle_max =
static_cast<float>(std::atof(argv[i + 2]));
136 else if (std::string(argv[i]) ==
"--calibrate") {
137 opt_calibrate =
true;
139 else if (std::string(argv[i]) ==
"--verbose" || std::string(argv[i]) ==
"-v") {
142 else if (std::string(argv[i]) ==
"--help" || std::string(argv[i]) ==
"-h") {
143 usage(argv, 0, opt_device, opt_baudrate, opt_channel, opt_pwm_min, opt_pwm_max, opt_angle_min, opt_angle_max);
147 usage(argv, i, opt_device, opt_baudrate, opt_channel, opt_pwm_min, opt_pwm_max, opt_angle_min, opt_angle_max);
152 std::chrono::seconds sec(1);
158 servo.connect(opt_device, opt_baudrate, opt_channel);
161 std::cout <<
"Proceed to calibration to determine pwm min and max values..." << std::endl;
162 std::cout <<
"WARNING: Calibration will move the servo at channel " << opt_channel <<
"!" << std::endl;
163 std::cout <<
"Press Enter to move to min and max pwm positions..." << std::endl;
166 unsigned short pwm_min, pwm_max;
167 servo.calibrate(pwm_min, pwm_max);
168 std::cout <<
"Servo on channel " << opt_channel <<
" has pwm range [" << pwm_min <<
", " << pwm_max <<
"]" << std::endl;
172 servo.setPwmRange(opt_pwm_min, opt_pwm_max);
173 servo.setAngularRange(opt_angle_min, opt_angle_max);
176 std::cout <<
"Move to zero position (deg): " <<
vpMath::deg(0) <<
" at max velocity" << std::endl;
177 servo.setAngularPosition(0, 0);
178 std::this_thread::sleep_for(std::chrono::seconds(3));
179 last_angle = servo.getAngularPosition();
180 std::cout <<
"Servo reached position (deg): " <<
vpMath::deg(last_angle) << std::endl;
185 std::cout <<
"Move at velocity (pwm): " << vel_pwm <<
" for 3 sec" << std::endl;
186 servo.setPwmVelocity(vel_pwm);
187 std::this_thread::sleep_for(3 * sec);
188 std::cout <<
"Servo reached position (pwm): " << servo.getPwmPosition() << std::endl;
191 std::cout <<
"Move at velocity (pwm): " << vel_pwm <<
" for 3 sec" << std::endl;
192 servo.setPwmVelocity(vel_pwm);
193 std::this_thread::sleep_for(3 * sec);
194 std::cout <<
"Servo reached position (pwm): " << servo.getPwmPosition() << std::endl;
195 std::cout <<
"End of velocity motion" << std::endl;
199 std::cout <<
"Move at velocity (deg/s): " <<
vpMath::deg(opt_velocity) <<
" for 3 sec" << std::endl;
200 servo.setAngularVelocity(opt_velocity);
201 std::this_thread::sleep_for(3 * sec);
202 std::cout <<
"Servo reached position (deg): " << servo.getPwmPosition() << std::endl;
204 std::cout <<
"Move at velocity (deg/s): " <<
vpMath::deg(-opt_velocity) <<
" for 3 sec" << std::endl;
205 servo.setAngularVelocity(-opt_velocity);
206 std::this_thread::sleep_for(3 * sec);
207 std::cout <<
"Servo reached position (deg): " << servo.getPwmPosition() << std::endl;
210 servo.stopVelocityCmd();
212 std::cout <<
"The end" << std::endl;
225 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.