cancel
Showing results for 
Search instead for 
Did you mean: 

Precise SPI Timing with Timer and DMA on H7

simae
Visitor

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)
  {
  }
}

 

 

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;
}

But this didn't work either.

0 REPLIES 0