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

Extend TimeSeriesCallback for TreeMesh1D/3D #1873

Merged
merged 12 commits into from
Mar 21, 2024
3 changes: 2 additions & 1 deletion NEWS.md
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,8 @@ for human readability.
## Changes in the v0.7 lifecycle

#### Added
- Implementation of `TimeSeriesCallback` for curvilinear meshes on `UnstructuredMesh2D`.
- Implementation of `TimeSeriesCallback` for curvilinear meshes on `UnstructuredMesh2D` and extension
to 1D and 3D on `TreeMesh`.


## Changes when updating to v0.7 from v0.6.x
Expand Down
3 changes: 3 additions & 0 deletions examples/tree_1d_dgsem/elixir_euler_source_terms.jl
Original file line number Diff line number Diff line change
Expand Up @@ -44,9 +44,12 @@ save_solution = SaveSolutionCallback(interval = 100,

stepsize_callback = StepsizeCallback(cfl = 0.8)

time_series = TimeSeriesCallback(semi, [0.0, 0.33, 1.0], interval = 10)

callbacks = CallbackSet(summary_callback,
analysis_callback, alive_callback,
save_solution,
time_series,
stepsize_callback)

###############################################################################
Expand Down
5 changes: 5 additions & 0 deletions examples/tree_3d_dgsem/elixir_euler_source_terms.jl
Original file line number Diff line number Diff line change
Expand Up @@ -41,9 +41,14 @@ save_solution = SaveSolutionCallback(interval = 100,

stepsize_callback = StepsizeCallback(cfl = 0.6)

time_series = TimeSeriesCallback(semi,
[(0.0, 0.0, 0.0), (0.33, 0.33, 0.33), (1.0, 1.0, 1.0)],
interval = 10)

callbacks = CallbackSet(summary_callback,
analysis_callback, alive_callback,
save_solution,
time_series,
stepsize_callback)

###############################################################################
Expand Down
6 changes: 3 additions & 3 deletions src/callbacks_step/time_series.jl
patrickersing marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
Expand Up @@ -23,8 +23,7 @@ After the last time step, the results are stored in an HDF5 file `filename` in d
The real data type `RealT` and data type for solution variables `uEltype` default to the respective
types used in the solver and the cache.

Currently this callback is only implemented for [`TreeMesh`](@ref) in 2D
and [`UnstructuredMesh2D`](@ref).
Currently this callback is only implemented for [`TreeMesh`](@ref) and [`UnstructuredMesh2D`](@ref).
"""
mutable struct TimeSeriesCallback{RealT <: Real, uEltype <: Real, SolutionVariables,
VariableNames, Cache}
Expand Down Expand Up @@ -218,5 +217,6 @@ function (time_series_callback::TimeSeriesCallback)(integrator)
end

include("time_series_dg.jl")
include("time_series_dg2d.jl")
include("time_series_dg_tree.jl")
include("time_series_dg_unstructured.jl")
end # @muladd
37 changes: 37 additions & 0 deletions src/callbacks_step/time_series_dg.jl
Original file line number Diff line number Diff line change
Expand Up @@ -34,4 +34,41 @@ function save_time_series_file(time_series_callback,
end
end
end

# Creates cache for time series callback
function create_cache_time_series(point_coordinates,
mesh::Union{TreeMesh, UnstructuredMesh2D},
dg, cache)
# Determine element ids for point coordinates
element_ids = get_elements_by_coordinates(point_coordinates, mesh, dg, cache)

# Calculate & store Lagrange interpolation polynomials
interpolating_polynomials = calc_interpolating_polynomials(point_coordinates,
element_ids, mesh,
dg, cache)

time_series_cache = (; element_ids, interpolating_polynomials)

return time_series_cache
end

function get_elements_by_coordinates(coordinates, mesh, dg, cache)
element_ids = Vector{Int}(undef, size(coordinates, 2))
get_elements_by_coordinates!(element_ids, coordinates, mesh, dg, cache)

return element_ids
end

function calc_interpolating_polynomials(coordinates, element_ids,
mesh::Union{TreeMesh, UnstructuredMesh2D},
dg, cache)
interpolating_polynomials = Array{real(dg), 3}(undef,
nnodes(dg), ndims(mesh),
length(element_ids))
calc_interpolating_polynomials!(interpolating_polynomials, coordinates, element_ids,
mesh, dg,
cache)

return interpolating_polynomials
end
end # @muladd
185 changes: 185 additions & 0 deletions src/callbacks_step/time_series_dg_tree.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,185 @@
# 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

# Find element ids containing coordinates given as a matrix [ndims, npoints]
function get_elements_by_coordinates!(element_ids, coordinates, mesh::TreeMesh, dg,
cache)
if length(element_ids) != size(coordinates, 2)
throw(DimensionMismatch("storage length for element ids does not match the number of coordinates"))
end

@unpack cell_ids = cache.elements
@unpack tree = mesh

# Reset element ids - 0 indicates "not (yet) found"
element_ids .= 0
found_elements = 0

# Iterate over all elements
for element in eachelement(dg, cache)
# Get cell id
cell_id = cell_ids[element]

# Iterate over coordinates
for index in 1:length(element_ids)
# Skip coordinates for which an element has already been found
if element_ids[index] > 0
continue
end

# Construct point
x = SVector(ntuple(i -> coordinates[i, index], ndims(mesh)))

# Skip if point is not in cell
if !is_point_in_cell(tree, x, cell_id)
continue
end

# Otherwise point is in cell and thus in element
element_ids[index] = element
found_elements += 1
end

# Exit loop if all elements have already been found
if found_elements == length(element_ids)
break
end
end

return element_ids
end

# Calculate the interpolating polynomials to extract data at the given coordinates
# The coordinates are known to be located in the respective element in `element_ids`
function calc_interpolating_polynomials!(interpolating_polynomials, coordinates,
element_ids,
mesh::TreeMesh, dg::DGSEM, cache)
@unpack tree = mesh
@unpack nodes = dg.basis

wbary = barycentric_weights(nodes)

for index in 1:length(element_ids)
# Construct point
x = SVector(ntuple(i -> coordinates[i, index], ndims(mesh)))

# Convert to unit coordinates
cell_id = cache.elements.cell_ids[element_ids[index]]
cell_coordinates_ = cell_coordinates(tree, cell_id)
cell_length = length_at_cell(tree, cell_id)
unit_coordinates = (x .- cell_coordinates_) * 2 / cell_length

# Calculate interpolating polynomial for each dimension, making use of tensor product structure
for d in 1:ndims(mesh)
interpolating_polynomials[:, d, index] .= lagrange_interpolating_polynomials(unit_coordinates[d],
nodes,
wbary)
end
end

return interpolating_polynomials
end

# Record the solution variables at each given point for the 1D case
function record_state_at_points!(point_data, u, solution_variables,
patrickersing marked this conversation as resolved.
Show resolved Hide resolved
n_solution_variables,
mesh::TreeMesh{1}, equations, dg::DG,
time_series_cache)
@unpack element_ids, interpolating_polynomials = time_series_cache
old_length = length(first(point_data))
new_length = old_length + n_solution_variables

# Loop over all points/elements that should be recorded
for index in 1:length(element_ids)
# Extract data array and element id
data = point_data[index]
element_id = element_ids[index]

# Make room for new data to be recorded
resize!(data, new_length)
data[(old_length + 1):new_length] .= zero(eltype(data))

# Loop over all nodes to compute their contribution to the interpolated values
for i in eachnode(dg)
u_node = solution_variables(get_node_vars(u, equations, dg, i,
element_id), equations)

for v in 1:length(u_node)
data[old_length + v] += (u_node[v] *
interpolating_polynomials[i, 1, index])
end
end
end
end

# Record the solution variables at each given point for the 2D case
function record_state_at_points!(point_data, u, solution_variables,
n_solution_variables,
mesh::TreeMesh{2},
patrickersing marked this conversation as resolved.
Show resolved Hide resolved
equations, dg::DG, time_series_cache)
@unpack element_ids, interpolating_polynomials = time_series_cache
old_length = length(first(point_data))
new_length = old_length + n_solution_variables

# Loop over all points/elements that should be recorded
for index in 1:length(element_ids)
# Extract data array and element id
data = point_data[index]
element_id = element_ids[index]

# Make room for new data to be recorded
resize!(data, new_length)
data[(old_length + 1):new_length] .= zero(eltype(data))

# Loop over all nodes to compute their contribution to the interpolated values
for j in eachnode(dg), i in eachnode(dg)
u_node = solution_variables(get_node_vars(u, equations, dg, i, j,
element_id), equations)

for v in 1:length(u_node)
data[old_length + v] += (u_node[v]
* interpolating_polynomials[i, 1, index]
* interpolating_polynomials[j, 2, index])
end
end
end
end

# Record the solution variables at each given point for the 3D case
function record_state_at_points!(point_data, u, solution_variables,
n_solution_variables,
mesh::TreeMesh{3}, equations, dg::DG,
time_series_cache)
@unpack element_ids, interpolating_polynomials = time_series_cache
old_length = length(first(point_data))
new_length = old_length + n_solution_variables

# Loop over all points/elements that should be recorded
for index in 1:length(element_ids)
# Extract data array and element id
data = point_data[index]
element_id = element_ids[index]

# Make room for new data to be recorded
resize!(data, new_length)
data[(old_length + 1):new_length] .= zero(eltype(data))

# Loop over all nodes to compute their contribution to the interpolated values
for k in eachnode(dg), j in eachnode(dg), i in eachnode(dg)
u_node = solution_variables(get_node_vars(u, equations, dg, i, j, k,
element_id), equations)

for v in 1:length(u_node)
data[old_length + v] += (u_node[v]
* interpolating_polynomials[i, 1, index]
* interpolating_polynomials[j, 2, index]
* interpolating_polynomials[k, 3, index])
end
end
end
end
end # @muladd
Loading
Loading