2019-01-08 10:29:25 +00:00
|
|
|
// Copyright 2015-2019 Espressif Systems (Shanghai) PTE LTD
|
|
|
|
//
|
|
|
|
// 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.
|
|
|
|
|
|
|
|
#include "spi_flash_defs.h"
|
|
|
|
#include "memspi_host_driver.h"
|
|
|
|
#include "string.h"
|
|
|
|
#include "esp_log.h"
|
|
|
|
#include "cache_utils.h"
|
|
|
|
#include "esp_flash_partitions.h"
|
|
|
|
|
|
|
|
static const char TAG[] = "memspi";
|
|
|
|
static const spi_flash_host_driver_t esp_flash_default_host = ESP_FLASH_DEFAULT_HOST_DRIVER();
|
|
|
|
|
2019-11-28 01:20:00 +00:00
|
|
|
#ifdef CONFIG_IDF_TARGET_ESP32S2BETA
|
|
|
|
extern void spi_flash_hal_gpspi_poll_cmd_done(spi_flash_host_driver_t *driver);
|
|
|
|
extern esp_err_t spi_flash_hal_gpspi_device_config(spi_flash_host_driver_t *driver);
|
|
|
|
esp_err_t spi_flash_hal_gpspi_configure_host_io_mode(
|
|
|
|
spi_flash_host_driver_t *host,
|
|
|
|
uint32_t command,
|
|
|
|
uint32_t addr_bitlen,
|
|
|
|
int dummy_cyclelen_base,
|
|
|
|
esp_flash_io_mode_t io_mode);
|
|
|
|
extern esp_err_t spi_flash_hal_gpspi_common_command(spi_flash_host_driver_t *chip_drv, spi_flash_trans_t *trans);
|
|
|
|
extern esp_err_t spi_flash_hal_gpspi_read(spi_flash_host_driver_t *chip_drv, void *buffer, uint32_t address, uint32_t read_len);
|
|
|
|
extern bool spi_flash_hal_gpspi_host_idle(spi_flash_host_driver_t *chip_drv);
|
|
|
|
extern bool spi_flash_hal_gpspi_supports_direct_write(spi_flash_host_driver_t *driver, const void *p);
|
|
|
|
extern bool spi_flash_hal_gpspi_supports_direct_read(spi_flash_host_driver_t *driver, const void *p);
|
|
|
|
|
|
|
|
/** Default configuration for GPSPI */
|
|
|
|
static const spi_flash_host_driver_t esp_flash_gpspi_host = {
|
|
|
|
.dev_config = spi_flash_hal_gpspi_device_config,
|
|
|
|
.common_command = spi_flash_hal_gpspi_common_command,
|
|
|
|
.read_id = memspi_host_read_id_hs,
|
|
|
|
.erase_chip = memspi_host_erase_chip,
|
|
|
|
.erase_sector = memspi_host_erase_sector,
|
|
|
|
.erase_block = memspi_host_erase_block,
|
|
|
|
.read_status = memspi_host_read_status_hs,
|
|
|
|
.set_write_protect = memspi_host_set_write_protect,
|
|
|
|
.supports_direct_write = spi_flash_hal_gpspi_supports_direct_write,
|
|
|
|
.supports_direct_read = spi_flash_hal_gpspi_supports_direct_read,
|
|
|
|
.program_page = memspi_host_program_page,
|
|
|
|
.max_write_bytes = SPI_FLASH_HAL_MAX_WRITE_BYTES,
|
|
|
|
.read = spi_flash_hal_gpspi_read,
|
|
|
|
.max_read_bytes = SPI_FLASH_HAL_MAX_READ_BYTES,
|
|
|
|
.host_idle = spi_flash_hal_gpspi_host_idle,
|
|
|
|
.configure_host_io_mode = spi_flash_hal_gpspi_configure_host_io_mode,
|
|
|
|
.poll_cmd_done = spi_flash_hal_gpspi_poll_cmd_done,
|
|
|
|
.flush_cache = NULL,
|
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2019-01-08 10:29:25 +00:00
|
|
|
esp_err_t memspi_host_init_pointers(spi_flash_host_driver_t *host, memspi_host_data_t *data, const memspi_host_config_t *cfg)
|
|
|
|
{
|
2019-11-28 01:20:00 +00:00
|
|
|
#ifdef CONFIG_IDF_TARGET_ESP32
|
2019-01-08 10:29:25 +00:00
|
|
|
memcpy(host, &esp_flash_default_host, sizeof(spi_flash_host_driver_t));
|
2019-11-28 01:20:00 +00:00
|
|
|
#elif CONFIG_IDF_TARGET_ESP32S2BETA
|
|
|
|
if (cfg->host_id == SPI_HOST)
|
|
|
|
memcpy(host, &esp_flash_default_host, sizeof(spi_flash_host_driver_t));
|
|
|
|
else {
|
|
|
|
memcpy(host, &esp_flash_gpspi_host, sizeof(spi_flash_host_driver_t));
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-01-08 10:29:25 +00:00
|
|
|
esp_err_t err = spi_flash_hal_init(data, cfg);
|
|
|
|
if (err != ESP_OK) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
|
|
|
host->driver_data = data;
|
|
|
|
//some functions are not required if not SPI1
|
2019-11-28 01:20:00 +00:00
|
|
|
if ((void*)data->spi != (void*)spi_flash_ll_get_hw(SPI_HOST)) {
|
2019-01-08 10:29:25 +00:00
|
|
|
host->flush_cache = NULL;
|
|
|
|
}
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
2019-09-05 05:17:11 +00:00
|
|
|
esp_err_t memspi_host_read_id_hs(spi_flash_host_driver_t *host, uint32_t *id)
|
2019-01-08 10:29:25 +00:00
|
|
|
{
|
2019-11-28 01:20:00 +00:00
|
|
|
uint32_t id_buf = 0;
|
2019-01-08 10:29:25 +00:00
|
|
|
spi_flash_trans_t t = {
|
|
|
|
.command = CMD_RDID,
|
2019-11-28 01:20:00 +00:00
|
|
|
.miso_len = 3,
|
|
|
|
.miso_data = ((uint8_t*) &id_buf),
|
2019-01-08 10:29:25 +00:00
|
|
|
};
|
2019-09-05 05:17:11 +00:00
|
|
|
host->common_command(host, &t);
|
2019-11-28 01:20:00 +00:00
|
|
|
|
|
|
|
uint32_t raw_flash_id = id_buf;
|
2019-01-08 10:29:25 +00:00
|
|
|
ESP_EARLY_LOGV(TAG, "raw_chip_id: %X\n", raw_flash_id);
|
|
|
|
if (raw_flash_id == 0xFFFFFF || raw_flash_id == 0) {
|
|
|
|
ESP_EARLY_LOGE(TAG, "no response\n");
|
|
|
|
return ESP_ERR_FLASH_NO_RESPONSE;
|
|
|
|
}
|
|
|
|
// Byte swap the flash id as it's usually written the other way around
|
|
|
|
uint8_t mfg_id = raw_flash_id & 0xFF;
|
|
|
|
uint16_t flash_id = (raw_flash_id >> 16) | (raw_flash_id & 0xFF00);
|
|
|
|
*id = ((uint32_t)mfg_id << 16) | flash_id;
|
|
|
|
ESP_EARLY_LOGV(TAG, "chip_id: %X\n", *id);
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_err_t memspi_host_read_status_hs(spi_flash_host_driver_t *driver, uint8_t *out_sr)
|
|
|
|
{
|
|
|
|
//NOTE: we do have a read id function, however it doesn't work in high freq
|
2019-11-28 01:20:00 +00:00
|
|
|
uint32_t stat_buf = 0;
|
2019-01-08 10:29:25 +00:00
|
|
|
spi_flash_trans_t t = {
|
|
|
|
.command = CMD_RDSR,
|
2019-11-28 01:20:00 +00:00
|
|
|
.miso_data = ((uint8_t*) &stat_buf),
|
|
|
|
.miso_len = 1
|
2019-01-08 10:29:25 +00:00
|
|
|
};
|
|
|
|
esp_err_t err = driver->common_command(driver, &t);
|
|
|
|
if (err != ESP_OK) {
|
|
|
|
return err;
|
|
|
|
}
|
2019-11-28 01:20:00 +00:00
|
|
|
*out_sr = stat_buf;
|
2019-01-08 10:29:25 +00:00
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_err_t memspi_host_flush_cache(spi_flash_host_driver_t* driver, uint32_t addr, uint32_t size)
|
|
|
|
{
|
2019-11-28 01:20:00 +00:00
|
|
|
if ((void*)((memspi_host_data_t*)(driver->driver_data))->spi == (void*) spi_flash_ll_get_hw(SPI_HOST)) {
|
2019-01-08 10:29:25 +00:00
|
|
|
spi_flash_check_and_flush_cache(addr, size);
|
|
|
|
}
|
|
|
|
return ESP_OK;
|
2019-11-28 01:20:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void memspi_host_erase_chip(spi_flash_host_driver_t *chip_drv)
|
|
|
|
{
|
|
|
|
spi_flash_trans_t t = { 0 };
|
|
|
|
t.command = CMD_CHIP_ERASE;
|
|
|
|
chip_drv->common_command(chip_drv, &t);
|
|
|
|
}
|
|
|
|
|
|
|
|
void memspi_host_erase_sector(spi_flash_host_driver_t *chip_drv, uint32_t start_address)
|
|
|
|
{
|
|
|
|
spi_flash_trans_t t = {
|
|
|
|
.command = CMD_SECTOR_ERASE,
|
|
|
|
.address_bitlen = 24,
|
|
|
|
.address = start_address
|
|
|
|
};
|
|
|
|
chip_drv->common_command(chip_drv, &t);
|
|
|
|
}
|
|
|
|
|
|
|
|
void memspi_host_erase_block(spi_flash_host_driver_t *chip_drv, uint32_t start_address)
|
|
|
|
{
|
|
|
|
spi_flash_trans_t t = {
|
|
|
|
.command = CMD_LARGE_BLOCK_ERASE,
|
|
|
|
.address_bitlen = 24,
|
|
|
|
.address = start_address,
|
|
|
|
};
|
|
|
|
chip_drv->common_command(chip_drv, &t);
|
|
|
|
}
|
|
|
|
|
|
|
|
void memspi_host_program_page(spi_flash_host_driver_t *chip_drv, const void *buffer, uint32_t address, uint32_t length)
|
|
|
|
{
|
|
|
|
spi_flash_trans_t t = {
|
|
|
|
.command = CMD_PROGRAM_PAGE,
|
|
|
|
.address_bitlen = 24,
|
|
|
|
.address = address,
|
|
|
|
.mosi_len = length,
|
|
|
|
.mosi_data = buffer
|
|
|
|
};
|
|
|
|
chip_drv->common_command(chip_drv, &t);
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_err_t memspi_host_read(spi_flash_host_driver_t *chip_drv, void *buffer, uint32_t address, uint32_t read_len)
|
|
|
|
{
|
|
|
|
spi_flash_trans_t t = {
|
|
|
|
.address_bitlen = 24,
|
|
|
|
.address = address,
|
|
|
|
.miso_len = read_len,
|
|
|
|
.miso_data = buffer
|
|
|
|
};
|
|
|
|
chip_drv->common_command(chip_drv, &t);
|
|
|
|
return ESP_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
esp_err_t memspi_host_set_write_protect(spi_flash_host_driver_t *chip_drv, bool wp)
|
|
|
|
{
|
|
|
|
spi_flash_trans_t t = {
|
|
|
|
.command = wp ? CMD_WRDI : CMD_WREN
|
|
|
|
};
|
|
|
|
chip_drv->common_command(chip_drv, &t);
|
|
|
|
return ESP_OK;
|
|
|
|
}
|