arm64: Add hwcaps for crypto and CRC32 extensions.
authorSteve Capper <steve.capper@linaro.org>
Mon, 16 Dec 2013 21:04:36 +0000 (21:04 +0000)
committerGreg Hackmann <ghackmann@google.com>
Thu, 11 Sep 2014 17:21:04 +0000 (17:21 +0000)
Advertise the optional cryptographic and CRC32 instructions to
user space where present. Several hwcap bits [3-7] are allocated.

Signed-off-by: Steve Capper <steve.capper@linaro.org>
[bit 2 is taken now so use bits 3-7 instead]
Signed-off-by: Ard Biesheuvel <ard.biesheuvel@linaro.org>
Signed-off-by: Catalin Marinas <catalin.marinas@arm.com>
arch/arm64/include/uapi/asm/hwcap.h
arch/arm64/kernel/setup.c

index 9b12476e9c8567545c493704075f56c8643bbae6..73cf0f54d57cc2459ce424f627f2a15d2ac496e6 100644 (file)
 #define HWCAP_FP               (1 << 0)
 #define HWCAP_ASIMD            (1 << 1)
 #define HWCAP_EVTSTRM          (1 << 2)
-
+#define HWCAP_AES              (1 << 3)
+#define HWCAP_PMULL            (1 << 4)
+#define HWCAP_SHA1             (1 << 5)
+#define HWCAP_SHA2             (1 << 6)
+#define HWCAP_CRC32            (1 << 7)
 
 #endif /* _UAPI__ASM_HWCAP_H */
index d6c340eb45cc3076b659d8125e53e60c1506d60f..a480eae24ec98e30905958caec24044011b525db 100644 (file)
@@ -110,6 +110,7 @@ void __init early_print(const char *str, ...)
 static void __init setup_processor(void)
 {
        struct cpu_info *cpu_info;
+       u64 features, block;
 
        /*
         * locate processor in the list of supported processor
@@ -130,6 +131,37 @@ static void __init setup_processor(void)
 
        sprintf(init_utsname()->machine, "aarch64");
        elf_hwcap = 0;
+
+       /*
+        * ID_AA64ISAR0_EL1 contains 4-bit wide signed feature blocks.
+        * The blocks we test below represent incremental functionality
+        * for non-negative values. Negative values are reserved.
+        */
+       features = read_cpuid(ID_AA64ISAR0_EL1);
+       block = (features >> 4) & 0xf;
+       if (!(block & 0x8)) {
+               switch (block) {
+               default:
+               case 2:
+                       elf_hwcap |= HWCAP_PMULL;
+               case 1:
+                       elf_hwcap |= HWCAP_AES;
+               case 0:
+                       break;
+               }
+       }
+
+       block = (features >> 8) & 0xf;
+       if (block && !(block & 0x8))
+               elf_hwcap |= HWCAP_SHA1;
+
+       block = (features >> 12) & 0xf;
+       if (block && !(block & 0x8))
+               elf_hwcap |= HWCAP_SHA2;
+
+       block = (features >> 16) & 0xf;
+       if (block && !(block & 0x8))
+               elf_hwcap |= HWCAP_CRC32;
 }
 
 static void __init setup_machine_fdt(phys_addr_t dt_phys)
@@ -320,6 +352,11 @@ static const char *hwcap_str[] = {
        "fp",
        "asimd",
        "evtstrm",
+       "aes",
+       "pmull",
+       "sha1",
+       "sha2",
+       "crc32",
        NULL
 };