32#include "./base/base_uses.f90"
37 CHARACTER(len=*),
PARAMETER,
PRIVATE :: moduleN =
'negf_control_types'
38 LOGICAL,
PARAMETER,
PRIVATE :: debug_this_module = .true.
49 INTEGER,
ALLOCATABLE,
DIMENSION(:) :: atomlist_bulk, atomlist_screening
52 DIMENSION(:) :: atomlist_cell
54 INTEGER :: force_env_index = -1
56 LOGICAL :: compute_fermi_level = .false.
58 LOGICAL :: refine_fermi_level = .false.
60 LOGICAL :: shift_fermi_level = .false.
62 LOGICAL :: read_write_hs = .false.
64 LOGICAL :: is_restart = .false.
66 REAL(kind=
dp) :: fermi_level = -1.0_dp
68 REAL(kind=
dp) :: fermi_level_shifted = -1.0_dp
70 REAL(kind=
dp) :: temperature = -1.0_dp
72 REAL(kind=
dp) :: v_external = 0.0_dp
82 DIMENSION(:) :: contacts
84 INTEGER,
ALLOCATABLE,
DIMENSION(:) :: atomlist_s
87 INTEGER,
ALLOCATABLE,
DIMENSION(:) :: atomlist_s_screening
89 LOGICAL :: read_write_hs = .false.
91 LOGICAL :: update_hs = .false.
93 LOGICAL :: is_dft_entire = .false.
95 LOGICAL :: is_restart = .false.
97 LOGICAL :: write_common_restart_file = .false.
99 LOGICAL :: disable_cache = .false.
101 REAL(kind=
dp) :: conv_density = -1.0_dp
103 REAL(kind=
dp) :: conv_green = -1.0_dp
105 REAL(kind=
dp) :: conv_scf = -1.0_dp
107 REAL(kind=
dp) :: eps_geometry = -1.0_dp
109 REAL(kind=
dp) :: v_bias = -1.0_dp
111 REAL(kind=
dp) :: energy_lbound = -1.0_dp
113 REAL(kind=
dp) :: eta = -1.0_dp
115 REAL(kind=
dp) :: homo_lumo_gap = -1.0_dp
117 INTEGER :: delta_npoles = -1
119 INTEGER :: gamma_kt = -1
121 INTEGER :: integr_method = -1
123 INTEGER :: integr_min_points = -1
125 INTEGER :: integr_max_points = -1
127 INTEGER :: max_scf = -1
129 INTEGER :: nprocs = -1
131 REAL(kind=
dp) :: v_shift = -1.0_dp
133 REAL(kind=
dp) :: v_shift_offset = -1.0_dp
135 INTEGER :: v_shift_maxiters = -1
138 PRIVATE :: read_negf_atomlist
151 CHARACTER(len=*),
PARAMETER :: routinen =
'negf_control_create'
155 cpassert(.NOT.
ASSOCIATED(negf_control))
156 CALL timeset(routinen, handle)
158 ALLOCATE (negf_control)
160 CALL timestop(handle)
172 CHARACTER(len=*),
PARAMETER :: routinen =
'negf_control_release'
174 INTEGER :: handle, i, j
176 CALL timeset(routinen, handle)
178 IF (
ASSOCIATED(negf_control))
THEN
179 IF (
ALLOCATED(negf_control%atomlist_S))
DEALLOCATE (negf_control%atomlist_S)
180 IF (
ALLOCATED(negf_control%atomlist_S_screening))
DEALLOCATE (negf_control%atomlist_S_screening)
182 IF (
ALLOCATED(negf_control%contacts))
THEN
183 DO i =
SIZE(negf_control%contacts), 1, -1
184 IF (
ALLOCATED(negf_control%contacts(i)%atomlist_bulk)) &
185 DEALLOCATE (negf_control%contacts(i)%atomlist_bulk)
187 IF (
ALLOCATED(negf_control%contacts(i)%atomlist_screening)) &
188 DEALLOCATE (negf_control%contacts(i)%atomlist_screening)
190 IF (
ALLOCATED(negf_control%contacts(i)%atomlist_cell))
THEN
191 DO j =
SIZE(negf_control%contacts(i)%atomlist_cell), 1, -1
192 IF (
ALLOCATED(negf_control%contacts(i)%atomlist_cell(j)%vector)) &
193 DEALLOCATE (negf_control%contacts(i)%atomlist_cell(j)%vector)
195 DEALLOCATE (negf_control%contacts(i)%atomlist_cell)
199 DEALLOCATE (negf_control%contacts)
202 DEALLOCATE (negf_control)
205 CALL timestop(handle)
219 CHARACTER(len=*),
PARAMETER :: routinen =
'read_negf_control'
221 CHARACTER(len=default_string_length) :: contact_id_str, eta_current_str, eta_max_str, &
222 npoles_current_str, npoles_min_str, temp_current_str, temp_min_str
223 INTEGER :: delta_npoles_min, handle, i2_rep, i_rep, &
224 n2_rep, n_rep, natoms_current, &
225 natoms_total, run_type
226 INTEGER,
ALLOCATABLE,
DIMENSION(:) :: inds
227 LOGICAL :: do_negf, is_explicit
228 REAL(kind=
dp) :: eta_max, temp_current, temp_min
230 negf_section, region_section, &
233 CALL timeset(routinen, handle)
241 CALL section_vals_get(contact_section, n_repetition=n_rep, explicit=is_explicit)
242 IF ((.NOT. is_explicit) .AND. do_negf)
THEN
243 CALL cp_abort(__location__, &
244 "At least one contact is needed for NEGF calculation.")
247 ALLOCATE (negf_control%contacts(n_rep))
252 IF ((.NOT. is_explicit) .AND. do_negf)
THEN
253 WRITE (contact_id_str,
'(I11)') i_rep
254 CALL cp_abort(__location__, &
255 "The screening region must be defined for the contact "//trim(adjustl(contact_id_str))//
".")
258 IF (is_explicit)
THEN
259 CALL read_negf_atomlist(negf_control%contacts(i_rep)%atomlist_screening, region_section, 1, subsys)
266 IF ((.NOT. is_explicit) .AND. do_negf)
THEN
267 WRITE (contact_id_str,
'(I11)') i_rep
268 CALL cp_abort(__location__, &
269 "The bulk region must be defined for the contact "//trim(adjustl(contact_id_str))//
".")
272 IF (is_explicit)
THEN
273 CALL read_negf_atomlist(negf_control%contacts(i_rep)%atomlist_bulk, region_section, 1, subsys)
277 i_val=negf_control%contacts(i_rep)%force_env_index, &
281 CALL section_vals_get(cell_section, n_repetition=n2_rep, explicit=is_explicit)
283 IF (((.NOT. is_explicit) .OR. n2_rep /= 2) .AND. negf_control%contacts(i_rep)%force_env_index <= 0 .AND. do_negf)
THEN
284 WRITE (contact_id_str,
'(I11)') i_rep
285 CALL cp_abort(__location__, &
286 "You must either provide indices of atoms belonging to two adjacent bulk unit cells "// &
287 "(BULK_REGION/CELL) for the contact, or the index of the FORCE_EVAL section (FORCE_EVAL_SECTION) "// &
288 "which will be used to construct Kohn-Sham matrix for the bulk contact "// &
289 trim(adjustl(contact_id_str))//
".")
292 IF (is_explicit .AND. n2_rep > 0)
THEN
293 ALLOCATE (negf_control%contacts(i_rep)%atomlist_cell(n2_rep))
295 DO i2_rep = 1, n2_rep
296 CALL read_negf_atomlist(negf_control%contacts(i_rep)%atomlist_cell(i2_rep)%vector, cell_section, i2_rep, subsys)
301 l_val=negf_control%contacts(i_rep)%refine_fermi_level, &
305 r_val=negf_control%contacts(i_rep)%fermi_level, &
306 i_rep_section=i_rep, explicit=is_explicit)
307 IF (.NOT. is_explicit) negf_control%contacts(i_rep)%refine_fermi_level = .false.
308 negf_control%contacts(i_rep)%compute_fermi_level = (.NOT. is_explicit) .OR. &
309 negf_control%contacts(i_rep)%refine_fermi_level
312 r_val=negf_control%contacts(i_rep)%fermi_level_shifted, &
313 i_rep_section=i_rep, explicit=is_explicit)
314 IF (is_explicit) negf_control%contacts(i_rep)%shift_fermi_level = .true.
317 r_val=negf_control%contacts(i_rep)%temperature, &
319 IF (negf_control%contacts(i_rep)%temperature <= 0.0_dp)
THEN
320 CALL cp_abort(__location__,
"Electronic temperature must be > 0")
324 r_val=negf_control%contacts(i_rep)%v_external, &
330 l_val=negf_control%contacts(i_rep)%read_write_HS, &
331 explicit=is_explicit)
332 IF (is_explicit) negf_control%contacts(i_rep)%read_write_HS = .true.
338 IF (is_explicit)
THEN
339 CALL read_negf_atomlist(negf_control%atomlist_S, region_section, 1, subsys)
344 l_val=negf_control%read_write_HS, &
345 explicit=is_explicit)
346 IF (is_explicit) negf_control%read_write_HS = .true.
363 CALL section_vals_val_get(negf_section,
"INTEGRATION_MIN_POINTS", i_val=negf_control%integr_min_points)
364 CALL section_vals_val_get(negf_section,
"INTEGRATION_MAX_POINTS", i_val=negf_control%integr_max_points)
366 IF (negf_control%integr_max_points < negf_control%integr_min_points) &
367 negf_control%integr_max_points = negf_control%integr_min_points
375 CALL section_vals_val_get(negf_section,
"V_SHIFT_MAX_ITERS", i_val=negf_control%v_shift_maxiters)
380 IF (negf_control%eta < 0.0_dp)
THEN
381 CALL cp_abort(__location__,
"ETA must be >= 0")
385 delta_npoles_min = nint(0.5_dp*(negf_control%eta/(
pi*maxval(negf_control%contacts(:)%temperature)) + 1.0_dp))
390 IF (negf_control%delta_npoles < delta_npoles_min)
THEN
392 eta_max = real(2*negf_control%delta_npoles - 1, kind=
dp)*
pi*maxval(negf_control%contacts(:)%temperature)
393 temp_current = maxval(negf_control%contacts(:)%temperature)*
kelvin
394 temp_min = negf_control%eta/(
pi*real(2*negf_control%delta_npoles - 1, kind=
dp))*
kelvin
396 WRITE (eta_current_str,
'(ES11.4E2)') negf_control%eta
397 WRITE (eta_max_str,
'(ES11.4E2)') eta_max
398 WRITE (npoles_current_str,
'(I11)') negf_control%delta_npoles
399 WRITE (npoles_min_str,
'(I11)') delta_npoles_min
400 WRITE (temp_current_str,
'(F11.3)') temp_current
401 WRITE (temp_min_str,
'(F11.3)') temp_min
403 CALL cp_abort(__location__, &
404 "Parameter DELTA_NPOLES must be at least "//trim(adjustl(npoles_min_str))// &
405 " (instead of "//trim(adjustl(npoles_current_str))// &
406 ") for given TEMPERATURE ("//trim(adjustl(temp_current_str))// &
407 " K) and ETA ("//trim(adjustl(eta_current_str))// &
408 "). Alternatively you can increase TEMPERATURE above "//trim(adjustl(temp_min_str))// &
409 " K, or decrease ETA below "//trim(adjustl(eta_max_str))// &
410 ". Please keep in mind that very tight ETA may result in dramatical precision loss"// &
411 " due to inversion of ill-conditioned matrices.")
414 negf_control%delta_npoles = delta_npoles_min
419 n_rep =
SIZE(negf_control%contacts)
420 IF (
ALLOCATED(negf_control%atomlist_S))
THEN
421 natoms_total =
SIZE(negf_control%atomlist_S)
427 IF (
ALLOCATED(negf_control%contacts(i_rep)%atomlist_screening))
THEN
428 IF (
ALLOCATED(negf_control%contacts(i_rep)%atomlist_screening)) &
429 natoms_total = natoms_total +
SIZE(negf_control%contacts(i_rep)%atomlist_screening)
433 IF (natoms_total > 0)
THEN
434 ALLOCATE (negf_control%atomlist_S_screening(natoms_total))
435 IF (
ALLOCATED(negf_control%atomlist_S))
THEN
436 natoms_total =
SIZE(negf_control%atomlist_S)
437 negf_control%atomlist_S_screening(1:natoms_total) = negf_control%atomlist_S(1:natoms_total)
443 IF (
ALLOCATED(negf_control%contacts(i_rep)%atomlist_screening))
THEN
444 natoms_current =
SIZE(negf_control%contacts(i_rep)%atomlist_screening)
446 negf_control%atomlist_S_screening(natoms_total + 1:natoms_total + natoms_current) = &
447 negf_control%contacts(i_rep)%atomlist_screening(1:natoms_current)
449 natoms_total = natoms_total + natoms_current
454 ALLOCATE (inds(natoms_total))
455 CALL sort(negf_control%atomlist_S_screening, natoms_total, inds)
459 DO i_rep = natoms_current + 1, natoms_total
460 IF (negf_control%atomlist_S_screening(i_rep) /= negf_control%atomlist_S_screening(natoms_current))
THEN
461 natoms_current = natoms_current + 1
462 negf_control%atomlist_S_screening(natoms_current) = negf_control%atomlist_S_screening(i_rep)
466 IF (natoms_current < natoms_total)
THEN
467 CALL move_alloc(negf_control%atomlist_S_screening, inds)
469 ALLOCATE (negf_control%atomlist_S_screening(natoms_current))
470 negf_control%atomlist_S_screening(1:natoms_current) = inds(1:natoms_current)
475 IF (do_negf .AND.
SIZE(negf_control%contacts) > 2)
THEN
476 CALL cp_abort(__location__, &
477 "General case (> 2 contacts) has not been implemented yet")
480 CALL timestop(handle)
490 SUBROUTINE read_negf_atomlist(atomlist, input_section, i_rep_section, subsys)
491 INTEGER,
ALLOCATABLE,
DIMENSION(:),
INTENT(out) :: atomlist
493 INTEGER,
INTENT(in) :: i_rep_section
496 CHARACTER(len=*),
PARAMETER :: routinen =
'read_negf_atomlist'
498 CHARACTER(len=default_string_length) :: index_str, natoms_str
499 CHARACTER(len=default_string_length), &
500 DIMENSION(:),
POINTER :: cptr
501 INTEGER :: first_atom, handle, iatom, ikind, imol, iname, irep, last_atom, natoms_current, &
502 natoms_max, natoms_total, nkinds, nmols, nnames, nrep_list, nrep_molname
503 INTEGER,
ALLOCATABLE,
DIMENSION(:) :: inds
504 INTEGER,
DIMENSION(:),
POINTER :: iptr
505 LOGICAL :: is_list, is_molname
512 CALL timeset(routinen, handle)
515 molecule_set=molecule_set, &
516 molecule_kind_set=molecule_kind_set)
517 natoms_max =
SIZE(particle_set)
518 nkinds =
SIZE(molecule_kind_set)
521 n_rep_val=nrep_list, explicit=is_list)
523 n_rep_val=nrep_molname, explicit=is_molname)
527 IF (is_list .AND. nrep_list > 0)
THEN
528 DO irep = 1, nrep_list
529 CALL section_vals_val_get(input_section,
"LIST", i_rep_section=i_rep_section, i_rep_val=irep, i_vals=iptr)
531 natoms_current =
SIZE(iptr)
532 DO iatom = 1, natoms_current
533 IF (iptr(iatom) > natoms_max)
THEN
536 CALL cp_abort(__location__, &
537 "NEGF: Atomic index "//trim(index_str)//
" given in section "// &
538 trim(input_section%section%name)//
" exceeds the maximum number of atoms ("// &
539 trim(natoms_str)//
").")
543 natoms_total = natoms_total + natoms_current
547 IF (is_molname .AND. nrep_molname > 0)
THEN
548 DO irep = 1, nrep_molname
549 CALL section_vals_val_get(input_section,
"MOLNAME", i_rep_section=i_rep_section, i_rep_val=irep, c_vals=cptr)
554 IF (molecule_kind_set(ikind)%name == cptr(iname))
EXIT
557 IF (ikind <= nkinds)
THEN
558 molecule_kind => molecule_kind_set(ikind)
562 molecule => molecule_set(iptr(imol))
563 CALL get_molecule(molecule, first_atom=first_atom, last_atom=last_atom)
564 natoms_current = last_atom - first_atom + 1
565 natoms_total = natoms_total + natoms_current
568 CALL cp_abort(__location__, &
569 "NEGF: A molecule with the name '"//trim(cptr(iname))//
"' mentioned in section "// &
570 trim(input_section%section%name)//
" has not been defined. Note that names are case sensitive.")
577 IF (natoms_total > 0)
THEN
578 ALLOCATE (atomlist(natoms_total))
582 IF (is_list .AND. nrep_list > 0)
THEN
583 DO irep = 1, nrep_list
584 CALL section_vals_val_get(input_section,
"LIST", i_rep_section=i_rep_section, i_rep_val=irep, i_vals=iptr)
586 natoms_current =
SIZE(iptr)
587 atomlist(natoms_total + 1:natoms_total + natoms_current) = iptr(1:natoms_current)
588 natoms_total = natoms_total + natoms_current
592 IF (is_molname .AND. nrep_molname > 0)
THEN
593 DO irep = 1, nrep_molname
594 CALL section_vals_val_get(input_section,
"MOLNAME", i_rep_section=i_rep_section, i_rep_val=irep, c_vals=cptr)
599 IF (molecule_kind_set(ikind)%name == cptr(iname))
EXIT
602 IF (ikind <= nkinds)
THEN
603 molecule_kind => molecule_kind_set(ikind)
607 molecule => molecule_set(iptr(imol))
608 CALL get_molecule(molecule, first_atom=first_atom, last_atom=last_atom)
610 DO natoms_current = first_atom, last_atom
611 natoms_total = natoms_total + 1
612 atomlist(natoms_total) = natoms_current
621 ALLOCATE (inds(natoms_total))
622 CALL sort(atomlist, natoms_total, inds)
626 DO iatom = natoms_current + 1, natoms_total
627 IF (atomlist(iatom) /= atomlist(natoms_current))
THEN
628 natoms_current = natoms_current + 1
629 atomlist(natoms_current) = atomlist(iatom)
633 IF (natoms_current < natoms_total)
THEN
634 CALL move_alloc(atomlist, inds)
636 ALLOCATE (atomlist(natoms_current))
637 atomlist(1:natoms_current) = inds(1:natoms_current)
642 CALL timestop(handle)
643 END SUBROUTINE read_negf_atomlist
types that represent a subsys, i.e. a part of the system
subroutine, public cp_subsys_get(subsys, ref_count, atomic_kinds, atomic_kind_set, particles, particle_set, local_particles, molecules, molecule_set, molecule_kinds, molecule_kind_set, local_molecules, para_env, colvar_p, shell_particles, core_particles, gci, multipoles, natom, nparticle, ncore, nshell, nkind, atprop, virial, results, cell)
returns information about various attributes of the given subsys
Defines the basic variable types.
integer, parameter, public dp
integer, parameter, public default_string_length
Definition of mathematical constants and functions.
real(kind=dp), parameter, public pi
Define the molecule kind structure types and the corresponding functionality.
subroutine, public get_molecule_kind(molecule_kind, atom_list, bond_list, bend_list, ub_list, impr_list, opbend_list, colv_list, fixd_list, g3x3_list, g4x6_list, vsite_list, torsion_list, shell_list, name, mass, charge, kind_number, natom, nbend, nbond, nub, nimpr, nopbend, nconstraint, nconstraint_fixd, nfixd, ncolv, ng3x3, ng4x6, nvsite, nfixd_restraint, ng3x3_restraint, ng4x6_restraint, nvsite_restraint, nrestraints, nmolecule, nsgf, nshell, ntorsion, molecule_list, nelectron, nelectron_alpha, nelectron_beta, bond_kind_set, bend_kind_set, ub_kind_set, impr_kind_set, opbend_kind_set, torsion_kind_set, molname_generated)
Get informations about a molecule kind.
Define the data structure for the molecule information.
subroutine, public get_molecule(molecule, molecule_kind, lmi, lci, lg3x3, lg4x6, lcolv, first_atom, last_atom, first_shell, last_shell)
Get components from a molecule data set.
Allocatable vectors for NEGF based quantum transport calculations.
Input control types for NEGF based quantum transport calculations.
subroutine, public negf_control_create(negf_control)
allocate control options for Non-equilibrium Green's Function calculation
subroutine, public read_negf_control(negf_control, input, subsys)
Read NEGF input parameters.
subroutine, public negf_control_release(negf_control)
release memory allocated for NEGF control options
Define the data structure for the particle information.
Definition of physical constants:
real(kind=dp), parameter, public kelvin
Utilities for string manipulations.
subroutine, public integer_to_string(inumber, string)
Converts an integer number to a string. The WRITE statement will return an error message,...
All kind of helpful little routines.
represents a system: atoms, molecules, their pos,vel,...
Allocatable 1-D integer vector.
Input parameters related to the NEGF run.