From b217d8889ff9d7ad501b7149b692a73be23072d0 Mon Sep 17 00:00:00 2001 From: Charles Kawczynski Date: Wed, 26 Jun 2024 16:26:14 -0400 Subject: [PATCH] Add convenience constructors for grids --- NEWS.md | 8 + docs/src/api.md | 34 ++ src/ClimaCore.jl | 1 + src/CommonGrids/CommonGrids.jl | 646 ++++++++++++++++++++++++++++++++ src/CommonGrids/Helpers.jl | 115 ++++++ test/CommonGrids/CommonGrids.jl | 121 ++++++ test/runtests.jl | 1 + 7 files changed, 926 insertions(+) create mode 100644 src/CommonGrids/CommonGrids.jl create mode 100644 src/CommonGrids/Helpers.jl create mode 100644 test/CommonGrids/CommonGrids.jl diff --git a/NEWS.md b/NEWS.md index d43cacf0b5..5a88b4a7d7 100644 --- a/NEWS.md +++ b/NEWS.md @@ -4,6 +4,14 @@ ClimaCore.jl Release Notes main ------- + - We've added new convenience constructors for grids PR [1848](https://github.com/CliMA/ClimaCore.jl/pull/1848). Here are links to the new constructors: + - [ExtrudedCubedSphereGrid]() + - [CubedSphereGrid]() + - [ColumnGrid]() + - [Box3DGrid]() + - [SliceXZGrid]() + - [RectangleXYGrid]() + - A `strict = true` keyword was added to `rcompare`, which checks that the types match. If `strict = false`, then `rcompare` will return `true` for `FieldVector`s and `NamedTuple`s with the same properties but permuted order. For example: - `rcompare((;a=1,b=2), (;b=2,a=1); strict = true)` will return `false` and - `rcompare((;a=1,b=2), (;b=2,a=1); strict = false)` will return `true` diff --git a/docs/src/api.md b/docs/src/api.md index 7d7aad6434..cfb9fd029b 100644 --- a/docs/src/api.md +++ b/docs/src/api.md @@ -40,6 +40,12 @@ DataLayouts.VIJHF ## Geometry +### Global Geometry +```@docs +Geometry.AbstractGlobalGeometry +Geometry.CartesianGlobalGeometry +``` + ### Coordinates ```@docs Geometry.AbstractPoint @@ -171,6 +177,34 @@ Topologies.local_neighboring_elements Topologies.ghost_neighboring_elements ``` +## Grids + +```@docs +Grids.ColumnGrid +Grids.FiniteDifferenceGrid +Grids.ExtrudedFiniteDifferenceGrid +Grids.SpectralElementGrid1D +Grids.SpectralElementGrid2D +``` + +## Hypsography + +```@docs +Grids.Flat +``` + +## CommonGrids + +```@docs +CommonGrids +CommonGrids.ExtrudedCubedSphereGrid +CommonGrids.CubedSphereGrid +CommonGrids.ColumnGrid +CommonGrids.Box3DGrid +CommonGrids.SliceXZGrid +CommonGrids.RectangleXYGrid +``` + ## Spaces A `Space` represents a discretized function space over some domain. Currently two main discretizations are supported: Spectral Element Discretization diff --git a/src/ClimaCore.jl b/src/ClimaCore.jl index b9b8b7e0b6..cb908bf7f1 100644 --- a/src/ClimaCore.jl +++ b/src/ClimaCore.jl @@ -23,6 +23,7 @@ include("Hypsography/Hypsography.jl") include("Limiters/Limiters.jl") include("InputOutput/InputOutput.jl") include("Remapping/Remapping.jl") +include("CommonGrids/CommonGrids.jl") include("deprecated.jl") diff --git a/src/CommonGrids/CommonGrids.jl b/src/CommonGrids/CommonGrids.jl new file mode 100644 index 0000000000..7322fb9779 --- /dev/null +++ b/src/CommonGrids/CommonGrids.jl @@ -0,0 +1,646 @@ +""" + CommonGrids + +CommonGrids contains convenience constructors for common +grids. Constructors in this module are sometimes dynamically +created. You may want to use a different constructor if you're +making the object in a performance-critical section, and if +you know the type parameters at compile time. + +If no convenience constructor exists, then you may need to +create a custom grid using our low-level compose-able API. + + +# Transitioning to using CommonGrids + +You may have constructed a grid in the following way: + +```julia +using ClimaComms +using ClimaCore: DataLayouts, Geometry, Topologies, Quadratures, Domains, Meshes, Grids +FT = Float64 +z_elem = 63 +z_min = FT(0) +z_max = FT(1) +radius = FT(6.371229e6) +h_elem = 15 +n_quad_points = 4 +device = ClimaComms.device() +context = ClimaComms.context(device) +hypsography = Grids.Flat() +global_geometry = Geometry.ShallowSphericalGlobalGeometry(radius) +quad = Quadratures.GLL{n_quad_points}() +h_mesh = Meshes.EquiangularCubedSphere(Domains.SphereDomain{FT}(radius), h_elem) +h_topology = Topologies.Topology2D(context, h_mesh) +z_boundary_names = (:bottom, :top) +z_domain = Domains.IntervalDomain( + Geometry.ZPoint{FT}(z_min), + Geometry.ZPoint{FT}(z_max); + boundary_names = z_boundary_names, +) +z_mesh = Meshes.IntervalMesh(z_domain; nelems = z_elem) +h_grid = Grids.SpectralElementGrid2D(h_topology, quad) +z_topology = Topologies.IntervalTopology(context, z_mesh) +z_grid = Grids.FiniteDifferenceGrid(z_topology) +grid = Grids.ExtrudedFiniteDifferenceGrid( + h_grid, + z_grid, + hypsography, + global_geometry, +) +``` + +You may re-write this as: + +```julia +using ClimaCore.CommonGrids: ExtrudedCubedSphereGrid +grid = ExtrudedCubedSphereGrid(; + z_elem = 63, + z_min = 0, + z_max = 1, + radius = 6.371229e6, + h_elem = 15, + n_quad_points = 4, +) +``` +""" +module CommonGrids + +export ExtrudedCubedSphereGrid, + CubedSphereGrid, ColumnGrid, Box3DGrid, SliceXZGrid, RectangleXYGrid + +import ClimaComms +import ..DataLayouts, + ..Meshes, ..Topologies, ..Geometry, ..Domains, ..Quadratures, ..Grids + +include("Helpers.jl") +import .Helpers.DefaultSliceXMesh +import .Helpers.DefaultZMesh +import .Helpers.DefaultRectangleXYMesh + +##### +##### Grids +##### + +""" + ExtrudedCubedSphereGrid( + ::Type{<:AbstractFloat}; # defaults to Float64 + z_elem::Integer, + z_min::Real, + z_max::Real, + radius::Real, + h_elem::Integer, + n_quad_points::Integer, + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + stretch::Meshes.StretchingRule = Meshes.Uniform(), + hypsography_fun = (h_grid, z_grid) -> Grids.Flat(), + global_geometry::Geometry.AbstractGlobalGeometry = Geometry.ShallowSphericalGlobalGeometry(radius), + quad::Quadratures.QuadratureStyle = Quadratures.GLL{n_quad_points}(), + h_mesh = Meshes.EquiangularCubedSphere(Domains.SphereDomain{FT}(radius), h_elem), + h_topology::Topologies.AbstractDistributedTopology = Topologies.Topology2D(context, h_mesh), + horizontal_layout_type = DataLayouts.IJFH, + z_mesh::Meshes.IntervalMesh = DefaultZMesh(FT; z_min, z_max, z_elem, stretch), + enable_bubble::Bool = false + ) + +A convenience constructor, which builds an +[`Grids.ExtrudedFiniteDifferenceGrid`](@ref), given: + + - `FT` the floating-point type (defaults to `Float64`) [`Float32`, `Float64`] + - `z_elem` the number of z-points + - `z_min` the domain minimum along the z-direction. + - `z_max` the domain maximum along the z-direction. + - `radius` the radius of the cubed sphere + - `h_elem` the number of horizontal elements per side of every panel (6 panels in total) + - `n_quad_points` the number of quadrature points per horizontal element + - `device` the `ClimaComms.device` + - `context` the `ClimaComms.context` + - `stretch` the mesh `Meshes.StretchingRule` (defaults to [`Meshes.Uniform`](@ref)) + - `hypsography_fun` a function or callable object (`hypsography_fun(h_grid, z_grid) -> hypsography`) for constructing the hypsography model. + - `global_geometry` the global geometry (defaults to [`Geometry.CartesianGlobalGeometry`](@ref)) + - `quad` the quadrature style (defaults to `Quadratures.GLL{n_quad_points}`) + - `h_mesh` the horizontal mesh (defaults to `Meshes.EquiangularCubedSphere`) + - `h_topology` the horizontal topology (defaults to `Topologies.Topology2D`) + - `horizontal_layout_type` the horizontal DataLayout type (defaults to `DataLayouts.IJFH`). This parameter describes how data is arranged in memory. See [`Grids.SpectralElementGrid2D`](@ref) for its use. + - `z_mesh` the vertical mesh, defaults to an `Meshes.IntervalMesh` along `z` with given `stretch` + - `enable_bubble` enables the "bubble correction" for more accurate element areas when computing the spectral element space. See [`Grids.SpectralElementGrid2D`](@ref) for more information. + +# Example usage + +```julia +using ClimaCore.CommonGrids +grid = ExtrudedCubedSphereGrid(; + z_elem = 10, + z_min = 0, + z_max = 1, + radius = 10, + h_elem = 10, + n_quad_points = 4, +) +``` +""" +ExtrudedCubedSphereGrid(; kwargs...) = + ExtrudedCubedSphereGrid(Float64; kwargs...) + +function ExtrudedCubedSphereGrid( + ::Type{FT}; + z_elem::Integer, + z_min::Real, + z_max::Real, + radius::Real, + h_elem::Integer, + n_quad_points::Integer, + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + stretch::Meshes.StretchingRule = Meshes.Uniform(), + hypsography_fun = (h_grid, z_grid) -> Grids.Flat(), + global_geometry::Geometry.AbstractGlobalGeometry = Geometry.ShallowSphericalGlobalGeometry( + radius, + ), + quad::Quadratures.QuadratureStyle = Quadratures.GLL{n_quad_points}(), + h_mesh = Meshes.EquiangularCubedSphere( + Domains.SphereDomain{FT}(radius), + h_elem, + ), + h_topology::Topologies.AbstractDistributedTopology = Topologies.Topology2D( + context, + h_mesh, + ), + horizontal_layout_type = DataLayouts.IJFH, + z_mesh::Meshes.IntervalMesh = DefaultZMesh( + FT; + z_min, + z_max, + z_elem, + stretch, + ), + enable_bubble::Bool = false, +) where {FT} + @assert horizontal_layout_type <: DataLayouts.AbstractData + @assert ClimaComms.device(context) == device "The given device and context device do not match." + + z_boundary_names = (:bottom, :top) + h_grid = Grids.SpectralElementGrid2D( + h_topology, + quad; + horizontal_layout_type, + enable_bubble, + ) + z_topology = Topologies.IntervalTopology(context, z_mesh) + z_grid = Grids.FiniteDifferenceGrid(z_topology) + return Grids.ExtrudedFiniteDifferenceGrid( + h_grid, + z_grid, + hypsography_fun(h_grid, z_grid), + global_geometry, + ) +end + +""" + CubedSphereGrid( + ::Type{<:AbstractFloat}; # defaults to Float64 + radius::Real, + h_elem::Integer, + n_quad_points::Integer, + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + quad::Quadratures.QuadratureStyle = Quadratures.GLL{n_quad_points}(), + h_mesh = Meshes.EquiangularCubedSphere(Domains.SphereDomain{FT}(radius), h_elem), + h_topology::Topologies.AbstractDistributedTopology = Topologies.Topology2D(context, h_mesh), + horizontal_layout_type = DataLayouts.IJFH, + ) + +A convenience constructor, which builds a +[`Grids.SpectralElementGrid2D`](@ref) given: + + - `FT` the floating-point type (defaults to `Float64`) [`Float32`, `Float64`] + - `radius` the radius of the cubed sphere + - `h_elem` the number of horizontal elements per side of every panel (6 panels in total) + - `n_quad_points` the number of quadrature points per horizontal element + - `device` the `ClimaComms.device` + - `context` the `ClimaComms.context` + - `quad` the quadrature style (defaults to `Quadratures.GLL{n_quad_points}`) + - `h_mesh` the horizontal mesh (defaults to `Meshes.EquiangularCubedSphere`) + - `h_topology` the horizontal topology (defaults to `Topologies.Topology2D`) + - `horizontal_layout_type` the horizontal DataLayout type (defaults to `DataLayouts.IJFH`). This parameter describes how data is arranged in memory. See [`Grids.SpectralElementGrid2D`](@ref) for its use. + +# Example usage + +```julia +using ClimaCore.CommonGrids +grid = CubedSphereGrid(; radius = 10, n_quad_points = 4, h_elem = 10) +``` +""" +CubedSphereGrid(; kwargs...) = CubedSphereGrid(Float64; kwargs...) +function CubedSphereGrid( + ::Type{FT}; + radius::Real, + h_elem::Integer, + n_quad_points::Integer, + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + quad::Quadratures.QuadratureStyle = Quadratures.GLL{n_quad_points}(), + h_mesh = Meshes.EquiangularCubedSphere( + Domains.SphereDomain{FT}(radius), + h_elem, + ), + h_topology::Topologies.AbstractDistributedTopology = Topologies.Topology2D( + context, + h_mesh, + ), + horizontal_layout_type = DataLayouts.IJFH, +) where {FT} + @assert horizontal_layout_type <: DataLayouts.AbstractData + @assert ClimaComms.device(context) == device "The given device and context device do not match." + return Grids.SpectralElementGrid2D(h_topology, quad; horizontal_layout_type) +end + +""" + ColumnGrid( + ::Type{<:AbstractFloat}; # defaults to Float64 + z_elem::Integer, + z_min::Real, + z_max::Real, + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + stretch::Meshes.StretchingRule = Meshes.Uniform(), + z_mesh::Meshes.IntervalMesh = DefaultZMesh(FT; z_min, z_max, z_elem, stretch), + ) + +A convenience constructor, which builds a +[`Grids.FiniteDifferenceGrid`](@ref). + +# Example usage + +```julia +using ClimaCore.CommonGrids +grid = ColumnGrid(; z_elem = 10, z_min = 0, z_max = 10) +``` +""" +ColumnGrid(; kwargs...) = ColumnGrid(Float64; kwargs...) +function ColumnGrid( + ::Type{FT}; + z_elem::Integer, + z_min::Real, + z_max::Real, + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + stretch::Meshes.StretchingRule = Meshes.Uniform(), + z_mesh::Meshes.IntervalMesh = DefaultZMesh( + FT; + z_min, + z_max, + z_elem, + stretch, + ), +) where {FT} + @assert ClimaComms.device(context) == device "The given device and context device do not match." + z_topology = Topologies.IntervalTopology(context, z_mesh) + return Grids.FiniteDifferenceGrid(z_topology) +end + +""" + Box3DGrid( + ::Type{<:AbstractFloat}; # defaults to Float64 + z_elem::Integer, + x_min::Real, + x_max::Real, + y_min::Real, + y_max::Real, + z_min::Real, + z_max::Real, + periodic_x::Bool, + periodic_y::Bool, + n_quad_points::Integer, + x_elem::Integer, + y_elem::Integer, + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + stretch::Meshes.StretchingRule = Meshes.Uniform(), + hypsography_fun = (h_grid, z_grid) -> Grids.Flat(), + global_geometry::Geometry.AbstractGlobalGeometry = Geometry.CartesianGlobalGeometry(), + quad::Quadratures.QuadratureStyle = Quadratures.GLL{n_quad_points}(), + horizontal_layout_type = DataLayouts.IJFH, + [h_topology::Topologies.AbstractDistributedTopology], # optional + [z_mesh::Meshes.IntervalMesh], # optional + enable_bubble::Bool = false, + ) + +A convenience constructor, which builds a +[`Grids.ExtrudedFiniteDifferenceGrid`](@ref) with a +[`Grids.FiniteDifferenceGrid`](@ref) vertical grid and a +[`Grids.SpectralElementGrid2D`](@ref) horizontal grid, given: + + - `z_elem` the number of z-points + - `x_min` the domain minimum along the x-direction. + - `x_max` the domain maximum along the x-direction. + - `y_min` the domain minimum along the y-direction. + - `y_max` the domain maximum along the y-direction. + - `z_min` the domain minimum along the z-direction. + - `z_max` the domain maximum along the z-direction. + - `periodic_x` Bool indicating to use periodic domain along x-direction + - `periodic_y` Bool indicating to use periodic domain along y-direction + - `n_quad_points` the number of quadrature points per horizontal element + - `x_elem` the number of x-points + - `y_elem` the number of y-points + - `device` the `ClimaComms.device` + - `context` the `ClimaComms.context` + - `stretch` the mesh `Meshes.StretchingRule` (defaults to [`Meshes.Uniform`](@ref)) + - `hypsography_fun` a function or callable object (`hypsography_fun(h_grid, z_grid) -> hypsography`) for constructing the hypsography model. + - `global_geometry` the global geometry (defaults to [`Geometry.CartesianGlobalGeometry`](@ref)) + - `quad` the quadrature style (defaults to `Quadratures.GLL{n_quad_points}`) + - `h_topology` the horizontal topology (defaults to `Topologies.Topology2D`) + - `z_mesh` the vertical mesh, defaults to an `Meshes.IntervalMesh` along `z` with given `stretch` + - `enable_bubble` enables the "bubble correction" for more accurate element areas when computing the spectral element space. See [`Grids.SpectralElementGrid2D`](@ref) for more information. + - `horizontal_layout_type` the horizontal DataLayout type (defaults to `DataLayouts.IJFH`). This parameter describes how data is arranged in memory. See [`Grids.SpectralElementGrid2D`](@ref) for its use. + +# Example usage + +```julia +using ClimaCore.CommonGrids +grid = Box3DGrid(; + z_elem = 10, + x_min = 0, + x_max = 1, + y_min = 0, + y_max = 1, + z_min = 0, + z_max = 10, + periodic_x = false, + periodic_y = false, + n_quad_points = 4, + x_elem = 3, + y_elem = 4, +) +``` +""" +Box3DGrid(; kwargs...) = Box3DGrid(Float64; kwargs...) +function Box3DGrid( + ::Type{FT}; + z_elem::Integer, + x_min::Real, + x_max::Real, + y_min::Real, + y_max::Real, + z_min::Real, + z_max::Real, + periodic_x::Bool, + periodic_y::Bool, + n_quad_points::Integer, + x_elem::Integer, + y_elem::Integer, + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + stretch::Meshes.StretchingRule = Meshes.Uniform(), + hypsography_fun = (h_grid, z_grid) -> Grids.Flat(), + global_geometry::Geometry.AbstractGlobalGeometry = Geometry.CartesianGlobalGeometry(), + quad::Quadratures.QuadratureStyle = Quadratures.GLL{n_quad_points}(), + h_topology::Topologies.AbstractDistributedTopology = Topologies.Topology2D( + context, + DefaultRectangleXYMesh( + FT; + x_min, + x_max, + y_min, + y_max, + x_elem, + y_elem, + periodic_x, + periodic_y, + ), + ), + z_mesh::Meshes.IntervalMesh = DefaultZMesh( + FT; + z_min, + z_max, + z_elem, + stretch, + ), + enable_bubble::Bool = false, + horizontal_layout_type = DataLayouts.IJFH, +) where {FT} + @assert horizontal_layout_type <: DataLayouts.AbstractData + @assert ClimaComms.device(context) == device "The given device and context device do not match." + h_grid = Grids.SpectralElementGrid2D( + h_topology, + quad; + horizontal_layout_type, + enable_bubble, + ) + z_topology = Topologies.IntervalTopology(context, z_mesh) + z_grid = Grids.FiniteDifferenceGrid(z_topology) + return Grids.ExtrudedFiniteDifferenceGrid( + h_grid, + z_grid, + hypsography_fun(h_grid, z_grid), + global_geometry, + ) +end + +""" + SliceXZGrid( + ::Type{<:AbstractFloat}; # defaults to Float64 + z_elem::Integer, + x_min::Real, + x_max::Real, + z_min::Real, + z_max::Real, + periodic_x::Bool, + n_quad_points::Integer, + x_elem::Integer, + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + stretch::Meshes.StretchingRule = Meshes.Uniform(), + hypsography_fun = (h_grid, z_grid) -> Grids.Flat(), + global_geometry::Geometry.AbstractGlobalGeometry = Geometry.CartesianGlobalGeometry(), + quad::Quadratures.QuadratureStyle = Quadratures.GLL{n_quad_points}(), + ) + +A convenience constructor, which builds a +[`Grids.ExtrudedFiniteDifferenceGrid`](@ref) with a +[`Grids.FiniteDifferenceGrid`](@ref) vertical grid and a +[`Grids.SpectralElementGrid1D`](@ref) horizontal grid, given: + + + - `FT` the floating-point type (defaults to `Float64`) [`Float32`, `Float64`] + - `z_elem` the number of z-points + - `x_min` the domain minimum along the x-direction. + - `x_max` the domain maximum along the x-direction. + - `z_min` the domain minimum along the z-direction. + - `z_max` the domain maximum along the z-direction. + - `periodic_x` Bool indicating to use periodic domain along x-direction + - `n_quad_points` the number of quadrature points per horizontal element + - `x_elem` the number of x-points + - `device` the `ClimaComms.device` + - `context` the `ClimaComms.context` + - `stretch` the mesh `Meshes.StretchingRule` (defaults to [`Meshes.Uniform`](@ref)) + - `hypsography_fun` a function or callable object (`hypsography_fun(h_grid, z_grid) -> hypsography`) for constructing the hypsography model. + - `global_geometry` the global geometry (defaults to [`Geometry.CartesianGlobalGeometry`](@ref)) + - `quad` the quadrature style (defaults to `Quadratures.GLL{n_quad_points}`) + +# Example usage + +```julia +using ClimaCore.CommonGrids +grid = SliceXZGrid(; + z_elem = 10, + x_min = 0, + x_max = 1, + z_min = 0, + z_max = 1, + periodic_x = false, + n_quad_points = 4, + x_elem = 4, +) +``` +""" +SliceXZGrid(; kwargs...) = SliceXZGrid(Float64; kwargs...) +function SliceXZGrid( + ::Type{FT}; + z_elem::Integer, + x_min::Real, + x_max::Real, + z_min::Real, + z_max::Real, + periodic_x::Bool, + n_quad_points::Integer, + x_elem::Integer, + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + stretch::Meshes.StretchingRule = Meshes.Uniform(), + hypsography_fun = (h_grid, z_grid) -> Grids.Flat(), + global_geometry::Geometry.AbstractGlobalGeometry = Geometry.CartesianGlobalGeometry(), + quad::Quadratures.QuadratureStyle = Quadratures.GLL{n_quad_points}(), + horizontal_layout_type = DataLayouts.IFH, + h_mesh::Meshes.IntervalMesh = DefaultSliceXMesh( + FT; + x_min, + x_max, + periodic_x, + x_elem, + ), + z_mesh::Meshes.IntervalMesh = DefaultZMesh( + FT; + z_min, + z_max, + z_elem, + stretch, + ), +) where {FT} + @assert horizontal_layout_type <: DataLayouts.AbstractData + @assert ClimaComms.device(context) == device "The given device and context device do not match." + + h_topology = Topologies.IntervalTopology(context, h_mesh) + h_grid = + Grids.SpectralElementGrid1D(h_topology, quad; horizontal_layout_type) + z_topology = Topologies.IntervalTopology(context, z_mesh) + z_grid = Grids.FiniteDifferenceGrid(z_topology) + return Grids.ExtrudedFiniteDifferenceGrid( + h_grid, + z_grid, + hypsography_fun(h_grid, z_grid), + global_geometry, + ) +end + +""" + RectangleXYGrid( + ::Type{<:AbstractFloat}; # defaults to Float64 + x_min::Real, + x_max::Real, + y_min::Real, + y_max::Real, + periodic_x::Bool, + periodic_y::Bool, + n_quad_points::Integer, + x_elem::Integer, # number of horizontal elements + y_elem::Integer, # number of horizontal elements + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + hypsography::Grids.HypsographyAdaption = Grids.Flat(), + global_geometry::Geometry.AbstractGlobalGeometry = Geometry.CartesianGlobalGeometry(), + quad::Quadratures.QuadratureStyle = Quadratures.GLL{n_quad_points}(), + ) + +A convenience constructor, which builds a +[`Grids.SpectralElementGrid2D`](@ref) with a horizontal +`RectilinearMesh` mesh, given: + + - `x_min` the domain minimum along the x-direction. + - `x_max` the domain maximum along the x-direction. + - `y_min` the domain minimum along the y-direction. + - `y_max` the domain maximum along the y-direction. + - `periodic_x` Bool indicating to use periodic domain along x-direction + - `periodic_y` Bool indicating to use periodic domain along y-direction + - `n_quad_points` the number of quadrature points per horizontal element + - `x_elem` the number of x-points + - `y_elem` the number of y-points + - `device` the `ClimaComms.device` + - `context` the `ClimaComms.context` + - `hypsography_fun` a function or callable object (`hypsography_fun(h_grid, z_grid) -> hypsography`) for constructing the hypsography model. + - `global_geometry` the global geometry (defaults to [`Geometry.CartesianGlobalGeometry`](@ref)) + - `quad` the quadrature style (defaults to `Quadratures.GLL{n_quad_points}`) + +# Example usage + +```julia +using ClimaCore.CommonGrids +grid = RectangleXYGrid(; + x_min = 0, + x_max = 1, + y_min = 0, + y_max = 1, + periodic_x = false, + periodic_y = false, + n_quad_points = 4, + x_elem = 3, + y_elem = 4, +) +``` +""" +RectangleXYGrid(; kwargs...) = RectangleXYGrid(Float64; kwargs...) +function RectangleXYGrid( + ::Type{FT}; + x_min::Real, + x_max::Real, + y_min::Real, + y_max::Real, + periodic_x::Bool, + periodic_y::Bool, + n_quad_points::Integer, + x_elem::Integer, # number of horizontal elements + y_elem::Integer, # number of horizontal elements + device::ClimaComms.AbstractDevice = ClimaComms.device(), + context::ClimaComms.AbstractCommsContext = ClimaComms.context(device), + hypsography::Grids.HypsographyAdaption = Grids.Flat(), + global_geometry::Geometry.AbstractGlobalGeometry = Geometry.CartesianGlobalGeometry(), + quad::Quadratures.QuadratureStyle = Quadratures.GLL{n_quad_points}(), + horizontal_layout_type = DataLayouts.IJFH, + h_topology::Topologies.AbstractDistributedTopology = Topologies.Topology2D( + context, + DefaultRectangleXYMesh( + FT; + x_min, + x_max, + y_min, + y_max, + x_elem, + y_elem, + periodic_x, + periodic_y, + ), + ), + enable_bubble::Bool = false, +) where {FT} + @assert horizontal_layout_type <: DataLayouts.AbstractData + @assert ClimaComms.device(context) == device "The given device and context device do not match." + return Grids.SpectralElementGrid2D( + h_topology, + quad; + horizontal_layout_type, + enable_bubble, + ) +end + +end # module diff --git a/src/CommonGrids/Helpers.jl b/src/CommonGrids/Helpers.jl new file mode 100644 index 0000000000..3ad6c7ecd9 --- /dev/null +++ b/src/CommonGrids/Helpers.jl @@ -0,0 +1,115 @@ +module Helpers + +import ...Meshes, ...Geometry, ...Domains + + +##### +##### Mesh helpers +##### + +""" + DefaultSliceXMesh( + ::Type{<:AbstractFloat}; # defaults to Float64 + x_min::Real, + x_max::Real, + periodic_x::Bool, + x_elem::Integer, + ) + +A convenience constructor, which builds an `IntervalMesh`. +""" +DefaultSliceXMesh(; kwargs...) = DefaultSliceXMesh(Float64; kwargs...) +function DefaultSliceXMesh( + ::Type{FT}; + x_min::Real, + x_max::Real, + periodic_x::Bool, + x_elem::Integer, +) where {FT} + + x1boundary = (:east, :west) + z_boundary_names = (:bottom, :top) + h_domain = Domains.IntervalDomain( + Geometry.XPoint{FT}(x_min), + Geometry.XPoint{FT}(x_max); + periodic = periodic_x, + boundary_names = (:east, :west), + ) + return Meshes.IntervalMesh(h_domain; nelems = x_elem) +end + +""" + DefaultZMesh( + ::Type{<:AbstractFloat}; # defaults to Float64 + z_min::Real, + z_max::Real, + z_elem::Integer, + stretch::Meshes.StretchingRule = Meshes.Uniform(), + ) + +A convenience constructor, which builds an `IntervalMesh`. +""" +DefaultZMesh(; kwargs...) = DefaultZMesh(Float64; kwargs...) +function DefaultZMesh( + ::Type{FT}; + z_min::Real, + z_max::Real, + z_elem::Integer, + stretch::Meshes.StretchingRule = Meshes.Uniform(), +) where {FT} + z_boundary_names = (:bottom, :top) + z_domain = Domains.IntervalDomain( + Geometry.ZPoint{FT}(z_min), + Geometry.ZPoint{FT}(z_max); + boundary_names = z_boundary_names, + ) + return Meshes.IntervalMesh(z_domain, stretch; nelems = z_elem) +end + +""" + DefaultRectangleXYMesh( + ::Type{<:AbstractFloat}; # defaults to Float64 + x_min::Real, + x_max::Real, + y_min::Real, + y_max::Real, + periodic_x::Bool, + periodic_y::Bool, + ) + +A convenience constructor, which builds a +`RectilinearMesh` with a rectangular domain +composed of interval domains. +""" +DefaultRectangleXYMesh(; kwargs...) = DefaultRectangleXYMesh(Float64; kwargs...) +function DefaultRectangleXYMesh( + ::Type{FT}; + x_min::Real, + x_max::Real, + y_min::Real, + y_max::Real, + x_elem::Integer, + y_elem::Integer, + periodic_x::Bool, + periodic_y::Bool, +) where {FT <: AbstractFloat} + x1boundary = (:east, :west) + x2boundary = (:south, :north) + domain = Domains.RectangleDomain( + Domains.IntervalDomain( + Geometry.XPoint{FT}(x_min), + Geometry.XPoint{FT}(x_max); + periodic = periodic_x, + boundary_names = x1boundary, + ), + Domains.IntervalDomain( + Geometry.YPoint{FT}(y_min), + Geometry.YPoint{FT}(y_max); + periodic = periodic_y, + boundary_names = x2boundary, + ), + ) + return Meshes.RectilinearMesh(domain, x_elem, y_elem) +end + +end # module diff --git a/test/CommonGrids/CommonGrids.jl b/test/CommonGrids/CommonGrids.jl new file mode 100644 index 0000000000..e3f6caa927 --- /dev/null +++ b/test/CommonGrids/CommonGrids.jl @@ -0,0 +1,121 @@ +#= +julia --project +using Revise; include(joinpath("test", "CommonGrids", "CommonGrids.jl")) +=# +import ClimaComms +ClimaComms.@import_required_backends +using ClimaCore.CommonGrids +using ClimaCore: + Geometry, + Hypsography, + Fields, + Spaces, + Grids, + Topologies, + Meshes, + DataLayouts +using Test + +@testset "Convenience constructors" begin + function warp_surface(coord) + # sin²(x) form ground elevation + x = Geometry.component(coord, 1) + FT = eltype(x) + hc = FT(500.0) + h = hc * FT(sin(π * x / 25000)^2) + return h + end + + grid = ExtrudedCubedSphereGrid(; + z_elem = 10, + z_min = 0, + z_max = 1, + radius = 10, + h_elem = 10, + n_quad_points = 4, + horizontal_layout_type = DataLayouts.IJHF, + ) + @test grid isa Grids.ExtrudedFiniteDifferenceGrid + @test grid.horizontal_grid isa Grids.SpectralElementGrid2D + @test Grids.topology(grid.horizontal_grid).mesh isa + Meshes.EquiangularCubedSphere + + function hypsography_fun(h_grid, z_grid) + h_space = Spaces.SpectralElementSpace2D(h_grid) + cf = Fields.coordinate_field(h_space) + warp_fn = warp_surface # closure + z_surface = map(cf) do coord + Geometry.ZPoint(warp_fn(coord)) + end + Hypsography.LinearAdaption(z_surface) + end + + grid = ExtrudedCubedSphereGrid(; + z_elem = 10, + z_min = 0, + z_max = 1, + radius = 10, + h_elem = 10, + n_quad_points = 4, + hypsography_fun, + ) + @test grid isa Grids.ExtrudedFiniteDifferenceGrid + @test grid.horizontal_grid isa Grids.SpectralElementGrid2D + @test Grids.topology(grid.horizontal_grid).mesh isa + Meshes.EquiangularCubedSphere + @test Grids.topology(grid.horizontal_grid).mesh isa + Meshes.EquiangularCubedSphere + + grid = CubedSphereGrid(; radius = 10, n_quad_points = 4, h_elem = 10) + @test grid isa Grids.SpectralElementGrid2D + @test Grids.topology(grid).mesh isa Meshes.EquiangularCubedSphere + + grid = ColumnGrid(; z_elem = 10, z_min = 0, z_max = 1) + @test grid isa Grids.FiniteDifferenceGrid + + grid = Box3DGrid(; + z_elem = 10, + x_min = 0, + x_max = 1, + y_min = 0, + y_max = 1, + z_min = 0, + z_max = 10, + periodic_x = false, + periodic_y = false, + n_quad_points = 4, + x_elem = 3, + y_elem = 4, + ) + @test grid isa Grids.ExtrudedFiniteDifferenceGrid + @test grid.horizontal_grid isa Grids.SpectralElementGrid2D + @test Grids.topology(grid.horizontal_grid).mesh isa Meshes.RectilinearMesh + + grid = SliceXZGrid(; + z_elem = 10, + x_min = 0, + x_max = 1, + z_min = 0, + z_max = 1, + periodic_x = false, + n_quad_points = 4, + x_elem = 4, + ) + @test grid isa Grids.ExtrudedFiniteDifferenceGrid + @test grid.horizontal_grid isa Grids.SpectralElementGrid1D + @test Grids.topology(grid.horizontal_grid).mesh isa Meshes.IntervalMesh + + grid = RectangleXYGrid(; + x_min = 0, + x_max = 1, + y_min = 0, + y_max = 1, + periodic_x = false, + periodic_y = false, + n_quad_points = 4, + x_elem = 3, + y_elem = 4, + ) + @test grid isa Grids.SpectralElementGrid2D + @test Grids.topology(grid).mesh isa Meshes.RectilinearMesh +end diff --git a/test/runtests.jl b/test/runtests.jl index 9ffc4d01b5..5e964ae478 100644 --- a/test/runtests.jl +++ b/test/runtests.jl @@ -36,6 +36,7 @@ UnitTest("Cubedsphere topology" ,"Topologies/cubedsphere.jl") UnitTest("Cubedsphere surface topology" ,"Topologies/cubedsphere_sfc.jl"), UnitTest("dss_transform" ,"Topologies/unit_dss_transform.jl"), UnitTest("Quadratures" ,"Quadratures/Quadratures.jl"), +UnitTest("CommonGrids" ,"CommonGrids/CommonGrids.jl"), UnitTest("Spaces" ,"Spaces/unit_spaces.jl"), UnitTest("dss" ,"Spaces/unit_dss.jl"), UnitTest("Spaces - serial CPU DSS" ,"Spaces/ddss1.jl"),