From 65c873b64c2792c5041639a4e6def63f108f3ce5 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Fri, 2 Aug 2024 15:09:38 +0200 Subject: [PATCH 01/12] add SWE-Exner1D --- Project.toml | 2 +- .../elixir_shallowwater_exner_channel.jl | 116 ++++ .../elixir_shallowwater_exner_source_terms.jl | 154 ++++++ ...elixir_shallowwater_exner_well_balanced.jl | 103 ++++ src/TrixiShallowWater.jl | 4 + src/equations/equations.jl | 2 + src/equations/shallow_water_exner_1d.jl | 513 ++++++++++++++++++ test/test_tree_1d.jl | 154 ++++++ test/test_unit.jl | 68 +++ 9 files changed, 1115 insertions(+), 1 deletion(-) create mode 100644 examples/tree_1d_dgsem/elixir_shallowwater_exner_channel.jl create mode 100644 examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl create mode 100644 examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl create mode 100644 src/equations/shallow_water_exner_1d.jl diff --git a/Project.toml b/Project.toml index 6ea3d75..1871d4c 100644 --- a/Project.toml +++ b/Project.toml @@ -16,6 +16,6 @@ LinearAlgebra = "1" MuladdMacro = "0.2.2" Static = "0.3, 0.4, 0.5, 0.6, 0.7, 0.8" StaticArrays = "1" -Trixi = "0.7" +Trixi = "0.7, 0.8" julia = "1.8" Printf = "1" diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_channel.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_channel.jl new file mode 100644 index 0000000..f6cc9ac --- /dev/null +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_channel.jl @@ -0,0 +1,116 @@ + +using OrdinaryDiffEq +using Trixi +using TrixiShallowWater +using Roots + +############################################################################### +# Semidiscretization of the shallow water exner equations for a channel flow problem +# with sediment transport + +equations = ShallowWaterExnerEquations1D(gravity_constant = 9.81, rho_f = 1.0, rho_s = 1.0, + porosity = 0.4, + sediment_model = GrassModel(A_g = 0.01)) + +# Initial condition for for a channel flow problem over a sediment hump. +# An asymptotic solution based on the method of characteristics was derived under a rigid-lid +# approximation in chapter 3.5.1 of the thesis: +# - Justin Hudson (2001) +# "Numerical Techniques for Morphodynamic Modelling" +# [PhD Thesis, University of Reading](https://citeseerx.ist.psu.edu/document?repid=rep1&type=pdf&doi=f78dbae9cfbb12ae823975c6ce9d2585b40417ba) +function initial_condition_channel(x, t, + equations::ShallowWaterExnerEquations1D) + (; sediment_model, porosity_inv) = equations + + H_ref = 10.0 # Reference water level + hv = 10.0 + + # Use the method of characteristics to find the asymptotic solution for the bed height, see + # Eq. 3.16 in the reference. + # First use an iterative method to predict x0 + f(x0) = x[1] - x0 - + sediment_model.A_g * porosity_inv * 3 * hv^3 * t * + (H_ref - sinpi((x0 - 300) / 200)^2)^(-4) + fx = Roots.ZeroProblem(f, 400.0) + x0 = Roots.solve(fx, atol = 0.0, rtol = 0.0) + + # If the result is outside 300 < x < 500 the result is invalid and instead compute x0 from + if x0 > 500 || x0 < 300 + x0 = x[1] - + sediment_model.A_g * porosity_inv * 3 * hv^3 * t * H_ref^(-4) + end + + # Compute the sediment and water height + 300 < x0 < 500 ? h_b = 0.1 + sinpi((x0 - 300) / 200)^2 : h_b = 0.1 + h = H_ref - h_b + + return SVector(h, hv, h_b) +end + +initial_condition = initial_condition_channel + +############################################################################### +# Get the DG approximation space + +volume_flux = (flux_ersing_etal, flux_nonconservative_ersing_etal) +surface_flux = (FluxPlusDissipation(flux_ersing_etal, dissipation_roe), + flux_nonconservative_ersing_etal) + +basis = LobattoLegendreBasis(6) + +indicator_sc = IndicatorHennemannGassner(equations, basis, + alpha_max = 0.5, + alpha_min = 0.001, + alpha_smooth = true, + variable = water_sediment_height) +volume_integral = VolumeIntegralShockCapturingHG(indicator_sc; + volume_flux_dg = volume_flux, + volume_flux_fv = surface_flux) + +solver = DGSEM(basis, surface_flux, volume_integral) + +############################################################################### +# Get the TreeMesh and setup a periodic mesh + +coordinates_min = 0.0 +coordinates_max = 1000.0 +mesh = TreeMesh(coordinates_min, coordinates_max, + initial_refinement_level = 4, + n_cells_max = 10_000, + periodicity = true) + +# Create the semi discretization object +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver) + +############################################################################### +# ODE solver + +tspan = (0.0, 30000) +ode = semidiscretize(semi, tspan) + +############################################################################### +# Callbacks + +summary_callback = SummaryCallback() + +analysis_interval = 10000 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +stepsize_callback = StepsizeCallback(cfl = 0.8) + +save_solution = SaveSolutionCallback(dt = 10000, + save_initial_solution = true, + save_final_solution = true) + +callbacks = CallbackSet(summary_callback, analysis_callback, alive_callback, + stepsize_callback, save_solution) + +############################################################################### +# run the simulation +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); + +summary_callback() # print the timer summary diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl new file mode 100644 index 0000000..02ce689 --- /dev/null +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl @@ -0,0 +1,154 @@ +using OrdinaryDiffEq +using Trixi +using TrixiShallowWater + +############################################################################### +# Semidiscretization of the SWE-Exner equations with source terms for convergence testing + +# Equations with MeyerPeterMueller model +equations_mpm = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, + rho_s = 1.0, porosity = 0.5, + friction = ManningFriction(n = 0.01), + sediment_model = MeyerPeterMueller(theta_c = 0.0, + d_s = 1e-3)) + +# Equations with Grass model +equations_grass = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, + rho_s = 1.0, porosity = 0.5, + friction = ManningFriction(n = 0.0), + sediment_model = GrassModel(A_g = 0.01)) + +equations = equations_grass + +# Smooth initial condition to test convergence +@inline function Trixi.initial_condition_convergence_test(x, t, + equations::ShallowWaterExnerEquations1D) + ω = sqrt(2) * pi + + h = 2.0 + cos(ω * x[1]) * cos(ω * t) + v = 0.5 + h_b = 2.0 + sin(ω * x[1]) * cos(ω * t) + + return SVector(h, h * v, h_b) +end + +""" + source_terms_convergence_test(u, x, t, equations::ShallowWaterExnerEquations1D{T, S, ShieldsStressModel{T}}) where {T, S}) + +Source terms used for convergence tests in combination with [`Trixi.initial_condition_convergence_test`](@extref) +when using the [`MeyerPeterMueller`](@ref) model. +""" +@inline function Trixi.source_terms_convergence_test(u, x, t, + equations::ShallowWaterExnerEquations1D{T, + S, + ShieldsStressModel{T}}) where { + T, + S + } + ω = sqrt(2.0) * pi + (; gravity, porosity_inv, rho_f, rho_s, r) = equations + + n = equations.friction.n + + # Constant expression from the MPM model + c = sqrt(gravity * (1 / r - 1)) * 8.0 * porosity_inv * + (rho_f / (rho_s - rho_f))^(3 / 2) * n^3 + + h = -cos(x[1] * ω) * sin(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω + + hv = ((5.0 * c * + (cos(x[1] * ω) * cos(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω)) / + ((2.0 + cos(x[1] * ω) * cos(t * ω))^0.5) - 0.5 * cos(x[1] * ω) * sin(t * ω) * ω - + 0.25 * sin(x[1] * ω) * cos(t * ω) * ω + + 10.0 * (2.0 + cos(x[1] * ω) * cos(t * ω)) * + (cos(x[1] * ω) * cos(t * ω) * ω - sin(x[1] * ω) * cos(t * ω) * ω)) + + h_b = ((0.5 * ((0.125 * c) / (2.0 + cos(x[1] * ω) * cos(t * ω))) * sin(x[1] * ω) * + cos(t * ω) * ω) / ((2.0 + cos(x[1] * ω) * cos(t * ω))^0.5) - + sin(x[1] * ω) * sin(t * ω) * ω) + + return SVector(h, hv, h_b) +end + +""" + source_terms_convergence_test(u, x, t, equations::ShallowWaterExnerEquations1D{T, S, GrassModel{T}}) where {T, S}) + +Source terms used for convergence tests in combination with [`initial_condition_convergence_test`](@extref) +when using the the [`GrassModel`](@ref) model. +""" +@inline function Trixi.source_terms_convergence_test(u, x, t, + equations::ShallowWaterExnerEquations1D{T, + S, + GrassModel{T}}) where { + T, + S + } + ω = sqrt(2.0) * pi + A_g = equations.sediment_model.A_g + + h = -cos(x[1] * ω) * sin(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω + hv = -0.5 * cos(x[1] * ω) * sin(t * ω) * ω - 0.25 * sin(x[1] * ω) * cos(t * ω) * ω + + 10.0 * A_g * + (cos(x[1] * ω) * cos(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω) + + 10.0 * (2.0 + cos(x[1] * ω) * cos(t * ω)) * + (cos(x[1] * ω) * cos(t * ω) * ω - sin(x[1] * ω) * cos(t * ω) * ω) + h_b = -sin(x[1] * ω) * sin(t * ω) * ω + return SVector(h, hv, h_b) +end + +initial_condition = initial_condition_convergence_test + +############################################################################### +# Get the DG approximation space + +volume_flux = (flux_ersing_etal, flux_nonconservative_ersing_etal) +surface_flux = (flux_ersing_etal, flux_nonconservative_ersing_etal) + +solver = DGSEM(polydeg = 4, + surface_flux = surface_flux, + volume_integral = VolumeIntegralFluxDifferencing(volume_flux)) + +############################################################################### +# Get the TreeMesh and setup a periodic mesh + +coordinates_min = 0.0 +coordinates_max = sqrt(2.0) +mesh = TreeMesh(coordinates_min, coordinates_max, + initial_refinement_level = 2, + n_cells_max = 10_000, + periodicity = true) + +# create the semi discretization object +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver, + source_terms = source_terms_convergence_test) + +############################################################################### +# ODE solvers, callbacks etc. + +tspan = (0.0, 1.0) +ode = semidiscretize(semi, tspan) + +summary_callback = SummaryCallback() + +analysis_interval = 200 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +stepsize_callback = StepsizeCallback(cfl = 1.0) + +save_solution = SaveSolutionCallback(interval = 200, + save_initial_solution = true, + save_final_solution = true) + +callbacks = CallbackSet(summary_callback, analysis_callback, alive_callback, + stepsize_callback, save_solution) + +############################################################################### +# run the simulation + +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); + +summary_callback() # print the timer summary diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl new file mode 100644 index 0000000..95f6137 --- /dev/null +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl @@ -0,0 +1,103 @@ + +using OrdinaryDiffEq +using Trixi +using TrixiShallowWater + +############################################################################### +# Semidiscretization of the SWE-Exner equations with a discontinous sediment bed +# to test well-balancedness + +equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, H0 = 1.0, + rho_f = 0.5, rho_s = 1.0, porosity = 0.5, + friction = ManningFriction(n = 0.01), + sediment_model = MeyerPeterMueller(theta_c = 0.047, + d_s = 1e-3)) + +function initial_condition_steady_state(x, t, + equations::ShallowWaterExnerEquations1D) + hv = 0.0 + + # discontinuous sediment bed + if -2.0 < x[1] < 0.0 + h_s = 0.3 + else + h_s = 0.1 + end + + h = 1.0 - h_s + + return SVector(h, hv, h_s) +end + +initial_condition = initial_condition_steady_state + +boundary_condition = boundary_condition_slip_wall + +############################################################################### +# Get the DG approximation space + +volume_flux = (flux_ersing_etal, flux_nonconservative_ersing_etal) +surface_flux = (FluxPlusDissipation(flux_ersing_etal, DissipationLocalLaxFriedrichs()), + flux_nonconservative_ersing_etal) + +basis = LobattoLegendreBasis(3) + +indicator_sc = IndicatorHennemannGassner(equations, basis, + alpha_max = 0.5, + alpha_min = 0.001, + alpha_smooth = true, + variable = water_sediment_height) +volume_integral = VolumeIntegralShockCapturingHG(indicator_sc; + volume_flux_dg = volume_flux, + volume_flux_fv = surface_flux) + +solver = DGSEM(basis, surface_flux, volume_integral) + +############################################################################### +# Get the TreeMesh + +coordinates_min = -2.0 +coordinates_max = 2.0 +mesh = TreeMesh(coordinates_min, coordinates_max, + initial_refinement_level = 4, + n_cells_max = 10_000, + periodicity = false) + +# Create the semi discretization object +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver, + source_terms = source_term_bottom_friction, + boundary_conditions = boundary_condition) + +############################################################################### +# ODE solver + +tspan = (0.0, 200.0) +ode = semidiscretize(semi, tspan) + +############################################################################### +# Callbacks + +summary_callback = SummaryCallback() + +analysis_interval = 10000 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval, + extra_analysis_integrals = (lake_at_rest_error,)) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +save_solution = SaveSolutionCallback(dt = 100, + save_initial_solution = true, + save_final_solution = true) + +stepsize_callback = StepsizeCallback(cfl = 1.0) + +callbacks = CallbackSet(summary_callback, analysis_callback, alive_callback, save_solution, + stepsize_callback) + +############################################################################### +# run the simulation +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); + +summary_callback() # print the timer summary diff --git a/src/TrixiShallowWater.jl b/src/TrixiShallowWater.jl index 5522fc5..86cd7e9 100644 --- a/src/TrixiShallowWater.jl +++ b/src/TrixiShallowWater.jl @@ -19,6 +19,7 @@ include("solvers/indicators.jl") # Export types/functions that define the public API of TrixiShallowWater.jl export ShallowWaterEquationsWetDry1D, ShallowWaterEquationsWetDry2D, + ShallowWaterExnerEquations1D, ShallowWaterTwoLayerEquations1D, ShallowWaterTwoLayerEquations2D, ShallowWaterMultiLayerEquations1D, ShallowWaterMultiLayerEquations2D @@ -26,6 +27,9 @@ export hydrostatic_reconstruction_chen_noelle, flux_nonconservative_chen_noelle, min_max_speed_chen_noelle, flux_hll_chen_noelle, flux_ersing_etal, flux_es_ersing_etal, hydrostatic_reconstruction_ersing_etal +export ManningFriction, MeyerPeterMueller, GrassModel, ShieldsStressModel, + dissipation_roe, water_sediment_height, source_term_bottom_friction + export nlayers, eachlayer export PositivityPreservingLimiterShallowWater diff --git a/src/equations/equations.jl b/src/equations/equations.jl index b2e69a6..b215b9a 100644 --- a/src/equations/equations.jl +++ b/src/equations/equations.jl @@ -11,6 +11,8 @@ include("shallow_water_wet_dry_1d.jl") include("shallow_water_wet_dry_2d.jl") +include("shallow_water_exner_1d.jl") + include("shallow_water_two_layer_1d.jl") include("shallow_water_two_layer_2d.jl") diff --git a/src/equations/shallow_water_exner_1d.jl b/src/equations/shallow_water_exner_1d.jl new file mode 100644 index 0000000..3a693b2 --- /dev/null +++ b/src/equations/shallow_water_exner_1d.jl @@ -0,0 +1,513 @@ +# 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 + +# Abstract type for the different bottom friction models +abstract type Friction{RealT} end + +""" + ManningFriction(; n) + +Creates a Manning friction model for the bottom friction with Manning coefficient `n`. +The type is used to dispatch on the respective friction law +[`shear_stress_coefficient(u, friction::ManningFriction)`](@ref) when computing the [`shear_stress`](@ref). +""" +struct ManningFriction{RealT} <: Friction{RealT} + n::RealT +end + +function ManningFriction(; n) + ManningFriction(n) +end + +# Abstract type for the different models to compute sediment discharge +abstract type SedimentModel{RealT} end + +""" + ShieldsStressModel(; m_1, m_2, m_3, k_1, k_2, k_3, theta_c, d_s) + +Create a Shields stress model to compute the sediment discharge [`q_s`](@ref) based on the generalized +formulation from equation (1.2) in the paper: +- E.D. Fernández-Nieto, T.M. de Luna, G. Narbona-Reina and J. de Dieu Zabsonré (2017) + Formal deduction of the Saint-Venant–Exner model including arbitrarily sloping sediment beds and + associated energy + [DOI: 10.1051/m2an/2016018](https://doi.org/10.1051/m2an/2016018) +""" +struct ShieldsStressModel{RealT} <: SedimentModel{RealT} + m_1::RealT + m_2::RealT + m_3::RealT + k_1::RealT + k_2::RealT + k_3::RealT + theta_c::RealT # critical shields stress + d_s::RealT # grain diameter +end + +""" + GrassModel(; A_g, m_g=3) + +Creates a Grass model to compute the sediment discharge [`q_s``](@ref) as +```math +q_s = A_g v^m_g +``` +with the coefficients `A_g` and `m_g`. + +An overview of different formulations to compute the sediment discharge can be found in: +- M.J. Castro Díaz, E.D. Fernández-Nieto, A.M. Ferreiro (2008) + Sediment transport models in Shallow Water equations and numerical approach by high order + finite volume methods + [DOI:10.1016/j.compfluid.2007.07.017](https://doi.org/10.1016/j.compfluid.2007.07.017) +""" +struct GrassModel{RealT} <: SedimentModel{RealT} + A_g::RealT + m_g::RealT +end + +function GrassModel(; A_g, m_g = 3.0) + return GrassModel(A_g, m_g) +end + +""" + MeyerPeterMueller(; theta_c, d_s) + +Creates a Meyer-Peter-Mueller model to compute the sediment discharge +[`q_s``](@ref) with the critical Shields stress `theta_c` and the grain diameter `d_s`. + +An overview of different formulations to compute the sediment discharge can be found in: +- M.J. Castro Díaz, E.D. Fernández-Nieto, A.M. Ferreiro (2008) + Sediment transport models in Shallow Water equations and numerical approach by high order + finite volume methods + [DOI:10.1016/j.compfluid.2007.07.017](https://doi.org/10.1016/j.compfluid.2007.07.017) +""" +function MeyerPeterMueller(; theta_c, d_s) + return ShieldsStressModel(0.0, 1.5, 0.0, 8.0, 1.0, 0.0, theta_c, d_s) +end + +@doc raw""" + ShallowWaterExnerEquations1D(;gravity_constant, H0 = 0.0, + friction = ManningFriction(n = 0.0), + sediment_model, + porosity, + rho_f, rho_s) +Entropy conservative formulation of the Shallow water-Exner equations in one space dimension. +The equations are given by +```math +\begin{cases} +\partial_t h + \partial_x hv = 0, \\ +\partial_t hv + \partial_x (hv^2) + gh\partial_x (h + h_b) + g\frac{1}{r}h_s\partial_x (rh + h_b) + \frac{\tau}{\rho_f} = 0,\\ +\partial_t h_b + \partial_x q_s = 0, +\end{cases} +``` +The unknown quantities are the water and sediment height ``h``, ``h_b`` and the velocity ``v``. +The sediment discharge ``q_s(h, hv)`` is determined by the ``sediment_model`` and is used to determine +the active sediment height ``h_s = q_s / v``. +Furthermore ``\tau`` denotes the shear stress at the water-sediment interface and is determined by +the ``friction`` model. +The gravitational constant is denoted by `g`, and ``rho_f`` and ``rho_s`` are the fluid and sediment +densities, respectively. The density ratio is given by ``r = \rho_f / \rho_s``. + +The conservative variable water height ``h`` is measured from the sediment height ``h_b``, therefore +one also defines the total water height as ``H = h + h_b``. + +The additional quantity ``H_0`` is also available to store a reference value for the total water +height that is useful to set initial conditions or test the "lake-at-rest" well-balancedness. + +The entropy conservative formulation has been derived in the paper: +- E.D. Fernández-Nieto, T.M. de Luna, G. Narbona-Reina and J. de Dieu Zabsonré (2017) + Formal deduction of the Saint-Venant–Exner model including arbitrarily sloping sediment beds and + associated energy + [DOI: 10.1051/m2an/2016018](https://doi.org/10.1051/m2an/2016018) +""" +struct ShallowWaterExnerEquations1D{RealT <: Real, + FrictionT <: Friction{RealT}, + SedimentT <: SedimentModel{RealT}} <: + Trixi.AbstractShallowWaterEquations{1, 3} + # This structure ensures that friction and sediment models are concrete types + # to prevent allocations + gravity::RealT # gravitational constant + H0::RealT # constant "lake-at-rest" total water height + friction::FrictionT + sediment_model::SedimentT + porosity_inv::RealT # 1/(1-porosity) + rho_f::RealT # density of fluid + rho_s::RealT # density of sediment + r::RealT # density ratio +end + +# Allow for flexibility to set the gravitational constant within an elixir depending on the +# application where `gravity_constant=1.0` or `gravity_constant=9.81` are common values. +# The reference total water height H0 defaults to 0.0 but is used for the "lake-at-rest" +# well-balancedness test cases. +function ShallowWaterExnerEquations1D(; gravity_constant, H0 = zero(gravity_constant), + friction = ManningFriction(n = 0.0), + sediment_model, + porosity, rho_f, rho_s) + # Precompute common expressions for the porosity and density ratio + porosity_inv = 1 / (1 - porosity) + r = rho_f / rho_s + return ShallowWaterExnerEquations1D(gravity_constant, H0, friction, sediment_model, + porosity_inv, rho_f, rho_s, r) +end + +Trixi.have_nonconservative_terms(::ShallowWaterExnerEquations1D) = True() +Trixi.varnames(::typeof(cons2cons), ::ShallowWaterExnerEquations1D) = ("h", "hv", "h_b") +# Note, we use the total water height, H = h + h_s, as the first primitive variable for easier +# visualization and setting initial conditions +Trixi.varnames(::typeof(cons2prim), ::ShallowWaterExnerEquations1D) = ("H", "v", "h_b") + +""" + boundary_condition_slip_wall(u_inner, orientation_or_normal, x, t, surface_flux_function, + equations::ShallowWaterExnerEquations1D) + +Create a boundary state by reflecting the normal velocity component and keep +the tangential velocity component unchanged. The boundary water height is taken from +the internal value. + +For details see Section 9.2.5 of the book: +- Eleuterio F. Toro (2001) + Shock-Capturing Methods for Free-Surface Shallow Flows + 1st edition + ISBN 0471987662 +""" +@inline function Trixi.boundary_condition_slip_wall(u_inner, orientation_or_normal, + direction, + x, t, + surface_flux_function, + equations::ShallowWaterExnerEquations1D) + # create the "external" boundary solution state + u_boundary = SVector(u_inner[1], + -u_inner[2], + u_inner[3]) + + # calculate the 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_or_normal, + equations) + else # u_boundary is "left" of boundary, u_inner is "right" of boundary + flux = surface_flux_function(u_boundary, u_inner, orientation_or_normal, + equations) + end + + return flux +end + +""" + source_term_bottom_friction(u, x, t, equations::ShallowWaterExnerEquations1D) + +Source term that accounts for the bottom friction in the [ShallowWaterExnerEquations1D](@ref). +The actual friction law is determined through the friction model in `equations.friction`. +""" +@inline function source_term_bottom_friction(u, x, t, + equations::ShallowWaterExnerEquations1D) + return SVector(zero(eltype(u)), + -shear_stress(u, equations), + zero(eltype(u))) +end + +# Calculate 1D flux for a single point +# Note, the bottom topography has no flux +@inline function Trixi.flux(u, orientation::Integer, + equations::ShallowWaterExnerEquations1D) + _, hv, _ = u + + v = velocity(u, equations) + + f1 = hv + f2 = hv * v + f3 = q_s(u, equations) + + return SVector(f1, f2, f3) +end + +""" + flux_nonconservative_ersing_etal(u_ll, u_rr, orientation, equations::ShallowWaterExnerEquations1D) + +Non-symmetric path-conservative two-point flux discretizing the nonconservative terms of the +[`ShallowWaterExnerEquations1D`](@ref) which consists of the hydrostatic pressure of the fluid +layer and an additional pressure contribution from the sediment layer to obtain an entropy +inequality. + +This non-conservative flux should be used together with [`flux_ersing_etal`](@ref) to create a +scheme that is entropy conservative and well-balanced. +""" +@inline function Trixi.flux_nonconservative_ersing_etal(u_ll, u_rr, + orientation::Integer, + equations::ShallowWaterExnerEquations1D) + # Pull the necessary left and right state information + h_ll, _, h_b_ll = u_ll + h_rr, _, h_b_rr = u_rr + + # Calculate jumps + h_jump = h_rr - h_ll + h_b_jump = h_b_rr - h_b_ll + + # Workaround to avoid division by zero, when computing the effective sediment height + if velocity(u_ll, equations) < eps() + h_s_ll = 0.0 + else + h_s_ll = q_s(u_ll, equations) / velocity(u_ll, equations) + end + + z = zero(eltype(u_ll)) + + f2 = equations.gravity * h_ll * (h_jump + h_b_jump) + # Additional nonconservative term to obtain entropy conservative formulation + f2 += (equations.gravity / equations.r * h_s_ll * + (equations.r * h_jump + h_b_jump)) + + return SVector(z, f2, z) +end + +""" + flux_ersing_etal(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterMultiLayerEquations1D) + +Entropy conservative split form, without the hydrostatic pressure. This flux should be used +together with the nonconservative [`flux_nonconservative_ersing_etal`](@ref) to create a scheme +that is entropy conservative and well-balanced. + +To obtain an entropy stable formulation the `surface_flux` can be set as +`FluxPlusDissipation(flux_ersing_etal, DissipationLocalLaxFriedrichs()), flux_nonconservative_ersing_etal`. +""" +@inline function flux_ersing_etal(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterExnerEquations1D) + # Unpack left and right state + _, h_v_ll, _ = u_ll + _, h_v_rr, _ = u_rr + + # Get the velocities on either side + v_ll = velocity(u_ll, equations) + v_rr = velocity(u_rr, equations) + + # Average each factor of products in flux + v_avg = 0.5 * (v_ll + v_rr) + + # Calculate fluxes depending on orientation + f1 = 0.5 * (h_v_ll + h_v_rr) + f2 = f1 * v_avg + f3 = 0.5 * (q_s(u_ll, equations) + q_s(u_rr, equations)) + + return SVector(f1, f2, f3) +end + +""" + dissipation_roe(u_ll, u_rr, orientation_or_normal_direction, + equations::ShallowWaterExnerEquations1D) +Roe-type dissipation term for the [`ShallowWaterExnerEquations1D`] with an approximate Roe average +for the sediment discharge `q_s`. +""" +@inline function dissipation_roe(u_ll, u_rr, orientation_or_normal_direction, + equations::ShallowWaterExnerEquations1D) + r = equations.r + g = equations.gravity + z = zero(eltype(u_ll)) + + # Get entropy variables and velocities + v_ll = velocity(u_ll, equations) + v_rr = velocity(u_rr, equations) + + # Compute approximate Roe averages. + # The actual Roe average for the sediment discharge `q_s` would depend on the sediment and + # friction model and can be hard to find. Therefore we only use an approximation here. + h_avg = 0.5 * (u_ll[1] + u_rr[1]) + v_avg = (sqrt(u_ll[1]) * v_ll + sqrt(u_rr[1]) * v_rr) / + (sqrt(u_ll[1]) + sqrt(u_rr[1])) + h_s_avg = q_s(SVector(h_avg, h_avg * v_avg, z), equations) / v_avg + + # Compute the eigenvalues using Cardano's formula + λ1, λ2, λ3 = eigvals_cardano(SVector(h_avg, h_avg * v_avg, h_s_avg), equations) + + # Precompute some common expressions + c1 = g * (h_avg + h_s_avg) + c2 = g * (h_avg + h_s_avg / r) + + # Eigenvector matrix + r31 = ((v_avg - λ1)^2 - c1) / c2 + r32 = ((v_avg - λ2)^2 - c1) / c2 + r33 = ((v_avg - λ3)^2 - c1) / c2 + R = @SMatrix [[1 1 1]; [λ1 λ2 λ3]; [r31 r32 r33]] + + # Inverse eigenvector matrix + d1 = (λ1 - λ2) * (λ1 - λ3) + d2 = (λ2 - λ1) * (λ2 - λ3) + d3 = (λ3 - λ2) * (λ3 - λ1) + R_inv = @SMatrix [(c1 - v_avg^2 + λ2 * λ3)/d1 (2 * v_avg - λ2 - λ3)/d1 c2/d1; + (c1 - v_avg^2 + λ1 * λ3)/d2 (2 * v_avg - λ1 - λ3)/d2 c2/d2; + (c1 - v_avg^2 + λ1 * λ2)/d3 (2 * v_avg - λ2 - λ1)/d3 c2/d3] + + # Eigenvalue absolute value matrix + Λ_abs = @SMatrix [abs(λ1) z z; z abs(λ2) z; z z abs(λ3)] + + # Compute dissipation + diss = SVector(-0.5 * R * Λ_abs * R_inv * (u_rr - u_ll)) + + return SVector(diss[1], diss[2], diss[3]) +end + +@inline function Trixi.max_abs_speed_naive(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterExnerEquations1D) + return max(eigvals_cardano(u_rr, equations)..., eigvals_cardano(u_ll, equations)...) +end + +@inline function Trixi.max_abs_speeds(u, equations::ShallowWaterExnerEquations1D) + return maximum(eigvals_cardano(u, equations)) +end + +#Helper function to extract the velocity vector from the conservative variables +@inline function Trixi.velocity(u, equations::ShallowWaterExnerEquations1D) + h, hv, _ = u + + v = hv / h + + return v +end + +# Compute the sediment discharge for Shields stress models +@inline function q_s(u, + equations::ShallowWaterExnerEquations1D{T, S, + ShieldsStressModel{T}}) where { + T, + S + } + (; gravity, rho_f, rho_s, porosity_inv) = equations + (; m_1, m_2, m_3, k_1, k_2, k_3, theta_c, d_s) = equations.sediment_model + + theta = rho_f * abs(shear_stress(u, equations)) / (gravity * (rho_s - rho_f) * d_s) # Shields stress + + Q = d_s * sqrt(gravity * (rho_s / rho_f - 1.0) * d_s) # Characteristic discharge + + return porosity_inv * Q * sign(theta) * k_1 * theta^m_1 * + (max(theta - k_2 * theta_c, 0.0))^m_2 * + (max(sqrt(theta) - k_3 * sqrt(theta_c), 0.0))^m_3 +end + +# Compute the sediment discharge for the Grass model +@inline function q_s(u, + equations::ShallowWaterExnerEquations1D{T, S, GrassModel{T}}) where { + T, + S + } + (; porosity_inv, sediment_model) = equations + return porosity_inv * sediment_model.A_g * velocity(u, equations)^3 +end + +# Shear stress formulation using a coefficient to take into account different friction models +@inline function shear_stress(u, equations::ShallowWaterExnerEquations1D) + v = velocity(u, equations) + return equations.gravity * shear_stress_coefficient(u, equations.friction) * v * + abs(v) +end + +# Model dependent shear stress coefficient +@inline function shear_stress_coefficient(u, friction::ManningFriction) + h, _, _ = u + return friction.n^2 / h^(1 / 3) +end + +# Convert conservative variables to primitive +@inline function Trixi.cons2prim(u, equations::ShallowWaterExnerEquations1D) + h, _, h_b = u + + H = h + h_b + v = velocity(u, equations) + + return SVector(H, v, h_b) +end + +# Convert conservative variables to entropy variables +@inline function Trixi.cons2entropy(u, equations::ShallowWaterExnerEquations1D) + h, _, h_b = u + (; gravity, r) = equations + + v = velocity(u, equations) + + w1 = r * (gravity * (h + h_b) - 0.5 * v^2) + w2 = r * v + w3 = gravity * (r * h + h_b) + + return SVector(w1, w2, w3) +end + +# Convert primitive to conservative variables +@inline function Trixi.prim2cons(prim, equations::ShallowWaterExnerEquations1D) + H, v, h_b = prim + + h = H - h_b + hv = h * v + + return SVector(h, hv, h_b) +end + +@inline function Trixi.waterheight(u, equations::ShallowWaterExnerEquations1D) + return u[1] +end + +# Indicator variable used for the shock capturing in `IndicatorHennemannGassnerShallowWater` +@inline function water_sediment_height(u, equations::ShallowWaterExnerEquations1D) + return equations.gravity * u[1] * u[3] +end + +# Entropy function +@inline function Trixi.entropy(u, equations::ShallowWaterExnerEquations1D) + h, _, h_b = u + v = velocity(u, equations) + (; gravity, r) = equations + + return 0.5 * r * h * v^2 + 0.5 * gravity * (r * h^2 + h_b^2) + r * gravity * h * h_b +end + +# Calculate the error for the "lake-at-rest" test case where H = h + h_b should +# be a constant value over time. +@inline function Trixi.lake_at_rest_error(u, equations::ShallowWaterExnerEquations1D) + h, _, h_b = u + return abs(equations.H0 - (h + h_b)) +end + +# Trigonometric version of Cardano's method to compute the eigenvalues of +# the [`ShallowWaterExnerEquations1D`[(@ref)] assuming only real roots. +@inline function eigvals_cardano(u, equations::ShallowWaterExnerEquations1D) + h, _, _ = u + v = velocity(u, equations) + g = equations.gravity + r = equations.r + + # Workaround to avoid division by zero, when computing the effective sediment height + if v > eps() + h_s = q_s(u, equations) / v + # Compute gradients of q_s using automatic differentiation. + # Introduces a closure to make q_s a function of u only. This is necessary since the + # gradient function only accepts functions of one variable. + dq_s_dh, dq_s_dhv, _ = Trixi.ForwardDiff.gradient(u -> q_s(u, equations), u) + else + h_s = 0.0 + dq_s_dh = 0.0 + dq_s_dhv = 0.0 + end + + # Coefficient for the original cubic equation ax^3 + bx^2 + cx + d + a = -1 + b = 2 * v + c = g * (h + 1 / r * h_s) * dq_s_dhv + g * (h + h_s) - v^2 + d = g * (h + 1 / r * h_s) * dq_s_dh + + # Coefficient of the depressed cubic equation t^3 + pt + q = 0 + p = (3 * a * c - b^2) / (3 * a^2) + q = (2 * b^3 - 9 * a * b * c + 27 * a^2 * d) / (27 * a^3) + + # Roots of the original cubic equation + λ1 = -b / (3 * a) + + 2 * sqrt(-p / 3) * cos(1 / 3 * acos(3 * q / (2 * p) * sqrt(-3 / p))) + λ2 = -b / (3 * a) + + 2 * sqrt(-p / 3) * + cos(1 / 3 * acos(3 * q / (2 * p) * sqrt(-3 / p)) - 2 * π * 1 / 3) + λ3 = -b / (3 * a) + + 2 * sqrt(-p / 3) * + cos(1 / 3 * acos(3 * q / (2 * p) * sqrt(-3 / p)) - 2 * π * 2 / 3) + + return SVector(λ1, λ2, λ3) +end +end # @muladd diff --git a/test/test_tree_1d.jl b/test/test_tree_1d.jl index d7cde02..d0b530b 100644 --- a/test/test_tree_1d.jl +++ b/test/test_tree_1d.jl @@ -823,6 +823,160 @@ end # 2LSWE end end end # MLSWE + +@testset "Shallow Water - Exner" begin + @trixi_testset "elixir_shallowwater_exner_source_terms.jl with EC fluxes/Grass" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_exner_source_terms.jl"), + l2=[ + 0.0004102960441666415, + 0.0024123111823754154, + 2.855259772927741e-5, + ], + linf=[ + 0.0008005791155958342, + 0.0075032017005062235, + 4.7151297207559395e-5, + ]) + # 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 + + @trixi_testset "elixir_shallowwater_exner_source_terms.jl with Roe/MPM" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_exner_source_terms.jl"), + l2=[ + 8.314469574617161e-5, + 0.00037371557342261787, + 2.8140625974847256e-5, + ], + linf=[ + 0.0003199264478970232, + 0.0017104276669870355, + 4.494237942265222e-5, + ], + surface_flux=(FluxPlusDissipation(flux_ersing_etal, + dissipation_roe), + flux_nonconservative_ersing_etal), + friction=ManningFriction(n = 0.01), + sediment_model=MeyerPeterMueller(theta_c = 0.0, + d_s = 1e-3)) + # 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 + + @trixi_testset "elixir_shallowwater_exner_source_terms.jl with LLF/MPM" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_exner_source_terms.jl"), + l2=[ + 8.494087939853228e-5, + 0.00037012479603853885, + 2.8139644904971178e-5, + ], + linf=[ + 0.0003106352175714644, + 0.0016775444674146378, + 4.4937259775723604e-5, + ], + surface_flux=(FluxPlusDissipation(flux_ersing_etal, + DissipationLocalLaxFriedrichs()), + flux_nonconservative_ersing_etal), + friction=ManningFriction(n = 0.01), + sediment_model=MeyerPeterMueller(theta_c = 0.0, + d_s = 1e-3)) + # 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 + + @trixi_testset "elixir_shallowwater_exner_well_balanced.jl" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_exner_well_balanced.jl"), + l2=[ + 0.0204167335836456, + 1.5412805715405497e-15, + 0.020416733583645628, + ], + linf=[ + 0.19999999999999907, + 2.957304143715833e-15, + 0.19999999999999998, + ]) + # 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 + + @trixi_testset "elixir_shallowwater_exner_channel.jl" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_exner_channel.jl"), + l2=[ + 0.061254948402258286, + 0.004292092112018592, + 0.06170746566026623, + ], + linf=[ + 0.555255659544267, + 0.009352017074599317, + 0.5499622869285615, + ]) + # 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 + + @trixi_testset "elixir_shallowwater_exner_channel.jl with EC fluxes" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_exner_channel.jl"), + l2=[ + 0.06511905620487073, + 0.021756041726745886, + 0.06488442680626008, + ], + linf=[ + 0.4644690482223428, + 0.058166305725594114, + 0.4604211411585378, + ], + surface_flux=(flux_ersing_etal, + flux_nonconservative_ersing_etal)) + # 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 # SWE-Exner end # TreeMesh1D # Clean up afterwards: delete TrixiShallowWater.jl output directory diff --git a/test/test_unit.jl b/test/test_unit.jl index 64002cf..e42157f 100644 --- a/test/test_unit.jl +++ b/test/test_unit.jl @@ -134,6 +134,74 @@ end end end +@timed_testset "SWE-Exner conversion between conservative/entropy variables" begin + h, v, h_b = (1.0, 0.3, 0.1) + + let equations = ShallowWaterExnerEquations1D(gravity_constant = 9.81, rho_f = 0.9, + rho_s = 1.0, porosity = 0.4, + sediment_model = GrassModel(A_g = 0.01)) + # Test conversion between primitive and conservative variables + prim_vars = SVector(h, v, h_b) + cons_vars = prim2cons(prim_vars, equations) + @test prim_vars ≈ cons2prim(cons_vars, equations) + + # Test conversion from conservative to entropy variables + entropy_vars = cons2entropy(cons_vars, equations) + @test entropy_vars ≈ + Trixi.ForwardDiff.gradient(u -> entropy(u, equations), cons_vars) + end +end + +@timed_testset "SWE-Exner eigenvalue / eigenvector computation" begin + h, v, h_b = (1.0, 0.3, 0.1) + + let equations = ShallowWaterExnerEquations1D(gravity_constant = 9.81, rho_f = 0.9, + rho_s = 1.0, porosity = 0.4, + sediment_model = GrassModel(A_g = 0.01)) + u = SVector(h, h * v, h_b) + r = equations.r + g = equations.gravity + + # Compute effective sediment height + h_s = TrixiShallowWater.q_s(SVector(h, h * v, 0.0), equations) / v + dq_s_dh, dq_s_dhv, _ = Trixi.ForwardDiff.gradient(u -> TrixiShallowWater.q_s(u, + equations), + u) + + # flux Jacobian + A = [0 1 0; (g * (h + h_s)-v^2) (2*v) (g*(h + 1 / equations.r * h_s)); + dq_s_dh dq_s_dhv 0] + + # Compute the eigenvalues using Cardano's formula + λ1, λ2, λ3 = TrixiShallowWater.eigvals_cardano(SVector(h, h * v, h_b), + equations) + + # Precompute some common expressions + c1 = g * (h + h_s) + c2 = g * (h + h_s / r) + + # Eigenvector matrix + r31 = ((v - λ1)^2 - c1) / c2 + r32 = ((v - λ2)^2 - c1) / c2 + r33 = ((v - λ3)^2 - c1) / c2 + R = [[1 1 1]; [λ1 λ2 λ3]; [r31 r32 r33]] + + # Inverse eigenvector matrix + d1 = (λ1 - λ2) * (λ1 - λ3) + d2 = (λ2 - λ1) * (λ2 - λ3) + d3 = (λ3 - λ2) * (λ3 - λ1) + R_inv = [(c1 - v^2 + λ2 * λ3)/d1 (2 * v - λ2 - λ3)/d1 c2/d1; + (c1 - v^2 + λ1 * λ3)/d2 (2 * v - λ1 - λ3)/d2 c2/d2; + (c1 - v^2 + λ1 * λ2)/d3 (2 * v - λ2 - λ1)/d3 c2/d3] + + # Eigenvalue vale matrix + Λ = [λ1 0 0; 0 λ2 0; 0 0 λ3] + + @test R * R_inv ≈ [1 0 0; 0 1 0; 0 0 1] + @test A ≈ R * Λ * R_inv + end +end + @timed_testset "Consistency check for waterheight_pressure" begin H, v1, v2, b = 3.5, 0.25, 0.1, 0.4 From 91e749dbaa1a731a23944bd513357080fb1459ae Mon Sep 17 00:00:00 2001 From: patrickersing Date: Fri, 2 Aug 2024 16:24:25 +0200 Subject: [PATCH 02/12] fix docstrings --- src/equations/shallow_water_exner_1d.jl | 52 ++++++++++++------------- 1 file changed, 26 insertions(+), 26 deletions(-) diff --git a/src/equations/shallow_water_exner_1d.jl b/src/equations/shallow_water_exner_1d.jl index 3a693b2..bc7319b 100644 --- a/src/equations/shallow_water_exner_1d.jl +++ b/src/equations/shallow_water_exner_1d.jl @@ -12,8 +12,8 @@ abstract type Friction{RealT} end ManningFriction(; n) Creates a Manning friction model for the bottom friction with Manning coefficient `n`. -The type is used to dispatch on the respective friction law -[`shear_stress_coefficient(u, friction::ManningFriction)`](@ref) when computing the [`shear_stress`](@ref). +The type is used to dispatch on the respective friction law through the +`shear_stress_coefficient` when computing the `shear_stress`. """ struct ManningFriction{RealT} <: Friction{RealT} n::RealT @@ -26,14 +26,14 @@ end # Abstract type for the different models to compute sediment discharge abstract type SedimentModel{RealT} end -""" +@doc raw""" ShieldsStressModel(; m_1, m_2, m_3, k_1, k_2, k_3, theta_c, d_s) -Create a Shields stress model to compute the sediment discharge [`q_s`](@ref) based on the generalized +Create a Shields stress model to compute the sediment discharge `q_s` based on the generalized formulation from equation (1.2) in the paper: -- E.D. Fernández-Nieto, T.M. de Luna, G. Narbona-Reina and J. de Dieu Zabsonré (2017) +- E.D. Fernández-Nieto, T.M. de Luna, G. Narbona-Reina and J. de Dieu Zabsonré (2017)\ Formal deduction of the Saint-Venant–Exner model including arbitrarily sloping sediment beds and - associated energy + associated energy\ [DOI: 10.1051/m2an/2016018](https://doi.org/10.1051/m2an/2016018) """ struct ShieldsStressModel{RealT} <: SedimentModel{RealT} @@ -47,19 +47,19 @@ struct ShieldsStressModel{RealT} <: SedimentModel{RealT} d_s::RealT # grain diameter end -""" +@doc raw""" GrassModel(; A_g, m_g=3) -Creates a Grass model to compute the sediment discharge [`q_s``](@ref) as +Creates a Grass model to compute the sediment discharge `q_s` as ```math -q_s = A_g v^m_g +q_s = A_g v^{m_g} ``` with the coefficients `A_g` and `m_g`. An overview of different formulations to compute the sediment discharge can be found in: -- M.J. Castro Díaz, E.D. Fernández-Nieto, A.M. Ferreiro (2008) +- M.J. Castro Díaz, E.D. Fernández-Nieto, A.M. Ferreiro (2008)\ Sediment transport models in Shallow Water equations and numerical approach by high order - finite volume methods + finite volume methods\ [DOI:10.1016/j.compfluid.2007.07.017](https://doi.org/10.1016/j.compfluid.2007.07.017) """ struct GrassModel{RealT} <: SedimentModel{RealT} @@ -71,16 +71,16 @@ function GrassModel(; A_g, m_g = 3.0) return GrassModel(A_g, m_g) end -""" +@doc raw""" MeyerPeterMueller(; theta_c, d_s) Creates a Meyer-Peter-Mueller model to compute the sediment discharge -[`q_s``](@ref) with the critical Shields stress `theta_c` and the grain diameter `d_s`. +`q_s` with the critical Shields stress `theta_c` and the grain diameter `d_s`. An overview of different formulations to compute the sediment discharge can be found in: -- M.J. Castro Díaz, E.D. Fernández-Nieto, A.M. Ferreiro (2008) +- M.J. Castro Díaz, E.D. Fernández-Nieto, A.M. Ferreiro (2008)\ Sediment transport models in Shallow Water equations and numerical approach by high order - finite volume methods + finite volume methods\ [DOI:10.1016/j.compfluid.2007.07.017](https://doi.org/10.1016/j.compfluid.2007.07.017) """ function MeyerPeterMueller(; theta_c, d_s) @@ -103,11 +103,11 @@ The equations are given by \end{cases} ``` The unknown quantities are the water and sediment height ``h``, ``h_b`` and the velocity ``v``. -The sediment discharge ``q_s(h, hv)`` is determined by the ``sediment_model`` and is used to determine +The sediment discharge ``q_s(h, hv)`` is determined by the `sediment_model` and is used to determine the active sediment height ``h_s = q_s / v``. Furthermore ``\tau`` denotes the shear stress at the water-sediment interface and is determined by -the ``friction`` model. -The gravitational constant is denoted by `g`, and ``rho_f`` and ``rho_s`` are the fluid and sediment +the `friction` model. +The gravitational constant is denoted by ``g``, and ``\rho_f`` and ``\rho_s`` are the fluid and sediment densities, respectively. The density ratio is given by ``r = \rho_f / \rho_s``. The conservative variable water height ``h`` is measured from the sediment height ``h_b``, therefore @@ -117,9 +117,9 @@ The additional quantity ``H_0`` is also available to store a reference value for height that is useful to set initial conditions or test the "lake-at-rest" well-balancedness. The entropy conservative formulation has been derived in the paper: -- E.D. Fernández-Nieto, T.M. de Luna, G. Narbona-Reina and J. de Dieu Zabsonré (2017) +- E.D. Fernández-Nieto, T.M. de Luna, G. Narbona-Reina and J. de Dieu Zabsonré (2017)\ Formal deduction of the Saint-Venant–Exner model including arbitrarily sloping sediment beds and - associated energy + associated energy\ [DOI: 10.1051/m2an/2016018](https://doi.org/10.1051/m2an/2016018) """ struct ShallowWaterExnerEquations1D{RealT <: Real, @@ -159,7 +159,7 @@ Trixi.varnames(::typeof(cons2cons), ::ShallowWaterExnerEquations1D) = ("h", "hv" # visualization and setting initial conditions Trixi.varnames(::typeof(cons2prim), ::ShallowWaterExnerEquations1D) = ("H", "v", "h_b") -""" +@doc raw""" boundary_condition_slip_wall(u_inner, orientation_or_normal, x, t, surface_flux_function, equations::ShallowWaterExnerEquations1D) @@ -168,9 +168,9 @@ the tangential velocity component unchanged. The boundary water height is taken the internal value. For details see Section 9.2.5 of the book: -- Eleuterio F. Toro (2001) - Shock-Capturing Methods for Free-Surface Shallow Flows - 1st edition +- Eleuterio F. Toro (2001)\ + Shock-Capturing Methods for Free-Surface Shallow Flows\ + 1st edition\ ISBN 0471987662 """ @inline function Trixi.boundary_condition_slip_wall(u_inner, orientation_or_normal, @@ -297,7 +297,7 @@ end """ dissipation_roe(u_ll, u_rr, orientation_or_normal_direction, equations::ShallowWaterExnerEquations1D) -Roe-type dissipation term for the [`ShallowWaterExnerEquations1D`] with an approximate Roe average +Roe-type dissipation term for the [`ShallowWaterExnerEquations1D`](@ref) with an approximate Roe average for the sediment discharge `q_s`. """ @inline function dissipation_roe(u_ll, u_rr, orientation_or_normal_direction, @@ -366,7 +366,7 @@ end return v end -# Compute the sediment discharge for Shields stress models +# Compute the sediment discharge for Shields stress models @inline function q_s(u, equations::ShallowWaterExnerEquations1D{T, S, ShieldsStressModel{T}}) where { From 23dcf500b9705634d20d3d120c40494696b438d3 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Fri, 2 Aug 2024 20:10:44 +0200 Subject: [PATCH 03/12] add zero velocity workaround to roe flux --- src/equations/shallow_water_exner_1d.jl | 7 ++++++- 1 file changed, 6 insertions(+), 1 deletion(-) diff --git a/src/equations/shallow_water_exner_1d.jl b/src/equations/shallow_water_exner_1d.jl index bc7319b..1b9b5ff 100644 --- a/src/equations/shallow_water_exner_1d.jl +++ b/src/equations/shallow_water_exner_1d.jl @@ -316,7 +316,12 @@ for the sediment discharge `q_s`. h_avg = 0.5 * (u_ll[1] + u_rr[1]) v_avg = (sqrt(u_ll[1]) * v_ll + sqrt(u_rr[1]) * v_rr) / (sqrt(u_ll[1]) + sqrt(u_rr[1])) - h_s_avg = q_s(SVector(h_avg, h_avg * v_avg, z), equations) / v_avg + # Workaround to avoid division by zero, when computing the effective sediment height + if v_avg < eps() + h_s_avg = 0.0 + else + h_s_avg = 0.5 * (q_s(u_ll, equations) / v_ll + q_s(u_rr, equations) / v_rr) + end # Compute the eigenvalues using Cardano's formula λ1, λ2, λ3 = eigvals_cardano(SVector(h_avg, h_avg * v_avg, h_s_avg), equations) From ecab38e2993d366fb216a28bb1a74e814c536a2f Mon Sep 17 00:00:00 2001 From: patrickersing Date: Fri, 2 Aug 2024 20:48:37 +0200 Subject: [PATCH 04/12] fix typo --- .../tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl index 95f6137..7c59ffc 100644 --- a/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl @@ -4,7 +4,7 @@ using Trixi using TrixiShallowWater ############################################################################### -# Semidiscretization of the SWE-Exner equations with a discontinous sediment bed +# Semidiscretization of the SWE-Exner equations with a discontinuous sediment bed # to test well-balancedness equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, H0 = 1.0, From d975b8dd39e6d85f9971481caddf35d98c12750b Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 12 Aug 2024 08:31:20 +0200 Subject: [PATCH 05/12] add roots --- Project.toml | 6 ++++-- test/Project.toml | 14 ++++++++------ 2 files changed, 12 insertions(+), 8 deletions(-) diff --git a/Project.toml b/Project.toml index 1871d4c..831c4d4 100644 --- a/Project.toml +++ b/Project.toml @@ -6,16 +6,18 @@ version = "0.1.0-pre" [deps] LinearAlgebra = "37e2e46d-f89d-539d-b4ee-838fcccc9c8e" MuladdMacro = "46d2c3a1-f734-5fdb-9937-b9b9aeba4221" +Printf = "de0858da-6303-5e67-8744-51eddeeeb8d7" +Roots = "f2b01f46-fcfa-551c-844a-d8ac1e96c665" Static = "aedffcd0-7271-4cad-89d0-dc628f76c6d3" StaticArrays = "90137ffa-7385-5640-81b9-e52037218182" Trixi = "a7f1ee26-1774-49b1-8366-f1abc58fbfcb" -Printf = "de0858da-6303-5e67-8744-51eddeeeb8d7" [compat] LinearAlgebra = "1" MuladdMacro = "0.2.2" +Printf = "1" +Roots = "2.1.6" Static = "0.3, 0.4, 0.5, 0.6, 0.7, 0.8" StaticArrays = "1" Trixi = "0.7, 0.8" julia = "1.8" -Printf = "1" diff --git a/test/Project.toml b/test/Project.toml index c45b5c0..172eaa4 100644 --- a/test/Project.toml +++ b/test/Project.toml @@ -1,16 +1,18 @@ [deps] -Test = "8dfed614-e22c-5e08-85e1-65c5234f0b40" -Trixi = "a7f1ee26-1774-49b1-8366-f1abc58fbfcb" -OrdinaryDiffEq = "1dea7af3-3e70-54e6-95c3-0bf5283fa5ed" Downloads = "f43a241f-c20a-4ad4-852c-f6b1247861c6" +OrdinaryDiffEq = "1dea7af3-3e70-54e6-95c3-0bf5283fa5ed" Printf = "de0858da-6303-5e67-8744-51eddeeeb8d7" +Roots = "f2b01f46-fcfa-551c-844a-d8ac1e96c665" +Test = "8dfed614-e22c-5e08-85e1-65c5234f0b40" +Trixi = "a7f1ee26-1774-49b1-8366-f1abc58fbfcb" [compat] -Test = "1" -Trixi = "0.7, 0.8" -OrdinaryDiffEq = "6.49.1" Downloads = "1" +OrdinaryDiffEq = "6.49.1" Printf = "1" +Roots = "2.1.6" +Test = "1" +Trixi = "0.7, 0.8" [preferences.OrdinaryDiffEq] PrecompileAutoSpecialize = false From 9973b06635bc3c98a76ea33332df7358514e4ff5 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 12 Aug 2024 08:40:49 +0200 Subject: [PATCH 06/12] fix convergence tests --- .../elixir_shallowwater_exner_source_terms.jl | 46 +++++++++---------- 1 file changed, 21 insertions(+), 25 deletions(-) diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl index 02ce689..eb3dcc5 100644 --- a/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl @@ -5,20 +5,11 @@ using TrixiShallowWater ############################################################################### # Semidiscretization of the SWE-Exner equations with source terms for convergence testing -# Equations with MeyerPeterMueller model -equations_mpm = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, - rho_s = 1.0, porosity = 0.5, - friction = ManningFriction(n = 0.01), - sediment_model = MeyerPeterMueller(theta_c = 0.0, - d_s = 1e-3)) - # Equations with Grass model -equations_grass = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, - rho_s = 1.0, porosity = 0.5, - friction = ManningFriction(n = 0.0), - sediment_model = GrassModel(A_g = 0.01)) - -equations = equations_grass +equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, + rho_s = 1.0, porosity = 0.5, + friction = ManningFriction(n = 0.0), + sediment_model = GrassModel(A_g = 0.01)) # Smooth initial condition to test convergence @inline function Trixi.initial_condition_convergence_test(x, t, @@ -32,12 +23,15 @@ equations = equations_grass return SVector(h, h * v, h_b) end -""" - source_terms_convergence_test(u, x, t, equations::ShallowWaterExnerEquations1D{T, S, ShieldsStressModel{T}}) where {T, S}) - -Source terms used for convergence tests in combination with [`Trixi.initial_condition_convergence_test`](@extref) -when using the [`MeyerPeterMueller`](@ref) model. -""" +# Source terms used for convergence tests in combination with [`Trixi.initial_condition_convergence_test`](@extref) +# when using the [`MeyerPeterMueller`](@ref) model. +# To use this source term the equations must be set to: +# +# equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, +# rho_s = 1.0, porosity = 0.5, +# friction = ManningFriction(n = 0.01), +# sediment_model = MeyerPeterMueller(theta_c = 0.0, +# d_s = 1e-3)) @inline function Trixi.source_terms_convergence_test(u, x, t, equations::ShallowWaterExnerEquations1D{T, S, @@ -70,12 +64,14 @@ when using the [`MeyerPeterMueller`](@ref) model. return SVector(h, hv, h_b) end -""" - source_terms_convergence_test(u, x, t, equations::ShallowWaterExnerEquations1D{T, S, GrassModel{T}}) where {T, S}) - -Source terms used for convergence tests in combination with [`initial_condition_convergence_test`](@extref) -when using the the [`GrassModel`](@ref) model. -""" +# Source terms used for convergence tests in combination with [`initial_condition_convergence_test`](@extref) +# when using the the [`GrassModel`](@ref) model. +# To use this source term the equations must be set to: +# +# equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, +# rho_s = 1.0, porosity = 0.5, +# friction = ManningFriction(n = 0.0), +# sediment_model = GrassModel(A_g = 0.01) @inline function Trixi.source_terms_convergence_test(u, x, t, equations::ShallowWaterExnerEquations1D{T, S, From a265e6ebd17cf5dba5c433665a66d340b1a6d934 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 12 Aug 2024 09:15:09 +0200 Subject: [PATCH 07/12] fix convergence tests 2 --- ...r_shallowwater_exner_source_terms_grass.jl | 109 ++++++++++++++++++ ...ir_shallowwater_exner_source_terms_mpm.jl} | 35 +----- test/test_tree_1d.jl | 22 ++-- 3 files changed, 121 insertions(+), 45 deletions(-) create mode 100644 examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_grass.jl rename examples/tree_1d_dgsem/{elixir_shallowwater_exner_source_terms.jl => elixir_shallowwater_exner_source_terms_mpm.jl} (73%) diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_grass.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_grass.jl new file mode 100644 index 0000000..9da9326 --- /dev/null +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_grass.jl @@ -0,0 +1,109 @@ +using OrdinaryDiffEq +using Trixi +using TrixiShallowWater + +############################################################################### +# Semidiscretization of the SWE-Exner equations with source terms for convergence testing + +# Equations with Grass model +equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, + rho_s = 1.0, porosity = 0.5, + friction = ManningFriction(n = 0.0), + sediment_model = GrassModel(A_g = 0.01)) + +# Smooth initial condition to test convergence +@inline function Trixi.initial_condition_convergence_test(x, t, + equations::ShallowWaterExnerEquations1D) + ω = sqrt(2) * pi + + h = 2.0 + cos(ω * x[1]) * cos(ω * t) + v = 0.5 + h_b = 2.0 + sin(ω * x[1]) * cos(ω * t) + + return SVector(h, h * v, h_b) +end + +# Source terms used for convergence tests in combination with [`initial_condition_convergence_test`](@extref) +# when using the the [`GrassModel`](@ref) model. +# To use this source term the equations must be set to: +# +# equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, +# rho_s = 1.0, porosity = 0.5, +# friction = ManningFriction(n = 0.0), +# sediment_model = GrassModel(A_g = 0.01) +@inline function Trixi.source_terms_convergence_test(u, x, t, + equations::ShallowWaterExnerEquations1D{T, + S, + GrassModel{T}}) where { + T, + S + } + ω = sqrt(2.0) * pi + A_g = equations.sediment_model.A_g + + h = -cos(x[1] * ω) * sin(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω + hv = -0.5 * cos(x[1] * ω) * sin(t * ω) * ω - 0.25 * sin(x[1] * ω) * cos(t * ω) * ω + + 10.0 * A_g * + (cos(x[1] * ω) * cos(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω) + + 10.0 * (2.0 + cos(x[1] * ω) * cos(t * ω)) * + (cos(x[1] * ω) * cos(t * ω) * ω - sin(x[1] * ω) * cos(t * ω) * ω) + h_b = -sin(x[1] * ω) * sin(t * ω) * ω + return SVector(h, hv, h_b) +end + +initial_condition = initial_condition_convergence_test + +############################################################################### +# Get the DG approximation space + +volume_flux = (flux_ersing_etal, flux_nonconservative_ersing_etal) +surface_flux = (flux_ersing_etal, flux_nonconservative_ersing_etal) + +solver = DGSEM(polydeg = 4, + surface_flux = surface_flux, + volume_integral = VolumeIntegralFluxDifferencing(volume_flux)) + +############################################################################### +# Get the TreeMesh and setup a periodic mesh + +coordinates_min = 0.0 +coordinates_max = sqrt(2.0) +mesh = TreeMesh(coordinates_min, coordinates_max, + initial_refinement_level = 2, + n_cells_max = 10_000, + periodicity = true) + +# create the semi discretization object +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver, + source_terms = source_terms_convergence_test) + +############################################################################### +# ODE solvers, callbacks etc. + +tspan = (0.0, 1.0) +ode = semidiscretize(semi, tspan) + +summary_callback = SummaryCallback() + +analysis_interval = 200 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +stepsize_callback = StepsizeCallback(cfl = 1.0) + +save_solution = SaveSolutionCallback(interval = 200, + save_initial_solution = true, + save_final_solution = true) + +callbacks = CallbackSet(summary_callback, analysis_callback, alive_callback, + stepsize_callback, save_solution) + +############################################################################### +# run the simulation + +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); + +summary_callback() # print the timer summary diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_mpm.jl similarity index 73% rename from examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl rename to examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_mpm.jl index eb3dcc5..32d7fa8 100644 --- a/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms.jl +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_mpm.jl @@ -5,11 +5,12 @@ using TrixiShallowWater ############################################################################### # Semidiscretization of the SWE-Exner equations with source terms for convergence testing -# Equations with Grass model +# Equations with Meyer-Peter-Mueller model equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, rho_s = 1.0, porosity = 0.5, - friction = ManningFriction(n = 0.0), - sediment_model = GrassModel(A_g = 0.01)) + friction = ManningFriction(n = 0.01), + sediment_model = MeyerPeterMueller(theta_c = 0.0, + d_s = 1e-3)) # Smooth initial condition to test convergence @inline function Trixi.initial_condition_convergence_test(x, t, @@ -64,34 +65,6 @@ end return SVector(h, hv, h_b) end -# Source terms used for convergence tests in combination with [`initial_condition_convergence_test`](@extref) -# when using the the [`GrassModel`](@ref) model. -# To use this source term the equations must be set to: -# -# equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, -# rho_s = 1.0, porosity = 0.5, -# friction = ManningFriction(n = 0.0), -# sediment_model = GrassModel(A_g = 0.01) -@inline function Trixi.source_terms_convergence_test(u, x, t, - equations::ShallowWaterExnerEquations1D{T, - S, - GrassModel{T}}) where { - T, - S - } - ω = sqrt(2.0) * pi - A_g = equations.sediment_model.A_g - - h = -cos(x[1] * ω) * sin(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω - hv = -0.5 * cos(x[1] * ω) * sin(t * ω) * ω - 0.25 * sin(x[1] * ω) * cos(t * ω) * ω + - 10.0 * A_g * - (cos(x[1] * ω) * cos(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω) + - 10.0 * (2.0 + cos(x[1] * ω) * cos(t * ω)) * - (cos(x[1] * ω) * cos(t * ω) * ω - sin(x[1] * ω) * cos(t * ω) * ω) - h_b = -sin(x[1] * ω) * sin(t * ω) * ω - return SVector(h, hv, h_b) -end - initial_condition = initial_condition_convergence_test ############################################################################### diff --git a/test/test_tree_1d.jl b/test/test_tree_1d.jl index d0b530b..f96df5c 100644 --- a/test/test_tree_1d.jl +++ b/test/test_tree_1d.jl @@ -825,9 +825,9 @@ end # 2LSWE end # MLSWE @testset "Shallow Water - Exner" begin - @trixi_testset "elixir_shallowwater_exner_source_terms.jl with EC fluxes/Grass" begin + @trixi_testset "elixir_shallowwater_exner_source_terms_grass.jl with EC fluxes" begin @test_trixi_include(joinpath(EXAMPLES_DIR, - "elixir_shallowwater_exner_source_terms.jl"), + "elixir_shallowwater_exner_source_terms_grass.jl"), l2=[ 0.0004102960441666415, 0.0024123111823754154, @@ -848,9 +848,9 @@ end # MLSWE end end - @trixi_testset "elixir_shallowwater_exner_source_terms.jl with Roe/MPM" begin + @trixi_testset "elixir_shallowwater_exner_source_terms_mpm.jl with Roe dissipation" begin @test_trixi_include(joinpath(EXAMPLES_DIR, - "elixir_shallowwater_exner_source_terms.jl"), + "elixir_shallowwater_exner_source_terms_mpm.jl"), l2=[ 8.314469574617161e-5, 0.00037371557342261787, @@ -863,10 +863,7 @@ end # MLSWE ], surface_flux=(FluxPlusDissipation(flux_ersing_etal, dissipation_roe), - flux_nonconservative_ersing_etal), - friction=ManningFriction(n = 0.01), - sediment_model=MeyerPeterMueller(theta_c = 0.0, - d_s = 1e-3)) + flux_nonconservative_ersing_etal)) # Ensure that we do not have excessive memory allocations # (e.g., from type instabilities) let @@ -877,9 +874,9 @@ end # MLSWE end end - @trixi_testset "elixir_shallowwater_exner_source_terms.jl with LLF/MPM" begin + @trixi_testset "elixir_shallowwater_exner_source_terms_mpm.jl with LLF dissipation" begin @test_trixi_include(joinpath(EXAMPLES_DIR, - "elixir_shallowwater_exner_source_terms.jl"), + "elixir_shallowwater_exner_source_terms_mpm.jl"), l2=[ 8.494087939853228e-5, 0.00037012479603853885, @@ -892,10 +889,7 @@ end # MLSWE ], surface_flux=(FluxPlusDissipation(flux_ersing_etal, DissipationLocalLaxFriedrichs()), - flux_nonconservative_ersing_etal), - friction=ManningFriction(n = 0.01), - sediment_model=MeyerPeterMueller(theta_c = 0.0, - d_s = 1e-3)) + flux_nonconservative_ersing_etal)) # Ensure that we do not have excessive memory allocations # (e.g., from type instabilities) let From d2be6c5111d9d325dd080dae0bb5d2d900a1ddb8 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 12 Aug 2024 09:49:13 +0200 Subject: [PATCH 08/12] fix convergence tests 3 --- ...r_shallowwater_exner_source_terms_grass.jl | 40 ------- ...xir_shallowwater_exner_source_terms_mpm.jl | 53 ---------- src/equations/shallow_water_exner_1d.jl | 100 ++++++++++++++++++ 3 files changed, 100 insertions(+), 93 deletions(-) diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_grass.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_grass.jl index 9da9326..4389ebe 100644 --- a/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_grass.jl +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_grass.jl @@ -11,46 +11,6 @@ equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, friction = ManningFriction(n = 0.0), sediment_model = GrassModel(A_g = 0.01)) -# Smooth initial condition to test convergence -@inline function Trixi.initial_condition_convergence_test(x, t, - equations::ShallowWaterExnerEquations1D) - ω = sqrt(2) * pi - - h = 2.0 + cos(ω * x[1]) * cos(ω * t) - v = 0.5 - h_b = 2.0 + sin(ω * x[1]) * cos(ω * t) - - return SVector(h, h * v, h_b) -end - -# Source terms used for convergence tests in combination with [`initial_condition_convergence_test`](@extref) -# when using the the [`GrassModel`](@ref) model. -# To use this source term the equations must be set to: -# -# equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, -# rho_s = 1.0, porosity = 0.5, -# friction = ManningFriction(n = 0.0), -# sediment_model = GrassModel(A_g = 0.01) -@inline function Trixi.source_terms_convergence_test(u, x, t, - equations::ShallowWaterExnerEquations1D{T, - S, - GrassModel{T}}) where { - T, - S - } - ω = sqrt(2.0) * pi - A_g = equations.sediment_model.A_g - - h = -cos(x[1] * ω) * sin(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω - hv = -0.5 * cos(x[1] * ω) * sin(t * ω) * ω - 0.25 * sin(x[1] * ω) * cos(t * ω) * ω + - 10.0 * A_g * - (cos(x[1] * ω) * cos(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω) + - 10.0 * (2.0 + cos(x[1] * ω) * cos(t * ω)) * - (cos(x[1] * ω) * cos(t * ω) * ω - sin(x[1] * ω) * cos(t * ω) * ω) - h_b = -sin(x[1] * ω) * sin(t * ω) * ω - return SVector(h, hv, h_b) -end - initial_condition = initial_condition_convergence_test ############################################################################### diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_mpm.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_mpm.jl index 32d7fa8..9022b7e 100644 --- a/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_mpm.jl +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_source_terms_mpm.jl @@ -12,59 +12,6 @@ equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, sediment_model = MeyerPeterMueller(theta_c = 0.0, d_s = 1e-3)) -# Smooth initial condition to test convergence -@inline function Trixi.initial_condition_convergence_test(x, t, - equations::ShallowWaterExnerEquations1D) - ω = sqrt(2) * pi - - h = 2.0 + cos(ω * x[1]) * cos(ω * t) - v = 0.5 - h_b = 2.0 + sin(ω * x[1]) * cos(ω * t) - - return SVector(h, h * v, h_b) -end - -# Source terms used for convergence tests in combination with [`Trixi.initial_condition_convergence_test`](@extref) -# when using the [`MeyerPeterMueller`](@ref) model. -# To use this source term the equations must be set to: -# -# equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, -# rho_s = 1.0, porosity = 0.5, -# friction = ManningFriction(n = 0.01), -# sediment_model = MeyerPeterMueller(theta_c = 0.0, -# d_s = 1e-3)) -@inline function Trixi.source_terms_convergence_test(u, x, t, - equations::ShallowWaterExnerEquations1D{T, - S, - ShieldsStressModel{T}}) where { - T, - S - } - ω = sqrt(2.0) * pi - (; gravity, porosity_inv, rho_f, rho_s, r) = equations - - n = equations.friction.n - - # Constant expression from the MPM model - c = sqrt(gravity * (1 / r - 1)) * 8.0 * porosity_inv * - (rho_f / (rho_s - rho_f))^(3 / 2) * n^3 - - h = -cos(x[1] * ω) * sin(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω - - hv = ((5.0 * c * - (cos(x[1] * ω) * cos(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω)) / - ((2.0 + cos(x[1] * ω) * cos(t * ω))^0.5) - 0.5 * cos(x[1] * ω) * sin(t * ω) * ω - - 0.25 * sin(x[1] * ω) * cos(t * ω) * ω + - 10.0 * (2.0 + cos(x[1] * ω) * cos(t * ω)) * - (cos(x[1] * ω) * cos(t * ω) * ω - sin(x[1] * ω) * cos(t * ω) * ω)) - - h_b = ((0.5 * ((0.125 * c) / (2.0 + cos(x[1] * ω) * cos(t * ω))) * sin(x[1] * ω) * - cos(t * ω) * ω) / ((2.0 + cos(x[1] * ω) * cos(t * ω))^0.5) - - sin(x[1] * ω) * sin(t * ω) * ω) - - return SVector(h, hv, h_b) -end - initial_condition = initial_condition_convergence_test ############################################################################### diff --git a/src/equations/shallow_water_exner_1d.jl b/src/equations/shallow_water_exner_1d.jl index 1b9b5ff..fdf6175 100644 --- a/src/equations/shallow_water_exner_1d.jl +++ b/src/equations/shallow_water_exner_1d.jl @@ -195,6 +195,106 @@ For details see Section 9.2.5 of the book: return flux end +# Set initial conditions at physical location `x` for time `t` +""" + initial_condition_convergence_test(x, t, equations::ShallowWaterExnerEquations1D) + +A smooth initial condition used for convergence tests in combination with +[`Trixi.source_terms_convergence_test`](@ref). +""" +@inline function Trixi.initial_condition_convergence_test(x, t, + equations::ShallowWaterExnerEquations1D) + ω = sqrt(2) * pi + + h = 2.0 + cos(ω * x[1]) * cos(ω * t) + v = 0.5 + h_b = 2.0 + sin(ω * x[1]) * cos(ω * t) + + return SVector(h, h * v, h_b) +end + +""" + source_terms_convergence_test(u, x, t, equations::ShallowWaterExnerEquations1D{T, S, GrassModel{T}}) where {T, S} + +Source terms used for convergence tests in combination with [`Trixi.initial_condition_convergence_test`](@ref) +when using the the [`GrassModel`](@ref) model. + +To use this source term the equations must be set to: +```julia +equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, + rho_s = 1.0, porosity = 0.5, + friction = ManningFriction(n = 0.0), + sediment_model = GrassModel(A_g = 0.01) +``` +""" +@inline function Trixi.source_terms_convergence_test(u, x, t, + equations::ShallowWaterExnerEquations1D{T, + S, + GrassModel{T}}) where { + T, + S + } + ω = sqrt(2.0) * pi + A_g = equations.sediment_model.A_g + + h = -cos(x[1] * ω) * sin(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω + hv = -0.5 * cos(x[1] * ω) * sin(t * ω) * ω - 0.25 * sin(x[1] * ω) * cos(t * ω) * ω + + 10.0 * A_g * + (cos(x[1] * ω) * cos(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω) + + 10.0 * (2.0 + cos(x[1] * ω) * cos(t * ω)) * + (cos(x[1] * ω) * cos(t * ω) * ω - sin(x[1] * ω) * cos(t * ω) * ω) + h_b = -sin(x[1] * ω) * sin(t * ω) * ω + return SVector(h, hv, h_b) +end + +""" + source_terms_convergence_test(u, x, t, equations::ShallowWaterExnerEquations1D{T, S, ShieldsStressModel{T}}) where {T, S} + +Source terms used for convergence tests in combination with [`Trixi.initial_condition_convergence_test`](@ref) +when using the [`MeyerPeterMueller`](@ref) model. + +To use this source term the equations must be set to: +```julia +equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, rho_f = 0.5, + rho_s = 1.0, porosity = 0.5, + friction = ManningFriction(n = 0.01), + sediment_model = MeyerPeterMueller(theta_c = 0.0, + d_s = 1e-3)) +``` +""" +@inline function Trixi.source_terms_convergence_test(u, x, t, + equations::ShallowWaterExnerEquations1D{T, + S, + ShieldsStressModel{T}}) where { + T, + S + } + ω = sqrt(2.0) * pi + (; gravity, porosity_inv, rho_f, rho_s, r) = equations + + n = equations.friction.n + + # Constant expression from the MPM model + c = sqrt(gravity * (1 / r - 1)) * 8.0 * porosity_inv * + (rho_f / (rho_s - rho_f))^(3 / 2) * n^3 + + h = -cos(x[1] * ω) * sin(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω + + hv = ((5.0 * c * + (cos(x[1] * ω) * cos(t * ω) * ω - 0.5 * sin(x[1] * ω) * cos(t * ω) * ω)) / + ((2.0 + cos(x[1] * ω) * cos(t * ω))^0.5) - + 0.5 * cos(x[1] * ω) * sin(t * ω) * ω - + 0.25 * sin(x[1] * ω) * cos(t * ω) * ω + + 10.0 * (2.0 + cos(x[1] * ω) * cos(t * ω)) * + (cos(x[1] * ω) * cos(t * ω) * ω - sin(x[1] * ω) * cos(t * ω) * ω)) + + h_b = ((0.5 * ((0.125 * c) / (2.0 + cos(x[1] * ω) * cos(t * ω))) * sin(x[1] * ω) * + cos(t * ω) * ω) / ((2.0 + cos(x[1] * ω) * cos(t * ω))^0.5) - + sin(x[1] * ω) * sin(t * ω) * ω) + + return SVector(h, hv, h_b) +end + """ source_term_bottom_friction(u, x, t, equations::ShallowWaterExnerEquations1D) From aec40d82939298cbd58df446bfbb246f6437ffc3 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 12 Aug 2024 10:11:18 +0200 Subject: [PATCH 09/12] update test_tree_1d reference values --- test/test_tree_1d.jl | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/test/test_tree_1d.jl b/test/test_tree_1d.jl index f96df5c..ff23d05 100644 --- a/test/test_tree_1d.jl +++ b/test/test_tree_1d.jl @@ -927,14 +927,14 @@ end # MLSWE @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_exner_channel.jl"), l2=[ - 0.061254948402258286, - 0.004292092112018592, - 0.06170746566026623, + 0.061254947613784645, + 0.0042920880585939165, + 0.06170746499938789, ], linf=[ - 0.555255659544267, - 0.009352017074599317, - 0.5499622869285615, + 0.5552555774807875, + 0.009352028888004682, + 0.549962205546136, ]) # Ensure that we do not have excessive memory allocations # (e.g., from type instabilities) From 025ccc0b31c6242172886674036250dd927cdc13 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 12 Aug 2024 10:23:06 +0200 Subject: [PATCH 10/12] add tests to increase coverage --- src/equations/shallow_water_exner_1d.jl | 2 +- test/test_tree_1d.jl | 40 ++++++++++++++++++++----- 2 files changed, 34 insertions(+), 8 deletions(-) diff --git a/src/equations/shallow_water_exner_1d.jl b/src/equations/shallow_water_exner_1d.jl index fdf6175..9e823d6 100644 --- a/src/equations/shallow_water_exner_1d.jl +++ b/src/equations/shallow_water_exner_1d.jl @@ -575,7 +575,7 @@ end # Trigonometric version of Cardano's method to compute the eigenvalues of # the [`ShallowWaterExnerEquations1D`[(@ref)] assuming only real roots. @inline function eigvals_cardano(u, equations::ShallowWaterExnerEquations1D) - h, _, _ = u + h = waterheight(u, equations) v = velocity(u, equations) g = equations.gravity r = equations.r diff --git a/test/test_tree_1d.jl b/test/test_tree_1d.jl index ff23d05..64752ac 100644 --- a/test/test_tree_1d.jl +++ b/test/test_tree_1d.jl @@ -852,16 +852,16 @@ end # MLSWE @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_exner_source_terms_mpm.jl"), l2=[ - 8.314469574617161e-5, - 0.00037371557342261787, - 2.8140625974847256e-5, + 8.314340397306541e-5, + 0.0003737050980420925, + 2.81406288308791e-5, ], linf=[ - 0.0003199264478970232, - 0.0017104276669870355, - 4.494237942265222e-5, + 0.000319905497986106, + 0.001710420951723135, + 4.494237163465975e-5, ], - surface_flux=(FluxPlusDissipation(flux_ersing_etal, + surface_flux=(FluxPlusDissipation(flux_central, dissipation_roe), flux_nonconservative_ersing_etal)) # Ensure that we do not have excessive memory allocations @@ -923,6 +923,32 @@ end # MLSWE end end + @trixi_testset "elixir_shallowwater_exner_well_balanced.jl with Roe dissipation" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_exner_well_balanced.jl"), + l2=[ + 0.010910894511791577, + 1.8877123431891935e-15, + 0.010910894511791757, + ], + linf=[ + 0.19999999999999674, + 3.752915460516524e-15, + 0.19999999999999998, + ], + surface_flux=(FluxPlusDissipation(flux_ersing_etal, + dissipation_roe), + flux_nonconservative_ersing_etal)) + # 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 + @trixi_testset "elixir_shallowwater_exner_channel.jl" begin @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_exner_channel.jl"), From 4ab15d37fd6b19bea815b894874995c10c188025 Mon Sep 17 00:00:00 2001 From: Patrick Ersing <114223904+patrickersing@users.noreply.github.com> Date: Tue, 13 Aug 2024 12:36:03 +0200 Subject: [PATCH 11/12] Apply suggestions from code review Co-authored-by: Andrew Winters --- .../elixir_shallowwater_exner_channel.jl | 4 +-- ...elixir_shallowwater_exner_well_balanced.jl | 1 + src/equations/shallow_water_exner_1d.jl | 33 ++++++++++++++----- 3 files changed, 27 insertions(+), 11 deletions(-) diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_channel.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_channel.jl index f6cc9ac..a62bc4d 100644 --- a/examples/tree_1d_dgsem/elixir_shallowwater_exner_channel.jl +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_channel.jl @@ -85,7 +85,7 @@ semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver) ############################################################################### # ODE solver -tspan = (0.0, 30000) +tspan = (0.0, 30_000.0) ode = semidiscretize(semi, tspan) ############################################################################### @@ -100,7 +100,7 @@ alive_callback = AliveCallback(analysis_interval = analysis_interval) stepsize_callback = StepsizeCallback(cfl = 0.8) -save_solution = SaveSolutionCallback(dt = 10000, +save_solution = SaveSolutionCallback(dt = 10_000.0, save_initial_solution = true, save_final_solution = true) diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl b/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl index 7c59ffc..facc9b6 100644 --- a/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl +++ b/examples/tree_1d_dgsem/elixir_shallowwater_exner_well_balanced.jl @@ -7,6 +7,7 @@ using TrixiShallowWater # Semidiscretization of the SWE-Exner equations with a discontinuous sediment bed # to test well-balancedness +# Equations with Meyer-Peter-Mueller sedimentation model equations = ShallowWaterExnerEquations1D(gravity_constant = 10.0, H0 = 1.0, rho_f = 0.5, rho_s = 1.0, porosity = 0.5, friction = ManningFriction(n = 0.01), diff --git a/src/equations/shallow_water_exner_1d.jl b/src/equations/shallow_water_exner_1d.jl index 9e823d6..35884a4 100644 --- a/src/equations/shallow_water_exner_1d.jl +++ b/src/equations/shallow_water_exner_1d.jl @@ -30,7 +30,15 @@ abstract type SedimentModel{RealT} end ShieldsStressModel(; m_1, m_2, m_3, k_1, k_2, k_3, theta_c, d_s) Create a Shields stress model to compute the sediment discharge `q_s` based on the generalized -formulation from equation (1.2) in the paper: +formulation from equation (1.2) in the given reference. + +The choice of the real constants ´m_1`, ´m_2`, ´m_3`, ´k_1`, ´k_2`, and ´k_3` creates +different models. For example, setting `m_1=0`, `m_2=1.5`, `m_3=0`, `k_1=8`, `k_2=1`, and `k_3=0` +yields the sedimentation model of Meyer-Peter and Müller as given in [`MeyerPeterMueller`](@ref) below. +The Shields stress represents the ratio of agitating and stabilizing forces in the sediment bed where +`theta_c` is the critical Shields stress for incipient motion and `d_s` is the mean diameter of +the sediment grain size. + - E.D. Fernández-Nieto, T.M. de Luna, G. Narbona-Reina and J. de Dieu Zabsonré (2017)\ Formal deduction of the Saint-Venant–Exner model including arbitrarily sloping sediment beds and associated energy\ @@ -55,6 +63,12 @@ Creates a Grass model to compute the sediment discharge `q_s` as q_s = A_g v^{m_g} ``` with the coefficients `A_g` and `m_g`. +with the coefficients `A_g` and `m_g`. The constant `A_g` lies in the interval ``[0,1]`` +and is a dimensional calibration constant that is usually measured experimental. It expresses +the kind of interaction between the fluid and the sediment the strength of which +increases as `A_g` approaches to 1. The factor `m_g` lies in the interval ``[1, 4]``. +Typically, one considers an odd integer value for `m_g` such that the sediment discharge +`q_s` can be differentiated and the model remains valid for all values of the velocity `v`. An overview of different formulations to compute the sediment discharge can be found in: - M.J. Castro Díaz, E.D. Fernández-Nieto, A.M. Ferreiro (2008)\ @@ -147,7 +161,7 @@ function ShallowWaterExnerEquations1D(; gravity_constant, H0 = zero(gravity_cons sediment_model, porosity, rho_f, rho_s) # Precompute common expressions for the porosity and density ratio - porosity_inv = 1 / (1 - porosity) + porosity_inv = inv(1 - porosity) r = rho_f / rho_s return ShallowWaterExnerEquations1D(gravity_constant, H0, friction, sediment_model, porosity_inv, rho_f, rho_s, r) @@ -155,7 +169,7 @@ end Trixi.have_nonconservative_terms(::ShallowWaterExnerEquations1D) = True() Trixi.varnames(::typeof(cons2cons), ::ShallowWaterExnerEquations1D) = ("h", "hv", "h_b") -# Note, we use the total water height, H = h + h_s, as the first primitive variable for easier +# Note, we use the total water height, H = h + h_b, as the first primitive variable for easier # visualization and setting initial conditions Trixi.varnames(::typeof(cons2prim), ::ShallowWaterExnerEquations1D) = ("H", "v", "h_b") @@ -309,7 +323,6 @@ The actual friction law is determined through the friction model in `equations.f end # Calculate 1D flux for a single point -# Note, the bottom topography has no flux @inline function Trixi.flux(u, orientation::Integer, equations::ShallowWaterExnerEquations1D) _, hv, _ = u @@ -412,7 +425,8 @@ for the sediment discharge `q_s`. # Compute approximate Roe averages. # The actual Roe average for the sediment discharge `q_s` would depend on the sediment and - # friction model and can be hard to find. Therefore we only use an approximation here. + # friction model and can be difficult to compute analytically. + # Therefore we only use an approximation here. h_avg = 0.5 * (u_ll[1] + u_rr[1]) v_avg = (sqrt(u_ll[1]) * v_ll + sqrt(u_rr[1]) * v_rr) / (sqrt(u_ll[1]) + sqrt(u_rr[1])) @@ -485,9 +499,9 @@ end Q = d_s * sqrt(gravity * (rho_s / rho_f - 1.0) * d_s) # Characteristic discharge - return porosity_inv * Q * sign(theta) * k_1 * theta^m_1 * + return (porosity_inv * Q * sign(theta) * k_1 * theta^m_1 * (max(theta - k_2 * theta_c, 0.0))^m_2 * - (max(sqrt(theta) - k_3 * sqrt(theta_c), 0.0))^m_3 + (max(sqrt(theta) - k_3 * sqrt(theta_c), 0.0))^m_3) end # Compute the sediment discharge for the Grass model @@ -572,8 +586,9 @@ end return abs(equations.H0 - (h + h_b)) end -# Trigonometric version of Cardano's method to compute the eigenvalues of -# the [`ShallowWaterExnerEquations1D`[(@ref)] assuming only real roots. +# Trigonometric version of Cardano's method for the roots of a cubic polynomial +# in order to compute the eigenvalues of the [`ShallowWaterExnerEquations1D`[(@ref)]. +# Note, assumes only real roots. @inline function eigvals_cardano(u, equations::ShallowWaterExnerEquations1D) h = waterheight(u, equations) v = velocity(u, equations) From f73f6e980707b325869773513dfc0f07da633710 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Tue, 13 Aug 2024 12:51:03 +0200 Subject: [PATCH 12/12] Modify docstrings, fix function q_s --- src/equations/shallow_water_exner_1d.jl | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/src/equations/shallow_water_exner_1d.jl b/src/equations/shallow_water_exner_1d.jl index 35884a4..cd043a6 100644 --- a/src/equations/shallow_water_exner_1d.jl +++ b/src/equations/shallow_water_exner_1d.jl @@ -32,7 +32,7 @@ abstract type SedimentModel{RealT} end Create a Shields stress model to compute the sediment discharge `q_s` based on the generalized formulation from equation (1.2) in the given reference. -The choice of the real constants ´m_1`, ´m_2`, ´m_3`, ´k_1`, ´k_2`, and ´k_3` creates +The choice of the real constants `m_1`, `m_2`, `m_3`, `k_1`, `k_2`, and `k_3` creates different models. For example, setting `m_1=0`, `m_2=1.5`, `m_3=0`, `k_1=8`, `k_2=1`, and `k_3=0` yields the sedimentation model of Meyer-Peter and Müller as given in [`MeyerPeterMueller`](@ref) below. The Shields stress represents the ratio of agitating and stabilizing forces in the sediment bed where @@ -62,10 +62,9 @@ Creates a Grass model to compute the sediment discharge `q_s` as ```math q_s = A_g v^{m_g} ``` -with the coefficients `A_g` and `m_g`. with the coefficients `A_g` and `m_g`. The constant `A_g` lies in the interval ``[0,1]`` and is a dimensional calibration constant that is usually measured experimental. It expresses -the kind of interaction between the fluid and the sediment the strength of which +the kind of interaction between the fluid and the sediment, the strength of which increases as `A_g` approaches to 1. The factor `m_g` lies in the interval ``[1, 4]``. Typically, one considers an odd integer value for `m_g` such that the sediment discharge `q_s` can be differentiated and the model remains valid for all values of the velocity `v`. @@ -107,8 +106,8 @@ end sediment_model, porosity, rho_f, rho_s) -Entropy conservative formulation of the Shallow water-Exner equations in one space dimension. -The equations are given by +Formulation of the Shallow water-Exner equations in one space dimension that possesses a mathematical +entropy inequality. The equations are given by ```math \begin{cases} \partial_t h + \partial_x hv = 0, \\ @@ -122,7 +121,8 @@ the active sediment height ``h_s = q_s / v``. Furthermore ``\tau`` denotes the shear stress at the water-sediment interface and is determined by the `friction` model. The gravitational constant is denoted by ``g``, and ``\rho_f`` and ``\rho_s`` are the fluid and sediment -densities, respectively. The density ratio is given by ``r = \rho_f / \rho_s``. +densities, respectively. The density ratio is given by ``r = \rho_f / \rho_s``, where ``r`` lies between +``0 < r < 1`` as the fluid density ``\rho_f`` should be smaller than the sediment density ``\rho_s``. The conservative variable water height ``h`` is measured from the sediment height ``h_b``, therefore one also defines the total water height as ``H = h + h_b``. @@ -500,8 +500,8 @@ end Q = d_s * sqrt(gravity * (rho_s / rho_f - 1.0) * d_s) # Characteristic discharge return (porosity_inv * Q * sign(theta) * k_1 * theta^m_1 * - (max(theta - k_2 * theta_c, 0.0))^m_2 * - (max(sqrt(theta) - k_3 * sqrt(theta_c), 0.0))^m_3) + (max(theta - k_2 * theta_c, 0.0))^m_2 * + (max(sqrt(theta) - k_3 * sqrt(theta_c), 0.0))^m_3) end # Compute the sediment discharge for the Grass model @@ -511,7 +511,7 @@ end S } (; porosity_inv, sediment_model) = equations - return porosity_inv * sediment_model.A_g * velocity(u, equations)^3 + return porosity_inv * sediment_model.A_g * velocity(u, equations)^sediment_model.m_g end # Shear stress formulation using a coefficient to take into account different friction models @@ -570,7 +570,7 @@ end return equations.gravity * u[1] * u[3] end -# Entropy function +# Mathematical entropy function @inline function Trixi.entropy(u, equations::ShallowWaterExnerEquations1D) h, _, h_b = u v = velocity(u, equations)