From e9c801f79bd6a43915918ca3c98f32b0a5571721 Mon Sep 17 00:00:00 2001 From: Pieter Wuille Date: Sun, 27 Oct 2019 12:52:31 -0700 Subject: [PATCH] [MOVEONLY] Move cpuid code from random & sha256 to compat/cpuid Cherry-picked from: 723c79666770b30cce9f962bed5ece8cc7d74580 --- src/Makefile.am | 1 + src/compat/cpuid.h | 24 ++++++++++++++++++++++++ src/crypto/sha256.cpp | 20 +++++--------------- src/random.cpp | 28 +++++++++++++++------------- 4 files changed, 45 insertions(+), 28 deletions(-) create mode 100644 src/compat/cpuid.h diff --git a/src/Makefile.am b/src/Makefile.am index 47127af7846..314c4a8189f 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -110,6 +110,7 @@ BITCOIN_CORE_H = \ coins.h \ compat.h \ compat/byteswap.h \ + compat/cpuid.h \ compat/endian.h \ compat/sanity.h \ compressor.h \ diff --git a/src/compat/cpuid.h b/src/compat/cpuid.h new file mode 100644 index 00000000000..0877ad47d30 --- /dev/null +++ b/src/compat/cpuid.h @@ -0,0 +1,24 @@ +// Copyright (c) 2017-2019 The Bitcoin Core developers +// Distributed under the MIT software license, see the accompanying +// file COPYING or http://www.opensource.org/licenses/mit-license.php. + +#ifndef BITCOIN_COMPAT_CPUID_H +#define BITCOIN_COMPAT_CPUID_H + +#if defined(__x86_64__) || defined(__amd64__) || defined(__i386__) +#define HAVE_GETCPUID + +#include + +// We can't use cpuid.h's __get_cpuid as it does not support subleafs. +void static inline GetCPUID(uint32_t leaf, uint32_t subleaf, uint32_t& a, uint32_t& b, uint32_t& c, uint32_t& d) +{ +#ifdef __GNUC__ + __cpuid_count(leaf, subleaf, a, b, c, d); +#else + __asm__ ("cpuid" : "=a"(a), "=b"(b), "=c"(c), "=d"(d) : "0"(leaf), "2"(subleaf)); +#endif +} + +#endif // defined(__x86_64__) || defined(__amd64__) || defined(__i386__) +#endif // BITCOIN_COMPAT_CPUID_H diff --git a/src/crypto/sha256.cpp b/src/crypto/sha256.cpp index f79a7b3cc3e..e386ff4bde7 100644 --- a/src/crypto/sha256.cpp +++ b/src/crypto/sha256.cpp @@ -67,9 +67,10 @@ static const uint32_t K[] = #include #endif +#include + #if defined(__x86_64__) || defined(__amd64__) || defined(__i386__) #if defined(USE_ASM) -#include namespace sha256_sse4 { void Transform(uint32_t* s, const unsigned char* chunk, size_t blocks); @@ -787,18 +788,7 @@ bool SelfTest() { return true; } - #if defined(USE_ASM) && (defined(__x86_64__) || defined(__amd64__) || defined(__i386__)) -// We can't use cpuid.h's __get_cpuid as it does not support subleafs. -void inline cpuid(uint32_t leaf, uint32_t subleaf, uint32_t& a, uint32_t& b, uint32_t& c, uint32_t& d) -{ -#ifdef __GNUC__ - __cpuid_count(leaf, subleaf, a, b, c, d); -#else - __asm__ ("cpuid" : "=a"(a), "=b"(b), "=c"(c), "=d"(d) : "0"(leaf), "2"(subleaf)); -#endif -} - /** Check whether the OS has enabled AVX registers. */ bool AVXEnabled() { @@ -811,7 +801,7 @@ bool AVXEnabled() /** Initialize the function pointer */ void inline Initialize_transform_ptr(void) { -#if defined(USE_ASM) && (defined(__x86_64__) || defined(__amd64__) || defined(__i386__)) +#if defined(USE_ASM) && defined(HAVE_GETCPUID) bool have_sse4 = false; bool have_xsave = false; bool have_avx = false; @@ -820,7 +810,7 @@ void inline Initialize_transform_ptr(void) [[maybe_unused]] bool enabled_avx = false; uint32_t eax, ebx, ecx, edx; - cpuid(1, 0, eax, ebx, ecx, edx); + GetCPUID(1, 0, eax, ebx, ecx, edx); have_sse4 = (ecx >> 19) & 1; have_xsave = (ecx >> 27) & 1; have_avx = (ecx >> 28) & 1; @@ -828,7 +818,7 @@ void inline Initialize_transform_ptr(void) enabled_avx = AVXEnabled(); } if (have_sse4) { - cpuid(7, 0, eax, ebx, ecx, edx); + GetCPUID(7, 0, eax, ebx, ecx, edx); have_avx2 = (ebx >> 5) & 1; have_x86_shani = (ebx >> 29) & 1; } diff --git a/src/random.cpp b/src/random.cpp index 1d162f82cd2..8ced2354369 100644 --- a/src/random.cpp +++ b/src/random.cpp @@ -5,6 +5,7 @@ #include "random.h" +#include "compat/cpuid.h" #include "crypto/sha512.h" #include "support/cleanse.h" #ifdef WIN32 @@ -40,14 +41,6 @@ #include #endif -#include - -#if defined(__x86_64__) || defined(__amd64__) || defined(__i386__) -#include -#endif - -#include - #include #include #include @@ -78,20 +71,29 @@ static inline int64_t GetPerformanceCounter() noexcept #endif } -#if defined(__x86_64__) || defined(__amd64__) || defined(__i386__) -static bool rdrand_supported = false; +#ifdef HAVE_GETCPUID +static bool g_rdrand_supported = false; +static bool g_rdseed_supported = false; static constexpr uint32_t CPUID_F1_ECX_RDRAND = 0x40000000; +static constexpr uint32_t CPUID_F7_EBX_RDSEED = 0x00040000; +#ifdef bit_RDRND +static_assert(CPUID_F1_ECX_RDRAND == bit_RDRND, "Unexpected value for bit_RDRND"); +#endif +#ifdef bit_RDSEED +static_assert(CPUID_F7_EBX_RDSEED == bit_RDSEED, "Unexpected value for bit_RDSEED"); +#endif + static void InitHardwareRand() { uint32_t eax, ebx, ecx, edx; if (__get_cpuid(1, &eax, &ebx, &ecx, &edx) && (ecx & CPUID_F1_ECX_RDRAND)) { - rdrand_supported = true; + g_rdrand_supported = true; } } static void ReportHardwareRand() { - if (rdrand_supported) { + if (g_rdrand_supported) { // This must be done in a separate function, as HWRandInit() may be indirectly called // from global constructors, before logging is initialized. LogPrintf("Using RdRand as an additional entropy source\n"); @@ -110,7 +112,7 @@ static void ReportHardwareRand() {} static bool GetHardwareRand(unsigned char* ent32) noexcept { #if defined(__x86_64__) || defined(__amd64__) || defined(__i386__) - if (rdrand_supported) { + if (g_rdrand_supported) { uint8_t ok; // Not all assemblers support the rdrand instruction, write it in hex. #ifdef __i386__