Skip to content

Commit

Permalink
Merge branch 'devel' into fix_unknown_size_vector
Browse files Browse the repository at this point in the history
  • Loading branch information
JafarAbdi authored Jan 3, 2025
2 parents f7ca0fd + 3ce5cf1 commit 9f07e2f
Show file tree
Hide file tree
Showing 8 changed files with 128 additions and 26 deletions.
17 changes: 17 additions & 0 deletions .github/workflows/update-flake-lock.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,17 @@
name: update-flake-lock

on:
workflow_dispatch:
schedule:
- cron: '0 1 5 * *'

jobs:
lockfile:
runs-on: ubuntu-latest
steps:
- name: Checkout repository
uses: actions/checkout@v4
- name: Install Nix
uses: DeterminateSystems/nix-installer-action@main
- name: Update flake.lock
uses: DeterminateSystems/update-flake-lock@main
1 change: 1 addition & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/).

### Fixed
- Fix mjcf Euler angle parsing: use xyz as a default value for eulerseq compiler option ([#2526](https://github.com/stack-of-tasks/pinocchio/pull/2526))
- Add parsing meshes with vertices for MJCF format ([#2537](https://github.com/stack-of-tasks/pinocchio/pull/2537))
- Fix mjcf parsing of keyframe qpos with newlines ([#2535](https://github.com/stack-of-tasks/pinocchio/pull/2535))

## [3.3.1] - 2024-12-13
Expand Down
20 changes: 10 additions & 10 deletions flake.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

8 changes: 2 additions & 6 deletions flake.nix
Original file line number Diff line number Diff line change
Expand Up @@ -20,11 +20,7 @@
devShells.default = pkgs.mkShell { inputsFrom = [ self'.packages.default ]; };
packages = {
default = self'.packages.pinocchio;
pinocchio = pkgs.python3Packages.pinocchio.overrideAttrs (super: {
# avoid SIGTRAP on macos github runners
cmakeFlags = super.cmakeFlags ++ pkgs.lib.optionals pkgs.stdenv.isDarwin [
"-DCMAKE_CTEST_ARGUMENTS=--exclude-regex;pinocchio-example-py-casadi-quadrotor-ocp"
];
pinocchio = pkgs.python3Packages.pinocchio.overrideAttrs {
src = pkgs.lib.fileset.toSource {
root = ./.;
fileset = pkgs.lib.fileset.unions [
Expand All @@ -42,7 +38,7 @@
./utils
];
};
});
};
};
};
};
Expand Down
2 changes: 2 additions & 0 deletions include/pinocchio/parsers/mjcf/mjcf-graph.hpp
Original file line number Diff line number Diff line change
Expand Up @@ -209,6 +209,8 @@ namespace pinocchio
Eigen::Vector3d scale = Eigen::Vector3d::Constant(1);
// Path to the mesh file
std::string filePath;
// Vertices of the mesh
Eigen::MatrixX3d vertices;
};

/// @brief All informations related to a texture are stored here
Expand Down
13 changes: 13 additions & 0 deletions src/parsers/mjcf/mjcf-graph-geom.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -75,6 +75,19 @@ namespace pinocchio
if (geom.geomType == "mesh")
{
MjcfMesh currentMesh = currentGraph.mapOfMeshes.at(geom.meshName);
if (currentMesh.vertices.size() > 0)
{
auto vertices = currentMesh.vertices;
// Scale vertices
for (std::size_t i = 0; i < vertices.rows(); ++i)
vertices.row(i) = vertices.row(i).cwiseProduct(currentMesh.scale.transpose());
auto model = std::make_shared<hpp::fcl::BVHModel<fcl::OBBRSS>>();
model->beginModel();
model->addVertices(vertices);
model->endModel();
model->buildConvexHull(true, "Qt");
return model->convex;
}
meshPath = currentMesh.filePath;
meshScale = currentMesh.scale;
hpp::fcl::BVHModelPtr_t bvh = meshLoader->load(meshPath, meshScale);
Expand Down
52 changes: 42 additions & 10 deletions src/parsers/mjcf/mjcf-graph.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -582,20 +582,52 @@ namespace pinocchio

MjcfMesh mesh;
auto file = el.get_optional<std::string>("<xmlattr>.file");
if (!file)
throw std::invalid_argument("Only meshes with files are supported");

fs::path filePath(*file);
std::string name = getName(el, filePath);

mesh.filePath =
updatePath(compilerInfo.strippath, compilerInfo.meshdir, modelPath, filePath).string();

auto scale = el.get_optional<std::string>("<xmlattr>.scale");
if (scale)
mesh.scale = internal::getVectorFromStream<3>(*scale);
if (file)
{
fs::path filePath(*file);
std::string name = getName(el, filePath);

mesh.filePath =
updatePath(compilerInfo.strippath, compilerInfo.meshdir, modelPath, filePath).string();
mapOfMeshes.insert(std::make_pair(name, mesh));
return;
}

// Handle vertex-based mesh
auto vertex = el.get_optional<std::string>("<xmlattr>.vertex");
if (!vertex)
{
PINOCCHIO_THROW_PRETTY(
std::invalid_argument, "Only meshes with files/vertices are supported.")
}

auto name = el.get_optional<std::string>("<xmlattr>.name");
if (!name)
{
PINOCCHIO_THROW_PRETTY(
std::invalid_argument, "Mesh with vertices without a name is not supported");
}

mapOfMeshes.insert(std::make_pair(name, mesh));
// Parse and validate vertices
Eigen::VectorXd meshVertices = internal::getUnknownSizeVectorFromStream(*vertex);
if (meshVertices.size() % 3 != 0)
{
PINOCCHIO_THROW_PRETTY(
std::invalid_argument, "Number of vertices is not a multiple of 3");
}

// Convert to 3D vertex matrix
const auto numVertices = meshVertices.size() / 3;
Eigen::MatrixX3d vertices(numVertices, 3);
for (auto i = 0; i < numVertices; ++i)
{
vertices.row(i) = meshVertices.segment<3>(3 * i).transpose();
}
mesh.vertices = vertices;
mapOfMeshes.insert(std::make_pair(*name, mesh));
}

void MjcfGraph::parseAsset(const ptree & el)
Expand Down
41 changes: 41 additions & 0 deletions unittest/mjcf.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1387,4 +1387,45 @@ BOOST_AUTO_TEST_CASE(test_get_unknown_size_vector_from_stream)
BOOST_CHECK(v3 == expected3);
}

/// @brief Test parsing a mesh with vertices
/// @param
BOOST_AUTO_TEST_CASE(parse_mesh_with_vertices)
{
std::istringstream xmlDataNoStrip(R"(<mujoco model="parseVertices">
<asset>
<mesh name="chasis" scale=".01 .006 .0015"
vertex=" 9 2 0
-10 10 10
9 -2 0
10 3 -10
10 -3 -10
-8 10 -10
-10 -10 10
-8 -10 -10
-5 0 20"/>
</asset>
</mujoco>)");

auto namefile = createTempFile(xmlDataNoStrip);

typedef ::pinocchio::mjcf::details::MjcfGraph MjcfGraph;
pinocchio::Model model_m;
MjcfGraph::UrdfVisitor visitor(model_m);

MjcfGraph graph(visitor, "/fakeMjcf/fake.xml");
graph.parseGraphFromXML(namefile.name());

// Test Meshes
pinocchio::mjcf::details::MjcfMesh mesh = graph.mapOfMeshes.at("chasis");
BOOST_CHECK_EQUAL(mesh.scale, Eigen::Vector3d(0.01, 0.006, 0.0015));
Eigen::MatrixX3d vertices(9, 3);
vertices << 9, 2, 0, -10, 10, 10, 9, -2, 0, 10, 3, -10, 10, -3, -10, -8, 10, -10, -10, -10, 10,
-8, -10, -10, -5, 0, 20;
BOOST_CHECK_EQUAL(mesh.vertices.rows(), 9);
for (auto i = 0; i < mesh.vertices.rows(); ++i)
{
BOOST_CHECK(mesh.vertices.row(i) == vertices.row(i));
}
}

BOOST_AUTO_TEST_SUITE_END()

0 comments on commit 9f07e2f

Please sign in to comment.