| /* | 
 |  * Copyright (c) 2021, Alliance for Open Media. All rights reserved | 
 |  * | 
 |  * This source code is subject to the terms of the BSD 3-Clause Clear License | 
 |  * and the Alliance for Open Media Patent License 1.0. If the BSD 3-Clause Clear | 
 |  * License was not distributed with this source code in the LICENSE file, you | 
 |  * can obtain it at aomedia.org/license/software-license/bsd-3-c-c/.  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 | 
 |  * aomedia.org/license/patent-license/. | 
 |  */ | 
 |  | 
 | #include <stdlib.h> | 
 | #include <string.h> | 
 | #include "aom_ports/arm.h" | 
 | #include "config/aom_config.h" | 
 |  | 
 | #ifdef WINAPI_FAMILY | 
 | #include <winapifamily.h> | 
 | #if !WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP) | 
 | #define getenv(x) NULL | 
 | #endif | 
 | #endif | 
 |  | 
 | static int arm_cpu_env_flags(int *flags) { | 
 |   char *env; | 
 |   env = getenv("AOM_SIMD_CAPS"); | 
 |   if (env && *env) { | 
 |     *flags = (int)strtol(env, NULL, 0); | 
 |     return 0; | 
 |   } | 
 |   *flags = 0; | 
 |   return -1; | 
 | } | 
 |  | 
 | static int arm_cpu_env_mask(void) { | 
 |   char *env; | 
 |   env = getenv("AOM_SIMD_CAPS_MASK"); | 
 |   return env && *env ? (int)strtol(env, NULL, 0) : ~0; | 
 | } | 
 |  | 
 | #if !CONFIG_RUNTIME_CPU_DETECT || defined(__APPLE__) | 
 |  | 
 | int aom_arm_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; | 
 |   int mask; | 
 |   if (!arm_cpu_env_flags(&flags)) { | 
 |     return flags; | 
 |   } | 
 |   mask = arm_cpu_env_mask(); | 
 | #if HAVE_NEON | 
 |   flags |= HAS_NEON; | 
 | #endif /* HAVE_NEON */ | 
 |   return flags & mask; | 
 | } | 
 |  | 
 | #elif defined(_MSC_VER) /* end !CONFIG_RUNTIME_CPU_DETECT || __APPLE__ */ | 
 | /*For GetExceptionCode() and EXCEPTION_ILLEGAL_INSTRUCTION.*/ | 
 | #define WIN32_LEAN_AND_MEAN | 
 | #define WIN32_EXTRA_LEAN | 
 | #include <windows.h> | 
 |  | 
 | int aom_arm_cpu_caps(void) { | 
 |   int flags; | 
 |   int mask; | 
 |   if (!arm_cpu_env_flags(&flags)) { | 
 |     return flags; | 
 |   } | 
 |   mask = arm_cpu_env_mask(); | 
 | /* 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. | 
 |  */ | 
 | #if HAVE_NEON | 
 |   if (mask & HAS_NEON) { | 
 |     __try { | 
 |       /*VORR q0,q0,q0*/ | 
 |       __emit(0xF2200150); | 
 |       flags |= HAS_NEON; | 
 |     } __except (GetExceptionCode() == EXCEPTION_ILLEGAL_INSTRUCTION) { | 
 |       /*Ignore exception.*/ | 
 |     } | 
 |   } | 
 | #endif /* HAVE_NEON */ | 
 |   return flags & mask; | 
 | } | 
 |  | 
 | #elif defined(__ANDROID__) /* end _MSC_VER */ | 
 | #include <cpu-features.h> | 
 |  | 
 | int aom_arm_cpu_caps(void) { | 
 |   int flags; | 
 |   int mask; | 
 |   uint64_t features; | 
 |   if (!arm_cpu_env_flags(&flags)) { | 
 |     return flags; | 
 |   } | 
 |   mask = arm_cpu_env_mask(); | 
 |   features = android_getCpuFeatures(); | 
 |  | 
 | #if HAVE_NEON | 
 |   if (features & ANDROID_CPU_ARM_FEATURE_NEON) flags |= HAS_NEON; | 
 | #endif /* HAVE_NEON */ | 
 |   return flags & mask; | 
 | } | 
 |  | 
 | #elif defined(__linux__) /* end __ANDROID__ */ | 
 |  | 
 | #include <stdio.h> | 
 |  | 
 | int aom_arm_cpu_caps(void) { | 
 |   FILE *fin; | 
 |   int flags; | 
 |   int mask; | 
 |   if (!arm_cpu_env_flags(&flags)) { | 
 |     return flags; | 
 |   } | 
 |   mask = arm_cpu_env_mask(); | 
 |   /* Reading /proc/self/auxv would be easier, but that doesn't work reliably | 
 |    *  on Android. | 
 |    * This also means that detection will fail in Scratchbox. | 
 |    */ | 
 |   fin = fopen("/proc/cpuinfo", "r"); | 
 |   if (fin != NULL) { | 
 |     /* 512 should be enough for anybody (it's even enough for all the flags | 
 |      * that x86 has accumulated... so far). | 
 |      */ | 
 |     char buf[512]; | 
 |     while (fgets(buf, 511, fin) != NULL) { | 
 | #if HAVE_NEON | 
 |       if (memcmp(buf, "Features", 8) == 0) { | 
 |         char *p; | 
 |         p = strstr(buf, " neon"); | 
 |         if (p != NULL && (p[5] == ' ' || p[5] == '\n')) { | 
 |           flags |= HAS_NEON; | 
 |         } | 
 |       } | 
 | #endif /* HAVE_NEON */ | 
 |     } | 
 |     fclose(fin); | 
 |   } | 
 |   return flags & mask; | 
 | } | 
 | #else  /* end __linux__ */ | 
 | #error \ | 
 |     "-DCONFIG_RUNTIME_CPU_DETECT=1 selected, but no CPU detection method " \ | 
 | "available for your platform. Reconfigure with -DCONFIG_RUNTIME_CPU_DETECT=0" | 
 | #endif |