Source code for pyscf.pbc.grad.krhf

#!/usr/bin/env python
# Copyright 2014-2020 The PySCF Developers. All Rights Reserved.
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
#
# Author: Yang Gao <younggao1994@gmail.com>

#
'''
Non-relativistic analytical nuclear gradients for restricted Hartree Fock with kpoints sampling
'''
import numpy as np
from pyscf import lib
from pyscf.lib import logger
from pyscf.grad import rhf as molgrad
from pyscf.pbc.gto.pseudo.pp import get_vlocG, get_alphas, get_projG, projG_li, _qli
from pyscf.pbc.dft.numint import eval_ao_kpts
from pyscf.pbc import gto, tools
from pyscf.gto import mole
from pyscf.pbc.gto.pseudo import pp_int
import scipy


[docs] def grad_elec(mf_grad, mo_energy=None, mo_coeff=None, mo_occ=None, atmlst=None): ''' Electronic part of KRHF/KRKS gradients Args: mf_grad : pbc.grad.krhf.Gradients or pbc.grad.krks.Gradients object ''' mf = mf_grad.base cell = mf_grad.cell kpts = mf.kpts nkpts = len(kpts) if mo_energy is None: mo_energy = mf.mo_energy if mo_occ is None: mo_occ = mf.mo_occ if mo_coeff is None: mo_coeff = mf.mo_coeff if atmlst is None: atmlst = range(cell.natm) log = logger.Logger(mf_grad.stdout, mf_grad.verbose) hcore_deriv = mf_grad.hcore_generator(cell, kpts) s1 = mf_grad.get_ovlp(cell, kpts) dm0 = mf.make_rdm1(mo_coeff, mo_occ) t0 = (logger.process_clock(), logger.perf_counter()) log.debug('Computing Gradients of NR-HF Coulomb repulsion') vhf = mf_grad.get_veff(dm0, kpts) log.timer('gradients of 2e part', *t0) dme0 = mf_grad.make_rdm1e(mo_energy, mo_coeff, mo_occ) aoslices = cell.aoslice_by_atom() de = np.zeros([len(atmlst),3]) for x, ia in enumerate(atmlst): p0, p1 = aoslices[ia,2:] h1ao = hcore_deriv(ia) de[x] += np.einsum('xkij,kji->x', h1ao, dm0).real # nabla was applied on bra in vhf, *2 for the contributions of nabla|ket> de[x] += np.einsum('xkij,kji->x', vhf[:,:,p0:p1], dm0[:,:,p0:p1]).real * 2 de[x] -= np.einsum('kxij,kji->x', s1[:,:,p0:p1], dme0[:,:,p0:p1]).real * 2 de[x] /= nkpts de[x] += mf_grad.extra_force(ia, locals()) de += pp_int.vppnl_nuc_grad(cell, dm0, kpts) / nkpts if log.verbose > logger.DEBUG: log.debug('gradients of electronic part') mf_grad._write(cell, de, atmlst) return de
def _make_fakemol(): fakemol = mole.Mole() fakemol._atm = np.zeros((1,mole.ATM_SLOTS), dtype=np.int32) fakemol._bas = np.zeros((1,mole.BAS_SLOTS), dtype=np.int32) ptr = mole.PTR_ENV_START fakemol._env = np.zeros(ptr+10) fakemol._bas[0,mole.NPRIM_OF ] = 1 fakemol._bas[0,mole.NCTR_OF ] = 1 fakemol._bas[0,mole.PTR_EXP ] = ptr+3 fakemol._bas[0,mole.PTR_COEFF] = ptr+4 return fakemol
[docs] def get_hcore(cell, kpts): ''' Part of the nuclear gradients of core Hamiltonian If pseudo potential is turned on, the local term is included, but the nonlocal term is not included. ''' h1 = np.asarray(cell.pbc_intor('int1e_ipkin', kpts=kpts)) dtype = h1.dtype if cell._pseudo: SI=cell.get_SI() coords = cell.get_uniform_grids() vlocG = get_vlocG(cell) vpplocG = -np.einsum('ij,ij->j', SI, vlocG) vpplocG[0] = np.sum(get_alphas(cell)) vpplocR = tools.ifft(vpplocG, cell.mesh).real for kn, kpt in enumerate(kpts): aos = eval_ao_kpts(cell, coords, kpt, deriv=1)[0] vloc = np.einsum('agi,g,gj->aij', aos[1:].conj(), vpplocR, aos[0]) if dtype == np.float64: h1[kn,:] += vloc.real else: h1[kn,:] += vloc else: raise NotImplementedError return h1
[docs] def get_ovlp(cell, kpts): return -np.asarray(cell.pbc_intor('int1e_ipovlp', kpts=kpts))
[docs] def hcore_generator(mf_grad, cell=None, kpts=None): ''' If pseudo potential is turned on, the local term is included, but the nonlocal term is not included. ''' if cell is None: cell = mf_grad.cell if kpts is None: kpts = mf_grad.kpts h1 = mf_grad.get_hcore(cell, kpts) aoslices = cell.aoslice_by_atom() SI=cell.get_SI() ##[natom ,grid] mesh = cell.mesh Gv = cell.Gv ##[grid, 3] coords = cell.get_uniform_grids() vlocG = get_vlocG(cell) ###[natom, grid] def hcore_deriv(atm_id): shl0, shl1, p0, p1 = aoslices[atm_id] vloc_g = 1j * np.einsum('ga,g->ag', Gv, SI[atm_id]*vlocG[atm_id]) nkpts, nao = h1.shape[0], h1.shape[2] hcore = np.zeros([3,nkpts,nao,nao], dtype=h1.dtype) for kn, kpt in enumerate(kpts): ao = eval_ao_kpts(cell, coords, kpt)[0] for ax in range(3): vloc_R = tools.ifft(vloc_g[ax], mesh).real vloc = np.einsum('gi,gj,g->ij', ao.conj(), ao, vloc_R, optimize=True) hcore[ax,kn] += vloc ao = None hcore[:,kn,p0:p1] -= h1[kn,:,p0:p1] hcore[:,kn,:,p0:p1] -= h1[kn,:,p0:p1].transpose(0,2,1).conj() return hcore return hcore_deriv
[docs] def grad_nuc(cell, atmlst): ''' Derivatives of nuclear repulsion energy wrt nuclear coordinates Notes: An optimized version of this function is available in `pbc.gto.ewald_methods.ewald_nuc_grad` ''' chargs = cell.atom_charges() ew_eta, ew_cut = cell.get_ewald_params() log_precision = np.log(cell.precision / (chargs.sum()*16*np.pi**2)) ke_cutoff = -2*ew_eta**2*log_precision mesh = cell.cutoff_to_mesh(ke_cutoff) logger.debug1(cell, 'mesh for ewald %s', mesh) coords = cell.atom_coords() Lall = cell.get_lattice_Ls() natom = len(chargs) ewovrl_grad = np.zeros([natom,3]) for i, qi in enumerate(chargs): ri = coords[i] for j in range(natom): if j == i: continue qj = chargs[j] rj = coords[j] r1 = ri-rj + Lall r = np.sqrt(np.einsum('ji,ji->j', r1, r1)) r = r.reshape(len(r),1) ewovrl_grad[i] += np.sum(- (qi * qj / r ** 3 * r1 * scipy.special.erfc(ew_eta * r).reshape(len(r),1)), axis = 0) ewovrl_grad[i] += np.sum(- qi * qj / r ** 2 * r1 * 2 * ew_eta / np.sqrt(np.pi) * np.exp(-ew_eta**2 * r ** 2).reshape(len(r),1), axis = 0) Gv, Gvbase, weights = cell.get_Gv_weights(mesh) absG2 = np.einsum('gi,gi->g', Gv, Gv) absG2[absG2==0] = 1e200 ewg_grad = np.zeros([natom,3]) SI = cell.get_SI(Gv) if cell.dimension != 2 or cell.low_dim_ft_type == 'inf_vacuum': coulG = 4*np.pi / absG2 coulG *= weights ZSI = np.einsum("i,ij->j", chargs, SI) ZexpG2 = coulG * np.exp(-absG2/(4*ew_eta**2)) ZexpG2_mod = ZexpG2.reshape(len(ZexpG2),1) * Gv else: raise NotImplementedError for i, qi in enumerate(chargs): Zfac = np.imag(ZSI * SI[i].conj()) * qi ewg_grad[i] = - np.sum(Zfac.reshape((len(Zfac),1)) * ZexpG2_mod, axis = 0) ew_grad = ewg_grad + ewovrl_grad if atmlst is not None: ew_grad = ew_grad[atmlst] return ew_grad
[docs] def get_jk(mf_grad, dm, kpts): '''J = ((-nabla i) j| kl) D_lk K = ((-nabla i) j| kl) D_jk ''' vj, vk = mf_grad.get_jk(dm, kpts) return vj, vk
[docs] def get_j(mf_grad, dm, kpts): return mf_grad.get_j(dm, kpts)
[docs] def get_k(mf_grad, dm, kpts): return mf_grad.get_k(dm, kpts)
[docs] def get_veff(mf_grad, dm, kpts): '''NR Hartree-Fock Coulomb repulsion''' vj, vk = mf_grad.get_jk(dm, kpts) return vj - vk * .5
[docs] def make_rdm1e(mo_energy, mo_coeff, mo_occ): '''Energy weighted density matrix''' nkpts = len(mo_occ) dm1e = [molgrad.make_rdm1e(mo_energy[k], mo_coeff[k], mo_occ[k]) for k in range(nkpts)] return np.asarray(dm1e)
[docs] class GradientsBase(molgrad.GradientsBase): ''' Basic nuclear gradient functions for non-relativistic methods ''' def __init__(self, method): self.cell = method.cell molgrad.GradientsBase.__init__(self, method) @property def kpts(self): return self.base.kpts
[docs] def reset(self, cell=None): if cell is not None: self.cell = cell self.base.reset(cell) return self
[docs] def get_hcore(self, cell=None, kpts=None): if cell is None: cell = self.cell if kpts is None: kpts = self.kpts return get_hcore(cell, kpts)
hcore_generator = hcore_generator
[docs] def get_ovlp(self, cell=None, kpts=None): if cell is None: cell = self.cell if kpts is None: kpts = self.kpts return get_ovlp(cell, kpts)
[docs] def get_jk(self, dm=None, kpts=None): if kpts is None: kpts = self.kpts if dm is None: dm = self.base.make_rdm1() exxdiv = self.base.exxdiv cpu0 = (logger.process_clock(), logger.perf_counter()) vj, vk = self.base.with_df.get_jk_e1(dm, kpts, exxdiv=exxdiv) logger.timer(self, 'vj and vk', *cpu0) return vj, vk
[docs] def get_j(self, dm=None, kpts=None): if kpts is None: kpts = self.kpts if dm is None: dm = self.base.make_rdm1() cpu0 = (logger.process_clock(), logger.perf_counter()) vj = self.base.with_df.get_j_e1(dm, kpts) logger.timer(self, 'vj', *cpu0) return vj
[docs] def get_k(self, dm=None, kpts=None, kpts_band=None): if kpts is None: kpts = self.kpts if dm is None: dm = self.base.make_rdm1() exxdiv = self.base.exxdiv cpu0 = (logger.process_clock(), logger.perf_counter()) vk = self.base.with_df.get_k_e1(dm, kpts, kpts_band, exxdiv) logger.timer(self, 'vk', *cpu0) return vk
[docs] def grad_nuc(self, cell=None, atmlst=None): if cell is None: cell = self.cell return grad_nuc(cell, atmlst)
[docs] def optimizer(self, solver='ase'): '''Geometry optimization solver ''' solver = solver.lower() if solver == 'ase': from pyscf.geomopt import ase_solver return ase_solver.GeometryOptimizer(self.base) else: raise RuntimeError(f'Optimization solver {solver} not supported')
[docs] def as_scanner(mf_grad): '''Generating a nuclear gradients scanner/solver (for geometry optimizer). The returned solver is a function. This function requires one argument "cell" as input and returns energy and first order nuclear derivatives. The solver will automatically use the results of last calculation as the initial guess of the new calculation. All parameters assigned in the nuc-grad object and SCF object (DIIS, conv_tol, max_memory etc) are automatically applied in the solver. Note scanner has side effects. It may change many underlying objects (_scf, with_df, with_x2c, ...) during calculation. ''' if isinstance(mf_grad, lib.GradScanner): return mf_grad logger.info(mf_grad, 'Create scanner for %s', mf_grad.__class__) name = mf_grad.__class__.__name__ + SCF_GradScanner.__name_mixin__ return lib.set_class(SCF_GradScanner(mf_grad), (SCF_GradScanner, mf_grad.__class__), name)
[docs] class SCF_GradScanner(lib.GradScanner): def __init__(self, g): lib.GradScanner.__init__(self, g) def __call__(self, cell_or_geom, **kwargs): if isinstance(cell_or_geom, gto.Cell): cell = cell_or_geom else: cell = self.cell.set_geom_(cell_or_geom, inplace=False) mf_scanner = self.base e_tot = mf_scanner(cell) self.cell = cell # If second integration grids are created for RKS and UKS # gradients if getattr(self, 'grids', None): self.grids.reset(cell) de = self.kernel(**kwargs) return e_tot, de
[docs] class Gradients(GradientsBase): '''Non-relativistic restricted Hartree-Fock gradients'''
[docs] def get_veff(self, dm=None, kpts=None): if kpts is None: kpts = self.kpts if dm is None: dm = self.base.make_rdm1() return get_veff(self, dm, kpts)
[docs] def make_rdm1e(self, mo_energy=None, mo_coeff=None, mo_occ=None): if mo_energy is None: mo_energy = self.base.mo_energy if mo_coeff is None: mo_coeff = self.base.mo_coeff if mo_occ is None: mo_occ = self.base.mo_occ return make_rdm1e(mo_energy, mo_coeff, mo_occ)
[docs] def extra_force(self, atom_id, envs): '''Hook for extra contributions in analytical gradients. Contributions like the response of auxiliary basis in density fitting method, the grid response in DFT numerical integration can be put in this function. ''' #1 force from exxdiv corrections when madelung constant has non-zero derivative #2 DFT grid response return 0
grad_elec = grad_elec as_scanner = as_scanner def _finalize(self): if self.verbose >= logger.NOTE: logger.note(self, '--------------- %s gradients ---------------', self.base.__class__.__name__) self._write(self.cell, self.de, self.atmlst) logger.note(self, '----------------------------------------------')
[docs] def kernel(self, mo_energy=None, mo_coeff=None, mo_occ=None, atmlst=None): cput0 = (logger.process_clock(), logger.perf_counter()) if mo_energy is None: mo_energy = self.base.mo_energy if mo_coeff is None: mo_coeff = self.base.mo_coeff if mo_occ is None: mo_occ = self.base.mo_occ if atmlst is None: atmlst = self.atmlst else: self.atmlst = atmlst if self.verbose >= logger.INFO: self.dump_flags() de = self.grad_elec(mo_energy, mo_coeff, mo_occ, atmlst) self.de = de + self.grad_nuc(atmlst=atmlst) logger.timer(self, 'SCF gradients', *cput0) self._finalize() return self.de
if __name__=='__main__': from pyscf.pbc import scf cell = gto.Cell() cell.atom = ''' He 0.0 0.0 0.0 He 1.0 1.1 1.2 ''' cell.basis = 'gth-dzv' cell.a = np.eye(3) * 3 cell.unit='bohr' cell.pseudo='gth-pade' cell.verbose=4 cell.build() nmp = [1,1,3] kpts = cell.make_kpts(nmp) kmf = scf.KRHF(cell, kpts, exxdiv=None) kmf.kernel() mygrad = Gradients(kmf) grad = mygrad.kernel()