On 10/17/2017 09:48 AM, Alexander Graf wrote:
> 
> 
> On 13.10.17 19:33, Heinrich Schuchardt wrote:
>> Environment variable efi_selftest is passed as load options
>> to the selftest application. It is used to select a single
>> test to be executed.
>>
>> Special value 'list' displays a list of all available tests.
>>
>> Tests get an on_request property. If this property is set
>> the tests are only executed if explicitly requested.
>>
>> The invocation of efi_selftest is changed to reflect that
>> bootefi selftest with efi_selftest = 'list' will call the
>> Exit bootservice.
>>
>> Environment variable bootargs is used as load options
>> for all other bootefi payloads.
>>
>> Signed-off-by: Heinrich Schuchardt <xypron.g...@gmx.de>
>> ---
>> v2
>>      use an environment variable to choose a test
>> ---
>>  cmd/bootefi.c                           | 46 ++++++++++++++++-
>>  include/efi_selftest.h                  | 18 +++++++
>>  lib/efi_selftest/efi_selftest.c         | 90 
>> +++++++++++++++++++++++++++++++--
>>  lib/efi_selftest/efi_selftest_console.c | 10 ++++
>>  lib/efi_selftest/efi_selftest_util.c    | 11 +++-
>>  5 files changed, 168 insertions(+), 7 deletions(-)
>>
>> diff --git a/cmd/bootefi.c b/cmd/bootefi.c
>> index 18176a1266..2d70137482 100644
>> --- a/cmd/bootefi.c
>> +++ b/cmd/bootefi.c
>> @@ -6,10 +6,12 @@
>>   *  SPDX-License-Identifier:     GPL-2.0+
>>   */
>>  
>> +#include <charset.h>
>>  #include <common.h>
>>  #include <command.h>
>>  #include <dm.h>
>>  #include <efi_loader.h>
>> +#include <efi_selftest.h>
>>  #include <errno.h>
>>  #include <libfdt.h>
>>  #include <libfdt_env.h>
>> @@ -50,6 +52,32 @@ static void efi_init_obj_list(void)
>>      efi_get_time_init();
>>  }
>>  
>> +/*
>> + * Set the load options of an image from an environment variable.
>> + *
>> + * @loaded_image_info:      the image
>> + * @env_var:                name of the environment variable
>> + */
>> +static void set_load_options(struct efi_loaded_image *loaded_image_info,
>> +                         const char *env_var)
>> +{
>> +    size_t size;
>> +    const char *env = env_get(env_var);
>> +
>> +    loaded_image_info->load_options = NULL;
>> +    loaded_image_info->load_options_size = 0;
>> +    if (!env)
>> +            return;
>> +    size = strlen(env) + 1;
>> +    loaded_image_info->load_options = calloc(size, sizeof(u16));
>> +    if (!loaded_image_info->load_options) {
>> +            printf("ERROR: Out of memory\n");
>> +            return;
>> +    }
>> +    utf8_to_utf16(loaded_image_info->load_options, (u8 *)env, size);
>> +    loaded_image_info->load_options_size = size * 2;
>> +}
>> +
>>  static void *copy_fdt(void *fdt)
>>  {
>>      u64 fdt_size = fdt_totalsize(fdt);
>> @@ -190,6 +218,8 @@ static unsigned long do_bootefi_exec(void *efi, void 
>> *fdt,
>>              efi_install_configuration_table(&fdt_guid, NULL);
>>      }
>>  
>> +    /* Transfer environment variable bootargs as load options */
>> +    set_load_options(&loaded_image_info, "bootargs");
> 
> While I really want to see that change, please try not to sneak it in
> with the selftest one :).
> 
> Just split that hunk out to a following patch and give it its own patch
> description. In case something goes wrong, we'd only need to revert a
> small patch then.
> 
>>      /* Load the EFI payload */
>>      entry = efi_load_pe(efi, &loaded_image_info);
>>      if (!entry) {
>> @@ -237,6 +267,7 @@ static unsigned long do_bootefi_exec(void *efi, void 
>> *fdt,
>>  
>>  exit:
>>      /* image has returned, loaded-image obj goes *poof*: */
>> +    free(loaded_image_info.load_options);
> 
> This too is a change that doesn't fit the patch description?
> 
>>      list_del(&loaded_image_info_obj.link);
>>  
>>      return ret;
>> @@ -301,17 +332,26 @@ static int do_bootefi(cmd_tbl_t *cmdtp, int flag, int 
>> argc, char * const argv[])
>>  
>>              efi_setup_loaded_image(&loaded_image_info,
>>                                     &loaded_image_info_obj,
>> -                                   bootefi_device_path, bootefi_image_path);
>> +                                   NULL, NULL);
> 
> Why?
> 
>>              /*
>>               * gd lives in a fixed register which may get clobbered while we
>>               * execute the payload. So save it here and restore it on every
>>               * callback entry
>>               */
>>              efi_save_gd();
>> +            loaded_image_info.image_code_type = EFI_LOADER_CODE;
>> +            loaded_image_info.image_data_type = EFI_LOADER_DATA;
> 
> Also unrelated? Please split it out.
> 
>>              /* Initialize and populate EFI object list */
>>              if (!efi_obj_list_initalized)
>>                      efi_init_obj_list();
>> -            return efi_selftest(&loaded_image_info, &systab);
>> +            /* Transfer environment variable efi_selftest as load options */
>> +            set_load_options(&loaded_image_info, "efi_selftest");
>> +            /* Execute the test */
>> +            r = efi_selftest(&loaded_image_info, &systab);
>> +            efi_restore_gd();
>> +            free(loaded_image_info.load_options);
>> +            list_del(&loaded_image_info_obj.link);
> 
> That change too is unrelated to the patch description. Please split out.
> 
>> +            return r;
>>      } else
>>  #endif
>>      if (!strcmp(argv[1], "bootmgr")) {
>> @@ -357,6 +397,8 @@ static char bootefi_help_text[] =
>>  #ifdef CONFIG_CMD_BOOTEFI_SELFTEST
>>      "bootefi selftest\n"
>>      "  - boot an EFI selftest application stored within U-Boot\n"
>> +    "    Use environment variable efi_selftest to select a single test.\n"
>> +    "    Use 'setenv efi_selftest list' to enumerate all tests.\n"
>>  #endif
>>      "bootmgr [fdt addr]\n"
>>      "  - load and boot EFI payload based on BootOrder/BootXXXX variables.\n"
>> diff --git a/include/efi_selftest.h b/include/efi_selftest.h
>> index 7ec42a0406..978ca2a7ea 100644
>> --- a/include/efi_selftest.h
>> +++ b/include/efi_selftest.h
>> @@ -12,6 +12,7 @@
>>  #include <common.h>
>>  #include <efi.h>
>>  #include <efi_api.h>
>> +#include <efi_loader.h>
>>  #include <linker_lists.h>
>>  
>>  #define EFI_ST_SUCCESS 0
>> @@ -71,6 +72,15 @@ void efi_st_printf(const char *fmt, ...)
>>   */
>>  int efi_st_memcmp(const void *buf1, const void *buf2, size_t length);
>>  
>> +/*
>> + * Compare an u16 string to a char string.
>> + *
>> + * @buf1:   u16 string
>> + * @buf2:   char string
>> + * @return: 0 if both buffers contain the same bytes
>> + */
>> +int efi_st_strcmp_16_8(const u16 *buf1, const char *buf2);
>> +
>>  /*
>>   * Reads an Unicode character from the input device.
>>   *
>> @@ -88,6 +98,7 @@ u16 efi_st_get_key(void);
>>   * @setup:  set up the unit test
>>   * @teardown:       tear down the unit test
>>   * @execute:        execute the unit test
>> + * @on_request:     test is only executed on request
>>   */
>>  struct efi_unit_test {
>>      const char *name;
>> @@ -96,10 +107,17 @@ struct efi_unit_test {
>>                   const struct efi_system_table *systable);
>>      int (*execute)(void);
>>      int (*teardown)(void);
>> +    bool on_request;
>>  };
>>  
>>  /* Declare a new EFI unit test */
>>  #define EFI_UNIT_TEST(__name)                                               
>> \
>>      ll_entry_declare(struct efi_unit_test, __name, efi_unit_test)
>>  
>> +#define EFI_SELFTEST_TABLE_GUID \
>> +    EFI_GUID(0xbc3ebe57, 0x09e5, 0xa59d, 0xdb, 0x87, \
>> +             0xf5, 0x79, 0x61, 0x62, 0x06, 0xde)
>> +
>> +extern const efi_guid_t efi_selftest_table_guid;
>> +
>>  #endif /* _EFI_SELFTEST_H */
>> diff --git a/lib/efi_selftest/efi_selftest.c 
>> b/lib/efi_selftest/efi_selftest.c
>> index 45d8d3d384..110284f9c7 100644
>> --- a/lib/efi_selftest/efi_selftest.c
>> +++ b/lib/efi_selftest/efi_selftest.c
>> @@ -15,6 +15,8 @@ static const struct efi_runtime_services *runtime;
>>  static efi_handle_t handle;
>>  static u16 reset_message[] = L"Selftest completed";
>>  
>> +const efi_guid_t efi_selftest_table_guid = EFI_SELFTEST_TABLE_GUID;
>> +
>>  /*
>>   * Exit the boot services.
>>   *
>> @@ -25,8 +27,8 @@ static u16 reset_message[] = L"Selftest completed";
>>   */
>>  void efi_st_exit_boot_services(void)
>>  {
>> -    unsigned long  map_size = 0;
>> -    unsigned long  map_key;
>> +    unsigned long map_size = 0;
>> +    unsigned long map_key;
> 
> Unrelated?
> 
>>      unsigned long desc_size;
>>      u32 desc_version;
>>      efi_status_t ret;
>> @@ -133,6 +135,41 @@ static int teardown(struct efi_unit_test *test, 
>> unsigned int *failures)
>>      return ret;
>>  }
>>  
>> +/*
>> + * Check that a test exists.
>> + *
>> + * @testname:       name of the test
>> + * @return: test
>> + */
>> +static struct efi_unit_test *find_test(const u16 *testname)
>> +{
>> +    struct efi_unit_test *test;
>> +
>> +    for (test = ll_entry_start(struct efi_unit_test, efi_unit_test);
>> +         test < ll_entry_end(struct efi_unit_test, efi_unit_test); ++test) {
>> +            if (!efi_st_strcmp_16_8(testname, test->name))
> 
> Why not just use UCS2 strings here and compare 16 to 16? Maybe you're
> using the name more often in normal situations?
> 
> Either way, not a biggie :)

I thought about defining the test names as UCS2. But wide strings need
double the space.

So I decided to stay with char * as far as possible to reduce code size.

I remember that reviewing one patch you asked me to rearrange function
arguments to save 16 bytes of compiled code size for a specific
architecture.

Best regards

Heinrich
_______________________________________________
U-Boot mailing list
U-Boot@lists.denx.de
https://lists.denx.de/listinfo/u-boot

Reply via email to