pc-bios/s390-ccw: fix non-sequential boot entries (eckd)

zIPL boot menu entries can be non-sequential. Let's account
for this issue for the s390 zIPL boot menu. Since this boot
menu is actually an imitation and is not completely capable
of everything the real zIPL menu can do, let's also print a
different banner to the user.

Signed-off-by: Collin Walling <walling@linux.ibm.com>
Reported-by: Vasily Gorbik <gor@linux.ibm.com>
Reviewed-by: Thomas Huth <thuth@redhat.com>
Reviewed-by: Janosch Frank <frankja@linux.ibm.com>
Signed-off-by: Thomas Huth <thuth@redhat.com>
This commit is contained in:
Collin Walling 2018-04-16 12:56:09 -04:00 committed by Thomas Huth
parent 074afe60d4
commit 7385e947fc

View File

@ -158,7 +158,7 @@ static void boot_menu_prompt(bool retry)
} }
} }
static int get_boot_index(int entries) static int get_boot_index(bool *valid_entries)
{ {
int boot_index; int boot_index;
bool retry = false; bool retry = false;
@ -168,7 +168,8 @@ static int get_boot_index(int entries)
boot_menu_prompt(retry); boot_menu_prompt(retry);
boot_index = get_index(); boot_index = get_index();
retry = true; retry = true;
} while (boot_index < 0 || boot_index >= entries); } while (boot_index < 0 || boot_index >= MAX_BOOT_ENTRIES ||
!valid_entries[boot_index]);
sclp_print("\nBooting entry #"); sclp_print("\nBooting entry #");
sclp_print(uitoa(boot_index, tmp, sizeof(tmp))); sclp_print(uitoa(boot_index, tmp, sizeof(tmp)));
@ -176,7 +177,8 @@ static int get_boot_index(int entries)
return boot_index; return boot_index;
} }
static void zipl_println(const char *data, size_t len) /* Returns the entry number that was printed */
static int zipl_print_entry(const char *data, size_t len)
{ {
char buf[len + 2]; char buf[len + 2];
@ -185,12 +187,15 @@ static void zipl_println(const char *data, size_t len)
buf[len + 1] = '\0'; buf[len + 1] = '\0';
sclp_print(buf); sclp_print(buf);
return buf[0] == ' ' ? atoui(buf + 1) : atoui(buf);
} }
int menu_get_zipl_boot_index(const char *menu_data) int menu_get_zipl_boot_index(const char *menu_data)
{ {
size_t len; size_t len;
int entries; int entry;
bool valid_entries[MAX_BOOT_ENTRIES] = {false};
uint16_t zipl_flag = *(uint16_t *)(menu_data - ZIPL_FLAG_OFFSET); uint16_t zipl_flag = *(uint16_t *)(menu_data - ZIPL_FLAG_OFFSET);
uint16_t zipl_timeout = *(uint16_t *)(menu_data - ZIPL_TIMEOUT_OFFSET); uint16_t zipl_timeout = *(uint16_t *)(menu_data - ZIPL_TIMEOUT_OFFSET);
@ -202,19 +207,25 @@ int menu_get_zipl_boot_index(const char *menu_data)
timeout = zipl_timeout * 1000; timeout = zipl_timeout * 1000;
} }
/* Print and count all menu items, including the banner */ /* Print banner */
for (entries = 0; *menu_data; entries++) { sclp_print("s390-ccw zIPL Boot Menu\n\n");
menu_data += strlen(menu_data) + 1;
/* Print entries */
while (*menu_data) {
len = strlen(menu_data); len = strlen(menu_data);
zipl_println(menu_data, len); entry = zipl_print_entry(menu_data, len);
menu_data += len + 1; menu_data += len + 1;
if (entries < 2) { valid_entries[entry] = true;
if (entry == 0) {
sclp_print("\n"); sclp_print("\n");
} }
} }
sclp_print("\n"); sclp_print("\n");
return get_boot_index(entries - 1); /* subtract 1 to exclude banner */ return get_boot_index(valid_entries);
} }