LCOV - code coverage report
Current view: top level - src/common - util.F (source / functions) Hit Total Coverage
Test: CP2K Regtests (git:4dc10b3) Lines: 117 117 100.0 %
Date: 2024-11-21 06:45:46 Functions: 8 8 100.0 %

          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 All kind of helpful little routines
      10             : !> \par History
      11             : !>      none
      12             : !> \author CJM & JGH
      13             : ! **************************************************************************************************
      14             : MODULE util
      15             :    USE cp_array_sort,                   ONLY: cp_1d_i4_sort,&
      16             :                                               cp_1d_i8_sort,&
      17             :                                               cp_1d_r_sort,&
      18             :                                               cp_1d_s_sort
      19             :    USE kinds,                           ONLY: dp
      20             : 
      21             :    IMPLICIT NONE
      22             : 
      23             :    PRIVATE
      24             :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'util'
      25             :    PUBLIC :: sort, &
      26             :              get_limit, &
      27             :              locate, &
      28             :              find_boundary, &
      29             :              sort_unique
      30             : 
      31             :    INTERFACE sort
      32             :       MODULE PROCEDURE cp_1d_s_sort, cp_1d_r_sort, cp_1d_i4_sort, cp_1d_i8_sort, &
      33             :          sort_cv, sort_im, sort_cm
      34             :    END INTERFACE
      35             : 
      36             :    INTERFACE sort_unique
      37             :       MODULE PROCEDURE sort_unique1
      38             :    END INTERFACE
      39             : 
      40             :    INTERFACE find_boundary
      41             :       MODULE PROCEDURE find_boundary1, find_boundary2
      42             :    END INTERFACE
      43             : 
      44             : CONTAINS
      45             : 
      46             : ! **************************************************************************************************
      47             : !> \brief Purpose: Given an array array(1:n), and given a value x, a value x_index
      48             : !>             is returned which is the index value of the array element equal
      49             : !>             to the value x: x = array(x_index)
      50             : !>             The array must be monotonic increasing.
      51             : !>             x_index = 0 is returned, if no array element equal to the value
      52             : !>             of x was found.
      53             : !> \param array ...
      54             : !> \param x ...
      55             : !> \return ...
      56             : !> \par History
      57             : !>      Derived from the locate function described in
      58             : !>      Numerical Recipes in Fortran 90 (09.01.2004,MK)
      59             : ! **************************************************************************************************
      60    10385168 :    PURE FUNCTION locate(array, x) RESULT(x_index)
      61             :       INTEGER, DIMENSION(:), INTENT(IN)                  :: array
      62             :       INTEGER, INTENT(IN)                                :: x
      63             :       INTEGER                                            :: x_index
      64             : 
      65             :       INTEGER                                            :: jl, jm, ju, n
      66             : 
      67    10385168 :       x_index = 0
      68             : 
      69    10385168 :       IF (x < array(1)) RETURN
      70    10385168 :       n = SIZE(array)
      71    10385168 :       IF (x > array(n)) RETURN
      72    10385094 :       jl = 0
      73    10385094 :       ju = n + 1
      74    36307462 :       DO WHILE (ju - jl > 1)
      75    25922368 :          jm = (ju + jl)/2
      76    36307462 :          IF (x >= array(jm)) THEN
      77             :             jl = jm
      78             :          ELSE
      79    18955669 :             ju = jm
      80             :          END IF
      81             :       END DO
      82    10385094 :       IF (x == array(jl)) x_index = jl
      83             :    END FUNCTION locate
      84             : 
      85             : ! **************************************************************************************************
      86             : !> \brief Sorts and returns a logical that checks if all elements are unique
      87             : !> \param arr ...
      88             : !> \param unique ...
      89             : !> \par History
      90             : !>      Teodoro Laino - Zurich University [tlaino] 04.2007
      91             : ! **************************************************************************************************
      92        1809 :    SUBROUTINE sort_unique1(arr, unique)
      93             :       INTEGER, DIMENSION(:), INTENT(INOUT)               :: arr
      94             :       LOGICAL, INTENT(OUT)                               :: unique
      95             : 
      96             :       INTEGER                                            :: i, n
      97        1809 :       INTEGER, ALLOCATABLE, DIMENSION(:)                 :: wrk
      98             : 
      99        1809 :       n = SIZE(arr)
     100        1809 :       unique = .TRUE.
     101        5027 :       ALLOCATE (wrk(n))
     102        1809 :       CALL sort(arr, n, wrk)
     103        2866 :       DO i = 2, n
     104        2866 :          IF (arr(i) == arr(i - 1)) THEN
     105         186 :             unique = .FALSE.
     106         186 :             EXIT
     107             :          END IF
     108             :       END DO
     109        1809 :       DEALLOCATE (wrk)
     110        1809 :    END SUBROUTINE sort_unique1
     111             : 
     112             : ! **************************************************************************************************
     113             : !> \brief Sorts an array of strings
     114             : !> \param arr ...
     115             : !> \param n ...
     116             : !> \param index ...
     117             : !> \author Teodoro Laino [tlaino] - University of Zurich  10.2008
     118             : ! **************************************************************************************************
     119       23190 :    SUBROUTINE sort_cv(arr, n, index)
     120             :       INTEGER, INTENT(IN)                                :: n
     121             :       CHARACTER(LEN=*), INTENT(INOUT)                    :: arr(1:n)
     122             :       INTEGER, INTENT(OUT)                               :: INDEX(1:n)
     123             : 
     124             :       INTEGER                                            :: i, j, max_length
     125             :       INTEGER, ALLOCATABLE, DIMENSION(:, :)              :: entries
     126             : 
     127       23190 :       max_length = 0
     128      726928 :       DO i = 1, n
     129      726928 :          max_length = MAX(max_length, LEN_TRIM(arr(i)))
     130             :       END DO
     131       92760 :       ALLOCATE (entries(max_length, SIZE(arr)))
     132      726928 :       DO i = 1, n
     133     3203396 :          DO j = 1, LEN_TRIM(arr(i))
     134     3203396 :             entries(j, i) = ICHAR(arr(i) (j:j))
     135             :          END DO
     136      726928 :          IF (j <= max_length) THEN
     137      627454 :             entries(j:max_length, i) = ICHAR(" ")
     138             :          END IF
     139             :       END DO
     140       23190 :       CALL sort_im(entries, istart=1, iend=n, j=1, jsize=max_length, INDEX=INDEX)
     141             :       ! Recover string once ordered
     142      726928 :       DO i = 1, n
     143     3635566 :          DO j = 1, max_length
     144     3612376 :             arr(i) (j:j) = CHAR(entries(j, i))
     145             :          END DO
     146             :       END DO
     147       23190 :       DEALLOCATE (entries)
     148       23190 :    END SUBROUTINE sort_cv
     149             : 
     150             : #:for argtype, itemtype in [['INTEGER', 'INTEGER'], ['CHARACTER(LEN=*)', 'CHARACTER(LEN=LEN(matrix))']]
     151             : ! **************************************************************************************************
     152             : !> \brief Sorts a multiple arrays M(j,i), ordering iteratively over i with fixed j
     153             : !> \param matrix ...
     154             : !> \param istart ...
     155             : !> \param iend ...
     156             : !> \param j ...
     157             : !> \param jsize ...
     158             : !> \param index ...
     159             : !> \author Teodoro Laino [tlaino] - University of Zurich  10.2008
     160             : ! **************************************************************************************************
     161      270110 :    RECURSIVE SUBROUTINE sort_${argtype[0]}$m(matrix, istart, iend, j, jsize, index)
     162             :       ${argtype}$, DIMENSION(:, :), INTENT(INOUT)            :: matrix
     163             :       INTEGER, INTENT(IN)                                    :: istart, iend, j, jsize
     164             :       INTEGER, DIMENSION(:), INTENT(INOUT)                   :: index
     165             : 
     166             : 
     167       23154 :       ${itemtype}$                                           :: item
     168      270110 :       ${itemtype}$, ALLOCATABLE, DIMENSION(:)                :: work, work2
     169             :       INTEGER                                                :: i, ind, isize, k, kend, kstart
     170      270110 :       INTEGER, ALLOCATABLE, DIMENSION(:)                     :: bck_index, tmp_index
     171             : 
     172      270110 :     isize = iend - istart + 1
     173             :     ! Initialize the INDEX array only for the first row..
     174      270110 :     IF (j == 1) THEN
     175      981376 :        DO i = 1, isize
     176      981376 :           INDEX(i) = i
     177             :        ENDDO
     178             :     END IF
     179             : 
     180             :     ! Allocate scratch arrays
     181     1666968 :     ALLOCATE (work(isize), work2(isize), tmp_index(isize), bck_index(isize))
     182     4119790 :     ind = 0
     183     4119790 :     DO i = istart, iend
     184     3849680 :        ind = ind + 1
     185     3849680 :        work(ind) = matrix(j, i)
     186     4119790 :        bck_index(ind) = INDEX(i)
     187             :     END DO
     188             : 
     189             :     ! Ordering row (j) interval istart..iend
     190      270110 :     CALL sort(work, isize, tmp_index)
     191             : 
     192             :     ! Copy into global INDEX array with a proper mapping
     193      270110 :     ind = 0
     194     4119790 :     DO i = istart, iend
     195     3849680 :        ind = ind + 1
     196     3849680 :        INDEX(i) = bck_index(tmp_index(ind))
     197     4119790 :        matrix(j, i) = work(ind)
     198             :     END DO
     199             : 
     200             :     ! Reorder the rest of the array according the present reordering
     201      975296 :     DO k = j + 1, jsize
     202             :        ind = 0
     203     9143732 :        DO i = istart, iend
     204     8438546 :           ind = ind + 1
     205     9143732 :           work2(ind) = matrix(k, i)
     206             :        END DO
     207             :        ind = 0
     208     9413842 :        DO i = istart, iend
     209     8438546 :           ind = ind + 1
     210     9143732 :           matrix(k, i) = work2(tmp_index(ind))
     211             :        END DO
     212             :     END DO
     213             : 
     214             :     ! There are more rows to order..
     215      270110 :     IF (j < jsize) THEN
     216      210142 :        kstart = istart
     217      210142 :        item = work(1)
     218      195582 :        ind = 0
     219     3141600 :        DO i = istart, iend
     220     2931458 :           ind = ind + 1
     221     3141600 :           IF (item /= work(ind)) THEN
     222       76526 :              kend = i - 1
     223       76526 :              IF (kstart /= kend) THEN
     224       54220 :                 CALL sort(matrix, kstart, kend, j + 1, jsize, INDEX)
     225             :              END IF
     226       76526 :              item = work(ind)
     227       76526 :              kstart = i
     228             :           END IF
     229             :        END DO
     230      210142 :        kend = i - 1
     231      210142 :        IF (kstart /= kend) THEN
     232      192592 :           CALL sort(matrix, kstart, kend, j + 1, jsize, INDEX)
     233             :        END IF
     234             :     END IF
     235      270110 :     DEALLOCATE (work, work2, tmp_index, bck_index)
     236             : 
     237      270110 :    END SUBROUTINE sort_${argtype[0]}$m
     238             : #:endfor
     239             : 
     240             : ! **************************************************************************************************
     241             : !> \brief divide m entries into n parts, return size of part me
     242             : !> \param m ...
     243             : !> \param n ...
     244             : !> \param me ...
     245             : !> \return ...
     246             : ! **************************************************************************************************
     247    10082708 :    PURE FUNCTION get_limit(m, n, me) RESULT(nlim)
     248             :       INTEGER, INTENT(IN)                                :: m, n, me
     249             :       INTEGER                                            :: nlim(2)
     250             : 
     251             :       INTEGER                                            :: nl, nu
     252             :       REAL(KIND=dp)                                      :: part
     253             : 
     254    10082708 :       part = REAL(m, KIND=dp)/REAL(n, KIND=dp)
     255    10082708 :       nl = NINT(REAL(me, KIND=dp)*part) + 1
     256    10082708 :       nu = NINT(REAL(me + 1, KIND=dp)*part)
     257    10082708 :       nlim(1) = MAX(1, nl)
     258    10082708 :       nlim(2) = MIN(m, nu)
     259             : 
     260    10082708 :    END FUNCTION get_limit
     261             : 
     262             : ! **************************************************************************************************
     263             : !> \brief finds boundary where element search starts and ends in a 1D array
     264             : !>      array1:      XXXXXAAAAAAAAAXXDGFSFGWDDDDDDDAAAWE
     265             : !>                        |       |
     266             : !>                     start     end  (searching for A)
     267             : !> \param num_array ...
     268             : !> \param ntot ...
     269             : !> \param first ...
     270             : !> \param last ...
     271             : !> \param search ...
     272             : ! **************************************************************************************************
     273        2480 :    PURE SUBROUTINE find_boundary1(num_array, ntot, first, last, search)
     274             :       INTEGER, POINTER                                   :: num_array(:)
     275             :       INTEGER, INTENT(IN)                                :: ntot
     276             :       INTEGER, INTENT(OUT)                               :: first, last
     277             :       INTEGER, INTENT(IN)                                :: search
     278             : 
     279             :       INTEGER                                            :: i
     280             :       LOGICAL                                            :: found
     281             : 
     282        2480 :       found = .FALSE.
     283        2480 :       first = 0
     284        2480 :       last = ntot
     285             : 
     286     1911983 :       DO i = 1, ntot
     287     1911983 :          IF (num_array(i) == search) THEN
     288      580855 :             IF (.NOT. found) THEN
     289        2480 :                first = i
     290             :             END IF
     291             :             found = .TRUE.
     292             :          ELSE
     293     1330432 :             IF (found) THEN
     294        1784 :                last = i - 1
     295        1784 :                EXIT
     296             :             END IF
     297             :             found = .FALSE.
     298             :          END IF
     299             :       END DO
     300             : 
     301        2480 :    END SUBROUTINE find_boundary1
     302             : 
     303             : ! **************************************************************************************************
     304             : !> \brief finds boundary where element search1 starts and ends in array1 checking
     305             : !>      at the same time search2 in array2
     306             : !>      array1:      XXXXXAAAAAAAAAXXDGFSFGWDDDDDDDAAAWE
     307             : !>      array2:      XXXXASDEYYYYASDEFAAAARGASGASRGAWRRR
     308             : !>                           |  |
     309             : !>                       start  end  (searching for A and Y)
     310             : !> \param num_array1 ...
     311             : !> \param num_array2 ...
     312             : !> \param ntot ...
     313             : !> \param first ...
     314             : !> \param last ...
     315             : !> \param search1 ...
     316             : !> \param search2 ...
     317             : ! **************************************************************************************************
     318        1228 :    PURE SUBROUTINE find_boundary2(num_array1, num_array2, ntot, first, last, search1, search2)
     319             :       INTEGER, POINTER                                   :: num_array1(:), num_array2(:)
     320             :       INTEGER, INTENT(IN)                                :: ntot
     321             :       INTEGER, INTENT(OUT)                               :: first, last
     322             :       INTEGER, INTENT(IN)                                :: search1, search2
     323             : 
     324             :       INTEGER                                            :: i, tfirst, tlast
     325             :       LOGICAL                                            :: found
     326             : 
     327        1228 :       found = .FALSE.
     328        1228 :       first = 0
     329        1228 :       last = ntot
     330             : 
     331        1228 :       CALL find_boundary(num_array1, ntot, tfirst, tlast, search1)
     332        1228 :       last = tlast
     333       41028 :       DO i = tfirst, tlast
     334       41028 :          IF (num_array2(i) == search2) THEN
     335       15986 :             IF (.NOT. found) THEN
     336        1228 :                first = i
     337             :             END IF
     338             :             found = .TRUE.
     339             :          ELSE
     340       24232 :             IF (found) THEN
     341         418 :                last = i - 1
     342         418 :                EXIT
     343             :             END IF
     344             :             found = .FALSE.
     345             :          END IF
     346             :       END DO
     347             : 
     348        1228 :    END SUBROUTINE find_boundary2
     349             : 
     350             : END MODULE util

Generated by: LCOV version 1.15