The Quantum Exact Simulation Toolkit v4.0.0
Loading...
Searching...
No Matches
types.h
1/** @file
2 * Definitions of the variable-precision API
3 * scalar types (qindex, qreal and qcomp) and
4 * convenience arithmetic operator overloads.
5 * Note both the user code and backend parse
6 * this file and may disagree on the underlying
7 * qcomp type which is no issue; we ensure the
8 * frontend-backend interface is agnostic to it.
9 *
10 * @author Tyson Jones
11 * @author Ali Rezaei (aided in design)
12 *
13 * @defgroup types Types
14 * @ingroup api
15 * @brief Macros for precision-agnostic real and complex arithmetic.
16 * @{
17 */
18
19#ifndef TYPES_H
20#define TYPES_H
21
22#include "quest/include/modes.h"
23#include "quest/include/precision.h"
24
25
26
27/*
28 * REAL TYPE ALIASES
29 */
30
31typedef FLOAT_TYPE qreal;
32typedef INDEX_TYPE qindex;
33
34
35
36/*
37 * COMPLEX TYPE ALIAS
38 */
39
40// when C++ parses this header during backend or C++ user-code compilation...
41#ifdef __cplusplus
42
43 // resolve qcomp as the standard C++ complex type
44 #include <complex>
45 typedef std::complex<FLOAT_TYPE> qcomp;
46
47// when C parses this header, during compilation of C user code...
48#else
49
50 // pretend that the API's qcomp is the C complex type
51 #include <complex.h>
52
53 // which is either MSVC's custom C complex...
54 #ifdef _MSC_VER
55
56 #if (FLOAT_PRECISION == 1)
57 typedef _Fcomplex qcomp;
58
59 #elif (FLOAT_PRECISION == 2)
60 typedef _Dcomplex qcomp;
61
62 #elif (FLOAT_PRECISION == 4)
63 typedef _Lcomplex qcomp;
64
65 #endif
66
67 // or that used by GNU & Clang
68 #else
69 typedef FLOAT_TYPE _Complex qcomp;
70
71 #endif
72
73#endif
74
75
76
77/*
78 * COMPLEX TYPE INSTANTIATION
79 *
80 * which is already elegantly possible in C++ using qcomp(re,im),
81 * but which cannot be extended nicely to C using a macro qcomp(),
82 * because it will break compilation of function signatures which
83 * accept funciton pointers which return qcomp. For example:
84 * myfunc( qcomp (*callback)(int,int) )
85 * So instead, we define a new, ugly 'getQcomp()' helper. Aw! :(
86 * We define it here in the header, inlining directly into user
87 * code, to avoid C & C++ qcomp interoperability issues.
88 */
89
90/// @notdoced
91static inline qcomp getQcomp(qreal re, qreal im) {
92
93 #if defined(__cplusplus)
94 return qcomp(re, im);
95
96 #elif defined(_MSC_VER)
97 return (qcomp) {re, im};
98
99 #else
100 return re + I*im;
101
102 #endif
103}
104
105
106
107/*
108 * COMPLEX TYPE LITERALS
109 */
110
111// MSVC C literals are literally impossible, and
112// C11 literals are already defined in complex header
113
114/// @cond EXCLUDE_FROM_DOXYGEN
115
116#ifdef __cplusplus
117
118 // enables C++14 literals like "3.5i" which are specifically double precision
119 using namespace std::complex_literals;
120
121 // the above literals are irksome for C++ users changing precision; for example, qcomp x = 3.5i
122 // will not compile in single precision, because complex<double> cannot be assigned to a
123 // complex<float>. To spare C++ users from having to change their literals when recompiling,
124 // we define variable-precision custom literals, enabling qcomp = 3.5_i. Note custom literal
125 // args always have max-precision (hence here are "long double" and "long long int")
126
127 static inline qcomp operator ""_i(long double y) {
128 return qcomp(0, static_cast<qreal>(y)); // discarding y precision
129 }
130
131 static inline qcomp operator ""_i(unsigned long long int y) {
132 return qcomp(0, static_cast<qreal>(y)); // discarding y precision
133 }
134
135#endif
136
137/// @endcond // EXCLUDE_FROM_DOXYGEN
138
139
140
141/*
142 * COMPLEX TYPE ARITHMETIC OVERLOADS
143 */
144
145// C11 arithmetic is already defined in complex header, and beautifully
146// permits mixing of parameterised types and precisions
147
148/// @cond EXCLUDE_FROM_DOXYGEN
149
150#ifdef __cplusplus
151
152 // <complex> defines overloads between complex and same-precision floats,
153 // so e.g. complex<double> + double work fine. However, for differing
154 // types and precisions, we must ourselves define operators which cast
155 // the non-complex type to a qreal (to match qcomp). We must do this for
156 // all operators the user might wish to use (e.g. + - * /) upon all
157 // natural literal precisions (typically double, e.g. "3.5"), plus all
158 // assignment forms (x += 2), and when their argument order is reversed.
159 // Furthermore, the user might do arithmetic on complex literals which are
160 // not the same precision as qcomp, so compilation will fail depending
161 // on the setting of PRECISION. To avoid this, we'll define overloads
162 // between all type/precision permutations, always returning qcomp.
163 // Via the unholy macros below, we create 312 overloads; since this will
164 // no doubt break somebody's build/integration, users can disable this
165 // attempt at precision-agnostic arithmetic via DEFINE_ARITHMETIC_OVERLOADS=0
166
167 #ifndef DEFINE_ARITHMETIC_OVERLOADS
168 #define DEFINE_ARITHMETIC_OVERLOADS 1
169 #endif
170
171 #if DEFINE_ARITHMETIC_OVERLOADS
172
173 // shortcuts for below overload definitions
174 #define COMP_TO_QCOMP(a) \
175 qcomp( \
176 static_cast<qreal>(std::real(a)), \
177 static_cast<qreal>(std::imag(a)))
178
179 #define REAL_TO_QCOMP(b) \
180 qcomp(static_cast<qreal>(b), 0)
181
182 // define operator between complex<compprec> and realtype, in either order, casting to qcomp
183 #define DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_REAL(op, compprec, realtype) \
184 static inline qcomp operator op (const std::complex<compprec>& a, const realtype& b) { \
185 return \
186 COMP_TO_QCOMP(a) \
187 op \
188 REAL_TO_QCOMP(b); \
189 } \
190 static inline qcomp operator op (const realtype& b, const std::complex<compprec>& a) { \
191 return \
192 REAL_TO_QCOMP(b) \
193 op \
194 COMP_TO_QCOMP(a); \
195 } \
196 static inline std::complex<compprec>& operator op##= (std::complex<compprec>& a, const realtype& b) { \
197 a = a op b; \
198 return a; \
199 }
200
201 // defines + - * / between complex<compprec> and the given real type
202 #define DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(compprec, realtype) \
203 DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_REAL(+, compprec, realtype) \
204 DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_REAL(-, compprec, realtype) \
205 DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_REAL(*, compprec, realtype) \
206 DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_REAL(/, compprec, realtype)
207
208 // defines + - * / between complex<compprec> and all (relevant) integer types
209 #define DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_INTEGER(compprec) \
210 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(compprec, int) \
211 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(compprec, long int) \
212 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(compprec, long long int) \
213 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(compprec, unsigned) \
214 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(compprec, long unsigned) \
215 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(compprec, long long unsigned)
216
217 // define arithmetic between any-precision-decimal complex and any-precision integer
218 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_INTEGER(float)
219 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_INTEGER(double)
220 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_INTEGER(long double)
221
222 // define arithmetic between any-precision decimal complex and any-differing-precision decimal real
223 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(float, double)
224 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(float, long double)
225 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(double, long double)
226 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(double, float)
227 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(long double, float)
228 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL(long double, double)
229
230 // defines arithmetic between differing-precision decimal complex, casting to qcomp, except assignment
231 #define DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_COMPLEX(op, precA, precB) \
232 static inline qcomp operator op (const std::complex<precA>& a, const std::complex<precB>& b) { \
233 return \
234 COMP_TO_QCOMP(a) \
235 op \
236 COMP_TO_QCOMP(b); \
237 }
238
239 // defines + - * / between the given differing-precision complex, casting to qcomp, except assignment
240 #define DEFINE_SINGLE_DIRECTION_ARITHMETIC_BETWEEN_COMPLEX_AND_COMPLEX(precA, precB) \
241 DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_COMPLEX(+, precA, precB) \
242 DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_COMPLEX(-, precA, precB) \
243 DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_COMPLEX(*, precA, precB) \
244 DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_COMPLEX(/, precA, precB)
245
246 #define DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_COMPLEX(precA, precB) \
247 DEFINE_SINGLE_DIRECTION_ARITHMETIC_BETWEEN_COMPLEX_AND_COMPLEX(precA, precB) \
248 DEFINE_SINGLE_DIRECTION_ARITHMETIC_BETWEEN_COMPLEX_AND_COMPLEX(precB, precA)
249
250 // define arithmetic between all differing-precision decimal complex, casting to qcomp, except assignment
251 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_COMPLEX(float, double)
252 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_COMPLEX(float, long double)
253 DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_COMPLEX(double, long double)
254
255 // keep these unholy macros away from the users
256 #undef COMP_TO_QCOMP
257 #undef REAL_TO_QCOMP
258 #undef DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_REAL
259 #undef DEFINE_OPERATOR_BETWEEN_COMPLEX_AND_COMPLEX
260 #undef DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_REAL
261 #undef DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_INTEGER
262 #undef DEFINE_ARITHMETIC_BETWEEN_COMPLEX_AND_COMPLEX
263 #undef DEFINE_SINGLE_DIRECTION_ARITHMETIC_BETWEEN_COMPLEX_AND_COMPLEX
264
265 #endif // DEFINE_ARITHMETIC_OVERLOADS
266
267#endif
268
269/// @endcond // EXCLUDE_FROM_DOXYGEN
270
271
272
273/*
274 * REPORTERS
275 */
276
277// both C and C++ declare "reportScalar" accepting
278// char array and a qcomp; but each language then
279// separately defines convenience overloads for
280// C++ strings and using C11 generics to overload
281
282#ifdef __cplusplus
283
284 #include <string>
285
286
287 /// @notdoced
288 /// @nottested
289 extern "C" void reportStr(const char* str);
290
291 /// @notdoced
292 /// @nottested
293 void reportStr(std::string str);
294
295 /// @notdoced
296 /// @nottested
297 extern "C" void reportScalar(const char* label, qcomp num);
298
299 /// @notdoced
300 /// @nottested
301 void reportScalar(const char* label, qreal num);
302
303 /// @notdoced
304 /// @nottested
305 void reportScalar(std::string label, qcomp num);
306
307 /// @notdoced
308 /// @nottested
309 void reportScalar(std::string label, qreal num);
310
311#else
312
313 /// @notdoced
314 /// @nottested
315 void reportStr(const char* str);
316
317 /// @notdoced
318 /// @nottested
319 void reportScalar (const char* label, qcomp num);
320
321 /// @private
322 void _reportScalar_real(const char* label, qreal num);
323
324 /// @notdoced
325 /// @nottested
326 #define reportScalar(label, num) \
327 _Generic((num), \
328 qcomp : reportScalar, \
329 qreal : _reportScalar_real, \
330 default : _reportScalar_real \
331 )(label, num)
332
333#endif
334
335
336
337#endif // TYPES_H
338
339/** @} (end doxygen defgroup) */
void reportStr(const char *str)
Definition types.cpp:29
static qcomp getQcomp(qreal re, qreal im)
Definition types.h:91
void reportScalar(const char *label, qcomp num)
Definition types.cpp:51