【求助】PWM转DA输出WAV音频的一个小程序

2020-01-12 17:25发布

本帖最后由 wicy001 于 2012-4-28 13:45 编辑

利用STC90C516RD+实现PWM输出WAV格式的音频,这是一个非常简单的程序

说明:1.此单片机没有集成PWM,需要使用定时器实现
         2.播放wav音频的格式为8bit,8Khz 单声道,大小<60K,放在程序存储器中。
         3.晶振需要 8Khz*(2^8bit)*12T=24.576Mhz,如果烧写使用6T方式,晶振可减半。
         4.端口接耳机,耳机另一端接+5V,这样声音听着会大一些。

问题:
       输出有噪音,噪音非常尖锐,而且声音很大,比声音信号要大。请各位大侠看看,问题出在哪,应该怎么解决?

-----------
声音尖锐,频率应该很高,试着加一个RC低通,但效果不明显。

---------------为方便大家快速阅读,将帖子相关内容整理如下-------------
图1:RC电路


图2:经过RC电路仿真波形      【如何去掉图中的尖刺?】


图3:未经RC电路前端口的仿真波形  【斜坡受RC影响,正常。如果不加RC电路,输出是平的】



/*************************程序****************************/
#include <reg52.h>   //MCU为STC89/STC90
#include "sound.h"    //wav格式数据

sbit BEEP = P0^0;         //耳机一端接在P0.0脚上,一端接+5V
unsigned int counter,wavedataL;
unsigned char currentdata,nextdata,i;




void sys_start()
{

    TMOD|= 0x02;  //T0,自动重装模式
        TH0 = currentdata;
        TL0 = currentdata;


    ET0 = 1;
        EA  = 1;       

        TR0 = 1;
        BEEP=0;
}



main()
{
  unsigned int tmp;

  tmp=sound[0x29];                           //从wav文件中取,数据长度。数据长度应该是4个字节,
  tmp=tmp<<8;
  wavedataL=tmp+sound[0x28];  //由于wav存在片内flash中,文件大小有限制,数据长度用int型足够表示了。

  counter=0x2c;          //wav文件第一个数据
  
  currentdata=sound[counter];       
  nextdata=sound[counter];

  //以上数字请参考wav格式说明

  sys_start();


  while(1)
  {
   
        if(counter>wavedataL)                //反复重播
                 {
                         counter=0x2c;      
                 }

        if(BEEP) TH0=0xFF-currentdata;
        else TH0=currentdata;

   }


}

/*************************************/

void Timer0int(void) interrupt 1    //定时器0中断
{
   
       
        BEEP=~BEEP;

        if(BEEP)
        {

                counter++;
                nextdata=sound[counter];

        }
        else
        currentdata=nextdata;
}


友情提示: 此问题已得到解决,问题已经关闭,关闭后问题禁止继续编辑,回答。
100条回答
vbvc6
2020-01-24 04:21
我用LMF4L120试了下,效果差的很,有很明显的尖噪音,我直接用IO驱动蜂鸣器,接音响效果好点,但噪声依然很大
50M/256 等效约200k的DA

void BeepLevelSet(int level);

#define VOICE_TIMER                         SYSCTL_PERIPH_TIMER2
#define VOICE_TIMER_BASE        TIMER2_BASE
#define VOICE_GPIO_PIN_CFG      GPIO_PB0_T2CCP0
#define VOICE_GPIO_PIN          GPIO_PIN_0
#define VOICE_GPIO_PERIPH       SYSCTL_PERIPH_GPIOB
#define VOICE_GPIO_BASE                  GPIO_PORTB_BASE

static unsigned int BufferPosMax=0;
static unsigned char *WaveBufferStart=0;

void IntTimer5HandlerA(void)
{
        static unsigned int BufferPos;
        HWREG(TIMER5_BASE + TIMER_O_ICR)   = 0x01;
        if(0==WaveBufferStart){
                BufferPos=0;
                return;
        }
        BeepLevelSet(WaveBufferStart[BufferPos]);
        BufferPos++;
        if(BufferPos>=BufferPosMax){
                BufferPos=0;
                WaveBufferStart=0;
        }
        return;
}
void BeepInit(void)
{
    //
    // Enable the GPIO Port and Timer for each LED
    //
        SysCtlPeripheralEnable(VOICE_GPIO_PERIPH);
        SysCtlPeripheralEnable(VOICE_TIMER);
        GPIOPinTypeGPIOOutput(GPIO_PORTB_BASE,GPIO_PIN_0);
    //
    // Configure each timer for output mode
    //
        HWREG(VOICE_TIMER_BASE + TIMER_O_CTL)   = 0x00;
    HWREG(VOICE_TIMER_BASE + TIMER_O_CFG)   = 0x04;
    HWREG(VOICE_TIMER_BASE + TIMER_O_TAMR)  = 0x0A;
    HWREG(VOICE_TIMER_BASE + TIMER_O_TAILR) = 0xFF;

    //
    // Invert the output signals.
    //
    HWREG(VOICE_TIMER_BASE + TIMER_O_CTL)   |= 0x40;
    //
    // Enable timers to begin counting
    //
    TimerEnable(VOICE_TIMER_BASE, TIMER_A);
    //
    // Reconfigure each LED's GPIO pad for timer control
    //
    GPIOPinConfigure(VOICE_GPIO_PIN_CFG);
    GPIOPinTypeTimer(VOICE_GPIO_BASE, VOICE_GPIO_PIN);
    GPIOPadConfigSet(VOICE_GPIO_BASE, VOICE_GPIO_PIN, GPIO_STRENGTH_2MA,
                     GPIO_PIN_TYPE_STD);

    BeepLevelSet(0);
    SysCtlPeripheralEnable(SYSCTL_PERIPH_TIMER5);
        HWREG(TIMER5_BASE + TIMER_O_CTL)   = 0x00;
    HWREG(TIMER5_BASE + TIMER_O_CFG)   = 0x04;
    HWREG(TIMER5_BASE + TIMER_O_TAMR)  = 0x02;
    HWREG(TIMER5_BASE + TIMER_O_TAILR) = 0x186A; //8000
    TimerMatchSet(TIMER5_BASE,TIMER_A,2);
    TimerEnable(TIMER5_BASE, TIMER_A);
    TimerIntEnable(TIMER5_BASE,TIMER_TIMA_TIMEOUT);//TIMER_TIMA_MATCH //TIMER_TIMA_TIMEOUT
    IntEnable(INT_TIMER5A);
}

void BeepLevelSet(int level)
{
        TimerMatchSet(VOICE_TIMER_BASE,TIMER_A,level);
}

void TestWaveOut(void)
{
        BufferPosMax=sizeof(wave_data);
        WaveBufferStart=(unsigned char*)wave_data;
}

一周热门 更多>