From a2bc514b83dedb0719aa2dde3c215c7b6816b754 Mon Sep 17 00:00:00 2001 From: Charles Kawczynski Date: Tue, 27 Aug 2024 11:49:44 -0400 Subject: [PATCH] Refactor DSS to better leverage DataLayouts Remove unused indices and methods Fix some conversions --- ext/cuda/topologies_dss.jl | 325 +++--------------- src/Topologies/dss.jl | 587 ++++++-------------------------- src/Topologies/dss_transform.jl | 28 -- 3 files changed, 160 insertions(+), 780 deletions(-) diff --git a/ext/cuda/topologies_dss.jl b/ext/cuda/topologies_dss.jl index 062917cde2..6c36d64071 100644 --- a/ext/cuda/topologies_dss.jl +++ b/ext/cuda/topologies_dss.jl @@ -130,8 +130,8 @@ function dss_local_kernel!( local_vertices::AbstractVector{Tuple{Int, Int}}, local_vertex_offset::AbstractVector{Int}, interior_faces::AbstractVector{Tuple{Int, Int, Int, Int, Bool}}, - perimeter::Topologies.Perimeter2D{Nq}, -) where {Nq} + perimeter::Topologies.Perimeter2D, +) FT = eltype(parent(perimeter_data)) gidx = threadIdx().x + (blockIdx().x - Int32(1)) * blockDim().x nlocalvertices = length(local_vertex_offset) - 1 @@ -183,41 +183,23 @@ function Topologies.dss_transform!( device::ClimaComms.CUDADevice, perimeter_data::DataLayouts.VIFH, data::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - ∂ξ∂x::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - ∂x∂ξ::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - weight::DataLayouts.IJFH, perimeter::Topologies.Perimeter2D, - scalarfidx::AbstractVector{Int}, - covariant12fidx::AbstractVector{Int}, - contravariant12fidx::AbstractVector{Int}, - covariant123fidx::AbstractVector{Int}, - contravariant123fidx::AbstractVector{Int}, + local_geometry::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, + weight::DataLayouts.IJFH, localelems::AbstractVector{Int}, ) nlocalelems = length(localelems) if nlocalelems > 0 - pdata = parent(data) - pweight = parent(weight) - p∂x∂ξ = parent(∂x∂ξ) - p∂ξ∂x = parent(∂ξ∂x) - pperimeter_data = parent(perimeter_data) - nmetric = cld(length(p∂ξ∂x), prod(size(∂ξ∂x))) - (nlevels, nperimeter, _, _) = DataLayouts.array_size(perimeter_data) + (nperimeter, _, _, nlevels, _) = + DataLayouts.universal_size(perimeter_data) nitems = nlevels * nperimeter * nlocalelems nthreads, nblocks = _configure_threadblock(nitems) args = ( perimeter_data, - pdata, - p∂ξ∂x, - p∂x∂ξ, - Val(nmetric), - pweight, + data, perimeter, - scalarfidx, - covariant12fidx, - contravariant12fidx, - covariant123fidx, - contravariant123fidx, + local_geometry, + weight, localelems, Val(nlocalelems), ) @@ -233,146 +215,30 @@ end function dss_transform_kernel!( perimeter_data::DataLayouts.VIFH, - pdata::Union{AbstractArray{FT, 4}, AbstractArray{FT, 5}}, - p∂ξ∂x::Union{AbstractArray{FT, 4}, AbstractArray{FT, 5}}, - p∂x∂ξ::Union{AbstractArray{FT, 4}, AbstractArray{FT, 5}}, - ::Val{nmetric}, - pweight::AbstractArray{FT, 4}, - perimeter::Topologies.Perimeter2D{Nq}, - scalarfidx::AbstractVector{Int}, - covariant12fidx::AbstractVector{Int}, - contravariant12fidx::AbstractVector{Int}, - covariant123fidx::AbstractVector{Int}, - contravariant123fidx::AbstractVector{Int}, + data::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, + perimeter::Topologies.Perimeter2D, + local_geometry::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, + weight::DataLayouts.IJFH, localelems::AbstractVector{Int}, ::Val{nlocalelems}, -) where {FT <: AbstractFloat, Nq, nmetric, nlocalelems} - pperimeter_data = parent(perimeter_data) +) where {nlocalelems} gidx = threadIdx().x + (blockIdx().x - Int32(1)) * blockDim().x - (nlevels, nperimeter, nfid, nelems) = - DataLayouts.farray_size(perimeter_data) + (nperimeter, _, _, nlevels, nelems) = + DataLayouts.universal_size(perimeter_data) + CI = CartesianIndex if gidx ≤ nlevels * nperimeter * nlocalelems sizet = (nlevels, nperimeter, nlocalelems) - sizet_data = (nlevels, Nq, Nq, nfid, nelems) - sizet_wt = (Nq, Nq, 1, nelems) - sizet_metric = (nlevels, Nq, Nq, nmetric, nelems) - (level, p, localelemno) = cart_ind(sizet, gidx).I elem = localelems[localelemno] (ip, jp) = perimeter[p] - - weight = pweight[linear_ind(sizet_wt, (ip, jp, 1, elem))] - for fidx in scalarfidx - data_idx = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - pperimeter_data[level, p, fidx, elem] = pdata[data_idx] * weight - end - for fidx in covariant12fidx - data_idx1 = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - (idx11, idx12, idx21, idx22) = - Topologies._get_idx_metric(sizet_metric, (level, ip, jp, elem)) - pperimeter_data[level, p, fidx, elem] = - ( - p∂ξ∂x[idx11] * pdata[data_idx1] + - p∂ξ∂x[idx12] * pdata[data_idx2] - ) * weight - pperimeter_data[level, p, fidx + 1, elem] = - ( - p∂ξ∂x[idx21] * pdata[data_idx1] + - p∂ξ∂x[idx22] * pdata[data_idx2] - ) * weight - end - for fidx in contravariant12fidx - data_idx1 = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - (idx11, idx12, idx21, idx22) = - Topologies._get_idx_metric(sizet_metric, (level, ip, jp, elem)) - pperimeter_data[level, p, fidx, elem] = - ( - p∂x∂ξ[idx11] * pdata[data_idx1] + - p∂x∂ξ[idx21] * pdata[data_idx2] - ) * weight - pperimeter_data[level, p, fidx + 1, elem] = - ( - p∂x∂ξ[idx12] * pdata[data_idx1] + - p∂x∂ξ[idx22] * pdata[data_idx2] - ) * weight - end - for fidx in covariant123fidx - data_idx1 = - Topologies.linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = Topologies.linear_ind( - sizet_data, - (level, ip, jp, fidx + 1, elem), - ) - data_idx3 = Topologies.linear_ind( - sizet_data, - (level, ip, jp, fidx + 2, elem), - ) - - (idx11, idx12, idx13, idx21, idx22, idx23, idx31, idx32, idx33) = - Topologies._get_idx_metric_3d( - sizet_metric, - (level, ip, jp, elem), - ) - - # Covariant to physical transformation - pperimeter_data[level, p, fidx, elem] = - ( - p∂ξ∂x[idx11] * pdata[data_idx1] + - p∂ξ∂x[idx12] * pdata[data_idx2] + - p∂ξ∂x[idx13] * pdata[data_idx3] - ) * weight - pperimeter_data[level, p, fidx + 1, elem] = - ( - p∂ξ∂x[idx21] * pdata[data_idx1] + - p∂ξ∂x[idx22] * pdata[data_idx2] + - p∂ξ∂x[idx23] * pdata[data_idx3] - ) * weight - pperimeter_data[level, p, fidx + 2, elem] = - ( - p∂ξ∂x[idx31] * pdata[data_idx1] + - p∂ξ∂x[idx32] * pdata[data_idx2] + - p∂ξ∂x[idx33] * pdata[data_idx3] - ) * weight - end - - for fidx in contravariant123fidx - data_idx1 = - Topologies.linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = Topologies.linear_ind( - sizet_data, - (level, ip, jp, fidx + 1, elem), - ) - data_idx3 = Topologies.linear_ind( - sizet_data, - (level, ip, jp, fidx + 2, elem), - ) - (idx11, idx12, idx13, idx21, idx22, idx23, idx31, idx32, idx33) = - Topologies._get_idx_metric_3d( - sizet_metric, - (level, ip, jp, elem), - ) - # Contravariant to physical transformation - pperimeter_data[level, p, fidx, elem] = - ( - p∂x∂ξ[idx11] * pdata[data_idx1] + - p∂x∂ξ[idx21] * pdata[data_idx2] + - p∂x∂ξ[idx31] * pdata[data_idx3] - ) * weight - pperimeter_data[level, p, fidx + 1, elem] = - ( - p∂x∂ξ[idx12] * pdata[data_idx1] + - p∂x∂ξ[idx22] * pdata[data_idx2] + - p∂x∂ξ[idx32] * pdata[data_idx3] - ) * weight - pperimeter_data[level, p, fidx + 2, elem] = - ( - p∂x∂ξ[idx13] * pdata[data_idx1] + - p∂x∂ξ[idx23] * pdata[data_idx2] + - p∂x∂ξ[idx33] * pdata[data_idx3] - ) * weight - end + loc = CI(ip, jp, 1, level, elem) + src = Topologies.dss_transform( + data[loc], + local_geometry[loc], + weight[loc], + ) + perimeter_data[CI(p, 1, 1, level, elem)] = + Topologies.drop_vert_dim(eltype(perimeter_data), src) end return nothing end @@ -381,37 +247,21 @@ function Topologies.dss_untransform!( device::ClimaComms.CUDADevice, perimeter_data::DataLayouts.VIFH, data::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - ∂ξ∂x::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - ∂x∂ξ::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - perimeter::Topologies.Perimeter2D{Nq}, - scalarfidx::AbstractVector{Int}, - covariant12fidx::AbstractVector{Int}, - contravariant12fidx::AbstractVector{Int}, - covariant123fidx::AbstractVector{Int}, - contravariant123fidx::AbstractVector{Int}, + local_geometry::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, + perimeter::Topologies.Perimeter2D, localelems::AbstractVector{Int}, -) where {Nq} +) nlocalelems = length(localelems) if nlocalelems > 0 - pdata = parent(data) - p∂x∂ξ = parent(∂x∂ξ) - p∂ξ∂x = parent(∂ξ∂x) - nmetric = cld(length(p∂ξ∂x), prod(size(∂ξ∂x))) - (nlevels, nperimeter, _, _) = DataLayouts.array_size(perimeter_data) + (nperimeter, _, _, nlevels, _) = + DataLayouts.universal_size(perimeter_data) nitems = nlevels * nperimeter * nlocalelems nthreads, nblocks = _configure_threadblock(nitems) args = ( perimeter_data, - pdata, - p∂ξ∂x, - p∂x∂ξ, - nmetric, + data, + local_geometry, perimeter, - scalarfidx, - covariant12fidx, - contravariant12fidx, - covariant123fidx, - contravariant123fidx, localelems, Val(nlocalelems), ) @@ -427,104 +277,27 @@ end function dss_untransform_kernel!( perimeter_data::DataLayouts.VIFH, - pdata::Union{AbstractArray{FT, 4}, AbstractArray{FT, 5}}, - p∂ξ∂x::Union{AbstractArray{FT, 4}, AbstractArray{FT, 5}}, - p∂x∂ξ::Union{AbstractArray{FT, 4}, AbstractArray{FT, 5}}, - nmetric::Int, - perimeter::Topologies.Perimeter2D{Nq}, - scalarfidx::AbstractVector{Int}, - covariant12fidx::AbstractVector{Int}, - contravariant12fidx::AbstractVector{Int}, - covariant123fidx::AbstractVector{Int}, - contravariant123fidx::AbstractVector{Int}, + data::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, + local_geometry::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, + perimeter::Topologies.Perimeter2D, localelems::AbstractVector{Int}, ::Val{nlocalelems}, -) where {FT <: AbstractFloat, Nq, nlocalelems} +) where {nlocalelems} gidx = threadIdx().x + (blockIdx().x - Int32(1)) * blockDim().x - (nlevels, nperimeter, nfid, nelems) = - DataLayouts.farray_size(perimeter_data) - pperimeter_data = parent(perimeter_data) + (nperimeter, _, _, nlevels, _) = DataLayouts.universal_size(perimeter_data) + CI = CartesianIndex if gidx ≤ nlevels * nperimeter * nlocalelems sizet = (nlevels, nperimeter, nlocalelems) - sizet_data = (nlevels, Nq, Nq, nfid, nelems) - sizet_wt = (Nq, Nq, 1, nelems) - sizet_metric = (nlevels, Nq, Nq, nmetric, nelems) - (level, p, localelemno) = cart_ind(sizet, gidx).I elem = localelems[localelemno] ip, jp = perimeter[p] - for fidx in scalarfidx - data_idx = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - pdata[data_idx] = pperimeter_data[level, p, fidx, elem] - end - for fidx in covariant12fidx - data_idx1 = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - (idx11, idx12, idx21, idx22) = - Topologies._get_idx_metric(sizet_metric, (level, ip, jp, elem)) - pdata[data_idx1] = - p∂x∂ξ[idx11] * pperimeter_data[level, p, fidx, elem] + - p∂x∂ξ[idx12] * pperimeter_data[level, p, fidx + 1, elem] - pdata[data_idx2] = - p∂x∂ξ[idx21] * pperimeter_data[level, p, fidx, elem] + - p∂x∂ξ[idx22] * pperimeter_data[level, p, fidx + 1, elem] - end - for fidx in contravariant12fidx - data_idx1 = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - (idx11, idx12, idx21, idx22) = - Topologies._get_idx_metric(sizet_metric, (level, ip, jp, elem)) - pdata[data_idx1] = - p∂ξ∂x[idx11] * pperimeter_data[level, p, fidx, elem] + - p∂ξ∂x[idx21] * pperimeter_data[level, p, fidx + 1, elem] - pdata[data_idx2] = - p∂ξ∂x[idx12] * pperimeter_data[level, p, fidx, elem] + - p∂ξ∂x[idx22] * pperimeter_data[level, p, fidx + 1, elem] - end - for fidx in covariant123fidx - data_idx1 = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - data_idx3 = linear_ind(sizet_data, (level, ip, jp, fidx + 2, elem)) - (idx11, idx12, idx13, idx21, idx22, idx23, idx31, idx32, idx33) = - Topologies._get_idx_metric_3d( - sizet_metric, - (level, ip, jp, elem), - ) - pdata[data_idx1] = - p∂x∂ξ[idx11] * pperimeter_data[level, p, fidx, elem] + - p∂x∂ξ[idx12] * pperimeter_data[level, p, fidx + 1, elem] + - p∂x∂ξ[idx13] * pperimeter_data[level, p, fidx + 2, elem] - pdata[data_idx2] = - p∂x∂ξ[idx21] * pperimeter_data[level, p, fidx, elem] + - p∂x∂ξ[idx22] * pperimeter_data[level, p, fidx + 1, elem] + - p∂x∂ξ[idx23] * pperimeter_data[level, p, fidx + 2, elem] - pdata[data_idx3] = - p∂x∂ξ[idx31] * pperimeter_data[level, p, fidx, elem] + - p∂x∂ξ[idx32] * pperimeter_data[level, p, fidx + 1, elem] + - p∂x∂ξ[idx33] * pperimeter_data[level, p, fidx + 2, elem] - end - for fidx in contravariant123fidx - data_idx1 = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - data_idx3 = linear_ind(sizet_data, (level, ip, jp, fidx + 2, elem)) - (idx11, idx12, idx13, idx21, idx22, idx23, idx31, idx32, idx33) = - Topologies._get_idx_metric_3d( - sizet_metric, - (level, ip, jp, elem), - ) - pdata[data_idx1] = - p∂ξ∂x[idx11] * pperimeter_data[level, p, fidx, elem] + - p∂ξ∂x[idx21] * pperimeter_data[level, p, fidx + 1, elem] + - p∂ξ∂x[idx31] * pperimeter_data[level, p, fidx + 2, elem] - pdata[data_idx2] = - p∂ξ∂x[idx12] * pperimeter_data[level, p, fidx, elem] + - p∂ξ∂x[idx22] * pperimeter_data[level, p, fidx + 1, elem] - p∂ξ∂x[idx32] * pperimeter_data[level, p, fidx + 2, elem] - pdata[data_idx3] = - p∂ξ∂x[idx13] * pperimeter_data[level, p, fidx, elem] + - p∂ξ∂x[idx23] * pperimeter_data[level, p, fidx + 1, elem] + - p∂ξ∂x[idx33] * pperimeter_data[level, p, fidx + 2, elem] - end + + loc = CI(ip, jp, 1, level, elem) + data[loc] = Topologies.dss_untransform( + eltype(data), + perimeter_data[CI(p, 1, 1, level, elem)], + local_geometry[loc], + ) end return nothing end @@ -563,8 +336,8 @@ function dss_local_ghost_kernel!( perimeter_data::DataLayouts.VIFH, ghost_vertices, ghost_vertex_offset, - perimeter::Topologies.Perimeter2D{Nq}, -) where {Nq} + perimeter::Topologies.Perimeter2D, +) gidx = threadIdx().x + (blockIdx().x - Int32(1)) * blockDim().x pperimeter_data = parent(perimeter_data) FT = eltype(pperimeter_data) @@ -722,8 +495,8 @@ function dss_ghost_kernel!( ghost_vertices, ghost_vertex_offset, repr_ghost_vertex, - perimeter::Topologies.Perimeter2D{Nq}, -) where {Nq} + perimeter::Topologies.Perimeter2D, +) pperimeter_data = parent(perimeter_data) FT = eltype(pperimeter_data) gidx = threadIdx().x + (blockIdx().x - Int32(1)) * blockDim().x diff --git a/src/Topologies/dss.jl b/src/Topologies/dss.jl index ef25cb9c37..e2848a17a7 100644 --- a/src/Topologies/dss.jl +++ b/src/Topologies/dss.jl @@ -22,16 +22,6 @@ struct DSSBuffer{S, G, D, A, B, VI} send_buf_idx::B "indexing array for loading (and summing) data from recv buffer to `perimeter_data`" recv_buf_idx::B - "field id for all scalar fields stored in the `data` array" - scalarfidx::VI - "field id for all covariant12vector fields stored in the `data` array" - covariant12fidx::VI - "field id for all contravariant12vector fields stored in the `data` array" - contravariant12fidx::VI - "internal local elements (lidx)" - covariant123fidx::VI - "field id for all contravariant123vector fields stored in the `data` array" - contravariant123fidx::VI "internal local elements (lidx)" internal_elems::VI "local elements (lidx) located on process boundary" @@ -69,14 +59,15 @@ function create_dss_buffer( Nf = DataLayouts.ncomponents(data) nfacedof = Nij - 2 T = eltype(parent(data)) - TS = _transformed_type(data, local_geometry, local_weights, DA) # extract transformed type # Add TS for Covariant123Vector # For DSS of Covariant123Vector, the third component is treated like a scalar # and is not transformed - if eltype(data) <: Geometry.Covariant123Vector - TS = Geometry.UVWVector{T} + TS = if eltype(data) <: Geometry.Covariant123Vector + Geometry.UVWVector{T} elseif eltype(data) <: Geometry.Contravariant123Vector - TS = Geometry.UVWVector{T} + Geometry.UVWVector{T} + else + _transformed_type(data, local_geometry, local_weights, DA) # extract transformed type end perimeter_data = DataLayouts.VIFH{TS, Nv, Np, Nh}(DA{T}(undef, Nv, Np, Nf, Nh)) @@ -111,92 +102,11 @@ function create_dss_buffer( internal_elems = DA(topology.internal_elems) perimeter_elems = DA(topology.perimeter_elems) end - scalarfidx, - covariant12fidx, - contravariant12fidx, - covariant123fidx, - contravariant123fidx = Int[], Int[], Int[], Int[], Int[] - supportedvectortypes = Union{ - Geometry.UVector, - Geometry.VVector, - Geometry.WVector, - Geometry.UVVector, - Geometry.UWVector, - Geometry.VWVector, - Geometry.UVWVector, - Geometry.Covariant12Vector, - Geometry.Covariant3Vector, - Geometry.Covariant123Vector, - Geometry.Contravariant123Vector, - Geometry.Contravariant12Vector, - Geometry.Contravariant3Vector, - } - - if S <: NamedTuple - for (i, fieldtype) in enumerate(S.parameters[2].types) - offset = DataLayouts.fieldtypeoffset(T, S, i) - ncomponents = DataLayouts.typesize(T, fieldtype) - if fieldtype <: Geometry.AxisVector # vector fields - if !(fieldtype <: supportedvectortypes) - @show fieldtype - @show supportedvectortypes - end - @assert fieldtype <: supportedvectortypes - if fieldtype <: Geometry.Covariant12Vector - push!(covariant12fidx, offset + 1) - elseif fieldtype <: Geometry.Covariant123Vector - push!(covariant123fidx, offset + 1) - elseif fieldtype <: Geometry.Contravariant12Vector - push!(contravariant12fidx, offset + 1) - elseif fieldtype <: Geometry.Contravariant123Vector - push!(contravariant123fidx, offset + 1) - else - append!( - scalarfidx, - Vector((offset + 1):(offset + ncomponents)), - ) - end - elseif fieldtype <: NTuple # support a NTuple of primitive types - append!(scalarfidx, Vector((offset + 1):(offset + ncomponents))) - else # scalar fields - push!(scalarfidx, offset + 1) - end - end - else # deals with simple type, with single field (e.g: S = Float64, S = CovariantVector12, etc.) - ncomponents = DataLayouts.typesize(T, S) - if S <: Geometry.AxisVector # vector field - if !(S <: supportedvectortypes) - @show S - @show supportedvectortypes - end - @assert S <: supportedvectortypes - if S <: Geometry.Covariant12Vector - push!(covariant12fidx, 1) - elseif S <: Geometry.Covariant123Vector - push!(covariant123fidx, 1) - elseif S <: Geometry.Contravariant12Vector - push!(contravariant12fidx, 1) - elseif S <: Geometry.Contravariant123Vector - push!(contravariant123fidx, 1) - else - append!(scalarfidx, Vector(1:ncomponents)) - end - elseif S <: NTuple # support a NTuple of primitive types - append!(scalarfidx, Vector(1:ncomponents)) - else # scalar field - push!(scalarfidx, 1) - end - end - scalarfidx = DA(scalarfidx) - covariant12fidx = DA(covariant12fidx) - covariant123fidx = DA(covariant123fidx) - contravariant12fidx = DA(contravariant12fidx) - contravariant123fidx = DA(contravariant123fidx) G = typeof(graph_context) D = typeof(perimeter_data) A = typeof(send_data) B = typeof(send_buf_idx) - VI = typeof(scalarfidx) + VI = typeof(perimeter_elems) return DSSBuffer{S, G, D, A, B, VI}( graph_context, perimeter_data, @@ -204,11 +114,6 @@ function create_dss_buffer( recv_data, send_buf_idx, recv_buf_idx, - scalarfidx, - covariant12fidx, - contravariant12fidx, - covariant123fidx, - contravariant123fidx, internal_elems, perimeter_elems, ) @@ -258,104 +163,45 @@ function dss_transform!( localelems::AbstractVector{Int}, ) if !isempty(localelems) - (; - scalarfidx, - covariant12fidx, - contravariant12fidx, - covariant123fidx, - contravariant123fidx, - perimeter_data, - ) = dss_buffer - (; ∂ξ∂x, ∂x∂ξ) = local_geometry dss_transform!( device, - perimeter_data, + dss_buffer.perimeter_data, data, - ∂ξ∂x, - ∂x∂ξ, - weight, perimeter, - scalarfidx, - covariant12fidx, - contravariant12fidx, - covariant123fidx, - contravariant123fidx, + local_geometry, + weight, localelems, ) end return nothing end -""" - dss_untransform!( - device::ClimaComms.AbstractDevice, - dss_buffer::DSSBuffer, - data::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, - local_geometry::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, - perimeter::AbstractPerimeter, - ) -Transforms the DSS'd local vectors back to Covariant12 vectors, and copies the DSS'd data from the -`perimeter_data` to `data`. This function calls the appropriate version of `dss_transform!` function -based on the data layout of the input arguments. - -Arguments: - -- `dss_buffer`: [`DSSBuffer`](@ref) generated by `create_dss_buffer` function for field data -- `data`: field data -- `local_geometry`: local metric information defined at each node -- `perimeter`: perimeter iterator -- `localelems`: list of local elements to perform transformation operations on +# For DSS of Covariant123Vector, the third component is treated like a scalar +# and is not transformed, therefore, we drop the vertical dimension: +""" + drop_vert_dim(::Type{T}, X) -Part of [`ClimaCore.Spaces.weighted_dss!`](@ref). +Convert the type of `X` to type `T` recursively +using `_drop_vert_dim`, which converts from `UVWVector` +to `UVVector` if `T <: UVVector`. """ -function dss_untransform!( - device::ClimaComms.AbstractDevice, - dss_buffer::DSSBuffer, - data::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, - local_geometry::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, - perimeter::Perimeter2D, - localelems::AbstractVector{Int}, -) - (; - scalarfidx, - covariant12fidx, - contravariant12fidx, - covariant123fidx, - contravariant123fidx, - perimeter_data, - ) = dss_buffer - (; ∂ξ∂x, ∂x∂ξ) = local_geometry - dss_untransform!( - device, - perimeter_data, - data, - ∂ξ∂x, - ∂x∂ξ, - perimeter, - scalarfidx, - covariant12fidx, - contravariant12fidx, - covariant123fidx, - contravariant123fidx, - localelems, - ) - return nothing -end +@inline drop_vert_dim(::Type{T}, X) where {T} = + RecursiveApply.rmap(RecursiveApply.rzero(T), X) do zero_value, x + _drop_vert_dim(typeof(zero_value), x) + end +@inline _drop_vert_dim( + ::Type{T}, + x::Geometry.UVWVector, +) where {T <: Geometry.UVVector} = Geometry.UVVector(x.u, x.v) +@inline _drop_vert_dim(::Type{T}, x::T) where {T} = x """ function dss_transform!( ::ClimaComms.AbstractCPUDevice, perimeter_data::DataLayouts.VIFH, data::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, - ∂ξ∂x::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, - ∂x∂ξ::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - weight::DataLayouts.IJFH, perimeter::AbstractPerimeter, - scalarfidx::Vector{Int}, - covariant12fidx::Vector{Int}, - contravariant12fidx::Vector{Int}, - covariant123fidx::Vector{Int}, - contravariant123fidx::Vector{Int}, + bc, localelems::Vector{Int}, ) @@ -366,12 +212,8 @@ Arguments: - `perimeter_data`: contains the perimeter field data, represented on the physical axis, corresponding to the full field data in `data` - `data`: field data -- `∂ξ∂x`: partial derivatives of the map from `x` to `ξ`: `∂ξ∂x[i,j]` is ∂ξⁱ/∂xʲ -- `weight`: local dss weights for horizontal space - `perimeter`: perimeter iterator -- `scalarfidx`: field index for scalar fields in the data layout -- `covariant12fidx`: field index for Covariant12 vector fields in the data layout -- `covariant123fidx`: field index for Covariant123 vector fields in the data layout +- `bc`: A broadcasted object representing the dss transform. - `localelems`: list of local elements to perform transformation operations on Part of [`ClimaCore.Spaces.weighted_dss!`](@ref). @@ -380,166 +222,82 @@ function dss_transform!( ::ClimaComms.AbstractCPUDevice, perimeter_data::DataLayouts.VIFH, data::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - ∂ξ∂x::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - ∂x∂ξ::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - weight::DataLayouts.IJFH, perimeter::Perimeter2D{Nq}, - scalarfidx::Vector{Int}, - covariant12fidx::Vector{Int}, - contravariant12fidx::Vector{Int}, - covariant123fidx::Vector{Int}, - contravariant123fidx::Vector{Int}, + local_geometry, + weight, localelems::Vector{Int}, ) where {Nq} - pdata = parent(data) - pweight = parent(weight) - p∂x∂ξ = parent(∂x∂ξ) - p∂ξ∂x = parent(∂ξ∂x) - pperimeter_data = parent(perimeter_data) - (nlevels, _, nfid, nelems) = DataLayouts.farray_size(perimeter_data) - - nmetric = cld(prod(DataLayouts.farray_size(∂ξ∂x)), prod(size(∂ξ∂x))) - sizet_data = (nlevels, Nq, Nq, nfid, nelems) - sizet_wt = (Nq, Nq, 1, nelems) - sizet_metric = (nlevels, Nq, Nq, nmetric, nelems) - + (_, _, _, nlevels, _) = DataLayouts.universal_size(perimeter_data) + CI = CartesianIndex + (; ∂ξ∂x) = local_geometry @inbounds for elem in localelems for (p, (ip, jp)) in enumerate(perimeter) - pw = pweight[linear_ind(sizet_wt, (ip, jp, 1, elem))] + for level in 1:nlevels + loc = CI(ip, jp, 1, level, elem) + src = dss_transform( + data[loc], + local_geometry[CI(ip, jp, 1, level, elem)], + weight[loc], + ) + perimeter_data[CI(p, 1, 1, level, elem)] = + drop_vert_dim(eltype(perimeter_data), src) - for fidx in scalarfidx, level in 1:nlevels - data_idx = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - pperimeter_data[level, p, fidx, elem] = pdata[data_idx] * pw end + end + end + return nothing +end - for fidx in covariant12fidx, level in 1:nlevels - data_idx1 = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = - linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - (idx11, idx12, idx21, idx22) = - _get_idx_metric(sizet_metric, (level, ip, jp, elem)) - pperimeter_data[level, p, fidx, elem] = - ( - p∂ξ∂x[idx11] * pdata[data_idx1] + - p∂ξ∂x[idx12] * pdata[data_idx2] - ) * pw - pperimeter_data[level, p, fidx + 1, elem] = - ( - p∂ξ∂x[idx21] * pdata[data_idx1] + - p∂ξ∂x[idx22] * pdata[data_idx2] - ) * pw - end +""" + dss_untransform!( + device::ClimaComms.AbstractDevice, + dss_buffer::DSSBuffer, + data::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, + local_geometry::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, + perimeter::AbstractPerimeter, + ) - for fidx in contravariant12fidx, level in 1:nlevels - data_idx1 = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = - linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - (idx11, idx12, idx21, idx22) = - _get_idx_metric(sizet_metric, (level, ip, jp, elem)) - pperimeter_data[level, p, fidx, elem] = - ( - p∂x∂ξ[idx11] * pdata[data_idx1] + - p∂x∂ξ[idx21] * pdata[data_idx2] - ) * pw - pperimeter_data[level, p, fidx + 1, elem] = - ( - p∂x∂ξ[idx12] * pdata[data_idx1] + - p∂x∂ξ[idx22] * pdata[data_idx2] - ) * pw - end +Transforms the DSS'd local vectors back to Covariant12 vectors, and copies the DSS'd data from the +`perimeter_data` to `data`. This function calls the appropriate version of `dss_transform!` function +based on the data layout of the input arguments. - for fidx in covariant123fidx, level in 1:nlevels - data_idx1 = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = - linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - data_idx3 = - linear_ind(sizet_data, (level, ip, jp, fidx + 2, elem)) +Arguments: - ( - idx11, - idx12, - idx13, - idx21, - idx22, - idx23, - idx31, - idx32, - idx33, - ) = _get_idx_metric_3d(sizet_metric, (level, ip, jp, elem)) +- `dss_buffer`: [`DSSBuffer`](@ref) generated by `create_dss_buffer` function for field data +- `data`: field data +- `local_geometry`: local metric information defined at each node +- `perimeter`: perimeter iterator +- `localelems`: list of local elements to perform transformation operations on - # Covariant to physical transformation - pperimeter_data[level, p, fidx, elem] = - ( - p∂ξ∂x[idx11] * pdata[data_idx1] + - p∂ξ∂x[idx12] * pdata[data_idx2] + - p∂ξ∂x[idx13] * pdata[data_idx3] - ) * pw - pperimeter_data[level, p, fidx + 1, elem] = - ( - p∂ξ∂x[idx21] * pdata[data_idx1] + - p∂ξ∂x[idx22] * pdata[data_idx2] + - p∂ξ∂x[idx23] * pdata[data_idx3] - ) * pw - pperimeter_data[level, p, fidx + 2, elem] = - ( - p∂ξ∂x[idx31] * pdata[data_idx1] + - p∂ξ∂x[idx32] * pdata[data_idx2] + - p∂ξ∂x[idx33] * pdata[data_idx3] - ) * pw - end +Part of [`ClimaCore.Spaces.weighted_dss!`](@ref). +""" +function dss_untransform!( + device::ClimaComms.AbstractDevice, + dss_buffer::DSSBuffer, + data::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, + local_geometry::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, + perimeter::Perimeter2D, + localelems::AbstractVector{Int}, +) - for fidx in contravariant123fidx, level in 1:nlevels - data_idx1 = linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = - linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - data_idx3 = - linear_ind(sizet_data, (level, ip, jp, fidx + 2, elem)) - ( - idx11, - idx12, - idx13, - idx21, - idx22, - idx23, - idx31, - idx32, - idx33, - ) = _get_idx_metric_3d(sizet_metric, (level, ip, jp, elem)) - # Contravariant to physical transformation - pperimeter_data[level, p, fidx, elem] = - ( - p∂x∂ξ[idx11] * pdata[data_idx1] + - p∂x∂ξ[idx21] * pdata[data_idx2] + - p∂x∂ξ[idx31] * pdata[data_idx3] - ) * pw - pperimeter_data[level, p, fidx + 1, elem] = - ( - p∂x∂ξ[idx12] * pdata[data_idx1] + - p∂x∂ξ[idx22] * pdata[data_idx2] + - p∂x∂ξ[idx32] * pdata[data_idx3] - ) * pw - pperimeter_data[level, p, fidx + 2, elem] = - ( - p∂x∂ξ[idx13] * pdata[data_idx1] + - p∂x∂ξ[idx23] * pdata[data_idx2] + - p∂x∂ξ[idx33] * pdata[data_idx3] - ) * pw - end - end - end + (; perimeter_data) = dss_buffer + dss_untransform!( + device, + perimeter_data, + data, + local_geometry, + perimeter, + localelems, + ) return nothing end + """ function dss_untransform!( ::ClimaComms.AbstractCPUDevice, perimeter_data::DataLayouts.VIFH, data::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, - ∂ξ∂x::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - ∂x∂ξ::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, - perimeter::AbstractPerimeter, - scalarfidx::Vector{Int}, - covariant12fidx::Vector{Int}, - contravariant12fidx::Vector{Int}, + local_geometry, localelems::Vector{Int}, ) @@ -550,10 +308,7 @@ Arguments: - `perimeter_data`: contains the perimeter field data, represented on the physical axis, corresponding to the full field data in `data` - `data`: field data -- `∂x∂ξ`: partial derivatives of the map from `ξ` to `x`: `∂x∂ξ[i,j]` is ∂xⁱ/∂ξʲ -- `perimeter`: perimeter iterator -- `scalarfidx`: field index for scalar fields in the data layout -- `covariant12fidx`: field index for Covariant12 vector fields in the data layout +- `local_geometry`: Field data containing local geometry Part of [`ClimaCore.Spaces.weighted_dss!`](@ref). """ @@ -562,136 +317,20 @@ function dss_untransform!( ::ClimaComms.AbstractCPUDevice, perimeter_data::DataLayouts.VIFH, data::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - ∂ξ∂x::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - ∂x∂ξ::Union{DataLayouts.VIJFH, DataLayouts.IJFH}, - perimeter::Perimeter2D{Nq}, - scalarfidx::Vector{Int}, - covariant12fidx::Vector{Int}, - contravariant12fidx::Vector{Int}, - covariant123fidx::Vector{Int}, - contravariant123fidx::Vector{Int}, + local_geometry::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, + perimeter::Perimeter2D, localelems::Vector{Int}, -) where {Nq} - pdata = parent(data) - p∂x∂ξ = parent(∂x∂ξ) - p∂ξ∂x = parent(∂ξ∂x) - pperimeter_data = parent(perimeter_data) - (nlevels, _, nfid, nelems) = DataLayouts.farray_size(perimeter_data) - nmetric = cld(prod(DataLayouts.farray_size(∂ξ∂x)), prod(size(∂ξ∂x))) - sizet_data = (nlevels, Nq, Nq, nfid, nelems) - sizet_metric = (nlevels, Nq, Nq, nmetric, nelems) - +) + (_, _, _, nlevels, _) = DataLayouts.universal_size(perimeter_data) + CI = CartesianIndex @inbounds for elem in localelems for (p, (ip, jp)) in enumerate(perimeter) - for fidx in scalarfidx - for level in 1:nlevels - data_idx = - linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - pdata[data_idx] = pperimeter_data[level, p, fidx, elem] - end - end - for fidx in covariant12fidx - for level in 1:nlevels - data_idx1 = - linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = - linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - (idx11, idx12, idx21, idx22) = - _get_idx_metric(sizet_metric, (level, ip, jp, elem)) - pdata[data_idx1] = - p∂x∂ξ[idx11] * pperimeter_data[level, p, fidx, elem] + - p∂x∂ξ[idx12] * pperimeter_data[level, p, fidx + 1, elem] - pdata[data_idx2] = - p∂x∂ξ[idx21] * pperimeter_data[level, p, fidx, elem] + - p∂x∂ξ[idx22] * pperimeter_data[level, p, fidx + 1, elem] - end - end - for fidx in contravariant12fidx - for level in 1:nlevels - data_idx1 = - linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = - linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - (idx11, idx12, idx21, idx22) = - _get_idx_metric(sizet_metric, (level, ip, jp, elem)) - pdata[data_idx1] = - p∂ξ∂x[idx11] * pperimeter_data[level, p, fidx, elem] + - p∂ξ∂x[idx21] * pperimeter_data[level, p, fidx + 1, elem] - pdata[data_idx2] = - p∂ξ∂x[idx12] * pperimeter_data[level, p, fidx, elem] + - p∂ξ∂x[idx22] * pperimeter_data[level, p, fidx + 1, elem] - end - end - for fidx in covariant123fidx - for level in 1:nlevels - data_idx1 = - linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = - linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - data_idx3 = - linear_ind(sizet_data, (level, ip, jp, fidx + 2, elem)) - ( - idx11, - idx12, - idx13, - idx21, - idx22, - idx23, - idx31, - idx32, - idx33, - ) = _get_idx_metric_3d(sizet_metric, (level, ip, jp, elem)) - pdata[data_idx1] = - p∂x∂ξ[idx11] * pperimeter_data[level, p, fidx, elem] + - p∂x∂ξ[idx12] * - pperimeter_data[level, p, fidx + 1, elem] + - p∂x∂ξ[idx13] * pperimeter_data[level, p, fidx + 2, elem] - pdata[data_idx2] = - p∂x∂ξ[idx21] * pperimeter_data[level, p, fidx, elem] + - p∂x∂ξ[idx22] * - pperimeter_data[level, p, fidx + 1, elem] + - p∂x∂ξ[idx23] * pperimeter_data[level, p, fidx + 2, elem] - pdata[data_idx3] = - p∂x∂ξ[idx31] * pperimeter_data[level, p, fidx, elem] + - p∂x∂ξ[idx32] * - pperimeter_data[level, p, fidx + 1, elem] + - p∂x∂ξ[idx33] * pperimeter_data[level, p, fidx + 2, elem] - end - end - for fidx in contravariant123fidx - for level in 1:nlevels - data_idx1 = - linear_ind(sizet_data, (level, ip, jp, fidx, elem)) - data_idx2 = - linear_ind(sizet_data, (level, ip, jp, fidx + 1, elem)) - data_idx3 = - linear_ind(sizet_data, (level, ip, jp, fidx + 2, elem)) - - ( - idx11, - idx12, - idx13, - idx21, - idx22, - idx23, - idx31, - idx32, - idx33, - ) = _get_idx_metric_3d(sizet_metric, (level, ip, jp, elem)) - pdata[data_idx1] = - p∂ξ∂x[idx11] * pperimeter_data[level, p, fidx, elem] + - p∂ξ∂x[idx21] * pperimeter_data[level, p, fidx + 1, elem] - p∂ξ∂x[idx31] * pperimeter_data[level, p, fidx + 2, elem] - pdata[data_idx2] = - p∂ξ∂x[idx12] * pperimeter_data[level, p, fidx, elem] + - p∂ξ∂x[idx22] * pperimeter_data[level, p, fidx + 1, elem] - p∂ξ∂x[idx32] * pperimeter_data[level, p, fidx + 2, elem] - pdata[data_idx3] = - p∂ξ∂x[idx13] * pperimeter_data[level, p, fidx, elem] + - p∂ξ∂x[idx23] * - pperimeter_data[level, p, fidx + 1, elem] + - p∂ξ∂x[idx33] * pperimeter_data[level, p, fidx + 2, elem] - end + for level in 1:nlevels + data[CI(ip, jp, 1, level, elem)] = dss_untransform( + eltype(data), + perimeter_data[CI(p, 1, 1, level, elem)], + local_geometry[CI(ip, jp, 1, level, elem)], + ) end end end @@ -702,17 +341,15 @@ function dss_load_perimeter_data!( ::ClimaComms.AbstractCPUDevice, dss_buffer::DSSBuffer, data::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, - perimeter::Perimeter2D{Nq}, -) where {Nq} + perimeter::Perimeter2D, +) (; perimeter_data) = dss_buffer - pperimeter_data = parent(perimeter_data) - pdata = parent(data) - (nlevels, _, nfid, nelems) = DataLayouts.farray_size(perimeter_data) - sizet = (nlevels, Nq, Nq, nfid, nelems) - for elem in 1:nelems, (p, (ip, jp)) in enumerate(perimeter) - for fidx in 1:nfid, level in 1:nlevels - idx = linear_ind(sizet, (level, ip, jp, fidx, elem)) - pperimeter_data[level, p, fidx, elem] = pdata[idx] + (_, _, _, nlevels, nelems) = DataLayouts.universal_size(perimeter_data) + CI = CartesianIndex + @inbounds for elem in 1:nelems, (p, (ip, jp)) in enumerate(perimeter) + for level in 1:nlevels + perimeter_data[CI(p, 1, 1, level, elem)] = + data[CI(ip, jp, 1, level, elem)] end end return nothing @@ -722,17 +359,15 @@ function dss_unload_perimeter_data!( ::ClimaComms.AbstractCPUDevice, data::Union{DataLayouts.IJFH, DataLayouts.VIJFH}, dss_buffer::DSSBuffer, - perimeter::Perimeter2D{Nq}, -) where {Nq} + perimeter::Perimeter2D, +) (; perimeter_data) = dss_buffer - pperimeter_data = parent(perimeter_data) - pdata = parent(data) - (nlevels, _, nfid, nelems) = DataLayouts.farray_size(perimeter_data) - sizet = (nlevels, Nq, Nq, nfid, nelems) - for elem in 1:nelems, (p, (ip, jp)) in enumerate(perimeter) - for fidx in 1:nfid, level in 1:nlevels - idx = linear_ind(sizet, (level, ip, jp, fidx, elem)) - pdata[idx] = pperimeter_data[level, p, fidx, elem] + (_, _, _, nlevels, nelems) = DataLayouts.universal_size(perimeter_data) + CI = CartesianIndex + @inbounds for elem in 1:nelems, (p, (ip, jp)) in enumerate(perimeter) + for level in 1:nlevels + data[CI(ip, jp, 1, level, elem)] = + perimeter_data[CI(p, 1, 1, level, elem)] end end return nothing diff --git a/src/Topologies/dss_transform.jl b/src/Topologies/dss_transform.jl index 5f48721a09..d1ca3f6268 100644 --- a/src/Topologies/dss_transform.jl +++ b/src/Topologies/dss_transform.jl @@ -201,34 +201,6 @@ end # helper functions for DSS2 -function _get_idx_metric(sizet::NTuple{5, Int}, loc::NTuple{4, Int}) - nmetric = sizet[4] - (i11, i12, i21, i22) = nmetric == 4 ? (1, 2, 3, 4) : (1, 2, 4, 5) - (level, i, j, elem) = loc - return ( - linear_ind(sizet, (level, i, j, i11, elem)), - linear_ind(sizet, (level, i, j, i12, elem)), - linear_ind(sizet, (level, i, j, i21, elem)), - linear_ind(sizet, (level, i, j, i22, elem)), - ) -end - -function _get_idx_metric_3d(sizet::NTuple{5, Int}, loc::NTuple{4, Int}) - nmetric = sizet[4] - (level, i, j, elem) = loc - return ( - linear_ind(sizet, (level, i, j, 1, elem)), - linear_ind(sizet, (level, i, j, 2, elem)), - linear_ind(sizet, (level, i, j, 3, elem)), - linear_ind(sizet, (level, i, j, 4, elem)), - linear_ind(sizet, (level, i, j, 5, elem)), - linear_ind(sizet, (level, i, j, 6, elem)), - linear_ind(sizet, (level, i, j, 7, elem)), - linear_ind(sizet, (level, i, j, 8, elem)), - linear_ind(sizet, (level, i, j, 9, elem)), - ) -end - function _representative_slab( data::Union{DataLayouts.AbstractData, Nothing}, ::Type{DA},