Skip to content

Commit

Permalink
Port shape_plugin from HOOMD-blue.
Browse files Browse the repository at this point in the history
  • Loading branch information
joaander committed Dec 2, 2024
1 parent d97980e commit d12b7dd
Show file tree
Hide file tree
Showing 9 changed files with 329 additions and 23 deletions.
22 changes: 11 additions & 11 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,9 +1,9 @@
# HOOMD-blue component template
# HOOMD-blue component template for HPMC shapes

`hoomd-component-template` provides a framework to develop components that extend
[**HOOMD-blue**](https://glotzerlab.engin.umich.edu/hoomd-blue/). It includes template C++ and
Python modules, an example unit test, CMake scripts to build the component, and GitHub Actions
workflows.
`hoomd-hpmc-shape-template` provides a framework to develop components that extend
[**HOOMD-blue**](https://glotzerlab.engin.umich.edu/hoomd-blue/) with new hard shapes.
It includes template C++ and Python modules, an example unit test, CMake scripts to
build the component, and GitHub Actions workflows.

## Building the component

Expand All @@ -12,19 +12,19 @@ To build this component:
1. Build and install **HOOMD-blue** from source.
2. Obtain the component's source.
```
$ git clone https://github.com/glotzerlab/hoomd-component-template
$ git clone https://github.com/glotzerlab/hoomd-hpmc-shape-template
```
3. Configure.
```
$ cmake -B build/hoomd-component-template -S hoomd-component-template
$ cmake -B build/hoomd-hpmc-shape-template -S hoomd-hpmc-shape-template
```
4. Build the component.
```
$ cmake --build build/hoomd-component-template
$ cmake --build build/hoomd-hpmc-shape-template
```
5. Install the component.
```
$ cmake --install build/hoomd-component-template
$ cmake --install build/hoomd-hpmc-shape-template
```
Once installed, the template is available for import via:
Expand All @@ -37,7 +37,7 @@ import hoomd.template
To create a new component:
1. Fork [hoomd-component-template](https://github.com/glotzerlab/hoomd-component-template/).
1. Fork [hoomd-hpmc-shape-template](https://github.com/glotzerlab/hoomd-hpmc-shape-template/).
2. Address all **TODO** comments (including those in `.github/`)
3. Add C++ and Python files to `src/`.
4. Add unit tests in `src/pytest`.
Expand All @@ -57,7 +57,7 @@ new GitHub release with automatically generated release notes.
## Maintaining your component
The HOOMD-blue developers will periodically update
[hoomd-component-template](https://github.com/glotzerlab/hoomd-component-template/), including
[hoomd-hpmc-shape-template](https://github.com/glotzerlab/hoomd-hpmc-shape-template/), including
updates to the GitHub Actions workflow, pre-commit configuration, and CMake scripts. Merge these
changes into your fork to support the latest version of HOOMD-blue.
Expand Down
2 changes: 2 additions & 0 deletions src/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -8,11 +8,13 @@ set(_${COMPONENT_NAME}_sources

# TODO: List all GPU C++ source code files in _${COMPONENT_NAME}_cu_sources.
set(_${COMPONENT_NAME}_cu_sources
kernels.cu
)

# TODO: List all Python modules in python_files.
set(python_files
__init__.py
integrate.py
version.py
)

Expand Down
175 changes: 175 additions & 0 deletions src/ShapeMySphere.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,175 @@
// Copyright (c) 2009-2024 The Regents of the University of Michigan.
// Part of HOOMD-blue, released under the BSD 3-Clause License.

#pragma once

#include "hoomd/AABB.h"
#include "hoomd/BoxDim.h"
#include "hoomd/HOOMDMath.h"
#include "hoomd/VectorMath.h"
#include "hoomd/hpmc/HPMCMiscFunctions.h"
#include "hoomd/hpmc/Moves.h"
#include "hoomd/hpmc/OBB.h"
#include "hoomd/hpmc/ShapeSphere.h"

#include <sstream>

#include <stdexcept>

#ifdef __HIPCC__
#define DEVICE __device__
#define HOSTDEVICE __host__ __device__
#else
#define DEVICE
#define HOSTDEVICE
#include <pybind11/pybind11.h>
#endif

namespace hoomd
{
namespace hpmc
{

struct MySphereParams : ShapeParams
{
/// The radius of the sphere
ShortReal radius;

/// True when move statistics should not be counted
bool ignore;

/// True when the shape may be oriented
bool isOriented;

#ifdef ENABLE_HIP
/// Set CUDA memory hints
void set_memory_hint() const { }
#endif

#ifndef __HIPCC__

/// Default constructor
MySphereParams() { }

/// Construct from a Python dictionary
MySphereParams(pybind11::dict v, bool managed = false)
{
ignore = v["ignore_statistics"].cast<bool>();
radius = v["radius"].cast<ShortReal>();
isOriented = v["orientable"].cast<bool>();
}

/// Convert parameters to a python dictionary
pybind11::dict asDict()
{
pybind11::dict v;
v["radius"] = radius;
v["orientable"] = isOriented;
v["ignore_statistics"] = ignore;
return v;
}

#endif
} __attribute__((aligned(32)));

struct ShapeMySphere
{
/// Define the parameter type
typedef MySphereParams param_type;

/// Construct a shape at a given orientation
DEVICE ShapeMySphere(const quat<Scalar>& _orientation, const param_type& _params)
: orientation(_orientation), params(_params)
{
}

/// Check if the shape may be rotated
DEVICE bool hasOrientation() const
{
return params.isOriented;
}

/// Check if this shape should be ignored in the move statistics
DEVICE bool ignoreStatistics() const
{
return params.ignore;
}

/// Get the circumsphere diameter of the shape
DEVICE ShortReal getCircumsphereDiameter() const
{
return params.radius * ShortReal(2.0);
}

/// Get the in-sphere radius of the shape
DEVICE ShortReal getInsphereRadius() const
{
return params.radius;
}

/// Return the bounding box of the shape in world coordinates
DEVICE hoomd::detail::AABB getAABB(const vec3<Scalar>& pos) const
{
return hoomd::detail::AABB(pos, params.radius);
}

/// Return a tight fitting OBB around the shape
DEVICE detail::OBB getOBB(const vec3<Scalar>& pos) const
{
return detail::OBB(pos, params.radius);
}

/// Returns true if this shape splits the overlap check over several threads of a warp using
/// threadIdx.x
HOSTDEVICE static bool isParallel()
{
return false;
}

/// Returns true if the overlap check supports sweeping both shapes by a sphere of given radius
HOSTDEVICE static bool supportsSweepRadius()
{
return true;
}

quat<Scalar> orientation; //!< Orientation of the sphere (unused)

/// MySphere parameters
const MySphereParams& params;
};

//! MySphere-MySphere overlap
/*! \param r_ab Vector defining the position of shape b relative to shape a (r_b - r_a)
\param a first shape
\param b second shape
\param err in/out variable incremented when error conditions occur in the overlap test
\returns true when *a* and *b* overlap, and false when they are disjoint
\ingroup shape
*/
template<>
DEVICE inline bool test_overlap<ShapeMySphere, ShapeMySphere>(const vec3<Scalar>& r_ab,
const ShapeMySphere& a,
const ShapeMySphere& b,
unsigned int& err)
{
vec3<ShortReal> dr(r_ab);

ShortReal rsq = dot(dr, dr);

ShortReal RaRb = a.params.radius + b.params.radius;
if (rsq < RaRb * RaRb)
{
return true;
}
else
{
return false;
}
}

} // end namespace hpmc
} // end namespace hoomd

#undef DEVICE
#undef HOSTDEVICE
1 change: 1 addition & 0 deletions src/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,3 +6,4 @@

# TODO: Import all Python modules in your component.
from . import version
from .integrate import MySphere
44 changes: 44 additions & 0 deletions src/integrate.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
# Copyright (c) 2009-2024 The Regents of the University of Michigan.
# Part of HOOMD-blue, released under the BSD 3-Clause License.

"""Example Shape Integrator."""

# Import the C++ module
from . import _template

import hoomd


class MySphere(hoomd.hpmc.integrate.HPMCIntegrator):
"""Example shape integrator."""

# set static class data
_ext_module = _template
_cpp_cls = "IntegratorHPMCMonoMySphere"

def __init__(
self,
default_d=0.1,
default_a=0.1,
translation_move_probability=0.5,
nselect=4,
kT=1.0,
):
# initialize base class
super().__init__(
default_d, default_a, translation_move_probability, nselect, kT
)

typeparam_shape = hoomd.data.typeparam.TypeParameter(
"shape",
type_kind="particle_types",
param_dict=hoomd.data.parameterdicts.TypeParameterDict(
radius=float, ignore_statistics=False, orientable=False, len_keys=1
),
)
self._add_typeparam(typeparam_shape)

@hoomd.logging.log(category="object", requires_run=True)
def type_shapes(self):
"""list[dict]: Description of shapes in ``type_shapes`` format."""
return super()._return_type_shapes()
39 changes: 39 additions & 0 deletions src/kernels.cu
Original file line number Diff line number Diff line change
@@ -0,0 +1,39 @@
// Copyright (c) 2009-2024 The Regents of the University of Michigan.
// Part of HOOMD-blue, released under the BSD 3-Clause License.

#include "ShapeMySphere.h"
#include "hoomd/hpmc/ComputeFreeVolumeGPU.cuh"
#include "hoomd/hpmc/IntegratorHPMCMonoGPU.cuh"
#include "hoomd/hpmc/IntegratorHPMCMonoGPUMoves.cuh"
#include "hoomd/hpmc/UpdaterGCAGPU.cuh"

namespace hoomd
{
namespace hpmc
{
namespace detail
{
template hipError_t
gpu_hpmc_free_volume<ShapeMySphere>(const hpmc_free_volume_args_t& args,
const typename ShapeMySphere::param_type* d_params);
}
namespace gpu
{
template void hpmc_gen_moves<ShapeMySphere>(const hpmc_args_t& args,
const ShapeMySphere::param_type* params);

template void hpmc_narrow_phase<ShapeMySphere>(const hpmc_args_t& args,
const ShapeMySphere::param_type* params);

template void hpmc_update_pdata<ShapeMySphere>(const hpmc_update_args_t& args,
const ShapeMySphere::param_type* params);

template void hpmc_cluster_overlaps<ShapeMySphere>(const cluster_args_t& args,
const ShapeMySphere::param_type* params);

template void transform_particles<ShapeMySphere>(const clusters_transform_args_t& args,
const ShapeMySphere::param_type* params);
} // namespace gpu

} // end namespace hpmc
} // end namespace hoomd
38 changes: 37 additions & 1 deletion src/module.cc
Original file line number Diff line number Diff line change
Expand Up @@ -4,21 +4,57 @@
// TODO: Include the header files of classes that will be exported to Python.

#include <pybind11/pybind11.h>
#include "hoomd/hpmc/ComputeFreeVolume.h"
#include "hoomd/hpmc/IntegratorHPMC.h"
#include "hoomd/hpmc/IntegratorHPMCMono.h"

#include "hoomd/hpmc/ComputeSDF.h"
#include "hoomd/hpmc/ShapeUnion.h"

#include "hoomd/hpmc/ExternalFieldHarmonic.h"
#include "hoomd/hpmc/ExternalFieldWall.h"

#include "hoomd/hpmc/UpdaterGCA.h"
#include "hoomd/hpmc/UpdaterMuVT.h"

#include "ShapeMySphere.h"

#ifdef ENABLE_HIP
#include "hoomd/hpmc/ComputeFreeVolumeGPU.h"
#include "hoomd/hpmc/IntegratorHPMCMonoGPU.h"
#include "hoomd/hpmc/UpdaterGCAGPU.h"
#endif

using namespace hoomd::hpmc::detail;

namespace hoomd
{
namespace md
namespace hpmc
{

// TODO: Set the name of the python module to match ${COMPONENT_NAME} (set in
// CMakeLists.txt), prefixed with an underscore.
PYBIND11_MODULE(_template, m)
{
// TODO: Call export_Class(m) for each C++ class to be exported to Python.
export_IntegratorHPMCMono<ShapeMySphere>(m, "IntegratorHPMCMonoMySphere");
export_ComputeFreeVolume<ShapeMySphere>(m, "ComputeFreeVolumeMySphere");
export_ComputeSDF<ShapeMySphere>(m, "ComputeSDFMySphere");
export_UpdaterMuVT<ShapeMySphere>(m, "UpdaterMuVTMySphere");
export_UpdaterGCA<ShapeMySphere>(m, "UpdaterGCAMySphere");

export_ExternalFieldWall<ShapeMySphere>(m, "WallMySphere");

pybind11::class_<MySphereParams, std::shared_ptr<MySphereParams>>(m, "MySphereParams")
.def(pybind11::init<pybind11::dict>())
.def("asDict", &MySphereParams::asDict);

#ifdef ENABLE_HIP
// TODO: Call export_ClassGPU(m) for each GPU enabled C++ class to be exported
// to Python.
export_IntegratorHPMCMonoGPU<ShapeMySphere>(m, "IntegratorHPMCMonoMySphereGPU");
export_ComputeFreeVolumeGPU<ShapeMySphere>(m, "ComputeFreeVolumeMySphereGPU");
export_UpdaterGCAGPU<ShapeMySphere>(m, "UpdaterGCAMySphereGPU");
#endif
}

Expand Down
Loading

0 comments on commit d12b7dd

Please sign in to comment.