17#include <gtest/gtest.h>
26static constexpr double kHomePose[7] = { 0.0, 0.2618, 3.1416, -2.2689, 0.0, 0.9599, 1.5708 };
27static constexpr double kTargetPose[7] = { 0.3, 0.5, 2.9, -2.0, 0.3, 1.2, 1.3 };
28static constexpr double kKv = 2.0;
29static constexpr double kMaxVel = 0.6;
30static constexpr double kTol = 0.02;
31static constexpr double kTimeout = 5.0;
33namespace fs = std::filesystem;
34static fs::path repo_root() {
return fs::path(__FILE__).parent_path().parent_path(); }
48 if (!fs::exists(
root_ /
"third_party/menagerie")) {
49 GTEST_SKIP() <<
"third_party/menagerie/ not found";
53 std::string arm_mjcf = (
root_ /
"third_party/menagerie/kinova_gen3/gen3.xml").
string();
57 r.
path = arm_mjcf.c_str();
65 KDL::JntArray q_home(
n_);
66 for (
unsigned i = 0; i <
n_; ++i) q_home(i) = kHomePose[i];
86 const double dt = model_->opt.timestep;
89 while (data_->time < kTimeout && !arrived) {
93 for (
unsigned i = 0; i < n_; ++i) {
94 double err = kTargetPose[i] - s_.jnt_pos_msr[i];
95 max_err = std::max(max_err, std::abs(err));
96 double vel = std::clamp(kKv * err, -kMaxVel, kMaxVel);
97 s_.jnt_pos_cmd[i] += vel * dt;
100 if (max_err < kTol) {
103 for (
unsigned i = 0; i < n_; ++i) s_.jnt_pos_cmd[i] = s_.jnt_pos_msr[i];
109 double max_err = 0.0;
110 for (
unsigned i = 0; i < n_; ++i)
111 max_err = std::max(max_err, std::abs(kTargetPose[i] - s_.jnt_pos_msr[i]));
114 "MJCF velocity ctrl: " << (arrived ?
"converged" :
"TIMEOUT") <<
" max_err=" << std::fixed
115 << std::setprecision(4) << max_err <<
" rad t=" << data_->time <<
" s"
118 EXPECT_TRUE(arrived) <<
"velocity controller did not converge within " << kTimeout <<
" s";
119 EXPECT_LE(max_err, kTol * 2);
122int main(
int argc,
char *argv[])
124 testing::InitGoogleTest(&argc, argv);
125 return RUN_ALL_TESTS();
bool init_robot_from_mjcf(Robot *r, mjModel *model, mjData *data, const char *base_body, const char *tip_body, const char *prefix="", const char *tool_body=nullptr)
void set_joint_pos(Robot *r, const KDL::JntArray &q, bool call_forward=true)
bool build_scene(mjModel **out_model, mjData **out_data, const SceneSpec *spec)
void destroy_scene(mjModel *model, mjData *data)
std::vector< double > jnt_pos_cmd
std::vector< RobotSpec > robots
int main(int argc, char *argv[])
TEST_F(MjcfVelCtrlTest, Convergence)