2020-04-10 17:24:08 +01:00
/*
2021-02-08 10:34:29 +00:00
support_esp . ino - ESP specific code for Tasmota
2020-04-10 17:24:08 +01:00
2021-01-01 12:44:04 +00:00
Copyright ( C ) 2021 Theo Arends / Jörg Schüler - Maroldt
2020-04-10 17:24:08 +01:00
This program 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 .
This program 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 this program . If not , see < http : //www.gnu.org/licenses/>.
*/
2021-02-08 10:34:29 +00:00
/*********************************************************************************************\
* ESP8266 and ESP32 specific code
*
* At the end the common Tasmota calls are provided
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2020-04-19 15:58:13 +01:00
/*********************************************************************************************\
* ESP8266 Support
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# ifdef ESP8266
extern " C " {
extern struct rst_info resetInfo ;
}
2022-12-29 13:46:34 +00:00
/*********************************************************************************************\
2022-12-29 15:10:41 +00:00
* Core overrides executed by core
2022-12-29 13:46:34 +00:00
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
// Add below line to tasmota_globals.h
// extern "C" void resetPins();
2022-12-29 15:10:41 +00:00
//
// This function overrules __resetPins() which is executed by core init() as initPins() in core_esp8266_wiring.cpp
//
// 20221229 - (v12.3.1.2) Enabled with additional check to execute on power on only to fix relay clicks on power on
// 20200321 - (v8.2.0.1) Disable core functionality to fix relay clicks on restart after OTA - make function return without setting pinMode
2022-12-29 13:46:34 +00:00
void resetPins ( ) {
if ( ( resetInfo . reason = = REASON_DEFAULT_RST ) | | ( resetInfo . reason = = REASON_EXT_SYS_RST ) ) {
// Only perform at power on
for ( int i = 0 ; i < = 5 ; + + i ) {
pinMode ( i , INPUT ) ;
}
2022-12-29 15:10:41 +00:00
// pins 6-11 are used for the SPI flash interface ESP8266
2022-12-29 13:46:34 +00:00
for ( int i = 12 ; i < = 16 ; + + i ) {
pinMode ( i , INPUT ) ;
}
}
}
/*********************************************************************************************\
* Hardware related
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
void HwWdtDisable ( void ) {
* ( ( volatile uint32_t * ) 0x60000900 ) & = ~ ( 1 ) ; // Hardware WDT OFF
}
void HwWdtEnable ( void ) {
* ( ( volatile uint32_t * ) 0x60000900 ) | = 1 ; // Hardware WDT ON
}
void WdtDisable ( void ) {
ESP . wdtDisable ( ) ;
HwWdtDisable ( ) ;
}
void WdtEnable ( void ) {
HwWdtEnable ( ) ;
ESP . wdtEnable ( 0 ) ;
}
/*********************************************************************************************\
* ESP8266 specifics
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2020-04-22 15:07:52 +01:00
uint32_t ESP_ResetInfoReason ( void ) {
2020-04-19 15:58:13 +01:00
return resetInfo . reason ;
}
2020-04-22 15:07:52 +01:00
String ESP_getResetReason ( void ) {
2020-04-19 15:58:13 +01:00
return ESP . getResetReason ( ) ;
}
2020-04-22 15:07:52 +01:00
uint32_t ESP_getChipId ( void ) {
2020-04-19 15:58:13 +01:00
return ESP . getChipId ( ) ;
}
2022-05-05 16:52:35 +01:00
uint32_t ESP_getFreeSketchSpace ( void ) {
return ESP . getFreeSketchSpace ( ) ;
}
2020-04-22 15:07:52 +01:00
uint32_t ESP_getSketchSize ( void ) {
2020-04-19 15:58:13 +01:00
return ESP . getSketchSize ( ) ;
}
2020-04-22 15:07:52 +01:00
uint32_t ESP_getFreeHeap ( void ) {
return ESP . getFreeHeap ( ) ;
}
2022-05-09 21:43:27 +01:00
uint32_t ESP_getFlashChipId ( void ) {
return ESP . getFlashChipId ( ) ;
}
uint32_t ESP_getFlashChipRealSize ( void ) {
return ESP . getFlashChipRealSize ( ) ;
}
2022-09-17 14:53:58 +01:00
uint32_t ESP_getFlashChipSize ( void ) {
return ESP . getFlashChipSize ( ) ;
}
2020-04-22 15:07:52 +01:00
void ESP_Restart ( void ) {
2020-04-19 15:58:13 +01:00
// ESP.restart(); // This results in exception 3 on restarts on core 2.3.0
ESP . reset ( ) ;
}
2020-11-22 16:35:04 +00:00
uint32_t FlashWriteStartSector ( void ) {
return ( ESP . getSketchSize ( ) / SPI_FLASH_SEC_SIZE ) + 2 ; // Stay on the safe side
}
uint32_t FlashWriteMaxSector ( void ) {
2021-01-11 16:56:18 +00:00
return ( ( ( uint32_t ) & _FS_start - 0x40200000 ) / SPI_FLASH_SEC_SIZE ) - 2 ;
2020-11-22 16:35:04 +00:00
}
uint8_t * FlashDirectAccess ( void ) {
return ( uint8_t * ) ( 0x40200000 + ( FlashWriteStartSector ( ) * SPI_FLASH_SEC_SIZE ) ) ;
}
2021-01-04 15:39:00 +00:00
void * special_malloc ( uint32_t size ) {
return malloc ( size ) ;
}
2021-03-20 17:44:35 +00:00
void * special_realloc ( void * ptr , size_t size ) {
return realloc ( ptr , size ) ;
}
2021-05-09 17:15:15 +01:00
void * special_calloc ( size_t num , size_t size ) {
return calloc ( num , size ) ;
}
2022-11-07 14:00:47 +00:00
String GetDeviceHardware ( void ) {
/*
ESP8266 SoCs
- 32 - bit MCU & 2.4 GHz Wi - Fi
- High - performance 160 MHz single - core CPU
- + 19.5 dBm output power ensures a good physical range
- Sleep current is less than 20 μ A , making it suitable for battery - powered and wearable - electronics applications
- Peripherals include UART , GPIO , I2C , I2S , SDIO , PWM , ADC and SPI
*/
// esptool.py get_efuses
uint32_t efuse0 = * ( uint32_t * ) ( 0x3FF00050 ) ;
// uint32_t efuse1 = *(uint32_t*)(0x3FF00054);
uint32_t efuse2 = * ( uint32_t * ) ( 0x3FF00058 ) ;
uint32_t efuse3 = * ( uint32_t * ) ( 0x3FF0005C ) ;
bool r0_4 = efuse0 & ( 1 < < 4 ) ; // ESP8285
bool r2_16 = efuse2 & ( 1 < < 16 ) ; // ESP8285
if ( r0_4 | | r2_16 ) { // ESP8285
2022-11-11 08:30:31 +00:00
// 1M 2M 2M 4M flash size
// r0_4 1 1 0 0
bool r3_25 = efuse3 & ( 1 < < 25 ) ; // flash matrix 0 0 1 1
bool r3_26 = efuse3 & ( 1 < < 26 ) ; // flash matrix 0 1 0 1
bool r3_27 = efuse3 & ( 1 < < 27 ) ; // flash matrix 0 0 0 0
2022-11-07 14:00:47 +00:00
uint32_t pkg_version = 0 ;
if ( ! r3_27 ) {
if ( r0_4 & & ! r3_25 ) {
pkg_version = ( r3_26 ) ? 2 : 1 ;
}
else if ( ! r0_4 & & r3_25 ) {
pkg_version = ( r3_26 ) ? 4 : 2 ;
}
}
bool max_temp = efuse0 & ( 1 < < 5 ) ; // Max flash temperature (0 = 85C, 1 = 105C)
switch ( pkg_version ) {
case 1 :
if ( max_temp ) { return F ( " ESP8285H08 " ) ; } // 1M flash
else { return F ( " ESP8285N08 " ) ; }
case 2 :
if ( max_temp ) { return F ( " ESP8285H16 " ) ; } // 2M flash
else { return F ( " ESP8285N16 " ) ; }
case 4 :
if ( max_temp ) { return F ( " ESP8285H32 " ) ; } // 4M flash
else { return F ( " ESP8285N32 " ) ; }
}
return F ( " ESP8285 " ) ;
}
return F ( " ESP8266EX " ) ;
}
2022-04-23 16:59:37 +01:00
String GetDeviceHardwareRevision ( void ) {
2022-04-24 10:24:01 +01:00
// No known revisions for ESP8266/85
return GetDeviceHardware ( ) ;
2022-04-23 16:59:37 +01:00
}
2022-10-04 16:52:15 +01:00
String GetCodeCores ( void ) {
return F ( " " ) ;
}
2020-04-19 15:58:13 +01:00
# endif
/*********************************************************************************************\
* ESP32 Support
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2020-04-10 17:24:08 +01:00
# ifdef ESP32
2022-08-21 19:22:39 +01:00
# include "soc/soc.h"
# include "soc/spi_reg.h"
2021-11-10 19:58:06 +00:00
// ESP32_ARCH contains the name of the architecture (used by autoconf)
# if CONFIG_IDF_TARGET_ESP32
2022-07-01 23:13:27 +01:00
# ifdef CORE32SOLO1
# define ESP32_ARCH "esp32solo1"
# else
# define ESP32_ARCH "esp32"
# endif
2021-11-10 19:58:06 +00:00
# elif CONFIG_IDF_TARGET_ESP32S2
# define ESP32_ARCH "esp32s2"
# elif CONFIG_IDF_TARGET_ESP32S3
# define ESP32_ARCH "esp32s3"
2023-08-21 15:00:20 +01:00
# elif CONFIG_IDF_TARGET_ESP32C2
# define ESP32_ARCH "esp32c2"
2021-11-10 19:58:06 +00:00
# elif CONFIG_IDF_TARGET_ESP32C3
# define ESP32_ARCH "esp32c3"
2023-08-21 15:00:20 +01:00
# elif CONFIG_IDF_TARGET_ESP32C6
# define ESP32_ARCH "esp32c6"
2021-11-10 19:58:06 +00:00
# else
# define ESP32_ARCH ""
# endif
2020-07-05 13:51:55 +01:00
// Handle 20k of NVM
2020-04-10 17:24:08 +01:00
# include <nvs.h>
2021-02-02 16:55:45 +00:00
2021-02-03 11:22:17 +00:00
// See libraries\ESP32\examples\ResetReason.ino
# if ESP_IDF_VERSION_MAJOR > 3 // IDF 4+
2023-08-05 14:11:10 +01:00
# include "esp_chip_info.h"
2021-02-03 11:22:17 +00:00
# if CONFIG_IDF_TARGET_ESP32 // ESP32/PICO-D4
# include "esp32/rom/rtc.h"
# elif CONFIG_IDF_TARGET_ESP32S2 // ESP32-S2
# include "esp32s2/rom/rtc.h"
2022-01-30 11:55:03 +00:00
# elif CONFIG_IDF_TARGET_ESP32S3 // ESP32-S3
# include "esp32s3/rom/rtc.h"
2023-08-21 15:00:20 +01:00
# elif CONFIG_IDF_TARGET_ESP32C2 // ESP32-C2
# include "esp32c2/rom/rtc.h"
2021-04-17 18:57:13 +01:00
# elif CONFIG_IDF_TARGET_ESP32C3 // ESP32-C3
# include "esp32c3/rom/rtc.h"
2023-08-21 15:00:20 +01:00
# elif CONFIG_IDF_TARGET_ESP32C6 // ESP32-C6
# include "esp32c6/rom/rtc.h"
2021-02-03 11:22:17 +00:00
# else
# error Target CONFIG_IDF_TARGET is not supported
# endif
# else // ESP32 Before IDF 4.0
# include "rom/rtc.h"
2021-02-02 16:55:45 +00:00
# endif
2023-08-05 19:34:24 +01:00
# if ESP_IDF_VERSION_MAJOR >= 5
# include "esp_chip_info.h"
# endif
2022-01-03 17:52:30 +00:00
// Set the Stacksize for Arduino core. Default is 8192, some builds may need a bigger one
size_t getArduinoLoopTaskStackSize ( void ) {
return SET_ESP32_STACK_SIZE ;
}
2020-11-12 16:50:34 +00:00
# include <esp_phy_init.h>
2020-04-10 17:24:08 +01:00
2021-12-12 17:04:46 +00:00
bool NvmLoad ( const char * sNvsName , const char * sName , void * pSettings , unsigned nSettingsLen ) {
nvs_handle_t handle ;
esp_err_t result = nvs_open ( sNvsName , NVS_READONLY , & handle ) ;
if ( result ! = ESP_OK ) {
AddLog ( LOG_LEVEL_DEBUG , PSTR ( " NVS: Error %d " ) , result ) ;
return false ;
}
2020-04-19 14:36:04 +01:00
size_t size = nSettingsLen ;
nvs_get_blob ( handle , sName , pSettings , & size ) ;
2020-04-10 17:24:08 +01:00
nvs_close ( handle ) ;
2021-12-12 17:04:46 +00:00
return true ;
2020-04-10 17:24:08 +01:00
}
2020-04-22 15:07:52 +01:00
void NvmSave ( const char * sNvsName , const char * sName , const void * pSettings , unsigned nSettingsLen ) {
2021-12-17 17:01:27 +00:00
# ifdef USE_WEBCAM
WcInterrupt ( 0 ) ; // Stop stream if active to fix TG1WDT_SYS_RESET
# endif
2021-12-12 17:04:46 +00:00
nvs_handle_t handle ;
esp_err_t result = nvs_open ( sNvsName , NVS_READWRITE , & handle ) ;
if ( result ! = ESP_OK ) {
AddLog ( LOG_LEVEL_DEBUG , PSTR ( " NVS: Error %d " ) , result ) ;
2021-12-17 17:01:27 +00:00
} else {
nvs_set_blob ( handle , sName , pSettings , nSettingsLen ) ;
nvs_commit ( handle ) ;
nvs_close ( handle ) ;
2021-12-12 17:04:46 +00:00
}
2021-12-17 17:01:27 +00:00
# ifdef USE_WEBCAM
WcInterrupt ( 1 ) ;
# endif
2020-04-10 17:24:08 +01:00
}
2020-11-12 17:37:01 +00:00
int32_t NvmErase ( const char * sNvsName ) {
2021-12-12 17:04:46 +00:00
nvs_handle_t handle ;
2020-11-12 17:37:01 +00:00
int32_t result = nvs_open ( sNvsName , NVS_READWRITE , & handle ) ;
2020-11-12 16:38:15 +00:00
if ( ESP_OK = = result ) { result = nvs_erase_all ( handle ) ; }
if ( ESP_OK = = result ) { result = nvs_commit ( handle ) ; }
2020-04-10 17:24:08 +01:00
nvs_close ( handle ) ;
2020-11-12 16:38:15 +00:00
return result ;
2020-04-10 17:24:08 +01:00
}
2020-04-22 15:07:52 +01:00
void SettingsErase ( uint8_t type ) {
2021-01-08 16:06:17 +00:00
// SDK and Tasmota data is held in default NVS partition
2021-01-12 13:54:12 +00:00
// Tasmota data is held also in file /.settings on default filesystem
2020-11-12 16:50:34 +00:00
// cal_data - SDK PHY calibration data as documented in esp_phy_init.h
// qpc - Tasmota Quick Power Cycle state
// main - Tasmota Settings data
2021-06-20 21:04:32 +01:00
int32_t r1 , r2 , r3 = 0 ;
2020-11-12 16:38:15 +00:00
switch ( type ) {
2021-01-13 11:38:35 +00:00
case 0 : // Reset 2 = Erase all flash from program end to end of physical flash
case 2 : // Reset 5, 6 = Erase all flash from program end to end of physical flash excluding filesystem
2020-11-12 16:38:15 +00:00
// nvs_flash_erase(); // Erase RTC, PHY, sta.mac, ap.sndchan, ap.mac, Tasmota etc.
r1 = NvmErase ( " qpc " ) ;
r2 = NvmErase ( " main " ) ;
2021-06-20 21:04:32 +01:00
# ifdef USE_UFILESYS
2021-01-08 16:06:17 +00:00
r3 = TfsDeleteFile ( TASM_FILE_SETTINGS ) ;
2021-06-20 21:04:32 +01:00
# endif
2021-01-23 15:26:23 +00:00
AddLog ( LOG_LEVEL_DEBUG , PSTR ( D_LOG_APPLICATION D_ERASE " Tasmota data (%d,%d,%d) " ) , r1 , r2 , r3 ) ;
2020-11-12 16:38:15 +00:00
break ;
2021-01-13 11:38:35 +00:00
case 1 : // Reset 3 = SDK parameter area
case 4 : // WIFI_FORCE_RF_CAL_ERASE = SDK parameter area
2020-11-12 16:50:34 +00:00
r1 = esp_phy_erase_cal_data_in_nvs ( ) ;
// r1 = NvmErase("cal_data");
2021-01-23 15:26:23 +00:00
AddLog ( LOG_LEVEL_DEBUG , PSTR ( D_LOG_APPLICATION D_ERASE " PHY data (%d) " ) , r1 ) ;
2020-11-12 16:38:15 +00:00
break ;
case 3 : // QPC Reached = QPC, Tasmota and SDK parameter area (0x0F3xxx - 0x0FFFFF)
// nvs_flash_erase(); // Erase RTC, PHY, sta.mac, ap.sndchan, ap.mac, Tasmota etc.
r1 = NvmErase ( " qpc " ) ;
r2 = NvmErase ( " main " ) ;
// r3 = esp_phy_erase_cal_data_in_nvs();
// r3 = NvmErase("cal_data");
2021-01-23 15:26:23 +00:00
// AddLog(LOG_LEVEL_DEBUG, PSTR(D_LOG_APPLICATION D_ERASE " Tasmota (%d,%d) and PHY data (%d)"), r1, r2, r3);
2021-06-20 21:04:32 +01:00
# ifdef USE_UFILESYS
2021-01-08 16:06:17 +00:00
r3 = TfsDeleteFile ( TASM_FILE_SETTINGS ) ;
2021-06-20 21:04:32 +01:00
# endif
2021-01-23 15:26:23 +00:00
AddLog ( LOG_LEVEL_DEBUG , PSTR ( D_LOG_APPLICATION D_ERASE " Tasmota data (%d,%d,%d) " ) , r1 , r2 , r3 ) ;
2020-11-12 16:38:15 +00:00
break ;
2020-04-19 14:36:04 +01:00
}
2020-04-10 17:24:08 +01:00
}
2021-01-08 15:22:06 +00:00
uint32_t SettingsRead ( void * data , size_t size ) {
2021-06-20 21:04:32 +01:00
# ifdef USE_UFILESYS
2021-12-12 17:04:46 +00:00
if ( TfsLoadFile ( TASM_FILE_SETTINGS , ( uint8_t * ) data , size ) ) {
return 2 ;
2021-01-08 15:22:06 +00:00
}
2021-06-20 21:04:32 +01:00
# endif
2021-12-12 17:04:46 +00:00
if ( NvmLoad ( " main " , " Settings " , data , size ) ) {
return 1 ;
} ;
return 0 ;
2020-04-10 17:24:08 +01:00
}
2020-04-22 15:07:52 +01:00
void SettingsWrite ( const void * pSettings , unsigned nSettingsLen ) {
2021-06-20 21:04:32 +01:00
# ifdef USE_UFILESYS
2021-01-08 16:06:17 +00:00
TfsSaveFile ( TASM_FILE_SETTINGS , ( const uint8_t * ) pSettings , nSettingsLen ) ;
2021-06-20 21:04:32 +01:00
# endif
2021-12-17 17:01:27 +00:00
NvmSave ( " main " , " Settings " , pSettings , nSettingsLen ) ;
2020-04-10 17:24:08 +01:00
}
2020-04-22 15:07:52 +01:00
void QPCRead ( void * pSettings , unsigned nSettingsLen ) {
2020-04-19 14:36:04 +01:00
NvmLoad ( " qpc " , " pcreg " , pSettings , nSettingsLen ) ;
2020-04-10 17:24:08 +01:00
}
2020-04-22 15:07:52 +01:00
void QPCWrite ( const void * pSettings , unsigned nSettingsLen ) {
2020-04-19 14:36:04 +01:00
NvmSave ( " qpc " , " pcreg " , pSettings , nSettingsLen ) ;
2020-04-10 17:24:08 +01:00
}
2022-05-10 21:21:34 +01:00
bool OtaFactoryRead ( void ) {
uint32_t pOtaLoader ;
NvmLoad ( " otal " , " otal " , & pOtaLoader , sizeof ( pOtaLoader ) ) ;
return pOtaLoader ;
}
void OtaFactoryWrite ( bool value ) {
uint32_t pOtaLoader = value ;
NvmSave ( " otal " , " otal " , & pOtaLoader , sizeof ( pOtaLoader ) ) ;
}
2020-12-15 15:24:01 +00:00
void NvsInfo ( void ) {
nvs_stats_t nvs_stats ;
nvs_get_stats ( NULL , & nvs_stats ) ;
2021-01-23 15:26:23 +00:00
AddLog ( LOG_LEVEL_INFO , PSTR ( " NVS: Used %d/%d entries, NameSpaces %d " ) ,
2020-12-16 11:01:10 +00:00
nvs_stats . used_entries , nvs_stats . total_entries , nvs_stats . namespace_count ) ;
2020-07-05 14:13:57 +01:00
}
2020-11-22 16:35:04 +00:00
//
// Flash memory mapping
//
2021-02-03 11:22:17 +00:00
// See Esp.cpp
2020-11-22 16:35:04 +00:00
# include "Esp.h"
2023-08-05 14:11:10 +01:00
# if ESP_IDF_VERSION_MAJOR >= 5
// esp_spi_flash.h is deprecated, please use spi_flash_mmap.h instead
# include "spi_flash_mmap.h"
# else
# include "esp_spi_flash.h"
# endif
2020-11-22 16:35:04 +00:00
# include <memory>
# include <soc/soc.h>
# include <soc/efuse_reg.h>
# include <esp_partition.h>
extern " C " {
# include "esp_ota_ops.h"
# include "esp_image_format.h"
}
2021-02-03 11:22:17 +00:00
# include "esp_system.h"
2023-08-23 12:02:13 +01:00
# include "esp_flash.h"
2021-02-03 11:22:17 +00:00
# if ESP_IDF_VERSION_MAJOR > 3 // IDF 4+
# if CONFIG_IDF_TARGET_ESP32 // ESP32/PICO-D4
# include "esp32/rom/spi_flash.h"
2022-09-17 14:53:58 +01:00
# define ESP_FLASH_IMAGE_BASE 0x1000 // Flash offset containing magic flash size and spi mode
2021-02-03 11:22:17 +00:00
# elif CONFIG_IDF_TARGET_ESP32S2 // ESP32-S2
# include "esp32s2/rom/spi_flash.h"
2022-09-17 14:53:58 +01:00
# define ESP_FLASH_IMAGE_BASE 0x1000 // Flash offset containing magic flash size and spi mode
2022-01-30 11:55:03 +00:00
# elif CONFIG_IDF_TARGET_ESP32S3 // ESP32-S3
# include "esp32s3/rom/spi_flash.h"
2022-09-17 14:53:58 +01:00
# define ESP_FLASH_IMAGE_BASE 0x0000 // Esp32s3 is located at 0x0000
2023-08-21 15:00:20 +01:00
# elif CONFIG_IDF_TARGET_ESP32C2 // ESP32-C2
# include "esp32c2/rom/spi_flash.h"
# define ESP_FLASH_IMAGE_BASE 0x0000 // Esp32c2 is located at 0x0000
2021-04-17 18:57:13 +01:00
# elif CONFIG_IDF_TARGET_ESP32C3 // ESP32-C3
# include "esp32c3/rom/spi_flash.h"
2022-09-17 14:53:58 +01:00
# define ESP_FLASH_IMAGE_BASE 0x0000 // Esp32c3 is located at 0x0000
2023-08-21 15:00:20 +01:00
# elif CONFIG_IDF_TARGET_ESP32C6 // ESP32-C6
# include "esp32c6/rom/spi_flash.h"
# define ESP_FLASH_IMAGE_BASE 0x0000 // Esp32c6 is located at 0x0000
2021-02-03 11:22:17 +00:00
# else
# error Target CONFIG_IDF_TARGET is not supported
# endif
# else // ESP32 Before IDF 4.0
# include "rom/spi_flash.h"
2022-09-17 14:53:58 +01:00
# define ESP_FLASH_IMAGE_BASE 0x1000
2021-02-03 11:22:17 +00:00
# endif
2023-08-05 14:11:10 +01:00
# if ESP_IDF_VERSION_MAJOR >= 5
# include "bootloader_common.h"
# endif
2020-11-22 16:35:04 +00:00
2022-05-14 16:41:52 +01:00
uint32_t EspProgramSize ( const char * label ) {
const esp_partition_t * part = esp_partition_find_first ( ESP_PARTITION_TYPE_APP , ESP_PARTITION_SUBTYPE_ANY , label ) ;
if ( ! part ) {
return 0 ;
}
const esp_partition_pos_t part_pos = {
. offset = part - > address ,
. size = part - > size ,
} ;
esp_image_metadata_t data ;
data . start_addr = part_pos . offset ;
esp_image_verify ( ESP_IMAGE_VERIFY , & part_pos , & data ) ;
return data . image_len ;
}
2022-05-01 15:12:54 +01:00
bool EspSingleOtaPartition ( void ) {
return ( 1 = = esp_ota_get_app_partition_count ( ) ) ;
}
2022-05-05 17:20:25 +01:00
uint32_t EspRunningFactoryPartition ( void ) {
2022-05-01 22:16:21 +01:00
const esp_partition_t * cur_part = esp_ota_get_running_partition ( ) ;
2022-05-05 17:20:25 +01:00
// return (cur_part->type == 0 && cur_part->subtype == 0);
if ( cur_part - > type = = 0 & & cur_part - > subtype = = 0 ) {
return cur_part - > size ;
}
return 0 ;
2022-05-01 22:16:21 +01:00
}
2022-05-05 10:19:39 +01:00
void EspPrepRestartToSafeBoot ( void ) {
2022-05-01 16:32:42 +01:00
const esp_partition_t * otadata_partition = esp_partition_find_first ( ESP_PARTITION_TYPE_DATA , ESP_PARTITION_SUBTYPE_DATA_OTA , NULL ) ;
if ( otadata_partition ) {
esp_partition_erase_range ( otadata_partition , 0 , SPI_FLASH_SEC_SIZE * 2 ) ;
}
2022-05-01 15:12:54 +01:00
}
2022-05-15 13:59:39 +01:00
bool EspPrepSwitchPartition ( uint32_t state ) {
2022-05-05 14:24:00 +01:00
bool valid = EspSingleOtaPartition ( ) ;
if ( valid ) {
bool running_factory = EspRunningFactoryPartition ( ) ;
2022-05-15 13:59:39 +01:00
switch ( state ) {
case 0 : // Off = safeboot
if ( ! running_factory ) {
EspPrepRestartToSafeBoot ( ) ;
} else {
valid = false ;
}
break ;
case 1 : // On = ota0
if ( running_factory ) {
const esp_partition_t * partition = esp_ota_get_next_update_partition ( nullptr ) ;
esp_ota_set_boot_partition ( partition ) ;
} else {
valid = false ;
}
break ;
case 2 : // Toggle
if ( ! running_factory ) {
EspPrepRestartToSafeBoot ( ) ;
} else {
const esp_partition_t * partition = esp_ota_get_next_update_partition ( nullptr ) ;
esp_ota_set_boot_partition ( partition ) ;
}
2022-05-05 14:24:00 +01:00
}
}
return valid ;
}
2020-11-22 16:35:04 +00:00
uint32_t EspFlashBaseAddress ( void ) {
2022-05-01 15:12:54 +01:00
if ( EspSingleOtaPartition ( ) ) { // Only one partition so start at end of sketch
2022-05-01 14:18:15 +01:00
const esp_partition_t * running = esp_ota_get_running_partition ( ) ;
if ( ! running ) { return 0 ; }
return running - > address + ESP_getSketchSize ( ) ;
} else { // Select other partition
const esp_partition_t * partition = esp_ota_get_next_update_partition ( nullptr ) ;
if ( ! partition ) { return 0 ; }
return partition - > address ; // For tasmota 0x00010000 or 0x00200000
}
2020-11-22 16:35:04 +00:00
}
uint32_t EspFlashBaseEndAddress ( void ) {
2022-05-01 15:12:54 +01:00
const esp_partition_t * partition = ( EspSingleOtaPartition ( ) ) ? esp_ota_get_running_partition ( ) : esp_ota_get_next_update_partition ( nullptr ) ;
2020-11-22 16:35:04 +00:00
if ( ! partition ) { return 0 ; }
return partition - > address + partition - > size ; // For tasmota 0x00200000 or 0x003F0000
}
uint8_t * EspFlashMmap ( uint32_t address ) {
static spi_flash_mmap_handle_t handle = 0 ;
if ( handle ) {
spi_flash_munmap ( handle ) ;
handle = 0 ;
}
const uint8_t * data ;
int32_t err = spi_flash_mmap ( address , 5 * SPI_FLASH_MMU_PAGE_SIZE , SPI_FLASH_MMAP_DATA , ( const void * * ) & data , & handle ) ;
/*
2021-01-23 15:26:23 +00:00
AddLog ( LOG_LEVEL_DEBUG , PSTR ( " DBG: Spi_flash_map %d " ) , err ) ;
2020-11-22 16:35:04 +00:00
spi_flash_mmap_dump ( ) ;
*/
return ( uint8_t * ) data ;
}
/*
int32_t EspPartitionMmap ( uint32_t action ) {
static spi_flash_mmap_handle_t handle ;
int32_t err = 0 ;
if ( 1 = = action ) {
const esp_partition_t * partition = esp_ota_get_running_partition ( ) ;
// const esp_partition_t* partition = esp_ota_get_next_update_partition(nullptr);
if ( ! partition ) { return 0 ; }
err = esp_partition_mmap ( partition , 0 , 4 * SPI_FLASH_MMU_PAGE_SIZE , SPI_FLASH_MMAP_DATA , ( const void * * ) & TasmotaGlobal_mmap_data , & handle ) ;
2021-01-23 15:26:23 +00:00
AddLog ( LOG_LEVEL_DEBUG , PSTR ( " DBG: Partition start 0x%08X, Partition end 0x%08X, Mmap data 0x%08X " ) , partition - > address , partition - > size , TasmotaGlobal_mmap_data ) ;
2020-11-22 16:35:04 +00:00
} else {
spi_flash_munmap ( handle ) ;
handle = 0 ;
}
return err ;
}
*/
2020-04-10 17:24:08 +01:00
2020-04-19 14:36:04 +01:00
//
// ESP32 specific
//
2023-08-21 15:00:20 +01:00
# ifdef CONFIG_IDF_TARGET_ESP32
2020-04-19 14:36:04 +01:00
# include "soc/soc.h"
# include "soc/rtc_cntl_reg.h"
2020-04-22 15:07:52 +01:00
void DisableBrownout ( void ) {
2020-04-19 14:36:04 +01:00
// https://github.com/espressif/arduino-esp32/issues/863#issuecomment-347179737
WRITE_PERI_REG ( RTC_CNTL_BROWN_OUT_REG , 0 ) ; // Disable brownout detector
}
2023-08-21 15:00:20 +01:00
# endif // ESP32
2020-04-19 14:36:04 +01:00
2020-04-19 15:58:13 +01:00
//
// ESP32 Alternatives
//
2020-04-22 15:07:52 +01:00
String ESP32GetResetReason ( uint32_t cpu_no ) {
2020-04-19 15:58:13 +01:00
// tools\sdk\include\esp32\rom\rtc.h
2021-06-29 17:15:22 +01:00
// tools\sdk\esp32\include\esp_rom\include\esp32c3\rom\rtc.h
2021-02-02 16:55:45 +00:00
// tools\sdk\esp32\include\esp_rom\include\esp32s2\rom\rtc.h
2021-06-29 17:15:22 +01:00
switch ( rtc_get_reset_reason ( cpu_no ) ) { // ESP32 ESP32-S / ESP32-C
case 1 : return F ( " Vbat power on reset " ) ; // 1 POWERON_RESET POWERON_RESET
case 3 : return F ( " Software reset digital core " ) ; // 3 SW_RESET RTC_SW_SYS_RESET
case 4 : return F ( " Legacy watch dog reset digital core " ) ; // 4 OWDT_RESET -
case 5 : return F ( " Deep Sleep reset digital core " ) ; // 5 DEEPSLEEP_RESET DEEPSLEEP_RESET
case 6 : return F ( " Reset by SLC module, reset digital core " ) ; // 6 SDIO_RESET
case 7 : return F ( " Timer Group0 Watch dog reset digital core " ) ; // 7 TG0WDT_SYS_RESET
case 8 : return F ( " Timer Group1 Watch dog reset digital core " ) ; // 8 TG1WDT_SYS_RESET
case 9 : return F ( " RTC Watch dog Reset digital core " ) ; // 9 RTCWDT_SYS_RESET
case 10 : return F ( " Instrusion tested to reset CPU " ) ; // 10 INTRUSION_RESET
case 11 : return F ( " Time Group0 reset CPU " ) ; // 11 TGWDT_CPU_RESET TG0WDT_CPU_RESET
case 12 : return F ( " Software reset CPU " ) ; // 12 SW_CPU_RESET RTC_SW_CPU_RESET
case 13 : return F ( " RTC Watch dog Reset CPU " ) ; // 13 RTCWDT_CPU_RESET
case 14 : return F ( " or APP CPU, reseted by PRO CPU " ) ; // 14 EXT_CPU_RESET -
case 15 : return F ( " Reset when the vdd voltage is not stable " ) ; // 15 RTCWDT_BROWN_OUT_RESET
case 16 : return F ( " RTC Watch dog reset digital core and rtc module " ) ; // 16 RTCWDT_RTC_RESET
case 17 : return F ( " Time Group1 reset CPU " ) ; // 17 - TG1WDT_CPU_RESET
case 18 : return F ( " Super watchdog reset digital core and rtc module " ) ; // 18 - SUPER_WDT_RESET
case 19 : return F ( " Glitch reset digital core and rtc module " ) ; // 19 - GLITCH_RTC_RESET
2022-04-04 10:23:40 +01:00
case 20 : return F ( " Efuse reset digital core " ) ; // 20 EFUSE_RESET
case 21 : return F ( " Usb uart reset digital core " ) ; // 21 USB_UART_CHIP_RESET
case 22 : return F ( " Usb jtag reset digital core " ) ; // 22 USB_JTAG_CHIP_RESET
case 23 : return F ( " Power glitch reset digital core and rtc module " ) ; // 23 POWER_GLITCH_RESET
2021-02-02 16:55:45 +00:00
}
2021-06-29 17:15:22 +01:00
return F ( " No meaning " ) ; // 0 and undefined
2020-04-19 15:58:13 +01:00
}
2020-04-22 15:07:52 +01:00
String ESP_getResetReason ( void ) {
2020-04-19 15:58:13 +01:00
return ESP32GetResetReason ( 0 ) ; // CPU 0
}
2020-04-22 15:07:52 +01:00
uint32_t ESP_ResetInfoReason ( void ) {
2020-04-19 15:58:13 +01:00
RESET_REASON reason = rtc_get_reset_reason ( 0 ) ;
2021-06-29 17:15:22 +01:00
if ( 1 = = reason ) { return REASON_DEFAULT_RST ; } // POWERON_RESET
if ( 12 = = reason ) { return REASON_SOFT_RESTART ; } // SW_CPU_RESET / RTC_SW_CPU_RESET
if ( 5 = = reason ) { return REASON_DEEP_SLEEP_AWAKE ; } // DEEPSLEEP_RESET
if ( 3 = = reason ) { return REASON_EXT_SYS_RST ; } // SW_RESET / RTC_SW_SYS_RESET
2020-11-15 08:57:23 +00:00
return - 1 ; //no "official error code", but should work with the current code base
2020-04-19 15:58:13 +01:00
}
2020-04-22 15:07:52 +01:00
uint32_t ESP_getChipId ( void ) {
2020-04-19 15:58:13 +01:00
uint32_t id = 0 ;
for ( uint32_t i = 0 ; i < 17 ; i = i + 8 ) {
id | = ( ( ESP . getEfuseMac ( ) > > ( 40 - i ) ) & 0xff ) < < i ;
}
return id ;
}
2023-08-23 12:02:13 +01:00
uint32_t ESP_getFlashChipMagicSize ( void ) {
2022-09-17 14:53:58 +01:00
esp_image_header_t fhdr ;
2023-08-23 12:02:13 +01:00
// if(ESP.flashRead(ESP_FLASH_IMAGE_BASE, (uint32_t*)&fhdr.magic, sizeof(esp_image_header_t)) && fhdr.magic != ESP_IMAGE_HEADER_MAGIC) {
// return 0;
// }
if ( esp_flash_read ( esp_flash_default_chip , ( void * ) & fhdr , ESP_FLASH_IMAGE_BASE , sizeof ( esp_image_header_t ) ) & & fhdr . magic ! = ESP_IMAGE_HEADER_MAGIC ) {
return 0 ;
2022-09-17 14:53:58 +01:00
}
return ESP_magicFlashChipSize ( fhdr . spi_size ) ;
}
uint32_t ESP_magicFlashChipSize ( uint8_t byte )
{
switch ( byte & 0x0F ) {
case 0x0 : // 8 MBit (1MB)
return 1048576 ;
case 0x1 : // 16 MBit (2MB)
return 2097152 ;
case 0x2 : // 32 MBit (4MB)
return 4194304 ;
case 0x3 : // 64 MBit (8MB)
return 8388608 ;
case 0x4 : // 128 MBit (16MB)
return 16777216 ;
default : // fail?
return 0 ;
}
}
2020-04-22 15:07:52 +01:00
uint32_t ESP_getSketchSize ( void ) {
2020-04-19 15:58:13 +01:00
static uint32_t sketchsize = 0 ;
if ( ! sketchsize ) {
sketchsize = ESP . getSketchSize ( ) ; // This takes almost 2 seconds on an ESP32
}
return sketchsize ;
}
2022-05-07 10:05:10 +01:00
uint32_t ESP_getFreeSketchSpace ( void ) {
if ( EspSingleOtaPartition ( ) ) {
uint32_t size = EspRunningFactoryPartition ( ) ;
if ( ! size ) {
size = ESP . getFreeSketchSpace ( ) ;
}
return size - ESP_getSketchSize ( ) ;
}
return ESP . getFreeSketchSpace ( ) ;
}
2020-04-22 15:07:52 +01:00
uint32_t ESP_getFreeHeap ( void ) {
2021-10-07 17:37:58 +01:00
// ESP_getFreeHeap() returns also IRAM which we don't use
return heap_caps_get_free_size ( MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT ) ;
2020-04-22 15:07:52 +01:00
}
2020-08-12 10:46:06 +01:00
uint32_t ESP_getMaxAllocHeap ( void ) {
2021-10-07 17:37:58 +01:00
// arduino returns IRAM but we want only DRAM
2023-05-10 13:02:51 +01:00
# ifdef USE_GT911 // GT911 IRQ crashes with heap_caps_get_largest_free_block
2022-12-18 13:06:04 +00:00
return ESP_getFreeHeap ( ) ;
# endif
2021-10-07 17:37:58 +01:00
uint32_t free_block_size = heap_caps_get_largest_free_block ( MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT ) ;
2020-08-12 11:11:47 +01:00
if ( free_block_size > 100 ) { free_block_size - = 100 ; }
return free_block_size ;
2020-08-12 10:46:06 +01:00
}
2021-10-08 10:52:50 +01:00
int32_t ESP_getHeapFragmentation ( void ) {
int32_t free_maxmem = 100 - ( int32_t ) ( ESP_getMaxAllocHeap ( ) * 100 / ESP_getFreeHeap ( ) ) ;
if ( free_maxmem < 0 ) { free_maxmem = 0 ; }
return free_maxmem ;
}
2020-04-22 15:07:52 +01:00
void ESP_Restart ( void ) {
2020-04-19 15:58:13 +01:00
ESP . restart ( ) ;
}
2020-11-22 16:35:04 +00:00
uint32_t FlashWriteStartSector ( void ) {
// Needs to be on SPI_FLASH_MMU_PAGE_SIZE (= 0x10000) alignment for mmap usage
uint32_t aligned_address = ( ( EspFlashBaseAddress ( ) + ( 2 * SPI_FLASH_MMU_PAGE_SIZE ) ) / SPI_FLASH_MMU_PAGE_SIZE ) * SPI_FLASH_MMU_PAGE_SIZE ;
return aligned_address / SPI_FLASH_SEC_SIZE ;
}
uint32_t FlashWriteMaxSector ( void ) {
// Needs to be on SPI_FLASH_MMU_PAGE_SIZE (= 0x10000) alignment for mmap usage
uint32_t aligned_end_address = ( EspFlashBaseEndAddress ( ) / SPI_FLASH_MMU_PAGE_SIZE ) * SPI_FLASH_MMU_PAGE_SIZE ;
return aligned_end_address / SPI_FLASH_SEC_SIZE ;
}
uint8_t * FlashDirectAccess ( void ) {
uint32_t address = FlashWriteStartSector ( ) * SPI_FLASH_SEC_SIZE ;
uint8_t * data = EspFlashMmap ( address ) ;
/*
uint8_t buf [ 32 ] ;
memcpy ( buf , data , sizeof ( buf ) ) ;
2022-11-11 08:30:31 +00:00
AddLog ( LOG_LEVEL_DEBUG , PSTR ( " DBG: Flash start address 0x%08X, Mmap address 0x%08X, Data %*_H " ) , address , data , sizeof ( buf ) , ( uint8_t * ) & buf ) ;
2020-11-22 16:35:04 +00:00
*/
return data ;
}
2021-01-01 16:04:36 +00:00
2021-09-01 21:48:02 +01:00
extern " C " {
2023-08-05 19:34:24 +01:00
# if ESP_IDF_VERSION_MAJOR >= 5
// bool IRAM_ATTR __attribute__((pure)) esp_psram_is_initialized(void)
bool esp_psram_is_initialized ( void ) ;
# else
bool esp_spiram_is_initialized ( void ) ;
# endif
2021-09-01 21:48:02 +01:00
}
// this function is a replacement for `psramFound()`.
// `psramFound()` can return true even if no PSRAM is actually installed
// This new version also checks `esp_spiram_is_initialized` to know if the PSRAM is initialized
bool FoundPSRAM ( void ) {
2023-08-21 15:00:20 +01:00
# if CONFIG_IDF_TARGET_ESP32C3 || CONFIG_IDF_TARGET_ESP32C6
2021-09-02 19:54:50 +01:00
return psramFound ( ) ;
# else
2023-08-05 19:34:24 +01:00
# if ESP_IDF_VERSION_MAJOR >= 5
return psramFound ( ) & & esp_psram_is_initialized ( ) ;
# else
return psramFound ( ) & & esp_spiram_is_initialized ( ) ;
# endif
2021-09-02 19:54:50 +01:00
# endif
2021-09-01 21:48:02 +01:00
}
2021-07-18 18:43:33 +01:00
// new function to check whether PSRAM is present and supported (i.e. required pacthes are present)
bool UsePSRAM ( void ) {
static bool can_use_psram = CanUsePSRAM ( ) ;
2021-09-01 21:48:02 +01:00
return FoundPSRAM ( ) & & can_use_psram ;
2021-07-18 18:43:33 +01:00
}
2021-01-04 15:39:00 +00:00
void * special_malloc ( uint32_t size ) {
2021-07-18 18:43:33 +01:00
if ( UsePSRAM ( ) ) {
2021-01-04 15:39:00 +00:00
return heap_caps_malloc ( size , MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT ) ;
} else {
return malloc ( size ) ;
}
}
2021-03-20 17:44:35 +00:00
void * special_realloc ( void * ptr , size_t size ) {
2021-07-18 18:43:33 +01:00
if ( UsePSRAM ( ) ) {
2021-03-20 17:44:35 +00:00
return heap_caps_realloc ( ptr , size , MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT ) ;
} else {
return realloc ( ptr , size ) ;
}
}
2021-05-09 17:15:15 +01:00
void * special_calloc ( size_t num , size_t size ) {
2021-07-18 18:43:33 +01:00
if ( UsePSRAM ( ) ) {
2021-05-09 17:15:15 +01:00
return heap_caps_calloc ( num , size , MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT ) ;
} else {
return calloc ( num , size ) ;
}
}
2021-01-04 15:39:00 +00:00
2022-01-06 12:41:51 +00:00
// Variants for IRAM heap, which need all accesses to be 32 bits aligned
void * special_malloc32 ( uint32_t size ) {
2022-02-15 21:51:11 +00:00
return heap_caps_malloc ( size , MALLOC_CAP_32BIT ) ;
2022-01-06 12:41:51 +00:00
}
2021-04-12 13:59:40 +01:00
float CpuTemperature ( void ) {
2021-10-10 10:22:53 +01:00
return ( float ) temperatureRead ( ) ; // In Celsius
2021-04-12 13:59:40 +01:00
}
2021-10-10 10:22:53 +01:00
/*
# ifdef __cplusplus
extern " C " {
# endif
uint8_t temprature_sens_read ( ) ;
# ifdef __cplusplus
}
# endif
# ifdef CONFIG_IDF_TARGET_ESP32
uint8_t temprature_sens_read ( ) ;
float CpuTemperature ( void ) {
uint8_t t = temprature_sens_read ( ) ;
AddLog ( LOG_LEVEL_DEBUG , PSTR ( " TMP: value %d " ) , t ) ;
return ( t - 32 ) / 1.8 ;
}
# else
float CpuTemperature ( void ) {
// Currently (20210801) repeated calls to temperatureRead() on ESP32C3 and ESP32S2 result in IDF error messages
static float t = NAN ;
if ( isnan ( t ) ) {
t = ( float ) temperatureRead ( ) ; // In Celsius
}
return t ;
}
# endif
*/
2021-04-17 16:32:20 +01:00
/*
# if CONFIG_IDF_TARGET_ESP32S2
# include "esp32s2/esp_efuse.h"
# elif CONFIG_IDF_TARGET_ESP32S3
# include "esp32s3/esp_efuse.h"
# elif CONFIG_IDF_TARGET_ESP32C3
# include "esp32c3/esp_efuse.h"
# endif
*/
2022-02-01 11:05:56 +00:00
// #include "esp_chip_info.h"
2021-02-08 10:34:29 +00:00
String GetDeviceHardware ( void ) {
2021-04-19 16:48:18 +01:00
// https://www.espressif.com/en/products/socs
2021-04-17 14:57:35 +01:00
/*
2023-08-25 09:55:20 +01:00
Source : esp - idf esp_system . h or arduino core esp_chip_info . h and esptool
2021-04-17 14:57:35 +01:00
typedef enum {
2022-02-01 11:05:56 +00:00
CHIP_ESP32 = 1 , //!< ESP32
CHIP_ESP32S2 = 2 , //!< ESP32-S2
CHIP_ESP32S3 = 9 , //!< ESP32-S3
CHIP_ESP32C3 = 5 , //!< ESP32-C3
CHIP_ESP32C2 = 12 , //!< ESP32-C2
2023-08-25 09:55:20 +01:00
CHIP_ESP32C6 = 13 , //!< ESP32-C6
CHIP_ESP32H2 = 16 , //!< ESP32-H2
CHIP_POSIX_LINUX = 999 , //!< The code is running on POSIX/Linux simulator
2021-04-17 14:57:35 +01:00
} esp_chip_model_t ;
// Chip feature flags, used in esp_chip_info_t
# define CHIP_FEATURE_EMB_FLASH BIT(0) //!< Chip has embedded flash memory
# define CHIP_FEATURE_WIFI_BGN BIT(1) //!< Chip has 2.4GHz WiFi
# define CHIP_FEATURE_BLE BIT(4) //!< Chip has Bluetooth LE
# define CHIP_FEATURE_BT BIT(5) //!< Chip has Bluetooth Classic
2022-02-01 11:05:56 +00:00
# define CHIP_FEATURE_IEEE802154 BIT(6) //!< Chip has IEEE 802.15.4
# define CHIP_FEATURE_EMB_PSRAM BIT(7) //!< Chip has embedded psram
2021-04-17 14:57:35 +01:00
// The structure represents information about the chip
typedef struct {
esp_chip_model_t model ; //!< chip model, one of esp_chip_model_t
uint32_t features ; //!< bit mask of CHIP_FEATURE_x feature flags
2023-08-25 09:55:20 +01:00
uint16_t revision ; //!< chip revision number (in format MXX; where M - wafer major version, XX - wafer minor version)
2021-04-17 14:57:35 +01:00
uint8_t cores ; //!< number of CPU cores
} esp_chip_info_t ;
2021-02-08 10:34:29 +00:00
2021-04-17 14:57:35 +01:00
*/
esp_chip_info_t chip_info ;
esp_chip_info ( & chip_info ) ;
uint32_t chip_model = chip_info . model ;
uint32_t chip_revision = chip_info . revision ;
// uint32_t chip_revision = ESP.getChipRevision();
2023-08-25 09:55:20 +01:00
// idf5 efuse_hal_chip_revision(void)
2023-08-25 11:48:35 +01:00
if ( chip_revision < 100 ) { chip_revision * = 100 ; } // Make <idf5 idf5
bool rev3 = ( chip_revision > = 300 ) ;
2021-04-17 14:57:35 +01:00
// bool single_core = (1 == ESP.getChipCores());
bool single_core = ( 1 = = chip_info . cores ) ;
2023-08-25 11:48:35 +01:00
uint32_t pkg_version = 0 ;
# if (ESP_IDF_VERSION_MAJOR >= 5)
pkg_version = bootloader_common_get_chip_ver_pkg ( ) ;
# endif
switch ( chip_model ) {
case 0 :
case 1 : { // ESP32
/*
ESP32 Series
- 32 - bit MCU & 2.4 GHz Wi - Fi & Bluetooth / Bluetooth LE
- Two or one CPU core ( s ) with adjustable clock frequency , ranging from 80 MHz to 240 MHz
- + 19.5 dBm output power ensures a good physical range
- Classic Bluetooth for legacy connections , also supporting L2CAP , SDP , GAP , SMP , AVDTP , AVCTP , A2DP ( SNK ) and AVRCP ( CT )
- Support for Bluetooth Low Energy ( Bluetooth LE ) profiles including L2CAP , GAP , GATT , SMP , and GATT - based profiles like BluFi , SPP - like , etc
- Bluetooth Low Energy ( Bluetooth LE ) connects to smart phones , broadcasting low - energy beacons for easy detection
- Sleep current is less than 5 μ A , making it suitable for battery - powered and wearable - electronics applications
- Peripherals include capacitive touch sensors , Hall sensor , SD card interface , Ethernet , high - speed SPI , UART , I2S and I2C
*/
2021-04-17 14:57:35 +01:00
# ifdef CONFIG_IDF_TARGET_ESP32
2023-08-25 11:48:35 +01:00
# if (ESP_IDF_VERSION_MAJOR < 5)
pkg_version = REG_GET_FIELD ( EFUSE_BLK0_RDATA3_REG , EFUSE_RD_CHIP_VER_PKG ) & 0x7 ;
2023-08-05 14:11:10 +01:00
# endif
2021-04-17 14:57:35 +01:00
2023-08-25 11:48:35 +01:00
// AddLog(LOG_LEVEL_DEBUG_MORE, PSTR("HDW: ESP32 Model %d, Revision %d, Core %d"), chip_info.model, chip_revision, chip_info.cores);
switch ( pkg_version ) {
case 0 :
if ( single_core ) { return F ( " ESP32-S0WDQ6 " ) ; } // Max 240MHz, Single core, QFN 6*6
else if ( rev3 ) { return F ( " ESP32-D0WDQ6-V3 " ) ; } // Max 240MHz, Dual core, QFN 6*6
else { return F ( " ESP32-D0WDQ6 " ) ; } // Max 240MHz, Dual core, QFN 6*6
case 1 :
if ( single_core ) { return F ( " ESP32-S0WD " ) ; } // Max 160MHz, Single core, QFN 5*5, ESP32-SOLO-1, ESP32-DevKitC
else if ( rev3 ) { return F ( " ESP32-D0WD-V3 " ) ; } // Max 240MHz, Dual core, QFN 5*5, ESP32-WROOM-32E, ESP32_WROVER-E, ESP32-DevKitC
else { return F ( " ESP32-D0WD " ) ; } // Max 240MHz, Dual core, QFN 5*5, ESP32-WROOM-32D, ESP32_WROVER-B, ESP32-DevKitC
case 2 : return F ( " ESP32-D2WD " ) ; // Max 160MHz, Dual core, QFN 5*5, 2MB embedded flash
case 3 :
if ( single_core ) { return F ( " ESP32-S0WD-OEM " ) ; } // Max 160MHz, Single core, QFN 5*5, Xiaomi Yeelight
else { return F ( " ESP32-D0WD-OEM " ) ; } // Max 240MHz, Dual core, QFN 5*5
case 4 :
if ( single_core ) { return F ( " ESP32-U4WDH-S " ) ; } // Max 160MHz, Single core, QFN 5*5, 4MB embedded flash, ESP32-MINI-1, ESP32-DevKitM-1
else { return F ( " ESP32-U4WDH-D " ) ; } // Max 240MHz, Dual core, QFN 5*5, 4MB embedded flash
case 5 :
if ( rev3 ) { return F ( " ESP32-PICO-V3 " ) ; } // Max 240MHz, Dual core, LGA 7*7, ESP32-PICO-V3-ZERO, ESP32-PICO-V3-ZERO-DevKit
else { return F ( " ESP32-PICO-D4 " ) ; } // Max 240MHz, Dual core, LGA 7*7, 4MB embedded flash, ESP32-PICO-KIT
case 6 : return F ( " ESP32-PICO-V3-02 " ) ; // Max 240MHz, Dual core, LGA 7*7, 8MB embedded flash, 2MB embedded PSRAM, ESP32-PICO-MINI-02, ESP32-PICO-DevKitM-2
case 7 : return F ( " ESP32-D0WDR2-V3 " ) ; // Max 240MHz, Dual core, QFN 5*5, ESP32-WROOM-32E, ESP32_WROVER-E, ESP32-DevKitC
}
2021-04-17 14:57:35 +01:00
# endif // CONFIG_IDF_TARGET_ESP32
2023-08-25 11:48:35 +01:00
return F ( " ESP32 " ) ;
}
case 2 : { // ESP32-S2
/*
ESP32 - S2 Series
- 32 - bit MCU & 2.4 GHz Wi - Fi
- High - performance 240 MHz single - core CPU
- Ultra - low - power performance : fine - grained clock gating , dynamic voltage and frequency scaling
- Security features : eFuse 、 flash encryption , secure boot , signature verification , integrated AES , SHA and RSA algorithms
- Peripherals include 43 GPIOs , 1 full - speed USB OTG interface , SPI , I2S , UART , I2C , LED PWM , LCD interface , camera interface , ADC , DAC , touch sensor , temperature sensor
- Availability of common cloud connectivity agents and common product features shortens the time to market
*/
2021-04-17 14:57:35 +01:00
# ifdef CONFIG_IDF_TARGET_ESP32S2
2023-08-25 11:48:35 +01:00
# if (ESP_IDF_VERSION_MAJOR < 5)
pkg_version = REG_GET_FIELD ( EFUSE_RD_MAC_SPI_SYS_3_REG , EFUSE_FLASH_VERSION ) & 0xF ;
# endif
uint32_t psram_ver = REG_GET_FIELD ( EFUSE_RD_MAC_SPI_SYS_3_REG , EFUSE_PSRAM_VERSION ) ;
pkg_version + = ( ( psram_ver & 0xF ) * 100 ) ;
// AddLog(LOG_LEVEL_DEBUG_MORE, PSTR("HDW: ESP32 Model %d, Revision %d, Core %d, Package %d"), chip_info.model, chip_revision, chip_info.cores, chip_ver);
switch ( pkg_version ) {
case 0 : return F ( " ESP32-S2 " ) ; // Max 240MHz, Single core, QFN 7*7, ESP32-S2-WROOM, ESP32-S2-WROVER, ESP32-S2-Saola-1, ESP32-S2-Kaluga-1
case 1 : return F ( " ESP32-S2FH2 " ) ; // Max 240MHz, Single core, QFN 7*7, 2MB embedded flash, ESP32-S2-MINI-1, ESP32-S2-DevKitM-1
case 2 : return F ( " ESP32-S2FH4 " ) ; // Max 240MHz, Single core, QFN 7*7, 4MB embedded flash
case 3 : return F ( " ESP32-S2FN4R2 " ) ; // Max 240MHz, Single core, QFN 7*7, 4MB embedded flash, 2MB embedded PSRAM, , ESP32-S2-MINI-1U, ESP32-S2-DevKitM-1U
case 100 : return F ( " ESP32-S2R2 " ) ;
case 102 : return F ( " ESP32-S2FNR2 " ) ; // Max 240MHz, Single core, QFN 7*7, 4MB embedded flash, 2MB embedded PSRAM, , Lolin S2 mini
}
2021-04-17 14:57:35 +01:00
# endif // CONFIG_IDF_TARGET_ESP32S2
2023-08-25 11:48:35 +01:00
return F ( " ESP32-S2 " ) ;
2021-04-17 14:57:35 +01:00
}
2023-08-25 11:48:35 +01:00
case 5 : { // ESP32-C3 = ESP8685 if embedded flash
/*
ESP32 - C3 Series
- 32 - bit RISC - V MCU & 2.4 GHz Wi - Fi & Bluetooth 5 ( LE )
- 32 - bit RISC - V single - core processor with a four - stage pipeline that operates at up to 160 MHz
- State - of - the - art power and RF performance
- 400 KB of SRAM and 384 KB of ROM on the chip , and SPI , Dual SPI , Quad SPI , and QPI interfaces that allow connection to flash
- Reliable security features ensured by RSA - 3072 - based secure boot , AES - 128 - XTS - based flash encryption , the innovative digital signature and the HMAC peripheral , hardware acceleration support for cryptographic algorithms
- Rich set of peripheral interfaces and GPIOs , ideal for various scenarios and complex applications
*/
# ifdef CONFIG_IDF_TARGET_ESP32C3
# if (ESP_IDF_VERSION_MAJOR < 5)
pkg_version = REG_GET_FIELD ( EFUSE_RD_MAC_SPI_SYS_3_REG , EFUSE_PKG_VERSION ) & 0x7 ;
# endif
switch ( pkg_version ) {
case 0 : return F ( " ESP32-C3 " ) ; // Max 160MHz, Single core, QFN 5*5, ESP32-C3-WROOM-02, ESP32-C3-DevKitC-02
// case 1: return F("ESP32-C3FH4"); // Max 160MHz, Single core, QFN 5*5, 4MB embedded flash, ESP32-C3-MINI-1, ESP32-C3-DevKitM-1
case 1 : return F ( " ESP8685 " ) ; // Max 160MHz, Single core, QFN 5*5, 4MB embedded flash, ESP32-C3-MINI-1, ESP32-C3-DevKitM-1
case 2 : return F ( " ESP32-C3 AZ " ) ; // QFN32
case 3 : return F ( " ESP8686 " ) ; // QFN24
}
2021-04-17 14:57:35 +01:00
# endif // CONFIG_IDF_TARGET_ESP32C3
2023-08-25 11:48:35 +01:00
return F ( " ESP32-C3 " ) ;
2023-08-24 19:52:36 +01:00
}
2023-08-25 11:48:35 +01:00
case 4 : // ESP32-S3(beta2)
case 6 : // ESP32-S3(beta3)
case 9 : { // ESP32-S3
/*
ESP32 - S3 Series
- 32 - bit MCU & 2.4 GHz Wi - Fi & Bluetooth 5 ( LE )
- Xtensa ® 32 - bit LX7 dual - core processor that operates at up to 240 MHz
- 512 KB of SRAM and 384 KB of ROM on the chip , and SPI , Dual SPI , Quad SPI , Octal SPI , QPI , and OPI interfaces that allow connection to flash and external RAM
- Additional support for vector instructions in the MCU , which provides acceleration for neural network computing and signal processing workloads
- Peripherals include 45 programmable GPIOs , SPI , I2S , I2C , PWM , RMT , ADC and UART , SD / MMC host and TWAITM
- Reliable security features ensured by RSA - based secure boot , AES - XTS - based flash encryption , the innovative digital signature and the HMAC peripheral , “ World Controller ”
*/
# ifdef CONFIG_IDF_TARGET_ESP32S3
# if (ESP_IDF_VERSION_MAJOR >= 5)
switch ( pkg_version ) {
2023-08-25 12:11:00 +01:00
case 0 : return F ( " ESP32-S3 " ) ; // QFN56
case 1 : return F ( " ESP32-S3-PICO-1 " ) ; // LGA56
2023-08-25 11:48:35 +01:00
}
2023-08-25 09:55:20 +01:00
# endif
# endif // CONFIG_IDF_TARGET_ESP32S3
2023-08-25 11:48:35 +01:00
return F ( " ESP32-S3 " ) ; // Max 240MHz, Dual core, QFN 7*7, ESP32-S3-WROOM-1, ESP32-S3-DevKitC-1
2023-08-25 09:55:20 +01:00
}
2023-08-25 11:48:35 +01:00
case 12 : { // ESP32-C2 = ESP8684 if embedded flash
/*
ESP32 - C2 Series
- 32 - bit RISC - V MCU & 2.4 GHz Wi - Fi & Bluetooth 5 ( LE )
- 32 - bit RISC - V single - core processor that operates at up to 120 MHz
- State - of - the - art power and RF performance
- 576 KB ROM , 272 KB SRAM ( 16 KB for cache ) on the chip
- 14 programmable GPIOs : SPI , UART , I2C , LED PWM controller , General DMA controller ( GDMA ) , SAR ADC , Temperature sensor
*/
# ifdef CONFIG_IDF_TARGET_ESP32C2
switch ( pkg_version ) {
case 0 : return F ( " ESP32-C2 " ) ;
case 1 : return F ( " ESP32-C2 " ) ;
}
2023-08-25 09:55:20 +01:00
# endif // CONFIG_IDF_TARGET_ESP32C2
2023-08-25 11:48:35 +01:00
return F ( " ESP32-C2 " ) ;
2023-08-25 09:55:20 +01:00
}
2023-08-25 11:48:35 +01:00
case 7 : // ESP32-C6(beta)
case 13 : { // ESP32-C6
/*
ESP32 - C6 Series
- 32 - bit RISC - V MCU & 2.4 GHz Wi - Fi 6 & Bluetooth 5 ( LE ) & IEEE 802.15 .4
- 32 - bit RISC - V single - core processor that operates at up to 160 MHz
- State - of - the - art power and RF performance
- 320 KB ROM , 512 KB SRAM , 16 KB Low - power SRAM on the chip , and works with external flash
- 30 ( QFN40 ) or 22 ( QFN32 ) programmable GPIOs , with support for SPI , UART , I2C , I2S , RMT , TWAI and PWM
*/
# ifdef CONFIG_IDF_TARGET_ESP32C6
switch ( pkg_version ) {
case 0 : return F ( " ESP32-C6 " ) ;
case 1 : return F ( " ESP32-C6FH4 " ) ;
}
2023-08-25 09:55:20 +01:00
# endif // CONFIG_IDF_TARGET_ESP32C6
2023-08-25 11:48:35 +01:00
return F ( " ESP32-C6 " ) ;
2023-08-25 09:55:20 +01:00
}
2023-08-25 11:48:35 +01:00
case 10 : // ESP32-H2(beta1)
case 14 : // ESP32-H2(beta2)
case 16 : { // ESP32-H2
# ifdef CONFIG_IDF_TARGET_ESP32H2
switch ( pkg_version ) {
case 0 : return F ( " ESP32-H2 " ) ;
}
2023-08-25 09:55:20 +01:00
# endif // CONFIG_IDF_TARGET_ESP32H2
2023-08-25 11:48:35 +01:00
return F ( " ESP32-H2 " ) ;
}
2023-08-25 12:11:00 +01:00
case 18 : { // ESP32-P4
# ifdef CONFIG_IDF_TARGET_ESP32P4
switch ( pkg_version ) {
case 0 : return F ( " ESP32-P4 " ) ;
}
# endif // CONFIG_IDF_TARGET_ESP32P4
return F ( " ESP32-P4 " ) ;
}
2023-08-25 09:55:20 +01:00
}
2021-04-17 14:57:35 +01:00
return F ( " ESP32 " ) ;
}
2021-02-08 10:34:29 +00:00
2022-04-23 16:59:37 +01:00
String GetDeviceHardwareRevision ( void ) {
2022-04-24 10:24:01 +01:00
// ESP32-S2
2023-08-25 11:48:35 +01:00
// ESP32-D0WDQ6 v1.0
2023-10-04 11:47:58 +01:00
// ESP32-C3 v0.3
2023-08-30 15:03:03 +01:00
// ESP32-C6FH4 v0.0
2022-04-24 10:24:01 +01:00
String result = GetDeviceHardware ( ) ; // ESP32-C3
2022-04-23 16:59:37 +01:00
esp_chip_info_t chip_info ;
esp_chip_info ( & chip_info ) ;
2023-10-04 11:47:58 +01:00
# if ESP_IDF_VERSION_MAJOR >= 5
uint32_t chip_revision = chip_info . revision ; // 16-bit chip revision number (in format MXX; where M - wafer major version, XX - wafer minor version)
# else
uint32_t chip_revision = chip_info . full_revision ; // 16-bit chip revision number (in format MXX; where M - wafer major version, XX - wafer minor version)
# endif
char revision [ 16 ] ;
snprintf_P ( revision , sizeof ( revision ) , PSTR ( " v%d.%d " ) , chip_revision / 100 , chip_revision % 100 ) ;
result + = revision ; // ESP32-C3 v0.3
2022-04-24 10:24:01 +01:00
return result ;
2022-04-23 16:59:37 +01:00
}
2022-10-04 16:52:15 +01:00
String GetCodeCores ( void ) {
# if defined(CORE32SOLO1)
return F ( " single-core " ) ;
# else
return F ( " " ) ;
# endif
}
2021-07-18 18:43:33 +01:00
/*
* ESP32 v1 and v2 needs some special patches to use PSRAM .
* Standard Tasmota 32 do not include those patches .
* If using ESP32 v1 , please add : ` - mfix - esp32 - psram - cache - issue - lc - psram - workaround - lm - psram - workaround `
2021-09-03 10:19:45 +01:00
*
* This function returns true if the chip supports PSRAM natively ( v3 ) or if the
2021-07-18 18:43:33 +01:00
* patches are present .
*/
bool CanUsePSRAM ( void ) {
2021-09-01 21:48:02 +01:00
if ( ! FoundPSRAM ( ) ) return false ;
2021-07-18 18:43:33 +01:00
# ifdef HAS_PSRAM_FIX
return true ;
# endif
# ifdef CONFIG_IDF_TARGET_ESP32
esp_chip_info_t chip_info ;
esp_chip_info ( & chip_info ) ;
2023-08-25 09:55:20 +01:00
uint32_t chip_revision = chip_info . revision ;
// idf5 efuse_hal_chip_revision(void)
2023-08-25 11:48:35 +01:00
if ( chip_revision < 100 ) { chip_revision * = 100 ; } // Make <idf5 idf5
if ( ( CHIP_ESP32 = = chip_info . model ) & & ( chip_revision < 300 ) ) {
2021-07-18 18:43:33 +01:00
return false ;
}
# endif // CONFIG_IDF_TARGET_ESP32
return true ;
}
2021-02-08 10:34:29 +00:00
# endif // ESP32
2021-04-17 14:57:35 +01:00
/*********************************************************************************************\
* ESP Support
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2021-02-08 10:34:29 +00:00
uint32_t ESP_getFreeHeap1024 ( void ) {
return ESP_getFreeHeap ( ) / 1024 ;
}
/*
float ESP_getFreeHeap1024 ( void ) {
return ( ( float ) ESP_getFreeHeap ( ) ) / 1024 ;
}
*/
/*********************************************************************************************\
* High entropy hardware random generator
* Thanks to DigitalAlchemist
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
// Based on code from https://raw.githubusercontent.com/espressif/esp-idf/master/components/esp32/hw_random.c
uint32_t HwRandom ( void ) {
# if ESP8266
// https://web.archive.org/web/20160922031242/http://esp8266-re.foogod.com/wiki/Random_Number_Generator
# define _RAND_ADDR 0x3FF20E44UL
# endif // ESP8266
# ifdef ESP32
# define _RAND_ADDR 0x3FF75144UL
# endif // ESP32
static uint32_t last_ccount = 0 ;
uint32_t ccount ;
uint32_t result = 0 ;
do {
ccount = ESP . getCycleCount ( ) ;
result ^ = * ( volatile uint32_t * ) _RAND_ADDR ;
} while ( ccount - last_ccount < 64 ) ;
last_ccount = ccount ;
return result ^ * ( volatile uint32_t * ) _RAND_ADDR ;
# undef _RAND_ADDR
2022-12-18 13:06:04 +00:00
}