blob: 1c7b5ca2b79d1ee82dda5743dca42a9685602ddc [file] [log] [blame]
Nicolas Capens0bac2852016-05-07 06:09:58 -04001// Copyright 2016 The SwiftShader Authors. All Rights Reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
Nicolas Capens48461502018-08-06 14:20:45 -040015#ifndef rr_Reactor_hpp
16#define rr_Reactor_hpp
Nicolas Capensd022e412016-09-26 13:30:14 -040017
Nicolas Capens0bac2852016-05-07 06:09:58 -040018#include "Nucleus.hpp"
19#include "Routine.hpp"
20
Nicolas Capens4dd1eff2017-08-04 09:33:04 -040021#include <cassert>
Nicolas Capens3bbc5e12016-09-27 10:49:52 -040022#include <cstddef>
Chris Forbes878d4b02019-01-21 10:48:35 -080023#include <cstdio>
Ben Clayton1bc7ee92019-02-14 18:43:22 +000024
25#include <string>
Ben Clayton169872e2019-02-27 23:58:35 +000026#include <tuple>
Nicolas Capens0192d152019-03-27 14:46:07 -040027#include <unordered_set>
Ben Clayton1bc7ee92019-02-14 18:43:22 +000028
Chris Forbesc2968062019-03-19 16:48:03 -070029#undef Bool // b/127920555
Nicolas Capens3bbc5e12016-09-27 10:49:52 -040030
Ben Clayton5875be52019-04-11 14:57:40 -040031#if !defined(NDEBUG)
Ben Clayton1bc7ee92019-02-14 18:43:22 +000032#define ENABLE_RR_PRINT 1 // Enables RR_PRINT(), RR_WATCH()
Ben Clayton5875be52019-04-11 14:57:40 -040033#endif // !defined(NDEBUG)
Ben Clayton1bc7ee92019-02-14 18:43:22 +000034
Ben Claytonac07ed82019-03-26 14:17:41 +000035#ifdef ENABLE_RR_DEBUG_INFO
36 // Functions used for generating JIT debug info.
37 // See docs/ReactorDebugInfo.md for more information.
38 namespace rr
39 {
40 // Update the current source location for debug.
41 void EmitDebugLocation();
42 // Bind value to its symbolic name taken from the backtrace.
43 void EmitDebugVariable(class Value* value);
44 // Flush any pending variable bindings before the line ends.
45 void FlushDebug();
46 }
47 #define RR_DEBUG_INFO_UPDATE_LOC() rr::EmitDebugLocation()
48 #define RR_DEBUG_INFO_EMIT_VAR(value) rr::EmitDebugVariable(value)
49 #define RR_DEBUG_INFO_FLUSH() rr::FlushDebug()
50#else
51 #define RR_DEBUG_INFO_UPDATE_LOC()
52 #define RR_DEBUG_INFO_EMIT_VAR(value)
53 #define RR_DEBUG_INFO_FLUSH()
54#endif // ENABLE_RR_DEBUG_INFO
55
Nicolas Capens48461502018-08-06 14:20:45 -040056namespace rr
Nicolas Capens0bac2852016-05-07 06:09:58 -040057{
Ben Claytonc7904162019-04-17 17:35:48 -040058 struct Capabilities
59 {
60 bool CallSupported; // Support for rr::Call()
61 };
62 extern const Capabilities Caps;
63
Nicolas Capenseb253d02016-11-18 14:40:40 -050064 class Bool;
Nicolas Capensd022e412016-09-26 13:30:14 -040065 class Byte;
66 class SByte;
67 class Byte4;
68 class SByte4;
69 class Byte8;
70 class SByte8;
71 class Byte16;
72 class SByte16;
73 class Short;
74 class UShort;
Nicolas Capens16b5f152016-10-13 13:39:01 -040075 class Short2;
76 class UShort2;
Nicolas Capensd022e412016-09-26 13:30:14 -040077 class Short4;
78 class UShort4;
79 class Short8;
80 class UShort8;
81 class Int;
82 class UInt;
83 class Int2;
84 class UInt2;
85 class Int4;
86 class UInt4;
87 class Long;
Alexis Hetu734e2572018-12-20 14:00:49 -050088 class Half;
Nicolas Capensd022e412016-09-26 13:30:14 -040089 class Float;
90 class Float2;
91 class Float4;
92
93 class Void
94 {
95 public:
96 static Type *getType();
97
98 static bool isVoid()
99 {
100 return true;
101 }
Nicolas Capensd022e412016-09-26 13:30:14 -0400102 };
103
104 template<class T>
105 class RValue;
106
107 template<class T>
108 class Pointer;
109
Nicolas Capens297d26e2016-11-18 12:52:17 -0500110 class Variable
111 {
Nicolas Capens0192d152019-03-27 14:46:07 -0400112 friend class Nucleus;
Ben Clayton1bc7ee92019-02-14 18:43:22 +0000113 friend class PrintValue;
Nicolas Capens5da8d8d2019-03-27 14:45:34 -0400114
Nicolas Capens0192d152019-03-27 14:46:07 -0400115 Variable() = delete;
Nicolas Capens5da8d8d2019-03-27 14:45:34 -0400116 Variable &operator=(const Variable&) = delete;
117
118 public:
Nicolas Capens0192d152019-03-27 14:46:07 -0400119 void materialize() const;
120
121 Value *loadValue() const;
122 Value *storeValue(Value *value) const;
123
124 Value *getBaseAddress() const;
125 Value *getElementPointer(Value *index, bool unsignedIndex) const;
Nicolas Capens5da8d8d2019-03-27 14:45:34 -0400126
127 protected:
Nicolas Capens0192d152019-03-27 14:46:07 -0400128 Variable(Type *type, int arraySize);
129 Variable(const Variable&) = default;
130
131 ~Variable();
132
133 private:
134 static void materializeAll();
135 static void killUnmaterialized();
136
137 static std::unordered_set<Variable*> unmaterializedVariables;
138
139 Type *const type;
140 const int arraySize;
141 mutable Value *rvalue = nullptr;
142 mutable Value *address = nullptr;
Nicolas Capens297d26e2016-11-18 12:52:17 -0500143 };
144
Nicolas Capens22479eb2016-09-28 22:34:26 -0400145 template<class T>
Nicolas Capens297d26e2016-11-18 12:52:17 -0500146 class LValue : public Variable
Nicolas Capensd022e412016-09-26 13:30:14 -0400147 {
148 public:
Nicolas Capens22479eb2016-09-28 22:34:26 -0400149 LValue(int arraySize = 0);
Nicolas Capensd022e412016-09-26 13:30:14 -0400150
Nicolas Capens297d26e2016-11-18 12:52:17 -0500151 RValue<Pointer<T>> operator&();
152
Nicolas Capensd022e412016-09-26 13:30:14 -0400153 static bool isVoid()
154 {
155 return false;
156 }
Nicolas Capensd022e412016-09-26 13:30:14 -0400157 };
158
159 template<class T>
160 class Reference
161 {
162 public:
163 explicit Reference(Value *pointer, int alignment = 1);
164
165 RValue<T> operator=(RValue<T> rhs) const;
166 RValue<T> operator=(const Reference<T> &ref) const;
167
168 RValue<T> operator+=(RValue<T> rhs) const;
169
Ben Claytonec255732019-02-27 13:51:22 +0000170 RValue<Pointer<T>> operator&() const { return RValue<Pointer<T>>(address); }
171
Nicolas Capensd022e412016-09-26 13:30:14 -0400172 Value *loadValue() const;
173 int getAlignment() const;
174
175 private:
176 Value *address;
177
178 const int alignment;
179 };
180
181 template<class T>
Ben Clayton35e90e22019-03-15 10:06:06 +0000182 struct BoolLiteral
Nicolas Capensd022e412016-09-26 13:30:14 -0400183 {
184 struct type;
185 };
186
Nicolas Capenseb253d02016-11-18 14:40:40 -0500187 template<>
Ben Clayton35e90e22019-03-15 10:06:06 +0000188 struct BoolLiteral<Bool>
Nicolas Capenseb253d02016-11-18 14:40:40 -0500189 {
190 typedef bool type;
191 };
192
Ben Clayton35e90e22019-03-15 10:06:06 +0000193 template<class T>
194 struct IntLiteral
195 {
196 struct type;
197 };
198
Nicolas Capenseb253d02016-11-18 14:40:40 -0500199 template<>
200 struct IntLiteral<Int>
Nicolas Capensd022e412016-09-26 13:30:14 -0400201 {
202 typedef int type;
203 };
204
Nicolas Capenseb253d02016-11-18 14:40:40 -0500205 template<>
206 struct IntLiteral<UInt>
Nicolas Capensd022e412016-09-26 13:30:14 -0400207 {
208 typedef unsigned int type;
209 };
210
Nicolas Capenseb253d02016-11-18 14:40:40 -0500211 template<>
212 struct IntLiteral<Long>
Nicolas Capensd022e412016-09-26 13:30:14 -0400213 {
214 typedef int64_t type;
215 };
216
217 template<class T>
218 struct FloatLiteral
219 {
220 struct type;
221 };
222
Nicolas Capenseb253d02016-11-18 14:40:40 -0500223 template<>
224 struct FloatLiteral<Float>
Nicolas Capensd022e412016-09-26 13:30:14 -0400225 {
226 typedef float type;
227 };
228
229 template<class T>
230 class RValue
231 {
232 public:
233 explicit RValue(Value *rvalue);
234
Ben Claytonac07ed82019-03-26 14:17:41 +0000235#ifdef ENABLE_RR_DEBUG_INFO
236 RValue(const RValue<T> &rvalue);
237#endif // ENABLE_RR_DEBUG_INFO
238
Nicolas Capensd022e412016-09-26 13:30:14 -0400239 RValue(const T &lvalue);
Ben Clayton35e90e22019-03-15 10:06:06 +0000240 RValue(typename BoolLiteral<T>::type i);
Nicolas Capensd022e412016-09-26 13:30:14 -0400241 RValue(typename IntLiteral<T>::type i);
242 RValue(typename FloatLiteral<T>::type f);
243 RValue(const Reference<T> &rhs);
244
245 RValue<T> &operator=(const RValue<T>&) = delete;
246
247 Value *value; // FIXME: Make private
248 };
249
250 template<typename T>
251 struct Argument
252 {
253 explicit Argument(Value *value) : value(value) {}
254
255 Value *value;
256 };
257
Nicolas Capens297d26e2016-11-18 12:52:17 -0500258 class Bool : public LValue<Bool>
Nicolas Capensd022e412016-09-26 13:30:14 -0400259 {
260 public:
261 Bool(Argument<Bool> argument);
262
Nicolas Capensa25311a2017-01-16 17:19:00 -0500263 Bool() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400264 Bool(bool x);
265 Bool(RValue<Bool> rhs);
266 Bool(const Bool &rhs);
267 Bool(const Reference<Bool> &rhs);
268
Nicolas Capens96d4e092016-11-18 14:22:38 -0500269 // RValue<Bool> operator=(bool rhs); // FIXME: Implement
270 RValue<Bool> operator=(RValue<Bool> rhs);
271 RValue<Bool> operator=(const Bool &rhs);
272 RValue<Bool> operator=(const Reference<Bool> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400273
274 static Type *getType();
275 };
276
277 RValue<Bool> operator!(RValue<Bool> val);
278 RValue<Bool> operator&&(RValue<Bool> lhs, RValue<Bool> rhs);
279 RValue<Bool> operator||(RValue<Bool> lhs, RValue<Bool> rhs);
Ben Claytonf3b57972019-03-15 09:56:47 +0000280 RValue<Bool> operator!=(RValue<Bool> lhs, RValue<Bool> rhs);
281 RValue<Bool> operator==(RValue<Bool> lhs, RValue<Bool> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400282
Nicolas Capens297d26e2016-11-18 12:52:17 -0500283 class Byte : public LValue<Byte>
Nicolas Capensd022e412016-09-26 13:30:14 -0400284 {
285 public:
286 Byte(Argument<Byte> argument);
287
288 explicit Byte(RValue<Int> cast);
289 explicit Byte(RValue<UInt> cast);
290 explicit Byte(RValue<UShort> cast);
291
Nicolas Capensa25311a2017-01-16 17:19:00 -0500292 Byte() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400293 Byte(int x);
294 Byte(unsigned char x);
295 Byte(RValue<Byte> rhs);
296 Byte(const Byte &rhs);
297 Byte(const Reference<Byte> &rhs);
298
Nicolas Capens96d4e092016-11-18 14:22:38 -0500299 // RValue<Byte> operator=(unsigned char rhs); // FIXME: Implement
300 RValue<Byte> operator=(RValue<Byte> rhs);
301 RValue<Byte> operator=(const Byte &rhs);
302 RValue<Byte> operator=(const Reference<Byte> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400303
304 static Type *getType();
305 };
306
307 RValue<Byte> operator+(RValue<Byte> lhs, RValue<Byte> rhs);
308 RValue<Byte> operator-(RValue<Byte> lhs, RValue<Byte> rhs);
309 RValue<Byte> operator*(RValue<Byte> lhs, RValue<Byte> rhs);
310 RValue<Byte> operator/(RValue<Byte> lhs, RValue<Byte> rhs);
311 RValue<Byte> operator%(RValue<Byte> lhs, RValue<Byte> rhs);
312 RValue<Byte> operator&(RValue<Byte> lhs, RValue<Byte> rhs);
313 RValue<Byte> operator|(RValue<Byte> lhs, RValue<Byte> rhs);
314 RValue<Byte> operator^(RValue<Byte> lhs, RValue<Byte> rhs);
315 RValue<Byte> operator<<(RValue<Byte> lhs, RValue<Byte> rhs);
316 RValue<Byte> operator>>(RValue<Byte> lhs, RValue<Byte> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500317 RValue<Byte> operator+=(Byte &lhs, RValue<Byte> rhs);
318 RValue<Byte> operator-=(Byte &lhs, RValue<Byte> rhs);
319 RValue<Byte> operator*=(Byte &lhs, RValue<Byte> rhs);
320 RValue<Byte> operator/=(Byte &lhs, RValue<Byte> rhs);
321 RValue<Byte> operator%=(Byte &lhs, RValue<Byte> rhs);
322 RValue<Byte> operator&=(Byte &lhs, RValue<Byte> rhs);
323 RValue<Byte> operator|=(Byte &lhs, RValue<Byte> rhs);
324 RValue<Byte> operator^=(Byte &lhs, RValue<Byte> rhs);
325 RValue<Byte> operator<<=(Byte &lhs, RValue<Byte> rhs);
326 RValue<Byte> operator>>=(Byte &lhs, RValue<Byte> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400327 RValue<Byte> operator+(RValue<Byte> val);
328 RValue<Byte> operator-(RValue<Byte> val);
329 RValue<Byte> operator~(RValue<Byte> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500330 RValue<Byte> operator++(Byte &val, int); // Post-increment
331 const Byte &operator++(Byte &val); // Pre-increment
332 RValue<Byte> operator--(Byte &val, int); // Post-decrement
333 const Byte &operator--(Byte &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400334 RValue<Bool> operator<(RValue<Byte> lhs, RValue<Byte> rhs);
335 RValue<Bool> operator<=(RValue<Byte> lhs, RValue<Byte> rhs);
336 RValue<Bool> operator>(RValue<Byte> lhs, RValue<Byte> rhs);
337 RValue<Bool> operator>=(RValue<Byte> lhs, RValue<Byte> rhs);
338 RValue<Bool> operator!=(RValue<Byte> lhs, RValue<Byte> rhs);
339 RValue<Bool> operator==(RValue<Byte> lhs, RValue<Byte> rhs);
340
Nicolas Capens297d26e2016-11-18 12:52:17 -0500341 class SByte : public LValue<SByte>
Nicolas Capensd022e412016-09-26 13:30:14 -0400342 {
343 public:
344 SByte(Argument<SByte> argument);
345
346 explicit SByte(RValue<Int> cast);
347 explicit SByte(RValue<Short> cast);
348
Nicolas Capensa25311a2017-01-16 17:19:00 -0500349 SByte() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400350 SByte(signed char x);
351 SByte(RValue<SByte> rhs);
352 SByte(const SByte &rhs);
353 SByte(const Reference<SByte> &rhs);
354
Nicolas Capens96d4e092016-11-18 14:22:38 -0500355 // RValue<SByte> operator=(signed char rhs); // FIXME: Implement
356 RValue<SByte> operator=(RValue<SByte> rhs);
357 RValue<SByte> operator=(const SByte &rhs);
358 RValue<SByte> operator=(const Reference<SByte> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400359
360 static Type *getType();
361 };
362
363 RValue<SByte> operator+(RValue<SByte> lhs, RValue<SByte> rhs);
364 RValue<SByte> operator-(RValue<SByte> lhs, RValue<SByte> rhs);
365 RValue<SByte> operator*(RValue<SByte> lhs, RValue<SByte> rhs);
366 RValue<SByte> operator/(RValue<SByte> lhs, RValue<SByte> rhs);
367 RValue<SByte> operator%(RValue<SByte> lhs, RValue<SByte> rhs);
368 RValue<SByte> operator&(RValue<SByte> lhs, RValue<SByte> rhs);
369 RValue<SByte> operator|(RValue<SByte> lhs, RValue<SByte> rhs);
370 RValue<SByte> operator^(RValue<SByte> lhs, RValue<SByte> rhs);
371 RValue<SByte> operator<<(RValue<SByte> lhs, RValue<SByte> rhs);
372 RValue<SByte> operator>>(RValue<SByte> lhs, RValue<SByte> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500373 RValue<SByte> operator+=(SByte &lhs, RValue<SByte> rhs);
374 RValue<SByte> operator-=(SByte &lhs, RValue<SByte> rhs);
375 RValue<SByte> operator*=(SByte &lhs, RValue<SByte> rhs);
376 RValue<SByte> operator/=(SByte &lhs, RValue<SByte> rhs);
377 RValue<SByte> operator%=(SByte &lhs, RValue<SByte> rhs);
378 RValue<SByte> operator&=(SByte &lhs, RValue<SByte> rhs);
379 RValue<SByte> operator|=(SByte &lhs, RValue<SByte> rhs);
380 RValue<SByte> operator^=(SByte &lhs, RValue<SByte> rhs);
381 RValue<SByte> operator<<=(SByte &lhs, RValue<SByte> rhs);
382 RValue<SByte> operator>>=(SByte &lhs, RValue<SByte> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400383 RValue<SByte> operator+(RValue<SByte> val);
384 RValue<SByte> operator-(RValue<SByte> val);
385 RValue<SByte> operator~(RValue<SByte> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500386 RValue<SByte> operator++(SByte &val, int); // Post-increment
387 const SByte &operator++(SByte &val); // Pre-increment
388 RValue<SByte> operator--(SByte &val, int); // Post-decrement
389 const SByte &operator--(SByte &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400390 RValue<Bool> operator<(RValue<SByte> lhs, RValue<SByte> rhs);
391 RValue<Bool> operator<=(RValue<SByte> lhs, RValue<SByte> rhs);
392 RValue<Bool> operator>(RValue<SByte> lhs, RValue<SByte> rhs);
393 RValue<Bool> operator>=(RValue<SByte> lhs, RValue<SByte> rhs);
394 RValue<Bool> operator!=(RValue<SByte> lhs, RValue<SByte> rhs);
395 RValue<Bool> operator==(RValue<SByte> lhs, RValue<SByte> rhs);
396
Nicolas Capens297d26e2016-11-18 12:52:17 -0500397 class Short : public LValue<Short>
Nicolas Capensd022e412016-09-26 13:30:14 -0400398 {
399 public:
400 Short(Argument<Short> argument);
401
402 explicit Short(RValue<Int> cast);
403
Nicolas Capensa25311a2017-01-16 17:19:00 -0500404 Short() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400405 Short(short x);
406 Short(RValue<Short> rhs);
407 Short(const Short &rhs);
408 Short(const Reference<Short> &rhs);
409
Nicolas Capens96d4e092016-11-18 14:22:38 -0500410 // RValue<Short> operator=(short rhs); // FIXME: Implement
411 RValue<Short> operator=(RValue<Short> rhs);
412 RValue<Short> operator=(const Short &rhs);
413 RValue<Short> operator=(const Reference<Short> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400414
415 static Type *getType();
416 };
417
418 RValue<Short> operator+(RValue<Short> lhs, RValue<Short> rhs);
419 RValue<Short> operator-(RValue<Short> lhs, RValue<Short> rhs);
420 RValue<Short> operator*(RValue<Short> lhs, RValue<Short> rhs);
421 RValue<Short> operator/(RValue<Short> lhs, RValue<Short> rhs);
422 RValue<Short> operator%(RValue<Short> lhs, RValue<Short> rhs);
423 RValue<Short> operator&(RValue<Short> lhs, RValue<Short> rhs);
424 RValue<Short> operator|(RValue<Short> lhs, RValue<Short> rhs);
425 RValue<Short> operator^(RValue<Short> lhs, RValue<Short> rhs);
426 RValue<Short> operator<<(RValue<Short> lhs, RValue<Short> rhs);
427 RValue<Short> operator>>(RValue<Short> lhs, RValue<Short> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500428 RValue<Short> operator+=(Short &lhs, RValue<Short> rhs);
429 RValue<Short> operator-=(Short &lhs, RValue<Short> rhs);
430 RValue<Short> operator*=(Short &lhs, RValue<Short> rhs);
431 RValue<Short> operator/=(Short &lhs, RValue<Short> rhs);
432 RValue<Short> operator%=(Short &lhs, RValue<Short> rhs);
433 RValue<Short> operator&=(Short &lhs, RValue<Short> rhs);
434 RValue<Short> operator|=(Short &lhs, RValue<Short> rhs);
435 RValue<Short> operator^=(Short &lhs, RValue<Short> rhs);
436 RValue<Short> operator<<=(Short &lhs, RValue<Short> rhs);
437 RValue<Short> operator>>=(Short &lhs, RValue<Short> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400438 RValue<Short> operator+(RValue<Short> val);
439 RValue<Short> operator-(RValue<Short> val);
440 RValue<Short> operator~(RValue<Short> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500441 RValue<Short> operator++(Short &val, int); // Post-increment
442 const Short &operator++(Short &val); // Pre-increment
443 RValue<Short> operator--(Short &val, int); // Post-decrement
444 const Short &operator--(Short &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400445 RValue<Bool> operator<(RValue<Short> lhs, RValue<Short> rhs);
446 RValue<Bool> operator<=(RValue<Short> lhs, RValue<Short> rhs);
447 RValue<Bool> operator>(RValue<Short> lhs, RValue<Short> rhs);
448 RValue<Bool> operator>=(RValue<Short> lhs, RValue<Short> rhs);
449 RValue<Bool> operator!=(RValue<Short> lhs, RValue<Short> rhs);
450 RValue<Bool> operator==(RValue<Short> lhs, RValue<Short> rhs);
451
Nicolas Capens297d26e2016-11-18 12:52:17 -0500452 class UShort : public LValue<UShort>
Nicolas Capensd022e412016-09-26 13:30:14 -0400453 {
454 public:
455 UShort(Argument<UShort> argument);
456
457 explicit UShort(RValue<UInt> cast);
458 explicit UShort(RValue<Int> cast);
459
Nicolas Capensa25311a2017-01-16 17:19:00 -0500460 UShort() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400461 UShort(unsigned short x);
462 UShort(RValue<UShort> rhs);
463 UShort(const UShort &rhs);
464 UShort(const Reference<UShort> &rhs);
465
Nicolas Capens96d4e092016-11-18 14:22:38 -0500466 // RValue<UShort> operator=(unsigned short rhs); // FIXME: Implement
467 RValue<UShort> operator=(RValue<UShort> rhs);
468 RValue<UShort> operator=(const UShort &rhs);
469 RValue<UShort> operator=(const Reference<UShort> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400470
471 static Type *getType();
472 };
473
474 RValue<UShort> operator+(RValue<UShort> lhs, RValue<UShort> rhs);
475 RValue<UShort> operator-(RValue<UShort> lhs, RValue<UShort> rhs);
476 RValue<UShort> operator*(RValue<UShort> lhs, RValue<UShort> rhs);
477 RValue<UShort> operator/(RValue<UShort> lhs, RValue<UShort> rhs);
478 RValue<UShort> operator%(RValue<UShort> lhs, RValue<UShort> rhs);
479 RValue<UShort> operator&(RValue<UShort> lhs, RValue<UShort> rhs);
480 RValue<UShort> operator|(RValue<UShort> lhs, RValue<UShort> rhs);
481 RValue<UShort> operator^(RValue<UShort> lhs, RValue<UShort> rhs);
482 RValue<UShort> operator<<(RValue<UShort> lhs, RValue<UShort> rhs);
483 RValue<UShort> operator>>(RValue<UShort> lhs, RValue<UShort> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500484 RValue<UShort> operator+=(UShort &lhs, RValue<UShort> rhs);
485 RValue<UShort> operator-=(UShort &lhs, RValue<UShort> rhs);
486 RValue<UShort> operator*=(UShort &lhs, RValue<UShort> rhs);
487 RValue<UShort> operator/=(UShort &lhs, RValue<UShort> rhs);
488 RValue<UShort> operator%=(UShort &lhs, RValue<UShort> rhs);
489 RValue<UShort> operator&=(UShort &lhs, RValue<UShort> rhs);
490 RValue<UShort> operator|=(UShort &lhs, RValue<UShort> rhs);
491 RValue<UShort> operator^=(UShort &lhs, RValue<UShort> rhs);
492 RValue<UShort> operator<<=(UShort &lhs, RValue<UShort> rhs);
493 RValue<UShort> operator>>=(UShort &lhs, RValue<UShort> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400494 RValue<UShort> operator+(RValue<UShort> val);
495 RValue<UShort> operator-(RValue<UShort> val);
496 RValue<UShort> operator~(RValue<UShort> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500497 RValue<UShort> operator++(UShort &val, int); // Post-increment
498 const UShort &operator++(UShort &val); // Pre-increment
499 RValue<UShort> operator--(UShort &val, int); // Post-decrement
500 const UShort &operator--(UShort &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400501 RValue<Bool> operator<(RValue<UShort> lhs, RValue<UShort> rhs);
502 RValue<Bool> operator<=(RValue<UShort> lhs, RValue<UShort> rhs);
503 RValue<Bool> operator>(RValue<UShort> lhs, RValue<UShort> rhs);
504 RValue<Bool> operator>=(RValue<UShort> lhs, RValue<UShort> rhs);
505 RValue<Bool> operator!=(RValue<UShort> lhs, RValue<UShort> rhs);
506 RValue<Bool> operator==(RValue<UShort> lhs, RValue<UShort> rhs);
507
Nicolas Capens297d26e2016-11-18 12:52:17 -0500508 class Byte4 : public LValue<Byte4>
Nicolas Capensd022e412016-09-26 13:30:14 -0400509 {
510 public:
Nicolas Capens16b5f152016-10-13 13:39:01 -0400511 explicit Byte4(RValue<Byte8> cast);
512
Nicolas Capensa25311a2017-01-16 17:19:00 -0500513 Byte4() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400514 // Byte4(int x, int y, int z, int w);
515 // Byte4(RValue<Byte4> rhs);
516 // Byte4(const Byte4 &rhs);
Nicolas Capens16b5f152016-10-13 13:39:01 -0400517 Byte4(const Reference<Byte4> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400518
Nicolas Capens96d4e092016-11-18 14:22:38 -0500519 // RValue<Byte4> operator=(RValue<Byte4> rhs);
520 // RValue<Byte4> operator=(const Byte4 &rhs);
521 // RValue<Byte4> operator=(const Reference<Byte4> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400522
523 static Type *getType();
524 };
525
526// RValue<Byte4> operator+(RValue<Byte4> lhs, RValue<Byte4> rhs);
527// RValue<Byte4> operator-(RValue<Byte4> lhs, RValue<Byte4> rhs);
528// RValue<Byte4> operator*(RValue<Byte4> lhs, RValue<Byte4> rhs);
529// RValue<Byte4> operator/(RValue<Byte4> lhs, RValue<Byte4> rhs);
530// RValue<Byte4> operator%(RValue<Byte4> lhs, RValue<Byte4> rhs);
531// RValue<Byte4> operator&(RValue<Byte4> lhs, RValue<Byte4> rhs);
532// RValue<Byte4> operator|(RValue<Byte4> lhs, RValue<Byte4> rhs);
533// RValue<Byte4> operator^(RValue<Byte4> lhs, RValue<Byte4> rhs);
534// RValue<Byte4> operator<<(RValue<Byte4> lhs, RValue<Byte4> rhs);
535// RValue<Byte4> operator>>(RValue<Byte4> lhs, RValue<Byte4> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500536// RValue<Byte4> operator+=(Byte4 &lhs, RValue<Byte4> rhs);
537// RValue<Byte4> operator-=(Byte4 &lhs, RValue<Byte4> rhs);
538// RValue<Byte4> operator*=(Byte4 &lhs, RValue<Byte4> rhs);
539// RValue<Byte4> operator/=(Byte4 &lhs, RValue<Byte4> rhs);
540// RValue<Byte4> operator%=(Byte4 &lhs, RValue<Byte4> rhs);
541// RValue<Byte4> operator&=(Byte4 &lhs, RValue<Byte4> rhs);
542// RValue<Byte4> operator|=(Byte4 &lhs, RValue<Byte4> rhs);
543// RValue<Byte4> operator^=(Byte4 &lhs, RValue<Byte4> rhs);
544// RValue<Byte4> operator<<=(Byte4 &lhs, RValue<Byte4> rhs);
545// RValue<Byte4> operator>>=(Byte4 &lhs, RValue<Byte4> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400546// RValue<Byte4> operator+(RValue<Byte4> val);
547// RValue<Byte4> operator-(RValue<Byte4> val);
548// RValue<Byte4> operator~(RValue<Byte4> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500549// RValue<Byte4> operator++(Byte4 &val, int); // Post-increment
550// const Byte4 &operator++(Byte4 &val); // Pre-increment
551// RValue<Byte4> operator--(Byte4 &val, int); // Post-decrement
552// const Byte4 &operator--(Byte4 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400553
Nicolas Capens297d26e2016-11-18 12:52:17 -0500554 class SByte4 : public LValue<SByte4>
Nicolas Capensd022e412016-09-26 13:30:14 -0400555 {
556 public:
Nicolas Capensa25311a2017-01-16 17:19:00 -0500557 SByte4() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400558 // SByte4(int x, int y, int z, int w);
559 // SByte4(RValue<SByte4> rhs);
560 // SByte4(const SByte4 &rhs);
561 // SByte4(const Reference<SByte4> &rhs);
562
Nicolas Capens96d4e092016-11-18 14:22:38 -0500563 // RValue<SByte4> operator=(RValue<SByte4> rhs);
564 // RValue<SByte4> operator=(const SByte4 &rhs);
565 // RValue<SByte4> operator=(const Reference<SByte4> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400566
567 static Type *getType();
568 };
569
570// RValue<SByte4> operator+(RValue<SByte4> lhs, RValue<SByte4> rhs);
571// RValue<SByte4> operator-(RValue<SByte4> lhs, RValue<SByte4> rhs);
572// RValue<SByte4> operator*(RValue<SByte4> lhs, RValue<SByte4> rhs);
573// RValue<SByte4> operator/(RValue<SByte4> lhs, RValue<SByte4> rhs);
574// RValue<SByte4> operator%(RValue<SByte4> lhs, RValue<SByte4> rhs);
575// RValue<SByte4> operator&(RValue<SByte4> lhs, RValue<SByte4> rhs);
576// RValue<SByte4> operator|(RValue<SByte4> lhs, RValue<SByte4> rhs);
577// RValue<SByte4> operator^(RValue<SByte4> lhs, RValue<SByte4> rhs);
578// RValue<SByte4> operator<<(RValue<SByte4> lhs, RValue<SByte4> rhs);
579// RValue<SByte4> operator>>(RValue<SByte4> lhs, RValue<SByte4> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500580// RValue<SByte4> operator+=(SByte4 &lhs, RValue<SByte4> rhs);
581// RValue<SByte4> operator-=(SByte4 &lhs, RValue<SByte4> rhs);
582// RValue<SByte4> operator*=(SByte4 &lhs, RValue<SByte4> rhs);
583// RValue<SByte4> operator/=(SByte4 &lhs, RValue<SByte4> rhs);
584// RValue<SByte4> operator%=(SByte4 &lhs, RValue<SByte4> rhs);
585// RValue<SByte4> operator&=(SByte4 &lhs, RValue<SByte4> rhs);
586// RValue<SByte4> operator|=(SByte4 &lhs, RValue<SByte4> rhs);
587// RValue<SByte4> operator^=(SByte4 &lhs, RValue<SByte4> rhs);
588// RValue<SByte4> operator<<=(SByte4 &lhs, RValue<SByte4> rhs);
589// RValue<SByte4> operator>>=(SByte4 &lhs, RValue<SByte4> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400590// RValue<SByte4> operator+(RValue<SByte4> val);
591// RValue<SByte4> operator-(RValue<SByte4> val);
592// RValue<SByte4> operator~(RValue<SByte4> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500593// RValue<SByte4> operator++(SByte4 &val, int); // Post-increment
594// const SByte4 &operator++(SByte4 &val); // Pre-increment
595// RValue<SByte4> operator--(SByte4 &val, int); // Post-decrement
596// const SByte4 &operator--(SByte4 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400597
Nicolas Capens297d26e2016-11-18 12:52:17 -0500598 class Byte8 : public LValue<Byte8>
Nicolas Capensd022e412016-09-26 13:30:14 -0400599 {
600 public:
Nicolas Capensa25311a2017-01-16 17:19:00 -0500601 Byte8() = default;
Nicolas Capens3bbc5e12016-09-27 10:49:52 -0400602 Byte8(uint8_t x0, uint8_t x1, uint8_t x2, uint8_t x3, uint8_t x4, uint8_t x5, uint8_t x6, uint8_t x7);
Nicolas Capensd022e412016-09-26 13:30:14 -0400603 Byte8(RValue<Byte8> rhs);
604 Byte8(const Byte8 &rhs);
605 Byte8(const Reference<Byte8> &rhs);
606
Nicolas Capens96d4e092016-11-18 14:22:38 -0500607 RValue<Byte8> operator=(RValue<Byte8> rhs);
608 RValue<Byte8> operator=(const Byte8 &rhs);
609 RValue<Byte8> operator=(const Reference<Byte8> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400610
611 static Type *getType();
612 };
613
614 RValue<Byte8> operator+(RValue<Byte8> lhs, RValue<Byte8> rhs);
615 RValue<Byte8> operator-(RValue<Byte8> lhs, RValue<Byte8> rhs);
616// RValue<Byte8> operator*(RValue<Byte8> lhs, RValue<Byte8> rhs);
617// RValue<Byte8> operator/(RValue<Byte8> lhs, RValue<Byte8> rhs);
618// RValue<Byte8> operator%(RValue<Byte8> lhs, RValue<Byte8> rhs);
619 RValue<Byte8> operator&(RValue<Byte8> lhs, RValue<Byte8> rhs);
620 RValue<Byte8> operator|(RValue<Byte8> lhs, RValue<Byte8> rhs);
621 RValue<Byte8> operator^(RValue<Byte8> lhs, RValue<Byte8> rhs);
622// RValue<Byte8> operator<<(RValue<Byte8> lhs, RValue<Byte8> rhs);
623// RValue<Byte8> operator>>(RValue<Byte8> lhs, RValue<Byte8> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500624 RValue<Byte8> operator+=(Byte8 &lhs, RValue<Byte8> rhs);
625 RValue<Byte8> operator-=(Byte8 &lhs, RValue<Byte8> rhs);
626// RValue<Byte8> operator*=(Byte8 &lhs, RValue<Byte8> rhs);
627// RValue<Byte8> operator/=(Byte8 &lhs, RValue<Byte8> rhs);
628// RValue<Byte8> operator%=(Byte8 &lhs, RValue<Byte8> rhs);
629 RValue<Byte8> operator&=(Byte8 &lhs, RValue<Byte8> rhs);
630 RValue<Byte8> operator|=(Byte8 &lhs, RValue<Byte8> rhs);
631 RValue<Byte8> operator^=(Byte8 &lhs, RValue<Byte8> rhs);
632// RValue<Byte8> operator<<=(Byte8 &lhs, RValue<Byte8> rhs);
633// RValue<Byte8> operator>>=(Byte8 &lhs, RValue<Byte8> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400634// RValue<Byte8> operator+(RValue<Byte8> val);
635// RValue<Byte8> operator-(RValue<Byte8> val);
636 RValue<Byte8> operator~(RValue<Byte8> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500637// RValue<Byte8> operator++(Byte8 &val, int); // Post-increment
638// const Byte8 &operator++(Byte8 &val); // Pre-increment
639// RValue<Byte8> operator--(Byte8 &val, int); // Post-decrement
640// const Byte8 &operator--(Byte8 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400641
642 RValue<Byte8> AddSat(RValue<Byte8> x, RValue<Byte8> y);
643 RValue<Byte8> SubSat(RValue<Byte8> x, RValue<Byte8> y);
644 RValue<Short4> Unpack(RValue<Byte4> x);
Nicolas Capens411273e2017-01-26 15:13:36 -0800645 RValue<Short4> Unpack(RValue<Byte4> x, RValue<Byte4> y);
Nicolas Capensd022e412016-09-26 13:30:14 -0400646 RValue<Short4> UnpackLow(RValue<Byte8> x, RValue<Byte8> y);
647 RValue<Short4> UnpackHigh(RValue<Byte8> x, RValue<Byte8> y);
648 RValue<Int> SignMask(RValue<Byte8> x);
649// RValue<Byte8> CmpGT(RValue<Byte8> x, RValue<Byte8> y);
650 RValue<Byte8> CmpEQ(RValue<Byte8> x, RValue<Byte8> y);
651
Nicolas Capens297d26e2016-11-18 12:52:17 -0500652 class SByte8 : public LValue<SByte8>
Nicolas Capensd022e412016-09-26 13:30:14 -0400653 {
654 public:
Nicolas Capensa25311a2017-01-16 17:19:00 -0500655 SByte8() = default;
Nicolas Capens3bbc5e12016-09-27 10:49:52 -0400656 SByte8(uint8_t x0, uint8_t x1, uint8_t x2, uint8_t x3, uint8_t x4, uint8_t x5, uint8_t x6, uint8_t x7);
Nicolas Capensd022e412016-09-26 13:30:14 -0400657 SByte8(RValue<SByte8> rhs);
658 SByte8(const SByte8 &rhs);
659 SByte8(const Reference<SByte8> &rhs);
660
Nicolas Capens96d4e092016-11-18 14:22:38 -0500661 RValue<SByte8> operator=(RValue<SByte8> rhs);
662 RValue<SByte8> operator=(const SByte8 &rhs);
663 RValue<SByte8> operator=(const Reference<SByte8> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400664
665 static Type *getType();
666 };
667
668 RValue<SByte8> operator+(RValue<SByte8> lhs, RValue<SByte8> rhs);
669 RValue<SByte8> operator-(RValue<SByte8> lhs, RValue<SByte8> rhs);
670// RValue<SByte8> operator*(RValue<SByte8> lhs, RValue<SByte8> rhs);
671// RValue<SByte8> operator/(RValue<SByte8> lhs, RValue<SByte8> rhs);
672// RValue<SByte8> operator%(RValue<SByte8> lhs, RValue<SByte8> rhs);
673 RValue<SByte8> operator&(RValue<SByte8> lhs, RValue<SByte8> rhs);
674 RValue<SByte8> operator|(RValue<SByte8> lhs, RValue<SByte8> rhs);
675 RValue<SByte8> operator^(RValue<SByte8> lhs, RValue<SByte8> rhs);
676// RValue<SByte8> operator<<(RValue<SByte8> lhs, RValue<SByte8> rhs);
677// RValue<SByte8> operator>>(RValue<SByte8> lhs, RValue<SByte8> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500678 RValue<SByte8> operator+=(SByte8 &lhs, RValue<SByte8> rhs);
679 RValue<SByte8> operator-=(SByte8 &lhs, RValue<SByte8> rhs);
680// RValue<SByte8> operator*=(SByte8 &lhs, RValue<SByte8> rhs);
681// RValue<SByte8> operator/=(SByte8 &lhs, RValue<SByte8> rhs);
682// RValue<SByte8> operator%=(SByte8 &lhs, RValue<SByte8> rhs);
683 RValue<SByte8> operator&=(SByte8 &lhs, RValue<SByte8> rhs);
684 RValue<SByte8> operator|=(SByte8 &lhs, RValue<SByte8> rhs);
685 RValue<SByte8> operator^=(SByte8 &lhs, RValue<SByte8> rhs);
686// RValue<SByte8> operator<<=(SByte8 &lhs, RValue<SByte8> rhs);
687// RValue<SByte8> operator>>=(SByte8 &lhs, RValue<SByte8> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400688// RValue<SByte8> operator+(RValue<SByte8> val);
689// RValue<SByte8> operator-(RValue<SByte8> val);
690 RValue<SByte8> operator~(RValue<SByte8> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500691// RValue<SByte8> operator++(SByte8 &val, int); // Post-increment
692// const SByte8 &operator++(SByte8 &val); // Pre-increment
693// RValue<SByte8> operator--(SByte8 &val, int); // Post-decrement
694// const SByte8 &operator--(SByte8 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400695
696 RValue<SByte8> AddSat(RValue<SByte8> x, RValue<SByte8> y);
697 RValue<SByte8> SubSat(RValue<SByte8> x, RValue<SByte8> y);
698 RValue<Short4> UnpackLow(RValue<SByte8> x, RValue<SByte8> y);
699 RValue<Short4> UnpackHigh(RValue<SByte8> x, RValue<SByte8> y);
700 RValue<Int> SignMask(RValue<SByte8> x);
701 RValue<Byte8> CmpGT(RValue<SByte8> x, RValue<SByte8> y);
702 RValue<Byte8> CmpEQ(RValue<SByte8> x, RValue<SByte8> y);
703
Nicolas Capens297d26e2016-11-18 12:52:17 -0500704 class Byte16 : public LValue<Byte16>
Nicolas Capensd022e412016-09-26 13:30:14 -0400705 {
706 public:
Nicolas Capensa25311a2017-01-16 17:19:00 -0500707 Byte16() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400708 // Byte16(int x, int y, int z, int w);
709 Byte16(RValue<Byte16> rhs);
710 Byte16(const Byte16 &rhs);
711 Byte16(const Reference<Byte16> &rhs);
712
Nicolas Capens96d4e092016-11-18 14:22:38 -0500713 RValue<Byte16> operator=(RValue<Byte16> rhs);
714 RValue<Byte16> operator=(const Byte16 &rhs);
715 RValue<Byte16> operator=(const Reference<Byte16> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400716
717 static Type *getType();
718 };
719
720// RValue<Byte16> operator+(RValue<Byte16> lhs, RValue<Byte16> rhs);
721// RValue<Byte16> operator-(RValue<Byte16> lhs, RValue<Byte16> rhs);
722// RValue<Byte16> operator*(RValue<Byte16> lhs, RValue<Byte16> rhs);
723// RValue<Byte16> operator/(RValue<Byte16> lhs, RValue<Byte16> rhs);
724// RValue<Byte16> operator%(RValue<Byte16> lhs, RValue<Byte16> rhs);
725// RValue<Byte16> operator&(RValue<Byte16> lhs, RValue<Byte16> rhs);
726// RValue<Byte16> operator|(RValue<Byte16> lhs, RValue<Byte16> rhs);
727// RValue<Byte16> operator^(RValue<Byte16> lhs, RValue<Byte16> rhs);
728// RValue<Byte16> operator<<(RValue<Byte16> lhs, RValue<Byte16> rhs);
729// RValue<Byte16> operator>>(RValue<Byte16> lhs, RValue<Byte16> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500730// RValue<Byte16> operator+=(Byte16 &lhs, RValue<Byte16> rhs);
731// RValue<Byte16> operator-=(Byte16 &lhs, RValue<Byte16> rhs);
732// RValue<Byte16> operator*=(Byte16 &lhs, RValue<Byte16> rhs);
733// RValue<Byte16> operator/=(Byte16 &lhs, RValue<Byte16> rhs);
734// RValue<Byte16> operator%=(Byte16 &lhs, RValue<Byte16> rhs);
735// RValue<Byte16> operator&=(Byte16 &lhs, RValue<Byte16> rhs);
736// RValue<Byte16> operator|=(Byte16 &lhs, RValue<Byte16> rhs);
737// RValue<Byte16> operator^=(Byte16 &lhs, RValue<Byte16> rhs);
738// RValue<Byte16> operator<<=(Byte16 &lhs, RValue<Byte16> rhs);
739// RValue<Byte16> operator>>=(Byte16 &lhs, RValue<Byte16> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400740// RValue<Byte16> operator+(RValue<Byte16> val);
741// RValue<Byte16> operator-(RValue<Byte16> val);
742// RValue<Byte16> operator~(RValue<Byte16> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500743// RValue<Byte16> operator++(Byte16 &val, int); // Post-increment
744// const Byte16 &operator++(Byte16 &val); // Pre-increment
745// RValue<Byte16> operator--(Byte16 &val, int); // Post-decrement
746// const Byte16 &operator--(Byte16 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400747
Nicolas Capens297d26e2016-11-18 12:52:17 -0500748 class SByte16 : public LValue<SByte16>
Nicolas Capensd022e412016-09-26 13:30:14 -0400749 {
750 public:
Nicolas Capensa25311a2017-01-16 17:19:00 -0500751 SByte16() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400752 // SByte16(int x, int y, int z, int w);
753 // SByte16(RValue<SByte16> rhs);
754 // SByte16(const SByte16 &rhs);
755 // SByte16(const Reference<SByte16> &rhs);
756
Nicolas Capens96d4e092016-11-18 14:22:38 -0500757 // RValue<SByte16> operator=(RValue<SByte16> rhs);
758 // RValue<SByte16> operator=(const SByte16 &rhs);
759 // RValue<SByte16> operator=(const Reference<SByte16> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400760
761 static Type *getType();
762 };
763
764// RValue<SByte16> operator+(RValue<SByte16> lhs, RValue<SByte16> rhs);
765// RValue<SByte16> operator-(RValue<SByte16> lhs, RValue<SByte16> rhs);
766// RValue<SByte16> operator*(RValue<SByte16> lhs, RValue<SByte16> rhs);
767// RValue<SByte16> operator/(RValue<SByte16> lhs, RValue<SByte16> rhs);
768// RValue<SByte16> operator%(RValue<SByte16> lhs, RValue<SByte16> rhs);
769// RValue<SByte16> operator&(RValue<SByte16> lhs, RValue<SByte16> rhs);
770// RValue<SByte16> operator|(RValue<SByte16> lhs, RValue<SByte16> rhs);
771// RValue<SByte16> operator^(RValue<SByte16> lhs, RValue<SByte16> rhs);
772// RValue<SByte16> operator<<(RValue<SByte16> lhs, RValue<SByte16> rhs);
773// RValue<SByte16> operator>>(RValue<SByte16> lhs, RValue<SByte16> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500774// RValue<SByte16> operator+=(SByte16 &lhs, RValue<SByte16> rhs);
775// RValue<SByte16> operator-=(SByte16 &lhs, RValue<SByte16> rhs);
776// RValue<SByte16> operator*=(SByte16 &lhs, RValue<SByte16> rhs);
777// RValue<SByte16> operator/=(SByte16 &lhs, RValue<SByte16> rhs);
778// RValue<SByte16> operator%=(SByte16 &lhs, RValue<SByte16> rhs);
779// RValue<SByte16> operator&=(SByte16 &lhs, RValue<SByte16> rhs);
780// RValue<SByte16> operator|=(SByte16 &lhs, RValue<SByte16> rhs);
781// RValue<SByte16> operator^=(SByte16 &lhs, RValue<SByte16> rhs);
782// RValue<SByte16> operator<<=(SByte16 &lhs, RValue<SByte16> rhs);
783// RValue<SByte16> operator>>=(SByte16 &lhs, RValue<SByte16> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400784// RValue<SByte16> operator+(RValue<SByte16> val);
785// RValue<SByte16> operator-(RValue<SByte16> val);
786// RValue<SByte16> operator~(RValue<SByte16> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500787// RValue<SByte16> operator++(SByte16 &val, int); // Post-increment
788// const SByte16 &operator++(SByte16 &val); // Pre-increment
789// RValue<SByte16> operator--(SByte16 &val, int); // Post-decrement
790// const SByte16 &operator--(SByte16 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400791
Nicolas Capens297d26e2016-11-18 12:52:17 -0500792 class Short2 : public LValue<Short2>
Nicolas Capens16b5f152016-10-13 13:39:01 -0400793 {
794 public:
795 explicit Short2(RValue<Short4> cast);
796
797 static Type *getType();
798 };
799
Nicolas Capens297d26e2016-11-18 12:52:17 -0500800 class UShort2 : public LValue<UShort2>
Nicolas Capens16b5f152016-10-13 13:39:01 -0400801 {
802 public:
803 explicit UShort2(RValue<UShort4> cast);
804
805 static Type *getType();
806 };
807
Nicolas Capens297d26e2016-11-18 12:52:17 -0500808 class Short4 : public LValue<Short4>
Nicolas Capensd022e412016-09-26 13:30:14 -0400809 {
810 public:
811 explicit Short4(RValue<Int> cast);
812 explicit Short4(RValue<Int4> cast);
813 // explicit Short4(RValue<Float> cast);
814 explicit Short4(RValue<Float4> cast);
815
Nicolas Capensa25311a2017-01-16 17:19:00 -0500816 Short4() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400817 Short4(short xyzw);
818 Short4(short x, short y, short z, short w);
819 Short4(RValue<Short4> rhs);
820 Short4(const Short4 &rhs);
821 Short4(const Reference<Short4> &rhs);
822 Short4(RValue<UShort4> rhs);
823 Short4(const UShort4 &rhs);
824 Short4(const Reference<UShort4> &rhs);
825
Nicolas Capens96d4e092016-11-18 14:22:38 -0500826 RValue<Short4> operator=(RValue<Short4> rhs);
827 RValue<Short4> operator=(const Short4 &rhs);
828 RValue<Short4> operator=(const Reference<Short4> &rhs);
829 RValue<Short4> operator=(RValue<UShort4> rhs);
830 RValue<Short4> operator=(const UShort4 &rhs);
831 RValue<Short4> operator=(const Reference<UShort4> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400832
833 static Type *getType();
834 };
835
836 RValue<Short4> operator+(RValue<Short4> lhs, RValue<Short4> rhs);
837 RValue<Short4> operator-(RValue<Short4> lhs, RValue<Short4> rhs);
838 RValue<Short4> operator*(RValue<Short4> lhs, RValue<Short4> rhs);
839// RValue<Short4> operator/(RValue<Short4> lhs, RValue<Short4> rhs);
840// RValue<Short4> operator%(RValue<Short4> lhs, RValue<Short4> rhs);
841 RValue<Short4> operator&(RValue<Short4> lhs, RValue<Short4> rhs);
842 RValue<Short4> operator|(RValue<Short4> lhs, RValue<Short4> rhs);
843 RValue<Short4> operator^(RValue<Short4> lhs, RValue<Short4> rhs);
844 RValue<Short4> operator<<(RValue<Short4> lhs, unsigned char rhs);
845 RValue<Short4> operator>>(RValue<Short4> lhs, unsigned char rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500846 RValue<Short4> operator+=(Short4 &lhs, RValue<Short4> rhs);
847 RValue<Short4> operator-=(Short4 &lhs, RValue<Short4> rhs);
848 RValue<Short4> operator*=(Short4 &lhs, RValue<Short4> rhs);
849// RValue<Short4> operator/=(Short4 &lhs, RValue<Short4> rhs);
850// RValue<Short4> operator%=(Short4 &lhs, RValue<Short4> rhs);
851 RValue<Short4> operator&=(Short4 &lhs, RValue<Short4> rhs);
852 RValue<Short4> operator|=(Short4 &lhs, RValue<Short4> rhs);
853 RValue<Short4> operator^=(Short4 &lhs, RValue<Short4> rhs);
854 RValue<Short4> operator<<=(Short4 &lhs, unsigned char rhs);
855 RValue<Short4> operator>>=(Short4 &lhs, unsigned char rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400856// RValue<Short4> operator+(RValue<Short4> val);
857 RValue<Short4> operator-(RValue<Short4> val);
858 RValue<Short4> operator~(RValue<Short4> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500859// RValue<Short4> operator++(Short4 &val, int); // Post-increment
860// const Short4 &operator++(Short4 &val); // Pre-increment
861// RValue<Short4> operator--(Short4 &val, int); // Post-decrement
862// const Short4 &operator--(Short4 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400863// RValue<Bool> operator<(RValue<Short4> lhs, RValue<Short4> rhs);
864// RValue<Bool> operator<=(RValue<Short4> lhs, RValue<Short4> rhs);
865// RValue<Bool> operator>(RValue<Short4> lhs, RValue<Short4> rhs);
866// RValue<Bool> operator>=(RValue<Short4> lhs, RValue<Short4> rhs);
867// RValue<Bool> operator!=(RValue<Short4> lhs, RValue<Short4> rhs);
868// RValue<Bool> operator==(RValue<Short4> lhs, RValue<Short4> rhs);
869
870 RValue<Short4> RoundShort4(RValue<Float4> cast);
871 RValue<Short4> Max(RValue<Short4> x, RValue<Short4> y);
872 RValue<Short4> Min(RValue<Short4> x, RValue<Short4> y);
873 RValue<Short4> AddSat(RValue<Short4> x, RValue<Short4> y);
874 RValue<Short4> SubSat(RValue<Short4> x, RValue<Short4> y);
875 RValue<Short4> MulHigh(RValue<Short4> x, RValue<Short4> y);
876 RValue<Int2> MulAdd(RValue<Short4> x, RValue<Short4> y);
Nicolas Capens33438a62017-09-27 11:47:35 -0400877 RValue<SByte8> PackSigned(RValue<Short4> x, RValue<Short4> y);
878 RValue<Byte8> PackUnsigned(RValue<Short4> x, RValue<Short4> y);
Nicolas Capensd022e412016-09-26 13:30:14 -0400879 RValue<Int2> UnpackLow(RValue<Short4> x, RValue<Short4> y);
880 RValue<Int2> UnpackHigh(RValue<Short4> x, RValue<Short4> y);
881 RValue<Short4> Swizzle(RValue<Short4> x, unsigned char select);
882 RValue<Short4> Insert(RValue<Short4> val, RValue<Short> element, int i);
883 RValue<Short> Extract(RValue<Short4> val, int i);
884 RValue<Short4> CmpGT(RValue<Short4> x, RValue<Short4> y);
885 RValue<Short4> CmpEQ(RValue<Short4> x, RValue<Short4> y);
886
Nicolas Capens297d26e2016-11-18 12:52:17 -0500887 class UShort4 : public LValue<UShort4>
Nicolas Capensd022e412016-09-26 13:30:14 -0400888 {
889 public:
890 explicit UShort4(RValue<Int4> cast);
891 explicit UShort4(RValue<Float4> cast, bool saturate = false);
892
Nicolas Capensa25311a2017-01-16 17:19:00 -0500893 UShort4() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -0400894 UShort4(unsigned short xyzw);
895 UShort4(unsigned short x, unsigned short y, unsigned short z, unsigned short w);
896 UShort4(RValue<UShort4> rhs);
897 UShort4(const UShort4 &rhs);
898 UShort4(const Reference<UShort4> &rhs);
899 UShort4(RValue<Short4> rhs);
900 UShort4(const Short4 &rhs);
901 UShort4(const Reference<Short4> &rhs);
902
Nicolas Capens96d4e092016-11-18 14:22:38 -0500903 RValue<UShort4> operator=(RValue<UShort4> rhs);
904 RValue<UShort4> operator=(const UShort4 &rhs);
905 RValue<UShort4> operator=(const Reference<UShort4> &rhs);
906 RValue<UShort4> operator=(RValue<Short4> rhs);
907 RValue<UShort4> operator=(const Short4 &rhs);
908 RValue<UShort4> operator=(const Reference<Short4> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400909
910 static Type *getType();
911 };
912
913 RValue<UShort4> operator+(RValue<UShort4> lhs, RValue<UShort4> rhs);
914 RValue<UShort4> operator-(RValue<UShort4> lhs, RValue<UShort4> rhs);
915 RValue<UShort4> operator*(RValue<UShort4> lhs, RValue<UShort4> rhs);
916// RValue<UShort4> operator/(RValue<UShort4> lhs, RValue<UShort4> rhs);
917// RValue<UShort4> operator%(RValue<UShort4> lhs, RValue<UShort4> rhs);
Nicolas Capens16b5f152016-10-13 13:39:01 -0400918 RValue<UShort4> operator&(RValue<UShort4> lhs, RValue<UShort4> rhs);
919 RValue<UShort4> operator|(RValue<UShort4> lhs, RValue<UShort4> rhs);
920 RValue<UShort4> operator^(RValue<UShort4> lhs, RValue<UShort4> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400921 RValue<UShort4> operator<<(RValue<UShort4> lhs, unsigned char rhs);
922 RValue<UShort4> operator>>(RValue<UShort4> lhs, unsigned char rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500923// RValue<UShort4> operator+=(UShort4 &lhs, RValue<UShort4> rhs);
924// RValue<UShort4> operator-=(UShort4 &lhs, RValue<UShort4> rhs);
925// RValue<UShort4> operator*=(UShort4 &lhs, RValue<UShort4> rhs);
926// RValue<UShort4> operator/=(UShort4 &lhs, RValue<UShort4> rhs);
927// RValue<UShort4> operator%=(UShort4 &lhs, RValue<UShort4> rhs);
928// RValue<UShort4> operator&=(UShort4 &lhs, RValue<UShort4> rhs);
929// RValue<UShort4> operator|=(UShort4 &lhs, RValue<UShort4> rhs);
930// RValue<UShort4> operator^=(UShort4 &lhs, RValue<UShort4> rhs);
931 RValue<UShort4> operator<<=(UShort4 &lhs, unsigned char rhs);
932 RValue<UShort4> operator>>=(UShort4 &lhs, unsigned char rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400933// RValue<UShort4> operator+(RValue<UShort4> val);
934// RValue<UShort4> operator-(RValue<UShort4> val);
935 RValue<UShort4> operator~(RValue<UShort4> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500936// RValue<UShort4> operator++(UShort4 &val, int); // Post-increment
937// const UShort4 &operator++(UShort4 &val); // Pre-increment
938// RValue<UShort4> operator--(UShort4 &val, int); // Post-decrement
939// const UShort4 &operator--(UShort4 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400940
941 RValue<UShort4> Max(RValue<UShort4> x, RValue<UShort4> y);
942 RValue<UShort4> Min(RValue<UShort4> x, RValue<UShort4> y);
943 RValue<UShort4> AddSat(RValue<UShort4> x, RValue<UShort4> y);
944 RValue<UShort4> SubSat(RValue<UShort4> x, RValue<UShort4> y);
945 RValue<UShort4> MulHigh(RValue<UShort4> x, RValue<UShort4> y);
946 RValue<UShort4> Average(RValue<UShort4> x, RValue<UShort4> y);
Nicolas Capensd022e412016-09-26 13:30:14 -0400947
Nicolas Capens297d26e2016-11-18 12:52:17 -0500948 class Short8 : public LValue<Short8>
Nicolas Capensd022e412016-09-26 13:30:14 -0400949 {
950 public:
Nicolas Capensa25311a2017-01-16 17:19:00 -0500951 Short8() = default;
Nicolas Capens3e7062b2017-01-17 14:01:33 -0500952 Short8(short c);
Nicolas Capensd022e412016-09-26 13:30:14 -0400953 Short8(short c0, short c1, short c2, short c3, short c4, short c5, short c6, short c7);
954 Short8(RValue<Short8> rhs);
955 // Short8(const Short8 &rhs);
956 Short8(const Reference<Short8> &rhs);
957 Short8(RValue<Short4> lo, RValue<Short4> hi);
958
Nicolas Capensf1beca42019-03-26 17:18:57 -0400959 RValue<Short8> operator=(RValue<Short8> rhs);
960 RValue<Short8> operator=(const Short8 &rhs);
961 RValue<Short8> operator=(const Reference<Short8> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400962
963 static Type *getType();
964 };
965
966 RValue<Short8> operator+(RValue<Short8> lhs, RValue<Short8> rhs);
967// RValue<Short8> operator-(RValue<Short8> lhs, RValue<Short8> rhs);
968// RValue<Short8> operator*(RValue<Short8> lhs, RValue<Short8> rhs);
969// RValue<Short8> operator/(RValue<Short8> lhs, RValue<Short8> rhs);
970// RValue<Short8> operator%(RValue<Short8> lhs, RValue<Short8> rhs);
971 RValue<Short8> operator&(RValue<Short8> lhs, RValue<Short8> rhs);
972// RValue<Short8> operator|(RValue<Short8> lhs, RValue<Short8> rhs);
973// RValue<Short8> operator^(RValue<Short8> lhs, RValue<Short8> rhs);
974 RValue<Short8> operator<<(RValue<Short8> lhs, unsigned char rhs);
975 RValue<Short8> operator>>(RValue<Short8> lhs, unsigned char rhs);
976// RValue<Short8> operator<<(RValue<Short8> lhs, RValue<Short8> rhs);
977// RValue<Short8> operator>>(RValue<Short8> lhs, RValue<Short8> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500978// RValue<Short8> operator+=(Short8 &lhs, RValue<Short8> rhs);
979// RValue<Short8> operator-=(Short8 &lhs, RValue<Short8> rhs);
980// RValue<Short8> operator*=(Short8 &lhs, RValue<Short8> rhs);
981// RValue<Short8> operator/=(Short8 &lhs, RValue<Short8> rhs);
982// RValue<Short8> operator%=(Short8 &lhs, RValue<Short8> rhs);
983// RValue<Short8> operator&=(Short8 &lhs, RValue<Short8> rhs);
984// RValue<Short8> operator|=(Short8 &lhs, RValue<Short8> rhs);
985// RValue<Short8> operator^=(Short8 &lhs, RValue<Short8> rhs);
986// RValue<Short8> operator<<=(Short8 &lhs, RValue<Short8> rhs);
987// RValue<Short8> operator>>=(Short8 &lhs, RValue<Short8> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -0400988// RValue<Short8> operator+(RValue<Short8> val);
989// RValue<Short8> operator-(RValue<Short8> val);
990// RValue<Short8> operator~(RValue<Short8> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -0500991// RValue<Short8> operator++(Short8 &val, int); // Post-increment
992// const Short8 &operator++(Short8 &val); // Pre-increment
993// RValue<Short8> operator--(Short8 &val, int); // Post-decrement
994// const Short8 &operator--(Short8 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -0400995// RValue<Bool> operator<(RValue<Short8> lhs, RValue<Short8> rhs);
996// RValue<Bool> operator<=(RValue<Short8> lhs, RValue<Short8> rhs);
997// RValue<Bool> operator>(RValue<Short8> lhs, RValue<Short8> rhs);
998// RValue<Bool> operator>=(RValue<Short8> lhs, RValue<Short8> rhs);
999// RValue<Bool> operator!=(RValue<Short8> lhs, RValue<Short8> rhs);
1000// RValue<Bool> operator==(RValue<Short8> lhs, RValue<Short8> rhs);
1001
1002 RValue<Short8> MulHigh(RValue<Short8> x, RValue<Short8> y);
1003 RValue<Int4> MulAdd(RValue<Short8> x, RValue<Short8> y);
1004 RValue<Int4> Abs(RValue<Int4> x);
1005
Nicolas Capens297d26e2016-11-18 12:52:17 -05001006 class UShort8 : public LValue<UShort8>
Nicolas Capensd022e412016-09-26 13:30:14 -04001007 {
1008 public:
Nicolas Capensa25311a2017-01-16 17:19:00 -05001009 UShort8() = default;
Nicolas Capens3e7062b2017-01-17 14:01:33 -05001010 UShort8(unsigned short c);
Nicolas Capensd022e412016-09-26 13:30:14 -04001011 UShort8(unsigned short c0, unsigned short c1, unsigned short c2, unsigned short c3, unsigned short c4, unsigned short c5, unsigned short c6, unsigned short c7);
1012 UShort8(RValue<UShort8> rhs);
1013 // UShort8(const UShort8 &rhs);
1014 UShort8(const Reference<UShort8> &rhs);
1015 UShort8(RValue<UShort4> lo, RValue<UShort4> hi);
1016
Nicolas Capens96d4e092016-11-18 14:22:38 -05001017 RValue<UShort8> operator=(RValue<UShort8> rhs);
1018 RValue<UShort8> operator=(const UShort8 &rhs);
1019 RValue<UShort8> operator=(const Reference<UShort8> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001020
1021 static Type *getType();
1022 };
1023
1024 RValue<UShort8> operator+(RValue<UShort8> lhs, RValue<UShort8> rhs);
1025// RValue<UShort8> operator-(RValue<UShort8> lhs, RValue<UShort8> rhs);
1026 RValue<UShort8> operator*(RValue<UShort8> lhs, RValue<UShort8> rhs);
1027// RValue<UShort8> operator/(RValue<UShort8> lhs, RValue<UShort8> rhs);
1028// RValue<UShort8> operator%(RValue<UShort8> lhs, RValue<UShort8> rhs);
1029 RValue<UShort8> operator&(RValue<UShort8> lhs, RValue<UShort8> rhs);
1030// RValue<UShort8> operator|(RValue<UShort8> lhs, RValue<UShort8> rhs);
1031// RValue<UShort8> operator^(RValue<UShort8> lhs, RValue<UShort8> rhs);
1032 RValue<UShort8> operator<<(RValue<UShort8> lhs, unsigned char rhs);
1033 RValue<UShort8> operator>>(RValue<UShort8> lhs, unsigned char rhs);
1034// RValue<UShort8> operator<<(RValue<UShort8> lhs, RValue<UShort8> rhs);
1035// RValue<UShort8> operator>>(RValue<UShort8> lhs, RValue<UShort8> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001036 RValue<UShort8> operator+=(UShort8 &lhs, RValue<UShort8> rhs);
1037// RValue<UShort8> operator-=(UShort8 &lhs, RValue<UShort8> rhs);
1038// RValue<UShort8> operator*=(UShort8 &lhs, RValue<UShort8> rhs);
1039// RValue<UShort8> operator/=(UShort8 &lhs, RValue<UShort8> rhs);
1040// RValue<UShort8> operator%=(UShort8 &lhs, RValue<UShort8> rhs);
1041// RValue<UShort8> operator&=(UShort8 &lhs, RValue<UShort8> rhs);
1042// RValue<UShort8> operator|=(UShort8 &lhs, RValue<UShort8> rhs);
1043// RValue<UShort8> operator^=(UShort8 &lhs, RValue<UShort8> rhs);
1044// RValue<UShort8> operator<<=(UShort8 &lhs, RValue<UShort8> rhs);
1045// RValue<UShort8> operator>>=(UShort8 &lhs, RValue<UShort8> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001046// RValue<UShort8> operator+(RValue<UShort8> val);
1047// RValue<UShort8> operator-(RValue<UShort8> val);
1048 RValue<UShort8> operator~(RValue<UShort8> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001049// RValue<UShort8> operator++(UShort8 &val, int); // Post-increment
1050// const UShort8 &operator++(UShort8 &val); // Pre-increment
1051// RValue<UShort8> operator--(UShort8 &val, int); // Post-decrement
1052// const UShort8 &operator--(UShort8 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -04001053// RValue<Bool> operator<(RValue<UShort8> lhs, RValue<UShort8> rhs);
1054// RValue<Bool> operator<=(RValue<UShort8> lhs, RValue<UShort8> rhs);
1055// RValue<Bool> operator>(RValue<UShort8> lhs, RValue<UShort8> rhs);
1056// RValue<Bool> operator>=(RValue<UShort8> lhs, RValue<UShort8> rhs);
1057// RValue<Bool> operator!=(RValue<UShort8> lhs, RValue<UShort8> rhs);
1058// RValue<Bool> operator==(RValue<UShort8> lhs, RValue<UShort8> rhs);
1059
1060 RValue<UShort8> Swizzle(RValue<UShort8> x, char select0, char select1, char select2, char select3, char select4, char select5, char select6, char select7);
1061 RValue<UShort8> MulHigh(RValue<UShort8> x, RValue<UShort8> y);
1062
Nicolas Capens297d26e2016-11-18 12:52:17 -05001063 class Int : public LValue<Int>
Nicolas Capensd022e412016-09-26 13:30:14 -04001064 {
1065 public:
1066 Int(Argument<Int> argument);
1067
1068 explicit Int(RValue<Byte> cast);
1069 explicit Int(RValue<SByte> cast);
1070 explicit Int(RValue<Short> cast);
1071 explicit Int(RValue<UShort> cast);
1072 explicit Int(RValue<Int2> cast);
1073 explicit Int(RValue<Long> cast);
1074 explicit Int(RValue<Float> cast);
1075
Nicolas Capensa25311a2017-01-16 17:19:00 -05001076 Int() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -04001077 Int(int x);
1078 Int(RValue<Int> rhs);
1079 Int(RValue<UInt> rhs);
1080 Int(const Int &rhs);
1081 Int(const UInt &rhs);
1082 Int(const Reference<Int> &rhs);
1083 Int(const Reference<UInt> &rhs);
1084
Nicolas Capens96d4e092016-11-18 14:22:38 -05001085 RValue<Int> operator=(int rhs);
1086 RValue<Int> operator=(RValue<Int> rhs);
1087 RValue<Int> operator=(RValue<UInt> rhs);
1088 RValue<Int> operator=(const Int &rhs);
1089 RValue<Int> operator=(const UInt &rhs);
1090 RValue<Int> operator=(const Reference<Int> &rhs);
1091 RValue<Int> operator=(const Reference<UInt> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001092
1093 static Type *getType();
1094 };
1095
1096 RValue<Int> operator+(RValue<Int> lhs, RValue<Int> rhs);
1097 RValue<Int> operator-(RValue<Int> lhs, RValue<Int> rhs);
1098 RValue<Int> operator*(RValue<Int> lhs, RValue<Int> rhs);
1099 RValue<Int> operator/(RValue<Int> lhs, RValue<Int> rhs);
1100 RValue<Int> operator%(RValue<Int> lhs, RValue<Int> rhs);
1101 RValue<Int> operator&(RValue<Int> lhs, RValue<Int> rhs);
1102 RValue<Int> operator|(RValue<Int> lhs, RValue<Int> rhs);
1103 RValue<Int> operator^(RValue<Int> lhs, RValue<Int> rhs);
1104 RValue<Int> operator<<(RValue<Int> lhs, RValue<Int> rhs);
1105 RValue<Int> operator>>(RValue<Int> lhs, RValue<Int> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001106 RValue<Int> operator+=(Int &lhs, RValue<Int> rhs);
1107 RValue<Int> operator-=(Int &lhs, RValue<Int> rhs);
1108 RValue<Int> operator*=(Int &lhs, RValue<Int> rhs);
1109 RValue<Int> operator/=(Int &lhs, RValue<Int> rhs);
1110 RValue<Int> operator%=(Int &lhs, RValue<Int> rhs);
1111 RValue<Int> operator&=(Int &lhs, RValue<Int> rhs);
1112 RValue<Int> operator|=(Int &lhs, RValue<Int> rhs);
1113 RValue<Int> operator^=(Int &lhs, RValue<Int> rhs);
1114 RValue<Int> operator<<=(Int &lhs, RValue<Int> rhs);
1115 RValue<Int> operator>>=(Int &lhs, RValue<Int> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001116 RValue<Int> operator+(RValue<Int> val);
1117 RValue<Int> operator-(RValue<Int> val);
1118 RValue<Int> operator~(RValue<Int> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001119 RValue<Int> operator++(Int &val, int); // Post-increment
1120 const Int &operator++(Int &val); // Pre-increment
1121 RValue<Int> operator--(Int &val, int); // Post-decrement
1122 const Int &operator--(Int &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -04001123 RValue<Bool> operator<(RValue<Int> lhs, RValue<Int> rhs);
1124 RValue<Bool> operator<=(RValue<Int> lhs, RValue<Int> rhs);
1125 RValue<Bool> operator>(RValue<Int> lhs, RValue<Int> rhs);
1126 RValue<Bool> operator>=(RValue<Int> lhs, RValue<Int> rhs);
1127 RValue<Bool> operator!=(RValue<Int> lhs, RValue<Int> rhs);
1128 RValue<Bool> operator==(RValue<Int> lhs, RValue<Int> rhs);
1129
1130 RValue<Int> Max(RValue<Int> x, RValue<Int> y);
1131 RValue<Int> Min(RValue<Int> x, RValue<Int> y);
1132 RValue<Int> Clamp(RValue<Int> x, RValue<Int> min, RValue<Int> max);
1133 RValue<Int> RoundInt(RValue<Float> cast);
1134
Nicolas Capens297d26e2016-11-18 12:52:17 -05001135 class Long : public LValue<Long>
Nicolas Capensd022e412016-09-26 13:30:14 -04001136 {
1137 public:
1138 // Long(Argument<Long> argument);
1139
1140 // explicit Long(RValue<Short> cast);
1141 // explicit Long(RValue<UShort> cast);
1142 explicit Long(RValue<Int> cast);
1143 explicit Long(RValue<UInt> cast);
1144 // explicit Long(RValue<Float> cast);
1145
Nicolas Capensa25311a2017-01-16 17:19:00 -05001146 Long() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -04001147 // Long(qword x);
1148 Long(RValue<Long> rhs);
1149 // Long(RValue<ULong> rhs);
1150 // Long(const Long &rhs);
1151 // Long(const Reference<Long> &rhs);
1152 // Long(const ULong &rhs);
1153 // Long(const Reference<ULong> &rhs);
1154
Nicolas Capens96d4e092016-11-18 14:22:38 -05001155 RValue<Long> operator=(int64_t rhs);
1156 RValue<Long> operator=(RValue<Long> rhs);
1157 // RValue<Long> operator=(RValue<ULong> rhs);
1158 RValue<Long> operator=(const Long &rhs);
1159 RValue<Long> operator=(const Reference<Long> &rhs);
1160 // RValue<Long> operator=(const ULong &rhs);
1161 // RValue<Long> operator=(const Reference<ULong> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001162
1163 static Type *getType();
1164 };
1165
1166 RValue<Long> operator+(RValue<Long> lhs, RValue<Long> rhs);
1167 RValue<Long> operator-(RValue<Long> lhs, RValue<Long> rhs);
Chris Forbesaa8f6992019-03-01 14:18:30 -08001168 RValue<Long> operator*(RValue<Long> lhs, RValue<Long> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001169// RValue<Long> operator/(RValue<Long> lhs, RValue<Long> rhs);
1170// RValue<Long> operator%(RValue<Long> lhs, RValue<Long> rhs);
1171// RValue<Long> operator&(RValue<Long> lhs, RValue<Long> rhs);
1172// RValue<Long> operator|(RValue<Long> lhs, RValue<Long> rhs);
1173// RValue<Long> operator^(RValue<Long> lhs, RValue<Long> rhs);
1174// RValue<Long> operator<<(RValue<Long> lhs, RValue<Long> rhs);
Chris Forbesaa8f6992019-03-01 14:18:30 -08001175 RValue<Long> operator>>(RValue<Long> lhs, RValue<Long> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001176 RValue<Long> operator+=(Long &lhs, RValue<Long> rhs);
1177 RValue<Long> operator-=(Long &lhs, RValue<Long> rhs);
1178// RValue<Long> operator*=(Long &lhs, RValue<Long> rhs);
1179// RValue<Long> operator/=(Long &lhs, RValue<Long> rhs);
1180// RValue<Long> operator%=(Long &lhs, RValue<Long> rhs);
1181// RValue<Long> operator&=(Long &lhs, RValue<Long> rhs);
1182// RValue<Long> operator|=(Long &lhs, RValue<Long> rhs);
1183// RValue<Long> operator^=(Long &lhs, RValue<Long> rhs);
1184// RValue<Long> operator<<=(Long &lhs, RValue<Long> rhs);
1185// RValue<Long> operator>>=(Long &lhs, RValue<Long> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001186// RValue<Long> operator+(RValue<Long> val);
1187// RValue<Long> operator-(RValue<Long> val);
1188// RValue<Long> operator~(RValue<Long> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001189// RValue<Long> operator++(Long &val, int); // Post-increment
1190// const Long &operator++(Long &val); // Pre-increment
1191// RValue<Long> operator--(Long &val, int); // Post-decrement
1192// const Long &operator--(Long &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -04001193// RValue<Bool> operator<(RValue<Long> lhs, RValue<Long> rhs);
1194// RValue<Bool> operator<=(RValue<Long> lhs, RValue<Long> rhs);
1195// RValue<Bool> operator>(RValue<Long> lhs, RValue<Long> rhs);
1196// RValue<Bool> operator>=(RValue<Long> lhs, RValue<Long> rhs);
1197// RValue<Bool> operator!=(RValue<Long> lhs, RValue<Long> rhs);
1198// RValue<Bool> operator==(RValue<Long> lhs, RValue<Long> rhs);
1199
1200// RValue<Long> RoundLong(RValue<Float> cast);
1201 RValue<Long> AddAtomic( RValue<Pointer<Long>> x, RValue<Long> y);
1202
Nicolas Capens297d26e2016-11-18 12:52:17 -05001203 class UInt : public LValue<UInt>
Nicolas Capensd022e412016-09-26 13:30:14 -04001204 {
1205 public:
1206 UInt(Argument<UInt> argument);
1207
1208 explicit UInt(RValue<UShort> cast);
1209 explicit UInt(RValue<Long> cast);
1210 explicit UInt(RValue<Float> cast);
1211
Nicolas Capensa25311a2017-01-16 17:19:00 -05001212 UInt() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -04001213 UInt(int x);
1214 UInt(unsigned int x);
1215 UInt(RValue<UInt> rhs);
1216 UInt(RValue<Int> rhs);
1217 UInt(const UInt &rhs);
1218 UInt(const Int &rhs);
1219 UInt(const Reference<UInt> &rhs);
1220 UInt(const Reference<Int> &rhs);
1221
Nicolas Capens96d4e092016-11-18 14:22:38 -05001222 RValue<UInt> operator=(unsigned int rhs);
1223 RValue<UInt> operator=(RValue<UInt> rhs);
1224 RValue<UInt> operator=(RValue<Int> rhs);
1225 RValue<UInt> operator=(const UInt &rhs);
1226 RValue<UInt> operator=(const Int &rhs);
1227 RValue<UInt> operator=(const Reference<UInt> &rhs);
1228 RValue<UInt> operator=(const Reference<Int> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001229
1230 static Type *getType();
1231 };
1232
1233 RValue<UInt> operator+(RValue<UInt> lhs, RValue<UInt> rhs);
1234 RValue<UInt> operator-(RValue<UInt> lhs, RValue<UInt> rhs);
1235 RValue<UInt> operator*(RValue<UInt> lhs, RValue<UInt> rhs);
1236 RValue<UInt> operator/(RValue<UInt> lhs, RValue<UInt> rhs);
1237 RValue<UInt> operator%(RValue<UInt> lhs, RValue<UInt> rhs);
1238 RValue<UInt> operator&(RValue<UInt> lhs, RValue<UInt> rhs);
1239 RValue<UInt> operator|(RValue<UInt> lhs, RValue<UInt> rhs);
1240 RValue<UInt> operator^(RValue<UInt> lhs, RValue<UInt> rhs);
1241 RValue<UInt> operator<<(RValue<UInt> lhs, RValue<UInt> rhs);
1242 RValue<UInt> operator>>(RValue<UInt> lhs, RValue<UInt> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001243 RValue<UInt> operator+=(UInt &lhs, RValue<UInt> rhs);
1244 RValue<UInt> operator-=(UInt &lhs, RValue<UInt> rhs);
1245 RValue<UInt> operator*=(UInt &lhs, RValue<UInt> rhs);
1246 RValue<UInt> operator/=(UInt &lhs, RValue<UInt> rhs);
1247 RValue<UInt> operator%=(UInt &lhs, RValue<UInt> rhs);
1248 RValue<UInt> operator&=(UInt &lhs, RValue<UInt> rhs);
1249 RValue<UInt> operator|=(UInt &lhs, RValue<UInt> rhs);
1250 RValue<UInt> operator^=(UInt &lhs, RValue<UInt> rhs);
1251 RValue<UInt> operator<<=(UInt &lhs, RValue<UInt> rhs);
1252 RValue<UInt> operator>>=(UInt &lhs, RValue<UInt> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001253 RValue<UInt> operator+(RValue<UInt> val);
1254 RValue<UInt> operator-(RValue<UInt> val);
1255 RValue<UInt> operator~(RValue<UInt> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001256 RValue<UInt> operator++(UInt &val, int); // Post-increment
1257 const UInt &operator++(UInt &val); // Pre-increment
1258 RValue<UInt> operator--(UInt &val, int); // Post-decrement
1259 const UInt &operator--(UInt &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -04001260 RValue<Bool> operator<(RValue<UInt> lhs, RValue<UInt> rhs);
1261 RValue<Bool> operator<=(RValue<UInt> lhs, RValue<UInt> rhs);
1262 RValue<Bool> operator>(RValue<UInt> lhs, RValue<UInt> rhs);
1263 RValue<Bool> operator>=(RValue<UInt> lhs, RValue<UInt> rhs);
1264 RValue<Bool> operator!=(RValue<UInt> lhs, RValue<UInt> rhs);
1265 RValue<Bool> operator==(RValue<UInt> lhs, RValue<UInt> rhs);
1266
1267 RValue<UInt> Max(RValue<UInt> x, RValue<UInt> y);
1268 RValue<UInt> Min(RValue<UInt> x, RValue<UInt> y);
1269 RValue<UInt> Clamp(RValue<UInt> x, RValue<UInt> min, RValue<UInt> max);
Chris Forbes17813932019-04-18 11:45:54 -07001270
1271 RValue<UInt> AddAtomic(RValue<Pointer<UInt>> x, RValue<UInt> y, std::memory_order memoryOrder);
Chris Forbes707ed992019-04-18 18:17:35 -07001272 RValue<UInt> SubAtomic(RValue<Pointer<UInt>> x, RValue<UInt> y, std::memory_order memoryOrder);
Chris Forbes17813932019-04-18 11:45:54 -07001273 RValue<UInt> AndAtomic(RValue<Pointer<UInt>> x, RValue<UInt> y, std::memory_order memoryOrder);
1274 RValue<UInt> OrAtomic(RValue<Pointer<UInt>> x, RValue<UInt> y, std::memory_order memoryOrder);
1275 RValue<UInt> XorAtomic(RValue<Pointer<UInt>> x, RValue<UInt> y, std::memory_order memoryOrder);
1276 RValue<Int> MinAtomic(RValue<Pointer<Int>> x, RValue<Int> y, std::memory_order memoryOrder);
1277 RValue<Int> MaxAtomic(RValue<Pointer<Int>> x, RValue<Int> y, std::memory_order memoryOrder);
1278 RValue<UInt> MinAtomic(RValue<Pointer<UInt>> x, RValue<UInt> y, std::memory_order memoryOrder);
1279 RValue<UInt> MaxAtomic(RValue<Pointer<UInt>> x, RValue<UInt> y, std::memory_order memoryOrder);
1280 RValue<UInt> ExchangeAtomic(RValue<Pointer<UInt>> x, RValue<UInt> y, std::memory_order memoryOrder);
Chris Forbesa16238d2019-04-18 16:31:54 -07001281 RValue<UInt> CompareExchangeAtomic(RValue<Pointer<UInt>> x, RValue<UInt> y, RValue<UInt> compare, std::memory_order memoryOrderEqual, std::memory_order memoryOrderUnequal);
Chris Forbes17813932019-04-18 11:45:54 -07001282
Nicolas Capensd022e412016-09-26 13:30:14 -04001283// RValue<UInt> RoundUInt(RValue<Float> cast);
1284
Nicolas Capens297d26e2016-11-18 12:52:17 -05001285 class Int2 : public LValue<Int2>
Nicolas Capensd022e412016-09-26 13:30:14 -04001286 {
1287 public:
1288 // explicit Int2(RValue<Int> cast);
1289 explicit Int2(RValue<Int4> cast);
1290
Nicolas Capensa25311a2017-01-16 17:19:00 -05001291 Int2() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -04001292 Int2(int x, int y);
1293 Int2(RValue<Int2> rhs);
1294 Int2(const Int2 &rhs);
1295 Int2(const Reference<Int2> &rhs);
1296 Int2(RValue<Int> lo, RValue<Int> hi);
1297
Nicolas Capens96d4e092016-11-18 14:22:38 -05001298 RValue<Int2> operator=(RValue<Int2> rhs);
1299 RValue<Int2> operator=(const Int2 &rhs);
1300 RValue<Int2> operator=(const Reference<Int2> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001301
1302 static Type *getType();
1303 };
1304
1305 RValue<Int2> operator+(RValue<Int2> lhs, RValue<Int2> rhs);
1306 RValue<Int2> operator-(RValue<Int2> lhs, RValue<Int2> rhs);
1307// RValue<Int2> operator*(RValue<Int2> lhs, RValue<Int2> rhs);
1308// RValue<Int2> operator/(RValue<Int2> lhs, RValue<Int2> rhs);
1309// RValue<Int2> operator%(RValue<Int2> lhs, RValue<Int2> rhs);
1310 RValue<Int2> operator&(RValue<Int2> lhs, RValue<Int2> rhs);
1311 RValue<Int2> operator|(RValue<Int2> lhs, RValue<Int2> rhs);
1312 RValue<Int2> operator^(RValue<Int2> lhs, RValue<Int2> rhs);
1313 RValue<Int2> operator<<(RValue<Int2> lhs, unsigned char rhs);
1314 RValue<Int2> operator>>(RValue<Int2> lhs, unsigned char rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001315 RValue<Int2> operator+=(Int2 &lhs, RValue<Int2> rhs);
1316 RValue<Int2> operator-=(Int2 &lhs, RValue<Int2> rhs);
1317// RValue<Int2> operator*=(Int2 &lhs, RValue<Int2> rhs);
1318// RValue<Int2> operator/=(Int2 &lhs, RValue<Int2> rhs);
1319// RValue<Int2> operator%=(Int2 &lhs, RValue<Int2> rhs);
1320 RValue<Int2> operator&=(Int2 &lhs, RValue<Int2> rhs);
1321 RValue<Int2> operator|=(Int2 &lhs, RValue<Int2> rhs);
1322 RValue<Int2> operator^=(Int2 &lhs, RValue<Int2> rhs);
1323 RValue<Int2> operator<<=(Int2 &lhs, unsigned char rhs);
1324 RValue<Int2> operator>>=(Int2 &lhs, unsigned char rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001325// RValue<Int2> operator+(RValue<Int2> val);
1326// RValue<Int2> operator-(RValue<Int2> val);
1327 RValue<Int2> operator~(RValue<Int2> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001328// RValue<Int2> operator++(Int2 &val, int); // Post-increment
1329// const Int2 &operator++(Int2 &val); // Pre-increment
1330// RValue<Int2> operator--(Int2 &val, int); // Post-decrement
1331// const Int2 &operator--(Int2 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -04001332// RValue<Bool> operator<(RValue<Int2> lhs, RValue<Int2> rhs);
1333// RValue<Bool> operator<=(RValue<Int2> lhs, RValue<Int2> rhs);
1334// RValue<Bool> operator>(RValue<Int2> lhs, RValue<Int2> rhs);
1335// RValue<Bool> operator>=(RValue<Int2> lhs, RValue<Int2> rhs);
1336// RValue<Bool> operator!=(RValue<Int2> lhs, RValue<Int2> rhs);
1337// RValue<Bool> operator==(RValue<Int2> lhs, RValue<Int2> rhs);
1338
1339// RValue<Int2> RoundInt(RValue<Float4> cast);
Nicolas Capens45f187a2016-12-02 15:30:56 -05001340 RValue<Short4> UnpackLow(RValue<Int2> x, RValue<Int2> y);
1341 RValue<Short4> UnpackHigh(RValue<Int2> x, RValue<Int2> y);
Nicolas Capensd022e412016-09-26 13:30:14 -04001342 RValue<Int> Extract(RValue<Int2> val, int i);
1343 RValue<Int2> Insert(RValue<Int2> val, RValue<Int> element, int i);
1344
Nicolas Capens297d26e2016-11-18 12:52:17 -05001345 class UInt2 : public LValue<UInt2>
Nicolas Capensd022e412016-09-26 13:30:14 -04001346 {
1347 public:
Nicolas Capensa25311a2017-01-16 17:19:00 -05001348 UInt2() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -04001349 UInt2(unsigned int x, unsigned int y);
1350 UInt2(RValue<UInt2> rhs);
1351 UInt2(const UInt2 &rhs);
1352 UInt2(const Reference<UInt2> &rhs);
1353
Nicolas Capens96d4e092016-11-18 14:22:38 -05001354 RValue<UInt2> operator=(RValue<UInt2> rhs);
1355 RValue<UInt2> operator=(const UInt2 &rhs);
1356 RValue<UInt2> operator=(const Reference<UInt2> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001357
1358 static Type *getType();
1359 };
1360
1361 RValue<UInt2> operator+(RValue<UInt2> lhs, RValue<UInt2> rhs);
1362 RValue<UInt2> operator-(RValue<UInt2> lhs, RValue<UInt2> rhs);
1363// RValue<UInt2> operator*(RValue<UInt2> lhs, RValue<UInt2> rhs);
1364// RValue<UInt2> operator/(RValue<UInt2> lhs, RValue<UInt2> rhs);
1365// RValue<UInt2> operator%(RValue<UInt2> lhs, RValue<UInt2> rhs);
1366 RValue<UInt2> operator&(RValue<UInt2> lhs, RValue<UInt2> rhs);
1367 RValue<UInt2> operator|(RValue<UInt2> lhs, RValue<UInt2> rhs);
1368 RValue<UInt2> operator^(RValue<UInt2> lhs, RValue<UInt2> rhs);
1369 RValue<UInt2> operator<<(RValue<UInt2> lhs, unsigned char rhs);
1370 RValue<UInt2> operator>>(RValue<UInt2> lhs, unsigned char rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001371 RValue<UInt2> operator+=(UInt2 &lhs, RValue<UInt2> rhs);
1372 RValue<UInt2> operator-=(UInt2 &lhs, RValue<UInt2> rhs);
1373// RValue<UInt2> operator*=(UInt2 &lhs, RValue<UInt2> rhs);
1374// RValue<UInt2> operator/=(UInt2 &lhs, RValue<UInt2> rhs);
1375// RValue<UInt2> operator%=(UInt2 &lhs, RValue<UInt2> rhs);
1376 RValue<UInt2> operator&=(UInt2 &lhs, RValue<UInt2> rhs);
1377 RValue<UInt2> operator|=(UInt2 &lhs, RValue<UInt2> rhs);
1378 RValue<UInt2> operator^=(UInt2 &lhs, RValue<UInt2> rhs);
1379 RValue<UInt2> operator<<=(UInt2 &lhs, unsigned char rhs);
1380 RValue<UInt2> operator>>=(UInt2 &lhs, unsigned char rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001381// RValue<UInt2> operator+(RValue<UInt2> val);
1382// RValue<UInt2> operator-(RValue<UInt2> val);
1383 RValue<UInt2> operator~(RValue<UInt2> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001384// RValue<UInt2> operator++(UInt2 &val, int); // Post-increment
1385// const UInt2 &operator++(UInt2 &val); // Pre-increment
1386// RValue<UInt2> operator--(UInt2 &val, int); // Post-decrement
1387// const UInt2 &operator--(UInt2 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -04001388// RValue<Bool> operator<(RValue<UInt2> lhs, RValue<UInt2> rhs);
1389// RValue<Bool> operator<=(RValue<UInt2> lhs, RValue<UInt2> rhs);
1390// RValue<Bool> operator>(RValue<UInt2> lhs, RValue<UInt2> rhs);
1391// RValue<Bool> operator>=(RValue<UInt2> lhs, RValue<UInt2> rhs);
1392// RValue<Bool> operator!=(RValue<UInt2> lhs, RValue<UInt2> rhs);
1393// RValue<Bool> operator==(RValue<UInt2> lhs, RValue<UInt2> rhs);
1394
1395// RValue<UInt2> RoundInt(RValue<Float4> cast);
1396
Nicolas Capenscb986762017-01-20 11:34:37 -05001397 template<class T>
1398 struct Scalar;
1399
1400 template<class Vector4>
1401 struct XYZW;
1402
1403 template<class Vector4, int T>
1404 class Swizzle2
1405 {
1406 friend Vector4;
1407
1408 public:
1409 operator RValue<Vector4>() const;
1410
1411 private:
1412 Vector4 *parent;
1413 };
1414
1415 template<class Vector4, int T>
1416 class Swizzle4
1417 {
1418 public:
1419 operator RValue<Vector4>() const;
1420
1421 private:
1422 Vector4 *parent;
1423 };
1424
1425 template<class Vector4, int T>
1426 class SwizzleMask4
1427 {
1428 friend XYZW<Vector4>;
1429
1430 public:
1431 operator RValue<Vector4>() const;
1432
1433 RValue<Vector4> operator=(RValue<Vector4> rhs);
1434 RValue<Vector4> operator=(RValue<typename Scalar<Vector4>::Type> rhs);
1435
1436 private:
1437 Vector4 *parent;
1438 };
1439
1440 template<>
1441 struct Scalar<Float4>
1442 {
1443 using Type = Float;
1444 };
1445
1446 template<>
1447 struct Scalar<Int4>
1448 {
1449 using Type = Int;
1450 };
1451
1452 template<>
1453 struct Scalar<UInt4>
1454 {
1455 using Type = UInt;
1456 };
1457
1458 template<class Vector4, int T>
1459 class SwizzleMask1
1460 {
1461 public:
1462 operator RValue<typename Scalar<Vector4>::Type>() const;
1463 operator RValue<Vector4>() const;
1464
1465 RValue<Vector4> operator=(float x);
1466 RValue<Vector4> operator=(RValue<Vector4> rhs);
1467 RValue<Vector4> operator=(RValue<typename Scalar<Vector4>::Type> rhs);
1468
1469 private:
1470 Float4 *parent;
1471 };
1472
1473 template<class Vector4, int T>
1474 class SwizzleMask2
1475 {
1476 friend class Float4;
1477
1478 public:
1479 operator RValue<Vector4>() const;
1480
1481 RValue<Vector4> operator=(RValue<Vector4> rhs);
1482
1483 private:
1484 Float4 *parent;
1485 };
1486
1487 template<class Vector4>
1488 struct XYZW
1489 {
1490 friend Vector4;
1491
1492 private:
1493 XYZW(Vector4 *parent)
1494 {
1495 xyzw.parent = parent;
1496 }
1497
1498 public:
1499 union
1500 {
1501 SwizzleMask1<Vector4, 0x00> x;
1502 SwizzleMask1<Vector4, 0x55> y;
1503 SwizzleMask1<Vector4, 0xAA> z;
1504 SwizzleMask1<Vector4, 0xFF> w;
1505 Swizzle2<Vector4, 0x00> xx;
1506 Swizzle2<Vector4, 0x01> yx;
1507 Swizzle2<Vector4, 0x02> zx;
1508 Swizzle2<Vector4, 0x03> wx;
1509 SwizzleMask2<Vector4, 0x54> xy;
1510 Swizzle2<Vector4, 0x55> yy;
1511 Swizzle2<Vector4, 0x56> zy;
1512 Swizzle2<Vector4, 0x57> wy;
1513 SwizzleMask2<Vector4, 0xA8> xz;
1514 SwizzleMask2<Vector4, 0xA9> yz;
1515 Swizzle2<Vector4, 0xAA> zz;
1516 Swizzle2<Vector4, 0xAB> wz;
1517 SwizzleMask2<Vector4, 0xFC> xw;
1518 SwizzleMask2<Vector4, 0xFD> yw;
1519 SwizzleMask2<Vector4, 0xFE> zw;
1520 Swizzle2<Vector4, 0xFF> ww;
1521 Swizzle4<Vector4, 0x00> xxx;
1522 Swizzle4<Vector4, 0x01> yxx;
1523 Swizzle4<Vector4, 0x02> zxx;
1524 Swizzle4<Vector4, 0x03> wxx;
1525 Swizzle4<Vector4, 0x04> xyx;
1526 Swizzle4<Vector4, 0x05> yyx;
1527 Swizzle4<Vector4, 0x06> zyx;
1528 Swizzle4<Vector4, 0x07> wyx;
1529 Swizzle4<Vector4, 0x08> xzx;
1530 Swizzle4<Vector4, 0x09> yzx;
1531 Swizzle4<Vector4, 0x0A> zzx;
1532 Swizzle4<Vector4, 0x0B> wzx;
1533 Swizzle4<Vector4, 0x0C> xwx;
1534 Swizzle4<Vector4, 0x0D> ywx;
1535 Swizzle4<Vector4, 0x0E> zwx;
1536 Swizzle4<Vector4, 0x0F> wwx;
1537 Swizzle4<Vector4, 0x50> xxy;
1538 Swizzle4<Vector4, 0x51> yxy;
1539 Swizzle4<Vector4, 0x52> zxy;
1540 Swizzle4<Vector4, 0x53> wxy;
1541 Swizzle4<Vector4, 0x54> xyy;
1542 Swizzle4<Vector4, 0x55> yyy;
1543 Swizzle4<Vector4, 0x56> zyy;
1544 Swizzle4<Vector4, 0x57> wyy;
1545 Swizzle4<Vector4, 0x58> xzy;
1546 Swizzle4<Vector4, 0x59> yzy;
1547 Swizzle4<Vector4, 0x5A> zzy;
1548 Swizzle4<Vector4, 0x5B> wzy;
1549 Swizzle4<Vector4, 0x5C> xwy;
1550 Swizzle4<Vector4, 0x5D> ywy;
1551 Swizzle4<Vector4, 0x5E> zwy;
1552 Swizzle4<Vector4, 0x5F> wwy;
1553 Swizzle4<Vector4, 0xA0> xxz;
1554 Swizzle4<Vector4, 0xA1> yxz;
1555 Swizzle4<Vector4, 0xA2> zxz;
1556 Swizzle4<Vector4, 0xA3> wxz;
1557 SwizzleMask4<Vector4, 0xA4> xyz;
1558 Swizzle4<Vector4, 0xA5> yyz;
1559 Swizzle4<Vector4, 0xA6> zyz;
1560 Swizzle4<Vector4, 0xA7> wyz;
1561 Swizzle4<Vector4, 0xA8> xzz;
1562 Swizzle4<Vector4, 0xA9> yzz;
1563 Swizzle4<Vector4, 0xAA> zzz;
1564 Swizzle4<Vector4, 0xAB> wzz;
1565 Swizzle4<Vector4, 0xAC> xwz;
1566 Swizzle4<Vector4, 0xAD> ywz;
1567 Swizzle4<Vector4, 0xAE> zwz;
1568 Swizzle4<Vector4, 0xAF> wwz;
1569 Swizzle4<Vector4, 0xF0> xxw;
1570 Swizzle4<Vector4, 0xF1> yxw;
1571 Swizzle4<Vector4, 0xF2> zxw;
1572 Swizzle4<Vector4, 0xF3> wxw;
1573 SwizzleMask4<Vector4, 0xF4> xyw;
1574 Swizzle4<Vector4, 0xF5> yyw;
1575 Swizzle4<Vector4, 0xF6> zyw;
1576 Swizzle4<Vector4, 0xF7> wyw;
1577 SwizzleMask4<Vector4, 0xF8> xzw;
1578 SwizzleMask4<Vector4, 0xF9> yzw;
1579 Swizzle4<Vector4, 0xFA> zzw;
1580 Swizzle4<Vector4, 0xFB> wzw;
1581 Swizzle4<Vector4, 0xFC> xww;
1582 Swizzle4<Vector4, 0xFD> yww;
1583 Swizzle4<Vector4, 0xFE> zww;
1584 Swizzle4<Vector4, 0xFF> www;
1585 Swizzle4<Vector4, 0x00> xxxx;
1586 Swizzle4<Vector4, 0x01> yxxx;
1587 Swizzle4<Vector4, 0x02> zxxx;
1588 Swizzle4<Vector4, 0x03> wxxx;
1589 Swizzle4<Vector4, 0x04> xyxx;
1590 Swizzle4<Vector4, 0x05> yyxx;
1591 Swizzle4<Vector4, 0x06> zyxx;
1592 Swizzle4<Vector4, 0x07> wyxx;
1593 Swizzle4<Vector4, 0x08> xzxx;
1594 Swizzle4<Vector4, 0x09> yzxx;
1595 Swizzle4<Vector4, 0x0A> zzxx;
1596 Swizzle4<Vector4, 0x0B> wzxx;
1597 Swizzle4<Vector4, 0x0C> xwxx;
1598 Swizzle4<Vector4, 0x0D> ywxx;
1599 Swizzle4<Vector4, 0x0E> zwxx;
1600 Swizzle4<Vector4, 0x0F> wwxx;
1601 Swizzle4<Vector4, 0x10> xxyx;
1602 Swizzle4<Vector4, 0x11> yxyx;
1603 Swizzle4<Vector4, 0x12> zxyx;
1604 Swizzle4<Vector4, 0x13> wxyx;
1605 Swizzle4<Vector4, 0x14> xyyx;
1606 Swizzle4<Vector4, 0x15> yyyx;
1607 Swizzle4<Vector4, 0x16> zyyx;
1608 Swizzle4<Vector4, 0x17> wyyx;
1609 Swizzle4<Vector4, 0x18> xzyx;
1610 Swizzle4<Vector4, 0x19> yzyx;
1611 Swizzle4<Vector4, 0x1A> zzyx;
1612 Swizzle4<Vector4, 0x1B> wzyx;
1613 Swizzle4<Vector4, 0x1C> xwyx;
1614 Swizzle4<Vector4, 0x1D> ywyx;
1615 Swizzle4<Vector4, 0x1E> zwyx;
1616 Swizzle4<Vector4, 0x1F> wwyx;
1617 Swizzle4<Vector4, 0x20> xxzx;
1618 Swizzle4<Vector4, 0x21> yxzx;
1619 Swizzle4<Vector4, 0x22> zxzx;
1620 Swizzle4<Vector4, 0x23> wxzx;
1621 Swizzle4<Vector4, 0x24> xyzx;
1622 Swizzle4<Vector4, 0x25> yyzx;
1623 Swizzle4<Vector4, 0x26> zyzx;
1624 Swizzle4<Vector4, 0x27> wyzx;
1625 Swizzle4<Vector4, 0x28> xzzx;
1626 Swizzle4<Vector4, 0x29> yzzx;
1627 Swizzle4<Vector4, 0x2A> zzzx;
1628 Swizzle4<Vector4, 0x2B> wzzx;
1629 Swizzle4<Vector4, 0x2C> xwzx;
1630 Swizzle4<Vector4, 0x2D> ywzx;
1631 Swizzle4<Vector4, 0x2E> zwzx;
1632 Swizzle4<Vector4, 0x2F> wwzx;
1633 Swizzle4<Vector4, 0x30> xxwx;
1634 Swizzle4<Vector4, 0x31> yxwx;
1635 Swizzle4<Vector4, 0x32> zxwx;
1636 Swizzle4<Vector4, 0x33> wxwx;
1637 Swizzle4<Vector4, 0x34> xywx;
1638 Swizzle4<Vector4, 0x35> yywx;
1639 Swizzle4<Vector4, 0x36> zywx;
1640 Swizzle4<Vector4, 0x37> wywx;
1641 Swizzle4<Vector4, 0x38> xzwx;
1642 Swizzle4<Vector4, 0x39> yzwx;
1643 Swizzle4<Vector4, 0x3A> zzwx;
1644 Swizzle4<Vector4, 0x3B> wzwx;
1645 Swizzle4<Vector4, 0x3C> xwwx;
1646 Swizzle4<Vector4, 0x3D> ywwx;
1647 Swizzle4<Vector4, 0x3E> zwwx;
1648 Swizzle4<Vector4, 0x3F> wwwx;
1649 Swizzle4<Vector4, 0x40> xxxy;
1650 Swizzle4<Vector4, 0x41> yxxy;
1651 Swizzle4<Vector4, 0x42> zxxy;
1652 Swizzle4<Vector4, 0x43> wxxy;
1653 Swizzle4<Vector4, 0x44> xyxy;
1654 Swizzle4<Vector4, 0x45> yyxy;
1655 Swizzle4<Vector4, 0x46> zyxy;
1656 Swizzle4<Vector4, 0x47> wyxy;
1657 Swizzle4<Vector4, 0x48> xzxy;
1658 Swizzle4<Vector4, 0x49> yzxy;
1659 Swizzle4<Vector4, 0x4A> zzxy;
1660 Swizzle4<Vector4, 0x4B> wzxy;
1661 Swizzle4<Vector4, 0x4C> xwxy;
1662 Swizzle4<Vector4, 0x4D> ywxy;
1663 Swizzle4<Vector4, 0x4E> zwxy;
1664 Swizzle4<Vector4, 0x4F> wwxy;
1665 Swizzle4<Vector4, 0x50> xxyy;
1666 Swizzle4<Vector4, 0x51> yxyy;
1667 Swizzle4<Vector4, 0x52> zxyy;
1668 Swizzle4<Vector4, 0x53> wxyy;
1669 Swizzle4<Vector4, 0x54> xyyy;
1670 Swizzle4<Vector4, 0x55> yyyy;
1671 Swizzle4<Vector4, 0x56> zyyy;
1672 Swizzle4<Vector4, 0x57> wyyy;
1673 Swizzle4<Vector4, 0x58> xzyy;
1674 Swizzle4<Vector4, 0x59> yzyy;
1675 Swizzle4<Vector4, 0x5A> zzyy;
1676 Swizzle4<Vector4, 0x5B> wzyy;
1677 Swizzle4<Vector4, 0x5C> xwyy;
1678 Swizzle4<Vector4, 0x5D> ywyy;
1679 Swizzle4<Vector4, 0x5E> zwyy;
1680 Swizzle4<Vector4, 0x5F> wwyy;
1681 Swizzle4<Vector4, 0x60> xxzy;
1682 Swizzle4<Vector4, 0x61> yxzy;
1683 Swizzle4<Vector4, 0x62> zxzy;
1684 Swizzle4<Vector4, 0x63> wxzy;
1685 Swizzle4<Vector4, 0x64> xyzy;
1686 Swizzle4<Vector4, 0x65> yyzy;
1687 Swizzle4<Vector4, 0x66> zyzy;
1688 Swizzle4<Vector4, 0x67> wyzy;
1689 Swizzle4<Vector4, 0x68> xzzy;
1690 Swizzle4<Vector4, 0x69> yzzy;
1691 Swizzle4<Vector4, 0x6A> zzzy;
1692 Swizzle4<Vector4, 0x6B> wzzy;
1693 Swizzle4<Vector4, 0x6C> xwzy;
1694 Swizzle4<Vector4, 0x6D> ywzy;
1695 Swizzle4<Vector4, 0x6E> zwzy;
1696 Swizzle4<Vector4, 0x6F> wwzy;
1697 Swizzle4<Vector4, 0x70> xxwy;
1698 Swizzle4<Vector4, 0x71> yxwy;
1699 Swizzle4<Vector4, 0x72> zxwy;
1700 Swizzle4<Vector4, 0x73> wxwy;
1701 Swizzle4<Vector4, 0x74> xywy;
1702 Swizzle4<Vector4, 0x75> yywy;
1703 Swizzle4<Vector4, 0x76> zywy;
1704 Swizzle4<Vector4, 0x77> wywy;
1705 Swizzle4<Vector4, 0x78> xzwy;
1706 Swizzle4<Vector4, 0x79> yzwy;
1707 Swizzle4<Vector4, 0x7A> zzwy;
1708 Swizzle4<Vector4, 0x7B> wzwy;
1709 Swizzle4<Vector4, 0x7C> xwwy;
1710 Swizzle4<Vector4, 0x7D> ywwy;
1711 Swizzle4<Vector4, 0x7E> zwwy;
1712 Swizzle4<Vector4, 0x7F> wwwy;
1713 Swizzle4<Vector4, 0x80> xxxz;
1714 Swizzle4<Vector4, 0x81> yxxz;
1715 Swizzle4<Vector4, 0x82> zxxz;
1716 Swizzle4<Vector4, 0x83> wxxz;
1717 Swizzle4<Vector4, 0x84> xyxz;
1718 Swizzle4<Vector4, 0x85> yyxz;
1719 Swizzle4<Vector4, 0x86> zyxz;
1720 Swizzle4<Vector4, 0x87> wyxz;
1721 Swizzle4<Vector4, 0x88> xzxz;
1722 Swizzle4<Vector4, 0x89> yzxz;
1723 Swizzle4<Vector4, 0x8A> zzxz;
1724 Swizzle4<Vector4, 0x8B> wzxz;
1725 Swizzle4<Vector4, 0x8C> xwxz;
1726 Swizzle4<Vector4, 0x8D> ywxz;
1727 Swizzle4<Vector4, 0x8E> zwxz;
1728 Swizzle4<Vector4, 0x8F> wwxz;
1729 Swizzle4<Vector4, 0x90> xxyz;
1730 Swizzle4<Vector4, 0x91> yxyz;
1731 Swizzle4<Vector4, 0x92> zxyz;
1732 Swizzle4<Vector4, 0x93> wxyz;
1733 Swizzle4<Vector4, 0x94> xyyz;
1734 Swizzle4<Vector4, 0x95> yyyz;
1735 Swizzle4<Vector4, 0x96> zyyz;
1736 Swizzle4<Vector4, 0x97> wyyz;
1737 Swizzle4<Vector4, 0x98> xzyz;
1738 Swizzle4<Vector4, 0x99> yzyz;
1739 Swizzle4<Vector4, 0x9A> zzyz;
1740 Swizzle4<Vector4, 0x9B> wzyz;
1741 Swizzle4<Vector4, 0x9C> xwyz;
1742 Swizzle4<Vector4, 0x9D> ywyz;
1743 Swizzle4<Vector4, 0x9E> zwyz;
1744 Swizzle4<Vector4, 0x9F> wwyz;
1745 Swizzle4<Vector4, 0xA0> xxzz;
1746 Swizzle4<Vector4, 0xA1> yxzz;
1747 Swizzle4<Vector4, 0xA2> zxzz;
1748 Swizzle4<Vector4, 0xA3> wxzz;
1749 Swizzle4<Vector4, 0xA4> xyzz;
1750 Swizzle4<Vector4, 0xA5> yyzz;
1751 Swizzle4<Vector4, 0xA6> zyzz;
1752 Swizzle4<Vector4, 0xA7> wyzz;
1753 Swizzle4<Vector4, 0xA8> xzzz;
1754 Swizzle4<Vector4, 0xA9> yzzz;
1755 Swizzle4<Vector4, 0xAA> zzzz;
1756 Swizzle4<Vector4, 0xAB> wzzz;
1757 Swizzle4<Vector4, 0xAC> xwzz;
1758 Swizzle4<Vector4, 0xAD> ywzz;
1759 Swizzle4<Vector4, 0xAE> zwzz;
1760 Swizzle4<Vector4, 0xAF> wwzz;
1761 Swizzle4<Vector4, 0xB0> xxwz;
1762 Swizzle4<Vector4, 0xB1> yxwz;
1763 Swizzle4<Vector4, 0xB2> zxwz;
1764 Swizzle4<Vector4, 0xB3> wxwz;
1765 Swizzle4<Vector4, 0xB4> xywz;
1766 Swizzle4<Vector4, 0xB5> yywz;
1767 Swizzle4<Vector4, 0xB6> zywz;
1768 Swizzle4<Vector4, 0xB7> wywz;
1769 Swizzle4<Vector4, 0xB8> xzwz;
1770 Swizzle4<Vector4, 0xB9> yzwz;
1771 Swizzle4<Vector4, 0xBA> zzwz;
1772 Swizzle4<Vector4, 0xBB> wzwz;
1773 Swizzle4<Vector4, 0xBC> xwwz;
1774 Swizzle4<Vector4, 0xBD> ywwz;
1775 Swizzle4<Vector4, 0xBE> zwwz;
1776 Swizzle4<Vector4, 0xBF> wwwz;
1777 Swizzle4<Vector4, 0xC0> xxxw;
1778 Swizzle4<Vector4, 0xC1> yxxw;
1779 Swizzle4<Vector4, 0xC2> zxxw;
1780 Swizzle4<Vector4, 0xC3> wxxw;
1781 Swizzle4<Vector4, 0xC4> xyxw;
1782 Swizzle4<Vector4, 0xC5> yyxw;
1783 Swizzle4<Vector4, 0xC6> zyxw;
1784 Swizzle4<Vector4, 0xC7> wyxw;
1785 Swizzle4<Vector4, 0xC8> xzxw;
1786 Swizzle4<Vector4, 0xC9> yzxw;
1787 Swizzle4<Vector4, 0xCA> zzxw;
1788 Swizzle4<Vector4, 0xCB> wzxw;
1789 Swizzle4<Vector4, 0xCC> xwxw;
1790 Swizzle4<Vector4, 0xCD> ywxw;
1791 Swizzle4<Vector4, 0xCE> zwxw;
1792 Swizzle4<Vector4, 0xCF> wwxw;
1793 Swizzle4<Vector4, 0xD0> xxyw;
1794 Swizzle4<Vector4, 0xD1> yxyw;
1795 Swizzle4<Vector4, 0xD2> zxyw;
1796 Swizzle4<Vector4, 0xD3> wxyw;
1797 Swizzle4<Vector4, 0xD4> xyyw;
1798 Swizzle4<Vector4, 0xD5> yyyw;
1799 Swizzle4<Vector4, 0xD6> zyyw;
1800 Swizzle4<Vector4, 0xD7> wyyw;
1801 Swizzle4<Vector4, 0xD8> xzyw;
1802 Swizzle4<Vector4, 0xD9> yzyw;
1803 Swizzle4<Vector4, 0xDA> zzyw;
1804 Swizzle4<Vector4, 0xDB> wzyw;
1805 Swizzle4<Vector4, 0xDC> xwyw;
1806 Swizzle4<Vector4, 0xDD> ywyw;
1807 Swizzle4<Vector4, 0xDE> zwyw;
1808 Swizzle4<Vector4, 0xDF> wwyw;
1809 Swizzle4<Vector4, 0xE0> xxzw;
1810 Swizzle4<Vector4, 0xE1> yxzw;
1811 Swizzle4<Vector4, 0xE2> zxzw;
1812 Swizzle4<Vector4, 0xE3> wxzw;
1813 SwizzleMask4<Vector4, 0xE4> xyzw;
1814 Swizzle4<Vector4, 0xE5> yyzw;
1815 Swizzle4<Vector4, 0xE6> zyzw;
1816 Swizzle4<Vector4, 0xE7> wyzw;
1817 Swizzle4<Vector4, 0xE8> xzzw;
1818 Swizzle4<Vector4, 0xE9> yzzw;
1819 Swizzle4<Vector4, 0xEA> zzzw;
1820 Swizzle4<Vector4, 0xEB> wzzw;
1821 Swizzle4<Vector4, 0xEC> xwzw;
1822 Swizzle4<Vector4, 0xED> ywzw;
1823 Swizzle4<Vector4, 0xEE> zwzw;
1824 Swizzle4<Vector4, 0xEF> wwzw;
1825 Swizzle4<Vector4, 0xF0> xxww;
1826 Swizzle4<Vector4, 0xF1> yxww;
1827 Swizzle4<Vector4, 0xF2> zxww;
1828 Swizzle4<Vector4, 0xF3> wxww;
1829 Swizzle4<Vector4, 0xF4> xyww;
1830 Swizzle4<Vector4, 0xF5> yyww;
1831 Swizzle4<Vector4, 0xF6> zyww;
1832 Swizzle4<Vector4, 0xF7> wyww;
1833 Swizzle4<Vector4, 0xF8> xzww;
1834 Swizzle4<Vector4, 0xF9> yzww;
1835 Swizzle4<Vector4, 0xFA> zzww;
1836 Swizzle4<Vector4, 0xFB> wzww;
1837 Swizzle4<Vector4, 0xFC> xwww;
1838 Swizzle4<Vector4, 0xFD> ywww;
1839 Swizzle4<Vector4, 0xFE> zwww;
1840 Swizzle4<Vector4, 0xFF> wwww;
1841 };
1842 };
1843
1844 class Int4 : public LValue<Int4>, public XYZW<Int4>
Nicolas Capensd022e412016-09-26 13:30:14 -04001845 {
1846 public:
1847 explicit Int4(RValue<Byte4> cast);
1848 explicit Int4(RValue<SByte4> cast);
1849 explicit Int4(RValue<Float4> cast);
1850 explicit Int4(RValue<Short4> cast);
1851 explicit Int4(RValue<UShort4> cast);
1852
Nicolas Capenscb986762017-01-20 11:34:37 -05001853 Int4();
Nicolas Capensd022e412016-09-26 13:30:14 -04001854 Int4(int xyzw);
1855 Int4(int x, int yzw);
1856 Int4(int x, int y, int zw);
1857 Int4(int x, int y, int z, int w);
1858 Int4(RValue<Int4> rhs);
1859 Int4(const Int4 &rhs);
1860 Int4(const Reference<Int4> &rhs);
1861 Int4(RValue<UInt4> rhs);
1862 Int4(const UInt4 &rhs);
1863 Int4(const Reference<UInt4> &rhs);
1864 Int4(RValue<Int2> lo, RValue<Int2> hi);
1865 Int4(RValue<Int> rhs);
1866 Int4(const Int &rhs);
1867 Int4(const Reference<Int> &rhs);
1868
Nicolas Capens96d4e092016-11-18 14:22:38 -05001869 RValue<Int4> operator=(RValue<Int4> rhs);
1870 RValue<Int4> operator=(const Int4 &rhs);
1871 RValue<Int4> operator=(const Reference<Int4> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001872
1873 static Type *getType();
1874
1875 private:
1876 void constant(int x, int y, int z, int w);
1877 };
1878
1879 RValue<Int4> operator+(RValue<Int4> lhs, RValue<Int4> rhs);
1880 RValue<Int4> operator-(RValue<Int4> lhs, RValue<Int4> rhs);
1881 RValue<Int4> operator*(RValue<Int4> lhs, RValue<Int4> rhs);
1882 RValue<Int4> operator/(RValue<Int4> lhs, RValue<Int4> rhs);
1883 RValue<Int4> operator%(RValue<Int4> lhs, RValue<Int4> rhs);
1884 RValue<Int4> operator&(RValue<Int4> lhs, RValue<Int4> rhs);
1885 RValue<Int4> operator|(RValue<Int4> lhs, RValue<Int4> rhs);
1886 RValue<Int4> operator^(RValue<Int4> lhs, RValue<Int4> rhs);
1887 RValue<Int4> operator<<(RValue<Int4> lhs, unsigned char rhs);
1888 RValue<Int4> operator>>(RValue<Int4> lhs, unsigned char rhs);
1889 RValue<Int4> operator<<(RValue<Int4> lhs, RValue<Int4> rhs);
1890 RValue<Int4> operator>>(RValue<Int4> lhs, RValue<Int4> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001891 RValue<Int4> operator+=(Int4 &lhs, RValue<Int4> rhs);
1892 RValue<Int4> operator-=(Int4 &lhs, RValue<Int4> rhs);
1893 RValue<Int4> operator*=(Int4 &lhs, RValue<Int4> rhs);
1894// RValue<Int4> operator/=(Int4 &lhs, RValue<Int4> rhs);
1895// RValue<Int4> operator%=(Int4 &lhs, RValue<Int4> rhs);
1896 RValue<Int4> operator&=(Int4 &lhs, RValue<Int4> rhs);
1897 RValue<Int4> operator|=(Int4 &lhs, RValue<Int4> rhs);
1898 RValue<Int4> operator^=(Int4 &lhs, RValue<Int4> rhs);
1899 RValue<Int4> operator<<=(Int4 &lhs, unsigned char rhs);
1900 RValue<Int4> operator>>=(Int4 &lhs, unsigned char rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001901 RValue<Int4> operator+(RValue<Int4> val);
1902 RValue<Int4> operator-(RValue<Int4> val);
1903 RValue<Int4> operator~(RValue<Int4> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001904// RValue<Int4> operator++(Int4 &val, int); // Post-increment
1905// const Int4 &operator++(Int4 &val); // Pre-increment
1906// RValue<Int4> operator--(Int4 &val, int); // Post-decrement
1907// const Int4 &operator--(Int4 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -04001908// RValue<Bool> operator<(RValue<Int4> lhs, RValue<Int4> rhs);
1909// RValue<Bool> operator<=(RValue<Int4> lhs, RValue<Int4> rhs);
1910// RValue<Bool> operator>(RValue<Int4> lhs, RValue<Int4> rhs);
1911// RValue<Bool> operator>=(RValue<Int4> lhs, RValue<Int4> rhs);
1912// RValue<Bool> operator!=(RValue<Int4> lhs, RValue<Int4> rhs);
1913// RValue<Bool> operator==(RValue<Int4> lhs, RValue<Int4> rhs);
1914
Nicolas Capens4b743732018-05-28 13:22:07 -04001915 inline RValue<Int4> operator+(RValue<Int> lhs, RValue<Int4> rhs)
1916 {
1917 return Int4(lhs) + rhs;
1918 }
1919
1920 inline RValue<Int4> operator+(RValue<Int4> lhs, RValue<Int> rhs)
1921 {
1922 return lhs + Int4(rhs);
1923 }
1924
Nicolas Capensd022e412016-09-26 13:30:14 -04001925 RValue<Int4> CmpEQ(RValue<Int4> x, RValue<Int4> y);
1926 RValue<Int4> CmpLT(RValue<Int4> x, RValue<Int4> y);
1927 RValue<Int4> CmpLE(RValue<Int4> x, RValue<Int4> y);
1928 RValue<Int4> CmpNEQ(RValue<Int4> x, RValue<Int4> y);
1929 RValue<Int4> CmpNLT(RValue<Int4> x, RValue<Int4> y);
1930 RValue<Int4> CmpNLE(RValue<Int4> x, RValue<Int4> y);
Ben Claytone95eeb12019-03-04 16:32:09 +00001931 inline RValue<Int4> CmpGT(RValue<Int4> x, RValue<Int4> y) { return CmpNLE(x, y); }
1932 inline RValue<Int4> CmpGE(RValue<Int4> x, RValue<Int4> y) { return CmpNLT(x, y); }
Nicolas Capensd022e412016-09-26 13:30:14 -04001933 RValue<Int4> Max(RValue<Int4> x, RValue<Int4> y);
1934 RValue<Int4> Min(RValue<Int4> x, RValue<Int4> y);
1935 RValue<Int4> RoundInt(RValue<Float4> cast);
Nicolas Capens33438a62017-09-27 11:47:35 -04001936 RValue<Short8> PackSigned(RValue<Int4> x, RValue<Int4> y);
1937 RValue<UShort8> PackUnsigned(RValue<Int4> x, RValue<Int4> y);
Nicolas Capensc94ab742016-11-08 15:15:31 -05001938 RValue<Int> Extract(RValue<Int4> val, int i);
Nicolas Capensd022e412016-09-26 13:30:14 -04001939 RValue<Int4> Insert(RValue<Int4> val, RValue<Int> element, int i);
1940 RValue<Int> SignMask(RValue<Int4> x);
1941 RValue<Int4> Swizzle(RValue<Int4> x, unsigned char select);
Chris Forbese86b6dc2019-03-01 09:08:47 -08001942 RValue<Int4> MulHigh(RValue<Int4> x, RValue<Int4> y);
Nicolas Capensd022e412016-09-26 13:30:14 -04001943
Nicolas Capenscb986762017-01-20 11:34:37 -05001944 class UInt4 : public LValue<UInt4>, public XYZW<UInt4>
Nicolas Capensd022e412016-09-26 13:30:14 -04001945 {
1946 public:
1947 explicit UInt4(RValue<Float4> cast);
1948
Nicolas Capenscb986762017-01-20 11:34:37 -05001949 UInt4();
Nicolas Capensd022e412016-09-26 13:30:14 -04001950 UInt4(int xyzw);
1951 UInt4(int x, int yzw);
1952 UInt4(int x, int y, int zw);
1953 UInt4(int x, int y, int z, int w);
Nicolas Capensd022e412016-09-26 13:30:14 -04001954 UInt4(RValue<UInt4> rhs);
1955 UInt4(const UInt4 &rhs);
1956 UInt4(const Reference<UInt4> &rhs);
1957 UInt4(RValue<Int4> rhs);
1958 UInt4(const Int4 &rhs);
1959 UInt4(const Reference<Int4> &rhs);
1960 UInt4(RValue<UInt2> lo, RValue<UInt2> hi);
1961
Nicolas Capens96d4e092016-11-18 14:22:38 -05001962 RValue<UInt4> operator=(RValue<UInt4> rhs);
1963 RValue<UInt4> operator=(const UInt4 &rhs);
1964 RValue<UInt4> operator=(const Reference<UInt4> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001965
1966 static Type *getType();
1967
1968 private:
1969 void constant(int x, int y, int z, int w);
1970 };
1971
1972 RValue<UInt4> operator+(RValue<UInt4> lhs, RValue<UInt4> rhs);
1973 RValue<UInt4> operator-(RValue<UInt4> lhs, RValue<UInt4> rhs);
1974 RValue<UInt4> operator*(RValue<UInt4> lhs, RValue<UInt4> rhs);
1975 RValue<UInt4> operator/(RValue<UInt4> lhs, RValue<UInt4> rhs);
1976 RValue<UInt4> operator%(RValue<UInt4> lhs, RValue<UInt4> rhs);
1977 RValue<UInt4> operator&(RValue<UInt4> lhs, RValue<UInt4> rhs);
1978 RValue<UInt4> operator|(RValue<UInt4> lhs, RValue<UInt4> rhs);
1979 RValue<UInt4> operator^(RValue<UInt4> lhs, RValue<UInt4> rhs);
1980 RValue<UInt4> operator<<(RValue<UInt4> lhs, unsigned char rhs);
1981 RValue<UInt4> operator>>(RValue<UInt4> lhs, unsigned char rhs);
1982 RValue<UInt4> operator<<(RValue<UInt4> lhs, RValue<UInt4> rhs);
1983 RValue<UInt4> operator>>(RValue<UInt4> lhs, RValue<UInt4> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001984 RValue<UInt4> operator+=(UInt4 &lhs, RValue<UInt4> rhs);
1985 RValue<UInt4> operator-=(UInt4 &lhs, RValue<UInt4> rhs);
1986 RValue<UInt4> operator*=(UInt4 &lhs, RValue<UInt4> rhs);
1987// RValue<UInt4> operator/=(UInt4 &lhs, RValue<UInt4> rhs);
1988// RValue<UInt4> operator%=(UInt4 &lhs, RValue<UInt4> rhs);
1989 RValue<UInt4> operator&=(UInt4 &lhs, RValue<UInt4> rhs);
1990 RValue<UInt4> operator|=(UInt4 &lhs, RValue<UInt4> rhs);
1991 RValue<UInt4> operator^=(UInt4 &lhs, RValue<UInt4> rhs);
1992 RValue<UInt4> operator<<=(UInt4 &lhs, unsigned char rhs);
1993 RValue<UInt4> operator>>=(UInt4 &lhs, unsigned char rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04001994 RValue<UInt4> operator+(RValue<UInt4> val);
1995 RValue<UInt4> operator-(RValue<UInt4> val);
1996 RValue<UInt4> operator~(RValue<UInt4> val);
Nicolas Capens96d4e092016-11-18 14:22:38 -05001997// RValue<UInt4> operator++(UInt4 &val, int); // Post-increment
1998// const UInt4 &operator++(UInt4 &val); // Pre-increment
1999// RValue<UInt4> operator--(UInt4 &val, int); // Post-decrement
2000// const UInt4 &operator--(UInt4 &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -04002001// RValue<Bool> operator<(RValue<UInt4> lhs, RValue<UInt4> rhs);
2002// RValue<Bool> operator<=(RValue<UInt4> lhs, RValue<UInt4> rhs);
2003// RValue<Bool> operator>(RValue<UInt4> lhs, RValue<UInt4> rhs);
2004// RValue<Bool> operator>=(RValue<UInt4> lhs, RValue<UInt4> rhs);
2005// RValue<Bool> operator!=(RValue<UInt4> lhs, RValue<UInt4> rhs);
2006// RValue<Bool> operator==(RValue<UInt4> lhs, RValue<UInt4> rhs);
2007
2008 RValue<UInt4> CmpEQ(RValue<UInt4> x, RValue<UInt4> y);
2009 RValue<UInt4> CmpLT(RValue<UInt4> x, RValue<UInt4> y);
2010 RValue<UInt4> CmpLE(RValue<UInt4> x, RValue<UInt4> y);
2011 RValue<UInt4> CmpNEQ(RValue<UInt4> x, RValue<UInt4> y);
2012 RValue<UInt4> CmpNLT(RValue<UInt4> x, RValue<UInt4> y);
2013 RValue<UInt4> CmpNLE(RValue<UInt4> x, RValue<UInt4> y);
Ben Claytone95eeb12019-03-04 16:32:09 +00002014 inline RValue<UInt4> CmpGT(RValue<UInt4> x, RValue<UInt4> y) { return CmpNLE(x, y); }
2015 inline RValue<UInt4> CmpGE(RValue<UInt4> x, RValue<UInt4> y) { return CmpNLT(x, y); }
Nicolas Capensd022e412016-09-26 13:30:14 -04002016 RValue<UInt4> Max(RValue<UInt4> x, RValue<UInt4> y);
2017 RValue<UInt4> Min(RValue<UInt4> x, RValue<UInt4> y);
Chris Forbese86b6dc2019-03-01 09:08:47 -08002018 RValue<UInt4> MulHigh(RValue<UInt4> x, RValue<UInt4> y);
Ben Claytonfc77af12019-04-09 10:48:00 -04002019 RValue<UInt> Extract(RValue<UInt4> val, int i);
2020 RValue<UInt4> Insert(RValue<UInt4> val, RValue<UInt> element, int i);
Nicolas Capensd022e412016-09-26 13:30:14 -04002021// RValue<UInt4> RoundInt(RValue<Float4> cast);
Nicolas Capensd022e412016-09-26 13:30:14 -04002022
Alexis Hetu734e2572018-12-20 14:00:49 -05002023 class Half : public LValue<Half>
2024 {
2025 public:
2026 explicit Half(RValue<Float> cast);
2027
2028 static Type *getType();
2029 };
2030
Nicolas Capens297d26e2016-11-18 12:52:17 -05002031 class Float : public LValue<Float>
Nicolas Capensd022e412016-09-26 13:30:14 -04002032 {
2033 public:
2034 explicit Float(RValue<Int> cast);
Alexis Hetucfd96322017-07-24 14:44:33 -04002035 explicit Float(RValue<UInt> cast);
Alexis Hetu734e2572018-12-20 14:00:49 -05002036 explicit Float(RValue<Half> cast);
Nicolas Capensd022e412016-09-26 13:30:14 -04002037
Nicolas Capensa25311a2017-01-16 17:19:00 -05002038 Float() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -04002039 Float(float x);
2040 Float(RValue<Float> rhs);
2041 Float(const Float &rhs);
2042 Float(const Reference<Float> &rhs);
Ben Claytonf3b57972019-03-15 09:56:47 +00002043 Float(Argument<Float> argument);
Nicolas Capensd022e412016-09-26 13:30:14 -04002044
2045 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002046 Float(const SwizzleMask1<Float4, T> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04002047
Nicolas Capens96d4e092016-11-18 14:22:38 -05002048 // RValue<Float> operator=(float rhs); // FIXME: Implement
2049 RValue<Float> operator=(RValue<Float> rhs);
2050 RValue<Float> operator=(const Float &rhs);
2051 RValue<Float> operator=(const Reference<Float> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04002052
2053 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002054 RValue<Float> operator=(const SwizzleMask1<Float4, T> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04002055
2056 static Type *getType();
2057 };
2058
2059 RValue<Float> operator+(RValue<Float> lhs, RValue<Float> rhs);
2060 RValue<Float> operator-(RValue<Float> lhs, RValue<Float> rhs);
2061 RValue<Float> operator*(RValue<Float> lhs, RValue<Float> rhs);
2062 RValue<Float> operator/(RValue<Float> lhs, RValue<Float> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05002063 RValue<Float> operator+=(Float &lhs, RValue<Float> rhs);
2064 RValue<Float> operator-=(Float &lhs, RValue<Float> rhs);
2065 RValue<Float> operator*=(Float &lhs, RValue<Float> rhs);
2066 RValue<Float> operator/=(Float &lhs, RValue<Float> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04002067 RValue<Float> operator+(RValue<Float> val);
2068 RValue<Float> operator-(RValue<Float> val);
2069 RValue<Bool> operator<(RValue<Float> lhs, RValue<Float> rhs);
2070 RValue<Bool> operator<=(RValue<Float> lhs, RValue<Float> rhs);
2071 RValue<Bool> operator>(RValue<Float> lhs, RValue<Float> rhs);
2072 RValue<Bool> operator>=(RValue<Float> lhs, RValue<Float> rhs);
2073 RValue<Bool> operator!=(RValue<Float> lhs, RValue<Float> rhs);
2074 RValue<Bool> operator==(RValue<Float> lhs, RValue<Float> rhs);
2075
2076 RValue<Float> Abs(RValue<Float> x);
2077 RValue<Float> Max(RValue<Float> x, RValue<Float> y);
2078 RValue<Float> Min(RValue<Float> x, RValue<Float> y);
2079 RValue<Float> Rcp_pp(RValue<Float> val, bool exactAtPow2 = false);
2080 RValue<Float> RcpSqrt_pp(RValue<Float> val);
2081 RValue<Float> Sqrt(RValue<Float> x);
2082 RValue<Float> Round(RValue<Float> val);
2083 RValue<Float> Trunc(RValue<Float> val);
2084 RValue<Float> Frac(RValue<Float> val);
2085 RValue<Float> Floor(RValue<Float> val);
2086 RValue<Float> Ceil(RValue<Float> val);
2087
Nicolas Capens297d26e2016-11-18 12:52:17 -05002088 class Float2 : public LValue<Float2>
Nicolas Capensd022e412016-09-26 13:30:14 -04002089 {
2090 public:
2091 // explicit Float2(RValue<Byte2> cast);
2092 // explicit Float2(RValue<Short2> cast);
2093 // explicit Float2(RValue<UShort2> cast);
2094 // explicit Float2(RValue<Int2> cast);
2095 // explicit Float2(RValue<UInt2> cast);
2096 explicit Float2(RValue<Float4> cast);
2097
Nicolas Capensa25311a2017-01-16 17:19:00 -05002098 Float2() = default;
Nicolas Capensd022e412016-09-26 13:30:14 -04002099 // Float2(float x, float y);
2100 // Float2(RValue<Float2> rhs);
2101 // Float2(const Float2 &rhs);
2102 // Float2(const Reference<Float2> &rhs);
2103 // Float2(RValue<Float> rhs);
2104 // Float2(const Float &rhs);
2105 // Float2(const Reference<Float> &rhs);
2106
2107 // template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002108 // Float2(const SwizzleMask1<T> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04002109
Nicolas Capens96d4e092016-11-18 14:22:38 -05002110 // RValue<Float2> operator=(float replicate);
2111 // RValue<Float2> operator=(RValue<Float2> rhs);
2112 // RValue<Float2> operator=(const Float2 &rhs);
2113 // RValue<Float2> operator=(const Reference<Float2> &rhs);
2114 // RValue<Float2> operator=(RValue<Float> rhs);
2115 // RValue<Float2> operator=(const Float &rhs);
2116 // RValue<Float2> operator=(const Reference<Float> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04002117
2118 // template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002119 // RValue<Float2> operator=(const SwizzleMask1<T> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04002120
2121 static Type *getType();
2122 };
2123
2124// RValue<Float2> operator+(RValue<Float2> lhs, RValue<Float2> rhs);
2125// RValue<Float2> operator-(RValue<Float2> lhs, RValue<Float2> rhs);
2126// RValue<Float2> operator*(RValue<Float2> lhs, RValue<Float2> rhs);
2127// RValue<Float2> operator/(RValue<Float2> lhs, RValue<Float2> rhs);
2128// RValue<Float2> operator%(RValue<Float2> lhs, RValue<Float2> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05002129// RValue<Float2> operator+=(Float2 &lhs, RValue<Float2> rhs);
2130// RValue<Float2> operator-=(Float2 &lhs, RValue<Float2> rhs);
2131// RValue<Float2> operator*=(Float2 &lhs, RValue<Float2> rhs);
2132// RValue<Float2> operator/=(Float2 &lhs, RValue<Float2> rhs);
2133// RValue<Float2> operator%=(Float2 &lhs, RValue<Float2> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04002134// RValue<Float2> operator+(RValue<Float2> val);
2135// RValue<Float2> operator-(RValue<Float2> val);
2136
2137// RValue<Float2> Abs(RValue<Float2> x);
2138// RValue<Float2> Max(RValue<Float2> x, RValue<Float2> y);
2139// RValue<Float2> Min(RValue<Float2> x, RValue<Float2> y);
2140// RValue<Float2> Swizzle(RValue<Float2> x, unsigned char select);
2141// RValue<Float2> Mask(Float2 &lhs, RValue<Float2> rhs, unsigned char select);
2142
Nicolas Capenscb986762017-01-20 11:34:37 -05002143 class Float4 : public LValue<Float4>, public XYZW<Float4>
Nicolas Capensa25311a2017-01-16 17:19:00 -05002144 {
2145 public:
2146 explicit Float4(RValue<Byte4> cast);
2147 explicit Float4(RValue<SByte4> cast);
2148 explicit Float4(RValue<Short4> cast);
2149 explicit Float4(RValue<UShort4> cast);
2150 explicit Float4(RValue<Int4> cast);
2151 explicit Float4(RValue<UInt4> cast);
2152
2153 Float4();
2154 Float4(float xyzw);
2155 Float4(float x, float yzw);
2156 Float4(float x, float y, float zw);
2157 Float4(float x, float y, float z, float w);
2158 Float4(RValue<Float4> rhs);
2159 Float4(const Float4 &rhs);
2160 Float4(const Reference<Float4> &rhs);
2161 Float4(RValue<Float> rhs);
2162 Float4(const Float &rhs);
2163 Float4(const Reference<Float> &rhs);
2164
2165 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002166 Float4(const SwizzleMask1<Float4, T> &rhs);
Nicolas Capensa25311a2017-01-16 17:19:00 -05002167 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002168 Float4(const Swizzle4<Float4, T> &rhs);
Nicolas Capensa25311a2017-01-16 17:19:00 -05002169 template<int X, int Y>
Nicolas Capenscb986762017-01-20 11:34:37 -05002170 Float4(const Swizzle2<Float4, X> &x, const Swizzle2<Float4, Y> &y);
Nicolas Capensa25311a2017-01-16 17:19:00 -05002171 template<int X, int Y>
Nicolas Capenscb986762017-01-20 11:34:37 -05002172 Float4(const SwizzleMask2<Float4, X> &x, const Swizzle2<Float4, Y> &y);
Nicolas Capensa25311a2017-01-16 17:19:00 -05002173 template<int X, int Y>
Nicolas Capenscb986762017-01-20 11:34:37 -05002174 Float4(const Swizzle2<Float4, X> &x, const SwizzleMask2<Float4, Y> &y);
Nicolas Capensa25311a2017-01-16 17:19:00 -05002175 template<int X, int Y>
Nicolas Capenscb986762017-01-20 11:34:37 -05002176 Float4(const SwizzleMask2<Float4, X> &x, const SwizzleMask2<Float4, Y> &y);
Nicolas Capensa25311a2017-01-16 17:19:00 -05002177
2178 RValue<Float4> operator=(float replicate);
2179 RValue<Float4> operator=(RValue<Float4> rhs);
2180 RValue<Float4> operator=(const Float4 &rhs);
2181 RValue<Float4> operator=(const Reference<Float4> &rhs);
2182 RValue<Float4> operator=(RValue<Float> rhs);
2183 RValue<Float4> operator=(const Float &rhs);
2184 RValue<Float4> operator=(const Reference<Float> &rhs);
2185
2186 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002187 RValue<Float4> operator=(const SwizzleMask1<Float4, T> &rhs);
Nicolas Capensa25311a2017-01-16 17:19:00 -05002188 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002189 RValue<Float4> operator=(const Swizzle4<Float4, T> &rhs);
Nicolas Capensa25311a2017-01-16 17:19:00 -05002190
2191 static Type *getType();
Nicolas Capensd022e412016-09-26 13:30:14 -04002192
2193 private:
2194 void constant(float x, float y, float z, float w);
2195 };
2196
2197 RValue<Float4> operator+(RValue<Float4> lhs, RValue<Float4> rhs);
2198 RValue<Float4> operator-(RValue<Float4> lhs, RValue<Float4> rhs);
2199 RValue<Float4> operator*(RValue<Float4> lhs, RValue<Float4> rhs);
2200 RValue<Float4> operator/(RValue<Float4> lhs, RValue<Float4> rhs);
2201 RValue<Float4> operator%(RValue<Float4> lhs, RValue<Float4> rhs);
Nicolas Capens96d4e092016-11-18 14:22:38 -05002202 RValue<Float4> operator+=(Float4 &lhs, RValue<Float4> rhs);
2203 RValue<Float4> operator-=(Float4 &lhs, RValue<Float4> rhs);
2204 RValue<Float4> operator*=(Float4 &lhs, RValue<Float4> rhs);
2205 RValue<Float4> operator/=(Float4 &lhs, RValue<Float4> rhs);
2206 RValue<Float4> operator%=(Float4 &lhs, RValue<Float4> rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04002207 RValue<Float4> operator+(RValue<Float4> val);
2208 RValue<Float4> operator-(RValue<Float4> val);
2209
2210 RValue<Float4> Abs(RValue<Float4> x);
2211 RValue<Float4> Max(RValue<Float4> x, RValue<Float4> y);
2212 RValue<Float4> Min(RValue<Float4> x, RValue<Float4> y);
2213 RValue<Float4> Rcp_pp(RValue<Float4> val, bool exactAtPow2 = false);
2214 RValue<Float4> RcpSqrt_pp(RValue<Float4> val);
2215 RValue<Float4> Sqrt(RValue<Float4> x);
Nicolas Capensc94ab742016-11-08 15:15:31 -05002216 RValue<Float4> Insert(RValue<Float4> val, RValue<Float> element, int i);
Nicolas Capensd022e412016-09-26 13:30:14 -04002217 RValue<Float> Extract(RValue<Float4> x, int i);
2218 RValue<Float4> Swizzle(RValue<Float4> x, unsigned char select);
2219 RValue<Float4> ShuffleLowHigh(RValue<Float4> x, RValue<Float4> y, unsigned char imm);
2220 RValue<Float4> UnpackLow(RValue<Float4> x, RValue<Float4> y);
2221 RValue<Float4> UnpackHigh(RValue<Float4> x, RValue<Float4> y);
2222 RValue<Float4> Mask(Float4 &lhs, RValue<Float4> rhs, unsigned char select);
2223 RValue<Int> SignMask(RValue<Float4> x);
Ben Claytonec1aeb82019-03-04 19:33:27 +00002224
2225 // Ordered comparison functions
Nicolas Capensd022e412016-09-26 13:30:14 -04002226 RValue<Int4> CmpEQ(RValue<Float4> x, RValue<Float4> y);
2227 RValue<Int4> CmpLT(RValue<Float4> x, RValue<Float4> y);
2228 RValue<Int4> CmpLE(RValue<Float4> x, RValue<Float4> y);
2229 RValue<Int4> CmpNEQ(RValue<Float4> x, RValue<Float4> y);
2230 RValue<Int4> CmpNLT(RValue<Float4> x, RValue<Float4> y);
2231 RValue<Int4> CmpNLE(RValue<Float4> x, RValue<Float4> y);
Ben Claytonec1aeb82019-03-04 19:33:27 +00002232 inline RValue<Int4> CmpGT(RValue<Float4> x, RValue<Float4> y) { return CmpNLE(x, y); }
2233 inline RValue<Int4> CmpGE(RValue<Float4> x, RValue<Float4> y) { return CmpNLT(x, y); }
2234
2235 // Unordered comparison functions
2236 RValue<Int4> CmpUEQ(RValue<Float4> x, RValue<Float4> y);
2237 RValue<Int4> CmpULT(RValue<Float4> x, RValue<Float4> y);
2238 RValue<Int4> CmpULE(RValue<Float4> x, RValue<Float4> y);
2239 RValue<Int4> CmpUNEQ(RValue<Float4> x, RValue<Float4> y);
2240 RValue<Int4> CmpUNLT(RValue<Float4> x, RValue<Float4> y);
2241 RValue<Int4> CmpUNLE(RValue<Float4> x, RValue<Float4> y);
2242 inline RValue<Int4> CmpUGT(RValue<Float4> x, RValue<Float4> y) { return CmpUNLE(x, y); }
2243 inline RValue<Int4> CmpUGE(RValue<Float4> x, RValue<Float4> y) { return CmpUNLT(x, y); }
2244
Alexis Hetu8ef6d102017-11-09 15:49:09 -05002245 RValue<Int4> IsInf(RValue<Float4> x);
2246 RValue<Int4> IsNan(RValue<Float4> x);
Nicolas Capensd022e412016-09-26 13:30:14 -04002247 RValue<Float4> Round(RValue<Float4> x);
2248 RValue<Float4> Trunc(RValue<Float4> x);
2249 RValue<Float4> Frac(RValue<Float4> x);
2250 RValue<Float4> Floor(RValue<Float4> x);
2251 RValue<Float4> Ceil(RValue<Float4> x);
2252
Ben Claytona2c8b772019-04-09 13:42:36 -04002253 // Trigonometric functions
2254 // TODO: Currentlhy unimplemented for Subzero.
2255 RValue<Float4> Sin(RValue<Float4> x);
Ben Clayton1b6f8c72019-04-09 13:47:43 -04002256 RValue<Float4> Cos(RValue<Float4> x);
Ben Clayton14740062019-04-09 13:48:41 -04002257 RValue<Float4> Tan(RValue<Float4> x);
Ben Claytonf9350d72019-04-09 14:19:02 -04002258 RValue<Float4> Asin(RValue<Float4> x);
Ben Claytoneafae472019-04-09 14:22:38 -04002259 RValue<Float4> Acos(RValue<Float4> x);
Ben Clayton749b4e02019-04-09 14:27:43 -04002260 RValue<Float4> Atan(RValue<Float4> x);
Ben Claytond9636972019-04-09 15:09:54 -04002261 RValue<Float4> Sinh(RValue<Float4> x);
Ben Clayton900ea2c2019-04-09 15:25:36 -04002262 RValue<Float4> Cosh(RValue<Float4> x);
Ben Clayton3928bd92019-04-09 15:27:41 -04002263 RValue<Float4> Tanh(RValue<Float4> x);
Ben Claytonf6d77ab2019-04-09 15:30:04 -04002264 RValue<Float4> Asinh(RValue<Float4> x);
Ben Clayton28ebcb02019-04-09 15:33:38 -04002265 RValue<Float4> Acosh(RValue<Float4> x);
Ben Claytonfa6a5392019-04-09 15:35:24 -04002266 RValue<Float4> Atanh(RValue<Float4> x);
Ben Claytona520c3e2019-04-09 15:43:45 -04002267 RValue<Float4> Atan2(RValue<Float4> x, RValue<Float4> y);
Ben Claytona2c8b772019-04-09 13:42:36 -04002268
Ben Claytonbfe94f02019-04-09 15:52:12 -04002269 // Exponential functions
2270 // TODO: Currentlhy unimplemented for Subzero.
2271 RValue<Float4> Pow(RValue<Float4> x, RValue<Float4> y);
Ben Clayton242f0022019-04-09 16:00:53 -04002272 RValue<Float4> Exp(RValue<Float4> x);
Ben Clayton2c1da722019-04-09 16:03:03 -04002273 RValue<Float4> Log(RValue<Float4> x);
Ben Claytonf40b56c2019-04-09 16:06:55 -04002274 RValue<Float4> Exp2(RValue<Float4> x);
Ben Claytone17acfe2019-04-09 16:09:13 -04002275 RValue<Float4> Log2(RValue<Float4> x);
Ben Claytonbfe94f02019-04-09 15:52:12 -04002276
Ben Clayton60958262019-04-10 14:53:30 -04002277 // Bit Manipulation functions.
2278 // TODO: Currentlhy unimplemented for Subzero.
2279
2280 // Count leading zeros.
2281 // Returns 32 when: isZeroUndef && x == 0.
2282 // Returns an undefined value when: !isZeroUndef && x == 0.
2283 RValue<UInt4> Ctlz(RValue<UInt4> x, bool isZeroUndef);
2284
Ben Clayton3f007c42019-04-10 14:54:23 -04002285 // Count trailing zeros.
2286 // Returns 32 when: isZeroUndef && x == 0.
2287 // Returns an undefined value when: !isZeroUndef && x == 0.
2288 RValue<UInt4> Cttz(RValue<UInt4> x, bool isZeroUndef);
2289
Nicolas Capensd022e412016-09-26 13:30:14 -04002290 template<class T>
Nicolas Capens297d26e2016-11-18 12:52:17 -05002291 class Pointer : public LValue<Pointer<T>>
Nicolas Capensd022e412016-09-26 13:30:14 -04002292 {
2293 public:
2294 template<class S>
2295 Pointer(RValue<Pointer<S>> pointerS, int alignment = 1) : alignment(alignment)
2296 {
2297 Value *pointerT = Nucleus::createBitCast(pointerS.value, Nucleus::getPointerType(T::getType()));
Nicolas Capens22479eb2016-09-28 22:34:26 -04002298 LValue<Pointer<T>>::storeValue(pointerT);
Nicolas Capensd022e412016-09-26 13:30:14 -04002299 }
2300
2301 template<class S>
2302 Pointer(const Pointer<S> &pointer, int alignment = 1) : alignment(alignment)
2303 {
Nicolas Capens4126b8e2017-07-26 13:34:36 -04002304 Value *pointerS = pointer.loadValue();
Nicolas Capensd022e412016-09-26 13:30:14 -04002305 Value *pointerT = Nucleus::createBitCast(pointerS, Nucleus::getPointerType(T::getType()));
Nicolas Capens22479eb2016-09-28 22:34:26 -04002306 LValue<Pointer<T>>::storeValue(pointerT);
Nicolas Capensd022e412016-09-26 13:30:14 -04002307 }
2308
2309 Pointer(Argument<Pointer<T>> argument);
Nicolas Capensd022e412016-09-26 13:30:14 -04002310
2311 Pointer();
2312 Pointer(RValue<Pointer<T>> rhs);
2313 Pointer(const Pointer<T> &rhs);
2314 Pointer(const Reference<Pointer<T>> &rhs);
2315
Nicolas Capens96d4e092016-11-18 14:22:38 -05002316 RValue<Pointer<T>> operator=(RValue<Pointer<T>> rhs);
2317 RValue<Pointer<T>> operator=(const Pointer<T> &rhs);
2318 RValue<Pointer<T>> operator=(const Reference<Pointer<T>> &rhs);
Nicolas Capensd022e412016-09-26 13:30:14 -04002319
2320 Reference<T> operator*();
2321 Reference<T> operator[](int index);
Nicolas Capensd294def2017-01-26 17:44:37 -08002322 Reference<T> operator[](unsigned int index);
Nicolas Capensd022e412016-09-26 13:30:14 -04002323 Reference<T> operator[](RValue<Int> index);
Nicolas Capensd294def2017-01-26 17:44:37 -08002324 Reference<T> operator[](RValue<UInt> index);
Nicolas Capensd022e412016-09-26 13:30:14 -04002325
2326 static Type *getType();
2327
2328 private:
2329 const int alignment;
2330 };
2331
2332 RValue<Pointer<Byte>> operator+(RValue<Pointer<Byte>> lhs, int offset);
2333 RValue<Pointer<Byte>> operator+(RValue<Pointer<Byte>> lhs, RValue<Int> offset);
2334 RValue<Pointer<Byte>> operator+(RValue<Pointer<Byte>> lhs, RValue<UInt> offset);
Nicolas Capens96d4e092016-11-18 14:22:38 -05002335 RValue<Pointer<Byte>> operator+=(Pointer<Byte> &lhs, int offset);
2336 RValue<Pointer<Byte>> operator+=(Pointer<Byte> &lhs, RValue<Int> offset);
2337 RValue<Pointer<Byte>> operator+=(Pointer<Byte> &lhs, RValue<UInt> offset);
Nicolas Capensd022e412016-09-26 13:30:14 -04002338
2339 RValue<Pointer<Byte>> operator-(RValue<Pointer<Byte>> lhs, int offset);
2340 RValue<Pointer<Byte>> operator-(RValue<Pointer<Byte>> lhs, RValue<Int> offset);
2341 RValue<Pointer<Byte>> operator-(RValue<Pointer<Byte>> lhs, RValue<UInt> offset);
Nicolas Capens96d4e092016-11-18 14:22:38 -05002342 RValue<Pointer<Byte>> operator-=(Pointer<Byte> &lhs, int offset);
2343 RValue<Pointer<Byte>> operator-=(Pointer<Byte> &lhs, RValue<Int> offset);
2344 RValue<Pointer<Byte>> operator-=(Pointer<Byte> &lhs, RValue<UInt> offset);
Nicolas Capensd022e412016-09-26 13:30:14 -04002345
Nicolas Capens86509d92019-03-21 13:23:50 -04002346 template<typename T>
2347 RValue<T> Load(RValue<Pointer<T>> pointer, unsigned int alignment, bool atomic, std::memory_order memoryOrder)
2348 {
2349 return RValue<T>(Nucleus::createLoad(pointer.value, T::getType(), false, alignment, atomic, memoryOrder));
2350 }
2351
2352 template<typename T>
Ben Clayton97035bd2019-04-16 11:35:38 -04002353 RValue<T> Load(Pointer<T> pointer, unsigned int alignment, bool atomic, std::memory_order memoryOrder)
2354 {
2355 return Load(RValue<Pointer<T>>(pointer), alignment, atomic, memoryOrder);
2356 }
2357
2358 template<typename T>
Nicolas Capens86509d92019-03-21 13:23:50 -04002359 void Store(RValue<T> value, RValue<Pointer<T>> pointer, unsigned int alignment, bool atomic, std::memory_order memoryOrder)
2360 {
2361 Nucleus::createStore(value.value, pointer.value, T::getType(), false, alignment, atomic, memoryOrder);
2362 }
2363
Ben Clayton97035bd2019-04-16 11:35:38 -04002364 template<typename T>
2365 void Store(RValue<T> value, Pointer<T> pointer, unsigned int alignment, bool atomic, std::memory_order memoryOrder)
2366 {
2367 Store(value, RValue<Pointer<T>>(pointer), alignment, atomic, memoryOrder);
2368 }
2369
2370 template<typename T>
2371 void Store(T value, Pointer<T> pointer, unsigned int alignment, bool atomic, std::memory_order memoryOrder)
2372 {
2373 Store(RValue<T>(value), RValue<Pointer<T>>(pointer), alignment, atomic, memoryOrder);
2374 }
2375
2376
Nicolas Capensd022e412016-09-26 13:30:14 -04002377 template<class T, int S = 1>
Nicolas Capens297d26e2016-11-18 12:52:17 -05002378 class Array : public LValue<T>
Nicolas Capensd022e412016-09-26 13:30:14 -04002379 {
2380 public:
2381 Array(int size = S);
2382
2383 Reference<T> operator[](int index);
Nicolas Capensd294def2017-01-26 17:44:37 -08002384 Reference<T> operator[](unsigned int index);
Nicolas Capensd022e412016-09-26 13:30:14 -04002385 Reference<T> operator[](RValue<Int> index);
Nicolas Capensd294def2017-01-26 17:44:37 -08002386 Reference<T> operator[](RValue<UInt> index);
Nicolas Capensd022e412016-09-26 13:30:14 -04002387 };
2388
Nicolas Capens96d4e092016-11-18 14:22:38 -05002389// RValue<Array<T>> operator++(Array<T> &val, int); // Post-increment
2390// const Array<T> &operator++(Array<T> &val); // Pre-increment
2391// RValue<Array<T>> operator--(Array<T> &val, int); // Post-decrement
2392// const Array<T> &operator--(Array<T> &val); // Pre-decrement
Nicolas Capensd022e412016-09-26 13:30:14 -04002393
Nicolas Capensf4eec2f2017-05-24 15:46:48 -04002394 void branch(RValue<Bool> cmp, BasicBlock *bodyBB, BasicBlock *endBB);
Nicolas Capensd022e412016-09-26 13:30:14 -04002395
2396 void Return();
Nicolas Capenseb253d02016-11-18 14:40:40 -05002397 void Return(RValue<Int> ret);
Nicolas Capensd022e412016-09-26 13:30:14 -04002398
2399 template<class T>
2400 void Return(const Pointer<T> &ret);
2401
2402 template<class T>
2403 void Return(RValue<Pointer<T>> ret);
2404
Nicolas Capensd022e412016-09-26 13:30:14 -04002405 // Generic template, leave undefined!
2406 template<typename FunctionType>
2407 class Function;
2408
2409 // Specialized for function types
2410 template<typename Return, typename... Arguments>
2411 class Function<Return(Arguments...)>
2412 {
2413 public:
2414 Function();
2415
2416 virtual ~Function();
2417
2418 template<int index>
Ben Clayton169872e2019-02-27 23:58:35 +00002419 Argument<typename std::tuple_element<index, std::tuple<Arguments...>>::type> Arg() const
Nicolas Capensd022e412016-09-26 13:30:14 -04002420 {
2421 Value *arg = Nucleus::getArgument(index);
Ben Clayton169872e2019-02-27 23:58:35 +00002422 return Argument<typename std::tuple_element<index, std::tuple<Arguments...>>::type>(arg);
Nicolas Capensd022e412016-09-26 13:30:14 -04002423 }
2424
Chris Forbes878d4b02019-01-21 10:48:35 -08002425 Routine *operator()(const char *name, ...);
Nicolas Capensd022e412016-09-26 13:30:14 -04002426
2427 protected:
2428 Nucleus *core;
2429 std::vector<Type*> arguments;
2430 };
2431
2432 template<typename Return>
2433 class Function<Return()> : public Function<Return(Void)>
2434 {
2435 };
2436
Nicolas Capensd022e412016-09-26 13:30:14 -04002437 RValue<Long> Ticks();
2438}
2439
Nicolas Capens48461502018-08-06 14:20:45 -04002440namespace rr
Nicolas Capensd022e412016-09-26 13:30:14 -04002441{
2442 template<class T>
Nicolas Capens0192d152019-03-27 14:46:07 -04002443 LValue<T>::LValue(int arraySize) : Variable(T::getType(), arraySize)
Nicolas Capens22479eb2016-09-28 22:34:26 -04002444 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002445#ifdef ENABLE_RR_DEBUG_INFO
2446 materialize();
2447#endif // ENABLE_RR_DEBUG_INFO
Nicolas Capens22479eb2016-09-28 22:34:26 -04002448 }
2449
Nicolas Capens0192d152019-03-27 14:46:07 -04002450 inline void Variable::materialize() const
Nicolas Capens22479eb2016-09-28 22:34:26 -04002451 {
Nicolas Capens0192d152019-03-27 14:46:07 -04002452 if(!address)
2453 {
2454 address = Nucleus::allocateStackVariable(type, arraySize);
Ben Claytonac07ed82019-03-26 14:17:41 +00002455 RR_DEBUG_INFO_EMIT_VAR(address);
Nicolas Capens0192d152019-03-27 14:46:07 -04002456
2457 if(rvalue)
2458 {
2459 storeValue(rvalue);
2460 rvalue = nullptr;
2461 }
2462 }
Nicolas Capens22479eb2016-09-28 22:34:26 -04002463 }
2464
Nicolas Capens0192d152019-03-27 14:46:07 -04002465 inline Value *Variable::loadValue() const
Nicolas Capens22479eb2016-09-28 22:34:26 -04002466 {
Nicolas Capens0192d152019-03-27 14:46:07 -04002467 if(rvalue)
2468 {
2469 return rvalue;
2470 }
2471
2472 if(!address)
2473 {
2474 // TODO: Return undef instead.
2475 materialize();
2476 }
2477
2478 return Nucleus::createLoad(address, type, false, 0);
Nicolas Capens22479eb2016-09-28 22:34:26 -04002479 }
2480
Nicolas Capens0192d152019-03-27 14:46:07 -04002481 inline Value *Variable::storeValue(Value *value) const
Nicolas Capens22479eb2016-09-28 22:34:26 -04002482 {
Nicolas Capens0192d152019-03-27 14:46:07 -04002483 if(address)
2484 {
2485 return Nucleus::createStore(value, address, type, false, 0);
2486 }
2487
2488 rvalue = value;
2489
2490 return value;
2491 }
2492
2493 inline Value *Variable::getBaseAddress() const
2494 {
2495 materialize();
2496
2497 return address;
2498 }
2499
2500 inline Value *Variable::getElementPointer(Value *index, bool unsignedIndex) const
2501 {
2502 return Nucleus::createGEP(getBaseAddress(), type, index, unsignedIndex);
Nicolas Capens22479eb2016-09-28 22:34:26 -04002503 }
2504
2505 template<class T>
Nicolas Capens297d26e2016-11-18 12:52:17 -05002506 RValue<Pointer<T>> LValue<T>::operator&()
Nicolas Capensd022e412016-09-26 13:30:14 -04002507 {
Nicolas Capens0192d152019-03-27 14:46:07 -04002508 return RValue<Pointer<T>>(getBaseAddress());
Nicolas Capensd022e412016-09-26 13:30:14 -04002509 }
2510
2511 template<class T>
2512 Reference<T>::Reference(Value *pointer, int alignment) : alignment(alignment)
2513 {
2514 address = pointer;
2515 }
2516
2517 template<class T>
2518 RValue<T> Reference<T>::operator=(RValue<T> rhs) const
2519 {
Nicolas Capens6d738712016-09-30 04:15:22 -04002520 Nucleus::createStore(rhs.value, address, T::getType(), false, alignment);
Nicolas Capensd022e412016-09-26 13:30:14 -04002521
2522 return rhs;
2523 }
2524
2525 template<class T>
2526 RValue<T> Reference<T>::operator=(const Reference<T> &ref) const
2527 {
Nicolas Capense12780d2016-09-27 14:18:07 -04002528 Value *tmp = Nucleus::createLoad(ref.address, T::getType(), false, ref.alignment);
Nicolas Capens6d738712016-09-30 04:15:22 -04002529 Nucleus::createStore(tmp, address, T::getType(), false, alignment);
Nicolas Capensd022e412016-09-26 13:30:14 -04002530
2531 return RValue<T>(tmp);
2532 }
2533
2534 template<class T>
2535 RValue<T> Reference<T>::operator+=(RValue<T> rhs) const
2536 {
2537 return *this = *this + rhs;
2538 }
2539
2540 template<class T>
2541 Value *Reference<T>::loadValue() const
2542 {
Nicolas Capense12780d2016-09-27 14:18:07 -04002543 return Nucleus::createLoad(address, T::getType(), false, alignment);
Nicolas Capensd022e412016-09-26 13:30:14 -04002544 }
2545
2546 template<class T>
2547 int Reference<T>::getAlignment() const
2548 {
2549 return alignment;
2550 }
2551
Ben Claytonac07ed82019-03-26 14:17:41 +00002552#ifdef ENABLE_RR_DEBUG_INFO
2553 template<class T>
2554 RValue<T>::RValue(const RValue<T> &rvalue) : value(rvalue.value)
2555 {
2556 RR_DEBUG_INFO_EMIT_VAR(value);
2557 }
2558#endif // ENABLE_RR_DEBUG_INFO
2559
Nicolas Capensd022e412016-09-26 13:30:14 -04002560 template<class T>
2561 RValue<T>::RValue(Value *rvalue)
2562 {
Nicolas Capensbea4dce2017-07-24 16:54:44 -04002563 assert(Nucleus::createBitCast(rvalue, T::getType()) == rvalue); // Run-time type should match T, so bitcast is no-op.
2564
Nicolas Capensd022e412016-09-26 13:30:14 -04002565 value = rvalue;
Ben Claytonac07ed82019-03-26 14:17:41 +00002566 RR_DEBUG_INFO_EMIT_VAR(value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002567 }
2568
2569 template<class T>
2570 RValue<T>::RValue(const T &lvalue)
2571 {
2572 value = lvalue.loadValue();
Ben Claytonac07ed82019-03-26 14:17:41 +00002573 RR_DEBUG_INFO_EMIT_VAR(value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002574 }
2575
2576 template<class T>
Ben Clayton35e90e22019-03-15 10:06:06 +00002577 RValue<T>::RValue(typename BoolLiteral<T>::type i)
2578 {
2579 value = Nucleus::createConstantBool(i);
Ben Claytonac07ed82019-03-26 14:17:41 +00002580 RR_DEBUG_INFO_EMIT_VAR(value);
Ben Clayton35e90e22019-03-15 10:06:06 +00002581 }
2582
2583 template<class T>
Nicolas Capensd022e412016-09-26 13:30:14 -04002584 RValue<T>::RValue(typename IntLiteral<T>::type i)
2585 {
Nicolas Capensa16473e2016-11-07 15:32:52 -05002586 value = Nucleus::createConstantInt(i);
Ben Claytonac07ed82019-03-26 14:17:41 +00002587 RR_DEBUG_INFO_EMIT_VAR(value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002588 }
2589
2590 template<class T>
2591 RValue<T>::RValue(typename FloatLiteral<T>::type f)
2592 {
Nicolas Capensa16473e2016-11-07 15:32:52 -05002593 value = Nucleus::createConstantFloat(f);
Ben Claytonac07ed82019-03-26 14:17:41 +00002594 RR_DEBUG_INFO_EMIT_VAR(value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002595 }
2596
2597 template<class T>
2598 RValue<T>::RValue(const Reference<T> &ref)
2599 {
2600 value = ref.loadValue();
Ben Claytonac07ed82019-03-26 14:17:41 +00002601 RR_DEBUG_INFO_EMIT_VAR(value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002602 }
2603
Nicolas Capenscb986762017-01-20 11:34:37 -05002604 template<class Vector4, int T>
2605 Swizzle2<Vector4, T>::operator RValue<Vector4>() const
Nicolas Capensd022e412016-09-26 13:30:14 -04002606 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002607 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002608 Value *vector = parent->loadValue();
2609
Nicolas Capenscb986762017-01-20 11:34:37 -05002610 return Swizzle(RValue<Vector4>(vector), T);
Nicolas Capensd022e412016-09-26 13:30:14 -04002611 }
2612
Nicolas Capenscb986762017-01-20 11:34:37 -05002613 template<class Vector4, int T>
2614 Swizzle4<Vector4, T>::operator RValue<Vector4>() const
Nicolas Capensd022e412016-09-26 13:30:14 -04002615 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002616 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002617 Value *vector = parent->loadValue();
2618
Nicolas Capenscb986762017-01-20 11:34:37 -05002619 return Swizzle(RValue<Vector4>(vector), T);
Nicolas Capensd022e412016-09-26 13:30:14 -04002620 }
2621
Nicolas Capenscb986762017-01-20 11:34:37 -05002622 template<class Vector4, int T>
2623 SwizzleMask4<Vector4, T>::operator RValue<Vector4>() const
Nicolas Capensd022e412016-09-26 13:30:14 -04002624 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002625 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002626 Value *vector = parent->loadValue();
2627
Nicolas Capenscb986762017-01-20 11:34:37 -05002628 return Swizzle(RValue<Vector4>(vector), T);
Nicolas Capensd022e412016-09-26 13:30:14 -04002629 }
2630
Nicolas Capenscb986762017-01-20 11:34:37 -05002631 template<class Vector4, int T>
2632 RValue<Vector4> SwizzleMask4<Vector4, T>::operator=(RValue<Vector4> rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002633 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002634 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002635 return Mask(*parent, rhs, T);
2636 }
2637
Nicolas Capenscb986762017-01-20 11:34:37 -05002638 template<class Vector4, int T>
2639 RValue<Vector4> SwizzleMask4<Vector4, T>::operator=(RValue<typename Scalar<Vector4>::Type> rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002640 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002641 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capenscb986762017-01-20 11:34:37 -05002642 return Mask(*parent, Vector4(rhs), T);
Nicolas Capensd022e412016-09-26 13:30:14 -04002643 }
2644
Nicolas Capenscb986762017-01-20 11:34:37 -05002645 template<class Vector4, int T>
2646 SwizzleMask1<Vector4, T>::operator RValue<typename Scalar<Vector4>::Type>() const // FIXME: Call a non-template function
Nicolas Capensd022e412016-09-26 13:30:14 -04002647 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002648 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002649 return Extract(*parent, T & 0x3);
2650 }
2651
Nicolas Capenscb986762017-01-20 11:34:37 -05002652 template<class Vector4, int T>
2653 SwizzleMask1<Vector4, T>::operator RValue<Vector4>() const
Nicolas Capensd022e412016-09-26 13:30:14 -04002654 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002655 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002656 Value *vector = parent->loadValue();
2657
Nicolas Capenscb986762017-01-20 11:34:37 -05002658 return Swizzle(RValue<Vector4>(vector), T);
Nicolas Capensd022e412016-09-26 13:30:14 -04002659 }
2660
Nicolas Capenscb986762017-01-20 11:34:37 -05002661 template<class Vector4, int T>
2662 RValue<Vector4> SwizzleMask1<Vector4, T>::operator=(float x)
Nicolas Capensd022e412016-09-26 13:30:14 -04002663 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002664 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensc94ab742016-11-08 15:15:31 -05002665 return *parent = Insert(*parent, Float(x), T & 0x3);
Nicolas Capensd022e412016-09-26 13:30:14 -04002666 }
2667
Nicolas Capenscb986762017-01-20 11:34:37 -05002668 template<class Vector4, int T>
2669 RValue<Vector4> SwizzleMask1<Vector4, T>::operator=(RValue<Vector4> rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002670 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002671 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002672 return Mask(*parent, Float4(rhs), T);
2673 }
2674
Nicolas Capenscb986762017-01-20 11:34:37 -05002675 template<class Vector4, int T>
2676 RValue<Vector4> SwizzleMask1<Vector4, T>::operator=(RValue<typename Scalar<Vector4>::Type> rhs) // FIXME: Call a non-template function
Nicolas Capensd022e412016-09-26 13:30:14 -04002677 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002678 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensc94ab742016-11-08 15:15:31 -05002679 return *parent = Insert(*parent, rhs, T & 0x3);
Nicolas Capensd022e412016-09-26 13:30:14 -04002680 }
2681
Nicolas Capenscb986762017-01-20 11:34:37 -05002682 template<class Vector4, int T>
2683 SwizzleMask2<Vector4, T>::operator RValue<Vector4>() const
Nicolas Capensd022e412016-09-26 13:30:14 -04002684 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002685 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002686 Value *vector = parent->loadValue();
2687
Nicolas Capense95d5342016-09-30 11:37:28 -04002688 return Swizzle(RValue<Float4>(vector), T);
Nicolas Capensd022e412016-09-26 13:30:14 -04002689 }
2690
Nicolas Capenscb986762017-01-20 11:34:37 -05002691 template<class Vector4, int T>
2692 RValue<Vector4> SwizzleMask2<Vector4, T>::operator=(RValue<Vector4> rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002693 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002694 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002695 return Mask(*parent, Float4(rhs), T);
2696 }
2697
2698 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002699 Float::Float(const SwizzleMask1<Float4, T> &rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002700 {
2701 *this = rhs.operator RValue<Float>();
2702 }
2703
2704 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002705 RValue<Float> Float::operator=(const SwizzleMask1<Float4, T> &rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002706 {
2707 return *this = rhs.operator RValue<Float>();
2708 }
2709
2710 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002711 Float4::Float4(const SwizzleMask1<Float4, T> &rhs) : XYZW(this)
Nicolas Capensd022e412016-09-26 13:30:14 -04002712 {
Nicolas Capensd022e412016-09-26 13:30:14 -04002713 *this = rhs.operator RValue<Float4>();
2714 }
2715
2716 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002717 Float4::Float4(const Swizzle4<Float4, T> &rhs) : XYZW(this)
Nicolas Capensd022e412016-09-26 13:30:14 -04002718 {
Nicolas Capensd022e412016-09-26 13:30:14 -04002719 *this = rhs.operator RValue<Float4>();
2720 }
2721
2722 template<int X, int Y>
Nicolas Capenscb986762017-01-20 11:34:37 -05002723 Float4::Float4(const Swizzle2<Float4, X> &x, const Swizzle2<Float4, Y> &y) : XYZW(this)
Nicolas Capensd022e412016-09-26 13:30:14 -04002724 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002725 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002726 *this = ShuffleLowHigh(*x.parent, *y.parent, (X & 0xF) | (Y & 0xF) << 4);
2727 }
2728
2729 template<int X, int Y>
Nicolas Capenscb986762017-01-20 11:34:37 -05002730 Float4::Float4(const SwizzleMask2<Float4, X> &x, const Swizzle2<Float4, Y> &y) : XYZW(this)
Nicolas Capensd022e412016-09-26 13:30:14 -04002731 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002732 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002733 *this = ShuffleLowHigh(*x.parent, *y.parent, (X & 0xF) | (Y & 0xF) << 4);
2734 }
2735
2736 template<int X, int Y>
Nicolas Capenscb986762017-01-20 11:34:37 -05002737 Float4::Float4(const Swizzle2<Float4, X> &x, const SwizzleMask2<Float4, Y> &y) : XYZW(this)
Nicolas Capensd022e412016-09-26 13:30:14 -04002738 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002739 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002740 *this = ShuffleLowHigh(*x.parent, *y.parent, (X & 0xF) | (Y & 0xF) << 4);
2741 }
2742
2743 template<int X, int Y>
Nicolas Capenscb986762017-01-20 11:34:37 -05002744 Float4::Float4(const SwizzleMask2<Float4, X> &x, const SwizzleMask2<Float4, Y> &y) : XYZW(this)
Nicolas Capensd022e412016-09-26 13:30:14 -04002745 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002746 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002747 *this = ShuffleLowHigh(*x.parent, *y.parent, (X & 0xF) | (Y & 0xF) << 4);
2748 }
2749
2750 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002751 RValue<Float4> Float4::operator=(const SwizzleMask1<Float4, T> &rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002752 {
2753 return *this = rhs.operator RValue<Float4>();
2754 }
2755
2756 template<int T>
Nicolas Capenscb986762017-01-20 11:34:37 -05002757 RValue<Float4> Float4::operator=(const Swizzle4<Float4, T> &rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002758 {
2759 return *this = rhs.operator RValue<Float4>();
2760 }
2761
2762 template<class T>
2763 Pointer<T>::Pointer(Argument<Pointer<T>> argument) : alignment(1)
2764 {
Nicolas Capens51d98672019-04-02 12:05:40 -04002765 LValue<Pointer<T>>::materialize(); // FIXME(b/129757459)
Nicolas Capens22479eb2016-09-28 22:34:26 -04002766 LValue<Pointer<T>>::storeValue(argument.value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002767 }
2768
2769 template<class T>
Nicolas Capensd022e412016-09-26 13:30:14 -04002770 Pointer<T>::Pointer() : alignment(1)
2771 {
Nicolas Capens22479eb2016-09-28 22:34:26 -04002772 LValue<Pointer<T>>::storeValue(Nucleus::createNullPointer(T::getType()));
Nicolas Capensd022e412016-09-26 13:30:14 -04002773 }
2774
2775 template<class T>
2776 Pointer<T>::Pointer(RValue<Pointer<T>> rhs) : alignment(1)
2777 {
Nicolas Capens22479eb2016-09-28 22:34:26 -04002778 LValue<Pointer<T>>::storeValue(rhs.value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002779 }
2780
2781 template<class T>
2782 Pointer<T>::Pointer(const Pointer<T> &rhs) : alignment(rhs.alignment)
2783 {
2784 Value *value = rhs.loadValue();
Nicolas Capens22479eb2016-09-28 22:34:26 -04002785 LValue<Pointer<T>>::storeValue(value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002786 }
2787
2788 template<class T>
2789 Pointer<T>::Pointer(const Reference<Pointer<T>> &rhs) : alignment(rhs.getAlignment())
2790 {
2791 Value *value = rhs.loadValue();
Nicolas Capens22479eb2016-09-28 22:34:26 -04002792 LValue<Pointer<T>>::storeValue(value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002793 }
2794
2795 template<class T>
Nicolas Capens96d4e092016-11-18 14:22:38 -05002796 RValue<Pointer<T>> Pointer<T>::operator=(RValue<Pointer<T>> rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002797 {
Nicolas Capens22479eb2016-09-28 22:34:26 -04002798 LValue<Pointer<T>>::storeValue(rhs.value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002799
2800 return rhs;
2801 }
2802
2803 template<class T>
Nicolas Capens96d4e092016-11-18 14:22:38 -05002804 RValue<Pointer<T>> Pointer<T>::operator=(const Pointer<T> &rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002805 {
2806 Value *value = rhs.loadValue();
Nicolas Capens22479eb2016-09-28 22:34:26 -04002807 LValue<Pointer<T>>::storeValue(value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002808
2809 return RValue<Pointer<T>>(value);
2810 }
2811
2812 template<class T>
Nicolas Capens96d4e092016-11-18 14:22:38 -05002813 RValue<Pointer<T>> Pointer<T>::operator=(const Reference<Pointer<T>> &rhs)
Nicolas Capensd022e412016-09-26 13:30:14 -04002814 {
2815 Value *value = rhs.loadValue();
Nicolas Capens22479eb2016-09-28 22:34:26 -04002816 LValue<Pointer<T>>::storeValue(value);
Nicolas Capensd022e412016-09-26 13:30:14 -04002817
2818 return RValue<Pointer<T>>(value);
2819 }
2820
2821 template<class T>
2822 Reference<T> Pointer<T>::operator*()
2823 {
Nicolas Capens22479eb2016-09-28 22:34:26 -04002824 return Reference<T>(LValue<Pointer<T>>::loadValue(), alignment);
Nicolas Capensd022e412016-09-26 13:30:14 -04002825 }
2826
2827 template<class T>
2828 Reference<T> Pointer<T>::operator[](int index)
2829 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002830 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd294def2017-01-26 17:44:37 -08002831 Value *element = Nucleus::createGEP(LValue<Pointer<T>>::loadValue(), T::getType(), Nucleus::createConstantInt(index), false);
2832
2833 return Reference<T>(element, alignment);
2834 }
2835
2836 template<class T>
2837 Reference<T> Pointer<T>::operator[](unsigned int index)
2838 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002839 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd294def2017-01-26 17:44:37 -08002840 Value *element = Nucleus::createGEP(LValue<Pointer<T>>::loadValue(), T::getType(), Nucleus::createConstantInt(index), true);
Nicolas Capensd022e412016-09-26 13:30:14 -04002841
2842 return Reference<T>(element, alignment);
2843 }
2844
2845 template<class T>
2846 Reference<T> Pointer<T>::operator[](RValue<Int> index)
2847 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002848 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd294def2017-01-26 17:44:37 -08002849 Value *element = Nucleus::createGEP(LValue<Pointer<T>>::loadValue(), T::getType(), index.value, false);
2850
2851 return Reference<T>(element, alignment);
2852 }
2853
2854 template<class T>
2855 Reference<T> Pointer<T>::operator[](RValue<UInt> index)
2856 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002857 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd294def2017-01-26 17:44:37 -08002858 Value *element = Nucleus::createGEP(LValue<Pointer<T>>::loadValue(), T::getType(), index.value, true);
Nicolas Capensd022e412016-09-26 13:30:14 -04002859
2860 return Reference<T>(element, alignment);
2861 }
2862
2863 template<class T>
2864 Type *Pointer<T>::getType()
2865 {
2866 return Nucleus::getPointerType(T::getType());
2867 }
2868
2869 template<class T, int S>
Nicolas Capens297d26e2016-11-18 12:52:17 -05002870 Array<T, S>::Array(int size) : LValue<T>(size)
Nicolas Capensd022e412016-09-26 13:30:14 -04002871 {
2872 }
2873
2874 template<class T, int S>
2875 Reference<T> Array<T, S>::operator[](int index)
2876 {
Nicolas Capens0192d152019-03-27 14:46:07 -04002877 Value *element = LValue<T>::getElementPointer(Nucleus::createConstantInt(index), false);
Nicolas Capensd294def2017-01-26 17:44:37 -08002878
2879 return Reference<T>(element);
2880 }
2881
2882 template<class T, int S>
2883 Reference<T> Array<T, S>::operator[](unsigned int index)
2884 {
Nicolas Capens0192d152019-03-27 14:46:07 -04002885 Value *element = LValue<T>::getElementPointer(Nucleus::createConstantInt(index), true);
Nicolas Capensd022e412016-09-26 13:30:14 -04002886
2887 return Reference<T>(element);
2888 }
2889
2890 template<class T, int S>
2891 Reference<T> Array<T, S>::operator[](RValue<Int> index)
2892 {
Nicolas Capens0192d152019-03-27 14:46:07 -04002893 Value *element = LValue<T>::getElementPointer(index.value, false);
Nicolas Capensd294def2017-01-26 17:44:37 -08002894
2895 return Reference<T>(element);
2896 }
2897
2898 template<class T, int S>
2899 Reference<T> Array<T, S>::operator[](RValue<UInt> index)
2900 {
Nicolas Capens0192d152019-03-27 14:46:07 -04002901 Value *element = LValue<T>::getElementPointer(index.value, true);
Nicolas Capensd022e412016-09-26 13:30:14 -04002902
2903 return Reference<T>(element);
2904 }
2905
2906// template<class T>
Nicolas Capens96d4e092016-11-18 14:22:38 -05002907// RValue<Array<T>> operator++(Array<T> &val, int)
Nicolas Capensd022e412016-09-26 13:30:14 -04002908// {
2909// // FIXME: Requires storing the address of the array
2910// }
2911
2912// template<class T>
Nicolas Capens96d4e092016-11-18 14:22:38 -05002913// const Array<T> &operator++(Array<T> &val)
Nicolas Capensd022e412016-09-26 13:30:14 -04002914// {
2915// // FIXME: Requires storing the address of the array
2916// }
2917
2918// template<class T>
Nicolas Capens96d4e092016-11-18 14:22:38 -05002919// RValue<Array<T>> operator--(Array<T> &val, int)
Nicolas Capensd022e412016-09-26 13:30:14 -04002920// {
2921// // FIXME: Requires storing the address of the array
2922// }
2923
2924// template<class T>
Nicolas Capens96d4e092016-11-18 14:22:38 -05002925// const Array<T> &operator--(Array<T> &val)
Nicolas Capensd022e412016-09-26 13:30:14 -04002926// {
2927// // FIXME: Requires storing the address of the array
2928// }
2929
2930 template<class T>
2931 RValue<T> IfThenElse(RValue<Bool> condition, RValue<T> ifTrue, RValue<T> ifFalse)
2932 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002933 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002934 return RValue<T>(Nucleus::createSelect(condition.value, ifTrue.value, ifFalse.value));
2935 }
2936
2937 template<class T>
2938 RValue<T> IfThenElse(RValue<Bool> condition, const T &ifTrue, RValue<T> ifFalse)
2939 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002940 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002941 Value *trueValue = ifTrue.loadValue();
2942
2943 return RValue<T>(Nucleus::createSelect(condition.value, trueValue, ifFalse.value));
2944 }
2945
2946 template<class T>
2947 RValue<T> IfThenElse(RValue<Bool> condition, RValue<T> ifTrue, const T &ifFalse)
2948 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002949 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002950 Value *falseValue = ifFalse.loadValue();
2951
2952 return RValue<T>(Nucleus::createSelect(condition.value, ifTrue.value, falseValue));
2953 }
2954
2955 template<class T>
2956 RValue<T> IfThenElse(RValue<Bool> condition, const T &ifTrue, const T &ifFalse)
2957 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002958 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002959 Value *trueValue = ifTrue.loadValue();
2960 Value *falseValue = ifFalse.loadValue();
2961
2962 return RValue<T>(Nucleus::createSelect(condition.value, trueValue, falseValue));
2963 }
2964
2965 template<class T>
2966 void Return(const Pointer<T> &ret)
2967 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002968 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capense12780d2016-09-27 14:18:07 -04002969 Nucleus::createRet(Nucleus::createLoad(ret.address, Pointer<T>::getType()));
Nicolas Capensd022e412016-09-26 13:30:14 -04002970 Nucleus::setInsertBlock(Nucleus::createBasicBlock());
Ben Claytonf3b57972019-03-15 09:56:47 +00002971 Nucleus::createUnreachable();
Nicolas Capensd022e412016-09-26 13:30:14 -04002972 }
2973
2974 template<class T>
2975 void Return(RValue<Pointer<T>> ret)
2976 {
Ben Claytonac07ed82019-03-26 14:17:41 +00002977 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04002978 Nucleus::createRet(ret.value);
2979 Nucleus::setInsertBlock(Nucleus::createBasicBlock());
Ben Claytonf3b57972019-03-15 09:56:47 +00002980 Nucleus::createUnreachable();
Nicolas Capensd022e412016-09-26 13:30:14 -04002981 }
2982
2983 template<typename Return, typename... Arguments>
2984 Function<Return(Arguments...)>::Function()
2985 {
2986 core = new Nucleus();
2987
2988 Type *types[] = {Arguments::getType()...};
2989 for(Type *type : types)
2990 {
2991 if(type != Void::getType())
2992 {
2993 arguments.push_back(type);
2994 }
2995 }
2996
2997 Nucleus::createFunction(Return::getType(), arguments);
2998 }
2999
3000 template<typename Return, typename... Arguments>
3001 Function<Return(Arguments...)>::~Function()
3002 {
3003 delete core;
3004 }
3005
3006 template<typename Return, typename... Arguments>
Chris Forbes878d4b02019-01-21 10:48:35 -08003007 Routine *Function<Return(Arguments...)>::operator()(const char *name, ...)
Nicolas Capensd022e412016-09-26 13:30:14 -04003008 {
Chris Forbes878d4b02019-01-21 10:48:35 -08003009 char fullName[1024 + 1];
Nicolas Capensd022e412016-09-26 13:30:14 -04003010
3011 va_list vararg;
3012 va_start(vararg, name);
Chris Forbes878d4b02019-01-21 10:48:35 -08003013 vsnprintf(fullName, 1024, name, vararg);
Nicolas Capensd022e412016-09-26 13:30:14 -04003014 va_end(vararg);
3015
3016 return core->acquireRoutine(fullName, true);
3017 }
3018
3019 template<class T, class S>
3020 RValue<T> ReinterpretCast(RValue<S> val)
3021 {
Ben Claytonac07ed82019-03-26 14:17:41 +00003022 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04003023 return RValue<T>(Nucleus::createBitCast(val.value, T::getType()));
3024 }
3025
Nicolas Capens22479eb2016-09-28 22:34:26 -04003026 template<class T, class S>
3027 RValue<T> ReinterpretCast(const LValue<S> &var)
Nicolas Capensd022e412016-09-26 13:30:14 -04003028 {
Ben Claytonac07ed82019-03-26 14:17:41 +00003029 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capensd022e412016-09-26 13:30:14 -04003030 Value *val = var.loadValue();
3031
3032 return RValue<T>(Nucleus::createBitCast(val, T::getType()));
3033 }
3034
3035 template<class T, class S>
3036 RValue<T> ReinterpretCast(const Reference<S> &var)
3037 {
3038 return ReinterpretCast<T>(RValue<S>(var));
3039 }
3040
Nicolas Capens70dfff42016-10-27 10:20:28 -04003041 template<class T>
3042 RValue<T> As(Value *val)
3043 {
Ben Claytonac07ed82019-03-26 14:17:41 +00003044 RR_DEBUG_INFO_UPDATE_LOC();
Nicolas Capens70dfff42016-10-27 10:20:28 -04003045 return RValue<T>(Nucleus::createBitCast(val, T::getType()));
3046 }
3047
Nicolas Capensd022e412016-09-26 13:30:14 -04003048 template<class T, class S>
3049 RValue<T> As(RValue<S> val)
3050 {
3051 return ReinterpretCast<T>(val);
3052 }
3053
Nicolas Capens22479eb2016-09-28 22:34:26 -04003054 template<class T, class S>
3055 RValue<T> As(const LValue<S> &var)
Nicolas Capensd022e412016-09-26 13:30:14 -04003056 {
3057 return ReinterpretCast<T>(var);
3058 }
3059
3060 template<class T, class S>
3061 RValue<T> As(const Reference<S> &val)
3062 {
3063 return ReinterpretCast<T>(val);
3064 }
3065
Ben Claytond853c122019-04-16 17:51:49 -04003066 template <typename T>
3067 inline Value* valueOf(RValue<T> v) { return v.value; }
3068
3069 template <typename T>
3070 inline Value* valueOf(LValue<T> v) { return valueOf(RValue<T>(v.loadValue())); }
3071
3072 template<typename T>
3073 struct CToReactor;
3074
3075 template<> struct CToReactor<void> { using type = Void; };
3076 template<> struct CToReactor<int> { using type = Int; };
3077 template<> struct CToReactor<float> { using type = Float; };
3078 template<> struct CToReactor<int*> { using type = Pointer<Int>; };
3079 template<> struct CToReactor<float*> { using type = Pointer<Float>; };
3080
3081 // Pointers to non-reactor types are treated as uint8_t*.
3082 template<typename T>
3083 struct CToReactor<T*> { using type = Pointer<Byte>; };
3084
3085 // Returns a reactor pointer to the fixed-address ptr.
3086 RValue<Pointer<Byte>> ConstantPointer(void const * ptr);
3087
3088 // Calls the function pointer fptr with the given arguments, return type
3089 // and parameter types. Returns the call's return value if the function has
3090 // a non-void return type.
3091 Value* Call(RValue<Pointer<Byte>> fptr, Type* retTy, std::initializer_list<Value*> args, std::initializer_list<Type*> paramTys);
3092
3093 template <typename F>
3094 class CallHelper {};
3095
3096 template<typename Return, typename ... Arguments>
3097 class CallHelper<Return(Arguments...)>
3098 {
3099 public:
3100 using RReturn = typename CToReactor<Return>::type;
3101
3102 static inline RReturn Call(Return(fptr)(Arguments...), typename CToReactor<Arguments>::type... args)
3103 {
3104 return RValue<RReturn>(rr::Call(
3105 ConstantPointer(reinterpret_cast<void*>(fptr)),
3106 RReturn::getType(),
3107 { valueOf(args) ... },
3108 { CToReactor<Arguments>::type::getType() ... }));
3109 }
3110
3111 static inline RReturn Call(Pointer<Byte> fptr, typename CToReactor<Arguments>::type... args)
3112 {
3113 return RValue<RReturn>(rr::Call(
3114 fptr,
3115 RReturn::getType(),
3116 { valueOf(args) ... },
3117 { CToReactor<Arguments>::type::getType() ... }));
3118 }
3119 };
3120
3121 template<typename ... Arguments>
3122 class CallHelper<void(Arguments...)>
3123 {
3124 public:
3125 static inline void Call(void(fptr)(Arguments...), typename CToReactor<Arguments>::type... args)
3126 {
3127 rr::Call(ConstantPointer(reinterpret_cast<void*>(fptr)),
3128 Void::getType(),
3129 { valueOf(args) ... },
3130 { CToReactor<Arguments>::type::getType() ... });
3131 }
3132
3133 static inline void Call(Pointer<Byte> fptr, typename CToReactor<Arguments>::type... args)
3134 {
3135 rr::Call(fptr,
3136 Void::getType(),
3137 { valueOf(args) ... },
3138 { CToReactor<Arguments>::type::getType() ... });
3139 }
3140 };
3141
3142 // Calls the function pointer fptr with the given arguments args.
3143 template<typename Return, typename ... Arguments>
3144 inline typename CToReactor<Return>::type Call(Return(fptr)(Arguments...), typename CToReactor<Arguments>::type... args)
3145 {
3146 return CallHelper<Return(Arguments...)>::Call(fptr, args...);
3147 }
3148
3149 // Calls the function pointer fptr with the signature FUNCTION_SIGNATURE and
3150 // arguments.
3151 template<typename FUNCTION_SIGNATURE, typename ... Arguments>
3152 inline void Call(Pointer<Byte> fptr, Arguments ... args)
3153 {
3154 CallHelper<FUNCTION_SIGNATURE>::Call(fptr, args...);
3155 }
3156
Ben Claytonac07ed82019-03-26 14:17:41 +00003157 // Breakpoint emits an instruction that will cause the application to trap.
3158 // This can be used to stop an attached debugger at the given call.
3159 void Breakpoint();
3160
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003161#ifdef ENABLE_RR_PRINT
3162 // PrintValue holds the printf format and value(s) for a single argument
3163 // to Print(). A single argument can be expanded into multiple printf
3164 // values - for example a Float4 will expand to "%f %f %f %f" and four
3165 // scalar values.
3166 // The PrintValue constructor accepts the following:
3167 // * Reactor LValues, RValues, Pointers.
3168 // * Standard Plain-Old-Value types (int, float, bool, etc)
3169 // * Custom types that specialize the PrintValue::Ty template struct.
3170 // * Static arrays in the form T[N] where T can be any of the above.
3171 class PrintValue
3172 {
3173 // Ty is a template that can be specialized for printing type T.
3174 // Each specialization must expose:
3175 // * A 'static constexpr const char* fmt' field that provides the
3176 // printf format specifier.
3177 // * A 'static std::vector<rr::Value*> val(const T& v)' method that
3178 // returns all the printf format values.
3179 template <typename T> struct Ty
3180 {
3181 // static constexpr const char* fmt;
3182 // static std::vector<rr::Value*> val(const T& v)
3183 };
3184
3185 // returns the printf value(s) for the given LValue.
3186 template <typename T>
Ben Claytonbcc71a32019-04-21 07:35:37 -04003187 static std::vector<Value*> val(const LValue<T>& v) { return val(RValue<T>(v.loadValue())); }
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003188
3189 // returns the printf value(s) for the given RValue.
3190 template <typename T>
Ben Claytonbcc71a32019-04-21 07:35:37 -04003191 static std::vector<Value*> val(const RValue<T>& v) { return Ty<T>::val(v); }
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003192
3193 // returns the printf value from for the given type with a
3194 // PrintValue::Ty<T> specialization.
3195 template <typename T>
Ben Claytonbcc71a32019-04-21 07:35:37 -04003196 static std::vector<Value*> val(const T& v) { return Ty<T>::val(v); }
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003197
3198 // returns the printf values for all the values in the given array.
3199 template <typename T>
3200 static std::vector<Value*> val(const T* list, int count) {
3201 std::vector<Value*> values;
3202 values.reserve(count);
3203 for (int i = 0; i < count; i++)
3204 {
3205 auto v = val(list[i]);
3206 values.insert(values.end(), v.begin(), v.end());
3207 }
3208 return values;
Ben Claytonbcc71a32019-04-21 07:35:37 -04003209 }
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003210
3211 // fmt returns a comma-delimited list of the string el repeated count
3212 // times enclosed in square brackets.
3213 static std::string fmt(const char* el, int count)
3214 {
3215 std::string out = "[";
3216 for (int i = 0; i < count; i++)
3217 {
3218 if (i > 0) { out += ", "; }
3219 out += el;
3220 }
3221 return out + "]";
3222 }
3223
Ben Clayton3ed8ba02019-04-04 22:01:51 +01003224 static std::string addr(const void* ptr)
3225 {
3226 char buf[32];
3227 snprintf(buf, sizeof(buf), "%p", ptr);
3228 return buf;
3229 }
3230
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003231 public:
3232 const std::string format;
3233 const std::vector<Value*> values;
3234
3235 // Constructs a PrintValue for the given value.
3236 template <typename T>
3237 PrintValue(const T& v) : format(Ty<T>::fmt), values(val(v)) {}
3238
3239 // Constructs a PrintValue for the given static array.
3240 template <typename T, int N>
3241 PrintValue(const T (&v)[N]) : format(fmt(Ty<T>::fmt, N)), values(val(&v[0], N)) {}
3242
3243 // Constructs a PrintValue for the given array starting at arr of length
3244 // len.
3245 template <typename T>
3246 PrintValue(const T* arr, int len) : format(fmt(Ty<T>::fmt, len)), values(val(arr, len)) {}
3247
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003248 // PrintValue constructors for plain-old-data values.
3249 PrintValue(bool v) : format(v ? "true" : "false") {}
3250 PrintValue(int8_t v) : format(std::to_string(v)) {}
3251 PrintValue(uint8_t v) : format(std::to_string(v)) {}
3252 PrintValue(int16_t v) : format(std::to_string(v)) {}
3253 PrintValue(uint16_t v) : format(std::to_string(v)) {}
3254 PrintValue(int32_t v) : format(std::to_string(v)) {}
3255 PrintValue(uint32_t v) : format(std::to_string(v)) {}
3256 PrintValue(int64_t v) : format(std::to_string(v)) {}
3257 PrintValue(uint64_t v) : format(std::to_string(v)) {}
3258 PrintValue(float v) : format(std::to_string(v)) {}
3259 PrintValue(double v) : format(std::to_string(v)) {}
3260 PrintValue(const char* v) : format(v) {}
3261 PrintValue(const std::string& v) : format(v) {}
3262
Ben Clayton3ed8ba02019-04-04 22:01:51 +01003263 template <typename T>
3264 PrintValue(const T* v) : format(addr(v)) {}
3265
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003266 // vals is a helper to build composite value lists.
3267 // vals returns the full, sequential list of printf argument values used
3268 // to print all the provided variadic values.
3269 // vals() is intended to be used by implementations of
3270 // PrintValue::Ty<>::vals() to help declare aggregate types.
3271 // For example, if you were declaring a PrintValue::Ty<> specialization
3272 // for a custom Mat4x4 matrix formed from four Vector4 values, you'd
3273 // write:
3274 //
3275 // namespace rr
3276 // {
3277 // template <> struct PrintValue::Ty<Mat4x4>
3278 // {
3279 // static constexpr const char* fmt =
3280 // "[a: <%f, %f, %f, %f>,"
3281 // " b: <%f, %f, %f, %f>,"
3282 // " c: <%f, %f, %f, %f>,"
3283 // " d: <%f, %f, %f, %f>]";
3284 // static std::vector<rr::Value*> val(const Mat4x4& v)
3285 // {
3286 // return PrintValue::vals(v.a, v.b, v.c, v.d);
3287 // }
3288 // };
3289 // }
3290 template<typename ... ARGS>
3291 static std::vector<Value*> vals(ARGS... v)
3292 {
3293 std::vector< std::vector<Value*> > lists = {val(v)...};
3294 std::vector<Value*> joined;
3295 for (const auto& list : lists)
3296 {
3297 joined.insert(joined.end(), list.begin(), list.end());
3298 }
3299 return joined;
3300 }
3301 };
3302
3303 // PrintValue::Ty<T> specializations for standard Reactor types.
Ben Claytonca56e8b2019-03-12 20:05:43 +00003304 template <> struct PrintValue::Ty<Bool>
3305 {
3306 static constexpr const char* fmt = "%d";
3307 static std::vector<Value*> val(const RValue<Bool>& v) { return {v.value}; }
3308 };
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003309 template <> struct PrintValue::Ty<Byte>
3310 {
3311 static constexpr const char* fmt = "%d";
3312 static std::vector<Value*> val(const RValue<Byte>& v) { return {v.value}; }
3313 };
3314 template <> struct PrintValue::Ty<Byte4>
3315 {
3316 static constexpr const char* fmt = "[%d, %d, %d, %d]";
3317 static std::vector<Value*> val(const RValue<Byte4>& v);
3318 };
3319 template <> struct PrintValue::Ty<Int>
3320 {
3321 static constexpr const char* fmt = "%d";
3322 static std::vector<Value*> val(const RValue<Int>& v) { return {v.value}; }
3323 };
3324 template <> struct PrintValue::Ty<Int4>
3325 {
3326 static constexpr const char* fmt = "[%d, %d, %d, %d]";
3327 static std::vector<Value*> val(const RValue<Int4>& v);
3328 };
3329 template <> struct PrintValue::Ty<UInt>
3330 {
3331 static constexpr const char* fmt = "%u";
3332 static std::vector<Value*> val(const RValue<UInt>& v) { return {v.value}; }
3333 };
3334 template <> struct PrintValue::Ty<UInt4>
3335 {
3336 static constexpr const char* fmt = "[%u, %u, %u, %u]";
3337 static std::vector<Value*> val(const RValue<UInt4>& v);
3338 };
3339 template <> struct PrintValue::Ty<Short>
3340 {
3341 static constexpr const char* fmt = "%d";
3342 static std::vector<Value*> val(const RValue<Short>& v) { return {v.value}; }
3343 };
3344 template <> struct PrintValue::Ty<Short4>
3345 {
3346 static constexpr const char* fmt = "[%d, %d, %d, %d]";
3347 static std::vector<Value*> val(const RValue<Short4>& v);
3348 };
3349 template <> struct PrintValue::Ty<UShort>
3350 {
3351 static constexpr const char* fmt = "%u";
3352 static std::vector<Value*> val(const RValue<UShort>& v) { return {v.value}; }
3353 };
3354 template <> struct PrintValue::Ty<UShort4>
3355 {
3356 static constexpr const char* fmt = "[%u, %u, %u, %u]";
3357 static std::vector<Value*> val(const RValue<UShort4>& v);
3358 };
3359 template <> struct PrintValue::Ty<Float>
3360 {
3361 static constexpr const char* fmt = "[%f]";
3362 static std::vector<Value*> val(const RValue<Float>& v);
3363 };
3364 template <> struct PrintValue::Ty<Float4>
3365 {
3366 static constexpr const char* fmt = "[%f, %f, %f, %f]";
3367 static std::vector<Value*> val(const RValue<Float4>& v);
3368 };
Ben Clayton7945a512019-04-15 23:00:43 -04003369 template <> struct PrintValue::Ty<Long>
3370 {
3371 static constexpr const char* fmt = "%lld";
3372 static std::vector<Value*> val(const RValue<Long>& v) { return {v.value}; }
3373 };
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003374 template <typename T> struct PrintValue::Ty< Pointer<T> >
3375 {
3376 static constexpr const char* fmt = "%p";
3377 static std::vector<Value*> val(const RValue<Pointer<T>>& v) { return {v.value}; }
3378 };
3379 template <typename T> struct PrintValue::Ty< Reference<T> >
3380 {
3381 static constexpr const char* fmt = PrintValue::Ty<T>::fmt;
3382 static std::vector<Value*> val(const Reference<T>& v) { return PrintValue::Ty<T>::val(v); }
3383 };
3384 template <typename T> struct PrintValue::Ty< RValue<T> >
3385 {
3386 static constexpr const char* fmt = PrintValue::Ty<T>::fmt;
3387 static std::vector<Value*> val(const RValue<T>& v) { return PrintValue::Ty<T>::val(v); }
3388 };
3389
3390 // Printv emits a call to printf() using the function, file and line,
3391 // message and optional values.
3392 // See Printv below.
3393 void Printv(const char* function, const char* file, int line, const char* msg, std::initializer_list<PrintValue> vals);
3394
3395 // Printv emits a call to printf() using the provided message and optional
3396 // values.
3397 // Printf replaces any bracketed indices in the message with string
3398 // representations of the corresponding value in vals.
3399 // For example:
3400 // Printv("{0} and {1}", "red", "green");
3401 // Would print the string:
3402 // "red and green"
3403 // Arguments can be indexed in any order.
3404 // Invalid indices are not substituted.
3405 inline void Printv(const char* msg, std::initializer_list<PrintValue> vals)
3406 {
3407 Printv(nullptr, nullptr, 0, msg, vals);
3408 }
3409
3410 // Print is a wrapper over Printv that wraps the variadic arguments into an
3411 // initializer_list before calling Printv.
3412 template <typename ... ARGS>
3413 void Print(const char* msg, const ARGS& ... vals) { Printv(msg, {vals...}); }
3414
3415 // Print is a wrapper over Printv that wraps the variadic arguments into an
3416 // initializer_list before calling Printv.
3417 template <typename ... ARGS>
3418 void Print(const char* function, const char* file, int line, const char* msg, const ARGS& ... vals)
3419 {
3420 Printv(function, file, line, msg, {vals...});
3421 }
3422
3423 // RR_LOG is a macro that calls Print(), automatically populating the
3424 // function, file and line parameters and appending a newline to the string.
3425 //
3426 // RR_LOG() is intended to be used for debugging JIT compiled code, and is
3427 // not intended for production use.
3428 #define RR_LOG(msg, ...) Print(__PRETTY_FUNCTION__, __FILE__, __LINE__, msg "\n", ##__VA_ARGS__)
3429
3430 // Macro magic to perform variadic dispatch.
3431 // See: https://renenyffenegger.ch/notes/development/languages/C-C-plus-plus/preprocessor/macros/__VA_ARGS__/count-arguments
3432 // Note, this doesn't attempt to use the ##__VA_ARGS__ trick to handle 0
3433 // args as this appears to still be broken on certain compilers.
Ben Clayton221459f2019-03-25 15:14:45 +00003434 // MSVC also has issues with variadic macros which requires the RR_VA_MSVC_BUG() work-around.
3435 // See: https://stackoverflow.com/a/48711060
3436 #define RR_VA_MSVC_BUG(MACRO, ARGS) MACRO ARGS
3437 #define RR_GET_NTH_ARG_EX(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12, _13, _14, _15, _16, N, ...) N
3438 #define RR_GET_NTH_ARG(...) RR_VA_MSVC_BUG(RR_GET_NTH_ARG_EX, (__VA_ARGS__))
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003439 #define RR_COUNT_ARGUMENTS(...) RR_GET_NTH_ARG(__VA_ARGS__, 16, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1, 0)
Ben Clayton60a3d6f2019-02-26 17:24:46 +00003440 static_assert(1 == RR_COUNT_ARGUMENTS(a), "RR_COUNT_ARGUMENTS broken"); // Sanity checks.
3441 static_assert(2 == RR_COUNT_ARGUMENTS(a, b), "RR_COUNT_ARGUMENTS broken");
3442 static_assert(3 == RR_COUNT_ARGUMENTS(a, b, c), "RR_COUNT_ARGUMENTS broken");
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003443
3444 // RR_WATCH_FMT(...) resolves to a string literal that lists all the
3445 // arguments by name. This string can be passed to LOG() to print each of
3446 // the arguments with their name and value.
3447 //
3448 // RR_WATCH_FMT(...) uses the RR_COUNT_ARGUMENTS helper macro to delegate to a
3449 // corresponding RR_WATCH_FMT_n specialization macro below.
3450 #define RR_WATCH_CONCAT(a, b) a ## b
3451 #define RR_WATCH_CONCAT2(a, b) RR_WATCH_CONCAT(a, b)
3452 #define RR_WATCH_FMT(...) RR_WATCH_CONCAT2(RR_WATCH_FMT_, RR_COUNT_ARGUMENTS(__VA_ARGS__))(__VA_ARGS__)
3453 #define RR_WATCH_FMT_1(_1) "\n " #_1 ": {0}"
Ben Clayton644a3502019-03-13 11:24:46 +00003454 #define RR_WATCH_FMT_2(_1, _2) RR_WATCH_FMT_1(_1) "\n " #_2 ": {1}"
3455 #define RR_WATCH_FMT_3(_1, _2, _3) RR_WATCH_FMT_2(_1, _2) "\n " #_3 ": {2}"
3456 #define RR_WATCH_FMT_4(_1, _2, _3, _4) RR_WATCH_FMT_3(_1, _2, _3) "\n " #_4 ": {3}"
3457 #define RR_WATCH_FMT_5(_1, _2, _3, _4, _5) RR_WATCH_FMT_4(_1, _2, _3, _4) "\n " #_5 ": {4}"
3458 #define RR_WATCH_FMT_6(_1, _2, _3, _4, _5, _6) RR_WATCH_FMT_5(_1, _2, _3, _4, _5) "\n " #_6 ": {5}"
3459 #define RR_WATCH_FMT_7(_1, _2, _3, _4, _5, _6, _7) RR_WATCH_FMT_6(_1, _2, _3, _4, _5, _6) "\n " #_7 ": {6}"
3460 #define RR_WATCH_FMT_8(_1, _2, _3, _4, _5, _6, _7, _8) RR_WATCH_FMT_7(_1, _2, _3, _4, _5, _6, _7) "\n " #_8 ": {7}"
3461 #define RR_WATCH_FMT_9(_1, _2, _3, _4, _5, _6, _7, _8, _9) RR_WATCH_FMT_8(_1, _2, _3, _4, _5, _6, _7, _8) "\n " #_9 ": {8}"
3462 #define RR_WATCH_FMT_10(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10) RR_WATCH_FMT_9(_1, _2, _3, _4, _5, _6, _7, _8, _9) "\n " #_10 ": {9}"
3463 #define RR_WATCH_FMT_11(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11) RR_WATCH_FMT_10(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10) "\n " #_11 ": {10}"
3464 #define RR_WATCH_FMT_12(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11, _12) RR_WATCH_FMT_11(_1, _2, _3, _4, _5, _6, _7, _8, _9, _10, _11) "\n " #_12 ": {11}"
Ben Clayton1bc7ee92019-02-14 18:43:22 +00003465
3466 // RR_WATCH() is a helper that prints the name and value of all the supplied
3467 // arguments.
3468 // For example, if you had the Int and bool variables 'foo' and 'bar' that
3469 // you want to print, you can simply write:
3470 // RR_WATCH(foo, bar)
3471 // When this JIT compiled code is executed, it will print the string
3472 // "foo: 1, bar: true" to stdout.
3473 //
3474 // RR_WATCH() is intended to be used for debugging JIT compiled code, and
3475 // is not intended for production use.
3476 #define RR_WATCH(...) RR_LOG(RR_WATCH_FMT(__VA_ARGS__), __VA_ARGS__)
3477#endif // ENABLE_RR_PRINT
3478
Nicolas Capens37ed9082016-11-16 17:40:48 -05003479 class ForData
3480 {
3481 public:
3482 ForData(bool init) : loopOnce(init)
3483 {
3484 }
3485
3486 operator bool()
3487 {
3488 return loopOnce;
3489 }
3490
3491 bool operator=(bool value)
3492 {
3493 return loopOnce = value;
3494 }
3495
3496 bool setup()
3497 {
Ben Claytonac07ed82019-03-26 14:17:41 +00003498 RR_DEBUG_INFO_FLUSH();
Nicolas Capens37ed9082016-11-16 17:40:48 -05003499 if(Nucleus::getInsertBlock() != endBB)
3500 {
3501 testBB = Nucleus::createBasicBlock();
3502
3503 Nucleus::createBr(testBB);
3504 Nucleus::setInsertBlock(testBB);
3505
3506 return true;
3507 }
3508
3509 return false;
3510 }
3511
3512 bool test(RValue<Bool> cmp)
3513 {
3514 BasicBlock *bodyBB = Nucleus::createBasicBlock();
3515 endBB = Nucleus::createBasicBlock();
3516
3517 Nucleus::createCondBr(cmp.value, bodyBB, endBB);
3518 Nucleus::setInsertBlock(bodyBB);
3519
3520 return true;
3521 }
3522
3523 void end()
3524 {
3525 Nucleus::createBr(testBB);
3526 Nucleus::setInsertBlock(endBB);
3527 }
3528
3529 private:
3530 BasicBlock *testBB = nullptr;
3531 BasicBlock *endBB = nullptr;
3532 bool loopOnce = true;
3533 };
3534
3535 class IfElseData
3536 {
3537 public:
3538 IfElseData(RValue<Bool> cmp) : iteration(0)
3539 {
3540 condition = cmp.value;
3541
3542 beginBB = Nucleus::getInsertBlock();
3543 trueBB = Nucleus::createBasicBlock();
3544 falseBB = nullptr;
3545 endBB = Nucleus::createBasicBlock();
3546
3547 Nucleus::setInsertBlock(trueBB);
3548 }
3549
3550 ~IfElseData()
3551 {
3552 Nucleus::createBr(endBB);
3553
3554 Nucleus::setInsertBlock(beginBB);
3555 Nucleus::createCondBr(condition, trueBB, falseBB ? falseBB : endBB);
3556
3557 Nucleus::setInsertBlock(endBB);
3558 }
3559
3560 operator int()
3561 {
3562 return iteration;
3563 }
3564
3565 IfElseData &operator++()
3566 {
3567 ++iteration;
3568
3569 return *this;
3570 }
3571
3572 void elseClause()
3573 {
3574 Nucleus::createBr(endBB);
3575
3576 falseBB = Nucleus::createBasicBlock();
3577 Nucleus::setInsertBlock(falseBB);
3578 }
3579
3580 private:
3581 Value *condition;
3582 BasicBlock *beginBB;
3583 BasicBlock *trueBB;
3584 BasicBlock *falseBB;
3585 BasicBlock *endBB;
3586 int iteration;
3587 };
3588
Nicolas Capensb0eb3772016-10-24 17:49:13 -04003589 #define For(init, cond, inc) \
Nicolas Capens8884a232016-11-16 15:03:18 -05003590 for(ForData for__ = true; for__; for__ = false) \
3591 for(init; for__.setup() && for__.test(cond); inc, for__.end())
Nicolas Capens0bac2852016-05-07 06:09:58 -04003592
Nicolas Capensb0eb3772016-10-24 17:49:13 -04003593 #define While(cond) For((void)0, cond, (void)0)
Nicolas Capens0bac2852016-05-07 06:09:58 -04003594
Nicolas Capensb0eb3772016-10-24 17:49:13 -04003595 #define Do \
3596 { \
3597 BasicBlock *body__ = Nucleus::createBasicBlock(); \
3598 Nucleus::createBr(body__); \
3599 Nucleus::setInsertBlock(body__);
3600
3601 #define Until(cond) \
3602 BasicBlock *end__ = Nucleus::createBasicBlock(); \
3603 Nucleus::createCondBr((cond).value, end__, body__); \
3604 Nucleus::setInsertBlock(end__); \
Nicolas Capens0bac2852016-05-07 06:09:58 -04003605 }
3606
Nicolas Capens37ed9082016-11-16 17:40:48 -05003607 enum {IF_BLOCK__, ELSE_CLAUSE__, ELSE_BLOCK__, IFELSE_NUM__};
Nicolas Capens0bac2852016-05-07 06:09:58 -04003608
Nicolas Capens37ed9082016-11-16 17:40:48 -05003609 #define If(cond) \
3610 for(IfElseData ifElse__(cond); ifElse__ < IFELSE_NUM__; ++ifElse__) \
3611 if(ifElse__ == IF_BLOCK__)
3612
3613 #define Else \
3614 else if(ifElse__ == ELSE_CLAUSE__) \
3615 { \
3616 ifElse__.elseClause(); \
3617 } \
3618 else // ELSE_BLOCK__
Nicolas Capensd022e412016-09-26 13:30:14 -04003619}
3620
Nicolas Capens48461502018-08-06 14:20:45 -04003621#endif // rr_Reactor_hpp