From 7676d35c97393cd87bd59d75da1231b4a824ea80 Mon Sep 17 00:00:00 2001 From: Daniel_Doehring Date: Thu, 11 Apr 2024 15:11:41 +0200 Subject: [PATCH 01/12] LinearizedEuler3D --- .../elixir_linearizedeuler_convergence.jl | 64 +++++ .../elixir_linearizedeuler_gauss_wall.jl | 66 +++++ src/Trixi.jl | 2 +- src/equations/equations.jl | 1 + src/equations/linearized_euler_1d.jl | 2 +- src/equations/linearized_euler_2d.jl | 2 +- src/equations/linearized_euler_3d.jl | 255 ++++++++++++++++++ test/test_p4est_3d.jl | 22 ++ test/test_tree_3d_linearizedeuler.jl | 35 +++ test/test_tree_3d_part2.jl | 3 + 10 files changed, 449 insertions(+), 3 deletions(-) create mode 100644 examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl create mode 100644 examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl create mode 100644 src/equations/linearized_euler_3d.jl create mode 100644 test/test_tree_3d_linearizedeuler.jl diff --git a/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl b/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl new file mode 100644 index 00000000000..608949f1546 --- /dev/null +++ b/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl @@ -0,0 +1,64 @@ +using OrdinaryDiffEq +using Trixi + +############################################################################### +# semidiscretization of the linearized Euler equations + +equations = LinearizedEulerEquations3D(v_mean_global = (0.0, 0.0, 0.0), c_mean_global = 1.0, + rho_mean_global = 1.0) + +initial_condition = initial_condition_convergence_test + +solver = DGSEM(polydeg = 3, surface_flux = flux_hll) + +coordinates_min = (-1.0, -1.0, -1.0) # minimum coordinates (min(x), min(y), min(z)) +coordinates_max = (1.0, 1.0, 1.0) # maximum coordinates (max(x), max(y), max(z)) + +# `initial_refinement_level` is provided here to allow for a +# convenient convergence test, see +# https://trixi-framework.github.io/Trixi.jl/stable/#Performing-a-convergence-analysis +trees_per_dimension = (4, 4, 4) +mesh = P4estMesh(trees_per_dimension, polydeg = 3, + coordinates_min = coordinates_min, + coordinates_max = coordinates_max, + initial_refinement_level = 0) + +# A semidiscretization collects data structures and functions for the spatial discretization +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver) + +############################################################################### +# ODE solvers, callbacks etc. + +# Create ODE problem with time span from 0.0 to 0.2 +tspan = (0.0, 0.2) +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() + +analysis_interval = 100 + +# The AnalysisCallback allows to analyse the solution in regular intervals and prints the results +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) + +# The AliveCallback prints short status information in regular intervals +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +# The StepsizeCallback handles the re-calculation of the maximum Δt after each time step +stepsize_callback = StepsizeCallback(cfl = 0.8) + +# 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, + stepsize_callback) + +############################################################################### +# run the simulation + +# OrdinaryDiffEq's `solve` method evolves the solution in time and executes the passed callbacks +sol = solve(ode, CarpenterKennedy2N54(williamson_condition = false), + dt = 1.0, # solve needs some value here but it will be overwritten by the stepsize_callback + save_everystep = false, callback = callbacks); + +# print the timer summary +summary_callback() # print the timer summary diff --git a/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl b/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl new file mode 100644 index 00000000000..bba868cbe51 --- /dev/null +++ b/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl @@ -0,0 +1,66 @@ + +using OrdinaryDiffEq +using Trixi + +############################################################################### +# semidiscretization of the linearized Euler equations + +equations = LinearizedEulerEquations3D(v_mean_global = (0.5, 0.5, 0.5), c_mean_global = 1.0, + rho_mean_global = 1.0) + +solver = DGSEM(polydeg = 5, surface_flux = flux_lax_friedrichs) + +coordinates_min = (0.0, 0.0, 0.0) +coordinates_max = (90.0, 90.0, 90.0) + +mesh = TreeMesh(coordinates_min, coordinates_max, + initial_refinement_level = 4, + n_cells_max = 100_000, + periodicity = false) + +# Initialize density and pressure perturbation with a Gaussian bump +# that splits and is advected with v - c and v + c. +function initial_condition_gauss_wall(x, t, equations::LinearizedEulerEquations3D) + v1_prime = 0.0 + v2_prime = 0.0 + v3_prime = 0.0 + rho_prime = p_prime = 2 * exp(-(x[1] - 45)^2 / 25) + return SVector(rho_prime, v1_prime, v2_prime, v3_prime, p_prime) +end +initial_condition = initial_condition_gauss_wall + +# A semidiscretization collects data structures and functions for the spatial discretization +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver, + boundary_conditions = boundary_condition_wall) + +############################################################################### +# ODE solvers, callbacks etc. + +# At t = 30, the wave moving with v + c crashes into the wall +tspan = (0.0, 30.0) +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_callback = AnalysisCallback(semi, interval = 100) + +# The StepsizeCallback handles the re-calculation of the maximum Δt after each time step +stepsize_callback = StepsizeCallback(cfl = 0.9) + +# Create a CallbackSet to collect all callbacks such that they can be passed to the ODE solver +callbacks = CallbackSet(summary_callback, analysis_callback, + stepsize_callback) + +############################################################################### +# run the simulation + +# OrdinaryDiffEq's `solve` method evolves the solution in time and executes the passed callbacks +sol = solve(ode, CarpenterKennedy2N54(williamson_condition = false), + dt = 1.0, # solve needs some value here but it will be overwritten by the stepsize_callback + save_everystep = false, callback = callbacks) + +# Print the timer summary +summary_callback() diff --git a/src/Trixi.jl b/src/Trixi.jl index 3e7fc5ee519..5511f7e02e2 100644 --- a/src/Trixi.jl +++ b/src/Trixi.jl @@ -160,7 +160,7 @@ export AcousticPerturbationEquations2D, LatticeBoltzmannEquations2D, LatticeBoltzmannEquations3D, ShallowWaterEquations1D, ShallowWaterEquations2D, ShallowWaterEquationsQuasi1D, - LinearizedEulerEquations1D, LinearizedEulerEquations2D, + LinearizedEulerEquations1D, LinearizedEulerEquations2D, LinearizedEulerEquations3D, PolytropicEulerEquations2D, TrafficFlowLWREquations1D diff --git a/src/equations/equations.jl b/src/equations/equations.jl index ef4dac8b1a5..69a54f7faf8 100644 --- a/src/equations/equations.jl +++ b/src/equations/equations.jl @@ -503,6 +503,7 @@ abstract type AbstractLinearizedEulerEquations{NDIMS, NVARS} <: AbstractEquations{NDIMS, NVARS} end include("linearized_euler_1d.jl") include("linearized_euler_2d.jl") +include("linearized_euler_3d.jl") abstract type AbstractEquationsParabolic{NDIMS, NVARS, GradientVariables} <: AbstractEquations{NDIMS, NVARS} end diff --git a/src/equations/linearized_euler_1d.jl b/src/equations/linearized_euler_1d.jl index 856a53a3d5d..19a3bdcb3bd 100644 --- a/src/equations/linearized_euler_1d.jl +++ b/src/equations/linearized_euler_1d.jl @@ -8,7 +8,7 @@ @doc raw""" LinearizedEulerEquations1D(v_mean_global, c_mean_global, rho_mean_global) -Linearized euler equations in one space dimension. The equations are given by +Linearized Euler equations in one space dimension. The equations are given by ```math \partial_t \begin{pmatrix} diff --git a/src/equations/linearized_euler_2d.jl b/src/equations/linearized_euler_2d.jl index d9d0d44e553..3df3093069d 100644 --- a/src/equations/linearized_euler_2d.jl +++ b/src/equations/linearized_euler_2d.jl @@ -8,7 +8,7 @@ @doc raw""" LinearizedEulerEquations2D(v_mean_global, c_mean_global, rho_mean_global) -Linearized euler equations in two space dimensions. The equations are given by +Linearized Euler equations in two space dimensions. The equations are given by ```math \partial_t \begin{pmatrix} diff --git a/src/equations/linearized_euler_3d.jl b/src/equations/linearized_euler_3d.jl new file mode 100644 index 00000000000..7d0f27133c3 --- /dev/null +++ b/src/equations/linearized_euler_3d.jl @@ -0,0 +1,255 @@ +# By default, Julia/LLVM does not use fused multiply-add operations (FMAs). +# Since these FMAs can increase the performance of many numerical algorithms, +# we need to opt-in explicitly. +# See https://ranocha.de/blog/Optimizing_EC_Trixi for further details. +@muladd begin +#! format: noindent + +@doc raw""" + LinearizedEulerEquations3D(v_mean_global, c_mean_global, rho_mean_global) + +Linearized Euler equations in two space dimensions. The equations are given by +```math +\partial_t +\begin{pmatrix} + \rho' \\ v_1' \\ v_2' \\ v_3' \\ p' +\end{pmatrix} ++ +\partial_x +\begin{pmatrix} + \bar{\rho} v_1' + \bar{v_1} \rho ' \\ + \bar{v_1} v_1' + \frac{p'}{\bar{\rho}} \\ + \bar{v_1} v_2' \\ + \bar{v_1} v_3' \\ + \bar{v_1} p' + c^2 \bar{\rho} v_1' +\end{pmatrix} ++ +\partial_y +\begin{pmatrix} + \bar{\rho} v_2' + \bar{v_2} \rho ' \\ + \bar{v_2} v_1' \\ + \bar{v_2} v_2' + \frac{p'}{\bar{\rho}} \\ + \bar{v_2} v_3' \\ + \bar{v_2} p' + c^2 \bar{\rho} v_2' +\end{pmatrix} ++ +\partial_z +\begin{pmatrix} + \bar{\rho} v_3' + \bar{v_3} \rho ' \\ + \bar{v_3} v_1' \\ + \bar{v_3} v_2' \\ + \bar{v_3} v_3' + + \frac{p'}{\bar{\rho}} \\ + \bar{v_3} p' + c^2 \bar{\rho} v_3' +\end{pmatrix} += +\begin{pmatrix} + 0 \\ 0 \\ 0 \\ 0 \\ 0 +\end{pmatrix} +``` +The bar ``\bar{(\cdot)}`` indicates uniform mean flow variables and ``c`` is the speed of sound. +The unknowns are the acoustic velocities ``v' = (v_1', v_2, v_3')``, the pressure ``p'`` and the density ``\rho'``. +""" +struct LinearizedEulerEquations3D{RealT <: Real} <: + AbstractLinearizedEulerEquations{3, 5} + v_mean_global::SVector{3, RealT} + c_mean_global::RealT + rho_mean_global::RealT +end + +function LinearizedEulerEquations3D(v_mean_global::NTuple{3, <:Real}, + c_mean_global::Real, rho_mean_global::Real) + if rho_mean_global < 0 + throw(ArgumentError("rho_mean_global must be non-negative")) + elseif c_mean_global < 0 + throw(ArgumentError("c_mean_global must be non-negative")) + end + + return LinearizedEulerEquations3D(SVector(v_mean_global), c_mean_global, + rho_mean_global) +end + +function LinearizedEulerEquations3D(; v_mean_global::NTuple{3, <:Real}, + c_mean_global::Real, rho_mean_global::Real) + return LinearizedEulerEquations3D(v_mean_global, c_mean_global, + rho_mean_global) +end + +function varnames(::typeof(cons2cons), ::LinearizedEulerEquations3D) + ("rho_prime", "v1_prime", "v2_prime", "v3_prime", "p_prime") +end +function varnames(::typeof(cons2prim), ::LinearizedEulerEquations3D) + ("rho_prime", "v1_prime", "v2_prime", "v3_prime", "p_prime") +end + +""" + initial_condition_convergence_test(x, t, equations::LinearizedEulerEquations3D) + +A smooth initial condition used for convergence tests. +""" +function initial_condition_convergence_test(x, t, equations::LinearizedEulerEquations3D) + rho_prime = -cospi(2 * t) * (sinpi(2 * x[1]) + sinpi(2 * x[2]) + sinpi(2 * x[3])) + v1_prime = sinpi(2 * t) * cospi(2 * x[1]) + v2_prime = sinpi(2 * t) * cospi(2 * x[2]) + v3_prime = sinpi(2 * t) * cospi(2 * x[3]) + p_prime = rho_prime + + return SVector(rho_prime, v1_prime, v2_prime, v3_prime, p_prime) +end + +""" + boundary_condition_wall(u_inner, orientation, direction, x, t, surface_flux_function, + equations::LinearizedEulerEquations3D) + +Boundary conditions for a solid wall. +""" +function boundary_condition_wall(u_inner, orientation, direction, x, t, + surface_flux_function, + equations::LinearizedEulerEquations3D) + # Boundary state is equal to the inner state except for the velocity. For boundaries + # in the -x/+x direction, we multiply the velocity in the x direction by -1. + # Similarly, for boundaries in the -y/+y direction, we multiply the velocity in the + # y direction by -1 + if direction in (1, 2) # x direction + u_boundary = SVector(u_inner[1], -u_inner[2], u_inner[3], u_inner[4], + u_inner[5]) + elseif direction in (3, 4) # y direction + u_boundary = SVector(u_inner[1], u_inner[2], -u_inner[3], u_inner[4], + u_inner[5]) + else # z direction = (5, 6) + u_boundary = SVector(u_inner[1], u_inner[2], u_inner[3], -u_inner[4], + u_inner[5]) + end + + # Calculate boundary flux depending on the orientation of the boundary + # Odd directions are in negative coordinate direction, + # even directions are in positive coordinate direction. + 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 + +# Calculate 1D flux for a single point +@inline function flux(u, orientation::Integer, equations::LinearizedEulerEquations3D) + @unpack v_mean_global, c_mean_global, rho_mean_global = equations + rho_prime, v1_prime, v2_prime, v3_prime, p_prime = u + if orientation == 1 + f1 = v_mean_global[1] * rho_prime + rho_mean_global * v1_prime + f2 = v_mean_global[1] * v1_prime + p_prime / rho_mean_global + f3 = v_mean_global[1] * v2_prime + f4 = v_mean_global[1] * v3_prime + f5 = v_mean_global[1] * p_prime + c_mean_global^2 * rho_mean_global * v1_prime + elseif orientation == 2 + f1 = v_mean_global[2] * rho_prime + rho_mean_global * v2_prime + f2 = v_mean_global[2] * v1_prime + f3 = v_mean_global[2] * v2_prime + p_prime / rho_mean_global + f4 = v_mean_global[2] * v3_prime + f5 = v_mean_global[2] * p_prime + c_mean_global^2 * rho_mean_global * v2_prime + else # orientation == 3 + f1 = v_mean_global[3] * rho_prime + rho_mean_global * v3_prime + f2 = v_mean_global[3] * v1_prime + f3 = v_mean_global[3] * v2_prime + f4 = v_mean_global[3] * v3_prime + p_prime / rho_mean_global + f5 = v_mean_global[3] * p_prime + c_mean_global^2 * rho_mean_global * v3_prime + end + + return SVector(f1, f2, f3, f4, f5) +end + +# Calculate 1D flux for a single point +@inline function flux(u, normal_direction::AbstractVector, + equations::LinearizedEulerEquations3D) + @unpack v_mean_global, c_mean_global, rho_mean_global = equations + rho_prime, v1_prime, v2_prime, v3_prime, p_prime = u + + v_mean_normal = v_mean_global[1] * normal_direction[1] + + v_mean_global[2] * normal_direction[2] + + v_mean_global[3] * normal_direction[3] + v_prime_normal = v1_prime * normal_direction[1] + v2_prime * normal_direction[2] + + v3_prime * normal_direction[3] + + f1 = v_mean_normal * rho_prime + rho_mean_global * v_prime_normal + f2 = v_mean_normal * v1_prime + normal_direction[1] * p_prime / rho_mean_global + f3 = v_mean_normal * v2_prime + normal_direction[2] * p_prime / rho_mean_global + f4 = v_mean_normal * v3_prime + normal_direction[3] * p_prime / rho_mean_global + f5 = v_mean_normal * p_prime + c_mean_global^2 * rho_mean_global * v_prime_normal + + return SVector(f1, f2, f3, f4, f5) +end + +@inline have_constant_speed(::LinearizedEulerEquations3D) = True() + +@inline function max_abs_speeds(equations::LinearizedEulerEquations3D) + @unpack v_mean_global, c_mean_global = equations + return abs(v_mean_global[1]) + c_mean_global, abs(v_mean_global[2]) + c_mean_global, + abs(v_mean_global[3]) + c_mean_global +end + +@inline function max_abs_speed_naive(u_ll, u_rr, orientation::Integer, + equations::LinearizedEulerEquations3D) + @unpack v_mean_global, c_mean_global = equations + if orientation == 1 + return abs(v_mean_global[1]) + c_mean_global + elseif orientation == 2 + return abs(v_mean_global[2]) + c_mean_global + else # orientation == 3 + return abs(v_mean_global[3]) + c_mean_global + end +end + +@inline function max_abs_speed_naive(u_ll, u_rr, normal_direction::AbstractVector, + equations::LinearizedEulerEquations3D) + @unpack v_mean_global, c_mean_global = equations + v_mean_normal = normal_direction[1] * v_mean_global[1] + + normal_direction[2] * v_mean_global[2] + + normal_direction[3] * v_mean_global[3] + return abs(v_mean_normal) + c_mean_global * norm(normal_direction) +end + +# Calculate estimate for minimum and maximum wave speeds for HLL-type fluxes +@inline function min_max_speed_naive(u_ll, u_rr, orientation::Integer, + equations::LinearizedEulerEquations3D) + min_max_speed_davis(u_ll, u_rr, orientation, equations) +end + +@inline function min_max_speed_naive(u_ll, u_rr, normal_direction::AbstractVector, + equations::LinearizedEulerEquations3D) + min_max_speed_davis(u_ll, u_rr, normal_direction, equations) +end + +# More refined estimates for minimum and maximum wave speeds for HLL-type fluxes +@inline function min_max_speed_davis(u_ll, u_rr, orientation::Integer, + equations::LinearizedEulerEquations3D) + @unpack v_mean_global, c_mean_global = equations + + λ_min = v_mean_global[orientation] - c_mean_global + λ_max = v_mean_global[orientation] + c_mean_global + + return λ_min, λ_max +end + +@inline function min_max_speed_davis(u_ll, u_rr, normal_direction::AbstractVector, + equations::LinearizedEulerEquations3D) + @unpack v_mean_global, c_mean_global = equations + + norm_ = norm(normal_direction) + + v_normal = v_mean_global[1] * normal_direction[1] + + v_mean_global[2] * normal_direction[2] + + v_mean_global[3] * normal_direction[3] + + # The v_normals are already scaled by the norm + λ_min = v_normal - c_mean_global * norm_ + λ_max = v_normal + c_mean_global * norm_ + + return λ_min, λ_max +end + +# Convert conservative variables to primitive +@inline cons2prim(u, equations::LinearizedEulerEquations3D) = u +@inline cons2entropy(u, ::LinearizedEulerEquations3D) = u +end # muladd diff --git a/test/test_p4est_3d.jl b/test/test_p4est_3d.jl index ea7d9193add..8b684b22d09 100644 --- a/test/test_p4est_3d.jl +++ b/test/test_p4est_3d.jl @@ -535,6 +535,28 @@ end @test (@allocated Trixi.rhs!(du_ode, u_ode, semi, t)) < 1000 end end + +@trixi_testset "elixir_linearizedeuler_convergence.jl" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_linearizedeuler_convergence.jl"), + l2=[ + 0.04452389418193219, 0.03688186699434862, + 0.03688186699434861, 0.03688186699434858, + 0.044523894181932186, + ], + linf=[ + 0.2295447498696467, 0.058369658071546704, + 0.05836965807154648, 0.05836965807154648, 0.2295447498696468, + ]) + # Ensure that we do not have excessive memory allocations + # (e.g., from type instabilities) + let + t = sol.t[end] + u_ode = sol.u[end] + du_ode = similar(u_ode) + @test (@allocated Trixi.rhs!(du_ode, u_ode, semi, t)) < 1000 + end +end end # Clean up afterwards: delete Trixi.jl output directory diff --git a/test/test_tree_3d_linearizedeuler.jl b/test/test_tree_3d_linearizedeuler.jl new file mode 100644 index 00000000000..002a77ce781 --- /dev/null +++ b/test/test_tree_3d_linearizedeuler.jl @@ -0,0 +1,35 @@ + +using Test +using Trixi + +include("test_trixi.jl") + +EXAMPLES_DIR = pkgdir(Trixi, "examples", "tree_3d_dgsem") + +@testset "Linearized Euler Equations 2D" begin +#! format: noindent + +@trixi_testset "elixir_linearizedeuler_gauss_wall.jl" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_linearizedeuler_gauss_wall.jl"), + l2=[ + 0.054201508303202306, 0.10347277056092179, + 8.10542634693489e-17, 8.273108096127844e-17, + 0.054201508303202306, + ], + linf=[ + 0.1774484851727698, 0.33221022902054875, + 1.9884884692276712e-15, 1.641012595221666e-15, + 0.1774484851727698, + ], + tspan=(0.0, 1.0)) + + # Ensure that we do not have excessive memory allocations + # (e.g., from type instabilities) + let + t = sol.t[end] + u_ode = sol.u[end] + du_ode = similar(u_ode) + @test (@allocated Trixi.rhs!(du_ode, u_ode, semi, t)) < 1000 + end +end +end diff --git a/test/test_tree_3d_part2.jl b/test/test_tree_3d_part2.jl index 5c7301654ad..4b9da039f98 100644 --- a/test/test_tree_3d_part2.jl +++ b/test/test_tree_3d_part2.jl @@ -22,6 +22,9 @@ isdir(outdir) && rm(outdir, recursive = true) # Compressible Euler with self-gravity include("test_tree_3d_eulergravity.jl") + + # Linearized Euler + include("test_tree_3d_linearizedeuler.jl") end @trixi_testset "Additional tests in 3D" begin From 53c789d6307099f28b6f15fedbbc28b359830f61 Mon Sep 17 00:00:00 2001 From: Daniel_Doehring Date: Thu, 11 Apr 2024 15:15:02 +0200 Subject: [PATCH 02/12] fmt --- examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl b/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl index 608949f1546..870f1b79824 100644 --- a/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl +++ b/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl @@ -19,7 +19,7 @@ coordinates_max = (1.0, 1.0, 1.0) # maximum coordinates (max(x), max(y), max(z)) # https://trixi-framework.github.io/Trixi.jl/stable/#Performing-a-convergence-analysis trees_per_dimension = (4, 4, 4) mesh = P4estMesh(trees_per_dimension, polydeg = 3, - coordinates_min = coordinates_min, + coordinates_min = coordinates_min, coordinates_max = coordinates_max, initial_refinement_level = 0) From 07a23c58bfcc3b6f4aab814720e40f859d13d307 Mon Sep 17 00:00:00 2001 From: Daniel Doehring Date: Fri, 12 Apr 2024 13:59:04 +0200 Subject: [PATCH 03/12] Apply suggestions from code review Co-authored-by: Lars Christmann --- src/equations/linearized_euler_3d.jl | 6 +++--- test/test_tree_3d_linearizedeuler.jl | 2 +- 2 files changed, 4 insertions(+), 4 deletions(-) diff --git a/src/equations/linearized_euler_3d.jl b/src/equations/linearized_euler_3d.jl index 7d0f27133c3..a03f13de36e 100644 --- a/src/equations/linearized_euler_3d.jl +++ b/src/equations/linearized_euler_3d.jl @@ -8,7 +8,7 @@ @doc raw""" LinearizedEulerEquations3D(v_mean_global, c_mean_global, rho_mean_global) -Linearized Euler equations in two space dimensions. The equations are given by +Linearized Euler equations in three space dimensions. The equations are given by ```math \partial_t \begin{pmatrix} @@ -108,8 +108,8 @@ function boundary_condition_wall(u_inner, orientation, direction, x, t, equations::LinearizedEulerEquations3D) # Boundary state is equal to the inner state except for the velocity. For boundaries # in the -x/+x direction, we multiply the velocity in the x direction by -1. - # Similarly, for boundaries in the -y/+y direction, we multiply the velocity in the - # y direction by -1 + # Similarly, for boundaries in the -y/+y or -z/+z direction, we multiply the + # velocity in the y or z direction by -1 if direction in (1, 2) # x direction u_boundary = SVector(u_inner[1], -u_inner[2], u_inner[3], u_inner[4], u_inner[5]) diff --git a/test/test_tree_3d_linearizedeuler.jl b/test/test_tree_3d_linearizedeuler.jl index 002a77ce781..ddfb17d9ee3 100644 --- a/test/test_tree_3d_linearizedeuler.jl +++ b/test/test_tree_3d_linearizedeuler.jl @@ -6,7 +6,7 @@ include("test_trixi.jl") EXAMPLES_DIR = pkgdir(Trixi, "examples", "tree_3d_dgsem") -@testset "Linearized Euler Equations 2D" begin +@testset "Linearized Euler Equations 3D" begin #! format: noindent @trixi_testset "elixir_linearizedeuler_gauss_wall.jl" begin From 1bc4893fcf93ff8443feced2c606a6571416700c Mon Sep 17 00:00:00 2001 From: Daniel_Doehring Date: Fri, 12 Apr 2024 14:08:53 +0200 Subject: [PATCH 04/12] nicer IC --- examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl b/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl index bba868cbe51..36150cd3b25 100644 --- a/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl +++ b/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl @@ -19,12 +19,12 @@ mesh = TreeMesh(coordinates_min, coordinates_max, periodicity = false) # Initialize density and pressure perturbation with a Gaussian bump -# that splits and is advected with v - c and v + c. +# that splits into radial waves which are advected with v - c and v + c. function initial_condition_gauss_wall(x, t, equations::LinearizedEulerEquations3D) v1_prime = 0.0 v2_prime = 0.0 v3_prime = 0.0 - rho_prime = p_prime = 2 * exp(-(x[1] - 45)^2 / 25) + rho_prime = p_prime = 2 * exp(-((x[1] - 45)^2 + (x[2] - 45)^2) / 25) return SVector(rho_prime, v1_prime, v2_prime, v3_prime, p_prime) end initial_condition = initial_condition_gauss_wall From a711b12cabefff6253cf548427606d1bff7a4d86 Mon Sep 17 00:00:00 2001 From: Daniel_Doehring Date: Fri, 12 Apr 2024 14:17:28 +0200 Subject: [PATCH 05/12] test vals --- test/test_tree_3d_linearizedeuler.jl | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/test/test_tree_3d_linearizedeuler.jl b/test/test_tree_3d_linearizedeuler.jl index ddfb17d9ee3..00f8d62dad9 100644 --- a/test/test_tree_3d_linearizedeuler.jl +++ b/test/test_tree_3d_linearizedeuler.jl @@ -12,14 +12,14 @@ EXAMPLES_DIR = pkgdir(Trixi, "examples", "tree_3d_dgsem") @trixi_testset "elixir_linearizedeuler_gauss_wall.jl" begin @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_linearizedeuler_gauss_wall.jl"), l2=[ - 0.054201508303202306, 0.10347277056092179, - 8.10542634693489e-17, 8.273108096127844e-17, - 0.054201508303202306, + 0.020380328336745232, 0.027122442311921492, + 0.02712244231192152, 8.273108096127844e-17, + 0.020380328336745232, ], linf=[ - 0.1774484851727698, 0.33221022902054875, - 1.9884884692276712e-15, 1.641012595221666e-15, - 0.1774484851727698, + 0.2916021983572774, 0.32763703462270843, + 0.32763703462270855, 1.641012595221666e-15, + 0.2916021983572774, ], tspan=(0.0, 1.0)) From 240fc91af7c4cc21c71ac3c488182d4ea0e794bc Mon Sep 17 00:00:00 2001 From: Daniel_Doehring Date: Fri, 12 Apr 2024 14:27:44 +0200 Subject: [PATCH 06/12] fmt --- src/equations/linearized_euler_3d.jl | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/src/equations/linearized_euler_3d.jl b/src/equations/linearized_euler_3d.jl index a03f13de36e..7059189eb15 100644 --- a/src/equations/linearized_euler_3d.jl +++ b/src/equations/linearized_euler_3d.jl @@ -38,8 +38,7 @@ Linearized Euler equations in three space dimensions. The equations are given by \bar{\rho} v_3' + \bar{v_3} \rho ' \\ \bar{v_3} v_1' \\ \bar{v_3} v_2' \\ - \bar{v_3} v_3' + - \frac{p'}{\bar{\rho}} \\ + \bar{v_3} v_3' + \frac{p'}{\bar{\rho}} \\ \bar{v_3} p' + c^2 \bar{\rho} v_3' \end{pmatrix} = From 8031fff1173ea1dba7d69fc2b379cb903d78f04c Mon Sep 17 00:00:00 2001 From: Daniel_Doehring Date: Sun, 14 Apr 2024 12:12:20 +0200 Subject: [PATCH 07/12] unit tests for full coverage --- test/test_unit.jl | 36 ++++++++++++++++++++++++++++++++++++ 1 file changed, 36 insertions(+) diff --git a/test/test_unit.jl b/test/test_unit.jl index 9a4ae2ede13..53349ba09f7 100644 --- a/test/test_unit.jl +++ b/test/test_unit.jl @@ -1536,6 +1536,42 @@ end end end +@testset "Equivalent Wave Speed Estimates" begin + @timed_testset "Linearized Euler 3D" begin + equations = LinearizedEulerEquations3D(v_mean_global = (0.42, 0.37, 0.7), + c_mean_global = 1.0, + rho_mean_global = 1.0) + + normal_x = SVector(1.0, 0.0, 0.0) + normal_y = SVector(0.0, 1.0, 0.0) + normal_z = SVector(0.0, 0.0, 1.0) + + u_ll = SVector(0.3, 0.5, -0.7, 0.1, 1.0) + u_rr = SVector(0.5, -0.2, 0.1, 0.2, 5.0) + + @test max_abs_speed_naive(u_ll, u_rr, 1, equations) == + max_abs_speed_naive(u_ll, u_rr, normal_x, equations) + @test max_abs_speed_naive(u_ll, u_rr, 2, equations) == + max_abs_speed_naive(u_ll, u_rr, normal_y, equations) + @test max_abs_speed_naive(u_ll, u_rr, 3, equations) == + max_abs_speed_naive(u_ll, u_rr, normal_z, equations) + + @test min_max_speed_naive(u_ll, u_rr, 1, equations) == + min_max_speed_naive(u_ll, u_rr, normal_x, equations) + @test min_max_speed_naive(u_ll, u_rr, 2, equations) == + min_max_speed_naive(u_ll, u_rr, normal_y, equations) + @test min_max_speed_naive(u_ll, u_rr, 3, equations) == + min_max_speed_naive(u_ll, u_rr, normal_z, equations) + + @test min_max_speed_davis(u_ll, u_rr, 1, equations) == + min_max_speed_davis(u_ll, u_rr, normal_x, equations) + @test min_max_speed_davis(u_ll, u_rr, 2, equations) == + min_max_speed_davis(u_ll, u_rr, normal_y, equations) + @test min_max_speed_davis(u_ll, u_rr, 3, equations) == + min_max_speed_davis(u_ll, u_rr, normal_z, equations) + end +end + @testset "SimpleKronecker" begin N = 3 From af9b0de19b0c4f840c21d4b62f49e3d0914d9725 Mon Sep 17 00:00:00 2001 From: Daniel Doehring Date: Sun, 14 Apr 2024 18:51:00 +0200 Subject: [PATCH 08/12] Update examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl --- examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl | 1 - 1 file changed, 1 deletion(-) diff --git a/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl b/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl index 36150cd3b25..2eb8ef822a3 100644 --- a/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl +++ b/examples/tree_3d_dgsem/elixir_linearizedeuler_gauss_wall.jl @@ -1,4 +1,3 @@ - using OrdinaryDiffEq using Trixi From 4856da9622ca90744855158b264ea58bdeca96ea Mon Sep 17 00:00:00 2001 From: Daniel_Doehring Date: Mon, 15 Apr 2024 10:59:15 +0200 Subject: [PATCH 09/12] whitespace --- examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl b/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl index 870f1b79824..fd066ef8955 100644 --- a/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl +++ b/examples/p4est_3d_dgsem/elixir_linearizedeuler_convergence.jl @@ -16,7 +16,7 @@ coordinates_max = (1.0, 1.0, 1.0) # maximum coordinates (max(x), max(y), max(z)) # `initial_refinement_level` is provided here to allow for a # convenient convergence test, see -# https://trixi-framework.github.io/Trixi.jl/stable/#Performing-a-convergence-analysis +# https://trixi-framework.github.io/Trixi.jl/stable/#Performing-a-convergence-analysis trees_per_dimension = (4, 4, 4) mesh = P4estMesh(trees_per_dimension, polydeg = 3, coordinates_min = coordinates_min, From 229dc7d6324304f2b7295dddf83241bda3e2c799 Mon Sep 17 00:00:00 2001 From: Daniel_Doehring Date: Mon, 15 Apr 2024 11:07:28 +0200 Subject: [PATCH 10/12] proper testing --- test/test_unit.jl | 58 +++++++++++++++++++++++++++++++---------------- 1 file changed, 38 insertions(+), 20 deletions(-) diff --git a/test/test_unit.jl b/test/test_unit.jl index 53349ba09f7..b23d4aa60e2 100644 --- a/test/test_unit.jl +++ b/test/test_unit.jl @@ -1549,26 +1549,44 @@ end u_ll = SVector(0.3, 0.5, -0.7, 0.1, 1.0) u_rr = SVector(0.5, -0.2, 0.1, 0.2, 5.0) - @test max_abs_speed_naive(u_ll, u_rr, 1, equations) == - max_abs_speed_naive(u_ll, u_rr, normal_x, equations) - @test max_abs_speed_naive(u_ll, u_rr, 2, equations) == - max_abs_speed_naive(u_ll, u_rr, normal_y, equations) - @test max_abs_speed_naive(u_ll, u_rr, 3, equations) == - max_abs_speed_naive(u_ll, u_rr, normal_z, equations) - - @test min_max_speed_naive(u_ll, u_rr, 1, equations) == - min_max_speed_naive(u_ll, u_rr, normal_x, equations) - @test min_max_speed_naive(u_ll, u_rr, 2, equations) == - min_max_speed_naive(u_ll, u_rr, normal_y, equations) - @test min_max_speed_naive(u_ll, u_rr, 3, equations) == - min_max_speed_naive(u_ll, u_rr, normal_z, equations) - - @test min_max_speed_davis(u_ll, u_rr, 1, equations) == - min_max_speed_davis(u_ll, u_rr, normal_x, equations) - @test min_max_speed_davis(u_ll, u_rr, 2, equations) == - min_max_speed_davis(u_ll, u_rr, normal_y, equations) - @test min_max_speed_davis(u_ll, u_rr, 3, equations) == - min_max_speed_davis(u_ll, u_rr, normal_z, equations) + @test all(isapprox(x, y) + for (x, y) in zip(max_abs_speed_naive(u_ll, u_rr, 1, equations), + max_abs_speed_naive(u_ll, u_rr, normal_x, + equations))) + @test all(isapprox(x, y) + for (x, y) in zip(max_abs_speed_naive(u_ll, u_rr, 2, equations), + max_abs_speed_naive(u_ll, u_rr, normal_y, + equations))) + @test all(isapprox(x, y) + for (x, y) in zip(max_abs_speed_naive(u_ll, u_rr, 3, equations), + max_abs_speed_naive(u_ll, u_rr, normal_z, + equations))) + + @test all(isapprox(x, y) + for (x, y) in zip(min_max_speed_naive(u_ll, u_rr, 1, equations), + min_max_speed_naive(u_ll, u_rr, normal_x, + equations))) + @test all(isapprox(x, y) + for (x, y) in zip(min_max_speed_naive(u_ll, u_rr, 2, equations), + min_max_speed_naive(u_ll, u_rr, normal_y, + equations))) + @test all(isapprox(x, y) + for (x, y) in zip(min_max_speed_naive(u_ll, u_rr, 3, equations), + min_max_speed_naive(u_ll, u_rr, normal_z, + equations))) + + @test all(isapprox(x, y) + for (x, y) in zip(min_max_speed_davis(u_ll, u_rr, 1, equations), + min_max_speed_davis(u_ll, u_rr, normal_x, + equations))) + @test all(isapprox(x, y) + for (x, y) in zip(min_max_speed_davis(u_ll, u_rr, 2, equations), + min_max_speed_davis(u_ll, u_rr, normal_y, + equations))) + @test all(isapprox(x, y) + for (x, y) in zip(min_max_speed_davis(u_ll, u_rr, 3, equations), + min_max_speed_davis(u_ll, u_rr, normal_z, + equations))) end end From efdf4f2f56522546d7d0db64a6b173aa284e690b Mon Sep 17 00:00:00 2001 From: Daniel Doehring Date: Mon, 15 Apr 2024 19:20:14 +0200 Subject: [PATCH 11/12] Update src/equations/linearized_euler_3d.jl Co-authored-by: Andrew Winters --- src/equations/linearized_euler_3d.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/equations/linearized_euler_3d.jl b/src/equations/linearized_euler_3d.jl index 7059189eb15..909a8944c11 100644 --- a/src/equations/linearized_euler_3d.jl +++ b/src/equations/linearized_euler_3d.jl @@ -159,7 +159,7 @@ end return SVector(f1, f2, f3, f4, f5) end -# Calculate 1D flux for a single point +# Calculate 3D flux for a single point @inline function flux(u, normal_direction::AbstractVector, equations::LinearizedEulerEquations3D) @unpack v_mean_global, c_mean_global, rho_mean_global = equations From fc7e86b5a2a5322ec26d8d6987e41a1003f21722 Mon Sep 17 00:00:00 2001 From: Daniel Doehring Date: Mon, 15 Apr 2024 19:20:49 +0200 Subject: [PATCH 12/12] Update src/equations/linearized_euler_3d.jl --- src/equations/linearized_euler_3d.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/equations/linearized_euler_3d.jl b/src/equations/linearized_euler_3d.jl index 909a8944c11..ab5f9863db8 100644 --- a/src/equations/linearized_euler_3d.jl +++ b/src/equations/linearized_euler_3d.jl @@ -132,7 +132,7 @@ function boundary_condition_wall(u_inner, orientation, direction, x, t, return flux end -# Calculate 1D flux for a single point +# Calculate 3D flux for a single point @inline function flux(u, orientation::Integer, equations::LinearizedEulerEquations3D) @unpack v_mean_global, c_mean_global, rho_mean_global = equations rho_prime, v1_prime, v2_prime, v3_prime, p_prime = u