-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathatom_admm_methods.F
469 lines (447 loc) · 25.6 KB
/
atom_admm_methods.F
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
!--------------------------------------------------------------------------------------------------!
! CP2K: A general program to perform molecular dynamics simulations !
! Copyright 2000-2024 CP2K developers group <https://cp2k.org> !
! !
! SPDX-License-Identifier: GPL-2.0-or-later !
!--------------------------------------------------------------------------------------------------!
! **************************************************************************************************
MODULE atom_admm_methods
USE atom_operators, ONLY: atom_basis_projection_overlap,&
atom_int_release,&
atom_int_setup
USE atom_output, ONLY: atom_print_basis
USE atom_types, ONLY: &
atom_basis_type, atom_integrals, atom_orbitals, atom_p_type, atom_type, create_atom_orbs, &
init_atom_basis, lmat, release_atom_basis, release_atom_orbs
USE atom_utils, ONLY: atom_consistent_method,&
atom_denmat,&
atom_trace,&
eeri_contract
USE atom_xc, ONLY: atom_dpot_lda,&
atom_vxc_lda,&
atom_vxc_lsd
USE input_constants, ONLY: do_rhf_atom,&
do_rks_atom,&
do_rohf_atom,&
do_uhf_atom,&
do_uks_atom,&
xc_funct_no_shortcut
USE input_section_types, ONLY: section_vals_duplicate,&
section_vals_get_subs_vals,&
section_vals_get_subs_vals2,&
section_vals_release,&
section_vals_remove_values,&
section_vals_type,&
section_vals_val_set
USE kinds, ONLY: dp
USE mathlib, ONLY: invmat_symm,&
jacobi
#include "./base/base_uses.f90"
IMPLICIT NONE
PRIVATE
PUBLIC :: atom_admm
CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'atom_admm_methods'
CONTAINS
! **************************************************************************************************
!> \brief Analysis of ADMM approximation to exact exchange.
!> \param atom_info information about the atomic kind. Two-dimensional array of size
!> (electronic-configuration, electronic-structure-method)
!> \param admm_section ADMM print section
!> \param iw output file unit
!> \par History
!> * 07.2016 created [Juerg Hutter]
! **************************************************************************************************
SUBROUTINE atom_admm(atom_info, admm_section, iw)
TYPE(atom_p_type), DIMENSION(:, :), POINTER :: atom_info
TYPE(section_vals_type), POINTER :: admm_section
INTEGER, INTENT(IN) :: iw
CHARACTER(LEN=2) :: btyp
INTEGER :: i, ifun, j, l, m, maxl, mo, n, na, nb, &
zval
LOGICAL :: pp_calc, rhf
REAL(KIND=dp) :: admm1_k_energy, admm2_k_energy, admmq_k_energy, dfexc_admm1, dfexc_admm2, &
dfexc_admmq, dxc, dxk, el1, el2, elq, elref, fexc_optx_admm1, fexc_optx_admm2, &
fexc_optx_admmq, fexc_optx_ref, fexc_pbex_admm1, fexc_pbex_admm2, fexc_pbex_admmq, &
fexc_pbex_ref, ref_energy, xsi
REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :) :: lamat
REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :, :) :: admm1_k, admm2_k, admm_xcmat, admm_xcmata, &
admm_xcmatb, admmq_k, ovlap, ref_k, ref_xcmat, ref_xcmata, ref_xcmatb, sinv, siref, tmat
TYPE(atom_basis_type), POINTER :: admm_basis, ref_basis
TYPE(atom_integrals), POINTER :: admm_int, ref_int
TYPE(atom_orbitals), POINTER :: admm1_orbs, admm2_orbs, admmq_orbs, &
ref_orbs
TYPE(atom_type), POINTER :: atom
TYPE(section_vals_type), POINTER :: basis_section, xc_fun, xc_fun_section, &
xc_optx, xc_pbex, xc_section
IF (iw > 0) THEN
WRITE (iw, '(/,T2,A)') &
'!-----------------------------------------------------------------------------!'
WRITE (iw, '(T30,A)') "Analysis of ADMM approximations"
WRITE (iw, '(T2,A)') &
'!-----------------------------------------------------------------------------!'
END IF
! setup an xc section
NULLIFY (xc_section, xc_pbex, xc_optx)
CALL section_vals_duplicate(atom_info(1, 1)%atom%xc_section, xc_section)
xc_fun_section => section_vals_get_subs_vals(xc_section, "XC_FUNCTIONAL")
! Overwrite possible shortcut
CALL section_vals_val_set(xc_fun_section, "_SECTION_PARAMETERS_", &
i_val=xc_funct_no_shortcut)
ifun = 0
DO
ifun = ifun + 1
xc_fun => section_vals_get_subs_vals2(xc_fun_section, i_section=ifun)
IF (.NOT. ASSOCIATED(xc_fun)) EXIT
CALL section_vals_remove_values(xc_fun)
END DO
! PBEX
CALL section_vals_duplicate(xc_section, xc_pbex)
xc_fun_section => section_vals_get_subs_vals(xc_pbex, "XC_FUNCTIONAL")
CALL section_vals_val_set(xc_fun_section, "PBE%_SECTION_PARAMETERS_", l_val=.TRUE.)
CALL section_vals_val_set(xc_fun_section, "PBE%SCALE_X", r_val=1.0_dp)
CALL section_vals_val_set(xc_fun_section, "PBE%SCALE_C", r_val=0.0_dp)
! OPTX
CALL section_vals_duplicate(xc_section, xc_optx)
xc_fun_section => section_vals_get_subs_vals(xc_optx, "XC_FUNCTIONAL")
CALL section_vals_val_set(xc_fun_section, "OPTX%_SECTION_PARAMETERS_", l_val=.TRUE.)
CALL section_vals_val_set(xc_fun_section, "OPTX%SCALE_X", r_val=1.0_dp)
! ADMM basis set
zval = atom_info(1, 1)%atom%z
pp_calc = atom_info(1, 1)%atom%pp_calc
btyp = "AE"
IF (pp_calc) btyp = "PP"
ALLOCATE (admm_basis)
basis_section => section_vals_get_subs_vals(admm_section, "ADMM_BASIS")
NULLIFY (admm_basis%grid)
CALL init_atom_basis(admm_basis, basis_section, zval, btyp)
IF (iw > 0) THEN
CALL atom_print_basis(admm_basis, iw, " ADMM Basis")
END IF
! reference basis set
ref_basis => atom_info(1, 1)%atom%basis
! integrals
ALLOCATE (ref_int, admm_int)
CALL atom_int_setup(ref_int, ref_basis, eri_exchange=.TRUE.)
CALL atom_int_setup(admm_int, admm_basis, eri_exchange=.TRUE.)
DO l = 0, lmat
IF (admm_int%n(l) /= admm_int%nne(l)) THEN
IF (iw > 0) WRITE (iw, *) "ADMM Basis is linear dependent ", l, admm_int%n(l), admm_int%nne(l)
CPABORT("ADMM basis is linear dependent")
END IF
END DO
! mixed overlap
na = MAXVAL(admm_basis%nbas(:))
nb = MAXVAL(ref_basis%nbas(:))
ALLOCATE (ovlap(1:na, 1:nb, 0:lmat))
CALL atom_basis_projection_overlap(ovlap, admm_basis, ref_basis)
! Inverse of ADMM overlap matrix
ALLOCATE (sinv(1:na, 1:na, 0:lmat))
DO l = 0, lmat
n = admm_basis%nbas(l)
IF (n < 1) CYCLE
sinv(1:n, 1:n, l) = admm_int%ovlp(1:n, 1:n, l)
CALL invmat_symm(sinv(1:n, 1:n, l))
END DO
! ADMM transformation matrix
ALLOCATE (tmat(1:na, 1:nb, 0:lmat))
DO l = 0, lmat
n = admm_basis%nbas(l)
m = ref_basis%nbas(l)
IF (n < 1 .OR. m < 1) CYCLE
tmat(1:n, 1:m, l) = MATMUL(sinv(1:n, 1:n, l), ovlap(1:n, 1:m, l))
END DO
! Inverse of REF overlap matrix
ALLOCATE (siref(1:nb, 1:nb, 0:lmat))
DO l = 0, lmat
n = ref_basis%nbas(l)
IF (n < 1) CYCLE
siref(1:n, 1:n, l) = ref_int%ovlp(1:n, 1:n, l)
CALL invmat_symm(siref(1:n, 1:n, l))
END DO
DO i = 1, SIZE(atom_info, 1)
DO j = 1, SIZE(atom_info, 2)
atom => atom_info(i, j)%atom
IF (atom_consistent_method(atom%method_type, atom%state%multiplicity)) THEN
ref_orbs => atom%orbitals
ALLOCATE (ref_k(1:nb, 1:nb, 0:lmat))
SELECT CASE (atom%method_type)
CASE (do_rks_atom, do_rhf_atom)
! restricted
rhf = .TRUE.
ref_k = 0.0_dp
CALL eeri_contract(ref_k, ref_int%eeri, ref_orbs%pmat, ref_basis%nbas)
ref_energy = 0.5_dp*atom_trace(ref_k, ref_orbs%pmat)
CASE (do_uks_atom, do_uhf_atom)
! unrestricted
rhf = .FALSE.
ref_k = 0.0_dp
CALL eeri_contract(ref_k, ref_int%eeri, ref_orbs%pmata, ref_basis%nbas)
ref_energy = atom_trace(ref_k, ref_orbs%pmata)
ref_k = 0.0_dp
CALL eeri_contract(ref_k, ref_int%eeri, ref_orbs%pmatb, ref_basis%nbas)
ref_energy = ref_energy + atom_trace(ref_k, ref_orbs%pmatb)
CASE (do_rohf_atom)
CPABORT("ADMM not available")
CASE DEFAULT
CPABORT("ADMM not available")
END SELECT
DEALLOCATE (ref_k)
! reference number of electrons
elref = atom_trace(ref_int%ovlp, ref_orbs%pmat)
! admm orbitals and density matrices
mo = MAXVAL(atom%state%maxn_calc)
NULLIFY (admm1_orbs, admm2_orbs, admmq_orbs)
CALL create_atom_orbs(admm1_orbs, na, mo)
CALL create_atom_orbs(admm2_orbs, na, mo)
CALL create_atom_orbs(admmq_orbs, na, mo)
ALLOCATE (lamat(1:mo, 1:mo))
ALLOCATE (admm1_k(1:na, 1:na, 0:lmat))
ALLOCATE (admm2_k(1:na, 1:na, 0:lmat))
ALLOCATE (admmq_k(1:na, 1:na, 0:lmat))
IF (rhf) THEN
DO l = 0, lmat
n = admm_basis%nbas(l)
m = ref_basis%nbas(l)
mo = atom%state%maxn_calc(l)
IF (n < 1 .OR. m < 1 .OR. mo < 1) CYCLE
admm2_orbs%wfn(1:n, 1:mo, l) = MATMUL(tmat(1:n, 1:m, l), ref_orbs%wfn(1:m, 1:mo, l))
CALL lowdin_matrix(admm2_orbs%wfn(1:n, 1:mo, l), lamat(1:mo, 1:mo), admm_int%ovlp(1:n, 1:n, l))
admm1_orbs%wfn(1:n, 1:mo, l) = MATMUL(admm2_orbs%wfn(1:n, 1:mo, l), lamat(1:mo, 1:mo))
END DO
CALL atom_denmat(admm1_orbs%pmat, admm1_orbs%wfn, admm_basis%nbas, atom%state%occupation, &
atom%state%maxl_occ, atom%state%maxn_occ)
CALL atom_denmat(admm2_orbs%pmat, admm2_orbs%wfn, admm_basis%nbas, atom%state%occupation, &
atom%state%maxl_occ, atom%state%maxn_occ)
el1 = atom_trace(admm_int%ovlp, admm1_orbs%pmat)
el2 = atom_trace(admm_int%ovlp, admm2_orbs%pmat)
xsi = elref/el2
admmq_orbs%pmat = xsi*admm2_orbs%pmat
elq = atom_trace(admm_int%ovlp, admmq_orbs%pmat)
admmq_orbs%wfn = SQRT(xsi)*admm2_orbs%wfn
!
admm1_k = 0.0_dp
CALL eeri_contract(admm1_k, admm_int%eeri, admm1_orbs%pmat, admm_basis%nbas)
admm1_k_energy = 0.5_dp*atom_trace(admm1_k, admm1_orbs%pmat)
admm2_k = 0.0_dp
CALL eeri_contract(admm2_k, admm_int%eeri, admm2_orbs%pmat, admm_basis%nbas)
admm2_k_energy = 0.5_dp*atom_trace(admm2_k, admm2_orbs%pmat)
admmq_k = 0.0_dp
CALL eeri_contract(admmq_k, admm_int%eeri, admmq_orbs%pmat, admm_basis%nbas)
admmq_k_energy = 0.5_dp*atom_trace(admmq_k, admmq_orbs%pmat)
ELSE
DO l = 0, lmat
n = admm_basis%nbas(l)
m = ref_basis%nbas(l)
mo = atom%state%maxn_calc(l)
IF (n < 1 .OR. m < 1 .OR. mo < 1) CYCLE
admm2_orbs%wfna(1:n, 1:mo, l) = MATMUL(tmat(1:n, 1:m, l), ref_orbs%wfna(1:m, 1:mo, l))
CALL lowdin_matrix(admm2_orbs%wfna(1:n, 1:mo, l), lamat, admm_int%ovlp(1:n, 1:n, l))
admm1_orbs%wfna(1:n, 1:mo, l) = MATMUL(admm2_orbs%wfna(1:n, 1:mo, l), lamat(1:mo, 1:mo))
admm2_orbs%wfnb(1:n, 1:mo, l) = MATMUL(tmat(1:n, 1:m, l), ref_orbs%wfnb(1:m, 1:mo, l))
CALL lowdin_matrix(admm2_orbs%wfnb(1:n, 1:mo, l), lamat, admm_int%ovlp(1:n, 1:n, l))
admm1_orbs%wfnb(1:n, 1:mo, l) = MATMUL(admm2_orbs%wfnb(1:n, 1:mo, l), lamat(1:mo, 1:mo))
END DO
CALL atom_denmat(admm1_orbs%pmata, admm1_orbs%wfna, admm_basis%nbas, atom%state%occa, &
atom%state%maxl_occ, atom%state%maxn_occ)
CALL atom_denmat(admm1_orbs%pmatb, admm1_orbs%wfnb, admm_basis%nbas, atom%state%occb, &
atom%state%maxl_occ, atom%state%maxn_occ)
admm1_orbs%pmat = admm1_orbs%pmata + admm1_orbs%pmatb
CALL atom_denmat(admm2_orbs%pmata, admm2_orbs%wfna, admm_basis%nbas, atom%state%occa, &
atom%state%maxl_occ, atom%state%maxn_occ)
CALL atom_denmat(admm2_orbs%pmatb, admm2_orbs%wfnb, admm_basis%nbas, atom%state%occb, &
atom%state%maxl_occ, atom%state%maxn_occ)
admm2_orbs%pmat = admm2_orbs%pmata + admm2_orbs%pmatb
elref = atom_trace(ref_int%ovlp, ref_orbs%pmata)
el2 = atom_trace(admm_int%ovlp, admm2_orbs%pmata)
xsi = elref/el2
admmq_orbs%pmata = xsi*admm2_orbs%pmata
admmq_orbs%wfna = SQRT(xsi)*admm2_orbs%wfna
elref = atom_trace(ref_int%ovlp, ref_orbs%pmatb)
el2 = atom_trace(admm_int%ovlp, admm2_orbs%pmatb)
xsi = elref/el2
admmq_orbs%pmatb = xsi*admm2_orbs%pmatb
admmq_orbs%wfnb = SQRT(xsi)*admm2_orbs%wfnb
admmq_orbs%pmat = admmq_orbs%pmata + admmq_orbs%pmatb
el1 = atom_trace(admm_int%ovlp, admm1_orbs%pmat)
el2 = atom_trace(admm_int%ovlp, admm2_orbs%pmat)
elq = atom_trace(admm_int%ovlp, admmq_orbs%pmat)
elref = atom_trace(ref_int%ovlp, ref_orbs%pmat)
!
admm1_k = 0.0_dp
CALL eeri_contract(admm1_k, admm_int%eeri, admm1_orbs%pmata, admm_basis%nbas)
admm1_k_energy = atom_trace(admm1_k, admm1_orbs%pmata)
admm1_k = 0.0_dp
CALL eeri_contract(admm1_k, admm_int%eeri, admm1_orbs%pmatb, admm_basis%nbas)
admm1_k_energy = admm1_k_energy + atom_trace(admm1_k, admm1_orbs%pmatb)
admm2_k = 0.0_dp
CALL eeri_contract(admm2_k, admm_int%eeri, admm2_orbs%pmata, admm_basis%nbas)
admm2_k_energy = atom_trace(admm2_k, admm2_orbs%pmata)
admm2_k = 0.0_dp
CALL eeri_contract(admm2_k, admm_int%eeri, admm2_orbs%pmatb, admm_basis%nbas)
admm2_k_energy = admm2_k_energy + atom_trace(admm2_k, admm2_orbs%pmatb)
admmq_k = 0.0_dp
CALL eeri_contract(admmq_k, admm_int%eeri, admmq_orbs%pmata, admm_basis%nbas)
admmq_k_energy = atom_trace(admmq_k, admmq_orbs%pmata)
admmq_k = 0.0_dp
CALL eeri_contract(admmq_k, admm_int%eeri, admmq_orbs%pmatb, admm_basis%nbas)
admmq_k_energy = admmq_k_energy + atom_trace(admmq_k, admmq_orbs%pmatb)
END IF
DEALLOCATE (lamat)
!
! ADMM correction terms
maxl = atom%state%maxl_occ
IF (rhf) THEN
ALLOCATE (ref_xcmat(1:nb, 1:nb, 0:lmat))
ALLOCATE (admm_xcmat(1:na, 1:na, 0:lmat))
! PBEX
CALL atom_vxc_lda(ref_basis, ref_orbs%pmat, maxl, xc_pbex, fexc_pbex_ref, ref_xcmat)
CALL atom_vxc_lda(admm_basis, admm1_orbs%pmat, maxl, xc_pbex, fexc_pbex_admm1, admm_xcmat)
CALL atom_vxc_lda(admm_basis, admm2_orbs%pmat, maxl, xc_pbex, fexc_pbex_admm2, admm_xcmat)
CALL atom_vxc_lda(admm_basis, admmq_orbs%pmat, maxl, xc_pbex, fexc_pbex_admmq, admm_xcmat)
! OPTX
CALL atom_vxc_lda(ref_basis, ref_orbs%pmat, maxl, xc_optx, fexc_optx_ref, ref_xcmat)
CALL atom_vxc_lda(admm_basis, admm1_orbs%pmat, maxl, xc_optx, fexc_optx_admm1, admm_xcmat)
CALL atom_vxc_lda(admm_basis, admm2_orbs%pmat, maxl, xc_optx, fexc_optx_admm2, admm_xcmat)
CALL atom_vxc_lda(admm_basis, admmq_orbs%pmat, maxl, xc_optx, fexc_optx_admmq, admm_xcmat)
! LINX
CALL atom_dpot_lda(ref_basis, ref_orbs%pmat, admm_basis, admm1_orbs%pmat, &
maxl, "LINX", dfexc_admm1)
CALL atom_dpot_lda(ref_basis, ref_orbs%pmat, admm_basis, admm2_orbs%pmat, &
maxl, "LINX", dfexc_admm2)
CALL atom_dpot_lda(ref_basis, ref_orbs%pmat, admm_basis, admmq_orbs%pmat, &
maxl, "LINX", dfexc_admmq)
DEALLOCATE (ref_xcmat, admm_xcmat)
ELSE
ALLOCATE (ref_xcmata(1:nb, 1:nb, 0:lmat))
ALLOCATE (ref_xcmatb(1:nb, 1:nb, 0:lmat))
ALLOCATE (admm_xcmata(1:na, 1:na, 0:lmat))
ALLOCATE (admm_xcmatb(1:na, 1:na, 0:lmat))
! PBEX
CALL atom_vxc_lsd(ref_basis, ref_orbs%pmata, ref_orbs%pmatb, maxl, xc_pbex, fexc_pbex_ref, &
ref_xcmata, ref_xcmatb)
CALL atom_vxc_lsd(admm_basis, admm1_orbs%pmata, admm1_orbs%pmatb, maxl, xc_pbex, &
fexc_pbex_admm1, admm_xcmata, admm_xcmatb)
CALL atom_vxc_lsd(admm_basis, admm2_orbs%pmata, admm2_orbs%pmatb, maxl, xc_pbex, &
fexc_pbex_admm2, admm_xcmata, admm_xcmatb)
CALL atom_vxc_lsd(admm_basis, admmq_orbs%pmata, admmq_orbs%pmatb, maxl, xc_pbex, &
fexc_pbex_admmq, admm_xcmata, admm_xcmatb)
CALL atom_vxc_lsd(ref_basis, ref_orbs%pmata, ref_orbs%pmatb, maxl, xc_optx, fexc_optx_ref, &
ref_xcmata, ref_xcmatb)
CALL atom_vxc_lsd(admm_basis, admm1_orbs%pmata, admm1_orbs%pmatb, maxl, xc_optx, &
fexc_optx_admm1, admm_xcmata, admm_xcmatb)
CALL atom_vxc_lsd(admm_basis, admm2_orbs%pmata, admm2_orbs%pmatb, maxl, xc_optx, &
fexc_optx_admm2, admm_xcmata, admm_xcmatb)
CALL atom_vxc_lsd(admm_basis, admmq_orbs%pmata, admmq_orbs%pmatb, maxl, xc_optx, &
fexc_optx_admmq, admm_xcmata, admm_xcmatb)
DEALLOCATE (ref_xcmata, ref_xcmatb, admm_xcmata, admm_xcmatb)
END IF
!
IF (iw > 0) THEN
WRITE (iw, "(/,A,I3,T48,A,I3)") " Electronic Structure Setting: ", i, &
" Electronic Structure Method: ", j
WRITE (iw, "(' Norm of ADMM Basis projection ',T61,F20.10)") el2/elref
WRITE (iw, "(' Reference Exchange Energy [Hartree]',T61,F20.10)") ref_energy
! ADMM1
dxk = ref_energy - admm1_k_energy
WRITE (iw, "(A,F20.10,T60,A,F13.10)") " ADMM1 METHOD: Energy ", admm1_k_energy, &
" Error: ", dxk
dxc = fexc_pbex_ref - fexc_pbex_admm1
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "PBEX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
dxc = fexc_optx_ref - fexc_optx_admm1
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "OPTX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
dxc = dfexc_admm1
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "LINX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
! ADMM2
dxk = ref_energy - admm2_k_energy
WRITE (iw, "(A,F20.10,T60,A,F13.10)") " ADMM2 METHOD: Energy ", admm2_k_energy, &
" Error: ", dxk
dxc = fexc_pbex_ref - fexc_pbex_admm2
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "PBEX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
dxc = fexc_optx_ref - fexc_optx_admm2
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "OPTX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
dxc = dfexc_admm2
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "LINX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
! ADMMQ
dxk = ref_energy - admmq_k_energy
WRITE (iw, "(A,F20.10,T60,A,F13.10)") " ADMMQ METHOD: Energy ", admmq_k_energy, &
" Error: ", dxk
dxc = fexc_pbex_ref - fexc_pbex_admmq
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "PBEX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
dxc = fexc_optx_ref - fexc_optx_admmq
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "OPTX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
dxc = dfexc_admmq
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "LINX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
! ADMMS
dxk = ref_energy - admmq_k_energy
WRITE (iw, "(A,F20.10,T60,A,F13.10)") " ADMMS METHOD: Energy ", admmq_k_energy, &
" Error: ", dxk
dxc = fexc_pbex_ref - fexc_pbex_admmq*xsi**(2._dp/3._dp)
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "PBEX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
dxc = fexc_optx_ref - fexc_optx_admmq*xsi**(2._dp/3._dp)
WRITE (iw, "(T10,A,F12.6,F12.3,'%',T60,A,F13.10)") "OPTX Correction ", dxc, dxc/dxk*100._dp, &
" Error: ", dxk - dxc
END IF
!
DEALLOCATE (admm1_k, admm2_k, admmq_k)
!
CALL release_atom_orbs(admm1_orbs)
CALL release_atom_orbs(admm2_orbs)
CALL release_atom_orbs(admmq_orbs)
END IF
END DO
END DO
! clean up
CALL atom_int_release(ref_int)
CALL atom_int_release(admm_int)
CALL release_atom_basis(admm_basis)
DEALLOCATE (ref_int, admm_int, admm_basis)
DEALLOCATE (ovlap, sinv, tmat, siref)
CALL section_vals_release(xc_pbex)
CALL section_vals_release(xc_optx)
CALL section_vals_release(xc_section)
IF (iw > 0) THEN
WRITE (iw, '(/,T2,A)') &
'!------------------------------End of ADMM analysis---------------------------!'
END IF
END SUBROUTINE atom_admm
! **************************************************************************************************
!> \brief ...
!> \param wfn ...
!> \param lamat ...
!> \param ovlp ...
! **************************************************************************************************
SUBROUTINE lowdin_matrix(wfn, lamat, ovlp)
REAL(KIND=dp), DIMENSION(:, :), INTENT(IN) :: wfn
REAL(KIND=dp), DIMENSION(:, :), INTENT(OUT) :: lamat
REAL(KIND=dp), DIMENSION(:, :), INTENT(IN) :: ovlp
INTEGER :: i, j, k, n
REAL(KIND=dp), ALLOCATABLE, DIMENSION(:) :: w
REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :) :: vmat
n = SIZE(wfn, 2)
IF (n > 0) THEN
lamat = MATMUL(TRANSPOSE(wfn), MATMUL(ovlp, wfn))
ALLOCATE (w(n), vmat(n, n))
CALL jacobi(lamat(1:n, 1:n), w(1:n), vmat(1:n, 1:n))
w(1:n) = 1.0_dp/SQRT(w(1:n))
DO i = 1, n
DO j = 1, n
lamat(i, j) = 0.0_dp
DO k = 1, n
lamat(i, j) = lamat(i, j) + vmat(i, k)*w(k)*vmat(j, k)
END DO
END DO
END DO
DEALLOCATE (vmat, w)
END IF
END SUBROUTINE lowdin_matrix
END MODULE atom_admm_methods