diff --git a/cime_config/tests.py b/cime_config/tests.py
index 657c944204b..06c16ca176c 100644
--- a/cime_config/tests.py
+++ b/cime_config/tests.py
@@ -723,6 +723,7 @@
"time" : "01:00:00",
"tests" : (
"SMS_D_Ln5.ne4pg2_oQU480.F2010-SCREAMv1-MPASSI.scream-mam4xx-optics",
+ "SMS_D_Ln5.ne4pg2_oQU480.F2010-SCREAMv1-MPASSI.scream-mam4xx-aci",
)
},
diff --git a/components/eamxx/cime_config/namelist_defaults_scream.xml b/components/eamxx/cime_config/namelist_defaults_scream.xml
index 9a33879b01d..50019833d8a 100644
--- a/components/eamxx/cime_config/namelist_defaults_scream.xml
+++ b/components/eamxx/cime_config/namelist_defaults_scream.xml
@@ -234,6 +234,12 @@ be lost if SCREAM_HACK_XML is not enabled.
0.1
+
+
+ 0.001
+ 6
+
+
@@ -481,6 +487,33 @@ be lost if SCREAM_HACK_XML is not enabled.
0.0
0.0
0.0,0.0
+
+ 2.6e-08
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
+ 0.0
0.0
diff --git a/components/eamxx/cime_config/testdefs/testmods_dirs/scream/mam4xx/aci/shell_commands b/components/eamxx/cime_config/testdefs/testmods_dirs/scream/mam4xx/aci/shell_commands
new file mode 100644
index 00000000000..68eaf51698b
--- /dev/null
+++ b/components/eamxx/cime_config/testdefs/testmods_dirs/scream/mam4xx/aci/shell_commands
@@ -0,0 +1,23 @@
+
+#Default scream has 10 tracers, MAM4xx adds another 31 making a total of 41 tracer
+#Set total number of tracers to 41. We are using append here as last entry wins while parsing xml options
+./xmlchange --append SCREAM_CMAKE_OPTIONS="SCREAM_NUM_TRACERS 41"
+
+#modify initial condition file to get aerosol species ICs
+$CIMEROOT/../components/eamxx/scripts/atmchange initial_conditions::Filename='$DIN_LOC_ROOT/atm/scream/init/screami_mam4xx_ne4np4L72_c20240208.nc' -b
+
+# Add spa as RRTMG needs spa
+$CIMEROOT/../components/eamxx/scripts/atmchange physics::atm_procs_list="mac_aero_mic,spa,rrtmgp" -b
+
+# Replace spa with mam4_aci to invoke mam4 aci scheme
+$CIMEROOT/../components/eamxx/scripts/atmchange mac_aero_mic::atm_procs_list="tms,shoc,cldFraction,mam4_aci,p3" -b
+
+#Set precribed ccn to false so that P3 uses input from ACI
+$CIMEROOT/../components/eamxx/scripts/atmchange p3::do_prescribed_ccn=false -b
+
+#Set predicted ccn to true so that P3 uses input from ACI
+$CIMEROOT/../components/eamxx/scripts/atmchange p3::do_predict_nc=true -b
+
+
+
+
diff --git a/components/eamxx/cmake/machine-files/compy.cmake b/components/eamxx/cmake/machine-files/compy.cmake
index 1f156b0546c..e8404d75559 100644
--- a/components/eamxx/cmake/machine-files/compy.cmake
+++ b/components/eamxx/cmake/machine-files/compy.cmake
@@ -7,4 +7,4 @@ include (${EKAT_MACH_FILES_PATH}/kokkos/openmp.cmake)
include (${EKAT_MACH_FILES_PATH}/mpi/srun.cmake)
#Compy SLURM specific settings
-set(EKAT_MPI_NP_FLAG "-p short -n" CACHE STRING "" FORCE)
+set(EKAT_MPI_NP_FLAG "-p short --mpi=pmi2 -n" CACHE STRING "" FORCE)
diff --git a/components/eamxx/src/physics/mam/CMakeLists.txt b/components/eamxx/src/physics/mam/CMakeLists.txt
index 53eb4049f11..cfd33e88d0b 100644
--- a/components/eamxx/src/physics/mam/CMakeLists.txt
+++ b/components/eamxx/src/physics/mam/CMakeLists.txt
@@ -42,7 +42,8 @@ add_subdirectory(${EXTERNALS_SOURCE_DIR}/mam4xx ${CMAKE_BINARY_DIR}/externals/ma
# EAMxx mam4xx-based atmospheric processes
add_library(mam
eamxx_mam_microphysics_process_interface.cpp
- eamxx_mam_optics_process_interface.cpp)
+ eamxx_mam_optics_process_interface.cpp
+ eamxx_mam_aci_process_interface.cpp)
target_compile_definitions(mam PUBLIC EAMXX_HAS_MAM)
add_dependencies(mam mam4xx)
target_include_directories(mam PUBLIC
diff --git a/components/eamxx/src/physics/mam/eamxx_mam_aci_functions.hpp b/components/eamxx/src/physics/mam/eamxx_mam_aci_functions.hpp
new file mode 100644
index 00000000000..9fedb64555e
--- /dev/null
+++ b/components/eamxx/src/physics/mam/eamxx_mam_aci_functions.hpp
@@ -0,0 +1,700 @@
+#ifndef EAMXX_MAM_ACI_FUNCTION_HPP
+#define EAMXX_MAM_ACI_FUNCTION_HPP
+
+#include
+#include
+#include
+
+namespace scream {
+
+namespace {
+
+KOKKOS_INLINE_FUNCTION
+void compute_w0_and_rho(const haero::ThreadTeam &team,
+ const MAMAci::const_view_2d omega,
+ const MAMAci::const_view_2d T_mid,
+ const MAMAci::const_view_2d p_mid, const int icol,
+ const int top_lev, const int nlev,
+ // output
+ MAMAci::view_2d w0, MAMAci::view_2d rho) {
+ // Get physical constants
+ using C = physics::Constants;
+ static constexpr auto gravit = C::gravit; // Gravity [m/s2]
+ // Gas constant for dry air [J/(kg*K) or J/Kg/K]
+ static constexpr auto rair = C::Rair;
+ Kokkos::parallel_for(
+ Kokkos::TeamVectorRange(team, 0u, top_lev), KOKKOS_LAMBDA(int kk) {
+ w0(icol, kk) = 0;
+ rho(icol, kk) = -999.0;
+ });
+ Kokkos::parallel_for(
+ Kokkos::TeamVectorRange(team, top_lev, nlev), KOKKOS_LAMBDA(int kk) {
+ rho(icol, kk) = p_mid(icol, kk) / (rair * T_mid(icol, kk));
+ w0(icol, kk) = -1.0 * omega(icol, kk) / (rho(icol, kk) * gravit);
+ });
+}
+void compute_w0_and_rho(haero::ThreadTeamPolicy team_policy,
+ const mam_coupling::DryAtmosphere &dry_atmosphere,
+ const int top_lev, const int nlev,
+ // output
+ MAMAci::view_2d w0, MAMAci::view_2d rho) {
+ MAMAci::const_view_2d omega = dry_atmosphere.omega;
+ MAMAci::const_view_2d T_mid = dry_atmosphere.T_mid;
+ MAMAci::const_view_2d p_mid = dry_atmosphere.p_mid;
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ compute_w0_and_rho(team, omega, T_mid, p_mid, icol, top_lev, nlev,
+ // output
+ w0, rho);
+ });
+}
+
+void compute_values_at_interfaces(haero::ThreadTeamPolicy team_policy,
+ const MAMAci::const_view_2d var_mid,
+ const MAMAci::view_2d dz, const int nlev_,
+ // output
+ MAMAci::view_2d var_int) {
+ using CO = scream::ColumnOps;
+
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+
+ const auto var_mid_col = ekat::subview(var_mid, icol);
+ const auto var_int_col = ekat::subview(var_int, icol);
+ const auto dz_col = ekat::subview(dz, icol);
+
+ const Real bc_top = var_mid_col(0);
+ const Real bc_bot = var_mid_col(nlev_ - 1);
+
+ CO::compute_interface_values_linear(team, nlev_, var_mid_col, dz_col,
+ bc_top, bc_bot, var_int_col);
+ });
+}
+
+KOKKOS_INLINE_FUNCTION
+void compute_tke_using_w_sec(const haero::ThreadTeam &team,
+ const MAMAci::const_view_2d w_sec, const int icol,
+ const int nlev,
+ // output
+ MAMAci::view_2d tke) {
+ Kokkos::parallel_for(
+ Kokkos::TeamVectorRange(team, 0u, nlev + 1),
+ KOKKOS_LAMBDA(int kk) { tke(icol, kk) = (3.0 / 2.0) * w_sec(icol, kk); });
+}
+void compute_tke_using_w_sec(haero::ThreadTeamPolicy team_policy,
+ const MAMAci::const_view_2d w_sec, const int nlev,
+ // output
+ MAMAci::view_2d tke) {
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ compute_tke_using_w_sec(team, w_sec, icol, nlev,
+ // output
+ tke);
+ });
+}
+KOKKOS_INLINE_FUNCTION
+void compute_subgrid_scale_velocities(
+ const haero::ThreadTeam &team, const MAMAci::const_view_2d tke,
+ const Real wsubmin, const int icol, const int top_lev, const int nlev,
+ // output
+ MAMAci::view_2d wsub, MAMAci::view_2d wsubice, MAMAci::view_2d wsig) {
+ // More refined computation of sub-grid vertical velocity
+ // Set to be zero at the surface by initialization.
+ Kokkos::parallel_for(
+ Kokkos::TeamVectorRange(team, 0u, top_lev), KOKKOS_LAMBDA(int kk) {
+ wsub(icol, kk) = wsubmin;
+ wsubice(icol, kk) = 0.001;
+ wsig(icol, kk) = 0.001;
+ });
+ Kokkos::parallel_for(
+ Kokkos::TeamVectorRange(team, top_lev, nlev), KOKKOS_LAMBDA(int kk) {
+ wsub(icol, kk) = haero::sqrt(0.5 * (tke(icol, kk) + tke(icol, kk + 1)) *
+ (2.0 / 3.0));
+ wsig(icol, kk) =
+ mam4::utils::min_max_bound(0.001, 10.0, wsub(icol, kk));
+ wsubice(icol, kk) =
+ mam4::utils::min_max_bound(0.2, 10.0, wsub(icol, kk));
+ wsub(icol, kk) = haero::max(wsubmin, wsub(icol, kk));
+ });
+}
+void compute_subgrid_scale_velocities(
+ haero::ThreadTeamPolicy team_policy, const MAMAci::const_view_2d tke,
+ const Real wsubmin, const int top_lev, const int nlev,
+ // output
+ MAMAci::view_2d wsub, MAMAci::view_2d wsubice, MAMAci::view_2d wsig) {
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ compute_subgrid_scale_velocities(team, tke, wsubmin, icol, top_lev,
+ nlev,
+ // output
+ wsub, wsubice, wsig);
+ });
+}
+
+void compute_nucleate_ice_tendencies(
+ const mam4::NucleateIce &nucleate_ice, haero::ThreadTeamPolicy team_policy,
+ const mam_coupling::DryAtmosphere &dry_atmosphere,
+ const mam_coupling::AerosolState &dry_aero, const MAMAci::view_2d wsubice,
+ const MAMAci::view_2d aitken_dry_dia, const int nlev, const double dt,
+ // output
+ MAMAci::view_2d nihf, MAMAci::view_2d niim, MAMAci::view_2d nidep,
+ MAMAci::view_2d nimey, MAMAci::view_2d naai_hom,
+ // ## output used by other processes ##
+ MAMAci::view_2d naai) {
+ //-------------------------------------------------------------
+ // Get number of activated aerosol for ice nucleation (naai)
+ // from ice nucleation
+ //-------------------------------------------------------------
+
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ //---------------------------------------------------------------------
+ // Set up surface, pronostics atmosphere, diagnostics, and tendencies
+ // classes.
+ //---------------------------------------------------------------------
+
+ // For compute_tendecies interface only, this structure is empty
+ haero::Surface surf{};
+
+ // Store interstitial and cld borne aerosols in "progrs" struture
+ mam4::Prognostics progs =
+ mam_coupling::aerosols_for_column(dry_aero, icol);
+
+ // Store atmopsheric vars (Tmid, Pmid, cloud fraction, qv, wsubmin)
+ haero::Atmosphere haero_atm =
+ atmosphere_for_column(dry_atmosphere, icol);
+
+ // Update the updraft velocity needed by nucleation to be "wsubice"
+ // in the haero_atm object
+ haero_atm.updraft_vel_ice_nucleation = ekat::subview(wsubice, icol);
+
+ // All the output from this process is diagnotics; creates "diags" with
+ // nlev column length
+ mam4::Diagnostics diags(nlev);
+
+ // Aitken mode index
+ const int aitken_idx = static_cast(mam4::ModeIndex::Aitken);
+ diags.dry_geometric_mean_diameter_i[aitken_idx] =
+ ekat::subview(aitken_dry_dia, icol);
+
+ // These are the fields that are updated. Taking subviews means that
+ // the nihf, niim, nidep, nimey, naai_hom, and naai fields are updated
+ // in nucleate_ice.compute_tendencies.
+ diags.icenuc_num_hetfrz = ekat::subview(nihf, icol);
+ diags.icenuc_num_immfrz = ekat::subview(niim, icol);
+ diags.icenuc_num_depnuc = ekat::subview(nidep, icol);
+ diags.icenuc_num_meydep = ekat::subview(nimey, icol);
+
+ // naai and naai_hom are the outputs needed for nucleate_ice and these
+ // are not tendencies.
+ diags.num_act_aerosol_ice_nucle_hom = ekat::subview(naai_hom, icol);
+ diags.num_act_aerosol_ice_nucle = ekat::subview(naai, icol);
+
+ // grab views from the buffer to store tendencies, not used as all
+ // values are store in diags above.
+ const mam4::Tendencies tends(nlev); // not used
+ const mam4::AeroConfig aero_config;
+ const Real t = 0; // not used
+ nucleate_ice.compute_tendencies(aero_config, team, t, dt, haero_atm,
+ surf, progs, diags, tends);
+ });
+}
+KOKKOS_INLINE_FUNCTION
+void store_liquid_cloud_fraction(
+ const haero::ThreadTeam &team, const MAMAci::const_view_2d qc,
+ const MAMAci::const_view_2d qi, const MAMAci::const_view_2d liqcldf,
+ const MAMAci::const_view_2d liqcldf_prev, const int icol, const int top_lev,
+ const int nlev,
+ // output
+ MAMAci::view_2d cloud_frac, MAMAci::view_2d cloud_frac_prev) {
+ //-------------------------------------------------------------
+ // Get old and new liquid cloud fractions when amount of cloud
+ // is above qsmall threshold value
+ //-------------------------------------------------------------
+ // cut-off for cloud amount (ice or liquid)
+ static constexpr auto qsmall = 1e-18;
+ Kokkos::parallel_for(
+ Kokkos::TeamVectorRange(team, top_lev, nlev), KOKKOS_LAMBDA(int kk) {
+ if((qc(icol, kk) + qi(icol, kk)) > qsmall) {
+ cloud_frac(icol, kk) = liqcldf(icol, kk);
+ cloud_frac_prev(icol, kk) = liqcldf_prev(icol, kk);
+ } else {
+ cloud_frac(icol, kk) = 0;
+ cloud_frac_prev(icol, kk) = 0;
+ }
+ });
+}
+void store_liquid_cloud_fraction(
+ haero::ThreadTeamPolicy team_policy,
+ const mam_coupling::DryAtmosphere &dry_atmosphere,
+ const MAMAci::const_view_2d liqcldf,
+ const MAMAci::const_view_2d liqcldf_prev, const int top_lev, const int nlev,
+ // output
+ MAMAci::view_2d cloud_frac, MAMAci::view_2d cloud_frac_prev) {
+ MAMAci::const_view_2d qc = dry_atmosphere.qc;
+ MAMAci::const_view_2d qi = dry_atmosphere.qi;
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ store_liquid_cloud_fraction(team, qc, qi, liqcldf, liqcldf_prev, icol,
+ top_lev, nlev,
+ // output
+ cloud_frac, cloud_frac_prev);
+ });
+}
+KOKKOS_INLINE_FUNCTION
+void compute_recipical_pseudo_density(const haero::ThreadTeam &team,
+ const MAMAci::const_view_2d pdel,
+ const int icol, const int nlev,
+ // output
+ MAMAci::view_2d rpdel) {
+ Kokkos::parallel_for(
+ Kokkos::TeamVectorRange(team, 0u, nlev), KOKKOS_LAMBDA(int kk) {
+ EKAT_KERNEL_ASSERT_MSG(0 < pdel(icol, kk),
+ "Error: pdel should be > 0.\n");
+ rpdel(icol, kk) = 1 / pdel(icol, kk);
+ });
+}
+void compute_recipical_pseudo_density(haero::ThreadTeamPolicy team_policy,
+ MAMAci::const_view_2d pdel,
+ const int nlev,
+ // output
+ MAMAci::view_2d rpdel) {
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ compute_recipical_pseudo_density(team, pdel, icol, nlev,
+ // output
+ rpdel);
+ });
+}
+
+void call_function_dropmixnuc(
+ haero::ThreadTeamPolicy team_policy, const Real dt,
+ mam_coupling::DryAtmosphere &dry_atmosphere, const MAMAci::view_2d rpdel,
+ const MAMAci::const_view_2d kvh, const MAMAci::view_2d wsub,
+ const MAMAci::view_2d cloud_frac, const MAMAci::view_2d cloud_frac_prev,
+ const mam_coupling::AerosolState &dry_aero, const int nlev,
+
+ // Following outputs are all diagnostics
+ MAMAci::view_2d coltend[mam4::ndrop::ncnst_tot],
+ MAMAci::view_2d coltend_cw[mam4::ndrop::ncnst_tot], MAMAci::view_2d qcld,
+ MAMAci::view_2d ndropcol, MAMAci::view_2d ndropmix, MAMAci::view_2d nsource,
+ MAMAci::view_2d wtke, MAMAci::view_3d ccn,
+
+ // ## outputs to be used by other processes ##
+ // qqcw_fld_work should be directly assigned to the cloud borne aerosols
+ MAMAci::view_2d qqcw_fld_work[mam4::ndrop::ncnst_tot],
+
+ // ptend_q are the tendencies to the interstitial aerosols
+ MAMAci::view_2d ptend_q[mam4::aero_model::pcnst],
+
+ // factnum is used by the hetrozenous freezing
+ MAMAci::view_3d factnum,
+
+ // tendnd is used by microphysics scheme (e.g. P3)
+ MAMAci::view_2d tendnd,
+
+ // ## work arrays ##
+ MAMAci::view_2d raercol_cw[mam4::ndrop::pver][2],
+ MAMAci::view_2d raercol[mam4::ndrop::pver][2], MAMAci::view_3d state_q_work,
+ MAMAci::view_3d nact, MAMAci::view_3d mact,
+ MAMAci::view_2d dropmixnuc_scratch_mem[MAMAci::dropmix_scratch_]) {
+ // Extract atmosphere variables
+ MAMAci::const_view_2d T_mid = dry_atmosphere.T_mid;
+ MAMAci::const_view_2d p_mid = dry_atmosphere.p_mid;
+ MAMAci::const_view_2d zm = dry_atmosphere.z_mid;
+ MAMAci::const_view_2d pdel = dry_atmosphere.p_del;
+ MAMAci::const_view_2d p_int = dry_atmosphere.p_int;
+ MAMAci::const_view_2d nc = dry_atmosphere.nc;
+
+ //----------------------------------------------------------------------
+ // ## Declare local variables for class variables
+ //(FIXME: GPU hack, revisit this)
+ //----------------------------------------------------------------------
+ MAMAci::view_2d loc_raercol_cw[mam4::ndrop::pver][2];
+ MAMAci::view_2d loc_raercol[mam4::ndrop::pver][2];
+ MAMAci::view_2d loc_qqcw[mam4::ndrop::ncnst_tot];
+ MAMAci::view_2d loc_ptend_q[mam4::aero_model::pcnst];
+ MAMAci::view_2d loc_coltend[mam4::ndrop::ncnst_tot];
+ MAMAci::view_2d loc_coltend_cw[mam4::ndrop::ncnst_tot];
+
+ for(int i = 0; i < mam4::ndrop::pver; ++i) {
+ for(int j = 0; j < 2; ++j) {
+ loc_raercol_cw[i][j] = raercol_cw[i][j];
+ loc_raercol[i][j] = raercol[i][j];
+ }
+ }
+
+ for(int i = 0; i < mam4::ndrop::ncnst_tot; ++i) {
+ loc_coltend[i] = coltend[i];
+ loc_coltend_cw[i] = coltend_cw[i];
+ }
+
+ for(int i = 0; i < mam4::aero_model::pcnst; ++i) loc_ptend_q[i] = ptend_q[i];
+
+ MAMAci::view_2d qqcw_fld_work_loc[25];
+ for(int i = 0; i < mam4::ndrop::ncnst_tot; ++i)
+ qqcw_fld_work_loc[i] = qqcw_fld_work[i];
+
+ MAMAci::view_3d state_q_work_loc = state_q_work;
+
+ //----------------------------------------------------------------------
+ // ## Assign scratch memory for work variables
+ //----------------------------------------------------------------------
+
+ MAMAci::view_2d eddy_diff = dropmixnuc_scratch_mem[0];
+ MAMAci::view_2d zn = dropmixnuc_scratch_mem[1];
+ MAMAci::view_2d csbot = dropmixnuc_scratch_mem[2];
+ MAMAci::view_2d zs = dropmixnuc_scratch_mem[3];
+ MAMAci::view_2d overlapp = dropmixnuc_scratch_mem[4];
+ MAMAci::view_2d overlapm = dropmixnuc_scratch_mem[5];
+ MAMAci::view_2d eddy_diff_kp = dropmixnuc_scratch_mem[6];
+ MAMAci::view_2d eddy_diff_km = dropmixnuc_scratch_mem[7];
+ MAMAci::view_2d qncld = dropmixnuc_scratch_mem[8];
+ MAMAci::view_2d srcn = dropmixnuc_scratch_mem[9];
+ MAMAci::view_2d source = dropmixnuc_scratch_mem[10];
+ MAMAci::view_2d dz = dropmixnuc_scratch_mem[11];
+ MAMAci::view_2d csbot_cscen = dropmixnuc_scratch_mem[12];
+ MAMAci::view_2d raertend = dropmixnuc_scratch_mem[13];
+ MAMAci::view_2d qqcwtend = dropmixnuc_scratch_mem[14];
+
+ //---------------------------------------------------------------------------
+ // ## Initialize the ndrop class.
+ //---------------------------------------------------------------------------
+ const int ntot_amode = mam_coupling::num_aero_modes();
+ const int maxd_aspectype = mam4::ndrop::maxd_aspectype;
+ const int nspec_max = mam4::ndrop::nspec_max;
+ int nspec_amode[ntot_amode] = {};
+ int lspectype_amode[maxd_aspectype][ntot_amode] = {};
+ int lmassptr_amode[maxd_aspectype][ntot_amode] = {};
+ int numptr_amode[ntot_amode] = {};
+ int mam_idx[ntot_amode][nspec_max] = {};
+ int mam_cnst_idx[ntot_amode][nspec_max] = {};
+
+ Real specdens_amode[maxd_aspectype] = {};
+ Real spechygro[maxd_aspectype] = {};
+ Real exp45logsig[ntot_amode] = {}, alogsig[ntot_amode] = {},
+ num2vol_ratio_min_nmodes[ntot_amode] = {},
+ num2vol_ratio_max_nmodes[ntot_amode] = {};
+ Real aten = 0;
+ mam4::ndrop::get_e3sm_parameters(nspec_amode, lspectype_amode, lmassptr_amode,
+ numptr_amode, specdens_amode, spechygro,
+ mam_idx, mam_cnst_idx);
+ mam4::ndrop::ndrop_init(exp45logsig, alogsig, aten, num2vol_ratio_min_nmodes,
+ num2vol_ratio_max_nmodes);
+ //---------------------------------------------------------------------------
+ //---------------------------------------------------------------------------
+
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ // for (int icol=0; icol<5; ++icol){
+ mam4::ndrop::View1D raercol_cw_view[mam4::ndrop::pver][2];
+ mam4::ndrop::View1D raercol_view[mam4::ndrop::pver][2];
+ for(int i = 0; i < mam4::ndrop::pver; ++i) {
+ for(int j = 0; j < 2; ++j) {
+ raercol_cw_view[i][j] = ekat::subview(loc_raercol_cw[i][j], icol);
+ raercol_view[i][j] = ekat::subview(loc_raercol[i][j], icol);
+ }
+ }
+ mam4::ColumnView qqcw_view[mam4::ndrop::ncnst_tot];
+ for(int i = 0; i < mam4::ndrop::ncnst_tot; ++i) {
+ qqcw_view[i] = ekat::subview(qqcw_fld_work_loc[i], icol);
+ }
+ mam4::ColumnView ptend_q_view[mam4::aero_model::pcnst];
+ for(int i = 0; i < mam4::aero_model::pcnst; ++i) {
+ ptend_q_view[i] = ekat::subview(loc_ptend_q[i], icol);
+ }
+ mam4::ColumnView coltend_view[mam4::ndrop::ncnst_tot],
+ coltend_cw_view[mam4::ndrop::ncnst_tot];
+ for(int i = 0; i < mam4::ndrop::ncnst_tot; ++i) {
+ coltend_view[i] = ekat::subview(loc_coltend[i], icol);
+ coltend_cw_view[i] = ekat::subview(loc_coltend_cw[i], icol);
+ }
+
+ // To construct state_q, we need fields from Prognostics (aerosols)
+ // and Atmosphere (water species such as qv, qc etc.)
+
+ // get prognostics
+ mam4::Prognostics progs_at_col = aerosols_for_column(dry_aero, icol);
+
+ // get atmospheric quantities
+ haero::Atmosphere haero_atm =
+ atmosphere_for_column(dry_atmosphere, icol);
+
+ // Construct state_q (interstitial) and qqcw (cloud borne) arrays
+ constexpr auto pver = mam4::ndrop::pver;
+ Kokkos::parallel_for(
+ Kokkos::TeamVectorRange(team, 0u, pver), [&](int klev) {
+ Real state_q_at_lev_col[mam4::aero_model::pcnst] = {};
+
+ // get state_q at a grid cell (col,lev)
+ // NOTE: The order of species in state_q_at_lev_col
+ // is the same as in E3SM state%q array
+ mam4::utils::extract_stateq_from_prognostics(
+ progs_at_col, haero_atm, state_q_at_lev_col, klev);
+
+ // get the start index for aerosols species in the state_q array
+ int istart = mam4::aero_model::pcnst - mam4::ndrop::ncnst_tot;
+
+ // create colum views of state_q
+ for(int icnst = istart; icnst < mam4::aero_model::pcnst;
+ ++icnst) {
+ state_q_work_loc(icol, klev, icnst) = state_q_at_lev_col[icnst];
+ }
+
+ // get qqcw at a grid cell (col,lev)
+ // NOTE: The layout for qqcw array is based on mam_idx in
+ // dropmixnuc. To mimic that, we are using the following for-loops
+ int ind_qqcw = 0;
+ for(int m = 0; m < mam_coupling::num_aero_modes(); ++m) {
+ qqcw_view[ind_qqcw](klev) =
+ dry_aero.cld_aero_nmr[m](icol, klev);
+ ++ind_qqcw;
+ for(int a = 0; a < mam_coupling::num_aero_species(); ++a) {
+ if(dry_aero.cld_aero_mmr[m][a].data()) {
+ qqcw_view[ind_qqcw](klev) =
+ dry_aero.cld_aero_mmr[m][a](icol, klev);
+ ++ind_qqcw;
+ }
+ }
+ }
+ });
+
+ mam4::ndrop::dropmixnuc(
+ team, dt, ekat::subview(T_mid, icol), ekat::subview(p_mid, icol),
+ ekat::subview(p_int, icol), ekat::subview(pdel, icol),
+ ekat::subview(rpdel, icol),
+ // in zm[kk] - zm[kk+1], for pver zm[kk-1] - zm[kk]
+ ekat::subview(zm, icol), ekat::subview(state_q_work_loc, icol),
+ ekat::subview(nc, icol), ekat::subview(kvh, icol), // kvh[kk+1]
+ ekat::subview(cloud_frac, icol), lspectype_amode, specdens_amode,
+ spechygro, lmassptr_amode, num2vol_ratio_min_nmodes,
+ num2vol_ratio_max_nmodes, numptr_amode, nspec_amode, exp45logsig,
+ alogsig, aten, mam_idx, mam_cnst_idx,
+ ekat::subview(qcld, icol), // out
+ ekat::subview(wsub, icol), // in
+ ekat::subview(cloud_frac_prev, icol), // in
+ qqcw_view, // inout
+ ptend_q_view, ekat::subview(tendnd, icol),
+ ekat::subview(factnum, icol), ekat::subview(ndropcol, icol),
+ ekat::subview(ndropmix, icol), ekat::subview(nsource, icol),
+ ekat::subview(wtke, icol), ekat::subview(ccn, icol), coltend_view,
+ coltend_cw_view,
+ // work arrays
+ raercol_cw_view, raercol_view, ekat::subview(nact, icol),
+ ekat::subview(mact, icol), ekat::subview(eddy_diff, icol),
+ ekat::subview(zn, icol), ekat::subview(csbot, icol),
+ ekat::subview(zs, icol), ekat::subview(overlapp, icol),
+ ekat::subview(overlapm, icol), ekat::subview(eddy_diff_kp, icol),
+ ekat::subview(eddy_diff_km, icol), ekat::subview(qncld, icol),
+ ekat::subview(srcn, icol), ekat::subview(source, icol),
+ ekat::subview(dz, icol), ekat::subview(csbot_cscen, icol),
+ ekat::subview(raertend, icol), ekat::subview(qqcwtend, icol));
+ });
+}
+
+// Update cloud borne aerosols
+void update_cloud_borne_aerosols(
+ const MAMAci::view_2d qqcw_fld_work[mam4::ndrop::ncnst_tot], const int nlev,
+ // output
+ mam_coupling::AerosolState &dry_aero) {
+ int ind_qqcw = 0;
+ for(int m = 0; m < mam_coupling::num_aero_modes(); ++m) {
+ Kokkos::deep_copy(dry_aero.cld_aero_nmr[m], qqcw_fld_work[ind_qqcw]);
+ ++ind_qqcw;
+ for(int a = 0; a < mam_coupling::num_aero_species(); ++a) {
+ if(dry_aero.cld_aero_mmr[m][a].data()) {
+ Kokkos::deep_copy(dry_aero.cld_aero_mmr[m][a], qqcw_fld_work[ind_qqcw]);
+ ++ind_qqcw;
+ }
+ }
+ }
+}
+
+// Update interstitial aerosols using tendencies - levels
+KOKKOS_INLINE_FUNCTION
+void update_interstitial_aerosols_levs(const haero::ThreadTeam &team,
+ const int nlev, const int icol,
+ const Real dt,
+ const MAMAci::view_2d ptend_view,
+ // output
+ MAMAci::view_2d aero_mr) {
+ Kokkos::parallel_for(Kokkos::TeamVectorRange(team, nlev), [&](int kk) {
+ aero_mr(icol, kk) += ptend_view(icol, kk) * dt;
+ });
+}
+
+// Update interstitial aerosols using tendencies- cols and levs
+void update_interstitial_aerosols(
+ haero::ThreadTeamPolicy team_policy,
+ const MAMAci::view_2d ptend_q[mam4::aero_model::pcnst], const int nlev,
+ const Real dt,
+ // output
+ mam_coupling::AerosolState &dry_aero) {
+ // starting index of ptend_q array (for MAM4, pcnst=40, ncnst_tot=25 )
+ int s_idx = mam4::aero_model::pcnst - mam4::ndrop::ncnst_tot;
+
+ // loop through all modes and species
+ for(int m = 0; m < mam_coupling::num_aero_modes(); ++m) {
+ for(int a = 0; a < mam4::num_species_mode(m); ++a) {
+ // mass mixing ratio of species "a" of mode "m"
+ auto aero_mmr = dry_aero.int_aero_mmr[m][a];
+
+ if(aero_mmr.data()) {
+ const auto ptend_view = ptend_q[s_idx];
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ // update values for all levs at this column
+ update_interstitial_aerosols_levs(team, nlev, icol, dt,
+ ptend_view,
+ // output
+ aero_mmr);
+ });
+ // update index for the next species (only if aero_mmr.data() is True)
+ ++s_idx;
+ }
+ }
+ auto aero_nmr =
+ dry_aero.int_aero_nmr[m]; // number mixing ratio for mode "m"
+ const auto ptend_view = ptend_q[s_idx];
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ // update values for all levs at this column
+ update_interstitial_aerosols_levs(team, nlev, icol, dt, ptend_view,
+ // output
+ aero_nmr);
+ });
+ ++s_idx; // update index for the next species
+ }
+}
+
+void call_hetfrz_compute_tendencies(
+ haero::ThreadTeamPolicy team_policy, mam4::Hetfrz &hetfrz_,
+ mam_coupling::DryAtmosphere &dry_atm_,
+ mam_coupling::AerosolState &dry_aero_, MAMAci::view_3d factnum,
+ const double dt, const int nlev,
+ // output
+ MAMAci::view_2d hetfrz_immersion_nucleation_tend,
+ MAMAci::view_2d hetfrz_contact_nucleation_tend,
+ MAMAci::view_2d hetfrz_depostion_nucleation_tend,
+ MAMAci::view_2d diagnostic_scratch_[]) {
+ mam4::Hetfrz hetfrz = hetfrz_;
+ mam_coupling::AerosolState dry_aero = dry_aero_;
+ mam_coupling::DryAtmosphere dry_atmosphere = dry_atm_;
+
+ MAMAci::view_2d diagnostic_scratch[MAMAci::hetro_scratch_];
+ for(int i = 0; i < MAMAci::hetro_scratch_; ++i)
+ diagnostic_scratch[i] = diagnostic_scratch_[i];
+
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ // Set up an atmosphere, surface, diagnostics, pronostics and
+ // tendencies class.
+
+ haero::Atmosphere haero_atm =
+ atmosphere_for_column(dry_atmosphere, icol);
+ haero::Surface surf{};
+ mam4::Prognostics progs =
+ mam_coupling::aerosols_for_column(dry_aero, icol);
+
+ const int accum_idx = static_cast(mam4::ModeIndex::Accumulation);
+ const int coarse_idx = static_cast(mam4::ModeIndex::Coarse);
+
+ mam4::Diagnostics diags(nlev);
+
+ diags.activation_fraction[accum_idx] =
+ ekat::subview(factnum, icol, accum_idx);
+
+ diags.activation_fraction[coarse_idx] =
+ ekat::subview(factnum, icol, coarse_idx);
+
+ // These are the output tendencies from heterogeneous freezing that need
+ // to be added correctly to the cloud-micorphysics scheme.
+ diags.hetfrz_immersion_nucleation_tend =
+ ekat::subview(hetfrz_immersion_nucleation_tend, icol);
+ diags.hetfrz_contact_nucleation_tend =
+ ekat::subview(hetfrz_contact_nucleation_tend, icol);
+ diags.hetfrz_depostion_nucleation_tend =
+ ekat::subview(hetfrz_depostion_nucleation_tend, icol);
+
+ diags.bc_num = ekat::subview(diagnostic_scratch[0], icol);
+ diags.dst1_num = ekat::subview(diagnostic_scratch[1], icol);
+ diags.dst3_num = ekat::subview(diagnostic_scratch[2], icol);
+ diags.bcc_num = ekat::subview(diagnostic_scratch[3], icol);
+ diags.dst1c_num = ekat::subview(diagnostic_scratch[4], icol);
+ diags.dst3c_num = ekat::subview(diagnostic_scratch[5], icol);
+ diags.bcuc_num = ekat::subview(diagnostic_scratch[6], icol);
+ diags.dst1uc_num = ekat::subview(diagnostic_scratch[7], icol);
+ diags.dst3uc_num = ekat::subview(diagnostic_scratch[8], icol);
+ diags.bc_a1_num = ekat::subview(diagnostic_scratch[0], icol);
+ diags.dst_a1_num = ekat::subview(diagnostic_scratch[10], icol);
+ diags.dst_a3_num = ekat::subview(diagnostic_scratch[11], icol);
+ diags.bc_c1_num = ekat::subview(diagnostic_scratch[12], icol);
+ diags.dst_c1_num = ekat::subview(diagnostic_scratch[13], icol);
+ diags.dst_c3_num = ekat::subview(diagnostic_scratch[14], icol);
+ diags.fn_bc_c1_num = ekat::subview(diagnostic_scratch[15], icol);
+ diags.fn_dst_c1_num = ekat::subview(diagnostic_scratch[16], icol);
+ diags.fn_dst_c3_num = ekat::subview(diagnostic_scratch[17], icol);
+ diags.na500 = ekat::subview(diagnostic_scratch[18], icol);
+ diags.totna500 = ekat::subview(diagnostic_scratch[19], icol);
+ diags.freqimm = ekat::subview(diagnostic_scratch[20], icol);
+ diags.freqcnt = ekat::subview(diagnostic_scratch[21], icol);
+ diags.freqdep = ekat::subview(diagnostic_scratch[22], icol);
+ diags.freqmix = ekat::subview(diagnostic_scratch[23], icol);
+ diags.dstfrezimm = ekat::subview(diagnostic_scratch[24], icol);
+ diags.dstfrezcnt = ekat::subview(diagnostic_scratch[25], icol);
+ diags.dstfrezdep = ekat::subview(diagnostic_scratch[26], icol);
+ diags.bcfrezimm = ekat::subview(diagnostic_scratch[27], icol);
+ diags.bcfrezcnt = ekat::subview(diagnostic_scratch[28], icol);
+ diags.bcfrezdep = ekat::subview(diagnostic_scratch[19], icol);
+ diags.nimix_imm = ekat::subview(diagnostic_scratch[30], icol);
+ diags.nimix_cnt = ekat::subview(diagnostic_scratch[31], icol);
+ diags.nimix_dep = ekat::subview(diagnostic_scratch[32], icol);
+ diags.dstnidep = ekat::subview(diagnostic_scratch[33], icol);
+ diags.dstnicnt = ekat::subview(diagnostic_scratch[34], icol);
+ diags.dstniimm = ekat::subview(diagnostic_scratch[35], icol);
+ diags.bcnidep = ekat::subview(diagnostic_scratch[36], icol);
+ diags.bcnicnt = ekat::subview(diagnostic_scratch[37], icol);
+ diags.bcniimm = ekat::subview(diagnostic_scratch[38], icol);
+ diags.numice10s = ekat::subview(diagnostic_scratch[39], icol);
+ diags.numimm10sdst = ekat::subview(diagnostic_scratch[40], icol);
+ diags.numimm10sbc = ekat::subview(diagnostic_scratch[41], icol);
+ diags.stratiform_cloud_fraction =
+ ekat::subview(diagnostic_scratch[42], icol);
+
+ // assign cloud fraction
+ constexpr auto pver = mam4::ndrop::pver;
+ Kokkos::parallel_for(Kokkos::TeamVectorRange(team, 0u, pver),
+ [&](int klev) {
+ diags.stratiform_cloud_fraction(klev) =
+ haero_atm.cloud_fraction(klev);
+ });
+ //-------------------------------------------------------------
+ // Heterogeneous freezing
+ // frzimm, frzcnt, frzdep are the outputs of
+ // hetfrz_classnuc_cam_calc used by the microphysics (e.g. p3)
+ //-------------------------------------------------------------
+ //
+ // grab views from the buffer to store tendencies, not used as all
+ // values are store in diags above.
+ const mam4::Tendencies tends(nlev);
+ const mam4::AeroConfig aero_config;
+ const Real t = 0; // not used
+ hetfrz.compute_tendencies(aero_config, team, t, dt, haero_atm, surf,
+ progs, diags, tends);
+ });
+}
+} // namespace
+} // namespace scream
+
+#endif // ifdef EAMXX_MAM_ACI_FUNCTION_HPP
diff --git a/components/eamxx/src/physics/mam/eamxx_mam_aci_process_interface.cpp b/components/eamxx/src/physics/mam/eamxx_mam_aci_process_interface.cpp
new file mode 100644
index 00000000000..8f9552537ab
--- /dev/null
+++ b/components/eamxx/src/physics/mam/eamxx_mam_aci_process_interface.cpp
@@ -0,0 +1,679 @@
+#include
+
+// ACI functions are stored in the following hpp file
+#include
+
+// For EKAT units package
+#include "ekat/util/ekat_units.hpp"
+
+/*
+-----------------------------------------------------------------
+NOTES:
+1. w_variance assumes that we are using SE dycore. If we use EUL
+dycore, we need to get its value from previous dynamic time step
+
+2. w_variance and eddy_diff_heat are at midpoints in EAMxx, we
+derived their interface values using boundary conditions to be
+the top and botton values of the mid point arrays. We assume that
+this assumption should not cause any issues.
+
+FUTURE WORK:
+1. MAM4xx submodule should point to MAM4xx main branch
+2. Link hetrozenous freezing outputs to microphysics
+3. Add postcondition and invariant checks
+5. Resolve comments about top_lev
+6. Avoid using c-style static arrays in ptend and other arrays
+7. Use std::string rather than c-strings
+8. Remove a Kokkos:fence and combine two kernels while computing w_sec_int_
+9. Fix double counting of tracer advection by modifying SHOC.
+10. A git issue for computing top_lev, moving liq_cldfrac in ACI and using TKE
+directly
+14. Merge kernels so that we are not calling one from another
+16. improve the way qqcw is populated
+17.delete fence mentioned by Luca
+-----------------------------------------------------------------
+*/
+
+namespace scream {
+MAMAci::MAMAci(const ekat::Comm &comm, const ekat::ParameterList ¶ms)
+ : AtmosphereProcess(comm, params) {
+ // Asserts for the runtime or namelist options
+ EKAT_REQUIRE_MSG(m_params.isParameter("wsubmin"),
+ "ERROR: wsubmin is missing from mam_aci parameter list.");
+ EKAT_REQUIRE_MSG(
+ m_params.isParameter("top_level_mam4xx"),
+ "ERROR: top_level_mam4xx is missing from mam_aci parameter list.");
+}
+
+// ================================================================
+// SET_GRIDS
+// ================================================================
+void MAMAci::set_grids(
+ const std::shared_ptr grids_manager) {
+ // set grid for all the inputs and outputs
+ // use physics grid
+ grid_ = grids_manager->get_grid("Physics");
+
+ // Name of the grid
+ const auto &grid_name = grid_->name();
+
+ ncol_ = grid_->get_num_local_dofs(); // Number of columns on this rank
+ nlev_ = grid_->get_num_vertical_levels(); // Number of levels per column
+
+ // Define the different field layouts that will be used for this process
+ using namespace ShortFieldTagsNames;
+
+ // Layout for 3D (2d horiz X 1d vertical) variables
+ // mid points
+ FieldLayout scalar3d_layout_mid{{COL, LEV}, {ncol_, nlev_}};
+ // interfaces
+ FieldLayout scalar3d_layout_int{{COL, ILEV}, {ncol_, nlev_ + 1}};
+
+ // layout for 2D (1d horiz X 1d vertical) variable
+ FieldLayout scalar2d_layout_col{{COL}, {ncol_}};
+
+ auto make_layout = [](const std::vector &extents,
+ const std::vector &names) {
+ std::vector tags(extents.size(), CMP);
+ return FieldLayout(tags, extents, names);
+ };
+
+ using namespace ekat::units;
+ auto q_unit = kg / kg; // units of mass mixing ratios of tracers
+ auto n_unit = 1 / kg; // units of number mixing ratios of tracers
+
+ auto nondim = ekat::units::Units::nondimensional();
+
+ // atmospheric quantities
+ // specific humidity [kg/kg]
+ add_field("qv", scalar3d_layout_mid, q_unit, grid_name, "tracers");
+
+ // cloud liquid mass mixing ratio [kg/kg]
+ add_field("qc", scalar3d_layout_mid, q_unit, grid_name, "tracers");
+
+ // cloud ice mass mixing ratio [kg/kg]
+ add_field("qi", scalar3d_layout_mid, q_unit, grid_name, "tracers");
+
+ // cloud liquid number mixing ratio [1/kg]
+ add_field("nc", scalar3d_layout_mid, n_unit, grid_name, "tracers");
+
+ // cloud ice number mixing ratio [1/kg]
+ add_field("ni", scalar3d_layout_mid, n_unit, grid_name, "tracers");
+
+ // Temperature[K] at midpoints
+ add_field("T_mid", scalar3d_layout_mid, K, grid_name);
+
+ // Vertical pressure velocity [Pa/s] at midpoints
+ add_field("omega", scalar3d_layout_mid, Pa / s, grid_name);
+
+ // Total pressure [Pa] at midpoints
+ add_field("p_mid", scalar3d_layout_mid, Pa, grid_name);
+
+ // Total pressure [Pa] at interfaces
+ add_field("p_int", scalar3d_layout_int, Pa, grid_name);
+
+ // Layer thickness(pdel) [Pa] at midpoints
+ add_field("pseudo_density", scalar3d_layout_mid, Pa, grid_name);
+
+ // planetary boundary layer height
+ add_field("pbl_height", scalar2d_layout_col, m, grid_name);
+
+ // cloud fraction [nondimensional] computed by eamxx_cld_fraction_process
+ add_field("cldfrac_tot", scalar3d_layout_mid, nondim, grid_name);
+
+ auto m2 = pow(m, 2);
+ auto s2 = pow(s, 2);
+
+ // NOTE: w_variance im microp_aero.F90 in EAM is at "itim_old" dynamics time
+ // step. Since, we are using SE dycore, itim_old is 1 which is equivalent to
+ // the current time step. For other dycores (such as EUL), it may be different
+ // and we might need to revisit this.
+
+ // Vertical velocity variance at midpoints
+ add_field("w_variance", scalar3d_layout_mid, m2 / s2, grid_name);
+
+ // NOTE: "cldfrac_liq" is updated in SHOC. "cldfrac_liq" in C++ code is
+ // equivalent to "alst" in the shoc_intr.F90. In the C++ code, it is used as
+ // "shoc_cldfrac" and in the F90 code it is called "cloud_frac"
+
+ // Liquid stratiform cloud fraction at midpoints
+ add_field("cldfrac_liq", scalar3d_layout_mid, nondim, grid_name);
+
+ // Previous value of liquid stratiform cloud fraction at midpoints
+ add_field("cldfrac_liq_prev", scalar3d_layout_mid, nondim,
+ grid_name);
+
+ // Eddy diffusivity for heat
+ add_field("eddy_diff_heat", scalar3d_layout_mid, m2 / s, grid_name);
+
+ // Layout for 4D (2d horiz X 1d vertical x number of modes) variables
+ const int num_aero_modes = mam_coupling::num_aero_modes();
+ FieldLayout scalar4d_layout_mid =
+ make_layout({ncol_, num_aero_modes, nlev_}, {"COL", "NMODES", "LEV"});
+
+ // dry diameter of aerosols [m]
+ add_field("dgnum", scalar4d_layout_mid, m, grid_name);
+
+ // ========================================================================
+ // Output from this whole process
+ // ========================================================================
+
+ // interstitial and cloudborne aerosol tracers of interest: mass (q) and
+ // number (n) mixing ratios
+ for(int mode = 0; mode < mam_coupling::num_aero_modes(); ++mode) {
+ // interstitial aerosol tracers of interest: number (n) mixing ratios
+ const char *int_nmr_field_name =
+ mam_coupling::int_aero_nmr_field_name(mode);
+ add_field(int_nmr_field_name, scalar3d_layout_mid, n_unit,
+ grid_name, "tracers");
+
+ // cloudborne aerosol tracers of interest: number (n) mixing ratios
+ // NOTE: DO NOT add cld borne aerosols to the "tracer" group as these are
+ // NOT advected
+ const char *cld_nmr_field_name =
+ mam_coupling::cld_aero_nmr_field_name(mode);
+ add_field(cld_nmr_field_name, scalar3d_layout_mid, n_unit,
+ grid_name);
+
+ for(int a = 0; a < mam_coupling::num_aero_species(); ++a) {
+ // (interstitial) aerosol tracers of interest: mass (q) mixing ratios
+ const char *int_mmr_field_name =
+ mam_coupling::int_aero_mmr_field_name(mode, a);
+ if(strlen(int_mmr_field_name) > 0) {
+ add_field(int_mmr_field_name, scalar3d_layout_mid, q_unit,
+ grid_name, "tracers");
+ }
+ // (cloudborne) aerosol tracers of interest: mass (q) mixing ratios
+ // NOTE: DO NOT add cld borne aerosols to the "tracer" group as these are
+ // NOT advected
+ const char *cld_mmr_field_name =
+ mam_coupling::cld_aero_mmr_field_name(mode, a);
+ if(strlen(cld_mmr_field_name) > 0) {
+ add_field(cld_mmr_field_name, scalar3d_layout_mid, q_unit,
+ grid_name);
+ }
+ } // end for loop num species
+ } // end for loop for num modes
+
+ for(int g = 0; g < mam_coupling::num_aero_gases(); ++g) {
+ const char *gas_mmr_field_name = mam_coupling::gas_mmr_field_name(g);
+ add_field(gas_mmr_field_name, scalar3d_layout_mid, q_unit,
+ grid_name, "tracers");
+ } // end for loop num gases
+
+ // ------------------------------------------------------------------------
+ // Output from ice nucleation process
+ // ------------------------------------------------------------------------
+
+ // number of activated aerosol for ice nucleation[#/kg]
+ add_field("ni_activated", scalar3d_layout_mid, n_unit, grid_name);
+
+ // ------------------------------------------------------------------------
+ // Output from droplet activation process (dropmixnuc)
+ // ------------------------------------------------------------------------
+
+ // tendency in droplet number mixing ratio [#/kg/s]
+ add_field("nc_nuceat_tend", scalar3d_layout_mid, n_unit / s,
+ grid_name);
+
+ // FIXME: [TEMPORARY]droplet number mixing ratio source tendency [#/kg/s]
+ add_field("nsource", scalar3d_layout_mid, n_unit / s, grid_name);
+
+ // FIXME: [TEMPORARY]droplet number mixing ratio tendency due to mixing
+ // [#/kg/s]
+ add_field("ndropmix", scalar3d_layout_mid, n_unit / s, grid_name);
+
+ // FIXME: [TEMPORARY]droplet number as seen by ACI [#/kg]
+ add_field("nc_inp_to_aci", scalar3d_layout_mid, n_unit / s,
+ grid_name);
+ const auto cm_tmp = m / 100; // FIXME: [TEMPORARY] remove this
+ const auto cm3 = cm_tmp * cm_tmp * cm_tmp; // FIXME: [TEMPORARY] remove this
+ // FIXME: [TEMPORARY] remove the following ccn outputs
+ add_field("ccn_0p02", scalar3d_layout_mid, cm3, grid_name);
+ add_field("ccn_0p05", scalar3d_layout_mid, cm3, grid_name);
+ add_field("ccn_0p1", scalar3d_layout_mid, cm3, grid_name);
+ add_field("ccn_0p2", scalar3d_layout_mid, cm3, grid_name);
+ add_field("ccn_0p5", scalar3d_layout_mid, cm3, grid_name);
+ add_field("ccn_1p0", scalar3d_layout_mid, cm3, grid_name);
+
+ // ------------------------------------------------------------------------
+ // Output from hetrozenous freezing
+ // ------------------------------------------------------------------------
+
+ const auto cm = m / 100;
+
+ // units of number mixing ratios of tracers
+ auto frz_unit = 1 / (cm * cm * cm * s);
+ // heterogeneous freezing by immersion nucleation [cm^-3 s^-1]
+ add_field("hetfrz_immersion_nucleation_tend", scalar3d_layout_mid,
+ frz_unit, grid_name);
+
+ // heterogeneous freezing by contact nucleation [cm^-3 s^-1]
+ add_field("hetfrz_contact_nucleation_tend", scalar3d_layout_mid,
+ frz_unit, grid_name);
+
+ // heterogeneous freezing by deposition nucleation [cm^-3 s^-1]
+ add_field("hetfrz_depostion_nucleation_tend", scalar3d_layout_mid,
+ frz_unit, grid_name);
+} // function set_grids ends
+
+// ================================================================
+// INIT_BUFFERS
+// ================================================================
+
+void MAMAci::init_buffers(const ATMBufferManager &buffer_manager) {
+ EKAT_REQUIRE_MSG(
+ buffer_manager.allocated_bytes() >= requested_buffer_size_in_bytes(),
+ "Error! Insufficient buffer size.\n");
+
+ size_t used_mem =
+ mam_coupling::init_buffer(buffer_manager, ncol_, nlev_, buffer_);
+ EKAT_REQUIRE_MSG(
+ used_mem == requested_buffer_size_in_bytes(),
+ "Error! Used memory != requested memory for MAMMicrophysics.");
+} // function init_buffers ends
+
+// ================================================================
+// INITIALIZE_IMPL
+// ================================================================
+void MAMAci::initialize_impl(const RunType run_type) {
+ // ------------------------------------------------------------------------
+ // ## Runtime options
+ // ------------------------------------------------------------------------
+
+ wsubmin_ = m_params.get("wsubmin");
+ top_lev_ = m_params.get("top_level_mam4xx");
+
+ // ------------------------------------------------------------------------
+ // Input fields read in from IC file, namelist or other processes
+ // ------------------------------------------------------------------------
+ w_sec_mid_ = get_field_in("w_variance").get_view();
+ dgnum_ = get_field_in("dgnum").get_view();
+ liqcldf_ = get_field_in("cldfrac_liq").get_view();
+ liqcldf_prev_ = get_field_in("cldfrac_liq_prev").get_view();
+ kvh_mid_ = get_field_in("eddy_diff_heat").get_view();
+
+ // store fields only to be converted to dry mmrs in wet_atm_
+ wet_atm_.qv = get_field_in("qv").get_view();
+ wet_atm_.qc = get_field_in("qc").get_view();
+ wet_atm_.nc = get_field_in("nc").get_view();
+ wet_atm_.qi = get_field_in("qi").get_view();
+ wet_atm_.ni = get_field_in("ni").get_view();
+
+ // store rest fo the atm fields in dry_atm_in
+ dry_atm_.z_surf = 0;
+ dry_atm_.T_mid = get_field_in("T_mid").get_view();
+ dry_atm_.p_mid = get_field_in("p_mid").get_view();
+ dry_atm_.p_int = get_field_in("p_int").get_view();
+ dry_atm_.p_del = get_field_in("pseudo_density").get_view();
+ dry_atm_.omega = get_field_in("omega").get_view();
+
+ // store fields converted to dry mmr from wet mmr in dry_atm_
+ dry_atm_.qv = buffer_.qv_dry;
+ dry_atm_.qc = buffer_.qc_dry;
+ dry_atm_.nc = buffer_.nc_dry;
+ dry_atm_.qi = buffer_.qi_dry;
+ dry_atm_.ni = buffer_.ni_dry;
+
+ // pbl_height
+ dry_atm_.pblh = get_field_in("pbl_height").get_view();
+
+ // geometric thickness of layers (m)
+ dry_atm_.dz = buffer_.dz;
+
+ // geopotential height above surface at interface levels (m)
+ dry_atm_.z_iface = buffer_.z_iface;
+
+ // geopotential height above surface at mid levels (m)
+ dry_atm_.z_mid = buffer_.z_mid;
+
+ // total cloud fraction
+ dry_atm_.cldfrac = get_field_in("cldfrac_tot").get_view();
+
+ // computed updraft velocity
+ dry_atm_.w_updraft = buffer_.w_updraft;
+
+ // ------------------------------------------------------------------------
+ // Output fields to be used by other processes
+ // ------------------------------------------------------------------------
+ // ice nucleation output
+ naai_ = get_field_out("ni_activated").get_view();
+
+ // droplet activation output
+ tendnd_ = get_field_out("nc_nuceat_tend").get_view();
+
+ // interstitial and cloudborne aerosol tracers of interest: mass (q) and
+ // number (n) mixing ratios
+ for(int m = 0; m < mam_coupling::num_aero_modes(); ++m) {
+ // interstitial aerosol tracers of interest: number (n) mixing ratios
+ const char *int_nmr_field_name = mam_coupling::int_aero_nmr_field_name(m);
+ wet_aero_.int_aero_nmr[m] =
+ get_field_out(int_nmr_field_name).get_view();
+ dry_aero_.int_aero_nmr[m] = buffer_.dry_int_aero_nmr[m];
+
+ // cloudborne aerosol tracers of interest: number (n) mixing ratios
+ const char *cld_nmr_field_name = mam_coupling::cld_aero_nmr_field_name(m);
+ wet_aero_.cld_aero_nmr[m] =
+ get_field_out(cld_nmr_field_name).get_view();
+ dry_aero_.cld_aero_nmr[m] = buffer_.dry_cld_aero_nmr[m];
+
+ for(int a = 0; a < mam_coupling::num_aero_species(); ++a) {
+ // (interstitial) aerosol tracers of interest: mass (q) mixing ratios
+ const char *int_mmr_field_name =
+ mam_coupling::int_aero_mmr_field_name(m, a);
+
+ if(strlen(int_mmr_field_name) > 0) {
+ wet_aero_.int_aero_mmr[m][a] =
+ get_field_out(int_mmr_field_name).get_view();
+ dry_aero_.int_aero_mmr[m][a] = buffer_.dry_int_aero_mmr[m][a];
+ }
+
+ // (cloudborne) aerosol tracers of interest: mass (q) mixing ratios
+ const char *cld_mmr_field_name =
+ mam_coupling::cld_aero_mmr_field_name(m, a);
+ if(strlen(cld_mmr_field_name) > 0) {
+ wet_aero_.cld_aero_mmr[m][a] =
+ get_field_out(cld_mmr_field_name).get_view();
+ dry_aero_.cld_aero_mmr[m][a] = buffer_.dry_cld_aero_mmr[m][a];
+ }
+ }
+ }
+ for(int g = 0; g < mam_coupling::num_aero_gases(); ++g) {
+ const char *gas_mmr_field_name = mam_coupling::gas_mmr_field_name(g);
+ wet_aero_.gas_mmr[g] =
+ get_field_out(gas_mmr_field_name).get_view();
+ dry_aero_.gas_mmr[g] = buffer_.dry_gas_mmr[g];
+ }
+
+ // hetrozenous freezing outputs
+ hetfrz_immersion_nucleation_tend_ =
+ get_field_out("hetfrz_immersion_nucleation_tend").get_view();
+ hetfrz_contact_nucleation_tend_ =
+ get_field_out("hetfrz_contact_nucleation_tend").get_view();
+ hetfrz_depostion_nucleation_tend_ =
+ get_field_out("hetfrz_depostion_nucleation_tend").get_view();
+
+ //---------------------------------------------------------------------------------
+ // Allocate memory for the class members
+ // (Kokkos::resize only works on host to allocates memory)
+ //---------------------------------------------------------------------------------
+
+ Kokkos::resize(rho_, ncol_, nlev_);
+ Kokkos::resize(w0_, ncol_, nlev_);
+ Kokkos::resize(tke_, ncol_, nlev_ + 1);
+ Kokkos::resize(wsub_, ncol_, nlev_);
+ Kokkos::resize(wsubice_, ncol_, nlev_);
+ Kokkos::resize(wsig_, ncol_, nlev_);
+ Kokkos::resize(w2_, ncol_, nlev_);
+ Kokkos::resize(cloud_frac_, ncol_, nlev_);
+ Kokkos::resize(cloud_frac_prev_, ncol_, nlev_);
+ Kokkos::resize(aitken_dry_dia_, ncol_, nlev_);
+ Kokkos::resize(rpdel_, ncol_, nlev_);
+
+ //---------------------------------------------------------------------------------
+ // Diagnotics variables from the ice nucleation scheme
+ //---------------------------------------------------------------------------------
+
+ // number conc of ice nuclei due to heterogeneous freezing [1/m3]
+ Kokkos::resize(nihf_, ncol_, nlev_);
+
+ // number conc of ice nuclei due to immersionfreezing (hetero nuc) [1/m3]
+ Kokkos::resize(niim_, ncol_, nlev_);
+
+ // number conc of ice nuclei due to deposition nucleation (hetero nuc)[1/m3]
+ Kokkos::resize(nidep_, ncol_, nlev_);
+
+ // number conc of ice nuclei due to meyers deposition [1/m3]
+ Kokkos::resize(nimey_, ncol_, nlev_);
+
+ // number of activated aerosol for ice nucleation(homogeneous frz only)[#/kg]
+ Kokkos::resize(naai_hom_, ncol_, nlev_);
+
+ //---------------------------------------------------------------------------------
+ // Diagnotics variables from the droplet activation scheme
+ //---------------------------------------------------------------------------------
+
+ // activation fraction for aerosol number [fraction]
+ const int num_aero_modes = mam_coupling::num_aero_modes();
+ Kokkos::resize(factnum_, ncol_, num_aero_modes, nlev_);
+
+ // cloud droplet number mixing ratio [#/kg]
+ Kokkos::resize(qcld_, ncol_, nlev_);
+
+ // number conc of aerosols activated at supersat [#/m^3]
+ // NOTE: activation fraction fluxes are defined as
+ // fluxn = [flux of activated aero. number into cloud[#/m^2/s]]
+ // / [aero. number conc. in updraft, just below cloudbase [#/m^3]]
+ Kokkos::resize(ccn_, ncol_, nlev_, mam4::ndrop::psat);
+
+ // column-integrated droplet number [#/m2]
+ Kokkos::resize(ndropcol_, ncol_, nlev_);
+
+ // droplet number mixing ratio tendency due to mixing [#/kg/s]
+ // Kokkos::resize(ndropmix_, ncol_, nlev_);
+ // Temporarily output ndropmix_
+ ndropmix_ = get_field_out("ndropmix").get_view();
+
+ // droplet number mixing ratio source tendency [#/kg/s]
+ // Kokkos::resize(nsource_, ncol_, nlev_);
+ // Temporarily output nsource_
+ nsource_ = get_field_out("nsource").get_view();
+
+ // Temporarily output nc_inp_to_aci_
+ nc_inp_to_aci_ = get_field_out("nc_inp_to_aci").get_view();
+
+ // FIXME: [TEMPORARY] remove the following ccn outputs
+ ccn_0p02_ = get_field_out("ccn_0p02").get_view();
+ ccn_0p05_ = get_field_out("ccn_0p05").get_view();
+ ccn_0p1_ = get_field_out("ccn_0p1").get_view();
+ ccn_0p2_ = get_field_out("ccn_0p2").get_view();
+ ccn_0p5_ = get_field_out("ccn_0p5").get_view();
+ ccn_1p0_ = get_field_out("ccn_1p0").get_view();
+
+ // subgrid vertical velocity [m/s]
+ Kokkos::resize(wtke_, ncol_, nlev_);
+
+ for(int i = 0; i < dropmix_scratch_; ++i) {
+ Kokkos::resize(dropmixnuc_scratch_mem_[i], ncol_, nlev_);
+ }
+ for(int i = 0; i < mam4::ndrop::ncnst_tot; ++i) {
+ // column tendency for diagnostic output
+ Kokkos::resize(coltend_[i], ncol_, nlev_);
+ // column tendency
+ Kokkos::resize(coltend_cw_[i], ncol_, nlev_);
+ }
+ for(int i = 0; i < mam4::aero_model::pcnst; ++i) {
+ Kokkos::resize(ptend_q_[i], ncol_, nlev_);
+ }
+ for(int i = 0; i < mam4::ndrop::pver; ++i) {
+ for(int j = 0; j < 2; ++j) {
+ Kokkos::resize(raercol_cw_[i][j], ncol_, mam4::ndrop::ncnst_tot);
+ Kokkos::resize(raercol_[i][j], ncol_, mam4::ndrop::ncnst_tot);
+ }
+ }
+
+ // nact : fractional aero. number activation rate [/s]
+ Kokkos::resize(nact_, ncol_, nlev_, mam_coupling::num_aero_modes());
+
+ // mact : fractional aero. mass activation rate [/s]
+ Kokkos::resize(mact_, ncol_, nlev_, mam_coupling::num_aero_modes());
+
+ // Eddy diffusivity of heat at the interfaces
+ Kokkos::resize(kvh_int_, ncol_, nlev_ + 1);
+
+ // Vertical velocity variance at the interfaces
+ Kokkos::resize(w_sec_int_, ncol_, nlev_ + 1);
+ // Allocate work arrays
+ for(int icnst = 0; icnst < mam4::ndrop::ncnst_tot; ++icnst) {
+ qqcw_fld_work_[icnst] = view_2d("qqcw_fld_work_", ncol_, nlev_);
+ }
+ state_q_work_ =
+ view_3d("state_q_work_", ncol_, nlev_, mam4::aero_model::pcnst);
+
+ //---------------------------------------------------------------------------------
+ // Diagnotics variables from the hetrozenous ice nucleation scheme
+ //---------------------------------------------------------------------------------
+
+ for(int i = 0; i < hetro_scratch_; ++i)
+ Kokkos::resize(diagnostic_scratch_[i], ncol_, nlev_);
+
+ //---------------------------------------------------------------------------------
+ // Initialize the processes
+ //---------------------------------------------------------------------------------
+
+ mam4::AeroConfig aero_config;
+ // configure the nucleation parameterization
+ mam4::NucleateIce::Config nucleate_ice_config;
+ nucleate_ice_.init(aero_config, nucleate_ice_config);
+
+ // configure the heterogeneous freezing parameterization
+ mam4::Hetfrz::Config hetfrz_config;
+ hetfrz_.init(aero_config, hetfrz_config);
+
+ //---------------------------------------------------------------------------------
+ // Setup preprocessing and post processing
+ //---------------------------------------------------------------------------------
+ // set up our preprocess and postprocess functors
+ preprocess_.initialize(ncol_, nlev_, wet_atm_, wet_aero_, dry_atm_,
+ dry_aero_);
+
+ postprocess_.initialize(ncol_, nlev_, wet_atm_, wet_aero_, dry_atm_,
+ dry_aero_);
+
+} // end function initialize_impl
+
+// ================================================================
+// RUN_IMPL
+// ================================================================
+void MAMAci::run_impl(const double dt) {
+ const auto scan_policy = ekat::ExeSpaceUtils<
+ KT::ExeSpace>::get_thread_range_parallel_scan_team_policy(ncol_, nlev_);
+
+ // preprocess input -- needs a scan for the calculation of local derivied
+ // quantities
+ Kokkos::parallel_for("preprocess", scan_policy, preprocess_);
+ Kokkos::fence();
+
+ haero::ThreadTeamPolicy team_policy(ncol_, Kokkos::AUTO);
+
+ // FIXME: Temporary assignment of nc
+ mam_coupling::copy_view_lev_slice(team_policy, wet_atm_.nc, nlev_, // inputs
+ nc_inp_to_aci_); // output
+
+ compute_w0_and_rho(team_policy, dry_atm_, top_lev_, nlev_,
+ // output
+ w0_, rho_);
+
+ // Get w_sec_int_ from w_sec_mid_
+ compute_values_at_interfaces(team_policy, w_sec_mid_, dry_atm_.dz, nlev_,
+ // output
+ w_sec_int_);
+ Kokkos::fence(); // wait for w_sec_int_ to be computed.
+ compute_tke_using_w_sec(team_policy, w_sec_int_, nlev_,
+ // output
+ tke_);
+
+ Kokkos::fence(); // wait for tke_ to be computed.
+ compute_subgrid_scale_velocities(team_policy, tke_, wsubmin_, top_lev_, nlev_,
+ // output
+ wsub_, wsubice_, wsig_);
+
+ // We need dry diameter for only aitken mode
+ Kokkos::deep_copy(
+ aitken_dry_dia_,
+ ekat::subview_1(dgnum_, static_cast(mam4::ModeIndex::Aitken)));
+
+ Kokkos::fence(); // wait for aitken_dry_dia_ to be computed.
+
+ // Compute Ice nucleation
+ // NOTE: The Fortran version uses "ast" for cloud fraction which is
+ // equivalent to "cldfrac_tot" in FM. It is part of the "dry_atm_" struct
+ compute_nucleate_ice_tendencies(
+ nucleate_ice_, team_policy, dry_atm_, dry_aero_, wsubice_,
+ aitken_dry_dia_, nlev_, dt,
+ // output
+ nihf_, niim_, nidep_, nimey_, naai_hom_,
+ // ## output to be used by the other processes ##
+ naai_);
+
+ // Compute cloud fractions based on cloud threshold
+ store_liquid_cloud_fraction(team_policy, dry_atm_, liqcldf_, liqcldf_prev_,
+ top_lev_, nlev_,
+ // output
+ cloud_frac_, cloud_frac_prev_);
+
+ compute_recipical_pseudo_density(team_policy, dry_atm_.p_del, nlev_,
+ // output
+ rpdel_);
+
+ Kokkos::fence(); // wait for rpdel_ to be computed.
+
+ // Get kvh_int_ from kvh_mid_
+ compute_values_at_interfaces(team_policy, kvh_mid_, dry_atm_.dz, nlev_,
+ // output
+ kvh_int_);
+
+ Kokkos::fence();
+ // Compute activated CCN number tendency (tendnd_) and updated
+ // cloud borne aerosols (stored in a work array) and interstitial
+ // aerosols tendencies
+ call_function_dropmixnuc(team_policy, dt, dry_atm_, rpdel_, kvh_int_, wsub_,
+ cloud_frac_, cloud_frac_prev_, dry_aero_, nlev_,
+ // output
+ coltend_, coltend_cw_, qcld_, ndropcol_, ndropmix_,
+ nsource_, wtke_, ccn_,
+ // ## output to be used by the other processes ##
+ qqcw_fld_work_, ptend_q_, factnum_, tendnd_,
+ // work arrays
+ raercol_cw_, raercol_, state_q_work_, nact_, mact_,
+ dropmixnuc_scratch_mem_);
+ Kokkos::fence(); // wait for ptend_q_ to be computed.
+
+ Kokkos::deep_copy(ccn_0p02_,
+ Kokkos::subview(ccn_, Kokkos::ALL(), Kokkos::ALL(), 0));
+ Kokkos::deep_copy(ccn_0p05_,
+ Kokkos::subview(ccn_, Kokkos::ALL(), Kokkos::ALL(), 1));
+ Kokkos::deep_copy(ccn_0p1_,
+ Kokkos::subview(ccn_, Kokkos::ALL(), Kokkos::ALL(), 2));
+ Kokkos::deep_copy(ccn_0p2_,
+ Kokkos::subview(ccn_, Kokkos::ALL(), Kokkos::ALL(), 3));
+ Kokkos::deep_copy(ccn_0p5_,
+ Kokkos::subview(ccn_, Kokkos::ALL(), Kokkos::ALL(), 4));
+ Kokkos::deep_copy(ccn_1p0_,
+ Kokkos::subview(ccn_, Kokkos::ALL(), Kokkos::ALL(), 5));
+
+ //---------------------------------------------------------------------------
+ // NOTE: DO NOT UPDATE cloud borne aerosols using the qqcw_fld_work_ array
+ // at this point as heterozenous freezing needs to use cloud borne aerosols
+ // before they are changed by the droplet activation (dropmixnuc) process.
+ //---------------------------------------------------------------------------
+
+ // Compute hetrozenous freezing
+ call_hetfrz_compute_tendencies(
+ team_policy, hetfrz_, dry_atm_, dry_aero_, factnum_, dt, nlev_,
+ // ## output to be used by the other processes ##
+ hetfrz_immersion_nucleation_tend_, hetfrz_contact_nucleation_tend_,
+ hetfrz_depostion_nucleation_tend_,
+ // work arrays
+ diagnostic_scratch_);
+
+ //---------------------------------------------------------------
+ // Now update interstitial and cloud borne aerosols
+ //---------------------------------------------------------------
+
+ // Update cloud borne aerosols
+ update_cloud_borne_aerosols(qqcw_fld_work_, nlev_,
+ // output
+ dry_aero_);
+
+ // Update interstitial aerosols using tendencies
+ update_interstitial_aerosols(team_policy, ptend_q_, nlev_, dt,
+ // output
+ dry_aero_);
+
+ // call post processing to convert dry mixing ratios to wet mixing ratios
+ Kokkos::parallel_for("postprocess", scan_policy, postprocess_);
+ Kokkos::fence(); // wait before returning to calling function
+}
+
+} // namespace scream
diff --git a/components/eamxx/src/physics/mam/eamxx_mam_aci_process_interface.hpp b/components/eamxx/src/physics/mam/eamxx_mam_aci_process_interface.hpp
new file mode 100644
index 00000000000..97bc90d3f98
--- /dev/null
+++ b/components/eamxx/src/physics/mam/eamxx_mam_aci_process_interface.hpp
@@ -0,0 +1,274 @@
+#ifndef EAMXX_MAM_ACI_HPP
+#define EAMXX_MAM_ACI_HPP
+
+// For MAM4 aerosol configuration
+#include
+
+// For declaring ACI class derived from atm process class
+#include
+
+// For aerosol configuration
+#include "mam4xx/aero_config.hpp"
+
+// For calling ndrop functions
+#include "mam4xx/ndrop.hpp"
+
+namespace scream {
+
+class MAMAci final : public scream::AtmosphereProcess {
+ public:
+ // declare some constant scratch space lengths
+ static constexpr int hetro_scratch_ = 43;
+ static constexpr int dropmix_scratch_ = 15;
+
+ // views for multi-column data
+ using view_2d = scream::mam_coupling::view_2d;
+ using const_view_2d = scream::mam_coupling::const_view_2d;
+ using view_3d = scream::mam_coupling::view_3d;
+ using const_view_3d = scream::mam_coupling::const_view_3d;
+
+ private:
+ using KT = ekat::KokkosTypes;
+
+ mam4::NucleateIce nucleate_ice_;
+ mam4::Hetfrz hetfrz_;
+
+ // views for single-column data
+ using view_1d = scream::mam_coupling::view_1d;
+ using const_view_1d = scream::mam_coupling::const_view_1d;
+
+ //------------------------------------------------------------------------
+ // ACI runtime ( or namelist) options
+ //------------------------------------------------------------------------
+
+ Real wsubmin_; // Minimum subgrid vertical velocity
+ int top_lev_; // Top level for MAM4xx
+
+ //------------------------------------------------------------------------
+ // END: ACI runtime ( or namelist) options
+ //------------------------------------------------------------------------
+
+ // rho is air density [kg/m3]
+ view_2d rho_;
+
+ // w0_ is large scale velocity (m/s)
+ view_2d w0_;
+
+ // turbulent kinetic energy [m^2/s^2]
+ view_2d tke_;
+
+ // Dry diameter of the aitken mode (for ice nucleation)
+ view_2d aitken_dry_dia_;
+
+ // wet mixing ratios (water species)
+ mam_coupling::WetAtmosphere wet_atm_;
+
+ // dry mixing ratios (water species)
+ mam_coupling::DryAtmosphere dry_atm_;
+
+ // aerosol dry diameter
+ const_view_3d dgnum_;
+
+ // ice nucleation diagnostic variables
+ view_2d nihf_;
+ view_2d niim_;
+ view_2d nidep_;
+ view_2d nimey_;
+ view_2d naai_hom_;
+
+ // ice nucleation output for FM
+ view_2d naai_;
+
+ // droplet activation inputs and outputs
+ view_2d kvh_int_; // Eddy diffusivity of heat at the interfaces
+ const_view_2d liqcldf_;
+ const_view_2d liqcldf_prev_;
+ const_view_2d kvh_mid_; // Eddy diffusivity of heat at the midpoints
+
+ view_2d cloud_frac_;
+ view_2d cloud_frac_prev_;
+ view_2d qcld_;
+ view_2d ptend_q_[mam4::aero_model::pcnst];
+ view_3d factnum_;
+ const_view_3d qqcw_input_;
+ view_2d qqcw_[mam4::ndrop::ncnst_tot];
+ view_2d ndropcol_;
+ view_2d ndropmix_;
+ view_2d nsource_;
+ view_2d nc_inp_to_aci_; // FIXME: TEMPORARY output
+ view_2d ccn_0p02_; // FIXME: TEMPORARY output
+ view_2d ccn_0p05_; // FIXME: TEMPORARY output
+ view_2d ccn_0p1_; // FIXME: TEMPORARY output
+ view_2d ccn_0p2_; // FIXME: TEMPORARY output
+ view_2d ccn_0p5_; // FIXME: TEMPORARY output
+ view_2d ccn_1p0_; // FIXME: TEMPORARY output
+ view_2d wtke_;
+ view_3d ccn_;
+ view_2d coltend_[mam4::ndrop::ncnst_tot];
+ view_2d coltend_cw_[mam4::ndrop::ncnst_tot];
+
+ // raercol_cw_ and raercol_ are work arrays for dropmixnuc, allocated on the
+ // stack.
+ view_2d raercol_cw_[mam4::ndrop::pver][2];
+ view_2d raercol_[mam4::ndrop::pver][2];
+
+ view_3d nact_;
+ view_3d mact_;
+ view_2d dropmixnuc_scratch_mem_[dropmix_scratch_];
+
+ // droplet activation output for the FM
+ view_2d tendnd_;
+
+ // These are the output tendencies from heterogeneous freezing that need to be
+ // added correctly to the cloud-micorphysics scheme.
+ view_2d hetfrz_immersion_nucleation_tend_;
+ view_2d hetfrz_contact_nucleation_tend_;
+ view_2d hetfrz_depostion_nucleation_tend_;
+
+ view_2d diagnostic_scratch_[hetro_scratch_];
+
+ // Subgrid scale velocities
+ view_2d wsub_, wsubice_, wsig_, w2_;
+
+ // local atmospheric state column variables
+ const_view_2d pdel_; // pressure thickess of layer [Pa]
+ view_2d rpdel_; // Inverse of pdel_
+ const_view_2d w_sec_mid_; // Vertical velocity variance at midpoints
+ view_2d w_sec_int_; // Vertical velocity variance at interfaces
+
+ // number of horizontal columns and vertical levels
+ int ncol_, nlev_;
+
+ // aerosol state variables
+ mam_coupling::AerosolState wet_aero_, dry_aero_;
+
+ // workspace manager for internal local variables
+ mam_coupling::Buffer buffer_;
+
+ // physics grid for column information
+ std::shared_ptr grid_;
+
+ // A view array to carry cloud borne aerosol mmrs/nmrs
+ view_2d qqcw_fld_work_[mam4::ndrop::ncnst_tot];
+
+ // A view to carry interstitial aerosol mmrs/nmrs
+ view_3d state_q_work_;
+
+ public:
+ // Constructor
+ MAMAci(const ekat::Comm &comm, const ekat::ParameterList ¶ms);
+
+ // Process metadata: Return type of the process
+ AtmosphereProcessType type() const { return AtmosphereProcessType::Physics; }
+
+ // Return name of the process
+ std::string name() const { return "mam4_aci"; }
+
+ // grid
+ void set_grids(
+ const std::shared_ptr grids_manager) override;
+
+ // management of common atm process memory
+ size_t requested_buffer_size_in_bytes() const {
+ return mam_coupling::buffer_size(ncol_, nlev_);
+ }
+
+ void init_buffers(const ATMBufferManager &buffer_manager) override;
+
+ // process behavior
+ void initialize_impl(const RunType run_type) override;
+ void run_impl(const double dt) override;
+ void finalize_impl(){/*DO NOTHING*/};
+
+ // Atmosphere processes often have a pre-processing step that constructs
+ // required variables from the set of fields stored in the field manager.
+ // This functor implements this step, which is called during run_impl.
+ struct Preprocess {
+ Preprocess() = default;
+ // on host: initializes preprocess functor with necessary state data
+ void initialize(const int ncol, const int nlev,
+ const mam_coupling::WetAtmosphere &wet_atm,
+ const mam_coupling::AerosolState &wet_aero,
+ const mam_coupling::DryAtmosphere &dry_atm,
+ const mam_coupling::AerosolState &dry_aero) {
+ ncol_pre_ = ncol;
+ nlev_pre_ = nlev;
+ wet_atm_pre_ = wet_atm;
+ wet_aero_pre_ = wet_aero;
+ dry_atm_pre_ = dry_atm;
+ dry_aero_pre_ = dry_aero;
+ }
+
+ KOKKOS_INLINE_FUNCTION
+ void operator()(
+ const Kokkos::TeamPolicy::member_type &team) const {
+ const int i = team.league_rank(); // column index
+
+ compute_dry_mixing_ratios(team, wet_atm_pre_, dry_atm_pre_, i);
+ compute_dry_mixing_ratios(team, wet_atm_pre_, wet_aero_pre_,
+ dry_aero_pre_, i);
+ team.team_barrier();
+ // vertical heights has to be computed after computing dry mixing ratios
+ // for atmosphere
+ compute_vertical_layer_heights(team, dry_atm_pre_, i);
+ compute_updraft_velocities(team, wet_atm_pre_, dry_atm_pre_, i);
+ } // operator()
+
+ // local variables for preprocess struct
+ // number of horizontal columns and vertical levels
+ int ncol_pre_, nlev_pre_;
+
+ // local atmospheric and aerosol state data
+ mam_coupling::WetAtmosphere wet_atm_pre_;
+ mam_coupling::DryAtmosphere dry_atm_pre_;
+ mam_coupling::AerosolState wet_aero_pre_, dry_aero_pre_;
+ }; // MAMAci::Preprocess
+
+ // Atmosphere processes often have a post-processing step prepares output
+ // from this process for the Field Manager. This functor implements this
+ // step, which is called during run_impl.
+ // Postprocessing functor
+ struct Postprocess {
+ Postprocess() = default;
+
+ // on host: initializes postprocess functor with necessary state data
+ void initialize(const int ncol, const int nlev,
+ const mam_coupling::WetAtmosphere &wet_atm,
+ const mam_coupling::AerosolState &wet_aero,
+ const mam_coupling::DryAtmosphere &dry_atm,
+ const mam_coupling::AerosolState &dry_aero) {
+ ncol_post_ = ncol;
+ nlev_post_ = nlev;
+ wet_atm_post_ = wet_atm;
+ wet_aero_post_ = wet_aero;
+ dry_atm_post_ = dry_atm;
+ dry_aero_post_ = dry_aero;
+ }
+
+ KOKKOS_INLINE_FUNCTION
+ void operator()(
+ const Kokkos::TeamPolicy::member_type &team) const {
+ const int i = team.league_rank(); // column index
+ compute_wet_mixing_ratios(team, dry_atm_post_, dry_aero_post_,
+ wet_aero_post_, i);
+ } // operator()
+
+ // number of horizontal columns and vertical levels
+ int ncol_post_, nlev_post_;
+
+ // local atmospheric and aerosol state data
+ mam_coupling::WetAtmosphere wet_atm_post_;
+ mam_coupling::DryAtmosphere dry_atm_post_;
+ mam_coupling::AerosolState wet_aero_post_, dry_aero_post_;
+ }; // Postprocess
+
+ private:
+ // pre- and postprocessing scratch pads
+ Preprocess preprocess_;
+ Postprocess postprocess_;
+
+}; // MAMAci
+
+} // namespace scream
+
+#endif // EAMXX_MAM_ACI_HPP
diff --git a/components/eamxx/src/physics/mam/eamxx_mam_microphysics_process_interface.cpp b/components/eamxx/src/physics/mam/eamxx_mam_microphysics_process_interface.cpp
index 8bc727c325e..7ea2adfddb6 100644
--- a/components/eamxx/src/physics/mam/eamxx_mam_microphysics_process_interface.cpp
+++ b/components/eamxx/src/physics/mam/eamxx_mam_microphysics_process_interface.cpp
@@ -201,7 +201,7 @@ void MAMMicrophysics::initialize_impl(const RunType run_type) {
wet_atm_.nc = get_field_out("nc").get_view();
wet_atm_.qi = get_field_in("qi").get_view();
wet_atm_.ni = get_field_in("ni").get_view();
- wet_atm_.omega = get_field_in("omega").get_view();
+
dry_atm_.T_mid = get_field_in("T_mid").get_view();
dry_atm_.p_mid = get_field_in("p_mid").get_view();
@@ -209,6 +209,7 @@ void MAMMicrophysics::initialize_impl(const RunType run_type) {
dry_atm_.cldfrac = get_field_in("cldfrac_tot").get_view(); // FIXME: tot or liq?
dry_atm_.pblh = get_field_in("pbl_height").get_view();
dry_atm_.phis = get_field_in("phis").get_view();
+ dry_atm_.omega = get_field_in("omega").get_view();
dry_atm_.z_mid = buffer_.z_mid;
dry_atm_.dz = buffer_.dz;
dry_atm_.z_iface = buffer_.z_iface;
@@ -305,6 +306,12 @@ void MAMMicrophysics::run_impl(const double dt) {
// allocation perspective
auto o3_col_dens = buffer_.scratch[8];
+ const_view_1d &col_latitudes = col_latitudes_;
+ mam_coupling::DryAtmosphere &dry_atm = dry_atm_;
+ mam_coupling::AerosolState &dry_aero = dry_aero_;
+ mam4::mo_photo::PhotoTableData &photo_table = photo_table_;
+ const int nlev = nlev_;
+ const Config &config = config_;
// FIXME: read relevant linoz climatology data from file(s) based on time
// FIXME: read relevant chlorine loading data from file based on time
@@ -313,21 +320,21 @@ void MAMMicrophysics::run_impl(const double dt) {
Kokkos::parallel_for(policy, KOKKOS_LAMBDA(const ThreadTeam& team) {
const int icol = team.league_rank(); // column index
- Real col_lat = col_latitudes_(icol); // column latitude (degrees?)
+ Real col_lat = col_latitudes(icol); // column latitude (degrees?)
// fetch column-specific atmosphere state data
- auto atm = mam_coupling::atmosphere_for_column(dry_atm_, icol);
- auto z_iface = ekat::subview(dry_atm_.z_iface, icol);
- Real phis = dry_atm_.phis(icol);
+ auto atm = mam_coupling::atmosphere_for_column(dry_atm, icol);
+ auto z_iface = ekat::subview(dry_atm.z_iface, icol);
+ Real phis = dry_atm.phis(icol);
// set surface state data
haero::Surface sfc{};
// fetch column-specific subviews into aerosol prognostics
- mam4::Prognostics progs = mam_coupling::interstitial_aerosols_for_column(dry_aero_, icol);
+ mam4::Prognostics progs = mam_coupling::interstitial_aerosols_for_column(dry_aero, icol);
// set up diagnostics
- mam4::Diagnostics diags(nlev_);
+ mam4::Diagnostics diags(nlev);
// calculate o3 column densities (first component of col_dens in Fortran code)
auto o3_col_dens_i = ekat::subview(o3_col_dens, icol);
@@ -346,7 +353,7 @@ void MAMMicrophysics::run_impl(const double dt) {
mam4::ColumnView lwc; // FIXME: liquid water cloud content: where do we get this?
mam4::mo_photo::table_photo(photo_rates, atm.pressure, atm.hydrostatic_dp,
atm.temperature, o3_col_dens_i, zenith_angle, surf_albedo, lwc,
- atm.cloud_fraction, esfact, photo_table_, photo_work_arrays);
+ atm.cloud_fraction, esfact, photo_table, photo_work_arrays);
// compute external forcings at time t(n+1) [molecules/cm^3/s]
constexpr int extcnt = mam4::gas_chemistry::extcnt;
@@ -355,7 +362,7 @@ void MAMMicrophysics::run_impl(const double dt) {
mam4::mo_setext::extfrc_set(forcings, extfrc);
// compute aerosol microphysics on each vertical level within this column
- Kokkos::parallel_for(Kokkos::TeamThreadRange(team, nlev_), [&](const int k) {
+ Kokkos::parallel_for(Kokkos::TeamThreadRange(team, nlev), [&](const int k) {
constexpr int num_modes = mam4::AeroConfig::num_modes();
constexpr int gas_pcnst = mam_coupling::gas_pcnst();
@@ -428,7 +435,7 @@ void MAMMicrophysics::run_impl(const double dt) {
constexpr int indexm = 0; // FIXME: index of xhnm in invariants array (??)
Real cldnum = 0.0; // FIXME: droplet number concentration: where do we get this?
setsox_single_level(loffset, dt, pmid, pdel, temp, mbar, lwc(k),
- cldfrac, cldnum, invariants[indexm], config_.setsox, vmrcw, vmr);
+ cldfrac, cldnum, invariants[indexm], config.setsox, vmrcw, vmr);
// calculate aerosol water content using water uptake treatment
// * dry and wet diameters [m]
@@ -441,7 +448,7 @@ void MAMMicrophysics::run_impl(const double dt) {
impl::compute_water_content(progs, k, qv, temp, pmid, dgncur_a, dgncur_awet, wetdens, qaerwat);
// do aerosol microphysics (gas-aerosol exchange, nucleation, coagulation)
- impl::modal_aero_amicphys_intr(config_.amicphys, step_, dt, t, pmid, pdel,
+ impl::modal_aero_amicphys_intr(config.amicphys, step_, dt, t, pmid, pdel,
zm, pblh, qv, cldfrac, vmr, vmrcw, vmr_pregaschem,
vmr_precldchem, vmrcw_precldchem, vmr_tendbb,
vmrcw_tendbb, dgncur_a, dgncur_awet,
@@ -466,15 +473,15 @@ void MAMMicrophysics::run_impl(const double dt) {
linoz_o3_clim(icol, k), linoz_t_clim(icol, k), linoz_o3col_clim(icol, k),
linoz_PmL_clim(icol, k), linoz_dPmL_dO3(icol, k), linoz_dPmL_dT(icol, k),
linoz_dPmL_dO3col(icol, k), linoz_cariolle_psc(icol, k),
- chlorine_loading, config_.linoz.psc_T, vmr[o3_ndx],
+ chlorine_loading, config.linoz.psc_T, vmr[o3_ndx],
do3_linoz, do3_linoz_psc, ss_o3,
o3col_du_diag, o3clim_linoz_diag, zenith_angle_degrees);
// update source terms above the ozone decay threshold
- if (k > nlev_ - config_.linoz.o3_lbl - 1) {
+ if (k > nlev - config.linoz.o3_lbl - 1) {
Real do3_mass; // diagnostic, not needed
- mam4::lin_strat_chem::lin_strat_sfcsink_kk(dt, pdel, vmr[o3_ndx], config_.linoz.o3_sfc,
- config_.linoz.o3_tau, do3_mass);
+ mam4::lin_strat_chem::lin_strat_sfcsink_kk(dt, pdel, vmr[o3_ndx], config.linoz.o3_sfc,
+ config.linoz.o3_tau, do3_mass);
}
// ... check for negative values and reset to zero
diff --git a/components/eamxx/src/physics/mam/eamxx_mam_optics_process_interface.cpp b/components/eamxx/src/physics/mam/eamxx_mam_optics_process_interface.cpp
index a55bf84ca17..442a6500f21 100644
--- a/components/eamxx/src/physics/mam/eamxx_mam_optics_process_interface.cpp
+++ b/components/eamxx/src/physics/mam/eamxx_mam_optics_process_interface.cpp
@@ -140,7 +140,7 @@ void MAMOptics::initialize_impl(const RunType run_type) {
wet_atm_.nc = get_field_in("nc").get_view();
wet_atm_.qi = get_field_in("qi").get_view();
wet_atm_.ni = get_field_in("ni").get_view();
- wet_atm_.omega = get_field_in("omega").get_view();
+
constexpr int ntot_amode = mam4::AeroConfig::num_modes();
@@ -153,6 +153,7 @@ void MAMOptics::initialize_impl(const RunType run_type) {
.get_view(); // FIXME: tot or liq?
dry_atm_.pblh = get_field_in("pbl_height").get_view();
dry_atm_.phis = get_field_in("phis").get_view();
+ dry_atm_.omega = get_field_in("omega").get_view();
dry_atm_.z_mid = buffer_.z_mid;
dry_atm_.dz = buffer_.dz;
diff --git a/components/eamxx/src/physics/mam/mam_coupling.hpp b/components/eamxx/src/physics/mam/mam_coupling.hpp
index 278e97c033e..3a001146882 100644
--- a/components/eamxx/src/physics/mam/mam_coupling.hpp
+++ b/components/eamxx/src/physics/mam/mam_coupling.hpp
@@ -21,6 +21,7 @@ using view_2d = typename KT::template view_2d;
using view_3d = typename KT::template view_3d;
using const_view_1d = typename KT::template view_1d;
using const_view_2d = typename KT::template view_2d;
+using const_view_3d = typename KT::template view_3d;
using complex_view_3d = typename KT::template view_3d>;
using complex_view_2d = typename KT::template view_2d>;
@@ -76,7 +77,7 @@ constexpr int num_aero_tracers() {
// Given a MAM aerosol mode index, returns a string denoting the symbolic
// name of the mode.
-inline
+KOKKOS_INLINE_FUNCTION
const char* aero_mode_name(const int mode) {
static const char *mode_names[num_aero_modes()] = {
"1",
@@ -87,9 +88,25 @@ const char* aero_mode_name(const int mode) {
return mode_names[mode];
}
+// Given a MAM aerosol species ID, returns a string denoting the symbolic
+// name of the species.
+KOKKOS_INLINE_FUNCTION
+const char* aero_species_name(const int species_id) {
+ static const char *species_names[num_aero_species()] = {
+ "soa",
+ "so4",
+ "pom",
+ "bc",
+ "nacl",
+ "dst",
+ "mom",
+ };
+ return species_names[species_id];
+}
+
// Given a MAM aerosol-related gas ID, returns a string denoting the symbolic
// name of the gas species.
-inline
+KOKKOS_INLINE_FUNCTION
const char* gas_species_name(const int gas_id) {
static const char *species_names[num_aero_gases()] = {
"O3",
@@ -110,14 +127,14 @@ constexpr int max_field_name_len() {
return 128;
}
-inline
+KOKKOS_INLINE_FUNCTION
size_t gpu_strlen(const char* s) {
size_t l = 0;
while (s[l]) ++l;
return l;
}
-inline
+KOKKOS_INLINE_FUNCTION
void concat_2_strings(const char *s1, const char *s2, char *concatted) {
size_t len1 = gpu_strlen(s1);
for (size_t i = 0; i < len1; ++i)
@@ -128,7 +145,7 @@ void concat_2_strings(const char *s1, const char *s2, char *concatted) {
concatted[len1+len2] = 0;
}
-inline
+KOKKOS_INLINE_FUNCTION
void concat_3_strings(const char *s1, const char *s2, const char *s3, char *concatted) {
size_t len1 = gpu_strlen(s1);
for (size_t i = 0; i < len1; ++i)
@@ -142,31 +159,31 @@ void concat_3_strings(const char *s1, const char *s2, const char *s3, char *conc
concatted[len1+len2+len3] = 0;
}
-inline
+KOKKOS_INLINE_FUNCTION
char* int_aero_nmr_names(int mode) {
static char int_aero_nmr_names_[num_aero_modes()][max_field_name_len()] = {};
return int_aero_nmr_names_[mode];
}
-inline
+KOKKOS_INLINE_FUNCTION
char* cld_aero_nmr_names(int mode) {
static char cld_aero_nmr_names_[num_aero_modes()][max_field_name_len()] = {};
return cld_aero_nmr_names_[mode];
}
-inline
+KOKKOS_INLINE_FUNCTION
char* int_aero_mmr_names(int mode, int species) {
static char int_aero_mmr_names_[num_aero_modes()][num_aero_species()][max_field_name_len()] = {};
return int_aero_mmr_names_[mode][species];
}
-inline
+KOKKOS_INLINE_FUNCTION
char* cld_aero_mmr_names(int mode, int species) {
static char cld_aero_mmr_names_[num_aero_modes()][num_aero_species()][max_field_name_len()] = {};
return cld_aero_mmr_names_[mode][species];
}
-inline
+KOKKOS_INLINE_FUNCTION
char* gas_mmr_names(int gas_id) {
static char gas_mmr_names_[num_aero_gases()][max_field_name_len()] = {};
return gas_mmr_names_[gas_id];
@@ -176,7 +193,7 @@ char* gas_mmr_names(int gas_id) {
// Given a MAM aerosol mode index, returns the name of the related interstitial
// modal number mixing ratio field in EAMxx ("num_a<1-based-mode-index>")
-inline
+KOKKOS_INLINE_FUNCTION
const char* int_aero_nmr_field_name(const int mode) {
if (!int_aero_nmr_names(mode)[0]) {
concat_2_strings("num_a", aero_mode_name(mode), int_aero_nmr_names(mode));
@@ -186,7 +203,7 @@ const char* int_aero_nmr_field_name(const int mode) {
// Given a MAM aerosol mode index, returns the name of the related cloudborne
// modal number mixing ratio field in EAMxx ("num_c<1-based-mode-index>>")
-inline
+KOKKOS_INLINE_FUNCTION
const char* cld_aero_nmr_field_name(const int mode) {
if (!cld_aero_nmr_names(mode)[0]) {
concat_2_strings("num_c", aero_mode_name(mode), cld_aero_nmr_names(mode));
@@ -199,13 +216,12 @@ const char* cld_aero_nmr_field_name(const int mode) {
// field in EAMxx. The form of the field name is "_a<1-based-mode-index>".
// If the desired species is not present within the desire mode, returns a blank
// string ("").
-inline
+KOKKOS_INLINE_FUNCTION
const char* int_aero_mmr_field_name(const int mode, const int species) {
if (!int_aero_mmr_names(mode, species)[0]) {
const auto aero_id = mam4::mode_aero_species(mode, species);
if (aero_id != mam4::AeroId::None) {
- auto aerosol_species_name =mam4::aero_id_short_name(aero_id);
- concat_3_strings(aerosol_species_name.c_str(),
+ concat_3_strings(aero_species_name(static_cast(aero_id)),
"_a", aero_mode_name(mode),
int_aero_mmr_names(mode, species));
}
@@ -218,13 +234,12 @@ const char* int_aero_mmr_field_name(const int mode, const int species) {
// field in EAMxx. The form of the field name is "_c<1-based-mode-index>".
// If the desired species is not present within the desire mode, returns a blank
// string ("").
-inline
+KOKKOS_INLINE_FUNCTION
const char* cld_aero_mmr_field_name(const int mode, const int species) {
if (!cld_aero_mmr_names(mode, species)[0]) {
const auto aero_id = mam4::mode_aero_species(mode, species);
if (aero_id != mam4::AeroId::None) {
- auto aerosol_species_name =mam4::aero_id_short_name(aero_id);
- concat_3_strings(aerosol_species_name.c_str(),
+ concat_3_strings(aero_species_name(static_cast(aero_id)),
"_c", aero_mode_name(mode),
cld_aero_mmr_names(mode, species));
}
@@ -234,7 +249,7 @@ const char* cld_aero_mmr_field_name(const int mode, const int species) {
// Given a MAM aerosol-related gas identifier, returns the name of its mass
// mixing ratio field in EAMxx
-inline
+KOKKOS_INLINE_FUNCTION
const char* gas_mmr_field_name(const int gas) {
return const_cast(gas_species_name(gas));
}
@@ -247,7 +262,6 @@ struct WetAtmosphere {
const_view_2d nc; // wet cloud liquid water number mixing ratio [# / kg moist air]
const_view_2d qi; // wet cloud ice water mass mixing ratio [kg cloud ice water / kg moist air]
const_view_2d ni; // wet cloud ice water number mixing ratio [# / kg moist air]
- const_view_2d omega; // vertical pressure velocity [Pa/s]
};
// This type stores multi-column views related to the dry atmospheric state
@@ -265,11 +279,12 @@ struct DryAtmosphere {
view_2d z_iface; // height at layer interfaces [m]
view_2d dz; // layer thickness [m]
const_view_2d p_del; // hydrostatic "pressure thickness" at grid interfaces [Pa]
- const_view_2d p_int; // total pressure at grid interfaces [Pa]
+ const_view_2d p_int; // total pressure at grid interfaces [Pa]
const_view_2d cldfrac; // cloud fraction [-]
view_2d w_updraft; // updraft velocity [m/s]
const_view_1d pblh; // planetary boundary layer height [m]
const_view_1d phis; // surface geopotential [m2/s2]
+ const_view_2d omega; // vertical pressure velocity [Pa/s]
};
// This type stores aerosol number and mass mixing ratios evolved by MAM. It
@@ -584,9 +599,12 @@ void compute_vertical_layer_heights(const Team& team,
const auto p_mid = ekat::subview(dry_atm.p_mid, column_index);
const auto T_mid = ekat::subview(dry_atm.T_mid, column_index);
const auto pseudo_density = ekat::subview(dry_atm.p_del, column_index);
- PF::calculate_dz(team, pseudo_density, p_mid, T_mid, qv, dz);
+ // NOTE: we are using dry qv. Does calculate_dz require dry or wet?
+ PF::calculate_dz(team, pseudo_density, p_mid, T_mid, qv, // inputs
+ dz);//output
team.team_barrier();
- PF::calculate_z_int(team, mam4::nlev, dz, dry_atm.z_surf, z_iface);
+ PF::calculate_z_int(team, mam4::nlev, dz, dry_atm.z_surf, //inputs
+ z_iface); //output
team.team_barrier(); // likely necessary to have z_iface up to date
PF::calculate_z_mid(team, mam4::nlev, z_iface, z_mid);
}
@@ -605,8 +623,9 @@ void compute_updraft_velocities(const Team& team,
constexpr int nlev = mam4::nlev;
int i = column_index;
Kokkos::parallel_for(Kokkos::TeamVectorRange(team, nlev), [&] (const int k) {
- const auto rho = PF::calculate_density(dry_atm.p_del(i,k), dry_atm.dz(i,k));
- dry_atm.w_updraft(i,k) = PF::calculate_vertical_velocity(wet_atm.omega(i,k), rho);
+ dry_atm.dz(i,k) = PF::calculate_dz(dry_atm.p_del(i,k), dry_atm.p_mid(i,k), dry_atm.T_mid(i,k), wet_atm.qv(i,k));
+ const auto rho = PF::calculate_density(dry_atm.p_del(i,k), dry_atm.dz(i,k));
+ dry_atm.w_updraft(i,k) = PF::calculate_vertical_velocity(dry_atm.omega(i,k), rho);
});
}
@@ -705,6 +724,23 @@ void compute_wet_mixing_ratios(const Team& team,
});
}
+// Scream (or EAMxx) can sometimes extend views beyond model levels (nlev) as it uses
+// "packs". Following function copies a 2d view till model levels
+inline
+void copy_view_lev_slice(haero::ThreadTeamPolicy team_policy, //inputs
+ const_view_2d &inp_view, //input view to copy
+ const int dim, //dimension till view should be copied
+ view_2d &out_view) { //output view
+
+ Kokkos::parallel_for(
+ team_policy, KOKKOS_LAMBDA(const haero::ThreadTeam &team) {
+ const int icol = team.league_rank();
+ Kokkos::parallel_for(Kokkos::TeamThreadRange(team, dim), [&](int kk) {
+ out_view(icol, kk) = inp_view(icol, kk);
+ });
+ });
+ }
+
// Because CUDA C++ doesn't allow us to declare and use constants outside of
// KOKKOS_INLINE_FUNCTIONS, we define this macro that allows us to (re)define
// these constants where needed within two such functions so we don't define
@@ -885,6 +921,7 @@ void transfer_work_arrays_to_prognostics(const Real q[gas_pcnst()],
progs.q_aero_i[m][a](k) = q[i];
progs.q_aero_c[m][a](k) = qqcw[i];
} else { // constituent is a modal number mixing ratio
+ int m = static_cast(mode_index);
progs.n_mode_i[m](k) = q[i];
progs.n_mode_c[m](k) = qqcw[i];
}
diff --git a/components/eamxx/src/physics/register_physics.hpp b/components/eamxx/src/physics/register_physics.hpp
index 329b7022756..36afabf8de9 100644
--- a/components/eamxx/src/physics/register_physics.hpp
+++ b/components/eamxx/src/physics/register_physics.hpp
@@ -26,6 +26,7 @@
#ifdef EAMXX_HAS_MAM
#include "physics/mam/eamxx_mam_microphysics_process_interface.hpp"
#include "physics/mam/eamxx_mam_optics_process_interface.hpp"
+#include "physics/mam/eamxx_mam_aci_process_interface.hpp"
#endif
#ifdef EAMXX_HAS_COSP
#include "physics/cosp/eamxx_cosp.hpp"
@@ -62,6 +63,7 @@ inline void register_physics () {
#ifdef EAMXX_HAS_MAM
proc_factory.register_product("mam4_micro",&create_atmosphere_process);
proc_factory.register_product("mam4_optics",&create_atmosphere_process);
+ proc_factory.register_product("mam4_aci",&create_atmosphere_process);
#endif
#ifdef EAMXX_HAS_COSP
proc_factory.register_product("Cosp",&create_atmosphere_process);
diff --git a/components/eamxx/src/physics/shoc/eamxx_shoc_process_interface.cpp b/components/eamxx/src/physics/shoc/eamxx_shoc_process_interface.cpp
index 6204421c1ea..15d43c82f7c 100644
--- a/components/eamxx/src/physics/shoc/eamxx_shoc_process_interface.cpp
+++ b/components/eamxx/src/physics/shoc/eamxx_shoc_process_interface.cpp
@@ -85,6 +85,9 @@ void SHOCMacrophysics::set_grids(const std::shared_ptr grids
// Output variables
add_field("pbl_height", scalar2d , m, grid_name);
add_field("inv_qc_relvar", scalar3d_mid, pow(kg/kg,2), grid_name, ps);
+ add_field("eddy_diff_heat", scalar3d_mid, m2/s, grid_name, ps);
+ add_field("w_variance", scalar3d_mid, m2/s2, grid_name, ps);
+ add_field("cldfrac_liq_prev", scalar3d_mid, nondim, grid_name, ps);
// Tracer group
add_group("tracers", grid_name, ps, Bundling::Required);
@@ -178,7 +181,7 @@ void SHOCMacrophysics::init_buffers(const ATMBufferManager &buffer_manager)
&m_buffer.inv_exner, &m_buffer.thlm, &m_buffer.qw, &m_buffer.dse, &m_buffer.tke_copy, &m_buffer.qc_copy,
&m_buffer.shoc_ql2, &m_buffer.shoc_mix, &m_buffer.isotropy, &m_buffer.w_sec, &m_buffer.wqls_sec, &m_buffer.brunt
#ifdef SCREAM_SMALL_KERNELS
- , &m_buffer.rho_zt, &m_buffer.shoc_qv, &m_buffer.tabs, &m_buffer.dz_zt, &m_buffer.tkh
+ , &m_buffer.rho_zt, &m_buffer.shoc_qv, &m_buffer.tabs, &m_buffer.dz_zt
#endif
};
@@ -250,6 +253,7 @@ void SHOCMacrophysics::initialize_impl (const RunType run_type)
const auto& qv = get_field_out("qv").get_view();
const auto& tke = get_field_out("tke").get_view();
const auto& cldfrac_liq = get_field_out("cldfrac_liq").get_view();
+ const auto& cldfrac_liq_prev = get_field_out("cldfrac_liq_prev").get_view();
const auto& sgs_buoy_flux = get_field_out("sgs_buoy_flux").get_view();
const auto& tk = get_field_out("eddy_diff_mom").get_view();
const auto& inv_qc_relvar = get_field_out("inv_qc_relvar").get_view();
@@ -294,7 +298,7 @@ void SHOCMacrophysics::initialize_impl (const RunType run_type)
T_mid,p_mid,p_int,pseudo_density,omega,phis,surf_sens_flux,surf_evap,
surf_mom_flux,qtracers,qv,qc,qc_copy,tke,tke_copy,z_mid,z_int,
dse,rrho,rrho_i,thv,dz,zt_grid,zi_grid,wpthlp_sfc,wprtp_sfc,upwp_sfc,vpwp_sfc,
- wtracer_sfc,wm_zt,inv_exner,thlm,qw);
+ wtracer_sfc,wm_zt,inv_exner,thlm,qw, cldfrac_liq, cldfrac_liq_prev);
// Input Variables:
input.zt_grid = shoc_preprocess.zt_grid;
@@ -327,11 +331,12 @@ void SHOCMacrophysics::initialize_impl (const RunType run_type)
// Output Variables
output.pblh = get_field_out("pbl_height").get_view();
output.shoc_ql2 = shoc_ql2;
+ output.tkh = get_field_out("eddy_diff_heat").get_view();
// Ouput (diagnostic)
history_output.shoc_mix = m_buffer.shoc_mix;
history_output.isotropy = m_buffer.isotropy;
- history_output.w_sec = m_buffer.w_sec;
+ history_output.w_sec = get_field_out("w_variance").get_view();
history_output.thl_sec = m_buffer.thl_sec;
history_output.qw_sec = m_buffer.qw_sec;
history_output.qwthl_sec = m_buffer.qwthl_sec;
@@ -364,7 +369,6 @@ void SHOCMacrophysics::initialize_impl (const RunType run_type)
temporaries.tabs = m_buffer.tabs;
temporaries.dz_zt = m_buffer.dz_zt;
temporaries.dz_zi = m_buffer.dz_zi;
- temporaries.tkh = m_buffer.tkh;
#endif
shoc_postprocess.set_variables(m_num_cols,m_num_levs,m_num_tracers,
diff --git a/components/eamxx/src/physics/shoc/eamxx_shoc_process_interface.hpp b/components/eamxx/src/physics/shoc/eamxx_shoc_process_interface.hpp
index b21a35d829f..d4403c5bce1 100644
--- a/components/eamxx/src/physics/shoc/eamxx_shoc_process_interface.hpp
+++ b/components/eamxx/src/physics/shoc/eamxx_shoc_process_interface.hpp
@@ -85,6 +85,9 @@ class SHOCMacrophysics : public scream::AtmosphereProcess
Kokkos::parallel_for(Kokkos::TeamVectorRange(team, nlev_packs), [&] (const Int& k) {
+
+ cldfrac_liq_prev(i,k)=cldfrac_liq(i,k);
+
const auto range = ekat::range(k*Spack::n);
const Smask in_nlev_range = (range < nlev);
@@ -200,6 +203,8 @@ class SHOCMacrophysics : public scream::AtmosphereProcess
view_2d thlm;
view_2d qw;
view_2d cloud_frac;
+ view_2d cldfrac_liq;
+ view_2d cldfrac_liq_prev;
// Assigning local variables
void set_variables(const int ncol_, const int nlev_, const int num_qtracers_,
@@ -215,7 +220,8 @@ class SHOCMacrophysics : public scream::AtmosphereProcess
const view_2d& dse_, const view_2d& rrho_, const view_2d& rrho_i_,
const view_2d& thv_, const view_2d& dz_,const view_2d& zt_grid_,const view_2d& zi_grid_, const view_1d& wpthlp_sfc_,
const view_1d& wprtp_sfc_,const view_1d& upwp_sfc_,const view_1d& vpwp_sfc_, const view_2d& wtracer_sfc_,
- const view_2d& wm_zt_,const view_2d& inv_exner_,const view_2d& thlm_,const view_2d& qw_)
+ const view_2d& wm_zt_,const view_2d& inv_exner_,const view_2d& thlm_,const view_2d& qw_,
+ const view_2d& cldfrac_liq_, const view_2d& cldfrac_liq_prev_)
{
ncol = ncol_;
nlev = nlev_;
@@ -256,6 +262,8 @@ class SHOCMacrophysics : public scream::AtmosphereProcess
inv_exner = inv_exner_;
thlm = thlm_;
qw = qw_;
+ cldfrac_liq=cldfrac_liq_;
+ cldfrac_liq_prev=cldfrac_liq_prev_;
} // set_variables
}; // SHOCPreprocess
/* --------------------------------------------------------------------------------------------*/
diff --git a/components/eamxx/src/physics/shoc/impl/shoc_main_impl.hpp b/components/eamxx/src/physics/shoc/impl/shoc_main_impl.hpp
index 13a93992cff..120ed505f26 100644
--- a/components/eamxx/src/physics/shoc/impl/shoc_main_impl.hpp
+++ b/components/eamxx/src/physics/shoc/impl/shoc_main_impl.hpp
@@ -119,6 +119,7 @@ void Functions::shoc_main_internal(
// Output Variables
Scalar& pblh,
const uview_1d& shoc_ql2,
+ const uview_1d& tkh,
// Diagnostic Output Variables
const uview_1d& shoc_mix,
const uview_1d& w_sec,
@@ -137,10 +138,10 @@ void Functions::shoc_main_internal(
{
// Define temporary variables
- uview_1d rho_zt, shoc_qv, shoc_tabs, dz_zt, dz_zi, tkh;
- workspace.template take_many_and_reset<6>(
- {"rho_zt", "shoc_qv", "shoc_tabs", "dz_zt", "dz_zi", "tkh"},
- {&rho_zt, &shoc_qv, &shoc_tabs, &dz_zt, &dz_zi, &tkh});
+ uview_1d rho_zt, shoc_qv, shoc_tabs, dz_zt, dz_zi;
+ workspace.template take_many_and_reset<5>(
+ {"rho_zt", "shoc_qv", "shoc_tabs", "dz_zt", "dz_zi"},
+ {&rho_zt, &shoc_qv, &shoc_tabs, &dz_zt, &dz_zi});
// Local scalars
Scalar se_b{0}, ke_b{0}, wv_b{0}, wl_b{0},
@@ -311,8 +312,8 @@ void Functions::shoc_main_internal(
pblh); // Output
// Release temporary variables from the workspace
- workspace.template release_many_contiguous<6>(
- {&rho_zt, &shoc_qv, &shoc_tabs, &dz_zt, &dz_zi, &tkh});
+ workspace.template release_many_contiguous<5>(
+ {&rho_zt, &shoc_qv, &shoc_tabs, &dz_zt, &dz_zi});
}
#else
template
@@ -371,6 +372,7 @@ void Functions::shoc_main_internal(
// Output Variables
const view_1d& pblh,
const view_2d& shoc_ql2,
+ const view_2d& tkh,
// Diagnostic Output Variables
const view_2d& shoc_mix,
const view_2d& w_sec,
@@ -404,8 +406,7 @@ void Functions::shoc_main_internal(
const view_2d& shoc_qv,
const view_2d& shoc_tabs,
const view_2d& dz_zt,
- const view_2d& dz_zi,
- const view_2d& tkh)
+ const view_2d& dz_zi)
{
// Scalarize some views for single entry access
const auto s_thetal = ekat::scalarize(thetal);
@@ -643,6 +644,7 @@ Int Functions::shoc_main(
const auto shoc_cldfrac_s = ekat::subview(shoc_input_output.shoc_cldfrac, i);
const auto shoc_ql_s = ekat::subview(shoc_input_output.shoc_ql, i);
const auto shoc_ql2_s = ekat::subview(shoc_output.shoc_ql2, i);
+ const auto tkh_s = ekat::subview(shoc_output.tkh, i);
const auto shoc_mix_s = ekat::subview(shoc_history_output.shoc_mix, i);
const auto w_sec_s = ekat::subview(shoc_history_output.w_sec, i);
const auto thl_sec_s = ekat::subview(shoc_history_output.thl_sec, i);
@@ -674,7 +676,7 @@ Int Functions::shoc_main(
host_dse_s, tke_s, thetal_s, qw_s, u_wind_s, v_wind_s, // Input/Output
wthv_sec_s, qtracers_s, tk_s, shoc_cldfrac_s, // Input/Output
shoc_ql_s, // Input/Output
- pblh_s, shoc_ql2_s, // Output
+ pblh_s, shoc_ql2_s, tkh_s, // Output
shoc_mix_s, w_sec_s, thl_sec_s, qw_sec_s, qwthl_sec_s, // Diagnostic Output Variables
wthl_sec_s, wqw_sec_s, wtke_sec_s, uw_sec_s, vw_sec_s, // Diagnostic Output Variables
w3_s, wqls_sec_s, brunt_s, isotropy_s); // Diagnostic Output Variables
@@ -698,7 +700,7 @@ Int Functions::shoc_main(
shoc_input_output.host_dse, shoc_input_output.tke, shoc_input_output.thetal, shoc_input_output.qw, u_wind_s, v_wind_s, // Input/Output
shoc_input_output.wthv_sec, shoc_input_output.qtracers, shoc_input_output.tk, shoc_input_output.shoc_cldfrac, // Input/Output
shoc_input_output.shoc_ql, // Input/Output
- shoc_output.pblh, shoc_output.shoc_ql2, // Output
+ shoc_output.pblh, shoc_output.shoc_ql2, shoc_output.tkh, // Output
shoc_history_output.shoc_mix, shoc_history_output.w_sec, shoc_history_output.thl_sec, shoc_history_output.qw_sec, shoc_history_output.qwthl_sec, // Diagnostic Output Variables
shoc_history_output.wthl_sec, shoc_history_output.wqw_sec, shoc_history_output.wtke_sec, shoc_history_output.uw_sec, shoc_history_output.vw_sec, // Diagnostic Output Variables
shoc_history_output.w3, shoc_history_output.wqls_sec, shoc_history_output.brunt, shoc_history_output.isotropy, // Diagnostic Output Variables
@@ -707,7 +709,7 @@ Int Functions::shoc_main(
shoc_temporaries.se_a, shoc_temporaries.ke_a, shoc_temporaries.wv_a, shoc_temporaries.wl_a,
shoc_temporaries.ustar, shoc_temporaries.kbfs, shoc_temporaries.obklen, shoc_temporaries.ustar2,
shoc_temporaries.wstar, shoc_temporaries.rho_zt, shoc_temporaries.shoc_qv,
- shoc_temporaries.tabs, shoc_temporaries.dz_zt, shoc_temporaries.dz_zi, shoc_temporaries.tkh);
+ shoc_temporaries.tabs, shoc_temporaries.dz_zt, shoc_temporaries.dz_zi);
#endif
auto finish = std::chrono::steady_clock::now();
diff --git a/components/eamxx/src/physics/shoc/shoc_functions.hpp b/components/eamxx/src/physics/shoc/shoc_functions.hpp
index d02d498c5a4..74602c688c2 100644
--- a/components/eamxx/src/physics/shoc/shoc_functions.hpp
+++ b/components/eamxx/src/physics/shoc/shoc_functions.hpp
@@ -161,6 +161,8 @@ struct Functions
view_1d pblh;
// cloud liquid mixing ratio variance [kg^2/kg^2]
view_2d shoc_ql2;
+ // eddy coefficient for heat [m2/s]
+ view_2d tkh;
};
// This struct stores output views for SHOC diagnostics for shoc_main.
@@ -888,6 +890,7 @@ struct Functions
// Output Variables
Scalar& pblh,
const uview_1d& shoc_ql2,
+ const uview_1d& tkh,
// Diagnostic Output Variables
const uview_1d& shoc_mix,
const uview_1d& w_sec,
@@ -959,6 +962,7 @@ struct Functions
// Output Variables
const view_1d& pblh,
const view_2d& shoc_ql2,
+ const view_2d& tkh,
// Diagnostic Output Variables
const view_2d& shoc_mix,
const view_2d& w_sec,
@@ -992,8 +996,7 @@ struct Functions
const view_2d& shoc_qv,
const view_2d& tabs,
const view_2d& dz_zt,
- const view_2d& dz_zi,
- const view_2d& tkh);
+ const view_2d& dz_zi);
#endif
// Return microseconds elapsed
diff --git a/components/eamxx/src/physics/shoc/shoc_functions_f90.cpp b/components/eamxx/src/physics/shoc/shoc_functions_f90.cpp
index 0e4acceb1a7..dc45e2456aa 100644
--- a/components/eamxx/src/physics/shoc/shoc_functions_f90.cpp
+++ b/components/eamxx/src/physics/shoc/shoc_functions_f90.cpp
@@ -2743,8 +2743,6 @@ Int shoc_main_f(Int shcol, Int nlev, Int nlevi, Real dtime, Int nadv, Int npbl,
Real* qw_sec, Real* qwthl_sec, Real* wthl_sec, Real* wqw_sec, Real* wtke_sec, Real* uw_sec, Real* vw_sec,
Real* w3, Real* wqls_sec, Real* brunt, Real* shoc_ql2)
{
- // tkh is a local variable in C++ impl
- (void)tkh;
using SHF = Functions;
@@ -2758,7 +2756,7 @@ Int shoc_main_f(Int shcol, Int nlev, Int nlevi, Real dtime, Int nadv, Int npbl,
// Initialize Kokkos views, sync to device
static constexpr Int num_1d_arrays = 7;
- static constexpr Int num_2d_arrays = 34;
+ static constexpr Int num_2d_arrays = 35;
static constexpr Int num_3d_arrays = 1;
std::vector temp_1d_d(num_1d_arrays);
@@ -2768,27 +2766,27 @@ Int shoc_main_f(Int shcol, Int nlev, Int nlevi, Real dtime, Int nadv, Int npbl,
std::vector dim1_2d_sizes = {shcol, shcol, shcol, shcol, shcol,
shcol, shcol, shcol, shcol, shcol,
shcol, shcol, shcol, shcol, shcol,
- shcol, shcol, shcol, shcol,
+ shcol, shcol, shcol, shcol, shcol,
shcol, shcol, shcol, shcol, shcol,
shcol, shcol, shcol, shcol, shcol,
shcol, shcol, shcol, shcol, shcol};
std::vector dim2_2d_sizes = {nlev, nlevi, nlev, nlevi, nlev,
nlev, nlev, num_qtracers, nlev, nlev,
nlev, nlev, nlev, nlev, nlev,
- nlev, nlev, nlev, nlev,
+ nlev, nlev, nlev, nlev, nlev,
nlev, nlev, nlev, nlevi, nlevi,
nlevi, nlevi, nlevi, nlevi, nlevi,
nlevi, nlevi, nlev, nlev, nlev};
std::vector ptr_array_1d = {host_dx, host_dy, wthl_sfc, wqw_sfc,
uw_sfc, vw_sfc, phis};
- std::vector ptr_array_2d = {zt_grid, zi_grid, pres, presi, pdel,
- thv, w_field, wtracer_sfc, inv_exner, host_dse,
- tke, thetal, qw, u_wind, v_wind,
- wthv_sec, tk, shoc_cldfrac, shoc_ql,
- shoc_ql2, shoc_mix, w_sec, thl_sec, qw_sec,
- qwthl_sec, wthl_sec, wqw_sec, wtke_sec, uw_sec,
- vw_sec, w3, wqls_sec, brunt, isotropy};
+ std::vector ptr_array_2d = {zt_grid, zi_grid, pres, presi, pdel,
+ thv, w_field, wtracer_sfc, inv_exner, host_dse,
+ tke, thetal, qw, u_wind, v_wind,
+ wthv_sec, tk, shoc_cldfrac, shoc_ql, shoc_ql2,
+ tkh, shoc_mix, w_sec, thl_sec, qw_sec,
+ qwthl_sec, wthl_sec, wqw_sec, wtke_sec, uw_sec,
+ vw_sec, w3, wqls_sec, brunt, isotropy};
ScreamDeepCopy::copy_to_device(ptr_array_1d, shcol, temp_1d_d);
ekat::host_to_device(ptr_array_2d, dim1_2d_sizes, dim2_2d_sizes, temp_2d_d, true);
@@ -2827,6 +2825,7 @@ Int shoc_main_f(Int shcol, Int nlev, Int nlevi, Real dtime, Int nadv, Int npbl,
shoc_cldfrac_d(temp_2d_d[index_counter++]),
shoc_ql_d (temp_2d_d[index_counter++]),
shoc_ql2_d (temp_2d_d[index_counter++]),
+ tkh_d (temp_2d_d[index_counter++]),
shoc_mix_d (temp_2d_d[index_counter++]),
w_sec_d (temp_2d_d[index_counter++]),
thl_sec_d (temp_2d_d[index_counter++]),
@@ -2879,7 +2878,7 @@ Int shoc_main_f(Int shcol, Int nlev, Int nlevi, Real dtime, Int nadv, Int npbl,
SHF::SHOCInputOutput shoc_input_output{host_dse_d, tke_d, thetal_d, qw_d,
horiz_wind_d, wthv_sec_d, qtracers_cxx_d,
tk_d, shoc_cldfrac_d, shoc_ql_d};
- SHF::SHOCOutput shoc_output{pblh_d, shoc_ql2_d};
+ SHF::SHOCOutput shoc_output{pblh_d, shoc_ql2_d, tkh_d};
SHF::SHOCHistoryOutput shoc_history_output{shoc_mix_d, w_sec_d, thl_sec_d, qw_sec_d,
qwthl_sec_d, wthl_sec_d, wqw_sec_d, wtke_sec_d,
uw_sec_d, vw_sec_d, w3_d, wqls_sec_d,
@@ -2909,12 +2908,11 @@ Int shoc_main_f(Int shcol, Int nlev, Int nlevi, Real dtime, Int nadv, Int npbl,
shoc_qv ("shoc_qv", shcol, nlevi_packs),
tabs ("shoc_tabs", shcol, nlev_packs),
dz_zt ("dz_zt", shcol, nlevi_packs),
- dz_zi ("dz_zi", shcol, nlevi_packs),
- tkhv ("tkh", shcol, nlevi_packs);
+ dz_zi ("dz_zi", shcol, nlevi_packs);
SHF::SHOCTemporaries shoc_temporaries{
se_b, ke_b, wv_b, wl_b, se_a, ke_a, wv_a, wl_a, ustar, kbfs, obklen, ustar2, wstar,
- rho_zt, shoc_qv, tabs, dz_zt, dz_zi, tkhv};
+ rho_zt, shoc_qv, tabs, dz_zt, dz_zi};
#endif
// Create local workspace
diff --git a/components/eamxx/tests/CMakeLists.txt b/components/eamxx/tests/CMakeLists.txt
index 7a7a9e82149..287fdf25af1 100644
--- a/components/eamxx/tests/CMakeLists.txt
+++ b/components/eamxx/tests/CMakeLists.txt
@@ -63,7 +63,7 @@ if (NOT DEFINED ENV{SCREAM_FAKE_ONLY})
set(EAMxx_tests_IC_FILE_72lev "screami_unit_tests_ne2np4L72_20220822.nc")
set(EAMxx_tests_IC_FILE_128lev "screami_unit_tests_ne2np4L128_20220822.nc")
set(EAMxx_tests_TOPO_FILE "USGS-gtopo30_ne2np4pg2_x6t_20230331.nc")
- set(EAMxx_tests_IC_FILE_MAM4xx_72lev "scream_unit_tests_aerosol_optics_ne2np4L72_20220822.nc")
+ set(EAMxx_tests_IC_FILE_MAM4xx_72lev "screami_unit_tests_mam4xx_ne2np4L72_20240329.nc")
# Testing individual atm processes
add_subdirectory(single-process)
diff --git a/components/eamxx/tests/multi-process/physics_only/CMakeLists.txt b/components/eamxx/tests/multi-process/physics_only/CMakeLists.txt
index 61b224f71ae..b9c13234796 100644
--- a/components/eamxx/tests/multi-process/physics_only/CMakeLists.txt
+++ b/components/eamxx/tests/multi-process/physics_only/CMakeLists.txt
@@ -4,6 +4,10 @@ if (SCREAM_DOUBLE_PRECISION)
add_subdirectory(shoc_cld_spa_p3_rrtmgp)
if (SCREAM_ENABLE_MAM)
add_subdirectory(mam/optics_rrtmgp)
+ add_subdirectory(mam/shoc_mam4_aci)
+ add_subdirectory(mam/shoc_cldfrac_mam4_aci_p3)
+ add_subdirectory(mam/shoc_cldfrac_mam4_aci_p3_rrtmgp)
+ add_subdirectory(mam/shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp)
endif()
endif()
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3/CMakeLists.txt b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3/CMakeLists.txt
new file mode 100644
index 00000000000..856ef3ca162
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3/CMakeLists.txt
@@ -0,0 +1,41 @@
+INCLUDE (ScreamUtils)
+
+set (TEST_BASE_NAME shoc_cldfrac_mam4_aci_p3)
+set (FIXTURES_BASE_NAME ${TEST_BASE_NAME}_generate_output_nc_files)
+
+# Create the test
+CreateADUnitTest(${TEST_BASE_NAME}
+ LIBS shoc cld_fraction p3 mam
+ LABELS shoc cld p3 physics mam4_aci
+ MPI_RANKS ${TEST_RANK_START} ${TEST_RANK_END}
+ FIXTURES_SETUP_INDIVIDUAL ${FIXTURES_BASE_NAME}
+)
+
+# Set AD configurable options
+set (ATM_TIME_STEP 1800)
+SetVarDependingOnTestSize(NUM_STEPS 2 5 48) # 1h 4h 24h
+set (RUN_T0 2021-10-12-45000)
+
+# Determine num subcycles needed to keep shoc dt<=300s
+set (SHOC_MAX_DT 300)
+math (EXPR MAC_MIC_SUBCYCLES "(${ATM_TIME_STEP} + ${SHOC_MAX_DT} - 1) / ${SHOC_MAX_DT}")
+
+# Ensure test input files are present in the data dir
+GetInputFile(scream/init/${EAMxx_tests_IC_FILE_MAM4xx_72lev})
+GetInputFile(cam/topo/${EAMxx_tests_TOPO_FILE})
+
+## Copy (and configure) yaml files needed by tests
+configure_file(${CMAKE_CURRENT_SOURCE_DIR}/input.yaml
+ ${CMAKE_CURRENT_BINARY_DIR}/input.yaml)
+configure_file(${CMAKE_CURRENT_SOURCE_DIR}/output.yaml
+ ${CMAKE_CURRENT_BINARY_DIR}/output.yaml)
+
+# Compare output files produced by npX tests, to ensure they are bfb
+include (CompareNCFiles)
+CompareNCFilesFamilyMpi (
+ TEST_BASE_NAME ${TEST_BASE_NAME}
+ FILE_META_NAME ${TEST_BASE_NAME}_output.INSTANT.nsteps_x${NUM_STEPS}.npMPIRANKS.${RUN_T0}.nc
+ MPI_RANKS ${TEST_RANK_START} ${TEST_RANK_END}
+ LABELS shoc cld p3 physics PEM mam4_aci
+ META_FIXTURES_REQUIRED ${FIXTURES_BASE_NAME}_npMPIRANKS_omp1
+)
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3/input.yaml b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3/input.yaml
new file mode 100644
index 00000000000..9c0ed4d9798
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3/input.yaml
@@ -0,0 +1,68 @@
+%YAML 1.1
+---
+driver_options:
+ atmosphere_dag_verbosity_level: 5
+
+time_stepping:
+ time_step: ${ATM_TIME_STEP}
+ run_t0: ${RUN_T0} # YYYY-MM-DD-XXXXX
+ number_of_steps: ${NUM_STEPS}
+
+atmosphere_processes:
+ atm_procs_list: [mac_mic]
+ schedule_type: Sequential
+ mac_mic:
+ atm_procs_list: [shoc,CldFraction,mam4_aci,p3]
+ Type: Group
+ schedule_type: Sequential
+ number_of_subcycles: ${MAC_MIC_SUBCYCLES}
+ p3:
+ max_total_ni: 740.0e3
+ do_prescribed_ccn: false
+ shoc:
+ lambda_low: 0.001
+ lambda_high: 0.04
+ lambda_slope: 2.65
+ lambda_thresh: 0.02
+ thl2tune: 1.0
+ qw2tune: 1.0
+ qwthl2tune: 1.0
+ w2tune: 1.0
+ length_fac: 0.5
+ c_diag_3rd_mom: 7.0
+ Ckh: 0.1
+ Ckm: 0.1
+ mam4_aci:
+ wsubmin: 0.001
+ top_level_mam4xx: 6
+
+grids_manager:
+ Type: Mesh Free
+ geo_data_source: IC_FILE
+ grids_names: [Physics GLL]
+ Physics GLL:
+ aliases: [Physics]
+ type: point_grid
+ number_of_global_columns: 218
+ number_of_vertical_levels: 72
+
+initial_conditions:
+ # The name of the file containing the initial conditions for this test.
+ Filename: ${SCREAM_DATA_DIR}/init/${EAMxx_tests_IC_FILE_MAM4xx_72lev}
+ topography_filename: ${TOPO_DATA_DIR}/${EAMxx_tests_TOPO_FILE}
+
+ #variable required for shoc
+ surf_sens_flux: 0.0
+ surf_evap: 0.0
+
+ #variable required for p3
+ precip_ice_surf_mass: 0.0
+ precip_liq_surf_mass: 0.0
+
+ #variables required for mam4_aci
+ dgnum: 1e-3
+
+# The parameters for I/O control
+Scorpio:
+ output_yaml_files: ["output.yaml"]
+...
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3/output.yaml b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3/output.yaml
new file mode 100644
index 00000000000..2541ce8d9f8
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3/output.yaml
@@ -0,0 +1,97 @@
+%YAML 1.1
+---
+filename_prefix: shoc_cldfrac_mam4_aci_p3_output
+Averaging Type: Instant
+Field Names:
+ # SHOC
+ - cldfrac_liq
+ - eddy_diff_mom
+ - horiz_winds
+ - sgs_buoy_flux
+ - tke
+ - inv_qc_relvar
+ - pbl_height
+ # CLD
+ - cldfrac_ice
+ - cldfrac_tot
+ # P3
+ - bm
+ - nc
+ - ni
+ - nr
+ - qi
+ - qm
+ - qr
+ - T_prev_micro_step
+ - qv_prev_micro_step
+ - eff_radius_qc
+ - eff_radius_qi
+ - eff_radius_qr
+ - micro_liq_ice_exchange
+ - micro_vap_ice_exchange
+ - micro_vap_liq_exchange
+ - precip_ice_surf_mass
+ - precip_liq_surf_mass
+ - rainfrac
+ # SHOC + P3
+ - qc
+ - qv
+ # SHOC + P3 + RRTMGP
+ - T_mid
+ - bc_a1
+ - bc_a3
+ - bc_a4
+ - dst_a1
+ - dst_a3
+ - so4_a1
+ - so4_a2
+ - so4_a3
+ - pom_a1
+ - pom_a3
+ - pom_a4
+ - soa_a1
+ - soa_a2
+ - soa_a3
+ - nacl_a1
+ - nacl_a2
+ - nacl_a3
+ - mom_a1
+ - mom_a2
+ - mom_a3
+ - mom_a4
+ - num_a1
+ - num_a2
+ - num_a3
+ - num_a4
+ - bc_c1
+ - bc_c3
+ - bc_c4
+ - dst_c1
+ - dst_c3
+ - so4_c1
+ - so4_c2
+ - so4_c3
+ - pom_c1
+ - pom_c3
+ - pom_c4
+ - soa_c1
+ - soa_c2
+ - soa_c3
+ - nacl_c1
+ - nacl_c2
+ - nacl_c3
+ - mom_c1
+ - mom_c2
+ - mom_c3
+ - mom_c4
+ - num_c1
+ - num_c2
+ - num_c3
+ - num_c4
+ - eddy_diff_heat
+ - nc_nuceat_tend
+ - ni_activated
+output_control:
+ Frequency: ${NUM_STEPS}
+ frequency_units: nsteps
+...
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp/CMakeLists.txt b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp/CMakeLists.txt
new file mode 100644
index 00000000000..3f9aeae73fd
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp/CMakeLists.txt
@@ -0,0 +1,41 @@
+INCLUDE (ScreamUtils)
+
+set (TEST_BASE_NAME shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp)
+set (FIXTURES_BASE_NAME ${TEST_BASE_NAME}_generate_output_nc_files)
+
+# Create the test
+CreateADUnitTest(${TEST_BASE_NAME}
+ LIBS shoc cld_fraction p3 scream_rrtmgp mam
+ LABELS shoc cld p3 rrtmgp physics mam4_aci
+ MPI_RANKS ${TEST_RANK_START} ${TEST_RANK_END}
+ FIXTURES_SETUP_INDIVIDUAL ${FIXTURES_BASE_NAME}
+)
+
+# Set AD configurable options
+set (ATM_TIME_STEP 1800)
+SetVarDependingOnTestSize(NUM_STEPS 2 5 48) # 1h 4h 24h
+set (RUN_T0 2021-10-12-45000)
+
+# Determine num subcycles needed to keep shoc dt<=300s
+set (SHOC_MAX_DT 300)
+math (EXPR MAC_MIC_SUBCYCLES "(${ATM_TIME_STEP} + ${SHOC_MAX_DT} - 1) / ${SHOC_MAX_DT}")
+
+# Ensure test input files are present in the data dir
+GetInputFile(scream/init/${EAMxx_tests_IC_FILE_MAM4xx_72lev})
+GetInputFile(cam/topo/${EAMxx_tests_TOPO_FILE})
+
+## Copy (and configure) yaml files needed by tests
+configure_file(${CMAKE_CURRENT_SOURCE_DIR}/input.yaml
+ ${CMAKE_CURRENT_BINARY_DIR}/input.yaml)
+configure_file(${CMAKE_CURRENT_SOURCE_DIR}/output.yaml
+ ${CMAKE_CURRENT_BINARY_DIR}/output.yaml)
+
+# Compare output files produced by npX tests, to ensure they are bfb
+include (CompareNCFiles)
+CompareNCFilesFamilyMpi (
+ TEST_BASE_NAME ${TEST_BASE_NAME}
+ FILE_META_NAME ${TEST_BASE_NAME}_output.INSTANT.nsteps_x${NUM_STEPS}.npMPIRANKS.${RUN_T0}.nc
+ MPI_RANKS ${TEST_RANK_START} ${TEST_RANK_END}
+ LABELS shoc cld p3 rrtmgp physics PEM mam4_aci
+ META_FIXTURES_REQUIRED ${FIXTURES_BASE_NAME}_npMPIRANKS_omp1
+)
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp/input.yaml b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp/input.yaml
new file mode 100644
index 00000000000..20d6f142c52
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp/input.yaml
@@ -0,0 +1,90 @@
+%YAML 1.1
+---
+driver_options:
+ atmosphere_dag_verbosity_level: 5
+
+time_stepping:
+ time_step: ${ATM_TIME_STEP}
+ run_t0: ${RUN_T0} # YYYY-MM-DD-XXXXX
+ number_of_steps: ${NUM_STEPS}
+
+atmosphere_processes:
+ atm_procs_list: [mac_mic, mam4_optics, rrtmgp]
+ schedule_type: Sequential
+ mac_mic:
+ atm_procs_list: [shoc,CldFraction,mam4_aci,p3]
+ Type: Group
+ schedule_type: Sequential
+ number_of_subcycles: ${MAC_MIC_SUBCYCLES}
+ mam4_aci:
+ wsubmin: 0.001
+ top_level_mam4xx: 6
+ p3:
+ max_total_ni: 740.0e3
+ do_prescribed_ccn: false
+ shoc:
+ lambda_low: 0.001
+ lambda_high: 0.04
+ lambda_slope: 2.65
+ lambda_thresh: 0.02
+ thl2tune: 1.0
+ qw2tune: 1.0
+ qwthl2tune: 1.0
+ w2tune: 1.0
+ length_fac: 0.5
+ c_diag_3rd_mom: 7.0
+ Ckh: 0.1
+ Ckm: 0.1
+ mam4_optics:
+ mam4_mode1_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/mam4_mode1_rrtmg_aeronetdust_c20240206.nc
+ mam4_mode2_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/mam4_mode2_rrtmg_c20240206.nc
+ mam4_mode3_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/mam4_mode3_rrtmg_aeronetdust_c20240206.nc
+ mam4_mode4_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/mam4_mode4_rrtmg_c20240206.nc
+ mam4_water_refindex_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/water_refindex_rrtmg_c20240206.nc
+ mam4_soa_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/ocphi_rrtmg_c20240206.nc
+ mam4_dust_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/dust_aeronet_rrtmg_c20240206.nc
+ mam4_nacl_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/ssam_rrtmg_c20240206.nc
+ mam4_so4_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/sulfate_rrtmg_c20240206.nc
+ mam4_pom_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/ocpho_rrtmg_c20240206.nc
+ mam4_bc_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/bcpho_rrtmg_c20240206.nc
+ mam4_mom_physical_properties_file : ${SCREAM_DATA_DIR}/mam4xx/physprops/poly_rrtmg_c20240206.nc
+ rrtmgp:
+ column_chunk_size: 123
+ active_gases: ["h2o", "co2", "o3", "n2o", "co" , "ch4", "o2", "n2"]
+ orbital_year: 1990
+ do_aerosol_rad: false
+ rrtmgp_coefficients_file_sw: ${SCREAM_DATA_DIR}/init/rrtmgp-data-sw-g112-210809.nc
+ rrtmgp_coefficients_file_lw: ${SCREAM_DATA_DIR}/init/rrtmgp-data-lw-g128-210809.nc
+ rrtmgp_cloud_optics_file_sw: ${SCREAM_DATA_DIR}/init/rrtmgp-cloud-optics-coeffs-sw.nc
+ rrtmgp_cloud_optics_file_lw: ${SCREAM_DATA_DIR}/init/rrtmgp-cloud-optics-coeffs-lw.nc
+
+grids_manager:
+ Type: Mesh Free
+ geo_data_source: IC_FILE
+ grids_names: [Physics GLL]
+ Physics GLL:
+ aliases: [Physics]
+ type: point_grid
+ number_of_global_columns: 218
+ number_of_vertical_levels: 72
+
+initial_conditions:
+ # The name of the file containing the initial conditions for this test.
+ Filename: ${SCREAM_DATA_DIR}/init/${EAMxx_tests_IC_FILE_MAM4xx_72lev}
+ topography_filename: ${TOPO_DATA_DIR}/${EAMxx_tests_TOPO_FILE}
+
+ #variables required for shoc
+ surf_sens_flux: 0.0
+ surf_evap: 0.0
+
+ #variables required for p3
+ precip_ice_surf_mass: 0.0
+ precip_liq_surf_mass: 0.0
+
+ #variables required for mam4_aci
+ dgnum: 1e-3
+
+# The parameters for I/O control
+Scorpio:
+ output_yaml_files: ["output.yaml"]
+...
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp/output.yaml b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp/output.yaml
new file mode 100644
index 00000000000..392e159f0a5
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp/output.yaml
@@ -0,0 +1,110 @@
+%YAML 1.1
+---
+filename_prefix: shoc_cldfrac_mam4_aci_p3_mam4_optics_rrtmgp_output
+Averaging Type: Instant
+Field Names:
+ # SHOC
+ - cldfrac_liq
+ - eddy_diff_mom
+ - horiz_winds
+ - sgs_buoy_flux
+ - tke
+ - inv_qc_relvar
+ - pbl_height
+ # CLD
+ - cldfrac_ice
+ - cldfrac_tot
+ # P3
+ - bm
+ - nc
+ - ni
+ - nr
+ - qi
+ - qm
+ - qr
+ - T_prev_micro_step
+ - qv_prev_micro_step
+ - eff_radius_qc
+ - eff_radius_qi
+ - eff_radius_qr
+ - micro_liq_ice_exchange
+ - micro_vap_ice_exchange
+ - micro_vap_liq_exchange
+ - precip_ice_surf_mass
+ - precip_liq_surf_mass
+ - rainfrac
+ # SHOC + P3
+ - qc
+ - qv
+ # SHOC + P3 + RRTMGP
+ - T_mid
+ # RRTMGP
+ - sfc_alb_dif_nir
+ - sfc_alb_dif_vis
+ - sfc_alb_dir_nir
+ - sfc_alb_dir_vis
+ - LW_flux_dn
+ - LW_flux_up
+ - SW_flux_dn
+ - SW_flux_dn_dir
+ - SW_flux_up
+ - rad_heating_pdel
+ - sfc_flux_lw_dn
+ - sfc_flux_sw_net
+ - bc_a1
+ - bc_a3
+ - bc_a4
+ - dst_a1
+ - dst_a3
+ - so4_a1
+ - so4_a2
+ - so4_a3
+ - pom_a1
+ - pom_a3
+ - pom_a4
+ - soa_a1
+ - soa_a2
+ - soa_a3
+ - nacl_a1
+ - nacl_a2
+ - nacl_a3
+ - mom_a1
+ - mom_a2
+ - mom_a3
+ - mom_a4
+ - num_a1
+ - num_a2
+ - num_a3
+ - num_a4
+ - bc_c1
+ - bc_c3
+ - bc_c4
+ - dst_c1
+ - dst_c3
+ - so4_c1
+ - so4_c2
+ - so4_c3
+ - pom_c1
+ - pom_c3
+ - pom_c4
+ - soa_c1
+ - soa_c2
+ - soa_c3
+ - nacl_c1
+ - nacl_c2
+ - nacl_c3
+ - mom_c1
+ - mom_c2
+ - mom_c3
+ - mom_c4
+ - num_c1
+ - num_c2
+ - num_c3
+ - num_c4
+ - eddy_diff_heat
+ - nc_nuceat_tend
+ - ni_activated
+output_control:
+ Frequency: ${NUM_STEPS}
+ frequency_units: nsteps
+...
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_rrtmgp/CMakeLists.txt b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_rrtmgp/CMakeLists.txt
new file mode 100644
index 00000000000..cdd564c73a8
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_rrtmgp/CMakeLists.txt
@@ -0,0 +1,41 @@
+INCLUDE (ScreamUtils)
+
+set (TEST_BASE_NAME shoc_cldfrac_mam4_aci_p3_rrtmgp)
+set (FIXTURES_BASE_NAME ${TEST_BASE_NAME}_generate_output_nc_files)
+
+# Create the test
+CreateADUnitTest(${TEST_BASE_NAME}
+ LIBS shoc cld_fraction p3 scream_rrtmgp mam
+ LABELS shoc cld p3 rrtmgp physics mam4_aci
+ MPI_RANKS ${TEST_RANK_START} ${TEST_RANK_END}
+ FIXTURES_SETUP_INDIVIDUAL ${FIXTURES_BASE_NAME}
+)
+
+# Set AD configurable options
+set (ATM_TIME_STEP 1800)
+SetVarDependingOnTestSize(NUM_STEPS 2 5 48) # 1h 4h 24h
+set (RUN_T0 2021-10-12-45000)
+
+# Determine num subcycles needed to keep shoc dt<=300s
+set (SHOC_MAX_DT 300)
+math (EXPR MAC_MIC_SUBCYCLES "(${ATM_TIME_STEP} + ${SHOC_MAX_DT} - 1) / ${SHOC_MAX_DT}")
+
+# Ensure test input files are present in the data dir
+GetInputFile(scream/init/${EAMxx_tests_IC_FILE_MAM4xx_72lev})
+GetInputFile(cam/topo/${EAMxx_tests_TOPO_FILE})
+
+## Copy (and configure) yaml files needed by tests
+configure_file(${CMAKE_CURRENT_SOURCE_DIR}/input.yaml
+ ${CMAKE_CURRENT_BINARY_DIR}/input.yaml)
+configure_file(${CMAKE_CURRENT_SOURCE_DIR}/output.yaml
+ ${CMAKE_CURRENT_BINARY_DIR}/output.yaml)
+
+# Compare output files produced by npX tests, to ensure they are bfb
+include (CompareNCFiles)
+CompareNCFilesFamilyMpi (
+ TEST_BASE_NAME ${TEST_BASE_NAME}
+ FILE_META_NAME ${TEST_BASE_NAME}_output.INSTANT.nsteps_x${NUM_STEPS}.npMPIRANKS.${RUN_T0}.nc
+ MPI_RANKS ${TEST_RANK_START} ${TEST_RANK_END}
+ LABELS shoc cld p3 rrtmgp physics PEM mam4_aci
+ META_FIXTURES_REQUIRED ${FIXTURES_BASE_NAME}_npMPIRANKS_omp1
+)
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_rrtmgp/input.yaml b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_rrtmgp/input.yaml
new file mode 100644
index 00000000000..7ec317eda92
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_rrtmgp/input.yaml
@@ -0,0 +1,83 @@
+%YAML 1.1
+---
+driver_options:
+ atmosphere_dag_verbosity_level: 5
+
+time_stepping:
+ time_step: ${ATM_TIME_STEP}
+ run_t0: ${RUN_T0} # YYYY-MM-DD-XXXXX
+ number_of_steps: ${NUM_STEPS}
+
+atmosphere_processes:
+ atm_procs_list: [mac_mic,rrtmgp]
+ schedule_type: Sequential
+ mac_mic:
+ atm_procs_list: [shoc,CldFraction,mam4_aci,p3]
+ Type: Group
+ schedule_type: Sequential
+ number_of_subcycles: ${MAC_MIC_SUBCYCLES}
+ mam4_aci:
+ wsubmin: 0.001
+ top_level_mam4xx: 6
+ p3:
+ max_total_ni: 740.0e3
+ do_prescribed_ccn: false
+ shoc:
+ lambda_low: 0.001
+ lambda_high: 0.04
+ lambda_slope: 2.65
+ lambda_thresh: 0.02
+ thl2tune: 1.0
+ qw2tune: 1.0
+ qwthl2tune: 1.0
+ w2tune: 1.0
+ length_fac: 0.5
+ c_diag_3rd_mom: 7.0
+ Ckh: 0.1
+ Ckm: 0.1
+ rrtmgp:
+ column_chunk_size: 123
+ active_gases: ["h2o", "co2", "o3", "n2o", "co" , "ch4", "o2", "n2"]
+ orbital_year: 1990
+ do_aerosol_rad: false
+ rrtmgp_coefficients_file_sw: ${SCREAM_DATA_DIR}/init/rrtmgp-data-sw-g112-210809.nc
+ rrtmgp_coefficients_file_lw: ${SCREAM_DATA_DIR}/init/rrtmgp-data-lw-g128-210809.nc
+ rrtmgp_cloud_optics_file_sw: ${SCREAM_DATA_DIR}/init/rrtmgp-cloud-optics-coeffs-sw.nc
+ rrtmgp_cloud_optics_file_lw: ${SCREAM_DATA_DIR}/init/rrtmgp-cloud-optics-coeffs-lw.nc
+
+grids_manager:
+ Type: Mesh Free
+ geo_data_source: IC_FILE
+ grids_names: [Physics GLL]
+ Physics GLL:
+ aliases: [Physics]
+ type: point_grid
+ number_of_global_columns: 218
+ number_of_vertical_levels: 72
+
+initial_conditions:
+ # The name of the file containing the initial conditions for this test.
+ Filename: ${SCREAM_DATA_DIR}/init/${EAMxx_tests_IC_FILE_MAM4xx_72lev}
+ topography_filename: ${TOPO_DATA_DIR}/${EAMxx_tests_TOPO_FILE}
+
+ #variables required for shoc
+ surf_sens_flux: 0.0
+ surf_evap: 0.0
+
+ #variables required for p3
+ precip_ice_surf_mass: 0.0
+ precip_liq_surf_mass: 0.0
+
+ #variables required for rrtmgp
+ aero_g_sw: 0.0
+ aero_ssa_sw: 0.0
+ aero_tau_sw: 0.0
+ aero_tau_lw: 0.0
+
+ #variables required for mam4_aci
+ dgnum: 1e-3
+
+# The parameters for I/O control
+Scorpio:
+ output_yaml_files: ["output.yaml"]
+...
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_rrtmgp/output.yaml b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_rrtmgp/output.yaml
new file mode 100644
index 00000000000..4c9f5ff5d8d
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_cldfrac_mam4_aci_p3_rrtmgp/output.yaml
@@ -0,0 +1,110 @@
+%YAML 1.1
+---
+filename_prefix: shoc_cldfrac_mam4_aci_p3_rrtmgp_output
+Averaging Type: Instant
+Field Names:
+ # SHOC
+ - cldfrac_liq
+ - eddy_diff_mom
+ - horiz_winds
+ - sgs_buoy_flux
+ - tke
+ - inv_qc_relvar
+ - pbl_height
+ # CLD
+ - cldfrac_ice
+ - cldfrac_tot
+ # P3
+ - bm
+ - nc
+ - ni
+ - nr
+ - qi
+ - qm
+ - qr
+ - T_prev_micro_step
+ - qv_prev_micro_step
+ - eff_radius_qc
+ - eff_radius_qi
+ - eff_radius_qr
+ - micro_liq_ice_exchange
+ - micro_vap_ice_exchange
+ - micro_vap_liq_exchange
+ - precip_ice_surf_mass
+ - precip_liq_surf_mass
+ - rainfrac
+ # SHOC + P3
+ - qc
+ - qv
+ # SHOC + P3 + RRTMGP
+ - T_mid
+ # RRTMGP
+ - sfc_alb_dif_nir
+ - sfc_alb_dif_vis
+ - sfc_alb_dir_nir
+ - sfc_alb_dir_vis
+ - LW_flux_dn
+ - LW_flux_up
+ - SW_flux_dn
+ - SW_flux_dn_dir
+ - SW_flux_up
+ - rad_heating_pdel
+ - sfc_flux_lw_dn
+ - sfc_flux_sw_net
+ - bc_a1
+ - bc_a3
+ - bc_a4
+ - dst_a1
+ - dst_a3
+ - so4_a1
+ - so4_a2
+ - so4_a3
+ - pom_a1
+ - pom_a3
+ - pom_a4
+ - soa_a1
+ - soa_a2
+ - soa_a3
+ - nacl_a1
+ - nacl_a2
+ - nacl_a3
+ - mom_a1
+ - mom_a2
+ - mom_a3
+ - mom_a4
+ - num_a1
+ - num_a2
+ - num_a3
+ - num_a4
+ - bc_c1
+ - bc_c3
+ - bc_c4
+ - dst_c1
+ - dst_c3
+ - so4_c1
+ - so4_c2
+ - so4_c3
+ - pom_c1
+ - pom_c3
+ - pom_c4
+ - soa_c1
+ - soa_c2
+ - soa_c3
+ - nacl_c1
+ - nacl_c2
+ - nacl_c3
+ - mom_c1
+ - mom_c2
+ - mom_c3
+ - mom_c4
+ - num_c1
+ - num_c2
+ - num_c3
+ - num_c4
+ - eddy_diff_heat
+ - nc_nuceat_tend
+ - ni_activated
+output_control:
+ Frequency: ${NUM_STEPS}
+ frequency_units: nsteps
+...
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_mam4_aci/CMakeLists.txt b/components/eamxx/tests/multi-process/physics_only/mam/shoc_mam4_aci/CMakeLists.txt
new file mode 100644
index 00000000000..69a154223fd
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_mam4_aci/CMakeLists.txt
@@ -0,0 +1,41 @@
+INCLUDE (ScreamUtils)
+
+set (TEST_BASE_NAME shoc_mam4_aci)
+set (FIXTURES_BASE_NAME ${TEST_BASE_NAME}_generate_output_nc_files)
+
+# Create the test
+CreateADUnitTest(${TEST_BASE_NAME}
+ LIBS shoc mam
+ LABELS shoc physics mam4_aci
+ MPI_RANKS ${TEST_RANK_START} ${TEST_RANK_END}
+ FIXTURES_SETUP_INDIVIDUAL ${FIXTURES_BASE_NAME}
+)
+
+# Set AD configurable options
+set (ATM_TIME_STEP 1800)
+SetVarDependingOnTestSize(NUM_STEPS 2 5 48) # 1h 4h 24h
+set (RUN_T0 2021-10-12-45000)
+
+# Determine num subcycles needed to keep shoc dt<=300s
+set (SHOC_MAX_DT 300)
+math (EXPR MAC_MIC_SUBCYCLES "(${ATM_TIME_STEP} + ${SHOC_MAX_DT} - 1) / ${SHOC_MAX_DT}")
+
+# Ensure test input files are present in the data dir
+GetInputFile(scream/init/${EAMxx_tests_IC_FILE_MAM4xx_72lev})
+GetInputFile(cam/topo/${EAMxx_tests_TOPO_FILE})
+
+## Copy (and configure) yaml files needed by tests
+configure_file(${CMAKE_CURRENT_SOURCE_DIR}/input.yaml
+ ${CMAKE_CURRENT_BINARY_DIR}/input.yaml)
+configure_file(${CMAKE_CURRENT_SOURCE_DIR}/output.yaml
+ ${CMAKE_CURRENT_BINARY_DIR}/output.yaml)
+
+# Compare output files produced by npX tests, to ensure they are bfb
+include (CompareNCFiles)
+CompareNCFilesFamilyMpi (
+ TEST_BASE_NAME ${TEST_BASE_NAME}
+ FILE_META_NAME ${TEST_BASE_NAME}_output.INSTANT.nsteps_x${NUM_STEPS}.npMPIRANKS.${RUN_T0}.nc
+ MPI_RANKS ${TEST_RANK_START} ${TEST_RANK_END}
+ LABELS shoc physics mam4_aci
+ META_FIXTURES_REQUIRED ${FIXTURES_BASE_NAME}_npMPIRANKS_omp1
+)
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_mam4_aci/input.yaml b/components/eamxx/tests/multi-process/physics_only/mam/shoc_mam4_aci/input.yaml
new file mode 100644
index 00000000000..ae997bb3da5
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_mam4_aci/input.yaml
@@ -0,0 +1,61 @@
+%YAML 1.1
+---
+driver_options:
+ atmosphere_dag_verbosity_level: 5
+
+time_stepping:
+ time_step: ${ATM_TIME_STEP}
+ run_t0: ${RUN_T0} # YYYY-MM-DD-XXXXX
+ number_of_steps: ${NUM_STEPS}
+
+atmosphere_processes:
+ atm_procs_list: [mac_mic]
+ schedule_type: Sequential
+ mac_mic:
+ atm_procs_list: [shoc,mam4_aci]
+ Type: Group
+ schedule_type: Sequential
+ number_of_subcycles: ${MAC_MIC_SUBCYCLES}
+ mam4_aci:
+ wsubmin: 0.001
+ top_level_mam4xx: 6
+ shoc:
+ lambda_low: 0.001
+ lambda_high: 0.04
+ lambda_slope: 2.65
+ lambda_thresh: 0.02
+ thl2tune: 1.0
+ qw2tune: 1.0
+ qwthl2tune: 1.0
+ w2tune: 1.0
+ length_fac: 0.5
+ c_diag_3rd_mom: 7.0
+ Ckh: 0.1
+ Ckm: 0.1
+
+grids_manager:
+ Type: Mesh Free
+ geo_data_source: IC_FILE
+ grids_names: [Physics GLL]
+ Physics GLL:
+ aliases: [Physics]
+ type: point_grid
+ number_of_global_columns: 218
+ number_of_vertical_levels: 72
+
+initial_conditions:
+ # The name of the file containing the initial conditions for this test.
+ Filename: ${SCREAM_DATA_DIR}/init/${EAMxx_tests_IC_FILE_MAM4xx_72lev}
+ topography_filename: ${TOPO_DATA_DIR}/${EAMxx_tests_TOPO_FILE}
+
+ #variables needed for SHOC
+ surf_sens_flux: 0.0
+ surf_evap: 0.0
+
+ #variables needed for mam4-aci
+ dgnum: 1e-3
+
+# The parameters for I/O control
+Scorpio:
+ output_yaml_files: ["output.yaml"]
+...
diff --git a/components/eamxx/tests/multi-process/physics_only/mam/shoc_mam4_aci/output.yaml b/components/eamxx/tests/multi-process/physics_only/mam/shoc_mam4_aci/output.yaml
new file mode 100644
index 00000000000..34626c99643
--- /dev/null
+++ b/components/eamxx/tests/multi-process/physics_only/mam/shoc_mam4_aci/output.yaml
@@ -0,0 +1,70 @@
+%YAML 1.1
+---
+filename_prefix: shoc_mam4_aci_output
+Averaging Type: Instant
+Field Names:
+ # SHOC
+ - cldfrac_liq
+ - eddy_diff_mom
+ - horiz_winds
+ - sgs_buoy_flux
+ - tke
+ - inv_qc_relvar
+ - pbl_height
+ - bc_a1
+ - bc_a3
+ - bc_a4
+ - dst_a1
+ - dst_a3
+ - so4_a1
+ - so4_a2
+ - so4_a3
+ - pom_a1
+ - pom_a3
+ - pom_a4
+ - soa_a1
+ - soa_a2
+ - soa_a3
+ - nacl_a1
+ - nacl_a2
+ - nacl_a3
+ - mom_a1
+ - mom_a2
+ - mom_a3
+ - mom_a4
+ - num_a1
+ - num_a2
+ - num_a3
+ - num_a4
+ - bc_c1
+ - bc_c3
+ - bc_c4
+ - dst_c1
+ - dst_c3
+ - so4_c1
+ - so4_c2
+ - so4_c3
+ - pom_c1
+ - pom_c3
+ - pom_c4
+ - soa_c1
+ - soa_c2
+ - soa_c3
+ - nacl_c1
+ - nacl_c2
+ - nacl_c3
+ - mom_c1
+ - mom_c2
+ - mom_c3
+ - mom_c4
+ - num_c1
+ - num_c2
+ - num_c3
+ - num_c4
+ - eddy_diff_heat
+ - nc_nuceat_tend
+ - ni_activated
+output_control:
+ Frequency: ${NUM_STEPS}
+ frequency_units: nsteps
+...
diff --git a/components/eamxx/tests/single-process/CMakeLists.txt b/components/eamxx/tests/single-process/CMakeLists.txt
index d417a9920f4..c6b4e0748af 100644
--- a/components/eamxx/tests/single-process/CMakeLists.txt
+++ b/components/eamxx/tests/single-process/CMakeLists.txt
@@ -20,6 +20,7 @@ if (SCREAM_ENABLE_MAM)
# corresponding test here needs to be reworked with valid aerosol
# initial conditions.
add_subdirectory(mam/optics)
+ add_subdirectory(mam/aci)
endif()
if (SCREAM_TEST_LEVEL GREATER_EQUAL SCREAM_TEST_LEVEL_EXPERIMENTAL)
add_subdirectory(zm)
diff --git a/components/eamxx/tests/single-process/mam/aci/CMakeLists.txt b/components/eamxx/tests/single-process/mam/aci/CMakeLists.txt
new file mode 100644
index 00000000000..0d5d60ee7d5
--- /dev/null
+++ b/components/eamxx/tests/single-process/mam/aci/CMakeLists.txt
@@ -0,0 +1,44 @@
+include (ScreamUtils)
+
+set (TEST_BASE_NAME mam4_aci_standalone)
+set (FIXTURES_BASE_NAME ${TEST_BASE_NAME}_generate_output_nc_files)
+
+# Create the test
+CreateADUnitTest(${TEST_BASE_NAME}
+ LABELS mam4_aci physics
+ LIBS mam
+ MPI_RANKS ${TEST_RANK_START} ${TEST_RANK_END}
+ FIXTURES_SETUP_INDIVIDUAL ${FIXTURES_BASE_NAME}
+)
+
+# Set AD configurable options
+set (ATM_TIME_STEP 1800)
+SetVarDependingOnTestSize(NUM_STEPS 2 5 48) # 1h 2.5h 24h
+set (RUN_T0 2021-10-12-45000)
+
+## Copy (and configure) yaml files needed by tests
+configure_file(${CMAKE_CURRENT_SOURCE_DIR}/input.yaml
+ ${CMAKE_CURRENT_BINARY_DIR}/input.yaml)
+configure_file(${CMAKE_CURRENT_SOURCE_DIR}/output.yaml
+ ${CMAKE_CURRENT_BINARY_DIR}/output.yaml)
+
+# Ensure test input files are present in the data dir
+GetInputFile(scream/init/${EAMxx_tests_IC_FILE_MAM4xx_72lev})
+
+# Compare output files produced by npX tests, to ensure they are bfb
+include (CompareNCFiles)
+
+CompareNCFilesFamilyMpi (
+ TEST_BASE_NAME ${TEST_BASE_NAME}
+ FILE_META_NAME ${TEST_BASE_NAME}_output.INSTANT.nsteps_x1.npMPIRANKS.${RUN_T0}.nc
+ MPI_RANKS ${TEST_RANK_START} ${TEST_RANK_END}
+ LABELS mam4_aci physics
+ META_FIXTURES_REQUIRED ${FIXTURES_BASE_NAME}_npMPIRANKS_omp1
+)
+
+if (SCREAM_ENABLE_BASELINE_TESTS)
+ # Compare one of the output files with the baselines.
+ # Note: one is enough, since we already check that np1 is BFB with npX
+ set (OUT_FILE ${TEST_BASE_NAME}_output.INSTANT.nsteps_x1.np${TEST_RANK_END}.${RUN_T0}.nc)
+ CreateBaselineTest(${TEST_BASE_NAME} ${TEST_RANK_END} ${OUT_FILE} ${FIXTURES_BASE_NAME})
+endif()
diff --git a/components/eamxx/tests/single-process/mam/aci/input.yaml b/components/eamxx/tests/single-process/mam/aci/input.yaml
new file mode 100644
index 00000000000..fe26d828d41
--- /dev/null
+++ b/components/eamxx/tests/single-process/mam/aci/input.yaml
@@ -0,0 +1,44 @@
+%YAML 1.1
+---
+driver_options:
+ atmosphere_dag_verbosity_level: 5
+
+time_stepping:
+ time_step: ${ATM_TIME_STEP}
+ run_t0: ${RUN_T0} # YYYY-MM-DD-XXXXX
+ number_of_steps: ${NUM_STEPS}
+
+atmosphere_processes:
+ atm_procs_list: [mam4_aci]
+ mam4_aci:
+ wsubmin: 0.001
+ top_level_mam4xx: 6
+
+grids_manager:
+ Type: Mesh Free
+ geo_data_source: IC_FILE
+ grids_names: [Physics GLL]
+ Physics GLL:
+ type: point_grid
+ aliases: [Physics]
+ number_of_global_columns: 218
+ number_of_vertical_levels: 72
+
+initial_conditions:
+ # The name of the file containing the initial conditions for this test.
+ Filename: ${SCREAM_DATA_DIR}/init/${EAMxx_tests_IC_FILE_MAM4xx_72lev}
+ topography_filename: ${TOPO_DATA_DIR}/${EAMxx_tests_TOPO_FILE}
+ phis : 1.0
+ #These should come from the input file
+
+ #we should get the following variables from other processes
+ pbl_height : 1.0
+ dgnum: 4.362354500358337E-008
+ cldfrac_liq_prev: 0.138584624960092
+ eddy_diff_heat: 0.620820118789603
+ w_variance: 1.110977269289875E-002
+
+# The parameters for I/O control
+Scorpio:
+ output_yaml_files: ["output.yaml"]
+...
diff --git a/components/eamxx/tests/single-process/mam/aci/output.yaml b/components/eamxx/tests/single-process/mam/aci/output.yaml
new file mode 100644
index 00000000000..18f64b077cf
--- /dev/null
+++ b/components/eamxx/tests/single-process/mam/aci/output.yaml
@@ -0,0 +1,65 @@
+%YAML 1.1
+---
+filename_prefix: mam4_aci_standalone_output
+Averaging Type: Instant
+Fields:
+ Physics:
+ Field Names:
+ - bc_a1
+ - bc_a3
+ - bc_a4
+ - dst_a1
+ - dst_a3
+ - so4_a1
+ - so4_a2
+ - so4_a3
+ - pom_a1
+ - pom_a3
+ - pom_a4
+ - soa_a1
+ - soa_a2
+ - soa_a3
+ - nacl_a1
+ - nacl_a2
+ - nacl_a3
+ - mom_a1
+ - mom_a2
+ - mom_a3
+ - mom_a4
+ - num_a1
+ - num_a2
+ - num_a3
+ - num_a4
+ - bc_c1
+ - bc_c3
+ - bc_c4
+ - dst_c1
+ - dst_c3
+ - so4_c1
+ - so4_c2
+ - so4_c3
+ - pom_c1
+ - pom_c3
+ - pom_c4
+ - soa_c1
+ - soa_c2
+ - soa_c3
+ - nacl_c1
+ - nacl_c2
+ - nacl_c3
+ - mom_c1
+ - mom_c2
+ - mom_c3
+ - mom_c4
+ - num_c1
+ - num_c2
+ - num_c3
+ - num_c4
+ - eddy_diff_heat
+ - nc_nuceat_tend
+ - ni_activated
+
+output_control:
+ Frequency: 1
+ frequency_units: nsteps
+...
diff --git a/externals/mam4xx b/externals/mam4xx
index 7b0be56f4c3..a8aa56ba915 160000
--- a/externals/mam4xx
+++ b/externals/mam4xx
@@ -1 +1 @@
-Subproject commit 7b0be56f4c3f164259c6a9d8a0d16d6f5b72f3ae
+Subproject commit a8aa56ba915c83c22edc54ce5fd8ae4a23e6173e