-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathldc_2d.py
108 lines (90 loc) · 4.13 KB
/
ldc_2d.py
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
from sympy import Symbol, Eq, Abs
from modulus.solver import Solver
from modulus.dataset import TrainDomain, ValidationDomain
from modulus.data import Validation
from modulus.sympy_utils.geometry_2d import Rectangle
from modulus.csv_utils.csv_rw import csv_to_dict
from modulus.PDES import NavierStokes
from modulus.controller import ModulusController
# params for domain
height = 0.1
width = 0.1
vel = 1.0
# define geometry (x y coordinates of bottom left and top right corners)
rec = Rectangle((-width / 2, -height / 2), (width / 2, height / 2))
geo = rec
# visualize geometry by Paraview after saving .vtu point cloud file
# from modulus.plot_utils.vtk import var_to_vtk
# samples = geo.sample_boundary(1000)
# var_to_vtk(samples, './geo')
# define sympy varaibles to parametize domain curves
x, y = Symbol('x'), Symbol('y')
class LDCTrain(TrainDomain):
def __init__(self, **config):
super(LDCTrain, self).__init__()
#top wall
# boundary_bc: sample the boundary
# outvar_sympy: define variables as key in dict
# batch_size_per_area: # points
# lambda_sympy: for point between top and other wall who get sharp discontinue velocity, Signed Distance Field
# criteria: points that y coor == height/2 (top)
topWall = geo.boundary_bc(outvar_sympy={'u': vel, 'v': 0},
batch_size_per_area=10000,
lambda_sympy={'lambda_u': 1.0 - 20 * Abs(x), # weight edges to be zero
'lambda_v': 1.0},
criteria=Eq(y, height / 2))
self.add(topWall, name="TopWall")
# no slip
bottomWall = geo.boundary_bc(outvar_sympy={'u': 0, 'v': 0},
batch_size_per_area=10000,
criteria=y < height / 2)
self.add(bottomWall, name="NoSlip")
# interior
interior = geo.interior_bc(outvar_sympy={'continuity': 0, 'momentum_x': 0, 'momentum_y': 0},
bounds={x: (-width / 2, width / 2),
y: (-height / 2, height / 2)},
lambda_sympy={'lambda_continuity': geo.sdf,
'lambda_momentum_x': geo.sdf,
'lambda_momentum_y': geo.sdf},
batch_size_per_area=400000)
self.add(interior, name="Interior")
# validation data, Data Driven from OpenFOAM
mapping = {'Points:0': 'x', 'Points:1': 'y', 'U:0': 'u', 'U:1': 'v', 'p': 'p'}
openfoam_var = csv_to_dict('openfoam/cavity_uniformVel0.csv', mapping)
openfoam_var['x'] += -width / 2 # center OpenFoam data
openfoam_var['y'] += -height / 2 # center OpenFoam data
openfoam_invar_numpy = {key: value for key, value in openfoam_var.items() if key in ['x', 'y']}
openfoam_outvar_numpy = {key: value for key, value in openfoam_var.items() if key in ['u', 'v']}
class LDCVal(ValidationDomain):
def __init__(self, **config):
super(LDCVal, self).__init__()
val = Validation.from_numpy(openfoam_invar_numpy, openfoam_outvar_numpy)
self.add(val, name='Val')
class LDCSolver(Solver):
train_domain = LDCTrain
val_domain = LDCVal
def __init__(self, **config):
super(LDCSolver, self).__init__(**config)
# solve continuity equation and momentum equations in x, y axis
# self.equations = {}
# set equations in dict
# set_equations['continuity'] = rho.diff(t) + (rho*u).diff(x) + (rho*v).diff(y) + (rho*w).diff(z)
# reference look NS equation
self.equations = NavierStokes(nu=0.01, rho=1.0, dim=2,
time=False).make_node()
# default netowrk 6 hidden layers with 512 nodes
flow_net = self.arch.make_node(name='flow_net',
inputs=['x', 'y'],
outputs=['u', 'v', 'p'])
self.nets = [flow_net]
@classmethod
# used to edit default network setting
def update_defaults(cls, defaults):
defaults.update({
'network_dir': './network_checkpoint_ldc_2d',
'decay_steps': 4000,
'max_steps': 400000
})
if __name__ == '__main__':
ctr = ModulusController(LDCSolver)
ctr.run()