blob: af6774c5d250927ec369cdbac53da03f56900c6f [file] [log] [blame]
/*
* Copyright (c) 2023, Alliance for Open Media. All rights reserved.
*
* This source code is subject to the terms of the BSD 2 Clause License and
* the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License
* was not distributed with this source code in the LICENSE file, you can
* obtain it at www.aomedia.org/license/software. If the Alliance for Open
* Media Patent License 1.0 was not distributed with this source code in the
* PATENTS file, you can obtain it at www.aomedia.org/license/patent.
*/
// Feature detection code for Armv7-A / AArch32.
#include "arm_cpudetect.h"
#if !CONFIG_RUNTIME_CPU_DETECT
static int arm_get_cpu_caps(void) {
// This function should actually be a no-op. There is no way to adjust any of
// these because the RTCD tables do not exist: the functions are called
// statically.
int flags = 0;
#if HAVE_NEON
flags |= HAS_NEON;
#endif // HAVE_NEON
return flags;
}
#elif defined(_MSC_VER) // end !CONFIG_RUNTIME_CPU_DETECT
static int arm_get_cpu_caps(void) {
int flags = 0;
#if HAVE_NEON
// MSVC has no inline __asm support for Arm, but it does let you __emit
// instructions via their assembled hex code.
// All of these instructions should be essentially nops.
__try {
// VORR q0,q0,q0
__emit(0xF2200150);
flags |= HAS_NEON;
} __except (GetExceptionCode() == EXCEPTION_ILLEGAL_INSTRUCTION) {
// Ignore exception.
}
#endif // HAVE_NEON
return flags;
}
#elif defined(AOM_USE_ANDROID_CPU_FEATURES)
static int arm_get_cpu_caps(void) {
int flags = 0;
#if HAVE_NEON
uint64_t features = android_getCpuFeatures();
if (features & ANDROID_CPU_ARM_FEATURE_NEON) flags |= HAS_NEON;
#endif // HAVE_NEON
return flags;
}
#elif defined(__linux__) // end defined(AOM_USE_ANDROID_CPU_FEATURES)
#include <sys/auxv.h>
// Define hwcap values ourselves: building with an old auxv header where these
// hwcap values are not defined should not prevent features from being enabled.
#define AOM_AARCH32_HWCAP_NEON (1 << 12)
static int arm_get_cpu_caps(void) {
int flags = 0;
unsigned long hwcap = getauxval(AT_HWCAP);
#if HAVE_NEON
if (hwcap & AOM_AARCH32_HWCAP_NEON) flags |= HAS_NEON;
#endif // HAVE_NEON
return flags;
}
#else // end __linux__
#error \
"Runtime CPU detection selected, but no CPU detection method " \
"available for your platform. Rerun cmake with -DCONFIG_RUNTIME_CPU_DETECT=0."
#endif
int aom_arm_cpu_caps(void) {
int flags = 0;
if (arm_cpu_env_flags(&flags)) {
return flags;
}
return arm_get_cpu_caps() & arm_cpu_env_mask();
}