Commit-ID:  3552fdf29f01e5a889e88202dc55b67aa6766620
Gitweb:     http://git.kernel.org/tip/3552fdf29f01e5a889e88202dc55b67aa6766620
Author:     Lukas Wunner <lu...@wunner.de>
AuthorDate: Sat, 12 Nov 2016 21:32:35 +0000
Committer:  Ingo Molnar <mi...@kernel.org>
CommitDate: Sun, 13 Nov 2016 08:23:16 +0100

efi: Allow bitness-agnostic protocol calls

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 <lu...@wunner.de>
Signed-off-by: Matt Fleming <m...@codeblueprint.co.uk>
Cc: Andreas Noever <andreas.noe...@gmail.com>
Cc: Ard Biesheuvel <ard.biesheu...@linaro.org>
Cc: Linus Torvalds <torva...@linux-foundation.org>
Cc: Peter Jones <pjo...@redhat.com>
Cc: Peter Zijlstra <pet...@infradead.org>
Cc: Thomas Gleixner <t...@linutronix.de>
Cc: linux-...@vger.kernel.org
Link: http://lkml.kernel.org/r/20161112213237.8804-8-m...@codeblueprint.co.uk
Signed-off-by: Ingo Molnar <mi...@kernel.org>
---
 arch/arm/include/asm/efi.h   |  3 +++
 arch/arm64/include/asm/efi.h |  3 +++
 arch/x86/include/asm/efi.h   | 16 +++++++++++-----
 3 files changed, 17 insertions(+), 5 deletions(-)

diff --git a/arch/arm/include/asm/efi.h b/arch/arm/include/asm/efi.h
index 766bf9b..0b06f53 100644
--- a/arch/arm/include/asm/efi.h
+++ b/arch/arm/include/asm/efi.h
@@ -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);
 
diff --git a/arch/arm64/include/asm/efi.h b/arch/arm64/include/asm/efi.h
index a9e54aa..771b3f0 100644
--- a/arch/arm64/include/asm/efi.h
+++ b/arch/arm64/include/asm/efi.h
@@ -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...)
 
diff --git a/arch/x86/include/asm/efi.h b/arch/x86/include/asm/efi.h
index 389d700..e99675b 100644
--- a/arch/x86/include/asm/efi.h
+++ b/arch/x86/include/asm/efi.h
@@ -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__);

Reply via email to