Skip to content

Commit

Permalink
setup_var patch
Browse files Browse the repository at this point in the history
1. add original setup_var patch
2. add brainsucker’s setup_var2 and lsefivar patch
3. add setup_var_3 patch
  • Loading branch information
datasone committed Aug 20, 2018
1 parent 6d273b6 commit e69e26a
Show file tree
Hide file tree
Showing 2 changed files with 339 additions and 0 deletions.
8 changes: 8 additions & 0 deletions grub-2.00+20130519/grub-core/Makefile.core.def
Original file line number Diff line number Diff line change
Expand Up @@ -635,6 +635,14 @@ module = {
enable = efi;
};

module = {
name = setup_var;

common = commands/efi/setup_var.c;

enable = efi;
};

module = {
name = lsefi;
common = commands/efi/lsefi.c;
Expand Down
331 changes: 331 additions & 0 deletions grub-2.00+20130519/grub-core/commands/setup_var.c
Original file line number Diff line number Diff line change
@@ -0,0 +1,331 @@
/* setup_var.c - InsydeH2o Setup variable modification tool, can modify single
* bytes within the Setup variable */
/* (c) 2009 by Bernhard Froemel
*
* GRUB -- GRand Unified Bootloader
* Copyright (C) 2002,2003,2005,2006,2007,2008,2009,2008 Free Software Foundation, Inc.
*
* GRUB is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* GRUB is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with GRUB. If not, see <http://www.gnu.org/licenses/>.
*/

/* mod by datasone adding setup_var_3 */

#include <grub/types.h>
#include <grub/dl.h>
#include <grub/misc.h>
#include <grub/command.h>
#include <grub/file.h>
#include <grub/efi/efi.h>
#include <grub/pci.h>

#define INSYDE_SETUP_VAR ((grub_efi_char16_t*)"S\0e\0t\0u\0p\0\0\0")
#define INSYDE_SETUP_VAR_NSIZE (12)
#define INSYDE_CUSTOM_VAR ((grub_efi_char16_t*)"C\0u\0s\0t\0o\0m\0\0\0")
#define INSYDE_CUSTOM_VAR_NSIZE (14)
#define INSYDE_SETUP_VAR_SIZE (0x2bc)
#define INSYDE_SETUP_VAR_GUID { 0xa04a27f4, 0xdf00, 0x4d42, { 0xb5, 0x52, 0x39, 0x51, 0x13, 0x02, 0x11, 0x3d } }
#define MAX_VARIABLE_SIZE (1024)
#define MAX_VAR_DATA_SIZE (65536)
#define CMDNAME_SETUP_VAR2 ("setup_var2")
#define CMDCHECK_SETUP_VAR2 (9)

#define CMDNAME_SETUP_VAR3 ("setup_var_3")
#define CMDCHECK_SETUP_VAR3 (10)

#define SETUP_VAR_SIZE_THRESHOLD (0x10)

GRUB_MOD_LICENSE("GPLv3+");

void print_varname(grub_efi_char16_t* str);

void print_varname(grub_efi_char16_t* str)
{
while(*str != 0x0)
{
grub_printf("%c", (grub_uint8_t) *str);
str++;
}
}

static grub_err_t
grub_cmd_setup_var (grub_command_t cmd,
int argc, char *argv[])
{
grub_efi_status_t status;
grub_efi_guid_t setup_var_guid = INSYDE_SETUP_VAR_GUID;
grub_efi_guid_t guid;
grub_uint8_t tmp_data[MAX_VAR_DATA_SIZE];
grub_uint16_t offset = 0x1af;
grub_efi_uintn_t setup_var_size = INSYDE_SETUP_VAR_SIZE;
grub_uint8_t set_value = 0x0;
grub_efi_uint32_t setup_var_attr = 0x7;
char* endptr;

grub_efi_char16_t name[MAX_VARIABLE_SIZE/2];
grub_efi_uintn_t name_size;

grub_uint16_t isMode2 = 0;
grub_uint16_t isMode3 = 0;
if (cmd->name[CMDCHECK_SETUP_VAR2] != 0) isMode2 = 1;
if (cmd->name[CMDCHECK_SETUP_VAR2] != 0 && cmd->name[CMDCHECK_SETUP_VAR3] != 0) isMode3 = 1;

if (argc == 0)
{
grub_printf("Hello!\n"
);
grub_printf(
"You may brick your InsydeH2o based laptop and need to send it in\n"
);
grub_printf(
"for repair.\n");
grub_printf(
"Some vendors (like Sony) are rather slow, and it could get expensive if they\n"
);
grub_printf(
"discover that you used this or similar tools.\nYou should be *very* sure what you do.\n"
);
grub_printf(
"\n\nThis Setup variable modification tool may only work with current (July 2009)\n"
);
grub_printf(
"Insyde H2o based firmware releases. Vendors may decide to lock the EFI\n"
);
grub_printf(
"environment completely out or introduce other security measures.\n"
);
grub_printf(
"Final warning: YOU MAY BRICK YOUR LAPTOP IF YOU USE THIS TOOL - I TAKE N O \n"
);
grub_printf(
"RESPONSIBILITY FOR Y O U R ACTIONS.\n"
);
grub_printf(
"\n\n(c) 2009 by Bernhard Froemel <[email protected]>\n"
);
}

name[0] = 0x0;
/* scan for Setup variable */
grub_printf("Looking for Setup variable...\n");
do
{
name_size = MAX_VARIABLE_SIZE;
status = efi_call_3(grub_efi_system_table->runtime_services->get_next_variable_name,
&name_size,
name,
&guid);

if(status == GRUB_EFI_NOT_FOUND)
{ /* finished traversing VSS */
break;
}

if(status)
{
grub_printf("status: 0x%02x\n", (grub_uint32_t) status);
}
if(! status && ((name_size == INSYDE_SETUP_VAR_NSIZE && 0 == grub_memcmp(name, INSYDE_SETUP_VAR, name_size)) ||
(isMode2 && name_size == INSYDE_CUSTOM_VAR_NSIZE && 0 == grub_memcmp(name, INSYDE_CUSTOM_VAR, name_size))))
{
grub_printf("var name: ");
print_varname(name);
grub_printf(", var size: %u, var guid: %08x-%04x-%04x - %02x-%02x-%02x-%02x-%02x-%02x-%02x-%02x\n\n",
(grub_uint32_t) name_size,
guid.data1,
guid.data2,
guid.data3,
guid.data4[0], guid.data4[1], guid.data4[2], guid.data4[3], guid.data4[4], guid.data4[5], guid.data4[6], guid.data4[7]
);

if(grub_memcmp(&guid, &setup_var_guid, sizeof(grub_efi_guid_t)) == 0)
{
grub_printf("--> GUID matches expected GUID\n");
}
else
{
grub_printf("--> GUID does not match expected GUID, taking it nevertheless...\n");
grub_memcpy(&setup_var_guid, &guid, sizeof(grub_efi_guid_t));
}


/* obtain current contents of Setup variable */
if(argc >= 1 && argc < 3)
{
grub_errno = 0;
offset = grub_strtoul(argv[0], &endptr, 16);
if(endptr == argv[0] || grub_errno != 0)
{
return grub_error(GRUB_ERR_BAD_ARGUMENT, "can't decode your first argument. Please provide a hex value (e.g. 0x1af).");
}
status = efi_call_5(grub_efi_system_table->runtime_services->get_variable,
name,
&setup_var_guid,
&setup_var_attr,
&setup_var_size,
tmp_data);
if(status == GRUB_EFI_BUFFER_TOO_SMALL)
{
grub_printf("expected a different size of the Setup variable (got %d (0x%x) bytes). Continue with care...\n", (int)setup_var_size, (int)setup_var_size);
status = efi_call_5(grub_efi_system_table->runtime_services->get_variable,
name,
&setup_var_guid,
&setup_var_attr,
&setup_var_size,
tmp_data);
}
if(status)
{
return grub_error(GRUB_ERR_INVALID_COMMAND, "can't get variable using efi (error: 0x%016x)", status);
}
grub_printf("successfully obtained \"Setup\" variable from VSS (got %d (0x%x) bytes).\n", (int)setup_var_size, (int)setup_var_size);
if(offset > setup_var_size)
{
/* When in mode 3 and the Setup variable size is too small(smaller than threshold, 0x10 here), supress the error and continue to the next Setup variable */
if (isMode3 && setup_var_size < SETUP_VAR_SIZE_THRESHOLD)
{
grub_printf("Too small Setup variable detected, ignoring.\n\n");
continue;
}
return grub_error(GRUB_ERR_BAD_ARGUMENT, "offset is out of range.");
}
grub_printf("offset 0x%02x is: 0x%02x\n", offset, tmp_data[offset]);
}
/* modify and write Setup variable, if user requests it */
if(argc == 2)
{
set_value = grub_strtoul(argv[1], &endptr, 16);
if(endptr == argv[1] || grub_errno != 0)
{
return grub_error(GRUB_ERR_BAD_ARGUMENT, "can't decode your second argument. Please provide a hex value (e.g. 0x01).");
}
grub_printf("setting offset 0x%02x to 0x%02x\n", offset, set_value);
tmp_data[offset] = set_value;
status = efi_call_5(grub_efi_system_table->runtime_services->set_variable,
name,
&setup_var_guid,
setup_var_attr,
setup_var_size,
tmp_data);
if(status)
{
return grub_error(GRUB_ERR_INVALID_COMMAND, "can't set variable using efi (error: 0x%016x)", status);
}
}


}
} while (! status);



if(argc == 0 || argc > 2)
{
return grub_error(GRUB_ERR_BAD_ARGUMENT, "Usage: %s offset [setval]", cmd->name);
}
return grub_errno;
}

static grub_err_t
grub_cmd_lsefivar (grub_command_t cmd __attribute__ ((unused)),
int argc __attribute__ ((unused)), char *argv[] __attribute__ ((unused)))
{
grub_efi_status_t status;
grub_efi_guid_t guid;
grub_uint8_t tmp_data[MAX_VAR_DATA_SIZE];
grub_efi_uintn_t setup_var_size = INSYDE_SETUP_VAR_SIZE;
grub_efi_uint32_t setup_var_attr = 0x7;

grub_efi_char16_t name[MAX_VARIABLE_SIZE/2];
grub_efi_uintn_t name_size;

name[0] = 0x0;
/* scan for Setup variable */
grub_printf("Listing EFI variables...\n");
do
{
name_size = MAX_VARIABLE_SIZE;
status = efi_call_3(grub_efi_system_table->runtime_services->get_next_variable_name,
&name_size,
name,
&guid);

if(status == GRUB_EFI_NOT_FOUND)
{ /* finished traversing VSS */
break;
}

if(status)
{
grub_printf("status: 0x%02x\n", (grub_uint32_t) status);
}
if(! status)
{
setup_var_size = 1;
status = efi_call_5(grub_efi_system_table->runtime_services->get_variable,
name,
&guid,
&setup_var_attr,
&setup_var_size,
tmp_data);
if (status && status != GRUB_EFI_BUFFER_TOO_SMALL)
{
grub_printf("error (0x%x) getting var size:\n ", (grub_uint32_t)status);
setup_var_size = 0;
}
status = 0;

grub_printf("name size: %02u, var size: %06u (0x%06x), var guid: %08x-%04x-%04x - %02x-%02x-%02x-%02x-%02x-%02x-%02x-%02x, name: ",
(grub_uint32_t) name_size, (grub_uint32_t) setup_var_size, (grub_uint32_t) setup_var_size,
guid.data1,
guid.data2,
guid.data3,
guid.data4[0], guid.data4[1], guid.data4[2], guid.data4[3], guid.data4[4], guid.data4[5], guid.data4[6], guid.data4[7]
);
print_varname(name);
grub_printf("\n");
}
} while (! status);

return grub_errno;
}


static grub_command_t cmd_setup_var;
static grub_command_t cmd_setup_var2;
static grub_command_t cmd_setup_var_3;
static grub_command_t cmd_setup_lsvar;

GRUB_MOD_INIT(setup_var)
{
cmd_setup_var = grub_register_command ("setup_var", grub_cmd_setup_var,
"setup_var offset [setval]",
"Read/Write specific (byte) offset of setup variable.");
cmd_setup_var2 = grub_register_command ("setup_var2", grub_cmd_setup_var,
"setup_var2 offset [setval]",
"Read/Write specific (byte) offset of setup and custom variables.");
cmd_setup_var_3 = grub_register_command ("setup_var_3", grub_cmd_setup_var,
"setup_var_3 offset [setval]",
"Read/Write specific (byte) offset of setup variables ignoring error, use with great caution!!!");
cmd_setup_lsvar = grub_register_command ("lsefivar", grub_cmd_lsefivar,
"lsefivar",
"Lists all efi variables.");
}

GRUB_MOD_FINI(setup_var)
{
grub_unregister_command (cmd_setup_var);
grub_unregister_command (cmd_setup_var2);
grub_unregister_command (cmd_setup_var_3);
grub_unregister_command(cmd_setup_lsvar);
}

0 comments on commit e69e26a

Please sign in to comment.