cancel
Showing results for 
Search instead for 
Did you mean: 

Precise SPI Timing with Timer and DMA on H7

simae
Associate

Hello everyone

I need to implement a protocol for an external ADC. An SPI read with 22 bits should be triggered via a timer in a 4us cycle.

spi_protocol.png

 

 

 

 

 

My procedure:

  • TIM2 triggers a DMA transfer from the TxBuffer into the SPI->TXDR (with some zeros)
  • Since CSTART is set and TSIZE is set to 0, the frame is immediately (approx. 196 ns delayed) written to the SPI bus
  • The received data should now be written to an RxBuffer via DMA
  • With DMA half complete or complete (after 125 samples in each case), an average value is to be formed.

The protocol on hardware side is working fine:

spi.png

 

My problem now is, that the DMA half complete or complete callback is never called (I used HAL_SPI_RxHalfCpltCallback and HAL_SPI_RxCpltCallback and overwrote them in the main). I have already read through some forum posts, the SPI seems to be a real pain. At least the write is now working via a DMA request from the timer, can anyone help me to get the read working?

Here is my code (a bit shortened), the configuration is generated with MXCube:

 

 

#include "main.h" #include "string.h" SPI_HandleTypeDef hspi3; DMA_HandleTypeDef hdma_spi3_rx; TIM_HandleTypeDef htim2; TIM_HandleTypeDef htim6; DMA_HandleTypeDef hdma_tim2_up; #define RX_BUF_LEN (250) #define RX_BUF_HALF_LEN (RX_BUF_LEN >> 1) static uint32_t tx_buffer; static uint32_t rx_buffer[RX_BUF_LEN]; uint32_t adc_value; void SystemClock_Config(void); static void MX_GPIO_Init(void); static void MX_DMA_Init(void); static void MX_SPI3_Init(void); static void MX_TIM6_Init(void); static void MX_TIM2_Init(void); int main(void) { memset(&tx_buffer, 0x00, sizeof(tx_buffer)); memset(&rx_buffer, 0x00, sizeof(rx_buffer)); memset(&adc_value, 0x00, sizeof(adc_value)); HAL_Init(); SystemClock_Config(); MX_GPIO_Init(); MX_DMA_Init(); MX_ETH_Init(); MX_USART3_UART_Init(); MX_USB_OTG_HS_USB_Init(); MX_SPI3_Init(); MX_TIM2_Init(); if (HAL_DMA_Init(&hdma_tim2_up) != HAL_OK) { Error_Handler(); } if (HAL_TIM_Base_Start(&htim2) != HAL_OK) { Error_Handler(); } if (HAL_TIM_PWM_Start(&htim2, TIM_CHANNEL_1)!= HAL_OK) { Error_Handler(); } // Prepare SPI for DMA transmit / receive tx_buffer = 0x0000; hspi3.Instance->CR2 = 0x0000; // Enable "auto-send" hspi3.Instance->CR1 = (0x0000 | (1<<12) | (1<<0)); // Enable SPI, Set CS hspi3.Instance->CR1 = (0x0000 | (1<<12) | (1<<9) | (1<<0)); // Enable CSTART, Set CS htim2.Instance->CCR1 = (htim2.Init.Period - 14); // NCS / CVST should go high aprox. 50 ns before read on spi if( HAL_SPI_Receive_DMA(&hspi3, (uint8_t*)rx_buffer, RX_BUF_LEN) != HAL_OK) { Error_Handler(); } if (HAL_DMA_Start(&hdma_tim2_up, (uint32_t)&tx_buffer, (uint32_t)&hspi3.Instance->TXDR, 1) != HAL_OK) { Error_Handler(); } __HAL_TIM_ENABLE_DMA(&htim2, TIM_DIER_UDE); HAL_GPIO_WritePin(LED_GREEN_GPIO_Port, LED_GREEN_Pin, GPIO_PIN_SET); while (1) { } } /** * @brief System Clock Configuration * @retval None */ void SystemClock_Config(void); // deleted for post /** * @brief SPI3 Initialization Function * None * @retval None */ static void MX_SPI3_Init(void) { hspi3.Instance = SPI3; hspi3.Init.Mode = SPI_MODE_MASTER; hspi3.Init.Direction = SPI_DIRECTION_2LINES; hspi3.Init.DataSize = SPI_DATASIZE_22BIT; hspi3.Init.CLKPolarity = SPI_POLARITY_LOW; hspi3.Init.CLKPhase = SPI_PHASE_1EDGE; hspi3.Init.NSS = SPI_NSS_SOFT; hspi3.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_8; hspi3.Init.FirstBit = SPI_FIRSTBIT_MSB; hspi3.Init.TIMode = SPI_TIMODE_DISABLE; hspi3.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE; hspi3.Init.CRCPolynomial = 0x0; hspi3.Init.NSSPMode = SPI_NSS_PULSE_DISABLE; hspi3.Init.NSSPolarity = SPI_NSS_POLARITY_LOW; hspi3.Init.FifoThreshold = SPI_FIFO_THRESHOLD_01DATA; hspi3.Init.TxCRCInitializationPattern = SPI_CRC_INITIALIZATION_ALL_ZERO_PATTERN; hspi3.Init.RxCRCInitializationPattern = SPI_CRC_INITIALIZATION_ALL_ZERO_PATTERN; hspi3.Init.MasterSSIdleness = SPI_MASTER_SS_IDLENESS_05CYCLE; hspi3.Init.MasterInterDataIdleness = SPI_MASTER_INTERDATA_IDLENESS_00CYCLE; hspi3.Init.MasterReceiverAutoSusp = SPI_MASTER_RX_AUTOSUSP_DISABLE; hspi3.Init.MasterKeepIOState = SPI_MASTER_KEEP_IO_STATE_ENABLE; hspi3.Init.IOSwap = SPI_IO_SWAP_DISABLE; if (HAL_SPI_Init(&hspi3) != HAL_OK) { Error_Handler(); } } /** * @brief TIM2 Initialization Function * None * @retval None */ static void MX_TIM2_Init(void) { TIM_MasterConfigTypeDef sMasterConfig = {0}; TIM_OC_InitTypeDef sConfigOC = {0}; htim2.Instance = TIM2; htim2.Init.Prescaler = 1-1; htim2.Init.CounterMode = TIM_COUNTERMODE_UP; htim2.Init.Period = 1375-1; htim2.Init.ClockDivision = TIM_CLOCKDIVISION_DIV1; htim2.Init.AutoReloadPreload = TIM_AUTORELOAD_PRELOAD_DISABLE; if (HAL_TIM_PWM_Init(&htim2) != HAL_OK) { Error_Handler(); } sMasterConfig.MasterOutputTrigger = TIM_TRGO_UPDATE; sMasterConfig.MasterSlaveMode = TIM_MASTERSLAVEMODE_DISABLE; if (HAL_TIMEx_MasterConfigSynchronization(&htim2, &sMasterConfig) != HAL_OK) { Error_Handler(); } sConfigOC.OCMode = TIM_OCMODE_PWM1; sConfigOC.Pulse = 0xFFFFFFFF; sConfigOC.OCPolarity = TIM_OCPOLARITY_LOW; sConfigOC.OCFastMode = TIM_OCFAST_DISABLE; if (HAL_TIM_PWM_ConfigChannel(&htim2, &sConfigOC, TIM_CHANNEL_1) != HAL_OK) { Error_Handler(); } HAL_TIM_MspPostInit(&htim2); } /** * Enable DMA controller clock */ static void MX_DMA_Init(void) { /* DMA controller clock enable */ __HAL_RCC_DMA1_CLK_ENABLE(); } /** * @brief GPIO Initialization Function * None * @retval None */ static void MX_GPIO_Init(void) { GPIO_InitTypeDef GPIO_InitStruct = {0}; /* USER CODE BEGIN MX_GPIO_Init_1 */ /* USER CODE END MX_GPIO_Init_1 */ /* GPIO Ports Clock Enable */ __HAL_RCC_GPIOC_CLK_ENABLE(); __HAL_RCC_GPIOH_CLK_ENABLE(); __HAL_RCC_GPIOA_CLK_ENABLE(); __HAL_RCC_GPIOB_CLK_ENABLE(); __HAL_RCC_GPIOD_CLK_ENABLE(); __HAL_RCC_GPIOG_CLK_ENABLE(); __HAL_RCC_GPIOE_CLK_ENABLE(); /*Configure GPIO pin Output Level */ HAL_GPIO_WritePin(GPIOB, LED_GREEN_Pin|LED_RED_Pin, GPIO_PIN_RESET); /*Configure GPIO pin Output Level */ HAL_GPIO_WritePin(USB_FS_PWR_EN_GPIO_Port, USB_FS_PWR_EN_Pin, GPIO_PIN_RESET); /*Configure GPIO pin Output Level */ HAL_GPIO_WritePin(LED_YELLOW_GPIO_Port, LED_YELLOW_Pin, GPIO_PIN_RESET); /*Configure GPIO pin : B1_Pin */ GPIO_InitStruct.Pin = B1_Pin; GPIO_InitStruct.Mode = GPIO_MODE_INPUT; GPIO_InitStruct.Pull = GPIO_NOPULL; HAL_GPIO_Init(B1_GPIO_Port, &GPIO_InitStruct); /*Configure GPIO pins : LED_GREEN_Pin LED_RED_Pin */ GPIO_InitStruct.Pin = LED_GREEN_Pin|LED_RED_Pin; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(GPIOB, &GPIO_InitStruct); /*Configure GPIO pin : USB_FS_PWR_EN_Pin */ GPIO_InitStruct.Pin = USB_FS_PWR_EN_Pin; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(USB_FS_PWR_EN_GPIO_Port, &GPIO_InitStruct); /*Configure GPIO pin : USB_FS_OVCR_Pin */ GPIO_InitStruct.Pin = USB_FS_OVCR_Pin; GPIO_InitStruct.Mode = GPIO_MODE_IT_RISING; GPIO_InitStruct.Pull = GPIO_NOPULL; HAL_GPIO_Init(USB_FS_OVCR_GPIO_Port, &GPIO_InitStruct); /*Configure GPIO pin : USB_FS_VBUS_Pin */ GPIO_InitStruct.Pin = USB_FS_VBUS_Pin; GPIO_InitStruct.Mode = GPIO_MODE_INPUT; GPIO_InitStruct.Pull = GPIO_NOPULL; HAL_GPIO_Init(USB_FS_VBUS_GPIO_Port, &GPIO_InitStruct); /*Configure GPIO pin : USB_FS_ID_Pin */ GPIO_InitStruct.Pin = USB_FS_ID_Pin; GPIO_InitStruct.Mode = GPIO_MODE_AF_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; GPIO_InitStruct.Alternate = GPIO_AF10_OTG1_HS; HAL_GPIO_Init(USB_FS_ID_GPIO_Port, &GPIO_InitStruct); /*Configure GPIO pin : LED_YELLOW_Pin */ GPIO_InitStruct.Pin = LED_YELLOW_Pin; GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP; GPIO_InitStruct.Pull = GPIO_NOPULL; GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW; HAL_GPIO_Init(LED_YELLOW_GPIO_Port, &GPIO_InitStruct); /* USER CODE BEGIN MX_GPIO_Init_2 */ /* USER CODE END MX_GPIO_Init_2 */ } void HAL_SPI_RxHalfCpltCallback(SPI_HandleTypeDef *hspi) { uint32_t mean = 0; for(uint16_t i=0; i < RX_BUF_HALF_LEN; i++) { mean += ( (rx_buffer[i] & 0x003FFFFF) >> 6); } adc_value = mean / ((uint32_t)RX_BUF_HALF_LEN); return; } void HAL_SPI_RxCpltCallback(SPI_HandleTypeDef *hspi) { uint32_t mean = 0; for(uint16_t i=RX_BUF_HALF_LEN; i < RX_BUF_LEN; i++) { mean += ( (rx_buffer[i] & 0x003FFFFF) >> 6); } adc_value = mean / ((uint32_t)RX_BUF_HALF_LEN); return; } void HAL_SPI_ErrorCallback(SPI_HandleTypeDef *hspi) { Error_Handler(); } /** * @brief This function is executed in case of error occurrence. * @retval None */ void Error_Handler(void) { HAL_GPIO_WritePin(LED_RED_GPIO_Port, LED_RED_Pin, GPIO_PIN_SET); __disable_irq(); while (1) { } }
View more

 

 

 

The second thing i tried is:

 

int main(void) { ... tx_buffer = 0x0000; hspi3.Instance->CR2 = 0x0000; // Enable "auto-send" hspi3.Instance->CR1 = (0x0000 | (1<<12) | (1<<0)); // Enable SPI, Set CS hspi3.Instance->CR1 = (0x0000 | (1<<12) | (1<<9) | (1<<0)); // Enable CSTART, Set CS htim2.Instance->CCR1 = (htim2.Init.Period - 14); // NCS / CVST should go high aprox. 50 ns before read on spi if( HAL_DMA_RegisterCallback(&hdma_spi3_rx, HAL_DMA_XFER_HALFCPLT_CB_ID, dma_half_complete_callback) ) { Error_Handler(); } if( HAL_DMA_RegisterCallback(&hdma_spi3_rx, HAL_DMA_XFER_CPLT_CB_ID, dma_complete_callback) ) { Error_Handler(); } if( HAL_DMA_Start_IT(&hdma_spi3_rx, (uint32_t)&hspi3.Instance->RXDR, (uint32_t)&rx_buffer, RX_BUF_LEN)) { Error_Handler(); } if (HAL_DMA_Start(&hdma_tim2_up, (uint32_t)&tx_buffer, (uint32_t)&hspi3.Instance->TXDR, 1) != HAL_OK) { Error_Handler(); } __HAL_TIM_ENABLE_DMA(&htim2, TIM_DIER_UDE); __HAL_SPI_ENABLE_IT(&hspi3, (SPI_IT_OVR | SPI_IT_FRE | SPI_IT_MODF | SPI_IT_RXP)); HAL_GPIO_WritePin(LED_GREEN_GPIO_Port, LED_GREEN_Pin, GPIO_PIN_SET); while (1) { } } ... void dma_half_complete_callback(DMA_HandleTypeDef *dmaHandle) { uint32_t mean = 0; for(uint16_t i=0; i < RX_BUF_HALF_LEN; i++) { mean += ( (rx_buffer[i] & 0x003FFFFF) >> 6); } adc_value = mean / ((uint32_t)RX_BUF_HALF_LEN); return; } void dma_complete_callback(DMA_HandleTypeDef *dmaHandle) { uint32_t mean = 0; for(uint16_t i=RX_BUF_HALF_LEN; i < RX_BUF_LEN; i++) { mean += ( (rx_buffer[i] & 0x003FFFFF) >> 6); } adc_value = mean / ((uint32_t)RX_BUF_HALF_LEN); return; }
View more

 

But this didn't work either.

0 REPLIES 0