25 #include <VirtualRobot/MathTools.h>
26 #include <VirtualRobot/RobotNodeSet.h>
44 NJointControllerRegistration<NJointTaskspaceVelocityController>
46 "NJointTaskspaceVelocityController");
54 arm->kinematicChainName = nodeSetName;
55 VirtualRobot::RobotNodeSetPtr rtRns =
56 rtGetRobot()->getRobotNodeSet(arm->kinematicChainName);
57 VirtualRobot::RobotNodeSetPtr nonRtRns =
58 nonRtRobotPtr->getRobotNodeSet(arm->kinematicChainName);
61 ARMARX_IMPORTANT <<
"Creating Taskspace Impedance controller with kinematic chain: "
62 << arm->kinematicChainName <<
" with num of joints: (RT robot) "
63 << rtRns->getSize() <<
", and (non-RT robot) " << nonRtRns->getSize();
65 arm->controller.initialize(nonRtRns, rtRns);
66 arm->controller.ftsensor.initialize(rtRns,
robotUnit, cfg.ftConfig);
67 arm->jointNames.clear();
68 for (
size_t i = 0; i < rtRns->getSize(); ++i)
70 std::string jointName = rtRns->getNode(i)->getName();
71 arm->jointNames.push_back(jointName);
75 auto* casted_ct = ct->
asA<ControlTarget1DoFActuatorVelocity>();
77 arm->targetsVel.push_back(casted_ct);
81 arm->sensorDevices.append(sv, jointName);
86 arm->nonRtConfig = cfg;
87 arm->rtStatus.reset(rtRns->getSize());
92 const NJointControllerConfigPtr& config,
98 ConfigPtrT cfg = ConfigPtrT::dynamicCast(config);
108 limb.emplace(pair.first, std::make_unique<ArmData>());
118 for (
auto& jointName :
hands->hands.at(pair.first)->jointNames)
120 hands->appendDevice(pair.first,
126 <<
hands->hands.at(pair.first)->kinematicChainName;
135 return "NJointTaskspaceVelocityController";
141 std::string taskName =
getClassName() +
"AdditionalTask";
153 while (
getState() == eManagedIceObjectStarted)
159 c.waitForCycleDuration();
167 arm->bufferConfigNonRtToRt.getWriteBuffer() = arm->nonRtConfig;
168 arm->bufferConfigNonRtToRt.commitWrite();
187 for (
auto& pair :
limb)
189 auto& arm = pair.second;
190 arm->nonRtConfig = arm->bufferConfigUserToNonRt.getUpToDateReadBuffer();
191 arm->rtStatusInNonRT = arm->bufferRtStatusToNonRt.getUpToDateReadBuffer();
192 rtSafe = rtSafe and pair.second->rtStatusInNonRT.rtSafe;
196 hands->nonRTUpdateStatus();
204 for (
auto& pair :
limb)
210 hands->nonRTSetTarget();
217 for (
auto& pair :
limb)
219 if (not pair.second->rtReady)
222 if (not pair.second->rtStatusInNonRT.rtTargetSafe)
224 pair.second->rtStatusInNonRT =
225 pair.second->bufferRtStatusToNonRt.getUpToDateReadBuffer();
227 pair.second->rtStatusInNonRT.currentPose,
228 pair.second->nonRtConfig.desiredPose,
239 arm->rtConfig = arm->bufferConfigNonRtToRt.getUpToDateReadBuffer();
240 arm->rtStatus.deltaT = deltaT;
241 arm->rtStatus.accumulateTime += deltaT;
243 arm->controller.ftsensor.updateStatus(arm->rtConfig.ftConfig,
244 arm->rtStatus.currentForceTorque,
245 arm->rtStatus.deltaT,
246 arm->rtFirstRun.load());
248 arm->sensorDevices.updateJointValues(
249 arm->rtStatus.jointPos, arm->rtStatus.jointVel, arm->rtStatus.jointTor);
252 arm->bufferRtStatusToNonRt.getWriteBuffer() = arm->rtStatus;
253 arm->bufferRtStatusToNonRt.commitWrite();
254 arm->bufferRtStatusToUser.getWriteBuffer() = arm->rtStatus;
255 arm->bufferRtStatusToUser.commitWrite();
260 const Eigen::VectorXf& targetVelocity)
262 for (
unsigned int i = 0; i < targetVelocity.size(); i++)
264 arm->targetsVel.at(i)->velocity = targetVelocity(i);
265 if (!arm->targetsVel.at(i)->isValid())
267 arm->targetsVel.at(i)->velocity = 0;
270 arm->bufferRtStatusToOnPublish.getWriteBuffer() = arm->rtStatus;
271 arm->bufferRtStatusToOnPublish.commitWrite();
273 arm->bufferConfigRtToOnPublish.getWriteBuffer() = arm->rtConfig;
274 arm->bufferConfigRtToOnPublish.commitWrite();
276 arm->bufferConfigRtToUser.getWriteBuffer() = arm->rtConfig;
277 arm->bufferConfigRtToUser.commitWrite();
279 if (arm->rtFirstRun.load())
281 arm->rtFirstRun.store(
false);
282 arm->rtReady.store(
true);
289 double time_measure = IceUtil::Time::now().toMicroSecondsDouble();
291 double time_update_status = IceUtil::Time::now().toMicroSecondsDouble() - time_measure;
293 arm->controller.run(arm->rtConfig, arm->rtStatus);
294 double time_run_rt = IceUtil::Time::now().toMicroSecondsDouble() - time_measure;
298 double time_set_target = IceUtil::Time::now().toMicroSecondsDouble() - time_measure;
299 time_measure = IceUtil::Time::now().toMicroSecondsDouble() - time_measure;
301 if (time_measure > 200)
304 "time_update_status: %.2f\n"
305 "run_rt_limb: %.2f\n"
306 "set_target_limb: %.2f\n"
309 time_run_rt - time_update_status,
310 time_set_target - time_run_rt,
312 .deactivateSpam(1.0f);
320 double deltaT = timeSinceLastIteration.toSecondsDouble();
321 for (
auto& pair :
limb)
323 limbRT(pair.second, deltaT);
327 hands->updateRTStatus(deltaT);
335 const Ice::Current& iceCurrent)
344 pair.second.desiredPose,
345 prevCfg.limbs.at(pair.first).desiredPose,
347 "previous desired pose",
348 pair.second.safeDistanceMMToGoal,
349 pair.second.safeRotAngleDegreeToGoal,
350 "updateConfig_" + pair.first))
353 pair.second.desiredPose = prevCfg.limbs.at(pair.first).desiredPose;
355 limb.at(pair.first)->bufferConfigUserToNonRt.getWriteBuffer() = pair.second;
356 limb.at(pair.first)->bufferConfigUserToNonRt.commitWrite();
367 for (
auto& pair :
limb)
370 pair.second->bufferConfigRtToUser.getUpToDateReadBuffer();
381 const bool forceGuard,
382 const bool torqueGuard,
383 const Ice::Current& iceCurrent)
388 it->second.ftConfig.enableSafeGuardForce = forceGuard;
389 it->second.ftConfig.enableSafeGuardTorque = torqueGuard;
390 limb.at(nodeSetName)->controller.ftsensor.enableSafeGuard(forceGuard, torqueGuard);
392 <<
"reset safe guard with force torque sensors: safe? "
393 <<
limb.at(nodeSetName)->controller.ftsensor.isSafe(it->second.ftConfig);
394 limb.at(nodeSetName)->bufferConfigUserToNonRt.getWriteBuffer() = it->second;
395 limb.at(nodeSetName)->bufferConfigUserToNonRt.commitWrite();
400 <<
" found in the controllers.";
407 const Ice::Current& iceCurrent)
412 return limb.at(nodeSetName)->controller.ftsensor.ftSafe.load();
417 <<
" found in the controllers.";
424 const Ice::Current& iceCurrent)
426 std::vector<float> tcpVel;
427 auto& arm =
limb.at(rns);
428 auto s = arm->bufferRtStatusToUser.getUpToDateReadBuffer();
429 for (
int i = 0; i <
s.currentTwist.size(); i++)
431 tcpVel.push_back(
s.currentTwist[i]);
440 const Ice::Current& iceCurrent)
444 for (
size_t i = 0; i < 4; ++i)
446 for (
int j = 0; j < 4; ++j)
448 pair.second.desiredPose(i, j) = targetPoseMap.at(pair.first)[i][j];
451 if (targetNullspaceMap.at(pair.first).size() > 0)
453 const size_t nDoF = pair.second.desiredNullspaceJointAngles.value().size();
455 <<
"the joint numbers does not match";
456 for (
size_t i = 0; i < nDoF; ++i)
458 pair.second.desiredNullspaceJointAngles.value()(i) =
459 targetNullspaceMap.at(pair.first)[i];
462 limb.at(pair.first)->bufferConfigUserToNonRt.getWriteBuffer() = pair.second;
463 limb.at(pair.first)->bufferConfigUserToNonRt.commitWrite();
471 const auto nDoF = arm->controller.numOfJoints;
474 const auto checkNonNegative = [](
const auto&
v) {
ARMARX_CHECK((
v.array() >= 0).all()); };
476 if (!configData.desiredNullspaceJointAngles.has_value())
480 ARMARX_INFO <<
"No user defined nullspace target, reset to zero with "
482 configData.desiredNullspaceJointAngles = Eigen::VectorXf::Zero(nDoF);
483 arm->reInitPreActivate.store(
true);
488 arm->bufferConfigRtToUser.getUpToDateReadBuffer().desiredNullspaceJointAngles;
490 <<
"Controller is active, but no user defined nullspace target. \n"
491 "You should first get up-to-date config of this controller and then update "
493 " auto cfg = ctrl->getConfig(); \n"
494 " cfg.desiredPose = xxx;\n"
495 " ctrl->updateConfig(cfg);\n"
496 "Now, I decide by myself to use the existing values of nullspace target\n"
497 << currentValue.value();
498 configData.desiredNullspaceJointAngles = currentValue;
502 checkSize(configData.desiredNullspaceJointAngles.value());
503 checkSize(configData.kdNullspaceVel);
504 checkSize(configData.kpNullspaceVel);
506 checkNonNegative(configData.kdNullspaceVel);
507 checkNonNegative(configData.kpNullspaceVel);
508 checkNonNegative(configData.kpNullspaceVel);
509 checkNonNegative(configData.kdNullspaceVel);
517 auto rtData = arm->bufferConfigRtToOnPublish.getUpToDateReadBuffer();
518 auto rtStatus = arm->bufferRtStatusToOnPublish.getUpToDateReadBuffer();
532 datafields,
"nullspaceTarget", rtData.desiredNullspaceJointAngles.value());
534 debugObs->setDebugChannel(
getClassName() +
"_" + arm->kinematicChainName, datafields);
542 for (
auto& pair :
limb)
544 if (not pair.second->rtReady.load())
553 for (
auto& pair :
limb)
555 pair.second->rtReady.store(
false);
562 VirtualRobot::RobotNodeSetPtr rns =
rtGetRobot()->getRobotNodeSet(arm->kinematicChainName);
564 auto quat = VirtualRobot::MathTools::eigen4f2quat(currentPose);
566 "[ %.2f, %.2f, %.2f, %.2f, %.2f, %.2f, %.2f ].",
575 if (arm->reInitPreActivate.load())
577 arm->rtConfig.desiredNullspaceJointAngles.value() = rns->getJointValuesEigen();
578 arm->rtConfig.desiredPose = currentPose;
580 arm->bufferConfigUserToNonRt.reinitAllBuffers(arm->rtConfig);
581 arm->reInitPreActivate.store(
false);
584 arm->nonRtConfig = arm->rtConfig;
585 arm->bufferConfigNonRtToRt.reinitAllBuffers(arm->rtConfig);
586 arm->bufferConfigRtToOnPublish.reinitAllBuffers(arm->rtConfig);
587 arm->bufferConfigRtToUser.reinitAllBuffers(arm->rtConfig);
590 arm->sensorDevices.updateJointValues(
591 arm->rtStatus.jointPos, arm->rtStatus.jointVel, arm->rtStatus.jointTor);
592 arm->rtStatus.rtPreActivate(currentPose);
594 arm->rtStatusInNonRT = arm->rtStatus;
595 arm->bufferRtStatusToOnPublish.reinitAllBuffers(arm->rtStatus);
596 arm->bufferRtStatusToNonRt.reinitAllBuffers(arm->rtStatus);
597 arm->bufferRtStatusToUser.reinitAllBuffers(arm->rtStatus);
600 arm->controller.preactivateInit(rns);
606 for (
auto& pair :
limb)
610 userConfig.limbs.at(pair.first) = pair.second->rtConfig;
614 hands->rtPreActivate();
631 const std::map<std::string, ConstControlDevicePtr>& controlDevices,
632 const std::map<std::string, ConstSensorDevicePtr>&)
638 ::armarx::WidgetDescription::WidgetSeq widgets;
641 LabelPtr label =
new Label;
642 label->text =
"select a controller config";
644 StringComboBoxPtr cfgBox =
new StringComboBox;
645 cfgBox->name =
"config_box";
646 cfgBox->defaultIndex = 0;
647 cfgBox->multiSelect =
false;
649 cfgBox->options = std::vector<std::string>{
"default",
"default_right",
"default_right_a7"};
652 layout->children.emplace_back(label);
653 layout->children.emplace_back(cfgBox);
656 layout->children.insert(layout->children.end(), widgets.begin(), widgets.end());
665 auto cfgName =
values.at(
"config_box")->getString();
668 "controller_config/NJointTaskspaceVelocityController/" + cfgName +
".json");
672 auto cfgDTO = armarx::readFromJson<ConfigDict>(configPath.
toSystemPath());
675 return new ConfigurableNJointControllerConfig{cfgDTO.toAronDTO()};