Spicy
driver.h
1 // Copyright (c) 2020-now by the Zeek Project. See LICENSE for details.
2 // p
3 
4 #pragma once
5 
6 #include <fstream>
7 #include <iostream>
8 #include <map>
9 #include <memory>
10 #include <set>
11 #include <string>
12 #include <string_view>
13 #include <unordered_map>
14 #include <unordered_set>
15 #include <utility>
16 #include <vector>
17 
18 #include <hilti/rt/filesystem.h>
19 
20 #include <hilti/base/logger.h>
21 #include <hilti/base/result.h>
22 #include <hilti/base/timing.h>
23 #include <hilti/compiler/context.h>
24 #include <hilti/compiler/unit.h>
25 
26 namespace hilti {
27 
28 class JIT;
29 struct Options;
30 
31 namespace driver {
32 
34 enum class Dependencies {
35  None,
36  All,
37  Code
38 };
39 
45 struct Options {
46  bool include_linker = false;
47  bool output_hilti = false;
48  bool output_prototypes = false;
49  bool output_cxx = false;
50  std::string output_cxx_prefix;
51  bool output_linker = false;
52  Dependencies output_dependencies = Dependencies::None;
53  bool execute_code = false;
54  bool show_backtraces = false;
55  bool abort_on_exceptions = false;
56  bool keep_tmps = false;
57  bool skip_dependencies = false;
58  bool report_resource_usage = false;
59  bool report_times = false;
60  bool dump_code = false;
61  bool enable_profiling = false;
62  std::vector<hilti::rt::filesystem::path>
64  hilti::rt::filesystem::path output_path;
65  std::unique_ptr<Logger>
68  Options() { logger = std::make_unique<Logger>(); }
69 };
70 
71 } // namespace driver
72 
85 class Driver {
86 public:
91  explicit Driver(std::string name);
92 
100  Driver(std::string name, const hilti::rt::filesystem::path& argv0);
101 
102  virtual ~Driver();
103 
104  Driver() = delete;
105  Driver(const Driver&) = delete;
106  Driver(Driver&&) noexcept = delete;
107  Driver& operator=(const Driver&) = delete;
108  Driver& operator=(Driver&&) noexcept = delete;
109 
120  Result<Nothing> parseOptions(int argc, char** argv);
121 
128  void registerUnit(const std::shared_ptr<Unit>& unit) { _addUnit(unit); }
129 
137  auto i = _units.find(uid);
138  return i != _units.end() ? i->second.get() : nullptr;
139  }
140 
151  void updateProcessExtension(const declaration::module::UID& uid, const hilti::rt::filesystem::path& ext);
152 
165  Result<Nothing> addInput(const hilti::rt::filesystem::path& path);
166 
178 
180  bool hasInputs() const { return ! (_units.empty() && _libraries.empty() && _external_cxxs.empty()); }
181 
183  const auto& driverOptions() const { return _driver_options; }
184 
186  const auto& hiltiOptions() const { return _compiler_options; }
187 
194 
201 
209 
218 
223  const auto& context() const { return _ctx; }
224 
229  auto* builder() const { return _builder.get(); }
230 
232  const auto& options() const { return _ctx->options(); }
233 
244 
252 
260 
270 
271 protected:
272  friend class ASTContext;
273 
278  void usage();
279 
286 
295 
302 
309 
316 
325  result::Error error(std::string_view msg, const hilti::rt::filesystem::path& p = "");
326 
335  result::Error augmentError(const result::Error& err, const hilti::rt::filesystem::path& p = "");
336 
345  Result<std::ofstream> openOutput(const hilti::rt::filesystem::path& p, bool binary = false, bool append = false);
346 
354  Result<Nothing> openInput(std::ifstream& in, const hilti::rt::filesystem::path& p);
355 
363  Result<Nothing> writeOutput(std::ifstream& in, const hilti::rt::filesystem::path& p);
364 
371  Result<std::stringstream> readInput(const hilti::rt::filesystem::path& p);
372 
381  Result<hilti::rt::filesystem::path> writeToTemp(std::ifstream& in, const std::string& name_hint,
382  const std::string& extension = "tmp");
383 
385  void dumpUnit(const Unit& unit);
386 
393 
402  virtual std::unique_ptr<Builder> createBuilder(ASTContext* ctx) const;
403 
408  virtual std::string hookAddCommandLineOptions() { return ""; }
409 
411  virtual bool hookProcessCommandLineOption(int opt, const char* optarg) { return false; }
412 
417  virtual std::string hookAugmentUsage() { return ""; }
418 
423  virtual void hookAddInput(std::shared_ptr<Unit> unit) {} // NOLINT(performance-unnecessary-value-param)
424 
429  virtual void hookAddInput(const hilti::rt::filesystem::path& path) {}
430 
436  virtual void hookNewASTPreCompilation(const Plugin& plugin, ASTRoot* root) {}
437 
453  virtual bool hookNewASTPostCompilation(const Plugin& plugin, ASTRoot* root) { return false; }
454 
460 
465  virtual void hookInitRuntime() {}
466 
471  virtual void hookFinishRuntime() {}
472 
473 private:
474  // Tracking the state of the compilation pipeline to catch out of order
475  // operation.
476  enum Stage { UNINITIALIZED, INITIALIZED, COMPILED, CODEGENED, LINKED, JITTED } _stage = UNINITIALIZED;
477 
478  // Backend for adding a new unit.
479  void _addUnit(const std::shared_ptr<Unit>& unit);
480 
481  // Run a specific plugin's AST passes on all units with the corresponding extension.
482  Result<Nothing> _resolveUnitsWithPlugin(const Plugin& plugin, std::vector<std::shared_ptr<Unit>> units, int& round);
483 
484  // Runs a specific plugin's transform step on a given set of units.
485  Result<Nothing> _transformUnitsWithPlugin(const Plugin& plugin, const std::vector<std::shared_ptr<Unit>>& units);
486 
487  // Run all plugins on current units, iterating until finished.
488  Result<Nothing> _resolveUnits();
489 
490  // Turns all HILTI units into C++.
491  Result<Nothing> _codegenUnits();
492 
493  // Performs global transformations on the generated code.
494  Result<Nothing> _optimizeUnits();
495 
496  // Sends a debug dump of a unit's AST to the global logger.
497  void _dumpAST(const std::shared_ptr<Unit>& unit, const logging::DebugStream& stream, const Plugin& plugin,
498  const std::string& prefix, int round);
499 
500  // Sends a debug dump of a unit's AST to the global logger.
501  void _dumpAST(const std::shared_ptr<Unit>& unit, const logging::DebugStream& stream, const std::string& prefix);
502 
503  // Sends a debug dump of a unit's AST to an output stream.
504  void _dumpAST(const std::shared_ptr<Unit>& unit, std::ostream& stream, const Plugin& plugin,
505  const std::string& prefix, int round);
506 
507  // Records a reduced debug dump of a unit's AST limited to just declarations.
508  void _dumpDeclarations(const std::shared_ptr<Unit>& unit, const Plugin& plugin);
509 
510  // Records a debug dump of a unit's AST to disk.
511  void _saveIterationAST(const std::shared_ptr<Unit>& unit, const Plugin& plugin, const std::string& prefix,
512  int round);
513 
514  // Records a debug dump of a unit's AST to disk.
515  void _saveIterationAST(const std::shared_ptr<Unit>& unit, const Plugin& plugin, const std::string& prefix,
516  const std::string& tag);
517 
518  // Sets the C++ namespaces for generated code from -x/-P prefix options.
519  Result<Nothing> _setCxxNamespacesFromPrefix(const char* prefix);
520 
527  static Result<void*> _symbol(const std::string& symbol);
528 
529  std::string _name;
530  driver::Options _driver_options;
531  hilti::Options _compiler_options;
532 
533  std::shared_ptr<Context> _ctx; // driver's compiler context
534  std::unique_ptr<hilti::Builder> _builder; // driver builder tied to its context
535  std::unique_ptr<hilti::JIT> _jit; // driver's JIT instance
536  std::shared_ptr<const hilti::rt::Library> _library; // Compiled code
537 
538  std::map<declaration::module::UID, std::shared_ptr<Unit>> _units;
539  std::vector<CxxCode> _generated_cxxs;
540  std::unordered_map<std::string, Library> _libraries;
541  std::vector<hilti::rt::filesystem::path> _external_cxxs;
542  std::vector<linker::MetaData> _mds;
543 
544  std::unordered_set<std::string> _processed_paths;
545 
546  bool _runtime_initialized = false; // true once initRuntime() has succeeded
547  std::set<std::string> _tmp_files; // all tmp files created, so that we can clean them up.
548 };
549 
550 } // namespace hilti
Definition: ast-context.h:121
Definition: ast-context.h:494
Definition: driver.h:85
Result< Nothing > compileUnits()
Definition: driver.cc:681
result::Error error(std::string_view msg, const hilti::rt::filesystem::path &p="")
Definition: driver.cc:143
Result< Nothing > linkUnits()
Definition: driver.cc:801
result::Error augmentError(const result::Error &err, const hilti::rt::filesystem::path &p="")
Definition: driver.cc:152
virtual std::string hookAugmentUsage()
Definition: driver.h:417
virtual void hookNewASTPreCompilation(const Plugin &plugin, ASTRoot *root)
Definition: driver.h:436
void printHiltiException(const hilti::rt::Exception &e)
Definition: driver.cc:932
Result< std::stringstream > readInput(const hilti::rt::filesystem::path &p)
Definition: driver.cc:184
const auto & driverOptions() const
Definition: driver.h:183
bool hasInputs() const
Definition: driver.h:180
virtual void hookAddInput(std::shared_ptr< Unit > unit)
Definition: driver.h:423
virtual bool hookProcessCommandLineOption(int opt, const char *optarg)
Definition: driver.h:411
virtual Result< Nothing > hookCompilationFinished(ASTRoot *root)
Definition: driver.h:459
Result< Nothing > finishRuntime()
Definition: driver.cc:1015
Result< Nothing > codegenUnits()
Result< Nothing > outputUnits()
Definition: driver.cc:840
const auto & options() const
Definition: driver.h:232
const auto & hiltiOptions() const
Definition: driver.h:186
Result< Nothing > jitUnits()
Definition: driver.cc:898
Result< Nothing > compile()
Definition: driver.cc:756
Result< Nothing > executeMain()
Definition: driver.cc:986
void dumpUnit(const Unit &unit)
Definition: driver.cc:229
Result< std::ofstream > openOutput(const hilti::rt::filesystem::path &p, bool binary=false, bool append=false)
Definition: driver.cc:156
void usage()
Definition: driver.cc:87
virtual void hookAddInput(const hilti::rt::filesystem::path &path)
Definition: driver.h:429
void updateProcessExtension(const declaration::module::UID &uid, const hilti::rt::filesystem::path &ext)
Definition: driver.cc:507
virtual void hookInitRuntime()
Definition: driver.h:465
const auto & context() const
Definition: driver.h:223
virtual std::unique_ptr< Builder > createBuilder(ASTContext *ctx) const
Definition: driver.cc:951
Result< Nothing > run()
Definition: driver.cc:719
Result< Nothing > addInput(const hilti::rt::filesystem::path &path)
Definition: driver.cc:560
Result< Nothing > openInput(std::ifstream &in, const hilti::rt::filesystem::path &p)
Definition: driver.cc:175
virtual std::string hookAddCommandLineOptions()
Definition: driver.h:408
Unit * lookupUnit(const declaration::module::UID &uid) const
Definition: driver.h:136
auto * builder() const
Definition: driver.h:229
void setDriverOptions(driver::Options options)
Definition: driver.cc:500
Result< hilti::rt::filesystem::path > writeToTemp(std::ifstream &in, const std::string &name_hint, const std::string &extension="tmp")
Definition: driver.cc:209
Result< Nothing > initialize()
Definition: driver.cc:470
void setCompilerOptions(hilti::Options options)
Definition: driver.cc:493
Result< Nothing > writeOutput(std::ifstream &in, const hilti::rt::filesystem::path &p)
Definition: driver.cc:197
Result< Nothing > parseOptions(int argc, char **argv)
Definition: driver.cc:270
virtual bool hookNewASTPostCompilation(const Plugin &plugin, ASTRoot *root)
Definition: driver.h:453
void registerUnit(const std::shared_ptr< Unit > &unit)
Definition: driver.h:128
Result< Nothing > initRuntime()
Definition: driver.cc:953
virtual void hookFinishRuntime()
Definition: driver.h:471
Definition: unit.h:47
Definition: exception.h:22
Definition: result.h:71
Definition: result.h:18
Definition: context.h:33
Definition: plugin.h:52
Definition: module-uid.h:20
Definition: driver.h:45
bool enable_profiling
Definition: driver.h:61
bool report_resource_usage
Definition: driver.h:58
bool report_times
Definition: driver.h:59
bool execute_code
Definition: driver.h:53
bool output_linker
Definition: driver.h:51
bool output_cxx
Definition: driver.h:49
std::vector< hilti::rt::filesystem::path > inputs
Definition: driver.h:63
bool show_backtraces
Definition: driver.h:54
hilti::rt::filesystem::path output_path
Definition: driver.h:64
bool output_hilti
Definition: driver.h:47
bool dump_code
Definition: driver.h:60
std::unique_ptr< Logger > logger
Definition: driver.h:66
bool keep_tmps
Definition: driver.h:56
std::string output_cxx_prefix
Definition: driver.h:50
bool include_linker
Definition: driver.h:46
bool abort_on_exceptions
Definition: driver.h:55
Dependencies output_dependencies
Definition: driver.h:52
bool output_prototypes
Definition: driver.h:48
bool skip_dependencies
Definition: driver.h:57
Definition: result.h:57