-
Notifications
You must be signed in to change notification settings - Fork 10
Adding New Functionality (Example)
On this Wiki page, we will sketch out how one might go about adding new functionality to NTPoly. The example we give will be based on trying to add a McWeeny purification module.
McWeeny purification can be used to compute the density matrix from a Hamiltonian using a very tight recursive scheme:
P_{k+1} = 3P_k^2 - 2P_k^3
P_0 = \frac{\lambda}{2}(\mu I - H) + \frac{1}{2}I
Here is a potential python implementation:
p_0 = (lambda_val/2.0) * (chemical_potential*identity_mat - hamiltonian) + 0.5*identity_mat
for i in range(0,100):
p2 = numpy.dot(p_0,p_0)
p3 = numpy.dot(p_0,p2)
p0 = 3.0*p2 - 2.0*p3
First, it is important to build an outline of the routine so that it will be easy to test. Let's begin by adding a new subroutine in Source/Fortran/DensityMatrixSolversModule.F90.
Fortran Code Outline
subroutine SolveMcWeeny(Hamiltonian, InverseSquareRoot, Density, &
& chemical_potential, energy_value_out solver_parameters_in)
type(Matrix_ps), intent(in) :: Hamiltonian
type(Matrix_ps), intent(in) :: InverseSquareRoot
integer, intent(in) :: nel
type(Matrix_ps), intent(inout) :: Density
real(ntreal), intent(out), optional :: energy_value_out
real(ntreal), intent(out) :: chemical_potential
type(SolverParameters_t), intent(in), optional :: solver_parameters_in
!! Handling Optional Parameters
type(SolverParameters_t) :: solver_parameters
!! Optional Parameters
if (present(solver_parameters_in)) then
solver_parameters = solver_parameters_in
else
solver_parameters = SolverParameters_t()
end if
end subroutine SolveMcWeeny
This new module will take in the necessary matrices along with the solver parameters and the chemical potential. Before coding the actual solver, we need to wrap up this module so it an be called from C++. This time, we go to the Source/Wrappers/ directory, and add to the DensityMatrixSolvers_wrp.f90 file.
Wrapper Code Outline
subroutine SolveMcWeeny_wrp(ih_Hamiltonian, ih_InverseSquareRoot, &
& ih_Density, chemical_potential, energy_value_out, ih_solver_parameters) &
& bind(c,name="SolveMcWeeny")
integer(kind=c_int), intent(in) :: ih_Hamiltonian(SIZE_wrp)
integer(kind=c_int), intent(in) :: ih_InverseSquareRoot(SIZE_wrp)
integer(kind=c_int), intent(inout) :: ih_Density(SIZE_wrp)
real(ntreal), intent(out) :: energy_value_out
real(ntreal), intent(in) :: chemical_potential
integer(kind=c_int), intent(in) :: ih_solver_parameters(SIZE_wrp)
type(Matrix_ps_wrp) :: h_Hamiltonian
type(Matrix_ps_wrp) :: h_InverseSquareRoot
type(Matrix_ps_wrp) :: h_Density
type(Matrix_ps_wrp) :: h_solver_parameters
h_Hamiltonian = transfer(ih_Hamiltonian,h_Hamiltonian)
h_InverseSquareRoot = transfer(ih_InverseSquareRoot,h_InverseSquareRoot)
h_Density = transfer(ih_Density,h_Density)
h_solver_parameters = transfer(ih_solver_parameters, h_solver_parameters)
call SolveMcWeeny(h_Hamiltonian%data, h_InverseSquareRoot%data, &
& h_Density%data, chemical_potential, h_solver_parameters%data)
end subroutine SolveMcWeeny_wrp
This wrapper is necessary to hide the distributed datatypes from C. Writing a subroutine like this is mostly just copy and pasting. For all the derived types, you need to pass an integer handle to it, and then use the transfer function to convert handle to the real object. We also need to add a signature to the C header file Source/C/DensityMatrixSolvers_c.h:
Next we add an object oriented C++ layer in Source/CPlusPlus.
C++ Header File
static void McWeeny(const Matrix_ps &Hamiltonian,
const Matrix_ps &InverseSquareRoot, int nel,
Matrix_ps &Density, double &chemical_potential,
double &energy_value_out,
const SolverParameters &solver_parameters);
C++ Source File
void DensityMatrixSolvers::McWeeny(const Matrix_ps &Hamiltonian,
const Matrix_ps &Overlap, int nel,
Matrix_ps &Density,
double &chemical_potential_out,
double &energy_value_out,
const SolverParameters &solver_parameters) {
McWeeny_wrp(GetIH(Hamiltonian), GetIH(Overlap), &nel, GetIH(Density),
&chemical_potential, &energy_value_out, GetIH(solver_parameters));
}
This is also pretty much just copy and pasting.
We've now added four new files. Each one of those files need to be added to the CMakeLists.txt files in their individual folders. Next, in the Swig directory, there is a file NTPolySwig.i . Here you need to add include statements for the new C++ header files.
#include "McWeenySolvers.h"
%include "McWeenySolvers.h"
Now let's add a new test that calls and verifies our McWeeny routine. In the UnitTests/Tests/ folder, there is testChemistry.py.in file. Here if we add a new test, Python will automatically add it to our unit testing setup. The Chemistry solvers have been specially set up to compare to some precomputed ground truth data.
def test_mcweeny(self):
fock_matrix = nt.DistributedSparseMatrix(self.hamiltonian)
overlap_matrix = nt.DistributedSparseMatrix(self.overlap)
inverse_sqrt_matrix = nt.DistributedSparseMatrix(
fock_matrix.GetActualDimension())
density_matrix = nt.DistributedSparseMatrix(
fock_matrix.GetActualDimension())
permutation = nt.Permutation(fock_matrix.GetLogicalDimension())
permutation.SetRandomPermutation()
self.solver_parameters.SetLoadBalance(permutation)
nt.SquareRootSolvers.InverseSquareRoot(overlap_matrix,inverse_sqrt_matrix,
self.solver_parameters)
chemical_potential = 0.0
nt.McWeenySolvers.Solve(fock_matrix, inverse_sqrt_matrix,
density_matrix, chemical_potential, self.solver_parameters)
density_matrix.WriteToMatrixMarket(self.result_file)
comm.barrier()
self.check_full()
Now when one runs make test
, this test will be automatically executed.
Note that this is a .py.in file. Any changes here require you to recompile
the code, even though it is python.
Finally we can return to the first Fortran file and implement the actual McWeeny purification routine. First let's set up the local variables we'll need.