forked from OpFlow-dev/OpFlow
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathLidDriven2D.cpp
99 lines (92 loc) · 4.79 KB
/
LidDriven2D.cpp
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
// Lid-driven flow in 99 lines by Luohao Wang 05.30.2021
#include <OpFlow>
int main(int argc, char* argv[]) {
using namespace OpFlow;
using Mesh = CartesianMesh<Meta::int_<2>>;
using Field = CartesianField<Real, Mesh>;
InitEnvironment(&argc, &argv);// clang-format off
// meshes & fields
auto n = 65;
auto mesh = MeshBuilder<Mesh>().newMesh(n, n).setMeshOfDim(0, 0., 1.).setMeshOfDim(1, 0., 1.).build();
auto builder = ExprBuilder<Field>().setMesh(mesh).setExt(1)
.setBC(0, DimPos::start, BCType::Dirc, 0.).setBC(0, DimPos::end, BCType::Dirc, 0.)
.setBC(1, DimPos::start, BCType::Dirc, 0.).setBC(1, DimPos::end, BCType::Dirc, 0.);
auto u = builder.setName("u").setBC(1, DimPos::end, BCType::Dirc, 1.)
.setLoc({LocOnMesh ::Corner, LocOnMesh ::Center}).build();
auto du = builder.setName("du").setBC(1, DimPos::end, BCType::Dirc, 0.).build();
auto v = builder.setName("v").setBC(1, DimPos::end, BCType::Dirc, 0.)
.setLoc({LocOnMesh ::Center, LocOnMesh ::Corner}).build();
auto dv = v; dv.name = "dv";
auto p = builder.setName("p")
.setBC(0, DimPos::start, BCType::Neum, 0.).setBC(0, DimPos::end, BCType::Neum, 0.)
.setBC(1, DimPos::start, BCType::Neum, 0.).setBC(1, DimPos::end, BCType::Neum, 0.)
.setLoc({LocOnMesh ::Center, LocOnMesh ::Center}).build();
auto dp = p; dp.name = "dp";
u = 0; du = 0; v = 0; dv = 0; p = 0; dp = 0;
// composite operators
auto conv_xx = [&](auto&& _1, auto&& _2) {
return dx<D1FirstOrderCentered>(d1IntpCornerToCenter<0>(_1) * d1IntpCornerToCenter<0>(_2));
};
auto conv_xy = [&](auto&& _1, auto&& _2) {
return dy<D1FirstOrderCentered>(d1IntpCenterToCorner<1>(_1) * d1IntpCenterToCorner<0>(_2));
};
auto conv_yx = [&](auto&& _1, auto&& _2) {
return dx<D1FirstOrderCentered>(d1IntpCenterToCorner<1>(_1) * d1IntpCenterToCorner<0>(_2));
};
auto conv_yy = [&](auto&& _1, auto&& _2) {
return dy<D1FirstOrderCentered>(d1IntpCornerToCenter<1>(_1) * d1IntpCornerToCenter<1>(_2));
};
// solvers
const Real dt = 0.5e-2, nu = 1.0e-2;
StructSolverParams<StructSolverType::GMRES> params; params.tol = 1e-10; params.maxIter = 100;
StructSolverParams<StructSolverType::GMRES> poisson_params = params;
StructSolverParams<StructSolverType::PFMG> p_params {.useZeroGuess = true, .relaxType = 1, .rapType = 0,
.numPreRelax = 1, .numPostRelax = 1, .skipRelax = 0}; p_params.tol = 1e-10;
auto solver = PrecondStructSolver<StructSolverType::GMRES, StructSolverType::PFMG>(params, p_params);
auto u_handler = makeEqnSolveHandler(
[&](auto&& e) {
return e / dt + conv_xx(u, e) + 0.5 * conv_xy(e, v)
== nu * (d2x<D2SecondOrderCentered>(u) + d2y<D2SecondOrderCentered>(u))
+ 0.5 * nu * (d2x<D2SecondOrderCentered>(e) + d2y<D2SecondOrderCentered>(e))
- (conv_xx(u, u) + conv_xy(u, v)) - dx<D1FirstOrderCentered>(p);
},
du, solver);
auto v_handler = makeEqnSolveHandler(
[&](auto&& e) {
return e / dt + conv_yy(v, e) + conv_yy(v, v) + conv_yx(u, v) + 0.5 * conv_yx(u, e)
+ 0.5 * conv_yx(du, v)
== nu * (d2x<D2SecondOrderCentered>(v) + d2y<D2SecondOrderCentered>(v))
+ 0.5 * nu * (d2x<D2SecondOrderCentered>(e) + d2y<D2SecondOrderCentered>(e))
- dy<D1FirstOrderCentered>(p);
},
dv, solver);
poisson_params.staticMat = true; poisson_params.pinValue = true;
auto p_solver = PrecondStructSolver<StructSolverType::GMRES, StructSolverType::PFMG>(poisson_params, p_params);
auto p_handler = makeEqnSolveHandler(
[&](auto&& e) {
return d2x<D2SecondOrderCentered>(e) + d2y<D2SecondOrderCentered>(e)
== (dx<D1FirstOrderCentered>(du) + dy<D1FirstOrderCentered>(dv)) / dt;
},
dp, p_solver);
// writers
Utils::TecplotASCIIStream uf("u.tec"), vf("v.tec"), pf("p.tec");
// main algorithm
auto t0 = std::chrono::system_clock::now();
for (auto i = 0; i < 1000; ++i) {
u_handler->solve();
v_handler->solve();
du = du - 0.5 * dt * conv_xy(u, dv);
u = u + du;
v = v + dv;
p_handler->solve();
u = u - dt * dx<D1FirstOrderCentered>(dp);
v = v - dt * dy<D1FirstOrderCentered>(dp);
p = p + dp;
uf << Utils::TimeStamp(i) << u; vf << Utils::TimeStamp(i) << v; pf << Utils::TimeStamp(i) << p;
OP_INFO("Current step {}", i);
}
auto t1 = std::chrono::system_clock::now();
OP_INFO("Elapsed time: {}ms", std::chrono::duration_cast<std::chrono::milliseconds>(t1 - t0).count());
FinalizeEnvironment();
return 0;
}