LCOV - code coverage report
Current view: top level - src/motion - bfgs_optimizer.F (source / functions) Hit Total Coverage
Test: CP2K Regtests (git:a24d8ac) Lines: 518 553 93.7 %
Date: 2025-01-02 07:24:57 Functions: 12 12 100.0 %

          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 Geometry optimization using BFGS algorithm
      10             : !> \par History
      11             : !>      Module modified by Pierre-André Cazade [pcazade] 01.2020 - University of Limerick.
      12             : !>      Modifications enable Space Group Symmetry.
      13             : ! **************************************************************************************************
      14             : MODULE bfgs_optimizer
      15             : 
      16             :    USE atomic_kind_list_types, ONLY: atomic_kind_list_type
      17             :    USE atomic_kind_types, ONLY: get_atomic_kind, &
      18             :                                 get_atomic_kind_set
      19             :    USE cell_types, ONLY: cell_type, &
      20             :                          pbc
      21             :    USE constraint_fxd, ONLY: fix_atom_control
      22             :    USE cp_blacs_env, ONLY: cp_blacs_env_create, &
      23             :                            cp_blacs_env_release, &
      24             :                            cp_blacs_env_type
      25             :    USE cp_external_control, ONLY: external_control
      26             :    USE cp_files, ONLY: close_file, &
      27             :                        open_file
      28             :    USE cp_fm_basic_linalg, ONLY: cp_fm_column_scale, &
      29             :                                  cp_fm_transpose
      30             :    USE cp_fm_diag, ONLY: choose_eigv_solver
      31             :    USE cp_fm_struct, ONLY: cp_fm_struct_create, &
      32             :                            cp_fm_struct_release, &
      33             :                            cp_fm_struct_type
      34             :    USE cp_fm_types, ONLY: &
      35             :       cp_fm_get_info, &
      36             :       cp_fm_read_unformatted, &
      37             :       cp_fm_set_all, &
      38             :       cp_fm_to_fm, &
      39             :       cp_fm_type, &
      40             :       cp_fm_write_unformatted, cp_fm_create, cp_fm_release
      41             :    USE parallel_gemm_api, ONLY: parallel_gemm
      42             :    USE cp_log_handling, ONLY: cp_get_default_logger, &
      43             :                               cp_logger_type, &
      44             :                               cp_to_string
      45             :    USE cp_output_handling, ONLY: cp_iterate, &
      46             :                                  cp_p_file, &
      47             :                                  cp_print_key_finished_output, &
      48             :                                  cp_print_key_should_output, &
      49             :                                  cp_print_key_unit_nr
      50             :    USE message_passing, ONLY: mp_para_env_type
      51             :    USE cp_subsys_types, ONLY: cp_subsys_get, &
      52             :                               cp_subsys_type
      53             :    USE force_env_types, ONLY: force_env_get, &
      54             :                               force_env_type
      55             :    USE global_types, ONLY: global_environment_type
      56             :    USE gopt_f_methods, ONLY: gopt_f_ii, &
      57             :                              gopt_f_io, &
      58             :                              gopt_f_io_finalize, &
      59             :                              gopt_f_io_init, &
      60             :                              print_geo_opt_header, &
      61             :                              print_geo_opt_nc
      62             :    USE gopt_f_types, ONLY: gopt_f_type
      63             :    USE gopt_param_types, ONLY: gopt_param_type
      64             :    USE input_constants, ONLY: default_cell_method_id, &
      65             :                               default_ts_method_id
      66             :    USE input_section_types, ONLY: section_vals_get_subs_vals, &
      67             :                                   section_vals_type, &
      68             :                                   section_vals_val_get, &
      69             :                                   section_vals_val_set
      70             :    USE kinds, ONLY: default_path_length, &
      71             :                     dp
      72             :    USE machine, ONLY: m_flush, &
      73             :                       m_walltime
      74             :    USE particle_list_types, ONLY: particle_list_type
      75             :    USE space_groups, ONLY: identify_space_group, &
      76             :                            print_spgr, &
      77             :                            spgr_apply_rotations_coord, &
      78             :                            spgr_apply_rotations_force
      79             :    USE space_groups_types, ONLY: spgr_type
      80             : #include "../base/base_uses.f90"
      81             : 
      82             :    IMPLICIT NONE
      83             :    PRIVATE
      84             : 
      85             :    #:include "gopt_f77_methods.fypp"
      86             : 
      87             :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'bfgs_optimizer'
      88             :    LOGICAL, PARAMETER                   :: debug_this_module = .TRUE.
      89             : 
      90             :    PUBLIC :: geoopt_bfgs
      91             : 
      92             : CONTAINS
      93             : 
      94             : ! **************************************************************************************************
      95             : !> \brief Main driver for BFGS geometry optimizations
      96             : !> \param force_env ...
      97             : !> \param gopt_param ...
      98             : !> \param globenv ...
      99             : !> \param geo_section ...
     100             : !> \param gopt_env ...
     101             : !> \param x0 ...
     102             : !> \par History
     103             : !>      01.2020 modified to perform Space Group Symmetry [pcazade]
     104             : ! **************************************************************************************************
     105        1246 :    RECURSIVE SUBROUTINE geoopt_bfgs(force_env, gopt_param, globenv, geo_section, gopt_env, x0)
     106             : 
     107             :       TYPE(force_env_type), POINTER                      :: force_env
     108             :       TYPE(gopt_param_type), POINTER                     :: gopt_param
     109             :       TYPE(global_environment_type), POINTER             :: globenv
     110             :       TYPE(section_vals_type), POINTER                   :: geo_section
     111             :       TYPE(gopt_f_type), POINTER                         :: gopt_env
     112             :       REAL(KIND=dp), DIMENSION(:), POINTER               :: x0
     113             : 
     114             :       CHARACTER(len=*), PARAMETER                        :: routineN = 'geoopt_bfgs'
     115             :       REAL(KIND=dp), PARAMETER                           :: one = 1.0_dp, zero = 0.0_dp
     116             : 
     117             :       CHARACTER(LEN=5)                                   :: wildcard
     118             :       CHARACTER(LEN=default_path_length)                 :: hes_filename
     119             :       INTEGER                                            :: handle, hesunit_read, indf, info, &
     120             :                                                             iter_nr, its, maxiter, ndf, nfree, &
     121             :                                                             output_unit
     122             :       LOGICAL                                            :: conv, hesrest, ionode, shell_present, &
     123             :                                                             should_stop, use_mod_hes, use_rfo
     124             :       REAL(KIND=dp)                                      :: ediff, emin, eold, etot, pred, rad, rat, &
     125             :                                                             step, t_diff, t_now, t_old
     126        1246 :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:)           :: dg, dr, dx, eigval, gold, work, xold
     127        1246 :       REAL(KIND=dp), DIMENSION(:), POINTER               :: g
     128             :       TYPE(atomic_kind_list_type), POINTER               :: atomic_kinds
     129             :       TYPE(cp_blacs_env_type), POINTER                   :: blacs_env
     130             :       TYPE(cp_fm_struct_type), POINTER                   :: fm_struct_hes
     131             :       TYPE(cp_fm_type)                          :: eigvec_mat, hess_mat, hess_tmp
     132             :       TYPE(cp_logger_type), POINTER                      :: logger
     133             :       TYPE(mp_para_env_type), POINTER                    :: para_env
     134             :       TYPE(cp_subsys_type), POINTER                      :: subsys
     135             :       TYPE(section_vals_type), POINTER                   :: print_key, root_section
     136             :       TYPE(spgr_type), POINTER                           :: spgr
     137             : 
     138        1246 :       NULLIFY (logger, g, blacs_env, spgr)
     139        2492 :       logger => cp_get_default_logger()
     140        1246 :       para_env => force_env%para_env
     141        1246 :       root_section => force_env%root_section
     142        1246 :       spgr => gopt_env%spgr
     143        1246 :       t_old = m_walltime()
     144             : 
     145        1246 :       CALL timeset(routineN, handle)
     146        1246 :       CALL section_vals_val_get(geo_section, "BFGS%TRUST_RADIUS", r_val=rad)
     147        1246 :       print_key => section_vals_get_subs_vals(geo_section, "BFGS%RESTART")
     148        1246 :       ionode = para_env%is_source()
     149        1246 :       maxiter = gopt_param%max_iter
     150        1246 :       conv = .FALSE.
     151        1246 :       rat = 0.0_dp
     152        1246 :       wildcard = " BFGS"
     153        1246 :       hes_filename = ""
     154             : 
     155             :       ! Stop if not yet implemented
     156        1246 :       SELECT CASE (gopt_env%type_id)
     157             :       CASE (default_ts_method_id)
     158        1246 :          CPABORT("BFGS method not yet working with DIMER")
     159             :       END SELECT
     160             : 
     161        1246 :       CALL section_vals_val_get(geo_section, "BFGS%USE_RAT_FUN_OPT", l_val=use_rfo)
     162        1246 :       CALL section_vals_val_get(geo_section, "BFGS%USE_MODEL_HESSIAN", l_val=use_mod_hes)
     163        1246 :       CALL section_vals_val_get(geo_section, "BFGS%RESTART_HESSIAN", l_val=hesrest)
     164             :       output_unit = cp_print_key_unit_nr(logger, geo_section, "PRINT%PROGRAM_RUN_INFO", &
     165        1246 :                                          extension=".geoLog")
     166        1246 :       IF (output_unit > 0) THEN
     167         629 :          IF (use_rfo) THEN
     168             :             WRITE (UNIT=output_unit, FMT="(/,T2,A,T78,A3)") &
     169          35 :                "BFGS| Use rational function optimization for step estimation: ", "YES"
     170             :          ELSE
     171             :             WRITE (UNIT=output_unit, FMT="(/,T2,A,T78,A3)") &
     172         594 :                "BFGS| Use rational function optimization for step estimation: ", " NO"
     173             :          END IF
     174         629 :          IF (use_mod_hes) THEN
     175             :             WRITE (UNIT=output_unit, FMT="(T2,A,T78,A3)") &
     176         525 :                "BFGS| Use model Hessian for initial guess: ", "YES"
     177             :          ELSE
     178             :             WRITE (UNIT=output_unit, FMT="(T2,A,T78,A3)") &
     179         104 :                "BFGS| Use model Hessian for initial guess: ", " NO"
     180             :          END IF
     181         629 :          IF (hesrest) THEN
     182             :             WRITE (UNIT=output_unit, FMT="(T2,A,T78,A3)") &
     183           1 :                "BFGS| Restart Hessian: ", "YES"
     184             :          ELSE
     185             :             WRITE (UNIT=output_unit, FMT="(T2,A,T78,A3)") &
     186         628 :                "BFGS| Restart Hessian: ", " NO"
     187             :          END IF
     188             :          WRITE (UNIT=output_unit, FMT="(T2,A,T61,F20.3)") &
     189         629 :             "BFGS| Trust radius: ", rad
     190             :       END IF
     191             : 
     192        1246 :       ndf = SIZE(x0)
     193        1246 :       nfree = gopt_env%nfree
     194        1246 :       IF (ndf > 3000) &
     195             :          CALL cp_warn(__LOCATION__, &
     196             :                       "The dimension of the Hessian matrix ("// &
     197             :                       TRIM(ADJUSTL(cp_to_string(ndf)))//") is greater than 3000. "// &
     198             :                       "The diagonalisation of the full Hessian  matrix needed for BFGS "// &
     199             :                       "is computationally expensive. You should consider to use the linear "// &
     200           0 :                       "scaling variant L-BFGS instead.")
     201             : 
     202             :       ! Initialize hessian (hes = unitary matrix or model hessian )
     203             :       CALL cp_blacs_env_create(blacs_env, para_env, globenv%blacs_grid_layout, &
     204        1246 :                                globenv%blacs_repeatable)
     205             :       CALL cp_fm_struct_create(fm_struct_hes, para_env=para_env, context=blacs_env, &
     206        1246 :                                nrow_global=ndf, ncol_global=ndf)
     207        1246 :       CALL cp_fm_create(hess_mat, fm_struct_hes, name="hess_mat")
     208        1246 :       CALL cp_fm_create(hess_tmp, fm_struct_hes, name="hess_tmp")
     209        1246 :       CALL cp_fm_create(eigvec_mat, fm_struct_hes, name="eigvec_mat")
     210        3738 :       ALLOCATE (eigval(ndf))
     211       89302 :       eigval(:) = 0.0_dp
     212             : 
     213        1246 :       CALL force_env_get(force_env=force_env, subsys=subsys)
     214        1246 :       CALL cp_subsys_get(subsys, atomic_kinds=atomic_kinds)
     215        1246 :       CALL get_atomic_kind_set(atomic_kind_set=atomic_kinds%els, shell_present=shell_present)
     216        1246 :       IF (use_mod_hes) THEN
     217        1038 :          IF (shell_present) THEN
     218             :             CALL cp_warn(__LOCATION__, &
     219             :                          "No model Hessian is available for core-shell models. "// &
     220           4 :                          "A unit matrix is used as the initial Hessian.")
     221           4 :             use_mod_hes = .FALSE.
     222             :          END IF
     223        1038 :          IF (gopt_env%type_id == default_cell_method_id) THEN
     224             :             CALL cp_warn(__LOCATION__, &
     225             :                          "No model Hessian is available for cell optimizations. "// &
     226           0 :                          "A unit matrix is used as the initial Hessian.")
     227           0 :             use_mod_hes = .FALSE.
     228             :          END IF
     229             :       END IF
     230             : 
     231        1246 :       IF (use_mod_hes) THEN
     232        1034 :          CALL cp_fm_set_all(hess_mat, alpha=zero, beta=0.00_dp)
     233        1034 :          CALL construct_initial_hess(gopt_env%force_env, hess_mat)
     234        1034 :          CALL cp_fm_to_fm(hess_mat, hess_tmp)
     235        1034 :          CALL choose_eigv_solver(hess_tmp, eigvec_mat, eigval, info=info)
     236             :          ! In rare cases the diagonalization of hess_mat fails (bug in scalapack?)
     237        1034 :          IF (info /= 0) THEN
     238           0 :             CALL cp_fm_set_all(hess_mat, alpha=zero, beta=one)
     239           0 :             IF (output_unit > 0) WRITE (output_unit, *) &
     240           0 :                "BFGS: Matrix diagonalization failed, using unity as model Hessian."
     241             :          ELSE
     242       55424 :             DO its = 1, SIZE(eigval)
     243       55424 :                IF (eigval(its) .LT. 0.1_dp) eigval(its) = 0.1_dp
     244             :             END DO
     245        1034 :             CALL cp_fm_to_fm(eigvec_mat, hess_tmp)
     246        1034 :             CALL cp_fm_column_scale(eigvec_mat, eigval)
     247        1034 :             CALL parallel_gemm("N", "T", ndf, ndf, ndf, one, hess_tmp, eigvec_mat, zero, hess_mat)
     248             :          END IF
     249             :       ELSE
     250         212 :          CALL cp_fm_set_all(hess_mat, alpha=zero, beta=one)
     251             :       END IF
     252             : 
     253        2492 :       ALLOCATE (xold(ndf))
     254       89302 :       xold(:) = x0(:)
     255             : 
     256        2492 :       ALLOCATE (g(ndf))
     257       89302 :       g(:) = 0.0_dp
     258             : 
     259        2492 :       ALLOCATE (gold(ndf))
     260       89302 :       gold(:) = 0.0_dp
     261             : 
     262        2492 :       ALLOCATE (dx(ndf))
     263       89302 :       dx(:) = 0.0_dp
     264             : 
     265        2492 :       ALLOCATE (dg(ndf))
     266       89302 :       dg(:) = 0.0_dp
     267             : 
     268        2492 :       ALLOCATE (work(ndf))
     269       89302 :       work(:) = 0.0_dp
     270             : 
     271        2492 :       ALLOCATE (dr(ndf))
     272       89302 :       dr(:) = 0.0_dp
     273             : 
     274             :       ! find space_group
     275        1246 :       CALL section_vals_val_get(geo_section, "KEEP_SPACE_GROUP", l_val=spgr%keep_space_group)
     276        1246 :       IF (spgr%keep_space_group) THEN
     277           4 :          CALL identify_space_group(subsys, geo_section, gopt_env, output_unit)
     278           4 :          CALL spgr_apply_rotations_coord(spgr, x0)
     279           4 :          CALL print_spgr(spgr)
     280             :       END IF
     281             : 
     282             :       ! Geometry optimization starts now
     283        1246 :       CALL cp_iterate(logger%iter_info, increment=0, iter_nr_out=iter_nr)
     284        1246 :       CALL print_geo_opt_header(gopt_env, output_unit, wildcard)
     285             : 
     286             :       ! Calculate Energy & Gradients
     287             :       CALL cp_eval_at(gopt_env, x0, etot, g, gopt_env%force_env%para_env%mepos, &
     288        1246 :                       .FALSE., gopt_env%force_env%para_env)
     289             : 
     290             :       ! Symmetrize coordinates and forces
     291        1246 :       IF (spgr%keep_space_group) THEN
     292           4 :          CALL spgr_apply_rotations_coord(spgr, x0)
     293           4 :          CALL spgr_apply_rotations_force(spgr, g)
     294             :       END IF
     295             : 
     296             :       ! Print info at time 0
     297        1246 :       emin = etot
     298        1246 :       t_now = m_walltime()
     299        1246 :       t_diff = t_now - t_old
     300        1246 :       t_old = t_now
     301        1246 :       CALL gopt_f_io_init(gopt_env, output_unit, etot, wildcard=wildcard, its=iter_nr, used_time=t_diff)
     302        7007 :       DO its = iter_nr + 1, maxiter
     303        6997 :          CALL cp_iterate(logger%iter_info, last=(its == maxiter))
     304        6997 :          CALL section_vals_val_set(geo_section, "STEP_START_VAL", i_val=its)
     305        6997 :          CALL gopt_f_ii(its, output_unit)
     306             : 
     307             :          ! Hessian update/restarting
     308        6997 :          IF (((its - iter_nr) == 1) .AND. hesrest) THEN
     309           2 :             IF (ionode) THEN
     310           1 :                CALL section_vals_val_get(geo_section, "BFGS%RESTART_FILE_NAME", c_val=hes_filename)
     311           1 :                IF (LEN_TRIM(hes_filename) == 0) THEN
     312             :                   ! Set default Hessian restart file name if no file name is defined
     313           0 :                   hes_filename = TRIM(logger%iter_info%project_name)//"-BFGS.Hessian"
     314             :                END IF
     315           1 :                IF (output_unit > 0) THEN
     316             :                   WRITE (UNIT=output_unit, FMT="(/,T2,A)") &
     317           1 :                      "BFGS| Checking for Hessian restart file <"//TRIM(ADJUSTL(hes_filename))//">"
     318             :                END IF
     319             :                CALL open_file(file_name=TRIM(hes_filename), file_status="OLD", &
     320           1 :                               file_form="UNFORMATTED", file_action="READ", unit_number=hesunit_read)
     321           1 :                IF (output_unit > 0) THEN
     322             :                   WRITE (UNIT=output_unit, FMT="(T2,A)") &
     323           1 :                      "BFGS| Hessian restart file read"
     324             :                END IF
     325             :             END IF
     326           2 :             CALL cp_fm_read_unformatted(hess_mat, hesunit_read)
     327           2 :             IF (ionode) CALL close_file(unit_number=hesunit_read)
     328             :          ELSE
     329        6995 :             IF ((its - iter_nr) > 1) THEN
     330             :                ! Symmetrize old coordinates and old forces
     331        5761 :                IF (spgr%keep_space_group) THEN
     332           0 :                   CALL spgr_apply_rotations_coord(spgr, xold)
     333           0 :                   CALL spgr_apply_rotations_force(spgr, gold)
     334             :                END IF
     335             : 
     336      595900 :                DO indf = 1, ndf
     337      590139 :                   dx(indf) = x0(indf) - xold(indf)
     338      595900 :                   dg(indf) = g(indf) - gold(indf)
     339             :                END DO
     340             : 
     341        5761 :                CALL bfgs(ndf, dx, dg, hess_mat, work, para_env)
     342             : 
     343             :                ! Symmetrize coordinates and forces change
     344        5761 :                IF (spgr%keep_space_group) THEN
     345           0 :                   CALL spgr_apply_rotations_force(spgr, dx)
     346           0 :                   CALL spgr_apply_rotations_force(spgr, dg)
     347             :                END IF
     348             : 
     349             :                !Possibly dump the Hessian file
     350        5761 :                IF (BTEST(cp_print_key_should_output(logger%iter_info, print_key), cp_p_file)) THEN
     351        3767 :                   CALL write_bfgs_hessian(geo_section, hess_mat, logger)
     352             :                END IF
     353             :             END IF
     354             :          END IF
     355             : 
     356             :          ! Symmetrize coordinates and forces
     357        6997 :          IF (spgr%keep_space_group) THEN
     358           4 :             CALL spgr_apply_rotations_coord(spgr, x0)
     359           4 :             CALL spgr_apply_rotations_force(spgr, g)
     360             :          END IF
     361             : 
     362             :          ! Setting the present positions & gradients as old
     363      685048 :          xold(:) = x0
     364      685048 :          gold(:) = g
     365             : 
     366             :          ! Copying hessian hes to (ndf x ndf) matrix hes_mat for diagonalization
     367        6997 :          CALL cp_fm_to_fm(hess_mat, hess_tmp)
     368             : 
     369        6997 :          CALL choose_eigv_solver(hess_tmp, eigvec_mat, eigval, info=info)
     370             : 
     371             :          ! In rare cases the diagonalization of hess_mat fails (bug in scalapack?)
     372        6997 :          IF (info /= 0) THEN
     373           0 :             IF (output_unit > 0) WRITE (output_unit, *) &
     374           0 :                "BFGS: Matrix diagonalization failed, resetting Hessian to unity."
     375           0 :             CALL cp_fm_set_all(hess_mat, alpha=zero, beta=one)
     376           0 :             CALL cp_fm_to_fm(hess_mat, hess_tmp)
     377           0 :             CALL choose_eigv_solver(hess_tmp, eigvec_mat, eigval)
     378             :          END IF
     379             : 
     380        6997 :          IF (use_rfo) THEN
     381        1350 :             CALL set_hes_eig(ndf, eigval, work)
     382      120078 :             dx(:) = eigval
     383        1350 :             CALL rat_fun_opt(ndf, dg, eigval, work, eigvec_mat, g, para_env)
     384             :          END IF
     385        6997 :          CALL geoopt_get_step(ndf, eigval, eigvec_mat, hess_tmp, dr, g, para_env, use_rfo)
     386             : 
     387             :          ! Symmetrize dr
     388        6997 :          IF (spgr%keep_space_group) THEN
     389           4 :             CALL spgr_apply_rotations_force(spgr, dr)
     390             :          END IF
     391             : 
     392        6997 :          CALL trust_radius(ndf, step, rad, rat, dr, output_unit)
     393             : 
     394             :          ! Update the atomic positions
     395      685048 :          x0 = x0 + dr
     396             : 
     397             :          ! Symmetrize coordinates
     398        6997 :          IF (spgr%keep_space_group) THEN
     399           4 :             CALL spgr_apply_rotations_coord(spgr, x0)
     400             :          END IF
     401             : 
     402        6997 :          CALL energy_predict(ndf, work, hess_mat, dr, g, conv, pred, para_env)
     403        6997 :          eold = etot
     404             : 
     405             :          ! Energy & Gradients at new step
     406             :          CALL cp_eval_at(gopt_env, x0, etot, g, gopt_env%force_env%para_env%mepos, &
     407        6997 :                          .FALSE., gopt_env%force_env%para_env)
     408             : 
     409        6997 :          ediff = etot - eold
     410             : 
     411             :          ! Symmetrize forces
     412        6997 :          IF (spgr%keep_space_group) THEN
     413           4 :             CALL spgr_apply_rotations_force(spgr, g)
     414             :          END IF
     415             : 
     416             :          ! check for an external exit command
     417        6997 :          CALL external_control(should_stop, "GEO", globenv=globenv)
     418        6997 :          IF (should_stop) EXIT
     419             : 
     420             :          ! Some IO and Convergence check
     421        6997 :          t_now = m_walltime()
     422        6997 :          t_diff = t_now - t_old
     423        6997 :          t_old = t_now
     424             :          CALL gopt_f_io(gopt_env, force_env, root_section, its, etot, output_unit, &
     425             :                         eold, emin, wildcard, gopt_param, ndf, dr, g, conv, pred, rat, &
     426        6997 :                         step, rad, used_time=t_diff)
     427             : 
     428        6997 :          IF (conv .OR. (its == maxiter)) EXIT
     429        5761 :          IF (etot < emin) emin = etot
     430       21001 :          IF (use_rfo) CALL update_trust_rad(rat, rad, step, ediff)
     431             :       END DO
     432             : 
     433        1246 :       IF (its == maxiter .AND. (.NOT. conv)) THEN
     434         597 :          CALL print_geo_opt_nc(gopt_env, output_unit)
     435             :       END IF
     436             : 
     437             :       ! Write final  information, if converged
     438        1246 :       CALL cp_iterate(logger%iter_info, last=.TRUE., increment=0)
     439        1246 :       CALL write_bfgs_hessian(geo_section, hess_mat, logger)
     440             :       CALL gopt_f_io_finalize(gopt_env, force_env, x0, conv, its, root_section, &
     441        1246 :                               gopt_env%force_env%para_env, gopt_env%force_env%para_env%mepos, output_unit)
     442             : 
     443        1246 :       CALL cp_fm_struct_release(fm_struct_hes)
     444        1246 :       CALL cp_fm_release(hess_mat)
     445        1246 :       CALL cp_fm_release(eigvec_mat)
     446        1246 :       CALL cp_fm_release(hess_tmp)
     447             : 
     448        1246 :       CALL cp_blacs_env_release(blacs_env)
     449        1246 :       DEALLOCATE (xold)
     450        1246 :       DEALLOCATE (g)
     451        1246 :       DEALLOCATE (gold)
     452        1246 :       DEALLOCATE (dx)
     453        1246 :       DEALLOCATE (dg)
     454        1246 :       DEALLOCATE (eigval)
     455        1246 :       DEALLOCATE (work)
     456        1246 :       DEALLOCATE (dr)
     457             : 
     458             :       CALL cp_print_key_finished_output(output_unit, logger, geo_section, &
     459        1246 :                                         "PRINT%PROGRAM_RUN_INFO")
     460        1246 :       CALL timestop(handle)
     461             : 
     462        6230 :    END SUBROUTINE geoopt_bfgs
     463             : 
     464             : ! **************************************************************************************************
     465             : !> \brief ...
     466             : !> \param ndf ...
     467             : !> \param dg ...
     468             : !> \param eigval ...
     469             : !> \param work ...
     470             : !> \param eigvec_mat ...
     471             : !> \param g ...
     472             : !> \param para_env ...
     473             : ! **************************************************************************************************
     474        2700 :    SUBROUTINE rat_fun_opt(ndf, dg, eigval, work, eigvec_mat, g, para_env)
     475             : 
     476             :       INTEGER, INTENT(IN)                                :: ndf
     477             :       REAL(KIND=dp), INTENT(INOUT)                       :: dg(ndf), eigval(ndf), work(ndf)
     478             :       TYPE(cp_fm_type), INTENT(IN)                       :: eigvec_mat
     479             :       REAL(KIND=dp), INTENT(INOUT)                       :: g(ndf)
     480             :       TYPE(mp_para_env_type), OPTIONAL, POINTER          :: para_env
     481             : 
     482             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'rat_fun_opt'
     483             :       REAL(KIND=dp), PARAMETER                           :: one = 1.0_dp
     484             : 
     485             :       INTEGER                                            :: handle, i, indf, iref, iter, j, k, l, &
     486             :                                                             maxit, ncol_local, nrow_local
     487        1350 :       INTEGER, DIMENSION(:), POINTER                     :: col_indices, row_indices
     488             :       LOGICAL                                            :: bisec, fail, set
     489             :       REAL(KIND=dp)                                      :: fun, fun1, fun2, fun3, fung, lam1, lam2, &
     490             :                                                             ln, lp, ssize, step, stol
     491        1350 :       REAL(KIND=dp), CONTIGUOUS, DIMENSION(:, :), POINTER            :: local_data
     492             : 
     493        1350 :       CALL timeset(routineN, handle)
     494             : 
     495        1350 :       stol = 1.0E-8_dp
     496        1350 :       ssize = 0.2_dp
     497        1350 :       maxit = 999
     498        1350 :       fail = .FALSE.
     499        1350 :       bisec = .FALSE.
     500             : 
     501      120078 :       dg = 0._dp
     502             : 
     503             :       CALL cp_fm_get_info(eigvec_mat, row_indices=row_indices, col_indices=col_indices, &
     504        1350 :                           local_data=local_data, nrow_local=nrow_local, ncol_local=ncol_local)
     505             : 
     506       76494 :       DO i = 1, nrow_local
     507       75144 :          j = row_indices(i)
     508    16701726 :          DO k = 1, ncol_local
     509    16625232 :             l = col_indices(k)
     510    16700376 :             dg(l) = dg(l) + local_data(i, k)*g(j)
     511             :          END DO
     512             :       END DO
     513        1350 :       CALL para_env%sum(dg)
     514             : 
     515        1350 :       set = .FALSE.
     516             : 
     517             :       DO
     518             : 
     519             : !   calculating Lambda
     520             : 
     521        1350 :          lp = 0.0_dp
     522        1350 :          iref = 1
     523        1350 :          ln = 0.0_dp
     524        1350 :          IF (eigval(iref) < 0.0_dp) ln = eigval(iref) - 0.01_dp
     525             : 
     526        1350 :          iter = 0
     527             :          DO
     528        4281 :             iter = iter + 1
     529        4281 :             fun = 0.0_dp
     530        4281 :             fung = 0.0_dp
     531      317355 :             DO indf = 1, ndf
     532      313074 :                fun = fun + dg(indf)**2/(ln - eigval(indf))
     533      317355 :                fung = fung - dg(indf)**2/(ln - eigval(indf)**2)
     534             :             END DO
     535        4281 :             fun = fun - ln
     536        4281 :             fung = fung - one
     537        4281 :             step = fun/fung
     538        4281 :             ln = ln - step
     539        4281 :             IF (ABS(step) < stol) GOTO 200
     540        2934 :             IF (iter >= maxit) EXIT
     541             :          END DO
     542             : 100      CONTINUE
     543           3 :          bisec = .TRUE.
     544           3 :          iter = 0
     545           3 :          maxit = 9999
     546           3 :          lam1 = 0.0_dp
     547           3 :          IF (eigval(iref) < 0.0_dp) lam1 = eigval(iref) - 0.01_dp
     548             :          fun1 = 0.0_dp
     549          93 :          DO indf = 1, ndf
     550          93 :             fun1 = fun1 + dg(indf)**2/(lam1 - eigval(indf))
     551             :          END DO
     552           3 :          fun1 = fun1 - lam1
     553           3 :          step = ABS(lam1)/1000.0_dp
     554           3 :          IF (step < ssize) step = ssize
     555             :          DO
     556           3 :             iter = iter + 1
     557           3 :             IF (iter > maxit) THEN
     558             :                ln = 0.0_dp
     559        1350 :                lp = 0.0_dp
     560             :                fail = .TRUE.
     561             :                GOTO 300
     562             :             END IF
     563           3 :             fun2 = 0.0_dp
     564           3 :             lam2 = lam1 - iter*step
     565          93 :             DO indf = 1, ndf
     566          93 :                fun2 = fun2 + eigval(indf)**2/(lam2 - eigval(indf))
     567             :             END DO
     568           3 :             fun2 = fun2 - lam2
     569        1353 :             IF (fun2*fun1 < 0.0_dp) THEN
     570             :                iter = 0
     571             :                DO
     572          75 :                   iter = iter + 1
     573          75 :                   IF (iter > maxit) THEN
     574             :                      ln = 0.0_dp
     575             :                      lp = 0.0_dp
     576             :                      fail = .TRUE.
     577             :                      GOTO 300
     578             :                   END IF
     579          75 :                   step = (lam1 + lam2)/2
     580          75 :                   fun3 = 0.0_dp
     581        2325 :                   DO indf = 1, ndf
     582        2325 :                      fun3 = fun3 + dg(indf)**2/(step - eigval(indf))
     583             :                   END DO
     584          75 :                   fun3 = fun3 - step
     585             : 
     586          75 :                   IF (ABS(step - lam2) < stol) THEN
     587             :                      ln = step
     588             :                      GOTO 200
     589             :                   END IF
     590             : 
     591          72 :                   IF (fun3*fun1 < stol) THEN
     592             :                      lam2 = step
     593             :                   ELSE
     594          72 :                      lam1 = step
     595             :                   END IF
     596             :                END DO
     597             :             END IF
     598             :          END DO
     599             : 
     600             : 200      CONTINUE
     601        1353 :          IF ((ln > eigval(iref)) .OR. ((ln > 0.0_dp) .AND. &
     602             :                                        (eigval(iref) > 0.0_dp))) THEN
     603             : 
     604           3 :             IF (.NOT. bisec) GOTO 100
     605             :             ln = 0.0_dp
     606             :             lp = 0.0_dp
     607             :             fail = .TRUE.
     608             :          END IF
     609             : 
     610             : 300      CONTINUE
     611             : 
     612        1350 :          IF (fail .AND. .NOT. set) THEN
     613           0 :             set = .TRUE.
     614           0 :             DO indf = 1, ndf
     615           0 :                eigval(indf) = eigval(indf)*work(indf)
     616             :             END DO
     617             :             CYCLE
     618             :          END IF
     619             : 
     620        1350 :          IF (.NOT. set) THEN
     621      120078 :             work(1:ndf) = one
     622             :          END IF
     623             : 
     624      120078 :          DO indf = 1, ndf
     625      120078 :             eigval(indf) = eigval(indf) - ln
     626             :          END DO
     627             :          EXIT
     628             :       END DO
     629             : 
     630        1350 :       CALL timestop(handle)
     631             : 
     632        1350 :    END SUBROUTINE rat_fun_opt
     633             : 
     634             : ! **************************************************************************************************
     635             : !> \brief ...
     636             : !> \param ndf ...
     637             : !> \param dx ...
     638             : !> \param dg ...
     639             : !> \param hess_mat ...
     640             : !> \param work ...
     641             : !> \param para_env ...
     642             : ! **************************************************************************************************
     643       11522 :    SUBROUTINE bfgs(ndf, dx, dg, hess_mat, work, para_env)
     644             :       INTEGER, INTENT(IN)                                :: ndf
     645             :       REAL(KIND=dp), INTENT(INOUT)                       :: dx(ndf), dg(ndf)
     646             :       TYPE(cp_fm_type), INTENT(IN)                       :: hess_mat
     647             :       REAL(KIND=dp), INTENT(INOUT)                       :: work(ndf)
     648             :       TYPE(mp_para_env_type), OPTIONAL, POINTER          :: para_env
     649             : 
     650             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'bfgs'
     651             :       REAL(KIND=dp), PARAMETER                           :: one = 1.0_dp, zero = 0.0_dp
     652             : 
     653             :       INTEGER                                            :: handle, i, j, k, l, ncol_local, &
     654             :                                                             nrow_local
     655        5761 :       INTEGER, DIMENSION(:), POINTER                     :: col_indices, row_indices
     656             :       REAL(KIND=dp)                                      :: DDOT, dxw, gdx
     657        5761 :       REAL(KIND=dp), CONTIGUOUS, DIMENSION(:, :), POINTER            :: local_hes
     658             : 
     659        5761 :       CALL timeset(routineN, handle)
     660             : 
     661             :       CALL cp_fm_get_info(hess_mat, row_indices=row_indices, col_indices=col_indices, &
     662        5761 :                           local_data=local_hes, nrow_local=nrow_local, ncol_local=ncol_local)
     663             : 
     664      595900 :       work = zero
     665      330859 :       DO i = 1, nrow_local
     666      325098 :          j = row_indices(i)
     667    54413263 :          DO k = 1, ncol_local
     668    54082404 :             l = col_indices(k)
     669    54407502 :             work(j) = work(j) + local_hes(i, k)*dx(l)
     670             :          END DO
     671             :       END DO
     672             : 
     673        5761 :       CALL para_env%sum(work)
     674             : 
     675        5761 :       gdx = DDOT(ndf, dg, 1, dx, 1)
     676        5761 :       gdx = one/gdx
     677        5761 :       dxw = DDOT(ndf, dx, 1, work, 1)
     678        5761 :       dxw = one/dxw
     679             : 
     680      330859 :       DO i = 1, nrow_local
     681      325098 :          j = row_indices(i)
     682    54413263 :          DO k = 1, ncol_local
     683    54082404 :             l = col_indices(k)
     684             :             local_hes(i, k) = local_hes(i, k) + gdx*dg(j)*dg(l) - &
     685    54407502 :                               dxw*work(j)*work(l)
     686             :          END DO
     687             :       END DO
     688             : 
     689        5761 :       CALL timestop(handle)
     690             : 
     691        5761 :    END SUBROUTINE bfgs
     692             : 
     693             : ! **************************************************************************************************
     694             : !> \brief ...
     695             : !> \param ndf ...
     696             : !> \param eigval ...
     697             : !> \param work ...
     698             : ! **************************************************************************************************
     699        1350 :    SUBROUTINE set_hes_eig(ndf, eigval, work)
     700             :       INTEGER, INTENT(IN)                                :: ndf
     701             :       REAL(KIND=dp), INTENT(INOUT)                       :: eigval(ndf), work(ndf)
     702             : 
     703             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'set_hes_eig'
     704             :       REAL(KIND=dp), PARAMETER                           :: max_neg = -0.5_dp, max_pos = 5.0_dp, &
     705             :                                                             min_eig = 0.005_dp, one = 1.0_dp
     706             : 
     707             :       INTEGER                                            :: handle, indf
     708             :       LOGICAL                                            :: neg
     709             : 
     710        1350 :       CALL timeset(routineN, handle)
     711             : 
     712      120078 :       DO indf = 1, ndf
     713      118728 :          IF (eigval(indf) < 0.0_dp) neg = .TRUE.
     714      120078 :          IF (eigval(indf) > 1000.0_dp) eigval(indf) = 1000.0_dp
     715             :       END DO
     716      120078 :       DO indf = 1, ndf
     717      120078 :          IF (eigval(indf) < 0.0_dp) THEN
     718           3 :             IF (eigval(indf) < max_neg) THEN
     719           1 :                eigval(indf) = max_neg
     720           2 :             ELSE IF (eigval(indf) > -min_eig) THEN
     721           1 :                eigval(indf) = -min_eig
     722             :             END IF
     723      118725 :          ELSE IF (eigval(indf) < 1000.0_dp) THEN
     724      118725 :             IF (eigval(indf) < min_eig) THEN
     725         427 :                eigval(indf) = min_eig
     726      118298 :             ELSE IF (eigval(indf) > max_pos) THEN
     727           0 :                eigval(indf) = max_pos
     728             :             END IF
     729             :          END IF
     730             :       END DO
     731             : 
     732      120078 :       DO indf = 1, ndf
     733      120078 :          IF (eigval(indf) < 0.0_dp) THEN
     734           3 :             work(indf) = -one
     735             :          ELSE
     736      118725 :             work(indf) = one
     737             :          END IF
     738             :       END DO
     739             : 
     740        1350 :       CALL timestop(handle)
     741             : 
     742        1350 :    END SUBROUTINE set_hes_eig
     743             : 
     744             : ! **************************************************************************************************
     745             : !> \brief ...
     746             : !> \param ndf ...
     747             : !> \param eigval ...
     748             : !> \param eigvec_mat ...
     749             : !> \param hess_tmp ...
     750             : !> \param dr ...
     751             : !> \param g ...
     752             : !> \param para_env ...
     753             : !> \param use_rfo ...
     754             : ! **************************************************************************************************
     755       20991 :    SUBROUTINE geoopt_get_step(ndf, eigval, eigvec_mat, hess_tmp, dr, g, para_env, use_rfo)
     756             : 
     757             :       INTEGER, INTENT(IN)                                :: ndf
     758             :       REAL(KIND=dp), INTENT(INOUT)                       :: eigval(ndf)
     759             :       TYPE(cp_fm_type), INTENT(IN)                       :: eigvec_mat, hess_tmp
     760             :       REAL(KIND=dp), INTENT(INOUT)                       :: dr(ndf), g(ndf)
     761             :       TYPE(mp_para_env_type), OPTIONAL, POINTER          :: para_env
     762             :       LOGICAL                                            :: use_rfo
     763             : 
     764             :       REAL(KIND=dp), PARAMETER                           :: one = 1.0_dp, zero = 0.0_dp
     765             : 
     766             :       INTEGER                                            :: i, indf, j, k, l, ncol_local, nrow_local
     767        6997 :       INTEGER, DIMENSION(:), POINTER                     :: col_indices, row_indices
     768        6997 :       REAL(KIND=dp), CONTIGUOUS, DIMENSION(:, :), POINTER            :: local_data
     769             :       TYPE(cp_fm_struct_type), POINTER                   :: matrix_struct
     770             :       TYPE(cp_fm_type)                          :: tmp
     771             : 
     772        6997 :       CALL cp_fm_to_fm(eigvec_mat, hess_tmp)
     773        6997 :       IF (use_rfo) THEN
     774      120078 :          DO indf = 1, ndf
     775      120078 :             eigval(indf) = one/eigval(indf)
     776             :          END DO
     777             :       ELSE
     778      564970 :          DO indf = 1, ndf
     779      564970 :             eigval(indf) = one/MAX(0.0001_dp, eigval(indf))
     780             :          END DO
     781             :       END IF
     782             : 
     783        6997 :       CALL cp_fm_column_scale(hess_tmp, eigval)
     784        6997 :       CALL cp_fm_get_info(eigvec_mat, matrix_struct=matrix_struct)
     785        6997 :       CALL cp_fm_create(tmp, matrix_struct, name="tmp")
     786             : 
     787        6997 :       CALL parallel_gemm("N", "T", ndf, ndf, ndf, one, hess_tmp, eigvec_mat, zero, tmp)
     788             : 
     789        6997 :       CALL cp_fm_transpose(tmp, hess_tmp)
     790        6997 :       CALL cp_fm_release(tmp)
     791             : 
     792             : !    ** New step **
     793             : 
     794             :       CALL cp_fm_get_info(hess_tmp, row_indices=row_indices, col_indices=col_indices, &
     795        6997 :                           local_data=local_data, nrow_local=nrow_local, ncol_local=ncol_local)
     796             : 
     797      685048 :       dr = 0.0_dp
     798      378301 :       DO i = 1, nrow_local
     799      371304 :          j = row_indices(i)
     800    64228585 :          DO k = 1, ncol_local
     801    63850284 :             l = col_indices(k)
     802    64221588 :             dr(j) = dr(j) - local_data(i, k)*g(l)
     803             :          END DO
     804             :       END DO
     805             : 
     806        6997 :       CALL para_env%sum(dr)
     807             : 
     808        6997 :    END SUBROUTINE geoopt_get_step
     809             : 
     810             : ! **************************************************************************************************
     811             : !> \brief ...
     812             : !> \param ndf ...
     813             : !> \param step ...
     814             : !> \param rad ...
     815             : !> \param rat ...
     816             : !> \param dr ...
     817             : !> \param output_unit ...
     818             : ! **************************************************************************************************
     819        6997 :    SUBROUTINE trust_radius(ndf, step, rad, rat, dr, output_unit)
     820             :       INTEGER, INTENT(IN)                                :: ndf
     821             :       REAL(KIND=dp), INTENT(INOUT)                       :: step, rad, rat, dr(ndf)
     822             :       INTEGER, INTENT(IN)                                :: output_unit
     823             : 
     824             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'trust_radius'
     825             :       REAL(KIND=dp), PARAMETER                           :: one = 1.0_dp
     826             : 
     827             :       INTEGER                                            :: handle
     828             :       REAL(KIND=dp)                                      :: scal
     829             : 
     830        6997 :       CALL timeset(routineN, handle)
     831             : 
     832      692045 :       step = MAXVAL(ABS(dr))
     833        6997 :       scal = MAX(one, rad/step)
     834             : 
     835        6997 :       IF (step > rad) THEN
     836         470 :          rat = rad/step
     837         470 :          CALL DSCAL(ndf, rat, dr, 1)
     838         470 :          step = rad
     839         470 :          IF (output_unit > 0) THEN
     840             :             WRITE (unit=output_unit, FMT="(/,T2,A,F8.5)") &
     841         219 :                " Step is scaled; Scaling factor = ", rat
     842         219 :             CALL m_flush(output_unit)
     843             :          END IF
     844             :       END IF
     845        6997 :       CALL timestop(handle)
     846             : 
     847        6997 :    END SUBROUTINE trust_radius
     848             : 
     849             : ! **************************************************************************************************
     850             : !> \brief ...
     851             : !> \param ndf ...
     852             : !> \param work ...
     853             : !> \param hess_mat ...
     854             : !> \param dr ...
     855             : !> \param g ...
     856             : !> \param conv ...
     857             : !> \param pred ...
     858             : !> \param para_env ...
     859             : ! **************************************************************************************************
     860       13994 :    SUBROUTINE energy_predict(ndf, work, hess_mat, dr, g, conv, pred, para_env)
     861             : 
     862             :       INTEGER, INTENT(IN)                                :: ndf
     863             :       REAL(KIND=dp), INTENT(INOUT)                       :: work(ndf)
     864             :       TYPE(cp_fm_type), INTENT(IN)                           :: hess_mat
     865             :       REAL(KIND=dp), INTENT(INOUT)                       :: dr(ndf), g(ndf)
     866             :       LOGICAL, INTENT(INOUT)                             :: conv
     867             :       REAL(KIND=dp), INTENT(INOUT)                       :: pred
     868             :       TYPE(mp_para_env_type), POINTER                    :: para_env
     869             : 
     870             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'energy_predict'
     871             :       REAL(KIND=dp), PARAMETER                           :: zero = 0.0_dp
     872             : 
     873             :       INTEGER                                            :: handle, i, j, k, l, ncol_local, &
     874             :                                                             nrow_local
     875        6997 :       INTEGER, DIMENSION(:), POINTER                     :: col_indices, row_indices
     876             :       REAL(KIND=dp)                                      :: DDOT, ener1, ener2
     877        6997 :       REAL(KIND=dp), CONTIGUOUS, DIMENSION(:, :), POINTER            :: local_data
     878             : 
     879        6997 :       CALL timeset(routineN, handle)
     880             : 
     881        6997 :       ener1 = DDOT(ndf, g, 1, dr, 1)
     882             : 
     883             :       CALL cp_fm_get_info(hess_mat, row_indices=row_indices, col_indices=col_indices, &
     884        6997 :                           local_data=local_data, nrow_local=nrow_local, ncol_local=ncol_local)
     885             : 
     886      685048 :       work = zero
     887      378301 :       DO i = 1, nrow_local
     888      371304 :          j = row_indices(i)
     889    64228585 :          DO k = 1, ncol_local
     890    63850284 :             l = col_indices(k)
     891    64221588 :             work(j) = work(j) + local_data(i, k)*dr(l)
     892             :          END DO
     893             :       END DO
     894             : 
     895        6997 :       CALL para_env%sum(work)
     896        6997 :       ener2 = DDOT(ndf, dr, 1, work, 1)
     897        6997 :       pred = ener1 + 0.5_dp*ener2
     898        6997 :       conv = .FALSE.
     899        6997 :       CALL timestop(handle)
     900             : 
     901        6997 :    END SUBROUTINE energy_predict
     902             : 
     903             : ! **************************************************************************************************
     904             : !> \brief ...
     905             : !> \param rat ...
     906             : !> \param rad ...
     907             : !> \param step ...
     908             : !> \param ediff ...
     909             : ! **************************************************************************************************
     910        1232 :    SUBROUTINE update_trust_rad(rat, rad, step, ediff)
     911             : 
     912             :       REAL(KIND=dp), INTENT(INOUT)                       :: rat, rad, step, ediff
     913             : 
     914             :       CHARACTER(LEN=*), PARAMETER                        :: routineN = 'update_trust_rad'
     915             :       REAL(KIND=dp), PARAMETER                           :: max_trust = 1.0_dp, min_trust = 0.1_dp
     916             : 
     917             :       INTEGER                                            :: handle
     918             : 
     919        1232 :       CALL timeset(routineN, handle)
     920             : 
     921        1232 :       IF (rat > 4.0_dp) THEN
     922           0 :          IF (ediff < 0.0_dp) THEN
     923           0 :             rad = step*0.5_dp
     924             :          ELSE
     925           0 :             rad = step*0.25_dp
     926             :          END IF
     927        1232 :       ELSE IF (rat > 2.0_dp) THEN
     928           0 :          IF (ediff < 0.0_dp) THEN
     929           0 :             rad = step*0.75_dp
     930             :          ELSE
     931           0 :             rad = step*0.5_dp
     932             :          END IF
     933        1232 :       ELSE IF (rat > 4.0_dp/3.0_dp) THEN
     934           0 :          IF (ediff < 0.0_dp) THEN
     935           0 :             rad = step
     936             :          ELSE
     937           0 :             rad = step*0.75_dp
     938             :          END IF
     939        1232 :       ELSE IF (rat > 10.0_dp/9.0_dp) THEN
     940           0 :          IF (ediff < 0.0_dp) THEN
     941           0 :             rad = step*1.25_dp
     942             :          ELSE
     943           0 :             rad = step
     944             :          END IF
     945        1232 :       ELSE IF (rat > 0.9_dp) THEN
     946          99 :          IF (ediff < 0.0_dp) THEN
     947          99 :             rad = step*1.5_dp
     948             :          ELSE
     949           0 :             rad = step*1.25_dp
     950             :          END IF
     951        1133 :       ELSE IF (rat > 0.75_dp) THEN
     952         232 :          IF (ediff < 0.0_dp) THEN
     953         230 :             rad = step*1.25_dp
     954             :          ELSE
     955           2 :             rad = step
     956             :          END IF
     957         901 :       ELSE IF (rat > 0.5_dp) THEN
     958         228 :          IF (ediff < 0.0_dp) THEN
     959         223 :             rad = step
     960             :          ELSE
     961           5 :             rad = step*0.75_dp
     962             :          END IF
     963         673 :       ELSE IF (rat > 0.25_dp) THEN
     964          39 :          IF (ediff < 0.0_dp) THEN
     965          39 :             rad = step*0.75_dp
     966             :          ELSE
     967           0 :             rad = step*0.5_dp
     968             :          END IF
     969         634 :       ELSE IF (ediff < 0.0_dp) THEN
     970         634 :          rad = step*0.5_dp
     971             :       ELSE
     972           0 :          rad = step*0.25_dp
     973             :       END IF
     974             : 
     975        1232 :       rad = MAX(rad, min_trust)
     976        1232 :       rad = MIN(rad, max_trust)
     977        1232 :       CALL timestop(handle)
     978             : 
     979        1232 :    END SUBROUTINE update_trust_rad
     980             : 
     981             : ! **************************************************************************************************
     982             : 
     983             : ! **************************************************************************************************
     984             : !> \brief ...
     985             : !> \param geo_section ...
     986             : !> \param hess_mat ...
     987             : !> \param logger ...
     988             : ! **************************************************************************************************
     989        5013 :    SUBROUTINE write_bfgs_hessian(geo_section, hess_mat, logger)
     990             : 
     991             :       TYPE(section_vals_type), POINTER                   :: geo_section
     992             :       TYPE(cp_fm_type), INTENT(IN)                          :: hess_mat
     993             :       TYPE(cp_logger_type), POINTER                      :: logger
     994             : 
     995             :       CHARACTER(LEN=*), PARAMETER :: routineN = 'write_bfgs_hessian'
     996             : 
     997             :       INTEGER                                            :: handle, hesunit
     998             : 
     999        5013 :       CALL timeset(routineN, handle)
    1000             : 
    1001             :       hesunit = cp_print_key_unit_nr(logger, geo_section, "BFGS%RESTART", &
    1002             :                                      extension=".Hessian", file_form="UNFORMATTED", file_action="WRITE", &
    1003        5013 :                                      file_position="REWIND")
    1004             : 
    1005        5013 :       CALL cp_fm_write_unformatted(hess_mat, hesunit)
    1006             : 
    1007        5013 :       CALL cp_print_key_finished_output(hesunit, logger, geo_section, "BFGS%RESTART")
    1008             : 
    1009        5013 :       CALL timestop(handle)
    1010             : 
    1011        5013 :    END SUBROUTINE write_bfgs_hessian
    1012             : ! **************************************************************************************************
    1013             : 
    1014             : ! **************************************************************************************************
    1015             : !> \brief ...
    1016             : !> \param force_env ...
    1017             : !> \param hess_mat ...
    1018             : ! **************************************************************************************************
    1019        1034 :    SUBROUTINE construct_initial_hess(force_env, hess_mat)
    1020             : 
    1021             :       TYPE(force_env_type), POINTER                      :: force_env
    1022             :       TYPE(cp_fm_type), INTENT(IN)                          :: hess_mat
    1023             : 
    1024             :       INTEGER                                            :: i, iat_col, iat_row, iglobal, iind, j, &
    1025             :                                                             jat_row, jglobal, jind, k, natom, &
    1026             :                                                             ncol_local, nrow_local, z
    1027        1034 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: at_row
    1028        1034 :       INTEGER, DIMENSION(:), POINTER                     :: col_indices, row_indices
    1029             :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :)        :: d_ij, rho_ij
    1030             :       REAL(KIND=dp), ALLOCATABLE, DIMENSION(:, :, :)     :: r_ij
    1031             :       REAL(KIND=dp), DIMENSION(3, 3)                     :: alpha, r0
    1032        1034 :       REAL(KIND=dp), CONTIGUOUS, DIMENSION(:, :), POINTER            :: fixed, local_data
    1033             :       TYPE(cell_type), POINTER                           :: cell
    1034             :       TYPE(cp_subsys_type), POINTER                      :: subsys
    1035             :       TYPE(particle_list_type), POINTER                  :: particles
    1036             : 
    1037        1034 :       CALL force_env_get(force_env=force_env, subsys=subsys, cell=cell)
    1038             :       CALL cp_subsys_get(subsys, &
    1039        1034 :                          particles=particles)
    1040             : 
    1041        4136 :       alpha(1, :) = (/1._dp, 0.3949_dp, 0.3949_dp/)
    1042        4136 :       alpha(2, :) = (/0.3494_dp, 0.2800_dp, 0.2800_dp/)
    1043        4136 :       alpha(3, :) = (/0.3494_dp, 0.2800_dp, 0.1800_dp/)
    1044             : 
    1045        4136 :       r0(1, :) = (/1.35_dp, 2.10_dp, 2.53_dp/)
    1046        4136 :       r0(2, :) = (/2.10_dp, 2.87_dp, 3.40_dp/)
    1047        4136 :       r0(3, :) = (/2.53_dp, 3.40_dp, 3.40_dp/)
    1048             : 
    1049             :       CALL cp_fm_get_info(hess_mat, row_indices=row_indices, col_indices=col_indices, &
    1050        1034 :                           local_data=local_data, nrow_local=nrow_local, ncol_local=ncol_local)
    1051        1034 :       natom = particles%n_els
    1052        3102 :       ALLOCATE (at_row(natom))
    1053        4136 :       ALLOCATE (rho_ij(natom, natom))
    1054        3102 :       ALLOCATE (d_ij(natom, natom))
    1055        5170 :       ALLOCATE (r_ij(natom, natom, 3))
    1056        3102 :       ALLOCATE (fixed(3, natom))
    1057       73554 :       fixed = 1.0_dp
    1058        1034 :       CALL fix_atom_control(force_env, fixed)
    1059        4136 :       DO i = 1, 3
    1060      112916 :          CALL hess_mat%matrix_struct%para_env%min(fixed(i, :))
    1061             :       END DO
    1062      918898 :       rho_ij = 0
    1063             :       !XXXX insert proper rows !XXX
    1064       19164 :       at_row = 3
    1065       19164 :       DO i = 1, natom
    1066       18130 :          CALL get_atomic_kind(atomic_kind=particles%els(i)%atomic_kind, z=z)
    1067       18130 :          IF (z .LE. 10) at_row(i) = 2
    1068       37294 :          IF (z .LE. 2) at_row(i) = 1
    1069             :       END DO
    1070       18130 :       DO i = 2, natom
    1071       17096 :          iat_row = at_row(i)
    1072      458932 :          DO j = 1, i - 1
    1073      440802 :             jat_row = at_row(j)
    1074             :             !pbc for a distance vector
    1075     1763208 :             r_ij(j, i, :) = pbc(particles%els(i)%r, particles%els(j)%r, cell)
    1076     1763208 :             r_ij(i, j, :) = -r_ij(j, i, :)
    1077     1763208 :             d_ij(j, i) = SQRT(DOT_PRODUCT(r_ij(j, i, :), r_ij(j, i, :)))
    1078      440802 :             d_ij(i, j) = d_ij(j, i)
    1079      440802 :             rho_ij(j, i) = EXP(alpha(jat_row, iat_row)*(r0(jat_row, iat_row)**2 - d_ij(j, i)**2))
    1080      457898 :             rho_ij(i, j) = rho_ij(j, i)
    1081             :          END DO
    1082             :       END DO
    1083       55424 :       DO i = 1, ncol_local
    1084       54390 :          iglobal = col_indices(i)
    1085       54390 :          iind = MOD(iglobal - 1, 3) + 1
    1086       54390 :          iat_col = (iglobal + 2)/3
    1087       54390 :          IF (iat_col .GT. natom) CYCLE
    1088     4203857 :          DO j = 1, nrow_local
    1089     4148433 :             jglobal = row_indices(j)
    1090     4148433 :             jind = MOD(jglobal - 1, 3) + 1
    1091     4148433 :             iat_row = (jglobal + 2)/3
    1092     4148433 :             IF (iat_row .GT. natom) CYCLE
    1093     4148433 :             IF (iat_row .NE. iat_col) THEN
    1094     4060098 :                IF (d_ij(iat_row, iat_col) .LT. 6.0_dp) &
    1095             :                   local_data(j, i) = local_data(j, i) + &
    1096      362736 :                                      angle_second_deriv(r_ij, d_ij, rho_ij, iind, jind, iat_col, iat_row, natom)
    1097             :             ELSE
    1098             :                local_data(j, i) = local_data(j, i) + &
    1099       88335 :                                   angle_second_deriv(r_ij, d_ij, rho_ij, iind, jind, iat_col, iat_row, natom)
    1100             :             END IF
    1101     4148433 :             IF (iat_col .NE. iat_row) THEN
    1102     4060098 :                IF (d_ij(iat_row, iat_col) .LT. 6.0_dp) &
    1103             :                   local_data(j, i) = local_data(j, i) - &
    1104             :                                      dist_second_deriv(r_ij(iat_col, iat_row, :), &
    1105     2539152 :                                                        iind, jind, d_ij(iat_row, iat_col), rho_ij(iat_row, iat_col))
    1106             :             ELSE
    1107     4236768 :                DO k = 1, natom
    1108     4148433 :                   IF (k == iat_col) CYCLE
    1109     4060098 :                   IF (d_ij(iat_row, k) .LT. 6.0_dp) &
    1110             :                      local_data(j, i) = local_data(j, i) + &
    1111             :                                         dist_second_deriv(r_ij(iat_col, k, :), &
    1112     2627487 :                                                           iind, jind, d_ij(iat_row, k), rho_ij(iat_row, k))
    1113             :                END DO
    1114             :             END IF
    1115     4202823 :             IF (fixed(jind, iat_row) .LT. 0.5_dp .OR. fixed(iind, iat_col) .LT. 0.5_dp) THEN
    1116       10161 :                local_data(j, i) = 0.0_dp
    1117       10161 :                IF (jind == iind .AND. iat_row == iat_col) local_data(j, i) = 1.0_dp
    1118             :             END IF
    1119             :          END DO
    1120             :       END DO
    1121        1034 :       DEALLOCATE (fixed)
    1122        1034 :       DEALLOCATE (rho_ij)
    1123        1034 :       DEALLOCATE (d_ij)
    1124        1034 :       DEALLOCATE (r_ij)
    1125        1034 :       DEALLOCATE (at_row)
    1126             : 
    1127        2068 :    END SUBROUTINE construct_initial_hess
    1128             : 
    1129             : ! **************************************************************************************************
    1130             : !> \brief ...
    1131             : !> \param r1 ...
    1132             : !> \param i ...
    1133             : !> \param j ...
    1134             : !> \param d ...
    1135             : !> \param rho ...
    1136             : !> \return ...
    1137             : ! **************************************************************************************************
    1138      725472 :    FUNCTION dist_second_deriv(r1, i, j, d, rho) RESULT(deriv)
    1139             :       REAL(KIND=dp), DIMENSION(3)                        :: r1
    1140             :       INTEGER                                            :: i, j
    1141             :       REAL(KIND=dp)                                      :: d, rho, deriv
    1142             : 
    1143      725472 :       deriv = 0.45_dp*rho*(r1(i)*r1(j))/d**2
    1144      725472 :    END FUNCTION
    1145             : 
    1146             : ! **************************************************************************************************
    1147             : !> \brief ...
    1148             : !> \param r_ij ...
    1149             : !> \param d_ij ...
    1150             : !> \param rho_ij ...
    1151             : !> \param idir ...
    1152             : !> \param jdir ...
    1153             : !> \param iat_der ...
    1154             : !> \param jat_der ...
    1155             : !> \param natom ...
    1156             : !> \return ...
    1157             : ! **************************************************************************************************
    1158      451071 :    FUNCTION angle_second_deriv(r_ij, d_ij, rho_ij, idir, jdir, iat_der, jat_der, natom) RESULT(deriv)
    1159             :       REAL(KIND=dp), DIMENSION(:, :, :)                  :: r_ij
    1160             :       REAL(KIND=dp), DIMENSION(:, :)                     :: d_ij, rho_ij
    1161             :       INTEGER                                            :: idir, jdir, iat_der, jat_der, natom
    1162             :       REAL(KIND=dp)                                      :: deriv
    1163             : 
    1164             :       INTEGER                                            :: i, iat, idr, j, jat, jdr
    1165             :       REAL(KIND=dp)                                      :: d12, d23, d31, D_mat(3, 2), denom1, &
    1166             :                                                             denom2, denom3, ka1, ka2, ka3, rho12, &
    1167             :                                                             rho23, rho31, rsst1, rsst2, rsst3
    1168             :       REAL(KIND=dp), DIMENSION(3)                        :: r12, r23, r31
    1169             : 
    1170      451071 :       deriv = 0._dp
    1171      451071 :       IF (iat_der == jat_der) THEN
    1172     4148433 :          DO i = 1, natom - 1
    1173     4060098 :             IF (rho_ij(iat_der, i) .LT. 0.00001) CYCLE
    1174    25786170 :             DO j = i + 1, natom
    1175    24745131 :                IF (rho_ij(iat_der, j) .LT. 0.00001) CYCLE
    1176     6095196 :                IF (i == iat_der .OR. j == iat_der) CYCLE
    1177     6095196 :                IF (iat_der .LT. i .OR. iat_der .GT. j) THEN
    1178    39212640 :                   r12 = r_ij(iat_der, i, :); r23 = r_ij(i, j, :); r31 = r_ij(j, iat_der, :)
    1179     3921264 :                   d12 = d_ij(iat_der, i); d23 = d_ij(i, j); d31 = d_ij(j, iat_der)
    1180     3921264 :                   rho12 = rho_ij(iat_der, i); rho23 = rho_ij(i, j); rho31 = rho_ij(j, iat_der)
    1181             :                ELSE
    1182    21739320 :                   r12 = r_ij(iat_der, j, :); r23 = r_ij(j, i, :); r31 = r_ij(i, iat_der, :)
    1183     2173932 :                   d12 = d_ij(iat_der, j); d23 = d_ij(j, i); d31 = d_ij(i, iat_der)
    1184     2173932 :                   rho12 = rho_ij(iat_der, j); rho23 = rho_ij(j, i); rho31 = rho_ij(i, iat_der)
    1185             :                END IF
    1186     6095196 :                ka1 = 0.15_dp*rho12*rho23; ka2 = 0.15_dp*rho23*rho31; ka3 = 0.15_dp*rho31*rho12
    1187    60951960 :                rsst1 = DOT_PRODUCT(r12, r23); rsst2 = DOT_PRODUCT(r23, r31); rsst3 = DOT_PRODUCT(r31, r12)
    1188     6095196 :                denom1 = 1.0_dp - rsst1**2/(d12**2*d23**2); denom2 = 1.0_dp - rsst2**2/(d23**2*d31**2)
    1189     6095196 :                denom3 = 1.0_dp - rsst3**2/(d31**2*d12**2)
    1190     6095196 :                denom1 = SIGN(1.0_dp, denom1)*MAX(ABS(denom1), 0.01_dp)
    1191     6095196 :                denom2 = SIGN(1.0_dp, denom2)*MAX(ABS(denom2), 0.01_dp)
    1192     6095196 :                denom3 = SIGN(1.0_dp, denom3)*MAX(ABS(denom3), 0.01_dp)
    1193     6095196 :                D_mat(1, 1) = r23(idir)/(d12*d23) - rsst1*r12(idir)/(d12**3*d23)
    1194     6095196 :                D_mat(1, 2) = r23(jdir)/(d12*d23) - rsst1*r12(jdir)/(d12**3*d23)
    1195     6095196 :                D_mat(2, 1) = -r23(idir)/(d23*d31) + rsst2*r31(idir)/(d23*d31**3)
    1196     6095196 :                D_mat(2, 2) = -r23(jdir)/(d23*d31) + rsst2*r31(jdir)/(d23*d31**3)
    1197             :                D_mat(3, 1) = (r31(idir) - r12(idir))/(d31*d12) + rsst3*r31(idir)/(d31**3*d12) - &
    1198     6095196 :                              rsst3*r12(idir)/(d31*d12**3)
    1199             :                D_mat(3, 2) = (r31(jdir) - r12(jdir))/(d31*d12) + rsst3*r31(jdir)/(d31**3*d12) - &
    1200     6095196 :                              rsst3*r12(jdir)/(d31*d12**3)
    1201     6095196 :                IF (ABS(denom1) .LE. 0.011_dp) D_mat(1, 1) = 0.0_dp
    1202     6095196 :                IF (ABS(denom2) .LE. 0.011_dp) D_mat(2, 1) = 0.0_dp
    1203     6095196 :                IF (ABS(denom3) .LE. 0.011_dp) D_mat(3, 1) = 0.0_dp
    1204             :                deriv = deriv + ka1*D_mat(1, 1)*D_mat(1, 2)/denom1 + &
    1205             :                        ka2*D_mat(2, 1)*D_mat(2, 2)/denom2 + &
    1206    28805229 :                        ka3*D_mat(3, 1)*D_mat(3, 2)/denom3
    1207             : 
    1208             :             END DO
    1209             :          END DO
    1210             :       ELSE
    1211    10515330 :          DO i = 1, natom
    1212    10152594 :             IF (i == iat_der .OR. i == jat_der) CYCLE
    1213     9427122 :             IF (jat_der .LT. iat_der) THEN
    1214     4713561 :                iat = jat_der; jat = iat_der; idr = jdir; jdr = idir
    1215             :             ELSE
    1216     4713561 :                iat = iat_der; jat = jat_der; idr = idir; jdr = jdir
    1217             :             END IF
    1218     9427122 :             IF (jat .LT. i .OR. iat .GT. i) THEN
    1219    71098920 :                r12 = r_ij(iat, jat, :); r23 = r_ij(jat, i, :); r31 = r_ij(i, iat, :)
    1220     7109892 :                d12 = d_ij(iat, jat); d23 = d_ij(jat, i); d31 = d_ij(i, iat)
    1221     7109892 :                rho12 = rho_ij(iat, jat); rho23 = rho_ij(jat, i); rho31 = rho_ij(i, iat)
    1222             :             ELSE
    1223    23172300 :                r12 = r_ij(iat, i, :); r23 = r_ij(i, jat, :); r31 = r_ij(jat, iat, :)
    1224     2317230 :                d12 = d_ij(iat, i); d23 = d_ij(i, jat); d31 = d_ij(jat, iat)
    1225     2317230 :                rho12 = rho_ij(iat, i); rho23 = rho_ij(i, jat); rho31 = rho_ij(jat, iat)
    1226             :             END IF
    1227     9427122 :             ka1 = 0.15_dp*rho12*rho23; ka2 = 0.15_dp*rho23*rho31; ka3 = 0.15_dp*rho31*rho12
    1228    94271220 :             rsst1 = DOT_PRODUCT(r12, r23); rsst2 = DOT_PRODUCT(r23, r31); rsst3 = DOT_PRODUCT(r31, r12)
    1229     9427122 :             denom1 = 1.0_dp - rsst1**2/(d12**2*d23**2); denom2 = 1.0_dp - rsst2**2/(d23**2*d31**2)
    1230     9427122 :             denom3 = 1.0_dp - rsst3**2/(d31**2*d12**2)
    1231     9427122 :             denom1 = SIGN(1.0_dp, denom1)*MAX(ABS(denom1), 0.01_dp)
    1232     9427122 :             denom2 = SIGN(1.0_dp, denom2)*MAX(ABS(denom2), 0.01_dp)
    1233     9427122 :             denom3 = SIGN(1.0_dp, denom3)*MAX(ABS(denom3), 0.01_dp)
    1234     9427122 :             D_mat(1, 1) = r23(idr)/(d12*d23) - rsst1*r12(idr)/(d12**3*d23)
    1235     9427122 :             D_mat(2, 1) = -r23(idr)/(d23*d31) + rsst2*r31(idr)/(d23*d31**3)
    1236             :             D_mat(3, 1) = (r31(idr) - r12(idr))/(d31*d12) + rsst3*r31(idr)/(d31**3*d12) - &
    1237     9427122 :                           rsst3*r12(idr)/(d31*d12**3)
    1238     9427122 :             IF (jat .LT. i .OR. iat .GT. i) THEN
    1239             :                D_mat(1, 2) = (r12(jdr) - r23(jdr))/(d12*d23) + rsst1*r12(jdr)/(d12**3*d23) - &
    1240     7109892 :                              rsst1*r23(jdr)/(d12*d23**3)
    1241     7109892 :                D_mat(2, 2) = r31(jdr)/(d23*d31) - rsst2*r23(jdr)/(d23**3*d31)
    1242     7109892 :                D_mat(3, 2) = -r31(jdr)/(d31*d12) + rsst3*r12(jdr)/(d31*d12**3)
    1243             :             ELSE
    1244     2317230 :                D_mat(1, 2) = -r12(jdr)/(d12*d23) + rsst1*r23(jdr)/(d12*d23**3)
    1245             :                D_mat(2, 2) = (r23(jdr) - r31(jdr))/(d23*d31) + rsst2*r23(jdr)/(d23**3*d31) - &
    1246     2317230 :                              rsst2*r31(jdr)/(d23*d31**3)
    1247     2317230 :                D_mat(3, 2) = r12(jdr)/(d31*d12) - rsst3*r31(jdr)/(d31**3*d12)
    1248             :             END IF
    1249     9427122 :             IF (ABS(denom1) .LE. 0.011_dp) D_mat(1, 1) = 0.0_dp
    1250     9427122 :             IF (ABS(denom2) .LE. 0.011_dp) D_mat(2, 1) = 0.0_dp
    1251     9427122 :             IF (ABS(denom3) .LE. 0.011_dp) D_mat(3, 1) = 0.0_dp
    1252             : 
    1253             :             deriv = deriv + ka1*D_mat(1, 1)*D_mat(1, 2)/denom1 + &
    1254             :                     ka2*D_mat(2, 1)*D_mat(2, 2)/denom2 + &
    1255    10515330 :                     ka3*D_mat(3, 1)*D_mat(3, 2)/denom3
    1256             :          END DO
    1257             :       END IF
    1258      451071 :       deriv = 0.25_dp*deriv
    1259             : 
    1260      451071 :    END FUNCTION angle_second_deriv
    1261             : 
    1262             : END MODULE bfgs_optimizer

Generated by: LCOV version 1.15