Added autoinstall and updating from extracted RNode Firmwares to rnodeconf
This commit is contained in:
parent
eed7698ed3
commit
2a1ec6592c
@ -867,87 +867,119 @@ selected_hash = None
|
|||||||
firmware_version_url = "https://unsigned.io/firmware/latest/?v="+program_version+"&variant="
|
firmware_version_url = "https://unsigned.io/firmware/latest/?v="+program_version+"&variant="
|
||||||
def ensure_firmware_file(fw_filename):
|
def ensure_firmware_file(fw_filename):
|
||||||
global selected_version, selected_hash, upd_nocheck
|
global selected_version, selected_hash, upd_nocheck
|
||||||
try:
|
if fw_filename == "extracted_rnode_firmware.zip":
|
||||||
if selected_version == None:
|
vfpath = EXT_DIR+"/extracted_rnode_firmware.version"
|
||||||
if not upd_nocheck:
|
if os.path.isfile(vfpath):
|
||||||
try:
|
required_files = [
|
||||||
urlretrieve(firmware_version_url+fw_filename, UPD_DIR+"/"+fw_filename+".version.latest")
|
"extracted_console_image.bin",
|
||||||
except Exception as e:
|
"extracted_rnode_firmware.bin",
|
||||||
RNS.log("Failed to retrive latest version information for your board.")
|
"extracted_rnode_firmware.boot_app0",
|
||||||
RNS.log("Check your internet connection and try again.")
|
"extracted_rnode_firmware.bootloader",
|
||||||
RNS.log("If you don't have Internet access currently, use the --fw-version option to manually specify a version.")
|
"extracted_rnode_firmware.partitions",
|
||||||
exit()
|
]
|
||||||
|
parts_missing = False
|
||||||
|
for rf in required_files:
|
||||||
|
if not os.path.isfile(EXT_DIR+"/"+rf):
|
||||||
|
parts_missing = True
|
||||||
|
|
||||||
import shutil
|
if parts_missing:
|
||||||
file = open(UPD_DIR+"/"+fw_filename+".version.latest", "rb")
|
RNS.log("One or more required firmware files are missing from the extracted RNode")
|
||||||
release_info = file.read().decode("utf-8").strip()
|
RNS.log("Firmware archive. Installation cannot continue. Please try extracting the")
|
||||||
selected_version = release_info.split()[0]
|
RNS.log("firmware again with the --extract-firmware option.")
|
||||||
selected_hash = release_info.split()[1]
|
exit(184)
|
||||||
if not os.path.isdir(UPD_DIR+"/"+selected_version):
|
|
||||||
os.makedirs(UPD_DIR+"/"+selected_version)
|
|
||||||
shutil.copy(UPD_DIR+"/"+fw_filename+".version.latest", UPD_DIR+"/"+selected_version+"/"+fw_filename+".version")
|
|
||||||
RNS.log("The latest firmware for this board is version "+selected_version)
|
|
||||||
|
|
||||||
else:
|
vf = open(vfpath, "rb")
|
||||||
RNS.log("Online firmware version check was disabled, but no firmware version specified for install.")
|
release_info = vf.read().decode("utf-8").strip()
|
||||||
RNS.log("use the --fw-version option to manually specify a version.")
|
selected_version = release_info.split()[0]
|
||||||
exit(98)
|
selected_hash = release_info.split()[1]
|
||||||
|
RNS.log("Using existing firmware file: "+fw_filename+" for version "+selected_version)
|
||||||
update_target_url = firmware_update_url+selected_version+"/"+fw_filename
|
else:
|
||||||
|
RNS.log("No extracted firmware is available, cannot continue.")
|
||||||
|
RNS.log("Extract a firmware from an existing RNode first, using the --extract-firmware option.")
|
||||||
|
exit(183)
|
||||||
|
|
||||||
|
else:
|
||||||
try:
|
try:
|
||||||
if not os.path.isdir(UPD_DIR+"/"+selected_version):
|
if selected_version == None:
|
||||||
os.makedirs(UPD_DIR+"/"+selected_version)
|
if not upd_nocheck:
|
||||||
|
try:
|
||||||
|
urlretrieve(firmware_version_url+fw_filename, UPD_DIR+"/"+fw_filename+".version.latest")
|
||||||
|
except Exception as e:
|
||||||
|
RNS.log("Failed to retrive latest version information for your board.")
|
||||||
|
RNS.log("Check your internet connection and try again.")
|
||||||
|
RNS.log("If you don't have Internet access currently, use the --fw-version option to manually specify a version.")
|
||||||
|
exit()
|
||||||
|
|
||||||
if not os.path.isfile(UPD_DIR+"/"+selected_version+"/"+fw_filename):
|
import shutil
|
||||||
RNS.log("Downloading missing firmware file: "+fw_filename+" for version "+selected_version)
|
file = open(UPD_DIR+"/"+fw_filename+".version.latest", "rb")
|
||||||
urlretrieve(update_target_url, UPD_DIR+"/"+selected_version+"/"+fw_filename)
|
release_info = file.read().decode("utf-8").strip()
|
||||||
RNS.log("Firmware file downloaded")
|
selected_version = release_info.split()[0]
|
||||||
else:
|
selected_hash = release_info.split()[1]
|
||||||
RNS.log("Using existing firmware file: "+fw_filename+" for version "+selected_version)
|
if not os.path.isdir(UPD_DIR+"/"+selected_version):
|
||||||
|
os.makedirs(UPD_DIR+"/"+selected_version)
|
||||||
|
shutil.copy(UPD_DIR+"/"+fw_filename+".version.latest", UPD_DIR+"/"+selected_version+"/"+fw_filename+".version")
|
||||||
|
RNS.log("The latest firmware for this board is version "+selected_version)
|
||||||
|
|
||||||
|
else:
|
||||||
|
RNS.log("Online firmware version check was disabled, but no firmware version specified for install.")
|
||||||
|
RNS.log("use the --fw-version option to manually specify a version.")
|
||||||
|
exit(98)
|
||||||
|
|
||||||
|
update_target_url = firmware_update_url+selected_version+"/"+fw_filename
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if selected_hash == None:
|
if not os.path.isdir(UPD_DIR+"/"+selected_version):
|
||||||
try:
|
os.makedirs(UPD_DIR+"/"+selected_version)
|
||||||
file = open(UPD_DIR+"/"+selected_version+"/"+fw_filename+".version", "rb")
|
|
||||||
release_info = file.read().decode("utf-8").strip()
|
|
||||||
selected_hash = release_info.split()[1]
|
|
||||||
except Exception as e:
|
|
||||||
RNS.log("Could not read locally cached release information.")
|
|
||||||
RNS.log("You can clear the cache with the --clear-cache option and try again.")
|
|
||||||
|
|
||||||
if selected_hash == None:
|
if not os.path.isfile(UPD_DIR+"/"+selected_version+"/"+fw_filename):
|
||||||
RNS.log("No release hash found for "+fw_filename+". The firmware integrity could not be verified.")
|
RNS.log("Downloading missing firmware file: "+fw_filename+" for version "+selected_version)
|
||||||
exit(97)
|
urlretrieve(update_target_url, UPD_DIR+"/"+selected_version+"/"+fw_filename)
|
||||||
|
RNS.log("Firmware file downloaded")
|
||||||
RNS.log("Verifying firmware integrity...")
|
|
||||||
fw_file = open(UPD_DIR+"/"+selected_version+"/"+fw_filename, "rb")
|
|
||||||
expected_hash = bytes.fromhex(selected_hash)
|
|
||||||
file_hash = hashlib.sha256(fw_file.read()).hexdigest()
|
|
||||||
if file_hash == selected_hash:
|
|
||||||
pass
|
|
||||||
else:
|
else:
|
||||||
RNS.log("")
|
RNS.log("Using existing firmware file: "+fw_filename+" for version "+selected_version)
|
||||||
RNS.log("Firmware corrupt. Try clearing the local firmware cache with: rnodeconf --clear-cache")
|
|
||||||
exit(96)
|
try:
|
||||||
|
if selected_hash == None:
|
||||||
|
try:
|
||||||
|
file = open(UPD_DIR+"/"+selected_version+"/"+fw_filename+".version", "rb")
|
||||||
|
release_info = file.read().decode("utf-8").strip()
|
||||||
|
selected_hash = release_info.split()[1]
|
||||||
|
except Exception as e:
|
||||||
|
RNS.log("Could not read locally cached release information.")
|
||||||
|
RNS.log("You can clear the cache with the --clear-cache option and try again.")
|
||||||
|
|
||||||
|
if selected_hash == None:
|
||||||
|
RNS.log("No release hash found for "+fw_filename+". The firmware integrity could not be verified.")
|
||||||
|
exit(97)
|
||||||
|
|
||||||
|
RNS.log("Verifying firmware integrity...")
|
||||||
|
fw_file = open(UPD_DIR+"/"+selected_version+"/"+fw_filename, "rb")
|
||||||
|
expected_hash = bytes.fromhex(selected_hash)
|
||||||
|
file_hash = hashlib.sha256(fw_file.read()).hexdigest()
|
||||||
|
if file_hash == selected_hash:
|
||||||
|
pass
|
||||||
|
else:
|
||||||
|
RNS.log("")
|
||||||
|
RNS.log("Firmware corrupt. Try clearing the local firmware cache with: rnodeconf --clear-cache")
|
||||||
|
exit(96)
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
RNS.log("An error occurred while checking firmware file integrity. The contained exception was:")
|
||||||
|
RNS.log(str(e))
|
||||||
|
exit(95)
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
RNS.log("An error occurred while checking firmware file integrity. The contained exception was:")
|
RNS.log("Could not download required firmware file: ")
|
||||||
|
RNS.log(str(update_target_url))
|
||||||
|
RNS.log("The contained exception was:")
|
||||||
RNS.log(str(e))
|
RNS.log(str(e))
|
||||||
exit(95)
|
exit()
|
||||||
|
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
RNS.log("Could not download required firmware file: ")
|
RNS.log("An error occurred while reading version information for "+str(fw_filename)+". The contained exception was:")
|
||||||
RNS.log(str(update_target_url))
|
|
||||||
RNS.log("The contained exception was:")
|
|
||||||
RNS.log(str(e))
|
RNS.log(str(e))
|
||||||
exit()
|
exit()
|
||||||
|
|
||||||
except Exception as e:
|
|
||||||
RNS.log("An error occurred while reading version information for "+str(fw_filename)+". The contained exception was:")
|
|
||||||
RNS.log(str(e))
|
|
||||||
exit()
|
|
||||||
|
|
||||||
def rnode_open_serial(port):
|
def rnode_open_serial(port):
|
||||||
import serial
|
import serial
|
||||||
return serial.Serial(
|
return serial.Serial(
|
||||||
@ -1114,7 +1146,6 @@ def main():
|
|||||||
port_product = selected_port.product
|
port_product = selected_port.product
|
||||||
port_serialno = selected_port.serial_number
|
port_serialno = selected_port.serial_number
|
||||||
|
|
||||||
clear()
|
|
||||||
print("\nOk, using device on "+str(port_path)+" ("+str(port_product)+", "+str(port_serialno)+")")
|
print("\nOk, using device on "+str(port_path)+" ("+str(port_product)+", "+str(port_serialno)+")")
|
||||||
|
|
||||||
else:
|
else:
|
||||||
@ -1457,6 +1488,7 @@ def main():
|
|||||||
print("\nWhat model is this RNode?\n")
|
print("\nWhat model is this RNode?\n")
|
||||||
print("[1] Handheld v2.x RNode, 410 - 525 MHz")
|
print("[1] Handheld v2.x RNode, 410 - 525 MHz")
|
||||||
print("[2] Handheld v2.x RNode, 820 - 1020 MHz")
|
print("[2] Handheld v2.x RNode, 820 - 1020 MHz")
|
||||||
|
print("")
|
||||||
print("[3] Original v1.x RNode, 410 - 525 MHz")
|
print("[3] Original v1.x RNode, 410 - 525 MHz")
|
||||||
print("[4] Original v1.x RNode, 820 - 1020 MHz")
|
print("[4] Original v1.x RNode, 820 - 1020 MHz")
|
||||||
# print("[5] Prototype v2 RNode, 410 - 525 MHz")
|
# print("[5] Prototype v2 RNode, 410 - 525 MHz")
|
||||||
@ -1618,6 +1650,9 @@ def main():
|
|||||||
print("")
|
print("")
|
||||||
exit()
|
exit()
|
||||||
|
|
||||||
|
if args.use_extracted:
|
||||||
|
fw_filename = "extracted_rnode_firmware.zip"
|
||||||
|
|
||||||
clear()
|
clear()
|
||||||
print("")
|
print("")
|
||||||
print("------------------------------------------------------------------------------")
|
print("------------------------------------------------------------------------------")
|
||||||
@ -2104,6 +2139,24 @@ def main():
|
|||||||
"0x10000", UPD_DIR+"/"+selected_version+"/rnode_firmware_ng21.bin",
|
"0x10000", UPD_DIR+"/"+selected_version+"/rnode_firmware_ng21.bin",
|
||||||
"0x8000", UPD_DIR+"/"+selected_version+"/rnode_firmware_ng21.partitions",
|
"0x8000", UPD_DIR+"/"+selected_version+"/rnode_firmware_ng21.partitions",
|
||||||
]
|
]
|
||||||
|
elif fw_filename == "extracted_rnode_firmware.zip":
|
||||||
|
return [
|
||||||
|
flasher,
|
||||||
|
"--chip", "esp32",
|
||||||
|
"--port", args.port,
|
||||||
|
"--baud", "921600",
|
||||||
|
"--before", "default_reset",
|
||||||
|
"--after", "hard_reset",
|
||||||
|
"write_flash", "-z",
|
||||||
|
"--flash_mode", "dio",
|
||||||
|
"--flash_freq", "80m",
|
||||||
|
"--flash_size", "4MB",
|
||||||
|
"0x1000", EXT_DIR+"/extracted_rnode_firmware.bootloader",
|
||||||
|
"0xe000", EXT_DIR+"/extracted_rnode_firmware.boot_app0",
|
||||||
|
"0x8000", EXT_DIR+"/extracted_rnode_firmware.partitions",
|
||||||
|
"0x10000", EXT_DIR+"/extracted_rnode_firmware.bin",
|
||||||
|
"0x210000",EXT_DIR+"/extracted_console_image.bin",
|
||||||
|
]
|
||||||
else:
|
else:
|
||||||
RNS.log("No flasher available for this board, cannot install firmware.")
|
RNS.log("No flasher available for this board, cannot install firmware.")
|
||||||
else:
|
else:
|
||||||
@ -2132,18 +2185,8 @@ def main():
|
|||||||
RNS.log("Missing parameters, cannot continue")
|
RNS.log("Missing parameters, cannot continue")
|
||||||
exit(68)
|
exit(68)
|
||||||
|
|
||||||
fw_src = UPD_DIR+"/"+selected_version+"/"
|
if fw_filename == "extracted_rnode_firmware.zip":
|
||||||
if os.path.isfile(fw_src+fw_filename):
|
|
||||||
try:
|
try:
|
||||||
if fw_filename.endswith(".zip"):
|
|
||||||
RNS.log("Extracting firmware...")
|
|
||||||
unzip_status = call(get_flasher_call("unzip", fw_filename))
|
|
||||||
if unzip_status == 0:
|
|
||||||
RNS.log("Firmware extracted")
|
|
||||||
else:
|
|
||||||
RNS.log("Could not extract firmware from downloaded zip file")
|
|
||||||
exit()
|
|
||||||
|
|
||||||
RNS.log("Flashing RNode firmware to device on "+args.port)
|
RNS.log("Flashing RNode firmware to device on "+args.port)
|
||||||
from subprocess import call
|
from subprocess import call
|
||||||
rc = get_flasher_call(args.platform, fw_filename)
|
rc = get_flasher_call(args.platform, fw_filename)
|
||||||
@ -2162,9 +2205,41 @@ def main():
|
|||||||
RNS.log("Error while flashing")
|
RNS.log("Error while flashing")
|
||||||
RNS.log(str(e))
|
RNS.log(str(e))
|
||||||
exit(1)
|
exit(1)
|
||||||
|
|
||||||
else:
|
else:
|
||||||
RNS.log("Firmware file not found")
|
fw_src = UPD_DIR+"/"+selected_version+"/"
|
||||||
exit()
|
if os.path.isfile(fw_src+fw_filename):
|
||||||
|
try:
|
||||||
|
if fw_filename.endswith(".zip"):
|
||||||
|
RNS.log("Extracting firmware...")
|
||||||
|
unzip_status = call(get_flasher_call("unzip", fw_filename))
|
||||||
|
if unzip_status == 0:
|
||||||
|
RNS.log("Firmware extracted")
|
||||||
|
else:
|
||||||
|
RNS.log("Could not extract firmware from downloaded zip file")
|
||||||
|
exit()
|
||||||
|
|
||||||
|
RNS.log("Flashing RNode firmware to device on "+args.port)
|
||||||
|
from subprocess import call
|
||||||
|
rc = get_flasher_call(args.platform, fw_filename)
|
||||||
|
flash_status = call(rc)
|
||||||
|
if flash_status == 0:
|
||||||
|
RNS.log("Done flashing")
|
||||||
|
args.rom = True
|
||||||
|
if args.platform == ROM.PLATFORM_ESP32:
|
||||||
|
wants_fw_provision = True
|
||||||
|
RNS.log("Waiting for ESP32 reset...")
|
||||||
|
time.sleep(7)
|
||||||
|
else:
|
||||||
|
exit()
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
RNS.log("Error while flashing")
|
||||||
|
RNS.log(str(e))
|
||||||
|
exit(1)
|
||||||
|
else:
|
||||||
|
RNS.log("Firmware file not found")
|
||||||
|
exit()
|
||||||
|
|
||||||
RNS.log("Opening serial port "+args.port+"...")
|
RNS.log("Opening serial port "+args.port+"...")
|
||||||
try:
|
try:
|
||||||
@ -2203,27 +2278,33 @@ def main():
|
|||||||
if rnode.model != ROM.MODEL_FF:
|
if rnode.model != ROM.MODEL_FF:
|
||||||
fw_filename = models[rnode.model][4]
|
fw_filename = models[rnode.model][4]
|
||||||
else:
|
else:
|
||||||
if rnode.platform == ROM.PLATFORM_AVR:
|
if args.use_extracted:
|
||||||
if rnode.mcu == ROM.MCU_1284P:
|
fw_filename = "extracted_rnode_firmware.zip"
|
||||||
fw_filename = "rnode_firmware.hex"
|
else:
|
||||||
elif rnode.mcu == ROM.MCU_2560:
|
if rnode.platform == ROM.PLATFORM_AVR:
|
||||||
fw_filename = "rnode_firmware_m2560.hex"
|
if rnode.mcu == ROM.MCU_1284P:
|
||||||
elif rnode.platform == ROM.PLATFORM_ESP32:
|
fw_filename = "rnode_firmware.hex"
|
||||||
if rnode.board == ROM.BOARD_HUZZAH32:
|
elif rnode.mcu == ROM.MCU_2560:
|
||||||
fw_filename = "rnode_firmware_featheresp32.zip"
|
fw_filename = "rnode_firmware_m2560.hex"
|
||||||
elif rnode.board == ROM.BOARD_GENERIC_ESP32:
|
elif rnode.platform == ROM.PLATFORM_ESP32:
|
||||||
fw_filename = "rnode_firmware_esp32_generic.zip"
|
if rnode.board == ROM.BOARD_HUZZAH32:
|
||||||
else:
|
fw_filename = "rnode_firmware_featheresp32.zip"
|
||||||
fw_filename = None
|
elif rnode.board == ROM.BOARD_GENERIC_ESP32:
|
||||||
if args.update:
|
fw_filename = "rnode_firmware_esp32_generic.zip"
|
||||||
RNS.log("ERROR: No firmware found for this board. Cannot update.")
|
else:
|
||||||
exit()
|
fw_filename = None
|
||||||
|
if args.update:
|
||||||
|
RNS.log("ERROR: No firmware found for this board. Cannot update.")
|
||||||
|
exit()
|
||||||
|
|
||||||
if args.update:
|
if args.update:
|
||||||
if not rnode.provisioned:
|
if not rnode.provisioned:
|
||||||
RNS.log("Device not provisioned. Cannot update device firmware.")
|
RNS.log("Device not provisioned. Cannot update device firmware.")
|
||||||
exit(1)
|
exit(1)
|
||||||
|
|
||||||
|
if args.use_extracted:
|
||||||
|
fw_filename = "extracted_rnode_firmware.zip"
|
||||||
|
|
||||||
from subprocess import call
|
from subprocess import call
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@ -2257,7 +2338,7 @@ def main():
|
|||||||
if not fw_file_ensured and selected_version != None:
|
if not fw_file_ensured and selected_version != None:
|
||||||
ensure_firmware_file(fw_filename)
|
ensure_firmware_file(fw_filename)
|
||||||
|
|
||||||
if fw_filename.endswith(".zip"):
|
if fw_filename.endswith(".zip") and not fw_filename == "extracted_rnode_firmware.zip":
|
||||||
RNS.log("Extracting firmware...")
|
RNS.log("Extracting firmware...")
|
||||||
unzip_status = call(get_flasher_call("unzip", fw_filename))
|
unzip_status = call(get_flasher_call("unzip", fw_filename))
|
||||||
if unzip_status == 0:
|
if unzip_status == 0:
|
||||||
@ -2271,12 +2352,22 @@ def main():
|
|||||||
RNS.log("The contained exception was: "+str(e))
|
RNS.log("The contained exception was: "+str(e))
|
||||||
exit()
|
exit()
|
||||||
|
|
||||||
if os.path.isfile(UPD_DIR+"/"+selected_version+"/"+fw_filename):
|
if fw_filename == "extracted_rnode_firmware.zip":
|
||||||
|
update_full_path = EXT_DIR+"/extracted_rnode_firmware.version"
|
||||||
|
else:
|
||||||
|
update_full_path = UPD_DIR+"/"+selected_version+"/"+fw_filename
|
||||||
|
if os.path.isfile(update_full_path):
|
||||||
try:
|
try:
|
||||||
args.info = False
|
args.info = False
|
||||||
RNS.log("Updating RNode firmware for device on "+args.port)
|
RNS.log("Updating RNode firmware for device on "+args.port)
|
||||||
partition_filename = fw_filename.replace(".zip", ".bin")
|
if fw_filename == "extracted_rnode_firmware.zip":
|
||||||
partition_hash = get_partition_hash(UPD_DIR+"/"+selected_version+"/"+partition_filename)
|
vf = open(update_full_path, "rb")
|
||||||
|
release_info = vf.read().decode("utf-8").strip()
|
||||||
|
partition_hash = bytes.fromhex(release_info.split()[1])
|
||||||
|
vf.close()
|
||||||
|
else:
|
||||||
|
partition_filename = fw_filename.replace(".zip", ".bin")
|
||||||
|
partition_hash = get_partition_hash(UPD_DIR+"/"+selected_version+"/"+partition_filename)
|
||||||
if partition_hash != None:
|
if partition_hash != None:
|
||||||
rnode.set_firmware_hash(partition_hash)
|
rnode.set_firmware_hash(partition_hash)
|
||||||
rnode.indicate_firmware_update()
|
rnode.indicate_firmware_update()
|
||||||
@ -2582,8 +2673,18 @@ def main():
|
|||||||
RNS.log("EEPROM written! Validating...")
|
RNS.log("EEPROM written! Validating...")
|
||||||
|
|
||||||
if wants_fw_provision:
|
if wants_fw_provision:
|
||||||
partition_filename = fw_filename.replace(".zip", ".bin")
|
partition_hash = None
|
||||||
partition_hash = get_partition_hash(UPD_DIR+"/"+selected_version+"/"+partition_filename)
|
|
||||||
|
if fw_filename == "extracted_rnode_firmware.zip":
|
||||||
|
update_full_path = EXT_DIR+"/extracted_rnode_firmware.version"
|
||||||
|
vf = open(update_full_path, "rb")
|
||||||
|
release_info = vf.read().decode("utf-8").strip()
|
||||||
|
partition_hash = bytes.fromhex(release_info.split()[1])
|
||||||
|
vf.close()
|
||||||
|
else:
|
||||||
|
partition_filename = fw_filename.replace(".zip", ".bin")
|
||||||
|
partition_hash = get_partition_hash(UPD_DIR+"/"+selected_version+"/"+partition_filename)
|
||||||
|
|
||||||
if partition_hash != None:
|
if partition_hash != None:
|
||||||
rnode.set_firmware_hash(partition_hash)
|
rnode.set_firmware_hash(partition_hash)
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user