improve inside and move it to utils.f90
This slightly improves the performance of inside. For a ~100 points contour the instructions cost is reduced by ~5%.
This commit is contained in:
parent
686f63b01a
commit
86ff5ecb06
@ -256,8 +256,7 @@ contains
|
|||||||
! ellipses in the transverse plane at the launch point (deg)
|
! ellipses in the transverse plane at the launch point (deg)
|
||||||
|
|
||||||
use gray_params, only : antenna_parameters
|
use gray_params, only : antenna_parameters
|
||||||
use utils, only : get_free_unit, intlin, locate
|
use utils, only : get_free_unit, intlin, locate, inside
|
||||||
use reflections, only : inside
|
|
||||||
use dierckx, only : curfit, splev, surfit, bispev
|
use dierckx, only : curfit, splev, surfit, bispev
|
||||||
use logger, only : log_error
|
use logger, only : log_error
|
||||||
|
|
||||||
@ -557,7 +556,7 @@ contains
|
|||||||
!
|
!
|
||||||
! check if (xcoord0, ycoord0) is out of table range
|
! check if (xcoord0, ycoord0) is out of table range
|
||||||
! incheck = 1 inside / 0 outside
|
! incheck = 1 inside / 0 outside
|
||||||
if(inside(xpolyg,ypolyg,npolyg,xcoord0,ycoord0)) incheck = 1
|
if(inside(xpolyg, ypolyg, xcoord0, ycoord0)) incheck = 1
|
||||||
end if
|
end if
|
||||||
deallocate(wrk,iwrk)
|
deallocate(wrk,iwrk)
|
||||||
!#######################################################################################
|
!#######################################################################################
|
||||||
@ -666,17 +665,17 @@ contains
|
|||||||
! v1 A v2
|
! v1 A v2
|
||||||
! (8) | (1) | (2)
|
! (8) | (1) | (2)
|
||||||
!
|
!
|
||||||
if(inside(xoutA,youtA,nxcoord+3,xcoord0,ycoord0)) then
|
if(inside(xoutA, youtA, xcoord0, ycoord0)) then
|
||||||
in = 1
|
in = 1
|
||||||
if(xcoord0.GT.xvert(2)) then
|
if(xcoord0.GT.xvert(2)) then
|
||||||
in = 2
|
in = 2
|
||||||
end if
|
end if
|
||||||
else if(inside(xoutB,youtB,nycoord+3,xcoord0,ycoord0)) then
|
else if(inside(xoutB, youtB, xcoord0, ycoord0)) then
|
||||||
in = 3
|
in = 3
|
||||||
if(ycoord0.GT.yvert(3)) then
|
if(ycoord0.GT.yvert(3)) then
|
||||||
in = 4
|
in = 4
|
||||||
end if
|
end if
|
||||||
else if(inside(xoutC,youtC,nxcoord+3,xcoord0,ycoord0)) then
|
else if(inside(xoutC, youtC, xcoord0, ycoord0)) then
|
||||||
in = 5
|
in = 5
|
||||||
if(xcoord0.LT.xvert(4)) then
|
if(xcoord0.LT.xvert(4)) then
|
||||||
in = 6
|
in = 6
|
||||||
|
@ -408,8 +408,7 @@ contains
|
|||||||
use const_and_precisions, only : zero, one
|
use const_and_precisions, only : zero, one
|
||||||
use gray_params, only : equilibrium_parameters, equilibrium_data
|
use gray_params, only : equilibrium_parameters, equilibrium_data
|
||||||
use gray_params, only : iequil
|
use gray_params, only : iequil
|
||||||
use reflections, only : inside
|
use utils, only : vmaxmin, vmaxmini, inside
|
||||||
use utils, only : vmaxmin, vmaxmini
|
|
||||||
use logger, only : log_info
|
use logger, only : log_info
|
||||||
|
|
||||||
implicit none
|
implicit none
|
||||||
@ -463,7 +462,7 @@ contains
|
|||||||
do j=1,nz
|
do j=1,nz
|
||||||
do i=1,nr
|
do i=1,nr
|
||||||
if (nbnd.gt.0) then
|
if (nbnd.gt.0) then
|
||||||
if(.not.inside(data%rbnd,data%zbnd,nbnd,data%rv(i),data%zv(j))) cycle
|
if(.not.inside(data%rbnd,data%zbnd,data%rv(i),data%zv(j))) cycle
|
||||||
else
|
else
|
||||||
if(data%psin(i,j).le.0.0d0) cycle
|
if(data%psin(i,j).le.0.0d0) cycle
|
||||||
end if
|
end if
|
||||||
@ -477,7 +476,7 @@ contains
|
|||||||
do j=1,nz
|
do j=1,nz
|
||||||
do i=1,nr
|
do i=1,nr
|
||||||
if (nbnd.gt.0) then
|
if (nbnd.gt.0) then
|
||||||
if(.not.inside(data%rbnd,data%zbnd,nbnd,data%rv(i),data%zv(j))) cycle
|
if(.not.inside(data%rbnd,data%zbnd,data%rv(i),data%zv(j))) cycle
|
||||||
else
|
else
|
||||||
if(data%psin(i,j).le.0.0d0) cycle
|
if(data%psin(i,j).le.0.0d0) cycle
|
||||||
end if
|
end if
|
||||||
@ -910,7 +909,7 @@ contains
|
|||||||
!
|
!
|
||||||
! Note: all output arguments are optional.
|
! Note: all output arguments are optional.
|
||||||
use gray_params, only : iequil
|
use gray_params, only : iequil
|
||||||
use reflections, only : inside
|
use utils, only : inside
|
||||||
use const_and_precisions, only : pi
|
use const_and_precisions, only : pi
|
||||||
|
|
||||||
implicit none
|
implicit none
|
||||||
@ -1037,7 +1036,7 @@ contains
|
|||||||
if (present(ddpsidrz)) ddpsidrz = ddpsidphidr * dphidz + dpsidphi * ddphidrdz
|
if (present(ddpsidrz)) ddpsidrz = ddpsidphidr * dphidz + dpsidphi * ddphidrdz
|
||||||
else
|
else
|
||||||
! Numerical data
|
! Numerical data
|
||||||
if (inside(psi_domain%R, psi_domain%z, psi_domain%npoints, R, z)) then
|
if (inside(psi_domain%R, psi_domain%z, R, z)) then
|
||||||
! Within the interpolation range
|
! Within the interpolation range
|
||||||
if (present(psi_n)) psi_n = psi_spline%eval(R, z)
|
if (present(psi_n)) psi_n = psi_spline%eval(R, z)
|
||||||
if (present(dpsidr)) dpsidr = psi_spline%deriv(R, z, 1, 0)
|
if (present(dpsidr)) dpsidr = psi_spline%deriv(R, z, 1, 0)
|
||||||
|
@ -20,8 +20,7 @@ contains
|
|||||||
use beamdata, only : init_btr, dealloc_beam
|
use beamdata, only : init_btr, dealloc_beam
|
||||||
use pec, only : pec_init, spec, postproc_profiles, dealloc_pec, &
|
use pec, only : pec_init, spec, postproc_profiles, dealloc_pec, &
|
||||||
rhop_tab, rhot_tab
|
rhop_tab, rhot_tab
|
||||||
use utils, only : vmaxmin
|
use utils, only : vmaxmin, inside
|
||||||
use reflections, only : inside
|
|
||||||
use multipass, only : alloc_multipass, dealloc_multipass, initbeam, &
|
use multipass, only : alloc_multipass, dealloc_multipass, initbeam, &
|
||||||
initmultipass, turnoffray, plasma_in, plasma_out, &
|
initmultipass, turnoffray, plasma_in, plasma_out, &
|
||||||
wall_out
|
wall_out
|
||||||
@ -232,7 +231,7 @@ contains
|
|||||||
rrm = sqrt(yw(1,jk)*yw(1,jk)+yw(2,jk)*yw(2,jk))*0.01_wp_
|
rrm = sqrt(yw(1,jk)*yw(1,jk)+yw(2,jk)*yw(2,jk))*0.01_wp_
|
||||||
|
|
||||||
if(inside(data%equilibrium%rlim, data%equilibrium%zlim, &
|
if(inside(data%equilibrium%rlim, data%equilibrium%zlim, &
|
||||||
nlim, rrm, zzm)) then ! * start propagation in/outside vessel?
|
rrm, zzm)) then ! * start propagation in/outside vessel?
|
||||||
iow(jk) = 1 ! + inside
|
iow(jk) = 1 ! + inside
|
||||||
else
|
else
|
||||||
iow(jk) = 0 ! + outside
|
iow(jk) = 0 ! + outside
|
||||||
@ -295,7 +294,7 @@ contains
|
|||||||
|
|
||||||
ins_pl = (psinv>=zero .and. psinv<params%profiles%psnbnd) ! in/out plasma?
|
ins_pl = (psinv>=zero .and. psinv<params%profiles%psnbnd) ! in/out plasma?
|
||||||
ins_wl = inside(data%equilibrium%rlim, data%equilibrium%zlim, &
|
ins_wl = inside(data%equilibrium%rlim, data%equilibrium%zlim, &
|
||||||
nlim,rrm,zzm) ! in/out vessel?
|
rrm, zzm) ! in/out vessel?
|
||||||
ent_pl = (mod(iop(jk),2) == 0 .and. ins_pl) ! enter plasma
|
ent_pl = (mod(iop(jk),2) == 0 .and. ins_pl) ! enter plasma
|
||||||
ext_pl = (mod(iop(jk),2) == 1 .and. .not.ins_pl) ! exit plasma
|
ext_pl = (mod(iop(jk),2) == 1 .and. .not.ins_pl) ! exit plasma
|
||||||
ent_wl = (mod(iow(jk),2) == 0 .and. ins_wl) ! enter vessel
|
ent_wl = (mod(iow(jk),2) == 0 .and. ins_wl) ! enter vessel
|
||||||
|
@ -62,7 +62,7 @@ subroutine gray_jetto1beam(ijetto, mr, mz, r, z, psin, psia, rax, zax, &
|
|||||||
|
|
||||||
! Set a simple limiter following the boundary of the data grid
|
! Set a simple limiter following the boundary of the data grid
|
||||||
simple_limiter: block
|
simple_limiter: block
|
||||||
use reflections, only : range2rect
|
use utils, only : range2rect
|
||||||
use limiter, only : limiter_set_globals=>set_globals
|
use limiter, only : limiter_set_globals=>set_globals
|
||||||
use const_and_precisions, only : cm
|
use const_and_precisions, only : cm
|
||||||
|
|
||||||
@ -73,7 +73,7 @@ subroutine gray_jetto1beam(ijetto, mr, mz, r, z, psin, psia, rax, zax, &
|
|||||||
|
|
||||||
! Convert to a list of R,z
|
! Convert to a list of R,z
|
||||||
call range2rect(xmin=params%misc%rwall, xmax=R_max, ymin=z(1), ymax=z(mz), &
|
call range2rect(xmin=params%misc%rwall, xmax=R_max, ymin=z(1), ymax=z(mz), &
|
||||||
xv=data%equilibrium%rlim, yv=data%equilibrium%zlim)
|
x=data%equilibrium%rlim, y=data%equilibrium%zlim)
|
||||||
|
|
||||||
! Set the global variables of `limiter`
|
! Set the global variables of `limiter`
|
||||||
call limiter_set_globals(data%equilibrium)
|
call limiter_set_globals(data%equilibrium)
|
||||||
|
@ -206,6 +206,7 @@ contains
|
|||||||
rhopjp=height
|
rhopjp=height
|
||||||
psinjp=height*height
|
psinjp=height*height
|
||||||
rhotjp=frhotor(rhopjp)
|
rhotjp=frhotor(rhopjp)
|
||||||
|
if (rhotjp /= rhotjp) print *, 'Nan!!!!!!!!!!'
|
||||||
psicon(jp)=height
|
psicon(jp)=height
|
||||||
|
|
||||||
call contours_psi(psinjp,rctemp,zctemp,rup,zup,rlw,zlw)
|
call contours_psi(psinjp,rctemp,zctemp,rup,zup,rlw,zlw)
|
||||||
|
@ -385,7 +385,7 @@ contains
|
|||||||
|
|
||||||
subroutine init_misc(params, data)
|
subroutine init_misc(params, data)
|
||||||
! Performs miscellanous initial tasks, before the gray_main subroutine.
|
! Performs miscellanous initial tasks, before the gray_main subroutine.
|
||||||
use reflections, only : range2rect
|
use utils, only : range2rect
|
||||||
use limiter, only : limiter_set_globals=>set_globals
|
use limiter, only : limiter_set_globals=>set_globals
|
||||||
use const_and_precisions, only : cm
|
use const_and_precisions, only : cm
|
||||||
|
|
||||||
@ -440,7 +440,7 @@ contains
|
|||||||
!
|
!
|
||||||
call range2rect(xmin=params%misc%rwall, xmax=R_max, &
|
call range2rect(xmin=params%misc%rwall, xmax=R_max, &
|
||||||
ymin=z_launch - delta_z, ymax=z_launch + delta_z, &
|
ymin=z_launch - delta_z, ymax=z_launch + delta_z, &
|
||||||
xv=data%equilibrium%rlim, yv=data%equilibrium%zlim)
|
x=data%equilibrium%rlim, y=data%equilibrium%zlim)
|
||||||
end block
|
end block
|
||||||
end if
|
end if
|
||||||
|
|
||||||
|
@ -3,9 +3,9 @@ module reflections
|
|||||||
implicit none
|
implicit none
|
||||||
|
|
||||||
private
|
private
|
||||||
public :: reflect,inters_linewall,inside
|
public :: reflect, inters_linewall
|
||||||
public :: linecone_coord,interssegm_coord,interssegm
|
public :: linecone_coord, interssegm_coord, interssegm
|
||||||
public :: wall_refl,range2rect
|
public :: wall_refl
|
||||||
|
|
||||||
contains
|
contains
|
||||||
|
|
||||||
@ -191,47 +191,9 @@ function interssegm(xa,ya,xb,yb)
|
|||||||
end function interssegm
|
end function interssegm
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
subroutine range2rect(xmin,xmax,ymin,ymax,xv,yv)
|
|
||||||
implicit none
|
|
||||||
real(wp_), intent(in) :: xmin,xmax,ymin,ymax
|
|
||||||
real(wp_), intent(out), dimension(5) :: xv,yv
|
|
||||||
xv=(/xmin,xmax,xmax,xmin,xmin/)
|
|
||||||
yv=(/ymin,ymin,ymax,ymax,ymin/)
|
|
||||||
end subroutine range2rect
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
function inside(xc,yc,n,x,y)
|
|
||||||
use utils, only : locatef, locate_unord, intlinf, bubble
|
|
||||||
implicit none
|
|
||||||
integer, intent(in) :: n
|
|
||||||
real(wp_), dimension(n), intent(in) :: xc,yc
|
|
||||||
real(wp_), intent(in) :: x,y
|
|
||||||
logical :: inside
|
|
||||||
integer, dimension(n) :: jint
|
|
||||||
real(wp_), dimension(n) :: xint
|
|
||||||
real(wp_), dimension(n+1) :: xclosed,yclosed
|
|
||||||
integer :: i,nj
|
|
||||||
xclosed(1:n)=xc(1:n)
|
|
||||||
yclosed(1:n)=yc(1:n)
|
|
||||||
xclosed(n+1)=xc(1)
|
|
||||||
yclosed(n+1)=yc(1)
|
|
||||||
call locate_unord(yclosed,n+1,y,jint,n,nj)
|
|
||||||
inside=.false.
|
|
||||||
if (nj==0) return
|
|
||||||
do i=1,nj
|
|
||||||
xint(i)=intlinf(yclosed(jint(i)),xclosed(jint(i)), &
|
|
||||||
yclosed(jint(i)+1),xclosed(jint(i)+1),y)
|
|
||||||
end do
|
|
||||||
call bubble(xint,nj)
|
|
||||||
inside=(mod(locatef(xint,nj,x),2)==1)
|
|
||||||
end function inside
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
subroutine wall_refl(xv,anv,ext,eyt,xvrfl,anvrfl,extr,eytr,walln,irfl)
|
subroutine wall_refl(xv,anv,ext,eyt,xvrfl,anvrfl,extr,eytr,walln,irfl)
|
||||||
use limiter, only : rlim,zlim,nlim
|
use limiter, only : rlim,zlim,nlim
|
||||||
|
use utils, only : inside
|
||||||
implicit none
|
implicit none
|
||||||
! arguments
|
! arguments
|
||||||
integer :: irfl
|
integer :: irfl
|
||||||
@ -253,7 +215,7 @@ subroutine wall_refl(xv,anv,ext,eyt,xvrfl,anvrfl,extr,eytr,walln,irfl)
|
|||||||
smax=smax*1.0e2_wp_
|
smax=smax*1.0e2_wp_
|
||||||
xvrfl=xv+smax*anv0
|
xvrfl=xv+smax*anv0
|
||||||
irfl=1
|
irfl=1
|
||||||
if (.not.inside(rlim,zlim,nlim,rrm,zzm)) then
|
if (.not.inside(rlim,zlim,rrm,zzm)) then
|
||||||
! first wall interface is outside-inside
|
! first wall interface is outside-inside
|
||||||
if (dot_product(walln,walln)<tiny(walln)) then
|
if (dot_product(walln,walln)<tiny(walln)) then
|
||||||
! wall never hit
|
! wall never hit
|
||||||
|
108
src/utils.f90
108
src/utils.f90
@ -85,23 +85,42 @@ contains
|
|||||||
j=jl
|
j=jl
|
||||||
end subroutine locatex
|
end subroutine locatex
|
||||||
|
|
||||||
subroutine locate_unord(a,n,x,j,m,nj)
|
|
||||||
|
subroutine locate_unord(array, value, locs, n, nlocs)
|
||||||
|
! Given an `array` of size `n` and a `value`, finds at most
|
||||||
|
! `n` locations `locs` such that `value` is between
|
||||||
|
! `array(locs(i))` and `array(locs(i+i))`, in whichever order.
|
||||||
|
|
||||||
implicit none
|
implicit none
|
||||||
integer, intent(in) :: n,m
|
|
||||||
integer, intent(out) :: nj
|
! subroutine arguments
|
||||||
real(wp_), dimension(n), intent(in) :: a
|
real(wp_), intent(in) :: array(:)
|
||||||
real(wp_), intent(in) :: x
|
real(wp_), intent(in) :: value
|
||||||
integer, dimension(m), intent(inout) :: j
|
integer, intent(inout) :: locs(n)
|
||||||
|
integer, intent(in) :: n
|
||||||
|
integer, intent(out) :: nlocs
|
||||||
|
|
||||||
|
! local variables
|
||||||
integer :: i
|
integer :: i
|
||||||
nj=0
|
logical :: larger_than_last
|
||||||
do i=1,n-1
|
|
||||||
if (x>a(i).neqv.x>a(i+1)) then
|
nlocs = 0
|
||||||
nj=nj+1
|
if (size(array) < 2) return
|
||||||
if (nj<=m) j(nj)=i
|
|
||||||
|
larger_than_last = value > array(1)
|
||||||
|
do i = 2, size(array)
|
||||||
|
! Note: the condition is equivalent to
|
||||||
|
! (array(i-1) < value < array(i))
|
||||||
|
! .or. (array(i) < value < array(i-1))
|
||||||
|
if (array(i) < value .neqv. larger_than_last) then
|
||||||
|
larger_than_last = value > array(i)
|
||||||
|
nlocs = nlocs + 1
|
||||||
|
if (nlocs <= n) locs(nlocs) = i - 1
|
||||||
end if
|
end if
|
||||||
end do
|
end do
|
||||||
end subroutine locate_unord
|
end subroutine locate_unord
|
||||||
|
|
||||||
|
|
||||||
function intlinf(x1,y1,x2,y2,x) result(y)
|
function intlinf(x1,y1,x2,y2,x) result(y)
|
||||||
!linear interpolation
|
!linear interpolation
|
||||||
!must be x1 != x2
|
!must be x1 != x2
|
||||||
@ -247,6 +266,72 @@ contains
|
|||||||
end subroutine bubble
|
end subroutine bubble
|
||||||
|
|
||||||
|
|
||||||
|
subroutine range2rect(xmin, xmax, ymin, ymax, x, y)
|
||||||
|
! Given two ranges [xmin, xmax], [ymin, ymax] builds
|
||||||
|
! the x and y vertices of the following rectangle:
|
||||||
|
!
|
||||||
|
! (xmin, ymax)╔═════╗(xmax, ymax)
|
||||||
|
! ║4 3║
|
||||||
|
! ║ ║
|
||||||
|
! ║1 2║
|
||||||
|
! (xmin, ymin)╚═════╝(xmin, ymax)
|
||||||
|
!
|
||||||
|
|
||||||
|
implicit none
|
||||||
|
|
||||||
|
! subroutine arguments
|
||||||
|
real(wp_), intent(in) :: xmin, xmax, ymin, ymax
|
||||||
|
real(wp_), intent(out), dimension(5) :: x, y
|
||||||
|
|
||||||
|
x = [xmin, xmax, xmax, xmin, xmin]
|
||||||
|
y = [ymin, ymin, ymax, ymax, ymin]
|
||||||
|
end subroutine range2rect
|
||||||
|
|
||||||
|
|
||||||
|
function inside(vertx, verty, x0, y0)
|
||||||
|
! Tests whether the point (`x0`, `y0`) lies inside the
|
||||||
|
! simple polygon of vertices `vertx`, `verty`.
|
||||||
|
|
||||||
|
implicit none
|
||||||
|
|
||||||
|
! subroutine arguments
|
||||||
|
real(wp_), dimension(:), intent(in) :: vertx, verty
|
||||||
|
real(wp_), intent(in) :: x0, y0
|
||||||
|
logical :: inside
|
||||||
|
|
||||||
|
! local variables
|
||||||
|
integer :: seg(size(vertx))
|
||||||
|
real(wp_) :: x, vertx_(size(vertx)+1), verty_(size(vertx)+1)
|
||||||
|
integer :: i, nsegs, n
|
||||||
|
|
||||||
|
! Ensure the first and last point are the same
|
||||||
|
n = size(vertx)
|
||||||
|
vertx_(1:n) = vertx(1:n)
|
||||||
|
verty_(1:n) = verty(1:n)
|
||||||
|
vertx_(n+1) = vertx(1)
|
||||||
|
verty_(n+1) = verty(1)
|
||||||
|
|
||||||
|
inside = .false.
|
||||||
|
|
||||||
|
! Find the `nsegs` segments that intersect the horizontal
|
||||||
|
! line y=y0, i.e. `verty(seg(i)) < y < verty(seg(i)+1)`
|
||||||
|
call locate_unord(verty_, y0, seg, n, nsegs)
|
||||||
|
|
||||||
|
! No intersections, it must be outside (above or below)
|
||||||
|
if (nsegs == 0) return
|
||||||
|
|
||||||
|
! Count the number of intersections that lie to the left
|
||||||
|
! (equivalently, to the right) of the point. An even number
|
||||||
|
! means that the point is outside the polygon.
|
||||||
|
do i = 1, nsegs
|
||||||
|
! coordinate of the intersection between segment and y=y0
|
||||||
|
x = intlinf(verty_(seg(i)), vertx_(seg(i)), &
|
||||||
|
verty_(seg(i)+1), vertx_(seg(i)+1), y0)
|
||||||
|
if (x < x0) inside = .not. inside
|
||||||
|
end do
|
||||||
|
end function inside
|
||||||
|
|
||||||
|
|
||||||
function get_free_unit(unit) result(i)
|
function get_free_unit(unit) result(i)
|
||||||
! Returns `unit` back or the first free unit
|
! Returns `unit` back or the first free unit
|
||||||
! number `i` if `unit` is absent.
|
! number `i` if `unit` is absent.
|
||||||
@ -299,6 +384,7 @@ contains
|
|||||||
end if
|
end if
|
||||||
end function dirname
|
end function dirname
|
||||||
|
|
||||||
|
|
||||||
function isrelative(filepath)
|
function isrelative(filepath)
|
||||||
! Check if `filepath` is a relative or an absolute path
|
! Check if `filepath` is a relative or an absolute path
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user