目前,需要做个功能 读取fllash中的语音文件,然后利用PWM播放出来; 现在我面临以前几个 问题 1、FLASH只有1M,我已经 存了字库了大概750kb;剩下的已经不多了(我需要的也就几句话,先搞一句话,空间的问题后面再说) 2、WAV如何导入到flash中 ;(需要把非语音字节去掉在导入,还是直接把WAV文件直接通过修改后缀的方式改成 BIN文件然后才导入)(我打算用 串口接收,然后写入到flash中) 3、我用 朗读女生成了我需要放出来的那几个字,但是看不到采样级别(我的电脑是WIN10 系统) |
250k字节空间,对8k/8bit/mono文件,可以存储约32秒音频。
2.可直接将wav文件写入flash中,但读取的时候,按照wav的格式,去掉头部即可,一般是前44字节。
播放时,每1/8k秒读取一个字节并输出到PWM上。如果可以,建议把PWM输出改为DAC输出,音效会更好。
3.用GoldWave打开文件即可看出文件的采样率和位宽等信息。
非常感谢;我就先去动手了;
不过,我还有个想法;因为我需要的语句并不多,只有几条;后面导入不可能单独导入,肯定是要做成一个文件去导入;我打算,把所有单独的语音文件,去掉前面的格式块的数据,将数据整合起来;如果能行,尽量做到一条语音占据一个page(或者是sector,这样的整个地址);这样通过地址偏移就能准确调用每一个想要的语音包
另外,PWM占空比如何通过当前数据来计算呢
去掉WAV格式数据,是因为我懒得解析
PWM占空比是这样计算么
差不多就这意思。
你把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("\r\n %d",speak_data);
TIM1->CCR1 = speak_data;
}
data*2180 这个相乘有没有考虑数据溢出?
额,是有这个可能;这样算 没错 的话,我先转换成浮点数,先做除法,在乘
{
// 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输出的是音频数据.
音频数据应该要转换成占空比输出吧;
这里PWM频率和定时器中断频率一直,在定时器中断中跟新数据,是这样吗