-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathMODULE_SOLVER.f90
238 lines (202 loc) · 9.09 KB
/
MODULE_SOLVER.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
!=================================================================================================
!> CARTESIAN ADAPTIVE FIVE EQUATION MODEL
!> AUTHOR: VAN-DAT THANG
!> E-MAIL: [email protected]
!> E-MAIL: [email protected]
!> SOURCE CODE LINK: https://github.com/dattv/2D_CARFIVE
!=================================================================================================
MODULE MODULE_SOLVER
use MODULE_PRECISION
use MODULE_QUADTREE
use MODULE_EXSOLVER
use MODULE_CFDMAINDATA
use MODULE_RESIDUAL
use MODULE_OUTPUT
!================== DEGINS PATTERN ==================
type, abstract :: abstract_solver
contains
procedure(generic_solver), deferred, pass(this) :: p_solver
end type abstract_solver
abstract interface
subroutine generic_solver(this, first, last, tree, dt)
use MODULE_PRECISION
use MODULE_QUADTREE
import :: abstract_solver
class(abstract_solver), intent(in) :: this
integer(ip), intent(in) :: first, last
type(quadtree), dimension(first:last), intent(inout) :: tree
real(rp), intent(out) :: dt
end subroutine generic_solver
end interface
type :: solver_factory
private
character(len = 20) :: c_solver_type
integer(ip) :: i_solver_type
class(abstract_solver), pointer :: solver
contains
procedure :: create_solver
end type solver_factory
type, extends(abstract_solver) :: first_order_explicit_solver
contains
procedure, pass(this) :: p_solver => first_EX_solver
end type first_order_explicit_solver
type, extends(abstract_solver) :: second_order_explicit_solver
contains
procedure, pass(this) :: p_solver => second_EX_solver
end type second_order_explicit_solver
type, extends(abstract_solver) :: runge_kutta_explicit_solver
contains
procedure, pass(this) :: p_solver => RK_EX_solver
end type runge_kutta_explicit_solver
type, extends(abstract_solver) :: second_order_implicit_solver
contains
procedure, pass(this) :: p_solver => second_IM_solver
end type second_order_implicit_solver
type, extends(abstract_solver) :: implicit_Explicit_solver
contains
procedure, pass(this) :: p_solver => IMEX_solver
end type implicit_Explicit_solver
!=========================== END OF FACTORY PATTERN ==========================
contains
!==================================================================================================
function create_solver(this, I_solver_type) result(ptr)
implicit none
class(solver_factory) :: this
class(abstract_solver), pointer :: ptr
integer(ip), intent(in) :: I_solver_type
if (I_solver_type == 1) then
! ===> first order explicit type <===
allocate(first_order_explicit_solver :: this%solver)
this%I_solver_type = 1
this%C_solver_type = "1st explicit"
ptr => this%solver
else if (I_solver_type == 2) then
! ===> second order explicit type <===
allocate(second_order_explicit_solver :: this%solver)
this%I_solver_type = 2
this%C_solver_type = "2st explicit"
ptr => this%solver
else if (I_solver_type == 3) then
! ===> runge kutta explicit type <===
allocate(runge_kutta_explicit_solver :: this%solver)
this%I_solver_type = 3
this%C_solver_type = "rk explicit"
ptr => this%solver
else if (I_solver_type == 4) then
! ===> second order implicit type <===
allocate(second_order_implicit_solver :: this%solver)
this%I_solver_type = 4
this%C_solver_type = "2st Implicit"
ptr => this%solver
else if (I_solver_type == 5) then
! ===> implicit explicit type <===
allocate(implicit_Explicit_solver :: this%solver)
this%I_solver_type = 5
this%C_solver_type = "IMEX"
ptr => this%solver
else
print*, "program have not supported this solver yet"
print*, "Stop"
pause
stop
end if
return
end function create_solver
!==================================================================================================
subroutine first_EX_solver(this, first, last, tree, dt)
implicit none
class(first_order_explicit_solver), intent(in) :: this
integer(ip), intent(in) :: first, last
type(quadtree), dimension(first:last), intent(inout) :: tree
real(rp), intent(out) :: dt
dt = zero
return
end subroutine first_EX_solver
!==================================================================================================
subroutine second_EX_solver(this, first, last, tree, dt)
implicit none
class(second_order_explicit_solver), intent(in) :: this
integer(ip), intent(in) :: first, last
type(quadtree), dimension(first:last), intent(inout) :: tree
real(rp), intent(out) :: dt
! ===> this function will re-direct to another module,
! ===> i made it to reduce this file's size
call second_EX(first, last, tree, dt)
return
end subroutine second_EX_solver
!==================================================================================================
subroutine RK_EX_solver(this, first, last, tree, dt)
implicit none
class(runge_kutta_explicit_solver), intent(in) :: this
integer(ip), intent(in) :: first, last
type(quadtree), dimension(first:last), intent(inout) :: tree
real(rp), intent(out) :: dt
dt = zero
return
end subroutine RK_EX_solver
!==================================================================================================
subroutine second_IM_solver(this, first, last, tree, dt)
implicit none
class(second_order_implicit_solver), intent(in) :: this
integer(ip), intent(in) :: first, last
type(quadtree), dimension(first:last), intent(inout) :: tree
real(rp), intent(out) :: dt
dt = zero
return
end subroutine second_IM_solver
!==================================================================================================
subroutine IMEX_solver(this, first, last, tree, dt)
implicit none
class(implicit_Explicit_solver), intent(in) :: this
integer(ip), intent(in) :: first, last
type(quadtree), dimension(first:last), intent(inout) :: tree
real(rp), intent(out) :: dt
dt = zero
return
end subroutine IMEX_solver
!==================================================================================================
subroutine solving(first, last, tree, I_solver_type)
implicit none
integer(ip), intent(in) :: first, last
class(quadtree), dimension(first:last), intent(inout) :: tree
integer(ip), intent(in) :: I_solver_type
integer(ip) :: i, iter
logical :: l_stop = .false.
real(rp) :: dt
real(rp), dimension(tree(first)%data%NQ,3) :: res_norm
character(len = 100) :: output, temp_out
type(solver_factory) :: factory
class(abstract_solver), pointer :: solver => null()
! ===> create solver with I_solver_type <===
solver => factory%create_solver(I_solver_type)
! ===> call solver <===
time = zero; iter = 0
do while (.not. l_stop)
iter = iter + 1
! ===> call solver
call solver%p_solver(first, last, tree, dt)
! ===> compute residual at each step
call residual_norm(first, last, tree, res_norm)
! ===> print out residual each step
if (iter == 1) write(*, 10)
if (mod(iter, frequency_dump) == 0) then
write(*, 20) time, iter, res_norm(:,1)
end if
! ===> check time stop
if (time >= t_final) l_stop = .true.
time = time + dt
! ===> print out result to data file
if (mod(iter, frequency_dump) == 0) then
output = "OUTPUT"
write(temp_out, "(10i)") iter
temp_out = adjustl(temp_out)
output = trim(temp_out)//"_"//trim(output)//".TEC"
call output_2D(output, last, tree)
end if
end do
return
10 format(47x,"Density1 Density2 x_momentum y_momentum Pressure A1")
20 format("Time =",f20.13, "steps =", i4, 6f20.13)
end subroutine solving
!==================================================================================================
END MODULE MODULE_SOLVER