Skip to content

Commit

Permalink
Merge branch 'main' into merge
Browse files Browse the repository at this point in the history
  • Loading branch information
Jason2866 authored Dec 13, 2024
2 parents ae315e3 + a6bceb7 commit a63da8d
Show file tree
Hide file tree
Showing 26 changed files with 258 additions and 373 deletions.
28 changes: 21 additions & 7 deletions esptool/bin_image.py
Original file line number Diff line number Diff line change
Expand Up @@ -270,13 +270,24 @@ def maybe_patch_segment_data(self, f, segment_data):
)
return segment_data

def save_segment(self, f, segment, checksum=None):
def save_segment(self, f, segment, checksum=None, segment_name=None):
"""
Save the next segment to the image file,
return next checksum value if provided
"""
segment_data = self.maybe_patch_segment_data(f, segment.data)
f.write(struct.pack("<II", segment.addr, len(segment_data)))
segment_len = len(segment_data)
segment_name = segment_name if segment_name is not None else ""
if segment_len & 3:
raise FatalError(
f"Invalid {segment_name} segment length {segment_len:#x}. It has to be multiple of 4."
)
SIXTEEN_MB = 0x1000000
if segment_len >= SIXTEEN_MB:
raise FatalError(
f"Invalid {segment_name} segment length {segment_len:#x}. The 16 MB limit has been exceeded."
)
f.write(struct.pack("<II", segment.addr, segment_len))
f.write(segment_data)
if checksum is not None:
return ESPLoader.checksum(segment_data, checksum)
Expand All @@ -293,7 +304,8 @@ def save_flash_segment(self, f, segment, checksum=None):
segment_len_remainder = segment_end_pos % self.IROM_ALIGN
if segment_len_remainder < 0x24:
segment.data += b"\x00" * (0x24 - segment_len_remainder)
return self.save_segment(f, segment, checksum)
segment_name = getattr(segment, "name", None)
return self.save_segment(f, segment, checksum, segment_name)

def read_checksum(self, f):
"""Return ESPLoader checksum from end of just-read image"""
Expand Down Expand Up @@ -748,7 +760,7 @@ def get_alignment_data_needed(segment):
# and checksum (ROM bootloader will only care for RAM segments and its
# correct checksums)
for segment in ram_segments:
checksum = self.save_segment(f, segment, checksum)
checksum = self.save_segment(f, segment, checksum, segment.name)
total_segments += 1
self.append_checksum(f, checksum)

Expand All @@ -769,7 +781,7 @@ def get_alignment_data_needed(segment):

pad_len -= self.ROM_LOADER.BOOTLOADER_FLASH_OFFSET
pad_segment = ImageSegment(0, b"\x00" * pad_len, f.tell())
self.save_segment(f, pad_segment)
self.save_segment(f, pad_segment, None, segment.name)
total_segments += 1
# check the alignment
assert (f.tell() + 8 + self.ROM_LOADER.BOOTLOADER_FLASH_OFFSET) % (
Expand All @@ -793,7 +805,9 @@ def get_alignment_data_needed(segment):
ram_segments.pop(0)
else:
pad_segment = ImageSegment(0, b"\x00" * pad_len, f.tell())
checksum = self.save_segment(f, pad_segment, checksum)
checksum = self.save_segment(
f, pad_segment, checksum, segment.name
)
total_segments += 1
else:
# write the flash segment
Expand All @@ -806,7 +820,7 @@ def get_alignment_data_needed(segment):

# flash segments all written, so write any remaining RAM segments
for segment in ram_segments:
checksum = self.save_segment(f, segment, checksum)
checksum = self.save_segment(f, segment, checksum, segment.name)
total_segments += 1

if self.secure_pad:
Expand Down
27 changes: 20 additions & 7 deletions esptool/cmds.py
Original file line number Diff line number Diff line change
Expand Up @@ -108,10 +108,10 @@ def check_if_stub(instance):
try:
print("Detecting chip type...", end="")
chip_id = detect_port.get_chip_id()
for cls in [
n for n in ROM_LIST if n.CHIP_NAME not in ("ESP8266", "ESP32", "ESP32-S2")
]:
for cls in ROM_LIST:
# cmd not supported on ESP8266 and ESP32 + ESP32-S2 doesn't return chip_id
if cls.USES_MAGIC_VALUE:
continue
if chip_id == cls.IMAGE_CHIP_ID:
inst = cls(detect_port._port, baud, trace_enabled=trace_enabled)
try:
Expand Down Expand Up @@ -144,11 +144,12 @@ def check_if_stub(instance):
)

for cls in ROM_LIST:
if chip_magic_value in cls.CHIP_DETECT_MAGIC_VALUE:
if not cls.USES_MAGIC_VALUE:
continue
if chip_magic_value == cls.MAGIC_VALUE:
inst = cls(detect_port._port, baud, trace_enabled=trace_enabled)
inst = check_if_stub(inst)
inst._post_connect()
inst.check_chip_id()
break
else:
err_msg = f"Unexpected chip magic value {chip_magic_value:#010x}."
Expand Down Expand Up @@ -875,7 +876,8 @@ def get_key_from_value(dict, val):
for idx, seg in enumerate(image.segments):
segs = seg.get_memory_type(image)
seg_name = ", ".join(segs)
if "DROM" in segs: # The DROM segment starts with the esp_app_desc_t struct
# The DROM segment starts with the esp_app_desc_t struct
if "DROM" in segs and app_desc is None:
app_desc = seg.data[:256]
elif "DRAM" in segs:
# The DRAM segment starts with the esp_bootloader_desc_t struct
Expand Down Expand Up @@ -914,7 +916,7 @@ def get_key_from_value(dict, val):
pass # ESP8266 image has no append_digest field

if app_desc:
APP_DESC_STRUCT_FMT = "<II" + "8s" + "32s32s16s16s32s32s" + "80s"
APP_DESC_STRUCT_FMT = "<II" + "8s" + "32s32s16s16s32s32sHHB" + "3s" + "72s"
(
magic_word,
secure_version,
Expand All @@ -925,6 +927,10 @@ def get_key_from_value(dict, val):
date,
idf_ver,
app_elf_sha256,
min_efuse_blk_rev_full,
max_efuse_blk_rev_full,
mmu_page_size,
reserv3,
reserv2,
) = struct.unpack(APP_DESC_STRUCT_FMT, app_desc)

Expand All @@ -938,6 +944,13 @@ def get_key_from_value(dict, val):
print(f'Compile time: {date.decode("utf-8")} {time.decode("utf-8")}')
print(f"ELF file SHA256: {hexify(app_elf_sha256, uppercase=False)}")
print(f'ESP-IDF: {idf_ver.decode("utf-8")}')
print(
f"Minimal eFuse block revision: {min_efuse_blk_rev_full // 100}.{min_efuse_blk_rev_full % 100}"
)
print(
f"Maximal eFuse block revision: {max_efuse_blk_rev_full // 100}.{max_efuse_blk_rev_full % 100}"
)
print(f"MMU page size: {2 ** mmu_page_size // 1024} KB")
print(f"Secure version: {secure_version}")

elif bootloader_desc:
Expand Down
118 changes: 65 additions & 53 deletions esptool/loader.py
Original file line number Diff line number Diff line change
Expand Up @@ -143,13 +143,6 @@ def stub_and_esp32_function_only(func):
)


def esp32s3_or_newer_function_only(func):
"""Attribute for a function only supported by ESP32S3 and later chips ROM"""
return check_supported_function(
func, lambda o: o.CHIP_NAME not in ["ESP8266", "ESP32", "ESP32-S2"]
)


class StubFlasher:
STUB_DIR = os.path.join(os.path.dirname(__file__), "targets", "stub_flasher")
# directories will be searched in the order of STUB_SUBDIRS
Expand Down Expand Up @@ -302,6 +295,9 @@ class ESPLoader(object):
# Number of attempts to write flash data
WRITE_FLASH_ATTEMPTS = 2

# Chip uses magic number for chip type autodetection
USES_MAGIC_VALUE = True

def __init__(self, port=DEFAULT_PORT, baud=ESP_ROM_BAUD, trace_enabled=False):
"""Base constructor for ESPLoader bootloader interaction
Expand Down Expand Up @@ -752,41 +748,75 @@ def connect(
)

if not detecting:
try:
from .targets import ROM_LIST
from .targets import ROM_LIST

# check the date code registers match what we expect to see
# Perform a dummy read_reg to check if the chip is in secure download mode
try:
chip_magic_value = self.read_reg(ESPLoader.CHIP_DETECT_MAGIC_REG_ADDR)
if chip_magic_value not in self.CHIP_DETECT_MAGIC_VALUE:
actually = None
for cls in ROM_LIST:
if chip_magic_value in cls.CHIP_DETECT_MAGIC_VALUE:
actually = cls
break
if warnings and actually is None:
print(
"WARNING: This chip doesn't appear to be a %s "
"(chip magic value 0x%08x). "
"Probably it is unsupported by this version of esptool."
% (self.CHIP_NAME, chip_magic_value)
)
else:
raise FatalError(
"This chip is %s not %s. Wrong --chip argument?"
% (actually.CHIP_NAME, self.CHIP_NAME)
)
except UnsupportedCommandError:
self.secure_download_mode = True

# Check if chip supports reading chip ID from the get_security_info command
try:
self.check_chip_id()
except UnsupportedCommandError:
# Fix for ROM not responding in SDM, reconnect and try again
if self.secure_download_mode:
self._connect_attempt(mode, reset_sequence[0])
self.check_chip_id()
chip_id = self.get_chip_id()
except (UnsupportedCommandError, struct.error, FatalError):
chip_id = None

detected = None
chip_arg_wrong = False

# If we can read chip ID (ESP32-S3 and later), verify the ID
if chip_id and (self.USES_MAGIC_VALUE or chip_id != self.IMAGE_CHIP_ID):
chip_arg_wrong = True
for cls in ROM_LIST:
if not cls.USES_MAGIC_VALUE and chip_id == cls.IMAGE_CHIP_ID:
detected = cls
break
# If we can't read chip ID (ESP8266, ESP32, ESP32-S2),
# try to verify the chip by magic value
elif (
not chip_id
and not self.secure_download_mode
and (not self.USES_MAGIC_VALUE or chip_magic_value != self.MAGIC_VALUE)
):
chip_arg_wrong = True
for cls in ROM_LIST:
if cls.USES_MAGIC_VALUE and chip_magic_value == cls.MAGIC_VALUE:
detected = cls
break
# If we can't read chip ID and the chip is in SDM (ESP32 or ESP32-S2),
# we can't verify
elif not chip_id and self.secure_download_mode:
if self.CHIP_NAME not in ["ESP32", "ESP32-S2"]:
chip_arg_wrong = True
detected = "ESP32 or ESP32-S2"
else:
raise
print(
f"WARNING: Can't verify this chip is {self.CHIP_NAME} "
"because of active Secure Download Mode. "
"Please check it manually."
)

if chip_arg_wrong:
if warnings and detected is None:
specifier = (
f"(read chip ID {chip_id})"
if chip_id
else f"(read chip magic value {chip_magic_value:#08x})"
)
print(
f"WARNING: This chip doesn't appear to be an {self.CHIP_NAME} "
f"{specifier}. Probably it is unsupported by this version "
"of esptool. Will attempt to continue anyway."
)
else:
chip_type = (
detected if isinstance(detected, str) else detected.CHIP_NAME
)
raise FatalError(
f"This chip is {chip_type}, not {self.CHIP_NAME}. "
"Wrong --chip argument?"
)
self._post_connect()

def _post_connect(self):
Expand Down Expand Up @@ -1002,7 +1032,6 @@ def get_security_info(self):
"api_version": None if esp32s2 else res[10],
}

@esp32s3_or_newer_function_only
def get_chip_id(self):
if self.cache["chip_id"] is None:
res = self.check_command(
Expand Down Expand Up @@ -1552,23 +1581,6 @@ def soft_reset(self, stay_in_bootloader):
# in the stub loader
self.command(self.ESP_RUN_USER_CODE, wait_response=False)

def check_chip_id(self):
try:
chip_id = self.get_chip_id()
if chip_id != self.IMAGE_CHIP_ID:
print(
"WARNING: Chip ID {} ({}) doesn't match expected Chip ID {}. "
"esptool may not work correctly.".format(
chip_id,
self.UNSUPPORTED_CHIPS.get(chip_id, "Unknown"),
self.IMAGE_CHIP_ID,
)
)
# Try to flash anyways by disabling stub
self.stub_is_disabled = True
except NotImplementedInROMError:
pass


def slip_reader(port, trace_function):
"""Generator to read SLIP packets from a serial port.
Expand Down
2 changes: 1 addition & 1 deletion esptool/targets/esp32.py
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ class ESP32ROM(ESPLoader):
IMAGE_CHIP_ID = 0
IS_STUB = False

CHIP_DETECT_MAGIC_VALUE = [0x00F01D83]
MAGIC_VALUE = 0x00F01D83

IROM_MAP_START = 0x400D0000
IROM_MAP_END = 0x40400000
Expand Down
3 changes: 0 additions & 3 deletions esptool/targets/esp32c2.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,9 +21,6 @@ class ESP32C2ROM(ESP32C3ROM):
DROM_MAP_START = 0x3C000000
DROM_MAP_END = 0x3C400000

# Magic value for ESP32C2 ECO0 , ECO1 and ECO4 respectively
CHIP_DETECT_MAGIC_VALUE = [0x6F51306F, 0x7C41A06F, 0x0C21E06F]

EFUSE_BASE = 0x60008800
EFUSE_BLOCK2_ADDR = EFUSE_BASE + 0x040
MAC_EFUSE_REG = EFUSE_BASE + 0x040
Expand Down
5 changes: 2 additions & 3 deletions esptool/targets/esp32c3.py
Original file line number Diff line number Diff line change
Expand Up @@ -30,10 +30,9 @@ class ESP32C3ROM(ESP32ROM):

SPI_ADDR_REG_MSB = False

BOOTLOADER_FLASH_OFFSET = 0x0
USES_MAGIC_VALUE = False

# Magic values for ESP32-C3 eco 1+2, eco 3, eco 6, and eco 7 respectively
CHIP_DETECT_MAGIC_VALUE = [0x6921506F, 0x1B31506F, 0x4881606F, 0x4361606F]
BOOTLOADER_FLASH_OFFSET = 0x0

UART_DATE_REG_ADDR = 0x60000000 + 0x7C

Expand Down
13 changes: 5 additions & 8 deletions esptool/targets/esp32c5.py
Original file line number Diff line number Diff line change
Expand Up @@ -44,19 +44,16 @@ class ESP32C5ROM(ESP32C6ROM):
EFUSE_SECURE_BOOT_EN_MASK = 1 << 20

IROM_MAP_START = 0x42000000
IROM_MAP_END = 0x42800000
DROM_MAP_START = 0x42800000
DROM_MAP_END = 0x43000000
IROM_MAP_END = 0x44000000
DROM_MAP_START = 0x42000000
DROM_MAP_END = 0x44000000

PCR_SYSCLK_CONF_REG = 0x60096110
PCR_SYSCLK_XTAL_FREQ_V = 0x7F << 24
PCR_SYSCLK_XTAL_FREQ_S = 24

UARTDEV_BUF_NO = 0x4085F51C # Variable in ROM .bss which indicates the port in use

# Magic values for ESP32C5 ECO0 and ECO1, respectively
CHIP_DETECT_MAGIC_VALUE = [0x1101406F, 0x63E1406F]

FLASH_FREQUENCY = {
"80m": 0xF,
"40m": 0x0,
Expand All @@ -65,12 +62,12 @@ class ESP32C5ROM(ESP32C6ROM):

MEMORY_MAP = [
[0x00000000, 0x00010000, "PADDING"],
[0x42800000, 0x43000000, "DROM"],
[0x42000000, 0x44000000, "DROM"],
[0x40800000, 0x40860000, "DRAM"],
[0x40800000, 0x40860000, "BYTE_ACCESSIBLE"],
[0x4003A000, 0x40040000, "DROM_MASK"],
[0x40000000, 0x4003A000, "IROM_MASK"],
[0x42000000, 0x42800000, "IROM"],
[0x42000000, 0x44000000, "IROM"],
[0x40800000, 0x40860000, "IRAM"],
[0x50000000, 0x50004000, "RTC_IRAM"],
[0x50000000, 0x50004000, "RTC_DRAM"],
Expand Down
3 changes: 0 additions & 3 deletions esptool/targets/esp32c5beta3.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,9 +19,6 @@ class ESP32C5BETA3ROM(ESP32C6ROM):
DROM_MAP_START = 0x41000000
DROM_MAP_END = 0x41800000

# Magic value for ESP32C5(beta3)
CHIP_DETECT_MAGIC_VALUE = [0xE10D8082]

FLASH_FREQUENCY = {
"80m": 0xF,
"40m": 0x0,
Expand Down
Loading

0 comments on commit a63da8d

Please sign in to comment.