Skip to content

Commit

Permalink
Support mesh optimization when using AttachMeshShapeFeature (#2417)
Browse files Browse the repository at this point in the history

Signed-off-by: Ian Chen <[email protected]>
  • Loading branch information
iche033 authored May 30, 2024
1 parent 8ca89f7 commit ae68e81
Show file tree
Hide file tree
Showing 3 changed files with 93 additions and 0 deletions.
7 changes: 7 additions & 0 deletions include/gz/sim/Util.hh
Original file line number Diff line number Diff line change
Expand Up @@ -316,6 +316,13 @@ namespace gz
/// \return The loaded mesh or null if the mesh can not be loaded.
GZ_SIM_VISIBLE const common::Mesh *loadMesh(const sdf::Mesh &_meshSdf);

/// \brief Optimize input mesh.
/// \param[in] _meshSdf Mesh SDF DOM with mesh optimization parameters
/// \param[in] _mesh Input mesh to optimize.
/// \return The optimized mesh or null if the mesh can not be optimized.
GZ_SIM_VISIBLE const common::Mesh *optimizeMesh(const sdf::Mesh &_meshSdf,
const common::Mesh &_mesh);

/// \brief Environment variable holding resource paths.
const std::string kResourcePathEnv{"GZ_SIM_RESOURCE_PATH"};

Expand Down
78 changes: 78 additions & 0 deletions src/Util.cc
Original file line number Diff line number Diff line change
Expand Up @@ -15,12 +15,17 @@
*
*/

#include <cstddef>
#include <string>
#include <vector>

#include <gz/msgs/entity.pb.h>

#include <gz/common/Filesystem.hh>
#include <gz/common/Mesh.hh>
#include <gz/common/MeshManager.hh>
#include <gz/common/StringUtils.hh>
#include <gz/common/SubMesh.hh>
#include <gz/common/URI.hh>
#include <gz/common/Util.hh>
#include <gz/math/Helpers.hh>
Expand Down Expand Up @@ -862,8 +867,81 @@ const common::Mesh *loadMesh(const sdf::Mesh &_meshSdf)
<< "]." << std::endl;
return nullptr;
}

if (mesh && _meshSdf.Optimization() != sdf::MeshOptimization::NONE)
{
const common::Mesh *optimizedMesh = optimizeMesh(_meshSdf, *mesh);
if (optimizedMesh)
return optimizedMesh;
else
gzwarn << "Failed to optimize Mesh " << mesh->Name() << std::endl;
}

return mesh;
}

const common::Mesh *optimizeMesh(const sdf::Mesh &_meshSdf,
const common::Mesh &_mesh)
{
if (_meshSdf.Optimization() !=
sdf::MeshOptimization::CONVEX_DECOMPOSITION &&
_meshSdf.Optimization() !=
sdf::MeshOptimization::CONVEX_HULL)
return nullptr;

auto &meshManager = *common::MeshManager::Instance();
std::size_t maxConvexHulls = 16u;
if (_meshSdf.Optimization() == sdf::MeshOptimization::CONVEX_HULL)
{
/// create 1 convex hull for the whole submesh
maxConvexHulls = 1u;
}
else if (_meshSdf.ConvexDecomposition())
{
// limit max number of convex hulls to generate
maxConvexHulls = _meshSdf.ConvexDecomposition()->MaxConvexHulls();
}
// Check if MeshManager contains the decomposed mesh already. If not
// add it to the MeshManager so we do not need to decompose it again.
const std::string convexMeshName =
_mesh.Name() + "_CONVEX_" + std::to_string(maxConvexHulls);
auto *optimizedMesh = meshManager.MeshByName(convexMeshName);
if (!optimizedMesh)
{
// Merge meshes before convex decomposition
auto mergedMesh = gz::common::MeshManager::MergeSubMeshes(_mesh);
if (mergedMesh && mergedMesh->SubMeshCount() == 1u)
{
// Decompose and add mesh to MeshManager
auto mergedSubmesh = mergedMesh->SubMeshByIndex(0u).lock();
std::vector<common::SubMesh> decomposed =
gz::common::MeshManager::ConvexDecomposition(
*mergedSubmesh.get(), maxConvexHulls);
gzdbg << "Optimizing mesh (" << _meshSdf.OptimizationStr() << "): "
<< _mesh.Name() << std::endl;
// Create decomposed mesh and add it to MeshManager
// Note: MeshManager will call delete on this mesh in its destructor
// \todo(iche033) Consider updating MeshManager to accept
// unique pointers instead
common::Mesh *convexMesh = new common::Mesh;
convexMesh->SetName(convexMeshName);
for (const auto & submesh : decomposed)
convexMesh->AddSubMesh(submesh);
meshManager.AddMesh(convexMesh);
if (decomposed.empty())
{
// Print an error if convex decomposition returned empty submeshes
// but still add it to MeshManager to avoid going through the
// expensive convex decomposition process for the same mesh again
gzerr << "Convex decomposition generated zero meshes: "
<< _mesh.Name() << std::endl;
}
optimizedMesh = meshManager.MeshByName(convexMeshName);
}
}
return optimizedMesh;
}

}
}
}
8 changes: 8 additions & 0 deletions src/Util_TEST.cc
Original file line number Diff line number Diff line change
Expand Up @@ -1023,4 +1023,12 @@ TEST_F(UtilTest, LoadMesh)
"test", "media", "duck.dae");
meshSdf.SetFilePath(filePath);
EXPECT_NE(nullptr, loadMesh(meshSdf));

EXPECT_TRUE(meshSdf.SetOptimization("convex_decomposition"));
sdf::ConvexDecomposition convexDecomp;
convexDecomp.SetMaxConvexHulls(16u);
meshSdf.SetConvexDecomposition(convexDecomp);
auto *optimizedMesh = loadMesh(meshSdf);
EXPECT_NE(nullptr, optimizedMesh);
EXPECT_EQ(16u, optimizedMesh->SubMeshCount());
}

0 comments on commit ae68e81

Please sign in to comment.