Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Navier-Stokes in 2D on DGMulti and TreeMesh #1165

Merged
merged 78 commits into from
Jul 29, 2022
Merged
Show file tree
Hide file tree
Changes from 11 commits
Commits
Show all changes
78 commits
Select commit Hold shift + click to select a range
60cce54
first draft of container for Navier-Stokes constants and fluxes
andrewwinters5000 Jun 7, 2022
46ae5fb
remove unneeded temperature computation
andrewwinters5000 Jun 7, 2022
4f6558f
draft of elixir with possible boundary condition structure
andrewwinters5000 Jun 7, 2022
dde54d3
added manufactured solution and source term
andrewwinters5000 Jun 7, 2022
daaeac3
fix typo in function name for MMS
andrewwinters5000 Jun 7, 2022
4a39883
update variable names for consistency. improve comments
andrewwinters5000 Jun 8, 2022
84d8454
Merge branch 'main' into nav_stokes_2d
andrewwinters5000 Jun 8, 2022
3879a3f
fix dumb typos in new equation system name
andrewwinters5000 Jun 8, 2022
435e081
actually export new equations
andrewwinters5000 Jun 8, 2022
7b59fef
add comment near variable_mapping declaration.
andrewwinters5000 Jun 8, 2022
d3b7870
Merge branch 'parabolic-treemesh' into nav_stokes_2d
ranocha Jun 8, 2022
5dc8301
Apply suggestions from code review
andrewwinters5000 Jun 9, 2022
f316684
parabolic equations now exported separately
andrewwinters5000 Jun 9, 2022
02c14e1
change name to CompressibleNavierStokesEquations2D
andrewwinters5000 Jun 9, 2022
dd733bb
export NS with proper name
andrewwinters5000 Jun 9, 2022
a1a48fe
explicitly require compressible Euler in the type parameter
andrewwinters5000 Jun 10, 2022
d8db203
name kinematic viscosity nu for consistency with Lattice-Boltzmann
andrewwinters5000 Jun 10, 2022
06e69b8
add promotion in constructor
andrewwinters5000 Jun 10, 2022
eda9737
make Reynolds, Prandtl, Mach, and kappa keyword arguments
andrewwinters5000 Jun 10, 2022
b469bce
update constructor call in elixir
andrewwinters5000 Jun 10, 2022
fb646b0
reduce computation by exploiting stress tensor symmetry
andrewwinters5000 Jun 10, 2022
f32ef8c
Merge remote-tracking branch 'origin/main' into nav_stokes_2d
jlchan Jun 14, 2022
f6bca76
fix unpacking of flux
jlchan Jun 14, 2022
9b5dd35
modifying parabolic cache creation
jlchan Jun 14, 2022
21e1c99
comments
jlchan Jun 15, 2022
19f004e
comments
jlchan Jun 15, 2022
de442bf
formatting and renaming equations to equations_hyperbolic
jlchan Jun 15, 2022
298b726
fix unpacking of gradients in flux
jlchan Jun 15, 2022
01f8325
adding CNS BCs
jlchan Jun 15, 2022
3f04e71
adding lid-driven cavity elixir
jlchan Jun 15, 2022
36e3b68
adding variable transform, editing cons2prim for CNS
jlchan Jun 15, 2022
6bf91eb
add prim2cons for NS (inconsistent right now)
jlchan Jun 15, 2022
a0ea3f0
add draft of DGMulti Navier Stokes convergence elixir
jlchan Jun 15, 2022
c103e2e
converging solution using elixir for TreeMesh with BCs
jlchan Jun 27, 2022
845c133
Merge branch 'main' into nav_stokes_2d
jlchan Jun 27, 2022
3d717a6
fixing DGMulti advection diffusion elixir convergence
jlchan Jun 27, 2022
8471273
naming equations as parabolic/hyperbolic
jlchan Jun 27, 2022
96667e6
generalizing transform_variables
jlchan Jun 27, 2022
73e4740
add TODO
jlchan Jun 27, 2022
2c05f66
additional checks on get_unsigned_normal
jlchan Jun 27, 2022
d5e7f9e
adding isothermal BC
jlchan Jun 27, 2022
89acae8
commenting out unused CNS functions for now
jlchan Jun 27, 2022
f503b85
fix call to transform_variables
jlchan Jun 27, 2022
41b41a9
comments and cleanup
jlchan Jun 27, 2022
1107def
changing default solver and Re for cavity
jlchan Jun 27, 2022
f90afb9
adding more advection diffusion tests
jlchan Jun 27, 2022
2e7019f
label tests
jlchan Jun 28, 2022
063b602
add gradient_variables field to SemidiscretizationHyperbolicParabolic
jlchan Jun 28, 2022
50d76df
Revert "add gradient_variables field to SemidiscretizationHyperbolicP…
jlchan Jun 28, 2022
5b112e9
Merge branch 'main' into nav_stokes_2d
jlchan Jul 11, 2022
549793c
allowing for specialization in transform_variables
jlchan Jul 12, 2022
3e3e41f
formatting and comments
jlchan Jul 13, 2022
6347151
reverting elixir
jlchan Jul 13, 2022
eb98568
comments
jlchan Jul 13, 2022
766cdc5
standardizing time tol
jlchan Jul 13, 2022
c865fb7
minor fix to CNS boundary flux for convenience
jlchan Jul 13, 2022
08768a7
formatting + comments
jlchan Jul 13, 2022
6c38bdb
using primitive variables in viscous flux instead of conservative
jlchan Jul 13, 2022
0f67dd0
minor formatting
jlchan Jul 13, 2022
6a1d908
add CNS tests
jlchan Jul 13, 2022
6d5be6c
fix test
jlchan Jul 13, 2022
bc58754
testing if scoping issues fix TreeMesh tests
jlchan Jul 13, 2022
f13045a
decreasing timestep tol for TreeMesh parabolic test
jlchan Jul 13, 2022
d29d583
enabling periodic BCs for TreeMesh + more tests
jlchan Jul 13, 2022
abd020b
fix density BC flux (unused, but could be useful)
jlchan Jul 13, 2022
663f465
adding non-working TreeMesh elixirs
jlchan Jul 15, 2022
5ae580b
adding AD checks
jlchan Jul 15, 2022
e0c3b84
Merge remote-tracking branch 'origin/main' into nav_stokes_2d
jlchan Jul 28, 2022
bfffc00
standardizing parameters in convergence elixirs
jlchan Jul 28, 2022
17eafbb
minor cleanup
jlchan Jul 28, 2022
0c4e098
revert DGMulti CNS elixir setup back to the one used in tests
jlchan Jul 28, 2022
b70bbc0
adding TreeMesh CNS convergence test
jlchan Jul 28, 2022
13e0c8f
removing redundant elixir
jlchan Jul 28, 2022
8cecb5f
add more tests
jlchan Jul 28, 2022
adb8eca
add more test
jlchan Jul 28, 2022
3281b49
Apply suggestions from code review
andrewwinters5000 Jul 29, 2022
98ea1bd
add docstrings
jlchan Jul 29, 2022
bf43508
Merge remote-tracking branch 'origin/parabolic-treemesh' into nav_sto…
ranocha Jul 29, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
272 changes: 272 additions & 0 deletions examples/tree_2d_dgsem/elixir_navier_stokes_source_terms.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,272 @@
using OrdinaryDiffEq
using Trixi

###############################################################################
# semidiscretization of the linear advection-diffusion equation
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved

equations = CompressibleEulerEquations2D(1.4)
# Note: If you change the Navier-Stokes parameters here, also change them in the initial condition
# I really do not like this structure but it should work for now
equations_parabolic = CompressibleNavierStokes2D(1.4, # gamma
1000, # Reynolds number
0.72, # Prandtl number
0.5, # free-stream Mach number
1.0, # thermal diffusivity
equations)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we need all of these parameters? If so, should we maybe create a constructor with keywords?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

  1. I am not sure if we need all these constants. For example, the Prandtl number basically never changes för the ideal gas assumption. I threw in all the constants I thought we would need and am open to trimming it down. Also, we could possibly store mu = 1 / Reynolds for the sake of code readability.
  2. Keywords is probably the best way to do this but I wasn't sure if having the constructor look like
function CompressibleNavierStokes2D(equations; gamma, Reynolds, Prandtl, Mach_freestream, kappa)

would be kosher to have the call structure

equations_parabolic = CompressibleNavierStokes2D(gamma=1.4,
                                                 Reynolds=1000,
                                                 Prandtl=0.72,
                                                 Mach_freestream=0.5,
                                                 kappa=1.0,  # thermal diffusivity
                                                 equations)

for the sake of consistency with the LaplaceEquation2D where equations comes last.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Keyword arguments need to come after positional ones, so it could be

CompressibleNavierStokes2D(equations; Reynolds, Prandtl, Mach_freestream, kappa)

where we should grab gamma from equations.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@sloede We should probably unify the argument names with the lattice Boltzmann stuff (using Ma and Re for now).

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done


# Create DG solver with polynomial degree = 3 and (local) Lax-Friedrichs/Rusanov flux as surface flux
solver = DGSEM(polydeg=3, surface_flux=flux_lax_friedrichs)

coordinates_min = (-1.0, -1.0) # minimum coordinates (min(x), min(y))
coordinates_max = ( 1.0, 1.0) # maximum coordinates (max(x), max(y))

# Create a uniformly refined mesh with periodic boundaries
mesh = TreeMesh(coordinates_min, coordinates_max,
initial_refinement_level=4,
n_cells_max=30_000) # set maximum capacity of tree data structure

# Define initial condition
# Note: If you change the parameters here, also change it in the corresponding source terms
function initial_condition_navier_stokes_convergence_test(x, t, equation::CompressibleEulerEquations2D)

# Amplitude and shift
A = 0.5
c = 2.0

# convenience values for trig. functions
pi_x = pi * x[1]
pi_y = pi * x[2]
pi_t = pi * t

rho = c + A * sin(pi_x) * cos(pi_y) * cos(pi_t)
v1 = sin(pi_x) * log(y + 2.0) * (1.0 - exp(-A * (y - 1.0)) ) * cos(pi_t)
v2 = v1
p = rho^2

return prim2cons(SVector(rho, v1, v2, p), equations)
end


initial_condition = initial_condition_navier_stokes_convergence_test


@inline function source_terms_navier_stokes_convergence_test(u, x, t,
equations::CompressibleNavierStokes2D)
# Same settings as in `initial_condition`
# Amplitude and shift
A = 0.5
c = 2.0

# convenience values for trig. functions
pi_x = pi * x[1]
pi_y = pi * x[2]
pi_t = pi * t

# compute the manufactured solution and all necessary derivatives
rho = c + A * sin(pi_x) * cos(pi_y) * cos(pi_t)
rho_t = -pi * A * sin(pi_x) * cos(pi_y) * sin(pi_t)
rho_x = pi * A * cos(pi_x) * cos(pi_y) * cos(pi_t)
rho_y = -pi * A * sin(pi_x) * sin(pi_y) * cos(pi_t)
rho_xx = -pi * pi * A * sin(pi_x) * cos(pi_y) * cos(pi_t)
rho_yy = -pi * pi * A * sin(pi_x) * cos(pi_y) * cos(pi_t)

v1 = sin(pi_x) * log(y + 2.0) * (1.0 - exp(-A * (y - 1.0))) * cos(pi_t)
v1_t = -pi * sin(pi_x) * log(y + 2.0) * (1.0 - exp(-A * (y - 1.0))) * sin(pi_t)
v1_x = pi * cos(pi_x) * log(y + 2.0) * (1.0 - exp(-A * (y - 1.0))) * cos(pi_t)
v1_y = sin(pi_x) * (A * log(y + 2.0) * exp(-A * (y - 1.0)) + (1.0 - exp(-A * (y - 1.0))) / (y + 2.0)) * cos(pi_t)
v1_xx = -pi * pi * sin(pi_x) * log(y + 2.0) * (1.0 - exp(-A * (y - 1.0))) * cos(pi_t)
v1_xy = pi * cos(pi_x) * (A * log(y + 2.0) * exp(-A * (y - 1.0)) + (1.0 - exp(-A * (y - 1.0))) / (y + 2.0)) * cos(pi_t)
v1_yy = (sin(pi_x) * ( 2.0 * A * exp(-A * (y - 1.0)) / (y + 2.0)
- A * A * log(y + 2.0) * exp(-A * (y - 1.0))
- (1.0 - exp(-A * (y - 1.0))) / ((y + 2.0) * (y + 2.0))) * cos(pi_t))
v2 = v1
v2_t = v1_t
v2_x = v1_x
v2_y = v1_y
v2_xx = v1_xx
v2_xy = v1_xy
v2_yy = v1_yy

p = rho * rho
p_t = 2.0 * rho * rho_t
p_x = 2.0 * rho * rho_x
p_y = 2.0 * rho * rho_y
p_xx = 2.0 * rho * rho_xx + 2.0 * rho_x * rho_x
p_yy = 2.0 * rho * rho_yy + 2.0 * rho_y * rho_y

# Note this simplifies slightly because the ansatz assumes that v1 = v2
E = p * equations.inv_gamma_minus_one + 0.5 * rho * (v1^2 + v2^2)
E_t = p_t * equations.inv_gamma_minus_one + rho_t * v1^2 + 2.0 * rho * v1 * v1_t
E_x = p_x * equations.inv_gamma_minus_one + rho_x * v1^2 + 2.0 * rho * v1 * v1_x
E_y = p_y * equations.inv_gamma_minus_one + rho_y * v1^2 + 2.0 * rho * v1 * v1_y

# Some convenience constants
T_const = equations.gamma * equations.inv_gamma_minus_one * equations.kappa / equations.Pr
inv_Re = 1.0 / equations.Re
inv_rho_cubed = 1.0 / (rho^3)

# compute the source terms
# density equation
du1 = rho_t + rho_x * v1 + rho * v1_x + rho_y * v2 + rho * v2_y

# x-momentum equation
du2 = ( rho_t * v1 + rho * v1_t + p_x + rho_x * v1^2
+ 2.0 * rho * v1 * v1_x
+ rho_y * v1 * v2
+ rho * v1_y * v2
+ rho * v1 * v2_y
# stress tensor from x-direction
- 4.0 / 3.0 * v1_xx * inv_Re
+ 2.0 / 3.0 * v2_xy * inv_Re
- v1_yy * inv_Re
- v2_xy * inv_Re )
# y-momentum equation
du3 = ( rho_t * v2 + rho * v2_t + p_y + rho_x * v1 * v2
+ rho * v1_x * v2
+ rho * v1 * v2_x
+ rho_y * v2^2
+ 2.0 * rho * v2 * v2_y
# stress tensor from y-direction
- v1_xy * inv_Re
- v2_xx * inv_Re
- 4.0 / 3.0 * v2_yy * inv_Re
+ 2.0 / 3.0 * v1_xy * inv_Re )
# total energy equation
du4 = ( E_t + v1_x * (E + p) + v1 * (E_x + p_x)
+ v2_y * (E + p) + v2 * (E_y + p_y)
# stress tensor and temperature gradient terms from x-direction
- 4.0 / 3.0 * v1_xx * v1 * inv_Re
+ 2.0 / 3.0 * v2_xy * v1 * inv_Re
- 4.0 / 3.0 * v1_x * v1_x * inv_Re
+ 2.0 / 3.0 * v2_y * v1_x * inv_Re
- v1_xy * v2 * inv_Re
- v2_xx * v2 * inv_Re
- v1_y * v2_x * inv_Re
- v2_x * v2_x * inv_Re
- T_const * inv_rho_cubed * ( p_xx * rho * rho
- 2.0 * p_x * rho * rho_x
+ 2.0 * p * rho_x * rho_x
- p * rho * rho_xx ) * inv_Re
# stress tensor and temperature gradient terms from y-direction
- v1_yy * v1 * inv_Re
- v2_xy * v1 * inv_Re
- v1_y * v1_y * inv_Re
- v2_x * v1_y * inv_Re
- 4.0 / 3.0 * v2_yy * v2 * inv_Re
+ 2.0 / 3.0 * v1_xy * v2 * inv_Re
- 4.0 / 3.0 * v2_y * v2_y * inv_Re
+ 2.0 / 3.0 * v1_x * v2_y * inv_Re
- T_const * inv_rho_cubed * ( p_yy * rho * rho
- 2.0 * p_y * rho * rho_y
+ 2.0 * p * rho_y * rho_y
- p * rho * rho_yy ) * inv_Re )

return SVector(du1, du2, du3, du4)
end


# TODO: Wasn't sure of the call structure, but this should be what we need
function boundary_condition_no_slip_adiabatic_wall(u_inner, orientation, direction,
x, t, surface_flux_function,
equations::CompressibleEulerEquations2D)
u_boundary = SVector(u_inner[1], -u_inner[2], -u_inner[3], u_inner[4])

# Calculate boundary flux
if iseven(direction) # u_inner is "left" of boundary, u_boundary is "right" of boundary
flux = surface_flux_function(u_inner, u_boundary, orientation, equations)
else # u_boundary is "left" of boundary, u_inner is "right" of boundary
flux = surface_flux_function(u_boundary, u_inner, orientation, equations)
end

return flux
end


# TODO: Wasn't sure of the call structure, but this should be what we need
function boundary_condition_no_slip_adiabatic_wall_neumann(grad_u_inner, orientation, direction,
x, t, surface_flux_function,
equations::CompressibleNavierStokes2D)
# Copy the inner gradients to an external state array
grad_u_boundary .= grad_u_inner
jlchan marked this conversation as resolved.
Show resolved Hide resolved

# Project out the appropriate temperature gradient pieces. Wasn't sure of array shape
if orientation == 1
grad_u_norm = grad_u[1,3] # temperature gradient in x-direction
u_x_tangent = grad_u[1,3] - grad_u_norm
u_y_tangent = grad_u[2,3]

# Update the external state gradients
grad_u_boundary[1,3] = u_x_tangent - grad_u_norm
grad_u_boundary[2,3] = u_y_tangent
else # orientation == 2
grad_u_norm = grad_u[2,3] # temperature gradient in y-direction
u_x_tangent = grad_u[1,3]
u_y_tangent = grad_u[2,3] - grad_u_norm

# Update the external state gradients
grad_u_boundary[1,3] = u_x_tangent
grad_u_boundary[2,3] = u_y_tangent - grad_u_norm
end

# Calculate boundary flux (just averages so has no orientation I think)
flux = surface_flux_function(grad_u_inner, grad_u_boundary, orientation, equations)

return flux
end


# Below is my best guess as to how to set periodic in x direction and walls
# in the y direcitons
boundary_condition= (
x_neg=boundary_condition_periodic,
x_pos=boundary_condition_periodic,
y_neg=boundary_condition_no_slip_adiabatic_wall,
y_pos=boundary_condition_no_slip_adiabatic_wall,
)
boundary_conditions_parabolic = (
x_neg=boundary_condition_periodic,
x_pos=boundary_condition_periodic,
y_neg=boundary_condition_no_slip_adiabatic_wall_neumann,
y_pos=boundary_condition_no_slip_adiabatic_wall_neumann,
)

# A semidiscretization collects data structures and functions for the spatial discretization
semi = SemidiscretizationHyperbolicParabolic(mesh,
(equations, equations_parabolic),
initial_condition, solver;
boundary_conditions=(boundary_conditions,
boundary_conditions_parabolic))


###############################################################################
# ODE solvers, callbacks etc.

# Create ODE problem with time span from 0.0 to 1.5
tspan = (0.0, 1.5)
ode = semidiscretize(semi, tspan);

# At the beginning of the main loop, the SummaryCallback prints a summary of the simulation setup
# and resets the timers
summary_callback = SummaryCallback()

# The AnalysisCallback allows to analyse the solution in regular intervals and prints the results
analysis_interval = 100
analysis_callback = AnalysisCallback(semi, interval=analysis_interval)

# The AliveCallback prints short status information in regular intervals
alive_callback = AliveCallback(analysis_interval=analysis_interval)

# Create a CallbackSet to collect all callbacks such that they can be passed to the ODE solver
callbacks = CallbackSet(summary_callback, analysis_callback, alive_callback)


###############################################################################
# run the simulation

# OrdinaryDiffEq's `solve` method evolves the solution in time and executes the passed callbacks
time_int_tol = 1.0e-8
sol = solve(ode, RDPK3SpFSAL49(), abstol=time_int_tol, reltol=time_int_tol,
save_everystep=false, callback=callbacks)

# Print the timer summary
summary_callback()
1 change: 1 addition & 0 deletions src/Trixi.jl
Original file line number Diff line number Diff line change
Expand Up @@ -133,6 +133,7 @@ export AcousticPerturbationEquations2D,
InviscidBurgersEquation1D,
LaplaceDiffusion2D,
LatticeBoltzmannEquations2D, LatticeBoltzmannEquations3D,
CompressibleNavierStokes2D,
ranocha marked this conversation as resolved.
Show resolved Hide resolved
ShallowWaterEquations1D, ShallowWaterEquations2D

export flux, flux_central, flux_lax_friedrichs, flux_hll, flux_hllc, flux_hlle, flux_godunov,
Expand Down
Loading