-
Notifications
You must be signed in to change notification settings - Fork 38
/
Copy pathHydroMech2D.jl
208 lines (198 loc) · 11.3 KB
/
HydroMech2D.jl
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
const USE_GPU = false # Use GPU? If this is set false, then no GPU needs to be available
using ParallelStencil
using ParallelStencil.FiniteDifferences2D
@static if USE_GPU
@init_parallel_stencil(CUDA, Float64, 2)
else
@init_parallel_stencil(Threads, Float64, 2)
end
using Plots, Printf, Statistics, LinearAlgebra
@parallel function update_old!(Phi_o::Data.Array, ∇V_o::Data.Array, Phi::Data.Array, ∇V::Data.Array)
@all(Phi_o) = @all(Phi)
@all(∇V_o) = @all(∇V)
return
end
@parallel function compute_params_∇!(EtaC::Data.Array, K_muf::Data.Array, Rog::Data.Array, ∇V::Data.Array, ∇qD::Data.Array, Phi::Data.Array, Pf::Data.Array, Pt::Data.Array, Vx::Data.Array, Vy::Data.Array, qDx::Data.Array, qDy::Data.Array, μs::Data.Number, η2μs::Data.Number, R::Data.Number, λPe::Data.Number, k_μf0::Data.Number, ϕ0::Data.Number, nperm::Data.Number, θ_e::Data.Number, θ_k::Data.Number, ρfg::Data.Number, ρsg::Data.Number, ρgBG::Data.Number, dx::Data.Number, dy::Data.Number)
@all(EtaC) = (1.0-θ_e)*@all(EtaC) + θ_e*( μs/@all(Phi)*η2μs*(1.0+0.5*(1.0/R-1.0)*(1.0+tanh((@all(Pf)-@all(Pt))/λPe))) )
@all(K_muf) = (1.0-θ_k)*@all(K_muf) + θ_k*( k_μf0 * (@all(Phi)/ϕ0)^nperm )
@all(Rog) = ρfg*@all(Phi) + ρsg*(1.0-@all(Phi)) - ρgBG
@all(∇V) = @d_xa(Vx)/dx + @d_ya(Vy)/dy
@all(∇qD) = @d_xa(qDx)/dx + @d_ya(qDy)/dy
return
end
@parallel function compute_RP!(dτPf::Data.Array, RPt::Data.Array, RPf::Data.Array, K_muf::Data.Array, ∇V::Data.Array, ∇qD::Data.Array, Pt::Data.Array, Pf::Data.Array, EtaC::Data.Array, Phi::Data.Array, Pfsc::Data.Number, Pfdmp::Data.Number, min_dxy2::Data.Number, dx::Data.Number, dy::Data.Number)
@inn(dτPf) = min_dxy2/@maxloc(K_muf)/4.1/Pfsc
@all(RPt) = - @all(∇V) - (@all(Pt) - @all(Pf))/(@all(EtaC)*(1.0-@all(Phi)))
@all(RPf) = @all(RPf)*Pfdmp - @all(∇qD) + (@all(Pt) - @all(Pf))/(@all(EtaC)*(1.0-@all(Phi)))
return
end
@parallel function compute_P_τ!(Pt::Data.Array, Pf::Data.Array, τxx::Data.Array, τyy::Data.Array, σxy::Data.Array, RPt::Data.Array, RPf::Data.Array, dτPf::Data.Array, Vx::Data.Array, Vy::Data.Array, ∇V::Data.Array, dτPt::Data.Number, μs::Data.Number, β_n::Data.Number, dx::Data.Number, dy::Data.Number)
@all(Pt) = @all(Pt) + dτPt *@all(RPt)
@all(Pf) = @all(Pf) + @all(dτPf)*@all(RPf)
@all(τxx) = 2.0*μs*( @d_xa(Vx)/dx - 1.0/3.0*@all(∇V) - β_n*@all(RPt) )
@all(τyy) = 2.0*μs*( @d_ya(Vy)/dy - 1.0/3.0*@all(∇V) - β_n*@all(RPt) )
@all(σxy) = 2.0*μs*(0.5*( @d_yi(Vx)/dy + @d_xi(Vy)/dx ))
return
end
@parallel function compute_res!(Rx::Data.Array, Ry::Data.Array, dVxdτ::Data.Array, dVydτ::Data.Array, τxx::Data.Array, τyy::Data.Array, σxy::Data.Array, Pt::Data.Array, Rog::Data.Array, dampX::Data.Number, dampY::Data.Number, dx::Data.Number, dy::Data.Number)
@all(Rx) = @d_xi(τxx)/dx + @d_ya(σxy)/dy - @d_xi(Pt)/dx
@all(Ry) = @d_yi(τyy)/dy + @d_xa(σxy)/dx - @d_yi(Pt)/dy - @av_yi(Rog)
@all(dVxdτ) = dampX*@all(dVxdτ) + @all(Rx)
@all(dVydτ) = dampY*@all(dVydτ) + @all(Ry)
return
end
@parallel function compute_update!(Vx::Data.Array, Vy::Data.Array, qDx::Data.Array, qDy::Data.Array, Phi::Data.Array, dVxdτ::Data.Array, dVydτ::Data.Array, K_muf::Data.Array, Pf::Data.Array, Phi_o::Data.Array, ∇V::Data.Array, ∇V_o::Data.Array, dτV::Data.Number, ρfg::Data.Number, ρgBG::Data.Number, CN::Data.Number, dt::Data.Number, dx::Data.Number, dy::Data.Number)
@inn(Vx) = @inn(Vx) + dτV*@all(dVxdτ)
@inn(Vy) = @inn(Vy) + dτV*@all(dVydτ)
@inn(qDx) = -@av_xi(K_muf)*(@d_xi(Pf)/dx)
@inn(qDy) = -@av_yi(K_muf)*(@d_yi(Pf)/dy + (ρfg - ρgBG))
@all(Phi) = @all(Phi_o) + (1.0-@all(Phi))*(CN*@all(∇V_o) + (1.0-CN)*@all(∇V))*dt
return
end
@parallel_indices (ix,iy) function bc_x!(A::Data.Array)
A[1 , iy] = A[2 , iy]
A[end, iy] = A[end-1, iy]
return
end
@parallel_indices (ix,iy) function bc_y!(A::Data.Array)
A[ix, 1 ] = A[ix, 2 ]
A[ix, end] = A[ix, end-1]
return
end
##################################################
@views function HydroMech2D()
# Physics - scales
ρfg = 1.0 # fluid rho*g
k_μf0 = 1.0 # reference permeability
ηC0 = 1.0 # reference bulk viscosity
# Physics - non-dimensional parameters
η2μs = 10.0 # bulk/shear viscosity ration
R = 500.0 # Compaction/decompaction strength ratio for bulk rheology
nperm = 3.0 # Carman-Kozeny exponent
ϕ0 = 0.01 # reference porosity
ra = 2 # radius of initil porosity perturbation
λ0 = 1.0 # standard deviation of initial porosity perturbation
t_tot = 0.02 # total time
# Physics - dependent scales
ρsg = 2.0*ρfg # solid rho*g
lx = 20.0 # domain size x
ly = ra*lx # domain size y
ϕA = 2*ϕ0 # amplitude of initial porosity perturbation
λPe = 0.01 # effective pressure transition zone
dt = 1e-5 # physical time-step
# Numerics
CN = 0.5 # Crank-Nicolson CN=0.5, Backward Euler CN=0.0
res = 128
nx, ny = res-1, ra*res-1 # numerical grid resolutions; should be a mulitple of 32-1 for optimal GPU perf
ε = 1e-5 # non-linear tolerance
iterMax = 5e3 # max nonlinear iterations
nout = 200 # error checking frequency
β_n = 1.0 # numerical compressibility
Vdmp = 5.0 # velocity damping for momentum equations
Pfdmp = 0.8 # fluid pressure damping for momentum equations
Vsc = 2.0 # reduction of PT steps for velocity
Ptsc = 2.0 # reduction of PT steps for total pressure
Pfsc = 4.0 # reduction of PT steps for fluid pressure
θ_e = 9e-1 # relaxation factor for non-linear viscosity
θ_k = 1e-1 # relaxation factor for non-linear permeability
dt_red = 1e-3 # reduction of physical timestep
# Derived physics
μs = ηC0*ϕ0/η2μs # solid shear viscosity
λ = λ0*sqrt(k_μf0*ηC0) # initial perturbation width
ρgBG = ρfg*ϕ0 + ρsg*(1.0-ϕ0) # Background density
# Derived numerics
dx, dy = lx/(nx-1), ly/(ny-1) # grid step in x, y
min_dxy2 = min(dx,dy)^2
dτV = min_dxy2/μs/(1.0+β_n)/4.1/Vsc # PT time step for velocity
dτPt = 4.1*μs*(1.0+β_n)/max(nx,ny)/Ptsc
dampX = 1.0-Vdmp/nx
dampY = 1.0-Vdmp/ny
# Array allocations
Phi_o = @zeros(nx ,ny )
Pt = @zeros(nx ,ny )
Pf = @zeros(nx ,ny )
Rog = @zeros(nx ,ny )
∇V = @zeros(nx ,ny )
∇V_o = @zeros(nx ,ny )
∇qD = @zeros(nx ,ny )
dτPf = @zeros(nx ,ny )
RPt = @zeros(nx ,ny )
RPf = @zeros(nx ,ny )
τxx = @zeros(nx ,ny )
τyy = @zeros(nx ,ny )
σxy = @zeros(nx-1,ny-1)
dVxdτ = @zeros(nx-1,ny-2)
dVydτ = @zeros(nx-2,ny-1)
Rx = @zeros(nx-1,ny-2)
Ry = @zeros(nx-2,ny-1)
Vx = @zeros(nx+1,ny )
Vy = @zeros(nx ,ny+1)
qDx = @zeros(nx+1,ny )
# Initial conditions
qDy = zeros(nx ,ny+1)
Phi = ϕ0*ones(nx ,ny )
Radc = zeros(nx ,ny )
Radc .= [(((ix-1)*dx-0.5*lx)/λ/4.0)^2 + (((iy-1)*dy-0.25*ly)/λ)^2 for ix=1:size(Radc,1), iy=1:size(Radc,2)]
Phi[Radc.<1.0] .= Phi[Radc.<1.0] .+ ϕA
EtaC = μs./Phi.*η2μs
K_muf = k_μf0.*(Phi./ϕ0)
ϕ0bc = mean.(Phi[:,end])
qDy[:,[1,end]] .= (ρsg.-ρfg).*(1.0.-ϕ0bc).*k_μf0.*(ϕ0bc./ϕ0).^nperm
Phi = Data.Array(Phi)
EtaC = Data.Array(EtaC)
K_muf = Data.Array(K_muf)
qDy = Data.Array(qDy)
t = 0.0
it = 1
# Preparation of visualisation
ENV["GKSwstype"]="nul"; if isdir("viz2D_out")==false mkdir("viz2D_out") end; loadpath = "./viz2D_out/"; anim = Animation(loadpath,String[])
println("Animation directory: $(anim.dir)")
X, Y, Yv = 0:dx:lx, 0:dy:ly, (-dy/2):dy:(ly+dy/2)
# Time loop
while t<t_tot
@parallel update_old!(Phi_o, ∇V_o, Phi, ∇V)
err=2*ε; iter=1; niter=0
while err > ε && iter <= iterMax
if (iter==11) global wtime0 = Base.time() end
@parallel compute_params_∇!(EtaC, K_muf, Rog, ∇V, ∇qD, Phi, Pf, Pt, Vx, Vy, qDx, qDy, μs, η2μs, R, λPe, k_μf0, ϕ0, nperm, θ_e, θ_k, ρfg, ρsg, ρgBG, dx, dy)
@parallel compute_RP!(dτPf, RPt, RPf, K_muf, ∇V, ∇qD, Pt, Pf, EtaC, Phi, Pfsc, Pfdmp, min_dxy2, dx, dy)
@parallel (1:size(dτPf,1), 1:size(dτPf,2)) bc_x!(dτPf)
@parallel (1:size(dτPf,1), 1:size(dτPf,2)) bc_y!(dτPf)
@parallel compute_P_τ!(Pt, Pf, τxx, τyy, σxy, RPt, RPf, dτPf, Vx, Vy, ∇V, dτPt, μs, β_n, dx, dy)
@parallel compute_res!(Rx, Ry, dVxdτ, dVydτ, τxx, τyy, σxy, Pt, Rog, dampX, dampY, dx, dy)
@parallel compute_update!(Vx, Vy, qDx, qDy, Phi, dVxdτ, dVydτ, K_muf, Pf, Phi_o, ∇V, ∇V_o, dτV, ρfg, ρgBG, CN, dt, dx, dy)
@parallel (1:size(Vx,1), 1:size(Vx,2)) bc_y!(Vx)
@parallel (1:size(Vy,1), 1:size(Vy,2)) bc_x!(Vy)
@parallel (1:size(qDx,1), 1:size(qDx,2)) bc_y!(qDx)
@parallel (1:size(qDy,1), 1:size(qDy,2)) bc_x!(qDy)
if mod(iter,nout)==0
global norm_Ry, norm_RPf
norm_Ry = norm(Ry)/length(Ry); norm_RPf = norm(RPf)/length(RPf); err = max(norm_Ry, norm_RPf)
# @printf("iter = %d, err = %1.3e [norm_Ry=%1.3e, norm_RPf=%1.3e] \n", iter, err, norm_Ry, norm_RPf)
end
iter+=1; niter+=1
end
# Performance
wtime = Base.time()-wtime0
A_eff = (8*2)/1e9*nx*ny*sizeof(Data.Number) # Effective main memory access per iteration [GB] (Lower bound of required memory access: Te has to be read and written: 2 whole-array memaccess; Ci has to be read: : 1 whole-array memaccess)
wtime_it = wtime/(niter-10) # Execution time per iteration [s]
T_eff = A_eff/wtime_it # Effective memory throughput [GB/s]
@printf("it = %d, time = %1.3e sec (@ T_eff = %1.2f GB/s) \n", it, wtime, round(T_eff, sigdigits=2))
# Visualisation
default(size=(500,700))
if mod(it,5)==0
p1 = heatmap(X, Y, Array(Phi)' , aspect_ratio=1, xlims=(X[1],X[end]), ylims=(Y[1],Y[end]), c=:viridis, title="porosity")
p2 = heatmap(X, Y, Array(Pt-Pf)', aspect_ratio=1, xlims=(X[1],X[end]), ylims=(Y[1],Y[end]), c=:viridis, title="effective pressure")
p3 = heatmap(X, Yv, Array(qDy)' , aspect_ratio=1, xlims=(X[1],X[end]), ylims=(Yv[1],Yv[end]), c=:viridis, title="vertical Darcy flux")
p4 = heatmap(X, Yv, Array(Vy)' , aspect_ratio=1, xlims=(X[1],X[end]), ylims=(Yv[1],Yv[end]), c=:viridis, title="vertical velocity")
display(plot(p1, p2, p3, p4)); frame(anim)
end
# Time
dt = dt_red/(1e-10+maximum(abs.(∇V)))
t = t + dt
it+=1
end
gif(anim, "HydroMech2D.gif", fps = 15)
return
end
HydroMech2D()