/* vim:set ts=2 sw=2 sts=2 et: */ /** * \author Marcus Holland-Moritz (github@mhxnet.de) * \copyright Copyright (c) Marcus Holland-Moritz * * This file is part of ricepp. * * ricepp 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. * * ricepp 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 ricepp. If not, see . */ #include #include #include #include #include #include #include "ricepp_cpuspecific.h" namespace ricepp { namespace { detail::cpu_variant get_cpu_variant_init() { #ifndef _WIN32 #if defined(__has_builtin) #if __has_builtin(__builtin_cpu_supports) && \ (defined(RICEPP_CPU_BMI2) || defined(RICEPP_CPU_BMI2_AVX512)) __builtin_cpu_init(); bool const has_avx512vl = __builtin_cpu_supports("avx512vl"); bool const has_avx512vbmi = __builtin_cpu_supports("avx512vbmi"); bool const has_bmi2 = __builtin_cpu_supports("bmi2"); if (has_avx512vl && has_avx512vbmi && has_bmi2) { return detail::cpu_variant::has_bmi2_avx512; } if (has_bmi2) { return detail::cpu_variant::has_bmi2; } #endif #endif #endif return detail::cpu_variant::fallback; } detail::cpu_variant get_cpu_variant() { static detail::cpu_variant const variant = get_cpu_variant_init(); return variant; } void show_cpu_variant(std::string_view variant) { if (std::getenv("RICEPP_SHOW_CPU_VARIANT")) { std::cerr << "ricepp: using " << variant << " CPU variant\n"; } } void show_cpu_variant_once(std::string_view variant) { static auto const _ = [&variant]() { show_cpu_variant(variant); return true; }(); } } // namespace template <> std::unique_ptr> create_codec(codec_config const& config) { switch (get_cpu_variant()) { #ifdef RICEPP_CPU_BMI2_AVX512 case detail::cpu_variant::has_bmi2_avx512: show_cpu_variant_once("BMI2+AVX512"); return detail::create_codec_cpuspecific_< uint16_t, detail::cpu_variant::has_bmi2_avx512>(config); #endif #ifdef RICEPP_CPU_BMI2 case detail::cpu_variant::has_bmi2: show_cpu_variant_once("BMI2"); return detail::create_codec_cpuspecific_( config); #endif default: show_cpu_variant_once("fallback"); return detail::create_codec_cpuspecific_( config); } throw std::runtime_error("internal error: unknown CPU variant"); } } // namespace ricepp