1*61145dc2SMartin Matuska // SPDX-License-Identifier: BSD-3-Clause OR GPL-2.0-only
2c03c5b1cSMartin Matuska /*
3c03c5b1cSMartin Matuska * Copyright (c) 2016-2020, Yann Collet, Facebook, Inc.
4c03c5b1cSMartin Matuska * All rights reserved.
5c03c5b1cSMartin Matuska *
6c03c5b1cSMartin Matuska * This source code is licensed under both the BSD-style license (found in the
7c03c5b1cSMartin Matuska * LICENSE file in the root directory of this source tree) and the GPLv2 (found
8c03c5b1cSMartin Matuska * in the COPYING file in the root directory of this source tree).
9c03c5b1cSMartin Matuska * You may select, at your option, one of the above-listed licenses.
10c03c5b1cSMartin Matuska */
11c03c5b1cSMartin Matuska
12c03c5b1cSMartin Matuska #ifndef MEM_H_MODULE
13c03c5b1cSMartin Matuska #define MEM_H_MODULE
14c03c5b1cSMartin Matuska
15c03c5b1cSMartin Matuska #if defined (__cplusplus)
16c03c5b1cSMartin Matuska extern "C" {
17c03c5b1cSMartin Matuska #endif
18c03c5b1cSMartin Matuska
19c03c5b1cSMartin Matuska /*-****************************************
20c03c5b1cSMartin Matuska * Dependencies
21c03c5b1cSMartin Matuska ******************************************/
22c03c5b1cSMartin Matuska #include <stddef.h> /* size_t, ptrdiff_t */
23c03c5b1cSMartin Matuska #include <string.h> /* memcpy */
24c03c5b1cSMartin Matuska
25c03c5b1cSMartin Matuska
26c03c5b1cSMartin Matuska /*-****************************************
27c03c5b1cSMartin Matuska * Compiler specifics
28c03c5b1cSMartin Matuska ******************************************/
29c03c5b1cSMartin Matuska #if defined(_MSC_VER) /* Visual Studio */
30c03c5b1cSMartin Matuska # include <stdlib.h> /* _byteswap_ulong */
31c03c5b1cSMartin Matuska # include <intrin.h> /* _byteswap_* */
32c03c5b1cSMartin Matuska #endif
33c03c5b1cSMartin Matuska #if defined(__GNUC__)
34c03c5b1cSMartin Matuska # define MEM_STATIC static __inline __attribute__((unused))
35c03c5b1cSMartin Matuska #elif defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */)
36c03c5b1cSMartin Matuska # define MEM_STATIC static inline
37c03c5b1cSMartin Matuska #elif defined(_MSC_VER)
38c03c5b1cSMartin Matuska # define MEM_STATIC static __inline
39c03c5b1cSMartin Matuska #else
40c03c5b1cSMartin Matuska # define MEM_STATIC static /* this version may generate warnings for unused static functions; disable the relevant warning */
41c03c5b1cSMartin Matuska #endif
42c03c5b1cSMartin Matuska
43c03c5b1cSMartin Matuska #ifndef __has_builtin
44c03c5b1cSMartin Matuska # define __has_builtin(x) 0 /* compat. with non-clang compilers */
45c03c5b1cSMartin Matuska #endif
46c03c5b1cSMartin Matuska
47c03c5b1cSMartin Matuska /* code only tested on 32 and 64 bits systems */
48c03c5b1cSMartin Matuska #define MEM_STATIC_ASSERT(c) { enum { MEM_static_assert = 1/(int)(!!(c)) }; }
MEM_check(void)49c03c5b1cSMartin Matuska MEM_STATIC void MEM_check(void) { MEM_STATIC_ASSERT((sizeof(size_t)==4) || (sizeof(size_t)==8)); }
50c03c5b1cSMartin Matuska
51c03c5b1cSMartin Matuska /* detects whether we are being compiled under msan */
52c03c5b1cSMartin Matuska #if defined (__has_feature)
53c03c5b1cSMartin Matuska # if __has_feature(memory_sanitizer)
54c03c5b1cSMartin Matuska # define MEMORY_SANITIZER 1
55c03c5b1cSMartin Matuska # endif
56c03c5b1cSMartin Matuska #endif
57c03c5b1cSMartin Matuska
58c03c5b1cSMartin Matuska #if defined (MEMORY_SANITIZER)
59c03c5b1cSMartin Matuska /* Not all platforms that support msan provide sanitizers/msan_interface.h.
60c03c5b1cSMartin Matuska * We therefore declare the functions we need ourselves, rather than trying to
61c03c5b1cSMartin Matuska * include the header file... */
62c03c5b1cSMartin Matuska
63c03c5b1cSMartin Matuska #include <stdint.h> /* intptr_t */
64c03c5b1cSMartin Matuska
65c03c5b1cSMartin Matuska /* Make memory region fully initialized (without changing its contents). */
66c03c5b1cSMartin Matuska void __msan_unpoison(const volatile void *a, size_t size);
67c03c5b1cSMartin Matuska
68c03c5b1cSMartin Matuska /* Make memory region fully uninitialized (without changing its contents).
69c03c5b1cSMartin Matuska This is a legacy interface that does not update origin information. Use
70c03c5b1cSMartin Matuska __msan_allocated_memory() instead. */
71c03c5b1cSMartin Matuska void __msan_poison(const volatile void *a, size_t size);
72c03c5b1cSMartin Matuska
73c03c5b1cSMartin Matuska /* Returns the offset of the first (at least partially) poisoned byte in the
74c03c5b1cSMartin Matuska memory range, or -1 if the whole range is good. */
75c03c5b1cSMartin Matuska intptr_t __msan_test_shadow(const volatile void *x, size_t size);
76c03c5b1cSMartin Matuska #endif
77c03c5b1cSMartin Matuska
78c03c5b1cSMartin Matuska /* detects whether we are being compiled under asan */
79c03c5b1cSMartin Matuska #if defined (ZFS_ASAN_ENABLED)
80c03c5b1cSMartin Matuska # define ADDRESS_SANITIZER 1
81c03c5b1cSMartin Matuska # define ZSTD_ASAN_DONT_POISON_WORKSPACE
82c03c5b1cSMartin Matuska #endif
83c03c5b1cSMartin Matuska
84c03c5b1cSMartin Matuska #if defined (ADDRESS_SANITIZER)
85c03c5b1cSMartin Matuska /* Not all platforms that support asan provide sanitizers/asan_interface.h.
86c03c5b1cSMartin Matuska * We therefore declare the functions we need ourselves, rather than trying to
87c03c5b1cSMartin Matuska * include the header file... */
88c03c5b1cSMartin Matuska
89c03c5b1cSMartin Matuska /**
90c03c5b1cSMartin Matuska * Marks a memory region (<c>[addr, addr+size)</c>) as unaddressable.
91c03c5b1cSMartin Matuska *
92c03c5b1cSMartin Matuska * This memory must be previously allocated by your program. Instrumented
93c03c5b1cSMartin Matuska * code is forbidden from accessing addresses in this region until it is
94c03c5b1cSMartin Matuska * unpoisoned. This function is not guaranteed to poison the entire region -
95c03c5b1cSMartin Matuska * it could poison only a subregion of <c>[addr, addr+size)</c> due to ASan
96c03c5b1cSMartin Matuska * alignment restrictions.
97c03c5b1cSMartin Matuska *
98c03c5b1cSMartin Matuska * \note This function is not thread-safe because no two threads can poison or
99c03c5b1cSMartin Matuska * unpoison memory in the same memory region simultaneously.
100c03c5b1cSMartin Matuska *
101c03c5b1cSMartin Matuska * \param addr Start of memory region.
102c03c5b1cSMartin Matuska * \param size Size of memory region. */
103c03c5b1cSMartin Matuska void __asan_poison_memory_region(void const volatile *addr, size_t size);
104c03c5b1cSMartin Matuska
105c03c5b1cSMartin Matuska /**
106c03c5b1cSMartin Matuska * Marks a memory region (<c>[addr, addr+size)</c>) as addressable.
107c03c5b1cSMartin Matuska *
108c03c5b1cSMartin Matuska * This memory must be previously allocated by your program. Accessing
109c03c5b1cSMartin Matuska * addresses in this region is allowed until this region is poisoned again.
110c03c5b1cSMartin Matuska * This function could unpoison a super-region of <c>[addr, addr+size)</c> due
111c03c5b1cSMartin Matuska * to ASan alignment restrictions.
112c03c5b1cSMartin Matuska *
113c03c5b1cSMartin Matuska * \note This function is not thread-safe because no two threads can
114c03c5b1cSMartin Matuska * poison or unpoison memory in the same memory region simultaneously.
115c03c5b1cSMartin Matuska *
116c03c5b1cSMartin Matuska * \param addr Start of memory region.
117c03c5b1cSMartin Matuska * \param size Size of memory region. */
118c03c5b1cSMartin Matuska void __asan_unpoison_memory_region(void const volatile *addr, size_t size);
119c03c5b1cSMartin Matuska #endif
120c03c5b1cSMartin Matuska
121c03c5b1cSMartin Matuska
122c03c5b1cSMartin Matuska /*-**************************************************************
123c03c5b1cSMartin Matuska * Basic Types
124c03c5b1cSMartin Matuska *****************************************************************/
125c03c5b1cSMartin Matuska #if !defined (__VMS) && (defined (__cplusplus) || (defined (__STDC_VERSION__) && (__STDC_VERSION__ >= 199901L) /* C99 */) )
126c03c5b1cSMartin Matuska # include <stdint.h>
127c03c5b1cSMartin Matuska typedef uint8_t BYTE;
128c03c5b1cSMartin Matuska typedef uint16_t U16;
129c03c5b1cSMartin Matuska typedef int16_t S16;
130c03c5b1cSMartin Matuska typedef uint32_t U32;
131c03c5b1cSMartin Matuska typedef int32_t S32;
132c03c5b1cSMartin Matuska typedef uint64_t U64;
133c03c5b1cSMartin Matuska typedef int64_t S64;
134c03c5b1cSMartin Matuska #else
135c03c5b1cSMartin Matuska # include <limits.h>
136c03c5b1cSMartin Matuska #if CHAR_BIT != 8
137c03c5b1cSMartin Matuska # error "this implementation requires char to be exactly 8-bit type"
138c03c5b1cSMartin Matuska #endif
139c03c5b1cSMartin Matuska typedef unsigned char BYTE;
140c03c5b1cSMartin Matuska #if USHRT_MAX != 65535
141c03c5b1cSMartin Matuska # error "this implementation requires short to be exactly 16-bit type"
142c03c5b1cSMartin Matuska #endif
143c03c5b1cSMartin Matuska typedef unsigned short U16;
144c03c5b1cSMartin Matuska typedef signed short S16;
145c03c5b1cSMartin Matuska #if UINT_MAX != 4294967295
146c03c5b1cSMartin Matuska # error "this implementation requires int to be exactly 32-bit type"
147c03c5b1cSMartin Matuska #endif
148c03c5b1cSMartin Matuska typedef unsigned int U32;
149c03c5b1cSMartin Matuska typedef signed int S32;
150c03c5b1cSMartin Matuska /* note : there are no limits defined for long long type in C90.
151c03c5b1cSMartin Matuska * limits exist in C99, however, in such case, <stdint.h> is preferred */
152c03c5b1cSMartin Matuska typedef unsigned long long U64;
153c03c5b1cSMartin Matuska typedef signed long long S64;
154c03c5b1cSMartin Matuska #endif
155c03c5b1cSMartin Matuska
156c03c5b1cSMartin Matuska
157c03c5b1cSMartin Matuska /*-**************************************************************
158c03c5b1cSMartin Matuska * Memory I/O
159c03c5b1cSMartin Matuska *****************************************************************/
160c03c5b1cSMartin Matuska /* MEM_FORCE_MEMORY_ACCESS :
161c03c5b1cSMartin Matuska * By default, access to unaligned memory is controlled by `memcpy()`, which is safe and portable.
162c03c5b1cSMartin Matuska * Unfortunately, on some target/compiler combinations, the generated assembly is sub-optimal.
163c03c5b1cSMartin Matuska * The below switch allow to select different access method for improved performance.
164c03c5b1cSMartin Matuska * Method 0 (default) : use `memcpy()`. Safe and portable.
165c03c5b1cSMartin Matuska * Method 1 : `__packed` statement. It depends on compiler extension (i.e., not portable).
166c03c5b1cSMartin Matuska * This method is safe if your compiler supports it, and *generally* as fast or faster than `memcpy`.
167c03c5b1cSMartin Matuska * Method 2 : direct access. This method is portable but violate C standard.
168c03c5b1cSMartin Matuska * It can generate buggy code on targets depending on alignment.
169c03c5b1cSMartin Matuska * In some circumstances, it's the only known way to get the most performance (i.e. GCC + ARMv6)
170c03c5b1cSMartin Matuska * See http://fastcompression.blogspot.fr/2015/08/accessing-unaligned-memory.html for details.
171c03c5b1cSMartin Matuska * Prefer these methods in priority order (0 > 1 > 2)
172c03c5b1cSMartin Matuska */
173c03c5b1cSMartin Matuska #ifndef MEM_FORCE_MEMORY_ACCESS /* can be defined externally, on command line for example */
174c03c5b1cSMartin Matuska # if defined(__GNUC__) && ( defined(__ARM_ARCH_6__) || defined(__ARM_ARCH_6J__) || defined(__ARM_ARCH_6K__) || defined(__ARM_ARCH_6Z__) || defined(__ARM_ARCH_6ZK__) || defined(__ARM_ARCH_6T2__) )
175c03c5b1cSMartin Matuska # define MEM_FORCE_MEMORY_ACCESS 2
176c03c5b1cSMartin Matuska # elif defined(__INTEL_COMPILER) || defined(__GNUC__) || defined(__ICCARM__)
177c03c5b1cSMartin Matuska # define MEM_FORCE_MEMORY_ACCESS 1
178c03c5b1cSMartin Matuska # endif
179c03c5b1cSMartin Matuska #endif
180c03c5b1cSMartin Matuska
MEM_32bits(void)181c03c5b1cSMartin Matuska MEM_STATIC unsigned MEM_32bits(void) { return sizeof(size_t)==4; }
MEM_64bits(void)182c03c5b1cSMartin Matuska MEM_STATIC unsigned MEM_64bits(void) { return sizeof(size_t)==8; }
183c03c5b1cSMartin Matuska
MEM_isLittleEndian(void)184c03c5b1cSMartin Matuska MEM_STATIC unsigned MEM_isLittleEndian(void)
185c03c5b1cSMartin Matuska {
186c03c5b1cSMartin Matuska const union { U32 u; BYTE c[4]; } one = { 1 }; /* don't use static : performance detrimental */
187c03c5b1cSMartin Matuska return one.c[0];
188c03c5b1cSMartin Matuska }
189c03c5b1cSMartin Matuska
190c03c5b1cSMartin Matuska #if defined(MEM_FORCE_MEMORY_ACCESS) && (MEM_FORCE_MEMORY_ACCESS==2)
191c03c5b1cSMartin Matuska
192c03c5b1cSMartin Matuska /* violates C standard, by lying on structure alignment.
193c03c5b1cSMartin Matuska Only use if no other choice to achieve best performance on target platform */
MEM_read16(const void * memPtr)194c03c5b1cSMartin Matuska MEM_STATIC U16 MEM_read16(const void* memPtr) { return *(const U16*) memPtr; }
MEM_read32(const void * memPtr)195c03c5b1cSMartin Matuska MEM_STATIC U32 MEM_read32(const void* memPtr) { return *(const U32*) memPtr; }
MEM_read64(const void * memPtr)196c03c5b1cSMartin Matuska MEM_STATIC U64 MEM_read64(const void* memPtr) { return *(const U64*) memPtr; }
MEM_readST(const void * memPtr)197c03c5b1cSMartin Matuska MEM_STATIC size_t MEM_readST(const void* memPtr) { return *(const size_t*) memPtr; }
198c03c5b1cSMartin Matuska
MEM_write16(void * memPtr,U16 value)199c03c5b1cSMartin Matuska MEM_STATIC void MEM_write16(void* memPtr, U16 value) { *(U16*)memPtr = value; }
MEM_write32(void * memPtr,U32 value)200c03c5b1cSMartin Matuska MEM_STATIC void MEM_write32(void* memPtr, U32 value) { *(U32*)memPtr = value; }
MEM_write64(void * memPtr,U64 value)201c03c5b1cSMartin Matuska MEM_STATIC void MEM_write64(void* memPtr, U64 value) { *(U64*)memPtr = value; }
202c03c5b1cSMartin Matuska
203c03c5b1cSMartin Matuska #elif defined(MEM_FORCE_MEMORY_ACCESS) && (MEM_FORCE_MEMORY_ACCESS==1)
204c03c5b1cSMartin Matuska
205c03c5b1cSMartin Matuska /* __pack instructions are safer, but compiler specific, hence potentially problematic for some compilers */
206c03c5b1cSMartin Matuska /* currently only defined for gcc and icc */
207c03c5b1cSMartin Matuska #if defined(_MSC_VER) || (defined(__INTEL_COMPILER) && defined(WIN32))
208c03c5b1cSMartin Matuska __pragma( pack(push, 1) )
209c03c5b1cSMartin Matuska typedef struct { U16 v; } unalign16;
210c03c5b1cSMartin Matuska typedef struct { U32 v; } unalign32;
211c03c5b1cSMartin Matuska typedef struct { U64 v; } unalign64;
212c03c5b1cSMartin Matuska typedef struct { size_t v; } unalignArch;
__pragma(pack (pop))213c03c5b1cSMartin Matuska __pragma( pack(pop) )
214c03c5b1cSMartin Matuska #else
215c03c5b1cSMartin Matuska typedef struct { U16 v; } __attribute__((packed)) unalign16;
216c03c5b1cSMartin Matuska typedef struct { U32 v; } __attribute__((packed)) unalign32;
217c03c5b1cSMartin Matuska typedef struct { U64 v; } __attribute__((packed)) unalign64;
218c03c5b1cSMartin Matuska typedef struct { size_t v; } __attribute__((packed)) unalignArch;
219c03c5b1cSMartin Matuska #endif
220c03c5b1cSMartin Matuska
221c03c5b1cSMartin Matuska MEM_STATIC U16 MEM_read16(const void* ptr) { return ((const unalign16*)ptr)->v; }
MEM_read32(const void * ptr)222c03c5b1cSMartin Matuska MEM_STATIC U32 MEM_read32(const void* ptr) { return ((const unalign32*)ptr)->v; }
MEM_read64(const void * ptr)223c03c5b1cSMartin Matuska MEM_STATIC U64 MEM_read64(const void* ptr) { return ((const unalign64*)ptr)->v; }
MEM_readST(const void * ptr)224c03c5b1cSMartin Matuska MEM_STATIC size_t MEM_readST(const void* ptr) { return ((const unalignArch*)ptr)->v; }
225c03c5b1cSMartin Matuska
MEM_write16(void * memPtr,U16 value)226c03c5b1cSMartin Matuska MEM_STATIC void MEM_write16(void* memPtr, U16 value) { ((unalign16*)memPtr)->v = value; }
MEM_write32(void * memPtr,U32 value)227c03c5b1cSMartin Matuska MEM_STATIC void MEM_write32(void* memPtr, U32 value) { ((unalign32*)memPtr)->v = value; }
MEM_write64(void * memPtr,U64 value)228c03c5b1cSMartin Matuska MEM_STATIC void MEM_write64(void* memPtr, U64 value) { ((unalign64*)memPtr)->v = value; }
229c03c5b1cSMartin Matuska
230c03c5b1cSMartin Matuska #else
231c03c5b1cSMartin Matuska
232c03c5b1cSMartin Matuska /* default method, safe and standard.
233c03c5b1cSMartin Matuska can sometimes prove slower */
234c03c5b1cSMartin Matuska
MEM_read16(const void * memPtr)235c03c5b1cSMartin Matuska MEM_STATIC U16 MEM_read16(const void* memPtr)
236c03c5b1cSMartin Matuska {
237c03c5b1cSMartin Matuska U16 val; memcpy(&val, memPtr, sizeof(val)); return val;
238c03c5b1cSMartin Matuska }
239c03c5b1cSMartin Matuska
MEM_read32(const void * memPtr)240c03c5b1cSMartin Matuska MEM_STATIC U32 MEM_read32(const void* memPtr)
241c03c5b1cSMartin Matuska {
242c03c5b1cSMartin Matuska U32 val; memcpy(&val, memPtr, sizeof(val)); return val;
243c03c5b1cSMartin Matuska }
244c03c5b1cSMartin Matuska
MEM_read64(const void * memPtr)245c03c5b1cSMartin Matuska MEM_STATIC U64 MEM_read64(const void* memPtr)
246c03c5b1cSMartin Matuska {
247c03c5b1cSMartin Matuska U64 val; memcpy(&val, memPtr, sizeof(val)); return val;
248c03c5b1cSMartin Matuska }
249c03c5b1cSMartin Matuska
MEM_readST(const void * memPtr)250c03c5b1cSMartin Matuska MEM_STATIC size_t MEM_readST(const void* memPtr)
251c03c5b1cSMartin Matuska {
252c03c5b1cSMartin Matuska size_t val; memcpy(&val, memPtr, sizeof(val)); return val;
253c03c5b1cSMartin Matuska }
254c03c5b1cSMartin Matuska
MEM_write16(void * memPtr,U16 value)255c03c5b1cSMartin Matuska MEM_STATIC void MEM_write16(void* memPtr, U16 value)
256c03c5b1cSMartin Matuska {
257c03c5b1cSMartin Matuska memcpy(memPtr, &value, sizeof(value));
258c03c5b1cSMartin Matuska }
259c03c5b1cSMartin Matuska
MEM_write32(void * memPtr,U32 value)260c03c5b1cSMartin Matuska MEM_STATIC void MEM_write32(void* memPtr, U32 value)
261c03c5b1cSMartin Matuska {
262c03c5b1cSMartin Matuska memcpy(memPtr, &value, sizeof(value));
263c03c5b1cSMartin Matuska }
264c03c5b1cSMartin Matuska
MEM_write64(void * memPtr,U64 value)265c03c5b1cSMartin Matuska MEM_STATIC void MEM_write64(void* memPtr, U64 value)
266c03c5b1cSMartin Matuska {
267c03c5b1cSMartin Matuska memcpy(memPtr, &value, sizeof(value));
268c03c5b1cSMartin Matuska }
269c03c5b1cSMartin Matuska
270c03c5b1cSMartin Matuska #endif /* MEM_FORCE_MEMORY_ACCESS */
271c03c5b1cSMartin Matuska
MEM_swap32(U32 in)272c03c5b1cSMartin Matuska MEM_STATIC U32 MEM_swap32(U32 in)
273c03c5b1cSMartin Matuska {
274c03c5b1cSMartin Matuska #if defined(_MSC_VER) /* Visual Studio */
275c03c5b1cSMartin Matuska return _byteswap_ulong(in);
276c03c5b1cSMartin Matuska #elif (defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403)) \
277c03c5b1cSMartin Matuska || (defined(__clang__) && __has_builtin(__builtin_bswap32))
278c03c5b1cSMartin Matuska return __builtin_bswap32(in);
279c03c5b1cSMartin Matuska #else
280c03c5b1cSMartin Matuska return ((in << 24) & 0xff000000 ) |
281c03c5b1cSMartin Matuska ((in << 8) & 0x00ff0000 ) |
282c03c5b1cSMartin Matuska ((in >> 8) & 0x0000ff00 ) |
283c03c5b1cSMartin Matuska ((in >> 24) & 0x000000ff );
284c03c5b1cSMartin Matuska #endif
285c03c5b1cSMartin Matuska }
286c03c5b1cSMartin Matuska
MEM_swap64(U64 in)287c03c5b1cSMartin Matuska MEM_STATIC U64 MEM_swap64(U64 in)
288c03c5b1cSMartin Matuska {
289c03c5b1cSMartin Matuska #if defined(_MSC_VER) /* Visual Studio */
290c03c5b1cSMartin Matuska return _byteswap_uint64(in);
291c03c5b1cSMartin Matuska #elif (defined (__GNUC__) && (__GNUC__ * 100 + __GNUC_MINOR__ >= 403)) \
292c03c5b1cSMartin Matuska || (defined(__clang__) && __has_builtin(__builtin_bswap64))
293c03c5b1cSMartin Matuska return __builtin_bswap64(in);
294c03c5b1cSMartin Matuska #else
295c03c5b1cSMartin Matuska return ((in << 56) & 0xff00000000000000ULL) |
296c03c5b1cSMartin Matuska ((in << 40) & 0x00ff000000000000ULL) |
297c03c5b1cSMartin Matuska ((in << 24) & 0x0000ff0000000000ULL) |
298c03c5b1cSMartin Matuska ((in << 8) & 0x000000ff00000000ULL) |
299c03c5b1cSMartin Matuska ((in >> 8) & 0x00000000ff000000ULL) |
300c03c5b1cSMartin Matuska ((in >> 24) & 0x0000000000ff0000ULL) |
301c03c5b1cSMartin Matuska ((in >> 40) & 0x000000000000ff00ULL) |
302c03c5b1cSMartin Matuska ((in >> 56) & 0x00000000000000ffULL);
303c03c5b1cSMartin Matuska #endif
304c03c5b1cSMartin Matuska }
305c03c5b1cSMartin Matuska
MEM_swapST(size_t in)306c03c5b1cSMartin Matuska MEM_STATIC size_t MEM_swapST(size_t in)
307c03c5b1cSMartin Matuska {
308c03c5b1cSMartin Matuska if (MEM_32bits())
309c03c5b1cSMartin Matuska return (size_t)MEM_swap32((U32)in);
310c03c5b1cSMartin Matuska else
311c03c5b1cSMartin Matuska return (size_t)MEM_swap64((U64)in);
312c03c5b1cSMartin Matuska }
313c03c5b1cSMartin Matuska
314c03c5b1cSMartin Matuska /*=== Little endian r/w ===*/
315c03c5b1cSMartin Matuska
MEM_readLE16(const void * memPtr)316c03c5b1cSMartin Matuska MEM_STATIC U16 MEM_readLE16(const void* memPtr)
317c03c5b1cSMartin Matuska {
318c03c5b1cSMartin Matuska if (MEM_isLittleEndian())
319c03c5b1cSMartin Matuska return MEM_read16(memPtr);
320c03c5b1cSMartin Matuska else {
321c03c5b1cSMartin Matuska const BYTE* p = (const BYTE*)memPtr;
322c03c5b1cSMartin Matuska return (U16)(p[0] + (p[1]<<8));
323c03c5b1cSMartin Matuska }
324c03c5b1cSMartin Matuska }
325c03c5b1cSMartin Matuska
MEM_writeLE16(void * memPtr,U16 val)326c03c5b1cSMartin Matuska MEM_STATIC void MEM_writeLE16(void* memPtr, U16 val)
327c03c5b1cSMartin Matuska {
328c03c5b1cSMartin Matuska if (MEM_isLittleEndian()) {
329c03c5b1cSMartin Matuska MEM_write16(memPtr, val);
330c03c5b1cSMartin Matuska } else {
331c03c5b1cSMartin Matuska BYTE* p = (BYTE*)memPtr;
332c03c5b1cSMartin Matuska p[0] = (BYTE)val;
333c03c5b1cSMartin Matuska p[1] = (BYTE)(val>>8);
334c03c5b1cSMartin Matuska }
335c03c5b1cSMartin Matuska }
336c03c5b1cSMartin Matuska
MEM_readLE24(const void * memPtr)337c03c5b1cSMartin Matuska MEM_STATIC U32 MEM_readLE24(const void* memPtr)
338c03c5b1cSMartin Matuska {
339c03c5b1cSMartin Matuska return MEM_readLE16(memPtr) + (((const BYTE*)memPtr)[2] << 16);
340c03c5b1cSMartin Matuska }
341c03c5b1cSMartin Matuska
MEM_writeLE24(void * memPtr,U32 val)342c03c5b1cSMartin Matuska MEM_STATIC void MEM_writeLE24(void* memPtr, U32 val)
343c03c5b1cSMartin Matuska {
344c03c5b1cSMartin Matuska MEM_writeLE16(memPtr, (U16)val);
345c03c5b1cSMartin Matuska ((BYTE*)memPtr)[2] = (BYTE)(val>>16);
346c03c5b1cSMartin Matuska }
347c03c5b1cSMartin Matuska
MEM_readLE32(const void * memPtr)348c03c5b1cSMartin Matuska MEM_STATIC U32 MEM_readLE32(const void* memPtr)
349c03c5b1cSMartin Matuska {
350c03c5b1cSMartin Matuska if (MEM_isLittleEndian())
351c03c5b1cSMartin Matuska return MEM_read32(memPtr);
352c03c5b1cSMartin Matuska else
353c03c5b1cSMartin Matuska return MEM_swap32(MEM_read32(memPtr));
354c03c5b1cSMartin Matuska }
355c03c5b1cSMartin Matuska
MEM_writeLE32(void * memPtr,U32 val32)356c03c5b1cSMartin Matuska MEM_STATIC void MEM_writeLE32(void* memPtr, U32 val32)
357c03c5b1cSMartin Matuska {
358c03c5b1cSMartin Matuska if (MEM_isLittleEndian())
359c03c5b1cSMartin Matuska MEM_write32(memPtr, val32);
360c03c5b1cSMartin Matuska else
361c03c5b1cSMartin Matuska MEM_write32(memPtr, MEM_swap32(val32));
362c03c5b1cSMartin Matuska }
363c03c5b1cSMartin Matuska
MEM_readLE64(const void * memPtr)364c03c5b1cSMartin Matuska MEM_STATIC U64 MEM_readLE64(const void* memPtr)
365c03c5b1cSMartin Matuska {
366c03c5b1cSMartin Matuska if (MEM_isLittleEndian())
367c03c5b1cSMartin Matuska return MEM_read64(memPtr);
368c03c5b1cSMartin Matuska else
369c03c5b1cSMartin Matuska return MEM_swap64(MEM_read64(memPtr));
370c03c5b1cSMartin Matuska }
371c03c5b1cSMartin Matuska
MEM_writeLE64(void * memPtr,U64 val64)372c03c5b1cSMartin Matuska MEM_STATIC void MEM_writeLE64(void* memPtr, U64 val64)
373c03c5b1cSMartin Matuska {
374c03c5b1cSMartin Matuska if (MEM_isLittleEndian())
375c03c5b1cSMartin Matuska MEM_write64(memPtr, val64);
376c03c5b1cSMartin Matuska else
377c03c5b1cSMartin Matuska MEM_write64(memPtr, MEM_swap64(val64));
378c03c5b1cSMartin Matuska }
379c03c5b1cSMartin Matuska
MEM_readLEST(const void * memPtr)380c03c5b1cSMartin Matuska MEM_STATIC size_t MEM_readLEST(const void* memPtr)
381c03c5b1cSMartin Matuska {
382c03c5b1cSMartin Matuska if (MEM_32bits())
383c03c5b1cSMartin Matuska return (size_t)MEM_readLE32(memPtr);
384c03c5b1cSMartin Matuska else
385c03c5b1cSMartin Matuska return (size_t)MEM_readLE64(memPtr);
386c03c5b1cSMartin Matuska }
387c03c5b1cSMartin Matuska
MEM_writeLEST(void * memPtr,size_t val)388c03c5b1cSMartin Matuska MEM_STATIC void MEM_writeLEST(void* memPtr, size_t val)
389c03c5b1cSMartin Matuska {
390c03c5b1cSMartin Matuska if (MEM_32bits())
391c03c5b1cSMartin Matuska MEM_writeLE32(memPtr, (U32)val);
392c03c5b1cSMartin Matuska else
393c03c5b1cSMartin Matuska MEM_writeLE64(memPtr, (U64)val);
394c03c5b1cSMartin Matuska }
395c03c5b1cSMartin Matuska
396c03c5b1cSMartin Matuska /*=== Big endian r/w ===*/
397c03c5b1cSMartin Matuska
MEM_readBE32(const void * memPtr)398c03c5b1cSMartin Matuska MEM_STATIC U32 MEM_readBE32(const void* memPtr)
399c03c5b1cSMartin Matuska {
400c03c5b1cSMartin Matuska if (MEM_isLittleEndian())
401c03c5b1cSMartin Matuska return MEM_swap32(MEM_read32(memPtr));
402c03c5b1cSMartin Matuska else
403c03c5b1cSMartin Matuska return MEM_read32(memPtr);
404c03c5b1cSMartin Matuska }
405c03c5b1cSMartin Matuska
MEM_writeBE32(void * memPtr,U32 val32)406c03c5b1cSMartin Matuska MEM_STATIC void MEM_writeBE32(void* memPtr, U32 val32)
407c03c5b1cSMartin Matuska {
408c03c5b1cSMartin Matuska if (MEM_isLittleEndian())
409c03c5b1cSMartin Matuska MEM_write32(memPtr, MEM_swap32(val32));
410c03c5b1cSMartin Matuska else
411c03c5b1cSMartin Matuska MEM_write32(memPtr, val32);
412c03c5b1cSMartin Matuska }
413c03c5b1cSMartin Matuska
MEM_readBE64(const void * memPtr)414c03c5b1cSMartin Matuska MEM_STATIC U64 MEM_readBE64(const void* memPtr)
415c03c5b1cSMartin Matuska {
416c03c5b1cSMartin Matuska if (MEM_isLittleEndian())
417c03c5b1cSMartin Matuska return MEM_swap64(MEM_read64(memPtr));
418c03c5b1cSMartin Matuska else
419c03c5b1cSMartin Matuska return MEM_read64(memPtr);
420c03c5b1cSMartin Matuska }
421c03c5b1cSMartin Matuska
MEM_writeBE64(void * memPtr,U64 val64)422c03c5b1cSMartin Matuska MEM_STATIC void MEM_writeBE64(void* memPtr, U64 val64)
423c03c5b1cSMartin Matuska {
424c03c5b1cSMartin Matuska if (MEM_isLittleEndian())
425c03c5b1cSMartin Matuska MEM_write64(memPtr, MEM_swap64(val64));
426c03c5b1cSMartin Matuska else
427c03c5b1cSMartin Matuska MEM_write64(memPtr, val64);
428c03c5b1cSMartin Matuska }
429c03c5b1cSMartin Matuska
MEM_readBEST(const void * memPtr)430c03c5b1cSMartin Matuska MEM_STATIC size_t MEM_readBEST(const void* memPtr)
431c03c5b1cSMartin Matuska {
432c03c5b1cSMartin Matuska if (MEM_32bits())
433c03c5b1cSMartin Matuska return (size_t)MEM_readBE32(memPtr);
434c03c5b1cSMartin Matuska else
435c03c5b1cSMartin Matuska return (size_t)MEM_readBE64(memPtr);
436c03c5b1cSMartin Matuska }
437c03c5b1cSMartin Matuska
MEM_writeBEST(void * memPtr,size_t val)438c03c5b1cSMartin Matuska MEM_STATIC void MEM_writeBEST(void* memPtr, size_t val)
439c03c5b1cSMartin Matuska {
440c03c5b1cSMartin Matuska if (MEM_32bits())
441c03c5b1cSMartin Matuska MEM_writeBE32(memPtr, (U32)val);
442c03c5b1cSMartin Matuska else
443c03c5b1cSMartin Matuska MEM_writeBE64(memPtr, (U64)val);
444c03c5b1cSMartin Matuska }
445c03c5b1cSMartin Matuska
446c03c5b1cSMartin Matuska
447c03c5b1cSMartin Matuska #if defined (__cplusplus)
448c03c5b1cSMartin Matuska }
449c03c5b1cSMartin Matuska #endif
450c03c5b1cSMartin Matuska
451c03c5b1cSMartin Matuska #endif /* MEM_H_MODULE */
452