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

Port z ifc #564

Open
wants to merge 74 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 70 commits
Commits
Show all changes
74 commits
Select commit Hold shift + click to select a range
991a0b1
port init_vert_coord
jcanton Sep 5, 2024
417d667
rename indexes
jcanton Sep 5, 2024
727814b
actually exit
jcanton Sep 6, 2024
7a83403
renaming
jcanton Sep 25, 2024
91c4d31
temp cached work on topo_smooth
jcanton Sep 25, 2024
40adcbf
porting with Nikki
jcanton Oct 1, 2024
5c42bdd
wip with Nikki
jcanton Oct 3, 2024
fb9299a
move to CellKField
jcanton Oct 3, 2024
2ff7134
more wip, `grid` works (not with savepoint); `icon_grid` doesn't (at …
jcanton Oct 3, 2024
1b0cc4e
wip testing
jcanton Oct 4, 2024
efbd294
fix KDim
jcanton Oct 4, 2024
fdf8622
wip backend issues
jcanton Oct 16, 2024
d2918c6
tests working, now cleanup and refactor
jcanton Oct 25, 2024
c5d2ca3
cleaned
jcanton Oct 25, 2024
a7d94f3
refactor
jcanton Oct 25, 2024
729e218
pre-commit
jcanton Oct 25, 2024
1020de3
more cleanup
jcanton Oct 25, 2024
3cced34
now init_vert_coord runs, wait for balfrin to be back online for seri…
jcanton Oct 25, 2024
06b52a2
2D nabla2 version
jcanton Nov 8, 2024
6a87418
add topo_c and topo_smt_c to serialbox and write the corresponding te…
jcanton Nov 8, 2024
1296dfb
test_init_vert_coord with serialized data working
jcanton Nov 8, 2024
ccd19a8
merge main
jcanton Nov 8, 2024
c4aa67c
these are not necessary anymore
jcanton Nov 8, 2024
423a74f
move topo data to its own savepoint
jcanton Nov 12, 2024
16af8d2
test_topography passes with serialized data
jcanton Nov 12, 2024
3a27f32
change savepoint in test
jcanton Nov 12, 2024
5a5824c
fix +1 bug
jcanton Nov 12, 2024
d82b858
Merge branch 'main' into port_z_ifc
jcanton Nov 12, 2024
51cb72e
pre-commit
jcanton Nov 12, 2024
1e2963a
move out of k loop
jcanton Nov 13, 2024
397219c
fix dims import
jcanton Nov 13, 2024
4546333
renaming and refactoring
jcanton Nov 13, 2024
d5b5e41
missing from previous
jcanton Nov 13, 2024
0c56879
xp not np
jcanton Nov 13, 2024
b2d3ae2
wip on artificial_topography testing
jcanton Nov 13, 2024
1d0adb3
@nikki review
jcanton Nov 14, 2024
ee013bb
save these two before moving to a separate branch
jcanton Nov 14, 2024
3e05c6e
remove these two, they'll end up in a new branch
jcanton Nov 14, 2024
7dfaa2c
Merge branch 'main' into port_z_ifc
jcanton Nov 14, 2024
e4f21a0
respect conventions
jcanton Nov 14, 2024
0086d92
remove unnecessary class
jcanton Nov 14, 2024
6ae8737
use gt4py for the update too
jcanton Nov 14, 2024
7d5cf88
@halungge comments
jcanton Nov 14, 2024
bcc54ea
renaming (thanks @halungge), added one forgotten test and numpy refer…
jcanton Nov 14, 2024
e07a404
keep np in tests instead of xp
jcanton Nov 14, 2024
15d8839
renaming
jcanton Nov 14, 2024
dbbcc1e
pre-commit and more renaming
jcanton Nov 14, 2024
1104571
type annotations
jcanton Nov 14, 2024
8d05431
this was not so unnecessary after all? + fix tests
jcanton Nov 15, 2024
6c72c4d
pre-commit
jcanton Nov 15, 2024
7e7b478
Merge branch 'main' into port_z_ifc
jcanton Nov 18, 2024
8b2618d
refactor: replace ConstantFieldsState with ExternalParameters and upd…
jcanton Nov 18, 2024
e48d56f
pre-commit
jcanton Nov 18, 2024
855aec7
refactor: reorganize nabla2 computation and introduce new stencil fun…
jcanton Nov 19, 2024
73f6c82
Merge branch 'main' into port_z_ifc
jcanton Nov 19, 2024
95b61c2
pre-commit
jcanton Nov 19, 2024
fcd808a
move dycore stencils to sub packages stencils (#595)
halungge Nov 19, 2024
f421d22
Fix parallel diffusion tests with dace orchestration (#572)
DropD Nov 20, 2024
554ffd2
most of magdalena's changes
jcanton Nov 20, 2024
fffaea3
tests on cpu
jcanton Nov 20, 2024
f62c3c1
fix the path to the stencil tests in the spack pytest config (#598)
halungge Nov 20, 2024
0659559
add docs
jcanton Nov 20, 2024
4bf99b0
Merge branch 'main' into port_z_ifc
jcanton Nov 20, 2024
1f2f18f
cleanup
jcanton Nov 20, 2024
45c16e0
mega refactoring
jcanton Nov 20, 2024
7e7f174
move to StencilTest
jcanton Nov 20, 2024
f0f4d7b
cleanup
jcanton Nov 20, 2024
c2d1471
feat: enhance common model utils with classes for implementing double…
egparedes Nov 20, 2024
7d5ff3d
Merge branch 'main' into port_z_ifc
jcanton Nov 20, 2024
34cfe29
added aquaplanet to test_vertical tests
jcanton Nov 20, 2024
ed72109
one change from Magdalena
jcanton Dec 16, 2024
ed61a3f
merge main and update xp
jcanton Dec 16, 2024
7c56b1e
actually update xp
jcanton Dec 16, 2024
6cc170f
pre-commit
jcanton Dec 16, 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
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ def calculate_nabla2_for_w(
vertical_start: gtx.int32,
vertical_end: gtx.int32,
):
# TODO: replace this by common/math/stencils/compute_nabla2_on_cell_k
_calculate_nabla2_for_w(
w,
geofac_n2s,
Expand Down
4 changes: 4 additions & 0 deletions model/common/src/icon4py/model/common/exceptions.py
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,10 @@
# SPDX-License-Identifier: BSD-3-Clause


class InvalidComputationError(Exception):
pass


class InvalidConfigError(Exception):
pass

Expand Down
19 changes: 19 additions & 0 deletions model/common/src/icon4py/model/common/external_parameters.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
# ICON4Py - ICON inspired code in Python and GT4Py
#
# Copyright (c) 2022-2024, ETH Zurich and MeteoSwiss
# All rights reserved.
#
# Please, refer to the LICENSE file in the root directory.
# SPDX-License-Identifier: BSD-3-Clause

import dataclasses

from icon4py.model.common import field_type_aliases as fa


@dataclasses.dataclass
class ExternalParameters:
"""Dataclass containing external parameters."""

topo_c: fa.CellField[float]
topo_smt_c: fa.CellField[float]
86 changes: 86 additions & 0 deletions model/common/src/icon4py/model/common/grid/topography.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,86 @@
# ICON4Py - ICON inspired code in Python and GT4Py
#
# Copyright (c) 2022-2024, ETH Zurich and MeteoSwiss
# All rights reserved.
#
# Please, refer to the LICENSE file in the root directory.
# SPDX-License-Identifier: BSD-3-Clause

import gt4py.next as gtx

from icon4py.model.common import dimension as dims, field_type_aliases as fa, type_alias as ta
from icon4py.model.common.grid import icon as icon_grid
from icon4py.model.common.math.stencils.compute_nabla2_on_cell import compute_nabla2_on_cell


@gtx.field_operator
def _update_smoothed_topography(
smoothed_topography: fa.CellField[ta.wpfloat],
nabla2_topo: fa.CellField[ta.wpfloat],
cell_areas: fa.CellField[ta.wpfloat],
) -> fa.CellField[ta.wpfloat]:
"""
Updates the smoothed topography field inside the loop.
"""
return smoothed_topography + 0.125 * nabla2_topo * cell_areas


@gtx.program(grid_type=gtx.GridType.UNSTRUCTURED)
def update_smoothed_topography(
nabla2_topo: fa.CellField[ta.wpfloat],
cell_areas: fa.CellField[ta.wpfloat],
smoothed_topography: fa.CellField[ta.wpfloat],
horizontal_start: gtx.int32,
horizontal_end: gtx.int32,
):
_update_smoothed_topography(
smoothed_topography=smoothed_topography,
nabla2_topo=nabla2_topo,
cell_areas=cell_areas,
out=smoothed_topography,
domain={
dims.CellDim: (horizontal_start, horizontal_end),
},
)


def smooth_topography(
topography: fa.CellField[ta.wpfloat],
grid: icon_grid.IconGrid,
cell_areas: fa.CellField[ta.wpfloat],
geofac_n2s: gtx.Field[gtx.Dims[dims.CellDim, dims.C2E2CODim], ta.wpfloat],
backend,
num_iterations: int = 25,
) -> fa.CellField[ta.wpfloat]:
"""
Computes the smoothed (laplacian-filtered) topography needed by the SLEVE
coordinate.
"""

# Make sure that it is copied (this should be topography.copy() but this is not supported by GT4Py)
smoothed_topography = gtx.as_field((dims.CellDim,), topography.ndarray.copy())

nabla2_topo = gtx.zeros(domain={dims.CellDim: range(grid.num_cells)}, dtype=ta.wpfloat)

for _ in range(num_iterations):
compute_nabla2_on_cell.with_backend(backend)(
psi_c=smoothed_topography,
geofac_n2s=geofac_n2s,
nabla2_psi_c=nabla2_topo,
horizontal_start=0,
horizontal_end=grid.num_cells,
offset_provider={
"C2E2CO": grid.get_offset_provider("C2E2CO"),
},
)

update_smoothed_topography.with_backend(backend)(
nabla2_topo=nabla2_topo,
cell_areas=cell_areas,
smoothed_topography=smoothed_topography,
horizontal_start=0,
horizontal_end=grid.num_cells,
offset_provider={},
)

return smoothed_topography
254 changes: 253 additions & 1 deletion model/common/src/icon4py/model/common/grid/vertical.py
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,13 @@
import gt4py.next as gtx

import icon4py.model.common.states.metadata as data
from icon4py.model.common import dimension as dims, exceptions, field_type_aliases as fa
from icon4py.model.common import (
dimension as dims,
exceptions,
field_type_aliases as fa,
type_alias as ta,
)
from icon4py.model.common.grid import icon as icon_grid, topography as topo
from icon4py.model.common.settings import xp


Expand Down Expand Up @@ -97,6 +103,23 @@ class VerticalGridConfig:
#: file name containing vct_a and vct_b table
file_path: pathlib.Path = None

# Parameters for setting up the decay function of the topographic signal for
# SLEVE. Default values from mo_sleve_nml.
#: Decay scale for large-scale topography component
SLEVE_decay_scale_1: Final[float] = 4000.0
#: Decay scale for small-scale topography component
SLEVE_decay_scale_2: Final[float] = 2500.0
#: Exponent for decay function
SLEVE_decay_exponent: Final[float] = 1.2
#: minimum absolute layer thickness 1 for SLEVE coordinates
SLEVE_minimum_layer_thickness_1: Final[float] = 100.0
#: minimum absolute layer thickness 2 for SLEVE coordinates
SLEVE_minimum_layer_thickness_2: Final[float] = 500.0
#: minimum relative layer thickness for nominal thicknesses <= SLEVE_minimum_layer_thickness_1
SLEVE_minimum_relative_layer_thickness_1: Final[float] = 1.0 / 3.0
#: minimum relative layer thickness for a nominal thickness of SLEVE_minimum_layer_thickness_2
SLEVE_minimum_relative_layer_thickness_2: Final[float] = 0.5


@dataclasses.dataclass(frozen=True)
class VerticalGrid:
Expand Down Expand Up @@ -517,3 +540,232 @@ def get_vct_a_and_vct_b(vertical_config: VerticalGridConfig) -> tuple[fa.KField,
if vertical_config.file_path
else _compute_vct_a_and_vct_b(vertical_config)
)


def compute_SLEVE_coordinate_from_vcta_and_topography(
jcanton marked this conversation as resolved.
Show resolved Hide resolved
vct_a: xp.ndarray,
topography: xp.ndarray,
cell_areas: fa.CellField[ta.wpfloat],
geofac_n2s: gtx.Field[gtx.Dims[dims.CellDim, dims.C2E2CODim], ta.wpfloat],
grid: icon_grid.IconGrid,
vertical_config: VerticalGridConfig,
vertical_geometry: VerticalGrid,
backend,
) -> xp.ndarray:
"""
Compute the 3D vertical coordinate field using the SLEVE coordinate
https://doi.org/10.1175/1520-0493(2002)130%3C2459:ANTFVC%3E2.0.CO;2

This is the same as vct_a in the flat levels (above nflatlev).
Below it is vct_a corrected by smothed and decaying topography such that it
blends smothly into the surface layer at num_lev + 1 which is the
topography.
"""

def _decay_func(
jcanton marked this conversation as resolved.
Show resolved Hide resolved
vct_a: xp.ndarray, model_top_height: float, decay_scale: float, decay_exponent: float
) -> xp.ndarray:
jcanton marked this conversation as resolved.
Show resolved Hide resolved
return xp.sinh(
(model_top_height / decay_scale) ** decay_exponent
- (vct_a / decay_scale) ** decay_exponent
) / xp.sinh((model_top_height / decay_scale) ** decay_exponent)

smoothed_topography = topo.smooth_topography(
topography=topography,
grid=grid,
cell_areas=cell_areas,
geofac_n2s=geofac_n2s,
backend=backend,
).ndarray
topography = topography.ndarray

vertical_coordinate = xp.zeros((grid.num_cells, grid.num_levels + 1), dtype=ta.wpfloat)
vertical_coordinate[:, grid.num_levels] = topography

# Small-scale topography (i.e. full topo - smooth topo)
small_scale_topography = topography - smoothed_topography

k = range(vertical_geometry.nflatlev + 1)
vertical_coordinate[:, k] = vct_a[k]

k = range(vertical_geometry.nflatlev + 1, grid.num_levels)
# Scaling factors for large-scale and small-scale topography
z_fac1 = _decay_func(
vct_a[k],
vertical_config.model_top_height,
vertical_config.SLEVE_decay_scale_1,
vertical_config.SLEVE_decay_exponent,
)
z_fac2 = _decay_func(
vct_a[k],
vertical_config.model_top_height,
vertical_config.SLEVE_decay_scale_2,
vertical_config.SLEVE_decay_exponent,
)
vertical_coordinate[:, k] = (
vct_a[k][xp.newaxis, :]
+ smoothed_topography[:, xp.newaxis] * z_fac1
+ small_scale_topography[:, xp.newaxis] * z_fac2
)

jcanton marked this conversation as resolved.
Show resolved Hide resolved
jcanton marked this conversation as resolved.
Show resolved Hide resolved
return vertical_coordinate


def check_and_correct_layer_thickness(
jcanton marked this conversation as resolved.
Show resolved Hide resolved
vertical_coordinate: xp.ndarray,
vct_a: xp.ndarray,
vertical_config: VerticalGridConfig,
grid: icon_grid.IconGrid,
) -> xp.ndarray:
ktop_thicklimit = xp.asarray(grid.num_cells * [grid.num_levels], dtype=ta.wpfloat)
# Ensure that layer thicknesses are not too small; this would potentially
# cause instabilities in vertical advection
for k in reversed(range(grid.num_levels)):
delta_vct_a = vct_a[k] - vct_a[k + 1]
if delta_vct_a < vertical_config.SLEVE_minimum_layer_thickness_1:
# limit layer thickness to SLEVE_minimum_relative_layer_thickness_1 times its nominal value
minimum_layer_thickness = (
vertical_config.SLEVE_minimum_relative_layer_thickness_1 * delta_vct_a
)
elif delta_vct_a < vertical_config.SLEVE_minimum_layer_thickness_2:
# limitation factor changes from SLEVE_minimum_relative_layer_thickness_1 to SLEVE_minimum_relative_layer_thickness_2
layer_thickness_adjustment_factor = (
(vertical_config.SLEVE_minimum_layer_thickness_2 - delta_vct_a)
/ (
vertical_config.SLEVE_minimum_layer_thickness_2
- vertical_config.SLEVE_minimum_layer_thickness_1
)
) ** 2
minimum_layer_thickness = (
vertical_config.SLEVE_minimum_relative_layer_thickness_1
* layer_thickness_adjustment_factor
+ vertical_config.SLEVE_minimum_relative_layer_thickness_2
* (1.0 - layer_thickness_adjustment_factor)
) * delta_vct_a
else:
# limitation factor decreases again
minimum_layer_thickness = (
vertical_config.SLEVE_minimum_relative_layer_thickness_2
* vertical_config.SLEVE_minimum_layer_thickness_2
* (delta_vct_a / vertical_config.SLEVE_minimum_layer_thickness_2) ** (1.0 / 3.0)
)

minimum_layer_thickness = max(
minimum_layer_thickness, min(50, vertical_config.lowest_layer_thickness)
)

# Ensure that the layer thickness is not too small, if so fix it and
# save the layer index
cell_ids = xp.argwhere(
vertical_coordinate[:, k + 1] + minimum_layer_thickness > vertical_coordinate[:, k]
)
vertical_coordinate[cell_ids, k] = (
vertical_coordinate[cell_ids, k + 1] + minimum_layer_thickness
)
ktop_thicklimit[cell_ids] = k

# Smooth layer thickness ratios in the transition layer of columns where the
# thickness limiter has been active (exclude lowest and highest layers)
cell_ids = xp.argwhere((ktop_thicklimit <= grid.num_levels - 3) & (ktop_thicklimit >= 3))
if cell_ids.size > 0:
delta_z1 = (
vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids] + 1]
- vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids] + 2]
)
delta_z2 = (
vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids] - 3]
- vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids] - 2]
)
stretching_factor = (delta_z2 / delta_z1) ** 0.25
delta_z3 = (
vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids] - 2]
- vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids] + 1]
) / (stretching_factor * (1.0 + stretching_factor * (1.0 + stretching_factor)))
vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids]] = xp.maximum(
vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids]],
vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids] + 1]
+ delta_z3 * stretching_factor,
)
vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids] - 1] = xp.maximum(
vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids] - 1],
vertical_coordinate[cell_ids, ktop_thicklimit[cell_ids]]
+ delta_z3 * stretching_factor**2,
)

# Check if ktop_thicklimit is sufficiently far away from the model top
if not xp.all(ktop_thicklimit > 2):
if vertical_config.num_levels > 6:
raise exceptions.InvalidConfigError(
f"Model top is too low and num_levels, {vertical_config.num_levels}, > 6."
)
else:
log.warning(
f"Model top is too low. But num_levels, {vertical_config.num_levels}, <= 6. "
)

return vertical_coordinate


def check_flatness_of_flat_level(
jcanton marked this conversation as resolved.
Show resolved Hide resolved
vertical_coordinate: xp.ndarray,
vct_a: xp.ndarray,
vertical_geometry: VerticalGrid,
) -> None:
# Check if level nflatlev is still flat
if not xp.all(
vertical_coordinate[:, vertical_geometry.nflatlev - 1]
== vct_a[vertical_geometry.nflatlev - 1]
):
raise exceptions.InvalidComputationError("Level nflatlev is not flat")


def compute_vertical_coordinate(
vct_a: fa.KField[ta.wpfloat],
topography: fa.CellField[ta.wpfloat],
cell_areas: fa.CellField[ta.wpfloat],
geofac_n2s: gtx.Field[gtx.Dims[dims.CellDim, dims.C2E2CODim], ta.wpfloat],
grid: icon_grid.IconGrid,
vertical_geometry: VerticalGrid,
backend,
) -> fa.CellKField[ta.wpfloat]:
"""
Compute the (Cell, K) vertical coordinate field starting from the
"flat/uniform/aquaplanet" vertical coordinate.

Args:
vct_a: Vertical coordinate with flat topography.
topography: Topography field.
cell_areas: Cell areas field.
geofac_n2s: Coefficients for nabla2 computation.
grid: Grid object.
vertical_geometry: Vertical grid object.
backend: Backend to use for computations.

Returns:
The (Cell, K) vertical coordinate field.

Raises:
exceptions.InvalidComputationError: If level nflatlev is not flat.
exceptions.InvalidConfigError: If model top is too low and num_levels > 6.
"""

vertical_config = vertical_geometry.config
vct_a = vct_a.ndarray

vertical_coordinate = compute_SLEVE_coordinate_from_vcta_and_topography(
vct_a,
topography,
cell_areas,
geofac_n2s,
grid,
vertical_config,
vertical_geometry,
backend,
)
vertical_coordinate = check_and_correct_layer_thickness(
vertical_coordinate, vct_a, vertical_config, grid
)

check_flatness_of_flat_level(vertical_coordinate, vct_a, vertical_geometry)

return gtx.as_field((dims.CellDim, dims.KDim), vertical_coordinate)
Loading
Loading