[U-Boot] [PATCH 1/3] efi_loader: Add an EFI_CALL() macro

Rather than open-coding EFI_EXIT() + callback + EFI_ENTRY(), introduce an EFI_CALL() macro. This makes callbacks into UEFI world (of which there will be more in the future) more concise and easier to locate in the code.
Signed-off-by: Rob Clark robdclark@gmail.com --- include/efi_loader.h | 17 +++++++++++++++++ lib/efi_loader/efi_boottime.c | 4 +--- 2 files changed, 18 insertions(+), 3 deletions(-)
diff --git a/include/efi_loader.h b/include/efi_loader.h index f384cbbe77..09bab7dbc6 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -15,16 +15,33 @@
#include <linux/list.h>
+/* + * Enter the u-boot world from UEFI: + */ #define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \ debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__); \ } while(0)
+/* + * Exit the u-boot world back to UEFI: + */ #define EFI_EXIT(ret) ({ \ debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) & ~EFI_ERROR_MASK)); \ efi_exit_func(ret); \ })
+/* + * Callback into UEFI world from u-boot: + */ +#define EFI_CALL(exp) do { \ + debug("EFI: Call: %s\n", #exp); \ + efi_exit_func(EFI_SUCCESS); \ + exp; \ + efi_restore_gd(); \ + debug("EFI: Return From: %s\n", #exp); \ + } while(0) + extern struct efi_runtime_services efi_runtime_services; extern struct efi_system_table systab;
diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index 9a1a93fade..76cafffc1d 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -120,9 +120,7 @@ void efi_signal_event(struct efi_event *event) return; event->signaled = 1; if (event->type & EVT_NOTIFY_SIGNAL) { - EFI_EXIT(EFI_SUCCESS); - event->notify_function(event, event->notify_context); - EFI_ENTRY("returning from notification function"); + EFI_CALL(event->notify_function(event, event->notify_context)); } }

Missing an EFI_ENTRY() or doubling up EFI_EXIT() leads to non-obvious crashes. Let's add some error checking.
Signed-off-by: Rob Clark robdclark@gmail.com --- include/efi_loader.h | 5 +++++ lib/efi_loader/efi_boottime.c | 13 +++++++++++++ 2 files changed, 18 insertions(+)
diff --git a/include/efi_loader.h b/include/efi_loader.h index 09bab7dbc6..4b49fac84b 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -15,11 +15,13 @@
#include <linux/list.h>
+int __efi_check_nesting(int delta); /* * Enter the u-boot world from UEFI: */ #define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \ + assert(__efi_check_nesting(+1)); \ debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__); \ } while(0)
@@ -28,6 +30,7 @@ */ #define EFI_EXIT(ret) ({ \ debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) & ~EFI_ERROR_MASK)); \ + assert(__efi_check_nesting(-1)); \ efi_exit_func(ret); \ })
@@ -36,10 +39,12 @@ */ #define EFI_CALL(exp) do { \ debug("EFI: Call: %s\n", #exp); \ + assert(__efi_check_nesting(-1)); \ efi_exit_func(EFI_SUCCESS); \ exp; \ efi_restore_gd(); \ debug("EFI: Return From: %s\n", #exp); \ + assert(__efi_check_nesting(+1)); \ } while(0)
extern struct efi_runtime_services efi_runtime_services; diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index 76cafffc1d..b21df7bd5d 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -57,6 +57,17 @@ void efi_save_gd(void) #endif }
+/* Check for incorrect use of EFI_ENTRY() / EFI_EXIT() */ +int __efi_check_nesting(int delta) +{ + static int entry_count; + /* post-increment, pre-decrement: */ + if (delta > 0) + return entry_count++ == 0; + else + return --entry_count == 0; +} + /* Called on every callback entry */ void efi_restore_gd(void) { @@ -716,7 +727,9 @@ static efi_status_t EFIAPI efi_start_image(efi_handle_t image_handle, return EFI_EXIT(info->exit_status); }
+ __efi_check_nesting(-1); entry(image_handle, &systab); + __efi_check_nesting(+1);
/* Should usually never get here */ return EFI_EXIT(EFI_SUCCESS);

On 26.07.17 15:55, Rob Clark wrote:
Missing an EFI_ENTRY() or doubling up EFI_EXIT() leads to non-obvious crashes. Let's add some error checking.
Signed-off-by: Rob Clark robdclark@gmail.com
include/efi_loader.h | 5 +++++ lib/efi_loader/efi_boottime.c | 13 +++++++++++++ 2 files changed, 18 insertions(+)
diff --git a/include/efi_loader.h b/include/efi_loader.h index 09bab7dbc6..4b49fac84b 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -15,11 +15,13 @@
#include <linux/list.h>
+int __efi_check_nesting(int delta); /*
- Enter the u-boot world from UEFI:
*/ #define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \
- assert(__efi_check_nesting(+1)); \ debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__); \ } while(0)
@@ -28,6 +30,7 @@ */ #define EFI_EXIT(ret) ({ \ debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) & ~EFI_ERROR_MASK)); \
- assert(__efi_check_nesting(-1)); \ efi_exit_func(ret); \ })
@@ -36,10 +39,12 @@ */ #define EFI_CALL(exp) do { \ debug("EFI: Call: %s\n", #exp); \
assert(__efi_check_nesting(-1)); \ efi_exit_func(EFI_SUCCESS); \ exp; \ efi_restore_gd(); \ debug("EFI: Return From: %s\n", #exp); \
assert(__efi_check_nesting(+1)); \ } while(0)
extern struct efi_runtime_services efi_runtime_services;
diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index 76cafffc1d..b21df7bd5d 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -57,6 +57,17 @@ void efi_save_gd(void) #endif }
+/* Check for incorrect use of EFI_ENTRY() / EFI_EXIT() */ +int __efi_check_nesting(int delta) +{
- static int entry_count;
I'd prefer if we marked globals as such (read: somewhere at the top of a .c file).
Also I think this function would be better off as a static inline in a header, as it should get compressed quite well.
- /* post-increment, pre-decrement: */
- if (delta > 0)
return entry_count++ == 0;
- else
return --entry_count == 0;
I have a hard time to wrap my head around the logic. At least, couldn't you split this into 2 functions? :)
Alex
+}
- /* Called on every callback entry */ void efi_restore_gd(void) {
@@ -716,7 +727,9 @@ static efi_status_t EFIAPI efi_start_image(efi_handle_t image_handle, return EFI_EXIT(info->exit_status); }
__efi_check_nesting(-1); entry(image_handle, &systab);
__efi_check_nesting(+1);
/* Should usually never get here */ return EFI_EXIT(EFI_SUCCESS);

On Wed, Jul 26, 2017 at 11:25 AM, Alexander Graf agraf@suse.de wrote:
On 26.07.17 15:55, Rob Clark wrote:
Missing an EFI_ENTRY() or doubling up EFI_EXIT() leads to non-obvious crashes. Let's add some error checking.
Signed-off-by: Rob Clark robdclark@gmail.com
include/efi_loader.h | 5 +++++ lib/efi_loader/efi_boottime.c | 13 +++++++++++++ 2 files changed, 18 insertions(+)
diff --git a/include/efi_loader.h b/include/efi_loader.h index 09bab7dbc6..4b49fac84b 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -15,11 +15,13 @@ #include <linux/list.h> +int __efi_check_nesting(int delta); /*
- Enter the u-boot world from UEFI:
*/ #define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \
@@ -28,6 +30,7 @@ */ #define EFI_EXIT(ret) ({ \ debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) &assert(__efi_check_nesting(+1)); \ debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__); \ } while(0)
~EFI_ERROR_MASK)); \
@@ -36,10 +39,12 @@ */ #define EFI_CALL(exp) do { \ debug("EFI: Call: %s\n", #exp); \assert(__efi_check_nesting(-1)); \ efi_exit_func(ret); \ })
assert(__efi_check_nesting(-1)); \ efi_exit_func(EFI_SUCCESS); \ exp; \ efi_restore_gd(); \ debug("EFI: Return From: %s\n", #exp); \
extern struct efi_runtime_services efi_runtime_services;assert(__efi_check_nesting(+1)); \ } while(0)
diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index 76cafffc1d..b21df7bd5d 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -57,6 +57,17 @@ void efi_save_gd(void) #endif } +/* Check for incorrect use of EFI_ENTRY() / EFI_EXIT() */ +int __efi_check_nesting(int delta) +{
static int entry_count;
I'd prefer if we marked globals as such (read: somewhere at the top of a .c file).
hmm, well that does increase the scope unnecessarily, which is why I made it static inside the fxn. But if you can prefer I guess I can put it just above __efi_check_nesting().
Also I think this function would be better off as a static inline in a header, as it should get compressed quite well.
That would mean making entry_count also non-static (ie. broader than function or file level scope), otherwise you end up with different copies of it in each .o file. (Which would mostly work, but it would fall apart in confusing ways in some edge cases..)
/* post-increment, pre-decrement: */
if (delta > 0)
return entry_count++ == 0;
else
return --entry_count == 0;
I have a hard time to wrap my head around the logic. At least, couldn't you split this into 2 functions? :)
Would that make the logic more clear, or just move it far enough apart that you don't notice it is confusing ;-)
BR, -R
Alex
+}
- /* Called on every callback entry */ void efi_restore_gd(void) {
@@ -716,7 +727,9 @@ static efi_status_t EFIAPI efi_start_image(efi_handle_t image_handle, return EFI_EXIT(info->exit_status); }
__efi_check_nesting(-1); entry(image_handle, &systab);
__efi_check_nesting(+1); /* Should usually never get here */ return EFI_EXIT(EFI_SUCCESS);

On 26.07.17 18:41, Rob Clark wrote:
On Wed, Jul 26, 2017 at 11:25 AM, Alexander Graf agraf@suse.de wrote:
On 26.07.17 15:55, Rob Clark wrote:
Missing an EFI_ENTRY() or doubling up EFI_EXIT() leads to non-obvious crashes. Let's add some error checking.
Signed-off-by: Rob Clark robdclark@gmail.com
include/efi_loader.h | 5 +++++ lib/efi_loader/efi_boottime.c | 13 +++++++++++++ 2 files changed, 18 insertions(+)
diff --git a/include/efi_loader.h b/include/efi_loader.h index 09bab7dbc6..4b49fac84b 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -15,11 +15,13 @@ #include <linux/list.h> +int __efi_check_nesting(int delta); /* * Enter the u-boot world from UEFI: */ #define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \
@@ -28,6 +30,7 @@ */ #define EFI_EXIT(ret) ({ \ debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) &assert(__efi_check_nesting(+1)); \ debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__); \ } while(0)
~EFI_ERROR_MASK)); \
@@ -36,10 +39,12 @@ */ #define EFI_CALL(exp) do { \ debug("EFI: Call: %s\n", #exp); \assert(__efi_check_nesting(-1)); \ efi_exit_func(ret); \ })
assert(__efi_check_nesting(-1)); \ efi_exit_func(EFI_SUCCESS); \ exp; \ efi_restore_gd(); \ debug("EFI: Return From: %s\n", #exp); \
extern struct efi_runtime_services efi_runtime_services;assert(__efi_check_nesting(+1)); \ } while(0)
diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index 76cafffc1d..b21df7bd5d 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -57,6 +57,17 @@ void efi_save_gd(void) #endif } +/* Check for incorrect use of EFI_ENTRY() / EFI_EXIT() */ +int __efi_check_nesting(int delta) +{
static int entry_count;
I'd prefer if we marked globals as such (read: somewhere at the top of a .c file).
hmm, well that does increase the scope unnecessarily, which is why I made it static inside the fxn. But if you can prefer I guess I can put it just above __efi_check_nesting().
It's a matter of taste, but in general I find it very useful to know at a glimpse where .bss, .rodata and .data come from.
Also I think this function would be better off as a static inline in a header, as it should get compressed quite well.
That would mean making entry_count also non-static (ie. broader than
Ah, no, entry_count would still be in efi_boottime.c, but the function using it would be in a .h file. That way we don't need to do the register save/restore dance we need to do for full remote function calls.
function or file level scope), otherwise you end up with different copies of it in each .o file. (Which would mostly work, but it would fall apart in confusing ways in some edge cases..)
/* post-increment, pre-decrement: */
if (delta > 0)
return entry_count++ == 0;
else
return --entry_count == 0;
I have a hard time to wrap my head around the logic. At least, couldn't you split this into 2 functions? :)
Would that make the logic more clear, or just move it far enough apart that you don't notice it is confusing ;-)
I can't really tell yet, but I'd say it's worth a try :).
Alex

On Wed, Jul 26, 2017 at 4:12 PM, Alexander Graf agraf@suse.de wrote:
On 26.07.17 18:41, Rob Clark wrote:
On Wed, Jul 26, 2017 at 11:25 AM, Alexander Graf agraf@suse.de wrote:
On 26.07.17 15:55, Rob Clark wrote:
Missing an EFI_ENTRY() or doubling up EFI_EXIT() leads to non-obvious crashes. Let's add some error checking.
Signed-off-by: Rob Clark robdclark@gmail.com
include/efi_loader.h | 5 +++++ lib/efi_loader/efi_boottime.c | 13 +++++++++++++ 2 files changed, 18 insertions(+)
diff --git a/include/efi_loader.h b/include/efi_loader.h index 09bab7dbc6..4b49fac84b 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -15,11 +15,13 @@ #include <linux/list.h> +int __efi_check_nesting(int delta); /* * Enter the u-boot world from UEFI: */ #define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \
assert(__efi_check_nesting(+1)); \ debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__);
\ } while(0) @@ -28,6 +30,7 @@ */ #define EFI_EXIT(ret) ({ \ debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) & ~EFI_ERROR_MASK)); \
@@ -36,10 +39,12 @@ */ #define EFI_CALL(exp) do { \ debug("EFI: Call: %s\n", #exp); \assert(__efi_check_nesting(-1)); \ efi_exit_func(ret); \ })
assert(__efi_check_nesting(-1)); \ efi_exit_func(EFI_SUCCESS); \ exp; \ efi_restore_gd(); \ debug("EFI: Return From: %s\n", #exp); \
extern struct efi_runtime_services efi_runtime_services;assert(__efi_check_nesting(+1)); \ } while(0)
diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index 76cafffc1d..b21df7bd5d 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -57,6 +57,17 @@ void efi_save_gd(void) #endif } +/* Check for incorrect use of EFI_ENTRY() / EFI_EXIT() */ +int __efi_check_nesting(int delta) +{
static int entry_count;
I'd prefer if we marked globals as such (read: somewhere at the top of a .c file).
hmm, well that does increase the scope unnecessarily, which is why I made it static inside the fxn. But if you can prefer I guess I can put it just above __efi_check_nesting().
It's a matter of taste, but in general I find it very useful to know at a glimpse where .bss, .rodata and .data come from.
Also I think this function would be better off as a static inline in a header, as it should get compressed quite well.
That would mean making entry_count also non-static (ie. broader than
Ah, no, entry_count would still be in efi_boottime.c, but the function using it would be in a .h file. That way we don't need to do the register save/restore dance we need to do for full remote function calls.
Ok.. I'm thinking about just moving these calls into efi_restore_gd()/efi_exit_func() but leaving the asserts in the macro (so the assert reflects the proper file/line). Minor downside is the assert would come before the debug() print, but since it has file/line I guess that isn't really important.
Fortunately the one place we need to special-case call the check_nesting() (ie. efi_start_image()) is in the same object file.
I'll give that a go this evening and see how it turns out.
BR, -R

On 07/26/2017 03:55 PM, Rob Clark wrote:
Missing an EFI_ENTRY() or doubling up EFI_EXIT() leads to non-obvious crashes. Let's add some error checking.
Signed-off-by: Rob Clark robdclark@gmail.com
include/efi_loader.h | 5 +++++ lib/efi_loader/efi_boottime.c | 13 +++++++++++++ 2 files changed, 18 insertions(+)
diff --git a/include/efi_loader.h b/include/efi_loader.h index 09bab7dbc6..4b49fac84b 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -15,11 +15,13 @@
#include <linux/list.h>
+int __efi_check_nesting(int delta); /*
- Enter the u-boot world from UEFI:
*/ #define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \
- assert(__efi_check_nesting(+1)); \ debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__); \ } while(0)
@@ -28,6 +30,7 @@ */ #define EFI_EXIT(ret) ({ \ debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) & ~EFI_ERROR_MASK)); \
- assert(__efi_check_nesting(-1)); \ efi_exit_func(ret); \ })
@@ -36,10 +39,12 @@ */ #define EFI_CALL(exp) do { \ debug("EFI: Call: %s\n", #exp); \
- assert(__efi_check_nesting(-1)); \ efi_exit_func(EFI_SUCCESS); \ exp; \ efi_restore_gd(); \ debug("EFI: Return From: %s\n", #exp); \
- assert(__efi_check_nesting(+1)); \ } while(0)
extern struct efi_runtime_services efi_runtime_services; diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index 76cafffc1d..b21df7bd5d 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -57,6 +57,17 @@ void efi_save_gd(void) #endif }
+/* Check for incorrect use of EFI_ENTRY() / EFI_EXIT() */ +int __efi_check_nesting(int delta) +{
- static int entry_count;
- /* post-increment, pre-decrement: */
- if (delta > 0)
return entry_count++ == 0;
- else
return --entry_count == 0;
+}
/* Called on every callback entry */ void efi_restore_gd(void) { @@ -716,7 +727,9 @@ static efi_status_t EFIAPI efi_start_image(efi_handle_t image_handle, return EFI_EXIT(info->exit_status); }
__efi_check_nesting(-1); entry(image_handle, &systab);
__efi_check_nesting(+1);
/* Should usually never get here */ return EFI_EXIT(EFI_SUCCESS);
Tested-by: Heinrich Schuchardt xypron.glpk@gmx.de

This should make it easier to see when a callback back to UEFI world calls back in to the u-boot world, and generally match up EFI_ENTRY() and EFI_EXIT() calls.
Signed-off-by: Rob Clark robdclark@gmail.com --- include/efi_loader.h | 12 ++++++++---- lib/efi_loader/efi_boottime.c | 28 ++++++++++++++++++++++++++++ 2 files changed, 36 insertions(+), 4 deletions(-)
diff --git a/include/efi_loader.h b/include/efi_loader.h index 4b49fac84b..88091d4914 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -16,20 +16,24 @@ #include <linux/list.h>
int __efi_check_nesting(int delta); +const char *__efi_nesting_level(int delta); + /* * Enter the u-boot world from UEFI: */ #define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \ + debug("%sEFI: Entry %s(" format ")\n", __efi_nesting_level(1), \ + __func__, ##__VA_ARGS__); \ assert(__efi_check_nesting(+1)); \ - debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__); \ } while(0)
/* * Exit the u-boot world back to UEFI: */ #define EFI_EXIT(ret) ({ \ - debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) & ~EFI_ERROR_MASK)); \ + debug("%sEFI: Exit: %s: %u\n", __efi_nesting_level(-1), \ + __func__, (u32)((ret) & ~EFI_ERROR_MASK)); \ assert(__efi_check_nesting(-1)); \ efi_exit_func(ret); \ }) @@ -38,12 +42,12 @@ int __efi_check_nesting(int delta); * Callback into UEFI world from u-boot: */ #define EFI_CALL(exp) do { \ - debug("EFI: Call: %s\n", #exp); \ + debug("%sEFI: Call: %s\n", __efi_nesting_level(1), #exp); \ assert(__efi_check_nesting(-1)); \ efi_exit_func(EFI_SUCCESS); \ exp; \ efi_restore_gd(); \ - debug("EFI: Return From: %s\n", #exp); \ + debug("%sEFI: Return From: %s\n", __efi_nesting_level(-1), #exp); \ assert(__efi_check_nesting(+1)); \ } while(0)
diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index b21df7bd5d..f6c4c162cf 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -92,6 +92,34 @@ efi_status_t efi_exit_func(efi_status_t ret) return ret; }
+/* + * Two spaces per indent level, maxing out at 10.. which ought to be + * enough for anyone ;-) + */ +static const char *indent_string(int level) +{ + static const char *indent = " "; + const int max = strlen(indent); + level = min(max, level * 2); + return &indent[max - level]; +} + +const char *__efi_nesting_level(int delta) +{ + static int nesting_level; + const char *s; + /* post-increment, pre-decrement */ + if (delta > 0) { + s = indent_string(nesting_level); + nesting_level += delta; + } else { + nesting_level += delta; + s = indent_string(nesting_level); + assert(nesting_level >= 0); + } + return s; +} + /* Low 32 bit */ #define EFI_LOW32(a) (a & 0xFFFFFFFFULL) /* High 32 bit */

On 26.07.17 15:55, Rob Clark wrote:
This should make it easier to see when a callback back to UEFI world calls back in to the u-boot world, and generally match up EFI_ENTRY() and EFI_EXIT() calls.
Signed-off-by: Rob Clark robdclark@gmail.com
include/efi_loader.h | 12 ++++++++---- lib/efi_loader/efi_boottime.c | 28 ++++++++++++++++++++++++++++ 2 files changed, 36 insertions(+), 4 deletions(-)
diff --git a/include/efi_loader.h b/include/efi_loader.h index 4b49fac84b..88091d4914 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -16,20 +16,24 @@ #include <linux/list.h>
int __efi_check_nesting(int delta); +const char *__efi_nesting_level(int delta);
- /*
*/ #define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \
- Enter the u-boot world from UEFI:
- debug("%sEFI: Entry %s(" format ")\n", __efi_nesting_level(1), \
assert(__efi_check_nesting(+1)); \__func__, ##__VA_ARGS__); \
debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__); \ } while(0)
/*
- Exit the u-boot world back to UEFI:
*/ #define EFI_EXIT(ret) ({ \
debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) & ~EFI_ERROR_MASK)); \
- debug("%sEFI: Exit: %s: %u\n", __efi_nesting_level(-1), \
assert(__efi_check_nesting(-1)); \ efi_exit_func(ret); \ })__func__, (u32)((ret) & ~EFI_ERROR_MASK)); \
@@ -38,12 +42,12 @@ int __efi_check_nesting(int delta);
- Callback into UEFI world from u-boot:
*/ #define EFI_CALL(exp) do { \
- debug("EFI: Call: %s\n", #exp); \
- debug("%sEFI: Call: %s\n", __efi_nesting_level(1), #exp); \ assert(__efi_check_nesting(-1)); \ efi_exit_func(EFI_SUCCESS); \ exp; \ efi_restore_gd(); \
- debug("EFI: Return From: %s\n", #exp); \
- debug("%sEFI: Return From: %s\n", __efi_nesting_level(-1), #exp); \ assert(__efi_check_nesting(+1)); \ } while(0)
diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index b21df7bd5d..f6c4c162cf 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -92,6 +92,34 @@ efi_status_t efi_exit_func(efi_status_t ret) return ret; }
+/*
- Two spaces per indent level, maxing out at 10.. which ought to be
- enough for anyone ;-)
- */
+static const char *indent_string(int level) +{
- static const char *indent = " ";
- const int max = strlen(indent);
- level = min(max, level * 2);
- return &indent[max - level];
+}
+const char *__efi_nesting_level(int delta) +{
- static int nesting_level;
- const char *s;
- /* post-increment, pre-decrement */
- if (delta > 0) {
s = indent_string(nesting_level);
nesting_level += delta;
- } else {
nesting_level += delta;
s = indent_string(nesting_level);
assert(nesting_level >= 0);
- }
Can you do the same here? Move globals to the top of the file and split the function?
In this particular case I think leaving them as functions is reasonable, as we should only call them when #define DEBUG is set, so we're not size constrained.
Alex

On 07/26/2017 03:55 PM, Rob Clark wrote:
This should make it easier to see when a callback back to UEFI world calls back in to the u-boot world, and generally match up EFI_ENTRY() and EFI_EXIT() calls.
Signed-off-by: Rob Clark robdclark@gmail.com
include/efi_loader.h | 12 ++++++++---- lib/efi_loader/efi_boottime.c | 28 ++++++++++++++++++++++++++++ 2 files changed, 36 insertions(+), 4 deletions(-)
diff --git a/include/efi_loader.h b/include/efi_loader.h index 4b49fac84b..88091d4914 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -16,20 +16,24 @@ #include <linux/list.h>
int __efi_check_nesting(int delta); +const char *__efi_nesting_level(int delta);
/*
- Enter the u-boot world from UEFI:
*/ #define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \
- debug("%sEFI: Entry %s(" format ")\n", __efi_nesting_level(1), \
assert(__efi_check_nesting(+1)); \__func__, ##__VA_ARGS__); \
- debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__); \ } while(0)
/*
- Exit the u-boot world back to UEFI:
*/ #define EFI_EXIT(ret) ({ \
- debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) & ~EFI_ERROR_MASK)); \
- debug("%sEFI: Exit: %s: %u\n", __efi_nesting_level(-1), \
assert(__efi_check_nesting(-1)); \ efi_exit_func(ret); \ })__func__, (u32)((ret) & ~EFI_ERROR_MASK)); \
@@ -38,12 +42,12 @@ int __efi_check_nesting(int delta);
- Callback into UEFI world from u-boot:
*/ #define EFI_CALL(exp) do { \
- debug("EFI: Call: %s\n", #exp); \
- debug("%sEFI: Call: %s\n", __efi_nesting_level(1), #exp); \ assert(__efi_check_nesting(-1)); \ efi_exit_func(EFI_SUCCESS); \ exp; \ efi_restore_gd(); \
- debug("EFI: Return From: %s\n", #exp); \
- debug("%sEFI: Return From: %s\n", __efi_nesting_level(-1), #exp); \ assert(__efi_check_nesting(+1)); \ } while(0)
diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index b21df7bd5d..f6c4c162cf 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -92,6 +92,34 @@ efi_status_t efi_exit_func(efi_status_t ret) return ret; }
+/*
- Two spaces per indent level, maxing out at 10.. which ought to be
- enough for anyone ;-)
- */
+static const char *indent_string(int level) +{
- static const char *indent = " ";
- const int max = strlen(indent);
- level = min(max, level * 2);
- return &indent[max - level];
+}
+const char *__efi_nesting_level(int delta) +{
- static int nesting_level;
- const char *s;
- /* post-increment, pre-decrement */
- if (delta > 0) {
s = indent_string(nesting_level);
nesting_level += delta;
- } else {
nesting_level += delta;
s = indent_string(nesting_level);
assert(nesting_level >= 0);
- }
- return s;
+}
/* Low 32 bit */ #define EFI_LOW32(a) (a & 0xFFFFFFFFULL) /* High 32 bit */
Tested-by: Heinrich Schuchardt xypron.glpk@gmx.de

On 07/26/2017 03:55 PM, Rob Clark wrote:
Rather than open-coding EFI_EXIT() + callback + EFI_ENTRY(), introduce an EFI_CALL() macro. This makes callbacks into UEFI world (of which there will be more in the future) more concise and easier to locate in the code.
Signed-off-by: Rob Clark robdclark@gmail.com
include/efi_loader.h | 17 +++++++++++++++++ lib/efi_loader/efi_boottime.c | 4 +--- 2 files changed, 18 insertions(+), 3 deletions(-)
diff --git a/include/efi_loader.h b/include/efi_loader.h index f384cbbe77..09bab7dbc6 100644 --- a/include/efi_loader.h +++ b/include/efi_loader.h @@ -15,16 +15,33 @@
#include <linux/list.h>
+/*
- Enter the u-boot world from UEFI:
- */
#define EFI_ENTRY(format, ...) do { \ efi_restore_gd(); \ debug("EFI: Entry %s(" format ")\n", __func__, ##__VA_ARGS__); \ } while(0)
+/*
- Exit the u-boot world back to UEFI:
- */
#define EFI_EXIT(ret) ({ \ debug("EFI: Exit: %s: %u\n", __func__, (u32)((ret) & ~EFI_ERROR_MASK)); \ efi_exit_func(ret); \ })
+/*
- Callback into UEFI world from u-boot:
- */
+#define EFI_CALL(exp) do { \
- debug("EFI: Call: %s\n", #exp); \
- efi_exit_func(EFI_SUCCESS); \
- exp; \
- efi_restore_gd(); \
- debug("EFI: Return From: %s\n", #exp); \
- } while(0)
extern struct efi_runtime_services efi_runtime_services; extern struct efi_system_table systab;
diff --git a/lib/efi_loader/efi_boottime.c b/lib/efi_loader/efi_boottime.c index 9a1a93fade..76cafffc1d 100644 --- a/lib/efi_loader/efi_boottime.c +++ b/lib/efi_loader/efi_boottime.c @@ -120,9 +120,7 @@ void efi_signal_event(struct efi_event *event) return; event->signaled = 1; if (event->type & EVT_NOTIFY_SIGNAL) {
EFI_EXIT(EFI_SUCCESS);
event->notify_function(event, event->notify_context);
EFI_ENTRY("returning from notification function");
}EFI_CALL(event->notify_function(event, event->notify_context));
}
Tested-by: Heinrich Schuchardt xypron.glpk@gmx.de
participants (3)
-
Alexander Graf
-
Heinrich Schuchardt
-
Rob Clark