blob: 9a5ab7a8e09a1688d415d983275e4cae6bb3562c [file]
/*
* Copyright © 2018-2022, VideoLAN and dav1d authors
* Copyright © 2018-2022, Two Orioles, LLC
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* 1. Redistributions of source code must retain the above copyright notice, this
* list of conditions and the following disclaimer.
*
* 2. Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#ifndef CHECKASM_CPU_H
#define CHECKASM_CPU_H
#include "checkasm_config.h"
#include <stddef.h>
#include <stdint.h>
#include "checkasm/attributes.h"
#include "checkasm/checkasm.h"
#include "internal.h"
#if ARCH_X86
PACKED(typedef struct { uint32_t eax, ebx, ecx, edx; }) CpuidRegisters;
CHECKASM_SELF_API void
checkasm_cpu_cpuid(CpuidRegisters *regs, unsigned leaf, unsigned subleaf);
CHECKASM_SELF_API uint64_t checkasm_cpu_xgetbv(unsigned xcr);
/* Initializes internal state for checkasm_checked_call(). */
void checkasm_init_x86(void);
/* Returns whether the vzeroupper state check is active. Exported only for use
* inside the selftest. */
CHECKASM_SELF_API int checkasm_get_check_vzeroupper(void);
/* Returns cpuid and model name. */
char *checkasm_get_x86_cpuid(char *buf, size_t buflen);
/* YMM and ZMM registers on x86 are turned off to save power when they haven't
* been used for some period of time. When they are used there will be a
* "warmup" period during which performance will be reduced and inconsistent
* which is problematic when trying to benchmark individual functions. We can
* work around this by periodically issuing "dummy" instructions that uses
* those registers to keep them powered on. */
void checkasm_simd_warmup(void);
#elif ARCH_RISCV
/* Gets the CPU identification registers. */
int checkasm_get_cpuids(uint32_t *vendor, uintptr_t *arch, uintptr_t *imp);
const char *checkasm_get_riscv_vendor_name(uint32_t vendorid);
const char *checkasm_get_riscv_arch_name(char *buf, size_t len,
uint32_t vendorid, uintptr_t archid);
/* Checks if floating point registers are supported. */
CHECKASM_SELF_API int checkasm_has_float(void);
/* Checks if vector registers are supported. */
CHECKASM_SELF_API int checkasm_has_vector(void);
/* Returns the vector length in bits, 0 if unavailable. */
unsigned long checkasm_get_vlen(void);
void checkasm_checked_call_i(void);
void checkasm_checked_call_if(void);
void checkasm_checked_call_iv(void);
void checkasm_checked_call_ifv(void);
#elif ARCH_AARCH64
/* Returns a nonzero value if SVE is available, 0 otherwise */
int checkasm_has_sve(void);
/* Returns a nonzero value if SME is available, 0 otherwise */
int checkasm_has_sme(void);
#if HAVE_SVE
/* Returns the SVE vector length in bits */
int checkasm_sve_length(void);
#endif
#if HAVE_SME
/* Returns the SME vector length in bits */
int checkasm_sme_length(void);
#endif
#elif ARCH_ARM
void checkasm_init_arm(void);
void checkasm_checked_call_vfp(void *func, int dummy, ...);
void checkasm_checked_call_novfp(void *func, int dummy, ...);
/* Returns a nonzero value if VFP is available, 0 otherwise */
CHECKASM_SELF_API int checkasm_has_vfp(void);
/* Returns a nonzero value if VFP has 32 registers, 0 otherwise */
CHECKASM_SELF_API int checkasm_has_vfpd32(void);
#endif
void checkasm_init_cpu(void);
unsigned long checkasm_getauxval(unsigned long);
const char *checkasm_get_jedec_vendor_name(unsigned bank, unsigned offset);
#if (ARCH_ARM || ARCH_AARCH64) && defined(__linux__)
const char *checkasm_get_arm_cpuinfo(char *buf, size_t buflen, int affinity);
#endif
#if (ARCH_ARM || ARCH_AARCH64) && defined(_WIN32)
const char *checkasm_get_arm_win32_reg(char *buf, size_t buflen, int affinity);
#endif
/* Iterate over all known CPU information and run the callback on each line */
void checkasm_cpu_info(void (*info_cb)(void *priv, const char *fmt, ...), void *priv,
const CheckasmConfig *config);
#endif /* CHECKASM_CPU_H */