From c9cb2ea008e98244fc453d6702e11cb5fede831b Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 18 Dec 2023 09:03:57 +0100 Subject: [PATCH 01/19] add equation SWEWetDry1D --- .JuliaFormatter.toml | 8 + Project.toml | 3 + src/TrixiShallowWater.jl | 14 +- src/equations/equations.jl | 13 + src/equations/shallow_water_wet_dry_1d.jl | 685 ++++++++++++++++++++++ 5 files changed, 718 insertions(+), 5 deletions(-) create mode 100644 .JuliaFormatter.toml create mode 100644 src/equations/equations.jl create mode 100644 src/equations/shallow_water_wet_dry_1d.jl diff --git a/.JuliaFormatter.toml b/.JuliaFormatter.toml new file mode 100644 index 0000000..8518d20 --- /dev/null +++ b/.JuliaFormatter.toml @@ -0,0 +1,8 @@ +# Use SciML style: https://github.com/SciML/SciMLStyle +style = "sciml" + +# Python style alignment. See https://github.com/domluna/JuliaFormatter.jl/pull/732. +yas_style_nesting = true + +# Align struct fields for better readability of large struct definitions +align_struct_field = true diff --git a/Project.toml b/Project.toml index a2e8f3d..a03795b 100644 --- a/Project.toml +++ b/Project.toml @@ -4,6 +4,9 @@ authors = ["Andrew R. Winters ", "Michael Schlottke- version = "0.1.0-pre" [deps] +MuladdMacro = "46d2c3a1-f734-5fdb-9937-b9b9aeba4221" +Static = "aedffcd0-7271-4cad-89d0-dc628f76c6d3" +StaticArrays = "90137ffa-7385-5640-81b9-e52037218182" Trixi = "a7f1ee26-1774-49b1-8366-f1abc58fbfcb" [compat] diff --git a/src/TrixiShallowWater.jl b/src/TrixiShallowWater.jl index b72b67e..e9561e7 100644 --- a/src/TrixiShallowWater.jl +++ b/src/TrixiShallowWater.jl @@ -1,10 +1,14 @@ module TrixiShallowWater - +# Decide between using Trixi: Trixi, import Trixi or using Trixi? using Trixi: Trixi +using MuladdMacro: @muladd +using StaticArrays: SVector +using Static: True, False + +include("equations/equations.jl") -# Write your package code here. -foo() = true -bar() = false -baz() = Trixi.examples_dir() +# export types/functions that define the public API of TrixiShallowWater.jl +export ShallowWaterEquationsWetDry1D +export hydrostatic_reconstruction_chen_noelle, flux_nonconservative_chen_noelle end diff --git a/src/equations/equations.jl b/src/equations/equations.jl new file mode 100644 index 0000000..a84bff8 --- /dev/null +++ b/src/equations/equations.jl @@ -0,0 +1,13 @@ +# 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 + +#################################################################################################### +# Include files with actual implementations for different systems of equations. +# Numerical flux formulations that are independent of the specific system of equations + +include("shallow_water_wet_dry_1d.jl") +end # @muladd diff --git a/src/equations/shallow_water_wet_dry_1d.jl b/src/equations/shallow_water_wet_dry_1d.jl new file mode 100644 index 0000000..dc3d9b4 --- /dev/null +++ b/src/equations/shallow_water_wet_dry_1d.jl @@ -0,0 +1,685 @@ +# 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""" + ShallowWaterEquationsWetDry1D(; gravity, H0 = 0, threshold_limiter = nothing threshold_wet = nothing) + +Shallow water equations (SWE) in one space dimension. The equations are given by +```math +\begin{aligned} + \frac{\partial h}{\partial t} + \frac{\partial}{\partial x}(h v) &= 0 \\ + \frac{\partial}{\partial t}(h v) + \frac{\partial}{\partial x}\left(h v^2 + \frac{g}{2}h^2\right) + + g h \frac{\partial b}{\partial x} &= 0 +\end{aligned} +``` +The unknown quantities of the SWE are the water height ``h`` and the velocity ``v``. +The gravitational constant is denoted by `g` and the (possibly) variable bottom topography function ``b(x)``. +Conservative variable water height ``h`` is measured from the bottom topography ``b``, therefore one +also defines the total water height as ``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. + +Also, there are two thresholds which prevent numerical problems as well as instabilities. Both of them do not +have to be passed, as default values are defined within the struct. The first one, `threshold_limiter`, is +used in [`PositivityPreservingLimiterShallowWater`](@ref) on the water height, as a (small) shift on the initial +condition and cutoff before the next time step. The second one, `threshold_wet`, is applied on the water height to +define when the flow is "wet" before calculating the numerical flux. + +The bottom topography function ``b(x)`` is set inside the initial condition routine +for a particular problem setup. To test the conservative form of the SWE one can set the bottom topography +variable `b` to zero. + +In addition to the unknowns, Trixi.jl currently stores the bottom topography values at the approximation points +despite being fixed in time. This is done for convenience of computing the bottom topography gradients +on the fly during the approximation as well as computing auxiliary quantities like the total water height ``H`` +or the entropy variables. +This affects the implementation and use of these equations in various ways: +* The flux values corresponding to the bottom topography must be zero. +* The bottom topography values must be included when defining initial conditions, boundary conditions or + source terms. +* [`AnalysisCallback`](@ref) analyzes this variable. +* Trixi.jl's visualization tools will visualize the bottom topography by default. + +References for the SWE are many but a good introduction is available in Chapter 13 of the book: +- Randall J. LeVeque (2002) + Finite Volume Methods for Hyperbolic Problems + [DOI: 10.1017/CBO9780511791253](https://doi.org/10.1017/CBO9780511791253) +""" +struct ShallowWaterEquationsWetDry1D{RealT <: Real} <: + Trixi.AbstractShallowWaterEquations{1, 3} + gravity::RealT # gravitational constant + H0::RealT # constant "lake-at-rest" total water height + # `threshold_limiter` used in `PositivityPreservingLimiterShallowWater` on water height, + # as a (small) shift on the initial condition and cutoff before the next time step. + # Default is 500*eps() which in double precision is ≈1e-13. + threshold_limiter::RealT + # `threshold_wet` applied on water height to define when the flow is "wet" + # before calculating the numerical flux. + # Default is 5*eps() which in double precision is ≈1e-15. + threshold_wet::RealT +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. +# Strict default values for thresholds that performed well in many numerical experiments +function ShallowWaterEquationsWetDry1D(; gravity_constant, H0 = zero(gravity_constant), + threshold_limiter = nothing, + threshold_wet = nothing) + T = promote_type(typeof(gravity_constant), typeof(H0)) + if threshold_limiter === nothing + threshold_limiter = 500 * eps(T) + end + if threshold_wet === nothing + threshold_wet = 5 * eps(T) + end + ShallowWaterEquationsWetDry1D(gravity_constant, H0, threshold_limiter, + threshold_wet) +end + +Trixi.have_nonconservative_terms(::ShallowWaterEquationsWetDry1D) = True() +function Trixi.varnames(::typeof(Trixi.cons2cons), ::ShallowWaterEquationsWetDry1D) + ("h", "h_v", "b") +end +# Note, we use the total water height, H = h + b, as the first primitive variable for easier +# visualization and setting initial conditions +function Trixi.varnames(::typeof(Trixi.cons2prim), ::ShallowWaterEquationsWetDry1D) + ("H", "v", "b") +end + +# Set initial conditions at physical location `x` for time `t` +""" + initial_condition_convergence_test(x, t, equations::ShallowWaterEquationsWetDry1D) + +A smooth initial condition used for convergence tests in combination with +[`source_terms_convergence_test`](@ref) +(and [`BoundaryConditionDirichlet(initial_condition_convergence_test)`](@ref) in non-periodic domains). +""" + +function Trixi.initial_condition_convergence_test(x, t, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.initial_condition_convergence_test(x, t, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), + eps())) +end + +""" + source_terms_convergence_test(u, x, t, equations::ShallowWaterEquationsWetDry1D) + +Source terms used for convergence tests in combination with +[`initial_condition_convergence_test`](@ref) +(and [`BoundaryConditionDirichlet(initial_condition_convergence_test)`](@ref) in non-periodic domains). + +This manufactured solution source term is specifically designed for the bottom topography function +`b(x) = 2.0 + 0.5 * sin(sqrt(2.0)*pi*x[1])` +as defined in [`initial_condition_convergence_test`](@ref). +""" + +@inline function Trixi.source_terms_convergence_test(u, x, t, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.source_terms_convergence_test(u, x, t, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), + eps())) +end + +""" + initial_condition_weak_blast_wave(x, t, equations::ShallowWaterEquationsWetDry1D) + +A weak blast wave discontinuity useful for testing, e.g., total energy conservation. +Note for the shallow water equations to the total energy acts as a mathematical entropy function. +""" +function Trixi.initial_condition_weak_blast_wave(x, t, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.initial_condition_weak_blast_wave(x, t, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), + eps())) +end + +""" + boundary_condition_slip_wall(u_inner, orientation_or_normal, x, t, surface_flux_function, + equations::ShallowWaterEquationsWetDry1D) + +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::ShallowWaterEquationsWetDry1D) + return Trixi.boundary_condition_slip_wall(u_inner, orientation_or_normal, direction, + x, t, + surface_flux_function, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), + eps())) +end + +# Calculate 1D flux for a single point +# Note, the bottom topography has no flux +@inline function Trixi.flux(u, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.flux(u, orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, equations.H0, + eps(), eps())) +end + +""" + flux_nonconservative_wintermeyer_etal(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + +Non-symmetric two-point volume flux discretizing the nonconservative (source) term +that contains the gradient of the bottom topography [`ShallowWaterEquationsWetDry1D`](@ref). + +Further details are available in the paper:#include("numerical_fluxes.jl") +- Niklas Wintermeyer, Andrew R. Winters, Gregor J. Gassner and David A. Kopriva (2017) + An entropy stable nodal discontinuous Galerkin method for the two dimensional + shallow water equations on unstructured curvilinear meshes with discontinuous bathymetry + [DOI: 10.1016/j.jcp.2017.03.036](https://doi.org/10.1016/j.jcp.2017.03.036) +""" +@inline function Trixi.flux_nonconservative_wintermeyer_etal(u_ll, u_rr, + orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.flux_nonconservative_wintermeyer_etal(u_ll, u_rr, orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), + eps())) +end + +""" + flux_nonconservative_fjordholm_etal(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + +Non-symmetric two-point surface flux discretizing the nonconservative (source) term of +that contains the gradient of the bottom topography [`ShallowWaterEquationsWetDry1D`](@ref). + +This contains additional terms compared to [`flux_nonconservative_wintermeyer_etal`](@ref) +that account for possible discontinuities in the bottom topography function. +Thus, this flux should be used in general at interfaces. For flux differencing volume terms, +[`flux_nonconservative_wintermeyer_etal`](@ref) is analytically equivalent but slightly +cheaper. + +Further details for the original finite volume formulation are available in +- Ulrik S. Fjordholm, Siddhartha Mishr and Eitan Tadmor (2011) + Well-balanced and energy stable schemes for the shallow water equations with discontinuous topography + [DOI: 10.1016/j.jcp.2011.03.042](https://doi.org/10.1016/j.jcp.2011.03.042) +and for curvilinear 2D case in the paper: +- Niklas Wintermeyer, Andrew R. Winters, Gregor J. Gassner and David A. Kopriva (2017) + An entropy stable nodal discontinuous Galerkin method for the two dimensional + shallow water equations on unstructured curvilinear meshes with discontinuous bathymetry + [DOI: 10.1016/j.jcp.2017.03.036](https://doi.org/10.1016/j.jcp.2017.03.036) +""" +@inline function Trixi.flux_nonconservative_fjordholm_etal(u_ll, u_rr, + orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.flux_nonconservative_fjordholm_etal(u_ll, u_rr, orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), + eps())) +end + +""" + flux_nonconservative_audusse_etal(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + +Non-symmetric two-point surface flux that discretizes the nonconservative (source) term. +The discretization uses the `hydrostatic_reconstruction_audusse_etal` on the conservative +variables. + +This hydrostatic reconstruction ensures that the finite volume numerical fluxes remain +well-balanced for discontinuous bottom topographies [`ShallowWaterEquationsWetDry1D`](@ref). +Should be used together with [`FluxHydrostaticReconstruction`](@ref) and +[`hydrostatic_reconstruction_audusse_etal`](@ref) in the surface flux to ensure consistency. + +Further details on the hydrostatic reconstruction and its motivation can be found in +- Emmanuel Audusse, François Bouchut, Marie-Odile Bristeau, Rupert Klein, and Benoit Perthame (2004) + A fast and stable well-balanced scheme with hydrostatic reconstruction for shallow water flows + [DOI: 10.1137/S1064827503431090](https://doi.org/10.1137/S1064827503431090) +""" +@inline function Trixi.flux_nonconservative_audusse_etal(u_ll, u_rr, + orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.flux_nonconservative_audusse_etal(u_ll, u_rr, + orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), + eps())) +end + +""" + flux_nonconservative_chen_noelle(u_ll, u_rr, + orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + +Non-symmetric two-point surface flux that discretizes the nonconservative (source) term. +The discretization uses the `hydrostatic_reconstruction_chen_noelle` on the conservative +variables. + +Should be used together with [`FluxHydrostaticReconstruction`](@ref) and +[`hydrostatic_reconstruction_chen_noelle`](@ref) in the surface flux to ensure consistency. + +Further details on the hydrostatic reconstruction and its motivation can be found in +- Guoxian Chen and Sebastian Noelle (2017) + A new hydrostatic reconstruction scheme based on subcell reconstructions + [DOI:10.1137/15M1053074](https://dx.doi.org/10.1137/15M1053074) +""" +@inline function flux_nonconservative_chen_noelle(u_ll, u_rr, + orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + + # Pull the water height and bottom topography on the left + h_ll, _, b_ll = u_ll + h_rr, _, b_rr = u_rr + + H_ll = h_ll + b_ll + H_rr = h_rr + b_rr + + b_star = min(max(b_ll, b_rr), min(H_ll, H_rr)) + + # Create the hydrostatic reconstruction for the left solution state + u_ll_star, _ = hydrostatic_reconstruction_chen_noelle(u_ll, u_rr, equations) + + # Copy the reconstructed water height for easier to read code + h_ll_star = u_ll_star[1] + + z = zero(eltype(u_ll)) + # Includes two parts: + # (i) Diagonal (consistent) term from the volume flux that uses `b_ll` to avoid + # cross-averaging across a discontinuous bottom topography + # (ii) True surface part that uses `h_ll` and `h_ll_star` to handle discontinuous bathymetry + return SVector(z, + equations.gravity * h_ll * b_ll - + equations.gravity * (h_ll_star + h_ll) * (b_ll - b_star), + z) +end + +""" + flux_nonconservative_ersing_etal(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + +!!! warning "Experimental code" + This numerical flux is experimental and may change in any future release. + +Non-symmetric path-conservative two-point volume flux discretizing the nonconservative (source) term +that contains the gradient of the bottom topography [`ShallowWaterEquationsWetDry1D`](@ref). + +This is a modified version of [`flux_nonconservative_wintermeyer_etal`](@ref) that gives entropy +conservation and well-balancedness in both the volume and surface when combined with +[`flux_wintermeyer_etal`](@ref). + +For further details see: +- Patrick Ersing, Andrew R. Winters (2023) + An entropy stable discontinuous Galerkin method for the two-layer shallow water equations on + curvilinear meshes + [DOI: 10.48550/arXiv.2306.12699](https://doi.org/10.48550/arXiv.2306.12699) +""" +@inline function Trixi.flux_nonconservative_ersing_etal(u_ll, u_rr, + orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.flux_nonconservative_ersing_etal(u_ll, u_rr, orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), + eps())) +end + +""" + flux_fjordholm_etal(u_ll, u_rr, orientation, + equations::ShallowWaterEquationsWetDry1D) + +Total energy conservative (mathematical entropy for shallow water equations). When the bottom topography +is nonzero this should only be used as a surface flux otherwise the scheme will not be well-balanced. +For well-balancedness in the volume flux use [`flux_wintermeyer_etal`](@ref). + +Details are available in Eq. (4.1) in the paper: +- Ulrik S. Fjordholm, Siddhartha Mishr and Eitan Tadmor (2011) + Well-balanced and energy stable schemes for the shallow water equations with discontinuous topography + [DOI: 10.1016/j.jcp.2011.03.042](https://doi.org/10.1016/j.jcp.2011.03.042) +""" +@inline function Trixi.flux_fjordholm_etal(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.flux_fjordholm_etal(u_ll, u_rr, orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), + eps())) +end + +""" + flux_wintermeyer_etal(u_ll, u_rr, orientation, + equations::ShallowWaterEquationsWetDry1D) + +Total energy conservative (mathematical entropy for shallow water equations) split form. +When the bottom topography is nonzero this scheme will be well-balanced when used as a `volume_flux`. +The `surface_flux` should still use, e.g., [`flux_fjordholm_etal`](@ref). + +Further details are available in Theorem 1 of the paper: +- Niklas Wintermeyer, Andrew R. Winters, Gregor J. Gassner and David A. Kopriva (2017) + An entropy stable nodal discontinuous Galerkin method for the two dimensional + shallow water equations on unstructured curvilinear meshes with discontinuous bathymetry + [DOI: 10.1016/j.jcp.2017.03.036](https://doi.org/10.1016/j.jcp.2017.03.036) +""" +@inline function Trixi.flux_wintermeyer_etal(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.flux_wintermeyer_etal(u_ll, u_rr, orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), eps())) +end + +""" + hydrostatic_reconstruction_audusse_etal(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + +A particular type of hydrostatic reconstruction on the water height to guarantee well-balancedness +for a general bottom topography [`ShallowWaterEquationsWetDry1D`](@ref). The reconstructed solution states +`u_ll_star` and `u_rr_star` variables are then used to evaluate the surface numerical flux at the interface. +Use in combination with the generic numerical flux routine [`FluxHydrostaticReconstruction`](@ref). + +Further details on this hydrostatic reconstruction and its motivation can be found in +- Emmanuel Audusse, François Bouchut, Marie-Odile Bristeau, Rupert Klein, and Benoit Perthame (2004) + A fast and stable well-balanced scheme with hydrostatic reconstruction for shallow water flows + [DOI: 10.1137/S1064827503431090](https://doi.org/10.1137/S1064827503431090) +""" +@inline function Trixi.hydrostatic_reconstruction_audusse_etal(u_ll, u_rr, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.hydrostatic_reconstruction_audusse_etal(u_ll, u_rr, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), + eps())) +end + +""" + hydrostatic_reconstruction_chen_noelle(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + +A particular type of hydrostatic reconstruction of the water height to guarantee well-balancedness +for a general bottom topography of the [`ShallowWaterEquationsWetDry1D`](@ref). The reconstructed solution states +`u_ll_star` and `u_rr_star` variables are used to evaluate the surface numerical flux at the interface. +The key idea is a linear reconstruction of the bottom and water height at the interfaces using subcells. +Use in combination with the generic numerical flux routine [`FluxHydrostaticReconstruction`](@ref). + +Further details on this hydrostatic reconstruction and its motivation can be found in +- Guoxian Chen and Sebastian Noelle (2017) + A new hydrostatic reconstruction scheme based on subcell reconstructions + [DOI:10.1137/15M1053074](https://dx.doi.org/10.1137/15M1053074) +""" +@inline function hydrostatic_reconstruction_chen_noelle(u_ll, u_rr, + equations::ShallowWaterEquationsWetDry1D) + # Unpack left and right water heights and bottom topographies + h_ll, _, b_ll = u_ll + h_rr, _, b_rr = u_rr + + # Get the velocities on either side + v_ll = velocity(u_ll, equations) + v_rr = velocity(u_rr, equations) + + H_ll = b_ll + h_ll + H_rr = b_rr + h_rr + + b_star = min(max(b_ll, b_rr), min(H_ll, H_rr)) + + # Compute the reconstructed water heights + h_ll_star = min(H_ll - b_star, h_ll) + h_rr_star = min(H_rr - b_star, h_rr) + + # Set the water height to be at least the value stored in the variable threshold after + # the hydrostatic reconstruction is applied and before the numerical flux is calculated + # to avoid numerical problem with arbitrary small values. Interfaces with a water height + # lower or equal to the threshold can be declared as dry. + # The default value for `threshold_wet` is ≈ 5*eps(), or 1e-15 in double precision, is set + # in the `ShallowWaterEquationsWetDry1D` struct. This threshold value can be changed in the constructor + # call of this equation struct in an elixir. + threshold = equations.threshold_wet + + if (h_ll_star <= threshold) + h_ll_star = threshold + v_ll = zero(v_ll) + end + + if (h_rr_star <= threshold) + h_rr_star = threshold + v_rr = zero(v_rr) + end + + # Create the conservative variables using the reconstruted water heights + u_ll_star = SVector(h_ll_star, h_ll_star * v_ll, b_ll) + u_rr_star = SVector(h_rr_star, h_rr_star * v_rr, b_rr) + + return u_ll_star, u_rr_star +end + +# Calculate maximum wave speed for local Lax-Friedrichs-type dissipation as the +# maximum velocity magnitude plus the maximum speed of sound +@inline function Trixi.max_abs_speed_naive(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.max_abs_speed_naive(u_ll, u_rr, orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), + eps())) +end + +# Specialized `DissipationLocalLaxFriedrichs` to avoid spurious dissipation in the bottom topography +@inline function (dissipation::Trixi.DissipationLocalLaxFriedrichs)(u_ll, u_rr, + orientation_or_normal_direction, + equations::ShallowWaterEquationsWetDry1D) + return (dissipation::Trixi.DissipationLocalLaxFriedrichs)(u_ll, u_rr, + orientation_or_normal_direction, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), + eps())) +end + +# Specialized `FluxHLL` to avoid spurious dissipation in the bottom topography +@inline function (numflux::Trixi.FluxHLL)(u_ll, u_rr, orientation_or_normal_direction, + equations::ShallowWaterEquationsWetDry1D) + return (numflux::Trixi.FluxHLL)(u_ll, u_rr, orientation_or_normal_direction, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), + eps())) +end + +# Calculate estimate for minimum and maximum wave speeds for HLL-type fluxes +@inline function Trixi.min_max_speed_naive(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.min_max_speed_naive(u_ll, u_rr, orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), + eps())) +end + +""" + min_max_speed_chen_noelle(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + +The approximated speeds for the HLL type numerical flux used by Chen and Noelle for their +hydrostatic reconstruction. As they state in the paper, these speeds are chosen for the numerical +flux to ensure positivity and to satisfy an entropy inequality. + +Further details on this hydrostatic reconstruction and its motivation can be found in +- Guoxian Chen and Sebastian Noelle (2017) + A new hydrostatic reconstruction scheme based on subcell reconstructions + [DOI:10.1137/15M1053074](https://dx.doi.org/10.1137/15M1053074) +""" +@inline function min_max_speed_chen_noelle(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + # Get the velocity quantities + v_ll = velocity(u_ll, equations) + v_rr = velocity(u_rr, equations) + + # Calculate the wave celerity on the left and right + h_ll = waterheight(u_ll, equations) + h_rr = waterheight(u_rr, equations) + + a_ll = sqrt(equations.gravity * h_ll) + a_rr = sqrt(equations.gravity * h_rr) + + λ_min = min(v_ll - a_ll, v_rr - a_rr, zero(eltype(u_ll))) + λ_max = max(v_ll + a_ll, v_rr + a_rr, zero(eltype(u_ll))) + + return λ_min, λ_max +end + +# More refined estimates for minimum and maximum wave speeds for HLL-type fluxes +@inline function Trixi.min_max_speed_davis(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.min_max_speed_davis(u_ll, u_rr, orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), + eps())) +end + +@inline function Trixi.min_max_speed_einfeldt(u_ll, u_rr, orientation::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.min_max_speed_einfeldt(u_ll, u_rr, orientation, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, + eps(), eps())) +end + +@inline function Trixi.max_abs_speeds(u, equations::ShallowWaterEquationsWetDry1D) + return Trixi.max_abs_speeds(u, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), + eps())) +end + +# Helper function to extract the velocity vector from the conservative variables +@inline function Trixi.velocity(u, equations::ShallowWaterEquationsWetDry1D) + return Trixi.velocity(u, + Trixi.ShallowWaterEquations1D(equations.gravity, equations.H0, + eps(), eps())) +end + +# Convert conservative variables to primitive +@inline function Trixi.cons2prim(u, equations::ShallowWaterEquationsWetDry1D) + return Trixi.cons2prim(u, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), eps())) +end + +# Convert conservative variables to entropy +# Note, only the first two are the entropy variables, the third entry still +# just carries the bottom topography values for convenience +@inline function Trixi.cons2entropy(u, equations::ShallowWaterEquationsWetDry1D) + return Trixi.cons2entropy(u, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), eps())) +end + +# Convert entropy variables to conservative +@inline function Trixi.entropy2cons(w, equations::ShallowWaterEquationsWetDry1D) + return Trixi.entropy2cons(w, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), eps())) +end + +# Convert primitive to conservative variables +@inline function Trixi.prim2cons(prim, equations::ShallowWaterEquationsWetDry1D) + return Trixi.prim2cons(prim, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), eps())) +end + +@inline function Trixi.waterheight(u, equations::ShallowWaterEquationsWetDry1D) + return Trixi.waterheight(u, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), eps())) +end + +@inline function Trixi.pressure(u, equations::ShallowWaterEquationsWetDry1D) + return Trixi.pressure(u, + Trixi.ShallowWaterEquations1D(equations.gravity, equations.H0, + eps(), eps())) +end + +@inline function Trixi.waterheight_pressure(u, equations::ShallowWaterEquationsWetDry1D) + return Trixi.waterheight_pressure(u, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), + eps())) +end + +""" + calc_wavespeed_roe(u_ll, u_rr, direction::Integer, + equations::ShallowWaterEquationsWetDry1D) + +Calculate Roe-averaged velocity `v_roe` and wavespeed `c_roe = sqrt{g * h_roe}` +See for instance equation (62) in +- Paul A. Ullrich, Christiane Jablonowski, and Bram van Leer (2010) + High-order finite-volume methods for the shallow-water equations on the sphere + [DOI: 10.1016/j.jcp.2010.04.044](https://doi.org/10.1016/j.jcp.2010.04.044) +Or equation (9.17) in [this lecture notes](https://metaphor.ethz.ch/x/2019/hs/401-4671-00L/literature/mishra_hyperbolic_pdes.pdf). +""" +@inline function Trixi.calc_wavespeed_roe(u_ll, u_rr, direction::Integer, + equations::ShallowWaterEquationsWetDry1D) + return Trixi.calc_wavespeed_roe(u_ll, u_rr, direction::Integer, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), + eps())) +end + +# Entropy function for the shallow water equations is the total energy +@inline function Trixi.entropy(cons, equations::ShallowWaterEquationsWetDry1D) + Trixi.energy_total(cons, equations) +end + +# Calculate total energy for a conservative state `cons` +@inline function Trixi.energy_total(cons, equations::ShallowWaterEquationsWetDry1D) + return Trixi.energy_total(cons, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), eps())) +end + +# Calculate kinetic energy for a conservative state `cons` +@inline function Trixi.energy_kinetic(u, equations::ShallowWaterEquationsWetDry1D) + return Trixi.energy_kinetic(u, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), + eps())) +end + +# Calculate potential energy for a conservative state `cons` +@inline function Trixi.energy_internal(cons, equations::ShallowWaterEquationsWetDry1D) + return Trixi.energy_total(cons, equations) - Trixi.energy_kinetic(cons, equations) +end + +# Calculate the error for the "lake-at-rest" test case where H = h+b should +# be a constant value over time. Note, assumes there is a single reference +# water height `H0` with which to compare. +@inline function Trixi.lake_at_rest_error(u, equations::ShallowWaterEquationsWetDry1D) + h, _, b = u + + # For well-balancedness testing with possible wet/dry regions the reference + # water height `H0` accounts for the possibility that the bottom topography + # can emerge out of the water as well as for the threshold offset to avoid + # division by a "hard" zero water heights as well. + H0_wet_dry = max(equations.H0, b + equations.threshold_limiter) + + return abs(H0_wet_dry - (h + b)) +end +end # @muladd From 6f23516cee63dfe30a180d4ec783ce7f95dfa15a Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 18 Dec 2023 09:05:24 +0100 Subject: [PATCH 02/19] add tests and elixirs --- .../tree_1d_dgsem/elixir_shallowwater_ec.jl | 93 ++++ .../elixir_shallowwater_source_terms.jl | 60 +++ ...xir_shallowwater_source_terms_dirichlet.jl | 63 +++ .../elixir_shallowwater_well_balanced.jl | 88 ++++ ..._shallowwater_well_balanced_nonperiodic.jl | 77 ++++ test/runtests.jl | 4 +- test/test_tree_1d.jl | 27 ++ test/test_tree_1d_shallowwater_wet_dry.jl | 396 ++++++++++++++++++ test/test_trixi.jl | 238 +++++++++++ 9 files changed, 1043 insertions(+), 3 deletions(-) create mode 100644 examples/tree_1d_dgsem/elixir_shallowwater_ec.jl create mode 100644 examples/tree_1d_dgsem/elixir_shallowwater_source_terms.jl create mode 100644 examples/tree_1d_dgsem/elixir_shallowwater_source_terms_dirichlet.jl create mode 100644 examples/tree_1d_dgsem/elixir_shallowwater_well_balanced.jl create mode 100644 examples/tree_1d_dgsem/elixir_shallowwater_well_balanced_nonperiodic.jl create mode 100644 test/test_tree_1d.jl create mode 100644 test/test_tree_1d_shallowwater_wet_dry.jl create mode 100644 test/test_trixi.jl diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_ec.jl b/examples/tree_1d_dgsem/elixir_shallowwater_ec.jl new file mode 100644 index 0000000..c976495 --- /dev/null +++ b/examples/tree_1d_dgsem/elixir_shallowwater_ec.jl @@ -0,0 +1,93 @@ + +using OrdinaryDiffEq +using Trixi +using TrixiShallowWater + +############################################################################### +# Semidiscretization of the shallow water equations with a discontinuous +# bottom topography function + +equations = ShallowWaterEquationsWetDry1D(gravity_constant = 9.81) + +# Initial condition with a truly discontinuous water height, velocity, and bottom +# topography function as an academic testcase for entropy conservation. +# The errors from the analysis callback are not important but `∑∂S/∂U ⋅ Uₜ` should +# be around machine roundoff. +# Works as intended for TreeMesh1D with `initial_refinement_level=4`. If the mesh +# refinement level is changed the initial condition below may need changed as well to +# ensure that the discontinuities lie on an element interface. +function initial_condition_ec_discontinuous_bottom(x, t, + equations::ShallowWaterEquationsWetDry1D) + # Set the background values + H = 4.25 + v = 0.0 + b = sin(x[1]) # arbitrary continuous function + + # Setup the discontinuous water height and velocity + if x[1] >= 0.125 && x[1] <= 0.25 + H = 5.0 + v = 0.1882 + end + + # Setup a discontinuous bottom topography + if x[1] >= -0.25 && x[1] <= -0.125 + b = 2.0 + 0.5 * sin(2.0 * pi * x[1]) + end + + return prim2cons(SVector(H, v, b), equations) +end + +initial_condition = initial_condition_ec_discontinuous_bottom + +############################################################################### +# Get the DG approximation space + +volume_flux = (flux_wintermeyer_etal, flux_nonconservative_wintermeyer_etal) +solver = DGSEM(polydeg = 4, + surface_flux = (flux_fjordholm_etal, flux_nonconservative_fjordholm_etal), + volume_integral = VolumeIntegralFluxDifferencing(volume_flux)) + +############################################################################### +# Get the TreeMesh and setup a periodic mesh + +coordinates_min = -1.0 +coordinates_max = 1.0 +mesh = TreeMesh(coordinates_min, coordinates_max, + initial_refinement_level = 4, + n_cells_max = 10_000) + +# Create the semi discretization object +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver) + +############################################################################### +# ODE solver + +tspan = (0.0, 2.0) +ode = semidiscretize(semi, tspan) + +############################################################################### +# Callbacks + +summary_callback = SummaryCallback() + +analysis_interval = 100 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +save_solution = SaveSolutionCallback(interval = 100, + save_initial_solution = true, + save_final_solution = true) + +stepsize_callback = StepsizeCallback(cfl = 3.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/examples/tree_1d_dgsem/elixir_shallowwater_source_terms.jl b/examples/tree_1d_dgsem/elixir_shallowwater_source_terms.jl new file mode 100644 index 0000000..1869a78 --- /dev/null +++ b/examples/tree_1d_dgsem/elixir_shallowwater_source_terms.jl @@ -0,0 +1,60 @@ + +using OrdinaryDiffEq +using Trixi +using TrixiShallowWater + +############################################################################### +# Semidiscretization of the shallow water equations + +equations = ShallowWaterEquationsWetDry1D(gravity_constant = 9.81) + +initial_condition = initial_condition_convergence_test + +############################################################################### +# Get the DG approximation space + +volume_flux = (flux_wintermeyer_etal, flux_nonconservative_wintermeyer_etal) +solver = DGSEM(polydeg = 3, + surface_flux = (flux_lax_friedrichs, flux_nonconservative_fjordholm_etal), + 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 = 3, + 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 = 500 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +save_solution = SaveSolutionCallback(interval = 200, + save_initial_solution = true, + save_final_solution = true) + +callbacks = CallbackSet(summary_callback, analysis_callback, alive_callback, save_solution) + +############################################################################### +# run the simulation + +# use a Runge-Kutta method with automatic (error based) time step size control +sol = solve(ode, RDPK3SpFSAL49(); abstol = 1.0e-8, reltol = 1.0e-8, + ode_default_options()..., callback = callbacks); +summary_callback() # print the timer summary diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_source_terms_dirichlet.jl b/examples/tree_1d_dgsem/elixir_shallowwater_source_terms_dirichlet.jl new file mode 100644 index 0000000..e623077 --- /dev/null +++ b/examples/tree_1d_dgsem/elixir_shallowwater_source_terms_dirichlet.jl @@ -0,0 +1,63 @@ + +using OrdinaryDiffEq +using Trixi +using TrixiShallowWater + +############################################################################### +# Semidiscretization of the shallow water equations + +equations = ShallowWaterEquationsWetDry1D(gravity_constant = 9.81) + +initial_condition = initial_condition_convergence_test + +boundary_condition = BoundaryConditionDirichlet(initial_condition) + +############################################################################### +# Get the DG approximation space + +volume_flux = (flux_wintermeyer_etal, flux_nonconservative_wintermeyer_etal) +surface_flux = (flux_lax_friedrichs, flux_nonconservative_fjordholm_etal) +solver = DGSEM(polydeg = 3, 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 = 3, + n_cells_max = 10_000, + periodicity = false) + +# create the semi discretization object +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver, + boundary_conditions = boundary_condition, + source_terms = source_terms_convergence_test) + +############################################################################### +# ODE solvers, callbacks etc. + +tspan = (0.0, 1.0) +ode = semidiscretize(semi, tspan) + +summary_callback = SummaryCallback() + +analysis_interval = 500 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +save_solution = SaveSolutionCallback(interval = 200, + save_initial_solution = true, + save_final_solution = true) + +callbacks = CallbackSet(summary_callback, analysis_callback, alive_callback, save_solution) + +############################################################################### +# run the simulation + +# use a Runge-Kutta method with automatic (error based) time step size control +sol = solve(ode, RDPK3SpFSAL49(); abstol = 1.0e-8, reltol = 1.0e-8, + ode_default_options()..., callback = callbacks); +summary_callback() # print the timer summary diff --git a/examples/tree_1d_dgsem/elixir_shallowwater_well_balanced.jl b/examples/tree_1d_dgsem/elixir_shallowwater_well_balanced.jl new file mode 100644 index 0000000..39b8166 --- /dev/null +++ b/examples/tree_1d_dgsem/elixir_shallowwater_well_balanced.jl @@ -0,0 +1,88 @@ + +using OrdinaryDiffEq +using Trixi +using TrixiShallowWater + +############################################################################### +# semidiscretization of the shallow water equations with a discontinuous +# bottom topography function + +equations = ShallowWaterEquationsWetDry1D(gravity_constant = 9.81, H0 = 3.25) + +# Setup a truly discontinuous bottom topography function for this academic +# testcase of well-balancedness. The errors from the analysis callback are +# not important but the error for this lake-at-rest test case +# `∑|H0-(h+b)|` should be around machine roundoff. +# Works as intended for TreeMesh1D with `initial_refinement_level=3`. If the mesh +# refinement level is changed the initial condition below may need changed as well to +# ensure that the discontinuities lie on an element interface. +function initial_condition_discontinuous_well_balancedness(x, t, + equations::ShallowWaterEquationsWetDry1D) + # Set the background values + H = equations.H0 + v = 0.0 + b = 0.0 + + # Setup a discontinuous bottom topography + if x[1] >= 0.5 && x[1] <= 0.75 + b = 2.0 + 0.5 * sin(2.0 * pi * x[1]) + end + + return prim2cons(SVector(H, v, b), equations) +end + +initial_condition = initial_condition_discontinuous_well_balancedness + +############################################################################### +# Get the DG approximation space + +volume_flux = (flux_wintermeyer_etal, flux_nonconservative_wintermeyer_etal) +surface_flux = (flux_fjordholm_etal, flux_nonconservative_fjordholm_etal) +solver = DGSEM(polydeg = 4, surface_flux = surface_flux, + volume_integral = VolumeIntegralFluxDifferencing(volume_flux)) + +############################################################################### +# Get the TreeMesh and setup a periodic mesh + +coordinates_min = -1.0 +coordinates_max = 1.0 +mesh = TreeMesh(coordinates_min, coordinates_max, + initial_refinement_level = 3, + n_cells_max = 10_000) + +# Create the semi discretization object +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver) + +############################################################################### +# ODE solver + +tspan = (0.0, 100.0) +ode = semidiscretize(semi, tspan) + +############################################################################### +# Callbacks + +summary_callback = SummaryCallback() + +analysis_interval = 1000 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval, + extra_analysis_integrals = (lake_at_rest_error,)) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +save_solution = SaveSolutionCallback(interval = 1000, + save_initial_solution = true, + save_final_solution = true) + +stepsize_callback = StepsizeCallback(cfl = 3.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/examples/tree_1d_dgsem/elixir_shallowwater_well_balanced_nonperiodic.jl b/examples/tree_1d_dgsem/elixir_shallowwater_well_balanced_nonperiodic.jl new file mode 100644 index 0000000..e3bc3fc --- /dev/null +++ b/examples/tree_1d_dgsem/elixir_shallowwater_well_balanced_nonperiodic.jl @@ -0,0 +1,77 @@ + +using OrdinaryDiffEq +using Trixi +using TrixiShallowWater + +############################################################################### +# Semidiscretization of the shallow water equations + +equations = ShallowWaterEquationsWetDry1D(gravity_constant = 1.0, H0 = 3.0) + +# An initial condition with constant total water height and zero velocities to test well-balancedness. +function initial_condition_well_balancedness(x, t, equations::ShallowWaterEquationsWetDry1D) + # Set the background values + H = equations.H0 + v = 0.0 + + b = (1.5 / exp(0.5 * ((x[1] - 1.0)^2)) + 0.75 / exp(0.5 * ((x[1] + 1.0)^2))) + + return prim2cons(SVector(H, v, b), equations) +end + +initial_condition = initial_condition_well_balancedness + +boundary_condition = BoundaryConditionDirichlet(initial_condition) + +############################################################################### +# Get the DG approximation space + +volume_flux = (flux_wintermeyer_etal, flux_nonconservative_wintermeyer_etal) +solver = DGSEM(polydeg = 4, surface_flux = (flux_hll, flux_nonconservative_fjordholm_etal), + 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 = 3, + n_cells_max = 10_000, + periodicity = false) + +# create the semi discretization object +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver, + boundary_conditions = boundary_condition) + +############################################################################### +# ODE solvers, callbacks etc. + +tspan = (0.0, 100.0) +ode = semidiscretize(semi, tspan) + +summary_callback = SummaryCallback() + +analysis_interval = 1000 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval, + save_analysis = true, + extra_analysis_integrals = (lake_at_rest_error,)) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +save_solution = SaveSolutionCallback(interval = 1000, + 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/test/runtests.jl b/test/runtests.jl index 909d0e2..fb8e5ef 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -11,9 +11,7 @@ const TRIXI_NTHREADS = clamp(Sys.CPU_THREADS, 2, 3) @time @testset "TrixiShallowWater.jl tests" begin @time if TRIXI_TEST == "all" - @test TrixiShallowWater.foo() == true - @test TrixiShallowWater.bar() == false - @test TrixiShallowWater.baz() isa String + include("test_tree_1d_shallowwater_wet_dry.jl") end @time if TRIXI_TEST == "all" || TRIXI_TEST == "upstream" diff --git a/test/test_tree_1d.jl b/test/test_tree_1d.jl new file mode 100644 index 0000000..7d96552 --- /dev/null +++ b/test/test_tree_1d.jl @@ -0,0 +1,27 @@ +module TestExamplesTree1D + +using Test +using TrixiShallowWater + +include("test_trixi.jl") + +EXAMPLES_DIR = pkgdir(TrixiShallowWater, "examples", "tree_1d_dgsem") + +# Start with a clean environment: remove Trixi.jl output directory if it exists +outdir = "out" +isdir(outdir) && rm(outdir, recursive = true) + +@testset "TreeMesh1D" begin +#! format: noindent + +# Run basic tests +@testset "Examples 1D" begin + # Shallow water + include("test_tree_1d_shallowwater_wet_dry.jl") +end + +# Clean up afterwards: delete Trixi.jl output directory +@test_nowarn rm(outdir, recursive = true) +end # TreeMesh1D + +end # module diff --git a/test/test_tree_1d_shallowwater_wet_dry.jl b/test/test_tree_1d_shallowwater_wet_dry.jl new file mode 100644 index 0000000..473bf05 --- /dev/null +++ b/test/test_tree_1d_shallowwater_wet_dry.jl @@ -0,0 +1,396 @@ +module TestExamples1DShallowWaterWetDry + +# TODO: TrixiShallowWater: move any wet/dry tests to new package + +using Test +using Trixi +using TrixiShallowWater + +# TODO: Right now this is a local copy of the file from Trixi.jl. How can I use @trixi_testset & +# @test_trixi_include from Trixi.jl? +include("test_trixi.jl") + +EXAMPLES_DIR = pkgdir(TrixiShallowWater, "examples", "tree_1d_dgsem") + +@testset "Shallow Water Wet Dry" begin +#! format: noindent + +@trixi_testset "elixir_shallowwater_ec.jl" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_ec.jl"), + l2=[0.244729018751225, 0.8583565222389505, 0.07330427577586297], + linf=[ + 2.1635021283528504, + 3.8717508164234453, + 1.7711213427919539, + ], + tspan=(0.0, 0.25)) + # 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_ec.jl with initial_condition_weak_blast_wave" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_ec.jl"), + l2=[ + 0.39464782107209717, + 2.03880864210846, + 4.1623084150546725e-10, + ], + linf=[ + 0.778905801278281, + 3.2409883402608273, + 7.419800190922032e-10, + ], + initial_condition=initial_condition_weak_blast_wave, + tspan=(0.0, 0.25)) + # 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_well_balanced.jl" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_well_balanced.jl"), + l2=[ + 0.10416666834254829, + 1.4352935256803184e-14, + 0.10416666834254838, + ], + linf=[1.9999999999999996, 3.248036646353028e-14, 2.0], + tspan=(0.0, 0.25)) + # 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_well_balanced.jl with FluxHydrostaticReconstruction" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_well_balanced.jl"), + l2=[ + 0.10416666834254835, + 1.1891029971551825e-14, + 0.10416666834254838, + ], + linf=[2.0000000000000018, 2.4019608337954543e-14, 2.0], + surface_flux=(FluxHydrostaticReconstruction(flux_lax_friedrichs, + hydrostatic_reconstruction_audusse_etal), + flux_nonconservative_audusse_etal), + tspan=(0.0, 0.25)) + # 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_well_balanced.jl with flux_nonconservative_ersing_etal" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_well_balanced.jl"), + l2=[ + 0.10416666834254838, + 1.6657566141935285e-14, + 0.10416666834254838, + ], + linf=[2.0000000000000004, 3.0610625110157164e-14, 2.0], + surface_flux=(flux_wintermeyer_etal, + flux_nonconservative_ersing_etal), + volume_flux=(flux_wintermeyer_etal, + flux_nonconservative_ersing_etal), + tspan=(0.0, 0.25)) + # 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 + +# TODO: Activate test when Wet_Dry functionality is moved +# @trixi_testset "elixir_shallowwater_well_balanced_wet_dry.jl with FluxHydrostaticReconstruction" begin +# @test_trixi_include(joinpath(EXAMPLES_DIR, +# "elixir_shallowwater_well_balanced_wet_dry.jl"), +# l2=[ +# 0.00965787167169024, +# 5.345454081916856e-14, +# 0.03857583749209928, +# ], +# linf=[ +# 0.4999999999998892, +# 2.2447689894899726e-13, +# 1.9999999999999714, +# ], +# tspan=(0.0, 0.25), +# # Soften the tolerance as test results vary between different CPUs +# atol=1000 * eps()) +# # 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_source_terms.jl" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_source_terms.jl"), + l2=[ + 0.0022363707373868713, + 0.01576799981934617, + 4.436491725585346e-5, + ], + linf=[ + 0.00893601803417754, + 0.05939797350246456, + 9.098379777405796e-5, + ], + tspan=(0.0, 0.025)) + # 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_source_terms.jl with flux_hll" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_source_terms.jl"), + l2=[ + 0.0022758146627220154, + 0.015864082886204556, + 4.436491725585346e-5, + ], + linf=[ + 0.008457195427364006, + 0.057201667446161064, + 9.098379777405796e-5, + ], + tspan=(0.0, 0.025), + surface_flux=(flux_hll, flux_nonconservative_fjordholm_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_source_terms.jl with flux_nonconservative_ersing_etal" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_source_terms.jl"), + l2=[ + 0.005774284062933275, + 0.017408601639513584, + 4.43649172561843e-5, + ], + linf=[ + 0.01639116193303547, + 0.05102877460799604, + 9.098379777450205e-5, + ], + surface_flux=(flux_wintermeyer_etal, + flux_nonconservative_ersing_etal), + volume_flux=(flux_wintermeyer_etal, + flux_nonconservative_ersing_etal), + tspan=(0.0, 0.025)) + # 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_source_terms_dirichlet.jl" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_source_terms_dirichlet.jl"), + l2=[ + 0.0022851099219788917, + 0.01560453773635554, + 4.43649172558535e-5, + ], + linf=[ + 0.008934615705174398, + 0.059403169140869405, + 9.098379777405796e-5, + ], + tspan=(0.0, 0.025)) + # 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_source_terms_dirichlet.jl with FluxHydrostaticReconstruction" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_source_terms_dirichlet.jl"), + l2=[ + 0.0022956052733432287, + 0.015540053559855601, + 4.43649172558535e-5, + ], + linf=[ + 0.008460440313118323, + 0.05720939349382359, + 9.098379777405796e-5, + ], + surface_flux=(FluxHydrostaticReconstruction(flux_hll, + hydrostatic_reconstruction_audusse_etal), + flux_nonconservative_audusse_etal), + tspan=(0.0, 0.025)) + # 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_well_balanced_nonperiodic.jl with Dirichlet boundary" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_well_balanced_nonperiodic.jl"), + l2=[ + 1.725964362045055e-8, + 5.0427180314307505e-16, + 1.7259643530442137e-8, + ], + linf=[ + 3.844551077492042e-8, + 3.469453422316143e-15, + 3.844551077492042e-8, + ], + tspan=(0.0, 0.25)) + # 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_well_balanced_nonperiodic.jl with wall boundary" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_shallowwater_well_balanced_nonperiodic.jl"), + l2=[ + 1.7259643614361866e-8, + 3.5519018243195145e-16, + 1.7259643530442137e-8, + ], + linf=[ + 3.844551010878661e-8, + 9.846474508971374e-16, + 3.844551077492042e-8, + ], + tspan=(0.0, 0.25), + boundary_condition=boundary_condition_slip_wall) + # 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 + +# TODO: Activate test when Wet_Dry functionality is moved +# @trixi_testset "elixir_shallowwater_shock_capturing.jl" begin +# @test_trixi_include(joinpath(EXAMPLES_DIR, +# "elixir_shallowwater_shock_capturing.jl"), +# l2=[0.07424140641160326, 0.2148642632748155, 0.0372579849000542], +# linf=[ +# 1.1209754279344226, +# 1.3230788645853582, +# 0.8646939843534251, +# ], +# tspan=(0.0, 0.05)) +# # 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 + +# TODO: Activate test when Wet_Dry functionality is moved +# @trixi_testset "elixir_shallowwater_beach.jl" begin +# @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_beach.jl"), +# l2=[ +# 0.17979210479598923, +# 1.2377495706611434, +# 6.289818963361573e-8, +# ], +# linf=[ +# 0.845938394800688, +# 3.3740800777086575, +# 4.4541473087633676e-7, +# ], +# tspan=(0.0, 0.05), +# atol=3e-10) # see https://github.com/trixi-framework/Trixi.jl/issues/1617 +# # 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 + +# TODO: Activate test when Wet_Dry functionality is moved +# @trixi_testset "elixir_shallowwater_parabolic_bowl.jl" begin +# @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_shallowwater_parabolic_bowl.jl"), +# l2=[ +# 8.965981683033589e-5, +# 1.8565707397810857e-5, +# 4.1043039226164336e-17, +# ], +# linf=[ +# 0.00041080213807871235, +# 0.00014823261488938177, +# 2.220446049250313e-16, +# ], +# tspan=(0.0, 0.05)) +# # 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 +end # module diff --git a/test/test_trixi.jl b/test/test_trixi.jl new file mode 100644 index 0000000..cebe216 --- /dev/null +++ b/test/test_trixi.jl @@ -0,0 +1,238 @@ +using Test: @test +import Trixi + +# Use a macro to avoid world age issues when defining new initial conditions etc. +# inside an elixir. +""" + @test_trixi_include(elixir; l2=nothing, linf=nothing, + atol=500*eps(), rtol=sqrt(eps()), + parameters...) + +Test Trixi by calling `trixi_include(elixir; parameters...)`. +By default, only the absence of error output is checked. +If `l2` or `linf` are specified, in addition the resulting L2/Linf errors +are compared approximately against these reference values, using `atol, rtol` +as absolute/relative tolerance. +""" +macro test_trixi_include(elixir, args...) + local l2 = get_kwarg(args, :l2, nothing) + local linf = get_kwarg(args, :linf, nothing) + local atol = get_kwarg(args, :atol, 500 * eps()) + local rtol = get_kwarg(args, :rtol, sqrt(eps())) + local skip_coverage = get_kwarg(args, :skip_coverage, false) + local coverage_override = expr_to_named_tuple(get_kwarg(args, :coverage_override, :())) + if !(:maxiters in keys(coverage_override)) + # maxiters in coverage_override defaults to 1 + coverage_override = (; coverage_override..., maxiters = 1) + end + + local cmd = string(Base.julia_cmd()) + local coverage = occursin("--code-coverage", cmd) && + !occursin("--code-coverage=none", cmd) + + local kwargs = Pair{Symbol, Any}[] + for arg in args + if (arg.head == :(=) && + !(arg.args[1] in (:l2, :linf, :atol, :rtol, :coverage_override, :skip_coverage)) + && !(coverage && arg.args[1] in keys(coverage_override))) + push!(kwargs, Pair(arg.args...)) + end + end + + if coverage + for key in keys(coverage_override) + push!(kwargs, Pair(key, coverage_override[key])) + end + end + + if coverage && skip_coverage + return quote + if Trixi.mpi_isroot() + println("═"^100) + println("Skipping coverage test of ", $elixir) + println("═"^100) + println("\n\n") + end + end + end + + quote + Trixi.mpi_isroot() && println("═"^100) + Trixi.mpi_isroot() && println($elixir) + + # if `maxiters` is set in tests, it is usually set to a small number to + # run only a few steps - ignore possible warnings coming from that + if any(==(:maxiters) ∘ first, $kwargs) + additional_ignore_content = [ + r"┌ Warning: Interrupted\. Larger maxiters is needed\..*\n└ @ SciMLBase .+\n", + r"┌ Warning: Interrupted\. Larger maxiters is needed\..*\n└ @ Trixi .+\n"] + else + additional_ignore_content = [] + end + + # evaluate examples in the scope of the module they're called from + @test_nowarn_mod trixi_include(@__MODULE__, $elixir; $kwargs...) additional_ignore_content + + # if present, compare l2 and linf errors against reference values + if !$coverage && (!isnothing($l2) || !isnothing($linf)) + l2_measured, linf_measured = analysis_callback(sol) + + if Trixi.mpi_isroot() && !isnothing($l2) + @test length($l2) == length(l2_measured) + for (l2_expected, l2_actual) in zip($l2, l2_measured) + @test isapprox(l2_expected, l2_actual, atol = $atol, rtol = $rtol) + end + end + + if Trixi.mpi_isroot() && !isnothing($linf) + @test length($linf) == length(linf_measured) + for (linf_expected, linf_actual) in zip($linf, linf_measured) + @test isapprox(linf_expected, linf_actual, atol = $atol, rtol = $rtol) + end + end + end + + Trixi.mpi_isroot() && println("═"^100) + Trixi.mpi_isroot() && println("\n\n") + end +end + +# Get the first value assigned to `keyword` in `args` and return `default_value` +# if there are no assignments to `keyword` in `args`. +function get_kwarg(args, keyword, default_value) + val = default_value + for arg in args + if arg.head == :(=) && arg.args[1] == keyword + val = arg.args[2] + break + end + end + return val +end + +function expr_to_named_tuple(expr) + result = (;) + + for arg in expr.args + if arg.head != :(=) + error("Invalid expression") + end + result = (; result..., arg.args[1] => arg.args[2]) + end + return result +end + +# Modified version of `@test_nowarn` that prints the content of `stderr` when +# it is not empty and ignores module replacements. +macro test_nowarn_mod(expr, additional_ignore_content = String[]) + quote + let fname = tempname() + try + ret = open(fname, "w") do f + redirect_stderr(f) do + $(esc(expr)) + end + end + stderr_content = read(fname, String) + if !isempty(stderr_content) + println("Content of `stderr`:\n", stderr_content) + end + + # Patterns matching the following ones will be ignored. Additional patterns + # passed as arguments can also be regular expressions, so we just use the + # type `Any` for `ignore_content`. + ignore_content = Any[ + # We need to ignore steady state information reported by our callbacks + r"┌ Info: Steady state tolerance reached\n│ steady_state_callback .+\n└ t = .+\n", + # We also ignore our own compilation messages + "[ Info: You just called `trixi_include`. Julia may now compile the code, please be patient.\n", + # TODO: Upstream (PlotUtils). This should be removed again once the + # deprecated stuff is fixed upstream. + "WARNING: importing deprecated binding Colors.RGB1 into Plots.\n", + "WARNING: importing deprecated binding Colors.RGB4 into Plots.\n", + r"┌ Warning: Keyword argument letter not supported with Plots.+\n└ @ Plots.+\n", + r"┌ Warning: `parse\(::Type, ::Coloarant\)` is deprecated.+\n│.+\n│.+\n└ @ Plots.+\n", + # TODO: Silence warning introduced by Flux v0.13.13. Should be properly fixed. + r"┌ Warning: Layer with Float32 parameters got Float64 input.+\n│.+\n│.+\n│.+\n└ @ Flux.+\n"] + append!(ignore_content, $additional_ignore_content) + for pattern in ignore_content + stderr_content = replace(stderr_content, pattern => "") + end + + # We also ignore simple module redefinitions for convenience. Thus, we + # check whether every line of `stderr_content` is of the form of a + # module replacement warning. + @test occursin(r"^(WARNING: replacing module .+\.\n)*$", stderr_content) + ret + finally + rm(fname, force = true) + end + end + end +end + +""" + @timed_testset "name of the testset" #= code to test #= + +Similar to `@testset`, but prints the name of the testset and its runtime +after execution. +""" +macro timed_testset(name, expr) + @assert name isa String + quote + local time_start = time_ns() + @testset $name $expr + local time_stop = time_ns() + if Trixi.mpi_isroot() + flush(stdout) + @info("Testset "*$name*" finished in " + *string(1.0e-9 * (time_stop - time_start))*" seconds.\n") + flush(stdout) + end + end +end + +""" + @trixi_testset "name of the testset" #= code to test #= + +Similar to `@testset`, but wraps the code inside a temporary module to avoid +namespace pollution. It also `include`s this file again to provide the +definition of `@test_trixi_include`. Moreover, it records the execution time +of the testset similarly to [`timed_testset`](@ref). +""" +macro trixi_testset(name, expr) + @assert name isa String + # TODO: `@eval` is evil + # We would like to use + # mod = gensym(name) + # ... + # module $mod + # to create new module names for every test set. However, this is not + # compatible with the dirty hack using `@eval` to get the mapping when + # loading structured, curvilinear meshes. Thus, we need to use a plain + # module name here. + quote + local time_start = time_ns() + @eval module TrixiTestModule + using Test + using Trixi + include(@__FILE__) + # We define `EXAMPLES_DIR` in (nearly) all test modules and use it to + # get the path to the elixirs to be tested. However, that's not required + # and we want to fail gracefully if it's not defined. + try + import ..EXAMPLES_DIR + catch + nothing + end + @testset $name $expr + end + local time_stop = time_ns() + if Trixi.mpi_isroot() + flush(stdout) + @info("Testset "*$name*" finished in " + *string(1.0e-9 * (time_stop - time_start))*" seconds.\n") + end + nothing + end +end From 49e1ef4af218d11bf46599892495c44332781263 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 18 Dec 2023 11:10:08 +0100 Subject: [PATCH 03/19] add OrdinaryDiffEq as a test dependency --- test/Project.toml | 12 ++++++++++++ 1 file changed, 12 insertions(+) diff --git a/test/Project.toml b/test/Project.toml index fbc64c5..b337aa6 100644 --- a/test/Project.toml +++ b/test/Project.toml @@ -1,6 +1,18 @@ [deps] Test = "8dfed614-e22c-5e08-85e1-65c5234f0b40" Trixi = "a7f1ee26-1774-49b1-8366-f1abc58fbfcb" +OrdinaryDiffEq = "1dea7af3-3e70-54e6-95c3-0bf5283fa5ed" [compat] Trixi = "0.5, 0.6" +OrdinaryDiffEq = "6.49.1" + +[preferences.OrdinaryDiffEq] +PrecompileAutoSpecialize = false +PrecompileAutoSwitch = false +PrecompileDefaultSpecialize = false +PrecompileFunctionWrapperSpecialize = false +PrecompileLowStorage = false +PrecompileNoSpecialize = false +PrecompileNonStiff = false +PrecompileStiff = false From 83f321ad383ac575a5ca2f6bd83ccf90740de0ab Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 18 Dec 2023 11:52:50 +0100 Subject: [PATCH 04/19] comment functions related to HR_chen_noelle --- src/TrixiShallowWater.jl | 3 +- src/equations/shallow_water_wet_dry_1d.jl | 263 +++++++++++----------- 2 files changed, 135 insertions(+), 131 deletions(-) diff --git a/src/TrixiShallowWater.jl b/src/TrixiShallowWater.jl index e9561e7..e44cf50 100644 --- a/src/TrixiShallowWater.jl +++ b/src/TrixiShallowWater.jl @@ -9,6 +9,7 @@ include("equations/equations.jl") # export types/functions that define the public API of TrixiShallowWater.jl export ShallowWaterEquationsWetDry1D -export hydrostatic_reconstruction_chen_noelle, flux_nonconservative_chen_noelle +# TODO: These function are currently exported by Trixi.jl. Needs to be uncommented when removed from Trixi.jl +#export hydrostatic_reconstruction_chen_noelle, flux_nonconservative_chen_noelle, min_max_speed_chen_noelle end diff --git a/src/equations/shallow_water_wet_dry_1d.jl b/src/equations/shallow_water_wet_dry_1d.jl index dc3d9b4..8937191 100644 --- a/src/equations/shallow_water_wet_dry_1d.jl +++ b/src/equations/shallow_water_wet_dry_1d.jl @@ -269,52 +269,53 @@ Further details on the hydrostatic reconstruction and its motivation can be foun eps())) end -""" - flux_nonconservative_chen_noelle(u_ll, u_rr, - orientation::Integer, - equations::ShallowWaterEquationsWetDry1D) - -Non-symmetric two-point surface flux that discretizes the nonconservative (source) term. -The discretization uses the `hydrostatic_reconstruction_chen_noelle` on the conservative -variables. - -Should be used together with [`FluxHydrostaticReconstruction`](@ref) and -[`hydrostatic_reconstruction_chen_noelle`](@ref) in the surface flux to ensure consistency. - -Further details on the hydrostatic reconstruction and its motivation can be found in -- Guoxian Chen and Sebastian Noelle (2017) - A new hydrostatic reconstruction scheme based on subcell reconstructions - [DOI:10.1137/15M1053074](https://dx.doi.org/10.1137/15M1053074) -""" -@inline function flux_nonconservative_chen_noelle(u_ll, u_rr, - orientation::Integer, - equations::ShallowWaterEquationsWetDry1D) - - # Pull the water height and bottom topography on the left - h_ll, _, b_ll = u_ll - h_rr, _, b_rr = u_rr - - H_ll = h_ll + b_ll - H_rr = h_rr + b_rr - - b_star = min(max(b_ll, b_rr), min(H_ll, H_rr)) - - # Create the hydrostatic reconstruction for the left solution state - u_ll_star, _ = hydrostatic_reconstruction_chen_noelle(u_ll, u_rr, equations) - - # Copy the reconstructed water height for easier to read code - h_ll_star = u_ll_star[1] - - z = zero(eltype(u_ll)) - # Includes two parts: - # (i) Diagonal (consistent) term from the volume flux that uses `b_ll` to avoid - # cross-averaging across a discontinuous bottom topography - # (ii) True surface part that uses `h_ll` and `h_ll_star` to handle discontinuous bathymetry - return SVector(z, - equations.gravity * h_ll * b_ll - - equations.gravity * (h_ll_star + h_ll) * (b_ll - b_star), - z) -end +# TODO: This function is currently exported by Trixi.jl. Needs to be uncommented when removed from Trixi.jl +# """ +# flux_nonconservative_chen_noelle(u_ll, u_rr, +# orientation::Integer, +# equations::ShallowWaterEquationsWetDry1D) + +# Non-symmetric two-point surface flux that discretizes the nonconservative (source) term. +# The discretization uses the `hydrostatic_reconstruction_chen_noelle` on the conservative +# variables. + +# Should be used together with [`FluxHydrostaticReconstruction`](@ref) and +# [`hydrostatic_reconstruction_chen_noelle`](@ref) in the surface flux to ensure consistency. + +# Further details on the hydrostatic reconstruction and its motivation can be found in +# - Guoxian Chen and Sebastian Noelle (2017) +# A new hydrostatic reconstruction scheme based on subcell reconstructions +# [DOI:10.1137/15M1053074](https://dx.doi.org/10.1137/15M1053074) +# """ +# @inline function flux_nonconservative_chen_noelle(u_ll, u_rr, +# orientation::Integer, +# equations::ShallowWaterEquationsWetDry1D) + +# # Pull the water height and bottom topography on the left +# h_ll, _, b_ll = u_ll +# h_rr, _, b_rr = u_rr + +# H_ll = h_ll + b_ll +# H_rr = h_rr + b_rr + +# b_star = min(max(b_ll, b_rr), min(H_ll, H_rr)) + +# # Create the hydrostatic reconstruction for the left solution state +# u_ll_star, _ = hydrostatic_reconstruction_chen_noelle(u_ll, u_rr, equations) + +# # Copy the reconstructed water height for easier to read code +# h_ll_star = u_ll_star[1] + +# z = zero(eltype(u_ll)) +# # Includes two parts: +# # (i) Diagonal (consistent) term from the volume flux that uses `b_ll` to avoid +# # cross-averaging across a discontinuous bottom topography +# # (ii) True surface part that uses `h_ll` and `h_ll_star` to handle discontinuous bathymetry +# return SVector(z, +# equations.gravity * h_ll * b_ll - +# equations.gravity * (h_ll_star + h_ll) * (b_ll - b_star), +# z) +# end """ flux_nonconservative_ersing_etal(u_ll, u_rr, orientation::Integer, @@ -412,65 +413,66 @@ Further details on this hydrostatic reconstruction and its motivation can be fou eps())) end -""" - hydrostatic_reconstruction_chen_noelle(u_ll, u_rr, orientation::Integer, - equations::ShallowWaterEquationsWetDry1D) - -A particular type of hydrostatic reconstruction of the water height to guarantee well-balancedness -for a general bottom topography of the [`ShallowWaterEquationsWetDry1D`](@ref). The reconstructed solution states -`u_ll_star` and `u_rr_star` variables are used to evaluate the surface numerical flux at the interface. -The key idea is a linear reconstruction of the bottom and water height at the interfaces using subcells. -Use in combination with the generic numerical flux routine [`FluxHydrostaticReconstruction`](@ref). - -Further details on this hydrostatic reconstruction and its motivation can be found in -- Guoxian Chen and Sebastian Noelle (2017) - A new hydrostatic reconstruction scheme based on subcell reconstructions - [DOI:10.1137/15M1053074](https://dx.doi.org/10.1137/15M1053074) -""" -@inline function hydrostatic_reconstruction_chen_noelle(u_ll, u_rr, - equations::ShallowWaterEquationsWetDry1D) - # Unpack left and right water heights and bottom topographies - h_ll, _, b_ll = u_ll - h_rr, _, b_rr = u_rr - - # Get the velocities on either side - v_ll = velocity(u_ll, equations) - v_rr = velocity(u_rr, equations) - - H_ll = b_ll + h_ll - H_rr = b_rr + h_rr - - b_star = min(max(b_ll, b_rr), min(H_ll, H_rr)) - - # Compute the reconstructed water heights - h_ll_star = min(H_ll - b_star, h_ll) - h_rr_star = min(H_rr - b_star, h_rr) - - # Set the water height to be at least the value stored in the variable threshold after - # the hydrostatic reconstruction is applied and before the numerical flux is calculated - # to avoid numerical problem with arbitrary small values. Interfaces with a water height - # lower or equal to the threshold can be declared as dry. - # The default value for `threshold_wet` is ≈ 5*eps(), or 1e-15 in double precision, is set - # in the `ShallowWaterEquationsWetDry1D` struct. This threshold value can be changed in the constructor - # call of this equation struct in an elixir. - threshold = equations.threshold_wet - - if (h_ll_star <= threshold) - h_ll_star = threshold - v_ll = zero(v_ll) - end - - if (h_rr_star <= threshold) - h_rr_star = threshold - v_rr = zero(v_rr) - end - - # Create the conservative variables using the reconstruted water heights - u_ll_star = SVector(h_ll_star, h_ll_star * v_ll, b_ll) - u_rr_star = SVector(h_rr_star, h_rr_star * v_rr, b_rr) - - return u_ll_star, u_rr_star -end +# TODO: This function is currently exported by Trixi.jl. Needs to be uncommented when removed from Trixi.jl +# """ +# hydrostatic_reconstruction_chen_noelle(u_ll, u_rr, orientation::Integer, +# equations::ShallowWaterEquationsWetDry1D) + +# A particular type of hydrostatic reconstruction of the water height to guarantee well-balancedness +# for a general bottom topography of the [`ShallowWaterEquationsWetDry1D`](@ref). The reconstructed solution states +# `u_ll_star` and `u_rr_star` variables are used to evaluate the surface numerical flux at the interface. +# The key idea is a linear reconstruction of the bottom and water height at the interfaces using subcells. +# Use in combination with the generic numerical flux routine [`FluxHydrostaticReconstruction`](@ref). + +# Further details on this hydrostatic reconstruction and its motivation can be found in +# - Guoxian Chen and Sebastian Noelle (2017) +# A new hydrostatic reconstruction scheme based on subcell reconstructions +# [DOI:10.1137/15M1053074](https://dx.doi.org/10.1137/15M1053074) +# """ +# @inline function hydrostatic_reconstruction_chen_noelle(u_ll, u_rr, +# equations::ShallowWaterEquationsWetDry1D) +# # Unpack left and right water heights and bottom topographies +# h_ll, _, b_ll = u_ll +# h_rr, _, b_rr = u_rr + +# # Get the velocities on either side +# v_ll = velocity(u_ll, equations) +# v_rr = velocity(u_rr, equations) + +# H_ll = b_ll + h_ll +# H_rr = b_rr + h_rr + +# b_star = min(max(b_ll, b_rr), min(H_ll, H_rr)) + +# # Compute the reconstructed water heights +# h_ll_star = min(H_ll - b_star, h_ll) +# h_rr_star = min(H_rr - b_star, h_rr) + +# # Set the water height to be at least the value stored in the variable threshold after +# # the hydrostatic reconstruction is applied and before the numerical flux is calculated +# # to avoid numerical problem with arbitrary small values. Interfaces with a water height +# # lower or equal to the threshold can be declared as dry. +# # The default value for `threshold_wet` is ≈ 5*eps(), or 1e-15 in double precision, is set +# # in the `ShallowWaterEquationsWetDry1D` struct. This threshold value can be changed in the constructor +# # call of this equation struct in an elixir. +# threshold = equations.threshold_wet + +# if (h_ll_star <= threshold) +# h_ll_star = threshold +# v_ll = zero(v_ll) +# end + +# if (h_rr_star <= threshold) +# h_rr_star = threshold +# v_rr = zero(v_rr) +# end + +# # Create the conservative variables using the reconstruted water heights +# u_ll_star = SVector(h_ll_star, h_ll_star * v_ll, b_ll) +# u_rr_star = SVector(h_rr_star, h_rr_star * v_rr, b_rr) + +# return u_ll_star, u_rr_star +# end # Calculate maximum wave speed for local Lax-Friedrichs-type dissipation as the # maximum velocity magnitude plus the maximum speed of sound @@ -512,37 +514,38 @@ end eps())) end -""" - min_max_speed_chen_noelle(u_ll, u_rr, orientation::Integer, - equations::ShallowWaterEquationsWetDry1D) +# TODO: This function is currently exported by Trixi.jl. Needs to be uncommented when removed from Trixi.jl +# """ +# min_max_speed_chen_noelle(u_ll, u_rr, orientation::Integer, +# equations::ShallowWaterEquationsWetDry1D) -The approximated speeds for the HLL type numerical flux used by Chen and Noelle for their -hydrostatic reconstruction. As they state in the paper, these speeds are chosen for the numerical -flux to ensure positivity and to satisfy an entropy inequality. +# The approximated speeds for the HLL type numerical flux used by Chen and Noelle for their +# hydrostatic reconstruction. As they state in the paper, these speeds are chosen for the numerical +# flux to ensure positivity and to satisfy an entropy inequality. -Further details on this hydrostatic reconstruction and its motivation can be found in -- Guoxian Chen and Sebastian Noelle (2017) - A new hydrostatic reconstruction scheme based on subcell reconstructions - [DOI:10.1137/15M1053074](https://dx.doi.org/10.1137/15M1053074) -""" -@inline function min_max_speed_chen_noelle(u_ll, u_rr, orientation::Integer, - equations::ShallowWaterEquationsWetDry1D) - # Get the velocity quantities - v_ll = velocity(u_ll, equations) - v_rr = velocity(u_rr, equations) +# Further details on this hydrostatic reconstruction and its motivation can be found in +# - Guoxian Chen and Sebastian Noelle (2017) +# A new hydrostatic reconstruction scheme based on subcell reconstructions +# [DOI:10.1137/15M1053074](https://dx.doi.org/10.1137/15M1053074) +# """ +# @inline function min_max_speed_chen_noelle(u_ll, u_rr, orientation::Integer, +# equations::ShallowWaterEquationsWetDry1D) +# # Get the velocity quantities +# v_ll = velocity(u_ll, equations) +# v_rr = velocity(u_rr, equations) - # Calculate the wave celerity on the left and right - h_ll = waterheight(u_ll, equations) - h_rr = waterheight(u_rr, equations) +# # Calculate the wave celerity on the left and right +# h_ll = waterheight(u_ll, equations) +# h_rr = waterheight(u_rr, equations) - a_ll = sqrt(equations.gravity * h_ll) - a_rr = sqrt(equations.gravity * h_rr) +# a_ll = sqrt(equations.gravity * h_ll) +# a_rr = sqrt(equations.gravity * h_rr) - λ_min = min(v_ll - a_ll, v_rr - a_rr, zero(eltype(u_ll))) - λ_max = max(v_ll + a_ll, v_rr + a_rr, zero(eltype(u_ll))) +# λ_min = min(v_ll - a_ll, v_rr - a_rr, zero(eltype(u_ll))) +# λ_max = max(v_ll + a_ll, v_rr + a_rr, zero(eltype(u_ll))) - return λ_min, λ_max -end +# return λ_min, λ_max +# end # More refined estimates for minimum and maximum wave speeds for HLL-type fluxes @inline function Trixi.min_max_speed_davis(u_ll, u_rr, orientation::Integer, From 02c28846fe1cce3dd4f8b2c5e74d3674bc2adf5d Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 18 Dec 2023 13:26:19 +0100 Subject: [PATCH 05/19] fix tests --- src/TrixiShallowWater.jl | 2 ++ 1 file changed, 2 insertions(+) diff --git a/src/TrixiShallowWater.jl b/src/TrixiShallowWater.jl index e44cf50..ca3748c 100644 --- a/src/TrixiShallowWater.jl +++ b/src/TrixiShallowWater.jl @@ -7,6 +7,8 @@ using Static: True, False include("equations/equations.jl") +baz() = Trixi.examples_dir() + # export types/functions that define the public API of TrixiShallowWater.jl export ShallowWaterEquationsWetDry1D # TODO: These function are currently exported by Trixi.jl. Needs to be uncommented when removed from Trixi.jl From b86c4ae5a793c20a554a18d72420b2cd71cb9ffd Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 18 Dec 2023 17:01:59 +0100 Subject: [PATCH 06/19] fix tests #2 --- .github/workflows/ci.yml | 22 +++++++++++++++++++--- 1 file changed, 19 insertions(+), 3 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 8dc9958..230d63a 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -31,19 +31,21 @@ concurrency: jobs: test: if: "!contains(github.event.head_commit.message, 'skip ci')" - name: Julia ${{ matrix.version }} - ${{ matrix.os }} - ${{ matrix.arch }} - ${{ github.event_name }} + name: ${{ matrix.trixi_test }} - ${{ matrix.os }} - ${{ matrix.arch }} - ${{ github.event_name }} runs-on: ${{ matrix.os }} strategy: fail-fast: false matrix: version: - - '1.8' + - '1.9' os: - ubuntu-latest - macOS-latest - windows-latest arch: - x64 + trixi_test: + - tree_part1 steps: - uses: actions/checkout@v4 - uses: julia-actions/setup-julia@v1 @@ -53,7 +55,21 @@ jobs: - run: julia -e 'using InteractiveUtils; versioninfo(verbose=true)' - uses: julia-actions/cache@v1 - uses: julia-actions/julia-buildpkg@v1 - - uses: julia-actions/julia-runtest@v1 + #- uses: julia-actions/julia-runtest@v1 + - name: Run tests without coverage + uses: julia-actions/julia-runtest@v1 + with: + coverage: false + env: + PYTHON: "" + TRIXI_TEST: ${{ matrix.trixi_test }} + - name: Run tests with coverage + uses: julia-actions/julia-runtest@v1 + with: + coverage: true + env: + PYTHON: "" + TRIXI_TEST: ${{ matrix.trixi_test }} - uses: julia-actions/julia-processcoverage@v1 - uses: codecov/codecov-action@v3 with: From 9f49340f352fecf03ce163ebdf909084490af821 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 18 Dec 2023 17:12:06 +0100 Subject: [PATCH 07/19] fix tests #3 --- test/runtests.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/test/runtests.jl b/test/runtests.jl index fb8e5ef..4a7f4f4 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -10,7 +10,7 @@ const TRIXI_MPI_NPROCS = clamp(Sys.CPU_THREADS, 2, 3) const TRIXI_NTHREADS = clamp(Sys.CPU_THREADS, 2, 3) @time @testset "TrixiShallowWater.jl tests" begin - @time if TRIXI_TEST == "all" + @time if TRIXI_TEST == "all" || TRIXI_TEST == "tree_part1" include("test_tree_1d_shallowwater_wet_dry.jl") end From df7d15e550766c8a27f8defbe3a07ab29b75f99a Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 18 Dec 2023 18:00:45 +0100 Subject: [PATCH 08/19] fix tests 4 --- .github/workflows/ci.yml | 2 +- src/TrixiShallowWater.jl | 2 -- test/runtests.jl | 6 ++---- 3 files changed, 3 insertions(+), 7 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 230d63a..47e8b02 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -45,7 +45,7 @@ jobs: arch: - x64 trixi_test: - - tree_part1 + - all steps: - uses: actions/checkout@v4 - uses: julia-actions/setup-julia@v1 diff --git a/src/TrixiShallowWater.jl b/src/TrixiShallowWater.jl index ca3748c..e44cf50 100644 --- a/src/TrixiShallowWater.jl +++ b/src/TrixiShallowWater.jl @@ -7,8 +7,6 @@ using Static: True, False include("equations/equations.jl") -baz() = Trixi.examples_dir() - # export types/functions that define the public API of TrixiShallowWater.jl export ShallowWaterEquationsWetDry1D # TODO: These function are currently exported by Trixi.jl. Needs to be uncommented when removed from Trixi.jl diff --git a/test/runtests.jl b/test/runtests.jl index 4a7f4f4..00acf9d 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -10,13 +10,11 @@ const TRIXI_MPI_NPROCS = clamp(Sys.CPU_THREADS, 2, 3) const TRIXI_NTHREADS = clamp(Sys.CPU_THREADS, 2, 3) @time @testset "TrixiShallowWater.jl tests" begin - @time if TRIXI_TEST == "all" || TRIXI_TEST == "tree_part1" + @time if TRIXI_TEST == "all" include("test_tree_1d_shallowwater_wet_dry.jl") end @time if TRIXI_TEST == "all" || TRIXI_TEST == "upstream" - @testset "baz()" begin - @test TrixiShallowWater.baz() isa String - end + end end From dba63bc1511dcad819372a603bc713cb9522050a Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 18 Dec 2023 18:38:17 +0100 Subject: [PATCH 09/19] fix tests 5 --- .github/workflows/ci.yml | 10 +--------- 1 file changed, 1 insertion(+), 9 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 47e8b02..b5a6cfa 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -31,7 +31,7 @@ concurrency: jobs: test: if: "!contains(github.event.head_commit.message, 'skip ci')" - name: ${{ matrix.trixi_test }} - ${{ matrix.os }} - ${{ matrix.arch }} - ${{ github.event_name }} + name: Julia ${{ matrix.version }} - ${{ matrix.os }} - ${{ matrix.arch }} - ${{ github.event_name }} runs-on: ${{ matrix.os }} strategy: fail-fast: false @@ -44,8 +44,6 @@ jobs: - windows-latest arch: - x64 - trixi_test: - - all steps: - uses: actions/checkout@v4 - uses: julia-actions/setup-julia@v1 @@ -60,16 +58,10 @@ jobs: uses: julia-actions/julia-runtest@v1 with: coverage: false - env: - PYTHON: "" - TRIXI_TEST: ${{ matrix.trixi_test }} - name: Run tests with coverage uses: julia-actions/julia-runtest@v1 with: coverage: true - env: - PYTHON: "" - TRIXI_TEST: ${{ matrix.trixi_test }} - uses: julia-actions/julia-processcoverage@v1 - uses: codecov/codecov-action@v3 with: From 5d5d29afffc677945c6c7bd2ebe2878489421ebf Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 18 Dec 2023 18:55:10 +0100 Subject: [PATCH 10/19] fix tests 6 --- .github/workflows/ci.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index b5a6cfa..2558db3 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -37,7 +37,7 @@ jobs: fail-fast: false matrix: version: - - '1.9' + - '1.8' os: - ubuntu-latest - macOS-latest From b58e902baf638041925b338b25b76a156ac2d9dc Mon Sep 17 00:00:00 2001 From: patrickersing Date: Tue, 19 Dec 2023 11:59:44 +0100 Subject: [PATCH 11/19] add unit tests and namespace conflict test --- src/equations/shallow_water_wet_dry_1d.jl | 19 -------- test/runtests.jl | 9 +++- test/test_unit.jl | 53 +++++++++++++++++++++++ 3 files changed, 61 insertions(+), 20 deletions(-) create mode 100644 test/test_unit.jl diff --git a/src/equations/shallow_water_wet_dry_1d.jl b/src/equations/shallow_water_wet_dry_1d.jl index 8937191..04bba57 100644 --- a/src/equations/shallow_water_wet_dry_1d.jl +++ b/src/equations/shallow_water_wet_dry_1d.jl @@ -627,25 +627,6 @@ end eps())) end -""" - calc_wavespeed_roe(u_ll, u_rr, direction::Integer, - equations::ShallowWaterEquationsWetDry1D) - -Calculate Roe-averaged velocity `v_roe` and wavespeed `c_roe = sqrt{g * h_roe}` -See for instance equation (62) in -- Paul A. Ullrich, Christiane Jablonowski, and Bram van Leer (2010) - High-order finite-volume methods for the shallow-water equations on the sphere - [DOI: 10.1016/j.jcp.2010.04.044](https://doi.org/10.1016/j.jcp.2010.04.044) -Or equation (9.17) in [this lecture notes](https://metaphor.ethz.ch/x/2019/hs/401-4671-00L/literature/mishra_hyperbolic_pdes.pdf). -""" -@inline function Trixi.calc_wavespeed_roe(u_ll, u_rr, direction::Integer, - equations::ShallowWaterEquationsWetDry1D) - return Trixi.calc_wavespeed_roe(u_ll, u_rr, direction::Integer, - Trixi.ShallowWaterEquations1D(equations.gravity, - equations.H0, eps(), - eps())) -end - # Entropy function for the shallow water equations is the total energy @inline function Trixi.entropy(cons, equations::ShallowWaterEquationsWetDry1D) Trixi.energy_total(cons, equations) diff --git a/test/runtests.jl b/test/runtests.jl index 00acf9d..0da42f1 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -1,3 +1,4 @@ +using Trixi using TrixiShallowWater using Test @@ -12,9 +13,15 @@ const TRIXI_NTHREADS = clamp(Sys.CPU_THREADS, 2, 3) @time @testset "TrixiShallowWater.jl tests" begin @time if TRIXI_TEST == "all" include("test_tree_1d_shallowwater_wet_dry.jl") + include("test_unit.jl") end @time if TRIXI_TEST == "all" || TRIXI_TEST == "upstream" - + @testset "Namespace conflicts" begin + # Test for namespace conflicts between TrixiShallowWater.jl and Trixi.jl + for name in names(Trixi) + @test !(name in names(TrixiShallowWater)) + end + end end end diff --git a/test/test_unit.jl b/test/test_unit.jl new file mode 100644 index 0000000..195e3bf --- /dev/null +++ b/test/test_unit.jl @@ -0,0 +1,53 @@ +module TestUnit + +using Test +using Trixi +using TrixiShallowWater + +include("test_trixi.jl") + +# Start with a clean environment: remove Trixi.jl output directory if it exists +outdir = "out" +isdir(outdir) && rm(outdir, recursive = true) + +# Run various unit (= non-elixir-triggered) tests +@testset "Unit tests" begin +#! format: noindent + +@timed_testset "Shallow water conversion between conservative/entropy variables" begin + H, v, b = 3.5, 0.25, 0.4 + + let equations = ShallowWaterEquations1D(gravity_constant = 9.8) + cons_vars = Trixi.prim2cons(SVector(H, v, b), equations) + entropy_vars = Trixi.cons2entropy(cons_vars, equations) + @test cons_vars ≈ Trixi.entropy2cons(entropy_vars, equations) + + total_energy = Trixi.energy_total(cons_vars, equations) + @test total_energy ≈ Trixi.entropy(cons_vars, equations) + + # test tuple args + cons_vars = Trixi.prim2cons((H, v, b), equations) + entropy_vars = Trixi.cons2entropy(cons_vars, equations) + @test cons_vars ≈ Trixi.entropy2cons(entropy_vars, equations) + end +end + +@timed_testset "Connectivity with Trixi.jl" begin + u = SVector(SVector(1, 0.5, 0.0)) + orientation = 1 + equations = ShallowWaterEquationsWetDry1D(gravity_constant = 9.81) + equations_trixi = ShallowWaterEquations1D(gravity_constant = 9.81) + + # We only need to check equivalence between the equation systems. The functionality is tested in + # Trixi.jl. We choose these specific ones to improve code coverage, as they are not tested in + # any elixirs. + @test min_max_speed_einfeldt(u, u, orientation, equations) == + min_max_speed_einfeldt(u, u, orientation, equations_trixi) + @test min_max_speed_davis(u, u, orientation, equations) == + min_max_speed_davis(u, u, orientation, equations_trixi) + @test max_abs_speed_naive(u, u, orientation, equations) == + max_abs_speed_naive(u, u, orientation, equations_trixi) +end +end + +end # module From d0eb6dd7eeed1e223c9fffc9b9b8c9f800a02a39 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Tue, 19 Dec 2023 13:05:32 +0100 Subject: [PATCH 12/19] update unit tests, comment wave speed estimates --- src/equations/shallow_water_wet_dry_1d.jl | 70 +++++++++++++---------- test/test_unit.jl | 36 ++++++------ 2 files changed, 58 insertions(+), 48 deletions(-) diff --git a/src/equations/shallow_water_wet_dry_1d.jl b/src/equations/shallow_water_wet_dry_1d.jl index 04bba57..fc22649 100644 --- a/src/equations/shallow_water_wet_dry_1d.jl +++ b/src/equations/shallow_water_wet_dry_1d.jl @@ -474,15 +474,17 @@ end # return u_ll_star, u_rr_star # end -# Calculate maximum wave speed for local Lax-Friedrichs-type dissipation as the -# maximum velocity magnitude plus the maximum speed of sound -@inline function Trixi.max_abs_speed_naive(u_ll, u_rr, orientation::Integer, - equations::ShallowWaterEquationsWetDry1D) - return Trixi.max_abs_speed_naive(u_ll, u_rr, orientation, - Trixi.ShallowWaterEquations1D(equations.gravity, - equations.H0, eps(), - eps())) -end +# TODO: Probably can be removed. Since we dispatch to ShallowWaterEquations1D we don't need +# equation specific wave speed estimates. +# # Calculate maximum wave speed for local Lax-Friedrichs-type dissipation as the +# # maximum velocity magnitude plus the maximum speed of sound +# @inline function Trixi.max_abs_speed_naive(u_ll, u_rr, orientation::Integer, +# equations::ShallowWaterEquationsWetDry1D) +# return Trixi.max_abs_speed_naive(u_ll, u_rr, orientation, +# Trixi.ShallowWaterEquations1D(equations.gravity, +# equations.H0, eps(), +# eps())) +# end # Specialized `DissipationLocalLaxFriedrichs` to avoid spurious dissipation in the bottom topography @inline function (dissipation::Trixi.DissipationLocalLaxFriedrichs)(u_ll, u_rr, @@ -505,14 +507,16 @@ end eps())) end -# Calculate estimate for minimum and maximum wave speeds for HLL-type fluxes -@inline function Trixi.min_max_speed_naive(u_ll, u_rr, orientation::Integer, - equations::ShallowWaterEquationsWetDry1D) - return Trixi.min_max_speed_naive(u_ll, u_rr, orientation, - Trixi.ShallowWaterEquations1D(equations.gravity, - equations.H0, eps(), - eps())) -end +# TODO: Probably can be removed. Since we dispatch to ShallowWaterEquations1D we don't need +# equation specific wave speed estimates. +# # Calculate estimate for minimum and maximum wave speeds for HLL-type fluxes +# @inline function Trixi.min_max_speed_naive(u_ll, u_rr, orientation::Integer, +# equations::ShallowWaterEquationsWetDry1D) +# return Trixi.min_max_speed_naive(u_ll, u_rr, orientation, +# Trixi.ShallowWaterEquations1D(equations.gravity, +# equations.H0, eps(), +# eps())) +# end # TODO: This function is currently exported by Trixi.jl. Needs to be uncommented when removed from Trixi.jl # """ @@ -547,22 +551,26 @@ end # return λ_min, λ_max # end +# TODO: Probably can be removed. Since we dispatch to ShallowWaterEquations1D we don't need +# equation specific wave speed estimates. # More refined estimates for minimum and maximum wave speeds for HLL-type fluxes -@inline function Trixi.min_max_speed_davis(u_ll, u_rr, orientation::Integer, - equations::ShallowWaterEquationsWetDry1D) - return Trixi.min_max_speed_davis(u_ll, u_rr, orientation, - Trixi.ShallowWaterEquations1D(equations.gravity, - equations.H0, eps(), - eps())) -end +# @inline function Trixi.min_max_speed_davis(u_ll, u_rr, orientation::Integer, +# equations::ShallowWaterEquationsWetDry1D) +# return Trixi.min_max_speed_davis(u_ll, u_rr, orientation, +# Trixi.ShallowWaterEquations1D(equations.gravity, +# equations.H0, eps(), +# eps())) +# end -@inline function Trixi.min_max_speed_einfeldt(u_ll, u_rr, orientation::Integer, - equations::ShallowWaterEquationsWetDry1D) - return Trixi.min_max_speed_einfeldt(u_ll, u_rr, orientation, - Trixi.ShallowWaterEquations1D(equations.gravity, - equations.H0, - eps(), eps())) -end +# TODO: Probably can be removed. Since we dispatch to ShallowWaterEquations1D we don't need +# equation specific wave speed estimates. +# @inline function Trixi.min_max_speed_einfeldt(u_ll, u_rr, orientation::Integer, +# equations::ShallowWaterEquationsWetDry1D) +# return Trixi.min_max_speed_einfeldt(u_ll, u_rr, orientation, +# Trixi.ShallowWaterEquations1D(equations.gravity, +# equations.H0, +# eps(), eps())) +# end @inline function Trixi.max_abs_speeds(u, equations::ShallowWaterEquationsWetDry1D) return Trixi.max_abs_speeds(u, diff --git a/test/test_unit.jl b/test/test_unit.jl index 195e3bf..a1474e3 100644 --- a/test/test_unit.jl +++ b/test/test_unit.jl @@ -17,7 +17,7 @@ isdir(outdir) && rm(outdir, recursive = true) @timed_testset "Shallow water conversion between conservative/entropy variables" begin H, v, b = 3.5, 0.25, 0.4 - let equations = ShallowWaterEquations1D(gravity_constant = 9.8) + let equations = ShallowWaterEquationsWetDry1D(gravity_constant = 9.8) cons_vars = Trixi.prim2cons(SVector(H, v, b), equations) entropy_vars = Trixi.cons2entropy(cons_vars, equations) @test cons_vars ≈ Trixi.entropy2cons(entropy_vars, equations) @@ -32,22 +32,24 @@ isdir(outdir) && rm(outdir, recursive = true) end end -@timed_testset "Connectivity with Trixi.jl" begin - u = SVector(SVector(1, 0.5, 0.0)) - orientation = 1 - equations = ShallowWaterEquationsWetDry1D(gravity_constant = 9.81) - equations_trixi = ShallowWaterEquations1D(gravity_constant = 9.81) - - # We only need to check equivalence between the equation systems. The functionality is tested in - # Trixi.jl. We choose these specific ones to improve code coverage, as they are not tested in - # any elixirs. - @test min_max_speed_einfeldt(u, u, orientation, equations) == - min_max_speed_einfeldt(u, u, orientation, equations_trixi) - @test min_max_speed_davis(u, u, orientation, equations) == - min_max_speed_davis(u, u, orientation, equations_trixi) - @test max_abs_speed_naive(u, u, orientation, equations) == - max_abs_speed_naive(u, u, orientation, equations_trixi) -end +# TODO: Probably can be removed. Since we dispatch to ShallowWaterEquations1D we don't need +# equation specific wave speed estimates. +# @timed_testset "Connectivity with Trixi.jl" begin +# u = SVector(SVector(1, 0.5, 0.0)) +# orientation = 1 +# equations = ShallowWaterEquationsWetDry1D(gravity_constant = 9.81) +# equations_trixi = ShallowWaterEquations1D(gravity_constant = 9.81) + +# # We only need to check equivalence between the equation systems. The functionality is tested in +# # Trixi.jl. We choose these specific ones to improve code coverage, as they are not tested in +# # any elixirs. +# @test min_max_speed_einfeldt(u, u, orientation, equations) == +# min_max_speed_einfeldt(u, u, orientation, equations_trixi) +# @test min_max_speed_davis(u, u, orientation, equations) == +# min_max_speed_davis(u, u, orientation, equations_trixi) +# @test max_abs_speed_naive(u, u, orientation, equations) == +# max_abs_speed_naive(u, u, orientation, equations_trixi) +# end end end # module From 5fc8eec4e3ebfb3fb1a6860aeea4f136f3bd4721 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Tue, 19 Dec 2023 13:38:39 +0100 Subject: [PATCH 13/19] update unit test --- test/test_unit.jl | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/test/test_unit.jl b/test/test_unit.jl index a1474e3..448f8fc 100644 --- a/test/test_unit.jl +++ b/test/test_unit.jl @@ -24,7 +24,9 @@ isdir(outdir) && rm(outdir, recursive = true) total_energy = Trixi.energy_total(cons_vars, equations) @test total_energy ≈ Trixi.entropy(cons_vars, equations) - + @test total_energy ≈ + Trixi.energy_internal(cons_vars, equations) + + energy_kinetic(cons_vars, equations) # test tuple args cons_vars = Trixi.prim2cons((H, v, b), equations) entropy_vars = Trixi.cons2entropy(cons_vars, equations) From 7e9f002cfbfcf924a2d04ab4904e2ee688fc3afe Mon Sep 17 00:00:00 2001 From: patrickersing Date: Wed, 20 Dec 2023 11:13:50 +0100 Subject: [PATCH 14/19] reset ci.yml --- .github/workflows/ci.yml | 12 ++---------- 1 file changed, 2 insertions(+), 10 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 2558db3..9807181 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -53,15 +53,7 @@ jobs: - run: julia -e 'using InteractiveUtils; versioninfo(verbose=true)' - uses: julia-actions/cache@v1 - uses: julia-actions/julia-buildpkg@v1 - #- uses: julia-actions/julia-runtest@v1 - - name: Run tests without coverage - uses: julia-actions/julia-runtest@v1 - with: - coverage: false - - name: Run tests with coverage - uses: julia-actions/julia-runtest@v1 - with: - coverage: true + - uses: julia-actions/julia-runtest@v1 - uses: julia-actions/julia-processcoverage@v1 - uses: codecov/codecov-action@v3 with: @@ -69,4 +61,4 @@ jobs: - uses: coverallsapp/github-action@master with: github-token: ${{ secrets.GITHUB_TOKEN }} - path-to-lcov: ./lcov.info + path-to-lcov: ./lcov.info \ No newline at end of file From 376023080ba79a25ae689032e4e73bf662578dd0 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Wed, 20 Dec 2023 11:34:32 +0100 Subject: [PATCH 15/19] delete coverage part from CI --- .github/workflows/ci.yml | 10 +--------- 1 file changed, 1 insertion(+), 9 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 9807181..48edccb 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -53,12 +53,4 @@ jobs: - run: julia -e 'using InteractiveUtils; versioninfo(verbose=true)' - uses: julia-actions/cache@v1 - uses: julia-actions/julia-buildpkg@v1 - - uses: julia-actions/julia-runtest@v1 - - uses: julia-actions/julia-processcoverage@v1 - - uses: codecov/codecov-action@v3 - with: - files: lcov.info - - uses: coverallsapp/github-action@master - with: - github-token: ${{ secrets.GITHUB_TOKEN }} - path-to-lcov: ./lcov.info \ No newline at end of file + - uses: julia-actions/julia-runtest@v1 \ No newline at end of file From 32c3e36eaa4f8ae067ba1161da6ecbc0027497b2 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Wed, 20 Dec 2023 12:09:30 +0100 Subject: [PATCH 16/19] remove cache --- .github/workflows/ci.yml | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 48edccb..4957dfa 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -51,6 +51,13 @@ jobs: version: ${{ matrix.version }} arch: ${{ matrix.arch }} - run: julia -e 'using InteractiveUtils; versioninfo(verbose=true)' - - uses: julia-actions/cache@v1 - uses: julia-actions/julia-buildpkg@v1 - - uses: julia-actions/julia-runtest@v1 \ No newline at end of file + - uses: julia-actions/julia-runtest@v1 + - uses: julia-actions/julia-processcoverage@v1 + - uses: codecov/codecov-action@v3 + with: + files: lcov.info + - uses: coverallsapp/github-action@master + with: + github-token: ${{ secrets.GITHUB_TOKEN }} + path-to-lcov: ./lcov.info \ No newline at end of file From b77a00fe9922f679166f2cd998e344325fba1160 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Fri, 12 Jan 2024 17:31:10 +0100 Subject: [PATCH 17/19] comment unused lines --- src/equations/shallow_water_wet_dry_1d.jl | 44 +++++++++++------------ 1 file changed, 22 insertions(+), 22 deletions(-) diff --git a/src/equations/shallow_water_wet_dry_1d.jl b/src/equations/shallow_water_wet_dry_1d.jl index fc22649..e71bf28 100644 --- a/src/equations/shallow_water_wet_dry_1d.jl +++ b/src/equations/shallow_water_wet_dry_1d.jl @@ -579,12 +579,12 @@ end eps())) end -# Helper function to extract the velocity vector from the conservative variables -@inline function Trixi.velocity(u, equations::ShallowWaterEquationsWetDry1D) - return Trixi.velocity(u, - Trixi.ShallowWaterEquations1D(equations.gravity, equations.H0, - eps(), eps())) -end +# # Helper function to extract the velocity vector from the conservative variables +# @inline function Trixi.velocity(u, equations::ShallowWaterEquationsWetDry1D) +# return Trixi.velocity(u, +# Trixi.ShallowWaterEquations1D(equations.gravity, equations.H0, +# eps(), eps())) +# end # Convert conservative variables to primitive @inline function Trixi.cons2prim(u, equations::ShallowWaterEquationsWetDry1D) @@ -616,24 +616,24 @@ end equations.H0, eps(), eps())) end -@inline function Trixi.waterheight(u, equations::ShallowWaterEquationsWetDry1D) - return Trixi.waterheight(u, - Trixi.ShallowWaterEquations1D(equations.gravity, - equations.H0, eps(), eps())) -end +# @inline function Trixi.waterheight(u, equations::ShallowWaterEquationsWetDry1D) +# return Trixi.waterheight(u, +# Trixi.ShallowWaterEquations1D(equations.gravity, +# equations.H0, eps(), eps())) +# end -@inline function Trixi.pressure(u, equations::ShallowWaterEquationsWetDry1D) - return Trixi.pressure(u, - Trixi.ShallowWaterEquations1D(equations.gravity, equations.H0, - eps(), eps())) -end +# @inline function Trixi.pressure(u, equations::ShallowWaterEquationsWetDry1D) +# return Trixi.pressure(u, +# Trixi.ShallowWaterEquations1D(equations.gravity, equations.H0, +# eps(), eps())) +# end -@inline function Trixi.waterheight_pressure(u, equations::ShallowWaterEquationsWetDry1D) - return Trixi.waterheight_pressure(u, - Trixi.ShallowWaterEquations1D(equations.gravity, - equations.H0, eps(), - eps())) -end +# @inline function Trixi.waterheight_pressure(u, equations::ShallowWaterEquationsWetDry1D) +# return Trixi.waterheight_pressure(u, +# Trixi.ShallowWaterEquations1D(equations.gravity, +# equations.H0, eps(), +# eps())) +# end # Entropy function for the shallow water equations is the total energy @inline function Trixi.entropy(cons, equations::ShallowWaterEquationsWetDry1D) From 95c595d781d49e4ee8c1eb26ec803f395da82dde Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 15 Jan 2024 16:23:57 +0100 Subject: [PATCH 18/19] add uncovered line, change ci --- .github/workflows/ci.yml | 2 ++ src/equations/shallow_water_wet_dry_1d.jl | 10 +++++----- 2 files changed, 7 insertions(+), 5 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index dd05983..2e37bfb 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -64,6 +64,8 @@ jobs: - uses: codecov/codecov-action@v3 with: files: lcov.info + fail_ci_if_error: false + token: ${{ secrets.CODECOV_TOKEN }} - uses: coverallsapp/github-action@master with: github-token: ${{ secrets.GITHUB_TOKEN }} diff --git a/src/equations/shallow_water_wet_dry_1d.jl b/src/equations/shallow_water_wet_dry_1d.jl index e71bf28..d5f1744 100644 --- a/src/equations/shallow_water_wet_dry_1d.jl +++ b/src/equations/shallow_water_wet_dry_1d.jl @@ -616,11 +616,11 @@ end equations.H0, eps(), eps())) end -# @inline function Trixi.waterheight(u, equations::ShallowWaterEquationsWetDry1D) -# return Trixi.waterheight(u, -# Trixi.ShallowWaterEquations1D(equations.gravity, -# equations.H0, eps(), eps())) -# end +@inline function Trixi.waterheight(u, equations::ShallowWaterEquationsWetDry1D) + return Trixi.waterheight(u, + Trixi.ShallowWaterEquations1D(equations.gravity, + equations.H0, eps(), eps())) +end # @inline function Trixi.pressure(u, equations::ShallowWaterEquationsWetDry1D) # return Trixi.pressure(u, From 8621d61013323b4c2b4946346689ffb1be333fa3 Mon Sep 17 00:00:00 2001 From: patrickersing Date: Mon, 15 Jan 2024 17:45:10 +0100 Subject: [PATCH 19/19] add codecov.yml --- .codecov.yml | 21 +++++++++++++++++++++ 1 file changed, 21 insertions(+) create mode 100644 .codecov.yml diff --git a/.codecov.yml b/.codecov.yml new file mode 100644 index 0000000..3bb0be1 --- /dev/null +++ b/.codecov.yml @@ -0,0 +1,21 @@ +# https://docs.codecov.io/docs/codecovyml-reference + +# We have +# 1 * [number of basic OS] + 1 * [number of additional OS] +# with +# [number of basic OS] = 1 (Linux) +# [number of additional OS] = 2 (Windows, MacOS) +codecov: + branch: main + notify: + after_n_builds: 3 +comment: + after_n_builds: 3 + +coverage: + range: 80..95 # set 95% and above as solid green, everything below 80% as red + round: nearest + precision: 2 + +github_checks: + annotations: false \ No newline at end of file