2014-12-30 19:47:44 +01:00
|
|
|
/******************************************************************************
|
|
|
|
* Flash api for NodeMCU
|
2014-12-31 01:08:31 +01:00
|
|
|
* NodeMCU Team
|
|
|
|
* 2014-12-31
|
2014-12-30 19:47:44 +01:00
|
|
|
*******************************************************************************/
|
|
|
|
#include "user_config.h"
|
|
|
|
#include "flash_api.h"
|
|
|
|
#include "spi_flash.h"
|
2016-05-26 10:36:20 +02:00
|
|
|
#include <stdio.h>
|
2014-12-30 19:47:44 +01:00
|
|
|
|
2015-03-15 17:48:28 +01:00
|
|
|
uint32_t flash_detect_size_byte(void)
|
|
|
|
{
|
2015-03-15 17:51:47 +01:00
|
|
|
#define FLASH_BUFFER_SIZE_DETECT 32
|
2015-03-15 17:48:28 +01:00
|
|
|
uint32_t dummy_size = FLASH_SIZE_256KBYTE;
|
2015-03-15 17:51:47 +01:00
|
|
|
uint8_t data_orig[FLASH_BUFFER_SIZE_DETECT] ICACHE_STORE_ATTR = {0};
|
|
|
|
uint8_t data_new[FLASH_BUFFER_SIZE_DETECT] ICACHE_STORE_ATTR = {0};
|
|
|
|
if (SPI_FLASH_RESULT_OK == flash_safe_read(0, (uint32 *)data_orig, FLASH_BUFFER_SIZE_DETECT))
|
2015-03-15 17:48:28 +01:00
|
|
|
{
|
|
|
|
dummy_size = FLASH_SIZE_256KBYTE;
|
|
|
|
while ((dummy_size < FLASH_SIZE_16MBYTE) &&
|
2015-03-15 17:51:47 +01:00
|
|
|
(SPI_FLASH_RESULT_OK == flash_safe_read(dummy_size, (uint32 *)data_new, FLASH_BUFFER_SIZE_DETECT)) &&
|
2016-05-26 10:36:20 +02:00
|
|
|
(0 != memcmp(data_orig, data_new, FLASH_BUFFER_SIZE_DETECT))
|
2015-03-15 17:51:47 +01:00
|
|
|
)
|
2015-03-15 17:48:28 +01:00
|
|
|
{
|
|
|
|
dummy_size *= 2;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
return dummy_size;
|
2015-03-15 17:51:47 +01:00
|
|
|
#undef FLASH_BUFFER_SIZE_DETECT
|
2015-03-15 17:48:28 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t flash_safe_get_size_byte(void)
|
|
|
|
{
|
|
|
|
static uint32_t flash_size = 0;
|
|
|
|
if (flash_size == 0)
|
|
|
|
{
|
|
|
|
flash_size = flash_detect_size_byte();
|
|
|
|
}
|
|
|
|
return flash_size;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t flash_safe_get_sec_num(void)
|
|
|
|
{
|
|
|
|
return (flash_safe_get_size_byte() / (SPI_FLASH_SEC_SIZE));
|
|
|
|
}
|
|
|
|
|
|
|
|
SpiFlashOpResult flash_safe_read(uint32 src_addr, uint32 *des_addr, uint32 size)
|
|
|
|
{
|
|
|
|
SpiFlashOpResult result = SPI_FLASH_RESULT_ERR;
|
|
|
|
FLASH_SAFEMODE_ENTER();
|
|
|
|
result = spi_flash_read(src_addr, (uint32 *) des_addr, size);
|
|
|
|
FLASH_SAFEMODE_LEAVE();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
SpiFlashOpResult flash_safe_write(uint32 des_addr, uint32 *src_addr, uint32 size)
|
|
|
|
{
|
|
|
|
SpiFlashOpResult result = SPI_FLASH_RESULT_ERR;
|
|
|
|
FLASH_SAFEMODE_ENTER();
|
|
|
|
result = spi_flash_write(des_addr, src_addr, size);
|
|
|
|
FLASH_SAFEMODE_LEAVE();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
SpiFlashOpResult flash_safe_erase_sector(uint16 sec)
|
|
|
|
{
|
|
|
|
SpiFlashOpResult result = SPI_FLASH_RESULT_ERR;
|
|
|
|
FLASH_SAFEMODE_ENTER();
|
|
|
|
result = spi_flash_erase_sector(sec);
|
|
|
|
FLASH_SAFEMODE_LEAVE();
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
SPIFlashInfo flash_rom_getinfo(void)
|
2015-01-05 03:09:51 +01:00
|
|
|
{
|
|
|
|
volatile SPIFlashInfo spi_flash_info ICACHE_STORE_ATTR;
|
2015-03-15 17:51:47 +01:00
|
|
|
spi_flash_read(0, (uint32 *)(& spi_flash_info), sizeof(spi_flash_info));
|
2015-01-05 03:09:51 +01:00
|
|
|
return spi_flash_info;
|
2014-12-30 19:47:44 +01:00
|
|
|
}
|
|
|
|
|
2015-03-15 17:48:28 +01:00
|
|
|
uint8_t flash_rom_get_size_type(void)
|
2014-12-31 01:08:31 +01:00
|
|
|
{
|
2015-03-15 17:48:28 +01:00
|
|
|
return flash_rom_getinfo().size;
|
2014-12-31 01:08:31 +01:00
|
|
|
}
|
|
|
|
|
2015-03-15 17:48:28 +01:00
|
|
|
uint32_t flash_rom_get_size_byte(void)
|
2014-12-30 19:47:44 +01:00
|
|
|
{
|
2015-03-15 17:51:47 +01:00
|
|
|
static uint32_t flash_size = 0;
|
|
|
|
if (flash_size == 0)
|
2014-12-30 19:47:44 +01:00
|
|
|
{
|
2015-03-15 17:51:47 +01:00
|
|
|
switch (flash_rom_getinfo().size)
|
|
|
|
{
|
|
|
|
case SIZE_2MBIT:
|
|
|
|
// 2Mbit, 256kByte
|
|
|
|
flash_size = 256 * 1024;
|
|
|
|
break;
|
|
|
|
case SIZE_4MBIT:
|
|
|
|
// 4Mbit, 512kByte
|
|
|
|
flash_size = 512 * 1024;
|
|
|
|
break;
|
|
|
|
case SIZE_8MBIT:
|
|
|
|
// 8Mbit, 1MByte
|
|
|
|
flash_size = 1 * 1024 * 1024;
|
|
|
|
break;
|
|
|
|
case SIZE_16MBIT:
|
|
|
|
// 16Mbit, 2MByte
|
|
|
|
flash_size = 2 * 1024 * 1024;
|
|
|
|
break;
|
|
|
|
case SIZE_32MBIT:
|
|
|
|
// 32Mbit, 4MByte
|
|
|
|
flash_size = 4 * 1024 * 1024;
|
|
|
|
break;
|
2015-12-30 02:49:01 +01:00
|
|
|
case SIZE_16MBIT_8M_8M:
|
|
|
|
// 16Mbit, 2MByte
|
|
|
|
flash_size = 2 * 1024 * 1024;
|
2015-03-15 17:51:47 +01:00
|
|
|
break;
|
2015-12-30 02:49:01 +01:00
|
|
|
case SIZE_32MBIT_8M_8M:
|
|
|
|
// 32Mbit, 4MByte
|
|
|
|
flash_size = 4 * 1024 * 1024;
|
2015-03-15 17:51:47 +01:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
// Unknown flash size, fall back mode.
|
|
|
|
flash_size = 512 * 1024;
|
|
|
|
break;
|
|
|
|
}
|
2014-12-30 19:47:44 +01:00
|
|
|
}
|
|
|
|
return flash_size;
|
|
|
|
}
|
|
|
|
|
2015-03-15 17:48:28 +01:00
|
|
|
bool flash_rom_set_size_type(uint8_t size)
|
2014-12-31 01:08:31 +01:00
|
|
|
{
|
|
|
|
// Dangerous, here are dinosaur infested!!!!!
|
|
|
|
// Reboot required!!!
|
|
|
|
// If you don't know what you're doing, your nodemcu may turn into stone ...
|
2015-03-15 22:40:43 +01:00
|
|
|
NODE_DBG("\nBEGIN SET FLASH HEADER\n");
|
2016-05-30 09:56:33 +02:00
|
|
|
uint8_t *data = malloc (SPI_FLASH_SEC_SIZE);
|
|
|
|
if (!data)
|
|
|
|
return false;
|
|
|
|
|
2015-03-15 22:40:43 +01:00
|
|
|
if (SPI_FLASH_RESULT_OK == spi_flash_read(0, (uint32 *)data, SPI_FLASH_SEC_SIZE))
|
|
|
|
{
|
|
|
|
((SPIFlashInfo *)(&data[0]))->size = size;
|
|
|
|
if (SPI_FLASH_RESULT_OK == spi_flash_erase_sector(0 * SPI_FLASH_SEC_SIZE))
|
|
|
|
{
|
|
|
|
NODE_DBG("\nERASE SUCCESS\n");
|
|
|
|
}
|
|
|
|
if (SPI_FLASH_RESULT_OK == spi_flash_write(0, (uint32 *)data, SPI_FLASH_SEC_SIZE))
|
|
|
|
{
|
|
|
|
NODE_DBG("\nWRITE SUCCESS, %u\n", size);
|
|
|
|
}
|
|
|
|
}
|
2016-05-30 09:56:33 +02:00
|
|
|
free (data);
|
2015-03-15 22:40:43 +01:00
|
|
|
NODE_DBG("\nEND SET FLASH HEADER\n");
|
2014-12-31 01:08:31 +01:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-03-15 17:48:28 +01:00
|
|
|
bool flash_rom_set_size_byte(uint32_t size)
|
2014-12-31 01:08:31 +01:00
|
|
|
{
|
|
|
|
// Dangerous, here are dinosaur infested!!!!!
|
|
|
|
// Reboot required!!!
|
|
|
|
// If you don't know what you're doing, your nodemcu may turn into stone ...
|
|
|
|
bool result = true;
|
|
|
|
uint32_t flash_size = 0;
|
|
|
|
switch (size)
|
|
|
|
{
|
|
|
|
case 256 * 1024:
|
|
|
|
// 2Mbit, 256kByte
|
|
|
|
flash_size = SIZE_2MBIT;
|
2015-03-15 17:48:28 +01:00
|
|
|
flash_rom_set_size_type(flash_size);
|
2014-12-31 01:08:31 +01:00
|
|
|
break;
|
|
|
|
case 512 * 1024:
|
|
|
|
// 4Mbit, 512kByte
|
|
|
|
flash_size = SIZE_4MBIT;
|
2015-03-15 17:48:28 +01:00
|
|
|
flash_rom_set_size_type(flash_size);
|
2014-12-31 01:08:31 +01:00
|
|
|
break;
|
|
|
|
case 1 * 1024 * 1024:
|
|
|
|
// 8Mbit, 1MByte
|
|
|
|
flash_size = SIZE_8MBIT;
|
2015-03-15 17:48:28 +01:00
|
|
|
flash_rom_set_size_type(flash_size);
|
2014-12-31 01:08:31 +01:00
|
|
|
break;
|
|
|
|
case 2 * 1024 * 1024:
|
|
|
|
// 16Mbit, 2MByte
|
|
|
|
flash_size = SIZE_16MBIT;
|
2015-03-15 17:48:28 +01:00
|
|
|
flash_rom_set_size_type(flash_size);
|
2014-12-31 01:08:31 +01:00
|
|
|
break;
|
|
|
|
case 4 * 1024 * 1024:
|
|
|
|
// 32Mbit, 4MByte
|
|
|
|
flash_size = SIZE_32MBIT;
|
2015-03-15 17:48:28 +01:00
|
|
|
flash_rom_set_size_type(flash_size);
|
2014-12-31 01:08:31 +01:00
|
|
|
break;
|
2015-12-30 02:49:01 +01:00
|
|
|
/*
|
2015-03-15 22:40:43 +01:00
|
|
|
case 8 * 1024 * 1024:
|
|
|
|
// 64Mbit, 8MByte
|
2015-12-30 02:49:01 +01:00
|
|
|
flash_size = SIZE_16MBIT_8M_8M;
|
2015-03-15 22:40:43 +01:00
|
|
|
flash_rom_set_size_type(flash_size);
|
|
|
|
break;
|
|
|
|
case 16 * 1024 * 1024:
|
|
|
|
// 128Mbit, 16MByte
|
2015-12-30 02:49:01 +01:00
|
|
|
flash_size = SIZE_32MBIT_8M_8M;
|
2015-03-15 22:40:43 +01:00
|
|
|
flash_rom_set_size_type(flash_size);
|
|
|
|
break;
|
2015-12-30 02:49:01 +01:00
|
|
|
*/
|
2014-12-31 01:08:31 +01:00
|
|
|
default:
|
|
|
|
// Unknown flash size.
|
|
|
|
result = false;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2015-03-15 17:48:28 +01:00
|
|
|
uint16_t flash_rom_get_sec_num(void)
|
2014-12-30 19:47:44 +01:00
|
|
|
{
|
2015-02-14 19:13:58 +01:00
|
|
|
//static uint16_t sec_num = 0;
|
2015-03-15 17:48:28 +01:00
|
|
|
// return flash_rom_get_size_byte() / (SPI_FLASH_SEC_SIZE);
|
2016-05-26 10:36:20 +02:00
|
|
|
// printf("\nflash_rom_get_size_byte()=%d\n", ( flash_rom_get_size_byte() / (SPI_FLASH_SEC_SIZE) ));
|
2015-02-14 19:13:58 +01:00
|
|
|
// if( sec_num == 0 )
|
|
|
|
//{
|
|
|
|
// sec_num = 4 * 1024 * 1024 / (SPI_FLASH_SEC_SIZE);
|
|
|
|
//}
|
|
|
|
//return sec_num;
|
2015-03-15 17:48:28 +01:00
|
|
|
return ( flash_rom_get_size_byte() / (SPI_FLASH_SEC_SIZE) );
|
2014-12-30 19:51:36 +01:00
|
|
|
}
|
2014-12-31 01:08:31 +01:00
|
|
|
|
2015-03-15 17:48:28 +01:00
|
|
|
uint8_t flash_rom_get_mode(void)
|
2014-12-31 01:08:31 +01:00
|
|
|
{
|
2015-03-15 17:48:28 +01:00
|
|
|
SPIFlashInfo spi_flash_info = flash_rom_getinfo();
|
2015-01-05 03:09:51 +01:00
|
|
|
switch (spi_flash_info.mode)
|
2014-12-31 01:08:31 +01:00
|
|
|
{
|
|
|
|
// Reserved for future use
|
|
|
|
case MODE_QIO:
|
|
|
|
break;
|
|
|
|
case MODE_QOUT:
|
|
|
|
break;
|
|
|
|
case MODE_DIO:
|
|
|
|
break;
|
|
|
|
case MODE_DOUT:
|
|
|
|
break;
|
|
|
|
}
|
2015-01-05 03:09:51 +01:00
|
|
|
return spi_flash_info.mode;
|
2014-12-31 01:08:31 +01:00
|
|
|
}
|
|
|
|
|
2015-03-15 17:48:28 +01:00
|
|
|
uint32_t flash_rom_get_speed(void)
|
2014-12-31 01:08:31 +01:00
|
|
|
{
|
|
|
|
uint32_t speed = 0;
|
2015-03-15 17:48:28 +01:00
|
|
|
SPIFlashInfo spi_flash_info = flash_rom_getinfo();
|
2015-01-05 03:09:51 +01:00
|
|
|
switch (spi_flash_info.speed)
|
2014-12-31 01:08:31 +01:00
|
|
|
{
|
|
|
|
case SPEED_40MHZ:
|
|
|
|
// 40MHz
|
|
|
|
speed = 40000000;
|
|
|
|
break;
|
|
|
|
case SPEED_26MHZ:
|
|
|
|
//26.7MHz
|
|
|
|
speed = 26700000;
|
|
|
|
break;
|
|
|
|
case SPEED_20MHZ:
|
|
|
|
// 20MHz
|
|
|
|
speed = 20000000;
|
|
|
|
break;
|
|
|
|
case SPEED_80MHZ:
|
|
|
|
//80MHz
|
|
|
|
speed = 80000000;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return speed;
|
|
|
|
}
|
|
|
|
|
2015-03-15 22:40:43 +01:00
|
|
|
bool flash_rom_set_speed(uint32_t speed)
|
|
|
|
{
|
|
|
|
// Dangerous, here are dinosaur infested!!!!!
|
|
|
|
// Reboot required!!!
|
|
|
|
// If you don't know what you're doing, your nodemcu may turn into stone ...
|
|
|
|
NODE_DBG("\nBEGIN SET FLASH HEADER\n");
|
2016-05-30 09:56:33 +02:00
|
|
|
uint8_t *data = malloc (SPI_FLASH_SEC_SIZE);
|
|
|
|
if (!data)
|
|
|
|
return false;
|
2015-03-15 22:40:43 +01:00
|
|
|
uint8_t speed_type = SPEED_40MHZ;
|
|
|
|
if (speed < 26700000)
|
|
|
|
{
|
|
|
|
speed_type = SPEED_20MHZ;
|
|
|
|
}
|
|
|
|
else if (speed < 40000000)
|
|
|
|
{
|
|
|
|
speed_type = SPEED_26MHZ;
|
|
|
|
}
|
|
|
|
else if (speed < 80000000)
|
|
|
|
{
|
|
|
|
speed_type = SPEED_40MHZ;
|
|
|
|
}
|
|
|
|
else if (speed >= 80000000)
|
|
|
|
{
|
|
|
|
speed_type = SPEED_80MHZ;
|
|
|
|
}
|
|
|
|
if (SPI_FLASH_RESULT_OK == spi_flash_read(0, (uint32 *)data, SPI_FLASH_SEC_SIZE))
|
|
|
|
{
|
|
|
|
((SPIFlashInfo *)(&data[0]))->speed = speed_type;
|
|
|
|
if (SPI_FLASH_RESULT_OK == spi_flash_erase_sector(0 * SPI_FLASH_SEC_SIZE))
|
|
|
|
{
|
|
|
|
NODE_DBG("\nERASE SUCCESS\n");
|
|
|
|
}
|
|
|
|
if (SPI_FLASH_RESULT_OK == spi_flash_write(0, (uint32 *)data, SPI_FLASH_SEC_SIZE))
|
|
|
|
{
|
|
|
|
NODE_DBG("\nWRITE SUCCESS, %u\n", speed_type);
|
|
|
|
}
|
|
|
|
}
|
2016-05-30 09:56:33 +02:00
|
|
|
free (data);
|
2015-03-15 22:40:43 +01:00
|
|
|
NODE_DBG("\nEND SET FLASH HEADER\n");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-02-14 17:02:53 +01:00
|
|
|
uint8_t byte_of_aligned_array(const uint8_t *aligned_array, uint32_t index)
|
2015-01-06 18:15:38 +01:00
|
|
|
{
|
2015-02-14 17:02:53 +01:00
|
|
|
if ( (((uint32_t)aligned_array) % 4) != 0 )
|
|
|
|
{
|
2015-01-06 18:15:38 +01:00
|
|
|
NODE_DBG("aligned_array is not 4-byte aligned.\n");
|
|
|
|
return 0;
|
|
|
|
}
|
2015-05-05 17:59:30 +02:00
|
|
|
volatile uint32_t v = ((uint32_t *)aligned_array)[ index / 4 ];
|
2015-01-06 18:15:38 +01:00
|
|
|
uint8_t *p = (uint8_t *) (&v);
|
2015-02-14 17:02:53 +01:00
|
|
|
return p[ (index % 4) ];
|
2015-01-06 18:15:38 +01:00
|
|
|
}
|
2015-03-15 22:40:43 +01:00
|
|
|
|
2015-05-05 17:59:30 +02:00
|
|
|
uint16_t word_of_aligned_array(const uint16_t *aligned_array, uint32_t index)
|
|
|
|
{
|
|
|
|
if ( (((uint32_t)aligned_array) % 4) != 0 )
|
|
|
|
{
|
|
|
|
NODE_DBG("aligned_array is not 4-byte aligned.\n");
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
volatile uint32_t v = ((uint32_t *)aligned_array)[ index / 2 ];
|
|
|
|
uint16_t *p = (uint16_t *) (&v);
|
|
|
|
return (index % 2 == 0) ? p[ 0 ] : p[ 1 ];
|
|
|
|
// return p[ (index % 2) ]; // -- why error???
|
|
|
|
// (byte_of_aligned_array((uint8_t *)aligned_array, index * 2 + 1) << 8) | byte_of_aligned_array((uint8_t *)aligned_array, index * 2);
|
|
|
|
}
|
|
|
|
|
2015-03-15 22:40:43 +01:00
|
|
|
// uint8_t flash_rom_get_checksum(void)
|
|
|
|
// {
|
|
|
|
// // SPIFlashInfo spi_flash_info ICACHE_STORE_ATTR = flash_rom_getinfo();
|
|
|
|
// // uint32_t address = sizeof(spi_flash_info) + spi_flash_info.segment_size;
|
|
|
|
// // uint32_t address_aligned_4bytes = (address + 3) & 0xFFFFFFFC;
|
|
|
|
// // uint8_t buffer[64] = {0};
|
|
|
|
// // spi_flash_read(address, (uint32 *) buffer, 64);
|
|
|
|
// // uint8_t i = 0;
|
2016-05-26 10:36:20 +02:00
|
|
|
// // printf("\nBEGIN DUMP\n");
|
2015-03-15 22:40:43 +01:00
|
|
|
// // for (i = 0; i < 64; i++)
|
|
|
|
// // {
|
2016-05-26 10:36:20 +02:00
|
|
|
// // printf("%02x," , buffer[i]);
|
2015-03-15 22:40:43 +01:00
|
|
|
// // }
|
|
|
|
// // i = (address + 0x10) & 0x10 - 1;
|
2016-05-26 10:36:20 +02:00
|
|
|
// // printf("\nSIZE:%d CHECK SUM:%02x\n", spi_flash_info.segment_size, buffer[i]);
|
|
|
|
// // printf("\nEND DUMP\n");
|
2015-03-15 22:40:43 +01:00
|
|
|
// // return buffer[0];
|
|
|
|
// return 0;
|
|
|
|
// }
|
|
|
|
|
|
|
|
// uint8_t flash_rom_calc_checksum(void)
|
|
|
|
// {
|
|
|
|
// return 0;
|
2015-11-12 02:57:07 +01:00
|
|
|
// }
|