LCOV - code coverage report
Current view: top level - src - topology_connectivity_util.F (source / functions) Hit Total Coverage
Test: CP2K Regtests (git:2fce0f8) Lines: 884 953 92.8 %
Date: 2024-12-21 06:28:57 Functions: 3 3 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 Collection of subroutine needed for topology related things
      10             : !> \par History
      11             : !>     jgh (23-05-2004) Last atom of molecule information added
      12             : ! **************************************************************************************************
      13             : 
      14             : MODULE topology_connectivity_util
      15             :    USE cp_log_handling,                 ONLY: cp_get_default_logger,&
      16             :                                               cp_logger_get_default_io_unit,&
      17             :                                               cp_logger_type
      18             :    USE cp_output_handling,              ONLY: cp_print_key_finished_output,&
      19             :                                               cp_print_key_unit_nr
      20             :    USE force_field_kind_types,          ONLY: do_ff_charmm,&
      21             :                                               do_ff_harmonic
      22             :    USE input_constants,                 ONLY: do_conn_g87,&
      23             :                                               do_conn_g96,&
      24             :                                               do_conn_user
      25             :    USE input_section_types,             ONLY: section_vals_type,&
      26             :                                               section_vals_val_get
      27             :    USE kinds,                           ONLY: default_string_length
      28             :    USE memory_utilities,                ONLY: reallocate
      29             :    USE molecule_kind_types,             ONLY: &
      30             :         allocate_molecule_kind_set, atom_type, bend_type, bond_type, get_molecule_kind, impr_type, &
      31             :         molecule_kind_type, opbend_type, set_molecule_kind, torsion_type, ub_type
      32             :    USE molecule_types,                  ONLY: allocate_molecule_set,&
      33             :                                               get_molecule,&
      34             :                                               local_states_type,&
      35             :                                               molecule_type,&
      36             :                                               set_molecule,&
      37             :                                               set_molecule_set
      38             :    USE string_table,                    ONLY: id2str
      39             :    USE topology_types,                  ONLY: atom_info_type,&
      40             :                                               connectivity_info_type,&
      41             :                                               topology_parameters_type
      42             :    USE util,                            ONLY: sort
      43             : #include "./base/base_uses.f90"
      44             : 
      45             :    IMPLICIT NONE
      46             : 
      47             :    CHARACTER(len=*), PARAMETER, PRIVATE :: moduleN = 'topology_connectivity_util'
      48             : 
      49             :    PRIVATE
      50             :    PUBLIC :: topology_connectivity_pack, topology_conn_multiple
      51             : 
      52             : CONTAINS
      53             : 
      54             : ! **************************************************************************************************
      55             : !> \brief topology connectivity pack
      56             : !> \param molecule_kind_set ...
      57             : !> \param molecule_set ...
      58             : !> \param topology ...
      59             : !> \param subsys_section ...
      60             : !> \par History 11/2009 (Louis Vanduyhuys): added Out of Plane bends based on
      61             : !>                                      impropers in topology
      62             : ! **************************************************************************************************
      63       10156 :    SUBROUTINE topology_connectivity_pack(molecule_kind_set, molecule_set, &
      64             :                                          topology, subsys_section)
      65             :       TYPE(molecule_kind_type), DIMENSION(:), POINTER    :: molecule_kind_set
      66             :       TYPE(molecule_type), DIMENSION(:), POINTER         :: molecule_set
      67             :       TYPE(topology_parameters_type), INTENT(INOUT)      :: topology
      68             :       TYPE(section_vals_type), POINTER                   :: subsys_section
      69             : 
      70             :       CHARACTER(len=*), PARAMETER :: routineN = 'topology_connectivity_pack'
      71             : 
      72             :       CHARACTER(LEN=default_string_length)               :: name
      73             :       INTEGER :: counter, first, handle, handle2, i, ibend, ibond, idim, iimpr, ikind, imol, &
      74             :          inter_bends, inter_bonds, inter_imprs, inter_torsions, inter_ubs, intra_bends, &
      75             :          intra_bonds, intra_imprs, intra_torsions, intra_ubs, inum, ires, istart_mol, istart_typ, &
      76             :          itorsion, ityp, iub, iw, j, j1, j2, j3, j4, jind, last, min_index, natom, nelectron, &
      77             :          nsgf, nval_tot1, nval_tot2, nvar1, nvar2, output_unit, stat
      78       10156 :       INTEGER, DIMENSION(:), POINTER :: c_var_a, c_var_b, c_var_c, c_var_d, c_var_type, &
      79       10156 :          first_list, last_list, map_atom_mol, map_atom_type, map_cvar_mol, map_cvars, map_var_mol, &
      80       10156 :          map_vars, molecule_list
      81       10156 :       INTEGER, DIMENSION(:, :), POINTER                  :: bnd_ctype, bnd_type
      82             :       LOGICAL                                            :: found, found_last
      83             :       TYPE(atom_info_type), POINTER                      :: atom_info
      84       10156 :       TYPE(atom_type), DIMENSION(:), POINTER             :: atom_list
      85       10156 :       TYPE(bend_type), DIMENSION(:), POINTER             :: bend_list
      86       10156 :       TYPE(bond_type), DIMENSION(:), POINTER             :: bond_list
      87             :       TYPE(connectivity_info_type), POINTER              :: conn_info
      88             :       TYPE(cp_logger_type), POINTER                      :: logger
      89       10156 :       TYPE(impr_type), DIMENSION(:), POINTER             :: impr_list
      90       10156 :       TYPE(local_states_type), DIMENSION(:), POINTER     :: lmi
      91             :       TYPE(molecule_kind_type), POINTER                  :: molecule_kind
      92             :       TYPE(molecule_type), POINTER                       :: molecule
      93       10156 :       TYPE(opbend_type), DIMENSION(:), POINTER           :: opbend_list
      94       10156 :       TYPE(torsion_type), DIMENSION(:), POINTER          :: torsion_list
      95       10156 :       TYPE(ub_type), DIMENSION(:), POINTER               :: ub_list
      96             : 
      97       10156 :       NULLIFY (logger)
      98       20312 :       logger => cp_get_default_logger()
      99       10156 :       output_unit = cp_logger_get_default_io_unit(logger)
     100             :       iw = cp_print_key_unit_nr(logger, subsys_section, "PRINT%TOPOLOGY_INFO/UTIL_INFO", &
     101       10156 :                                 extension=".subsysLog")
     102       10156 :       CALL timeset(routineN, handle)
     103             : 
     104       10156 :       atom_info => topology%atom_info
     105       10156 :       conn_info => topology%conn_info
     106       30468 :       ALLOCATE (map_atom_mol(topology%natoms))
     107       20312 :       ALLOCATE (map_atom_type(topology%natoms))
     108             :       !-----------------------------------------------------------------------------
     109             :       !-----------------------------------------------------------------------------
     110             :       ! 1. Set the topology%[nmol_type,nmol,nmol_conn]
     111             :       !-----------------------------------------------------------------------------
     112       10156 :       CALL timeset(routineN//"_1", handle2)
     113       10156 :       natom = topology%natoms
     114       10156 :       topology%nmol = 1
     115       10156 :       topology%nmol_type = 1
     116       10156 :       topology%nmol_conn = 0
     117      763297 :       map_atom_mol = -1
     118      763297 :       map_atom_type = -1
     119       10156 :       map_atom_mol(1) = 1
     120       10156 :       map_atom_type(1) = 1
     121      753141 :       DO i = 1, natom - 1
     122      742985 :          IF ((atom_info%map_mol_typ(i + 1) /= atom_info%map_mol_typ(i)) .OR. &
     123             :              ((atom_info%map_mol_res(i + 1) /= atom_info%map_mol_res(i)) .AND. &
     124             :               (.NOT. (topology%conn_type == do_conn_user)))) THEN
     125      124855 :             topology%nmol_type = topology%nmol_type + 1
     126             :          END IF
     127      742985 :          map_atom_type(i + 1) = topology%nmol_type
     128             :          IF ((atom_info%map_mol_typ(i + 1) /= atom_info%map_mol_typ(i)) .OR. &
     129      742985 :              (atom_info%map_mol_num(i + 1) /= atom_info%map_mol_num(i)) .OR. &
     130             :              (atom_info%map_mol_res(i + 1) /= atom_info%map_mol_res(i))) THEN
     131      292808 :             topology%nmol = topology%nmol + 1
     132             :          END IF
     133      742985 :          map_atom_mol(i + 1) = topology%nmol
     134             :          IF ((atom_info%map_mol_typ(i + 1) == atom_info%map_mol_typ(i)) .AND. &
     135      742985 :              (atom_info%map_mol_num(i + 1) == atom_info%map_mol_num(i)) .AND. &
     136       10156 :              (atom_info%map_mol_res(i + 1) /= atom_info%map_mol_res(i))) THEN
     137        2832 :             topology%nmol_conn = topology%nmol_conn + 1
     138             :          END IF
     139             :       END DO
     140       10156 :       IF (iw > 0) WRITE (iw, *) "topology%nmol ::", topology%nmol
     141       10156 :       IF (iw > 0) WRITE (iw, *) "topology%nmol_type ::", topology%nmol_type
     142       10156 :       IF (iw > 0) WRITE (iw, *) "topology%nmol_conn ::", topology%nmol_conn
     143       10156 :       CALL timestop(handle2)
     144             :       !-----------------------------------------------------------------------------
     145             :       !-----------------------------------------------------------------------------
     146             :       ! 1.1 Clean the temporary arrays to avoid quadratic loops around
     147             :       !     after this fix all topology_pack will be linear scaling
     148             :       !-----------------------------------------------------------------------------
     149       10156 :       CALL timeset(routineN//"_1.1", handle2)
     150       10156 :       istart_mol = map_atom_mol(1)
     151       10156 :       istart_typ = map_atom_type(1)
     152      753141 :       DO i = 2, natom
     153      753141 :          IF ((map_atom_mol(i) /= istart_mol) .AND. (map_atom_type(i) == istart_typ)) THEN
     154      493297 :             map_atom_mol(i) = -map_atom_mol(i)
     155      249688 :          ELSE IF (map_atom_type(i) /= istart_typ) THEN
     156      124855 :             istart_mol = map_atom_mol(i)
     157      124855 :             istart_typ = map_atom_type(i)
     158             :          END IF
     159             :       END DO
     160       10156 :       CALL timestop(handle2)
     161             :       !-----------------------------------------------------------------------------
     162             :       !-----------------------------------------------------------------------------
     163             :       ! 2. Allocate the molecule_kind_set
     164             :       !-----------------------------------------------------------------------------
     165       10156 :       CALL timeset(routineN//"_2", handle2)
     166       10156 :       IF (topology%nmol_type <= 0) THEN
     167           0 :          CPABORT("No molecule kind defined")
     168             :       ELSE
     169       10156 :          NULLIFY (molecule_kind_set)
     170       10156 :          i = topology%nmol_type
     171       10156 :          CALL allocate_molecule_kind_set(molecule_kind_set, i)
     172       10182 :          IF (iw > 0) WRITE (iw, *) "    Allocated molecule_kind_set, Dimenstion of ", &
     173          52 :             SIZE(molecule_kind_set)
     174             :       END IF
     175       10156 :       CALL timestop(handle2)
     176             :       !-----------------------------------------------------------------------------
     177             :       !-----------------------------------------------------------------------------
     178             :       ! 3. Allocate the molecule_set
     179             :       !-----------------------------------------------------------------------------
     180       10156 :       CALL timeset(routineN//"_3", handle2)
     181       10156 :       IF (topology%nmol <= 0) THEN
     182           0 :          CPABORT("No molecule defined")
     183             :       ELSE
     184       10156 :          NULLIFY (molecule_set)
     185       10156 :          i = topology%nmol
     186       10156 :          CALL allocate_molecule_set(molecule_set, i)
     187       10182 :          IF (iw > 0) WRITE (iw, *) "    Allocated molecule_set, dimension of ", &
     188          52 :             topology%nmol
     189             :       END IF
     190       10156 :       CALL timestop(handle2)
     191             :       !-----------------------------------------------------------------------------
     192             :       !-----------------------------------------------------------------------------
     193             :       ! 4. Set the molecule_kind_set%[kind_number,name,nsgf,nelectron]
     194             :       !-----------------------------------------------------------------------------
     195       10156 :       CALL timeset(routineN//"_4", handle2)
     196       10156 :       natom = topology%natoms
     197       10156 :       ikind = -1
     198      763297 :       DO i = 1, natom
     199      763297 :          IF (ikind /= map_atom_type(i)) THEN
     200      135011 :             ikind = map_atom_type(i)
     201      135011 :             molecule_kind => molecule_kind_set(ikind)
     202      135011 :             nsgf = 0
     203      135011 :             nelectron = 0
     204      135011 :             name = TRIM(id2str(atom_info%id_molname(i)))
     205             :             CALL set_molecule_kind(molecule_kind=molecule_kind, &
     206             :                                    kind_number=ikind, &
     207             :                                    molname_generated=topology%molname_generated, &
     208             :                                    name=TRIM(name), &
     209             :                                    nsgf=nsgf, &
     210      135011 :                                    nelectron=nelectron)
     211             :          END IF
     212             :       END DO
     213       10156 :       CALL timestop(handle2)
     214             :       !-----------------------------------------------------------------------------
     215             :       !-----------------------------------------------------------------------------
     216             :       ! 5. Set the molecule_list for molecule_kind in molecule_kind_set
     217             :       !-----------------------------------------------------------------------------
     218       10156 :       CALL timeset(routineN//"_5", handle2)
     219       10156 :       natom = topology%natoms
     220       10156 :       ikind = map_atom_type(1)
     221       10156 :       imol = ABS(map_atom_mol(1))
     222       10156 :       counter = 0
     223      288580 :       DO i = 1, natom - 1
     224      281149 :          IF (ikind /= map_atom_type(i + 1)) THEN
     225      124855 :             found = .TRUE.
     226      124855 :             found_last = .FALSE.
     227      124855 :             imol = ABS(map_atom_mol(i))
     228      156294 :          ELSEIF (ikind == topology%nmol_type) THEN
     229        2725 :             found = .TRUE.
     230        2725 :             found_last = .TRUE.
     231        2725 :             imol = ABS(map_atom_mol(natom))
     232             :          ELSE
     233             :             found = .FALSE.
     234             :             found_last = .FALSE.
     235             :          END IF
     236             : 
     237       10156 :          IF (found) THEN
     238      382740 :             ALLOCATE (molecule_list(imol - counter))
     239      423113 :             DO j = 1, SIZE(molecule_list)
     240      423113 :                molecule_list(j) = j + counter
     241             :             END DO
     242      127580 :             molecule_kind => molecule_kind_set(ikind)
     243             :             CALL set_molecule_kind(molecule_kind=molecule_kind, &
     244      127580 :                                    molecule_list=molecule_list)
     245      127580 :             IF (iw > 0) WRITE (iw, *) "      molecule_list", ikind, molecule_list(:)
     246      127580 :             IF (found_last) EXIT
     247      124855 :             counter = imol
     248      124855 :             ikind = map_atom_type(i + 1)
     249             :          END IF
     250             :       END DO
     251             :       ! Treat separately the case in which the last atom is also a molecule
     252       10156 :       IF (i == natom) THEN
     253        7431 :          imol = ABS(map_atom_mol(natom))
     254             :          ! Last atom is also a molecule by itself
     255       22293 :          ALLOCATE (molecule_list(imol - counter))
     256       14862 :          DO j = 1, SIZE(molecule_list)
     257       14862 :             molecule_list(j) = j + counter
     258             :          END DO
     259        7431 :          molecule_kind => molecule_kind_set(ikind)
     260             :          CALL set_molecule_kind(molecule_kind=molecule_kind, &
     261        7431 :                                 molecule_list=molecule_list)
     262        7431 :          IF (iw > 0) WRITE (iw, *) "      molecule_list", ikind, molecule_list(:)
     263             :       END IF
     264       10156 :       CALL timestop(handle2)
     265             :       !-----------------------------------------------------------------------------
     266             :       !-----------------------------------------------------------------------------
     267             :       ! 6. Set the molecule_set(imol)%molecule_kind via set_molecule
     268             :       !-----------------------------------------------------------------------------
     269       10156 :       CALL timeset(routineN//"_6", handle2)
     270      145167 :       DO ikind = 1, SIZE(molecule_kind_set)
     271      135011 :          molecule_kind => molecule_kind_set(ikind)
     272             :          CALL get_molecule_kind(molecule_kind=molecule_kind, &
     273      135011 :                                 molecule_list=molecule_list)
     274      448131 :          DO i = 1, SIZE(molecule_list)
     275      302964 :             molecule => molecule_set(molecule_list(i))
     276      437975 :             CALL set_molecule(molecule, molecule_kind=molecule_kind)
     277             :          END DO
     278             :       END DO
     279       10156 :       CALL timestop(handle2)
     280             :       !-----------------------------------------------------------------------------
     281             :       !-----------------------------------------------------------------------------
     282             :       ! 7. Set the molecule_set(imol)%[first_atom,last_atom] via set_molecule_set
     283             :       !-----------------------------------------------------------------------------
     284       30468 :       ALLOCATE (first_list(SIZE(molecule_set)))
     285       20312 :       ALLOCATE (last_list(SIZE(molecule_set)))
     286       10156 :       CALL timeset(routineN//"_7", handle2)
     287      313120 :       first_list(:) = 0
     288      313120 :       last_list(:) = 0
     289       10156 :       ityp = atom_info%map_mol_typ(1)
     290       10156 :       inum = atom_info%map_mol_num(1)
     291       10156 :       ires = atom_info%map_mol_res(1)
     292       10156 :       imol = 1
     293       10156 :       first_list(1) = 1
     294      753141 :       DO j = 2, natom
     295             :          IF ((atom_info%map_mol_typ(j) /= ityp) .OR. &
     296      742985 :              (atom_info%map_mol_num(j) /= inum) .OR. &
     297       10156 :              (atom_info%map_mol_res(j) /= ires)) THEN
     298      292808 :             ityp = atom_info%map_mol_typ(j)
     299      292808 :             inum = atom_info%map_mol_num(j)
     300      292808 :             ires = atom_info%map_mol_res(j)
     301      292808 :             imol = imol + 1
     302      292808 :             first_list(imol) = j
     303             :          END IF
     304             :       END DO
     305       10156 :       CPASSERT(imol == topology%nmol)
     306      302964 :       DO ikind = 1, topology%nmol - 1
     307      302964 :          last_list(ikind) = first_list(ikind + 1) - 1
     308             :       END DO
     309       10156 :       last_list(topology%nmol) = topology%natoms
     310       10156 :       CALL set_molecule_set(molecule_set, first_list, last_list)
     311       10156 :       CALL timestop(handle2)
     312             :       !-----------------------------------------------------------------------------
     313             :       !-----------------------------------------------------------------------------
     314             :       ! 8. Set and NULLIFY the molecule_set(imol)%lmi via set_molecule_set
     315             :       !-----------------------------------------------------------------------------
     316       10156 :       CALL timeset(routineN//"_8", handle2)
     317      313120 :       DO imol = 1, SIZE(molecule_set)
     318      302964 :          molecule => molecule_set(imol)
     319      302964 :          NULLIFY (lmi)
     320      313120 :          CALL set_molecule(molecule, lmi=lmi)
     321             :       END DO
     322       10156 :       CALL timestop(handle2)
     323             :       !-----------------------------------------------------------------------------
     324             :       !-----------------------------------------------------------------------------
     325             :       ! 9. Set the atom_list for molecule_kind in molecule_kind_set (PART 1)
     326             :       !-----------------------------------------------------------------------------
     327       10156 :       CALL timeset(routineN//"_9", handle2)
     328       10156 :       counter = 0
     329      313120 :       DO imol = 1, SIZE(molecule_set)
     330      302964 :          molecule => molecule_set(imol)
     331      302964 :          molecule_kind => molecule_set(imol)%molecule_kind
     332             :          CALL get_molecule_kind(molecule_kind=molecule_kind, &
     333      302964 :                                 kind_number=i)
     334      313120 :          IF (counter /= i) THEN
     335      135011 :             counter = i
     336             :             CALL get_molecule(molecule=molecule, &
     337      135011 :                               first_atom=first, last_atom=last)
     338      135011 :             natom = 0
     339      135011 :             IF (first /= 0 .AND. last /= 0) natom = last - first + 1
     340      664877 :             ALLOCATE (atom_list(natom))
     341      394855 :             DO i = 1, natom
     342             :                !Atomic kind information will be filled in (PART 2)
     343      259844 :                NULLIFY (atom_list(i)%atomic_kind)
     344      259844 :                atom_list(i)%id_name = atom_info%id_atmname(i + first - 1)
     345      260634 :                IF (iw > 0) WRITE (iw, '(5X,A,3I5,1X,A5)') "atom_list ", &
     346      136591 :                   imol, counter, i, TRIM(id2str(atom_list(i)%id_name))
     347             :             END DO
     348      135011 :             CALL set_molecule_kind(molecule_kind=molecule_kind, atom_list=atom_list)
     349             :          END IF
     350             :       END DO
     351       10156 :       CALL timestop(handle2)
     352             :       !-----------------------------------------------------------------------------
     353             :       !-----------------------------------------------------------------------------
     354             :       ! 10. Set the molecule_kind%[nbond,bond_list] via set_molecule_kind
     355             :       !-----------------------------------------------------------------------------
     356       10156 :       CALL timeset(routineN//"_10", handle2)
     357             :       ! First map bonds on molecules
     358       10156 :       nvar1 = 0
     359       10156 :       nvar2 = 0
     360       10156 :       NULLIFY (map_vars, map_cvars, bnd_type, bnd_ctype)
     361       10156 :       IF (ASSOCIATED(conn_info%bond_a)) nvar1 = SIZE(conn_info%bond_a)
     362       10156 :       IF (ASSOCIATED(conn_info%c_bond_a)) nvar2 = SIZE(conn_info%c_bond_a)
     363       10156 :       nval_tot1 = nvar1
     364       10156 :       nval_tot2 = 0
     365       22831 :       ALLOCATE (map_var_mol(nvar1))
     366       20466 :       ALLOCATE (map_cvar_mol(nvar2))
     367      518345 :       map_var_mol = -1
     368       13022 :       map_cvar_mol = -1
     369      518345 :       DO i = 1, nvar1
     370      508189 :          j1 = map_atom_mol(conn_info%bond_a(i))
     371      508189 :          j2 = map_atom_mol(conn_info%bond_b(i))
     372      518345 :          IF (j1 == j2) THEN
     373      505323 :             IF (j1 > 0) map_var_mol(i) = map_atom_type(conn_info%bond_a(i))
     374             :          END IF
     375             :       END DO
     376       13022 :       DO i = 1, nvar2
     377        2866 :          min_index = MIN(conn_info%c_bond_a(i), conn_info%c_bond_b(i))
     378        2866 :          j1 = map_atom_mol(min_index)
     379       13022 :          IF (j1 > 0) map_cvar_mol(i) = map_atom_type(min_index)
     380             :       END DO
     381       10156 :       CALL find_bnd_typ(topology%nmol_type, map_vars, map_var_mol, bnd_type, nvar1)
     382       10156 :       CALL find_bnd_typ(topology%nmol_type, map_cvars, map_cvar_mol, bnd_ctype, nvar2)
     383      145167 :       DO i = 1, topology%nmol_type
     384      135011 :          intra_bonds = 0
     385      135011 :          inter_bonds = 0
     386      195529 :          IF (ALL(bnd_type(:, i) > 0)) THEN
     387       30259 :             intra_bonds = bnd_type(2, i) - bnd_type(1, i) + 1
     388             :          END IF
     389      140679 :          IF (ALL(bnd_ctype(:, i) > 0)) THEN
     390        2834 :             inter_bonds = bnd_ctype(2, i) - bnd_ctype(1, i) + 1
     391             :          END IF
     392      135011 :          ibond = intra_bonds + inter_bonds
     393      135011 :          IF (iw > 0) THEN
     394         434 :             WRITE (iw, *) "    Total number bonds for molecule type ", i, " :", ibond
     395         434 :             WRITE (iw, *) "    intra (bonds inside  molecules) :: ", intra_bonds
     396         434 :             WRITE (iw, *) "    inter (bonds between molecules) :: ", inter_bonds
     397             :          END IF
     398      135011 :          molecule_kind => molecule_kind_set(i)
     399      135011 :          nval_tot2 = nval_tot2 + ibond*SIZE(molecule_kind%molecule_list)
     400             : 
     401      416436 :          ALLOCATE (bond_list(ibond))
     402      135011 :          ibond = 0
     403      353050 :          DO j = bnd_type(1, i), bnd_type(2, i)
     404      218039 :             IF (j == 0) CYCLE
     405      113287 :             ibond = ibond + 1
     406      113287 :             jind = map_vars(j)
     407      113287 :             first = first_list(map_atom_mol(conn_info%bond_a(jind)))
     408      113287 :             bond_list(ibond)%a = conn_info%bond_a(jind) - first + 1
     409      113287 :             bond_list(ibond)%b = conn_info%bond_b(jind) - first + 1
     410             :             ! Set by default id_type to charmm and modify when handling the forcefield
     411      113287 :             bond_list(ibond)%id_type = do_ff_charmm
     412      113287 :             IF ((topology%conn_type == do_conn_g96) .OR. (topology%conn_type == do_conn_g87)) THEN
     413         132 :                bond_list(ibond)%itype = conn_info%bond_type(jind)
     414             :             END IF
     415             :             !point this to the right bond_kind_type if using force field
     416      113287 :             NULLIFY (bond_list(ibond)%bond_kind)
     417      248298 :             IF (iw > 0) THEN
     418         135 :                WRITE (iw, '(7X,A,I3,1X,A,I5,I5,1X,A,I5,I5)') "molecule_kind", &
     419         135 :                   i, "  intra bond", &
     420         135 :                   conn_info%bond_a(jind), &
     421         135 :                   conn_info%bond_b(jind), &
     422         135 :                   "offset number at", &
     423         135 :                   conn_info%bond_a(jind) - first + 1, &
     424         270 :                   conn_info%bond_b(jind) - first + 1
     425             :             END IF
     426             :          END DO
     427      270054 :          DO j = bnd_ctype(1, i), bnd_ctype(2, i)
     428      135043 :             IF (j == 0) CYCLE
     429        2866 :             ibond = ibond + 1
     430        2866 :             jind = map_cvars(j)
     431        2866 :             min_index = MIN(conn_info%c_bond_a(jind), conn_info%c_bond_b(jind))
     432        2866 :             first = first_list(map_atom_mol(min_index))
     433        2866 :             bond_list(ibond)%a = conn_info%c_bond_a(jind) - first + 1
     434        2866 :             bond_list(ibond)%b = conn_info%c_bond_b(jind) - first + 1
     435             :             ! Set by default id_type to charmm and modify when handling the forcefield
     436        2866 :             bond_list(ibond)%id_type = do_ff_charmm
     437        2866 :             IF ((topology%conn_type == do_conn_g96) .OR. (topology%conn_type == do_conn_g87)) THEN
     438           0 :                bond_list(ibond)%itype = conn_info%c_bond_type(jind)
     439             :             END IF
     440             :             !point this to the right bond_kind_type if using force field
     441        2866 :             NULLIFY (bond_list(ibond)%bond_kind)
     442      137877 :             IF (iw > 0) THEN
     443           2 :                WRITE (iw, '(7X,A,I3,1X,A,I5,I5,1X,A,I5,I5)') "molecule_kind", &
     444           2 :                   i, " inter bond", &
     445           2 :                   conn_info%c_bond_a(jind), &
     446           2 :                   conn_info%c_bond_b(jind), &
     447           2 :                   "offset number at", &
     448           2 :                   conn_info%c_bond_a(jind) - first + 1, &
     449           4 :                   conn_info%c_bond_b(jind) - first + 1
     450             :             END IF
     451             :          END DO
     452             :          CALL set_molecule_kind(molecule_kind=molecule_kind, &
     453      145167 :                                 nbond=SIZE(bond_list), bond_list=bond_list)
     454             :       END DO
     455       10156 :       IF ((nval_tot1 /= nval_tot2) .AND. (output_unit > 0)) THEN
     456           0 :          WRITE (output_unit, '(/)')
     457           0 :          WRITE (output_unit, '(T5,A)') "ERROR| Mismatching found between the total number  of atoms"
     458           0 :          WRITE (output_unit, '(T5,A)') "ERROR| and the number of atoms computed multiplying the Nr."
     459           0 :          WRITE (output_unit, '(T5,A)') "ERROR| of molecules by the  number of atoms  building  that"
     460           0 :          WRITE (output_unit, '(T5,A)') "ERROR| kind of molecule."
     461           0 :          WRITE (output_unit, '(T5,A)') "ERROR| This happens when the connectivity is wrongly  built"
     462           0 :          WRITE (output_unit, '(T5,A)') "ERROR| One example could be two same kind of molecules have"
     463           0 :          WRITE (output_unit, '(T5,A)') "ERROR| a different number of atoms. Check the connectivity!"
     464             :       END IF
     465       10156 :       CPASSERT(nval_tot1 == nval_tot2)
     466       10156 :       DEALLOCATE (map_var_mol)
     467       10156 :       DEALLOCATE (map_cvar_mol)
     468       10156 :       DEALLOCATE (map_vars)
     469       10156 :       DEALLOCATE (map_cvars)
     470       10156 :       DEALLOCATE (bnd_type)
     471       10156 :       DEALLOCATE (bnd_ctype)
     472       10156 :       CALL timestop(handle2)
     473             :       !-----------------------------------------------------------------------------
     474             :       !-----------------------------------------------------------------------------
     475             :       ! 11. Set the molecule_kind%[nbend,bend_list] via set_molecule_kind
     476             :       !-----------------------------------------------------------------------------
     477             :       ! Allocate c_var_a, c_var_b, c_var_c, c_var_type
     478       10156 :       CALL timeset(routineN//"_11_pre", handle2)
     479       10156 :       idim = 0
     480       10156 :       ALLOCATE (c_var_a(idim))
     481       10156 :       ALLOCATE (c_var_b(idim))
     482       10156 :       ALLOCATE (c_var_c(idim))
     483       10156 :       found = ASSOCIATED(conn_info%theta_type)
     484       10156 :       IF (found) THEN
     485          14 :          ALLOCATE (c_var_type(idim))
     486             :       END IF
     487       10156 :       IF (ASSOCIATED(conn_info%c_bond_a) .AND. ASSOCIATED(conn_info%theta_a)) THEN
     488      230110 :          DO j = 1, SIZE(conn_info%theta_a)
     489      222271 :             j1 = map_atom_mol(conn_info%theta_a(j))
     490      222271 :             j2 = map_atom_mol(conn_info%theta_b(j))
     491      222271 :             j3 = map_atom_mol(conn_info%theta_c(j))
     492      230110 :             IF (j1 /= j2 .OR. j2 /= j3) THEN
     493       11392 :                idim = idim + 1
     494             :             END IF
     495             :          END DO
     496        7839 :          CALL reallocate(c_var_a, 1, idim)
     497        7839 :          CALL reallocate(c_var_b, 1, idim)
     498        7839 :          CALL reallocate(c_var_c, 1, idim)
     499        7839 :          IF (found) THEN
     500           0 :             CALL reallocate(c_var_type, 1, idim)
     501             :          END IF
     502        7839 :          idim = 0
     503      230110 :          DO j = 1, SIZE(conn_info%theta_a)
     504      222271 :             j1 = map_atom_mol(conn_info%theta_a(j))
     505      222271 :             j2 = map_atom_mol(conn_info%theta_b(j))
     506      222271 :             j3 = map_atom_mol(conn_info%theta_c(j))
     507      230110 :             IF (j1 /= j2 .OR. j2 /= j3) THEN
     508       11392 :                idim = idim + 1
     509       11392 :                c_var_a(idim) = conn_info%theta_a(j)
     510       11392 :                c_var_b(idim) = conn_info%theta_b(j)
     511       11392 :                c_var_c(idim) = conn_info%theta_c(j)
     512       11392 :                IF (found) THEN
     513           0 :                   c_var_type(idim) = conn_info%theta_type(j)
     514             :                END IF
     515             :             END IF
     516             :          END DO
     517             :       END IF
     518       10156 :       CALL timestop(handle2)
     519       10156 :       CALL timeset(routineN//"_11", handle2)
     520             :       ! map bends on molecules
     521       10156 :       nvar1 = 0
     522       10156 :       nvar2 = 0
     523             :       NULLIFY (map_vars, map_cvars, bnd_type, bnd_ctype)
     524       10156 :       IF (ASSOCIATED(conn_info%theta_a)) nvar1 = SIZE(conn_info%theta_a)
     525       10156 :       IF (ASSOCIATED(c_var_a)) nvar2 = SIZE(c_var_a)
     526       10156 :       nval_tot1 = nvar1
     527       10156 :       nval_tot2 = 0
     528       22661 :       ALLOCATE (map_var_mol(nvar1))
     529       20466 :       ALLOCATE (map_cvar_mol(nvar2))
     530      281653 :       map_var_mol = -1
     531       21548 :       map_cvar_mol = -1
     532      281653 :       DO i = 1, nvar1
     533      271497 :          j1 = map_atom_mol(conn_info%theta_a(i))
     534      271497 :          j2 = map_atom_mol(conn_info%theta_b(i))
     535      271497 :          j3 = map_atom_mol(conn_info%theta_c(i))
     536      281653 :          IF (j1 == j2 .AND. j2 == j3) THEN
     537      260105 :             IF (j1 > 0) map_var_mol(i) = map_atom_type(conn_info%theta_a(i))
     538             :          END IF
     539             :       END DO
     540       21548 :       DO i = 1, nvar2
     541       11392 :          min_index = MIN(c_var_a(i), c_var_b(i), c_var_c(i))
     542       11392 :          j1 = map_atom_mol(min_index)
     543       21548 :          IF (j1 > 0) map_cvar_mol(i) = map_atom_type(min_index)
     544             :       END DO
     545       10156 :       CALL find_bnd_typ(topology%nmol_type, map_vars, map_var_mol, bnd_type, nvar1)
     546       10156 :       CALL find_bnd_typ(topology%nmol_type, map_cvars, map_cvar_mol, bnd_ctype, nvar2)
     547      145167 :       DO i = 1, topology%nmol_type
     548      135011 :          intra_bends = 0
     549      135011 :          inter_bends = 0
     550      194517 :          IF (ALL(bnd_type(:, i) > 0)) THEN
     551       29753 :             intra_bends = bnd_type(2, i) - bnd_type(1, i) + 1
     552             :          END IF
     553      140679 :          IF (ALL(bnd_ctype(:, i) > 0)) THEN
     554        2834 :             inter_bends = bnd_ctype(2, i) - bnd_ctype(1, i) + 1
     555             :          END IF
     556      135011 :          ibend = intra_bends + inter_bends
     557      135011 :          IF (iw > 0) THEN
     558         434 :             WRITE (iw, *) "    Total number of angles for molecule type ", i, " :", ibend
     559         434 :             WRITE (iw, *) "    intra (angles inside  molecules) :: ", intra_bends
     560         434 :             WRITE (iw, *) "    inter (angles between molecules) :: ", inter_bends
     561             :          END IF
     562      135011 :          molecule_kind => molecule_kind_set(i)
     563      135011 :          nval_tot2 = nval_tot2 + ibend*SIZE(molecule_kind%molecule_list)
     564      440479 :          ALLOCATE (bend_list(ibend))
     565      135011 :          ibend = 0
     566      369579 :          DO j = bnd_type(1, i), bnd_type(2, i)
     567      234568 :             IF (j == 0) CYCLE
     568      129310 :             ibend = ibend + 1
     569      129310 :             jind = map_vars(j)
     570      129310 :             first = first_list(map_atom_mol(conn_info%theta_a(jind)))
     571      129310 :             bend_list(ibend)%a = conn_info%theta_a(jind) - first + 1
     572      129310 :             bend_list(ibend)%b = conn_info%theta_b(jind) - first + 1
     573      129310 :             bend_list(ibend)%c = conn_info%theta_c(jind) - first + 1
     574             :             ! Set by default id_type to charmm and modify when handling the forcefield
     575      129310 :             bend_list(ibend)%id_type = do_ff_charmm
     576      129310 :             IF ((topology%conn_type == do_conn_g96) .OR. (topology%conn_type == do_conn_g87)) THEN
     577         158 :                bend_list(ibend)%itype = conn_info%theta_type(jind)
     578             :             END IF
     579             :             !point this to the right bend_kind_type if using force field
     580      129310 :             NULLIFY (bend_list(ibend)%bend_kind)
     581      264321 :             IF (iw > 0) THEN
     582             :                WRITE (iw, '(7X,A,I3,1X,A,I5,I5,I5,1X,A,I5,I5,I5)') &
     583         229 :                   "molecule_kind", ikind, "intra bend", &
     584         229 :                   conn_info%theta_a(jind), &
     585         229 :                   conn_info%theta_b(jind), &
     586         229 :                   conn_info%theta_c(jind), &
     587         229 :                   "offset number at", &
     588         229 :                   conn_info%theta_a(jind) - first + 1, &
     589         229 :                   conn_info%theta_b(jind) - first + 1, &
     590         458 :                   conn_info%theta_c(jind) - first + 1
     591             :             END IF
     592             :          END DO
     593      278580 :          DO j = bnd_ctype(1, i), bnd_ctype(2, i)
     594      143569 :             IF (j == 0) CYCLE
     595       11392 :             ibend = ibend + 1
     596       11392 :             jind = map_cvars(j)
     597       11392 :             min_index = MIN(c_var_a(jind), c_var_b(jind), c_var_c(jind))
     598       11392 :             first = first_list(map_atom_mol(min_index))
     599       11392 :             bend_list(ibend)%a = c_var_a(jind) - first + 1
     600       11392 :             bend_list(ibend)%b = c_var_b(jind) - first + 1
     601       11392 :             bend_list(ibend)%c = c_var_c(jind) - first + 1
     602             :             ! Set by default id_type to charmm and modify when handling the forcefield
     603       11392 :             bend_list(ibend)%id_type = do_ff_charmm
     604       11392 :             IF ((topology%conn_type == do_conn_g96) .OR. (topology%conn_type == do_conn_g87)) THEN
     605           0 :                bend_list(ibend)%itype = c_var_type(jind)
     606             :             END IF
     607             :             !point this to the right bend_kind_type if using force field
     608       11392 :             NULLIFY (bend_list(ibend)%bend_kind)
     609      146403 :             IF (iw > 0) THEN
     610             :                WRITE (iw, '(7X,A,I3,1X,A,I5,I5,I5,1X,A,I5,I5,I5)') &
     611           8 :                   "molecule_kind", ikind, "inter bend", &
     612           8 :                   c_var_a(jind), &
     613           8 :                   c_var_b(jind), &
     614           8 :                   c_var_c(jind), &
     615           8 :                   "offset number at", &
     616           8 :                   c_var_a(jind) - first + 1, &
     617           8 :                   c_var_b(jind) - first + 1, &
     618          16 :                   c_var_c(jind) - first + 1
     619             :             END IF
     620             :          END DO
     621             :          CALL set_molecule_kind(molecule_kind=molecule_kind, &
     622      145167 :                                 nbend=SIZE(bend_list), bend_list=bend_list)
     623             :       END DO
     624       10156 :       CPASSERT(nval_tot1 == nval_tot2)
     625       10156 :       DEALLOCATE (map_var_mol)
     626       10156 :       DEALLOCATE (map_cvar_mol)
     627       10156 :       DEALLOCATE (map_vars)
     628       10156 :       DEALLOCATE (map_cvars)
     629       10156 :       DEALLOCATE (bnd_type)
     630       10156 :       DEALLOCATE (bnd_ctype)
     631       10156 :       DEALLOCATE (c_var_a)
     632       10156 :       DEALLOCATE (c_var_b)
     633       10156 :       DEALLOCATE (c_var_c)
     634       10156 :       IF (found) THEN
     635          14 :          DEALLOCATE (c_var_type)
     636             :       END IF
     637       10156 :       CALL timestop(handle2)
     638             :       !-----------------------------------------------------------------------------
     639             :       !-----------------------------------------------------------------------------
     640             :       ! 12. Set the molecule_kind%[nub,ub_list] via set_molecule_kind
     641             :       !-----------------------------------------------------------------------------
     642       10156 :       CALL timeset(routineN//"_12_pre", handle2)
     643       10156 :       idim = 0
     644       10156 :       ALLOCATE (c_var_a(idim))
     645       10156 :       ALLOCATE (c_var_b(idim))
     646       10156 :       ALLOCATE (c_var_c(idim))
     647       10156 :       IF (ASSOCIATED(conn_info%c_bond_a) .AND. ASSOCIATED(conn_info%ub_a)) THEN
     648      230110 :          DO j = 1, SIZE(conn_info%ub_a)
     649      222271 :             j1 = map_atom_mol(conn_info%ub_a(j))
     650      222271 :             j2 = map_atom_mol(conn_info%ub_b(j))
     651      222271 :             j3 = map_atom_mol(conn_info%ub_c(j))
     652      230110 :             IF (j1 /= j2 .OR. j2 /= j3) THEN
     653       11392 :                idim = idim + 1
     654             :             END IF
     655             :          END DO
     656        7839 :          CALL reallocate(c_var_a, 1, idim)
     657        7839 :          CALL reallocate(c_var_b, 1, idim)
     658        7839 :          CALL reallocate(c_var_c, 1, idim)
     659        7839 :          idim = 0
     660      230110 :          DO j = 1, SIZE(conn_info%ub_a)
     661      222271 :             j1 = map_atom_mol(conn_info%ub_a(j))
     662      222271 :             j2 = map_atom_mol(conn_info%ub_b(j))
     663      222271 :             j3 = map_atom_mol(conn_info%ub_c(j))
     664      230110 :             IF (j1 /= j2 .OR. j2 /= j3) THEN
     665       11392 :                idim = idim + 1
     666       11392 :                c_var_a(idim) = conn_info%ub_a(j)
     667       11392 :                c_var_b(idim) = conn_info%ub_b(j)
     668       11392 :                c_var_c(idim) = conn_info%ub_c(j)
     669             :             END IF
     670             :          END DO
     671             :       END IF
     672       10156 :       CALL timestop(handle2)
     673       10156 :       CALL timeset(routineN//"_12", handle2)
     674             :       ! map UBs on molecules
     675       10156 :       nvar1 = 0
     676       10156 :       nvar2 = 0
     677             :       NULLIFY (map_vars, map_cvars, bnd_type, bnd_ctype)
     678       10156 :       IF (ASSOCIATED(conn_info%ub_a)) nvar1 = SIZE(conn_info%ub_a)
     679       10156 :       IF (ASSOCIATED(c_var_a)) nvar2 = SIZE(c_var_a)
     680       10156 :       nval_tot1 = nvar1
     681       10156 :       nval_tot2 = 0
     682       22647 :       ALLOCATE (map_var_mol(nvar1))
     683       20466 :       ALLOCATE (map_cvar_mol(nvar2))
     684      281495 :       map_var_mol = -1
     685       21548 :       map_cvar_mol = -1
     686      281495 :       DO i = 1, nvar1
     687      271339 :          j1 = map_atom_mol(conn_info%ub_a(i))
     688      271339 :          j2 = map_atom_mol(conn_info%ub_b(i))
     689      271339 :          j3 = map_atom_mol(conn_info%ub_c(i))
     690      281495 :          IF (j1 == j2 .AND. j2 == j3) THEN
     691      259947 :             IF (j1 > 0) map_var_mol(i) = map_atom_type(conn_info%ub_a(i))
     692             :          END IF
     693             :       END DO
     694       21548 :       DO i = 1, nvar2
     695       11392 :          min_index = MIN(c_var_a(i), c_var_b(i), c_var_c(i))
     696       11392 :          j1 = map_atom_mol(min_index)
     697       21548 :          IF (j1 > 0) map_cvar_mol(i) = map_atom_type(min_index)
     698             :       END DO
     699       10156 :       CALL find_bnd_typ(topology%nmol_type, map_vars, map_var_mol, bnd_type, nvar1)
     700       10156 :       CALL find_bnd_typ(topology%nmol_type, map_cvars, map_cvar_mol, bnd_ctype, nvar2)
     701      145167 :       DO i = 1, topology%nmol_type
     702      135011 :          intra_ubs = 0
     703      135011 :          inter_ubs = 0
     704      194489 :          IF (ALL(bnd_type(:, i) > 0)) THEN
     705       29739 :             intra_ubs = bnd_type(2, i) - bnd_type(1, i) + 1
     706             :          END IF
     707      140679 :          IF (ALL(bnd_ctype(:, i) > 0)) THEN
     708        2834 :             inter_ubs = bnd_ctype(2, i) - bnd_ctype(1, i) + 1
     709             :          END IF
     710      135011 :          iub = intra_ubs + inter_ubs
     711      135011 :          IF (iw > 0) THEN
     712         434 :             WRITE (iw, *) "    Total number of Urey-Bradley for molecule type ", i, " :", iub
     713         434 :             WRITE (iw, *) "    intra (UB inside  molecules) :: ", intra_ubs
     714         434 :             WRITE (iw, *) "    inter (UB between molecules) :: ", inter_ubs
     715             :          END IF
     716      135011 :          molecule_kind => molecule_kind_set(i)
     717      135011 :          nval_tot2 = nval_tot2 + iub*SIZE(molecule_kind%molecule_list)
     718      440307 :          ALLOCATE (ub_list(iub))
     719      135011 :          iub = 0
     720      369435 :          DO j = bnd_type(1, i), bnd_type(2, i)
     721      234424 :             IF (j == 0) CYCLE
     722      129152 :             iub = iub + 1
     723      129152 :             jind = map_vars(j)
     724      129152 :             first = first_list(map_atom_mol(conn_info%ub_a(jind)))
     725      129152 :             ub_list(iub)%a = conn_info%ub_a(jind) - first + 1
     726      129152 :             ub_list(iub)%b = conn_info%ub_b(jind) - first + 1
     727      129152 :             ub_list(iub)%c = conn_info%ub_c(jind) - first + 1
     728      129152 :             ub_list(iub)%id_type = do_ff_charmm
     729             :             !point this to the right ub_kind_type if using force field
     730      129152 :             NULLIFY (ub_list(iub)%ub_kind)
     731      264163 :             IF (iw > 0) THEN
     732             :                WRITE (iw, '(7X,A,I3,1X,A,I5,I5,I5,1X,A,I5,I5,I5)') &
     733         229 :                   "molecule_kind", i, "intra UB", &
     734         229 :                   conn_info%ub_a(jind), &
     735         229 :                   conn_info%ub_b(jind), &
     736         229 :                   conn_info%ub_c(jind), &
     737         229 :                   "offset number at", &
     738         229 :                   conn_info%ub_a(jind) - first + 1, &
     739         229 :                   conn_info%ub_b(jind) - first + 1, &
     740         458 :                   conn_info%ub_c(jind) - first + 1
     741             :             END IF
     742             :          END DO
     743      278580 :          DO j = bnd_ctype(1, i), bnd_ctype(2, i)
     744      143569 :             IF (j == 0) CYCLE
     745       11392 :             iub = iub + 1
     746       11392 :             jind = map_cvars(j)
     747       11392 :             min_index = MIN(c_var_a(jind), c_var_b(jind), c_var_c(jind))
     748       11392 :             first = first_list(map_atom_mol(min_index))
     749       11392 :             ub_list(iub)%a = c_var_a(jind) - first + 1
     750       11392 :             ub_list(iub)%b = c_var_b(jind) - first + 1
     751       11392 :             ub_list(iub)%c = c_var_c(jind) - first + 1
     752       11392 :             ub_list(iub)%id_type = do_ff_charmm
     753             :             !point this to the right ub_kind_type if using force field
     754       11392 :             NULLIFY (ub_list(iub)%ub_kind)
     755      146403 :             IF (iw > 0) THEN
     756             :                WRITE (iw, '(7X,A,I3,1X,A,I5,I5,I5,1X,A,I5,I5,I5)') &
     757           8 :                   "molecule_kind", i, "inter UB", &
     758           8 :                   c_var_a(jind), &
     759           8 :                   c_var_b(jind), &
     760           8 :                   c_var_c(jind), &
     761           8 :                   "offset number at", &
     762           8 :                   c_var_a(jind) - first + 1, &
     763           8 :                   c_var_b(jind) - first + 1, &
     764          16 :                   c_var_c(jind) - first + 1
     765             :             END IF
     766             :          END DO
     767             :          CALL set_molecule_kind(molecule_kind=molecule_kind, &
     768      145167 :                                 nub=SIZE(ub_list), ub_list=ub_list)
     769             :       END DO
     770       10156 :       CPASSERT(nval_tot1 == nval_tot2)
     771       10156 :       DEALLOCATE (map_var_mol)
     772       10156 :       DEALLOCATE (map_cvar_mol)
     773       10156 :       DEALLOCATE (map_vars)
     774       10156 :       DEALLOCATE (map_cvars)
     775       10156 :       DEALLOCATE (bnd_type)
     776       10156 :       DEALLOCATE (bnd_ctype)
     777       10156 :       DEALLOCATE (c_var_a)
     778       10156 :       DEALLOCATE (c_var_b)
     779       10156 :       DEALLOCATE (c_var_c)
     780       10156 :       CALL timestop(handle2)
     781             :       !-----------------------------------------------------------------------------
     782             :       !-----------------------------------------------------------------------------
     783             :       ! 13. Set the molecule_kind%[ntorsion,torsion_list] via set_molecule_kind
     784             :       !-----------------------------------------------------------------------------
     785             :       ! Allocate c_var_a, c_var_b, c_var_c, c_var_d, c_var_type
     786       10156 :       CALL timeset(routineN//"_13_pre", handle2)
     787       10156 :       idim = 0
     788       10156 :       ALLOCATE (c_var_a(idim))
     789       10156 :       ALLOCATE (c_var_b(idim))
     790       10156 :       ALLOCATE (c_var_c(idim))
     791       10156 :       ALLOCATE (c_var_d(idim))
     792       10156 :       found = ASSOCIATED(conn_info%phi_type)
     793       10156 :       IF (found) THEN
     794          14 :          ALLOCATE (c_var_type(idim))
     795             :       END IF
     796       10156 :       IF (ASSOCIATED(conn_info%c_bond_a) .AND. ASSOCIATED(conn_info%phi_a)) THEN
     797      142198 :          DO j = 1, SIZE(conn_info%phi_a)
     798      134359 :             j1 = map_atom_mol(conn_info%phi_a(j))
     799      134359 :             j2 = map_atom_mol(conn_info%phi_b(j))
     800      134359 :             j3 = map_atom_mol(conn_info%phi_c(j))
     801      134359 :             j4 = map_atom_mol(conn_info%phi_d(j))
     802      142198 :             IF (j1 /= j2 .OR. j2 /= j3 .OR. j3 /= j4) THEN
     803       31630 :                idim = idim + 1
     804             :             END IF
     805             :          END DO
     806        7839 :          CALL reallocate(c_var_a, 1, idim)
     807        7839 :          CALL reallocate(c_var_b, 1, idim)
     808        7839 :          CALL reallocate(c_var_c, 1, idim)
     809        7839 :          CALL reallocate(c_var_d, 1, idim)
     810        7839 :          IF (found) THEN
     811           0 :             CALL reallocate(c_var_type, 1, idim)
     812             :          END IF
     813        7839 :          idim = 0
     814      142198 :          DO j = 1, SIZE(conn_info%phi_a)
     815      134359 :             j1 = map_atom_mol(conn_info%phi_a(j))
     816      134359 :             j2 = map_atom_mol(conn_info%phi_b(j))
     817      134359 :             j3 = map_atom_mol(conn_info%phi_c(j))
     818      134359 :             j4 = map_atom_mol(conn_info%phi_d(j))
     819      142198 :             IF (j1 /= j2 .OR. j2 /= j3 .OR. j3 /= j4) THEN
     820       31630 :                idim = idim + 1
     821       31630 :                c_var_a(idim) = conn_info%phi_a(j)
     822       31630 :                c_var_b(idim) = conn_info%phi_b(j)
     823       31630 :                c_var_c(idim) = conn_info%phi_c(j)
     824       31630 :                c_var_d(idim) = conn_info%phi_d(j)
     825       31630 :                IF (found) THEN
     826           0 :                   c_var_type(idim) = conn_info%phi_type(j)
     827             :                END IF
     828             :             END IF
     829             :          END DO
     830             :       END IF
     831       10156 :       CALL timestop(handle2)
     832       10156 :       CALL timeset(routineN//"_13", handle2)
     833             :       ! map torsions on molecules
     834       10156 :       nvar1 = 0
     835       10156 :       nvar2 = 0
     836             :       NULLIFY (map_vars, map_cvars, bnd_type, bnd_ctype)
     837       10156 :       IF (ASSOCIATED(conn_info%phi_a)) nvar1 = SIZE(conn_info%phi_a)
     838       10156 :       IF (ASSOCIATED(c_var_a)) nvar2 = SIZE(c_var_a)
     839       10156 :       nval_tot1 = nvar1
     840       10156 :       nval_tot2 = 0
     841       20958 :       ALLOCATE (map_var_mol(nvar1))
     842       20464 :       ALLOCATE (map_cvar_mol(nvar2))
     843      175593 :       map_var_mol = -1
     844       41786 :       map_cvar_mol = -1
     845      175593 :       DO i = 1, nvar1
     846      165437 :          j1 = map_atom_mol(conn_info%phi_a(i))
     847      165437 :          j2 = map_atom_mol(conn_info%phi_b(i))
     848      165437 :          j3 = map_atom_mol(conn_info%phi_c(i))
     849      165437 :          j4 = map_atom_mol(conn_info%phi_d(i))
     850      175593 :          IF (j1 == j2 .AND. j2 == j3 .AND. j3 == j4) THEN
     851      133807 :             IF (j1 > 0) map_var_mol(i) = map_atom_type(conn_info%phi_a(i))
     852             :          END IF
     853             :       END DO
     854       41786 :       DO i = 1, nvar2
     855       31630 :          min_index = MIN(c_var_a(i), c_var_b(i), c_var_c(i), c_var_d(i))
     856       31630 :          j1 = map_atom_mol(min_index)
     857       41786 :          IF (j1 > 0) map_cvar_mol(i) = map_atom_type(min_index)
     858             :       END DO
     859       10156 :       CALL find_bnd_typ(topology%nmol_type, map_vars, map_var_mol, bnd_type, nvar1)
     860       10156 :       CALL find_bnd_typ(topology%nmol_type, map_cvars, map_cvar_mol, bnd_ctype, nvar2)
     861      145167 :       DO i = 1, topology%nmol_type
     862      135011 :          intra_torsions = 0
     863      135011 :          inter_torsions = 0
     864      146187 :          IF (ALL(bnd_type(:, i) > 0)) THEN
     865        5588 :             intra_torsions = bnd_type(2, i) - bnd_type(1, i) + 1
     866             :          END IF
     867      140675 :          IF (ALL(bnd_ctype(:, i) > 0)) THEN
     868        2832 :             inter_torsions = bnd_ctype(2, i) - bnd_ctype(1, i) + 1
     869             :          END IF
     870      135011 :          itorsion = intra_torsions + inter_torsions
     871      135011 :          IF (iw > 0) THEN
     872         434 :             WRITE (iw, *) "    Total number of torsions for molecule type ", i, " :", itorsion
     873         434 :             WRITE (iw, *) "    intra (torsions inside  molecules) :: ", intra_torsions
     874         434 :             WRITE (iw, *) "    inter (torsions between molecules) :: ", inter_torsions
     875             :          END IF
     876      135011 :          molecule_kind => molecule_kind_set(i)
     877      135011 :          nval_tot2 = nval_tot2 + itorsion*SIZE(molecule_kind%molecule_list)
     878      432199 :          ALLOCATE (torsion_list(itorsion))
     879      135011 :          itorsion = 0
     880      389393 :          DO j = bnd_type(1, i), bnd_type(2, i)
     881      254382 :             IF (j == 0) CYCLE
     882      124959 :             itorsion = itorsion + 1
     883      124959 :             jind = map_vars(j)
     884      124959 :             first = first_list(map_atom_mol(conn_info%phi_a(jind)))
     885      124959 :             torsion_list(itorsion)%a = conn_info%phi_a(jind) - first + 1
     886      124959 :             torsion_list(itorsion)%b = conn_info%phi_b(jind) - first + 1
     887      124959 :             torsion_list(itorsion)%c = conn_info%phi_c(jind) - first + 1
     888      124959 :             torsion_list(itorsion)%d = conn_info%phi_d(jind) - first + 1
     889             :             ! Set by default id_type to charmm and modify when handling the forcefield
     890      124959 :             torsion_list(itorsion)%id_type = do_ff_charmm
     891      124959 :             IF ((topology%conn_type == do_conn_g96) .OR. (topology%conn_type == do_conn_g87)) THEN
     892         312 :                torsion_list(itorsion)%itype = conn_info%phi_type(jind)
     893             :             END IF
     894             :             !point this to the right torsion_kind_type if using force field
     895      124959 :             NULLIFY (torsion_list(itorsion)%torsion_kind)
     896      259970 :             IF (iw > 0) THEN
     897             :                WRITE (iw, '(7X,A,I3,1X,A,I4,I4,I4,I4,1X,A,I4,I4,I4,I4)') &
     898         366 :                   "molecule_kind", i, "intra TOR", &
     899         366 :                   conn_info%phi_a(jind), &
     900         366 :                   conn_info%phi_b(jind), &
     901         366 :                   conn_info%phi_c(jind), &
     902         366 :                   conn_info%phi_d(jind), &
     903         366 :                   "offset number at", &
     904         366 :                   conn_info%phi_a(jind) - first + 1, &
     905         366 :                   conn_info%phi_b(jind) - first + 1, &
     906         366 :                   conn_info%phi_c(jind) - first + 1, &
     907         732 :                   conn_info%phi_d(jind) - first + 1
     908             :             END IF
     909             :          END DO
     910      298820 :          DO j = bnd_ctype(1, i), bnd_ctype(2, i)
     911      163809 :             IF (j == 0) CYCLE
     912       31630 :             itorsion = itorsion + 1
     913       31630 :             jind = map_cvars(j)
     914       31630 :             min_index = MIN(c_var_a(jind), c_var_b(jind), c_var_c(jind), c_var_d(jind))
     915       31630 :             first = first_list(map_atom_mol(min_index))
     916       31630 :             torsion_list(itorsion)%a = c_var_a(jind) - first + 1
     917       31630 :             torsion_list(itorsion)%b = c_var_b(jind) - first + 1
     918       31630 :             torsion_list(itorsion)%c = c_var_c(jind) - first + 1
     919       31630 :             torsion_list(itorsion)%d = c_var_d(jind) - first + 1
     920             :             ! Set by default id_type to charmm and modify when handling the forcefield
     921       31630 :             torsion_list(itorsion)%id_type = do_ff_charmm
     922       31630 :             IF ((topology%conn_type == do_conn_g96) .OR. (topology%conn_type == do_conn_g87)) THEN
     923           0 :                torsion_list(itorsion)%itype = c_var_type(jind)
     924             :             END IF
     925             :             !point this to the right torsion_kind_type if using force field
     926       31630 :             NULLIFY (torsion_list(itorsion)%torsion_kind)
     927      166641 :             IF (iw > 0) THEN
     928             :                WRITE (iw, '(7X,A,I3,1X,A,I4,I4,I4,I4,1X,A,I4,I4,I4,I4)') &
     929          34 :                   "molecule_kind", i, "inter TOR", &
     930          34 :                   c_var_a(jind), &
     931          34 :                   c_var_b(jind), &
     932          34 :                   c_var_c(jind), &
     933          34 :                   c_var_d(jind), &
     934          34 :                   "offset number at", &
     935          34 :                   c_var_a(jind) - first + 1, &
     936          34 :                   c_var_b(jind) - first + 1, &
     937          34 :                   c_var_c(jind) - first + 1, &
     938          68 :                   c_var_d(jind) - first + 1
     939             :             END IF
     940             :          END DO
     941             :          CALL set_molecule_kind(molecule_kind=molecule_kind, &
     942      145167 :                                 ntorsion=SIZE(torsion_list), torsion_list=torsion_list)
     943             :       END DO
     944       10156 :       CPASSERT(nval_tot1 == nval_tot2)
     945       10156 :       DEALLOCATE (map_var_mol)
     946       10156 :       DEALLOCATE (map_cvar_mol)
     947       10156 :       DEALLOCATE (map_vars)
     948       10156 :       DEALLOCATE (map_cvars)
     949       10156 :       DEALLOCATE (bnd_type)
     950       10156 :       DEALLOCATE (bnd_ctype)
     951       10156 :       DEALLOCATE (c_var_a)
     952       10156 :       DEALLOCATE (c_var_b)
     953       10156 :       DEALLOCATE (c_var_c)
     954       10156 :       DEALLOCATE (c_var_d)
     955       10156 :       IF (found) THEN
     956          14 :          DEALLOCATE (c_var_type)
     957             :       END IF
     958       10156 :       CALL timestop(handle2)
     959             :       !-----------------------------------------------------------------------------
     960             :       !-----------------------------------------------------------------------------
     961             :       ! 14. Set the molecule_kind%[nimpr,impr_list] via set_molecule_kind
     962             :       !     Also set the molecule_kind%[nopbend,opbend_list]
     963             :       !-----------------------------------------------------------------------------
     964             :       ! Allocate c_var_a, c_var_b, c_var_c, c_var_d, c_var_type
     965       10156 :       CALL timeset(routineN//"_14_pre", handle2)
     966       10156 :       idim = 0
     967       10156 :       ALLOCATE (c_var_a(idim))
     968       10156 :       ALLOCATE (c_var_b(idim))
     969       10156 :       ALLOCATE (c_var_c(idim))
     970       10156 :       ALLOCATE (c_var_d(idim))
     971       10156 :       found = ASSOCIATED(conn_info%impr_type)
     972       10156 :       IF (found) THEN
     973          14 :          ALLOCATE (c_var_type(idim))
     974             :       END IF
     975       10156 :       IF (ASSOCIATED(conn_info%c_bond_a) .AND. ASSOCIATED(conn_info%impr_a)) THEN
     976       12253 :          DO j = 1, SIZE(conn_info%impr_a)
     977        4414 :             j1 = map_atom_mol(conn_info%impr_a(j))
     978        4414 :             j2 = map_atom_mol(conn_info%impr_b(j))
     979        4414 :             j3 = map_atom_mol(conn_info%impr_c(j))
     980        4414 :             j4 = map_atom_mol(conn_info%impr_d(j))
     981       12253 :             IF (j1 /= j2 .OR. j2 /= j3 .OR. j3 /= j4) THEN
     982        3124 :                idim = idim + 1
     983             :             END IF
     984             :          END DO
     985        7839 :          CALL reallocate(c_var_a, 1, idim)
     986        7839 :          CALL reallocate(c_var_b, 1, idim)
     987        7839 :          CALL reallocate(c_var_c, 1, idim)
     988        7839 :          CALL reallocate(c_var_d, 1, idim)
     989        7839 :          IF (found) THEN
     990           0 :             CALL reallocate(c_var_type, 1, idim)
     991             :          END IF
     992        7839 :          idim = 0
     993       12253 :          DO j = 1, SIZE(conn_info%impr_a)
     994        4414 :             j1 = map_atom_mol(conn_info%impr_a(j))
     995        4414 :             j2 = map_atom_mol(conn_info%impr_b(j))
     996        4414 :             j3 = map_atom_mol(conn_info%impr_c(j))
     997        4414 :             j4 = map_atom_mol(conn_info%impr_d(j))
     998       12253 :             IF (j1 /= j2 .OR. j2 /= j3 .OR. j3 /= j4) THEN
     999        3124 :                idim = idim + 1
    1000        3124 :                c_var_a(idim) = conn_info%impr_a(j)
    1001        3124 :                c_var_b(idim) = conn_info%impr_b(j)
    1002        3124 :                c_var_c(idim) = conn_info%impr_c(j)
    1003        3124 :                c_var_d(idim) = conn_info%impr_d(j)
    1004        3124 :                IF (found) THEN
    1005           0 :                   c_var_type(idim) = conn_info%impr_type(j)
    1006             :                END IF
    1007             :             END IF
    1008             :          END DO
    1009             :       END IF
    1010       10156 :       CALL timestop(handle2)
    1011       10156 :       CALL timeset(routineN//"_14", handle2)
    1012             :       ! map imprs on molecules
    1013       10156 :       nvar1 = 0
    1014       10156 :       nvar2 = 0
    1015             :       NULLIFY (map_vars, map_cvars, bnd_type, bnd_ctype)
    1016       10156 :       IF (ASSOCIATED(conn_info%impr_a)) nvar1 = SIZE(conn_info%impr_a)
    1017       10156 :       IF (ASSOCIATED(c_var_a)) nvar2 = SIZE(c_var_a)
    1018       10156 :       nval_tot1 = nvar1
    1019       10156 :       nval_tot2 = 0
    1020       20458 :       ALLOCATE (map_var_mol(nvar1))
    1021       20352 :       ALLOCATE (map_cvar_mol(nvar2))
    1022       15566 :       map_var_mol = -1
    1023       13280 :       map_cvar_mol = -1
    1024       15566 :       DO i = 1, nvar1
    1025        5410 :          j1 = map_atom_mol(conn_info%impr_a(i))
    1026        5410 :          j2 = map_atom_mol(conn_info%impr_b(i))
    1027        5410 :          j3 = map_atom_mol(conn_info%impr_c(i))
    1028        5410 :          j4 = map_atom_mol(conn_info%impr_d(i))
    1029       15566 :          IF (j1 == j2 .AND. j2 == j3 .AND. j3 == j4) THEN
    1030        2286 :             IF (j1 > 0) map_var_mol(i) = map_atom_type(conn_info%impr_a(i))
    1031             :          END IF
    1032             :       END DO
    1033       13280 :       DO i = 1, nvar2
    1034        3124 :          min_index = MIN(c_var_a(i), c_var_b(i), c_var_c(i), c_var_d(i))
    1035        3124 :          j1 = map_atom_mol(min_index)
    1036       13280 :          IF (j1 > 0) map_cvar_mol(i) = map_atom_type(min_index)
    1037             :       END DO
    1038       10156 :       CALL find_bnd_typ(topology%nmol_type, map_vars, map_var_mol, bnd_type, nvar1)
    1039       10156 :       CALL find_bnd_typ(topology%nmol_type, map_cvars, map_cvar_mol, bnd_ctype, nvar2)
    1040      145167 :       DO i = 1, topology%nmol_type
    1041      135011 :          intra_imprs = 0
    1042      135011 :          inter_imprs = 0
    1043      136099 :          IF (ALL(bnd_type(:, i) > 0)) THEN
    1044         544 :             intra_imprs = bnd_type(2, i) - bnd_type(1, i) + 1
    1045             :          END IF
    1046      138135 :          IF (ALL(bnd_ctype(:, i) > 0)) THEN
    1047        1562 :             inter_imprs = bnd_ctype(2, i) - bnd_ctype(1, i) + 1
    1048             :          END IF
    1049      135011 :          iimpr = intra_imprs + inter_imprs
    1050      135011 :          IF (iw > 0) THEN
    1051         434 :             WRITE (iw, *) "    Total number of imprs for molecule type ", i, " :", iimpr
    1052         434 :             WRITE (iw, *) "    intra (imprs inside  molecules) :: ", intra_imprs
    1053         434 :             WRITE (iw, *) "    inter (imprs between molecules) :: ", inter_imprs
    1054         434 :             WRITE (iw, *) "    Total number of opbends for molecule type ", i, " :", iimpr
    1055         434 :             WRITE (iw, *) "    intra (opbends inside  molecules) :: ", intra_imprs
    1056         434 :             WRITE (iw, *) "    inter (opbends between molecules) :: ", inter_imprs
    1057             :          END IF
    1058      135011 :          molecule_kind => molecule_kind_set(i)
    1059      135011 :          nval_tot2 = nval_tot2 + iimpr*SIZE(molecule_kind%molecule_list)
    1060      277094 :          ALLOCATE (impr_list(iimpr), STAT=stat)
    1061      142083 :          ALLOCATE (opbend_list(iimpr), STAT=stat)
    1062      135011 :          CPASSERT(stat == 0)
    1063      135011 :          iimpr = 0
    1064      271740 :          DO j = bnd_type(1, i), bnd_type(2, i)
    1065      136729 :             IF (j == 0) CYCLE
    1066        2262 :             iimpr = iimpr + 1
    1067        2262 :             jind = map_vars(j)
    1068        2262 :             first = first_list(map_atom_mol(conn_info%impr_a(jind)))
    1069        2262 :             impr_list(iimpr)%a = conn_info%impr_a(jind) - first + 1
    1070        2262 :             impr_list(iimpr)%b = conn_info%impr_b(jind) - first + 1
    1071        2262 :             impr_list(iimpr)%c = conn_info%impr_c(jind) - first + 1
    1072        2262 :             impr_list(iimpr)%d = conn_info%impr_d(jind) - first + 1
    1073             :             ! Atom sequence for improper is A B C D in which A is central atom,
    1074             :             ! B is deviating atom and C & D are secondairy atoms. Atom sequence for
    1075             :             ! opbend is B D C A in which A is central atom, B is deviating. Hence
    1076             :             ! to create an opbend out of an improper, B and D need to be interchanged.
    1077        2262 :             opbend_list(iimpr)%a = conn_info%impr_b(jind) - first + 1
    1078        2262 :             opbend_list(iimpr)%b = conn_info%impr_d(jind) - first + 1
    1079        2262 :             opbend_list(iimpr)%c = conn_info%impr_c(jind) - first + 1
    1080        2262 :             opbend_list(iimpr)%d = conn_info%impr_a(jind) - first + 1
    1081             :             ! Set by default id_type of improper to charmm and modify when handling the forcefield
    1082        2262 :             impr_list(iimpr)%id_type = do_ff_charmm
    1083             :             ! Set by default id_type of opbend to harmonic and modify when handling the forcefield
    1084        2262 :             opbend_list(iimpr)%id_type = do_ff_harmonic
    1085        2262 :             IF ((topology%conn_type == do_conn_g96) .OR. (topology%conn_type == do_conn_g87)) THEN
    1086           0 :                impr_list(iimpr)%itype = conn_info%impr_type(jind)
    1087             :             END IF
    1088             :             !point this to the right impr_kind_type if using force field
    1089        2262 :             NULLIFY (impr_list(iimpr)%impr_kind)
    1090        2262 :             NULLIFY (opbend_list(iimpr)%opbend_kind)
    1091      137273 :             IF (iw > 0) THEN
    1092             :                WRITE (iw, '(7X,A,I3,1X,A,I4,I4,I4,I4,1X,A,I4,I4,I4,I4)') &
    1093           0 :                   "molecule_kind", i, "intra IMPR", &
    1094           0 :                   conn_info%impr_a(jind), &
    1095           0 :                   conn_info%impr_b(jind), &
    1096           0 :                   conn_info%impr_c(jind), &
    1097           0 :                   conn_info%impr_d(jind), &
    1098           0 :                   "offset number at", &
    1099           0 :                   conn_info%impr_a(jind) - first + 1, &
    1100           0 :                   conn_info%impr_b(jind) - first + 1, &
    1101           0 :                   conn_info%impr_c(jind) - first + 1, &
    1102           0 :                   conn_info%impr_d(jind) - first + 1
    1103             :                WRITE (iw, '(7X,A,I3,1X,A,I4,I4,I4,I4,1X,A,I4,I4,I4,I4)') &
    1104           0 :                   "molecule_kind", i, "intra OPBEND", &
    1105           0 :                   conn_info%impr_b(jind), &
    1106           0 :                   conn_info%impr_d(jind), &
    1107           0 :                   conn_info%impr_c(jind), &
    1108           0 :                   conn_info%impr_a(jind), &
    1109           0 :                   "offset number at", &
    1110           0 :                   conn_info%impr_b(jind) - first + 1, &
    1111           0 :                   conn_info%impr_d(jind) - first + 1, &
    1112           0 :                   conn_info%impr_c(jind) - first + 1, &
    1113           0 :                   conn_info%impr_a(jind) - first + 1
    1114             :             END IF
    1115             :          END DO
    1116      271584 :          DO j = bnd_ctype(1, i), bnd_ctype(2, i)
    1117      136573 :             IF (j == 0) CYCLE
    1118        3124 :             iimpr = iimpr + 1
    1119        3124 :             jind = map_cvars(j)
    1120        3124 :             min_index = MIN(c_var_a(jind), c_var_b(jind), c_var_c(jind), c_var_d(jind))
    1121        3124 :             first = first_list(map_atom_mol(min_index))
    1122        3124 :             impr_list(iimpr)%a = c_var_a(jind) - first + 1
    1123        3124 :             impr_list(iimpr)%b = c_var_b(jind) - first + 1
    1124        3124 :             impr_list(iimpr)%c = c_var_c(jind) - first + 1
    1125        3124 :             impr_list(iimpr)%d = c_var_d(jind) - first + 1
    1126        3124 :             opbend_list(iimpr)%a = c_var_b(jind) - first + 1
    1127        3124 :             opbend_list(iimpr)%b = c_var_d(jind) - first + 1
    1128        3124 :             opbend_list(iimpr)%c = c_var_c(jind) - first + 1
    1129        3124 :             opbend_list(iimpr)%d = c_var_a(jind) - first + 1
    1130             :             ! Set by default id_type of improper to charmm and modify when handling the forcefield
    1131        3124 :             impr_list(iimpr)%id_type = do_ff_charmm
    1132             :             ! Set by default id_type of opbend to harmonic and modify when handling the forcefield
    1133        3124 :             opbend_list(iimpr)%id_type = do_ff_harmonic
    1134        3124 :             IF ((topology%conn_type == do_conn_g96) .OR. (topology%conn_type == do_conn_g87)) THEN
    1135           0 :                impr_list(iimpr)%itype = c_var_type(jind)
    1136             :             END IF
    1137             :             !point this to the right impr_kind_type and opbend_kind_type if using force field
    1138        3124 :             NULLIFY (impr_list(iimpr)%impr_kind)
    1139        3124 :             NULLIFY (opbend_list(iimpr)%opbend_kind)
    1140      138135 :             IF (iw > 0) THEN
    1141             :                WRITE (iw, '(7X,A,I3,1X,A,I4,I4,I4,I4,1X,A,I4,I4,I4,I4)') &
    1142           0 :                   "molecule_kind", i, "inter IMPR", &
    1143           0 :                   c_var_a(jind), &
    1144           0 :                   c_var_b(jind), &
    1145           0 :                   c_var_c(jind), &
    1146           0 :                   c_var_d(jind), &
    1147           0 :                   "offset number at", &
    1148           0 :                   c_var_a(jind) - first + 1, &
    1149           0 :                   c_var_b(jind) - first + 1, &
    1150           0 :                   c_var_c(jind) - first + 1, &
    1151           0 :                   c_var_d(jind) - first + 1
    1152             :                WRITE (iw, '(7X,A,I3,1X,A,I4,I4,I4,I4,1X,A,I4,I4,I4,I4)') &
    1153           0 :                   "molecule_kind", i, "inter OPBEND", &
    1154           0 :                   c_var_b(jind), &
    1155           0 :                   c_var_d(jind), &
    1156           0 :                   c_var_c(jind), &
    1157           0 :                   c_var_a(jind), &
    1158           0 :                   "offset number at", &
    1159           0 :                   c_var_b(jind) - first + 1, &
    1160           0 :                   c_var_d(jind) - first + 1, &
    1161           0 :                   c_var_c(jind) - first + 1, &
    1162           0 :                   c_var_a(jind) - first + 1
    1163             :             END IF
    1164             :          END DO
    1165             :          CALL set_molecule_kind(molecule_kind=molecule_kind, &
    1166      135011 :                                 nimpr=SIZE(impr_list), impr_list=impr_list)
    1167             :          CALL set_molecule_kind(molecule_kind=molecule_kind, &
    1168      145167 :                                 nopbend=SIZE(opbend_list), opbend_list=opbend_list)
    1169             :       END DO
    1170       10156 :       CPASSERT(nval_tot1 == nval_tot2)
    1171       10156 :       DEALLOCATE (map_var_mol)
    1172       10156 :       DEALLOCATE (map_cvar_mol)
    1173       10156 :       DEALLOCATE (map_vars)
    1174       10156 :       DEALLOCATE (map_cvars)
    1175       10156 :       DEALLOCATE (bnd_type)
    1176       10156 :       DEALLOCATE (bnd_ctype)
    1177       10156 :       DEALLOCATE (c_var_a)
    1178       10156 :       DEALLOCATE (c_var_b)
    1179       10156 :       DEALLOCATE (c_var_c)
    1180       10156 :       DEALLOCATE (c_var_d)
    1181       10156 :       IF (found) THEN
    1182          14 :          DEALLOCATE (c_var_type)
    1183             :       END IF
    1184       10156 :       CALL timestop(handle2)
    1185             :       !-----------------------------------------------------------------------------
    1186             :       !-----------------------------------------------------------------------------
    1187             :       ! Finally deallocate some stuff.
    1188             :       !-----------------------------------------------------------------------------
    1189       10156 :       DEALLOCATE (first_list)
    1190       10156 :       DEALLOCATE (last_list)
    1191       10156 :       DEALLOCATE (map_atom_mol)
    1192       10156 :       DEALLOCATE (map_atom_type)
    1193       10156 :       CALL timestop(handle)
    1194             :       CALL cp_print_key_finished_output(iw, logger, subsys_section, &
    1195       10156 :                                         "PRINT%TOPOLOGY_INFO/UTIL_INFO")
    1196      213276 :    END SUBROUTINE topology_connectivity_pack
    1197             : 
    1198             : ! **************************************************************************************************
    1199             : !> \brief used to achieve linear scaling in the connectivity_pack
    1200             : !> \param nmol_type ...
    1201             : !> \param map_vars ...
    1202             : !> \param map_var_mol ...
    1203             : !> \param bnd_type ...
    1204             : !> \param nvar1 ...
    1205             : !> \par History
    1206             : !>      Teodoro Laino
    1207             : ! **************************************************************************************************
    1208      101560 :    SUBROUTINE find_bnd_typ(nmol_type, map_vars, map_var_mol, bnd_type, nvar1)
    1209             :       INTEGER, INTENT(IN)                                :: nmol_type
    1210             :       INTEGER, DIMENSION(:), POINTER                     :: map_vars, map_var_mol
    1211             :       INTEGER, DIMENSION(:, :), POINTER                  :: bnd_type
    1212             :       INTEGER, INTENT(IN)                                :: nvar1
    1213             : 
    1214             :       INTEGER                                            :: i, ibond, j
    1215             : 
    1216      211769 :       ALLOCATE (map_vars(nvar1))
    1217      101560 :       CALL sort(map_var_mol, nvar1, map_vars)
    1218      304680 :       ALLOCATE (bnd_type(2, nmol_type))
    1219     4151890 :       bnd_type = 0
    1220      101560 :       IF (nvar1 == 0) RETURN
    1221      731551 :       DO j = 1, nvar1
    1222      731551 :          IF (map_var_mol(j) /= -1) EXIT
    1223             :       END DO
    1224        8649 :       IF (j == nvar1 + 1) RETURN
    1225        8617 :       i = map_var_mol(j)
    1226        8617 :       bnd_type(1, i) = j
    1227      567991 :       DO ibond = j, nvar1
    1228      567991 :          IF (map_var_mol(ibond) /= i) THEN
    1229      100162 :             bnd_type(2, i) = ibond - 1
    1230      100162 :             i = map_var_mol(ibond)
    1231      100162 :             bnd_type(1, i) = ibond
    1232             :          END IF
    1233             :       END DO
    1234        8617 :       bnd_type(2, i) = nvar1
    1235             : 
    1236             :    END SUBROUTINE find_bnd_typ
    1237             : 
    1238             : ! **************************************************************************************************
    1239             : !> \brief   Handles the multiple unit cell option for the connectivity
    1240             : !> \param topology ...
    1241             : !> \param subsys_section ...
    1242             : !> \author  Teodoro Laino [tlaino] - 06.2009
    1243             : ! **************************************************************************************************
    1244       10156 :    SUBROUTINE topology_conn_multiple(topology, subsys_section)
    1245             :       TYPE(topology_parameters_type), INTENT(INOUT)      :: topology
    1246             :       TYPE(section_vals_type), POINTER                   :: subsys_section
    1247             : 
    1248             :       INTEGER                                            :: a, fac, i, ind, j, k, m, natoms_orig, &
    1249             :                                                             nbond, nbond_c, nimpr, nonfo, nphi, &
    1250             :                                                             ntheta, nub
    1251       10156 :       INTEGER, DIMENSION(:), POINTER                     :: multiple_unit_cell
    1252             :       TYPE(connectivity_info_type), POINTER              :: conn_info
    1253             : 
    1254       10156 :       NULLIFY (multiple_unit_cell)
    1255             :       CALL section_vals_val_get(subsys_section, "TOPOLOGY%MULTIPLE_UNIT_CELL", &
    1256       10156 :                                 i_vals=multiple_unit_cell)
    1257       40190 :       IF (ANY(multiple_unit_cell /= 1)) THEN
    1258         592 :          fac = PRODUCT(multiple_unit_cell)
    1259         148 :          conn_info => topology%conn_info
    1260             : 
    1261         148 :          nbond = 0
    1262         148 :          IF (ASSOCIATED(conn_info%bond_a)) THEN
    1263         148 :             nbond = SIZE(conn_info%bond_a)
    1264         148 :             CALL reallocate(conn_info%bond_a, 1, nbond*fac)
    1265         148 :             CALL reallocate(conn_info%bond_b, 1, nbond*fac)
    1266             :          END IF
    1267             : 
    1268         148 :          ntheta = 0
    1269         148 :          IF (ASSOCIATED(conn_info%theta_a)) THEN
    1270         148 :             ntheta = SIZE(conn_info%theta_a)
    1271         148 :             CALL reallocate(conn_info%theta_a, 1, ntheta*fac)
    1272         148 :             CALL reallocate(conn_info%theta_b, 1, ntheta*fac)
    1273         148 :             CALL reallocate(conn_info%theta_c, 1, ntheta*fac)
    1274             :          END IF
    1275             : 
    1276         148 :          nphi = 0
    1277         148 :          IF (ASSOCIATED(conn_info%phi_a)) THEN
    1278         148 :             nphi = SIZE(conn_info%phi_a)
    1279         148 :             CALL reallocate(conn_info%phi_a, 1, nphi*fac)
    1280         148 :             CALL reallocate(conn_info%phi_b, 1, nphi*fac)
    1281         148 :             CALL reallocate(conn_info%phi_c, 1, nphi*fac)
    1282         148 :             CALL reallocate(conn_info%phi_d, 1, nphi*fac)
    1283             :          END IF
    1284             : 
    1285         148 :          nimpr = 0
    1286         148 :          IF (ASSOCIATED(conn_info%impr_a)) THEN
    1287         148 :             nimpr = SIZE(conn_info%impr_a)
    1288         148 :             CALL reallocate(conn_info%impr_a, 1, nimpr*fac)
    1289         148 :             CALL reallocate(conn_info%impr_b, 1, nimpr*fac)
    1290         148 :             CALL reallocate(conn_info%impr_c, 1, nimpr*fac)
    1291         148 :             CALL reallocate(conn_info%impr_d, 1, nimpr*fac)
    1292             :          END IF
    1293             : 
    1294         148 :          nbond_c = 0
    1295         148 :          IF (ASSOCIATED(conn_info%c_bond_a)) THEN
    1296         116 :             nbond_c = SIZE(conn_info%c_bond_a)
    1297         116 :             CALL reallocate(conn_info%c_bond_a, 1, nbond_c*fac)
    1298         116 :             CALL reallocate(conn_info%c_bond_b, 1, nbond_c*fac)
    1299             :          END IF
    1300             : 
    1301         148 :          nub = 0
    1302         148 :          IF (ASSOCIATED(conn_info%ub_a)) THEN
    1303         148 :             nub = SIZE(conn_info%ub_a)
    1304         148 :             CALL reallocate(conn_info%ub_a, 1, nub*fac)
    1305         148 :             CALL reallocate(conn_info%ub_b, 1, nub*fac)
    1306         148 :             CALL reallocate(conn_info%ub_c, 1, nub*fac)
    1307             :          END IF
    1308             : 
    1309         148 :          nonfo = 0
    1310         148 :          IF (ASSOCIATED(conn_info%onfo_a)) THEN
    1311         148 :             nonfo = SIZE(conn_info%onfo_a)
    1312         148 :             CALL reallocate(conn_info%onfo_a, 1, nonfo*fac)
    1313         148 :             CALL reallocate(conn_info%onfo_b, 1, nonfo*fac)
    1314             :          END IF
    1315             : 
    1316         148 :          natoms_orig = topology%natoms/fac
    1317         148 :          ind = 0
    1318         556 :          DO k = 1, multiple_unit_cell(3)
    1319        1870 :             DO j = 1, multiple_unit_cell(2)
    1320        6698 :                DO i = 1, multiple_unit_cell(1)
    1321        4976 :                   ind = ind + 1
    1322        4976 :                   IF (ind == 1) CYCLE
    1323        4828 :                   a = (ind - 1)*natoms_orig
    1324             : 
    1325             :                   ! Bonds
    1326        4828 :                   IF (nbond > 0) THEN
    1327        3208 :                      m = (ind - 1)*nbond
    1328       34024 :                      conn_info%bond_a(m + 1:m + nbond) = conn_info%bond_a(1:nbond) + a
    1329       37232 :                      conn_info%bond_b(m + 1:m + nbond) = conn_info%bond_b(1:nbond) + a
    1330             :                   END IF
    1331             :                   ! Theta
    1332        4828 :                   IF (ntheta > 0) THEN
    1333        2168 :                      m = (ind - 1)*ntheta
    1334       37428 :                      conn_info%theta_a(m + 1:m + ntheta) = conn_info%theta_a(1:ntheta) + a
    1335       39596 :                      conn_info%theta_b(m + 1:m + ntheta) = conn_info%theta_b(1:ntheta) + a
    1336       39596 :                      conn_info%theta_c(m + 1:m + ntheta) = conn_info%theta_c(1:ntheta) + a
    1337             :                   END IF
    1338             :                   ! Phi
    1339        4828 :                   IF (nphi > 0) THEN
    1340          14 :                      m = (ind - 1)*nphi
    1341       35756 :                      conn_info%phi_a(m + 1:m + nphi) = conn_info%phi_a(1:nphi) + a
    1342       35770 :                      conn_info%phi_b(m + 1:m + nphi) = conn_info%phi_b(1:nphi) + a
    1343       35770 :                      conn_info%phi_c(m + 1:m + nphi) = conn_info%phi_c(1:nphi) + a
    1344       35770 :                      conn_info%phi_d(m + 1:m + nphi) = conn_info%phi_d(1:nphi) + a
    1345             :                   END IF
    1346             :                   ! Impropers
    1347        4828 :                   IF (nimpr > 0) THEN
    1348           0 :                      m = (ind - 1)*nimpr
    1349           0 :                      conn_info%impr_a(m + 1:m + nimpr) = conn_info%impr_a(1:nimpr) + a
    1350           0 :                      conn_info%impr_b(m + 1:m + nimpr) = conn_info%impr_b(1:nimpr) + a
    1351           0 :                      conn_info%impr_c(m + 1:m + nimpr) = conn_info%impr_c(1:nimpr) + a
    1352           0 :                      conn_info%impr_d(m + 1:m + nimpr) = conn_info%impr_d(1:nimpr) + a
    1353             :                   END IF
    1354             :                   ! Para_res
    1355        4828 :                   IF (nbond_c > 0) THEN
    1356           0 :                      m = (ind - 1)*nbond_c
    1357           0 :                      conn_info%c_bond_a(m + 1:m + nbond_c) = conn_info%c_bond_a(1:nbond_c) + a
    1358           0 :                      conn_info%c_bond_b(m + 1:m + nbond_c) = conn_info%c_bond_b(1:nbond_c) + a
    1359             :                   END IF
    1360             :                   ! Urey-Bradley
    1361        4828 :                   IF (nub > 0) THEN
    1362        2168 :                      m = (ind - 1)*nub
    1363       37428 :                      conn_info%ub_a(m + 1:m + nub) = conn_info%ub_a(1:nub) + a
    1364       39596 :                      conn_info%ub_b(m + 1:m + nub) = conn_info%ub_b(1:nub) + a
    1365       39596 :                      conn_info%ub_c(m + 1:m + nub) = conn_info%ub_c(1:nub) + a
    1366             :                   END IF
    1367             :                   ! ONFO
    1368        6142 :                   IF (nonfo > 0) THEN
    1369          14 :                      m = (ind - 1)*nonfo
    1370       28364 :                      conn_info%onfo_a(m + 1:m + nonfo) = conn_info%onfo_a(1:nonfo) + a
    1371       28378 :                      conn_info%onfo_b(m + 1:m + nonfo) = conn_info%onfo_b(1:nonfo) + a
    1372             :                   END IF
    1373             :                END DO
    1374             :             END DO
    1375             :          END DO
    1376             :       END IF
    1377             : 
    1378       10156 :    END SUBROUTINE topology_conn_multiple
    1379             : 
    1380             : END MODULE topology_connectivity_util

Generated by: LCOV version 1.15