2022-06-26 22:53:35 +00:00
|
|
|
/* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
|
|
|
2010-05-24 10:41:50 +00:00
|
|
|
/***************************************************************************
|
|
|
|
* Copyright (C) 2010 by Spencer Oliver *
|
|
|
|
* spen@spen-soft.co.uk *
|
|
|
|
***************************************************************************/
|
2012-01-31 17:55:03 +00:00
|
|
|
|
2010-05-24 10:41:50 +00:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
|
|
|
#include "imp.h"
|
|
|
|
|
2012-01-31 17:55:03 +00:00
|
|
|
static struct flash_bank *virtual_get_master_bank(struct flash_bank *bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
{
|
2012-01-31 17:55:03 +00:00
|
|
|
struct flash_bank *master_bank;
|
2010-05-24 10:41:50 +00:00
|
|
|
|
2010-05-24 11:32:58 +00:00
|
|
|
master_bank = get_flash_bank_by_name_noprobe(bank->driver_priv);
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2012-01-31 17:55:03 +00:00
|
|
|
LOG_ERROR("master flash bank '%s' does not exist", (char *)bank->driver_priv);
|
2010-05-24 10:41:50 +00:00
|
|
|
|
|
|
|
return master_bank;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void virtual_update_bank_info(struct flash_bank *bank)
|
|
|
|
{
|
|
|
|
struct flash_bank *master_bank = virtual_get_master_bank(bank);
|
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
/* update the info we do not have */
|
|
|
|
bank->size = master_bank->size;
|
|
|
|
bank->chip_width = master_bank->chip_width;
|
|
|
|
bank->bus_width = master_bank->bus_width;
|
2016-05-08 21:49:07 +00:00
|
|
|
bank->erased_value = master_bank->erased_value;
|
2013-09-19 11:07:06 +00:00
|
|
|
bank->default_padded_value = master_bank->default_padded_value;
|
2018-04-27 13:09:51 +00:00
|
|
|
bank->write_start_alignment = master_bank->write_start_alignment;
|
|
|
|
bank->write_end_alignment = master_bank->write_end_alignment;
|
|
|
|
bank->minimal_write_gap = master_bank->minimal_write_gap;
|
2010-05-24 10:41:50 +00:00
|
|
|
bank->num_sectors = master_bank->num_sectors;
|
|
|
|
bank->sectors = master_bank->sectors;
|
2018-04-27 13:09:51 +00:00
|
|
|
bank->num_prot_blocks = master_bank->num_prot_blocks;
|
|
|
|
bank->prot_blocks = master_bank->prot_blocks;
|
2010-05-24 10:41:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
FLASH_BANK_COMMAND_HANDLER(virtual_flash_bank_command)
|
|
|
|
{
|
|
|
|
if (CMD_ARGC < 7)
|
2011-12-16 06:48:39 +00:00
|
|
|
return ERROR_COMMAND_SYNTAX_ERROR;
|
2010-05-24 10:41:50 +00:00
|
|
|
|
|
|
|
/* get the master flash bank */
|
|
|
|
const char *bank_name = CMD_ARGV[6];
|
2010-05-24 11:32:58 +00:00
|
|
|
struct flash_bank *master_bank = get_flash_bank_by_name_noprobe(bank_name);
|
2010-05-24 10:41:50 +00:00
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank) {
|
2010-05-24 10:41:50 +00:00
|
|
|
LOG_ERROR("master flash bank '%s' does not exist", bank_name);
|
|
|
|
return ERROR_FLASH_OPERATION_FAILED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* save master bank name - use this to get settings later */
|
|
|
|
bank->driver_priv = strdup(bank_name);
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2020-06-07 15:00:13 +00:00
|
|
|
static int virtual_protect(struct flash_bank *bank, int set, unsigned int first,
|
|
|
|
unsigned int last)
|
2010-05-24 10:41:50 +00:00
|
|
|
{
|
|
|
|
struct flash_bank *master_bank = virtual_get_master_bank(bank);
|
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
return ERROR_FLASH_OPERATION_FAILED;
|
|
|
|
|
2020-07-17 21:44:35 +00:00
|
|
|
return flash_driver_protect(master_bank, set, first, last);
|
2010-05-24 10:41:50 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static int virtual_protect_check(struct flash_bank *bank)
|
|
|
|
{
|
|
|
|
struct flash_bank *master_bank = virtual_get_master_bank(bank);
|
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
return ERROR_FLASH_OPERATION_FAILED;
|
|
|
|
|
2021-07-03 19:29:32 +00:00
|
|
|
if (!master_bank->driver->protect_check)
|
2020-07-17 21:44:35 +00:00
|
|
|
return ERROR_FLASH_OPER_UNSUPPORTED;
|
2010-05-24 10:41:50 +00:00
|
|
|
|
2020-07-17 21:44:35 +00:00
|
|
|
/* call master handler */
|
|
|
|
return master_bank->driver->protect_check(master_bank);
|
2010-05-24 10:41:50 +00:00
|
|
|
}
|
|
|
|
|
2020-06-07 15:00:13 +00:00
|
|
|
static int virtual_erase(struct flash_bank *bank, unsigned int first,
|
|
|
|
unsigned int last)
|
2010-05-24 10:41:50 +00:00
|
|
|
{
|
|
|
|
struct flash_bank *master_bank = virtual_get_master_bank(bank);
|
|
|
|
int retval;
|
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
return ERROR_FLASH_OPERATION_FAILED;
|
|
|
|
|
|
|
|
/* call master handler */
|
2012-01-31 17:55:03 +00:00
|
|
|
retval = master_bank->driver->erase(master_bank, first, last);
|
|
|
|
if (retval != ERROR_OK)
|
2010-05-24 10:41:50 +00:00
|
|
|
return retval;
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2014-03-10 21:23:07 +00:00
|
|
|
static int virtual_write(struct flash_bank *bank, const uint8_t *buffer,
|
2010-05-24 10:41:50 +00:00
|
|
|
uint32_t offset, uint32_t count)
|
|
|
|
{
|
|
|
|
struct flash_bank *master_bank = virtual_get_master_bank(bank);
|
|
|
|
int retval;
|
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
return ERROR_FLASH_OPERATION_FAILED;
|
|
|
|
|
|
|
|
/* call master handler */
|
2012-01-31 17:55:03 +00:00
|
|
|
retval = master_bank->driver->write(master_bank, buffer, offset, count);
|
|
|
|
if (retval != ERROR_OK)
|
2010-05-24 10:41:50 +00:00
|
|
|
return retval;
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int virtual_probe(struct flash_bank *bank)
|
|
|
|
{
|
|
|
|
struct flash_bank *master_bank = virtual_get_master_bank(bank);
|
|
|
|
int retval;
|
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
return ERROR_FLASH_OPERATION_FAILED;
|
|
|
|
|
|
|
|
/* call master handler */
|
2012-01-31 17:55:03 +00:00
|
|
|
retval = master_bank->driver->probe(master_bank);
|
|
|
|
if (retval != ERROR_OK)
|
2010-05-24 10:41:50 +00:00
|
|
|
return retval;
|
|
|
|
|
|
|
|
/* update the info we do not have */
|
|
|
|
virtual_update_bank_info(bank);
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int virtual_auto_probe(struct flash_bank *bank)
|
|
|
|
{
|
|
|
|
struct flash_bank *master_bank = virtual_get_master_bank(bank);
|
|
|
|
int retval;
|
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
return ERROR_FLASH_OPERATION_FAILED;
|
|
|
|
|
|
|
|
/* call master handler */
|
2012-01-31 17:55:03 +00:00
|
|
|
retval = master_bank->driver->auto_probe(master_bank);
|
|
|
|
if (retval != ERROR_OK)
|
2010-05-24 10:41:50 +00:00
|
|
|
return retval;
|
|
|
|
|
|
|
|
/* update the info we do not have */
|
|
|
|
virtual_update_bank_info(bank);
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2021-04-23 08:47:17 +00:00
|
|
|
static int virtual_info(struct flash_bank *bank, struct command_invocation *cmd)
|
2010-05-24 10:41:50 +00:00
|
|
|
{
|
|
|
|
struct flash_bank *master_bank = virtual_get_master_bank(bank);
|
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
return ERROR_FLASH_OPERATION_FAILED;
|
|
|
|
|
2021-04-23 08:47:17 +00:00
|
|
|
command_print_sameline(cmd, "%s driver for flash bank %s at " TARGET_ADDR_FMT,
|
2010-05-24 10:41:50 +00:00
|
|
|
bank->driver->name, master_bank->name, master_bank->base);
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2010-06-20 09:38:19 +00:00
|
|
|
static int virtual_blank_check(struct flash_bank *bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
{
|
|
|
|
struct flash_bank *master_bank = virtual_get_master_bank(bank);
|
|
|
|
int retval;
|
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
return ERROR_FLASH_OPERATION_FAILED;
|
|
|
|
|
|
|
|
/* call master handler */
|
2012-01-31 17:55:03 +00:00
|
|
|
retval = master_bank->driver->erase_check(master_bank);
|
|
|
|
if (retval != ERROR_OK)
|
2010-05-24 10:41:50 +00:00
|
|
|
return retval;
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2010-06-20 09:38:19 +00:00
|
|
|
static int virtual_flash_read(struct flash_bank *bank,
|
2010-05-24 10:41:50 +00:00
|
|
|
uint8_t *buffer, uint32_t offset, uint32_t count)
|
|
|
|
{
|
|
|
|
struct flash_bank *master_bank = virtual_get_master_bank(bank);
|
|
|
|
int retval;
|
|
|
|
|
2021-07-03 16:51:20 +00:00
|
|
|
if (!master_bank)
|
2010-05-24 10:41:50 +00:00
|
|
|
return ERROR_FLASH_OPERATION_FAILED;
|
|
|
|
|
|
|
|
/* call master handler */
|
2012-01-31 17:55:03 +00:00
|
|
|
retval = master_bank->driver->read(master_bank, buffer, offset, count);
|
|
|
|
if (retval != ERROR_OK)
|
2010-05-24 10:41:50 +00:00
|
|
|
return retval;
|
|
|
|
|
|
|
|
return ERROR_OK;
|
|
|
|
}
|
|
|
|
|
2018-12-13 19:53:59 +00:00
|
|
|
const struct flash_driver virtual_flash = {
|
2010-05-24 10:41:50 +00:00
|
|
|
.name = "virtual",
|
|
|
|
.flash_bank_command = virtual_flash_bank_command,
|
|
|
|
.erase = virtual_erase,
|
|
|
|
.protect = virtual_protect,
|
|
|
|
.write = virtual_write,
|
|
|
|
.read = virtual_flash_read,
|
|
|
|
.probe = virtual_probe,
|
|
|
|
.auto_probe = virtual_auto_probe,
|
|
|
|
.erase_check = virtual_blank_check,
|
|
|
|
.protect_check = virtual_protect_check,
|
|
|
|
.info = virtual_info,
|
2018-02-15 09:25:50 +00:00
|
|
|
.free_driver_priv = default_flash_free_driver_priv,
|
2010-05-24 10:41:50 +00:00
|
|
|
};
|