cell_list_m.f90 Source File


This file depends on

sourcefile~~cell_list_m.f90~~EfferentGraph sourcefile~cell_list_m.f90 cell_list_m.f90 sourcefile~simbox_m.f90 simbox_m.f90 sourcefile~cell_list_m.f90->sourcefile~simbox_m.f90 sourcefile~vector_m.f90 vector_m.f90 sourcefile~cell_list_m.f90->sourcefile~vector_m.f90 sourcefile~constants_m.f90 constants_m.f90 sourcefile~cell_list_m.f90->sourcefile~constants_m.f90 sourcefile~simbox_m.f90->sourcefile~constants_m.f90 sourcefile~random_m.f90 random_m.f90 sourcefile~simbox_m.f90->sourcefile~random_m.f90 sourcefile~vector_m.f90->sourcefile~constants_m.f90 sourcefile~qsort_m.f90 qsort_m.f90 sourcefile~vector_m.f90->sourcefile~qsort_m.f90 sourcefile~random_m.f90->sourcefile~constants_m.f90 sourcefile~qsort_m.f90->sourcefile~constants_m.f90

Files dependent on this one

sourcefile~~cell_list_m.f90~~AfferentGraph sourcefile~cell_list_m.f90 cell_list_m.f90 sourcefile~pairtab_m.fpp pairtab_m.fpp sourcefile~pairtab_m.fpp->sourcefile~cell_list_m.f90 sourcefile~interaction_m.f90 interaction_m.f90 sourcefile~interaction_m.f90->sourcefile~pairtab_m.fpp sourcefile~setup_m.f90 setup_m.f90 sourcefile~setup_m.f90->sourcefile~interaction_m.f90 sourcefile~bd_solver_m.f90 bd_solver_m.f90 sourcefile~setup_m.f90->sourcefile~bd_solver_m.f90 sourcefile~bd_solver_m.f90->sourcefile~interaction_m.f90 sourcefile~main.f90 main.f90 sourcefile~main.f90->sourcefile~setup_m.f90

Contents

Source Code


Source Code

module cell_list_m
    !! Sorts atoms using a cell list.
    !!
    !! The algorithm to build the cell list partially follows the techniques in Watanabe et
    !! al, 2011, “Efficient Implementations of Molecular Dynamics Simulations for
    !! Lennard-Jones Systems,” Prog. Theor. Phys. 126, 203–235.
    !!
    !! The pairlist is not explicitly built, rather the cells are directly
    !! looped over during force calculation.

use constants_m
use vector_m
use simbox_m

implicit none

private

public :: cl_init, cl_delete, cl_print, &
          cl_set_cell_size, cl_build, cl_build_cell_nbrs, &
          cl_get_num_cells, cl_get_contents, cl_get_nbr_cells 

real(rp), dimension(3)   :: cell_size = 0.0_rp
    !! Cell size along *x*, *y*, & *z*.
integer,  dimension(3)   :: nc_max = 0
    !! Maximum number of cells along *x*, *y*, & *z*.
integer,  dimension(3)   :: nc = 0
    !! Number of cells along *x*, *y*, & *z*.
integer  :: nct_max = 0
    !! Maximum total number of cells.
integer  :: nct = 0
    !! Total of cells.
integer, dimension(:), allocatable, target :: cells
    !! *(na_max,)* array. Listing atoms in each cell.
integer, dimension(:), allocatable :: cells_pos
    !! *(0:nct_max,)* index array. Note: 0-based indexing.
type(ivector_t) :: cell_nbrs
    !! Lists neighbor cells for each cell.
integer, dimension(:), allocatable :: cell_nbrs_pos
    !! *(0:nct_max,)* index array. Note: 0-based indexing.
integer, dimension(:), allocatable :: host_cells
    !! *(na_max,)* array. *host_cells(i)* stores the linear index of the cell
    !! containing atom *i*. *na_max* is the total number of atoms under consideration.
integer, dimension(:), allocatable :: cell_pop
    !! *(0:nct_max-1,)* array storing population of each cell. Note: 0-based indexing.
integer, dimension(3,13), parameter :: d = reshape( [           &
                &                            1, 0, 0,           &
                &    1, 1, 0,   -1, 1, 0,    0, 1, 0,           &
                &    0, 0, 1,   -1, 0, 1,    1, 0, 1,           &
                &   -1,-1, 1,    0,-1, 1,    1,-1, 1,           &
                &   -1, 1, 1,    0, 1, 1,    1, 1, 1 ], [ 3, 13] )

contains

!******************************************************************************

subroutine cl_init(na_max, cs_min, simbox)
    !! Initializes a cell list.

    integer, intent(in) :: na_max
        !! Maximum number of atoms to be handled.
    real(rp), intent(in) :: cs_min
        !! Minimum size (i.e. length) of a cell.
    type(smbx_t), intent(in) :: simbox
        !! Simulation box
    integer :: nelem
    
    !The following defines an upper bound on the number of cells
    nc_max(1) = ceiling( simbox%basis(1,1)/cs_min )
    nc_max(2) = ceiling( simbox%basis(2,2)/cs_min )
    nc_max(3) = ceiling( simbox%basis(3,3)/cs_min )

    nct_max = product(nc_max)

    allocate(cells(na_max))
    allocate(cells_pos(0:nct_max))

    allocate(host_cells(na_max))
    allocate(cell_pop(0:nct_max-1))

    !Provisional allocation with nelem (arbitrary) elements,
    !will expand/contract later.
    nelem = 8
    call ivector_init(cell_nbrs, nelem)
    allocate(cell_nbrs_pos(0:nelem))

    end subroutine

!******************************************************************************

subroutine cl_set_cell_size(cs, simbox)
    !! Sets the cell size. The actual cell size may be slightly larger.

    real(rp), intent(in) :: cs
    type(smbx_t), intent(in) :: simbox

    nc(1) = floor( simbox%basis(1,1)/cs )
    nc(2) = floor( simbox%basis(2,2)/cs )
    nc(3) = floor( simbox%basis(3,3)/cs )

    nct = product(nc)

    cell_size(1) = simbox%basis(1,1)/nc(1)
    cell_size(2) = simbox%basis(2,2)/nc(2)
    cell_size(3) = simbox%basis(3,3)/nc(3)

    end subroutine

!******************************************************************************

subroutine cl_build_cell_nbrs()
    !! Makes a table of neighboring cells.

    integer :: j
    integer :: ncx, ncy, ncz
    integer :: icx, icy, icz, ic
    integer :: jcx, jcy, jcz, jc

    if ( size(cell_nbrs_pos) < (nct+1) ) then
        deallocate(cell_nbrs_pos)
        allocate(cell_nbrs_pos(0:nct))
    end if

    ncx = nc(1); ncy = nc(2); ncz = nc(3)

    cell_nbrs_pos(0) = 1

    do icz = 0, ncz-1
        do icy = 0, ncy-1
            do icx = 0, ncx-1
                ic = icz*ncx*ncy + icy*ncx + icx
                cell_nbrs_pos(ic+1) = cell_nbrs_pos(ic)
                do j = 1, 13
                    jcx = modulo( icx + d(1,j), ncx )
                    jcy = modulo( icy + d(2,j), ncy )
                    jcz = modulo( icz + d(3,j), ncz )
                    jc = jcz*ncx*ncy + jcy*ncx + jcx
                    call cell_nbrs%append(jc)
                    cell_nbrs_pos(ic+1) = cell_nbrs_pos(ic+1) + 1
                end do
            end do
        end do
    end do

    call cell_nbrs%shrink_to_fit()

    end subroutine

!******************************************************************************

subroutine cl_delete()
    !! Deallocates memory allocated in `cl_init`.

    if (allocated(cells))         deallocate(cells)
    if (allocated(cells_pos))     deallocate(cells_pos)

    call cell_nbrs%delete()
    if (allocated(cell_nbrs_pos)) deallocate(cell_nbrs_pos)

    if (allocated(host_cells))    deallocate(host_cells)
    if (allocated(cell_pop))      deallocate(cell_pop)

    cell_size = 0.0_rp
    nc_max = 0; nc = 0
    nct_max = 0; nct = 0

    end subroutine

!******************************************************************************

subroutine cl_build(coords)
    !! Sorts atoms into cells for calculating short-range interations

    real(rp), dimension(:,:), intent(in) :: coords
    real(rp), dimension(3) :: ri
    integer :: na, iatm, j
    integer :: ncx, ncy, ncz
    integer :: icx, icy, icz, ic

    cells = 0
    host_cells = 0
    cell_pop = 0
    ncx = nc(1); ncy = nc(2); ncz = nc(3)
    na = size(coords,2)

    !Loop over particles and put into cells
    do iatm = 1, na
        ri = coords(:,iatm)
        icx = int( ri(1)/cell_size(1) )
        icy = int( ri(2)/cell_size(2) )
        icz = int( ri(3)/cell_size(3) )

        if ((icx<0) .or. (icy<0) .or. (icz<0)) then
            print*, ri
            stop
        end if

        !If atoms are exactly on the box edge
        if ( icx > (ncx-1) ) icx = ncx - 1
        if ( icy > (ncy-1) ) icy = ncy - 1
        if ( icz > (ncz-1) ) icz = ncz - 1

        ic = icz*ncx*ncy + icy*ncx + icx
        host_cells(iatm) = ic
        cell_pop(ic) = cell_pop(ic) + 1
    end do

    !Loop over all cells to set up pointers
    cells_pos(0) = 1
    do ic = 0, (nct-1)
        cells_pos(ic+1) = cells_pos(ic) + cell_pop(ic)
    end do

    !Loop over all atoms
    do iatm = 1, na
        ic = host_cells(iatm)
        j = cells_pos(ic)
        cells(j) = iatm
        cells_pos(ic) = cells_pos(ic) + 1
    end do

    !Loop over all cells to set up pointers
    cells_pos(0) = 1
    do ic = 0, (nct-1)
        cells_pos(ic+1) = cells_pos(ic) + cell_pop(ic)
    end do

    end subroutine

!******************************************************************************

function cl_get_num_cells() result (res)
    !! Returns the total number of cells

    integer :: res

    res = nct

    end function

!******************************************************************************

subroutine cl_get_contents(ic, res)
    !! Returns a pointer to the entries of cell with linear index *ic*.

    integer, intent(in) :: ic
    integer, dimension(:), pointer, intent(out) :: res
    integer :: ibeg, iend

    res => null()
    ibeg = cells_pos(ic); iend = cells_pos(ic+1) - 1
    res => cells(ibeg:iend)

    end subroutine

!******************************************************************************

subroutine cl_get_nbr_cells(ic, res)
    !! Returns a pointer to the neighbor cells of cell with linear index *ic*.

    integer, intent(in) :: ic
    integer, dimension(:), pointer, intent(out) :: res
    integer :: ibeg, iend

    res => null()
    ibeg = cell_nbrs_pos(ic); iend = cell_nbrs_pos(ic+1) - 1
    call cell_nbrs%get_data(res, ibeg, iend)

    end subroutine

!********************************************************************************

subroutine cl_print()
    !! Prints a cell list

    integer, dimension(:), pointer :: aic => null()
    integer, dimension(:), pointer :: nbrc => null()
    integer :: ncx, ncy, ncz
    integer :: icx, icy, icz, ic

    ncx = nc(1); ncy = nc(2); ncz = nc(3)

    write(*,'("ncx: ", i0, " ncy: ", i0, " ncz: ", i0)') ncx, ncy, ncz
    write(*,'("lcx: ", g0.6, " lcy: ", g0.6, " lcz: ", g0.6)') cell_size
    write(*, *) 'CELL CONTENTS'
    do icz = 0, ncz-1
        do icy = 0, ncy-1
            do icx = 0, ncx-1
                ic = icz*ncx*ncy + icy*ncx + icx
                call cl_get_contents(ic, aic)
                if ( size(aic) > 0 ) then
                    write(*,'("(",i0,",",i0,",",i0,")[",i0,"] ")', advance='no') icx, icy, icz, ic
                    write(*, *) size(aic), aic
                end if
            end do
        end do
    end do

    write(*, *)
    write(*, *) 'NBR CELLS'
    do icz = 0, ncz-1
        do icy = 0, ncy-1
            do icx = 0, ncx-1
                ic = icz*ncx*ncy + icy*ncx + icx
                call cl_get_nbr_cells(ic, nbrc)
                write(*,'("(",i0,",",i0,",",i0,")[",i0,"] ")', advance='no') icx, icy, icz, ic
                write(*, *) size(nbrc), nbrc
            end do
        end do
    end do

    end subroutine

!********************************************************************************

end module cell_list_m