mirror of
https://gitlab.freedesktop.org/pipewire/pipewire.git
synced 2025-11-02 09:01:50 -05:00
Including C headers inside of `extern "C"` breaks use from C++. Hoist the includes of standard C headers above the block so we don't try to mangle the stdlib. I initially tried to scope this with a targeted change but it's too hard to do correctly that way. This way, we avoid whack-a-mole. Firefox is working around this in their e21461b7b8b39cc31ba53c47d4f6f310c673ff2f commit. Bug: https://bugzilla.mozilla.org/1953080
209 lines
5.8 KiB
C
209 lines
5.8 KiB
C
/* Simple Plugin API */
|
|
/* SPDX-FileCopyrightText: Copyright © 2018 Wim Taymans */
|
|
/* SPDX-License-Identifier: MIT */
|
|
|
|
#ifndef SPA_CPU_H
|
|
#define SPA_CPU_H
|
|
|
|
#include <stdarg.h>
|
|
#include <errno.h>
|
|
|
|
#include <spa/utils/defs.h>
|
|
#include <spa/utils/hook.h>
|
|
|
|
#ifdef __cplusplus
|
|
extern "C" {
|
|
#endif
|
|
|
|
#ifndef SPA_API_CPU
|
|
#ifdef SPA_API_IMPL
|
|
#define SPA_API_CPU SPA_API_IMPL
|
|
#else
|
|
#define SPA_API_CPU static inline
|
|
#endif
|
|
#endif
|
|
|
|
/** \defgroup spa_cpu CPU
|
|
* Querying CPU properties
|
|
*/
|
|
|
|
/**
|
|
* \addtogroup spa_cpu
|
|
* \{
|
|
*/
|
|
|
|
/**
|
|
* The CPU features interface
|
|
*/
|
|
#define SPA_TYPE_INTERFACE_CPU SPA_TYPE_INFO_INTERFACE_BASE "CPU"
|
|
|
|
#define SPA_VERSION_CPU 0
|
|
struct spa_cpu { struct spa_interface iface; };
|
|
|
|
/* x86 specific */
|
|
#define SPA_CPU_FLAG_MMX (1<<0) /**< standard MMX */
|
|
#define SPA_CPU_FLAG_MMXEXT (1<<1) /**< SSE integer or AMD MMX ext */
|
|
#define SPA_CPU_FLAG_3DNOW (1<<2) /**< AMD 3DNOW */
|
|
#define SPA_CPU_FLAG_SSE (1<<3) /**< SSE */
|
|
#define SPA_CPU_FLAG_SSE2 (1<<4) /**< SSE2 */
|
|
#define SPA_CPU_FLAG_3DNOWEXT (1<<5) /**< AMD 3DNowExt */
|
|
#define SPA_CPU_FLAG_SSE3 (1<<6) /**< Prescott SSE3 */
|
|
#define SPA_CPU_FLAG_SSSE3 (1<<7) /**< Conroe SSSE3 */
|
|
#define SPA_CPU_FLAG_SSE41 (1<<8) /**< Penryn SSE4.1 */
|
|
#define SPA_CPU_FLAG_SSE42 (1<<9) /**< Nehalem SSE4.2 */
|
|
#define SPA_CPU_FLAG_AESNI (1<<10) /**< Advanced Encryption Standard */
|
|
#define SPA_CPU_FLAG_AVX (1<<11) /**< AVX */
|
|
#define SPA_CPU_FLAG_XOP (1<<12) /**< Bulldozer XOP */
|
|
#define SPA_CPU_FLAG_FMA4 (1<<13) /**< Bulldozer FMA4 */
|
|
#define SPA_CPU_FLAG_CMOV (1<<14) /**< supports cmov */
|
|
#define SPA_CPU_FLAG_AVX2 (1<<15) /**< AVX2 */
|
|
#define SPA_CPU_FLAG_FMA3 (1<<16) /**< Haswell FMA3 */
|
|
#define SPA_CPU_FLAG_BMI1 (1<<17) /**< Bit Manipulation Instruction Set 1 */
|
|
#define SPA_CPU_FLAG_BMI2 (1<<18) /**< Bit Manipulation Instruction Set 2 */
|
|
#define SPA_CPU_FLAG_AVX512 (1<<19) /**< AVX-512 */
|
|
#define SPA_CPU_FLAG_SLOW_UNALIGNED (1<<20) /**< unaligned loads/stores are slow */
|
|
|
|
/* PPC specific */
|
|
#define SPA_CPU_FLAG_ALTIVEC (1<<0) /**< standard */
|
|
#define SPA_CPU_FLAG_VSX (1<<1) /**< ISA 2.06 */
|
|
#define SPA_CPU_FLAG_POWER8 (1<<2) /**< ISA 2.07 */
|
|
|
|
/* ARM specific */
|
|
#define SPA_CPU_FLAG_ARMV5TE (1 << 0)
|
|
#define SPA_CPU_FLAG_ARMV6 (1 << 1)
|
|
#define SPA_CPU_FLAG_ARMV6T2 (1 << 2)
|
|
#define SPA_CPU_FLAG_VFP (1 << 3)
|
|
#define SPA_CPU_FLAG_VFPV3 (1 << 4)
|
|
#define SPA_CPU_FLAG_NEON (1 << 5)
|
|
#define SPA_CPU_FLAG_ARMV8 (1 << 6)
|
|
|
|
/* RISCV specific */
|
|
#define SPA_CPU_FLAG_RISCV_V (1 << 0)
|
|
|
|
#define SPA_CPU_FORCE_AUTODETECT ((uint32_t)-1)
|
|
|
|
#define SPA_CPU_VM_NONE (0)
|
|
#define SPA_CPU_VM_OTHER (1 << 0)
|
|
#define SPA_CPU_VM_KVM (1 << 1)
|
|
#define SPA_CPU_VM_QEMU (1 << 2)
|
|
#define SPA_CPU_VM_BOCHS (1 << 3)
|
|
#define SPA_CPU_VM_XEN (1 << 4)
|
|
#define SPA_CPU_VM_UML (1 << 5)
|
|
#define SPA_CPU_VM_VMWARE (1 << 6)
|
|
#define SPA_CPU_VM_ORACLE (1 << 7)
|
|
#define SPA_CPU_VM_MICROSOFT (1 << 8)
|
|
#define SPA_CPU_VM_ZVM (1 << 9)
|
|
#define SPA_CPU_VM_PARALLELS (1 << 10)
|
|
#define SPA_CPU_VM_BHYVE (1 << 11)
|
|
#define SPA_CPU_VM_QNX (1 << 12)
|
|
#define SPA_CPU_VM_ACRN (1 << 13)
|
|
#define SPA_CPU_VM_POWERVM (1 << 14)
|
|
|
|
SPA_API_CPU const char *spa_cpu_vm_type_to_string(uint32_t vm_type)
|
|
{
|
|
switch(vm_type) {
|
|
case SPA_CPU_VM_NONE:
|
|
return NULL;
|
|
case SPA_CPU_VM_KVM:
|
|
return "kvm";
|
|
case SPA_CPU_VM_QEMU:
|
|
return "qemu";
|
|
case SPA_CPU_VM_BOCHS:
|
|
return "bochs";
|
|
case SPA_CPU_VM_XEN:
|
|
return "xen";
|
|
case SPA_CPU_VM_UML:
|
|
return "uml";
|
|
case SPA_CPU_VM_VMWARE:
|
|
return "vmware";
|
|
case SPA_CPU_VM_ORACLE:
|
|
return "oracle";
|
|
case SPA_CPU_VM_MICROSOFT:
|
|
return "microsoft";
|
|
case SPA_CPU_VM_ZVM:
|
|
return "zvm";
|
|
case SPA_CPU_VM_PARALLELS:
|
|
return "parallels";
|
|
case SPA_CPU_VM_BHYVE:
|
|
return "bhyve";
|
|
case SPA_CPU_VM_QNX:
|
|
return "qnx";
|
|
case SPA_CPU_VM_ACRN:
|
|
return "acrn";
|
|
case SPA_CPU_VM_POWERVM:
|
|
return "powervm";
|
|
case SPA_CPU_VM_OTHER:
|
|
return "other";
|
|
default:
|
|
return "unknown";
|
|
}
|
|
}
|
|
|
|
/**
|
|
* methods
|
|
*/
|
|
struct spa_cpu_methods {
|
|
/** the version of the methods. This can be used to expand this
|
|
structure in the future */
|
|
#define SPA_VERSION_CPU_METHODS 2
|
|
uint32_t version;
|
|
|
|
/** get CPU flags */
|
|
uint32_t (*get_flags) (void *object);
|
|
|
|
/** force CPU flags, use SPA_CPU_FORCE_AUTODETECT to autodetect CPU flags */
|
|
int (*force_flags) (void *object, uint32_t flags);
|
|
|
|
/** get number of CPU cores */
|
|
uint32_t (*get_count) (void *object);
|
|
|
|
/** get maximum required alignment of data */
|
|
uint32_t (*get_max_align) (void *object);
|
|
|
|
/* check if running in a VM. Since:1 */
|
|
uint32_t (*get_vm_type) (void *object);
|
|
|
|
/* denormals will be handled as zero, either with FTZ or DAZ.
|
|
* Since:2 */
|
|
int (*zero_denormals) (void *object, bool enable);
|
|
};
|
|
|
|
SPA_API_CPU uint32_t spa_cpu_get_flags(struct spa_cpu *c)
|
|
{
|
|
return spa_api_method_r(uint32_t, 0, spa_cpu, &c->iface, get_flags, 0);
|
|
}
|
|
SPA_API_CPU int spa_cpu_force_flags(struct spa_cpu *c, uint32_t flags)
|
|
{
|
|
return spa_api_method_r(int, -ENOTSUP, spa_cpu, &c->iface, force_flags, 0, flags);
|
|
}
|
|
SPA_API_CPU uint32_t spa_cpu_get_count(struct spa_cpu *c)
|
|
{
|
|
return spa_api_method_r(uint32_t, 0, spa_cpu, &c->iface, get_count, 0);
|
|
}
|
|
SPA_API_CPU uint32_t spa_cpu_get_max_align(struct spa_cpu *c)
|
|
{
|
|
return spa_api_method_r(uint32_t, 0, spa_cpu, &c->iface, get_max_align, 0);
|
|
}
|
|
SPA_API_CPU uint32_t spa_cpu_get_vm_type(struct spa_cpu *c)
|
|
{
|
|
return spa_api_method_r(uint32_t, 0, spa_cpu, &c->iface, get_vm_type, 1);
|
|
}
|
|
SPA_API_CPU int spa_cpu_zero_denormals(struct spa_cpu *c, bool enable)
|
|
{
|
|
return spa_api_method_r(int, -ENOTSUP, spa_cpu, &c->iface, zero_denormals, 2, enable);
|
|
}
|
|
|
|
/** keys can be given when initializing the cpu handle */
|
|
#define SPA_KEY_CPU_FORCE "cpu.force" /**< force cpu flags */
|
|
#define SPA_KEY_CPU_VM_TYPE "cpu.vm.type" /**< force a VM type */
|
|
#define SPA_KEY_CPU_ZERO_DENORMALS "cpu.zero.denormals" /**< zero denormals */
|
|
|
|
/**
|
|
* \}
|
|
*/
|
|
|
|
#ifdef __cplusplus
|
|
} /* extern "C" */
|
|
#endif
|
|
|
|
#endif /* SPA_CPU_H */
|