PlayPlatformTrajectory.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  * @package RobotSkillTemplates::PlatformGroup
17  * @author Stefan Reither ( stef dot reither at web dot de )
18  * @date 2017
19  * @copyright http://www.gnu.org/licenses/gpl-2.0.txt
20  * GNU General Public License
21  */
22 
23 #include "PlayPlatformTrajectory.h"
24 
27 
29 
31 
32 using namespace armarx;
33 using namespace PlatformGroup;
34 
35 #define STATE_POSITION 0
36 #define STATE_VELOCITY 1
37 
38 #define MAX_TRANSLATION_ACCEL 0.5 // m/s²
39 #define MAX_ROTATION_ACCEL 0.75 // rad/s²
40 
41 #define FACTOR_BETWEEN_TRANS_AND_ROT 500.0
42 
43 // DO NOT EDIT NEXT LINE
44 PlayPlatformTrajectory::SubClassRegistry
45  PlayPlatformTrajectory::Registry(PlayPlatformTrajectory::GetName(),
47 
50 {
51  ARMARX_INFO << t->output();
52  double length = t->getLength(0);
53  ARMARX_INFO << VAROUT(length);
54  double timelength = t->getTimeLength();
55  ARMARX_INFO << VAROUT(timelength);
56  auto timestamps = t->getTimestamps();
57  ARMARX_INFO << VAROUT(timestamps);
58  Ice::DoubleSeq newTimestamps;
59  newTimestamps.push_back(0);
60  for (size_t var = 0; var < timestamps.size() - 1; ++var)
61  {
62  double tBefore = timestamps.at(var);
63  double tAfter = (timestamps.at(var + 1));
64  ARMARX_INFO << VAROUT(tBefore) << VAROUT(tAfter);
65  double partLength = t->getLength(0, tBefore, tAfter);
66  double lengthPortion = partLength / length;
67  ARMARX_INFO << VAROUT(partLength) << VAROUT(lengthPortion);
68  newTimestamps.push_back(*newTimestamps.rbegin() + timelength * lengthPortion);
69  }
70  ARMARX_INFO << VAROUT(newTimestamps);
71  TrajectoryPtr newTraj = new Trajectory();
72  for (size_t d = 0; d < t->dim(); ++d)
73  {
74  newTraj->addDimension(t->getDimensionData(d), newTimestamps, t->getDimensionName(d));
75  }
76  ARMARX_INFO << newTraj->output();
77 
78  return newTraj;
79 }
80 
82 normalizeTimestamps(TrajectoryPtr t, float maxTransVel, float maxRotVel)
83 {
84  double totalTime = 0.0;
85  auto timestamps = t->getTimestamps();
86  auto dimNames = t->getDimensionNames();
87 
88  for (size_t var = 0; var < timestamps.size() - 1; ++var)
89  {
90  double xLength =
91  t->getLength(std::find(dimNames.begin(), dimNames.end(), "x") - dimNames.begin(),
92  0,
93  timestamps.at(var),
94  timestamps.at(var + 1)) /
95  1000; // conversion mm to m
96  ARMARX_INFO << VAROUT(xLength);
97  double yLength =
98  t->getLength(std::find(dimNames.begin(), dimNames.end(), "y") - dimNames.begin(),
99  0,
100  timestamps.at(var),
101  timestamps.at(var + 1)) /
102  1000; // conversion mm to m
103  ARMARX_INFO << VAROUT(yLength);
104  double alphaLength =
105  t->getLength(std::find(dimNames.begin(), dimNames.end(), "alpha") - dimNames.begin(),
106  0,
107  timestamps.at(var),
108  timestamps.at(var + 1));
109  ARMARX_INFO << VAROUT(alphaLength);
110 
111  double xyLength = std::sqrt(fabs(xLength * xLength + yLength * yLength));
112  ARMARX_INFO << VAROUT(xyLength);
113 
114  double sAcc = 0.5 * maxTransVel * maxTransVel / MAX_TRANSLATION_ACCEL;
115 
116  double xyTime = 0.0;
117 
118  if (xyLength <= 2 * sAcc)
119  {
120  xyTime = 2 * std::sqrt(xyLength / MAX_TRANSLATION_ACCEL);
121  }
122  else
123  {
124  xyTime = 2 * maxTransVel / MAX_TRANSLATION_ACCEL + (xyLength - 2 * sAcc) / maxTransVel;
125  }
126 
127  ARMARX_INFO << VAROUT(xyTime);
128  double rotTime =
129  2 * maxRotVel / MAX_ROTATION_ACCEL +
130  (alphaLength - 0.5 * maxRotVel * maxRotVel / MAX_ROTATION_ACCEL) / maxRotVel;
131  ARMARX_INFO << VAROUT(rotTime);
132 
133  double dtime = (xyTime > rotTime) ? xyTime : rotTime;
134  totalTime += dtime;
135  }
136 
137  t = t->normalize(0.0, totalTime);
138  ARMARX_INFO << t->output();
139 
140  return t;
141 }
142 
145 {
146  Ice::DoubleSeq factors = {1.0, 1.0, factor};
147  ARMARX_INFO << VAROUT(factors);
148  t->scaleValue(factors);
149  return t;
150 }
151 
152 void
154 {
155  TrajectoryPtr platformTraj = in.getTrajectory();
156  auto dimNames = platformTraj->getDimensionNames();
157  if (dimNames.at(0) != "x" || dimNames.at(1) != "y" || dimNames.at(2) != "alpha")
158  {
159  ARMARX_WARNING << "Trajectory misses at least one of the following needed dimensions: 'x', "
160  "'y' and 'alpha'.";
161  emitFailure();
162  }
163 }
164 
165 void
167 {
168  PlatformContext* c = getContext<PlatformContext>();
169  PlatformUnitInterfacePrx platformUnit = c->platformUnitPrx;
170  TrajectoryPtr platformTraj = in.getTrajectory();
171  auto dimNames = platformTraj->getDimensionNames();
172  float maxTransVel = in.getMaxTranslationVelocity();
173  float maxRotVel = in.getMaxRotationVelocity();
174 
175  if (in.getBalanceTrajectory())
176  {
177  // Scaling rotation dimension of the trajectory to fit to the translation trajectory
178 
179  platformTraj = scaleRotationLengths(platformTraj, FACTOR_BETWEEN_TRANS_AND_ROT);
180 
181  // Balancing timestamps of the trajetory
182 
183  platformTraj = balanceTimestamps(platformTraj);
184 
185  // Rescaling rotation dimension of the trajectory
186 
187  platformTraj =
188  scaleRotationLengths(platformTraj, (float)(1 / FACTOR_BETWEEN_TRANS_AND_ROT));
189 
190  // calculate approximation of duration of the movement given a maximal velocity and a maximal acceleration
191 
192  platformTraj = normalizeTimestamps(platformTraj, maxTransVel, maxRotVel);
193  }
194 
195  // Executing trajectory
196 
197 
198  MultiDimPIDController pidTrans(in.getPTranslation(),
199  in.getITranslation(),
200  in.getDTranslation(),
201  maxTransVel,
202  MAX_TRANSLATION_ACCEL * 1000);
203  PIDController pidRot(in.getPOrientation(),
204  in.getIOrientation(),
205  in.getDOrientation(),
206  maxRotVel,
208  DatafieldRefPtr posXRef =
209  new DatafieldRef(c->platformUnitObserverPrx, "platformPose", "positionX");
210  DatafieldRefPtr posYRef =
211  new DatafieldRef(c->platformUnitObserverPrx, "platformPose", "positionY");
212  DatafieldRefPtr oriRef =
213  new DatafieldRef(c->platformUnitObserverPrx, "platformPose", "rotation");
214 
215  NameValueMap targetValues;
217  double currentTime = 0.0;
218  bool finalPoseReached = false;
219  int cycleTime = in.getCycleTimeMS();
220  CycleUtil cycle(cycleTime);
221 
222  c->debugDrawer->removeLayer("ActualRobotPath");
223 
224  // Vector3Ptr positionOld;
225 
226  IceUtil::Time cycleStart;
227  while (!isRunningTaskStopped()) // stop run function if returning true
228  {
229  cycleStart = armarx::TimeUtil::GetTime();
230 
231  currentTime = (armarx::TimeUtil::GetTime() - startTime).toSecondsDouble();
232  if (currentTime > platformTraj->getTimeLength())
233  {
234  currentTime = platformTraj->getTimeLength();
235  finalPoseReached = true;
236  }
237 
238  float posx = posXRef->getDataField()->getFloat();
239  float posy = posYRef->getDataField()->getFloat();
240  float ori = oriRef->getDataField()->getFloat();
241 
242  // Vector3Ptr v = new Vector3 {posx, posy, 50.f};
243  // if (positionOld)
244  // {
245  // c->debugDrawer->setLineVisu("ActualRobotPath", boost::lexical_cast<std::string>(currentTime), positionOld, v, 2.0, DrawColor {0, 1, 0, 1});
246  // }
247  // positionOld = v;
248 
249  std::vector<Ice::DoubleSeq> allStates = platformTraj->getAllStates(currentTime, 1);
250  for (size_t i = 0; i < platformTraj->dim(); ++i)
251  {
252  targetValues[dimNames.at(i)] = allStates[i][STATE_POSITION];
253  }
254 
255  Eigen::Vector3f pos{posx, posy, 0.0f};
256  Eigen::Vector3f targetPos{targetValues["x"], targetValues["y"], 0.0f};
257 
258  pidTrans.update(pos, targetPos);
259  pidRot.update(ori, targetValues["alpha"]);
260  float newRotVel = pidRot.getControlValue();
261  Eigen::Vector3f globalNewTransVel = {
262  pidTrans.getControlValue()[0], pidTrans.getControlValue()[1], 0};
263  Eigen::Matrix3f m;
264  m = Eigen::AngleAxisf(ori, Eigen::Vector3f::UnitZ());
265  Eigen::Vector3f localVel = m.inverse() * globalNewTransVel;
266 
267 
268  ARMARX_INFO << "New values: x: " << localVel[0] << " , y= " << localVel[1]
269  << " , alpha= " << newRotVel;
270 
271  platformUnit->move(localVel[0], localVel[1], newRotVel);
272 
273  if (finalPoseReached && fabs(pidTrans.previousError) < in.getPositionalAccuracy() &&
274  fabs(pidRot.previousError) < in.getOrientationalAccuracy())
275  {
276  disableRunFunction();
277  platformUnit->move(0, 0, 0);
278  ARMARX_INFO << "Success";
279  emitSuccess();
280  }
281  else
282  {
283  cycle.waitForCycleDuration();
284  }
285  }
286 }
287 
288 void
290 {
291  // Setting velocities to zero
292  PlatformContext* c = getContext<PlatformContext>();
293  PlatformUnitInterfacePrx platformUnit = c->platformUnitPrx;
294  platformUnit->move(0.0, 0.0, 0.0);
295 }
296 
297 // DO NOT EDIT NEXT FUNCTION
300 {
301  return XMLStateFactoryBasePtr(new PlayPlatformTrajectory(stateData));
302 }
armarx::PlatformGroup::PlayPlatformTrajectory::onEnter
void onEnter() override
Definition: PlayPlatformTrajectory.cpp:153
MAX_TRANSLATION_ACCEL
#define MAX_TRANSLATION_ACCEL
Definition: PlayPlatformTrajectory.cpp:38
armarx::MultiDimPIDControllerTemplate::update
void update(const double deltaSec, const PIDVectorX &measuredValue, const PIDVectorX &targetValue)
Definition: MultiDimPIDController.h:79
armarx::MultiDimPIDControllerTemplate<>
armarx::XMLStateConstructorParams
Definition: XMLState.h:49
armarx::PlatformGroup::PlayPlatformTrajectory::run
void run() override
Definition: PlayPlatformTrajectory.cpp:166
armarx::CycleUtil
This util class helps with keeping a cycle time during a control cycle.
Definition: CycleUtil.h:40
c
constexpr T c
Definition: UnscentedKalmanFilterTest.cpp:46
armarx::PIDController::getControlValue
double getControlValue() const
Definition: PIDController.cpp:297
armarx::PIDController::update
void update(double deltaSec, double measuredValue, double targetValue)
Definition: PIDController.cpp:232
PlayPlatformTrajectory.h
armarx::XMLStateFactoryBasePtr
IceInternal::Handle< XMLStateFactoryBase > XMLStateFactoryBasePtr
Definition: XMLState.h:64
armarx::MultiDimPIDControllerTemplate::previousError
double previousError
Definition: MultiDimPIDController.h:232
scaleRotationLengths
TrajectoryPtr scaleRotationLengths(TrajectoryPtr t, float factor)
Definition: PlayPlatformTrajectory.cpp:144
IceInternal::Handle< Trajectory >
DatafieldRef.h
normalizeTimestamps
TrajectoryPtr normalizeTimestamps(TrajectoryPtr t, float maxTransVel, float maxRotVel)
Definition: PlayPlatformTrajectory.cpp:82
armarx::PIDController
Definition: PIDController.h:43
balanceTimestamps
TrajectoryPtr balanceTimestamps(TrajectoryPtr t)
Definition: PlayPlatformTrajectory.cpp:49
armarx::PlatformGroup::PlayPlatformTrajectory::PlayPlatformTrajectory
PlayPlatformTrajectory(const XMLStateConstructorParams &stateData)
Definition: PlayPlatformTrajectory.h:35
armarx::CycleUtil::waitForCycleDuration
IceUtil::Time waitForCycleDuration()
This function will wait (virtual or system time) until the cycle time is reached.
Definition: CycleUtil.cpp:53
armarx::PIDController::previousError
double previousError
Definition: PIDController.h:73
armarx::armem::Time
armarx::core::time::DateTime Time
Definition: forward_declarations.h:13
armarx::VariantType::Trajectory
const VariantTypeId Trajectory
Definition: Trajectory.h:44
armarx::TimeUtil::GetTime
static IceUtil::Time GetTime(TimeMode timeMode=TimeMode::VirtualTime)
Get the current time.
Definition: TimeUtil.cpp:42
armarx::VariantType::DatafieldRef
const VariantTypeId DatafieldRef
Definition: DatafieldRef.h:197
CycleUtil.h
PIDController.h
armarx::control::njoint_controller::platform::platform_follower_controller::NameValueMap
std::map< std::string, float > NameValueMap
Definition: PlatformFollowerController.h:88
GfxTL::sqrt
VectorXD< D, T > sqrt(const VectorXD< D, T > &a)
Definition: VectorXD.h:704
PlatformContext.h
ARMARX_INFO
#define ARMARX_INFO
Definition: Logging.h:181
armarx::MultiDimPIDControllerTemplate::getControlValue
const PIDVectorX & getControlValue() const
Definition: MultiDimPIDController.h:201
VAROUT
#define VAROUT(x)
Definition: StringHelpers.h:198
armarx::PlatformContext
Definition: PlatformContext.h:77
MAX_ROTATION_ACCEL
#define MAX_ROTATION_ACCEL
Definition: PlayPlatformTrajectory.cpp:39
GfxTL::Matrix3f
MatrixXX< 3, 3, float > Matrix3f
Definition: MatrixXX.h:649
ARMARX_WARNING
#define ARMARX_WARNING
Definition: Logging.h:193
FACTOR_BETWEEN_TRANS_AND_ROT
#define FACTOR_BETWEEN_TRANS_AND_ROT
Definition: PlayPlatformTrajectory.cpp:41
armarx::PlatformGroup::PlayPlatformTrajectory::CreateInstance
static XMLStateFactoryBasePtr CreateInstance(XMLStateConstructorParams stateData)
Definition: PlayPlatformTrajectory.cpp:299
armarx
This file offers overloads of toIce() and fromIce() functions for STL container types.
Definition: ArmarXTimeserver.cpp:27
armarx::PlatformGroup::PlayPlatformTrajectory::onExit
void onExit() override
Definition: PlayPlatformTrajectory.cpp:289
armarx::PlatformGroup::PlayPlatformTrajectory::Registry
static SubClassRegistry Registry
Definition: PlayPlatformTrajectory.h:49
STATE_POSITION
#define STATE_POSITION
Definition: PlayPlatformTrajectory.cpp:35