1 // 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 
15 #ifndef sw_Types_hpp
16 #define sw_Types_hpp
17 
18 #include <assert.h>
19 #include <limits>
20 #include <type_traits>
21 
22 // GCC warns against bitfields not fitting the entire range of an enum with a fixed underlying type of unsigned int, which gets promoted to an error with -Werror and cannot be suppressed.
23 // However, GCC already defaults to using unsigned int as the underlying type of an unscoped enum without a fixed underlying type. So we can just omit it.
24 #if defined(__GNUC__) && !defined(__clang__)
25 namespace {enum E {}; static_assert(!std::numeric_limits<std::underlying_type<E>::type>::is_signed, "expected unscoped enum whose underlying type is not fixed to be unsigned");}
26 #define ENUM_UNDERLYING_TYPE_UNSIGNED_INT
27 #else
28 #define ENUM_UNDERLYING_TYPE_UNSIGNED_INT : unsigned int
29 #endif
30 
31 #if defined(_MSC_VER)
32 	typedef signed __int8 int8_t;
33 	typedef signed __int16 int16_t;
34 	typedef signed __int32 int32_t;
35 	typedef signed __int64 int64_t;
36 	typedef unsigned __int8 uint8_t;
37 	typedef unsigned __int16 uint16_t;
38 	typedef unsigned __int32 uint32_t;
39 	typedef unsigned __int64 uint64_t;
40 	#define ALIGN(bytes, type) __declspec(align(bytes)) type
41 #else
42 	#include <stdint.h>
43 	#define ALIGN(bytes, type) type __attribute__((aligned(bytes)))
44 #endif
45 
46 namespace sw
47 {
48 	typedef ALIGN(1, uint8_t) byte;
49 	typedef ALIGN(2, uint16_t) word;
50 	typedef ALIGN(4, uint32_t) dword;
51 	typedef ALIGN(8, uint64_t) qword;
52 	typedef ALIGN(16, uint64_t) qword2[2];
53 	typedef ALIGN(4, uint8_t) byte4[4];
54 	typedef ALIGN(8, uint8_t) byte8[8];
55 	typedef ALIGN(16, uint8_t) byte16[16];
56 	typedef ALIGN(8, uint16_t) word4[4];
57 	typedef ALIGN(8, uint32_t) dword2[2];
58 	typedef ALIGN(16, uint32_t) dword4[4];
59 	typedef ALIGN(16, uint64_t) xword[2];
60 
61 	typedef ALIGN(1, int8_t) sbyte;
62 	typedef ALIGN(4, int8_t) sbyte4[4];
63 	typedef ALIGN(8, int8_t) sbyte8[8];
64 	typedef ALIGN(16, int8_t) sbyte16[16];
65 	typedef ALIGN(8, short) short4[4];
66 	typedef ALIGN(8, unsigned short) ushort4[4];
67 	typedef ALIGN(16, short) short8[8];
68 	typedef ALIGN(16, unsigned short) ushort8[8];
69 	typedef ALIGN(8, int) int2[2];
70 	typedef ALIGN(8, unsigned int) uint2[2];
71 	typedef ALIGN(16, unsigned int) uint4[4];
72 
73 	typedef ALIGN(8, float) float2[2];
74 
75 	ALIGN(16, struct int4
76 	{
77 		int x;
78 		int y;
79 		int z;
80 		int w;
81 
82 		int &operator[](int i)
83 		{
84 			return (&x)[i];
85 		}
86 
87 		const int &operator[](int i) const
88 		{
89 			return (&x)[i];
90 		}
91 
92 		bool operator!=(const int4 &rhs)
93 		{
94 			return x != rhs.x || y != rhs.y || z != rhs.z || w != rhs.w;
95 		}
96 
97 		bool operator==(const int4 &rhs)
98 		{
99 			return x == rhs.x && y == rhs.y && z == rhs.z && w == rhs.w;
100 		}
101 	});
102 
103 	ALIGN(16, struct float4
104 	{
105 		float x;
106 		float y;
107 		float z;
108 		float w;
109 
110 		float &operator[](int i)
111 		{
112 			return (&x)[i];
113 		}
114 
115 		const float &operator[](int i) const
116 		{
117 			return (&x)[i];
118 		}
119 
120 		bool operator!=(const float4 &rhs)
121 		{
122 			return x != rhs.x || y != rhs.y || z != rhs.z || w != rhs.w;
123 		}
124 
125 		bool operator==(const float4 &rhs)
126 		{
127 			return x == rhs.x && y == rhs.y && z == rhs.z && w == rhs.w;
128 		}
129 	});
130 
vector(float x,float y,float z,float w)131 	inline float4 vector(float x, float y, float z, float w)
132 	{
133 		float4 v;
134 
135 		v.x = x;
136 		v.y = y;
137 		v.z = z;
138 		v.w = w;
139 
140 		return v;
141 	}
142 
replicate(float f)143 	inline float4 replicate(float f)
144 	{
145 		float4 v;
146 
147 		v.x = f;
148 		v.y = f;
149 		v.z = f;
150 		v.w = f;
151 
152 		return v;
153 	}
154 
155 	template <int limit> class BoundedIndex
156 	{
157 	public:
BoundedIndex(int index)158 		BoundedIndex(int index) : index(index) {}
159 
operator ++(int)160 		inline int operator++(int) { return index++; }
operator --(int)161 		inline int operator--(int) { return index--; }
operator =(int i)162 		inline void operator=(int i) { index = i; }
163 
operator ==(int i)164 		inline bool operator==(int i) { return index == i; }
operator !=(int i)165 		inline bool operator!=(int i) { return index != i; }
operator <(int i)166 		inline bool operator<(int i) { return index < i; }
operator >(int i)167 		inline bool operator>(int i) { return index > i; }
operator <=(int i)168 		inline bool operator<=(int i) { return index <= i; }
operator >=(int i)169 		inline bool operator>=(int i) { return index >= i; }
170 
operator int()171 		inline operator int()
172 		{
173 			if(index < 0)
174 			{
175 #if !defined(NDEBUG) || defined(DCHECK_ALWAYS_ON)
176 				assert(false);
177 #endif
178 				return 0;
179 			}
180 			else if(index >= limit)
181 			{
182 #if !defined(NDEBUG) || defined(DCHECK_ALWAYS_ON)
183 				assert(false);
184 #endif
185 				return limit - 1;
186 			}
187 
188 			return index;
189 		}
190 
191 	private:
192 		int index = 0;
193 	};
194 
195 	// The OFFSET macro is a generalization of the offsetof() macro defined in <cstddef>.
196 	// It allows e.g. getting the offset of array elements, even when indexed dynamically.
197 	// We cast the address '32' and subtract it again, because null-dereference is undefined behavior.
198 	#define OFFSET(s,m) ((int)(size_t)&reinterpret_cast<const volatile char&>((((s*)32)->m)) - 32)
199 }
200 
201 #endif   // sw_Types_hpp
202