Skip to content

Latest commit

 

History

History
109 lines (58 loc) · 11.7 KB

README.md

File metadata and controls

109 lines (58 loc) · 11.7 KB

2d continuum physics for the GPU with Taichi

Peek 2022-02-25 15-09

Above: a Neo-Hookean elastic bar with one end fixed


This repo presents a didactic implementation of the Total Langranian Material Point Method (TLMPM) as described in the papers:

  • "A Total-Lagrangian Material Point Method for solid mechanics problems involving large deformation", (DOI: 10.1016/j.cma.2019.112783)
  • "Modelling contacts with a total Lagrangian material point method" (DOI: 10.13140/RG.2.2.10187.62245). Note: in this repo, we'll use the shorthand name "TLMPM Contacts" to refer to this paper

(These papers can be found in the papers folder of this repo.)

The objective of the contens is to present a simple TLMPM implementation beginning from a very literal translation of Algorithm 2 from "TLMPM Contacts" based on the famous mpm128.py implementation of MLS-MPM provided by Yuanming Hu and the Taichi team, and then to make a series of small attempts to improve the implementation, ultimately arriving at a much more performant version. This reflects my process while attempting to learn both the TLMPM algortithm itself and some of the optimizations one might try when attempting to write performant code for the GPU in general and Taichi in specific. I am by no means an expert in any of these areas, but this code is offered here nonetheless in the hopes that it might be helpful to someone else attempting to learn about these topics in the future.

Ideally, in the first step you should be able to see how the algorithm from "TLMPM Contacts" maps to the code, and since we've gone step-by-step making small changes along the way, you'll hopefully be able to understand and reconstruct why the final code looks the way it does even though it looks quite different from the initial code.

To run the demos

You'll need to pip install taichi, then simply run any of the scripts, e.g. python TLMPM_beam_fixed_end/TLMPM_beam_fixed_end.py.

This code has been tested on Taichi 0.8.11 with CUDA. You will very likely need install CUDA if you want to use GPU, I have not tried the non-CUDA backends.

How to read this repo

To follow along with the implementation improvements, it's recommended that you clone/download this repo to your local machine and then use you favorite text editor to compare the different TLMPM implementations in the folder TLMPM_perf_optimization. The files are all named starting with a version number, e.g. v{n}_*.py; in each case, the changes between v{n}_*.py and v{n+1}_*.py is meant to be one discrete set of improvements that should be easy to follow and understand.

(Note: to compare versions in VSCode, click on v{n}_*.py and then control+click v{n+1}_*.py, then right click and select "Compare selected".)

These files follow the strategy I took when implementing TLMPM: start with the working code in mpm128.py, and adapt it to Algorithm 2 in "TLMPM Contacts", then make incremental improvements from there. This is a little awkward at first, because these algorithms are different enough that shoehorning TLMPM into the mpm128.py framework resulted in some code that is not what you would want if you we're approaching TLMPM from scratch, but I as a relatively new user of Taichi, it was helpful to have this scaffolding.

Below we describe the changes made between each version, starting with the file v0_mpm128_jelly_spinning_beam.py. The FPS numbers given are from running the simulation with 294912 particles, allowing 5 seconds of warm up and then taking 20 seconds of FPS measurements. (Note that these FPS numbers should not be taken to suggest anything about the relative merits of these MLS-MPM vs TLMPM, as the MLS-MPM implementation is not necessarily fully optimized, and in any case the comparison is apples-to-oranges in quite a number of ways.)

v0_mpm128_jelly_spinning_beam.py

Avg FPS: 15.59

In This first file, we make a number of changes to mpm128.py to give create a baseline MLS-MPM scenario that will allow us to track the progress of our TLMPM implementation:

  • because TLMPM is best suited for solids that don't undergo plastic deformation, the water and snow materials are not really appropriate so we use only the "jelly" material;
  • because the handling of contact is not automatic in TLMPM, we want a scenario in which the object simulated won't have to bounce off of the bounds of the simulation area, so we initialize the particles as a spinning beam;
  • because we want to be able to be able to use exactly the same number of particles in our MLS-MPM and TLMPM simulations, we initialize the particles on a Cartesian lattice rather than randomly.

To satisfy these requirements, we'll simulate a spinning beam. This is a very uninteresting scenario for a continuum simulation, but the main point here is to focus on the changes to the implementation as we refine it, not the scenario itself.

v1_from_paper.py

Avg FPS: 8.29

This initial implementation of TLMPM is based on the particle-oriented MLS-MPM implementation from v0_mpm128_jelly_spinning_beam.py, in which the p2g and g2p phases "iterates" over particles; in p2g particles scatter information to the grid (using atomic adds), and in g2p they gather from the grid. This is not optimal for TLMPM (as we will see below), but it follows from the template

However, there are a lot of code changes between v0_mpm128_jelly_spinning_beam.py and v1_from_paper.py (so a direct diff between them may not be super illuminating at this point).

For this file, I'd recommend reviewing Algorithm 2 in "TLMPM Contacts"; the file is heavily annotated with comments indicating exactly which line of code maps to which line in the algorithm from the paper. This is likely to be the most helpful way of understanding how the TLMPM algorithm works, but a few of the main conceptual changes from v0_mpm128_jelly_spinning_beam.py reflected in this code are:

  • TLMPM calculates it's deformation particle/grid transfer weights using the positions of particles and grid cells in configuration space, therefore fields must be added to capture this data alongside the world positions of particles

  • Since weights and weight gradients are calculated WRT unchanging configuration space positions, they are calculated and stored once before the simulation begins.

    • In this code, all of the relevant initialization kernels and functions are coded with clarity rather than performance in mind since they will be run only once. Furthermore, they are not optimized in any following variants.
    • These weights and gradients are just stored in a Taichi field, one set of weights and gradients for each particle. These are then retrieved via index lookup in e.g. the p2g and g2p loops, rather than being calculated on the fly for each particle-grid node pair, as must be done in MLS-MPM.
  • Unlike MLS-MPM, in which the particle/grid weight function has a support radius large enough to cover a 3x3 grid node neighborhood for each particle, the linear hat function kernel recommended for TLMPM will cover at most a 2x2 neighborhood for each particle.

  • TLMPM uses a PIC+FLIP blend to improve conservation of angular momentum. This necessitates the storage of temporary velocities during each timestep as well as a bit of extra computation to handle the PIC+FLIP update.

v2_grid_of_particles_in_config_space.py

Avg FPS: 19.21

Comparing this version to the previous one, it will initially look like a lot of changes have been made. However, in actuality there is really only one change: since the positions of particles are fixed in configuration space in TLMPM, and moreover since they are fixed on a cartesian lattice with a fixed relationship to the MPM background grid (4 particles per grid cell), it makes sense to store the particles in a 2d field and access them by their index in the lattice. The actual position in configuration space can be easily calculated with the index in the particle field and the spacing of the background grid.

This also substantially simplifies some of the details regarding which grid nodes each particle maps to, allowing us to easily use a background grid that is sized specifically for the initial particle distribution in configuration space. This dramatically reduces the amount of memory needed for the grid and the amount of computation required for kernels that operate on the entire grid.

Note: starting with this file we always use the notation (i,j) to refer to indices of the MPM background grid, and (f,g) to refer to indices of particles in configuration space.

v3_particle_grid_to_cell_grid_accessors.py

Avg FPS: 20.84

This version makes one very small change. In mpm128.py, there is the concept of the grid base that corresponds to each particle; this is the bottom-left-most grid node of the 3x3 range of nodes that need to be iterated over for each particle during p2g and g2p. For TLMPM, this is not really needed because, since particles have a fixed position in configuration space and can be store on a fixed grid, the base need not be calculated from the configuration space position, it can calculated using the indices od the particle in the field of particles. It's interesting to see that even a seemingly tiny change like this (replacing a couple casts and floating point multiplications with an integer division) can result in several percentage points of performance improvement.

v4_gather.py

Avg FPS: 32.31

This file has the most important changes of any so far, and massively improves the performance of the implementation. Since TLMPM operates in configuration space and each particle is permanently assigned to a background grid cell, rather than doing scattered atomic add operation from particles to grid cells, each grid cell can gather the information it needs from it's neighboring particles. In this file, we make that change in p2g and g2p.

Additionally, since doing p2g as a gather requires gathering from a 4x4 neighborhood of surrounding particles, we take this opportunity to notice that for each grid node, the surrounding particles will always have the same offset positions. Therefore we only need to calculate a single 4x4 stencil for weights and weight gradients, and that this stencil can be applied at every grid node. This dramatically reduces memory use, and should also perhaps improve memory locality.

v4.1_more_gather_DEAD_END.py

Avg FPS: 5.63

This variant attempted to complete the process of replacing scattered atomic adds with gathers by updating the calculation of grid_mv to use a gather. This unexpectedly destroys the performance of the implentation. I don't understand why this should be the case at this; it remains a question to be investigated.

v5_update_momentum_in_p2g.py

Avg FPS: 33.63

Note: this file should be compared with v4, not v4.1

We make only one small change in this file, rolling the update_momenta() kernel into p2g, which removes an addtional kernel dispatch and prevents two additional passes over the background grid.

v7_memory_placement.py

Avg FPS: 33.36

Here we attempt to use some of Taichi's more advanced memory layout features to improve performance by arranging the particles field into blocks. This should increase the number of particles that are close to each other spatially (and therefore frequently accessed and processed together) are that are also close to each other in the machine's linear memory. This attempt was made based on some resources I'd read which suggested that such optimization might theoretically improve performance by improving cache coherence/locality or something along those lines, but in practice the performance is very slightly reduced with the best block size and moderately reduced in the worst case.