30 use,
intrinsic :: iso_fortran_env
82 real(real64),
allocatable :: center(:)
83 real(real64) :: radius =
m_zero
84 class(box_t),
pointer :: box => null()
86 integer,
allocatable :: map(:)
87 integer :: num_regions
88 integer,
allocatable :: regions(:)
89 type(accel_mem_t) :: buff_map
90 real(real64),
allocatable :: rel_x(:,:)
91 real(real64),
allocatable :: r(:)
92 type(mesh_t),
pointer :: mesh => null()
95 integer :: np_global = -1
96 real(real64),
allocatable :: rel_x_global(:,:)
97 integer,
allocatable :: part_v(:)
98 integer,
allocatable :: global2local(:)
100 type(mesh_cube_map_t) :: cube_map
117 recursive real(real64) function f_n(dims)
result(fn)
122 else if (dims == 1)
then
131 subroutine submesh_init(this, space, mesh, latt, center, rc)
132 type(submesh_t),
intent(inout) :: this
133 class(space_t),
intent(in) :: space
134 class(mesh_t),
target,
intent(in) :: mesh
135 type(lattice_vectors_t),
intent(in) :: latt
136 real(real64),
intent(in) :: center(1:space%dim)
137 real(real64),
intent(in) :: rc
139 real(real64) :: r2, rc2, xx(space%dim), rc_norm_n
140 real(real64),
allocatable :: center_copies(:,:), xtmp(:, :), rtmp(:)
141 integer :: icell, is, ip, ix, iy, iz
142 integer(int64) :: max_elements_count
143 type(lattice_iterator_t) :: latt_iter
144 integer,
allocatable :: map_inv(:), map_temp(:)
145 integer :: nmax(3), nmin(3)
146 real(real64),
parameter :: tol = 1e-13_real64
152 assert(space%dim <= 3)
156 safe_allocate(this%center(1:space%dim))
157 this%center(:) = center(:)
165 if (.not. space%is_periodic())
then
168 safe_allocate(map_inv(0:this%mesh%np))
169 map_inv(0:this%mesh%np) = 0
175 nmin(1:space%dim) = int((center(1:space%dim) - abs(rc))/mesh%spacing(1:space%dim)) - 1
176 nmax(1:space%dim) = int((center(1:space%dim) + abs(rc))/mesh%spacing(1:space%dim)) + 1
180 nmin(1:space%dim) = max(mesh%idx%nr(1, 1:space%dim), nmin(1:space%dim))
181 nmax(1:space%dim) = min(mesh%idx%nr(2, 1:space%dim), nmax(1:space%dim))
185 do iz = nmin(3), nmax(3)
186 do iy = nmin(2), nmax(2)
187 do ix = nmin(1), nmax(1)
189 if (ip == 0 .or. ip > mesh%np) cycle
190 r2 = sum((mesh%x(:, ip) - center)**2)
202 safe_allocate(this%map(1:this%np))
203 safe_allocate(xtmp(1:space%dim, 1:this%np))
204 safe_allocate(rtmp(1:this%np))
207 do iz = nmin(3), nmax(3)
208 do iy = nmin(2), nmax(2)
209 do ix = nmin(1), nmax(1)
211 if (ip == 0 .or. ip > mesh%np) cycle
215 xtmp(:, is) = mesh%x(:, ip) - center
216 rtmp(is) = norm2(xtmp(:,is))
221 safe_deallocate_a(map_inv)
234 safe_allocate(center_copies(1:space%dim, 1:latt_iter%n_cells))
235 do icell = 1, latt_iter%n_cells
236 center_copies(:, icell) = center + latt_iter%get(icell)
241 rc_norm_n = product(ceiling(rc / mesh%spacing(1:space%dim), int64) +
m_one)
242 if (mesh%use_curvilinear) rc_norm_n = rc_norm_n / mesh%coord_system%min_mesh_scaling_product
243 max_elements_count = 3**space%dim * int(
m_pi**
floor(0.5 * space%dim) * rc_norm_n *
f_n(space%dim), int64)
246 safe_allocate(map_temp(1:max_elements_count))
247 safe_allocate(xtmp(1:space%dim, 1:max_elements_count))
248 safe_allocate(rtmp(1:max_elements_count))
252 do icell = 1, latt_iter%n_cells
253 xx = mesh%x(:,ip) - center_copies(:, icell)
254 if(any(abs(xx)>rc+tol)) cycle
264 assert(is < huge(is))
267 safe_allocate(this%map(1:this%np))
268 this%map(1:this%np) = map_temp(1:this%np)
271 safe_deallocate_a(map_temp)
272 safe_deallocate_a(center_copies)
279 safe_deallocate_a(xtmp)
280 safe_deallocate_a(rtmp)
287 class(
space_t),
intent(in) :: space
288 real(real64),
intent(in) :: xtmp(:, :), rtmp(:)
290 integer :: ip, i_region, offset
291 integer,
allocatable :: order(:), order_new(:)
292 integer,
allocatable :: map_new(:)
293 integer,
allocatable :: np_region(:), tmp_array(:)
300 safe_allocate(order(1:this%np))
301 safe_allocate(this%rel_x(1:space%dim, 1:this%np))
302 safe_allocate(this%r(1:this%np))
309 call sort(this%map, order)
312 this%overlap = .false.
313 do ip = 1, this%np - 1
314 if (this%map(ip) == this%map(ip + 1))
then
316 this%overlap = .
true.
324 if(this%overlap)
then
328 safe_allocate(tmp_array(1:this%np))
329 safe_allocate(order_new(1:this%np))
330 safe_allocate(np_region(1:this%np))
331 safe_allocate(map_new( 1:this%np))
338 if (this%map(ip) == this%map(ip - 1))
then
339 i_region = i_region + 1
340 if (i_region > this%num_regions)
then
341 this%num_regions = i_region
342 np_region(i_region) = 0
347 tmp_array(ip) = i_region
348 np_region(i_region) = np_region(i_region) + 1
351 assert( .not.
allocated(this%regions))
354 safe_allocate(this%regions(1:this%num_regions+1))
358 if(this%num_regions > 1)
then
359 do i_region = 1, this%num_regions
360 this%regions(i_region + 1) = this%regions(i_region) + np_region(i_region)
363 this%regions(2) = this%np + 1
366 np_region(1:this%np) = 0
367 order_new(1:this%np) = -1
368 map_new(1:this%np) = -1
373 i_region = tmp_array(ip)
374 np_region(i_region) = np_region(i_region) + 1
375 offset = this%regions(i_region) - 1
376 map_new( offset + np_region(i_region) ) = this%map(ip)
377 order_new( offset + np_region(i_region) ) = order(ip)
380 order(1:this%np) = order_new(1:this%np)
381 this%map(1:this%np) = map_new(1:this%np)
383 safe_deallocate_a(tmp_array)
384 safe_deallocate_a(order_new)
385 safe_deallocate_a(np_region)
386 safe_deallocate_a(map_new)
391 safe_allocate(this%regions(1:2))
393 this%regions(2) = this%np + 1
398 this%rel_x(:, ip) = xtmp(:, order(ip))
399 this%r(ip) = rtmp(order(ip))
402 safe_deallocate_a(order)
413 class(
space_t),
intent(in) :: space
414 class(
mesh_t),
target,
intent(in) :: mesh
417 real(real64),
optional,
intent(in) :: shift(:)
421 real(real64) :: xx(space%dim), diff_centers(space%dim)
428 safe_allocate(this%center(1:space%dim))
429 this%center(:) = sm1%center(:)
430 this%radius = sm1%radius
434 this%overlap = sm1%overlap .or. sm2%overlap
436 diff_centers = sm1%center - sm2%center
437 if (
present(shift)) diff_centers = diff_centers - shift
444 xx = sm2%rel_x(:, ip) - diff_centers
446 if (sum(xx**2) > sm1%radius**2) is = is + 1
451 safe_allocate(this%map(1:this%np))
452 safe_allocate(this%rel_x(1:space%dim, 1:this%np))
453 safe_allocate(this%r(1:this%np))
454 this%map(1:sm1%np) = sm1%map(1:sm1%np)
455 this%rel_x(:, 1:sm1%np) = sm1%rel_x(:, 1:sm1%np)
456 this%r(1:sm1%np) = sm1%r(1:sm1%np)
461 xx = sm2%rel_x(:, ip) - diff_centers
463 if (r2 > sm1%radius**2)
then
465 this%map(is) = sm2%map(ip)
466 this%r(is) =
sqrt(r2)
467 this%rel_x(:, is) = xx
479 class(
space_t),
intent(in) :: space
480 real(real64),
intent(in) :: newcenter(:)
482 real(real64) :: xx(space%dim), diff_centers(space%dim), oldcenter(space%dim)
488 oldcenter = this%center
489 this%center(:) = newcenter(:)
491 diff_centers = newcenter - oldcenter
494 xx = this%rel_x(:, ip) - diff_centers
495 this%r(ip) = norm2(xx)
496 this%rel_x(:, ip) = xx
507 type(
mesh_t),
target,
intent(in) :: mesh
508 real(real64),
intent(in) :: center(1:space%dim)
509 real(real64),
intent(in) :: radius
510 integer,
intent(in) :: root
513 integer :: nparray(1:3)
518 if (root /= mpi_grp%rank)
then
520 safe_allocate(this%center(1:space%dim))
521 this%center(:) = center(:)
525 if (mpi_grp%size > 1)
then
527 if (root == mpi_grp%rank)
then
529 nparray(2) = this%num_regions
530 if (this%overlap)
then
537 call mpi_grp%bcast(nparray, 3, mpi_integer, root)
539 this%num_regions = nparray(2)
540 this%overlap = (nparray(3) == 1)
542 if (root /= mpi_grp%rank)
then
543 safe_allocate(this%map(1:this%np))
544 safe_allocate(this%rel_x(1:space%dim, 1:this%np))
545 safe_allocate(this%r(1:this%np))
546 safe_allocate(this%regions(1:this%num_regions+1))
549 call mpi_grp%bcast(this%regions(1), this%num_regions+1, mpi_integer, root)
551 if (this%np > 0)
then
552 call mpi_grp%bcast(this%map(1), this%np, mpi_integer, root)
553 call mpi_grp%bcast(this%rel_x(1, 1), this%np*space%dim, mpi_double_precision, root)
554 call mpi_grp%bcast(this%r(1), this%np, mpi_double_precision, root)
566 real(real64),
intent(in) :: radius
567 real(real64),
intent(in) :: center(:)
568 real(real64),
intent(in) :: dx
571 if (
allocated(this%center))
then
574 if (radius <= this%radius+dx*1e-6_real64 .and. all(abs(this%center - center) < 0.25*dx))
then
587 if (this%np /= -1)
then
590 safe_deallocate_a(this%center)
591 safe_deallocate_a(this%map)
592 safe_deallocate_a(this%rel_x)
593 safe_deallocate_a(this%r)
594 safe_deallocate_a(this%regions)
609 class(
space_t),
intent(in) :: space
611 integer :: ii, jj, dd
616 if (.not. space%is_periodic())
then
618 distance = sum((sm1%center - sm2%center)**2)
619 overlap =
distance <= (1.5_real64*(sm1%radius + sm2%radius))**2
622 if (.not. overlap)
return
630 do while(ii <= sm1%np .and. jj <= sm2%np)
631 dd = sm1%map(ii) - sm2%map(jj)
634 else if (dd > 0)
then
642 if (sm1%mesh%parallel_in_domains)
then
643 call sm1%mesh%mpi_grp%allreduce_inplace(overlap, 1, mpi_logical, mpi_lor)
651 class(
space_t),
intent(in) :: space
653 integer,
allocatable :: part_np(:)
654 integer :: ipart, ind, ip
658 if (.not. this%mesh%parallel_in_domains)
then
659 this%np_global = this%np
664 safe_allocate(part_np(this%mesh%pv%npart))
666 part_np(this%mesh%pv%partno) = this%np
668 call this%mesh%allreduce(part_np)
669 this%np_global = sum(part_np)
671 safe_allocate(this%rel_x_global(1:space%dim, 1:this%np_global))
672 safe_allocate(this%part_v(1:this%np_global))
673 safe_allocate(this%global2local(1:this%np_global))
674 this%rel_x_global(1:space%dim, 1:this%np_global) =
m_zero
675 this%part_v(1:this%np_global) = 0
676 this%global2local(1:this%np_global) = 0
679 do ipart = 1, this%mesh%pv%npart
680 if (ipart == this%mesh%pv%partno)
then
682 this%rel_x_global(:, ind + ip) = this%rel_x(:, ip)
683 this%part_v(ind + ip) = this%mesh%pv%partno
684 this%global2local(ind + ip) = ip
687 ind = ind + part_np(ipart)
690 call this%mesh%allreduce(this%rel_x_global)
691 call this%mesh%allreduce(this%part_v)
692 call this%mesh%allreduce(this%global2local)
694 safe_deallocate_a(part_np)
705 safe_deallocate_a(this%rel_x_global)
707 safe_deallocate_a(this%part_v)
708 safe_deallocate_a(this%global2local)
717 complex(real64),
intent(in) :: sphi(:)
718 complex(real64),
intent(inout) :: phi(:)
719 complex(real64),
optional,
intent(in) :: factor
723 push_sub(zzdsubmesh_add_to_mesh)
725 if (
present(factor))
then
729 phi(this%map(ip)) = phi(this%map(ip)) + factor*sphi(ip)
731 if (this%np.ge.4)
then
732 do ip = m+1, this%np, 4
733 phi(this%map(ip)) = phi(this%map(ip)) + factor*sphi(ip)
734 phi(this%map(ip+1)) = phi(this%map(ip+1)) + factor*sphi(ip+1)
735 phi(this%map(ip+2)) = phi(this%map(ip+2)) + factor*sphi(ip+2)
736 phi(this%map(ip+3)) = phi(this%map(ip+3)) + factor*sphi(ip+3)
742 phi(this%map(ip)) = phi(this%map(ip)) + sphi(ip)
744 if (this%np.ge.4)
then
745 do ip = m+1, this%np, 4
746 phi(this%map(ip)) = phi(this%map(ip)) + sphi(ip)
747 phi(this%map(ip+1)) = phi(this%map(ip+1)) + sphi(ip+1)
748 phi(this%map(ip+2)) = phi(this%map(ip+2)) + sphi(ip+2)
749 phi(this%map(ip+3)) = phi(this%map(ip+3)) + sphi(ip+3)
754 pop_sub(zzdsubmesh_add_to_mesh)
760 complex(real64),
intent(in) :: sphi(:)
761 complex(real64),
intent(in) :: phi(:)
762 logical,
optional,
intent(in) :: reduce
770 if (this%mesh%use_curvilinear)
then
772 dotp = dotp + this%mesh%vol_pp(this%map(is))*phi(this%map(is))*conjg(sphi(is))
777 dotp = dotp + phi(this%map(ip))*conjg(sphi(ip))
779 if (this%np.ge.4)
then
780 do ip = m+1, this%np, 4
781 dotp = dotp + phi(this%map(ip))*conjg(sphi(ip)) &
782 + phi(this%map(ip+1))*conjg(sphi(ip+1)) &
783 + phi(this%map(ip+2))*conjg(sphi(ip+2)) &
784 + phi(this%map(ip+3))*conjg(sphi(ip+3))
787 dotp = dotp*this%mesh%vol_pp(1)
792 call this%mesh%allreduce(dotp)
802 type(
submesh_t),
target,
intent(in) :: sm
803 class(
space_t),
intent(in) :: space
804 integer,
intent(out) :: db(1:space%dim)
807 real(real64) :: chi(space%dim), max_chi(space%dim)
808 integer :: db_red(1:space%dim)
809 real(real64),
parameter :: tol=1.0e-10_real64
813 max_chi = abs(sm%mesh%coord_system%from_cartesian(sm%rel_x(:, 1)))/sm%mesh%spacing
815 chi = sm%mesh%coord_system%from_cartesian(sm%rel_x(:, ip))
816 do idir = 1, space%dim
817 max_chi(idir) = max(max_chi(idir), abs(chi(idir))/sm%mesh%spacing(idir))
821 do idir = 1, space%dim
822 db(idir) = nint(max_chi(idir)-tol)
825 if(sm%mesh%parallel_in_domains)
then
826 call sm%mesh%mpi_grp%allreduce(db(1), db_red(1), space%dim, mpi_integer, mpi_max)
827 db(1:space%dim) = db_red(1:space%dim)
837 type(
submesh_t),
target,
intent(inout) :: sm
838 class(
space_t),
intent(in) :: space
842 real(real64) :: chi(space%dim), shift(space%dim)
846 sm%cube_map%nmap = sm%np
848 safe_allocate(sm%cube_map%map(1:space%dim, 1:sm%cube_map%nmap))
852 chi = sm%mesh%coord_system%from_cartesian(sm%center)
853 do idir = 1, space%dim
854 shift(idir) = nint(chi(idir)/sm%mesh%spacing(idir))*sm%mesh%spacing(idir)
856 shift = sm%mesh%coord_system%to_cartesian(shift)
857 shift = shift - sm%center
859 do ip = 1, sm%cube_map%nmap
860 chi = sm%mesh%coord_system%from_cartesian(sm%rel_x(:,ip) - shift)
861 do idir = 1, space%dim
862 sm%cube_map%map(idir, ip) = nint(chi(idir)/sm%mesh%spacing(idir))
868 call accel_write_buffer(sm%cube_map%map_buffer, space%dim, sm%cube_map%nmap, sm%cube_map%map)
893 class(
batch_t),
intent(in) :: ss
894 class(
batch_t),
intent(inout) :: mm
896 integer :: ist, idim, jdim, is
900 assert(.not. mm%is_packed())
903 assert(ss%nst_linear == mm%nst_linear)
904 assert(ss%status() == mm%status())
905 assert(ss%dim == mm%dim)
907 assert(mm%nst == ss%nst)
912 jdim = min(idim, ss%dim)
915 mm%zff(this%map(is), idim, ist) = &
916 mm%zff(this%map(is), idim, ist) + ss%dff(is, jdim, ist)
929#include "submesh_inc.F90"
932#include "complex.F90"
933#include "submesh_inc.F90"
This is the common interface to a sorting routine. It performs the shell algorithm,...
double sqrt(double __x) __attribute__((__nothrow__
double floor(double __x) __attribute__((__nothrow__
double fn(const gsl_vector *v, void *params)
subroutine, public accel_free_buffer(this, async)
pure logical function, public accel_is_enabled()
integer, parameter, public accel_mem_read_only
This module implements batches of mesh functions.
Module implementing boundary conditions in Octopus.
real(real64), parameter, public m_two
real(real64), parameter, public m_zero
real(real64), parameter, public m_pi
some mathematical constants
complex(real64), parameter, public m_z0
real(real64), parameter, public m_one
This module implements the index, used for the mesh points.
subroutine, public mesh_cube_map_end(this)
This module defines the meshes, which are used in Octopus.
integer function, public mesh_local_index_from_coords(mesh, ix)
This function returns the local index of the point for a given vector of integer coordinates.
Some general things and nomenclature:
subroutine, public profiling_out(label)
Increment out counter and sum up difference between entry and exit time.
subroutine, public profiling_in(label, exclude)
Increment in counter and save entry time.
This module is intended to contain "only mathematical" functions and procedures.
real(real64) function, public dsm_integrate_frommesh(mesh, sm, ff, reduce)
subroutine zdsubmesh_add_to_mesh(this, sphi, phi, factor)
subroutine, public submesh_init_cube_map(sm, space)
logical function, public submesh_overlap(sm1, sm2, space)
subroutine, public zsubmesh_batch_dotp_matrix(this, mm, ss, dot, reduce)
real(real64) function, public dsm_nrm2(sm, ff, reduce)
this function returns the the norm of a vector
subroutine zzsubmesh_add_to_mesh(this, sphi, phi, factor)
subroutine, public submesh_end_global(this)
subroutine, public zsubmesh_copy_from_mesh(this, phi, sphi, conjugate)
subroutine submesh_reorder_points(this, space, xtmp, rtmp)
complex(real64) function, public zsm_integrate(mesh, sm, ff, reduce)
subroutine, public submesh_shift_center(this, space, newcenter)
complex(real64) function zzsubmesh_to_mesh_dotp(this, sphi, phi, reduce)
real(real64) function, public dsm_integrate(mesh, sm, ff, reduce)
subroutine, public dsubmesh_batch_add_matrix(this, factor, ss, mm)
The following functions takes a batch of functions defined in submesh (ss) and adds all of them to ea...
recursive real(real64) function f_n(dims)
subroutine, public dsubmesh_copy_from_mesh(this, phi, sphi, conjugate)
real(real64) function ddsubmesh_to_mesh_dotp(this, sphi, phi, reduce)
subroutine, public submesh_merge(this, space, mesh, sm1, sm2, shift)
subroutine, public dzsubmesh_batch_add(this, ss, mm)
The following function takes a batch of functions defined in submesh (ss) and adds one of them to eac...
subroutine, public zsubmesh_batch_add(this, ss, mm)
The following function takes a batch of functions defined in submesh (ss) and adds one of them to eac...
real(real64) function, public zsm_nrm2(sm, ff, reduce)
this function returns the the norm of a vector
subroutine, public zsubmesh_batch_add_matrix(this, factor, ss, mm)
The following functions takes a batch of functions defined in submesh (ss) and adds all of them to ea...
subroutine, public submesh_end_cube_map(sm)
subroutine, public submesh_end(this)
subroutine ddsubmesh_add_to_mesh(this, sphi, phi, factor)
subroutine, public submesh_get_cube_dim(sm, space, db)
finds the dimension of a box containing the submesh
subroutine, public dsubmesh_batch_dotp_matrix(this, mm, ss, dot, reduce)
complex(real64) function, public zsm_integrate_frommesh(mesh, sm, ff, reduce)
logical function, public submesh_compatible(this, radius, center, dx)
subroutine, public submesh_build_global(this, space)
subroutine, public submesh_broadcast(this, space, mesh, center, radius, root, mpi_grp)
subroutine, public dsubmesh_batch_add(this, ss, mm)
The following function takes a batch of functions defined in submesh (ss) and adds one of them to eac...
subroutine, public submesh_init(this, space, mesh, latt, center, rc)
complex(real64) function zdsubmesh_to_mesh_dotp(this, sphi, phi, reduce)
type(type_t), public type_float
type(type_t), public type_cmplx
type(type_t), public type_integer
Class defining batches of mesh functions.
The following class implements a lattice iterator. It allows one to loop over all cells that are with...
Describes mesh distribution to nodes.
This is defined even when running serial.
A submesh is a type of mesh, used for the projectors in the pseudopotentials It contains points on a ...
void distance(const int iatom, const int jatom, const double coordinates[], double *rr, double *rr2, double *rr6, double *rr7)