forked from microsoft/mu_feature_config
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathWriteConfVarListToUefiVars.py
147 lines (115 loc) · 4.15 KB
/
WriteConfVarListToUefiVars.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
# @file
#
# Set a dmpstore formatted set of variables to NVRAM
#
# Copyright (c), Microsoft Corporation
# SPDX-License-Identifier: BSD-2-Clause-Patent
import os
import sys
import logging
import argparse
import struct
import uuid
import ctypes
from SettingSupport.UefiVariablesSupportLib import UefiVariable
gEfiGlobalVariableGuid = "8BE4DF61-93CA-11D2-AA0D-00E098032B8C"
def option_parser():
parser = argparse.ArgumentParser()
parser.add_argument(
"-l",
"--load",
dest="setting_file",
required=True,
type=str,
default='""',
help="""Specify the input setting file""",
)
arguments = parser.parse_args()
if not os.path.isfile(arguments.setting_file):
print("Invalid input file: %s" % arguments.setting_file)
sys.exit(1)
return arguments
#
# Create an unpack statement formatted to address the variable length
# parameters in the var store (Data and Unicode Name)
def create_unpack_statement(NameStringSize, DataBufferSize):
#
# Dmpstore Format taken from AppendSingleVariableToFile() in
# ShellPkg\Library\UefiShellDebug1CommandsLib\DmpStore.c
# NameSize
# DataSize
# Name[NameSize]
# EFI_GUID
# Attributes
# Data[DataSize]
# CRC32
unpack_statement = "<II"
unpack_statement = unpack_statement + f"{NameStringSize}s"
unpack_statement = unpack_statement + "16s"
unpack_statement = unpack_statement + "I"
unpack_statement = unpack_statement + f"{DataBufferSize}s"
unpack_statement = unpack_statement + "I"
logging.debug(f"Created unpack statement {unpack_statement}")
return unpack_statement
#
# Using the passed byte array, extract the variable data and
# write it into nvram
#
# Return the size of the un
#
def extract_single_var_from_file_and_write_nvram(var):
# check that the passed byte array has at least enough space for the NameSize and DataSize
if len(var) > 8:
(NameSize, DataSize) = struct.unpack("<II", var[0:8])
unpack_statement = create_unpack_statement(NameSize, DataSize)
# check that the input byte array has at least enough space for unpack statement
if struct.calcsize(unpack_statement) > len(var):
logging.critical("Input File Parsing error: input buffer is smaller than unpack size")
return len(var)
result = struct.unpack(unpack_statement, var[0: struct.calcsize(unpack_statement)])
VarName = result[2].decode('utf16')
Guid = uuid.UUID(bytes_le=result[3])
Attributes = result[4]
Data = result[5]
logging.debug(f"Found Variable: {VarName} {Guid} {Attributes}")
UefiVar = UefiVariable()
(rc, err, error_string) = UefiVar.SetUefiVar(
VarName,
Guid,
Data,
Attributes,
)
if rc == 0:
logging.debug(f"Error returned from SetUefiVar: {rc}")
return struct.calcsize(unpack_statement)
else:
logging.critical("var buffer was too small to be a valid dmpstore")
return len(var)
#
# main script function
#
def main():
arguments = option_parser()
# read the entire file
with open(arguments.setting_file, "rb") as file:
var = file.read()
# go through the entire file parsing each dmpstore variable
start = 0
while len(var[start:]) != 0:
start = start + extract_single_var_from_file_and_write_nvram(var[start:])
return 0
if __name__ == "__main__":
# setup main console as logger
logger = logging.getLogger("")
logger.setLevel(logging.INFO)
formatter = logging.Formatter("%(levelname)s - %(message)s")
console = logging.StreamHandler()
console.setLevel(logging.CRITICAL)
# check the privilege level and report error
if not ctypes.windll.shell32.IsUserAnAdmin():
print("Administrator privilege required. Please launch from an Administrator privilege level.")
sys.exit(1)
# call main worker function
retcode = main()
logging.shutdown()
sys.exit(retcode)