Проблема связи с флэш-памятью SPI NAND (STM32L4, QSPI)

Я пытаюсь использовать внешнюю флеш-память (TC58CVG1S3HxAIx) с платой NUCLEO-L476RG. Я не могу заставить QSPI работать. Я использовал STM32CubeMx для настройки QSPI:

void MX_QUADSPI_Init(void)
{

  hqspi.Instance = QUADSPI;
  hqspi.Init.ClockPrescaler = 255;
  hqspi.Init.FifoThreshold = 1;
  hqspi.Init.SampleShifting = QSPI_SAMPLE_SHIFTING_NONE;
  hqspi.Init.FlashSize = 30;
  hqspi.Init.ChipSelectHighTime = QSPI_CS_HIGH_TIME_1_CYCLE;
  hqspi.Init.ClockMode = QSPI_CLOCK_MODE_0;
  if (HAL_QSPI_Init(&hqspi) != HAL_OK)
  {
    _Error_Handler(__FILE__, __LINE__);
  }

}

void HAL_QSPI_MspInit(QSPI_HandleTypeDef* qspiHandle)
{

  GPIO_InitTypeDef GPIO_InitStruct;
  if(qspiHandle->Instance==QUADSPI)
  {
  /* USER CODE BEGIN QUADSPI_MspInit 0 */

  /* USER CODE END QUADSPI_MspInit 0 */
    /* QUADSPI clock enable */
    __HAL_RCC_QSPI_CLK_ENABLE();


    /**QUADSPI GPIO Configuration    
    PA6     ------> QUADSPI_BK1_IO3
    PA7     ------> QUADSPI_BK1_IO2
    PB0     ------> QUADSPI_BK1_IO1
    PB1     ------> QUADSPI_BK1_IO0
    PB10     ------> QUADSPI_CLK
    PB11     ------> QUADSPI_NCS 
    */
    GPIO_InitStruct.Pin = GPIO_PIN_6|GPIO_PIN_7;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;
    GPIO_InitStruct.Alternate = GPIO_AF10_QUADSPI;
    HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);

    GPIO_InitStruct.Pin = GPIO_PIN_0|GPIO_PIN_1|GPIO_PIN_10|GPIO_PIN_11;
    GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
    GPIO_InitStruct.Pull = GPIO_NOPULL;
    GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;
    GPIO_InitStruct.Alternate = GPIO_AF10_QUADSPI;
    HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);

  /* USER CODE BEGIN QUADSPI_MspInit 1 */

  /* USER CODE END QUADSPI_MspInit 1 */
  }
}

И я написал простую программу для получения id из флеш-памяти:

#include "main.h"
#include "stm32l4xx_hal.h"
#include "quadspi.h"
#include "gpio.h"

uint8_t ReadAddr = 0x9F;
uint8_t pBuffer[3];

int main(void)
{
  HAL_Init();
  SystemClock_Config();
  MX_GPIO_Init();
  MX_QUADSPI_Init();

  while (1)
  {
  HAL_QSPI_Transmit(&hqspi, &ReadAddr, 5);
  HAL_QSPI_Receive(&hqspi, pBuffer, 5);
  }
}

Часы qspi не работают. Я проверил на осциллографе, и все, что я вижу, - это прямая линия. Если я изменю режим часов qspi на высокий, проблема все еще существует, но я вижу линию на 3 В, а не на 0 В. Сигнал CS работает, на осциллографе я четко вижу время, когда uC должен передавать и принимать.

Я проверил регистр AHB3ENR, он правильно инициализирован. Понятия не имею, что еще может быть не так.

Может ли кто-нибудь мне помочь?


person awyzlin    schedule 18.09.2018    source источник
comment
Вы пробовали режим отладки? Это СУПЕР полезно, потому что вы можете просматривать значения всех регистров в устройстве. Вы также можете изменить значения регистров во время выполнения без необходимости повторно загружать код. Убедитесь, что параметры в регистрах QSPI кажутся нормальными в соответствии со справочным руководством для вашего конкретного чипа STM32.   -  person George Troulis    schedule 19.09.2018
comment
Да, пробовал режим отладки. Мне все значения регистров QSPI кажутся хорошими, но я использую QSPI впервые, поэтому, возможно, я что-то упускаю.   -  person awyzlin    schedule 20.09.2018


Ответы (1)


Похоже на некорректное использование драйвера HAL.

QSPI - более сложное периферийное устройство, чем обычный модуль SPI.

Связь через QSPI состоит из нескольких этапов (инструкция, адрес и данные), как описано в справочном руководстве.

Длина этих фаз должна быть записана в регистры длины данных и конфигурации связи.

Это нужно сделать, позвонив HAL_QSPI_Command(), если вы все еще используете драйверы HAL.

Вы можете найти примеры в драйвере внешней памяти HAL (например, stm32h743i_eval_qspi.c, но я не знаю, есть ли аналог HAL STM32L476).

Также прочтите очень полезный документ AN4760 Application note Quad-SPI (QSPI) interface on STM32 microcontrollers

person Alexey Esaulenko    schedule 26.09.2018
comment
Алексей, я использовал HAL_QSPI_Command () и все работает! Спасибо за совет. - person awyzlin; 28.09.2018