STM32/STM8技术论坛
直播中

张玉英

7年用户 194经验值
私信 关注
[问答]

读取flash中的语音文件WAV如何导入到flash中

目前,需要做个功能 读取fllash中的语音文件,然后利用PWM播放出来;
现在我面临以前几个 问题
1、FLASH只有1M,我已经 存了字库了大概750kb;剩下的已经不多了(我需要的也就几句话,先搞一句话,空间的问题后面再说)
2、WAV如何导入到flash中 ;(需要把非语音字节去掉在导入,还是直接把WAV文件直接通过修改后缀的方式改成 BIN文件然后才导入)(我打算用 串口接收,然后写入到flash中)
3、我用 朗读女生成了我需要放出来的那几个字,但是看不到采样级别(我的电脑是WIN10 系统)

回帖(15)

张玉英

2018-12-6 09:08:07
说实话,一头雾水,网上 好多都是SD的方式;
举报

周丽

2018-12-6 09:22:55
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 09:36:30
引用: 吕少大大 发表于 2018-12-6 20:23
1.如果存储空间不够,可以将wav文件转换成8k采样率8bit深度单声道文件,下个GoldWave软件即可。
250k字节空间,对8k/8bit/mono文件,可以存储约32秒音频。
2.可直接将wav文件写入flash中,但读取的时候,按照wav的格式,去掉头部即可,一般是前44字节。

非常感谢;我就先去动手了;
不过,我还有个想法;因为我需要的语句并不多,只有几条;后面导入不可能单独导入,肯定是要做成一个文件去导入;我打算,把所有单独的语音文件,去掉前面的格式块的数据,将数据整合起来;如果能行,尽量做到一条语音占据一个page(或者是sector,这样的整个地址);这样通过地址偏移就能准确调用每一个想要的语音包
举报

张玉英

2018-12-6 09:53:02
引用: 吕少大大 发表于 2018-12-6 20:23
1.如果存储空间不够,可以将wav文件转换成8k采样率8bit深度单声道文件,下个GoldWave软件即可。
250k字节空间,对8k/8bit/mono文件,可以存储约32秒音频。
2.可直接将wav文件写入flash中,但读取的时候,按照wav的格式,去掉头部即可,一般是前44字节。

另外,PWM占空比如何通过当前数据来计算呢
举报

张玉英

2018-12-6 10:12:30
引用: sdfsgsd 发表于 2018-12-6 20:36
非常感谢;我就先去动手了;
不过,我还有个想法;因为我需要的语句并不多,只有几条;后面导入不可能单独导入,肯定是要做成一个文件去导入;我打算,把所有单独的语音文件,去掉前面的格式块的数据,将数据整合起来;如果能行,尽量做到一条语音占据一个page(或者是sector,这样的整个地址);这样通过地址偏移就能准确 ...

去掉WAV格式数据,是因为我懒得解析
举报

张玉英

2018-12-6 10:28:58
8位量化的脉冲宽度为:音频数据/255*pwm周期

PWM占空比是这样计算么
举报

周丽

2018-12-6 10:41:01
引用: sdfsgsd 发表于 2018-12-6 21:29
8位量化的脉冲宽度为:音频数据/255*pwm周期

PWM占空比是这样计算么

差不多就这意思。
你把PWM理解成DAC就行了。
举报

张玉英

2018-12-6 10:54:11
引用: 吕少大大 发表于 2018-12-6 21:41
差不多就这意思。
你把PWM理解成DAC就行了。

应该是这样吧,16位数据相当于转换精度,ARR的值相当于基准电压;
data/(0xFFFF)*ARR;最后得到一个比较值,与ARR比较,产生占空比
举报

张玉英

2018-12-6 11:03:18
引用: 吕少大大 发表于 2018-12-6 21:41
差不多就这意思。
你把PWM理解成DAC就行了。

可能还是哪里有问题;语音输出一直是杂波; 我一共试了三种采样率的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;
}
举报

周丽

2018-12-6 11:11:04
引用: sdfsgsd 发表于 2018-12-6 22:03
可能还是哪里有问题;语音输出一直是杂波; 我一共试了三种采样率的WAV文件;
这个代码是11K 8位数据的WAV文件,红色部分是转换到CCR1寄存器的写入值;

data*2180 这个相乘有没有考虑数据溢出?
举报

张玉英

2018-12-6 11:26:10
引用: 吕少大大 发表于 2018-12-6 22:11
data*2180 这个相乘有没有考虑数据溢出?

额,是有这个可能;这样算 没错 的话,我先转换成浮点数,先做除法,在乘
举报

王鹏

2018-12-6 11:44:14
__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输出的是音频数据.
举报

张玉英

2018-12-6 11:58:31
引用: qwer34 发表于 2018-12-6 22:44
__interrupt VectorNumber_Vtimch0 void _Sampling_Interrupt(void)
{            
  // clear interrupt and call oninterrupt function

音频数据应该要转换成占空比输出吧;
举报

张玉英

2018-12-6 12:09:46
引用: qwer34 发表于 2018-12-6 22:44
__interrupt VectorNumber_Vtimch0 void _Sampling_Interrupt(void)
{            
  // clear interrupt and call oninterrupt function

这里PWM频率和定时器中断频率一直,在定时器中断中跟新数据,是这样吗
举报

张玉英

2018-12-6 12:20:32
引用: qwer34 发表于 2018-12-6 22:44
__interrupt VectorNumber_Vtimch0 void _Sampling_Interrupt(void)
{            
  // clear interrupt and call oninterrupt function

我感觉应该他妈的挺简单的,不知道为什么就是搞不出来,
我用的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);//
举报

更多回帖

发帖
×
20
完善资料,
赚取积分