/src/spirv-tools/source/spirv_endian.cpp
Line | Count | Source (jump to first uncovered line) |
1 | | // Copyright (c) 2015-2016 The Khronos Group Inc. |
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 | | #include "source/spirv_endian.h" |
16 | | |
17 | | #include <cstring> |
18 | | |
19 | | enum { |
20 | | I32_ENDIAN_LITTLE = 0x03020100ul, |
21 | | I32_ENDIAN_BIG = 0x00010203ul, |
22 | | }; |
23 | | |
24 | | // This constant value allows the detection of the host machine's endianness. |
25 | | // Accessing it through the "value" member is valid due to C++11 section 3.10 |
26 | | // paragraph 10. |
27 | | static const union { |
28 | | unsigned char bytes[4]; |
29 | | uint32_t value; |
30 | | } o32_host_order = {{0, 1, 2, 3}}; |
31 | | |
32 | 0 | #define I32_ENDIAN_HOST (o32_host_order.value) |
33 | | |
34 | 0 | uint32_t spvFixWord(const uint32_t word, const spv_endianness_t endian) { |
35 | 0 | if ((SPV_ENDIANNESS_LITTLE == endian && I32_ENDIAN_HOST == I32_ENDIAN_BIG) || |
36 | 0 | (SPV_ENDIANNESS_BIG == endian && I32_ENDIAN_HOST == I32_ENDIAN_LITTLE)) { |
37 | 0 | return (word & 0x000000ff) << 24 | (word & 0x0000ff00) << 8 | |
38 | 0 | (word & 0x00ff0000) >> 8 | (word & 0xff000000) >> 24; |
39 | 0 | } |
40 | | |
41 | 0 | return word; |
42 | 0 | } |
43 | | |
44 | | uint64_t spvFixDoubleWord(const uint32_t low, const uint32_t high, |
45 | 0 | const spv_endianness_t endian) { |
46 | 0 | return (uint64_t(spvFixWord(high, endian)) << 32) | spvFixWord(low, endian); |
47 | 0 | } |
48 | | |
49 | | spv_result_t spvBinaryEndianness(spv_const_binary binary, |
50 | 0 | spv_endianness_t* pEndian) { |
51 | 0 | if (!binary->code || !binary->wordCount) return SPV_ERROR_INVALID_BINARY; |
52 | 0 | if (!pEndian) return SPV_ERROR_INVALID_POINTER; |
53 | | |
54 | 0 | uint8_t bytes[4]; |
55 | 0 | memcpy(bytes, binary->code, sizeof(uint32_t)); |
56 | |
|
57 | 0 | if (0x03 == bytes[0] && 0x02 == bytes[1] && 0x23 == bytes[2] && |
58 | 0 | 0x07 == bytes[3]) { |
59 | 0 | *pEndian = SPV_ENDIANNESS_LITTLE; |
60 | 0 | return SPV_SUCCESS; |
61 | 0 | } |
62 | | |
63 | 0 | if (0x07 == bytes[0] && 0x23 == bytes[1] && 0x02 == bytes[2] && |
64 | 0 | 0x03 == bytes[3]) { |
65 | 0 | *pEndian = SPV_ENDIANNESS_BIG; |
66 | 0 | return SPV_SUCCESS; |
67 | 0 | } |
68 | | |
69 | 0 | return SPV_ERROR_INVALID_BINARY; |
70 | 0 | } |
71 | | |
72 | 0 | bool spvIsHostEndian(spv_endianness_t endian) { |
73 | 0 | return ((SPV_ENDIANNESS_LITTLE == endian) && |
74 | 0 | (I32_ENDIAN_LITTLE == I32_ENDIAN_HOST)) || |
75 | 0 | ((SPV_ENDIANNESS_BIG == endian) && |
76 | 0 | (I32_ENDIAN_BIG == I32_ENDIAN_HOST)); |
77 | 0 | } |