-
Notifications
You must be signed in to change notification settings - Fork 38
Expand file tree
/
Copy pathMHD1D_SND.f90
More file actions
executable file
·217 lines (162 loc) · 6.12 KB
/
Copy pathMHD1D_SND.f90
File metadata and controls
executable file
·217 lines (162 loc) · 6.12 KB
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
program MHD1D_SND
!A 'CLASSICAL HYDRODYNAMIC' SOLVER (WAVES PROPAGATED THRU SOURCE TERMS)
!FOR THE IDEAL MHD EQUATIONS IN 1.66 DIMENSIONS. M. ZETTERGREN, T. SYMONS
use advec
implicit none
integer, parameter :: method=1
integer, parameter :: npts=1501 !hard-coded for SND problem
real(wp), parameter :: tcfl=0.75
real(wp), parameter :: pi=3.14
real(wp), parameter :: amu=1.67e-27, kb=1.38e-23, gammap=5.0/3.0, mu0=4.0*pi*1e-7
real(wp), parameter :: stride=2e3
real(wp), dimension(npts) :: dx1i
real(wp), dimension(-1:npts+2) :: x1
real(wp), dimension(-1:npts+2) :: rhom,rhou1,rhou2,rhou3,B1,B2,B3,rhoeps,u1,u2,u3
real(wp), dimension(npts+1) :: x1i,v1i
real(wp), dimension(0:npts+2) :: dx1
integer :: lx1,it,ix1, u
real(wp) :: t=0,dt=1e-6,dtout=5,tdur=300,xi=2
real(wp) :: toutnext
real(wp), dimension(npts) :: Q,p,grad1B1u3,rhoepshalf,sourceterm,du1full,grad1u1,vA,vsnd !'work' arrays
real(wp) :: sinwt,sinwt2 !'work' vars
real(wp) :: vsnd2,rhom1,u11,B10,rhom0,vA0,omega0,T0,tfwhm,tdel,Ti=8000 !linear wave params
!GRID CONSTRUCTION
x1=[ (ix1*stride, ix1=-2,npts+1) ]
lx1=size(x1)-4 !exclude ghost cells in count
dx1=x1(0:lx1+2)-x1(-1:lx1+1) !backward diffs
x1i(1:lx1+1)=0.5*(x1(0:lx1)+x1(1:lx1+1)) !interface data
dx1i=x1i(2:lx1+1)-x1i(1:lx1)
!PREP OUTPUT FILE
open(newunit=u,file='MHD1D.dat',status='replace', access='stream', action='write')
write(u,*) lx1
call writearray(u,x1)
!INITIAL CONDITIONS
rhom=1e9*amu
rhou1=0
rhou2=0
rhou3=0
B1=10e-9
B2=0
B3=0
rhoeps=rhom*kb*Ti/amu/(gammap-1)
u1=rhou1/rhom
u2=rhou2/rhom
u3=rhou3/rhom
!STATIC INFO FOR CALCULATING BOUNDARY CONDITIONS
u11=5.0
B10=sum(B1)/size(B1)
rhom0=sum(rhom)/size(rhom)
vsnd2=gammap*kb*Ti/amu
rhom1=rhom0/sqrt(vsnd2)*u11
vA0=B10/sqrt(mu0*rhom0)
omega0=0.15*2*pi
T0=2*pi/omega0
tfwhm=5*T0
tdel=10*T0
!MAIN LOOP
toutnext=dtout
do while (t<tdur)
!ART. VISCOSITY DIFFS OF U1
du1full=rhou1(2:lx1+1)/rhom(2:lx1+1)-rhou1(0:lx1-1)/rhom(0:lx1-1)
!TIME STEP DETERMINATION
vA=sqrt((B1(1:lx1)**2+B2(1:lx1)**2+B3(1:lx1)**2)/mu0/rhom(1:lx1))
p=rhoeps(1:lx1)*(gammap-1)
vsnd=sqrt(gammap*p/rhom(1:lx1))
dt=tcfl*minval(dx1i/(vA+vsnd+abs(u1(1:lx1))))
t=t+dt !time after this step
write(*,*) 'Time: ',t,' Time step: ',dt, &
' Alfven speed: ',maxval(vA/1e3), &
' sound speed: ',maxval(vsnd/1e3), &
' matter speed: ',maxval(abs(u1(1:lx1))/1e3)
!BOUNDARY CONDITIONS
sinwt=exp(-(t-tdel)**2/tfwhm**2)*sin(omega0*t)
sinwt2=exp(-(t-tdel)**2/(tfwhm/2)**2)*sin(2*omega0*t)
rhom(-1:0)=rhom0+rhom1*sinwt
rhom(lx1+1:lx1+2)=rhom0+rhom1*sinwt2
rhou1(-1:0)=2*rhom0*u11*sinwt-rhou1(1)
rhou1(lx1+1:lx1+2)=-2*rhom0*u11*sinwt2-rhou1(lx1)
rhou2(-1:0)=rhou2(1)
rhou2(lx1+1:lx1+2)=rhou2(lx1)
rhou3(-1:0)=rhou3(1)
rhou3(lx1+1:lx1+2)=rhou3(lx1)
B2(-1:0)=B2(1)
B2(lx1+1:lx1+2)=B2(lx1)
B3(-1:0)=B3(1)
B3(lx1+1:lx1+2)=B3(lx1)
rhoeps(-1:0)=rhoeps(1)
rhoeps(lx1+1:lx1+2)=rhoeps(lx1)
!ADVECTION SUBSTEP
v1i(:)=0.5*(rhou1(0:lx1)/rhom(0:lx1)+rhou1(1:lx1+1)/rhom(1:lx1+1)) !CELL INTERFACE SPEEDS (LEFT WALL OF ITH CELL)
rhom=advec1D_MC(rhom,v1i,dt,dx1,dx1i)
rhou1=advec1D_MC(rhou1,v1i,dt,dx1,dx1i)
rhou2=advec1D_MC(rhou2,v1i,dt,dx1,dx1i)
rhou3=advec1D_MC(rhou3,v1i,dt,dx1,dx1i)
B2=advec1D_MC(B2,v1i,dt,dx1,dx1i)
B3=advec1D_MC(B3,v1i,dt,dx1,dx1i)
rhoeps=advec1D_MC(rhoeps,v1i,dt,dx1,dx1i)
!VON NEUMANN-RICHTMYER ARTIFICIAL VISCOSITY ('CLEANS UP' SOLUTIONS
!WITH STRONG SHOCKS) [POTTER, 1970; STONE ET AL 1992]
Q=0.25*xi**2*(min(du1full,0.0))**2*rhom(1:lx1)
!SOURCE TERMS FOR 1-COMP OF MOMENTUM
p=rhoeps(1:lx1)*(gammap-1)+(B1(1:lx1)**2+B2(1:lx1)**2+B3(1:lx1)**2)/2/mu0+Q
rhou1(1:lx1)=rhou1(1:lx1)+dt*(-1*derivative(p,dx1(1:lx1)))
!SOURCE TERMS FOR 2-COMP OF MOMENTUM
rhou2(1:lx1)=rhou2(1:lx1)+dt*(1/mu0*B1(1:lx1)*derivative(B2(1:lx1),dx1(1:lx1)))
!SOURCE TERMS FOR Y-COMP OF MOMENTUM
rhou3(1:lx1)=rhou3(1:lx1)+dt*(1/mu0*B1(1:lx1)*derivative(B3(1:lx1),dx1(1:lx1)))
!PARTIALLY UPDATED FLOWS FROM MOMENTA
u1=rhou1/rhom
u2=rhou2/rhom
u3=rhou3/rhom
!SOURCE TERMS FOR X-COMP OF B-FIELD
B2(1:lx1)=B2(1:lx1)+dt*derivative(B1(1:lx1)*u2(1:lx1),dx1(1:lx1))
!SOURCE TERMS FOR Y-COMP OF B-FIELD
grad1B1u3=derivative(B1(1:lx1)*u3(1:lx1),dx1(1:lx1))
! grad1B1u3(1)=(B1(2)*u3(2)-B1(1)*u3(0))/(dx1(1)+dx1(2)) !USE GHOST CELL BCS TO COUPLE WAVE PERTURBATIONS ONTO MESH (check B1(1) --> B1(0))
! grad1B1u3(lx1)=(B1(lx1)*u3(lx1+1)-B1(lx1-1)*u3(lx1-1))/(dx1(lx1+1)+dx1(lx1)) !DITTO FOR RIGHT BOUNDARY
grad1B1u3(1)=(B1(2)*u3(2)-B1(0)*u3(0))/(dx1(2)+dx1(1))
grad1B1u3(lx1)=(B1(lx1+1)*u3(lx1+1)-B1(lx1-1)*u3(lx1-1))/(dx1(lx1+1)+dx1(lx1))
B3(1:lx1)=B3(1:lx1)+dt*grad1B1u3
!SOURCE TERMS FOR INTERNAL ENERGY (RK2 STEPPING, I THINK THIS MIGHT ACTUALLY BE DOABLE IN ONE STEP...)
p=rhoeps(1:lx1)*(gammap-1)+Q
grad1u1=derivative(u1(1:lx1),dx1(1:lx1))
sourceterm=-p*grad1u1
rhoepshalf=rhoeps(1:lx1)+dt/2*sourceterm
p=rhoepshalf*(gammap-1)+Q
sourceterm=-p*grad1u1
rhoeps(1:lx1)=rhoeps(1:lx1)+dt*sourceterm
!OUTPUT
if (t>toutnext) then
write(u,*) t
call writearray(u,rhom/amu)
call writearray(u,u1)
call writearray(u,u2)
call writearray(u,u3)
call writearray(u,B1)
call writearray(u,B2)
call writearray(u,B3)
call writearray(u,rhoeps*(gammap-1))
toutnext=toutnext+dtout
end if
end do
close(u)
contains
subroutine writearray(fileunit,array)
integer, intent(in) :: fileunit
real(wp), dimension(:), intent(in) :: array
integer :: k
do k=1,size(array)
write(fileunit,*) array(k)
end do
end subroutine writearray
function derivative(f,dx)
real(wp), dimension(:), intent(in) :: dx !presumed backward diffs.
real(wp), dimension(:), intent(in) :: f
integer :: lx
real(wp), dimension(1:size(f)) :: derivative
lx=size(f)
derivative(1)=(f(2)-f(1))/dx(2)
derivative(2:lx-1)=(f(3:lx)-f(1:lx-2))/(dx(3:lx)+dx(2:lx-1))
derivative(lx)=(f(lx)-f(lx-1))/dx(lx)
end function derivative
end program MHD1D_SND