/
SDL_endian.h
235 lines (215 loc) · 6.4 KB
1
2
/*
SDL - Simple DirectMedia Layer
3
Copyright (C) 1997-2012 Sam Lantinga
4
5
This library is free software; you can redistribute it and/or
6
modify it under the terms of the GNU Lesser General Public
7
License as published by the Free Software Foundation; either
8
version 2.1 of the License, or (at your option) any later version.
9
10
11
12
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13
Lesser General Public License for more details.
14
15
16
17
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
18
19
Sam Lantinga
20
slouken@libsdl.org
21
22
*/
23
24
25
26
/**
* @file SDL_endian.h
* Functions for reading and writing endian-specific values
*/
27
28
29
30
#ifndef _SDL_endian_h
#define _SDL_endian_h
31
32
#include "SDL_stdinc.h"
33
34
35
36
/** @name SDL_ENDIANs
* The two types of endianness
*/
/*@{*/
37
38
#define SDL_LIL_ENDIAN 1234
#define SDL_BIG_ENDIAN 4321
40
41
#ifndef SDL_BYTEORDER /* Not defined in SDL_config.h? */
42
43
44
45
#ifdef __linux__
#include <endian.h>
#define SDL_BYTEORDER __BYTE_ORDER
#else /* __linux __ */
46
#if defined(__hppa__) || \
47
defined(__m68k__) || defined(mc68000) || defined(_M_M68K) || \
48
(defined(__MIPS__) && defined(__MISPEB__)) || \
49
defined(__ppc__) || defined(__POWERPC__) || defined(_M_PPC) || \
50
defined(__sparc__)
51
#define SDL_BYTEORDER SDL_BIG_ENDIAN
52
53
#else
#define SDL_BYTEORDER SDL_LIL_ENDIAN
54
#endif
55
#endif /* __linux __ */
56
#endif /* !SDL_BYTEORDER */
57
58
59
60
61
62
63
64
#include "begin_code.h"
/* Set up for C function definitions, even when using C++ */
#ifdef __cplusplus
extern "C" {
#endif
65
66
67
68
69
70
71
72
/**
* @name SDL_Swap Functions
* Use inline functions for compilers that support them, and static
* functions for those that do not. Because these functions become
* static for compilers that do not support inline functions, this
* header should only be included in files that actually use them.
*/
/*@{*/
73
#if defined(__GNUC__) && defined(__i386__) && \
74
!(__GNUC__ == 2 && __GNUC_MINOR__ <= 95 /* broken gcc version */)
75
static __inline__ Uint16 SDL_Swap16(Uint16 x)
76
{
77
78
79
80
81
82
__asm__("xchgb %b0,%h0" : "=q" (x) : "0" (x));
return x;
}
#elif defined(__GNUC__) && defined(__x86_64__)
static __inline__ Uint16 SDL_Swap16(Uint16 x)
{
83
__asm__("xchgb %b0,%h0" : "=Q" (x) : "0" (x));
84
85
return x;
}
86
#elif defined(__GNUC__) && (defined(__powerpc__) || defined(__ppc__))
87
88
static __inline__ Uint16 SDL_Swap16(Uint16 x)
{
89
int result;
90
91
__asm__("rlwimi %0,%2,8,16,23" : "=&r" (result) : "0" (x >> 8), "r" (x));
92
return (Uint16)result;
93
}
94
#elif defined(__GNUC__) && (defined(__m68k__) && !defined(__mcoldfire__))
95
96
97
98
99
static __inline__ Uint16 SDL_Swap16(Uint16 x)
{
__asm__("rorw #8,%0" : "=d" (x) : "0" (x) : "cc");
return x;
}
100
101
102
103
104
105
#elif defined(__WATCOMC__) && defined(__386__)
extern _inline Uint16 SDL_Swap16(Uint16);
#pragma aux SDL_Swap16 = \
"xchg al, ah" \
parm [ax] \
modify [ax];
106
#else
107
static __inline__ Uint16 SDL_Swap16(Uint16 x) {
108
return SDL_static_cast(Uint16, ((x<<8)|(x>>8)));
109
110
}
#endif
111
112
113
#if defined(__GNUC__) && defined(__i386__) && \
!(__GNUC__ == 2 && __GNUC_MINOR__ <= 95 /* broken gcc version */)
114
static __inline__ Uint32 SDL_Swap32(Uint32 x)
115
{
116
117
118
119
120
121
122
123
124
__asm__("bswap %0" : "=r" (x) : "0" (x));
return x;
}
#elif defined(__GNUC__) && defined(__x86_64__)
static __inline__ Uint32 SDL_Swap32(Uint32 x)
{
__asm__("bswapl %0" : "=r" (x) : "0" (x));
return x;
}
125
#elif defined(__GNUC__) && (defined(__powerpc__) || defined(__ppc__))
126
127
128
129
130
131
132
133
static __inline__ Uint32 SDL_Swap32(Uint32 x)
{
Uint32 result;
__asm__("rlwimi %0,%2,24,16,23" : "=&r" (result) : "0" (x>>24), "r" (x));
__asm__("rlwimi %0,%2,8,8,15" : "=&r" (result) : "0" (result), "r" (x));
__asm__("rlwimi %0,%2,24,0,7" : "=&r" (result) : "0" (result), "r" (x));
return result;
134
}
135
#elif defined(__GNUC__) && (defined(__m68k__) && !defined(__mcoldfire__))
136
static __inline__ Uint32 SDL_Swap32(Uint32 x)
137
{
138
__asm__("rorw #8,%0\n\tswap %0\n\trorw #8,%0" : "=d" (x) : "0" (x) : "cc");
139
140
return x;
}
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
#elif defined(__WATCOMC__) && defined(__386__)
extern _inline Uint32 SDL_Swap32(Uint32);
#ifndef __SW_3 /* 486+ */
#pragma aux SDL_Swap32 = \
"bswap eax" \
parm [eax] \
modify [eax];
#else /* 386-only */
#pragma aux SDL_Swap32 = \
"xchg al, ah" \
"ror eax, 16" \
"xchg al, ah" \
parm [eax] \
modify [eax];
#endif
156
#else
157
static __inline__ Uint32 SDL_Swap32(Uint32 x) {
158
return SDL_static_cast(Uint32, ((x<<24)|((x<<8)&0x00FF0000)|((x>>8)&0x0000FF00)|(x>>24)));
159
160
}
#endif
161
162
#ifdef SDL_HAS_64BIT_TYPE
163
164
#if defined(__GNUC__) && defined(__i386__) && \
!(__GNUC__ == 2 && __GNUC_MINOR__ <= 95 /* broken gcc version */)
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
static __inline__ Uint64 SDL_Swap64(Uint64 x)
{
union {
struct { Uint32 a,b; } s;
Uint64 u;
} v;
v.u = x;
__asm__("bswapl %0 ; bswapl %1 ; xchgl %0,%1"
: "=r" (v.s.a), "=r" (v.s.b)
: "0" (v.s.a), "1" (v.s.b));
return v.u;
}
#elif defined(__GNUC__) && defined(__x86_64__)
static __inline__ Uint64 SDL_Swap64(Uint64 x)
{
__asm__("bswapq %0" : "=r" (x) : "0" (x));
return x;
}
#else
static __inline__ Uint64 SDL_Swap64(Uint64 x)
{
186
187
188
Uint32 hi, lo;
/* Separate into high and low 32-bit values and swap them */
189
lo = SDL_static_cast(Uint32, x & 0xFFFFFFFF);
190
x >>= 32;
191
hi = SDL_static_cast(Uint32, x & 0xFFFFFFFF);
192
193
194
x = SDL_Swap32(lo);
x <<= 32;
x |= SDL_Swap32(hi);
195
return (x);
196
197
198
199
}
#endif
#else
/* This is mainly to keep compilers from complaining in SDL code.
200
201
202
* If there is no real 64-bit datatype, then compilers will complain about
* the fake 64-bit datatype that SDL provides when it compiles user code.
*/
203
204
#define SDL_Swap64(X) (X)
#endif /* SDL_HAS_64BIT_TYPE */
206
207
208
209
210
211
/**
* @name SDL_SwapLE and SDL_SwapBE Functions
* Byteswap item from the specified endianness to the native endianness
*/
/*@{*/
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
#if SDL_BYTEORDER == SDL_LIL_ENDIAN
#define SDL_SwapLE16(X) (X)
#define SDL_SwapLE32(X) (X)
#define SDL_SwapLE64(X) (X)
#define SDL_SwapBE16(X) SDL_Swap16(X)
#define SDL_SwapBE32(X) SDL_Swap32(X)
#define SDL_SwapBE64(X) SDL_Swap64(X)
#else
#define SDL_SwapLE16(X) SDL_Swap16(X)
#define SDL_SwapLE32(X) SDL_Swap32(X)
#define SDL_SwapLE64(X) SDL_Swap64(X)
#define SDL_SwapBE16(X) (X)
#define SDL_SwapBE32(X) (X)
#define SDL_SwapBE64(X) (X)
#endif
228
229
230
231
232
233
234
235
/* Ends C function definitions when using C++ */
#ifdef __cplusplus
}
#endif
#include "close_code.h"
#endif /* _SDL_endian_h */