Loading [MathJax]/extensions/tex2jax.js
PISM, A Parallel Ice Sheet Model 2.2.2-d6b3a29ca committed by Constantine Khrulev on 2025-03-28
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
diagnostics.cc
Go to the documentation of this file.
1// Copyright (C) 2010--2025 Constantine Khroulev
2//
3// This file is part of PISM.
4//
5// PISM is free software; you can redistribute it and/or modify it under the
6// terms of the GNU General Public License as published by the Free Software
7// Foundation; either version 3 of the License, or (at your option) any later
8// version.
9//
10// PISM is distributed in the hope that it will be useful, but WITHOUT ANY
11// WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
12// FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
13// details.
14//
15// You should have received a copy of the GNU General Public License
16// along with PISM; if not, write to the Free Software
17// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18
19#include <algorithm>
20#include <cassert>
21#include <memory>
22
23#include "pism/age/AgeModel.hh"
24#include "pism/energy/EnergyModel.hh"
25#include "pism/energy/utilities.hh"
26#include "pism/geometry/grounded_cell_fraction.hh"
27#include "pism/geometry/part_grid_threshold_thickness.hh"
28#include "pism/icemodel/IceModel.hh"
29#include "pism/rheology/FlowLaw.hh"
30#include "pism/stressbalance/SSB_Modifier.hh"
31#include "pism/stressbalance/ShallowStressBalance.hh"
32#include "pism/stressbalance/StressBalance.hh"
33#include "pism/util/Diagnostic.hh"
34#include "pism/util/EnthalpyConverter.hh"
35#include "pism/util/error_handling.hh"
36#include "pism/util/pism_utilities.hh"
37#include "pism/util/projection.hh"
38
39#if (Pism_USE_PROJ == 1)
40#include "pism/util/Proj.hh"
41#endif
42
43// Flux balance code
44namespace pism {
45namespace diagnostics {
46
48
49//! @brief Computes tendency_of_ice_amount, the ice amount rate of change.
50class TendencyOfIceAmount : public Diag<IceModel> {
51public:
53 : Diag<IceModel>(Model),
54 m_kind(kind),
55 m_last_amount(m_grid, "last_ice_amount"),
57
58 std::string name = "tendency_of_ice_amount", long_name = "rate of change of the ice amount",
59 internal_units = "kg m^-2 second^-1", external_units = "kg m^-2 year^-1";
60 if (kind == MASS) {
61 name = "tendency_of_ice_mass";
62 long_name = "rate of change of the ice mass";
63 internal_units = "kg second^-1";
64 external_units = "Gt year^-1";
65 }
66
67 // set metadata:
68 m_vars = { { m_sys, name } };
69 m_vars[0].long_name(long_name).units(internal_units).output_units(external_units);
70
71 auto large_number = to_internal(1e6);
72 m_vars[0]["valid_range"] = { -large_number, large_number };
73 m_vars[0]["_FillValue"] = { to_internal(m_fill_value) };
74 m_vars[0]["cell_methods"] = "time: mean";
75
77 .long_name("ice amount at the time of the last report of " + name)
78 .units(internal_units + " second");
79 }
80
81protected:
82 std::shared_ptr<array::Array> compute_impl() const {
83
84 auto result = std::make_shared<array::Scalar>(m_grid, "");
85 result->metadata() = m_vars[0];
86
87 if (m_interval_length > 0.0) {
88 double ice_density = m_config->get_number("constants.ice.density");
89
90 auto cell_area = m_grid->cell_area();
91
92 const auto &thickness = model->geometry().ice_thickness;
93 const auto &area_specific_volume = model->geometry().ice_area_specific_volume;
94
95 array::AccessScope list{ result.get(), &thickness, &area_specific_volume, &m_last_amount };
96
97 for (auto p = m_grid->points(); p; p.next()) {
98 const int i = p.i(), j = p.j();
99
100 // m * (kg / m^3) = kg / m^2
101 double amount = (thickness(i, j) + area_specific_volume(i, j)) * ice_density;
102
103 (*result)(i, j) = (amount - m_last_amount(i, j)) / m_interval_length;
104
105 if (m_kind == MASS) {
106 // kg / m^2 * m^2 = kg
107 (*result)(i, j) *= cell_area;
108 }
109 }
110 } else {
111 result->set(m_fill_value);
112 }
113
114 return result;
115 }
116
117 void reset_impl() {
118 m_interval_length = 0.0;
119
120 const array::Scalar &thickness = model->geometry().ice_thickness;
121 const array::Scalar &area_specific_volume = model->geometry().ice_area_specific_volume;
122
123 double ice_density = m_config->get_number("constants.ice.density");
124
125 array::AccessScope list{ &m_last_amount, &thickness, &area_specific_volume };
126
127 for (auto p = m_grid->points(); p; p.next()) {
128 const int i = p.i(), j = p.j();
129
130 // m * (kg / m^3) = kg / m^2
131 m_last_amount(i, j) = (thickness(i, j) + area_specific_volume(i, j)) * ice_density;
132 }
133 }
134
135 void update_impl(double dt) {
136 m_interval_length += dt;
137 }
138
142};
143
144//! @brief Computes tendency_of_ice_amount_due_to_flow, the rate of change of ice amount due to
145//! flow.
146/*! @brief Report rate of change of ice amount due to flow. */
148public:
150 : DiagAverageRate<IceModel>(Model,
151 kind == AMOUNT ? "tendency_of_ice_amount_due_to_flow" :
152 "tendency_of_ice_mass_due_to_flow",
154 m_kind(kind) {
155
156 std::string name = "tendency_of_ice_amount_due_to_flow",
157 long_name = "rate of change of ice amount due to flow",
158 accumulator_units = "kg m^-2", internal_units = "kg m^-2 second^-1",
159 external_units = "kg m^-2 year^-1";
160
161 if (kind == MASS) {
162 name = "tendency_of_ice_mass_due_to_flow";
163 long_name = "rate of change of ice mass due to flow";
164 accumulator_units = "kg";
165 internal_units = "kg second^-1";
166 external_units = "Gt year^-1";
167 }
168
169 m_factor = m_config->get_number("constants.ice.density");
170
171 m_accumulator.metadata().units(accumulator_units);
172
173 m_vars = { { m_sys, name } };
174 m_vars[0].long_name(long_name).units(internal_units).output_units(external_units);
175 m_vars[0]["cell_methods"] = "time: mean";
176 m_vars[0]["_FillValue"] = { to_internal(m_fill_value) };
177 m_vars[0]["comment"] = "positive flux corresponds to ice gain";
178 }
179
180protected:
181 void update_impl(double dt) {
182 const array::Scalar &dH = model->geometry_evolution().thickness_change_due_to_flow(),
183 &dV = model->geometry_evolution().area_specific_volume_change_due_to_flow();
184
185 auto cell_area = m_grid->cell_area();
186
187 array::AccessScope list{ &m_accumulator, &dH, &dV };
188
189 for (auto p = m_grid->points(); p; p.next()) {
190 const int i = p.i(), j = p.j();
191
192 double C = m_factor * (m_kind == AMOUNT ? 1.0 : cell_area);
193
194 m_accumulator(i, j) += C * (dH(i, j) + dV(i, j));
195 }
196
197 m_interval_length += dt;
198 }
200};
201
202/*! @brief Report surface mass balance flux, averaged over the reporting interval */
203class SurfaceFlux : public DiagAverageRate<IceModel> {
204public:
207 kind == AMOUNT ?
208 "tendency_of_ice_amount_due_to_surface_mass_flux" :
209 "tendency_of_ice_mass_due_to_surface_mass_flux",
211 m_kind(kind) {
212 m_factor = m_config->get_number("constants.ice.density");
213
214 auto ismip6 = m_config->get_flag("output.ISMIP6");
215
216 std::string name = ismip6 ? "acabf" : "tendency_of_ice_amount_due_to_surface_mass_flux",
217 accumulator_units = "kg m^-2",
218 long_name = "average surface mass flux over reporting interval",
219 standard_name = "land_ice_surface_specific_mass_balance_flux",
220 internal_units = "kg m^-2 s^-1", external_units = "kg m^-2 year^-1";
221 if (kind == MASS) {
222 name = "tendency_of_ice_mass_due_to_surface_mass_flux", accumulator_units = "kg",
223 long_name = "average surface mass flux over reporting interval", standard_name = "",
224 internal_units = "kg second^-1", external_units = "Gt year^-1";
225 }
226
227 m_accumulator.metadata()["units"] = accumulator_units;
228
229 m_vars = { { m_sys, name } };
230 m_vars[0]
231 .long_name(long_name)
232 .standard_name(standard_name)
233 .units(internal_units)
234 .output_units(external_units);
235 m_vars[0]["_FillValue"] = { to_internal(m_fill_value) };
236 m_vars[0]["cell_methods"] = "time: mean";
237 m_vars[0]["comment"] = "positive flux corresponds to ice gain";
238 }
239
240protected:
241 void update_impl(double dt) {
242 const array::Scalar &SMB = model->geometry_evolution().top_surface_mass_balance();
243
244 auto cell_area = m_grid->cell_area();
245
247
248 for (auto p = m_grid->points(); p; p.next()) {
249 const int i = p.i(), j = p.j();
250
251 double C = m_factor * (m_kind == AMOUNT ? 1.0 : cell_area);
252
253 m_accumulator(i, j) += C * SMB(i, j);
254 }
255
256 m_interval_length += dt;
257 }
259};
260
261/*! @brief Report basal mass balance flux, averaged over the reporting interval */
262class BasalFlux : public DiagAverageRate<IceModel> {
263public:
266 kind == AMOUNT ? "tendency_of_ice_amount_due_to_basal_mass_flux" :
267 "tendency_of_ice_mass_due_to_basal_mass_flux",
269 m_kind(kind) {
270 m_factor = m_config->get_number("constants.ice.density");
271
272 std::string name = "tendency_of_ice_amount_due_to_basal_mass_flux",
273 accumulator_units = "kg m^-2",
274 long_name = "average basal mass flux over reporting interval", standard_name,
275 internal_units = "kg m^-2 second^-1", external_units = "kg m^-2 year^-1";
276 if (kind == MASS) {
277 name = "tendency_of_ice_mass_due_to_basal_mass_flux", accumulator_units = "kg",
278 long_name = "average basal mass flux over reporting interval",
279 standard_name = "tendency_of_land_ice_mass_due_to_basal_mass_balance",
280 internal_units = "kg second^-1", external_units = "Gt year^-1";
281 }
282 m_accumulator.metadata()["units"] = accumulator_units;
283
284 m_vars = { { m_sys, name } };
285 m_vars[0]
286 .long_name(long_name)
287 .standard_name(standard_name)
288 .units(internal_units)
289 .output_units(external_units);
290 m_vars[0]["_FillValue"] = { to_internal(m_fill_value) };
291 m_vars[0]["cell_methods"] = "time: mean";
292 m_vars[0]["comment"] = "positive flux corresponds to ice gain";
293 }
294
295protected:
296 void update_impl(double dt) {
297 const array::Scalar &BMB = model->geometry_evolution().bottom_surface_mass_balance();
298
299 auto cell_area = m_grid->cell_area();
300
302
303 for (auto p = m_grid->points(); p; p.next()) {
304 const int i = p.i(), j = p.j();
305
306 double C = m_factor * (m_kind == AMOUNT ? 1.0 : cell_area);
307
308 m_accumulator(i, j) += C * BMB(i, j);
309 }
310
311 m_interval_length += dt;
312 }
314};
315
316class ConservationErrorFlux : public DiagAverageRate<IceModel> {
317public:
320 kind == AMOUNT ?
321 "tendency_of_ice_amount_due_to_conservation_error" :
322 "tendency_of_ice_mass_due_to_conservation_error",
324 m_kind(kind) {
325 m_factor = m_config->get_number("constants.ice.density");
326
327 std::string name = "tendency_of_ice_amount_due_to_conservation_error",
328 accumulator_units = "kg m^-2",
329 long_name = "average mass conservation error flux over reporting interval",
330 internal_units = "kg m^-2 second^-1", external_units = "kg m^-2 year^-1";
331 if (kind == MASS) {
332 name = "tendency_of_ice_mass_due_to_conservation_error", accumulator_units = "kg",
333 long_name = "average mass conservation error flux over reporting interval",
334 internal_units = "kg second^-1", external_units = "Gt year^-1";
335 }
336
337 m_accumulator.metadata()["units"] = accumulator_units;
338
339 m_vars = { { m_sys, name } };
340 m_vars[0]
341 .long_name(long_name)
342 .units(internal_units)
343 .output_units(external_units);
344 m_vars[0]["_FillValue"] = {to_internal(m_fill_value)};
345 m_vars[0]["cell_methods"] = "time: mean";
346 m_vars[0]["comment"] = "positive flux corresponds to ice gain";
347 }
348
349protected:
350 void update_impl(double dt) {
351 const array::Scalar
352 &error = model->geometry_evolution().conservation_error();
353
354 array::AccessScope list{&m_accumulator, &error};
355
356 auto cell_area = m_grid->cell_area();
357
358 for (auto p = m_grid->points(); p; p.next()) {
359 const int i = p.i(), j = p.j();
360
361 double C = m_factor * (m_kind == AMOUNT ? 1.0 : cell_area);
362
363 m_accumulator(i, j) += C * error(i, j);
364 }
365
366 m_interval_length += dt;
367 }
369};
370
372
373static void accumulate_changes(const IceModel *model, double factor, ChangeKind kind,
374 array::Scalar &accumulator) {
375
376 const auto &calving = model->calving();
377 const auto &frontal_melt = model->frontal_melt();
378 const auto &forced_retreat = model->forced_retreat();
379
380 auto grid = accumulator.grid();
381
382 bool add_calving = (kind == CALVING or kind == TOTAL_DISCHARGE);
383 bool add_frontal_melt = (kind == FRONTAL_MELT or kind == TOTAL_DISCHARGE);
384 bool add_forced_retreat = (kind == FORCED_RETREAT or kind == TOTAL_DISCHARGE);
385
386 array::AccessScope scope{ &accumulator };
387 if (add_calving) {
388 scope.add(calving);
389 }
390 if (add_frontal_melt) {
391 scope.add(frontal_melt);
392 }
393 if (add_forced_retreat) {
394 scope.add(forced_retreat);
395 }
396
397 for (auto p = grid->points(); p; p.next()) {
398 const int i = p.i(), j = p.j();
399
400 if (add_calving) {
401 accumulator(i, j) += factor * calving(i, j);
402 }
403 if (add_frontal_melt) {
404 accumulator(i, j) += factor * frontal_melt(i, j);
405 }
406 if (add_forced_retreat) {
407 accumulator(i, j) += factor * forced_retreat(i, j);
408 }
409 }
410}
411
412
413/*! @brief Report discharge (calving and frontal melt) flux. */
414class DischargeFlux : public DiagAverageRate<IceModel> {
415public:
418 kind == AMOUNT ? "tendency_of_ice_amount_due_to_discharge" :
419 "tendency_of_ice_mass_due_to_discharge",
421 m_kind(kind) {
422
423 m_factor = m_config->get_number("constants.ice.density");
424
425 auto ismip6 = m_config->get_flag("output.ISMIP6");
426
427 std::string name = ismip6 ? "lifmassbf" : "tendency_of_ice_amount_due_to_discharge",
428 long_name = "discharge flux (calving, frontal melt, forced retreat)",
429 accumulator_units = "kg m^-2",
430 standard_name = "land_ice_specific_mass_flux_due_to_calving_and_ice_front_melting",
431 internal_units = "kg m^-2 s^-1", external_units = "kg m^-2 year^-1";
432 if (kind == MASS) {
433 name = "tendency_of_ice_mass_due_to_discharge";
434 long_name = "discharge flux (calving, frontal melt, forced retreat)";
435 accumulator_units = "kg";
436 standard_name = "";
437 internal_units = "kg second^-1";
438 external_units = "Gt year^-1";
439 }
440
441 m_accumulator.metadata()["units"] = accumulator_units;
442
443 m_vars = { { m_sys, name } };
444 m_vars[0]
445 .long_name(long_name)
446 .standard_name(standard_name)
447 .units(internal_units)
448 .output_units(external_units);
449 m_vars[0]["cell_methods"] = "time: mean";
450 m_vars[0]["_FillValue"] = { to_internal(m_fill_value) };
451 m_vars[0]["comment"] = "positive flux corresponds to ice gain";
452 }
453
454protected:
455 void update_impl(double dt) {
457 m_factor * (m_kind == AMOUNT ? 1.0 : m_grid->cell_area()),
460
461 m_interval_length += dt;
462 }
464};
465
466/*! @brief Report the calving flux. */
467class CalvingFlux : public DiagAverageRate<IceModel>
468{
469public:
472 kind == AMOUNT
473 ? "tendency_of_ice_amount_due_to_calving"
474 : "tendency_of_ice_mass_due_to_calving",
476 m_kind(kind) {
477
478 m_factor = m_config->get_number("constants.ice.density");
479
480 auto ismip6 = m_config->get_flag("output.ISMIP6");
481
482 std::string
483 name = ismip6 ? "licalvf" : "tendency_of_ice_amount_due_to_calving",
484 long_name = "calving flux",
485 accumulator_units = "kg m^-2",
486 standard_name = "land_ice_specific_mass_flux_due_to_calving",
487 internal_units = "kg m^-2 s^-1",
488 external_units = "kg m^-2 year^-1";
489 if (kind == MASS) {
490 name = "tendency_of_ice_mass_due_to_calving";
491 long_name = "calving flux";
492 accumulator_units = "kg";
493 standard_name = "";
494 internal_units = "kg second^-1";
495 external_units = "Gt year^-1";
496 }
497
498 m_accumulator.metadata().units(accumulator_units);
499
500 m_vars = { { m_sys, name } };
501 m_vars[0]
502 .long_name(long_name)
503 .standard_name(standard_name)
504 .units(internal_units)
505 .output_units(external_units);
506 m_vars[0]["cell_methods"] = "time: mean";
507
508 m_vars[0]["_FillValue"] = {to_internal(m_fill_value)};
509 m_vars[0]["comment"] = "positive flux corresponds to ice gain";
510 }
511
512protected:
513 void update_impl(double dt) {
514
516 m_factor * (m_kind == AMOUNT ? 1.0 : m_grid->cell_area()),
517 CALVING,
519
520 m_interval_length += dt;
521 }
523};
524
525/*! @brief Report the frontal melt flux. */
526class FrontalMeltFlux : public DiagAverageRate<IceModel>
527{
528public:
531 kind == AMOUNT
532 ? "tendency_of_ice_amount_due_to_frontal_melt"
533 : "tendency_of_ice_mass_due_to_frontal_melt",
535 m_kind(kind) {
536
537 m_factor = m_config->get_number("constants.ice.density");
538
539 std::string name = "tendency_of_ice_amount_due_to_frontal_melt", accumulator_units = "kg m^-2",
540 internal_units = "kg m^-2 s^-1", external_units = "kg m^-2 year^-1";
541 if (kind == MASS) {
542 name = "tendency_of_ice_mass_due_to_frontal_melt";
543 accumulator_units = "kg";
544 internal_units = "kg second^-1";
545 external_units = "Gt year^-1";
546 }
547
548 m_accumulator.metadata().units(accumulator_units);
549
550 m_vars = { { m_sys, name } };
551 m_vars[0].long_name("frontal melt flux").units(internal_units).output_units(external_units);
552 m_vars[0]["cell_methods"] = "time: mean";
553 m_vars[0]["_FillValue"] = { to_internal(m_fill_value) };
554 m_vars[0]["comment"] = "positive flux corresponds to ice gain";
555 }
556
557protected:
558 void update_impl(double dt) {
559
561 m_factor * (m_kind == AMOUNT ? 1.0 : m_grid->cell_area()),
564
565 m_interval_length += dt;
566 }
568};
569
570/*! @brief Report the frontal melt flux. */
571class ForcedRetreatFlux : public DiagAverageRate<IceModel>
572{
573public:
576 kind == AMOUNT ? "tendency_of_ice_amount_due_to_forced_retreat" :
577 "tendency_of_ice_mass_due_to_forced_retreat",
579 m_kind(kind) {
580
581 m_factor = m_config->get_number("constants.ice.density");
582
583 std::string name = "tendency_of_ice_amount_due_to_forced_retreat", accumulator_units = "kg m^-2",
584 internal_units = "kg m^-2 s^-1", external_units = "kg m^-2 year^-1";
585 if (kind == MASS) {
586 name = "tendency_of_ice_mass_due_to_forced_retreat";
587 accumulator_units = "kg";
588 internal_units = "kg second^-1";
589 external_units = "Gt year^-1";
590 }
591
592 m_accumulator.metadata().units(accumulator_units);
593
594 m_vars = { { m_sys, name } };
595 m_vars[0]
596 .long_name("forced (prescribed) retreat flux")
597 .units(internal_units)
598 .output_units(external_units);
599 m_vars[0]["cell_methods"] = "time: mean";
600 m_vars[0]["_FillValue"] = { to_internal(m_fill_value) };
601 m_vars[0]["comment"] = "positive flux corresponds to ice gain";
602 }
603
604protected:
605 void update_impl(double dt) {
606
608 m_factor * (m_kind == AMOUNT ? 1.0 : m_grid->cell_area()),
611
612 m_interval_length += dt;
613 }
615};
616
617} // end of namespace diagnostics
618} // end of namespace pism
619
620namespace pism {
621
622namespace details {
624
625static double ice_volume(const array::Scalar &ice_thickness,
626 const array::Array3D &ice_enthalpy,
627 IceKind kind,
628 double thickness_threshold) {
629
630 auto grid = ice_thickness.grid();
631 auto ctx = grid->ctx();
632 auto EC = ctx->enthalpy_converter();
633
634 auto cell_area = grid->cell_area();
635 const auto& z = grid->z();
636
637 double volume = 0.0;
638
639 // count the volume of a 3D grid cell if
640 //
641 // - it is temperate and we're asked for the temperate ice volume
642 // - it is cold and we're asked for the cold ice volume
643 //
644 // return zero otherwise
645 //
646 // uses the depth at the *bottom* of a cell to compute pressure
647 auto volume_counter = [EC, kind, cell_area](double z_min, double z_max, double H, double E) {
648 double depth = H - z_min;
649 double P = EC->pressure(depth);
650 double V = cell_area * (z_max - z_min);
651 bool temperate = EC->is_temperate_relaxed(E, P); // FIXME issue #15
652
653 switch (kind) {
654 case ICE_TEMPERATE:
655 return temperate ? V : 0.0;
656 default:
657 case ICE_COLD:
658 return (not temperate) ? V : 0.0;
659 }
660 };
661
662 array::AccessScope list{&ice_thickness, &ice_enthalpy};
663 ParallelSection loop(grid->com);
664 try {
665 for (auto p = grid->points(); p; p.next()) {
666 const int i = p.i(), j = p.j();
667
668 double H = ice_thickness(i, j);
669
670 if (H >= thickness_threshold) {
671 auto ks = grid->kBelowHeight(H);
672 const double *E = ice_enthalpy.get_column(i, j);
673
674 for (unsigned int k = 0; k < ks; ++k) {
675 volume += volume_counter(z[k], z[k + 1], H, E[k]);
676 }
677
678 volume += volume_counter(z[ks], H, H, E[ks]);
679 }
680 }
681 } catch (...) {
682 loop.failed();
683 }
684 loop.check();
685
686 return GlobalSum(grid->com, volume);
687}
688
689static double base_area(const array::Scalar &ice_thickness,
690 const array::Array3D &ice_enthalpy,
691 IceKind kind,
692 double thickness_threshold) {
693
694 auto grid = ice_thickness.grid();
695 auto ctx = grid->ctx();
696 auto EC = ctx->enthalpy_converter();
697
698 auto cell_area = grid->cell_area();
699
700 double area = 0.0;
701
702 array::AccessScope list{&ice_thickness, &ice_enthalpy};
703 ParallelSection loop(grid->com);
704 try {
705 for (auto p = grid->points(); p; p.next()) {
706 const int i = p.i(), j = p.j();
707
708 double thickness = ice_thickness(i, j);
709
710 if (thickness >= thickness_threshold) {
711 double basal_enthalpy = ice_enthalpy.get_column(i, j)[0];
712
713 bool temperate = EC->is_temperate_relaxed(basal_enthalpy,
714 EC->pressure(thickness)); // FIXME issue #15
715
716 switch (kind) {
717 case ICE_TEMPERATE:
718 area += temperate ? cell_area : 0.0;
719 break;
720 default:
721 case ICE_COLD:
722 area += (not temperate) ? cell_area : 0.0;
723 }
724 }
725 }
726 } catch (...) {
727 loop.failed();
728 }
729 loop.check();
730
731 return GlobalSum(grid->com, area);
732}
733
734} // end of namespace details
735
736// Horrendous names used by InitMIP (and ISMIP6, and CMIP5). Ugh.
737static const char* land_ice_area_fraction_name = "sftgif";
738static const char* grounded_ice_sheet_area_fraction_name = "sftgrf";
739static const char* floating_ice_sheet_area_fraction_name = "sftflf";
740
741namespace diagnostics {
742
744
746
747/*! @brief Ocean pressure difference at calving fronts. Used to debug CF boundary conditins. */
748class IceMarginPressureDifference : public Diag<IceModel>
749{
750public:
752
753protected:
754 std::shared_ptr<array::Array> compute_impl() const;
755};
756
758
759 /* set metadata: */
760 m_vars = { { m_sys, "ice_margin_pressure_difference" } };
761 m_vars[0]["_FillValue"] = { m_fill_value };
762 m_vars[0]
763 .long_name(
764 "vertically-averaged pressure difference at ice margins (including calving fronts)")
765 .units("Pa");
766}
767
768std::shared_ptr<array::Array> IceMarginPressureDifference::compute_impl() const {
769
770 auto result = allocate<array::Scalar>("ice_margin_pressure_difference");
771
772 array::CellType1 mask(m_grid, "mask");
773
774 const auto &H = model->geometry().ice_thickness;
775 const auto &bed = model->geometry().bed_elevation;
776 const auto &sea_level = model->geometry().sea_level_elevation;
777
778 {
779 const double H_threshold = m_config->get_number("stress_balance.ice_free_thickness_standard");
781 gc.set_icefree_thickness(H_threshold);
782
783 gc.compute_mask(sea_level, bed, H, mask);
784 }
785
786 const double rho_ice = m_config->get_number("constants.ice.density"),
787 rho_ocean = m_config->get_number("constants.sea_water.density"),
788 g = m_config->get_number("constants.standard_gravity");
789
790 array::AccessScope list{ &H, &bed, &mask, &sea_level, result.get() };
791
792 ParallelSection loop(m_grid->com);
793 try {
794 for (auto p = m_grid->points(); p; p.next()) {
795 const int i = p.i(), j = p.j();
796
797 double delta_p = 0.0;
798 if (mask.grounded_ice(i, j) and grid::domain_edge(*m_grid, i, j)) {
799 delta_p = 0.0;
800 } else if (mask.icy(i, j) and mask.next_to_ice_free_ocean(i, j)) {
801 double P_ice = 0.5 * rho_ice * g * H(i, j),
802 P_water = average_water_column_pressure(H(i, j), bed(i, j), sea_level(i, j), rho_ice,
803 rho_ocean, g);
804
805 delta_p = P_ice - P_water;
806 } else {
807 delta_p = m_fill_value;
808 }
809
810 (*result)(i, j) = delta_p;
811 }
812 } catch (...) {
813 loop.failed();
814 }
815 loop.check();
816
817
818 return result;
819}
820
821/*! @brief Report average basal mass balance flux over the reporting interval (grounded or floating
822 areas) */
823class BMBSplit : public DiagAverageRate<IceModel> {
824public:
825 BMBSplit(const IceModel *m, AreaType flag)
827 m, flag == GROUNDED ? "basal_mass_flux_grounded" : "basal_mass_flux_floating",
829 m_kind(flag) {
830 assert(flag != BOTH);
831
832 auto ismip6 = m_config->get_flag("output.ISMIP6");
833
834 std::string name, description, standard_name;
835 if (m_kind == GROUNDED) {
836 name = ismip6 ? "libmassbfgr" : "basal_mass_flux_grounded";
837 description = "average basal mass flux over the reporting interval (grounded areas)";
838 standard_name = ismip6 ? "land_ice_basal_specific_mass_balance_flux" : "";
839 } else {
840 name = ismip6 ? "libmassbffl" : "basal_mass_flux_floating";
841 description = "average basal mass flux over the reporting interval (floating areas)";
842 standard_name = ismip6 ? "land_ice_basal_specific_mass_balance_flux" : "";
843 }
844
845 m_accumulator.metadata()["units"] = "kg m^-2";
846
847 m_vars = { { m_sys, name } };
848 m_vars[0]
849 .long_name(description)
850 .standard_name(standard_name)
851 .units("kg m^-2 s^-1")
852 .output_units("kg m^-2 year^-1");
853 m_vars[0]["cell_methods"] = "time: mean";
854 m_vars[0]["_FillValue"] = { to_internal(m_fill_value) };
855 m_vars[0]["comment"] = "positive flux corresponds to ice gain";
856 }
857
858protected:
860 void update_impl(double dt) {
861 const array::Scalar &input = model->geometry_evolution().bottom_surface_mass_balance();
862 const auto &cell_type = model->geometry().cell_type;
863
864 double ice_density = m_config->get_number("constants.ice.density");
865
866 // the accumulator has the units of kg/m^2, computed as
867 //
868 // accumulator += BMB (m) * ice_density (kg / m^3)
869
870 array::AccessScope list{ &input, &cell_type, &m_accumulator };
871
872 for (auto p = m_grid->points(); p; p.next()) {
873 const int i = p.i(), j = p.j();
874
875 if (m_kind == GROUNDED and cell_type.grounded(i, j)) {
876 m_accumulator(i, j) += input(i, j) * ice_density;
877 } else if (m_kind == SHELF and cell_type.ocean(i, j)) {
878 m_accumulator(i, j) += input(i, j) * ice_density;
879 } else {
880 m_accumulator(i, j) = 0.0;
881 }
882 }
883
884 m_interval_length += dt;
885 }
886};
887
888//! \brief Computes vertically-averaged ice hardness.
889class HardnessAverage : public Diag<IceModel> {
890public:
891 HardnessAverage(const IceModel *m);
892
893protected:
894 virtual std::shared_ptr<array::Array> compute_impl() const;
895};
896
898
899 // set metadata:
900 m_vars = { { m_sys, "hardav" } };
901 m_vars[0]
902 .long_name("vertical average of ice hardness")
903 .set_units_without_validation(
904 "Pa s^(1/n)"); // n is the Glen exponent used by the SSA (shallow stress balance) flow law
905 m_vars[0]["valid_min"] = { 0.0 };
906 m_vars[0]["_FillValue"] = { m_fill_value };
907 m_vars[0]["comment"] = "units depend on the Glen exponent used by the flow law";
908}
909
910//! \brief Computes vertically-averaged ice hardness.
911std::shared_ptr<array::Array> HardnessAverage::compute_impl() const {
912
913 const rheology::FlowLaw *flow_law = model->stress_balance()->shallow()->flow_law().get();
914 if (flow_law == NULL) {
915 flow_law = model->stress_balance()->modifier()->flow_law().get();
916 if (flow_law == NULL) {
918 "Can't compute vertically-averaged hardness: no flow law is used.");
919 }
920 }
921
922 auto result = std::make_shared<array::Scalar>(m_grid, "hardav");
923 result->metadata() = m_vars[0];
924
925 const auto &cell_type = model->geometry().cell_type;
926
927 const array::Array3D &ice_enthalpy = model->energy_balance_model()->enthalpy();
928 const array::Scalar &ice_thickness = model->geometry().ice_thickness;
929
930 array::AccessScope list{ &cell_type, &ice_enthalpy, &ice_thickness, result.get() };
931 ParallelSection loop(m_grid->com);
932 try {
933 for (auto p = m_grid->points(); p; p.next()) {
934 const int i = p.i(), j = p.j();
935
936 const double *Eij = ice_enthalpy.get_column(i, j);
937 const double H = ice_thickness(i, j);
938 if (cell_type.icy(i, j)) {
939 (*result)(i, j) = rheology::averaged_hardness(*flow_law, H, m_grid->kBelowHeight(H),
940 m_grid->z().data(), Eij);
941 } else { // put negative value below valid range
942 (*result)(i, j) = m_fill_value;
943 }
944 }
945 } catch (...) {
946 loop.failed();
947 }
948 loop.check();
949
950 return result;
951}
952
953//! \brief Computes a diagnostic field filled with processor rank values.
954class Rank : public Diag<IceModel> {
955public:
956 Rank(const IceModel *m);
957
958protected:
959 virtual std::shared_ptr<array::Array> compute_impl() const;
960};
961
963 m_vars = { { m_sys, "rank" } };
964 m_vars[0]
965 .long_name("processor rank")
966 .units("1")
967 .set_time_independent(true)
968 .set_output_type(io::PISM_INT);
969}
970
971std::shared_ptr<array::Array> Rank::compute_impl() const {
972
973 auto result = std::make_shared<array::Scalar>(m_grid, "rank");
974 result->metadata() = m_vars[0];
975
976 array::AccessScope list{ result.get() };
977
978 for (auto p = m_grid->points(); p; p.next()) {
979 (*result)(p.i(), p.j()) = m_grid->rank();
980 }
981
982 return result;
983}
984
985//! \brief Computes CTS, CTS = E/E_s(p).
986class CTS : public Diag<IceModel> {
987public:
988 CTS(const IceModel *m);
989
990protected:
991 virtual std::shared_ptr<array::Array> compute_impl() const;
992};
993
995 m_vars = { { m_sys, "cts", m_grid->z() } };
996 m_vars[0]
997 .long_name("cts = E/E_s(p), so cold-temperate transition surface is at cts = 1")
998 .units("1");
999}
1000
1001std::shared_ptr<array::Array> CTS::compute_impl() const {
1002
1003 std::shared_ptr<array::Array3D> result(new array::Array3D(m_grid, "cts", array::WITHOUT_GHOSTS, m_grid->z()));
1004 result->metadata() = m_vars[0];
1005
1007 *result);
1008
1009 return result;
1010}
1011
1012//! \brief Computes ice temperature from enthalpy.
1013class Temperature : public Diag<IceModel> {
1014public:
1015 Temperature(const IceModel *m);
1016
1017protected:
1018 virtual std::shared_ptr<array::Array> compute_impl() const;
1019};
1020
1022 m_vars = { { m_sys, "temp", m_grid->z() } };
1023 m_vars[0]
1024 .long_name("ice temperature")
1025 .standard_name("land_ice_temperature")
1026 .units("kelvin");
1027 m_vars[0]["valid_min"] = { 0.0 };
1028}
1029
1030std::shared_ptr<array::Array> Temperature::compute_impl() const {
1031
1032 std::shared_ptr<array::Array3D> result(
1034 result->metadata() = m_vars[0];
1035
1036 const auto &thickness = model->geometry().ice_thickness;
1037 const auto &enthalpy = model->energy_balance_model()->enthalpy();
1038
1039 EnthalpyConverter::Ptr EC = model->ctx()->enthalpy_converter();
1040
1041 double *Tij;
1042 const double *Enthij; // columns of these values
1043
1044 array::AccessScope list{result.get(), &enthalpy, &thickness};
1045
1046 ParallelSection loop(m_grid->com);
1047 try {
1048 for (auto p = m_grid->points(); p; p.next()) {
1049 const int i = p.i(), j = p.j();
1050
1051 Tij = result->get_column(i,j);
1052 Enthij = enthalpy.get_column(i,j);
1053 for (unsigned int k=0; k <m_grid->Mz(); ++k) {
1054 const double depth = thickness(i,j) - m_grid->z(k);
1055 Tij[k] = EC->temperature(Enthij[k], EC->pressure(depth));
1056 }
1057 }
1058 } catch (...) {
1059 loop.failed();
1060 }
1061 loop.check();
1062
1063 return result;
1064}
1065
1066//! \brief Compute the pressure-adjusted temperature in degrees C corresponding
1067//! to ice temperature.
1068class TemperaturePA : public Diag<IceModel>
1069{
1070public:
1071 TemperaturePA(const IceModel *m);
1072protected:
1073 virtual std::shared_ptr<array::Array> compute_impl() const;
1074};
1075
1076
1078 : Diag<IceModel>(m) {
1079 m_vars = {{m_sys, "temp_pa", m_grid->z()}};
1080 m_vars[0]
1081 .long_name("pressure-adjusted ice temperature (degrees above pressure-melting point)")
1082 .units("deg_C");
1083 m_vars[0]["valid_max"] = {0};
1084}
1085
1086std::shared_ptr<array::Array> TemperaturePA::compute_impl() const {
1087 bool cold_mode = member(m_config->get_string("energy.model"), {"cold", "none"});
1088 double melting_point_temp = m_config->get_number("constants.fresh_water.melting_point_temperature");
1089
1090 auto result = std::make_shared<array::Array3D>(m_grid, "temp_pa", array::WITHOUT_GHOSTS, m_grid->z());
1091 result->metadata() = m_vars[0];
1092
1093 const auto &thickness = model->geometry().ice_thickness;
1094 const auto &enthalpy = model->energy_balance_model()->enthalpy();
1095
1096 EnthalpyConverter::Ptr EC = model->ctx()->enthalpy_converter();
1097
1098 double *Tij;
1099 const double *Enthij; // columns of these values
1100
1101 array::AccessScope list{result.get(), &enthalpy, &thickness};
1102
1103 ParallelSection loop(m_grid->com);
1104 try {
1105 for (auto pt = m_grid->points(); pt; pt.next()) {
1106 const int i = pt.i(), j = pt.j();
1107
1108 Tij = result->get_column(i,j);
1109 Enthij = enthalpy.get_column(i,j);
1110 for (unsigned int k=0; k < m_grid->Mz(); ++k) {
1111 const double depth = thickness(i,j) - m_grid->z(k),
1112 p = EC->pressure(depth);
1113 Tij[k] = EC->pressure_adjusted_temperature(Enthij[k], p);
1114
1115 if (cold_mode) { // if ice is temperate then its pressure-adjusted temp
1116 // is 273.15
1117 if (EC->is_temperate_relaxed(Enthij[k],p) && (thickness(i,j) > 0)) {
1118 Tij[k] = melting_point_temp;
1119 }
1120 }
1121
1122 }
1123 }
1124 } catch (...) {
1125 loop.failed();
1126 }
1127 loop.check();
1128
1129 result->shift(-melting_point_temp);
1130
1131 return result;
1132}
1133
1134//! \brief Computes basal values of the pressure-adjusted temperature.
1135class TemperaturePABasal : public Diag<IceModel>
1136{
1137public:
1138 TemperaturePABasal(const IceModel *m);
1139protected:
1140 virtual std::shared_ptr<array::Array> compute_impl() const;
1141};
1142
1144 : Diag<IceModel>(m) {
1145 m_vars = { { m_sys, "temppabase" } };
1146 m_vars[0].long_name("pressure-adjusted ice temperature at the base of ice").units("degree_Celsius");
1147}
1148
1149std::shared_ptr<array::Array> TemperaturePABasal::compute_impl() const {
1150
1151 bool cold_mode = member(m_config->get_string("energy.model"), {"cold", "none"});
1152 double melting_point_temp = m_config->get_number("constants.fresh_water.melting_point_temperature");
1153
1154 auto result = std::make_shared<array::Scalar>(m_grid, "temp_pa_base");
1155 result->metadata() = m_vars[0];
1156
1157 const auto &thickness = model->geometry().ice_thickness;
1158 const auto &enthalpy = model->energy_balance_model()->enthalpy();
1159
1160 EnthalpyConverter::Ptr EC = model->ctx()->enthalpy_converter();
1161
1162 array::AccessScope list{result.get(), &enthalpy, &thickness};
1163
1164 ParallelSection loop(m_grid->com);
1165 try {
1166 for (auto pt = m_grid->points(); pt; pt.next()) {
1167 const int i = pt.i(), j = pt.j();
1168
1169 const auto *Enthij = enthalpy.get_column(i,j);
1170
1171 const double depth = thickness(i,j),
1172 p = EC->pressure(depth);
1173 (*result)(i,j) = EC->pressure_adjusted_temperature(Enthij[0], p);
1174
1175 if (cold_mode) { // if ice is temperate then its pressure-adjusted temp
1176 // is 273.15
1177 if (EC->is_temperate_relaxed(Enthij[0],p) && (thickness(i,j) > 0)) {
1178 (*result)(i,j) = melting_point_temp;
1179 }
1180 }
1181 }
1182 } catch (...) {
1183 loop.failed();
1184 }
1185 loop.check();
1186
1187 result->shift(-melting_point_temp);
1188
1189 return result;
1190}
1191
1192//! \brief Computes surface values of ice enthalpy.
1193class IceEnthalpySurface : public Diag<IceModel>
1194{
1195public:
1196 IceEnthalpySurface(const IceModel *m);
1197protected:
1198 virtual std::shared_ptr<array::Array> compute_impl() const;
1199};
1200
1202 : Diag<IceModel>(m) {
1203 m_vars = { { m_sys, "enthalpysurf" } };
1204 m_vars[0].long_name("ice enthalpy at 1m below the ice surface").units("J kg^-1");
1205 m_vars[0]["_FillValue"] = {m_fill_value};
1206}
1207
1208std::shared_ptr<array::Array> IceEnthalpySurface::compute_impl() const {
1209
1210 auto result = std::make_shared<array::Scalar>(m_grid, "enthalpysurf");
1211 result->metadata() = m_vars[0];
1212
1213 // compute levels corresponding to 1 m below the ice surface:
1214
1215 const array::Array3D& ice_enthalpy = model->energy_balance_model()->enthalpy();
1216 const array::Scalar& ice_thickness = model->geometry().ice_thickness;
1217
1218 array::AccessScope list{&ice_thickness, result.get()};
1219
1220 for (auto p = m_grid->points(); p; p.next()) {
1221 const int i = p.i(), j = p.j();
1222
1223 (*result)(i,j) = std::max(ice_thickness(i,j) - 1.0, 0.0);
1224 }
1225
1226 extract_surface(ice_enthalpy, *result, *result); // slice at 1 m below the surface
1227
1228 for (auto p = m_grid->points(); p; p.next()) {
1229 const int i = p.i(), j = p.j();
1230
1231 if (ice_thickness(i,j) <= 1.0) {
1232 (*result)(i,j) = m_fill_value;
1233 }
1234 }
1235
1236 return result;
1237}
1238
1239//! \brief Computes enthalpy at the base of the ice.
1240class IceEnthalpyBasal : public Diag<IceModel>
1241{
1242public:
1243 IceEnthalpyBasal(const IceModel *m);
1244protected:
1245 virtual std::shared_ptr<array::Array> compute_impl() const;
1246};
1247
1249 : Diag<IceModel>(m) {
1250 m_vars = {{m_sys, "enthalpybase"}};
1251 m_vars[0].long_name("ice enthalpy at the base of ice").units("J kg^-1");
1252 m_vars[0]["_FillValue"] = {m_fill_value};
1253}
1254
1255std::shared_ptr<array::Array> IceEnthalpyBasal::compute_impl() const {
1256
1257 auto result = std::make_shared<array::Scalar>(m_grid, "enthalpybase");
1258 result->metadata() = m_vars[0];
1259
1260 extract_surface(model->energy_balance_model()->enthalpy(), 0.0, *result); // z=0 slice
1261
1262 apply_mask(model->geometry().ice_thickness, m_fill_value, *result);
1263
1264 return result;
1265}
1266
1267//! \brief Computes ice temperature at the base of the ice.
1268class TemperatureBasal : public Diag<IceModel>
1269{
1270public:
1271 TemperatureBasal(const IceModel *m, AreaType area_type);
1272private:
1273 std::shared_ptr<array::Array> compute_impl() const;
1274
1276};
1277
1279 : Diag<IceModel>(m), m_area_type(area_type) {
1280
1281 std::string name, long_name, standard_name;
1282 switch (area_type) {
1283 case GROUNDED:
1284 name = "litempbotgr";
1285 long_name = "ice temperature at the bottom surface of grounded ice";
1286 standard_name = "temperature_at_base_of_ice_sheet_model";
1287 break;
1288 case SHELF:
1289 name = "litempbotfl";
1290 long_name = "ice temperature at the bottom surface of floating ice";
1291 standard_name = "temperature_at_base_of_ice_sheet_model";
1292 break;
1293 case BOTH:
1294 name = "tempbase";
1295 long_name = "ice temperature at the base of ice";
1296 standard_name = "land_ice_basal_temperature";
1297 break;
1298 }
1299
1300 m_vars = { { m_sys, name } };
1301 m_vars[0].long_name(long_name).standard_name(standard_name).units("kelvin");
1302 m_vars[0]["_FillValue"] = { m_fill_value };
1303}
1304
1305std::shared_ptr<array::Array> TemperatureBasal::compute_impl() const {
1306
1307 auto result = allocate<array::Scalar>("basal_temperature");
1308
1309 const auto &thickness = model->geometry().ice_thickness;
1310
1311 EnthalpyConverter::Ptr EC = model->ctx()->enthalpy_converter();
1312
1313 extract_surface(model->energy_balance_model()->enthalpy(), 0.0, *result); // z=0 (basal) slice
1314 // Now result contains basal enthalpy.
1315
1316 const auto &cell_type = model->geometry().cell_type;
1317
1318 array::AccessScope list{ &cell_type, result.get(), &thickness };
1319
1320 ParallelSection loop(m_grid->com);
1321 try {
1322 for (auto p = m_grid->points(); p; p.next()) {
1323 const int i = p.i(), j = p.j();
1324
1325 double depth = thickness(i, j), pressure = EC->pressure(depth),
1326 T = EC->temperature((*result)(i, j), pressure);
1327
1328 if ((m_area_type == BOTH and cell_type.icy(i, j)) or
1329 (m_area_type == GROUNDED and cell_type.grounded_ice(i, j)) or
1330 (m_area_type == SHELF and cell_type.floating_ice(i, j))) {
1331 (*result)(i, j) = T;
1332 } else {
1333 (*result)(i, j) = m_fill_value;
1334 }
1335 }
1336 } catch (...) {
1337 loop.failed();
1338 }
1339 loop.check();
1340
1341 return result;
1342}
1343
1344//! \brief Computes ice temperature at the surface of the ice.
1345class TemperatureSurface : public Diag<IceModel> {
1346public:
1347 TemperatureSurface(const IceModel *m);
1348
1349protected:
1350 virtual std::shared_ptr<array::Array> compute_impl() const;
1351};
1352
1354 m_vars = { { m_sys, "tempsurf" } };
1355 m_vars[0]
1356 .long_name("ice temperature at 1m below the ice surface")
1357 .standard_name("temperature_at_ground_level_in_snow_or_firn") // InitMIP "standard" name
1358 .units("kelvin");
1359 m_vars[0]["_FillValue"] = { m_fill_value };
1360}
1361
1362std::shared_ptr<array::Array> TemperatureSurface::compute_impl() const {
1363
1364 const array::Scalar &thickness = model->geometry().ice_thickness;
1365
1366 auto enth = IceEnthalpySurface(model).compute();
1367 auto result = array::cast<array::Scalar>(enth);
1368
1369 EnthalpyConverter::Ptr EC = model->ctx()->enthalpy_converter();
1370
1371 // result contains surface enthalpy; note that it is allocated by
1372 // IceEnthalpySurface::compute().
1373
1374 array::AccessScope list{ result.get(), &thickness };
1375
1376 double depth = 1.0, pressure = EC->pressure(depth);
1377 ParallelSection loop(m_grid->com);
1378 try {
1379 for (auto p = m_grid->points(); p; p.next()) {
1380 const int i = p.i(), j = p.j();
1381
1382 if (thickness(i, j) > 1) {
1383 (*result)(i, j) = EC->temperature((*result)(i, j), pressure);
1384 } else {
1385 (*result)(i, j) = m_fill_value;
1386 }
1387 }
1388 } catch (...) {
1389 loop.failed();
1390 }
1391 loop.check();
1392
1393 result->metadata(0) = m_vars[0];
1394 return result;
1395}
1396
1397//! \brief Computes the liquid water fraction.
1398class LiquidFraction : public Diag<IceModel> {
1399public:
1400 LiquidFraction(const IceModel *m);
1401
1402protected:
1403 virtual std::shared_ptr<array::Array> compute_impl() const;
1404};
1405
1407 m_vars = { { m_sys, "liqfrac", m_grid->z() } };
1408 m_vars[0].long_name("liquid water fraction in ice (between 0 and 1)").units("1");
1409 m_vars[0]["valid_range"] = { 0.0, 1.0 };
1410}
1411
1412std::shared_ptr<array::Array> LiquidFraction::compute_impl() const {
1413
1414 std::shared_ptr<array::Array3D> result(
1415 new array::Array3D(m_grid, "liqfrac", array::WITHOUT_GHOSTS, m_grid->z()));
1416 result->metadata(0) = m_vars[0];
1417
1418 bool cold_mode = member(m_config->get_string("energy.model"), {"cold", "none"});
1419
1420 if (cold_mode) {
1421 result->set(0.0);
1422 } else {
1424 model->geometry().ice_thickness, *result);
1425 }
1426
1427 return result;
1428}
1429
1430//! \brief Computes the total thickness of temperate ice in a column.
1431class TemperateIceThickness : public Diag<IceModel> {
1432public:
1434
1435protected:
1436 virtual std::shared_ptr<array::Array> compute_impl() const;
1437};
1438
1440 m_vars = { { m_sys, "tempicethk" } };
1441 m_vars[0].long_name("temperate ice thickness (total column content)").units("m");
1442 m_vars[0]["_FillValue"] = { m_fill_value };
1443}
1444
1445std::shared_ptr<array::Array> TemperateIceThickness::compute_impl() const {
1446
1447 auto result = allocate<array::Scalar>("tempicethk");
1448
1449 const auto &cell_type = model->geometry().cell_type;
1450 const auto &ice_enthalpy = model->energy_balance_model()->enthalpy();
1451 const auto &ice_thickness = model->geometry().ice_thickness;
1452
1453 array::AccessScope list{ &cell_type, result.get(), &ice_enthalpy, &ice_thickness };
1454
1455 EnthalpyConverter::Ptr EC = model->ctx()->enthalpy_converter();
1456
1457 ParallelSection loop(m_grid->com);
1458 try {
1459 for (auto p = m_grid->points(); p; p.next()) {
1460 const int i = p.i(), j = p.j();
1461
1462 if (cell_type.icy(i, j)) {
1463 const double *Enth = ice_enthalpy.get_column(i, j);
1464 double H_temperate = 0.0;
1465 const double H = ice_thickness(i, j);
1466 const unsigned int ks = m_grid->kBelowHeight(H);
1467
1468 for (unsigned int k = 0; k < ks; ++k) { // FIXME issue #15
1469 double pressure = EC->pressure(H - m_grid->z(k));
1470
1471 if (EC->is_temperate_relaxed(Enth[k], pressure)) {
1472 H_temperate += m_grid->z(k + 1) - m_grid->z(k);
1473 }
1474 }
1475
1476 double pressure = EC->pressure(H - m_grid->z(ks));
1477 if (EC->is_temperate_relaxed(Enth[ks], pressure)) {
1478 H_temperate += H - m_grid->z(ks);
1479 }
1480
1481 (*result)(i, j) = H_temperate;
1482 } else {
1483 // ice-free
1484 (*result)(i, j) = m_fill_value;
1485 }
1486 }
1487 } catch (...) {
1488 loop.failed();
1489 }
1490 loop.check();
1491
1492 return result;
1493}
1494
1495//! \brief Computes the thickness of the basal layer of temperate ice.
1496class TemperateIceThicknessBasal : public Diag<IceModel> {
1497public:
1499
1500protected:
1501 virtual std::shared_ptr<array::Array> compute_impl() const;
1502};
1503
1505 m_vars = { { m_sys, "tempicethk_basal" } };
1506 m_vars[0].long_name("thickness of the basal layer of temperate ice").units("m");
1507 m_vars[0]["_FillValue"] = { m_fill_value };
1508}
1509
1510/*!
1511 * Uses linear interpolation to go beyond vertical grid resolution.
1512 */
1513std::shared_ptr<array::Array> TemperateIceThicknessBasal::compute_impl() const {
1514
1515 auto result = allocate<array::Scalar>("tempicethk_basal");
1516
1517 EnthalpyConverter::Ptr EC = model->ctx()->enthalpy_converter();
1518
1519 const auto &cell_type = model->geometry().cell_type;
1520 const auto &ice_enthalpy = model->energy_balance_model()->enthalpy();
1521 const auto &ice_thickness = model->geometry().ice_thickness;
1522
1523 array::AccessScope list{ &cell_type, result.get(), &ice_thickness, &ice_enthalpy };
1524
1525 ParallelSection loop(m_grid->com);
1526 try {
1527 for (auto p = m_grid->points(); p; p.next()) {
1528 const int i = p.i(), j = p.j();
1529
1530 double H = ice_thickness(i, j);
1531
1532 // if we have no ice, go on to the next grid point (this cell will be
1533 // marked as "missing" later)
1534 if (cell_type.ice_free(i, j)) {
1535 (*result)(i, j) = m_fill_value;
1536 continue;
1537 }
1538
1539 const double *Enth = ice_enthalpy.get_column(i, j);
1540
1541 unsigned int ks = m_grid->kBelowHeight(H);
1542
1543 unsigned int k = 0;
1544 double pressure = EC->pressure(H - m_grid->z(k));
1545 while (k <= ks) { // FIXME issue #15
1546 pressure = EC->pressure(H - m_grid->z(k));
1547
1548 if (EC->is_temperate_relaxed(Enth[k], pressure)) {
1549 k++;
1550 } else {
1551 break;
1552 }
1553 }
1554 // after this loop 'pressure' is equal to the pressure at the first level
1555 // that is cold
1556
1557 // no temperate ice at all; go to the next grid point
1558 if (k == 0) {
1559 (*result)(i, j) = 0.0;
1560 continue;
1561 }
1562
1563 // the whole column is temperate (except, possibly, some ice between
1564 // z(ks) and the total thickness; we ignore it)
1565 if (k == ks + 1) {
1566 (*result)(i, j) = m_grid->z(ks);
1567 continue;
1568 }
1569
1570 double pressure_0 = EC->pressure(H - m_grid->z(k - 1)), dz = m_grid->z(k) - m_grid->z(k - 1),
1571 slope1 = (Enth[k] - Enth[k - 1]) / dz,
1572 slope2 = (EC->enthalpy_cts(pressure) - EC->enthalpy_cts(pressure_0)) / dz;
1573
1574 if (slope1 != slope2) {
1575 (*result)(i, j) =
1576 m_grid->z(k - 1) + (EC->enthalpy_cts(pressure_0) - Enth[k - 1]) / (slope1 - slope2);
1577
1578 // check if the resulting thickness is valid:
1579 (*result)(i, j) = std::max((*result)(i, j), m_grid->z(k - 1));
1580 (*result)(i, j) = std::min((*result)(i, j), m_grid->z(k));
1581 } else {
1583 "Linear interpolation of the thickness of"
1584 " the basal temperate layer failed:\n"
1585 "(i=%d, j=%d, k=%d, ks=%d)\n",
1586 i, j, k, ks);
1587 }
1588 }
1589 } catch (...) {
1590 loop.failed();
1591 }
1592 loop.check();
1593
1594
1595 return result;
1596}
1597
1598namespace scalar {
1599
1600//! \brief Computes the total ice volume in glacierized areas.
1601class IceVolumeGlacierized : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1602public:
1604 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_volume_glacierized") {
1605
1606 set_units("m^3", "m^3");
1607 m_variable["long_name"] = "volume of the ice in glacierized areas";
1608 m_variable["valid_min"] = { 0.0 };
1609 }
1610 double compute() {
1611 return ice_volume(model->geometry(),
1612 m_config->get_number("output.ice_free_thickness_standard"));
1613 }
1614};
1615
1616//! \brief Computes the total ice volume.
1617class IceVolume : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1618public:
1620
1621 set_units("m^3", "m^3");
1622 m_variable["long_name"] = "volume of the ice, including seasonal cover";
1623 m_variable["valid_min"] = { 0.0 };
1624 }
1625
1626 double compute() {
1627 return ice_volume(model->geometry(), 0.0);
1628 }
1629};
1630
1631//! \brief Computes the total ice volume which is relevant for sea-level
1632class SeaLevelRisePotential : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1633public:
1635 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "sea_level_rise_potential") {
1636
1637 set_units("m", "m");
1638 m_variable["long_name"] = "the sea level rise that would result if all the ice were melted";
1639 m_variable["valid_min"] = { 0.0 };
1640 }
1641
1642 double compute() {
1644 m_config->get_number("output.ice_free_thickness_standard"));
1645 }
1646};
1647
1648//! \brief Computes the rate of change of the total ice volume in glacierized areas.
1649class IceVolumeRateOfChangeGlacierized : public TSDiag<TSRateDiagnostic, IceModel> {
1650public:
1652 : TSDiag<TSRateDiagnostic, IceModel>(m, "tendency_of_ice_volume_glacierized") {
1653
1654 set_units("m^3 s^-1", "m^3 year^-1");
1655 m_variable["long_name"] = "rate of change of the ice volume in glacierized areas";
1656 }
1657
1658 double compute() {
1659 return ice_volume(model->geometry(),
1660 m_config->get_number("output.ice_free_thickness_standard"));
1661 }
1662};
1663
1664//! \brief Computes the rate of change of the total ice volume.
1665class IceVolumeRateOfChange : public TSDiag<TSRateDiagnostic, IceModel> {
1666public:
1668 : TSDiag<TSRateDiagnostic, IceModel>(m, "tendency_of_ice_volume") {
1669
1670 set_units("m^3 s^-1", "m^3 year^-1");
1671 m_variable["long_name"] = "rate of change of the ice volume, including seasonal cover";
1672 }
1673
1674 double compute() {
1675 return ice_volume(model->geometry(), 0.0);
1676 }
1677};
1678
1679//! \brief Computes the total ice area.
1680class IceAreaGlacierized : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1681public:
1683 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_area_glacierized") {
1684
1685 set_units("m^2", "m^2");
1686 m_variable["long_name"] = "glacierized area";
1687 m_variable["valid_min"] = { 0.0 };
1688 }
1689
1690 double compute() {
1691 return ice_area(model->geometry(), m_config->get_number("output.ice_free_thickness_standard"));
1692 }
1693};
1694
1695//! \brief Computes the total mass of the ice not displacing sea water.
1696class IceMassNotDisplacingSeaWater : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1697public:
1699 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "limnsw") {
1700
1701 set_units("kg", "kg");
1702 m_variable["long_name"] = "mass of the ice not displacing sea water";
1703 m_variable["standard_name"] = "land_ice_mass_not_displacing_sea_water";
1704 m_variable["valid_min"] = { 0.0 };
1705 }
1706
1707 double compute() {
1708
1709 const double thickness_standard = m_config->get_number("output.ice_free_thickness_standard"),
1710 ice_density = m_config->get_number("constants.ice.density"),
1711 ice_volume =
1712 ice_volume_not_displacing_seawater(model->geometry(), thickness_standard),
1713 ice_mass = ice_volume * ice_density;
1714
1715 return ice_mass;
1716 }
1717};
1718
1719//! \brief Computes the total ice mass in glacierized areas.
1720class IceMassGlacierized : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1721public:
1723 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_mass_glacierized") {
1724
1725 set_units("kg", "kg");
1726 m_variable["long_name"] = "mass of the ice in glacierized areas";
1727 m_variable["valid_min"] = { 0.0 };
1728 }
1729
1730 double compute() {
1731 double ice_density = m_config->get_number("constants.ice.density"),
1732 thickness_standard = m_config->get_number("output.ice_free_thickness_standard");
1733 return ice_volume(model->geometry(), thickness_standard) * ice_density;
1734 }
1735};
1736
1737//! \brief Computes the total ice mass.
1738class IceMass : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1739public:
1741
1742 if (m_config->get_flag("output.ISMIP6")) {
1743 m_variable.set_name("lim");
1744 }
1745
1746 set_units("kg", "kg");
1747 m_variable["long_name"] = "mass of the ice, including seasonal cover";
1748 m_variable["standard_name"] = "land_ice_mass";
1749 m_variable["valid_min"] = { 0.0 };
1750 }
1751
1752 double compute() {
1753 return (ice_volume(model->geometry(), 0.0) * m_config->get_number("constants.ice.density"));
1754 }
1755};
1756
1757//! \brief Computes the rate of change of the total ice mass in glacierized areas.
1758class IceMassRateOfChangeGlacierized : public TSDiag<TSRateDiagnostic, IceModel> {
1759public:
1761 : TSDiag<TSRateDiagnostic, IceModel>(m, "tendency_of_ice_mass_glacierized") {
1762
1763 set_units("kg s^-1", "Gt year^-1");
1764 m_variable["long_name"] = "rate of change of the ice mass in glacierized areas";
1765 }
1766
1767 double compute() {
1768 double ice_density = m_config->get_number("constants.ice.density"),
1769 thickness_threshold = m_config->get_number("output.ice_free_thickness_standard");
1770 return ice_volume(model->geometry(), thickness_threshold) * ice_density;
1771 }
1772};
1773
1774//! \brief Computes the rate of change of the total ice mass due to flow (influx due to
1775//! prescribed constant-in-time ice thickness).
1776/*!
1777 * This is the change in mass resulting from prescribing (fixing) ice thickness.
1778 */
1779class IceMassRateOfChangeDueToFlow : public TSDiag<TSFluxDiagnostic, IceModel> {
1780public:
1782 : TSDiag<TSFluxDiagnostic, IceModel>(m, "tendency_of_ice_mass_due_to_flow") {
1783
1784 set_units("kg s^-1", "Gt year^-1");
1785 m_variable["long_name"] = "rate of change of the mass of ice due to flow"
1786 " (i.e. prescribed ice thickness)";
1787 }
1788
1789 double compute() {
1790
1791 const double ice_density = m_config->get_number("constants.ice.density");
1792
1795
1796 auto cell_area = m_grid->cell_area();
1797
1798 array::AccessScope list{ &dH, &dV };
1799
1800 double volume_change = 0.0;
1801 for (auto p = m_grid->points(); p; p.next()) {
1802 const int i = p.i(), j = p.j();
1803 // m * m^2 = m^3
1804 volume_change += (dH(i, j) + dV(i, j)) * cell_area;
1805 }
1806
1807 // (kg/m^3) * m^3 = kg
1808 return ice_density * GlobalSum(m_grid->com, volume_change);
1809 }
1810};
1811
1812//! \brief Computes the rate of change of the total ice mass.
1813class IceMassRateOfChange : public TSDiag<TSRateDiagnostic, IceModel> {
1814public:
1815 IceMassRateOfChange(IceModel *m) : TSDiag<TSRateDiagnostic, IceModel>(m, "tendency_of_ice_mass") {
1816
1817 set_units("kg s^-1", "Gt year^-1");
1818 m_variable["long_name"] = "rate of change of the mass of ice, including seasonal cover";
1819 }
1820
1821 double compute() {
1822 const double ice_density = m_config->get_number("constants.ice.density");
1823 return ice_volume(model->geometry(), 0.0) * ice_density;
1824 }
1825};
1826
1827
1828//! \brief Computes the total volume of the temperate ice in glacierized areas.
1829class IceVolumeGlacierizedTemperate : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1830public:
1832 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_volume_glacierized_temperate") {
1833
1834 set_units("m^3", "m^3");
1835 m_variable["long_name"] = "volume of temperate ice in glacierized areas";
1836 m_variable["valid_min"] = { 0.0 };
1837 }
1838
1839 double compute() {
1840 auto thickness_threshold = m_config->get_number("output.ice_free_thickness_standard");
1843 thickness_threshold);
1844 }
1845};
1846
1847//! \brief Computes the total volume of the temperate ice.
1848class IceVolumeTemperate : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1849public:
1851 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_volume_temperate") {
1852
1853 set_units("m^3", "m^3");
1854 m_variable["long_name"] = "volume of temperate ice, including seasonal cover";
1855 m_variable["valid_min"] = { 0.0 };
1856 }
1857
1863};
1864
1865//! \brief Computes the total volume of the cold ice in glacierized areas.
1866class IceVolumeGlacierizedCold : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1867public:
1869 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_volume_glacierized_cold") {
1870
1871 set_units("m^3", "m^3");
1872 m_variable["long_name"] = "volume of cold ice in glacierized areas";
1873 m_variable["valid_min"] = { 0.0 };
1874 }
1875
1876 double compute() {
1877 auto thickness_threshold = m_config->get_number("output.ice_free_thickness_standard");
1880 thickness_threshold);
1881 }
1882};
1883
1884//! \brief Computes the total volume of the cold ice.
1885class IceVolumeCold : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1886public:
1888
1889 set_units("m^3", "m^3");
1890 m_variable["long_name"] = "volume of cold ice, including seasonal cover";
1891 m_variable["valid_min"] = { 0.0 };
1892 }
1893
1899};
1900
1901//! \brief Computes the total area of the temperate ice.
1902class IceAreaGlacierizedTemperateBase : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1903public:
1905 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_area_glacierized_temperate_base") {
1906
1907 set_units("m^2", "m^2");
1908 m_variable["long_name"] = "glacierized area where basal ice is temperate";
1909 m_variable["valid_min"] = { 0.0 };
1910 }
1911
1912 double compute() {
1913 auto thickness_threshold = m_config->get_number("output.ice_free_thickness_standard");
1916 thickness_threshold);
1917 }
1918};
1919
1920//! \brief Computes the total area of the cold ice.
1921class IceAreaGlacierizedColdBase : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1922public:
1924 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_area_glacierized_cold_base") {
1925
1926 set_units("m^2", "m^2");
1927 m_variable["long_name"] = "glacierized area where basal ice is cold";
1928 m_variable["valid_min"] = { 0.0 };
1929 }
1930
1931 double compute() {
1932 auto thickness_threshold = m_config->get_number("output.ice_free_thickness_standard");
1935 thickness_threshold);
1936 }
1937};
1938
1939//! \brief Computes the total ice enthalpy in glacierized areas.
1940class IceEnthalpyGlacierized : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1941public:
1943 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_enthalpy_glacierized") {
1944
1945 set_units("J", "J");
1946 m_variable["long_name"] = "enthalpy of the ice in glacierized areas";
1947 m_variable["valid_min"] = { 0.0 };
1948 }
1949
1950 double compute() {
1951 return energy::total_ice_enthalpy(m_config->get_number("output.ice_free_thickness_standard"),
1954 }
1955};
1956
1957//! \brief Computes the total ice enthalpy.
1958class IceEnthalpy : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1959public:
1961
1962 set_units("J", "J");
1963 m_variable["long_name"] = "enthalpy of the ice, including seasonal cover";
1964 m_variable["valid_min"] = { 0.0 };
1965 }
1966
1971};
1972
1973//! \brief Computes the total grounded ice area.
1974class IceAreaGlacierizedGrounded : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1975public:
1977 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_area_glacierized_grounded") {
1978
1979 if (m_config->get_flag("output.ISMIP6")) {
1980 m_variable.set_name("iareagr");
1981 }
1982
1983 set_units("m^2", "m^2");
1984 m_variable["long_name"] = "area of grounded ice in glacierized areas";
1985 m_variable["standard_name"] = "grounded_ice_sheet_area";
1986 m_variable["valid_min"] = { 0.0 };
1987 }
1988
1989 double compute() {
1991 m_config->get_number("output.ice_free_thickness_standard"));
1992 }
1993};
1994
1995//! \brief Computes the total floating ice area.
1996class IceAreaGlacierizedShelf : public TSDiag<TSSnapshotDiagnostic, IceModel> {
1997public:
1999 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_area_glacierized_floating") {
2000
2001 if (m_config->get_flag("output.ISMIP6")) {
2002 m_variable.set_name("iareafl");
2003 }
2004
2005 set_units("m^2", "m^2");
2006 m_variable["long_name"] = "area of ice shelves in glacierized areas";
2007 m_variable["standard_name"] = "floating_ice_shelf_area";
2008 m_variable["valid_min"] = { 0.0 };
2009 }
2010
2011 double compute() {
2013 m_config->get_number("output.ice_free_thickness_standard"));
2014 }
2015};
2016
2017//! \brief Computes the total grounded ice volume.
2018class IceVolumeGlacierizedGrounded : public TSDiag<TSSnapshotDiagnostic, IceModel> {
2019public:
2021 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_volume_glacierized_grounded") {
2022
2023 set_units("m^3", "m^3");
2024 m_variable["long_name"] = "volume of grounded ice in glacierized areas";
2025 m_variable["valid_min"] = { 0.0 };
2026 }
2027
2028 double compute() {
2029 const auto &cell_type = model->geometry().cell_type;
2030
2031 const array::Scalar &ice_thickness = model->geometry().ice_thickness;
2032
2033 const double thickness_threshold = m_config->get_number("output.ice_free_thickness_standard"),
2034 cell_area = m_grid->cell_area();
2035
2036 array::AccessScope list{ &ice_thickness, &cell_type };
2037
2038 double volume = 0.0;
2039 for (auto p = m_grid->points(); p; p.next()) {
2040 const int i = p.i(), j = p.j();
2041
2042 const double H = ice_thickness(i, j);
2043
2044 if (cell_type.grounded(i, j) and H >= thickness_threshold) {
2045 volume += cell_area * H;
2046 }
2047 }
2048
2049 return GlobalSum(m_grid->com, volume);
2050 }
2051};
2052
2053//! \brief Computes the total floating ice volume.
2054class IceVolumeGlacierizedShelf : public TSDiag<TSSnapshotDiagnostic, IceModel> {
2055public:
2057 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "ice_volume_glacierized_floating") {
2058
2059 set_units("m^3", "m^3");
2060 m_variable["long_name"] = "volume of ice shelves in glacierized areas";
2061 m_variable["valid_min"] = { 0.0 };
2062 }
2063
2064 double compute() {
2065 const auto &cell_type = model->geometry().cell_type;
2066
2067 const array::Scalar &ice_thickness = model->geometry().ice_thickness;
2068
2069 const double thickness_threshold = m_config->get_number("output.ice_free_thickness_standard"),
2070 cell_area = m_grid->cell_area();
2071
2072 array::AccessScope list{ &ice_thickness, &cell_type };
2073
2074 double volume = 0.0;
2075 for (auto p = m_grid->points(); p; p.next()) {
2076 const int i = p.i(), j = p.j();
2077
2078 const double H = ice_thickness(i, j);
2079
2080 if (cell_type.ocean(i, j) and H >= thickness_threshold) {
2081 volume += cell_area * H;
2082 }
2083 }
2084
2085 return GlobalSum(m_grid->com, volume);
2086 }
2087};
2088
2089//! \brief Reports the mass continuity time step.
2090class TimeStepLength : public TSDiag<TSSnapshotDiagnostic, IceModel> {
2091public:
2093
2094 set_units("second", "year");
2095 m_variable["long_name"] = "mass continuity time step";
2096 m_variable["valid_min"] = { 0.0 };
2097 }
2098
2099 double compute() {
2100 return model->dt();
2101 }
2102};
2103
2104//! \brief Reports the mass continuity time step.
2105class TimeStepRatio : public TSDiag<TSSnapshotDiagnostic, IceModel> {
2106public:
2108
2109 set_units("1", "1");
2110 m_variable["long_name"] = "ratio of max. allowed time steps "
2111 "according to CFL and SIA diffusivity criteria";
2112 m_variable["valid_min"] = { 0.0 };
2113 m_variable["_FillValue"] = { -1.0 };
2114 }
2115
2116 double compute() {
2117
2118 const auto *stress_balance = model->stress_balance();
2119
2120 auto cfl_2d = stress_balance->max_timestep_cfl_2d();
2121 auto cfl_3d = stress_balance->max_timestep_cfl_3d();
2122
2123 auto dt_diff = max_timestep_diffusivity(stress_balance->max_diffusivity(),
2124 model->grid()->dx(),
2125 model->grid()->dy(),
2126 m_config->get_number("time_stepping.adaptive_ratio"));
2127
2128 auto dt_cfl = std::min(cfl_2d.dt_max, cfl_3d.dt_max);
2129
2130 if (dt_cfl.finite() and dt_diff.finite() and dt_diff.value() > 0.0) {
2131 return dt_cfl.value() / dt_diff.value();
2132 }
2133
2134 return -1.0;
2135 }
2136};
2137
2138//! \brief Reports maximum diffusivity.
2139class MaxDiffusivity : public TSDiag<TSSnapshotDiagnostic, IceModel> {
2140public:
2141 MaxDiffusivity(const IceModel *m) : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "max_diffusivity") {
2142
2143 set_units("m^2 s^-1", "m^2 s^-1");
2144 m_variable["long_name"] = "maximum diffusivity of the flow (usually from the SIA model)";
2145 m_variable["valid_min"] = { 0.0 };
2146 }
2147
2148 double compute() {
2150 }
2151};
2152
2153//! \brief Reports the maximum horizontal absolute velocity component over the grid.
2154/*!
2155 * This is the value used by the adaptive time-stepping code in the CFL condition
2156 * for horizontal advection (i.e. in energy and mass conservation time steps).
2157 *
2158 * This is not the maximum horizontal speed, but rather the maximum of components.
2159 *
2160 * Note that this picks up the value computed during the time-step taken at a
2161 * reporting time. (It is not the "average over the reporting interval computed using
2162 * differencing in time", as other rate-of-change diagnostics.)
2163 */
2164class MaxHorizontalVelocity : public TSDiag<TSSnapshotDiagnostic, IceModel> {
2165public:
2167 : TSDiag<TSSnapshotDiagnostic, IceModel>(m, "max_horizontal_vel") {
2168
2169 set_units("m second^-1", "m year^-1");
2170 m_variable["long_name"] = "max(max(abs(u)), max(abs(v))) for the horizontal velocity of ice"
2171 " over volume of the ice in last time step during time-series reporting interval";
2172 m_variable["valid_min"] = { 0.0 };
2173 }
2174
2175 double compute() {
2177
2178 return std::max(cfl.u_max, cfl.v_max);
2179 }
2180};
2181
2182/*!
2183 * Return total mass change due to one of the terms in the mass continuity equation.
2184 *
2185 * Possible terms are
2186 *
2187 * - SMB: surface mass balance
2188 * - BMB: basal mass balance
2189 * - FLOW: ice flow
2190 * - ERROR: numerical flux needed to preserve non-negativity of thickness
2191 *
2192 * This computation can be restricted to grounded and floating areas
2193 * using the `area` argument.
2194 *
2195 * - BOTH: include all contributions
2196 * - GROUNDED: include grounded areas only
2197 * - SHELF: include floating areas only
2198 *
2199 * When computing mass changes due to flow it is important to remember
2200 * that ice mass in a cell can be represented by its thickness *or* an
2201 * "area specific volume". Transferring mass from one representation
2202 * to the other does not change the mass in a cell. This explains the
2203 * special case used when `term == FLOW`. (Note that surface and basal
2204 * mass balances do not affect the area specific volume field.)
2205 */
2206double mass_change(const IceModel *model, TermType term, AreaType area) {
2207 const Grid &grid = *model->grid();
2208 const Config &config = *grid.ctx()->config();
2209
2210 const double ice_density = config.get_number("constants.ice.density"),
2211 cell_area = grid.cell_area();
2212
2213 const auto &cell_type = model->geometry().cell_type;
2214
2215 const array::Scalar *thickness_change = nullptr;
2216
2217 switch (term) {
2218 case FLOW:
2219 thickness_change = &model->geometry_evolution().thickness_change_due_to_flow();
2220 break;
2221 case SMB:
2222 thickness_change = &model->geometry_evolution().top_surface_mass_balance();
2223 break;
2224 case BMB:
2225 thickness_change = &model->geometry_evolution().bottom_surface_mass_balance();
2226 break;
2227 case ERROR:
2228 thickness_change = &model->geometry_evolution().conservation_error();
2229 break;
2230 default:
2231 // can't happen
2232 throw RuntimeError::formatted(PISM_ERROR_LOCATION, "invalid term type");
2233 }
2234
2235 const array::Scalar &dV_flow =
2237
2238 array::AccessScope list{ &cell_type, thickness_change };
2239
2240 if (term == FLOW) {
2241 list.add(dV_flow);
2242 }
2243
2244 double volume_change = 0.0;
2245 for (auto p = grid.points(); p; p.next()) {
2246 const int i = p.i(), j = p.j();
2247
2248 if ((area == BOTH) or (area == GROUNDED and cell_type.grounded(i, j)) or
2249 (area == SHELF and cell_type.ocean(i, j))) {
2250
2251 double dV = term == FLOW ? dV_flow(i, j) : 0.0;
2252
2253 // m^3 = m^2 * m
2254 volume_change += cell_area * ((*thickness_change)(i, j) + dV);
2255 }
2256 }
2257
2258 // (kg / m^3) * m^3 = kg
2259 return ice_density * GlobalSum(grid.com, volume_change);
2260}
2261
2262//! \brief Reports the total bottom surface ice flux.
2263class IceMassFluxBasal : public TSDiag<TSFluxDiagnostic, IceModel> {
2264public:
2266 : TSDiag<TSFluxDiagnostic, IceModel>(m, "tendency_of_ice_mass_due_to_basal_mass_flux") {
2267
2268 if (m_config->get_flag("output.ISMIP6")) {
2269 m_variable.set_name("tendlibmassbf");
2270 }
2271
2272 set_units("kg s^-1", "Gt year^-1");
2273 m_variable["long_name"] = "total over ice domain of bottom surface ice mass flux";
2274 m_variable["standard_name"] = "tendency_of_land_ice_mass_due_to_basal_mass_balance";
2275 m_variable["comment"] = "positive means ice gain";
2276 }
2277
2278 double compute() {
2279 return mass_change(model, BMB, BOTH);
2280 }
2281};
2282
2283//! \brief Reports the total top surface ice flux.
2284class IceMassFluxSurface : public TSDiag<TSFluxDiagnostic, IceModel> {
2285public:
2287 : TSDiag<TSFluxDiagnostic, IceModel>(m, "tendency_of_ice_mass_due_to_surface_mass_flux") {
2288
2289 if (m_config->get_flag("output.ISMIP6")) {
2290 m_variable.set_name("tendacabf");
2291 }
2292
2293 set_units("kg s^-1", "Gt year^-1");
2294 m_variable["long_name"] = "total over ice domain of top surface ice mass flux";
2295 m_variable["standard_name"] = "tendency_of_land_ice_mass_due_to_surface_mass_balance";
2296 m_variable["comment"] = "positive means ice gain";
2297 }
2298
2299 double compute() {
2300 return mass_change(model, SMB, BOTH);
2301 }
2302};
2303
2304//! \brief Reports the total basal ice flux over the grounded region.
2305class IceMassFluxBasalGrounded : public TSDiag<TSFluxDiagnostic, IceModel> {
2306public:
2308 : TSDiag<TSFluxDiagnostic, IceModel>(m, "basal_mass_flux_grounded") {
2309
2310 set_units("kg s^-1", "Gt year^-1");
2311 m_variable["long_name"] = "total over grounded ice domain of basal mass flux";
2312 m_variable["standard_name"] = "tendency_of_land_ice_mass_due_to_basal_mass_balance";
2313 m_variable["comment"] = "positive means ice gain";
2314 }
2315
2316 double compute() {
2317 return mass_change(model, BMB, GROUNDED);
2318 }
2319};
2320
2321//! \brief Reports the total sub-shelf ice flux.
2322class IceMassFluxBasalFloating : public TSDiag<TSFluxDiagnostic, IceModel> {
2323public:
2325 : TSDiag<TSFluxDiagnostic, IceModel>(m, "basal_mass_flux_floating") {
2326
2327 if (m_config->get_flag("output.ISMIP6")) {
2328 m_variable.set_name("tendlibmassbffl");
2329 }
2330
2331 set_units("kg s^-1", "Gt year^-1");
2332 m_variable["long_name"] = "total sub-shelf ice flux";
2333 m_variable["standard_name"] = "tendency_of_land_ice_mass_due_to_basal_mass_balance";
2334 m_variable["comment"] = "positive means ice gain";
2335 }
2336
2337 double compute() {
2338 return mass_change(model, BMB, SHELF);
2339 }
2340};
2341
2342//! \brief Reports the total numerical mass flux needed to preserve
2343//! non-negativity of ice thickness.
2344class IceMassFluxConservationError : public TSDiag<TSFluxDiagnostic, IceModel> {
2345public:
2347 : TSDiag<TSFluxDiagnostic, IceModel>(m, "tendency_of_ice_mass_due_to_conservation_error") {
2348
2349 set_units("kg s^-1", "Gt year^-1");
2350 m_variable["long_name"] = "total numerical flux needed to preserve non-negativity"
2351 " of ice thickness";
2352 m_variable["comment"] = "positive means ice gain";
2353 }
2354
2355 double compute() {
2356 return mass_change(model, ERROR, BOTH);
2357 }
2358};
2359
2360//! \brief Reports the total discharge flux.
2361class IceMassFluxDischarge : public TSDiag<TSFluxDiagnostic, IceModel> {
2362public:
2364 : TSDiag<TSFluxDiagnostic, IceModel>(m, "tendency_of_ice_mass_due_to_discharge") {
2365
2366 if (m_config->get_flag("output.ISMIP6")) {
2367 m_variable.set_name("tendlifmassbf");
2368 }
2369
2370 set_units("kg s^-1", "Gt year^-1");
2371 m_variable["long_name"] = "discharge flux (frontal melt, calving, forced retreat)";
2372 m_variable["standard_name"] = "tendency_of_land_ice_mass_due_to_calving_and_ice_front_melting";
2373 m_variable["comment"] = "positive means ice gain";
2374 }
2375
2376 double compute() {
2377 const double ice_density = m_config->get_number("constants.ice.density");
2378
2379 const array::Scalar &calving = model->calving();
2380 const array::Scalar &frontal_melt = model->frontal_melt();
2381 const array::Scalar &forced_retreat = model->forced_retreat();
2382
2383 auto cell_area = m_grid->cell_area();
2384
2385 double volume_change = 0.0;
2386
2387 array::AccessScope list{ &calving, &frontal_melt, &forced_retreat };
2388
2389 for (auto p = m_grid->points(); p; p.next()) {
2390 const int i = p.i(), j = p.j();
2391 // m^2 * m = m^3
2392 volume_change += cell_area * (calving(i, j) + frontal_melt(i, j) + forced_retreat(i, j));
2393 }
2394
2395 // (kg/m^3) * m^3 = kg
2396 return ice_density * GlobalSum(m_grid->com, volume_change);
2397 }
2398};
2399
2400//! \brief Reports the total calving flux.
2401class IceMassFluxCalving : public TSDiag<TSFluxDiagnostic, IceModel> {
2402public:
2404 : TSDiag<TSFluxDiagnostic, IceModel>(m, "tendency_of_ice_mass_due_to_calving") {
2405
2406 if (m_config->get_flag("output.ISMIP6")) {
2407 m_variable.set_name("tendlicalvf");
2408 }
2409
2410 set_units("kg s^-1", "Gt year^-1");
2411 m_variable["long_name"] = "calving flux";
2412 m_variable["standard_name"] = "tendency_of_land_ice_mass_due_to_calving";
2413 m_variable["comment"] = "positive means ice gain";
2414 }
2415
2416 double compute() {
2417 const double ice_density = m_config->get_number("constants.ice.density");
2418
2419 const array::Scalar &calving = model->calving();
2420
2421 auto cell_area = m_grid->cell_area();
2422
2423 double volume_change = 0.0;
2424
2425 array::AccessScope list{ &calving };
2426
2427 for (auto p = m_grid->points(); p; p.next()) {
2428 const int i = p.i(), j = p.j();
2429 // m^2 * m = m^3
2430 volume_change += cell_area * calving(i, j);
2431 }
2432
2433 // (kg/m^3) * m^3 = kg
2434 return ice_density * GlobalSum(m_grid->com, volume_change);
2435 }
2436};
2437
2438//! @brief Reports the total flux across the grounding line.
2439class IceMassFluxAtGroundingLine : public TSDiag<TSFluxDiagnostic, IceModel> {
2440public:
2442 : TSDiag<TSFluxDiagnostic, IceModel>(m, "grounding_line_flux") {
2443
2444 if (m_config->get_flag("output.ISMIP6")) {
2445 m_variable.set_name("tendligroundf");
2446 m_variable["standard_name"] = "tendency_of_grounded_ice_mass";
2447 }
2448
2449 set_units("kg s^-1", "Gt year^-1");
2450 m_variable["long_name"] = "total ice flux across the grounding line";
2451 m_variable["comment"] = "negative flux corresponds to ice loss into the ocean";
2452 }
2453
2458};
2459
2460} // end of namespace scalar
2461
2462
2463//! \brief Computes dHdt, the ice thickness rate of change.
2464class ThicknessRateOfChange : public Diag<IceModel> {
2465public:
2467 : Diag<IceModel>(m), m_last_thickness(m_grid, "last_ice_thickness"), m_interval_length(0.0) {
2468
2469 auto ismip6 = m_config->get_flag("output.ISMIP6");
2470
2471 // set metadata:
2472 m_vars = { { m_sys, ismip6 ? "dlithkdt" : "dHdt" } };
2473 m_vars[0]
2474 .long_name("ice thickness rate of change")
2475 .standard_name("tendency_of_land_ice_thickness")
2476 .units("m s^-1")
2477 .output_units("m year^-1");
2478
2479 auto large_number = to_internal(1e6);
2480
2481 m_vars[0]["valid_range"] = { -large_number, large_number };
2482 m_vars[0]["_FillValue"] = { to_internal(m_fill_value) };
2483 m_vars[0]["cell_methods"] = "time: mean";
2484
2486 .long_name(
2487 "ice thickness at the time of the last report of the rate of change of ice thickness")
2488 .units("m")
2489 .standard_name("land_ice_thickness");
2490 }
2491
2492protected:
2493 std::shared_ptr<array::Array> compute_impl() const {
2494
2495 auto result = std::make_shared<array::Scalar>(m_grid, "dHdt");
2496 result->metadata() = m_vars[0];
2497
2498 if (m_interval_length > 0.0) {
2499 model->geometry().ice_thickness.add(-1.0, m_last_thickness, *result);
2500 result->scale(1.0 / m_interval_length);
2501 } else {
2502 result->set(m_fill_value);
2503 }
2504
2505 return result;
2506 }
2507
2512
2513 void update_impl(double dt) {
2514 m_interval_length += dt;
2515 }
2516
2519};
2520
2521//! \brief Computes latitude and longitude bounds.
2522class LatLonBounds : public Diag<IceModel> {
2523public:
2524 LatLonBounds(const IceModel *m, const std::string &var_name, const std::string &proj_string);
2525
2526protected:
2527 virtual std::shared_ptr<array::Array> compute_impl() const;
2528
2530};
2531
2532LatLonBounds::LatLonBounds(const IceModel *m, const std::string &var_name,
2533 const std::string &proj_string)
2534 : Diag<IceModel>(m) {
2535 assert(var_name == "lat" || var_name == "lon");
2536 m_var_name = var_name;
2537
2538 // set metadata:
2539 m_vars = { { m_sys, m_var_name + "_bnds", { 0.0, 1.0, 2.0, 3.0 } } };
2540 m_vars[0].z().clear().set_name("nv4");
2541
2542 m_vars[0].set_time_independent(true);
2543 if (m_var_name == "lon") {
2544 m_vars[0].long_name("longitude bounds").units("degree_east");
2545 m_vars[0]["valid_range"] = { -180, 180 };
2546 } else {
2547 m_vars[0].long_name("latitude bounds").units("degree_north");
2548 m_vars[0]["valid_range"] = { -90, 90 };
2549 }
2550
2551 m_proj_string = proj_string;
2552
2553#if (Pism_USE_PROJ == 1)
2554 // create the transformation from the provided projection to lat,lon to check if
2555 // proj_string is valid.
2556 Proj crs(m_proj_string, "EPSG:4326");
2557#endif
2558 // If PISM_USE_PROJ is not 1 we don't need to check validity of m_proj_string: this diagnostic
2559 // will not be available and so this code will not run.
2560}
2561
2562std::shared_ptr<array::Array> LatLonBounds::compute_impl() const {
2563 std::shared_ptr<array::Array3D> result(new array::Array3D(
2564 m_grid, m_var_name + "_bnds", array::WITHOUT_GHOSTS, { 0.0, 1.0, 2.0, 3.0 }));
2565 result->metadata(0) = m_vars[0];
2566
2567 if (m_var_name == "lat") {
2569 } else {
2571 }
2572
2573 return result;
2574}
2575
2576class IceAreaFraction : public Diag<IceModel> {
2577public:
2578 IceAreaFraction(const IceModel *m);
2579
2580protected:
2581 virtual std::shared_ptr<array::Array> compute_impl() const;
2582};
2583
2586 m_vars[0]
2587 .long_name("fraction of a grid cell covered by ice (grounded or floating)")
2588 .standard_name("land_ice_area_fraction") // InitMIP "standard" name
2589 .units("1");
2590}
2591
2592std::shared_ptr<array::Array> IceAreaFraction::compute_impl() const {
2593
2594 auto result = allocate<array::Scalar>(land_ice_area_fraction_name);
2595
2596 const array::Scalar1 &thickness = model->geometry().ice_thickness,
2597 &surface_elevation = model->geometry().ice_surface_elevation,
2598 &bed_topography = model->geometry().bed_elevation;
2599
2600 const array::CellType1 &cell_type = model->geometry().cell_type;
2601
2602 array::AccessScope list{ &thickness, &surface_elevation, &bed_topography, &cell_type,
2603 result.get() };
2604
2605 const bool do_part_grid = m_config->get_flag("geometry.part_grid.enabled");
2607 ;
2608 if (do_part_grid) {
2609 list.add(Href);
2610 }
2611
2612 ParallelSection loop(m_grid->com);
2613 try {
2614 for (auto p = m_grid->points(); p; p.next()) {
2615 const int i = p.i(), j = p.j();
2616
2617 if (cell_type.icy(i, j)) {
2618 // an "icy" cell: the area fraction is one
2619 (*result)(i, j) = 1.0;
2620 } else if (cell_type.ice_free_ocean(i, j)) {
2621 // an ice-free ocean cell may be "partially-filled", in which case we need to compute its
2622 // ice area fraction by dividing Href by the threshold thickness.
2623
2624 double H_reference = do_part_grid ? Href(i, j) : 0.0;
2625
2626 if (H_reference > 0.0) {
2627 const double H_threshold =
2628 part_grid_threshold_thickness(cell_type.star_int(i, j), thickness.star(i, j),
2629 surface_elevation.star(i, j), bed_topography(i, j));
2630 // protect from a division by zero
2631 if (H_threshold > 0.0) {
2632 (*result)(i, j) = std::min(H_reference / H_threshold, 1.0);
2633 } else {
2634 (*result)(i, j) = 1.0;
2635 }
2636 } else {
2637 // H_reference is zero
2638 (*result)(i, j) = 0.0;
2639 }
2640 } else {
2641 // an ice-free-ground cell: the area fraction is zero
2642 (*result)(i, j) = 0.0;
2643 }
2644 } // end of the loop over grid points
2645 } catch (...) {
2646 loop.failed();
2647 }
2648 loop.check();
2649
2650 return result;
2651}
2652
2653class IceAreaFractionGrounded : public Diag<IceModel> {
2654public:
2656
2657protected:
2658 virtual std::shared_ptr<array::Array> compute_impl() const;
2659};
2660
2663 m_vars[0]
2664 .long_name("fraction of a grid cell covered by grounded ice")
2665 .standard_name("grounded_ice_sheet_area_fraction") // InitMIP "standard" name
2666 .units("1");
2667}
2668
2669std::shared_ptr<array::Array> IceAreaFractionGrounded::compute_impl() const {
2670 auto result = std::make_shared<array::Scalar>(m_grid, grounded_ice_sheet_area_fraction_name);
2671 result->metadata() = m_vars[0];
2672
2673 const double ice_density = m_config->get_number("constants.ice.density"),
2674 ocean_density = m_config->get_number("constants.sea_water.density");
2675
2676 const auto &ice_thickness = model->geometry().ice_thickness;
2677 const auto &sea_level = model->geometry().sea_level_elevation;
2678 const auto &bed_topography = model->geometry().bed_elevation;
2679
2680 const auto &cell_type = model->geometry().cell_type;
2681
2682 compute_grounded_cell_fraction(ice_density, ocean_density, sea_level, ice_thickness,
2683 bed_topography, *result);
2684
2685 // All grounded areas have the grounded cell fraction of one, so now we make sure that ice-free
2686 // areas get the value of 0 (they are grounded but not covered by a grounded ice sheet).
2687
2688 array::AccessScope list{ &cell_type, result.get() };
2689
2690 ParallelSection loop(m_grid->com);
2691 try {
2692 for (auto p = m_grid->points(); p; p.next()) {
2693 const int i = p.i(), j = p.j();
2694 if (cell_type.ice_free(i, j)) {
2695 (*result)(i, j) = 0.0;
2696 }
2697 }
2698 } catch (...) {
2699 loop.failed();
2700 }
2701 loop.check();
2702
2703 return result;
2704}
2705
2706class IceAreaFractionFloating : public Diag<IceModel> {
2707public:
2709
2710protected:
2711 virtual std::shared_ptr<array::Array> compute_impl() const;
2712};
2713
2716 m_vars[0]
2717 .long_name("fraction of a grid cell covered by floating ice")
2718 .standard_name("floating_ice_shelf_area_fraction")
2719 .units("1");
2720}
2721
2722std::shared_ptr<array::Array> IceAreaFractionFloating::compute_impl() const {
2723
2724 auto ice_area_fraction = IceAreaFraction(model).compute();
2726
2727 auto result = ice_area_fraction;
2728 result->metadata() = m_vars[0];
2729
2730 // Floating area fraction is total area fraction minus grounded area fraction.
2731 result->add(-1.0, *grounded_area_fraction);
2732
2733 return result;
2734}
2735
2736//! \brief Computes the 2D height above flotation.
2737class HeightAboveFloatation : public Diag<IceModel> {
2738public:
2740
2741protected:
2742 virtual std::shared_ptr<array::Array> compute_impl() const;
2743};
2744
2746
2747 // set metadata:
2748 m_vars = { { m_sys, "height_above_flotation" } };
2749 m_vars[0].long_name("ice thickness in excess of the maximum floating ice thickness").units("m");
2750 m_vars[0]["_FillValue"] = { m_fill_value };
2751 m_vars[0]["comment"] = "shows how close to floatation the ice is at a given location";
2752}
2753
2754std::shared_ptr<array::Array> HeightAboveFloatation::compute_impl() const {
2755
2756 auto result = allocate<array::Scalar>("height_above_flotation");
2757
2758 const auto &cell_type = model->geometry().cell_type;
2759
2760 const double ice_density = m_config->get_number("constants.ice.density"),
2761 ocean_density = m_config->get_number("constants.sea_water.density");
2762
2763 const auto &sea_level = model->geometry().sea_level_elevation;
2764 const auto &ice_thickness = model->geometry().ice_thickness;
2765 const auto &bed_topography = model->geometry().bed_elevation;
2766
2767 array::AccessScope list{ &cell_type, result.get(), &ice_thickness, &bed_topography, &sea_level };
2768
2769 ParallelSection loop(m_grid->com);
2770 try {
2771 for (auto p = m_grid->points(); p; p.next()) {
2772 const int i = p.i(), j = p.j();
2773
2774 const double thickness = ice_thickness(i, j), bed = bed_topography(i, j),
2775 ocean_depth = sea_level(i, j) - bed;
2776
2777 if (cell_type.icy(i, j) and ocean_depth > 0.0) {
2778 const double max_floating_thickness = ocean_depth * (ocean_density / ice_density);
2779 (*result)(i, j) = thickness - max_floating_thickness;
2780 } else {
2781 (*result)(i, j) = m_fill_value;
2782 }
2783 }
2784 } catch (...) {
2785 loop.failed();
2786 }
2787 loop.check();
2788
2789 return result;
2790}
2791
2792//! \brief Computes the mass per cell.
2793class IceMass : public Diag<IceModel> {
2794public:
2795 IceMass(const IceModel *m);
2796
2797protected:
2798 virtual std::shared_ptr<array::Array> compute_impl() const;
2799};
2800
2802 m_vars = { { m_sys, "ice_mass" } };
2803 m_vars[0].long_name("ice mass per cell").units("kg");
2804 m_vars[0]["_FillValue"] = { m_fill_value };
2805}
2806
2807std::shared_ptr<array::Array> IceMass::compute_impl() const {
2808
2809 auto result = allocate<array::Scalar>("ice_mass");
2810
2811 const auto &cell_type = model->geometry().cell_type;
2812
2813 const double ice_density = m_config->get_number("constants.ice.density");
2814
2815 const array::Scalar &ice_thickness = model->geometry().ice_thickness;
2816
2817 auto cell_area = m_grid->cell_area();
2818
2819 array::AccessScope list{ &cell_type, result.get(), &ice_thickness };
2820
2821 ParallelSection loop(m_grid->com);
2822 try {
2823 for (auto p = m_grid->points(); p; p.next()) {
2824 const int i = p.i(), j = p.j();
2825
2826 // count all ice, including cells which have so little they
2827 // are considered "ice-free"
2828 if (ice_thickness(i, j) > 0.0) {
2829 (*result)(i, j) = ice_density * ice_thickness(i, j) * cell_area;
2830 } else {
2831 (*result)(i, j) = m_fill_value;
2832 }
2833 } // end of loop over grid points
2834
2835 } catch (...) {
2836 loop.failed();
2837 }
2838 loop.check();
2839
2840 // Add the mass of ice in Href:
2841 if (m_config->get_flag("geometry.part_grid.enabled")) {
2843 list.add(Href);
2844 for (auto p = m_grid->points(); p; p.next()) {
2845 const int i = p.i(), j = p.j();
2846
2847 if (ice_thickness(i, j) <= 0.0 and Href(i, j) > 0.0) {
2848 (*result)(i, j) = ice_density * Href(i, j) * cell_area;
2849 }
2850 }
2851 }
2852
2853 return result;
2854}
2855
2856/*! @brief Sea-level adjusted bed topography (zero at sea level). */
2857class BedTopographySeaLevelAdjusted : public Diag<IceModel> {
2858public:
2860
2861protected:
2862 std::shared_ptr<array::Array> compute_impl() const;
2863};
2864
2866 : Diag<IceModel>(m) {
2867 m_vars = { { m_sys, "topg_sl_adjusted" } };
2868 m_vars[0].long_name("sea-level adjusted bed topography (zero is at sea level)").units("meters");
2869}
2870
2871std::shared_ptr<array::Array> BedTopographySeaLevelAdjusted::compute_impl() const {
2872
2873 auto result = allocate<array::Scalar>("topg_sl_adjusted");
2874
2875 const auto &bed = model->geometry().bed_elevation;
2876 const auto &sea_level = model->geometry().sea_level_elevation;
2877
2878 array::AccessScope list{ &bed, &sea_level, result.get() };
2879
2880 for (auto p = m_grid->points(); p; p.next()) {
2881 const int i = p.i(), j = p.j();
2882
2883 (*result)(i, j) = bed(i, j) - sea_level(i, j);
2884 }
2885
2886 return result;
2887}
2888
2889/*! @brief Ice hardness computed using the SIA flow law. */
2890class IceHardness : public Diag<IceModel> {
2891public:
2892 IceHardness(const IceModel *m);
2893
2894protected:
2895 std::shared_ptr<array::Array> compute_impl() const;
2896};
2897
2899 m_vars = { { m_sys, "hardness", m_grid->z() } };
2900 m_vars[0]
2901 .long_name("ice hardness computed using the SIA flow law")
2902 .set_units_without_validation(
2903 "Pa s^(1/n)"); // n is the Glen exponent used by the SIA (modifier) flow law
2904 m_vars[0]["comment"] = "units depend on the Glen exponent used by the flow law";
2905}
2906
2907std::shared_ptr<array::Array> IceHardness::compute_impl() const {
2908
2909 std::shared_ptr<array::Array3D> result(
2910 new array::Array3D(m_grid, "hardness", array::WITHOUT_GHOSTS, m_grid->z()));
2911 result->metadata(0) = m_vars[0];
2912
2913 EnthalpyConverter::Ptr EC = m_grid->ctx()->enthalpy_converter();
2914
2915 const array::Array3D &ice_enthalpy = model->energy_balance_model()->enthalpy();
2916 const array::Scalar &ice_thickness = model->geometry().ice_thickness;
2917
2918 const rheology::FlowLaw &flow_law = *model->stress_balance()->modifier()->flow_law();
2919
2920 array::AccessScope list{ &ice_enthalpy, &ice_thickness, result.get() };
2921
2922 const unsigned int Mz = m_grid->Mz();
2923
2924 ParallelSection loop(m_grid->com);
2925 try {
2926 for (auto p = m_grid->points(); p; p.next()) {
2927 const int i = p.i(), j = p.j();
2928 const double *E = ice_enthalpy.get_column(i, j);
2929 const double H = ice_thickness(i, j);
2930
2931 double *hardness = result->get_column(i, j);
2932
2933 for (unsigned int k = 0; k < Mz; ++k) {
2934 const double depth = H - m_grid->z(k);
2935
2936 // EC->pressure() handles negative depths correctly
2937 const double pressure = EC->pressure(depth);
2938
2939 hardness[k] = flow_law.hardness(E[k], pressure);
2940 }
2941 }
2942 } catch (...) {
2943 loop.failed();
2944 }
2945 loop.check();
2946
2947 return result;
2948}
2949
2950/*! @brief Effective viscosity of ice (3D). */
2951class IceViscosity : public Diag<IceModel> {
2952public:
2954
2955protected:
2956 std::shared_ptr<array::Array> compute_impl() const;
2957};
2958
2960 m_vars = { { m_sys, "effective_viscosity", m_grid->z() } };
2961 m_vars[0]
2962 .long_name("effective viscosity of ice")
2963 .units("Pascal second")
2964 .output_units("kPascal second");
2965 m_vars[0]["valid_min"] = { 0.0 };
2966 m_vars[0]["_FillValue"] = { m_fill_value };
2967}
2968
2969static inline double square(double x) {
2970 return x * x;
2971}
2972
2973std::shared_ptr<array::Array> IceViscosity::compute_impl() const {
2974
2975 std::shared_ptr<array::Array3D> result(
2976 new array::Array3D(m_grid, "effective_viscosity", array::WITHOUT_GHOSTS, m_grid->z()));
2977 result->metadata(0) = m_vars[0];
2978
2980
2981 using mask::ice_free;
2982
2983 EnthalpyConverter::Ptr EC = m_grid->ctx()->enthalpy_converter();
2984
2985 const rheology::FlowLaw &flow_law = *model->stress_balance()->modifier()->flow_law();
2986
2987 const array::Scalar &ice_thickness = model->geometry().ice_thickness;
2988
2989 const array::Array3D &ice_enthalpy = model->energy_balance_model()->enthalpy(),
2990 &U = model->stress_balance()->velocity_u(),
2991 &V = model->stress_balance()->velocity_v(),
2992 &W_without_ghosts = model->stress_balance()->velocity_w();
2993
2994 W.copy_from(W_without_ghosts);
2995
2996 const unsigned int Mz = m_grid->Mz();
2997 const double dx = m_grid->dx(), dy = m_grid->dy();
2998 const std::vector<double> &z = m_grid->z();
2999
3000 const array::CellType1 &mask = model->geometry().cell_type;
3001
3002 array::AccessScope list{ &U, &V, &W, &ice_enthalpy, &ice_thickness, &mask, result.get() };
3003
3004 ParallelSection loop(m_grid->com);
3005 try {
3006 for (auto p = m_grid->points(); p; p.next()) {
3007 const int i = p.i(), j = p.j();
3008
3009 const double *E = ice_enthalpy.get_column(i, j);
3010 const double H = ice_thickness(i, j);
3011
3012 const double *u = U.get_column(i, j), *u_n = U.get_column(i, j + 1),
3013 *u_e = U.get_column(i + 1, j), *u_s = U.get_column(i, j - 1),
3014 *u_w = U.get_column(i - 1, j);
3015
3016 const double *v = V.get_column(i, j), *v_n = V.get_column(i, j + 1),
3017 *v_e = V.get_column(i + 1, j), *v_s = V.get_column(i, j - 1),
3018 *v_w = V.get_column(i - 1, j);
3019
3020 const double *w = W.get_column(i, j), *w_n = W.get_column(i, j + 1),
3021 *w_e = W.get_column(i + 1, j), *w_s = W.get_column(i, j - 1),
3022 *w_w = W.get_column(i - 1, j);
3023
3024 auto m = mask.star_int(i, j);
3025 const unsigned int east = ice_free(m.e) ? 0 : 1, west = ice_free(m.w) ? 0 : 1,
3026 south = ice_free(m.s) ? 0 : 1, north = ice_free(m.n) ? 0 : 1;
3027
3028 double *viscosity = result->get_column(i, j);
3029
3030 if (ice_free(m.c)) {
3031 result->set_column(i, j, m_fill_value);
3032 continue;
3033 }
3034
3035 for (unsigned int k = 0; k < Mz; ++k) {
3036 const double depth = H - z[k];
3037
3038 if (depth < 0.0) {
3039 viscosity[k] = m_fill_value;
3040 continue;
3041 }
3042
3043 // EC->pressure() handles negative depths correctly
3044 const double pressure = EC->pressure(depth);
3045
3046 const double hardness = flow_law.hardness(E[k], pressure);
3047
3048 double u_x = 0.0, v_x = 0.0, w_x = 0.0;
3049 if (west + east > 0) {
3050 const double D = 1.0 / (dx * (west + east));
3051 u_x = D * (west * (u[k] - u_w[k]) + east * (u_e[k] - u[k]));
3052 v_x = D * (west * (v[k] - v_w[k]) + east * (v_e[k] - v[k]));
3053 w_x = D * (west * (w[k] - w_w[k]) + east * (w_e[k] - w[k]));
3054 }
3055
3056 double u_y = 0.0, v_y = 0.0, w_y = 0.0;
3057 if (south + north > 0) {
3058 const double D = 1.0 / (dy * (south + north));
3059 u_y = D * (south * (u[k] - u_s[k]) + north * (u_n[k] - u[k]));
3060 v_y = D * (south * (v[k] - v_s[k]) + north * (v_n[k] - v[k]));
3061 w_y = D * (south * (w[k] - w_s[k]) + north * (w_n[k] - w[k]));
3062 }
3063
3064 double u_z = 0.0, v_z = 0.0, w_z = 0.0;
3065
3066 if (k == 0) {
3067 const double dz = z[1] - z[0];
3068 u_z = (u[1] - u[0]) / dz;
3069 v_z = (v[1] - v[0]) / dz;
3070 w_z = (w[1] - w[0]) / dz;
3071 } else if (k == Mz - 1) {
3072 const double dz = z[Mz - 1] - z[Mz - 2];
3073 u_z = (u[Mz - 1] - u[Mz - 2]) / dz;
3074 v_z = (v[Mz - 1] - v[Mz - 2]) / dz;
3075 w_z = (w[Mz - 1] - w[Mz - 2]) / dz;
3076 } else {
3077 const double dz_p = z[k + 1] - z[k], dz_m = z[k] - z[k - 1];
3078 u_z = 0.5 * ((u[k + 1] - u[k]) / dz_p + (u[k] - u[k - 1]) / dz_m);
3079 v_z = 0.5 * ((v[k + 1] - v[k]) / dz_p + (v[k] - v[k - 1]) / dz_m);
3080 w_z = 0.5 * ((w[k + 1] - w[k]) / dz_p + (w[k] - w[k - 1]) / dz_m);
3081 }
3082
3083 // These should be "epsilon dot", but that's just too long.
3084 const double eps_xx = u_x, eps_yy = v_y, eps_zz = w_z, eps_xy = 0.5 * (u_y + v_x),
3085 eps_xz = 0.5 * (u_z + w_x), eps_yz = 0.5 * (v_z + w_y);
3086
3087 // The second invariant of the 3D strain rate tensor; see equation 4.8 in [@ref
3088 // GreveBlatter2009]. Unlike secondInvariant_2D(), this code does not make assumptions about
3089 // the input velocity field: we do not ignore w_x and w_y and do not assume that u_z and v_z
3090 // are zero.
3091 const double gamma = (square(eps_xx) + square(eps_yy) + square(eps_zz) +
3092 2.0 * (square(eps_xy) + square(eps_xz) + square(eps_yz)));
3093
3094 double nu = 0.0;
3095 // Note: in PISM gamma has an extra factor of 1/2; compare to
3096 flow_law.effective_viscosity(hardness, 0.5 * gamma, &nu, NULL);
3097
3098 viscosity[k] = nu;
3099 }
3100 }
3101 } catch (...) {
3102 loop.failed();
3103 }
3104 loop.check();
3105
3106 return result;
3107}
3108
3109/*! @brief Report ice thickness */
3110class IceThickness : public Diag<IceModel> {
3111public:
3113
3114 auto ismip6 = m_config->get_flag("output.ISMIP6");
3115
3116 m_vars = { { m_sys, ismip6 ? "lithk" : "thk" } };
3117
3118 m_vars[0].long_name("land ice thickness").standard_name("land_ice_thickness").units("m");
3119 m_vars[0]["valid_min"] = { 0.0 };
3120 }
3121
3122protected:
3123 std::shared_ptr<array::Array> compute_impl() const {
3124
3125 auto result = allocate<array::Scalar>("thk");
3126
3127 result->copy_from(model->geometry().ice_thickness);
3128
3129 return result;
3130 }
3131};
3132
3133/*! @brief Report ice top surface elevation */
3134class IceBottomSurfaceElevation : public Diag<IceModel> {
3135public:
3137
3138 auto ismip6 = m_config->get_flag("output.ISMIP6");
3139
3140 m_vars = { { m_sys, ismip6 ? "base" : "ice_base_elevation" } };
3141 m_vars[0].long_name("ice bottom surface elevation").units("m");
3142 }
3143
3144protected:
3145 std::shared_ptr<array::Array> compute_impl() const {
3146
3147 auto result = allocate<array::Scalar>("ice_base_elevation");
3148
3149 ice_bottom_surface(model->geometry(), *result);
3150
3151 return result;
3152 }
3153};
3154
3155/*! @brief Report ice top surface elevation */
3156class IceSurfaceElevation : public Diag<IceModel> {
3157public:
3159
3160 auto ismip6 = m_config->get_flag("output.ISMIP6");
3161
3162 m_vars = { { m_sys, ismip6 ? "orog" : "usurf" } };
3163 m_vars[0].long_name("ice top surface elevation").standard_name("surface_altitude").units("m");
3164 }
3165
3166protected:
3167 std::shared_ptr<array::Array> compute_impl() const {
3168 auto result = allocate<array::Scalar>("usurf");
3169
3170 result->copy_from(model->geometry().ice_surface_elevation);
3171
3172 return result;
3173 }
3174};
3175
3176/*! @brief Report grounding line flux. */
3177class GroundingLineFlux : public DiagAverageRate<IceModel> {
3178public:
3179 GroundingLineFlux(const IceModel *m) : DiagAverageRate<IceModel>(m, "grounding_line_flux", RATE) {
3180
3181 m_accumulator.metadata()["units"] = "kg m^-2";
3182
3183 auto ismip6 = m_config->get_flag("output.ISMIP6");
3184
3185 m_vars = { { m_sys, ismip6 ? "ligroundf" : "grounding_line_flux" } };
3186
3187 m_vars[0]
3188 .long_name("grounding line flux")
3189 .units("kg m^-2 second^-1")
3190 .output_units("kg m^-2 year^-1");
3191 m_vars[0]["cell_methods"] = "time: mean";
3192
3193 m_vars[0]["_FillValue"] = {to_internal(m_fill_value)};
3194 m_vars[0]["comment"] =
3195 "Positive flux corresponds to mass moving from the ocean to"
3196 " an icy grounded area. This convention makes it easier to compare"
3197 " grounding line flux to the total discharge into the ocean";
3198 }
3199
3200protected:
3201 void update_impl(double dt) {
3202 auto grid = m_accumulator.grid();
3203 auto cell_area = grid->cell_area(); // units: m^2
3204 auto ice_density =
3205 grid->ctx()->config()->get_number("constants.ice.density"); // units: kg / m^3
3206
3207 // factor used to convert from m^3/s to kg/m^2
3208 double unit_conversion_factor = dt * (ice_density / cell_area); // units: kg * s / m^5
3209
3210 ice_flow_rate_across_grounding_line(model->geometry().cell_type,
3211 model->geometry_evolution().flux_staggered(),
3212 unit_conversion_factor, m_accumulator);
3213
3214 m_interval_length += dt;
3215 }
3216};
3217
3219public:
3221 : DiagAverageRate<IceModel>(m, "ice_mass_transport_across_grounding_line", RATE) {
3222
3223 m_accumulator.metadata()["units"] = "kg";
3224
3225 m_vars = { { m_sys, "ice_mass_transport_across_grounding_line" } };
3226
3227 m_vars[0]
3228 .long_name("ice mass flow rate across the grounding line")
3229 .units("kg s^-1")
3230 .output_units("Gt year^-1");
3231 m_vars[0]["cell_methods"] = "time: mean";
3232
3233 m_vars[0]["_FillValue"] = { to_internal(m_fill_value) };
3234 m_vars[0]["comment"] =
3235 "Negative values correspond to mass moving from an icy grounded area into a lake or ocean."
3236 " This convention makes it easier to compare to calving, frontal melt, and discharge fluxes.";
3237 }
3238
3239protected:
3240 void update_impl(double dt) {
3241 auto grid = model->grid();
3242 double ice_density =
3243 grid->ctx()->config()->get_number("constants.ice.density"); // units: kg / m^3
3244
3245 // factor used to convert from m^3/s to kg
3246 double unit_conversion_factor = dt * ice_density; // units: kg * s / m^3
3247
3248 ice_flow_rate_across_grounding_line(model->geometry().cell_type,
3249 model->geometry_evolution().flux_staggered(),
3250 unit_conversion_factor, m_accumulator);
3251
3252 m_interval_length += dt;
3253 }
3254};
3255
3256
3257} // end of namespace diagnostics
3258
3260
3261 using namespace diagnostics;
3262
3263 typedef Diagnostic d;
3264 typedef Diagnostic::Ptr f; // "f" for "field"
3265 m_diagnostics = {
3266 // geometry
3267 {"cell_grounded_fraction", d::wrap(m_geometry.cell_grounded_fraction)},
3268 {"height_above_flotation", f(new HeightAboveFloatation(this))},
3269 {"ice_area_specific_volume", d::wrap(m_geometry.ice_area_specific_volume)},
3270 {"ice_mass", f(new IceMass(this))},
3271 {"lat", d::wrap(m_geometry.latitude)},
3272 {"lon", d::wrap(m_geometry.longitude)},
3273 {"mask", d::wrap(m_geometry.cell_type)},
3274 {"thk", f(new IceThickness(this))},
3275 {"topg_sl_adjusted", f(new BedTopographySeaLevelAdjusted(this))},
3276 {"usurf", f(new IceSurfaceElevation(this))},
3277 {"ice_base_elevation", f(new IceBottomSurfaceElevation(this))},
3278 {floating_ice_sheet_area_fraction_name, f(new IceAreaFractionFloating(this))},
3279 {grounded_ice_sheet_area_fraction_name, f(new IceAreaFractionGrounded(this))},
3280 {land_ice_area_fraction_name, f(new IceAreaFraction(this))},
3281
3282 // temperature, enthalpy, and liquid water fraction
3283 {"enthalpybase", f(new IceEnthalpyBasal(this))},
3284 {"enthalpysurf", f(new IceEnthalpySurface(this))},
3285 {"bedtoptemp", d::wrap(m_bedtoptemp)},
3286 {"cts", f(new CTS(this))},
3287 {"liqfrac", f(new LiquidFraction(this))},
3288 {"temp", f(new Temperature(this))},
3289 {"temp_pa", f(new TemperaturePA(this))},
3290 {"tempbase", f(new TemperatureBasal(this, BOTH))},
3291 {"temppabase", f(new TemperaturePABasal(this))},
3292 {"tempsurf", f(new TemperatureSurface(this))},
3293
3294 // rheology-related stuff
3295 {"tempicethk", f(new TemperateIceThickness(this))},
3296 {"tempicethk_basal", f(new TemperateIceThicknessBasal(this))},
3297 {"hardav", f(new HardnessAverage(this))},
3298 {"hardness", f(new IceHardness(this))},
3299 {"effective_viscosity", f(new IceViscosity(this))},
3300
3301 // boundary conditions
3302 {"vel_bc_mask", d::wrap(m_velocity_bc_mask)},
3303 {"vel_bc_values", d::wrap(m_velocity_bc_values)},
3304 {"ice_margin_pressure_difference", f(new IceMarginPressureDifference(this))},
3305 {"thk_bc_mask", d::wrap(m_ice_thickness_bc_mask)},
3306
3307 // balancing the books
3308 // tendency_of_ice_amount = (tendency_of_ice_amount_due_to_flow +
3309 // tendency_of_ice_amount_due_to_conservation_error +
3310 // tendency_of_ice_amount_due_to_surface_mass_balance +
3311 // tendency_of_ice_amount_due_to_basal_mass_balance +
3312 // tendency_of_ice_amount_due_to_discharge)
3313 //
3314 // Also,
3315 // tendency_of_ice_amount_due_to_discharge = (tendency_of_ice_amount_due_to_calving +
3316 // tendency_of_ice_amount_due_to_frontal_melt +
3317 // tendency_of_ice_amount_due_to_forced_retreat)
3318 {"tendency_of_ice_amount", f(new TendencyOfIceAmount(this, AMOUNT))},
3319 {"tendency_of_ice_amount_due_to_flow", f(new TendencyOfIceAmountDueToFlow(this, AMOUNT))},
3320 {"tendency_of_ice_amount_due_to_conservation_error", f(new ConservationErrorFlux(this, AMOUNT))},
3321 {"tendency_of_ice_amount_due_to_surface_mass_flux", f(new SurfaceFlux(this, AMOUNT))},
3322 {"tendency_of_ice_amount_due_to_basal_mass_flux", f(new BasalFlux(this, AMOUNT))},
3323 {"tendency_of_ice_amount_due_to_discharge", f(new DischargeFlux(this, AMOUNT))},
3324 {"tendency_of_ice_amount_due_to_calving", f(new CalvingFlux(this, AMOUNT))},
3325 {"tendency_of_ice_amount_due_to_frontal_melt", f(new FrontalMeltFlux(this, AMOUNT))},
3326 {"tendency_of_ice_amount_due_to_forced_retreat", f(new ForcedRetreatFlux(this, AMOUNT))},
3327
3328 // same, in terms of mass
3329 // tendency_of_ice_mass = (tendency_of_ice_mass_due_to_flow +
3330 // tendency_of_ice_mass_due_to_conservation_error +
3331 // tendency_of_ice_mass_due_to_surface_mass_flux +
3332 // tendency_of_ice_mass_due_to_basal_mass_balance +
3333 // tendency_of_ice_mass_due_to_discharge)
3334 //
3335 // Also,
3336 // tendency_of_ice_mass_due_to_discharge = (tendency_of_ice_mass_due_to_calving +
3337 // tendency_of_ice_mass_due_to_frontal_melt +
3338 // tendency_of_ice_mass_due_to_forced_retreat)
3339 {"tendency_of_ice_mass", f(new TendencyOfIceAmount(this, MASS))},
3340 {"tendency_of_ice_mass_due_to_flow", f(new TendencyOfIceAmountDueToFlow(this, MASS))},
3341 {"tendency_of_ice_mass_due_to_conservation_error", f(new ConservationErrorFlux(this, MASS))},
3342 {"tendency_of_ice_mass_due_to_surface_mass_flux", f(new SurfaceFlux(this, MASS))},
3343 {"tendency_of_ice_mass_due_to_basal_mass_flux", f(new BasalFlux(this, MASS))},
3344 {"tendency_of_ice_mass_due_to_discharge", f(new DischargeFlux(this, MASS))},
3345 {"tendency_of_ice_mass_due_to_calving", f(new CalvingFlux(this, MASS))},
3346 {"tendency_of_ice_mass_due_to_frontal_melt", f(new FrontalMeltFlux(this, MASS))},
3347 {"tendency_of_ice_mass_due_to_forced_retreat", f(new ForcedRetreatFlux(this, MASS))},
3348
3349 // other rates and fluxes
3350 {"basal_mass_flux_grounded", f(new BMBSplit(this, GROUNDED))},
3351 {"basal_mass_flux_floating", f(new BMBSplit(this, SHELF))},
3352 {"dHdt", f(new ThicknessRateOfChange(this))},
3353 {"bmelt", d::wrap(m_basal_melt_rate)},
3354 {"grounding_line_flux", f(new GroundingLineFlux(this))},
3355 {"ice_mass_transport_across_grounding_line", f(new MassTransportAcrossGroundingLine(this))},
3356
3357 // misc
3358 {"rank", f(new Rank(this))},
3359 };
3360
3361#if (Pism_USE_PROJ==1)
3362 std::string proj = m_grid->get_mapping_info().proj_string;
3363 if (not proj.empty()) {
3364 m_diagnostics["lat_bnds"] = f(new LatLonBounds(this, "lat", proj));
3365 m_diagnostics["lon_bnds"] = f(new LatLonBounds(this, "lon", proj));
3366 }
3367#endif
3368
3369 // add ISMIP6 variable names
3370 if (m_config->get_flag("output.ISMIP6")) {
3371 m_diagnostics["base"] = m_diagnostics["ice_base_elevation"];
3372 m_diagnostics["lithk"] = m_diagnostics["thk"];
3373 m_diagnostics["dlithkdt"] = m_diagnostics["dHdt"];
3374 m_diagnostics["orog"] = m_diagnostics["usurf"];
3375 m_diagnostics["acabf"] = m_diagnostics["tendency_of_ice_amount_due_to_surface_mass_flux"];
3376 m_diagnostics["libmassbfgr"] = m_diagnostics["basal_mass_flux_grounded"];
3377 m_diagnostics["libmassbffl"] = m_diagnostics["basal_mass_flux_floating"];
3378 m_diagnostics["lifmassbf"] = m_diagnostics["tendency_of_ice_amount_due_to_discharge"];
3379 m_diagnostics["licalvf"] = m_diagnostics["tendency_of_ice_amount_due_to_calving"];
3380 m_diagnostics["litempbotgr"] = f(new TemperatureBasal(this, GROUNDED));
3381 m_diagnostics["litempbotfl"] = f(new TemperatureBasal(this, SHELF));
3382 m_diagnostics["ligroundf"] = m_diagnostics["grounding_line_flux"];
3383 }
3384
3385 typedef TSDiagnostic::Ptr s; // "s" for "scalar"
3387 // area
3388 {"ice_area_glacierized", s(new scalar::IceAreaGlacierized(this))},
3389 {"ice_area_glacierized_cold_base", s(new scalar::IceAreaGlacierizedColdBase(this))},
3390 {"ice_area_glacierized_grounded", s(new scalar::IceAreaGlacierizedGrounded(this))},
3391 {"ice_area_glacierized_floating", s(new scalar::IceAreaGlacierizedShelf(this))},
3392 {"ice_area_glacierized_temperate_base", s(new scalar::IceAreaGlacierizedTemperateBase(this))},
3393 // mass
3394 {"ice_mass_glacierized", s(new scalar::IceMassGlacierized(this))},
3395 {"ice_mass", s(new scalar::IceMass(this))},
3396 {"tendency_of_ice_mass_glacierized", s(new scalar::IceMassRateOfChangeGlacierized(this))},
3397 {"limnsw", s(new scalar::IceMassNotDisplacingSeaWater(this))},
3398 // volume
3399 {"ice_volume_glacierized", s(new scalar::IceVolumeGlacierized(this))},
3400 {"ice_volume_glacierized_cold", s(new scalar::IceVolumeGlacierizedCold(this))},
3401 {"ice_volume_glacierized_grounded", s(new scalar::IceVolumeGlacierizedGrounded(this))},
3402 {"ice_volume_glacierized_floating", s(new scalar::IceVolumeGlacierizedShelf(this))},
3403 {"ice_volume_glacierized_temperate", s(new scalar::IceVolumeGlacierizedTemperate(this))},
3404 {"ice_volume", s(new scalar::IceVolume(this))},
3405 {"ice_volume_cold", s(new scalar::IceVolumeCold(this))},
3406 {"ice_volume_temperate", s(new scalar::IceVolumeTemperate(this))},
3407 {"tendency_of_ice_volume_glacierized", s(new scalar::IceVolumeRateOfChangeGlacierized(this))},
3408 {"tendency_of_ice_volume", s(new scalar::IceVolumeRateOfChange(this))},
3409 {"sea_level_rise_potential", s(new scalar::SeaLevelRisePotential(this))},
3410 // energy
3411 {"ice_enthalpy_glacierized", s(new scalar::IceEnthalpyGlacierized(this))},
3412 {"ice_enthalpy", s(new scalar::IceEnthalpy(this))},
3413 // time-stepping
3414 {"max_diffusivity", s(new scalar::MaxDiffusivity(this))},
3415 {"max_horizontal_vel", s(new scalar::MaxHorizontalVelocity(this))},
3416 {"dt", s(new scalar::TimeStepLength(this))},
3417 {"dt_ratio", s(new scalar::TimeStepRatio(this))},
3418 // balancing the books
3419 {"tendency_of_ice_mass", s(new scalar::IceMassRateOfChange(this))},
3420 {"tendency_of_ice_mass_due_to_flow", s(new scalar::IceMassRateOfChangeDueToFlow(this))},
3421 {"tendency_of_ice_mass_due_to_conservation_error", s(new scalar::IceMassFluxConservationError(this))},
3422 {"tendency_of_ice_mass_due_to_basal_mass_flux", s(new scalar::IceMassFluxBasal(this))},
3423 {"tendency_of_ice_mass_due_to_surface_mass_flux", s(new scalar::IceMassFluxSurface(this))},
3424 {"tendency_of_ice_mass_due_to_discharge", s(new scalar::IceMassFluxDischarge(this))},
3425 {"tendency_of_ice_mass_due_to_calving", s(new scalar::IceMassFluxCalving(this))},
3426 // other fluxes
3427 {"basal_mass_flux_grounded", s(new scalar::IceMassFluxBasalGrounded(this))},
3428 {"basal_mass_flux_floating", s(new scalar::IceMassFluxBasalFloating(this))},
3429 {"grounding_line_flux", s(new scalar::IceMassFluxAtGroundingLine(this))},
3430 };
3431
3432 // add ISMIP6 variable names
3433 if (m_config->get_flag("output.ISMIP6")) {
3434 m_ts_diagnostics["iareafl"] = m_ts_diagnostics["ice_area_glacierized_floating"];
3435 m_ts_diagnostics["iareagr"] = m_ts_diagnostics["ice_area_glacierized_grounded"];
3436 m_ts_diagnostics["lim"] = m_ts_diagnostics["ice_mass"];
3437 m_ts_diagnostics["tendacabf"] = m_ts_diagnostics["tendency_of_ice_mass_due_to_surface_mass_flux"];
3438 m_ts_diagnostics["tendlibmassbf"] = m_ts_diagnostics["tendency_of_ice_mass_due_to_basal_mass_flux"];
3439 m_ts_diagnostics["tendlibmassbffl"] = m_ts_diagnostics["basal_mass_flux_floating"];
3440 m_ts_diagnostics["tendlicalvf"] = m_ts_diagnostics["tendency_of_ice_mass_due_to_calving"];
3441 m_ts_diagnostics["tendlifmassbf"] = m_ts_diagnostics["tendency_of_ice_mass_due_to_discharge"];
3442 m_ts_diagnostics["tendligroundf"] = m_ts_diagnostics["grounding_line_flux"];
3443 }
3444
3445 // get diagnostics from submodels
3446 for (const auto& m : m_submodels) {
3447 m_diagnostics = pism::combine(m_diagnostics, m.second->diagnostics());
3448 m_ts_diagnostics = pism::combine(m_ts_diagnostics, m.second->ts_diagnostics());
3449 }
3450}
3451
3452typedef std::map<std::string, std::vector<VariableMetadata>> Metadata;
3453
3454static void print_diagnostics(const Logger &log, const Metadata &list) {
3455 for (const auto &d : list) {
3456 const std::string &name = d.first;
3457 log.message(1, " Name: %s\n", name.c_str());
3458
3459 for (const auto &v : d.second) {
3460
3461 std::string
3462 var_name = v.get_name(),
3463 units = v["units"],
3464 output_units = v["output_units"],
3465 long_name = v["long_name"],
3466 comment = v["comment"];
3467
3468 if (not output_units.empty()) {
3469 units = output_units;
3470 }
3471
3472 log.message(1, " %s [%s]\n", var_name.c_str(), units.c_str());
3473 log.message(1, " %s\n", long_name.c_str());
3474 if (not comment.empty()) {
3475 log.message(1, " %s\n", comment.c_str());
3476 }
3477 }
3478 log.message(1, "\n");
3479 }
3480}
3481
3482static void print_diagnostics_json(const Logger &log, const Metadata &list) {
3483 log.message(1, "{\n");
3484 bool first_diagnostic = true;
3485 for (const auto &d : list) {
3486
3487 if (not first_diagnostic) {
3488 log.message(1, ",\n");
3489 } else {
3490 first_diagnostic = false;
3491 }
3492
3493 log.message(1, "\"%s\" : [\n", d.first.c_str());
3494
3495 bool first_variable = true;
3496 for (const auto &variable : d.second) {
3497
3498 std::string
3499 var_name = variable.get_name(),
3500 units = variable["units"],
3501 output_units = variable["output_units"],
3502 long_name = variable["long_name"],
3503 standard_name = variable["standard_name"],
3504 comment = variable["comment"];
3505
3506 if (not output_units.empty()) {
3507 units = output_units;
3508 }
3509
3510 if (not first_variable) {
3511 log.message(1, ",\n");
3512 } else {
3513 first_variable = false;
3514 }
3515
3516 log.message(1, "[\"%s\", \"%s\", \"%s\", \"%s\", \"%s\"]",
3517 var_name.c_str(), units.c_str(), long_name.c_str(), standard_name.c_str(), comment.c_str());
3518 }
3519 log.message(1, "]");
3520 }
3521 log.message(1, "}\n");
3522}
3523
3524/*!
3525 * Return metadata of 2D and 3D diagnostics.
3526 */
3527static Metadata diag_metadata(const std::map<std::string,Diagnostic::Ptr> &diags) {
3528 Metadata result;
3529
3530 for (const auto& f : diags) {
3531 Diagnostic::Ptr diag = f.second;
3532
3533 for (unsigned int k = 0; k < diag->n_variables(); ++k) {
3534 result[f.first].push_back(diag->metadata(k));
3535 }
3536 }
3537
3538 return result;
3539}
3540
3541/*!
3542 * Return metadata of scalar diagnostics.
3543 */
3544static Metadata ts_diag_metadata(const std::map<std::string,TSDiagnostic::Ptr> &ts_diags) {
3545 Metadata result;
3546
3547 for (const auto& d : ts_diags) {
3548 // always one variable per diagnostic
3549 result[d.first] = {d.second->metadata()};
3550 }
3551
3552 return result;
3553}
3554
3555void IceModel::list_diagnostics(const std::string &list_type) const {
3556
3557 if (list_type == "json") {
3558 m_log->message(1, "{\n");
3559
3560 m_log->message(1, "\"spatial\" :\n");
3562
3563 m_log->message(1, ",\n"); // separator
3564
3565 m_log->message(1, "\"scalar\" :\n");
3567
3568 m_log->message(1, "}\n");
3569
3570 return;
3571 }
3572
3573 if (member(list_type, {"all", "spatial"})) {
3574 m_log->message(1, "\n");
3575 m_log->message(1, "======== Available 2D and 3D diagnostics ========\n");
3576
3578 }
3579
3580 if (member(list_type, {"all", "scalar"})) {
3581 // scalar time-series
3582 m_log->message(1, "======== Available time-series ========\n");
3583
3585 }
3586}
3587
3588/*!
3589 Computes fraction of the base which is melted.
3590
3591 Communication occurs here.
3592 */
3593double IceModel::compute_temperate_base_fraction(double total_ice_area) {
3594
3595 EnthalpyConverter::Ptr EC = m_ctx->enthalpy_converter();
3596
3597 auto cell_area = m_grid->cell_area();
3598
3599 double result = 0.0, meltarea = 0.0;
3600
3601 const array::Array3D &enthalpy = m_energy_model->enthalpy();
3602
3604 ParallelSection loop(m_grid->com);
3605 try {
3606 for (auto p = m_grid->points(); p; p.next()) {
3607 const int i = p.i(), j = p.j();
3608
3609 if (m_geometry.cell_type.icy(i, j)) {
3610 const double
3611 E_basal = enthalpy.get_column(i, j)[0],
3612 pressure = EC->pressure(m_geometry.ice_thickness(i,j)); // FIXME issue #15
3613 // accumulate area of base which is at melt point
3614 if (EC->is_temperate_relaxed(E_basal, pressure)) {
3615 meltarea += cell_area;
3616 }
3617 }
3618 }
3619 } catch (...) {
3620 loop.failed();
3621 }
3622 loop.check();
3623
3624 // convert from m2 to km2
3625 meltarea = units::convert(m_sys, meltarea, "m^2", "km^2");
3626 // communication
3627 result = GlobalSum(m_grid->com, meltarea);
3628
3629 // normalize fraction correctly
3630 if (total_ice_area > 0.0) {
3631 result = result / total_ice_area;
3632 } else {
3633 result = 0.0;
3634 }
3635 return result;
3636}
3637
3638
3639/*!
3640 Computes fraction of the ice which is as old as the start of the run (original).
3641 Communication occurs here.
3642 */
3643double IceModel::compute_original_ice_fraction(double total_ice_volume) {
3644
3645 double result = -1.0; // result value if not age.enabled
3646
3647 if (not m_age_model) {
3648 return result; // leave now
3649 }
3650
3651 const double a = m_grid->cell_area() * 1e-3 * 1e-3, // area unit (km^2)
3652 currtime = m_time->current(); // seconds
3653
3654 const array::Array3D &ice_age = m_age_model->age();
3655
3657
3658 const double one_year = units::convert(m_sys, 1.0, "year", "seconds");
3659 double original_ice_volume = 0.0;
3660
3661 // compute local original volume
3662 ParallelSection loop(m_grid->com);
3663 try {
3664 for (auto p = m_grid->points(); p; p.next()) {
3665 const int i = p.i(), j = p.j();
3666
3667 if (m_geometry.cell_type.icy(i, j)) {
3668 // accumulate volume of ice which is original
3669 const double *age = ice_age.get_column(i, j);
3670 auto ks = m_grid->kBelowHeight(m_geometry.ice_thickness(i,j));
3671 for (unsigned int k = 1; k <= ks; k++) {
3672 // ice in segment is original if it is as old as one year less than current time
3673 if (0.5 * (age[k - 1] + age[k]) > currtime - one_year) {
3674 original_ice_volume += a * 1.0e-3 * (m_grid->z(k) - m_grid->z(k - 1));
3675 }
3676 }
3677 }
3678 }
3679 } catch (...) {
3680 loop.failed();
3681 }
3682 loop.check();
3683
3684
3685 // communicate to turn into global original fraction
3686 result = GlobalSum(m_grid->com, original_ice_volume);
3687
3688 // normalize fraction correctly
3689 if (total_ice_volume > 0.0) {
3690 result = result / total_ice_volume;
3691 } else {
3692 result = 0.0;
3693 }
3694 return result;
3695}
3696
3697} // end of namespace pism
double get_number(const std::string &name, UseFlag flag=REMEMBER_THIS_USE) const
A class for storing and accessing PISM configuration flags and parameters.
const IceModel * model
A template derived from Diagnostic, adding a "Model".
double m_fill_value
fill value (used often enough to justify storing it)
const units::System::Ptr m_sys
the unit system
double to_internal(double x) const
Definition Diagnostic.cc:59
std::vector< SpatialVariableMetadata > m_vars
metadata corresponding to NetCDF variables
std::shared_ptr< Diagnostic > Ptr
Definition Diagnostic.hh:65
std::shared_ptr< const Grid > m_grid
the grid
std::shared_ptr< array::Array > compute() const
Compute a diagnostic quantity and return a pointer to a newly-allocated Array.
const Config::ConstPtr m_config
Configuration flags and parameters.
Class representing diagnostic computations in PISM.
Definition Diagnostic.hh:60
std::shared_ptr< EnthalpyConverter > Ptr
void compute_mask(const array::Scalar &sea_level, const array::Scalar &bed, const array::Scalar &thickness, array::Scalar &result) const
Definition Mask.cc:36
void set_icefree_thickness(double threshold)
Definition Mask.hh:81
const array::Scalar & thickness_change_due_to_flow() const
const array::Scalar & bottom_surface_mass_balance() const
const array::Scalar & top_surface_mass_balance() const
const array::Scalar & area_specific_volume_change_due_to_flow() const
const array::Scalar & conservation_error() const
const array::Staggered1 & flux_staggered() const
array::Scalar1 sea_level_elevation
Definition Geometry.hh:48
array::Scalar cell_grounded_fraction
Definition Geometry.hh:56
array::Scalar2 ice_surface_elevation
Definition Geometry.hh:57
array::Scalar1 ice_area_specific_volume
Definition Geometry.hh:52
array::CellType2 cell_type
Definition Geometry.hh:55
array::Scalar2 ice_thickness
Definition Geometry.hh:51
array::Scalar longitude
Definition Geometry.hh:42
array::Scalar2 bed_elevation
Definition Geometry.hh:47
array::Scalar latitude
Definition Geometry.hh:41
Describes the PISM grid and the distribution of data across processors.
Definition Grid.hh:290
std::map< std::string, const Component * > m_submodels
the list of sub-models, for writing model states and obtaining diagnostics
Definition IceModel.hh:252
virtual double compute_temperate_base_fraction(double ice_area)
const Geometry & geometry() const
Definition IceModel.cc:877
std::map< std::string, TSDiagnostic::Ptr > m_ts_diagnostics
Requested scalar diagnostics.
Definition IceModel.hh:419
const Time::Ptr m_time
Time manager.
Definition IceModel.hh:246
Geometry m_geometry
Definition IceModel.hh:288
const GeometryEvolution & geometry_evolution() const
Definition IceModel.cc:881
const stressbalance::StressBalance * stress_balance() const
Definition IceModel.cc:885
virtual double compute_original_ice_fraction(double ice_volume)
std::shared_ptr< Context > m_ctx
Execution context.
Definition IceModel.hh:240
array::Scalar m_basal_melt_rate
rate of production of basal meltwater (ice-equivalent); no ghosts
Definition IceModel.hh:295
std::shared_ptr< Grid > grid() const
Return the grid used by this model.
Definition utilities.cc:121
const Logger::Ptr m_log
Logger.
Definition IceModel.hh:244
array::Scalar m_bedtoptemp
temperature at the top surface of the bedrock thermal layer
Definition IceModel.hh:297
const energy::EnergyModel * energy_balance_model() const
Definition IceModel.cc:897
const array::Scalar & frontal_melt() const
Definition IceModel.cc:919
const array::Scalar & forced_retreat() const
Definition IceModel.cc:926
double dt() const
Definition IceModel.cc:126
array::Scalar2 m_velocity_bc_mask
mask to determine Dirichlet boundary locations for the sliding velocity
Definition IceModel.hh:302
Config::Ptr m_config
Configuration flags and parameters.
Definition IceModel.hh:238
array::Scalar1 m_ice_thickness_bc_mask
Mask prescribing locations where ice thickness is held constant.
Definition IceModel.hh:307
std::shared_ptr< AgeModel > m_age_model
Definition IceModel.hh:262
const units::System::Ptr m_sys
Unit system.
Definition IceModel.hh:242
array::Vector2 m_velocity_bc_values
Dirichlet boundary velocities.
Definition IceModel.hh:304
void list_diagnostics(const std::string &list_type) const
std::map< std::string, Diagnostic::Ptr > m_diagnostics
Requested spatially-variable diagnostics.
Definition IceModel.hh:417
const array::Scalar & calving() const
Definition IceModel.cc:912
std::shared_ptr< Context > ctx() const
Return the context this model is running in.
Definition utilities.cc:126
virtual void init_diagnostics()
std::shared_ptr< energy::EnergyModel > m_energy_model
Definition IceModel.hh:260
const std::shared_ptr< Grid > m_grid
Computational grid.
Definition IceModel.hh:236
void message(int threshold, const char format[],...) const __attribute__((format(printf
Print a message to the log.
Definition Logger.cc:49
A basic logging class.
Definition Logger.hh:40
void failed()
Indicates a failure of a parallel section.
A wrapper for PJ that makes sure pj_destroy is called.
Definition Proj.hh:32
static RuntimeError formatted(const ErrorLocation &location, const char format[],...) __attribute__((format(printf
build a RuntimeError with a formatted message
void set_units(const std::string &units, const std::string &output_units)
VariableMetadata m_variable
const Config::ConstPtr m_config
Configuration flags and parameters.
std::shared_ptr< const Grid > m_grid
the grid
std::shared_ptr< TSDiagnostic > Ptr
Scalar diagnostic reporting a "flux".
Scalar diagnostic reporting the rate of change of a quantity modeled by PISM.
Scalar diagnostic reporting a snapshot of a quantity modeled by PISM.
VariableMetadata & long_name(const std::string &input)
VariableMetadata & units(const std::string &input)
VariableMetadata & set_name(const std::string &name)
VariableMetadata & standard_name(const std::string &input)
VariableMetadata & output_units(const std::string &input)
Makes sure that we call begin_access() and end_access() for all accessed array::Arrays.
Definition Array.hh:64
void copy_from(const Array2D< T > &source)
Definition Array2D.hh:73
stencils::Star< T > star(int i, int j) const
Definition Array2D.hh:79
void add(double alpha, const Array2D< T > &x)
Definition Array2D.hh:65
void copy_from(const Array3D &input)
Definition Array3D.cc:209
double * get_column(int i, int j)
Definition Array3D.cc:121
A virtual class collecting methods common to ice and bedrock 3D fields.
Definition Array3D.hh:33
std::shared_ptr< const Grid > grid() const
Definition Array.cc:131
SpatialVariableMetadata & metadata(unsigned int N=0)
Returns a reference to the SpatialVariableMetadata object containing metadata for the compoment N.
Definition Array.cc:476
stencils::Star< int > star_int(int i, int j) const
Definition Scalar.hh:72
bool next_to_ice_free_ocean(int i, int j) const
Definition CellType.hh:106
bool ice_free_ocean(int i, int j) const
Definition CellType.hh:58
bool grounded_ice(int i, int j) const
Definition CellType.hh:46
bool icy(int i, int j) const
Definition CellType.hh:42
BMBSplit(const IceModel *m, AreaType flag)
Report average basal mass balance flux over the reporting interval (grounded or floating areas)
BasalFlux(const IceModel *m, AmountKind kind)
Report basal mass balance flux, averaged over the reporting interval.
std::shared_ptr< array::Array > compute_impl() const
Sea-level adjusted bed topography (zero at sea level).
CTS(const IceModel *m)
virtual std::shared_ptr< array::Array > compute_impl() const
Computes CTS, CTS = E/E_s(p).
CalvingFlux(const IceModel *m, AmountKind kind)
Report the calving flux.
ConservationErrorFlux(const IceModel *m, AmountKind kind)
DischargeFlux(const IceModel *m, AmountKind kind)
Report discharge (calving and frontal melt) flux.
ForcedRetreatFlux(const IceModel *m, AmountKind kind)
Report the frontal melt flux.
FrontalMeltFlux(const IceModel *m, AmountKind kind)
Report the frontal melt flux.
Report grounding line flux.
virtual std::shared_ptr< array::Array > compute_impl() const
Computes vertically-averaged ice hardness.
Computes vertically-averaged ice hardness.
virtual std::shared_ptr< array::Array > compute_impl() const
Computes the 2D height above flotation.
virtual std::shared_ptr< array::Array > compute_impl() const
virtual std::shared_ptr< array::Array > compute_impl() const
virtual std::shared_ptr< array::Array > compute_impl() const
std::shared_ptr< array::Array > compute_impl() const
Report ice top surface elevation.
virtual std::shared_ptr< array::Array > compute_impl() const
Computes enthalpy at the base of the ice.
virtual std::shared_ptr< array::Array > compute_impl() const
Computes surface values of ice enthalpy.
std::shared_ptr< array::Array > compute_impl() const
Ice hardness computed using the SIA flow law.
std::shared_ptr< array::Array > compute_impl() const
Ocean pressure difference at calving fronts. Used to debug CF boundary conditins.
IceMass(const IceModel *m)
virtual std::shared_ptr< array::Array > compute_impl() const
Computes the mass per cell.
std::shared_ptr< array::Array > compute_impl() const
Report ice top surface elevation.
std::shared_ptr< array::Array > compute_impl() const
std::shared_ptr< array::Array > compute_impl() const
Effective viscosity of ice (3D).
LatLonBounds(const IceModel *m, const std::string &var_name, const std::string &proj_string)
virtual std::shared_ptr< array::Array > compute_impl() const
Computes latitude and longitude bounds.
virtual std::shared_ptr< array::Array > compute_impl() const
Computes the liquid water fraction.
Rank(const IceModel *m)
virtual std::shared_ptr< array::Array > compute_impl() const
Computes a diagnostic field filled with processor rank values.
SurfaceFlux(const IceModel *m, AmountKind kind)
Report surface mass balance flux, averaged over the reporting interval.
virtual std::shared_ptr< array::Array > compute_impl() const
Computes the thickness of the basal layer of temperate ice.
virtual std::shared_ptr< array::Array > compute_impl() const
Computes the total thickness of temperate ice in a column.
std::shared_ptr< array::Array > compute_impl() const
TemperatureBasal(const IceModel *m, AreaType area_type)
Computes ice temperature at the base of the ice.
virtual std::shared_ptr< array::Array > compute_impl() const
Computes basal values of the pressure-adjusted temperature.
virtual std::shared_ptr< array::Array > compute_impl() const
Compute the pressure-adjusted temperature in degrees C corresponding to ice temperature.
virtual std::shared_ptr< array::Array > compute_impl() const
Computes ice temperature at the surface of the ice.
virtual std::shared_ptr< array::Array > compute_impl() const
Computes ice temperature from enthalpy.
TendencyOfIceAmountDueToFlow(const IceModel *Model, AmountKind kind)
Computes tendency_of_ice_amount_due_to_flow, the rate of change of ice amount due to flow.
std::shared_ptr< array::Array > compute_impl() const
TendencyOfIceAmount(const IceModel *Model, AmountKind kind)
Computes tendency_of_ice_amount, the ice amount rate of change.
std::shared_ptr< array::Array > compute_impl() const
Computes dHdt, the ice thickness rate of change.
Computes the total area of the cold ice.
Computes the total grounded ice area.
Computes the total floating ice area.
Computes the total area of the temperate ice.
Computes the total ice enthalpy in glacierized areas.
Computes the total ice enthalpy.
Reports the total flux across the grounding line.
Reports the total sub-shelf ice flux.
Reports the total basal ice flux over the grounded region.
Reports the total bottom surface ice flux.
Reports the total numerical mass flux needed to preserve non-negativity of ice thickness.
Reports the total discharge flux.
Reports the total top surface ice flux.
Computes the total ice mass in glacierized areas.
Computes the total mass of the ice not displacing sea water.
Computes the rate of change of the total ice mass due to flow (influx due to prescribed constant-in-t...
Computes the rate of change of the total ice mass in glacierized areas.
Computes the rate of change of the total ice mass.
Computes the total ice mass.
Computes the total volume of the cold ice.
Computes the total volume of the cold ice in glacierized areas.
Computes the total grounded ice volume.
Computes the total floating ice volume.
Computes the total volume of the temperate ice in glacierized areas.
Computes the total ice volume in glacierized areas.
Computes the rate of change of the total ice volume in glacierized areas.
Computes the rate of change of the total ice volume.
Computes the total volume of the temperate ice.
Computes the total ice volume.
Reports the maximum horizontal absolute velocity component over the grid.
Computes the total ice volume which is relevant for sea-level.
Reports the mass continuity time step.
Reports the mass continuity time step.
const array::Array3D & enthalpy() const
void effective_viscosity(double hardness, double gamma, double *nu, double *dnu) const
Computes the regularized effective viscosity and its derivative with respect to the second invariant ...
Definition FlowLaw.cc:162
double hardness(double E, double p) const
Definition FlowLaw.cc:117
std::shared_ptr< const rheology::FlowLaw > flow_law() const
std::shared_ptr< const rheology::FlowLaw > flow_law() const
const array::Array3D & velocity_w() const
const SSB_Modifier * modifier() const
Returns a pointer to a stress balance modifier implementation.
const array::Array3D & velocity_u() const
Get components of the the 3D velocity field.
double max_diffusivity() const
Get the max diffusivity (for the adaptive time-stepping).
const ShallowStressBalance * shallow() const
Returns a pointer to a shallow stress balance solver implementation.
const array::Array3D & velocity_v() const
#define PISM_ERROR_LOCATION
const double rho_ice
Definition exactTestK.c:31
@ WITH_GHOSTS
Definition Array.hh:61
@ WITHOUT_GHOSTS
Definition Array.hh:61
static double ice_volume(const array::Scalar &ice_thickness, const array::Array3D &ice_enthalpy, IceKind kind, double thickness_threshold)
static double base_area(const array::Scalar &ice_thickness, const array::Array3D &ice_enthalpy, IceKind kind, double thickness_threshold)
double mass_change(const IceModel *model, TermType term, AreaType area)
static double square(double x)
static void accumulate_changes(const IceModel *model, double factor, ChangeKind kind, array::Scalar &accumulator)
void compute_liquid_water_fraction(const array::Array3D &enthalpy, const array::Scalar &ice_thickness, array::Array3D &result)
Compute the liquid fraction corresponding to enthalpy and ice_thickness.
Definition utilities.cc:136
void compute_cts(const array::Array3D &ice_enthalpy, const array::Scalar &ice_thickness, array::Array3D &result)
Compute the CTS field, CTS = E/E_s(p), from ice_enthalpy and ice_thickness, and put in result.
Definition utilities.cc:179
double total_ice_enthalpy(double thickness_threshold, const array::Array3D &ice_enthalpy, const array::Scalar &ice_thickness)
Computes the total ice enthalpy in J.
Definition utilities.cc:218
bool domain_edge(const Grid &grid, int i, int j)
Definition Grid.hh:409
bool ice_free(int M)
Ice-free cell (grounded or ocean).
Definition Mask.hh:58
double averaged_hardness(const FlowLaw &ice, double thickness, unsigned int kbelowH, const double *zlevels, const double *enthalpy)
Computes vertical average of B(E, p) ice hardness, namely .
Definition FlowLaw.cc:213
double convert(System::Ptr system, double input, const std::string &spec1, const std::string &spec2)
Convert a quantity from unit1 to unit2.
Definition Units.cc:70
double ice_volume_not_displacing_seawater(const Geometry &geometry, double thickness_threshold)
Definition Geometry.cc:287
double ice_area(const Geometry &geometry, double thickness_threshold)
Computes ice area, in m^2.
Definition Geometry.cc:336
double grounded_area_fraction(double a, double b, double c)
static const char * grounded_ice_sheet_area_fraction_name
static Metadata diag_metadata(const std::map< std::string, Diagnostic::Ptr > &diags)
double average_water_column_pressure(double ice_thickness, double bed, double floatation_level, double rho_ice, double rho_water, double g)
static const double g
Definition exactTestP.cc:36
void compute_grounded_cell_fraction(double ice_density, double ocean_density, const array::Scalar1 &sea_level, const array::Scalar1 &ice_thickness, const array::Scalar1 &bed_topography, array::Scalar &result)
double ice_area_floating(const Geometry &geometry, double thickness_threshold)
Computes floating ice area, in m^2.
Definition Geometry.cc:353
double part_grid_threshold_thickness(stencils::Star< int > cell_type, stencils::Star< double > ice_thickness, stencils::Star< double > surface_elevation, double bed_elevation)
Compute threshold thickness used when deciding if a partially-filled cell should be considered 'full'...
double sea_level_rise_potential(const Geometry &geometry, double thickness_threshold)
Computes the sea level rise that would result if all the ice were melted.
Definition Geometry.cc:362
double total_grounding_line_flux(const array::CellType1 &cell_type, const array::Staggered1 &flux, double dt)
static const double k
Definition exactTestP.cc:42
static void print_diagnostics_json(const Logger &log, const Metadata &list)
double ice_volume(const Geometry &geometry, double thickness_threshold)
Computes the ice volume, in m^3.
Definition Geometry.cc:254
void ice_flow_rate_across_grounding_line(const array::CellType1 &cell_type, const array::Staggered1 &flux, double unit_conversion_factor, array::Scalar &output)
void ice_bottom_surface(const Geometry &geometry, array::Scalar &result)
Definition Geometry.cc:218
bool member(const std::string &string, const std::set< std::string > &set)
MaxTimestep max_timestep_diffusivity(double D_max, double dx, double dy, double adaptive_timestepping_ratio)
static Metadata ts_diag_metadata(const std::map< std::string, TSDiagnostic::Ptr > &ts_diags)
static void print_diagnostics(const Logger &log, const Metadata &list)
double ice_area_grounded(const Geometry &geometry, double thickness_threshold)
Computes grounded ice area, in m^2.
Definition Geometry.cc:344
T combine(const T &a, const T &b)
void compute_lat_bounds(const std::string &projection, array::Array3D &result)
void GlobalSum(MPI_Comm comm, double *local, double *result, int count)
void compute_lon_bounds(const std::string &projection, array::Array3D &result)
std::map< std::string, std::vector< VariableMetadata > > Metadata
static const char * land_ice_area_fraction_name
static const char * floating_ice_sheet_area_fraction_name