BSzFL-Ethernet/Drivers/BSP/stm32h735g_discovery_audio.c
2025-03-18 10:54:44 +01:00

4236 lines
138 KiB
C

/**
******************************************************************************
* @file stm32h735g_discovery_audio.c
* @author MCD Application Team
* @brief This file provides the Audio driver for the STM32H735G_DISCO
* board.
@verbatim
How To use this driver:
-----------------------
+ This driver supports stm32h7xx devices on STM32H735G_DISCO (MB1520) boards.
+ Call the function BSP_AUDIO_OUT_Init() for AUDIO OUT initialization:
Instance: Select the output instance. It can only be 0 (SAI)
AudioInit: Audio Out structure to select the following parameters
- Device: Select the output device (headphone, speaker, hdmi ..)
- SampleRate: Select the output sample rate (8Khz .. 96Khz)
- BitsPerSample: Select the output resolution (16 or 32bits per sample)
- ChannelsNbr: Select the output channels number(1 for mono, 2 for stereo)
- Volume: Select the output volume(0% .. 100%)
This function configures all the hardware required for the audio application (codec, I2C, I2S ,SAI,
GPIOs, DMA and interrupt if needed). This function returns BSP_ERROR_NONE if configuration is OK.
If the returned value is different from BSP_ERROR_NONE or the function is stuck then the communication with
the codec or the MFX has failed (try to un-plug the power or reset device in this case).
User can update the I2S/SAI or the clock configurations by overriding the weak MX functions MX_I2S6_Init(),
MX_I2S6_ClockConfig(),MX_SAI1_Block_B_Init() and MX_SAI1_ClockConfig()
User can override the default MSP configuration and register his own MSP callbacks (defined at application level)
by calling BSP_AUDIO_OUT_RegisterMspCallbacks() function
User can restore the default MSP configuration by calling BSP_AUDIO_OUT_RegisterDefaultMspCallbacks()
To use these two functions, user have to enable USE_HAL_I2S_REGISTER_CALLBACKS and USE_HAL_SAI_REGISTER_CALLBACKS
within stm32h7xx_hal_conf.h file
+ Call the function BSP_AUDIO_OUT_Play() to play audio stream:
Instance: Select the output instance. It can only be 0 (SAI)
pBuf: pointer to the audio data file address
NbrOfBytes: Total size of the buffer to be sent in Bytes
+ Call the function BSP_AUDIO_OUT_Pause() to pause playing
+ Call the function BSP_AUDIO_OUT_Resume() to resume playing.
Note. After calling BSP_AUDIO_OUT_Pause() function for pause, only BSP_AUDIO_OUT_Resume() should be called
for resume (it is not allowed to call BSP_AUDIO_OUT_Play() in this case).
Note. This function should be called only when the audio file is played or paused (not stopped).
+ Call the function BSP_AUDIO_OUT_Stop() to stop playing.
+ Call the function BSP_AUDIO_OUT_Mute() to mute the player.
+ Call the function BSP_AUDIO_OUT_UnMute() to unmute the player.
+ Call the function BSP_AUDIO_OUT_IsMute() to get the mute state(BSP_AUDIO_MUTE_ENABLED or BSP_AUDIO_MUTE_DISABLED).
+ Call the function BSP_AUDIO_OUT_SetDevice() to update the AUDIO OUT device.
+ Call the function BSP_AUDIO_OUT_GetDevice() to get the AUDIO OUT device.
+ Call the function BSP_AUDIO_OUT_SetSampleRate() to update the AUDIO OUT sample rate.
+ Call the function BSP_AUDIO_OUT_GetSampleRate() to get the AUDIO OUT sample rate.
+ Call the function BSP_AUDIO_OUT_SetBitsPerSample() to update the AUDIO OUT resolution.
+ Call the function BSP_AUDIO_OUT_GetBitPerSample() to get the AUDIO OUT resolution.
+ Call the function BSP_AUDIO_OUT_SetChannelsNbr() to update the AUDIO OUT number of channels.
+ Call the function BSP_AUDIO_OUT_GetChannelsNbr() to get the AUDIO OUT number of channels.
+ Call the function BSP_AUDIO_OUT_SetVolume() to update the AUDIO OUT volume.
+ Call the function BSP_AUDIO_OUT_GetVolume() to get the AUDIO OUT volume.
+ Call the function BSP_AUDIO_OUT_GetState() to get the AUDIO OUT state.
+ BSP_AUDIO_OUT_SetDevice(), BSP_AUDIO_OUT_SetSampleRate(), BSP_AUDIO_OUT_SetBitsPerSample() and
BSP_AUDIO_OUT_SetChannelsNbr() cannot be called while the state is AUDIO_OUT_STATE_PLAYING.
+ For each mode, you may need to implement the relative callback functions into your code.
The Callback functions are named AUDIO_OUT_XXX_CallBack() and only their prototypes are declared in
the STM32H735G_DISCO_audio.h file. (refer to the example for more details on the callbacks implementations)
+ Call the function BSP_AUDIO_IN_Init() for AUDIO IN initialization:
Instance : Select the input instance. Can be 0 (SAI), 1 (I2S) or 2 (DFSDM)
AudioInit: Audio In structure to select the following parameters
- Device: Select the input device (analog, digital micx)
- SampleRate: Select the input sample rate (8Khz .. 96Khz)
- BitsPerSample: Select the input resolution (16 or 32bits per sample)
- ChannelsNbr: Select the input channels number(1 for mono, 2 for stereo)
- Volume: Select the input volume(0% .. 100%)
This function configures all the hardware required for the audio application (codec, I2C, SAI, I2S, DFSDM
GPIOs, DMA and interrupt if needed). This function returns BSP_ERROR_NONE if configuration is OK.
If the returned value is different from BSP_ERROR_NONE or the function is stuck then the communication with
the codec or the MFX has failed (try to un-plug the power or reset device in this case).
User can update the DFSDM/SAI or the clock configurations by overriding the weak MX functions MX_SAIx_Init(),
MX_SAIx_ClockConfig(), MX_DFSDMx_Init() and MX_DFSDMx_ClockConfig()
User can override the default MSP configuration and register his own MSP callbacks (defined at application level)
by calling BSP_AUDIO_IN_RegisterMspCallbacks() function
User can restore the default MSP configuration by calling BSP_AUDIO_IN_RegisterDefaultMspCallbacks()
To use these two functions, user have to enable USE_HAL_SAI_REGISTER_CALLBACKS and/or USE_HAL_DFSDM_REGISTER_CALLBACKS
within stm32h7xx_hal_conf.h file
+ Call the function BSP_AUDIO_IN_Record() to record audio stream. The recorded data are stored to user buffer in raw
(L, R, L, R ...)
Instance : Select the input instance. Can be 0 (SAI), 1 (I2S) or 2 (DFSDM)
pBuf: pointer to user buffer
NbrOfBytes: Total size of the buffer to be sent in Bytes
+ Call the function BSP_AUDIO_IN_Pause() to pause recording
+ Call the function BSP_AUDIO_IN_Resume() to resume recording.
+ Call the function BSP_AUDIO_IN_Stop() to stop recording.
+ Call the function BSP_AUDIO_IN_SetDevice() to update the AUDIO IN device.
+ Call the function BSP_AUDIO_IN_GetDevice() to get the AUDIO IN device.
+ Call the function BSP_AUDIO_IN_SetSampleRate() to update the AUDIO IN sample rate.
+ Call the function BSP_AUDIO_IN_GetSampleRate() to get the AUDIO IN sample rate.
+ Call the function BSP_AUDIO_IN_SetBitPerSample() to update the AUDIO IN resolution.
+ Call the function BSP_AUDIO_IN_GetBitPerSample() to get the AUDIO IN resolution.
+ Call the function BSP_AUDIO_IN_SetChannelsNbr() to update the AUDIO IN number of channels.
+ Call the function BSP_AUDIO_IN_GetChannelsNbr() to get the AUDIO IN number of channels.
+ Call the function BSP_AUDIO_IN_SetVolume() to update the AUDIO IN volume.
+ Call the function BSP_AUDIO_IN_GetVolume() to get the AUDIO IN volume.
+ Call the function BSP_AUDIO_IN_GetState() to get the AUDIO IN state.
+ Call the function BSP_AUDIO_IN_RecordChannels() to record audio stream. The recorded data are stored to user buffers separately
(L, L, ...) (R, R ...). User has to process his data at application level.
Instance : Select the input instance. Can be 2 (DFSDM)
pBuf: pointer to user buffers table
NbrOfBytes: Total size of the buffer to be sent in Bytes
+ Call the function BSP_AUDIO_IN_PauseChannels() to pause recording:
Instance : Select the input instance. Can be 2 (DFSDM)
Device: Select the input device (digital micX)
+ Call the function BSP_AUDIO_IN_ResumeChannels() to resume recording.
Instance : Select the input instance. Can be 2 (DFSDM)
Device: Select the input device (digital micX)
+ Call the function BSP_AUDIO_IN_StopChannels() to stop recording.
Instance : Select the input instance. Can be 2 (DFSDM)
Device: Select the input device (digital micX)
+ For each mode, you may need to implement the relative callback functions into your code.
The Callback functions are named AUDIO_IN_XXX_CallBack() and only their prototypes are declared in
the stm32h735g_discovery_audio.h file. (refer to the example for more details on the callbacks implementations)
+ The driver API and the callback functions are at the end of the stm32h735g_discovery_audio.h file.
@endverbatim
******************************************************************************
* @attention
*
* Copyright (c) 2019 STMicroelectronics.
* All rights reserved.
*
* This software is licensed under terms that can be found in the LICENSE file
* in the root directory of this software component.
* If no LICENSE file comes with this software, it is provided AS-IS.
*
******************************************************************************
*/
/* Includes ------------------------------------------------------------------*/
#include "stm32h735g_discovery_audio.h"
#include "stm32h735g_discovery_bus.h"
/** @addtogroup BSP
* @{
*/
/** @addtogroup STM32H735G_DISCO
* @{
*/
/** @defgroup STM32H735G_DISCO_AUDIO AUDIO
* @brief This file includes the low layer driver for cs42l51 Audio Codec
* available on STM32H735G_DISCO board(MB1332).
* @{
*/
/** @defgroup STM32H735G_DISCO_AUDIO_Private_Defines STM32H735G_DISCO_AUDIO Private Defines
* @{
*/
/**
* @}
*/
/** @defgroup STM32H735G_DISCO_AUDIO_Private_Macros AUDIO Private Macros
* @{
*/
/*### RECORD ###*/
#define DFSDM_OVER_SAMPLING(__FREQUENCY__) \
((__FREQUENCY__) == (AUDIO_FREQUENCY_8K)) ? (256U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_11K)) ? (256U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_16K)) ? (128U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_22K)) ? (128U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_32K)) ? (64U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_44K)) ? (64U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_48K)) ? (32U) : (16U)
#define DFSDM_CLOCK_DIVIDER(__FREQUENCY__) \
((__FREQUENCY__) == (AUDIO_FREQUENCY_8K)) ? (24U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_11K)) ? (4U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_16K)) ? (24U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_22K)) ? (4U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_32K)) ? (24U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_44K)) ? (4U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_48K)) ? (32U) : (32U)
#define DFSDM_FILTER_ORDER(__FREQUENCY__) \
((__FREQUENCY__) == (AUDIO_FREQUENCY_8K)) ? (DFSDM_FILTER_SINC3_ORDER) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_11K)) ? (DFSDM_FILTER_SINC3_ORDER) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_16K)) ? (DFSDM_FILTER_SINC3_ORDER) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_22K)) ? (DFSDM_FILTER_SINC3_ORDER) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_32K)) ? (DFSDM_FILTER_SINC4_ORDER) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_44K)) ? (DFSDM_FILTER_SINC3_ORDER) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_48K)) ? (DFSDM_FILTER_SINC4_ORDER) : (DFSDM_FILTER_SINC5_ORDER)
#define DFSDM_MIC_BIT_SHIFT(__FREQUENCY__) \
((__FREQUENCY__) == (AUDIO_FREQUENCY_8K)) ? (5U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_11K)) ? (6U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_16K)) ? (3U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_22K)) ? (3U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_32K)) ? (6U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_44K)) ? (0U) \
: ((__FREQUENCY__) == (AUDIO_FREQUENCY_48K)) ? (2U) : (2U)
/* Saturate the record PCM sample */
#define SaturaLH(N, L, H) (((N)<(L))?(L):(((N)>(H))?(H):(N)))
/**
* @}
*/
/** @defgroup STM32H735G_DISCO_AUDIO_Exported_Variables AUDIO Exported Variables
* @{
*/
/* Audio in and out component object */
void *Audio_CompObj = NULL;
/* Play */
SAI_HandleTypeDef haudio_out_sai = {0};
AUDIO_OUT_Ctx_t Audio_Out_Ctx[AUDIO_OUT_INSTANCES_NBR];
/* Record */
DFSDM_Filter_HandleTypeDef haudio_in_dfsdm_filter[DFSDM_MIC_NUMBER];
DFSDM_Channel_HandleTypeDef haudio_in_dfsdm_channel[DFSDM_MIC_NUMBER];
SAI_HandleTypeDef haudio_in_sai[2] = {0};
AUDIO_IN_Ctx_t Audio_In_Ctx[AUDIO_IN_INSTANCES_NBR];
/**
* @}
*/
/** @defgroup STM32H735G_DISCO_AUDIO_Private_Variables AUDIO Private Variables
* @{
*/
/* Audio in and out driver */
static AUDIO_Drv_t *Audio_Drv = NULL;
/* Recording DFSDM DMA handles */
static DMA_HandleTypeDef hDmaDfsdm[DFSDM_MIC_NUMBER];
/* Recording Buffer Trigger */
static __IO uint32_t RecBuffTrigger = 0;
static __IO uint32_t RecBuffHalf = 0;
ALIGN_32BYTES(static int32_t MicRecBuff[2][DEFAULT_AUDIO_IN_BUFFER_SIZE]);
static __IO uint32_t MicBuffIndex[DFSDM_MIC_NUMBER];
#if (USE_BSP_PDM_LIB_FEATURE > 0U)
/* PDM filters params */
static PDM_Filter_Handler_t PDM_FilterHandler[2];
static PDM_Filter_Config_t PDM_FilterConfig[2];
#endif
/**
* @}
*/
/** @defgroup STM32H735G_DISCO_AUDIO_Private_Function_Prototypes AUDIO Private Function Prototypes
* @{
*/
/* SAI Msp config */
static void SAI_MspInit(SAI_HandleTypeDef *hsai);
static void SAI_MspDeInit(SAI_HandleTypeDef *hsai);
/* SAI callbacks */
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
static void SAI_TxCpltCallback(SAI_HandleTypeDef *hsai);
static void SAI_TxHalfCpltCallback(SAI_HandleTypeDef *hsai);
static void SAI_RxCpltCallback(SAI_HandleTypeDef *hsai);
static void SAI_RxHalfCpltCallback(SAI_HandleTypeDef *hsai);
static void SAI_ErrorCallback(SAI_HandleTypeDef *hsai);
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
/* DFSDM Channel Msp config */
static void DFSDM_ChannelMspInit(DFSDM_Channel_HandleTypeDef *hDfsdmChannel);
static void DFSDM_ChannelMspDeInit(DFSDM_Channel_HandleTypeDef *hDfsdmChannel);
/* DFSDM Filter Msp config */
static void DFSDM_FilterMspInit(DFSDM_Filter_HandleTypeDef *hDfsdmFilter);
static void DFSDM_FilterMspDeInit(DFSDM_Filter_HandleTypeDef *hDfsdmFilter);
/* DFSDM Filter conversion callbacks */
#if (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1)
static void DFSDM_FilterRegConvHalfCpltCallback(DFSDM_Filter_HandleTypeDef *hdfsdm_filter);
static void DFSDM_FilterRegConvCpltCallback(DFSDM_Filter_HandleTypeDef *hdfsdm_filter);
#endif /* (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1) */
#if (USE_AUDIO_CODEC_WM8994 == 1)
static int32_t WM8994_Probe(void);
#endif
/**
* @}
*/
/** @defgroup STM32H735G_DISCO_AUDIO_OUT_Exported_Functions AUDIO OUT Exported Functions
* @{
*/
/**
* @brief Configures the audio peripherals.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param AudioInit AUDIO OUT init Structure
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_Init(uint32_t Instance, BSP_AUDIO_Init_t* AudioInit)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Fill Audio_Out_Ctx structure */
Audio_Out_Ctx[Instance].Device = AudioInit->Device;
Audio_Out_Ctx[Instance].Instance = Instance;
Audio_Out_Ctx[Instance].SampleRate = AudioInit->SampleRate;
Audio_Out_Ctx[Instance].BitsPerSample = AudioInit->BitsPerSample;
Audio_Out_Ctx[Instance].ChannelsNbr = AudioInit->ChannelsNbr;
Audio_Out_Ctx[Instance].Volume = AudioInit->Volume;
Audio_Out_Ctx[Instance].State = AUDIO_OUT_STATE_RESET;
#if (USE_AUDIO_CODEC_WM8994 == 1)
if (Audio_In_Ctx[0].State == AUDIO_IN_STATE_RESET)
{
if(WM8994_Probe() != BSP_ERROR_NONE)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
}
#endif
if(ret == BSP_ERROR_NONE)
{
/* PLL clock is set depending by the AudioFreq (44.1khz vs 48khz groups) */
if(MX_SAI1_ClockConfig(&haudio_out_sai, AudioInit->SampleRate) != HAL_OK)
{
ret = BSP_ERROR_CLOCK_FAILURE;
}
else
{
/* SAI data transfer preparation:
Prepare the Media to be used for the audio transfer from memory to SAI peripheral */
haudio_out_sai.Instance = AUDIO_OUT_SAIx;
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/* Register the MSP Callbacks */
if(Audio_Out_Ctx[Instance].IsMspCallbacksValid == 0U)
{
if(BSP_AUDIO_OUT_RegisterDefaultMspCallbacks(Instance) != BSP_ERROR_NONE)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
#else
SAI_MspInit(&haudio_out_sai);
#endif
}
if(ret == BSP_ERROR_NONE)
{
MX_SAI_Config mx_sai_config;
/* Prepare haudio_out_sai handle */
mx_sai_config.AudioFrequency = AudioInit->SampleRate;
mx_sai_config.AudioMode = SAI_MODEMASTER_TX;
mx_sai_config.ClockStrobing = SAI_CLOCKSTROBING_RISINGEDGE;
mx_sai_config.MonoStereoMode = (AudioInit->ChannelsNbr == 1U) ? SAI_MONOMODE : SAI_STEREOMODE;
if (AudioInit->BitsPerSample == AUDIO_RESOLUTION_32B)
{
mx_sai_config.DataSize = SAI_DATASIZE_32;
mx_sai_config.FrameLength = 128;
mx_sai_config.ActiveFrameLength = 64;
}
else
{
mx_sai_config.DataSize = SAI_DATASIZE_16;
mx_sai_config.FrameLength = 64;
mx_sai_config.ActiveFrameLength = 32;
}
mx_sai_config.OutputDrive = SAI_OUTPUTDRIVE_ENABLE;
mx_sai_config.Synchro = SAI_ASYNCHRONOUS;
mx_sai_config.SynchroExt = SAI_SYNCEXT_DISABLE;
switch(AudioInit->Device)
{
case AUDIO_OUT_DEVICE_SPK_HP:
case AUDIO_OUT_DEVICE_AUTO:
mx_sai_config.SlotActive = CODEC_AUDIOFRAME_SLOT_0123;
break;
case AUDIO_OUT_DEVICE_SPEAKER:
mx_sai_config.SlotActive = CODEC_AUDIOFRAME_SLOT_13;
break;
case AUDIO_OUT_DEVICE_HEADPHONE:
default:
mx_sai_config.SlotActive = CODEC_AUDIOFRAME_SLOT_02;
break;
}
/* SAI peripheral initialization: this __weak function can be redefined by the application */
if(MX_SAI1_Block_B_Init(&haudio_out_sai, &mx_sai_config) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/* Register SAI TC, HT and Error callbacks */
else if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_TX_COMPLETE_CB_ID, SAI_TxCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_TX_HALFCOMPLETE_CB_ID, SAI_TxHalfCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_ERROR_CB_ID, SAI_ErrorCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
#endif
if(ret == BSP_ERROR_NONE)
{
#if (USE_AUDIO_CODEC_WM8994 == 1)
WM8994_Init_t codec_init;
/* Fill codec_init structure */
codec_init.InputDevice = (Audio_In_Ctx[0].State == AUDIO_IN_STATE_RESET) ? WM8994_IN_NONE : WM8994_IN_LINE1;
switch (AudioInit->Device)
{
case AUDIO_OUT_DEVICE_SPK_HP:
codec_init.OutputDevice = WM8994_OUT_SPEAKER;
break;
case AUDIO_OUT_DEVICE_AUTO:
codec_init.OutputDevice = WM8994_OUT_AUTO;
break;
case AUDIO_OUT_DEVICE_SPEAKER:
codec_init.OutputDevice = WM8994_OUT_SPEAKER;
break;
case AUDIO_OUT_DEVICE_HEADPHONE:
default:
codec_init.OutputDevice = WM8994_OUT_HEADPHONE;
break;
}
codec_init.Frequency = AudioInit->SampleRate;
codec_init.Resolution = (AudioInit->BitsPerSample == AUDIO_RESOLUTION_32B) ? WM8994_RESOLUTION_32b : WM8994_RESOLUTION_16b;
/* Convert volume before sending to the codec */
codec_init.Volume = VOLUME_OUT_CONVERT(AudioInit->Volume);
/* Initialize the codec internal registers */
if(Audio_Drv->Init(Audio_CompObj, &codec_init) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
#endif
/* Update BSP AUDIO OUT state */
Audio_Out_Ctx[Instance].State = AUDIO_OUT_STATE_STOP;
}
}
}
}
return ret;
}
/**
* @brief De-initializes the audio out peripheral.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @retval None
*/
int32_t BSP_AUDIO_OUT_DeInit(uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 0U)
SAI_MspDeInit(&haudio_out_sai);
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 0U) */
/* Initialize the haudio_out_sai Instance parameter */
haudio_out_sai.Instance = AUDIO_OUT_SAIx;
/* Call the Media layer stop function */
if(Audio_Drv->DeInit(Audio_CompObj) != 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
else if(HAL_SAI_DeInit(&haudio_out_sai) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
/* Update BSP AUDIO OUT state */
Audio_Out_Ctx[Instance].State = AUDIO_OUT_STATE_RESET;
}
}
/* Return BSP status */
return ret;
}
/**
* @brief Initializes the Audio Codec audio out instance (SAI).
* @param hsai SAI handle
* @param MXConfig SAI confiruration structure
* @note Being __weak it can be overwritten by the application
* @retval HAL status
*/
__weak HAL_StatusTypeDef MX_SAI1_Block_B_Init(SAI_HandleTypeDef* hsai, MX_SAI_Config *MXConfig)
{
HAL_StatusTypeDef ret = HAL_OK;
/* Disable SAI peripheral to allow access to SAI internal registers */
__HAL_SAI_DISABLE(hsai);
/* Configure SAI1_Block_B */
hsai->Init.MonoStereoMode = MXConfig->MonoStereoMode;
hsai->Init.AudioFrequency = MXConfig->AudioFrequency;
hsai->Init.AudioMode = MXConfig->AudioMode;
hsai->Init.NoDivider = SAI_MASTERDIVIDER_ENABLE;
hsai->Init.Protocol = SAI_FREE_PROTOCOL;
hsai->Init.DataSize = MXConfig->DataSize;
hsai->Init.FirstBit = SAI_FIRSTBIT_MSB;
hsai->Init.ClockStrobing = MXConfig->ClockStrobing;
hsai->Init.Synchro = MXConfig->Synchro;
hsai->Init.OutputDrive = MXConfig->OutputDrive;
hsai->Init.FIFOThreshold = SAI_FIFOTHRESHOLD_1QF;
hsai->Init.SynchroExt = MXConfig->SynchroExt;
hsai->Init.CompandingMode = SAI_NOCOMPANDING;
hsai->Init.TriState = SAI_OUTPUT_NOTRELEASED;
hsai->Init.Mckdiv = 0;
hsai->Init.MckOutput = SAI_MCK_OUTPUT_ENABLE;
hsai->Init.MckOverSampling = SAI_MCK_OVERSAMPLING_DISABLE;
hsai->Init.PdmInit.Activation = DISABLE;
/* Configure SAI_Block_x Frame */
hsai->FrameInit.FrameLength = MXConfig->FrameLength;
hsai->FrameInit.ActiveFrameLength = MXConfig->ActiveFrameLength;
hsai->FrameInit.FSDefinition = SAI_FS_CHANNEL_IDENTIFICATION;
hsai->FrameInit.FSPolarity = SAI_FS_ACTIVE_LOW;
hsai->FrameInit.FSOffset = SAI_FS_BEFOREFIRSTBIT;
/* Configure SAI Block_x Slot */
hsai->SlotInit.FirstBitOffset = 0;
if ((MXConfig->DataSize == AUDIO_RESOLUTION_24B) || (MXConfig->DataSize == AUDIO_RESOLUTION_32B))
{
hsai->SlotInit.SlotSize = SAI_SLOTSIZE_32B;
}
else
{
hsai->SlotInit.SlotSize = SAI_SLOTSIZE_16B;
}
hsai->SlotInit.SlotNumber = 4;
hsai->SlotInit.SlotActive = MXConfig->SlotActive;
if(HAL_SAI_Init(hsai) != HAL_OK)
{
ret = HAL_ERROR;
}
__HAL_SAI_ENABLE(hsai);
return ret;
}
/**
* @brief SAI clock Config.
* @param hsai SAI handle
* @param SampleRate Audio frequency used to play the audio stream.
* @note This API is called by BSP_AUDIO_OUT_Init() and BSP_AUDIO_OUT_SetFrequency()
* Being __weak it can be overwritten by the application
* @retval HAL status
*/
__weak HAL_StatusTypeDef MX_SAI1_ClockConfig(SAI_HandleTypeDef *hsai, uint32_t SampleRate)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hsai);
RCC_PeriphCLKInitTypeDef rcc_ex_clk_init_struct;
HAL_RCCEx_GetPeriphCLKConfig(&rcc_ex_clk_init_struct);
/* Set the PLL configuration according to the audio frequency */
/* Set the PLL configuration according to the audio frequency */
if((SampleRate == AUDIO_FREQUENCY_11K) || (SampleRate == AUDIO_FREQUENCY_22K) || (SampleRate == AUDIO_FREQUENCY_44K))
{
rcc_ex_clk_init_struct.PLL2.PLL2P = 36;
rcc_ex_clk_init_struct.PLL2.PLL2Q = 36;
}
else /* AUDIO_FREQUENCY_8K, AUDIO_FREQUENCY_16K, AUDIO_FREQUENCY_32K, AUDIO_FREQUENCY_48K, AUDIO_FREQUENCY_96K */
{
rcc_ex_clk_init_struct.PLL2.PLL2P = 8;
rcc_ex_clk_init_struct.PLL2.PLL2Q = 8;
}
rcc_ex_clk_init_struct.PLL2.PLL2N = 80;
rcc_ex_clk_init_struct.PeriphClockSelection = RCC_PERIPHCLK_SAI1;
rcc_ex_clk_init_struct.Sai1ClockSelection = RCC_SAI1CLKSOURCE_PLL2;
rcc_ex_clk_init_struct.PLL2.PLL2R = 2;
rcc_ex_clk_init_struct.PLL2.PLL2M = 5;
rcc_ex_clk_init_struct.PLL2.PLL2RGE = RCC_PLL2VCIRANGE_0;
rcc_ex_clk_init_struct.PLL2.PLL2VCOSEL = RCC_PLL2VCOMEDIUM;
rcc_ex_clk_init_struct.PLL2.PLL2FRACN = 0;
return HAL_RCCEx_PeriphCLKConfig(&rcc_ex_clk_init_struct);
}
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/**
* @brief Default BSP AUDIO OUT Msp Callbacks
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_RegisterDefaultMspCallbacks (uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
__HAL_SAI_RESET_HANDLE_STATE(&haudio_out_sai);
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/* Register MspInit/MspDeInit Callbacks */
if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_MSPINIT_CB_ID,SAI_MspInit) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_MSPDEINIT_CB_ID,SAI_MspDeInit) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
Audio_Out_Ctx[Instance].IsMspCallbacksValid = 1;
}
#endif
}
/* Return BSP status */
return ret;
}
/**
* @brief BSP AUDIO OUT Msp Callback registering
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param CallBacks pointer to MspInit/MspDeInit callbacks functions
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_RegisterMspCallbacks (uint32_t Instance, BSP_AUDIO_OUT_Cb_t *CallBacks)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
__HAL_SAI_RESET_HANDLE_STATE(&haudio_out_sai);
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/* Register MspInit/MspDeInit Callbacks */
if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_MSPINIT_CB_ID, CallBacks->pMspSaiInitCb) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_MSPDEINIT_CB_ID, CallBacks->pMspSaiDeInitCb) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
Audio_Out_Ctx[Instance].IsMspCallbacksValid = 1;
}
#endif
}
/* Return BSP status */
return ret;
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
/**
* @brief Starts playing audio stream from a data buffer for a determined size.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param pData pointer on data address
* @param NbrOfBytes Size of total samples in bytes
* BitsPerSample: 16 or 32
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_Play(uint32_t Instance, uint8_t* pData, uint32_t NbrOfBytes)
{
int32_t ret = BSP_ERROR_NONE;
if((Instance >= AUDIO_OUT_INSTANCES_NBR) || (((NbrOfBytes / (Audio_Out_Ctx[Instance].BitsPerSample/8U)) > 0xFFFFU)))
{
ret = BSP_ERROR_WRONG_PARAM;
}
else if (Audio_Out_Ctx[Instance].State != AUDIO_OUT_STATE_STOP)
{
ret = BSP_ERROR_BUSY;
}
else if(HAL_SAI_Transmit_DMA(&haudio_out_sai, pData, (uint16_t)(NbrOfBytes /(Audio_Out_Ctx[Instance].BitsPerSample/8U))) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(Audio_Drv->Play(Audio_CompObj) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
else
{
/* Update BSP AUDIO OUT state */
Audio_Out_Ctx[Instance].State = AUDIO_OUT_STATE_PLAYING;
}
/* Return BSP status */
return ret;
}
/**
* @brief This function Pauses the audio file stream. In case
* of using DMA, the DMA Pause feature is used.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @note When calling BSP_AUDIO_OUT_Pause() function for pause, only
* BSP_AUDIO_OUT_Resume() function should be called for resume (use of BSP_AUDIO_OUT_Play()
* function for resume could lead to unexpected behavior).
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_Pause(uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
if (Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio out state */
else if (Audio_Out_Ctx[Instance].State != AUDIO_OUT_STATE_PLAYING)
{
ret = BSP_ERROR_BUSY;
}
/* Call the audio codec pause function */
else if (Audio_Drv->Pause(Audio_CompObj) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}/* Pause DMA transfer of audio samples towards the serial audio interface */
else if(HAL_SAI_DMAPause(&haudio_out_sai) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
/* Update BSP AUDIO OUT state */
Audio_Out_Ctx[Instance].State = AUDIO_OUT_STATE_PAUSE;
}
/* Return BSP status */
return ret;
}
/**
* @brief Resumes the audio file stream.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @note When calling BSP_AUDIO_OUT_Pause() function for pause, only
* BSP_AUDIO_OUT_Resume() function should be called for resume (use of BSP_AUDIO_OUT_Play()
* function for resume could lead to unexpected behavior).
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_Resume(uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
if (Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio out state */
else if (Audio_Out_Ctx[Instance].State != AUDIO_OUT_STATE_PAUSE)
{
ret = BSP_ERROR_BUSY;
}
/* Call the audio codec resume function */
else if (Audio_Drv->Resume(Audio_CompObj) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}/* Resume DMA transfer of audio samples towards the serial audio interface */
else if(HAL_SAI_DMAResume(&haudio_out_sai) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
/* Update BSP AUDIO OUT state */
Audio_Out_Ctx[Instance].State = AUDIO_OUT_STATE_PLAYING;
}
/* Return BSP status */
return ret;
}
/**
* @brief Stops audio playing and Power down the Audio Codec.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_Stop(uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
if (Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio out state */
else if (Audio_Out_Ctx[Instance].State == AUDIO_OUT_STATE_STOP)
{
/* Nothing to do */
}
else if ((Audio_Out_Ctx[Instance].State != AUDIO_OUT_STATE_PLAYING) &&
(Audio_Out_Ctx[Instance].State != AUDIO_OUT_STATE_PAUSE))
{
ret = BSP_ERROR_BUSY;
}
/* Call the Media layer stop function */
else if(Audio_Drv->Stop(Audio_CompObj, CODEC_PDWN_SW) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
else if(HAL_SAI_DMAStop(&haudio_out_sai)!= HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
/* Update BSP AUDIO OUT state */
Audio_Out_Ctx[Instance].State = AUDIO_OUT_STATE_STOP;
}
/* Return BSP status */
return ret;
}
/**
* @brief Controls the current audio volume level.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param Volume Volume level to be set in percentage from 0% to 100% (0 for
* Mute and 100 for Max volume level).
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_SetVolume(uint32_t Instance, uint32_t Volume)
{
int32_t ret = BSP_ERROR_NONE;
if ((Instance >= AUDIO_OUT_INSTANCES_NBR) || (Volume > 100U))
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Call the codec volume control function with converted volume value */
if(Audio_Drv->SetVolume(Audio_CompObj, AUDIO_VOLUME_OUTPUT, Volume) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
else if(Volume == 0U)
{
/* Update Mute State */
Audio_Out_Ctx[Instance].IsMute = BSP_AUDIO_MUTE_ENABLED;
}
else
{
/* Update Mute State */
Audio_Out_Ctx[Instance].IsMute = BSP_AUDIO_MUTE_DISABLED;
}
Audio_Out_Ctx[Instance].Volume = Volume;
}
/* Return BSP status */
return ret;
}
/**
* @brief Get the current audio volume level.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param Volume pointer to volume to be returned
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_GetVolume(uint32_t Instance, uint32_t *Volume)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else if (Audio_Out_Ctx[Instance].State == AUDIO_OUT_STATE_RESET)
{
ret = BSP_ERROR_BUSY;
}
else
{
*Volume = Audio_Out_Ctx[Instance].Volume;
}
/* Return BSP status */
return ret;
}
/**
* @brief Enables the MUTE
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_Mute(uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
if (Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio out state */
else if (Audio_Out_Ctx[Instance].State == AUDIO_OUT_STATE_RESET)
{
ret = BSP_ERROR_BUSY;
}
/* Check audio out mute status */
else if (Audio_Out_Ctx[Instance].IsMute == 1U)
{
/* Nothing to do */
}
/* Call the audio codec mute function */
else if (Audio_Drv->SetMute(Audio_CompObj, CODEC_MUTE_ON) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
else
{
/* Update audio out mute status */
Audio_Out_Ctx[Instance].IsMute = 1U;
}
/* Return BSP status */
return ret;
}
/**
* @brief Disables the MUTE mode
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_UnMute(uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
if (Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio out state */
else if (Audio_Out_Ctx[Instance].State == AUDIO_OUT_STATE_RESET)
{
ret = BSP_ERROR_BUSY;
}
/* Check audio out mute status */
else if (Audio_Out_Ctx[Instance].IsMute == 0U)
{
/* Nothing to do */
}
/* Call the audio codec mute function */
else if (Audio_Drv->SetMute(Audio_CompObj, CODEC_MUTE_OFF) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
else
{
/* Update audio out mute status */
Audio_Out_Ctx[Instance].IsMute = 0U;
}
/* Return BSP status */
return ret;
}
/**
* @brief Check whether the MUTE mode is enabled or not
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param IsMute pointer to mute state
* @retval Mute status
*/
int32_t BSP_AUDIO_OUT_IsMute(uint32_t Instance, uint32_t *IsMute)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else if (Audio_Out_Ctx[Instance].State == AUDIO_OUT_STATE_RESET)
{
ret = BSP_ERROR_BUSY;
}
else
{
*IsMute = Audio_Out_Ctx[Instance].IsMute;
}
/* Return BSP status */
return ret;
}
/**
* @brief Switch dynamically (while audio file is played) the output target
* (speaker or headphone).
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param Device The audio output device
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_SetDevice(uint32_t Instance, uint32_t Device)
{
int32_t ret = BSP_ERROR_NONE;
UNUSED(Device);
if (Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio out state */
else if (Audio_Out_Ctx[Instance].State != AUDIO_OUT_STATE_STOP)
{
ret = BSP_ERROR_BUSY;
}
else
{
/* Nothing to do because there is only one device (AUDIO_OUT_DEVICE_HEADPHONE) */
}
/* Return BSP status */
return ret;
}
/**
* @brief Get the Output Device
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param Device The audio output device
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_GetDevice(uint32_t Instance, uint32_t *Device)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else if (Audio_Out_Ctx[Instance].State == AUDIO_OUT_STATE_RESET)
{
ret = BSP_ERROR_BUSY;
}
else
{
/* Get Audio_Out_Ctx Device */
*Device = Audio_Out_Ctx[Instance].Device;
}
/* Return BSP status */
return ret;
}
/**
* @brief Updates the audio frequency.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param SampleRate Audio frequency used to play the audio stream.
* @note This API should be called after the BSP_AUDIO_OUT_Init() to adjust the
* audio frequency.
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_SetSampleRate(uint32_t Instance, uint32_t SampleRate)
{
int32_t ret = BSP_ERROR_NONE;
if (Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio out state */
else if (Audio_Out_Ctx[Instance].State != AUDIO_OUT_STATE_STOP)
{
ret = BSP_ERROR_BUSY;
}
/* Check if record on instance 0 is on going and corresponding sample rate */
else if ((Audio_In_Ctx[0].State != AUDIO_IN_STATE_RESET) &&
(Audio_In_Ctx[0].SampleRate != SampleRate))
{
ret = BSP_ERROR_FEATURE_NOT_SUPPORTED;
}/* Check if sample rate is modified */
else if (Audio_Out_Ctx[Instance].SampleRate == SampleRate)
{
/* Nothing to do */
}
else
{
/* Update the SAI audio frequency configuration */
haudio_out_sai.Init.AudioFrequency = SampleRate;
/* PLL clock is set depending by the AudioFreq (44.1khz vs 48khz groups) */
if(MX_SAI1_ClockConfig(&haudio_out_sai, SampleRate) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_Init(&haudio_out_sai) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/* Register SAI TC, HT and Error callbacks */
if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_TX_COMPLETE_CB_ID, SAI_TxCpltCallback) != HAL_OK)
{
return BSP_ERROR_PERIPH_FAILURE;
}
if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_TX_HALFCOMPLETE_CB_ID, SAI_TxHalfCpltCallback) != HAL_OK)
{
return BSP_ERROR_PERIPH_FAILURE;
}
if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_ERROR_CB_ID, SAI_ErrorCallback) != HAL_OK)
{
return BSP_ERROR_PERIPH_FAILURE;
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
/* Store new sample rate */
Audio_Out_Ctx[Instance].SampleRate = SampleRate;
}
}
/* Return BSP status */
return ret;
}
/**
* @brief Get the audio frequency.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param SampleRate Audio frequency used to play the audio stream.
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_GetSampleRate(uint32_t Instance, uint32_t *SampleRate)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else if (Audio_Out_Ctx[Instance].State == AUDIO_OUT_STATE_RESET)
{
ret = BSP_ERROR_BUSY;
}
else
{
*SampleRate = Audio_Out_Ctx[Instance].SampleRate;
}
/* Return BSP status */
return ret;
}
/**
* @brief Get the audio Resolution.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param BitsPerSample Audio Resolution used to play the audio stream.
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_SetBitsPerSample(uint32_t Instance, uint32_t BitsPerSample)
{
int32_t ret = BSP_ERROR_NONE;
if (Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else if ((Instance == 0U) && ((BitsPerSample == AUDIO_RESOLUTION_24B) || (BitsPerSample == AUDIO_RESOLUTION_8B)))
{
ret = BSP_ERROR_FEATURE_NOT_SUPPORTED;
}
else if ((Instance == 0U) && (Audio_In_Ctx[0].State != AUDIO_IN_STATE_RESET) &&
(Audio_In_Ctx[0].BitsPerSample != BitsPerSample))
{
ret = BSP_ERROR_FEATURE_NOT_SUPPORTED;
}
/* Check audio out state */
else if (Audio_Out_Ctx[Instance].State != AUDIO_OUT_STATE_STOP)
{
ret = BSP_ERROR_BUSY;
}
else
{
/* Store new bits per sample on audio out context */
Audio_Out_Ctx[Instance].BitsPerSample = BitsPerSample;
/* Update data size, frame length and active frame length parameters of SAI handle */
if (BitsPerSample == AUDIO_RESOLUTION_32B)
{
haudio_out_sai.Init.DataSize = SAI_DATASIZE_32;
haudio_out_sai.FrameInit.FrameLength = 128;
haudio_out_sai.FrameInit.ActiveFrameLength = 64;
}
else
{
haudio_out_sai.Init.DataSize = SAI_DATASIZE_16;
haudio_out_sai.FrameInit.FrameLength = 64;
haudio_out_sai.FrameInit.ActiveFrameLength = 32;
}
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/* Update SAI state only to keep current MSP functions */
haudio_out_sai.State = HAL_SAI_STATE_RESET;
#else
SAI_MspInit(&haudio_out_sai);
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
/* Re-initialize SAI1 with new parameters */
if (HAL_SAI_Init(&haudio_out_sai) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/* Register SAI TC, HT and Error callbacks */
else if (HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_TX_COMPLETE_CB_ID, SAI_TxCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if (HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_TX_HALFCOMPLETE_CB_ID, SAI_TxHalfCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
if (HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_ERROR_CB_ID, SAI_ErrorCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
}
return ret;
}
/**
* @brief Get the audio Resolution.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param BitsPerSample Audio Resolution used to play the audio stream.
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_GetBitsPerSample(uint32_t Instance, uint32_t *BitsPerSample)
{
int32_t ret = BSP_ERROR_NONE;
if (Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio out state */
else if (Audio_Out_Ctx[Instance].State == AUDIO_OUT_STATE_RESET)
{
ret = BSP_ERROR_BUSY;
}
/* Get the current bits per sample of audio out stream */
else
{
*BitsPerSample = Audio_Out_Ctx[Instance].BitsPerSample;
}
return ret;
}
/**
* @brief Set the audio Channels number.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param ChannelNbr Audio Channels number used to play the audio stream (It can only be 2U for I2S Instance)
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_SetChannelsNbr(uint32_t Instance, uint32_t ChannelNbr)
{
int32_t ret = BSP_ERROR_NONE;
if (Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio out state */
else if (Audio_Out_Ctx[Instance].State != AUDIO_OUT_STATE_STOP)
{
ret = BSP_ERROR_BUSY;
}
else
{
/* Update mono or stereo mode of SAI handle */
haudio_out_sai.Init.MonoStereoMode = (ChannelNbr == 1U) ? SAI_MONOMODE : SAI_STEREOMODE;
/* Re-initialize SAI1 with new parameter */
if (HAL_SAI_Init(&haudio_out_sai) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/* Register SAI TC, HT and Error callbacks */
else if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_TX_COMPLETE_CB_ID, SAI_TxCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_TX_HALFCOMPLETE_CB_ID, SAI_TxHalfCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_ERROR_CB_ID, SAI_ErrorCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
else
{
/* Store new Channel number */
Audio_Out_Ctx[Instance].ChannelsNbr = ChannelNbr;
}
}
/* Return BSP status */
return ret;
}
/**
* @brief Get the audio Channels number.
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param ChannelNbr Audio Channels number used to play the audio stream.
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_GetChannelsNbr(uint32_t Instance, uint32_t *ChannelNbr)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else if (Audio_Out_Ctx[Instance].State == AUDIO_OUT_STATE_RESET)
{
ret = BSP_ERROR_BUSY;
}
else
{
/* Get the audio Channels number */
*ChannelNbr = Audio_Out_Ctx[Instance].ChannelsNbr;
}
/* Return BSP status */
return ret;
}
/**
* @brief Get Audio Out state
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @param State Audio Out state
* @retval BSP status
*/
int32_t BSP_AUDIO_OUT_GetState(uint32_t Instance, uint32_t *State)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_OUT_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Return audio Output State */
*State = Audio_Out_Ctx[Instance].State;
}
/* Return BSP status */
return ret;
}
/**
* @brief This function handles Audio Out DMA interrupt requests.
* @param Instance Audio OUT instance
* @retval None
*/
void BSP_AUDIO_OUT_IRQHandler(uint32_t Instance)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(Instance);
HAL_DMA_IRQHandler(haudio_out_sai.hdmatx);
}
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 0) || !defined (USE_HAL_SAI_REGISTER_CALLBACKS)
/**
* @brief Tx Transfer completed callbacks.
* @param hsai SAI handle
* @retval None
*/
void HAL_SAI_TxCpltCallback(SAI_HandleTypeDef *hsai)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hsai);
/* Manage the remaining file size and new address offset: This function
should be coded by user (its prototype is already declared in stm32h735g_discovery_audio.h) */
BSP_AUDIO_OUT_TransferComplete_CallBack(0);
}
/**
* @brief Tx Half Transfer completed callbacks.
* @param hsai SAI handle
* @retval None
*/
void HAL_SAI_TxHalfCpltCallback(SAI_HandleTypeDef *hsai)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hsai);
/* Manage the remaining file size and new address offset: This function
should be coded by user (its prototype is already declared in stm32h735g_discovery_audio.h) */
BSP_AUDIO_OUT_HalfTransfer_CallBack(0);
}
/**
* @brief SAI error callbacks.
* @param hsai SAI handle
* @retval None
*/
void HAL_SAI_ErrorCallback(SAI_HandleTypeDef *hsai)
{
if(hsai->Instance == AUDIO_OUT_SAIx)
{
BSP_AUDIO_OUT_Error_CallBack(0);
}
else
{
BSP_AUDIO_IN_Error_CallBack(0);
}
}
#endif
/**
* @brief Manages the DMA full Transfer complete event
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @retval None
*/
__weak void BSP_AUDIO_OUT_TransferComplete_CallBack(uint32_t Instance)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(Instance);
}
/**
* @brief Manages the DMA Half Transfer complete event
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @retval None
*/
__weak void BSP_AUDIO_OUT_HalfTransfer_CallBack(uint32_t Instance)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(Instance);
}
/**
* @brief Manages the DMA FIFO error event
* @param Instance AUDIO OUT Instance. It can only be 0 (SAI)
* @retval None
*/
__weak void BSP_AUDIO_OUT_Error_CallBack(uint32_t Instance)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(Instance);
}
/**
* @}
*/
/** @defgroup STM32H735G_DISCO_AUDIO_OUT_Private_Functions AUDIO OUT Private Functions
* @{
*/
#if (USE_AUDIO_CODEC_WM8994 == 1)
/**
* @brief Register Bus IOs if component ID is OK
* @retval error status
*/
static int32_t WM8994_Probe(void)
{
int32_t ret = BSP_ERROR_NONE;
WM8994_IO_t IOCtx;
static WM8994_Object_t WM8994Obj;
uint32_t id;
/* Configure the audio driver */
IOCtx.Address = AUDIO_I2C_ADDRESS;
IOCtx.Init = BSP_I2C4_Init;
IOCtx.DeInit = BSP_I2C4_DeInit;
IOCtx.ReadReg = BSP_I2C4_ReadReg16;
IOCtx.WriteReg = BSP_I2C4_WriteReg16;
IOCtx.GetTick = BSP_GetTick;
if(WM8994_RegisterBusIO (&WM8994Obj, &IOCtx) != WM8994_OK)
{
ret = BSP_ERROR_BUS_FAILURE;
}
else
{
/* Reset the codec */
if(WM8994_Reset(&WM8994Obj) != WM8994_OK)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
else if(WM8994_ReadID(&WM8994Obj, &id) != WM8994_OK)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
else if(id != WM8994_ID)
{
ret = BSP_ERROR_UNKNOWN_COMPONENT;
}
else
{
Audio_Drv = (AUDIO_Drv_t *) &WM8994_Driver;
Audio_CompObj = &WM8994Obj;
}
}
return ret;
}
#endif
/**
* @brief Initialize BSP_AUDIO_OUT MSP.
* @param hsai SAI handle
* @retval None
*/
static void SAI_MspInit(SAI_HandleTypeDef *hsai)
{
GPIO_InitTypeDef gpio_init_structure;
static DMA_HandleTypeDef hdma_sai_tx, hdma_sai_rx;
/* Enable SAI clock */
AUDIO_OUT_SAIx_CLK_ENABLE();
/* Enable GPIO clock */
AUDIO_OUT_SAIx_MCLK_ENABLE();
AUDIO_OUT_SAIx_SCK_ENABLE();
AUDIO_OUT_SAIx_SD_ENABLE();
AUDIO_OUT_SAIx_FS_ENABLE();
/* CODEC_SAI pins configuration: FS, SCK, MCK and SD pins ------------------*/
gpio_init_structure.Pin = AUDIO_OUT_SAIx_FS_PIN;
gpio_init_structure.Mode = GPIO_MODE_AF_PP;
gpio_init_structure.Pull = GPIO_NOPULL;
gpio_init_structure.Speed = GPIO_SPEED_FREQ_VERY_HIGH;
gpio_init_structure.Alternate = AUDIO_OUT_SAIx_FS_AF;
HAL_GPIO_Init(AUDIO_OUT_SAIx_FS_GPIO_PORT, &gpio_init_structure);
gpio_init_structure.Pin = AUDIO_OUT_SAIx_SCK_PIN;
gpio_init_structure.Alternate = AUDIO_OUT_SAIx_SCK_AF;
HAL_GPIO_Init(AUDIO_OUT_SAIx_SCK_GPIO_PORT, &gpio_init_structure);
gpio_init_structure.Pin = AUDIO_OUT_SAIx_SD_PIN;
gpio_init_structure.Alternate = AUDIO_OUT_SAIx_SD_AF;
HAL_GPIO_Init(AUDIO_OUT_SAIx_SD_GPIO_PORT, &gpio_init_structure);
if(hsai->Instance != AUDIO_IN_SAI_PDMx)
{
if(haudio_in_sai[0].State != HAL_SAI_STATE_READY)
{
AUDIO_OUT_SAIx_MCLK_ENABLE();
gpio_init_structure.Pin = AUDIO_OUT_SAIx_MCLK_PIN;
gpio_init_structure.Alternate = AUDIO_OUT_SAIx_MCLK_AF;
HAL_GPIO_Init(AUDIO_OUT_SAIx_MCLK_GPIO_PORT, &gpio_init_structure);
}
}
if(hsai->Instance == AUDIO_OUT_SAIx)
{
/* Enable the DMA clock */
AUDIO_OUT_SAIx_DMAx_CLK_ENABLE();
/* Configure the hdma_saiTx handle parameters */
hdma_sai_tx.Init.Request = AUDIO_OUT_SAIx_DMAx_REQUEST;
hdma_sai_tx.Init.Direction = DMA_MEMORY_TO_PERIPH;
hdma_sai_tx.Init.PeriphInc = DMA_PINC_DISABLE;
hdma_sai_tx.Init.MemInc = DMA_MINC_ENABLE;
if (Audio_Out_Ctx[0].BitsPerSample == AUDIO_RESOLUTION_16B)
{
hdma_sai_tx.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;
hdma_sai_tx.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
}
else /* AUDIO_RESOLUTION_32B */
{
hdma_sai_tx.Init.PeriphDataAlignment = DMA_PDATAALIGN_WORD;
hdma_sai_tx.Init.MemDataAlignment = DMA_MDATAALIGN_WORD;
}
hdma_sai_tx.Init.Mode = DMA_CIRCULAR;
hdma_sai_tx.Init.Priority = DMA_PRIORITY_HIGH;
hdma_sai_tx.Init.FIFOMode = DMA_FIFOMODE_ENABLE;
hdma_sai_tx.Init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;
hdma_sai_tx.Init.MemBurst = DMA_MBURST_SINGLE;
hdma_sai_tx.Init.PeriphBurst = DMA_PBURST_SINGLE;
hdma_sai_tx.Instance = AUDIO_OUT_SAIx_DMAx_STREAM;
/* Associate the DMA handle */
__HAL_LINKDMA(hsai, hdmatx, hdma_sai_tx);
/* Deinitialize the Stream for new transfer */
(void)HAL_DMA_DeInit(&hdma_sai_tx);
/* Configure the DMA Stream */
(void)HAL_DMA_Init(&hdma_sai_tx);
/* SAI DMA IRQ Channel configuration */
HAL_NVIC_SetPriority(AUDIO_OUT_SAIx_DMAx_IRQ, BSP_AUDIO_OUT_IT_PRIORITY, 0);
HAL_NVIC_EnableIRQ(AUDIO_OUT_SAIx_DMAx_IRQ);
}
/* Audio In Msp initialization */
if(hsai->Instance == AUDIO_IN_SAIx)
{
/* Enable SAI clock */
AUDIO_IN_SAIx_CLK_ENABLE();
/* Enable SD GPIO clock */
AUDIO_IN_SAIx_SD_ENABLE();
/* CODEC_SAI pin configuration: SD pin */
gpio_init_structure.Pin = AUDIO_IN_SAIx_SD_PIN;
gpio_init_structure.Mode = GPIO_MODE_AF_PP;
gpio_init_structure.Pull = GPIO_NOPULL;
gpio_init_structure.Speed = GPIO_SPEED_FREQ_HIGH;
gpio_init_structure.Alternate = AUDIO_IN_SAIx_AF;
HAL_GPIO_Init(AUDIO_IN_SAIx_SD_GPIO_PORT, &gpio_init_structure);
/* Enable the DMA clock */
AUDIO_IN_SAIx_DMAx_CLK_ENABLE();
/* Configure the hdma_sai_rx handle parameters */
hdma_sai_rx.Instance = AUDIO_IN_SAIx_DMAx_STREAM;
hdma_sai_rx.Init.Request = AUDIO_IN_SAIx_DMAx_REQUEST;
hdma_sai_rx.Init.Mode = DMA_CIRCULAR;
hdma_sai_rx.Init.Priority = DMA_PRIORITY_HIGH;
hdma_sai_rx.Init.Direction = DMA_PERIPH_TO_MEMORY;
hdma_sai_rx.Init.PeriphInc = DMA_PINC_DISABLE;
hdma_sai_rx.Init.MemInc = DMA_MINC_ENABLE;
hdma_sai_rx.Init.FIFOMode = DMA_FIFOMODE_DISABLE;
hdma_sai_rx.Init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;
hdma_sai_rx.Init.MemBurst = DMA_MBURST_SINGLE;
hdma_sai_rx.Init.PeriphBurst = DMA_MBURST_SINGLE;
if(Audio_In_Ctx[0].BitsPerSample == AUDIO_RESOLUTION_16B)
{
hdma_sai_rx.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;
hdma_sai_rx.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
}
else
{
hdma_sai_rx.Init.PeriphDataAlignment = DMA_PDATAALIGN_WORD;
hdma_sai_rx.Init.MemDataAlignment = DMA_MDATAALIGN_WORD;
}
/* Associate the DMA handle */
__HAL_LINKDMA(hsai, hdmarx, hdma_sai_rx);
/* Deinitialize the Stream for new transfer */
(void)HAL_DMA_DeInit(&hdma_sai_rx);
/* Configure the DMA Stream */
(void)HAL_DMA_Init(&hdma_sai_rx);
/* SAI DMA IRQ Channel configuration */
HAL_NVIC_SetPriority(AUDIO_IN_SAIx_DMAx_IRQ, BSP_AUDIO_IN_IT_PRIORITY, 0);
HAL_NVIC_EnableIRQ(AUDIO_IN_SAIx_DMAx_IRQ);
}
if(hsai->Instance == AUDIO_IN_SAI_PDMx)
{
/* Enable SAI clock */
AUDIO_IN_SAI_PDMx_CLK_ENABLE();
AUDIO_IN_SAI_PDMx_CLK_IN_ENABLE();
AUDIO_IN_SAI_PDMx_DATA_IN_ENABLE();
gpio_init_structure.Pin = AUDIO_IN_SAI_PDMx_CLK_IN_PIN;
gpio_init_structure.Mode = GPIO_MODE_AF_PP;
gpio_init_structure.Pull = GPIO_NOPULL;
gpio_init_structure.Speed = GPIO_SPEED_FREQ_HIGH;
gpio_init_structure.Alternate = AUDIO_IN_SAI_PDMx_DATA_CLK_AF;
HAL_GPIO_Init(AUDIO_IN_SAI_PDMx_CLK_IN_PORT, &gpio_init_structure);
gpio_init_structure.Pull = GPIO_PULLUP;
gpio_init_structure.Speed = GPIO_SPEED_FREQ_MEDIUM;
gpio_init_structure.Pin = AUDIO_IN_SAI_PDMx_DATA_IN_PIN;
HAL_GPIO_Init(AUDIO_IN_SAI_PDMx_DATA_IN_PORT, &gpio_init_structure);
AUDIO_IN_SAI_PDMx_FS_SCK_ENABLE();
/* CODEC_SAI pins configuration: FS, SCK, MCK and SD pins ------------------*/
gpio_init_structure.Pin = AUDIO_IN_SAI_PDMx_FS_PIN | AUDIO_IN_SAI_PDMx_SCK_PIN;
gpio_init_structure.Mode = GPIO_MODE_AF_PP;
gpio_init_structure.Pull = GPIO_NOPULL;
gpio_init_structure.Speed = GPIO_SPEED_FREQ_HIGH;
gpio_init_structure.Alternate = AUDIO_IN_SAI_PDMx_FS_SCK_AF;
HAL_GPIO_Init(AUDIO_IN_SAI_PDMx_FS_SCK_GPIO_PORT, &gpio_init_structure);
/* Enable the DMA clock */
AUDIO_IN_SAI_PDMx_DMAx_CLK_ENABLE();
/* Configure the hdma_sai_rx handle parameters */
hdma_sai_rx.Init.Request = AUDIO_IN_SAI_PDMx_DMAx_REQUEST;
hdma_sai_rx.Init.Direction = DMA_PERIPH_TO_MEMORY;
hdma_sai_rx.Init.PeriphInc = DMA_PINC_DISABLE;
hdma_sai_rx.Init.MemInc = DMA_MINC_ENABLE;
hdma_sai_rx.Init.PeriphDataAlignment = AUDIO_IN_SAI_PDMx_DMAx_PERIPH_DATA_SIZE;
hdma_sai_rx.Init.MemDataAlignment = AUDIO_IN_SAI_PDMx_DMAx_MEM_DATA_SIZE;
hdma_sai_rx.Init.Mode = DMA_CIRCULAR;
hdma_sai_rx.Init.Priority = DMA_PRIORITY_HIGH;
hdma_sai_rx.Init.FIFOMode = DMA_FIFOMODE_DISABLE;
hdma_sai_rx.Init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;
hdma_sai_rx.Init.MemBurst = DMA_MBURST_SINGLE;
hdma_sai_rx.Init.PeriphBurst = DMA_MBURST_SINGLE;
hdma_sai_rx.Instance = AUDIO_IN_SAI_PDMx_DMAx_STREAM;
/* Associate the DMA handle */
__HAL_LINKDMA(hsai, hdmarx, hdma_sai_rx);
/* Deinitialize the Stream for new transfer */
HAL_DMA_DeInit(&hdma_sai_rx);
/* Configure the DMA Stream */
HAL_DMA_Init(&hdma_sai_rx);
/* SAI DMA IRQ Channel configuration */
HAL_NVIC_SetPriority(AUDIO_IN_SAI_PDMx_DMAx_IRQ, BSP_AUDIO_IN_IT_PRIORITY, 0);
HAL_NVIC_EnableIRQ(AUDIO_IN_SAI_PDMx_DMAx_IRQ);
}
}
/**
* @brief Deinitializes SAI MSP.
* @param hsai SAI handle
* @retval HAL status
*/
static void SAI_MspDeInit(SAI_HandleTypeDef *hsai)
{
if(hsai->Instance == AUDIO_OUT_SAIx)
{
/* SAI DMA IRQ Channel deactivation */
HAL_NVIC_DisableIRQ(AUDIO_OUT_SAIx_DMAx_IRQ);
/* Deinitialize the DMA stream */
(void)HAL_DMA_DeInit(hsai->hdmatx);
/* De-initialize FS, SCK, MCK and SD pins */
HAL_GPIO_DeInit(AUDIO_OUT_SAIx_FS_GPIO_PORT, AUDIO_OUT_SAIx_FS_PIN);
HAL_GPIO_DeInit(AUDIO_OUT_SAIx_SCK_GPIO_PORT, AUDIO_OUT_SAIx_SCK_PIN);
HAL_GPIO_DeInit(AUDIO_OUT_SAIx_SD_GPIO_PORT, AUDIO_OUT_SAIx_SD_PIN);
HAL_GPIO_DeInit(AUDIO_OUT_SAIx_MCLK_GPIO_PORT, AUDIO_OUT_SAIx_MCLK_PIN);
}
if(hsai->Instance == AUDIO_IN_SAIx)
{
/* SAI DMA IRQ Channel deactivation */
HAL_NVIC_DisableIRQ(AUDIO_IN_SAIx_DMAx_IRQ);
/* Deinitialize the DMA stream */
(void)HAL_DMA_DeInit(hsai->hdmarx);
/* De-initialize SD pin */
HAL_GPIO_DeInit(AUDIO_IN_SAIx_SD_GPIO_PORT, AUDIO_IN_SAIx_SD_PIN);
/* Disable SAI clock */
AUDIO_IN_SAIx_CLK_DISABLE();
}
}
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/**
* @brief Tx Transfer completed callbacks.
* @param hsai SAI handle
* @retval None
*/
static void SAI_TxCpltCallback(SAI_HandleTypeDef *hsai)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hsai);
/* Manage the remaining file size and new address offset: This function
should be coded by user (its prototype is already declared in stm32h735g_discovery_audio.h) */
BSP_AUDIO_OUT_TransferComplete_CallBack(0);
}
/**
* @brief Tx Half Transfer completed callbacks.
* @param hsai SAI handle
* @retval None
*/
static void SAI_TxHalfCpltCallback(SAI_HandleTypeDef *hsai)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hsai);
/* Manage the remaining file size and new address offset: This function
should be coded by user (its prototype is already declared in stm32h735g_discovery_audio.h) */
BSP_AUDIO_OUT_HalfTransfer_CallBack(0);
}
/**
* @brief SAI error callbacks.
* @param hsai SAI handle
* @retval None
*/
static void SAI_ErrorCallback(SAI_HandleTypeDef *hsai)
{
if(hsai->Instance == AUDIO_OUT_SAIx)
{
BSP_AUDIO_OUT_Error_CallBack(0);
}
else
{
BSP_AUDIO_IN_Error_CallBack(0);
}
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
/**
* @}
*/
/** @defgroup STM32H735G_DISCO_AUDIO_IN_Exported_Functions STM32H735G_DISCO_AUDIO_IN Exported Functions
* @{
*/
/**
* @brief Initialize wave recording.
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param AudioInit Init structure
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_Init(uint32_t Instance, BSP_AUDIO_Init_t* AudioInit)
{
uint32_t i;
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Store the audio record context */
Audio_In_Ctx[Instance].Device = AudioInit->Device;
Audio_In_Ctx[Instance].ChannelsNbr = AudioInit->ChannelsNbr;
Audio_In_Ctx[Instance].SampleRate = AudioInit->SampleRate;
Audio_In_Ctx[Instance].BitsPerSample = AudioInit->BitsPerSample;
Audio_In_Ctx[Instance].Volume = AudioInit->Volume;
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_RESET;
if(Instance == 0U)
{
/* PLL clock is set depending by the AudioFreq (44.1khz vs 48khz groups) */
if(MX_SAI1_ClockConfig(&haudio_in_sai[Instance], AudioInit->SampleRate) != HAL_OK)
{
ret = BSP_ERROR_CLOCK_FAILURE;
}
else
{
haudio_in_sai[Instance].Instance = AUDIO_IN_SAIx;
haudio_out_sai.Instance = AUDIO_OUT_SAIx;
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/* Register the default SAI MSP callbacks */
if(Audio_In_Ctx[Instance].IsMspCallbacksValid == 0U)
{
if(BSP_AUDIO_IN_RegisterDefaultMspCallbacks(Instance) != BSP_ERROR_NONE)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
#else
SAI_MspInit(&haudio_in_sai[Instance]);
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
if (ret == BSP_ERROR_NONE)
{
MX_SAI_Config mx_config;
/* Prepare haudio_in_sai handle */
mx_config.AudioFrequency = Audio_In_Ctx[Instance].SampleRate;
mx_config.AudioMode = SAI_MODESLAVE_RX;
mx_config.ClockStrobing = SAI_CLOCKSTROBING_FALLINGEDGE;
mx_config.MonoStereoMode = (AudioInit->ChannelsNbr == 1U) ? SAI_MONOMODE : SAI_STEREOMODE;
mx_config.DataSize = SAI_DATASIZE_16;
mx_config.FrameLength = 128;
mx_config.ActiveFrameLength = 64;
mx_config.OutputDrive = SAI_OUTPUTDRIVE_ENABLE;
mx_config.Synchro = SAI_SYNCHRONOUS;
mx_config.SynchroExt = SAI_SYNCEXT_DISABLE;
mx_config.SlotActive = SAI_SLOTACTIVE_0 | SAI_SLOTACTIVE_1;
if(MX_SAI1_Block_A_Init(&haudio_in_sai[Instance], &mx_config) != HAL_OK)
{
/* Return BSP_ERROR_PERIPH_FAILURE when operations are not correctly done */
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
/* Prepare haudio_out_sai handle */
mx_config.AudioMode = SAI_MODEMASTER_TX;
mx_config.Synchro = SAI_ASYNCHRONOUS;
if(MX_SAI1_Block_B_Init(&haudio_out_sai, &mx_config) != HAL_OK)
{
/* Return BSP_ERROR_PERIPH_FAILURE when operations are not correctly done */
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
if (ret == BSP_ERROR_NONE)
{
/* Register SAI TC, HT and Error callbacks */
if(HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_RX_COMPLETE_CB_ID, SAI_RxCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_RX_HALFCOMPLETE_CB_ID, SAI_RxHalfCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
if(HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_ERROR_CB_ID, SAI_ErrorCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
#if (USE_AUDIO_CODEC_WM8994 == 1)
if (ret == BSP_ERROR_NONE)
{
/* Initialize the codec internal registers */
if (WM8994_Probe() == BSP_ERROR_NONE)
{
WM8994_Init_t codec_init;
/* Fill codec_init structure */
codec_init.OutputDevice = (Audio_Out_Ctx[0].State == AUDIO_OUT_STATE_RESET) ? WM8994_OUT_NONE : WM8994_OUT_HEADPHONE;
codec_init.Frequency = AudioInit->SampleRate;
codec_init.Resolution = (AudioInit->BitsPerSample == AUDIO_RESOLUTION_32B) ? WM8994_RESOLUTION_32b : WM8994_RESOLUTION_16b;
codec_init.InputDevice = (AudioInit->Device == AUDIO_IN_DEVICE_ANALOG_MIC) ? WM8994_IN_LINE1 : WM8994_IN_MIC2;
/* Convert volume before sending to the codec */
codec_init.Volume = VOLUME_IN_CONVERT(AudioInit->Volume);
/* Initialize the codec internal registers */
if(Audio_Drv->Init(Audio_CompObj, &codec_init) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
else
{
/* Update audio in context state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_STOP;
}
}
}
#endif /*USE_AUDIO_CODEC_WM8994 == 1)*/
}
}
}
else if(Instance == 1)
{
if(Audio_In_Ctx[Instance].Device != AUDIO_IN_DEVICE_DIGITAL_MIC1)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* PLL clock is set depending by the AudioFreq (44.1khz vs 48khz groups) */
if(MX_SAI4_ClockConfig(&haudio_in_sai[Instance], AudioInit->SampleRate) != HAL_OK)
{
return BSP_ERROR_CLOCK_FAILURE;
}
haudio_in_sai[Instance].Instance = AUDIO_IN_SAI_PDMx;
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1U)
/* Register the default SAI MSP callbacks */
if(Audio_In_Ctx[Instance].IsMspCallbacksValid == 0U)
{
if(BSP_AUDIO_IN_RegisterDefaultMspCallbacks(Instance) != BSP_ERROR_NONE)
{
return BSP_ERROR_PERIPH_FAILURE;
}
}
#else
SAI_MspInit(&haudio_in_sai[Instance]);
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1U) */
MX_SAI_Config mx_config;
/* Prepare haudio_in_sai handle */
mx_config.MonoStereoMode = SAI_STEREOMODE;
mx_config.DataSize = SAI_DATASIZE_16;
mx_config.FrameLength = 16;
mx_config.ActiveFrameLength = 1;
mx_config.OutputDrive = SAI_OUTPUTDRIVE_DISABLE;
mx_config.SlotActive = SAI_SLOTACTIVE_0;
mx_config.AudioFrequency = Audio_In_Ctx[Instance].SampleRate*8;
mx_config.AudioMode = SAI_MODEMASTER_RX;
mx_config.ClockStrobing = SAI_CLOCKSTROBING_FALLINGEDGE;
mx_config.Synchro = SAI_ASYNCHRONOUS;
mx_config.SynchroExt = SAI_SYNCEXT_DISABLE;
if(MX_SAI4_Block_A_Init(&haudio_in_sai[Instance], &mx_config) != HAL_OK)
{
/* Return BSP_ERROR_PERIPH_FAILURE when operations are not correctly done */
return BSP_ERROR_PERIPH_FAILURE;
}
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1U)
/* Register SAI TC, HT and Error callbacks */
if(HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_RX_COMPLETE_CB_ID, SAI_RxCpltCallback) != HAL_OK)
{
return BSP_ERROR_PERIPH_FAILURE;
}
if(HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_RX_HALFCOMPLETE_CB_ID, SAI_RxHalfCpltCallback) != HAL_OK)
{
return BSP_ERROR_PERIPH_FAILURE;
}
if(HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_ERROR_CB_ID, SAI_ErrorCallback) != HAL_OK)
{
return BSP_ERROR_PERIPH_FAILURE;
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1U) */
#if (USE_BSP_PDM_LIB_FEATURE > 0U)
if(BSP_AUDIO_IN_PDMToPCM_Init(Instance, AudioInit->SampleRate, Audio_In_Ctx[Instance].ChannelsNbr, Audio_In_Ctx[Instance].ChannelsNbr) != BSP_ERROR_NONE)
{
return BSP_ERROR_NO_INIT;
}
#endif
}
}
else /* (Instance == 2U) */
{
DFSDM_Filter_TypeDef* FilterInstnace[DFSDM_MIC_NUMBER] = {AUDIO_DFSDMx_MIC1_FILTER, AUDIO_DFSDMx_MIC2_FILTER,AUDIO_DFSDMx_MIC3_FILTER,AUDIO_DFSDMx_MIC4_FILTER};
DFSDM_Channel_TypeDef* ChannelInstance[DFSDM_MIC_NUMBER] = {AUDIO_DFSDMx_MIC1_CHANNEL, AUDIO_DFSDMx_MIC2_CHANNEL,AUDIO_DFSDMx_MIC3_CHANNEL,AUDIO_DFSDMx_MIC4_CHANNEL};
uint32_t DigitalMicPins[DFSDM_MIC_NUMBER] = {DFSDM_CHANNEL_SAME_CHANNEL_PINS, DFSDM_CHANNEL_FOLLOWING_CHANNEL_PINS, DFSDM_CHANNEL_SAME_CHANNEL_PINS, DFSDM_CHANNEL_FOLLOWING_CHANNEL_PINS};
uint32_t DigitalMicType[DFSDM_MIC_NUMBER] = {DFSDM_CHANNEL_SPI_RISING, DFSDM_CHANNEL_SPI_FALLING, DFSDM_CHANNEL_SPI_RISING, DFSDM_CHANNEL_SPI_FALLING};
uint32_t Channel4Filter[DFSDM_MIC_NUMBER] = {AUDIO_DFSDMx_MIC1_CHANNEL_FOR_FILTER, AUDIO_DFSDMx_MIC2_CHANNEL_FOR_FILTER,AUDIO_DFSDMx_MIC3_CHANNEL_FOR_FILTER,AUDIO_DFSDMx_MIC4_CHANNEL_FOR_FILTER};
MX_DFSDM_Config dfsdm_config;
/* PLL clock is set depending on the AudioFreq (44.1khz vs 48khz groups) */
if(MX_DFSDM1_ClockConfig(&haudio_in_dfsdm_channel[0], AudioInit->SampleRate) != HAL_OK)
{
ret = BSP_ERROR_CLOCK_FAILURE;
}
if(ret == BSP_ERROR_NONE)
{
#if (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1)
/* Register the default DFSDM MSP callbacks */
if(Audio_In_Ctx[Instance].IsMspCallbacksValid == 0U)
{
if(BSP_AUDIO_IN_RegisterDefaultMspCallbacks(Instance) != BSP_ERROR_NONE)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
#else
DFSDM_FilterMspInit(&haudio_in_dfsdm_filter[1]);
DFSDM_ChannelMspInit(&haudio_in_dfsdm_channel[1]);
#endif /* (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1) */
if(ret == BSP_ERROR_NONE)
{
for(i = 0; i < DFSDM_MIC_NUMBER; i ++)
{
dfsdm_config.FilterInstance = FilterInstnace[i];
dfsdm_config.ChannelInstance = ChannelInstance[i];
dfsdm_config.DigitalMicPins = DigitalMicPins[i];
dfsdm_config.DigitalMicType = DigitalMicType[i];
dfsdm_config.Channel4Filter = Channel4Filter[i];
dfsdm_config.RegularTrigger = DFSDM_FILTER_SW_TRIGGER;
/* MIC2, MIC3 and MIC4 should be synchronized to MIC1 if it's used */
if((i >= 1U) && (i < 4U) && ((Audio_In_Ctx[Instance].Device & AUDIO_IN_DEVICE_DIGITAL_MIC1) == AUDIO_IN_DEVICE_DIGITAL_MIC1))
{
dfsdm_config.RegularTrigger = DFSDM_FILTER_SYNC_TRIGGER;
}
dfsdm_config.SincOrder = DFSDM_FILTER_ORDER(Audio_In_Ctx[Instance].SampleRate);
dfsdm_config.Oversampling = DFSDM_OVER_SAMPLING(Audio_In_Ctx[Instance].SampleRate);
dfsdm_config.ClockDivider = DFSDM_CLOCK_DIVIDER(Audio_In_Ctx[Instance].SampleRate);
dfsdm_config.RightBitShift = DFSDM_MIC_BIT_SHIFT(Audio_In_Ctx[Instance].SampleRate);
if(((AudioInit->Device >> i) & AUDIO_IN_DEVICE_DIGITAL_MIC1) == AUDIO_IN_DEVICE_DIGITAL_MIC1)
{
/* Default configuration of DFSDM filters and channels */
if(MX_DFSDM1_Init(&haudio_in_dfsdm_filter[i], &haudio_in_dfsdm_channel[i], &dfsdm_config) != HAL_OK)
{
/* Return BSP_ERROR_PERIPH_FAILURE when operations are not correctly done */
ret = BSP_ERROR_PERIPH_FAILURE;
}
#if (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1)
/* Register filter regular conversion callbacks */
if(HAL_DFSDM_Filter_RegisterCallback(&haudio_in_dfsdm_filter[i], HAL_DFSDM_FILTER_REGCONV_COMPLETE_CB_ID, DFSDM_FilterRegConvCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
if(HAL_DFSDM_Filter_RegisterCallback(&haudio_in_dfsdm_filter[i], HAL_DFSDM_FILTER_REGCONV_HALFCOMPLETE_CB_ID, DFSDM_FilterRegConvHalfCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
#endif /* (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1) */
}
if(ret != BSP_ERROR_NONE)
{
break;
}
}
}
}
}
/* Update BSP AUDIO IN state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_STOP;
}
/* Return BSP status */
return ret;
}
/**
* @brief Deinit the audio IN peripherals.
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_DeInit(uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
uint32_t i;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
if(Instance == 0U)
{
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 0U)
SAI_MspDeInit(&haudio_in_sai[Instance]);
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 0U) */
/* Initialize the haudio_in_sai Instance parameter */
haudio_in_sai[Instance].Instance = AUDIO_IN_SAIx;
/* Initialize the haudio_out_sai Instance parameter */
haudio_out_sai.Instance = AUDIO_OUT_SAIx;
if(Audio_Drv->DeInit(Audio_CompObj) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}/* De-Initializes SAI handles */
else if(HAL_SAI_DeInit(&haudio_in_sai[Instance]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
if(HAL_SAI_DeInit(&haudio_out_sai) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
}
else if (Instance == 1U)
{
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 0U)
SAI_MspDeInit(&haudio_in_sai[Instance]);
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 0U) */
/* Initialize the haudio_in_sai Instance parameter */
haudio_in_sai[Instance].Instance = AUDIO_IN_SAI_PDMx;
/* Initialize the haudio_out_sai Instance parameter */
haudio_out_sai.Instance = AUDIO_OUT_SAIx;
if(HAL_SAI_DeInit(&haudio_in_sai[Instance]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
if(HAL_SAI_DeInit(&haudio_out_sai) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
}
else /* if(Instance == 2U) */
{
for(i = 0U; i < DFSDM_MIC_NUMBER; i++)
{
/* De-initializes DFSDM Filter handle */
if(haudio_in_dfsdm_filter[i].Instance != NULL)
{
#if (USE_HAL_DFSDM_REGISTER_CALLBACKS == 0U)
DFSDM_FilterMspDeInit(&haudio_in_dfsdm_filter[i]);
#endif /* (USE_HAL_DFSDM_REGISTER_CALLBACKS == 0U) */
if(HAL_OK != HAL_DFSDM_FilterDeInit(&haudio_in_dfsdm_filter[i]))
{
return BSP_ERROR_PERIPH_FAILURE;
}
haudio_in_dfsdm_filter[i].Instance = NULL;
}
/* De-initializes DFSDM Channel handle */
if(haudio_in_dfsdm_channel[i].Instance != NULL)
{
#if (USE_HAL_DFSDM_REGISTER_CALLBACKS == 0U)
DFSDM_ChannelMspDeInit(&haudio_in_dfsdm_channel[i]);
#endif /* (USE_HAL_DFSDM_REGISTER_CALLBACKS == 0U) */
if(HAL_OK != HAL_DFSDM_ChannelDeInit(&haudio_in_dfsdm_channel[i]))
{
return BSP_ERROR_PERIPH_FAILURE;
}
haudio_in_dfsdm_channel[i].Instance = NULL;
}
}
/* Reset Audio_In_Ctx[1].IsMultiBuff if any */
Audio_In_Ctx[1].IsMultiBuff = 0;
}
/* Update BSP AUDIO IN state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_RESET;
}
/* Return BSP status */
return ret;
}
/**
* @brief Clock Config.
* @param hDfsdmChannel DFSDM Channel Handle
* @param SampleRate Audio frequency to be configured for the DFSDM Channel.
* @note This API is called by BSP_AUDIO_IN_Init()
* Being __weak it can be overwritten by the application
* @retval HAL_status
*/
__weak HAL_StatusTypeDef MX_DFSDM1_ClockConfig(DFSDM_Channel_HandleTypeDef *hDfsdmChannel, uint32_t SampleRate)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hDfsdmChannel);
HAL_StatusTypeDef ret = HAL_OK;
RCC_PeriphCLKInitTypeDef rcc_ex_clk_init_struct;
HAL_RCCEx_GetPeriphCLKConfig(&rcc_ex_clk_init_struct);
/* Configure the PLL2 according to the requested audio frequency if not already done by other instances */
if ((Audio_Out_Ctx[0].State == AUDIO_OUT_STATE_RESET) && (Audio_In_Ctx[0].State == AUDIO_IN_STATE_RESET))
{
ret = MX_SAI1_ClockConfig(&haudio_in_sai[0], SampleRate);
}
return ret;
}
/**
* @brief SAI clock Config.
* @param hsai SAI handle
* @param SampleRate Audio frequency used to play the audio stream.
* @note This API is called by BSP_AUDIO_OUT_Init() and BSP_AUDIO_OUT_SetFrequency()
* Being __weak it can be overwritten by the application
* @retval HAL status
*/
__weak HAL_StatusTypeDef MX_SAI4_ClockConfig(SAI_HandleTypeDef *hsai, uint32_t SampleRate)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hsai);
HAL_StatusTypeDef ret = HAL_OK;
RCC_PeriphCLKInitTypeDef rcc_ex_clk_init_struct;
HAL_RCCEx_GetPeriphCLKConfig(&rcc_ex_clk_init_struct);
/* Set the PLL configuration according to the audio frequency */
if((SampleRate == AUDIO_FREQUENCY_11K) || (SampleRate == AUDIO_FREQUENCY_22K) || (SampleRate == AUDIO_FREQUENCY_44K))
{
rcc_ex_clk_init_struct.PLL2.PLL2P = 24;
rcc_ex_clk_init_struct.PLL2.PLL2N = 271;
}
else /* AUDIO_FREQUENCY_8K, AUDIO_FREQUENCY_16K, AUDIO_FREQUENCY_32K, AUDIO_FREQUENCY_48K, AUDIO_FREQUENCY_96K */
{
rcc_ex_clk_init_struct.PLL2.PLL2P = 7;
rcc_ex_clk_init_struct.PLL2.PLL2N = 344;
}
/* SAI clock config */
rcc_ex_clk_init_struct.PeriphClockSelection = RCC_PERIPHCLK_SAI4A;
rcc_ex_clk_init_struct.Sai4AClockSelection = RCC_SAI4ACLKSOURCE_PLL2;
rcc_ex_clk_init_struct.PLL2.PLL2Q = 1;
rcc_ex_clk_init_struct.PLL2.PLL2R = 1;
rcc_ex_clk_init_struct.PLL2.PLL2M = 25;
if(HAL_RCCEx_PeriphCLKConfig(&rcc_ex_clk_init_struct) != HAL_OK)
{
ret = HAL_ERROR;
}
return ret;
}
/**
* @brief Initializes the Audio instance (DFSDM).
* @param hDfsdmFilter DFSDM Filter Handle
* @param hDfsdmChannel DFSDM Channel Handle
* @param MXConfig DFSDM configuration structure
* @note Being __weak it can be overwritten by the application
* @note Channel output Clock Divider and Filter Oversampling are calculated as follow:
* - Clock_Divider = CLK(input DFSDM)/CLK(micro) with
* 1MHZ < CLK(micro) < 3.2MHZ (TYP 2.4MHZ for MP34DT01TR)
* - Oversampling = CLK(input DFSDM)/(Clock_Divider * AudioFreq)
* @retval HAL_status
*/
__weak HAL_StatusTypeDef MX_DFSDM1_Init(DFSDM_Filter_HandleTypeDef *hDfsdmFilter, DFSDM_Channel_HandleTypeDef *hDfsdmChannel, MX_DFSDM_Config *MXConfig)
{
/* MIC filters initialization */
hDfsdmFilter->Instance = MXConfig->FilterInstance;
hDfsdmFilter->Init.RegularParam.Trigger = MXConfig->RegularTrigger;
hDfsdmFilter->Init.RegularParam.FastMode = ENABLE;
hDfsdmFilter->Init.RegularParam.DmaMode = ENABLE;
hDfsdmFilter->Init.InjectedParam.Trigger = DFSDM_FILTER_SW_TRIGGER;
hDfsdmFilter->Init.InjectedParam.ScanMode = DISABLE;
hDfsdmFilter->Init.InjectedParam.DmaMode = DISABLE;
hDfsdmFilter->Init.InjectedParam.ExtTrigger = DFSDM_FILTER_EXT_TRIG_TIM8_TRGO;
hDfsdmFilter->Init.InjectedParam.ExtTriggerEdge = DFSDM_FILTER_EXT_TRIG_BOTH_EDGES;
hDfsdmFilter->Init.FilterParam.SincOrder = MXConfig->SincOrder;
hDfsdmFilter->Init.FilterParam.Oversampling = MXConfig->Oversampling;
hDfsdmFilter->Init.FilterParam.IntOversampling = 1;
if(HAL_DFSDM_FilterInit(hDfsdmFilter) != HAL_OK)
{
return HAL_ERROR;
}
/* MIC channels initialization */
hDfsdmChannel->Instance = MXConfig->ChannelInstance;
hDfsdmChannel->Init.OutputClock.Activation = ENABLE;
hDfsdmChannel->Init.OutputClock.Selection = DFSDM_CHANNEL_OUTPUT_CLOCK_AUDIO;
hDfsdmChannel->Init.OutputClock.Divider = MXConfig->ClockDivider;
hDfsdmChannel->Init.Input.Multiplexer = DFSDM_CHANNEL_EXTERNAL_INPUTS;
hDfsdmChannel->Init.Input.DataPacking = DFSDM_CHANNEL_STANDARD_MODE;
hDfsdmChannel->Init.SerialInterface.SpiClock = DFSDM_CHANNEL_SPI_CLOCK_INTERNAL;
hDfsdmChannel->Init.Awd.FilterOrder = DFSDM_CHANNEL_SINC1_ORDER;
hDfsdmChannel->Init.Awd.Oversampling = 10;
hDfsdmChannel->Init.Offset = 0;
hDfsdmChannel->Init.RightBitShift = MXConfig->RightBitShift;
hDfsdmChannel->Init.Input.Pins = MXConfig->DigitalMicPins;
hDfsdmChannel->Init.SerialInterface.Type = MXConfig->DigitalMicType;
if(HAL_OK != HAL_DFSDM_ChannelInit(hDfsdmChannel))
{
return HAL_ERROR;
}
/* Configure injected channel */
if(HAL_DFSDM_FilterConfigRegChannel(hDfsdmFilter, MXConfig->Channel4Filter, DFSDM_CONTINUOUS_CONV_ON) != HAL_OK)
{
return HAL_ERROR;
}
return HAL_OK;
}
/**
* @brief Initializes the Audio Codec audio in instance (SAI).
* @param hsai SAI handle
* @param MXConfig SAI configuration structure
* @note Being __weak it can be overwritten by the application
* @retval HAL status
*/
__weak HAL_StatusTypeDef MX_SAI1_Block_A_Init(SAI_HandleTypeDef* hsai, MX_SAI_Config *MXConfig)
{
HAL_StatusTypeDef ret = HAL_OK;
/* Disable SAI peripheral to allow access to SAI internal registers */
__HAL_SAI_DISABLE(hsai);
/* Configure SAI1_Block_B */
hsai->Init.AudioFrequency = MXConfig->AudioFrequency;
hsai->Init.MonoStereoMode = MXConfig->MonoStereoMode;
hsai->Init.AudioMode = MXConfig->AudioMode;
hsai->Init.NoDivider = SAI_MASTERDIVIDER_ENABLE;
hsai->Init.Protocol = SAI_FREE_PROTOCOL;
hsai->Init.DataSize = MXConfig->DataSize;
hsai->Init.FirstBit = SAI_FIRSTBIT_MSB;
hsai->Init.ClockStrobing = MXConfig->ClockStrobing;
hsai->Init.Synchro = MXConfig->Synchro;
hsai->Init.OutputDrive = MXConfig->OutputDrive;
hsai->Init.FIFOThreshold = SAI_FIFOTHRESHOLD_1QF;
hsai->Init.SynchroExt = MXConfig->SynchroExt;
hsai->Init.CompandingMode = SAI_NOCOMPANDING;
hsai->Init.TriState = SAI_OUTPUT_RELEASED;
hsai->Init.Mckdiv = 0;
hsai->Init.MckOutput = SAI_MCK_OUTPUT_ENABLE;
hsai->Init.MckOverSampling = SAI_MCK_OVERSAMPLING_DISABLE;
hsai->Init.PdmInit.Activation = DISABLE;
/* Configure SAI_Block_x Frame */
hsai->FrameInit.FrameLength = MXConfig->FrameLength;
hsai->FrameInit.ActiveFrameLength = MXConfig->ActiveFrameLength;
hsai->FrameInit.FSDefinition = SAI_FS_CHANNEL_IDENTIFICATION;
hsai->FrameInit.FSPolarity = SAI_FS_ACTIVE_LOW;
hsai->FrameInit.FSOffset = SAI_FS_BEFOREFIRSTBIT;
/* Configure SAI Block_x Slot */
hsai->SlotInit.FirstBitOffset = 0;
if ((MXConfig->DataSize == AUDIO_RESOLUTION_24B) || (MXConfig->DataSize == AUDIO_RESOLUTION_32B))
{
hsai->SlotInit.SlotSize = SAI_SLOTSIZE_32B;
}
else
{
hsai->SlotInit.SlotSize = SAI_SLOTSIZE_16B;
}
hsai->SlotInit.SlotNumber = 2;
hsai->SlotInit.SlotActive = MXConfig->SlotActive;
if(HAL_SAI_Init(hsai) != HAL_OK)
{
ret = HAL_ERROR;
}
return ret;
}
/**
* @brief Initializes the Audio Codec audio in instance (SAI).
* @param MXConfig SAI configuration structure
* @note Being __weak it can be overwritten by the application
* @retval HAL status
*/
__weak HAL_StatusTypeDef MX_SAI4_Block_A_Init(SAI_HandleTypeDef* hsai, MX_SAI_Config *MXConfig)
{
HAL_StatusTypeDef ret = HAL_OK;
/* Disable SAI peripheral to allow access to SAI internal registers */
__HAL_SAI_DISABLE(hsai);
/* Configure SAI4_Block_A */
hsai->Init.AudioFrequency = MXConfig->AudioFrequency;
hsai->Init.MonoStereoMode = MXConfig->MonoStereoMode;
hsai->Init.AudioMode = MXConfig->AudioMode;
hsai->Init.NoDivider = SAI_MASTERDIVIDER_DISABLE;
hsai->Init.Protocol = SAI_FREE_PROTOCOL;
hsai->Init.DataSize = MXConfig->DataSize;
hsai->Init.FirstBit = SAI_FIRSTBIT_LSB;
hsai->Init.ClockStrobing = MXConfig->ClockStrobing;
hsai->Init.Synchro = MXConfig->Synchro;
hsai->Init.OutputDrive = MXConfig->OutputDrive;
hsai->Init.FIFOThreshold = SAI_FIFOTHRESHOLD_1QF;
hsai->Init.SynchroExt = MXConfig->SynchroExt;
hsai->Init.CompandingMode = SAI_NOCOMPANDING;
hsai->Init.TriState = SAI_OUTPUT_RELEASED;
hsai->Init.Mckdiv = 0;
hsai->Init.PdmInit.Activation = ENABLE;
hsai->Init.PdmInit.MicPairsNbr = 1;
hsai->Init.PdmInit.ClockEnable = SAI_PDM_CLOCK1_ENABLE;
/* Configure SAI_Block_x Frame */
hsai->FrameInit.FrameLength = MXConfig->FrameLength;
hsai->FrameInit.ActiveFrameLength = MXConfig->ActiveFrameLength;
hsai->FrameInit.FSDefinition = SAI_FS_STARTFRAME;
hsai->FrameInit.FSPolarity = SAI_FS_ACTIVE_HIGH;
hsai->FrameInit.FSOffset = SAI_FS_FIRSTBIT;
/* Configure SAI Block_x Slot */
hsai->SlotInit.FirstBitOffset = 0;
hsai->SlotInit.SlotSize = SAI_SLOTSIZE_DATASIZE;
hsai->SlotInit.SlotNumber = 1;
hsai->SlotInit.SlotActive = MXConfig->SlotActive;
if(HAL_SAI_Init(hsai) != HAL_OK)
{
ret = HAL_ERROR;
}
/* Enable SAI peripheral */
__HAL_SAI_ENABLE(hsai);
return ret;
}
#if ((USE_HAL_DFSDM_REGISTER_CALLBACKS == 1U) || (USE_HAL_SAI_REGISTER_CALLBACKS == 1U))
/**
* @brief Default BSP AUDIO IN Msp Callbacks
* @param Instance BSP AUDIO IN Instance
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_RegisterDefaultMspCallbacks (uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
uint32_t i;
if(Instance == 1U)
{
for(i = 0; i < DFSDM_MIC_NUMBER; i ++)
{
if(((Audio_In_Ctx[Instance].Device >> i) & AUDIO_IN_DEVICE_DIGITAL_MIC1) == AUDIO_IN_DEVICE_DIGITAL_MIC1)
{
__HAL_DFSDM_CHANNEL_RESET_HANDLE_STATE(&haudio_in_dfsdm_channel[i]);
__HAL_DFSDM_FILTER_RESET_HANDLE_STATE(&haudio_in_dfsdm_filter[i]);
#if (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1)
/* Register MspInit/MspDeInit Callbacks */
if(HAL_DFSDM_CHANNEL_RegisterCallback(&haudio_in_dfsdm_channel[i], HAL_DFSDM_CHANNEL_MSPINIT_CB_ID, DFSDM_ChannelMspInit) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_DFSDM_FILTER_RegisterCallback(&haudio_in_dfsdm_filter[i], HAL_DFSDM_FILTER_MSPINIT_CB_ID, DFSDM_FilterMspInit) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_DFSDM_CHANNEL_RegisterCallback(&haudio_in_dfsdm_channel[i], HAL_DFSDM_CHANNEL_MSPDEINIT_CB_ID, DFSDM_ChannelMspDeInit) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_DFSDM_FILTER_RegisterCallback(&haudio_in_dfsdm_filter[i], HAL_DFSDM_FILTER_MSPDEINIT_CB_ID, DFSDM_FilterMspDeInit) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
}
#endif /* (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1) */
}
}
}
else if(Instance == 0U)
{
__HAL_SAI_RESET_HANDLE_STATE(&haudio_in_sai[Instance]);
__HAL_SAI_RESET_HANDLE_STATE(&haudio_out_sai);
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1U)
/* Register MspInit/MspDeInit Callbacks */
if(HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_MSPINIT_CB_ID, SAI_MspInit) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_MSPINIT_CB_ID, SAI_MspInit) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_MSPDEINIT_CB_ID, SAI_MspDeInit) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_RegisterCallback(&haudio_out_sai, HAL_SAI_MSPDEINIT_CB_ID, SAI_MspDeInit) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1U) */
}
else
{
ret = BSP_ERROR_WRONG_PARAM;
}
if(ret == BSP_ERROR_NONE)
{
Audio_In_Ctx[Instance].IsMspCallbacksValid = 1;
}
/* Return BSP status */
return ret;
}
/**
* @brief BSP AUDIO In Filter Msp Callback registering
* @param Instance AUDIO IN Instance
* @param CallBacks pointer to filter MspInit/MspDeInit functions
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_RegisterMspCallbacks (uint32_t Instance, BSP_AUDIO_IN_Cb_t *CallBacks)
{
int32_t ret = BSP_ERROR_NONE;
uint32_t i;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
if(Instance == 1U)
{
#if (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1)
for(i = 0U; i < DFSDM_MIC_NUMBER; i ++)
{
__HAL_DFSDM_FILTER_RESET_HANDLE_STATE(&haudio_in_dfsdm_filter[i]);
/* Register MspInit/MspDeInit Callback */
if(HAL_DFSDM_FILTER_RegisterCallback(&haudio_in_dfsdm_filter[i], HAL_DFSDM_FILTER_MSPINIT_CB_ID, CallBacks->pMspFltrInitCb) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_DFSDM_FILTER_RegisterCallback(&haudio_in_dfsdm_filter[i], HAL_DFSDM_FILTER_MSPDEINIT_CB_ID, CallBacks->pMspFltrDeInitCb) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_DFSDM_CHANNEL_RegisterCallback(&haudio_in_dfsdm_channel[i], HAL_DFSDM_CHANNEL_MSPINIT_CB_ID, CallBacks->pMspChInitCb) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_DFSDM_CHANNEL_RegisterCallback(&haudio_in_dfsdm_channel[i], HAL_DFSDM_CHANNEL_MSPDEINIT_CB_ID, CallBacks->pMspChDeInitCb) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
}
}
#endif /* (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1) */
}
else /* (Instance == 0U) */
{
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1U)
if(HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_MSPINIT_CB_ID, CallBacks->pMspSaiInitCb) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_MSPDEINIT_CB_ID, CallBacks->pMspSaiDeInitCb) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1U) */
}
if(ret == BSP_ERROR_NONE)
{
Audio_In_Ctx[Instance].IsMspCallbacksValid = 1;
}
}
/* Return BSP status */
return ret;
}
#endif /* ((USE_HAL_DFSDM_REGISTER_CALLBACKS == 1) || (USE_HAL_SAI_REGISTER_CALLBACKS == 1U)) */
#if (USE_BSP_PDM_LIB_FEATURE > 0U)
/**
* @brief Initialize the PDM library.
* @param Instance AUDIO IN Instance
* @param AudioFreq Audio sampling frequency
* @param ChnlNbrIn Number of input audio channels in the PDM buffer
* @param ChnlNbrOut Number of desired output audio channels in the resulting PCM buffer
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_PDMToPCM_Init(uint32_t Instance, uint32_t AudioFreq, uint32_t ChnlNbrIn, uint32_t ChnlNbrOut)
{
uint32_t index = 0;
if(Instance != 1U)
{
return BSP_ERROR_WRONG_PARAM;
}
else
{
/* Enable CRC peripheral to unlock the PDM library */
__HAL_RCC_CRC_CLK_ENABLE();
for(index = 0; index < ChnlNbrIn; index++)
{
/* Init PDM filters */
PDM_FilterHandler[index].bit_order = PDM_FILTER_BIT_ORDER_MSB;
PDM_FilterHandler[index].endianness = PDM_FILTER_ENDIANNESS_LE;
PDM_FilterHandler[index].high_pass_tap = 2122358088;
PDM_FilterHandler[index].out_ptr_channels = ChnlNbrOut;
PDM_FilterHandler[index].in_ptr_channels = ChnlNbrIn;
PDM_Filter_Init((PDM_Filter_Handler_t *)(&PDM_FilterHandler[index]));
/* PDM lib config phase */
PDM_FilterConfig[index].output_samples_number = AudioFreq/1000;
PDM_FilterConfig[index].mic_gain = 24;
PDM_FilterConfig[index].decimation_factor = PDM_FILTER_DEC_FACTOR_64;
PDM_Filter_setConfig((PDM_Filter_Handler_t *)&PDM_FilterHandler[index], &PDM_FilterConfig[index]);
}
}
return BSP_ERROR_NONE;
}
/**
* @brief Converts audio format from PDM to PCM.
* @param Instance AUDIO IN Instance
* @param PDMBuf Pointer to PDM buffer data
* @param PCMBuf Pointer to PCM buffer data
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_PDMToPCM(uint32_t Instance, uint16_t *PDMBuf, uint16_t *PCMBuf)
{
uint32_t index = 0;
if(Instance != 1U)
{
return BSP_ERROR_WRONG_PARAM;
}
else
{
for(index = 0; index < Audio_In_Ctx[Instance].ChannelsNbr; index++)
{
PDM_Filter(&((uint8_t*)(PDMBuf))[index], (uint16_t*)&(PCMBuf[index]), &PDM_FilterHandler[index]);
}
}
return BSP_ERROR_NONE;
}
#endif
/**
* @brief Start audio recording.
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param pBuf Main buffer pointer for the recorded data storing
* @param NbrOfBytes Size of the record buffer in bytes
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_Record(uint32_t Instance, uint8_t* pBuf, uint32_t NbrOfBytes)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else if(Instance != 2U)
{
uint8_t TxData[2] = {0x00U, 0x00U};
if(HAL_SAI_Transmit(&haudio_out_sai, TxData, 2, 1000) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
/* Call the audio Codec Play function */
if (Audio_Drv->Play(Audio_CompObj) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
/* Start the process receive DMA */
if(HAL_SAI_Receive_DMA(&haudio_in_sai[Instance], (uint8_t*)pBuf, (uint16_t)(NbrOfBytes/(Audio_In_Ctx[Instance].BitsPerSample/8U))) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
}
else
{
Audio_In_Ctx[Instance].pBuff = (uint16_t*)pBuf;
Audio_In_Ctx[Instance].Size = NbrOfBytes;
/* Reset Buffer Trigger */
RecBuffTrigger = 0;
RecBuffHalf = 0;
/* Call the Media layer start function for MIC2 channel */
if(HAL_DFSDM_FilterRegularStart_DMA(&haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)], \
(int32_t*)MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)], DEFAULT_AUDIO_IN_BUFFER_SIZE) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if(HAL_DFSDM_FilterRegularStart_DMA(&haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)], \
(int32_t*)MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)], DEFAULT_AUDIO_IN_BUFFER_SIZE) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
/* Update BSP AUDIO IN state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_RECORDING;
}
}
/* Return BSP status */
return ret;
}
/**
* @brief Stop audio recording.
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_Stop(uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio in state */
else if (Audio_In_Ctx[Instance].State == AUDIO_IN_STATE_STOP)
{
/* Nothing to do */
}
else if ((Audio_In_Ctx[Instance].State != AUDIO_IN_STATE_RECORDING) &&
(Audio_In_Ctx[Instance].State != AUDIO_IN_STATE_PAUSE))
{
ret = BSP_ERROR_BUSY;
}
else
{
if(Instance == 2U)
{
/* Call the Media layer stop function */
if(HAL_DFSDM_FilterRegularStop_DMA(&haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
if(HAL_DFSDM_FilterRegularStop_DMA(&haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
}
else
{
if(Instance == 0U)
{
/* Call the Media layer stop function */
if(Audio_Drv->Stop(Audio_CompObj, CODEC_PDWN_SW) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
}
if(ret == BSP_ERROR_NONE)
{
if(HAL_SAI_DMAStop(&haudio_in_sai[Instance]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
}
/* Update BSP AUDIO IN state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_STOP;
}
/* Return BSP status */
return ret;
}
/**
* @brief Pause the audio file stream.
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_Pause(uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
if(Instance == 2U)
{
/* Call the Media layer stop function */
if(HAL_DFSDM_FilterRegularStop_DMA(&haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
if(HAL_DFSDM_FilterRegularStop_DMA(&haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
}
else /* 0 or 1*/
{
if(Instance == 0U)
{
/* Call Audio Codec Pause function */
if(Audio_Drv->Pause(Audio_CompObj) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
}
if(ret == BSP_ERROR_NONE)
{
if(HAL_SAI_DMAPause(&haudio_in_sai[Instance]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
}
/* Update BSP AUDIO IN state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_PAUSE;
}
/* Return BSP status */
return ret;
}
/**
* @brief Resume the audio file stream.
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_Resume(uint32_t Instance)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
if(Instance == 2U)
{
/* Call the Media layer start function for MIC2/MIC1 channel */
if(HAL_DFSDM_FilterRegularMsbStart_DMA(&haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)], \
(int16_t*)MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)], DEFAULT_AUDIO_IN_BUFFER_SIZE) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
if(HAL_DFSDM_FilterRegularMsbStart_DMA(&haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)], \
(int16_t*)MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)], DEFAULT_AUDIO_IN_BUFFER_SIZE) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
}
else /* 0 or 1 */
{
if(Instance == 0U)
{
/* Call Audio Codec Resume function */
if(Audio_Drv->Resume(Audio_CompObj) < 0)
{
ret = BSP_ERROR_COMPONENT_FAILURE;
}
}
if(ret == BSP_ERROR_NONE)
{
if(HAL_SAI_DMAResume(&haudio_in_sai[Instance]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
}
/* Update BSP AUDIO IN state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_RECORDING;
}
/* Return BSP status */
return ret;
}
/**
* @brief Starts audio recording.
* @param Instance AUDIO IN Instance. It can be 2(DFSDM used)
* @param pBuf Main buffer pointer for the recorded data storing
* @param NbrOfBytes Size of the recorded buffer in bytes
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_RecordChannels(uint32_t Instance, uint8_t **pBuf, uint32_t NbrOfBytes)
{
uint16_t i;
int32_t ret = BSP_ERROR_NONE;
uint32_t mic_init[DFSDM_MIC_NUMBER] = {0};
uint32_t audio_in_digital_mic = AUDIO_IN_DEVICE_DIGITAL_MIC1, pbuf_index = 0;
uint32_t enabled_mic = 0;
if(Instance != 2U)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Get the number of activated microphones */
for(i = 0U; i < DFSDM_MIC_NUMBER; i++)
{
if((Audio_In_Ctx[Instance].Device & audio_in_digital_mic) == audio_in_digital_mic)
{
enabled_mic++;
}
audio_in_digital_mic = audio_in_digital_mic << 1;
}
Audio_In_Ctx[Instance].pMultiBuff = pBuf;
Audio_In_Ctx[Instance].Size = NbrOfBytes;
Audio_In_Ctx[Instance].IsMultiBuff = 1;
audio_in_digital_mic = AUDIO_IN_DEVICE_DIGITAL_MIC_LAST;
for(i = 0U; i < DFSDM_MIC_NUMBER; i++)
{
if((mic_init[POS_VAL(audio_in_digital_mic)] != 1U) && ((Audio_In_Ctx[Instance].Device & audio_in_digital_mic) == audio_in_digital_mic))
{
/* Call the Media layer start function for MICx channel */
if(HAL_DFSDM_FilterRegularStart_DMA(&haudio_in_dfsdm_filter[POS_VAL(audio_in_digital_mic)], (int32_t*)pBuf[enabled_mic - 1U - pbuf_index], NbrOfBytes) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else
{
mic_init[POS_VAL(audio_in_digital_mic)] = 1;
pbuf_index++;
}
}
audio_in_digital_mic = audio_in_digital_mic >> 1;
}
/* Update BSP AUDIO IN state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_RECORDING;
}
/* Return BSP status */
return ret;
}
/**
* @brief Stop audio recording.
* @param Instance AUDIO IN Instance. It can be 1(DFSDM used)
* @param Device Digital input device to be stopped
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_StopChannels(uint32_t Instance, uint32_t Device)
{
int32_t ret;
/* Stop selected devices */
ret = BSP_AUDIO_IN_PauseChannels(Instance, Device);
if(ret == BSP_ERROR_NONE)
{
/* Update BSP AUDIO IN state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_STOP;
}
/* Return BSP status */
return ret;
}
/**
* @brief Pause the audio file stream.
* @param Instance AUDIO IN Instance. It can be 1(DFSDM used)
* @param Device Digital mic to be paused
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_PauseChannels(uint32_t Instance, uint32_t Device)
{
uint32_t audio_in_digital_mic = AUDIO_IN_DEVICE_DIGITAL_MIC1;
uint32_t i;
int32_t ret = BSP_ERROR_NONE;
if((Instance != 2U) || ((Device < AUDIO_IN_DEVICE_DIGITAL_MIC1) && (Device > AUDIO_IN_DEVICE_DIGITAL_MIC_LAST)))
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
for(i = 0; i < DFSDM_MIC_NUMBER; i++)
{
if((Device & audio_in_digital_mic) == audio_in_digital_mic)
{
/* Call the Media layer stop function */
if(HAL_DFSDM_FilterRegularStop_DMA(&haudio_in_dfsdm_filter[POS_VAL(audio_in_digital_mic)]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
audio_in_digital_mic = audio_in_digital_mic << 1;
}
if(ret == BSP_ERROR_NONE)
{
/* Update BSP AUDIO IN state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_PAUSE;
}
}
/* Return BSP status */
return ret;
}
/**
* @brief Resume the audio file stream
* @param Instance AUDIO IN Instance. It can be 1(DFSDM used)
* @param Device Digital mic to be resumed
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_ResumeChannels(uint32_t Instance, uint32_t Device)
{
int32_t ret = BSP_ERROR_NONE;
uint32_t audio_in_digital_mic = AUDIO_IN_DEVICE_DIGITAL_MIC_LAST;
uint32_t i;
if((Instance != 2U) || ((Device < AUDIO_IN_DEVICE_DIGITAL_MIC1) && (Device > AUDIO_IN_DEVICE_DIGITAL_MIC_LAST)))
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
for(i = 0; i < DFSDM_MIC_NUMBER; i++)
{
if((Device & audio_in_digital_mic) == audio_in_digital_mic)
{
/* Start selected device channel */
if(HAL_DFSDM_FilterRegularStart_DMA(&haudio_in_dfsdm_filter[POS_VAL(audio_in_digital_mic)],\
(int32_t*)Audio_In_Ctx[Instance].pMultiBuff[POS_VAL(audio_in_digital_mic)], Audio_In_Ctx[Instance].Size) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
audio_in_digital_mic = audio_in_digital_mic >> 1;
}
if(ret == BSP_ERROR_NONE)
{
/* Update BSP AUDIO IN state */
Audio_In_Ctx[Instance].State = AUDIO_IN_STATE_RECORDING;
}
}
/* Return BSP status */
return ret;
}
/**
* @brief Start audio recording.
* @param Instance AUDIO IN SAI PDM Instance. It can be only 2
* @param pbuf Main buffer pointer for the recorded data storing
* @param Size Size of the record buffer
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_RecordPDM(uint32_t Instance, uint8_t* pBuf, uint32_t NbrOfBytes)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance != 1U)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Start the process receive DMA */
if(HAL_SAI_Receive_DMA(&haudio_in_sai[Instance], (uint8_t*)pBuf, (uint16_t)(NbrOfBytes/(Audio_In_Ctx[Instance].BitsPerSample/8U))) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
}
/* Return BSP status */
return ret;
}
/**
* @brief Set Audio In device
* @param Instance AUDIO IN Instance. It can be 0 when SAI is used or 2 if DFSDM is used
* @param Device The audio input device to be used
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_SetDevice(uint32_t Instance, uint32_t Device)
{
int32_t ret = BSP_ERROR_NONE;
uint32_t i;
BSP_AUDIO_Init_t audio_init;
if((Instance >= AUDIO_IN_INSTANCES_NBR) || ((Instance == 0U) && (Device != AUDIO_IN_DEVICE_ANALOG_LINE1) &&\
(Device != AUDIO_IN_DEVICE_DIGITAL_MIC1))\
|| ((Instance == 1U) && (Device != AUDIO_IN_DEVICE_DIGITAL_MIC1))\
|| ((Instance == 2U) && (Device < AUDIO_IN_DEVICE_DIGITAL_MIC1) &&\
(Device > AUDIO_IN_DEVICE_DIGITAL_MIC_LAST)))
{
ret = BSP_ERROR_WRONG_PARAM;
}
else if(Audio_In_Ctx[Instance].State == AUDIO_IN_STATE_STOP)
{
if(Instance == 2U)
{
for(i = 0; i < DFSDM_MIC_NUMBER; i ++)
{
if(((Device >> i) & AUDIO_IN_DEVICE_DIGITAL_MIC1) == AUDIO_IN_DEVICE_DIGITAL_MIC1)
{
if(HAL_DFSDM_ChannelDeInit(&haudio_in_dfsdm_channel[i]) != HAL_OK)
{
return BSP_ERROR_PERIPH_FAILURE;
}
}
}
}
audio_init.Device = Device;
audio_init.ChannelsNbr = Audio_In_Ctx[Instance].ChannelsNbr;
audio_init.SampleRate = Audio_In_Ctx[Instance].SampleRate;
audio_init.BitsPerSample = Audio_In_Ctx[Instance].BitsPerSample;
audio_init.Volume = Audio_In_Ctx[Instance].Volume;
if(BSP_AUDIO_IN_Init(Instance, &audio_init) != BSP_ERROR_NONE)
{
ret = BSP_ERROR_NO_INIT;
}
}
else
{
ret = BSP_ERROR_BUSY;
}
/* Return BSP status */
return ret;
}
/**
* @brief Get Audio In device
* @param Instance AUDIO IN Instance. It can be 0 when SAI is used 1 when I2S is used or 2 if DFSDM is used
* @param Device The audio input device used
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_GetDevice(uint32_t Instance, uint32_t *Device)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Return audio Input Device */
*Device = Audio_In_Ctx[Instance].Device;
}
return ret;
}
/**
* @brief Set Audio In frequency
* @param Instance Audio IN instance
* @param SampleRate Input frequency to be set
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_SetSampleRate(uint32_t Instance, uint32_t SampleRate)
{
int32_t ret = BSP_ERROR_NONE;
if (Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
/* Check audio in state */
else if (Audio_In_Ctx[Instance].State != AUDIO_IN_STATE_STOP)
{
ret = BSP_ERROR_BUSY;
}
/* Check if playback on instance 0 or 1 is on going and corresponding sample rate */
else if ((Audio_Out_Ctx[0].State != AUDIO_OUT_STATE_RESET) && (Audio_Out_Ctx[0].SampleRate != SampleRate))
{
ret = BSP_ERROR_FEATURE_NOT_SUPPORTED;
}
/* Check if sample rate is modified */
else if (Audio_In_Ctx[Instance].SampleRate == SampleRate)
{
/* Nothing to do */
}
else if (Instance != 2U)
{
/* Update SAI1 clock config */
haudio_in_sai[Instance].Init.AudioFrequency = SampleRate;
haudio_out_sai.Init.AudioFrequency = SampleRate;
if (MX_SAI1_ClockConfig(&haudio_in_sai[Instance], SampleRate) != HAL_OK)
{
ret = BSP_ERROR_CLOCK_FAILURE;
}
/* Re-initialize SAI1 with new sample rate */
else if (HAL_SAI_Init(&haudio_in_sai[Instance]) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if (HAL_SAI_Init(&haudio_out_sai) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/* Register SAI TC, HT and Error callbacks */
else if (HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_RX_COMPLETE_CB_ID, SAI_RxCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if (HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_RX_HALFCOMPLETE_CB_ID, SAI_RxHalfCpltCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
else if (HAL_SAI_RegisterCallback(&haudio_in_sai[Instance], HAL_SAI_ERROR_CB_ID, SAI_ErrorCallback) != HAL_OK)
{
ret = BSP_ERROR_PERIPH_FAILURE;
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
/* Store new sample rate on audio in context */
else
{
Audio_In_Ctx[Instance].SampleRate = SampleRate;
}
}
else /* Instance = 2 */
{
uint32_t i;
BSP_AUDIO_Init_t audio_init;
for(i = 0; i < DFSDM_MIC_NUMBER; i ++)
{
if(((Audio_In_Ctx[Instance].Device >> i) & AUDIO_IN_DEVICE_DIGITAL_MIC1) == AUDIO_IN_DEVICE_DIGITAL_MIC1)
{
if(HAL_DFSDM_ChannelDeInit(&haudio_in_dfsdm_channel[i]) != HAL_OK)
{
return BSP_ERROR_PERIPH_FAILURE;
}
if(HAL_DFSDM_FilterDeInit(&haudio_in_dfsdm_filter[i]) != HAL_OK)
{
return BSP_ERROR_PERIPH_FAILURE;
}
}
}
audio_init.Device = Audio_In_Ctx[Instance].Device;
audio_init.ChannelsNbr = Audio_In_Ctx[Instance].ChannelsNbr;
audio_init.SampleRate = SampleRate;
audio_init.BitsPerSample = Audio_In_Ctx[Instance].BitsPerSample;
audio_init.Volume = Audio_In_Ctx[Instance].Volume;
if(BSP_AUDIO_IN_Init(Instance, &audio_init) != BSP_ERROR_NONE)
{
ret = BSP_ERROR_NO_INIT;
}
}
/* Return BSP status */
return ret;
}
/**
* @brief Get Audio In frequency
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param SampleRate Audio Input frequency to be returned
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_GetSampleRate(uint32_t Instance, uint32_t *SampleRate)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Return audio in frequency */
*SampleRate = Audio_In_Ctx[Instance].SampleRate;
}
/* Return BSP status */
return ret;
}
/**
* @brief Set Audio In Resolution
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param BitsPerSample Input resolution to be set
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_SetBitsPerSample(uint32_t Instance, uint32_t BitsPerSample)
{
uint32_t i;
int32_t ret = BSP_ERROR_NONE;
BSP_AUDIO_Init_t audio_init;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else if(Audio_In_Ctx[Instance].State == AUDIO_IN_STATE_STOP)
{
if(Instance == 2U)
{
for(i = 0; i < DFSDM_MIC_NUMBER; i ++)
{
if(((Audio_In_Ctx[Instance].Device >> i) & AUDIO_IN_DEVICE_DIGITAL_MIC1) == AUDIO_IN_DEVICE_DIGITAL_MIC1)
{
if(HAL_DFSDM_ChannelDeInit(&haudio_in_dfsdm_channel[i]) != HAL_OK)
{
return BSP_ERROR_PERIPH_FAILURE;
}
}
}
}
audio_init.Device = Audio_In_Ctx[Instance].Device;
audio_init.ChannelsNbr = Audio_In_Ctx[Instance].ChannelsNbr;
audio_init.SampleRate = Audio_In_Ctx[Instance].SampleRate;
audio_init.BitsPerSample = BitsPerSample;
audio_init.Volume = Audio_In_Ctx[Instance].Volume;
if(BSP_AUDIO_IN_Init(Instance, &audio_init) != BSP_ERROR_NONE)
{
ret = BSP_ERROR_NO_INIT;
}
}
else
{
ret = BSP_ERROR_BUSY;
}
/* Return BSP status */
return ret;
}
/**
* @brief Get Audio In Resolution
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param BitsPerSample Input resolution to be returned
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_GetBitsPerSample(uint32_t Instance, uint32_t *BitsPerSample)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Return audio in resolution */
*BitsPerSample = Audio_In_Ctx[Instance].BitsPerSample;
}
return ret;
}
/**
* @brief Set Audio In Channel number
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param ChannelNbr Channel number to be used
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_SetChannelsNbr(uint32_t Instance, uint32_t ChannelNbr)
{
int32_t ret = BSP_ERROR_NONE;
if((Instance >= AUDIO_IN_INSTANCES_NBR) || (ChannelNbr > 2U))
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Update AudioIn Context */
Audio_In_Ctx[Instance].ChannelsNbr = ChannelNbr;
}
/* Return BSP status */
return ret;
}
/**
* @brief Get Audio In Channel number
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param ChannelNbr Channel number to be used
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_GetChannelsNbr(uint32_t Instance, uint32_t *ChannelNbr)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Channel number to be returned */
*ChannelNbr = Audio_In_Ctx[Instance].ChannelsNbr;
}
return ret;
}
/**
* @brief Set the current audio in volume level.
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param Volume Volume level to be returned
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_SetVolume(uint32_t Instance, uint32_t Volume)
{
int32_t ret;
if ((Instance >= AUDIO_IN_INSTANCES_NBR) || (Volume > 100U))
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Feature not supported */
ret = BSP_ERROR_FEATURE_NOT_SUPPORTED;
}
/* Return BSP status */
return ret;
}
/**
* @brief Get the current audio in volume level.
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param Volume Volume level to be returned
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_GetVolume(uint32_t Instance, uint32_t *Volume)
{
int32_t ret;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Input Volume to be returned */
*Volume = 0U;
/* Feature not supported */
ret = BSP_ERROR_FEATURE_NOT_SUPPORTED;
}
/* Return BSP status */
return ret;
}
/**
* @brief Get Audio In device
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param State Audio Out state
* @retval BSP status
*/
int32_t BSP_AUDIO_IN_GetState(uint32_t Instance, uint32_t *State)
{
int32_t ret = BSP_ERROR_NONE;
if(Instance >= AUDIO_IN_INSTANCES_NBR)
{
ret = BSP_ERROR_WRONG_PARAM;
}
else
{
/* Input State to be returned */
*State = Audio_In_Ctx[Instance].State;
}
return ret;
}
/**
* @brief This function handles Audio Out DMA interrupt requests.
* @param Instance Audio IN instance: 0 for SAI, 1 for SAI PDM and 2 for DFSDM
* @param InputDevice Can be:
* - AUDIO_IN_DEVICE_ANALOG_MIC
* - AUDIO_IN_DEVICE_DIGITAL_MIC1
* - AUDIO_IN_DEVICE_DIGITAL_MIC2
* - AUDIO_IN_DEVICE_DIGITAL_MIC3
* - AUDIO_IN_DEVICE_DIGITAL_MIC4
* @retval None
*/
void BSP_AUDIO_IN_IRQHandler(uint32_t Instance, uint32_t InputDevice)
{
if(((Instance == 0U) && (InputDevice == AUDIO_IN_DEVICE_DIGITAL_MIC1)) ||\
((Instance == 0U) && (InputDevice == AUDIO_IN_DEVICE_ANALOG_MIC)) ||\
((Instance == 1U) && (InputDevice == AUDIO_IN_DEVICE_DIGITAL_MIC1)))
{
HAL_DMA_IRQHandler(haudio_in_sai[Instance].hdmarx);
}
else
{
if((Instance == 2U) && (InputDevice >= AUDIO_IN_DEVICE_DIGITAL_MIC1) &&\
(InputDevice <= AUDIO_IN_DEVICE_DIGITAL_MIC_LAST))
{
HAL_DMA_IRQHandler(haudio_in_dfsdm_filter[POS_VAL(InputDevice)].hdmaReg);
}
}
}
#if (USE_HAL_DFSDM_REGISTER_CALLBACKS == 0) || !defined (USE_HAL_DFSDM_REGISTER_CALLBACKS)
/**
* @brief Regular conversion complete callback.
* @note In interrupt mode, user has to read conversion value in this function
using HAL_DFSDM_FilterGetRegularValue.
* @param hdfsdm_filter DFSDM filter handle.
* @retval None
*/
void HAL_DFSDM_FilterRegConvCpltCallback(DFSDM_Filter_HandleTypeDef *hdfsdm_filter)
{
uint32_t index;
static uint32_t DmaRecBuffCplt[DFSDM_MIC_NUMBER] = {0};
int32_t tmp;
if(Audio_In_Ctx[2].IsMultiBuff == 1U)
{
/* Call the record update function to get the second half */
BSP_AUDIO_IN_TransferComplete_CallBack(2);
}
else
{
if(hdfsdm_filter == &haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)])
{
DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] = 1;
}
if(hdfsdm_filter == &haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)])
{
DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] = 1;
}
if(DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] == 1U)
{
if(DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] == 1U)
{
#if (USE_BSP_CPU_CACHE_MAINTENANCE == 1)
SCB_InvalidateDCache_by_Addr((uint32_t *)&MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)][DEFAULT_AUDIO_IN_BUFFER_SIZE/2U], ((int32_t)DEFAULT_AUDIO_IN_BUFFER_SIZE/2)*4);
SCB_InvalidateDCache_by_Addr((uint32_t *)&MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)][DEFAULT_AUDIO_IN_BUFFER_SIZE/2U], ((int32_t)DEFAULT_AUDIO_IN_BUFFER_SIZE/2)*4);
#endif /* USE_BSP_CPU_CACHE_MAINTENANCE */
for(index = (DEFAULT_AUDIO_IN_BUFFER_SIZE/2U) ; index < DEFAULT_AUDIO_IN_BUFFER_SIZE; index++)
{
if(Audio_In_Ctx[2].ChannelsNbr == 2U)
{
tmp = MicRecBuff[0][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger] = (uint16_t)(tmp);
tmp = MicRecBuff[1][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger + 1U] = (uint16_t)(tmp);
}
else
{
tmp = MicRecBuff[0][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger] = (uint16_t)(tmp);
Audio_In_Ctx[2].pBuff[RecBuffTrigger + 1U] = (uint16_t)(tmp);
}
RecBuffTrigger +=2U;
}
#if (USE_BSP_CPU_CACHE_MAINTENANCE == 1)
SCB_CleanDCache_by_Addr((uint32_t *)Audio_In_Ctx[2].pBuff, (int32_t)Audio_In_Ctx[2].Size*2);
#endif /* USE_BSP_CPU_CACHE_MAINTENANCE */
DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] = 0;
DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] = 0;
}
}
/* Call Half Transfer Complete callback */
if(RecBuffTrigger == (Audio_In_Ctx[2].Size/4U))
{
if(RecBuffHalf == 0U)
{
RecBuffHalf = 1;
BSP_AUDIO_IN_HalfTransfer_CallBack(2);
}
}
/* Call Transfer Complete callback */
if(RecBuffTrigger == Audio_In_Ctx[2].Size/2U)
{
/* Reset Application Buffer Trigger */
RecBuffTrigger = 0;
RecBuffHalf = 0;
/* Call the record update function to get the next buffer to fill and its size (size is ignored) */
BSP_AUDIO_IN_TransferComplete_CallBack(2);
}
}
}
/**
* @brief Half regular conversion complete callback.
* @param hdfsdm_filter DFSDM filter handle.
* @retval None
*/
void HAL_DFSDM_FilterRegConvHalfCpltCallback(DFSDM_Filter_HandleTypeDef *hdfsdm_filter)
{
uint32_t index;
static uint32_t DmaRecHalfBuffCplt[DFSDM_MIC_NUMBER] = {0};
int32_t tmp;
if(Audio_In_Ctx[2].IsMultiBuff == 1U)
{
/* Call the record update function to get the first half */
BSP_AUDIO_IN_HalfTransfer_CallBack(2);
}
else
{
if(hdfsdm_filter == &haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)])
{
DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] = 1;
}
if(hdfsdm_filter == &haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)])
{
DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] = 1;
}
if(DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] == 1U)
{
if(DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] == 1U)
{
SCB_InvalidateDCache_by_Addr((uint32_t *)&MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)][0], ((int32_t)DEFAULT_AUDIO_IN_BUFFER_SIZE/2)*4);
SCB_InvalidateDCache_by_Addr((uint32_t *)&MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)][0], ((int32_t)DEFAULT_AUDIO_IN_BUFFER_SIZE/2)*4);
for(index = 0 ; index < (DEFAULT_AUDIO_IN_BUFFER_SIZE/2U); index++)
{
if(Audio_In_Ctx[2].ChannelsNbr == 2U)
{
tmp = MicRecBuff[0][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger] = (uint16_t)(tmp);
tmp = MicRecBuff[1][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger + 1U] = (uint16_t)(tmp);
}
else
{
tmp = MicRecBuff[0][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger] = (uint16_t)(tmp);
Audio_In_Ctx[2].pBuff[RecBuffTrigger + 1U] = (uint16_t)(tmp);
}
RecBuffTrigger +=2U;
}
#if (USE_BSP_CPU_CACHE_MAINTENANCE == 1)
SCB_CleanDCache_by_Addr((uint32_t *)Audio_In_Ctx[2].pBuff, ((int32_t)Audio_In_Ctx[2].Size*2));
#endif /* USE_BSP_CPU_CACHE_MAINTENANCE */
DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] = 0;
DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] = 0;
}
}
/* Call Half Transfer Complete callback */
if(RecBuffTrigger == (Audio_In_Ctx[2].Size/4U))
{
if(RecBuffHalf == 0U)
{
RecBuffHalf = 1;
BSP_AUDIO_IN_HalfTransfer_CallBack(2);
}
}
/* Call Transfer Complete callback */
if(RecBuffTrigger == Audio_In_Ctx[2].Size/2U)
{
/* Reset Application Buffer Trigger */
RecBuffTrigger = 0;
RecBuffHalf = 0;
/* Call the record update function to get the next buffer to fill and its size (size is ignored) */
BSP_AUDIO_IN_TransferComplete_CallBack(2);
}
}
}
#endif /* (USE_HAL_DFSDM_REGISTER_CALLBACKS == 0) || !defined (USE_HAL_DFSDM_REGISTER_CALLBACKS) */
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 0) || !defined (USE_HAL_SAI_REGISTER_CALLBACKS)
/**
* @brief Half reception complete callback.
* @param hsai SAI handle.
* @retval None
*/
void HAL_SAI_RxHalfCpltCallback(SAI_HandleTypeDef *hsai)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hsai);
/* Call the record update function to get the first half */
BSP_AUDIO_IN_HalfTransfer_CallBack(0);
}
/**
* @brief Reception complete callback.
* @param hsai SAI handle.
* @retval None
*/
void HAL_SAI_RxCpltCallback(SAI_HandleTypeDef *hsai)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hsai);
/* Call the record update function to get the second half */
BSP_AUDIO_IN_TransferComplete_CallBack(0);
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 0) || !defined (USE_HAL_SAI_REGISTER_CALLBACKS) */
/**
* @brief User callback when record buffer is filled.
* @retval None
*/
__weak void BSP_AUDIO_IN_TransferComplete_CallBack(uint32_t Instance)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(Instance);
/* This function should be implemented by the user application.
It is called into this driver when the current buffer is filled
to prepare the next buffer pointer and its size. */
}
/**
* @brief Manages the DMA Half Transfer complete event.
* @retval None
*/
__weak void BSP_AUDIO_IN_HalfTransfer_CallBack(uint32_t Instance)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(Instance);
/* This function should be implemented by the user application.
It is called into this driver when the current buffer is filled
to prepare the next buffer pointer and its size. */
}
/**
* @brief Audio IN Error callback function.
* @retval None
*/
__weak void BSP_AUDIO_IN_Error_CallBack(uint32_t Instance)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(Instance);
/* This function is called when an Interrupt due to transfer error on or peripheral
error occurs. */
}
/**
* @}
*/
/** @defgroup STM32H735G_DISCO_AUDIO_IN_Private_Functions AUDIO IN Private Functions
* @{
*/
#if (USE_HAL_SAI_REGISTER_CALLBACKS == 1)
/**
* @brief Half reception complete callback.
* @param hsai SAI handle.
* @retval None
*/
static void SAI_RxHalfCpltCallback(SAI_HandleTypeDef *hsai)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hsai);
/* Call the record update function to get the first half */
BSP_AUDIO_IN_HalfTransfer_CallBack(0);
}
/**
* @brief Reception complete callback.
* @param hsai SAI handle.
* @retval None
*/
static void SAI_RxCpltCallback(SAI_HandleTypeDef *hsai)
{
/* Prevent unused argument(s) compilation warning */
UNUSED(hsai);
/* Call the record update function to get the second half */
BSP_AUDIO_IN_TransferComplete_CallBack(0);
}
#endif /* (USE_HAL_SAI_REGISTER_CALLBACKS == 1) */
#if (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1)
/**
* @brief Regular conversion complete callback.
* @note In interrupt mode, user has to read conversion value in this function
* using HAL_DFSDM_FilterGetRegularValue.
* @param hdfsdm_filter DFSDM filter handle.
* @retval None
*/
static void DFSDM_FilterRegConvCpltCallback(DFSDM_Filter_HandleTypeDef *hdfsdm_filter)
{
uint32_t index;
static uint32_t DmaRecBuffCplt[DFSDM_MIC_NUMBER] = {0};
int32_t tmp;
if(Audio_In_Ctx[2].IsMultiBuff == 1U)
{
/* Call the record update function to get the second half */
BSP_AUDIO_IN_TransferComplete_CallBack(2);
}
else
{
if(hdfsdm_filter == &haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)])
{
DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] = 1;
}
if(hdfsdm_filter == &haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)])
{
DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] = 1;
}
if(DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] == 1U)
{
if(DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] == 1U)
{
#if (USE_BSP_CPU_CACHE_MAINTENANCE == 1)
SCB_InvalidateDCache_by_Addr((uint32_t *)&MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)][DEFAULT_AUDIO_IN_BUFFER_SIZE/2U], (int32_t)((int32_t)DEFAULT_AUDIO_IN_BUFFER_SIZE/2)*4);
SCB_InvalidateDCache_by_Addr((uint32_t *)&MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)][DEFAULT_AUDIO_IN_BUFFER_SIZE/2U], (int32_t)((int32_t)DEFAULT_AUDIO_IN_BUFFER_SIZE/2)*4);
#endif /* USE_BSP_CPU_CACHE_MAINTENANCE */
for(index = (DEFAULT_AUDIO_IN_BUFFER_SIZE/2U) ; index < DEFAULT_AUDIO_IN_BUFFER_SIZE; index++)
{
if(Audio_In_Ctx[2].ChannelsNbr == 2U)
{
tmp = MicRecBuff[0][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger] = (uint16_t)(tmp);
tmp = MicRecBuff[1][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger + 1U] = (uint16_t)(tmp);
}
else
{
tmp = MicRecBuff[0][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger] = (uint16_t)(tmp);
Audio_In_Ctx[2].pBuff[RecBuffTrigger + 1U] = (uint16_t)(tmp);
}
RecBuffTrigger +=2U;
}
#if (USE_BSP_CPU_CACHE_MAINTENANCE == 1)
SCB_CleanDCache_by_Addr((uint32_t *)Audio_In_Ctx[2].pBuff, (int32_t)Audio_In_Ctx[2].Size*2);
#endif /* USE_BSP_CPU_CACHE_MAINTENANCE */
DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] = 0;
DmaRecBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] = 0;
}
}
/* Call Half Transfer Complete callback */
if(RecBuffTrigger == (Audio_In_Ctx[2].Size/4U))
{
if(RecBuffHalf == 0U)
{
RecBuffHalf = 1;
BSP_AUDIO_IN_HalfTransfer_CallBack(2);
}
}
/* Call Transfer Complete callback */
if(RecBuffTrigger == Audio_In_Ctx[2].Size/2U)
{
/* Reset Application Buffer Trigger */
RecBuffTrigger = 0;
RecBuffHalf = 0;
/* Call the record update function to get the next buffer to fill and its size (size is ignored) */
BSP_AUDIO_IN_TransferComplete_CallBack(2);
}
}
}
/**
* @brief Half regular conversion complete callback.
* @param hdfsdm_filter DFSDM filter handle.
* @retval None
*/
static void DFSDM_FilterRegConvHalfCpltCallback(DFSDM_Filter_HandleTypeDef *hdfsdm_filter)
{
uint32_t index;
static uint32_t DmaRecHalfBuffCplt[DFSDM_MIC_NUMBER] = {0};
int32_t tmp;
if(Audio_In_Ctx[2].IsMultiBuff == 1U)
{
/* Call the record update function to get the first half */
BSP_AUDIO_IN_HalfTransfer_CallBack(2);
}
else
{
if(hdfsdm_filter == &haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)])
{
DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] = 1;
}
if(hdfsdm_filter == &haudio_in_dfsdm_filter[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)])
{
DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] = 1;
}
if(DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] == 1U)
{
if(DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] == 1U)
{
#if (USE_BSP_CPU_CACHE_MAINTENANCE == 1)
SCB_InvalidateDCache_by_Addr((uint32_t *)&MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)][0], (int32_t)((int32_t)DEFAULT_AUDIO_IN_BUFFER_SIZE/2)*4);
SCB_InvalidateDCache_by_Addr((uint32_t *)&MicRecBuff[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)][0], (int32_t)((int32_t)DEFAULT_AUDIO_IN_BUFFER_SIZE/2)*4);
#endif /* USE_BSP_CPU_CACHE_MAINTENANCE */
for(index = 0 ; index < (DEFAULT_AUDIO_IN_BUFFER_SIZE/2U); index++)
{
if(Audio_In_Ctx[2].ChannelsNbr == 2U)
{
tmp = MicRecBuff[0][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger] = (uint16_t)(tmp);
tmp = MicRecBuff[1][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger + 1U] = (uint16_t)(tmp);
}
else
{
tmp = MicRecBuff[0][index] / 256;
tmp = SaturaLH(tmp, -32768, 32767);
Audio_In_Ctx[2].pBuff[RecBuffTrigger] = (uint16_t)(tmp);
Audio_In_Ctx[2].pBuff[RecBuffTrigger + 1U] = (uint16_t)(tmp);
}
RecBuffTrigger +=2U;
}
#if (USE_BSP_CPU_CACHE_MAINTENANCE == 1)
SCB_CleanDCache_by_Addr((uint32_t *)Audio_In_Ctx[2].pBuff, (int32_t)Audio_In_Ctx[2].Size*2);
#endif /* USE_BSP_CPU_CACHE_MAINTENANCE */
DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] = 0;
DmaRecHalfBuffCplt[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] = 0;
}
}
/* Call Half Transfer Complete callback */
if(RecBuffTrigger == (Audio_In_Ctx[2].Size/4U))
{
if(RecBuffHalf == 0U)
{
RecBuffHalf = 1;
BSP_AUDIO_IN_HalfTransfer_CallBack(2);
}
}
/* Call Transfer Complete callback */
if(RecBuffTrigger == Audio_In_Ctx[2].Size/2U)
{
/* Reset Application Buffer Trigger */
RecBuffTrigger = 0;
RecBuffHalf = 0;
/* Call the record update function to get the next buffer to fill and its size (size is ignored) */
BSP_AUDIO_IN_TransferComplete_CallBack(2);
}
}
}
#endif /* (USE_HAL_DFSDM_REGISTER_CALLBACKS == 1) */
/**
* @brief Initialize the DFSDM channel MSP.
* @param hDfsdmChannel DFSDM Channel handle
* @retval None
*/
static void DFSDM_ChannelMspInit(DFSDM_Channel_HandleTypeDef *hDfsdmChannel)
{
GPIO_InitTypeDef GPIO_InitStruct;
/* Prevent unused argument(s) compilation warning */
UNUSED(hDfsdmChannel);
/* DFSDM pins configuration: DFSDM_CKOUT, DMIC_DATIN pins ------------------*/
GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
GPIO_InitStruct.Pull = GPIO_NOPULL;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_VERY_HIGH;
/* Enable DFSDM clock */
AUDIO_DFSDM1_CLK_ENABLE();
/* Enable GPIO clock */
AUDIO_DFSDM1_CKOUT_GPIO_CLK_ENABLE();
AUDIO_DFSDMx_DATIN_MIC1_GPIO_CLK_ENABLE();
AUDIO_DFSDMx_DATIN_MIC2_GPIO_CLK_ENABLE();
AUDIO_DFSDMx_DATIN_MIC3_GPIO_CLK_ENABLE();
AUDIO_DFSDMx_DATIN_MIC4_GPIO_CLK_ENABLE();
GPIO_InitStruct.Pin = AUDIO_DFSDM1_CKOUT_PIN;
GPIO_InitStruct.Alternate = AUDIO_DFSDM1_CKOUT_AF;
HAL_GPIO_Init(AUDIO_DFSDM1_CKOUT_GPIO_PORT, &GPIO_InitStruct);
GPIO_InitStruct.Pin = AUDIO_DFSDMx_DATIN_MIC1_PIN;
GPIO_InitStruct.Alternate = AUDIO_DFSDMx_DATIN_MIC1_AF;
HAL_GPIO_Init(AUDIO_DFSDMx_DATIN_MIC1_GPIO_PORT, &GPIO_InitStruct);
GPIO_InitStruct.Pin = AUDIO_DFSDMx_DATIN_MIC2_PIN;
GPIO_InitStruct.Alternate = AUDIO_DFSDMx_DATIN_MIC2_AF;
HAL_GPIO_Init(AUDIO_DFSDMx_DATIN_MIC2_GPIO_PORT, &GPIO_InitStruct);
GPIO_InitStruct.Pin = AUDIO_DFSDMx_DATIN_MIC3_PIN;
GPIO_InitStruct.Alternate = AUDIO_DFSDMx_DATIN_MIC3_AF;
HAL_GPIO_Init(AUDIO_DFSDMx_DATIN_MIC3_GPIO_PORT, &GPIO_InitStruct);
GPIO_InitStruct.Pin = AUDIO_DFSDMx_DATIN_MIC4_PIN;
GPIO_InitStruct.Alternate = AUDIO_DFSDMx_DATIN_MIC4_AF;
HAL_GPIO_Init(AUDIO_DFSDMx_DATIN_MIC4_GPIO_PORT, &GPIO_InitStruct);
}
/**
* @brief DeInitialize the DFSDM channel MSP.
* @param hDfsdmChannel DFSDM Channel handle
* @retval None
*/
static void DFSDM_ChannelMspDeInit(DFSDM_Channel_HandleTypeDef *hDfsdmChannel)
{
GPIO_InitTypeDef GPIO_InitStruct;
/* Prevent unused argument(s) compilation warning */
UNUSED(hDfsdmChannel);
/* DFSDM pins configuration: DFSDM_CKOUT, DMIC_DATIN pins ------------------*/
GPIO_InitStruct.Pin = AUDIO_DFSDM1_CKOUT_PIN;
HAL_GPIO_DeInit(AUDIO_DFSDM1_CKOUT_GPIO_PORT, GPIO_InitStruct.Pin);
GPIO_InitStruct.Pin = AUDIO_DFSDMx_DATIN_MIC1_PIN;
HAL_GPIO_DeInit(AUDIO_DFSDMx_DATIN_MIC1_GPIO_PORT, GPIO_InitStruct.Pin);
GPIO_InitStruct.Pin = AUDIO_DFSDMx_DATIN_MIC2_PIN;
HAL_GPIO_DeInit(AUDIO_DFSDMx_DATIN_MIC2_GPIO_PORT, GPIO_InitStruct.Pin);
GPIO_InitStruct.Pin = AUDIO_DFSDMx_DATIN_MIC3_PIN;
HAL_GPIO_DeInit(AUDIO_DFSDMx_DATIN_MIC3_GPIO_PORT, GPIO_InitStruct.Pin);
GPIO_InitStruct.Pin = AUDIO_DFSDMx_DATIN_MIC4_PIN;
HAL_GPIO_DeInit(AUDIO_DFSDMx_DATIN_MIC4_GPIO_PORT, GPIO_InitStruct.Pin);
}
/**
* @brief Initialize the DFSDM filter MSP.
* @param hDfsdmFilter DFSDM Filter handle
* @retval None
*/
static void DFSDM_FilterMspInit(DFSDM_Filter_HandleTypeDef *hDfsdmFilter)
{
uint32_t i, mic_num, mic_init[DFSDM_MIC_NUMBER] = {0};
IRQn_Type AUDIO_DFSDM_DMAx_MIC_IRQHandler[DFSDM_MIC_NUMBER] = {AUDIO_DFSDMx_DMAx_MIC1_IRQ, AUDIO_DFSDMx_DMAx_MIC2_IRQ, AUDIO_DFSDMx_DMAx_MIC3_IRQ, AUDIO_DFSDMx_DMAx_MIC4_IRQ};
DMA_Stream_TypeDef* AUDIO_DFSDMx_DMAx_MIC_STREAM[DFSDM_MIC_NUMBER] = {AUDIO_DFSDMx_DMAx_MIC1_STREAM, AUDIO_DFSDMx_DMAx_MIC2_STREAM, AUDIO_DFSDMx_DMAx_MIC3_STREAM, AUDIO_DFSDMx_DMAx_MIC4_STREAM};
uint32_t AUDIO_DFSDMx_DMAx_MIC_REQUEST[DFSDM_MIC_NUMBER] = {AUDIO_DFSDMx_DMAx_MIC1_REQUEST, AUDIO_DFSDMx_DMAx_MIC2_REQUEST, AUDIO_DFSDMx_DMAx_MIC3_REQUEST, AUDIO_DFSDMx_DMAx_MIC4_REQUEST};
/* Prevent unused argument(s) compilation warning */
UNUSED(hDfsdmFilter);
/* Enable DFSDM clock */
AUDIO_DFSDM1_CLK_ENABLE();
/* Enable the DMA clock */
AUDIO_DFSDM1_DMAx_CLK_ENABLE();
for(i = 0; i < DFSDM_MIC_NUMBER; i++)
{
if((mic_init[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC1)] != 1U) && ((Audio_In_Ctx[2].Device & AUDIO_IN_DEVICE_DIGITAL_MIC1) == AUDIO_IN_DEVICE_DIGITAL_MIC1))
{
mic_num = 0U;
mic_init[mic_num] = 1;
}
else if((mic_init[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC2)] != 1U) && ((Audio_In_Ctx[2].Device & AUDIO_IN_DEVICE_DIGITAL_MIC2) == AUDIO_IN_DEVICE_DIGITAL_MIC2))
{
mic_num = 1U;
mic_init[mic_num] = 1;
}
else if((mic_init[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC3)] != 1U) &&((Audio_In_Ctx[2].Device & AUDIO_IN_DEVICE_DIGITAL_MIC3) == AUDIO_IN_DEVICE_DIGITAL_MIC3))
{
mic_num = 2U;
mic_init[mic_num] = 1;
}
else if((mic_init[POS_VAL(AUDIO_IN_DEVICE_DIGITAL_MIC4)] != 1U) && ((Audio_In_Ctx[2].Device & AUDIO_IN_DEVICE_DIGITAL_MIC4) == AUDIO_IN_DEVICE_DIGITAL_MIC4))
{
mic_num = 3U;
mic_init[mic_num] = 1;
}
else
{
break;
}
/* Configure the hDmaDfsdm[i] handle parameters */
hDmaDfsdm[mic_num].Init.Request = AUDIO_DFSDMx_DMAx_MIC_REQUEST[mic_num];
hDmaDfsdm[mic_num].Instance = AUDIO_DFSDMx_DMAx_MIC_STREAM[mic_num];
hDmaDfsdm[mic_num].Init.Direction = DMA_PERIPH_TO_MEMORY;
hDmaDfsdm[mic_num].Init.PeriphInc = DMA_PINC_DISABLE;
hDmaDfsdm[mic_num].Init.MemInc = DMA_MINC_ENABLE;
hDmaDfsdm[mic_num].Init.PeriphDataAlignment = DMA_PDATAALIGN_WORD;
hDmaDfsdm[mic_num].Init.MemDataAlignment = DMA_MDATAALIGN_WORD;
hDmaDfsdm[mic_num].Init.Mode = DMA_CIRCULAR;
hDmaDfsdm[mic_num].Init.Priority = DMA_PRIORITY_HIGH;
hDmaDfsdm[mic_num].Init.FIFOMode = DMA_FIFOMODE_DISABLE;
hDmaDfsdm[mic_num].Init.MemBurst = DMA_MBURST_SINGLE;
hDmaDfsdm[mic_num].Init.PeriphBurst = DMA_PBURST_SINGLE;
hDmaDfsdm[mic_num].State = HAL_DMA_STATE_RESET;
/* Associate the DMA handle */
__HAL_LINKDMA(&haudio_in_dfsdm_filter[mic_num], hdmaReg, hDmaDfsdm[mic_num]);
/* Reset DMA handle state */
__HAL_DMA_RESET_HANDLE_STATE(&hDmaDfsdm[mic_num]);
/* Configure the DMA Channel */
(void)HAL_DMA_Init(&hDmaDfsdm[mic_num]);
/* DMA IRQ Channel configuration */
HAL_NVIC_SetPriority(AUDIO_DFSDM_DMAx_MIC_IRQHandler[mic_num], BSP_AUDIO_IN_IT_PRIORITY, 0);
HAL_NVIC_EnableIRQ(AUDIO_DFSDM_DMAx_MIC_IRQHandler[mic_num]);
}
}
/**
* @brief DeInitialize the DFSDM filter MSP.
* @param hDfsdmFilter DFSDM Filter handle
* @retval None
*/
static void DFSDM_FilterMspDeInit(DFSDM_Filter_HandleTypeDef *hDfsdmFilter)
{
uint32_t i;
/* Prevent unused argument(s) compilation warning */
UNUSED(hDfsdmFilter);
/* Configure the DMA Channel */
for(i = 0; i < DFSDM_MIC_NUMBER; i++)
{
if(hDmaDfsdm[i].Instance != NULL)
{
(void)HAL_DMA_DeInit(&hDmaDfsdm[i]);
}
}
}
/**
* @}
*/
/**
* @}
*/
/**
* @}
*/
/**
* @}
*/