|  | /* | 
|  | * Copyright (c) 2020 Raspberry Pi (Trading) Ltd. | 
|  | * Copyright (c) 2022 Yonatan Schachter | 
|  | * | 
|  | * SPDX-License-Identifier: Apache-2.0 | 
|  | */ | 
|  |  | 
|  | #include <errno.h> | 
|  | #include <stdint.h> | 
|  | #include <stddef.h> | 
|  |  | 
|  | #include <zephyr/device.h> | 
|  | #include <zephyr/drivers/flash.h> | 
|  | #include <zephyr/logging/log.h> | 
|  | #include <zephyr/irq.h> | 
|  | #include <zephyr/toolchain.h> | 
|  |  | 
|  | #include <hardware/flash.h> | 
|  | #include <hardware/regs/io_qspi.h> | 
|  | #include <hardware/regs/pads_qspi.h> | 
|  | #include <hardware/structs/ssi.h> | 
|  | #include <hardware/structs/xip_ctrl.h> | 
|  | #include <hardware/resets.h> | 
|  | #include <pico/bootrom.h> | 
|  |  | 
|  | LOG_MODULE_REGISTER(flash_rpi_pico, CONFIG_FLASH_LOG_LEVEL); | 
|  |  | 
|  | #define DT_DRV_COMPAT raspberrypi_pico_flash_controller | 
|  |  | 
|  | #define PAGE_SIZE 256 | 
|  | #define SECTOR_SIZE DT_PROP(DT_CHOSEN(zephyr_flash), erase_block_size) | 
|  | #define ERASE_VALUE 0xff | 
|  | #define FLASH_SIZE KB(CONFIG_FLASH_SIZE) | 
|  | #define FLASH_BASE CONFIG_FLASH_BASE_ADDRESS | 
|  | #define SSI_BASE_ADDRESS DT_REG_ADDR(DT_CHOSEN(zephyr_flash_controller)) | 
|  |  | 
|  | static const struct flash_parameters flash_rpi_parameters = { | 
|  | .write_block_size = 1, | 
|  | .erase_value = ERASE_VALUE, | 
|  | }; | 
|  |  | 
|  | /** | 
|  | * Low level flash functions are based on: | 
|  | * github.com/raspberrypi/pico-bootrom/blob/master/bootrom/program_flash_generic.c | 
|  | * and | 
|  | * github.com/raspberrypi/pico-sdk/blob/master/src/rp2_common/hardware_flash/flash.c | 
|  | */ | 
|  |  | 
|  | #define FLASHCMD_PAGE_PROGRAM 0x02 | 
|  | #define FLASHCMD_READ_STATUS 0x05 | 
|  | #define FLASHCMD_WRITE_ENABLE 0x06 | 
|  | #define BOOT2_SIZE_WORDS 64 | 
|  |  | 
|  | enum outover { | 
|  | OUTOVER_NORMAL = 0, | 
|  | OUTOVER_INVERT, | 
|  | OUTOVER_LOW, | 
|  | OUTOVER_HIGH | 
|  | }; | 
|  |  | 
|  | static ssi_hw_t *const ssi = (ssi_hw_t *)SSI_BASE_ADDRESS; | 
|  | static uint32_t boot2_copyout[BOOT2_SIZE_WORDS]; | 
|  | static bool boot2_copyout_valid; | 
|  | static uint8_t flash_ram_buffer[PAGE_SIZE]; | 
|  |  | 
|  | static void __no_inline_not_in_flash_func(flash_init_boot2_copyout)(void) | 
|  | { | 
|  | if (boot2_copyout_valid) { | 
|  | return; | 
|  | } | 
|  | for (int i = 0; i < BOOT2_SIZE_WORDS; ++i) { | 
|  | boot2_copyout[i] = ((uint32_t *)FLASH_BASE)[i]; | 
|  | } | 
|  | __compiler_memory_barrier(); | 
|  | boot2_copyout_valid = true; | 
|  | } | 
|  |  | 
|  | static void __no_inline_not_in_flash_func(flash_enable_xip_via_boot2)(void) | 
|  | { | 
|  | ((void (*)(void))((uint32_t)boot2_copyout+1))(); | 
|  | } | 
|  |  | 
|  | void __no_inline_not_in_flash_func(flash_cs_force)(enum outover over) | 
|  | { | 
|  | io_rw_32 *reg = (io_rw_32 *) (IO_QSPI_BASE + IO_QSPI_GPIO_QSPI_SS_CTRL_OFFSET); | 
|  | *reg = (*reg & ~IO_QSPI_GPIO_QSPI_SS_CTRL_OUTOVER_BITS) | 
|  | | (over << IO_QSPI_GPIO_QSPI_SS_CTRL_OUTOVER_LSB); | 
|  | (void) *reg; | 
|  | } | 
|  |  | 
|  | int __no_inline_not_in_flash_func(flash_was_aborted)() | 
|  | { | 
|  | return *(io_rw_32 *) (IO_QSPI_BASE + IO_QSPI_GPIO_QSPI_SD1_CTRL_OFFSET) | 
|  | & IO_QSPI_GPIO_QSPI_SD1_CTRL_INOVER_BITS; | 
|  | } | 
|  |  | 
|  | void __no_inline_not_in_flash_func(flash_put_get)(const uint8_t *tx, uint8_t *rx, size_t count, | 
|  | size_t rx_skip) | 
|  | { | 
|  | const uint max_in_flight = 16 - 2; | 
|  | size_t tx_count = count; | 
|  | size_t rx_count = count; | 
|  | bool did_something; | 
|  | uint32_t tx_level; | 
|  | uint32_t rx_level; | 
|  | uint8_t rxbyte; | 
|  |  | 
|  | while (tx_count || rx_skip || rx_count) { | 
|  | tx_level = ssi_hw->txflr; | 
|  | rx_level = ssi_hw->rxflr; | 
|  | did_something = false; | 
|  | if (tx_count && tx_level + rx_level < max_in_flight) { | 
|  | ssi->dr0 = (uint32_t) (tx ? *tx++ : 0); | 
|  | --tx_count; | 
|  | did_something = true; | 
|  | } | 
|  | if (rx_level) { | 
|  | rxbyte = ssi->dr0; | 
|  | did_something = true; | 
|  | if (rx_skip) { | 
|  | --rx_skip; | 
|  | } else { | 
|  | if (rx) { | 
|  | *rx++ = rxbyte; | 
|  | } | 
|  | --rx_count; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (!did_something && __builtin_expect(flash_was_aborted(), 0)) { | 
|  | break; | 
|  | } | 
|  | } | 
|  | flash_cs_force(OUTOVER_HIGH); | 
|  | } | 
|  |  | 
|  | void __no_inline_not_in_flash_func(flash_put_get_wrapper)(uint8_t cmd, const uint8_t *tx, | 
|  | uint8_t *rx, size_t count) | 
|  | { | 
|  | flash_cs_force(OUTOVER_LOW); | 
|  | ssi->dr0 = cmd; | 
|  | flash_put_get(tx, rx, count, 1); | 
|  | } | 
|  |  | 
|  | static ALWAYS_INLINE void flash_put_cmd_addr(uint8_t cmd, uint32_t addr) | 
|  | { | 
|  | flash_cs_force(OUTOVER_LOW); | 
|  | addr |= cmd << 24; | 
|  | for (int i = 0; i < 4; ++i) { | 
|  | ssi->dr0 = addr >> 24; | 
|  | addr <<= 8; | 
|  | } | 
|  | } | 
|  |  | 
|  | void __no_inline_not_in_flash_func(flash_write_partial_internal)(uint32_t addr, const uint8_t *data, | 
|  | size_t size) | 
|  | { | 
|  | uint8_t status_reg; | 
|  |  | 
|  | flash_put_get_wrapper(FLASHCMD_WRITE_ENABLE, NULL, NULL, 0); | 
|  | flash_put_cmd_addr(FLASHCMD_PAGE_PROGRAM, addr); | 
|  | flash_put_get(data, NULL, size, 4); | 
|  |  | 
|  | do { | 
|  | flash_put_get_wrapper(FLASHCMD_READ_STATUS, NULL, &status_reg, 1); | 
|  | } while (status_reg & 0x1 && !flash_was_aborted()); | 
|  | } | 
|  |  | 
|  | void __no_inline_not_in_flash_func(flash_write_partial)(uint32_t flash_offs, const uint8_t *data, | 
|  | size_t count) | 
|  | { | 
|  | rom_connect_internal_flash_fn connect_internal_flash = (rom_connect_internal_flash_fn) | 
|  | rom_func_lookup_inline(ROM_FUNC_CONNECT_INTERNAL_FLASH); | 
|  | rom_flash_exit_xip_fn exit_xip = (rom_flash_exit_xip_fn) | 
|  | rom_func_lookup_inline(ROM_FUNC_FLASH_EXIT_XIP); | 
|  | rom_flash_flush_cache_fn flush_cache = (rom_flash_flush_cache_fn) | 
|  | rom_func_lookup_inline(ROM_FUNC_FLASH_FLUSH_CACHE); | 
|  |  | 
|  | flash_init_boot2_copyout(); | 
|  |  | 
|  | __compiler_memory_barrier(); | 
|  |  | 
|  | connect_internal_flash(); | 
|  | exit_xip(); | 
|  | flash_write_partial_internal(flash_offs, data, count); | 
|  | flush_cache(); | 
|  | flash_enable_xip_via_boot2(); | 
|  | } | 
|  |  | 
|  | static bool is_valid_range(off_t offset, uint32_t size) | 
|  | { | 
|  | return (offset >= 0) && ((offset + size) <= FLASH_SIZE); | 
|  | } | 
|  |  | 
|  | static int flash_rpi_read(const struct device *dev, off_t offset, void *data, size_t size) | 
|  | { | 
|  | if (size == 0) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | if (!is_valid_range(offset, size)) { | 
|  | LOG_ERR("Read range exceeds the flash boundaries"); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | memcpy(data, (uint8_t *)(CONFIG_FLASH_BASE_ADDRESS + offset), size); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int flash_rpi_write(const struct device *dev, off_t offset, const void *data, size_t size) | 
|  | { | 
|  | uint32_t key; | 
|  | size_t bytes_to_write; | 
|  | uint8_t *data_pointer = (uint8_t *)data; | 
|  |  | 
|  | if (size == 0) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | if (!is_valid_range(offset, size)) { | 
|  | LOG_ERR("Write range exceeds the flash boundaries. Offset=%#lx, Size=%u", | 
|  | offset, size); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | key = irq_lock(); | 
|  |  | 
|  | if ((offset & (PAGE_SIZE - 1)) > 0) { | 
|  | bytes_to_write = MIN(PAGE_SIZE - (offset & (PAGE_SIZE - 1)), size); | 
|  | memcpy(flash_ram_buffer, data_pointer, bytes_to_write); | 
|  | flash_write_partial(offset, flash_ram_buffer, bytes_to_write); | 
|  | data_pointer += bytes_to_write; | 
|  | size -= bytes_to_write; | 
|  | offset += bytes_to_write; | 
|  | } | 
|  |  | 
|  | while (size >= PAGE_SIZE) { | 
|  | bytes_to_write = PAGE_SIZE; | 
|  | memcpy(flash_ram_buffer, data_pointer, bytes_to_write); | 
|  | flash_range_program(offset, flash_ram_buffer, bytes_to_write); | 
|  | data_pointer += bytes_to_write; | 
|  | size -= bytes_to_write; | 
|  | offset += bytes_to_write; | 
|  | } | 
|  |  | 
|  | if (size > 0) { | 
|  | memcpy(flash_ram_buffer, data_pointer, size); | 
|  | flash_write_partial(offset, flash_ram_buffer, size); | 
|  | } | 
|  |  | 
|  | irq_unlock(key); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static int flash_rpi_erase(const struct device *dev, off_t offset, size_t size) | 
|  | { | 
|  | uint32_t key; | 
|  |  | 
|  | if (size == 0) { | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | if (!is_valid_range(offset, size)) { | 
|  | LOG_ERR("Erase range exceeds the flash boundaries. Offset=%#lx, Size=%u", | 
|  | offset, size); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | if ((offset % SECTOR_SIZE) || (size % SECTOR_SIZE)) { | 
|  | LOG_ERR("Erase range is not a multiple of the sector size. Offset=%#lx, Size=%u", | 
|  | offset, size); | 
|  | return -EINVAL; | 
|  | } | 
|  |  | 
|  | key = irq_lock(); | 
|  |  | 
|  | flash_range_erase(offset, size); | 
|  |  | 
|  | irq_unlock(key); | 
|  |  | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | static const struct flash_parameters *flash_rpi_get_parameters(const struct device *dev) | 
|  | { | 
|  | ARG_UNUSED(dev); | 
|  |  | 
|  | return &flash_rpi_parameters; | 
|  | } | 
|  |  | 
|  | #if CONFIG_FLASH_PAGE_LAYOUT | 
|  |  | 
|  | static const struct flash_pages_layout flash_rpi_pages_layout = { | 
|  | .pages_count = FLASH_SIZE / SECTOR_SIZE, | 
|  | .pages_size = SECTOR_SIZE, | 
|  | }; | 
|  |  | 
|  | void flash_rpi_page_layout(const struct device *dev, const struct flash_pages_layout **layout, | 
|  | size_t *layout_size) | 
|  | { | 
|  | *layout = &flash_rpi_pages_layout; | 
|  | *layout_size = 1; | 
|  | } | 
|  |  | 
|  | #endif /* CONFIG_FLASH_PAGE_LAYOUT */ | 
|  |  | 
|  | static DEVICE_API(flash, flash_rpi_driver_api) = { | 
|  | .read = flash_rpi_read, | 
|  | .write = flash_rpi_write, | 
|  | .erase = flash_rpi_erase, | 
|  | .get_parameters = flash_rpi_get_parameters, | 
|  | #ifdef CONFIG_FLASH_PAGE_LAYOUT | 
|  | .page_layout = flash_rpi_page_layout, | 
|  | #endif /* CONFIG_FLASH_PAGE_LAYOUT */ | 
|  | }; | 
|  |  | 
|  | DEVICE_DT_INST_DEFINE(0, NULL, NULL, NULL, NULL, POST_KERNEL, | 
|  | CONFIG_FLASH_INIT_PRIORITY, &flash_rpi_driver_api); |