1*83d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+ 2c7ae3dfdSSimon Glass /* 3c7ae3dfdSSimon Glass * EFI hello world 4c7ae3dfdSSimon Glass * 5c7ae3dfdSSimon Glass * Copyright (c) 2016 Google, Inc 6c7ae3dfdSSimon Glass * Written by Simon Glass <sjg@chromium.org> 7c7ae3dfdSSimon Glass * 8bbf75dd9SHeinrich Schuchardt * This program demonstrates calling a boottime service. 9bbf75dd9SHeinrich Schuchardt * It writes a greeting and the load options to the console. 10c7ae3dfdSSimon Glass */ 11c7ae3dfdSSimon Glass 12c7ae3dfdSSimon Glass #include <common.h> 13c7ae3dfdSSimon Glass #include <efi_api.h> 14c7ae3dfdSSimon Glass 15ae67dca5SAlexander Graf static const efi_guid_t loaded_image_guid = LOADED_IMAGE_GUID; 160aaabbb2SHeinrich Schuchardt static const efi_guid_t fdt_guid = EFI_FDT_GUID; 170aaabbb2SHeinrich Schuchardt static const efi_guid_t smbios_guid = SMBIOS_TABLE_GUID; 180aaabbb2SHeinrich Schuchardt 190aaabbb2SHeinrich Schuchardt static int hw_memcmp(const void *buf1, const void *buf2, size_t length) 200aaabbb2SHeinrich Schuchardt { 210aaabbb2SHeinrich Schuchardt const u8 *pos1 = buf1; 220aaabbb2SHeinrich Schuchardt const u8 *pos2 = buf2; 230aaabbb2SHeinrich Schuchardt 240aaabbb2SHeinrich Schuchardt for (; length; --length) { 250aaabbb2SHeinrich Schuchardt if (*pos1 != *pos2) 260aaabbb2SHeinrich Schuchardt return *pos1 - *pos2; 270aaabbb2SHeinrich Schuchardt ++pos1; 280aaabbb2SHeinrich Schuchardt ++pos2; 290aaabbb2SHeinrich Schuchardt } 300aaabbb2SHeinrich Schuchardt return 0; 310aaabbb2SHeinrich Schuchardt } 32ae67dca5SAlexander Graf 33bbf75dd9SHeinrich Schuchardt /* 34bbf75dd9SHeinrich Schuchardt * Entry point of the EFI application. 35bbf75dd9SHeinrich Schuchardt * 36bbf75dd9SHeinrich Schuchardt * @handle handle of the loaded image 37bbf75dd9SHeinrich Schuchardt * @systable system table 38bbf75dd9SHeinrich Schuchardt * @return status code 39bbf75dd9SHeinrich Schuchardt */ 40c7ae3dfdSSimon Glass efi_status_t EFIAPI efi_main(efi_handle_t handle, 41c7ae3dfdSSimon Glass struct efi_system_table *systable) 42c7ae3dfdSSimon Glass { 43c7ae3dfdSSimon Glass struct efi_simple_text_output_protocol *con_out = systable->con_out; 44c7ae3dfdSSimon Glass struct efi_boot_services *boottime = systable->boottime; 45bbf75dd9SHeinrich Schuchardt struct efi_loaded_image *loaded_image; 46bbf75dd9SHeinrich Schuchardt efi_status_t ret; 470aaabbb2SHeinrich Schuchardt efi_uintn_t i; 48f7c342f4SHeinrich Schuchardt u16 rev[] = L"0.0.0"; 49c7ae3dfdSSimon Glass 50c7ae3dfdSSimon Glass con_out->output_string(con_out, L"Hello, world!\n"); 51c7ae3dfdSSimon Glass 52f7c342f4SHeinrich Schuchardt /* Print the revision number */ 53f7c342f4SHeinrich Schuchardt rev[0] = (systable->hdr.revision >> 16) + '0'; 54f7c342f4SHeinrich Schuchardt rev[4] = systable->hdr.revision & 0xffff; 55f7c342f4SHeinrich Schuchardt for (; rev[4] >= 10;) { 56f7c342f4SHeinrich Schuchardt rev[4] -= 10; 57f7c342f4SHeinrich Schuchardt ++rev[2]; 58f7c342f4SHeinrich Schuchardt } 59f7c342f4SHeinrich Schuchardt /* Third digit is only to be shown if non-zero */ 60f7c342f4SHeinrich Schuchardt if (rev[4]) 61f7c342f4SHeinrich Schuchardt rev[4] += '0'; 62f7c342f4SHeinrich Schuchardt else 63f7c342f4SHeinrich Schuchardt rev[3] = 0; 64f7c342f4SHeinrich Schuchardt 65f7c342f4SHeinrich Schuchardt con_out->output_string(con_out, L"Running on UEFI "); 66f7c342f4SHeinrich Schuchardt con_out->output_string(con_out, rev); 67f7c342f4SHeinrich Schuchardt con_out->output_string(con_out, L"\n"); 68f7c342f4SHeinrich Schuchardt 69bbf75dd9SHeinrich Schuchardt /* Get the loaded image protocol */ 70bbf75dd9SHeinrich Schuchardt ret = boottime->handle_protocol(handle, &loaded_image_guid, 71bbf75dd9SHeinrich Schuchardt (void **)&loaded_image); 72bbf75dd9SHeinrich Schuchardt if (ret != EFI_SUCCESS) { 73bbf75dd9SHeinrich Schuchardt con_out->output_string(con_out, 74bbf75dd9SHeinrich Schuchardt L"Cannot open loaded image protocol\n"); 75bbf75dd9SHeinrich Schuchardt goto out; 76bbf75dd9SHeinrich Schuchardt } 770aaabbb2SHeinrich Schuchardt /* Find configuration tables */ 780aaabbb2SHeinrich Schuchardt for (i = 0; i < systable->nr_tables; ++i) { 790aaabbb2SHeinrich Schuchardt if (!hw_memcmp(&systable->tables[i].guid, &fdt_guid, 800aaabbb2SHeinrich Schuchardt sizeof(efi_guid_t))) 810aaabbb2SHeinrich Schuchardt con_out->output_string(con_out, L"Have device tree\n"); 820aaabbb2SHeinrich Schuchardt if (!hw_memcmp(&systable->tables[i].guid, &smbios_guid, 830aaabbb2SHeinrich Schuchardt sizeof(efi_guid_t))) 840aaabbb2SHeinrich Schuchardt con_out->output_string(con_out, L"Have SMBIOS table\n"); 850aaabbb2SHeinrich Schuchardt } 86bbf75dd9SHeinrich Schuchardt /* Output the load options */ 87bbf75dd9SHeinrich Schuchardt con_out->output_string(con_out, L"Load options: "); 88bbf75dd9SHeinrich Schuchardt if (loaded_image->load_options_size && loaded_image->load_options) 89bbf75dd9SHeinrich Schuchardt con_out->output_string(con_out, 90bbf75dd9SHeinrich Schuchardt (u16 *)loaded_image->load_options); 91bbf75dd9SHeinrich Schuchardt else 92bbf75dd9SHeinrich Schuchardt con_out->output_string(con_out, L"<none>"); 93bbf75dd9SHeinrich Schuchardt con_out->output_string(con_out, L"\n"); 94bbf75dd9SHeinrich Schuchardt 95bbf75dd9SHeinrich Schuchardt out: 96bbf75dd9SHeinrich Schuchardt boottime->exit(handle, ret, 0, NULL); 97bbf75dd9SHeinrich Schuchardt 98bbf75dd9SHeinrich Schuchardt /* We should never arrive here */ 99bbf75dd9SHeinrich Schuchardt return ret; 100c7ae3dfdSSimon Glass } 101