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"
39#if (Pism_USE_PROJ == 1)
40#include "pism/util/Proj.hh"
45namespace diagnostics {
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";
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";
69 m_vars[0].long_name(long_name).units(internal_units).output_units(external_units);
72 m_vars[0][
"valid_range"] = { -large_number, large_number };
74 m_vars[0][
"cell_methods"] =
"time: mean";
77 .
long_name(
"ice amount at the time of the last report of " + name)
78 .
units(internal_units +
" second");
84 auto result = std::make_shared<array::Scalar>(
m_grid,
"");
85 result->metadata() =
m_vars[0];
88 double ice_density =
m_config->get_number(
"constants.ice.density");
90 auto cell_area =
m_grid->cell_area();
97 for (
auto p =
m_grid->points(); p; p.next()) {
98 const int i = p.i(), j = p.j();
101 double amount = (thickness(i, j) + area_specific_volume(i, j)) * ice_density;
107 (*result)(i, j) *= cell_area;
123 double ice_density =
m_config->get_number(
"constants.ice.density");
127 for (
auto p =
m_grid->points(); p; p.next()) {
128 const int i = p.i(), j = p.j();
131 m_last_amount(i, j) = (thickness(i, j) + area_specific_volume(i, j)) * ice_density;
151 kind ==
AMOUNT ?
"tendency_of_ice_amount_due_to_flow" :
152 "tendency_of_ice_mass_due_to_flow",
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";
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";
175 m_vars[0][
"cell_methods"] =
"time: mean";
177 m_vars[0][
"comment"] =
"positive flux corresponds to ice gain";
183 &dV =
model->geometry_evolution().area_specific_volume_change_due_to_flow();
185 auto cell_area =
m_grid->cell_area();
189 for (
auto p =
m_grid->points(); p; p.next()) {
190 const int i = p.i(), j = p.j();
208 "tendency_of_ice_amount_due_to_surface_mass_flux" :
209 "tendency_of_ice_mass_due_to_surface_mass_flux",
214 auto ismip6 =
m_config->get_flag(
"output.ISMIP6");
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";
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";
233 .
units(internal_units)
236 m_vars[0][
"cell_methods"] =
"time: mean";
237 m_vars[0][
"comment"] =
"positive flux corresponds to ice gain";
244 auto cell_area =
m_grid->cell_area();
248 for (
auto p =
m_grid->points(); p; p.next()) {
249 const int i = p.i(), j = p.j();
266 kind ==
AMOUNT ?
"tendency_of_ice_amount_due_to_basal_mass_flux" :
267 "tendency_of_ice_mass_due_to_basal_mass_flux",
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";
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";
288 .
units(internal_units)
291 m_vars[0][
"cell_methods"] =
"time: mean";
292 m_vars[0][
"comment"] =
"positive flux corresponds to ice gain";
299 auto cell_area =
m_grid->cell_area();
303 for (
auto p =
m_grid->points(); p; p.next()) {
304 const int i = p.i(), j = p.j();
321 "tendency_of_ice_amount_due_to_conservation_error" :
322 "tendency_of_ice_mass_due_to_conservation_error",
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";
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";
342 .
units(internal_units)
345 m_vars[0][
"cell_methods"] =
"time: mean";
346 m_vars[0][
"comment"] =
"positive flux corresponds to ice gain";
352 &error =
model->geometry_evolution().conservation_error();
356 auto cell_area =
m_grid->cell_area();
358 for (
auto p =
m_grid->points(); p; p.next()) {
359 const int i = p.i(), j = p.j();
376 const auto &calving = model->
calving();
380 auto grid = accumulator.
grid();
390 if (add_frontal_melt) {
391 scope.add(frontal_melt);
393 if (add_forced_retreat) {
394 scope.add(forced_retreat);
397 for (
auto p = grid->points(); p; p.next()) {
398 const int i = p.i(), j = p.j();
401 accumulator(i, j) += factor * calving(i, j);
403 if (add_frontal_melt) {
404 accumulator(i, j) += factor * frontal_melt(i, j);
406 if (add_forced_retreat) {
407 accumulator(i, j) += factor * forced_retreat(i, j);
418 kind ==
AMOUNT ?
"tendency_of_ice_amount_due_to_discharge" :
419 "tendency_of_ice_mass_due_to_discharge",
425 auto ismip6 =
m_config->get_flag(
"output.ISMIP6");
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";
433 name =
"tendency_of_ice_mass_due_to_discharge";
434 long_name =
"discharge flux (calving, frontal melt, forced retreat)";
435 accumulator_units =
"kg";
437 internal_units =
"kg second^-1";
438 external_units =
"Gt year^-1";
447 .
units(internal_units)
449 m_vars[0][
"cell_methods"] =
"time: mean";
451 m_vars[0][
"comment"] =
"positive flux corresponds to ice gain";
473 ?
"tendency_of_ice_amount_due_to_calving"
474 :
"tendency_of_ice_mass_due_to_calving",
480 auto ismip6 =
m_config->get_flag(
"output.ISMIP6");
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";
490 name =
"tendency_of_ice_mass_due_to_calving";
491 long_name =
"calving flux";
492 accumulator_units =
"kg";
494 internal_units =
"kg second^-1";
495 external_units =
"Gt year^-1";
504 .
units(internal_units)
506 m_vars[0][
"cell_methods"] =
"time: mean";
509 m_vars[0][
"comment"] =
"positive flux corresponds to ice gain";
532 ?
"tendency_of_ice_amount_due_to_frontal_melt"
533 :
"tendency_of_ice_mass_due_to_frontal_melt",
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";
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";
552 m_vars[0][
"cell_methods"] =
"time: mean";
554 m_vars[0][
"comment"] =
"positive flux corresponds to ice gain";
576 kind ==
AMOUNT ?
"tendency_of_ice_amount_due_to_forced_retreat" :
577 "tendency_of_ice_mass_due_to_forced_retreat",
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";
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";
596 .
long_name(
"forced (prescribed) retreat flux")
597 .
units(internal_units)
599 m_vars[0][
"cell_methods"] =
"time: mean";
601 m_vars[0][
"comment"] =
"positive flux corresponds to ice gain";
628 double thickness_threshold) {
630 auto grid = ice_thickness.
grid();
631 auto ctx = grid->ctx();
632 auto EC = ctx->enthalpy_converter();
634 auto cell_area = grid->cell_area();
635 const auto& z = grid->z();
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);
655 return temperate ? V : 0.0;
658 return (not temperate) ? V : 0.0;
665 for (
auto p = grid->points(); p; p.next()) {
666 const int i = p.i(), j = p.j();
668 double H = ice_thickness(i, j);
670 if (H >= thickness_threshold) {
671 auto ks = grid->kBelowHeight(H);
672 const double *E = ice_enthalpy.
get_column(i, j);
674 for (
unsigned int k = 0;
k < ks; ++
k) {
675 volume += volume_counter(z[
k], z[
k + 1], H, E[
k]);
678 volume += volume_counter(z[ks], H, H, E[ks]);
692 double thickness_threshold) {
694 auto grid = ice_thickness.
grid();
695 auto ctx = grid->ctx();
696 auto EC = ctx->enthalpy_converter();
698 auto cell_area = grid->cell_area();
705 for (
auto p = grid->points(); p; p.next()) {
706 const int i = p.i(), j = p.j();
708 double thickness = ice_thickness(i, j);
710 if (thickness >= thickness_threshold) {
711 double basal_enthalpy = ice_enthalpy.
get_column(i, j)[0];
713 bool temperate = EC->is_temperate_relaxed(basal_enthalpy,
714 EC->pressure(thickness));
718 area += temperate ? cell_area : 0.0;
722 area += (not temperate) ? cell_area : 0.0;
741namespace diagnostics {
760 m_vars = { {
m_sys,
"ice_margin_pressure_difference" } };
764 "vertically-averaged pressure difference at ice margins (including calving fronts)")
770 auto result = allocate<array::Scalar>(
"ice_margin_pressure_difference");
779 const double H_threshold =
m_config->get_number(
"stress_balance.ice_free_thickness_standard");
787 rho_ocean =
m_config->get_number(
"constants.sea_water.density"),
788 g =
m_config->get_number(
"constants.standard_gravity");
794 for (
auto p =
m_grid->points(); p; p.next()) {
795 const int i = p.i(), j = p.j();
797 double delta_p = 0.0;
801 double P_ice = 0.5 *
rho_ice *
g * H(i, j),
805 delta_p = P_ice - P_water;
810 (*result)(i, j) = delta_p;
827 m, flag ==
GROUNDED ?
"basal_mass_flux_grounded" :
"basal_mass_flux_floating",
830 assert(flag !=
BOTH);
832 auto ismip6 =
m_config->get_flag(
"output.ISMIP6");
834 std::string name, description, standard_name;
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" :
"";
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" :
"";
851 .
units(
"kg m^-2 s^-1")
853 m_vars[0][
"cell_methods"] =
"time: mean";
855 m_vars[0][
"comment"] =
"positive flux corresponds to ice gain";
861 const array::Scalar &input =
model->geometry_evolution().bottom_surface_mass_balance();
862 const auto &cell_type =
model->geometry().cell_type;
864 double ice_density =
m_config->get_number(
"constants.ice.density");
872 for (
auto p =
m_grid->points(); p; p.next()) {
873 const int i = p.i(), j = p.j();
877 }
else if (
m_kind ==
SHELF and cell_type.ocean(i, j)) {
894 virtual std::shared_ptr<array::Array>
compute_impl()
const;
902 .long_name(
"vertical average of ice hardness")
903 .set_units_without_validation(
905 m_vars[0][
"valid_min"] = { 0.0 };
907 m_vars[0][
"comment"] =
"units depend on the Glen exponent used by the flow law";
914 if (flow_law == NULL) {
916 if (flow_law == NULL) {
918 "Can't compute vertically-averaged hardness: no flow law is used.");
922 auto result = std::make_shared<array::Scalar>(
m_grid,
"hardav");
923 result->metadata() =
m_vars[0];
933 for (
auto p =
m_grid->points(); p; p.next()) {
934 const int i = p.i(), j = p.j();
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)) {
959 virtual std::shared_ptr<array::Array>
compute_impl()
const;
965 .long_name(
"processor rank")
967 .set_time_independent(
true)
973 auto result = std::make_shared<array::Scalar>(
m_grid,
"rank");
974 result->metadata() =
m_vars[0];
978 for (
auto p =
m_grid->points(); p; p.next()) {
979 (*result)(p.i(), p.j()) =
m_grid->rank();
991 virtual std::shared_ptr<array::Array>
compute_impl()
const;
997 .long_name(
"cts = E/E_s(p), so cold-temperate transition surface is at cts = 1")
1004 result->metadata() =
m_vars[0];
1018 virtual std::shared_ptr<array::Array>
compute_impl()
const;
1024 .long_name(
"ice temperature")
1025 .standard_name(
"land_ice_temperature")
1027 m_vars[0][
"valid_min"] = { 0.0 };
1032 std::shared_ptr<array::Array3D> result(
1034 result->metadata() =
m_vars[0];
1042 const double *Enthij;
1048 for (
auto p =
m_grid->points(); p; p.next()) {
1049 const int i = p.i(), j = p.j();
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));
1073 virtual std::shared_ptr<array::Array>
compute_impl()
const;
1081 .long_name(
"pressure-adjusted ice temperature (degrees above pressure-melting point)")
1083 m_vars[0][
"valid_max"] = {0};
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");
1091 result->metadata() =
m_vars[0];
1099 const double *Enthij;
1105 for (
auto pt =
m_grid->points(); pt; pt.next()) {
1106 const int i = pt.i(), j = pt.j();
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);
1117 if (EC->is_temperate_relaxed(Enthij[
k],p) && (thickness(i,j) > 0)) {
1118 Tij[
k] = melting_point_temp;
1129 result->shift(-melting_point_temp);
1140 virtual std::shared_ptr<array::Array>
compute_impl()
const;
1146 m_vars[0].long_name(
"pressure-adjusted ice temperature at the base of ice").units(
"degree_Celsius");
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");
1154 auto result = std::make_shared<array::Scalar>(
m_grid,
"temp_pa_base");
1155 result->metadata() =
m_vars[0];
1166 for (
auto pt =
m_grid->points(); pt; pt.next()) {
1167 const int i = pt.i(), j = pt.j();
1169 const auto *Enthij = enthalpy.get_column(i,j);
1171 const double depth = thickness(i,j),
1172 p = EC->pressure(depth);
1173 (*result)(i,j) = EC->pressure_adjusted_temperature(Enthij[0], p);
1177 if (EC->is_temperate_relaxed(Enthij[0],p) && (thickness(i,j) > 0)) {
1178 (*result)(i,j) = melting_point_temp;
1187 result->shift(-melting_point_temp);
1198 virtual std::shared_ptr<array::Array>
compute_impl()
const;
1204 m_vars[0].long_name(
"ice enthalpy at 1m below the ice surface").units(
"J kg^-1");
1210 auto result = std::make_shared<array::Scalar>(
m_grid,
"enthalpysurf");
1211 result->metadata() =
m_vars[0];
1220 for (
auto p =
m_grid->points(); p; p.next()) {
1221 const int i = p.i(), j = p.j();
1223 (*result)(i,j) = std::max(ice_thickness(i,j) - 1.0, 0.0);
1226 extract_surface(ice_enthalpy, *result, *result);
1228 for (
auto p =
m_grid->points(); p; p.next()) {
1229 const int i = p.i(), j = p.j();
1231 if (ice_thickness(i,j) <= 1.0) {
1245 virtual std::shared_ptr<array::Array>
compute_impl()
const;
1251 m_vars[0].long_name(
"ice enthalpy at the base of ice").units(
"J kg^-1");
1257 auto result = std::make_shared<array::Scalar>(
m_grid,
"enthalpybase");
1258 result->metadata() =
m_vars[0];
1281 std::string name, long_name, standard_name;
1282 switch (area_type) {
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";
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";
1295 long_name =
"ice temperature at the base of ice";
1296 standard_name =
"land_ice_basal_temperature";
1301 m_vars[0].long_name(long_name).standard_name(standard_name).units(
"kelvin");
1307 auto result = allocate<array::Scalar>(
"basal_temperature");
1322 for (
auto p =
m_grid->points(); p; p.next()) {
1323 const int i = p.i(), j = p.j();
1325 double depth = thickness(i, j), pressure = EC->pressure(depth),
1326 T = EC->temperature((*result)(i, j), pressure);
1331 (*result)(i, j) = T;
1350 virtual std::shared_ptr<array::Array>
compute_impl()
const;
1356 .long_name(
"ice temperature at 1m below the ice surface")
1357 .standard_name(
"temperature_at_ground_level_in_snow_or_firn")
1367 auto result = array::cast<array::Scalar>(enth);
1376 double depth = 1.0, pressure = EC->pressure(depth);
1379 for (
auto p =
m_grid->points(); p; p.next()) {
1380 const int i = p.i(), j = p.j();
1382 if (thickness(i, j) > 1) {
1383 (*result)(i, j) = EC->temperature((*result)(i, j), pressure);
1393 result->metadata(0) =
m_vars[0];
1403 virtual std::shared_ptr<array::Array>
compute_impl()
const;
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 };
1414 std::shared_ptr<array::Array3D> result(
1416 result->metadata(0) =
m_vars[0];
1418 bool cold_mode =
member(
m_config->get_string(
"energy.model"), {
"cold",
"none"});
1436 virtual std::shared_ptr<array::Array>
compute_impl()
const;
1441 m_vars[0].long_name(
"temperate ice thickness (total column content)").units(
"m");
1447 auto result = allocate<array::Scalar>(
"tempicethk");
1459 for (
auto p =
m_grid->points(); p; p.next()) {
1460 const int i = p.i(), j = p.j();
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);
1468 for (
unsigned int k = 0;
k < ks; ++
k) {
1469 double pressure = EC->pressure(H -
m_grid->z(
k));
1471 if (EC->is_temperate_relaxed(Enth[
k], pressure)) {
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);
1481 (*result)(i, j) = H_temperate;
1501 virtual std::shared_ptr<array::Array>
compute_impl()
const;
1506 m_vars[0].long_name(
"thickness of the basal layer of temperate ice").units(
"m");
1515 auto result = allocate<array::Scalar>(
"tempicethk_basal");
1527 for (
auto p =
m_grid->points(); p; p.next()) {
1528 const int i = p.i(), j = p.j();
1530 double H = ice_thickness(i, j);
1534 if (cell_type.ice_free(i, j)) {
1539 const double *Enth = ice_enthalpy.get_column(i, j);
1541 unsigned int ks =
m_grid->kBelowHeight(H);
1544 double pressure = EC->pressure(H -
m_grid->z(
k));
1546 pressure = EC->pressure(H -
m_grid->z(
k));
1548 if (EC->is_temperate_relaxed(Enth[
k], pressure)) {
1559 (*result)(i, j) = 0.0;
1566 (*result)(i, j) =
m_grid->z(ks);
1571 slope1 = (Enth[
k] - Enth[
k - 1]) / dz,
1572 slope2 = (EC->enthalpy_cts(pressure) - EC->enthalpy_cts(pressure_0)) / dz;
1574 if (slope1 != slope2) {
1576 m_grid->z(
k - 1) + (EC->enthalpy_cts(pressure_0) - Enth[
k - 1]) / (slope1 - slope2);
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));
1583 "Linear interpolation of the thickness of"
1584 " the basal temperate layer failed:\n"
1585 "(i=%d, j=%d, k=%d, ks=%d)\n",
1607 m_variable[
"long_name"] =
"volume of the ice in glacierized areas";
1612 m_config->get_number(
"output.ice_free_thickness_standard"));
1622 m_variable[
"long_name"] =
"volume of the ice, including seasonal cover";
1638 m_variable[
"long_name"] =
"the sea level rise that would result if all the ice were melted";
1644 m_config->get_number(
"output.ice_free_thickness_standard"));
1655 m_variable[
"long_name"] =
"rate of change of the ice volume in glacierized areas";
1660 m_config->get_number(
"output.ice_free_thickness_standard"));
1671 m_variable[
"long_name"] =
"rate of change of the ice volume, including seasonal cover";
1686 m_variable[
"long_name"] =
"glacierized area";
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";
1709 const double thickness_standard =
m_config->get_number(
"output.ice_free_thickness_standard"),
1710 ice_density =
m_config->get_number(
"constants.ice.density"),
1726 m_variable[
"long_name"] =
"mass of the ice in glacierized areas";
1731 double ice_density =
m_config->get_number(
"constants.ice.density"),
1732 thickness_standard =
m_config->get_number(
"output.ice_free_thickness_standard");
1742 if (
m_config->get_flag(
"output.ISMIP6")) {
1747 m_variable[
"long_name"] =
"mass of the ice, including seasonal cover";
1748 m_variable[
"standard_name"] =
"land_ice_mass";
1764 m_variable[
"long_name"] =
"rate of change of the ice mass in glacierized areas";
1768 double ice_density =
m_config->get_number(
"constants.ice.density"),
1769 thickness_threshold =
m_config->get_number(
"output.ice_free_thickness_standard");
1785 m_variable[
"long_name"] =
"rate of change of the mass of ice due to flow"
1786 " (i.e. prescribed ice thickness)";
1791 const double ice_density =
m_config->get_number(
"constants.ice.density");
1796 auto cell_area =
m_grid->cell_area();
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();
1804 volume_change += (dH(i, j) + dV(i, j)) * cell_area;
1818 m_variable[
"long_name"] =
"rate of change of the mass of ice, including seasonal cover";
1822 const double ice_density =
m_config->get_number(
"constants.ice.density");
1835 m_variable[
"long_name"] =
"volume of temperate ice in glacierized areas";
1840 auto thickness_threshold =
m_config->get_number(
"output.ice_free_thickness_standard");
1843 thickness_threshold);
1854 m_variable[
"long_name"] =
"volume of temperate ice, including seasonal cover";
1872 m_variable[
"long_name"] =
"volume of cold ice in glacierized areas";
1877 auto thickness_threshold =
m_config->get_number(
"output.ice_free_thickness_standard");
1880 thickness_threshold);
1890 m_variable[
"long_name"] =
"volume of cold ice, including seasonal cover";
1908 m_variable[
"long_name"] =
"glacierized area where basal ice is temperate";
1913 auto thickness_threshold =
m_config->get_number(
"output.ice_free_thickness_standard");
1916 thickness_threshold);
1927 m_variable[
"long_name"] =
"glacierized area where basal ice is cold";
1932 auto thickness_threshold =
m_config->get_number(
"output.ice_free_thickness_standard");
1935 thickness_threshold);
1946 m_variable[
"long_name"] =
"enthalpy of the ice in glacierized areas";
1963 m_variable[
"long_name"] =
"enthalpy of the ice, including seasonal cover";
1979 if (
m_config->get_flag(
"output.ISMIP6")) {
1984 m_variable[
"long_name"] =
"area of grounded ice in glacierized areas";
1985 m_variable[
"standard_name"] =
"grounded_ice_sheet_area";
1991 m_config->get_number(
"output.ice_free_thickness_standard"));
2001 if (
m_config->get_flag(
"output.ISMIP6")) {
2006 m_variable[
"long_name"] =
"area of ice shelves in glacierized areas";
2007 m_variable[
"standard_name"] =
"floating_ice_shelf_area";
2013 m_config->get_number(
"output.ice_free_thickness_standard"));
2024 m_variable[
"long_name"] =
"volume of grounded ice in glacierized areas";
2033 const double thickness_threshold =
m_config->get_number(
"output.ice_free_thickness_standard"),
2034 cell_area =
m_grid->cell_area();
2038 double volume = 0.0;
2039 for (
auto p =
m_grid->points(); p; p.next()) {
2040 const int i = p.i(), j = p.j();
2042 const double H = ice_thickness(i, j);
2044 if (cell_type.grounded(i, j) and H >= thickness_threshold) {
2045 volume += cell_area * H;
2060 m_variable[
"long_name"] =
"volume of ice shelves in glacierized areas";
2069 const double thickness_threshold =
m_config->get_number(
"output.ice_free_thickness_standard"),
2070 cell_area =
m_grid->cell_area();
2074 double volume = 0.0;
2075 for (
auto p =
m_grid->points(); p; p.next()) {
2076 const int i = p.i(), j = p.j();
2078 const double H = ice_thickness(i, j);
2080 if (cell_type.ocean(i, j) and H >= thickness_threshold) {
2081 volume += cell_area * H;
2095 m_variable[
"long_name"] =
"mass continuity time step";
2110 m_variable[
"long_name"] =
"ratio of max. allowed time steps "
2111 "according to CFL and SIA diffusivity criteria";
2121 auto cfl_3d = stress_balance->max_timestep_cfl_3d();
2126 m_config->get_number(
"time_stepping.adaptive_ratio"));
2128 auto dt_cfl = std::min(cfl_2d.dt_max, cfl_3d.dt_max);
2130 if (dt_cfl.finite() and dt_diff.finite() and dt_diff.value() > 0.0) {
2131 return dt_cfl.value() / dt_diff.value();
2144 m_variable[
"long_name"] =
"maximum diffusivity of the flow (usually from the SIA model)";
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";
2208 const Config &config = *grid.ctx()->config();
2210 const double ice_density = config.
get_number(
"constants.ice.density"),
2211 cell_area = grid.cell_area();
2244 double volume_change = 0.0;
2245 for (
auto p = grid.points(); p; p.next()) {
2246 const int i = p.i(), j = p.j();
2248 if ((area ==
BOTH) or (area ==
GROUNDED and cell_type.grounded(i, j)) or
2249 (area ==
SHELF and cell_type.ocean(i, j))) {
2251 double dV = term ==
FLOW ? dV_flow(i, j) : 0.0;
2254 volume_change += cell_area * ((*thickness_change)(i, j) + dV);
2259 return ice_density *
GlobalSum(grid.com, volume_change);
2268 if (
m_config->get_flag(
"output.ISMIP6")) {
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";
2289 if (
m_config->get_flag(
"output.ISMIP6")) {
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";
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";
2327 if (
m_config->get_flag(
"output.ISMIP6")) {
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";
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";
2366 if (
m_config->get_flag(
"output.ISMIP6")) {
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";
2377 const double ice_density =
m_config->get_number(
"constants.ice.density");
2383 auto cell_area =
m_grid->cell_area();
2385 double volume_change = 0.0;
2389 for (
auto p =
m_grid->points(); p; p.next()) {
2390 const int i = p.i(), j = p.j();
2392 volume_change += cell_area * (calving(i, j) + frontal_melt(i, j) + forced_retreat(i, j));
2406 if (
m_config->get_flag(
"output.ISMIP6")) {
2412 m_variable[
"standard_name"] =
"tendency_of_land_ice_mass_due_to_calving";
2413 m_variable[
"comment"] =
"positive means ice gain";
2417 const double ice_density =
m_config->get_number(
"constants.ice.density");
2421 auto cell_area =
m_grid->cell_area();
2423 double volume_change = 0.0;
2427 for (
auto p =
m_grid->points(); p; p.next()) {
2428 const int i = p.i(), j = p.j();
2430 volume_change += cell_area * calving(i, j);
2444 if (
m_config->get_flag(
"output.ISMIP6")) {
2446 m_variable[
"standard_name"] =
"tendency_of_grounded_ice_mass";
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";
2469 auto ismip6 =
m_config->get_flag(
"output.ISMIP6");
2472 m_vars = { {
m_sys, ismip6 ?
"dlithkdt" :
"dHdt" } };
2474 .long_name(
"ice thickness rate of change")
2475 .standard_name(
"tendency_of_land_ice_thickness")
2477 .output_units(
"m year^-1");
2481 m_vars[0][
"valid_range"] = { -large_number, large_number };
2483 m_vars[0][
"cell_methods"] =
"time: mean";
2487 "ice thickness at the time of the last report of the rate of change of ice thickness")
2495 auto result = std::make_shared<array::Scalar>(
m_grid,
"dHdt");
2496 result->metadata() =
m_vars[0];
2527 virtual std::shared_ptr<array::Array>
compute_impl()
const;
2533 const std::string &proj_string)
2535 assert(var_name ==
"lat" || var_name ==
"lon");
2540 m_vars[0].z().clear().set_name(
"nv4");
2542 m_vars[0].set_time_independent(
true);
2544 m_vars[0].long_name(
"longitude bounds").units(
"degree_east");
2545 m_vars[0][
"valid_range"] = { -180, 180 };
2547 m_vars[0].long_name(
"latitude bounds").units(
"degree_north");
2548 m_vars[0][
"valid_range"] = { -90, 90 };
2553#if (Pism_USE_PROJ == 1)
2565 result->metadata(0) =
m_vars[0];
2581 virtual std::shared_ptr<array::Array>
compute_impl()
const;
2587 .long_name(
"fraction of a grid cell covered by ice (grounded or floating)")
2588 .standard_name(
"land_ice_area_fraction")
2602 array::AccessScope list{ &thickness, &surface_elevation, &bed_topography, &cell_type,
2605 const bool do_part_grid =
m_config->get_flag(
"geometry.part_grid.enabled");
2614 for (
auto p =
m_grid->points(); p; p.next()) {
2615 const int i = p.i(), j = p.j();
2617 if (cell_type.
icy(i, j)) {
2619 (*result)(i, j) = 1.0;
2624 double H_reference = do_part_grid ? Href(i, j) : 0.0;
2626 if (H_reference > 0.0) {
2627 const double H_threshold =
2629 surface_elevation.star(i, j), bed_topography(i, j));
2631 if (H_threshold > 0.0) {
2632 (*result)(i, j) = std::min(H_reference / H_threshold, 1.0);
2634 (*result)(i, j) = 1.0;
2638 (*result)(i, j) = 0.0;
2642 (*result)(i, j) = 0.0;
2658 virtual std::shared_ptr<array::Array>
compute_impl()
const;
2664 .long_name(
"fraction of a grid cell covered by grounded ice")
2665 .standard_name(
"grounded_ice_sheet_area_fraction")
2671 result->metadata() =
m_vars[0];
2673 const double ice_density =
m_config->get_number(
"constants.ice.density"),
2674 ocean_density =
m_config->get_number(
"constants.sea_water.density");
2683 bed_topography, *result);
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;
2711 virtual std::shared_ptr<array::Array>
compute_impl()
const;
2717 .long_name(
"fraction of a grid cell covered by floating ice")
2718 .standard_name(
"floating_ice_shelf_area_fraction")
2727 auto result = ice_area_fraction;
2728 result->metadata() =
m_vars[0];
2742 virtual std::shared_ptr<array::Array>
compute_impl()
const;
2749 m_vars[0].long_name(
"ice thickness in excess of the maximum floating ice thickness").units(
"m");
2751 m_vars[0][
"comment"] =
"shows how close to floatation the ice is at a given location";
2756 auto result = allocate<array::Scalar>(
"height_above_flotation");
2760 const double ice_density =
m_config->get_number(
"constants.ice.density"),
2761 ocean_density =
m_config->get_number(
"constants.sea_water.density");
2767 array::AccessScope list{ &cell_type, result.get(), &ice_thickness, &bed_topography, &sea_level };
2771 for (
auto p =
m_grid->points(); p; p.next()) {
2772 const int i = p.i(), j = p.j();
2774 const double thickness = ice_thickness(i, j), bed = bed_topography(i, j),
2775 ocean_depth = sea_level(i, j) - bed;
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;
2798 virtual std::shared_ptr<array::Array>
compute_impl()
const;
2803 m_vars[0].long_name(
"ice mass per cell").units(
"kg");
2809 auto result = allocate<array::Scalar>(
"ice_mass");
2813 const double ice_density =
m_config->get_number(
"constants.ice.density");
2817 auto cell_area =
m_grid->cell_area();
2823 for (
auto p =
m_grid->points(); p; p.next()) {
2824 const int i = p.i(), j = p.j();
2828 if (ice_thickness(i, j) > 0.0) {
2829 (*result)(i, j) = ice_density * ice_thickness(i, j) * cell_area;
2841 if (
m_config->get_flag(
"geometry.part_grid.enabled")) {
2844 for (
auto p =
m_grid->points(); p; p.next()) {
2845 const int i = p.i(), j = p.j();
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;
2868 m_vars[0].long_name(
"sea-level adjusted bed topography (zero is at sea level)").units(
"meters");
2873 auto result = allocate<array::Scalar>(
"topg_sl_adjusted");
2880 for (
auto p =
m_grid->points(); p; p.next()) {
2881 const int i = p.i(), j = p.j();
2883 (*result)(i, j) = bed(i, j) - sea_level(i, j);
2901 .long_name(
"ice hardness computed using the SIA flow law")
2902 .set_units_without_validation(
2904 m_vars[0][
"comment"] =
"units depend on the Glen exponent used by the flow law";
2909 std::shared_ptr<array::Array3D> result(
2911 result->metadata(0) =
m_vars[0];
2922 const unsigned int Mz =
m_grid->Mz();
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);
2931 double *hardness = result->get_column(i, j);
2933 for (
unsigned int k = 0;
k < Mz; ++
k) {
2934 const double depth = H -
m_grid->z(
k);
2937 const double pressure = EC->pressure(depth);
2939 hardness[
k] = flow_law.
hardness(E[
k], pressure);
2962 .long_name(
"effective viscosity of ice")
2963 .units(
"Pascal second")
2964 .output_units(
"kPascal second");
2965 m_vars[0][
"valid_min"] = { 0.0 };
2975 std::shared_ptr<array::Array3D> result(
2977 result->metadata(0) =
m_vars[0];
2996 const unsigned int Mz =
m_grid->Mz();
2998 const std::vector<double> &z =
m_grid->z();
3002 array::AccessScope list{ &U, &V, &W, &ice_enthalpy, &ice_thickness, &mask, result.get() };
3006 for (
auto p =
m_grid->points(); p; p.next()) {
3007 const int i = p.i(), j = p.j();
3009 const double *E = ice_enthalpy.
get_column(i, j);
3010 const double H = ice_thickness(i, j);
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);
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);
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;
3028 double *viscosity = result->get_column(i, j);
3030 if (ice_free(m.c)) {
3035 for (
unsigned int k = 0;
k < Mz; ++
k) {
3036 const double depth = H - z[
k];
3044 const double pressure = EC->pressure(depth);
3046 const double hardness = flow_law.
hardness(E[
k], pressure);
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]));
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]));
3064 double u_z = 0.0, v_z = 0.0, w_z = 0.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;
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);
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);
3114 auto ismip6 =
m_config->get_flag(
"output.ISMIP6");
3118 m_vars[0].long_name(
"land ice thickness").standard_name(
"land_ice_thickness").units(
"m");
3119 m_vars[0][
"valid_min"] = { 0.0 };
3125 auto result = allocate<array::Scalar>(
"thk");
3138 auto ismip6 =
m_config->get_flag(
"output.ISMIP6");
3140 m_vars = { {
m_sys, ismip6 ?
"base" :
"ice_base_elevation" } };
3141 m_vars[0].long_name(
"ice bottom surface elevation").units(
"m");
3147 auto result = allocate<array::Scalar>(
"ice_base_elevation");
3160 auto ismip6 =
m_config->get_flag(
"output.ISMIP6");
3162 m_vars = { {
m_sys, ismip6 ?
"orog" :
"usurf" } };
3163 m_vars[0].long_name(
"ice top surface elevation").standard_name(
"surface_altitude").units(
"m");
3168 auto result = allocate<array::Scalar>(
"usurf");
3183 auto ismip6 =
m_config->get_flag(
"output.ISMIP6");
3185 m_vars = { {
m_sys, ismip6 ?
"ligroundf" :
"grounding_line_flux" } };
3189 .
units(
"kg m^-2 second^-1")
3191 m_vars[0][
"cell_methods"] =
"time: mean";
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";
3203 auto cell_area = grid->cell_area();
3205 grid->ctx()->config()->get_number(
"constants.ice.density");
3208 double unit_conversion_factor = dt * (ice_density / cell_area);
3211 model->geometry_evolution().flux_staggered(),
3225 m_vars = { {
m_sys,
"ice_mass_transport_across_grounding_line" } };
3228 .
long_name(
"ice mass flow rate across the grounding line")
3231 m_vars[0][
"cell_methods"] =
"time: mean";
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.";
3241 auto grid =
model->grid();
3242 double ice_density =
3243 grid->ctx()->config()->get_number(
"constants.ice.density");
3246 double unit_conversion_factor = dt * ice_density;
3249 model->geometry_evolution().flux_staggered(),
3261 using namespace diagnostics;
3268 {
"height_above_flotation", f(
new HeightAboveFloatation(
this))},
3270 {
"ice_mass", f(
new IceMass(
this))},
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))},
3283 {
"enthalpybase", f(
new IceEnthalpyBasal(
this))},
3284 {
"enthalpysurf", f(
new IceEnthalpySurface(
this))},
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))},
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))},
3304 {
"ice_margin_pressure_difference", f(
new IceMarginPressureDifference(
this))},
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))},
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))},
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))},
3354 {
"grounding_line_flux", f(
new GroundingLineFlux(
this))},
3355 {
"ice_mass_transport_across_grounding_line", f(
new MassTransportAcrossGroundingLine(
this))},
3358 {
"rank", f(
new Rank(
this))},
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));
3370 if (
m_config->get_flag(
"output.ISMIP6")) {
3380 m_diagnostics[
"litempbotgr"] = f(
new TemperatureBasal(
this, GROUNDED));
3381 m_diagnostics[
"litempbotfl"] = f(
new TemperatureBasal(
this, SHELF));
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))},
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))},
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))},
3411 {
"ice_enthalpy_glacierized", s(
new scalar::IceEnthalpyGlacierized(
this))},
3412 {
"ice_enthalpy", s(
new scalar::IceEnthalpy(
this))},
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))},
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))},
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))},
3433 if (
m_config->get_flag(
"output.ISMIP6")) {
3452typedef std::map<std::string, std::vector<VariableMetadata>>
Metadata;
3455 for (
const auto &d : list) {
3456 const std::string &name = d.first;
3457 log.
message(1,
" Name: %s\n", name.c_str());
3459 for (
const auto &v : d.second) {
3462 var_name = v.get_name(),
3464 output_units = v[
"output_units"],
3465 long_name = v[
"long_name"],
3466 comment = v[
"comment"];
3468 if (not output_units.empty()) {
3469 units = output_units;
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());
3484 bool first_diagnostic =
true;
3485 for (
const auto &d : list) {
3487 if (not first_diagnostic) {
3490 first_diagnostic =
false;
3493 log.
message(1,
"\"%s\" : [\n", d.first.c_str());
3495 bool first_variable =
true;
3496 for (
const auto &variable : d.second) {
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"];
3506 if (not output_units.empty()) {
3507 units = output_units;
3510 if (not first_variable) {
3513 first_variable =
false;
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());
3530 for (
const auto& f : diags) {
3533 for (
unsigned int k = 0;
k < diag->n_variables(); ++
k) {
3534 result[f.first].push_back(diag->metadata(
k));
3547 for (
const auto& d : ts_diags) {
3549 result[d.first] = {d.second->metadata()};
3557 if (list_type ==
"json") {
3558 m_log->message(1,
"{\n");
3560 m_log->message(1,
"\"spatial\" :\n");
3563 m_log->message(1,
",\n");
3565 m_log->message(1,
"\"scalar\" :\n");
3568 m_log->message(1,
"}\n");
3573 if (
member(list_type, {
"all",
"spatial"})) {
3574 m_log->message(1,
"\n");
3575 m_log->message(1,
"======== Available 2D and 3D diagnostics ========\n");
3580 if (
member(list_type, {
"all",
"scalar"})) {
3582 m_log->message(1,
"======== Available time-series ========\n");
3597 auto cell_area =
m_grid->cell_area();
3599 double result = 0.0, meltarea = 0.0;
3606 for (
auto p =
m_grid->points(); p; p.next()) {
3607 const int i = p.i(), j = p.j();
3614 if (EC->is_temperate_relaxed(E_basal, pressure)) {
3615 meltarea += cell_area;
3630 if (total_ice_area > 0.0) {
3631 result = result / total_ice_area;
3645 double result = -1.0;
3651 const double a =
m_grid->cell_area() * 1e-3 * 1e-3,
3652 currtime =
m_time->current();
3659 double original_ice_volume = 0.0;
3664 for (
auto p =
m_grid->points(); p; p.next()) {
3665 const int i = p.i(), j = p.j();
3669 const double *age = ice_age.
get_column(i, j);
3671 for (
unsigned int k = 1;
k <= ks;
k++) {
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));
3689 if (total_ice_volume > 0.0) {
3690 result = result / total_ice_volume;
double get_number(const std::string &name, UseFlag flag=REMEMBER_THIS_USE) const
A class for storing and accessing PISM configuration flags and parameters.
array::Scalar m_accumulator
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
std::vector< SpatialVariableMetadata > m_vars
metadata corresponding to NetCDF variables
std::shared_ptr< Diagnostic > Ptr
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.
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
void set_icefree_thickness(double threshold)
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
array::Scalar cell_grounded_fraction
array::Scalar2 ice_surface_elevation
array::Scalar1 ice_area_specific_volume
array::CellType2 cell_type
array::Scalar2 ice_thickness
array::Scalar2 bed_elevation
Describes the PISM grid and the distribution of data across processors.
std::map< std::string, const Component * > m_submodels
the list of sub-models, for writing model states and obtaining diagnostics
virtual double compute_temperate_base_fraction(double ice_area)
const Geometry & geometry() const
std::map< std::string, TSDiagnostic::Ptr > m_ts_diagnostics
Requested scalar diagnostics.
const Time::Ptr m_time
Time manager.
const GeometryEvolution & geometry_evolution() const
const stressbalance::StressBalance * stress_balance() const
virtual double compute_original_ice_fraction(double ice_volume)
std::shared_ptr< Context > m_ctx
Execution context.
array::Scalar m_basal_melt_rate
rate of production of basal meltwater (ice-equivalent); no ghosts
std::shared_ptr< Grid > grid() const
Return the grid used by this model.
const Logger::Ptr m_log
Logger.
array::Scalar m_bedtoptemp
temperature at the top surface of the bedrock thermal layer
const energy::EnergyModel * energy_balance_model() const
const array::Scalar & frontal_melt() const
const array::Scalar & forced_retreat() const
array::Scalar2 m_velocity_bc_mask
mask to determine Dirichlet boundary locations for the sliding velocity
Config::Ptr m_config
Configuration flags and parameters.
array::Scalar1 m_ice_thickness_bc_mask
Mask prescribing locations where ice thickness is held constant.
std::shared_ptr< AgeModel > m_age_model
const units::System::Ptr m_sys
Unit system.
array::Vector2 m_velocity_bc_values
Dirichlet boundary velocities.
void list_diagnostics(const std::string &list_type) const
std::map< std::string, Diagnostic::Ptr > m_diagnostics
Requested spatially-variable diagnostics.
const array::Scalar & calving() const
std::shared_ptr< Context > ctx() const
Return the context this model is running in.
virtual void init_diagnostics()
std::shared_ptr< energy::EnergyModel > m_energy_model
const std::shared_ptr< Grid > m_grid
Computational grid.
void message(int threshold, const char format[],...) const __attribute__((format(printf
Print a message to the log.
void failed()
Indicates a failure of a parallel section.
A wrapper for PJ that makes sure pj_destroy is called.
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.
Makes sure that we call begin_access() and end_access() for all accessed array::Arrays.
void copy_from(const Array2D< T > &source)
stencils::Star< T > star(int i, int j) const
void add(double alpha, const Array2D< T > &x)
void copy_from(const Array3D &input)
double * get_column(int i, int j)
A virtual class collecting methods common to ice and bedrock 3D fields.
std::shared_ptr< const Grid > grid() const
SpatialVariableMetadata & metadata(unsigned int N=0)
Returns a reference to the SpatialVariableMetadata object containing metadata for the compoment N.
stencils::Star< int > star_int(int i, int j) const
bool next_to_ice_free_ocean(int i, int j) const
bool ice_free_ocean(int i, int j) const
bool grounded_ice(int i, int j) const
bool icy(int i, int j) const
BMBSplit(const IceModel *m, AreaType flag)
void update_impl(double dt)
Report average basal mass balance flux over the reporting interval (grounded or floating areas)
BasalFlux(const IceModel *m, AmountKind kind)
void update_impl(double dt)
Report basal mass balance flux, averaged over the reporting interval.
BedTopographySeaLevelAdjusted(const IceModel *m)
std::shared_ptr< array::Array > compute_impl() const
Sea-level adjusted bed topography (zero at sea level).
virtual std::shared_ptr< array::Array > compute_impl() const
Computes CTS, CTS = E/E_s(p).
void update_impl(double dt)
CalvingFlux(const IceModel *m, AmountKind kind)
void update_impl(double dt)
ConservationErrorFlux(const IceModel *m, AmountKind kind)
void update_impl(double dt)
DischargeFlux(const IceModel *m, AmountKind kind)
Report discharge (calving and frontal melt) flux.
ForcedRetreatFlux(const IceModel *m, AmountKind kind)
void update_impl(double dt)
Report the frontal melt flux.
FrontalMeltFlux(const IceModel *m, AmountKind kind)
void update_impl(double dt)
Report the frontal melt flux.
void update_impl(double dt)
GroundingLineFlux(const IceModel *m)
Report grounding line flux.
HardnessAverage(const IceModel *m)
virtual std::shared_ptr< array::Array > compute_impl() const
Computes vertically-averaged ice hardness.
Computes vertically-averaged ice hardness.
HeightAboveFloatation(const IceModel *m)
virtual std::shared_ptr< array::Array > compute_impl() const
Computes the 2D height above flotation.
IceAreaFractionFloating(const IceModel *m)
virtual std::shared_ptr< array::Array > compute_impl() const
IceAreaFractionGrounded(const IceModel *m)
virtual std::shared_ptr< array::Array > compute_impl() const
IceAreaFraction(const IceModel *m)
virtual std::shared_ptr< array::Array > compute_impl() const
std::shared_ptr< array::Array > compute_impl() const
IceBottomSurfaceElevation(const IceModel *m)
Report ice top surface elevation.
virtual std::shared_ptr< array::Array > compute_impl() const
IceEnthalpyBasal(const IceModel *m)
Computes enthalpy at the base of the ice.
IceEnthalpySurface(const IceModel *m)
virtual std::shared_ptr< array::Array > compute_impl() const
Computes surface values of ice enthalpy.
IceHardness(const IceModel *m)
std::shared_ptr< array::Array > compute_impl() const
Ice hardness computed using the SIA flow law.
std::shared_ptr< array::Array > compute_impl() const
IceMarginPressureDifference(IceModel *m)
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.
IceSurfaceElevation(const IceModel *m)
std::shared_ptr< array::Array > compute_impl() const
Report ice top surface elevation.
IceThickness(const IceModel *m)
std::shared_ptr< array::Array > compute_impl() const
std::shared_ptr< array::Array > compute_impl() const
IceViscosity(IceModel *m)
Effective viscosity of ice (3D).
LatLonBounds(const IceModel *m, const std::string &var_name, const std::string &proj_string)
std::string m_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
LiquidFraction(const IceModel *m)
Computes the liquid water fraction.
MassTransportAcrossGroundingLine(const IceModel *m)
void update_impl(double dt)
virtual std::shared_ptr< array::Array > compute_impl() const
Computes a diagnostic field filled with processor rank values.
SurfaceFlux(const IceModel *m, AmountKind kind)
void update_impl(double dt)
Report surface mass balance flux, averaged over the reporting interval.
TemperateIceThicknessBasal(const IceModel *m)
virtual std::shared_ptr< array::Array > compute_impl() const
Computes the thickness of the basal layer of temperate ice.
TemperateIceThickness(const IceModel *m)
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
TemperaturePABasal(const IceModel *m)
Computes basal values of the pressure-adjusted temperature.
virtual std::shared_ptr< array::Array > compute_impl() const
TemperaturePA(const IceModel *m)
Compute the pressure-adjusted temperature in degrees C corresponding to ice temperature.
virtual std::shared_ptr< array::Array > compute_impl() const
TemperatureSurface(const IceModel *m)
Computes ice temperature at the surface of the ice.
virtual std::shared_ptr< array::Array > compute_impl() const
Temperature(const IceModel *m)
Computes ice temperature from enthalpy.
void update_impl(double dt)
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
void update_impl(double dt)
TendencyOfIceAmount(const IceModel *Model, AmountKind kind)
array::Scalar m_last_amount
Computes tendency_of_ice_amount, the ice amount rate of change.
array::Scalar m_last_thickness
ThicknessRateOfChange(const IceModel *m)
void update_impl(double dt)
std::shared_ptr< array::Array > compute_impl() const
Computes dHdt, the ice thickness rate of change.
IceAreaGlacierizedColdBase(IceModel *m)
Computes the total area of the cold ice.
IceAreaGlacierizedGrounded(IceModel *m)
Computes the total grounded ice area.
IceAreaGlacierizedShelf(IceModel *m)
Computes the total floating ice area.
IceAreaGlacierizedTemperateBase(IceModel *m)
Computes the total area of the temperate ice.
IceAreaGlacierized(IceModel *m)
Computes the total ice area.
IceEnthalpyGlacierized(IceModel *m)
Computes the total ice enthalpy in glacierized areas.
Computes the total ice enthalpy.
IceMassFluxAtGroundingLine(const IceModel *m)
Reports the total flux across the grounding line.
IceMassFluxBasalFloating(const IceModel *m)
Reports the total sub-shelf ice flux.
IceMassFluxBasalGrounded(const IceModel *m)
Reports the total basal ice flux over the grounded region.
IceMassFluxBasal(const IceModel *m)
Reports the total bottom surface ice flux.
IceMassFluxCalving(const IceModel *m)
Reports the total calving flux.
IceMassFluxConservationError(const IceModel *m)
Reports the total numerical mass flux needed to preserve non-negativity of ice thickness.
IceMassFluxDischarge(const IceModel *m)
Reports the total discharge flux.
IceMassFluxSurface(const IceModel *m)
Reports the total top surface ice flux.
IceMassGlacierized(IceModel *m)
Computes the total ice mass in glacierized areas.
IceMassNotDisplacingSeaWater(const IceModel *m)
Computes the total mass of the ice not displacing sea water.
IceMassRateOfChangeDueToFlow(IceModel *m)
Computes the rate of change of the total ice mass due to flow (influx due to prescribed constant-in-t...
IceMassRateOfChangeGlacierized(IceModel *m)
Computes the rate of change of the total ice mass in glacierized areas.
IceMassRateOfChange(IceModel *m)
Computes the rate of change of the total ice mass.
Computes the total ice mass.
IceVolumeCold(IceModel *m)
Computes the total volume of the cold ice.
IceVolumeGlacierizedCold(IceModel *m)
Computes the total volume of the cold ice in glacierized areas.
IceVolumeGlacierizedGrounded(IceModel *m)
Computes the total grounded ice volume.
IceVolumeGlacierizedShelf(IceModel *m)
Computes the total floating ice volume.
IceVolumeGlacierizedTemperate(IceModel *m)
Computes the total volume of the temperate ice in glacierized areas.
IceVolumeGlacierized(IceModel *m)
Computes the total ice volume in glacierized areas.
IceVolumeRateOfChangeGlacierized(IceModel *m)
Computes the rate of change of the total ice volume in glacierized areas.
IceVolumeRateOfChange(IceModel *m)
Computes the rate of change of the total ice volume.
IceVolumeTemperate(IceModel *m)
Computes the total volume of the temperate ice.
Computes the total ice volume.
MaxDiffusivity(const IceModel *m)
Reports maximum diffusivity.
MaxHorizontalVelocity(const IceModel *m)
Reports the maximum horizontal absolute velocity component over the grid.
SeaLevelRisePotential(const IceModel *m)
Computes the total ice volume which is relevant for sea-level.
TimeStepLength(const IceModel *m)
Reports the mass continuity time step.
TimeStepRatio(const IceModel *m)
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 ...
double hardness(double E, double p) const
std::shared_ptr< const rheology::FlowLaw > flow_law() const
std::shared_ptr< const rheology::FlowLaw > flow_law() const
CFLData max_timestep_cfl_3d() 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.
CFLData max_timestep_cfl_2d() const
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
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.
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.
double total_ice_enthalpy(double thickness_threshold, const array::Array3D &ice_enthalpy, const array::Scalar &ice_thickness)
Computes the total ice enthalpy in J.
bool domain_edge(const Grid &grid, int i, int j)
bool ice_free(int M)
Ice-free cell (grounded or ocean).
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 .
double convert(System::Ptr system, double input, const std::string &spec1, const std::string &spec2)
Convert a quantity from unit1 to unit2.
double ice_volume_not_displacing_seawater(const Geometry &geometry, double thickness_threshold)
double ice_area(const Geometry &geometry, double thickness_threshold)
Computes ice area, in m^2.
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)
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.
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.
double total_grounding_line_flux(const array::CellType1 &cell_type, const array::Staggered1 &flux, double dt)
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.
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)
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.
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