diff --git a/docs/Development/ReleaseNotes.md b/docs/Development/ReleaseNotes.md index 04b090d87..9acad3afa 100644 --- a/docs/Development/ReleaseNotes.md +++ b/docs/Development/ReleaseNotes.md @@ -74,7 +74,12 @@ This is a major update from VIC 4. The VIC 5.0.0 release aims to have nearly ide - `BINARY_STATE_FILE` (TRUE or FALSE) has been changed to `STATE_FORMAT` (BINARY or ASCII) - `BINARY_OUTPUT` (TRUE or FALSE) has been changed to `OUT_FORMAT` (BINARY or ASCII) -3. Classic Driver Output Variables ([GH#352](https://github.com/UW-Hydro/VIC/pull/352)) +3. State files now include seconds ([GH#464] (https://github.com/UW-Hydro/VIC/pull/464)) + + - There is a new global parameter option, `STATESEC`. This specifies the time step at the end of which state will be saved, in units of seconds. In other words, if you have an hourly time step (3600 sec) and you want to save state at the end of the final time step of the day (which is 86400 seconds long), subtract 3600 from 86400 to get a STATESEC of 82800. This corresponds to the first second of the final time step. State will be saved at the end of that time step. + - When the state save date is appended to state filenames, STATESEC will be included so that the date will have the format YYYYMMDD_SSSSS. + +4. Classic Driver Output Variables ([GH#352](https://github.com/UW-Hydro/VIC/pull/352)) Computation of potential evapotranspiration (PET) has been simplified, reducing the number of output variables from 6 to 1. The following output variables have been removed: diff --git a/samples/global.param.sample.classic.txt b/samples/global.param.sample.classic.txt index a20c025f1..b6877b235 100644 --- a/samples/global.param.sample.classic.txt +++ b/samples/global.param.sample.classic.txt @@ -97,10 +97,11 @@ FROZEN_SOIL FALSE # TRUE = calculate frozen soils. Default = FALSE. # State Files and Parameters ####################################################################### #INIT_STATE (put the initial state path/filename here) # Initial state path/file -#STATENAME (put the path/prefix of output state file here) # Output state file path/prefix. The date (STATEYEAR,STATEMONTH,STATEDAY) will be appended to the prefix automatically in the format yyyymmdd. +#STATENAME (put the path/prefix of output state file here) # Output state file path/prefix. The date (STATEYEAR,STATEMONTH,STATEDAY,STATESEC) will be appended to the prefix automatically in the format yyyymmdd_sssss. #STATEYEAR 2000 # year to save model state #STATEMONTH 12 # month to save model state #STATEDAY 31 # day to save model state +#STATESEC 75600 # second of day to save model state; numbering begins at 0 and STATESEC must be the second of the day corresponding to the beginning of the timestep whose state will be saved #STATE_FORMAT ASCII # BINARY OR ASCII ####################################################################### diff --git a/samples/global.param.sample.image.txt b/samples/global.param.sample.image.txt index f06332b59..8c3cc5570 100644 --- a/samples/global.param.sample.image.txt +++ b/samples/global.param.sample.image.txt @@ -108,10 +108,11 @@ FROZEN_SOIL FALSE # TRUE = calculate frozen soils. Default = FALSE. # State Files and Parameters ####################################################################### #INIT_STATE (put the initial state path/filename here) # Initial state path/file -#STATENAME (put the path/prefix of output state file here) # Output state file path/prefix. The date (STATEYEAR,STATEMONTH,STATEDAY) will be appended to the prefix automatically in the format yyyymmdd. +#STATENAME (put the path/prefix of output state file here) # Output state file path/prefix. The date (STATEYEAR,STATEMONTH,STATEDAY,STATESEC) will be appended to the prefix automatically in the format yyyymmdd_sssss. #STATEYEAR 2000 # year to save model state #STATEMONTH 12 # month to save model state #STATEDAY 31 # day to save model state +#STATESEC 75600 # second of day to save model state; numbering begins at 0 and STATESEC must be the second of the day corresponding to the beginning of the timestep whose state will be saved #STATE_FORMAT NETCDF4_CLASSIC # State file format, valid options: NETCDF3_CLASSIC, NETCDF3_64BIT_OFFSET, NETCDF4_CLASSIC, NETCDF4 ####################################################################### diff --git a/vic/drivers/cesm/src/vic_populate_model_state.c b/vic/drivers/cesm/src/vic_populate_model_state.c index 3c77534a3..62bb24d2b 100644 --- a/vic/drivers/cesm/src/vic_populate_model_state.c +++ b/vic/drivers/cesm/src/vic_populate_model_state.c @@ -33,6 +33,7 @@ void vic_populate_model_state(char *runtype_str) { extern all_vars_struct *all_vars; + extern lake_con_struct *lake_con; extern domain_struct local_domain; extern option_struct options; extern soil_con_struct *soil_con; @@ -40,19 +41,13 @@ vic_populate_model_state(char *runtype_str) extern filenames_struct filenames; - double surf_temp; size_t i; - size_t nveg; unsigned short int runtype; - debug("In vic_restore"); + debug("In vic_populate_model_state"); runtype = start_type_from_char(runtype_str); - // read first forcing timestep (used in restoring model state) - // reset the forcing offset to what it was before - vic_force(); - // read the model state from the netcdf file if there is one if (runtype == CESM_RUNTYPE_RESTART || runtype == CESM_RUNTYPE_BRANCH) { // Get restart file from rpointer file @@ -64,11 +59,20 @@ vic_populate_model_state(char *runtype_str) else if (runtype == CESM_RUNTYPE_CLEANSTART) { // run type is clean start for (i = 0; i < local_domain.ncells_active; i++) { - // TBD: do something sensible for surf_temp - surf_temp = 0.; - nveg = veg_con[i][0].vegetat_type_num; - initialize_model_state(&(all_vars[i]), nveg, options.Nnode, - surf_temp, &(soil_con[i]), veg_con[i]); + generate_default_state(&(all_vars[i]), &(soil_con[i]), veg_con[i]); + if (options.LAKES) { + generate_default_lake_state(&(all_vars[i]), &(soil_con[i]), + lake_con[i]); + } + } + } + + // compute those state variables that are derived from the others + for (i = 0; i < local_domain.ncells_active; i++) { + compute_derived_state_vars(&(all_vars[i]), &(soil_con[i]), veg_con[i]); + if (options.LAKES) { + compute_derived_lake_dimensions(&(all_vars[i].lake_var), + lake_con[i]); } } } diff --git a/vic/drivers/classic/include/vic_driver_classic.h b/vic/drivers/classic/include/vic_driver_classic.h index 166b355f1..7b69d3d99 100644 --- a/vic/drivers/classic/include/vic_driver_classic.h +++ b/vic/drivers/classic/include/vic_driver_classic.h @@ -39,6 +39,7 @@ void alloc_veg_hist(int nrecs, int nveg, veg_hist_struct ***veg_hist); void calc_netlongwave(double *, double, double, double); double calc_netshort(double, int, double, double *); void check_files(filep_struct *, filenames_struct *); +bool check_save_state_flag(dmy_struct *, size_t); FILE *check_state_file(char *, size_t, size_t, int *); void close_files(filep_struct *, out_data_file_struct *, filenames_struct *); size_t count_n_outfiles(FILE *gp); @@ -53,10 +54,6 @@ void get_force_type(char *, int, int *); void get_global_param(FILE *); void init_output_list(out_data_struct *, int, char *, int, double); void initialize_forcing_files(void); -int initialize_model_state(all_vars_struct *, global_param_struct *, - filep_struct, size_t, size_t, size_t, double, - soil_con_struct *, veg_con_struct *, - lake_con_struct); void make_in_and_outfiles(filep_struct *, filenames_struct *, soil_con_struct *, out_data_file_struct *); FILE *open_state_file(global_param_struct *, filenames_struct, int, int); @@ -66,7 +63,7 @@ void read_atmos_data(FILE *, global_param_struct, int, int, double **, double ***); double **read_forcing_data(FILE **, global_param_struct, double ****); void read_initial_model_state(FILE *, all_vars_struct *, int, int, int, - soil_con_struct *); + soil_con_struct *, lake_con_struct); lake_con_struct read_lakeparam(FILE *, soil_con_struct, veg_con_struct *); void read_snowband(FILE *, soil_con_struct *); soil_con_struct read_soilparam(FILE *, char *, char *); @@ -75,6 +72,9 @@ veg_con_struct *read_vegparam(FILE *, int, size_t); out_data_file_struct *set_output_defaults(out_data_struct *); void vic_force(atmos_data_struct *, dmy_struct *, FILE **, veg_con_struct *, veg_hist_struct **, soil_con_struct *); +void vic_populate_model_state(all_vars_struct *, filep_struct, size_t, + soil_con_struct *, veg_con_struct *, + lake_con_struct); void write_data(out_data_file_struct *, out_data_struct *, dmy_struct *, double); void write_forcing_file(atmos_data_struct *, int, out_data_file_struct *, diff --git a/vic/drivers/classic/src/check_save_state_flag.c b/vic/drivers/classic/src/check_save_state_flag.c new file mode 100644 index 000000000..fa600f977 --- /dev/null +++ b/vic/drivers/classic/src/check_save_state_flag.c @@ -0,0 +1,49 @@ +/****************************************************************************** + * @section DESCRIPTION + * + * Function to check whether model state should be saved for the current + * time step + * + * @section LICENSE + * + * The Variable Infiltration Capacity (VIC) macroscale hydrological model + * Copyright (C) 2014 The Land Surface Hydrology Group, Department of Civil + * and Environmental Engineering, University of Washington. + * + * The VIC model is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + *****************************************************************************/ + +#include + +/****************************************************************************** + * @brief Function to check whether model state should be saved for the + * current time step + *****************************************************************************/ +bool +check_save_state_flag(dmy_struct *dmy, + size_t current) +{ + extern global_param_struct global_param; + + if (dmy[current].year == global_param.stateyear && + dmy[current].month == global_param.statemonth && + dmy[current].day == global_param.stateday && + dmy[current].dayseconds == global_param.statesec) { + return true; + } + else { + return false; + } +} diff --git a/vic/drivers/classic/src/display_current_settings.c b/vic/drivers/classic/src/display_current_settings.c index fe83275f7..0fcb33b0f 100644 --- a/vic/drivers/classic/src/display_current_settings.c +++ b/vic/drivers/classic/src/display_current_settings.c @@ -417,6 +417,7 @@ display_current_settings(int mode) fprintf(LOG_DEST, "STATEYEAR\t\t%d\n", global_param.stateyear); fprintf(LOG_DEST, "STATEMONTH\t\t%d\n", global_param.statemonth); fprintf(LOG_DEST, "STATEDAY\t\t%d\n", global_param.stateday); + fprintf(LOG_DEST, "STATESEC\t\t%u\n", global_param.statesec); if (options.STATE_FORMAT == BINARY) { fprintf(LOG_DEST, "STATE_FORMAT\tBINARY\n"); } diff --git a/vic/drivers/classic/src/get_global_param.c b/vic/drivers/classic/src/get_global_param.c index b0b1a58f3..a7657b71c 100644 --- a/vic/drivers/classic/src/get_global_param.c +++ b/vic/drivers/classic/src/get_global_param.c @@ -416,6 +416,9 @@ get_global_param(FILE *gp) else if (strcasecmp("STATEDAY", optstr) == 0) { sscanf(cmdstr, "%*s %hu", &global_param.stateday); } + else if (strcasecmp("STATESEC", optstr) == 0) { + sscanf(cmdstr, "%*s %u", &global_param.statesec); + } else if (strcasecmp("STATE_FORMAT", optstr) == 0) { sscanf(cmdstr, "%*s %s", flgstr); if (strcasecmp("BINARY", flgstr) == 0) { @@ -1397,32 +1400,36 @@ get_global_param(FILE *gp) if (global_param.stateyear == 0 || global_param.statemonth == 0 || global_param.stateday == 0) { log_err("Incomplete specification of the date to save state " - "for state file (%s). Specified date (yyyy-mm-dd): " - "%04d-%02d-%02d Make sure STATEYEAR, STATEMONTH, and " - "STATEDAY are set correctly in your global parameter " + "for state file (%s).\nSpecified date (yyyy-mm-dd-sssss): " + "%04d-%02d-%02d-%05u\nMake sure STATEYEAR, STATEMONTH, " + "and STATEDAY are set correctly in your global parameter " "file.", filenames.statefile, global_param.stateyear, - global_param.statemonth, global_param.stateday); + global_param.statemonth, global_param.stateday, + global_param.statesec); } // Check for month, day in range make_lastday(global_param.stateyear, global_param.calendar, lastday); if (global_param.stateday > lastday[global_param.statemonth - 1] || - global_param.statemonth > MONTHS_PER_YEAR || global_param.statemonth < 1 || - global_param.stateday < 1) { - log_err("Unusual specification of the date to save state for " - "state file (%s). Specified date (yyyy-mm-dd): " - "%04d-%02d-%02d Make sure STATEYEAR, STATEMONTH, and " - "STATEDAY are set correctly in your global parameter " - "file.", filenames.statefile, global_param.stateyear, - global_param.statemonth, global_param.stateday); + global_param.statemonth > MONTHS_PER_YEAR || + global_param.stateday < 1 || global_param.stateday > 31 || + global_param.statesec > SEC_PER_DAY) { + log_err("Unusual specification of the date to save state " + "for state file (%s).\nSpecified date (yyyy-mm-dd-sssss): " + "%04d-%02d-%02d-%05u\nMake sure STATEYEAR, STATEMONTH, " + "STATEDAY and STATESEC are set correctly in your global " + "parameter file.", filenames.statefile, + global_param.stateyear, global_param.statemonth, + global_param.stateday, global_param.statesec); } } // Set the statename here to be able to compare with INIT_STATE name if (options.SAVE_STATE) { - sprintf(filenames.statefile, "%s_%04i%02i%02i", filenames.statefile, - global_param.stateyear, global_param.statemonth, - global_param.stateday); + sprintf(filenames.statefile, "%s_%04i%02i%02i_%05u", + filenames.statefile, global_param.stateyear, + global_param.statemonth, global_param.stateday, + global_param.statesec); } if (options.INIT_STATE && options.SAVE_STATE && (strcmp(filenames.init_state, filenames.statefile) == 0)) { diff --git a/vic/drivers/classic/src/initialize_model_state.c b/vic/drivers/classic/src/initialize_model_state.c deleted file mode 100644 index ea3354a5f..000000000 --- a/vic/drivers/classic/src/initialize_model_state.c +++ /dev/null @@ -1,693 +0,0 @@ -/****************************************************************************** - * @section DESCRIPTION - * - * This routine initializes the model state (energy balance, water balance, and - * snow components). - * - * If a state file is provided to the model than its - * contents are checked to see if it agrees with the current simulation set-up, - * if so it is used to initialize the model state. If no state file is - * provided the model initializes all variables with defaults and the user - * should expect to throw out the beginning of the simulation period as model - * start-up. - * - * @section LICENSE - * - * The Variable Infiltration Capacity (VIC) macroscale hydrological model - * Copyright (C) 2014 The Land Surface Hydrology Group, Department of Civil - * and Environmental Engineering, University of Washington. - * - * The VIC model is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * as published by the Free Software Foundation; either version 2 - * of the License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License along with - * this program; if not, write to the Free Software Foundation, Inc., - * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. - *****************************************************************************/ - -#include - -/****************************************************************************** - * @brief Initialize the model state (energy balance, water balance, and - * snow components). - *****************************************************************************/ -int -initialize_model_state(all_vars_struct *all_vars, - global_param_struct *global_param, - filep_struct filep, - size_t cellnum, - size_t Nveg, - size_t Nnodes, - double surf_temp, - soil_con_struct *soil_con, - veg_con_struct *veg_con, - lake_con_struct lake_con) -{ - extern option_struct options; - extern parameters_struct param; - - char FIRST_VEG; - size_t veg, index; - size_t lidx; - double tmp_moist[MAX_LAYERS]; - double tmp_runoff; - size_t band; - size_t frost_area; - int ErrorFlag; - double Cv; - double Zsum, dp; - double tmpdp, tmpadj, Bexp; - double Tair; - double tmp; - double moist[MAX_VEG][MAX_BANDS][MAX_LAYERS]; - double ice[MAX_VEG][MAX_BANDS][MAX_LAYERS][MAX_FROST_AREAS - ]; - double surf_swq; - double pack_swq; - double dt_thresh; - int tmp_lake_idx; - - cell_data_struct **cell; - energy_bal_struct **energy; - lake_var_struct *lake_var; - snow_data_struct **snow; - veg_var_struct **veg_var; - - cell = all_vars->cell; - energy = all_vars->energy; - lake_var = &all_vars->lake_var; - snow = all_vars->snow; - veg_var = all_vars->veg_var; - - // Initialize soil depths - dp = soil_con->dp; - - FIRST_VEG = true; - - // increase initial soil surface temperature if air is very cold - Tair = surf_temp; - if (surf_temp < -1.) { - surf_temp = -1.; - } - - /******************************************** - Initialize all snow pack variables - - some may be reset if state file present - ********************************************/ - - initialize_snow(snow, Nveg); - - /******************************************** - Initialize all soil layer variables - - some may be reset if state file present - ********************************************/ - - initialize_soil(cell, soil_con, Nveg); - - /******************************************** - Initialize all vegetation variables - - some may be reset if state file present - ********************************************/ - - initialize_veg(veg_var, Nveg); - - /******************************************** - Initialize all lake variables - ********************************************/ - - if (options.LAKES) { - tmp_lake_idx = lake_con.lake_idx; - if (tmp_lake_idx < 0) { - tmp_lake_idx = 0; - } - ErrorFlag = - initialize_lake(lake_var, lake_con, soil_con, - &(cell[tmp_lake_idx][0]), 0); - if (ErrorFlag == ERROR) { - return(ErrorFlag); - } - } - - /******************************************** - Initialize all spatial frost variables - ********************************************/ - - for (frost_area = 0; frost_area < options.Nfrost; frost_area++) { - if (options.Nfrost == 1) { - soil_con->frost_fract[frost_area] = 1.; - } - else if (options.Nfrost == 2) { - soil_con->frost_fract[frost_area] = 0.5; - } - else { - soil_con->frost_fract[frost_area] = 1. / (options.Nfrost - 1); - if (frost_area == 0 || frost_area == options.Nfrost - 1) { - soil_con->frost_fract[frost_area] /= 2.; - } - } - } - - /******************************************************** - Compute grid cell fractions for all subareas used in - spatial distribution of soil frost routines. - ********************************************************/ - - /************************************************************************ - CASE 1: Not using quick ground heat flux, and initial conditions files - provided - ************************************************************************/ - - if (options.INIT_STATE) { - read_initial_model_state(filep.init_state, all_vars, Nveg, - options.SNOW_BAND, cellnum, soil_con); - - /******Check that soil moisture does not exceed maximum allowed************/ - for (veg = 0; veg <= Nveg; veg++) { - for (band = 0; band < options.SNOW_BAND; band++) { - for (lidx = 0; lidx < options.Nlayer; lidx++) { - if (cell[veg][band].layer[lidx].moist > - soil_con->max_moist[lidx]) { - log_warn("Initial soil moisture (%f mm) exceeds " - "maximum (%f mm) in layer %zu for veg tile " - "%zu and snow band%zu. Resetting to maximum.", - cell[veg][band].layer[lidx].moist, - soil_con->max_moist[lidx], lidx, veg, band); - for (frost_area = 0; - frost_area < options.Nfrost; - frost_area++) { - cell[veg][band].layer[lidx].ice[frost_area] *= - soil_con->max_moist[lidx] / - cell[veg][band].layer[lidx].moist; - } - cell[veg][band].layer[lidx].moist = - soil_con->max_moist[lidx]; - } - - for (frost_area = 0; - frost_area < options.Nfrost; - frost_area++) { - if (cell[veg][band].layer[lidx].ice[frost_area] > - cell[veg][band].layer[lidx].moist) { - cell[veg][band].layer[lidx].ice[frost_area] = - cell[veg][band].layer[lidx].moist; - } - } - tmp_moist[lidx] = cell[veg][band].layer[lidx].moist; - } - compute_runoff_and_asat(soil_con, tmp_moist, 0, - &(cell[veg][band].asat), &tmp_runoff); - } - - // Override possible bad values of soil moisture under lake coming from state file - // (ideally we wouldn't store these in the state file in the first place) - if (options.LAKES && (int) veg == lake_con.lake_idx) { - for (lidx = 0; lidx < options.Nlayer; lidx++) { - lake_var->soil.layer[lidx].moist = - soil_con->max_moist[lidx]; - for (frost_area = 0; - frost_area < options.Nfrost; - frost_area++) { - if (lake_var->soil.layer[lidx].ice[frost_area] > - lake_var->soil.layer[lidx].moist) { - lake_var->soil.layer[lidx].ice[frost_area] = - lake_var->soil.layer[lidx].moist; - } - } - } - } - } - - - /****** initialize moist and ice ************/ - for (veg = 0; veg <= Nveg; veg++) { - // Initialize soil for existing vegetation types - Cv = veg_con[veg].Cv; - - if (Cv > 0) { - for (band = 0; band < options.SNOW_BAND; band++) { - for (lidx = 0; lidx < options.Nlayer; lidx++) { - moist[veg][band][lidx] = - cell[veg][band].layer[lidx].moist; - - for (frost_area = 0; - frost_area < options.Nfrost; - frost_area++) { - ice[veg][band][lidx][frost_area] = - cell[veg][band].layer[lidx].ice[frost_area]; - } - } - } - } - } - - /******Check that snow pack terms are self-consistent************/ - for (veg = 0; veg <= Nveg; veg++) { - for (band = 0; band < options.SNOW_BAND; band++) { - if (snow[veg][band].swq > param.SNOW_MAX_SURFACE_SWE) { - pack_swq = snow[veg][band].swq - param.SNOW_MAX_SURFACE_SWE; - surf_swq = param.SNOW_MAX_SURFACE_SWE; - } - else { - pack_swq = 0; - surf_swq = snow[veg][band].swq; - snow[veg][band].pack_temp = 0; - } - if (snow[veg][band].surf_water > - param.SNOW_LIQUID_WATER_CAPACITY * - surf_swq) { - snow[veg][band].pack_water += snow[veg][band].surf_water - - (param. - SNOW_LIQUID_WATER_CAPACITY * - surf_swq); - snow[veg][band].surf_water = - param.SNOW_LIQUID_WATER_CAPACITY * - surf_swq; - } - if (snow[veg][band].pack_water > - param.SNOW_LIQUID_WATER_CAPACITY * - pack_swq) { - snow[veg][band].pack_water = - param.SNOW_LIQUID_WATER_CAPACITY * - pack_swq; - } - } - } - } - - /************************************************************************ - CASE 2: Initialize soil if using quick heat flux, and no initial - soil properties file given - ************************************************************************/ - - else if (options.QUICK_FLUX) { - Nnodes = options.Nnode; - - /* Initialize soil node thicknesses */ - soil_con->dz_node[0] = soil_con->depth[0]; - soil_con->dz_node[1] = soil_con->depth[0]; - soil_con->dz_node[2] = 2. * (dp - 1.5 * soil_con->depth[0]); - soil_con->Zsum_node[0] = 0; - soil_con->Zsum_node[1] = soil_con->depth[0]; - soil_con->Zsum_node[2] = dp; - - for (veg = 0; veg <= Nveg; veg++) { - // Initialize soil for existing vegetation types - Cv = veg_con[veg].Cv; - - if (Cv > 0) { - for (band = 0; band < options.SNOW_BAND; band++) { - /* Initialize soil node temperatures */ - energy[veg][band].T[0] = surf_temp; - energy[veg][band].T[1] = surf_temp; - energy[veg][band].T[2] = soil_con->avg_temp; - - /* Initialize soil layer moisture and ice contents */ - for (lidx = 0; lidx < options.Nlayer; lidx++) { - moist[veg][band][lidx] = - cell[veg][band].layer[lidx].moist; - for (frost_area = 0; - frost_area < options.Nfrost; - frost_area++) { - ice[veg][band][lidx][frost_area] = 0.; - } - } - } - } - } - } - - /***************************************************************** - CASE 3: Initialize Energy Balance Variables if not using quick - ground heat flux, and no Initial Condition File Given - *****************************************************************/ - else if (!options.QUICK_FLUX) { - for (veg = 0; veg <= Nveg; veg++) { - // Initialize soil for existing vegetation types - Cv = veg_con[veg].Cv; - - if (Cv > 0) { - for (band = 0; band < options.SNOW_BAND; band++) { - if (!options.EXP_TRANS) { - /* Initialize soil node temperatures and thicknesses - Nodes set at surface, the depth of the first layer, - twice the depth of the first layer, and at the - damping depth. Extra nodes are placed equal distance - between the damping depth and twice the depth of the - first layer. */ - - soil_con->dz_node[0] = soil_con->depth[0]; - soil_con->dz_node[1] = soil_con->depth[0]; - soil_con->dz_node[2] = soil_con->depth[0]; - - soil_con->Zsum_node[0] = 0; - soil_con->Zsum_node[1] = soil_con[0].depth[0]; - Zsum = 2. * soil_con[0].depth[0]; - soil_con->Zsum_node[2] = Zsum; - tmpdp = dp - soil_con[0].depth[0] * 2.5; - tmpadj = 3.5; - for (index = 3; index < Nnodes - 1; index++) { - if (FIRST_VEG) { - soil_con->dz_node[index] = tmpdp / - (((double) Nnodes - - tmpadj)); - } - Zsum += (soil_con->dz_node[index] + - soil_con->dz_node[index - 1]) / 2.; - soil_con->Zsum_node[index] = Zsum; - } - energy[veg][band].T[0] = surf_temp; - for (index = 1; index < Nnodes; index++) { - energy[veg][band].T[index] = soil_con->avg_temp; - } - if (FIRST_VEG) { - FIRST_VEG = false; - soil_con->dz_node[Nnodes - 1] = (dp - Zsum - - soil_con->dz_node[ - Nnodes - 2] / - 2.) * 2.; - Zsum += (soil_con->dz_node[Nnodes - 2] + - soil_con->dz_node[Nnodes - 1]) / 2.; - soil_con->Zsum_node[Nnodes - 1] = Zsum; - if ((int) (Zsum * MM_PER_M + 0.5) != - (int) (dp * MM_PER_M + 0.5)) { - log_err("Sum of thermal node thicknesses (%f) " - "do not equal dp (%f), check " - "initialization procedure", Zsum, dp); - } - } - } - else { /* exponential grid transformation, EXP_TRANS = TRUE*/ - /*calculate exponential function parameter */ - if (FIRST_VEG) { - Bexp = logf(dp + 1.) / (double) (Nnodes - 1); // to force Zsum=dp at bottom node - - /* validate Nnodes by requiring that there be at - least 3 nodes in the top 50cm */ - if (Nnodes < 5 * logf(dp + 1.) + 1) { - log_err("The number of soil thermal nodes " - "(%zu) is too small for the supplied " - "damping depth (%f) with EXP_TRANS " - "set to TRUE, leading to fewer than 3 " - "nodes in the top 50 cm of the soil " - "column. For EXP_TRANS=TRUE, Nnodes " - "and dp must follow the relationship:" - "\n5*ln(dp+1)Zsum_node[index] = - expf(Bexp * index) - 1.; - } - if (soil_con->Zsum_node[0] > soil_con->depth[0]) { - log_err("Depth of first thermal node (%f)" - "initialize_model_state is greater " - "than depth of first soil layer (%f); " - "increase the number of nodes or " - "decrease the thermal damping depth " - "dp (%f)", soil_con->Zsum_node[0], - soil_con->depth[0], dp); - } - } - - // top node - index = 0; - if (FIRST_VEG) { - soil_con->dz_node[index] = - soil_con->Zsum_node[index + - 1] - - soil_con->Zsum_node[index]; - } - energy[veg][band].T[index] = surf_temp; - // middle nodes - for (index = 1; index < Nnodes - 1; index++) { - if (FIRST_VEG) { - soil_con->dz_node[index] = - (soil_con->Zsum_node[index + - 1] - - soil_con->Zsum_node[index]) / 2. + - (soil_con->Zsum_node[index] - - soil_con->Zsum_node[index - 1]) / 2.; - } - energy[veg][band].T[index] = soil_con->avg_temp; - } - // bottom node - index = Nnodes - 1; - if (FIRST_VEG) { - soil_con->dz_node[index] = - soil_con->Zsum_node[index] - - soil_con->Zsum_node[index - 1]; - } - energy[veg][band].T[index] = soil_con->avg_temp; - } // end if !EXP_TRANS - - // initialize moisture and ice for each soil layer - for (lidx = 0; lidx < options.Nlayer; lidx++) { - moist[veg][band][lidx] = - cell[veg][band].layer[lidx].moist; - for (frost_area = 0; - frost_area < options.Nfrost; - frost_area++) { - ice[veg][band][lidx][frost_area] = 0.; - } - } - } - } - } - } - - /********************************* - CASE 4: Unknown option - *********************************/ - else { - for (veg = 0; veg <= Nveg; veg++) { - // Initialize soil for existing vegetation types - Cv = veg_con[veg].Cv; - - if (Cv > 0) { - for (band = 0; band < options.SNOW_BAND; band++) { - // Initialize soil for existing snow elevation bands - if (soil_con->AreaFract[band] > 0.) { - for (index = 0; index < options.Nlayer; index++) { - soil_con->dz_node[index] = 1.; - } - } - } - } - } - } - - /****************************************** - Initialize soil thermal node properties - ******************************************/ - - FIRST_VEG = true; - for (veg = 0; veg <= Nveg; veg++) { - // Initialize soil for existing vegetation types - Cv = veg_con[veg].Cv; - - if (Cv > 0) { - for (band = 0; band < options.SNOW_BAND; band++) { - // Initialize soil for existing snow elevation bands - if (soil_con->AreaFract[band] > 0.) { - /** Set soil properties for all soil nodes **/ - if (FIRST_VEG) { - FIRST_VEG = false; - set_node_parameters(soil_con->Zsum_node, - soil_con->max_moist_node, - soil_con->expt_node, - soil_con->bubble_node, - soil_con->alpha, soil_con->beta, - soil_con->gamma, soil_con->depth, - soil_con->max_moist, soil_con->expt, - soil_con->bubble, - Nnodes, options.Nlayer); - } - - /* set soil moisture properties for all soil thermal nodes */ - ErrorFlag = - distribute_node_moisture_properties( - energy[veg][band].moist, - energy[veg][ - band].ice, - energy[veg][ - band].kappa_node, - energy[veg][ - band].Cs_node, - soil_con->Zsum_node, - energy[veg][ - band].T, - soil_con->max_moist_node, - soil_con->expt_node, - soil_con->bubble_node, - moist[veg][ - band], - soil_con->depth, - soil_con->soil_dens_min, - soil_con->bulk_dens_min, - soil_con->quartz, - soil_con->soil_density, - soil_con->bulk_density, - soil_con->organic, - Nnodes, options.Nlayer, - soil_con->FS_ACTIVE); - if (ErrorFlag == ERROR) { - return (ErrorFlag); - } - - /* Check node spacing v time step */ - - /* (note this is only approximate since heat capacity and - conductivity can change considerably during the - simulation depending on soil moisture and ice content) */ - if ((options.FROZEN_SOIL && - !options.QUICK_FLUX) && !options.IMPLICIT) { - dt_thresh = 0.5 * energy[veg][band].Cs_node[1] / - energy[veg][band].kappa_node[1] * - pow((soil_con->dz_node[1]), - 2); // in seconds - if (global_param->dt > dt_thresh) { - log_err("You are currently running " - "FROZEN SOIL with an explicit method " - "(IMPLICIT is set to FALSE). For the " - "explicit method to be stable, time step " - "%f seconds is too large for the given " - "thermal node spacing %f m, soil heat " - "capacity %f J/m3/K, and soil thermal " - "conductivity %f J/m/s/K. Either set " - "IMPLICIT to TRUE in your global " - "parameter file (this is the recommended " - "action), or decrease time step length to " - "<= %f seconds, or decrease the number of " - "soil thermal nodes.", global_param->dt, - soil_con->dz_node[1], - energy[veg][band].Cs_node[1], - energy[veg][band].kappa_node[1], dt_thresh); - } - } - - /* initialize layer moistures and ice contents */ - for (lidx = 0; lidx < options.Nlayer; lidx++) { - cell[veg][band].layer[lidx].moist = - moist[veg][band][lidx]; - for (frost_area = 0; - frost_area < options.Nfrost; - frost_area++) { - cell[veg][band].layer[lidx].ice[frost_area] = - ice[veg][band][lidx][frost_area]; - } - } - if (options.QUICK_FLUX) { - ErrorFlag = - estimate_layer_ice_content_quick_flux( - cell[veg][band].layer, - soil_con->depth, soil_con->dp, - energy[ - veg][band].T[0], energy[veg][band].T[1], - soil_con->avg_temp, soil_con->max_moist, - soil_con->expt, soil_con->bubble, - soil_con->frost_fract, soil_con->frost_slope, - soil_con->FS_ACTIVE); - } - else { - ErrorFlag = estimate_layer_ice_content( - cell[veg][band].layer, - soil_con->Zsum_node, - energy[veg][band].T, - soil_con->depth, - soil_con->max_moist, - soil_con->expt, - soil_con->bubble, - soil_con->frost_fract, - soil_con->frost_slope, - Nnodes, options.Nlayer, - soil_con->FS_ACTIVE); - } - - /* Find freezing and thawing front depths */ - if (!options.QUICK_FLUX && soil_con->FS_ACTIVE) { - find_0_degree_fronts(&energy[veg][band], - soil_con->Zsum_node, - energy[veg][band].T, Nnodes); - } - } - } - } - } - - // initialize miscellaneous energy balance terms - for (veg = 0; veg <= Nveg; veg++) { - for (band = 0; band < options.SNOW_BAND; band++) { - /* Set fluxes to 0 */ - energy[veg][band].advected_sensible = 0.0; - energy[veg][band].advection = 0.0; - energy[veg][band].AtmosError = 0.0; - energy[veg][band].AtmosLatent = 0.0; - energy[veg][band].AtmosLatentSub = 0.0; - energy[veg][band].AtmosSensible = 0.0; - energy[veg][band].canopy_advection = 0.0; - energy[veg][band].canopy_latent = 0.0; - energy[veg][band].canopy_latent_sub = 0.0; - energy[veg][band].canopy_refreeze = 0.0; - energy[veg][band].canopy_sensible = 0.0; - energy[veg][band].deltaCC = 0.0; - energy[veg][band].deltaH = 0.0; - energy[veg][band].error = 0.0; - energy[veg][band].fusion = 0.0; - energy[veg][band].grnd_flux = 0.0; - energy[veg][band].latent = 0.0; - energy[veg][band].latent_sub = 0.0; - energy[veg][band].longwave = 0.0; - energy[veg][band].LongOverIn = 0.0; - energy[veg][band].LongUnderIn = 0.0; - energy[veg][band].LongUnderOut = 0.0; - energy[veg][band].melt_energy = 0.0; - energy[veg][band].NetLongAtmos = 0.0; - energy[veg][band].NetLongOver = 0.0; - energy[veg][band].NetLongUnder = 0.0; - energy[veg][band].NetShortAtmos = 0.0; - energy[veg][band].NetShortGrnd = 0.0; - energy[veg][band].NetShortOver = 0.0; - energy[veg][band].NetShortUnder = 0.0; - energy[veg][band].out_long_canopy = 0.0; - energy[veg][band].out_long_surface = 0.0; - energy[veg][band].refreeze_energy = 0.0; - energy[veg][band].sensible = 0.0; - energy[veg][band].shortwave = 0.0; - energy[veg][band].ShortOverIn = 0.0; - energy[veg][band].ShortUnderIn = 0.0; - energy[veg][band].snow_flux = 0.0; - /* Initial estimate of LongUnderOut for use by snow_intercept() */ - tmp = energy[veg][band].T[0] + CONST_TKFRZ; - energy[veg][band].LongUnderOut = calc_outgoing_longwave(tmp, - param.EMISS_SNOW); - energy[veg][band].Tfoliage = Tair + soil_con->Tfactor[band]; - } - } - - // initialize Tfallback counters - for (veg = 0; veg <= Nveg; veg++) { - for (band = 0; band < options.SNOW_BAND; band++) { - energy[veg][band].Tfoliage_fbcount = 0; - energy[veg][band].Tcanopy_fbcount = 0; - energy[veg][band].Tsurf_fbcount = 0; - for (index = 0; index < Nnodes - 1; index++) { - energy[veg][band].T_fbcount[index] = 0; - } - } - } - - return(0); -} diff --git a/vic/drivers/classic/src/read_initial_model_state.c b/vic/drivers/classic/src/read_initial_model_state.c index c542b379f..e27dcff82 100644 --- a/vic/drivers/classic/src/read_initial_model_state.c +++ b/vic/drivers/classic/src/read_initial_model_state.c @@ -42,29 +42,30 @@ read_initial_model_state(FILE *init_state, int Nveg, int Nbands, int cellnum, - soil_con_struct *soil_con) + soil_con_struct *soil_con, + lake_con_struct lake_con) { - extern option_struct options; + extern option_struct options; - char tmpstr[MAXSTRING]; - char tmpchar; - int veg, iveg; - int band, iband; - size_t lidx; - size_t nidx; - int tmp_cellnum; - int tmp_Nveg; - int tmp_Nband; - int tmp_char; - int byte, Nbytes; - int node; - size_t frost_area; + char tmpstr[MAXSTRING]; + char tmpchar; + int veg, iveg; + int band, iband; + size_t lidx; + size_t nidx; + int tmp_cellnum; + int tmp_Nveg; + int tmp_Nband; + int tmp_char; + int byte, Nbytes; + int node; + size_t frost_area; - cell_data_struct **cell; - snow_data_struct **snow; - energy_bal_struct **energy; - veg_var_struct **veg_var; - lake_var_struct *lake_var; + cell_data_struct **cell; + snow_data_struct **snow; + energy_bal_struct **energy; + veg_var_struct **veg_var; + lake_var_struct *lake_var; cell = all_vars->cell; veg_var = all_vars->veg_var; @@ -688,4 +689,58 @@ read_initial_model_state(FILE *init_state, } } } + + // Check that soil moisture does not exceed maximum allowed + for (veg = 0; veg <= Nveg; veg++) { + for (band = 0; band < Nbands; band++) { + for (lidx = 0; lidx < options.Nlayer; lidx++) { + if (cell[veg][band].layer[lidx].moist > + soil_con->max_moist[lidx]) { + log_warn("Initial soil moisture (%f mm) exceeds " + "maximum (%f mm) in layer %zu for veg tile " + "%d and snow band %d. Resetting to maximum.", + cell[veg][band].layer[lidx].moist, + soil_con->max_moist[lidx], lidx, veg, band); + for (frost_area = 0; + frost_area < options.Nfrost; + frost_area++) { + cell[veg][band].layer[lidx].ice[frost_area] *= + soil_con->max_moist[lidx] / + cell[veg][band].layer[lidx].moist; + } + cell[veg][band].layer[lidx].moist = + soil_con->max_moist[lidx]; + } + + for (frost_area = 0; + frost_area < options.Nfrost; + frost_area++) { + if (cell[veg][band].layer[lidx].ice[frost_area] > + cell[veg][band].layer[lidx].moist) { + cell[veg][band].layer[lidx].ice[frost_area] = + cell[veg][band].layer[lidx].moist; + } + } + } + } + + // Override possible bad values of soil moisture under lake coming from state file + // (ideally we wouldn't store these in the state file in the first place) + if (options.LAKES && (int) veg == lake_con.lake_idx) { + for (lidx = 0; lidx < options.Nlayer; lidx++) { + lake_var->soil.layer[lidx].moist = + soil_con->max_moist[lidx]; + for (frost_area = 0; + frost_area < options.Nfrost; + frost_area++) { + if (lake_var->soil.layer[lidx].ice[frost_area] > + lake_var->soil.layer[lidx].moist) { + lake_var->soil.layer[lidx].ice[frost_area] = + lake_var->soil.layer[lidx].moist; + } + } + } + } + } + } diff --git a/vic/drivers/classic/src/read_soilparam.c b/vic/drivers/classic/src/read_soilparam.c index eee4d74e3..51f30ac9b 100644 --- a/vic/drivers/classic/src/read_soilparam.c +++ b/vic/drivers/classic/src/read_soilparam.c @@ -62,6 +62,10 @@ read_soilparam(FILE *soilparam, double tmp_moist; double w_avg; soil_con_struct temp; + double Zsum, dp; + double tmpdp, tmpadj, Bexp; + size_t k; + size_t Nnodes; /** Read plain ASCII soil parameter file **/ if ((fscanf(soilparam, "%d", &flag)) != EOF) { @@ -675,6 +679,20 @@ read_soilparam(FILE *soilparam, temp.frost_slope); } } + for (k = 0; k < options.Nfrost; k++) { + if (options.Nfrost == 1) { + temp.frost_fract[k] = 1.; + } + else if (options.Nfrost == 2) { + temp.frost_fract[k] = 0.5; + } + else { + temp.frost_fract[k] = 1. / (options.Nfrost - 1); + if (k == 0 || k == options.Nfrost - 1) { + temp.frost_fract[k] /= 2.; + } + } + } /************************************************* If BASEFLOW = NIJSSEN2001 then convert NIJSSEN2001 @@ -692,6 +710,108 @@ read_soilparam(FILE *soilparam, temp.Ws = temp.Ws / temp.max_moist[layer]; } + // Soil thermal node thicknesses and positions + Nnodes = options.Nnode; + dp = temp.dp; + if (options.QUICK_FLUX) { + /* node thicknesses */ + temp.dz_node[0] = temp.depth[0]; + temp.dz_node[1] = temp.depth[0]; + temp.dz_node[2] = 2. * (dp - 1.5 * temp.depth[0]); + + /* node depths (positions) */ + temp.Zsum_node[0] = 0; + temp.Zsum_node[1] = temp.depth[0]; + temp.Zsum_node[2] = dp; + } + else { + if (!options.EXP_TRANS) { + /* Compute soil node thicknesses + Nodes set at surface, the depth of the first layer, + twice the depth of the first layer, and at the + damping depth. Extra nodes are placed equal distance + between the damping depth and twice the depth of the + first layer. */ + + temp.dz_node[0] = temp.depth[0]; + temp.dz_node[1] = temp.depth[0]; + temp.dz_node[2] = temp.depth[0]; + temp.Zsum_node[0] = 0; + temp.Zsum_node[1] = temp.depth[0]; + Zsum = 2. * temp.depth[0]; + temp.Zsum_node[2] = Zsum; + tmpdp = dp - temp.depth[0] * 2.5; + tmpadj = 3.5; + for (k = 3; k < Nnodes - 1; k++) { + temp.dz_node[k] = tmpdp / (((double) Nnodes - tmpadj)); + Zsum += (temp.dz_node[k] + temp.dz_node[k - 1]) / 2.; + temp.Zsum_node[k] = Zsum; + } + temp.dz_node[Nnodes - + 1] = (dp - Zsum - temp.dz_node[Nnodes - 2] / 2.) * + 2.; + Zsum += (temp.dz_node[Nnodes - 2] + temp.dz_node[Nnodes - 1]) / + 2.; + temp.Zsum_node[Nnodes - 1] = Zsum; + if ((int) (Zsum * MM_PER_M + 0.5) != + (int) (dp * MM_PER_M + 0.5)) { + log_err("Sum of thermal node thicknesses (%f) " + "in initialize_model_state do not " + "equal dp (%f), check initialization " + "procedure", Zsum, dp); + } + } + else { + // exponential grid transformation, EXP_TRANS = TRUE + // calculate exponential function parameter + // to force Zsum=dp at bottom node + Bexp = logf(dp + 1.) / (double) (Nnodes - 1); + // validate Nnodes by requiring that there be at + // least 3 nodes in the top 50cm + if (Nnodes < 5 * logf(dp + 1.) + 1) { + log_err("The number of soil thermal nodes (%zu) " + "is too small for the supplied damping " + "depth (%f) with EXP_TRANS set to " + "TRUE, leading to fewer than 3 nodes " + "in the top 50 cm of the soil column. " + "For EXP_TRANS=TRUE, Nnodes and dp " + "must follow the relationship:\n" + "5*ln(dp+1) temp.depth[0]) { + log_err("Depth of first thermal node (%f) in " + "initialize_model_state is greater " + "than depth of first soil layer (%f); " + "increase the number of nodes or " + "decrease the thermal damping depth " + "dp (%f)", temp.Zsum_node[0], temp.depth[0], dp); + } + + // top node + k = 0; + temp.dz_node[k] = temp.Zsum_node[k + 1] - temp.Zsum_node[k]; + // middle nodes + for (k = 1; k < Nnodes - 1; k++) { + temp.dz_node[k] = + (temp.Zsum_node[k + 1] - temp.Zsum_node[k]) / 2. + + (temp.Zsum_node[k] - temp.Zsum_node[k - 1]) / 2.; + } + // bottom node + k = Nnodes - 1; + temp.dz_node[k] = temp.Zsum_node[k] - temp.Zsum_node[k - 1]; + } // end if !EXP_TRANS + } + /******************************************************************* Calculate grid cell area. ******************************************************************/ diff --git a/vic/drivers/classic/src/vic_classic.c b/vic/drivers/classic/src/vic_classic.c index 334aee104..cc415bcff 100644 --- a/vic/drivers/classic/src/vic_classic.c +++ b/vic/drivers/classic/src/vic_classic.c @@ -205,31 +205,8 @@ main(int argc, Initialize Energy Balance and Snow Variables **************************************************/ - ErrorFlag = initialize_model_state(&all_vars, &global_param, - filep, soil_con.gridcel, - veg_con[0].vegetat_type_num, - options.Nnode, - atmos[0].air_temp[NR], - &soil_con, veg_con, - lake_con); - if (ErrorFlag == ERROR) { - if (options.CONTINUEONERROR) { - // Handle grid cell solution error - log_warn("ERROR: Grid cell %i failed in record %zu so " - "the simulation has not finished. An " - "incomplete output file has been generated, " - "check your inputs before rerunning the " - "simulation.\n", soil_con.gridcel, rec); - break; - } - else { - // Else exit program on cell solution error as in previous versions - log_err("ERROR: Grid cell %i failed in record %zu so " - "the simulation has ended. Check your inputs " - "before rerunning the simulation.\n", - soil_con.gridcel, rec); - } - } + vic_populate_model_state(&all_vars, filep, soil_con.gridcel, + &soil_con, veg_con, lake_con); /** Update Error Handling Structure **/ Error.filep = filep; @@ -285,11 +262,7 @@ main(int argc, (after the final time step of the assigned date) ************************************/ if (filep.statefile != NULL && - (dmy[rec].year == global_param.stateyear && - dmy[rec].month == global_param.statemonth && - dmy[rec].day == global_param.stateday && - (rec + 1 == global_param.nrecs || - dmy[rec + 1].day != global_param.stateday))) { + check_save_state_flag(dmy, rec)) { write_model_state(&all_vars, veg_con->vegetat_type_num, soil_con.gridcel, &filep, &soil_con); } diff --git a/vic/drivers/classic/src/vic_populate_model_state.c b/vic/drivers/classic/src/vic_populate_model_state.c new file mode 100644 index 000000000..0ebb18681 --- /dev/null +++ b/vic/drivers/classic/src/vic_populate_model_state.c @@ -0,0 +1,101 @@ +/****************************************************************************** + * @section DESCRIPTION + * + * This routine initializes the model state (energy balance, water balance, and + * snow components). + * + * If a state file is provided to the model then its + * contents are checked to see if it agrees with the current simulation set-up, + * if so it is used to initialize the model state. If no state file is + * provided the model initializes all variables with defaults and the user + * should expect to throw out the beginning of the simulation period as model + * start-up. + * + * @section LICENSE + * + * The Variable Infiltration Capacity (VIC) macroscale hydrological model + * Copyright (C) 2014 The Land Surface Hydrology Group, Department of Civil + * and Environmental Engineering, University of Washington. + * + * The VIC model is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + *****************************************************************************/ + +#include + +/****************************************************************************** + * @brief Initialize the model state (energy balance, water balance, and + * snow components). + *****************************************************************************/ +void +vic_populate_model_state(all_vars_struct *all_vars, + filep_struct filep, + size_t cellnum, + soil_con_struct *soil_con, + veg_con_struct *veg_con, + lake_con_struct lake_con) +{ + extern option_struct options; + + size_t Nveg; + int tmp_lake_idx; + + cell_data_struct **cell; + energy_bal_struct **energy; + lake_var_struct *lake; + snow_data_struct **snow; + veg_var_struct **veg_var; + + cell = all_vars->cell; + energy = all_vars->energy; + lake = &all_vars->lake_var; + snow = all_vars->snow; + veg_var = all_vars->veg_var; + + Nveg = veg_con[0].vegetat_type_num; + + // Initialize all data structures to 0 + initialize_soil(cell, soil_con, Nveg); + initialize_snow(snow, Nveg); + initialize_veg(veg_var, Nveg); + if (options.LAKES) { + tmp_lake_idx = lake_con.lake_idx; + if (tmp_lake_idx < 0) { + tmp_lake_idx = 0; + } + initialize_lake(lake, lake_con, soil_con, &(cell[tmp_lake_idx][0]), + false); + } + initialize_energy(energy, Nveg); + + // Read initial state from a file if provided + if (options.INIT_STATE) { + read_initial_model_state(filep.init_state, all_vars, Nveg, + options.SNOW_BAND, cellnum, soil_con, + lake_con); + } + else { + // else generate a default state + generate_default_state(all_vars, soil_con, veg_con); + if (options.LAKES) { + generate_default_lake_state(all_vars, soil_con, lake_con); + } + } + + // compute those state variables that are derived from the others + compute_derived_state_vars(all_vars, soil_con, veg_con); + if (options.LAKES) { + compute_derived_lake_dimensions(lake, lake_con); + } +} diff --git a/vic/drivers/image/include/vic_driver_image.h b/vic/drivers/image/include/vic_driver_image.h index 045d62e5c..041b9c157 100644 --- a/vic/drivers/image/include/vic_driver_image.h +++ b/vic/drivers/image/include/vic_driver_image.h @@ -31,6 +31,7 @@ #define VIC_DRIVER "Image" +bool check_save_state_flag(size_t); void display_current_settings(int); void get_forcing_file_info(param_set_struct *param_set, size_t file_num); void get_global_param(FILE *); diff --git a/vic/drivers/image/src/check_save_state_flag.c b/vic/drivers/image/src/check_save_state_flag.c new file mode 100644 index 000000000..6578546f7 --- /dev/null +++ b/vic/drivers/image/src/check_save_state_flag.c @@ -0,0 +1,49 @@ +/****************************************************************************** + * @section DESCRIPTION + * + * Function to check whether model state should be saved for the current + * time step + * + * @section LICENSE + * + * The Variable Infiltration Capacity (VIC) macroscale hydrological model + * Copyright (C) 2014 The Land Surface Hydrology Group, Department of Civil + * and Environmental Engineering, University of Washington. + * + * The VIC model is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + *****************************************************************************/ + +#include + +/****************************************************************************** + * @brief Function to check whether model state should be saved for the + * current time step + *****************************************************************************/ +bool +check_save_state_flag(size_t current) +{ + extern global_param_struct global_param; + extern dmy_struct *dmy; + + if (dmy[current].year == global_param.stateyear && + dmy[current].month == global_param.statemonth && + dmy[current].day == global_param.stateday && + dmy[current].dayseconds == global_param.statesec) { + return true; + } + else { + return false; + } +} diff --git a/vic/drivers/image/src/display_current_settings.c b/vic/drivers/image/src/display_current_settings.c index 43b0d4381..1b566b5d2 100644 --- a/vic/drivers/image/src/display_current_settings.c +++ b/vic/drivers/image/src/display_current_settings.c @@ -409,6 +409,7 @@ display_current_settings(int mode) fprintf(LOG_DEST, "STATEYEAR\t\t%d\n", global_param.stateyear); fprintf(LOG_DEST, "STATEMONTH\t\t%d\n", global_param.statemonth); fprintf(LOG_DEST, "STATEDAY\t\t%d\n", global_param.stateday); + fprintf(LOG_DEST, "STATESEC\t\t%u\n", global_param.statesec); if (options.STATE_FORMAT == NETCDF3_CLASSIC) { fprintf(LOG_DEST, "STATE_FORMAT\t\tNETCDF3_CLASSIC\n"); } diff --git a/vic/drivers/image/src/get_global_param.c b/vic/drivers/image/src/get_global_param.c index 348072caa..f21041316 100644 --- a/vic/drivers/image/src/get_global_param.c +++ b/vic/drivers/image/src/get_global_param.c @@ -388,6 +388,9 @@ get_global_param(FILE *gp) else if (strcasecmp("STATEDAY", optstr) == 0) { sscanf(cmdstr, "%*s %hu", &global_param.stateday); } + else if (strcasecmp("STATESEC", optstr) == 0) { + sscanf(cmdstr, "%*s %u", &global_param.statesec); + } else if (strcasecmp("STATE_FORMAT", optstr) == 0) { sscanf(cmdstr, "%*s %s", flgstr); if (strcasecmp("NETCDF3_CLASSIC", flgstr) == 0) { @@ -1168,35 +1171,39 @@ get_global_param(FILE *gp) if (global_param.stateyear == 0 || global_param.statemonth == 0 || global_param.stateday == 0) { log_err("Incomplete specification of the date to save state " - "for state file (%s).\nSpecified date (yyyy-mm-dd): " - "%04d-%02d-%02d\nMake sure STATEYEAR, STATEMONTH, and " - "STATEDAY are set correctly in your global parameter " + "for state file (%s).\nSpecified date (yyyy-mm-dd-sssss): " + "%04d-%02d-%02d-%05u\nMake sure STATEYEAR, STATEMONTH, " + "and STATEDAY are set correctly in your global parameter " "file.", filenames.statefile, global_param.stateyear, - global_param.statemonth, global_param.stateday); + global_param.statemonth, global_param.stateday, + global_param.statesec); } // Check for month, day in range make_lastday(global_param.stateyear, global_param.calendar, lastday); if (global_param.stateday > lastday[global_param.statemonth - 1] || - global_param.statemonth > MONTHS_PER_YEAR || global_param.statemonth < 1 || - global_param.stateday < 1) { - log_err("Unusual specification of the date to save state for " - "state file (%s).\nSpecified date (yyyy-mm-dd): " - "%04d-%02d-%02d\nMake sure STATEYEAR, STATEMONTH, and " - "STATEDAY are set correctly in your global parameter " - "file.", filenames.statefile, global_param.stateyear, - global_param.statemonth, global_param.stateday); + global_param.statemonth > MONTHS_PER_YEAR || + global_param.stateday < 1 || global_param.stateday > 31 || + global_param.statesec > SEC_PER_DAY) { + log_err("Unusual specification of the date to save state " + "for state file (%s).\nSpecified date (yyyy-mm-dd-sssss): " + "%04d-%02d-%02d-%05u\nMake sure STATEYEAR, STATEMONTH, " + "STATEDAY and STATESEC are set correctly in your global " + "parameter file.", filenames.statefile, + global_param.stateyear, global_param.statemonth, + global_param.stateday, global_param.statesec); } } - // Set the statename here to be able to compare with INIT_STATE name + // Set the statename here temporarily to compare with INIT_STATE name if (options.SAVE_STATE) { - sprintf(filenames.statefile, "%s_%04i%02i%02i", filenames.statefile, - global_param.stateyear, global_param.statemonth, - global_param.stateday); + sprintf(flgstr2, "%s.%04i%02i%02i_%05u.nc", + filenames.statefile, global_param.stateyear, + global_param.statemonth, global_param.stateday, + global_param.statesec); } if (options.INIT_STATE && options.SAVE_STATE && - (strcmp(filenames.init_state, filenames.statefile) == 0)) { + (strcmp(filenames.init_state, flgstr2) == 0)) { log_err("The save state file (%s) has the same name as the " "initialize state file (%s). The initialize state file " "will be destroyed when the save state file is opened.", diff --git a/vic/drivers/image/src/vic_image.c b/vic/drivers/image/src/vic_image.c index 7828c7c65..87f755891 100644 --- a/vic/drivers/image/src/vic_image.c +++ b/vic/drivers/image/src/vic_image.c @@ -36,9 +36,9 @@ dmy_struct *dmy = NULL; filenames_struct filenames; filep_struct filep; domain_struct global_domain; -domain_struct local_domain; global_param_struct global_param; -lake_con_struct lake_con; +lake_con_struct *lake_con = NULL; +domain_struct local_domain; MPI_Comm MPI_COMM_VIC = MPI_COMM_WORLD; MPI_Datatype mpi_global_struct_type; MPI_Datatype mpi_filenames_struct_type; @@ -53,10 +53,10 @@ int mpi_size; nc_file_struct nc_hist_file; nc_var_struct nc_vars[N_OUTVAR_TYPES]; option_struct options; -parameters_struct param; out_data_struct **out_data; -save_data_struct *save_data; +parameters_struct param; param_set_struct param_set; +save_data_struct *save_data; soil_con_struct *soil_con = NULL; veg_con_map_struct *veg_con_map = NULL; veg_con_struct **veg_con = NULL; @@ -123,10 +123,10 @@ main(int argc, vic_write(&(dmy[current])); } - // if save: TBD needs to be fixed - not working in MPI - // if (current == global_param.nrecs - 1) { - // vic_store(); - // } + // if save: + if (check_save_state_flag(current)) { + vic_store(&(dmy[current])); + } } // clean up diff --git a/vic/drivers/image/src/vic_populate_model_state.c b/vic/drivers/image/src/vic_populate_model_state.c index 1c728b708..da344af30 100644 --- a/vic/drivers/image/src/vic_populate_model_state.c +++ b/vic/drivers/image/src/vic_populate_model_state.c @@ -1,7 +1,13 @@ /****************************************************************************** * @section DESCRIPTION * - * Populate model state. + * This function populates the model state. + * + * If a state file is provided to the model then its contents are checked + * to see if it agrees with the current simulation set-up, if so it is used + * to initialize the model state. If no state file is provided then the + * model initializes all variables with defaults and the user should expect + * to throw out the beginning of the simulation period as model spin-up. * * @section LICENSE * @@ -32,35 +38,36 @@ void vic_populate_model_state(void) { - extern all_vars_struct *all_vars; - extern domain_struct local_domain; - extern global_param_struct global_param; - extern option_struct options; - extern soil_con_struct *soil_con; - extern veg_con_struct **veg_con; - - double surf_temp; - int store_offset; - size_t i; - size_t nveg; + extern all_vars_struct *all_vars; + extern lake_con_struct *lake_con; + extern domain_struct local_domain; + extern option_struct options; + extern soil_con_struct *soil_con; + extern veg_con_struct **veg_con; - // read first forcing timestep (used in restoring model state) - // reset the forcing offset to what it was before - store_offset = global_param.forceoffset[0]; - vic_force(); - global_param.forceoffset[0] = store_offset; + size_t i; // read the model state from the netcdf file if there is one if (options.INIT_STATE) { vic_restore(); } + else { + // else generate a default state + for (i = 0; i < local_domain.ncells_active; i++) { + generate_default_state(&(all_vars[i]), &(soil_con[i]), veg_con[i]); + if (options.LAKES) { + generate_default_lake_state(&(all_vars[i]), &(soil_con[i]), + lake_con[i]); + } + } + } - // run through the remaining VIC initialization routines + // compute those state variables that are derived from the others for (i = 0; i < local_domain.ncells_active; i++) { - // TBD: do something sensible for surf_temp - surf_temp = 0.; - nveg = veg_con[i][0].vegetat_type_num; - initialize_model_state(&(all_vars[i]), nveg, options.Nnode, - surf_temp, &(soil_con[i]), veg_con[i]); + compute_derived_state_vars(&(all_vars[i]), &(soil_con[i]), veg_con[i]); + if (options.LAKES) { + compute_derived_lake_dimensions(&(all_vars[i].lake_var), + lake_con[i]); + } } } diff --git a/vic/drivers/python/vic_headers.py b/vic/drivers/python/vic_headers.py index 8fd30d805..700584b83 100644 --- a/vic/drivers/python/vic_headers.py +++ b/vic/drivers/python/vic_headers.py @@ -408,11 +408,12 @@ size_t nrecs; unsigned short int skipyear; unsigned short int startday; - unsigned int startsec; unsigned short int startmonth; + unsigned int startsec; unsigned short int startyear; unsigned short int stateday; unsigned short int statemonth; + unsigned int statesec; unsigned short int stateyear; unsigned short int calendar; unsigned short int time_units; diff --git a/vic/drivers/shared_all/include/vic_driver_shared_all.h b/vic/drivers/shared_all/include/vic_driver_shared_all.h index 8efb3a1a7..35adebad7 100644 --- a/vic/drivers/shared_all/include/vic_driver_shared_all.h +++ b/vic/drivers/shared_all/include/vic_driver_shared_all.h @@ -449,6 +449,8 @@ void collect_eb_terms(energy_bal_struct, snow_data_struct, cell_data_struct, void collect_wb_terms(cell_data_struct, veg_var_struct, snow_data_struct, double, double, double, int, double, int, double *, double *, out_data_struct *); +void compute_derived_state_vars(all_vars_struct *, soil_con_struct *, + veg_con_struct *); void compute_lake_params(lake_con_struct *, soil_con_struct); void compute_treeline(atmos_data_struct *, dmy_struct *, double, double *, bool *); @@ -473,9 +475,14 @@ void free_dmy(dmy_struct **dmy); void free_out_data_files(out_data_file_struct **); void free_out_data(out_data_struct **); void free_vegcon(veg_con_struct **veg_con); +void generate_default_state(all_vars_struct *, soil_con_struct *, + veg_con_struct *); +void generate_default_lake_state(all_vars_struct *, soil_con_struct *, + lake_con_struct); void get_parameters(FILE *paramfile); void init_output_list(out_data_struct *out_data, int write, char *format, int type, double mult); +void initialize_energy(energy_bal_struct **energy, size_t nveg); void initialize_filenames(void); void initialize_fileps(void); void initialize_global(void); diff --git a/vic/drivers/shared_all/src/compute_derived_state_vars.c b/vic/drivers/shared_all/src/compute_derived_state_vars.c new file mode 100644 index 000000000..7bb542b44 --- /dev/null +++ b/vic/drivers/shared_all/src/compute_derived_state_vars.c @@ -0,0 +1,242 @@ +/****************************************************************************** + * @section DESCRIPTION + * + * This routine computes the state variables (energy balance, water balance, + * and snow components) that are derived from the variables that are stored in + * state files. + * + * @section LICENSE + * + * The Variable Infiltration Capacity (VIC) macroscale hydrological model + * Copyright (C) 2014 The Land Surface Hydrology Group, Department of Civil + * and Environmental Engineering, University of Washington. + * + * The VIC model is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + *****************************************************************************/ + +#include + +/****************************************************************************** + * @brief Compute the state variables (energy balance, water balance, + * and snow components) that are derived from the variables that + * are stored in state files. + *****************************************************************************/ +void +compute_derived_state_vars(all_vars_struct *all_vars, + soil_con_struct *soil_con, + veg_con_struct *veg_con) +{ + extern global_param_struct global_param; + extern option_struct options; + + char FIRST_VEG; + size_t Nveg; + size_t veg; + size_t lidx; + size_t band; + size_t frost_area; + int ErrorFlag; + double Cv; + double moist[MAX_VEG][MAX_BANDS][MAX_LAYERS]; + double ice[MAX_VEG][MAX_BANDS][MAX_LAYERS][ + MAX_FROST_AREAS]; + double dt_thresh; + double tmp_runoff; + + cell_data_struct **cell; + energy_bal_struct **energy; + + cell = all_vars->cell; + energy = all_vars->energy; + Nveg = veg_con[0].vegetat_type_num; + + /****************************************** + Compute derived soil layer vars + ******************************************/ + for (veg = 0; veg <= Nveg; veg++) { + // Initialize soil for existing vegetation types + Cv = veg_con[veg].Cv; + + if (Cv > 0) { + for (band = 0; band < options.SNOW_BAND; band++) { + // Initialize soil for existing snow elevation bands + if (soil_con->AreaFract[band] > 0.) { + // set up temporary moist and ice arrays + for (lidx = 0; lidx < options.Nlayer; lidx++) { + moist[veg][band][lidx] = + cell[veg][band].layer[lidx].moist; + for (frost_area = 0; + frost_area < options.Nfrost; + frost_area++) { + ice[veg][band][lidx][frost_area] = + cell[veg][band].layer[lidx].ice[frost_area]; + } + } + + // compute saturated area and water table + compute_runoff_and_asat(soil_con, moist[veg][band], 0, + &(cell[veg][band].asat), + &tmp_runoff); + wrap_compute_zwt(soil_con, &(cell[veg][band])); + } + } + } + } + + /****************************************** + Compute soil thermal node properties + ******************************************/ + + FIRST_VEG = true; + for (veg = 0; veg <= Nveg; veg++) { + // Initialize soil for existing vegetation types + Cv = veg_con[veg].Cv; + + if (Cv > 0) { + for (band = 0; band < options.SNOW_BAND; band++) { + // Initialize soil for existing snow elevation bands + if (soil_con->AreaFract[band] > 0.) { + /** Set soil properties for all soil nodes **/ + if (FIRST_VEG) { + FIRST_VEG = false; + set_node_parameters(soil_con->Zsum_node, + soil_con->max_moist_node, + soil_con->expt_node, + soil_con->bubble_node, + soil_con->alpha, soil_con->beta, + soil_con->gamma, soil_con->depth, + soil_con->max_moist, soil_con->expt, + soil_con->bubble, + options.Nnode, options.Nlayer); + } + + // set soil moisture properties for all soil thermal nodes + ErrorFlag = + distribute_node_moisture_properties( + energy[veg][band].moist, + energy[veg][band].ice, + energy[veg][band].kappa_node, + energy[veg][band].Cs_node, + soil_con->Zsum_node, + energy[veg][band].T, + soil_con->max_moist_node, + soil_con->expt_node, + soil_con->bubble_node, + moist[veg][band], + soil_con->depth, + soil_con->soil_dens_min, + soil_con->bulk_dens_min, + soil_con->quartz, + soil_con->soil_density, + soil_con->bulk_density, + soil_con->organic, + options.Nnode, options.Nlayer, + soil_con->FS_ACTIVE); + if (ErrorFlag == ERROR) { + log_err("Error setting physical properties for " + "soil thermal nodes"); + } + + // Check node spacing v time step + // (note this is only approximate since heat capacity and + // conductivity can change considerably during the + // simulation depending on soil moisture and ice content) + if ((options.FROZEN_SOIL && + !options.QUICK_FLUX) && !options.IMPLICIT) { + // in seconds + dt_thresh = 0.5 * energy[veg][band].Cs_node[1] / + energy[veg][band].kappa_node[1] * + pow((soil_con->dz_node[1]), + 2); + if (global_param.dt > dt_thresh) { + log_err("You are currently running FROZEN SOIL " + "with an explicit method (IMPLICIT is " + "set to FALSE). For the explicit method " + "to be stable, time step %f seconds is too " + "large for the given thermal node spacing " + "%f m, soil heat capacity %f J/m3/K, and " + "soil thermal conductivity %f J/m/s/K. " + "Either set IMPLICIT to TRUE in your " + "global parameter file (this is the " + "recommended action), or decrease time " + "step length to <= %f seconds, or decrease " + "the number of soil thermal nodes.", + global_param.dt, + soil_con->dz_node[1], + energy[veg][band].Cs_node[1], + energy[veg][band].kappa_node[1], dt_thresh); + } + } + + /* initialize layer moistures and ice contents */ + for (lidx = 0; lidx < options.Nlayer; lidx++) { + cell[veg][band].layer[lidx].moist = + moist[veg][band][lidx]; + for (frost_area = 0; + frost_area < options.Nfrost; + frost_area++) { + cell[veg][band].layer[lidx].ice[frost_area] = + ice[veg][band][lidx][frost_area]; + } + } + if (options.QUICK_FLUX) { + ErrorFlag = + estimate_layer_ice_content_quick_flux( + cell[veg][band].layer, + soil_con->depth, soil_con->dp, + energy[ + veg][band].T[0], energy[veg][band].T[1], + soil_con->avg_temp, soil_con->max_moist, + soil_con->expt, soil_con->bubble, + soil_con->frost_fract, soil_con->frost_slope, + soil_con->FS_ACTIVE); + if (ErrorFlag == ERROR) { + log_err("Error in " + "estimate_layer_ice_content_quick_flux"); + } + } + else { + ErrorFlag = estimate_layer_ice_content( + cell[veg][band].layer, + soil_con->Zsum_node, + energy[veg][band].T, + soil_con->depth, + soil_con->max_moist, + soil_con->expt, + soil_con->bubble, + soil_con->frost_fract, + soil_con->frost_slope, + options.Nnode, + options.Nlayer, + soil_con->FS_ACTIVE); + if (ErrorFlag == ERROR) { + log_err("Error in " + "estimate_layer_ice_content"); + } + } + + /* Find freezing and thawing front depths */ + if (!options.QUICK_FLUX && soil_con->FS_ACTIVE) { + find_0_degree_fronts(&energy[veg][band], + soil_con->Zsum_node, + energy[veg][band].T, + options.Nnode); + } + } + } + } + } + +} diff --git a/vic/drivers/shared_all/src/generate_default_lake_state.c b/vic/drivers/shared_all/src/generate_default_lake_state.c new file mode 100644 index 000000000..55be62978 --- /dev/null +++ b/vic/drivers/shared_all/src/generate_default_lake_state.c @@ -0,0 +1,61 @@ +/****************************************************************************** + * @section DESCRIPTION + * + * This routine initializes the lake model state (energy balance, water balance, + * snow components) to default values. + * + * @section LICENSE + * + * The Variable Infiltration Capacity (VIC) macroscale hydrological model + * Copyright (C) 2014 The Land Surface Hydrology Group, Department of Civil + * and Environmental Engineering, University of Washington. + * + * The VIC model is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + *****************************************************************************/ + +#include + +/****************************************************************************** + * @brief Initialize the lake model state (energy balance, water balance, + * and snow components) to default values. + *****************************************************************************/ +void +generate_default_lake_state(all_vars_struct *all_vars, + soil_con_struct *soil_con, + lake_con_struct lake_con) +{ + extern option_struct options; + + size_t k; + + lake_var_struct lake; + + lake = all_vars->lake_var; + + /************************************************************************ + Initialize lake state variables + TBD: currently setting depth to depth_in from parameter file, but + in future we should initialize to mindepth as default and + eliminate depth_in (require user to use a state file if they + want control over initial depth) + ************************************************************************/ + if (options.LAKES) { + lake.ldepth = lake_con.depth_in; + for (k = 0; k < lake.activenod; k++) { + // lake model requires FULL_ENERGY set to true + lake.temp[k] = soil_con->avg_temp; + } + } +} diff --git a/vic/drivers/shared_all/src/generate_default_state.c b/vic/drivers/shared_all/src/generate_default_state.c new file mode 100644 index 000000000..7d638437e --- /dev/null +++ b/vic/drivers/shared_all/src/generate_default_state.c @@ -0,0 +1,113 @@ +/****************************************************************************** + * @section DESCRIPTION + * + * This routine initializes the model state (energy balance, water balance, and + * snow components) to default values. + * + * @section LICENSE + * + * The Variable Infiltration Capacity (VIC) macroscale hydrological model + * Copyright (C) 2014 The Land Surface Hydrology Group, Department of Civil + * and Environmental Engineering, University of Washington. + * + * The VIC model is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + *****************************************************************************/ + +#include + +/****************************************************************************** + * @brief Initialize the model state (energy balance, water balance, and + * snow components) to default values. + *****************************************************************************/ +void +generate_default_state(all_vars_struct *all_vars, + soil_con_struct *soil_con, + veg_con_struct *veg_con) +{ + extern option_struct options; + extern parameters_struct param; + + size_t Nveg; + size_t veg; + size_t band; + size_t lidx; + size_t k; + double Cv; + double tmp; + + cell_data_struct **cell; + energy_bal_struct **energy; + + cell = all_vars->cell; + energy = all_vars->energy; + Nveg = veg_con[0].vegetat_type_num; + + /************************************************************************ + Initialize soil moistures + TBD: currently setting moist to init_moist from parameter file, but + in future we should initialize to max_moist as default and + eliminate init_moist (require user to use a state file if they + want control over initial moist) + ************************************************************************/ + + for (veg = 0; veg <= Nveg; veg++) { + Cv = veg_con[veg].Cv; + if (Cv > 0) { + for (band = 0; band < options.SNOW_BAND; band++) { + if (soil_con->AreaFract[band] > 0.) { + /* Initialize soil node temperatures */ + for (lidx = 0; lidx < options.Nlayer; lidx++) { + cell[veg][band].layer[lidx].moist = + soil_con->init_moist[lidx]; + if (cell[veg][band].layer[lidx].moist > + soil_con->max_moist[lidx]) { + cell[veg][band].layer[lidx].moist = + soil_con->max_moist[lidx]; + } + } + } + } + } + } + + /************************************************************************ + Initialize soil temperatures + ************************************************************************/ + + for (veg = 0; veg <= Nveg; veg++) { + Cv = veg_con[veg].Cv; + if (Cv > 0) { + for (band = 0; band < options.SNOW_BAND; band++) { + if (soil_con->AreaFract[band] > 0.) { + /* Initialize soil node temperatures */ + for (k = 0; k < options.Nnode; k++) { + if (options.FULL_ENERGY || options.FROZEN_SOIL) { + energy[veg][band].T[k] = soil_con->avg_temp; + } + else { + energy[veg][band].T[k] = 0; + } + } + /* Initial estimate of LongUnderOut for use by snow_intercept() */ + tmp = energy[veg][band].T[0] + CONST_TKFRZ; + energy[veg][band].LongUnderOut = calc_outgoing_longwave(tmp, + param.EMISS_SNOW); + energy[veg][band].Tfoliage = energy[veg][band].T[0] + + soil_con->Tfactor[band]; + } + } + } + } +} diff --git a/vic/drivers/shared_image/src/initialize_energy.c b/vic/drivers/shared_all/src/initialize_energy.c similarity index 99% rename from vic/drivers/shared_image/src/initialize_energy.c rename to vic/drivers/shared_all/src/initialize_energy.c index 0c99b7899..cf5c1c863 100644 --- a/vic/drivers/shared_image/src/initialize_energy.c +++ b/vic/drivers/shared_all/src/initialize_energy.c @@ -24,7 +24,7 @@ * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. *****************************************************************************/ -#include +#include /****************************************************************************** * @brief Initialize energy structure. diff --git a/vic/drivers/shared_all/src/initialize_soil.c b/vic/drivers/shared_all/src/initialize_soil.c index 05005400a..94dc9a30a 100644 --- a/vic/drivers/shared_all/src/initialize_soil.c +++ b/vic/drivers/shared_all/src/initialize_soil.c @@ -47,13 +47,7 @@ initialize_soil(cell_data_struct **cell, cell[veg][band].runoff = 0; for (lindex = 0; lindex < options.Nlayer; lindex++) { cell[veg][band].layer[lindex].evap = 0; - cell[veg][band].layer[lindex].moist = - soil_con->init_moist[lindex]; - if (cell[veg][band].layer[lindex].moist > - soil_con->max_moist[lindex]) { - cell[veg][band].layer[lindex].moist = - soil_con->max_moist[lindex]; - } + cell[veg][band].layer[lindex].moist = 0; tmp_moist[lindex] = cell[veg][band].layer[lindex].moist; for (frost_area = 0; frost_area < options.Nfrost; frost_area++) { @@ -68,18 +62,4 @@ initialize_soil(cell_data_struct **cell, cell[veg][band].CSlow = 0; } } - for (frost_area = 0; frost_area < options.Nfrost; frost_area++) { - if (options.Nfrost == 1) { - soil_con->frost_fract[frost_area] = 1.; - } - else if (options.Nfrost == 2) { - soil_con->frost_fract[frost_area] = 0.5; - } - else { - soil_con->frost_fract[frost_area] = 1. / (options.Nfrost - 1); - if (frost_area == 0 || frost_area == options.Nfrost - 1) { - soil_con->frost_fract[frost_area] /= 2.; - } - } - } } diff --git a/vic/drivers/shared_image/include/vic_driver_shared_image.h b/vic/drivers/shared_image/include/vic_driver_shared_image.h index 15ab60b8e..75f5c01dd 100644 --- a/vic/drivers/shared_image/include/vic_driver_shared_image.h +++ b/vic/drivers/shared_image/include/vic_driver_shared_image.h @@ -169,7 +169,6 @@ int get_nc_field_int(char *nc_name, char *var_name, size_t *start, int get_nc_mode(unsigned short int format); void initialize_domain(domain_struct *domain); void initialize_domain_info(domain_info_struct *info); -void initialize_energy(energy_bal_struct **energy, size_t nveg); void initialize_global_structures(void); void initialize_history_file(nc_file_struct *nc); void initialize_location(location_struct *location); @@ -177,7 +176,6 @@ int initialize_model_state(all_vars_struct *all_vars, size_t Nveg, size_t Nnodes, double surf_temp, soil_con_struct *soil_con, veg_con_struct *veg_con); void initialize_soil_con(soil_con_struct *soil_con); -void initialize_state_file(nc_file_struct *nc, dmy_struct *dmy_current); void initialize_veg_con(veg_con_struct *veg_con); int parse_output_info(FILE *gp, out_data_struct **out_data); void print_atmos_data(atmos_data_struct *atmos); diff --git a/vic/drivers/shared_image/src/initialize_model_state.c b/vic/drivers/shared_image/src/initialize_model_state.c deleted file mode 100644 index 9a45dbb19..000000000 --- a/vic/drivers/shared_image/src/initialize_model_state.c +++ /dev/null @@ -1,434 +0,0 @@ -/****************************************************************************** - * @section DESCRIPTION - * - * This routine initializes the model state (energy balance, water balance, and - * snow components). - * - * If a state file is provided to the model than its - * contents are checked to see if it agrees with the current simulation set-up, - * if so it is used to initialize the model state. If no state file is - * provided the model initializes all variables with defaults and the user - * should expect to throw out the beginning of the simulation period as model - * start-up. - * - * @section LICENSE - * - * The Variable Infiltration Capacity (VIC) macroscale hydrological model - * Copyright (C) 2014 The Land Surface Hydrology Group, Department of Civil - * and Environmental Engineering, University of Washington. - * - * The VIC model is free software; you can redistribute it and/or - * modify it under the terms of the GNU General Public License - * as published by the Free Software Foundation; either version 2 - * of the License, or (at your option) any later version. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the - * GNU General Public License for more details. - * - * You should have received a copy of the GNU General Public License along with - * this program; if not, write to the Free Software Foundation, Inc., - * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. - *****************************************************************************/ - -#include - -/****************************************************************************** - * @brief Initialize the model state (energy balance, water balance, and - * snow components). - *****************************************************************************/ -int -initialize_model_state(all_vars_struct *all_vars, - size_t Nveg, - size_t Nnodes, - double surf_temp, - soil_con_struct *soil_con, - veg_con_struct *veg_con) -{ - extern global_param_struct global_param; - extern option_struct options; - - char FIRST_VEG; - size_t veg; - size_t index; - size_t lidx; - size_t band; - size_t frost_area; - int ErrorFlag; - double Cv; - double Zsum, dp; - double tmpdp, tmpadj, Bexp; - double moist[MAX_VEG][MAX_BANDS][MAX_LAYERS]; - double ice[MAX_VEG][MAX_BANDS][MAX_LAYERS][ - MAX_FROST_AREAS]; - double dt_thresh; - - cell_data_struct **cell; - energy_bal_struct **energy; - - cell = all_vars->cell; - energy = all_vars->energy; - - // Initialize soil depths - dp = soil_con->dp; - - FIRST_VEG = true; - - // increase initial soil surface temperature if air is very cold - if (surf_temp < -1.) { - surf_temp = -1.; - } - - /************************************************************************ - CASE 1: initial conditions files provided -- handled in - vic_populate_model_state() - ************************************************************************/ - - /************************************************************************ - CASE 2: Initialize soil if using quick heat flux, and no initial - soil properties file given - ************************************************************************/ - - if (!options.INIT_STATE && options.QUICK_FLUX) { - Nnodes = options.Nnode; - - /* Initialize soil node thicknesses */ - soil_con->dz_node[0] = soil_con->depth[0]; - soil_con->dz_node[1] = soil_con->depth[0]; - soil_con->dz_node[2] = 2. * (dp - 1.5 * soil_con->depth[0]); - soil_con->Zsum_node[0] = 0; - soil_con->Zsum_node[1] = soil_con->depth[0]; - soil_con->Zsum_node[2] = dp; - - for (veg = 0; veg <= Nveg; veg++) { - // Initialize soil for existing vegetation types - Cv = veg_con[veg].Cv; - - if (Cv > 0) { - for (band = 0; band < options.SNOW_BAND; band++) { - /* Initialize soil node temperatures */ - energy[veg][band].T[0] = surf_temp; - energy[veg][band].T[1] = surf_temp; - energy[veg][band].T[2] = soil_con->avg_temp; - - /* Initialize soil layer moisture and ice contents */ - for (lidx = 0; lidx < options.Nlayer; lidx++) { - moist[veg][band][lidx] = - cell[veg][band].layer[lidx].moist; - for (frost_area = 0; - frost_area < options.Nfrost; - frost_area++) { - ice[veg][band][lidx][frost_area] = 0.; - } - } - } - } - } - } - - /***************************************************************** - CASE 3: Initialize Energy Balance Variables if not using quick - ground heat flux, and no Initial Condition File Given - *****************************************************************/ - else if (!options.INIT_STATE && !options.QUICK_FLUX) { - for (veg = 0; veg <= Nveg; veg++) { - // Initialize soil for existing vegetation types - Cv = veg_con[veg].Cv; - - if (Cv > 0) { - for (band = 0; band < options.SNOW_BAND; band++) { - if (!options.EXP_TRANS) { - /* Initialize soil node temperatures and thicknesses - Nodes set at surface, the depth of the first layer, - twice the depth of the first layer, and at the - damping depth. Extra nodes are placed equal distance - between the damping depth and twice the depth of the - first layer. */ - - soil_con->dz_node[0] = soil_con->depth[0]; - soil_con->dz_node[1] = soil_con->depth[0]; - soil_con->dz_node[2] = soil_con->depth[0]; - - soil_con->Zsum_node[0] = 0; - soil_con->Zsum_node[1] = soil_con[0].depth[0]; - Zsum = 2. * soil_con[0].depth[0]; - soil_con->Zsum_node[2] = Zsum; - tmpdp = dp - soil_con[0].depth[0] * 2.5; - tmpadj = 3.5; - for (index = 3; index < Nnodes - 1; index++) { - if (FIRST_VEG) { - soil_con->dz_node[index] = tmpdp / - (((double) Nnodes - - tmpadj)); - } - Zsum += (soil_con->dz_node[index] + - soil_con->dz_node[index - 1]) / 2.; - soil_con->Zsum_node[index] = Zsum; - } - energy[veg][band].T[0] = surf_temp; - for (index = 1; index < Nnodes; index++) { - energy[veg][band].T[index] = soil_con->avg_temp; - } - if (FIRST_VEG) { - FIRST_VEG = false; - soil_con->dz_node[Nnodes - 1] = (dp - Zsum - - soil_con->dz_node[ - Nnodes - 2] / - 2.) * 2.; - Zsum += (soil_con->dz_node[Nnodes - 2] + - soil_con->dz_node[Nnodes - 1]) / 2.; - soil_con->Zsum_node[Nnodes - 1] = Zsum; - if ((int) (Zsum * MM_PER_M + 0.5) != - (int) (dp * MM_PER_M + 0.5)) { - log_err("Sum of thermal node thicknesses (%f) " - "in initialize_model_state do not " - "equal dp (%f), check initialization " - "procedure", Zsum, dp); - } - } - } - else { // exponential grid transformation, EXP_TRANS = TRUE - // calculate exponential function parameter */ - if (FIRST_VEG) { - // to force Zsum=dp at bottom node - Bexp = logf(dp + 1.) / (double) (Nnodes - 1); - // validate Nnodes by requiring that there be at - // least 3 nodes in the top 50cm - if (Nnodes < 5 * logf(dp + 1.) + 1) { - log_err("The number of soil thermal nodes (%zu) " - "is too small for the supplied damping " - "depth (%f) with EXP_TRANS set to " - "TRUE, leading to fewer than 3 nodes " - "in the top 50 cm of the soil column. " - "For EXP_TRANS=TRUE, Nnodes and dp " - "must follow the relationship:\n" - "5*ln(dp+1)Zsum_node[index] = - expf(Bexp * index) - 1.; - } - if (soil_con->Zsum_node[0] > soil_con->depth[0]) { - log_err("Depth of first thermal node (%f) in " - "initialize_model_state is greater " - "than depth of first soil layer (%f); " - "increase the number of nodes or " - "decrease the thermal damping depth " - "dp (%f)", - soil_con->Zsum_node[0], - soil_con->depth[0], dp); - } - } - - // top node - index = 0; - if (FIRST_VEG) { - soil_con->dz_node[index] = - soil_con->Zsum_node[index + - 1] - - soil_con->Zsum_node[index]; - } - energy[veg][band].T[index] = surf_temp; - // middle nodes - for (index = 1; index < Nnodes - 1; index++) { - if (FIRST_VEG) { - soil_con->dz_node[index] = - (soil_con->Zsum_node[index + - 1] - - soil_con->Zsum_node[index]) / 2. + - (soil_con->Zsum_node[index] - - soil_con->Zsum_node[index - 1]) / 2.; - } - energy[veg][band].T[index] = soil_con->avg_temp; - } - // bottom node - index = Nnodes - 1; - if (FIRST_VEG) { - soil_con->dz_node[index] = - soil_con->Zsum_node[index] - - soil_con->Zsum_node[index - 1]; - } - energy[veg][band].T[index] = soil_con->avg_temp; - } // end if !EXP_TRANS - - // initialize moisture and ice for each soil layer - for (lidx = 0; lidx < options.Nlayer; lidx++) { - moist[veg][band][lidx] = - cell[veg][band].layer[lidx].moist; - for (frost_area = 0; - frost_area < options.Nfrost; - frost_area++) { - ice[veg][band][lidx][frost_area] = 0.; - } - } - } - } - } - } - - /********************************* - CASE 4: Unknown option - *********************************/ - else if (!options.INIT_STATE) { - for (veg = 0; veg <= Nveg; veg++) { - // Initialize soil for existing vegetation types - Cv = veg_con[veg].Cv; - - if (Cv > 0) { - for (band = 0; band < options.SNOW_BAND; band++) { - // Initialize soil for existing snow elevation bands - if (soil_con->AreaFract[band] > 0.) { - for (index = 0; index < options.Nlayer; index++) { - soil_con->dz_node[index] = 1.; - } - } - } - } - } - } - - /****************************************** - Initialize soil thermal node properties - ******************************************/ - - FIRST_VEG = true; - for (veg = 0; veg <= Nveg; veg++) { - // Initialize soil for existing vegetation types - Cv = veg_con[veg].Cv; - - if (Cv > 0) { - for (band = 0; band < options.SNOW_BAND; band++) { - // Initialize soil for existing snow elevation bands - if (soil_con->AreaFract[band] > 0.) { - /** Set soil properties for all soil nodes **/ - if (FIRST_VEG) { - FIRST_VEG = false; - set_node_parameters(soil_con->Zsum_node, - soil_con->max_moist_node, - soil_con->expt_node, - soil_con->bubble_node, - soil_con->alpha, soil_con->beta, - soil_con->gamma, soil_con->depth, - soil_con->max_moist, soil_con->expt, - soil_con->bubble, - Nnodes, options.Nlayer); - } - - // set soil moisture properties for all soil thermal nodes - ErrorFlag = - distribute_node_moisture_properties( - energy[veg][band].moist, - energy[veg][band].ice, - energy[veg][band].kappa_node, - energy[veg][band].Cs_node, - soil_con->Zsum_node, - energy[veg][band].T, - soil_con->max_moist_node, - soil_con->expt_node, - soil_con->bubble_node, - moist[veg][band], - soil_con->depth, - soil_con->soil_dens_min, - soil_con->bulk_dens_min, - soil_con->quartz, - soil_con->soil_density, - soil_con->bulk_density, - soil_con->organic, - Nnodes, options.Nlayer, - soil_con->FS_ACTIVE); - if (ErrorFlag == ERROR) { - return (ErrorFlag); - } - - // Check node spacing v time step - // (note this is only approximate since heat capacity and - // conductivity can change considerably during the - // simulation depending on soil moisture and ice content) - if ((options.FROZEN_SOIL && - !options.QUICK_FLUX) && !options.IMPLICIT) { - // in seconds - dt_thresh = 0.5 * energy[veg][band].Cs_node[1] / - energy[veg][band].kappa_node[1] * - pow((soil_con->dz_node[1]), - 2); - if (global_param.dt > dt_thresh) { - log_err("You are currently running FROZEN SOIL " - "with an explicit method (IMPLICIT is " - "set to FALSE). For the explicit method " - "to be stable, time step %f seconds is too " - "large for the given thermal node spacing " - "%f m, soil heat capacity %f J/m3/K, and " - "soil thermal conductivity %f J/m/s/K. " - "Either set IMPLICIT to TRUE in your " - "global parameter file (this is the " - "recommended action), or decrease time " - "step length to <= %f seconds, or decrease " - "the number of soil thermal nodes.", - global_param.dt, - soil_con->dz_node[1], - energy[veg][band].Cs_node[1], - energy[veg][band].kappa_node[1], dt_thresh); - } - } - - /* initialize layer moistures and ice contents */ - for (lidx = 0; lidx < options.Nlayer; lidx++) { - cell[veg][band].layer[lidx].moist = - moist[veg][band][lidx]; - for (frost_area = 0; - frost_area < options.Nfrost; - frost_area++) { - cell[veg][band].layer[lidx].ice[frost_area] = - ice[veg][band][lidx][frost_area]; - } - } - if (options.QUICK_FLUX) { - ErrorFlag = - estimate_layer_ice_content_quick_flux( - cell[veg][band].layer, - soil_con->depth, soil_con->dp, - energy[ - veg][band].T[0], energy[veg][band].T[1], - soil_con->avg_temp, soil_con->max_moist, - soil_con->expt, soil_con->bubble, - soil_con->frost_fract, soil_con->frost_slope, - soil_con->FS_ACTIVE); - } - else { - ErrorFlag = estimate_layer_ice_content( - cell[veg][band].layer, - soil_con->Zsum_node, - energy[veg][band].T, - soil_con->depth, - soil_con->max_moist, - soil_con->expt, - soil_con->bubble, - soil_con->frost_fract, - soil_con->frost_slope, - Nnodes, options.Nlayer, - soil_con->FS_ACTIVE); - } - - /* Find freezing and thawing front depths */ - if (!options.QUICK_FLUX && soil_con->FS_ACTIVE) { - find_0_degree_fronts(&energy[veg][band], - soil_con->Zsum_node, - energy[veg][band].T, Nnodes); - } - } - } - } - } - - return(0); -} diff --git a/vic/drivers/shared_image/src/put_nc_field.c b/vic/drivers/shared_image/src/put_nc_field.c index a790a75f7..e4a19e5aa 100644 --- a/vic/drivers/shared_image/src/put_nc_field.c +++ b/vic/drivers/shared_image/src/put_nc_field.c @@ -104,7 +104,7 @@ put_nc_field_double(char *nc_name, } /****************************************************************************** - * @brief Put interger data field to netCDF. + * @brief Put integer data field to netCDF. *****************************************************************************/ int put_nc_field_int(char *nc_name, diff --git a/vic/drivers/shared_image/src/vic_init.c b/vic/drivers/shared_image/src/vic_init.c index 3b81962d6..9c028003a 100644 --- a/vic/drivers/shared_image/src/vic_init.c +++ b/vic/drivers/shared_image/src/vic_init.c @@ -58,6 +58,7 @@ vic_init(void) size_t m; size_t nveg; size_t max_numnod; + size_t Nnodes; int vidx; size_t d2count[2]; size_t d2start[2]; @@ -66,6 +67,8 @@ vic_init(void) size_t d4count[4]; size_t d4start[4]; int tmp_lake_idx; + double Zsum, dp; + double tmpdp, tmpadj, Bexp; // allocate memory for Cv_sum Cv_sum = malloc(local_domain.ncells_active * sizeof(*Cv_sum)); @@ -676,7 +679,10 @@ vic_init(void) for (i = 0; i < local_domain.ncells_active; i++) { soil_con[i].max_snow_distrib_slope = (double) dvar[i]; } + } + // spatial frost + if (options.SPATIAL_FROST) { // frost_slope: slope of frozen soil distribution get_scatter_nc_field_double(filenames.soil, "frost_slope", d2start, d2count, dvar); @@ -684,10 +690,25 @@ vic_init(void) soil_con[i].frost_slope = (double) dvar[i]; } } + for (k = 0; k < options.Nfrost; k++) { + if (options.Nfrost == 1) { + soil_con[i].frost_fract[k] = 1.; + } + else if (options.Nfrost == 2) { + soil_con[i].frost_fract[k] = 0.5; + } + else { + soil_con[i].frost_fract[k] = 1. / (options.Nfrost - 1); + if (k == 0 || k == options.Nfrost - 1) { + soil_con[i].frost_fract[k] /= 2.; + } + } + } // TODO: read avgJulyAirTemp for compute treeline option // Additional processing of the soil variables + // (compute derived parameters) for (i = 0; i < local_domain.ncells_active; i++) { for (j = 0; j < options.Nlayer; j++) { // compute layer properties @@ -792,6 +813,120 @@ vic_init(void) soil_moisture_from_water_table(&(soil_con[i]), options.Nlayer); + // Soil thermal node thicknesses and positions + Nnodes = options.Nnode; + dp = soil_con[i].dp; + if (options.QUICK_FLUX) { + /* node thicknesses */ + soil_con[i].dz_node[0] = soil_con[i].depth[0]; + soil_con[i].dz_node[1] = soil_con[i].depth[0]; + soil_con[i].dz_node[2] = 2. * (dp - 1.5 * soil_con[i].depth[0]); + + /* node depths (positions) */ + soil_con[i].Zsum_node[0] = 0; + soil_con[i].Zsum_node[1] = soil_con[i].depth[0]; + soil_con[i].Zsum_node[2] = dp; + } + else { + if (!options.EXP_TRANS) { + /* Compute soil node thicknesses + Nodes set at surface, the depth of the first layer, + twice the depth of the first layer, and at the + damping depth. Extra nodes are placed equal distance + between the damping depth and twice the depth of the + first layer. */ + + soil_con[i].dz_node[0] = soil_con[i].depth[0]; + soil_con[i].dz_node[1] = soil_con[i].depth[0]; + soil_con[i].dz_node[2] = soil_con[i].depth[0]; + soil_con[i].Zsum_node[0] = 0; + soil_con[i].Zsum_node[1] = soil_con[0].depth[0]; + Zsum = 2. * soil_con[0].depth[0]; + soil_con[i].Zsum_node[2] = Zsum; + tmpdp = dp - soil_con[0].depth[0] * 2.5; + tmpadj = 3.5; + for (j = 3; j < Nnodes - 1; j++) { + soil_con[i].dz_node[j] = tmpdp / + (((double) Nnodes - tmpadj)); + Zsum += + (soil_con[i].dz_node[j] + soil_con[i].dz_node[j - 1]) / + 2.; + soil_con[i].Zsum_node[j] = Zsum; + } + soil_con[i].dz_node[Nnodes - + 1] = + (dp - Zsum - soil_con[i].dz_node[Nnodes - 2] / 2.) * 2.; + Zsum += + (soil_con[i].dz_node[Nnodes - 2] + + soil_con[i].dz_node[Nnodes - 1]) / 2.; + soil_con[i].Zsum_node[Nnodes - 1] = Zsum; + if ((int) (Zsum * MM_PER_M + 0.5) != + (int) (dp * MM_PER_M + 0.5)) { + log_err("Sum of thermal node thicknesses (%f) " + "in initialize_model_state do not " + "equal dp (%f), check initialization " + "procedure", Zsum, dp); + } + } + else { + // exponential grid transformation, EXP_TRANS = TRUE + // calculate exponential function parameter + // to force Zsum=dp at bottom node + Bexp = logf(dp + 1.) / (double) (Nnodes - 1); + // validate Nnodes by requiring that there be at + // least 3 nodes in the top 50cm + if (Nnodes < 5 * logf(dp + 1.) + 1) { + log_err("The number of soil thermal nodes (%zu) " + "is too small for the supplied damping " + "depth (%f) with EXP_TRANS set to " + "TRUE, leading to fewer than 3 nodes " + "in the top 50 cm of the soil column. " + "For EXP_TRANS=TRUE, Nnodes and dp " + "must follow the relationship:\n" + "5*ln(dp+1) soil_con[i].depth[0]) { + log_err("Depth of first thermal node (%f) in " + "initialize_model_state is greater " + "than depth of first soil layer (%f); " + "increase the number of nodes or " + "decrease the thermal damping depth " + "dp (%f)", soil_con[i].Zsum_node[0], + soil_con[i].depth[0], dp); + } + + // top node + j = 0; + soil_con[i].dz_node[j] = soil_con[i].Zsum_node[j + 1] - + soil_con[i].Zsum_node[j]; + // middle nodes + for (j = 1; j < Nnodes - 1; j++) { + soil_con[i].dz_node[j] = + (soil_con[i].Zsum_node[j + 1] - + soil_con[i].Zsum_node[j]) / + 2. + + (soil_con[i].Zsum_node[j] - + soil_con[i].Zsum_node[j - 1]) / + 2.; + } + // bottom node + j = Nnodes - 1; + soil_con[i].dz_node[j] = soil_con[i].Zsum_node[j] - + soil_con[i].Zsum_node[j - 1]; + } // end if !EXP_TRANS + } + + // Carbon parameters if (options.CARBON) { // TBD Remove hardcoded parameter values soil_con[i].AlbedoPar = 0.92 * param.ALBEDO_BARE_SOIL - 0.015; @@ -1396,7 +1531,7 @@ vic_init(void) } } - // initialize structures with default values + // initialize state variables with default values for (i = 0; i < local_domain.ncells_active; i++) { nveg = veg_con[i][0].vegetat_type_num; initialize_snow(all_vars[i].snow, nveg); @@ -1409,7 +1544,7 @@ vic_init(void) } initialize_lake(&(all_vars[i].lake_var), lake_con[i], &(soil_con[i]), - &(all_vars[i].cell[tmp_lake_idx][0]), 0); + &(all_vars[i].cell[tmp_lake_idx][0]), false); } initialize_energy(all_vars[i].energy, nveg); } diff --git a/vic/drivers/shared_image/src/vic_restore.c b/vic/drivers/shared_image/src/vic_restore.c index f0ef77205..f85bbeae2 100644 --- a/vic/drivers/shared_image/src/vic_restore.c +++ b/vic/drivers/shared_image/src/vic_restore.c @@ -837,6 +837,7 @@ check_init_state_file(void) int status; size_t dimlen; size_t i; + size_t j; size_t d1count[1]; size_t d1start[1]; size_t d2count[2]; @@ -924,6 +925,7 @@ check_init_state_file(void) log_err("Error reading data from \"%s\" in %s", global_domain.info.lon_var, filenames.init_state); } + // implicitly nested loop over ni and nj with j set to 0 for (i = 0; i < global_domain.n_nx; i++) { if (!assert_close_double(dvar[i], global_domain.locations[i].longitude, rtol, @@ -945,9 +947,13 @@ check_init_state_file(void) log_err("Error reading data from \"%s\" in %s", global_domain.info.lat_var, filenames.init_state); } - for (i = 0; i < global_domain.n_ny; i++) { - if (!assert_close_double(dvar[i], - global_domain.locations[i].latitude, rtol, + // implicitly nested loop over ni and nj with i set to 0; + // j stride = n_nx + for (j = 0; j < global_domain.n_ny; j++) { + if (!assert_close_double(dvar[j], + global_domain.locations[j * + global_domain.n_nx] + .latitude, rtol, abs_tol)) { log_err("Latitudes in initial state file do not " "match parameter file"); diff --git a/vic/drivers/shared_image/src/vic_store.c b/vic/drivers/shared_image/src/vic_store.c index f7cb8201f..bdcba1608 100644 --- a/vic/drivers/shared_image/src/vic_store.c +++ b/vic/drivers/shared_image/src/vic_store.c @@ -32,14 +32,19 @@ void vic_store(dmy_struct *dmy_current) { + extern size_t current; + extern dmy_struct *dmy; + extern filenames_struct filenames; extern all_vars_struct *all_vars; extern domain_struct global_domain; extern domain_struct local_domain; extern option_struct options; extern soil_con_struct *soil_con; extern veg_con_map_struct *veg_con_map; + extern int mpi_rank; int status; + int old_fill_mode; size_t dcount[MAXDIMS]; size_t dstart[MAXDIMS]; int dimids[MAXDIMS]; @@ -71,281 +76,384 @@ vic_store(dmy_struct *dmy_current) nc_file_struct nc_state_file; - // create netcdf file for storing model state - keep the file open - initialize_state_file(&nc_state_file, dmy_current); - - // initialize dimids to invalid values - for (i = 0; i < MAXDIMS; i++) { - dimids[i] = -1; + // create netcdf file for storing model state + sprintf(nc_state_file.fname, "%s.%04d%02d%02d_%05u.nc", + filenames.statefile, dmy[current].year, dmy[current].month, + dmy[current].day, dmy[current].dayseconds); + + nc_state_file.c_fillvalue = NC_FILL_CHAR; + nc_state_file.i_fillvalue = NC_FILL_INT; + nc_state_file.d_fillvalue = NC_FILL_DOUBLE; + nc_state_file.f_fillvalue = NC_FILL_FLOAT; + + nc_state_file.ni_size = global_domain.n_nx; + nc_state_file.nj_size = global_domain.n_ny; + nc_state_file.veg_size = options.NVEGTYPES; + nc_state_file.band_size = options.SNOW_BAND; + nc_state_file.layer_size = options.Nlayer; + nc_state_file.frost_size = options.Nfrost; + nc_state_file.node_size = options.Nnode; + if (options.LAKES) { + nc_state_file.lake_node_size = options.NLAKENODES; } - // write dimension variables + // only open and initialize the netcdf file on the first thread + if (mpi_rank == 0) { + // open the netcdf file + status = nc_create(nc_state_file.fname, get_nc_mode( + options.STATE_FORMAT), &(nc_state_file.nc_id)); + if (status != NC_NOERR) { + log_err("Error creating %s", nc_state_file.fname); + } + nc_state_file.open = true; - // Coordinate variables - ndims = global_domain.info.n_coord_dims; - dstart[0] = 0; - dstart[1] = 0; + // set the NC_FILL attribute + status = nc_set_fill(nc_state_file.nc_id, NC_FILL, &old_fill_mode); + if (status != NC_NOERR) { + log_err("Error setting fill value in %s", nc_state_file.fname); + } - if (global_domain.info.n_coord_dims == 1) { - dimids[0] = nc_state_file.ni_dimid; - dcount[0] = nc_state_file.ni_size; - } - else if (global_domain.info.n_coord_dims == 2) { - dimids[0] = nc_state_file.nj_dimid; - dcount[0] = nc_state_file.nj_size; + // define netcdf dimensions + status = nc_def_dim(nc_state_file.nc_id, global_domain.info.x_dim, + nc_state_file.ni_size, &(nc_state_file.ni_dimid)); + if (status != NC_NOERR) { + log_err("Error defining \"%s\" in %s", global_domain.info.x_dim, + nc_state_file.fname); + } - dimids[1] = nc_state_file.ni_dimid; - dcount[1] = nc_state_file.ni_size; - } - else { - log_err("COORD_DIMS_OUT should be 1 or 2"); - } + status = nc_def_dim(nc_state_file.nc_id, global_domain.info.y_dim, + nc_state_file.nj_size, &(nc_state_file.nj_dimid)); + if (status != NC_NOERR) { + log_err("Error defining \"%s\" in %s", global_domain.info.y_dim, + nc_state_file.fname); + } - // define the netcdf variable longitude - status = nc_def_var(nc_state_file.nc_id, global_domain.info.lon_var, - NC_DOUBLE, ndims, - dimids, &(lon_var_id)); - if (status != NC_NOERR) { - log_err("Error defining lon variable in %s", nc_state_file.fname); - } + status = nc_def_dim(nc_state_file.nc_id, "veg_class", + nc_state_file.veg_size, &(nc_state_file.veg_dimid)); + if (status != NC_NOERR) { + log_err("Error defining veg_class in %s", nc_state_file.fname); + } - status = nc_put_att_text(nc_state_file.nc_id, lon_var_id, "long_name", - strlen("longitude"), "longitude"); - if (status != NC_NOERR) { - log_err("Error adding attribute in %s", nc_state_file.fname); - } - status = nc_put_att_text(nc_state_file.nc_id, lon_var_id, "units", - strlen("degrees_east"), "degrees_east"); - if (status != NC_NOERR) { - log_err("Error adding attribute in %s", nc_state_file.fname); - } - status = nc_put_att_text(nc_state_file.nc_id, lon_var_id, "standard_name", - strlen("longitude"), "longitude"); - if (status != NC_NOERR) { - log_err("Error adding attribute in %s", nc_state_file.fname); - } + status = nc_def_dim(nc_state_file.nc_id, "snow_band", + nc_state_file.band_size, + &(nc_state_file.band_dimid)); + if (status != NC_NOERR) { + log_err("Error defining snow_band in %s", nc_state_file.fname); + } - if (global_domain.info.n_coord_dims == 1) { - dimids[0] = nc_state_file.nj_dimid; - dcount[0] = nc_state_file.nj_size; - } + status = nc_def_dim(nc_state_file.nc_id, "nlayer", + nc_state_file.layer_size, + &(nc_state_file.layer_dimid)); + if (status != NC_NOERR) { + log_err("Error defining nlayer in %s", nc_state_file.fname); + } - // define the netcdf variable latitude - status = nc_def_var(nc_state_file.nc_id, global_domain.info.lat_var, - NC_DOUBLE, ndims, - dimids, &(lat_var_id)); - if (status != NC_NOERR) { - log_err("Error defining lat variable in %s", nc_state_file.fname); - } - status = nc_put_att_text(nc_state_file.nc_id, lat_var_id, "long_name", - strlen("latitude"), "latitude"); - if (status != NC_NOERR) { - log_err("Error adding attribute in %s", nc_state_file.fname); - } - status = nc_put_att_text(nc_state_file.nc_id, lat_var_id, "units", - strlen("degrees_north"), "degrees_north"); - if (status != NC_NOERR) { - log_err("Error adding attribute in %s", nc_state_file.fname); - } - status = nc_put_att_text(nc_state_file.nc_id, lat_var_id, "standard_name", - strlen("latitude"), "latitude"); - if (status != NC_NOERR) { - log_err("Error adding attribute in %s", nc_state_file.fname); - } + status = nc_def_dim(nc_state_file.nc_id, "frost_area", + nc_state_file.frost_size, + &(nc_state_file.frost_dimid)); + if (status != NC_NOERR) { + log_err("Error defining frost_area in %s", nc_state_file.fname); + } - // populate lat/lon - if (global_domain.info.n_coord_dims == 1) { - dvar = calloc(nc_state_file.ni_size, sizeof(*dvar)); - if (dvar == NULL) { - log_err("Memory allocation error in vic_store()."); + status = nc_def_dim(nc_state_file.nc_id, "soil_node", + nc_state_file.node_size, + &(nc_state_file.node_dimid)); + if (status != NC_NOERR) { + log_err("Error defining soil_node in %s", nc_state_file.fname); } - dcount[0] = nc_state_file.ni_size; - for (i = 0; i < nc_state_file.ni_size; i++) { - dvar[i] = (double) global_domain.locations[i].longitude; + if (options.LAKES) { + status = nc_def_dim(nc_state_file.nc_id, "lake_node", + nc_state_file.lake_node_size, + &(nc_state_file.lake_node_dimid)); + if (status != NC_NOERR) { + log_err("Error defining lake_node in %s", nc_state_file.fname); + } } - status = - nc_put_vara_double(nc_state_file.nc_id, lon_var_id, dstart, dcount, - dvar); - if (status != NC_NOERR) { - log_err("Error adding data to lon in %s", nc_state_file.fname); + + // initialize dimids to invalid values + for (i = 0; i < MAXDIMS; i++) { + dimids[i] = -1; } - free(dvar); - dvar = calloc(nc_state_file.nj_size, sizeof(*dvar)); - if (dvar == NULL) { - log_err("Memory allocation error in vic_store()."); + // write dimension variables + + // Coordinate variables + ndims = global_domain.info.n_coord_dims; + dstart[0] = 0; + dstart[1] = 0; + + if (global_domain.info.n_coord_dims == 1) { + dimids[0] = nc_state_file.ni_dimid; + dcount[0] = nc_state_file.ni_size; + } + else if (global_domain.info.n_coord_dims == 2) { + dimids[0] = nc_state_file.nj_dimid; + dcount[0] = nc_state_file.nj_size; + + dimids[1] = nc_state_file.ni_dimid; + dcount[1] = nc_state_file.ni_size; } - dcount[0] = nc_state_file.nj_size; - for (i = 0; i < nc_state_file.nj_size; i++) { - dvar[i] = - (double) global_domain.locations[i].latitude; + else { + log_err("COORD_DIMS_OUT should be 1 or 2"); } - status = - nc_put_vara_double(nc_state_file.nc_id, lat_var_id, dstart, dcount, - dvar); + // define the netcdf variable longitude + status = nc_def_var(nc_state_file.nc_id, global_domain.info.lon_var, + NC_DOUBLE, ndims, dimids, &(lon_var_id)); if (status != NC_NOERR) { - log_err("Error adding data to lon in %s", nc_state_file.fname); - } - free(dvar); - } - else if (global_domain.info.n_coord_dims == 2) { - dvar = - calloc(nc_state_file.nj_size * nc_state_file.ni_size, - sizeof(*dvar)); - if (dvar == NULL) { - log_err("Memory allocation error in vic_store()."); + log_err("Error defining lon variable in %s", nc_state_file.fname); } - for (i = 0; i < nc_state_file.nj_size * nc_state_file.ni_size; i++) { - dvar[i] = (double) global_domain.locations[i].longitude; + status = nc_put_att_text(nc_state_file.nc_id, lon_var_id, "long_name", strlen( + "longitude"), "longitude"); + if (status != NC_NOERR) { + log_err("Error adding attribute in %s", nc_state_file.fname); } - status = - nc_put_vara_double(nc_state_file.nc_id, lon_var_id, dstart, dcount, - dvar); + status = nc_put_att_text(nc_state_file.nc_id, lon_var_id, "units", strlen( + "degrees_east"), "degrees_east"); if (status != NC_NOERR) { - log_err("Error adding data to lon in %s", nc_state_file.fname); + log_err("Error adding attribute in %s", nc_state_file.fname); + } + status = nc_put_att_text(nc_state_file.nc_id, lon_var_id, + "standard_name", strlen( + "longitude"), "longitude"); + if (status != NC_NOERR) { + log_err("Error adding attribute in %s", nc_state_file.fname); } - for (i = 0; i < nc_state_file.nj_size * nc_state_file.ni_size; i++) { - dvar[i] = (double) global_domain.locations[i].latitude; + if (global_domain.info.n_coord_dims == 1) { + dimids[0] = nc_state_file.nj_dimid; + dcount[0] = nc_state_file.nj_size; } - status = - nc_put_vara_double(nc_state_file.nc_id, lat_var_id, dstart, dcount, - dvar); + + // define the netcdf variable latitude + status = nc_def_var(nc_state_file.nc_id, global_domain.info.lat_var, + NC_DOUBLE, ndims, dimids, &(lat_var_id)); if (status != NC_NOERR) { - log_err("Error adding data to lat in %s", nc_state_file.fname); + log_err("Error defining lat variable in %s", nc_state_file.fname); + } + status = nc_put_att_text(nc_state_file.nc_id, lat_var_id, "long_name", strlen( + "latitude"), "latitude"); + if (status != NC_NOERR) { + log_err("Error adding attribute in %s", nc_state_file.fname); + } + status = nc_put_att_text(nc_state_file.nc_id, lat_var_id, "units", strlen( + "degrees_north"), "degrees_north"); + if (status != NC_NOERR) { + log_err("Error adding attribute in %s", nc_state_file.fname); + } + status = nc_put_att_text(nc_state_file.nc_id, lat_var_id, + "standard_name", strlen("latitude"), + "latitude"); + if (status != NC_NOERR) { + log_err("Error adding attribute in %s", nc_state_file.fname); } - free(dvar); - } - else { - log_err("COORD_DIMS_OUT should be 1 or 2"); - } + // leave define mode + status = nc_enddef(nc_state_file.nc_id); + if (status != NC_NOERR) { + log_err("Error leaving define mode for %s", nc_state_file.fname); + } - // Variables for other dimensions (all 1-dimensional) - ndims = 1; - d1start[0] = 0; + // populate lat/lon + if (global_domain.info.n_coord_dims == 1) { + dvar = calloc(nc_state_file.ni_size, sizeof(*dvar)); + if (dvar == NULL) { + log_err("Memory allocation error in vic_store()."); + } - // vegetation classes - dimids[0] = nc_state_file.veg_dimid; - ivar = malloc(options.NVEGTYPES * sizeof(*ivar)); - if (ivar == NULL) { - log_err("Memory allocation error in vic_store()."); - } - for (j = 0; j < options.NVEGTYPES; j++) { - ivar[j] = (int) j; - } - gather_put_nc_field_int(nc_state_file.fname, &(nc_state_file.open), - &(nc_state_file.nc_id), - nc_state_file.d_fillvalue, - dimids, ndims, "veg_class", d1start, d1count, - ivar); - for (i = 0; i < ndims; i++) { - dimids[i] = -1; - } - free(ivar); + dcount[0] = nc_state_file.ni_size; + // implicitly nested loop over ni and nj with j set to 0 + for (i = 0; i < nc_state_file.ni_size; i++) { + dvar[i] = (double) global_domain.locations[i].longitude; + } + status = nc_put_vara_double(nc_state_file.nc_id, lon_var_id, dstart, + dcount, dvar); + if (status != NC_NOERR) { + log_err("Error adding data to lon in %s", nc_state_file.fname); + } + free(dvar); - // snow bands - dimids[0] = nc_state_file.band_dimid; - ivar = malloc(options.SNOW_BAND * sizeof(*ivar)); - if (ivar == NULL) { - log_err("Memory allocation error in vic_store()."); - } - for (j = 0; j < options.SNOW_BAND; j++) { - ivar[j] = (int) j; - } - gather_put_nc_field_int(nc_state_file.fname, &(nc_state_file.open), - &(nc_state_file.nc_id), - nc_state_file.d_fillvalue, - dimids, ndims, "snow_band", d1start, d1count, - ivar); - for (i = 0; i < ndims; i++) { - dimids[i] = -1; - } - free(ivar); + dvar = calloc(nc_state_file.nj_size, sizeof(*dvar)); + if (dvar == NULL) { + log_err("Memory allocation error in vic_store()."); + } + dcount[0] = nc_state_file.nj_size; + // implicitly nested loop over ni and nj with i set to 0; + // j stride = ni_size + for (j = 0; j < nc_state_file.nj_size; j++) { + dvar[j] = + (double) global_domain.locations[j * + nc_state_file.ni_size]. + latitude; + } - // soil layers - dimids[0] = nc_state_file.layer_dimid; - ivar = malloc(options.Nlayer * sizeof(*ivar)); - if (ivar == NULL) { - log_err("Memory allocation error in vic_store()."); - } - for (j = 0; j < options.Nlayer; j++) { - ivar[j] = (int) j; - } - gather_put_nc_field_int(nc_state_file.fname, &(nc_state_file.open), - &(nc_state_file.nc_id), - nc_state_file.d_fillvalue, - dimids, ndims, "layer", d1start, d1count, - ivar); - for (i = 0; i < ndims; i++) { - dimids[i] = -1; - } - free(ivar); + status = nc_put_vara_double(nc_state_file.nc_id, lat_var_id, dstart, + dcount, dvar); + if (status != NC_NOERR) { + log_err("Error adding data to lon in %s", nc_state_file.fname); + } + free(dvar); + } + else if (global_domain.info.n_coord_dims == 2) { + dvar = + calloc(nc_state_file.nj_size * nc_state_file.ni_size, + sizeof(*dvar)); + if (dvar == NULL) { + log_err("Memory allocation error in vic_store()."); + } - // frost areas - dimids[0] = nc_state_file.frost_dimid; - ivar = malloc(options.Nfrost * sizeof(*ivar)); - if (ivar == NULL) { - log_err("Memory allocation error in vic_store()."); - } - for (j = 0; j < options.Nfrost; j++) { - ivar[j] = (int) j; - } - gather_put_nc_field_int(nc_state_file.fname, &(nc_state_file.open), - &(nc_state_file.nc_id), - nc_state_file.d_fillvalue, - dimids, ndims, "frost_area", d1start, d1count, - ivar); - for (i = 0; i < ndims; i++) { - dimids[i] = -1; - } - free(ivar); + for (i = 0; i < nc_state_file.nj_size * nc_state_file.ni_size; + i++) { + dvar[i] = (double) global_domain.locations[i].longitude; + } + status = nc_put_vara_double(nc_state_file.nc_id, lon_var_id, dstart, + dcount, dvar); + if (status != NC_NOERR) { + log_err("Error adding data to lon in %s", nc_state_file.fname); + } - // soil thermal node deltas - dimids[0] = nc_state_file.node_dimid; - gather_put_nc_field_double(nc_state_file.fname, &(nc_state_file.open), - &(nc_state_file.nc_id), - nc_state_file.d_fillvalue, - dimids, ndims, "dz_node", d1start, d1count, - soil_con[0].dz_node); - for (i = 0; i < ndims; i++) { - dimids[i] = -1; - } + for (i = 0; i < nc_state_file.nj_size * nc_state_file.ni_size; + i++) { + dvar[i] = (double) global_domain.locations[i].latitude; + } + status = nc_put_vara_double(nc_state_file.nc_id, lat_var_id, dstart, + dcount, dvar); + if (status != NC_NOERR) { + log_err("Error adding data to lat in %s", nc_state_file.fname); + } - // soil thermal node depths - dimids[0] = nc_state_file.node_dimid; - gather_put_nc_field_double(nc_state_file.fname, &(nc_state_file.open), - &(nc_state_file.nc_id), - nc_state_file.d_fillvalue, - dimids, ndims, "node_depth", d1start, d1count, - soil_con[0].Zsum_node); - for (i = 0; i < ndims; i++) { - dimids[i] = -1; - } + free(dvar); + } + else { + log_err("COORD_DIMS_OUT should be 1 or 2"); + } - if (options.LAKES) { - // lake nodes - dimids[0] = nc_state_file.lake_node_dimid; - ivar = malloc(options.NLAKENODES * sizeof(*ivar)); + // Variables for other dimensions (all 1-dimensional) + ndims = 1; + d1start[0] = 0; + + // vegetation classes + dimids[0] = nc_state_file.veg_dimid; + d1count[0] = nc_state_file.veg_size; + ivar = malloc(nc_state_file.veg_size * sizeof(*ivar)); if (ivar == NULL) { log_err("Memory allocation error in vic_store()."); } - for (j = 0; j < options.SNOW_BAND; j++) { + for (j = 0; j < nc_state_file.veg_size; j++) { + ivar[j] = (int) j + 1; + } + put_nc_field_int(nc_state_file.fname, &(nc_state_file.open), + &(nc_state_file.nc_id), nc_state_file.d_fillvalue, + dimids, ndims, + "veg_class", d1start, d1count, ivar); + for (i = 0; i < ndims; i++) { + dimids[i] = -1; + } + free(ivar); + + // snow bands + dimids[0] = nc_state_file.band_dimid; + d1count[0] = nc_state_file.band_size; + ivar = malloc(nc_state_file.band_size * sizeof(*ivar)); + if (ivar == NULL) { + log_err("Memory allocation error in vic_store()."); + } + for (j = 0; j < nc_state_file.band_size; j++) { ivar[j] = (int) j; } - gather_put_nc_field_int(nc_state_file.fname, &(nc_state_file.open), - &(nc_state_file.nc_id), - nc_state_file.d_fillvalue, - dimids, ndims, "lake_node", d1start, d1count, - ivar); + put_nc_field_int(nc_state_file.fname, &(nc_state_file.open), + &(nc_state_file.nc_id), nc_state_file.d_fillvalue, + dimids, ndims, + "snow_band", d1start, d1count, ivar); for (i = 0; i < ndims; i++) { dimids[i] = -1; } free(ivar); - } + + // soil layers + dimids[0] = nc_state_file.layer_dimid; + d1count[0] = nc_state_file.layer_size; + ivar = malloc(nc_state_file.layer_size * sizeof(*ivar)); + if (ivar == NULL) { + log_err("Memory allocation error in vic_store()."); + } + for (j = 0; j < nc_state_file.layer_size; j++) { + ivar[j] = (int) j; + } + put_nc_field_int(nc_state_file.fname, &(nc_state_file.open), + &(nc_state_file.nc_id), nc_state_file.d_fillvalue, + dimids, ndims, + "layer", d1start, d1count, ivar); + for (i = 0; i < ndims; i++) { + dimids[i] = -1; + } + free(ivar); + + // frost areas + dimids[0] = nc_state_file.frost_dimid; + d1count[0] = nc_state_file.frost_size; + ivar = malloc(nc_state_file.frost_size * sizeof(*ivar)); + if (ivar == NULL) { + log_err("Memory allocation error in vic_store()."); + } + for (j = 0; j < nc_state_file.frost_size; j++) { + ivar[j] = (int) j; + } + put_nc_field_int(nc_state_file.fname, &(nc_state_file.open), + &(nc_state_file.nc_id), nc_state_file.d_fillvalue, + dimids, ndims, + "frost_area", d1start, d1count, ivar); + for (i = 0; i < ndims; i++) { + dimids[i] = -1; + } + free(ivar); + + // soil thermal node deltas + dimids[0] = nc_state_file.node_dimid; + d1count[0] = nc_state_file.node_size; + put_nc_field_double(nc_state_file.fname, &(nc_state_file.open), + &(nc_state_file.nc_id), nc_state_file.d_fillvalue, + dimids, ndims, + "dz_node", d1start, d1count, soil_con[0].dz_node); + for (i = 0; i < ndims; i++) { + dimids[i] = -1; + } + + // soil thermal node depths + dimids[0] = nc_state_file.node_dimid; + d1count[0] = nc_state_file.node_size; + put_nc_field_double(nc_state_file.fname, &(nc_state_file.open), + &(nc_state_file.nc_id), nc_state_file.d_fillvalue, + dimids, ndims, + "node_depth", d1start, d1count, + soil_con[0].Zsum_node); + for (i = 0; i < ndims; i++) { + dimids[i] = -1; + } + + if (options.LAKES) { + // lake nodes + dimids[0] = nc_state_file.lake_node_dimid; + d1count[0] = nc_state_file.lake_node_size; + ivar = malloc(nc_state_file.lake_node_size * sizeof(*ivar)); + if (ivar == NULL) { + log_err("Memory allocation error in vic_store()."); + } + for (j = 0; j < nc_state_file.lake_node_size; j++) { + ivar[j] = (int) j; + } + put_nc_field_int(nc_state_file.fname, &(nc_state_file.open), + &(nc_state_file.nc_id), nc_state_file.d_fillvalue, + dimids, ndims, + "lake_node", d1start, d1count, ivar); + for (i = 0; i < ndims; i++) { + dimids[i] = -1; + } + free(ivar); + } + } // end if (mpi_rank == 0) // write state variables @@ -372,9 +480,6 @@ vic_store(dmy_struct *dmy_current) } // initialize starts and counts - d1start[0] = 0; - d1count[0] = 1; - d2start[0] = 0; d2start[1] = 0; d2count[0] = global_domain.n_ny; @@ -1969,111 +2074,3 @@ vic_store(dmy_struct *dmy_current) free(dvar); free(fvar); } - -void -initialize_state_file(nc_file_struct *nc, - dmy_struct *dmy_current) -{ - extern size_t current; - extern filenames_struct filenames; - extern domain_struct global_domain; - extern option_struct options; - - int status; - int old_fill_mode; - - sprintf(nc->fname, "%s.%04d%02d%02d_%05u.nc", - filenames.statefile, dmy_current->year, dmy_current->month, - dmy_current->day, dmy_current->dayseconds); - - nc->c_fillvalue = NC_FILL_CHAR; - nc->i_fillvalue = NC_FILL_INT; - nc->d_fillvalue = NC_FILL_DOUBLE; - nc->f_fillvalue = NC_FILL_FLOAT; - - nc->ni_size = global_domain.n_nx; - nc->nj_size = global_domain.n_ny; - nc->veg_size = options.NVEGTYPES; - nc->band_size = options.SNOW_BAND; - nc->layer_size = options.Nlayer; - nc->frost_size = options.Nfrost; - nc->node_size = options.Nnode; - if (options.LAKES) { - nc->lake_node_size = options.NLAKENODES; - } - - // open the netcdf file - status = nc_create(nc->fname, get_nc_mode(options.STATE_FORMAT), - &(nc->nc_id)); - if (status != NC_NOERR) { - log_err("Error creating %s", nc->fname); - } - nc->open = true; - - // set the NC_FILL attribute - status = nc_set_fill(nc->nc_id, NC_FILL, &old_fill_mode); - if (status != NC_NOERR) { - log_err("Error setting fill value in %s", nc->fname); - } - - // define netcdf dimensions - status = - nc_def_dim(nc->nc_id, global_domain.info.x_dim, nc->ni_size, - &(nc->ni_dimid)); - if (status != NC_NOERR) { - log_err("Error defining \"%s\" in %s", global_domain.info.x_dim, - nc->fname); - } - - status = - nc_def_dim(nc->nc_id, global_domain.info.y_dim, nc->nj_size, - &(nc->nj_dimid)); - if (status != NC_NOERR) { - log_err("Error defining \"%s\" in %s", global_domain.info.y_dim, - nc->fname); - } - - status = nc_def_dim(nc->nc_id, "veg_class", nc->veg_size, - &(nc->veg_dimid)); - if (status != NC_NOERR) { - log_err("Error defining veg_class in %s", nc->fname); - } - - status = nc_def_dim(nc->nc_id, "snow_band", nc->band_size, - &(nc->band_dimid)); - if (status != NC_NOERR) { - log_err("Error defining snow_band in %s", nc->fname); - } - - status = nc_def_dim(nc->nc_id, "nlayer", nc->layer_size, - &(nc->layer_dimid)); - if (status != NC_NOERR) { - log_err("Error defining nlayer in %s", nc->fname); - } - - status = nc_def_dim(nc->nc_id, "frost_area", nc->frost_size, - &(nc->frost_dimid)); - if (status != NC_NOERR) { - log_err("Error defining frost_area in %s", nc->fname); - } - - status = - nc_def_dim(nc->nc_id, "soil_node", nc->node_size, &(nc->node_dimid)); - if (status != NC_NOERR) { - log_err("Error defining soil_node in %s", nc->fname); - } - - if (options.LAKES) { - status = nc_def_dim(nc->nc_id, "lake_node", nc->lake_node_size, - &(nc->lake_node_dimid)); - if (status != NC_NOERR) { - log_err("Error defining lake_node in %s", nc->fname); - } - } - - // leave define mode - status = nc_enddef(nc->nc_id); - if (status != NC_NOERR) { - log_err("Error leaving define mode for %s", nc->fname); - } -} diff --git a/vic/vic_run/include/vic_def.h b/vic/vic_run/include/vic_def.h index ed42d998d..935468b5c 100644 --- a/vic/vic_run/include/vic_def.h +++ b/vic/vic_run/include/vic_def.h @@ -361,14 +361,15 @@ typedef struct { unsigned short int skipyear; /**< Number of years to skip before writing output data */ unsigned short int startday; /**< Starting day of the simulation */ + unsigned short int startmonth; /**< Starting month of the simulation */ unsigned int startsec; /**< Seconds since midnight when simulation will start */ - unsigned short int startmonth; /**< Starting month of the simulation */ unsigned short int startyear; /**< Starting year of the simulation */ unsigned short int stateday; /**< Day of the simulation at which to save model state */ unsigned short int statemonth; /**< Month of the simulation at which to save model state */ + unsigned int statesec; /**< Seconds since midnight at which to save state */ unsigned short int stateyear; /**< Year of the simulation at which to save model state */ unsigned short int calendar; /**< Date/time calendar */ diff --git a/vic/vic_run/include/vic_run.h b/vic/vic_run/include/vic_run.h index 5d72ac2fa..9cf4bc18e 100644 --- a/vic/vic_run/include/vic_run.h +++ b/vic/vic_run/include/vic_run.h @@ -108,6 +108,7 @@ double canopy_evap(layer_data_struct *, veg_var_struct *, bool, void colavg(double *, double *, double *, double, double *, int, double, double); double compute_coszen(double, double, double, unsigned short int, unsigned int); +void compute_derived_lake_dimensions(lake_var_struct *, lake_con_struct); void compute_pot_evap(size_t, double, double, double, double, double, double, double, double, double, double *, char, double, double, double, double *); @@ -182,8 +183,8 @@ double IceEnergyBalance(double, va_list); void iceform(double *, double *, double, double, double *, int, double, double, double, double *, double *, double *, double *, double); void icerad(double, double, double, double *, double *, double *); -int initialize_lake(lake_var_struct *, lake_con_struct, soil_con_struct *, - cell_data_struct *, int); +void initialize_lake(lake_var_struct *, lake_con_struct, soil_con_struct *, + cell_data_struct *, bool); int lakeice(double, double, double, double, double, double *, double, double *, double *, double, double); void latent_heat_from_snow(double, double, double, double, double, double, diff --git a/vic/vic_run/src/compute_derived_lake_dimensions.c b/vic/vic_run/src/compute_derived_lake_dimensions.c new file mode 100644 index 000000000..5fcf1cbae --- /dev/null +++ b/vic/vic_run/src/compute_derived_lake_dimensions.c @@ -0,0 +1,105 @@ +/****************************************************************************** + * @section DESCRIPTION + * + * This routine computes those lake variables that are completely dependent + * on lake depth and basin dimensions. + * + * @section LICENSE + * + * The Variable Infiltration Capacity (VIC) macroscale hydrological model + * Copyright (C) 2014 The Land Surface Hydrology Group, Department of Civil + * and Environmental Engineering, University of Washington. + * + * The VIC model is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License + * as published by the Free Software Foundation; either version 2 + * of the License, or (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License along with + * this program; if not, write to the Free Software Foundation, Inc., + * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. + *****************************************************************************/ + +#include + +/****************************************************************************** + * @brief This routine computes those lake variables that are completely + * dependent on lake depth and basin dimensions. + *****************************************************************************/ +void +compute_derived_lake_dimensions(lake_var_struct *lake, + lake_con_struct lake_con) +{ + extern parameters_struct param; + + int k; + int status; + double depth; + double tmp_volume; + + /* number and thicknesses of lake layers */ + if (lake->ldepth > param.LAKE_MAX_SURFACE && lake->ldepth < 2 * + param.LAKE_MAX_SURFACE) { + /* Not quite enough for two full layers. */ + lake->surfdz = lake->ldepth / 2.; + lake->dz = lake->ldepth / 2.; + lake->activenod = 2; + } + else if (lake->ldepth >= 2 * param.LAKE_MAX_SURFACE) { + /* More than two layers. */ + lake->surfdz = param.LAKE_MAX_SURFACE; + lake->activenod = (int) (lake->ldepth / param.LAKE_MAX_SURFACE); + if (lake->activenod > MAX_LAKE_NODES) { + lake->activenod = MAX_LAKE_NODES; + } + lake->dz = (lake->ldepth - lake->surfdz) / + ((double) (lake->activenod - 1)); + } + else if (lake->ldepth > DBL_EPSILON) { + lake->surfdz = lake->ldepth; + lake->dz = 0.0; + lake->activenod = 1; + } + else { + lake->surfdz = 0.0; + lake->dz = 0.0; + lake->activenod = 0; + lake->ldepth = 0.0; + } + + // lake_con.basin equals the surface area at specific depths as input by + // the user in the lake parameter file or calculated in read_lakeparam(), + // lake->surface equals the area at the top of each dynamic solution layer + + for (k = 0; k <= lake->activenod; k++) { + if (k == 0) { + depth = lake->ldepth; + } + else { + depth = lake->dz * (lake->activenod - k); + } + status = get_sarea(lake_con, depth, &(lake->surface[k])); + if (status < 0) { + log_err("Error in get_sarea: record = %d, depth = %f, " + "sarea = %e", 0, depth, lake->surface[k]); + } + } + + lake->sarea = lake->surface[0]; + status = get_volume(lake_con, lake->ldepth, &tmp_volume); + if (status < 0) { + log_err("Error in get_volume: record = %d, depth = %f, " + "volume = %e", 0, depth, tmp_volume); + } + else if (status > 0) { + log_err("Warning in get_volume: lake depth exceeds maximum; " + "setting to maximum; record = %d", 0); + } + lake->volume = tmp_volume + lake->ice_water_eq; + +} diff --git a/vic/drivers/shared_all/src/initialize_lake.c b/vic/vic_run/src/initialize_lake.c similarity index 66% rename from vic/drivers/shared_all/src/initialize_lake.c rename to vic/vic_run/src/initialize_lake.c index 593f8ea4b..ecb341044 100644 --- a/vic/drivers/shared_all/src/initialize_lake.c +++ b/vic/vic_run/src/initialize_lake.c @@ -27,118 +27,32 @@ #include /****************************************************************************** - * @brief This routine initializes the lake variables for each new grid cell + * @brief This routine initializes the lake variables. + * lake->ldepth and lake->temp[] + * must have been defined outside this function *****************************************************************************/ -int +void initialize_lake(lake_var_struct *lake, lake_con_struct lake_con, soil_con_struct *soil_con, cell_data_struct *cell, - int skip_hydro) + bool preserve_essentials) { - extern option_struct options; - extern parameters_struct param; + extern option_struct options; - size_t i, j; - int k; - int status; - double depth; - double tmp_volume; + size_t i, j; - for (i = 0; i < MAX_LAKE_NODES; i++) { - lake->temp[i] = max(soil_con->avg_temp, 0.0); - } - - lake->areai = 0.0; - lake->coldcontent = 0.0; - lake->hice = 0.0; - lake->ice_water_eq = 0.0; - lake->new_ice_area = 0.0; - lake->pack_temp = 0.0; - lake->pack_water = 0.0; - lake->SAlbedo = 0.0; - lake->sdepth = 0.0; - lake->surf_temp = 0.0; - lake->surf_water = 0.0; - lake->swe = 0.0; - lake->swe_save = 0.0; - lake->tempi = 0.0; - - /********************************************************************/ - /* Initialize lake physical parameters. */ - /********************************************************************/ - - if (!skip_hydro) { - if (lake_con.depth_in > lake_con.z[0]) { - lake_con.depth_in = lake_con.z[0]; - } - - lake->ldepth = lake_con.depth_in; - - if (lake->ldepth > param.LAKE_MAX_SURFACE && lake->ldepth < 2 * - param.LAKE_MAX_SURFACE) { - /* Not quite enough for two full layers. */ - lake->surfdz = lake->ldepth / 2.; - lake->dz = lake->ldepth / 2.; - lake->activenod = 2; - } - else if (lake->ldepth >= 2 * param.LAKE_MAX_SURFACE) { - /* More than two layers. */ - lake->surfdz = param.LAKE_MAX_SURFACE; - lake->activenod = (int) (lake->ldepth / param.LAKE_MAX_SURFACE); - if (lake->activenod > MAX_LAKE_NODES) { - lake->activenod = MAX_LAKE_NODES; - } - lake->dz = - (lake->ldepth - - lake->surfdz) / ((double) (lake->activenod - 1)); - } - else if (lake->ldepth > 0.0) { - lake->surfdz = lake->ldepth; - lake->dz = 0.0; - lake->activenod = 1; - } - else { - lake->surfdz = 0.0; - lake->dz = 0.0; - lake->activenod = 0; - lake->ldepth = 0.0; - } - - // lake_con.basin equals the surface area at specific depths as input by - // the user in the lake parameter file or calculated in read_lakeparam(), - // lake->surface equals the area at the top of each dynamic solution layer - - for (k = 0; k <= lake->activenod; k++) { - if (k == 0) { - depth = lake->ldepth; - } - else { - depth = lake->dz * (lake->activenod - k); - } - status = get_sarea(lake_con, depth, &(lake->surface[k])); - if (status < 0) { - log_err("Error in get_sarea: record = %d, depth = %f, " - "sarea = %e", 0, depth, lake->surface[k]); - } - } - - lake->sarea = lake->surface[0]; + if (!preserve_essentials) { + // Initialize lake depth and the + // variables that depend on it + lake->ldepth = 0.0; + lake->ice_water_eq = 0.0; // needed for lake volume + compute_derived_lake_dimensions(lake, lake_con); lake->sarea_save = lake->sarea; - status = get_volume(lake_con, lake->ldepth, &tmp_volume); - if (status < 0) { - log_err("Error in get_volume: record = %d, depth = %f, " - "volume = %e", 0, depth, tmp_volume); - return(status); - } - else if (status > 0) { - log_err("Warning in get_volume: lake depth exceeds maximum; " - "setting to maximum; record = %d", 0); - } - lake->volume = tmp_volume + lake->ice_water_eq; + lake->swe_save = lake->swe; lake->volume_save = lake->volume; - // Initialize lake moisture fluxes to 0 + // Initialize lake moisture fluxes lake->baseflow_in = 0.0; lake->baseflow_out = 0.0; lake->channel_in = 0.0; @@ -149,13 +63,31 @@ initialize_lake(lake_var_struct *lake, lake->runoff_out = 0.0; lake->snowmlt = 0.0; lake->vapor_flux = 0.0; - } // if (!skip_hydro) - // Initialize other miscellaneous lake properties - lake->aero_resist = 0; - for (k = 0; k < lake->activenod; k++) { - lake->density[k] = CONST_RHOFW; + // Initialize other terms + lake->aero_resist = 0; + lake->soil.pot_evap = 0.0; + } + + // Initialize other lake state variables + lake->areai = 0.0; + lake->coldcontent = 0.0; + for (i = 0; i < MAX_LAKE_NODES; i++) { + lake->density[i] = CONST_RHOFW; + lake->temp[i] = 0; } + lake->hice = 0.0; + lake->ice_throughfall = 0.0; + lake->new_ice_area = lake->areai; + lake->pack_temp = 0.0; + lake->pack_water = 0.0; + lake->SAlbedo = 0.0; + lake->sdepth = 0.0; + lake->surf_temp = 0.0; + lake->surf_water = 0.0; + lake->swe = 0.0; + lake->tempavg = 0.0; + lake->tempi = 0.0; // Initialize the snow, energy, and soil components of lake structure // If we implement heat flux between lake and underlying soil, we will need to initialize these more correctly @@ -268,11 +200,9 @@ initialize_lake(lake_var_struct *lake, lake->energy.snow_flux = 0.0; // Soil states and fluxes lake->soil.asat = 1.0; - if (!skip_hydro) { - lake->soil.baseflow = 0.0; - lake->soil.inflow = 0.0; - lake->soil.runoff = 0.0; - } + lake->soil.baseflow = 0.0; + lake->soil.inflow = 0.0; + lake->soil.runoff = 0.0; lake->soil.rootmoist = 0.0; lake->soil.wetness = 1.0; for (i = 0; i < 2; i++) { @@ -292,9 +222,6 @@ initialize_lake(lake_var_struct *lake, } lake->soil.zwt = 0.0; lake->soil.zwt_lumped = 0.0; - if (!skip_hydro) { - lake->soil.pot_evap = 0.0; - } if (options.CARBON) { lake->soil.RhLitter = 0.0; lake->soil.RhLitter2Atm = 0.0; @@ -305,6 +232,4 @@ initialize_lake(lake_var_struct *lake, lake->soil.CInter = 0.0; lake->soil.CSlow = 0.0; } - - return(0); } diff --git a/vic/vic_run/src/lakes.eb.c b/vic/vic_run/src/lakes.eb.c index 5d056ed6d..426200dcd 100644 --- a/vic/vic_run/src/lakes.eb.c +++ b/vic/vic_run/src/lakes.eb.c @@ -1823,7 +1823,7 @@ water_balance(lake_var_struct *lake, int isave_n; double inflow_volume; double surfacearea, ldepth; - double i; + double i_dbl; double index; size_t j, k, frost_area; double Tnew[MAX_LAKE_NODES]; @@ -2132,56 +2132,7 @@ water_balance(lake_var_struct *lake, * 4. Adjust the activenodes and lake area array. **********************************************************************/ - if (lake->ldepth > param.LAKE_MAX_SURFACE && lake->ldepth < 2 * - param.LAKE_MAX_SURFACE) { - /* Not quite enough for two full layers. */ - lake->surfdz = lake->ldepth / 2.; - lake->dz = lake->ldepth / 2.; - lake->activenod = 2; - } - else if (lake->ldepth >= 2 * param.LAKE_MAX_SURFACE) { - /* More than two layers. */ - lake->surfdz = param.LAKE_MAX_SURFACE; - lake->activenod = (int) (lake->ldepth / param.LAKE_MAX_SURFACE); - if (lake->activenod > MAX_LAKE_NODES) { - lake->activenod = MAX_LAKE_NODES; - } - lake->dz = - (lake->ldepth - lake->surfdz) / ((double) (lake->activenod - 1)); - } - else if (lake->ldepth > DBL_EPSILON) { - lake->surfdz = lake->ldepth; - lake->dz = 0.0; - lake->activenod = 1; - } - else { - lake->runoff_out += (lake->volume - lake->ice_water_eq); - lake->volume = lake->ice_water_eq; - lake->surfdz = 0.0; - lake->dz = 0.0; - lake->activenod = 0; - lake->ldepth = 0.0; - } - - // lake_con.basin equals the surface area at specific depths as input by - // the user in the lake parameter file or calculated in read_lakeparam(), - // lake->surface equals the area at the top of each dynamic solution layer - - /* Re-calculate lake surface area */ - for (k = 0; k <= lake->activenod; k++) { - if (k == 0) { - ldepth = lake->ldepth; - } - else { - ldepth = lake->dz * (lake->activenod - k); - } - - ErrorFlag = get_sarea(lake_con, ldepth, &(lake->surface[k])); - if (ErrorFlag == ERROR) { - log_err("Something went wrong in get_sarea; depth = " - "%f, sarea = %e", ldepth, lake->surface[k]); - } - } + compute_derived_lake_dimensions(lake, lake_con); // Final lakefraction if (lake->new_ice_area > lake->surface[0]) { @@ -2202,7 +2153,7 @@ water_balance(lake_var_struct *lake, if (lake->activenod != isave_n) { for (k = 0; k < lake->activenod; k++) { Tnew[k] = 0.0; - for (i = 0; i < isave_n; i++) { + for (i_dbl = 0; i_dbl < isave_n; i_dbl++) { index += (1. / lake->activenod); Tnew[k] += lake->temp[(int) floor(index)]; } @@ -2305,7 +2256,25 @@ water_balance(lake_var_struct *lake, } } else { // lake didn't exist at beginning of time step; create new lake - initialize_lake(lake, lake_con, &soil_con, &(cell[iveg][band]), 1); + // Reset all non-essential lake variables + initialize_lake(lake, lake_con, &soil_con, &(cell[iveg][band]), + true); + + // Compute lake dimensions from current lake depth + compute_derived_lake_dimensions(lake, lake_con); + + // Assign initial temperatures + for (k = 0; k < lake->activenod; k++) { + lake->temp[k] = all_vars->energy[iveg][band].T[0]; + } + lake->tempavg = lake->temp[0]; + lake->energy.Tsurf = all_vars->energy[iveg][band].Tsurf; + for (k = 0; k < options.Nnode; k++) { + lake->energy.T[k] = all_vars->energy[iveg][band].T[k]; + } + for (k = 0; k < options.Nlayer; k++) { + lake->soil.layer[k].T = all_vars->cell[iveg][band].layer[k].T; + } } } else if (lakefrac > 0.0) { // lake is gone at end of time step, but existed at beginning of step