LCOV - code coverage report
Current view: top level - src - pao_io.F (source / functions) Coverage Total Hit
Test: CP2K Regtests (git:fd1133a) Lines: 82.2 % 320 263
Test Date: 2025-12-07 06:28:01 Functions: 71.4 % 14 10

            Line data    Source code
       1              : !--------------------------------------------------------------------------------------------------!
       2              : !   CP2K: A general program to perform molecular dynamics simulations                              !
       3              : !   Copyright 2000-2025 CP2K developers group <https://cp2k.org>                                   !
       4              : !                                                                                                  !
       5              : !   SPDX-License-Identifier: GPL-2.0-or-later                                                      !
       6              : !--------------------------------------------------------------------------------------------------!
       7              : 
       8              : ! **************************************************************************************************
       9              : !> \brief Routines for reading and writing restart files.
      10              : !> \author Ole Schuett
      11              : ! **************************************************************************************************
      12              : MODULE pao_io
      13              :    USE atomic_kind_types,               ONLY: atomic_kind_type,&
      14              :                                               get_atomic_kind
      15              :    USE basis_set_types,                 ONLY: gto_basis_set_type
      16              :    USE cell_types,                      ONLY: cell_type
      17              :    USE cp_dbcsr_api,                    ONLY: &
      18              :         dbcsr_convert_dbcsr_to_csr, dbcsr_copy, dbcsr_csr_create_from_dbcsr, &
      19              :         dbcsr_csr_dbcsr_blkrow_dist, dbcsr_csr_destroy, dbcsr_csr_type, dbcsr_csr_write, &
      20              :         dbcsr_desymmetrize, dbcsr_get_block_p, dbcsr_get_info, dbcsr_has_symmetry, dbcsr_release, &
      21              :         dbcsr_type
      22              :    USE cp_files,                        ONLY: close_file,&
      23              :                                               open_file
      24              :    USE cp_log_handling,                 ONLY: cp_get_default_logger,&
      25              :                                               cp_logger_get_default_io_unit,&
      26              :                                               cp_logger_type
      27              :    USE cp_output_handling,              ONLY: cp_p_file,&
      28              :                                               cp_print_key_finished_output,&
      29              :                                               cp_print_key_should_output,&
      30              :                                               cp_print_key_unit_nr
      31              :    USE dm_ls_scf_types,                 ONLY: ls_scf_env_type
      32              :    USE input_section_types,             ONLY: section_vals_get_subs_vals,&
      33              :                                               section_vals_type,&
      34              :                                               section_vals_val_get
      35              :    USE kinds,                           ONLY: default_path_length,&
      36              :                                               default_string_length,&
      37              :                                               dp
      38              :    USE message_passing,                 ONLY: mp_para_env_type
      39              :    USE pao_input,                       ONLY: id2str
      40              :    USE pao_param,                       ONLY: pao_param_count
      41              :    USE pao_types,                       ONLY: pao_env_type
      42              :    USE particle_types,                  ONLY: particle_type
      43              :    USE physcon,                         ONLY: angstrom
      44              :    USE qs_environment_types,            ONLY: get_qs_env,&
      45              :                                               qs_environment_type
      46              :    USE qs_kind_types,                   ONLY: get_qs_kind,&
      47              :                                               pao_potential_type,&
      48              :                                               qs_kind_type
      49              : #include "./base/base_uses.f90"
      50              : 
      51              :    IMPLICIT NONE
      52              : 
      53              :    PRIVATE
      54              : 
      55              :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'pao_io'
      56              : 
      57              :    PUBLIC :: pao_read_restart, pao_write_restart
      58              :    PUBLIC :: pao_read_raw, pao_kinds_ensure_equal
      59              :    PUBLIC :: pao_ioblock_type, pao_iokind_type
      60              :    PUBLIC :: pao_write_ks_matrix_csr, pao_write_s_matrix_csr
      61              :    PUBLIC :: pao_write_hcore_matrix_csr, pao_write_p_matrix_csr
      62              : 
      63              :    ! data types used by pao_read_raw()
      64              :    TYPE pao_ioblock_type
      65              :       REAL(dp), DIMENSION(:, :), ALLOCATABLE    :: p
      66              :    END TYPE pao_ioblock_type
      67              : 
      68              :    TYPE pao_iokind_type
      69              :       CHARACTER(LEN=default_string_length)     :: name = ""
      70              :       INTEGER                                  :: z = -1
      71              :       CHARACTER(LEN=default_string_length)     :: prim_basis_name = ""
      72              :       INTEGER                                  :: prim_basis_size = -1
      73              :       INTEGER                                  :: pao_basis_size = -1
      74              :       INTEGER                                  :: nparams = -1
      75              :       TYPE(pao_potential_type), ALLOCATABLE, DIMENSION(:) :: pao_potentials
      76              :    END TYPE pao_iokind_type
      77              : 
      78              :    INTEGER, PARAMETER, PRIVATE :: file_format_version = 4
      79              : 
      80              : CONTAINS
      81              : 
      82              : ! **************************************************************************************************
      83              : !> \brief Reads restart file
      84              : !> \param pao ...
      85              : !> \param qs_env ...
      86              : ! **************************************************************************************************
      87            8 :    SUBROUTINE pao_read_restart(pao, qs_env)
      88              :       TYPE(pao_env_type), POINTER                        :: pao
      89              :       TYPE(qs_environment_type), POINTER                 :: qs_env
      90              : 
      91              :       CHARACTER(LEN=default_string_length)               :: param
      92              :       INTEGER                                            :: iatom, ikind, natoms
      93            8 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: atom2kind
      94            8 :       INTEGER, DIMENSION(:), POINTER                     :: col_blk_sizes, row_blk_sizes
      95              :       LOGICAL                                            :: found
      96              :       REAL(dp)                                           :: diff
      97            8 :       REAL(dp), ALLOCATABLE, DIMENSION(:, :)             :: hmat, positions
      98            8 :       REAL(dp), DIMENSION(:, :), POINTER                 :: block_X, buffer
      99              :       TYPE(cell_type), POINTER                           :: cell
     100              :       TYPE(mp_para_env_type), POINTER                    :: para_env
     101            8 :       TYPE(pao_ioblock_type), ALLOCATABLE, DIMENSION(:)  :: xblocks
     102            8 :       TYPE(pao_iokind_type), ALLOCATABLE, DIMENSION(:)   :: kinds
     103            8 :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     104              : 
     105            0 :       CPASSERT(LEN_TRIM(pao%restart_file) > 0)
     106            8 :       IF (pao%iw > 0) WRITE (pao%iw, '(A,A)') " PAO| Reading matrix_X from restart file: ", TRIM(pao%restart_file)
     107              : 
     108              :       CALL get_qs_env(qs_env, &
     109              :                       para_env=para_env, &
     110              :                       natom=natoms, &
     111              :                       cell=cell, &
     112            8 :                       particle_set=particle_set)
     113              : 
     114              :       ! read and check restart file on first rank only
     115            8 :       IF (para_env%is_source()) THEN
     116            4 :          CALL pao_read_raw(pao%restart_file, param, hmat, kinds, atom2kind, positions, xblocks)
     117              : 
     118              :          ! check cell
     119           52 :          IF (MAXVAL(ABS(hmat - cell%hmat)) > 1e-10) THEN
     120            0 :             CPWARN("Restarting from different cell")
     121              :          END IF
     122              : 
     123              :          ! check parametrization
     124            4 :          IF (TRIM(param) /= TRIM(ADJUSTL(id2str(pao%parameterization)))) &
     125            4 :             CPABORT("Restart PAO parametrization does not match")
     126              : 
     127              :          ! check kinds
     128           11 :          DO ikind = 1, SIZE(kinds)
     129           11 :             CALL pao_kinds_ensure_equal(pao, qs_env, ikind, kinds(ikind))
     130              :          END DO
     131              : 
     132              :          ! check number of atoms
     133            4 :          IF (SIZE(positions, 1) /= natoms) &
     134            0 :             CPABORT("Number of atoms do not match")
     135              : 
     136              :          ! check atom2kind
     137           15 :          DO iatom = 1, natoms
     138           11 :             IF (atom2kind(iatom) /= particle_set(iatom)%atomic_kind%kind_number) &
     139            4 :                CPABORT("Restart atomic kinds do not match.")
     140              :          END DO
     141              : 
     142              :          ! check positions, warning only
     143            4 :          diff = 0.0_dp
     144           15 :          DO iatom = 1, natoms
     145           59 :             diff = MAX(diff, MAXVAL(ABS(positions(iatom, :) - particle_set(iatom)%r)))
     146              :          END DO
     147            4 :          CPWARN_IF(diff > 1e-10, "Restarting from different atom positions")
     148              : 
     149              :       END IF
     150              : 
     151              :       ! scatter xblocks across ranks to fill pao%matrix_X
     152              :       ! this could probably be done more efficiently
     153            8 :       CALL dbcsr_get_info(pao%matrix_X, row_blk_size=row_blk_sizes, col_blk_size=col_blk_sizes)
     154           30 :       DO iatom = 1, natoms
     155           88 :          ALLOCATE (buffer(row_blk_sizes(iatom), col_blk_sizes(iatom)))
     156           22 :          IF (para_env%is_source()) THEN
     157           11 :             CPASSERT(row_blk_sizes(iatom) == SIZE(xblocks(iatom)%p, 1))
     158           11 :             CPASSERT(col_blk_sizes(iatom) == SIZE(xblocks(iatom)%p, 2))
     159          193 :             buffer = xblocks(iatom)%p
     160              :          END IF
     161          750 :          CALL para_env%bcast(buffer)
     162           22 :          CALL dbcsr_get_block_p(matrix=pao%matrix_X, row=iatom, col=iatom, block=block_X, found=found)
     163           22 :          IF (ASSOCIATED(block_X)) &
     164          375 :             block_X = buffer
     165           52 :          DEALLOCATE (buffer)
     166              :       END DO
     167              : 
     168              :       ! ALLOCATABLEs deallocate themselves
     169              : 
     170           34 :    END SUBROUTINE pao_read_restart
     171              : 
     172              : ! **************************************************************************************************
     173              : !> \brief Reads a restart file into temporary datastructures
     174              : !> \param filename ...
     175              : !> \param param ...
     176              : !> \param hmat ...
     177              : !> \param kinds ...
     178              : !> \param atom2kind ...
     179              : !> \param positions ...
     180              : !> \param xblocks ...
     181              : !> \param ml_range ...
     182              : ! **************************************************************************************************
     183           21 :    SUBROUTINE pao_read_raw(filename, param, hmat, kinds, atom2kind, positions, xblocks, ml_range)
     184              :       CHARACTER(LEN=default_path_length), INTENT(IN)     :: filename
     185              :       CHARACTER(LEN=default_string_length), INTENT(OUT)  :: param
     186              :       REAL(dp), ALLOCATABLE, DIMENSION(:, :)             :: hmat
     187              :       TYPE(pao_iokind_type), ALLOCATABLE, DIMENSION(:)   :: kinds
     188              :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: atom2kind
     189              :       REAL(dp), ALLOCATABLE, DIMENSION(:, :)             :: positions
     190              :       TYPE(pao_ioblock_type), ALLOCATABLE, DIMENSION(:)  :: xblocks
     191              :       INTEGER, DIMENSION(2), INTENT(OUT), OPTIONAL       :: ml_range
     192              : 
     193              :       CHARACTER(LEN=default_string_length)               :: label, str_in
     194              :       INTEGER                                            :: i1, i2, iatom, ikind, ipot, natoms, &
     195              :                                                             nkinds, nparams, unit_nr, xblocks_read
     196              :       REAL(dp)                                           :: r1, r2
     197              :       REAL(dp), DIMENSION(3)                             :: pos_in
     198              :       REAL(dp), DIMENSION(3, 3)                          :: hmat_angstrom
     199              : 
     200           21 :       CPASSERT(.NOT. ALLOCATED(hmat))
     201           21 :       CPASSERT(.NOT. ALLOCATED(kinds))
     202           21 :       CPASSERT(.NOT. ALLOCATED(atom2kind))
     203           21 :       CPASSERT(.NOT. ALLOCATED(positions))
     204           21 :       CPASSERT(.NOT. ALLOCATED(xblocks))
     205              : 
     206           21 :       natoms = -1
     207           21 :       nkinds = -1
     208           21 :       xblocks_read = 0
     209              : 
     210              :       CALL open_file(file_name=filename, file_status="OLD", file_form="FORMATTED", &
     211           21 :                      file_action="READ", unit_number=unit_nr)
     212              : 
     213              :       ! check if file starts with proper header !TODO: introduce a more unique header
     214           21 :       READ (unit_nr, fmt=*) label, i1
     215           21 :       IF (TRIM(label) /= "Version") &
     216            0 :          CPABORT("PAO restart file appears to be corrupted.")
     217           21 :       IF (i1 /= file_format_version) CPABORT("Restart PAO file format version is wrong")
     218              : 
     219              :       DO WHILE (.TRUE.)
     220          377 :          READ (unit_nr, fmt=*) label
     221          377 :          BACKSPACE (unit_nr)
     222              : 
     223          398 :          IF (TRIM(label) == "Parametrization") THEN
     224           21 :             READ (unit_nr, fmt=*) label, str_in
     225           21 :             param = str_in
     226              : 
     227          356 :          ELSE IF (TRIM(label) == "Cell") THEN
     228           21 :             READ (unit_nr, fmt=*) label, hmat_angstrom
     229           21 :             ALLOCATE (hmat(3, 3))
     230          273 :             hmat(:, :) = hmat_angstrom(:, :)/angstrom
     231              : 
     232          335 :          ELSE IF (TRIM(label) == "Nkinds") THEN
     233           21 :             READ (unit_nr, fmt=*) label, nkinds
     234           87 :             ALLOCATE (kinds(nkinds))
     235              : 
     236          314 :          ELSE IF (TRIM(label) == "Kind") THEN
     237           24 :             READ (unit_nr, fmt=*) label, ikind, str_in, i1
     238           24 :             CPASSERT(ALLOCATED(kinds))
     239           24 :             kinds(ikind)%name = str_in
     240           24 :             kinds(ikind)%z = i1
     241              : 
     242          290 :          ELSE IF (TRIM(label) == "PrimBasis") THEN
     243           24 :             READ (unit_nr, fmt=*) label, ikind, i1, str_in
     244           24 :             CPASSERT(ALLOCATED(kinds))
     245           24 :             kinds(ikind)%prim_basis_size = i1
     246           24 :             kinds(ikind)%prim_basis_name = str_in
     247              : 
     248          266 :          ELSE IF (TRIM(label) == "PaoBasis") THEN
     249           24 :             READ (unit_nr, fmt=*) label, ikind, i1
     250           24 :             CPASSERT(ALLOCATED(kinds))
     251           24 :             kinds(ikind)%pao_basis_size = i1
     252              : 
     253          242 :          ELSE IF (TRIM(label) == "NPaoPotentials") THEN
     254           24 :             READ (unit_nr, fmt=*) label, ikind, i1
     255           24 :             CPASSERT(ALLOCATED(kinds))
     256           88 :             ALLOCATE (kinds(ikind)%pao_potentials(i1))
     257              : 
     258          218 :          ELSE IF (TRIM(label) == "PaoPotential") THEN
     259           20 :             READ (unit_nr, fmt=*) label, ikind, ipot, i1, i2, r1, r2
     260           20 :             CPASSERT(ALLOCATED(kinds(ikind)%pao_potentials))
     261           20 :             kinds(ikind)%pao_potentials(ipot)%maxl = i1
     262           20 :             kinds(ikind)%pao_potentials(ipot)%max_projector = i2
     263           20 :             kinds(ikind)%pao_potentials(ipot)%beta = r1
     264           20 :             kinds(ikind)%pao_potentials(ipot)%weight = r2
     265              : 
     266          198 :          ELSE IF (TRIM(label) == "NParams") THEN
     267           24 :             READ (unit_nr, fmt=*) label, ikind, i1
     268           24 :             CPASSERT(ALLOCATED(kinds))
     269           24 :             kinds(ikind)%nparams = i1
     270              : 
     271          174 :          ELSE IF (TRIM(label) == "Natoms") THEN
     272           21 :             READ (unit_nr, fmt=*) label, natoms
     273          192 :             ALLOCATE (positions(natoms, 3), atom2kind(natoms), xblocks(natoms))
     274          264 :             positions = 0.0_dp; atom2kind = -1
     275           55 :             IF (PRESENT(ml_range)) ml_range = [1, natoms]
     276              : 
     277          153 :          ELSE IF (TRIM(label) == "MLRange") THEN
     278              :             ! Natoms entry has to come first
     279            0 :             CPASSERT(natoms > 0)
     280              :             ! range of atoms whose xblocks are used for machine learning
     281            0 :             READ (unit_nr, fmt=*) label, i1, i2
     282            0 :             IF (PRESENT(ml_range)) ml_range = [i1, i2]
     283              : 
     284          153 :          ELSE IF (TRIM(label) == "Atom") THEN
     285           45 :             READ (unit_nr, fmt=*) label, iatom, str_in, pos_in
     286           45 :             CPASSERT(ALLOCATED(kinds))
     287           51 :             DO ikind = 1, nkinds
     288           51 :                IF (TRIM(kinds(ikind)%name) == TRIM(str_in)) EXIT
     289              :             END DO
     290           45 :             CPASSERT(ALLOCATED(atom2kind) .AND. ALLOCATED(positions))
     291           45 :             atom2kind(iatom) = ikind
     292          180 :             positions(iatom, :) = pos_in/angstrom
     293              : 
     294          108 :          ELSE IF (TRIM(label) == "Xblock") THEN
     295           45 :             READ (unit_nr, fmt=*) label, iatom
     296           45 :             CPASSERT(ALLOCATED(kinds) .AND. ALLOCATED(atom2kind))
     297           45 :             ikind = atom2kind(iatom)
     298           45 :             nparams = kinds(ikind)%nparams
     299           45 :             CPASSERT(nparams >= 0)
     300          135 :             ALLOCATE (xblocks(iatom)%p(nparams, 1))
     301           45 :             BACKSPACE (unit_nr)
     302          499 :             READ (unit_nr, fmt=*) label, iatom, xblocks(iatom)%p
     303           45 :             xblocks_read = xblocks_read + 1
     304           45 :             CPASSERT(iatom == xblocks_read) ! ensure blocks are read in order
     305              : 
     306           63 :          ELSE IF (TRIM(label) == "THE_END") THEN
     307              :             EXIT
     308              :          ELSE
     309              :             !CPWARN("Skipping restart header with label: "//TRIM(label))
     310           42 :             READ (unit_nr, fmt=*) label ! just read again and ignore
     311              :          END IF
     312              :       END DO
     313           21 :       CALL close_file(unit_number=unit_nr)
     314              : 
     315           21 :       CPASSERT(xblocks_read == natoms) ! ensure we read all blocks
     316              : 
     317           21 :    END SUBROUTINE pao_read_raw
     318              : 
     319              : ! **************************************************************************************************
     320              : !> \brief Ensure that the kind read from the restart is equal to the kind curretly in use.
     321              : !> \param pao ...
     322              : !> \param qs_env ...
     323              : !> \param ikind ...
     324              : !> \param pao_kind ...
     325              : ! **************************************************************************************************
     326           96 :    SUBROUTINE pao_kinds_ensure_equal(pao, qs_env, ikind, pao_kind)
     327              :       TYPE(pao_env_type), POINTER                        :: pao
     328              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     329              :       INTEGER, INTENT(IN)                                :: ikind
     330              :       TYPE(pao_iokind_type), INTENT(IN)                  :: pao_kind
     331              : 
     332              :       CHARACTER(LEN=default_string_length)               :: name
     333              :       INTEGER                                            :: ipot, nparams, pao_basis_size, z
     334           24 :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
     335              :       TYPE(gto_basis_set_type), POINTER                  :: basis_set
     336           24 :       TYPE(pao_potential_type), DIMENSION(:), POINTER    :: pao_potentials
     337           24 :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
     338              : 
     339              :       CALL get_qs_env(qs_env, &
     340              :                       atomic_kind_set=atomic_kind_set, &
     341           24 :                       qs_kind_set=qs_kind_set)
     342              : 
     343           24 :       IF (ikind > SIZE(atomic_kind_set) .OR. ikind > SIZE(qs_kind_set)) &
     344            0 :          CPABORT("Some kinds are missing.")
     345              : 
     346           24 :       CALL get_atomic_kind(atomic_kind_set(ikind), z=z, name=name)
     347              :       CALL get_qs_kind(qs_kind_set(ikind), &
     348              :                        basis_set=basis_set, &
     349              :                        pao_basis_size=pao_basis_size, &
     350           24 :                        pao_potentials=pao_potentials)
     351           24 :       CALL pao_param_count(pao, qs_env, ikind=ikind, nparams=nparams)
     352              : 
     353           24 :       IF (pao_kind%nparams /= nparams) &
     354            0 :          CPABORT("Number of parameters do not match")
     355           24 :       IF (TRIM(pao_kind%name) /= TRIM(name)) &
     356            0 :          CPABORT("Kind names do not match")
     357           24 :       IF (pao_kind%z /= z) &
     358            0 :          CPABORT("Atomic numbers do not match")
     359           24 :       IF (TRIM(pao_kind%prim_basis_name) /= TRIM(basis_set%name)) &
     360            0 :          CPABORT("Primary Basis-set name does not match")
     361           24 :       IF (pao_kind%prim_basis_size /= basis_set%nsgf) &
     362            0 :          CPABORT("Primary Basis-set size does not match")
     363           24 :       IF (pao_kind%pao_basis_size /= pao_basis_size) &
     364            0 :          CPABORT("PAO basis size does not match")
     365           24 :       IF (SIZE(pao_kind%pao_potentials) /= SIZE(pao_potentials)) &
     366            0 :          CPABORT("Number of PAO_POTENTIALS does not match")
     367              : 
     368           44 :       DO ipot = 1, SIZE(pao_potentials)
     369           20 :          IF (pao_kind%pao_potentials(ipot)%maxl /= pao_potentials(ipot)%maxl) THEN
     370            0 :             CPABORT("PAO_POT_MAXL does not match")
     371              :          END IF
     372           20 :          IF (pao_kind%pao_potentials(ipot)%max_projector /= pao_potentials(ipot)%max_projector) THEN
     373            0 :             CPABORT("PAO_POT_MAX_PROJECTOR does not match")
     374              :          END IF
     375           20 :          IF (pao_kind%pao_potentials(ipot)%beta /= pao_potentials(ipot)%beta) THEN
     376            0 :             CPWARN("PAO_POT_BETA does not match")
     377              :          END IF
     378           44 :          IF (pao_kind%pao_potentials(ipot)%weight /= pao_potentials(ipot)%weight) THEN
     379            0 :             CPWARN("PAO_POT_WEIGHT does not match")
     380              :          END IF
     381              :       END DO
     382              : 
     383           24 :    END SUBROUTINE pao_kinds_ensure_equal
     384              : 
     385              : ! **************************************************************************************************
     386              : !> \brief Writes restart file
     387              : !> \param pao ...
     388              : !> \param qs_env ...
     389              : !> \param energy ...
     390              : ! **************************************************************************************************
     391          254 :    SUBROUTINE pao_write_restart(pao, qs_env, energy)
     392              :       TYPE(pao_env_type), POINTER                        :: pao
     393              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     394              :       REAL(dp)                                           :: energy
     395              : 
     396              :       CHARACTER(len=*), PARAMETER :: printkey_section = 'DFT%LS_SCF%PAO%PRINT%RESTART', &
     397              :          routineN = 'pao_write_restart'
     398              : 
     399              :       INTEGER                                            :: handle, unit_max, unit_nr
     400              :       TYPE(cp_logger_type), POINTER                      :: logger
     401              :       TYPE(mp_para_env_type), POINTER                    :: para_env
     402              :       TYPE(section_vals_type), POINTER                   :: input
     403              : 
     404          254 :       CALL timeset(routineN, handle)
     405          254 :       logger => cp_get_default_logger()
     406              : 
     407          254 :       CALL get_qs_env(qs_env, input=input, para_env=para_env)
     408              : 
     409              :       ! open file
     410              :       unit_nr = cp_print_key_unit_nr(logger, &
     411              :                                      input, &
     412              :                                      printkey_section, &
     413              :                                      extension=".pao", &
     414              :                                      file_action="WRITE", &
     415              :                                      file_position="REWIND", &
     416              :                                      file_status="UNKNOWN", &
     417          254 :                                      do_backup=.TRUE.)
     418              : 
     419              :       ! although just rank-0 writes the trajectory it requires collective MPI calls
     420          254 :       unit_max = unit_nr
     421          254 :       CALL para_env%max(unit_max)
     422          254 :       IF (unit_max > 0) THEN
     423          104 :          IF (pao%iw > 0) WRITE (pao%iw, '(A,A)') " PAO| Writing restart file."
     424          104 :          IF (unit_nr > 0) &
     425           52 :             CALL write_restart_header(pao, qs_env, energy, unit_nr)
     426              : 
     427          104 :          CALL pao_write_diagonal_blocks(para_env, pao%matrix_X, "Xblock", unit_nr)
     428              : 
     429              :       END IF
     430              : 
     431              :       ! close file
     432          254 :       IF (unit_nr > 0) WRITE (unit_nr, '(A)') "THE_END"
     433          254 :       CALL cp_print_key_finished_output(unit_nr, logger, input, printkey_section)
     434              : 
     435          254 :       CALL timestop(handle)
     436          254 :    END SUBROUTINE pao_write_restart
     437              : 
     438              : ! **************************************************************************************************
     439              : !> \brief Write the digonal blocks of given DBCSR matrix into the provided unit_nr
     440              : !> \param para_env ...
     441              : !> \param matrix ...
     442              : !> \param label ...
     443              : !> \param unit_nr ...
     444              : ! **************************************************************************************************
     445          104 :    SUBROUTINE pao_write_diagonal_blocks(para_env, matrix, label, unit_nr)
     446              :       TYPE(mp_para_env_type), POINTER                    :: para_env
     447              :       TYPE(dbcsr_type)                                   :: matrix
     448              :       CHARACTER(LEN=*), INTENT(IN)                       :: label
     449              :       INTEGER, INTENT(IN)                                :: unit_nr
     450              : 
     451              :       INTEGER                                            :: iatom, natoms
     452          104 :       INTEGER, DIMENSION(:), POINTER                     :: col_blk_sizes, row_blk_sizes
     453              :       LOGICAL                                            :: found
     454          104 :       REAL(dp), DIMENSION(:, :), POINTER                 :: local_block, mpi_buffer
     455              : 
     456              :       !TODO: this is a serial algorithm
     457          104 :       CALL dbcsr_get_info(matrix, row_blk_size=row_blk_sizes, col_blk_size=col_blk_sizes)
     458          104 :       CPASSERT(SIZE(row_blk_sizes) == SIZE(col_blk_sizes))
     459          104 :       natoms = SIZE(row_blk_sizes)
     460              : 
     461          352 :       DO iatom = 1, natoms
     462          984 :          ALLOCATE (mpi_buffer(row_blk_sizes(iatom), col_blk_sizes(iatom)))
     463          248 :          NULLIFY (local_block)
     464          248 :          CALL dbcsr_get_block_p(matrix=matrix, row=iatom, col=iatom, block=local_block, found=found)
     465          248 :          IF (ASSOCIATED(local_block)) THEN
     466          372 :             IF (SIZE(local_block) > 0) & ! catch corner-case
     467         4204 :                mpi_buffer(:, :) = local_block(:, :)
     468              :          ELSE
     469         2110 :             mpi_buffer(:, :) = 0.0_dp
     470              :          END IF
     471              : 
     472         8192 :          CALL para_env%sum(mpi_buffer)
     473          248 :          IF (unit_nr > 0) THEN
     474          124 :             WRITE (unit_nr, fmt="(A,1X,I10,1X)", advance='no') label, iatom
     475         2110 :             WRITE (unit_nr, *) mpi_buffer
     476              :          END IF
     477          600 :          DEALLOCATE (mpi_buffer)
     478              :       END DO
     479              : 
     480              :       ! flush
     481          104 :       IF (unit_nr > 0) FLUSH (unit_nr)
     482              : 
     483          104 :    END SUBROUTINE pao_write_diagonal_blocks
     484              : 
     485              : ! **************************************************************************************************
     486              : !> \brief Writes header of restart file
     487              : !> \param pao ...
     488              : !> \param qs_env ...
     489              : !> \param energy ...
     490              : !> \param unit_nr ...
     491              : ! **************************************************************************************************
     492           52 :    SUBROUTINE write_restart_header(pao, qs_env, energy, unit_nr)
     493              :       TYPE(pao_env_type), POINTER                        :: pao
     494              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     495              :       REAL(dp)                                           :: energy
     496              :       INTEGER, INTENT(IN)                                :: unit_nr
     497              : 
     498              :       CHARACTER(LEN=default_string_length)               :: kindname
     499              :       INTEGER                                            :: iatom, ikind, ipot, nparams, &
     500              :                                                             pao_basis_size, z
     501           52 :       TYPE(atomic_kind_type), DIMENSION(:), POINTER      :: atomic_kind_set
     502              :       TYPE(cell_type), POINTER                           :: cell
     503              :       TYPE(gto_basis_set_type), POINTER                  :: basis_set
     504           52 :       TYPE(pao_potential_type), DIMENSION(:), POINTER    :: pao_potentials
     505           52 :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     506           52 :       TYPE(qs_kind_type), DIMENSION(:), POINTER          :: qs_kind_set
     507              : 
     508              :       CALL get_qs_env(qs_env, &
     509              :                       cell=cell, &
     510              :                       particle_set=particle_set, &
     511              :                       atomic_kind_set=atomic_kind_set, &
     512           52 :                       qs_kind_set=qs_kind_set)
     513              : 
     514           52 :       WRITE (unit_nr, "(A,5X,I0)") "Version", file_format_version
     515           52 :       WRITE (unit_nr, "(A,5X,F20.10)") "Energy", energy
     516           52 :       WRITE (unit_nr, "(A,5X,I0)") "Step", pao%istep
     517           52 :       WRITE (unit_nr, "(A,5X,A)") "Parametrization", id2str(pao%parameterization)
     518              : 
     519              :       ! write kinds
     520           52 :       WRITE (unit_nr, "(A,5X,I0)") "Nkinds", SIZE(atomic_kind_set)
     521          124 :       DO ikind = 1, SIZE(atomic_kind_set)
     522           72 :          CALL get_atomic_kind(atomic_kind_set(ikind), name=kindname, z=z)
     523              :          CALL get_qs_kind(qs_kind_set(ikind), &
     524              :                           pao_basis_size=pao_basis_size, &
     525              :                           pao_potentials=pao_potentials, &
     526           72 :                           basis_set=basis_set)
     527           72 :          CALL pao_param_count(pao, qs_env, ikind, nparams)
     528           72 :          WRITE (unit_nr, "(A,5X,I10,1X,A,1X,I3)") "Kind", ikind, TRIM(kindname), z
     529           72 :          WRITE (unit_nr, "(A,5X,I10,1X,I3)") "NParams", ikind, nparams
     530           72 :          WRITE (unit_nr, "(A,5X,I10,1X,I10,1X,A)") "PrimBasis", ikind, basis_set%nsgf, TRIM(basis_set%name)
     531           72 :          WRITE (unit_nr, "(A,5X,I10,1X,I3)") "PaoBasis", ikind, pao_basis_size
     532           72 :          WRITE (unit_nr, "(A,5X,I10,1X,I3)") "NPaoPotentials", ikind, SIZE(pao_potentials)
     533          245 :          DO ipot = 1, SIZE(pao_potentials)
     534           49 :             WRITE (unit_nr, "(A,5X,I10,1X,I3)", advance='no') "PaoPotential", ikind, ipot
     535           49 :             WRITE (unit_nr, "(1X,I3)", advance='no') pao_potentials(ipot)%maxl
     536           49 :             WRITE (unit_nr, "(1X,I3)", advance='no') pao_potentials(ipot)%max_projector
     537           49 :             WRITE (unit_nr, "(1X,F20.16)", advance='no') pao_potentials(ipot)%beta
     538          121 :             WRITE (unit_nr, "(1X,F20.16)") pao_potentials(ipot)%weight
     539              :          END DO
     540              :       END DO
     541              : 
     542              :       ! write cell
     543           52 :       WRITE (unit_nr, fmt="(A,5X)", advance='no') "Cell"
     544          676 :       WRITE (unit_nr, *) cell%hmat*angstrom
     545              : 
     546              :       ! write atoms
     547           52 :       WRITE (unit_nr, "(A,5X,I0)") "Natoms", SIZE(particle_set)
     548          176 :       DO iatom = 1, SIZE(particle_set)
     549          124 :          kindname = particle_set(iatom)%atomic_kind%name
     550          124 :          WRITE (unit_nr, fmt="(A,5X,I10,5X,A,1X)", advance='no') "Atom ", iatom, TRIM(kindname)
     551          548 :          WRITE (unit_nr, *) particle_set(iatom)%r*angstrom
     552              :       END DO
     553              : 
     554           52 :    END SUBROUTINE write_restart_header
     555              : 
     556              : !**************************************************************************************************
     557              : !> \brief writing the KS matrix (in terms of the PAO basis) in csr format into a file
     558              : !> \param qs_env qs environment
     559              : !> \param ls_scf_env ls environment
     560              : !> \author Mohammad Hossein Bani-Hashemian
     561              : ! **************************************************************************************************
     562          294 :    SUBROUTINE pao_write_ks_matrix_csr(qs_env, ls_scf_env)
     563              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     564              :       TYPE(ls_scf_env_type), TARGET                      :: ls_scf_env
     565              : 
     566              :       CHARACTER(len=*), PARAMETER :: routineN = 'pao_write_ks_matrix_csr'
     567              : 
     568              :       CHARACTER(LEN=default_path_length)                 :: file_name, fileformat
     569              :       INTEGER                                            :: handle, ispin, output_unit, unit_nr
     570              :       LOGICAL                                            :: bin, do_kpoints, do_ks_csr_write, uptr
     571              :       REAL(KIND=dp)                                      :: thld
     572              :       TYPE(cp_logger_type), POINTER                      :: logger
     573              :       TYPE(dbcsr_csr_type)                               :: ks_mat_csr
     574              :       TYPE(dbcsr_type)                                   :: matrix_ks_nosym
     575              :       TYPE(section_vals_type), POINTER                   :: dft_section, input
     576              : 
     577          294 :       CALL timeset(routineN, handle)
     578              : 
     579          294 :       NULLIFY (dft_section)
     580              : 
     581          294 :       logger => cp_get_default_logger()
     582          294 :       output_unit = cp_logger_get_default_io_unit(logger)
     583              : 
     584          294 :       CALL get_qs_env(qs_env, input=input)
     585          294 :       dft_section => section_vals_get_subs_vals(input, "DFT")
     586              :       do_ks_csr_write = BTEST(cp_print_key_should_output(logger%iter_info, dft_section, &
     587          294 :                                                          "PRINT%KS_CSR_WRITE"), cp_p_file)
     588              : 
     589              :       ! NOTE: k-points has to be treated differently later. k-points has KS matrix as double pointer.
     590          294 :       CALL get_qs_env(qs_env=qs_env, do_kpoints=do_kpoints)
     591              : 
     592          294 :       IF (do_ks_csr_write .AND. (.NOT. do_kpoints)) THEN
     593            0 :          CALL section_vals_val_get(dft_section, "PRINT%KS_CSR_WRITE%THRESHOLD", r_val=thld)
     594            0 :          CALL section_vals_val_get(dft_section, "PRINT%KS_CSR_WRITE%UPPER_TRIANGULAR", l_val=uptr)
     595            0 :          CALL section_vals_val_get(dft_section, "PRINT%KS_CSR_WRITE%BINARY", l_val=bin)
     596              : 
     597            0 :          IF (bin) THEN
     598            0 :             fileformat = "UNFORMATTED"
     599              :          ELSE
     600            0 :             fileformat = "FORMATTED"
     601              :          END IF
     602              : 
     603            0 :          DO ispin = 1, SIZE(ls_scf_env%matrix_ks)
     604              : 
     605            0 :             IF (dbcsr_has_symmetry(ls_scf_env%matrix_ks(ispin))) THEN
     606            0 :                CALL dbcsr_desymmetrize(ls_scf_env%matrix_ks(ispin), matrix_ks_nosym)
     607              :             ELSE
     608            0 :                CALL dbcsr_copy(matrix_ks_nosym, ls_scf_env%matrix_ks(ispin))
     609              :             END IF
     610              : 
     611            0 :             CALL dbcsr_csr_create_from_dbcsr(matrix_ks_nosym, ks_mat_csr, dbcsr_csr_dbcsr_blkrow_dist)
     612            0 :             CALL dbcsr_convert_dbcsr_to_csr(matrix_ks_nosym, ks_mat_csr)
     613              : 
     614            0 :             WRITE (file_name, '(A,I0)') "PAO_KS_SPIN_", ispin
     615              :             unit_nr = cp_print_key_unit_nr(logger, dft_section, "PRINT%KS_CSR_WRITE", &
     616              :                                            extension=".csr", middle_name=TRIM(file_name), &
     617            0 :                                            file_status="REPLACE", file_form=fileformat)
     618            0 :             CALL dbcsr_csr_write(ks_mat_csr, unit_nr, upper_triangle=uptr, threshold=thld, binary=bin)
     619              : 
     620            0 :             CALL cp_print_key_finished_output(unit_nr, logger, dft_section, "PRINT%KS_CSR_WRITE")
     621              : 
     622            0 :             CALL dbcsr_csr_destroy(ks_mat_csr)
     623            0 :             CALL dbcsr_release(matrix_ks_nosym)
     624              :          END DO
     625              :       END IF
     626              : 
     627          294 :       CALL timestop(handle)
     628              : 
     629          294 :    END SUBROUTINE pao_write_ks_matrix_csr
     630              : 
     631              : !**************************************************************************************************
     632              : !> \brief writing the overlap matrix (in terms of the PAO basis) in csr format into a file
     633              : !> \param qs_env qs environment
     634              : !> \param ls_scf_env ls environment
     635              : !> \author Mohammad Hossein Bani-Hashemian
     636              : ! **************************************************************************************************
     637          294 :    SUBROUTINE pao_write_s_matrix_csr(qs_env, ls_scf_env)
     638              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     639              :       TYPE(ls_scf_env_type), TARGET                      :: ls_scf_env
     640              : 
     641              :       CHARACTER(len=*), PARAMETER :: routineN = 'pao_write_s_matrix_csr'
     642              : 
     643              :       CHARACTER(LEN=default_path_length)                 :: file_name, fileformat
     644              :       INTEGER                                            :: handle, output_unit, unit_nr
     645              :       LOGICAL                                            :: bin, do_kpoints, do_s_csr_write, uptr
     646              :       REAL(KIND=dp)                                      :: thld
     647              :       TYPE(cp_logger_type), POINTER                      :: logger
     648              :       TYPE(dbcsr_csr_type)                               :: s_mat_csr
     649              :       TYPE(dbcsr_type)                                   :: matrix_s_nosym
     650              :       TYPE(section_vals_type), POINTER                   :: dft_section, input
     651              : 
     652          294 :       CALL timeset(routineN, handle)
     653              : 
     654          294 :       NULLIFY (dft_section)
     655              : 
     656          294 :       logger => cp_get_default_logger()
     657          294 :       output_unit = cp_logger_get_default_io_unit(logger)
     658              : 
     659          294 :       CALL get_qs_env(qs_env, input=input)
     660          294 :       dft_section => section_vals_get_subs_vals(input, "DFT")
     661              :       do_s_csr_write = BTEST(cp_print_key_should_output(logger%iter_info, dft_section, &
     662          294 :                                                         "PRINT%S_CSR_WRITE"), cp_p_file)
     663              : 
     664              :       ! NOTE: k-points has to be treated differently later. k-points has overlap matrix as double pointer.
     665          294 :       CALL get_qs_env(qs_env=qs_env, do_kpoints=do_kpoints)
     666              : 
     667          294 :       IF (do_s_csr_write .AND. (.NOT. do_kpoints)) THEN
     668            0 :          CALL section_vals_val_get(dft_section, "PRINT%S_CSR_WRITE%THRESHOLD", r_val=thld)
     669            0 :          CALL section_vals_val_get(dft_section, "PRINT%S_CSR_WRITE%UPPER_TRIANGULAR", l_val=uptr)
     670            0 :          CALL section_vals_val_get(dft_section, "PRINT%S_CSR_WRITE%BINARY", l_val=bin)
     671              : 
     672            0 :          IF (bin) THEN
     673            0 :             fileformat = "UNFORMATTED"
     674              :          ELSE
     675            0 :             fileformat = "FORMATTED"
     676              :          END IF
     677              : 
     678            0 :          IF (dbcsr_has_symmetry(ls_scf_env%matrix_s)) THEN
     679            0 :             CALL dbcsr_desymmetrize(ls_scf_env%matrix_s, matrix_s_nosym)
     680              :          ELSE
     681            0 :             CALL dbcsr_copy(matrix_s_nosym, ls_scf_env%matrix_s)
     682              :          END IF
     683              : 
     684            0 :          CALL dbcsr_csr_create_from_dbcsr(matrix_s_nosym, s_mat_csr, dbcsr_csr_dbcsr_blkrow_dist)
     685            0 :          CALL dbcsr_convert_dbcsr_to_csr(matrix_s_nosym, s_mat_csr)
     686              : 
     687            0 :          WRITE (file_name, '(A,I0)') "PAO_S"
     688              :          unit_nr = cp_print_key_unit_nr(logger, dft_section, "PRINT%S_CSR_WRITE", &
     689              :                                         extension=".csr", middle_name=TRIM(file_name), &
     690            0 :                                         file_status="REPLACE", file_form=fileformat)
     691            0 :          CALL dbcsr_csr_write(s_mat_csr, unit_nr, upper_triangle=uptr, threshold=thld, binary=bin)
     692              : 
     693            0 :          CALL cp_print_key_finished_output(unit_nr, logger, dft_section, "PRINT%S_CSR_WRITE")
     694              : 
     695            0 :          CALL dbcsr_csr_destroy(s_mat_csr)
     696            0 :          CALL dbcsr_release(matrix_s_nosym)
     697              :       END IF
     698              : 
     699          294 :       CALL timestop(handle)
     700              : 
     701          294 :    END SUBROUTINE pao_write_s_matrix_csr
     702              : 
     703              : !**************************************************************************************************
     704              : !> \brief writing the core Hamiltonian matrix (NYA)
     705              : !> \param qs_env qs environment
     706              : !> \param ls_scf_env ls environment
     707              : !> \author Mohammad Hossein Bani-Hashemian
     708              : ! **************************************************************************************************
     709          294 :    SUBROUTINE pao_write_hcore_matrix_csr(qs_env, ls_scf_env)
     710              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     711              :       TYPE(ls_scf_env_type), TARGET                      :: ls_scf_env
     712              : 
     713              :       CHARACTER(len=*), PARAMETER :: routineN = 'pao_write_hcore_matrix_csr'
     714              : 
     715              :       INTEGER                                            :: handle, output_unit
     716              :       LOGICAL                                            :: do_h_csr_write, do_kpoints
     717              :       TYPE(cp_logger_type), POINTER                      :: logger
     718              :       TYPE(section_vals_type), POINTER                   :: dft_section, input
     719              : 
     720              :       MARK_USED(ls_scf_env)
     721              : 
     722          294 :       CALL timeset(routineN, handle)
     723              : 
     724          294 :       NULLIFY (dft_section)
     725              : 
     726          294 :       logger => cp_get_default_logger()
     727          294 :       output_unit = cp_logger_get_default_io_unit(logger)
     728              : 
     729          294 :       CALL get_qs_env(qs_env, input=input)
     730          294 :       dft_section => section_vals_get_subs_vals(input, "DFT")
     731              :       do_h_csr_write = BTEST(cp_print_key_should_output(logger%iter_info, dft_section, &
     732          294 :                                                         "PRINT%HCORE_CSR_WRITE"), cp_p_file)
     733              : 
     734              :       ! NOTE: k-points has to be treated differently later. k-points has KS matrix as double pointer.
     735          294 :       CALL get_qs_env(qs_env=qs_env, do_kpoints=do_kpoints)
     736              : 
     737          294 :       IF (do_h_csr_write .AND. (.NOT. do_kpoints)) THEN
     738            0 :          CALL cp_warn(__LOCATION__, "Writing the PAO Core Hamiltonian matrix in CSR format NYA")
     739              :       END IF
     740              : 
     741          294 :       CALL timestop(handle)
     742              : 
     743          294 :    END SUBROUTINE pao_write_hcore_matrix_csr
     744              : 
     745              : !**************************************************************************************************
     746              : !> \brief writing the density matrix (NYA)
     747              : !> \param qs_env qs environment
     748              : !> \param ls_scf_env ls environment
     749              : !> \author Mohammad Hossein Bani-Hashemian
     750              : ! **************************************************************************************************
     751          294 :    SUBROUTINE pao_write_p_matrix_csr(qs_env, ls_scf_env)
     752              :       TYPE(qs_environment_type), POINTER                 :: qs_env
     753              :       TYPE(ls_scf_env_type), TARGET                      :: ls_scf_env
     754              : 
     755              :       CHARACTER(len=*), PARAMETER :: routineN = 'pao_write_p_matrix_csr'
     756              : 
     757              :       INTEGER                                            :: handle, output_unit
     758              :       LOGICAL                                            :: do_kpoints, do_p_csr_write
     759              :       TYPE(cp_logger_type), POINTER                      :: logger
     760              :       TYPE(section_vals_type), POINTER                   :: dft_section, input
     761              : 
     762              :       MARK_USED(ls_scf_env)
     763              : 
     764          294 :       CALL timeset(routineN, handle)
     765              : 
     766          294 :       NULLIFY (dft_section)
     767              : 
     768          294 :       logger => cp_get_default_logger()
     769          294 :       output_unit = cp_logger_get_default_io_unit(logger)
     770              : 
     771          294 :       CALL get_qs_env(qs_env, input=input)
     772          294 :       dft_section => section_vals_get_subs_vals(input, "DFT")
     773              :       do_p_csr_write = BTEST(cp_print_key_should_output(logger%iter_info, dft_section, &
     774          294 :                                                         "PRINT%P_CSR_WRITE"), cp_p_file)
     775              : 
     776              :       ! NOTE: k-points has to be treated differently later. k-points has KS matrix as double pointer.
     777          294 :       CALL get_qs_env(qs_env=qs_env, do_kpoints=do_kpoints)
     778              : 
     779          294 :       IF (do_p_csr_write .AND. (.NOT. do_kpoints)) THEN
     780            0 :          CALL cp_warn(__LOCATION__, "Writing the PAO density matrix in CSR format NYA")
     781              :       END IF
     782              : 
     783          294 :       CALL timestop(handle)
     784              : 
     785          294 :    END SUBROUTINE pao_write_p_matrix_csr
     786              : 
     787            0 : END MODULE pao_io
        

Generated by: LCOV version 2.0-1