mirror of https://github.com/arendst/Tasmota.git
409 lines
11 KiB
C++
409 lines
11 KiB
C++
#include "TasUpdate.h"
|
|
#include "Arduino.h"
|
|
#if ESP_IDF_VERSION_MAJOR >= 5
|
|
#include "spi_flash_mmap.h"
|
|
#else
|
|
#include "esp_spi_flash.h"
|
|
#endif
|
|
#include "esp_ota_ops.h"
|
|
#include "esp_image_format.h"
|
|
|
|
static const char * _err2str(uint8_t _error){
|
|
if(_error == UPDATE_ERROR_OK){
|
|
return ("No Error");
|
|
} else if(_error == UPDATE_ERROR_WRITE){
|
|
return ("Flash Write Failed");
|
|
} else if(_error == UPDATE_ERROR_ERASE){
|
|
return ("Flash Erase Failed");
|
|
} else if(_error == UPDATE_ERROR_READ){
|
|
return ("Flash Read Failed");
|
|
} else if(_error == UPDATE_ERROR_SPACE){
|
|
return ("Not Enough Space");
|
|
} else if(_error == UPDATE_ERROR_SIZE){
|
|
return ("Bad Size Given");
|
|
} else if(_error == UPDATE_ERROR_STREAM){
|
|
return ("Stream Read Timeout");
|
|
} else if(_error == UPDATE_ERROR_MD5){
|
|
return ("MD5 Check Failed");
|
|
} else if(_error == UPDATE_ERROR_MAGIC_BYTE){
|
|
return ("Wrong Magic Byte");
|
|
} else if(_error == UPDATE_ERROR_ACTIVATE){
|
|
return ("Could Not Activate The Firmware");
|
|
} else if(_error == UPDATE_ERROR_NO_PARTITION){
|
|
return ("Partition Could Not be Found");
|
|
} else if(_error == UPDATE_ERROR_BAD_ARGUMENT){
|
|
return ("Bad Argument");
|
|
} else if(_error == UPDATE_ERROR_ABORT){
|
|
return ("Aborted");
|
|
}
|
|
return ("UNKNOWN");
|
|
}
|
|
|
|
static bool _partitionIsBootable(const esp_partition_t* partition){
|
|
uint8_t buf[ENCRYPTED_BLOCK_SIZE];
|
|
if(!partition){
|
|
return false;
|
|
}
|
|
if(!ESP.partitionRead(partition, 0, (uint32_t*)buf, ENCRYPTED_BLOCK_SIZE)) {
|
|
return false;
|
|
}
|
|
|
|
if(buf[0] != ESP_IMAGE_HEADER_MAGIC) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool TasUpdateClass::_enablePartition(const esp_partition_t* partition){
|
|
if(!partition){
|
|
return false;
|
|
}
|
|
return ESP.partitionWrite(partition, 0, (uint32_t*) _skipBuffer, ENCRYPTED_BLOCK_SIZE);
|
|
}
|
|
|
|
TasUpdateClass::TasUpdateClass()
|
|
: _error(0)
|
|
, _buffer(0)
|
|
, _bufferLen(0)
|
|
, _size(0)
|
|
, _progress_callback(NULL)
|
|
, _progress(0)
|
|
, _paroffset(0)
|
|
, _command(U_FLASH)
|
|
, _partition(NULL)
|
|
{
|
|
}
|
|
|
|
TasUpdateClass& TasUpdateClass::onProgress(THandlerFunction_Progress fn) {
|
|
_progress_callback = fn;
|
|
return *this;
|
|
}
|
|
|
|
void TasUpdateClass::_reset() {
|
|
if (_buffer)
|
|
delete[] _buffer;
|
|
_buffer = 0;
|
|
_bufferLen = 0;
|
|
_progress = 0;
|
|
_size = 0;
|
|
_command = U_FLASH;
|
|
|
|
if(_ledPin != -1) {
|
|
digitalWrite(_ledPin, !_ledOn); // off
|
|
}
|
|
}
|
|
|
|
bool TasUpdateClass::canRollBack(){
|
|
if(_buffer){ //Update is running
|
|
return false;
|
|
}
|
|
const esp_partition_t* partition = esp_ota_get_next_update_partition(NULL);
|
|
return _partitionIsBootable(partition);
|
|
}
|
|
|
|
bool TasUpdateClass::rollBack(){
|
|
if(_buffer){ //Update is running
|
|
return false;
|
|
}
|
|
const esp_partition_t* partition = esp_ota_get_next_update_partition(NULL);
|
|
return _partitionIsBootable(partition) && !esp_ota_set_boot_partition(partition);
|
|
}
|
|
|
|
// Start Tasmota Factory patch
|
|
//bool UpdateClass::begin(size_t size, int command, int ledPin, uint8_t ledOn, const char *label) {
|
|
bool TasUpdateClass::begin(size_t size, int command, int ledPin, uint8_t ledOn, const char *label, bool factory) {
|
|
// End Tasmota Factory patch
|
|
if(_size > 0){
|
|
log_w("already running");
|
|
return false;
|
|
}
|
|
|
|
_ledPin = ledPin;
|
|
_ledOn = !!ledOn; // 0(LOW) or 1(HIGH)
|
|
|
|
_reset();
|
|
_error = 0;
|
|
_target_md5 = emptyString;
|
|
_md5 = MD5Builder();
|
|
|
|
if(size == 0) {
|
|
_error = UPDATE_ERROR_SIZE;
|
|
return false;
|
|
}
|
|
|
|
if (command == U_FLASH) {
|
|
// Start Tasmota Factory patch
|
|
// _partition = esp_ota_get_next_update_partition(NULL);
|
|
if (factory) {
|
|
_partition = esp_partition_find_first(ESP_PARTITION_TYPE_APP, ESP_PARTITION_SUBTYPE_APP_FACTORY, NULL);
|
|
} else {
|
|
_partition = esp_ota_get_next_update_partition(NULL);
|
|
}
|
|
// End Tasmota Factory patch
|
|
if(!_partition){
|
|
_error = UPDATE_ERROR_NO_PARTITION;
|
|
return false;
|
|
}
|
|
log_d("OTA Partition: %s", _partition->label);
|
|
}
|
|
else if (command == U_SPIFFS) {
|
|
_partition = esp_partition_find_first(ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_SPIFFS, label);
|
|
_paroffset = 0;
|
|
if(!_partition){
|
|
_partition = esp_partition_find_first(ESP_PARTITION_TYPE_DATA, ESP_PARTITION_SUBTYPE_DATA_FAT, NULL);
|
|
_paroffset = 0x1000; //Offset for ffat, assuming size is already corrected
|
|
if(!_partition){
|
|
_error = UPDATE_ERROR_NO_PARTITION;
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
else {
|
|
_error = UPDATE_ERROR_BAD_ARGUMENT;
|
|
log_e("bad command %u", command);
|
|
return false;
|
|
}
|
|
|
|
if(size == UPDATE_SIZE_UNKNOWN){
|
|
size = _partition->size;
|
|
} else if(size > _partition->size){
|
|
_error = UPDATE_ERROR_SIZE;
|
|
log_e("too large %u > %u", size, _partition->size);
|
|
return false;
|
|
}
|
|
|
|
//initialize
|
|
_buffer = (uint8_t*)malloc(SPI_FLASH_SEC_SIZE);
|
|
if(!_buffer){
|
|
log_e("malloc failed");
|
|
return false;
|
|
}
|
|
_size = size;
|
|
_command = command;
|
|
_md5.begin();
|
|
return true;
|
|
}
|
|
|
|
void TasUpdateClass::_abort(uint8_t err){
|
|
_reset();
|
|
_error = err;
|
|
}
|
|
|
|
void TasUpdateClass::abort(){
|
|
_abort(UPDATE_ERROR_ABORT);
|
|
}
|
|
|
|
bool TasUpdateClass::_writeBuffer(){
|
|
//first bytes of new firmware
|
|
uint8_t skip = 0;
|
|
if(!_progress && _command == U_FLASH){
|
|
//check magic
|
|
if(_buffer[0] != ESP_IMAGE_HEADER_MAGIC){
|
|
_abort(UPDATE_ERROR_MAGIC_BYTE);
|
|
return false;
|
|
}
|
|
|
|
//Stash the first 16 bytes of data and set the offset so they are
|
|
//not written at this point so that partially written firmware
|
|
//will not be bootable
|
|
skip = ENCRYPTED_BLOCK_SIZE;
|
|
_skipBuffer = (uint8_t*)malloc(skip);
|
|
if(!_skipBuffer){
|
|
log_e("malloc failed");
|
|
return false;
|
|
}
|
|
memcpy(_skipBuffer, _buffer, skip);
|
|
}
|
|
if (!_progress && _progress_callback) {
|
|
_progress_callback(0, _size);
|
|
}
|
|
if(!ESP.partitionEraseRange(_partition, _progress, SPI_FLASH_SEC_SIZE)){
|
|
_abort(UPDATE_ERROR_ERASE);
|
|
return false;
|
|
}
|
|
if (!ESP.partitionWrite(_partition, _progress + skip, (uint32_t*)_buffer + skip/sizeof(uint32_t), _bufferLen - skip)) {
|
|
_abort(UPDATE_ERROR_WRITE);
|
|
return false;
|
|
}
|
|
//restore magic or md5 will fail
|
|
if(!_progress && _command == U_FLASH){
|
|
_buffer[0] = ESP_IMAGE_HEADER_MAGIC;
|
|
}
|
|
_md5.add(_buffer, _bufferLen);
|
|
_progress += _bufferLen;
|
|
_bufferLen = 0;
|
|
if (_progress_callback) {
|
|
_progress_callback(_progress, _size);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool TasUpdateClass::_verifyHeader(uint8_t data) {
|
|
if(_command == U_FLASH) {
|
|
if(data != ESP_IMAGE_HEADER_MAGIC) {
|
|
_abort(UPDATE_ERROR_MAGIC_BYTE);
|
|
return false;
|
|
}
|
|
return true;
|
|
} else if(_command == U_SPIFFS) {
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool TasUpdateClass::_verifyEnd() {
|
|
if(_command == U_FLASH) {
|
|
if(!_enablePartition(_partition) || !_partitionIsBootable(_partition)) {
|
|
_abort(UPDATE_ERROR_READ);
|
|
return false;
|
|
}
|
|
|
|
if(esp_ota_set_boot_partition(_partition)){
|
|
_abort(UPDATE_ERROR_ACTIVATE);
|
|
return false;
|
|
}
|
|
_reset();
|
|
return true;
|
|
} else if(_command == U_SPIFFS) {
|
|
_reset();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool TasUpdateClass::setMD5(const char * expected_md5){
|
|
if(strlen(expected_md5) != 32)
|
|
{
|
|
return false;
|
|
}
|
|
_target_md5 = expected_md5;
|
|
return true;
|
|
}
|
|
|
|
bool TasUpdateClass::end(bool evenIfRemaining){
|
|
if(hasError() || _size == 0){
|
|
return false;
|
|
}
|
|
|
|
if(!isFinished() && !evenIfRemaining){
|
|
log_e("premature end: res:%u, pos:%u/%u\n", getError(), progress(), _size);
|
|
_abort(UPDATE_ERROR_ABORT);
|
|
return false;
|
|
}
|
|
|
|
if(evenIfRemaining) {
|
|
if(_bufferLen > 0) {
|
|
_writeBuffer();
|
|
}
|
|
_size = progress();
|
|
}
|
|
|
|
_md5.calculate();
|
|
if(_target_md5.length()) {
|
|
if(_target_md5 != _md5.toString()){
|
|
_abort(UPDATE_ERROR_MD5);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return _verifyEnd();
|
|
}
|
|
|
|
size_t TasUpdateClass::write(uint8_t *data, size_t len) {
|
|
if(hasError() || !isRunning()){
|
|
return 0;
|
|
}
|
|
|
|
if(len > remaining()){
|
|
_abort(UPDATE_ERROR_SPACE);
|
|
return 0;
|
|
}
|
|
|
|
size_t left = len;
|
|
|
|
while((_bufferLen + left) > SPI_FLASH_SEC_SIZE) {
|
|
size_t toBuff = SPI_FLASH_SEC_SIZE - _bufferLen;
|
|
memcpy(_buffer + _bufferLen, data + (len - left), toBuff);
|
|
_bufferLen += toBuff;
|
|
if(!_writeBuffer()){
|
|
return len - left;
|
|
}
|
|
left -= toBuff;
|
|
}
|
|
memcpy(_buffer + _bufferLen, data + (len - left), left);
|
|
_bufferLen += left;
|
|
if(_bufferLen == remaining()){
|
|
if(!_writeBuffer()){
|
|
return len - left;
|
|
}
|
|
}
|
|
return len;
|
|
}
|
|
|
|
size_t TasUpdateClass::writeStream(Stream &data) {
|
|
size_t written = 0;
|
|
size_t toRead = 0;
|
|
int timeout_failures = 0;
|
|
|
|
if(hasError() || !isRunning())
|
|
return 0;
|
|
|
|
if(!_verifyHeader(data.peek())) {
|
|
_reset();
|
|
return 0;
|
|
}
|
|
|
|
if(_ledPin != -1) {
|
|
pinMode(_ledPin, OUTPUT);
|
|
}
|
|
|
|
while(remaining()) {
|
|
if(_ledPin != -1) {
|
|
digitalWrite(_ledPin, _ledOn); // Switch LED on
|
|
}
|
|
size_t bytesToRead = SPI_FLASH_SEC_SIZE - _bufferLen;
|
|
if(bytesToRead > remaining()) {
|
|
bytesToRead = remaining();
|
|
}
|
|
|
|
/*
|
|
Init read&timeout counters and try to read, if read failed, increase counter,
|
|
wait 100ms and try to read again. If counter > 300 (30 sec), give up/abort
|
|
*/
|
|
toRead = 0;
|
|
timeout_failures = 0;
|
|
while(!toRead) {
|
|
toRead = data.readBytes(_buffer + _bufferLen, bytesToRead);
|
|
if(toRead == 0) {
|
|
timeout_failures++;
|
|
if (timeout_failures >= 300) {
|
|
_abort(UPDATE_ERROR_STREAM);
|
|
return written;
|
|
}
|
|
delay(100);
|
|
}
|
|
}
|
|
|
|
if(_ledPin != -1) {
|
|
digitalWrite(_ledPin, !_ledOn); // Switch LED off
|
|
}
|
|
_bufferLen += toRead;
|
|
if((_bufferLen == remaining() || _bufferLen == SPI_FLASH_SEC_SIZE) && !_writeBuffer())
|
|
return written;
|
|
written += toRead;
|
|
|
|
delay(1); // Fix solo WDT
|
|
}
|
|
return written;
|
|
}
|
|
|
|
void TasUpdateClass::printError(Print &out){
|
|
out.println(_err2str(_error));
|
|
}
|
|
|
|
const char * TasUpdateClass::errorString(){
|
|
return _err2str(_error);
|
|
}
|
|
|
|
TasUpdateClass TasUpdate;
|