clm5.0/bld/CLMBuildNamelist.pm
2025-01-12 20:48:10 +08:00

4670 lines
205 KiB
Perl
Executable File

# build-namelist
#
# This script builds the namelists for CLM
#
# The simplest use of build-namelist is to execute it from the build directory where configure
# was run. By default it will use the config_cache.xml file that was written by configure to
# determine the build time properties of the executable, and will write the files that contain
# the output namelists in that same directory. But if multiple runs are to made using the
# same executable, successive invocations of build-namelist will overwrite previously generated
# namelist files. So generally the best strategy is to invoke build-namelist from the run
# directory and use the -config option to provide the filepath of the config_cache.xml file.
#
#
# Date Contributor Modification
# -------------------------------------------------------------------------------------------
# 2009-01-20 Vertenstein Original version
# 2010-04-27 Kluzek Add ndep streams capability
# 2011-07-25 Kluzek Add multiple ensemble's of namelists
# 2012-03-23 Kluzek Add megan namelist and do checking on it
# 2012-07-01 Kluzek Add some common CESM namelist options
# 2013-12 Andre Refactor everything into subroutines
# 2013-12 Muszala Add Ecosystem Demography functionality
#--------------------------------------------------------------------------------------------
package CLMBuildNamelist;
require 5;
use strict;
#use warnings;
#use diagnostics;
use Cwd qw(getcwd abs_path);
use File::Basename qw(dirname);
use English;
use Getopt::Long;
use IO::File;
use File::Glob ':glob';
#-------------------------------------------------------------------------------
#
# Define a small number of global variables
#
#-------------------------------------------------------------------------------
(my $ProgName = $0) =~ s!(.*)/!!; # name of this script
my $ProgDir = $1;
$ProgName = "CLM " . "$ProgName";
my $cwd = abs_path(getcwd()); # absolute path of the current working directory
my $log; # Log messages object -- will be set in main, declaring it global here means it can be used everywhere
#-------------------------------------------------------------------------------
sub usage {
die <<EOF;
SYNOPSIS
build-namelist [options]
Create the namelist for CLM
REQUIRED OPTIONS
-cimeroot "directory" Path to cime directory
-config "filepath" Read the given CLM configuration cache file.
Default: "config_cache.xml".
-d "directory" Directory where output namelist file will be written
Default: current working directory.
-envxml_dir "directory" Directory name of env_*.xml case files to read in.
(if read they allow user_nl_clm and CLM_BLDNML_OPTS to expand
variables [for example to use \$DIN_LOC_ROOT])
(default current directory)
-lnd_frac "domainfile" Land fraction file (the input domain file)
-res "resolution" Specify horizontal grid. Use nlatxnlon for spectral grids;
dlatxdlon for fv grids (dlat and dlon are the grid cell size
in degrees for latitude and longitude respectively)
"-res list" to list valid resolutions.
(default: 0.9x1.25)
-sim_year "year" Year to simulate for input datasets
(i.e. PtVg, 1850, 2000, 2010, 1850-2000, 1850-2100)
"-sim_year list" to list valid simulation years
(default 2000)
OPTIONS
-bgc "value" Build CLM with BGC package [ sp | cn | bgc | fates ]
(default is sp).
CLM Biogeochemistry mode
sp = Satellite Phenology (SP)
This toggles off the namelist variable: use_cn
cn = Carbon Nitrogen model (CN)
(or CLM45CN if phys=clm4_5/clm5_0)
This toggles on the namelist variable: use_cn
bgc = Carbon Nitrogen with methane, nitrification, vertical soil C,
CENTURY decomposition
(or CLM45BGC if phys=clm4_5/clm5_0)
This toggles on the namelist variables:
use_cn, use_lch4, use_nitrif_denitrif, use_vertsoilc, use_century_decomp
fates = FATES/Ecosystem Demography with below ground BGC
This toggles on the namelist variables:
use_fates, use_vertsoilc, use_century_decomp
(Only for CLM4.5/CLM5.0)
-[no-]chk_res Also check [do NOT check] to make sure the resolution and
land-mask is valid.
-clm_accelerated_spinup "on|off" Setup in a configuration to run as fast as possible for doing a throw-away
simulation in order to get the model to a spun-up state. So do things like
turn off expensive options and setup for a low level of history output.
If CLM4.5/CLM5.0 and bgc it also includes a prognostic Carbon model (cn or bgc)
, also by default turn on Accelerated Decomposition mode which
is controlled by the namelist variable spinup_state.
BGC Spinup for CLM4.5/5.0 Only (for CLM4.0 BGC spinup is controlled from configure)
Turn on given spinup mode for BGC setting of CN
on : Turn on Accelerated Decomposition (spinup_state = 1 or 2)
off : run in normal mode (spinup_state = 0)
Default is set by clm_accelerated_spinup mode.
Spinup is now a two step procedure. First, run the model
with clm_accelerated_spinup = "on". Then run the model for a while with
spinup_state = 0. The exit spinup step happens automatically
on the first timestep when using a restart file from spinup
mode.
The spinup state is saved to the restart file.
If the values match between the model and the restart
file it proceeds as directed.
If the restart file is in spinup mode and the model is in
normal mode, then it performs the exit spinup step
and proceeds in normal mode after that.
If the restart file has normal mode and the model is in
spinup, then it enters spinup. This is useful if you change
a parameter and want to rapidly re-equilibrate without doing
a cold start.
-clm_demand "list" List of variables to require on clm namelist besides the usuals.
"-clm_demand list" to list valid options.
(can include a list member "null" which does nothing)
-clm_start_type "type" Start type of simulation
(default, cold, arb_ic, startup, continue, or branch)
(default=do the default type for this configuration)
(cold=always start with arbitrary initial conditions)
(arb_ic=start with arbitrary initial conditions if
initial conditions do not exist)
(startup=ensure that initial conditions are being used)
-clm_usr_name "name" Dataset resolution/descriptor for personal datasets.
Default: not used
Example: 1x1pt_boulderCO_c090722 to describe location,
number of pts, and date files created
-co2_type "value" Set CO2 the type of CO2 variation to use.
-co2_ppmv "value" Set CO2 concentration to use when co2_type is constant (ppmv).
-crop Toggle for prognostic crop model. (default is off)
(can ONLY be turned on when BGC type is CN or BGC)
This turns on the namelist variable: use_crop
-csmdata "dir" Root directory of CESM input data.
Can also be set by using the CSMDATA environment variable.
-drydep Produce a drydep_inparm namelist that will go into the
"drv_flds_in" file for the driver to pass dry-deposition to the atm.
Default: -no-drydep
(Note: buildnml copies the file for use by the driver)
-dynamic_vegetation Toggle for dynamic vegetation model. (default is off)
(can ONLY be turned on when BGC type is 'cn' or 'bgc')
This turns on the namelist variable: use_cndv
-fire_emis Produce a fire_emis_nl namelist that will go into the
"drv_flds_in" file for the driver to pass fire emissions to the atm.
(Note: buildnml copies the file for use by the driver)
-glc_nec <name> Glacier number of elevation classes [0 | 3 | 5 | 10 | 36]
(default is 0) (standard option with land-ice model is 10)
-help [or -h] Print usage to STDOUT.
-light_res <value> Resolution of lightning dataset to use for CN fire (hcru or T62)
-ignore_ic_date Ignore the date on the initial condition files
when determining what input initial condition file to use.
-ignore_ic_year Ignore just the year part of the date on the initial condition files
when determining what input initial condition file to use.
-ignore_warnings Allow build-namelist to continue, rather than stopping on
warnings
-infile "filepath" Specify a file (or list of files) containing namelists to
read values from.
If used with a CLM build with multiple ensembles (ninst_lnd>1)
and the filename entered is a directory to files of the
form filepath/filepath and filepath/filepath_\$n where \$n
is the ensemble member number. the "filepath/filepath"
input namelist file is the master input namelist file
that is applied to ALL ensemble members.
(by default for CESM this is setup for files of the
form \$CASEDIR/user_nl_clm/user_nl_clm_????)
-inputdata "filepath" Writes out a list containing pathnames for required input datasets in
file specified.
-irrig "value" If .true. week surface datasets with irrigation turned on. (only allowed for CLM4.0 physics)
Default: .false.
(for CLM4.5/CLM5.0 physics set the namelist flag irrigate=.true.)
-l_ncpl "LND_NCPL" Number of CLM coupling time-steps in a day.
-lnd_tuning_mode "value" Use the parameters tuned for the given configuration (CLM version and atmospheric forcing)
-mask "landmask" Type of land-mask (default, navy, gx3v5, gx1v5 etc.)
"-mask list" to list valid land masks.
-namelist "namelist" Specify namelist settings directly on the commandline by supplying
a string containing FORTRAN namelist syntax, e.g.,
-namelist "&clm_inparm dt=1800 /"
-no-megan DO NOT PRODUCE a megan_emis_nl namelist that will go into the
"drv_flds_in" file for the driver to pass VOCs to the atm.
MEGAN (Model of Emissions of Gases and Aerosols from Nature)
(Note: buildnml copies the file for use by the driver)
-[no-]note Add note to output namelist [do NOT add note] about the
arguments to build-namelist.
-output_reals <file> Output real parameters to the given output file.
-ssp_rcp "value" Shared Socioeconomic Pathway (SSP) and
Representative Concentration Pathway (RCP) combination to use for
future scenarios.
"-ssp_rcp list" to list valid ssp_rcp settings.
-s Turns on silent mode - only fatal messages issued.
-test Enable checking that input datasets exist on local filesystem.
-use_case "case" Specify a use case which will provide default values.
"-use_case list" to list valid use-cases.
-verbose [or -v] Turn on verbose echoing of informational messages.
-version Echo the SVN tag name used to check out this CLM distribution.
-vichydro Toggle to turn on VIC hydrologic parameterizations (default is off)
This turns on the namelist variable: use_vichydro
Note: The precedence for setting the values of namelist variables is (highest to lowest):
0. namelist values set by specific command-line options, like, -d, -sim_year
(i.e. compset choice and CLM_BLDNML_OPTS, CLM_ACCELERATED_SPINUP, LND_TUNING_MODE env_run variables)
(NOTE: If you try to contradict these settings by methods below, an error will be triggered)
1. values set on the command-line using the -namelist option,
(i.e. CLM_NAMELIST_OPTS env_run variable)
2. values read from the file(s) specified by -infile,
(i.e. user_nl_clm files)
3. datasets from the -clm_usr_name option,
(i.e. CLM_USRDAT_NAME env_run variable)
4. values set from a use-case scenario, e.g., -use_case
(i.e. CLM_NML_USE_CASE env_run variable)
5. values from the namelist defaults file.
EOF
}
#-------------------------------------------------------------------------------
sub process_commandline {
# Process command-line options and return the hash
my ($nl_flags) = @_;
# Save the command line arguments to the script. NOTE: this must be
# before GetOptions() is called because items are removed from from
# the array!
$nl_flags->{'cmdline'} = "@ARGV";
my %opts = ( cimeroot => undef,
config => "config_cache.xml",
csmdata => undef,
clm_usr_name => undef,
co2_type => undef,
co2_ppmv => undef,
clm_demand => "null",
help => 0,
glc_nec => "default",
light_res => "default",
l_ncpl => undef,
lnd_tuning_mode => "default",
lnd_frac => undef,
dir => "$cwd",
ssp_rcp => "default",
sim_year => "default",
clm_accelerated_spinup=> "default",
chk_res => undef,
note => undef,
drydep => 0,
output_reals_filename => undef,
fire_emis => 0,
megan => "default",
irrig => "default",
res => "default",
silent => 0,
ignore_warnings => 0,
mask => "default",
test => 0,
bgc => "default",
crop => 0,
dynamic_vegetation => 0,
envxml_dir => ".",
vichydro => 0,
maxpft => "default",
);
GetOptions(
"cimeroot=s" => \$opts{'cimeroot'},
"clm_demand=s" => \$opts{'clm_demand'},
"co2_ppmv=f" => \$opts{'co2_ppmv'},
"co2_type=s" => \$opts{'co2_type'},
"config=s" => \$opts{'config'},
"csmdata=s" => \$opts{'csmdata'},
"clm_usr_name=s" => \$opts{'clm_usr_name'},
"envxml_dir=s" => \$opts{'envxml_dir'},
"drydep!" => \$opts{'drydep'},
"fire_emis!" => \$opts{'fire_emis'},
"ignore_warnings!" => \$opts{'ignore_warnings'},
"chk_res!" => \$opts{'chk_res'},
"note!" => \$opts{'note'},
"megan!" => \$opts{'megan'},
"glc_nec=i" => \$opts{'glc_nec'},
"light_res=s" => \$opts{'light_res'},
"irrig=s" => \$opts{'irrig'},
"d:s" => \$opts{'dir'},
"h|help" => \$opts{'help'},
"ignore_ic_date" => \$opts{'ignore_ic_date'},
"ignore_ic_year" => \$opts{'ignore_ic_year'},
"infile=s" => \$opts{'infile'},
"lnd_frac=s" => \$opts{'lnd_frac'},
"lnd_tuning_mode=s" => \$opts{'lnd_tuning_mode'},
"l_ncpl=i" => \$opts{'l_ncpl'},
"inputdata=s" => \$opts{'inputdata'},
"mask=s" => \$opts{'mask'},
"namelist=s" => \$opts{'namelist'},
"res=s" => \$opts{'res'},
"ssp_rcp=s" => \$opts{'ssp_rcp'},
"s|silent" => \$opts{'silent'},
"sim_year=s" => \$opts{'sim_year'},
"output_reals=s" => \$opts{'output_reals_filename'},
"clm_accelerated_spinup=s" => \$opts{'clm_accelerated_spinup'},
"clm_start_type=s" => \$opts{'clm_start_type'},
"test" => \$opts{'test'},
"use_case=s" => \$opts{'use_case'},
"bgc=s" => \$opts{'bgc'},
"crop!" => \$opts{'crop'},
"dynamic_vegetation" => \$opts{'dynamic_vegetation'},
"vichydro" => \$opts{'vichydro'},
"maxpft=i" => \$opts{'maxpft'},
"v|verbose" => \$opts{'verbose'},
"version" => \$opts{'version'},
) or usage();
# Give usage message.
usage() if $opts{'help'};
# Check for unparsed arguments
if (@ARGV) {
print "ERROR: unrecognized arguments: @ARGV\n";
usage();
}
return %opts;
}
#-------------------------------------------------------------------------------
sub check_for_perl_utils {
my $cfgdir = shift;
my $opts_ref = shift;
# Determine CIME root directory and perl5lib root directory
my $cimeroot = $opts_ref->{'cimeroot'};
if ( ! defined($cimeroot) ) {
$cimeroot = "$cfgdir/../cime";
if ( -d $cimeroot ) {
} elsif ( -d "$cfgdir/../../../cime" ) {
$cimeroot = "$cfgdir/../../../cime";
} else {
die <<"EOF";
** Cannot find the root of the cime directory enter it using the -cimeroot option
Did you run the checkout_externals scripts?
EOF
}
}
my $perl5lib_dir = "$cimeroot/utils/perl5lib";
#-----------------------------------------------------------------------------
# Add $perl5lib_dir to the list of paths that Perl searches for modules
my @dirs = ( $ProgDir, $cfgdir, "$perl5lib_dir");
unshift @INC, @dirs;
require config_files::clm_phys_vers;
require namelist_files::LogMessages;
my $locallog = namelist_files::LogMessages->new( $ProgName, $opts_ref );
# The XML::Lite module is required to parse the XML files.
(-f "$perl5lib_dir/XML/Lite.pm") or
$locallog->fatal_error("Cannot find perl module \"XML/Lite.pm\" in directory\n" .
"\"$perl5lib_dir\"");
# The Build::Config module provides utilities to access the configuration information
# in the config_cache.xml file
(-f "$perl5lib_dir/Build/Config.pm") or
$locallog->fatal_error("Cannot find perl module \"Build/Config.pm\" in directory\n" .
"\"$perl5lib_dir\"");
# The Build::NamelistDefinition module provides utilities to validate that the output
# namelists are consistent with the namelist definition file
(-f "$perl5lib_dir/Build/NamelistDefinition.pm") or
$locallog->fatal_error("Cannot find perl module \"Build/NamelistDefinition.pm\" in directory\n" .
"\"$perl5lib_dir\"");
# The Build::NamelistDefaults module provides a utility to obtain default values of namelist
# variables based on finding a best fit with the attributes specified in the defaults file.
(-f "$perl5lib_dir/Build/NamelistDefaults.pm") or
$locallog->fatal_error("Cannot find perl module \"Build/NamelistDefaults.pm\" in directory\n" .
"\"$perl5lib_dir\"");
# The Build::Namelist module provides utilities to parse input namelists, to query and modify
# namelists, and to write output namelists.
(-f "$perl5lib_dir/Build/Namelist.pm") or
$locallog->fatal_error("Cannot find perl module \"Build/Namelist.pm\" in directory\n" .
"\"$perl5lib_dir\"");
# required cesm perl modules
require XML::Lite;
require Build::Config;
require Build::NamelistDefinition;
require Build::NamelistDefaults;
require Build::Namelist;
require Config::SetupTools;
}
#-------------------------------------------------------------------------------
sub read_configure_definition {
# Read the configure definition and specific config_cache file for this case
# configure are the build-time settings for CLM
my ($cfgdir, $opts) = @_;
$log->verbose_message("Setting CLM configuration script directory to $cfgdir");
# Create a configuration object from the default config_definition file
my $configfile;
if ( -f $opts->{'config'} ) {
$configfile = $opts->{'config'};
} else {
$configfile = "$cfgdir/config_files/config_definition.xml";
}
# Check that configuration cache file exists.
$log->verbose_message("Using CLM configuration cache file $opts->{'config'}");
if ( $configfile ne $opts->{'config'} ) {
$log->fatal_error("Cannot find configuration cache file: \"$opts->{'config'}\"");
}
my $cfg = Build::Config->new("$configfile");
return $cfg;
}
#-----------------------------------------------------------------------------------------------
sub read_namelist_definition {
my ($cfgdir, $opts, $nl_flags, $physv) = @_;
# The namelist definition file contains entries for all namelist
# variables that can be output by build-namelist.
my $phys = $physv->as_filename( );
my @nl_definition_files = ( "$cfgdir/namelist_files/namelist_definition_drv.xml",
"$cfgdir/namelist_files/namelist_definition_drv_flds.xml",
"$cfgdir/namelist_files/namelist_definition_$phys.xml" );
foreach my $nl_defin_file ( @nl_definition_files ) {
(-f "$nl_defin_file") or $log->fatal_error("Cannot find namelist definition file \"$nl_defin_file\"");
$log->verbose_message("Using namelist definition file $nl_defin_file");
}
# Create a namelist definition object. This object provides a
# method for verifying that the output namelist variables are in the
# definition file, and are output in the correct namelist groups.
my $definition = Build::NamelistDefinition->new( shift(@nl_definition_files) );
foreach my $nl_defin_file ( @nl_definition_files ) {
$definition->add( "$nl_defin_file" );
}
return $definition;
}
#-----------------------------------------------------------------------------------------------
sub read_envxml_case_files {
# read the contents of the env*.xml files in the case directory
my ($opts) = @_;
my %envxml = ();
if ( defined($opts->{'envxml_dir'}) ) {
(-d $opts->{'envxml_dir'}) or $log->fatal_error( "envxml_dir is not a directory" );
my @files = glob( $opts->{'envxml_dir'}."/env_*xml" );
($#files >= 0) or $log->fatal_error( "there are no env_*xml files in the envxml_dir" );
foreach my $file (@files) {
$log->verbose_message( "Open env.xml file: $file" );
my $xml = XML::Lite->new( "$file" );
my @e = $xml->elements_by_name('entry');
while ( my $e = shift @e ) {
my %a = $e->get_attributes();
$envxml{$a{'id'}} = $a{'value'};
}
}
foreach my $attr (keys %envxml) {
if ( $envxml{$attr} =~ m/\$/ ) {
$envxml{$attr} = SetupTools::expand_xml_var( $envxml{$attr}, \%envxml );
}
}
} else {
$log->fatal_error( "The -envxml_dir option was NOT given and it is a REQUIRED option" );
}
return( %envxml );
}
#-----------------------------------------------------------------------------------------------
sub read_namelist_defaults {
my ($cfgdir, $opts, $nl_flags, $cfg, $physv) = @_;
my $phys = $physv->as_filename( );
# The namelist defaults file contains default values for all required namelist variables.
my @nl_defaults_files = ( "$cfgdir/namelist_files/namelist_defaults_overall.xml",
"$cfgdir/namelist_files/namelist_defaults_$phys.xml",
"$cfgdir/namelist_files/namelist_defaults_drv.xml",
"$cfgdir/namelist_files/namelist_defaults_fire_emis.xml",
"$cfgdir/namelist_files/namelist_defaults_drydep.xml" );
# Add the location of the use case defaults files to the options hash
$opts->{'use_case_dir'} = "$cfgdir/namelist_files/use_cases";
if (defined $opts->{'use_case'}) {
if ( $opts->{'use_case'} ne "list" ) {
unshift( @nl_defaults_files, "$opts->{'use_case_dir'}/$opts->{'use_case'}.xml" );
}
}
foreach my $nl_defaults_file ( @nl_defaults_files ) {
(-f "$nl_defaults_file") or $log->fatal_error("Cannot find namelist defaults file \"$nl_defaults_file\"");
$log->verbose_message("Using namelist defaults file $nl_defaults_file");
}
# Create a namelist defaults object. This object provides default
# values for variables contained in the input defaults file. The
# configuration object provides attribute values that are relevent
# for the CLM executable for which the namelist is being produced.
my $defaults = Build::NamelistDefaults->new( shift( @nl_defaults_files ), $cfg);
foreach my $nl_defaults_file ( @nl_defaults_files ) {
$defaults->add( "$nl_defaults_file" );
}
return $defaults;
}
#-------------------------------------------------------------------------------
sub check_cesm_inputdata {
# Check that the CESM inputdata root directory has been specified. This must be
# a local or nfs mounted directory.
my ($opts, $nl_flags) = @_;
$nl_flags->{'inputdata_rootdir'} = undef;
if (defined($opts->{'csmdata'})) {
$nl_flags->{'inputdata_rootdir'} = $opts->{'csmdata'};
}
elsif (defined $ENV{'CSMDATA'}) {
$nl_flags->{'inputdata_rootdir'} = $ENV{'CSMDATA'};
}
else {
$log->fatal_error("CESM inputdata root directory must be specified by either -csmdata\n" .
"argument or by the CSMDATA environment variable.");
}
if ( ! defined($ENV{'DIN_LOC_ROOT'}) ) {
$ENV{'DIN_LOC_ROOT'} = $nl_flags->{'inputdata_rootdir'};
}
if ($opts->{'test'}) {
(-d $nl_flags->{'inputdata_rootdir'}) or $log->fatal_error("CESM inputdata root is not a directory: \"$nl_flags->{'inputdata_rootdir'}\"");
}
$log->verbose_message("CESM inputdata root directory: $nl_flags->{'inputdata_rootdir'}");
}
#-------------------------------------------------------------------------------
sub process_namelist_user_input {
# Process the user input in general by order of precedence. At each point
# we'll only add new values to the namelist and not overwrite
# previously specified specified values which have higher
# precedence. The one exception to this rule are the specifc command-line
# options which are done last as if the user contradicts these settings
# CLM build-namelist will abort with an error.
#
# 1. values set on the command-line using the -namelist option,
# (i.e. CLM_NAMELIST_OPTS env_run variable)
# 2. values read from the file(s) specified by -infile,
# (i.e. user_nl_clm files)
# After the above are done the command line options are processed and they
# are made sure the user hasn't contradicted any of their settings with
# anything above. Because of this they are condsidered to have the highest
# precedence.
# 0. namelist values set by specific command-line options, like, -d, -sim_year
# (i.e. CLM_BLDNML_OPTS env_run variable)
# The results of these are needed for the final two user input
# 3. datasets from the -clm_usr_name option,
# (i.e. CLM_USRDAT_NAME env_run variable)
# 4. values set from a use-case scenario, e.g., -use_case
# (i.e. CLM_NML_USE_CASE env_run variable)
#
# Finally after all the above is done, the defaults are found from the
# namelist defaults file (outside of this routine).
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $envxml_ref, $physv) = @_;
# Get the inputs that will be coming from the user...
process_namelist_commandline_namelist($opts, $definition, $nl, $envxml_ref);
process_namelist_commandline_infile($opts, $definition, $nl, $envxml_ref);
# Apply the commandline options and make sure the user didn't change it above
process_namelist_commandline_options($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv);
# The last two process command line arguments for usr_name and use_case
# They require that process_namelist_commandline_options was called before this
process_namelist_commandline_clm_usr_name($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $envxml_ref);
process_namelist_commandline_use_case($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $envxml_ref, $physv);
# Set the start_type by the command line setting for clm_start_type
process_namelist_commandline_clm_start_type($opts, $nl_flags, $definition, $defaults, $nl);
}
#-------------------------------------------------------------------------------
sub process_namelist_commandline_options {
# First process the commandline args that provide specific namelist values.
#
# First get the command-line specified overall values or their defaults
# Obtain default values for the following build-namelist input arguments
# : res, mask, ssp_rcp, sim_year, sim_year_range, and clm_accelerated_spinup.
#
# NOTE: cfg only needs to be passed to functions that work with
# clm4_0 compile time functionality!
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv) = @_;
setup_cmdl_chk_res($opts, $defaults);
setup_cmdl_resolution($opts, $nl_flags, $definition, $defaults);
setup_cmdl_mask($opts, $nl_flags, $definition, $defaults, $nl);
setup_cmdl_bgc($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv);
setup_cmdl_fire_light_res($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv);
setup_cmdl_spinup($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv);
setup_cmdl_crop($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv);
setup_cmdl_maxpft($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv);
setup_cmdl_glc_nec($opts, $nl_flags, $definition, $defaults, $nl);
setup_cmdl_irrigation($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_cmdl_ssp_rcp($opts, $nl_flags, $definition, $defaults, $nl);
setup_cmdl_simulation_year($opts, $nl_flags, $definition, $defaults, $nl);
setup_cmdl_dynamic_vegetation($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_cmdl_fates_mode($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_cmdl_vichydro($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_cmdl_run_type($opts, $nl_flags, $definition, $defaults, $nl);
setup_cmdl_output_reals($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_lnd_tuning($opts, $nl_flags, $definition, $defaults, $nl, $physv);
}
#-------------------------------------------------------------------------------
sub setup_cmdl_chk_res {
my ($opts, $defaults) = @_;
my $var = "chk_res";
if ( ! defined($opts->{$var}) ) {
$opts->{$var} = $defaults->get_value($var);
}
}
sub setup_cmdl_resolution {
my ($opts, $nl_flags, $definition, $defaults) = @_;
my $var = "res";
my $val;
if ( $opts->{$var} ne "default" ) {
$val = $opts->{$var};
} else {
$val= $defaults->get_value($var);
}
$nl_flags->{'res'} = $val;
$log->verbose_message("CLM atm resolution is $nl_flags->{'res'}");
$opts->{$var} = $val;
if ( $opts->{'chk_res'} ) {
$val = &quote_string( $nl_flags->{'res'} );
if ( ! $definition->is_valid_value( $var, $val ) ) {
my @valid_values = $definition->get_valid_values( $var );
if ( ! defined($opts->{'clm_usr_name'}) || $nl_flags->{'res'} ne $opts->{'clm_usr_name'} ) {
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_cmdl_mask {
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
my $var = "mask";
my $val;
if ( $opts->{$var} ne "default" ) {
$val = $opts->{$var};
} else {
my %tmp = ( 'hgrid'=>$nl_flags->{'res'} );
$val = $defaults->get_value($var, \%tmp );
}
$nl_flags->{'mask'} = $val;
$opts->{'mask'} = $nl_flags->{'mask'};
if ( $opts->{'chk_res'} ) {
$val = &quote_string( $val );
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val);
if ( ! $definition->is_valid_value( $var, $val ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
}
$log->verbose_message("CLM land mask is $nl_flags->{'mask'}");
}
#-------------------------------------------------------------------------------
sub setup_cmdl_fates_mode {
#
# call this at least after crop check is called
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $val;
my $var = "bgc_mode";
if ( $physv->as_long() == $physv->as_long("clm4_0") || $nl_flags->{'crop'} eq "on" ) {
if ( $nl_flags->{$var} eq "fates" ) {
# ED is not a clm4_0 option and should not be used with crop and not with clm4_0
$log->fatal_error("** Cannot turn fates mode on with crop or with clm4_0 physics." );
}
} elsif ($nl_flags->{"bgc_mode"} eq "fates" && ! &value_is_true($nl_flags->{"use_fates"}) ) {
$log->fatal_error("DEV_ERROR: internal logic error: bgc_mode = fates and use_fates = false.");
} else {
$var = "use_fates";
if ( &value_is_true($nl_flags->{$var}) ) {
# This section is a place-holder to test for modules that are not allowed with ED
# the defaults which are set in the logic section of the namelist builder will
# automatically set these correctly (well that is the assumption), but here we
# want to set a catch to fail and warn users if they explicitly set incompatible user namelist
# options
# my $var = "use_somevar";
# $val = $nl_flags->{$var};
# if ( defined($nl->get_value($var)) ) {
# if ( &value_is_true($nl->get_value($var)) ) {
# $log->fatal_error("$var was set to .true., which is incompatible when -bgc fates option is used.");
# }
# }
# The following variables may be set by the user and are compatible with use_fates
# no need to set defaults, covered in a different routine
my @list = ( "use_vertsoilc", "use_century_decomp", "use_lch4" );
foreach my $var ( @list ) {
if ( defined($nl->get_value($var)) ) {
$nl_flags->{$var} = $nl->get_value($var);
$val = $nl_flags->{$var};
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val);
if ( ! $definition->is_valid_value( $var, $val ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
}
}
} else {
# dis-allow fates specific namelist items with non-fates runs
my @list = ( "use_fates_spitfire", "use_fates_planthydro", "use_fates_ed_st3", "use_fates_ed_prescribed_phys",
"use_fates_inventory_init", "fates_inventory_ctrl_filename","use_fates_logging","fates_parteh_mode" );
foreach my $var ( @list ) {
if ( defined($nl->get_value($var)) ) {
$log->fatal_error("$var is being set, but can ONLY be set when -bgc fates option is used.\n");
}
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_cmdl_bgc {
# BGC - alias for group of biogeochemistry related use_XXX namelists
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv) = @_;
my $val;
my $var = "bgc";
$val = $opts->{$var};
$nl_flags->{'bgc_mode'} = $val;
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
if ( $nl_flags->{'bgc_mode'} ne "default" ) {
$log->fatal_error("-bgc option used with clm4_0 physics. -bgc can ONLY be used with clm4_5/clm5_0 physics");
}
$nl_flags->{'bgc_mode'} = $cfg->get($var);
} else {
my $var = "bgc_mode";
if ( $nl_flags->{$var} eq "default" ) {
$nl_flags->{$var} = $defaults->get_value($var);
}
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, quote_string( $nl_flags->{$var} ) );
if ( ! $definition->is_valid_value( $var, quote_string( $nl_flags->{$var}) ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value (".$nl_flags->{$var}.") that is NOT valid. Valid values are: @valid_values");
}
$log->verbose_message("Using $nl_flags->{$var} for bgc.");
# now set the actual name list variables based on the bgc alias
if ($nl_flags->{$var} eq "cn" ) {
$nl_flags->{'use_cn'} = ".true.";
$nl_flags->{'use_fates'} = ".false.";
} elsif ($nl_flags->{$var} eq "bgc" ) {
$nl_flags->{'use_cn'} = ".true.";
$nl_flags->{'use_fates'} = ".false.";
} elsif ($nl_flags->{$var} eq "fates" ) {
$nl_flags->{'use_cn'} = ".false.";
$nl_flags->{'use_fates'} = ".true.";
} else {
$nl_flags->{'use_cn'} = ".false.";
$nl_flags->{'use_fates'} = ".false.";
}
if ( defined($nl->get_value("use_cn")) && ($nl_flags->{'use_cn'} ne $nl->get_value("use_cn")) ) {
$log->fatal_error("The namelist variable use_cn is inconsistent with the -bgc option");
}
if ( defined($nl->get_value("use_fates")) && ($nl_flags->{'use_fates'} ne $nl->get_value("use_fates")) ) {
$log->fatal_error("The namelist variable use_fates is inconsistent with the -bgc option");
}
{
# If the variable has already been set use it, if not set to the value defined by the bgc_mode
my @list = ( "use_lch4", "use_nitrif_denitrif", "use_vertsoilc", "use_century_decomp" );
my $ndiff = 0;
my %settings = ( 'bgc_mode'=>$nl_flags->{'bgc_mode'} );
foreach my $var ( @list ) {
my $default_setting = $defaults->get_value($var, \%settings );
if ( ! defined($nl->get_value($var)) ) {
$nl_flags->{$var} = $default_setting;
} else {
if ( $nl->get_value($var) ne $default_setting ) {
$ndiff += 1;
}
$nl_flags->{$var} = $nl->get_value($var);
}
$val = $nl_flags->{$var};
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val);
if ( ! $definition->is_valid_value( $var, $val ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
}
# If all the variables are different report it as an error
if ( $ndiff == ($#list + 1) ) {
$log->fatal_error("You are contradicting the -bgc setting with the namelist variables: @list" );
}
}
# Now set use_cn and use_fates
foreach $var ( "use_cn", "use_fates" ) {
$val = $nl_flags->{$var};
$group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val);
if ( ! $definition->is_valid_value( $var, $val ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
}
}
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
my $var = "use_fun";
if ( ! defined($nl->get_value($var)) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var,
'phys'=>$nl_flags->{'phys'}, 'use_cn'=>$nl_flags->{'use_cn'},
'use_nitrif_denitrif'=>$nl_flags->{'use_nitrif_denitrif'} );
}
if ( (! &value_is_true($nl_flags->{'use_nitrif_denitrif'}) ) && &value_is_true($nl->get_value('use_fun')) ) {
$log->fatal_error("When FUN is on, use_nitrif_denitrif MUST also be on!");
}
}
} # end bgc
#-------------------------------------------------------------------------------
sub setup_cmdl_fire_light_res {
# light_res - alias for lightning resolution
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv) = @_;
my $var = "light_res";
my $val = $opts->{$var};
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
if ( $val !~ /default|none/ ) {
$log->fatal_error("-$var option used with clm4_0 physics. -$var can ONLY be used with clm4_5/clm5_0 physics");
}
} else {
if ( $val eq "default" ) {
$nl_flags->{$var} = remove_leading_and_trailing_quotes($defaults->get_value($var));
} else {
my $fire_method = remove_leading_and_trailing_quotes( $nl->get_value('fire_method') );
if ( defined($fire_method) && $val ne "none" ) {
if ( $fire_method eq "nofire" ) {
$log->fatal_error("-$var option used with fire_method='nofire'. -$var can ONLY be used without the nofire option");
}
}
my $stream_fldfilename_lightng = remove_leading_and_trailing_quotes( $nl->get_value('stream_fldfilename_lightng') );
if ( defined($stream_fldfilename_lightng) && $val ne "none" ) {
$log->fatal_error("-$var option used while also explicitly setting stream_fldfilename_lightng filename which is a contradiction. Use one or the other not both.");
}
if ( ! &value_is_true($nl->get_value('use_cn')) ) {
$log->fatal_error("-$var option used CN is NOT on. -$var can only be used when CN is on (with bgc: cn or bgc)");
}
if ( &value_is_true($nl->get_value('use_cn')) && $val eq "none" ) {
$log->fatal_error("-$var option is set to none, but CN is on (with bgc: cn or bgc) which is a contradiction");
}
$nl_flags->{$var} = $val;
}
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, quote_string($nl_flags->{$var}) );
if ( ! $definition->is_valid_value( $var, $nl_flags->{$var}, 'noquotes'=>1 ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value (".$nl_flags->{$var}.") that is NOT valid. Valid values are: @valid_values");
}
$log->verbose_message("Using $nl_flags->{$var} for $var.");
#
# Set flag if cn-fires are on or not
#
$var = "cnfireson";
if ( $physv->as_long() >= $physv->as_long("clm4_5") && &value_is_true($nl->get_value('use_cn')) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fire_method');
}
my $fire_method = remove_leading_and_trailing_quotes( $nl->get_value('fire_method') );
if ( defined($fire_method) && ! &value_is_true($nl_flags->{'use_cn'}) ) {
$log->fatal_error("fire_method is being set even though bgc is NOT cn or bgc.");
}
if ( defined($fire_method) && $fire_method eq "nofire" ) {
$nl_flags->{$var} = ".false.";
} elsif ( &value_is_true($nl->get_value('use_cn')) ) {
$nl_flags->{$var} = ".true.";
} else {
$nl_flags->{$var} = ".false.";
}
}
}
#-------------------------------------------------------------------------------
sub setup_cmdl_crop {
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv) = @_;
$nl_flags->{'use_crop'} = ".false.";
my $val;
my $var = "crop";
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
$nl_flags->{'crop'} = $cfg->get($var);
if ( $nl_flags->{'crop'} eq "on" ) {
$nl_flags->{'use_crop'} = ".true.";
}
} else {
$val = $opts->{$var};
$nl_flags->{'crop'} = $val;
if ( $nl_flags->{'crop'} eq 1 ) {
$nl_flags->{'use_crop'} = ".true.";
}
if ( defined($nl->get_value("use_crop")) && ($nl_flags->{'use_crop'} ne $nl->get_value("use_crop")) ) {
$log->fatal_error("Namelist item use_crop contradicts the command-line option -crop, use the command line option");
}
if ( ($nl_flags->{'crop'} eq 1 ) && ($nl_flags->{'bgc_mode'} eq "sp") ) {
$log->fatal_error("** Cannot turn crop mode on mode bgc=sp\n" .
"**\n" .
"** Set the bgc mode to 'cn' or 'bgc' by the following means from highest to lowest precedence:\n" .
"** * by the command-line options -bgc cn\n" .
"** * by a default configuration file, specified by -defaults");
}
$var = "use_crop";
$val = ".false.";
if ($nl_flags->{'crop'} eq 1) {
$val = ".true.";
}
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val);
if ( ! $definition->is_valid_value( $var, $val ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
}
}
#-------------------------------------------------------------------------------
sub setup_cmdl_maxpft {
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv) = @_;
my $val;
my $var = "maxpft";
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
$nl_flags->{'maxpft'} = $cfg->get($var);
# NOTE: maxpatchpft sizes already checked for clm4_0 by configure.
} else {
my %maxpatchpft;
$maxpatchpft{'.true.'} = 79;
$maxpatchpft{'.false.'} = 17;
if ( $opts->{$var} ne "default") {
$val = $opts->{$var};
} else {
$val = $maxpatchpft{$nl_flags->{'use_crop'}};
}
$nl_flags->{'maxpft'} = $val;
if ( ($nl_flags->{'bgc_mode'} ne "sp") && ($nl_flags->{'maxpft'} != $maxpatchpft{$nl_flags->{'use_crop'}}) ) {
$log->fatal_error("** For CN or BGC mode you MUST set max patch PFT's to $maxpatchpft{$nl_flags->{'use_crop'}}\n" .
"**\n" .
"** When the crop model is on then it must be set to $maxpatchpft{'crop'} otherwise to $maxpatchpft{'nocrop'}\n" .
"** Set the bgc mode, crop and maxpft by the following means from highest to lowest precedence:\n" .
"** * by the command-line options -bgc, -crop and -maxpft\n" .
"** * by a default configuration file, specified by -defaults\n" .
"**");
}
if ( $nl_flags->{'maxpft'} > $maxpatchpft{$nl_flags->{'use_crop'}} ) {
$log->fatal_error("** Max patch PFT's can NOT exceed $maxpatchpft{$nl_flags->{'use_crop'}}\n" .
"**\n" .
"** Set maxpft by the following means from highest to lowest precedence:\n" .
"** * by the command-line options -maxpft\n" .
"** * by a default configuration file, specified by -defaults\n" .
"**");
}
if ( $nl_flags->{'maxpft'} != $maxpatchpft{$nl_flags->{'use_crop'}} ) {
$log->warning("running with maxpft NOT equal to $maxpatchpft{$nl_flags->{'use_crop'}} is " .
"NOT validated / scientifically supported." );
}
$log->verbose_message("Using $nl_flags->{'maxpft'} for maxpft.");
$var = "maxpatch_pft";
my $group = $definition->get_group_name($var);
if ( ! defined($nl->get_variable_value($group, $var)) ) {
$val = $nl_flags->{'maxpft'};
$nl->set_variable_value($group, $var, $val);
}
$val = $nl->get_variable_value($group, $var);
my @valid_values = ($maxpatchpft{'.true.'}, $maxpatchpft{'.false.'} );
my $found = 0;
foreach my $valid_val ( @valid_values ) {
if ( $val == $valid_val ) {
$found = 1;
last;
}
}
if ( ! $found ) {
$log->warning("$var has a value ($val) that is normally NOT valid. Normal valid values are: @valid_values");
}
}
}
#-------------------------------------------------------------------------------
sub setup_cmdl_glc_nec {
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
my $val;
my $var = "glc_nec";
if ( $opts->{$var} ne "default" ) {
$val = $opts->{$var};
} else {
$val = $defaults->get_value($var);
}
$nl_flags->{'glc_nec'} = $val;
$opts->{'glc_nec'} = $val;
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val);
if ( ! $definition->is_valid_value( $var, $val ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
$log->verbose_message("Glacier number of elevation classes is $val");
}
#-------------------------------------------------------------------------------
sub setup_cmdl_irrigation {
# Must be after setup_cmdl_crop
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $var = "irrig";
if ( $opts->{$var} eq "default" ) {
my %settings;
$settings{'use_crop'} = $nl_flags->{'use_crop'};
$nl_flags->{$var} = $defaults->get_value($var, \%settings);
} else {
$nl_flags->{$var} = $opts->{$var};
}
my $val = $nl_flags->{$var};
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val);
if ( ! $definition->is_valid_value( $var, $val ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
$log->verbose_message("Irrigation $val");
if ( &value_is_true($nl_flags->{'irrig'}) && &value_is_true($nl_flags->{'use_crop'}) ) {
$log->fatal_error("You've turned on both irrigation and crop.\n" .
"Irrigation is only applied to generic crop currently,\n" .
"which negates it's practical usage.\n." .
"We also have a known problem when both are on " .
"(see bug 1326 in the components/clm/doc/KnownBugs file)\n" .
"both irrigation and crop can NOT be on.");
}
} elsif ( $opts->{$var} ne "default" ) {
$log->fatal_error("The -irrig option can ONLY be used with clm4_0 physics");
}
}
#-------------------------------------------------------------------------------
sub setup_cmdl_ssp_rcp {
# shared socioeconmic pathway and representative concentration pathway combination
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
my $val;
my $var = "ssp_rcp";
if ( $opts->{$var} ne "default" ) {
$val = $opts->{$var};
} else {
$val = remove_leading_and_trailing_quotes( $defaults->get_value($var) );
}
$nl_flags->{'ssp_rcp'} = $val;
$opts->{'ssp_rcp'} = $nl_flags->{'ssp_rcp'};
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, quote_string($val) );
if ( ! $definition->is_valid_value( $var, $val, 'noquotes'=>1 ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
$log->verbose_message("CLM future scenario SSP-RCP combination is $nl_flags->{'ssp_rcp'}");
}
#-------------------------------------------------------------------------------
sub setup_cmdl_spinup {
# CLM 4.0 --> BGC spinup mode controlled from "spinup" in configure
# CLM 4.5/5.0 --> BGC spinup mode controlled from "clm_accelerated_spinup" in build-namelist
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $physv) = @_;
my $val;
my $var;
$nl_flags->{'spinup'} = undef;
$var = "clm_accelerated_spinup";
if ( $opts->{$var} ne "default" ) {
$val = $opts->{$var};
} else {
$val = $defaults->get_value($var);
}
$nl_flags->{$var} = $val;
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, quote_string($val) );
if ( ! $definition->is_valid_value( $var, $val , 'noquotes' => 1) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has an invalid value ($val). Valid values are: @valid_values");
}
$log->verbose_message("CLM accelerated spinup mode is $val");
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
$nl_flags->{'spinup'} = $cfg->get('spinup');
} elsif ( $physv->as_long() >= $physv->as_long("clm4_5")) {
if ( &value_is_true($nl_flags->{'use_cn'}) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition,
$defaults, $nl, "spinup_state", clm_accelerated_spinup=>$nl_flags->{$var},
use_cn=>$nl_flags->{'use_cn'}, use_fates=>$nl_flags->{'use_fates'} );
if ( $nl->get_value("spinup_state") ne 0 ) {
$nl_flags->{'bgc_spinup'} = "on";
if ( $nl_flags->{'bgc_mode'} eq "sp" ) {
$log->fatal_error("spinup_state is accelerated (=1 or 2) which is for a BGC mode of CN or BGC," .
" but the BGC mode is Satellite Phenology, change one or the other");
}
if ( $nl_flags->{'clm_accelerated_spinup'} eq "off" ) {
$log->fatal_error("spinup_state is accelerated, but clm_accelerated_spinup is off, change one or the other");
}
}
} else {
$nl_flags->{'bgc_spinup'} = "off";
$val = $defaults->get_value($var);
}
$nl_flags->{$var} = $val;
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, quote_string($val) );
if ( ! $definition->is_valid_value( $var, $val , 'noquotes' => 1) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has an invalid value ($val). Valid values are: @valid_values");
}
if ( $nl_flags->{'bgc_spinup'} eq "on" && (not &value_is_true( $nl_flags->{'use_cn'} )) && (not &value_is_true($nl_flags->{'use_fates'})) ) {
$log->fatal_error("$var can not be '$nl_flags->{'bgc_spinup'}' if neither CN nor ED is turned on (use_cn=$nl_flags->{'use_cn'}, use_fates=$nl_flags->{'use_fates'}).");
}
if ( $nl->get_value("spinup_state") eq 0 && $nl_flags->{'bgc_spinup'} eq "on" ) {
$log->fatal_error("Namelist spinup_state contradicts the command line option bgc_spinup" );
}
if ( $nl->get_value("spinup_state") eq 1 && $nl_flags->{'bgc_spinup'} eq "off" ) {
$log->fatal_error("Namelist spinup_state contradicts the command line option bgc_spinup" );
}
}
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
$val = $nl_flags->{'spinup'};
} else {
$val = $nl_flags->{'bgc_spinup'};
}
$log->verbose_message("CLM CN bgc_spinup mode is $val");
}
#-------------------------------------------------------------------------------
sub setup_cmdl_simulation_year {
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg) = @_;
my $val;
my $var = "sim_year";
if ( $opts->{$var} ne "default" ) {
$val = $opts->{$var};
} else {
$val = $defaults->get_value($var);
}
$nl_flags->{'sim_year_range'} = $defaults->get_value("sim_year_range");
$nl_flags->{'sim_year'} = &remove_leading_and_trailing_quotes($val);
if ( $val =~ /([0-9]+)-([0-9]+)/ ) {
$nl_flags->{'sim_year'} = $1;
$nl_flags->{'sim_year_range'} = $val;
}
$val = $nl_flags->{'sim_year'};
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, "'$val'" );
if ( ! $definition->is_valid_value( $var, $val, 'noquotes'=>1 ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var of $val is NOT valid. Valid values are: @valid_values");
}
$nl->set_variable_value($group, $var, "'$val'" );
$log->verbose_message("CLM sim_year is $nl_flags->{'sim_year'}");
$var = "sim_year_range";
$val = $nl_flags->{'sim_year_range'};
if ( $val ne "constant" ) {
$opts->{$var} = $val;
$group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val );
if ( ! $definition->is_valid_value( $var, $val, 'noquotes'=>1 ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var of $val is NOT valid. Valid values are: @valid_values");
}
$val = "'".$defaults->get_value($var)."'";
$nl->set_variable_value($group, $var, $val );
$log->verbose_message("CLM sim_year_range is $nl_flags->{'sim_year_range'}");
}
}
#-------------------------------------------------------------------------------
sub setup_cmdl_run_type {
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
my $val;
my $var = "clm_start_type";
my $ic_date = $nl->get_value('start_ymd');
my $st_year = int( $ic_date / 10000);
if (defined $opts->{$var}) {
if ($opts->{$var} eq "default" ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var,
'use_cndv'=>$nl_flags->{'use_cndv'}, 'use_fates'=>$nl_flags->{'use_fates'},
'sim_year'=>$st_year );
} else {
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, quote_string( $opts->{$var} ) );
}
} else {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var,
'use_cndv'=>$nl_flags->{'use_cndv'}, 'use_fates'=>$nl_flags->{'use_fates'},
'sim_year'=>$st_year );
}
$nl_flags->{'clm_start_type'} = $nl->get_value($var);
}
#-------------------------------------------------------------------------------
sub setup_cmdl_dynamic_vegetation {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $val;
my $var = "dynamic_vegetation";
$val = $opts->{$var};
$nl_flags->{'dynamic_vegetation'} = $val;
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
# not applicable
if ( $nl_flags->{'dynamic_vegetation'}eq 1) {
$log->fatal_error("** Turn dynamic_vegetation mode on with CLM_CONFIG_OPTS (-bgc cndv) for clm4_0 physics." );
}
} else {
if ( ($nl_flags->{'dynamic_vegetation'} eq 1 ) && ($nl_flags->{'bgc_mode'} eq "sp") ) {
$log->fatal_error("** Cannot turn dynamic_vegetation mode on with bgc=sp.\n" .
"**\n" .
"** Set the bgc mode to 'cn' or 'bgc' by the following means from highest to lowest precedence:" .
"** * by the command-line options -bgc cn\n");
}
$var = "use_cndv";
$nl_flags->{$var} = ".false.";
if ($nl_flags->{'dynamic_vegetation'} eq 1) {
$val = ".true.";
$nl_flags->{$var} = $val;
}
if ( defined($nl->get_value($var)) && $nl->get_value($var) ne $val ) {
$log->fatal_error("$var is inconsistent with the commandline setting of -dynamic_vegetation");
}
if ( &value_is_true($nl_flags->{$var}) ) {
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val);
if ( ! $definition->is_valid_value( $var, $val ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_cmdl_output_reals {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $var = "output_reals_filename";
my $file = $opts->{$var};
if ( defined($file) ) {
# Make sure can open file and if not die with an error
my $fh = IO::File->new($file, '>') or $log->fatal_error("can't create real parameter filename: $file");
$fh->close();
}
}
#-------------------------------------------------------------------------------
sub setup_cmdl_vichydro {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $val;
my $var = "vichydro";
$val = $opts->{$var};
$nl_flags->{'vichydro'} = $val;
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
# not relevant in clm4_0
if ( $nl_flags->{'vichydro'}eq 1) {
$log->fatal_error("** Cannot turn vichydro on with clm4_0 physics." );
}
} else {
if ($nl_flags->{'vichydro'} eq 1) {
$log->verbose_message("Using VIC hydrology for runoff calculations.");
}
$var = "use_vichydro";
$val = $nl->get_value($var);
if ($nl_flags->{'vichydro'} eq 1) {
my $group = $definition->get_group_name($var);
my $set = ".true.";
if ( defined($val) && $set ne $val ) {
$log->fatal_error("$var contradicts the command-line -vichydro option" );
}
$nl->set_variable_value($group, $var, $set);
if ( ! $definition->is_valid_value($var, $val) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value ($val) that is NOT valid. Valid values are: @valid_values");
}
}
}
}
#-------------------------------------------------------------------------------
sub process_namelist_commandline_namelist {
# Process the commandline '-namelist' arg.
my ($opts, $definition, $nl, $envxml_ref) = @_;
if (defined $opts->{'namelist'}) {
# Parse commandline namelist
my $nl_arg = Build::Namelist->new($opts->{'namelist'});
# Validate input namelist -- trap exceptions
my $nl_arg_valid;
eval { $nl_arg_valid = $definition->validate($nl_arg); };
if ($@) {
$log->fatal_error("Invalid namelist variable in commandline arg '-namelist'.\n $@");
}
# Go through all variables and expand any XML env settings in them
expand_xml_variables_in_namelist( $nl_arg_valid, $envxml_ref );
# Merge input values into namelist. Previously specified values have higher precedence
# and are not overwritten.
$nl->merge_nl($nl_arg_valid);
}
}
#-------------------------------------------------------------------------------
sub process_namelist_commandline_infile {
# Process the commandline '-infile' arg.
my ($opts, $definition, $nl, $envxml_ref) = @_;
if (defined $opts->{'infile'}) {
my @infiles = split( /,/, $opts->{'infile'} );
foreach my $infile ( @infiles ) {
# Make sure a valid file was found
if ( -f "$infile" ) {
# Otherwise abort as a valid file doesn't exist
} else {
$log->fatal_error("input namelist file does NOT exist $infile.\n $@");
}
# Parse namelist input from the next file
my $nl_infile = Build::Namelist->new($infile);
# Validate input namelist -- trap exceptions
my $nl_infile_valid;
eval { $nl_infile_valid = $definition->validate($nl_infile); };
if ($@) {
$log->fatal_error("Invalid namelist variable in '-infile' $infile.\n $@");
}
# Go through all variables and expand any XML env settings in them
expand_xml_variables_in_namelist( $nl_infile_valid, $envxml_ref );
# Merge input values into namelist. Previously specified values have higher precedence
# and are not overwritten.
$nl->merge_nl($nl_infile_valid);
}
}
}
#-------------------------------------------------------------------------------
sub process_namelist_commandline_clm_usr_name {
# Process the -clm_usr_name argument
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $envxml_ref) = @_;
if (defined $opts->{'clm_usr_name'}) {
# The user files definition is contained in an xml file with the same format as the defaults file.
# The one difference is that variables are expanded.
# Create a new NamelistDefaults object.
my $nl_defaults_file = "$nl_flags->{'cfgdir'}/namelist_files/namelist_defaults_usr_files.xml";
my $uf_defaults = Build::NamelistDefaults->new("$nl_defaults_file", $cfg );
# Loop over the variables specified in the user files
# Add each one to the namelist.
my @vars = $uf_defaults->get_variable_names();
my %settings;
$settings{'mask'} = $nl_flags->{'mask'};
$settings{'sim_year'} = $nl_flags->{'sim_year'};
$settings{'ssp_rcp'} = $nl_flags->{'ssp_rcp'};
$settings{'sim_year_range'} = $nl_flags->{'sim_year_range'};
$settings{'clm_accelerated_spinup'} = $nl_flags->{'clm_accelerated_spinup'};
$settings{'clm_usr_name'} = $opts->{'clm_usr_name'};
if ( $nl_flags->{'inputdata_rootdir'} eq "\$DIN_LOC_ROOT" ) {
$settings{'csmdata'} = $ENV{'DIN_LOC_ROOT'};
} else {
$settings{'csmdata'} = $nl_flags->{'inputdata_rootdir'};
}
my $nvars = 0;
my $nl_usrfile = Build::Namelist->new();
foreach my $var (@vars) {
my $val = $uf_defaults->get_usr_file($var, $definition, \%settings);
if ($val) {
$log->message("adding clm user file defaults for var $var with val $val");
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl_usrfile, $var, 'val'=>$val);
$nvars++;
}
}
if ( $nvars == 0 ) {
$log->message("setting clm_usr_name -- but did NOT find any user datasets: $opts->{'clm_usr_name'}", $opts);
}
# Go through all variables and expand any XML env settings in them
expand_xml_variables_in_namelist( $nl_usrfile, $envxml_ref );
# Merge input values into namelist. Previously specified values have higher precedence
# and are not overwritten.
$nl->merge_nl($nl_usrfile);
}
}
#-------------------------------------------------------------------------------
sub process_namelist_commandline_use_case {
# Now process the -use_case arg.
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $envxml_ref, $physv) = @_;
if (defined $opts->{'use_case'}) {
# The use case definition is contained in an xml file with the same format as the defaults file.
# Create a new NamelistDefaults object.
my $uc_defaults = Build::NamelistDefaults->new("$opts->{'use_case_dir'}/$opts->{'use_case'}.xml", $cfg);
my %settings;
$settings{'res'} = $nl_flags->{'res'};
$settings{'ssp_rcp'} = $nl_flags->{'ssp_rcp'};
$settings{'mask'} = $nl_flags->{'mask'};
$settings{'sim_year'} = $nl_flags->{'sim_year'};
$settings{'sim_year_range'} = $nl_flags->{'sim_year_range'};
$settings{'phys'} = $nl_flags->{'phys'};
$settings{'lnd_tuning_mode'}= $nl_flags->{'lnd_tuning_mode'};
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
$settings{'use_cn'} = $nl_flags->{'use_cn'};
$settings{'use_cndv'} = $nl_flags->{'use_cndv'};
$settings{'use_crop'} = $nl_flags->{'use_crop'};
$settings{'cnfireson'} = $nl_flags->{'cnfireson'};
} else {
$settings{'bgc'} = $nl_flags->{'bgc_mode'};
}
# Loop over the variables specified in the use case.
# Add each one to the namelist.
my @vars = $uc_defaults->get_variable_names();
my $nl_usecase = Build::Namelist->new();
foreach my $var (@vars) {
my $val = $uc_defaults->get_value($var, \%settings );
if ( defined($val) ) {
$log->message("CLM adding use_case $opts->{'use_case'} defaults for var '$var' with val '$val'");
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl_usecase, $var, 'val'=>$val);
}
}
# Go through all variables and expand any XML env settings in them
expand_xml_variables_in_namelist( $nl_usecase, $envxml_ref );
# Merge input values into namelist. Previously specified values have higher precedence
# and are not overwritten.
$nl->merge_nl($nl_usecase);
}
}
#-------------------------------------------------------------------------------
sub process_namelist_commandline_clm_start_type {
# Set the start_type according to the command line clm_start_type option
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
# Run type for driver namelist - note that arb_ic implies that the run is startup
my $var = "start_type";
if ($nl_flags->{'clm_start_type'} eq "'cold'" || $nl_flags->{'clm_start_type'} eq "'arb_ic'") {
# Add default is used here, but the value is explicitly set
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, 'val'=>'startup' );
} else {
# Add default is used here, but the value is explicitly set
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, 'val'=>$nl_flags->{'clm_start_type'} );
}
}
#-------------------------------------------------------------------------------
sub process_namelist_inline_logic {
# Use the namelist default object to add default values for required
# namelist variables that have not been previously set.
my ($opts, $nl_flags, $definition, $defaults, $nl, $cfg, $envxml_ref, $physv) = @_;
##############################
# namelist group: clm_inparm #
##############################
setup_logic_site_specific($nl_flags, $definition, $nl, $physv);
setup_logic_lnd_frac($opts, $nl_flags, $definition, $defaults, $nl, $envxml_ref);
setup_logic_co2_type($opts, $nl_flags, $definition, $defaults, $nl);
setup_logic_irrigate($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_start_type($opts, $nl_flags, $nl);
setup_logic_delta_time($opts, $nl_flags, $definition, $defaults, $nl);
setup_logic_decomp_performance($opts, $nl_flags, $definition, $defaults, $nl);
setup_logic_snow($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_glacier($opts, $nl_flags, $definition, $defaults, $nl, $envxml_ref, $physv);
setup_logic_dynamic_plant_nitrogen_alloc($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_luna($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_hydrstress($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_dynamic_roots($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_params_file($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_create_crop_landunit($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_subgrid($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_fertilizer($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_grainproduct($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_soilstate($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_demand($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_surface_dataset($opts, $nl_flags, $definition, $defaults, $nl, $physv);
if ( remove_leading_and_trailing_quotes($nl_flags->{'clm_start_type'}) ne "branch" ) {
setup_logic_initial_conditions($opts, $nl_flags, $definition, $defaults, $nl, $physv);
}
setup_logic_dynamic_subgrid($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_spinup($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_supplemental_nitrogen($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_snowpack($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_fates($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#########################################
# namelist group: atm2lnd_inparm
#########################################
setup_logic_atm_forcing($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#########################################
# namelist group: lnd2atm_inparm
#########################################
setup_logic_lnd2atm($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#########################################
# namelist group: clm_humanindex_inparm #
#########################################
setup_logic_humanindex($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#################################
# namelist group: cnfire_inparm #
#################################
setup_logic_cnfire($opts, $nl_flags, $definition, $defaults, $nl, $physv);
######################################
# namelist group: cnprecision_inparm #
######################################
setup_logic_cnprec($opts, $nl_flags, $definition, $defaults, $nl, $physv);
###############################
# namelist group: clmu_inparm #
###############################
setup_logic_urban($opts, $nl_flags, $definition, $defaults, $nl, $physv);
###############################
# namelist group: crop #
###############################
setup_logic_crop($opts, $nl_flags, $definition, $defaults, $nl, $physv);
###############################
# namelist group: ch4par_in #
###############################
setup_logic_methane($opts, $nl_flags, $definition, $defaults, $nl);
setup_logic_c_isotope($opts, $nl_flags, $definition, $defaults, $nl, $physv);
###############################
# namelist group: ndepdyn_nml #
###############################
setup_logic_nitrogen_deposition($opts, $nl_flags, $definition, $defaults, $nl, $physv);
##########################################
# namelist group: soil_moisture_streams #
##########################################
setup_logic_soilm_streams($opts, $nl_flags, $definition, $defaults, $nl, $physv);
##################################
# namelist group: cnmresp_inparm #
##################################
setup_logic_cnmresp($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#################################
# namelist group: nitrif_inparm #
#################################
setup_logic_nitrif_params( $nl_flags, $definition, $defaults, $nl );
####################################
# namelist group: photosyns_inparm #
####################################
setup_logic_photosyns($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#################################
# namelist group: popd_streams #
#################################
setup_logic_popd_streams($opts, $nl_flags, $definition, $defaults, $nl, $physv);
####################################
# namelist group: urbantv_streams #
####################################
setup_logic_urbantv_streams($opts, $nl_flags, $definition, $defaults, $nl, $physv);
##################################
# namelist group: light_streams #
##################################
setup_logic_lightning_streams($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#################################
# namelist group: drydep_inparm #
#################################
setup_logic_dry_deposition($opts, $nl_flags, $definition, $defaults, $nl);
#################################
# namelist group: fire_emis_nl #
#################################
setup_logic_fire_emis($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#################################
# namelist group: megan_emis_nl #
#################################
setup_logic_megan($opts, $nl_flags, $definition, $defaults, $nl);
##################################
# namelist group: lai_streams #
##################################
setup_logic_lai_streams($opts, $nl_flags, $definition, $defaults, $nl, $physv);
##################################
# namelist group: bgc_shared
##################################
setup_logic_bgc_shared($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#############################################
# namelist group: soilwater_movement_inparm #
#############################################
setup_logic_soilwater_movement($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#############################################
# namelist group: rooting_profile_inparm #
#############################################
setup_logic_rooting_profile($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#############################################
# namelist group: friction_velocity #
#############################################
setup_logic_friction_vel($opts, $nl_flags, $definition, $defaults, $nl, $physv);
################################################
# namelist group: century_soilbgcdecompcascade #
################################################
setup_logic_century_soilbgcdecompcascade($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#############################
# namelist group: cngeneral #
#############################
setup_logic_cngeneral($opts, $nl_flags, $definition, $defaults, $nl, $physv);
####################################
# namelist group: cnvegcarbonstate #
####################################
setup_logic_cnvegcarbonstate($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#############################################
# namelist group: soil_resis_inparm #
#############################################
setup_logic_soil_resis($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#############################################
# namelist group: canopyfluxes_inparm #
#############################################
setup_logic_canopyfluxes($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#############################################
# namelist group: canopyhydrology_inparm #
#############################################
setup_logic_canopyhydrology($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#####################################
# namelist group: clm_canopy_inparm #
#####################################
setup_logic_canopy($opts, $nl_flags, $definition, $defaults, $nl, $physv);
########################################
# namelist group: soilhydrology_inparm #
########################################
setup_logic_hydrology_params($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#####################################
# namelist group: irrigation_inparm #
#####################################
setup_logic_irrigation_parameters($opts, $nl_flags, $definition, $defaults, $nl, $physv);
#######################################################################
# namelist groups: clm_hydrology1_inparm and clm_soilhydrology_inparm #
#######################################################################
setup_logic_hydrology_switches($nl, $physv);
#########################################
# namelist group: clm_initinterp_inparm #
#########################################
setup_logic_initinterp($opts, $nl_flags, $definition, $defaults, $nl, $physv);
}
#-------------------------------------------------------------------------------
sub setup_logic_site_specific {
# site specific requirements
my ($nl_flags, $definition, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
# res check prevents polluting the namelist with an unnecessary
# false variable for every run
if ($nl_flags->{'res'} eq "1x1_vancouverCAN") {
my $var = "use_vancouver";
my $val = ".true.";
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val);
}
}
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
# res check prevents polluting the namelist with an unnecessary
# false variable for every run
if ($nl_flags->{'res'} eq "1x1_mexicocityMEX") {
my $var = "use_mexicocity";
my $val = ".true.";
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $val);
}
}
if ( $physv->as_long() >= $physv->as_long("clm4_5") && $nl_flags->{'res'} eq "1x1_smallvilleIA") {
if (! &value_is_true($nl_flags->{'use_cn'}) || ! &value_is_true($nl_flags->{'use_crop'})) {
$log->fatal_error("1x1_smallvilleIA grids must use a compset with CN and CROP turned on.");
}
}
if ( $physv->as_long() >= $physv->as_long("clm4_5") && $nl_flags->{'res'} eq "1x1_numaIA") {
if (! &value_is_true($nl_flags->{'use_cn'}) || ! &value_is_true($nl_flags->{'use_crop'})) {
$log->fatal_error("1x1_numaIA grids must use a compset with CN and CROP turned on.");
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_lnd_tuning {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $var = "lnd_tuning_mode";
if ( $opts->{$var} eq "default" ) {
my %settings;
$settings{'phys'} = $nl_flags->{'phys'};
$nl_flags->{$var} = $defaults->get_value($var, \%settings );
} else {
$nl_flags->{$var} = $opts->{$var};
}
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, quote_string( $nl_flags->{$var} ) );
if ( ! $definition->is_valid_value( $var, quote_string( $nl_flags->{$var}) ) ) {
my @valid_values = $definition->get_valid_values( $var );
$log->fatal_error("$var has a value (".$nl_flags->{$var}.") that is NOT valid. Valid values are: @valid_values");
}
$log->verbose_message("Using $nl_flags->{$var} for lnd_tuning_mode");
my $phys = $physv->as_string();
if ( $nl_flags->{$var} !~ /^${phys}_/ ) {
$log->fatal_error("First part of lnd_tuning_mode MUST match the CLM version you are using.");
}
}
#-------------------------------------------------------------------------------
sub setup_logic_lnd_frac {
my ($opts, $nl_flags, $definition, $defaults, $nl, $envxml_ref) = @_;
my $var = "lnd_frac";
if ( defined($opts->{$var}) ) {
if ( defined($nl->get_value('fatmlndfrc')) ) {
$log->fatal_error("Can NOT set both -lnd_frac option (set via LND_DOMAIN_PATH/LND_DOMAIN_FILE " .
"env variables) AND fatmlndfrac on namelist");
}
my $lnd_frac = SetupTools::expand_xml_var( $opts->{$var}, $envxml_ref);
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fatmlndfrc','val'=>$lnd_frac );
}
# Get the fraction file
if (defined $nl->get_value('fatmlndfrc')) {
# do nothing - use value provided by config_grid.xml and clm.cpl7.template
} else {
$log->fatal_error("fatmlndfrc was NOT sent into CLM build-namelist.");
}
}
#-------------------------------------------------------------------------------
sub setup_logic_co2_type {
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
my $var = "co2_type";
if ( defined($opts->{$var}) ) {
if ( ! defined($nl->get_value($var)) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'co2_type','val'=>"$opts->{'co2_type'}");
} else {
$log->fatal_error("co2_type set on namelist as well as -co2_type option.");
}
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'co2_type');
if ( $nl->get_value('co2_type') =~ /constant/ ) {
my $var = 'co2_ppmv';
if ( defined($opts->{$var}) ) {
if ( $opts->{$var} <= 0.0 ) {
$log->fatal_error("co2_ppmv can NOT be less than or equal to zero.");
}
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, $opts->{$var});
} else {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, 'sim_year'=>$nl_flags->{'sim_year'},
'ssp_rcp'=>$nl_flags->{'ssp_rcp'} );
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_irrigate {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'irrigate',
'use_crop'=>$nl_flags->{'use_crop'}, 'use_cndv'=>$nl_flags->{'use_cndv'},
'sim_year'=>$nl_flags->{'sim_year'}, 'sim_year_range'=>$nl_flags->{'sim_year_range'}, );
if ( &value_is_true($nl->get_value('irrigate') ) ) {
$nl_flags->{'irrigate'} = ".true."
} else {
$nl_flags->{'irrigate'} = ".false."
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_start_type {
my ($opts, $nl_flags, $nl) = @_;
my $var = "start_type";
my $drv_start_type = $nl->get_value($var);
my $my_start_type = $nl_flags->{'clm_start_type'};
my $nsrest = $nl->get_value('override_nsrest');
if ( defined($nsrest) ) {
if ( $nsrest == 0 ) { $my_start_type = "startup"; }
if ( $nsrest == 1 ) { $my_start_type = "continue"; }
if ( $nsrest == 3 ) { $my_start_type = "branch"; }
if ( "$my_start_type" eq "$drv_start_type" ) {
$log->fatal_error("no need to set override_nsrest to same as start_type.");
}
if ( "$drv_start_type" !~ /startup/ ) {
$log->fatal_error("can NOT set override_nsrest if driver is NOT a startup type.");
}
}
if ( $my_start_type =~ /branch/ ) {
if (not defined $nl->get_value('nrevsn')) {
$log->fatal_error("nrevsn is required for a branch type.");
}
if (defined $nl->get_value('use_init_interp')) {
if ( &value_is_true($nl->get_value('use_init_interp') ) ) {
# Always print this warning, but don't stop if it happens
print "\nWARNING: use_init_interp will NOT happen for a branch case.\n\n";
}
}
} else {
if (defined $nl->get_value('nrevsn')) {
$log->fatal_error("nrevsn should ONLY be set for a branch type.");
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_delta_time {
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
if ( defined($opts->{'l_ncpl'}) ) {
my $l_ncpl = $opts->{'l_ncpl'};
if ( $l_ncpl <= 0 ) {
$log->fatal_error("bad value for -l_ncpl option.");
}
my $val = ( 3600 * 24 ) / $l_ncpl;
my $dtime = $nl->get_value('dtime');
if ( ! defined($dtime) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'dtime', 'val'=>$val);
} elsif ( $dtime ne $val ) {
$log->fatal_error("can NOT set both -l_ncpl option (via LND_NCPL env variable) AND dtime namelist variable.");
}
} else {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'dtime', 'hgrid'=>$nl_flags->{'res'});
}
}
#-------------------------------------------------------------------------------
sub setup_logic_decomp_performance {
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
# Set the number of segments per clump
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'nsegspc', 'hgrid'=>$nl_flags->{'res'});
}
#-------------------------------------------------------------------------------
sub setup_logic_snow {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'snowveg_flag', 'phys'=>$nl_flags->{'phys'} );
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fsnowoptics' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fsnowaging' );
}
#-------------------------------------------------------------------------------
sub setup_logic_glacier {
#
# Glacier multiple elevation class options
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $envxml_ref, $physv) = @_;
my $clm_upvar = "GLC_TWO_WAY_COUPLING";
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
# glc_do_dynglacier is set via GLC_TWO_WAY_COUPLING; it cannot be set via
# user_nl_clm (this is because we might eventually want the coupler and glc
# to also respond to GLC_TWO_WAY_COUPLING, by not bothering to send / map
# these fields - so we want to ensure that CLM is truly listening to this
# shared xml variable and not overriding it)
my $var = "glc_do_dynglacier";
my $val = logical_to_fortran($envxml_ref->{$clm_upvar});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, 'val'=>$val);
if (lc($nl->get_value($var)) ne lc($val)) {
$log->fatal_error("glc_do_dynglacier can only be set via the env variable $clm_upvar: it can NOT be set in user_nl_clm");
}
} else {
# Otherwise if CLM4.0 physics and GLC_TWO_WAY_COUPLING is TRUE -- trigger an error
if ( &value_is_true(logical_to_fortran($envxml_ref->{$clm_upvar})) ) {
$log->fatal_error( "clm4_0 physics are being used, but $clm_upvar variable is set to true. $clm_upvar can ONLY be set for physics after clm4_5" );
}
}
my $var = "maxpatch_glcmec";
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, 'val'=>$nl_flags->{'glc_nec'} );
my $val = $nl->get_value($var);
if ( $val != $nl_flags->{'glc_nec'} ) {
$log->fatal_error("$var set to $val does NOT agree with -glc_nec argument of $nl_flags->{'glc_nec'} (set with GLC_NEC env variable)");
}
if ( $physv->as_long >= $physv->as_long("clm4_5") ) {
if ( $nl_flags->{'glc_nec'} < 1 ) {
$log->fatal_error("For clm4_5 and later, GLC_NEC must be at least 1.");
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'glc_snow_persistence_max_days');
} else {
# clm4_0
if ( $nl_flags->{'glc_nec'} > 0 ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'flndtopo' , 'hgrid'=>$nl_flags->{'res'}, 'mask'=>$nl_flags->{'mask'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fglcmask' , 'hgrid'=>$nl_flags->{'res'});
} else {
# glc_nec == 0
# Error checking for glacier multiple elevation class options when glc_mec off
# Make sure various glc_mec-specific logicals are not true, and fglcmask is not set
my $glc_dyntopo= $nl->get_value('glc_dyntopo');
if ( defined($glc_dyntopo) ) {
if ( &value_is_true($glc_dyntopo) ) {
$log->fatal_error("glc_dyntopo is true, but glc_nec is equal to zero");
}
}
my $fglcmask = $nl->get_value('fglcmask');
if ( defined($fglcmask) ) {
$log->fatal_error("fglcmask is set, but glc_nec is equal to zero");
}
}
}
# Dependence of albice on glc_nec has gone away starting in CLM4_5. Thus, we
# can remove glc_nec from the following call once we ditch CLM4_0.
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'albice', 'glc_nec'=>$nl_flags->{'glc_nec'});
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'glacier_region_behavior');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'glacier_region_melt_behavior');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'glacier_region_ice_runoff_behavior');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'glacier_region_rain_to_snow_behavior');
}
}
#-------------------------------------------------------------------------------
sub setup_logic_params_file {
# get param data. For 4_0, pft-physiology, for 4_5 old
# pft-physiology was used but now now includes CN and BGC century
# parameters.
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'paramfile',
'phys'=>$nl_flags->{'phys'},
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
} else {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fpftcon');
}
}
#-------------------------------------------------------------------------------
sub setup_logic_create_crop_landunit {
# Create crop land unit
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $var = 'create_crop_landunit';
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
if ( $nl_flags->{'crop'} eq "on" ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var );
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, 'irrig'=>$nl_flags->{'irrig'} );
} else {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var,
'use_fates'=>$nl_flags->{'use_fates'} );
if ( &value_is_true($nl_flags->{'use_fates'}) && &value_is_true($nl->get_value($var)) ) {
$log->fatal_error( "$var is true and yet use_fates is being set, which contradicts that (use_fates requires $var to be .false." );
}
if ( (! &value_is_true($nl_flags->{'use_fates'})) && (! &value_is_true($nl->get_value($var))) ) {
$log->fatal_error( "$var is false which is ONLY allowed when FATES is being used" );
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_subgrid {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $var = 'run_zero_weight_urban';
if ($physv->as_long() >= $physv->as_long("clm4_5")) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var);
}
}
#-------------------------------------------------------------------------------
sub setup_logic_cnfire {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my @fire_consts = ( "rh_low", "rh_hgh", "bt_min", "bt_max", "cli_scale", "boreal_peatfire_c", "non_boreal_peatfire_c",
"pot_hmn_ign_counts_alpha", "cropfire_a1", "occur_hi_gdp_tree", "lfuel", "ufuel", "cmb_cmplt_fact" );
if ( $physv->as_long() >= $physv->as_long("clm4_5") && &value_is_true($nl->get_value('use_cn')) ) {
foreach my $item ( @fire_consts ) {
if ( ! &value_is_true($nl_flags->{'cnfireson'} ) ) {
if ( defined($nl->get_value($item)) ) {
$log->fatal_error( "fire_method is no_fire and yet $item is being set, which contradicts that" );
}
} else {
my $fire_method = remove_leading_and_trailing_quotes( $nl->get_value('fire_method') );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $item,
'lnd_tuning_mode'=>$nl_flags->{'lnd_tuning_mode'},
'fire_method'=>$fire_method );
}
}
} elsif ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
foreach my $item ( @fire_consts ) {
if ( defined($nl->get_value($item)) ) {
$log->fatal_error( "CN is off which implies that cnfire is off and yet a fire constant ($item) is being set, which contradicts that" );
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_cnprec {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") && &value_is_true($nl_flags->{'use_cn'}) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults,
$nl, 'ncrit', 'use_cn'=>$nl_flags->{'use_cn'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults,
$nl, 'cnegcrit', 'use_cn'=>$nl_flags->{'use_cn'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults,
$nl, 'nnegcrit', 'use_cn'=>$nl_flags->{'use_cn'});
}
}
#-------------------------------------------------------------------------------
sub setup_logic_humanindex {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'calc_human_stress_indices');
} else {
if ( defined($nl->get_value('calc_human_stress_indices')) ) {
$log->fatal_error( "calc_human_stress_indices can NOT be set, for physics versions less than clm4_5" );
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_urban {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'building_temp_method');
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'urban_hac');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'urban_traffic');
}
#-------------------------------------------------------------------------------
sub setup_logic_crop {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
if ( &value_is_true($nl->get_value('use_crop')) ) {
my $maptype = 'baset_mapping';
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $maptype,
'use_crop'=>$nl->get_value('use_crop') );
my $baset_mapping = remove_leading_and_trailing_quotes( $nl->get_value($maptype) );
if ( $baset_mapping eq "varytropicsbylat" ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, "baset_latvary_slope",
'use_crop'=>$nl->get_value('use_crop'), $maptype=>$baset_mapping );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, "baset_latvary_intercept",
'use_crop'=>$nl->get_value('use_crop'), $maptype=>$baset_mapping );
} else {
error_if_set( $nl, "Can only be set if $maptype == varytropicsbylat", "baset_latvary_slope", "baset_latvary_intercept" );
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, "initial_seed_at_planting",
'use_crop'=>$nl->get_value('use_crop') );
} else {
error_if_set( $nl, "Can NOT be set without crop on", "baset_mapping", "baset_latvary_slope", "baset_latvary_intercept" );
}
}
}
#-------------------------------------------------------------------------------
sub error_if_set {
# do a fatal_error and exit if any of the input variable names are set
my ($nl, $error, @list) = @_;
foreach my $var ( @list ) {
if ( defined($nl->get_value($var)) ) {
$log->fatal_error( "$var $error" );
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_soilstate {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'organic_frac_squared' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'soil_layerstruct' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_bedrock',
'use_fates'=>$nl_flags->{'use_fates'}, 'vichydro'=>$nl_flags->{'vichydro'} );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_demand {
#
# Deal with options that the user has said are required...
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my %settings;
$settings{'hgrid'} = $nl_flags->{'res'};
$settings{'sim_year'} = $nl_flags->{'sim_year'};
$settings{'sim_year_range'} = $nl_flags->{'sim_year_range'};
$settings{'mask'} = $nl_flags->{'mask'};
$settings{'crop'} = $nl_flags->{'crop'};
$settings{'ssp_rcp'} = $nl_flags->{'ssp_rcp'};
$settings{'glc_nec'} = $nl_flags->{'glc_nec'};
if ( $physv->as_long() >= $physv->as_long("clm4_5")) {
# necessary for demand to be set correctly (flanduse_timeseries requires
# use_crop, maybe other options require other flags?)!
$settings{'irrigate'} = $nl_flags->{'irrigate'};
$settings{'use_cn'} = $nl_flags->{'use_cn'};
$settings{'use_cndv'} = $nl_flags->{'use_cndv'};
$settings{'use_lch4'} = $nl_flags->{'use_lch4'};
$settings{'use_nitrif_denitrif'} = $nl_flags->{'use_nitrif_denitrif'};
$settings{'use_vertsoilc'} = $nl_flags->{'use_vertsoilc'};
$settings{'use_century_decomp'} = $nl_flags->{'use_century_decomp'};
$settings{'use_crop'} = $nl_flags->{'use_crop'};
} elsif ( $physv->as_long() == $physv->as_long("clm4_0")) {
$settings{'irrig'} = $nl_flags->{'irrig'};
}
my $demand = $nl->get_value('clm_demand');
if (defined($demand)) {
$demand =~ s/\'//g; # Remove quotes
if ( $demand =~ /.+/ ) {
$opts->{'clm_demand'} .= ",$demand";
}
}
$demand = $defaults->get_value('clm_demand', \%settings);
if (defined($demand)) {
$demand =~ s/\'//g; # Remove quotes
if ( $demand =~ /.+/ ) {
$opts->{'clm_demand'} .= ",$demand";
}
}
my @demandlist = split( ",", $opts->{'clm_demand'} );
foreach my $item ( @demandlist ) {
if ( $item eq "null" ) {
next;
}
if ( $item eq "finidat" ) {
$log->fatal_error( "Do NOT put findat in the clm_demand list, set the clm_start_type=startup so initial conditions are required");
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $item, %settings );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_surface_dataset {
#
# Get surface dataset after flanduse_timeseries so that we can get surface data
# consistent with it
# MUST BE AFTER: setup_logic_demand which is where flanduse_timeseries is set
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
$nl_flags->{'flanduse_timeseries'} = "null";
my $flanduse_timeseries = $nl->get_value('flanduse_timeseries');
if (defined($flanduse_timeseries)) {
$flanduse_timeseries =~ s!(.*)/!!;
$flanduse_timeseries =~ s/\'//;
$flanduse_timeseries =~ s/\"//;
if ( $flanduse_timeseries ne "" ) {
$nl_flags->{'flanduse_timeseries'} = $flanduse_timeseries;
}
}
$flanduse_timeseries = $nl_flags->{'flanduse_timeseries'};
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
if ($flanduse_timeseries ne "null" && $nl_flags->{'bgc_mode'} eq "cndv" ) {
$log->fatal_error( "dynamic PFT's (setting flanduse_timeseries) are incompatible with dynamic vegetation ('-bgc cndv' in CLM_CONFIG_OPTS)." );
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fsurdat',
'hgrid'=>$nl_flags->{'res'}, 'ssp_rcp'=>$nl_flags->{'ssp_rcp'},
'sim_year'=>$nl_flags->{'sim_year'}, 'irrig'=>$nl_flags->{'irrig'},
'crop'=>$nl_flags->{'crop'}, 'glc_nec'=>$nl_flags->{'glc_nec'});
} else{
if ($flanduse_timeseries ne "null" && &value_is_true($nl_flags->{'use_cndv'}) ) {
$log->fatal_error( "dynamic PFT's (setting flanduse_timeseries) are incompatible with dynamic vegetation (use_cndv=.true)." );
}
if ($flanduse_timeseries ne "null" && &value_is_true($nl_flags->{'use_fates'}) ) {
$log->fatal_error( "dynamic PFT's (setting flanduse_timeseries) are incompatible with ecosystem dynamics (use_fates=.true)." );
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fsurdat',
'hgrid'=>$nl_flags->{'res'}, 'ssp_rcp'=>$nl_flags->{'ssp_rcp'},
'sim_year'=>$nl_flags->{'sim_year'}, 'irrigate'=>$nl_flags->{'irrigate'},
'use_crop'=>$nl_flags->{'use_crop'}, 'glc_nec'=>$nl_flags->{'glc_nec'});
}
}
#-------------------------------------------------------------------------------
sub setup_logic_initial_conditions {
# Initial conditions
# The initial date is an attribute in the defaults file which should be matched unless
# the user explicitly requests to ignore the initial date via the -ignore_ic_date option,
# or just ignore the year of the initial date via the -ignore_ic_year option.
#
# MUST BE AFTER: setup_logic_demand which is where flanduse_timeseries is set
# AFTER: setup_logic_irrigate which is where irrig (or irrigate) is set
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $var = "finidat";
my $finidat = $nl->get_value($var);
if ( $nl_flags->{'clm_start_type'} =~ /cold/ ) {
if (defined $finidat ) {
$log->warning("setting $var (either explicitly in your user_nl_clm or by doing a hybrid or branch RUN_TYPE)\n is incomptable with using a cold start" .
" (by setting CLM_FORCE_COLDSTART=on)." );
$log->warning("Overridding input $var file with one specifying that this is a cold start from arbitrary initial conditions." );
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, "' '" );
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl,
$var, 'val'=>"' '", 'no_abspath'=>1);
$finidat = $nl->get_value($var);
} elsif ( defined $finidat ) {
if ( string_is_undef_or_empty($finidat) ) {
print "You are setting $var to blank which signals arbitrary initial conditions.\n";
print "But, CLM_FORCE_COLDSTART is off which is a contradiction. For arbitrary initial conditions just use the CLM_FORCE_COLDSTART option\n";
$log->fatal_error("To do a cold-start set ./xmlchange CLM_FORCE_COLDSTART=on, and remove the setting of $var in the user_nl_clm file");
}
}
my $useinitvar = "use_init_interp";
if (not defined $finidat ) {
my $ic_date = $nl->get_value('start_ymd');
my $st_year = int( $ic_date / 10000);
my $nofail = 1;
my %settings;
$settings{'hgrid'} = $nl_flags->{'res'};
$settings{'phys'} = $physv->as_string();
$settings{'nofail'} = $nofail;
my $fsurdat = $nl->get_value('fsurdat');
$fsurdat =~ s!(.*)/!!;
$settings{'fsurdat'} = $fsurdat;
#
# If not transient use sim_year, otherwise use date
#
if (string_is_undef_or_empty($nl->get_value('flanduse_timeseries'))) {
$settings{'sim_year'} = $nl_flags->{'sim_year'};
$opts->{'ignore_ic_year'} = 1;
} else {
delete( $settings{'sim_year'} );
}
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
$settings{'bgc'} = $nl_flags->{'bgc_mode'};
foreach my $item ( "mask", "maxpft", "irrig", "glc_nec", "crop", "lnd_tuning_mode" ) {
$settings{$item} = $nl_flags->{$item};
}
} else {
foreach my $item ( "mask", "maxpft", "irrigate", "glc_nec", "use_crop", "use_cn", "use_cndv",
"use_nitrif_denitrif", "use_vertsoilc", "use_century_decomp", "use_fates",
"lnd_tuning_mode"
) {
$settings{$item} = $nl_flags->{$item};
}
}
if ($opts->{'ignore_ic_date'}) {
if ( &value_is_true($nl_flags->{'use_crop'}) ) {
$log->warning("using ignore_ic_date is incompatable with crop! If you choose to ignore this error, " .
"the counters since planting for crops will be messed up. \nSo you should ignore at " .
"least the first season for crops. And since it will impact the 20 year means, ideally the " .
"first 20 years should be ignored.");
}
} elsif ($opts->{'ignore_ic_year'}) {
$settings{'ic_md'} = $ic_date;
} else {
$settings{'ic_ymd'} = $ic_date;
}
my $try = 0;
my $done = 2;
my $use_init_interp_default = $nl->get_value($useinitvar);
$settings{$useinitvar} = $use_init_interp_default;
if ( string_is_undef_or_empty( $use_init_interp_default ) ) {
$use_init_interp_default = $defaults->get_value($useinitvar, \%settings);
$settings{$useinitvar} = ".false.";
}
do {
$try++;
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, %settings );
# If couldn't find a matching finidat file, check if can turn on interpolation and try to find one again
$finidat = $nl->get_value($var);
if ( (not defined $finidat ) && ($physv->as_long() >= $physv->as_long("clm4_5")) ) {
# Delete any date settings, except for crop
delete( $settings{'ic_ymd'} );
delete( $settings{'ic_md'} );
#if ( &value_is_true($nl_flags->{'use_crop'}) ) {
#$settings{'ic_md'} = $ic_date;
#}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, "init_interp_sim_years" );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, "init_interp_how_close" );
my $close = $nl->get_value("init_interp_how_close");
foreach my $sim_yr ( split( /,/, $nl->get_value("init_interp_sim_years") )) {
my $how_close = undef;
if ( $nl_flags->{'sim_year'} eq "PtVg" ) {
$how_close = abs(1850 - $sim_yr);
} else {
$how_close = abs($st_year - $sim_yr);
}
if ( ($how_close < $nl->get_value("init_interp_how_close")) && ($how_close < $close) ) {
$close = $how_close;
$settings{'sim_year'} = $sim_yr;
}
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $useinitvar,
'use_cndv'=>$nl_flags->{'use_cndv'}, 'phys'=>$physv->as_string(),
'sim_year'=>$settings{'sim_year'}, 'nofail'=>1, 'lnd_tuning_mode'=>$nl_flags->{'lnd_tuning_mode'},
'use_fates'=>$nl_flags->{'use_fates'} );
$settings{$useinitvar} = $nl->get_value($useinitvar);
if ( $try > 1 ) {
my $group = $definition->get_group_name($useinitvar);
$nl->set_variable_value($group, $useinitvar, $use_init_interp_default );
}
if ( &value_is_true($nl->get_value($useinitvar) ) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, "init_interp_attributes",
'sim_year'=>$settings{'sim_year'}, 'use_cndv'=>$nl_flags->{'use_cndv'},
'glc_nec'=>$nl_flags->{'glc_nec'}, 'use_fates'=>$nl_flags->{'use_fates'},
'use_cn'=>$nl_flags->{'use_cn'}, 'lnd_tuning_mode'=>$nl_flags->{'lnd_tuning_mode'},'nofail'=>1 );
my $attributes_string = remove_leading_and_trailing_quotes($nl->get_value("init_interp_attributes"));
foreach my $pair ( split( /\s/, $attributes_string) ) {
if ( $pair =~ /^([a-z_]+)=([a-z._0-9]+)$/ ) {
$settings{$1} = $2;
} else {
$log->fatal_error("Problem interpreting init_interp_attributes");
}
}
} else {
if ( $nl_flags->{'clm_start_type'} =~ /startup/ ) {
$log->fatal_error("clm_start_type is startup so an initial conditions ($var) file is required, but can't find one without $useinitvar being set to true");
}
$try = $done;
}
} else {
$try = $done
}
} while( ($try < $done) && (not defined $finidat ) );
if ( not defined $finidat ) {
my $group = $definition->get_group_name($var);
$nl->set_variable_value($group, $var, "' '" );
}
}
$finidat = $nl->get_value($var);
if ( &value_is_true($nl->get_value($useinitvar) ) && string_is_undef_or_empty($finidat) ) {
$log->fatal_error("$useinitvar is set BUT $var is NOT, need to set both" );
}
} # end initial conditions
#-------------------------------------------------------------------------------
sub setup_logic_dynamic_subgrid {
#
# Options controlling which parts of flanduse_timeseries to use
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
setup_logic_do_transient_pfts($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_do_transient_crops($opts, $nl_flags, $definition, $defaults, $nl, $physv);
setup_logic_do_harvest($opts, $nl_flags, $definition, $defaults, $nl, $physv);
}
sub setup_logic_do_transient_pfts {
#
# Set do_transient_pfts default value, and perform error checking on do_transient_pfts
#
# Assumes the following are already set in the namelist (although it's okay
# for them to be unset if that will be their final state):
# - flanduse_timeseries
# - use_cndv
# - use_fates
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $var = 'do_transient_pfts';
if ($physv->as_long() >= $physv->as_long("clm4_5")) {
# Start by assuming a default value of '.true.'. Then check a number of
# conditions under which do_transient_pfts cannot be true. Under these
# conditions: (1) set default value to '.false.'; (2) make sure that the
# value is indeed false (e.g., that the user didn't try to set it to true).
my $default_val = ".true.";
# cannot_be_true will be set to a non-empty string in any case where
# do_transient_pfts should not be true; if it turns out that
# do_transient_pfts IS true in any of these cases, a fatal error will be
# generated
my $cannot_be_true = "";
if (string_is_undef_or_empty($nl->get_value('flanduse_timeseries'))) {
$cannot_be_true = "$var can only be set to true when running a transient case (flanduse_timeseries non-blank)";
}
elsif (&value_is_true($nl->get_value('use_cndv'))) {
$cannot_be_true = "$var cannot be combined with use_cndv";
}
elsif (&value_is_true($nl->get_value('use_fates'))) {
$cannot_be_true = "$var cannot be combined with use_fates";
}
if ($cannot_be_true) {
$default_val = ".false.";
}
if (!$cannot_be_true) {
# Note that, if the variable cannot be true, we don't call add_default
# - so that we don't clutter up the namelist with variables that don't
# matter for this case
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, val=>$default_val);
}
# Make sure the value is false when it needs to be false - i.e., that the
# user hasn't tried to set a true value at an inappropriate time.
if (&value_is_true($nl->get_value($var)) && $cannot_be_true) {
$log->fatal_error($cannot_be_true);
}
}
}
sub setup_logic_do_transient_crops {
#
# Set do_transient_crops default value, and perform error checking on do_transient_crops
#
# Assumes the following are already set in the namelist (although it's okay
# for them to be unset if that will be their final state):
# - flanduse_timeseries
# - use_crop
# - use_fates
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $var = 'do_transient_crops';
if ($physv->as_long() >= $physv->as_long("clm4_5")) {
# Start by assuming a default value of '.true.'. Then check a number of
# conditions under which do_transient_crops cannot be true. Under these
# conditions: (1) set default value to '.false.'; (2) make sure that the
# value is indeed false (e.g., that the user didn't try to set it to true).
my $default_val = ".true.";
# cannot_be_true will be set to a non-empty string in any case where
# do_transient_crops should not be true; if it turns out that
# do_transient_crops IS true in any of these cases, a fatal error will be
# generated
my $cannot_be_true = "";
if (string_is_undef_or_empty($nl->get_value('flanduse_timeseries'))) {
$cannot_be_true = "$var can only be set to true when running a transient case (flanduse_timeseries non-blank)";
}
elsif (&value_is_true($nl->get_value('use_fates'))) {
# In principle, use_fates should be compatible with
# do_transient_crops. However, this hasn't been tested, so to be safe,
# we are not allowing this combination for now.
$cannot_be_true = "$var has not been tested with ED, so for now these two options cannot be combined";
}
if ($cannot_be_true) {
$default_val = ".false.";
}
if (!$cannot_be_true) {
# Note that, if the variable cannot be true, we don't call add_default
# - so that we don't clutter up the namelist with variables that don't
# matter for this case
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, val=>$default_val);
}
# Make sure the value is false when it needs to be false - i.e., that the
# user hasn't tried to set a true value at an inappropriate time.
if (&value_is_true($nl->get_value($var)) && $cannot_be_true) {
$log->fatal_error($cannot_be_true);
}
my $dopft = "do_transient_pfts";
# Make sure the value agrees with the do_transient_pft flag
if ( ( &value_is_true($nl->get_value($var))) && (! &value_is_true($nl->get_value($dopft))) ||
(! &value_is_true($nl->get_value($var))) && ( &value_is_true($nl->get_value($dopft))) ) {
$log->fatal_error("$var and $dopft do NOT agree and need to");
}
}
}
sub setup_logic_do_harvest {
#
# Set do_harvest default value, and perform error checking on do_harvest
#
# Assumes the following are already set in the namelist (although it's okay
# for them to be unset if that will be their final state):
# - flanduse_timeseries
# - use_cn
# - use_fates
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $var = 'do_harvest';
if ($physv->as_long() >= $physv->as_long("clm4_5")) {
# Start by assuming a default value of '.true.'. Then check a number of
# conditions under which do_harvest cannot be true. Under these
# conditions: (1) set default value to '.false.'; (2) make sure that the
# value is indeed false (e.g., that the user didn't try to set it to true).
my $default_val = ".true.";
# cannot_be_true will be set to a non-empty string in any case where
# do_harvest should not be true; if it turns out that do_harvest IS true
# in any of these cases, a fatal error will be generated
my $cannot_be_true = "";
if (string_is_undef_or_empty($nl->get_value('flanduse_timeseries'))) {
$cannot_be_true = "$var can only be set to true when running a transient case (flanduse_timeseries non-blank)";
}
elsif (!&value_is_true($nl->get_value('use_cn'))) {
$cannot_be_true = "$var can only be set to true when running with CN (use_cn = true)";
}
elsif (&value_is_true($nl->get_value('use_fates'))) {
$cannot_be_true = "$var currently doesn't work with ED";
}
if ($cannot_be_true) {
$default_val = ".false.";
}
if (!$cannot_be_true) {
# Note that, if the variable cannot be true, we don't call add_default
# - so that we don't clutter up the namelist with variables that don't
# matter for this case
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, val=>$default_val);
}
# Make sure the value is false when it needs to be false - i.e., that the
# user hasn't tried to set a true value at an inappropriate time.
if (&value_is_true($nl->get_value($var)) && $cannot_be_true) {
$log->fatal_error($cannot_be_true);
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_spinup {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5")) {
if ( $nl_flags->{'bgc_mode'} eq "sp" && defined($nl->get_value('override_bgc_restart_mismatch_dump'))) {
$log->fatal_error("CN must be on if override_bgc_restart_mismatch_dump is set.");
}
}
if ( $nl_flags->{'clm_accelerated_spinup'} eq "on" ) {
foreach my $var ( "hist_nhtfrq", "hist_fincl1", "hist_empty_htapes", "hist_mfilt" ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl,
$var, use_cn=>$nl_flags->{'use_cn'}, use_fates=>$nl_flags->{'use_fates'},
use_cndv=>$nl_flags->{'use_cndv'} );
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_bgc_shared {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5")) {
if ( $nl_flags->{'bgc_mode'} ne "sp" ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'constrain_stress_deciduous_onset', 'phys'=>$physv->as_string() );
}
# FIXME(bja, 201606) the logic around fates / bgc_mode /
# use_century_decomp is confusing and messed up. This is a hack
# workaround.
if ( &value_is_true($nl_flags->{'use_century_decomp'}) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'decomp_depth_efolding', 'phys'=>$physv->as_string() );
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_supplemental_nitrogen {
#
# Supplemental Nitrogen for prognostic crop cases
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $nl_flags->{'bgc_mode'} ne "sp" && $nl_flags->{'bgc_mode'} ne "fates" && &value_is_true($nl_flags->{'use_crop'}) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl,
'suplnitro', 'use_cn'=>$nl_flags->{'use_cn'}, 'use_crop'=>$nl_flags->{'use_crop'});
}
#
# Error checking for suplnitro
#
my $suplnitro = $nl->get_value('suplnitro');
if ( defined($suplnitro) ) {
if ( $nl_flags->{'bgc_mode'} eq "sp" ) {
$log->fatal_error("supplemental Nitrogen (suplnitro) is set, but neither CN nor CNDV is active!");
}
if ( ! &value_is_true($nl_flags->{'use_crop'}) && $suplnitro =~ /PROG_CROP_ONLY/i ) {
$log->fatal_error("supplemental Nitrogen is set to run over prognostic crops, but prognostic crop is NOT active!");
}
if ( $suplnitro =~ /ALL/i ) {
if ( $physv->as_long() == $physv->as_long("clm4_0") && $nl_flags->{'spinup'} ne "normal" ) {
$log->fatal_error("There is no need to use a spinup mode when supplemental Nitrogen is on for all PFT's, as these modes spinup Nitrogen\n" .
"when spinup != normal you can NOT set supplemental Nitrogen (suplnitro) to ALL");
}
if ( $physv->as_long() >= $physv->as_long("clm4_5") && $nl_flags->{'bgc_spinup'} eq "on" ) {
$log->warning("There is no need to use a bgc_spinup mode when supplemental Nitrogen is on for all PFT's, as these modes spinup Nitrogen" );
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_hydrology_params {
#
# Logic for hydrology parameters
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5")) {
my $lower = $nl->get_value( 'lower_boundary_condition' );
my $var = "baseflow_scalar";
if ( $lower == 1 || $lower == 2 ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl,
$var, 'lower_boundary_condition' => $lower );
}
my $val = $nl->get_value( $var );
if ( defined($val) ) {
if ( $lower != 1 && $lower != 2 ) {
$log->fatal_error("baseflow_scalar is only used for lower_boundary_condition of flux or zero-flux");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_irrigation_parameters {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5")) {
my $var;
foreach $var ("irrig_min_lai", "irrig_start_time", "irrig_length",
"irrig_target_smp", "irrig_depth", "irrig_threshold_fraction",
"limit_irrigation_if_rof_enabled") {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var);
}
$var = "irrig_river_volume_threshold";
if ( &value_is_true($nl->get_value("limit_irrigation_if_rof_enabled")) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var);
} else {
if (defined($nl->get_value($var))) {
$log->fatal_error("$var can only be set if limit_irrigation_if_rof_enabled is true");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_nitrif_params {
#
# Logic for nitrification parameters
#
my ($nl_flags, $definition, $defaults, $nl) = @_;
if ( ! &value_is_true($nl_flags->{'use_nitrif_denitrif'}) ) {
my @vars = ( "k_nitr_max", "denitrif_respiration_coefficient", "denitrif_respiration_exponent",
"denitrif_nitrateconc_coefficient", "denitrif_nitrateconc_exponent" );
foreach my $var ( @vars ) {
if ( defined($nl->get_value( $var ) ) ) {
$log->fatal_error("$var is only used when use_nitrif_denitrif is turned on");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_hydrology_switches {
#
# Check on Switches for hydrology
#
my ($nl, $physv) = @_;
my $subgrid = $nl->get_value('subgridflag' );
my $origflag = $nl->get_value('origflag' );
my $h2osfcflag = $nl->get_value('h2osfcflag' );
if ( $origflag == 1 && $subgrid == 1 ) {
$log->fatal_error("if origflag is ON, subgridflag can NOT also be on!");
}
if ( $h2osfcflag == 1 && $subgrid != 1 ) {
$log->fatal_error("if h2osfcflag is ON, subgridflag can NOT be off!");
}
# These should NOT be set for CLM5.0 and beyond
if ( $physv->as_long() > $physv->as_long("clm4_5") ) {
foreach my $var ( "origflag", "h2osfcflag", "oldfflag" ) {
my $val = $nl->get_value($var);
if ( defined($val) ) {
$log->fatal_error( "ERROR:: $var=$val is deprecated and can only be used with CLM4.5" );
}
}
}
# Test bad configurations
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
my $lower = $nl->get_value( 'lower_boundary_condition' );
my $use_vic = $nl->get_value( 'use_vichydro' );
my $use_bed = $nl->get_value( 'use_bedrock' );
my $soilmtd = $nl->get_value( 'soilwater_movement_method' );
if ( defined($soilmtd) && defined($lower) && $soilmtd == 0 && $lower != 4 ) {
$log->fatal_error( "If soil water movement method is zeng-decker -- lower_boundary_condition can only be aquifer" );
}
if ( defined($soilmtd) && defined($lower) && $soilmtd == 1 && $lower == 4 ) {
$log->fatal_error( "If soil water movement method is adaptive -- lower_boundary_condition can NOT be aquifer" );
}
if ( defined($use_bed) && defined($lower) && (&value_is_true($use_bed)) && $lower != 2 ) {
$log->fatal_error( "If use_bedrock is on -- lower_boundary_condition can only be flux" );
}
if ( defined($use_vic) && defined($lower) && (&value_is_true($use_vic)) && $lower != 3 && $lower != 4) {
$log->fatal_error( "If use_vichydro is on -- lower_boundary_condition can only be table or aquifer" );
}
if ( defined($origflag) && defined($use_vic) && (&value_is_true($use_vic)) && $origflag == 1 ) {
$log->fatal_error( "If use_vichydro is on -- origflag can NOT be equal to 1" );
}
if ( defined($h2osfcflag) && defined($lower) && $h2osfcflag == 0 && $lower != 4 ) {
$log->fatal_error( "If h2osfcflag is 0 lower_boundary_condition can only be aquifer" );
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_methane {
#
# CH4 model if bgc=CN or CNDV
#
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
if ( &value_is_true($nl_flags->{'use_lch4'}) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'finundation_method',
'use_cn'=>$nl_flags->{'use_cn'}, 'use_fates'=>$nl_flags->{'use_fates'} );
#
# Get resolution to read streams file for
#
my $finundation_method = remove_leading_and_trailing_quotes($nl->get_value('finundation_method' ));
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'finundation_res',
'finundation_method'=>$finundation_method );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_fldfilename_ch4finundated',
'finundation_method'=>$finundation_method,
'finundation_res'=>$nl->get_value('finundation_res') );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_aereoxid_prog',
'use_cn'=>$nl_flags->{'use_cn'}, 'use_fates'=>$nl_flags->{'use_fates'} );
#
# Check if use_aereoxid_prog is set. If no, then read value of aereoxid from
# parameters file
#
my $use_aereoxid_prog = $nl->get_value('use_aereoxid_prog');
if ( defined($use_aereoxid_prog) && ! &value_is_true($use_aereoxid_prog) ) {
$log->warning("Using aereoxid value from parameters file." );
}
} else {
my @vars = $nl->get_variable_names('ch4par_in');
if ( $#vars >= 0 ) {
$log->fatal_error("ch4par_in namelist variables were set, but Methane model NOT defined in the configuration (use_lch4)");
}
}
#
# Ch4 namelist checking
#
if ( &value_is_true($nl_flags->{'use_lch4'}) ) {
my $allowlakeprod = $nl->get_value('allowlakeprod');
if ( ! defined($allowlakeprod) ||
(defined($allowlakeprod) && ! &value_is_true($allowlakeprod)) ) {
if ( defined($nl->get_value('lake_decomp_fact')) ) {
$log->fatal_error("lake_decomp_fact set without allowlakeprod=.true.");
}
}
my $anoxia = $nl->get_value('anoxia');
if ( ! defined($anoxia) ||
(defined($anoxia) && ! &value_is_true($anoxia)) ) {
if ( defined($nl->get_value('anoxia_wtsat')) ) {
$log->fatal_error("anoxia_wtsat set without anoxia=.true.");
}
}
my $pftspec_rootprof = $nl->get_value('pftspecific_rootingprofile');
if ( ! defined($pftspec_rootprof) ||
(defined($pftspec_rootprof) && &value_is_true($pftspec_rootprof) ) ) {
if ( defined($nl->get_value('rootprof_exp')) ) {
$log->fatal_error("rootprof_exp set without pftspecific_rootingprofile=.false.");
}
}
} else {
my @vars = ( "allowlakeprod", "anoxia", "anoxia_wtsat", "pftspecific_rootingprofile" );
foreach my $var ( @vars ) {
if ( defined($nl->get_value($var)) ) {
$log->fatal_error("$var set without methane model configuration on (use_lch4)");
}
}
}
} # end methane
#-------------------------------------------------------------------------------
sub setup_logic_dynamic_plant_nitrogen_alloc {
#
# dynamic plant nitrogen allocation model, bgc=bgc
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") &&
&value_is_true($nl_flags->{'use_cn'}) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_flexibleCN',
'phys'=>$physv->as_string(), 'use_cn'=>$nl_flags->{'use_cn'} );
$nl_flags->{'use_flexibleCN'} = $nl->get_value('use_flexibleCN');
if ( &value_is_true($nl_flags->{'use_flexibleCN'}) ) {
# TODO(bja, 2015-04) make this depend on > clm 5.0 and bgc mode at some point.
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'MM_Nuptake_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'downreg_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'plant_ndemand_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'substrate_term_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'nscalar_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'temp_scalar_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'CNratio_floating',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'reduce_dayl_factor',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'vcmax_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'CN_residual_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'CN_partition_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'CN_evergreen_phenology_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'carbon_resp_opt',
'use_flexibleCN'=>$nl_flags->{'use_flexibleCN'}, 'use_fun'=>$nl->get_value('use_fun') );
if ( $nl->get_value('carbon_resp_opt') == 1 && &value_is_true($nl->get_value('use_fun')) ) {
$log->fatal_error("carbon_resp_opt should NOT be set to 1 when FUN is also on");
}
}
} elsif ( $physv->as_long() >= $physv->as_long("clm4_5") && ! &value_is_true($nl_flags->{'use_cn'}) ) {
if ( &value_is_true($nl->get_value('use_flexibleCN')) ) {
$log->fatal_error("use_flexibleCN can ONLY be set if CN is on");
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_luna {
#
# LUNA model to calculate photosynthetic capacities based on environmental conditions
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_luna',
'phys'=>$physv->as_string(), 'use_cn'=>$nl_flags->{'use_cn'}, 'use_fates'=>$nl_flags->{'use_fates'},
'use_nitrif_denitrif'=>$nl_flags->{'use_nitrif_denitrif'} );
if ( &value_is_true( $nl_flags->{'use_cn'} ) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_nguardrail',
'use_cn'=>$nl_flags->{'use_cn'} );
}
$nl_flags->{'use_luna'} = $nl->get_value('use_luna');
my $vcmax_opt= $nl->get_value('vcmax_opt');
# lnc_opt only applies if luna is on or for vcmax_opt=3/4
if ( &value_is_true( $nl_flags->{'use_luna'} ) || $vcmax_opt == 3 || $vcmax_opt == 4 ) {
# lnc_opt can be set for both CN on and off
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'lnc_opt',
'use_cn'=>$nl_flags->{'use_cn'} );
}
if ( &value_is_true($nl->get_value('lnc_opt') ) && not &value_is_true( $nl_flags->{'use_cn'}) ) {
$log->fatal_error("Cannot turn lnc_opt to true when bgc=sp" );
}
my $var = "jmaxb1";
if ( $physv->as_long() >= $physv->as_long("clm5_0") && &value_is_true( $nl_flags->{'use_luna'} ) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var,
'use_luna'=>$nl_flags->{'use_luna'} );
}
my $val = $nl->get_value($var);
if ( $physv->as_long() >= $physv->as_long("clm4_5") && ! &value_is_true( $nl_flags->{'use_luna'} ) ) {
if ( defined($val) ) {
$log->fatal_error("Cannot set $var when use_luna is NOT on" );
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_hydrstress {
#
# Plant hydraulic stress model
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
# TODO(kwo, 2015-09) make this depend on > clm 5.0 at some point.
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_hydrstress',
'use_fates'=>$nl_flags->{'use_fates'} );
$nl_flags->{'use_hydrstress'} = $nl->get_value('use_hydrstress');
if ( &value_is_true( $nl_flags->{'use_fates'} ) && &value_is_true( $nl_flags->{'use_hydrstress'} ) ) {
$log->fatal_error("Cannot turn use_hydrstress on when use_fates is on" );
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_fertilizer {
#
# Flags to control fertilizer application
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_fertilizer',
'use_crop'=>$nl_flags->{'use_crop'} );
my $use_fert = $nl->get_value('use_fertilizer');
if ( (! &value_is_true($nl_flags->{'use_crop'})) && &value_is_true($use_fert) ) {
$log->fatal_error("use_ferilizer can NOT be on without prognostic crop\n" );
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_grainproduct {
#
# Flags to control 1-year grain product pool
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_grainproduct',
'use_crop'=>$nl_flags->{'use_crop'}, 'phys'=>$physv->as_string() );
if ( (! &value_is_true($nl_flags->{'use_crop'})) && &value_is_true($nl->get_value('use_grainproduct') ) ) {
$log->fatal_error("use_grainproduct can NOT be on without prognostic crop\n" );
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_dynamic_roots {
#
# dynamic root model
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_dynroot', 'phys'=>$physv->as_string(), 'bgc_mode'=>$nl_flags->{'bgc_mode'});
my $use_dynroot = $nl->get_value('use_dynroot');
if ( &value_is_true($use_dynroot) && ($nl_flags->{'bgc_mode'} eq "sp") ) {
$log->fatal_error("Cannot turn dynroot mode on mode bgc=sp\n" .
"Set the bgc mode to 'cn' or 'bgc'.");
}
if ( &value_is_true( $use_dynroot ) && &value_is_true( $nl_flags->{'use_hydrstress'} ) ) {
$log->fatal_error("Cannot turn use_dynroot on when use_hydrstress is on" );
}
} # else - not relevant in clm4_0, not part of namelist definition, will not run.
}
#-------------------------------------------------------------------------------
sub setup_logic_c_isotope {
#
# Error checking for C-isotope options
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
my $use_c13 = $nl->get_value('use_c13');
my $use_c14 = $nl->get_value('use_c14');
if ( $nl_flags->{'bgc_mode'} ne "sp" && $nl_flags->{'bgc_mode'} ne "fates" ) {
if ( $nl_flags->{'bgc_mode'} ne "bgc" ) {
if ( defined($use_c13) && &value_is_true($use_c13) ) {
$log->warning("use_c13 is ONLY scientifically validated with the bgc=BGC configuration" );
}
if ( defined($use_c14) && &value_is_true($use_c14) ) {
$log->warning("use_c14 is ONLY scientifically validated with the bgc=BGC configuration" );
}
}
if ( defined($use_c14) ) {
if ( &value_is_true($use_c14) ) {
my $use_c14_bombspike = $nl->get_value('use_c14_bombspike');
if ( defined($use_c14_bombspike) && &value_is_true($use_c14_bombspike) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'atm_c14_filename',
'use_c14'=>$use_c14, 'use_cn'=>$nl_flags->{'use_cn'}, 'use_c14_bombspike'=>$nl->get_value('use_c14_bombspike'),
'ssp_rcp'=>$nl_flags->{'ssp_rcp'} );
}
} else {
if ( defined($nl->get_value('use_c14_bombspike')) ||
defined($nl->get_value('atm_c14_filename')) ) {
$log->fatal_error("use_c14 is FALSE and use_c14_bombspike or atm_c14_filename set");
}
}
} else {
if ( defined($nl->get_value('use_c14_bombspike')) ||
defined($nl->get_value('atm_c14_filename')) ) {
$log->fatal_error("use_c14 NOT set to .true., but use_c14_bompspike/atm_c14_filename defined.");
}
}
if ( defined($use_c13) ) {
if ( &value_is_true($use_c13) ) {
my $use_c13_timeseries = $nl->get_value('use_c13_timeseries');
if ( defined($use_c13_timeseries) && &value_is_true($use_c13_timeseries) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'atm_c13_filename',
'use_c13'=>$use_c13, 'use_cn'=>$nl_flags->{'use_cn'}, 'use_c13_timeseries'=>$nl->get_value('use_c13_timeseries'),
'ssp_rcp'=>$nl_flags->{'ssp_rcp'} );
}
} else {
if ( defined($nl->get_value('use_c13_timeseries')) ||
defined($nl->get_value('atm_c13_filename')) ) {
$log->fatal_error("use_c13 is FALSE and use_c13_timeseries or atm_c13_filename set");
}
}
} else {
if ( defined($nl->get_value('use_c13_timeseries')) ||
defined($nl->get_value('atm_c13_filename')) ) {
$log->fatal_error("use_c13 NOT set to .true., but use_c13_bompspike/atm_c13_filename defined.");
}
}
} else {
if ( defined($use_c13) ||
defined($use_c14) ||
defined($nl->get_value('use_c14_bombspike')) ||
defined($nl->get_value('atm_c14_filename')) ||
defined($nl->get_value('use_c13_timeseries')) ||
defined($nl->get_value('atm_c13_filename')) ) {
$log->fatal_error("bgc=sp and C isotope namelist variables were set, both can't be used at the same time");
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_nitrogen_deposition {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
#
# Nitrogen deposition for bgc=CN
#
if ( $physv->as_long() == $physv->as_long("clm4_0") && $nl_flags->{'bgc_mode'} ne "none" ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'ndepmapalgo', 'phys'=>$nl_flags->{'phys'},
'bgc'=>$nl_flags->{'bgc_mode'}, 'hgrid'=>$nl_flags->{'res'},
'clm_accelerated_spinup'=>$nl_flags->{'clm_accelerated_spinup'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_first_ndep', 'phys'=>$nl_flags->{'phys'},
'bgc'=>$nl_flags->{'bgc_mode'}, 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_last_ndep', 'phys'=>$nl_flags->{'phys'},
'bgc'=>$nl_flags->{'bgc_mode'}, 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
# Set align year, if first and last years are different
if ( $nl->get_value('stream_year_first_ndep') != $nl->get_value('stream_year_last_ndep') ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'model_year_align_ndep', 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_fldfilename_ndep', 'phys'=>$nl_flags->{'phys'},
'bgc'=>$nl_flags->{'bgc_mode'}, 'ssp_rcp'=>$nl_flags->{'ssp_rcp'},
'hgrid'=>"1.9x2.5" );
} elsif ( $physv->as_long() >= $physv->as_long("clm4_5") && $nl_flags->{'bgc_mode'} =~/cn|bgc/ ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'ndepmapalgo', 'phys'=>$nl_flags->{'phys'},
'use_cn'=>$nl_flags->{'use_cn'}, 'hgrid'=>$nl_flags->{'res'},
'clm_accelerated_spinup'=>$nl_flags->{'clm_accelerated_spinup'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'ndep_taxmode', 'phys'=>$nl_flags->{'phys'},
'use_cn'=>$nl_flags->{'use_cn'},
'lnd_tuning_mode'=>$nl_flags->{'lnd_tuning_mode'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'ndep_varlist', 'phys'=>$nl_flags->{'phys'},
'use_cn'=>$nl_flags->{'use_cn'},
'lnd_tuning_mode'=>$nl_flags->{'lnd_tuning_mode'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_first_ndep', 'phys'=>$nl_flags->{'phys'},
'use_cn'=>$nl_flags->{'use_cn'}, 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_last_ndep', 'phys'=>$nl_flags->{'phys'},
'use_cn'=>$nl_flags->{'use_cn'}, 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
# Set align year, if first and last years are different
if ( $nl->get_value('stream_year_first_ndep') != $nl->get_value('stream_year_last_ndep') ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'model_year_align_ndep', 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_fldfilename_ndep', 'phys'=>$nl_flags->{'phys'},
'use_cn'=>$nl_flags->{'use_cn'}, 'lnd_tuning_mode'=>$nl_flags->{'lnd_tuning_mode'}, 'sim_year_range'=>$nl_flags->{'sim_year_range'},
'hgrid'=>"0.9x1.25", 'ssp_rcp'=>$nl_flags->{'ssp_rcp'}, 'nofail'=>1 );
if ( ! defined($nl->get_value('stream_fldfilename_ndep') ) ) {
# Also check at f19 resolution
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_fldfilename_ndep', 'phys'=>$nl_flags->{'phys'},
'use_cn'=>$nl_flags->{'use_cn'}, 'lnd_tuning_mode'=>$nl_flags->{'lnd_tuning_mode'}, 'sim_year_range'=>$nl_flags->{'sim_year_range'},
'hgrid'=>"1.9x2.5", 'ssp_rcp'=>$nl_flags->{'ssp_rcp'}, 'nofail'=>1 );
# If not found report an error
if ( ! defined($nl->get_value('stream_fldfilename_ndep') ) ) {
$log->warning("Did NOT find the Nitrogen-deposition forcing file (stream_fldfilename_ndep) for this ssp_rcp\n" .
"One way to get around this is to point to a file for another existing ssp_rcp in your user_nl_clm file.\n" .
"If you are running with CAM and WACCM chemistry Nitrogen deposition will come through the coupler.\n" .
"This file won't be used, so it doesn't matter what it points to -- but it's required to point to something.\n" )
}
}
} else {
# If bgc is NOT CN/CNDV then make sure none of the ndep settings are set!
if ( defined($nl->get_value('stream_year_first_ndep')) ||
defined($nl->get_value('stream_year_last_ndep')) ||
defined($nl->get_value('model_year_align_ndep')) ||
defined($nl->get_value('ndep_taxmode' )) ||
defined($nl->get_value('ndep_varlist' )) ||
defined($nl->get_value('stream_fldfilename_ndep'))
) {
$log->fatal_error("When bgc is NOT CN or CNDV none of: stream_year_first_ndep," .
"stream_year_last_ndep, model_year_align_ndep, ndep_taxmod, " .
"ndep_varlist, nor stream_fldfilename_ndep" .
" can be set!");
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_cnmresp {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
#
# CN Maintence respiration for bgc=CN
#
if ( $physv->as_long() >= $physv->as_long("clm4_5") && $nl_flags->{'bgc_mode'} ne "sp" ) {
# When FUN is on and it's clm5_0 get a default value
if ( &value_is_true( $nl->get_value('use_fun') ) && $physv->as_long() >= $physv->as_long("clm5_0")) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults,
$nl, 'br_root', 'phys'=>$nl_flags->{'phys'},
'use_fun'=>$nl->get_value('use_fun'),
'use_cn'=>$nl_flags->{'use_cn'} );
}
} else {
# If bgc is NOT CN/CNDV then make sure not set
if ( defined($nl->get_value('br_root'))) {
$log->fatal_error("br_root can NOT be set when phys==clm4_0 or bgc_mode==sp!");
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_photosyns {
# MUST be after use_hydrstress is set
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
#
# Photo synthesis
#
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults,
$nl, 'rootstem_acc', 'phys'=>$nl_flags->{'phys'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults,
$nl, 'light_inhibit', 'phys'=>$nl_flags->{'phys'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults,
$nl, 'leafresp_method', 'phys'=>$nl_flags->{'phys'},
'use_cn'=>$nl_flags->{'use_cn'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults,
$nl, 'modifyphoto_and_lmr_forcrop', 'phys'=>$nl_flags->{'phys'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults,
$nl, 'stomatalcond_method', 'phys'=>$nl_flags->{'phys'},
'use_hydrstress'=>$nl_flags->{'use_hydrstress'} );
# When CN on, must NOT be scaled by vcmax25top
if ( &value_is_true( $nl_flags->{'use_cn'} ) ) {
if ( $nl->get_value('leafresp_method') == 0 ) {
$log->fatal_error("leafresp_method can NOT be set to scaled to vcmax (0) when CN is on!");
}
# And when CN off, must NOT be anything besides scaled by vxmac25top
} else {
if ( $nl->get_value('leafresp_method') != 0 ) {
$log->fatal_error("leafresp_method can NOT be set to anything besides scaled to vcmax (0) when bgc_mode==sp!");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_canopy {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
#
# Canopy state
#
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults,
$nl, 'leaf_mr_vcm', 'phys'=>$nl_flags->{'phys'} )
}
}
#-------------------------------------------------------------------------------
sub setup_logic_popd_streams {
# population density streams require clm4_5/clm5_0 and CN/BGC
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
if ( &value_is_true($nl_flags->{'cnfireson'}) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'popdensmapalgo', 'hgrid'=>$nl_flags->{'res'},
'clm_accelerated_spinup'=>$nl_flags->{'clm_accelerated_spinup'}, 'cnfireson'=>$nl_flags->{'cnfireson'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_first_popdens', 'phys'=>$nl_flags->{'phys'},
'cnfireson'=>$nl_flags->{'cnfireson'}, 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_last_popdens', 'phys'=>$nl_flags->{'phys'},
'cnfireson'=>$nl_flags->{'cnfireson'}, 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
# Set align year, if first and last years are different
if ( $nl->get_value('stream_year_first_popdens') !=
$nl->get_value('stream_year_last_popdens') ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'model_year_align_popdens', 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'}, 'cnfireson'=>$nl_flags->{'cnfireson'});
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_fldfilename_popdens', 'phys'=>$nl_flags->{'phys'},
'cnfireson'=>$nl_flags->{'cnfireson'}, 'hgrid'=>"0.5x0.5", 'ssp_rcp'=>$nl_flags->{'ssp_rcp'} );
} else {
# If bgc is NOT CN/CNDV or fire_method==nofire then make sure none of the popdens settings are set
if ( defined($nl->get_value('stream_year_first_popdens')) ||
defined($nl->get_value('stream_year_last_popdens')) ||
defined($nl->get_value('model_year_align_popdens')) ||
defined($nl->get_value('popdens_tintalgo' )) ||
defined($nl->get_value('stream_fldfilename_popdens')) ) {
$log->fatal_error("When bgc is SP (NOT CN or BGC) or fire_method==nofire none of: stream_year_first_popdens,\n" .
"stream_year_last_popdens, model_year_align_popdens, popdens_tintalgo nor\n" .
"stream_fldfilename_popdens can be set!");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_urbantv_streams {
# urban time varying streams require clm4_5/clm5_0
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'urbantvmapalgo',
'hgrid'=>$nl_flags->{'res'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_first_urbantv', 'phys'=>$nl_flags->{'phys'},
'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_last_urbantv', 'phys'=>$nl_flags->{'phys'},
'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
# Set align year, if first and last years are different
if ( $nl->get_value('stream_year_first_urbantv') !=
$nl->get_value('stream_year_last_urbantv') ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl,
'model_year_align_urbantv', 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_fldfilename_urbantv', 'phys'=>$nl_flags->{'phys'},
'hgrid'=>"0.9x1.25" );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_lightning_streams {
# lightning streams require clm4_5/clm5_0 and CN/BGC
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
if ( &value_is_true($nl_flags->{'cnfireson'}) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'lightngmapalgo', 'use_cn'=>$nl_flags->{'use_cn'},
'hgrid'=>$nl_flags->{'res'},
'clm_accelerated_spinup'=>$nl_flags->{'clm_accelerated_spinup'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_first_lightng', 'use_cn'=>$nl_flags->{'use_cn'},
'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_last_lightng', 'use_cn'=>$nl_flags->{'use_cn'},
'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
# Set align year, if first and last years are different
if ( $nl->get_value('stream_year_first_lightng') !=
$nl->get_value('stream_year_last_lightng') ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'model_year_align_lightng', 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_fldfilename_lightng', 'use_cn'=>$nl_flags->{'use_cn'},
'hgrid'=>$nl_flags->{'light_res'} );
} else {
# If bgc is NOT CN/CNDV then make sure none of the Lightng settings are set
if ( defined($nl->get_value('stream_year_first_lightng')) ||
defined($nl->get_value('stream_year_last_lightng')) ||
defined($nl->get_value('model_year_align_lightng')) ||
defined($nl->get_value('lightng_tintalgo' )) ||
defined($nl->get_value('stream_fldfilename_lightng')) ) {
$log->fatal_error("When bgc is SP (NOT CN or BGC) or fire_method==nofire none of: stream_year_first_lightng,\n" .
"stream_year_last_lightng, model_year_align_lightng, lightng_tintalgo nor\n" .
"stream_fldfilename_lightng can be set!");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_dry_deposition {
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
if ($opts->{'drydep'} ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'drydep_list');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'drydep_method');
} else {
if ( defined($nl->get_value('drydep_list')) ||
defined($nl->get_value('drydep_method')) ) {
$log->fatal_error("drydep_list or drydep_method defined, but drydep option NOT set");
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_fire_emis {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ($opts->{'fire_emis'} ) {
if ( $physv->as_long() < $physv->as_long("clm4_5") ) {
$log->fatal_error("fire_emis option can NOT be set for CLM versions before clm4_5");
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fire_emis_factors_file');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fire_emis_specifier');
} else {
if ( defined($nl->get_value('fire_emis_elevated')) ||
defined($nl->get_value('fire_emis_factors_file')) ||
defined($nl->get_value('fire_emis_specifier')) ) {
$log->fatal_error("fire_emission setting defined: fire_emis_elevated, fire_emis_factors_file, or fire_emis_specifier, but fire_emis option NOT set");
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_megan {
my ($opts, $nl_flags, $definition, $defaults, $nl) = @_;
my $var = "megan";
if ( $opts->{$var} eq "default" ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl,
'megan', clm_accelerated_spinup=>$nl_flags->{'clm_accelerated_spinup'} );
$nl_flags->{$var} = $nl->get_value($var);
} else {
$nl_flags->{$var} = $opts->{$var};
}
if ($nl_flags->{'megan'} ) {
if ( &value_is_true( $nl_flags->{'use_fates'} ) ) {
$log->fatal_error("MEGAN can NOT be on when ED is also on.\n" .
" Use the '-no-megan' option when '-bgc fates' is activated");
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'megan_specifier');
check_megan_spec( $opts, $nl, $definition );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'megan_factors_file');
} else {
if ( defined($nl->get_value('megan_specifier')) ||
defined($nl->get_value('megan_factors_file')) ) {
$log->fatal_error("megan_specifier or megan_factors_file defined, but megan option NOT set");
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_soilm_streams {
# prescribed soil moisture streams require clm4_5/clm5_0
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_soil_moisture_streams');
if ( &value_is_true( $nl->get_value('use_soil_moisture_streams') ) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'soilm_tintalgo',
'hgrid'=>$nl_flags->{'res'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'soilm_offset',
'hgrid'=>$nl_flags->{'res'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_first_soilm', 'phys'=>$nl_flags->{'phys'},
'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_last_soilm', 'phys'=>$nl_flags->{'phys'},
'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
# Set align year, if first and last years are different
if ( $nl->get_value('stream_year_first_soilm') !=
$nl->get_value('stream_year_last_soilm') ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl,
'model_year_align_soilm', 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_fldfilename_soilm', 'phys'=>$nl_flags->{'phys'},
'hgrid'=>$nl_flags->{'res'} );
if ( ($opts->{'use_case'} =~ /_transient$/) &&
(remove_leading_and_trailing_quotes($nl->get_value("soilm_tintalgo")) eq "linear") ) {
$log->warning("For a transient case, soil moisture streams, should NOT use soilm_tintalgo='linear'" .
" since vegetated areas could go from missing to not missing or vice versa" );
}
} else {
if ( defined($nl->get_value('stream_year_first_soilm')) ||
defined($nl->get_value('model_year_align_soilm')) ||
defined($nl->get_value('stream_fldfilename_soilm')) ||
defined($nl->get_value('soilm_tintalgo')) ||
defined($nl->get_value('soilm_offset')) ||
defined($nl->get_value('stream_year_last_soilm')) ) {
$log->fatal_error("One of the soilm streams namelist items (stream_year_first_soilm, " .
" model_year_align_soilm, stream_fldfilename_soilm, stream_fldfilename_soilm)" .
" soilm_tintalgo soilm_offset" .
" is defined, but use_soil_moisture_streams option NOT set to true");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_lai_streams {
# lai streams require clm4_5/clm5_0
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_lai_streams');
if ( &value_is_true($nl_flags->{'use_crop'}) && &value_is_true($nl->get_value('use_lai_streams')) ) {
$log->fatal_error("turning use_lai_streams on is incompatable with use_crop set to true.");
}
if ( $nl_flags->{'bgc_mode'} eq "sp" ) {
if ( &value_is_true($nl->get_value('use_lai_streams')) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'lai_mapalgo',
'hgrid'=>$nl_flags->{'res'} );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_first_lai',
'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_year_last_lai',
'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
# Set align year, if first and last years are different
if ( $nl->get_value('stream_year_first_lai') !=
$nl->get_value('stream_year_last_lai') ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl,
'model_year_align_lai', 'sim_year'=>$nl_flags->{'sim_year'},
'sim_year_range'=>$nl_flags->{'sim_year_range'});
}
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'stream_fldfilename_lai',
'hgrid'=>"360x720cru" );
}
} else {
# If bgc is CN/CNDV then make sure none of the LAI settings are set
if ( defined($nl->get_value('stream_year_first_lai')) ||
defined($nl->get_value('stream_year_last_lai')) ||
defined($nl->get_value('model_year_align_lai')) ||
defined($nl->get_value('lai_tintalgo' )) ||
defined($nl->get_value('stream_fldfilename_lai')) ) {
$log->fatal_error("When bgc is NOT SP none of the following can be set: stream_year_first_lai,\n" .
"stream_year_last_lai, model_year_align_lai, lai_tintalgo nor\n" .
"stream_fldfilename_lai (eg. don't use this option with BGC,CN,CNDV nor BGDCV).");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_soilwater_movement {
# soilwater_movement require clm4_5/clm5_0
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'soilwater_movement_method' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'upper_boundary_condition' );
my $soilmtd = $nl->get_value("soilwater_movement_method");
my $use_bed = $nl->get_value('use_bedrock' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl,
'lower_boundary_condition', 'vichydro'=>$nl_flags->{'vichydro'},
'soilwater_movement_method'=>$soilmtd, 'use_bedrock'=>$use_bed
);
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'dtmin' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'verySmall' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'xTolerUpper' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'xTolerLower' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'expensive' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'inexpensive' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'flux_calculation' );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_century_soilbgcdecompcascade {
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") &&
(&value_is_true($nl->get_value('use_cn')) || &value_is_true($nl->get_value('use_fates'))) &&
&value_is_true($nl->get_value('use_century_decomp')) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'initial_Cstocks',
'use_cn' => $nl->get_value('use_cn'), 'use_fates' => $nl->get_value('use_fates'),
'use_century_decomp' => $nl->get_value('use_century_decomp') );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'initial_Cstocks_depth',
'use_cn' => $nl->get_value('use_cn'), 'use_fates' => $nl->get_value('use_fates'),
'use_century_decomp' => $nl->get_value('use_century_decomp') );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_cnvegcarbonstate {
# MUST be AFTER: setup_logic_dynamic_plant_nitrogen_alloc as depends on mm_nuptake_opt which is set there
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") && &value_is_true($nl->get_value('use_cn')) ) {
my $mmnuptake = $nl->get_value('mm_nuptake_opt');
if ( ! defined($mmnuptake) ) { $mmnuptake = ".false."; }
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'initial_vegC',
'use_cn' => $nl->get_value('use_cn'), 'mm_nuptake_opt' => $mmnuptake );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_cngeneral {
# Must be set after setup_logic_co2_type
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") && &value_is_true($nl->get_value('use_cn')) ) {
if ( &value_is_true($nl->get_value('use_crop')) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'dribble_crophrv_xsmrpool_2atm',
'co2_type' => remove_leading_and_trailing_quotes($nl->get_value('co2_type')),
'use_crop' => $nl->get_value('use_crop') );
} else {
if ( defined($nl->get_value('dribble_crophrv_xsmrpool_2atm')) ) {
$log->fatal_error("When CROP is NOT on dribble_crophrv_xsmrpool_2atm can NOT be set\n" );
}
}
} else {
if ( defined($nl->get_value('reseed_dead_plants')) ||
defined($nl->get_value('dribble_crophrv_xsmrpool_2atm')) ) {
$log->fatal_error("When CN is not on none of the following can be set: ,\n" .
"dribble_crophrv_xsmrpool_2atm nor reseed_dead_plantsr\n" .
"(eg. don't use these options with SP mode).");
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_rooting_profile {
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'rooting_profile_method_water' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'rooting_profile_method_carbon' );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_friction_vel {
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'zetamaxstable' );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_soil_resis {
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'soil_resis_method' );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_canopyfluxes {
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_undercanopy_stability' );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_canopyhydrology {
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'interception_fraction' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'maximum_leaf_wetted_fraction' );
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'use_clm5_fpi' );
}
}
#-------------------------------------------------------------------------------
sub setup_logic_snowpack {
#
# Snowpack related options
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ($physv->as_long() >= $physv->as_long("clm4_5")) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'nlevsno');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'h2osno_max');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'int_snow_max');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'n_melt_glcmec');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'wind_dependent_snow_density');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'snow_overburden_compaction_method');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'lotmp_snowdensity_method');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'upplim_destruct_metamorph');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fresh_snw_rds_max');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'reset_snow');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'reset_snow_glc');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'reset_snow_glc_ela');
if (remove_leading_and_trailing_quotes($nl->get_value('snow_overburden_compaction_method')) eq 'Vionnet2012') {
# overburden_compress_tfactor isn't used if we're using the Vionnet2012
# snow overburden compaction method, so make sure the user hasn't tried
# to set it
if (defined($nl->get_value('overburden_compress_tfactor'))) {
$log->fatal_error('overburden_compress_tfactor is set, but does not apply when using snow_overburden_compaction_method=Vionnet2012');
}
} else {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'overburden_compress_tfactor');
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_atm_forcing {
#
# Options related to atmospheric forcings
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ($physv->as_long() >= $physv->as_long("clm4_5")) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'glcmec_downscale_longwave');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'repartition_rain_snow');
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'lapse_rate');
my $var;
foreach $var ("lapse_rate_longwave",
"longwave_downscaling_limit") {
if ( &value_is_true($nl->get_value("glcmec_downscale_longwave")) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var);
} else {
if (defined($nl->get_value($var))) {
$log->fatal_error("$var can only be set if glcmec_downscale_longwave is true");
}
}
}
foreach $var ("precip_repartition_glc_all_snow_t",
"precip_repartition_glc_all_rain_t",
"precip_repartition_nonglc_all_snow_t",
"precip_repartition_nonglc_all_rain_t") {
if ( &value_is_true($nl->get_value("repartition_rain_snow")) ){
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var);
} else {
if (defined($nl->get_value($var))) {
$log->fatal_error("$var can only be set if repartition_rain_snow is true");
}
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_lnd2atm {
#
# Options related to fields sent to atmosphere
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ($physv->as_long() >= $physv->as_long("clm4_5")) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'melt_non_icesheet_ice_runoff');
}
}
#-------------------------------------------------------------------------------
sub setup_logic_initinterp {
#
# Options related to init_interp
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ($physv->as_long() >= $physv->as_long("clm4_5")) {
my $var = 'init_interp_method';
if ( &value_is_true($nl->get_value("use_init_interp"))) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var);
} else {
if (defined($nl->get_value($var))) {
$log->fatal_error("$var can only be set if use_init_interp is true");
}
}
}
}
#-------------------------------------------------------------------------------
sub setup_logic_fates {
#
# Set some default options related to Ecosystem Demography
#
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
if ($physv->as_long() >= $physv->as_long("clm4_5") && &value_is_true( $nl_flags->{'use_fates'}) ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, 'fates_paramfile', 'phys'=>$nl_flags->{'phys'});
my @list = ( "use_fates_spitfire", "use_fates_planthydro", "use_fates_ed_st3", "use_fates_ed_prescribed_phys",
"use_fates_inventory_init", "use_fates_logging","fates_parteh_mode" );
foreach my $var ( @list ) {
add_default($opts, $nl_flags->{'inputdata_rootdir'}, $definition, $defaults, $nl, $var, 'use_fates'=>$nl_flags->{'use_fates'} );
}
my $var = "use_fates_inventory_init";
if ( defined($nl->get_value($var)) ) {
if ( &value_is_true($nl->get_value($var)) ) {
$var = "fates_inventory_ctrl_filename";
my $fname = substr $nl->get_value($var), 1, -1; # ignore first and last positions of string because those are quote characters
if ( ! defined($nl->get_value($var)) ) {
$log->fatal_error("$var is required when use_fates_inventory_init is set" );
} elsif ( ! -f "$fname" ) {
$log->fatal_error("$fname does NOT point to a valid filename" );
}
}
}
}
}
#-------------------------------------------------------------------------------
sub write_output_files {
my ($opts, $nl_flags, $defaults, $nl, $physv) = @_;
my $note = "";
my $var = "note";
if ( ! defined($opts->{$var}) ) {
$opts->{$var} = $defaults->get_value($var);
}
if ( $opts->{$var} ) {
$note = "Comment:\n" .
"This namelist was created using the following command-line:\n" .
" $nl_flags->{'cfgdir'}/$ProgName $nl_flags->{'cmdline'}\n" .
"For help on options use: $nl_flags->{'cfgdir'}/$ProgName -help";
}
# CLM component
my @groups;
if ( $physv->as_long() == $physv->as_long("clm4_0") ) {
@groups = qw(clm_inparm);
# Eventually only list namelists that are actually used when CN on
#if ( $nl_flags->{'bgc_mode'} eq "cn" ) {
push @groups, "ndepdyn_nml";
#}
} else {
@groups = qw(clm_inparm ndepdyn_nml popd_streams urbantv_streams light_streams
soil_moisture_streams lai_streams atm2lnd_inparm lnd2atm_inparm clm_canopyhydrology_inparm cnphenology
clm_soilhydrology_inparm dynamic_subgrid cnvegcarbonstate
finidat_consistency_checks dynpft_consistency_checks
clm_initinterp_inparm century_soilbgcdecompcascade
soilhydrology_inparm luna friction_velocity mineral_nitrogen_dynamics
soilwater_movement_inparm rooting_profile_inparm
soil_resis_inparm bgc_shared canopyfluxes_inparm aerosol
clmu_inparm clm_soilstate_inparm clm_nitrogen clm_snowhydrology_inparm
cnprecision_inparm clm_glacier_behavior crop irrigation_inparm);
#@groups = qw(clm_inparm clm_canopyhydrology_inparm clm_soilhydrology_inparm
# finidat_consistency_checks dynpft_consistency_checks);
# Eventually only list namelists that are actually used when CN on
#if ( $nl_flags->{'bgc_mode'} eq "cn" ) {
# push @groups, qw(ndepdyn_nml popd_streams light_streams);
#}
if ( &value_is_true($nl_flags->{'use_lch4'}) ) {
push @groups, "ch4par_in";
}
if ( $physv->as_long() >= $physv->as_long("clm4_5") ) {
push @groups, "clm_humanindex_inparm";
push @groups, "cnmresp_inparm";
push @groups, "photosyns_inparm";
push @groups, "cnfire_inparm";
push @groups, "cn_general";
push @groups, "nitrif_inparm";
push @groups, "lifire_inparm";
push @groups, "ch4finundated";
push @groups, "clm_canopy_inparm";
}
}
my $outfile;
$outfile = "$opts->{'dir'}/lnd_in";
$nl->write($outfile, 'groups'=>\@groups, 'note'=>"$note" );
$log->verbose_message("Writing clm namelist to $outfile");
# Drydep, fire-emission or MEGAN namelist for driver
@groups = qw(drydep_inparm megan_emis_nl fire_emis_nl carma_inparm);
$outfile = "$opts->{'dir'}/drv_flds_in";
$nl->write($outfile, 'groups'=>\@groups, 'note'=>"$note" );
$log->verbose_message("Writing @groups namelists to $outfile");
}
sub write_output_real_parameter_file {
my ($opts, $nl_flags, $definition, $defaults, $nl, $physv) = @_;
# Output real parameters
if ( defined($opts->{'output_reals_filename'}) ) {
my $file = $opts->{'output_reals_filename'};
my $fh = IO::File->new($file, '>>') or $log->fatal_error("can't create real parameter filename: $file");
foreach my $var ( $definition->get_var_names() ) {
my $type = $definition->get_var_type($var);
my $doc = $definition->get_var_doc($var);
$doc =~ s/\n//g;
if ( $type =~ /real/ ) {
my $val = $nl->get_value($var);
if ( ! defined($val) ) { $val = "?.??"; }
print $fh "\! $doc\n$var = $val\n";
}
}
$fh->close();
}
}
#-------------------------------------------------------------------------------
sub add_default {
# Add a value for the specified variable to the specified namelist object. The variables
# already in the object have the higher precedence, so if the specified variable is already
# defined in the object then don't overwrite it, just return.
#
# This method checks the definition file and adds the variable to the correct
# namelist group.
#
# The value can be provided by using the optional argument key 'val' in the
# calling list. Otherwise a default value is obtained from the namelist
# defaults object. If no default value is found this method throws an exception
# unless the 'nofail' option is set true.
#
# Example 1: Specify the default value $val for the namelist variable $var in namelist
# object $nl:
#
# add_default($inputdata_rootdir, $definition, $defaults, $nl, $var, 'val'=>$val)
#
# Example 2: Add a default for variable $var if an appropriate value is found. Otherwise
# don't add the variable
#
# add_default($inputdata_rootdir, $definition, $defaults, $nl, $var, 'nofail'=>1)
#
#
# ***** N.B. ***** This routine assumes the following variables are in package main::
# $definition -- the namelist definition object
# $defaults -- the namelist defaults object
# $inputdata_rootdir -- CESM inputdata root directory
my $opts = shift;
my $inputdata_rootdir = shift;
my $definition = shift;
my $defaults = shift;
my $nl = shift;
my $var = shift;
my %settings = @_;
my $test_files = $opts->{'test'};
#my $nl = shift; # namelist object
#my $var = shift; # name of namelist variable
#my %settings = @_; # options
# If variable has quotes around it
if ( $var =~ /'(.+)'/ ) {
$var = $1;
}
# Query the definition to find which group the variable belongs to. Exit if not found.
my $group = $definition->get_group_name($var);
unless ($group) {
my $fname = $definition->get_file_name();
$log->fatal_error("variable \"$var\" not found in namelist definition file $fname.");
}
# check whether the variable has a value in the namelist object -- if so then skip to end
my $val = $nl->get_variable_value($group, $var);
if (! defined $val) {
# Look for a specified value in the options hash
if (defined $settings{'val'}) {
$val = $settings{'val'};
}
# or else get a value from namelist defaults object.
# Note that if the 'val' key isn't in the hash, then just pass anything else
# in %settings to the get_value method to be used as attributes that are matched
# when looking for default values.
else {
$val = $defaults->get_value($var, \%settings);
# Truncate model_version appropriately
if ( $var eq "model_version" ) {
$val =~ /(URL: https:\/\/[a-zA-Z0-9._-]+\/)([a-zA-Z0-9\/._-]+)(\/bld\/.+)/;
$val = $2;
}
}
# if no value is found then exit w/ error (unless 'nofail' option set)
unless ( defined($val) ) {
unless ($settings{'nofail'}) {
if ($var eq 'finidat') {
$log->message("No default value found for $var.\n" .
" Are defaults provided for this resolution and land mask?" );
} else {
$log->fatal_error("No default value found for $var.\n" .
" Are defaults provided for this resolution and land mask?");
}
}
else {
return;
}
}
# query the definition to find out if the variable is an input pathname
my $is_input_pathname = $definition->is_input_pathname($var);
# The default values for input pathnames are relative. If the namelist
# variable is defined to be an absolute pathname, then prepend
# the CESM inputdata root directory.
if (not defined $settings{'no_abspath'}) {
if (defined $settings{'set_abspath'}) {
$val = set_abs_filepath($val, $settings{'set_abspath'});
} else {
if ($is_input_pathname eq 'abs') {
$val = set_abs_filepath($val, $inputdata_rootdir);
if ( $test_files and ($val !~ /null/) and (! -f "$val") ) {
$log->fatal_error("file not found: $var = $val");
}
}
}
}
# query the definition to find out if the variable takes a string value.
# The returned string length will be >0 if $var is a string, and 0 if not.
my $str_len = $definition->get_str_len($var);
# If the variable is a string, then add quotes if they're missing
if ($str_len > 0) {
$val = quote_string($val);
}
# set the value in the namelist
$nl->set_variable_value($group, $var, $val);
}
}
#-------------------------------------------------------------------------------
sub expand_xml_variables_in_namelist {
# Go through all variables in the namelist and expand any XML env settings in them
my ($nl, $xmlvar_ref) = @_;
foreach my $group ( $nl->get_group_names() ) {
foreach my $var ( $nl->get_variable_names($group) ) {
my $val = $nl->get_variable_value($group, $var);
my $newval = SetupTools::expand_xml_var( $val, $xmlvar_ref );
if ( $newval ne $val ) {
$nl->set_variable_value($group, $var, $newval);
}
}
}
}
#-------------------------------------------------------------------------------
sub check_input_files {
# For each variable in the namelist which is an input dataset, check to see if it
# exists locally.
#
# ***** N.B. ***** This routine assumes the following variables are in package main::
# $definition -- the namelist definition object
# $nl -- namelist object
# $inputdata_rootdir -- if false prints test, else creates inputdata file
my ($nl, $inputdata_rootdir, $outfile, $definition) = @_;
open(OUTFILE, ">>$outfile") if defined $inputdata_rootdir;
# Look through all namelist groups
my @groups = $nl->get_group_names();
foreach my $group (@groups) {
# Look through all variables in each group
my @vars = $nl->get_variable_names($group);
foreach my $var (@vars) {
# Is the variable an input dataset?
my $input_pathname_type = $definition->is_input_pathname($var);
# If it is, check whether it exists locally and print status
if ($input_pathname_type) {
# Get pathname of input dataset
my $pathname = $nl->get_variable_value($group, $var);
# Need to strip the quotes
$pathname =~ s/['"]//g;
if ($input_pathname_type eq 'abs') {
if ($inputdata_rootdir) {
#MV $pathname =~ s:$inputdata_rootdir::;
print OUTFILE "$var = $pathname\n";
}
else {
if (-e $pathname) { # use -e rather than -f since the absolute pathname
# might be a directory
print "OK -- found $var = $pathname\n";
}
else {
print "NOT FOUND: $var = $pathname\n";
}
}
}
elsif ($input_pathname_type =~ m/rel:(.+)/o) {
# The match provides the namelist variable that contains the
# root directory for a relative filename
my $rootdir_var = $1;
my $rootdir = $nl->get_variable_value($group, $rootdir_var);
$rootdir =~ s/['"]//g;
if ($inputdata_rootdir) {
$pathname = "$rootdir/$pathname";
#MV $pathname =~ s:$inputdata_rootdir::;
print OUTFILE "$var = $pathname\n";
}
else {
if (-f "$rootdir/$pathname") {
print "OK -- found $var = $rootdir/$pathname\n";
}
else {
print "NOT FOUND: $var = $rootdir/$pathname\n";
}
}
}
}
}
}
close OUTFILE if defined $inputdata_rootdir;
return 0 if defined $inputdata_rootdir;
}
#-------------------------------------------------------------------------------
sub set_abs_filepath {
# check whether the input filepath is an absolute path, and if it isn't then
# prepend a root directory
my ($filepath, $rootdir) = @_;
# strip any leading/trailing whitespace and quotes
$filepath = trim($filepath);
$filepath = remove_leading_and_trailing_quotes($filepath);
$rootdir = trim($rootdir);
$rootdir = remove_leading_and_trailing_quotes($rootdir);
my $out = $filepath;
unless ( $filepath =~ /^\// ) { # unless $filepath starts with a /
$out = "$rootdir/$filepath"; # prepend the root directory
}
return $out;
}
#-------------------------------------------------------------------------------
sub valid_option {
my ($val, @expect) = @_;
my $expect;
$val = trim($val);
foreach $expect (@expect) {
if ($val =~ /^$expect$/i) { return $expect; }
}
return undef;
}
#-------------------------------------------------------------------------------
sub check_use_case_name {
#
# Check the use-case name and ensure it follows the naming convention.
#
my ($use_case) = @_;
my $diestring = "bad use_case name $use_case, follow the conventions " .
"in namelist_files/use_cases/README\n";
my $desc = "[a-zA-Z0-9]*";
my $ssp_rcp = "SSP[0-9]-[0-9\.]+";
if ( $use_case =~ /^[0-9]+-[0-9]+([a-zA-Z0-9_\.-]*)_transient$/ ) {
my $string = $1;
if ( $string =~ /^_($ssp_rcp)_*($desc)$/ ) {
# valid name
} elsif ( $string =~ /^_*($desc)$/ ) {
# valid name
} else {
$log->fatal_error($diestring);
}
} elsif ( $use_case =~ /^20thC([a-zA-Z0-9_\.]*)_transient$/ ) {
my $string = $1;
if ( $string =~ /^_($ssp_rcp)_*($desc)$/ ) {
# valid name
} elsif ( $string =~ /^_*($desc)$/ ) {
# valid name
} else {
$log->fatal_error($diestring);
}
} elsif ( $use_case =~ /^[0-9]+([a-zA-Z0-9_\.-]*)_control$/ ) {
my $string = $1;
if ( $string =~ /^_($ssp_rcp)_*($desc)$/ ) {
# valid name
} elsif ( $string =~ /^_*($desc)$/ ) {
# valid name
} else {
$log->fatal_error($diestring);
}
} elsif ( $use_case =~ /^([0-9]+)_*($desc)_control$/ ) {
# valid name
} elsif ( $use_case =~ /^($desc)_pd$/ ) {
# valid name
} else {
$log->fatal_error($diestring);
}
}
#-------------------------------------------------------------------------------
sub validate_options {
# $source -- text string declaring the source of the options being validated
# $cfg -- configure object
# $opts -- reference to hash that contains the options
my ($source, $cfg, $opts) = @_;
my ($opt, $old, @expect);
# use_case
$opt = 'use_case';
if (defined $opts->{$opt}) {
if ( $opts->{$opt} ne "list" ) {
# create the @expect array by listing the files in $use_case_dir
# and strip off the ".xml" part of the filename
@expect = ();
my @files = glob("$opts->{'use_case_dir'}/*.xml");
foreach my $file (@files) {
$file =~ m{.*/(.*)\.xml};
&check_use_case_name( $1 );
push @expect, $1;
}
$old = $opts->{$opt};
$opts->{$opt} = valid_option($old, @expect)
or $log->fatal_error("invalid value of $opt ($old) specified in $source\n" .
"expected one of: @expect");
} else {
print "Use cases are:...\n\n";
my @ucases;
foreach my $file( sort( glob($opts->{'use_case_dir'}."/*.xml") ) ) {
my $use_case;
if ( $file =~ /\/([^\/]+)\.xml$/ ) {
&check_use_case_name( $1 );
$use_case = $1;
} else {
$log->fatal_error("Bad name for use case file = $file");
}
my $uc_defaults = Build::NamelistDefaults->new("$file", $cfg);
printf "%15s = %s\n", $use_case, $uc_defaults->get_value("use_case_desc");
push @ucases, $use_case;
}
$log->exit_message("use cases : @ucases");
}
}
}
#-------------------------------------------------------------------------------
sub list_options {
#
# List the options for different command line values if asked for
#
my ($opts_cmdl, $definition, $defaults) = @_;
# options to list values that are in the defaults files
my @opts_list = ( "res", "mask", "sim_year", "ssp_rcp" );
my %opts_local;
foreach my $var ( "res", "mask", "sim_year", "ssp_rcp" ) {
my $val;
if ( $opts_cmdl->{$var} eq "list" ) {
$val = "default";
} elsif ( $opts_cmdl->{$var} eq "default" ) {
$val = $defaults->get_value($var, \%opts_local );
} else {
$val = $opts_cmdl->{$var};
}
my $vname = $var;
if ( $vname eq "res" ) { $vname = "hgrid"; }
$opts_local{$vname} = $val;
}
foreach my $var ( @opts_list ) {
if (defined $opts_cmdl->{$var}) {
if ( $opts_cmdl->{$var} eq "list" ) {
my @valid_values = $definition->get_valid_values( $var );
if ( $var eq "sim_year" ) {
unshift( @valid_values,
$definition->get_valid_values( "sim_year_range" ) );
}
unshift( @valid_values, "default" );
# Strip out quotes and the constant value
for( my $i = 0; $i <= $#valid_values; $i++ ) {
$valid_values[$i] =~ s/('|')//g;
if ( $valid_values[$i] eq "constant" ) { $valid_values[$i] = undef; }
}
my $val= $defaults->get_value($var, \%opts_local);
my $doc = $definition->get_var_doc( $var );
$doc =~ s/\n//;
chomp( $doc );
$log->exit_message("valid values for $var ($doc) :\n" .
" Values: @valid_values\n" .
" Default = $val\n" .
" (NOTE: resolution and mask and other settings may influence what the default is)");
}
}
}
# clm_demand
my $var = 'clm_demand';
if (defined $opts_cmdl->{$var}) {
if ( $opts_cmdl->{$var} eq "list" ) {
my @vars = $definition->get_var_names( );
my @demands = ( "null" );
foreach my $var ( @vars ) {
if ( $definition->get_group_name( $var ) ne "clm_inparm" ) { next; }
if ( defined($defaults->get_value($var, $opts_cmdl ) ) ) {
push( @demands, $var );
}
}
my $doc = $definition->get_var_doc( 'clm_demand' );
$doc =~ s/\n//;
chomp( $doc );
$log->exit_message("valid values for $var ($doc) :\n" .
"Namelist options to require: @demands\n" .
"any valid namelist item for clm_inparm can be set. However, not all are\n" .
"available in the clm defaults file. The defaults are also dependent on\n" .
"resolution and landmask, as well as other settings. Hence, the list above\n" .
"will vary depending on what you set for resolution and landmask.");
}
}
}
#-------------------------------------------------------------------------------
sub check_megan_spec {
#
# Check the megan specifier setting
#
my ($opts, $nl, $definition) = @_;
my $megan_spec = $nl->get_value('megan_specifier');
my @megan_spec_list = split( /\s*,\s*/, $megan_spec );
foreach $megan_spec ( @megan_spec_list ) {
if ( $megan_spec =~ /^['"]+[A-Za-z0-9]+\s*\=\s*([\sA-Za-z0-9+_-]+)["']+$/ ) {
my $megan_list = $1;
my @megan_cmpds = split( /\s*\+\s*/, $megan_list );
my $var = "megan_cmpds";
my $warn = 0;
foreach my $megan_cmpd ( @megan_cmpds ) {
if ( ! $definition->is_valid_value( $var, $megan_cmpd, 'noquotes'=>1 ) ) {
$log->warning("megan_compound $megan_cmpd NOT found in list" );
$warn++;
}
}
if ( $warn > 0 ) {
my @valid_values = $definition->get_valid_values( $var, 'noquotes'=>1 );
$log->warning("list of megan compounds includes:\n" .
"@valid_values\n" .
"Does your megan_factors_file include more compounds?\n" .
"If NOT your simulation will fail." );
}
} else {
$log->fatal_error("Bad format for megan_specifier = $megan_spec");
}
}
}
#-------------------------------------------------------------------------------
sub trim {
# remove leading and trailing whitespace from a string.
my ($str) = @_;
$str =~ s/^\s+//;
$str =~ s/\s+$//;
return $str;
}
#-------------------------------------------------------------------------------
sub quote_string {
# Add quotes around a string, unless they are already there
my ($str) = @_;
$str = trim($str);
unless ($str =~ /^['"]/) { #"'
$str = "\'$str\'";
}
return $str;
}
#-------------------------------------------------------------------------------
sub remove_leading_and_trailing_quotes {
# Remove leading and trailing single and double quotes from a string. Also
# removes leading spaces before the leading quotes, and trailing spaces after
# the trailing quotes.
my ($str) = @_;
$str = trim($str);
# strip any leading/trailing quotes
$str =~ s/^['"]+//;
$str =~ s/["']+$//;
return $str;
}
#-------------------------------------------------------------------------------
sub logical_to_fortran {
# Given a logical variable ('true' / 'false'), convert it to a fortran-style logical ('.true.' / '.false.')
# The result will be lowercase, regardless of the case of the input.
my ($var) = @_;
my $result;
if (lc($var) eq 'true') {
$result = ".true.";
}
elsif (lc($var) eq 'false') {
$result = ".false.";
}
else {
$log->fatal_error("Unexpected value in logical_to_fortran: $var");
}
return $result;
}
#-------------------------------------------------------------------------------
sub string_is_undef_or_empty {
# Return true if the given string is undefined or only spaces, false otherwise.
# A quoted empty string (' ' or " ") is treated as being empty.
my ($str) = @_;
if (!defined($str)) {
return 1;
}
else {
$str = remove_leading_and_trailing_quotes($str);
if ($str =~ /^\s*$/) {
return 1;
}
else {
return 0;
}
}
}
#-------------------------------------------------------------------------------
sub value_is_true {
# Return true if the given namelist value is .true.
# An undefined value is treated as false (with the assumption that false is the default in the code)
my ($val) = @_;
# Some regular expressions...
###my $TRUE = qr/\.true\./i;
###my $FALSE = qr/\.false\./i;
# **N.B.** the use of qr// for precompiling regexps isn't supported until perl 5.005.
my $TRUE = '\.?true\.?|[t]';
my $FALSE = '\.?false\.?|[f]';
my $is_true = 0;
if (defined($val)) {
if ($val =~ /$TRUE/i) {
$is_true = 1;
}
}
return $is_true;
}
#-------------------------------------------------------------------------------
sub version {
# The version is found in CLM ChangeLog file.
# $cfgdir is set by the configure script to the name of its directory.
my ($cfgdir) = @_;
my $logfile = "$cfgdir/../doc/ChangeLog";
my $fh = IO::File->new($logfile, '<') or $log->fatal_error("can't open ChangeLog file: $logfile");
while (my $line = <$fh>) {
if ($line =~ /^Tag name:\s*([a-zA-Z0-9_. -]*[clmcesm0-9_.-]+)$/ ) {
$log->exit_message("$1");
}
}
}
#-------------------------------------------------------------------------------
sub main {
my %nl_flags;
$nl_flags{'cfgdir'} = dirname(abs_path($0));
my %opts = process_commandline(\%nl_flags);
my $cfgdir = $nl_flags{'cfgdir'};
check_for_perl_utils($cfgdir, \%opts);
$log = namelist_files::LogMessages->new( $ProgName, \%opts ); # global
version($cfgdir) if $opts{'version'};
my $cfg = read_configure_definition($cfgdir, \%opts);
my $physv = config_files::clm_phys_vers->new( $cfg->get('phys') );
my $definition = read_namelist_definition($cfgdir, \%opts, \%nl_flags, $physv);
my $defaults = read_namelist_defaults($cfgdir, \%opts, \%nl_flags, $cfg, $physv);
# List valid values if asked for
list_options(\%opts, $definition, $defaults);
# Validate some of the commandline option values.
validate_options("commandline", $cfg, \%opts);
# Create an empty namelist object.
my $nl = Build::Namelist->new();
check_cesm_inputdata(\%opts, \%nl_flags);
# Read in the env_*.xml files
my %env_xml = read_envxml_case_files( \%opts );
# Process the user inputs
process_namelist_user_input(\%opts, \%nl_flags, $definition, $defaults, $nl, $cfg, \%env_xml, $physv );
# Get any other defaults needed from the namelist defaults file
process_namelist_inline_logic(\%opts, \%nl_flags, $definition, $defaults, $nl, $cfg, \%env_xml, $physv);
# Validate that the entire resultant namelist is valid
$definition->validate($nl);
write_output_files(\%opts, \%nl_flags, $defaults, $nl, $physv);
write_output_real_parameter_file(\%opts, \%nl_flags, $definition, $defaults, $nl, $physv);
if ($opts{'inputdata'}) {
check_input_files($nl, $nl_flags{'inputdata_rootdir'}, $opts{'inputdata'}, $definition);
}
$log->final_exit("Successfully made CLM namelist file");
}
#-------------------------------------------------------------------------------
1;