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