Energy minimisation
ChemShell contains a wide range of geometry optimisation algorithms through its close integration with the DL-FIND geometry optimisation library. The simplest optimisation task that can be carried out is to minimise the energy of a system on its potential energy surface, which we demonstrate in this tutorial.
Minimisation of water
In the previous tutorial we calculated the single-point energy of a water
molecule. We will now look at minimisation of the same molecule using the
example inputs in the directory samples/minimisation/
.
For the NWChem case, the example is min_water.nwchem.py
,
with equivalents also available for GAMESS-UK and ORCA.
Take a look at the input. You will see that most of the commands are exactly the same as in the single-point case, with the same geometry being read in and the same theory definition. The only difference is the lines requesting the task to be carried out:
# Request an optimisation using DL-FIND
opt = Opt(theory=my_theory)
opt.run()
In this case, we have requested a DL-FIND geometry optimisation by defining an
Opt()
object. As we haven’t specified any options in this object besides
the name of the theory object, it will run an energy minisation by default.
As before, calling the run()
method will execute the calculation.
When the calculation is run, DL-FIND will print information on the energy, step taken and gradient changes at each cycle, and will continue until convergence on all these measures is reached. When the calculation has finished, compare the energy output at the end with the energy reported in the first cycle to check that it has indeed been lowered.
After the optimisation has completed, the save()
method can be used to
save the optimised geometry to disk for further analysis:
# Save the optimised geometry to disk
my_mol.save("water_opt.xyz")
Load water.xyz
and water_opt.xyz
into your visualiser. Note that
the starting geometry was close to the minimum in order to make sure that the
example ran quickly, but you should still see a small difference in the atomic
positions between the starting and optimised geometries.
Minimisation of nitrogen dioxide
In the second example, we look at how charged and open shell calculations can be performed and how these affect the resulting optimised structure for the case of nitrogen dioxide.
First let’s look at the file min_no2.nwchem.py
which runs an optimisation
on uncharged NO2. First we read in the starting structure as before:
# Read in NO2 geometry from disk
my_mol = Fragment(coords="no2.xyz")
Then we define the level of theory. NO2 is an open shell doublet system, which
we select with the options scftype="uhf"
, which specifies an unrestricted
HF or DFT calculation, and mult=2
to set the doublet spin multiplicity:
# Set up the QM calculation
my_theory = NWChem(frag=my_mol, method="dft", functional="blyp", basis="3-21g",
scftype="uhf", mult=2)
The minimisation is then requested in the same way as for water, except this time we will introduce further DL-FIND options:
# Request an optimisation using DL-FIND
opt = Opt(theory=my_theory, tolerance=0.0003, save_path=True)
opt.run()
Here, we show how to set a tighter threshold for convergence of the
optimisation, using the tolerance
option, and include the option
save_path
which instructs ChemShell/DL-FIND to save further input and output
files that can be used to help monitor the course of the optimisation. These
will be saved in a folder called _dl_find
. Of particular interest is the
file path.xyz
, which contains a series of XYZ geometries corresponding to
the full optimisation path.
When the calculation is finished, open _dl_find/path.xyz
in your visualiser
and take a look at how the structure changed as the optimisation progressed.
Note the final conformation that the molecule adopts, which will be the same as
the geometry saved to no2_opt.xyz
.
Now let’s take a look at min_no2p.nwchem.py
. Positively charged NO2 is a
closed shell system, so we use the default restricted SCF calculation.
The charge is specified by the option charge=1
:
# Set up the QM calculation
my_theory = NWChem(frag=my_mol, method="dft", functional="blyp", basis="3-21g",
charge=1)
The optimisation is then run in the same way as in the uncharged system.
When it is finished read in the two optimised geometries no2_opt.xyz
and no2p_opt.xyz
into your visualiser. How does the positively charged
structure compare with the uncharged case?