aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFun Maker <funmaker95@gmail.com>2024-04-21 20:48:35 +0200
committerSébastien Crozet <sebastien@crozet.re>2024-05-05 16:34:55 +0200
commit4332818e021644aa716e7ef0cca774cab09f4860 (patch)
treee8ddfb1ea02661a9d70ad8570488f661205aa054
parenta2fdeab7e16282c56b1b9a7b2bcafbb420663c39 (diff)
downloadrapier-4332818e021644aa716e7ef0cca774cab09f4860.tar.gz
rapier-4332818e021644aa716e7ef0cca774cab09f4860.tar.bz2
rapier-4332818e021644aa716e7ef0cca774cab09f4860.zip
Fix joint limits not being flipped in one body constrains. (#549)
-rw-r--r--src/dynamics/joint/generic_joint.rs20
-rw-r--r--src/dynamics/solver/joint_constraint/joint_constraint_builder.rs2
-rw-r--r--src/dynamics/solver/joint_constraint/joint_generic_constraint_builder.rs22
3 files changed, 28 insertions, 16 deletions
diff --git a/src/dynamics/joint/generic_joint.rs b/src/dynamics/joint/generic_joint.rs
index 76a7fe1..98f9311 100644
--- a/src/dynamics/joint/generic_joint.rs
+++ b/src/dynamics/joint/generic_joint.rs
@@ -496,6 +496,26 @@ impl GenericJoint {
self.motors[i].damping = damping;
self
}
+
+ /// Flips the orientation of the joint, including limits and motors.
+ pub fn flip(&mut self) -> &mut Self {
+ std::mem::swap(&mut self.local_frame1, &mut self.local_frame2);
+
+ let coupled_bits = self.coupled_axes.bits();
+
+ for dim in 0..SPATIAL_DIM {
+ if coupled_bits & (1 << dim) == 0 {
+ let limit = self.limits[dim];
+ self.limits[dim].min = -limit.max;
+ self.limits[dim].max = -limit.min;
+ }
+
+ self.motors[dim].target_vel = -self.motors[dim].target_vel;
+ self.motors[dim].target_pos = -self.motors[dim].target_pos;
+ }
+
+ self
+ }
}
macro_rules! joint_conversion_methods(
diff --git a/src/dynamics/solver/joint_constraint/joint_constraint_builder.rs b/src/dynamics/solver/joint_constraint/joint_constraint_builder.rs
index 44716b2..40613c1 100644
--- a/src/dynamics/solver/joint_constraint/joint_constraint_builder.rs
+++ b/src/dynamics/solver/joint_constraint/joint_constraint_builder.rs
@@ -198,7 +198,7 @@ impl JointOneBodyConstraintBuilder {
if flipped {
std::mem::swap(&mut handle1, &mut handle2);
- std::mem::swap(&mut joint_data.local_frame1, &mut joint_data.local_frame2);
+ joint_data.flip();
};
let rb1 = &bodies[handle1];
diff --git a/src/dynamics/solver/joint_constraint/joint_generic_constraint_builder.rs b/src/dynamics/solver/joint_constraint/joint_generic_constraint_builder.rs
index 34256a2..9876195 100644
--- a/src/dynamics/solver/joint_constraint/joint_generic_constraint_builder.rs
+++ b/src/dynamics/solver/joint_constraint/joint_generic_constraint_builder.rs
@@ -319,7 +319,6 @@ pub struct JointGenericVelocityOneBodyExternalConstraintBuilder {
joint_id: JointIndex,
joint: GenericJoint,
j_id: usize,
- flipped: bool,
constraint_id: usize,
// These are solver body for both joints, except that
// the world_com is actually in local-space.
@@ -337,21 +336,20 @@ impl JointGenericVelocityOneBodyExternalConstraintBuilder {
jacobians: &mut DVector<Real>,
out_constraint_id: &mut usize,
) {
+ let mut joint_data = joint.data;
let mut handle1 = joint.body1;
let mut handle2 = joint.body2;
let flipped = !bodies[handle2].is_dynamic();
- let local_frame1 = if flipped {
+ if flipped {
std::mem::swap(&mut handle1, &mut handle2);
- joint.data.local_frame2
- } else {
- joint.data.local_frame1
- };
+ joint_data.flip();
+ }
let rb1 = &bodies[handle1];
let rb2 = &bodies[handle2];
- let frame1 = rb1.pos.position * local_frame1;
+ let frame1 = rb1.pos.position * joint_data.local_frame1;
let starting_j_id = *j_id;
@@ -394,11 +392,10 @@ impl JointGenericVelocityOneBodyExternalConstraintBuilder {
body1,
link2,
joint_id,
- joint: joint.data,
+ joint: joint_data,
j_id: starting_j_id,
frame1,
local_body2,
- flipped,
constraint_id: *out_constraint_id,
});
@@ -417,12 +414,7 @@ impl JointGenericVelocityOneBodyExternalConstraintBuilder {
// constraints. Could we make this more incremental?
let mb2 = &multibodies[self.link2.multibody];
let pos2 = &mb2.link(self.link2.id).unwrap().local_to_world;
- let frame2 = pos2
- * if self.flipped {
- self.joint.local_frame1
- } else {
- self.joint.local_frame2
- };
+ let frame2 = pos2 * self.joint.local_frame2;
let joint_body2 = JointSolverBody {
world_com: pos2 * self.local_body2.world_com, // the world_com was stored in local-space.