Merge pull request #11189 from klutvott123/move-telemetry-displayport-init
[betaflight.git] / lib / main / STM32H7 / Drivers / STM32H7xx_HAL_Driver / Src / stm32h7xx_hal_spi.c
blob584adf0a50891452d512cef940d0a761bbfdcc64
1 /**
2 ******************************************************************************
3 * @file stm32h7xx_hal_spi.c
4 * @author MCD Application Team
5 * @brief SPI HAL module driver.
6 * This file provides firmware functions to manage the following
7 * functionalities of the Serial Peripheral Interface (SPI) peripheral:
8 * + Initialization and de-initialization functions
9 * + IO operation functions
10 * + Peripheral Control functions
11 * + Peripheral State functions
13 @verbatim
14 ==============================================================================
15 ##### How to use this driver #####
16 ==============================================================================
17 [..]
18 The SPI HAL driver can be used as follows:
20 (#) Declare a SPI_HandleTypeDef handle structure, for example:
21 SPI_HandleTypeDef hspi;
23 (#)Initialize the SPI low level resources by implementing the HAL_SPI_MspInit() API:
24 (##) Enable the SPIx interface clock
25 (##) SPI pins configuration
26 (+++) Enable the clock for the SPI GPIOs
27 (+++) Configure these SPI pins as alternate function push-pull
28 (##) NVIC configuration if you need to use interrupt process or DMA process
29 (+++) Configure the SPIx interrupt priority
30 (+++) Enable the NVIC SPI IRQ handle
31 (##) DMA Configuration if you need to use DMA process
32 (+++) Declare a DMA_HandleTypeDef handle structure for the transmit or receive Stream/Channel
33 (+++) Enable the DMAx clock
34 (+++) Configure the DMA handle parameters
35 (+++) Configure the DMA Tx or Rx Stream/Channel
36 (+++) Associate the initialized hdma_tx handle to the hspi DMA Tx or Rx handle
37 (+++) Configure the priority and enable the NVIC for the transfer complete interrupt on the DMA Tx or Rx Stream/Channel
39 (#) Program the Mode, BidirectionalMode , Data size, Baudrate Prescaler, NSS
40 management, Clock polarity and phase, FirstBit and CRC configuration in the hspi Init structure.
42 (#) Initialize the SPI registers by calling the HAL_SPI_Init() API:
43 (++) This API configures also the low level Hardware GPIO, CLOCK, CORTEX...etc)
44 by calling the customized HAL_SPI_MspInit() API.
45 [..]
46 Callback registration:
48 (#) The compilation flag USE_HAL_SPI_REGISTER_CALLBACKS when set to 1UL
49 allows the user to configure dynamically the driver callbacks.
50 Use Functions HAL_SPI_RegisterCallback() to register an interrupt callback.
52 Function HAL_SPI_RegisterCallback() allows to register following callbacks:
53 (+) TxCpltCallback : SPI Tx Completed callback
54 (+) RxCpltCallback : SPI Rx Completed callback
55 (+) TxRxCpltCallback : SPI TxRx Completed callback
56 (+) TxHalfCpltCallback : SPI Tx Half Completed callback
57 (+) RxHalfCpltCallback : SPI Rx Half Completed callback
58 (+) TxRxHalfCpltCallback : SPI TxRx Half Completed callback
59 (+) ErrorCallback : SPI Error callback
60 (+) AbortCpltCallback : SPI Abort callback
61 (+) MspInitCallback : SPI Msp Init callback
62 (+) MspDeInitCallback : SPI Msp DeInit callback
63 This function takes as parameters the HAL peripheral handle, the Callback ID
64 and a pointer to the user callback function.
67 (#) Use function HAL_SPI_UnRegisterCallback to reset a callback to the default
68 weak function.
69 HAL_SPI_UnRegisterCallback takes as parameters the HAL peripheral handle,
70 and the Callback ID.
71 This function allows to reset following callbacks:
72 (+) TxCpltCallback : SPI Tx Completed callback
73 (+) RxCpltCallback : SPI Rx Completed callback
74 (+) TxRxCpltCallback : SPI TxRx Completed callback
75 (+) TxHalfCpltCallback : SPI Tx Half Completed callback
76 (+) RxHalfCpltCallback : SPI Rx Half Completed callback
77 (+) TxRxHalfCpltCallback : SPI TxRx Half Completed callback
78 (+) ErrorCallback : SPI Error callback
79 (+) AbortCpltCallback : SPI Abort callback
80 (+) MspInitCallback : SPI Msp Init callback
81 (+) MspDeInitCallback : SPI Msp DeInit callback
83 By default, after the HAL_SPI_Init() and when the state is HAL_SPI_STATE_RESET
84 all callbacks are set to the corresponding weak functions:
85 examples HAL_SPI_MasterTxCpltCallback(), HAL_SPI_MasterRxCpltCallback().
86 Exception done for MspInit and MspDeInit functions that are
87 reset to the legacy weak functions in the HAL_SPI_Init()/ HAL_SPI_DeInit() only when
88 these callbacks are null (not registered beforehand).
89 If MspInit or MspDeInit are not null, the HAL_SPI_Init()/ HAL_SPI_DeInit()
90 keep and use the user MspInit/MspDeInit callbacks (registered beforehand) whatever the state.
92 Callbacks can be registered/unregistered in HAL_SPI_STATE_READY state only.
93 Exception done MspInit/MspDeInit functions that can be registered/unregistered
94 in HAL_SPI_STATE_READY or HAL_SPI_STATE_RESET state,
95 thus registered (user) MspInit/DeInit callbacks can be used during the Init/DeInit.
96 Then, the user first registers the MspInit/MspDeInit user callbacks
97 using HAL_SPI_RegisterCallback() before calling HAL_SPI_DeInit()
98 or HAL_SPI_Init() function.
100 When The compilation define USE_HAL_PPP_REGISTER_CALLBACKS is set to 0 or
101 not defined, the callback registering feature is not available
102 and weak (surcharged) callbacks are used.
105 [..]
106 Circular mode restriction:
107 (+) The DMA circular mode cannot be used when the SPI is configured in these modes:
108 (++) Master 2Lines RxOnly
109 (++) Master 1Line Rx
110 (+) The CRC feature is not managed when the DMA circular mode is enabled
111 (+) The functions HAL_SPI_DMAPause()/ HAL_SPI_DMAResume() are not supported. Return always
112 HAL_ERROR with ErrorCode set to HAL_SPI_ERROR_NOT_SUPPORTED.
113 Those functions are maintained for backward compatibility reasons.
115 @endverbatim
116 ******************************************************************************
117 * @attention
119 * <h2><center>&copy; Copyright (c) 2017 STMicroelectronics.
120 * All rights reserved.</center></h2>
122 * This software component is licensed by ST under BSD 3-Clause license,
123 * the "License"; You may not use this file except in compliance with the
124 * License. You may obtain a copy of the License at:
125 * opensource.org/licenses/BSD-3-Clause
127 ******************************************************************************
130 /* Includes ------------------------------------------------------------------*/
131 #include "stm32h7xx_hal.h"
133 /** @addtogroup STM32H7xx_HAL_Driver
134 * @{
137 /** @defgroup SPI SPI
138 * @brief SPI HAL module driver
139 * @{
141 #ifdef HAL_SPI_MODULE_ENABLED
143 /* Private typedef -----------------------------------------------------------*/
144 /* Private defines -----------------------------------------------------------*/
145 /** @defgroup SPI_Private_Constants SPI Private Constants
146 * @{
148 #define SPI_DEFAULT_TIMEOUT 100UL
150 * @}
153 /* Private macros ------------------------------------------------------------*/
154 /* Private variables ---------------------------------------------------------*/
155 /* Private function prototypes -----------------------------------------------*/
156 /** @defgroup SPI_Private_Functions SPI Private Functions
157 * @{
159 static void SPI_DMATransmitCplt(DMA_HandleTypeDef *hdma);
160 static void SPI_DMAReceiveCplt(DMA_HandleTypeDef *hdma);
161 static void SPI_DMATransmitReceiveCplt(DMA_HandleTypeDef *hdma);
162 static void SPI_DMAHalfTransmitCplt(DMA_HandleTypeDef *hdma);
163 static void SPI_DMAHalfReceiveCplt(DMA_HandleTypeDef *hdma);
164 static void SPI_DMAHalfTransmitReceiveCplt(DMA_HandleTypeDef *hdma);
165 static void SPI_DMAError(DMA_HandleTypeDef *hdma);
166 static void SPI_DMAAbortOnError(DMA_HandleTypeDef *hdma);
167 static void SPI_DMATxAbortCallback(DMA_HandleTypeDef *hdma);
168 static void SPI_DMARxAbortCallback(DMA_HandleTypeDef *hdma);
169 static HAL_StatusTypeDef SPI_WaitOnFlagUntilTimeout(SPI_HandleTypeDef *hspi, uint32_t Flag, FlagStatus FlagStatus,
170 uint32_t Timeout, uint32_t Tickstart);
171 static void SPI_TxISR_8BIT(SPI_HandleTypeDef *hspi);
172 static void SPI_TxISR_16BIT(SPI_HandleTypeDef *hspi);
173 static void SPI_TxISR_32BIT(SPI_HandleTypeDef *hspi);
174 static void SPI_RxISR_8BIT(SPI_HandleTypeDef *hspi);
175 static void SPI_RxISR_16BIT(SPI_HandleTypeDef *hspi);
176 static void SPI_RxISR_32BIT(SPI_HandleTypeDef *hspi);
177 static void SPI_AbortTransfer(SPI_HandleTypeDef *hspi);
178 static void SPI_CloseTransfer(SPI_HandleTypeDef *hspi);
179 static uint32_t SPI_GetPacketSize(SPI_HandleTypeDef *hspi);
183 * @}
186 /* Exported functions --------------------------------------------------------*/
187 /** @defgroup SPI_Exported_Functions SPI Exported Functions
188 * @{
191 /** @defgroup SPI_Exported_Functions_Group1 Initialization and de-initialization functions
192 * @brief Initialization and Configuration functions
194 @verbatim
195 ===============================================================================
196 ##### Initialization and de-initialization functions #####
197 ===============================================================================
198 [..] This subsection provides a set of functions allowing to initialize and
199 de-initialize the SPIx peripheral:
201 (+) User must implement HAL_SPI_MspInit() function in which he configures
202 all related peripherals resources (CLOCK, GPIO, DMA, IT and NVIC ).
204 (+) Call the function HAL_SPI_Init() to configure the selected device with
205 the selected configuration:
206 (++) Mode
207 (++) Direction
208 (++) Data Size
209 (++) Clock Polarity and Phase
210 (++) NSS Management
211 (++) BaudRate Prescaler
212 (++) FirstBit
213 (++) TIMode
214 (++) CRC Calculation
215 (++) CRC Polynomial if CRC enabled
216 (++) CRC Length, used only with Data8 and Data16
217 (++) FIFO reception threshold
218 (++) FIFO transmission threshold
220 (+) Call the function HAL_SPI_DeInit() to restore the default configuration
221 of the selected SPIx peripheral.
223 @endverbatim
224 * @{
228 * @brief Initialize the SPI according to the specified parameters
229 * in the SPI_InitTypeDef and initialize the associated handle.
230 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
231 * the configuration information for SPI module.
232 * @retval HAL status
234 HAL_StatusTypeDef HAL_SPI_Init(SPI_HandleTypeDef *hspi)
236 uint32_t crc_length = 0UL;
237 uint32_t packet_length;
239 /* Check the SPI handle allocation */
240 if (hspi == NULL)
242 return HAL_ERROR;
245 /* Check the parameters */
246 assert_param(IS_SPI_ALL_INSTANCE(hspi->Instance));
247 assert_param(IS_SPI_MODE(hspi->Init.Mode));
248 assert_param(IS_SPI_DIRECTION(hspi->Init.Direction));
249 assert_param(IS_SPI_DATASIZE(hspi->Init.DataSize));
250 assert_param(IS_SPI_FIFOTHRESHOLD(hspi->Init.FifoThreshold));
251 assert_param(IS_SPI_NSS(hspi->Init.NSS));
252 assert_param(IS_SPI_NSSP(hspi->Init.NSSPMode));
253 assert_param(IS_SPI_BAUDRATE_PRESCALER(hspi->Init.BaudRatePrescaler));
254 assert_param(IS_SPI_FIRST_BIT(hspi->Init.FirstBit));
255 assert_param(IS_SPI_TIMODE(hspi->Init.TIMode));
256 if (hspi->Init.TIMode == SPI_TIMODE_DISABLE)
258 assert_param(IS_SPI_CPOL(hspi->Init.CLKPolarity));
259 assert_param(IS_SPI_CPHA(hspi->Init.CLKPhase));
261 #if (USE_SPI_CRC != 0UL)
262 assert_param(IS_SPI_CRC_CALCULATION(hspi->Init.CRCCalculation));
263 if (hspi->Init.CRCCalculation == SPI_CRCCALCULATION_ENABLE)
265 assert_param(IS_SPI_CRC_POLYNOMIAL(hspi->Init.CRCPolynomial));
266 assert_param(IS_SPI_CRC_LENGTH(hspi->Init.CRCLength));
267 assert_param(IS_SPI_CRC_INITIALIZATION_PATTERN(hspi->Init.TxCRCInitializationPattern));
268 assert_param(IS_SPI_CRC_INITIALIZATION_PATTERN(hspi->Init.RxCRCInitializationPattern));
270 #else
271 hspi->Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE;
272 #endif /* USE_SPI_CRC */
274 /* Verify that the SPI instance supports Data Size higher than 16bits */
275 if ((!IS_SPI_HIGHEND_INSTANCE(hspi->Instance)) && (hspi->Init.DataSize > SPI_DATASIZE_16BIT))
277 return HAL_ERROR;
280 /* Verify that the SPI instance supports requested data packing */
281 packet_length = SPI_GetPacketSize(hspi);
282 if (((!IS_SPI_HIGHEND_INSTANCE(hspi->Instance)) && (packet_length > SPI_LOWEND_FIFO_SIZE)) ||
283 ((IS_SPI_HIGHEND_INSTANCE(hspi->Instance)) && (packet_length > SPI_HIGHEND_FIFO_SIZE)))
285 return HAL_ERROR;
288 #if (USE_SPI_CRC != 0UL)
289 if (hspi->Init.CRCCalculation == SPI_CRCCALCULATION_ENABLE)
291 /* Verify that the SPI instance supports CRC Length higher than 16bits */
292 if ((!IS_SPI_HIGHEND_INSTANCE(hspi->Instance)) && (hspi->Init.CRCLength > SPI_CRC_LENGTH_16BIT))
294 return HAL_ERROR;
297 /* Align the CRC Length on the data size */
298 if (hspi->Init.CRCLength == SPI_CRC_LENGTH_DATASIZE)
300 crc_length = (hspi->Init.DataSize >> SPI_CFG1_DSIZE_Pos) << SPI_CFG1_CRCSIZE_Pos;
302 else
304 crc_length = hspi->Init.CRCLength;
307 /* Verify that the CRC Length is higher than DataSize */
308 if ((hspi->Init.DataSize >> SPI_CFG1_DSIZE_Pos) > (crc_length >> SPI_CFG1_CRCSIZE_Pos))
310 return HAL_ERROR;
313 #endif /* USE_SPI_CRC */
315 if (hspi->State == HAL_SPI_STATE_RESET)
317 /* Allocate lock resource and initialize it */
318 hspi->Lock = HAL_UNLOCKED;
320 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
321 /* Init the SPI Callback settings */
322 hspi->TxCpltCallback = HAL_SPI_TxCpltCallback; /* Legacy weak TxCpltCallback */
323 hspi->RxCpltCallback = HAL_SPI_RxCpltCallback; /* Legacy weak RxCpltCallback */
324 hspi->TxRxCpltCallback = HAL_SPI_TxRxCpltCallback; /* Legacy weak TxRxCpltCallback */
325 hspi->TxHalfCpltCallback = HAL_SPI_TxHalfCpltCallback; /* Legacy weak TxHalfCpltCallback */
326 hspi->RxHalfCpltCallback = HAL_SPI_RxHalfCpltCallback; /* Legacy weak RxHalfCpltCallback */
327 hspi->TxRxHalfCpltCallback = HAL_SPI_TxRxHalfCpltCallback; /* Legacy weak TxRxHalfCpltCallback */
328 hspi->ErrorCallback = HAL_SPI_ErrorCallback; /* Legacy weak ErrorCallback */
329 hspi->AbortCpltCallback = HAL_SPI_AbortCpltCallback; /* Legacy weak AbortCpltCallback */
331 if (hspi->MspInitCallback == NULL)
333 hspi->MspInitCallback = HAL_SPI_MspInit; /* Legacy weak MspInit */
336 /* Init the low level hardware : GPIO, CLOCK, NVIC... */
337 hspi->MspInitCallback(hspi);
338 #else
339 /* Init the low level hardware : GPIO, CLOCK, NVIC... */
340 HAL_SPI_MspInit(hspi);
341 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
344 hspi->State = HAL_SPI_STATE_BUSY;
346 /* Disable the selected SPI peripheral */
347 __HAL_SPI_DISABLE(hspi);
349 /*----------------------- SPIx CR1 & CR2 Configuration ---------------------*/
350 /* Configure : SPI Mode, Communication Mode, Clock polarity and phase, NSS management,
351 Communication speed, First bit, CRC calculation state, CRC Length */
353 if ((hspi->Init.NSS == SPI_NSS_SOFT) && (hspi->Init.Mode == SPI_MODE_MASTER) && (hspi->Init.NSSPolarity == SPI_NSS_POLARITY_LOW))
355 SET_BIT(hspi->Instance->CR1, SPI_CR1_SSI);
358 /* SPIx CFG1 Configuration */
359 WRITE_REG(hspi->Instance->CFG1, (hspi->Init.BaudRatePrescaler | hspi->Init.CRCCalculation | crc_length |
360 hspi->Init.FifoThreshold | hspi->Init.DataSize));
362 /* SPIx CFG2 Configuration */
363 WRITE_REG(hspi->Instance->CFG2, (hspi->Init.NSSPMode | hspi->Init.TIMode | hspi->Init.NSSPolarity |
364 hspi->Init.NSS | hspi->Init.CLKPolarity | hspi->Init.CLKPhase |
365 hspi->Init.FirstBit | hspi->Init.Mode | hspi->Init.MasterInterDataIdleness |
366 hspi->Init.Direction | hspi->Init.MasterSSIdleness | hspi->Init.IOSwap));
368 #if (USE_SPI_CRC != 0UL)
369 /*---------------------------- SPIx CRCPOLY Configuration ------------------*/
370 /* Configure : CRC Polynomial */
371 if (hspi->Init.CRCCalculation == SPI_CRCCALCULATION_ENABLE)
373 /* Initialize TXCRC Pattern Initial Value */
374 if (hspi->Init.TxCRCInitializationPattern == SPI_CRC_INITIALIZATION_ALL_ONE_PATTERN)
376 SET_BIT(hspi->Instance->CR1, SPI_CR1_TCRCINI);
378 else
380 CLEAR_BIT(hspi->Instance->CR1, SPI_CR1_TCRCINI);
383 /* Initialize RXCRC Pattern Initial Value */
384 if (hspi->Init.RxCRCInitializationPattern == SPI_CRC_INITIALIZATION_ALL_ONE_PATTERN)
386 SET_BIT(hspi->Instance->CR1, SPI_CR1_RCRCINI);
388 else
390 CLEAR_BIT(hspi->Instance->CR1, SPI_CR1_RCRCINI);
393 /* Enable 33/17 bits CRC computation */
394 if (((!IS_SPI_HIGHEND_INSTANCE(hspi->Instance)) && (crc_length == SPI_CRC_LENGTH_16BIT)) ||
395 ((IS_SPI_HIGHEND_INSTANCE(hspi->Instance)) && (crc_length == SPI_CRC_LENGTH_32BIT)))
397 SET_BIT(hspi->Instance->CR1, SPI_CR1_CRC33_17);
399 else
401 CLEAR_BIT(hspi->Instance->CR1, SPI_CR1_CRC33_17);
404 /* Write CRC polynomial in SPI Register */
405 WRITE_REG(hspi->Instance->CRCPOLY, hspi->Init.CRCPolynomial);
407 #endif /* USE_SPI_CRC */
409 /* Insure that Underrun configuration is managed only by Salve */
410 if (hspi->Init.Mode == SPI_MODE_SLAVE)
412 /* Set Default Underrun configuration */
413 #if (USE_SPI_CRC != 0UL)
414 if (hspi->Init.CRCCalculation == SPI_CRCCALCULATION_DISABLE)
415 #endif
417 MODIFY_REG(hspi->Instance->CFG1, SPI_CFG1_UDRDET, SPI_CFG1_UDRDET_0);
419 MODIFY_REG(hspi->Instance->CFG1, SPI_CFG1_UDRCFG, SPI_CFG1_UDRCFG_1);
422 #if defined(SPI_I2SCFGR_I2SMOD)
423 /* Activate the SPI mode (Make sure that I2SMOD bit in I2SCFGR register is reset) */
424 CLEAR_BIT(hspi->Instance->I2SCFGR, SPI_I2SCFGR_I2SMOD);
425 #endif /* SPI_I2SCFGR_I2SMOD */
427 /* Insure that AFCNTR is managed only by Master */
428 if ((hspi->Init.Mode & SPI_MODE_MASTER) == SPI_MODE_MASTER)
430 /* Alternate function GPIOs control */
431 MODIFY_REG(hspi->Instance->CFG2, SPI_CFG2_AFCNTR, (hspi->Init.MasterKeepIOState));
434 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
435 hspi->State = HAL_SPI_STATE_READY;
437 return HAL_OK;
441 * @brief De-Initialize the SPI peripheral.
442 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
443 * the configuration information for SPI module.
444 * @retval HAL status
446 HAL_StatusTypeDef HAL_SPI_DeInit(SPI_HandleTypeDef *hspi)
448 /* Check the SPI handle allocation */
449 if (hspi == NULL)
451 return HAL_ERROR;
454 /* Check SPI Instance parameter */
455 assert_param(IS_SPI_ALL_INSTANCE(hspi->Instance));
457 hspi->State = HAL_SPI_STATE_BUSY;
459 /* Disable the SPI Peripheral Clock */
460 __HAL_SPI_DISABLE(hspi);
462 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
463 if (hspi->MspDeInitCallback == NULL)
465 hspi->MspDeInitCallback = HAL_SPI_MspDeInit; /* Legacy weak MspDeInit */
468 /* DeInit the low level hardware: GPIO, CLOCK, NVIC... */
469 hspi->MspDeInitCallback(hspi);
470 #else
471 /* DeInit the low level hardware: GPIO, CLOCK, NVIC... */
472 HAL_SPI_MspDeInit(hspi);
473 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
475 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
476 hspi->State = HAL_SPI_STATE_RESET;
478 /* Release Lock */
479 __HAL_UNLOCK(hspi);
481 return HAL_OK;
485 * @brief Initialize the SPI MSP.
486 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
487 * the configuration information for SPI module.
488 * @retval None
490 __weak void HAL_SPI_MspInit(SPI_HandleTypeDef *hspi)
492 /* Prevent unused argument(s) compilation warning */
493 UNUSED(hspi);
495 /* NOTE : This function should not be modified, when the callback is needed,
496 the HAL_SPI_MspInit should be implemented in the user file
501 * @brief De-Initialize the SPI MSP.
502 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
503 * the configuration information for SPI module.
504 * @retval None
506 __weak void HAL_SPI_MspDeInit(SPI_HandleTypeDef *hspi)
508 /* Prevent unused argument(s) compilation warning */
509 UNUSED(hspi);
511 /* NOTE : This function should not be modified, when the callback is needed,
512 the HAL_SPI_MspDeInit should be implemented in the user file
516 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
518 * @brief Register a User SPI Callback
519 * To be used instead of the weak predefined callback
520 * @param hspi Pointer to a SPI_HandleTypeDef structure that contains
521 * the configuration information for the specified SPI.
522 * @param CallbackID ID of the callback to be registered
523 * @param pCallback pointer to the Callback function
524 * @retval HAL status
526 HAL_StatusTypeDef HAL_SPI_RegisterCallback(SPI_HandleTypeDef *hspi, HAL_SPI_CallbackIDTypeDef CallbackID, pSPI_CallbackTypeDef pCallback)
528 HAL_StatusTypeDef status = HAL_OK;
530 if (pCallback == NULL)
532 /* Update the error code */
533 hspi->ErrorCode |= HAL_SPI_ERROR_INVALID_CALLBACK;
535 return HAL_ERROR;
537 /* Process locked */
538 __HAL_LOCK(hspi);
540 if (HAL_SPI_STATE_READY == hspi->State)
542 switch (CallbackID)
544 case HAL_SPI_TX_COMPLETE_CB_ID :
545 hspi->TxCpltCallback = pCallback;
546 break;
548 case HAL_SPI_RX_COMPLETE_CB_ID :
549 hspi->RxCpltCallback = pCallback;
550 break;
552 case HAL_SPI_TX_RX_COMPLETE_CB_ID :
553 hspi->TxRxCpltCallback = pCallback;
554 break;
556 case HAL_SPI_TX_HALF_COMPLETE_CB_ID :
557 hspi->TxHalfCpltCallback = pCallback;
558 break;
560 case HAL_SPI_RX_HALF_COMPLETE_CB_ID :
561 hspi->RxHalfCpltCallback = pCallback;
562 break;
564 case HAL_SPI_TX_RX_HALF_COMPLETE_CB_ID :
565 hspi->TxRxHalfCpltCallback = pCallback;
566 break;
568 case HAL_SPI_ERROR_CB_ID :
569 hspi->ErrorCallback = pCallback;
570 break;
572 case HAL_SPI_ABORT_CB_ID :
573 hspi->AbortCpltCallback = pCallback;
574 break;
576 case HAL_SPI_MSPINIT_CB_ID :
577 hspi->MspInitCallback = pCallback;
578 break;
580 case HAL_SPI_MSPDEINIT_CB_ID :
581 hspi->MspDeInitCallback = pCallback;
582 break;
584 default :
585 /* Update the error code */
586 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_INVALID_CALLBACK);
588 /* Return error status */
589 status = HAL_ERROR;
590 break;
593 else if (HAL_SPI_STATE_RESET == hspi->State)
595 switch (CallbackID)
597 case HAL_SPI_MSPINIT_CB_ID :
598 hspi->MspInitCallback = pCallback;
599 break;
601 case HAL_SPI_MSPDEINIT_CB_ID :
602 hspi->MspDeInitCallback = pCallback;
603 break;
605 default :
606 /* Update the error code */
607 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_INVALID_CALLBACK);
609 /* Return error status */
610 status = HAL_ERROR;
611 break;
614 else
616 /* Update the error code */
617 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_INVALID_CALLBACK);
619 /* Return error status */
620 status = HAL_ERROR;
623 /* Release Lock */
624 __HAL_UNLOCK(hspi);
625 return status;
629 * @brief Unregister an SPI Callback
630 * SPI callback is redirected to the weak predefined callback
631 * @param hspi Pointer to a SPI_HandleTypeDef structure that contains
632 * the configuration information for the specified SPI.
633 * @param CallbackID ID of the callback to be unregistered
634 * @retval HAL status
636 HAL_StatusTypeDef HAL_SPI_UnRegisterCallback(SPI_HandleTypeDef *hspi, HAL_SPI_CallbackIDTypeDef CallbackID)
638 HAL_StatusTypeDef status = HAL_OK;
640 /* Process locked */
641 __HAL_LOCK(hspi);
643 if (HAL_SPI_STATE_READY == hspi->State)
645 switch (CallbackID)
647 case HAL_SPI_TX_COMPLETE_CB_ID :
648 hspi->TxCpltCallback = HAL_SPI_TxCpltCallback; /* Legacy weak TxCpltCallback */
649 break;
651 case HAL_SPI_RX_COMPLETE_CB_ID :
652 hspi->RxCpltCallback = HAL_SPI_RxCpltCallback; /* Legacy weak RxCpltCallback */
653 break;
655 case HAL_SPI_TX_RX_COMPLETE_CB_ID :
656 hspi->TxRxCpltCallback = HAL_SPI_TxRxCpltCallback; /* Legacy weak TxRxCpltCallback */
657 break;
659 case HAL_SPI_TX_HALF_COMPLETE_CB_ID :
660 hspi->TxHalfCpltCallback = HAL_SPI_TxHalfCpltCallback; /* Legacy weak TxHalfCpltCallback */
661 break;
663 case HAL_SPI_RX_HALF_COMPLETE_CB_ID :
664 hspi->RxHalfCpltCallback = HAL_SPI_RxHalfCpltCallback; /* Legacy weak RxHalfCpltCallback */
665 break;
667 case HAL_SPI_TX_RX_HALF_COMPLETE_CB_ID :
668 hspi->TxRxHalfCpltCallback = HAL_SPI_TxRxHalfCpltCallback; /* Legacy weak TxRxHalfCpltCallback */
669 break;
671 case HAL_SPI_ERROR_CB_ID :
672 hspi->ErrorCallback = HAL_SPI_ErrorCallback; /* Legacy weak ErrorCallback */
673 break;
675 case HAL_SPI_ABORT_CB_ID :
676 hspi->AbortCpltCallback = HAL_SPI_AbortCpltCallback; /* Legacy weak AbortCpltCallback */
677 break;
679 case HAL_SPI_MSPINIT_CB_ID :
680 hspi->MspInitCallback = HAL_SPI_MspInit; /* Legacy weak MspInit */
681 break;
683 case HAL_SPI_MSPDEINIT_CB_ID :
684 hspi->MspDeInitCallback = HAL_SPI_MspDeInit; /* Legacy weak MspDeInit */
685 break;
687 default :
688 /* Update the error code */
689 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_INVALID_CALLBACK);
691 /* Return error status */
692 status = HAL_ERROR;
693 break;
696 else if (HAL_SPI_STATE_RESET == hspi->State)
698 switch (CallbackID)
700 case HAL_SPI_MSPINIT_CB_ID :
701 hspi->MspInitCallback = HAL_SPI_MspInit; /* Legacy weak MspInit */
702 break;
704 case HAL_SPI_MSPDEINIT_CB_ID :
705 hspi->MspDeInitCallback = HAL_SPI_MspDeInit; /* Legacy weak MspDeInit */
706 break;
708 default :
709 /* Update the error code */
710 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_INVALID_CALLBACK);
712 /* Return error status */
713 status = HAL_ERROR;
714 break;
717 else
719 /* Update the error code */
720 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_INVALID_CALLBACK);
722 /* Return error status */
723 status = HAL_ERROR;
726 /* Release Lock */
727 __HAL_UNLOCK(hspi);
728 return status;
730 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
732 * @}
735 /** @defgroup SPI_Exported_Functions_Group2 IO operation functions
736 * @brief Data transfers functions
738 @verbatim
739 ==============================================================================
740 ##### IO operation functions #####
741 ===============================================================================
742 [..]
743 This subsection provides a set of functions allowing to manage the SPI
744 data transfers.
746 [..] The SPI supports master and slave mode :
748 (#) There are two modes of transfer:
749 (##) Blocking mode: The communication is performed in polling mode.
750 The HAL status of all data processing is returned by the same function
751 after finishing transfer.
752 (##) No-Blocking mode: The communication is performed using Interrupts
753 or DMA, These APIs return the HAL status.
754 The end of the data processing will be indicated through the
755 dedicated SPI IRQ when using Interrupt mode or the DMA IRQ when
756 using DMA mode.
757 The HAL_SPI_TxCpltCallback(), HAL_SPI_RxCpltCallback() and HAL_SPI_TxRxCpltCallback() user callbacks
758 will be executed respectively at the end of the transmit or Receive process
759 The HAL_SPI_ErrorCallback()user callback will be executed when a communication error is detected
761 (#) APIs provided for these 2 transfer modes (Blocking mode or Non blocking mode using either Interrupt or DMA)
762 exist for 1Line (simplex) and 2Lines (full duplex) modes.
764 @endverbatim
765 * @{
769 * @brief Transmit an amount of data in blocking mode.
770 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
771 * the configuration information for SPI module.
772 * @param pData : pointer to data buffer
773 * @param Size : amount of data to be sent
774 * @param Timeout: Timeout duration
775 * @retval HAL status
777 HAL_StatusTypeDef HAL_SPI_Transmit(SPI_HandleTypeDef *hspi, const uint8_t *pData, uint16_t Size, uint32_t Timeout)
779 #if defined (__GNUC__)
780 __IO uint16_t *ptxdr_16bits = (__IO uint16_t *)(&(hspi->Instance->TXDR));
781 #endif /* __GNUC__ */
783 uint32_t tickstart;
784 HAL_StatusTypeDef errorcode = HAL_OK;
786 /* Check Direction parameter */
787 assert_param(IS_SPI_DIRECTION_2LINES_OR_1LINE_2LINES_TXONLY(hspi->Init.Direction));
789 /* Process Locked */
790 __HAL_LOCK(hspi);
792 /* Init tickstart for timeout management*/
793 tickstart = HAL_GetTick();
795 if (hspi->State != HAL_SPI_STATE_READY)
797 errorcode = HAL_BUSY;
798 __HAL_UNLOCK(hspi);
799 return errorcode;
802 if ((pData == NULL) || (Size == 0UL))
804 errorcode = HAL_ERROR;
805 __HAL_UNLOCK(hspi);
806 return errorcode;
809 /* Set the transaction information */
810 hspi->State = HAL_SPI_STATE_BUSY_TX;
811 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
812 hspi->pTxBuffPtr = (uint8_t *)pData;
813 hspi->TxXferSize = Size;
814 hspi->TxXferCount = Size;
816 /*Init field not used in handle to zero */
817 hspi->pRxBuffPtr = NULL;
818 hspi->RxXferSize = (uint16_t) 0UL;
819 hspi->RxXferCount = (uint16_t) 0UL;
820 hspi->TxISR = NULL;
821 hspi->RxISR = NULL;
823 /* Configure communication direction : 1Line */
824 if (hspi->Init.Direction == SPI_DIRECTION_1LINE)
826 SPI_1LINE_TX(hspi);
829 /* Set the number of data at current transfer */
830 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, Size);
832 /* Enable SPI peripheral */
833 __HAL_SPI_ENABLE(hspi);
835 if (hspi->Init.Mode == SPI_MODE_MASTER)
837 /* Master transfer start */
838 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSTART);
841 /* Transmit data in 32 Bit mode */
842 if (hspi->Init.DataSize > SPI_DATASIZE_16BIT)
844 /* Transmit data in 32 Bit mode */
845 while (hspi->TxXferCount > 0UL)
847 /* Wait until TXP flag is set to send data */
848 if (__HAL_SPI_GET_FLAG(hspi, SPI_FLAG_TXP))
850 *((__IO uint32_t *)&hspi->Instance->TXDR) = *((uint32_t *)hspi->pTxBuffPtr);
851 hspi->pTxBuffPtr += sizeof(uint32_t);
852 hspi->TxXferCount--;
854 else
856 /* Timeout management */
857 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U))
859 /* Call standard close procedure with error check */
860 SPI_CloseTransfer(hspi);
862 /* Process Unlocked */
863 __HAL_UNLOCK(hspi);
865 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_TIMEOUT);
866 hspi->State = HAL_SPI_STATE_READY;
867 return HAL_ERROR;
872 /* Transmit data in 16 Bit mode */
873 else if (hspi->Init.DataSize > SPI_DATASIZE_8BIT)
875 /* Transmit data in 16 Bit mode */
876 while (hspi->TxXferCount > 0UL)
878 /* Wait until TXP flag is set to send data */
879 if (__HAL_SPI_GET_FLAG(hspi, SPI_FLAG_TXP))
881 if ((hspi->TxXferCount > 1UL) && (hspi->Init.FifoThreshold > SPI_FIFO_THRESHOLD_01DATA))
883 *((__IO uint32_t *)&hspi->Instance->TXDR) = *((uint32_t *)hspi->pTxBuffPtr);
884 hspi->pTxBuffPtr += sizeof(uint32_t);
885 hspi->TxXferCount -= (uint16_t)2UL;
887 else
889 #if defined (__GNUC__)
890 *ptxdr_16bits = *((uint16_t *)hspi->pTxBuffPtr);
891 #else
892 *((__IO uint16_t *)&hspi->Instance->TXDR) = *((uint16_t *)hspi->pTxBuffPtr);
893 #endif /* __GNUC__ */
894 hspi->pTxBuffPtr += sizeof(uint16_t);
895 hspi->TxXferCount--;
898 else
900 /* Timeout management */
901 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U))
903 /* Call standard close procedure with error check */
904 SPI_CloseTransfer(hspi);
906 /* Process Unlocked */
907 __HAL_UNLOCK(hspi);
909 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_TIMEOUT);
910 hspi->State = HAL_SPI_STATE_READY;
911 return HAL_ERROR;
916 /* Transmit data in 8 Bit mode */
917 else
919 while (hspi->TxXferCount > 0UL)
921 /* Wait until TXP flag is set to send data */
922 if (__HAL_SPI_GET_FLAG(hspi, SPI_FLAG_TXP))
924 if ((hspi->TxXferCount > 3UL) && (hspi->Init.FifoThreshold > SPI_FIFO_THRESHOLD_03DATA))
926 *((__IO uint32_t *)&hspi->Instance->TXDR) = *((uint32_t *)hspi->pTxBuffPtr);
927 hspi->pTxBuffPtr += sizeof(uint32_t);
928 hspi->TxXferCount -= (uint16_t)4UL;
930 else if ((hspi->TxXferCount > 1UL) && (hspi->Init.FifoThreshold > SPI_FIFO_THRESHOLD_01DATA))
932 #if defined (__GNUC__)
933 *ptxdr_16bits = *((uint16_t *)hspi->pTxBuffPtr);
934 #else
935 *((__IO uint16_t *)&hspi->Instance->TXDR) = *((uint16_t *)hspi->pTxBuffPtr);
936 #endif /* __GNUC__ */
937 hspi->pTxBuffPtr += sizeof(uint16_t);
938 hspi->TxXferCount -= (uint16_t)2UL;
940 else
942 *((__IO uint8_t *)&hspi->Instance->TXDR) = *((uint8_t *)hspi->pTxBuffPtr);
943 hspi->pTxBuffPtr += sizeof(uint8_t);
944 hspi->TxXferCount--;
947 else
949 /* Timeout management */
950 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U))
952 /* Call standard close procedure with error check */
953 SPI_CloseTransfer(hspi);
955 /* Process Unlocked */
956 __HAL_UNLOCK(hspi);
958 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_TIMEOUT);
959 hspi->State = HAL_SPI_STATE_READY;
960 return HAL_ERROR;
966 /* Wait for Tx (and CRC) data to be sent */
967 if (SPI_WaitOnFlagUntilTimeout(hspi, SPI_FLAG_EOT, RESET, tickstart, Timeout) != HAL_OK)
969 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_FLAG);
972 /* Call standard close procedure with error check */
973 SPI_CloseTransfer(hspi);
975 /* Process Unlocked */
976 __HAL_UNLOCK(hspi);
978 hspi->State = HAL_SPI_STATE_READY;
980 if (hspi->ErrorCode != HAL_SPI_ERROR_NONE)
982 return HAL_ERROR;
984 return errorcode;
988 * @brief Receive an amount of data in blocking mode.
989 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
990 * the configuration information for SPI module.
991 * @param pData : pointer to data buffer
992 * @param Size : amount of data to be received
993 * @param Timeout: Timeout duration
994 * @retval HAL status
996 HAL_StatusTypeDef HAL_SPI_Receive(SPI_HandleTypeDef *hspi, uint8_t *pData, uint16_t Size, uint32_t Timeout)
998 uint32_t tickstart;
999 HAL_StatusTypeDef errorcode = HAL_OK;
1000 #if defined (__GNUC__)
1001 __IO uint16_t *prxdr_16bits = (__IO uint16_t *)(&(hspi->Instance->RXDR));
1002 #endif /* __GNUC__ */
1004 /* Check Direction parameter */
1005 assert_param(IS_SPI_DIRECTION_2LINES_OR_1LINE_2LINES_RXONLY(hspi->Init.Direction));
1007 if ((hspi->Init.Mode == SPI_MODE_MASTER) && (hspi->Init.Direction == SPI_DIRECTION_2LINES))
1009 hspi->State = HAL_SPI_STATE_BUSY_RX;
1010 /* Call transmit-receive function to send Dummy data on Tx line and generate clock on CLK line */
1011 return HAL_SPI_TransmitReceive(hspi, pData, pData, Size, Timeout);
1014 /* Process Locked */
1015 __HAL_LOCK(hspi);
1017 /* Init tickstart for timeout management*/
1018 tickstart = HAL_GetTick();
1020 if (hspi->State != HAL_SPI_STATE_READY)
1022 errorcode = HAL_BUSY;
1023 __HAL_UNLOCK(hspi);
1024 return errorcode;
1027 if ((pData == NULL) || (Size == 0UL))
1029 errorcode = HAL_ERROR;
1030 __HAL_UNLOCK(hspi);
1031 return errorcode;
1034 /* Set the transaction information */
1035 hspi->State = HAL_SPI_STATE_BUSY_RX;
1036 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
1037 hspi->pRxBuffPtr = (uint8_t *)pData;
1038 hspi->RxXferSize = Size;
1039 hspi->RxXferCount = Size;
1041 /*Init field not used in handle to zero */
1042 hspi->pTxBuffPtr = NULL;
1043 hspi->TxXferSize = (uint16_t) 0UL;
1044 hspi->TxXferCount = (uint16_t) 0UL;
1045 hspi->RxISR = NULL;
1046 hspi->TxISR = NULL;
1048 /* Configure communication direction: 1Line */
1049 if (hspi->Init.Direction == SPI_DIRECTION_1LINE)
1051 SPI_1LINE_RX(hspi);
1054 /* Set the number of data at current transfer */
1055 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, Size);
1057 /* Enable SPI peripheral */
1058 __HAL_SPI_ENABLE(hspi);
1060 if (hspi->Init.Mode == SPI_MODE_MASTER)
1062 /* Master transfer start */
1063 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSTART);
1066 /* Receive data in 32 Bit mode */
1067 if (hspi->Init.DataSize > SPI_DATASIZE_16BIT)
1069 /* Transfer loop */
1070 while (hspi->RxXferCount > 0UL)
1072 /* Check the RXWNE/EOT flag */
1073 if ((hspi->Instance->SR & (SPI_FLAG_RXWNE | SPI_FLAG_EOT)) != 0UL)
1075 *((uint32_t *)hspi->pRxBuffPtr) = *((__IO uint32_t *)&hspi->Instance->RXDR);
1076 hspi->pRxBuffPtr += sizeof(uint32_t);
1077 hspi->RxXferCount--;
1079 else
1081 /* Timeout management */
1082 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U))
1084 /* Call standard close procedure with error check */
1085 SPI_CloseTransfer(hspi);
1087 /* Process Unlocked */
1088 __HAL_UNLOCK(hspi);
1090 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_TIMEOUT);
1091 hspi->State = HAL_SPI_STATE_READY;
1092 return HAL_ERROR;
1097 /* Receive data in 16 Bit mode */
1098 else if (hspi->Init.DataSize > SPI_DATASIZE_8BIT)
1100 /* Transfer loop */
1101 while (hspi->RxXferCount > 0UL)
1103 /* Check the RXWNE/FRLVL flag */
1104 if ((hspi->Instance->SR & (SPI_FLAG_RXWNE | SPI_FLAG_FRLVL)) != 0UL)
1106 if ((hspi->Instance->SR & SPI_FLAG_RXWNE) != 0UL)
1108 *((uint32_t *)hspi->pRxBuffPtr) = *((__IO uint32_t *)&hspi->Instance->RXDR);
1109 hspi->pRxBuffPtr += sizeof(uint32_t);
1110 hspi->RxXferCount -= (uint16_t)2UL;
1112 else
1114 #if defined (__GNUC__)
1115 *((uint16_t *)hspi->pRxBuffPtr) = *prxdr_16bits;
1116 #else
1117 *((uint16_t *)hspi->pRxBuffPtr) = *((__IO uint16_t *)&hspi->Instance->RXDR);
1118 #endif /* __GNUC__ */
1119 hspi->pRxBuffPtr += sizeof(uint16_t);
1120 hspi->RxXferCount--;
1123 else
1125 /* Timeout management */
1126 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U))
1128 /* Call standard close procedure with error check */
1129 SPI_CloseTransfer(hspi);
1131 /* Process Unlocked */
1132 __HAL_UNLOCK(hspi);
1134 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_TIMEOUT);
1135 hspi->State = HAL_SPI_STATE_READY;
1136 return HAL_ERROR;
1141 /* Receive data in 8 Bit mode */
1142 else
1144 /* Transfer loop */
1145 while (hspi->RxXferCount > 0UL)
1147 /* Check the RXWNE/FRLVL flag */
1148 if ((hspi->Instance->SR & (SPI_FLAG_RXWNE | SPI_FLAG_FRLVL)) != 0UL)
1150 if ((hspi->Instance->SR & SPI_FLAG_RXWNE) != 0UL)
1152 *((uint32_t *)hspi->pRxBuffPtr) = *((__IO uint32_t *)&hspi->Instance->RXDR);
1153 hspi->pRxBuffPtr += sizeof(uint32_t);
1154 hspi->RxXferCount -= (uint16_t)4UL;
1156 else if ((hspi->Instance->SR & SPI_FLAG_FRLVL) > SPI_RX_FIFO_1PACKET)
1158 #if defined (__GNUC__)
1159 *((uint16_t *)hspi->pRxBuffPtr) = *prxdr_16bits;
1160 #else
1161 *((uint16_t *)hspi->pRxBuffPtr) = *((__IO uint16_t *)&hspi->Instance->RXDR);
1162 #endif /* __GNUC__ */
1163 hspi->pRxBuffPtr += sizeof(uint16_t);
1164 hspi->RxXferCount -= (uint16_t)2UL;
1166 else
1168 *((uint8_t *)hspi->pRxBuffPtr) = *((__IO uint8_t *)&hspi->Instance->RXDR);
1169 hspi->pRxBuffPtr += sizeof(uint8_t);
1170 hspi->RxXferCount--;
1173 else
1175 /* Timeout management */
1176 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U))
1178 /* Call standard close procedure with error check */
1179 SPI_CloseTransfer(hspi);
1181 /* Process Unlocked */
1182 __HAL_UNLOCK(hspi);
1184 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_TIMEOUT);
1185 hspi->State = HAL_SPI_STATE_READY;
1186 return HAL_ERROR;
1192 #if (USE_SPI_CRC != 0UL)
1193 if (hspi->Init.CRCCalculation == SPI_CRCCALCULATION_ENABLE)
1195 /* Wait for crc data to be received */
1196 if (SPI_WaitOnFlagUntilTimeout(hspi, SPI_FLAG_EOT, RESET, tickstart, Timeout) != HAL_OK)
1198 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_FLAG);
1201 #endif /* USE_SPI_CRC */
1203 /* Call standard close procedure with error check */
1204 SPI_CloseTransfer(hspi);
1206 /* Process Unlocked */
1207 __HAL_UNLOCK(hspi);
1209 hspi->State = HAL_SPI_STATE_READY;
1211 if (hspi->ErrorCode != HAL_SPI_ERROR_NONE)
1213 return HAL_ERROR;
1215 return errorcode;
1219 * @brief Transmit and Receive an amount of data in blocking mode.
1220 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
1221 * the configuration information for SPI module.
1222 * @param pTxData: pointer to transmission data buffer
1223 * @param pRxData: pointer to reception data buffer
1224 * @param Size : amount of data to be sent and received
1225 * @param Timeout: Timeout duration
1226 * @retval HAL status
1228 HAL_StatusTypeDef HAL_SPI_TransmitReceive(SPI_HandleTypeDef *hspi, const uint8_t *pTxData, uint8_t *pRxData, uint16_t Size,
1229 uint32_t Timeout)
1231 HAL_SPI_StateTypeDef tmp_state;
1232 HAL_StatusTypeDef errorcode = HAL_OK;
1233 #if defined (__GNUC__)
1234 __IO uint16_t *ptxdr_16bits = (__IO uint16_t *)(&(hspi->Instance->TXDR));
1235 __IO uint16_t *prxdr_16bits = (__IO uint16_t *)(&(hspi->Instance->RXDR));
1236 #endif /* __GNUC__ */
1238 uint32_t tickstart;
1239 uint32_t tmp_mode;
1240 uint16_t initial_TxXferCount;
1241 uint16_t initial_RxXferCount;
1243 /* Check Direction parameter */
1244 assert_param(IS_SPI_DIRECTION_2LINES(hspi->Init.Direction));
1246 /* Process Locked */
1247 __HAL_LOCK(hspi);
1249 /* Init tickstart for timeout management*/
1250 tickstart = HAL_GetTick();
1252 initial_TxXferCount = Size;
1253 initial_RxXferCount = Size;
1254 tmp_state = hspi->State;
1255 tmp_mode = hspi->Init.Mode;
1257 if (!((tmp_state == HAL_SPI_STATE_READY) || \
1258 ((tmp_mode == SPI_MODE_MASTER) && (hspi->Init.Direction == SPI_DIRECTION_2LINES) && (tmp_state == HAL_SPI_STATE_BUSY_RX))))
1260 errorcode = HAL_BUSY;
1261 __HAL_UNLOCK(hspi);
1262 return errorcode;
1265 if ((pTxData == NULL) || (pRxData == NULL) || (Size == 0UL))
1267 errorcode = HAL_ERROR;
1268 __HAL_UNLOCK(hspi);
1269 return errorcode;
1272 /* Don't overwrite in case of HAL_SPI_STATE_BUSY_RX */
1273 if (hspi->State != HAL_SPI_STATE_BUSY_RX)
1275 hspi->State = HAL_SPI_STATE_BUSY_TX_RX;
1278 /* Set the transaction information */
1279 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
1280 hspi->pRxBuffPtr = (uint8_t *)pRxData;
1281 hspi->RxXferCount = Size;
1282 hspi->RxXferSize = Size;
1283 hspi->pTxBuffPtr = (uint8_t *)pTxData;
1284 hspi->TxXferCount = Size;
1285 hspi->TxXferSize = Size;
1287 /*Init field not used in handle to zero */
1288 hspi->RxISR = NULL;
1289 hspi->TxISR = NULL;
1291 /* Set the number of data at current transfer */
1292 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, Size);
1294 __HAL_SPI_ENABLE(hspi);
1296 if (hspi->Init.Mode == SPI_MODE_MASTER)
1298 /* Master transfer start */
1299 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSTART);
1302 /* Transmit and Receive data in 32 Bit mode */
1303 if (hspi->Init.DataSize > SPI_DATASIZE_16BIT)
1305 while ((initial_TxXferCount > 0UL) || (initial_RxXferCount > 0UL))
1307 /* Check TXP flag */
1308 if ((__HAL_SPI_GET_FLAG(hspi, SPI_FLAG_TXP)) && (initial_TxXferCount > 0UL))
1310 *((__IO uint32_t *)&hspi->Instance->TXDR) = *((uint32_t *)hspi->pTxBuffPtr);
1311 hspi->pTxBuffPtr += sizeof(uint32_t);
1312 hspi->TxXferCount --;
1313 initial_TxXferCount = hspi->TxXferCount;
1316 /* Check RXWNE/EOT flag */
1317 if (((hspi->Instance->SR & (SPI_FLAG_RXWNE | SPI_FLAG_EOT)) != 0UL) && (initial_RxXferCount > 0UL))
1319 *((uint32_t *)hspi->pRxBuffPtr) = *((__IO uint32_t *)&hspi->Instance->RXDR);
1320 hspi->pRxBuffPtr += sizeof(uint32_t);
1321 hspi->RxXferCount --;
1322 initial_RxXferCount = hspi->RxXferCount;
1325 /* Timeout management */
1326 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U))
1328 /* Call standard close procedure with error check */
1329 SPI_CloseTransfer(hspi);
1331 /* Process Unlocked */
1332 __HAL_UNLOCK(hspi);
1334 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_TIMEOUT);
1335 hspi->State = HAL_SPI_STATE_READY;
1336 return HAL_ERROR;
1340 /* Transmit and Receive data in 16 Bit mode */
1341 else if (hspi->Init.DataSize > SPI_DATASIZE_8BIT)
1343 while ((initial_TxXferCount > 0UL) || (initial_RxXferCount > 0UL))
1345 /* Check TXP flag */
1346 if (__HAL_SPI_GET_FLAG(hspi, SPI_FLAG_TXP) && (initial_TxXferCount > 0UL))
1348 if ((initial_TxXferCount > 1UL) && (hspi->Init.FifoThreshold > SPI_FIFO_THRESHOLD_01DATA))
1350 *((__IO uint32_t *)&hspi->Instance->TXDR) = *((uint32_t *)hspi->pTxBuffPtr);
1351 hspi->pTxBuffPtr += sizeof(uint32_t);
1352 hspi->TxXferCount -= (uint16_t)2UL;
1353 initial_TxXferCount = hspi->TxXferCount;
1355 else
1357 #if defined (__GNUC__)
1358 *ptxdr_16bits = *((uint16_t *)hspi->pTxBuffPtr);
1359 #else
1360 *((__IO uint16_t *)&hspi->Instance->TXDR) = *((uint16_t *)hspi->pTxBuffPtr);
1361 #endif /* __GNUC__ */
1362 hspi->pTxBuffPtr += sizeof(uint16_t);
1363 hspi->TxXferCount--;
1364 initial_TxXferCount = hspi->TxXferCount;
1368 /* Check RXWNE/FRLVL flag */
1369 if (((hspi->Instance->SR & (SPI_FLAG_RXWNE | SPI_FLAG_FRLVL)) != 0UL) && (initial_RxXferCount > 0UL))
1371 if ((hspi->Instance->SR & SPI_FLAG_RXWNE) != 0UL)
1373 *((uint32_t *)hspi->pRxBuffPtr) = *((__IO uint32_t *)&hspi->Instance->RXDR);
1374 hspi->pRxBuffPtr += sizeof(uint32_t);
1375 hspi->RxXferCount -= (uint16_t)2UL;
1376 initial_RxXferCount = hspi->RxXferCount;
1378 else
1380 #if defined (__GNUC__)
1381 *((uint16_t *)hspi->pRxBuffPtr) = *prxdr_16bits;
1382 #else
1383 *((uint16_t *)hspi->pRxBuffPtr) = *((__IO uint16_t *)&hspi->Instance->RXDR);
1384 #endif /* __GNUC__ */
1385 hspi->pRxBuffPtr += sizeof(uint16_t);
1386 hspi->RxXferCount--;
1387 initial_RxXferCount = hspi->RxXferCount;
1391 /* Timeout management */
1392 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U))
1394 /* Call standard close procedure with error check */
1395 SPI_CloseTransfer(hspi);
1397 /* Process Unlocked */
1398 __HAL_UNLOCK(hspi);
1400 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_TIMEOUT);
1401 hspi->State = HAL_SPI_STATE_READY;
1402 return HAL_ERROR;
1406 /* Transmit and Receive data in 8 Bit mode */
1407 else
1409 while ((initial_TxXferCount > 0UL) || (initial_RxXferCount > 0UL))
1411 /* check TXP flag */
1412 if ((__HAL_SPI_GET_FLAG(hspi, SPI_FLAG_TXP)) && (initial_TxXferCount > 0UL))
1414 if ((initial_TxXferCount > 3UL) && (hspi->Init.FifoThreshold > SPI_FIFO_THRESHOLD_03DATA))
1416 *((__IO uint32_t *)&hspi->Instance->TXDR) = *((uint32_t *)hspi->pTxBuffPtr);
1417 hspi->pTxBuffPtr += sizeof(uint32_t);
1418 hspi->TxXferCount -= (uint16_t)4UL;
1419 initial_TxXferCount = hspi->TxXferCount;
1421 else if ((initial_TxXferCount > 1UL) && (hspi->Init.FifoThreshold > SPI_FIFO_THRESHOLD_01DATA))
1423 #if defined (__GNUC__)
1424 *ptxdr_16bits = *((uint16_t *)hspi->pTxBuffPtr);
1425 #else
1426 *((__IO uint16_t *)&hspi->Instance->TXDR) = *((uint16_t *)hspi->pTxBuffPtr);
1427 #endif /* __GNUC__ */
1428 hspi->pTxBuffPtr += sizeof(uint16_t);
1429 hspi->TxXferCount -= (uint16_t)2UL;
1430 initial_TxXferCount = hspi->TxXferCount;
1432 else
1434 *((__IO uint8_t *)&hspi->Instance->TXDR) = *((uint8_t *)hspi->pTxBuffPtr);
1435 hspi->pTxBuffPtr += sizeof(uint8_t);
1436 hspi->TxXferCount--;
1437 initial_TxXferCount = hspi->TxXferCount;
1441 /* Wait until RXWNE/FRLVL flag is reset */
1442 if (((hspi->Instance->SR & (SPI_FLAG_RXWNE | SPI_FLAG_FRLVL)) != 0UL) && (initial_RxXferCount > 0UL))
1444 if ((hspi->Instance->SR & SPI_FLAG_RXWNE) != 0UL)
1446 *((uint32_t *)hspi->pRxBuffPtr) = *((__IO uint32_t *)&hspi->Instance->RXDR);
1447 hspi->pRxBuffPtr += sizeof(uint32_t);
1448 hspi->RxXferCount -= (uint16_t)4UL;
1449 initial_RxXferCount = hspi->RxXferCount;
1451 else if ((hspi->Instance->SR & SPI_FLAG_FRLVL) > SPI_RX_FIFO_1PACKET)
1453 #if defined (__GNUC__)
1454 *((uint16_t *)hspi->pRxBuffPtr) = *prxdr_16bits;
1455 #else
1456 *((uint16_t *)hspi->pRxBuffPtr) = *((__IO uint16_t *)&hspi->Instance->RXDR);
1457 #endif /* __GNUC__ */
1458 hspi->pRxBuffPtr += sizeof(uint16_t);
1459 hspi->RxXferCount -= (uint16_t)2UL;
1460 initial_RxXferCount = hspi->RxXferCount;
1462 else
1464 *((uint8_t *)hspi->pRxBuffPtr) = *((__IO uint8_t *)&hspi->Instance->RXDR);
1465 hspi->pRxBuffPtr += sizeof(uint8_t);
1466 hspi->RxXferCount--;
1467 initial_RxXferCount = hspi->RxXferCount;
1471 /* Timeout management */
1472 if ((((HAL_GetTick() - tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U))
1474 /* Call standard close procedure with error check */
1475 SPI_CloseTransfer(hspi);
1477 /* Process Unlocked */
1478 __HAL_UNLOCK(hspi);
1480 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_TIMEOUT);
1481 hspi->State = HAL_SPI_STATE_READY;
1482 return HAL_ERROR;
1487 /* Wait for Tx/Rx (and CRC) data to be sent/received */
1488 if (SPI_WaitOnFlagUntilTimeout(hspi, SPI_FLAG_EOT, RESET, tickstart, Timeout) != HAL_OK)
1490 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_FLAG);
1493 /* Call standard close procedure with error check */
1494 SPI_CloseTransfer(hspi);
1496 /* Process Unlocked */
1497 __HAL_UNLOCK(hspi);
1499 hspi->State = HAL_SPI_STATE_READY;
1501 if (hspi->ErrorCode != HAL_SPI_ERROR_NONE)
1503 return HAL_ERROR;
1505 return errorcode;
1509 * @brief Transmit an amount of data in non-blocking mode with Interrupt.
1510 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
1511 * the configuration information for SPI module.
1512 * @param pData: pointer to data buffer
1513 * @param Size : amount of data to be sent
1514 * @retval HAL status
1516 HAL_StatusTypeDef HAL_SPI_Transmit_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, uint16_t Size)
1518 HAL_StatusTypeDef errorcode = HAL_OK;
1520 /* Check Direction parameter */
1521 assert_param(IS_SPI_DIRECTION_2LINES_OR_1LINE_2LINES_TXONLY(hspi->Init.Direction));
1523 /* Process Locked */
1524 __HAL_LOCK(hspi);
1526 if ((pData == NULL) || (Size == 0UL))
1528 errorcode = HAL_ERROR;
1529 __HAL_UNLOCK(hspi);
1530 return errorcode;
1533 if (hspi->State != HAL_SPI_STATE_READY)
1535 errorcode = HAL_BUSY;
1536 __HAL_UNLOCK(hspi);
1537 return errorcode;
1540 /* Set the transaction information */
1541 hspi->State = HAL_SPI_STATE_BUSY_TX;
1542 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
1543 hspi->pTxBuffPtr = (uint8_t *)pData;
1544 hspi->TxXferSize = Size;
1545 hspi->TxXferCount = Size;
1547 /* Init field not used in handle to zero */
1548 hspi->pRxBuffPtr = NULL;
1549 hspi->RxXferSize = (uint16_t) 0UL;
1550 hspi->RxXferCount = (uint16_t) 0UL;
1551 hspi->RxISR = NULL;
1553 /* Set the function for IT treatment */
1554 if (hspi->Init.DataSize > SPI_DATASIZE_16BIT)
1556 hspi->TxISR = SPI_TxISR_32BIT;
1558 else if (hspi->Init.DataSize > SPI_DATASIZE_8BIT)
1560 hspi->TxISR = SPI_TxISR_16BIT;
1562 else
1564 hspi->TxISR = SPI_TxISR_8BIT;
1567 /* Configure communication direction : 1Line */
1568 if (hspi->Init.Direction == SPI_DIRECTION_1LINE)
1570 SPI_1LINE_TX(hspi);
1573 /* Set the number of data at current transfer */
1574 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, Size);
1576 /* Enable SPI peripheral */
1577 __HAL_SPI_ENABLE(hspi);
1579 /* Enable EOT, TXP, FRE, MODF, UDR and TSERF interrupts */
1580 __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_EOT | SPI_IT_TXP | SPI_IT_UDR | SPI_IT_FRE | SPI_IT_MODF | SPI_IT_TSERF));
1582 if (hspi->Init.Mode == SPI_MODE_MASTER)
1584 /* Master transfer start */
1585 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSTART);
1588 __HAL_UNLOCK(hspi);
1589 return errorcode;
1593 * @brief Receive an amount of data in non-blocking mode with Interrupt.
1594 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
1595 * the configuration information for SPI module.
1596 * @param pData: pointer to data buffer
1597 * @param Size : amount of data to be sent
1598 * @retval HAL status
1600 HAL_StatusTypeDef HAL_SPI_Receive_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, uint16_t Size)
1602 HAL_StatusTypeDef errorcode = HAL_OK;
1604 /* Check Direction parameter */
1605 assert_param(IS_SPI_DIRECTION_2LINES_OR_1LINE_2LINES_RXONLY(hspi->Init.Direction));
1607 if ((hspi->Init.Direction == SPI_DIRECTION_2LINES) && (hspi->Init.Mode == SPI_MODE_MASTER))
1609 hspi->State = HAL_SPI_STATE_BUSY_RX;
1610 /* Call transmit-receive function to send Dummy data on Tx line and generate clock on CLK line */
1611 return HAL_SPI_TransmitReceive_IT(hspi, pData, pData, Size);
1614 /* Process Locked */
1615 __HAL_LOCK(hspi);
1617 if (hspi->State != HAL_SPI_STATE_READY)
1619 errorcode = HAL_BUSY;
1620 __HAL_UNLOCK(hspi);
1621 return errorcode;
1624 if ((pData == NULL) || (Size == 0UL))
1626 errorcode = HAL_ERROR;
1627 __HAL_UNLOCK(hspi);
1628 return errorcode;
1631 /* Set the transaction information */
1632 hspi->State = HAL_SPI_STATE_BUSY_RX;
1633 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
1634 hspi->pRxBuffPtr = (uint8_t *)pData;
1635 hspi->RxXferSize = Size;
1636 hspi->RxXferCount = Size;
1638 /* Init field not used in handle to zero */
1639 hspi->pTxBuffPtr = NULL;
1640 hspi->TxXferSize = (uint16_t) 0UL;
1641 hspi->TxXferCount = (uint16_t) 0UL;
1642 hspi->TxISR = NULL;
1644 /* Set the function for IT treatment */
1645 if (hspi->Init.DataSize > SPI_DATASIZE_16BIT)
1647 hspi->RxISR = SPI_RxISR_32BIT;
1649 else if (hspi->Init.DataSize > SPI_DATASIZE_8BIT)
1651 hspi->RxISR = SPI_RxISR_16BIT;
1653 else
1655 hspi->RxISR = SPI_RxISR_8BIT;
1658 /* Configure communication direction : 1Line */
1659 if (hspi->Init.Direction == SPI_DIRECTION_1LINE)
1661 SPI_1LINE_RX(hspi);
1664 /* Note : The SPI must be enabled after unlocking current process
1665 to avoid the risk of SPI interrupt handle execution before current
1666 process unlock */
1668 /* Set the number of data at current transfer */
1669 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, Size);
1671 /* Enable SPI peripheral */
1672 __HAL_SPI_ENABLE(hspi);
1674 /* Enable EOT, RXP, OVR, FRE, MODF and TSERF interrupts */
1675 __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_EOT | SPI_IT_RXP | SPI_IT_OVR | SPI_IT_FRE | SPI_IT_MODF | SPI_IT_TSERF));
1677 if (hspi->Init.Mode == SPI_MODE_MASTER)
1679 /* Master transfer start */
1680 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSTART);
1683 /* Process Unlocked */
1684 __HAL_UNLOCK(hspi);
1685 return errorcode;
1689 * @brief Transmit and Receive an amount of data in non-blocking mode with Interrupt.
1690 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
1691 * the configuration information for SPI module.
1692 * @param pTxData: pointer to transmission data buffer
1693 * @param pRxData: pointer to reception data buffer
1694 * @param Size : amount of data to be sent and received
1695 * @retval HAL status
1697 HAL_StatusTypeDef HAL_SPI_TransmitReceive_IT(SPI_HandleTypeDef *hspi, uint8_t *pTxData, uint8_t *pRxData, uint16_t Size)
1699 HAL_SPI_StateTypeDef tmp_state;
1700 HAL_StatusTypeDef errorcode = HAL_OK;
1702 uint32_t tmp_mode;
1704 /* Check Direction parameter */
1705 assert_param(IS_SPI_DIRECTION_2LINES(hspi->Init.Direction));
1707 /* Process locked */
1708 __HAL_LOCK(hspi);
1710 /* Init temporary variables */
1711 tmp_state = hspi->State;
1712 tmp_mode = hspi->Init.Mode;
1714 if (!((tmp_state == HAL_SPI_STATE_READY) || \
1715 ((tmp_mode == SPI_MODE_MASTER) && (hspi->Init.Direction == SPI_DIRECTION_2LINES) && (tmp_state == HAL_SPI_STATE_BUSY_RX))))
1717 errorcode = HAL_BUSY;
1718 __HAL_UNLOCK(hspi);
1719 return errorcode;
1722 if ((pTxData == NULL) || (pRxData == NULL) || (Size == 0UL))
1724 errorcode = HAL_ERROR;
1725 __HAL_UNLOCK(hspi);
1726 return errorcode;
1729 /* Don't overwrite in case of HAL_SPI_STATE_BUSY_RX */
1730 if (hspi->State != HAL_SPI_STATE_BUSY_RX)
1732 hspi->State = HAL_SPI_STATE_BUSY_TX_RX;
1735 /* Set the transaction information */
1736 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
1737 hspi->pTxBuffPtr = (uint8_t *)pTxData;
1738 hspi->TxXferSize = Size;
1739 hspi->TxXferCount = Size;
1740 hspi->pRxBuffPtr = (uint8_t *)pRxData;
1741 hspi->RxXferSize = Size;
1742 hspi->RxXferCount = Size;
1744 /* Set the function for IT treatment */
1745 if (hspi->Init.DataSize > SPI_DATASIZE_16BIT)
1747 hspi->TxISR = SPI_TxISR_32BIT;
1748 hspi->RxISR = SPI_RxISR_32BIT;
1750 else if (hspi->Init.DataSize > SPI_DATASIZE_8BIT)
1752 hspi->RxISR = SPI_RxISR_16BIT;
1753 hspi->TxISR = SPI_TxISR_16BIT;
1755 else
1757 hspi->RxISR = SPI_RxISR_8BIT;
1758 hspi->TxISR = SPI_TxISR_8BIT;
1761 /* Set the number of data at current transfer */
1762 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, Size);
1764 /* Enable SPI peripheral */
1765 __HAL_SPI_ENABLE(hspi);
1767 /* Enable EOT, RXP, TXP, DXP, UDR, OVR, FRE, MODF and TSERF interrupts */
1768 __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_EOT | SPI_IT_RXP | SPI_IT_TXP | SPI_IT_DXP | SPI_IT_UDR | SPI_IT_OVR | SPI_IT_FRE | SPI_IT_MODF | SPI_IT_TSERF));
1770 if (hspi->Init.Mode == SPI_MODE_MASTER)
1772 /* Master transfer start */
1773 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSTART);
1776 /* Process Unlocked */
1777 __HAL_UNLOCK(hspi);
1778 return errorcode;
1781 #if defined(USE_SPI_RELOAD_TRANSFER)
1783 * @brief Transmit an additional amount of data in blocking mode.
1784 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
1785 * the configuration information for SPI module.
1786 * @param pData: pointer to data buffer
1787 * @param Size : amount of data to be sent
1788 * @retval HAL status
1790 HAL_StatusTypeDef HAL_SPI_Reload_Transmit_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, uint16_t Size)
1792 HAL_StatusTypeDef errorcode = HAL_OK;
1793 HAL_SPI_StateTypeDef tmp_state;
1795 /* Lock the process */
1796 __HAL_LOCK(hspi);
1798 if ((pData == NULL) || (Size == 0UL))
1800 errorcode = HAL_ERROR;
1801 __HAL_UNLOCK(hspi);
1802 return errorcode;
1805 if (hspi->State == HAL_SPI_STATE_BUSY_TX)
1807 /* check if there is already a request to reload */
1808 if (hspi->Reload.Requested == 1UL)
1810 errorcode = HAL_ERROR;
1811 __HAL_UNLOCK(hspi);
1812 return errorcode;
1815 /* Insert the new number of data to be sent just after the current one */
1816 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSER, (Size & 0xFFFFFFFFUL) << 16UL);
1818 /* Set the transaction information */
1819 hspi->Reload.Requested = 1UL;
1820 hspi->Reload.pTxBuffPtr = (uint8_t *)pData;
1821 hspi->Reload.TxXferSize = Size;
1823 tmp_state = hspi->State;
1825 /* Check if the current transmit is already completed */
1826 if (((hspi->Instance->CR2 & SPI_CR2_TSER) != 0UL) && (tmp_state == HAL_SPI_STATE_READY))
1828 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_TSERF);
1829 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSER, 0UL);
1830 hspi->Reload.Requested = 0UL;
1831 errorcode = HAL_ERROR;
1832 __HAL_UNLOCK(hspi);
1833 return errorcode;
1836 else
1838 errorcode = HAL_ERROR;
1839 return errorcode;
1842 __HAL_UNLOCK(hspi);
1843 return errorcode;
1845 #endif /* USE_HSPI_RELOAD_TRANSFER */
1847 #if defined(USE_SPI_RELOAD_TRANSFER)
1849 * @brief Receive an additional amount of data in blocking mode.
1850 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
1851 * the configuration information for SPI module.
1852 * @param pData: pointer to data buffer
1853 * @param Size : amount of data to be sent
1854 * @retval HAL status
1856 HAL_StatusTypeDef HAL_SPI_Reload_Receive_IT(SPI_HandleTypeDef *hspi, uint8_t *pData, uint16_t Size)
1858 HAL_StatusTypeDef errorcode = HAL_OK;
1859 HAL_SPI_StateTypeDef tmp_state;
1861 /* Lock the process */
1862 __HAL_LOCK(hspi);
1864 if ((pData == NULL) || (Size == 0UL))
1866 errorcode = HAL_ERROR;
1867 __HAL_UNLOCK(hspi);
1868 return errorcode;
1871 if (hspi->State == HAL_SPI_STATE_BUSY_RX)
1873 /* check if there is already a request to reload */
1874 if (hspi->Reload.Requested == 1UL)
1876 errorcode = HAL_ERROR;
1877 __HAL_UNLOCK(hspi);
1878 return errorcode;
1881 /* Insert the new number of data that will be received just after the current one */
1882 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSER, (Size & 0xFFFFFFFFUL) << 16UL);
1884 /* Set the transaction information */
1885 hspi->Reload.Requested = 1UL;
1886 hspi->Reload.pRxBuffPtr = (uint8_t *)pData;
1887 hspi->Reload.RxXferSize = Size;
1889 tmp_state = hspi->State;
1891 /* Check if the current reception is already completed */
1892 if (((hspi->Instance->CR2 & SPI_CR2_TSER) != 0UL) && (tmp_state == HAL_SPI_STATE_READY))
1894 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_TSERF);
1895 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSER, 0UL);
1896 hspi->Reload.Requested = 0UL;
1897 errorcode = HAL_ERROR;
1898 __HAL_UNLOCK(hspi);
1899 return errorcode;
1902 else
1904 errorcode = HAL_ERROR;
1905 return errorcode;
1908 __HAL_UNLOCK(hspi);
1909 return errorcode;
1911 #endif /* USE_HSPI_RELOAD_TRANSFER */
1913 #if defined(USE_SPI_RELOAD_TRANSFER)
1915 * @brief Transmit and receive an additional amount of data in blocking mode.
1916 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
1917 * the configuration information for SPI module.
1918 * @param pTxData: pointer to transmission data buffer
1919 * @param pRxData: pointer to reception data buffer
1920 * @param Size : amount of data to be sent and received
1921 * @retval HAL status
1923 HAL_StatusTypeDef HAL_SPI_Reload_TransmitReceive_IT(SPI_HandleTypeDef *hspi, uint8_t *pTxData, uint8_t *pRxData, uint16_t Size)
1925 HAL_StatusTypeDef errorcode = HAL_OK;
1926 HAL_SPI_StateTypeDef tmp_state;
1928 /* Lock the process */
1929 __HAL_LOCK(hspi);
1931 if ((pTxData == NULL) || (pRxData == NULL) || (Size == 0UL))
1933 errorcode = HAL_ERROR;
1934 __HAL_UNLOCK(hspi);
1935 return errorcode;
1938 if (hspi->State == HAL_SPI_STATE_BUSY_TX_RX)
1940 /* check if there is already a request to reload */
1941 if (hspi->Reload.Requested == 1UL)
1943 errorcode = HAL_ERROR;
1944 __HAL_UNLOCK(hspi);
1945 return errorcode;
1948 /* Insert the new number of data that will be sent and received just after the current one */
1949 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSER, (Size & 0xFFFFFFFFUL) << 16UL);
1951 /* Set the transaction information */
1952 hspi->Reload.Requested = 1UL;
1953 hspi->Reload.pTxBuffPtr = (uint8_t *)pTxData;
1954 hspi->Reload.TxXferSize = Size;
1955 hspi->Reload.pRxBuffPtr = (uint8_t *)pRxData;
1956 hspi->Reload.RxXferSize = Size;
1958 tmp_state = hspi->State;
1960 /* Check if the current transmit is already completed */
1961 if (((hspi->Instance->CR2 & SPI_CR2_TSER) != 0UL) && (tmp_state == HAL_SPI_STATE_READY))
1963 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_TSERF);
1964 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSER, 0UL);
1965 hspi->Reload.Requested = 0UL;
1966 errorcode = HAL_ERROR;
1967 __HAL_UNLOCK(hspi);
1968 return errorcode;
1971 else
1973 errorcode = HAL_ERROR;
1974 return errorcode;
1977 __HAL_UNLOCK(hspi);
1978 return errorcode;
1980 #endif /* USE_HSPI_RELOAD_TRANSFER */
1983 * @brief Transmit an amount of data in non-blocking mode with DMA.
1984 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
1985 * the configuration information for SPI module.
1986 * @param pData: pointer to data buffer
1987 * @param Size : amount of data to be sent
1988 * @retval HAL status
1990 HAL_StatusTypeDef HAL_SPI_Transmit_DMA(SPI_HandleTypeDef *hspi, uint8_t *pData, uint16_t Size)
1992 HAL_StatusTypeDef errorcode = HAL_OK;
1994 /* Check Direction parameter */
1995 assert_param(IS_SPI_DIRECTION_2LINES_OR_1LINE_2LINES_TXONLY(hspi->Init.Direction));
1997 /* Process Locked */
1998 __HAL_LOCK(hspi);
2000 if (hspi->State != HAL_SPI_STATE_READY)
2002 errorcode = HAL_BUSY;
2003 __HAL_UNLOCK(hspi);
2004 return errorcode;
2007 if ((pData == NULL) || (Size == 0UL))
2009 errorcode = HAL_ERROR;
2010 __HAL_UNLOCK(hspi);
2011 return errorcode;
2014 /* Set the transaction information */
2015 hspi->State = HAL_SPI_STATE_BUSY_TX;
2016 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
2017 hspi->pTxBuffPtr = (uint8_t *)pData;
2018 hspi->TxXferSize = Size;
2019 hspi->TxXferCount = Size;
2021 /* Init field not used in handle to zero */
2022 hspi->pRxBuffPtr = NULL;
2023 hspi->TxISR = NULL;
2024 hspi->RxISR = NULL;
2025 hspi->RxXferSize = (uint16_t)0UL;
2026 hspi->RxXferCount = (uint16_t)0UL;
2028 /* Configure communication direction : 1Line */
2029 if (hspi->Init.Direction == SPI_DIRECTION_1LINE)
2031 SPI_1LINE_TX(hspi);
2034 /* Packing mode management is enabled by the DMA settings */
2035 if (((hspi->Init.DataSize > SPI_DATASIZE_16BIT) && (hspi->hdmatx->Init.MemDataAlignment != DMA_MDATAALIGN_WORD)) || \
2036 ((hspi->Init.DataSize > SPI_DATASIZE_8BIT) && ((hspi->hdmatx->Init.MemDataAlignment != DMA_MDATAALIGN_HALFWORD) && \
2037 (hspi->hdmatx->Init.MemDataAlignment != DMA_MDATAALIGN_WORD))))
2039 /* Restriction the DMA data received is not allowed in this mode */
2040 errorcode = HAL_ERROR;
2041 __HAL_UNLOCK(hspi);
2042 return errorcode;
2045 /* Adjust XferCount according to DMA alignment / Data size */
2046 if (hspi->Init.DataSize <= SPI_DATASIZE_8BIT)
2048 if (hspi->hdmatx->Init.MemDataAlignment == DMA_MDATAALIGN_HALFWORD)
2050 hspi->TxXferCount = (hspi->TxXferCount + (uint16_t) 1UL) >> 1UL;
2052 if (hspi->hdmatx->Init.MemDataAlignment == DMA_MDATAALIGN_WORD)
2054 hspi->TxXferCount = (hspi->TxXferCount + (uint16_t) 3UL) >> 2UL;
2057 else if (hspi->Init.DataSize <= SPI_DATASIZE_16BIT)
2059 if (hspi->hdmatx->Init.MemDataAlignment == DMA_MDATAALIGN_WORD)
2061 hspi->TxXferCount = (hspi->TxXferCount + (uint16_t) 1UL) >> 1UL;
2064 else
2066 /* Adjustment done */
2069 /* Set the SPI TxDMA Half transfer complete callback */
2070 hspi->hdmatx->XferHalfCpltCallback = SPI_DMAHalfTransmitCplt;
2072 /* Set the SPI TxDMA transfer complete callback */
2073 hspi->hdmatx->XferCpltCallback = SPI_DMATransmitCplt;
2075 /* Set the DMA error callback */
2076 hspi->hdmatx->XferErrorCallback = SPI_DMAError;
2078 /* Set the DMA AbortCpltCallback */
2079 hspi->hdmatx->XferAbortCallback = NULL;
2081 /* Clear TXDMAEN bit*/
2082 CLEAR_BIT(hspi->Instance->CFG1, SPI_CFG1_TXDMAEN);
2084 /* Enable the Tx DMA Stream/Channel */
2085 if (HAL_OK != HAL_DMA_Start_IT(hspi->hdmatx, (uint32_t)hspi->pTxBuffPtr, (uint32_t)&hspi->Instance->TXDR, hspi->TxXferCount))
2087 /* Update SPI error code */
2088 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_DMA);
2089 errorcode = HAL_ERROR;
2090 hspi->State = HAL_SPI_STATE_READY;
2091 return errorcode;
2094 /* Set the number of data at current transfer */
2095 if (hspi->hdmatx->Init.Mode == DMA_CIRCULAR)
2097 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, 0UL);
2099 else
2101 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, Size);
2104 /* Enable Tx DMA Request */
2105 SET_BIT(hspi->Instance->CFG1, SPI_CFG1_TXDMAEN);
2107 /* Enable the SPI Error Interrupt Bit */
2108 __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_UDR | SPI_IT_FRE | SPI_IT_MODF));
2110 /* Enable SPI peripheral */
2111 __HAL_SPI_ENABLE(hspi);
2113 if (hspi->Init.Mode == SPI_MODE_MASTER)
2115 /* Master transfer start */
2116 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSTART);
2119 /* Process Unlocked */
2120 __HAL_UNLOCK(hspi);
2121 return errorcode;
2125 * @brief Receive an amount of data in non-blocking mode with DMA.
2126 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
2127 * the configuration information for SPI module.
2128 * @param pData: pointer to data buffer
2129 * @param Size : amount of data to be sent
2130 * @note When the CRC feature is enabled the pData Length must be Size + 1.
2131 * @retval HAL status
2133 HAL_StatusTypeDef HAL_SPI_Receive_DMA(SPI_HandleTypeDef *hspi, uint8_t *pData, uint16_t Size)
2135 HAL_StatusTypeDef errorcode = HAL_OK;
2137 /* Check Direction parameter */
2138 assert_param(IS_SPI_DIRECTION_2LINES_OR_1LINE_2LINES_RXONLY(hspi->Init.Direction));
2140 if ((hspi->Init.Direction == SPI_DIRECTION_2LINES) && (hspi->Init.Mode == SPI_MODE_MASTER))
2142 hspi->State = HAL_SPI_STATE_BUSY_RX;
2143 /* Call transmit-receive function to send Dummy data on Tx line and generate clock on CLK line */
2144 return HAL_SPI_TransmitReceive_DMA(hspi, pData, pData, Size);
2147 /* Process Locked */
2148 __HAL_LOCK(hspi);
2150 if (hspi->State != HAL_SPI_STATE_READY)
2152 errorcode = HAL_BUSY;
2153 __HAL_UNLOCK(hspi);
2154 return errorcode;
2157 if ((pData == NULL) || (Size == 0UL))
2159 errorcode = HAL_ERROR;
2160 __HAL_UNLOCK(hspi);
2161 return errorcode;
2164 /* Set the transaction information */
2165 hspi->State = HAL_SPI_STATE_BUSY_RX;
2166 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
2167 hspi->pRxBuffPtr = (uint8_t *)pData;
2168 hspi->RxXferSize = Size;
2169 hspi->RxXferCount = Size;
2171 /*Init field not used in handle to zero */
2172 hspi->RxISR = NULL;
2173 hspi->TxISR = NULL;
2174 hspi->TxXferSize = (uint16_t) 0UL;
2175 hspi->TxXferCount = (uint16_t) 0UL;
2177 /* Configure communication direction : 1Line */
2178 if (hspi->Init.Direction == SPI_DIRECTION_1LINE)
2180 SPI_1LINE_RX(hspi);
2183 /* Packing mode management is enabled by the DMA settings */
2184 if (((hspi->Init.DataSize > SPI_DATASIZE_16BIT) && (hspi->hdmarx->Init.MemDataAlignment != DMA_MDATAALIGN_WORD)) || \
2185 ((hspi->Init.DataSize > SPI_DATASIZE_8BIT) && ((hspi->hdmarx->Init.MemDataAlignment != DMA_MDATAALIGN_HALFWORD) && \
2186 (hspi->hdmarx->Init.MemDataAlignment != DMA_MDATAALIGN_WORD))))
2188 /* Restriction the DMA data received is not allowed in this mode */
2189 errorcode = HAL_ERROR;
2190 __HAL_UNLOCK(hspi);
2191 return errorcode;
2194 /* Clear RXDMAEN bit */
2195 CLEAR_BIT(hspi->Instance->CFG1, SPI_CFG1_RXDMAEN);
2197 /* Adjust XferCount according to DMA alignment / Data size */
2198 if (hspi->Init.DataSize <= SPI_DATASIZE_8BIT)
2200 if (hspi->hdmarx->Init.MemDataAlignment == DMA_MDATAALIGN_HALFWORD)
2202 hspi->RxXferCount = (hspi->RxXferCount + (uint16_t) 1UL) >> 1UL;
2204 if (hspi->hdmarx->Init.MemDataAlignment == DMA_MDATAALIGN_WORD)
2206 hspi->RxXferCount = (hspi->RxXferCount + (uint16_t) 3UL) >> 2UL;
2209 else if (hspi->Init.DataSize <= SPI_DATASIZE_16BIT)
2211 if (hspi->hdmarx->Init.MemDataAlignment == DMA_MDATAALIGN_WORD)
2213 hspi->RxXferCount = (hspi->RxXferCount + (uint16_t) 1UL) >> 1UL;
2216 else
2218 /* Adjustment done */
2221 /* Set the SPI RxDMA Half transfer complete callback */
2222 hspi->hdmarx->XferHalfCpltCallback = SPI_DMAHalfReceiveCplt;
2224 /* Set the SPI Rx DMA transfer complete callback */
2225 hspi->hdmarx->XferCpltCallback = SPI_DMAReceiveCplt;
2227 /* Set the DMA error callback */
2228 hspi->hdmarx->XferErrorCallback = SPI_DMAError;
2230 /* Set the DMA AbortCpltCallback */
2231 hspi->hdmarx->XferAbortCallback = NULL;
2233 /* Enable the Rx DMA Stream/Channel */
2234 if (HAL_OK != HAL_DMA_Start_IT(hspi->hdmarx, (uint32_t)&hspi->Instance->RXDR, (uint32_t)hspi->pRxBuffPtr, hspi->RxXferCount))
2236 /* Update SPI error code */
2237 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_DMA);
2238 errorcode = HAL_ERROR;
2239 hspi->State = HAL_SPI_STATE_READY;
2240 return errorcode;
2243 /* Set the number of data at current transfer */
2244 if (hspi->hdmarx->Init.Mode == DMA_CIRCULAR)
2246 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, 0UL);
2248 else
2250 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, Size);
2253 /* Enable Rx DMA Request */
2254 SET_BIT(hspi->Instance->CFG1, SPI_CFG1_RXDMAEN);
2256 /* Enable the SPI Error Interrupt Bit */
2257 __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_OVR | SPI_IT_FRE | SPI_IT_MODF));
2259 /* Enable SPI peripheral */
2260 __HAL_SPI_ENABLE(hspi);
2262 if (hspi->Init.Mode == SPI_MODE_MASTER)
2264 /* Master transfer start */
2265 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSTART);
2268 /* Process Unlocked */
2269 __HAL_UNLOCK(hspi);
2270 return errorcode;
2274 * @brief Transmit and Receive an amount of data in non-blocking mode with DMA.
2275 * @param hspi : pointer to a SPI_HandleTypeDef structure that contains
2276 * the configuration information for SPI module.
2277 * @param pTxData: pointer to transmission data buffer
2278 * @param pRxData: pointer to reception data buffer
2279 * @param Size : amount of data to be sent
2280 * @note When the CRC feature is enabled the pRxData Length must be Size + 1
2281 * @retval HAL status
2283 HAL_StatusTypeDef HAL_SPI_TransmitReceive_DMA(SPI_HandleTypeDef *hspi, uint8_t *pTxData, uint8_t *pRxData,
2284 uint16_t Size)
2286 HAL_SPI_StateTypeDef tmp_state;
2287 HAL_StatusTypeDef errorcode = HAL_OK;
2289 uint32_t tmp_mode;
2291 /* Check Direction parameter */
2292 assert_param(IS_SPI_DIRECTION_2LINES(hspi->Init.Direction));
2294 /* Process locked */
2295 __HAL_LOCK(hspi);
2297 /* Init temporary variables */
2298 tmp_state = hspi->State;
2299 tmp_mode = hspi->Init.Mode;
2301 if (!(((tmp_mode == SPI_MODE_MASTER) && (hspi->Init.Direction == SPI_DIRECTION_2LINES) && (tmp_state == HAL_SPI_STATE_BUSY_RX)) || (tmp_state == HAL_SPI_STATE_READY)))
2303 errorcode = HAL_BUSY;
2304 __HAL_UNLOCK(hspi);
2305 return errorcode;
2308 if ((pTxData == NULL) || (pRxData == NULL) || (Size == 0UL))
2310 errorcode = HAL_ERROR;
2311 __HAL_UNLOCK(hspi);
2312 return errorcode;
2315 /* Don't overwrite in case of HAL_SPI_STATE_BUSY_RX */
2316 if (hspi->State != HAL_SPI_STATE_BUSY_RX)
2318 hspi->State = HAL_SPI_STATE_BUSY_TX_RX;
2321 /* Set the transaction information */
2322 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
2323 hspi->pTxBuffPtr = (uint8_t *)pTxData;
2324 hspi->TxXferSize = Size;
2325 hspi->TxXferCount = Size;
2326 hspi->pRxBuffPtr = (uint8_t *)pRxData;
2327 hspi->RxXferSize = Size;
2328 hspi->RxXferCount = Size;
2330 /* Init field not used in handle to zero */
2331 hspi->RxISR = NULL;
2332 hspi->TxISR = NULL;
2334 /* Reset the Tx/Rx DMA bits */
2335 CLEAR_BIT(hspi->Instance->CFG1, SPI_CFG1_TXDMAEN | SPI_CFG1_RXDMAEN);
2337 /* Packing mode management is enabled by the DMA settings */
2338 if (((hspi->Init.DataSize > SPI_DATASIZE_16BIT) && (hspi->hdmarx->Init.MemDataAlignment != DMA_MDATAALIGN_WORD)) || \
2339 ((hspi->Init.DataSize > SPI_DATASIZE_8BIT) && ((hspi->hdmarx->Init.MemDataAlignment != DMA_MDATAALIGN_HALFWORD) && \
2340 (hspi->hdmarx->Init.MemDataAlignment != DMA_MDATAALIGN_WORD))))
2342 /* Restriction the DMA data received is not allowed in this mode */
2343 errorcode = HAL_ERROR;
2344 /* Process Unlocked */
2345 __HAL_UNLOCK(hspi);
2346 return errorcode;
2349 /* Adjust XferCount according to DMA alignment / Data size */
2350 if (hspi->Init.DataSize <= SPI_DATASIZE_8BIT)
2352 if (hspi->hdmatx->Init.MemDataAlignment == DMA_MDATAALIGN_HALFWORD)
2354 hspi->TxXferCount = (hspi->TxXferCount + (uint16_t) 1UL) >> 1UL;
2356 if (hspi->hdmatx->Init.MemDataAlignment == DMA_MDATAALIGN_WORD)
2358 hspi->TxXferCount = (hspi->TxXferCount + (uint16_t) 3UL) >> 2UL;
2360 if (hspi->hdmarx->Init.MemDataAlignment == DMA_MDATAALIGN_HALFWORD)
2362 hspi->RxXferCount = (hspi->RxXferCount + (uint16_t) 1UL) >> 1UL;
2364 if (hspi->hdmarx->Init.MemDataAlignment == DMA_MDATAALIGN_WORD)
2366 hspi->RxXferCount = (hspi->RxXferCount + (uint16_t) 3UL) >> 2UL;
2369 else if (hspi->Init.DataSize <= SPI_DATASIZE_16BIT)
2371 if (hspi->hdmatx->Init.MemDataAlignment == DMA_MDATAALIGN_WORD)
2373 hspi->TxXferCount = (hspi->TxXferCount + (uint16_t) 1UL) >> 1UL;
2375 if (hspi->hdmarx->Init.MemDataAlignment == DMA_MDATAALIGN_WORD)
2377 hspi->RxXferCount = (hspi->RxXferCount + (uint16_t) 1UL) >> 1UL;
2380 else
2382 /* Adjustment done */
2385 /* Check if we are in Rx only or in Rx/Tx Mode and configure the DMA transfer complete callback */
2386 if (hspi->State == HAL_SPI_STATE_BUSY_RX)
2388 /* Set the SPI Rx DMA Half transfer complete callback */
2389 hspi->hdmarx->XferHalfCpltCallback = SPI_DMAHalfReceiveCplt;
2390 hspi->hdmarx->XferCpltCallback = SPI_DMAReceiveCplt;
2392 else
2394 /* Set the SPI Tx/Rx DMA Half transfer complete callback */
2395 hspi->hdmarx->XferHalfCpltCallback = SPI_DMAHalfTransmitReceiveCplt;
2396 hspi->hdmarx->XferCpltCallback = SPI_DMATransmitReceiveCplt;
2399 /* Set the DMA error callback */
2400 hspi->hdmarx->XferErrorCallback = SPI_DMAError;
2402 /* Set the DMA AbortCallback */
2403 hspi->hdmarx->XferAbortCallback = NULL;
2405 /* Enable the Rx DMA Stream/Channel */
2406 if (HAL_OK != HAL_DMA_Start_IT(hspi->hdmarx, (uint32_t)&hspi->Instance->RXDR, (uint32_t)hspi->pRxBuffPtr, hspi->RxXferCount))
2408 /* Update SPI error code */
2409 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_DMA);
2410 errorcode = HAL_ERROR;
2411 hspi->State = HAL_SPI_STATE_READY;
2412 return errorcode;
2415 /* Enable Rx DMA Request */
2416 SET_BIT(hspi->Instance->CFG1, SPI_CFG1_RXDMAEN);
2418 /* Set the SPI Tx DMA transfer complete callback as NULL because the communication closing
2419 is performed in DMA reception complete callback */
2420 hspi->hdmatx->XferHalfCpltCallback = NULL;
2421 hspi->hdmatx->XferCpltCallback = NULL;
2422 hspi->hdmatx->XferErrorCallback = NULL;
2423 hspi->hdmatx->XferAbortCallback = NULL;
2425 /* Enable the Tx DMA Stream/Channel */
2426 if (HAL_OK != HAL_DMA_Start_IT(hspi->hdmatx, (uint32_t)hspi->pTxBuffPtr, (uint32_t)&hspi->Instance->TXDR, hspi->TxXferCount))
2428 /* Update SPI error code */
2429 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_DMA);
2430 errorcode = HAL_ERROR;
2431 hspi->State = HAL_SPI_STATE_READY;
2432 return errorcode;
2435 if (hspi->hdmatx->Init.Mode == DMA_CIRCULAR)
2437 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, 0UL);
2439 else
2441 MODIFY_REG(hspi->Instance->CR2, SPI_CR2_TSIZE, Size);
2444 /* Enable Tx DMA Request */
2445 SET_BIT(hspi->Instance->CFG1, SPI_CFG1_TXDMAEN);
2447 /* Enable the SPI Error Interrupt Bit */
2448 __HAL_SPI_ENABLE_IT(hspi, (SPI_IT_OVR | SPI_IT_UDR | SPI_IT_FRE | SPI_IT_MODF));
2450 /* Enable SPI peripheral */
2451 __HAL_SPI_ENABLE(hspi);
2453 if (hspi->Init.Mode == SPI_MODE_MASTER)
2455 /* Master transfer start */
2456 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSTART);
2459 /* Process Unlocked */
2460 __HAL_UNLOCK(hspi);
2461 return errorcode;
2465 * @brief Abort ongoing transfer (blocking mode).
2466 * @param hspi SPI handle.
2467 * @note This procedure could be used for aborting any ongoing transfer (Tx and Rx),
2468 * started in Interrupt or DMA mode.
2469 * @note This procedure performs following operations :
2470 * + Disable SPI Interrupts (depending of transfer direction)
2471 * + Disable the DMA transfer in the peripheral register (if enabled)
2472 * + Abort DMA transfer by calling HAL_DMA_Abort (in case of transfer in DMA mode)
2473 * + Set handle State to READY.
2474 * @note This procedure is executed in blocking mode : when exiting function, Abort is considered as completed.
2475 * @retval HAL status
2477 HAL_StatusTypeDef HAL_SPI_Abort(SPI_HandleTypeDef *hspi)
2479 HAL_StatusTypeDef errorcode;
2481 __IO uint32_t count;
2483 /* Process locked */
2484 __HAL_LOCK(hspi);
2486 /* Set hspi->state to aborting to avoid any interaction */
2487 hspi->State = HAL_SPI_STATE_ABORT;
2489 /* Initialized local variable */
2490 errorcode = HAL_OK;
2491 count = SPI_DEFAULT_TIMEOUT * (SystemCoreClock / 24UL / 1000UL);
2493 /* If master communication on going, make sure current frame is done before closing the connection */
2494 if (HAL_IS_BIT_SET(hspi->Instance->CR1, SPI_CR1_CSTART))
2496 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSUSP);
2499 count--;
2500 if (count == 0UL)
2502 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_ABORT);
2503 break;
2506 while (HAL_IS_BIT_SET(hspi->Instance->CR1, SPI_CR1_CSTART));
2509 /* Disable the SPI DMA Tx request if enabled */
2510 if (HAL_IS_BIT_SET(hspi->Instance->CFG1, SPI_CFG1_TXDMAEN))
2512 if (hspi->hdmatx != NULL)
2514 /* Abort the SPI DMA Tx Stream/Channel : use blocking DMA Abort API (no callback) */
2515 hspi->hdmatx->XferAbortCallback = NULL;
2517 /* Abort DMA Tx Handle linked to SPI Peripheral */
2518 if (HAL_DMA_Abort(hspi->hdmatx) != HAL_OK)
2520 if (HAL_DMA_GetError(hspi->hdmatx) == HAL_DMA_ERROR_TIMEOUT)
2522 hspi->ErrorCode = HAL_SPI_ERROR_ABORT;
2528 /* Disable the SPI DMA Rx request if enabled */
2529 if (HAL_IS_BIT_SET(hspi->Instance->CFG1, SPI_CFG1_RXDMAEN))
2531 if (hspi->hdmarx != NULL)
2533 /* Abort the SPI DMA Rx Stream/Channel : use blocking DMA Abort API (no callback) */
2534 hspi->hdmarx->XferAbortCallback = NULL;
2536 /* Abort DMA Rx Handle linked to SPI Peripheral */
2537 if (HAL_DMA_Abort(hspi->hdmarx) != HAL_OK)
2539 if (HAL_DMA_GetError(hspi->hdmarx) == HAL_DMA_ERROR_TIMEOUT)
2541 hspi->ErrorCode = HAL_SPI_ERROR_ABORT;
2547 /* Proceed with abort procedure */
2548 SPI_AbortTransfer(hspi);
2550 /* Check error during Abort procedure */
2551 if (hspi->ErrorCode == HAL_SPI_ERROR_ABORT)
2553 /* return HAL_Error in case of error during Abort procedure */
2554 errorcode = HAL_ERROR;
2556 else
2558 /* Reset errorCode */
2559 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
2562 /* Process Unlocked */
2563 __HAL_UNLOCK(hspi);
2565 /* Restore hspi->state to ready */
2566 hspi->State = HAL_SPI_STATE_READY;
2568 return errorcode;
2572 * @brief Abort ongoing transfer (Interrupt mode).
2573 * @param hspi SPI handle.
2574 * @note This procedure could be used for aborting any ongoing transfer (Tx and Rx),
2575 * started in Interrupt or DMA mode.
2576 * @note This procedure performs following operations :
2577 * + Disable SPI Interrupts (depending of transfer direction)
2578 * + Disable the DMA transfer in the peripheral register (if enabled)
2579 * + Abort DMA transfer by calling HAL_DMA_Abort_IT (in case of transfer in DMA mode)
2580 * + Set handle State to READY
2581 * + At abort completion, call user abort complete callback.
2582 * @note This procedure is executed in Interrupt mode, meaning that abort procedure could be
2583 * considered as completed only when user abort complete callback is executed (not when exiting function).
2584 * @retval HAL status
2586 HAL_StatusTypeDef HAL_SPI_Abort_IT(SPI_HandleTypeDef *hspi)
2588 HAL_StatusTypeDef errorcode;
2589 __IO uint32_t count;
2590 uint32_t dma_tx_abort_done = 1UL, dma_rx_abort_done = 1UL;
2592 /* Set hspi->state to aborting to avoid any interaction */
2593 hspi->State = HAL_SPI_STATE_ABORT;
2595 /* Initialized local variable */
2596 errorcode = HAL_OK;
2597 count = SPI_DEFAULT_TIMEOUT * (SystemCoreClock / 24UL / 1000UL);
2599 /* If master communication on going, make sure current frame is done before closing the connection */
2600 if (HAL_IS_BIT_SET(hspi->Instance->CR1, SPI_CR1_CSTART))
2602 SET_BIT(hspi->Instance->CR1, SPI_CR1_CSUSP);
2605 count--;
2606 if (count == 0UL)
2608 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_ABORT);
2609 break;
2612 while (HAL_IS_BIT_SET(hspi->Instance->CR1, SPI_CR1_CSTART));
2615 /* If DMA Tx and/or DMA Rx Handles are associated to SPI Handle, DMA Abort complete callbacks should be initialized
2616 before any call to DMA Abort functions */
2618 if(hspi->hdmatx != NULL)
2620 if (HAL_IS_BIT_SET(hspi->Instance->CFG1, SPI_CFG1_TXDMAEN))
2622 /* Set DMA Abort Complete callback if SPI DMA Tx request if enabled */
2623 hspi->hdmatx->XferAbortCallback = SPI_DMATxAbortCallback;
2625 dma_tx_abort_done = 0UL;
2627 /* Abort DMA Tx Handle linked to SPI Peripheral */
2628 if (HAL_DMA_Abort_IT(hspi->hdmatx) != HAL_OK)
2630 if (HAL_DMA_GetError(hspi->hdmatx) == HAL_DMA_ERROR_NO_XFER)
2632 dma_tx_abort_done = 1UL;
2633 hspi->hdmatx->XferAbortCallback = NULL;
2637 else
2639 hspi->hdmatx->XferAbortCallback = NULL;
2643 if(hspi->hdmarx != NULL)
2645 if (HAL_IS_BIT_SET(hspi->Instance->CFG1, SPI_CFG1_RXDMAEN))
2647 /* Set DMA Abort Complete callback if SPI DMA Rx request if enabled */
2648 hspi->hdmarx->XferAbortCallback = SPI_DMARxAbortCallback;
2650 dma_rx_abort_done = 0UL;
2652 /* Abort DMA Rx Handle linked to SPI Peripheral */
2653 if (HAL_DMA_Abort_IT(hspi->hdmarx) != HAL_OK)
2655 if (HAL_DMA_GetError(hspi->hdmarx) == HAL_DMA_ERROR_NO_XFER)
2657 dma_rx_abort_done = 1UL;
2658 hspi->hdmarx->XferAbortCallback = NULL;
2662 else
2664 hspi->hdmarx->XferAbortCallback = NULL;
2668 /* If no running DMA transfer, finish cleanup and call callbacks */
2669 if ((dma_tx_abort_done == 1UL) && (dma_rx_abort_done == 1UL))
2671 /* Proceed with abort procedure */
2672 SPI_AbortTransfer(hspi);
2674 /* Check error during Abort procedure */
2675 if (hspi->ErrorCode == HAL_SPI_ERROR_ABORT)
2677 /* return HAL_Error in case of error during Abort procedure */
2678 errorcode = HAL_ERROR;
2680 else
2682 /* Reset errorCode */
2683 hspi->ErrorCode = HAL_SPI_ERROR_NONE;
2686 /* Restore hspi->state to ready */
2687 hspi->State = HAL_SPI_STATE_READY;
2689 /* Call user Abort complete callback */
2690 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
2691 hspi->AbortCpltCallback(hspi);
2692 #else
2693 HAL_SPI_AbortCpltCallback(hspi);
2694 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
2697 return errorcode;
2701 * @brief Pause the DMA Transfer.
2702 * This API is not supported, it is maintained for backward compatibility.
2703 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
2704 * the configuration information for the specified SPI module.
2705 * @retval HAL_ERROR
2707 HAL_StatusTypeDef HAL_SPI_DMAPause(SPI_HandleTypeDef *hspi)
2709 /* Set error code to not supported */
2710 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_NOT_SUPPORTED);
2712 return HAL_ERROR;
2716 * @brief Resume the DMA Transfer.
2717 * This API is not supported, it is maintained for backward compatibility.
2718 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
2719 * the configuration information for the specified SPI module.
2720 * @retval HAL_ERROR
2722 HAL_StatusTypeDef HAL_SPI_DMAResume(SPI_HandleTypeDef *hspi)
2724 /* Set error code to not supported */
2725 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_NOT_SUPPORTED);
2727 return HAL_ERROR;
2731 * @brief Stop the DMA Transfer.
2732 * This API is not supported, it is maintained for backward compatibility.
2733 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
2734 * the configuration information for the specified SPI module.
2735 * @retval HAL_ERROR
2737 HAL_StatusTypeDef HAL_SPI_DMAStop(SPI_HandleTypeDef *hspi)
2739 /* Set error code to not supported */
2740 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_NOT_SUPPORTED);
2742 return HAL_ERROR;
2746 * @brief Handle SPI interrupt request.
2747 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
2748 * the configuration information for the specified SPI module.
2749 * @retval None
2751 void HAL_SPI_IRQHandler(SPI_HandleTypeDef *hspi)
2753 uint32_t itsource = hspi->Instance->IER;
2754 uint32_t itflag = hspi->Instance->SR;
2755 uint32_t trigger = itsource & itflag;
2756 uint32_t cfg1 = hspi->Instance->CFG1;
2757 uint32_t handled = 0UL;
2759 HAL_SPI_StateTypeDef State = hspi->State;
2760 #if defined (__GNUC__)
2761 __IO uint16_t *prxdr_16bits = (__IO uint16_t *)(&(hspi->Instance->RXDR));
2762 #endif /* __GNUC__ */
2765 /* SPI in mode Transmitter and Receiver ------------------------------------*/
2766 if (HAL_IS_BIT_CLR(trigger, SPI_FLAG_OVR) && HAL_IS_BIT_CLR(trigger, SPI_FLAG_UDR) && HAL_IS_BIT_SET(trigger, SPI_FLAG_DXP))
2768 hspi->TxISR(hspi);
2769 hspi->RxISR(hspi);
2770 handled = 1UL;
2773 /* SPI in mode Receiver ----------------------------------------------------*/
2774 if (HAL_IS_BIT_CLR(trigger, SPI_FLAG_OVR) && HAL_IS_BIT_SET(trigger, SPI_FLAG_RXP) && HAL_IS_BIT_CLR(trigger, SPI_FLAG_DXP))
2776 hspi->RxISR(hspi);
2777 handled = 1UL;
2780 /* SPI in mode Transmitter -------------------------------------------------*/
2781 if (HAL_IS_BIT_CLR(trigger, SPI_FLAG_UDR) && HAL_IS_BIT_SET(trigger, SPI_FLAG_TXP) && HAL_IS_BIT_CLR(trigger, SPI_FLAG_DXP))
2783 hspi->TxISR(hspi);
2784 handled = 1UL;
2787 #if defined(USE_SPI_RELOAD_TRANSFER)
2788 /* SPI Reload -------------------------------------------------*/
2789 if (HAL_IS_BIT_SET(trigger, SPI_FLAG_TSERF))
2791 hspi->Reload.Requested = 0UL;
2792 __HAL_SPI_CLEAR_TSERFFLAG(hspi);
2794 #endif /* USE_HSPI_RELOAD_TRANSFER */
2796 if (handled != 0UL)
2798 return;
2801 /* SPI End Of Transfer: DMA or IT based transfer */
2802 if (HAL_IS_BIT_SET(trigger, SPI_FLAG_EOT))
2804 /* Clear EOT/TXTF/SUSP flag */
2805 __HAL_SPI_CLEAR_EOTFLAG(hspi);
2806 __HAL_SPI_CLEAR_TXTFFLAG(hspi);
2807 __HAL_SPI_CLEAR_SUSPFLAG(hspi);
2809 /* Disable EOT interrupt */
2810 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_EOT);
2812 /* DMA Normal Mode */
2813 if (HAL_IS_BIT_CLR(cfg1, SPI_CFG1_TXDMAEN | SPI_CFG1_RXDMAEN) || // IT based transfer is done
2814 ((State != HAL_SPI_STATE_BUSY_RX) && (hspi->hdmatx->Init.Mode == DMA_NORMAL)) || // DMA is used in normal mode
2815 ((State != HAL_SPI_STATE_BUSY_TX) && (hspi->hdmarx->Init.Mode == DMA_NORMAL))) // DMA is used in normal mode
2817 /* For the IT based receive extra polling maybe required for last packet */
2818 if (HAL_IS_BIT_CLR(hspi->Instance->CFG1, SPI_CFG1_TXDMAEN | SPI_CFG1_RXDMAEN))
2820 /* Pooling remaining data */
2821 while (hspi->RxXferCount != 0UL)
2823 /* Receive data in 32 Bit mode */
2824 if (hspi->Init.DataSize > SPI_DATASIZE_16BIT)
2826 *((uint32_t *)hspi->pRxBuffPtr) = *((__IO uint32_t *)&hspi->Instance->RXDR);
2827 hspi->pRxBuffPtr += sizeof(uint32_t);
2829 /* Receive data in 16 Bit mode */
2830 else if (hspi->Init.DataSize > SPI_DATASIZE_8BIT)
2832 #if defined (__GNUC__)
2833 *((uint16_t *)hspi->pRxBuffPtr) = *prxdr_16bits;
2834 #else
2835 *((uint16_t *)hspi->pRxBuffPtr) = *((__IO uint16_t *)&hspi->Instance->RXDR);
2836 #endif /* __GNUC__ */
2837 hspi->pRxBuffPtr += sizeof(uint16_t);
2839 /* Receive data in 8 Bit mode */
2840 else
2842 *((uint8_t *)hspi->pRxBuffPtr) = *((__IO uint8_t *)&hspi->Instance->RXDR);
2843 hspi->pRxBuffPtr += sizeof(uint8_t);
2846 hspi->RxXferCount--;
2850 /* Call SPI Standard close procedure */
2851 SPI_CloseTransfer(hspi);
2853 hspi->State = HAL_SPI_STATE_READY;
2854 if (hspi->ErrorCode != HAL_SPI_ERROR_NONE)
2856 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
2857 hspi->ErrorCallback(hspi);
2858 #else
2859 HAL_SPI_ErrorCallback(hspi);
2860 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
2861 return;
2865 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
2866 /* Call appropriate user callback */
2867 if (State == HAL_SPI_STATE_BUSY_TX_RX)
2869 hspi->TxRxCpltCallback(hspi);
2871 else if (State == HAL_SPI_STATE_BUSY_RX)
2873 hspi->RxCpltCallback(hspi);
2875 else if (State == HAL_SPI_STATE_BUSY_TX)
2877 hspi->TxCpltCallback(hspi);
2879 #else
2880 /* Call appropriate user callback */
2881 if (State == HAL_SPI_STATE_BUSY_TX_RX)
2883 HAL_SPI_TxRxCpltCallback(hspi);
2885 else if (State == HAL_SPI_STATE_BUSY_RX)
2887 HAL_SPI_RxCpltCallback(hspi);
2889 else if (State == HAL_SPI_STATE_BUSY_TX)
2891 HAL_SPI_TxCpltCallback(hspi);
2893 else
2895 /* end of the appropriate call */
2897 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
2899 return;
2902 if (HAL_IS_BIT_SET(itflag, SPI_FLAG_SUSP) && HAL_IS_BIT_SET(itsource, SPI_FLAG_EOT))
2904 /* Abort on going, clear SUSP flag to avoid infinite looping */
2905 __HAL_SPI_CLEAR_SUSPFLAG(hspi);
2907 return;
2910 /* SPI in Error Treatment --------------------------------------------------*/
2911 if ((trigger & (SPI_FLAG_MODF | SPI_FLAG_OVR | SPI_FLAG_FRE | SPI_FLAG_UDR)) != 0UL)
2913 /* SPI Overrun error interrupt occurred ----------------------------------*/
2914 if ((trigger & SPI_FLAG_OVR) != 0UL)
2916 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_OVR);
2917 __HAL_SPI_CLEAR_OVRFLAG(hspi);
2920 /* SPI Mode Fault error interrupt occurred -------------------------------*/
2921 if ((trigger & SPI_FLAG_MODF) != 0UL)
2923 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_MODF);
2924 __HAL_SPI_CLEAR_MODFFLAG(hspi);
2927 /* SPI Frame error interrupt occurred ------------------------------------*/
2928 if ((trigger & SPI_FLAG_FRE) != 0UL)
2930 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_FRE);
2931 __HAL_SPI_CLEAR_FREFLAG(hspi);
2934 /* SPI Underrun error interrupt occurred ------------------------------------*/
2935 if ((trigger & SPI_FLAG_UDR) != 0UL)
2937 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_UDR);
2938 __HAL_SPI_CLEAR_UDRFLAG(hspi);
2941 if (hspi->ErrorCode != HAL_SPI_ERROR_NONE)
2943 /* Disable SPI peripheral */
2944 __HAL_SPI_DISABLE(hspi);
2946 /* Disable all interrupts */
2947 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_EOT | SPI_IT_RXP | SPI_IT_TXP | SPI_IT_MODF | SPI_IT_OVR | SPI_IT_FRE | SPI_IT_UDR);
2949 /* Disable the SPI DMA requests if enabled */
2950 if (HAL_IS_BIT_SET(cfg1, SPI_CFG1_TXDMAEN | SPI_CFG1_RXDMAEN))
2952 /* Disable the SPI DMA requests */
2953 CLEAR_BIT(hspi->Instance->CFG1, SPI_CFG1_TXDMAEN | SPI_CFG1_RXDMAEN);
2955 /* Abort the SPI DMA Rx channel */
2956 if (hspi->hdmarx != NULL)
2958 /* Set the SPI DMA Abort callback :
2959 will lead to call HAL_SPI_ErrorCallback() at end of DMA abort procedure */
2960 hspi->hdmarx->XferAbortCallback = SPI_DMAAbortOnError;
2961 if (HAL_OK != HAL_DMA_Abort_IT(hspi->hdmarx))
2963 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_ABORT);
2966 /* Abort the SPI DMA Tx channel */
2967 if (hspi->hdmatx != NULL)
2969 /* Set the SPI DMA Abort callback :
2970 will lead to call HAL_SPI_ErrorCallback() at end of DMA abort procedure */
2971 hspi->hdmatx->XferAbortCallback = SPI_DMAAbortOnError;
2972 if (HAL_OK != HAL_DMA_Abort_IT(hspi->hdmatx))
2974 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_ABORT);
2978 else
2980 /* Restore hspi->State to Ready */
2981 hspi->State = HAL_SPI_STATE_READY;
2983 /* Call user error callback */
2984 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
2985 hspi->ErrorCallback(hspi);
2986 #else
2987 HAL_SPI_ErrorCallback(hspi);
2988 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
2991 return;
2996 * @brief Tx Transfer completed callback.
2997 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
2998 * the configuration information for SPI module.
2999 * @retval None
3001 __weak void HAL_SPI_TxCpltCallback(SPI_HandleTypeDef *hspi)
3003 /* Prevent unused argument(s) compilation warning */
3004 UNUSED(hspi);
3006 /* NOTE : This function should not be modified, when the callback is needed,
3007 the HAL_SPI_TxCpltCallback should be implemented in the user file
3012 * @brief Rx Transfer completed callback.
3013 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3014 * the configuration information for SPI module.
3015 * @retval None
3017 __weak void HAL_SPI_RxCpltCallback(SPI_HandleTypeDef *hspi)
3019 /* Prevent unused argument(s) compilation warning */
3020 UNUSED(hspi);
3022 /* NOTE : This function should not be modified, when the callback is needed,
3023 the HAL_SPI_RxCpltCallback should be implemented in the user file
3028 * @brief Tx and Rx Transfer completed callback.
3029 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3030 * the configuration information for SPI module.
3031 * @retval None
3033 __weak void HAL_SPI_TxRxCpltCallback(SPI_HandleTypeDef *hspi)
3035 /* Prevent unused argument(s) compilation warning */
3036 UNUSED(hspi);
3038 /* NOTE : This function should not be modified, when the callback is needed,
3039 the HAL_SPI_TxRxCpltCallback should be implemented in the user file
3044 * @brief Tx Half Transfer completed callback.
3045 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3046 * the configuration information for SPI module.
3047 * @retval None
3049 __weak void HAL_SPI_TxHalfCpltCallback(SPI_HandleTypeDef *hspi)
3051 /* Prevent unused argument(s) compilation warning */
3052 UNUSED(hspi);
3054 /* NOTE : This function should not be modified, when the callback is needed,
3055 the HAL_SPI_TxHalfCpltCallback should be implemented in the user file
3060 * @brief Rx Half Transfer completed callback.
3061 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3062 * the configuration information for SPI module.
3063 * @retval None
3065 __weak void HAL_SPI_RxHalfCpltCallback(SPI_HandleTypeDef *hspi)
3067 /* Prevent unused argument(s) compilation warning */
3068 UNUSED(hspi);
3070 /* NOTE : This function should not be modified, when the callback is needed,
3071 the HAL_SPI_RxHalfCpltCallback() should be implemented in the user file
3076 * @brief Tx and Rx Half Transfer callback.
3077 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3078 * the configuration information for SPI module.
3079 * @retval None
3081 __weak void HAL_SPI_TxRxHalfCpltCallback(SPI_HandleTypeDef *hspi)
3083 /* Prevent unused argument(s) compilation warning */
3084 UNUSED(hspi);
3086 /* NOTE : This function should not be modified, when the callback is needed,
3087 the HAL_SPI_TxRxHalfCpltCallback() should be implemented in the user file
3092 * @brief SPI error callback.
3093 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3094 * the configuration information for SPI module.
3095 * @retval None
3097 __weak void HAL_SPI_ErrorCallback(SPI_HandleTypeDef *hspi)
3099 /* Prevent unused argument(s) compilation warning */
3100 UNUSED(hspi);
3102 /* NOTE : This function should not be modified, when the callback is needed,
3103 the HAL_SPI_ErrorCallback should be implemented in the user file
3105 /* NOTE : The ErrorCode parameter in the hspi handle is updated by the SPI processes
3106 and user can use HAL_SPI_GetError() API to check the latest error occurred
3111 * @brief SPI Abort Complete callback.
3112 * @param hspi SPI handle.
3113 * @retval None
3115 __weak void HAL_SPI_AbortCpltCallback(SPI_HandleTypeDef *hspi)
3117 /* Prevent unused argument(s) compilation warning */
3118 UNUSED(hspi);
3120 /* NOTE : This function should not be modified, when the callback is needed,
3121 the HAL_SPI_AbortCpltCallback can be implemented in the user file.
3126 * @}
3129 /** @defgroup SPI_Exported_Functions_Group3 Peripheral State and Errors functions
3130 * @brief SPI control functions
3132 @verbatim
3133 ===============================================================================
3134 ##### Peripheral State and Errors functions #####
3135 ===============================================================================
3136 [..]
3137 This subsection provides a set of functions allowing to control the SPI.
3138 (+) HAL_SPI_GetState() API can be helpful to check in run-time the state of the SPI peripheral
3139 (+) HAL_SPI_GetError() check in run-time Errors occurring during communication
3140 @endverbatim
3141 * @{
3145 * @brief Return the SPI handle state.
3146 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3147 * the configuration information for SPI module.
3148 * @retval SPI state
3150 HAL_SPI_StateTypeDef HAL_SPI_GetState(SPI_HandleTypeDef *hspi)
3152 /* Return SPI handle state */
3153 return hspi->State;
3157 * @brief Return the SPI error code.
3158 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3159 * the configuration information for SPI module.
3160 * @retval SPI error code in bitmap format
3162 uint32_t HAL_SPI_GetError(SPI_HandleTypeDef *hspi)
3164 /* Return SPI ErrorCode */
3165 return hspi->ErrorCode;
3169 * @}
3173 * @}
3176 /** @addtogroup SPI_Private_Functions
3177 * @brief Private functions
3178 * @{
3182 * @brief DMA SPI transmit process complete callback.
3183 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
3184 * the configuration information for the specified DMA module.
3185 * @retval None
3187 static void SPI_DMATransmitCplt(DMA_HandleTypeDef *hdma)
3189 SPI_HandleTypeDef *hspi = (SPI_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent;
3191 if (hspi->State != HAL_SPI_STATE_ABORT)
3193 if (hspi->hdmatx->Init.Mode == DMA_CIRCULAR)
3195 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
3196 hspi->TxCpltCallback(hspi);
3197 #else
3198 HAL_SPI_TxCpltCallback(hspi);
3199 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
3201 else
3203 /* Enable EOT interrupt */
3204 __HAL_SPI_ENABLE_IT(hspi, SPI_IT_EOT);
3210 * @brief DMA SPI receive process complete callback.
3211 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
3212 * the configuration information for the specified DMA module.
3213 * @retval None
3215 static void SPI_DMAReceiveCplt(DMA_HandleTypeDef *hdma)
3217 SPI_HandleTypeDef *hspi = (SPI_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent;
3219 if (hspi->State != HAL_SPI_STATE_ABORT)
3221 if (hspi->hdmarx->Init.Mode == DMA_CIRCULAR)
3223 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
3224 hspi->RxCpltCallback(hspi);
3225 #else
3226 HAL_SPI_RxCpltCallback(hspi);
3227 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
3229 else
3231 /* Enable EOT interrupt */
3232 __HAL_SPI_ENABLE_IT(hspi, SPI_IT_EOT);
3238 * @brief DMA SPI transmit receive process complete callback.
3239 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
3240 * the configuration information for the specified DMA module.
3241 * @retval None
3243 static void SPI_DMATransmitReceiveCplt(DMA_HandleTypeDef *hdma)
3245 SPI_HandleTypeDef *hspi = (SPI_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent;
3247 if (hspi->State != HAL_SPI_STATE_ABORT)
3249 if (hspi->hdmatx->Init.Mode == DMA_CIRCULAR)
3251 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
3252 hspi->TxRxCpltCallback(hspi);
3253 #else
3254 HAL_SPI_TxRxCpltCallback(hspi);
3255 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
3257 else
3259 /* Enable EOT interrupt */
3260 __HAL_SPI_ENABLE_IT(hspi, SPI_IT_EOT);
3266 * @brief DMA SPI half transmit process complete callback.
3267 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
3268 * the configuration information for the specified DMA module.
3269 * @retval None
3271 static void SPI_DMAHalfTransmitCplt(DMA_HandleTypeDef *hdma)
3273 SPI_HandleTypeDef *hspi = (SPI_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent;
3275 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
3276 hspi->TxHalfCpltCallback(hspi);
3277 #else
3278 HAL_SPI_TxHalfCpltCallback(hspi);
3279 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
3283 * @brief DMA SPI half receive process complete callback
3284 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
3285 * the configuration information for the specified DMA module.
3286 * @retval None
3288 static void SPI_DMAHalfReceiveCplt(DMA_HandleTypeDef *hdma)
3290 SPI_HandleTypeDef *hspi = (SPI_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent;
3292 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
3293 hspi->RxHalfCpltCallback(hspi);
3294 #else
3295 HAL_SPI_RxHalfCpltCallback(hspi);
3296 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
3300 * @brief DMA SPI half transmit receive process complete callback.
3301 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
3302 * the configuration information for the specified DMA module.
3303 * @retval None
3305 static void SPI_DMAHalfTransmitReceiveCplt(DMA_HandleTypeDef *hdma)
3307 SPI_HandleTypeDef *hspi = (SPI_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent;
3309 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
3310 hspi->TxRxHalfCpltCallback(hspi);
3311 #else
3312 HAL_SPI_TxRxHalfCpltCallback(hspi);
3313 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
3317 * @brief DMA SPI communication error callback.
3318 * @param hdma: pointer to a DMA_HandleTypeDef structure that contains
3319 * the configuration information for the specified DMA module.
3320 * @retval None
3322 static void SPI_DMAError(DMA_HandleTypeDef *hdma)
3324 SPI_HandleTypeDef *hspi = (SPI_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent;
3326 /* if DMA error is FIFO error ignore it */
3327 if (HAL_DMA_GetError(hdma) != HAL_DMA_ERROR_FE)
3329 /* Call SPI standard close procedure */
3330 SPI_CloseTransfer(hspi);
3332 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_DMA);
3333 hspi->State = HAL_SPI_STATE_READY;
3334 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
3335 hspi->ErrorCallback(hspi);
3336 #else
3337 HAL_SPI_ErrorCallback(hspi);
3338 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
3343 * @brief DMA SPI communication abort callback, when initiated by HAL services on Error
3344 * (To be called at end of DMA Abort procedure following error occurrence).
3345 * @param hdma DMA handle.
3346 * @retval None
3348 static void SPI_DMAAbortOnError(DMA_HandleTypeDef *hdma)
3350 SPI_HandleTypeDef *hspi = (SPI_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent;
3351 hspi->RxXferCount = (uint16_t) 0UL;
3352 hspi->TxXferCount = (uint16_t) 0UL;
3354 /* Restore hspi->State to Ready */
3355 hspi->State = HAL_SPI_STATE_READY;
3357 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
3358 hspi->ErrorCallback(hspi);
3359 #else
3360 HAL_SPI_ErrorCallback(hspi);
3361 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
3365 * @brief DMA SPI Tx communication abort callback, when initiated by user
3366 * (To be called at end of DMA Tx Abort procedure following user abort request).
3367 * @note When this callback is executed, User Abort complete call back is called only if no
3368 * Abort still ongoing for Rx DMA Handle.
3369 * @param hdma DMA handle.
3370 * @retval None
3372 static void SPI_DMATxAbortCallback(DMA_HandleTypeDef *hdma)
3374 SPI_HandleTypeDef *hspi = (SPI_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent;
3376 hspi->hdmatx->XferAbortCallback = NULL;
3378 /* Check if an Abort process is still ongoing */
3379 if (hspi->hdmarx != NULL)
3381 if (hspi->hdmarx->XferAbortCallback != NULL)
3383 return;
3387 /* Call the Abort procedure */
3388 SPI_AbortTransfer(hspi);
3390 /* Restore hspi->State to Ready */
3391 hspi->State = HAL_SPI_STATE_READY;
3393 /* Call user Abort complete callback */
3394 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
3395 hspi->AbortCpltCallback(hspi);
3396 #else
3397 HAL_SPI_AbortCpltCallback(hspi);
3398 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
3402 * @brief DMA SPI Rx communication abort callback, when initiated by user
3403 * (To be called at end of DMA Rx Abort procedure following user abort request).
3404 * @note When this callback is executed, User Abort complete call back is called only if no
3405 * Abort still ongoing for Tx DMA Handle.
3406 * @param hdma DMA handle.
3407 * @retval None
3409 static void SPI_DMARxAbortCallback(DMA_HandleTypeDef *hdma)
3411 SPI_HandleTypeDef *hspi = (SPI_HandleTypeDef *)((DMA_HandleTypeDef *)hdma)->Parent;
3413 hspi->hdmarx->XferAbortCallback = NULL;
3415 /* Check if an Abort process is still ongoing */
3416 if (hspi->hdmatx != NULL)
3418 if (hspi->hdmatx->XferAbortCallback != NULL)
3420 return;
3424 /* Call the Abort procedure */
3425 SPI_AbortTransfer(hspi);
3427 /* Restore hspi->State to Ready */
3428 hspi->State = HAL_SPI_STATE_READY;
3430 /* Call user Abort complete callback */
3431 #if (USE_HAL_SPI_REGISTER_CALLBACKS == 1UL)
3432 hspi->AbortCpltCallback(hspi);
3433 #else
3434 HAL_SPI_AbortCpltCallback(hspi);
3435 #endif /* USE_HAL_SPI_REGISTER_CALLBACKS */
3439 * @brief Manage the receive 8-bit in Interrupt context.
3440 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3441 * the configuration information for SPI module.
3442 * @retval None
3444 static void SPI_RxISR_8BIT(SPI_HandleTypeDef *hspi)
3446 /* Receive data in 8 Bit mode */
3447 *((uint8_t *)hspi->pRxBuffPtr) = (*(__IO uint8_t *)&hspi->Instance->RXDR);
3448 hspi->pRxBuffPtr += sizeof(uint8_t);
3449 hspi->RxXferCount--;
3451 /* Disable IT if no more data excepted */
3452 if (hspi->RxXferCount == 0UL)
3454 #if defined(USE_SPI_RELOAD_TRANSFER)
3455 /* Check if there is any request to reload */
3456 if (hspi->Reload.Requested == 1UL)
3458 hspi->RxXferSize = hspi->Reload.RxXferSize;
3459 hspi->RxXferCount = hspi->Reload.RxXferSize;
3460 hspi->pRxBuffPtr = hspi->Reload.pRxBuffPtr;
3462 else
3464 /* Disable RXP interrupts */
3465 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_RXP);
3467 #else
3468 /* Disable RXP interrupts */
3469 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_RXP);
3470 #endif /* USE_HSPI_RELOAD_TRANSFER */
3476 * @brief Manage the 16-bit receive in Interrupt context.
3477 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3478 * the configuration information for SPI module.
3479 * @retval None
3481 static void SPI_RxISR_16BIT(SPI_HandleTypeDef *hspi)
3483 /* Receive data in 16 Bit mode */
3484 #if defined (__GNUC__)
3485 __IO uint16_t *prxdr_16bits = (__IO uint16_t *)(&(hspi->Instance->RXDR));
3487 *((uint16_t *)hspi->pRxBuffPtr) = *prxdr_16bits;
3488 #else
3489 *((uint16_t *)hspi->pRxBuffPtr) = (*(__IO uint16_t *)&hspi->Instance->RXDR);
3490 #endif /* __GNUC__ */
3491 hspi->pRxBuffPtr += sizeof(uint16_t);
3492 hspi->RxXferCount--;
3494 /* Disable IT if no more data excepted */
3495 if (hspi->RxXferCount == 0UL)
3497 #if defined(USE_SPI_RELOAD_TRANSFER)
3498 /* Check if there is any request to reload */
3499 if (hspi->Reload.Requested == 1UL)
3501 hspi->RxXferSize = hspi->Reload.RxXferSize;
3502 hspi->RxXferCount = hspi->Reload.RxXferSize;
3503 hspi->pRxBuffPtr = hspi->Reload.pRxBuffPtr;
3505 else
3507 /* Disable RXP interrupts */
3508 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_RXP);
3510 #else
3511 /* Disable RXP interrupts */
3512 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_RXP);
3513 #endif /* USE_HSPI_RELOAD_TRANSFER */
3519 * @brief Manage the 32-bit receive in Interrupt context.
3520 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3521 * the configuration information for SPI module.
3522 * @retval None
3524 static void SPI_RxISR_32BIT(SPI_HandleTypeDef *hspi)
3526 /* Receive data in 32 Bit mode */
3527 *((uint32_t *)hspi->pRxBuffPtr) = (*(__IO uint32_t *)&hspi->Instance->RXDR);
3528 hspi->pRxBuffPtr += sizeof(uint32_t);
3529 hspi->RxXferCount--;
3531 /* Disable IT if no more data excepted */
3532 if (hspi->RxXferCount == 0UL)
3534 #if defined(USE_SPI_RELOAD_TRANSFER)
3535 /* Check if there is any request to reload */
3536 if (hspi->Reload.Requested == 1UL)
3538 hspi->RxXferSize = hspi->Reload.RxXferSize;
3539 hspi->RxXferCount = hspi->Reload.RxXferSize;
3540 hspi->pRxBuffPtr = hspi->Reload.pRxBuffPtr;
3542 else
3544 /* Disable RXP interrupts */
3545 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_RXP);
3547 #else
3548 /* Disable RXP interrupts */
3549 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_RXP);
3550 #endif /* USE_HSPI_RELOAD_TRANSFER */
3556 * @brief Handle the data 8-bit transmit in Interrupt mode.
3557 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3558 * the configuration information for SPI module.
3559 * @retval None
3561 static void SPI_TxISR_8BIT(SPI_HandleTypeDef *hspi)
3563 /* Transmit data in 8 Bit mode */
3564 *(__IO uint8_t *)&hspi->Instance->TXDR = *((uint8_t *)hspi->pTxBuffPtr);
3565 hspi->pTxBuffPtr += sizeof(uint8_t);
3566 hspi->TxXferCount--;
3568 /* Disable IT if no more data excepted */
3569 if (hspi->TxXferCount == 0UL)
3571 #if defined(USE_SPI_RELOAD_TRANSFER)
3572 /* Check if there is any request to reload */
3573 if (hspi->Reload.Requested == 1UL)
3575 hspi->TxXferSize = hspi->Reload.TxXferSize;
3576 hspi->TxXferCount = hspi->Reload.TxXferSize;
3577 hspi->pTxBuffPtr = hspi->Reload.pTxBuffPtr;
3579 else
3581 /* Disable TXP interrupts */
3582 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_TXP);
3584 #else
3585 /* Disable TXP interrupts */
3586 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_TXP);
3587 #endif /* USE_HSPI_RELOAD_TRANSFER */
3592 * @brief Handle the data 16-bit transmit in Interrupt mode.
3593 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3594 * the configuration information for SPI module.
3595 * @retval None
3597 static void SPI_TxISR_16BIT(SPI_HandleTypeDef *hspi)
3599 /* Transmit data in 16 Bit mode */
3600 #if defined (__GNUC__)
3601 __IO uint16_t *ptxdr_16bits = (__IO uint16_t *)(&(hspi->Instance->TXDR));
3603 *ptxdr_16bits = *((uint16_t *)hspi->pTxBuffPtr);
3604 #else
3605 *((__IO uint16_t *)&hspi->Instance->TXDR) = *((uint16_t *)hspi->pTxBuffPtr);
3606 #endif /* __GNUC__ */
3607 hspi->pTxBuffPtr += sizeof(uint16_t);
3608 hspi->TxXferCount--;
3610 /* Disable IT if no more data excepted */
3611 if (hspi->TxXferCount == 0UL)
3613 #if defined(USE_SPI_RELOAD_TRANSFER)
3614 /* Check if there is any request to reload */
3615 if (hspi->Reload.Requested == 1UL)
3617 hspi->TxXferSize = hspi->Reload.TxXferSize;
3618 hspi->TxXferCount = hspi->Reload.TxXferSize;
3619 hspi->pTxBuffPtr = hspi->Reload.pTxBuffPtr;
3621 else
3623 /* Disable TXP interrupts */
3624 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_TXP);
3626 #else
3627 /* Disable TXP interrupts */
3628 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_TXP);
3629 #endif /* USE_HSPI_RELOAD_TRANSFER */
3634 * @brief Handle the data 32-bit transmit in Interrupt mode.
3635 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3636 * the configuration information for SPI module.
3637 * @retval None
3639 static void SPI_TxISR_32BIT(SPI_HandleTypeDef *hspi)
3641 /* Transmit data in 32 Bit mode */
3642 *((__IO uint32_t *)&hspi->Instance->TXDR) = *((uint32_t *)hspi->pTxBuffPtr);
3643 hspi->pTxBuffPtr += sizeof(uint32_t);
3644 hspi->TxXferCount--;
3646 /* Disable IT if no more data excepted */
3647 if (hspi->TxXferCount == 0UL)
3649 #if defined(USE_SPI_RELOAD_TRANSFER)
3650 /* Check if there is any request to reload */
3651 if (hspi->Reload.Requested == 1UL)
3653 hspi->TxXferSize = hspi->Reload.TxXferSize;
3654 hspi->TxXferCount = hspi->Reload.TxXferSize;
3655 hspi->pTxBuffPtr = hspi->Reload.pTxBuffPtr;
3657 else
3659 /* Disable TXP interrupts */
3660 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_TXP);
3662 #else
3663 /* Disable TXP interrupts */
3664 __HAL_SPI_DISABLE_IT(hspi, SPI_IT_TXP);
3665 #endif /* USE_HSPI_RELOAD_TRANSFER */
3670 * @brief Abort Transfer and clear flags.
3671 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3672 * the configuration information for SPI module.
3673 * @retval None
3675 static void SPI_AbortTransfer(SPI_HandleTypeDef *hspi)
3677 /* Disable SPI peripheral */
3678 __HAL_SPI_DISABLE(hspi);
3680 /* Disable ITs */
3681 __HAL_SPI_DISABLE_IT(hspi, (SPI_IT_EOT | SPI_IT_TXP | SPI_IT_RXP | SPI_IT_DXP | SPI_IT_UDR | SPI_IT_OVR | SPI_IT_FRE | SPI_IT_MODF));
3683 /* Clear the Status flags in the SR register */
3684 __HAL_SPI_CLEAR_EOTFLAG(hspi);
3685 __HAL_SPI_CLEAR_TXTFFLAG(hspi);
3687 /* Disable Tx DMA Request */
3688 CLEAR_BIT(hspi->Instance->CFG1, SPI_CFG1_TXDMAEN | SPI_CFG1_RXDMAEN);
3690 /* Clear the Error flags in the SR register */
3691 __HAL_SPI_CLEAR_OVRFLAG(hspi);
3692 __HAL_SPI_CLEAR_UDRFLAG(hspi);
3693 __HAL_SPI_CLEAR_FREFLAG(hspi);
3694 __HAL_SPI_CLEAR_MODFFLAG(hspi);
3695 __HAL_SPI_CLEAR_SUSPFLAG(hspi);
3697 #if (USE_SPI_CRC != 0U)
3698 __HAL_SPI_CLEAR_CRCERRFLAG(hspi);
3699 #endif /* USE_SPI_CRC */
3701 hspi->TxXferCount = (uint16_t)0UL;
3702 hspi->RxXferCount = (uint16_t)0UL;
3707 * @brief Close Transfer and clear flags.
3708 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3709 * the configuration information for SPI module.
3710 * @retval HAL_ERROR: if any error detected
3711 * HAL_OK: if nothing detected
3713 static void SPI_CloseTransfer(SPI_HandleTypeDef *hspi)
3715 uint32_t itflag = hspi->Instance->SR;
3717 __HAL_SPI_CLEAR_EOTFLAG(hspi);
3718 __HAL_SPI_CLEAR_TXTFFLAG(hspi);
3720 /* Disable SPI peripheral */
3721 __HAL_SPI_DISABLE(hspi);
3723 /* Disable ITs */
3724 __HAL_SPI_DISABLE_IT(hspi, (SPI_IT_EOT | SPI_IT_TXP | SPI_IT_RXP | SPI_IT_DXP | SPI_IT_UDR | SPI_IT_OVR | SPI_IT_FRE | SPI_IT_MODF));
3726 /* Disable Tx DMA Request */
3727 CLEAR_BIT(hspi->Instance->CFG1, SPI_CFG1_TXDMAEN | SPI_CFG1_RXDMAEN);
3729 /* Report UnderRun error for non RX Only communication */
3730 if (hspi->State != HAL_SPI_STATE_BUSY_RX)
3732 if ((itflag & SPI_FLAG_UDR) != 0UL)
3734 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_UDR);
3735 __HAL_SPI_CLEAR_UDRFLAG(hspi);
3739 /* Report OverRun error for non TX Only communication */
3740 if (hspi->State != HAL_SPI_STATE_BUSY_TX)
3742 if ((itflag & SPI_FLAG_OVR) != 0UL)
3744 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_OVR);
3745 __HAL_SPI_CLEAR_OVRFLAG(hspi);
3748 #if (USE_SPI_CRC != 0UL)
3749 /* Check if CRC error occurred */
3750 if (hspi->Init.CRCCalculation == SPI_CRCCALCULATION_ENABLE)
3752 if ((itflag & SPI_FLAG_CRCERR) != 0UL)
3754 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_CRC);
3755 __HAL_SPI_CLEAR_CRCERRFLAG(hspi);
3758 #endif /* USE_SPI_CRC */
3761 /* SPI Mode Fault error interrupt occurred -------------------------------*/
3762 if ((itflag & SPI_FLAG_MODF) != 0UL)
3764 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_MODF);
3765 __HAL_SPI_CLEAR_MODFFLAG(hspi);
3768 /* SPI Frame error interrupt occurred ------------------------------------*/
3769 if ((itflag & SPI_FLAG_FRE) != 0UL)
3771 SET_BIT(hspi->ErrorCode, HAL_SPI_ERROR_FRE);
3772 __HAL_SPI_CLEAR_FREFLAG(hspi);
3775 hspi->TxXferCount = (uint16_t)0UL;
3776 hspi->RxXferCount = (uint16_t)0UL;
3780 * @brief Handle SPI Communication Timeout.
3781 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3782 * the configuration information for SPI module.
3783 * @param Flag: SPI flag to check
3784 * @param Status: flag state to check
3785 * @param Timeout: Timeout duration
3786 * @param Tickstart: Tick start value
3787 * @retval HAL status
3789 static HAL_StatusTypeDef SPI_WaitOnFlagUntilTimeout(SPI_HandleTypeDef *hspi, uint32_t Flag, FlagStatus Status,
3790 uint32_t Tickstart, uint32_t Timeout)
3792 /* Wait until flag is set */
3793 while ((__HAL_SPI_GET_FLAG(hspi, Flag) ? SET : RESET) == Status)
3795 /* Check for the Timeout */
3796 if ((((HAL_GetTick() - Tickstart) >= Timeout) && (Timeout != HAL_MAX_DELAY)) || (Timeout == 0U))
3798 return HAL_TIMEOUT;
3801 return HAL_OK;
3805 * @brief Compute configured packet size from fifo perspective.
3806 * @param hspi: pointer to a SPI_HandleTypeDef structure that contains
3807 * the configuration information for SPI module.
3808 * @retval Packet size occupied in the fifo
3810 static uint32_t SPI_GetPacketSize(SPI_HandleTypeDef *hspi)
3812 uint32_t fifo_threashold = (hspi->Init.FifoThreshold >> SPI_CFG1_FTHLV_Pos) + 1UL;
3813 uint32_t data_size = (hspi->Init.DataSize >> SPI_CFG1_DSIZE_Pos) + 1UL;
3815 /* Convert data size to Byte */
3816 data_size = (data_size + 7UL) / 8UL;
3818 return data_size * fifo_threashold;
3823 * @}
3826 #endif /* HAL_SPI_MODULE_ENABLED */
3829 * @}
3833 * @}
3836 /************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/