Replies: 2 comments
-
2D models cannot have surface stress because there are no z-derivatives in a 2D model. A surface stress is a vertical flux across the top grid cell. To describe concisely what happens consider a simplified version of the x-momentum equation: where where A two-dimensional model has vertical direction and therefore no vertical flux of anything. There is no way to add a surface stress. Often times we think of two-dimensional models as idealizing the evolution of the vertical average or something. So we can derive a two-dimensional momentum equation for example by taking the vertical average of the equation I wrote above, but over an entire dimension, say an ocean of depth where So one can think of "surface forcing" as being a body force acting on the vertically-averaged equations, which are integrated in But even in this case I strongly recommend to refer to the physical effect as a body force in 2D, rather than a "surface stress". It's not a surface stress, because a 2D model has no surface. We've developed the code intentionally to help users (but also people who read user's code) understand clearly the physics that are being applied. In particular, a 2D model has no grid spacing in the vertical direction (it's not easy to think of this as a model with just 1 grid point in the vertical direction, which is usually the way that people simplify 3D to 2D). So we can't apply a surface stress to the "top" grid cell, there is no In summary, surface or bottom boundary conditions cannot be applied to models that are 2D in x, y. However, it is likely that the effect one is trying to achieve would use a body |
Beta Was this translation helpful? Give feedback.
-
@glwagner Thank you so much for that detailed explanation. It makes perfect sense now why I can't apply a surface forcing. Instead of applying a body using Pkg
Pkg.activate(".")
using Dates
using Random
using Printf
using Oceananigans
using Oceananigans.Units: minute, minutes, hour, hours, day, days, meter, meters, kilometer, kilometers
# --------------------------------------------------
# computing architecture
# --------------------------------------------------
architecture = CPU()
# --------------------------------------------------
# grid
# --------------------------------------------------
const Nx = 100 # number of x points
const Ny = 100 # number of y points
const Nz = 1 # number of z points
const Lx = 100kilometers # (m) domain x extent
const Ly = 100kilometers # (m) domain y extent
const Lz = 10meters # (m) domain depth
grid = RectilinearGrid(
architecture,
topology = (Periodic, Periodic, Bounded),
size = (Nx, Nx, Nz),
halo = (3,3,2),
x = (0, Lx),
y = (0, Ly),
z = (-Lz, 0)
)
# --------------------------------------------------
# coriolis
# --------------------------------------------------
coriolis = FPlane(f=1e-4)
# --------------------------------------------------
# closure
# --------------------------------------------------
closure = AnisotropicMinimumDissipation()
# --------------------------------------------------
# seawater buoyancy
# --------------------------------------------------
buoyancy = SeawaterBuoyancy(
equation_of_state = LinearEquationOfState(
thermal_expansion = 2e-4,
haline_contraction = 8e-4)
)
# --------------------------------------------------
# surface stress
# --------------------------------------------------
u₁₀ = 10 # m s⁻¹, average wind velocity 10 meters above the ocean
ρₒ = 1026.0 # kg m⁻³, average density at the surface of the world ocean
cᴰ = 2.5e-3 # dimensionless drag coefficient
ρₐ = 1.225 # kg m⁻³, average density of air at sea-level
@inline wind_stress_x(x, y, t, p) = (- ρₐ / ρₒ * cᴰ * u₁₀ * abs(u₁₀)) * rand()
@inline wind_stress_y(x, y, t, p) = (- ρₐ / ρₒ * cᴰ * u₁₀ * abs(u₁₀)) * rand()
# --------------------------------------------------
# boundary conditions
# --------------------------------------------------
u_top_bcs = FluxBoundaryCondition(wind_stress_x, parameters=(k=4π, ω=3.0, τ=1e-4))
v_top_bcs = FluxBoundaryCondition(wind_stress_y, parameters=(k=4π, ω=3.0, τ=1e-4))
u_bcs = FieldBoundaryConditions(top = u_top_bcs)
v_bcs = FieldBoundaryConditions(top = v_top_bcs)
T_bcs = FieldBoundaryConditions()
S_bcs = FieldBoundaryConditions()
c_bcs = FieldBoundaryConditions()
# --------------------------------------------------
# instantiate model
# --------------------------------------------------
model = NonhydrostaticModel(; grid, buoyancy,
timestepper = :QuasiAdamsBashforth2,
advection = WENO(),
tracers = (:T, :S, :c),
coriolis = coriolis,
closure = closure,
boundary_conditions = (u=u_bcs, v=v_bcs, T=T_bcs, S=S_bcs, c=c_bcs))
# --------------------------------------------------
# initial conditions
# --------------------------------------------------
const u_initial = 0.0
const v_initial = 0.0
const T_initial = 20.0
const S_initial = 35.0
# initial concentration value
# center indices (x, y)
center_x_index = Int(grid.Nx / 2) # center index in the x-direction
center_y_index = Int(grid.Ny / 2) # center index in the y-direction
surface_z_index = grid.Nz # index of surface z grid cell
# initialize tracer values to zero
c_initial = model.tracers.c
# grid cell volume
# note: this assumes grid cells all same size
dx = Lx / Nx # (m) grid cell x length
dy = Ly / Ny # (m) grid cell y length
dz = Lz / Nz # (m) grid cell z length
dv = dx * dy * dz # (m3) grid cell volume
# set initial concentration to 1 mol / m3 at center grid cell
c_initial[center_x_index, center_y_index, surface_z_index] = 1
set!(model, u=u_initial, v=v_initial, T=T_initial, S=S_initial, c=c_initial)
# --------------------------------------------------
# simulation and outputs
# --------------------------------------------------
simulation = Simulation(model, Δt=10minutes, stop_time=365days)
wizard = TimeStepWizard(cfl=0.2)
simulation.callbacks[:wizard] = Callback(wizard, IterationInterval(10))
progress_message(sim) = @printf("Iteration: %04d, time: %s, Δt: %s, max(|u|) = %.1e ms⁻¹, wall time: %s\n",
iteration(sim), prettytime(sim), prettytime(sim.Δt),
maximum(abs, sim.model.velocities.u), prettytime(sim.run_wall_time))
add_callback!(simulation, progress_message, IterationInterval(20))
# --------------------------------------------------
# output writer
# --------------------------------------------------
filename = "output-tracer-release_$(string(today()))"
simulation.output_writers[:surface_slice_writer] = NetCDFOutputWriter(
model,
(; model.velocities.u, model.velocities.v, model.tracers.S, model.tracers.T, model.tracers.c),
filename = filename * ".nc",
schedule = AveragedTimeInterval(1hour, window=1hour),
indices=(:, :, grid.Nz),
overwrite_existing = true
)
# --------------------------------------------------
# run simulation
# --------------------------------------------------
run!(simulation) |
Beta Was this translation helpful? Give feedback.
-
I am just learning Oceananigans.jl and Julia. To get acquainted with the framework i'm tryig to advect around a passive tracer on a 2D field. Basically integrate forward the advection diffusion equation. I tried modifying the wind stress example in the docs to work with a (Periodic, Periodic, Flat) topology (i.e. a torus). But I am learning you can't apply a
FluxBoundaryCondition
in theFlat
direction. Any guidance would be appreciated. Thanks! Here is my code so far:For what it's worth, here the complete error code:
Beta Was this translation helpful? Give feedback.
All reactions