LLVM 22.0.0git
Endian.h
Go to the documentation of this file.
1//===- Endian.h - Utilities for IO with endian specific data ----*- C++ -*-===//
2//
3// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6//
7//===----------------------------------------------------------------------===//
8//
9// This file declares generic functions to read and write endian specific data.
10//
11//===----------------------------------------------------------------------===//
12
13#ifndef LLVM_SUPPORT_ENDIAN_H
14#define LLVM_SUPPORT_ENDIAN_H
15
16#include "llvm/ADT/bit.h"
19#include <cassert>
20#include <cstddef>
21#include <cstdint>
22#include <cstring>
23#include <type_traits>
24
25namespace llvm {
26namespace support {
27
28// These are named values for common alignments.
29enum {aligned = 0, unaligned = 1};
30
31namespace detail {
32
33/// ::value is either alignment, or alignof(T) if alignment is 0.
34template<class T, int alignment>
36 enum { value = alignment == 0 ? alignof(T) : alignment };
37};
38
39} // end namespace detail
40
41namespace endian {
42
43template <typename value_type>
44[[nodiscard]] inline value_type byte_swap(value_type value, endianness endian) {
46 sys::swapByteOrder(value);
47 return value;
48}
49
50/// Swap the bytes of value to match the given endianness.
51template <typename value_type, endianness endian>
52[[nodiscard]] inline value_type byte_swap(value_type value) {
53 return byte_swap(value, endian);
54}
55
56/// Read a value of a particular endianness from memory.
57template <typename value_type, std::size_t alignment = unaligned>
58[[nodiscard]] inline value_type read(const void *memory, endianness endian) {
59 value_type ret;
60
61 memcpy(static_cast<void *>(&ret),
64 sizeof(value_type));
65 return byte_swap<value_type>(ret, endian);
66}
67
68template <typename value_type, endianness endian, std::size_t alignment>
69[[nodiscard]] inline value_type read(const void *memory) {
70 return read<value_type, alignment>(memory, endian);
71}
72
73/// Read a value of a particular endianness from a buffer, and increment the
74/// buffer past that value.
75template <typename value_type, std::size_t alignment = unaligned,
76 typename CharT>
77[[nodiscard]] inline value_type readNext(const CharT *&memory,
79 value_type ret = read<value_type, alignment>(memory, endian);
80 memory += sizeof(value_type);
81 return ret;
82}
83
84template <typename value_type, endianness endian,
85 std::size_t alignment = unaligned, typename CharT>
86[[nodiscard]] inline value_type readNext(const CharT *&memory) {
88}
89
90/// Write a value to memory with a particular endianness.
91template <typename value_type, std::size_t alignment = unaligned>
92inline void write(void *memory, value_type value, endianness endian) {
93 value = byte_swap<value_type>(value, endian);
96 &value, sizeof(value_type));
97}
98
99template <typename value_type, endianness endian, std::size_t alignment>
100LLVM_DEPRECATED("Pass endian as a function argument instead", "write")
101inline void write(void *memory, value_type value) {
102 write<value_type, alignment>(memory, value, endian);
103}
104
105/// Write a value of a particular endianness, and increment the buffer past that
106/// value.
107template <typename value_type, std::size_t alignment = unaligned,
108 typename CharT>
109inline void writeNext(CharT *&memory, value_type value, endianness endian) {
110 write(memory, value, endian);
111 memory += sizeof(value_type);
112}
113
114template <typename value_type, endianness endian,
115 std::size_t alignment = unaligned, typename CharT>
116inline void writeNext(CharT *&memory, value_type value) {
118}
119
120template <typename value_type>
121using make_unsigned_t = std::make_unsigned_t<value_type>;
122
123/// Read a value of a particular endianness from memory, for a location
124/// that starts at the given bit offset within the first byte.
125template <typename value_type, endianness endian, std::size_t alignment>
126[[nodiscard]] inline value_type readAtBitAlignment(const void *memory,
127 uint64_t startBit) {
128 assert(startBit < 8);
129 if (startBit == 0)
131 else {
132 // Read two values and compose the result from them.
133 value_type val[2];
134 memcpy(&val[0],
137 sizeof(value_type) * 2);
138 val[0] = byte_swap<value_type, endian>(val[0]);
139 val[1] = byte_swap<value_type, endian>(val[1]);
140
141 // Shift bits from the lower value into place.
142 make_unsigned_t<value_type> lowerVal = val[0] >> startBit;
143 // Mask off upper bits after right shift in case of signed type.
144 make_unsigned_t<value_type> numBitsFirstVal =
145 (sizeof(value_type) * 8) - startBit;
146 lowerVal &= ((make_unsigned_t<value_type>)1 << numBitsFirstVal) - 1;
147
148 // Get the bits from the upper value.
150 val[1] & (((make_unsigned_t<value_type>)1 << startBit) - 1);
151 // Shift them in to place.
152 upperVal <<= numBitsFirstVal;
153
154 return lowerVal | upperVal;
155 }
156}
157
158/// Write a value to memory with a particular endianness, for a location
159/// that starts at the given bit offset within the first byte.
160template <typename value_type, endianness endian, std::size_t alignment>
161inline void writeAtBitAlignment(void *memory, value_type value,
162 uint64_t startBit) {
163 assert(startBit < 8);
164 if (startBit == 0)
165 write<value_type, alignment>(memory, value, endian);
166 else {
167 // Read two values and shift the result into them.
168 value_type val[2];
169 memcpy(&val[0],
172 sizeof(value_type) * 2);
173 val[0] = byte_swap<value_type, endian>(val[0]);
174 val[1] = byte_swap<value_type, endian>(val[1]);
175
176 // Mask off any existing bits in the upper part of the lower value that
177 // we want to replace.
178 val[0] &= ((make_unsigned_t<value_type>)1 << startBit) - 1;
179 make_unsigned_t<value_type> numBitsFirstVal =
180 (sizeof(value_type) * 8) - startBit;
181 make_unsigned_t<value_type> lowerVal = value;
182 if (startBit > 0) {
183 // Mask off the upper bits in the new value that are not going to go into
184 // the lower value. This avoids a left shift of a negative value, which
185 // is undefined behavior.
186 lowerVal &= (((make_unsigned_t<value_type>)1 << numBitsFirstVal) - 1);
187 // Now shift the new bits into place
188 lowerVal <<= startBit;
189 }
190 val[0] |= lowerVal;
191
192 // Mask off any existing bits in the lower part of the upper value that
193 // we want to replace.
194 val[1] &= ~(((make_unsigned_t<value_type>)1 << startBit) - 1);
195 // Next shift the bits that go into the upper value into position.
196 make_unsigned_t<value_type> upperVal = value >> numBitsFirstVal;
197 // Mask off upper bits after right shift in case of signed type.
198 upperVal &= ((make_unsigned_t<value_type>)1 << startBit) - 1;
199 val[1] |= upperVal;
200
201 // Finally, rewrite values.
202 val[0] = byte_swap<value_type, endian>(val[0]);
203 val[1] = byte_swap<value_type, endian>(val[1]);
204 memcpy(LLVM_ASSUME_ALIGNED(
206 &val[0], sizeof(value_type) * 2);
207 }
208}
209
210} // end namespace endian
211
212namespace detail {
213
214template <typename ValueType, endianness Endian, std::size_t Alignment,
218 static constexpr endianness endian = Endian;
219 static constexpr std::size_t alignment = Alignment;
220
222
223 explicit packed_endian_specific_integral(value_type val) { *this = val; }
224
227 (const void*)Value.buffer);
228 }
229 operator value_type() const { return value(); }
230
231 void operator=(value_type newValue) {
232 endian::write<value_type, alignment>((void *)Value.buffer, newValue,
233 endian);
234 }
235
237 *this = *this + newValue;
238 return *this;
239 }
240
242 *this = *this - newValue;
243 return *this;
244 }
245
247 *this = *this | newValue;
248 return *this;
249 }
250
252 *this = *this & newValue;
253 return *this;
254 }
255
256private:
257 struct {
258 alignas(ALIGN) char buffer[sizeof(value_type)];
259 } Value;
260
261public:
262 struct ref {
263 explicit ref(void *Ptr) : Ptr(Ptr) {}
264
265 operator value_type() const {
267 }
268
269 void operator=(value_type NewValue) {
271 }
272
273 private:
274 void *Ptr;
275 };
276};
277
278} // end namespace detail
279
282 unaligned>;
285 unaligned>;
288 unaligned>;
291 unaligned>;
292
295 unaligned>;
298 unaligned>;
301 unaligned>;
302
305 aligned>;
308 aligned>;
311 aligned>;
312
315 aligned>;
318 aligned>;
321 aligned>;
322
323using ubig16_t =
325 unaligned>;
326using ubig32_t =
328 unaligned>;
329using ubig64_t =
331 unaligned>;
332
333using big16_t =
335 unaligned>;
336using big32_t =
338 unaligned>;
339using big64_t =
341 unaligned>;
342
345 aligned>;
348 aligned>;
351 aligned>;
352
355 aligned>;
358 aligned>;
361 aligned>;
362
365 unaligned>;
368 unaligned>;
371 unaligned>;
372
375 unaligned>;
378 unaligned>;
381 unaligned>;
382
383template <typename T>
384using little_t =
386 unaligned>;
387template <typename T>
389 unaligned>;
390
391template <typename T>
394 aligned>;
395template <typename T>
398
399namespace endian {
400
401template <typename T, endianness E> [[nodiscard]] inline T read(const void *P) {
403}
404
405[[nodiscard]] inline uint16_t read16(const void *P, endianness E) {
406 return read<uint16_t>(P, E);
407}
408[[nodiscard]] inline uint32_t read32(const void *P, endianness E) {
409 return read<uint32_t>(P, E);
410}
411[[nodiscard]] inline uint64_t read64(const void *P, endianness E) {
412 return read<uint64_t>(P, E);
413}
414
415template <endianness E> [[nodiscard]] inline uint16_t read16(const void *P) {
416 return read<uint16_t, E>(P);
417}
418template <endianness E> [[nodiscard]] inline uint32_t read32(const void *P) {
419 return read<uint32_t, E>(P);
420}
421template <endianness E> [[nodiscard]] inline uint64_t read64(const void *P) {
422 return read<uint64_t, E>(P);
423}
424
425[[nodiscard]] inline uint16_t read16le(const void *P) {
427}
428[[nodiscard]] inline uint32_t read32le(const void *P) {
430}
431[[nodiscard]] inline uint64_t read64le(const void *P) {
433}
434[[nodiscard]] inline uint16_t read16be(const void *P) {
436}
437[[nodiscard]] inline uint32_t read32be(const void *P) {
439}
440[[nodiscard]] inline uint64_t read64be(const void *P) {
442}
443
444template <typename T, endianness E> inline void write(void *P, T V) {
446}
447
448inline void write16(void *P, uint16_t V, endianness E) {
449 write<uint16_t>(P, V, E);
450}
451inline void write32(void *P, uint32_t V, endianness E) {
452 write<uint32_t>(P, V, E);
453}
454inline void write64(void *P, uint64_t V, endianness E) {
455 write<uint64_t>(P, V, E);
456}
457
458template <endianness E> inline void write16(void *P, uint16_t V) {
460}
461template <endianness E> inline void write32(void *P, uint32_t V) {
463}
464template <endianness E> inline void write64(void *P, uint64_t V) {
466}
467
468inline void write16le(void *P, uint16_t V) {
470}
471inline void write32le(void *P, uint32_t V) {
473}
474inline void write64le(void *P, uint64_t V) {
476}
477inline void write16be(void *P, uint16_t V) {
479}
480inline void write32be(void *P, uint32_t V) {
482}
483inline void write64be(void *P, uint64_t V) {
485}
486
487} // end namespace endian
488
489} // end namespace support
490} // end namespace llvm
491
492#endif // LLVM_SUPPORT_ENDIAN_H
assert(UImm &&(UImm !=~static_cast< T >(0)) &&"Invalid immediate!")
always inline
static GCRegistry::Add< CoreCLRGC > E("coreclr", "CoreCLR-compatible GC")
#define LLVM_DEPRECATED(MSG, FIX)
Definition Compiler.h:252
#define LLVM_ASSUME_ALIGNED(p, a)
\macro LLVM_ASSUME_ALIGNED Returns a pointer with an assumed alignment.
Definition Compiler.h:504
#define T
#define P(N)
This file implements the C++20 <bit> header.
LLVM Value Representation.
Definition Value.h:75
uint64_t read64le(const void *P)
Definition Endian.h:431
value_type byte_swap(value_type value, endianness endian)
Definition Endian.h:44
uint16_t read16le(const void *P)
Definition Endian.h:425
uint32_t read32(const void *P, endianness E)
Definition Endian.h:408
void writeNext(CharT *&memory, value_type value, endianness endian)
Write a value of a particular endianness, and increment the buffer past that value.
Definition Endian.h:109
void write16be(void *P, uint16_t V)
Definition Endian.h:477
void write64le(void *P, uint64_t V)
Definition Endian.h:474
uint64_t read64be(const void *P)
Definition Endian.h:440
void write32le(void *P, uint32_t V)
Definition Endian.h:471
void write32(void *P, uint32_t V, endianness E)
Definition Endian.h:451
void writeAtBitAlignment(void *memory, value_type value, uint64_t startBit)
Write a value to memory with a particular endianness, for a location that starts at the given bit off...
Definition Endian.h:161
value_type readAtBitAlignment(const void *memory, uint64_t startBit)
Read a value of a particular endianness from memory, for a location that starts at the given bit offs...
Definition Endian.h:126
void write32be(void *P, uint32_t V)
Definition Endian.h:480
uint64_t read64(const void *P, endianness E)
Definition Endian.h:411
uint32_t read32be(const void *P)
Definition Endian.h:437
std::make_unsigned_t< value_type > make_unsigned_t
Definition Endian.h:121
void write16(void *P, uint16_t V, endianness E)
Definition Endian.h:448
void write16le(void *P, uint16_t V)
Definition Endian.h:468
void write64be(void *P, uint64_t V)
Definition Endian.h:483
value_type read(const void *memory, endianness endian)
Read a value of a particular endianness from memory.
Definition Endian.h:58
void write64(void *P, uint64_t V, endianness E)
Definition Endian.h:454
uint16_t read16be(const void *P)
Definition Endian.h:434
void write(void *memory, value_type value, endianness endian)
Write a value to memory with a particular endianness.
Definition Endian.h:92
value_type readNext(const CharT *&memory, endianness endian)
Read a value of a particular endianness from a buffer, and increment the buffer past that value.
Definition Endian.h:77
uint32_t read32le(const void *P)
Definition Endian.h:428
uint16_t read16(const void *P, endianness E)
Definition Endian.h:405
detail::packed_endian_specific_integral< int64_t, llvm::endianness::little, aligned > aligned_little64_t
Definition Endian.h:319
detail::packed_endian_specific_integral< uint64_t, llvm::endianness::little, unaligned > ulittle64_t
Definition Endian.h:289
detail::packed_endian_specific_integral< int16_t, llvm::endianness::little, unaligned > little16_t
Definition Endian.h:293
detail::packed_endian_specific_integral< int32_t, llvm::endianness::big, aligned > aligned_big32_t
Definition Endian.h:356
detail::packed_endian_specific_integral< uint32_t, llvm::endianness::little, unaligned > ulittle32_t
Definition Endian.h:286
detail::packed_endian_specific_integral< uint64_t, llvm::endianness::big, unaligned > ubig64_t
Definition Endian.h:329
detail::packed_endian_specific_integral< int16_t, llvm::endianness::little, aligned > aligned_little16_t
Definition Endian.h:313
detail::packed_endian_specific_integral< int32_t, llvm::endianness::big, unaligned > big32_t
Definition Endian.h:336
detail::packed_endian_specific_integral< int64_t, llvm::endianness::big, aligned > aligned_big64_t
Definition Endian.h:359
detail::packed_endian_specific_integral< int16_t, llvm::endianness::native, unaligned > unaligned_int16_t
Definition Endian.h:373
detail::packed_endian_specific_integral< uint8_t, llvm::endianness::little, unaligned > ulittle8_t
Definition Endian.h:280
detail::packed_endian_specific_integral< uint16_t, llvm::endianness::big, unaligned > ubig16_t
Definition Endian.h:323
detail::packed_endian_specific_integral< uint64_t, llvm::endianness::native, unaligned > unaligned_uint64_t
Definition Endian.h:369
detail::packed_endian_specific_integral< uint16_t, llvm::endianness::little, unaligned > ulittle16_t
Definition Endian.h:283
detail::packed_endian_specific_integral< int64_t, llvm::endianness::native, unaligned > unaligned_int64_t
Definition Endian.h:379
detail::packed_endian_specific_integral< uint64_t, llvm::endianness::little, aligned > aligned_ulittle64_t
Definition Endian.h:309
detail::packed_endian_specific_integral< uint64_t, llvm::endianness::big, aligned > aligned_ubig64_t
Definition Endian.h:349
detail::packed_endian_specific_integral< T, llvm::endianness::big, aligned > aligned_big_t
Definition Endian.h:396
detail::packed_endian_specific_integral< uint16_t, llvm::endianness::big, aligned > aligned_ubig16_t
Definition Endian.h:343
detail::packed_endian_specific_integral< int32_t, llvm::endianness::little, unaligned > little32_t
Definition Endian.h:296
detail::packed_endian_specific_integral< uint16_t, llvm::endianness::little, aligned > aligned_ulittle16_t
Definition Endian.h:303
detail::packed_endian_specific_integral< int16_t, llvm::endianness::big, unaligned > big16_t
Definition Endian.h:333
detail::packed_endian_specific_integral< int16_t, llvm::endianness::big, aligned > aligned_big16_t
Definition Endian.h:353
detail::packed_endian_specific_integral< T, llvm::endianness::little, aligned > aligned_little_t
Definition Endian.h:392
detail::packed_endian_specific_integral< uint32_t, llvm::endianness::big, unaligned > ubig32_t
Definition Endian.h:326
detail::packed_endian_specific_integral< int64_t, llvm::endianness::little, unaligned > little64_t
Definition Endian.h:299
detail::packed_endian_specific_integral< uint32_t, llvm::endianness::native, unaligned > unaligned_uint32_t
Definition Endian.h:366
detail::packed_endian_specific_integral< int64_t, llvm::endianness::big, unaligned > big64_t
Definition Endian.h:339
detail::packed_endian_specific_integral< int32_t, llvm::endianness::native, unaligned > unaligned_int32_t
Definition Endian.h:376
detail::packed_endian_specific_integral< uint16_t, llvm::endianness::native, unaligned > unaligned_uint16_t
Definition Endian.h:363
detail::packed_endian_specific_integral< uint32_t, llvm::endianness::big, aligned > aligned_ubig32_t
Definition Endian.h:346
detail::packed_endian_specific_integral< uint32_t, llvm::endianness::little, aligned > aligned_ulittle32_t
Definition Endian.h:306
detail::packed_endian_specific_integral< T, llvm::endianness::big, unaligned > big_t
Definition Endian.h:388
detail::packed_endian_specific_integral< T, llvm::endianness::little, unaligned > little_t
Definition Endian.h:384
detail::packed_endian_specific_integral< int32_t, llvm::endianness::little, aligned > aligned_little32_t
Definition Endian.h:316
void swapByteOrder(T &Value)
This is an optimization pass for GlobalISel generic memory operations.
PointerUnion< const Value *, const PseudoSourceValue * > ValueType
endianness
Definition bit.h:71
value is either alignment, or alignof(T) if alignment is 0.
Definition Endian.h:35
packed_endian_specific_integral & operator+=(value_type newValue)
Definition Endian.h:236
packed_endian_specific_integral & operator&=(value_type newValue)
Definition Endian.h:251
packed_endian_specific_integral & operator|=(value_type newValue)
Definition Endian.h:246
packed_endian_specific_integral & operator-=(value_type newValue)
Definition Endian.h:241