Skip to content

Commit

Permalink
[mpm] Add ParticleSorter
Browse files Browse the repository at this point in the history
In addition, SpGrid reports the flags it uses, and add the missing
set_zero() method for GridData.
  • Loading branch information
xuchenhan-tri committed Jan 14, 2025
1 parent 70016f2 commit d06fcbf
Show file tree
Hide file tree
Showing 8 changed files with 496 additions and 0 deletions.
26 changes: 26 additions & 0 deletions multibody/mpm/BUILD.bazel
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ drake_cc_package_library(
":bspline_weights",
":grid_data",
":particle_data",
":particle_sorter",
],
)

Expand Down Expand Up @@ -60,6 +61,21 @@ drake_cc_library(
],
)

drake_cc_library(
name = "particle_sorter",
srcs = [
"particle_sorter.cc",
],
hdrs = [
"particle_sorter.h",
],
deps = [
":bspline_weights",
"//common:essential",
"//math:gradient",
],
)

# TODO(xuchenhan-tri): when we enable SPGrid in our releases, we also need to
# install its license file in drake/tools/workspace/BUILD.bazel.

Expand Down Expand Up @@ -99,6 +115,16 @@ drake_cc_googletest(
],
)

drake_cc_googletest_linux_only(
name = "particle_sorter_test",
deps = [
":bspline_weights",
":grid_data",
":particle_sorter",
":spgrid",
],
)

drake_cc_googletest_linux_only(
name = "spgrid_test",
deps = [
Expand Down
8 changes: 8 additions & 0 deletions multibody/mpm/grid_data.h
Original file line number Diff line number Diff line change
Expand Up @@ -112,6 +112,14 @@ struct GridData {
are set to NAN and the index is inactive. */
void reset() { *this = {}; }

/* Sets all floating point data to zero, and set the index to be inactive.*/
void set_zero() {
v.setZero();
m = 0;
scratch.setZero();
index_or_flag.set_inactive();
}

/* Returns true iff `this` GridData is bit-wise equal to `other`. */
bool operator==(const GridData<T>& other) const {
return std::memcmp(this, &other, sizeof(GridData<T>)) == 0;
Expand Down
27 changes: 27 additions & 0 deletions multibody/mpm/particle_sorter.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
#include "drake/multibody/mpm/particle_sorter.h"

namespace drake {
namespace multibody {
namespace mpm {
namespace internal {

void ConvertToRanges(const std::vector<int>& data, Ranges* ranges) {
ranges->clear();
ranges->reserve(data.size() - 1);
for (int i = 1; i < ssize(data); ++i) {
ranges->push_back({data[i - 1], data[i]});
}
}

std::vector<uint64_t> ParticleSorter::GetActiveBlockOffsets() const {
std::vector<uint64_t> offsets(ssize(sentinel_particles_) - 1);
for (int i = 0; i < ssize(sentinel_particles_) - 1; ++i) {
offsets[i] = base_node_offsets_[sentinel_particles_[i]];
}
return offsets;
}

} // namespace internal
} // namespace mpm
} // namespace multibody
} // namespace drake
231 changes: 231 additions & 0 deletions multibody/mpm/particle_sorter.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,231 @@
#pragma once

#include <algorithm>
#include <array>
#include <bitset>
#include <vector>

#include "drake/common/drake_assert.h"
#include "drake/common/eigen_types.h"
#include "drake/common/ssize.h"
#include "drake/math/autodiff_gradient.h"
#include "drake/multibody/mpm/bspline_weights.h"

namespace drake {
namespace multibody {
namespace mpm {
namespace internal {

/* A range of integers [start, end). */
struct Range {
int start;
int end;
};

using Ranges = std::vector<Range>;

/* Given a sorted vector of integers key points, converts it to a vector of
consecutive ranges. For example, given the input {1, 3, 6, 9}, the output
should be {{1, 3}, {3, 6}, {6, 9}}.
@pre data is sorted in ascending order and has at least two elements.
@pre ranges != nullptr */
void ConvertToRanges(const std::vector<int>& data, Ranges* ranges);

/* ParticleSorter sorts MPM particle data based on their positions within the
grid.
Given a set of MPM particles and the background grid, recall that we define the
base node of a particle to be the grid node that is closest to the particle
(see multibody::mpm::internal::ComputeBaseNode).
ParticleSorter provides a `Sort()` function that sorts the particles first
based on their base node offsets (recall that the offset of a node is the 1D
index of the node; see multibody::mpm::internal::SpGrid for the full
definition) in the grid; if they are the same, it then sorts the particles by
their original indices. This sorting is useful for efficient data access in the
MPM transfer kernels.
After `Sort()` is called, the ParticleSorter provides information about the
particles and its background grid based on the result of the sort. */
class ParticleSorter {
public:
DRAKE_DEFAULT_COPY_AND_MOVE_AND_ASSIGN(ParticleSorter);
ParticleSorter() = default;

// TODO(xuchenhan-tri): The implementation of this function can be moved to
// the .cc file when SpGrid is no longer an open template.
template <typename T, typename SpGrid>
void Sort(const SpGrid& spgrid, double dx,
const std::vector<Vector3<T>>& particle_positions) {
const auto& flags = spgrid.flags();
const int log2_max_grid_size = flags.log2_max_grid_size;
const int data_bits = flags.data_bits;
const int log2_page = flags.log2_page;

const int num_particles = particle_positions.size();
data_indices_.resize(num_particles);
base_node_offsets_.resize(num_particles);
particle_sorters_.resize(num_particles);

/* We sort particles first based on their base node offsets in the spgrid;
if they are the same, then we sort the particles by their original indices.
To do that efficiently , we notice that the base node offset in the spgrid
of the particle (a 64 bit unsigned integer) looks like
page bits | block bits | data bits
with all the data bits being equal to zero.
In addition, a few bits (starting from the left) in the page bits are zero
because we cap the total number of allowed grid nodes. There are at most
2^(3*log2_max_grid_size) number of grid nodes and that takes up
3*log2_max_grid_size bits. The page bits and block bits have 64 - data bits
in total, so the left most 64 - data bits - 3 * log2_max_grid_size bits are
zero. So we left shift the base node offset by that amount and now we get
the lowest 64 - 3 * log2_max_grid_size bits (which we name `index_bits`) to
be zero. We use these bits to store the original index of the particles.
With a typical log2_max_grid_size == 10, we have 44 bits to work with, more
than enough to store the particle indices. We then sort the resulting 64
bit unsigned integers which is enough to achieve the sorting objective. */
const int index_bits = 64 - 3 * log2_max_grid_size;
// TODO(xuchenhan-tri): This should be enforced at registration time.
/* Make sure that we have enough bits to store the index of the particles.
*/
DRAKE_DEMAND(uint64_t(1) << index_bits >
static_cast<uint64_t>(num_particles));
const int zero_page_bits = 64 - data_bits - 3 * log2_max_grid_size;

for (int p = 0; p < num_particles; ++p) {
/* We allow T == AutoDiffXd only for testing purpose. The positions in
those tests are guaranteed to have zero gradients. */
const auto& particle_x = [&]() -> const Vector3<double>& {
if constexpr (std::is_same_v<T, double>) {
return particle_positions[p];
} else if constexpr (std::is_same_v<T, float>) {
return particle_positions[p].template cast<double>();
} else {
return math::DiscardZeroGradient(particle_positions[p]);
}
}();
const Vector3<int> base_node = ComputeBaseNode<double>(particle_x / dx);
base_node_offsets_[p] =
spgrid.CoordinateToOffset(base_node[0], base_node[1], base_node[2]);
data_indices_[p] = p;
/* Confirm the data bits of the base node offset are all zero. */
DRAKE_ASSERT((base_node_offsets_[p] & ((uint64_t(1) << data_bits) - 1)) ==
0);
/* Confirm the left most bits in the page bits are unused. */
DRAKE_ASSERT((base_node_offsets_[p] &
~((uint64_t(1) << (64 - zero_page_bits)) - 1)) == 0);
particle_sorters_[p] =
(base_node_offsets_[p] << zero_page_bits) + data_indices_[p];
}

// TODO(xuchenhan-tri): use a more efficient sorting algorithm. This can be
// a bottleneck for large number of particles.
std::sort(particle_sorters_.begin(), particle_sorters_.end());

/* Peel off the data indices and the base node offsets from
particle_sorters_. Meanwhile, reorder the data indices and the base node
offsets based on the sorting results. */
const uint64_t index_mask = ((uint64_t(1) << index_bits) - 1);
for (int p = 0; p < ssize(particle_sorters_); ++p) {
data_indices_[p] = particle_sorters_[p] & index_mask;
base_node_offsets_[p] = (particle_sorters_[p] >> index_bits) << data_bits;
}

/* Record the sentinel particles and the coloring of the blocks. */
sentinel_particles_.clear();
for (int b = 0; b < 8; ++b) {
colored_ranges_[b].clear();
}
uint64_t previous_page{};
int range_index = 0;
for (int p = 0; p < num_particles; ++p) {
/* The bits in the offset is ordered as follows:
page bits | block bits | data bits
block bits and data bits add up to log2_page bits.
We right shift to get the page bits. */
const uint64_t page = base_node_offsets_[p] >> log2_page;
if (p == 0 || previous_page != page) {
previous_page = page;
sentinel_particles_.push_back(p);
const int color = spgrid.get_color(page);
ranges_indices_[color].push_back(range_index++);
}
}
sentinel_particles_.push_back(num_particles);

ConvertToRanges(sentinel_particles_, &ranges_);
for (int c = 0; c < 8; ++c) {
colored_ranges_[c].clear();
for (int r : ranges_indices_[c]) {
colored_ranges_[c].push_back(ranges_[r]);
}
}
}

/* Returns a vector of representative grid node offsets for all active SpGrid
blocks.
A grid block is considered "active" if it contains at least one particle. The
returned vector has exactly one offset per active block, allowing the caller
to identify or access the blocks that contain particles. */
std::vector<uint64_t> GetActiveBlockOffsets() const;

/* The order in which the particle data is preferrablly accessed.
typical usage:
const std::vector<int>& data_indices = particle_sorter.data_indices();
const int num_particles = data_indices.ssize();
// Loop over some particle data `foo`.
for (int p = 0; p < num_particles; ++p) {
const auto& particle_data = foo_data[data_indices[p]];
// do something with particle_data.
}
*/
const std::vector<int>& data_indices() const { return data_indices_; }

/* Returns the offsets of the base nodes for the particles in sorted order. */
const std::vector<uint64_t>& base_node_offsets() const {
return base_node_offsets_;
}

/* Returns an array of eight Ranges objects, each representing a distinct
color partition of the particle data. After sorting, the disjoint subsets in
each partition can be processed safely in parallel using MPM transfer
kernels, minimizing write hazards between concurrent threads.
Specifically, for each i from 0 to 7, colored_ranges()[i] is a collection of
non-overlapping sorted particle data indices ranges (Cᵢ). Using these ranges,
you can partition the particle data into segments that can be handled by
different threads without risking simultaneous writes to the same memory
locations. */
const std::array<Ranges, 8>& colored_ranges() const {
return colored_ranges_;
}

private:
/* All but last entry store (sorted) indices of particles marking the boundary
of a new block. The last entry stores the number of particles. */
std::vector<int> sentinel_particles_;
std::vector<int> data_indices_;
std::vector<uint64_t> base_node_offsets_;
/* The data being sorted -- it encodes the base node offset and the original
index of the particle. */
std::vector<uint64_t> particle_sorters_;
std::array<Ranges, 8> colored_ranges_;
/* The ranges of the particle data indices, each range corresponds to
particles inside a single SpGrid page. */
Ranges ranges_;
/* Helper data structure for building `colored_ranges_`. */
std::array<std::vector<int>, 8> ranges_indices_;
};

} // namespace internal
} // namespace mpm
} // namespace multibody
} // namespace drake
34 changes: 34 additions & 0 deletions multibody/mpm/spgrid.h
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,16 @@ namespace internal {
template <typename T>
using Pad = std::array<std::array<std::array<T, 3>, 3>, 3>;

/* A subset of SPGrid flags and configs we care about. */
struct SpGridFlags {
int log2_page{12}; // 4KB page size.
int log2_max_grid_size{10}; // Largest grid size is 1024 x 1024 x 1024.
int data_bits{6}; // Number of bits to represent GridData.
int num_nodes_in_block_x{4}; // Number of nodes in a block in x direction.
int num_nodes_in_block_y{4}; // Number of nodes in a block in y direction.
int num_nodes_in_block_z{4}; // Number of nodes in a block in z direction.
};

/* SpGrid is a wrapper class around the SPGrid library designed for managing
sparse grid data in 3D space.
Expand Down Expand Up @@ -191,6 +201,9 @@ class SpGrid {
const uint64_t world_space_offset = Mask::Linear_Offset(x, y, z);
return Mask::Packed_Add(world_space_offset, origin_offset_);
}
Offset CoordinateToOffset(const Vector3<int>& coord) const {
return CoordinateToOffset(coord[0], coord[1], coord[2]);
}

/* Returns the 3D grid coordinates in world space given the offset (1D
index) of a grid node.
Expand Down Expand Up @@ -316,6 +329,27 @@ class SpGrid {
}
}

/* Returns the flags associated with `this` SpGrid. */
SpGridFlags flags() const {
return SpGridFlags{.log2_page = kLog2Page,
.log2_max_grid_size = log2_max_grid_size,
.data_bits = kDataBits,
.num_nodes_in_block_x = kNumNodesInBlockX,
.num_nodes_in_block_y = kNumNodesInBlockY,
.num_nodes_in_block_z = kNumNodesInBlockZ};
}

/* Returns the color of the block given the page bits of the node offset.
Blocks with different colors are guaranteed to be non-adjacent. */
static int get_color(uint64_t page) {
/* According to [Setaluri et al. 2014], the blocks are arranged in a 3D
space using Z-order curve. Blocks with 8 blocks apart are guaranteed to be
non-adjacent. */
constexpr int kNumColors = 8;
int color = (page & (kNumColors - 1));
return color;
}

private:
/* Returns reference to the data at the given grid offset.
@pre Given `offset` is valid. */
Expand Down
15 changes: 15 additions & 0 deletions multibody/mpm/test/grid_data_test.cc
Original file line number Diff line number Diff line change
Expand Up @@ -97,6 +97,21 @@ TYPED_TEST(GridDataTest, Reset) {
EXPECT_TRUE(std::isnan(data.m));
}

TYPED_TEST(GridDataTest, SetZero) {
using T = TypeParam;
GridData<T> data;
data.index_or_flag.set_index(123);
data.scratch = Vector3<T>::Ones();
data.v = Vector3<T>::Ones();
data.m = 1;

data.set_zero();
EXPECT_TRUE(data.index_or_flag.is_inactive());
EXPECT_EQ(data.scratch, Vector3<T>::Zero());
EXPECT_EQ(data.v, Vector3<T>::Zero());
EXPECT_EQ(data.m, 0);
}

TYPED_TEST(GridDataTest, Equality) {
using T = TypeParam;
GridData<T> data1;
Expand Down
Loading

0 comments on commit d06fcbf

Please sign in to comment.