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