LCOV - code coverage report
Current view: top level - src/motion - neb_io.F (source / functions) Coverage Total Hit
Test: CP2K Regtests (git:85b8a9b) Lines: 82.1 % 307 252
Test Date: 2026-06-14 06:48:14 Functions: 100.0 % 7 7

            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              : !> \brief I/O Module for Nudged Elastic Band Calculation
      10              : !> \note
      11              : !>      Numerical accuracy for parallel runs:
      12              : !>       Each replica starts the SCF run from the one optimized
      13              : !>       in a previous run. It may happen then energies and derivatives
      14              : !>       of a serial run and a parallel run could be slightly different
      15              : !>       'cause of a different starting density matrix.
      16              : !>       Exact results are obtained using:
      17              : !>          EXTRAPOLATION USE_GUESS in QS section (Teo 09.2006)
      18              : !> \author Teodoro Laino 10.2006
      19              : ! **************************************************************************************************
      20              : MODULE neb_io
      21              :    USE cell_types,                      ONLY: cell_type
      22              :    USE cp2k_info,                       ONLY: get_runtime_info
      23              :    USE cp_files,                        ONLY: close_file,&
      24              :                                               open_file
      25              :    USE cp_log_handling,                 ONLY: cp_add_default_logger,&
      26              :                                               cp_get_default_logger,&
      27              :                                               cp_logger_type,&
      28              :                                               cp_rm_default_logger,&
      29              :                                               cp_to_string
      30              :    USE cp_output_handling,              ONLY: cp_print_key_finished_output,&
      31              :                                               cp_print_key_generate_filename,&
      32              :                                               cp_print_key_unit_nr
      33              :    USE cp_units,                        ONLY: cp_unit_from_cp2k
      34              :    USE f77_interface,                   ONLY: f_env_add_defaults,&
      35              :                                               f_env_rm_defaults,&
      36              :                                               f_env_type
      37              :    USE force_env_types,                 ONLY: force_env_get,&
      38              :                                               use_mixed_force
      39              :    USE header,                          ONLY: cp2k_footer
      40              :    USE input_constants,                 ONLY: band_md_opt,&
      41              :                                               do_sm,&
      42              :                                               dump_extxyz,&
      43              :                                               dump_xmol,&
      44              :                                               pot_neb_fe,&
      45              :                                               pot_neb_full,&
      46              :                                               pot_neb_me
      47              :    USE input_cp2k_neb,                  ONLY: create_band_section
      48              :    USE input_cp2k_restarts,             ONLY: write_restart
      49              :    USE input_enumeration_types,         ONLY: enum_i2c,&
      50              :                                               enumeration_type
      51              :    USE input_keyword_types,             ONLY: keyword_get,&
      52              :                                               keyword_type
      53              :    USE input_section_types,             ONLY: section_get_keyword,&
      54              :                                               section_release,&
      55              :                                               section_type,&
      56              :                                               section_vals_get,&
      57              :                                               section_vals_get_subs_vals,&
      58              :                                               section_vals_type,&
      59              :                                               section_vals_val_get,&
      60              :                                               section_vals_val_set
      61              :    USE kinds,                           ONLY: default_path_length,&
      62              :                                               default_string_length,&
      63              :                                               dp
      64              :    USE machine,                         ONLY: m_flush
      65              :    USE neb_md_utils,                    ONLY: get_temperatures
      66              :    USE neb_types,                       ONLY: neb_type,&
      67              :                                               neb_var_type
      68              :    USE particle_methods,                ONLY: write_particle_coordinates
      69              :    USE particle_types,                  ONLY: get_particle_pos_or_vel,&
      70              :                                               particle_type
      71              :    USE physcon,                         ONLY: angstrom
      72              :    USE replica_types,                   ONLY: replica_env_type
      73              : #include "../base/base_uses.f90"
      74              : 
      75              :    IMPLICIT NONE
      76              :    PRIVATE
      77              :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'neb_io'
      78              : 
      79              :    PUBLIC :: read_neb_section, &
      80              :              dump_neb_final, &
      81              :              dump_neb_info, &
      82              :              dump_replica_coordinates, &
      83              :              handle_band_file_names, &
      84              :              neb_rep_env_map_info
      85              : 
      86              : CONTAINS
      87              : 
      88              : ! **************************************************************************************************
      89              : !> \brief Read data from the NEB input section
      90              : !> \param neb_env ...
      91              : !> \param neb_section ...
      92              : !> \author Teodoro Laino 09.2006
      93              : ! **************************************************************************************************
      94           34 :    SUBROUTINE read_neb_section(neb_env, neb_section)
      95              :       TYPE(neb_type), POINTER                            :: neb_env
      96              :       TYPE(section_vals_type), POINTER                   :: neb_section
      97              : 
      98              :       LOGICAL                                            :: explicit
      99              :       TYPE(section_vals_type), POINTER                   :: wrk_section
     100              : 
     101           34 :       CPASSERT(ASSOCIATED(neb_env))
     102           34 :       neb_env%istep = 0
     103           34 :       CALL section_vals_val_get(neb_section, "BAND_TYPE", i_val=neb_env%id_type)
     104           34 :       CALL section_vals_val_get(neb_section, "NUMBER_OF_REPLICA", i_val=neb_env%number_of_replica)
     105           34 :       CALL section_vals_val_get(neb_section, "K_SPRING", r_val=neb_env%K)
     106           34 :       CALL section_vals_val_get(neb_section, "ROTATE_FRAMES", l_val=neb_env%rotate_frames)
     107           34 :       CALL section_vals_val_get(neb_section, "ALIGN_FRAMES", l_val=neb_env%align_frames)
     108           34 :       CALL section_vals_val_get(neb_section, "OPTIMIZE_BAND%OPTIMIZE_END_POINTS", l_val=neb_env%optimize_end_points)
     109              :       ! Climb Image NEB
     110           34 :       CALL section_vals_val_get(neb_section, "CI_NEB%NSTEPS_IT", i_val=neb_env%nsteps_it)
     111              :       ! Band Optimization Type
     112           34 :       CALL section_vals_val_get(neb_section, "OPTIMIZE_BAND%OPT_TYPE", i_val=neb_env%opt_type)
     113              :       ! Use colvars
     114           34 :       CALL section_vals_val_get(neb_section, "USE_COLVARS", l_val=neb_env%use_colvar)
     115           34 :       CALL section_vals_val_get(neb_section, "POT_TYPE", i_val=neb_env%pot_type)
     116              :       ! Before continuing let's do some consistency check between keywords
     117           34 :       IF (neb_env%pot_type /= pot_neb_full) THEN
     118              :          ! Requires the use of colvars
     119            4 :          IF (.NOT. neb_env%use_colvar) &
     120              :             CALL cp_abort(__LOCATION__, &
     121              :                           "A potential energy function based on free energy or minimum energy"// &
     122              :                           " was requested without enabling the usage of COLVARS. Both methods"// &
     123            0 :                           " are based on COLVARS definition.")
     124              :          ! Moreover let's check if the proper sections have been defined..
     125            4 :          SELECT CASE (neb_env%pot_type)
     126              :          CASE (pot_neb_fe)
     127            0 :             wrk_section => section_vals_get_subs_vals(neb_env%root_section, "MOTION%MD")
     128            0 :             CALL section_vals_get(wrk_section, explicit=explicit)
     129            0 :             IF (.NOT. explicit) &
     130              :                CALL cp_abort(__LOCATION__, &
     131              :                              "A free energy BAND (colvars projected) calculation is requested"// &
     132            0 :                              " but NONE MD section was defined in the input.")
     133              :          CASE (pot_neb_me)
     134            4 :             wrk_section => section_vals_get_subs_vals(neb_env%root_section, "MOTION%GEO_OPT")
     135            4 :             CALL section_vals_get(wrk_section, explicit=explicit)
     136            4 :             IF (.NOT. explicit) &
     137              :                CALL cp_abort(__LOCATION__, &
     138              :                              "A minimum energy BAND (colvars projected) calculation is requested"// &
     139            8 :                              " but NONE GEO_OPT section was defined in the input.")
     140              :          END SELECT
     141              :       ELSE
     142           30 :          IF (neb_env%use_colvar) &
     143              :             CALL cp_abort(__LOCATION__, &
     144              :                           "A band calculation was requested with a full potential energy. USE_COLVAR cannot"// &
     145            0 :                           " be set for this kind of calculation!")
     146              :       END IF
     147              :       ! String Method
     148           34 :       CALL section_vals_val_get(neb_section, "STRING_METHOD%SMOOTHING", r_val=neb_env%smoothing)
     149           34 :       CALL section_vals_val_get(neb_section, "STRING_METHOD%SPLINE_ORDER", i_val=neb_env%spline_order)
     150           34 :       neb_env%reparametrize_frames = .FALSE.
     151           34 :       IF (neb_env%id_type == do_sm) THEN
     152            2 :          neb_env%reparametrize_frames = .TRUE.
     153              :       END IF
     154           34 :    END SUBROUTINE read_neb_section
     155              : 
     156              : ! **************************************************************************************************
     157              : !> \brief dump final structures after a NEB run
     158              : !> \param neb_env ...
     159              : !> \param energies ...
     160              : !> \param coords ...
     161              : !> \param particle_set ...
     162              : !> \param logger ...
     163              : !> \param output_unit ...
     164              : !> \param converged ...
     165              : !> \par
     166              : !>          History
     167              : !>          06.2026 - Created
     168              : !> \author  HE Zilong
     169              : !> \version 1.0
     170              : ! **************************************************************************************************
     171           34 :    SUBROUTINE dump_neb_final(neb_env, energies, coords, particle_set, logger, output_unit, converged)
     172              :       TYPE(neb_type), POINTER                            :: neb_env
     173              :       REAL(KIND=dp), DIMENSION(:), INTENT(IN)            :: energies
     174              :       TYPE(neb_var_type), POINTER                        :: coords
     175              :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     176              :       TYPE(cp_logger_type), POINTER                      :: logger
     177              :       INTEGER, INTENT(IN)                                :: output_unit
     178              :       LOGICAL                                            :: converged
     179              : 
     180              :       CHARACTER(len=*), PARAMETER                        :: routineN = 'dump_neb_final'
     181              : 
     182              :       CHARACTER(LEN=1024)                                :: cell_str, ener_str, lm_str, record, &
     183              :                                                             replica_str, title
     184              :       CHARACTER(LEN=4)                                   :: l_ener
     185              :       CHARACTER(LEN=5)                                   :: pbc_str
     186              :       INTEGER                                            :: irep, iw
     187              :       LOGICAL                                            :: print_kind
     188              :       REAL(KIND=dp)                                      :: unit_conv
     189              :       TYPE(cell_type), POINTER                           :: cell
     190              :       TYPE(section_vals_type), POINTER                   :: final_band_section
     191              : 
     192           34 :       NULLIFY (final_band_section)
     193           34 :       final_band_section => section_vals_get_subs_vals(neb_env%neb_section, "FINAL_BAND")
     194           34 :       CALL force_env_get(neb_env%force_env, cell=cell) ! For now NEB has constant cell
     195           34 :       pbc_str = "F F F"
     196           34 :       IF (cell%perd(1) == 1) pbc_str(1:1) = "T"
     197           34 :       IF (cell%perd(2) == 1) pbc_str(3:3) = "T"
     198           34 :       IF (cell%perd(3) == 1) pbc_str(5:5) = "T"
     199              :       WRITE (UNIT=cell_str, FMT="(9(1X,F19.10))") &
     200          340 :          cell%hmat(:, 1)*angstrom, cell%hmat(:, 2)*angstrom, cell%hmat(:, 3)*angstrom
     201           34 :       unit_conv = cp_unit_from_cp2k(1.0_dp, "angstrom")
     202              : 
     203              :       ! Print a message to log
     204              :       record = cp_print_key_generate_filename(logger, final_band_section, &
     205              :                                               extension=".xyz", &
     206           34 :                                               my_local=.FALSE.)
     207           34 :       IF (output_unit > 0) THEN
     208           17 :          IF (converged) THEN
     209              :             WRITE (UNIT=output_unit, FMT="(/,T2,A)") &
     210            2 :                routineN//": Band task converged, writing XYZ trajectory gladly:"
     211              :          ELSE
     212              :             WRITE (UNIT=output_unit, FMT="(/,T2,A)") &
     213           15 :                routineN//": Band task not yet converged, writing XYZ trajectory anyway:"
     214              :          END IF
     215           17 :          WRITE (UNIT=output_unit, FMT="(T3,A)") TRIM(record)
     216              :       END IF
     217              : 
     218              :       ! Write actual trajectory file
     219              :       iw = cp_print_key_unit_nr(logger, neb_env%neb_section, "FINAL_BAND", &
     220           34 :                                 extension=".xyz", file_form="FORMATTED", file_status="REPLACE")
     221              :       CALL section_vals_val_get(neb_env%neb_section, "FINAL_BAND%PRINT_ATOM_KIND", &
     222           34 :                                 l_val=print_kind)
     223          246 :       DO irep = 1, neb_env%number_of_replica
     224          212 :          l_ener = "(**)"
     225          212 :          IF (irep > 1) THEN
     226          178 :             IF (energies(irep) - energies(irep - 1) > 0) THEN
     227          106 :                l_ener(2:2) = "+"
     228              :             ELSE
     229           72 :                l_ener(2:2) = "-"
     230              :             END IF
     231              :          END IF
     232          212 :          IF (irep < neb_env%number_of_replica) THEN
     233          178 :             IF (energies(irep + 1) - energies(irep) < 0) THEN
     234           72 :                l_ener(3:3) = "+"
     235              :             ELSE
     236          106 :                l_ener(3:3) = "-"
     237              :             END IF
     238              :          END IF
     239           46 :          SELECT CASE (l_ener)
     240              :          CASE ("(++)") ! local maximum
     241           46 :             WRITE (lm_str, '(A)') "Ener_loc_max=T Ener_loc_min=F"
     242              :          CASE ("(--)") ! local minimum
     243           34 :             WRITE (lm_str, '(A)') "Ener_loc_max=F Ener_loc_min=T"
     244              :          CASE DEFAULT
     245          212 :             WRITE (lm_str, '(A)') "Ener_loc_max=F Ener_loc_min=F"
     246              :          END SELECT
     247          212 :          WRITE (UNIT=replica_str, FMT="(I8)") irep
     248          212 :          WRITE (UNIT=ener_str, FMT="(F20.10)") energies(irep)
     249              :          WRITE (UNIT=title, FMT="(A)") &
     250              :             'Lattice="'//TRIM(ADJUSTL(cell_str))//'" '// &
     251              :             'Properties=species:S:1:pos:R:3 '// &
     252              :             'pbc="'//pbc_str//'" '// &
     253              :             'Replica='//TRIM(ADJUSTL(replica_str))//' '// &
     254              :             'Energy='//TRIM(ADJUSTL(ener_str))//' '// &
     255          212 :             TRIM(ADJUSTL(lm_str))
     256          246 :          IF (iw > 0) THEN
     257              :             ! The iw condition does not hold for certain ranks/processes
     258              :             ! that write to <proj>-BAND<n>.out where n > neb_env%number_of_replica
     259              :             CALL write_particle_coordinates(particle_set, iw, dump_extxyz, "POS", title, &
     260              :                                             cell=cell, array=coords%xyz(:, irep), unit_conv=unit_conv, &
     261          106 :                                             print_kind=print_kind)
     262          106 :             CALL m_flush(iw)
     263              :          END IF
     264              :       END DO
     265              : 
     266           34 :       IF (output_unit > 0) &
     267              :          WRITE (UNIT=output_unit, FMT='(/,T2,A)') &
     268           17 :          routineN//": Done!"
     269              : 
     270           34 :       CALL cp_print_key_finished_output(iw, logger, neb_env%neb_section, "FINAL_BAND")
     271              : 
     272           34 :    END SUBROUTINE dump_neb_final
     273              : 
     274              : ! **************************************************************************************************
     275              : !> \brief dump print info of a NEB run
     276              : !> \param neb_env ...
     277              : !> \param coords ...
     278              : !> \param vels ...
     279              : !> \param forces ...
     280              : !> \param particle_set ...
     281              : !> \param logger ...
     282              : !> \param istep ...
     283              : !> \param energies ...
     284              : !> \param distances ...
     285              : !> \param output_unit ...
     286              : !> \author Teodoro Laino 09.2006
     287              : ! **************************************************************************************************
     288          578 :    SUBROUTINE dump_neb_info(neb_env, coords, vels, forces, particle_set, logger, &
     289          578 :                             istep, energies, distances, output_unit)
     290              :       TYPE(neb_type), POINTER                            :: neb_env
     291              :       TYPE(neb_var_type), POINTER                        :: coords
     292              :       TYPE(neb_var_type), OPTIONAL, POINTER              :: vels, forces
     293              :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     294              :       TYPE(cp_logger_type), POINTER                      :: logger
     295              :       INTEGER, INTENT(IN)                                :: istep
     296              :       REAL(KIND=dp), DIMENSION(:), INTENT(IN)            :: energies, distances
     297              :       INTEGER, INTENT(IN)                                :: output_unit
     298              : 
     299              :       CHARACTER(len=*), PARAMETER                        :: routineN = 'dump_neb_info'
     300              : 
     301              :       CHARACTER(LEN=20)                                  :: mytype
     302              :       CHARACTER(LEN=4)                                   :: l_ener
     303              :       CHARACTER(LEN=default_string_length)               :: line, title, unit_str
     304              :       INTEGER                                            :: crd, ener, frc, handle, i, irep, n_max, &
     305              :                                                             n_min, ndig, ndigl, plt, ttst, vel
     306              :       LOGICAL                                            :: explicit, lval, plot_rel_energy, &
     307              :                                                             print_kind
     308              :       REAL(KIND=dp)                                      :: ener_min, ener_range, f_ann, tmp_r1, &
     309              :                                                             unit_conv
     310          578 :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:)           :: ekin, temperatures
     311              :       TYPE(cell_type), POINTER                           :: cell
     312              :       TYPE(enumeration_type), POINTER                    :: enum
     313              :       TYPE(keyword_type), POINTER                        :: keyword
     314              :       TYPE(section_type), POINTER                        :: section
     315              :       TYPE(section_vals_type), POINTER                   :: run_info_section, tc_section, vc_section
     316              : 
     317          578 :       CALL timeset(routineN, handle)
     318          578 :       ndig = CEILING(LOG10(REAL(neb_env%number_of_replica + 1, KIND=dp)))
     319          578 :       CALL force_env_get(neb_env%force_env, cell=cell)
     320         4152 :       DO irep = 1, neb_env%number_of_replica
     321         3574 :          ndigl = CEILING(LOG10(REAL(irep + 1, KIND=dp)))
     322         3574 :          WRITE (line, '(A,'//cp_to_string(ndig)//'("0"),T'//cp_to_string(11 + ndig + 1 - ndigl)//',I0)') "Replica_nr_", irep
     323              :          crd = cp_print_key_unit_nr(logger, neb_env%motion_print_section, "TRAJECTORY", &
     324         3574 :                                     extension=".xyz", file_form="FORMATTED", middle_name="pos-"//TRIM(line))
     325         3574 :          IF (PRESENT(vels)) THEN
     326              :             vel = cp_print_key_unit_nr(logger, neb_env%motion_print_section, "VELOCITIES", &
     327         3574 :                                        extension=".xyz", file_form="FORMATTED", middle_name="vel-"//TRIM(line))
     328              :          END IF
     329         3574 :          IF (PRESENT(forces)) THEN
     330              :             frc = cp_print_key_unit_nr(logger, neb_env%motion_print_section, "FORCES", &
     331         3574 :                                        extension=".xyz", file_form="FORMATTED", middle_name="force-"//TRIM(line))
     332              :          END IF
     333              :          ! Dump Trajectory
     334         3574 :          IF (crd > 0) THEN
     335              :             ! Gather units of measure for output
     336              :             CALL section_vals_val_get(neb_env%motion_print_section, "TRAJECTORY%UNIT", &
     337         1565 :                                       c_val=unit_str)
     338              :             CALL section_vals_val_get(neb_env%motion_print_section, "TRAJECTORY%PRINT_ATOM_KIND", &
     339         1565 :                                       l_val=print_kind)
     340         1565 :             unit_conv = cp_unit_from_cp2k(1.0_dp, TRIM(unit_str))
     341              :             ! This information can be digested by Molden
     342         1565 :             WRITE (UNIT=title, FMT="(A,I8,A,F20.10)") " i =", istep, ", E =", energies(irep)
     343              :             CALL write_particle_coordinates(particle_set, crd, dump_xmol, "POS", title, &
     344              :                                             cell=cell, array=coords%xyz(:, irep), unit_conv=unit_conv, &
     345         1565 :                                             print_kind=print_kind)
     346         1565 :             CALL m_flush(crd)
     347              :          END IF
     348              :          ! Dump Velocities
     349         3574 :          IF (vel > 0 .AND. PRESENT(vels)) THEN
     350              :             ! Gather units of measure for output
     351              :             CALL section_vals_val_get(neb_env%motion_print_section, "VELOCITIES%UNIT", &
     352            0 :                                       c_val=unit_str)
     353              :             CALL section_vals_val_get(neb_env%motion_print_section, "VELOCITIES%PRINT_ATOM_KIND", &
     354            0 :                                       l_val=print_kind)
     355            0 :             unit_conv = cp_unit_from_cp2k(1.0_dp, TRIM(unit_str))
     356            0 :             WRITE (UNIT=title, FMT="(A,I8,A,F20.10)") " i =", istep, ", E =", energies(irep)
     357              :             CALL write_particle_coordinates(particle_set, vel, dump_xmol, "VEL", title, &
     358              :                                             cell=cell, array=vels%xyz(:, irep), unit_conv=unit_conv, &
     359            0 :                                             print_kind=print_kind)
     360            0 :             CALL m_flush(vel)
     361              :          END IF
     362              :          ! Dump Forces
     363         3574 :          IF (frc > 0 .AND. PRESENT(forces)) THEN
     364              :             ! Gather units of measure for output
     365              :             CALL section_vals_val_get(neb_env%motion_print_section, "FORCES%UNIT", &
     366            0 :                                       c_val=unit_str)
     367              :             CALL section_vals_val_get(neb_env%motion_print_section, "FORCES%PRINT_ATOM_KIND", &
     368            0 :                                       l_val=print_kind)
     369            0 :             unit_conv = cp_unit_from_cp2k(1.0_dp, TRIM(unit_str))
     370            0 :             WRITE (UNIT=title, FMT="(A,I8,A,F20.10)") " i =", istep, ", E =", energies(irep)
     371              :             CALL write_particle_coordinates(particle_set, frc, dump_xmol, "FRC", title, &
     372              :                                             cell=cell, array=forces%xyz(:, irep), unit_conv=unit_conv, &
     373            0 :                                             print_kind=print_kind)
     374            0 :             CALL m_flush(frc)
     375              :          END IF
     376              :          CALL cp_print_key_finished_output(crd, logger, neb_env%motion_print_section, &
     377         3574 :                                            "TRAJECTORY")
     378         3574 :          IF (PRESENT(vels)) THEN
     379              :             CALL cp_print_key_finished_output(vel, logger, neb_env%motion_print_section, &
     380         3574 :                                               "VELOCITIES")
     381              :          END IF
     382         4152 :          IF (PRESENT(forces)) THEN
     383              :             CALL cp_print_key_finished_output(frc, logger, neb_env%motion_print_section, &
     384         3574 :                                               "FORCES")
     385              :          END IF
     386              :       END DO
     387              :       ! NEB summary info on screen
     388          578 :       IF (output_unit > 0) THEN
     389          289 :          tc_section => section_vals_get_subs_vals(neb_env%neb_section, "OPTIMIZE_BAND%MD%TEMP_CONTROL")
     390          289 :          vc_section => section_vals_get_subs_vals(neb_env%neb_section, "OPTIMIZE_BAND%MD%VEL_CONTROL")
     391          289 :          run_info_section => section_vals_get_subs_vals(neb_env%neb_section, "PROGRAM_RUN_INFO")
     392          289 :          CALL section_vals_val_get(run_info_section, "PLOT_REL_ENERGY", l_val=plot_rel_energy)
     393          867 :          ALLOCATE (temperatures(neb_env%number_of_replica))
     394          578 :          ALLOCATE (ekin(neb_env%number_of_replica))
     395          289 :          CALL get_temperatures(vels, particle_set, temperatures, ekin=ekin)
     396          289 :          WRITE (output_unit, '(/)', ADVANCE="NO")
     397          289 :          WRITE (output_unit, FMT='(A,A)') ' **************************************', &
     398          578 :             '*****************************************'
     399          289 :          NULLIFY (section, keyword, enum)
     400          289 :          CALL create_band_section(section)
     401          289 :          keyword => section_get_keyword(section, "BAND_TYPE")
     402          289 :          CALL keyword_get(keyword, enum=enum)
     403          289 :          mytype = TRIM(enum_i2c(enum, neb_env%id_type))
     404              :          WRITE (output_unit, FMT='(A,T61,A)') &
     405          289 :             ' BAND TYPE                     =', ADJUSTR(mytype)
     406          289 :          CALL section_release(section)
     407              :          WRITE (output_unit, FMT='(A,T61,A)') &
     408          289 :             ' BAND TYPE OPTIMIZATION        =', ADJUSTR(neb_env%opt_type_label(1:20))
     409              :          WRITE (output_unit, '( A,T71,I10 )') &
     410          289 :             ' STEP NUMBER                   =', istep
     411          289 :          IF (neb_env%rotate_frames) WRITE (output_unit, '( A,T71,L10 )') &
     412           80 :             ' RMSD DISTANCE DEFINITION      =', neb_env%rotate_frames
     413              :          ! velocity control parameters output
     414          289 :          CALL section_vals_get(vc_section, explicit=explicit)
     415          289 :          IF (explicit) THEN
     416           88 :             CALL section_vals_val_get(vc_section, "PROJ_VELOCITY_VERLET", l_val=lval)
     417           88 :             IF (lval) WRITE (output_unit, '( A,T71,L10 )') &
     418           77 :                ' PROJECTED VELOCITY VERLET     =', lval
     419           88 :             CALL section_vals_val_get(vc_section, "SD_LIKE", l_val=lval)
     420           88 :             IF (lval) WRITE (output_unit, '( A,T71,L10)') &
     421            0 :                ' STEEPEST DESCENT LIKE         =', lval
     422           88 :             CALL section_vals_val_get(vc_section, "ANNEALING", r_val=f_ann)
     423           88 :             IF (f_ann /= 1.0_dp) THEN
     424              :                WRITE (output_unit, '( A,T71,F10.5)') &
     425           88 :                   ' ANNEALING FACTOR              = ', f_ann
     426              :             END IF
     427              :          END IF
     428              :          ! temperature control parameters output
     429          289 :          CALL section_vals_get(tc_section, explicit=explicit)
     430          289 :          IF (explicit) THEN
     431           32 :             CALL section_vals_val_get(tc_section, "TEMP_TOL_STEPS", i_val=ttst)
     432           32 :             IF (istep <= ttst) THEN
     433           22 :                CALL section_vals_val_get(tc_section, "TEMPERATURE", r_val=f_ann)
     434           22 :                tmp_r1 = cp_unit_from_cp2k(f_ann, "K")
     435              :                WRITE (output_unit, '( A,T71,F10.5)') &
     436           22 :                   ' TEMPERATURE TARGET            =', tmp_r1
     437              :             END IF
     438              :          END IF
     439              :          WRITE (output_unit, '( A,T71,I10 )') &
     440          289 :             ' NUMBER OF NEB REPLICA         =', neb_env%number_of_replica
     441              :          ! switch between a longer visual format and a compact data-only print format
     442          289 :          IF (plot_rel_energy) THEN
     443            0 :             CPASSERT(SIZE(distances) == neb_env%number_of_replica - 1)
     444            0 :             CPASSERT(SIZE(energies) == neb_env%number_of_replica)
     445            0 :             CPASSERT(SIZE(temperatures) == neb_env%number_of_replica)
     446            0 :             ener_min = MINVAL(energies(:))
     447            0 :             ener_range = MAXVAL(energies(:)) - ener_min
     448            0 :             n_max = 0
     449            0 :             n_min = 0
     450              :             WRITE (output_unit, '(T2,A,T22,A,T35,A,T52,A)') &
     451            0 :                'REPLICA', 'ENERGY [au]', 'TEMPERATURE [K]', 'o-------------------------> E'
     452            0 :             DO i = 1, SIZE(distances)
     453            0 :                plt = FLOOR((energies(i) - ener_min)/ener_range*25)
     454            0 :                l_ener = "(**)"
     455            0 :                IF (i > 1) THEN
     456            0 :                   IF (energies(i) - energies(i - 1) > 0) THEN
     457            0 :                      l_ener(2:2) = "+"
     458              :                   ELSE
     459            0 :                      l_ener(2:2) = "-"
     460              :                   END IF
     461              :                END IF
     462            0 :                IF (energies(i + 1) - energies(i) < 0) THEN
     463            0 :                   l_ener(3:3) = "+"
     464              :                ELSE
     465            0 :                   l_ener(3:3) = "-"
     466              :                END IF
     467            0 :                SELECT CASE (l_ener)
     468              :                CASE ("(++)") ! local maximum
     469            0 :                   n_max = n_max + 1
     470            0 :                   WRITE (line, '(A,A,A)') "|", REPEAT(" ", plt), "X"
     471              :                CASE ("(--)") ! local minimum
     472            0 :                   n_min = n_min + 1
     473            0 :                   WRITE (line, '(A,A,A)') "|", REPEAT(" ", plt), "x"
     474              :                CASE DEFAULT
     475            0 :                   WRITE (line, '(A,A,A)') "|", REPEAT(" ", plt), "O"
     476              :                END SELECT
     477              :                WRITE (output_unit, '(T2,I7,T10,F18.8,1X,A,T34,F16.6,T52,A)') &
     478            0 :                   i, energies(i), l_ener, temperatures(i), TRIM(line)
     479              :                WRITE (output_unit, '(T2,A,1X,F16.6,T52,A)') &
     480            0 :                   "DISTANCE = ", distances(i), "|"
     481              :             END DO
     482            0 :             plt = FLOOR((energies(neb_env%number_of_replica) - ener_min)/ener_range*25)
     483            0 :             l_ener = "(**)"
     484            0 :             IF (energies(neb_env%number_of_replica) - energies(neb_env%number_of_replica - 1) > 0) THEN
     485            0 :                l_ener(2:2) = "+"
     486              :             ELSE
     487            0 :                l_ener(2:2) = "-"
     488              :             END IF
     489              :             ! The last point would not be local maximum or minimum, as is the first
     490            0 :             WRITE (line, '(A,A,A)') "|", REPEAT(" ", plt), "O"
     491              :             WRITE (output_unit, '(T2,I7,T10,F18.8,1X,A,T34,F16.6,T52,A)') &
     492            0 :                neb_env%number_of_replica, energies(neb_env%number_of_replica), &
     493            0 :                l_ener, temperatures(neb_env%number_of_replica), TRIM(line)
     494            0 :             WRITE (output_unit, '(T52,A)') "v Nr."
     495              :             WRITE (output_unit, '(T2,A,T44,2(1X,I4))') &
     496            0 :                "NUMBER OF LOCAL MAXIMA (X) and MINIMA (x):", n_max, n_min
     497              :          ELSE
     498              :             WRITE (output_unit, '( A,T17,4F16.6)') &
     499          289 :                ' DISTANCES REP =', distances(1:MIN(4, SIZE(distances)))
     500          289 :             IF (SIZE(distances) > 4) THEN
     501           74 :                WRITE (output_unit, '( T17,4F16.6)') distances(5:SIZE(distances))
     502              :             END IF
     503              :             WRITE (output_unit, '( A,T17,4F16.6)') &
     504          289 :                ' ENERGIES [au] =', energies(1:MIN(4, SIZE(energies)))
     505          289 :             IF (SIZE(energies) > 4) THEN
     506          198 :                WRITE (output_unit, '( T17,4F16.6)') energies(5:SIZE(energies))
     507              :             END IF
     508          289 :             IF (neb_env%opt_type == band_md_opt) THEN
     509              :                WRITE (output_unit, '( A,T33,4(1X,F11.5))') &
     510           88 :                   ' REPLICA TEMPERATURES (K)      =', temperatures(1:MIN(4, SIZE(temperatures)))
     511          187 :                DO i = 5, SIZE(temperatures), 4
     512              :                   WRITE (output_unit, '( T33,4(1X,F11.5))') &
     513          187 :                      temperatures(i:MIN(i + 3, SIZE(temperatures)))
     514              :                END DO
     515              :             END IF
     516              :          END IF
     517              :          WRITE (output_unit, '( A,T56,F25.14)') &
     518          289 :             ' BAND TOTAL ENERGY [au]        =', SUM(energies(:) + ekin(:)) + &
     519         2365 :             neb_env%spring_energy
     520          289 :          WRITE (output_unit, FMT='(A,A)') ' **************************************', &
     521          578 :             '*****************************************'
     522          289 :          DEALLOCATE (ekin)
     523         1156 :          DEALLOCATE (temperatures)
     524              :       END IF
     525              :       ! Ener file
     526              :       ener = cp_print_key_unit_nr(logger, neb_env%neb_section, "ENERGY", &
     527          578 :                                   extension=".ener", file_form="FORMATTED")
     528          578 :       IF (ener > 0) THEN
     529          289 :          WRITE (line, '(I0)') 2*neb_env%number_of_replica - 1
     530          289 :          WRITE (ener, '(I10,'//TRIM(line)//'(1X,F20.9))') istep, &
     531          578 :             energies, distances
     532              :       END IF
     533              :       CALL cp_print_key_finished_output(ener, logger, neb_env%neb_section, &
     534          578 :                                         "ENERGY")
     535              : 
     536              :       ! Dump Restarts
     537          578 :       CALL cp_add_default_logger(logger)
     538              :       CALL write_restart(force_env=neb_env%force_env, &
     539              :                          root_section=neb_env%root_section, &
     540              :                          coords=coords, &
     541          578 :                          vels=vels)
     542          578 :       CALL cp_rm_default_logger()
     543              : 
     544          578 :       CALL timestop(handle)
     545              : 
     546          578 :    END SUBROUTINE dump_neb_info
     547              : 
     548              : ! **************************************************************************************************
     549              : !> \brief dump coordinates of a replica NEB
     550              : !> \param particle_set ...
     551              : !> \param coords ...
     552              : !> \param i_rep ...
     553              : !> \param ienum ...
     554              : !> \param iw ...
     555              : !> \param use_colvar ...
     556              : !> \author Teodoro Laino 09.2006
     557              : ! **************************************************************************************************
     558          212 :    SUBROUTINE dump_replica_coordinates(particle_set, coords, i_rep, ienum, iw, use_colvar)
     559              : 
     560              :       TYPE(particle_type), DIMENSION(:), POINTER         :: particle_set
     561              :       TYPE(neb_var_type), POINTER                        :: coords
     562              :       INTEGER, INTENT(IN)                                :: i_rep, ienum, iw
     563              :       LOGICAL, INTENT(IN)                                :: use_colvar
     564              : 
     565              :       INTEGER                                            :: iatom, j
     566              :       REAL(KIND=dp), DIMENSION(3)                        :: r
     567              : 
     568          212 :       IF (iw > 0) THEN
     569           18 :          WRITE (iw, '(/,T2,"NEB|",75("*"))')
     570              :          WRITE (iw, '(T2,"NEB|",1X,A,I0,A)') &
     571           18 :             "Geometry for Replica Nr. ", ienum, " in Angstrom"
     572          948 :          DO iatom = 1, SIZE(particle_set)
     573          930 :             r(1:3) = get_particle_pos_or_vel(iatom, particle_set, coords%xyz(:, i_rep))
     574              :             WRITE (iw, '(T2,"NEB|",1X,A10,5X,3F15.9)') &
     575         4668 :                TRIM(particle_set(iatom)%atomic_kind%name), r(1:3)*angstrom
     576              :          END DO
     577           18 :          IF (use_colvar) THEN
     578           10 :             WRITE (iw, '(/,T2,"NEB|",1X,A10)') "COLLECTIVE VARIABLES:"
     579              :             WRITE (iw, '(T2,"NEB|",16X,3F15.9)') &
     580           20 :                (coords%int(j, i_rep), j=1, SIZE(coords%int(:, :), 1))
     581              :          END IF
     582           18 :          WRITE (iw, '(T2,"NEB|",75("*"))')
     583           18 :          CALL m_flush(iw)
     584              :       END IF
     585              : 
     586          212 :    END SUBROUTINE dump_replica_coordinates
     587              : 
     588              : ! **************************************************************************************************
     589              : !> \brief Handles the correct file names during a band calculation
     590              : !> \param rep_env ...
     591              : !> \param irep ...
     592              : !> \param n_rep ...
     593              : !> \param istep ...
     594              : !> \author Teodoro Laino  06.2009
     595              : ! **************************************************************************************************
     596         8376 :    SUBROUTINE handle_band_file_names(rep_env, irep, n_rep, istep)
     597              :       TYPE(replica_env_type), POINTER                    :: rep_env
     598              :       INTEGER, INTENT(IN)                                :: irep, n_rep, istep
     599              : 
     600              :       CHARACTER(len=*), PARAMETER :: routineN = 'handle_band_file_names'
     601              : 
     602              :       CHARACTER(LEN=default_path_length)                 :: output_file_path, replica_proj_name
     603              :       INTEGER                                            :: handle, handle2, i, ierr, j, lp, unit_nr
     604              :       TYPE(cp_logger_type), POINTER                      :: logger, sub_logger
     605              :       TYPE(f_env_type), POINTER                          :: f_env
     606              :       TYPE(section_vals_type), POINTER                   :: root_section
     607              : 
     608         2792 :       CALL timeset(routineN, handle)
     609              :       CALL f_env_add_defaults(f_env_id=rep_env%f_env_id, f_env=f_env, &
     610         2792 :                               handle=handle2)
     611         2792 :       logger => cp_get_default_logger()
     612         2792 :       CALL force_env_get(f_env%force_env, root_section=root_section)
     613         2792 :       j = irep + (rep_env%local_rep_indices(1) - 1)
     614              :       ! Get replica_project_name
     615         2792 :       replica_proj_name = get_replica_project_name(rep_env, n_rep, j)
     616         2792 :       lp = LEN_TRIM(replica_proj_name)
     617              :       CALL section_vals_val_set(root_section, "GLOBAL%PROJECT_NAME", &
     618         2792 :                                 c_val=TRIM(replica_proj_name))
     619         2792 :       logger%iter_info%project_name = TRIM(replica_proj_name)
     620              : 
     621              :       ! We change the file on which is pointing the global logger and error
     622         2792 :       output_file_path = replica_proj_name(1:lp)//".out"
     623              :       CALL section_vals_val_set(root_section, "GLOBAL%OUTPUT_FILE_NAME", &
     624         2792 :                                 c_val=TRIM(output_file_path))
     625         2792 :       IF (logger%default_global_unit_nr > 0) THEN
     626         2777 :          CALL close_file(logger%default_global_unit_nr)
     627              :          CALL open_file(file_name=output_file_path, file_status="UNKNOWN", &
     628              :                         file_action="WRITE", file_position="APPEND", &
     629              :                         unit_number=logger%default_global_unit_nr, &
     630         2777 :                         skip_get_unit_number=.TRUE.)
     631              :          WRITE (UNIT=logger%default_global_unit_nr, FMT="(/,(T2,A79))") &
     632         2777 :             "*******************************************************************************", &
     633         2777 :             "**                 BAND EVALUATION OF ENERGIES AND FORCES                    **", &
     634         5554 :             "*******************************************************************************"
     635         2777 :          WRITE (UNIT=logger%default_global_unit_nr, FMT="(T2,A,T79,A)") "**", "**"
     636         2777 :          WRITE (UNIT=logger%default_global_unit_nr, FMT="(T2,A,T79,A)") "**", "**"
     637              :          WRITE (UNIT=logger%default_global_unit_nr, FMT="(T2,A,I5,T41,A,I5,T79,A)") &
     638         2777 :             "** Replica Env Nr. :", rep_env%local_rep_indices(1) - 1, "Replica Band Nr. :", j, "**"
     639              :          WRITE (UNIT=logger%default_global_unit_nr, FMT="(T2,A,I5,T79,A)") &
     640         2777 :             "** Band  Step  Nr. :", istep, "**"
     641              :          WRITE (UNIT=logger%default_global_unit_nr, FMT="(T2,A79)") &
     642         2777 :             "*******************************************************************************"
     643              :       END IF
     644              : 
     645              :       ! Handle specific case for mixed_env
     646         2822 :       SELECT CASE (f_env%force_env%in_use)
     647              :       CASE (use_mixed_force)
     648         2852 :          DO i = 1, f_env%force_env%mixed_env%ngroups
     649           60 :             IF (MODULO(i - 1, f_env%force_env%mixed_env%ngroups) == &
     650           30 :                 f_env%force_env%mixed_env%group_distribution(f_env%force_env%mixed_env%para_env%mepos)) THEN
     651           30 :                sub_logger => f_env%force_env%mixed_env%sub_logger(i)%p
     652           30 :                sub_logger%iter_info%project_name = replica_proj_name(1:lp)//"-r-"//TRIM(ADJUSTL(cp_to_string(i)))
     653              : 
     654           30 :                unit_nr = sub_logger%default_global_unit_nr
     655           30 :                IF (unit_nr > 0) THEN
     656           30 :                   CALL close_file(unit_nr)
     657              : 
     658           30 :                   output_file_path = replica_proj_name(1:lp)//"-r-"//TRIM(ADJUSTL(cp_to_string(i)))//".out"
     659              :                   CALL open_file(file_name=output_file_path, file_status="UNKNOWN", &
     660              :                                  file_action="WRITE", file_position="APPEND", &
     661           30 :                                  unit_number=unit_nr, skip_get_unit_number=.TRUE.)
     662              :                END IF
     663              :             END IF
     664              :          END DO
     665              :       END SELECT
     666              : 
     667         2792 :       CALL f_env_rm_defaults(f_env=f_env, ierr=ierr, handle=handle2)
     668         2792 :       CPASSERT(ierr == 0)
     669         2792 :       CALL timestop(handle)
     670              : 
     671         2792 :    END SUBROUTINE handle_band_file_names
     672              : 
     673              : ! **************************************************************************************************
     674              : !> \brief  Constructs project names for BAND replicas
     675              : !> \param rep_env ...
     676              : !> \param n_rep ...
     677              : !> \param j ...
     678              : !> \return ...
     679              : !> \author Teodoro Laino  06.2009
     680              : ! **************************************************************************************************
     681         2916 :    FUNCTION get_replica_project_name(rep_env, n_rep, j) RESULT(replica_proj_name)
     682              :       TYPE(replica_env_type), POINTER                    :: rep_env
     683              :       INTEGER, INTENT(IN)                                :: n_rep, j
     684              :       CHARACTER(LEN=default_path_length)                 :: replica_proj_name
     685              : 
     686              :       CHARACTER(LEN=default_string_length)               :: padding
     687              :       INTEGER                                            :: i, lp, ndigits
     688              : 
     689              : ! Setup new replica project name and output file
     690              : 
     691         2916 :       replica_proj_name = rep_env%original_project_name
     692              :       ! Find padding
     693              :       ndigits = CEILING(LOG10(REAL(n_rep + 1, KIND=dp))) - &
     694         2916 :                 CEILING(LOG10(REAL(j + 1, KIND=dp)))
     695         2916 :       padding = ""
     696         3618 :       DO i = 1, ndigits
     697         3618 :          padding(i:i) = "0"
     698              :       END DO
     699         2916 :       lp = LEN_TRIM(replica_proj_name)
     700              :       replica_proj_name(lp + 1:LEN(replica_proj_name)) = "-BAND"// &
     701         2916 :                                                          TRIM(padding)//ADJUSTL(cp_to_string(j))
     702         2916 :    END FUNCTION get_replica_project_name
     703              : 
     704              : ! **************************************************************************************************
     705              : !> \brief  Print some mapping infos in the replica_env setup output files
     706              : !>         i.e. prints in which files one can find information for each band
     707              : !>         replica
     708              : !> \param rep_env ...
     709              : !> \param neb_env ...
     710              : !> \author Teodoro Laino  06.2009
     711              : ! **************************************************************************************************
     712           68 :    SUBROUTINE neb_rep_env_map_info(rep_env, neb_env)
     713              :       TYPE(replica_env_type), POINTER                    :: rep_env
     714              :       TYPE(neb_type), POINTER                            :: neb_env
     715              : 
     716              :       CHARACTER(LEN=default_path_length)                 :: replica_proj_name
     717              :       INTEGER                                            :: handle2, ierr, irep, n_rep, n_rep_neb, &
     718              :                                                             output_unit
     719              :       TYPE(cp_logger_type), POINTER                      :: logger
     720              :       TYPE(f_env_type), POINTER                          :: f_env
     721              : 
     722           34 :       n_rep_neb = neb_env%number_of_replica
     723           34 :       n_rep = rep_env%nrep
     724              :       CALL f_env_add_defaults(f_env_id=rep_env%f_env_id, f_env=f_env, &
     725           34 :                               handle=handle2)
     726           34 :       logger => cp_get_default_logger()
     727           34 :       output_unit = logger%default_global_unit_nr
     728           34 :       IF (output_unit > 0) THEN
     729              :          WRITE (UNIT=output_unit, FMT='(/,(T2,A79))') &
     730           33 :             "*******************************************************************************", &
     731           33 :             "**                  MAPPING OF BAND REPLICA TO REPLICA ENV                   **", &
     732           66 :             "*******************************************************************************"
     733              :          WRITE (UNIT=output_unit, FMT='(T2,A,I6,T32,A,T79,A)') &
     734           33 :             "** Replica Env Nr.: ", rep_env%local_rep_indices(1) - 1, &
     735           66 :             "working on the following BAND replicas", "**"
     736              :          WRITE (UNIT=output_unit, FMT='(T2,A79)') &
     737           33 :             "**                                                                           **"
     738              :       END IF
     739          158 :       DO irep = 1, n_rep_neb, n_rep
     740          124 :          replica_proj_name = get_replica_project_name(rep_env, n_rep_neb, irep + rep_env%local_rep_indices(1) - 1)
     741          158 :          IF (output_unit > 0) THEN
     742              :             WRITE (UNIT=output_unit, FMT='(T2,A,I6,T32,A,T79,A)') &
     743          119 :                "** Band Replica   Nr.: ", irep + rep_env%local_rep_indices(1) - 1, &
     744          238 :                "Output available on file: "//TRIM(replica_proj_name)//".out", "**"
     745              :          END IF
     746              :       END DO
     747           34 :       IF (output_unit > 0) THEN
     748              :          WRITE (UNIT=output_unit, FMT='(T2,A79)') &
     749           33 :             "**                                                                           **", &
     750           66 :             "*******************************************************************************"
     751           33 :          WRITE (UNIT=output_unit, FMT='(/)')
     752              :       END IF
     753              :       ! update runtime info before printing the footer
     754           34 :       CALL get_runtime_info()
     755              :       ! print footer
     756           34 :       CALL cp2k_footer(output_unit)
     757           34 :       CALL f_env_rm_defaults(f_env=f_env, ierr=ierr, handle=handle2)
     758           34 :       CPASSERT(ierr == 0)
     759           34 :    END SUBROUTINE neb_rep_env_map_info
     760              : 
     761              : END MODULE neb_io
        

Generated by: LCOV version 2.0-1