2019-02-02 10:41:15 +01:00
// SPDX-License-Identifier: GPL-2.0
2016-04-25 21:06:57 +01:00
/*
* efibc : control EFI bootloaders which obey LoaderEntryOneShot var
* Copyright ( c ) 2013 - 2016 , Intel Corporation .
*/
# define pr_fmt(fmt) "efibc: " fmt
# include <linux/efi.h>
# include <linux/module.h>
# include <linux/reboot.h>
2016-05-06 22:39:28 +01:00
# include <linux/slab.h>
2016-04-25 21:06:57 +01:00
static void efibc_str_to_str16 ( const char * str , efi_char16_t * str16 )
{
size_t i ;
for ( i = 0 ; i < strlen ( str ) ; i + + )
str16 [ i ] = str [ i ] ;
str16 [ i ] = ' \0 ' ;
}
2016-05-06 22:39:28 +01:00
static int efibc_set_variable ( const char * name , const char * value )
2016-04-25 21:06:57 +01:00
{
int ret ;
efi_guid_t guid = LINUX_EFI_LOADER_ENTRY_GUID ;
2016-05-06 22:39:28 +01:00
struct efivar_entry * entry ;
2016-04-25 21:06:57 +01:00
size_t size = ( strlen ( value ) + 1 ) * sizeof ( efi_char16_t ) ;
2016-05-06 22:39:28 +01:00
if ( size > sizeof ( entry - > var . Data ) ) {
2016-06-25 08:20:24 +01:00
pr_err ( " value is too large (%zu bytes) for '%s' EFI variable \n " , size , name ) ;
2016-05-06 22:39:28 +01:00
return - EINVAL ;
}
2016-04-25 21:06:57 +01:00
2016-05-06 22:39:28 +01:00
entry = kmalloc ( sizeof ( * entry ) , GFP_KERNEL ) ;
if ( ! entry ) {
2016-06-25 08:20:24 +01:00
pr_err ( " failed to allocate efivar entry for '%s' EFI variable \n " , name ) ;
2016-05-06 22:39:28 +01:00
return - ENOMEM ;
}
2016-04-25 21:06:57 +01:00
2016-05-06 22:39:28 +01:00
efibc_str_to_str16 ( name , entry - > var . VariableName ) ;
efibc_str_to_str16 ( value , ( efi_char16_t * ) entry - > var . Data ) ;
memcpy ( & entry - > var . VendorGuid , & guid , sizeof ( guid ) ) ;
2019-06-12 16:18:10 +08:00
ret = efivar_entry_set_safe ( entry - > var . VariableName ,
entry - > var . VendorGuid ,
EFI_VARIABLE_NON_VOLATILE
| EFI_VARIABLE_BOOTSERVICE_ACCESS
| EFI_VARIABLE_RUNTIME_ACCESS ,
false , size , entry - > var . Data ) ;
2016-04-25 21:06:57 +01:00
if ( ret )
pr_err ( " failed to set %s EFI variable: 0x%x \n " ,
name , ret ) ;
2016-05-06 22:39:28 +01:00
kfree ( entry ) ;
return ret ;
2016-04-25 21:06:57 +01:00
}
static int efibc_reboot_notifier_call ( struct notifier_block * notifier ,
unsigned long event , void * data )
{
const char * reason = " shutdown " ;
2016-05-06 22:39:28 +01:00
int ret ;
2016-04-25 21:06:57 +01:00
if ( event = = SYS_RESTART )
reason = " reboot " ;
2016-05-06 22:39:28 +01:00
ret = efibc_set_variable ( " LoaderEntryRebootReason " , reason ) ;
if ( ret | | ! data )
2016-04-25 21:06:57 +01:00
return NOTIFY_DONE ;
efibc_set_variable ( " LoaderEntryOneShot " , ( char * ) data ) ;
return NOTIFY_DONE ;
}
static struct notifier_block efibc_reboot_notifier = {
. notifier_call = efibc_reboot_notifier_call ,
} ;
static int __init efibc_init ( void )
{
int ret ;
if ( ! efi_enabled ( EFI_RUNTIME_SERVICES ) )
return - ENODEV ;
ret = register_reboot_notifier ( & efibc_reboot_notifier ) ;
if ( ret )
pr_err ( " unable to register reboot notifier \n " ) ;
return ret ;
}
module_init ( efibc_init ) ;
static void __exit efibc_exit ( void )
{
unregister_reboot_notifier ( & efibc_reboot_notifier ) ;
}
module_exit ( efibc_exit ) ;
MODULE_AUTHOR ( " Jeremy Compostella <jeremy.compostella@intel.com> " ) ;
MODULE_AUTHOR ( " Matt Gumbel <matthew.k.gumbel@intel.com " ) ;
MODULE_DESCRIPTION ( " EFI Bootloader Control " ) ;
MODULE_LICENSE ( " GPL v2 " ) ;