LCOV - code coverage report
Current view: top level - src - atom_upf.F (source / functions) Hit Total Coverage
Test: CP2K Regtests (git:2fce0f8) Lines: 331 396 83.6 %
Date: 2024-12-21 06:28:57 Functions: 10 13 76.9 %

          Line data    Source code
       1             : !--------------------------------------------------------------------------------------------------!
       2             : !   CP2K: A general program to perform molecular dynamics simulations                              !
       3             : !   Copyright 2000-2024 CP2K developers group <https://cp2k.org>                                   !
       4             : !                                                                                                  !
       5             : !   SPDX-License-Identifier: GPL-2.0-or-later                                                      !
       6             : !--------------------------------------------------------------------------------------------------!
       7             : 
       8             : ! **************************************************************************************************
       9             : !> \brief Routines that process Quantum Espresso UPF files.
      10             : !> \par History
      11             : !>    * 07.2018 CP2K-SIRIUS interface [Juerg Hutter]
      12             : !>    * 02.2016 created [Juerg Hutter]
      13             : ! **************************************************************************************************
      14             : MODULE atom_upf
      15             :    USE cp_parser_methods,               ONLY: parser_get_next_line,&
      16             :                                               parser_get_object,&
      17             :                                               parser_test_next_token
      18             :    USE cp_parser_types,                 ONLY: cp_parser_type,&
      19             :                                               parser_create,&
      20             :                                               parser_release
      21             :    USE kinds,                           ONLY: default_string_length,&
      22             :                                               dp
      23             :    USE periodic_table,                  ONLY: get_ptable_info,&
      24             :                                               ptable
      25             : #include "./base/base_uses.f90"
      26             : 
      27             :    IMPLICIT NONE
      28             : 
      29             :    ! use same value as in atom_types!
      30             :    INTEGER, PARAMETER                                :: lmat = 3
      31             : 
      32             :    TYPE atom_upfpot_type
      33             :       CHARACTER(LEN=2)                               :: symbol = ""
      34             :       CHARACTER(LEN=default_string_length)           :: pname = ""
      35             :       INTEGER, DIMENSION(0:lmat)                     :: econf = 0
      36             :       REAL(dp)                                       :: zion = 0.0_dp
      37             :       CHARACTER(LEN=default_string_length)           :: version = ""
      38             :       CHARACTER(LEN=default_string_length)           :: filename = ""
      39             :       ! <INFO>
      40             :       INTEGER                                        :: maxinfo = 100
      41             :       CHARACTER(LEN=default_string_length), DIMENSION(100) &
      42             :          :: info = ""
      43             :       ! <HEADER>
      44             :       CHARACTER(LEN=default_string_length)           :: generated = ""
      45             :       CHARACTER(LEN=default_string_length)           :: author = ""
      46             :       CHARACTER(LEN=default_string_length)           :: date = ""
      47             :       CHARACTER(LEN=default_string_length)           :: comment = ""
      48             :       CHARACTER(LEN=4)                               :: pseudo_type = ""
      49             :       CHARACTER(LEN=15)                              :: relativistic = ""
      50             :       CHARACTER(LEN=default_string_length)           :: functional = ""
      51             :       LOGICAL                                        :: is_ultrasoft = .FALSE.
      52             :       LOGICAL                                        :: is_paw = .FALSE.
      53             :       LOGICAL                                        :: is_coulomb = .FALSE.
      54             :       LOGICAL                                        :: has_so = .FALSE.
      55             :       LOGICAL                                        :: has_wfc = .FALSE.
      56             :       LOGICAL                                        :: has_gipaw = .FALSE.
      57             :       LOGICAL                                        :: paw_as_gipaw = .FALSE.
      58             :       LOGICAL                                        :: core_correction = .FALSE.
      59             :       REAL(dp)                                       :: total_psenergy = 0.0_dp
      60             :       REAL(dp)                                       :: wfc_cutoff = 0.0_dp
      61             :       REAL(dp)                                       :: rho_cutoff = 0.0_dp
      62             :       INTEGER                                        :: l_max = -100
      63             :       INTEGER                                        :: l_max_rho = -1
      64             :       INTEGER                                        :: l_local = -1
      65             :       INTEGER                                        :: mesh_size = -1
      66             :       INTEGER                                        :: number_of_wfc = -1
      67             :       INTEGER                                        :: number_of_proj = -1
      68             :       ! <MESH>
      69             :       REAL(dp)                                       :: dx = 0.0_dp
      70             :       REAL(dp)                                       :: xmin = 0.0_dp
      71             :       REAL(dp)                                       :: rmax = 0.0_dp
      72             :       REAL(dp)                                       :: zmesh = 0.0_dp
      73             :       REAL(dp), DIMENSION(:), ALLOCATABLE            :: r, rab
      74             :       ! <NLCC>
      75             :       REAL(dp), DIMENSION(:), ALLOCATABLE            :: rho_nlcc
      76             :       ! <LOCAL>
      77             :       REAL(dp), DIMENSION(:), ALLOCATABLE            :: vlocal
      78             :       ! <NONLOCAL>
      79             :       REAL(dp), DIMENSION(:, :), ALLOCATABLE         :: dion
      80             :       REAL(dp), DIMENSION(:, :), ALLOCATABLE         :: beta
      81             :       INTEGER, DIMENSION(:), ALLOCATABLE             :: lbeta
      82             :       ! <SEMILOCAL>
      83             :       REAL(dp), DIMENSION(:, :), ALLOCATABLE         :: vsemi
      84             :    END TYPE atom_upfpot_type
      85             : 
      86             :    PRIVATE
      87             :    PUBLIC  :: atom_read_upf, atom_upfpot_type, atom_release_upf
      88             : 
      89             :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'atom_upf'
      90             : 
      91             : ! **************************************************************************************************
      92             : 
      93             : CONTAINS
      94             : 
      95             : ! **************************************************************************************************
      96             : !> \brief ...
      97             : !> \param pot ...
      98             : !> \param upf_filename ...
      99             : !> \param read_header ...
     100             : ! **************************************************************************************************
     101          16 :    SUBROUTINE atom_read_upf(pot, upf_filename, read_header)
     102             : 
     103             :       TYPE(atom_upfpot_type)                             :: pot
     104             :       CHARACTER(len=*), INTENT(IN)                       :: upf_filename
     105             :       LOGICAL, INTENT(IN), OPTIONAL                      :: read_header
     106             : 
     107             :       CHARACTER(LEN=2)                                   :: symbol
     108             :       INTEGER                                            :: l, ncore, nel
     109             :       LOGICAL                                            :: readall
     110             : 
     111          16 :       IF (PRESENT(read_header)) THEN
     112           0 :          readall = .NOT. read_header
     113             :       ELSE
     114          16 :          readall = .TRUE.
     115             :       END IF
     116             : 
     117             :       ! filename
     118          16 :       pot%filename = ADJUSTL(TRIM(upf_filename))
     119             : 
     120             :       ! Ignore json potentials as SIRIUS will parse those on its own.
     121          16 :       l = LEN_TRIM(pot%filename)
     122          16 :       IF (pot%filename(l - 4:l) == '.json') THEN
     123           0 :          pot%zion = 0.0
     124           0 :          RETURN
     125             :       END IF
     126             : 
     127          16 :       CALL atom_read_upf_v2(pot, upf_filename, readall)
     128             : 
     129             :       ! set up econf
     130          80 :       IF (SUM(pot%econf) == 0) THEN
     131          16 :          symbol = ADJUSTL(TRIM(pot%symbol))
     132          16 :          CALL get_ptable_info(symbol, number=ncore)
     133          80 :          pot%econf(0:3) = ptable(ncore)%e_conv(0:3)
     134          16 :          nel = NINT(ncore - pot%zion)
     135           0 :          SELECT CASE (nel)
     136             :          CASE DEFAULT
     137           0 :             CPABORT("Unknown Core State")
     138             :          CASE (0)
     139             :             ! no core electron
     140             :          CASE (2)
     141          50 :             pot%econf(0:3) = pot%econf(0:3) - ptable(2)%e_conv(0:3)
     142             :          CASE (10)
     143           0 :             pot%econf(0:3) = pot%econf(0:3) - ptable(10)%e_conv(0:3)
     144             :          CASE (18)
     145           0 :             pot%econf(0:3) = pot%econf(0:3) - ptable(18)%e_conv(0:3)
     146             :          CASE (28)
     147           0 :             pot%econf(0:3) = pot%econf(0:3) - ptable(18)%e_conv(0:3)
     148           0 :             pot%econf(2) = pot%econf(2) - 10
     149             :          CASE (36)
     150           0 :             pot%econf(0:3) = pot%econf(0:3) - ptable(36)%e_conv(0:3)
     151             :          CASE (46)
     152           0 :             pot%econf(0:3) = pot%econf(0:3) - ptable(36)%e_conv(0:3)
     153           0 :             pot%econf(2) = pot%econf(2) - 10
     154             :          CASE (54)
     155           0 :             pot%econf(0:3) = pot%econf(0:3) - ptable(54)%e_conv(0:3)
     156             :          CASE (60)
     157           0 :             pot%econf(0:3) = pot%econf(0:3) - ptable(36)%e_conv(0:3)
     158           0 :             pot%econf(2) = pot%econf(2) - 10
     159           0 :             pot%econf(3) = pot%econf(3) - 14
     160             :          CASE (68)
     161           0 :             pot%econf(0:3) = pot%econf(0:3) - ptable(54)%e_conv(0:3)
     162           0 :             pot%econf(3) = pot%econf(3) - 14
     163             :          CASE (78)
     164           0 :             pot%econf(0:3) = pot%econf(0:3) - ptable(54)%e_conv(0:3)
     165           0 :             pot%econf(2) = pot%econf(2) - 10
     166          16 :             pot%econf(3) = pot%econf(3) - 14
     167             :          END SELECT
     168             :          !
     169          80 :          CPASSERT(ALL(pot%econf >= 0))
     170             :       END IF
     171             : 
     172             :       ! name
     173          16 :       IF (pot%pname == "") THEN
     174          16 :          pot%pname = ADJUSTL(TRIM(pot%symbol))
     175             :       END IF
     176             : 
     177             :    END SUBROUTINE atom_read_upf
     178             : 
     179             : ! **************************************************************************************************
     180             : !> \brief ...
     181             : !> \param pot ...
     182             : !> \param upf_filename ...
     183             : !> \param readall ...
     184             : ! **************************************************************************************************
     185          16 :    SUBROUTINE atom_read_upf_v2(pot, upf_filename, readall)
     186             : 
     187             :       TYPE(atom_upfpot_type)                             :: pot
     188             :       CHARACTER(len=*), INTENT(IN)                       :: upf_filename
     189             :       LOGICAL, INTENT(IN)                                :: readall
     190             : 
     191             :       CHARACTER(LEN=default_string_length)               :: nametag
     192             :       INTEGER                                            :: ib, ntag
     193             :       LOGICAL                                            :: at_end
     194             :       TYPE(cp_parser_type)                               :: parser
     195             : 
     196          16 :       ntag = 0
     197          16 :       CALL parser_create(parser, upf_filename)
     198             :       DO
     199             :          at_end = .FALSE.
     200       10788 :          CALL parser_get_next_line(parser, 1, at_end)
     201       10788 :          IF (at_end) EXIT
     202       10788 :          CALL parser_get_object(parser, nametag, lower_to_upper=.TRUE.)
     203       10788 :          IF (nametag(1:1) /= "<") CYCLE
     204         302 :          IF (ntag == 0) THEN
     205             :             ! we are looking for UPF tag
     206          16 :             IF (nametag(2:4) == "UPF") THEN
     207          16 :                CALL parser_get_object(parser, nametag, lower_to_upper=.TRUE.)
     208             :                ! read UPF file version
     209          16 :                CALL parser_get_object(parser, nametag, lower_to_upper=.TRUE.)
     210          16 :                pot%version = TRIM(nametag)
     211          16 :                CPASSERT(nametag(1:5) == "2.0.1")
     212          16 :                CALL parser_get_object(parser, nametag, lower_to_upper=.TRUE.)
     213          16 :                CPASSERT(nametag(1:1) == ">")
     214             :                ntag = 1
     215             :             END IF
     216         270 :          ELSE IF (ntag == 1) THEN
     217             :             ! we are looking for 1st level tags
     218         270 :             IF (nametag(2:8) == "PP_INFO") THEN
     219          16 :                CPASSERT(nametag(9:9) == ">")
     220          16 :                CALL upf_info_section(parser, pot)
     221         254 :             ELSEIF (nametag(2:10) == "PP_HEADER") THEN
     222          16 :                IF (.NOT. (nametag(11:11) == ">")) THEN
     223          16 :                   CALL upf_header_option(parser, pot)
     224             :                END IF
     225         238 :             ELSEIF (nametag(2:8) == "PP_MESH") THEN
     226          16 :                IF (.NOT. (nametag(9:9) == ">")) THEN
     227          16 :                   CALL upf_mesh_option(parser, pot)
     228             :                END IF
     229          16 :                CALL upf_mesh_section(parser, pot)
     230         222 :             ELSEIF (nametag(2:8) == "PP_NLCC") THEN
     231           0 :                IF (nametag(9:9) == ">") THEN
     232           0 :                   CALL upf_nlcc_section(parser, pot, .FALSE.)
     233             :                ELSE
     234           0 :                   CALL upf_nlcc_section(parser, pot, .TRUE.)
     235             :                END IF
     236         222 :             ELSEIF (nametag(2:9) == "PP_LOCAL") THEN
     237          16 :                IF (nametag(10:10) == ">") THEN
     238           0 :                   CALL upf_local_section(parser, pot, .FALSE.)
     239             :                ELSE
     240          16 :                   CALL upf_local_section(parser, pot, .TRUE.)
     241             :                END IF
     242         206 :             ELSEIF (nametag(2:12) == "PP_NONLOCAL") THEN
     243          16 :                CPASSERT(nametag(13:13) == ">")
     244          16 :                CALL upf_nonlocal_section(parser, pot)
     245         190 :             ELSEIF (nametag(2:13) == "PP_SEMILOCAL") THEN
     246           2 :                CALL upf_semilocal_section(parser, pot)
     247         188 :             ELSEIF (nametag(2:9) == "PP_PSWFC") THEN
     248             :                ! skip section for now
     249         172 :             ELSEIF (nametag(2:11) == "PP_RHOATOM") THEN
     250             :                ! skip section for now
     251         156 :             ELSEIF (nametag(2:7) == "PP_PAW") THEN
     252             :                ! skip section for now
     253         156 :             ELSEIF (nametag(2:6) == "/UPF>") THEN
     254             :                EXIT
     255             :             END IF
     256             :          END IF
     257             :       END DO
     258          16 :       CALL parser_release(parser)
     259             : 
     260          16 :       CPASSERT(ntag > 0)
     261             : 
     262             :       ! rescale projectors
     263          16 :       IF (ALLOCATED(pot%beta)) THEN
     264          30 :          DO ib = 1, pot%number_of_proj
     265          30 :             IF (pot%r(1) == 0.0_dp) THEN
     266           0 :                pot%beta(2:, ib) = pot%beta(2:, ib)/pot%r(2:)
     267             :             ELSE
     268       11452 :                pot%beta(:, ib) = pot%beta(:, ib)/pot%r(:)
     269             :             END IF
     270             :          END DO
     271             :       END IF
     272             : 
     273             :       ! test for not supported options
     274          16 :       IF (readall) THEN
     275          16 :          IF (pot%is_ultrasoft) THEN
     276           0 :             CPABORT("UPF ultrasoft pseudopotential not implemented")
     277             :          END IF
     278          16 :          IF (pot%is_paw) THEN
     279           0 :             CPABORT("UPF PAW potential not implemented")
     280             :          END IF
     281             :       END IF
     282             : 
     283          48 :    END SUBROUTINE atom_read_upf_v2
     284             : 
     285             : ! **************************************************************************************************
     286             : !> \brief ...
     287             : !> \param parser ...
     288             : !> \param pot ...
     289             : ! **************************************************************************************************
     290          16 :    SUBROUTINE upf_info_section(parser, pot)
     291             :       TYPE(cp_parser_type), INTENT(INOUT)                :: parser
     292             :       TYPE(atom_upfpot_type)                             :: pot
     293             : 
     294             :       CHARACTER(LEN=default_string_length)               :: line, string
     295             :       INTEGER                                            :: icount, iline
     296             :       LOGICAL                                            :: at_end
     297             : 
     298          16 :       icount = 0
     299             :       DO
     300         264 :          CALL parser_get_next_line(parser, 1, at_end)
     301         264 :          CPASSERT(.NOT. at_end)
     302         264 :          iline = parser%buffer%present_line_number
     303         264 :          line = TRIM(parser%buffer%input_lines(iline))
     304         264 :          CALL parser_get_object(parser, string)
     305         264 :          IF (string(1:10) == "</PP_INFO>") EXIT
     306         248 :          icount = icount + 1
     307         248 :          IF (icount > pot%maxinfo) CYCLE
     308         248 :          pot%info(icount) = line
     309             :       END DO
     310          16 :       pot%maxinfo = icount
     311             : 
     312          16 :    END SUBROUTINE upf_info_section
     313             : 
     314             : ! **************************************************************************************************
     315             : !> \brief ...
     316             : !> \param parser ...
     317             : !> \param pot ...
     318             : ! **************************************************************************************************
     319          16 :    SUBROUTINE upf_header_option(parser, pot)
     320             :       TYPE(cp_parser_type), INTENT(INOUT)                :: parser
     321             :       TYPE(atom_upfpot_type)                             :: pot
     322             : 
     323             :       CHARACTER(LEN=default_string_length)               :: string
     324             :       LOGICAL                                            :: at_end
     325             : 
     326             :       DO
     327         432 :          IF (parser_test_next_token(parser) == "EOL") THEN
     328         400 :             CALL parser_get_next_line(parser, 1, at_end)
     329         832 :             CPASSERT(.NOT. at_end)
     330             :          END IF
     331         432 :          CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     332         432 :          IF (string == "/>") EXIT
     333          16 :          SELECT CASE (string)
     334             :          CASE ("GENERATED")
     335          16 :             CALL parser_get_object(parser, pot%generated)
     336             :          CASE ("AUTHOR")
     337          16 :             CALL parser_get_object(parser, pot%author)
     338             :          CASE ("DATE")
     339          16 :             CALL parser_get_object(parser, pot%date)
     340             :          CASE ("COMMENT")
     341          16 :             CALL parser_get_object(parser, pot%comment)
     342             :          CASE ("ELEMENT")
     343          16 :             CALL parser_get_object(parser, pot%symbol)
     344          16 :             CPASSERT(2 <= LEN(pot%symbol))
     345             :          CASE ("PSEUDO_TYPE")
     346          16 :             CALL parser_get_object(parser, pot%pseudo_type)
     347             :          CASE ("RELATIVISTIC")
     348          16 :             CALL parser_get_object(parser, pot%relativistic)
     349             :          CASE ("IS_ULTRASOFT")
     350          16 :             CALL parser_get_object(parser, pot%is_ultrasoft)
     351             :          CASE ("IS_PAW")
     352          16 :             CALL parser_get_object(parser, pot%is_paw)
     353             :          CASE ("IS_COULOMB")
     354          16 :             CALL parser_get_object(parser, pot%is_coulomb)
     355             :          CASE ("HAS_SO")
     356          16 :             CALL parser_get_object(parser, pot%has_so)
     357             :          CASE ("HAS_WFC")
     358          16 :             CALL parser_get_object(parser, pot%has_wfc)
     359             :          CASE ("HAS_GIPAW")
     360          16 :             CALL parser_get_object(parser, pot%has_gipaw)
     361             :          CASE ("PAW_AS_GIPAW")
     362          16 :             CALL parser_get_object(parser, pot%paw_as_gipaw)
     363             :          CASE ("CORE_CORRECTION")
     364          16 :             CALL parser_get_object(parser, pot%core_correction)
     365             :          CASE ("FUNCTIONAL")
     366          16 :             CALL parser_get_object(parser, pot%functional)
     367             :          CASE ("Z_VALENCE")
     368          16 :             CALL parser_get_object(parser, pot%zion)
     369             :          CASE ("TOTAL_PSENERGY")
     370          16 :             CALL parser_get_object(parser, pot%total_psenergy)
     371             :          CASE ("WFC_CUTOFF")
     372          16 :             CALL parser_get_object(parser, pot%wfc_cutoff)
     373             :          CASE ("RHO_CUTOFF")
     374          16 :             CALL parser_get_object(parser, pot%rho_cutoff)
     375             :          CASE ("L_MAX")
     376          16 :             CALL parser_get_object(parser, pot%l_max)
     377             :          CASE ("L_MAX_RHO")
     378          16 :             CALL parser_get_object(parser, pot%l_max_rho)
     379             :          CASE ("L_LOCAL")
     380          16 :             CALL parser_get_object(parser, pot%l_local)
     381             :          CASE ("MESH_SIZE")
     382          16 :             CALL parser_get_object(parser, pot%mesh_size)
     383             :          CASE ("NUMBER_OF_WFC")
     384          16 :             CALL parser_get_object(parser, pot%number_of_wfc)
     385             :          CASE ("NUMBER_OF_PROJ")
     386          16 :             CALL parser_get_object(parser, pot%number_of_proj)
     387             :          CASE DEFAULT
     388           0 :             CPWARN(string)
     389             :             CALL cp_abort(__LOCATION__, "Error while parsing UPF header: "// &
     390         416 :                           "Adjust format of delimiters ... only double quotes are admissible.")
     391             :          END SELECT
     392             :       END DO
     393             : 
     394          16 :    END SUBROUTINE upf_header_option
     395             : 
     396             : ! **************************************************************************************************
     397             : !> \brief ...
     398             : !> \param parser ...
     399             : !> \param pot ...
     400             : ! **************************************************************************************************
     401          16 :    SUBROUTINE upf_mesh_option(parser, pot)
     402             :       TYPE(cp_parser_type), INTENT(INOUT)                :: parser
     403             :       TYPE(atom_upfpot_type)                             :: pot
     404             : 
     405             :       CHARACTER(LEN=default_string_length)               :: string
     406             :       INTEGER                                            :: jj
     407             :       LOGICAL                                            :: at_end
     408             : 
     409             :       DO
     410          96 :          IF (parser_test_next_token(parser) == "EOL") THEN
     411          16 :             CALL parser_get_next_line(parser, 1, at_end)
     412         112 :             CPASSERT(.NOT. at_end)
     413             :          END IF
     414          96 :          CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     415          96 :          IF (string == ">") EXIT
     416          16 :          SELECT CASE (string)
     417             :          CASE ("DX")
     418          16 :             CALL parser_get_object(parser, pot%dx)
     419             :          CASE ("XMIN")
     420          16 :             CALL parser_get_object(parser, pot%xmin)
     421             :          CASE ("RMAX")
     422          16 :             CALL parser_get_object(parser, pot%rmax)
     423             :          CASE ("MESH")
     424          16 :             CALL parser_get_object(parser, jj)
     425          16 :             CPASSERT(pot%mesh_size == jj)
     426             :          CASE ("ZMESH")
     427          16 :             CALL parser_get_object(parser, pot%zmesh)
     428             :          CASE DEFAULT
     429          80 :             CPABORT("Unknown UPF PP_MESH option <"//TRIM(string)//"> found")
     430             :          END SELECT
     431             : 
     432             :       END DO
     433             : 
     434          16 :    END SUBROUTINE upf_mesh_option
     435             : 
     436             : ! **************************************************************************************************
     437             : !> \brief ...
     438             : !> \param parser ...
     439             : !> \param pot ...
     440             : ! **************************************************************************************************
     441          16 :    SUBROUTINE upf_mesh_section(parser, pot)
     442             :       TYPE(cp_parser_type), INTENT(INOUT)                :: parser
     443             :       TYPE(atom_upfpot_type)                             :: pot
     444             : 
     445             :       CHARACTER(LEN=default_string_length)               :: line, string, string2
     446             :       INTEGER                                            :: icount, m, mc, ms
     447             :       LOGICAL                                            :: at_end
     448             : 
     449             :       DO
     450          80 :          CALL parser_get_next_line(parser, 1, at_end)
     451          80 :          CPASSERT(.NOT. at_end)
     452          80 :          CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     453          16 :          SELECT CASE (string)
     454             :          CASE ("<PP_R")
     455          16 :             m = pot%mesh_size
     456          16 :             ms = pot%mesh_size
     457          16 :             mc = 1
     458          16 :             IF (string(6:6) /= ">") THEN
     459             :                ! options
     460             :                DO
     461          64 :                   IF (parser_test_next_token(parser) == "EOL") THEN
     462           0 :                      CALL parser_get_next_line(parser, 1, at_end)
     463          64 :                      CPASSERT(.NOT. at_end)
     464             :                   END IF
     465          64 :                   CALL parser_get_object(parser, string2, lower_to_upper=.TRUE.)
     466          64 :                   IF (string2 == ">") EXIT
     467          16 :                   SELECT CASE (string2)
     468             :                   CASE ("TYPE")
     469          16 :                      CALL parser_get_object(parser, line, lower_to_upper=.TRUE.)
     470          16 :                      CPASSERT(line == "REAL")
     471             :                   CASE ("SIZE")
     472          16 :                      CALL parser_get_object(parser, ms)
     473          16 :                      CPASSERT(ms <= m)
     474             :                   CASE ("COLUMNS")
     475          16 :                      CALL parser_get_object(parser, mc)
     476             :                   CASE DEFAULT
     477          48 :                      CPABORT("Unknown UPF PP_R option <"//TRIM(string2)//"> found")
     478             :                   END SELECT
     479             :                END DO
     480             :             END IF
     481          48 :             ALLOCATE (pot%r(m))
     482       15184 :             pot%r = 0.0_dp
     483             :             icount = 1
     484          16 :             DO
     485       18970 :                IF (parser_test_next_token(parser) == "EOL") THEN
     486        3802 :                   CALL parser_get_next_line(parser, 1, at_end)
     487        3802 :                   CPASSERT(.NOT. at_end)
     488       34138 :                ELSE IF (parser_test_next_token(parser) == "FLT") THEN
     489       15168 :                   CALL parser_get_object(parser, pot%r(icount))
     490       30336 :                   icount = icount + 1
     491             :                END IF
     492       18970 :                IF (icount > ms) EXIT
     493             :             END DO
     494             :          CASE ("<PP_RAB")
     495          16 :             IF (string(6:6) /= ">") THEN
     496             :                ! options
     497             :                DO
     498          64 :                   IF (parser_test_next_token(parser) == "EOL") THEN
     499           0 :                      CALL parser_get_next_line(parser, 1, at_end)
     500          64 :                      CPASSERT(.NOT. at_end)
     501             :                   END IF
     502          64 :                   CALL parser_get_object(parser, string2, lower_to_upper=.TRUE.)
     503          64 :                   IF (string2 == ">") EXIT
     504          16 :                   SELECT CASE (string2)
     505             :                   CASE ("TYPE")
     506          16 :                      CALL parser_get_object(parser, line, lower_to_upper=.TRUE.)
     507          16 :                      CPASSERT(line == "REAL")
     508             :                   CASE ("SIZE")
     509          16 :                      CALL parser_get_object(parser, ms)
     510          16 :                      CPASSERT(ms <= m)
     511             :                   CASE ("COLUMNS")
     512          16 :                      CALL parser_get_object(parser, mc)
     513             :                   CASE DEFAULT
     514          48 :                      CPABORT("Unknown UPF PP_RAB option <"//TRIM(string2)//"> found")
     515             :                   END SELECT
     516             :                END DO
     517             :             END IF
     518          48 :             ALLOCATE (pot%rab(m))
     519       15184 :             pot%rab = 0.0_dp
     520             :             icount = 1
     521             :             DO
     522       18970 :                IF (parser_test_next_token(parser) == "EOL") THEN
     523        3802 :                   CALL parser_get_next_line(parser, 1, at_end)
     524        3802 :                   CPASSERT(.NOT. at_end)
     525       34138 :                ELSE IF (parser_test_next_token(parser) == "FLT") THEN
     526       15168 :                   CALL parser_get_object(parser, pot%rab(icount))
     527       30336 :                   icount = icount + 1
     528             :                END IF
     529       18970 :                IF (icount > ms) EXIT
     530             :             END DO
     531             :          CASE ("</PP_MESH>")
     532          80 :             EXIT
     533             :          CASE DEFAULT
     534             :             !
     535             :          END SELECT
     536             :       END DO
     537             : 
     538          16 :    END SUBROUTINE upf_mesh_section
     539             : 
     540             : ! **************************************************************************************************
     541             : !> \brief ...
     542             : !> \param parser ...
     543             : !> \param pot ...
     544             : !> \param options ...
     545             : ! **************************************************************************************************
     546           0 :    SUBROUTINE upf_nlcc_section(parser, pot, options)
     547             :       TYPE(cp_parser_type), INTENT(INOUT)                :: parser
     548             :       TYPE(atom_upfpot_type)                             :: pot
     549             :       LOGICAL, INTENT(IN)                                :: options
     550             : 
     551             :       CHARACTER(LEN=default_string_length)               :: line, string
     552             :       INTEGER                                            :: icount, m, mc, ms
     553             :       LOGICAL                                            :: at_end
     554             : 
     555           0 :       m = pot%mesh_size
     556           0 :       ms = m
     557           0 :       mc = 1
     558           0 :       IF (options) THEN
     559             :          DO
     560           0 :             IF (parser_test_next_token(parser) == "EOL") THEN
     561           0 :                CALL parser_get_next_line(parser, 1, at_end)
     562           0 :                CPASSERT(.NOT. at_end)
     563             :             END IF
     564           0 :             CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     565           0 :             IF (string == ">") EXIT
     566           0 :             SELECT CASE (string)
     567             :             CASE ("TYPE")
     568           0 :                CALL parser_get_object(parser, line, lower_to_upper=.TRUE.)
     569           0 :                CPASSERT(line == "REAL")
     570             :             CASE ("SIZE")
     571           0 :                CALL parser_get_object(parser, ms)
     572           0 :                CPASSERT(ms <= m)
     573             :             CASE ("COLUMNS")
     574           0 :                CALL parser_get_object(parser, mc)
     575             :             CASE DEFAULT
     576           0 :                CPABORT("Unknown UPF PP_NLCC option <"//TRIM(string)//"> found")
     577             :             END SELECT
     578             :          END DO
     579             :       END IF
     580             : 
     581           0 :       ALLOCATE (pot%rho_nlcc(m))
     582           0 :       pot%rho_nlcc = 0.0_dp
     583             :       icount = 1
     584             :       DO
     585           0 :          IF (parser_test_next_token(parser) == "EOL") THEN
     586           0 :             CALL parser_get_next_line(parser, 1, at_end)
     587           0 :             CPASSERT(.NOT. at_end)
     588           0 :          ELSE IF (parser_test_next_token(parser) == "FLT") THEN
     589           0 :             CALL parser_get_object(parser, pot%rho_nlcc(icount))
     590           0 :             icount = icount + 1
     591             :          END IF
     592           0 :          IF (icount > ms) EXIT
     593             :       END DO
     594             : 
     595           0 :       CALL parser_get_next_line(parser, 1, at_end)
     596           0 :       CPASSERT(.NOT. at_end)
     597           0 :       CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     598           0 :       CPASSERT(string == "</PP_NLCC>")
     599             : 
     600           0 :    END SUBROUTINE upf_nlcc_section
     601             : 
     602             : ! **************************************************************************************************
     603             : !> \brief ...
     604             : !> \param parser ...
     605             : !> \param pot ...
     606             : !> \param options ...
     607             : ! **************************************************************************************************
     608          16 :    SUBROUTINE upf_local_section(parser, pot, options)
     609             :       TYPE(cp_parser_type), INTENT(INOUT)                :: parser
     610             :       TYPE(atom_upfpot_type)                             :: pot
     611             :       LOGICAL, INTENT(IN)                                :: options
     612             : 
     613             :       CHARACTER(LEN=default_string_length)               :: line, string
     614             :       INTEGER                                            :: icount, m, mc, ms
     615             :       LOGICAL                                            :: at_end
     616             : 
     617          16 :       m = pot%mesh_size
     618          16 :       ms = m
     619          16 :       mc = 1
     620          16 :       IF (options) THEN
     621             :          DO
     622          64 :             IF (parser_test_next_token(parser) == "EOL") THEN
     623           0 :                CALL parser_get_next_line(parser, 1, at_end)
     624          64 :                CPASSERT(.NOT. at_end)
     625             :             END IF
     626          64 :             CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     627          64 :             IF (string == ">") EXIT
     628          16 :             SELECT CASE (string)
     629             :             CASE ("TYPE")
     630          16 :                CALL parser_get_object(parser, line, lower_to_upper=.TRUE.)
     631          16 :                CPASSERT(line == "REAL")
     632             :             CASE ("SIZE")
     633          16 :                CALL parser_get_object(parser, ms)
     634          16 :                CPASSERT(ms <= m)
     635             :             CASE ("COLUMNS")
     636          16 :                CALL parser_get_object(parser, mc)
     637             :             CASE DEFAULT
     638          48 :                CPABORT("Unknown UPF PP_LOCAL option <"//TRIM(string)//"> found")
     639             :             END SELECT
     640             :          END DO
     641             :       END IF
     642             : 
     643          48 :       ALLOCATE (pot%vlocal(m))
     644       15184 :       pot%vlocal = 0.0_dp
     645             :       icount = 1
     646             :       DO
     647       18970 :          IF (parser_test_next_token(parser) == "EOL") THEN
     648        3802 :             CALL parser_get_next_line(parser, 1, at_end)
     649        3802 :             CPASSERT(.NOT. at_end)
     650       34138 :          ELSE IF (parser_test_next_token(parser) == "FLT") THEN
     651       15168 :             CALL parser_get_object(parser, pot%vlocal(icount))
     652       30336 :             icount = icount + 1
     653             :          END IF
     654       18970 :          IF (icount > ms) EXIT
     655             :       END DO
     656             : 
     657             :       ! Ry -> Hartree
     658       15184 :       pot%vlocal = 0.5_dp*pot%vlocal
     659             : 
     660          16 :       CALL parser_get_next_line(parser, 1, at_end)
     661          16 :       CPASSERT(.NOT. at_end)
     662          16 :       CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     663          16 :       CPASSERT(string == "</PP_LOCAL>")
     664             : 
     665          16 :    END SUBROUTINE upf_local_section
     666             : 
     667             : ! **************************************************************************************************
     668             : !> \brief ...
     669             : !> \param parser ...
     670             : !> \param pot ...
     671             : ! **************************************************************************************************
     672          16 :    SUBROUTINE upf_nonlocal_section(parser, pot)
     673             :       TYPE(cp_parser_type), INTENT(INOUT)                :: parser
     674             :       TYPE(atom_upfpot_type)                             :: pot
     675             : 
     676             :       CHARACTER(LEN=default_string_length)               :: line, string
     677             :       INTEGER                                            :: i1, i2, ibeta, icount, la, m, mc, ms, &
     678             :                                                             nbeta
     679             :       LOGICAL                                            :: at_end
     680             : 
     681          16 :       m = pot%mesh_size
     682          16 :       nbeta = pot%number_of_proj
     683         120 :       ALLOCATE (pot%dion(nbeta, nbeta), pot%beta(m, nbeta), pot%lbeta(nbeta))
     684          56 :       pot%dion = 0.0_dp
     685       11468 :       pot%beta = 0.0_dp
     686          30 :       pot%lbeta = -1
     687             : 
     688             :       ibeta = 0
     689             :       DO
     690          70 :          CALL parser_get_next_line(parser, 1, at_end)
     691          70 :          CPASSERT(.NOT. at_end)
     692          70 :          CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     693          86 :          IF (string(1:8) == "<PP_BETA") THEN
     694          14 :             ms = m
     695          14 :             ibeta = ibeta + 1
     696          14 :             i1 = ibeta
     697          14 :             la = 0
     698          14 :             CPASSERT(ibeta <= nbeta)
     699             :             DO
     700         140 :                IF (parser_test_next_token(parser) == "EOL") THEN
     701          14 :                   CALL parser_get_next_line(parser, 1, at_end)
     702         154 :                   CPASSERT(.NOT. at_end)
     703             :                END IF
     704         140 :                CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     705         140 :                IF (string == ">") EXIT
     706          14 :                SELECT CASE (string)
     707             :                CASE ("TYPE")
     708          14 :                   CALL parser_get_object(parser, line, lower_to_upper=.TRUE.)
     709          14 :                   CPASSERT(line == "REAL")
     710             :                CASE ("SIZE")
     711          14 :                   CALL parser_get_object(parser, ms)
     712          14 :                   CPASSERT(ms <= m)
     713             :                CASE ("COLUMNS")
     714          14 :                   CALL parser_get_object(parser, mc)
     715             :                CASE ("INDEX")
     716          14 :                   CALL parser_get_object(parser, i1)
     717          14 :                   CPASSERT(i1 <= nbeta)
     718             :                CASE ("ANGULAR_MOMENTUM")
     719          28 :                   CALL parser_get_object(parser, la)
     720             :                CASE ("LABEL")
     721          14 :                   CALL parser_get_object(parser, line)
     722             :                   ! not used currently
     723             :                CASE ("CUTOFF_RADIUS_INDEX")
     724          14 :                   CALL parser_get_object(parser, line)
     725             :                   ! not used currently
     726             :                CASE ("CUTOFF_RADIUS")
     727          14 :                   CALL parser_get_object(parser, line)
     728             :                   ! not used currently
     729             :                CASE ("ULTRASOFT_CUTOFF_RADIUS")
     730          14 :                   CALL parser_get_object(parser, line)
     731             :                   ! not used currently
     732             :                CASE DEFAULT
     733         126 :                   CPABORT("Unknown UPF PP_BETA option <"//TRIM(string)//"> found")
     734             :                END SELECT
     735             :             END DO
     736          14 :             pot%lbeta(i1) = la
     737          14 :             icount = 1
     738             :             DO
     739       14306 :                IF (parser_test_next_token(parser) == "EOL") THEN
     740        2868 :                   CALL parser_get_next_line(parser, 1, at_end)
     741        2868 :                   CPASSERT(.NOT. at_end)
     742       25744 :                ELSE IF (parser_test_next_token(parser) == "FLT") THEN
     743       11438 :                   CALL parser_get_object(parser, pot%beta(icount, i1))
     744       22876 :                   icount = icount + 1
     745             :                END IF
     746       14306 :                IF (icount > ms) EXIT
     747             :             END DO
     748          56 :          ELSE IF (string(1:7) == "<PP_DIJ") THEN
     749          16 :             ms = nbeta*nbeta
     750             :             DO
     751          64 :                IF (parser_test_next_token(parser) == "EOL") THEN
     752           0 :                   CALL parser_get_next_line(parser, 1, at_end)
     753          64 :                   CPASSERT(.NOT. at_end)
     754             :                END IF
     755          64 :                CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     756          64 :                IF (string == ">") EXIT
     757          16 :                SELECT CASE (string)
     758             :                CASE ("TYPE")
     759          16 :                   CALL parser_get_object(parser, line, lower_to_upper=.TRUE.)
     760          16 :                   CPASSERT(line == "REAL")
     761             :                CASE ("SIZE")
     762          16 :                   CALL parser_get_object(parser, ms)
     763          16 :                   CPASSERT(ms <= m)
     764             :                CASE ("COLUMNS")
     765          16 :                   CALL parser_get_object(parser, mc)
     766             :                CASE DEFAULT
     767          48 :                   CPABORT("Unknown UPF PP_DIJ option <"//TRIM(string)//"> found")
     768             :                END SELECT
     769             :             END DO
     770             :             icount = 1
     771             :             DO
     772          46 :                IF (parser_test_next_token(parser) == "EOL") THEN
     773          20 :                   CALL parser_get_next_line(parser, 1, at_end)
     774          20 :                   CPASSERT(.NOT. at_end)
     775          72 :                ELSE IF (parser_test_next_token(parser) == "FLT") THEN
     776          26 :                   i1 = (icount - 1)/nbeta + 1
     777          26 :                   i2 = MOD(icount - 1, nbeta) + 1
     778          26 :                   CALL parser_get_object(parser, pot%dion(i1, i2))
     779          52 :                   icount = icount + 1
     780             :                END IF
     781          46 :                IF (icount > ms) EXIT
     782             :             END DO
     783             :          ELSE IF (string(1:7) == "<PP_QIJL") THEN
     784             :             ! skip this option
     785          40 :          ELSE IF (string(1:14) == "</PP_NONLOCAL>") THEN
     786             :             EXIT
     787             :          END IF
     788             :       END DO
     789             : 
     790             :       ! change units and scaling, beta is still r*beta
     791          56 :       pot%dion = 2.0_dp*pot%dion
     792       11468 :       pot%beta = 0.5_dp*pot%beta
     793             : 
     794          16 :    END SUBROUTINE upf_nonlocal_section
     795             : 
     796             : ! **************************************************************************************************
     797             : !> \brief ...
     798             : !> \param parser ...
     799             : !> \param pot ...
     800             : ! **************************************************************************************************
     801           2 :    SUBROUTINE upf_semilocal_section(parser, pot)
     802             :       TYPE(cp_parser_type), INTENT(INOUT)                :: parser
     803             :       TYPE(atom_upfpot_type)                             :: pot
     804             : 
     805             :       CHARACTER(LEN=default_string_length)               :: line, string
     806             :       INTEGER                                            :: i1, ib, icount, la, lmax, m, mc, ms
     807             :       LOGICAL                                            :: at_end
     808             : 
     809           2 :       m = pot%mesh_size
     810           2 :       lmax = pot%l_max
     811           8 :       ALLOCATE (pot%vsemi(m, lmax + 1))
     812        3698 :       pot%vsemi = 0.0_dp
     813             : 
     814             :       ib = 0
     815             :       DO
     816          14 :          CALL parser_get_next_line(parser, 1, at_end)
     817          14 :          CPASSERT(.NOT. at_end)
     818          14 :          CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     819          16 :          IF (string(1:7) == "<PP_VNL") THEN
     820           6 :             ms = m
     821           6 :             ib = ib + 1
     822           6 :             i1 = ib
     823           6 :             la = 0
     824           6 :             CPASSERT(ib <= lmax + 1)
     825             :             DO
     826          30 :                IF (parser_test_next_token(parser) == "EOL") THEN
     827           0 :                   CALL parser_get_next_line(parser, 1, at_end)
     828          30 :                   CPASSERT(.NOT. at_end)
     829             :                END IF
     830          30 :                CALL parser_get_object(parser, string, lower_to_upper=.TRUE.)
     831          30 :                IF (string == ">") EXIT
     832           6 :                SELECT CASE (string)
     833             :                CASE ("TYPE")
     834           6 :                   CALL parser_get_object(parser, line, lower_to_upper=.TRUE.)
     835           6 :                   CPASSERT(line == "REAL")
     836             :                CASE ("SIZE")
     837           6 :                   CALL parser_get_object(parser, ms)
     838           6 :                   CPASSERT(ms <= m)
     839             :                CASE ("COLUMNS")
     840           6 :                   CALL parser_get_object(parser, mc)
     841             :                CASE ("L")
     842           6 :                   CALL parser_get_object(parser, la)
     843             :                CASE DEFAULT
     844          24 :                   CPABORT("Unknown UPF PP_VNL option <"//TRIM(string)//"> found")
     845             :                END SELECT
     846             :             END DO
     847           6 :             i1 = la + 1
     848           6 :             icount = 1
     849             :             DO
     850        3462 :                IF (parser_test_next_token(parser) == "EOL") THEN
     851         696 :                   CALL parser_get_next_line(parser, 1, at_end)
     852         696 :                   CPASSERT(.NOT. at_end)
     853        6228 :                ELSE IF (parser_test_next_token(parser) == "FLT") THEN
     854        2766 :                   CALL parser_get_object(parser, pot%vsemi(icount, i1))
     855        5532 :                   icount = icount + 1
     856             :                END IF
     857        3462 :                IF (icount > ms) EXIT
     858             :             END DO
     859           8 :          ELSEIF (string(1:15) == "</PP_SEMILOCAL>") THEN
     860             :             EXIT
     861             :          ELSE
     862             :             !
     863             :          END IF
     864             :       END DO
     865             :       ! Ry -> Hartree
     866        3698 :       pot%vsemi = 0.5_dp*pot%vsemi
     867             : 
     868           2 :    END SUBROUTINE upf_semilocal_section
     869             : 
     870             : ! **************************************************************************************************
     871             : !> \brief ...
     872             : !> \param upfpot ...
     873             : ! **************************************************************************************************
     874        9326 :    PURE SUBROUTINE atom_release_upf(upfpot)
     875             : 
     876             :       TYPE(atom_upfpot_type), INTENT(INOUT)              :: upfpot
     877             : 
     878        9326 :       IF (ALLOCATED(upfpot%r)) DEALLOCATE (upfpot%r)
     879        9326 :       IF (ALLOCATED(upfpot%rab)) DEALLOCATE (upfpot%rab)
     880        9326 :       IF (ALLOCATED(upfpot%vlocal)) DEALLOCATE (upfpot%vlocal)
     881        9326 :       IF (ALLOCATED(upfpot%dion)) DEALLOCATE (upfpot%dion)
     882        9326 :       IF (ALLOCATED(upfpot%beta)) DEALLOCATE (upfpot%beta)
     883        9326 :       IF (ALLOCATED(upfpot%lbeta)) DEALLOCATE (upfpot%lbeta)
     884        9326 :       IF (ALLOCATED(upfpot%vsemi)) DEALLOCATE (upfpot%vsemi)
     885             : 
     886        9326 :    END SUBROUTINE atom_release_upf
     887             : ! **************************************************************************************************
     888             : 
     889           0 : END MODULE atom_upf

Generated by: LCOV version 1.15