183d290c5STom 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; 17*47cae019SBin Meng static const efi_guid_t acpi_guid = EFI_ACPI_TABLE_GUID; 180aaabbb2SHeinrich Schuchardt static const efi_guid_t smbios_guid = SMBIOS_TABLE_GUID; 190aaabbb2SHeinrich Schuchardt 200aaabbb2SHeinrich Schuchardt static int hw_memcmp(const void *buf1, const void *buf2, size_t length) 210aaabbb2SHeinrich Schuchardt { 220aaabbb2SHeinrich Schuchardt const u8 *pos1 = buf1; 230aaabbb2SHeinrich Schuchardt const u8 *pos2 = buf2; 240aaabbb2SHeinrich Schuchardt 250aaabbb2SHeinrich Schuchardt for (; length; --length) { 260aaabbb2SHeinrich Schuchardt if (*pos1 != *pos2) 270aaabbb2SHeinrich Schuchardt return *pos1 - *pos2; 280aaabbb2SHeinrich Schuchardt ++pos1; 290aaabbb2SHeinrich Schuchardt ++pos2; 300aaabbb2SHeinrich Schuchardt } 310aaabbb2SHeinrich Schuchardt return 0; 320aaabbb2SHeinrich Schuchardt } 33ae67dca5SAlexander Graf 34bbf75dd9SHeinrich Schuchardt /* 35bbf75dd9SHeinrich Schuchardt * Entry point of the EFI application. 36bbf75dd9SHeinrich Schuchardt * 37bbf75dd9SHeinrich Schuchardt * @handle handle of the loaded image 38bbf75dd9SHeinrich Schuchardt * @systable system table 39bbf75dd9SHeinrich Schuchardt * @return status code 40bbf75dd9SHeinrich Schuchardt */ 41c7ae3dfdSSimon Glass efi_status_t EFIAPI efi_main(efi_handle_t handle, 42c7ae3dfdSSimon Glass struct efi_system_table *systable) 43c7ae3dfdSSimon Glass { 44c7ae3dfdSSimon Glass struct efi_simple_text_output_protocol *con_out = systable->con_out; 45c7ae3dfdSSimon Glass struct efi_boot_services *boottime = systable->boottime; 46bbf75dd9SHeinrich Schuchardt struct efi_loaded_image *loaded_image; 47bbf75dd9SHeinrich Schuchardt efi_status_t ret; 480aaabbb2SHeinrich Schuchardt efi_uintn_t i; 49f7c342f4SHeinrich Schuchardt u16 rev[] = L"0.0.0"; 50c7ae3dfdSSimon Glass 51c7ae3dfdSSimon Glass con_out->output_string(con_out, L"Hello, world!\n"); 52c7ae3dfdSSimon Glass 53f7c342f4SHeinrich Schuchardt /* Print the revision number */ 54f7c342f4SHeinrich Schuchardt rev[0] = (systable->hdr.revision >> 16) + '0'; 55f7c342f4SHeinrich Schuchardt rev[4] = systable->hdr.revision & 0xffff; 56f7c342f4SHeinrich Schuchardt for (; rev[4] >= 10;) { 57f7c342f4SHeinrich Schuchardt rev[4] -= 10; 58f7c342f4SHeinrich Schuchardt ++rev[2]; 59f7c342f4SHeinrich Schuchardt } 60f7c342f4SHeinrich Schuchardt /* Third digit is only to be shown if non-zero */ 61f7c342f4SHeinrich Schuchardt if (rev[4]) 62f7c342f4SHeinrich Schuchardt rev[4] += '0'; 63f7c342f4SHeinrich Schuchardt else 64f7c342f4SHeinrich Schuchardt rev[3] = 0; 65f7c342f4SHeinrich Schuchardt 66f7c342f4SHeinrich Schuchardt con_out->output_string(con_out, L"Running on UEFI "); 67f7c342f4SHeinrich Schuchardt con_out->output_string(con_out, rev); 68f7c342f4SHeinrich Schuchardt con_out->output_string(con_out, L"\n"); 69f7c342f4SHeinrich Schuchardt 70bbf75dd9SHeinrich Schuchardt /* Get the loaded image protocol */ 71bbf75dd9SHeinrich Schuchardt ret = boottime->handle_protocol(handle, &loaded_image_guid, 72bbf75dd9SHeinrich Schuchardt (void **)&loaded_image); 73bbf75dd9SHeinrich Schuchardt if (ret != EFI_SUCCESS) { 74bbf75dd9SHeinrich Schuchardt con_out->output_string(con_out, 75bbf75dd9SHeinrich Schuchardt L"Cannot open loaded image protocol\n"); 76bbf75dd9SHeinrich Schuchardt goto out; 77bbf75dd9SHeinrich Schuchardt } 780aaabbb2SHeinrich Schuchardt /* Find configuration tables */ 790aaabbb2SHeinrich Schuchardt for (i = 0; i < systable->nr_tables; ++i) { 800aaabbb2SHeinrich Schuchardt if (!hw_memcmp(&systable->tables[i].guid, &fdt_guid, 810aaabbb2SHeinrich Schuchardt sizeof(efi_guid_t))) 820aaabbb2SHeinrich Schuchardt con_out->output_string(con_out, L"Have device tree\n"); 83*47cae019SBin Meng if (!hw_memcmp(&systable->tables[i].guid, &acpi_guid, 84*47cae019SBin Meng sizeof(efi_guid_t))) 85*47cae019SBin Meng con_out->output_string(con_out, L"Have ACPI 2.0 table\n"); 860aaabbb2SHeinrich Schuchardt if (!hw_memcmp(&systable->tables[i].guid, &smbios_guid, 870aaabbb2SHeinrich Schuchardt sizeof(efi_guid_t))) 880aaabbb2SHeinrich Schuchardt con_out->output_string(con_out, L"Have SMBIOS table\n"); 890aaabbb2SHeinrich Schuchardt } 90bbf75dd9SHeinrich Schuchardt /* Output the load options */ 91bbf75dd9SHeinrich Schuchardt con_out->output_string(con_out, L"Load options: "); 92bbf75dd9SHeinrich Schuchardt if (loaded_image->load_options_size && loaded_image->load_options) 93bbf75dd9SHeinrich Schuchardt con_out->output_string(con_out, 94bbf75dd9SHeinrich Schuchardt (u16 *)loaded_image->load_options); 95bbf75dd9SHeinrich Schuchardt else 96bbf75dd9SHeinrich Schuchardt con_out->output_string(con_out, L"<none>"); 97bbf75dd9SHeinrich Schuchardt con_out->output_string(con_out, L"\n"); 98bbf75dd9SHeinrich Schuchardt 99bbf75dd9SHeinrich Schuchardt out: 100bbf75dd9SHeinrich Schuchardt boottime->exit(handle, ret, 0, NULL); 101bbf75dd9SHeinrich Schuchardt 102bbf75dd9SHeinrich Schuchardt /* We should never arrive here */ 103bbf75dd9SHeinrich Schuchardt return ret; 104c7ae3dfdSSimon Glass } 105