iCub-main
Loading...
Searching...
No Matches
icub-main
src
modules
cartesianInterfaceControl
main.cpp
Go to the documentation of this file.
1
/*
2
* Copyright (C) 2010 RobotCub Consortium, European Commission FP6 Project IST-004370
3
* Author: Ugo Pattacini
4
* email: ugo.pattacini@iit.it
5
* website: www.robotcub.org
6
* Permission is granted to copy, distribute, and/or modify this program
7
* under the terms of the GNU General Public License, version 2 or any
8
* later version published by the Free Software Foundation.
9
*
10
* A copy of the license can be found at
11
* http://www.robotcub.org/icub/license/gpl.txt
12
*
13
* This program is distributed in the hope that it will be useful, but
14
* WITHOUT ANY WARRANTY; without even the implied warranty of
15
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General
16
* Public License for more details
17
*/
18
115
#include <cstdio>
116
#include <string>
117
#include <map>
118
119
#include <yarp/os/Network.h>
120
#include <yarp/os/RFModule.h>
121
#include <yarp/os/PeriodicThread.h>
122
#include <yarp/os/Bottle.h>
123
#include <yarp/os/BufferedPort.h>
124
#include <yarp/os/Time.h>
125
#include <yarp/sig/Vector.h>
126
#include <yarp/math/Math.h>
127
128
#include <yarp/dev/Drivers.h>
129
#include <yarp/dev/CartesianControl.h>
130
#include <yarp/dev/PolyDriver.h>
131
132
#include <
iCub/iKin/iKinFwd.h
>
133
134
#define MAX_TORSO_PITCH 30.0
// [deg]
135
#define EXECTIME_THRESDIST 0.3
// [m]
136
#define PRINT_STATUS_PER 1.0
// [s]
137
138
using namespace
std;
139
using namespace
yarp::os;
140
using namespace
yarp::dev
;
141
using namespace
yarp::sig;
142
using namespace
yarp::math;
143
using namespace
iCub::iKin
;
144
145
146
/************************************************************************/
147
class
CtrlThread
:
public
PeriodicThread {
148
protected
:
149
ResourceFinder&
rf
;
150
PolyDriver
driver
;
151
ICartesianControl*
iarm
;
152
BufferedPort<Bottle>
port_xd
;
153
154
bool
ctrlCompletePose
;
155
bool
reinstateContext
;
156
string
remote
;
157
string
local
;
158
159
int
startup_context_id
;
160
int
task_context_id
;
161
162
Vector
xd
;
163
Vector
od
;
164
165
double
defaultExecTime
;
166
double
t0
;
167
168
public
:
169
/************************************************************************/
170
CtrlThread
(
unsigned
int
_period, ResourceFinder& _rf,
171
string
_remote,
string
_local) :
172
PeriodicThread((double)_period / 1000.0),
rf
(_rf),
173
remote
(_remote),
local
(_local) { }
174
175
/************************************************************************/
176
bool
threadInit
() {
177
// get params from the RF
178
ctrlCompletePose
= !
rf
.check(
"onlyXYZ"
);
179
reinstateContext
=
rf
.check(
"reinstate_context"
);
180
181
// open the client
182
Property option(
"(device cartesiancontrollerclient)"
);
183
option.put(
"remote"
,
remote
);
184
option.put(
"local"
,
local
);
185
if
(!
driver
.open(option))
186
return
false
;
187
188
// open the view
189
driver
.view(
iarm
);
190
191
// latch the controller context
192
iarm
->storeContext(&
startup_context_id
);
193
194
// set trajectory time
195
defaultExecTime
=
rf
.check(
"T"
, Value(2.0)).asFloat64();
196
197
Bottle info;
198
iarm
->getInfo(info);
199
iKinLimbVersion
hwver(info.find(
"arm_version"
).asString());
200
printf(
"Detected arm kinematics version %s\n"
, hwver.
get_version
().c_str());
201
202
map<string, int> torsoJointsRemap{{
"pitch"
, 0 }, {
"roll"
, 1 }, {
"yaw"
, 2 }};
203
if
(hwver >=
iKinLimbVersion
(
"3.0"
)) {
204
torsoJointsRemap[
"roll"
] = 0;
205
torsoJointsRemap[
"pitch"
] = 1;
206
}
207
208
// set torso dofs
209
Vector newDof, curDof;
210
iarm
->getDOF(curDof);
211
newDof = curDof;
212
213
if
(curDof.length() > 7) {
// consider only robots w/ torso
214
if
(
rf
.check(
"DOF10"
)) {
215
// torso joints completely enabled
216
newDof[torsoJointsRemap[
"pitch"
]] = 1;
217
newDof[torsoJointsRemap[
"roll"
]] = 1;
218
newDof[torsoJointsRemap[
"yaw"
]] = 1;
219
220
limitTorsoPitch
(torsoJointsRemap[
"pitch"
]);
221
}
else
if
(
rf
.check(
"DOF9"
)) {
222
// torso yaw and pitch enabled
223
newDof[torsoJointsRemap[
"pitch"
]] = 1;
224
newDof[torsoJointsRemap[
"roll"
]] = 0;
225
newDof[torsoJointsRemap[
"yaw"
]] = 1;
226
227
limitTorsoPitch
(torsoJointsRemap[
"pitch"
]);
228
}
else
if
(
rf
.check(
"DOF8"
)) {
229
// only torso yaw enabled
230
newDof[torsoJointsRemap[
"pitch"
]] = 0;
231
newDof[torsoJointsRemap[
"roll"
]] = 0;
232
newDof[torsoJointsRemap[
"yaw"
]] = 1;
233
}
else
{
234
// torso joints completely disabled
235
newDof[torsoJointsRemap[
"pitch"
]] = 0;
236
newDof[torsoJointsRemap[
"roll"
]] = 0;
237
newDof[torsoJointsRemap[
"yaw"
]] = 0;
238
}
239
}
240
241
iarm
->setDOF(newDof, curDof);
242
243
// set tracking mode
244
iarm
->setTrackingMode(
false
);
245
246
// set cartesian tolerance
247
if
(
rf
.check(
"tol"
))
248
iarm
->setInTargetTol(
rf
.find(
"tol"
).asFloat64());
249
250
// latch the controller context for our task
251
iarm
->storeContext(&
task_context_id
);
252
253
// init variables
254
while
(
true
) {
255
if
(
iarm
->getPose(
xd
,
od
))
256
break
;
257
258
Time::delay(0.1);
259
}
260
261
// open ports
262
port_xd
.open(
local
+
"/xd:i"
);
263
264
return
true
;
265
}
266
267
/************************************************************************/
268
void
afterStart
(
bool
s) {
269
if
(s)
270
printf(
"Thread started successfully\n"
);
271
else
272
printf(
"Thread did not start\n"
);
273
274
t0
= Time::now();
275
}
276
277
/************************************************************************/
278
void
run
() {
279
if
(Bottle* b =
port_xd
.read(
false
)) {
280
if
(b->size() >= 3) {
281
for
(
int
i = 0; i < 3; i++)
282
xd
[i] = b->get(i).asFloat64();
283
284
if
(
ctrlCompletePose
&& (b->size() >= 7)) {
285
for
(
int
i = 0; i < 4; i++)
286
od
[i] = b->get(3 + i).asFloat64();
287
}
288
289
if
(
reinstateContext
)
290
iarm
->restoreContext(
task_context_id
);
291
292
const
double
execTime =
calcExecTime
(
xd
);
293
if
(
ctrlCompletePose
)
294
iarm
->goToPose(
xd
,
od
, execTime);
295
else
296
iarm
->goToPosition(
xd
, execTime);
297
}
298
}
299
300
printStatus
();
301
}
302
303
/************************************************************************/
304
void
threadRelease
() {
305
iarm
->stopControl();
306
iarm
->restoreContext(
startup_context_id
);
307
driver
.close();
308
309
port_xd
.interrupt();
310
port_xd
.close();
311
}
312
313
/************************************************************************/
314
void
limitTorsoPitch
(
const
int
axis) {
315
double
min, max;
316
iarm
->getLimits(axis, &min, &max);
317
iarm
->setLimits(axis, min,
MAX_TORSO_PITCH
);
318
}
319
320
/************************************************************************/
321
double
calcExecTime
(
const
Vector&
xd
) {
322
Vector
x
, o;
323
iarm
->getPose(
x
, o);
324
325
if
(norm(
xd
-
x
) <
EXECTIME_THRESDIST
)
326
return
defaultExecTime
;
327
else
328
return
1.5 *
defaultExecTime
;
329
}
330
331
/************************************************************************/
332
void
printStatus
() {
333
double
t = Time::now();
334
335
if
(t -
t0
>=
PRINT_STATUS_PER
) {
336
Vector
x
, o, xdot, odot;
337
Vector xdhat, odhat, qdhat;
338
339
iarm
->getPose(
x
, o);
340
iarm
->getTaskVelocities(xdot, odot);
341
iarm
->getDesired(xdhat, odhat, qdhat);
342
double
e_x = norm(xdhat -
x
);
343
344
printf(
"xd [m] = %s\n"
,
xd
.toString().c_str());
345
printf(
"xdhat [m] = %s\n"
, xdhat.toString().c_str());
346
printf(
"x [m] = %s\n"
,
x
.toString().c_str());
347
printf(
"xdot [m/s] = %s\n"
, xdot.toString().c_str());
348
printf(
"norm(e_x) [m] = %g\n"
, e_x);
349
350
if
(
ctrlCompletePose
) {
351
Vector e_o = dcm2axis(axis2dcm(odhat) * axis2dcm(o).transposed());
352
e_o *= e_o[3];
353
e_o.pop_back();
354
355
printf(
"od [rad] = %s\n"
,
od
.toString().c_str());
356
printf(
"odhat [rad] = %s\n"
, odhat.toString().c_str());
357
printf(
"o [rad] = %s\n"
, o.toString().c_str());
358
printf(
"odot [rad/s] = %s\n"
, odot.toString().c_str());
359
printf(
"norm(e_o) [rad] = %g\n"
, norm(e_o));
360
}
361
362
printf(
"\n"
);
363
364
t0
= t;
365
}
366
}
367
};
368
369
370
/************************************************************************/
371
class
CtrlModule
:
public
RFModule {
372
protected
:
373
CtrlThread
*
thr
;
374
Port
rpcPort
;
375
376
public
:
377
/************************************************************************/
378
bool
configure
(ResourceFinder& rf) {
379
string
slash =
"/"
;
380
string
name;
381
string
robot;
382
string
part;
383
string
remote;
384
string
local;
385
386
// get params from the RF
387
name = rf.check(
"name"
, Value(
"armCtrl"
)).asString();
388
robot = rf.check(
"robot"
, Value(
"icub"
)).asString();
389
part = rf.check(
"part"
, Value(
"right_arm"
)).asString();
390
391
remote = slash + robot +
"/cartesianController/"
+ part;
392
local = slash + name + slash + part;
393
394
thr
=
new
CtrlThread
(20, rf, remote, local);
395
if
(!
thr
->start()) {
396
delete
thr
;
397
return
false
;
398
}
399
400
rpcPort
.open(local +
"/rpc"
);
401
attach(
rpcPort
);
402
403
return
true
;
404
}
405
406
/************************************************************************/
407
bool
close
() {
408
thr
->stop();
409
delete
thr
;
410
411
rpcPort
.interrupt();
412
rpcPort
.close();
413
414
return
true
;
415
}
416
417
/************************************************************************/
418
double
getPeriod
() {
419
return
1.0;
420
}
421
422
/************************************************************************/
423
bool
updateModule
() {
424
return
true
;
425
}
426
};
427
428
429
/************************************************************************/
430
int
main
(
int
argc,
char
* argv[]) {
431
Network
yarp
;
432
433
ResourceFinder rf;
434
rf.configure(argc, argv);
435
436
if
(rf.check(
"help"
)) {
437
printf(
"Options:\n"
);
438
printf(
"\t--name name : controller name (default armCtrl)\n"
);
439
printf(
"\t--robot name : robot name to connect to (default: icub)\n"
);
440
printf(
"\t--part type : robot arm type, left_arm or right_arm (default: right_arm)\n"
);
441
printf(
"\t--T time : specify the task execution time in seconds (default: 2.0)\n"
);
442
printf(
"\t--tol tolerance: specify the cartesian tolerance in meters\n"
);
443
printf(
"\t--DOF10 : control the torso yaw/roll/pitch as well\n"
);
444
printf(
"\t--DOF9 : control the torso yaw/pitch as well\n"
);
445
printf(
"\t--DOF8 : control the torso yaw as well\n"
);
446
printf(
"\t--onlyXYZ : disable orientation control\n"
);
447
printf(
"\t--reinstate_context : reinstate controller context upon target reception\n"
);
448
printf(
"\n"
);
449
450
return
0;
451
}
452
453
if
(!
yarp
.checkNetwork()) {
454
printf(
"YARP server not available!\n"
);
455
return
-1;
456
}
457
458
CtrlModule
mod;
459
return
mod.runModule(rf);
460
}
461
462
463
CtrlModule
Definition
main.cpp:371
CtrlModule::updateModule
bool updateModule()
Definition
main.cpp:423
CtrlModule::close
bool close()
Definition
main.cpp:407
CtrlModule::thr
CtrlThread * thr
Definition
main.cpp:373
CtrlModule::getPeriod
double getPeriod()
Definition
main.cpp:418
CtrlModule::rpcPort
Port rpcPort
Definition
main.cpp:374
CtrlModule::configure
bool configure(ResourceFinder &rf)
Definition
main.cpp:378
CtrlThread
Definition
main.cpp:147
CtrlThread::CtrlThread
CtrlThread(unsigned int _period, ResourceFinder &_rf, string _remote, string _local)
Definition
main.cpp:170
CtrlThread::t0
double t0
Definition
main.cpp:166
CtrlThread::od
Vector od
Definition
main.cpp:163
CtrlThread::remote
string remote
Definition
main.cpp:156
CtrlThread::defaultExecTime
double defaultExecTime
Definition
main.cpp:165
CtrlThread::xd
Vector xd
Definition
main.cpp:162
CtrlThread::task_context_id
int task_context_id
Definition
main.cpp:160
CtrlThread::rf
ResourceFinder & rf
Definition
main.cpp:149
CtrlThread::afterStart
void afterStart(bool s)
Definition
main.cpp:268
CtrlThread::run
void run()
Definition
main.cpp:278
CtrlThread::local
string local
Definition
main.cpp:157
CtrlThread::startup_context_id
int startup_context_id
Definition
main.cpp:159
CtrlThread::port_xd
BufferedPort< Bottle > port_xd
Definition
main.cpp:152
CtrlThread::threadInit
bool threadInit()
Definition
main.cpp:176
CtrlThread::iarm
ICartesianControl * iarm
Definition
main.cpp:151
CtrlThread::reinstateContext
bool reinstateContext
Definition
main.cpp:155
CtrlThread::ctrlCompletePose
bool ctrlCompletePose
Definition
main.cpp:154
CtrlThread::limitTorsoPitch
void limitTorsoPitch(const int axis)
Definition
main.cpp:314
CtrlThread::threadRelease
void threadRelease()
Definition
main.cpp:304
CtrlThread::driver
PolyDriver driver
Definition
main.cpp:150
CtrlThread::calcExecTime
double calcExecTime(const Vector &xd)
Definition
main.cpp:321
CtrlThread::printStatus
void printStatus()
Definition
main.cpp:332
iCub::iKin::iKinLimbVersion
A class for defining the versions of the iCub limbs.
Definition
iKinFwd.h:1045
iCub::iKin::iKinLimbVersion::get_version
std::string get_version() const
Return the version string.
Definition
iKinFwd.cpp:1600
x
x
Definition
compute_ekf_sym.m:21
iKinFwd.h
EXECTIME_THRESDIST
#define EXECTIME_THRESDIST
Definition
main.cpp:135
MAX_TORSO_PITCH
#define MAX_TORSO_PITCH
Definition
main.cpp:134
PRINT_STATUS_PER
#define PRINT_STATUS_PER
Definition
main.cpp:104
main
int main()
Definition
main.cpp:67
iCub::iKin
Definition
iKinFwd.h:71
yarp::dev
Definition
DebugInterfaces.h:52
yarp
Copyright (C) 2008 RobotCub Consortium.
Definition
DebugInterfaces.h:51
Generated on Mon Jan 13 2025 14:57:40 for iCub-main by
1.9.8