Skip to content

Commit

Permalink
ports/rp2: Set flash divisor appropriately.
Browse files Browse the repository at this point in the history
Signed-off-by: Mike Bell <mdb036@gmail.com>
  • Loading branch information
MichaelBell authored and Gadgetoid committed Sep 5, 2024
1 parent f10906d commit 3cd6781
Show file tree
Hide file tree
Showing 4 changed files with 92 additions and 0 deletions.
4 changes: 4 additions & 0 deletions ports/rp2/main.c
Original file line number Diff line number Diff line change
Expand Up @@ -27,6 +27,7 @@
#include <stdio.h>

#include "rp2_psram.h"
#include "rp2_flash.h"
#include "py/compile.h"
#include "py/cstack.h"
#include "py/runtime.h"
Expand Down Expand Up @@ -91,6 +92,9 @@ int main(int argc, char **argv) {
// Set the MCU frequency and as a side effect the peripheral clock to 48 MHz.
set_sys_clock_khz(125000, false);

// Set the flash divisor to an appropriate value
rp2_flash_set_timing();

#if MICROPY_HW_ENABLE_UART_REPL
bi_decl(bi_program_feature("UART REPL"))
setup_default_uart();
Expand Down
12 changes: 12 additions & 0 deletions ports/rp2/modmachine.c
Original file line number Diff line number Diff line change
Expand Up @@ -32,6 +32,7 @@
#include "modmachine.h"
#include "uart.h"
#include "rp2_psram.h"
#include "rp2_flash.h"
#include "clocks_extra.h"
#include "hardware/pll.h"
#include "hardware/structs/rosc.h"
Expand Down Expand Up @@ -95,6 +96,11 @@ static mp_obj_t mp_machine_get_freq(void) {

static void mp_machine_set_freq(size_t n_args, const mp_obj_t *args) {
mp_int_t freq = mp_obj_get_int(args[0]);

// If necessary, increase the flash divider before increasing the clock speed
const int old_freq = clock_get_hz(clk_sys);
rp2_flash_set_timing_for_freq(MAX(freq, old_freq));

if (!set_sys_clock_khz(freq / 1000, false)) {
mp_raise_ValueError(MP_ERROR_TEXT("cannot change frequency"));
}
Expand All @@ -112,6 +118,12 @@ static void mp_machine_set_freq(size_t n_args, const mp_obj_t *args) {
}
}
}

// If clock speed was reduced, maybe we can reduce the flash divider
if (freq < old_freq) {
rp2_flash_set_timing_for_freq(freq);
}

#if MICROPY_HW_ENABLE_UART_REPL
setup_default_uart();
mp_uart_init();
Expand Down
69 changes: 69 additions & 0 deletions ports/rp2/rp2_flash.c
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,12 @@
#include "modrp2.h"
#include "hardware/flash.h"
#include "pico/binary_info.h"
#ifdef PICO_RP2350
#include "hardware/structs/ioqspi.h"
#include "hardware/structs/qmi.h"
#else
#include "hardware/structs/ssi.h"
#endif

#define BLOCK_SIZE_BYTES (FLASH_SECTOR_SIZE)

Expand Down Expand Up @@ -71,6 +77,48 @@ bi_decl(bi_block_device(
BINARY_INFO_BLOCK_DEV_FLAG_WRITE |
BINARY_INFO_BLOCK_DEV_FLAG_PT_UNKNOWN));

// Function to set the flash divisor to the correct divisor, assumes interrupts disabled
// and core1 locked out if relevant.
static void __no_inline_not_in_flash_func(rp2_flash_set_timing_internal)(int clock_hz) {

// Use the minimum divisor assuming a 133MHz flash.
const int max_flash_freq = 133000000;
int divisor = (clock_hz + max_flash_freq - 1) / max_flash_freq;

#if PICO_RP2350
// Make sure flash is deselected - QMI doesn't appear to have a busy flag(!)
while ((ioqspi_hw->io[1].status & IO_QSPI_GPIO_QSPI_SS_STATUS_OUTTOPAD_BITS) != IO_QSPI_GPIO_QSPI_SS_STATUS_OUTTOPAD_BITS) {
;
}

// RX delay equal to the divisor means sampling at the same time as the next falling edge of SCK after the
// falling edge that generated the data. This is pretty tight at 133MHz but seems to work with the Winbond flash chips.
const int rxdelay = divisor;
qmi_hw->m[0].timing = (1 << QMI_M0_TIMING_COOLDOWN_LSB) |
rxdelay << QMI_M1_TIMING_RXDELAY_LSB |
divisor << QMI_M1_TIMING_CLKDIV_LSB;

// Force a read through XIP to ensure the timing is applied
volatile uint32_t *ptr = (volatile uint32_t *)0x14000000;
(void)*ptr;
#else
// RP2040 SSI hardware only supports even divisors
if (divisor & 1) {
divisor += 1;
}

// Wait for SSI not busy
while (ssi_hw->sr & SSI_SR_BUSY_BITS) {
;
}

// Disable, set the new divisor, and re-enable
hw_clear_bits(&ssi_hw->ssienr, SSI_SSIENR_SSI_EN_BITS);
ssi_hw->baudr = divisor;
hw_set_bits(&ssi_hw->ssienr, SSI_SSIENR_SSI_EN_BITS);
#endif
}

// Flash erase and write must run with interrupts disabled and the other core suspended,
// because the XIP bit gets disabled.
static uint32_t begin_critical_flash_section(void) {
Expand All @@ -94,6 +142,7 @@ static void end_critical_flash_section(uint32_t state) {
#if defined(MICROPY_HW_PSRAM_CS_PIN) && MICROPY_HW_ENABLE_PSRAM
psram_init(MICROPY_HW_PSRAM_CS_PIN);
#endif
rp2_flash_set_timing_internal(clock_get_hz(clk_sys));
restore_interrupts(state);
if (multicore_lockout_victim_is_initialized(1 - get_core_num())) {
multicore_lockout_end_blocking();
Expand Down Expand Up @@ -250,3 +299,23 @@ MP_DEFINE_CONST_OBJ_TYPE(
make_new, rp2_flash_make_new,
locals_dict, &rp2_flash_locals_dict
);

// Modify the flash timing. Ensure flash access is suspended while
// the timings are altered.
void rp2_flash_set_timing_for_freq(int clock_hz) {
if (multicore_lockout_victim_is_initialized(1 - get_core_num())) {
multicore_lockout_start_blocking();
}
uint32_t state = save_and_disable_interrupts();

rp2_flash_set_timing_internal(clock_hz);

restore_interrupts(state);
if (multicore_lockout_victim_is_initialized(1 - get_core_num())) {
multicore_lockout_end_blocking();
}
}

void rp2_flash_set_timing() {
rp2_flash_set_timing_for_freq(clock_get_hz(clk_sys));
}
7 changes: 7 additions & 0 deletions ports/rp2/rp2_flash.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
#ifndef MICROPY_INCLUDED_RP2_MACHINE_FLASH_H
#define MICROPY_INCLUDED_RP2_MACHINE_FLASH_H

extern void rp2_flash_set_timing_for_freq(int clock_hz);
extern void rp2_flash_set_timing();

#endif

0 comments on commit 3cd6781

Please sign in to comment.