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

Add functionality for TimeSeries callback on UnstructuredMesh2D #1855

Merged
Merged
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
35 commits
Select commit Hold shift + click to select a range
afac6d8
add functionality for TimeSeries callback on UnstructuredMesh2D
andrewwinters5000 Mar 1, 2024
89ba0e8
Update src/callbacks_step/time_series_dg.jl
andrewwinters5000 Mar 1, 2024
deeea14
Merge branch 'main' into unstructured-time-series
andrewwinters5000 Mar 1, 2024
96b9928
Merge branch 'main' into unstructured-time-series
andrewwinters5000 Mar 6, 2024
970ddaa
Apply suggestions from code review
andrewwinters5000 Mar 7, 2024
ef91125
add strategy to correctly locate a gauge point within a curvilinear e…
andrewwinters5000 Mar 7, 2024
b684048
add sanity check that the Newton solution is correct
andrewwinters5000 Mar 7, 2024
bcbcffc
run formatter
andrewwinters5000 Mar 7, 2024
25071a3
implement a more general approach that also works on curved element w…
andrewwinters5000 Mar 7, 2024
f2aa34a
Merge branch 'main' into unstructured-time-series
andrewwinters5000 Mar 7, 2024
735ad18
run formatter
andrewwinters5000 Mar 7, 2024
935d01e
forgot to format the examples
andrewwinters5000 Mar 7, 2024
b8b99db
Apply suggestions from code review
andrewwinters5000 Mar 8, 2024
818edbf
working version of the element finding routine
andrewwinters5000 Mar 8, 2024
6f5e24e
run formatter
andrewwinters5000 Mar 8, 2024
00d9b83
add new elixir for the time series callback
andrewwinters5000 Mar 8, 2024
7f52bed
add additional test for the time series callback on an unstructured mesh
andrewwinters5000 Mar 8, 2024
76d0dcb
add appropriate test
andrewwinters5000 Mar 8, 2024
749f179
Merge branch 'main' into unstructured-time-series
andrewwinters5000 Mar 8, 2024
ce44fe6
update docstring
andrewwinters5000 Mar 8, 2024
bb4ce38
Merge branch 'unstructured-time-series' of github.com:andrewwinters50…
andrewwinters5000 Mar 8, 2024
04ffc1a
add comment about the barycenter computation
andrewwinters5000 Mar 8, 2024
0682471
add simplifications and comments from code review
andrewwinters5000 Mar 12, 2024
5435e1b
adjust variable name to avoid ugly formatting
andrewwinters5000 Mar 12, 2024
b002920
Apply suggestions from code review
andrewwinters5000 Mar 12, 2024
7147a10
fix variable name
andrewwinters5000 Mar 12, 2024
fe4f431
remove Experimental status from the TimeSeriesCallback
andrewwinters5000 Mar 12, 2024
fa5daa8
move new TimeSeries test into the unit testing
andrewwinters5000 Mar 12, 2024
771a3c5
add output_directory creation if not already done. Necessary if this …
andrewwinters5000 Mar 12, 2024
e652079
formatting
andrewwinters5000 Mar 12, 2024
823891b
update test mesh to have one straight-sided element to trigger invers…
andrewwinters5000 Mar 13, 2024
afc6859
update test values
andrewwinters5000 Mar 13, 2024
e73e3e7
add news item
andrewwinters5000 Mar 13, 2024
099050d
forgot to update all new test values on the new mesh
andrewwinters5000 Mar 13, 2024
d81b79e
update tests and use coverage override to avoid redundancy
andrewwinters5000 Mar 13, 2024
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
6 changes: 5 additions & 1 deletion examples/unstructured_2d_dgsem/elixir_euler_basic.jl
Original file line number Diff line number Diff line change
Expand Up @@ -58,12 +58,16 @@ save_solution = SaveSolutionCallback(interval = 10,

stepsize_callback = StepsizeCallback(cfl = 0.9)

time_series = TimeSeriesCallback(semi, [(2.0, -0.5), (0.28, 1 / 3), (1.87, 2 / 3)];
interval = 10)
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved

callbacks = CallbackSet(summary_callback,
analysis_callback,
alive_callback,
save_restart,
save_solution,
stepsize_callback)
stepsize_callback,
time_series)
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved

###############################################################################
# run the simulation
Expand Down
6 changes: 4 additions & 2 deletions src/callbacks_step/time_series_dg.jl
Original file line number Diff line number Diff line change
Expand Up @@ -5,8 +5,10 @@
@muladd begin
#! format: noindent

# Store time series file for a TreeMesh with a DG solver
function save_time_series_file(time_series_callback, mesh::TreeMesh, equations, dg::DG)
# Store time series file for a DG solver
function save_time_series_file(time_series_callback,
mesh::Union{TreeMesh, UnstructuredMesh2D},
sloede marked this conversation as resolved.
Show resolved Hide resolved
equations, dg::DG)
@unpack (interval, solution_variables, variable_names,
output_directory, filename, point_coordinates,
point_data, time, step, time_series_cache) = time_series_callback
Expand Down
186 changes: 181 additions & 5 deletions src/callbacks_step/time_series_dg2d.jl
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,9 @@
#! format: noindent

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

Expand Down Expand Up @@ -68,6 +70,52 @@ function get_elements_by_coordinates!(element_ids, coordinates, mesh::TreeMesh,
return element_ids
end

# Elements on an `UnstructuredMesh2D` are possibly curved. Assume that each
# element is convex, i.e., all interior angles are less than 180 degrees.
# We use the barycenter of each element to determine if a given coordinate (x,y)
# lies within said element. The shortest distance between the point and all the
# barycenters "wins".
function get_elements_by_coordinates!(element_ids, coordinates,
mesh::UnstructuredMesh2D,
dg, cache)
if length(element_ids) != size(coordinates, 2)
throw(DimensionMismatch("storage length for element ids does not match the number of coordinates"))
end

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

# Compute and save the barycentric coordinate on each element
bary_centers = zeros(eltype(mesh.corners), 2, mesh.n_elements)
distances = zeros(eltype(mesh.corners), mesh.n_elements)
calc_bary_centers!(bary_centers, dg, cache)

# Iterate over coordinates
for index in 1:length(element_ids)
# Compute the distance between the current point and all the bary centers.
for element in eachelement(dg, cache)
distances[element] = norm(coordinates[:, index] .- bary_centers[:, element])
end

# Locate the minimal distance as this gives the element containing the point
element_ids[index] = argmin(distances)
end
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved

return element_ids
end

# Use the available `node_coordinates` on each element to compute and save the barycenter.
@inline function calc_bary_centers!(bary_centers, dg, cache)
n = nnodes(dg)
for element in eachelement(dg, cache)
bary_centers[1, element] = sum(cache.elements.node_coordinates[1, :, :,
element]) / n^2
bary_centers[2, element] = sum(cache.elements.node_coordinates[2, :, :,
element]) / n^2
DanielDoehring marked this conversation as resolved.
Show resolved Hide resolved
end
return nothing
end
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved

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)
Expand Down Expand Up @@ -106,8 +154,136 @@ function calc_interpolating_polynomials!(interpolating_polynomials, coordinates,
return interpolating_polynomials
end

function calc_interpolating_polynomials(coordinates, element_ids, mesh::TreeMesh, dg,
cache)
function calc_interpolating_polynomials!(interpolating_polynomials, coordinates,
element_ids,
mesh::UnstructuredMesh2D, dg::DGSEM, cache)
@unpack nodes = dg.basis

wbary = barycentric_weights(nodes)

# Helper array for a straight-sided quadrilateral element
corners = zeros(eltype(mesh.corners), 4, 2)

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

# Convert to unit coordinates; procedure differs for straight-sided
# versus curvilinear elements
element = element_ids[index]
if !mesh.element_is_curved[element]
for j in 1:2, i in 1:4
# Pull the (x,y) values of the element corners from the global corners array
corners[i, j] = mesh.corners[j, mesh.element_node_ids[i, element]]
end
unit_coordinates = invert_bilinear_interpolation(mesh, x, corners)
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved

# Sanity check that the computed `unit_coordinates` indeed recover the desired point `x`
x_check = straight_side_quad_map(unit_coordinates[1], unit_coordinates[2],
corners)
if !isapprox(x[1], x_check[1], atol = 1e-13) ||
!isapprox(x[2], x_check[2], atol = 1e-13)
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved
error("failed to compute computational coordinates for the time series point $(x), closet candidate was $(x_check)")
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved
end
else # mesh.element_is_curved[element]
unit_coordinates = invert_transfinite_interpolation(mesh, x,
view(mesh.surface_curves,
:, element))

# Sanity check that the computed `unit_coordinates` indeed recover the desired point `x`
x_check = transfinite_quad_map(unit_coordinates[1], unit_coordinates[2],
view(mesh.surface_curves, :, element))
if !isapprox(x[1], x_check[1], atol = 1e-13) ||
!isapprox(x[2], x_check[2], atol = 1e-13)
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved
error("failed to compute computational coordinates for the time series point $(x), closet candidate was $(x_check)")
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved
end
end

# 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

# Use a Newton iteration to determine the computational coordinates
# (xi, eta) of given (x,y) `point` that is given in physical coordinates
# by inverting the transformation. For straight-sided elements this
# amounts to inverting a bi-linear interpolation. For curved
# elements we invert the transfinite interpolation with linear blending.
# The residual function for the Newton iteration is
# r(xi,eta) = X(xi,eta) - point
# and the Jacobian entries are computed accordingly from either
# `straight_side_quad_map_metrics` or `transfinite_quad_map_metrics`.
# We exploit the 2x2 nature of the problem and directly compute the matrix
# inverse to make things faster. The implementations below are inspired by
# an answer on Stack Overflow (https://stackoverflow.com/a/18332009) where
# the author explicitly states that their code is released to the public domain.
sloede marked this conversation as resolved.
Show resolved Hide resolved
@inline function invert_bilinear_interpolation(mesh::UnstructuredMesh2D, point,
element_corners)
# Initial guess for the point (center of the reference element)
xi = zero(eltype(point))
eta = zero(eltype(point))
for k in 1:5 # Newton's method should converge quickly
# Compute current x and y coordinate and the Jacobian matrix
# J = (X_xi, X_eta; Y_xi, Y_eta)
x, y = straight_side_quad_map(xi, eta, element_corners)
J11, J12, J21, J22 = straight_side_quad_map_metrics(xi, eta, element_corners)

# Compute residuals for the Newton teration for the current (x, y) coordinate
r1 = x - point[1]
r2 = y - point[2]

# Newton update that directly applies the inverse of the 2x2 Jacobian matrix
inv_detJ = inv(J11 * J22 - J12 * J21)

xi = xi - inv_detJ * (J22 * r1 - J12 * r2)
eta = eta - inv_detJ * (-J21 * r1 + J11 * r2)
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved

# Ensure updated point is in the reference element
xi = min(max(xi, -1), 1)
eta = min(max(eta, -1), 1)
end

return SVector(xi, eta)
end

@inline function invert_transfinite_interpolation(mesh::UnstructuredMesh2D, point,
surface_curves::AbstractVector{<:CurvedSurface})
# Initial guess for the point (center of the reference element)
xi = zero(eltype(point))
eta = zero(eltype(point))
for k in 1:5 # Newton's method should converge quickly
# Compute current x and y coordinate and the Jacobian matrix
# J = (X_xi, X_eta; Y_xi, Y_eta)
x, y = transfinite_quad_map(xi, eta, surface_curves)
J11, J12, J21, J22 = transfinite_quad_map_metrics(xi, eta, surface_curves)

# Compute residuals for the Newton teration for the current (x,y) coordinate
r1 = x - point[1]
r2 = y - point[2]

# Newton update that directly applies the inverse of the 2x2 Jacobian matrix
inv_detJ = inv(J11 * J22 - J12 * J21)

xi = xi - inv_detJ * (J22 * r1 - J12 * r2)
eta = eta - inv_detJ * (-J21 * r1 + J11 * r2)
andrewwinters5000 marked this conversation as resolved.
Show resolved Hide resolved

# Ensure updated point is in the reference element
xi = min(max(xi, -1), 1)
eta = min(max(eta, -1), 1)
end

return SVector(xi, eta)
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))
Expand All @@ -121,8 +297,8 @@ end
# Record the solution variables at each given point
function record_state_at_points!(point_data, u, solution_variables,
n_solution_variables,
mesh::TreeMesh{2}, equations, dg::DG,
time_series_cache)
mesh::Union{TreeMesh{2}, UnstructuredMesh2D},
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
Expand Down
Loading