WipingMixImpVelColController.cpp
Go to the documentation of this file.
1 /**
2  * This file is part of ArmarX.
3  *
4  * ArmarX is free software; you can redistribute it and/or modify
5  * it under the terms of the GNU General Public License version 2 as
6  * published by the Free Software Foundation.
7  *
8  * ArmarX is distributed in the hope that it will be useful, but
9  * WITHOUT ANY WARRANTY; without even the implied warranty of
10  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11  * GNU General Public License for more details.
12  *
13  * You should have received a copy of the GNU General Public License
14  * along with this program. If not, see <http://www.gnu.org/licenses/>.
15  *
16  * @author Jianfeng Gao ( jianfeng dot gao at kit dot edu )
17  * @date 2024
18  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
19  * GNU General Public License
20  */
21 
23 
25 #include <RobotAPI/components/units/RobotUnit/RobotUnit.h> // FIXME avoid this
26 
28 
30 {
31  NJointControllerRegistration<NJointWipingMixImpVelColMPController>
33  "NJointWipingMixImpVelColMPController");
34 
36  const RobotUnitPtr& robotUnit,
37  const NJointControllerConfigPtr& config,
38  const VirtualRobot::RobotPtr& robot) :
41  {
42  ARMARX_IMPORTANT << getClassName() + " construction done";
43  }
44 
45  std::string
47  {
48  return "NJointWipingMixImpVelColMPController";
49  }
50 
51  void
54  const Ice::Current& iceCurrent)
55  {
57  {
58  for (auto& pair : limb)
59  {
60  pair.second->rtFirstRun.store(true);
61  }
62  }
63  }
64 
65  void
68  const Ice::Current& iceCurrent)
69  {
70  adaptionCfg = arondto::WipingAdaptionConfigs::FromAron(dto);
71  for (auto& pair : limb)
72  {
73  auto search = adaptionCfg.cfg.find(pair.first);
74  if (search == adaptionCfg.cfg.end())
75  {
76  ARMARX_ERROR << "Arm " << pair.first << " is not included in the adaption config";
77  }
78  // ARMARX_CHECK_NOT_EQUAL(search, adaptionCfg.cfg.end());
79  }
80  adaptionReady.store(true);
81  }
82 
83  void
85  {
88  // for (auto& pair: limb)
89  // {
90  // adaptionCfg.cfg[pair.first] = arondto::WipingAdaptionConfigs();
91  // }
92  for (auto& pair : limb)
93  {
94  adaptionStatus[pair.first] = AdaptionStatus();
95  }
96  }
97 
98  void
100  {
101  if (not adaptionReady.load())
102  {
103  return;
104  }
105  std::lock_guard<std::recursive_mutex> lock(mtx_mps);
106  bool rtSafe = additionalTaskUpdateStatus();
107 
108  // when ft guard is enabled and rt is not safe due to ft trigger, then we stop the corresponding mp
109  std::map<std::string, bool> flagMPToStop;
110  for (auto& pair : limb)
111  {
112  flagMPToStop[pair.first] = limb.at(pair.first)->controller.ftsensor.ftSafe.load();
113  }
114 
115  std::map<std::string, bool> mpRunning;
116  for (auto& _mp : mps)
117  {
118  const auto mpNodeSet = _mp.second.mp->getNodeSetName();
119  auto search = flagMPToStop.find(mpNodeSet);
120  if (search != flagMPToStop.end() and flagMPToStop.at(mpNodeSet) and
121  _mp.second.mp->isRunning() and
122  (limb.at(mpNodeSet)->controller.ftsensor.isForceGuardEnabled() or
123  limb.at(mpNodeSet)->controller.ftsensor.isTorqueGuardEnabled()))
124  {
125  _mp.second.mp->stop();
126  ARMARX_INFO << "Due to force/torque safe guard, MP " << _mp.second.mp->getMPName()
127  << " is stopped at " << _mp.second.mp->getCanonicalValue();
128  }
129  mpRunning[_mp.second.mp->getMPName()] = _mp.second.mp->isRunning();
130  if (_mp.second.mp->isFinished())
131  {
132  continue;
133  }
134  if (_mp.second.mp->getRole() == "taskspace")
135  {
136  auto& arm = limb.at(_mp.second.mp->getNodeSetName());
137  if (_mp.second.mp->isFirstRun())
138  {
139  ARMARX_INFO << "checking TSMP initial status ...";
140  /// Here we check if the TSMP start pose is too far away from the current pose, if so, we would
141  /// prefer to reset the mp start from the current pose.
142  const std::vector<double> mpStart = _mp.second.mp->getStartVec();
143  const auto mpStartPose = common::dVecToMat4(mpStart);
145  arm->rtStatusInNonRT.currentPose,
146  mpStartPose,
147  "current pose",
148  "TSMP initial start pose",
149  arm->nonRtConfig.safeDistanceMMToGoal,
150  arm->nonRtConfig.safeRotAngleDegreeToGoal,
151  _mp.second.mp->getMPName() + "mp_set_target"))
152  {
154  << "-- deviation from current pose too large, reset MP start pose";
155  _mp.second.mp->validateInitialState(
156  common::mat4ToDVec(arm->rtStatusInNonRT.desiredPose));
157  }
158  ARMARX_INFO << "done";
159  }
160  mp::TSMPInputPtr in = std::dynamic_pointer_cast<mp::TSMPInput>(_mp.second.input);
161  in->pose = arm->rtStatusInNonRT.currentPose;
162  in->vel = arm->rtStatusInNonRT.currentTwist;
163  in->deltaT = arm->rtStatusInNonRT.deltaT;
164  }
165  else if (_mp.second.mp->getRole() == "nullspace")
166  {
167  auto& arm = limb.at(_mp.second.mp->getNodeSetName());
168  mp::JSMPInputPtr in = std::dynamic_pointer_cast<mp::JSMPInput>(_mp.second.input);
169  ARMARX_CHECK_EQUAL(arm->rtStatusInNonRT.numJoints,
170  arm->rtStatusInNonRT.jointPos.size());
171  in->angleRadian = arm->rtStatusInNonRT.jointPos;
172  in->angularVel = arm->rtStatusInNonRT.qvelFiltered;
173  in->deltaT = arm->rtStatusInNonRT.deltaT;
174  }
175  else if (_mp.second.mp->getRole() == "hand")
176  {
177  auto& hand = hands->hands.at(_mp.second.mp->getNodeSetName());
178  mp::JSMPInputPtr in = std::dynamic_pointer_cast<mp::JSMPInput>(_mp.second.input);
179  in->angleRadian =
180  hand->bufferRTToNonRT.getUpToDateReadBuffer().jointPosition.value();
181  in->deltaT = hand->bufferRTToNonRT.getReadBuffer().deltaT;
182  }
183  }
184  if (not rtSafe)
185  {
187 
188  // return;
189  // Make sure that you do not call additionalTaskSetTarget before return;
190  // since we overwrite the arm->nonRtConfig in additionalTaskUpdateStatus() with user config, which does not
191  // align with the up-to-date mp status. userConfig is only updated to align with MP status everytime a MP
192  // finishes execution. Only then, user can use this controller as if no mp is there.
193  }
194  runMPs(rtSafe);
195  std::lock_guard<std::recursive_mutex> adaptLock(adaptionMtx);
196  /// set mp target to nonRT config data structure
197  for (auto& _mp : mps)
198  {
199  if (not mpRunning.at(_mp.second.mp->getMPName()))
200  {
201  continue;
202  }
203  for (auto& ftGuard : mpConfig.ftGuard)
204  {
205  if (ftGuard.mpName == _mp.second.mp->getMPName())
206  {
207  bool const forceGuard =
208  (ftGuard.force.has_value() and
209  ftGuard.force.value() >= _mp.second.mp->getCanonicalValue());
210  bool const torqueGuard =
211  (ftGuard.torque.has_value() and
212  ftGuard.torque.value() >= _mp.second.mp->getCanonicalValue());
213 
214  auto& arm = limb.at(_mp.second.mp->getNodeSetName());
215  bool const resetForce =
216  not arm->controller.ftsensor.isForceGuardEnabled() and forceGuard;
217  bool const resetTorque =
218  not arm->controller.ftsensor.isTorqueGuardEnabled() and torqueGuard;
219  if (resetForce or resetTorque)
220  {
221  ARMARX_INFO << "Triggering force torque safety guard for "
222  << arm->kinematicChainName << " at can value "
223  << _mp.second.mp->getCanonicalValue();
224  }
225 
226  arm->nonRtConfig.ftConfig.enableSafeGuardForce = forceGuard;
227  arm->nonRtConfig.ftConfig.enableSafeGuardTorque = torqueGuard;
228  arm->controller.ftsensor.enableSafeGuard(resetForce, resetTorque);
229  }
230  }
231 
232  if (_mp.second.mp->getRole() == "taskspace")
233  {
234  auto& nodeSetName = _mp.second.mp->getNodeSetName();
235  auto& cfg = adaptionCfg.cfg.at(nodeSetName);
236  auto& s = adaptionStatus.at(nodeSetName);
237  auto& arm = limb.at(nodeSetName);
238  mp::TSMPOutputPtr out =
239  std::dynamic_pointer_cast<mp::TSMPOutput>(_mp.second.output);
240 
241  /// Note that here we only report warning but still set target to RT, we rely on MP phase stop and
242  /// RT safety mechanism to stop or pause the action if needed
244  arm->rtStatusInNonRT.currentPose,
245  arm->nonRtConfig.desiredPose,
246  "current pose",
247  "desiredPose for RT",
248  arm->nonRtConfig.safeDistanceMMToGoal,
249  arm->nonRtConfig.safeRotAngleDegreeToGoal,
250  _mp.second.mp->getMPName() + "mp_set_target");
251  arm->nonRtConfig.desiredPose = out->pose;
252  arm->nonRtConfig.desiredTwist = out->vel;
253 
254  /// Note: use force to generate acceleration target and aggregate to the target velocity
255  float desiredVelZ =
256  cfg.kpf * (cfg.targetForce - arm->rtStatusInNonRT.currentForceTorque(2));
257  arm->nonRtConfig.desiredTwist(2) -= desiredVelZ;
258  arm->nonRtConfig.desiredTwist.tail<3>().setZero();
259 
260  float dT = static_cast<float>(arm->rtStatusInNonRT.deltaT);
261 
262  /// check phase stop
263  auto& adaptKImp = arm->nonRtConfig.kpImpedance;
264  auto& adaptKVel = arm->nonRtConfig.kpCartesianVel;
265  /// TODO find a way to keep the original coefficient parameter
266  Eigen::Vector6f originalKpImp = Eigen::Vector6f::Zero();
267  Eigen::Vector6f originalKpVel = Eigen::Vector6f::Zero();
268  if (tooFar)
269  {
270  Eigen::Vector2f currentForceXY =
271  arm->rtStatusInNonRT.currentForceTorque.head<2>();
272  Eigen::Vector2f dragForce =
273  currentForceXY - cfg.deadZoneForce * currentForceXY / currentForceXY.norm();
274  adaptKImp.head<2>() =
275  (adaptKImp.head<2>().array() - dT * cfg.adaptForceCoeff * dragForce.norm())
276  .max(0.0f);
277  adaptKVel.head<2>() =
278  (adaptKVel.head<2>().array() - dT * cfg.adaptForceCoeff * dragForce.norm())
279  .max(0.0f);
280  // lastDiff = diff;
281  }
282  else
283  {
284  adaptKImp.head<2>() = (adaptKImp.head<2>().array() + dT * cfg.adaptCoeff)
285  .min(originalKpImp.head<2>().array());
286  adaptKVel.head<2>() = (adaptKVel.head<2>().array() + dT * cfg.adaptCoeff)
287  .min(originalKpVel.head<2>().array());
288  }
289  adaptKImp(2) = originalKpImp(2);
290  adaptKImp(2) = originalKpVel(2);
291 
292  /// Note: integrate the target pose
293  arm->nonRtConfig.desiredPose.block<3, 1>(0, 3) +=
294  dT * arm->nonRtConfig.desiredTwist.head<3>();
295 
296  /// Note: checking the duration that arm is tooFar to indicate a new location for intended wiping
297  if (tooFar)
298  {
299  Eigen::Vector2f currentXY = arm->rtStatusInNonRT.currentPose.block<2, 1>(0, 3);
300  if ((s.lastPosition - currentXY).norm() < cfg.changePositionTolerance)
301  {
302  s.changeTimer += dT;
303  }
304  else
305  {
306  s.lastPosition = currentXY;
307  s.changeTimer = 0;
308  }
309 
310  if (s.changeTimer > cfg.changeTimerThreshold)
311  {
312  arm->nonRtConfig.desiredPose.block<2, 1>(0, 3) = currentXY;
313  // isPhaseStop = false; /// Check if desiredPose above in report deviation is correct
314  s.changeTimer = 0;
315  }
316  }
317  else
318  {
319  s.changeTimer = 0;
320  }
321  }
322  else if (_mp.second.mp->getRole() == "nullspace")
323  {
324  auto& arm = limb.at(_mp.second.mp->getNodeSetName());
325  mp::JSMPOutputPtr out =
326  std::dynamic_pointer_cast<mp::JSMPOutput>(_mp.second.output);
327  ARMARX_CHECK_EQUAL(arm->rtStatusInNonRT.numJoints, out->angleRadian.size());
328  arm->nonRtConfig.desiredNullspaceJointAngles.value() = out->angleRadian;
329  }
330  else if (_mp.second.mp->getRole() == "hand")
331  {
332  mp::JSMPOutputPtr out =
333  std::dynamic_pointer_cast<mp::JSMPOutput>(_mp.second.output);
334  auto& hand = hands->hands.at(_mp.second.mp->getNodeSetName());
335  mp::JSMPInputPtr in = std::dynamic_pointer_cast<mp::JSMPInput>(_mp.second.input);
336  ARMARX_CHECK_EQUAL(hand->targetNonRT.jointPosition.value().size(),
337  out->angleRadian.size());
338  hand->targetNonRT.jointPosition.value() = out->angleRadian;
339  }
340  if (mpRunning.at(_mp.second.mp->getMPName()) and _mp.second.mp->isFinished())
341  {
342  ARMARX_INFO << "reset buffer for nonRtConfig of " << _mp.second.mp->getMPName();
343  for (auto& pair : limb)
344  {
345  auto& arm = pair.second;
346  arm->bufferConfigUserToNonRt.reinitAllBuffers(arm->nonRtConfig);
347  }
348  for (auto& pair : limb)
349  {
350  userConfig.limbs.at(pair.first) = pair.second->nonRtConfig;
351  }
352  if (hands)
353  {
354  hands->reinitBuffer(userConfig.hands);
355  }
356  }
357  }
359  }
360 } // namespace armarx::control::njoint_mp_controller::task_space
armarx::control::njoint_controller::task_space::NJointTaskspaceMixedImpedanceVelocityController::userConfig
ConfigDict userConfig
Definition: MixedImpedanceVelocityController.h:174
ARMARX_IMPORTANT
#define ARMARX_IMPORTANT
Definition: Logging.h:190
armarx::control::njoint_controller::task_space::NJointTaskspaceMixedImpedanceVelocityController::limb
std::map< std::string, ArmPtr > limb
Definition: MixedImpedanceVelocityController.h:171
armarx::control::common::mp::JSMPInputPtr
std::shared_ptr< JSMPInput > JSMPInputPtr
Definition: JSMP.h:47
RobotUnit.h
armarx::control::njoint_controller::task_space::NJointTaskspaceMixedImpedanceVelocityController::handleRTNotSafeInNonRT
void handleRTNotSafeInNonRT()
Definition: MixedImpedanceVelocityController.cpp:255
armarx::control::njoint_controller::task_space::NJointTaskspaceMixedImpedanceVelocityController::controllableNodeSets
std::map< std::string, VirtualRobot::RobotNodeSetPtr > controllableNodeSets
Definition: MixedImpedanceVelocityController.h:176
armarx::control::common::detectAndReportPoseDeviationWarning
bool detectAndReportPoseDeviationWarning(const Eigen::Matrix4f &pose1, const Eigen::Matrix4f &pose2, const std::string &namePose1, const std::string &namePose2, float positionThrMM, float angleThrDeg, const std::string &who)
Definition: utils.cpp:646
armarx::max
std::vector< T > max(const std::vector< T > &v1, const std::vector< T > &v2)
Definition: VectorHelpers.h:297
armarx::control::njoint_controller::task_space::NJointTaskspaceCollisionAvoidanceMixedImpedanceVelocityController
Brief description of class NJointTaskspaceCollisionAvoidanceMixedImpedanceVelocityController.
Definition: CollisionAvoidanceMixedImpedanceVelocityController.h:55
armarx::control::njoint_mp_controller::task_space::NJointWipingMixImpVelColMPController::rtPreActivateController
void rtPreActivateController() override
NJointControllerBase interface.
Definition: WipingMixImpVelColController.cpp:84
armarx::control::common::mp::MPPool::runMPs
void runMPs(const bool rtSafe)
Definition: MPPool.cpp:87
armarx::control::common::mp::JSMPOutputPtr
std::shared_ptr< JSMPOutput > JSMPOutputPtr
Definition: JSMP.h:48
armarx::control::common::mp::MPPool::resetMPs
bool resetMPs(const ::armarx::aron::data::dto::DictPtr &dto, const std::map< std::string, VirtualRobot::RobotNodeSetPtr > &rnsMap)
Definition: MPPool.cpp:503
armarx::control::common::mp::TSMPOutputPtr
std::shared_ptr< TSMPOutput > TSMPOutputPtr
Definition: TSMP.h:47
armarx::control::njoint_controller::task_space::NJointTaskspaceMixedImpedanceVelocityController
Brief description of class NJointTaskspaceMixedImpedanceVelocityController.
Definition: MixedImpedanceVelocityController.h:50
WipingMixImpVelColController.h
armarx::control::common::mp::MPPool::mtx_mps
std::recursive_mutex mtx_mps
Definition: MPPool.h:128
armarx::control::njoint_mp_controller::task_space::NJointWipingMixImpVelColMPController::NJointWipingMixImpVelColMPController
NJointWipingMixImpVelColMPController(const RobotUnitPtr &robotUnit, const NJointControllerConfigPtr &config, const VirtualRobot::RobotPtr &)
Definition: WipingMixImpVelColController.cpp:35
armarx::control::common::dVecToMat4
Eigen::Matrix4f dVecToMat4(const mplib::core::DVec &dvec)
create Eigen:Matrix4f from 7D double vector.
Definition: utils.cpp:230
ARMARX_ERROR
#define ARMARX_ERROR
Definition: Logging.h:196
armarx::control::njoint_controller::task_space::NJointTaskspaceMixedImpedanceVelocityController::additionalTaskUpdateStatus
bool additionalTaskUpdateStatus()
Definition: MixedImpedanceVelocityController.cpp:223
utils.h
armarx::control::njoint_mp_controller::task_space
This file is part of ArmarX.
Definition: AdmittanceController.cpp:29
armarx::control::common::MPType::hand
@ hand
armarx::aron::data::DictPtr
std::shared_ptr< Dict > DictPtr
Definition: Dict.h:41
armarx::control::njoint_mp_controller::task_space::NJointWipingMixImpVelColMPController::updateAdaptiveConfig
void updateAdaptiveConfig(const ::armarx::aron::data::dto::DictPtr &dto, const Ice::Current &iceCurrent=Ice::emptyCurrent) override
Definition: WipingMixImpVelColController.cpp:66
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
armarx::control::njoint_controller::task_space::NJointTaskspaceCollisionAvoidanceMixedImpedanceVelocityController::rtPreActivateController
void rtPreActivateController() override
NJointControllerBase interface.
Definition: CollisionAvoidanceMixedImpedanceVelocityController.cpp:630
IceUtil::Handle< class RobotUnit >
armarx::min
std::vector< T > min(const std::vector< T > &v1, const std::vector< T > &v2)
Definition: VectorHelpers.h:327
armarx::control::njoint_controller::task_space::NJointTaskspaceMixedImpedanceVelocityController::hands
core::HandControlPtr hands
Definition: MixedImpedanceVelocityController.h:175
armarx::control::njoint_mp_controller::task_space::NJointWipingMixImpVelColMPController::getClassName
std::string getClassName(const Ice::Current &=Ice::emptyCurrent) const override
Definition: WipingMixImpVelColController.cpp:46
armarx::control::common::mat4ToDVec
mplib::core::DVec mat4ToDVec(const Eigen::Matrix4f &mat)
convert Eigen:Matrix4f to 7D double vector.
Definition: utils.cpp:252
Eigen::Matrix< float, 6, 1 >
armarx::control::njoint_mp_controller::task_space::NJointWipingMixImpVelColMPController::updateMPConfig
void updateMPConfig(const ::armarx::aron::data::dto::DictPtr &dto, const Ice::Current &iceCurrent=Ice::emptyCurrent) override
Definition: WipingMixImpVelColController.cpp:52
armarx::control::common::mp::MPPool::mps
std::map< std::string, MPInputOutput > mps
Definition: MPPool.h:127
armarx::control::njoint_controller::task_space::NJointTaskspaceMixedImpedanceVelocityController::additionalTaskSetTarget
void additionalTaskSetTarget()
Definition: MixedImpedanceVelocityController.cpp:242
armarx::control::common::mp::TSMPInputPtr
std::shared_ptr< TSMPInput > TSMPInputPtr
Definition: TSMP.h:46
ARMARX_CHECK_EQUAL
#define ARMARX_CHECK_EQUAL(lhs, rhs)
This macro evaluates whether lhs is equal (==) rhs and if it turns out to be false it will throw an E...
Definition: ExpressionException.h:130
armarx::control::njoint_controller::task_space::NJointTaskspaceMixedImpedanceVelocityController::robotUnit
RobotUnitPtr robotUnit
Definition: MixedImpedanceVelocityController.h:173
armarx::ctrlutil::s
double s(double t, double s0, double v0, double a0, double j)
Definition: CtrlUtil.h:33
NJointControllerRegistry.h
armarx::control::njoint_mp_controller::task_space::NJointWipingMixImpVelColMPController::AdaptionStatus
Definition: WipingMixImpVelColController.h:54
VirtualRobot::RobotPtr
std::shared_ptr< class Robot > RobotPtr
Definition: Bus.h:19
armarx::control::njoint_mp_controller::task_space::NJointWipingMixImpVelColMPController::additionalTask
void additionalTask() override
Definition: WipingMixImpVelColController.cpp:99
armarx::control::common::mp::MPPool::mpConfig
MPListConfig mpConfig
this variable is only needed when constructing the MP instances, therefore you don't need to use trip...
Definition: MPPool.h:134
armarx::control::njoint_mp_controller::task_space::registrationControllerNJointWipingMixImpVelColMPController
NJointControllerRegistration< NJointWipingMixImpVelColMPController > registrationControllerNJointWipingMixImpVelColMPController("NJointWipingMixImpVelColMPController")