Files
platform-espressif32/platform.py
T

269 lines
11 KiB
Python
Raw Normal View History

# Copyright 2014-present PlatformIO <contact@platformio.org>
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
2020-10-26 18:44:10 +02:00
import os
2022-04-13 18:49:20 +03:00
import urllib
import sys
import json
import re
import requests
2022-05-27 17:56:49 +03:00
from platformio.public import PlatformBase, to_unix_path
2022-05-27 17:27:53 +03:00
IS_WINDOWS = sys.platform.startswith("win")
2024-09-04 19:08:11 +02:00
# Set Platformio env var to use windows_amd64 for all windows architectures
# only windows_amd64 native espressif toolchains are available
# needs platformio core >= 6.1.16b2 or pioarduino core 6.1.16+test
if IS_WINDOWS:
os.environ["PLATFORMIO_SYSTEM_TYPE"] = "windows_amd64"
class Espressif32Platform(PlatformBase):
def configure_default_packages(self, variables, targets):
2020-05-11 21:17:48 +03:00
if not variables.get("board"):
2022-05-27 17:27:53 +03:00
return super().configure_default_packages(variables, targets)
2020-05-11 21:17:48 +03:00
board_config = self.board_config(variables.get("board"))
2020-09-01 21:26:55 +03:00
mcu = variables.get("board_build.mcu", board_config.get("build.mcu", "esp32"))
2024-12-14 12:42:42 +01:00
board_sdkconfig = variables.get("board_espidf.custom_sdkconfig", board_config.get("espidf.custom_sdkconfig", ""))
core_variant_board = ''.join(variables.get("board_build.extra_flags", board_config.get("build.extra_flags", "")))
core_variant_board = core_variant_board.replace("-D", " ")
core_variant_build = (''.join(variables.get("build_flags", []))).replace("-D", " ")
2020-09-01 21:26:55 +03:00
frameworks = variables.get("pioframework", [])
2021-11-05 13:49:21 +02:00
if "arduino" in frameworks:
self.packages["framework-arduinoespressif32"]["optional"] = False
self.packages["framework-arduinoespressif32-libs"]["optional"] = False
# use latest espressif Arduino libs
2025-02-13 15:02:06 +01:00
URL = "https://raw.githubusercontent.com/espressif/arduino-esp32/master/package/package_esp32_index.template.json"
packjdata = requests.get(URL).json()
dyn_lib_url = packjdata['packages'][0]['tools'][0]['systems'][0]['url']
self.packages["framework-arduinoespressif32-libs"]["version"] = dyn_lib_url
2024-12-14 12:42:42 +01:00
if variables.get("custom_sdkconfig") is not None or len(str(board_sdkconfig)) > 3:
frameworks.append("espidf")
self.packages["framework-espidf"]["optional"] = False
if mcu == "esp32c2":
self.packages["framework-arduino-c2-skeleton-lib"]["optional"] = False
2025-01-15 15:37:33 +01:00
# Enable check tools only when "check_tool" is active
for p in self.packages:
if p in ("tool-cppcheck", "tool-clangtidy", "tool-pvs-studio"):
self.packages[p]["optional"] = False if str(variables.get("check_tool")).strip("['']") in p else True
2018-05-10 23:08:50 +03:00
if "buildfs" in targets:
filesystem = variables.get("board_build.filesystem", "littlefs")
2022-04-17 23:58:10 +03:00
if filesystem == "littlefs":
self.packages["tool-mklittlefs"]["optional"] = False
elif filesystem == "fatfs":
self.packages["tool-mkfatfs"]["optional"] = False
else:
self.packages["tool-mkspiffs"]["optional"] = False
2018-05-10 23:08:50 +03:00
if variables.get("upload_protocol"):
2021-01-21 13:42:23 +02:00
self.packages["tool-openocd-esp32"]["optional"] = False
2020-10-26 18:44:10 +02:00
if os.path.isdir("ulp"):
2021-01-21 13:42:23 +02:00
self.packages["toolchain-esp32ulp"]["optional"] = False
if "downloadfs" in targets:
filesystem = variables.get("board_build.filesystem", "littlefs")
if filesystem == "littlefs":
# Use Tasmota mklittlefs v4.0.0 to unpack, older version is incompatible
self.packages["tool-mklittlefs"]["version"] = "~4.0.0"
2023-08-01 19:38:55 +03:00
# Currently only Arduino Nano ESP32 uses the dfuutil tool as uploader
if variables.get("board") == "arduino_nano_esp32":
self.packages["tool-dfuutil-arduino"]["optional"] = False
else:
del self.packages["tool-dfuutil-arduino"]
# Starting from v12, Espressif's toolchains are shipped without
# bundled GDB. Instead, it's distributed as separate packages for Xtensa
# and RISC-V targets.
for gdb_package in ("tool-xtensa-esp-elf-gdb", "tool-riscv32-esp-elf-gdb"):
self.packages[gdb_package]["optional"] = False
2024-10-25 15:40:06 +02:00
# if IS_WINDOWS:
# Note: On Windows GDB v12 is not able to
# launch a GDB server in pipe mode while v11 works fine
2024-10-25 15:40:06 +02:00
# self.packages[gdb_package]["version"] = "~11.2.0"
# Common packages for IDF and mixed Arduino+IDF projects
2022-04-13 18:49:20 +03:00
if "espidf" in frameworks:
self.packages["toolchain-esp32ulp"]["optional"] = False
2021-11-09 12:12:06 +02:00
for p in self.packages:
2024-12-14 12:42:42 +01:00
if p in ("tool-scons", "tool-cmake", "tool-ninja"):
2020-05-11 21:17:48 +03:00
self.packages[p]["optional"] = False
2024-12-14 12:42:42 +01:00
# elif p in ("tool-mconf", "tool-idf") and IS_WINDOWS:
# self.packages[p]["optional"] = False
2022-04-13 18:49:20 +03:00
2024-07-30 15:24:08 +02:00
if mcu in ("esp32", "esp32s2", "esp32s3"):
self.packages["toolchain-xtensa-esp-elf"]["optional"] = False
else:
self.packages.pop("toolchain-xtensa-esp-elf", None)
2024-07-30 15:24:08 +02:00
if mcu in ("esp32s2", "esp32s3", "esp32c2", "esp32c3", "esp32c6", "esp32h2", "esp32p4"):
if mcu in ("esp32c2", "esp32c3", "esp32c6", "esp32h2", "esp32p4"):
self.packages.pop("toolchain-esp32ulp", None)
# RISC-V based toolchain for ESP32C3, ESP32C6 ESP32S2, ESP32S3 ULP
2022-04-13 18:49:20 +03:00
self.packages["toolchain-riscv32-esp"]["optional"] = False
2022-05-27 17:27:53 +03:00
return super().configure_default_packages(variables, targets)
2018-03-21 23:37:51 +02:00
def get_boards(self, id_=None):
2022-05-27 17:27:53 +03:00
result = super().get_boards(id_)
2018-03-21 23:37:51 +02:00
if not result:
return result
if id_:
2019-05-11 22:13:26 +03:00
return self._add_dynamic_options(result)
2018-03-21 23:37:51 +02:00
else:
for key, value in result.items():
2019-05-11 22:13:26 +03:00
result[key] = self._add_dynamic_options(result[key])
2018-03-21 23:37:51 +02:00
return result
2019-05-11 22:13:26 +03:00
def _add_dynamic_options(self, board):
# upload protocols
if not board.get("upload.protocols", []):
2021-01-21 13:42:23 +02:00
board.manifest["upload"]["protocols"] = ["esptool", "espota"]
2019-05-11 22:13:26 +03:00
if not board.get("upload.protocol", ""):
2021-01-21 13:42:23 +02:00
board.manifest["upload"]["protocol"] = "esptool"
2019-05-11 22:13:26 +03:00
# debug tools
debug = board.manifest.get("debug", {})
2024-02-16 18:02:36 +02:00
non_debug_protocols = ["esptool", "espota"]
supported_debug_tools = [
2022-04-25 13:36:01 +03:00
"cmsis-dap",
"esp-prog",
2022-06-14 14:55:32 +03:00
"esp-bridge",
"iot-bus-jtag",
"jlink",
"minimodule",
"olimex-arm-usb-tiny-h",
"olimex-arm-usb-ocd-h",
"olimex-arm-usb-ocd",
"olimex-jtag-tiny",
2021-01-21 13:42:23 +02:00
"tumpa",
]
# A special case for the Kaluga board that has a separate interface config
if board.id == "esp32-s2-kaluga-1":
supported_debug_tools.append("ftdi")
if board.get("build.mcu", "") in ("esp32c3", "esp32c6", "esp32s3", "esp32h2"):
2022-06-14 14:55:32 +03:00
supported_debug_tools.append("esp-builtin")
upload_protocol = board.manifest.get("upload", {}).get("protocol")
2021-01-21 13:42:23 +02:00
upload_protocols = board.manifest.get("upload", {}).get("protocols", [])
if debug:
upload_protocols.extend(supported_debug_tools)
if upload_protocol and upload_protocol not in upload_protocols:
upload_protocols.append(upload_protocol)
2021-01-21 13:42:23 +02:00
board.manifest["upload"]["protocols"] = upload_protocols
if "tools" not in debug:
2021-01-21 13:42:23 +02:00
debug["tools"] = {}
2018-10-26 01:37:16 +03:00
for link in upload_protocols:
2021-01-21 13:42:23 +02:00
if link in non_debug_protocols or link in debug["tools"]:
2018-03-21 23:37:51 +02:00
continue
if link in ("jlink", "cmsis-dap"):
2022-04-25 13:36:01 +03:00
openocd_interface = link
2018-10-19 20:06:47 +03:00
elif link in ("esp-prog", "ftdi"):
if board.id == "esp32-s2-kaluga-1":
openocd_interface = "ftdi/esp32s2_kaluga_v1"
else:
openocd_interface = "ftdi/esp32_devkitj_v1"
2022-06-14 14:55:32 +03:00
elif link == "esp-bridge":
openocd_interface = "esp_usb_bridge"
elif link == "esp-builtin":
openocd_interface = "esp_usb_jtag"
2018-10-19 20:06:47 +03:00
else:
openocd_interface = "ftdi/" + link
2018-05-09 19:38:25 +03:00
2018-03-21 23:37:51 +02:00
server_args = [
2021-01-21 13:42:23 +02:00
"-s",
"$PACKAGE_DIR/share/openocd/scripts",
"-f",
"interface/%s.cfg" % openocd_interface,
"-f",
"%s/%s"
% (
("target", debug.get("openocd_target"))
if "openocd_target" in debug
else ("board", debug.get("openocd_board"))
),
2018-03-21 23:37:51 +02:00
]
2021-01-21 13:42:23 +02:00
debug["tools"][link] = {
2018-03-21 23:37:51 +02:00
"server": {
"package": "tool-openocd-esp32",
"executable": "bin/openocd",
2021-01-21 13:42:23 +02:00
"arguments": server_args,
2018-03-22 00:12:15 +02:00
},
"init_break": "thb app_main",
2018-03-22 00:12:15 +02:00
"init_cmds": [
"define pio_reset_halt_target",
" monitor reset halt",
2018-06-28 20:18:47 +03:00
" flushregs",
"end",
"define pio_reset_run_target",
" monitor reset",
"end",
2018-03-22 00:12:15 +02:00
"target extended-remote $DEBUG_PORT",
"$LOAD_CMDS",
"pio_reset_halt_target",
2021-01-21 13:42:23 +02:00
"$INIT_BREAK",
2018-03-22 00:12:15 +02:00
],
"onboard": link in debug.get("onboard_tools", []),
2021-01-21 13:42:23 +02:00
"default": link == debug.get("default_tool"),
2018-03-21 23:37:51 +02:00
}
2023-08-01 19:38:55 +03:00
# Avoid erasing Arduino Nano bootloader by preloading app binary
if board.id == "arduino_nano_esp32":
debug["tools"][link]["load_cmds"] = "preload"
2021-01-21 13:42:23 +02:00
board.manifest["debug"] = debug
2018-03-21 23:37:51 +02:00
return board
2020-10-26 18:44:10 +02:00
def configure_debug_session(self, debug_config):
build_extra_data = debug_config.build_data.get("extra", {})
flash_images = build_extra_data.get("flash_images", [])
if "openocd" in (debug_config.server or {}).get("executable", ""):
debug_config.server["arguments"].extend(
2022-09-27 12:30:15 +03:00
["-c", "adapter speed %s" % (debug_config.speed or "5000")]
)
ignore_conds = [
debug_config.load_cmds != ["load"],
not flash_images,
not all([os.path.isfile(item["path"]) for item in flash_images]),
]
if any(ignore_conds):
return
load_cmds = [
'monitor program_esp "{{{path}}}" {offset} verify'.format(
path=to_unix_path(item["path"]), offset=item["offset"]
)
for item in flash_images
]
load_cmds.append(
'monitor program_esp "{%s.bin}" %s verify'
% (
to_unix_path(debug_config.build_data["prog_path"][:-4]),
build_extra_data.get("application_offset", "0x10000"),
)
)
debug_config.load_cmds = load_cmds