/* XMRig * Copyright 2010 Jeff Garzik * Copyright 2012-2014 pooler * Copyright 2014 Lucas Jones * Copyright 2014-2016 Wolf9466 * Copyright 2016 Jay D Dee * Copyright 2017-2019 XMR-Stak , * Copyright 2018-2019 SChernykh * Copyright 2016-2019 XMRig * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ #include #include #include #include #include #include "3rdparty/libcpuid/libcpuid.h" #include "backend/cpu/platform/AdvancedCpuInfo.h" namespace xmrig { static inline void cpu_brand_string(char out[64], const char *in) { size_t pos = 0; const size_t size = strlen(in); for (size_t i = 0; i < size; ++i) { if (in[i] == ' ' && ((pos > 0 && out[pos - 1] == ' ') || pos == 0)) { continue; } out[pos++] = in[i]; } if (pos > 0 && out[pos - 1] == ' ') { out[pos - 1] = '\0'; } } } // namespace xmrig xmrig::AdvancedCpuInfo::AdvancedCpuInfo() : m_brand() { struct cpu_raw_data_t raw = {}; struct cpu_id_t data = {}; cpuid_get_raw_data(&raw); cpu_identify(&raw, &data); cpu_brand_string(m_brand, data.brand_str); snprintf(m_backend, sizeof m_backend, "libcpuid/%s", cpuid_lib_version()); m_threads = static_cast(data.total_logical_cpus); m_packages = std::max(threads() / static_cast(data.num_logical_cpus), 1); m_cores = static_cast(data.num_cores) * m_packages; m_L3 = data.l3_cache > 0 ? static_cast(data.l3_cache) * m_packages : 0; const size_t l2 = static_cast(data.l2_cache); // Workaround for AMD CPUs https://github.com/anrieff/libcpuid/issues/97 if (data.vendor == VENDOR_AMD && data.ext_family >= 0x15 && data.ext_family < 0x17) { m_L2 = l2 * (cores() / 2) * m_packages; m_L2_exclusive = true; } // Workaround for Intel Pentium Dual-Core, Core Duo, Core 2 Duo, Core 2 Quad and their Xeon homologue // These processors have L2 cache shared by 2 cores. else if (data.vendor == VENDOR_INTEL && data.ext_family == 0x06 && (data.ext_model == 0x0E || data.ext_model == 0x0F || data.ext_model == 0x17)) { size_t l2_count_per_socket = cores() > 1 ? cores() / 2 : 1; m_L2 = data.l2_cache > 0 ? l2 * l2_count_per_socket * m_packages : 0; } else{ m_L2 = data.l2_cache > 0 ? l2 * cores() * m_packages : 0; } m_L2 *= 1024; m_L3 *= 1024; if (data.flags[CPU_FEATURE_AES]) { m_aes = true; if (data.vendor == VENDOR_AMD) { m_assembly = (data.ext_family >= 23) ? Assembly::RYZEN : Assembly::BULLDOZER; } else if (data.vendor == VENDOR_INTEL) { m_assembly = Assembly::INTEL; } } m_avx2 = data.flags[CPU_FEATURE_AVX2] && data.flags[CPU_FEATURE_OSXSAVE]; } xmrig::CpuThreads xmrig::AdvancedCpuInfo::threads(const Algorithm &algorithm) const { if (threads() == 1) { return 1; } # ifdef XMRIG_ALGO_CN_GPU if (algorithm == Algorithm::CN_GPU) { return CpuThreads(threads()); } # endif size_t cache = 0; size_t count = 0; if (m_L3) { cache = m_L2_exclusive ? (m_L2 + m_L3) : m_L3; } else { cache = m_L2; } if (cache) { const size_t memory = algorithm.l3(); assert(memory > 0); count = cache / memory; if (cache % memory >= memory / 2) { count++; } } else { count = threads() / 2; } uint32_t intensity = algorithm.maxIntensity() == 1 ? 0 : 1; # ifdef XMRIG_ALGO_CN_PICO if (algorithm == Algorithm::CN_PICO_0 && (count / cores()) >= 2) { intensity = 2; } # endif return CpuThreads(std::max(std::min(count, threads()), 1), intensity); }