xref: /aosp_15_r20/external/libaom/aom_ports/ppc_cpudetect.c (revision 77c1e3ccc04c968bd2bc212e87364f250e820521)
1*77c1e3ccSAndroid Build Coastguard Worker /*
2*77c1e3ccSAndroid Build Coastguard Worker  * Copyright (c) 2018, Alliance for Open Media. All rights reserved.
3*77c1e3ccSAndroid Build Coastguard Worker  *
4*77c1e3ccSAndroid Build Coastguard Worker  * This source code is subject to the terms of the BSD 2 Clause License and
5*77c1e3ccSAndroid Build Coastguard Worker  * the Alliance for Open Media Patent License 1.0. If the BSD 2 Clause License
6*77c1e3ccSAndroid Build Coastguard Worker  * was not distributed with this source code in the LICENSE file, you can
7*77c1e3ccSAndroid Build Coastguard Worker  * obtain it at www.aomedia.org/license/software. If the Alliance for Open
8*77c1e3ccSAndroid Build Coastguard Worker  * Media Patent License 1.0 was not distributed with this source code in the
9*77c1e3ccSAndroid Build Coastguard Worker  * PATENTS file, you can obtain it at www.aomedia.org/license/patent.
10*77c1e3ccSAndroid Build Coastguard Worker  */
11*77c1e3ccSAndroid Build Coastguard Worker 
12*77c1e3ccSAndroid Build Coastguard Worker #include <fcntl.h>
13*77c1e3ccSAndroid Build Coastguard Worker #include <unistd.h>
14*77c1e3ccSAndroid Build Coastguard Worker #include <stdint.h>
15*77c1e3ccSAndroid Build Coastguard Worker #include <asm/cputable.h>
16*77c1e3ccSAndroid Build Coastguard Worker #include <linux/auxvec.h>
17*77c1e3ccSAndroid Build Coastguard Worker 
18*77c1e3ccSAndroid Build Coastguard Worker #include "config/aom_config.h"
19*77c1e3ccSAndroid Build Coastguard Worker 
20*77c1e3ccSAndroid Build Coastguard Worker #include "aom_ports/ppc.h"
21*77c1e3ccSAndroid Build Coastguard Worker 
22*77c1e3ccSAndroid Build Coastguard Worker #if CONFIG_RUNTIME_CPU_DETECT
cpu_env_flags(int * flags)23*77c1e3ccSAndroid Build Coastguard Worker static int cpu_env_flags(int *flags) {
24*77c1e3ccSAndroid Build Coastguard Worker   char *env;
25*77c1e3ccSAndroid Build Coastguard Worker   env = getenv("AOM_SIMD_CAPS");
26*77c1e3ccSAndroid Build Coastguard Worker   if (env && *env) {
27*77c1e3ccSAndroid Build Coastguard Worker     *flags = (int)strtol(env, NULL, 0);
28*77c1e3ccSAndroid Build Coastguard Worker     return 0;
29*77c1e3ccSAndroid Build Coastguard Worker   }
30*77c1e3ccSAndroid Build Coastguard Worker   *flags = 0;
31*77c1e3ccSAndroid Build Coastguard Worker   return -1;
32*77c1e3ccSAndroid Build Coastguard Worker }
33*77c1e3ccSAndroid Build Coastguard Worker 
cpu_env_mask(void)34*77c1e3ccSAndroid Build Coastguard Worker static int cpu_env_mask(void) {
35*77c1e3ccSAndroid Build Coastguard Worker   char *env;
36*77c1e3ccSAndroid Build Coastguard Worker   env = getenv("AOM_SIMD_CAPS_MASK");
37*77c1e3ccSAndroid Build Coastguard Worker   return env && *env ? (int)strtol(env, NULL, 0) : ~0;
38*77c1e3ccSAndroid Build Coastguard Worker }
39*77c1e3ccSAndroid Build Coastguard Worker 
ppc_simd_caps(void)40*77c1e3ccSAndroid Build Coastguard Worker int ppc_simd_caps(void) {
41*77c1e3ccSAndroid Build Coastguard Worker   int flags;
42*77c1e3ccSAndroid Build Coastguard Worker   int mask;
43*77c1e3ccSAndroid Build Coastguard Worker   int fd;
44*77c1e3ccSAndroid Build Coastguard Worker   ssize_t count;
45*77c1e3ccSAndroid Build Coastguard Worker   unsigned int i;
46*77c1e3ccSAndroid Build Coastguard Worker   uint64_t buf[64];
47*77c1e3ccSAndroid Build Coastguard Worker 
48*77c1e3ccSAndroid Build Coastguard Worker   // If AOM_SIMD_CAPS_MASK is set then allow only those capabilities.
49*77c1e3ccSAndroid Build Coastguard Worker   if (!cpu_env_flags(&flags)) {
50*77c1e3ccSAndroid Build Coastguard Worker     return flags;
51*77c1e3ccSAndroid Build Coastguard Worker   }
52*77c1e3ccSAndroid Build Coastguard Worker 
53*77c1e3ccSAndroid Build Coastguard Worker   mask = cpu_env_mask();
54*77c1e3ccSAndroid Build Coastguard Worker 
55*77c1e3ccSAndroid Build Coastguard Worker   fd = open("/proc/self/auxv", O_RDONLY);
56*77c1e3ccSAndroid Build Coastguard Worker   if (fd < 0) {
57*77c1e3ccSAndroid Build Coastguard Worker     return 0;
58*77c1e3ccSAndroid Build Coastguard Worker   }
59*77c1e3ccSAndroid Build Coastguard Worker 
60*77c1e3ccSAndroid Build Coastguard Worker   while ((count = read(fd, buf, sizeof(buf))) > 0) {
61*77c1e3ccSAndroid Build Coastguard Worker     for (i = 0; i < (count / sizeof(*buf)); i += 2) {
62*77c1e3ccSAndroid Build Coastguard Worker       if (buf[i] == AT_HWCAP) {
63*77c1e3ccSAndroid Build Coastguard Worker #if HAVE_VSX
64*77c1e3ccSAndroid Build Coastguard Worker         if (buf[i + 1] & PPC_FEATURE_HAS_VSX) {
65*77c1e3ccSAndroid Build Coastguard Worker           flags |= HAS_VSX;
66*77c1e3ccSAndroid Build Coastguard Worker         }
67*77c1e3ccSAndroid Build Coastguard Worker #endif  // HAVE_VSX
68*77c1e3ccSAndroid Build Coastguard Worker         goto out_close;
69*77c1e3ccSAndroid Build Coastguard Worker       } else if (buf[i] == AT_NULL) {
70*77c1e3ccSAndroid Build Coastguard Worker         goto out_close;
71*77c1e3ccSAndroid Build Coastguard Worker       }
72*77c1e3ccSAndroid Build Coastguard Worker     }
73*77c1e3ccSAndroid Build Coastguard Worker   }
74*77c1e3ccSAndroid Build Coastguard Worker out_close:
75*77c1e3ccSAndroid Build Coastguard Worker   close(fd);
76*77c1e3ccSAndroid Build Coastguard Worker   return flags & mask;
77*77c1e3ccSAndroid Build Coastguard Worker }
78*77c1e3ccSAndroid Build Coastguard Worker #else
79*77c1e3ccSAndroid Build Coastguard Worker // If there is no RTCD the function pointers are not used and can not be
80*77c1e3ccSAndroid Build Coastguard Worker // changed.
ppc_simd_caps(void)81*77c1e3ccSAndroid Build Coastguard Worker int ppc_simd_caps(void) { return 0; }
82*77c1e3ccSAndroid Build Coastguard Worker #endif  // CONFIG_RUNTIME_CPU_DETECT
83