OOPS
Run.cc
Go to the documentation of this file.
1 /*
2  * (C) Copyright 2009-2016 ECMWF.
3  *
4  * This software is licensed under the terms of the Apache Licence Version 2.0
5  * which can be obtained at http://www.apache.org/licenses/LICENSE-2.0.
6  * In applying this licence, ECMWF does not waive the privileges and immunities
7  * granted to it by virtue of its status as an intergovernmental organisation nor
8  * does it submit to any jurisdiction.
9  */
10 
11 #include "oops/runs/Run.h"
12 
13 #include <string>
14 
15 #include "eckit/config/LocalConfiguration.h"
16 #include "eckit/config/YAMLConfiguration.h"
17 #include "eckit/exception/Exceptions.h"
18 
19 #include "oops/runs/Application.h"
20 #include "oops/util/LibOOPS.h"
21 #include "oops/util/Logger.h"
22 #include "oops/util/ObjectCountHelper.h"
23 #include "oops/util/printRunStats.h"
24 #include "oops/util/TimerHelper.h"
25 
26 #ifdef ENABLE_GPTL
27 #include <gptl.h>
28 #endif
29 
30 // GPTL implementation in JEDI requires retrieval of both integer and string env. var.
31 int getEnv(const std::string& env, int default_value) {
32  if (::getenv(env.c_str())) {return eckit::Translator<std::string, int>()(::getenv(env.c_str()));}
33  return default_value;
34 }
35 
36 std::string getEnv(const std::string& env, std::string default_value) {
37  if ((::getenv(env.c_str()))) {return ::getenv(env.c_str());}
38  return default_value;
39 }
40 
41 namespace oops {
42 
43  // -----------------------------------------------------------------------------
44 
45 Run::Run(int argc, char** argv) : eckit::Main(argc, argv, "OOPS_HOME"), config_(), timer_() {
46  // Initialize MPI and LibOOPS variables that require eckit::Main
47 #ifdef ENABLE_GPTL
48  int do_profile = getEnv("OOPS_PROFILE", 0); // Default is profiling disabled
49  if (do_profile) {
50  int ret;
51  // All GPTLsetutr() and GPTLsetoption() calls must precede GPTLinitialize()
52 
53  // Default JEDI behavior: Change the GPTL default from expensive gettimeofday() to inexpensive
54  // register read ("nanotime"). User can control with $OOPS_UNDERLYING_TIMER
55  std::string utr = getEnv("OOPS_UNDERLYING_TIMER", "nanotime");
56  if (utr == "nanotime") {
57  ret = GPTLsetutr(GPTLnanotime); // Use x86-specific register read to gather timings
58  } else if (utr == "gettimeofday") {
59  ret = GPTLsetutr(GPTLgettimeofday); // Use gettimeofday() to gather timings
60  } else {
61  Log::warning() << "OOPS_UNDERLYING_TIMER=" << utr << " is invalid: ignoring" << std::endl;
62  }
63 
64  // Set printout method. Another common method is full_tree, but that can create tons of output
65  ret = GPTLsetoption(GPTLprint_method, GPTLmost_frequent);
66 
67  // For MPI codes, synchronize and time collectives and receives to avoid mis-assignment
68  // of load imbalance to true MPI time. Disable by changing 1 to 0 here or via env. var.
69  int sync_mpi = getEnv("OOPS_SYNC_MPI", 1);
70  if (sync_mpi)
71  ret = GPTLsetoption(GPTLsync_mpi, 1);
72 
73  // Default behavior is do not print hi-water mark increases as the process runs
74  int dopr_memusage = getEnv("OOPS_MEMUSAGE", 0);
75  if (dopr_memusage)
76  ret = GPTLsetoption(GPTLdopr_memusage, 1); // Print growth of resident set size (RSS)
77 
78  ret = GPTLinitialize(); // Initialize GPTL timing library
79  }
80 #endif
81 
82  LibOOPS::instance().initialise();
83 
84 // Get configuration file and optional output file from command line
85  ASSERT(argc >= 2);
86  eckit::PathName configfile = argv[1];
87  if (argc == 3) {
88  eckit::PathName outputfile;
89  outputfile = argv[2];
90  LibOOPS::instance().teeOutput(outputfile);
91  }
92 
93 // Read configuration
94  config_.reset(new eckit::YAMLConfiguration(configfile));
95 
96  // Configure TestReference with "test:" sub-config
97  if (config_->has("test"))
98  LibOOPS::instance().testReferenceInitialise(config_->getSubConfiguration("test"));
99 
100  Log::info() << "Configuration input file is: " << configfile << std::endl;
101  Log::info() << "Full configuration is:" << *config_ << std::endl;
102 
103 // Start measuring performance
104  util::TimerHelper::start();
105  util::ObjectCountHelper::start();
106 }
107 
108 // -----------------------------------------------------------------------------
109 
111  LibOOPS::instance().finalise(); // Finalize MPI and logs
112 }
113 
114 // -----------------------------------------------------------------------------
115 
116 int Run::execute(const Application & app) {
117  util::printRunStats("Run start", true);
118  int status = 1;
119  Log::info() << "Run: Starting " << app << std::endl;
120  try {
121  status = app.execute(*config_);
122  }
123  catch(const eckit::Exception & e) {
124  status = 1;
125  Log::error() << e.what() << " caught in " << Here() << std::endl;
126  Log::error() << "Exception: " << app << " terminating..." << std::endl;
127  eckit::Exception::exceptionStack(eckit::Log::error(), true);
128  }
129  catch(const std::exception & e) {
130  status = 1;
131  Log::error() << "Exception: " << e.what() << std::endl;
132  Log::error() << "Exception: " << app << " terminating..." << std::endl;
133  }
134  catch(...) {
135  status = 1;
136  Log::error() << "Unknown exception: " << app << " terminating..." << std::endl;
137  }
138  Log::info() << std::endl << "Run: Finishing " << app << std::endl;
139 
140 // Performance diagnostics
141  util::ObjectCountHelper::stop();
142  util::TimerHelper::stop();
143  util::printRunStats("Run end", true);
144 
145  Log::info() << "Run: Finishing " << app << " with status = " << status << std::endl;
146  return status;
147 }
148 
149 // -----------------------------------------------------------------------------
150 
151 } // namespace oops
int getEnv(const std::string &env, int default_value)
Definition: Run.cc:31
virtual int execute(const eckit::Configuration &) const =0
int execute(const Application &)
Definition: Run.cc:116
Run(int argc, char **argv)
Definition: Run.cc:45
virtual ~Run()
Definition: Run.cc:110
std::unique_ptr< const eckit::YAMLConfiguration > config_
Definition: Run.h:39
int error
Definition: compare.py:168
Definition: FieldL95.h:22
The namespace for the main oops code.