Line data Source code
1 : !--------------------------------------------------------------------------------------------------!
2 : ! CP2K: A general program to perform molecular dynamics simulations !
3 : ! Copyright 2000-2026 CP2K developers group <https://cp2k.org> !
4 : ! !
5 : ! SPDX-License-Identifier: GPL-2.0-or-later !
6 : !--------------------------------------------------------------------------------------------------!
7 :
8 : ! **************************************************************************************************
9 : !> \par History
10 : !> none
11 : !> \author MI (08.01.2004)
12 : ! **************************************************************************************************
13 : MODULE soft_basis_set
14 :
15 : USE ao_util, ONLY: exp_radius
16 : USE basis_set_types, ONLY: copy_gto_basis_set,&
17 : deallocate_gto_basis_set,&
18 : get_gto_basis_set,&
19 : gto_basis_set_type,&
20 : init_cphi_and_sphi
21 : USE kinds, ONLY: default_string_length,&
22 : dp
23 : USE memory_utilities, ONLY: reallocate
24 : USE orbital_pointers, ONLY: indco,&
25 : nco,&
26 : ncoset,&
27 : nso
28 : USE orbital_symbols, ONLY: cgf_symbol,&
29 : sgf_symbol
30 : #include "../base/base_uses.f90"
31 :
32 : IMPLICIT NONE
33 :
34 : PRIVATE
35 :
36 : ! *** Global parameters (only in this module)
37 :
38 : CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'soft_basis_set'
39 :
40 : INTEGER, PARAMETER :: max_name_length = 60
41 :
42 : ! *** Public subroutines ***
43 :
44 : PUBLIC :: create_soft_basis
45 :
46 : CONTAINS
47 :
48 : ! **************************************************************************************************
49 : !> \brief create the soft basis from a GTO basis
50 : !> \param orb_basis ...
51 : !> \param soft_basis ...
52 : !> \param eps_fit ...
53 : !> \param rc ...
54 : !> \param paw_atom ...
55 : !> \param paw_type_forced ...
56 : !> \param gpw_r3d_rs_type_forced ...
57 : !> \version 1.0
58 : ! **************************************************************************************************
59 2556 : SUBROUTINE create_soft_basis(orb_basis, soft_basis, eps_fit, rc, paw_atom, &
60 : paw_type_forced, gpw_r3d_rs_type_forced)
61 :
62 : TYPE(gto_basis_set_type), POINTER :: orb_basis, soft_basis
63 : REAL(dp), INTENT(IN) :: eps_fit, rc
64 : LOGICAL, INTENT(OUT) :: paw_atom
65 : LOGICAL, INTENT(IN) :: paw_type_forced, gpw_r3d_rs_type_forced
66 :
67 : CHARACTER(LEN=default_string_length) :: bsname
68 : INTEGER :: ico, ipgf, ipgf_s, iset, iset_s, ishell, lshell, lshell_old, m, maxco, maxpgf, &
69 : maxpgf_s, maxshell, maxshell_s, ncgf, nset, nset_s, nsgf
70 2556 : INTEGER, ALLOCATABLE, DIMENSION(:) :: iset_s2h
71 2556 : INTEGER, DIMENSION(:), POINTER :: lmax, lmin, npgf, nshell
72 2556 : INTEGER, DIMENSION(:, :), POINTER :: l, n
73 : LOGICAL :: my_gpw_r3d_rs_type_forced
74 : REAL(KIND=dp) :: minzet, radius
75 2556 : REAL(KIND=dp), DIMENSION(:, :), POINTER :: zet
76 2556 : REAL(KIND=dp), DIMENSION(:, :, :), POINTER :: gcc
77 :
78 2556 : NULLIFY (gcc, l, lmax, lmin, n, npgf, nshell, zet)
79 2556 : paw_atom = .FALSE.
80 2556 : my_gpw_r3d_rs_type_forced = gpw_r3d_rs_type_forced
81 2556 : IF (paw_type_forced) THEN
82 72 : paw_atom = .TRUE.
83 : my_gpw_r3d_rs_type_forced = .FALSE.
84 : END IF
85 2484 : IF (.NOT. my_gpw_r3d_rs_type_forced) THEN
86 : CALL get_gto_basis_set(gto_basis_set=orb_basis, name=bsname, &
87 2528 : maxpgf=maxpgf, maxshell=maxshell, nset=nset)
88 :
89 2528 : soft_basis%name = TRIM(bsname)//"_soft"
90 :
91 2528 : CALL reallocate(npgf, 1, nset)
92 2528 : CALL reallocate(nshell, 1, nset)
93 2528 : CALL reallocate(lmax, 1, nset)
94 2528 : CALL reallocate(lmin, 1, nset)
95 :
96 2528 : CALL reallocate(n, 1, maxshell, 1, nset)
97 2528 : CALL reallocate(l, 1, maxshell, 1, nset)
98 :
99 2528 : CALL reallocate(zet, 1, maxpgf, 1, nset)
100 2528 : CALL reallocate(gcc, 1, maxpgf, 1, maxshell, 1, nset)
101 :
102 7584 : ALLOCATE (iset_s2h(nset))
103 :
104 9722 : iset_s2h = 0
105 2528 : iset_s = 0
106 2528 : maxpgf_s = 0
107 2528 : maxshell_s = 0
108 :
109 9722 : DO iset = 1, nset ! iset
110 7194 : minzet = orb_basis%zet(orb_basis%npgf(iset), iset)
111 17296 : DO ipgf = orb_basis%npgf(iset) - 1, 1, -1
112 17296 : IF (orb_basis%zet(ipgf, iset) < minzet) THEN
113 72 : minzet = orb_basis%zet(ipgf, iset)
114 : END IF
115 : END DO
116 7194 : radius = exp_radius(orb_basis%lmax(iset), minzet, eps_fit, 1.0_dp)
117 :
118 : ! The soft basis contains this set
119 7194 : iset_s = iset_s + 1
120 7194 : nshell(iset_s) = orb_basis%nshell(iset)
121 7194 : lmax(iset_s) = orb_basis%lmax(iset)
122 7194 : lmin(iset_s) = orb_basis%lmin(iset)
123 :
124 7194 : iset_s2h(iset_s) = iset
125 :
126 18898 : DO ishell = 1, nshell(iset_s)
127 11704 : n(ishell, iset_s) = orb_basis%n(ishell, iset)
128 18898 : l(ishell, iset_s) = orb_basis%l(ishell, iset)
129 : END DO
130 :
131 7194 : IF (nshell(iset_s) > maxshell_s) THEN
132 3010 : maxshell_s = nshell(iset_s)
133 : END IF
134 :
135 7194 : IF (radius < rc) THEN
136 : ! The soft basis does not contain this set
137 : ! For the moment I keep the set as a dummy set
138 : ! with no exponents, in order to have the right number of contractions
139 : ! In a second time it can be taken away, by creating a pointer
140 : ! which connects the remaining sets to the right contraction index
141 620 : paw_atom = .TRUE.
142 620 : npgf(iset_s) = 0
143 620 : CYCLE
144 : END IF
145 :
146 6574 : ipgf_s = 0
147 21470 : DO ipgf = 1, orb_basis%npgf(iset) ! ipgf
148 14896 : IF (orb_basis%zet(ipgf, iset) > 100.0_dp) THEN
149 : ! The soft basis does not contain this exponent
150 836 : paw_atom = .TRUE.
151 836 : CYCLE
152 : END IF
153 :
154 : radius = exp_radius(orb_basis%lmax(iset), orb_basis%zet(ipgf, iset), &
155 14060 : eps_fit, 1.0_dp)
156 14060 : IF (radius < rc) THEN
157 : ! The soft basis does not contain this exponent
158 3158 : paw_atom = .TRUE.
159 3158 : CYCLE
160 : END IF
161 :
162 : ! The soft basis contains this exponent
163 10902 : ipgf_s = ipgf_s + 1
164 10902 : zet(ipgf_s, iset_s) = orb_basis%zet(ipgf, iset)
165 :
166 10902 : lshell_old = orb_basis%l(1, iset)
167 10902 : radius = exp_radius(lshell_old, zet(ipgf_s, iset_s), eps_fit, 1.0_dp)
168 :
169 41488 : DO ishell = 1, nshell(iset_s)
170 24012 : lshell = orb_basis%l(ishell, iset)
171 24012 : IF (lshell == lshell_old) THEN
172 : ELSE
173 4554 : lshell_old = lshell
174 4554 : radius = exp_radius(lshell_old, zet(ipgf_s, iset_s), eps_fit, 1.0_dp)
175 : END IF
176 34914 : IF (radius < rc) THEN
177 4 : gcc(ipgf_s, ishell, iset_s) = 0.0_dp
178 4 : paw_atom = .TRUE.
179 : ELSE
180 24008 : gcc(ipgf_s, ishell, iset_s) = orb_basis%gcc(ipgf, ishell, iset)
181 : END IF
182 : END DO
183 : END DO
184 6574 : npgf(iset_s) = ipgf_s
185 9102 : IF (ipgf_s > maxpgf_s) THEN
186 2850 : maxpgf_s = ipgf_s
187 : END IF
188 : END DO
189 2528 : nset_s = iset_s
190 2528 : IF (paw_atom) THEN
191 2196 : soft_basis%nset = nset_s
192 2196 : CALL reallocate(soft_basis%lmax, 1, nset_s)
193 2196 : CALL reallocate(soft_basis%lmin, 1, nset_s)
194 2196 : CALL reallocate(soft_basis%npgf, 1, nset_s)
195 2196 : CALL reallocate(soft_basis%nshell, 1, nset_s)
196 2196 : CALL reallocate(soft_basis%n, 1, maxshell_s, 1, nset_s)
197 2196 : CALL reallocate(soft_basis%l, 1, maxshell_s, 1, nset_s)
198 2196 : CALL reallocate(soft_basis%zet, 1, maxpgf_s, 1, nset_s)
199 2196 : CALL reallocate(soft_basis%gcc, 1, maxpgf_s, 1, maxshell_s, 1, nset_s)
200 :
201 : ! *** Copy the basis set information into the data structure ***
202 :
203 8804 : DO iset = 1, nset_s
204 6608 : soft_basis%lmax(iset) = lmax(iset)
205 6608 : soft_basis%lmin(iset) = lmin(iset)
206 6608 : soft_basis%npgf(iset) = npgf(iset)
207 6608 : soft_basis%nshell(iset) = nshell(iset)
208 17252 : DO ishell = 1, nshell(iset)
209 10644 : soft_basis%n(ishell, iset) = n(ishell, iset)
210 10644 : soft_basis%l(ishell, iset) = l(ishell, iset)
211 38404 : DO ipgf = 1, npgf(iset)
212 31796 : soft_basis%gcc(ipgf, ishell, iset) = gcc(ipgf, ishell, iset)
213 : END DO
214 : END DO
215 18556 : DO ipgf = 1, npgf(iset)
216 16360 : soft_basis%zet(ipgf, iset) = zet(ipgf, iset)
217 : END DO
218 : END DO
219 :
220 : ! *** Initialise the depending soft_basis pointers ***
221 2196 : CALL reallocate(soft_basis%set_radius, 1, nset_s)
222 2196 : CALL reallocate(soft_basis%pgf_radius, 1, maxpgf_s, 1, nset_s)
223 2196 : CALL reallocate(soft_basis%first_cgf, 1, maxshell_s, 1, nset_s)
224 2196 : CALL reallocate(soft_basis%first_sgf, 1, maxshell_s, 1, nset_s)
225 2196 : CALL reallocate(soft_basis%last_cgf, 1, maxshell_s, 1, nset_s)
226 2196 : CALL reallocate(soft_basis%last_sgf, 1, maxshell_s, 1, nset_s)
227 2196 : CALL reallocate(soft_basis%ncgf_set, 1, nset_s)
228 2196 : CALL reallocate(soft_basis%nsgf_set, 1, nset_s)
229 :
230 2196 : maxco = 0
231 2196 : ncgf = 0
232 2196 : nsgf = 0
233 :
234 8804 : DO iset = 1, nset_s
235 6608 : soft_basis%ncgf_set(iset) = 0
236 6608 : soft_basis%nsgf_set(iset) = 0
237 17252 : DO ishell = 1, nshell(iset)
238 10644 : lshell = soft_basis%l(ishell, iset)
239 10644 : soft_basis%first_cgf(ishell, iset) = ncgf + 1
240 10644 : ncgf = ncgf + nco(lshell)
241 10644 : soft_basis%last_cgf(ishell, iset) = ncgf
242 : soft_basis%ncgf_set(iset) = &
243 10644 : soft_basis%ncgf_set(iset) + nco(lshell)
244 10644 : soft_basis%first_sgf(ishell, iset) = nsgf + 1
245 10644 : nsgf = nsgf + nso(lshell)
246 10644 : soft_basis%last_sgf(ishell, iset) = nsgf
247 : soft_basis%nsgf_set(iset) = &
248 17252 : soft_basis%nsgf_set(iset) + nso(lshell)
249 : END DO
250 8804 : maxco = MAX(maxco, npgf(iset)*ncoset(lmax(iset)))
251 : END DO
252 2196 : soft_basis%ncgf = ncgf
253 2196 : soft_basis%nsgf = nsgf
254 :
255 2196 : CALL reallocate(soft_basis%cphi, 1, maxco, 1, ncgf)
256 2196 : CALL reallocate(soft_basis%sphi, 1, maxco, 1, nsgf)
257 2196 : CALL reallocate(soft_basis%scon, 1, maxco, 1, nsgf)
258 2196 : CALL reallocate(soft_basis%lx, 1, ncgf)
259 2196 : CALL reallocate(soft_basis%ly, 1, ncgf)
260 2196 : CALL reallocate(soft_basis%lz, 1, ncgf)
261 2196 : CALL reallocate(soft_basis%m, 1, nsgf)
262 2196 : CALL reallocate(soft_basis%norm_cgf, 1, ncgf)
263 6588 : ALLOCATE (soft_basis%cgf_symbol(ncgf))
264 6588 : ALLOCATE (soft_basis%sgf_symbol(nsgf))
265 :
266 2196 : ncgf = 0
267 2196 : nsgf = 0
268 :
269 8804 : DO iset = 1, nset_s
270 19448 : DO ishell = 1, nshell(iset)
271 10644 : lshell = soft_basis%l(ishell, iset)
272 35872 : DO ico = ncoset(lshell - 1) + 1, ncoset(lshell)
273 25228 : ncgf = ncgf + 1
274 25228 : soft_basis%lx(ncgf) = indco(1, ico)
275 25228 : soft_basis%ly(ncgf) = indco(2, ico)
276 25228 : soft_basis%lz(ncgf) = indco(3, ico)
277 : soft_basis%cgf_symbol(ncgf) = &
278 : cgf_symbol(n(ishell, iset), [soft_basis%lx(ncgf), &
279 : soft_basis%ly(ncgf), &
280 111556 : soft_basis%lz(ncgf)])
281 : END DO
282 40948 : DO m = -lshell, lshell
283 23696 : nsgf = nsgf + 1
284 23696 : soft_basis%m(nsgf) = m
285 : soft_basis%sgf_symbol(nsgf) = &
286 34340 : sgf_symbol(n(ishell, iset), lshell, m)
287 : END DO
288 : END DO
289 : END DO
290 :
291 : ! *** Normalization factor of the contracted Gaussians ***
292 2196 : soft_basis%norm_type = orb_basis%norm_type
293 52652 : soft_basis%norm_cgf = orb_basis%norm_cgf
294 : ! *** Initialize the transformation matrices ***
295 2196 : CALL init_cphi_and_sphi(soft_basis)
296 : END IF
297 :
298 5056 : DEALLOCATE (gcc, l, lmax, lmin, n, npgf, nshell, zet, iset_s2h)
299 : END IF
300 :
301 2556 : IF (.NOT. paw_atom) THEN
302 360 : CALL deallocate_gto_basis_set(soft_basis)
303 360 : CALL copy_gto_basis_set(orb_basis, soft_basis)
304 360 : CALL get_gto_basis_set(gto_basis_set=orb_basis, name=bsname)
305 360 : soft_basis%name = TRIM(bsname)//"_soft"
306 : END IF
307 :
308 2556 : END SUBROUTINE create_soft_basis
309 :
310 : END MODULE soft_basis_set
|