Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

KinematicTrajectoryOptimization adds a linear constraint on velocity. #22377

Open
wants to merge 1 commit into
base: master
Choose a base branch
from

Conversation

hongkai-dai
Copy link
Contributor

@hongkai-dai hongkai-dai commented Jan 2, 2025

This imposes a linear constraint on our decision variables.

This is inspired by the stackoverflow question https://stackoverflow.com/questions/79317170/question-about-kinematictrajectoryoptimization-addpathvelocityconstraint


This change is Reviewable

Copy link
Contributor Author

@hongkai-dai hongkai-dai left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

+@RussTedrake for feature review please, thanks!

Reviewable status: LGTM missing from assignee RussTedrake(platform), needs at least two assigned reviewers, missing label for release notes (waiting on @hongkai-dai)

Copy link
Contributor

@RussTedrake RussTedrake left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Checkpoint. Maybe best to answer the API questions first, in case it changes the implementation?

Reviewed all commit messages.
Reviewable status: 3 unresolved discussions, LGTM missing from assignee RussTedrake(platform), needs at least two assigned reviewers, missing label for release notes (waiting on @hongkai-dai)


planning/trajectory_optimization/kinematic_trajectory_optimization.cc line 352 at r1 (raw file):

  auto [vars_vel, _] = symbolic::ExtractVariablesFromExpression(rdot);
  Eigen::MatrixXd M_vel(num_positions(), vars_vel.size());
  symbolic::DecomposeLinearExpressions(rdot, vars_vel, &M_vel);

btw -- for bezier curve, we got rid of the symbolic (see AsLinearInControlPoints), but I guess we don't have it for bsplinebasis yet. :-(


planning/trajectory_optimization/kinematic_trajectory_optimization.h line 147 at r1 (raw file):

  /** Adds a linear constraint on trajectory velocity `q̇(t)`, evaluated at
  `s`. The constraint will be evaluated as if it is bound with variables
  corresponding to `q̇(T*s)` (as opposed to [q(T*s), q̇(T*s)] in

oh. that seems very confusing, no? I would think that the bindings should still be [q, qdot] here? there is no real cost to having some of the elements of the constraint be zero?


planning/trajectory_optimization/kinematic_trajectory_optimization.h line 157 at r1 (raw file):

  solvers::Binding<solvers::LinearConstraint>
  AddVelocityLinearConstraintAtNormalizedTime(
      const std::shared_ptr<solvers::LinearConstraint>& constraint, double s);

Wouldn't it be more clear/helpful to just take A and lb/ub?

This imposes a linear constraint on our decision variables.
@hongkai-dai hongkai-dai force-pushed the kin_trajopt_add_velocity_linear_constraint branch from 4f95a67 to 6363af3 Compare January 3, 2025 05:27
Copy link
Contributor Author

@hongkai-dai hongkai-dai left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewable status: 2 unresolved discussions, LGTM missing from assignee RussTedrake(platform), needs at least two assigned reviewers, missing label for release notes


planning/trajectory_optimization/kinematic_trajectory_optimization.h line 147 at r1 (raw file):

Previously, RussTedrake (Russ Tedrake) wrote…

oh. that seems very confusing, no? I would think that the bindings should still be [q, qdot] here? there is no real cost to having some of the elements of the constraint be zero?

The issue of associating the constraint with q is that the constraint becomes nonlinear.

If we were to associate the constraint with [q, qdot], then the constraint becomes

lb <= A * [q; qdot] <= ub

since q = r, qdot = rdot / T, we have

lb <= A * [r, rdot/T] <= ub

rearranging the equation we get

A * [r * T, rdot] - lb * T >= 0
A * [r * T, rodt] - ub * T <= 0

Notice that r and T multiplies together, making the constraint nonlinear.

Maybe I should rename the function? What about AddVelocityLinearConstraint?


planning/trajectory_optimization/kinematic_trajectory_optimization.h line 157 at r1 (raw file):

Previously, RussTedrake (Russ Tedrake) wrote…

Wouldn't it be more clear/helpful to just take A and lb/ub?

My rationale was that if we want to specify lb <= qdot <= ub, then it is easier to pass a BoundingBoxConstraint (which only requires specifying lb and ub, but not A), than passing A, lb, ub. Does that make sense?


planning/trajectory_optimization/kinematic_trajectory_optimization.cc line 352 at r1 (raw file):

Previously, RussTedrake (Russ Tedrake) wrote…

btw -- for bezier curve, we got rid of the symbolic (see AsLinearInControlPoints), but I guess we don't have it for bsplinebasis yet. :-(

Done. I added a TODO here.

Copy link
Contributor

@RussTedrake RussTedrake left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewed 1 of 5 files at r1, 1 of 1 files at r2.
Reviewable status: 2 unresolved discussions, LGTM missing from assignee RussTedrake(platform), needs at least two assigned reviewers, missing label for release notes (waiting on @hongkai-dai)


planning/trajectory_optimization/kinematic_trajectory_optimization.h line 147 at r1 (raw file):

Previously, hongkai-dai (Hongkai Dai) wrote…

The issue of associating the constraint with q is that the constraint becomes nonlinear.

If we were to associate the constraint with [q, qdot], then the constraint becomes

lb <= A * [q; qdot] <= ub

since q = r, qdot = rdot / T, we have

lb <= A * [r, rdot/T] <= ub

rearranging the equation we get

A * [r * T, rdot] - lb * T >= 0
A * [r * T, rodt] - ub * T <= 0

Notice that r and T multiplies together, making the constraint nonlinear.

Maybe I should rename the function? What about AddVelocityLinearConstraint?

I see. Thanks for the clear explanation, and sorry I was slow!

I do think that the shortest path forward would be to find yet another name to distinguish this. But as I look at the overall API right now, what I actually think would be better is if we make some placeholder variables for q and qdot, etc, and then allowed people to pass bindings instead of constraints to this method. Then they could keep the same name AddVelocityConstraintAtNormalizedTime(), but the variable assignments would be clear. (and this one would have to throw if the binding included q vars). For that matter, the method you've written already can fail fast if the number of variables doesn't match.

Proposal: if you were interested/willing to do the placeholder variables now, that would be my first pick. But to move this forward, I would be ok adding the method with your originally proposed naming, and we add a TODO to replace these with placeholder variable variants in the near future?

Wdyt?


planning/trajectory_optimization/kinematic_trajectory_optimization.h line 157 at r1 (raw file):

Previously, hongkai-dai (Hongkai Dai) wrote…

My rationale was that if we want to specify lb <= qdot <= ub, then it is easier to pass a BoundingBoxConstraint (which only requires specifying lb and ub, but not A), than passing A, lb, ub. Does that make sense?

I would think that lb <= qdot <= ub should probably be a special case that deserves it's own entry point, since we already have so many methods like that (e.g. AddPathVelocityConstraint). But I take your point that this could work for linearcontraint, linearequalityconstraint, and bounding box constraints.

Copy link
Contributor Author

@hongkai-dai hongkai-dai left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewable status: 2 unresolved discussions, LGTM missing from assignee RussTedrake(platform), needs at least two assigned reviewers, missing label for release notes


planning/trajectory_optimization/kinematic_trajectory_optimization.h line 147 at r1 (raw file):

Previously, RussTedrake (Russ Tedrake) wrote…

I see. Thanks for the clear explanation, and sorry I was slow!

I do think that the shortest path forward would be to find yet another name to distinguish this. But as I look at the overall API right now, what I actually think would be better is if we make some placeholder variables for q and qdot, etc, and then allowed people to pass bindings instead of constraints to this method. Then they could keep the same name AddVelocityConstraintAtNormalizedTime(), but the variable assignments would be clear. (and this one would have to throw if the binding included q vars). For that matter, the method you've written already can fail fast if the number of variables doesn't match.

Proposal: if you were interested/willing to do the placeholder variables now, that would be my first pick. But to move this forward, I would be ok adding the method with your originally proposed naming, and we add a TODO to replace these with placeholder variable variants in the near future?

Wdyt?

Sorry for my belated reply.

Just want to make sure that we are on the same page, here is what I think the proposal is

  1. Add placeholder variables q and qdot.
  2. Use one function call
Binding<Constraint> AddVelocityConstraintAtNormalizedTime(const Binding<Constraint>& binding, double s);

binding can associate with some or all of placeholder q and qdot. The behavior of this function can be

  1. If binding associates with only qdot, and binding.evaluator() is a linear constraint, then we impose a linear constraint on the actual decision variables.
  2. If binding associates with both q and qdot, then we impose a generic nonlinear constraint (using the WrappedVelocityConstraint you have implemented).
  3. If binding associates with only q, then we can still impose the constraint, but maybe it is better to throw an error here, as the function name suggests that it should be associated with qdot.

What do you think?

Copy link
Contributor

@RussTedrake RussTedrake left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Reviewed all commit messages.
Reviewable status: 2 unresolved discussions, LGTM missing from assignee RussTedrake(platform), needs at least two assigned reviewers, missing label for release notes (waiting on @hongkai-dai)


planning/trajectory_optimization/kinematic_trajectory_optimization.h line 147 at r1 (raw file):

Previously, hongkai-dai (Hongkai Dai) wrote…

Sorry for my belated reply.

Just want to make sure that we are on the same page, here is what I think the proposal is

  1. Add placeholder variables q and qdot.
  2. Use one function call
Binding<Constraint> AddVelocityConstraintAtNormalizedTime(const Binding<Constraint>& binding, double s);

binding can associate with some or all of placeholder q and qdot. The behavior of this function can be

  1. If binding associates with only qdot, and binding.evaluator() is a linear constraint, then we impose a linear constraint on the actual decision variables.
  2. If binding associates with both q and qdot, then we impose a generic nonlinear constraint (using the WrappedVelocityConstraint you have implemented).
  3. If binding associates with only q, then we can still impose the constraint, but maybe it is better to throw an error here, as the function name suggests that it should be associated with qdot.

What do you think?

That sounds great.
I like still having a new entry point AddVelocityLinearConstraintAtNormalizedTime (although maybe AddLinearVelocityConstraintAtNormalizedTime is better?). Like in MathematicalProgram, this will let people declare ... "I think this should be linear" and get a note when it's not. Otherwise sounds good.

Once we have the placeholders workflow, then we don't have to worry about the different entry points assuming the same ordering of the variables, etc.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants