4236 lines
138 KiB
C
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]);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* @}
|
|
*/
|