Merge pull request #2935 from ErnestChen1/master
[bsp][stm32][stm32f429-atk-apollo][audio]add f429 audio driver
This commit is contained in:
commit
33e774358f
File diff suppressed because one or more lines are too long
|
@ -57,7 +57,7 @@
|
||||||
/* #define HAL_LTDC_MODULE_ENABLED */
|
/* #define HAL_LTDC_MODULE_ENABLED */
|
||||||
/* #define HAL_RNG_MODULE_ENABLED */
|
/* #define HAL_RNG_MODULE_ENABLED */
|
||||||
#define HAL_RTC_MODULE_ENABLED
|
#define HAL_RTC_MODULE_ENABLED
|
||||||
/* #define HAL_SAI_MODULE_ENABLED */
|
#define HAL_SAI_MODULE_ENABLED
|
||||||
#define HAL_SD_MODULE_ENABLED
|
#define HAL_SD_MODULE_ENABLED
|
||||||
/* #define HAL_MMC_MODULE_ENABLED */
|
/* #define HAL_MMC_MODULE_ENABLED */
|
||||||
#define HAL_SPI_MODULE_ENABLED
|
#define HAL_SPI_MODULE_ENABLED
|
||||||
|
|
|
@ -14,114 +14,120 @@ KeepUserPlacement=false
|
||||||
Mcu.Family=STM32F4
|
Mcu.Family=STM32F4
|
||||||
Mcu.IP0=ADC1
|
Mcu.IP0=ADC1
|
||||||
Mcu.IP1=CAN1
|
Mcu.IP1=CAN1
|
||||||
Mcu.IP10=SPI2
|
Mcu.IP10=SPI1
|
||||||
Mcu.IP11=SPI5
|
Mcu.IP11=SPI2
|
||||||
Mcu.IP12=SYS
|
Mcu.IP12=SPI5
|
||||||
Mcu.IP13=TIM2
|
Mcu.IP13=SYS
|
||||||
Mcu.IP14=TIM11
|
Mcu.IP14=TIM2
|
||||||
Mcu.IP15=TIM13
|
Mcu.IP15=TIM11
|
||||||
Mcu.IP16=TIM14
|
Mcu.IP16=TIM13
|
||||||
Mcu.IP17=USART1
|
Mcu.IP17=TIM14
|
||||||
Mcu.IP18=USART2
|
Mcu.IP18=USART1
|
||||||
Mcu.IP19=USART3
|
Mcu.IP19=USART2
|
||||||
Mcu.IP2=ETH
|
Mcu.IP2=ETH
|
||||||
|
Mcu.IP20=USART3
|
||||||
Mcu.IP3=FMC
|
Mcu.IP3=FMC
|
||||||
Mcu.IP4=IWDG
|
Mcu.IP4=IWDG
|
||||||
Mcu.IP5=NVIC
|
Mcu.IP5=NVIC
|
||||||
Mcu.IP6=RCC
|
Mcu.IP6=RCC
|
||||||
Mcu.IP7=RTC
|
Mcu.IP7=RTC
|
||||||
Mcu.IP8=SDIO
|
Mcu.IP8=SAI1
|
||||||
Mcu.IP9=SPI1
|
Mcu.IP9=SDIO
|
||||||
Mcu.IPNb=20
|
Mcu.IPNb=21
|
||||||
Mcu.Name=STM32F429I(E-G)Tx
|
Mcu.Name=STM32F429I(E-G)Tx
|
||||||
Mcu.Package=LQFP176
|
Mcu.Package=LQFP176
|
||||||
Mcu.Pin0=PC14/OSC32_IN
|
Mcu.Pin0=PE2
|
||||||
Mcu.Pin1=PC15/OSC32_OUT
|
Mcu.Pin1=PE3
|
||||||
Mcu.Pin10=PF9
|
Mcu.Pin10=PF3
|
||||||
Mcu.Pin11=PH0/OSC_IN
|
Mcu.Pin11=PF4
|
||||||
Mcu.Pin12=PH1/OSC_OUT
|
Mcu.Pin12=PF5
|
||||||
Mcu.Pin13=PC0
|
Mcu.Pin13=PF7
|
||||||
Mcu.Pin14=PC1
|
Mcu.Pin14=PF8
|
||||||
Mcu.Pin15=PC2
|
Mcu.Pin15=PF9
|
||||||
Mcu.Pin16=PC3
|
Mcu.Pin16=PH0/OSC_IN
|
||||||
Mcu.Pin17=PA1
|
Mcu.Pin17=PH1/OSC_OUT
|
||||||
Mcu.Pin18=PA2
|
Mcu.Pin18=PC0
|
||||||
Mcu.Pin19=PA3
|
Mcu.Pin19=PC1
|
||||||
Mcu.Pin2=PF0
|
Mcu.Pin2=PE4
|
||||||
Mcu.Pin20=PA5
|
Mcu.Pin20=PC2
|
||||||
Mcu.Pin21=PA6
|
Mcu.Pin21=PC3
|
||||||
Mcu.Pin22=PA7
|
Mcu.Pin22=PA1
|
||||||
Mcu.Pin23=PC4
|
Mcu.Pin23=PA2
|
||||||
Mcu.Pin24=PC5
|
Mcu.Pin24=PA3
|
||||||
Mcu.Pin25=PF11
|
Mcu.Pin25=PA5
|
||||||
Mcu.Pin26=PF12
|
Mcu.Pin26=PA6
|
||||||
Mcu.Pin27=PF13
|
Mcu.Pin27=PA7
|
||||||
Mcu.Pin28=PF14
|
Mcu.Pin28=PC4
|
||||||
Mcu.Pin29=PF15
|
Mcu.Pin29=PC5
|
||||||
Mcu.Pin3=PF1
|
Mcu.Pin3=PE5
|
||||||
Mcu.Pin30=PG0
|
Mcu.Pin30=PF11
|
||||||
Mcu.Pin31=PG1
|
Mcu.Pin31=PF12
|
||||||
Mcu.Pin32=PE7
|
Mcu.Pin32=PF13
|
||||||
Mcu.Pin33=PE8
|
Mcu.Pin33=PF14
|
||||||
Mcu.Pin34=PE9
|
Mcu.Pin34=PF15
|
||||||
Mcu.Pin35=PE10
|
Mcu.Pin35=PG0
|
||||||
Mcu.Pin36=PE11
|
Mcu.Pin36=PG1
|
||||||
Mcu.Pin37=PE12
|
Mcu.Pin37=PE7
|
||||||
Mcu.Pin38=PE13
|
Mcu.Pin38=PE8
|
||||||
Mcu.Pin39=PE14
|
Mcu.Pin39=PE9
|
||||||
Mcu.Pin4=PF2
|
Mcu.Pin4=PE6
|
||||||
Mcu.Pin40=PE15
|
Mcu.Pin40=PE10
|
||||||
Mcu.Pin41=PB10
|
Mcu.Pin41=PE11
|
||||||
Mcu.Pin42=PB11
|
Mcu.Pin42=PE12
|
||||||
Mcu.Pin43=PB13
|
Mcu.Pin43=PE13
|
||||||
Mcu.Pin44=PB14
|
Mcu.Pin44=PE14
|
||||||
Mcu.Pin45=PB15
|
Mcu.Pin45=PE15
|
||||||
Mcu.Pin46=PD8
|
Mcu.Pin46=PB10
|
||||||
Mcu.Pin47=PD9
|
Mcu.Pin47=PB11
|
||||||
Mcu.Pin48=PD10
|
Mcu.Pin48=PB13
|
||||||
Mcu.Pin49=PD14
|
Mcu.Pin49=PB14
|
||||||
Mcu.Pin5=PF3
|
Mcu.Pin5=PC14/OSC32_IN
|
||||||
Mcu.Pin50=PD15
|
Mcu.Pin50=PB15
|
||||||
Mcu.Pin51=PG2
|
Mcu.Pin51=PD8
|
||||||
Mcu.Pin52=PG4
|
Mcu.Pin52=PD9
|
||||||
Mcu.Pin53=PG5
|
Mcu.Pin53=PD10
|
||||||
Mcu.Pin54=PG8
|
Mcu.Pin54=PD14
|
||||||
Mcu.Pin55=PC8
|
Mcu.Pin55=PD15
|
||||||
Mcu.Pin56=PC9
|
Mcu.Pin56=PG2
|
||||||
Mcu.Pin57=PA9
|
Mcu.Pin57=PG4
|
||||||
Mcu.Pin58=PA10
|
Mcu.Pin58=PG5
|
||||||
Mcu.Pin59=PA11
|
Mcu.Pin59=PG8
|
||||||
Mcu.Pin6=PF4
|
Mcu.Pin6=PC15/OSC32_OUT
|
||||||
Mcu.Pin60=PA12
|
Mcu.Pin60=PC8
|
||||||
Mcu.Pin61=PA13
|
Mcu.Pin61=PC9
|
||||||
Mcu.Pin62=PA14
|
Mcu.Pin62=PA9
|
||||||
Mcu.Pin63=PC10
|
Mcu.Pin63=PA10
|
||||||
Mcu.Pin64=PC11
|
Mcu.Pin64=PA11
|
||||||
Mcu.Pin65=PC12
|
Mcu.Pin65=PA12
|
||||||
Mcu.Pin66=PD0
|
Mcu.Pin66=PA13
|
||||||
Mcu.Pin67=PD1
|
Mcu.Pin67=PA14
|
||||||
Mcu.Pin68=PD2
|
Mcu.Pin68=PC10
|
||||||
Mcu.Pin69=PD5
|
Mcu.Pin69=PC11
|
||||||
Mcu.Pin7=PF5
|
Mcu.Pin7=PF0
|
||||||
Mcu.Pin70=PD6
|
Mcu.Pin70=PC12
|
||||||
Mcu.Pin71=PG11
|
Mcu.Pin71=PD0
|
||||||
Mcu.Pin72=PG13
|
Mcu.Pin72=PD1
|
||||||
Mcu.Pin73=PG14
|
Mcu.Pin73=PD2
|
||||||
Mcu.Pin74=PG15
|
Mcu.Pin74=PD5
|
||||||
Mcu.Pin75=PB3
|
Mcu.Pin75=PD6
|
||||||
Mcu.Pin76=PB5
|
Mcu.Pin76=PG11
|
||||||
Mcu.Pin77=PE0
|
Mcu.Pin77=PG13
|
||||||
Mcu.Pin78=PE1
|
Mcu.Pin78=PG14
|
||||||
Mcu.Pin79=VP_IWDG_VS_IWDG
|
Mcu.Pin79=PG15
|
||||||
Mcu.Pin8=PF7
|
Mcu.Pin8=PF1
|
||||||
Mcu.Pin80=VP_RTC_VS_RTC_Activate
|
Mcu.Pin80=PB3
|
||||||
Mcu.Pin81=VP_SYS_VS_Systick
|
Mcu.Pin81=PB5
|
||||||
Mcu.Pin82=VP_TIM2_VS_ClockSourceINT
|
Mcu.Pin82=PE0
|
||||||
Mcu.Pin83=VP_TIM11_VS_ClockSourceINT
|
Mcu.Pin83=PE1
|
||||||
Mcu.Pin84=VP_TIM13_VS_ClockSourceINT
|
Mcu.Pin84=VP_IWDG_VS_IWDG
|
||||||
Mcu.Pin85=VP_TIM14_VS_ClockSourceINT
|
Mcu.Pin85=VP_RTC_VS_RTC_Activate
|
||||||
Mcu.Pin9=PF8
|
Mcu.Pin86=VP_SYS_VS_Systick
|
||||||
Mcu.PinsNb=86
|
Mcu.Pin87=VP_TIM2_VS_ClockSourceINT
|
||||||
|
Mcu.Pin88=VP_TIM11_VS_ClockSourceINT
|
||||||
|
Mcu.Pin89=VP_TIM13_VS_ClockSourceINT
|
||||||
|
Mcu.Pin9=PF2
|
||||||
|
Mcu.Pin90=VP_TIM14_VS_ClockSourceINT
|
||||||
|
Mcu.PinsNb=91
|
||||||
Mcu.ThirdPartyNb=0
|
Mcu.ThirdPartyNb=0
|
||||||
Mcu.UserConstants=
|
Mcu.UserConstants=
|
||||||
Mcu.UserName=STM32F429IGTx
|
Mcu.UserName=STM32F429IGTx
|
||||||
|
@ -230,6 +236,16 @@ PE12.Signal=FMC_D9_DA9
|
||||||
PE13.Signal=FMC_D10_DA10
|
PE13.Signal=FMC_D10_DA10
|
||||||
PE14.Signal=FMC_D11_DA11
|
PE14.Signal=FMC_D11_DA11
|
||||||
PE15.Signal=FMC_D12_DA12
|
PE15.Signal=FMC_D12_DA12
|
||||||
|
PE2.Mode=SAI_A_MasterWithClock
|
||||||
|
PE2.Signal=SAI1_MCLK_A
|
||||||
|
PE3.Mode=SAI_B_SyncSlave
|
||||||
|
PE3.Signal=SAI1_SD_B
|
||||||
|
PE4.Mode=SAI_A_MasterWithClock
|
||||||
|
PE4.Signal=SAI1_FS_A
|
||||||
|
PE5.Mode=SAI_A_MasterWithClock
|
||||||
|
PE5.Signal=SAI1_SCK_A
|
||||||
|
PE6.Mode=SAI_A_MasterWithClock
|
||||||
|
PE6.Signal=SAI1_SD_A
|
||||||
PE7.Signal=FMC_D4_DA4
|
PE7.Signal=FMC_D4_DA4
|
||||||
PE8.Signal=FMC_D5_DA5
|
PE8.Signal=FMC_D5_DA5
|
||||||
PE9.Signal=FMC_D6_DA6
|
PE9.Signal=FMC_D6_DA6
|
||||||
|
@ -313,8 +329,8 @@ RCC.HCLKFreq_Value=180000000
|
||||||
RCC.HSE_VALUE=25000000
|
RCC.HSE_VALUE=25000000
|
||||||
RCC.HSI_VALUE=16000000
|
RCC.HSI_VALUE=16000000
|
||||||
RCC.I2SClocksFreq_Value=160000000
|
RCC.I2SClocksFreq_Value=160000000
|
||||||
RCC.IPParameters=48MHZClocksFreq_Value,AHBFreq_Value,APB1CLKDivider,APB1Freq_Value,APB1TimFreq_Value,APB2CLKDivider,APB2Freq_Value,APB2TimFreq_Value,CortexFreq_Value,EthernetFreq_Value,FCLKCortexFreq_Value,FamilyName,HCLKFreq_Value,HSE_VALUE,HSI_VALUE,I2SClocksFreq_Value,LCDTFTFreq_Value,LSI_VALUE,MCO2PinFreq_Value,PLLCLKFreq_Value,PLLM,PLLN,PLLQ,PLLQCLKFreq_Value,PLLSourceVirtual,RCC_RTC_Clock_Source,RCC_RTC_Clock_SourceVirtual,RTCFreq_Value,RTCHSEDivFreq_Value,SAI_AClocksFreq_Value,SAI_BClocksFreq_Value,SYSCLKFreq_VALUE,SYSCLKSource,VCOI2SOutputFreq_Value,VCOInputFreq_Value,VCOOutputFreq_Value,VCOSAIOutputFreq_Value,VCOSAIOutputFreq_ValueQ,VCOSAIOutputFreq_ValueR,VcooutputI2S,VcooutputI2SQ
|
RCC.IPParameters=48MHZClocksFreq_Value,AHBFreq_Value,APB1CLKDivider,APB1Freq_Value,APB1TimFreq_Value,APB2CLKDivider,APB2Freq_Value,APB2TimFreq_Value,CortexFreq_Value,EthernetFreq_Value,FCLKCortexFreq_Value,FamilyName,HCLKFreq_Value,HSE_VALUE,HSI_VALUE,I2SClocksFreq_Value,LCDTFTFreq_Value,LSI_VALUE,MCO2PinFreq_Value,PLLCLKFreq_Value,PLLM,PLLN,PLLQ,PLLQCLKFreq_Value,PLLSAIN,PLLSourceVirtual,RCC_RTC_Clock_Source,RCC_RTC_Clock_SourceVirtual,RTCFreq_Value,RTCHSEDivFreq_Value,SAI_AClocksFreq_Value,SAI_BClocksFreq_Value,SYSCLKFreq_VALUE,SYSCLKSource,VCOI2SOutputFreq_Value,VCOInputFreq_Value,VCOOutputFreq_Value,VCOSAIOutputFreq_Value,VCOSAIOutputFreq_ValueQ,VCOSAIOutputFreq_ValueR,VcooutputI2S,VcooutputI2SQ
|
||||||
RCC.LCDTFTFreq_Value=20416666.666666668
|
RCC.LCDTFTFreq_Value=25000000
|
||||||
RCC.LSI_VALUE=32000
|
RCC.LSI_VALUE=32000
|
||||||
RCC.MCO2PinFreq_Value=180000000
|
RCC.MCO2PinFreq_Value=180000000
|
||||||
RCC.PLLCLKFreq_Value=180000000
|
RCC.PLLCLKFreq_Value=180000000
|
||||||
|
@ -322,23 +338,35 @@ RCC.PLLM=15
|
||||||
RCC.PLLN=216
|
RCC.PLLN=216
|
||||||
RCC.PLLQ=8
|
RCC.PLLQ=8
|
||||||
RCC.PLLQCLKFreq_Value=45000000
|
RCC.PLLQCLKFreq_Value=45000000
|
||||||
|
RCC.PLLSAIN=60
|
||||||
RCC.PLLSourceVirtual=RCC_PLLSOURCE_HSE
|
RCC.PLLSourceVirtual=RCC_PLLSOURCE_HSE
|
||||||
RCC.RCC_RTC_Clock_Source=RCC_RTCCLKSOURCE_LSE
|
RCC.RCC_RTC_Clock_Source=RCC_RTCCLKSOURCE_LSE
|
||||||
RCC.RCC_RTC_Clock_SourceVirtual=RCC_RTCCLKSOURCE_LSE
|
RCC.RCC_RTC_Clock_SourceVirtual=RCC_RTCCLKSOURCE_LSE
|
||||||
RCC.RTCFreq_Value=32768
|
RCC.RTCFreq_Value=32768
|
||||||
RCC.RTCHSEDivFreq_Value=12500000
|
RCC.RTCHSEDivFreq_Value=12500000
|
||||||
RCC.SAI_AClocksFreq_Value=20416666.666666668
|
RCC.SAI_AClocksFreq_Value=25000000
|
||||||
RCC.SAI_BClocksFreq_Value=20416666.666666668
|
RCC.SAI_BClocksFreq_Value=25000000
|
||||||
RCC.SYSCLKFreq_VALUE=180000000
|
RCC.SYSCLKFreq_VALUE=180000000
|
||||||
RCC.SYSCLKSource=RCC_SYSCLKSOURCE_PLLCLK
|
RCC.SYSCLKSource=RCC_SYSCLKSOURCE_PLLCLK
|
||||||
RCC.VCOI2SOutputFreq_Value=320000000
|
RCC.VCOI2SOutputFreq_Value=320000000
|
||||||
RCC.VCOInputFreq_Value=1666666.6666666667
|
RCC.VCOInputFreq_Value=1666666.6666666667
|
||||||
RCC.VCOOutputFreq_Value=360000000
|
RCC.VCOOutputFreq_Value=360000000
|
||||||
RCC.VCOSAIOutputFreq_Value=81666666.66666667
|
RCC.VCOSAIOutputFreq_Value=100000000
|
||||||
RCC.VCOSAIOutputFreq_ValueQ=20416666.666666668
|
RCC.VCOSAIOutputFreq_ValueQ=25000000
|
||||||
RCC.VCOSAIOutputFreq_ValueR=40833333.333333336
|
RCC.VCOSAIOutputFreq_ValueR=50000000
|
||||||
RCC.VcooutputI2S=160000000
|
RCC.VcooutputI2S=160000000
|
||||||
RCC.VcooutputI2SQ=160000000
|
RCC.VcooutputI2SQ=160000000
|
||||||
|
SAI1.ErrorAudioFreq-SAI_A_MasterWithClock=-49.13 %
|
||||||
|
SAI1.FrameLength-SAI_B_SyncSlave=8
|
||||||
|
SAI1.IPParameters=Instance-SAI_B_SyncSlave,VirtualMode-SAI_B_SyncSlave,Synchro-SAI_B_SyncSlave,FrameLength-SAI_B_SyncSlave,Instance-SAI_A_MasterWithClock,VirtualMode-SAI_A_MasterWithClock,Synchro-SAI_A_MasterWithClock,MClockEnable-SAI_A_MasterWithClock,RealAudioFreq-SAI_A_MasterWithClock,ErrorAudioFreq-SAI_A_MasterWithClock
|
||||||
|
SAI1.Instance-SAI_A_MasterWithClock=SAI$Index_Block_A
|
||||||
|
SAI1.Instance-SAI_B_SyncSlave=SAI$Index_Block_B
|
||||||
|
SAI1.MClockEnable-SAI_A_MasterWithClock=SAI_MASTERCLOCK_ENABLE
|
||||||
|
SAI1.RealAudioFreq-SAI_A_MasterWithClock=97.656 KHz
|
||||||
|
SAI1.Synchro-SAI_A_MasterWithClock=SAI_ASYNCHRONOUS
|
||||||
|
SAI1.Synchro-SAI_B_SyncSlave=SAI_SYNCHRONOUS
|
||||||
|
SAI1.VirtualMode-SAI_A_MasterWithClock=VM_MASTER
|
||||||
|
SAI1.VirtualMode-SAI_B_SyncSlave=VM_SLAVE
|
||||||
SH.ADCx_IN5.0=ADC1_IN5,IN5
|
SH.ADCx_IN5.0=ADC1_IN5,IN5
|
||||||
SH.ADCx_IN5.ConfNb=1
|
SH.ADCx_IN5.ConfNb=1
|
||||||
SH.FMC_A0.0=FMC_A0,13b-sda1
|
SH.FMC_A0.0=FMC_A0,13b-sda1
|
||||||
|
|
|
@ -72,6 +72,9 @@ IWDG_HandleTypeDef hiwdg;
|
||||||
|
|
||||||
RTC_HandleTypeDef hrtc;
|
RTC_HandleTypeDef hrtc;
|
||||||
|
|
||||||
|
SAI_HandleTypeDef hsai_BlockA1;
|
||||||
|
SAI_HandleTypeDef hsai_BlockB1;
|
||||||
|
|
||||||
SD_HandleTypeDef hsd;
|
SD_HandleTypeDef hsd;
|
||||||
|
|
||||||
SPI_HandleTypeDef hspi1;
|
SPI_HandleTypeDef hspi1;
|
||||||
|
@ -112,6 +115,7 @@ static void MX_SPI1_Init(void);
|
||||||
static void MX_SPI2_Init(void);
|
static void MX_SPI2_Init(void);
|
||||||
static void MX_SPI5_Init(void);
|
static void MX_SPI5_Init(void);
|
||||||
static void MX_CAN1_Init(void);
|
static void MX_CAN1_Init(void);
|
||||||
|
static void MX_SAI1_Init(void);
|
||||||
static void MX_USART2_UART_Init(void);
|
static void MX_USART2_UART_Init(void);
|
||||||
static void MX_USART3_UART_Init(void);
|
static void MX_USART3_UART_Init(void);
|
||||||
/* USER CODE BEGIN PFP */
|
/* USER CODE BEGIN PFP */
|
||||||
|
@ -168,6 +172,7 @@ int main(void)
|
||||||
MX_SPI2_Init();
|
MX_SPI2_Init();
|
||||||
MX_SPI5_Init();
|
MX_SPI5_Init();
|
||||||
MX_CAN1_Init();
|
MX_CAN1_Init();
|
||||||
|
MX_SAI1_Init();
|
||||||
MX_USART2_UART_Init();
|
MX_USART2_UART_Init();
|
||||||
MX_USART3_UART_Init();
|
MX_USART3_UART_Init();
|
||||||
/* USER CODE BEGIN 2 */
|
/* USER CODE BEGIN 2 */
|
||||||
|
@ -197,6 +202,12 @@ void SystemClock_Config(void)
|
||||||
RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
|
RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
|
||||||
RCC_PeriphCLKInitTypeDef PeriphClkInitStruct = {0};
|
RCC_PeriphCLKInitTypeDef PeriphClkInitStruct = {0};
|
||||||
|
|
||||||
|
/** Macro to configure SAI1BlockB clock source selection
|
||||||
|
*/
|
||||||
|
__HAL_RCC_SAI_BLOCKBCLKSOURCE_CONFIG(SAI_CLKSOURCE_PLLSAI);
|
||||||
|
/** Macro to configure SAI1BlockA clock source selection
|
||||||
|
*/
|
||||||
|
__HAL_RCC_SAI_BLOCKACLKSOURCE_CONFIG(SAI_CLKSOURCE_PLLSAI);
|
||||||
/** Configure the main internal regulator output voltage
|
/** Configure the main internal regulator output voltage
|
||||||
*/
|
*/
|
||||||
__HAL_RCC_PWR_CLK_ENABLE();
|
__HAL_RCC_PWR_CLK_ENABLE();
|
||||||
|
@ -237,7 +248,10 @@ void SystemClock_Config(void)
|
||||||
{
|
{
|
||||||
Error_Handler();
|
Error_Handler();
|
||||||
}
|
}
|
||||||
PeriphClkInitStruct.PeriphClockSelection = RCC_PERIPHCLK_RTC;
|
PeriphClkInitStruct.PeriphClockSelection = RCC_PERIPHCLK_SAI_PLLSAI|RCC_PERIPHCLK_RTC;
|
||||||
|
PeriphClkInitStruct.PLLSAI.PLLSAIN = 60;
|
||||||
|
PeriphClkInitStruct.PLLSAI.PLLSAIQ = 4;
|
||||||
|
PeriphClkInitStruct.PLLSAIDivQ = 1;
|
||||||
PeriphClkInitStruct.RTCClockSelection = RCC_RTCCLKSOURCE_LSE;
|
PeriphClkInitStruct.RTCClockSelection = RCC_RTCCLKSOURCE_LSE;
|
||||||
if (HAL_RCCEx_PeriphCLKConfig(&PeriphClkInitStruct) != HAL_OK)
|
if (HAL_RCCEx_PeriphCLKConfig(&PeriphClkInitStruct) != HAL_OK)
|
||||||
{
|
{
|
||||||
|
@ -439,6 +453,72 @@ static void MX_RTC_Init(void)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief SAI1 Initialization Function
|
||||||
|
* @param None
|
||||||
|
* @retval None
|
||||||
|
*/
|
||||||
|
static void MX_SAI1_Init(void)
|
||||||
|
{
|
||||||
|
|
||||||
|
/* USER CODE BEGIN SAI1_Init 0 */
|
||||||
|
|
||||||
|
/* USER CODE END SAI1_Init 0 */
|
||||||
|
|
||||||
|
/* USER CODE BEGIN SAI1_Init 1 */
|
||||||
|
|
||||||
|
/* USER CODE END SAI1_Init 1 */
|
||||||
|
hsai_BlockA1.Instance = SAI1_Block_A;
|
||||||
|
hsai_BlockA1.Init.Protocol = SAI_FREE_PROTOCOL;
|
||||||
|
hsai_BlockA1.Init.AudioMode = SAI_MODEMASTER_TX;
|
||||||
|
hsai_BlockA1.Init.DataSize = SAI_DATASIZE_24;
|
||||||
|
hsai_BlockA1.Init.FirstBit = SAI_FIRSTBIT_MSB;
|
||||||
|
hsai_BlockA1.Init.ClockStrobing = SAI_CLOCKSTROBING_FALLINGEDGE;
|
||||||
|
hsai_BlockA1.Init.OutputDrive = SAI_OUTPUTDRIVE_DISABLE;
|
||||||
|
hsai_BlockA1.Init.NoDivider = SAI_MASTERDIVIDER_ENABLE;
|
||||||
|
hsai_BlockA1.Init.FIFOThreshold = SAI_FIFOTHRESHOLD_EMPTY;
|
||||||
|
hsai_BlockA1.Init.ClockSource = SAI_CLKSOURCE_PLLSAI;
|
||||||
|
hsai_BlockA1.Init.AudioFrequency = SAI_AUDIO_FREQUENCY_192K;
|
||||||
|
hsai_BlockA1.FrameInit.FrameLength = 8;
|
||||||
|
hsai_BlockA1.FrameInit.ActiveFrameLength = 1;
|
||||||
|
hsai_BlockA1.FrameInit.FSDefinition = SAI_FS_STARTFRAME;
|
||||||
|
hsai_BlockA1.FrameInit.FSPolarity = SAI_FS_ACTIVE_LOW;
|
||||||
|
hsai_BlockA1.FrameInit.FSOffset = SAI_FS_FIRSTBIT;
|
||||||
|
hsai_BlockA1.SlotInit.FirstBitOffset = 0;
|
||||||
|
hsai_BlockA1.SlotInit.SlotSize = SAI_SLOTSIZE_DATASIZE;
|
||||||
|
hsai_BlockA1.SlotInit.SlotNumber = 1;
|
||||||
|
hsai_BlockA1.SlotInit.SlotActive = 0x00000000;
|
||||||
|
if (HAL_SAI_Init(&hsai_BlockA1) != HAL_OK)
|
||||||
|
{
|
||||||
|
Error_Handler();
|
||||||
|
}
|
||||||
|
hsai_BlockB1.Instance = SAI1_Block_B;
|
||||||
|
hsai_BlockB1.Init.Protocol = SAI_FREE_PROTOCOL;
|
||||||
|
hsai_BlockB1.Init.AudioMode = SAI_MODESLAVE_RX;
|
||||||
|
hsai_BlockB1.Init.DataSize = SAI_DATASIZE_24;
|
||||||
|
hsai_BlockB1.Init.FirstBit = SAI_FIRSTBIT_MSB;
|
||||||
|
hsai_BlockB1.Init.ClockStrobing = SAI_CLOCKSTROBING_FALLINGEDGE;
|
||||||
|
hsai_BlockB1.Init.OutputDrive = SAI_OUTPUTDRIVE_DISABLE;
|
||||||
|
hsai_BlockB1.Init.FIFOThreshold = SAI_FIFOTHRESHOLD_EMPTY;
|
||||||
|
hsai_BlockB1.FrameInit.FrameLength = 8;
|
||||||
|
hsai_BlockB1.FrameInit.ActiveFrameLength = 1;
|
||||||
|
hsai_BlockB1.FrameInit.FSDefinition = SAI_FS_STARTFRAME;
|
||||||
|
hsai_BlockB1.FrameInit.FSPolarity = SAI_FS_ACTIVE_LOW;
|
||||||
|
hsai_BlockB1.FrameInit.FSOffset = SAI_FS_FIRSTBIT;
|
||||||
|
hsai_BlockB1.SlotInit.FirstBitOffset = 0;
|
||||||
|
hsai_BlockB1.SlotInit.SlotSize = SAI_SLOTSIZE_DATASIZE;
|
||||||
|
hsai_BlockB1.SlotInit.SlotNumber = 1;
|
||||||
|
hsai_BlockB1.SlotInit.SlotActive = 0x00000000;
|
||||||
|
if (HAL_SAI_Init(&hsai_BlockB1) != HAL_OK)
|
||||||
|
{
|
||||||
|
Error_Handler();
|
||||||
|
}
|
||||||
|
/* USER CODE BEGIN SAI1_Init 2 */
|
||||||
|
|
||||||
|
/* USER CODE END SAI1_Init 2 */
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief SDIO Initialization Function
|
* @brief SDIO Initialization Function
|
||||||
* @param None
|
* @param None
|
||||||
|
@ -896,12 +976,12 @@ static void MX_GPIO_Init(void)
|
||||||
{
|
{
|
||||||
|
|
||||||
/* GPIO Ports Clock Enable */
|
/* GPIO Ports Clock Enable */
|
||||||
|
__HAL_RCC_GPIOE_CLK_ENABLE();
|
||||||
__HAL_RCC_GPIOC_CLK_ENABLE();
|
__HAL_RCC_GPIOC_CLK_ENABLE();
|
||||||
__HAL_RCC_GPIOF_CLK_ENABLE();
|
__HAL_RCC_GPIOF_CLK_ENABLE();
|
||||||
__HAL_RCC_GPIOH_CLK_ENABLE();
|
__HAL_RCC_GPIOH_CLK_ENABLE();
|
||||||
__HAL_RCC_GPIOA_CLK_ENABLE();
|
__HAL_RCC_GPIOA_CLK_ENABLE();
|
||||||
__HAL_RCC_GPIOG_CLK_ENABLE();
|
__HAL_RCC_GPIOG_CLK_ENABLE();
|
||||||
__HAL_RCC_GPIOE_CLK_ENABLE();
|
|
||||||
__HAL_RCC_GPIOB_CLK_ENABLE();
|
__HAL_RCC_GPIOB_CLK_ENABLE();
|
||||||
__HAL_RCC_GPIOD_CLK_ENABLE();
|
__HAL_RCC_GPIOD_CLK_ENABLE();
|
||||||
|
|
||||||
|
|
|
@ -1105,6 +1105,97 @@ void HAL_SDRAM_MspDeInit(SDRAM_HandleTypeDef* hsdram){
|
||||||
/* USER CODE END SDRAM_MspDeInit 1 */
|
/* USER CODE END SDRAM_MspDeInit 1 */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static uint32_t SAI1_client =0;
|
||||||
|
|
||||||
|
void HAL_SAI_MspInit(SAI_HandleTypeDef* hsai)
|
||||||
|
{
|
||||||
|
|
||||||
|
GPIO_InitTypeDef GPIO_InitStruct;
|
||||||
|
/* SAI1 */
|
||||||
|
if(hsai->Instance==SAI1_Block_A)
|
||||||
|
{
|
||||||
|
/* Peripheral clock enable */
|
||||||
|
if (SAI1_client == 0)
|
||||||
|
{
|
||||||
|
__HAL_RCC_SAI1_CLK_ENABLE();
|
||||||
|
}
|
||||||
|
SAI1_client ++;
|
||||||
|
|
||||||
|
/**SAI1_A_Block_A GPIO Configuration
|
||||||
|
PE2 ------> SAI1_MCLK_A
|
||||||
|
PE4 ------> SAI1_FS_A
|
||||||
|
PE5 ------> SAI1_SCK_A
|
||||||
|
PE6 ------> SAI1_SD_A
|
||||||
|
*/
|
||||||
|
GPIO_InitStruct.Pin = GPIO_PIN_2|GPIO_PIN_4|GPIO_PIN_5|GPIO_PIN_6;
|
||||||
|
GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
|
||||||
|
GPIO_InitStruct.Pull = GPIO_NOPULL;
|
||||||
|
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
|
||||||
|
GPIO_InitStruct.Alternate = GPIO_AF6_SAI1;
|
||||||
|
HAL_GPIO_Init(GPIOE, &GPIO_InitStruct);
|
||||||
|
|
||||||
|
}
|
||||||
|
if(hsai->Instance==SAI1_Block_B)
|
||||||
|
{
|
||||||
|
/* Peripheral clock enable */
|
||||||
|
if (SAI1_client == 0)
|
||||||
|
{
|
||||||
|
__HAL_RCC_SAI1_CLK_ENABLE();
|
||||||
|
}
|
||||||
|
SAI1_client ++;
|
||||||
|
|
||||||
|
/**SAI1_B_Block_B GPIO Configuration
|
||||||
|
PE3 ------> SAI1_SD_B
|
||||||
|
*/
|
||||||
|
GPIO_InitStruct.Pin = GPIO_PIN_3;
|
||||||
|
GPIO_InitStruct.Mode = GPIO_MODE_AF_PP;
|
||||||
|
GPIO_InitStruct.Pull = GPIO_NOPULL;
|
||||||
|
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
|
||||||
|
GPIO_InitStruct.Alternate = GPIO_AF6_SAI1;
|
||||||
|
HAL_GPIO_Init(GPIOE, &GPIO_InitStruct);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void HAL_SAI_MspDeInit(SAI_HandleTypeDef* hsai)
|
||||||
|
{
|
||||||
|
|
||||||
|
/* SAI1 */
|
||||||
|
if(hsai->Instance==SAI1_Block_A)
|
||||||
|
{
|
||||||
|
SAI1_client --;
|
||||||
|
if (SAI1_client == 0)
|
||||||
|
{
|
||||||
|
/* Peripheral clock disable */
|
||||||
|
__HAL_RCC_SAI1_CLK_DISABLE();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**SAI1_A_Block_A GPIO Configuration
|
||||||
|
PE2 ------> SAI1_MCLK_A
|
||||||
|
PE4 ------> SAI1_FS_A
|
||||||
|
PE5 ------> SAI1_SCK_A
|
||||||
|
PE6 ------> SAI1_SD_A
|
||||||
|
*/
|
||||||
|
HAL_GPIO_DeInit(GPIOE, GPIO_PIN_2|GPIO_PIN_4|GPIO_PIN_5|GPIO_PIN_6);
|
||||||
|
|
||||||
|
}
|
||||||
|
if(hsai->Instance==SAI1_Block_B)
|
||||||
|
{
|
||||||
|
SAI1_client --;
|
||||||
|
if (SAI1_client == 0)
|
||||||
|
{
|
||||||
|
/* Peripheral clock disable */
|
||||||
|
__HAL_RCC_SAI1_CLK_DISABLE();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**SAI1_B_Block_B GPIO Configuration
|
||||||
|
PE3 ------> SAI1_SD_B
|
||||||
|
*/
|
||||||
|
HAL_GPIO_DeInit(GPIOE, GPIO_PIN_3);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* USER CODE BEGIN 1 */
|
/* USER CODE BEGIN 1 */
|
||||||
|
|
||||||
/* USER CODE END 1 */
|
/* USER CODE END 1 */
|
||||||
|
|
|
@ -60,7 +60,24 @@ menu "Onboard Peripheral Drivers"
|
||||||
select RT_USING_DFS
|
select RT_USING_DFS
|
||||||
select RT_USING_DFS_ELMFAT
|
select RT_USING_DFS_ELMFAT
|
||||||
default n
|
default n
|
||||||
|
|
||||||
|
config BSP_USING_AUDIO
|
||||||
|
bool "Enable AUDIO (WM8978)"
|
||||||
|
select BSP_USING_I2C1
|
||||||
|
select RT_USING_AUDIO
|
||||||
|
default n
|
||||||
|
|
||||||
|
if BSP_USING_AUDIO
|
||||||
|
config BSP_USING_AUDIO_PLAY
|
||||||
|
bool "Enable Audio Play"
|
||||||
|
default y
|
||||||
|
|
||||||
|
config BSP_USING_AUDIO_RECORD
|
||||||
|
bool "Enable Audio Record"
|
||||||
|
select BSP_USING_AUDIO_PLAY
|
||||||
|
default n
|
||||||
|
endif
|
||||||
|
|
||||||
endmenu
|
endmenu
|
||||||
|
|
||||||
menu "On-chip Peripheral Drivers"
|
menu "On-chip Peripheral Drivers"
|
||||||
|
|
|
@ -20,11 +20,21 @@ if GetDepend(['BSP_USING_SPI_FLASH']):
|
||||||
|
|
||||||
if GetDepend(['BSP_USING_SDCARD']):
|
if GetDepend(['BSP_USING_SDCARD']):
|
||||||
src += Glob('ports/sdcard_port.c')
|
src += Glob('ports/sdcard_port.c')
|
||||||
|
|
||||||
|
if GetDepend(['BSP_USING_AUDIO']):
|
||||||
|
src += Glob('ports/audio/drv_wm8978.c')
|
||||||
|
src += Glob('ports/audio/drv_sound.c')
|
||||||
|
|
||||||
|
if GetDepend(['BSP_USING_AUDIO_RECORD']):
|
||||||
|
src += Glob('ports/audio/drv_mic.c')
|
||||||
|
|
||||||
path = [cwd]
|
path = [cwd]
|
||||||
path += [cwd + '/CubeMX_Config/Inc']
|
path += [cwd + '/CubeMX_Config/Inc']
|
||||||
path += [cwd + '/ports']
|
path += [cwd + '/ports']
|
||||||
|
|
||||||
|
if GetDepend(['BSP_USING_AUDIO']):
|
||||||
|
path += [cwd + '/ports/audio']
|
||||||
|
|
||||||
startup_path_prefix = SDK_LIB
|
startup_path_prefix = SDK_LIB
|
||||||
|
|
||||||
if rtconfig.CROSS_TOOL == 'gcc':
|
if rtconfig.CROSS_TOOL == 'gcc':
|
||||||
|
|
|
@ -0,0 +1,385 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2006-2018, RT-Thread Development Team
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
|
*
|
||||||
|
* Change Logs:
|
||||||
|
* Date Author Notes
|
||||||
|
* 2019-07-28 Ernest the first version
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "board.h"
|
||||||
|
#include "drv_mic.h"
|
||||||
|
#include "drv_wm8978.h"
|
||||||
|
#include "drv_sound.h"
|
||||||
|
|
||||||
|
#define DBG_TAG "drv.mic"
|
||||||
|
#define DBG_LVL DBG_INFO
|
||||||
|
#include <rtdbg.h>
|
||||||
|
|
||||||
|
#define CODEC_I2C_NAME ("i2c1")
|
||||||
|
#define RX_DMA_FIFO_SIZE (2048)
|
||||||
|
|
||||||
|
extern struct drv_sai _sai_a;
|
||||||
|
static struct drv_sai _sai_b = {0};
|
||||||
|
|
||||||
|
struct stm32_mic
|
||||||
|
{
|
||||||
|
struct rt_i2c_bus_device *i2c_bus;
|
||||||
|
struct rt_audio_device audio;
|
||||||
|
struct rt_audio_configure config;
|
||||||
|
rt_uint8_t *rx_fifo;
|
||||||
|
rt_bool_t startup;
|
||||||
|
};
|
||||||
|
static struct stm32_mic _stm32_audio_record = {0};
|
||||||
|
|
||||||
|
static rt_err_t SAIB_samplerate_set(rt_uint32_t freq)
|
||||||
|
{
|
||||||
|
__HAL_SAI_DISABLE(&_sai_b.hsai);
|
||||||
|
_sai_b.hsai.Init.AudioFrequency = freq;
|
||||||
|
HAL_SAI_Init(&_sai_b.hsai);
|
||||||
|
__HAL_SAI_ENABLE(&_sai_b.hsai);
|
||||||
|
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SAIB_channels_set(rt_uint16_t channels)
|
||||||
|
{
|
||||||
|
if (channels == 2)
|
||||||
|
{
|
||||||
|
_sai_b.hsai.Init.MonoStereoMode = SAI_STEREOMODE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_sai_b.hsai.Init.MonoStereoMode = SAI_MONOMODE;
|
||||||
|
}
|
||||||
|
__HAL_SAI_DISABLE(&_sai_b.hsai);
|
||||||
|
HAL_SAI_Init(&_sai_b.hsai);
|
||||||
|
__HAL_SAI_ENABLE(&_sai_b.hsai);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SAIB_samplebits_set(rt_uint16_t samplebits)
|
||||||
|
{
|
||||||
|
switch (samplebits)
|
||||||
|
{
|
||||||
|
case 16:
|
||||||
|
_sai_b.hsai.Init.DataSize = SAI_DATASIZE_16;
|
||||||
|
break;
|
||||||
|
case 24:
|
||||||
|
_sai_b.hsai.Init.DataSize = SAI_DATASIZE_24;
|
||||||
|
break;
|
||||||
|
case 32:
|
||||||
|
_sai_b.hsai.Init.DataSize = SAI_DATASIZE_32;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
_sai_b.hsai.Init.DataSize = SAI_DATASIZE_16;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
__HAL_SAI_DISABLE(&_sai_b.hsai);
|
||||||
|
HAL_SAI_Init(&_sai_b.hsai);
|
||||||
|
__HAL_SAI_ENABLE(&_sai_b.hsai);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SAIB_config_set(struct rt_audio_configure config)
|
||||||
|
{
|
||||||
|
SAIB_channels_set(config.channels);
|
||||||
|
SAIB_samplerate_set(config.samplerate);
|
||||||
|
SAIB_samplebits_set(config.samplebits);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void SAIB_config_init()
|
||||||
|
{
|
||||||
|
_sai_b.hsai.Instance = SAI1_Block_B;
|
||||||
|
_sai_b.hsai.Init.AudioMode = SAI_MODESLAVE_RX;
|
||||||
|
_sai_b.hsai.Init.Synchro = SAI_SYNCHRONOUS;
|
||||||
|
_sai_b.hsai.Init.OutputDrive = SAI_OUTPUTDRIVE_ENABLE;
|
||||||
|
_sai_b.hsai.Init.NoDivider = SAI_MASTERDIVIDER_ENABLE;
|
||||||
|
_sai_b.hsai.Init.FIFOThreshold = SAI_FIFOTHRESHOLD_1QF;
|
||||||
|
_sai_b.hsai.Init.ClockSource = SAI_CLKSOURCE_PLLI2S;
|
||||||
|
_sai_b.hsai.Init.MonoStereoMode = SAI_STEREOMODE;
|
||||||
|
_sai_b.hsai.Init.Protocol = SAI_FREE_PROTOCOL;
|
||||||
|
_sai_b.hsai.Init.DataSize = SAI_DATASIZE_16;
|
||||||
|
_sai_b.hsai.Init.FirstBit = SAI_FIRSTBIT_MSB;
|
||||||
|
_sai_b.hsai.Init.ClockStrobing = SAI_CLOCKSTROBING_RISINGEDGE;
|
||||||
|
|
||||||
|
/* frame */
|
||||||
|
_sai_b.hsai.FrameInit.FrameLength = 64;
|
||||||
|
_sai_b.hsai.FrameInit.ActiveFrameLength = 32;
|
||||||
|
_sai_b.hsai.FrameInit.FSDefinition = SAI_FS_CHANNEL_IDENTIFICATION;
|
||||||
|
_sai_b.hsai.FrameInit.FSPolarity = SAI_FS_ACTIVE_LOW;
|
||||||
|
_sai_b.hsai.FrameInit.FSOffset = SAI_FS_BEFOREFIRSTBIT;
|
||||||
|
|
||||||
|
/* slot */
|
||||||
|
_sai_b.hsai.SlotInit.FirstBitOffset = 0;
|
||||||
|
_sai_b.hsai.SlotInit.SlotSize = SAI_SLOTSIZE_32B;
|
||||||
|
_sai_b.hsai.SlotInit.SlotNumber = 2;
|
||||||
|
_sai_b.hsai.SlotInit.SlotActive = SAI_SLOTACTIVE_0 | SAI_SLOTACTIVE_1;
|
||||||
|
|
||||||
|
HAL_SAI_Init(&_sai_b.hsai);
|
||||||
|
__HAL_SAI_ENABLE(&_sai_b.hsai);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void SAIB_tx_dma(void)
|
||||||
|
{
|
||||||
|
__HAL_RCC_DMA2_CLK_ENABLE();
|
||||||
|
__HAL_LINKDMA(&_sai_b.hsai, hdmarx, _sai_b.hdma);
|
||||||
|
_sai_b.hdma.Instance = DMA2_Stream5;
|
||||||
|
_sai_b.hdma.Init.Channel = DMA_CHANNEL_0;
|
||||||
|
_sai_b.hdma.Init.Direction = DMA_PERIPH_TO_MEMORY;
|
||||||
|
_sai_b.hdma.Init.PeriphInc = DMA_PINC_DISABLE;
|
||||||
|
_sai_b.hdma.Init.MemInc = DMA_MINC_ENABLE;
|
||||||
|
_sai_b.hdma.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;
|
||||||
|
_sai_b.hdma.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
|
||||||
|
_sai_b.hdma.Init.Mode = DMA_CIRCULAR;
|
||||||
|
_sai_b.hdma.Init.Priority = DMA_PRIORITY_MEDIUM;
|
||||||
|
_sai_b.hdma.Init.FIFOMode = DMA_FIFOMODE_DISABLE;
|
||||||
|
_sai_b.hdma.Init.MemBurst = DMA_MBURST_SINGLE;
|
||||||
|
_sai_b.hdma.Init.PeriphBurst = DMA_PBURST_SINGLE;
|
||||||
|
HAL_DMA_DeInit(&_sai_b.hdma);
|
||||||
|
HAL_DMA_Init(&_sai_b.hdma);
|
||||||
|
|
||||||
|
__HAL_DMA_DISABLE(&_sai_b.hdma);
|
||||||
|
|
||||||
|
__HAL_DMA_CLEAR_FLAG(&_sai_b.hdma, DMA_FLAG_TCIF1_5);
|
||||||
|
__HAL_DMA_ENABLE_IT(&_sai_b.hdma, DMA_IT_TC);
|
||||||
|
|
||||||
|
HAL_NVIC_SetPriority(DMA2_Stream5_IRQn, 5, 1);
|
||||||
|
HAL_NVIC_EnableIRQ(DMA2_Stream5_IRQn);
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t sai_record_init()
|
||||||
|
{
|
||||||
|
SAIA_config_init();
|
||||||
|
SAIB_config_init();
|
||||||
|
|
||||||
|
/* set record samplerate */
|
||||||
|
SAIA_config_set(_stm32_audio_record.config);
|
||||||
|
SAIB_config_set(_stm32_audio_record.config);
|
||||||
|
SAIA_tx_dma();
|
||||||
|
SAIB_tx_dma();
|
||||||
|
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
void DMA2_Stream5_IRQHandler(void)
|
||||||
|
{
|
||||||
|
rt_interrupt_enter();
|
||||||
|
HAL_DMA_IRQHandler(_sai_b.hsai.hdmarx);
|
||||||
|
rt_interrupt_leave();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HAL_SAI_RxCpltCallback(SAI_HandleTypeDef *hsai)
|
||||||
|
{
|
||||||
|
rt_audio_rx_done(&(_stm32_audio_record.audio), &_stm32_audio_record.rx_fifo[0], RX_DMA_FIFO_SIZE / 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HAL_SAI_RxHalfCpltCallback(SAI_HandleTypeDef *hsai)
|
||||||
|
{
|
||||||
|
|
||||||
|
rt_audio_rx_done(&(_stm32_audio_record.audio), &_stm32_audio_record.rx_fifo[RX_DMA_FIFO_SIZE / 2], RX_DMA_FIFO_SIZE / 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t stm32_mic_getcaps(struct rt_audio_device *audio, struct rt_audio_caps *caps)
|
||||||
|
{
|
||||||
|
rt_err_t result = RT_EOK;
|
||||||
|
|
||||||
|
LOG_D("%s:main_type: %d, sub_type: %d", __FUNCTION__, caps->main_type, caps->sub_type);
|
||||||
|
|
||||||
|
switch (caps->main_type)
|
||||||
|
{
|
||||||
|
/* Provide capabilities of INTPUT unit */
|
||||||
|
case AUDIO_TYPE_INPUT:
|
||||||
|
{
|
||||||
|
switch (caps->sub_type)
|
||||||
|
{
|
||||||
|
case AUDIO_DSP_PARAM:
|
||||||
|
caps->udata.config.channels = _stm32_audio_record.config.channels;
|
||||||
|
caps->udata.config.samplebits = _stm32_audio_record.config.samplebits;
|
||||||
|
caps->udata.config.samplerate = _stm32_audio_record.config.samplerate;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AUDIO_DSP_SAMPLERATE:
|
||||||
|
caps->udata.config.samplerate = _stm32_audio_record.config.samplerate;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AUDIO_DSP_CHANNELS:
|
||||||
|
caps->udata.config.channels = _stm32_audio_record.config.channels;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AUDIO_DSP_SAMPLEBITS:
|
||||||
|
caps->udata.config.samplebits = _stm32_audio_record.config.samplebits;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
result = -RT_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
result = -RT_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void start_record_mode(void)
|
||||||
|
{
|
||||||
|
rt_uint8_t temp[4] = {0};
|
||||||
|
|
||||||
|
HAL_SAI_DMAStop(&_sai_b.hsai);
|
||||||
|
HAL_SAI_Transmit(&_sai_a.hsai, temp, 4, 0);
|
||||||
|
HAL_SAI_Receive_DMA(&_sai_b.hsai, _stm32_audio_record.rx_fifo, RX_DMA_FIFO_SIZE / 2);
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t stm32_mic_configure(struct rt_audio_device *audio, struct rt_audio_caps *caps)
|
||||||
|
{
|
||||||
|
rt_err_t result = RT_EOK;
|
||||||
|
|
||||||
|
LOG_D("%s:main_type: %d, sub_type: %d", __FUNCTION__, caps->main_type, caps->sub_type);
|
||||||
|
|
||||||
|
switch (caps->main_type)
|
||||||
|
{
|
||||||
|
case AUDIO_TYPE_INPUT:
|
||||||
|
{
|
||||||
|
switch (caps->sub_type)
|
||||||
|
{
|
||||||
|
|
||||||
|
case AUDIO_DSP_PARAM:
|
||||||
|
{
|
||||||
|
_stm32_audio_record.config.samplerate = caps->udata.config.samplerate;
|
||||||
|
_stm32_audio_record.config.channels = caps->udata.config.channels;
|
||||||
|
_stm32_audio_record.config.samplebits = caps->udata.config.samplebits;
|
||||||
|
|
||||||
|
HAL_SAI_DMAStop(&_sai_b.hsai);
|
||||||
|
SAIA_config_set(caps->udata.config);
|
||||||
|
SAIB_config_set(caps->udata.config);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AUDIO_DSP_SAMPLERATE:
|
||||||
|
{
|
||||||
|
_stm32_audio_record.config.samplerate = caps->udata.config.samplerate;
|
||||||
|
SAIA_samplerate_set(caps->udata.config.samplerate);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case AUDIO_DSP_CHANNELS:
|
||||||
|
{
|
||||||
|
_stm32_audio_record.config.channels = caps->udata.config.channels;
|
||||||
|
SAIA_channels_set(caps->udata.config.channels);
|
||||||
|
SAIB_channels_set(caps->udata.config.channels);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AUDIO_DSP_SAMPLEBITS:
|
||||||
|
{
|
||||||
|
_stm32_audio_record.config.samplebits = caps->udata.config.samplebits;
|
||||||
|
SAIA_samplebits_set(caps->udata.config.samplebits);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
result = -RT_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
/* After set config, MCLK will stop */
|
||||||
|
start_record_mode();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t stm32_mic_init(struct rt_audio_device *audio)
|
||||||
|
{
|
||||||
|
rt_err_t result = RT_EOK;
|
||||||
|
|
||||||
|
/* initialize wm8978 */
|
||||||
|
_stm32_audio_record.i2c_bus = (struct rt_i2c_bus_device *)rt_device_find(CODEC_I2C_NAME);
|
||||||
|
if (_stm32_audio_record.i2c_bus != RT_NULL)
|
||||||
|
{
|
||||||
|
LOG_D("Find device i2c1 success");
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
LOG_E("Find device i2c1 error");
|
||||||
|
return -RT_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
result = wm8978_init(_stm32_audio_record.i2c_bus);
|
||||||
|
if (result != RT_EOK)
|
||||||
|
{
|
||||||
|
LOG_E("initialize wm8978 failed");
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
sai_record_init();
|
||||||
|
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t stm32_mic_start(struct rt_audio_device *audio, int stream)
|
||||||
|
{
|
||||||
|
rt_err_t result = RT_EOK;
|
||||||
|
|
||||||
|
if (stream == AUDIO_STREAM_RECORD)
|
||||||
|
{
|
||||||
|
/* set mic start */
|
||||||
|
wm8978_record_start(_stm32_audio_record.i2c_bus);
|
||||||
|
/* start transfer data */
|
||||||
|
start_record_mode();
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t stm32_mic_stop(struct rt_audio_device *audio, int stream)
|
||||||
|
{
|
||||||
|
if (stream == AUDIO_STREAM_RECORD)
|
||||||
|
{
|
||||||
|
HAL_SAI_DMAStop(&_sai_b.hsai);
|
||||||
|
HAL_SAI_DMAStop(&_sai_a.hsai);
|
||||||
|
wm8978_mic_enabled(_stm32_audio_record.i2c_bus, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct rt_audio_ops _mic_audio_ops =
|
||||||
|
{
|
||||||
|
.getcaps = stm32_mic_getcaps,
|
||||||
|
.configure = stm32_mic_configure,
|
||||||
|
.init = stm32_mic_init,
|
||||||
|
.start = stm32_mic_start,
|
||||||
|
.stop = stm32_mic_stop,
|
||||||
|
.transmit = RT_NULL,
|
||||||
|
.buffer_info = RT_NULL,
|
||||||
|
};
|
||||||
|
|
||||||
|
int rt_hw_mic_init(void)
|
||||||
|
{
|
||||||
|
struct rt_audio_device *audio = &_stm32_audio_record.audio;
|
||||||
|
/* mic default */
|
||||||
|
_stm32_audio_record.rx_fifo = rt_calloc(1, RX_DMA_FIFO_SIZE);
|
||||||
|
if (_stm32_audio_record.rx_fifo == RT_NULL)
|
||||||
|
{
|
||||||
|
return -RT_ENOMEM;
|
||||||
|
}
|
||||||
|
|
||||||
|
_stm32_audio_record.config.channels = 1;
|
||||||
|
_stm32_audio_record.config.samplerate = 16000;
|
||||||
|
_stm32_audio_record.config.samplebits = 16;
|
||||||
|
|
||||||
|
/* register mic device */
|
||||||
|
audio->ops = &_mic_audio_ops;
|
||||||
|
rt_audio_register(audio, "mic0", RT_DEVICE_FLAG_RDONLY, &_stm32_audio_record);
|
||||||
|
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
INIT_DEVICE_EXPORT(rt_hw_mic_init);
|
|
@ -0,0 +1,17 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2006-2018, RT-Thread Development Team
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
|
*
|
||||||
|
* Change Logs:
|
||||||
|
* Date Author Notes
|
||||||
|
* 2019-07-28 Ernest the first version
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __DRV_MIC_H_
|
||||||
|
#define __DRV_MIC_H_
|
||||||
|
|
||||||
|
#include <rtthread.h>
|
||||||
|
#include <rtdevice.h>
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,470 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2006-2018, RT-Thread Development Team
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
|
*
|
||||||
|
* Change Logs:
|
||||||
|
* Date Author Notes
|
||||||
|
* 2019-07-28 Ernest the first version
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include "board.h"
|
||||||
|
#include "drv_wm8978.h"
|
||||||
|
#include "drv_sound.h"
|
||||||
|
|
||||||
|
#define DBG_TAG "drv.sound"
|
||||||
|
#define DBG_LVL DBG_INFO
|
||||||
|
#include <rtdbg.h>
|
||||||
|
|
||||||
|
#define CODEC_I2C_NAME ("i2c1")
|
||||||
|
|
||||||
|
#define TX_DMA_FIFO_SIZE (2048)
|
||||||
|
|
||||||
|
struct drv_sai _sai_a = {0};
|
||||||
|
|
||||||
|
struct stm32_audio
|
||||||
|
{
|
||||||
|
struct rt_i2c_bus_device *i2c_bus;
|
||||||
|
struct rt_audio_device audio;
|
||||||
|
struct rt_audio_configure replay_config;
|
||||||
|
int replay_volume;
|
||||||
|
rt_uint8_t *tx_fifo;
|
||||||
|
rt_bool_t startup;
|
||||||
|
};
|
||||||
|
struct stm32_audio _stm32_audio_play = {0};
|
||||||
|
|
||||||
|
/* sample_rate, PLLI2SN(50.7), PLLI2SQ, PLLI2SDivQ, MCKDIV */
|
||||||
|
const rt_uint32_t SAI_PSC_TBL[][5] =
|
||||||
|
{
|
||||||
|
{AUDIO_FREQUENCY_048K, 206, 7, 0, 12},
|
||||||
|
{AUDIO_FREQUENCY_044K, 257, 2, 18, 2},
|
||||||
|
{AUDIO_FREQUENCY_032K, 206, 7, 0, 6},
|
||||||
|
{AUDIO_FREQUENCY_022K, 257, 2, 18, 1},
|
||||||
|
{AUDIO_FREQUENCY_016K, 206, 7, 0, 3},
|
||||||
|
{AUDIO_FREQUENCY_011K, 257, 2, 18, 0},
|
||||||
|
{AUDIO_FREQUENCY_008K, 206, 7, 0, 2},
|
||||||
|
};
|
||||||
|
|
||||||
|
void SAIA_samplerate_set(rt_uint32_t freq)
|
||||||
|
{
|
||||||
|
RCC_PeriphCLKInitTypeDef PeriphClkInitStruct;
|
||||||
|
int i;
|
||||||
|
|
||||||
|
/* check frequence */
|
||||||
|
for (i = 0; i < (sizeof(SAI_PSC_TBL) / sizeof(SAI_PSC_TBL[0])); i++)
|
||||||
|
{
|
||||||
|
if ((freq) == SAI_PSC_TBL[i][0])break;
|
||||||
|
}
|
||||||
|
if (i == (sizeof(SAI_PSC_TBL) / sizeof(SAI_PSC_TBL[0])))
|
||||||
|
{
|
||||||
|
LOG_E("Can not support this frequence: %d.", freq);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
PeriphClkInitStruct.PeriphClockSelection = RCC_PERIPHCLK_SAI_PLLI2S;
|
||||||
|
PeriphClkInitStruct.PLLI2S.PLLI2SN = SAI_PSC_TBL[i][1];
|
||||||
|
PeriphClkInitStruct.PLLI2S.PLLI2SQ = SAI_PSC_TBL[i][2];
|
||||||
|
PeriphClkInitStruct.PLLI2SDivQ = SAI_PSC_TBL[i][3] + 1;
|
||||||
|
|
||||||
|
HAL_RCCEx_PeriphCLKConfig(&PeriphClkInitStruct);
|
||||||
|
|
||||||
|
__HAL_RCC_SAI_BLOCKACLKSOURCE_CONFIG(RCC_SAIACLKSOURCE_PLLI2S);
|
||||||
|
|
||||||
|
__HAL_SAI_DISABLE(&_sai_a.hsai);
|
||||||
|
_sai_a.hsai.Init.AudioFrequency = freq;
|
||||||
|
HAL_SAI_Init(&_sai_a.hsai);
|
||||||
|
__HAL_SAI_ENABLE(&_sai_a.hsai);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SAIA_channels_set(rt_uint16_t channels)
|
||||||
|
{
|
||||||
|
if (channels == 2)
|
||||||
|
{
|
||||||
|
_sai_a.hsai.Init.MonoStereoMode = SAI_STEREOMODE;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_sai_a.hsai.Init.MonoStereoMode = SAI_MONOMODE;
|
||||||
|
}
|
||||||
|
__HAL_SAI_DISABLE(&_sai_a.hsai);
|
||||||
|
HAL_SAI_Init(&_sai_a.hsai);
|
||||||
|
__HAL_SAI_ENABLE(&_sai_a.hsai);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SAIA_samplebits_set(rt_uint16_t samplebits)
|
||||||
|
{
|
||||||
|
switch (samplebits)
|
||||||
|
{
|
||||||
|
case 16:
|
||||||
|
_sai_a.hsai.Init.DataSize = SAI_DATASIZE_16;
|
||||||
|
break;
|
||||||
|
case 24:
|
||||||
|
_sai_a.hsai.Init.DataSize = SAI_DATASIZE_24;
|
||||||
|
break;
|
||||||
|
case 32:
|
||||||
|
_sai_a.hsai.Init.DataSize = SAI_DATASIZE_32;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
_sai_a.hsai.Init.DataSize = SAI_DATASIZE_16;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
__HAL_SAI_DISABLE(&_sai_a.hsai);
|
||||||
|
HAL_SAI_Init(&_sai_a.hsai);
|
||||||
|
__HAL_SAI_ENABLE(&_sai_a.hsai);
|
||||||
|
}
|
||||||
|
|
||||||
|
void SAIA_config_set(struct rt_audio_configure config)
|
||||||
|
{
|
||||||
|
SAIA_channels_set(config.channels);
|
||||||
|
SAIA_samplerate_set(config.samplerate);
|
||||||
|
SAIA_samplebits_set(config.samplebits);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* initial sai A */
|
||||||
|
rt_err_t SAIA_config_init(void)
|
||||||
|
{
|
||||||
|
_sai_a.hsai.Instance = SAI1_Block_A;
|
||||||
|
_sai_a.hsai.Init.AudioMode = SAI_MODEMASTER_TX;
|
||||||
|
_sai_a.hsai.Init.Synchro = SAI_ASYNCHRONOUS;
|
||||||
|
_sai_a.hsai.Init.OutputDrive = SAI_OUTPUTDRIVE_ENABLE;
|
||||||
|
_sai_a.hsai.Init.NoDivider = SAI_MASTERDIVIDER_ENABLE;
|
||||||
|
_sai_a.hsai.Init.FIFOThreshold = SAI_FIFOTHRESHOLD_1QF;
|
||||||
|
_sai_a.hsai.Init.ClockSource = SAI_CLKSOURCE_PLLI2S;
|
||||||
|
|
||||||
|
_sai_a.hsai.Init.Protocol = SAI_FREE_PROTOCOL;
|
||||||
|
_sai_a.hsai.Init.DataSize = SAI_DATASIZE_16;
|
||||||
|
_sai_a.hsai.Init.FirstBit = SAI_FIRSTBIT_MSB;
|
||||||
|
_sai_a.hsai.Init.ClockStrobing = SAI_CLOCKSTROBING_RISINGEDGE;
|
||||||
|
|
||||||
|
//frame
|
||||||
|
_sai_a.hsai.FrameInit.FrameLength = 64;
|
||||||
|
_sai_a.hsai.FrameInit.ActiveFrameLength = 32;
|
||||||
|
_sai_a.hsai.FrameInit.FSDefinition = SAI_FS_CHANNEL_IDENTIFICATION;
|
||||||
|
_sai_a.hsai.FrameInit.FSPolarity = SAI_FS_ACTIVE_LOW;
|
||||||
|
_sai_a.hsai.FrameInit.FSOffset = SAI_FS_BEFOREFIRSTBIT;
|
||||||
|
|
||||||
|
//slot
|
||||||
|
_sai_a.hsai.SlotInit.FirstBitOffset = 0;
|
||||||
|
_sai_a.hsai.SlotInit.SlotSize = SAI_SLOTSIZE_32B;
|
||||||
|
_sai_a.hsai.SlotInit.SlotNumber = 2;
|
||||||
|
_sai_a.hsai.SlotInit.SlotActive = SAI_SLOTACTIVE_0 | SAI_SLOTACTIVE_1;
|
||||||
|
|
||||||
|
HAL_SAI_Init(&_sai_a.hsai);
|
||||||
|
__HAL_SAI_ENABLE(&_sai_a.hsai);
|
||||||
|
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
rt_err_t SAIA_tx_dma(void)
|
||||||
|
{
|
||||||
|
__HAL_RCC_DMA2_CLK_ENABLE();
|
||||||
|
__HAL_LINKDMA(&_sai_a.hsai, hdmatx, _sai_a.hdma);
|
||||||
|
|
||||||
|
_sai_a.hdma.Instance = DMA2_Stream3;
|
||||||
|
_sai_a.hdma.Init.Channel = DMA_CHANNEL_0;
|
||||||
|
_sai_a.hdma.Init.Direction = DMA_MEMORY_TO_PERIPH;
|
||||||
|
_sai_a.hdma.Init.PeriphInc = DMA_PINC_DISABLE;
|
||||||
|
_sai_a.hdma.Init.MemInc = DMA_MINC_ENABLE;
|
||||||
|
|
||||||
|
_sai_a.hdma.Init.PeriphDataAlignment = DMA_PDATAALIGN_HALFWORD;
|
||||||
|
_sai_a.hdma.Init.MemDataAlignment = DMA_MDATAALIGN_HALFWORD;
|
||||||
|
|
||||||
|
_sai_a.hdma.Init.Mode = DMA_CIRCULAR;
|
||||||
|
_sai_a.hdma.Init.Priority = DMA_PRIORITY_HIGH;
|
||||||
|
_sai_a.hdma.Init.FIFOMode = DMA_FIFOMODE_DISABLE;
|
||||||
|
_sai_a.hdma.Init.FIFOThreshold = DMA_FIFO_THRESHOLD_FULL;
|
||||||
|
_sai_a.hdma.Init.MemBurst = DMA_MBURST_SINGLE;
|
||||||
|
_sai_a.hdma.Init.PeriphBurst = DMA_PBURST_SINGLE;
|
||||||
|
|
||||||
|
HAL_DMA_DeInit(&_sai_a.hdma);
|
||||||
|
HAL_DMA_Init(&_sai_a.hdma);
|
||||||
|
|
||||||
|
__HAL_DMA_DISABLE(&_sai_a.hdma);
|
||||||
|
|
||||||
|
__HAL_DMA_ENABLE_IT(&_sai_a.hdma, DMA_IT_TC);
|
||||||
|
__HAL_DMA_CLEAR_FLAG(&_sai_a.hdma, DMA_FLAG_TCIF3_7);
|
||||||
|
/* set nvic */
|
||||||
|
HAL_NVIC_SetPriority(DMA2_Stream3_IRQn, 5, 0);
|
||||||
|
HAL_NVIC_EnableIRQ(DMA2_Stream3_IRQn);
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
void DMA2_Stream3_IRQHandler(void)
|
||||||
|
{
|
||||||
|
rt_interrupt_enter();
|
||||||
|
HAL_DMA_IRQHandler(_sai_a.hsai.hdmatx);
|
||||||
|
rt_interrupt_leave();
|
||||||
|
}
|
||||||
|
|
||||||
|
void HAL_SAI_TxHalfCpltCallback(SAI_HandleTypeDef *hsai)
|
||||||
|
{
|
||||||
|
rt_audio_tx_complete(&_stm32_audio_play.audio);
|
||||||
|
}
|
||||||
|
|
||||||
|
void HAL_SAI_TxCpltCallback(SAI_HandleTypeDef *hsai)
|
||||||
|
{
|
||||||
|
rt_audio_tx_complete(&_stm32_audio_play.audio);
|
||||||
|
}
|
||||||
|
|
||||||
|
rt_err_t sai_a_init()
|
||||||
|
{
|
||||||
|
/* set sai_a DMA */
|
||||||
|
SAIA_tx_dma();
|
||||||
|
SAIA_config_init();
|
||||||
|
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t stm32_player_getcaps(struct rt_audio_device *audio, struct rt_audio_caps *caps)
|
||||||
|
{
|
||||||
|
rt_err_t result = RT_EOK;
|
||||||
|
struct stm32_audio *st_audio = (struct stm32_audio *)audio->parent.user_data;
|
||||||
|
|
||||||
|
LOG_D("%s:main_type: %d, sub_type: %d", __FUNCTION__, caps->main_type, caps->sub_type);
|
||||||
|
|
||||||
|
switch (caps->main_type)
|
||||||
|
{
|
||||||
|
case AUDIO_TYPE_QUERY: /* query the types of hw_codec device */
|
||||||
|
{
|
||||||
|
switch (caps->sub_type)
|
||||||
|
{
|
||||||
|
case AUDIO_TYPE_QUERY:
|
||||||
|
caps->udata.mask = AUDIO_TYPE_OUTPUT | AUDIO_TYPE_MIXER;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
result = -RT_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AUDIO_TYPE_OUTPUT: /* Provide capabilities of OUTPUT unit */
|
||||||
|
{
|
||||||
|
switch (caps->sub_type)
|
||||||
|
{
|
||||||
|
case AUDIO_DSP_PARAM:
|
||||||
|
caps->udata.config.channels = st_audio->replay_config.channels;
|
||||||
|
caps->udata.config.samplebits = st_audio->replay_config.samplebits;
|
||||||
|
caps->udata.config.samplerate = st_audio->replay_config.samplerate;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AUDIO_DSP_SAMPLERATE:
|
||||||
|
caps->udata.config.samplerate = st_audio->replay_config.samplerate;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AUDIO_DSP_CHANNELS:
|
||||||
|
caps->udata.config.channels = st_audio->replay_config.channels;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AUDIO_DSP_SAMPLEBITS:
|
||||||
|
caps->udata.config.samplebits = st_audio->replay_config.samplebits;
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
result = -RT_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AUDIO_TYPE_MIXER: /* report the Mixer Units */
|
||||||
|
{
|
||||||
|
switch (caps->sub_type)
|
||||||
|
{
|
||||||
|
case AUDIO_MIXER_QUERY:
|
||||||
|
caps->udata.mask = AUDIO_MIXER_VOLUME | AUDIO_MIXER_LINE;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AUDIO_MIXER_VOLUME:
|
||||||
|
caps->udata.value = st_audio->replay_volume;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case AUDIO_MIXER_LINE:
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
result = -RT_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
result = -RT_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t stm32_player_configure(struct rt_audio_device *audio, struct rt_audio_caps *caps)
|
||||||
|
{
|
||||||
|
rt_err_t result = RT_EOK;
|
||||||
|
struct stm32_audio *st_audio = (struct stm32_audio *)audio->parent.user_data;
|
||||||
|
|
||||||
|
LOG_D("%s:main_type: %d, sub_type: %d", __FUNCTION__, caps->main_type, caps->sub_type);
|
||||||
|
|
||||||
|
switch (caps->main_type)
|
||||||
|
{
|
||||||
|
case AUDIO_TYPE_MIXER:
|
||||||
|
{
|
||||||
|
switch (caps->sub_type)
|
||||||
|
{
|
||||||
|
case AUDIO_MIXER_MUTE:
|
||||||
|
{
|
||||||
|
/* set mute mode */
|
||||||
|
wm8978_mute_enabled(_stm32_audio_play.i2c_bus, RT_FALSE);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AUDIO_MIXER_VOLUME:
|
||||||
|
{
|
||||||
|
int volume = caps->udata.value;
|
||||||
|
|
||||||
|
st_audio->replay_volume = volume;
|
||||||
|
/* set mixer volume */
|
||||||
|
wm8978_set_volume(_stm32_audio_play.i2c_bus, volume);
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
result = -RT_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AUDIO_TYPE_OUTPUT:
|
||||||
|
{
|
||||||
|
switch (caps->sub_type)
|
||||||
|
{
|
||||||
|
case AUDIO_DSP_PARAM:
|
||||||
|
{
|
||||||
|
struct rt_audio_configure config = caps->udata.config;
|
||||||
|
|
||||||
|
st_audio->replay_config.samplerate = config.samplerate;
|
||||||
|
st_audio->replay_config.samplebits = config.samplebits;
|
||||||
|
st_audio->replay_config.channels = config.channels;
|
||||||
|
|
||||||
|
SAIA_config_set(config);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AUDIO_DSP_SAMPLERATE:
|
||||||
|
{
|
||||||
|
st_audio->replay_config.samplerate = caps->udata.config.samplerate;
|
||||||
|
SAIA_samplerate_set(caps->udata.config.samplerate);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AUDIO_DSP_CHANNELS:
|
||||||
|
{
|
||||||
|
st_audio->replay_config.channels = caps->udata.config.channels;
|
||||||
|
SAIA_channels_set(caps->udata.config.channels);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case AUDIO_DSP_SAMPLEBITS:
|
||||||
|
{
|
||||||
|
st_audio->replay_config.samplebits = caps->udata.config.samplebits;
|
||||||
|
SAIA_samplebits_set(caps->udata.config.samplebits);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
result = -RT_ERROR;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t stm32_player_init(struct rt_audio_device *audio)
|
||||||
|
{
|
||||||
|
/* initialize wm8978 */
|
||||||
|
_stm32_audio_play.i2c_bus = (struct rt_i2c_bus_device *)rt_device_find(CODEC_I2C_NAME);
|
||||||
|
|
||||||
|
sai_a_init();
|
||||||
|
wm8978_init(_stm32_audio_play.i2c_bus);
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t stm32_player_start(struct rt_audio_device *audio, int stream)
|
||||||
|
{
|
||||||
|
if (stream == AUDIO_STREAM_REPLAY)
|
||||||
|
{
|
||||||
|
HAL_SAI_Transmit_DMA(&_sai_a.hsai, _stm32_audio_play.tx_fifo, TX_DMA_FIFO_SIZE / 2);
|
||||||
|
wm8978_player_start(_stm32_audio_play.i2c_bus);
|
||||||
|
}
|
||||||
|
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static rt_err_t stm32_player_stop(struct rt_audio_device *audio, int stream)
|
||||||
|
{
|
||||||
|
if (stream == AUDIO_STREAM_REPLAY)
|
||||||
|
{
|
||||||
|
HAL_SAI_DMAStop(&_sai_a.hsai);
|
||||||
|
}
|
||||||
|
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void stm32_player_buffer_info(struct rt_audio_device *audio, struct rt_audio_buf_info *info)
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* TX_FIFO
|
||||||
|
* +----------------+----------------+
|
||||||
|
* | block1 | block2 |
|
||||||
|
* +----------------+----------------+
|
||||||
|
* \ block_size /
|
||||||
|
*/
|
||||||
|
info->buffer = _stm32_audio_play.tx_fifo;
|
||||||
|
info->total_size = TX_DMA_FIFO_SIZE;
|
||||||
|
info->block_size = TX_DMA_FIFO_SIZE / 2;
|
||||||
|
info->block_count = 2;
|
||||||
|
}
|
||||||
|
static struct rt_audio_ops _p_audio_ops =
|
||||||
|
{
|
||||||
|
.getcaps = stm32_player_getcaps,
|
||||||
|
.configure = stm32_player_configure,
|
||||||
|
.init = stm32_player_init,
|
||||||
|
.start = stm32_player_start,
|
||||||
|
.stop = stm32_player_stop,
|
||||||
|
.transmit = RT_NULL,
|
||||||
|
.buffer_info = stm32_player_buffer_info,
|
||||||
|
};
|
||||||
|
|
||||||
|
int rt_hw_sound_init(void)
|
||||||
|
{
|
||||||
|
rt_uint8_t *tx_fifo;
|
||||||
|
|
||||||
|
/* player */
|
||||||
|
tx_fifo = rt_malloc(TX_DMA_FIFO_SIZE);
|
||||||
|
if (tx_fifo == RT_NULL)
|
||||||
|
{
|
||||||
|
return -RT_ENOMEM;
|
||||||
|
}
|
||||||
|
rt_memset(tx_fifo, 0, TX_DMA_FIFO_SIZE);
|
||||||
|
_stm32_audio_play.tx_fifo = tx_fifo;
|
||||||
|
|
||||||
|
/* register sound device */
|
||||||
|
_stm32_audio_play.audio.ops = &_p_audio_ops;
|
||||||
|
rt_audio_register(&_stm32_audio_play.audio, "sound0", RT_DEVICE_FLAG_WRONLY, &_stm32_audio_play);
|
||||||
|
|
||||||
|
return RT_EOK;
|
||||||
|
}
|
||||||
|
|
||||||
|
INIT_DEVICE_EXPORT(rt_hw_sound_init);
|
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2006-2018, RT-Thread Development Team
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
|
*
|
||||||
|
* Change Logs:
|
||||||
|
* Date Author Notes
|
||||||
|
* 2019-07-28 Ernest the first version
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __DRV_SOUND_H_
|
||||||
|
#define __DRV_SOUND_H_
|
||||||
|
|
||||||
|
#include <rtthread.h>
|
||||||
|
#include <rtdevice.h>
|
||||||
|
#include <board.h>
|
||||||
|
|
||||||
|
#define AUDIO_FREQUENCY_048K ((rt_uint32_t) 48000)
|
||||||
|
#define AUDIO_FREQUENCY_044K ((rt_uint32_t) 44100)
|
||||||
|
#define AUDIO_FREQUENCY_032K ((rt_uint32_t) 32000)
|
||||||
|
#define AUDIO_FREQUENCY_022K ((rt_uint32_t) 22050)
|
||||||
|
#define AUDIO_FREQUENCY_016K ((rt_uint32_t) 16000)
|
||||||
|
#define AUDIO_FREQUENCY_011K ((rt_uint32_t) 11025)
|
||||||
|
#define AUDIO_FREQUENCY_008K ((rt_uint32_t) 8000)
|
||||||
|
|
||||||
|
struct drv_sai
|
||||||
|
{
|
||||||
|
SAI_HandleTypeDef hsai;
|
||||||
|
DMA_HandleTypeDef hdma;
|
||||||
|
};
|
||||||
|
|
||||||
|
void SAIA_samplerate_set(rt_uint32_t samplerate);
|
||||||
|
void SAIA_channels_set(rt_uint16_t channels);
|
||||||
|
void SAIA_samplebits_set(rt_uint16_t samplebits);
|
||||||
|
void SAIA_config_set(struct rt_audio_configure config);
|
||||||
|
|
||||||
|
rt_err_t SAIA_tx_dma(void);
|
||||||
|
rt_err_t SAIA_config_init(void);
|
||||||
|
|
||||||
|
#endif
|
File diff suppressed because it is too large
Load Diff
|
@ -0,0 +1,58 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2006-2018, RT-Thread Development Team
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: Apache-2.0
|
||||||
|
*
|
||||||
|
* Change Logs:
|
||||||
|
* Date Author Notes
|
||||||
|
* 2018-11-14 ZeroFree first implementation
|
||||||
|
* 2019-07-28 Ernest perfect player, add record and other APIs
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifndef __DRV_WM8978_H__
|
||||||
|
#define __DRV_WM8978_H__
|
||||||
|
|
||||||
|
#include <rtthread.h>
|
||||||
|
#include <rtdevice.h>
|
||||||
|
|
||||||
|
enum data_fomat_select
|
||||||
|
{
|
||||||
|
RIGHT_FOMAT_SELECT,
|
||||||
|
LEFT_FOMAT_SELECT,
|
||||||
|
I2S_FOMAT_SELECT,
|
||||||
|
PCM_FOMAT_SELECT,
|
||||||
|
};
|
||||||
|
|
||||||
|
rt_err_t wm8978_init(struct rt_i2c_bus_device *dev);
|
||||||
|
void wm8978_player_start(struct rt_i2c_bus_device *dev);
|
||||||
|
void wm8978_record_start(struct rt_i2c_bus_device *dev);
|
||||||
|
|
||||||
|
int wm8978_set_volume(struct rt_i2c_bus_device *dev, int vol);
|
||||||
|
|
||||||
|
void wm8978_reset(struct rt_i2c_bus_device *dev);
|
||||||
|
/* enable ADC/DAC */
|
||||||
|
void wm8978_ADC_enabled(struct rt_i2c_bus_device *dev, rt_bool_t bool);
|
||||||
|
void wm8978_DAC_enabled(struct rt_i2c_bus_device *dev, rt_bool_t bool);
|
||||||
|
|
||||||
|
void wm8978_mic_enabled(struct rt_i2c_bus_device *dev, rt_bool_t bool);
|
||||||
|
void wm8978_linein_enabled(struct rt_i2c_bus_device *dev, rt_bool_t bool);
|
||||||
|
void wm8978_aux_enabled(struct rt_i2c_bus_device *dev, rt_bool_t bool);
|
||||||
|
void wm8978_linein_gain(struct rt_i2c_bus_device *dev, rt_uint8_t value);
|
||||||
|
void wm8978_aux_gain(struct rt_i2c_bus_device *dev, rt_uint8_t value);
|
||||||
|
void wm8978_mic_gain(struct rt_i2c_bus_device *dev, rt_uint8_t gain);
|
||||||
|
void wm8978_output_set(struct rt_i2c_bus_device *dev, rt_bool_t dac, rt_bool_t bypass);
|
||||||
|
void wm8978_hpvol_set(struct rt_i2c_bus_device *dev, rt_uint8_t volume);
|
||||||
|
void wm8978_spkvol_set(struct rt_i2c_bus_device *dev, rt_uint8_t volume);
|
||||||
|
|
||||||
|
/* set interface mode */
|
||||||
|
void wm8978_interface_cfg(struct rt_i2c_bus_device *dev, enum data_fomat_select fmt, rt_uint32_t bitBand);
|
||||||
|
|
||||||
|
void wm8978_mute_enabled(struct rt_i2c_bus_device *dev, rt_bool_t enable);
|
||||||
|
rt_err_t wm8978_set_EQ1(struct rt_i2c_bus_device *dev, rt_uint8_t freq, rt_uint8_t gain);
|
||||||
|
rt_err_t wm8978_set_EQ2(struct rt_i2c_bus_device *dev, rt_uint8_t freq, rt_uint8_t gain);
|
||||||
|
rt_err_t wm8978_set_EQ3(struct rt_i2c_bus_device *dev, rt_uint8_t freq, rt_uint8_t gain);
|
||||||
|
rt_err_t wm8978_set_EQ4(struct rt_i2c_bus_device *dev, rt_uint8_t freq, rt_uint8_t gain);
|
||||||
|
rt_err_t wm8978_set_EQ5(struct rt_i2c_bus_device *dev, rt_uint8_t freq, rt_uint8_t gain);
|
||||||
|
void wm8978_3D_Set(struct rt_i2c_bus_device *dev, rt_uint8_t depth);
|
||||||
|
|
||||||
|
#endif
|
Loading…
Reference in New Issue