-
Notifications
You must be signed in to change notification settings - Fork 203
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
Implement an implicit free surface solver in the NonhydrostaticModel #3968
base: main
Are you sure you want to change the base?
Conversation
So here's how the algorithm changes with an implicit free surface (which is all I'd like to attempt for the time being):
I think the simplest way to implement this is therefore to add the implicit free surface solve as a precursor to the pressure correction. |
Okay, this script: using Oceananigans
using Oceananigans.Models.HydrostaticFreeSurfaceModels: ImplicitFreeSurface
using GLMakie
grid = RectilinearGrid(size=(128, 32), halo=(4, 4), x=(-5, 5), z=(0, 1), topology=(Bounded, Flat, Bounded))
mountain(x) = (x - 3) / 2
grid = ImmersedBoundaryGrid(grid, GridFittedBottom(mountain))
Fu(x, z, t) = sin(t)
free_surface = ImplicitFreeSurface(gravitational_acceleration=10)
model = NonhydrostaticModel(; grid, free_surface, advection=WENO(order=5), forcing=(; u=Fu))
simulation = Simulation(model, Δt=0.1, stop_time=20*2π)
conjure_time_step_wizard!(simulation, cfl=0.7)
progress(sim) = @info string(iteration(sim), ": ", time(sim))
add_callback!(simulation, progress, IterationInterval(100))
ow = JLD2OutputWriter(model, merge(model.velocities, (; η=model.free_surface.η)),
filename = "nonhydrostatic_internal_tide.jld2",
schedule = TimeInterval(0.1),
overwrite_existing = true)
simulation.output_writers[:jld2] = ow
run!(simulation)
fig = Figure()
axη = Axis(fig[1, 1], xlabel="x", ylabel="Free surface \n displacement")
axw = Axis(fig[2, 1], xlabel="x", ylabel="Surface vertical velocity")
axu = Axis(fig[3, 1], xlabel="x", ylabel="z")
ut = FieldTimeSeries("nonhydrostatic_internal_tide.jld2", "u")
wt = FieldTimeSeries("nonhydrostatic_internal_tide.jld2", "w")
ηt = FieldTimeSeries("nonhydrostatic_internal_tide.jld2", "η")
Nt = length(wt)
slider = Slider(fig[4, 1], range=1:Nt, startvalue=1)
n = slider.value
Nz = size(ut.grid, 3)
u = @lift ut[$n]
η = @lift interior(ηt[$n], :, 1, 1)
w = @lift interior(wt[$n], :, 1, Nz+1)
x = xnodes(wt)
ulim = maximum(abs, ut) * 3/4
lines!(axη, x, η)
lines!(axw, x, w)
heatmap!(axu, u)
ylims!(axη, -0.1, 0.1)
ylims!(axw, -0.01, 0.01)
record(fig, "nonhydrostatic_internal_tide.mp4", 1:Nt) do nn
@info "Drawing frame $nn of $Nt..."
n[] = nn
end produces this movie nonhydrostatic_internal_tide.mp4some weird grid artifacts in there but maybe higher resolution will help with that. |
@shriyafruitwala let me know if this code works for the problem you are interested in. The implementation is fairly clean, but there are a few things we could consider before merging, like tests, some validation in the constructor. @simone-silvestri I feel this code exposes some messiness with the peformance optimization stuff regarding kernel parameters. Please check it over to make sure what I did will work and add any tests that may be missing... |
|
||
for (wpar, ppar, κpar) in zip(w_parameters, p_parameters, κ_parameters) | ||
if !isnothing(model.free_surface) | ||
compute_w_from_continuity!(model; parameters = wpar) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I am unsure of the algorithm, but wouldn't this replace the w-velocity that should be prognostic?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
For sure. That's what is written in MITgcm docs. @jm-c can you confirm?
it seems that you are missing |
@shriyafruitwala to set η = model.free_surface.η
set!(η, value) |
It's been discussed elsewhere (and there are notes lying around -- hopefully they can be shared here) that the implementation in this PR is incorrect. In particular, the pressure satisfies a Robin boundary condition (not a Neumann condition) at the surface; implementing a Robin boundary condition requires changing the pressure solver. It's possible that the current implementation represents some kind of approximation to the true problem which may be revealed by comparing to an exact solution. In thinking about a different problem, I realized that --- I think --- a Robin boundary condition can be implemented using If true then it should be relatively straightforward to develop a direct solver for the flat bottom case. Hopefully, this would help in developing a CG solver for the case with bathymetry. I'd also like to point out there seem to be a few issues with the CG solver currently, as documented in #4007 and #3848. |
Hey everyone! I have set up a test of the nonhydrostatic free surface code of a deep water surface gravity wave initialized with a Gaussian bump. Here, G = 0, so per the MITgcm docs, the free surface equation (2.52) essentially becomes a diffusion equation, with K = gH \Delta t. We see exactly this in the test case, where the velocities are essentially zero, and the evolution of the free surface is diffusive. nonhydrostatic_deepwater_test.jld2.mp4hydrostatic_deepwater_test.jld2.mp4 |
Great work. Can you include the code that you used to run the simulation? (The code that generates the animation may be helpful too if anyone would like to reproduce your work.) The hydrostatic model can also be used with |
Sure! Here is the code: using Oceananigans
using Oceananigans.Models.HydrostaticFreeSurfaceModels: ImplicitFreeSurface
using GLMakie
using Oceananigans.Units
Nx, Nz = 50, 50
const H = 50meters
const L = 50meters
const g = 10
k = 2
ω² = g*abs(k)
ω = sqrt(ω²)
coriolis = FPlane(latitude=28)
f=coriolis.f
grid = RectilinearGrid(size = (Nx, Nz),
x = (-25meters, 25meters),
z = (-H, 0),
halo = (4,4),
topology = (Periodic, Flat, Bounded))
free_surface = ImplicitFreeSurface(gravitational_acceleration=10)
model = NonhydrostaticModel(; grid, free_surface, coriolis, advection=WENO(order=5))
#model = HydrostaticFreeSurfaceModel(; grid, free_surface, coriolis, momentum_advection=WENO(order=5))
#initialize with gaussian bump surface
η = model.free_surface.η
bump(x) = exp(-(x^2)/(2 * (5meters)^2))
x_vals = LinRange(-L/2, L/2, Nx)
η₀_vals = bump.(x_vals)
η₀ = reshape(η₀_vals, (Nx,1,1))
set!(η, η₀)
#sets up simulation run
simulation = Simulation(model, Δt=0.1, stop_time=50)
progress(sim) = @info string(iteration(sim), ": ", time(sim))
add_callback!(simulation, progress, IterationInterval(100))
output_file = "hydrostatic_deepwater_test_k2_implicit.jld2"
ow = JLD2OutputWriter(model, merge(model.velocities, (; η=model.free_surface.η)),
filename = output_file,
schedule = TimeInterval(0.1),
overwrite_existing = true)
simulation.output_writers[:jld2] = ow
run!(simulation)
#produces animation
fig = Figure()
axη = Axis(fig[1, 1], xlabel="x", ylabel="η", width=400, height=75, title="sea surface height")
axw = Axis(fig[2, 1], xlabel="x", ylabel="z", width=400, height=75, title = "w velocity")
axu = Axis(fig[3, 1], xlabel="x", ylabel="z", width=400, height=75, title = "u velocity")
ut = FieldTimeSeries(output_file, "u")
wt = FieldTimeSeries(output_file, "w")
ηt = FieldTimeSeries(output_file, "η")
Nt = length(wt)
n = Observable(1)
u = @lift ut[$n]
η = @lift interior(ηt[$n], :, 1, 1)
w = @lift wt[$n]
x = xnodes(wt)
ulim = maximum(abs, ut)
wlim = maximum(abs, wt)
lines!(axη, x, η)
hm_w = heatmap!(axw, w, colorrange=(-wlim,wlim))
Colorbar(fig[2, 2], hm_w, label = "m s⁻¹")
hm_u = heatmap!(axu, u, colorrange=(-ulim,ulim))
Colorbar(fig[3, 2], hm_u, label = "m s⁻¹")
ylims!(axη, -1, 1)
record(fig, output_file * "title.mp4", 1:Nt) do nn
@info "Drawing frame $nn of $Nt..."
n[] = nn
end Using hydrostatic_deepwater_test_k2_explicit.mp4 |
Closes #3946