aboutsummaryrefslogtreecommitdiffhomepage
path: root/src/common/src/crc.h
diff options
context:
space:
mode:
Diffstat (limited to 'src/common/src/crc.h')
-rw-r--r--src/common/src/crc.h81
1 files changed, 81 insertions, 0 deletions
diff --git a/src/common/src/crc.h b/src/common/src/crc.h
new file mode 100644
index 00000000..2bb3210d
--- /dev/null
+++ b/src/common/src/crc.h
@@ -0,0 +1,81 @@
+#ifndef COMMON_CRC_H_
+#define COMMON_CRC_H_
+
+#include "types.h"
+#include "platform.h"
+
+#define CRC_ROTL(crc) crc32_table[3][((crc) & 0xFF)] ^ crc32_table[2][((crc >> 8) & 0xFF)] ^ \
+ crc32_table[1][((crc >> 16) & 0xFF)] ^ crc32_table[0][((crc >> 24))]
+
+// Some definitions for using the X86 CRC32 instruction on different platforms. Keep in mind, you
+// should check for X86/X64 architecture support before using these, as well as for SSE 4.2 (see the
+// x86_utils module).
+
+#if defined(EMU_ARCHITECTURE_X86) || defined(EMU_ARCHITECTURE_X64)
+
+#if EMU_PLATFORM == PLATFORM_WINDOWS
+
+#include <nmmintrin.h>
+
+#ifdef EMU_ARCHITECTURE_X64
+static inline u64 InlineCrc32_U64(u64 crc, u64 value) {
+ return _mm_crc32_u64(crc, value);
+}
+#endif
+static inline u32 InlineCrc32_U32(u32 crc, u64 value) {
+ return _mm_crc32_u32(crc, static_cast<u32>(value));
+}
+
+static inline u32 InlineCrc32_U8(u32 crc, u8 value) {
+ return _mm_crc32_u8(crc, value);
+}
+
+#elif GCC_VERSION_AVAILABLE(4, 5) && defined(__SSE4_2__)
+
+extern inline unsigned int __attribute__((
+ __gnu_inline__, __always_inline__, __artificial__))
+InlineCrc32_U8(unsigned int __C, unsigned char __V) {
+ return __builtin_ia32_crc32qi(__C, __V);
+}
+#ifdef EMU_ARCHITECTURE_X64
+extern inline unsigned long long __attribute__((
+ __gnu_inline__, __always_inline__, __artificial__))
+InlineCrc32_U64(unsigned long long __C, unsigned long long __V) {
+ return __builtin_ia32_crc32di(__C, __V);
+}
+#else
+extern inline unsigned int __attribute__((
+ __gnu_inline__, __always_inline__, __artificial__))
+InlineCrc32_U32(unsigned int __C, unsigned int __V) {
+ return __builtin_ia32_crc32si (__C, __V);
+}
+#endif // EMU_ARCHITECTURE_X64
+
+#else
+
+// GCC 4.4.x and earlier: use inline asm, or msse4.2 flag not set
+
+static inline u64 InlineCrc32_U64(u64 crc, u64 value) {
+ asm("crc32q %[value], %[crc]\n" : [crc] "+r" (crc) : [value] "rm" (value));
+ return crc;
+}
+
+static inline u32 InlineCrc32_U32(u32 crc, u64 value) {
+ asm("crc32l %[value], %[crc]\n" : [crc] "+r" (crc) : [value] "rm" (value));
+ return crc;
+}
+
+static inline u32 InlineCrc32_U8(u32 crc, u8 value) {
+ asm("crc32b %[value], %[crc]\n" : [crc] "+r" (crc) : [value] "rm" (value));
+ return crc;
+}
+#endif
+
+#endif // EMU_ARCHITECTURE_X86 or EMU_ARCHITECTURE_X64
+
+extern u32 crc32_table[4][256];
+
+void Init_CRC32_Table();
+u32 GenerateCRC(u8 *StartAddr, u32 Len);
+
+#endif // COMMON_CRC_H_