-
Notifications
You must be signed in to change notification settings - Fork 1
/
material.F90
264 lines (222 loc) · 8.34 KB
/
material.F90
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
!>
!! \brief This module contains data and routines for handling the material properties on the grid (1D)
!!
!! These properties are; density, temperature, clumping, ionization fractions
!!
!! \b Author: Garrelt Mellema
!!
!! \b Date:
!!
!! \b Version: 1D test problems\n
!! Problem 1: constant density (Strömgren problem)\n
!! Problem 2: 1/r density\n
!! Problem 3: 1/r^2 density (with core of radius r_core)\n
!! Problem 4: cosmological constant density (Shapiro & Giroux problem)\n
module material
! This module contains the grid data and routines for initializing them.
! These are
! - mat_ini : initializes temperature and ionization fractions at start
! Version: 1D test problems
! Problem 1: constant density (Strömgren problem)
! Problem 2: 1/r density
! Problem 3: 1/r^2 density (with core of radius r_core)
! Problem 4: cosmological constant density (Shapiro & Giroux problem)
use precision, only: dp,si
use cgsconstants, only: bh00, albpow
use astroconstants, only: YEAR
use sizes, only: mesh
use file_admin, only: stdinput, file_input
use my_mpi
use grid, only: r,vol
use cgsconstants, only: m_p
use abundances, only: mu
use cosmology, only: cosmology_init,H0,t0,zred_t0
implicit none
save
! ndens - number density (cm^-3) of a cell
! temper - temperature (K) of a cell
! xh - ionization fractions for one cell
real(kind=dp) :: ndens(mesh) !< number density (cm^-3) of a cell
real(kind=dp) :: temper(mesh) !< temperature (K) of a cell
real(kind=dp) :: xh(mesh,0:1) !< ionization fractions for one cell
real(kind=dp) :: clumping !< global clumping factor
real(kind=dp) :: r_core !< core radius (for problems 2 and 3)
real(kind=dp) :: dens_core !< core density (for problems 2 and 3)
integer :: testnum !< number of test problem (1 to 4)
logical :: isothermal !< is the run isothermal?
real(kind=dp) :: gamma_uvb_h
! needed for analytical solution of cosmological Ifront
real(kind=dp) :: t1 !< parameter for analytical solution of test 4
real(kind=dp) :: eta !< parameter for analytical solution of test 4
#ifdef MPI
integer,private :: ierror !< MPI error flag
#endif
contains
! ============================================================================
!> Initializes material properties on grid\n
!! \b Author: Garrelt Mellema\n
!! \b Date: 20-Aug-2006 (f77 21-May-2005 (derives from mat_ini_cosmo2.f))\n
!! \b Version:
!! - 1D\n
!! - Four different test problems\n
!! - Initially completely neutral\n
subroutine mat_ini (restart)
! Initializes material properties on grid
! Author: Garrelt Mellema
! Date: 20-Aug-2006 (f77 21-May-2005 (derives from mat_ini_cosmo2.f))
! Version:
! - 1D
! - Four different test problems
! - Initially completely neutral
integer,intent(out) :: restart !< will be /= 0 if a restart is intended
integer :: i,n ! loop counters
real(kind=dp) :: dens_val
real(kind=dp) :: temper_val
real(kind=dp) :: alpha
real(kind=dp) :: zfactor
real(kind=dp) :: xions
character(len=1) :: answer
! restart
restart=0 ! no restart by default
! Ask for input
if (rank == 0) then
if (.not.file_input) then
write(*,'(A,$)') 'Which test? (1-4): '
endif
read(stdinput,*) testnum
#ifdef MPI
! Distribute the input parameters to the other nodes
call MPI_BCAST(testnum,1,MPI_INTEGER,0,MPI_COMM_NEW,ierror)
#endif
endif
! Set alpha according to test problem
select case (testnum)
case(1,4)
alpha=0.0
case(2)
alpha=-1.0
case(3)
alpha=-2.0
end select
if (rank == 0) then
if (testnum == 1 .or. testnum == 4) then
if (.not.file_input) write(*,'(A,$)') 'Enter density (cm^-3): '
read(stdinput,*) dens_val
elseif (testnum == 2 .or. testnum == 3) then
if (.not.file_input) write(*,'(A,$)') 'Enter reference (core) radius (cm): '
read(stdinput,*) r_core
if (.not.file_input) write(*,'(A,$)') 'Enter density at reference (core)', &
' radius(cm^-3): '
read(stdinput,*) dens_val
endif
if (.not.file_input) write(*,'(A,$)') 'Enter clumping factor: '
read(stdinput,*) clumping
if (.not.file_input) write(*,'(A,$)') 'Enter initial temperature (K): '
read(stdinput,*) temper_val
if (.not.file_input) write(*,'(A,$)') 'Isothermal? (y/n): '
read(stdinput,*) answer
! Isothermal?
if (answer == 'y' .or. answer == 'Y') then
isothermal=.true.
else
isothermal=.false.
endif
if (.not.file_input) write(*,'(A,$)') 'Ionizing background (s^-1): '
read(stdinput,*) gamma_uvb_h
endif
#ifdef MPI
! Distribute the input parameters to the other nodes
call MPI_BCAST(dens_val,1,MPI_DOUBLE_PRECISION,0,MPI_COMM_NEW,ierror)
if (testnum == 2.or.testnum == 3) &
call MPI_BCAST(r_core,1,MPI_DOUBLE_PRECISION,0,MPI_COMM_NEW,ierror)
call MPI_BCAST(clumping,1,MPI_DOUBLE_PRECISION,0,MPI_COMM_NEW,ierror)
call MPI_BCAST(temper_val,1,MPI_DOUBLE_PRECISION,0,MPI_COMM_NEW,ierror)
call MPI_BCAST(isothermal,1,MPI_LOGICAL,0,MPI_COMM_NEW,ierror)
call MPI_BCAST(gamma_uvb_h,1,MPI_DOUBLE_PRECISION,0,MPI_COMM_NEW,ierror)
#endif
! For test problem 4: cosmological parameters
if (testnum == 4) then
call cosmology_init (.true.)
else
call cosmology_init (.false.)
endif
! Assign density and temperature to grid
select case (testnum)
case(1)
do i=1,mesh
ndens(i)=dens_val
temper(i)=temper_val
enddo
case(2,3)
dens_core=dens_val
do i=1,mesh
! This is an attempt to make the initial conditions more
! correct: the density of a cell is the integrated density
! (mass) divided by the volume. Seems to give a worse fit
! to the analytical solution (slightly)
! rl=r(i)-0.5*dr
! rr=r(i)+0.5*dr
! ndens(i)=4.0*pi*dens_val*r_core**(-alpha)/vol(i)*
! $ (rr**(3.0+alpha)-rl**(3.0+alpha))/(3.0+alpha)
! This is just a straight sampling of the density distribution
! using the value at the cell centre.
if (testnum == 3 .and. r(i) <= r_core) then
! Flat core for test 3
ndens(i)=dens_val
else
ndens(i)=dens_val*(r(i)/r_core)**alpha
endif
temper(i)=temper_val
enddo
case(4)
! For cosmological simulations, mean IGM
dens_core=dens_val ! save initial density value
! Parameters needed for the analytical solution
! Recombination time at z0
t1 = 1./(bh00*clumping*dens_core)
eta = t0/t1*(1.+zred_t0)**3
! Set the density to the comoving value
! (as is the spatial coordinate)
! evol_cosmo will set it to proper values.
do i=1,mesh
ndens(i)=dens_val
temper(i)=temper_val
enddo
dens_val=dens_val*(1.+zred_t0)**3 !otherwise recombination time
!scale below would be wrong
end select
! Initialize zfactor for cosmological problems
if (testnum == 4) then
zfactor=(1.+zred_t0)**3
else
zfactor=1.0
endif
! Assign ionization fractions
! Use Gamma_UVB_H for this if it is not zero
if (gamma_uvb_h > 0.0) then
do i=1,mesh
call find_ionfractions_from_uvb(i, ndens(i), temper(i), xions)
xh(i,0)=xions
xh(i,1)=1.0-xh(i,0)
enddo
else
do i=1,mesh
xh(i,0)=1.0-1e-8
xh(i,1)=1e-8
enddo
endif
! Report recombination time scale (in case of screen input)
if (.not.file_input) write(*,'(A,1pe10.3,A)') 'Recombination time scale: ', &
1.0/(dens_val*clumping*bh00*YEAR),' years'
end subroutine mat_ini
subroutine find_ionfractions_from_uvb (ii,nnd,ttemp,xions)
integer,intent(in) :: ii
real(kind=dp),intent(in) :: nnd
real(kind=dp),intent(in) :: ttemp
real(kind=dp) :: two_na
real(kind=dp),intent(out) :: xions
two_na=2.0 * nnd * clumping * bh00 * (temper(ii)/1e4)**albpow
xions=1.0-(sqrt(gamma_uvb_h*(gamma_uvb_h + 2 * two_na))-gamma_uvb_h)/ &
two_na
end subroutine find_ionfractions_from_uvb
end module material