src/coreprofiles.f90: use derived types
This commit is contained in:
parent
948a512254
commit
f56e1cbc05
@ -2,11 +2,11 @@ module coreprofiles
|
||||
use const_and_precisions, only : wp_,zero,one
|
||||
implicit none
|
||||
|
||||
INTEGER, SAVE :: npp,nsfd
|
||||
REAL(wp_), SAVE :: psdbnd,psnpp,denpp,ddenpp,d2denpp
|
||||
REAL(wp_), DIMENSION(:), ALLOCATABLE, SAVE :: tfn,cfn,psrad
|
||||
REAL(wp_), DIMENSION(:,:), ALLOCATABLE, SAVE :: ct,cz
|
||||
REAL(wp_), SAVE :: dens0,aln1,aln2,te0,dte0,alt1,alt2,zeffan
|
||||
integer, save :: npp, nsfd
|
||||
real(wp_), save :: psdbnd, psnpp, denpp, ddenpp, d2denpp
|
||||
real(wp_), dimension(:), allocatable, save :: tfn, cfn, psrad
|
||||
real(wp_), dimension(:, :), allocatable, save :: ct, cz
|
||||
real(wp_), save :: dens0, aln1, aln2, te0, dte0, alt1, alt2, zeffan
|
||||
|
||||
contains
|
||||
|
||||
@ -124,36 +124,58 @@ contains
|
||||
fzeff=spli(cz,npp,k,dps)
|
||||
endif
|
||||
end function fzeff
|
||||
|
||||
|
||||
subroutine read_profiles(filenm,psin,te,ne,zeff,unit)
|
||||
use utils, only : get_free_unit
|
||||
subroutine read_profiles(filenm, data, unit)
|
||||
! Reads the radial plasma profiles from `file` and store them
|
||||
! into `data`. If given, the file is opened in the `unit` number.
|
||||
! Format notes:
|
||||
! 1. The file is formatted as a table with the following columns:
|
||||
! radial coordinate, temperature, density, effective charge.
|
||||
! 2. The first line is a header specifying the number of rows.
|
||||
use utils, only : get_free_unit
|
||||
use gray_params, only : profiles_data
|
||||
|
||||
implicit none
|
||||
! arguments
|
||||
character(len=*), intent(in) :: filenm
|
||||
real(wp_), dimension(:), allocatable, intent(out) :: psin,te,ne,zeff
|
||||
integer, optional, intent(in) :: unit
|
||||
! local variables
|
||||
integer :: u, i, n
|
||||
|
||||
|
||||
! subroutine arguments
|
||||
character(len=*), intent(in) :: filenm
|
||||
type(profiles_data), intent(out) :: data
|
||||
integer, optional, intent(in) :: unit
|
||||
|
||||
! local variables
|
||||
integer :: u, i, nrows
|
||||
|
||||
! Free the arrays when already allocated
|
||||
if(allocated(data%psrad)) deallocate(data%psrad)
|
||||
if(allocated(data%terad)) deallocate(data%terad)
|
||||
if(allocated(data%derad)) deallocate(data%derad)
|
||||
if(allocated(data%zfc)) deallocate(data%zfc)
|
||||
|
||||
if (present(unit)) then
|
||||
u=unit
|
||||
u = unit
|
||||
else
|
||||
u=get_free_unit()
|
||||
u = get_free_unit()
|
||||
end if
|
||||
open(file=trim(filenm),status='old',action='read',unit=u)
|
||||
read(u,*) n
|
||||
if(allocated(psin)) deallocate(psin)
|
||||
if(allocated(te)) deallocate(te)
|
||||
if(allocated(ne)) deallocate(ne)
|
||||
if(allocated(zeff)) deallocate(zeff)
|
||||
allocate(psin(n),te(n),ne(n),zeff(n))
|
||||
do i=1,n
|
||||
read(u,*) psin(i),te(i),ne(i),zeff(i)
|
||||
|
||||
! Read number of rows and allocate the arrays
|
||||
open(file=trim(filenm), status='old', action='read', unit=u)
|
||||
read(u, *) nrows
|
||||
allocate(data%psrad(nrows), data%terad(nrows), &
|
||||
data%derad(nrows), data%zfc(nrows))
|
||||
|
||||
! Read the table rows
|
||||
do i=1,nrows
|
||||
read(u, *) data%psrad(i), data%terad(i), data%derad(i), data%zfc(i)
|
||||
end do
|
||||
psin(1)=max(psin(1),zero)
|
||||
close(u)
|
||||
|
||||
! ??
|
||||
data%psrad(1) = max(data%psrad(1), zero)
|
||||
|
||||
end subroutine read_profiles
|
||||
|
||||
|
||||
subroutine read_profiles_an(filenm,te,ne,zeff,unit)
|
||||
use utils, only : get_free_unit
|
||||
implicit none
|
||||
@ -182,47 +204,66 @@ contains
|
||||
close(u)
|
||||
end subroutine read_profiles_an
|
||||
|
||||
subroutine tene_scal(te,ne,tfact,nfact,bfact,iscal,iprof)
|
||||
|
||||
subroutine scale_profiles(params, factb, data)
|
||||
! Rescales the temperature and density profiles by `params%factte`
|
||||
! and `params%factne`, respectively, according to the model
|
||||
! specified by `params%iscal`.
|
||||
! See the GRAY user manual for the explanation.
|
||||
use gray_params, only : profiles_parameters, profiles_data
|
||||
|
||||
implicit none
|
||||
! arguments
|
||||
real(wp_), dimension(:), intent(inout) :: te,ne
|
||||
real(wp_), intent(in) :: tfact,nfact,bfact
|
||||
integer, intent(in) :: iscal,iprof
|
||||
! local variables
|
||||
real(wp_) :: aat,aan,ffact
|
||||
integer :: lastte,lastne
|
||||
|
||||
if (iscal==0) then
|
||||
aat=2.0_wp_/3.0_wp_
|
||||
aan=4.0_wp_/3.0_wp_
|
||||
else
|
||||
aat=1.0_wp_
|
||||
aan=1.0_wp_
|
||||
end if
|
||||
if(iscal==2) then
|
||||
ffact=1.0_wp_
|
||||
else
|
||||
ffact=bfact
|
||||
end if
|
||||
if (iprof==0) then
|
||||
lastte=2
|
||||
lastne=1
|
||||
else
|
||||
lastte=size(te)
|
||||
lastne=size(ne)
|
||||
end if
|
||||
te(1:lastte)=te(1:lastte)*ffact**aat*tfact
|
||||
ne(1:lastne)=ne(1:lastne)*ffact**aan*nfact
|
||||
end subroutine tene_scal
|
||||
! subroutine arguments
|
||||
type(profiles_parameters), intent(in) :: params
|
||||
real(wp_), intent(in) :: factb
|
||||
type(profiles_data), intent(inout) :: data
|
||||
|
||||
subroutine set_prfspl(psin,te,ne,zeff,ssplne,psdbndmx)
|
||||
! local variables
|
||||
real(wp_) :: aat, aan, ffact
|
||||
integer :: last_te, last_ne
|
||||
|
||||
if (params%iscal==0) then
|
||||
aat = 2.0_wp_/3.0_wp_
|
||||
aan = 4.0_wp_/3.0_wp_
|
||||
else
|
||||
aat = one
|
||||
aan = one
|
||||
end if
|
||||
|
||||
if(params%iscal==2) then
|
||||
ffact = one
|
||||
else
|
||||
ffact = factb
|
||||
end if
|
||||
|
||||
if(params%iprof==0) then
|
||||
last_te = 2
|
||||
last_ne = 1
|
||||
else
|
||||
last_te = size(data%terad)
|
||||
last_ne = size(data%derad)
|
||||
end if
|
||||
|
||||
data%terad(1:last_te) = data%terad(1:last_te) * ffact**aat * params%factte
|
||||
data%derad(1:last_ne) = data%derad(1:last_ne) * ffact**aan * params%factne
|
||||
end subroutine scale_profiles
|
||||
|
||||
|
||||
subroutine set_prfspl(params, data)
|
||||
! Computes splines for the plasma profiles data and stores them
|
||||
! in their respective global variables, see the top of this file.
|
||||
use simplespline, only : difcs
|
||||
use dierckx, only : curfit, splev, splder
|
||||
use dierckx, only : curfit, splev, splder
|
||||
use gray_params, only : profiles_parameters, profiles_data
|
||||
|
||||
implicit none
|
||||
! arguments
|
||||
real(wp_), dimension(:), intent(in) :: psin,te,ne,zeff
|
||||
real(wp_), intent(in) :: ssplne,psdbndmx
|
||||
! local variables
|
||||
|
||||
! subroutine arguments
|
||||
type(profiles_parameters), intent(in) :: params
|
||||
type(profiles_data), intent(inout) :: data
|
||||
|
||||
! local variables
|
||||
integer, parameter :: iopt=0, kspl=3
|
||||
integer :: n, npest, lwrkf, ier
|
||||
real(wp_) :: xb, xe, fp, xnv, xxp,xxm,delta2,ssplne_loc
|
||||
@ -230,14 +271,14 @@ contains
|
||||
integer, dimension(:), allocatable :: iwrkf
|
||||
real(wp_), dimension(1) :: dedge,ddedge,d2dedge
|
||||
|
||||
n=size(psin)
|
||||
n=size(data%psrad)
|
||||
npest=n+4
|
||||
lwrkf=n*4+npest*16
|
||||
allocate(wrkf(lwrkf),iwrkf(npest),wf(n))
|
||||
|
||||
ssplne_loc=ssplne
|
||||
ssplne_loc=params%sspld
|
||||
|
||||
! if necessary, reallocate spline arrays
|
||||
! if necessary, reallocate spline arrays
|
||||
if(.not.allocated(psrad)) then
|
||||
allocate(psrad(n),ct(n,4),cz(n,4))
|
||||
else
|
||||
@ -255,37 +296,37 @@ contains
|
||||
end if
|
||||
end if
|
||||
|
||||
! spline approximation of temperature and Zeff
|
||||
call difcs(psin,te, n,iopt,ct,ier)
|
||||
call difcs(psin,zeff,n,iopt,cz,ier)
|
||||
psrad=psin
|
||||
! spline approximation of temperature and data%zfc
|
||||
call difcs(data%psrad,data%terad, n,iopt,ct,ier)
|
||||
call difcs(data%psrad,data%zfc,n,iopt,cz,ier)
|
||||
psrad=data%psrad
|
||||
npp=n
|
||||
|
||||
! spline approximation of density
|
||||
! spline approximation of density
|
||||
xb=zero
|
||||
xe=psin(n)
|
||||
xe=data%psrad(n)
|
||||
wf(:)=one
|
||||
call curfit(iopt,n,psin,ne,wf,xb,xe,kspl,ssplne_loc,npest, &
|
||||
call curfit(iopt,n,data%psrad,data%derad,wf,xb,xe,kspl,ssplne_loc,npest, &
|
||||
nsfd,tfn,cfn,fp,wrkf,lwrkf,iwrkf,ier)
|
||||
! if ier=-1 data are re-fitted using sspl=0
|
||||
! if ier=-1 data are re-fitted using sspl=0
|
||||
if(ier==-1) then
|
||||
write(*,*) 'density curfit: ier=-1. Re-fitting with interpolating spline'
|
||||
ssplne_loc=0.0_wp_
|
||||
call curfit(iopt,n,psin,ne,wf,xb,xe,kspl,ssplne_loc,npest, &
|
||||
call curfit(iopt,n,data%psrad,data%derad,wf,xb,xe,kspl,ssplne_loc,npest, &
|
||||
nsfd,tfn,cfn,fp,wrkf,lwrkf,iwrkf,ier)
|
||||
end if
|
||||
|
||||
! compute polinomial extrapolation matching the spline boundary up to the
|
||||
! 2nd order derivative, extending the profile up to psi=psdbnd where
|
||||
! ne=ne'=ne''=0
|
||||
! spline value and derivatives at the edge
|
||||
call splev(tfn,nsfd,cfn,kspl,psin(n:n),dedge(1:1),1,ier)
|
||||
call splder(tfn,nsfd,cfn,kspl,1,psin(n:n),ddedge(1:1), 1,wrkf(1:nsfd),ier)
|
||||
call splder(tfn,nsfd,cfn,kspl,2,psin(n:n),d2dedge(1:1),1,wrkf(1:nsfd),ier)
|
||||
! determination of the boundary
|
||||
psdbnd=psdbndmx
|
||||
! compute polinomial extrapolation matching the spline boundary up to the
|
||||
! 2nd order derivative, extending the profile up to psi=psdbnd where
|
||||
! data%derad=data%derad'=data%derad''=0
|
||||
! spline value and derivatives at the edge
|
||||
call splev(tfn,nsfd,cfn,kspl,data%psrad(n:n),dedge(1:1),1,ier)
|
||||
call splder(tfn,nsfd,cfn,kspl,1,data%psrad(n:n),ddedge(1:1), 1,wrkf(1:nsfd),ier)
|
||||
call splder(tfn,nsfd,cfn,kspl,2,data%psrad(n:n),d2dedge(1:1),1,wrkf(1:nsfd),ier)
|
||||
! determination of the boundary
|
||||
psdbnd=params%psnbnd
|
||||
|
||||
psnpp=psin(n)
|
||||
psnpp=data%psrad(n)
|
||||
denpp=dedge(1)
|
||||
ddenpp=ddedge(1)
|
||||
d2denpp=d2dedge(1)
|
||||
@ -293,7 +334,7 @@ contains
|
||||
delta2=(ddenpp/d2denpp)**2-2.0_wp_*denpp/d2denpp
|
||||
xnv=psnpp-ddenpp/d2denpp
|
||||
if(delta2 < zero) then
|
||||
! if(xnv > psnpp) psdbnd=min(psdbnd,xnv)
|
||||
! if(xnv > psnpp) psdbnd=min(psdbnd,xnv)
|
||||
else
|
||||
xxm=xnv-sqrt(delta2)
|
||||
xxp=xnv+sqrt(delta2)
|
||||
@ -302,12 +343,13 @@ contains
|
||||
else if (xxp > psnpp) then
|
||||
psdbnd=min(psdbnd,xxp)
|
||||
end if
|
||||
write(*,*) 'density psdbnd=',psdbnd
|
||||
write(*,'(a,f5.3)') 'density psdbnd=', psdbnd
|
||||
end if
|
||||
|
||||
deallocate(iwrkf,wrkf,wf)
|
||||
end subroutine set_prfspl
|
||||
|
||||
|
||||
subroutine unset_prfspl
|
||||
implicit none
|
||||
|
||||
|
63
src/main.f90
63
src/main.f90
@ -21,7 +21,7 @@ program main
|
||||
! Read the input data into set the global variables
|
||||
! of the respective module. Note: order matters.
|
||||
call init_equilibrium(params, data)
|
||||
call init_profiles(params, data)
|
||||
call init_profiles(params%profiles, params%equilibrium%factb, data%profiles)
|
||||
call init_antenna(params%antenna)
|
||||
call init_misc(params, data)
|
||||
|
||||
@ -182,63 +182,50 @@ contains
|
||||
end subroutine deinit_equilibrium
|
||||
|
||||
|
||||
subroutine init_profiles(params, data)
|
||||
subroutine init_profiles(params, factb, data)
|
||||
! Reads the plasma kinetic profiles file (containing the elecron
|
||||
! temperature, density and plasma effective charge) and initialises
|
||||
! the respective GRAY data structure.
|
||||
use gray_params, only : profiles_parameters, profiles_data
|
||||
use equilibrium, only : frhopolv
|
||||
use coreprofiles, only : read_profiles_an, read_profiles, tene_scal, &
|
||||
set_prfan, set_prfspl
|
||||
use coreprofiles, only : read_profiles_an, read_profiles, &
|
||||
scale_profiles, set_prfan, set_prfspl
|
||||
implicit none
|
||||
|
||||
! subroutine arguments
|
||||
type(gray_parameters), intent(in) :: params
|
||||
type(gray_data), intent(inout), target :: data
|
||||
type(profiles_parameters), intent(in) :: params
|
||||
real(wp_), intent(in) :: factb
|
||||
type(profiles_data), intent(out) :: data
|
||||
|
||||
! local variables
|
||||
type(profiles_parameters) :: profp
|
||||
type(profiles_data), pointer :: profd
|
||||
|
||||
! Radial coordinate (depending on profp%irho: ρ_t, ρ_p, or ψ)
|
||||
real(wp_), allocatable :: xrad(:)
|
||||
|
||||
profp = params%profiles
|
||||
profd => data%profiles
|
||||
|
||||
if(params%profiles%iprof == 0) then
|
||||
if(params%iprof == 0) then
|
||||
! Analytical profiles
|
||||
! TODO: rewrite using derived type
|
||||
call read_profiles_an(profp%filenm, profd%terad, profd%derad, profd%zfc)
|
||||
call read_profiles_an(params%filenm, data%terad, data%derad, data%zfc)
|
||||
else
|
||||
! Numerical profiles
|
||||
call read_profiles(profp%filenm, xrad, profd%terad, profd%derad, profd%zfc)
|
||||
call read_profiles(params%filenm, data)
|
||||
|
||||
allocate(profd%psrad(size(xrad)))
|
||||
|
||||
select case (profp%irho)
|
||||
case (0) ! xrad is rhot
|
||||
profd%psrad = frhopolv(xrad)**2
|
||||
case (1) ! xrad is rhop
|
||||
profd%psrad = xrad**2
|
||||
case default ! xrad is psi
|
||||
profd%psrad = xrad
|
||||
! Convert psrad to ψ
|
||||
select case (params%irho)
|
||||
case (0) ! psrad is ρ_t
|
||||
data%psrad = frhopolv(data%psrad)**2
|
||||
case (1) ! psrad is ρ_p
|
||||
data%psrad = data%psrad**2
|
||||
case default ! psrad is already ψ
|
||||
end select
|
||||
deallocate(xrad)
|
||||
|
||||
end if
|
||||
|
||||
! Rescale input data
|
||||
! TODO: rewrite using derived type
|
||||
call tene_scal(profd%terad, profd%derad, profp%factte, profp%factne, &
|
||||
params%equilibrium%factb, profp%iscal, profp%iprof)
|
||||
call scale_profiles(params, factb, data)
|
||||
|
||||
! Set global variables
|
||||
! TODO: rewrite using derived type
|
||||
if(params%profiles%iprof == 0) then
|
||||
call set_prfan(profd%terad, profd%derad, profd%zfc)
|
||||
if(params%iprof == 0) then
|
||||
! Analytical profiles
|
||||
! TODO: rewrite using derived type
|
||||
call set_prfan(data%terad, data%derad, data%zfc)
|
||||
else
|
||||
call set_prfspl(profd%psrad, profd%terad, profd%derad, profd%zfc, &
|
||||
profp%sspld, profp%psnbnd)
|
||||
! Numerical profiles
|
||||
call set_prfspl(params, data)
|
||||
end if
|
||||
end subroutine init_profiles
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user