diff --git a/examples/t8code_2d_fv/elixir_advection_basic.jl b/examples/t8code_2d_fv/elixir_advection_basic.jl index 198c30fd22a..e16fc5a7ff1 100644 --- a/examples/t8code_2d_fv/elixir_advection_basic.jl +++ b/examples/t8code_2d_fv/elixir_advection_basic.jl @@ -16,6 +16,8 @@ solver = FV(order = 2, extended_reconstruction_stencil = false, # Afterwards and only inside Trixi, `tree_node_coordinates` are mapped back to [-1, 1]^2. # But, this variable is not used for the FV method. # That's why I use the cmesh interface in all other elixirs. + +# Option 1: coordinates coordinates_min = (0.0, 0.0) # minimum coordinates (min(x), min(y)) coordinates_max = (8.0, 8.0) # maximum coordinates (max(x), max(y)) # Note and TODO: The plan is to move the auxiliary routine f and the macro to a different place. diff --git a/examples/t8code_2d_fv/elixir_advection_gauss.jl b/examples/t8code_2d_fv/elixir_advection_gauss.jl index 177fce928c6..d4394451df6 100644 --- a/examples/t8code_2d_fv/elixir_advection_gauss.jl +++ b/examples/t8code_2d_fv/elixir_advection_gauss.jl @@ -11,9 +11,9 @@ initial_condition = initial_condition_gauss solver = FV(order = 2, surface_flux = flux_lax_friedrichs) initial_refinement_level = 4 +# Note: The initial_condition is set up for a domain [-5,5]^2. +# This becomes important when using a non-periodic mesh. cmesh = Trixi.cmesh_new_periodic_hybrid() -# Note: A non-periodic run with the tri mesh is unstable. Same as in `elixir_advection_nonperiodic.jl` -# cmesh = Trixi.cmesh_new_tri(periodicity = (false, false)) mesh = T8codeMesh(cmesh, solver, initial_refinement_level = initial_refinement_level) diff --git a/examples/t8code_2d_fv/elixir_advection_nonperiodic.jl b/examples/t8code_2d_fv/elixir_advection_nonperiodic.jl index 7e956fe1754..fbe7940617d 100644 --- a/examples/t8code_2d_fv/elixir_advection_nonperiodic.jl +++ b/examples/t8code_2d_fv/elixir_advection_nonperiodic.jl @@ -23,7 +23,7 @@ solver = FV(order = 2, surface_flux = flux_lax_friedrichs) initial_refinement_level = 5 cmesh = Trixi.cmesh_new_quad(periodicity = (false, false)) -# **Note**: A non-periodic run with the tri mesh is unstable. (Same happens for a non-periodic run with `elixir_advection_gauss.jl`) +# **Note**: A non-periodic run with the tri mesh is unstable. # - This only happens with **2nd order** # - When increasing refinement_level to 6 and lower CFL to 0.4, it seems like the simulation is stable again (except of some smaller noises at the corners) # - With a lower resolution (5) I cannot get the simulation stable. Even with a cfl of 0.01 etc. diff --git a/examples/t8code_3d_fv/elixir_advection_basic.jl b/examples/t8code_3d_fv/elixir_advection_basic.jl new file mode 100644 index 00000000000..4fe2f147fd4 --- /dev/null +++ b/examples/t8code_3d_fv/elixir_advection_basic.jl @@ -0,0 +1,96 @@ +using OrdinaryDiffEq +using Trixi +using T8code + +advection_velocity = (0.2, -0.7, 0.5) +equations = LinearScalarAdvectionEquation3D(advection_velocity) + +initial_condition = initial_condition_convergence_test + +solver = FV(order = 2, extended_reconstruction_stencil = false, + surface_flux = flux_lax_friedrichs) + +# Option 1: coordinates +# For all problems see the 2D file... +coordinates_min = (0.0, 0.0, 0.0) # minimum coordinates (min(x), min(y), min(z)) +coordinates_max = (8.0, 8.0, 8.0) # maximum coordinates (max(x), max(y), max(z)) + +mapping_coordinates = Trixi.coordinates2mapping(coordinates_min, coordinates_max) + +# Option 3: classic mapping +function mapping(xi, eta, zeta) + x = 4 * (xi + 1) + y = 4 * (eta + 1) + z = 4 * (zeta + 1) + + return SVector(x, y, z) +end + +function trixi_t8_mapping(cmesh, gtreeid, ref_coords, num_coords, out_coords, + tree_data, user_data) + ltreeid = t8_cmesh_get_local_id(cmesh, gtreeid) + eclass = t8_cmesh_get_tree_class(cmesh, ltreeid) + T8code.t8_geom_compute_linear_geometry(eclass, tree_data, + ref_coords, num_coords, out_coords) + + for i in 1:num_coords + offset_3d = 3 * (i - 1) + 1 + + xi = unsafe_load(out_coords, offset_3d) + eta = unsafe_load(out_coords, offset_3d + 1) + zeta = unsafe_load(out_coords, offset_3d + 2) + # xyz = mapping_coordinates(xi, eta, zeta) + xyz = mapping(xi, eta, zeta) + + unsafe_store!(out_coords, xyz[1], offset_3d) + unsafe_store!(out_coords, xyz[2], offset_3d + 1) + unsafe_store!(out_coords, xyz[3], offset_3d + 2) + end + + return nothing +end + +function trixi_t8_mapping_c() + @cfunction($trixi_t8_mapping, Cvoid, + (t8_cmesh_t, t8_gloidx_t, Ptr{Cdouble}, Csize_t, + Ptr{Cdouble}, Ptr{Cvoid}, Ptr{Cvoid})) +end + +trees_per_dimension = (2, 2, 2) + +eclass = T8_ECLASS_HEX +mesh = T8codeMesh(trees_per_dimension, eclass; + mapping = trixi_t8_mapping_c(), + # Plan is to use either + # coordinates_max = coordinates_max, coordinates_min = coordinates_min, + # or at least + # mapping = mapping, + initial_refinement_level = 5) + +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver) + +tspan = (0.0, 1.0) +ode = semidiscretize(semi, tspan); + +summary_callback = SummaryCallback() + +analysis_interval = 100 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +save_solution = SaveSolutionCallback(interval = 10, + solution_variables = cons2prim) + +stepsize_callback = StepsizeCallback(cfl = 0.9) + +callbacks = CallbackSet(summary_callback, analysis_callback, alive_callback, + stepsize_callback, save_solution) + +############################################################################### +# run the simulation + +sol = solve(ode, CarpenterKennedy2N54(williamson_condition = false), + dt = 1.0, # solve needs some value here but it will be overwritten by the stepsize_callback + save_everystep = false, callback = callbacks); +summary_callback() diff --git a/examples/t8code_3d_fv/elixir_advection_basic_hybrid.jl b/examples/t8code_3d_fv/elixir_advection_basic_hybrid.jl new file mode 100644 index 00000000000..8d6dddca1e0 --- /dev/null +++ b/examples/t8code_3d_fv/elixir_advection_basic_hybrid.jl @@ -0,0 +1,42 @@ +using OrdinaryDiffEq +using Trixi + +advection_velocity = (0.2, -0.7, 0.5) +equations = LinearScalarAdvectionEquation3D(advection_velocity) + +initial_condition = initial_condition_convergence_test + +solver = FV(order = 2, extended_reconstruction_stencil = false, + surface_flux = flux_lax_friedrichs) + +# TODO: There are no other cmesh functions implemented yet in 3d. +cmesh = Trixi.cmesh_new_quad_3d(periodicity = (true, true, true)) +mesh = T8codeMesh(cmesh, solver, + initial_refinement_level = 4) + +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver) + +ode = semidiscretize(semi, (0.0, 1.0)); + +summary_callback = SummaryCallback() + +analysis_interval = 100 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +save_solution = SaveSolutionCallback(interval = 1, + solution_variables = cons2prim) + +stepsize_callback = StepsizeCallback(cfl = 0.9) + +callbacks = CallbackSet(summary_callback, analysis_callback, alive_callback, + stepsize_callback, save_solution) + +############################################################################### +# run the simulation + +sol = solve(ode, CarpenterKennedy2N54(williamson_condition = false), + dt = 1.0, # solve needs some value here but it will be overwritten by the stepsize_callback + save_everystep = false, callback = callbacks); +summary_callback() diff --git a/examples/t8code_3d_fv/elixir_advection_gauss.jl b/examples/t8code_3d_fv/elixir_advection_gauss.jl new file mode 100644 index 00000000000..71e6d82eca6 --- /dev/null +++ b/examples/t8code_3d_fv/elixir_advection_gauss.jl @@ -0,0 +1,47 @@ +using OrdinaryDiffEq +using Trixi + +#################################################### + +advection_velocity = (0.2, -0.7, 0.5) +equations = LinearScalarAdvectionEquation3D(advection_velocity) + +initial_condition = initial_condition_gauss + +solver = FV(order = 2, surface_flux = flux_lax_friedrichs) + +initial_refinement_level = 4 + +# TODO: There are no other cmesh functions implemented yet in 3d. +cmesh = Trixi.cmesh_new_quad_3d(coordinates_min = (-5.0, -5.0, -5.0), + coordinates_max = (5.0, 5.0, 5.0), + periodicity = (true, true, true)) +mesh = T8codeMesh(cmesh, solver, initial_refinement_level = initial_refinement_level) + +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver) + +tspan = (0.0, 5.0) +ode = semidiscretize(semi, tspan); + +summary_callback = SummaryCallback() + +analysis_interval = 100 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) + +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +save_solution = SaveSolutionCallback(interval = 10, + solution_variables = cons2prim) + +stepsize_callback = StepsizeCallback(cfl = 0.7) + +callbacks = CallbackSet(summary_callback, save_solution, analysis_callback, alive_callback, + stepsize_callback) + +############################################################################### +# run the simulation + +sol = solve(ode, CarpenterKennedy2N54(williamson_condition = false),#Euler(), + dt = 5.0e-2, # solve needs some value here but it will be overwritten by the stepsize_callback + save_everystep = false, callback = callbacks); +summary_callback() diff --git a/examples/t8code_3d_fv/elixir_advection_nonperiodic.jl b/examples/t8code_3d_fv/elixir_advection_nonperiodic.jl new file mode 100644 index 00000000000..414fcbe4a8f --- /dev/null +++ b/examples/t8code_3d_fv/elixir_advection_nonperiodic.jl @@ -0,0 +1,53 @@ +using OrdinaryDiffEq +using Trixi + +############################################################################### +# Semidiscretization of the linear advection equation. + +advection_velocity = (0.2, -0.7, 0.5) +equations = LinearScalarAdvectionEquation3D(advection_velocity) + +initial_condition = initial_condition_convergence_test + +boundary_condition = BoundaryConditionDirichlet(initial_condition) +boundary_conditions = Dict(:all => boundary_condition) +# Problem: T8codeMesh interface with parameter cmesh cannot distinguish between boundaries +# boundary_conditions = Dict(:x_neg => boundary_condition, +# :x_pos => boundary_condition, +# :y_neg => boundary_condition_periodic, +# :y_pos => boundary_condition_periodic) + +solver = FV(order = 2, surface_flux = flux_lax_friedrichs) + +# TODO: When using mesh construction as in elixir_advection_basic.jl boundary Symbol :all is not defined +initial_refinement_level = 5 +cmesh = Trixi.cmesh_new_quad_3d(periodicity = (false, false, false)) +mesh = T8codeMesh(cmesh, solver, initial_refinement_level = initial_refinement_level) + +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver, + boundary_conditions = boundary_conditions) + +############################################################################### +# ODE solvers, callbacks etc. + +tspan = (0.0, 1.0) +ode = semidiscretize(semi, tspan) + +summary_callback = SummaryCallback() + +analysis_interval = 100 +analysis_callback = AnalysisCallback(semi, interval = analysis_interval) +alive_callback = AliveCallback(analysis_interval = analysis_interval) + +stepsize_callback = StepsizeCallback(cfl = 0.8) + +callbacks = CallbackSet(summary_callback, analysis_callback, alive_callback, + 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() diff --git a/examples/t8code_3d_fv/elixir_euler_source_terms.jl b/examples/t8code_3d_fv/elixir_euler_source_terms.jl new file mode 100644 index 00000000000..e588e24674f --- /dev/null +++ b/examples/t8code_3d_fv/elixir_euler_source_terms.jl @@ -0,0 +1,51 @@ + +using OrdinaryDiffEq +using Trixi + +equations = CompressibleEulerEquations3D(1.4) + +initial_condition = initial_condition_convergence_test + +# boundary_condition = BoundaryConditionDirichlet(initial_condition) +# boundary_conditions = Dict(:all => boundary_condition) + +source_terms = source_terms_convergence_test + +solver = FV(order = 2, extended_reconstruction_stencil = false, + surface_flux = flux_lax_friedrichs) + +# TODO: There are no other cmesh functions implemented yet in 3d. +cmesh = Trixi.cmesh_new_quad_3d(periodicity = (true, true, true)) +mesh = T8codeMesh(cmesh, solver, + initial_refinement_level = 3) + +semi = SemidiscretizationHyperbolic(mesh, equations, initial_condition, solver, + source_terms = source_terms) +# boundary_conditions = boundary_conditions) + +tspan = (0.0, 2.0) +ode = semidiscretize(semi, tspan) + +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, + solution_variables = cons2prim) + +stepsize_callback = StepsizeCallback(cfl = 0.8) + +callbacks = CallbackSet(summary_callback, + analysis_callback, alive_callback, + save_solution, + stepsize_callback) + +sol = solve(ode, CarpenterKennedy2N54(williamson_condition = false), + dt = 1.0, # solve needs some value here but it will be overwritten by the stepsize_callback + save_everystep = false, callback = callbacks); +summary_callback() # print the timer summary diff --git a/src/equations/linear_scalar_advection_3d.jl b/src/equations/linear_scalar_advection_3d.jl index 088f934cc3e..9c68ce6cd47 100644 --- a/src/equations/linear_scalar_advection_3d.jl +++ b/src/equations/linear_scalar_advection_3d.jl @@ -64,14 +64,24 @@ function initial_condition_convergence_test(x, t, return SVector(scalar) end +# Calculates translated coordinates `x` for a periodic domain +function x_trans_periodic_3d(x, domain_length = SVector(10, 10, 10), + center = SVector(0, 0, 0)) + x_normalized = x .- center + x_shifted = x_normalized .% domain_length + x_offset = ((x_shifted .< -0.5f0 * domain_length) - + (x_shifted .> 0.5f0 * domain_length)) .* domain_length + return center + x_shifted + x_offset +end + """ - initial_condition_gauss(x, t, equations::LinearScalarAdvectionEquation1D) + initial_condition_gauss(x, t, equations::LinearScalarAdvectionEquation3D) A Gaussian pulse. """ function initial_condition_gauss(x, t, equation::LinearScalarAdvectionEquation3D) # Store translated coordinate for easy use of exact solution - x_trans = x - equation.advection_velocity * t + x_trans = x_trans_periodic_3d(x - equation.advection_velocity * t) scalar = exp(-(x_trans[1]^2 + x_trans[2]^2 + x_trans[3]^2)) return SVector(scalar) diff --git a/src/meshes/t8code_mesh.jl b/src/meshes/t8code_mesh.jl index 51078f8eece..47d0eb0304f 100644 --- a/src/meshes/t8code_mesh.jl +++ b/src/meshes/t8code_mesh.jl @@ -384,22 +384,23 @@ function T8codeMesh(trees_per_dimension, eclass; end do_partition = 0 - if NDIMS == 2 - cmesh_ref = Ref(t8_cmesh_t()) - t8_cmesh_init(cmesh_ref) - cmesh = cmesh_ref[] + cmesh_ref = Ref(t8_cmesh_t()) + t8_cmesh_init(cmesh_ref) + cmesh = cmesh_ref[] - @assert(trees_per_dimension[1]==trees_per_dimension[2], - "Different trees per dimensions are not supported for quad mesh. `trees_per_dimension`: $trees_per_dimension") - num_trees = prod(trees_per_dimension) * 1 # 1 = number of trees for single hypercube with quads + @assert(allequal(trees_per_dimension), + "Different trees per dimensions are not supported for quad mesh. `trees_per_dimension`: $trees_per_dimension") + num_trees = prod(trees_per_dimension) * 1 # 1 = number of trees for single hypercube with quads + vertices_per_tree = 2^NDIMS # number of vertices (=corners) in one tree + vertices = Vector{Cdouble}(undef, 3 * vertices_per_tree * num_trees) # 3 (dimensions) * vertices_per_tree) * num_trees + if NDIMS == 2 coordinates_tree_x = range(-1.0, 1.0, length = trees_per_dimension[1] + 1) coordinates_tree_y = range(-1.0, 1.0, length = trees_per_dimension[2] + 1) - vertices = Vector{Cdouble}(undef, 3 * 4 * num_trees) # 3 (dimensions) * 4 (vertices per tree) * num_trees for itree_y in 1:trees_per_dimension[2], itree_x in 1:trees_per_dimension[1] - index = trees_per_dimension[1] * 3 * 4 * (itree_y - 1) + - 3 * 4 * (itree_x - 1) + 1 + index = trees_per_dimension[1] * 3 * vertices_per_tree * (itree_y - 1) + + 3 * vertices_per_tree * (itree_x - 1) + 1 vertices[index] = coordinates_tree_x[itree_x] vertices[index + 1] = coordinates_tree_y[itree_y] vertices[index + 2] = 0.0 @@ -419,32 +420,88 @@ function T8codeMesh(trees_per_dimension, eclass; vertices[index + 1] = coordinates_tree_y[itree_y + 1] vertices[index + 2] = 0.0 end + elseif NDIMS == 3 + coordinates_tree_x = range(-1.0, 1.0, length = trees_per_dimension[1] + 1) + coordinates_tree_y = range(-1.0, 1.0, length = trees_per_dimension[2] + 1) + coordinates_tree_z = range(-1.0, 1.0, length = trees_per_dimension[3] + 1) + + for itree_z in 1:trees_per_dimension[3], itree_y in 1:trees_per_dimension[2], + itree_x in 1:trees_per_dimension[1] + + index = trees_per_dimension[1] * trees_per_dimension[2] * 3 * + vertices_per_tree * (itree_z - 1) + + trees_per_dimension[1] * 3 * vertices_per_tree * (itree_y - 1) + + 3 * vertices_per_tree * (itree_x - 1) + 1 + + vertices[index] = coordinates_tree_x[itree_x] + vertices[index + 1] = coordinates_tree_y[itree_y] + vertices[index + 2] = coordinates_tree_z[itree_z] + + index += 3 + vertices[index] = coordinates_tree_x[itree_x + 1] + vertices[index + 1] = coordinates_tree_y[itree_y] + vertices[index + 2] = coordinates_tree_z[itree_z] + + index += 3 + vertices[index] = coordinates_tree_x[itree_x] + vertices[index + 1] = coordinates_tree_y[itree_y + 1] + vertices[index + 2] = coordinates_tree_z[itree_z] - # analytical = trixi_t8_mapping_c(mapping) - analytical = mapping - name = "" - user_data = vertices - # user_data = C_NULL - - jacobian = C_NULL # type: t8_geom_analytic_jacobian_fn - # TODO: Since @t8_load_tree_data is not yet included into T8code.jl, precompiling Trixi fails because of this line. - # load_tree_data = @t8_load_tree_data(t8_geom_load_tree_data_vertices) # type: t8_geom_load_tree_data_fn - # For now, I remove it and pass C_NULL. Then, `elixir_advection_basic.jl` will fail with a SegFault. - load_tree_data = C_NULL - tree_negative_volume = C_NULL # type: t8_geom_tree_negative_volume_fn - - geometry = t8_geometry_analytic_new(NDIMS, name, analytical, jacobian, - load_tree_data, tree_negative_volume, user_data) - - t8_cmesh_register_geometry(cmesh, geometry) - - for itree in 1:num_trees - offset_vertices = 3 * 4 * (itree - 1) - t8_cmesh_set_tree_class(cmesh, itree - 1, eclass) - t8_cmesh_set_tree_vertices(cmesh, itree - 1, - @views(vertices[(1 + offset_vertices):end]), 4) + index += 3 + vertices[index] = coordinates_tree_x[itree_x + 1] + vertices[index + 1] = coordinates_tree_y[itree_y + 1] + vertices[index + 2] = coordinates_tree_z[itree_z] + + # itree_z + 1 + index += 3 + vertices[index] = coordinates_tree_x[itree_x] + vertices[index + 1] = coordinates_tree_y[itree_y] + vertices[index + 2] = coordinates_tree_z[itree_z + 1] + + index += 3 + vertices[index] = coordinates_tree_x[itree_x + 1] + vertices[index + 1] = coordinates_tree_y[itree_y] + vertices[index + 2] = coordinates_tree_z[itree_z + 1] + + index += 3 + vertices[index] = coordinates_tree_x[itree_x] + vertices[index + 1] = coordinates_tree_y[itree_y + 1] + vertices[index + 2] = coordinates_tree_z[itree_z + 1] + + index += 3 + vertices[index] = coordinates_tree_x[itree_x + 1] + vertices[index + 1] = coordinates_tree_y[itree_y + 1] + vertices[index + 2] = coordinates_tree_z[itree_z + 1] end + end + # analytical = trixi_t8_mapping_c(mapping) + analytical = mapping + name = "" + user_data = vertices + # user_data = C_NULL + + jacobian = C_NULL # type: t8_geom_analytic_jacobian_fn + # TODO: Since @t8_load_tree_data is not yet included into T8code.jl, precompiling Trixi fails because of this line. + # load_tree_data = @t8_load_tree_data(t8_geom_load_tree_data_vertices) # type: t8_geom_load_tree_data_fn + # For now, I remove it and pass C_NULL. Then, `elixir_advection_basic.jl` will fail with a SegFault. + load_tree_data = C_NULL + tree_negative_volume = C_NULL # type: t8_geom_tree_negative_volume_fn + + geometry = t8_geometry_analytic_new(NDIMS, name, analytical, jacobian, + load_tree_data, tree_negative_volume, user_data) + + t8_cmesh_register_geometry(cmesh, geometry) + + for itree in 1:num_trees + offset_vertices = 3 * vertices_per_tree * (itree - 1) + t8_cmesh_set_tree_class(cmesh, itree - 1, eclass) + t8_cmesh_set_tree_vertices(cmesh, itree - 1, + @views(vertices[(1 + offset_vertices):end]), + vertices_per_tree) + end + + if NDIMS == 2 # Note and TODO: # Only hardcoded to `trees_per_dimension = (1, 1) or (2, 2)` if num_trees == 1 @@ -467,11 +524,44 @@ function T8codeMesh(trees_per_dimension, eclass; else error("Not supported trees_per_dimension") end - - t8_cmesh_commit(cmesh, mpi_comm()) elseif NDIMS == 3 - error("3D not supported") + # Note and TODO: + # Only hardcoded to `trees_per_dimension = (1, 1, 1) or (2, 2, 2)` + if num_trees == 1 + if periodicity[1] + t8_cmesh_set_join(cmesh, 0, 0, 0, 1, 0) + end + if periodicity[2] + t8_cmesh_set_join(cmesh, 0, 0, 2, 3, 0) + end + if periodicity[3] + t8_cmesh_set_join(cmesh, 0, 0, 4, 5, 0) + end + elseif num_trees == 8 + if periodicity[1] + t8_cmesh_set_join(cmesh, 0, 1, 0, 1, 0) + t8_cmesh_set_join(cmesh, 2, 3, 0, 1, 0) + t8_cmesh_set_join(cmesh, 4, 5, 0, 1, 0) + t8_cmesh_set_join(cmesh, 6, 7, 0, 1, 0) + end + if periodicity[2] + t8_cmesh_set_join(cmesh, 0, 2, 2, 3, 0) + t8_cmesh_set_join(cmesh, 1, 3, 2, 3, 0) + t8_cmesh_set_join(cmesh, 4, 6, 2, 3, 0) + t8_cmesh_set_join(cmesh, 5, 7, 2, 3, 0) + end + if periodicity[3] + t8_cmesh_set_join(cmesh, 0, 4, 4, 5, 0) + t8_cmesh_set_join(cmesh, 1, 5, 4, 5, 0) + t8_cmesh_set_join(cmesh, 2, 6, 4, 5, 0) + t8_cmesh_set_join(cmesh, 3, 7, 4, 5, 0) + end + t8_cmesh_set_join_by_stash(cmesh, C_NULL, 0) + else + error("Not supported trees_per_dimension") + end end + t8_cmesh_commit(cmesh, mpi_comm()) do_face_ghost = mpi_isparallel() scheme = t8_scheme_new_default_cxx() @@ -1721,6 +1811,44 @@ function cmesh_new_quad(; trees_per_dimension = (1, 1), return cmesh end +function cmesh_new_quad_3d(; trees_per_dimension = (1, 1, 1), + coordinates_min = (-1.0, -1.0, -1.0), + coordinates_max = (1.0, 1.0, 1.0), + periodicity = (true, true, true))::t8_cmesh_t + # This is how the cmesh looks like. The numbers are the tree numbers: + # +---+ + # / /| + # +---+ | + # | | | + # | 0 | + + # | |/ + # +---+ + # + + polygons_x, polygons_y, polygons_z = trees_per_dimension + periodic_x, periodic_y, periodic_z = periodicity + boundary = [coordinates_min[1], coordinates_min[2], coordinates_min[3], + coordinates_max[1], coordinates_min[2], coordinates_min[3], + coordinates_min[1], coordinates_max[2], coordinates_min[3], + coordinates_max[1], coordinates_max[2], coordinates_min[3], + coordinates_min[1], coordinates_min[2], coordinates_max[3], + coordinates_max[1], coordinates_min[2], coordinates_max[3], + coordinates_min[1], coordinates_max[2], coordinates_max[3], + coordinates_max[1], coordinates_max[2], coordinates_max[3]] + + comm = mpi_comm() + set_partition = 0 + offset = 0.0 + use_axis_aligned = 0 + eclass = T8_ECLASS_HEX + cmesh = t8_cmesh_new_hypercube_pad_ext(eclass, comm, boundary, + polygons_x, polygons_y, polygons_z, + periodic_x, periodic_y, periodic_z, + use_axis_aligned, set_partition, offset) + + return cmesh +end + # TODO: The structure of `cmesh_new_quad` and `cmesh_new_tri` is equal and only differs for `eclass`. # Use only one routine! function cmesh_new_tri(; trees_per_dimension = (1, 1), @@ -1758,56 +1886,6 @@ function cmesh_new_tri(; trees_per_dimension = (1, 1), return cmesh end -# # Old version if triangular mesh -# function cmesh_new_periodic_tri(; periodicity = (true, true), comm = mpi_comm())::t8_cmesh_t -# n_dims = 2 -# vertices = [ # Just all vertices of all trees. partly duplicated -# -1.0, -1.0, 0, # tree 0, triangle -# 1.0, -1.0, 0, -# 1.0, 1.0, 0, -# -1.0, -1.0, 0, # tree 1, triangle -# 1.0, 1.0, 0, -# -1.0, 1.0, 0, -# ] - -# # Generally, one can define other geometries. But besides linear the other -# # geometries in t8code do not have C interface yet. -# linear_geom = t8_geometry_linear_new(n_dims) - -# # This is how the cmesh looks like. The numbers are the tree numbers: -# # -# # +---+ -# # |1 /| -# # | / | -# # |/0 | -# # +---+ -# # - -# cmesh_ref = Ref(t8_cmesh_t()) -# t8_cmesh_init(cmesh_ref) -# cmesh = cmesh_ref[] - -# # Use linear geometry -# t8_cmesh_register_geometry(cmesh, linear_geom) -# t8_cmesh_set_tree_class(cmesh, 0, T8_ECLASS_TRIANGLE) -# t8_cmesh_set_tree_class(cmesh, 1, T8_ECLASS_TRIANGLE) - -# t8_cmesh_set_tree_vertices(cmesh, 0, @views(vertices[(1 + 0):end]), 3) -# t8_cmesh_set_tree_vertices(cmesh, 1, @views(vertices[(1 + 9):end]), 3) - -# t8_cmesh_set_join(cmesh, 0, 1, 1, 2, 0) -# if periodicity[1] -# t8_cmesh_set_join(cmesh, 0, 1, 0, 1, 0) -# end -# if periodicity[2] -# t8_cmesh_set_join(cmesh, 0, 1, 2, 0, 1) -# end - -# t8_cmesh_commit(cmesh, comm) - -# return cmesh -# end - function cmesh_new_periodic_tri2(; comm = mpi_comm())::t8_cmesh_t n_dims = 2 vertices = [ # Just all vertices of all trees. partly duplicated diff --git a/src/solvers/fv_t8code/containers.jl b/src/solvers/fv_t8code/containers.jl index 18007d493d6..52278e0b69a 100644 --- a/src/solvers/fv_t8code/containers.jl +++ b/src/solvers/fv_t8code/containers.jl @@ -248,7 +248,7 @@ function init_elements!(elements, mesh::T8codeMesh, solver::FV) return nothing end -@inline function init_extended_reconstruction_stencil!(corners, elements, solver) +@inline function init_extended_reconstruction_stencil!(corners, elements, solver::FV) if solver.order != 2 return nothing end @@ -327,7 +327,7 @@ end function init_reconstruction_stencil!(elements, interfaces, boundaries, communication_data, - mesh, equations, solver) + mesh, equations, solver::FV) if solver.order != 2 return nothing end @@ -352,7 +352,7 @@ function init_reconstruction_stencil!(elements, interfaces, boundaries, face_midpoint_element1 = domain_data[element1].face_midpoints[face_element1] face_midpoint_element2 = domain_data[element2].face_midpoints[face_element2] - # How to handle periodic boundaries? + # TODO: How to handle periodic boundaries? if isapprox(face_midpoint_element1, face_midpoint_element2) distance = midpoint_element2 .- midpoint_element1 else diff --git a/src/solvers/fv_t8code/fv.jl b/src/solvers/fv_t8code/fv.jl index d3effbe7d77..e531c6ee8a5 100644 --- a/src/solvers/fv_t8code/fv.jl +++ b/src/solvers/fv_t8code/fv.jl @@ -216,7 +216,7 @@ function rhs!(du, u, t, mesh::T8codeMesh, equations, return nothing end -function calc_gradient_reconstruction!(u, mesh, equations, solver, cache) +function calc_gradient_reconstruction!(u, mesh::T8codeMesh{2}, equations, solver, cache) if solver.order == 1 return nothing elseif solver.order > 2 @@ -237,7 +237,7 @@ function calc_gradient_reconstruction!(u, mesh, equations, solver, cache) # (A^T A)^-1 = determinant_factor * [a3 -a2; -a2, a1] # A^T b = [d1; d2] - # Note: A^T b depends on the variable v. Using structure [d/e, v] + # Note: A^T b depends on the variable v. Using structure [1/2, v] a = zeros(eltype(u), 3) # [a1, a2, a3] d = zeros(eltype(u), size(u, 1), 2) # [d1(v), d2(v)] @@ -248,9 +248,12 @@ function calc_gradient_reconstruction!(u, mesh, equations, solver, cache) epsilon = 1.0e-13 for element in eachelement(mesh, solver, cache) + # The actual number of used stencils is `n_stencil_neighbors + 1`, since the full stencil is additionally used once. if solver.extended_reconstruction_stencil + # Number of faces = Number of corners n_stencil_neighbors = elements.num_faces[element] else + # Number of direct (edge) neighbors n_stencil_neighbors = length(reconstruction_stencil[element]) end @@ -262,7 +265,7 @@ function calc_gradient_reconstruction!(u, mesh, equations, solver, cache) d .= zero(eltype(u)) if solver.extended_reconstruction_stencil - calc_gradient_extended!(stencil, n_stencil_neighbors, element, a, d, + calc_gradient_extended!(stencil, element, a, d, reconstruction_stencil, reconstruction_distance, reconstruction_corner_elements, solution_data, equations, solver, cache) @@ -302,14 +305,14 @@ function calc_gradient_reconstruction!(u, mesh, equations, solver, cache) indicator = sum(gradient .^ 2) lambda_j = (j == 1) ? lambda[1] : lambda[2] weight = (lambda_j / (indicator + epsilon))^r - for x in axes(reconstruction_gradient_limited, 1) - reconstruction_gradient_limited[x, v, element] += weight * - gradient[x] + for dim in axes(reconstruction_gradient_limited, 1) + reconstruction_gradient_limited[dim, v, element] += weight * + gradient[dim] end weight_sum += weight end - for x in axes(reconstruction_gradient_limited, 1) - reconstruction_gradient_limited[x, v, element] /= weight_sum + for dim in axes(reconstruction_gradient_limited, 1) + reconstruction_gradient_limited[dim, v, element] /= weight_sum end end end @@ -322,7 +325,7 @@ end @inline function calc_gradient_simple!(stencil, n_stencil_neighbors, element, a, d, reconstruction_stencil, reconstruction_distance, - solution_data, equations) + solution_data, equations::AbstractEquations{2}) for i in 1:n_stencil_neighbors # stencil=1 contains information from all neighbors # stencil=2,...,n_stencil_neighbors+1 is computed without (stencil+1)-th neighbor's information @@ -347,11 +350,11 @@ end return nothing end -@inline function calc_gradient_extended!(stencil, n_stencil_neighbors, element, a, d, +@inline function calc_gradient_extended!(stencil, element, a, d, reconstruction_stencil, reconstruction_distance, reconstruction_corner_elements, solution_data, - equations, solver, cache) + equations::AbstractEquations{2}, solver, cache) if stencil == 1 # Full stencil for i in eachindex(reconstruction_stencil[element]) @@ -397,6 +400,233 @@ end return nothing end +function calc_gradient_reconstruction!(u, mesh::T8codeMesh{3}, equations, solver, cache) + if solver.order == 1 + return nothing + elseif solver.order > 2 + error("Order $(solver.order) is not supported yet!") + end + + (; elements, communication_data) = cache + (; reconstruction_stencil, reconstruction_distance, reconstruction_corner_elements, reconstruction_gradient, reconstruction_gradient_limited) = elements + (; solution_data) = communication_data + + # A N x 3 Matrix, where N is the number of stencil neighbors + # A^T A 3 x 3 Matrix + # b N Vector + # A^T b 3 Vector + + # Matrix/vector notation + # A^T A = [a11 a12 a13; a12 a22 a23; a13 a23 a33] + # det(A^T A) = a11 * (a22*a33 - a23^2) - a12 * (a12*a33 - a13*a23) + a13 * (a12*a23 - a22*a13) + # (A^T A)^-1 = 1/det(A^T A) * [(a33*a22-a23^2) -(a33*a12-a23*a13) (a23*a12-a22*a13) ; + # *** (a11*a33-a13^2) -(a23*a11-a12*a13); + # *** *** (a11*a22-a12^2) ] + + # A^T b = [d1; d2; d3] + # Note: A^T b depends on the variable v. Using structure [1/2/3, v] + a = zeros(eltype(u), 6) # [a11, a12, a13, a22, a23, a33] + d = zeros(eltype(u), size(u, 1), 3) # [d1(v), d2(v), d3(v)] + + # Parameter for limiting weights + lambda = [0.0, 1.0] + # lambda = [1.0, 0.0] # No limiting + r = 4 + epsilon = 1.0e-13 + + for element in eachelement(mesh, solver, cache) + # The actual number of used stencils is `n_stencil_neighbors + 1`, since the full stencil is additionally used once. + if solver.extended_reconstruction_stencil + # Number of faces = Number of corners + n_stencil_neighbors = elements.num_faces[element] + else + # Number of direct (surface) neighbors + n_stencil_neighbors = length(reconstruction_stencil[element]) + end + + for stencil in 1:(n_stencil_neighbors + 1) + # Reset variables + a .= zero(eltype(u)) + # A^T b = [d1(v), d2(v), d3(v)] + # Note: A^T b depends on the variable v. Using vectors with d[v, 1/2/3] + d .= zero(eltype(u)) + + if solver.extended_reconstruction_stencil + calc_gradient_extended!(stencil, element, a, d, + reconstruction_stencil, reconstruction_distance, + reconstruction_corner_elements, solution_data, + equations, solver, cache) + else + calc_gradient_simple!(stencil, n_stencil_neighbors, element, a, d, + reconstruction_stencil, reconstruction_distance, + solution_data, equations) + end + + # Divide by determinant + # Determinant = a11 * (a22 * a33 - a23^2 ) - a12 * (a12 * a33 - a13 * a23 ) + a13 * (a12 * a23 - a22 * a13 ) + AT_A_determinant = a[1] * (a[4] * a[6] - a[5]^2) - + a[2] * (a[2] * a[6] - a[3] * a[5]) + + a[3] * (a[2] * a[5] - a[4] * a[3]) + if isapprox(AT_A_determinant, 0.0) + for v in eachvariable(equations) + reconstruction_gradient[:, v, stencil, element] .= 0.0 + end + continue + end + + # det(A^T A) = a11 * (a22*a33 - a23^2) - a12 * (a12*a33 - a13*a23) + a13 * (a12*a23 - a22*a13) + # (A^T A)^-1 = 1/det(A^T A) * [(a33*a22-a23^2) -(a33*a12-a23*a13) (a23*a12-a22*a13) ; + # *** (a11*a33-a13^2) -(a23*a11-a12*a13); + # *** *** (a11*a22-a12^2) ] + # = [m11 m12 m13; m12 m22 m23; m13 m23 m33] + m11 = a[6] * a[4] - a[5]^2 + m12 = -(a[6] * a[2] - a[5] * a[3]) + m13 = (a[5] * a[2] - a[4] * a[3]) + m22 = (a[1] * a[6] - a[3]^2) + m23 = -(a[5] * a[1] - a[2] * a[3]) + m33 = (a[1] * a[4] - a[2]^2) + + # Solving least square problem + for v in eachvariable(equations) + reconstruction_gradient[1, v, stencil, element] = 1 / AT_A_determinant * + (m11 * d[v, 1] + + m12 * d[v, 2] + + m13 * d[v, 3]) + reconstruction_gradient[2, v, stencil, element] = 1 / AT_A_determinant * + (m12 * d[v, 1] + + m22 * d[v, 2] + + m23 * d[v, 3]) + reconstruction_gradient[3, v, stencil, element] = 1 / AT_A_determinant * + (m13 * d[v, 1] + + m23 * d[v, 2] + + m33 * d[v, 3]) + end + end + + # Get limited reconstruction gradient by weighting the just computed + weight_sum = zero(eltype(reconstruction_gradient)) + for v in eachvariable(equations) + reconstruction_gradient_limited[:, v, element] .= zero(eltype(reconstruction_gradient_limited)) + weight_sum = zero(eltype(reconstruction_gradient)) + for j in 1:(n_stencil_neighbors + 1) + gradient = get_node_coords(reconstruction_gradient, equations, solver, + v, j, element) + indicator = sum(gradient .^ 2) + lambda_j = (j == 1) ? lambda[1] : lambda[2] + weight = (lambda_j / (indicator + epsilon))^r + for dim in axes(reconstruction_gradient_limited, 1) + reconstruction_gradient_limited[dim, v, element] += weight * + gradient[dim] + end + weight_sum += weight + end + for dim in axes(reconstruction_gradient_limited, 1) + reconstruction_gradient_limited[dim, v, element] /= weight_sum + end + end + end + + exchange_gradient_data!(reconstruction_gradient_limited, mesh, equations, solver, + cache) + + return nothing +end + +@inline function calc_gradient_simple!(stencil, n_stencil_neighbors, element, a, d, + reconstruction_stencil, reconstruction_distance, + solution_data, equations::AbstractEquations{3}) + for i in 1:n_stencil_neighbors + # stencil=1 contains information from all neighbors + # stencil=2,...,n_stencil_neighbors+1 is computed without (stencil+1)-th neighbor's information + if i + 1 != stencil + neighbor = reconstruction_stencil[element][i] + distance = reconstruction_distance[element][i] + a[1] += distance[1]^2 # = a11 + a[2] += distance[1] * distance[2] # = a12 + a[3] += distance[1] * distance[3] + a[4] += distance[2]^2 + a[5] += distance[2] * distance[3] + a[6] += distance[3]^2 + + for v in eachvariable(equations) + d[v, 1] += distance[1] * + (solution_data[neighbor].u[v] - + solution_data[element].u[v]) + d[v, 2] += distance[2] * + (solution_data[neighbor].u[v] - + solution_data[element].u[v]) + d[v, 3] += distance[3] * + (solution_data[neighbor].u[v] - + solution_data[element].u[v]) + end + end + end + + return nothing +end + +@inline function calc_gradient_extended!(stencil, element, a, d, + reconstruction_stencil, + reconstruction_distance, + reconstruction_corner_elements, solution_data, + equations::AbstractEquations{3}, solver, cache) + if stencil == 1 + # Full stencil + for i in eachindex(reconstruction_stencil[element]) + neighbor = reconstruction_stencil[element][i] + distance = reconstruction_distance[element][i] + a[1] += distance[1]^2 # = a11 + a[2] += distance[1] * distance[2] # = a12 + a[3] += distance[1] * distance[3] + a[4] += distance[2]^2 + a[5] += distance[2] * distance[3] + a[6] += distance[3]^2 + + for v in eachvariable(equations) + d[v, 1] += distance[1] * + (solution_data[neighbor].u[v] - + solution_data[element].u[v]) + d[v, 2] += distance[2] * + (solution_data[neighbor].u[v] - + solution_data[element].u[v]) + d[v, 3] += distance[3] * + (solution_data[neighbor].u[v] - + solution_data[element].u[v]) + end + end + else + # Partial stencils + midpoint_element = get_node_coords(cache.elements.midpoint, equations, solver, + element) + for neighbor in reconstruction_corner_elements[stencil - 1, element] + midpoint_neighbor = get_node_coords(cache.elements.midpoint, equations, + solver, neighbor) + distance = midpoint_neighbor .- midpoint_element + + a[1] += distance[1]^2 # = a11 + a[2] += distance[1] * distance[2] # = a12 + a[3] += distance[1] * distance[3] + a[4] += distance[2]^2 + a[5] += distance[2] * distance[3] + a[6] += distance[3]^2 + + for v in eachvariable(equations) + d[v, 1] += distance[1] * + (solution_data[neighbor].u[v] - + solution_data[element].u[v]) + d[v, 2] += distance[2] * + (solution_data[neighbor].u[v] - + solution_data[element].u[v]) + d[v, 3] += distance[3] * + (solution_data[neighbor].u[v] - + solution_data[element].u[v]) + end + end + end + + return nothing +end + function prolong2interfaces!(cache, mesh::T8codeMesh, equations, solver::FV) (; interfaces, communication_data) = cache (; solution_data, domain_data, gradient_data) = communication_data diff --git a/test/runtests.jl b/test/runtests.jl index 1ecb13c7fab..75bd210d0eb 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -94,7 +94,8 @@ const TRIXI_NTHREADS = clamp(Sys.CPU_THREADS, 2, 3) end @time if TRIXI_TEST == "all" || TRIXI_TEST == "t8code_fv" - include("test_t8code_fv.jl") + include("test_t8code_fv_2d.jl") + include("test_t8code_fv_3d.jl") end @time if TRIXI_TEST == "all" || TRIXI_TEST == "unstructured_dgmulti" diff --git a/test/test_mpi.jl b/test/test_mpi.jl index f36b669d3c3..4f9eb65ecd8 100644 --- a/test/test_mpi.jl +++ b/test/test_mpi.jl @@ -23,10 +23,11 @@ CI_ON_WINDOWS = (get(ENV, "GITHUB_ACTIONS", false) == "true") && Sys.iswindows() # P4estMesh and T8codeMesh tests include("test_mpi_p4est_2d.jl") include("test_mpi_t8code_2d.jl") - include("test_mpi_t8code_fv.jl") + include("test_mpi_t8code_fv_2d.jl") if !CI_ON_WINDOWS # see comment on `CI_ON_WINDOWS` above include("test_mpi_p4est_3d.jl") include("test_mpi_t8code_3d.jl") + include("test_mpi_t8code_fv_3d.jl") end end # MPI diff --git a/test/test_mpi_t8code_fv.jl b/test/test_mpi_t8code_fv_2d.jl similarity index 90% rename from test/test_mpi_t8code_fv.jl rename to test/test_mpi_t8code_fv_2d.jl index 14d918794d2..54b4cd83333 100644 --- a/test/test_mpi_t8code_fv.jl +++ b/test/test_mpi_t8code_fv_2d.jl @@ -13,37 +13,37 @@ const EXAMPLES_DIR = pkgdir(Trixi, "examples", "t8code_2d_fv") # Run basic tests @testset "Examples 2D" begin # Linear scalar advection - # @trixi_testset "elixir_advection_basic.jl" begin - # @trixi_testset "first-order FV" begin - # @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), - # order=1, - # l2=[0.08551397247817498], - # linf=[0.12087467695430498]) - # # 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 "second-order FV" begin - # @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), - # l2=[0.008142380494734171], - # linf=[0.018687916234976898]) - # # 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 - # # The extended reconstruction stencil is currently not mpi parallel. - # # The current version runs through but an error occurs on some rank. - # end + @trixi_testset "elixir_advection_basic.jl" begin + @trixi_testset "first-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), + order=1, + l2=[0.08551397247817498], + linf=[0.12087467695430498]) + # 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 "second-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), + l2=[0.008142380494734171], + linf=[0.018687916234976898]) + # 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 + # The extended reconstruction stencil is currently not mpi parallel. + # The current version runs through but an error occurs on some rank. + end @trixi_testset "elixir_advection_gauss.jl" begin @trixi_testset "first-order FV" begin diff --git a/test/test_mpi_t8code_fv_3d.jl b/test/test_mpi_t8code_fv_3d.jl new file mode 100644 index 00000000000..89faeb6ac57 --- /dev/null +++ b/test/test_mpi_t8code_fv_3d.jl @@ -0,0 +1,97 @@ +module TestExamplesMPIT8codeMesh2D + +using Test +using Trixi + +include("test_trixi.jl") + +const EXAMPLES_DIR = pkgdir(Trixi, "examples", "t8code_3d_fv") + +@testset "T8codeMesh MPI FV" begin +#! format: noindent + +# Run basic tests +@testset "Examples 3D" begin + @trixi_testset "elixir_advection_basic.jl" begin + @trixi_testset "first-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), + order=1, + initial_refinement_level=4, + l2=[0.2848617953369851], + linf=[0.3721898718954475]) + # 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 "second-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), + initial_refinement_level=4, + l2=[0.10381089565603231], + linf=[0.13787405651527007]) + # 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 + # The extended reconstruction stencil is currently not mpi parallel. + # The current version runs through but an error occurs on some rank. + end + + @trixi_testset "elixir_advection_basic_hybrid.jl" begin + @trixi_testset "first-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_advection_basic_hybrid.jl"), + order=1, + l2=[0.20282363730327146], + linf=[0.28132446651281295]) + # 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 "second-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_advection_basic_hybrid.jl"), + l2=[0.02153993127089835], + linf=[0.039109618097251886]) + # 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 + + @trixi_testset "elixir_advection_nonperiodic.jl" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_nonperiodic.jl"), + l2=[0.022202106950138526], + linf=[0.0796166790338586]) + # 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 # T8codeMesh MPI + +end # module diff --git a/test/test_t8code_fv.jl b/test/test_t8code_fv_2d.jl similarity index 87% rename from test/test_t8code_fv.jl rename to test/test_t8code_fv_2d.jl index 46330d2d98e..47f607252ad 100644 --- a/test/test_t8code_fv.jl +++ b/test/test_t8code_fv_2d.jl @@ -14,6 +14,12 @@ outdir = "out" isdir(outdir) && rm(outdir, recursive = true) mkdir(outdir) +# The are some reasons why the FV tests fail on the CI runs: +# - T8code.jl is no dependency of the Trixi test environment. So `using T8code` within the `advection_basic` elixir throws an error. +# (But: When removing this line, e.g. code like `trixi_t8_mapping_c` (with unknown types) or unknown eclasses fails) +# - `load_tree_data = @t8_load_tree_data(t8_geom_load_tree_data_vertices)` is required. +# Sadly, the macro is not yet in the latest T8code.jl release (but for instance in `jmark/bumb-t8code-3.0.0` or `bennibolm/t8-trixi-fv-scheme`). + @testset "T8codeMesh2D" begin #! format: noindent @@ -44,49 +50,49 @@ mkdir(outdir) # end # end -# @trixi_testset "elixir_advection_basic.jl" begin -# @trixi_testset "first-order FV" begin -# @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), -# order=1, -# l2=[0.08551397247817498], -# linf=[0.12087467695430498]) -# # 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 "second-order FV" begin -# @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), -# l2=[0.008142380494734171], -# linf=[0.018687916234976898]) -# # 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 "second-order FV, extended reconstruction stencil" begin -# @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), -# extended_reconstruction_stencil=true, -# l2=[0.028436326251639936], -# linf=[0.08696815845435057]) -# # 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 +@trixi_testset "elixir_advection_basic.jl" begin + @trixi_testset "first-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), + order=1, + l2=[0.08551397247817498], + linf=[0.12087467695430498]) + # 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 "second-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), + l2=[0.008142380494734171], + linf=[0.018687916234976898]) + # 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 "second-order FV, extended reconstruction stencil" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), + extended_reconstruction_stencil=true, + l2=[0.028436326251639936], + linf=[0.08696815845435057]) + # 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 @trixi_testset "elixir_advection_gauss.jl" begin @trixi_testset "first-order FV" begin diff --git a/test/test_t8code_fv_3d.jl b/test/test_t8code_fv_3d.jl new file mode 100644 index 00000000000..a3150a3ab29 --- /dev/null +++ b/test/test_t8code_fv_3d.jl @@ -0,0 +1,256 @@ +module TestExamplesT8codeMesh2D + +using Test +using Trixi + +include("test_trixi.jl") + +# I added this temporary test file for constantly testing while developing. +# The tests have to be adapted at the end. +EXAMPLES_DIR = joinpath(examples_dir(), "t8code_3d_fv") + +# Start with a clean environment: remove Trixi.jl output directory if it exists +outdir = "out" +isdir(outdir) && rm(outdir, recursive = true) +mkdir(outdir) + +@testset "T8codeMesh3D" begin +#! format: noindent + +# @trixi_testset "test save_mesh_file" begin +# @test_throws Exception begin +# # Save mesh file support will be added in the future. The following +# # lines of code are here for satisfying code coverage. + +# # Create dummy mesh. +# mesh = T8codeMesh((1, 1), polydeg = 1, +# mapping = Trixi.coordinates2mapping((-1.0, -1.0), (1.0, 1.0)), +# initial_refinement_level = 1) + +# # This call throws an error. +# Trixi.save_mesh_file(mesh, "dummy") +# end +# end + +# @trixi_testset "test check_for_negative_volumes" begin +# @test_warn "Discovered negative volumes" begin +# # Unstructured mesh with six cells which have left-handed node ordering. +# mesh_file = Trixi.download("https://gist.githubusercontent.com/jmark/bfe0d45f8e369298d6cc637733819013/raw/cecf86edecc736e8b3e06e354c494b2052d41f7a/rectangle_with_negative_volumes.msh", +# joinpath(EXAMPLES_DIR, +# "rectangle_with_negative_volumes.msh")) + +# # This call should throw a warning about negative volumes detected. +# mesh = T8codeMesh(mesh_file, 2) +# end +# end + +@trixi_testset "elixir_advection_basic.jl" begin + @trixi_testset "first-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), + order=1, + initial_refinement_level=4, + l2=[0.2848617953369851], + linf=[0.3721898718954475]) + # 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 "second-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), + initial_refinement_level=4, + l2=[0.10381089565603231], + linf=[0.13787405651527007]) + # 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 "second-order FV, extended reconstruction stencil" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic.jl"), + initial_refinement_level=3, + extended_reconstruction_stencil=true, + l2=[0.3282177575292713], + linf=[0.39002345444858333]) + # 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 + +@trixi_testset "elixir_advection_gauss.jl" begin + @trixi_testset "first-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_advection_gauss.jl"), + order=1, + l2=[0.03422041780251932], + linf=[0.7655163504661142]) + # 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 "second-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, + "elixir_advection_gauss.jl"), + l2=[0.02129708543319383], + linf=[0.5679262915623222]) + # 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 + +@trixi_testset "elixir_advection_basic_hybrid.jl" begin + @trixi_testset "first-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic_hybrid.jl"), + order=1, + l2=[0.20282363730327146], + linf=[0.28132446651281295]) + # 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 "second-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_basic_hybrid.jl"), + l2=[0.02153993127089835], + linf=[0.039109618097251886]) + # 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 + +@trixi_testset "elixir_advection_nonperiodic.jl" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_advection_nonperiodic.jl"), + l2=[0.022202106950138526], + linf=[0.0796166790338586]) + # 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_euler_source_terms.jl" begin + @trixi_testset "first-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_euler_source_terms.jl"), + order=1, + l2=[ + 0.050763790354290725, + 0.0351299673616484, + 0.0351299673616484, + 0.03512996736164839, + 0.1601847269543808 + ], + linf=[ + 0.07175521415072939, + 0.04648499338897771, + 0.04648499338897816, + 0.04648499338897816, + 0.2235470564880404 + ]) + # 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 "second-order FV" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_euler_source_terms.jl"), + order=2, + l2=[ + 0.012308219704695382, + 0.010791416898840429, + 0.010791416898840464, + 0.010791416898840377, + 0.036995680347196136 + ], + linf=[ + 0.01982294164697862, + 0.01840725612418126, + 0.01840725612418148, + 0.01840725612418148, + 0.05736595182767079 + ]) + # 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 "second-order FV, extended reconstruction stencil" begin + @test_trixi_include(joinpath(EXAMPLES_DIR, "elixir_euler_source_terms.jl"), + order=2, + extended_reconstruction_stencil=true, + l2=[ + 0.05057867333486591, + 0.03596196296013507, + 0.03616867188152877, + 0.03616867188152873, + 0.14939041550302212 + ], + linf=[ + 0.07943789383956079, + 0.06389365911606859, + 0.06469291944863809, + 0.0646929194486372, + 0.23507781748792533 + ]) + # 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 + +# Clean up afterwards: delete Trixi.jl output directory +@test_nowarn rm(outdir, recursive = true) + +end # module