|
@@ -0,0 +1,3020 @@
|
|
|
+/*==================================================================================================
|
|
|
+* Project : RTD AUTOSAR 4.4
|
|
|
+* Platform : CORTEXM
|
|
|
+* Peripheral : LPSPI
|
|
|
+* Dependencies :
|
|
|
+*
|
|
|
+* Autosar Version : 4.4.0
|
|
|
+* Autosar Revision : ASR_REL_4_4_REV_0000
|
|
|
+* Autosar Conf.Variant :
|
|
|
+* SW Version : 1.0.0
|
|
|
+* Build Version : S32K1_RTD_1_0_0_HF01_D2109_ASR_REL_4_4_REV_0000_20210907
|
|
|
+*
|
|
|
+* (c) Copyright 2020-2021 NXP Semiconductors
|
|
|
+* All Rights Reserved.
|
|
|
+*
|
|
|
+* NXP Confidential. This software is owned or controlled by NXP and may only be
|
|
|
+* used strictly in accordance with the applicable license terms. By expressly
|
|
|
+* accepting such terms or by downloading, installing, activating and/or otherwise
|
|
|
+* using the software, you are agreeing that you have read, and that you agree to
|
|
|
+* comply with and are bound by, such license terms. If you do not agree to be
|
|
|
+* bound by the applicable license terms, then you may not retain, install,
|
|
|
+* activate or otherwise use the software.
|
|
|
+==================================================================================================*/
|
|
|
+
|
|
|
+/**
|
|
|
+* @file Lpspi_Ip.c
|
|
|
+*
|
|
|
+*
|
|
|
+* @brief LPSPI low-level driver implementations.
|
|
|
+* @details LPSPI low-level driver implementations.
|
|
|
+*
|
|
|
+* @addtogroup LPSPI_DRIVER Lpspi Driver
|
|
|
+* @{
|
|
|
+*/
|
|
|
+
|
|
|
+#ifdef __cplusplus
|
|
|
+extern "C"
|
|
|
+{
|
|
|
+#endif
|
|
|
+
|
|
|
+/*==================================================================================================
|
|
|
+* INCLUDE FILES
|
|
|
+* 1) system and project includes
|
|
|
+* 2) needed interfaces from external units
|
|
|
+* 3) internal and external interfaces from this unit
|
|
|
+==================================================================================================*/
|
|
|
+#include "Mcal.h"
|
|
|
+#include "Lpspi_Ip.h"
|
|
|
+#include "Lpspi_Ip_Cfg.h"
|
|
|
+#include "OsIf.h"
|
|
|
+#if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ #include "Dma_Ip.h"
|
|
|
+#endif
|
|
|
+#if (STD_ON == LPSPI_IP_ENABLE_USER_MODE_SUPPORT)
|
|
|
+ #define USER_MODE_REG_PROT_ENABLED (LPSPI_IP_ENABLE_USER_MODE_SUPPORT)
|
|
|
+ #include "RegLockMacros.h"
|
|
|
+#endif
|
|
|
+#include "SchM_Spi.h"
|
|
|
+#if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ #include "Devassert.h"
|
|
|
+#endif
|
|
|
+
|
|
|
+/*==================================================================================================
|
|
|
+* SOURCE FILE VERSION INFORMATION
|
|
|
+==================================================================================================*/
|
|
|
+#define LPSPI_IP_VENDOR_ID_C 43
|
|
|
+#define LPSPI_IP_AR_RELEASE_MAJOR_VERSION_C 4
|
|
|
+#define LPSPI_IP_AR_RELEASE_MINOR_VERSION_C 4
|
|
|
+#define LPSPI_IP_AR_RELEASE_REVISION_VERSION_C 0
|
|
|
+#define LPSPI_IP_SW_MAJOR_VERSION_C 1
|
|
|
+#define LPSPI_IP_SW_MINOR_VERSION_C 0
|
|
|
+#define LPSPI_IP_SW_PATCH_VERSION_C 0
|
|
|
+/*==================================================================================================
|
|
|
+* FILE VERSION CHECKS
|
|
|
+==================================================================================================*/
|
|
|
+#ifndef DISABLE_MCAL_INTERMODULE_ASR_CHECK
|
|
|
+ /* Check if current file and Mcal header file are of the same Autosar version */
|
|
|
+ #if ((LPSPI_IP_AR_RELEASE_MAJOR_VERSION_C != MCAL_AR_RELEASE_MAJOR_VERSION) || \
|
|
|
+ (LPSPI_IP_AR_RELEASE_MINOR_VERSION_C != MCAL_AR_RELEASE_MINOR_VERSION))
|
|
|
+ #error "AutoSar Version Numbers of Lpspi_Ip.c and Mcal.h are different"
|
|
|
+ #endif
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ /* Check if current file and Dma_Ip header file are of the same Autosar version */
|
|
|
+ #if ((DMA_IP_AR_RELEASE_MAJOR_VERSION_H != LPSPI_IP_AR_RELEASE_MAJOR_VERSION_C) || \
|
|
|
+ (DMA_IP_AR_RELEASE_MINOR_VERSION_H != LPSPI_IP_AR_RELEASE_MINOR_VERSION_C))
|
|
|
+ #error "AutoSar Version Numbers of Lpspi_Ip.c and Dma_Ip.h are different"
|
|
|
+ #endif
|
|
|
+ #endif
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_ENABLE_USER_MODE_SUPPORT)
|
|
|
+ #if ((LPSPI_IP_AR_RELEASE_MAJOR_VERSION_C != REGLOCKMACROS_AR_RELEASE_MAJOR_VERSION) || \
|
|
|
+ (LPSPI_IP_AR_RELEASE_MINOR_VERSION_C != REGLOCKMACROS_AR_RELEASE_MINOR_VERSION))
|
|
|
+ #error "AutoSar Version Numbers of Lpspi_Ip.c and RegLockMacros.h are different"
|
|
|
+ #endif
|
|
|
+ #endif
|
|
|
+
|
|
|
+ /* Check if the current file and SchM_Spi.h header file are of the same version */
|
|
|
+ #if ((LPSPI_IP_AR_RELEASE_MAJOR_VERSION_C != SCHM_SPI_AR_RELEASE_MAJOR_VERSION) || \
|
|
|
+ (LPSPI_IP_AR_RELEASE_MINOR_VERSION_C != SCHM_SPI_AR_RELEASE_MINOR_VERSION) \
|
|
|
+ )
|
|
|
+ #error "AutoSar Version Numbers of Lpspi_Ip.c and SchM_Spi.h are different"
|
|
|
+ #endif
|
|
|
+
|
|
|
+ /* Check if the current file and OsIf.h header file are of the same version */
|
|
|
+ #if ((LPSPI_IP_AR_RELEASE_MAJOR_VERSION_C != OSIF_AR_RELEASE_MAJOR_VERSION) || \
|
|
|
+ (LPSPI_IP_AR_RELEASE_MINOR_VERSION_C != OSIF_AR_RELEASE_MINOR_VERSION) \
|
|
|
+ )
|
|
|
+ #error "AutoSar Version Numbers of Lpspi_Ip.c and OsIf.h are different"
|
|
|
+ #endif
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ /* Check if the current file and Devassert.h header file are of the same version */
|
|
|
+ #if ((LPSPI_IP_AR_RELEASE_MAJOR_VERSION_C != DEVASSERT_AR_RELEASE_MAJOR_VERSION) || \
|
|
|
+ (LPSPI_IP_AR_RELEASE_MINOR_VERSION_C != DEVASSERT_AR_RELEASE_MINOR_VERSION) \
|
|
|
+ )
|
|
|
+ #error "AutoSar Version Numbers of Lpspi_Ip.c and Devassert.h are different"
|
|
|
+ #endif
|
|
|
+ #endif /* (STD_ON == LPSPI_IP_DEV_ERROR_DETECT) */
|
|
|
+#endif
|
|
|
+
|
|
|
+/* Check if Lpspi_Ip.h and Lpspi_Ip.c are of the same vendor */
|
|
|
+#if (LPSPI_IP_VENDOR_ID != LPSPI_IP_VENDOR_ID_C)
|
|
|
+ #error "Lpspi_Ip.h and Lpspi_Ip.c have different vendor ids"
|
|
|
+#endif
|
|
|
+/* Check if Lpspi_Ip.h file and Lpspi_Ip.c file are of the same Autosar version */
|
|
|
+#if ((LPSPI_IP_AR_RELEASE_MAJOR_VERSION != LPSPI_IP_AR_RELEASE_MAJOR_VERSION_C) || \
|
|
|
+ (LPSPI_IP_AR_RELEASE_MINOR_VERSION != LPSPI_IP_AR_RELEASE_MINOR_VERSION_C) || \
|
|
|
+ (LPSPI_IP_AR_RELEASE_REVISION_VERSION != LPSPI_IP_AR_RELEASE_REVISION_VERSION_C))
|
|
|
+#error "AutoSar Version Numbers of Lpspi_Ip.h and Lpspi_Ip.c are different"
|
|
|
+#endif
|
|
|
+#if ((LPSPI_IP_SW_MAJOR_VERSION != LPSPI_IP_SW_MAJOR_VERSION_C) || \
|
|
|
+ (LPSPI_IP_SW_MINOR_VERSION != LPSPI_IP_SW_MINOR_VERSION_C) || \
|
|
|
+ (LPSPI_IP_SW_PATCH_VERSION != LPSPI_IP_SW_PATCH_VERSION_C))
|
|
|
+#error "Software Version Numbers of Lpspi_Ip.h and Lpspi_Ip.c are different"
|
|
|
+#endif
|
|
|
+
|
|
|
+/* Check if Lpspi_Ip_Cfg.h and Lpspi_Ip.c are of the same vendor */
|
|
|
+#if (LPSPI_IP_VENDOR_ID_CFG != LPSPI_IP_VENDOR_ID_C)
|
|
|
+ #error "Lpspi_Ip_Cfg.h and Lpspi_Ip.c have different vendor ids"
|
|
|
+#endif
|
|
|
+/* Check if Lpspi_Ip_Cfg.h file and Lpspi_Ip.c file are of the same Autosar version */
|
|
|
+#if ((LPSPI_IP_AR_RELEASE_MAJOR_VERSION_CFG != LPSPI_IP_AR_RELEASE_MAJOR_VERSION_C) || \
|
|
|
+ (LPSPI_IP_AR_RELEASE_MINOR_VERSION_CFG != LPSPI_IP_AR_RELEASE_MINOR_VERSION_C) || \
|
|
|
+ (LPSPI_IP_AR_RELEASE_REVISION_VERSION_CFG != LPSPI_IP_AR_RELEASE_REVISION_VERSION_C))
|
|
|
+#error "AutoSar Version Numbers of Lpspi_Ip_Cfg.h and Lpspi_Ip.c are different"
|
|
|
+#endif
|
|
|
+#if ((LPSPI_IP_SW_MAJOR_VERSION_CFG != LPSPI_IP_SW_MAJOR_VERSION_C) || \
|
|
|
+ (LPSPI_IP_SW_MINOR_VERSION_CFG != LPSPI_IP_SW_MINOR_VERSION_C) || \
|
|
|
+ (LPSPI_IP_SW_PATCH_VERSION_CFG != LPSPI_IP_SW_PATCH_VERSION_C))
|
|
|
+#error "Software Version Numbers of Lpspi_Ip_Cfg.h and Lpspi_Ip.c are different"
|
|
|
+#endif
|
|
|
+/*==================================================================================================
|
|
|
+* LOCAL TYPEDEFS (STRUCTURES, UNIONS, ENUMS)
|
|
|
+==================================================================================================*/
|
|
|
+/*==================================================================================================
|
|
|
+* LOCAL MACROS
|
|
|
+==================================================================================================*/
|
|
|
+#if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ /* the maximum of Major loop when Minor loop Channel Linking Disabled */
|
|
|
+ #define LPSPI_IP_DMA_MAX_ITER_CNT_U16 ((uint16)0x7FFFu)
|
|
|
+#endif
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief SR mask
|
|
|
+*/
|
|
|
+#define LPSPI_IP_SR_W1C_MASK_U32 (uint32)((uint32)LPSPI_SR_WCF_MASK | (uint32)LPSPI_SR_FCF_MASK | (uint32)LPSPI_SR_TCF_MASK | \
|
|
|
+ (uint32)LPSPI_SR_TEF_MASK | (uint32)LPSPI_SR_REF_MASK | (uint32)LPSPI_SR_DMF_MASK)
|
|
|
+
|
|
|
+/*==================================================================================================
|
|
|
+* LOCAL CONSTANTS
|
|
|
+==================================================================================================*/
|
|
|
+/*==================================================================================================
|
|
|
+* LOCAL VARIABLES
|
|
|
+==================================================================================================*/
|
|
|
+/*==================================================================================================
|
|
|
+* GLOBAL CONSTANTS
|
|
|
+==================================================================================================*/
|
|
|
+/*==================================================================================================
|
|
|
+ GLOBAL VARIABLES
|
|
|
+==================================================================================================*/
|
|
|
+#if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ #define SPI_START_SEC_VAR_CLEARED_UNSPECIFIED_NO_CACHEABLE
|
|
|
+#else
|
|
|
+ #define SPI_START_SEC_VAR_CLEARED_UNSPECIFIED
|
|
|
+#endif
|
|
|
+#include "Spi_MemMap.h"
|
|
|
+static Lpspi_Ip_StateStructureType Lpspi_Ip_axStateStructure[LPSPI_IP_NUMBER_OF_INSTANCES];
|
|
|
+#if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ #define SPI_STOP_SEC_VAR_CLEARED_UNSPECIFIED_NO_CACHEABLE
|
|
|
+#else
|
|
|
+ #define SPI_STOP_SEC_VAR_CLEARED_UNSPECIFIED
|
|
|
+#endif
|
|
|
+#include "Spi_MemMap.h"
|
|
|
+
|
|
|
+#if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ #define SPI_START_SEC_VAR_CLEARED_32_NO_CACHEABLE
|
|
|
+ #include "Spi_MemMap.h"
|
|
|
+static uint32 Lpspi_Ip_u32DiscardData;
|
|
|
+ #define SPI_STOP_SEC_VAR_CLEARED_32_NO_CACHEABLE
|
|
|
+ #include "Spi_MemMap.h"
|
|
|
+#endif
|
|
|
+
|
|
|
+#define SPI_START_SEC_VAR_CLEARED_UNSPECIFIED
|
|
|
+#include "Spi_MemMap.h"
|
|
|
+
|
|
|
+Lpspi_Ip_StateStructureType* Lpspi_Ip_apxStateStructureArray[LPSPI_INSTANCE_COUNT];
|
|
|
+
|
|
|
+#define SPI_STOP_SEC_VAR_CLEARED_UNSPECIFIED
|
|
|
+#include "Spi_MemMap.h"
|
|
|
+
|
|
|
+#define SPI_START_SEC_CONST_UNSPECIFIED
|
|
|
+#include "Spi_MemMap.h"
|
|
|
+
|
|
|
+static LPSPI_Type* const Lpspi_Ip_apxBases[LPSPI_INSTANCE_COUNT] = IP_LPSPI_BASE_PTRS;
|
|
|
+
|
|
|
+#define SPI_STOP_SEC_CONST_UNSPECIFIED
|
|
|
+#include "Spi_MemMap.h"
|
|
|
+/*==================================================================================================
|
|
|
+* LOCAL FUNCTION PROTOTYPES
|
|
|
+==================================================================================================*/
|
|
|
+#define SPI_START_SEC_CODE
|
|
|
+#include "Spi_MemMap.h"
|
|
|
+
|
|
|
+LOCAL_INLINE void Lpspi_Ip_ReadDataFromFifo(uint8 Instance, uint8 NumberOfReads);
|
|
|
+LOCAL_INLINE void Lpspi_Ip_PushDataToFifo(uint8 Instance, uint8 NumberOfWrites);
|
|
|
+static void Lpspi_Ip_ChannelFinished(uint8 Instance, boolean ErrorFlag);
|
|
|
+#if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+static void Lpspi_Ip_TransferProcessHalfDuplex(uint8 Instance);
|
|
|
+static boolean Lpspi_Ip_HalfDupleTransmitChannelComplete(uint8 Instance);
|
|
|
+LOCAL_INLINE boolean Lpspi_Ip_HalfDuplexTransmitProcess(uint8 Instance, uint8 NumberOfWrites);
|
|
|
+LOCAL_INLINE void Lpspi_Ip_HalfDuplexPushDataToFifo(uint8 Instance, uint8 NumberOfWrites);
|
|
|
+static void Lpspi_Ip_HalfDuplexPrepare(uint8 Instance, uint8* Buffer);
|
|
|
+#endif
|
|
|
+static void Lpspi_Ip_TransferProcess(uint8 Instance);
|
|
|
+static void Lpspi_TransmitTxInit(uint8 Instance,
|
|
|
+ uint8* TxBuffer,
|
|
|
+ uint8 TxFrameSize,
|
|
|
+ boolean TxLsb,
|
|
|
+ uint16 NumberOfFrames
|
|
|
+ );
|
|
|
+static void Lpspi_TransmitRxInit(uint8 Instance,
|
|
|
+ uint8* RxBuffer,
|
|
|
+ uint8 RxFrameSize,
|
|
|
+ uint16 NumberOfFrames
|
|
|
+ );
|
|
|
+#if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+static void Lpspi_Ip_TxDmaConfig(uint8 Instance);
|
|
|
+static void Lpspi_Ip_RxDmaConfig(uint8 Instance);
|
|
|
+static void Lpspi_Ip_TxDmaContinueTransfer(uint8 Instance);
|
|
|
+static void Lpspi_Ip_RxDmaContinueTransfer(uint8 Instance);
|
|
|
+
|
|
|
+#if (STD_ON == LPSPI_IP_ENABLE_DMAFASTTRANSFER_SUPPORT)
|
|
|
+static void Lpspi_Ip_DmaFastConfig(uint8 Instance, const Lpspi_Ip_FastTransferType *FastTransferCfg, uint8 NumberOfTransfer);
|
|
|
+static void Lpspi_Ip_RxDmaTcdSGConfig(uint8 Instance, uint8 TcdSgIndex, uint8 DisHwReq);
|
|
|
+static void Lpspi_Ip_RxDmaTcdSGInit(uint8 Instance);
|
|
|
+static void Lpspi_Ip_TxDmaTcdSGConfig(uint8 Instance, uint8 TcdSgIndex, uint8 DisHwReq, const uint32 *DefaultDataAddress);
|
|
|
+static void Lpspi_Ip_TxDmaTcdSGInit(uint8 Instance);
|
|
|
+static void Lpspi_Ip_CmdTxDmaTcdSGConfig(uint8 Instance, uint8 TcdSgIndex, uint32 CmdAdd, uint8 DisHwReq);
|
|
|
+#endif
|
|
|
+#endif
|
|
|
+#if (STD_ON == LPSPI_IP_ENABLE_USER_MODE_SUPPORT)
|
|
|
+static void Lpspi_Ip_SetUserAccess(uint8 Instance);
|
|
|
+static void Lpspi_Ip_SetUserAccessAllowed(uint8 Instance);
|
|
|
+#endif /* LPSPI_IP_ENABLE_USER_MODE_SUPPORT */
|
|
|
+#if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+static void Lpspi_Ip_CheckValidParameters
|
|
|
+(
|
|
|
+ const Lpspi_Ip_ExternalDeviceType *ExternalDevice,
|
|
|
+ uint16 Length
|
|
|
+);
|
|
|
+#endif
|
|
|
+#if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+static void Lpspi_Ip_TxDmaFinishTransfer
|
|
|
+(
|
|
|
+ const uint8 Instance
|
|
|
+);
|
|
|
+#endif /* (STD_ON == LPSPI_IP_DMA_USED) */
|
|
|
+/*==================================================================================================
|
|
|
+* LOCAL FUNCTIONS
|
|
|
+==================================================================================================*/
|
|
|
+/**
|
|
|
+* @brief This function will read Data from RX FIFO.
|
|
|
+* @details This function will read Data from RX FIFO.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @param[in] NumberOfReads Number of Data can be read from RX FIFO.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+LOCAL_INLINE void Lpspi_Ip_ReadDataFromFifo(uint8 Instance, uint8 NumberOfReads)
|
|
|
+{
|
|
|
+ const LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ uint32 Data = 0u;
|
|
|
+ uint8 Index = 0u;
|
|
|
+
|
|
|
+ /* To increase the performance of transfer, we will reduce the checking conditions in per "for" statement so we should have more than 1 "for" statement here */
|
|
|
+ if (NULL_PTR != State->RxBuffer)
|
|
|
+ {
|
|
|
+ if (State->ExternalDevice->DeviceParams->FrameSize < 9u)
|
|
|
+ {
|
|
|
+ for (Index = 0; Index < NumberOfReads; Index++)
|
|
|
+ {
|
|
|
+ Data = Base->RDR;
|
|
|
+ *((uint8*)(&State->RxBuffer[State->RxIndex + Index])) = (uint8)Data;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else if (State->ExternalDevice->DeviceParams->FrameSize < 17u)
|
|
|
+ {
|
|
|
+ for (Index = 0; Index < NumberOfReads; Index++)
|
|
|
+ {
|
|
|
+ Data = Base->RDR;
|
|
|
+ *((uint16*)(&State->RxBuffer[2u * (State->RxIndex + Index)])) = (uint16)Data;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ for (Index = 0; Index < NumberOfReads; Index++)
|
|
|
+ {
|
|
|
+ Data = Base->RDR;
|
|
|
+ *((uint32*)(&State->RxBuffer[4u * (State->RxIndex + Index)])) = (uint32)Data;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ for (Index = 0; Index < NumberOfReads; Index++)
|
|
|
+ {
|
|
|
+ /* Read RDR register to discard data */
|
|
|
+ (void)Base->RDR;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ State->RxIndex += NumberOfReads;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function will push Data into TX FIFO.
|
|
|
+* @details This function will push Data into TX FIFO.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @param[in] NumberOfWrites Number of Data can be pushed to TX FIFO.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+LOCAL_INLINE void Lpspi_Ip_PushDataToFifo(uint8 Instance, uint8 NumberOfWrites)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ uint32 Data = 0u;
|
|
|
+ uint8 Index = 0u;
|
|
|
+
|
|
|
+ /* Default data will be sent if TX buffer is null */
|
|
|
+ Data = State->ExternalDevice->DeviceParams->DefaultData;
|
|
|
+ /* To increase the performance of transfer, we will reduce the checking conditions in per "for" statement so we should have more than 1 "for" statement here */
|
|
|
+ if (NULL_PTR != State->TxBuffer)
|
|
|
+ {
|
|
|
+ if (State->TxFrameSize < 9u)
|
|
|
+ {
|
|
|
+ for (Index = 0; Index < NumberOfWrites; Index++)
|
|
|
+ {
|
|
|
+ Data = *((uint8*)(&State->TxBuffer[State->TxIndex + Index]));
|
|
|
+ Base->TDR = Data;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else if (State->TxFrameSize < 17u)
|
|
|
+ {
|
|
|
+ for (Index = 0; Index < NumberOfWrites; Index++)
|
|
|
+ {
|
|
|
+ Data = *((uint16*)(&State->TxBuffer[2u * (State->TxIndex + Index)]));
|
|
|
+ Base->TDR = Data;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ for (Index = 0; Index < NumberOfWrites; Index++)
|
|
|
+ {
|
|
|
+ Data = *((uint32*)(&State->TxBuffer[4u * (State->TxIndex + Index)]));
|
|
|
+ Base->TDR = Data;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ for (Index = 0; Index < NumberOfWrites; Index++)
|
|
|
+ {
|
|
|
+ Base->TDR = Data;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ State->TxIndex += NumberOfWrites;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function will finish transfer of a channel.
|
|
|
+* @details This function will finish transfer of a channel.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @param[in] ErrorFlag Save the status of transfer error flags
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_ChannelFinished(uint8 Instance, boolean ErrorFlag)
|
|
|
+{
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+
|
|
|
+ if(TRUE == ErrorFlag)
|
|
|
+ {
|
|
|
+ State->Status = LPSPI_IP_FAULT;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->Status = LPSPI_IP_IDLE;
|
|
|
+ }
|
|
|
+ if (State->Callback != NULL_PTR)
|
|
|
+ {
|
|
|
+ if(TRUE == ErrorFlag)
|
|
|
+ {
|
|
|
+ State->Callback(Instance, LPSPI_IP_EVENT_FAULT);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->Callback(Instance, LPSPI_IP_EVENT_END_TRANSFER);
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+#if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+/**
|
|
|
+* @brief This function is called by Lpspi_Ip_IrqHandler or Lpspi_Ip_ManageBuffers. It will process transfer in interrupt mode or polling mode.
|
|
|
+* @details This function will fill Data into TX FIFO and read Data in RX FIFO fill to Rx Buffers.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_TransferProcessHalfDuplex(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ uint32 SrStatusRegister;
|
|
|
+ uint8 NumberOfWrites = 0u;
|
|
|
+ uint8 NumberOfReads = 0u;
|
|
|
+ boolean ErrorFlag = FALSE;
|
|
|
+ Lpspi_Ip_HalfDuplexType TransferType;
|
|
|
+ boolean TransferCompleted = FALSE;
|
|
|
+ uint8 NumberOfFramesTxFifo;
|
|
|
+
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ /* Read Status and clear all flags. */
|
|
|
+ SrStatusRegister = Base->SR;
|
|
|
+ Base->SR &= LPSPI_IP_SR_W1C_MASK_U32;
|
|
|
+ TransferType = State->ExternalDevice->DeviceParams->TransferType;
|
|
|
+
|
|
|
+ if ((SrStatusRegister & (LPSPI_SR_REF_MASK | LPSPI_SR_TEF_MASK)) != 0u)
|
|
|
+ {
|
|
|
+ /* mark error flag */
|
|
|
+ ErrorFlag = TRUE;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* TX of latest channel in job have done */
|
|
|
+ /* Check TX FIFO is available to fill CMD */
|
|
|
+ if((State->ExpectedFifoWrites == State->TxIndex) && (LPSPI_IP_HALF_DUPLEX_TRANSMIT == TransferType))
|
|
|
+ {
|
|
|
+ ErrorFlag = Lpspi_Ip_HalfDupleTransmitChannelComplete(Instance);
|
|
|
+ /* In Transmit only mode, masks the transfer has finish */
|
|
|
+ TransferCompleted = TRUE;
|
|
|
+ }
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_RECEIVE == TransferType)
|
|
|
+ {
|
|
|
+ /* RECEIVE DATA */
|
|
|
+ /* Read all Data available in receive HW fifo. */
|
|
|
+ NumberOfReads = (uint8)(((Base->FSR) & LPSPI_FSR_RXCOUNT_MASK) >> LPSPI_FSR_RXCOUNT_SHIFT);
|
|
|
+ if (NumberOfReads > (State->ExpectedFifoReads - State->RxIndex))
|
|
|
+ {
|
|
|
+ NumberOfReads = (uint8)(State->ExpectedFifoReads - State->RxIndex);
|
|
|
+ }
|
|
|
+ if (0u != NumberOfReads)
|
|
|
+ {
|
|
|
+ Lpspi_Ip_ReadDataFromFifo(Instance, NumberOfReads);
|
|
|
+ }
|
|
|
+ /* In slave mode, do not need push CMD */
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if (FALSE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ /* PUSH CMD */
|
|
|
+ /* Only write to TCR when TX FIFO is empty in receive mode */
|
|
|
+ if ((0u == ((Base->FSR) & LPSPI_FSR_TXCOUNT_MASK)) && (State->ExpectedFifoWrites != State->TxIndex))
|
|
|
+ {
|
|
|
+ /* Push one CMD to start transfer. */
|
|
|
+ Base->TCR = State->HalfDuplexTcrCommand;
|
|
|
+ State->TxIndex++;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* TRANSMIT */
|
|
|
+ /* Push Data until HW fifo is full or transfer is done. */
|
|
|
+ NumberOfFramesTxFifo = (uint8)(((Base->FSR) & LPSPI_FSR_TXCOUNT_MASK) >> LPSPI_FSR_TXCOUNT_SHIFT);
|
|
|
+ NumberOfWrites = LPSPI_IP_FIFO_SIZE_U8 - NumberOfFramesTxFifo;
|
|
|
+ if((0u != NumberOfWrites) && (State->ExpectedFifoWrites != State->TxIndex))
|
|
|
+ {
|
|
|
+ Lpspi_Ip_HalfDuplexPushDataToFifo(Instance, NumberOfWrites);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /* End of transfer */
|
|
|
+ if(
|
|
|
+ ((LPSPI_IP_HALF_DUPLEX_TRANSMIT == TransferType) && (TRUE == TransferCompleted)) ||
|
|
|
+ ((State->RxIndex == State->ExpectedFifoReads) && (LPSPI_IP_HALF_DUPLEX_RECEIVE == TransferType)) ||
|
|
|
+ (TRUE == ErrorFlag)
|
|
|
+ )
|
|
|
+ {
|
|
|
+ /* Disable interrupts */
|
|
|
+ Base->IER = 0u;
|
|
|
+ Lpspi_Ip_ChannelFinished(Instance, ErrorFlag);
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function to complete TX channel in HD mode.
|
|
|
+* @details This function to complete TX channel in HD mode.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+static boolean Lpspi_Ip_HalfDupleTransmitChannelComplete(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ const Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ uint32 TimeoutTicks = OsIf_MicrosToTicks(LPSPI_IP_HALF_DUPLEX_TIMEOUT_COUNTER, LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ uint32 CurrentTicks = 0u; /* initialize current counter */
|
|
|
+ uint32 ElapsedTicks = 0u; /* elapsed will give timeout */
|
|
|
+ boolean TimeOutError = FALSE;
|
|
|
+
|
|
|
+ /* TX of latest channel in job have done */
|
|
|
+ if (FALSE == State->KeepCs)
|
|
|
+ {
|
|
|
+ /* No need clear for slave mode */
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if (TRUE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ /* Clear CS */
|
|
|
+ Base->TCR &= ~(LPSPI_TCR_CONT_MASK | LPSPI_TCR_CONTC_MASK);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if ((TRUE == State->NextChannelIsRX) || (FALSE == State->KeepCs))
|
|
|
+ {
|
|
|
+ /* If this is latest channel, waiting for all frames in TXFIFO have moved out before finish transfer for next job */
|
|
|
+ /* If next channel is in receive mode, waiting for TX of this channel to complete before start next channel */
|
|
|
+ while ((((Base->FSR) & LPSPI_FSR_TXCOUNT_MASK) != 0u) && (ElapsedTicks < TimeoutTicks))
|
|
|
+ {
|
|
|
+ CurrentTicks = OsIf_GetCounter(LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ ElapsedTicks += OsIf_GetElapsed(&CurrentTicks, LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ }
|
|
|
+ if(ElapsedTicks >= TimeoutTicks)
|
|
|
+ {
|
|
|
+ /* mark error flag */
|
|
|
+ TimeOutError = TRUE;
|
|
|
+ /* Clear TX FIFO */
|
|
|
+ Base->CR |= LPSPI_CR_RTF_MASK;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return TimeOutError;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function will process to transmit data.
|
|
|
+* @details This function will process to transmit data.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+LOCAL_INLINE boolean Lpspi_Ip_HalfDuplexTransmitProcess(uint8 Instance, uint8 NumberOfWrites)
|
|
|
+{
|
|
|
+ const Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ boolean TransferCompleted = FALSE;
|
|
|
+
|
|
|
+ if(State->ExpectedFifoWrites != State->TxIndex)
|
|
|
+ {
|
|
|
+ Lpspi_Ip_HalfDuplexPushDataToFifo(Instance, NumberOfWrites);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* At least 1 TX FIFO slot must be available at here to make sure that driver can push one more CMD to TX FIFO to clear CONT at the end of latest channel in a job */
|
|
|
+ if((FALSE == State->KeepCs))
|
|
|
+ {
|
|
|
+ /* Clear CS if there is request to clear CONT bit, make sure that is having at least 1 entry TX FIFO is available to fill CMD by check TDF flag */
|
|
|
+ if(0u != (Base->TCR & LPSPI_TCR_CONT_MASK))
|
|
|
+ {
|
|
|
+ Base->TCR &= ~(LPSPI_TCR_CONT_MASK | LPSPI_TCR_CONTC_MASK);
|
|
|
+ /* Set transfer flag is done to break from while loop to finish transfer */
|
|
|
+ }
|
|
|
+ }
|
|
|
+ /* Transfer is finished */
|
|
|
+ TransferCompleted = TRUE;
|
|
|
+ }
|
|
|
+ return TransferCompleted;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function will perform Transmit data in half duplex mode.
|
|
|
+* @details This function will perform Transmit data in half duplex mode.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+LOCAL_INLINE void Lpspi_Ip_HalfDuplexPushDataToFifo(uint8 Instance, uint8 NumberOfWrites)
|
|
|
+{
|
|
|
+ const Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ uint8 WriteToFifoTime = NumberOfWrites;
|
|
|
+
|
|
|
+ /* Limits to remaining frames. */
|
|
|
+ if (NumberOfWrites > (State->ExpectedFifoWrites - State->TxIndex))
|
|
|
+ {
|
|
|
+ WriteToFifoTime = (uint8)(State->ExpectedFifoWrites - State->TxIndex);
|
|
|
+ }
|
|
|
+ /* Push Data into TX FIFO */
|
|
|
+ Lpspi_Ip_PushDataToFifo(Instance, WriteToFifoTime);
|
|
|
+}
|
|
|
+/**
|
|
|
+* @brief This function will prepare to transfer in half duplex mode.
|
|
|
+* @details This function will prepare to transfer in half duplex mode.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @param[in] Buffer Transfer buffer.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_HalfDuplexPrepare(uint8 Instance, uint8* Buffer)
|
|
|
+{
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ uint8 LsbWriteValue = State->TxLsb ? 1U : 0U;
|
|
|
+
|
|
|
+ /* In half duplex mode, CFGR1 register should be updated to configure PCSCFG, OUTCFG, PINCFG fields. This bit fields should be reset before write new value */
|
|
|
+ Base->CFGR1 &= ~(LPSPI_CFGR1_PCSCFG_MASK | LPSPI_CFGR1_OUTCFG_MASK | LPSPI_CFGR1_PINCFG_MASK);
|
|
|
+ Base->CFGR1 |= State->ExternalDevice->HalfDuplexCfgr1;
|
|
|
+ /* This variable will be used only in polling mode, It do not determine the number of frames in TX FIFO in half mode. it only makes sure that in Slave receive mode, TDF is set but it will not be used to execute process function */
|
|
|
+ /* Set this variable to different 0 if is not in Slave Receive mode */
|
|
|
+ State->CurrentTxFifoSlot = 1u;
|
|
|
+ if(LPSPI_IP_HALF_DUPLEX_TRANSMIT == State->ExternalDevice->DeviceParams->TransferType)
|
|
|
+ {
|
|
|
+ State->TxBuffer = Buffer;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->RxBuffer = Buffer;
|
|
|
+
|
|
|
+ /* Update TCR value to using in ISR function */
|
|
|
+ #if (STD_ON == LPSPI_IP_DUAL_CLOCK_MODE)
|
|
|
+ State->HalfDuplexTcrCommand = State->ExternalDevice->Tcr[State->ClockMode] | LPSPI_TCR_FRAMESZ((uint32)State->ExternalDevice->DeviceParams->FrameSize - 1u) | LPSPI_TCR_LSBF(LsbWriteValue);
|
|
|
+ #else
|
|
|
+ State->HalfDuplexTcrCommand = State->ExternalDevice->Tcr | LPSPI_TCR_FRAMESZ((uint32)State->ExternalDevice->DeviceParams->FrameSize - 1u) | LPSPI_TCR_LSBF(LsbWriteValue);
|
|
|
+ #endif
|
|
|
+ /* In Slave mode, CONT bit should be cleared */
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if(TRUE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ {
|
|
|
+ State->HalfDuplexTcrCommand &= ~LPSPI_TCR_CONT_MASK;
|
|
|
+ /* This variable will be used only in polling mode, It do not determine the number of frames in TX FIFO in half mode. it only makes sure that in Slave receive mode, TDF is set but it will not be used to execute process function */
|
|
|
+ /* Set this variable to 0 in Slave Receive mode */
|
|
|
+ State->CurrentTxFifoSlot = 0u;
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ /* Because of not supporting CS continous mode so nothing need to be done with CONT bit */
|
|
|
+ /* Mark TX FIFO */
|
|
|
+ State->HalfDuplexTcrCommand |= LPSPI_TCR_TXMSK_MASK;
|
|
|
+ }
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function is called by Lpspi_Ip_IrqHandler or Lpspi_Ip_ManageBuffers. It will process transfer in interrupt mode or polling mode.
|
|
|
+* @details This function will fill Data into TX FIFO and read Data in RX FIFO fill to Rx Buffers.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_TransferProcess(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ uint32 SrStatusRegister;
|
|
|
+ uint8 NumberOfWrites = 0u;
|
|
|
+ uint8 NumberOfReads = 0u;
|
|
|
+ boolean ErrorFlag = FALSE;
|
|
|
+
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ /* Read Status and clear all flags. */
|
|
|
+ SrStatusRegister = Base->SR;
|
|
|
+ Base->SR &= LPSPI_IP_SR_W1C_MASK_U32;
|
|
|
+
|
|
|
+ if ((SrStatusRegister & (LPSPI_SR_REF_MASK | LPSPI_SR_TEF_MASK)) != 0u)
|
|
|
+ {
|
|
|
+ /* mark error flag */
|
|
|
+ ErrorFlag = TRUE;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Read all Data available in receive HW fifo. */
|
|
|
+ NumberOfReads = (uint8)(((Base->FSR) & LPSPI_FSR_RXCOUNT_MASK) >> LPSPI_FSR_RXCOUNT_SHIFT);
|
|
|
+ if (NumberOfReads != 0u)
|
|
|
+ {
|
|
|
+ if (NumberOfReads > (State->ExpectedFifoReads - State->RxIndex))
|
|
|
+ {
|
|
|
+ NumberOfReads = (uint8)(State->ExpectedFifoReads - State->RxIndex);
|
|
|
+ }
|
|
|
+ /* If these are the first frames of this channel. Current TXFIFO slot must be plus 1 because the slot of CMD have moved out and
|
|
|
+ CurrentTxFifoSlot was minus 1 when prepare TX channel */
|
|
|
+ if (0u == State->RxIndex)
|
|
|
+ {
|
|
|
+ State->CurrentTxFifoSlot += 1u;
|
|
|
+ }
|
|
|
+ /* Read Data from RX FIFO */
|
|
|
+ Lpspi_Ip_ReadDataFromFifo(Instance, NumberOfReads);
|
|
|
+ /* Update current FIFO slots are available to fill .*/
|
|
|
+ State->CurrentTxFifoSlot += NumberOfReads;
|
|
|
+ }
|
|
|
+
|
|
|
+ /*TRANSMIT*/
|
|
|
+ /* Push Data until HW fifo is full or transfer is done. */
|
|
|
+ /* After driver code read all frames in RX FIFO, if there are still some frames in TX FIFO, at the time before driver code check number of frames available in TX FIFO to prepare
|
|
|
+ to fill TX FIFO. At that time, another interrupt occurred and preemptive current interrupt, and the time to process that interrupt is longer than the time to transfer all frames
|
|
|
+ in TX FIFO. So TX FIFO will be empty and some frames received in RX FIFO, then the program is returned from that interrupt and fill TX FIFO until full and exist SPI interrupt function.
|
|
|
+ And if there is a interrupt occurred with higher priority of SPI interrupt and the time to process that interrupt is longer than the time to transfer all frames in TX FIFO.
|
|
|
+ So, RX FIFO can be overflow due to SPI interrupt function is not serviced to read RX FIFO.
|
|
|
+ State->CurrentTxFifoSlot variable is used to hanlde number of frames are "on bus transfer". They are always less than FIFO size */
|
|
|
+ if((State->CurrentTxFifoSlot != 0u) && (State->TxDoneFlag != TRUE))
|
|
|
+ {
|
|
|
+ if(State->ExpectedFifoWrites != State->TxIndex)
|
|
|
+ {
|
|
|
+ NumberOfWrites = State->CurrentTxFifoSlot;
|
|
|
+ /* Limits to remaining frames. */
|
|
|
+ if (NumberOfWrites > (State->ExpectedFifoWrites - State->TxIndex))
|
|
|
+ {
|
|
|
+ NumberOfWrites = (uint8)(State->ExpectedFifoWrites - State->TxIndex);
|
|
|
+ }
|
|
|
+ /* Push Data into TX FIFO */
|
|
|
+ Lpspi_Ip_PushDataToFifo(Instance, NumberOfWrites);
|
|
|
+ State->CurrentTxFifoSlot -= NumberOfWrites;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ if(TRUE == State->NextTransferConfigAvailable)
|
|
|
+ {
|
|
|
+ /* Initialize next transfer */
|
|
|
+ State->ExternalDevice->DeviceParams->DefaultData = State->DefaultDataNext;
|
|
|
+ State->FirstCmd = FALSE;
|
|
|
+ Lpspi_TransmitTxInit(Instance, State->TxBufferNext, State->FrameSizeNext, State->LsbNext, State->LengthNext);
|
|
|
+ State->NextTransferConfigAvailable = FALSE;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->TxDoneFlag = TRUE;
|
|
|
+ /* Disable TX interrupt */
|
|
|
+ Base->IER &= ~LPSPI_IER_TDIE_MASK;
|
|
|
+ if((FALSE == State->KeepCs) && (0u != (Base->TCR & LPSPI_TCR_CONT_MASK)))
|
|
|
+ {
|
|
|
+ /* Clear CS */
|
|
|
+ Base->TCR &= ~(LPSPI_TCR_CONT_MASK | LPSPI_TCR_CONTC_MASK);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /* End of transfer */
|
|
|
+ if((State->RxIndex == State->ExpectedFifoReads) || (TRUE == ErrorFlag))
|
|
|
+ {
|
|
|
+ /* In slave mode. After Slave devices to finish its transfer then SPI module masks rx to avoid un-expected extra Data from Master device */
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if ( ((FALSE == State->KeepCs) || (TRUE == ErrorFlag)) && (TRUE == State->PhyUnitConfig->SlaveMode) )
|
|
|
+ {
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_12();
|
|
|
+ /* Mask Rx */
|
|
|
+ Base->TCR |= LPSPI_TCR_RXMSK(1);
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_12();
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ /* Disable interrupts */
|
|
|
+ Base->IER = 0u;
|
|
|
+ Lpspi_Ip_ChannelFinished(Instance, ErrorFlag);
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+#if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+/**
|
|
|
+* @brief This function will verify the validation of some input parameters of transmision functions.
|
|
|
+* @details This function will verify the validation of some input parameters of transmision functions.
|
|
|
+*
|
|
|
+* @param[in] ExternalDevice Pointer to the external device where data is transmitted.
|
|
|
+* @param[in] Length Number of bytes to be sent.
|
|
|
+* @return LPSPI_IP_STATUS_SUCCESS: Don't have any errors was found.
|
|
|
+* LPSPI_IP_STATUS_FAIL: Transmission command has not been accepted.
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_CheckValidParameters(
|
|
|
+ const Lpspi_Ip_ExternalDeviceType *ExternalDevice,
|
|
|
+ uint16 Length
|
|
|
+ )
|
|
|
+{
|
|
|
+
|
|
|
+ if (ExternalDevice->DeviceParams->FrameSize > 16u)
|
|
|
+ {
|
|
|
+ DevAssert((Length%4) == 0u);
|
|
|
+ }
|
|
|
+ else if (ExternalDevice->DeviceParams->FrameSize > 8u)
|
|
|
+ {
|
|
|
+ DevAssert((Length%2) == 0u);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* do nothing */
|
|
|
+ }
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+#if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+#if (STD_ON == LPSPI_IP_ENABLE_DMAFASTTRANSFER_SUPPORT)
|
|
|
+static void Lpspi_Ip_TxDmaTcdSGInit(uint8 Instance)
|
|
|
+{
|
|
|
+ const Lpspi_Ip_StateStructureType* State = (const Lpspi_Ip_StateStructureType*)Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[9u];
|
|
|
+ uint8 TcdSgIndex = 0u;
|
|
|
+
|
|
|
+ /* initialze configuration software TCD Scatter Gather for Tx DMA channel */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_SOURCE_ADDRESS;
|
|
|
+ DmaTcdList[1u].Param = DMA_IP_CH_SET_DESTINATION_ADDRESS;
|
|
|
+ DmaTcdList[2u].Param = DMA_IP_CH_SET_SOURCE_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[3u].Param = DMA_IP_CH_SET_SOURCE_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[4u].Param = DMA_IP_CH_SET_DESTINATION_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[5u].Param = DMA_IP_CH_SET_MINORLOOP_SIZE;
|
|
|
+ DmaTcdList[6u].Param = DMA_IP_CH_SET_DESTINATION_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[7u].Param = DMA_IP_CH_SET_MAJORLOOP_COUNT;
|
|
|
+ DmaTcdList[8u].Param = DMA_IP_CH_SET_CONTROL_DIS_AUTO_REQUEST;
|
|
|
+
|
|
|
+ DmaTcdList[1u].Value = 0u; /* dummy dest address write, will be updated latter according to ScatterGather to update TCR or TDR */
|
|
|
+ DmaTcdList[2u].Value = 1u; /* dummy src offset is 1 byte, will be updated latter according to frame size and transfer default Data */
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* dummy 1 byte src transfer size, will be updated latter according to frame size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* dummy 1 byte dest transfer size, will be updated latter according to frame size */
|
|
|
+ DmaTcdList[5u].Value = 1u; /* dummy bytes to transfer for each request, will be updated latter according to frame size */
|
|
|
+ DmaTcdList[6u].Value = 0u; /* no dest offset */
|
|
|
+ DmaTcdList[7u].Value = 0u; /* dummy iteration count, will be updated latter according to number of frames */
|
|
|
+ DmaTcdList[8u].Value = 1u; /* dummy disable hardware request when major loop complete, will be updated latter according to last transfer or not */
|
|
|
+ DmaTcdList[0u].Value = 0u; /* dummy src address read, will be updated latter Base on TxBuffer */
|
|
|
+
|
|
|
+ for(TcdSgIndex = 0u; TcdSgIndex < State->PhyUnitConfig->MaxNumOfFastTransfer; TcdSgIndex++)
|
|
|
+ {
|
|
|
+ /* Update software TX DMA TCD Scatter Gather */
|
|
|
+ (void)Dma_Ip_SetLogicChannelScatterGatherList(State->PhyUnitConfig->TxDmaChannel, State->PhyUnitConfig->TxDmaFastSGId[TcdSgIndex], DmaTcdList, 9u);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void Lpspi_Ip_CmdTxDmaTcdSGConfig(uint8 Instance, uint8 TcdSgIndex, uint32 CmdAdd, uint8 DisHwReq)
|
|
|
+{
|
|
|
+ const LPSPI_Type* Base = (const LPSPI_Type *)Lpspi_Ip_apxBases[Instance];
|
|
|
+ const Lpspi_Ip_StateStructureType* State = (const Lpspi_Ip_StateStructureType *)Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[9u];
|
|
|
+
|
|
|
+ /* configure TX DMA TCD Scatter Gather to update transfer command TCR */
|
|
|
+ /* No need to configure dest offset due to it are already set by Lpspi_Ip_TxDmaTcdSGInit */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_SOURCE_ADDRESS;
|
|
|
+ DmaTcdList[1u].Param = DMA_IP_CH_SET_DESTINATION_ADDRESS;
|
|
|
+ DmaTcdList[2u].Param = DMA_IP_CH_SET_SOURCE_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[3u].Param = DMA_IP_CH_SET_SOURCE_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[4u].Param = DMA_IP_CH_SET_DESTINATION_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[5u].Param = DMA_IP_CH_SET_MINORLOOP_SIZE;
|
|
|
+ DmaTcdList[6u].Param = DMA_IP_CH_SET_MAJORLOOP_COUNT;
|
|
|
+ DmaTcdList[7u].Param = DMA_IP_CH_SET_CONTROL_DIS_AUTO_REQUEST;
|
|
|
+ DmaTcdList[8u].Param = DMA_IP_CH_SET_CONTROL_EN_MAJOR_INTERRUPT;
|
|
|
+
|
|
|
+ DmaTcdList[0u].Value = CmdAdd; /* src address read */
|
|
|
+ DmaTcdList[1u].Value = (uint32)&Base->TCR; /* dest address write*/
|
|
|
+ DmaTcdList[2u].Value = 0u; /* src offset is 0 byte */
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 4u; /* bytes to transfer for each request */
|
|
|
+ DmaTcdList[6u].Value = 1u; /* iteration count */
|
|
|
+ DmaTcdList[7u].Value = DisHwReq; /* disable hardware request when major loop complete */
|
|
|
+ DmaTcdList[8u].Value = DisHwReq; /* Enable Major interrupt at the end of transfer sequence(meanning when DisHwReq = 1u) */
|
|
|
+
|
|
|
+ /* Update software TX DMA TCD Scatter Gather */
|
|
|
+ (void)Dma_Ip_SetLogicChannelScatterGatherList(State->PhyUnitConfig->TxDmaChannel, State->PhyUnitConfig->TxDmaFastSGId[TcdSgIndex], DmaTcdList, 9u);
|
|
|
+}
|
|
|
+
|
|
|
+static void Lpspi_Ip_TxDmaTcdSGConfig(uint8 Instance, uint8 TcdSgIndex, uint8 DisHwReq, const uint32 *DefaultDataAddress)
|
|
|
+{
|
|
|
+ const LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[9u];
|
|
|
+
|
|
|
+ /* Update buffers index */
|
|
|
+ State->TxIndex = State->ExpectedFifoWrites;
|
|
|
+
|
|
|
+ /* configure TX DMA TCD Scatter Gather to fill tx Data to TDR */
|
|
|
+ /* No need to configure dest offset due to it are already set by Lpspi_Ip_TxDmaTcdSGInit */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_SOURCE_ADDRESS;
|
|
|
+ DmaTcdList[1u].Param = DMA_IP_CH_SET_DESTINATION_ADDRESS;
|
|
|
+ DmaTcdList[2u].Param = DMA_IP_CH_SET_SOURCE_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[3u].Param = DMA_IP_CH_SET_SOURCE_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[4u].Param = DMA_IP_CH_SET_DESTINATION_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[5u].Param = DMA_IP_CH_SET_MINORLOOP_SIZE;
|
|
|
+ DmaTcdList[6u].Param = DMA_IP_CH_SET_MAJORLOOP_COUNT;
|
|
|
+ DmaTcdList[7u].Param = DMA_IP_CH_SET_CONTROL_DIS_AUTO_REQUEST;
|
|
|
+ DmaTcdList[8u].Param = DMA_IP_CH_SET_CONTROL_EN_MAJOR_INTERRUPT;
|
|
|
+ DmaTcdList[1u].Value = (uint32)&Base->TDR; /* dest address write*/
|
|
|
+ if (State->ExternalDevice->DeviceParams->FrameSize < 9u)
|
|
|
+ {
|
|
|
+ DmaTcdList[2u].Value = 1u; /* src offset is 1 byte */
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* 1 byte src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* 1 byte dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 1u; /* bytes to transfer for each request */
|
|
|
+ }
|
|
|
+ else if (State->ExternalDevice->DeviceParams->FrameSize < 17u)
|
|
|
+ {
|
|
|
+ DmaTcdList[2u].Value = 2u; /* src offset is 2 bytes */
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_2_BYTE; /* 2 bytes src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_2_BYTE; /* 2 bytes dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 2u; /* bytes to transfer for each request */
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ DmaTcdList[2u].Value = 4u; /* src offset is 4 bytes */
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 4u; /* bytes to transfer for each request */
|
|
|
+ }
|
|
|
+ DmaTcdList[6u].Value = State->ExpectedFifoWrites; /* iteration count */
|
|
|
+ DmaTcdList[7u].Value = DisHwReq; /* disable hardware request when major loop complete */
|
|
|
+ DmaTcdList[8u].Value = DisHwReq; /* Enable Major interrupt at the end of transfer sequence(meanning when DisHwReq = 1u) */
|
|
|
+ if (NULL_PTR == State->TxBuffer)
|
|
|
+ {
|
|
|
+ /* send default Data */
|
|
|
+ DmaTcdList[0u].Value = (uint32)DefaultDataAddress; /* src address read */
|
|
|
+ DmaTcdList[2u].Value = 0u; /* src offset is 0 byte */
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ DmaTcdList[0u].Value = (uint32)State->TxBuffer; /* src address read */
|
|
|
+ /* cast to avoid CW */
|
|
|
+ (void)DefaultDataAddress;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Update software TX DMA TCD Scatter Gather */
|
|
|
+ (void)Dma_Ip_SetLogicChannelScatterGatherList(State->PhyUnitConfig->TxDmaChannel, State->PhyUnitConfig->TxDmaFastSGId[TcdSgIndex], DmaTcdList, 9u);
|
|
|
+}
|
|
|
+
|
|
|
+static void Lpspi_Ip_RxDmaTcdSGInit(uint8 Instance)
|
|
|
+{
|
|
|
+ const LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ const Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[9u];
|
|
|
+ uint8 TcdSgIndex = 0u;
|
|
|
+
|
|
|
+ /* initialze configuration software TCD Scatter Gather for Rx DMA channel */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_SOURCE_ADDRESS;
|
|
|
+ DmaTcdList[1u].Param = DMA_IP_CH_SET_DESTINATION_ADDRESS;
|
|
|
+ DmaTcdList[2u].Param = DMA_IP_CH_SET_SOURCE_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[3u].Param = DMA_IP_CH_SET_SOURCE_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[4u].Param = DMA_IP_CH_SET_DESTINATION_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[5u].Param = DMA_IP_CH_SET_MINORLOOP_SIZE;
|
|
|
+ DmaTcdList[6u].Param = DMA_IP_CH_SET_DESTINATION_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[7u].Param = DMA_IP_CH_SET_MAJORLOOP_COUNT;
|
|
|
+ DmaTcdList[8u].Param = DMA_IP_CH_SET_CONTROL_DIS_AUTO_REQUEST;
|
|
|
+
|
|
|
+ DmaTcdList[0u].Value = (uint32)&Base->RDR; /* src address read */
|
|
|
+ DmaTcdList[2u].Value = 0u; /* no src offset */
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* dummy 1 byte src transfer size, will be updated latter Base on frame size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* dummy 1 byte dest transfer size, will be updated latter Base on frame size */
|
|
|
+ DmaTcdList[5u].Value = 1u; /* dummy 1 byte to transfer for each request, will be updated latter Base on frame size */
|
|
|
+ DmaTcdList[6u].Value = 1u; /* dummy dest offset is 1 byte, will be updated latter Base on frame size and discard info */
|
|
|
+ DmaTcdList[1u].Value = 0u; /* dummy dest address write, will be updated latter Base on RxBuffer */
|
|
|
+ DmaTcdList[7u].Value = 0u; /* dummy iteration count, will be updated latter Base on number of frames */
|
|
|
+ DmaTcdList[8u].Value = 1u; /* dummy disable hardware request when major loop complete, will be updated latter according to last transfer or not */
|
|
|
+
|
|
|
+ for(TcdSgIndex = 0u; TcdSgIndex < State->PhyUnitConfig->MaxNumOfFastTransfer; TcdSgIndex++)
|
|
|
+ {
|
|
|
+ /* Update software RX DMA TCD Scatter Gather */
|
|
|
+ (void)Dma_Ip_SetLogicChannelScatterGatherList(State->PhyUnitConfig->RxDmaChannel, State->PhyUnitConfig->RxDmaFastSGId[TcdSgIndex], DmaTcdList, 9u);
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void Lpspi_Ip_RxDmaTcdSGConfig(uint8 Instance, uint8 TcdSgIndex, uint8 DisHwReq)
|
|
|
+{
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[8u];
|
|
|
+
|
|
|
+ /* Update buffers index */
|
|
|
+ State->RxIndex = State->ExpectedFifoReads;
|
|
|
+
|
|
|
+ /* configure RX DMA TCD Scatter Gather */
|
|
|
+ /* No need to configure src address and src offset due to they are already set by Lpspi_Ip_RxDmaTcdSGInit */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_DESTINATION_ADDRESS;
|
|
|
+ DmaTcdList[1u].Param = DMA_IP_CH_SET_SOURCE_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[2u].Param = DMA_IP_CH_SET_DESTINATION_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[3u].Param = DMA_IP_CH_SET_MINORLOOP_SIZE;
|
|
|
+ DmaTcdList[4u].Param = DMA_IP_CH_SET_DESTINATION_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[5u].Param = DMA_IP_CH_SET_MAJORLOOP_COUNT;
|
|
|
+ DmaTcdList[6u].Param = DMA_IP_CH_SET_CONTROL_DIS_AUTO_REQUEST;
|
|
|
+ DmaTcdList[7u].Param = DMA_IP_CH_SET_CONTROL_EN_MAJOR_INTERRUPT;
|
|
|
+
|
|
|
+ if (State->ExternalDevice->DeviceParams->FrameSize < 9u)
|
|
|
+ {
|
|
|
+ DmaTcdList[1u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* 1 byte src transfer size */
|
|
|
+ DmaTcdList[2u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* 1 byte dest transfer size */
|
|
|
+ DmaTcdList[3u].Value = 1u; /* 1 byte to transfer for each request */
|
|
|
+ DmaTcdList[4u].Value = 1u; /* dest offset is 1 bytes */
|
|
|
+ }
|
|
|
+ else if (State->ExternalDevice->DeviceParams->FrameSize < 17u)
|
|
|
+ {
|
|
|
+ DmaTcdList[1u].Value = DMA_IP_TRANSFER_SIZE_2_BYTE; /* 2 bytes src transfer size */
|
|
|
+ DmaTcdList[2u].Value = DMA_IP_TRANSFER_SIZE_2_BYTE; /* 2 bytes dest transfer size */
|
|
|
+ DmaTcdList[3u].Value = 2u; /* 2 bytes to transfer for each request */
|
|
|
+ DmaTcdList[4u].Value = 2u; /* dest offset is 2 bytes */
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ DmaTcdList[1u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes src transfer size */
|
|
|
+ DmaTcdList[2u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes dest transfer size */
|
|
|
+ DmaTcdList[3u].Value = 4u; /* 4 bytes to transfer for each request */
|
|
|
+ DmaTcdList[4u].Value = 4u; /* dest offset is 4 bytes */
|
|
|
+ }
|
|
|
+ if (NULL_PTR == State->RxBuffer)
|
|
|
+ {
|
|
|
+ /* Discard Data */
|
|
|
+ DmaTcdList[0u].Value = (uint32)&Lpspi_Ip_u32DiscardData; /* dest address write*/
|
|
|
+ DmaTcdList[4u].Value = 0u; /* dest offset is 0 bytes */
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ DmaTcdList[0u].Value = (uint32)State->RxBuffer; /* dest address write*/
|
|
|
+ }
|
|
|
+ DmaTcdList[5u].Value = State->ExpectedFifoReads; /* iteration count */
|
|
|
+ DmaTcdList[6u].Value = DisHwReq; /* disable hardware request when major loop complete */
|
|
|
+ DmaTcdList[7u].Value = DisHwReq; /* Enable Major interrupt at the end of transfer sequence(meanning when DisHwReq = 1u) */
|
|
|
+
|
|
|
+ /* Update software RX DMA TCD Scatter Gather */
|
|
|
+ (void)Dma_Ip_SetLogicChannelScatterGatherList(State->PhyUnitConfig->RxDmaChannel, State->PhyUnitConfig->RxDmaFastSGId[TcdSgIndex], DmaTcdList, 8u);
|
|
|
+}
|
|
|
+#endif /* (STD_ON == LPSPI_IP_ENABLE_DMAFASTTRANSFER_SUPPORT) */
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function will configure hardware TCDs for the channels TX DMA, RX DMA
|
|
|
+* according to current transfer configuration. DMA channels will be started at the end of the function.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_TxDmaConfig(uint8 Instance)
|
|
|
+{
|
|
|
+ const LPSPI_Type* Base = (const LPSPI_Type*)Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[10u];
|
|
|
+ uint16 NumberDmaIterWrite = State->ExpectedFifoWrites;
|
|
|
+
|
|
|
+ /* Limits number of major count */
|
|
|
+ if (LPSPI_IP_DMA_MAX_ITER_CNT_U16 < NumberDmaIterWrite)
|
|
|
+ {
|
|
|
+ NumberDmaIterWrite = LPSPI_IP_DMA_MAX_ITER_CNT_U16;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Nothing to do */
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Update buffers index */
|
|
|
+ State->TxIndex = NumberDmaIterWrite;
|
|
|
+
|
|
|
+ /* configure TX DMA channel */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_SOURCE_ADDRESS;
|
|
|
+ DmaTcdList[1u].Param = DMA_IP_CH_SET_DESTINATION_ADDRESS;
|
|
|
+ DmaTcdList[2u].Param = DMA_IP_CH_SET_SOURCE_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[3u].Param = DMA_IP_CH_SET_SOURCE_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[4u].Param = DMA_IP_CH_SET_DESTINATION_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[5u].Param = DMA_IP_CH_SET_MINORLOOP_SIZE;
|
|
|
+ DmaTcdList[6u].Param = DMA_IP_CH_SET_DESTINATION_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[7u].Param = DMA_IP_CH_SET_MAJORLOOP_COUNT;
|
|
|
+ DmaTcdList[8u].Param = DMA_IP_CH_SET_CONTROL_DIS_AUTO_REQUEST;
|
|
|
+ DmaTcdList[9u].Param = DMA_IP_CH_SET_DESTINATION_SIGNED_LAST_ADDR_ADJ;
|
|
|
+#if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ /* In Half duplex receive mode. TX DMA will be used to push CMD to TXFIFO to start transfer */
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_RECEIVE == State->ExternalDevice->DeviceParams->TransferType)
|
|
|
+ {
|
|
|
+ DmaTcdList[0u].Value = (uint32)&State->HalfDuplexTcrCommand; /* src address read */
|
|
|
+ DmaTcdList[1u].Value = (uint32)&Base->TCR; /* dest address write*/
|
|
|
+ DmaTcdList[2u].Value = 0u; /* src offset is 0 bytes */
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 4u; /* bytes to transfer for each request */
|
|
|
+ }
|
|
|
+ else
|
|
|
+#endif
|
|
|
+ {
|
|
|
+ DmaTcdList[1u].Value = (uint32)&Base->TDR; /* dest address write*/
|
|
|
+ if (State->TxFrameSize < 9u)
|
|
|
+ {
|
|
|
+ DmaTcdList[2u].Value = 1u; /* src offset is 1 byte */
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* 1 byte src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* 1 byte dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 1u; /* bytes to transfer for each request */
|
|
|
+ }
|
|
|
+ else if (State->TxFrameSize < 17u)
|
|
|
+ {
|
|
|
+ DmaTcdList[2u].Value = 2u; /* src offset is 2 bytes */
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_2_BYTE; /* 2 bytes src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_2_BYTE; /* 2 bytes dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 2u; /* bytes to transfer for each request */
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ DmaTcdList[2u].Value = 4u; /* src offset is 4 bytes */
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 4u; /* bytes to transfer for each request */
|
|
|
+ }
|
|
|
+ if (NULL_PTR == State->TxBuffer)
|
|
|
+ {
|
|
|
+ /* send default Data */
|
|
|
+ DmaTcdList[0u].Value = (uint32)&State->ExternalDevice->DeviceParams->DefaultData; /* src address read */
|
|
|
+ DmaTcdList[2u].Value = 0u; /* src offset is 0 byte */
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ DmaTcdList[0u].Value = (uint32)State->TxBuffer; /* src address read */
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ DmaTcdList[6u].Value = 0u; /* no dest offset */
|
|
|
+ DmaTcdList[7u].Value = NumberDmaIterWrite; /* iteration count */
|
|
|
+ DmaTcdList[8u].Value = 1u; /* disable hardware request when major loop complete */
|
|
|
+
|
|
|
+ /* Set DESTINATION_SIGNED_LAST_ADDR_ADJ = 0 to avoid the case it still stored from previous TCD Scatter Gather */
|
|
|
+ DmaTcdList[9u].Value = 0u; /* No adjust DADD when major loop completed */
|
|
|
+ /* write TCD for TX DMA channel */
|
|
|
+ (void)Dma_Ip_SetLogicChannelTransferList(State->PhyUnitConfig->TxDmaChannel, DmaTcdList, 10u);
|
|
|
+
|
|
|
+ /* Enable TX DMA HW request */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->TxDmaChannel, DMA_IP_CH_SET_HARDWARE_REQUEST);
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function will configure hardware TCDs for the channels TX DMA, RX DMA
|
|
|
+* according to current transfer configuration. DMA channels will be started at the end of the function.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_RxDmaConfig(uint8 Instance)
|
|
|
+{
|
|
|
+ const LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[10u];
|
|
|
+ uint16 NumberDmaIterRead = State->ExpectedFifoReads;
|
|
|
+
|
|
|
+ /* Limits number of major count */
|
|
|
+ if (LPSPI_IP_DMA_MAX_ITER_CNT_U16 < NumberDmaIterRead)
|
|
|
+ {
|
|
|
+ NumberDmaIterRead = LPSPI_IP_DMA_MAX_ITER_CNT_U16;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Nothing to do */
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Update buffers index */
|
|
|
+ State->RxIndex = NumberDmaIterRead;
|
|
|
+
|
|
|
+ /* configure RX DMA channel */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_SOURCE_ADDRESS;
|
|
|
+ DmaTcdList[1u].Param = DMA_IP_CH_SET_DESTINATION_ADDRESS;
|
|
|
+ DmaTcdList[2u].Param = DMA_IP_CH_SET_SOURCE_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[3u].Param = DMA_IP_CH_SET_SOURCE_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[4u].Param = DMA_IP_CH_SET_DESTINATION_TRANSFER_SIZE;
|
|
|
+ DmaTcdList[5u].Param = DMA_IP_CH_SET_MINORLOOP_SIZE;
|
|
|
+ DmaTcdList[6u].Param = DMA_IP_CH_SET_DESTINATION_SIGNED_OFFSET;
|
|
|
+ DmaTcdList[7u].Param = DMA_IP_CH_SET_MAJORLOOP_COUNT;
|
|
|
+ DmaTcdList[8u].Param = DMA_IP_CH_SET_CONTROL_DIS_AUTO_REQUEST;
|
|
|
+ DmaTcdList[9u].Param = DMA_IP_CH_SET_DESTINATION_SIGNED_LAST_ADDR_ADJ;
|
|
|
+
|
|
|
+ DmaTcdList[0u].Value = (uint32)&Base->RDR; /* src address read */
|
|
|
+ DmaTcdList[2u].Value = 0u; /* no src offset */
|
|
|
+ if (State->ExternalDevice->DeviceParams->FrameSize < 9u)
|
|
|
+ {
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* 1 byte src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_1_BYTE; /* 1 byte dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 1u; /* 1 byte to transfer for each request */
|
|
|
+ DmaTcdList[6u].Value = 1u; /* dest offset is 1 bytes */
|
|
|
+ }
|
|
|
+ else if (State->ExternalDevice->DeviceParams->FrameSize < 17u)
|
|
|
+ {
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_2_BYTE; /* 2 bytes src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_2_BYTE; /* 2 bytes dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 2u; /* 2 bytes to transfer for each request */
|
|
|
+ DmaTcdList[6u].Value = 2u; /* dest offset is 2 bytes */
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ DmaTcdList[3u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes src transfer size */
|
|
|
+ DmaTcdList[4u].Value = DMA_IP_TRANSFER_SIZE_4_BYTE; /* 4 bytes dest transfer size */
|
|
|
+ DmaTcdList[5u].Value = 4u; /* 4 bytes to transfer for each request */
|
|
|
+ DmaTcdList[6u].Value = 4u; /* dest offset is 4 bytes */
|
|
|
+ }
|
|
|
+ if (NULL_PTR == State->RxBuffer)
|
|
|
+ {
|
|
|
+ /* Discard Data */
|
|
|
+ DmaTcdList[1u].Value = (uint32)&Lpspi_Ip_u32DiscardData; /* dest address write*/
|
|
|
+ DmaTcdList[6u].Value = 0u; /* dest offset is 0 bytes */
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ DmaTcdList[1u].Value = (uint32)State->RxBuffer; /* dest address write*/
|
|
|
+ }
|
|
|
+ DmaTcdList[7u].Value = NumberDmaIterRead; /* iteration count */
|
|
|
+ DmaTcdList[8u].Value = 1u; /* disable hardware request when major loop complete */
|
|
|
+ /* Set DESTINATION_SIGNED_LAST_ADDR_ADJ = 0 to avoid the case it still stored from previous TCD Scatter Gather */
|
|
|
+ DmaTcdList[9u].Value = 0u; /* No adjust DADD when major loop completed */
|
|
|
+ /* write TCD for RX DMA channel */
|
|
|
+ (void)Dma_Ip_SetLogicChannelTransferList(State->PhyUnitConfig->RxDmaChannel, DmaTcdList, 10u);
|
|
|
+
|
|
|
+ /* Enable RX DMA HW request */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->RxDmaChannel, DMA_IP_CH_SET_HARDWARE_REQUEST);
|
|
|
+}
|
|
|
+
|
|
|
+static void Lpspi_Ip_TxDmaContinueTransfer(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[1u];
|
|
|
+ uint16 NumberDmaIterWrite = State->ExpectedFifoWrites - State->TxIndex;
|
|
|
+
|
|
|
+ /* Limits number of major count */
|
|
|
+ if (LPSPI_IP_DMA_MAX_ITER_CNT_U16 < NumberDmaIterWrite)
|
|
|
+ {
|
|
|
+ NumberDmaIterWrite = LPSPI_IP_DMA_MAX_ITER_CNT_U16;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Nothing to do */
|
|
|
+ }
|
|
|
+
|
|
|
+ State->TxIndex += NumberDmaIterWrite;
|
|
|
+
|
|
|
+ /* Update TX DMA channel */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_MAJORLOOP_COUNT;
|
|
|
+ DmaTcdList[0u].Value = NumberDmaIterWrite; /* iteration count */
|
|
|
+ (void)Dma_Ip_SetLogicChannelTransferList(State->PhyUnitConfig->TxDmaChannel, DmaTcdList, 1u);
|
|
|
+
|
|
|
+ /* Disable TX DMA request to avoid overflow because RX DMA needs time to be initialized for next transfer,
|
|
|
+ TX DMA request will be enabled later when RX DMA complete by Lpspi_Ip_RxDmaContinueTransfer. */
|
|
|
+ Base->DER &= ~LPSPI_DER_TDDE_MASK;
|
|
|
+ /* Enable TX DMA HW request for TX DMA channel */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->TxDmaChannel, DMA_IP_CH_SET_HARDWARE_REQUEST);
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if(TRUE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ {
|
|
|
+ /* Push first frame to TX FIFO to ensure that TX FIFO is not empty and CS can be kept in the case of CS continue */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->TxDmaChannel, DMA_IP_CH_SET_SOFTWARE_REQUEST);
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+}
|
|
|
+
|
|
|
+static void Lpspi_Ip_RxDmaContinueTransfer(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[1u];
|
|
|
+ uint16 NumberDmaIterRead = State->ExpectedFifoReads - State->RxIndex;
|
|
|
+
|
|
|
+ /* Limits number of major count */
|
|
|
+ if (LPSPI_IP_DMA_MAX_ITER_CNT_U16 < NumberDmaIterRead)
|
|
|
+ {
|
|
|
+ NumberDmaIterRead = LPSPI_IP_DMA_MAX_ITER_CNT_U16;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Nothing to do */
|
|
|
+ }
|
|
|
+
|
|
|
+ State->RxIndex += NumberDmaIterRead;
|
|
|
+
|
|
|
+ /* Update RX DMA channel */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_MAJORLOOP_COUNT;
|
|
|
+ DmaTcdList[0u].Value = NumberDmaIterRead; /* iteration count */
|
|
|
+ (void)Dma_Ip_SetLogicChannelTransferList(State->PhyUnitConfig->RxDmaChannel, DmaTcdList, 1u);
|
|
|
+
|
|
|
+ /* Enable DMA HW request for RX DMA channel */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->RxDmaChannel, DMA_IP_CH_SET_HARDWARE_REQUEST);
|
|
|
+ /* Enable TX DMA request due to it is disabled in Lpspi_Ip_TxDmaContinueTransfer */
|
|
|
+ Base->DER |= LPSPI_DER_TDDE_MASK;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function will finish channel transmission via DMA.
|
|
|
+* @details This function will finish channel transmission via DMA.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @return void
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_TxDmaFinishTransfer(const uint8 Instance)
|
|
|
+{
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ Lpspi_Ip_HalfDuplexType TransferType;
|
|
|
+ uint32 CurrentTicks = 0u; /* initialize current counter */
|
|
|
+ uint32 ElapsedTicks = 0u; /* elapsed will give timeout */
|
|
|
+ uint32 TimeoutTicks = OsIf_MicrosToTicks(LPSPI_IP_HALF_DUPLEX_TIMEOUT_COUNTER, LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ #endif
|
|
|
+ LPSPI_Type *Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType *State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ TransferType = State->ExternalDevice->DeviceParams->TransferType;
|
|
|
+ if (((TRUE == State->NextChannelIsRX) && (LPSPI_IP_HALF_DUPLEX_TRANSMIT == TransferType)) || (FALSE == State->KeepCs))
|
|
|
+ #else
|
|
|
+ if (FALSE == State->KeepCs)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ if (FALSE == State->KeepCs)
|
|
|
+ {
|
|
|
+ /* Clear CS, in slave mode CONT bit is always 0 so nothing will be done. */
|
|
|
+ if (0u != (Base->TCR & LPSPI_TCR_CONT_MASK))
|
|
|
+ {
|
|
|
+ Base->TCR &= ~(LPSPI_TCR_CONT_MASK | LPSPI_TCR_CONTC_MASK);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_TRANSMIT == TransferType)
|
|
|
+ {
|
|
|
+ /* If this is latest channel, waiting for all frames in TXFIFO have moved out before finish transfer for next job */
|
|
|
+ /* If next channel is in receive mode, waiting for TX of this channel to complete before start next channel */
|
|
|
+ while ((((Base->FSR) & LPSPI_FSR_TXCOUNT_MASK) != 0u) && (ElapsedTicks < TimeoutTicks))
|
|
|
+ {
|
|
|
+ CurrentTicks = OsIf_GetCounter(LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ ElapsedTicks += OsIf_GetElapsed(&CurrentTicks, LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ }
|
|
|
+ if(ElapsedTicks >= TimeoutTicks)
|
|
|
+ {
|
|
|
+ /* Clear TX FIFO */
|
|
|
+ Base->CR |= LPSPI_CR_RTF_MASK;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* In Half duplex trasnmit mode, we do not need to care about received datas of current channel so do not need to configure next channel here */
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ if (LPSPI_IP_FULL_DUPLEX == TransferType)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ if (TRUE == State->NextTransferConfigAvailable)
|
|
|
+ {
|
|
|
+ /* Initialize next transfer */
|
|
|
+ State->ExternalDevice->DeviceParams->DefaultData = State->DefaultDataNext;
|
|
|
+ State->FirstCmd = FALSE;
|
|
|
+ Lpspi_TransmitTxInit(Instance, State->TxBufferNext, State->FrameSizeNext, State->LsbNext, State->LengthNext);
|
|
|
+ State->NextTransferConfigAvailable = FALSE;
|
|
|
+ /* Disable TX DMA request to avoid overflow because RX DMA needs time to be initialized for next transfer,
|
|
|
+ TX DMA request will be enabled later when RX DMA completed and next transfer initialized by Lpspi_Ip_AsyncTransmit. */
|
|
|
+ Base->DER &= ~LPSPI_DER_TDDE_MASK;
|
|
|
+ Lpspi_Ip_TxDmaConfig(Instance);
|
|
|
+ /* Push first frame to TX FIFO to ensure that previous received frame will be pushed to RX FIFO and RX DMA can be completed in the case of CS continue */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->TxDmaChannel, DMA_IP_CH_SET_SOFTWARE_REQUEST);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Set flag next transfer has been completed */
|
|
|
+ State->NextTransferDone = TRUE;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/*==================================================================================================
|
|
|
+* GLOBAL FUNCTIONS
|
|
|
+==================================================================================================*/
|
|
|
+/**
|
|
|
+* @brief This function will process TX DMA transfer complete interrupt.
|
|
|
+* @details This function will process continue transfer or end of transfer via TX DMA.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @return void
|
|
|
+* @implements Lpspi_Ip_IrqTxDmaHandler_Activity
|
|
|
+*/
|
|
|
+void Lpspi_Ip_IrqTxDmaHandler(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ uint32 SrStatusRegister = 0u;
|
|
|
+ boolean ErrorFlag = FALSE;
|
|
|
+#if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ boolean EndOfTransferFlag = FALSE;
|
|
|
+ Lpspi_Ip_HalfDuplexType TransferType;
|
|
|
+ Lpspi_Ip_EventType EventState = LPSPI_IP_EVENT_FAULT;
|
|
|
+#endif
|
|
|
+
|
|
|
+ if (NULL_PTR != State)
|
|
|
+ {
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ TransferType = State->ExternalDevice->DeviceParams->TransferType;
|
|
|
+ #endif
|
|
|
+ /* Read Status an clear all flags. */
|
|
|
+ SrStatusRegister = Base->SR;
|
|
|
+ Base->SR &= LPSPI_IP_SR_W1C_MASK_U32;
|
|
|
+
|
|
|
+ if ((SrStatusRegister & (LPSPI_SR_REF_MASK | LPSPI_SR_TEF_MASK)) != 0u)
|
|
|
+ {
|
|
|
+ /* mark error flag */
|
|
|
+ ErrorFlag = TRUE;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ if (State->ExpectedFifoWrites != State->TxIndex)
|
|
|
+ {
|
|
|
+ /* Transfer is not finished => update TX pointers */
|
|
|
+ Lpspi_Ip_TxDmaContinueTransfer(Instance);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Mask EndOfTransferFlag to TRUE to finish transfer if transmit only mode */
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ TransferType = State->ExternalDevice->DeviceParams->TransferType;
|
|
|
+ /* In Transmit only mode, the transfer can be finished after TX have done transfer */
|
|
|
+ /* In Receive only mode, the trasfer need to wait for RX finish */
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_TRANSMIT == TransferType)
|
|
|
+ {
|
|
|
+ EndOfTransferFlag = TRUE;
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ Lpspi_Ip_TxDmaFinishTransfer(Instance);
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ if ((TRUE == ErrorFlag) || (TRUE == EndOfTransferFlag))
|
|
|
+ #else
|
|
|
+ if (TRUE == ErrorFlag)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ /* In slave mode. After Slave device finish its transfer or Transfer errors apprear then SPI module masks Rx to avoid un-expected extra Data from Master device */
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if(TRUE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ {
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_13();
|
|
|
+ /* Mask Rx */
|
|
|
+ Base->TCR |= LPSPI_TCR_RXMSK(1);
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_13();
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ /* Disable DMA requests. */
|
|
|
+ Base->DER = 0u;
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_TRANSMIT == TransferType)
|
|
|
+ {
|
|
|
+ if(TRUE == ErrorFlag)
|
|
|
+ {
|
|
|
+ State->Status = LPSPI_IP_FAULT;
|
|
|
+ EventState = LPSPI_IP_EVENT_FAULT;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->Status = LPSPI_IP_IDLE;
|
|
|
+ EventState = LPSPI_IP_EVENT_END_TRANSFER;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (NULL_PTR != State->Callback)
|
|
|
+ {
|
|
|
+ State->Callback(Instance, EventState);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ /* Disable RX DMA HW request because may RX DMA is not completed, no need to apply for TX DMA due to DMA HW request is cleared automatically */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->RxDmaChannel, DMA_IP_CH_CLEAR_HARDWARE_REQUEST);
|
|
|
+ State->Status = LPSPI_IP_FAULT;
|
|
|
+
|
|
|
+ if (State->Callback != NULL_PTR)
|
|
|
+ {
|
|
|
+ State->Callback(Instance, LPSPI_IP_EVENT_FAULT);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Driver is initialized but there was no poll request*/
|
|
|
+ /* nothing to do */
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function will process RX DMA transfer complete interrupt.
|
|
|
+* @details This function will process continue transfer or end of transfer via RX DMA.
|
|
|
+*
|
|
|
+* @param[in] Instance Index of the hardware instance.
|
|
|
+* @return void
|
|
|
+* @implements Lpspi_Ip_IrqRxDmaHandler_Activity
|
|
|
+*/
|
|
|
+void Lpspi_Ip_IrqRxDmaHandler(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ uint32 SrStatusRegister = 0u;
|
|
|
+ boolean ErrorFlag = FALSE;
|
|
|
+ boolean EndOfTransferFlag = FALSE;
|
|
|
+
|
|
|
+ if (NULL_PTR != State)
|
|
|
+ {
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ /* Read Status an clear all flags. */
|
|
|
+ SrStatusRegister = Base->SR;
|
|
|
+ Base->SR &= LPSPI_IP_SR_W1C_MASK_U32;
|
|
|
+
|
|
|
+ if ((SrStatusRegister & (LPSPI_SR_REF_MASK | LPSPI_SR_TEF_MASK)) != 0u)
|
|
|
+ {
|
|
|
+ /* mark error flag */
|
|
|
+ ErrorFlag = TRUE;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ if (State->ExpectedFifoReads != State->RxIndex)
|
|
|
+ {
|
|
|
+ /* Transfer is not finished => update RX pointers */
|
|
|
+ Lpspi_Ip_RxDmaContinueTransfer(Instance);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ EndOfTransferFlag = TRUE;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ if ((TRUE == EndOfTransferFlag) || (TRUE == ErrorFlag))
|
|
|
+ {
|
|
|
+ /* In slave mode. After Slave device finish its transfer or Transfer errors apprear then SPI module masks Rx to avoid un-expected extra Data from Master device */
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if(TRUE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ {
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_14();
|
|
|
+ /* Mask Rx */
|
|
|
+ Base->TCR |= LPSPI_TCR_RXMSK(1);
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_14();
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ /* Disable DMA requests. */
|
|
|
+ Base->DER = 0u;
|
|
|
+ if(TRUE == ErrorFlag)
|
|
|
+ {
|
|
|
+ State->Status = LPSPI_IP_FAULT;
|
|
|
+ /* Disable TX DMA HW request because may TX DMA is not completed, no need to apply for RX DMA due to DMA HW request is cleared automatically */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->TxDmaChannel, DMA_IP_CH_CLEAR_HARDWARE_REQUEST);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->Status = LPSPI_IP_IDLE;
|
|
|
+ }
|
|
|
+ if (State->Callback != NULL_PTR)
|
|
|
+ {
|
|
|
+ if(TRUE == ErrorFlag)
|
|
|
+ {
|
|
|
+ State->Callback(Instance, LPSPI_IP_EVENT_FAULT);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->Callback(Instance, LPSPI_IP_EVENT_END_TRANSFER);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Driver is initialized but there was no poll request*/
|
|
|
+ /* nothing to do */
|
|
|
+ }
|
|
|
+}
|
|
|
+#endif /* (STD_ON == LPSPI_IP_DMA_USED) */
|
|
|
+
|
|
|
+#if (STD_ON == LPSPI_IP_ENABLE_USER_MODE_SUPPORT)
|
|
|
+/**
|
|
|
+* @brief This function will set UAA bit in REG_PROT for SPI unit
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_SetUserAccess(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+
|
|
|
+ SET_USER_ACCESS_ALLOWED((uint32)Base,LPSPI_IP_PROT_MEM_U32);
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function will enable writing all SPI registers under protection in User mode by configuring REG_PROT
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_SetUserAccessAllowed(uint8 Instance)
|
|
|
+{
|
|
|
+ OsIf_Trusted_Call1param(Lpspi_Ip_SetUserAccess, Instance);
|
|
|
+}
|
|
|
+#endif /* LPSPI_IP_ENABLE_USER_MODE_SUPPORT */
|
|
|
+/*================================================================================================*/
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_Init(const Lpspi_Ip_ConfigType *PhyUnitConfigPtr)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base;
|
|
|
+ Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+ uint8 Instance = 0u;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(PhyUnitConfigPtr != NULL_PTR);
|
|
|
+ #endif
|
|
|
+ Instance = PhyUnitConfigPtr->Instance;
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(State == NULL_PTR);
|
|
|
+ #endif
|
|
|
+ Lpspi_Ip_apxStateStructureArray[Instance] = &Lpspi_Ip_axStateStructure[PhyUnitConfigPtr->StateIndex];
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ State->PhyUnitConfig = PhyUnitConfigPtr;
|
|
|
+ /* enable in debug mode to ensure CS will be kept when CPU halts at breakpoint */
|
|
|
+ Base->CR = PhyUnitConfigPtr->Cr | LPSPI_CR_DBGEN_MASK;
|
|
|
+ Base->CFGR1 = PhyUnitConfigPtr->Cfgr1;
|
|
|
+ /* Set TX WATER. it will be set again in DMA mode */
|
|
|
+ Base->FCR = LPSPI_FCR_TXWATER((uint32)LPSPI_IP_FIFO_SIZE_U8 - (uint32)1u);
|
|
|
+ #if (STD_ON == LPSPI_IP_DUAL_CLOCK_MODE)
|
|
|
+ State->ClockMode = LPSPI_IP_NORMAL_CLOCK;
|
|
|
+ #endif
|
|
|
+ State->KeepCs = FALSE;
|
|
|
+ State->FirstCmd = TRUE;
|
|
|
+ #if ((STD_ON == LPSPI_IP_DMA_USED) && (STD_ON == LPSPI_IP_ENABLE_DMAFASTTRANSFER_SUPPORT))
|
|
|
+ Lpspi_Ip_TxDmaTcdSGInit(Instance);
|
|
|
+ Lpspi_Ip_RxDmaTcdSGInit(Instance);
|
|
|
+ #endif
|
|
|
+ /* set State to idle */
|
|
|
+ State->Status = LPSPI_IP_IDLE;
|
|
|
+ (void)Lpspi_Ip_UpdateTransferMode(Instance, PhyUnitConfigPtr->TransferMode);
|
|
|
+
|
|
|
+ /* Enable SPI module */
|
|
|
+ Base->CR |= LPSPI_CR_MEN_MASK;
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+/*================================================================================================*/
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_DeInit(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base;
|
|
|
+ const Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(Instance < LPSPI_INSTANCE_COUNT);
|
|
|
+ #endif
|
|
|
+ Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != State);
|
|
|
+ #endif
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Use reset hardware feature. */
|
|
|
+ Base->CR |= LPSPI_CR_RST(1u);
|
|
|
+ Base->CR = 0;
|
|
|
+
|
|
|
+ Lpspi_Ip_apxStateStructureArray[Instance] = NULL_PTR;
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+
|
|
|
+/*================================================================================================*/
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_SyncTransmit(
|
|
|
+ const Lpspi_Ip_ExternalDeviceType *ExternalDevice,
|
|
|
+ uint8 *TxBuffer,
|
|
|
+ uint8 *RxBuffer,
|
|
|
+ uint16 Length,
|
|
|
+ uint32 TimeOut
|
|
|
+ )
|
|
|
+{
|
|
|
+ LPSPI_Type *Base;
|
|
|
+ Lpspi_Ip_StateStructureType *State;
|
|
|
+ uint8 NumberOfWrites, NumberOfReads;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+ uint32 TimeoutTicks = OsIf_MicrosToTicks(TimeOut, LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ uint32 CurrentTicks = 0u; /* initialize current counter */
|
|
|
+ uint32 ElapsedTicks = 0u; /* elapsed will give timeout */
|
|
|
+ uint8 Instance = 0u;
|
|
|
+ boolean TxDoneFlag = FALSE;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(ExternalDevice != NULL_PTR);
|
|
|
+ DevAssert(0u != Length);
|
|
|
+ DevAssert(0u != TimeOut);
|
|
|
+ Lpspi_Ip_CheckValidParameters(ExternalDevice, Length);
|
|
|
+ #endif
|
|
|
+ Instance = ExternalDevice->Instance;
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(State != NULL_PTR);
|
|
|
+ #endif
|
|
|
+
|
|
|
+ Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_08();
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_08();
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ /* Clear some bits which support for half duplex mode at previous transfer */
|
|
|
+ Base->CFGR1 &= ~(LPSPI_CFGR1_PCSCFG_MASK | LPSPI_CFGR1_OUTCFG_MASK | LPSPI_CFGR1_PINCFG_MASK);
|
|
|
+ #endif
|
|
|
+
|
|
|
+ /* Mark the hardware as busy. */
|
|
|
+ State->Status = LPSPI_IP_BUSY;
|
|
|
+ State->ExternalDevice = ExternalDevice;
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_08();
|
|
|
+
|
|
|
+ /* Disable DMA requests and all interrupts */
|
|
|
+ Base->DER = 0u;
|
|
|
+ Base->IER = 0u;
|
|
|
+
|
|
|
+ /* Update State structure. */
|
|
|
+ State->NextTransferConfigAvailable = State->KeepCs;
|
|
|
+
|
|
|
+ /* Set clock configuration */
|
|
|
+ if(TRUE == State->FirstCmd)
|
|
|
+ {
|
|
|
+ /* Make sure that FIFOs will be empty before start new transfer session */
|
|
|
+ Base->CR |= LPSPI_CR_RTF_MASK|LPSPI_CR_RRF_MASK;
|
|
|
+ /* clear all flags */
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DUAL_CLOCK_MODE)
|
|
|
+ Base->CCR = ExternalDevice->Ccr[State->ClockMode];
|
|
|
+ #else
|
|
|
+ Base->CCR = ExternalDevice->Ccr;
|
|
|
+ #endif
|
|
|
+ /* Reset current FIFO slots are available to fill at beginning of job (HLD).*/
|
|
|
+ State->CurrentTxFifoSlot = LPSPI_IP_FIFO_SIZE_U8;
|
|
|
+ Lpspi_TransmitTxInit(Instance, TxBuffer, State->ExternalDevice->DeviceParams->FrameSize, State->ExternalDevice->DeviceParams->Lsb, Length);
|
|
|
+ }
|
|
|
+ Lpspi_TransmitRxInit(Instance, RxBuffer, State->ExternalDevice->DeviceParams->FrameSize, Length);
|
|
|
+ /* initialize current counter */
|
|
|
+ CurrentTicks = OsIf_GetCounter(LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ while(State->RxIndex != State->ExpectedFifoReads)
|
|
|
+ {
|
|
|
+ /* RECEIVE DATA */
|
|
|
+ /* The receiving should be performed first because maybe have a last frame in RX FIFO from previous channel
|
|
|
+ , it should be read to clear RXFIFO before start a new write to TXFIFO */
|
|
|
+ /* Read all Data available in receive HW fifo. */
|
|
|
+ NumberOfReads = (uint8)(((Base->FSR) & LPSPI_FSR_RXCOUNT_MASK) >> LPSPI_FSR_RXCOUNT_SHIFT);
|
|
|
+ /* Limits to remaining frames. */
|
|
|
+
|
|
|
+ if (NumberOfReads != 0u)
|
|
|
+ {
|
|
|
+ if (NumberOfReads > (State->ExpectedFifoReads - State->RxIndex))
|
|
|
+ {
|
|
|
+ NumberOfReads = (uint8)(State->ExpectedFifoReads - State->RxIndex);
|
|
|
+ }
|
|
|
+ /* If these are the first frames of this channel. Current TXFIFO slot must be plus 1 because the slot of CMD have moved out */
|
|
|
+ if (0u == State->RxIndex)
|
|
|
+ {
|
|
|
+ State->CurrentTxFifoSlot += 1u;
|
|
|
+ }
|
|
|
+ /* Read Data from RX FIFO */
|
|
|
+ Lpspi_Ip_ReadDataFromFifo(Instance, NumberOfReads);
|
|
|
+ /* Update current FIFO slots are available to fill .*/
|
|
|
+ State->CurrentTxFifoSlot += NumberOfReads;
|
|
|
+ ElapsedTicks = 0u;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* TRANSMIT DATA */
|
|
|
+ /* Push Data until HW fifo is full or transfer is done. */
|
|
|
+ /* After driver code read all frames in RX FIFO, if there are still some frames in TX FIFO, at the time before driver code check number of frames available in TX FIFO
|
|
|
+ to prepare to fill TX FIFO. At that time, interrupt occurred, and the time to process interrupt is longer than the time to transfer all frames in TX FIFO.
|
|
|
+ So TX FIFO will be empty and some frames received in RX FIFO, then the program is returned from interrupt and fill TX FIFO until full.
|
|
|
+ And there is a interrupt occurred after that before read all frames in RX FIFO, and the time to process interrupt is longer than the time to transfer all frames in TX FIFO.
|
|
|
+ State->CurrentTxFifoSlot variable is used to hanlde number of frames are "on bus transfer". They are always less than FIFO size */
|
|
|
+ if((State->CurrentTxFifoSlot != 0u) && (TxDoneFlag != TRUE))
|
|
|
+ {
|
|
|
+ if(State->ExpectedFifoWrites != State->TxIndex)
|
|
|
+ {
|
|
|
+ NumberOfWrites = State->CurrentTxFifoSlot;
|
|
|
+ /* Limits to remaining frames. */
|
|
|
+ if (NumberOfWrites > (State->ExpectedFifoWrites - State->TxIndex))
|
|
|
+ {
|
|
|
+ NumberOfWrites = (uint8)(State->ExpectedFifoWrites - State->TxIndex);
|
|
|
+ }
|
|
|
+ /* Push Data into TX FIFO */
|
|
|
+ Lpspi_Ip_PushDataToFifo(Instance, NumberOfWrites);
|
|
|
+ State->CurrentTxFifoSlot -= NumberOfWrites;
|
|
|
+ ElapsedTicks = 0u;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ if(TRUE == State->NextTransferConfigAvailable)
|
|
|
+ {
|
|
|
+ /* Initialize next transfer */
|
|
|
+ State->ExternalDevice->DeviceParams->DefaultData = State->DefaultDataNext;
|
|
|
+ State->FirstCmd = FALSE;
|
|
|
+ Lpspi_TransmitTxInit(Instance, State->TxBufferNext, State->FrameSizeNext, State->LsbNext, State->LengthNext);
|
|
|
+ State->NextTransferConfigAvailable = FALSE;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ TxDoneFlag = TRUE;
|
|
|
+ if((FALSE == State->KeepCs) && (0u != (Base->TCR & LPSPI_TCR_CONT_MASK)))
|
|
|
+ {
|
|
|
+ /* Clear CS */
|
|
|
+ Base->TCR &= ~(LPSPI_TCR_CONT_MASK | LPSPI_TCR_CONTC_MASK);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ ElapsedTicks += OsIf_GetElapsed(&CurrentTicks, LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ if (ElapsedTicks >= TimeoutTicks)
|
|
|
+ {
|
|
|
+ Status = LPSPI_IP_TIMEOUT;
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if (LPSPI_IP_STATUS_SUCCESS != Status)
|
|
|
+ {
|
|
|
+ State->Status = LPSPI_IP_FAULT;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->Status = LPSPI_IP_IDLE;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+
|
|
|
+#if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_SyncTransmitHalfDuplex(
|
|
|
+ const Lpspi_Ip_ExternalDeviceType *ExternalDevice,
|
|
|
+ uint8 *Buffer,
|
|
|
+ uint16 Length,
|
|
|
+ Lpspi_Ip_HalfDuplexType TransferType,
|
|
|
+ uint32 TimeOut
|
|
|
+ )
|
|
|
+{
|
|
|
+ LPSPI_Type *Base;
|
|
|
+ Lpspi_Ip_StateStructureType *State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+ uint32 TimeoutTicks = OsIf_MicrosToTicks(TimeOut, LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ uint32 CurrentTicks = 0u; /* initialize current counter */
|
|
|
+ uint32 ElapsedTicks = 0u; /* elapsed will give timeout */
|
|
|
+ uint8 Instance = 0u;
|
|
|
+ boolean TransferCompleted = FALSE;
|
|
|
+ uint8 NumberOfFramesTxFifo;
|
|
|
+ uint8 NumberOfWrites;
|
|
|
+ uint8 NumberOfReads;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(ExternalDevice != NULL_PTR);
|
|
|
+ DevAssert(0u != Length);
|
|
|
+ Lpspi_Ip_CheckValidParameters(ExternalDevice, Length);
|
|
|
+ #endif
|
|
|
+ Instance = ExternalDevice->Instance;
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(State != NULL_PTR);
|
|
|
+ #endif
|
|
|
+
|
|
|
+ Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_08();
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_08();
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Mark the hardware as busy. */
|
|
|
+ State->Status = LPSPI_IP_BUSY;
|
|
|
+ State->ExternalDevice = ExternalDevice;
|
|
|
+ /* Update TransferType */
|
|
|
+ State->ExternalDevice->DeviceParams->TransferType = TransferType;
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_08();
|
|
|
+
|
|
|
+ /* Prepare the transfer: Write to CFGR1 register the bit fields supporting Half duplex, get transfer buffer, get CMD value in reeive mode.*/
|
|
|
+ Lpspi_Ip_HalfDuplexPrepare(Instance, Buffer);
|
|
|
+
|
|
|
+ /* Disable DMA requests and all interrupts */
|
|
|
+ Base->DER = 0u;
|
|
|
+ Base->IER = 0u;
|
|
|
+
|
|
|
+ /* Update State structure. */
|
|
|
+ State->NextTransferConfigAvailable = State->KeepCs;
|
|
|
+
|
|
|
+ /* Set clock configuration */
|
|
|
+ if(TRUE == State->FirstCmd)
|
|
|
+ {
|
|
|
+ /* Make sure that FIFOs will be empty before start new transfer session */
|
|
|
+ Base->CR |= LPSPI_CR_RTF_MASK|LPSPI_CR_RRF_MASK;
|
|
|
+ /* clear all flags */
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DUAL_CLOCK_MODE)
|
|
|
+ Base->CCR = ExternalDevice->Ccr[State->ClockMode];
|
|
|
+ #else
|
|
|
+ Base->CCR = ExternalDevice->Ccr;
|
|
|
+ #endif
|
|
|
+ }
|
|
|
+ /* Init TX channel */
|
|
|
+ Lpspi_TransmitTxInit(Instance, NULL_PTR, State->ExternalDevice->DeviceParams->FrameSize, State->ExternalDevice->DeviceParams->Lsb, Length);
|
|
|
+ /* If in half duplex Transmit only mode, RX do not need to be configured */
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_RECEIVE == TransferType)
|
|
|
+ {
|
|
|
+ Lpspi_TransmitRxInit(Instance, NULL_PTR, State->ExternalDevice->DeviceParams->FrameSize, Length);
|
|
|
+ }
|
|
|
+ /* initialize current counter */
|
|
|
+ CurrentTicks = OsIf_GetCounter(LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ while((TRUE != TransferCompleted) && (LPSPI_IP_STATUS_SUCCESS == Status))
|
|
|
+ {
|
|
|
+ if(LPSPI_IP_HALF_DUPLEX_RECEIVE == TransferType)
|
|
|
+ {
|
|
|
+ /* RECEIVE DATA */
|
|
|
+ /* Read all Data available in receive HW fifo. */
|
|
|
+ NumberOfReads = (uint8)(((Base->FSR) & LPSPI_FSR_RXCOUNT_MASK) >> LPSPI_FSR_RXCOUNT_SHIFT);
|
|
|
+ if (NumberOfReads > (State->ExpectedFifoReads - State->RxIndex))
|
|
|
+ {
|
|
|
+ NumberOfReads = (uint8)(State->ExpectedFifoReads - State->RxIndex);
|
|
|
+ }
|
|
|
+ if (0u != NumberOfReads)
|
|
|
+ {
|
|
|
+ Lpspi_Ip_ReadDataFromFifo(Instance, NumberOfReads);
|
|
|
+ }
|
|
|
+ /* PUSH CMD */
|
|
|
+ /* Only write to TCR when TX FIFO is empty in receive mode */
|
|
|
+ if ((0u == ((Base->FSR) & LPSPI_FSR_TXCOUNT_MASK)) && (State->ExpectedFifoWrites != State->TxIndex))
|
|
|
+ {
|
|
|
+ /* Push one CMD to start transfer. */
|
|
|
+ Base->TCR = State->HalfDuplexTcrCommand;
|
|
|
+ State->TxIndex++;
|
|
|
+ }
|
|
|
+ ElapsedTicks = 0u;
|
|
|
+ if (State->ExpectedFifoReads == State->RxIndex)
|
|
|
+ {
|
|
|
+ /* if all frames have received, break from the while loop to finish transfer */
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* TRANSMIT DATA */
|
|
|
+ /* Push Data until HW fifo is full or transfer is done. */
|
|
|
+ NumberOfFramesTxFifo = (uint8)(((Base->FSR) & LPSPI_FSR_TXCOUNT_MASK) >> LPSPI_FSR_TXCOUNT_SHIFT);
|
|
|
+ NumberOfWrites = LPSPI_IP_FIFO_SIZE_U8 - NumberOfFramesTxFifo;
|
|
|
+ if(0u != NumberOfWrites)
|
|
|
+ {
|
|
|
+ TransferCompleted = Lpspi_Ip_HalfDuplexTransmitProcess(Instance, NumberOfWrites);
|
|
|
+ }
|
|
|
+ /* If this is latest channel, waiting for all frames in TXFIFO have moved out before finish transfer for next job */
|
|
|
+ /* If next channel is in receive mode, waiting for TX of this channel to complete before start next channel */
|
|
|
+ if ((TRUE == TransferCompleted) && ((TRUE == State->NextChannelIsRX) || (FALSE == State->KeepCs)))
|
|
|
+ {
|
|
|
+ TransferCompleted = FALSE;
|
|
|
+ if (((Base->FSR) & LPSPI_FSR_TXCOUNT_MASK) == 0u)
|
|
|
+ {
|
|
|
+ TransferCompleted = TRUE;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ ElapsedTicks = 0u;
|
|
|
+ }
|
|
|
+ ElapsedTicks += OsIf_GetElapsed(&CurrentTicks, LPSPI_IP_TIMEOUT_METHOD);
|
|
|
+ if (ElapsedTicks >= TimeoutTicks)
|
|
|
+ {
|
|
|
+ Status = LPSPI_IP_TIMEOUT;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ if (LPSPI_IP_STATUS_SUCCESS != Status)
|
|
|
+ {
|
|
|
+ State->Status = LPSPI_IP_FAULT;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->Status = LPSPI_IP_IDLE;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+static void Lpspi_TransmitTxInit(uint8 Instance,
|
|
|
+ uint8* TxBuffer,
|
|
|
+ uint8 TxFrameSize,
|
|
|
+ boolean TxLsb,
|
|
|
+ uint16 NumberOfFrames
|
|
|
+ )
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ uint32 TransferCommand = 0u;
|
|
|
+ uint8 LsbWriteValue = TxLsb ? 1U : 0U;
|
|
|
+
|
|
|
+ State->TxFrameSize = TxFrameSize;
|
|
|
+ State->TxLsb = TxLsb;
|
|
|
+ /* Get transfer command */
|
|
|
+#if (STD_ON == LPSPI_IP_DUAL_CLOCK_MODE)
|
|
|
+ TransferCommand = State->ExternalDevice->Tcr[State->ClockMode] | LPSPI_TCR_FRAMESZ((uint32)TxFrameSize - 1u) | LPSPI_TCR_LSBF(LsbWriteValue);
|
|
|
+#else
|
|
|
+ TransferCommand = State->ExternalDevice->Tcr | LPSPI_TCR_FRAMESZ((uint32)TxFrameSize - 1u) | LPSPI_TCR_LSBF(LsbWriteValue);
|
|
|
+#endif
|
|
|
+#if ((STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT) && (STD_ON == LPSPI_IP_SLAVE_SUPPORT))
|
|
|
+ /* In Receive only mode of Slave, a CMD with TXMSK should be push here */
|
|
|
+ if ((LPSPI_IP_HALF_DUPLEX_RECEIVE == State->ExternalDevice->DeviceParams->TransferType) && (TRUE == State->PhyUnitConfig->SlaveMode))
|
|
|
+ {
|
|
|
+ /* Mask TX in Transmit only mode */
|
|
|
+ TransferCommand |= LPSPI_TCR_TXMSK_MASK;
|
|
|
+ TransferCommand &= ~LPSPI_TCR_CONT_MASK;
|
|
|
+ Base->TCR = TransferCommand;
|
|
|
+ }
|
|
|
+#endif
|
|
|
+#if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ /* In RX only mode, TX FIFO will be masked, Push CMD to TX FIFO to transfer and pushing CMD will be done in transfer time. Push nothing to TCR register here */
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_RECEIVE != State->ExternalDevice->DeviceParams->TransferType)
|
|
|
+#endif
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_TRANSMIT == State->ExternalDevice->DeviceParams->TransferType)
|
|
|
+ {
|
|
|
+ /* Mask RX FIFO in Transmit only mode */
|
|
|
+ TransferCommand |= LPSPI_TCR_RXMSK_MASK;
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ /* In Slave mode, CONT bit should be cleared */
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if(TRUE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ {
|
|
|
+ TransferCommand &= ~LPSPI_TCR_CONT_MASK;
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+
|
|
|
+ if(TRUE == State->FirstCmd)
|
|
|
+ {
|
|
|
+ /* Set transfer command */
|
|
|
+ Base->TCR = TransferCommand;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ if (0u != (TransferCommand & LPSPI_TCR_CONT_MASK))
|
|
|
+ {
|
|
|
+ Base->TCR = TransferCommand | LPSPI_TCR_CONTC_MASK;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ Base->TCR = TransferCommand;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ if (LPSPI_IP_FULL_DUPLEX == State->ExternalDevice->DeviceParams->TransferType)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ /* CMD will take 1 slot in TXFIFO, update current TX FIFO slot. it will be plus 1 when the first frames of channel have recieved */
|
|
|
+ State->CurrentTxFifoSlot -= 1u;
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Update State structure. */
|
|
|
+ State->TxIndex = 0u;
|
|
|
+
|
|
|
+ /* In half duplex mode, TX buffer have updated */
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ if (LPSPI_IP_FULL_DUPLEX == State->ExternalDevice->DeviceParams->TransferType)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ State->TxBuffer = TxBuffer;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (TxFrameSize < 9u)
|
|
|
+ {
|
|
|
+ State->ExpectedFifoWrites = NumberOfFrames;
|
|
|
+ }
|
|
|
+ else if (TxFrameSize < 17u)
|
|
|
+ {
|
|
|
+ State->ExpectedFifoWrites = NumberOfFrames/2u;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->ExpectedFifoWrites = NumberOfFrames/4u;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+static void Lpspi_TransmitRxInit(uint8 Instance,
|
|
|
+ uint8* RxBuffer,
|
|
|
+ uint8 RxFrameSize,
|
|
|
+ uint16 NumberOfFrames
|
|
|
+ )
|
|
|
+{
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+
|
|
|
+ /* Update State structure. */
|
|
|
+ State->RxIndex = 0u;
|
|
|
+
|
|
|
+ /* In half duplex mode, RX buffer have updated */
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ if (LPSPI_IP_FULL_DUPLEX != State->ExternalDevice->DeviceParams->TransferType)
|
|
|
+ {
|
|
|
+ (void)RxBuffer;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ State->RxBuffer = RxBuffer;
|
|
|
+ }
|
|
|
+
|
|
|
+ if (RxFrameSize < 9u)
|
|
|
+ {
|
|
|
+ State->ExpectedFifoReads = NumberOfFrames;
|
|
|
+ }
|
|
|
+ else if (RxFrameSize < 17u)
|
|
|
+ {
|
|
|
+ State->ExpectedFifoReads = NumberOfFrames/2u;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->ExpectedFifoReads = NumberOfFrames/4u;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_AsyncTransmit(
|
|
|
+ const Lpspi_Ip_ExternalDeviceType *ExternalDevice,
|
|
|
+ uint8 *TxBuffer,
|
|
|
+ uint8 *RxBuffer,
|
|
|
+ uint16 Length,
|
|
|
+ Lpspi_Ip_CallbackType EndCallback
|
|
|
+ )
|
|
|
+{
|
|
|
+ LPSPI_Type* Base;
|
|
|
+ Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+ uint8 Instance = 0u;
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[1u];
|
|
|
+ #endif
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(ExternalDevice != NULL_PTR);
|
|
|
+ DevAssert(0u != Length);
|
|
|
+ Lpspi_Ip_CheckValidParameters(ExternalDevice, Length);
|
|
|
+ #endif
|
|
|
+ Instance = ExternalDevice->Instance;
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(State != NULL_PTR);
|
|
|
+ #endif
|
|
|
+
|
|
|
+ Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_09();
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_09();
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ /* Clear some bits which support for half duplex mode at previous transfer */
|
|
|
+ Base->CFGR1 &= ~(LPSPI_CFGR1_PCSCFG_MASK | LPSPI_CFGR1_OUTCFG_MASK | LPSPI_CFGR1_PINCFG_MASK);
|
|
|
+ #endif
|
|
|
+ /* Mark the hardware as busy. */
|
|
|
+ State->Status = LPSPI_IP_BUSY;
|
|
|
+ State->ExternalDevice = ExternalDevice;
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_09();
|
|
|
+
|
|
|
+ /* Update State structure. */
|
|
|
+ State->Callback = EndCallback;
|
|
|
+ State->NextTransferConfigAvailable = State->KeepCs;
|
|
|
+
|
|
|
+ /* Reset TX done flag */
|
|
|
+ State->TxDoneFlag = FALSE;
|
|
|
+
|
|
|
+ /* Set clock configuration */
|
|
|
+ if(TRUE == State->FirstCmd)
|
|
|
+ {
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_11();
|
|
|
+ /* Make sure that FIFOs will be empty before start new transfer session */
|
|
|
+ Base->CR |= LPSPI_CR_RTF_MASK|LPSPI_CR_RRF_MASK;
|
|
|
+ /* clear all flags */
|
|
|
+ Base->SR &= LPSPI_IP_SR_W1C_MASK_U32;
|
|
|
+ /* Does not use the Clock Configuration Register (CCR) for Slave mode */
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if(FALSE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_DUAL_CLOCK_MODE)
|
|
|
+ Base->CCR = ExternalDevice->Ccr[State->ClockMode];
|
|
|
+ #else
|
|
|
+ Base->CCR = ExternalDevice->Ccr;
|
|
|
+ #endif
|
|
|
+ }
|
|
|
+ /* Update current FIFO slots are available to fill .*/
|
|
|
+ State->CurrentTxFifoSlot = LPSPI_IP_FIFO_SIZE_U8;
|
|
|
+ /* In setting up Transmit command register, the RXMSK is also cleared */
|
|
|
+ Lpspi_TransmitTxInit(Instance, TxBuffer, State->ExternalDevice->DeviceParams->FrameSize, State->ExternalDevice->DeviceParams->Lsb, Length);
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_11();
|
|
|
+ }
|
|
|
+ Lpspi_TransmitRxInit(Instance, RxBuffer, State->ExternalDevice->DeviceParams->FrameSize, Length);
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ if (FALSE == State->PhyUnitConfig->DmaUsed)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if (TRUE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ {
|
|
|
+ Base->FCR = LPSPI_FCR_TXWATER(0u);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ /* Set TX water */
|
|
|
+ Base->FCR = LPSPI_FCR_TXWATER((uint32)LPSPI_IP_FIFO_SIZE_U8 - (uint32)1u);
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Disable DMA requests */
|
|
|
+ Base->DER = 0u;
|
|
|
+ switch (State->TransferMode)
|
|
|
+ {
|
|
|
+ case LPSPI_IP_POLLING:
|
|
|
+ /* Disable interrupts. */
|
|
|
+ Base->IER = 0u;
|
|
|
+ break;
|
|
|
+ case LPSPI_IP_INTERRUPT:
|
|
|
+ Base->IER = LPSPI_IER_RDIE_MASK | LPSPI_IER_TDIE_MASK | LPSPI_IER_REIE_MASK | LPSPI_IER_TEIE_MASK;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ /* Nothing to do */
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Disable all interrupts */
|
|
|
+ Base->IER = 0u;
|
|
|
+ /* Activate TX DMA and RX DMA interrupt in interrupt mode or disable then in polling mode. */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_CONTROL_EN_MAJOR_INTERRUPT;
|
|
|
+ switch(State->TransferMode)
|
|
|
+ {
|
|
|
+ case LPSPI_IP_POLLING:
|
|
|
+ /* Disable DMA major interrupt. */
|
|
|
+ DmaTcdList[0u].Value = 0u;
|
|
|
+ break;
|
|
|
+ case LPSPI_IP_INTERRUPT:
|
|
|
+ /* Enable DMA major interrupt. */
|
|
|
+ DmaTcdList[0u].Value = 1u;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ /* Nothing to do */
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ /* TX WATER should be set again to LPSPI_IP_FIFO_SIZE_U8 - 3 to make sure when TX DMA finish, at least 2 TX FIFO slots are available to
|
|
|
+ fill CMD and the first frame of next channel */
|
|
|
+ Base->FCR = LPSPI_FCR_TXWATER((uint32)LPSPI_IP_FIFO_SIZE_U8 - (uint32)3u);
|
|
|
+
|
|
|
+ (void)Dma_Ip_SetLogicChannelTransferList(State->PhyUnitConfig->TxDmaChannel, DmaTcdList, 1u);
|
|
|
+ (void)Dma_Ip_SetLogicChannelTransferList(State->PhyUnitConfig->RxDmaChannel, DmaTcdList, 1u);
|
|
|
+
|
|
|
+ /* Initialize DMA configuration for RX before TX*/
|
|
|
+ Lpspi_Ip_RxDmaConfig(Instance);
|
|
|
+ if(TRUE == State->FirstCmd)
|
|
|
+ {
|
|
|
+ /* Call function to configure TX DMA channel only for the first channel, others channels will be done in TX DMA notification function */
|
|
|
+ Lpspi_Ip_TxDmaConfig(Instance);
|
|
|
+ }
|
|
|
+ /* Enable DMA request. */
|
|
|
+ Base->DER = LPSPI_DER_RDDE_MASK | LPSPI_DER_TDDE_MASK;
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+
|
|
|
+#if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_AsyncTransmitHalfDuplex(
|
|
|
+ const Lpspi_Ip_ExternalDeviceType *ExternalDevice,
|
|
|
+ uint8 *Buffer,
|
|
|
+ uint16 Length,
|
|
|
+ Lpspi_Ip_HalfDuplexType TransferType,
|
|
|
+ Lpspi_Ip_CallbackType EndCallback
|
|
|
+ )
|
|
|
+{
|
|
|
+ LPSPI_Type* Base;
|
|
|
+ Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+ uint8 Instance = 0u;
|
|
|
+ uint32 InterruptEnable = 0u;
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[1u];
|
|
|
+ #endif
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(ExternalDevice != NULL_PTR);
|
|
|
+ DevAssert(0u != Length);
|
|
|
+ Lpspi_Ip_CheckValidParameters(ExternalDevice, Length);
|
|
|
+ #endif
|
|
|
+ Instance = ExternalDevice->Instance;
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(State != NULL_PTR);
|
|
|
+ #endif
|
|
|
+
|
|
|
+ Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_09();
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_09();
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Mark the hardware as busy. */
|
|
|
+ State->Status = LPSPI_IP_BUSY;
|
|
|
+ State->ExternalDevice = ExternalDevice;
|
|
|
+ /* Update TransferType */
|
|
|
+ State->ExternalDevice->DeviceParams->TransferType = TransferType;
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_09();
|
|
|
+
|
|
|
+ /* Prepare the transfer: Write to CFGR1 register the bit fields supporting Half duplex, get transfer buffer, get CMD value in reeive mode.*/
|
|
|
+ Lpspi_Ip_HalfDuplexPrepare(Instance, Buffer);
|
|
|
+
|
|
|
+ /* Update State structure. */
|
|
|
+ State->Callback = EndCallback;
|
|
|
+ State->NextTransferConfigAvailable = State->KeepCs;
|
|
|
+
|
|
|
+ /* Set clock configuration */
|
|
|
+ if(TRUE == State->FirstCmd)
|
|
|
+ {
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_11();
|
|
|
+ /* Make sure that FIFOs will be empty before start new transfer session */
|
|
|
+ Base->CR |= LPSPI_CR_RTF_MASK|LPSPI_CR_RRF_MASK;
|
|
|
+ /* clear all flags */
|
|
|
+ Base->SR &= LPSPI_IP_SR_W1C_MASK_U32;
|
|
|
+ /* Does not use the Clock Configuration Register (CCR) for Slave mode */
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if(FALSE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_DUAL_CLOCK_MODE)
|
|
|
+ Base->CCR = ExternalDevice->Ccr[State->ClockMode];
|
|
|
+ #else
|
|
|
+ Base->CCR = ExternalDevice->Ccr;
|
|
|
+ #endif
|
|
|
+ }
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_11();
|
|
|
+ }
|
|
|
+ /* Init TX channel */
|
|
|
+ Lpspi_TransmitTxInit(Instance, NULL_PTR, State->ExternalDevice->DeviceParams->FrameSize, State->ExternalDevice->DeviceParams->Lsb, Length);
|
|
|
+ /* If in Transmit only mode, RX do not need to be configured */
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_RECEIVE == TransferType)
|
|
|
+ {
|
|
|
+ Lpspi_TransmitRxInit(Instance, NULL_PTR, State->ExternalDevice->DeviceParams->FrameSize, Length);
|
|
|
+ }
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if ((TRUE == State->PhyUnitConfig->SlaveMode) && (LPSPI_IP_HALF_DUPLEX_RECEIVE == TransferType))
|
|
|
+ {
|
|
|
+ /* In slave receive mode, only use RX Flags */
|
|
|
+ InterruptEnable = LPSPI_IER_RDIE_MASK | LPSPI_IER_REIE_MASK;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ /* Others mode still use TX flag to trigger to push CMDs */
|
|
|
+ InterruptEnable = LPSPI_IER_RDIE_MASK | LPSPI_IER_TDIE_MASK | LPSPI_IER_REIE_MASK | LPSPI_IER_TEIE_MASK;
|
|
|
+ }
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ if (FALSE == State->PhyUnitConfig->DmaUsed)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ /* In Receive mode, Next CMD could only be pushed to TXFIFO when previous CMD have moved out (TXFIFO is empty)
|
|
|
+ On Slave receive mode, do not care about TX WATER because TDIE flag is not set */
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_RECEIVE == TransferType)
|
|
|
+ {
|
|
|
+ Base->FCR = LPSPI_FCR_TXWATER(0u);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ Base->FCR = LPSPI_FCR_TXWATER((uint32)LPSPI_IP_FIFO_SIZE_U8 - (uint32)1u);
|
|
|
+ }
|
|
|
+ /* Disable DMA requests */
|
|
|
+ Base->DER = 0u;
|
|
|
+ switch (State->TransferMode)
|
|
|
+ {
|
|
|
+ case LPSPI_IP_POLLING:
|
|
|
+ /* Disable interrupts. */
|
|
|
+ Base->IER = 0u;
|
|
|
+ break;
|
|
|
+ case LPSPI_IP_INTERRUPT:
|
|
|
+ Base->IER = InterruptEnable;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ /* Nothing to do */
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Disable all interrupts */
|
|
|
+ Base->IER = 0u;
|
|
|
+ /* Activate TX DMA and RX DMA interrupt in interrupt mode or disable then in polling mode. */
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_CONTROL_EN_MAJOR_INTERRUPT;
|
|
|
+ switch(State->TransferMode)
|
|
|
+ {
|
|
|
+ case LPSPI_IP_POLLING:
|
|
|
+ /* Disable DMA major interrupt. */
|
|
|
+ DmaTcdList[0u].Value = 0u;
|
|
|
+ break;
|
|
|
+ case LPSPI_IP_INTERRUPT:
|
|
|
+ /* Enable DMA major interrupt. */
|
|
|
+ DmaTcdList[0u].Value = 1u;
|
|
|
+ break;
|
|
|
+ default:
|
|
|
+ /* Nothing to do */
|
|
|
+ break;
|
|
|
+ }
|
|
|
+ if (LPSPI_IP_HALF_DUPLEX_TRANSMIT == TransferType)
|
|
|
+ {
|
|
|
+ (void)Dma_Ip_SetLogicChannelTransferList(State->PhyUnitConfig->TxDmaChannel, DmaTcdList, 1u);
|
|
|
+
|
|
|
+ /* TX WATER should be set again to LPSPI_IP_FIFO_SIZE_U8 - 2 to make sure when TX DMA finish, at least 1 TX FIFO slots are available to
|
|
|
+ fill CMD (in the case clear CONT bit) at latest channel in job */
|
|
|
+ Base->FCR = LPSPI_FCR_TXWATER((uint32)LPSPI_IP_FIFO_SIZE_U8 - (uint32)2u);
|
|
|
+
|
|
|
+ /* Call function to configure TX DMA channel only for the first channel, others channels will be done in TX DMA notification function */
|
|
|
+ /* In Transmit only mode, this function only is called here */
|
|
|
+ Lpspi_Ip_TxDmaConfig(Instance);
|
|
|
+ /* Enable DMA request. */
|
|
|
+ Base->DER = LPSPI_DER_TDDE_MASK;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Reveive only mode still need TX channel to sent CMD to provide clock out */
|
|
|
+ (void)Dma_Ip_SetLogicChannelTransferList(State->PhyUnitConfig->TxDmaChannel, DmaTcdList, 1u);
|
|
|
+ (void)Dma_Ip_SetLogicChannelTransferList(State->PhyUnitConfig->RxDmaChannel, DmaTcdList, 1u);
|
|
|
+
|
|
|
+ /* TX WATER sets to 0 to make sure that TX DMA only write 1 CMD and wait for this CMD have moved out to perform to push next CMD */
|
|
|
+ Base->FCR = LPSPI_FCR_TXWATER(0u);
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_SLAVE_SUPPORT)
|
|
|
+ if (TRUE == State->PhyUnitConfig->SlaveMode)
|
|
|
+ {
|
|
|
+ /* In Half duplex slave receive mode, only RX DMA channel needed */
|
|
|
+ Lpspi_Ip_RxDmaConfig(Instance);
|
|
|
+ /* Enable DMA request. */
|
|
|
+ Base->DER = LPSPI_DER_RDDE_MASK;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ /* In receive only mode, TX DMA Channel will be used to push CMD to start transfer */
|
|
|
+ Lpspi_Ip_RxDmaConfig(Instance);
|
|
|
+ Lpspi_Ip_TxDmaConfig(Instance);
|
|
|
+ /* Enable DMA request. */
|
|
|
+ Base->DER = LPSPI_DER_RDDE_MASK | LPSPI_DER_TDDE_MASK;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+#if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+#if (STD_ON == LPSPI_IP_ENABLE_DMAFASTTRANSFER_SUPPORT)
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_AsyncTransmitFast(
|
|
|
+ const Lpspi_Ip_FastTransferType *FastTransferCfg,
|
|
|
+ uint8 NumberOfTransfer,
|
|
|
+ Lpspi_Ip_CallbackType EndCallback
|
|
|
+ )
|
|
|
+{
|
|
|
+ LPSPI_Type* Base;
|
|
|
+ Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+ uint8 Instance = 0u;
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ uint8 Count = 0u;
|
|
|
+
|
|
|
+ DevAssert(NULL_PTR != FastTransferCfg);
|
|
|
+ DevAssert(NULL_PTR != FastTransferCfg[0u].ExternalDevice);
|
|
|
+ #endif
|
|
|
+ Instance = FastTransferCfg[0u].ExternalDevice->Instance;
|
|
|
+ Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR == State);
|
|
|
+ DevAssert(State->TransferMode == LPSPI_IP_INTERRUPT);
|
|
|
+ DevAssert(NumberOfTransfer <= State->PhyUnitConfig->MaxNumOfFastTransfer);
|
|
|
+ for(Count = 0u; Count < NumberOfTransfer; Count++)
|
|
|
+ {
|
|
|
+ DevAssert(NULL_PTR != FastTransferCfg[Count].ExternalDevice);
|
|
|
+ DevAssert(0u != FastTransferCfg[Count].Length);
|
|
|
+ DevAssert(FastTransferCfg[0u].ExternalDevice->DeviceParams->FrameSize <= 16u);
|
|
|
+ if (FastTransferCfg[0u].ExternalDevice->DeviceParams->FrameSize > 8u)
|
|
|
+ {
|
|
|
+ DevAssert((FastTransferCfg[Count].Length%2) == 0u);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ DevAssert(LPSPI_IP_DMA_MAX_ITER_CNT_U16 >= FastTransferCfg[Count].Length);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_15();
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_15();
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ /* Clear some bits which support for half duplex mode at previous transfer */
|
|
|
+ Base->CFGR1 &= ~(LPSPI_CFGR1_PCSCFG_MASK | LPSPI_CFGR1_OUTCFG_MASK | LPSPI_CFGR1_PINCFG_MASK);
|
|
|
+ #endif
|
|
|
+
|
|
|
+ /* Mark the hardware as busy. */
|
|
|
+ State->Status = LPSPI_IP_BUSY;
|
|
|
+
|
|
|
+ /* Update State structure. */
|
|
|
+ /* For Dma Fast transfer, All transfers use the same HWUnit and in Master mode only.
|
|
|
+ Some parameters such as Baudrate, Delays timming SpiTimeClk2Cs, SpiTimeCs2Clk, SpiTimeCs2Cs, SpiDataWidth, SpiTransferStart configuration
|
|
|
+ must be the same between transfers. So, make sure they are configured the same in each External Device allocated for Dma Fast Transfers.
|
|
|
+ And all those attributes can be got from first transfer in FastTransferCfg */
|
|
|
+ State->ExternalDevice = FastTransferCfg[0u].ExternalDevice;
|
|
|
+ State->Callback = EndCallback;
|
|
|
+
|
|
|
+ /* Disable module before configure CCR */
|
|
|
+ Base->CR &= ~LPSPI_CR_MEN_MASK;
|
|
|
+ /* Make sure that FIFOs will be empty before start new transfer session */
|
|
|
+ Base->CR |= LPSPI_CR_RTF_MASK|LPSPI_CR_RRF_MASK;
|
|
|
+
|
|
|
+ /* clear all flags */
|
|
|
+ Base->SR &= LPSPI_IP_SR_W1C_MASK_U32;
|
|
|
+ #if (STD_ON == LPSPI_IP_DUAL_CLOCK_MODE)
|
|
|
+ Base->CCR = FastTransferCfg[0u].ExternalDevice->Ccr[State->ClockMode];
|
|
|
+ #else
|
|
|
+ Base->CCR = FastTransferCfg[0u].ExternalDevice->Ccr;
|
|
|
+ #endif
|
|
|
+ Base->CR |= LPSPI_CR_MEN_MASK;
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_15();
|
|
|
+
|
|
|
+ /* Disable all interrupts */
|
|
|
+ Base->IER = 0u;
|
|
|
+ /* Enable DMA request. */
|
|
|
+ Base->DER = LPSPI_DER_RDDE_MASK | LPSPI_DER_TDDE_MASK;
|
|
|
+
|
|
|
+ Lpspi_Ip_DmaFastConfig(Instance, FastTransferCfg, NumberOfTransfer);
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+
|
|
|
+/**
|
|
|
+* @brief This function will configure Scatter/Gather TCDs for the channels TX DMA, RX DMA and CMD DMA
|
|
|
+* according to Dma Fast transfers configuration. DMA channels will be started at the end of the function.
|
|
|
+*/
|
|
|
+static void Lpspi_Ip_DmaFastConfig(uint8 Instance, const Lpspi_Ip_FastTransferType *FastTransferCfg, uint8 NumberOfTransfer)
|
|
|
+{
|
|
|
+ Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ boolean ClearCS = FALSE;
|
|
|
+ uint8 DisHwRequest = 0u;
|
|
|
+ uint8 Count = 0u;
|
|
|
+ uint8 TxDmaTCDSGIndex = 0u;
|
|
|
+ uint32 TransferCommand = 0u;
|
|
|
+ uint32 LSBBit = 0UL;
|
|
|
+ boolean FirstCmd = TRUE;
|
|
|
+ Dma_Ip_LogicChannelTransferListType DmaTcdList[1u];
|
|
|
+
|
|
|
+ DmaTcdList[0u].Param = DMA_IP_CH_SET_CONTROL_EN_MAJOR_INTERRUPT;
|
|
|
+ DmaTcdList[0u].Value = 1u;
|
|
|
+
|
|
|
+ for(Count = 0u; Count < NumberOfTransfer; Count++)
|
|
|
+ {
|
|
|
+ /* Update State structure. */
|
|
|
+ State->RxIndex = 0u;
|
|
|
+ State->TxIndex = 0u;
|
|
|
+ State->TxBuffer = FastTransferCfg[Count].TxBuffer;
|
|
|
+ State->RxBuffer = FastTransferCfg[Count].RxBuffer;
|
|
|
+ if (State->ExternalDevice->DeviceParams->FrameSize < 9u)
|
|
|
+ {
|
|
|
+ State->ExpectedFifoWrites = FastTransferCfg[Count].Length;
|
|
|
+ }
|
|
|
+ else if (State->ExternalDevice->DeviceParams->FrameSize < 17u)
|
|
|
+ {
|
|
|
+ State->ExpectedFifoWrites = FastTransferCfg[Count].Length/2u;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ State->ExpectedFifoWrites = FastTransferCfg[Count].Length/4u;
|
|
|
+ }
|
|
|
+ State->ExpectedFifoReads = State->ExpectedFifoWrites;
|
|
|
+ State->PhyUnitConfig->CmdDmaFast[Count].DefaultData = FastTransferCfg[Count].DefaultData;
|
|
|
+ LSBBit = State->ExternalDevice->DeviceParams->Lsb ? 1UL : 0UL;
|
|
|
+
|
|
|
+ /* Get transfer command */
|
|
|
+ #if (STD_ON == LPSPI_IP_DUAL_CLOCK_MODE)
|
|
|
+ TransferCommand = FastTransferCfg[Count].ExternalDevice->Tcr[State->ClockMode] | LPSPI_TCR_FRAMESZ((uint32)State->ExternalDevice->DeviceParams->FrameSize - 1u) | LPSPI_TCR_LSBF(LSBBit);
|
|
|
+ #else
|
|
|
+ TransferCommand = FastTransferCfg[Count].ExternalDevice->Tcr | LPSPI_TCR_FRAMESZ((uint32)State->ExternalDevice->DeviceParams->FrameSize - 1u) | LPSPI_TCR_LSBF(LSBBit);
|
|
|
+ #endif
|
|
|
+
|
|
|
+ if (0u != (TransferCommand & LPSPI_TCR_CONT_MASK))
|
|
|
+ {
|
|
|
+ if(TRUE == FirstCmd)
|
|
|
+ {
|
|
|
+ FirstCmd = FALSE;
|
|
|
+ /* In continue CS, no need to set CONTC for first command */
|
|
|
+ State->PhyUnitConfig->CmdDmaFast[Count].DmaFastTcrCmd = TransferCommand;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* In continue CS, need to set CONTC for next command */
|
|
|
+ State->PhyUnitConfig->CmdDmaFast[Count].DmaFastTcrCmd = TransferCommand | LPSPI_TCR_CONTC_MASK;
|
|
|
+ }
|
|
|
+ State->PhyUnitConfig->CmdDmaFast[Count].DmaFastTcrCmdLast = TransferCommand & (~(LPSPI_TCR_CONT_MASK | LPSPI_TCR_CONTC_MASK));
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* In non-continue CS, set DmaFastTcrCmd = DmaFastTcrCmdLast */
|
|
|
+ State->PhyUnitConfig->CmdDmaFast[Count].DmaFastTcrCmd = TransferCommand;
|
|
|
+ State->PhyUnitConfig->CmdDmaFast[Count].DmaFastTcrCmdLast = TransferCommand;
|
|
|
+ }
|
|
|
+ if(FALSE == FastTransferCfg[Count].KeepCs)
|
|
|
+ {
|
|
|
+ /* After CS is cleared, next transfer will be set as first command */
|
|
|
+ FirstCmd = TRUE;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* CS will be cleared for last transfer or depend on KeepCs if not last transfer.
|
|
|
+ Disable DMA HW request at the end of transfer. */
|
|
|
+ if (Count == (NumberOfTransfer - 1u))
|
|
|
+ {
|
|
|
+ ClearCS = TRUE;
|
|
|
+ DisHwRequest = 1u;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ if(FALSE == FastTransferCfg[Count].KeepCs)
|
|
|
+ {
|
|
|
+ ClearCS = TRUE;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ ClearCS = FALSE;
|
|
|
+ }
|
|
|
+ DisHwRequest = 0u;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Configure software TCDs Scatter Gather for TX DMA channel to update TCR */
|
|
|
+ /* No disable dma hw request for transfer command */
|
|
|
+ Lpspi_Ip_CmdTxDmaTcdSGConfig(Instance, TxDmaTCDSGIndex, (uint32)&State->PhyUnitConfig->CmdDmaFast[Count].DmaFastTcrCmd, 0u);
|
|
|
+ TxDmaTCDSGIndex++;
|
|
|
+
|
|
|
+ if((TRUE == ClearCS) && (0u != (TransferCommand & LPSPI_TCR_CONT_MASK)))
|
|
|
+ {
|
|
|
+ /* Configure software TCDs Scatter Gather for TX DMA channel to fill TDR, no disable dma hw request */
|
|
|
+ Lpspi_Ip_TxDmaTcdSGConfig(Instance, TxDmaTCDSGIndex, 0u, (const uint32 *)&State->PhyUnitConfig->CmdDmaFast[Count].DefaultData);
|
|
|
+ TxDmaTCDSGIndex++;
|
|
|
+ /* disable dma hw request for last transfer command */
|
|
|
+ Lpspi_Ip_CmdTxDmaTcdSGConfig(Instance, TxDmaTCDSGIndex, (uint32)&State->PhyUnitConfig->CmdDmaFast[Count].DmaFastTcrCmdLast, DisHwRequest);
|
|
|
+ TxDmaTCDSGIndex++;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Configure software TCDs Scatter Gather for TX DMA channel to fill TDR */
|
|
|
+ Lpspi_Ip_TxDmaTcdSGConfig(Instance, TxDmaTCDSGIndex, DisHwRequest, (const uint32 *)&State->PhyUnitConfig->CmdDmaFast[Count].DefaultData);
|
|
|
+ TxDmaTCDSGIndex++;
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Configure software TCDs Scatter Gather for RX DMA channel */
|
|
|
+ Lpspi_Ip_RxDmaTcdSGConfig(Instance, Count, DisHwRequest);
|
|
|
+ }
|
|
|
+
|
|
|
+ /* When all transfers session are completed and next TCD ScatterGather is loaded to HW. If next TCD ScatterGather loaded has INTMAJOR=0 then
|
|
|
+ Dma_Ip interrupt function will not call Spi Dma notification due to it understood as spurious interrupt(Done flag = 1, INTMAJOR=0).
|
|
|
+ So, the workaround is set INTMAJOR=1 for next TCD ScatterGather. */
|
|
|
+ if (TxDmaTCDSGIndex < State->PhyUnitConfig->NumberTxSG)
|
|
|
+ {
|
|
|
+ /* Set INTMAJOR=1 for next TX TCD ScatterGather */
|
|
|
+ (void)Dma_Ip_SetLogicChannelScatterGatherList(State->PhyUnitConfig->TxDmaChannel, State->PhyUnitConfig->TxDmaFastSGId[TxDmaTCDSGIndex], DmaTcdList, 1u);
|
|
|
+ }
|
|
|
+ if (NumberOfTransfer < State->PhyUnitConfig->NumberRxSG)
|
|
|
+ {
|
|
|
+ /* Set INTMAJOR=1 for next RX TCD ScatterGather */
|
|
|
+ (void)Dma_Ip_SetLogicChannelScatterGatherList(State->PhyUnitConfig->RxDmaChannel, State->PhyUnitConfig->RxDmaFastSGId[NumberOfTransfer], DmaTcdList, 1u);
|
|
|
+ }
|
|
|
+
|
|
|
+ /* Load first software TCD to hardware TCD for TX DMA channel */
|
|
|
+ (void)Dma_Ip_SetLogicChannelScatterGatherConfig(State->PhyUnitConfig->TxDmaChannel, State->PhyUnitConfig->TxDmaFastSGId[0u]);
|
|
|
+ /* Load first software TCD to hardware TCD for RX DMA channel */
|
|
|
+ (void)Dma_Ip_SetLogicChannelScatterGatherConfig(State->PhyUnitConfig->RxDmaChannel, State->PhyUnitConfig->RxDmaFastSGId[0u]);
|
|
|
+
|
|
|
+ /* Enable HW request for RX DMA channel before TX DMA channel */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->RxDmaChannel, DMA_IP_CH_SET_HARDWARE_REQUEST);
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->TxDmaChannel, DMA_IP_CH_SET_HARDWARE_REQUEST);
|
|
|
+}
|
|
|
+#endif
|
|
|
+#endif
|
|
|
+
|
|
|
+void Lpspi_Ip_ManageBuffers(uint8 Instance)
|
|
|
+{
|
|
|
+ const LPSPI_Type* Base;
|
|
|
+ const Lpspi_Ip_StateStructureType* State;
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ Dma_Ip_LogicChannelStatusType DmaChannelStatus;
|
|
|
+ #endif
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(Instance < LPSPI_INSTANCE_COUNT);
|
|
|
+ #endif
|
|
|
+ Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != State);
|
|
|
+ #endif
|
|
|
+ if (LPSPI_IP_POLLING == State->TransferMode)
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ if(FALSE == State->PhyUnitConfig->DmaUsed)
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ /* In half duplex master mode:
|
|
|
+ - Transmit: TDF is set only, RDF will never be set by RX FIFO have masked.
|
|
|
+ - Receive: Both TDF (push CDM) and DRF (receive data) are set.
|
|
|
+ In half duplex Slave mode:
|
|
|
+ - Transmit: TDF is set only, RDF will never be set by RX FIFO have masked.
|
|
|
+ - Receive: Both TDF and RDF are set. But only process for RDF to receive data. In this case, State->CurrentTxFifoSlot will be set to 0 */
|
|
|
+ if(
|
|
|
+ (0u != (Base->SR & LPSPI_SR_RDF_MASK)) ||
|
|
|
+ ((0u != (Base->SR & LPSPI_SR_TDF_MASK)) && (0u!= State->CurrentTxFifoSlot))
|
|
|
+ )
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ if (LPSPI_IP_FULL_DUPLEX != State->ExternalDevice->DeviceParams->TransferType)
|
|
|
+ {
|
|
|
+ Lpspi_Ip_TransferProcessHalfDuplex(Instance);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ Lpspi_Ip_TransferProcess(Instance);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ }
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Polling RX before TX */
|
|
|
+ (void)Dma_Ip_GetLogicChannelStatus(State->PhyUnitConfig->RxDmaChannel, &DmaChannelStatus);
|
|
|
+ if(TRUE == DmaChannelStatus.Done)
|
|
|
+ {
|
|
|
+ /* Clear DONE bit */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->RxDmaChannel, DMA_IP_CH_CLEAR_DONE);
|
|
|
+ Lpspi_Ip_IrqRxDmaHandler(Instance);
|
|
|
+ }
|
|
|
+
|
|
|
+ (void)Dma_Ip_GetLogicChannelStatus(State->PhyUnitConfig->TxDmaChannel, &DmaChannelStatus);
|
|
|
+ if(TRUE == DmaChannelStatus.Done)
|
|
|
+ {
|
|
|
+ /* Clear DONE bit */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->TxDmaChannel, DMA_IP_CH_CLEAR_DONE);
|
|
|
+ Lpspi_Ip_IrqTxDmaHandler(Instance);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ }
|
|
|
+}
|
|
|
+/*================================================================================================*/
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_UpdateFrameSize(const Lpspi_Ip_ExternalDeviceType *ExternalDevice, uint8 FrameSize)
|
|
|
+{
|
|
|
+ const Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != ExternalDevice);
|
|
|
+ DevAssert(LPSPI_IP_FRAMESIZE_MAX_U8 >= FrameSize);
|
|
|
+ DevAssert(LPSPI_IP_FRAMESIZE_MIN_U8 <= FrameSize);
|
|
|
+ #endif
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[ExternalDevice->Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != State);
|
|
|
+ #endif
|
|
|
+ /* Frame size can be changed when no transfers are in progress. */
|
|
|
+ if (State->Status != LPSPI_IP_BUSY)
|
|
|
+ {
|
|
|
+ ExternalDevice->DeviceParams->FrameSize = FrameSize;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_UpdateLsb(const Lpspi_Ip_ExternalDeviceType *ExternalDevice, boolean Lsb)
|
|
|
+{
|
|
|
+ const Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != ExternalDevice);
|
|
|
+ #endif
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[ExternalDevice->Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != State);
|
|
|
+ #endif
|
|
|
+ /* Bite order can be changed when no transfers are in progress. */
|
|
|
+ if (State->Status != LPSPI_IP_BUSY)
|
|
|
+ {
|
|
|
+ ExternalDevice->DeviceParams->Lsb = Lsb;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_UpdateDefaultTransmitData(const Lpspi_Ip_ExternalDeviceType *ExternalDevice, uint32 DefaultData)
|
|
|
+{
|
|
|
+ const Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != ExternalDevice);
|
|
|
+ #endif
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[ExternalDevice->Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != State);
|
|
|
+ #endif
|
|
|
+ /* Bite order can be changed when no transfers are in progress. */
|
|
|
+ if (State->Status != LPSPI_IP_BUSY)
|
|
|
+ {
|
|
|
+ ExternalDevice->DeviceParams->DefaultData = DefaultData;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_UpdateTransferMode(uint8 Instance, Lpspi_Ip_ModeType Mode)
|
|
|
+{
|
|
|
+ Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(Instance < LPSPI_INSTANCE_COUNT);
|
|
|
+ #endif
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != State);
|
|
|
+ #endif
|
|
|
+ /* Transfer mode can be changed when no transfers are in progress. */
|
|
|
+ if (State->Status != LPSPI_IP_BUSY)
|
|
|
+ {
|
|
|
+ State->TransferMode = Mode;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+
|
|
|
+void Lpspi_Ip_Cancel(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base;
|
|
|
+ Lpspi_Ip_StateStructureType* State;
|
|
|
+ uint32 Cfgr1 = 0u;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(Instance < LPSPI_INSTANCE_COUNT);
|
|
|
+ #endif
|
|
|
+ Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != State);
|
|
|
+ #endif
|
|
|
+ SchM_Enter_Spi_SPI_EXCLUSIVE_AREA_10();
|
|
|
+ if (LPSPI_IP_BUSY == State->Status)
|
|
|
+ {
|
|
|
+ /* Mask Rx to discard received data */
|
|
|
+ Base->TCR |= LPSPI_TCR_RXMSK(1);
|
|
|
+ /* store CFGR1 and restore after all registers are reset */
|
|
|
+ Cfgr1 = Base->CFGR1;
|
|
|
+ /* Disable interrupts and DMA requests. */
|
|
|
+ /* Clear FIFO */
|
|
|
+ /* RTF and RRF will not clear shifter, so RST must be used to ensure old Data in shifter will also be cleared. */
|
|
|
+ Base->CR |= LPSPI_CR_RST_MASK;
|
|
|
+ Base->CR &= ~LPSPI_CR_RST_MASK;
|
|
|
+ /* restore CFGR1 */
|
|
|
+ Base->CFGR1 = Cfgr1;
|
|
|
+ #if (STD_ON == LPSPI_IP_DMA_USED)
|
|
|
+ if(TRUE == State->PhyUnitConfig->DmaUsed)
|
|
|
+ {
|
|
|
+ /* Disable all HW request */
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->RxDmaChannel, DMA_IP_CH_CLEAR_HARDWARE_REQUEST);
|
|
|
+ (void)Dma_Ip_SetLogicChannelCommand(State->PhyUnitConfig->TxDmaChannel, DMA_IP_CH_CLEAR_HARDWARE_REQUEST);
|
|
|
+ }
|
|
|
+ #endif
|
|
|
+ /* set State to idle */
|
|
|
+ State->Status = LPSPI_IP_IDLE;
|
|
|
+ }
|
|
|
+ SchM_Exit_Spi_SPI_EXCLUSIVE_AREA_10();
|
|
|
+}
|
|
|
+/**
|
|
|
+* @brief This function is called by LSPI ISRs.
|
|
|
+* @details This function will process activities for flags TDF, RDF, REF and TEF.
|
|
|
+*
|
|
|
+* @param[in] Instance Instance of the hardware unit.
|
|
|
+*
|
|
|
+* @implements Lpspi_Ip_IrqHandler_Activity
|
|
|
+*/
|
|
|
+void Lpspi_Ip_IrqHandler(uint8 Instance)
|
|
|
+{
|
|
|
+ LPSPI_Type* Base = Lpspi_Ip_apxBases[Instance];
|
|
|
+ const Lpspi_Ip_StateStructureType* State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ uint32 IrqFlags = 0u;
|
|
|
+
|
|
|
+ if (NULL_PTR != State)
|
|
|
+ {
|
|
|
+ /* the driver has been initialized */
|
|
|
+ IrqFlags = Base->SR & (LPSPI_SR_TDF_MASK | LPSPI_SR_RDF_MASK | LPSPI_SR_TEF_MASK | LPSPI_SR_REF_MASK);
|
|
|
+ IrqFlags &= Base->IER & (LPSPI_IER_TDIE_MASK | LPSPI_IER_RDIE_MASK | LPSPI_IER_TEIE_MASK | LPSPI_IER_REIE_MASK);
|
|
|
+ if (0u != IrqFlags)
|
|
|
+ {
|
|
|
+ #if (STD_ON == LPSPI_IP_HALF_DUPLEX_MODE_SUPPORT)
|
|
|
+ if (
|
|
|
+ (LPSPI_IP_HALF_DUPLEX_TRANSMIT == State->ExternalDevice->DeviceParams->TransferType) ||
|
|
|
+ (LPSPI_IP_HALF_DUPLEX_RECEIVE == State->ExternalDevice->DeviceParams->TransferType)
|
|
|
+ )
|
|
|
+ {
|
|
|
+ Lpspi_Ip_TransferProcessHalfDuplex(Instance);
|
|
|
+ }
|
|
|
+ else
|
|
|
+ #endif
|
|
|
+ {
|
|
|
+ Lpspi_Ip_TransferProcess(Instance);
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* Driver has been initialized and received an unconfigured interrupt, clear all flags */
|
|
|
+ Base->SR &= LPSPI_IP_SR_W1C_MASK_U32;
|
|
|
+ }
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ /* the driver has not been initialized */
|
|
|
+ /* clear all flags */
|
|
|
+ Base->SR &= LPSPI_IP_SR_W1C_MASK_U32;
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+Lpspi_Ip_HwStatusType Lpspi_Ip_GetStatus(uint8 Instance)
|
|
|
+{
|
|
|
+ const Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_HwStatusType Status = LPSPI_IP_UNINIT;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(Instance < LPSPI_INSTANCE_COUNT);
|
|
|
+ #endif
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ if (State != NULL_PTR)
|
|
|
+ {
|
|
|
+ Status = State->Status;
|
|
|
+ }
|
|
|
+
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+
|
|
|
+#if (STD_ON == LPSPI_IP_DUAL_CLOCK_MODE)
|
|
|
+Lpspi_Ip_StatusType Lpspi_Ip_SetClockMode(uint8 Instance, Lpspi_Ip_DualClockModeType ClockMode)
|
|
|
+{
|
|
|
+ Lpspi_Ip_StateStructureType* State;
|
|
|
+ Lpspi_Ip_StatusType Status = LPSPI_IP_STATUS_SUCCESS;
|
|
|
+
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(Instance < LPSPI_INSTANCE_COUNT);
|
|
|
+ #endif
|
|
|
+ State = Lpspi_Ip_apxStateStructureArray[Instance];
|
|
|
+ #if (STD_ON == LPSPI_IP_DEV_ERROR_DETECT)
|
|
|
+ DevAssert(NULL_PTR != State);
|
|
|
+ #endif
|
|
|
+ /* Clock mode can be changed when no transfers are in progress. */
|
|
|
+ if (State->Status != LPSPI_IP_BUSY)
|
|
|
+ {
|
|
|
+ State->ClockMode = ClockMode;
|
|
|
+ }
|
|
|
+ else
|
|
|
+ {
|
|
|
+ Status = LPSPI_IP_STATUS_FAIL;
|
|
|
+ }
|
|
|
+ return Status;
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+#define SPI_STOP_SEC_CODE
|
|
|
+#include "Spi_MemMap.h"
|
|
|
+
|
|
|
+#ifdef __cplusplus
|
|
|
+}
|
|
|
+#endif
|
|
|
+
|
|
|
+/** @} */
|