blob: 261e6f22e1671ba0d079f43b597203d7489dce4e [file] [log] [blame]
/*
* Copyright 2016 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
#ifndef SkCpu_DEFINED
#define SkCpu_DEFINED
#include "include/core/SkTypes.h"
#include <cstdint>
struct SkCpu {
enum {
SSE1 = 1 << 0,
SSE2 = 1 << 1,
SSE3 = 1 << 2,
SSSE3 = 1 << 3,
SSE41 = 1 << 4,
SSE42 = 1 << 5,
AVX = 1 << 6,
F16C = 1 << 7,
FMA = 1 << 8,
AVX2 = 1 << 9,
BMI1 = 1 << 10,
BMI2 = 1 << 11,
// Handy alias for all the cool Haswell+ instructions.
HSW = AVX2 | BMI1 | BMI2 | F16C | FMA,
AVX512F = 1 << 12,
AVX512DQ = 1 << 13,
AVX512IFMA = 1 << 14,
AVX512PF = 1 << 15,
AVX512ER = 1 << 16,
AVX512CD = 1 << 17,
AVX512BW = 1 << 18,
AVX512VL = 1 << 19,
// Handy alias for all the cool Skylake Xeon+ instructions.
SKX = AVX512F | AVX512DQ | AVX512CD | AVX512BW | AVX512VL,
ERMS = 1 << 20,
};
enum {
LOONGARCH_SX = 1 << 0,
LOONGARCH_ASX = 1 << 1,
};
static void CacheRuntimeFeatures();
static bool Supports(uint32_t);
private:
static uint32_t gCachedFeatures;
};
inline bool SkCpu::Supports(uint32_t mask) {
uint32_t features = gCachedFeatures;
// If we mask in compile-time known lower limits, the compiler can
// often compile away this entire function.
#if SK_CPU_X86
#if SK_CPU_SSE_LEVEL >= SK_CPU_SSE_LEVEL_SSE1
features |= SSE1;
#endif
#if SK_CPU_SSE_LEVEL >= SK_CPU_SSE_LEVEL_SSE2
features |= SSE2;
#endif
#if SK_CPU_SSE_LEVEL >= SK_CPU_SSE_LEVEL_SSE3
features |= SSE3;
#endif
#if SK_CPU_SSE_LEVEL >= SK_CPU_SSE_LEVEL_SSSE3
features |= SSSE3;
#endif
#if SK_CPU_SSE_LEVEL >= SK_CPU_SSE_LEVEL_SSE41
features |= SSE41;
#endif
#if SK_CPU_SSE_LEVEL >= SK_CPU_SSE_LEVEL_SSE42
features |= SSE42;
#endif
#if SK_CPU_SSE_LEVEL >= SK_CPU_SSE_LEVEL_AVX
features |= AVX;
#endif
// F16C goes here if we add SK_CPU_SSE_LEVEL_F16C
#if SK_CPU_SSE_LEVEL >= SK_CPU_SSE_LEVEL_AVX2
features |= AVX2;
#endif
#if SK_CPU_SSE_LEVEL >= SK_CPU_SSE_LEVEL_SKX
features |= (AVX512F | AVX512DQ | AVX512CD | AVX512BW | AVX512VL);
#endif
// FMA doesn't fit neatly into this total ordering.
// It's available on Haswell+ just like AVX2, but it's technically a different bit.
// TODO: circle back on this if we find ourselves limited by lack of compile-time FMA
#if defined(SK_CPU_LIMIT_AVX)
features &= (SSE1 | SSE2 | SSE3 | SSSE3 | SSE41 | SSE42 | AVX);
#elif defined(SK_CPU_LIMIT_SSE41)
features &= (SSE1 | SSE2 | SSE3 | SSSE3 | SSE41);
#elif defined(SK_CPU_LIMIT_SSE2)
features &= (SSE1 | SSE2);
#endif
#elif SK_CPU_LOONGARCH
#if SK_CPU_LSX_LEVEL >= SK_CPU_LSX_LEVEL_LSX
features |= LOONGARCH_SX;
#endif
#if SK_CPU_LSX_LEVEL >= SK_CPU_LSX_LEVEL_LASX
features |= LOONGARCH_ASX;
#endif
#endif
return (features & mask) == mask;
}
#endif//SkCpu_DEFINED