Loading [MathJax]/extensions/tex2jax.js
PISM, A Parallel Ice Sheet Model 2.2.2-d6b3a29ca committed by Constantine Khrulev on 2025-03-28
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Pages
Time.cc
Go to the documentation of this file.
1// Copyright (C) 2011-2024 Constantine Khroulev
2//
3// This file is part of PISM.
4//
5// PISM is free software; you can redistribute it and/or modify it under the
6// terms of the GNU General Public License as published by the Free Software
7// Foundation; either version 3 of the License, or (at your option) any later
8// version.
9//
10// PISM is distributed in the hope that it will be useful, but WITHOUT ANY
11// WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
12// FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
13// details.
14//
15// You should have received a copy of the GNU General Public License
16// along with PISM; if not, write to the Free Software
17// Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18
19#include <cmath>
20#include <cassert> // assert
21#include <cstring> // strlen()
22#include <limits>
23
24#include "pism/util/Time.hh"
25
26#include "pism/external/calcalcs/calcalcs.h"
27
28#include "pism/util/ConfigInterface.hh"
29#include "pism/util/VariableMetadata.hh"
30#include "pism/util/pism_utilities.hh"
31#include "pism/util/error_handling.hh"
32#include "pism/util/io/File.hh"
33#include "pism/util/io/io_helpers.hh"
34#include "pism/util/Logger.hh"
35#include "pism/util/io/IO_Flags.hh"
36
37namespace pism {
38
39//! Get the reference date from a file.
40static std::string reference_date_from_file(const File &file,
41 const std::string &time_name,
42 const std::string &default_value,
43 bool stop_on_error) {
44
45 if (file.variable_exists(time_name)) {
46 std::string time_units = file.read_text_attribute(time_name, "units");
47
48 if (not time_units.empty()) {
49 // Check if the time_units includes a reference date.
50 size_t position = time_units.find("since");
51
52 if (position != std::string::npos) {
53 return string_strip(time_units.substr(position + strlen("since")));
54 }
55
56 if (stop_on_error) {
57
59 "%s:units = \"%s\" in '%s' does not contain a reference date",
60 time_name.c_str(),
61 time_units.c_str(),
62 file.name().c_str());
63 }
64 } else if (stop_on_error) {
66 "the '%s' variable in '%s' has no units",
67 time_name.c_str(), file.name().c_str());
68 }
69 } else if (stop_on_error) {
71 "'%s' variable is not present in '%s'.",
72 time_name.c_str(), file.name().c_str());
73 }
74
75 return default_value;
76}
77
78//! Get the calendar name from a file.
79static std::string calendar_from_file(const File &file,
80 const std::string &time_name,
81 const std::string &default_value,
82 bool stop_on_error) {
83
84 if (file.variable_exists(time_name)) {
85 std::string calendar_name = file.read_text_attribute(time_name, "calendar");
86
87 if (not calendar_name.empty()) {
88 return calendar_name;
89 }
90
91 if (stop_on_error) {
93 "the '%s' variable in '%s' has no calendar attribute",
94 time_name.c_str(), file.name().c_str());
95 }
96
97 } else if (stop_on_error) {
99 "'%s' variable is not present in '%s'.",
100 time_name.c_str(), file.name().c_str());
101 }
102
103 return default_value;
104}
105
106/*!
107 * Get the reference date from a file or the configuration parameter.
108 */
109static std::string reference_date(const File *input_file,
110 const Config &config,
111 const Logger &log) {
112
113 auto default_reference_date = config.get_string("time.reference_date");
114
115 if (input_file != nullptr) {
116 // input file is not empty
117
118 auto time = config.get_string("time.dimension_name");
119
120 if (not config.get_flag("input.bootstrap")) {
121 // restarting from a file: use the reference date in this file
122 bool stop_on_error = true;
123 return reference_date_from_file(*input_file, time, default_reference_date, stop_on_error);
124 }
125
126 // Bootstrapping: use the configuration parameter and warn about mismatches
127 bool stop_on_error = false;
128 auto ref_date = reference_date_from_file(*input_file, time, default_reference_date, stop_on_error);
129
130 if (ref_date != default_reference_date) {
131 log.message(2,
132 "WARNING: Using reference date %s\n"
133 " instead of the one present in the input file '%s' (%s)\n",
134 default_reference_date.c_str(), input_file->name().c_str(), ref_date.c_str());
135 }
136
137 return ref_date;
138 }
139
140 return default_reference_date;
141}
142
143/*!
144 * Get the calendar name from a file or a configuration parameter.
145 */
146static std::string calendar(const File *input_file,
147 const Config &config,
148 const Logger &log) {
149 auto default_calendar = config.get_string("time.calendar");
150
151 if (input_file != nullptr) {
152 // input file is not empty
153
154 auto time = config.get_string("time.dimension_name");
155
156 if (not config.get_flag("input.bootstrap")) {
157 // restarting from a file: use the calendar in this file
158 bool stop_on_error = true;
159 return calendar_from_file(*input_file, time, default_calendar, stop_on_error);
160 }
161
162 // Bootstrapping: use the configuration parameter and warn about mismatches
163 bool stop_on_error = false;
164 auto calendar = calendar_from_file(*input_file, time, default_calendar, stop_on_error);
165
166 if (calendar != default_calendar) {
167 log.message(2,
168 "WARNING: Using calendar %s\n"
169 " instead of the one present in the input file '%s' (%s)\n",
170 default_calendar.c_str(), input_file->name().c_str(), calendar.c_str());
171 }
172
173 return default_calendar;
174 }
175
176 return default_calendar;
177}
178
179/*!
180 * Increment the date corresponding to `T` by `years` years.
181 */
182static double increment_date(const units::Unit &time_units,
183 const std::string &calendar,
184 double T, double years) {
185
186 double whole_years_double = std::floor(years);
187 if (whole_years_double > std::numeric_limits<int>::max() or
188 whole_years_double < std::numeric_limits<int>::min()) {
190 "time offset of %f years does not fit in an 'int'",
191 whole_years_double);
192 }
193
194 int whole_years = static_cast<int>(whole_years_double);
195 double year_fraction = years - whole_years;
196 const double day_length = 86400.0;
197
198 // Get the date corresponding to time T:
199 auto date = time_units.date(T, calendar);
200
201 // shift the date by the number of whole years requested
202 date.year += whole_years;
203
204 // check if the resulting year is a leap year:
205 int leap = 0;
206 {
208 assert(cal != NULL);
209 int errcode = ccs_isleap(cal, date.year, &leap);
210 if (errcode != 0) {
211 throw RuntimeError::formatted(PISM_ERROR_LOCATION, "CalCalcs error: %s",
212 ccs_err_str(errcode));
213 }
215 }
216
217 double result = 0.0;
218 if (leap == 0 and date.month == 2 and date.day == 29) {
219 // avoid passing an impossible date to UDUNITS (no February 29 in non-leap years):
220 date.day -= 1;
221 result = time_units.time(date, calendar);
222 // add back the day we substracted above
223 result += day_length;
224 } else {
225 result = time_units.time(date, calendar);
226 }
227
228 int year_length = 360;
229 if (calendar != "360_day") {
230 year_length = (leap == 1) ? 366 : 365;
231 }
232
233 result += year_fraction * (year_length * day_length);
234
235 return result;
236}
237
238/*!
239 * Parse the date.
240 *
241 * `input` can be
242 *
243 * - a YYYY-MM-DD date (YYYY can be negative)
244 *
245 * - a number (interpreted as the number of years since the reference date in
246 * `time_units`)
247 *
248 * - a number with units attached ("1 day", etc) interpreted as time since the reference
249 * date in `time_units`
250 */
251static double parse_date(const std::string &input,
252 const units::Unit &time_units,
253 const std::string &calendar) {
254
255 std::string spec = string_strip(input);
256
257 if (spec.empty()) {
259 "got an empty date specification: '%s'",
260 input.c_str());
261 }
262
263 // We need to remember if the year was negative in the input string: split() will ignore
264 // empty tokens separated by "-", so "-1000-1-1" will produce ["1000", "1", "1"].
265 bool year_is_negative = (spec[0] == '-');
266
267 if (year_is_negative and not member(calendar, {"365_day", "360_day", "noleap"})) {
269 "negative dates such as '%s' are not allowed with the '%s' calendar.\n"
270 "Please submit a bug report if this is a problem.",
271 spec.c_str(), calendar.c_str());
272 }
273
274 auto parts = split(spec, '-');
275
276 if (parts.size() == 3) {
277 std::vector<int> numbers;
278 for (const auto &p : parts) {
279 try {
280 long int n = parse_integer(p);
281
282 if (n > std::numeric_limits<int>::max() or
283 n < std::numeric_limits<int>::min()) {
285 "%ld does not fit in an 'int'",
286 n);
287 }
288
289 numbers.push_back(static_cast<int>(n));
290 } catch (RuntimeError &e) {
291 e.add_context("parsing a date specification %s",
292 spec.c_str());
293 throw;
294 }
295 }
296
297 if (year_is_negative) {
298 numbers[0] *= -1;
299 }
300
301 // Validate the calendar string and the date in this calendar:
302 {
304 if (cal == NULL) {
306 "calendar string '%s' is invalid",
307 calendar.c_str());
308 }
309
310 int dummy = 0;
311 int errcode = ccs_date2jday(cal, numbers[0], numbers[1], numbers[2], &dummy);
312 if (errcode != 0) {
315 "date %s is invalid in the %s calendar",
316 spec.c_str(), calendar.c_str());
317 }
319 }
320
321 units::DateTime d{numbers[0], numbers[1], numbers[2], 0, 0, 0.0};
322
323 return time_units.time(d, calendar);
324 } // end of the block processing dates written as Y-M-D
325
326 // "spec" must be a number or a number with units attached to it
327 try {
328 // check if strtod() can parse it:
329 char *endptr = NULL;
330 double t = strtod(spec.c_str(), &endptr);
331 if (*endptr == '\0') {
332 // strtod() parsed it successfully: assume that it is in years. This will return
333 // time in seconds.
334 return increment_date(time_units, calendar, 0, t);
335 }
336
337 // strtod() failed -- assume that this is a number followed by units compatible
338 // with seconds
339 return units::convert(time_units.system(), 1.0, spec, "seconds");
340 } catch (RuntimeError &e) {
341 e.add_context("parsing the date " + spec);
342 throw;
343 }
344}
345
346/*!
347 * Return the start time.
348 */
349static double start_time(const Config &config,
350 const File *file,
351 const std::string &reference_date,
352 const std::string &calendar,
353 const units::Unit &time_units) {
354
355 auto time_start = config.get_string("time.start");
356
357 if (not time_start.empty()) {
358 return parse_date(time_start, time_units, calendar);
359 }
360
361 if (file == nullptr) {
362 // 0.0 corresponds to the reference date
363 return 0.0;
364 }
365
366 // get the calendar in this file
367 auto time_name = config.get_string("time.dimension_name");
368 bool stop_on_error = false;
369 auto file_calendar = calendar_from_file(*file, time_name, calendar, stop_on_error);
370 auto ref_date = reference_date_from_file(*file, time_name, reference_date, stop_on_error);
371
372 if (file_calendar != calendar) {
374 "calendar in '%s' (%s) does not match the selected calendar (%s)",
375 file->name().c_str(), file_calendar.c_str(), calendar.c_str());
376 }
377
378 if (ref_date != reference_date) {
380 "reference date in '%s' (%s) does not match the selected date (%s)",
381 file->name().c_str(), ref_date.c_str(), reference_date.c_str());
382 }
383
384 // FIXME: it would make sense to get the length of the time dimension and read the last
385 // number instead.
386 if (file->dimension_length(time_name) > 0) {
387 auto time = io::read_1d_variable(*file, time_name, time_units.format(), time_units.system());
388
389 return time.back();
390 }
391
392 return 0.0;
393}
394
395static double end_time(const Config &config,
396 double time_start,
397 const std::string &calendar,
398 const units::Unit &time_units) {
399 auto time_end = config.get_string("time.end");
400
401 if (not time_end.empty()) {
402 // parse use time_end and use it
403 return parse_date(time_end, time_units, calendar);
404 }
405
406 // use time_start and run_length
407 auto run_length = config.get_number("time.run_length", "seconds");
408 return time_start + run_length;
409}
410
411//! Convert model years into seconds using the year length
412//! corresponding to the current calendar.
413/*! Do not use this to convert quantities other than time intervals!
414 */
415double Time::years_to_seconds(double input) const {
416 return input * m_year_length;
417}
418
419//! Convert seconds into model years using the year length
420//! corresponding to the current calendar.
421/*! Do not use this to convert quantities other than time intervals!
422 */
423double Time::seconds_to_years(double input) const {
424 return input / m_year_length;
425}
426
427void Time::init_calendar(const std::string &calendar_string) {
428
429 if (not pism_is_valid_calendar_name(calendar_string)) {
431 "unsupported calendar: %s", calendar_string.c_str());
432 }
433
434 m_calendar_string = calendar_string;
435
436 double seconds_per_day = convert(m_unit_system, 1.0, "day", "seconds");
437 if (calendar_string == "360_day") {
438 m_year_length = 360 * seconds_per_day;
439 } else if (member(calendar_string, {"365_day", "noleap"})) {
440 m_year_length = 365 * seconds_per_day;
441 } else {
442 // use the ~365.2524-day year
443 m_year_length = convert(m_unit_system, 1.0, "year", "seconds");
444 }
445}
446
447//! \brief Sets the current time (in seconds since the reference time).
448void Time::set(double new_time) {
449 m_time_in_seconds = new_time;
450}
451
452void Time::set_start(double new_start) {
453 m_run_start = new_start;
454}
455
456void Time::set_end(double new_end) {
457 m_run_end = new_end;
458}
459
460//! \brief Current time, in seconds.
461double Time::current() const {
462 return m_time_in_seconds;
463}
464
465double Time::start() const {
466 return m_run_start;
467}
468
469double Time::end() const {
470 return m_run_end;
471}
472
473std::string Time::units_string() const {
474 return m_time_units.format();
475}
476
478 return m_time_units;
479}
480
481//! \brief Returns the calendar string.
482std::string Time::calendar() const {
483 return m_calendar_string;
484}
485
486void Time::step(double delta_t) {
487 m_time_in_seconds += delta_t;
488
489 // If we are less than m_t_eps seconds from the end of the run, reset
490 // m_time_in_seconds to avoid taking a very small (and useless) time step.
494 }
495}
496
497std::string Time::run_length() const {
499}
500
501double Time::day_of_the_year_to_year_fraction(unsigned int day) const {
502 const double sperd = 86400.0;
503 return (sperd / m_year_length) * (double) day;
504}
505
506std::vector<double> Time::parse_times(const std::string &spec) const {
507 if (spec.find(',') != std::string::npos) {
508 // a list will always contain a comma because at least two numbers are
509 // needed to specify reporting intervals
510 return parse_list(spec);
511 }
512
513 // it must be a range specification
514 return parse_range(spec);
515}
516
517std::vector<double> Time::parse_list(const std::string &spec) const {
518 std::vector<double> result;
519
520 try {
521 for (const auto &s : split(spec, ',')) {
522 result.emplace_back(parse_date(s, m_time_units, m_calendar_string));
523 }
524 } catch (RuntimeError &e) {
525 e.add_context("parsing a list of dates %s", spec.c_str());
526 throw;
527 }
528
529 return result;
530}
531
532/**
533 * Parses an interval specification string.
534 *
535 * @param[in] spec specification string
536 *
537 */
538auto Time::parse_interval_length(const std::string &spec) const -> Interval {
539
540 // check if it is a keyword
541 if (spec == "hourly") {
542 return {3600.0, SIMPLE};
543 }
544
545 if (spec == "daily") {
546 return {86400.0, SIMPLE};
547 }
548
549 if (spec == "monthly") {
550 return {1.0, MONTHLY};
551 }
552
553 if (spec == "yearly") {
554 return {1.0, YEARLY};
555 }
556
557 if (not m_simple_calendar) {
558 if (spec.find("year") != std::string::npos or spec.find("month") != std::string::npos) {
560 "interval length '%s' with the calendar '%s' is not supported",
561 spec.c_str(), m_calendar_string.c_str());
562 }
563 }
564
565 try {
566 units::Unit seconds(m_unit_system, "seconds");
567 units::Unit one(m_unit_system, "1");
568 units::Unit tmp(m_unit_system, spec);
569
570 // Check if these units are compatible with "seconds" or "1". The
571 // latter allows intervals of the form "0.5", which stands for "half
572 // of a model year". This also discards interval specs such as "days
573 // since 1-1-1", even though "days" is compatible with "seconds".
574 if (units::are_convertible(tmp, seconds)) {
575 units::Converter c(tmp, seconds);
576
577 return {c(1.0), SIMPLE};
578 }
579
580 if (units::are_convertible(tmp, one)) {
581 units::Converter c(tmp, one);
582
583 // convert from years to seconds without using UDUNITS-2 (this
584 // way we handle 360-day and 365-day years correctly)
585 return {years_to_seconds(c(1.0)), SIMPLE};
586 }
587 } catch (RuntimeError &e) {
588 e.add_context("processing interval length " + spec);
589 throw;
590 }
591
593 "interval length '%s' is invalid", spec.c_str());
594}
595
596
597std::vector<double> Time::parse_range(const std::string &spec) const {
598 double
599 time_start = m_run_start,
600 time_end = m_run_end;
601
602 Interval I{0.0, SIMPLE};
603
604 if (spec == "hourly") {
605 I = {3600.0, SIMPLE};
606 } else if (spec == "daily") {
607 I = {86400.0, SIMPLE};
608 } else if (spec == "monthly") {
609 I = {1.0, MONTHLY};
610 } else if (spec == "yearly") {
611 I = {1.0, YEARLY};
612 } else {
613
614 auto parts = pism::split(spec, ':');
615
616 if (parts.size() == 1) {
617 I = parse_interval_length(parts[0]);
618
619 } else if (parts.size() == 3) {
620 time_start = parse_date(parts[0], m_time_units, m_calendar_string);
621 I = parse_interval_length(parts[1]);
622 time_end = parse_date(parts[2], m_time_units, m_calendar_string);
623 } else {
625 "a time range must consist of exactly 3 parts separated by colons (got '%s').",
626 spec.c_str());
627 }
628 }
629
630 std::vector<double> result;
631 compute_times(time_start, time_end, I, result);
632 return result;
633}
634
635/**
636 * Compute times corresponding to a "simple" time range.
637 *
638 * This is a time range with a fixed step ("hourly" is an example).
639 *
640 * @param[in] time_start beginning of the time interval, in seconds
641 * @param[in] delta step, in seconds
642 * @param[in] time_end end of the interval, in seconds
643 * @param[out] result list of model times
644 *
645 */
646void Time::compute_times_simple(double time_start, double delta, double time_end,
647 std::vector<double> &result) const {
648 if (time_start >= time_end) {
649 throw RuntimeError::formatted(PISM_ERROR_LOCATION, "a >= b in time range a:dt:b (got a = %s, b = %s)",
650 this->date(time_start).c_str(), this->date(time_end).c_str());
651 }
652
653 if (delta <= 0) {
654 throw RuntimeError::formatted(PISM_ERROR_LOCATION, "dt <= 0 in time range a:dt:b (got dt = %f seconds)", delta);
655 }
656
657 int k = 0;
658 double t = time_start;
659
660 result.clear();
661 do {
662 if (t >= this->start() && t <= this->end()) {
663 result.push_back(t);
664 }
665 k += 1;
666 t = time_start + k * delta;
667 } while (t <= time_end);
668}
669
670double Time::convert_time_interval(double T, const std::string &units) const {
671 if (member(units, {"year", "years", "yr", "a"})) {
672 return this->seconds_to_years(T); // uses year length here
673 }
674 return convert(m_unit_system, T, "seconds", units);
675}
676
677/*!
678
679 See http://meteora.ucsd.edu/~pierce/calcalcs/index.html and
680 http://cfconventions.org/Data/cf-conventions/cf-conventions-1.8/cf-conventions.html#calendar
681
682 for more details about supported calendars.
683 */
684Time::Time(MPI_Comm com,
685 Config::ConstPtr config,
686 const Logger &log,
687 units::System::Ptr unit_system)
688 : m_config(config),
689 m_unit_system(unit_system),
690 m_time_units(unit_system, "seconds since 1-1-1") {
691
692 m_t_eps = config->get_number("time_stepping.resolution", "seconds");
693
694 auto input_file = config->get_string("input.file");
695
696 std::unique_ptr<File> file{};
697 if (not input_file.empty()) {
698 file.reset(new File(com, input_file, io::PISM_NETCDF3, io::PISM_READONLY));
699 }
700
701 // set the reference date
702 auto ref_date = reference_date(file.get(), *config, log);
703
704 try {
705 // this will validate the reference date
706 m_time_units = units::Unit(m_unit_system, "seconds since " + ref_date);
707 } catch (RuntimeError &e) {
708 e.add_context("setting time units");
709 throw;
710 }
711
712 m_calendar_string = ::pism::calendar(file.get(), *config, log);
714 m_simple_calendar = member(m_calendar_string, {"360_day", "365_day", "no_leap"});
715
716 m_run_start = start_time(*config,
717 file.get(),
718 ref_date,
721
722 m_run_end = end_time(*config,
726
727 if (m_run_start > m_run_end) {
728 auto start = date(m_run_start);
729 auto end = date(m_run_end);
731 "Negative run length. Start: %s, end: %s.",
732 start.c_str(), end.c_str());
733 }
734
736
737 auto time_file = config->get_string("time.file");
738 bool continue_run = config->get_flag("time.file.continue");
739
740 if (not time_file.empty()) {
741 log.message(2,
742 "* Setting time from '%s'...\n",
743 time_file.c_str());
744 init_from_file(com, time_file, log, not continue_run);
745 }
746}
747
748//! \brief Sets the time from a NetCDF file with a time dimension (`-time_file`).
749/*!
750 * Sets
751 * - calendar
752 * - reference date
753 * - start time
754 * - current time
755 * - end time
756 *
757 * This allows running PISM for the duration of the available forcing.
758 */
759void Time::init_from_file(MPI_Comm com,
760 const std::string &filename,
761 const Logger &log,
762 bool set_start_time) {
763 try {
764 std::string time_name = m_config->get_string("time.dimension_name");
765
766 File file(com, filename, io::PISM_NETCDF3, io::PISM_READONLY); // OK to use netcdf3
767
768 // Set the calendar name from file.
769 std::string new_calendar = file.read_text_attribute(time_name, "calendar");
770 if (not new_calendar.empty()) {
771 init_calendar(new_calendar);
772 }
773
774 // Set the reference date of internal units.
775 {
776 bool stop_on_error = true;
777 std::string date_string = reference_date_from_file(file,
778 time_name,
779 "irrelevant (not used)",
780 stop_on_error);
781 m_time_units = units::Unit(m_unit_system, "seconds since " + date_string);
782 }
783
784 // Read time information from the file.
785 std::vector<double> time;
786 std::string time_bounds_name = file.read_text_attribute(time_name, "bounds");
787 if (not time_bounds_name.empty()) {
788 // use the time bounds
789 time = io::read_bounds(file, time_bounds_name, m_time_units.format(), m_unit_system);
790 } else {
791 // use the time axis
792 time = io::read_1d_variable(file, time_name, m_time_units.format(), m_unit_system);
793 }
794
795 // Set time.
796 if (set_start_time) {
797 this->set_start(time.front());
798 this->set(time.front());
799 } else {
800 log.message(2, "* Using start time from an -i file to continue an interrupted run.\n");
801 }
802 this->set_end(time.back());
803 } catch (RuntimeError &e) {
804 e.add_context("initializing model time from \"%s\"", filename.c_str());
805 throw;
806 }
807}
808
809double Time::year_fraction(double T) const {
810
812
813 units::DateTime D2{D.year, 1, 1, 0, 0, 0.0};
814
815 auto year_start = m_time_units.time(D2, m_calendar_string);
816
817 D2.year += 1;
818 auto next_year_start = m_time_units.time(D2, m_calendar_string);
819
820 return (T - year_start) / (next_year_start - year_start);
821}
822
823std::string Time::date(double T) const {
825
826 double hour = date.hour + date.minute / 60.0 + date.second / 3600.0;
827
828 return pism::printf("%04d-%02d-%02d %06.3fh",
829 date.year, date.month, date.day, hour);
830}
831
832double Time::calendar_year_start(double T) const {
834
835 units::DateTime D2{D.year, 1, 1, 0, 0, 0.0};
836
838}
839
840
841double Time::increment_date(double T, double years) const {
842 return ::pism::increment_date(m_time_units, m_calendar_string, T, years);
843}
844
845void Time::compute_times_monthly(std::vector<double> &result) const {
846
847 double time = m_run_start;
848
849 // get the date corresponding to the current time
851
852 // beginning of the current month
853 units::DateTime d{date.year, date.month, 1, 0, 0, 0.0};
854
855 result.clear();
856 while (true) {
857 // find the time corresponding to the beginning of the current month
859
860 if (time > m_run_end) {
861 break;
862 }
863
864 if (time >= m_run_start and time <= m_run_end) {
865 result.push_back(time);
866 }
867
868 if (d.month == 12) {
869 d.year += 1;
870 d.month = 1;
871 } else {
872 d.month += 1;
873 }
874 }
875}
876
877void Time::compute_times_yearly(std::vector<double> &result) const {
878
879 double time = m_run_start;
880 // get the date corresponding to the current time
882
883 units::DateTime d{date.year, 1, 1, 0, 0, 0.0};
884
885 result.clear();
886 while (true) {
887 // find the time corresponding to the beginning of the current year
889
890 if (time > m_run_end) {
891 break;
892 }
893
894 if (time >= m_run_start and time <= m_run_end) {
895 result.push_back(time);
896 }
897
898 d.year += 1;
899 }
900}
901
902void Time::compute_times(double time_start, double time_end,
903 const Interval &interval,
904 std::vector<double> &result) const {
905 switch (interval.type) {
906 case SIMPLE:
907 compute_times_simple(time_start, interval.dt, time_end, result);
908 break;
909 case MONTHLY:
910 compute_times_monthly(result);
911 break;
912 case YEARLY:
913 compute_times_yearly(result);
914 break;
915 }
916}
917
918/*!
919 * Check if the modeled time interval is a subset of the time interval in a forcing file.
920 *
921 * Returns silently if it is, otherwise throws an exception with an error message.
922 */
924 double forcing_start,
925 double forcing_end) {
926
927 double run_start = time.start();
928 double run_end = time.end();
929
930 if (not (run_start >= forcing_start and
931 run_end <= forcing_end)) {
933 "A time-dependent forcing has to span the whole length of the simulation\n"
934 " Run time: [%s, %s]\n"
935 " Forcing data: [%s, %s]",
936 time.date(run_start).c_str(),
937 time.date(run_end).c_str(),
938 time.date(forcing_start).c_str(),
939 time.date(forcing_end).c_str());
940 }
941}
942
943} // end of namespace pism
char * ccs_err_str(int error_code)
Definition calcalcs.c:908
int ccs_isleap(calcalcs_cal *calendar, int year, int *leap)
Definition calcalcs.c:336
void ccs_free_calendar(calcalcs_cal *cc)
Definition calcalcs.c:1338
int ccs_date2jday(calcalcs_cal *calendar, int year, int month, int day, int *jday)
Definition calcalcs.c:442
calcalcs_cal * ccs_init_calendar(const char *calname)
Definition calcalcs.c:104
double get_number(const std::string &name, UseFlag flag=REMEMBER_THIS_USE) const
std::shared_ptr< const Config > ConstPtr
std::string get_string(const std::string &name, UseFlag flag=REMEMBER_THIS_USE) const
bool get_flag(const std::string &name, UseFlag flag=REMEMBER_THIS_USE) const
A class for storing and accessing PISM configuration flags and parameters.
bool variable_exists(const std::string &short_name) const
Checks if a variable exists.
Definition File.cc:378
std::string name() const
Definition File.cc:274
unsigned int dimension_length(const std::string &name) const
Get the length of a dimension.
Definition File.cc:420
std::string read_text_attribute(const std::string &var_name, const std::string &att_name) const
Get a text attribute.
Definition File.cc:645
High-level PISM I/O class.
Definition File.hh:55
void message(int threshold, const char format[],...) const __attribute__((format(printf
Print a message to the log.
Definition Logger.cc:49
A basic logging class.
Definition Logger.hh:40
void add_context(const std::string &message)
Add a message providing some context. This way we can (sort of) get a stack trace even though C++ exc...
static RuntimeError formatted(const ErrorLocation &location, const char format[],...) __attribute__((format(printf
build a RuntimeError with a formatted message
double increment_date(double T, double years) const
Definition Time.cc:841
std::vector< double > parse_list(const std::string &spec) const
Definition Time.cc:517
std::string run_length() const
Returns the length of the current run, in years.
Definition Time.cc:497
const Config::ConstPtr m_config
Definition Time.hh:151
std::string units_string() const
Internal time units as a string.
Definition Time.cc:473
void compute_times_monthly(std::vector< double > &result) const
Definition Time.cc:845
double years_to_seconds(double input) const
Definition Time.cc:415
void set(double new_time)
Sets the current time (in seconds since the reference time).
Definition Time.cc:448
double convert_time_interval(double T, const std::string &units) const
Convert time interval from seconds to given units. Handle 'years' using the year length corresponding...
Definition Time.cc:670
double m_year_length
number of seconds in a year, for "year fraction"
Definition Time.hh:159
void set_end(double new_end)
Definition Time.cc:456
void compute_times_simple(double time_start, double delta, double time_end, std::vector< double > &result) const
Definition Time.cc:646
void init_calendar(const std::string &calendar)
Definition Time.cc:427
double start() const
Definition Time.cc:465
double calendar_year_start(double T) const
Returns the model time in seconds corresponding to the beginning of the year T falls into.
Definition Time.cc:832
void compute_times(double time_start, double time_end, const Interval &interval, std::vector< double > &result) const
Definition Time.cc:902
@ MONTHLY
Definition Time.hh:129
double end() const
Definition Time.cc:469
std::vector< double > parse_range(const std::string &spec) const
Definition Time.cc:597
double m_run_end
run end tim, in seconds since the reference time
Definition Time.hh:168
double year_fraction(double T) const
Returns the fraction of a year that passed since the last beginning of a year. Only useful in codes w...
Definition Time.cc:809
double m_time_in_seconds
current time, in seconds since the reference time
Definition Time.hh:162
Interval parse_interval_length(const std::string &spec) const
Definition Time.cc:538
std::string m_calendar_string
CF calendar string.
Definition Time.hh:171
bool m_simple_calendar
Definition Time.hh:173
double seconds_to_years(double input) const
Definition Time.cc:423
double current() const
Current time, in seconds.
Definition Time.cc:461
units::Unit units() const
Definition Time.cc:477
std::vector< double > parse_times(const std::string &spec) const
Definition Time.cc:506
Time(MPI_Comm com, Config::ConstPtr config, const Logger &log, units::System::Ptr unit_system)
Definition Time.cc:684
void compute_times_yearly(std::vector< double > &result) const
Definition Time.cc:877
double day_of_the_year_to_year_fraction(unsigned int day) const
Convert the day number to the year fraction.
Definition Time.cc:501
void step(double delta_t)
Advance by delta_t seconds.
Definition Time.cc:486
void set_start(double new_start)
Definition Time.cc:452
std::string date(double T) const
Returns the date corresponding to time T.
Definition Time.cc:823
double m_run_start
run start time, in seconds since the reference time
Definition Time.hh:165
const units::System::Ptr m_unit_system
Definition Time.hh:152
units::Unit m_time_units
Definition Time.hh:153
std::string calendar() const
Returns the calendar string.
Definition Time.cc:482
void init_from_file(MPI_Comm com, const std::string &filename, const Logger &log, bool set_start_time)
Sets the time from a NetCDF file with a time dimension (-time_file).
Definition Time.cc:759
double m_t_eps
Time resolution, in seconds.
Definition Time.hh:156
Time management class.
Definition Time.hh:55
std::shared_ptr< System > Ptr
Definition Units.hh:47
DateTime date(double T, const std::string &calendar) const
Definition Units.cc:153
std::string format() const
Definition Units.cc:141
System::Ptr system() const
Definition Units.cc:149
double time(const DateTime &d, const std::string &calendar) const
Definition Units.cc:168
#define PISM_ERROR_LOCATION
#define n
Definition exactTestM.c:37
@ PISM_NETCDF3
Definition IO_Flags.hh:57
@ PISM_READONLY
open an existing file for reading only
Definition IO_Flags.hh:68
std::vector< double > read_1d_variable(const File &file, const std::string &variable_name, const std::string &units, std::shared_ptr< units::System > unit_system)
std::vector< double > read_bounds(const File &file, const std::string &bounds_variable_name, const std::string &internal_units, std::shared_ptr< units::System > unit_system)
bool are_convertible(const Unit &u1, const Unit &u2)
Definition Units.cc:242
double convert(System::Ptr system, double input, const std::string &spec1, const std::string &spec2)
Convert a quantity from unit1 to unit2.
Definition Units.cc:70
static double increment_date(const units::Unit &time_units, const std::string &calendar, double T, double years)
Definition Time.cc:182
std::string printf(const char *format,...)
static const double k
Definition exactTestP.cc:42
static double end_time(const Config &config, double time_start, const std::string &calendar, const units::Unit &time_units)
Definition Time.cc:395
std::string string_strip(const std::string &input)
void check_forcing_duration(const Time &time, double forcing_start, double forcing_end)
Definition Time.cc:923
static std::string calendar_from_file(const File &file, const std::string &time_name, const std::string &default_value, bool stop_on_error)
Get the calendar name from a file.
Definition Time.cc:79
static std::string reference_date(const File *input_file, const Config &config, const Logger &log)
Definition Time.cc:109
static std::string calendar(const File *input_file, const Config &config, const Logger &log)
Definition Time.cc:146
bool member(const std::string &string, const std::set< std::string > &set)
static double start_time(const Config &config, const File *file, const std::string &reference_date, const std::string &calendar, const units::Unit &time_units)
Definition Time.cc:349
static double parse_date(const std::string &input, const units::Unit &time_units, const std::string &calendar)
Definition Time.cc:251
long int parse_integer(const std::string &input)
static std::string reference_date_from_file(const File &file, const std::string &time_name, const std::string &default_value, bool stop_on_error)
Get the reference date from a file.
Definition Time.cc:40
bool pism_is_valid_calendar_name(const std::string &name)
Definition Time.hh:34
std::vector< std::string > split(const std::string &input, char separator)
Transform a separator-separated list (a string) into a vector of strings.
IntervalType type
Definition Time.hh:133