A Python library for nonadiabatic molecular dynamics, implementing Fewest Switches Surface Hopping (FSSH) and related methods. Includes Tully's model problems (Tully, J.C. J. Chem. Phys. (1990) 93 1061) as well as interfaces to ab initio electronic structure codes. Supports diabatic and adiabatic models with two or more electronic states and one or more dimensional potentials.
mudslidepackage that contains- nonadiabatic and adiabatic dynamics methods
SurfaceHoppingMD- Standard FSSH implementationEhrenfest- Ehrenfest dynamicsAugmentedFSSH- Augmented FSSH (A-FSSH) implementationEvenSamplingTrajectory- FSSH with even sampling of phase spaceAdiabaticMD- Adiabatic (ground state) molecular dynamics
- collection of 1D model potentials
TullySimpleAvoidedCrossingTullyDualAvoidedCrossingTullyExtendedCouplingReflectionSuperExchangeSubotnikModelXSubotnikModelSSubotnikModelWSubotnikModelZShinMetiuLinearVibronic
- 2D models
Subotnik2D
- ab initio interfaces
TMModel- Turbomole interface for TDDFT-based NAMDOpenMM- OpenMM interface for classical MDQMMM- QM/MM combining Turbomole and OpenMMHarmonicModel- Harmonic approximation from Hessian data
- nonadiabatic and adiabatic dynamics methods
mudslidescript that runs simple model trajectoriesmudscript providing a unified CLI with subcommandsmudslide-surfacescript that prints 1D surfaces and couplings
- numpy
- scipy (for Shin-Metiu model)
- pyyaml
Note: turboparse is bundled internally within mudslide.
Mudslide has switched to a proper python package structure, which means to work properly it now needs to be "installed". The most straightforward way to do this is
cd /path/to/mudslide
pip install --user -e .
which install into your user installation dir. You can find out your user installation directory with the command
python -m site --user-base
To set up your PATH and PYTHONPATH to be able to use both the command line scripts
and the python package, use
export PATH=$(python -m site --user-base)/bin:$PATH
export PYTHONPATH=$(python -m site --user-base):$PYTHONPATH
Sets of simulations are run using the BatchedTraj class. A BatchedTraj object must be instantiated by passing a model object
(handles electronic PESs and couplings), and a traj_gen
generator that generates new initial conditions. Some simple canned examples are provided for traj_gen. All
other options are passed as keyword arguments to the constructor. The compute() function of the
BatchedTraj object returns a TraceManager object that contains all the results, but functionally behaves
like a python dictionary. Custom TraceManagers can also be
provided. For example:
import mudslide
simple_model = mudslide.models.TullySimpleAvoidedCrossing()
# Generates trajectories always with starting position -5, starting momentum 10.0, on ground state
traj_gen = mudslide.TrajGenConst(-5.0, 10.0, 0)
simulator = mudslide.BatchedTraj(simple_model, traj_gen, mudslide.SurfaceHoppingMD, samples = 4, bounds=[[-4],[4]])
results = simulator.compute()
outcomes = results.outcomes
print("Probability of reflection on the ground state: %12.4f" % outcomes[0,0])
print("Probability of transmission on the ground state: %12.4f" % outcomes[0,1])
print("Probability of reflection on the excited state: %12.4f" % outcomes[1,0])
print("Probability of transmission on the excited state: %12.4f" % outcomes[1,1])
will run 4 scattering simulations with a particle starting in the ground state (0) at x=-5.0 a.u. and traveling with an initial momentum of 10.0 a.u.
initial_state- specify the initial electronic state- 0 (default) - start on the ground state
mass- particle mass (default: 2000 a.u.)initial_time- starting value oftimevariable (default: 0.0 a.u.)dt- timestep (default: abs(0.05 / velocity)))total_time- total simulation length (default: 2 * abs(position/velocity))samples- number of trajectories to run (default: 2000)seed- random seed for trajectories (defaults however numpy does)electronic_integration- method used to propagate electronic wavefunction- "exp" (default) - apply exponentiated Hamiltonian via diagonalization
- "linear-rk4" - interpolated RK4 integration
outcome_type- how to count statistics at the end of a trajectory- "state" (default) - use the state attribute of the simulation only
- "populations" - use the diagonals of the density matrix
trace_every- save snapshot data every nth step (i.e., whennsteps%trace_every==0)
A primary goal of mudslide is to include the flexibility for users to provide their own
model classes. To interface with mudslide, a model class should derive from mudslide.ElectronicModel_
and should implement
- a
compute()function that computes energies, forces, and derivative couplings - an
nstates()function that returns the number of electronic states in the model - an
ndof()function that returns the number of classical (vibrational) degrees of freedom in the model
The compute() function needs to have the following signature:
def compute(self, X: ArrayLike, couplings: Any = None, gradients: Any = None, reference: Any = None) -> None
The X input to the compute() function is an array of the positions. All other inputs are ignored for now
but will eventually be used to allow the trajectory to enumerate precisely which quantities are desired at
each call.
At the end of the compute() function, the object must store
self._hamiltonian- Annstates x nstatesarray of the Hamiltonian (nstatesis the number of electronic states)self._force- Annstates x ndimarray of the force on each PES (ndimis the number of classical degrees of freedom)self._derivative_coupling- Annstates x nstates x ndimarray wherederivative_coupling[i,j,:]contains <i|d/dR|j>self._forces_available- A boolean array of lengthnstatesindicating which forces were computedself._derivative_couplings_available- Annstates x nstatesboolean array indicating which couplings were computed
See the file mudslide/turbomole_model.py for an example of a standalone ab initio model.
For diabatic models, such as Tully's scattering models, there is a helper class, mudslide.DiabaticModel_ to simplify construction of model classes.
To use it, have your model derive from mudslide.DiabaticModel_ and then you just need to implement two helper functions:
V(self, x)--- returns (nstates,nstates)-shaped numpy array containing the diabatic Hamiltonian matrix at nuclear positionxdV(self, x)--- returns (nstates,nstates,ndim)-shaped numpy array containing the gradient of the diabatic Hamiltonian matrix at nuclear positionx
And further define two class variables:
nstates_--- the number of electronic states in the modelndim_--- the number nuclear degrees of freedom in the model
In all cases, the input x ought to be a numpy array with ndim elements.
See the file mudslide/scattering_models.py for examples, which includes, among other models, the original Tully models.
Additional models included are:
SuperExchange- Oleg Prezhdo's three-state super exchange model from Wang, Trivedi, Prezhdo JCTC (2014) doi:10.1021/ct5003835SubotnikModelX- 'Model X' from Subotnik JPCA (2011) doi: 10.1021/jp206557hSubotnikModelS- 'Model S' from Subotnik JPCA (2011) doi: 10.1021/jp206557hSubotnik2D- 2D model from Subotnik JPCA (2011) doi: 10.1021/jp206557hShin-Metiu
For batch runs, one must tell BatchedTraj how to decide on new initial conditions
and how to decide when a trajectory has finished. The basic requirements for each of those
is simple.
This should be a generator function that accepts a number of samples and returns a dictionary with starting conditions filled in, e.g., the yield statement should be something like
yield { "position" : x0, "momentum" : p0, "initial_state" : 0 }
See TrajGenConst for an example.
Scans of the surfaces can be printed using the mudslide-surface command that is included
for the installation. For usage, make sure your PATH includes the installation
directory for scripts (e.g., by running
export PATH=$(python -m site --user-base)/bin:$PATH) and run
mudslide-surface -h