完善资料让更多小伙伴认识你,还能领取20积分哦, 立即完善>
目前,需要做个功能 读取fllash中的语音文件,然后利用PWM播放出来;
现在我面临以前几个 问题 1、FLASH只有1M,我已经 存了字库了大概750kb;剩下的已经不多了(我需要的也就几句话,先搞一句话,空间的问题后面再说) 2、WAV如何导入到flash中 ;(需要把非语音字节去掉在导入,还是直接把WAV文件直接通过修改后缀的方式改成 BIN文件然后才导入)(我打算用 串口接收,然后写入到flash中) 3、我用 朗读女生成了我需要放出来的那几个字,但是看不到采样级别(我的电脑是WIN10 系统) |
|
相关推荐
15个回答
|
|
说实话,一头雾水,网上 好多都是SD的方式;
|
|
|
|
1.如果存储空间不够,可以将wav文件转换成8k采样率8bit深度单声道文件,下个GoldWave软件即可。
250k字节空间,对8k/8bit/mono文件,可以存储约32秒音频。 2.可直接将wav文件写入flash中,但读取的时候,按照wav的格式,去掉头部即可,一般是前44字节。 播放时,每1/8k秒读取一个字节并输出到PWM上。如果可以,建议把PWM输出改为DAC输出,音效会更好。 3.用GoldWave打开文件即可看出文件的采样率和位宽等信息。 |
|
|
|
吕少大大 发表于 2018-12-6 20:23 非常感谢;我就先去动手了; 不过,我还有个想法;因为我需要的语句并不多,只有几条;后面导入不可能单独导入,肯定是要做成一个文件去导入;我打算,把所有单独的语音文件,去掉前面的格式块的数据,将数据整合起来;如果能行,尽量做到一条语音占据一个page(或者是sector,这样的整个地址);这样通过地址偏移就能准确调用每一个想要的语音包 |
|
|
|
吕少大大 发表于 2018-12-6 20:23 另外,PWM占空比如何通过当前数据来计算呢 |
|
|
|
sdfsgsd 发表于 2018-12-6 20:36 去掉WAV格式数据,是因为我懒得解析 |
|
|
|
|
|
|
|
差不多就这意思。 你把PWM理解成DAC就行了。 |
|
|
|
应该是这样吧,16位数据相当于转换精度,ARR的值相当于基准电压; data/(0xFFFF)*ARR;最后得到一个比较值,与ARR比较,产生占空比 |
|
|
|
可能还是哪里有问题;语音输出一直是杂波; 我一共试了三种采样率的WAV文件; 这个代码是11K 8位数据的WAV文件,红色部分是转换到CCR1寄存器的写入值; //PWM WAV配置----TIM1 --CH1 void Speak_Init(void) { TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure; TIM_OCInitTypeDef TIM_OCInitStructure; GPIO_InitTypeDef GPIO_InitStructure; RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA,ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1,ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8; //GPIOA.8 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA,&GPIO_InitStructure); GPIO_PinAFConfig(GPIOA,GPIO_PinSource8,GPIO_AF_2); //GPIO_AF_2 //wav采样率 24KHz = 48000KHz/(999+1)(1+1); // 16KHz = 48000KHz/(999+1)(2+1) // 11KHz = 48000KHz/(2180+1)(1+1) TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM_TimeBaseInitStructure.TIM_Period = 999; // TIM_TimeBaseInitStructure.TIM_Prescaler = 2; //16K TIM_TimeBaseInitStructure.TIM_Period = 2180; TIM_TimeBaseInitStructure.TIM_Prescaler = 1; //11K TIM_TimeBaseInit(TIM1,&TIM_TimeBaseInitStructure); TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //TIM_OCInitStructure.TIM_Pulse = 500; //测试 TIM_OC1Init(TIM1,&TIM_OCInitStructure);//TIM1--CH1 TIM_Cmd(TIM1,ENABLE); TIM_CtrlPWMOutputs(TIM1,ENABLE);// } uint8_t Speak_Buffer[1024];// 1KB为单位读取 void Speak_DataSet(uint8_t data) { uint16_t speak_data; speak_data = (data*2180)/255; //--8位数据 //printf("rn %d",speak_data); TIM1->CCR1 = speak_data; } |
|
|
|
sdfsgsd 发表于 2018-12-6 22:03 data*2180 这个相乘有没有考虑数据溢出? |
|
|
|
额,是有这个可能;这样算 没错 的话,我先转换成浮点数,先做除法,在乘 |
|
|
|
__interrupt VectorNumber_Vtimch0 void _Sampling_Interrupt(void)
{ // clear interrupt and call oninterrupt function TC0 = g_iEngineSpeedReal + TCNT;// + rand(); TFLG1 = TFLG1_C0F_MASK; PWMDTY2 = g_pSound[g_iSoundIndex]; g_iSoundIndex++; if (g_iSoundIndex >= g_iSoundSize) { g_iSoundIndex = 0; } } 飞思卡尔芯片做过,定时器中断就是采样频率.PWM输出的是音频数据. |
|
|
|
qwer34 发表于 2018-12-6 22:44 音频数据应该要转换成占空比输出吧; |
|
|
|
qwer34 发表于 2018-12-6 22:44 这里PWM频率和定时器中断频率一直,在定时器中断中跟新数据,是这样吗 |
|
|
|
qwer34 发表于 2018-12-6 22:44 我感觉应该他妈的挺简单的,不知道为什么就是搞不出来, 我用的STM32F030C8T6,利用TIM1产生PWM和定时器中断,PWM的频率设置为11KHZ; TIM1->ARR = 2180; TIM1->PSC = 1 ; 然后定时器中断将WAV的数据转换成TIM1->CCR1的值; TIM_TimeBaseInitTypeDef TIM_TimeBaseInitStructure; TIM_OCInitTypeDef TIM_OCInitStructure; GPIO_InitTypeDef GPIO_InitStructure; NVIC_InitTypeDef NVIC_InirStructure; RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA,ENABLE); RCC_APB2PeriphClockCmd(RCC_APB2Periph_TIM1,ENABLE); GPIO_InitStructure.GPIO_Pin = GPIO_Pin_8; //GPIOA.8 GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF; GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_NOPULL; GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz; GPIO_Init(GPIOA,&GPIO_InitStructure); GPIO_PinAFConfig(GPIOA,GPIO_PinSource8,GPIO_AF_2); //GPIO_AF_2 NVIC_InirStructure.NVIC_IRQChannel = TIM1_BRK_UP_TRG_COM_IRQn; NVIC_InirStructure.NVIC_IRQChannelPriority = 3; NVIC_InirStructure.NVIC_IRQChannelCmd = ENABLE; NVIC_Init(&NVIC_InirStructure); //wav采样率 24KHz = 48000KHz/(999+1)(1+1); // 16KHz = 48000KHz/(999+1)(2+1) // 11KHz = 48000KHz/(2180+1)(1+1) TIM_TimeBaseInitStructure.TIM_ClockDivision = TIM_CKD_DIV1; TIM_TimeBaseInitStructure.TIM_CounterMode = TIM_CounterMode_Up; //TIM_TimeBaseInitStructure.TIM_Period = 999; // TIM_TimeBaseInitStructure.TIM_Prescaler = 2; //16K TIM_TimeBaseInitStructure.TIM_Period = 2180; TIM_TimeBaseInitStructure.TIM_Prescaler = 1; //11K TIM_TimeBaseInit(TIM1,&TIM_TimeBaseInitStructure); TIM_OCInitStructure.TIM_OCMode = TIM_OCMode_PWM1; TIM_OCInitStructure.TIM_OCPolarity = TIM_OCPolarity_High; TIM_OCInitStructure.TIM_OutputState = TIM_OutputState_Enable; //TIM_OCInitStructure.TIM_Pulse = 500; //测试 TIM_OC1Init(TIM1,&TIM_OCInitStructure);//TIM1--CH1 TIM_ITConfig(TIM1,TIM_IT_Update,ENABLE);//更新中断 TIM_Cmd(TIM1,ENABLE); TIM_CtrlPWMOutputs(TIM1,ENABLE);// |
|
|
|
你正在撰写答案
如果你是对答案或其他答案精选点评或询问,请使用“评论”功能。
1049 浏览 0 评论
AD7686芯片不传输数据给STM32,但是手按住就会有数据。
1017 浏览 2 评论
2123 浏览 0 评论
如何解决MPU-9250与STM32通讯时,出现HAL_ERROR = 0x01U
1219 浏览 1 评论
hal库中i2c卡死在HAL_I2C_Master_Transmit
1639 浏览 1 评论
小黑屋| 手机版| Archiver| 电子发烧友 ( 湘ICP备2023018690号 )
GMT+8, 2024-11-24 14:01 , Processed in 0.852423 second(s), Total 72, Slave 65 queries .
Powered by 电子发烧友网
© 2015 bbs.elecfans.com
关注我们的微信
下载发烧友APP
电子发烧友观察
版权所有 © 湖南华秋数字科技有限公司
电子发烧友 (电路图) 湘公网安备 43011202000918 号 电信与信息服务业务经营许可证:合字B2-20210191 工商网监 湘ICP备2023018690号