Calculation of translational, rotational and vibrational density of states
-
CBLAS (openblas can be problematic in combination with OPENMP)
-
LAPACKE (Do not use 3.9.0! there is a bug that causes wrong eigenvectors. 3.8, 3.9.1 are fine.)
-
FFTW
-
Chemfiles (below 0.9.3 does not contain .trr reader)
# needed if libxdrfile and/or cJSON are not installed globally
CMAKE_PREFIX_PATH=${CMAKE_PREFIX_PATH}:/path/to/libxdrfile
CMAKE_PREFIX_PATH=${CMAKE_PREFIX_PATH}:/path/to/cJSON
export CMAKE_PREFIX_PATH
INSTALL_PREFIX=/choose/destination/for/dos-calc
mkdir build
pushd build
cmake -DCMAKE_INSTALL_PREFIX=$INSTALL_PREFIX ..
make
make install
popd
By default RPATH is used for the location of libxdrfile.so and libcjson.so.1, but it can be turned off using -DUSE_RPATH=OFF
.
If turned off, libxdrfile.so and libcjson.so.1 have to be in a standard directory or in LD_LIBRARY_PATH
at runtime.
There is also a scripts folder, but those scripts are not automatically installed anywhere.
Check dos-calc --help
for command line options.
In any case one needs to provide a parameter file in JSON format (e.g. params.json
) and a trajectory in any format Chemfiles supports.
The output is be written to the JSON file dos.json
or whatever filename specified with -o
.
A example params.json of a mixture of three-point model water with united atom methanol.
{
"nsamples": 2,
"nblocks": 3,
"nblocksteps": 1000,
"moltypes": [
{
"nmols": 1950,
"atom_masses": [ 15.9994, 1.008, 1.008],
"rot_treat": "f",
"abc_indicators": [1, 2, 0, -1]
},
{
"nmols": 50,
"atom_masses": [ 15.9994, 1.008, 15.035 ],
"rot_treat": "f",
"abc_indicators": [1, 2, 0, -1]
}
],
"cross_spectra": [
{
"name": "water_trn water_vib",
"type": "i",
"dof_pairs":
[
[{"moltype": 0, "dof_type": "t", "dof_list": [0, 1, 2]}, {"moltype": 0, "dof_type": "v", "dof_list": [0, 1, 2, 3, 4, 5, 6, 7, 8]}]
]
}
]
}
The following visualisation shows the effect of nsamples = 2
and nblocks = 3
.
---------------------trajectory----------------------
----------sample0--------- ----------sample1---------
-block0- -block1- -block2- -block0- -block1- -block2-
In the consequence the trajectory must have equal or more than nsamples * nblocks * nblocksteps
frames with positions and velocities.
For each sample DosCalc is generating power spectra in the output file. Each sample can consist of multipe blocks that contribute to the sample's DoS (for example to reduce noise).
The order of moltypes and the atoms must reflect the structure of the trajectory!
nmols
is the number of molecules of that type.atom_masses
indicates the atoms of each molecule of that type.
One can define less atoms in total, than are present in the trajectory (produces a warning) but not more (produces an error).
For each moltype there is also:
-
rot_treat
, which is a char that determines the method used for the rotational DoS that is calculated. -
abc_indicators
, which is a list of four integers, which indicate two pairs of atoms (zero indexed). If the number is -1 it stands for the center of mass of the molecule, but this is only allowed for the second number of each pair. These atom pairs define the helping vectors a, b and c, that also form the auxiliary frame. The first two numbers define a, the second two define b'. c is the cross product of a and b', b is the cross product of c and a. Whenever principal axis are calculated in DosCalc, the auxiliary vectors will be used to switch them to point in a consistent direction. Therefore a, b, c should point roughly in the direction of the principal axes (from lowest to highest moment of inertia). They can also be used to directly output the rotational DoS with respact to the auxiliary frame (in this case their order is not important).
Depending on the point group of the molecule the following needs to be defined:
-
For atoms and monoatomic ions (if
atom_masses
has only one element)rot_treat
andabc_indicators
both are ignored. -
For linear molecules set
"rot_treat": "l"
.abc_indicators
is ignored. The rotational DoS is calculated with regard to the axes x, y, and z (but still be calledrot_a
, ... in the output file. Tested only for diatomic molecules. -
For molecules where the principal axis can swap order by vibration, for example ammonia, use the auxiliary frame by setting
"rot_treat": "a"
and"abc_indicators": [1, 2, 0 -1]
. Ammonia has atoms N H H H. One can not use the actual principal axes of the molecule, because they swap order during vibration. Therefore 1 2 defines vector a between two hydrogens. 0 -1 defines vector b' from the nitrogen atom to the COM and thereby along the symmetry axis. The rotational DoS is calculated be with regard to sqrt(I^aux_l) * ω^aux_l where l is one of the auxiliary axis a, b and c. Also the moments of inertia are given with respect to those axes (ignoring off-diagonal elements of the moments of inertia tensor). Note, that this does yield unusable results, if a, b and c are not close to the actual principal axes (but the order does not matter). -
For other molecules, for example water, set
"rot_treat": "f"
and"abc_indicators": [1, 2, 0 -1]
. Water has atoms O H H. Therefore 1 2 defines vector a between the two hydrogens. 0 -1 defines vector b' along the symmetry axis. The rotational DoS is calculated with regard to the principal axes of rotation. The vectors a, b and c are used to ensure, that the actual axis derived from the moment of inertia tensor, always point in the same direction. This will likely be the correct choice for most molecules, especially larger ones without symmetries. -
If one does not want any velocity separation
"rot_treat": "u"
can be used and the unseparated DoS is written tovib_{x,y,z}
.abc_indicators
is ignored. -
For Eckart separation use one of
e,E,p,E
forrot_treat
.p
andP
are used for planar,e
andE
for tree-dimensionsal molecules. The capital letters use sqrt(I^aux_l) * ω^aux_l where l is one of the auxiliary vectors for output. Use these when the princincipal axis frame is not unique as described above for"rot_treat": "a"
. The lower-case letters will return sqrt(I^pa_l) * ω^pa where l is one of the principal axis. For Eckart separation you also need to provide a reference structure for each molecule. It can for example be obtained by running a steepest decent run with all intermolecular interactions turned off. Note that if the molecule has rotating sub-groups such as methyl-groups then the alignment of the reference structure to the molecule will produce artifacts in the rotational DoS.
The name
of a cross spectrum
can be up to 79 characters long and can help identify the specified cross correlation in the output file.
The type
of a cross spectrum
can be "i" for inside or "e" for external.
Option "i" does result in a cross correlation of degrees of freedom within molecules and an average over molecules.
Option "e" does correlate all specified degrees of freedom from all matching molecules (can take a long time).
The variable dof_type
can be one of:
- "t" for translational xyz-component of the molecule. Possible indices for dof_list: 0, 1, 2.
- "r" for rotational xyz-component of each atom in the molecule. Possible indices for dof_list: 0, 1, ..., 3*mol_natoms.
- "v" for vibrational xyz-component of each atom in the molecule. Possible indices for dof_list: 0, 1, ..., 3*mol_natoms.
- "o" for rotational abc-component of the molecule. Possible indices for dof_list: 0, 1, 2.
- "c" for rotational-vibrational-coupled xyz-component of each atom in the molecule. Possible indices for dof_list: 0, 1, ..., 3*mol_natoms.
Here, mol_natoms indicates the number of atoms in one molecule of the chosen moleculetype.
A example dos.json corresponding to the example above with long lists of numbers shortened to [...]
.
{
"frequencies": [...],
"moltypes": [{
"spectra": {
"trn_x": [[...], [...]],
"trn_y": [[...], [...]],
"trn_z": [[...], [...]],
"rot_x": [[...], [...]],
"rot_y": [[...], [...]],
"rot_z": [[...], [...]],
"vib_x": [[...], [...]],
"vib_y": [[...], [...]],
"vib_z": [[...], [...]],
"rot_a": [[...], [...]],
"rot_b": [[...], [...]],
"rot_c": [[...], [...]]
},
"moments_of_inertia": [[...], [...]]
}, {
"spectra": {
"trn_x": [[...], [...]],
"trn_y": [[...], [...]],
"trn_z": [[...], [...]],
"rot_x": [[...], [...]],
"rot_y": [[...], [...]],
"rot_z": [[...], [...]],
"vib_x": [[...], [...]],
"vib_y": [[...], [...]],
"vib_z": [[...], [...]],
"rot_a": [[...], [...]],
"rot_b": [[...], [...]],
"rot_c": [[...], [...]]
},
"moments_of_inertia": [[...], [...]],
"moments_of_inertia_std": [[...], [...]],
"coriolis": [...]
}],
"cross_spectra": {
"water_trn water_vib": [[...], [...]]
}
}
The first dimension of each spectrum, moments_of_inertia
, moments_of_inertia_std
, and coriolis
is determined by nsamples
.
The second dimension of each spectrum list is the number of frequencies, which is floor(nblocksteps / 2) + 1
.
The Coriolis term will only be non-zero if Eckart decomposition is used.
A verbal (and therefore not exact) description of the DoS components:
trn_x
is the power spectrum of the x component of the velocities of the molecules center of mass translational motion.rot_x
is the power spectrum of the x component of the velocities of the atoms due to molecular rotational motion.vib_x
is the power spectrum of the x component of the velocities of the atoms due to molecular vibrational motion.rot_x
is the power spectrum of the angular velocity times square root of the moment of inertia, both with respect to lab axis 'x'.roto_a
is the power spectrum of the angular velocity times square root of the moment of inertia, both with respect to axis 'a' which is either a principal axis or a user defined axis.
Masses are assumed to be provided in u.
The framelength will be read from the trajectory for .trr files or from the command line argument and is assumed to be in picoseconds.
DosCalc relies on Chemfiles for reading the trajectory. Chemfiles usually converts units to Å
and Å/ps
. DosCalc converts those to nm
and nm/ps
by dividing positions and velocities by 10. Thereby the Gromacs unit system is established and the unit of energy is [E] = u * nm²/ps² = kJ/mol
.
However, for some formats, like lammps dumps, chemfiles can not infer the unit of the velocity and proviedes it unmodified. The energy will have the unit [E] = u * ([v] * 10)²
. So for example if lammps runs with units real, then [v] = Å/fs
and therefore [E] = u * Ų/fs² * 100 = u * nm²/fs² = 10^6 kJ/mol
.
The unit of the DoS is [S] = [E] * ps
.
- Was tested only with Lammps and Gromacs trajectories.
- PBC recombination does not work for non-orthorhombic boxes, will produce an error.
However one can make all molecules whole in the trajectory (for example with
gmx trjconv -pbc mol
) before using DosCalc and then use the--no-pbc
option. Whole molecules are alowed to jump between frames so a full unwrapping of the trajectory is not necessary. - The program uses single precision (if you find this is not sufficient, it should be simple replacing all
float
withdouble
or make it a compile option). - Linear molecules are assumed to be diatomic.
- Not tested on periodic molecules.
Check show-doses.py --help
for command line options.