mirror of
https://github.com/espressif/esp-idf.git
synced 2024-10-05 20:47:46 -04:00
Merge branch 'bugfix/fixed_sdmmc_high_speed_ddr_timing_issue_on_esp32s3_v4.4' into 'release/v4.4'
sdmmc: fixed incorrect clock phase settings on esp32s3 (v4.4) See merge request espressif/esp-idf!23261
This commit is contained in:
commit
f87bdd4a9f
@ -107,32 +107,53 @@ void sdmmc_host_reset(void)
|
||||
|
||||
static void sdmmc_host_set_clk_div(int div)
|
||||
{
|
||||
// Set frequency to 160MHz / div
|
||||
// div = p + 1
|
||||
// duty cycle = (h + 1)/(p + 1) (should be = 1/2)
|
||||
/**
|
||||
* Set frequency to 160MHz / div
|
||||
*
|
||||
* n: counter resets at div_factor_n.
|
||||
* l: negedge when counter equals div_factor_l.
|
||||
* h: posedge when counter equals div_factor_h.
|
||||
*
|
||||
* We set the duty cycle to 1/2
|
||||
*/
|
||||
#if CONFIG_IDF_TARGET_ESP32
|
||||
assert (div > 1 && div <= 16);
|
||||
int p = div - 1;
|
||||
int h = div - 1;
|
||||
int l = div / 2 - 1;
|
||||
|
||||
SDMMC.clock.div_factor_h = h;
|
||||
SDMMC.clock.div_factor_l = l;
|
||||
SDMMC.clock.div_factor_n = h;
|
||||
|
||||
// Set phases for in/out clocks
|
||||
// 180 degree phase on input and output clocks
|
||||
SDMMC.clock.phase_dout = 4;
|
||||
SDMMC.clock.phase_din = 4;
|
||||
SDMMC.clock.phase_core = 0;
|
||||
|
||||
#elif CONFIG_IDF_TARGET_ESP32S3
|
||||
assert (div > 1 && div <= 16);
|
||||
int l = div - 1;
|
||||
int h = div / 2 - 1;
|
||||
|
||||
SDMMC.clock.div_factor_p = p;
|
||||
SDMMC.clock.div_factor_h = h;
|
||||
SDMMC.clock.div_factor_m = p;
|
||||
SDMMC.clock.div_factor_l = l;
|
||||
SDMMC.clock.div_factor_n = l;
|
||||
|
||||
// Make sure 160 MHz source clock is used
|
||||
#if SOC_SDMMC_SUPPORT_XTAL_CLOCK
|
||||
SDMMC.clock.clk_sel = 1;
|
||||
#endif
|
||||
#if SOC_SDMMC_USE_GPIO_MATRIX
|
||||
// 90 degree phase on input and output clocks
|
||||
const int inout_clock_phase = 1;
|
||||
#else
|
||||
// 180 degree phase on input and output clocks
|
||||
const int inout_clock_phase = 4;
|
||||
#endif
|
||||
// Set phases for in/out clocks
|
||||
SDMMC.clock.phase_dout = inout_clock_phase;
|
||||
SDMMC.clock.phase_din = inout_clock_phase;
|
||||
|
||||
SDMMC.clock.phase_core = 0;
|
||||
/* 90 deg. delay for cclk_out to satisfy large hold time for SDR12 (up to 25MHz) and SDR25 (up to 50MHz) modes.
|
||||
* Whether this delayed clock will be used depends on use_hold_reg bit in CMD structure,
|
||||
* determined when sending out the command.
|
||||
*/
|
||||
SDMMC.clock.phase_dout = 1;
|
||||
SDMMC.clock.phase_din = 0;
|
||||
#endif //CONFIG_IDF_TARGET_ESP32S3
|
||||
|
||||
// Wait for the clock to propagate
|
||||
esp_rom_delay_us(10);
|
||||
}
|
||||
@ -246,6 +267,9 @@ esp_err_t sdmmc_host_start_command(int slot, sdmmc_hw_cmd_t cmd, uint32_t arg) {
|
||||
if (cmd.data_expected && cmd.rw && (SDMMC.wrtprt.cards & BIT(slot)) != 0) {
|
||||
return ESP_ERR_INVALID_STATE;
|
||||
}
|
||||
/* Outputs should be synchronized to cclk_out */
|
||||
cmd.use_hold_reg = 1;
|
||||
|
||||
while (SDMMC.cmd.start_command == 1) {
|
||||
;
|
||||
}
|
||||
|
@ -304,7 +304,6 @@ static sdmmc_hw_cmd_t make_hw_cmd(sdmmc_command_t* cmd)
|
||||
if (cmd->flags & SCF_RSP_CRC) {
|
||||
res.check_response_crc = 1;
|
||||
}
|
||||
res.use_hold_reg = 1;
|
||||
if (cmd->data) {
|
||||
res.data_expected = 1;
|
||||
if ((cmd->flags & SCF_CMD_READ) == 0) {
|
||||
|
@ -379,9 +379,9 @@ typedef volatile struct sdmmc_dev_s {
|
||||
uint32_t phase_dout: 3; ///< phase of data output clock (0x0: 0, 0x1: 90, 0x4: 180, 0x6: 270)
|
||||
uint32_t phase_din: 3; ///< phase of data input clock
|
||||
uint32_t phase_core: 3; ///< phase of the clock to SDMMC peripheral
|
||||
uint32_t div_factor_p: 4; ///< controls clock period; it will be (div_factor_p + 1) / 160MHz
|
||||
uint32_t div_factor_h: 4; ///< controls length of high pulse; it will be (div_factor_h + 1) / 160MHz
|
||||
uint32_t div_factor_m: 4; ///< should be equal to div_factor_p
|
||||
uint32_t div_factor_l: 4; ///< controls clock period; it will be (div_factor_l + 1) / 160MHz
|
||||
uint32_t div_factor_n: 4; ///< should be equal to div_factor_l
|
||||
uint32_t reserved21: 11;
|
||||
};
|
||||
uint32_t val;
|
||||
|
@ -376,9 +376,9 @@ typedef volatile struct sdmmc_dev_s {
|
||||
uint32_t phase_dout: 3; ///< phase of data output clock (0x0: 0, 0x1: 90, 0x4: 180, 0x6: 270)
|
||||
uint32_t phase_din: 3; ///< phase of data input clock
|
||||
uint32_t phase_core: 3; ///< phase of the clock to SDMMC peripheral
|
||||
uint32_t div_factor_p: 4; ///< controls clock period; it will be (div_factor_p + 1) / 160MHz
|
||||
uint32_t div_factor_h: 4; ///< controls length of high pulse; it will be (div_factor_h + 1) / 160MHz
|
||||
uint32_t div_factor_m: 4; ///< should be equal to div_factor_p
|
||||
uint32_t div_factor_l: 4; ///< controls clock period; it will be (div_factor_l + 1) / 160MHz
|
||||
uint32_t div_factor_n: 4; ///< should be equal to div_factor_l
|
||||
uint32_t reserved1 : 2;
|
||||
uint32_t clk_sel : 1; ///< clock source select (0: XTAL, 1: 160 MHz from PLL)
|
||||
uint32_t reserved24: 8;
|
||||
|
Loading…
x
Reference in New Issue
Block a user