| /* |
| * SPDX-License-Identifier: GPL-2.0-or-later |
| * Host specific cpu identification for ppc. |
| */ |
| |
| #include "qemu/osdep.h" |
| #include "host/cpuinfo.h" |
| |
| #ifdef CONFIG_LINUX |
| # include <asm/cputable.h> |
| # ifdef CONFIG_GETAUXVAL |
| # include <sys/auxv.h> |
| # else |
| # include "elf.h" |
| # endif |
| #endif |
| #if defined(CONFIG_ELF_AUX_INFO) |
| # include <sys/auxv.h> |
| # include <machine/cpu.h> |
| # ifndef PPC_FEATURE2_ARCH_3_1 |
| # define PPC_FEATURE2_ARCH_3_1 0 |
| # endif |
| # define PPC_FEATURE2_VEC_CRYPTO PPC_FEATURE2_HAS_VEC_CRYPTO |
| #endif |
| |
| unsigned cpuinfo; |
| |
| /* Called both as constructor and (possibly) via other constructors. */ |
| unsigned __attribute__((constructor)) cpuinfo_init(void) |
| { |
| unsigned info = cpuinfo; |
| |
| if (info) { |
| return info; |
| } |
| |
| info = CPUINFO_ALWAYS; |
| |
| #if defined(CONFIG_LINUX) || defined(CONFIG_ELF_AUX_INFO) |
| unsigned long hwcap = qemu_getauxval(AT_HWCAP); |
| unsigned long hwcap2 = qemu_getauxval(AT_HWCAP2); |
| |
| /* Version numbers are monotonic, and so imply all lower versions. */ |
| if (hwcap2 & PPC_FEATURE2_ARCH_3_1) { |
| info |= CPUINFO_V3_1 | CPUINFO_V3_0 | CPUINFO_V2_07 | CPUINFO_V2_06; |
| } else if (hwcap2 & PPC_FEATURE2_ARCH_3_00) { |
| info |= CPUINFO_V3_0 | CPUINFO_V2_07 | CPUINFO_V2_06; |
| } else if (hwcap2 & PPC_FEATURE2_ARCH_2_07) { |
| info |= CPUINFO_V2_07 | CPUINFO_V2_06; |
| } else if (hwcap & PPC_FEATURE_ARCH_2_06) { |
| info |= CPUINFO_V2_06; |
| } |
| |
| if (hwcap2 & PPC_FEATURE2_ISEL) { |
| info |= CPUINFO_ISEL; |
| } |
| if (hwcap & PPC_FEATURE_HAS_ALTIVEC) { |
| info |= CPUINFO_ALTIVEC; |
| /* We only care about the portion of VSX that overlaps Altivec. */ |
| if (hwcap & PPC_FEATURE_HAS_VSX) { |
| info |= CPUINFO_VSX; |
| /* |
| * We use VSX especially for little-endian, but we should |
| * always have both anyway, since VSX came with Power7 |
| * and crypto came with Power8. |
| */ |
| if (hwcap2 & PPC_FEATURE2_VEC_CRYPTO) { |
| info |= CPUINFO_CRYPTO; |
| } |
| } |
| } |
| #endif |
| |
| cpuinfo = info; |
| return info; |
| } |