Detecting Islands in Tetrahedral Meshes
In the following video a teapot made of 149210 tetrahedra is generated from a regular surface-bounded mesh. Fifty tetrahedra are removed each frame. After each removal the mesh is traversed in search of “islands” - groups of adjacent tetrahedra which share no faces with other groups - which are indicated by color.
I am beginning work on an FEM based model for deforming and fracturing physics objects. Such a model requires that objects to be composed of assemblies of tetrahedrons (theoretically FEM could work on more complex element shapes than the tetrahedron, but I will make no such attempt :P).
Data Structure
In an effort to keep things real-time and prepare for GPU parallelization, I have required that my tetrahedral mesh implementation keep node and tetrahedron memory in mostly contiguous arrays. This will make GPU data binding more feasible later on.
For mesh traversal, some adjacency information is needed between tetrahedrons. I have seen two methods used for this. One is to link nodes to tetrahedra, but this would require either that each node to contain dynamic data (a list of pointers to the elements which contain them), or that the number of tetrahedra per node is limited.
The other method is to track tetrahedron face adjacencies. Since each tetrahedron has at most four face-adjacencies, tetrahedron and node data can still be packed tightly in memory with a predictable stride.
Algorithms
Removal of a tetrahedron is simple, and can be done in constant time. The removal of tetrahedron i
is performed by moving the last tetrahedron (indexed by n-1
where n
is the number of tetrahedra) into slot i
, and correcting it’s neighboring tetrahedra’ adjacencies. This can be done in constant time.
Traversing the structure for the detection and removal of an island is fairly straight forward. I use a stack rather than recursion, and record the indexes of traversed tetraheda and nodes using two large bitmasks.
When a node is copied into a new mesh, it’s index will have changed, and so the indexes stored in each copied tetrahedron which references that node must also change. A simple array of integers, constructed during node-copy, can be used to map the old indexes to the new ones.
Tetrahedron indexes will also change. The same index-mapping method is used to update tetrahedron neighbors in the new mesh.
Each of the traversal/island removal phases has a complexity of O(N)
where N
is the number of nodes. If M
elements are deleted, then in the worst case N*M*4
islands might be generated, so complete island separation has a complexity of O(N*M)
.
Summary
The main purpose of this approach is to keep connected tetrahedral mesh data contiguous in memory.
There are faster ways to split tetrahedral meshes (for example, keeping a node pool instead of allowing a single mesh to “own” its nodes’ memory, and adding an inverted node-tetrahedron relationship).
However, the method I described here will result in a memory layout which is close to ideal for doing per-mesh calculations such as those which are optimal in FEM techniques. Furthermore, mesh breakage will not happen at every frame, but stress calculations will.