mirror of
https://git.proxmox.com/git/mirror_zfs.git
synced 2025-01-15 04:30:33 +03:00
c65aaa8387
Intel SPR erratum SPR4 says that if you trip into a vmexit while doing FPU save/restore, your AMX register state might misbehave... and by misbehave, I mean save all zeroes incorrectly, leading to explosions if you restore it. Since we're not using AMX for anything, the simple way to avoid this is to just not save/restore those when we do anything, since we're killing preemption of any sort across our save/restores. If we ever decide to use AMX, it's not clear that we have any way to mitigate this, on Linux...but I am not an expert. Reviewed-by: Brian Behlendorf <behlendorf1@llnl.gov> Signed-off-by: Rich Ercolani <rincebrain@gmail.com> Closes #14989 Closes #15168
767 lines
17 KiB
C
767 lines
17 KiB
C
/*
|
||
* CDDL HEADER START
|
||
*
|
||
* The contents of this file are subject to the terms of the
|
||
* Common Development and Distribution License (the "License").
|
||
* You may not use this file except in compliance with the License.
|
||
*
|
||
* You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE
|
||
* or https://opensource.org/licenses/CDDL-1.0.
|
||
* See the License for the specific language governing permissions
|
||
* and limitations under the License.
|
||
*
|
||
* When distributing Covered Code, include this CDDL HEADER in each
|
||
* file and include the License file at usr/src/OPENSOLARIS.LICENSE.
|
||
* If applicable, add the following below this CDDL HEADER, with the
|
||
* fields enclosed by brackets "[]" replaced with your own identifying
|
||
* information: Portions Copyright [yyyy] [name of copyright owner]
|
||
*
|
||
* CDDL HEADER END
|
||
*/
|
||
/*
|
||
* Copyright (C) 2016 Gvozden Neskovic <neskovic@compeng.uni-frankfurt.de>.
|
||
*/
|
||
|
||
/*
|
||
* USER API:
|
||
*
|
||
* Kernel fpu methods:
|
||
* kfpu_allowed()
|
||
* kfpu_begin()
|
||
* kfpu_end()
|
||
* kfpu_init()
|
||
* kfpu_fini()
|
||
*
|
||
* SIMD support:
|
||
*
|
||
* Following functions should be called to determine whether CPU feature
|
||
* is supported. All functions are usable in kernel and user space.
|
||
* If a SIMD algorithm is using more than one instruction set
|
||
* all relevant feature test functions should be called.
|
||
*
|
||
* Supported features:
|
||
* zfs_sse_available()
|
||
* zfs_sse2_available()
|
||
* zfs_sse3_available()
|
||
* zfs_ssse3_available()
|
||
* zfs_sse4_1_available()
|
||
* zfs_sse4_2_available()
|
||
*
|
||
* zfs_avx_available()
|
||
* zfs_avx2_available()
|
||
*
|
||
* zfs_bmi1_available()
|
||
* zfs_bmi2_available()
|
||
*
|
||
* zfs_shani_available()
|
||
*
|
||
* zfs_avx512f_available()
|
||
* zfs_avx512cd_available()
|
||
* zfs_avx512er_available()
|
||
* zfs_avx512pf_available()
|
||
* zfs_avx512bw_available()
|
||
* zfs_avx512dq_available()
|
||
* zfs_avx512vl_available()
|
||
* zfs_avx512ifma_available()
|
||
* zfs_avx512vbmi_available()
|
||
*
|
||
* NOTE(AVX-512VL): If using AVX-512 instructions with 128Bit registers
|
||
* also add zfs_avx512vl_available() to feature check.
|
||
*/
|
||
|
||
#ifndef _LINUX_SIMD_X86_H
|
||
#define _LINUX_SIMD_X86_H
|
||
|
||
/* only for __x86 */
|
||
#if defined(__x86)
|
||
|
||
#include <sys/types.h>
|
||
#include <asm/cpufeature.h>
|
||
|
||
/*
|
||
* Disable the WARN_ON_FPU() macro to prevent additional dependencies
|
||
* when providing the kfpu_* functions. Relevant warnings are included
|
||
* as appropriate and are unconditionally enabled.
|
||
*/
|
||
#if defined(CONFIG_X86_DEBUG_FPU) && !defined(KERNEL_EXPORTS_X86_FPU)
|
||
#undef CONFIG_X86_DEBUG_FPU
|
||
#endif
|
||
|
||
/*
|
||
* The following cases are for kernels which export either the
|
||
* kernel_fpu_* or __kernel_fpu_* functions.
|
||
*/
|
||
#if defined(KERNEL_EXPORTS_X86_FPU)
|
||
|
||
#if defined(HAVE_KERNEL_FPU_API_HEADER)
|
||
#include <asm/fpu/api.h>
|
||
#if defined(HAVE_KERNEL_FPU_INTERNAL_HEADER)
|
||
#include <asm/fpu/internal.h>
|
||
#endif
|
||
#else
|
||
#include <asm/i387.h>
|
||
#endif
|
||
|
||
#define kfpu_allowed() 1
|
||
#define kfpu_init() 0
|
||
#define kfpu_fini() ((void) 0)
|
||
|
||
#if defined(HAVE_UNDERSCORE_KERNEL_FPU)
|
||
#define kfpu_begin() \
|
||
{ \
|
||
preempt_disable(); \
|
||
__kernel_fpu_begin(); \
|
||
}
|
||
#define kfpu_end() \
|
||
{ \
|
||
__kernel_fpu_end(); \
|
||
preempt_enable(); \
|
||
}
|
||
|
||
#elif defined(HAVE_KERNEL_FPU)
|
||
#define kfpu_begin() kernel_fpu_begin()
|
||
#define kfpu_end() kernel_fpu_end()
|
||
|
||
#else
|
||
/*
|
||
* This case is unreachable. When KERNEL_EXPORTS_X86_FPU is defined then
|
||
* either HAVE_UNDERSCORE_KERNEL_FPU or HAVE_KERNEL_FPU must be defined.
|
||
*/
|
||
#error "Unreachable kernel configuration"
|
||
#endif
|
||
|
||
#else /* defined(KERNEL_EXPORTS_X86_FPU) */
|
||
|
||
/*
|
||
* When the kernel_fpu_* symbols are unavailable then provide our own
|
||
* versions which allow the FPU to be safely used.
|
||
*/
|
||
#if defined(HAVE_KERNEL_FPU_INTERNAL)
|
||
|
||
/*
|
||
* For kernels not exporting *kfpu_{begin,end} we have to use inline assembly
|
||
* with the XSAVE{,OPT,S} instructions, so we need the toolchain to support at
|
||
* least XSAVE.
|
||
*/
|
||
#if !defined(HAVE_XSAVE)
|
||
#error "Toolchain needs to support the XSAVE assembler instruction"
|
||
#endif
|
||
|
||
#ifndef XFEATURE_MASK_XTILE
|
||
/*
|
||
* For kernels where this doesn't exist yet, we still don't want to break
|
||
* by save/restoring this broken nonsense.
|
||
* See issue #14989 or Intel errata SPR4 for why
|
||
*/
|
||
#define XFEATURE_MASK_XTILE 0x60000
|
||
#endif
|
||
|
||
#include <linux/mm.h>
|
||
#include <linux/slab.h>
|
||
|
||
extern uint8_t **zfs_kfpu_fpregs;
|
||
|
||
/*
|
||
* Return the size in bytes required by the XSAVE instruction for an
|
||
* XSAVE area containing all the user state components supported by this CPU.
|
||
* See: Intel 64 and IA-32 Architectures Software Developer’s Manual.
|
||
* Dec. 2021. Vol. 2A p. 3-222.
|
||
*/
|
||
static inline uint32_t
|
||
get_xsave_area_size(void)
|
||
{
|
||
if (!boot_cpu_has(X86_FEATURE_OSXSAVE)) {
|
||
return (0);
|
||
}
|
||
/*
|
||
* Call CPUID with leaf 13 and subleaf 0. The size is in ecx.
|
||
* We don't need to check for cpuid_max here, since if this CPU has
|
||
* OSXSAVE set, it has leaf 13 (0x0D) as well.
|
||
*/
|
||
uint32_t eax, ebx, ecx, edx;
|
||
|
||
eax = 13U;
|
||
ecx = 0U;
|
||
__asm__ __volatile__("cpuid"
|
||
: "=a" (eax), "=b" (ebx), "=c" (ecx), "=d" (edx)
|
||
: "a" (eax), "c" (ecx));
|
||
|
||
return (ecx);
|
||
}
|
||
|
||
/*
|
||
* Return the allocation order of the maximum buffer size required to save the
|
||
* FPU state on this architecture. The value returned is the same as Linux'
|
||
* get_order() function would return (i.e. 2^order = nr. of pages required).
|
||
* Currently this will always return 0 since the save area is below 4k even for
|
||
* a full fledged AVX-512 implementation.
|
||
*/
|
||
static inline int
|
||
get_fpuregs_save_area_order(void)
|
||
{
|
||
size_t area_size = (size_t)get_xsave_area_size();
|
||
|
||
/*
|
||
* If we are dealing with a CPU not supporting XSAVE,
|
||
* get_xsave_area_size() will return 0. Thus the maximum memory
|
||
* required is the FXSAVE area size which is 512 bytes. See: Intel 64
|
||
* and IA-32 Architectures Software Developer’s Manual. Dec. 2021.
|
||
* Vol. 2A p. 3-451.
|
||
*/
|
||
if (area_size == 0) {
|
||
area_size = 512;
|
||
}
|
||
return (get_order(area_size));
|
||
}
|
||
|
||
/*
|
||
* Initialize per-cpu variables to store FPU state.
|
||
*/
|
||
static inline void
|
||
kfpu_fini(void)
|
||
{
|
||
int cpu;
|
||
int order = get_fpuregs_save_area_order();
|
||
|
||
for_each_possible_cpu(cpu) {
|
||
if (zfs_kfpu_fpregs[cpu] != NULL) {
|
||
free_pages((unsigned long)zfs_kfpu_fpregs[cpu], order);
|
||
}
|
||
}
|
||
|
||
kfree(zfs_kfpu_fpregs);
|
||
}
|
||
|
||
static inline int
|
||
kfpu_init(void)
|
||
{
|
||
zfs_kfpu_fpregs = kzalloc(num_possible_cpus() * sizeof (uint8_t *),
|
||
GFP_KERNEL);
|
||
|
||
if (zfs_kfpu_fpregs == NULL)
|
||
return (-ENOMEM);
|
||
|
||
/*
|
||
* The fxsave and xsave operations require 16-/64-byte alignment of
|
||
* the target memory. Since kmalloc() provides no alignment
|
||
* guarantee instead use alloc_pages_node().
|
||
*/
|
||
int cpu;
|
||
int order = get_fpuregs_save_area_order();
|
||
|
||
for_each_possible_cpu(cpu) {
|
||
struct page *page = alloc_pages_node(cpu_to_node(cpu),
|
||
GFP_KERNEL | __GFP_ZERO, order);
|
||
if (page == NULL) {
|
||
kfpu_fini();
|
||
return (-ENOMEM);
|
||
}
|
||
|
||
zfs_kfpu_fpregs[cpu] = page_address(page);
|
||
}
|
||
|
||
return (0);
|
||
}
|
||
|
||
#define kfpu_allowed() 1
|
||
|
||
/*
|
||
* FPU save and restore instructions.
|
||
*/
|
||
#define __asm __asm__ __volatile__
|
||
#define kfpu_fxsave(addr) __asm("fxsave %0" : "=m" (*(addr)))
|
||
#define kfpu_fxsaveq(addr) __asm("fxsaveq %0" : "=m" (*(addr)))
|
||
#define kfpu_fnsave(addr) __asm("fnsave %0; fwait" : "=m" (*(addr)))
|
||
#define kfpu_fxrstor(addr) __asm("fxrstor %0" : : "m" (*(addr)))
|
||
#define kfpu_fxrstorq(addr) __asm("fxrstorq %0" : : "m" (*(addr)))
|
||
#define kfpu_frstor(addr) __asm("frstor %0" : : "m" (*(addr)))
|
||
#define kfpu_fxsr_clean(rval) __asm("fnclex; emms; fildl %P[addr]" \
|
||
: : [addr] "m" (rval));
|
||
|
||
#define kfpu_do_xsave(instruction, addr, mask) \
|
||
{ \
|
||
uint32_t low, hi; \
|
||
\
|
||
low = mask; \
|
||
hi = (uint64_t)(mask) >> 32; \
|
||
__asm(instruction " %[dst]\n\t" \
|
||
: \
|
||
: [dst] "m" (*(addr)), "a" (low), "d" (hi) \
|
||
: "memory"); \
|
||
}
|
||
|
||
static inline void
|
||
kfpu_save_fxsr(uint8_t *addr)
|
||
{
|
||
if (IS_ENABLED(CONFIG_X86_32))
|
||
kfpu_fxsave(addr);
|
||
else
|
||
kfpu_fxsaveq(addr);
|
||
}
|
||
|
||
static inline void
|
||
kfpu_save_fsave(uint8_t *addr)
|
||
{
|
||
kfpu_fnsave(addr);
|
||
}
|
||
|
||
static inline void
|
||
kfpu_begin(void)
|
||
{
|
||
/*
|
||
* Preemption and interrupts must be disabled for the critical
|
||
* region where the FPU state is being modified.
|
||
*/
|
||
preempt_disable();
|
||
local_irq_disable();
|
||
|
||
/*
|
||
* The current FPU registers need to be preserved by kfpu_begin()
|
||
* and restored by kfpu_end(). They are stored in a dedicated
|
||
* per-cpu variable, not in the task struct, this allows any user
|
||
* FPU state to be correctly preserved and restored.
|
||
*/
|
||
uint8_t *state = zfs_kfpu_fpregs[smp_processor_id()];
|
||
#if defined(HAVE_XSAVES)
|
||
if (static_cpu_has(X86_FEATURE_XSAVES)) {
|
||
kfpu_do_xsave("xsaves", state, ~XFEATURE_MASK_XTILE);
|
||
return;
|
||
}
|
||
#endif
|
||
#if defined(HAVE_XSAVEOPT)
|
||
if (static_cpu_has(X86_FEATURE_XSAVEOPT)) {
|
||
kfpu_do_xsave("xsaveopt", state, ~XFEATURE_MASK_XTILE);
|
||
return;
|
||
}
|
||
#endif
|
||
if (static_cpu_has(X86_FEATURE_XSAVE)) {
|
||
kfpu_do_xsave("xsave", state, ~XFEATURE_MASK_XTILE);
|
||
} else if (static_cpu_has(X86_FEATURE_FXSR)) {
|
||
kfpu_save_fxsr(state);
|
||
} else {
|
||
kfpu_save_fsave(state);
|
||
}
|
||
}
|
||
|
||
#define kfpu_do_xrstor(instruction, addr, mask) \
|
||
{ \
|
||
uint32_t low, hi; \
|
||
\
|
||
low = mask; \
|
||
hi = (uint64_t)(mask) >> 32; \
|
||
__asm(instruction " %[src]" \
|
||
: \
|
||
: [src] "m" (*(addr)), "a" (low), "d" (hi) \
|
||
: "memory"); \
|
||
}
|
||
|
||
static inline void
|
||
kfpu_restore_fxsr(uint8_t *addr)
|
||
{
|
||
/*
|
||
* On AuthenticAMD K7 and K8 processors the fxrstor instruction only
|
||
* restores the _x87 FOP, FIP, and FDP registers when an exception
|
||
* is pending. Clean the _x87 state to force the restore.
|
||
*/
|
||
if (unlikely(static_cpu_has_bug(X86_BUG_FXSAVE_LEAK)))
|
||
kfpu_fxsr_clean(addr);
|
||
|
||
if (IS_ENABLED(CONFIG_X86_32)) {
|
||
kfpu_fxrstor(addr);
|
||
} else {
|
||
kfpu_fxrstorq(addr);
|
||
}
|
||
}
|
||
|
||
static inline void
|
||
kfpu_restore_fsave(uint8_t *addr)
|
||
{
|
||
kfpu_frstor(addr);
|
||
}
|
||
|
||
static inline void
|
||
kfpu_end(void)
|
||
{
|
||
uint8_t *state = zfs_kfpu_fpregs[smp_processor_id()];
|
||
#if defined(HAVE_XSAVES)
|
||
if (static_cpu_has(X86_FEATURE_XSAVES)) {
|
||
kfpu_do_xrstor("xrstors", state, ~XFEATURE_MASK_XTILE);
|
||
goto out;
|
||
}
|
||
#endif
|
||
if (static_cpu_has(X86_FEATURE_XSAVE)) {
|
||
kfpu_do_xrstor("xrstor", state, ~XFEATURE_MASK_XTILE);
|
||
} else if (static_cpu_has(X86_FEATURE_FXSR)) {
|
||
kfpu_restore_fxsr(state);
|
||
} else {
|
||
kfpu_restore_fsave(state);
|
||
}
|
||
out:
|
||
local_irq_enable();
|
||
preempt_enable();
|
||
|
||
}
|
||
|
||
#else
|
||
|
||
#error "Exactly one of KERNEL_EXPORTS_X86_FPU or HAVE_KERNEL_FPU_INTERNAL" \
|
||
" must be defined"
|
||
|
||
#endif /* defined(HAVE_KERNEL_FPU_INTERNAL */
|
||
#endif /* defined(KERNEL_EXPORTS_X86_FPU) */
|
||
|
||
/*
|
||
* Linux kernel provides an interface for CPU feature testing.
|
||
*/
|
||
|
||
/*
|
||
* Detect register set support
|
||
*/
|
||
|
||
/*
|
||
* Check if OS supports AVX and AVX2 by checking XCR0
|
||
* Only call this function if CPUID indicates that AVX feature is
|
||
* supported by the CPU, otherwise it might be an illegal instruction.
|
||
*/
|
||
static inline uint64_t
|
||
zfs_xgetbv(uint32_t index)
|
||
{
|
||
uint32_t eax, edx;
|
||
/* xgetbv - instruction byte code */
|
||
__asm__ __volatile__(".byte 0x0f; .byte 0x01; .byte 0xd0"
|
||
: "=a" (eax), "=d" (edx)
|
||
: "c" (index));
|
||
|
||
return ((((uint64_t)edx)<<32) | (uint64_t)eax);
|
||
}
|
||
|
||
|
||
static inline boolean_t
|
||
__simd_state_enabled(const uint64_t state)
|
||
{
|
||
boolean_t has_osxsave;
|
||
uint64_t xcr0;
|
||
|
||
#if defined(X86_FEATURE_OSXSAVE)
|
||
has_osxsave = !!boot_cpu_has(X86_FEATURE_OSXSAVE);
|
||
#else
|
||
has_osxsave = B_FALSE;
|
||
#endif
|
||
if (!has_osxsave)
|
||
return (B_FALSE);
|
||
|
||
xcr0 = zfs_xgetbv(0);
|
||
return ((xcr0 & state) == state);
|
||
}
|
||
|
||
#define _XSTATE_SSE_AVX (0x2 | 0x4)
|
||
#define _XSTATE_AVX512 (0xE0 | _XSTATE_SSE_AVX)
|
||
|
||
#define __ymm_enabled() __simd_state_enabled(_XSTATE_SSE_AVX)
|
||
#define __zmm_enabled() __simd_state_enabled(_XSTATE_AVX512)
|
||
|
||
/*
|
||
* Check if SSE instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_sse_available(void)
|
||
{
|
||
return (!!boot_cpu_has(X86_FEATURE_XMM));
|
||
}
|
||
|
||
/*
|
||
* Check if SSE2 instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_sse2_available(void)
|
||
{
|
||
return (!!boot_cpu_has(X86_FEATURE_XMM2));
|
||
}
|
||
|
||
/*
|
||
* Check if SSE3 instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_sse3_available(void)
|
||
{
|
||
return (!!boot_cpu_has(X86_FEATURE_XMM3));
|
||
}
|
||
|
||
/*
|
||
* Check if SSSE3 instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_ssse3_available(void)
|
||
{
|
||
return (!!boot_cpu_has(X86_FEATURE_SSSE3));
|
||
}
|
||
|
||
/*
|
||
* Check if SSE4.1 instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_sse4_1_available(void)
|
||
{
|
||
return (!!boot_cpu_has(X86_FEATURE_XMM4_1));
|
||
}
|
||
|
||
/*
|
||
* Check if SSE4.2 instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_sse4_2_available(void)
|
||
{
|
||
return (!!boot_cpu_has(X86_FEATURE_XMM4_2));
|
||
}
|
||
|
||
/*
|
||
* Check if AVX instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx_available(void)
|
||
{
|
||
return (boot_cpu_has(X86_FEATURE_AVX) && __ymm_enabled());
|
||
}
|
||
|
||
/*
|
||
* Check if AVX2 instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx2_available(void)
|
||
{
|
||
return (boot_cpu_has(X86_FEATURE_AVX2) && __ymm_enabled());
|
||
}
|
||
|
||
/*
|
||
* Check if BMI1 instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_bmi1_available(void)
|
||
{
|
||
#if defined(X86_FEATURE_BMI1)
|
||
return (!!boot_cpu_has(X86_FEATURE_BMI1));
|
||
#else
|
||
return (B_FALSE);
|
||
#endif
|
||
}
|
||
|
||
/*
|
||
* Check if BMI2 instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_bmi2_available(void)
|
||
{
|
||
#if defined(X86_FEATURE_BMI2)
|
||
return (!!boot_cpu_has(X86_FEATURE_BMI2));
|
||
#else
|
||
return (B_FALSE);
|
||
#endif
|
||
}
|
||
|
||
/*
|
||
* Check if AES instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_aes_available(void)
|
||
{
|
||
#if defined(X86_FEATURE_AES)
|
||
return (!!boot_cpu_has(X86_FEATURE_AES));
|
||
#else
|
||
return (B_FALSE);
|
||
#endif
|
||
}
|
||
|
||
/*
|
||
* Check if PCLMULQDQ instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_pclmulqdq_available(void)
|
||
{
|
||
#if defined(X86_FEATURE_PCLMULQDQ)
|
||
return (!!boot_cpu_has(X86_FEATURE_PCLMULQDQ));
|
||
#else
|
||
return (B_FALSE);
|
||
#endif
|
||
}
|
||
|
||
/*
|
||
* Check if MOVBE instruction is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_movbe_available(void)
|
||
{
|
||
#if defined(X86_FEATURE_MOVBE)
|
||
return (!!boot_cpu_has(X86_FEATURE_MOVBE));
|
||
#else
|
||
return (B_FALSE);
|
||
#endif
|
||
}
|
||
|
||
/*
|
||
* Check if SHA_NI instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_shani_available(void)
|
||
{
|
||
#if defined(X86_FEATURE_SHA_NI)
|
||
return (!!boot_cpu_has(X86_FEATURE_SHA_NI));
|
||
#else
|
||
return (B_FALSE);
|
||
#endif
|
||
}
|
||
|
||
/*
|
||
* AVX-512 family of instruction sets:
|
||
*
|
||
* AVX512F Foundation
|
||
* AVX512CD Conflict Detection Instructions
|
||
* AVX512ER Exponential and Reciprocal Instructions
|
||
* AVX512PF Prefetch Instructions
|
||
*
|
||
* AVX512BW Byte and Word Instructions
|
||
* AVX512DQ Double-word and Quadword Instructions
|
||
* AVX512VL Vector Length Extensions
|
||
*
|
||
* AVX512IFMA Integer Fused Multiply Add (Not supported by kernel 4.4)
|
||
* AVX512VBMI Vector Byte Manipulation Instructions
|
||
*/
|
||
|
||
/*
|
||
* Check if AVX512F instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx512f_available(void)
|
||
{
|
||
boolean_t has_avx512 = B_FALSE;
|
||
|
||
#if defined(X86_FEATURE_AVX512F)
|
||
has_avx512 = !!boot_cpu_has(X86_FEATURE_AVX512F);
|
||
#endif
|
||
return (has_avx512 && __zmm_enabled());
|
||
}
|
||
|
||
/*
|
||
* Check if AVX512CD instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx512cd_available(void)
|
||
{
|
||
boolean_t has_avx512 = B_FALSE;
|
||
|
||
#if defined(X86_FEATURE_AVX512CD)
|
||
has_avx512 = boot_cpu_has(X86_FEATURE_AVX512F) &&
|
||
boot_cpu_has(X86_FEATURE_AVX512CD);
|
||
#endif
|
||
return (has_avx512 && __zmm_enabled());
|
||
}
|
||
|
||
/*
|
||
* Check if AVX512ER instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx512er_available(void)
|
||
{
|
||
boolean_t has_avx512 = B_FALSE;
|
||
|
||
#if defined(X86_FEATURE_AVX512ER)
|
||
has_avx512 = boot_cpu_has(X86_FEATURE_AVX512F) &&
|
||
boot_cpu_has(X86_FEATURE_AVX512ER);
|
||
#endif
|
||
return (has_avx512 && __zmm_enabled());
|
||
}
|
||
|
||
/*
|
||
* Check if AVX512PF instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx512pf_available(void)
|
||
{
|
||
boolean_t has_avx512 = B_FALSE;
|
||
|
||
#if defined(X86_FEATURE_AVX512PF)
|
||
has_avx512 = boot_cpu_has(X86_FEATURE_AVX512F) &&
|
||
boot_cpu_has(X86_FEATURE_AVX512PF);
|
||
#endif
|
||
return (has_avx512 && __zmm_enabled());
|
||
}
|
||
|
||
/*
|
||
* Check if AVX512BW instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx512bw_available(void)
|
||
{
|
||
boolean_t has_avx512 = B_FALSE;
|
||
|
||
#if defined(X86_FEATURE_AVX512BW)
|
||
has_avx512 = boot_cpu_has(X86_FEATURE_AVX512F) &&
|
||
boot_cpu_has(X86_FEATURE_AVX512BW);
|
||
#endif
|
||
|
||
return (has_avx512 && __zmm_enabled());
|
||
}
|
||
|
||
/*
|
||
* Check if AVX512DQ instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx512dq_available(void)
|
||
{
|
||
boolean_t has_avx512 = B_FALSE;
|
||
|
||
#if defined(X86_FEATURE_AVX512DQ)
|
||
has_avx512 = boot_cpu_has(X86_FEATURE_AVX512F) &&
|
||
boot_cpu_has(X86_FEATURE_AVX512DQ);
|
||
#endif
|
||
return (has_avx512 && __zmm_enabled());
|
||
}
|
||
|
||
/*
|
||
* Check if AVX512VL instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx512vl_available(void)
|
||
{
|
||
boolean_t has_avx512 = B_FALSE;
|
||
|
||
#if defined(X86_FEATURE_AVX512VL)
|
||
has_avx512 = boot_cpu_has(X86_FEATURE_AVX512F) &&
|
||
boot_cpu_has(X86_FEATURE_AVX512VL);
|
||
#endif
|
||
return (has_avx512 && __zmm_enabled());
|
||
}
|
||
|
||
/*
|
||
* Check if AVX512IFMA instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx512ifma_available(void)
|
||
{
|
||
boolean_t has_avx512 = B_FALSE;
|
||
|
||
#if defined(X86_FEATURE_AVX512IFMA)
|
||
has_avx512 = boot_cpu_has(X86_FEATURE_AVX512F) &&
|
||
boot_cpu_has(X86_FEATURE_AVX512IFMA);
|
||
#endif
|
||
return (has_avx512 && __zmm_enabled());
|
||
}
|
||
|
||
/*
|
||
* Check if AVX512VBMI instruction set is available
|
||
*/
|
||
static inline boolean_t
|
||
zfs_avx512vbmi_available(void)
|
||
{
|
||
boolean_t has_avx512 = B_FALSE;
|
||
|
||
#if defined(X86_FEATURE_AVX512VBMI)
|
||
has_avx512 = boot_cpu_has(X86_FEATURE_AVX512F) &&
|
||
boot_cpu_has(X86_FEATURE_AVX512VBMI);
|
||
#endif
|
||
return (has_avx512 && __zmm_enabled());
|
||
}
|
||
|
||
#endif /* defined(__x86) */
|
||
|
||
#endif /* _LINUX_SIMD_X86_H */
|