13cab2bb3Spatrick //===-- checksum.cpp --------------------------------------------*- C++ -*-===//
23cab2bb3Spatrick //
33cab2bb3Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
43cab2bb3Spatrick // See https://llvm.org/LICENSE.txt for license information.
53cab2bb3Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
63cab2bb3Spatrick //
73cab2bb3Spatrick //===----------------------------------------------------------------------===//
83cab2bb3Spatrick 
93cab2bb3Spatrick #include "checksum.h"
103cab2bb3Spatrick #include "atomic_helpers.h"
11*810390e3Srobert #include "chunk.h"
123cab2bb3Spatrick 
133cab2bb3Spatrick #if defined(__x86_64__) || defined(__i386__)
143cab2bb3Spatrick #include <cpuid.h>
153cab2bb3Spatrick #elif defined(__arm__) || defined(__aarch64__)
163cab2bb3Spatrick #if SCUDO_FUCHSIA
173cab2bb3Spatrick #include <zircon/features.h>
183cab2bb3Spatrick #include <zircon/syscalls.h>
193cab2bb3Spatrick #else
203cab2bb3Spatrick #include <sys/auxv.h>
213cab2bb3Spatrick #endif
223cab2bb3Spatrick #endif
233cab2bb3Spatrick 
243cab2bb3Spatrick namespace scudo {
253cab2bb3Spatrick 
263cab2bb3Spatrick Checksum HashAlgorithm = {Checksum::BSD};
273cab2bb3Spatrick 
283cab2bb3Spatrick #if defined(__x86_64__) || defined(__i386__)
293cab2bb3Spatrick // i386 and x86_64 specific code to detect CRC32 hardware support via CPUID.
303cab2bb3Spatrick // CRC32 requires the SSE 4.2 instruction set.
313cab2bb3Spatrick #ifndef bit_SSE4_2
323cab2bb3Spatrick #define bit_SSE4_2 bit_SSE42 // clang and gcc have different defines.
333cab2bb3Spatrick #endif
343cab2bb3Spatrick 
351f9cb04fSpatrick #ifndef signature_HYGON_ebx // They are not defined in gcc.
361f9cb04fSpatrick // HYGON: "HygonGenuine".
371f9cb04fSpatrick #define signature_HYGON_ebx 0x6f677948
381f9cb04fSpatrick #define signature_HYGON_edx 0x6e65476e
391f9cb04fSpatrick #define signature_HYGON_ecx 0x656e6975
401f9cb04fSpatrick #endif
411f9cb04fSpatrick 
hasHardwareCRC32()423cab2bb3Spatrick bool hasHardwareCRC32() {
433cab2bb3Spatrick   u32 Eax, Ebx = 0, Ecx = 0, Edx = 0;
443cab2bb3Spatrick   __get_cpuid(0, &Eax, &Ebx, &Ecx, &Edx);
453cab2bb3Spatrick   const bool IsIntel = (Ebx == signature_INTEL_ebx) &&
463cab2bb3Spatrick                        (Edx == signature_INTEL_edx) &&
473cab2bb3Spatrick                        (Ecx == signature_INTEL_ecx);
483cab2bb3Spatrick   const bool IsAMD = (Ebx == signature_AMD_ebx) && (Edx == signature_AMD_edx) &&
493cab2bb3Spatrick                      (Ecx == signature_AMD_ecx);
501f9cb04fSpatrick   const bool IsHygon = (Ebx == signature_HYGON_ebx) &&
511f9cb04fSpatrick                        (Edx == signature_HYGON_edx) &&
521f9cb04fSpatrick                        (Ecx == signature_HYGON_ecx);
531f9cb04fSpatrick   if (!IsIntel && !IsAMD && !IsHygon)
543cab2bb3Spatrick     return false;
553cab2bb3Spatrick   __get_cpuid(1, &Eax, &Ebx, &Ecx, &Edx);
563cab2bb3Spatrick   return !!(Ecx & bit_SSE4_2);
573cab2bb3Spatrick }
583cab2bb3Spatrick #elif defined(__arm__) || defined(__aarch64__)
593cab2bb3Spatrick #ifndef AT_HWCAP
603cab2bb3Spatrick #define AT_HWCAP 16
613cab2bb3Spatrick #endif
623cab2bb3Spatrick #ifndef HWCAP_CRC32
633cab2bb3Spatrick #define HWCAP_CRC32 (1U << 7) // HWCAP_CRC32 is missing on older platforms.
643cab2bb3Spatrick #endif
653cab2bb3Spatrick 
hasHardwareCRC32()663cab2bb3Spatrick bool hasHardwareCRC32() {
673cab2bb3Spatrick #if SCUDO_FUCHSIA
683cab2bb3Spatrick   u32 HWCap;
693cab2bb3Spatrick   const zx_status_t Status =
703cab2bb3Spatrick       zx_system_get_features(ZX_FEATURE_KIND_CPU, &HWCap);
713cab2bb3Spatrick   if (Status != ZX_OK)
723cab2bb3Spatrick     return false;
733cab2bb3Spatrick   return !!(HWCap & ZX_ARM64_FEATURE_ISA_CRC32);
743cab2bb3Spatrick #else
753cab2bb3Spatrick   return !!(getauxval(AT_HWCAP) & HWCAP_CRC32);
763cab2bb3Spatrick #endif // SCUDO_FUCHSIA
773cab2bb3Spatrick }
783cab2bb3Spatrick #else
793cab2bb3Spatrick // No hardware CRC32 implemented in Scudo for other architectures.
hasHardwareCRC32()803cab2bb3Spatrick bool hasHardwareCRC32() { return false; }
813cab2bb3Spatrick #endif // defined(__x86_64__) || defined(__i386__)
823cab2bb3Spatrick 
833cab2bb3Spatrick } // namespace scudo
84