Skip to content
New issue

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

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

Already on GitHub? Sign in to your account

Covariant formulation of the shallow water equations #58

Open
wants to merge 25 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
25 commits
Select commit Hold shift + click to select a range
c63f514
add additional aux vars
tristanmontoya Dec 12, 2024
32f49a9
compute christoffel symbols
tristanmontoya Dec 18, 2024
668d63d
fix typo in christoffel calculation
tristanmontoya Dec 19, 2024
9508cc2
covariant SWE weak form runs now
tristanmontoya Dec 23, 2024
1b5e975
covariant SWE weak form runs now
tristanmontoya Dec 23, 2024
b620511
no longer pass surface_integral into prolong2mortars
tristanmontoya Dec 23, 2024
b8410a0
bump compat for Trixi from 0.9 to 0.9.9
tristanmontoya Dec 23, 2024
ee9572b
Merge branch 'tm/remove_surface_integral_from_prolong2mortars' into t…
tristanmontoya Dec 25, 2024
1e725bc
add test for spherical SWE
tristanmontoya Dec 25, 2024
f741b92
approach to automatically transform to desired coordinate system
tristanmontoya Dec 30, 2024
a90bfa0
added missing aux_vars to prim2cons
tristanmontoya Dec 31, 2024
b9c3591
both spherical and cartesian works now
tristanmontoya Dec 31, 2024
9c720dd
add EC formulation
tristanmontoya Jan 2, 2025
7317ed3
add tests and improve elixirs/docs/cases
tristanmontoya Jan 2, 2025
4272119
more detailed docs
tristanmontoya Jan 3, 2025
2a5d2a7
fix typo in docs
tristanmontoya Jan 3, 2025
9021899
improve docs more
tristanmontoya Jan 3, 2025
0356b55
add docstring for covariant SWE
tristanmontoya Jan 3, 2025
192c84b
sqrtG -> J
tristanmontoya Jan 3, 2025
bacf745
fix typo in quadrilaterals
tristanmontoya Jan 3, 2025
33590da
Merge remote-tracking branch 'origin/main' into tm/new_shallow_water
tristanmontoya Jan 3, 2025
4622df4
add vorticity calculation for covariant form
tristanmontoya Jan 5, 2025
18d9e91
improve docs and comments
tristanmontoya Jan 6, 2025
0f4c546
fix mistake in comment
tristanmontoya Jan 7, 2025
5491153
Rossby-Haurwitz is actually Case 6, not Case 5
tristanmontoya Jan 8, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
81 changes: 81 additions & 0 deletions examples/elixir_shallowwater_covariant_geostrophic_balance.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
###############################################################################
# DGSEM for the shallow water equations in covariant form on the cubed sphere
###############################################################################

using OrdinaryDiffEq, Trixi, TrixiAtmo

###############################################################################
# Parameters

initial_condition = initial_condition_geostrophic_balance
polydeg = 3
cells_per_dimension = 5
n_saves = 10
tspan = (0.0, 5.0 * SECONDS_PER_DAY)

###############################################################################
# Spatial discretization

mesh = P4estMeshCubedSphere2D(cells_per_dimension, EARTH_RADIUS, polydeg = polydeg,
initial_refinement_level = 0,
element_local_mapping = true)

equations = CovariantShallowWaterEquations2D(EARTH_GRAVITATIONAL_ACCELERATION,
EARTH_ROTATION_RATE,
global_coordinate_system = GlobalSphericalCoordinates())

# The covariant shallow water equations are treated as a nonconservative system in order to
# handle flux-differencing formulations of the covariant derivative. With
# VolumeIntegralWeakForm, there are actually no nonconservative terms, but we must still
# pass a no-op function "flux_nonconservative_weak_form" as the nonconservative surface flux
surface_flux = (flux_lax_friedrichs, flux_nonconservative_weak_form)

# Create DG solver with polynomial degree = p
solver = DGSEM(polydeg = polydeg, volume_integral = VolumeIntegralWeakForm(),
surface_flux = surface_flux)

# Transform the initial condition to the proper set of conservative variables
initial_condition_transformed = transform_initial_condition(initial_condition, equations)

# A semidiscretization collects data structures and functions for the spatial discretization
semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition_transformed, solver,
source_terms = source_terms_weak_form)

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

# Create ODE problem with time span from 0 to T
ode = semidiscretize(semi, tspan)

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

# The AnalysisCallback allows to analyse the solution in regular intervals and prints the
# results
analysis_callback = AnalysisCallback(semi, interval = 200,
save_analysis = true,
extra_analysis_errors = (:conservation_error,))

# The SaveSolutionCallback allows to save the solution to a file in regular intervals
save_solution = SaveSolutionCallback(dt = (tspan[2] - tspan[1]) / n_saves,
solution_variables = cons2cons)

# The StepsizeCallback handles the re-calculation of the maximum Δt after each time step
stepsize_callback = StepsizeCallback(cfl = 0.4)

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

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

# OrdinaryDiffEq's `solve` method evolves the solution in time and executes the passed
# callbacks
sol = solve(ode, CarpenterKennedy2N54(williamson_condition = false),
dt = 100.0, save_everystep = false, callback = callbacks);

# Print the timer summary
summary_callback()
84 changes: 84 additions & 0 deletions examples/elixir_shallowwater_covariant_rossby_haurwitz_EC.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,84 @@
###############################################################################
# Entropy-conservative DGSEM for the shallow water equations in covariant form
# on the cubed sphere
###############################################################################

using OrdinaryDiffEq, Trixi, TrixiAtmo

###############################################################################
# Parameters

initial_condition = initial_condition_rossby_haurwitz
polydeg = 3
cells_per_dimension = 5
n_saves = 10
tspan = (0.0, 7.0 * SECONDS_PER_DAY)

###############################################################################
# Spatial discretization

mesh = P4estMeshCubedSphere2D(cells_per_dimension, EARTH_RADIUS, polydeg = polydeg,
initial_refinement_level = 0,
element_local_mapping = true)

equations = CovariantShallowWaterEquations2D(EARTH_GRAVITATIONAL_ACCELERATION,
EARTH_ROTATION_RATE,
global_coordinate_system = GlobalCartesianCoordinates())

# Use entropy-conservative two-point fluxes for volume and surface terms
volume_flux = (flux_split_covariant, flux_nonconservative_split_covariant)
surface_flux = (flux_split_covariant, flux_nonconservative_split_covariant)

# The following flux should be used to add interface dissipation:
# surface_flux = (flux_split_covariant_lax_friedrichs, flux_nonconservative_split_covariant)

# Create DG solver with polynomial degree = p
solver = DGSEM(polydeg = polydeg, surface_flux = surface_flux,
volume_integral = VolumeIntegralFluxDifferencing(volume_flux))

# Transform the initial condition to the proper set of conservative variables
initial_condition_transformed = transform_initial_condition(initial_condition, equations)

# A semidiscretization collects data structures and functions for the spatial discretization
semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition_transformed, solver,
source_terms = source_terms_split_covariant)

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

# Create ODE problem with time span from 0 to T
ode = semidiscretize(semi, tspan)

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

# The AnalysisCallback allows to analyse the solution in regular intervals and prints the
# results. Note that entropy should be conserved at the semi-discrete level.
analysis_callback = AnalysisCallback(semi, interval = 200,
save_analysis = true,
extra_analysis_errors = (:conservation_error,),
extra_analysis_integrals = (entropy,))

# The SaveSolutionCallback allows to save the solution to a file in regular intervals
save_solution = SaveSolutionCallback(dt = (tspan[2] - tspan[1]) / n_saves,
solution_variables = cons2prim_and_vorticity)

# The StepsizeCallback handles the re-calculation of the maximum Δt after each time step
stepsize_callback = StepsizeCallback(cfl = 0.4)

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

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

# OrdinaryDiffEq's `solve` method evolves the solution in time and executes the passed
# callbacks
sol = solve(ode, CarpenterKennedy2N54(williamson_condition = false),
dt = 100.0, save_everystep = false, callback = callbacks);

# Print the timer summary
summary_callback()
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@ mesh = P4estMeshCubedSphere2D(cells_per_dimension[1], EARTH_RADIUS, polydeg = 3,
#initial_refinement_level = 0, # Comment to use cells_per_dimension in the convergence test
element_local_mapping = false)

# Transform the initial condition to the proper set of conservative variables
initial_condition_transformed = transform_initial_condition(initial_condition, equations)

# A semidiscretization collects data structures and functions for the spatial discretization
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,7 @@ mesh = P4estMeshCubedSphere2D(cells_per_dimension[1], EARTH_RADIUS,
polydeg = Trixi.polydeg(solver),
element_local_mapping = true)

# Transform the initial condition to the proper set of conservative variables
initial_condition_transformed = transform_initial_condition(initial_condition, equations)

# A semidiscretization collects data structures and functions for the spatial discretization
Expand Down
20 changes: 12 additions & 8 deletions examples/elixir_spherical_advection_covariant_quad_icosahedron.jl
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
###############################################################################
# DGSEM for the linear advection equation on the cubed sphere
# DGSEM for the linear advection equation on a quadrilateral icosahedral grid
###############################################################################
# To run a convergence test, use
# convergence_test("../examples/elixir_spherical_advection_covariant_quad_icosahedron.jl", 4, cells_per_dimension = (1,1))
Expand All @@ -18,12 +18,13 @@ equations = CovariantLinearAdvectionEquation2D(global_coordinate_system = Global
solver = DGSEM(polydeg = 3, surface_flux = flux_lax_friedrichs,
volume_integral = VolumeIntegralWeakForm())

# Create a 2D cubed sphere mesh the size of the Earth. For the covariant form to work
# properly, we currently need polydeg to equal that of the solver, and
# Create a 2D quadrilateral icosahedral mesh the size of the Earth. For the covariant form
# to work properly, we currently need polydeg to equal that of the solver, and
# initial_refinement_level = 0 (default)
mesh = P4estMeshQuadIcosahedron2D(cells_per_dimension[1], EARTH_RADIUS,
polydeg = Trixi.polydeg(solver))

# Transform the initial condition to the proper set of conservative variables
initial_condition_transformed = transform_initial_condition(initial_condition, equations)

# A semidiscretization collects data structures and functions for the spatial discretization
Expand All @@ -35,11 +36,12 @@ semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition_transform
# Create ODE problem with time span from 0 to T
ode = semidiscretize(semi, (0.0, 12 * SECONDS_PER_DAY))

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

# The AnalysisCallback allows to analyse the solution in regular intervals and prints the results
# The AnalysisCallback allows to analyse the solution in regular intervals and prints the
# results
analysis_callback = AnalysisCallback(semi, interval = 10,
save_analysis = true,
extra_analysis_errors = (:conservation_error,))
Expand All @@ -51,14 +53,16 @@ save_solution = SaveSolutionCallback(interval = 10,
# The StepsizeCallback handles the re-calculation of the maximum Δt after each time step
stepsize_callback = StepsizeCallback(cfl = 0.7)

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

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

# OrdinaryDiffEq's `solve` method evolves the solution in time and executes the passed callbacks
# OrdinaryDiffEq's `solve` method evolves the solution in time and executes the passed
# callbacks
sol = solve(ode, CarpenterKennedy2N54(williamson_condition = false),
dt = 1.0, save_everystep = false, callback = callbacks);

Expand Down
19 changes: 14 additions & 5 deletions src/TrixiAtmo.jl
Original file line number Diff line number Diff line change
Expand Up @@ -31,22 +31,31 @@ include("semidiscretization/semidiscretization_hyperbolic_2d_manifold_in_3d.jl")
include("callbacks_step/callbacks_step.jl")

export CompressibleMoistEulerEquations2D, ShallowWaterEquations3D,
CovariantLinearAdvectionEquation2D
CovariantLinearAdvectionEquation2D, CovariantShallowWaterEquations2D

export GlobalCartesianCoordinates, GlobalSphericalCoordinates

export flux_chandrashekar, flux_LMARS
export flux_chandrashekar, flux_LMARS, flux_split_covariant, flux_nonconservative_weak_form,
flux_nonconservative_split_covariant, flux_split_covariant_lax_friedrichs,
source_terms_split_covariant

export velocity, waterheight, pressure, energy_total, energy_kinetic, energy_internal,
lake_at_rest_error, source_terms_lagrange_multiplier,
lake_at_rest_error, source_terms_lagrange_multiplier, source_terms_weak_form,
clean_solution_lagrange_multiplier!

export cons2prim_and_vorticity

export P4estMeshCubedSphere2D, P4estMeshQuadIcosahedron2D, MetricTermsCrossProduct,
MetricTermsInvariantCurl

export EARTH_RADIUS, EARTH_GRAVITATIONAL_ACCELERATION,
EARTH_ROTATION_RATE, SECONDS_PER_DAY
export global2contravariant, contravariant2global, spherical2cartesian,

export global2contravariant, contravariant2global, spherical2cartesian, cartesian2spherical,
transform_initial_condition
export initial_condition_gaussian, initial_condition_gaussian_cartesian

export initial_condition_gaussian, initial_condition_geostrophic_balance,
initial_condition_rossby_haurwitz

export examples_dir
end # module TrixiAtmo
38 changes: 29 additions & 9 deletions src/callbacks_step/analysis_covariant.jl
Original file line number Diff line number Diff line change
@@ -1,9 +1,28 @@
@muladd begin
#! format: noindent

# When the equations are of type AbstractCovariantEquations, the functions which we would
# like to integrate depend on the solution as well as the auxiliary variables
function Trixi.integrate(func::Func, u,
mesh::Union{TreeMesh{2}, StructuredMesh{2},
StructuredMeshView{2},
UnstructuredMesh2D, P4estMesh{2}, T8codeMesh{2}},
equations::AbstractCovariantEquations{2}, dg::DG,
cache; normalize = true) where {Func}
(; aux_node_vars) = cache.auxiliary_variables

Trixi.integrate_via_indices(u, mesh, equations, dg, cache;
normalize = normalize) do u, i, j, element, equations,
dg
u_local = Trixi.get_node_vars(u, equations, dg, i, j, element)
aux_local = get_node_aux_vars(aux_node_vars, equations, dg, i, j, element)
return func(u_local, aux_local, equations)
end
end

# For the covariant form, we want to integrate using the exact area element
# √G = (det(AᵀA))^(1/2), which is stored in cache.auxiliary_variables, not the approximate
# area element used in the Cartesian formulation, which stored in cache.elements.
# J = (det(AᵀA))^(1/2), which is stored in cache.auxiliary_variables, not the approximate
# area element used in the Cartesian formulation, which stored in cache.elements
function Trixi.integrate_via_indices(func::Func, u,
mesh::Union{StructuredMesh{2},
StructuredMeshView{2},
Expand All @@ -23,10 +42,10 @@ function Trixi.integrate_via_indices(func::Func, u,
for element in eachelement(dg, cache)
for j in eachnode(dg), i in eachnode(dg)
aux_node = get_node_aux_vars(aux_node_vars, equations, dg, i, j, element)
sqrtG = area_element(aux_node, equations)
integral += weights[i] * weights[j] * sqrtG *
J = area_element(aux_node, equations)
integral += weights[i] * weights[j] * J *
func(u, i, j, element, equations, dg, args...)
total_volume += weights[i] * weights[j] * sqrtG
total_volume += weights[i] * weights[j] * J
end
end

Expand Down Expand Up @@ -65,6 +84,7 @@ function Trixi.calc_error_norms(func, u, t, analyzer, mesh::P4estMesh{2},
(; weights) = dg.basis
(; node_coordinates) = cache.elements
(; aux_node_vars) = cache.auxiliary_variables

# Set up data structures
l2_error = zero(func(Trixi.get_node_vars(u, equations, dg, 1, 1, 1), equations))
linf_error = copy(l2_error)
Expand All @@ -88,14 +108,14 @@ function Trixi.calc_error_norms(func, u, t, analyzer, mesh::P4estMesh{2},
diff = func(u_exact, equations) - func(u_numerical, equations)

# For the L2 error, integrate with respect to area element stored in aux vars
sqrtG = area_element(aux_node, equations)
l2_error += diff .^ 2 * (weights[i] * weights[j] * sqrtG)
J = area_element(aux_node, equations)
l2_error += diff .^ 2 * (weights[i] * weights[j] * J)

# Compute Linf error as usual
linf_error = @. max(linf_error, abs(diff))

# Increment total volume according to the volume element stored in aux vars
total_volume += weights[i] * weights[j] * sqrtG
total_volume += weights[i] * weights[j] * J
end
end

Expand All @@ -104,4 +124,4 @@ function Trixi.calc_error_norms(func, u, t, analyzer, mesh::P4estMesh{2},

return l2_error, linf_error
end
end # muladd
end # @muladd
13 changes: 8 additions & 5 deletions src/callbacks_step/save_solution_2d_manifold_in_3d_cartesian.jl
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,8 @@

# Convert to another set of variables using a solution_variables function
function convert_variables(u, solution_variables, mesh::P4estMesh{2},
equations::AbstractEquations{3}, dg, cache)
equations::AbstractEquations{3},
dg, cache)
(; contravariant_vectors) = cache.elements
# Extract the number of solution variables to be output
# (may be different than the number of conservative variables)
Expand Down Expand Up @@ -129,7 +130,7 @@ end

u_node = Trixi.get_node_vars(u, equations, dg, i, j, element)

# Return th solution variables
# Return the solution variables
return SVector(cons2prim(u_node, equations)..., relative_vorticity)
end

Expand Down Expand Up @@ -177,7 +178,9 @@ end
end

# Variable names for cons2prim_and_vorticity
Trixi.varnames(::typeof(cons2prim_and_vorticity), equations::ShallowWaterEquations3D) = (varnames(cons2prim,
equations)...,
"vorticity")
function Trixi.varnames(::typeof(cons2prim_and_vorticity),
equations::Union{ShallowWaterEquations3D,
AbstractCovariantEquations{2}})
return (varnames(cons2prim, equations)..., "vorticity")
end
end # @muladd
Loading
Loading