efi: Allow bitness-agnostic protocol calls
authorLukas Wunner <lukas@wunner.de>
Sat, 12 Nov 2016 21:32:35 +0000 (21:32 +0000)
committerIngo Molnar <mingo@kernel.org>
Sun, 13 Nov 2016 07:23:16 +0000 (08:23 +0100)
We already have a macro to invoke boot services which on x86 adapts
automatically to the bitness of the EFI firmware:  efi_call_early().

The macro allows sharing of functions across arches and bitness variants
as long as those functions only call boot services.  However in practice
functions in the EFI stub contain a mix of boot services calls and
protocol calls.

Add an efi_call_proto() macro for bitness-agnostic protocol calls to
allow sharing more code across arches as well as deduplicating 32 bit
and 64 bit code paths.

On x86, implement it using a new efi_table_attr() macro for bitness-
agnostic table lookups.  Refactor efi_call_early() to make use of the
same macro.  (The resulting object code remains identical.)

Signed-off-by: Lukas Wunner <lukas@wunner.de>
Signed-off-by: Matt Fleming <matt@codeblueprint.co.uk>
Cc: Andreas Noever <andreas.noever@gmail.com>
Cc: Ard Biesheuvel <ard.biesheuvel@linaro.org>
Cc: Linus Torvalds <torvalds@linux-foundation.org>
Cc: Peter Jones <pjones@redhat.com>
Cc: Peter Zijlstra <peterz@infradead.org>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: linux-efi@vger.kernel.org
Link: http://lkml.kernel.org/r/20161112213237.8804-8-matt@codeblueprint.co.uk
Signed-off-by: Ingo Molnar <mingo@kernel.org>
arch/arm/include/asm/efi.h
arch/arm64/include/asm/efi.h
arch/x86/include/asm/efi.h

index 766bf9b781601f285bd648788ee53202ebcb4550..0b06f5341b451533c349de4b8f04e277b415cfd9 100644 (file)
@@ -57,6 +57,9 @@ void efi_virtmap_unload(void);
 #define __efi_call_early(f, ...)       f(__VA_ARGS__)
 #define efi_is_64bit()                 (false)
 
+#define efi_call_proto(protocol, f, instance, ...)                     \
+       ((protocol##_t *)instance)->f(instance, ##__VA_ARGS__)
+
 struct screen_info *alloc_screen_info(efi_system_table_t *sys_table_arg);
 void free_screen_info(efi_system_table_t *sys_table, struct screen_info *si);
 
index a9e54aad15ef401ae7e0badcdcbffadfe7a97d9f..771b3f0bc7570bbd05f6a9538935428822dee5d7 100644 (file)
@@ -51,6 +51,9 @@ int efi_set_mapping_permissions(struct mm_struct *mm, efi_memory_desc_t *md);
 #define __efi_call_early(f, ...)       f(__VA_ARGS__)
 #define efi_is_64bit()                 (true)
 
+#define efi_call_proto(protocol, f, instance, ...)                     \
+       ((protocol##_t *)instance)->f(instance, ##__VA_ARGS__)
+
 #define alloc_screen_info(x...)                &screen_info
 #define free_screen_info(x...)
 
index 389d700b961e48028a1a7b1797c41aa2fb339bf7..e99675b9c861dad48a9c4ba46cc63876255c2748 100644 (file)
@@ -210,12 +210,18 @@ static inline bool efi_is_64bit(void)
        return __efi_early()->is64;
 }
 
+#define efi_table_attr(table, attr, instance)                          \
+       (efi_is_64bit() ?                                               \
+               ((table##_64_t *)(unsigned long)instance)->attr :       \
+               ((table##_32_t *)(unsigned long)instance)->attr)
+
+#define efi_call_proto(protocol, f, instance, ...)                     \
+       __efi_early()->call(efi_table_attr(protocol, f, instance),      \
+               instance, ##__VA_ARGS__)
+
 #define efi_call_early(f, ...)                                         \
-       __efi_early()->call(efi_is_64bit() ?                            \
-               ((efi_boot_services_64_t *)(unsigned long)              \
-                       __efi_early()->boot_services)->f :              \
-               ((efi_boot_services_32_t *)(unsigned long)              \
-                       __efi_early()->boot_services)->f, __VA_ARGS__)
+       __efi_early()->call(efi_table_attr(efi_boot_services, f,        \
+               __efi_early()->boot_services), __VA_ARGS__)
 
 #define __efi_call_early(f, ...)                                       \
        __efi_early()->call((unsigned long)f, __VA_ARGS__);