mpi_cpp_tools
dynamical_systems.hpp
Go to the documentation of this file.
1 
9 #pragma once
10 
11 #include <cmath>
12 #include <math.h>
13 #include <Eigen/Eigen>
14 #include <iostream>
15 
17 #include "mpi_cpp_tools/math.hpp"
18 
19 
20 namespace mct
21 {
22 
23 
24 
25 
26 
27 
29 {
30 public:
31  typedef Eigen::Matrix<double, Eigen::Dynamic, 1, 0, 10> Vector;
32 
33  LinearDynamics(Eigen::Vector4d parameters): LinearDynamics(parameters[0],
34  parameters[1],
35  parameters[2],
36  parameters[3]) { }
37 
38 
39  LinearDynamics(double jerk,
40  double initial_acceleration,
41  double initial_velocity,
42  double initial_position)
43  {
44  jerk_ = jerk;
45  initial_acceleration_ = initial_acceleration;
46  initial_velocity_ = initial_velocity;
47  initial_position_= initial_position;
48  }
49  double get_acceleration(mct::NonnegDouble t) const
50  {
51  return jerk_ * t +
52  initial_acceleration_;
53  }
54  double get_velocity(mct::NonnegDouble t) const
55  {
56  return jerk_ * 0.5 * t * t +
57  initial_acceleration_ * t +
58  initial_velocity_;
59  }
60  double get_position(mct::NonnegDouble t) const
61  {
62  return jerk_ * 0.5 * 1./3. * t * t * t +
63  initial_acceleration_ * 0.5 * t * t +
64  initial_velocity_ * t +
65  initial_position_;
66  }
67 
68  Vector find_t_given_velocity(double velocity) const
69  {
70  double a = jerk_ * 0.5;
71  double b = initial_acceleration_;
72  double c = initial_velocity_ - velocity;
73 
74  double determinant = b * b - 4 * a * c;
75 
76  Vector solutions(Vector::Index(0));
77  if(a == 0)
78  {
79  if(b != 0)
80  {
81  solutions.resize(1);
82  solutions[0] = - c / b;
83  }
84 
85  }
86  else if(determinant == 0)
87  {
88  solutions.resize(1);
89  solutions[0] = -b / 2 / a;
90  }
91  else if(determinant > 0)
92  {
93  double determinant_sqrt = std::sqrt(determinant);
94  solutions.resize(2);
95  solutions[0] = (-b + determinant_sqrt) / 2 / a;
96  solutions[1] = (-b - determinant_sqrt) / 2 / a;
97  }
98 
99  Vector positive_solutions(Vector::Index(0));
100  for(size_t i = 0; i < solutions.size(); i++)
101  {
102  if(solutions[i] >= 0)
103  {
104  mct::append_to_vector(positive_solutions, solutions[i]);
105  }
106  }
107 
108  return positive_solutions;
109  }
110 
111 protected:
112  double jerk_;
113  double initial_acceleration_;
114  double initial_velocity_;
115  double initial_position_;
116 };
117 
118 
119 
120 
121 
122 
124 {
125 public:
126 
127  void print_parameters() const
128  {
129  std::cout << "-------------------------------------------" << std::endl;
130  std::cout << "jerk: " << jerk_ << std::endl
131  << "initial_acceleration: " << initial_acceleration_ << std::endl
132  << "initial_velocity: " << initial_velocity_ << std::endl
133  << "initial_position: " << initial_position_ << std::endl
134  << "acceleration_limit: " << acceleration_limit_ << std::endl
135  << "jerk_duration: " << jerk_duration_ << std::endl;
136  std::cout << "-------------------------------------------" << std::endl;
137  }
138 
139  typedef LinearDynamics::Vector Vector;
140 
141  typedef Eigen::Matrix<double, Eigen::Dynamic, Eigen::Dynamic, 0, 10, 10>
142  Matrix;
143 
144 
145  LinearDynamicsWithAccelerationConstraint(Eigen::Matrix<double, 5, 1> parameters):
147  parameters[1],
148  parameters[2],
149  parameters[3],
150  parameters[4]) { }
151 
153  double initial_acceleration,
154  double initial_velocity,
155  double initial_position,
156  mct::NonnegDouble abs_acceleration_limit):
157  LinearDynamics(jerk,
158  initial_acceleration,
159  initial_velocity,
160  initial_position)
161  {
162  if(jerk_ > 0)
163  acceleration_limit_ = abs_acceleration_limit;
164  else
165  acceleration_limit_ = -abs_acceleration_limit;
166 
167  set_initial_acceleration(initial_acceleration);
168  }
169 
170 
171  void set_initial_acceleration(double initial_acceleration)
172  {
173  if(std::fabs(initial_acceleration) > std::fabs(acceleration_limit_))
174  throw std::invalid_argument("expected "
175  "std::fabs(initial_acceleration) > "
176  "abs_acceleration_limit");
177  initial_acceleration_ = initial_acceleration;
178  jerk_duration_ =
179  (acceleration_limit_ - initial_acceleration_) / jerk_;
180  }
181 
182  double get_acceleration(mct::NonnegDouble t) const
183  {
184  if(t < jerk_duration_)
185  {
186  return LinearDynamics::get_acceleration(t);
187  }
188  else
189  {
190  return acceleration_limit_;
191  }
192  }
193  double get_velocity(mct::NonnegDouble t) const
194  {
195  if(t < jerk_duration_)
196  {
197  return LinearDynamics::get_velocity(t);
198  }
199  else
200  {
201  return LinearDynamics::get_velocity(jerk_duration_) +
202  acceleration_limit_ * (t - jerk_duration_);
203  }
204  }
205  double get_position(mct::NonnegDouble t) const
206  {
207  if(t < jerk_duration_)
208  {
209  return LinearDynamics::get_position(t);
210  }
211  else
212  {
213  return LinearDynamics::get_position(jerk_duration_) +
214  LinearDynamics::get_velocity(jerk_duration_) * (t - jerk_duration_) +
215  acceleration_limit_ * 0.5 * (t - jerk_duration_) * (t - jerk_duration_);
216  }
217  }
218 
219  template<typename Array>
220  Array get_positions(const Array& times) const
221  {
222  Array positions(times.size());
223  for(size_t i = 0; i < times.size(); i++)
224  {
225  positions[i] = get_position(times[i]);
226  }
227  return positions;
228  }
229 
230  Vector find_t_given_velocity(double velocity) const
231  {
232  Vector potential_solutions =
233  LinearDynamics::find_t_given_velocity(velocity);
234 
235  Vector solutions(Vector::Index(0));
236  for(size_t i = 0; i < potential_solutions.size(); i++)
237  {
238  if(potential_solutions[i] <= jerk_duration_)
239  {
240  mct::append_to_vector(solutions, potential_solutions[i]);
241  }
242  }
243 
244  double potential_solution = jerk_duration_
245  + (velocity - LinearDynamics::get_velocity(jerk_duration_))
246  / acceleration_limit_;
247  if(potential_solution > jerk_duration_ &&
248  !(mct::contains(solutions, jerk_duration_) &&
249  mct::approx_equal(potential_solution, jerk_duration_)))
250  {
251  mct::append_to_vector(solutions, potential_solution);
252  }
253 
254 
255 
256  if(solutions.size() > 2)
257  {
258  std::cout << "too many solutions, something went wrong!!!"
259  << std::endl;
260  print_parameters();
261 
262  std::cout << "potential_solutions[0]: " << potential_solutions[0] << std::endl;
263 
264  std::cout << "potential_solutions size: " << potential_solutions.size() << " content: " << potential_solutions.transpose() << std::endl;
265 
266 
267  std::cout << "solutions size: " << solutions.size() << " content: " << solutions.transpose() << std::endl;
268  exit(-1);
269  }
270  return solutions;
271  }
272 
273  bool will_exceed_jointly(const double& max_velocity,
274  const double& max_position) const
275  {
276  double certificate_time;
277  return will_exceed_jointly(max_velocity, max_position, certificate_time);
278  }
279 
280 
281  bool will_exceed_jointly(const double& max_velocity,
282  const double& max_position,
283  double& certificate_time) const
284  {
285  if(max_velocity == std::numeric_limits<double>::infinity() ||
286  max_position == std::numeric_limits<double>::infinity())
287  {
288  return false;
289  }
290  if(jerk_ > 0)
291  {
292  certificate_time = std::numeric_limits<double>::infinity();
293  return true;
294  }
295  if(jerk_ == 0)
296  {
297  throw std::domain_error("not implemented for jerk == 0");
298  }
299 
300  // find maximum achieved position --------------------------------------
302  // Matrix candidate_points(0, 0);
303  // Vector candidate_times(0);
304 
305  // mct::append_rows_to_matrix(candidate_points,
306  // Eigen::Vector2d(initial_velocity_,
307  // initial_position_).transpose());
308  // mct::append_to_vector(candidate_times, 0);
309 
310 
311  if(initial_velocity_ > max_velocity &&
312  initial_position_ > max_position)
313  {
314  certificate_time = 0;
315  return true;
316  }
317 
318  Vector t_given_zero_velocity = find_t_given_velocity(0);
319  if(t_given_zero_velocity.size() > 0)
320  {
321  Vector position_given_zero_velocity =
322  get_positions(t_given_zero_velocity);
323 
324  Vector::Index max_index;
325  double max_achieved_position =
326  position_given_zero_velocity.maxCoeff(&max_index);
327  if(max_achieved_position < max_position)
328  {
329  return false;
330  }
331  if(max_velocity < 0)
332  {
333  certificate_time = t_given_zero_velocity[max_index];
334  return true;
335  }
336  }
337 
338  Vector t_given_max_velocity =
339  find_t_given_velocity(max_velocity);
340  Vector position_given_max_velocity =
341  get_positions(t_given_max_velocity);
342 
343  for(size_t i = 0; i < position_given_max_velocity.size(); i++)
344  {
345  if(position_given_max_velocity[i] > max_position)
346  {
347  certificate_time = t_given_max_velocity[i];
348  return true;
349  }
350  }
351 
352  return false;
353  }
354 
355 
356  bool will_deceed_jointly(const double& min_velocity,
357  const double& min_position) const
358  {
359  double certificate_time;
360  return will_deceed_jointly(min_velocity, min_position, certificate_time);
361  }
362 
363  bool will_deceed_jointly(const double& min_velocity,
364  const double& min_position,
365  double& certificate_time) const
366  {
368  flipped_dynamics(-jerk_,
369  -initial_acceleration_,
370  -initial_velocity_,
371  -initial_position_,
372  std::fabs(acceleration_limit_));
373 
374  return flipped_dynamics.will_exceed_jointly(-min_velocity,
375  -min_position,
376  certificate_time);
377  }
378 
379 
380 private:
381  double acceleration_limit_;
382  mct::NonnegDouble jerk_duration_;
383 };
384 
385 
386 
388  const double& initial_velocity,
389  const double& initial_position,
390  const double& max_velocity,
391  const double& max_position,
392  const mct::NonnegDouble& abs_jerk_limit,
393  const mct::NonnegDouble& abs_acceleration_limit)
394 {
395  double lower = -abs_acceleration_limit;
396  double upper = abs_acceleration_limit;
397 
398 
399  LinearDynamicsWithAccelerationConstraint dynamics(-abs_jerk_limit,
400  lower,
401  initial_velocity,
402  initial_position,
403  abs_acceleration_limit);
404 
405 
406  if(dynamics.will_exceed_jointly(max_velocity, max_position))
407  {
409  return lower;
410  }
411 
412  dynamics.set_initial_acceleration(upper);
413  if(!dynamics.will_exceed_jointly(max_velocity, max_position))
414  {
415  return upper;
416  }
417 
418  for(size_t i = 0; i < 20; i++)
419  {
420  double middle = (lower + upper) / 2.0;
421 
422  dynamics.set_initial_acceleration(middle);
423  if(dynamics.will_exceed_jointly(max_velocity, max_position))
424  {
425  upper = middle;
426  }
427  else
428  {
429  lower = middle;
430  }
431  }
432  return lower;
433 }
434 
435 
436 
437 double find_min_admissible_acceleration(
438  const double& initial_velocity,
439  const double& initial_position,
440  const double& min_velocity,
441  const double& min_position,
442  const mct::NonnegDouble& abs_jerk_limit,
443  const mct::NonnegDouble& abs_acceleration_limit)
444 {
445  return -find_max_admissible_acceleration(-initial_velocity,
446  -initial_position,
447  -min_velocity,
448  -min_position,
449  abs_jerk_limit,
450  abs_acceleration_limit);
451 }
452 
453 
454 
456 {
457 public:
459  {
460  min_velocity_ = -std::numeric_limits<double>::infinity();
461  min_position_ = -std::numeric_limits<double>::infinity();
462  max_velocity_ = std::numeric_limits<double>::infinity();
463  max_position_ = std::numeric_limits<double>::infinity();
464  max_torque_ = 1.0;
465  max_jerk_ = 1.0;
466  inertia_ = 1.0;
467  }
468 
469  SafetyConstraint(double min_velocity,
470  double min_position,
471  double max_velocity,
472  double max_position,
473  mct::NonnegDouble max_torque,
474  mct::NonnegDouble max_jerk,
475  mct::NonnegDouble inertia)
476  {
477  min_velocity_ = min_velocity;
478  min_position_ = min_position;
479  max_velocity_ = max_velocity;
480  max_position_ = max_position;
481  max_torque_ = max_torque;
482  max_jerk_ = max_jerk;
483  inertia_ = inertia;
484  }
485 
486 
487  double get_safe_torque(const double& torque,
488  const double& velocity,
489  const double& position)
490  {
491  double safe_torque = mct::clamp(torque, -max_torque_, max_torque_);
492 
493 // std::cout << "safe_torque: " << safe_torque << std::endl;
494 
495 
496  mct::NonnegDouble max_achievable_acc = max_torque_ / inertia_;
497 
498 
499 
500 
501  double max_admissible_acc =
503  position,
504  max_velocity_,
505  max_position_,
506  max_jerk_,
507  max_achievable_acc);
508  double max_admissible_torque = max_admissible_acc * inertia_;
509 
510 
511 
512 // LinearDynamicsWithAccelerationConstraint
513 // test_dynamics(- max_jerk_,
514 // max_achievable_acc,
515 // velocity,
516 // position,
517 // max_achievable_acc);
518 
519 // test_dynamics.print_parameters();
520 // std::cout << "will exceed: " << test_dynamics.will_exceed_jointly(max_velocity_, max_position_) << std::endl;
521 // std::cout << "max_admissible_acc: " << max_admissible_acc << std::endl;
522 
523 
524 // std::cout << "max_achievable_acc: " << max_achievable_acc << std::endl;
525 // std::cout << "max_admissible_acc: " << max_admissible_acc << std::endl;
526 // std::cout << "max_admissible_torque: " << max_admissible_torque << std::endl;
527 
528 
529 
530  double min_admissible_acc =
531  find_min_admissible_acceleration(velocity,
532  position,
533  min_velocity_,
534  min_position_,
535  max_jerk_,
536  max_achievable_acc);
537  double min_admissible_torque = min_admissible_acc * inertia_;
538 
539 // std::cout << "min_admissible_acc: " << min_admissible_acc << std::endl;
540 // std::cout << "min_admissible_torque: " << min_admissible_torque << std::endl;
541 
542 
543 
544  if(min_admissible_torque > max_admissible_torque)
545  {
546  std::cout << "min_admissible_torque > max_admissible_torque!!!!"
547  << std::endl;
548  return 0;
549  }
550 
551  safe_torque = mct::clamp(safe_torque,
552  min_admissible_torque, max_admissible_torque);
553 
554 
555  if(safe_torque > max_torque_ || safe_torque < -max_torque_)
556  {
557  std::cout << "something went horribly horribly wrong " << std::endl;
558  return 0;
559  }
560 
561  return safe_torque;
562  }
563 
564  double min_velocity_;
565  double min_position_;
566  double max_velocity_;
567  double max_position_;
568  mct::NonnegDouble max_torque_;
569  mct::NonnegDouble max_jerk_;
570  mct::NonnegDouble inertia_;
571 };
572 
573 
574 
575 }
Definition: basic_tools.hpp:14
double find_max_admissible_acceleration(const double &initial_velocity, const double &initial_position, const double &max_velocity, const double &max_position, const mct::NonnegDouble &abs_jerk_limit, const mct::NonnegDouble &abs_acceleration_limit)
Definition: dynamical_systems.hpp:387
Definition: dynamical_systems.hpp:28
bool will_exceed_jointly(const double &max_velocity, const double &max_position, double &certificate_time) const
Definition: dynamical_systems.hpp:281
Definition: basic_tools.hpp:17
Definition: dynamical_systems.hpp:455
Definition: dynamical_systems.hpp:123