|  | /* | 
|  | * 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(); | 
|  | } |