blob: 7e6bd8628348f2113de269f42ec86e0834befa77 [file] [log] [blame]
jat@google.com134be542009-08-03 15:30:11 +00001#ifndef __H_ByteOrder
2#define __H_ByteOrder
3/*
4 * Copyright 2008 Google Inc.
5 *
6 * Licensed under the Apache License, Version 2.0 (the "License"); you may not
7 * use this file except in compliance with the License. You may obtain a copy of
8 * the License at
9 *
10 * http://www.apache.org/licenses/LICENSE-2.0
11 *
12 * Unless required by applicable law or agreed to in writing, software
13 * distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
14 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
15 * License for the specific language governing permissions and limitations under
16 * the License.
17 */
18
19#include "Platform.h"
20#include <cstring>
21
22class ByteOrder {
23private:
24 enum FloatByteOrder {
25 FLOAT_BIG_ENDIAN,
26 FLOAT_LITTLE_ENDIAN,
27 // TODO(jat): do we need to consider anything other than straight
28 // big-endian or little-endian? PDP-11 (irrelevant), ARM (probably
29 // relevant for mobile devices), MIPS (probably not relevant, but maybe)
30 // and others have some intermediate endianess. Also, FP-endianess is not
31 // necessarily the same as that for integers.
32 };
33#ifdef PLATFORM_FLOAT_ENDIANESS
34 static const FloatByteOrder floatByteOrder = PLATFORM_FLOAT_ENDIANESS;
35#else
36 FloatByteOrder floatByteOrder;
37#endif
38
39 typedef union {
40 float v;
41 char b[sizeof(float)];
42 } FloatUnion;
43
44 typedef union {
45 double v;
46 char b[sizeof(double)];
47 } DoubleUnion;
48
49 /**
50 * Copy src to dest, reversing the order of the bytes.
51 * Assumes src and dest do not overlap.
52 */
53 void memcpyrev(char* dest, const char* src, size_t n) {
54 src += n;
55 while (n-- > 0) {
56 *dest++ = *--src;
57 }
58 }
59public:
60 ByteOrder() {
61#ifndef PLATFORM_FLOAT_ENDIANESS
62 DoubleUnion u;
63 memset(u.b, 0, sizeof(u.b));
64 u.b[0] = (char) 0x80;
65 u.b[7] = (char) 0x02;
66 // TODO(jat): add more tests here if we support other endianess
67 floatByteOrder = u.v > 0 ? FLOAT_LITTLE_ENDIAN : FLOAT_BIG_ENDIAN;
68 if (Debug::level(Debug::Debugging)) {
69 std::string str = "Unknown";
70 switch (floatByteOrder) {
71 case FLOAT_LITTLE_ENDIAN:
72 str = "little-endian";
73 break;
74 case FLOAT_BIG_ENDIAN:
75 str = "big-endian";
76 break;
77 }
78 Debug::log(Debug::Debugging) << "Dynamically detected float byte order: "
79 << str << Debug::flush;
80 }
81#endif
82 }
83
84 void bytesFromDouble(double v, char* bytes) {
85 DoubleUnion u;
86 u.v = v;
87 switch (floatByteOrder) {
88 case FLOAT_LITTLE_ENDIAN:
89 memcpyrev(bytes, u.b, sizeof(u.b));
90 break;
91 case FLOAT_BIG_ENDIAN:
92 memcpy(bytes, u.b, sizeof(u.b));
93 break;
94 }
95 }
96
97 void bytesFromFloat(float v, char* bytes) {
98 FloatUnion u;
99 u.v = v;
100 switch (floatByteOrder) {
101 case FLOAT_LITTLE_ENDIAN:
102 memcpyrev(bytes, u.b, sizeof(u.b));
103 break;
104 case FLOAT_BIG_ENDIAN:
105 memcpy(bytes, u.b, sizeof(u.b));
106 break;
107 }
108 }
109
110 double doubleFromBytes(const char* bytes) {
111 DoubleUnion u;
112 switch (floatByteOrder) {
113 case FLOAT_LITTLE_ENDIAN:
114 memcpyrev(u.b, bytes, sizeof(u.b));
115 break;
116 case FLOAT_BIG_ENDIAN:
117 // TODO(jat): find a way to avoid the extra copy while keeping the
118 // compiler happy.
119 memcpy(u.b, bytes, sizeof(u.b));
120 break;
121 }
122 return u.v;
123 }
124
125 float floatFromBytes(const char* bytes) {
126 FloatUnion u;
127 switch (floatByteOrder) {
128 case FLOAT_LITTLE_ENDIAN:
129 memcpyrev(u.b, bytes, sizeof(u.b));
130 break;
131 case FLOAT_BIG_ENDIAN:
132 // TODO(jat): find a way to avoid the extra copy while keeping the
133 // compiler happy.
134 memcpy(u.b, bytes, sizeof(u.b));
135 break;
136 }
137 return u.v;
138 }
139};
140
141#endif