Scine::Sparrow  5.0.0
Library for fast and agile quantum chemical calculations with semiempirical methods.
 All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Pages
Common.h
Go to the documentation of this file.
1 
7 #ifndef INCLUDE_SPARROW_EMBED_COMMON_H
8 #define INCLUDE_SPARROW_EMBED_COMMON_H
9 
10 #include "boost/optional.hpp"
11 #include <iso646.h>
12 #include <algorithm>
13 #include <fstream>
14 #include <iostream>
15 #include <vector>
16 
17 inline std::ostream& sep(std::ostream& os) {
18  os << ", ";
19  return os;
20 }
21 
22 inline std::ostream& nl(std::ostream& os) {
23  os << "\n";
24  return os;
25 }
26 
27 inline std::string sp(unsigned i) {
28  return std::string(i, ' ');
29 }
30 
32  std::vector<std::string> includes;
33  std::vector<std::string> namespaces;
34  boost::optional<std::string> guard = boost::none;
35 
36  void writeHeader(std::ostream& os) {
37  /* We need to obfuscate the doxygen copyright statement from our license
38  * checker script, which allows only a single such statement in each file.
39  */
40  os << R"delim(/** * @file * @copy)delim"
41  << R"delim(right This code is licensed under the 3-clause BSD license.\n * Copyright ETH Zurich, Department of Chemistry and Applied Biosciences, Reiher Group.\n * See LICENSE.txt for details. * * @note This file was generated by the Embed binary from runtime values. * Prefer improving the generator over editing this file whenever possible. * )delim";
42 
43  // Write some more details for cpp files
44  if (!guard) {
45  os << R"delim( * This file contains functions generating runtime values. It was * generated from runtime values of its return type. It is not intended to be * human-readable. A small guide: Return values are directly brace-initialized * in deep nesting to keep file size to a minimum. Types are annotated only when * necessary. Floating point values are represented in hexadecimal (see * std::hexfloat) to ensure serialization does not cause loss of accuracy. * * The functions defined here might be declared and called elsewhere entirely. * )delim";
46  }
47 
48  os << R"delim( */ )delim";
49 
50  if (guard) {
51  os << "#ifndef " << guard.value() << nl;
52  os << "#define " << guard.value() << nl << nl;
53  }
54 
55  for (auto& include : includes) {
56  if (include[0] == '<' || include[0] == '\"') {
57  os << "#include " << include << nl;
58  }
59  else {
60  os << "#include \"" << include << "\"" << nl;
61  }
62  }
63  os << nl;
64 
65  for (auto& name : namespaces) {
66  os << "namespace " << name << " {" << nl;
67  }
68 
69  os << nl;
70  }
71 
72  void writeFooter(std::ostream& os) {
73  os << nl;
74  for (auto iter = std::rbegin(namespaces); iter != std::rend(namespaces); ++iter) {
75  os << "} // namespace " << *iter << nl;
76  }
77 
78  if (guard) {
79  os << nl << "#endif";
80  }
81  }
82 };
83 
84 inline std::string lower(std::string a) {
85  std::transform(std::begin(a), std::end(a), std::begin(a), [](unsigned char c) { return std::tolower(c); });
86  return a;
87 }
88 
89 inline std::string upper(std::string a) {
90  std::transform(std::begin(a), std::end(a), std::begin(a), [](unsigned char c) { return std::toupper(c); });
91  return a;
92 }
93 
94 namespace detail {
95 
96 template<typename Arg>
97 void writePack(std::ostream& os, Arg arg) {
98  os << arg;
99 }
100 
101 template<typename Arg, typename... Args>
102 void writePack(std::ostream& os, Arg arg1, Args... args) {
103  os << arg1 << sep;
104  writePack(os, args...);
105 }
106 
107 template<typename Tuple, std::size_t... Inds>
108 void writeTuplePackHelper(std::ostream& os, const Tuple& tup, std::index_sequence<Inds...> /* inds */
109 ) {
110  writePack(os, std::get<Inds>(tup)...);
111 }
112 
113 template<typename Tuple>
114 void writeTuplePack(std::ostream& os, const Tuple& tup) {
115  writeTuplePackHelper(os, tup, std::make_index_sequence<std::tuple_size<Tuple>::value>{});
116 }
117 
118 } // namespace detail
119 
120 template<typename... Args>
121 struct Joiner {
122  Joiner(const Args&... args) : bound(args...) {
123  }
124 
125  std::tuple<const Args&...> bound;
126 };
127 
128 template<typename... Args>
129 auto join(const Args&... args) {
130  return Joiner<Args...>(args...);
131 }
132 
133 template<typename... Args>
134 std::ostream& operator<<(std::ostream& os, const Joiner<Args...>& join) {
135  detail::writeTuplePack(os, join.bound);
136  return os;
137 }
138 
139 /* Note: We need to wrap optionals because boost provides its own output
140  * operators that don't serve our purpose. We can't use the type directly
141  * because there's already a declaration available for it by inclusion of boost
142  * optional.
143  */
144 template<typename T>
145 struct OptionalWrapper {
146  boost::optional<T> optional;
147 };
148 
149 template<typename T>
150 auto wrapOptional(const boost::optional<T>& optional) {
151  return OptionalWrapper<T>{optional};
152 }
153 
154 template<typename T>
155 std::ostream& operator<<(std::ostream& os, const OptionalWrapper<T>& o) {
156  if (o.optional) {
157  os << o.optional.value();
158  }
159  else {
160  os << "boost::none";
161  }
162  return os;
163 }
165 #endif
166 
Definition: Common.h:164
Definition: Common.h:31
Definition: Common.h:140