This assignment will give you the chance to implement a simple cloth simulation. We will leverage our new found expertise on finite element methods to build a FEM cloth simulation. This simulation will use triangles, rather than tetrahedron as the finite elements and will use a principal stretch-based model for the cloth material. You will also implement your first contact response model, a simple velocity filter that can be bolted onto standard time integration schemes.
On all platforms, we will assume you have installed cmake and a modern c++ compiler on Mac OS X¹, Linux², or Windows³.
We also assume that you have cloned this repository using the --recursive
flag (if not then issue git submodule update --init --recursive
).
Note: We only officially support these assignments on Ubuntu Linux 18.04 (the OS the teaching labs are running) and OSX 10.13 (the OS I use on my personal laptop). While they should work on other operating systems, we make no guarantees.
All grading of assignments is done on Linux 18.04
All assignments will have a similar directory and file layout:
README.md
CMakeLists.txt
main.cpp
assignment_setup.h
include/
function1.h
function2.h
...
src/
function1.cpp
function2.cpp
...
data/
...
...
The README.md
file will describe the background, contents and tasks of the
assignment.
The CMakeLists.txt
file setups up the cmake build routine for this
assignment.
The main.cpp
file will include the headers in the include/
directory and
link to the functions compiled in the src/
directory. This file contains the
main
function that is executed when the program is run from the command line.
The include/
directory contains one file for each function that you will
implement as part of the assignment.
The src/
directory contains empty implementations of the functions
specified in the include/
directory. This is where you will implement the
parts of the assignment.
The data/
directory contains sample input data for your program. Keep in
mind you should create your own test data to verify your program as you write
it. It is not necessarily sufficient that your program only works on the given
sample data.
This and all following assignments will follow a typical cmake/make build routine. Starting in this directory, issue:
mkdir build
cd build
cmake ..
If you are using Mac or Linux, then issue:
make
Compiling the code in the above manner will yield working, but very slow executables. To run the code at full speed, you should compile it in release mode. Starting in the build directory, do the following:
cmake .. -DCMAKE_BUILD_TYPE=Release
Followed by:
make
Your code should now run significantly (sometimes as much as ten times) faster.
If you are using Windows, then running cmake ..
should have created a Visual Studio solution file
called a4-cloth-simulation.sln
that you can open and build from there. Building the project will generate an .exe file.
Why don't you try this right now?
Once built, you can execute the assignment from inside the build/
using
./a4-cloth-simulation
While running, you can activate or de-activate the collision sphere by pressing c
.
In this assignment we will move from the simulation of volumetric objects to the simulation of thin sheets or thin shells. For thin objects, rather than discretize the volume with tetrahedra (as was done in assignment 2 and assignment 3) we discretize the medial surface of the cloth. This surface takes the form of a two-dimensional (2d) manifold embedded in (3d) space. While many of the concepts you have already learned will carry over to this assignment, the major confounding factor will be evaluating the material model for the cloth on this manifold. To make things more interesting, we will see our first material model expressed in terms of the "principal stretches" of the deformation.
In order to allow for more interesting interactions with the cloth, we will also implement collision detection and resolution with an analytical sphere. We will implement a simple collision resolution scheme, via velocity filter. These algorithms try to prevent collisions by "filtering" a previously computed velocity to remove any components that might make collisions worse. While running the assignment code, you can activate or de-activate the collision sphere by pressing c
.
Somewhat sadly, it is difficult to find a comprehensive resource for modern cloth simulation. However, there are a collection of seminal papers that are helpful to peruse. The first, Large-Steps in Cloth Simulation is widely recognized for introducing the linearly-implicit time integrator to graphics. While we won't deal with bending stiffness (think the difference between a Kleenex and a steel sheet), Discrete quadratic curvature energies are still the state-of-the-art approach for triangle mesh cloth. Finally, modern cloth simulators rely on aggressive remeshing schemes to capture detail while retaining performance.
Our major challenge in this assignment arises from the difference in the dimensionality of the cloth material and the world (deformed) space (NOTE: I use the terms world and deformed space interchangeably). Cloth is locally two-dimensional (2d) while the world is 3d. What will be comforting is that, a relatively straight-forward application of the finite-element-method (FEM) will allow us to build a passable dynamic cloth simulator.
The previous assignment applied FEM to volumetric simulation -- the simulation of objects with geometry of dimension equal to that of the world space (i.e our bunny and armadillo were 3d as was the world). In this case our finite elements were also volumetric ... they were tetrahedra in the undeformed space of the simulated object.
In the case of cloth, the underformed geometry is of different dimension (2d) than the world space (3d). Because our finite elements divide up the undeformed space, they also need to be 2d. As such we will use triangles, not tetrahedra as our elements.
Just as in the previous assignment, we need to choose basis, or shape functions with which to approximate functions on our, now triangular, mesh. A triangle as 3 nodes our approximations become
where are the barycentric coordinates for a 2D triangle and is the 2d coordinate in the undeformed space.
However, cloth is really a thin volumetric construct, of which our triangle only represents a small part. We might need information lying slightly off the triangle surface. To account for this we will need to modify our FEM model a bit. First, let's assume our triangle is actually embedded in a 3D undeformed space . Let's try and and build and appropriate mapping from this space to the world space.
Given any point in the undeformed space, we can compute the barycentric coordinates of the nearest point on our triangle by solving
,where is a matrix of edge vectors. We use the constriaint to reconstruct . This equation finds the barycentric coordinates of the nearest point on the triangle to in a least squares fashion. The error in this least squares solve will be orthogonal to the column space of , our triangle. For any point we can work out its offset from the triangle by computing , where is the first vertex of our triangle and is the undeformed surface normal of the triangle. Because our triangle has a constant normal, we don't need to worry about where we compute it, which makes this all very convenient.
Let's assume that our point maintains a constant offset from the triangle when deformed. This implies we can reconstruct the world space position by offsetting our point the same distance along the world space normal . This gives us the following mapping from reference space to world space
.Now we can choose the generalized coordinates () to be the stacked vector of vertex positions, which defines the generalized velocities as the stacked 9d vector of per-vertex velocities.
There are lots of ways to handle build a cloth deformation gradient in literature. In this assignment we will be able to avoid these more complicated solution due to our particular choice of undeformed to world space mapping which allows us to directly compute a deformation gradient as
Armed with the generalized velocities, the formula for the per-triangle kinetic energy is eerily similar to that of assignment 3. It's an integral of the local kinetic energy over the entire triangle, multiplied by the thickness of the cloth, . For this assignment you are free to assume the thickness of the cloth is .
and can be compute analytically using a symbolic math package. The per-element mass matrices for every cloth triangle can then be assembled into the mass matrix for the entire mesh.
For this assignment we will use a different type of material model to describe the elastic behaviour of the cloth. This is motivated by the fact that cloth is typically very resistant to stretching. For these materials, a linear stress-strain relationship is often desirable. Unfortunately, cloth triangles also rotate a lot (every time they fold-over for instance). Rotations are NOT linear and so a purely linear relationship will suffer from severe artifacts. To avoid this we will build a material model that only measures the in plane deformation of the cloth via its principal stretches.
Recall that in the previous assignment we used the right Cauchy strain tensor () to measure deformation and the rationale for using this was that it measures the squared deformed length of an arbitrary, infinitesimal line of material, . In other words, . Because is symmetric and positive definite, we can perform an eigendecomposition such that where is the orthogonal matrix of eigenvectors and is the diagonal matrix of eigenvalues. This means we can think of this squared length as where . In other words, if we transform just right, its deformation is completely characterized by .
are the eigenvalues of and also the squared singular values of . We call these singular values of the principal stretches. They measure deformation independently of the orientation (or rotation/reflection) of the finite element.
Now we can formulate a linear elastic model using the principal stretches which "filters out" any rotational components. Much like the Neohookean material model, this model will have one energy term which measures deformation and one energy term that tries to preserve volume (well area in the case of cloth). We already know we can measure deformation using the principal stretches. We also know that the determinant of measures the change in volume of a 3D object. In the volumetric case this determinant is just the product of the principal stretches.
where and are the material properties for the cloth. The first term in this model attempts to keep and close to one (limiting deformation) while the second term is attempting to preserve the volume of the deformed triangle (it's a linearization of the determinant). This model is called co-rotational linear elasticity because it is linear in the principal stretches but rotates with each finite element. When we use energy models to measure the in-plane stretching of the cloth (or membrane), we often refer to them as membrane energies.
The strain energy density for principal stretch models, like the one above, are relatively easy to implement and understand. This is a big reason we like them in graphics. We'll also see that the gradient of this model (needed for force computation) is also pretty easy to compute.
Really, the derivative we need to understand how to compute is . Once we have this we can use to compute the gradient wrt to the generalized coordinates. Conveniently, we have the following for principal stretch models.
where is the singular value decomposition.
Unfortunately, the gradient of the principal stretch energy is not enough. That's because our favourite implicit integrators require second order information to provide the stability and performance we crave in computer graphics. This is where things get messy. The good news is that, if we can just compute then we can use to compute our Hessian wrt to the generalized coordinates (this follows from the linearity of the FEM formulation wrt to the generalized coordinates). This formula is going to get ugly so, in an attempt to make it somewhat clear, we are going to consider derivatives wrt to single entries of , denoted . In this context we are trying to compute
Here takes a vector as input and converts it into a diagonal matrix, with the entries of the matrix on the diagonal. In our case, we define as
When looking at this formula DON'T PANIC. It's a straight forward application of the chain rule, just a little nastier than usual. Also remember that I am giving you the code to compute SVD derivatives in dsvd.h/dsvd.cpp.
If we define the svd of a matrix as , this code returns , and . Yes this code returns 3 and four dimensional tensors storing this quantities, yes I said never to do this in class, consider this the exception that makes the rule. The latter two indices on each tensor are the and indices used in the formula above.
The hardest part of implementing this gradient correctly is handling the SVD terms. These gradients have a different form based on whether your matrix is square or rectangular. This is one big reason that the deformation gradient we use in this assignment is desirable. It allows one to use the same singular value decomposition code for volumetric and cloth models.
To make this assignment a little more visually interesting, you will implement simple collision detection and resolution with an analytical sphere. Collision Detection is the process of detecting contact between two or more objects in the scene and Collision Resolution is the process of modifying the motion of the object in response to these detected collisions.
For this assignment we will implement per-vertex collision detection with the sphere. This is as simple as detecting if the distance from the center of the sphere to any vertex in your mesh is less than the radius of the sphere. If you detect such a collision, you need to store an index to the colliding vertex, along with the outward facing contact normal(). In this case, the outward facing contact normal is the sphere normal at the point of contact.
The minimal goal of any collision resolution algorithm is to prevent collisions from getting worse locally. To do this we will implement a simple velocity filter approach. Velocity filters are so named because the "filter out" components of an objects velocity that will increase the severity of a collision. Given a vertex that is colliding with our sphere, the only way that the collision can get worse locally is if that vertex moves into the sphere. One way we can check if this is happening is to compute the projection of the vertex velocity onto the outward facing contact normal (, selects the contacting vertex). If this number is we are OK, the vertex is moving away from the sphere. If this number is we better do something.
The thing we will do is project out, or filter out, the component of the velocity moving in the negative, normal direction like so:
This "fixes" the collision. This approach to collision resolution is fast but for more complicated scenes it is fraught with peril. For instance it doesn't take into account how it is deforming the simulated object which can cause big headaches when objects are stiff or rigid. We'll see a cleaner mathematical approach to content in the final assignment of the course.
In this assignment you will reuse your linearly implicit integrator to time step the dynamic system. Also, we will eschew implementing the strain energy density function, quadrature rule and potential energies separately. Instead functions for potential energy and its derivatives should directly compute the integrated values for a triangular element.
Piecewise constant gradient matrix for linear shape functions. Row of the returned matrix contains the gradient of the shape function.
The kinetic energy of the whole cost mesh.
Gradient of potential energy due to gravity
Potential energy for the cloth stretching force
Gradient of the cloth stretching energy.
Hessian matrix of the cloth stretching energy
Use your code from the last assignment
Use your code from the last assignment
Assemble the full mass matrix for the entire tetrahedral mesh.
Assemble the global force vector for the finite element mesh.
Assemble the global stiffness matrix for the finite element mesh.
Use your code from the last assignment
Use your code from the last assignment
Detect if any mesh vertex falls inside a sphere centered at (0,0,0.4) with radius 0.22
Project out components of the per-vertex velocities which are in the positive direction of the contact normal
Use your code from the last assignment