用51单片机实现音频信号的频谱显示

发布者:心境恬淡最新更新时间:2015-04-27 来源: 51hei关键字:51单片机  音频信号  频谱显示 手机看文章 扫描二维码
随时随地手机看文章
思路:外来音频信号经过51单片机,在单片机中进行频谱分析,并将结果显示在LCD(12864或1602)上

要求:频谱显示如同千千静听播放音乐时的频谱显示

希望各位高手能给出详细的解决方案,感激。。。。。。
 51做FFT有些困难,可以使用增强型(RAM)的51机子进行

参考程序:

#include
#define uchar unsigned char
#define uint unsigned int  
#define  channel  0x01     //设置AD通道为 P1.1
//---------------------------------------------------------------------

sbit  SDA_R=P1^2;
sbit  SDA_R_TOP=P1^3;
sbit  SDA_G=P1^4;    
sbit  SDA_G_TOP=P1^5; 
sbit  STCP=P1^6;
sbit  SHCP=P1^7;
//---------------------------------------------------------------------
//----------------------------------------------------------------------------------------------------------------------
//放大128倍后的sin整数表(128) 
code char SIN_TAB[128] = { 0, 6, 12, 18, 24, 30, 36, 42, 48, 54, 59, 65, 70, 75, 80, 85, 89, 94, 98, 102,

105, 108, 112, 114, 117, 119, 121, 123, 124, 125, 126, 126, 126, 126, 126, 125, 124, 123, 121, 119, 117, 114, 112,

108, 105, 102, 98, 94, 89, 85, 80, 75, 70, 65, 59, 54, 48, 42, 36, 30, 24, 18, 12, 6, 0, -6, -12, -18, -24, -30,

-36, -42, -48, -54, -59, -65, -70, -75, -80, -85, -89, -94, -98, -102, -105, -108, -112, -114, -117, -119, -121,

-123, -124, -125, -126, -126, -126, -126, -126, -125, -124, -123, -121, -119, -117, -114, -112, -108, -105, -102,

-98, -94, -89, -85, -80, -75, -70, -65, -59, -54, -48, -42, -36, -30, -24, -18, -12, -6 };

//放大128倍后的cos整数表(128)
code char COS_TAB[128] = { 127, 126, 126, 125, 124, 123, 121, 119, 117, 114, 112, 108, 105, 102, 98, 94,

89, 85, 80, 75, 70, 65, 59, 54, 48, 42, 36, 30, 24, 18, 12, 6, 0, -6, -12, -18, -24, -30, -36, -42, -48, -54, -59,

-65, -70, -75, -80, -85, -89, -94, -98, -102, -105, -108, -112, -114, -117, -119, -121, -123, -124, -125, -126, -

126, -126, -126, -126, -125, -124, -123, -121, -119, -117, -114, -112, -108, -105, -102, -98, -94, -89, -85, -80,

-75, -70, -65, -59, -54, -48, -42, -36, -30, -24, -18, -12, -6, 0, 6, 12, 18, 24, 30, 36, 42, 48, 54, 59, 65, 70,

75, 80, 85, 89, 94, 98, 102, 105, 108, 112, 114, 117, 119, 121, 123, 124, 125, 126, 126 };

//采样存储序列表
code char LIST_TAB[128] = { 0, 64, 32, 96, 16, 80, 48, 112,
8, 72, 40, 104, 24, 88, 56, 120,
4, 68, 36, 100, 20, 84, 52, 116,
12, 76, 44, 108, 28, 92, 60, 124,
2, 66, 34, 98, 18, 82, 50, 114,
10, 74, 42, 106, 26, 90, 58, 122,
6, 70, 38, 102, 22, 86, 54, 118,
14, 78, 46, 110, 30, 94, 62, 126,
1, 65, 33, 97, 17, 81, 49, 113,
9, 73, 41, 105, 25, 89, 57, 121,
5, 69, 37, 101, 21, 85, 53, 117,
13, 77, 45, 109, 29, 93, 61, 125,
3, 67, 35, 99, 19, 83, 51, 115,
11, 75, 43, 107, 27, 91, 59, 123,
7, 71, 39, 103, 23, 87, 55, 119,
15, 79, 47, 111, 31, 95, 63, 127
};


uchar COUNT=0,COUNT1=0,ADC_Count=0,LINE=15,G,T;
uchar i,j,k,b,p;                 
int Temp_Real,Temp_Imag,temp;                // 中间临时变量  
uint TEMP1; 
int xdata Fft_Real[128]; 
int xdata Fft_Image[128];               // fft的虚部 
uchar xdata LED_TAB2[64];    //记录 漂浮物 是否需要 停顿一下
uchar xdata LED_TAB[64];    //记录红色柱状 
uchar xdata LED_TAB1[64];    //记录 漂浮点


void Delay(uint a)
{
 while(a--);
}

void FFT()
{    //uchar X;               
    for( i=1; i<=7; i++)                            /* for(1) */
    { 
        b=1;
        b <<=(i-1);                                       //碟式运算,用于计算 隔多少行计算 例如 第一极 1和2行计算,,第二级 
        for( j=0; j<=b-1; j++)                              /* for (2) */
        { 
            p=1;
            p <<= (7-i);            
            p = p*j;
            for( k=j; k<128; k=k+2*b)                /* for (3) 基二fft */
            { 
                Temp_Real = Fft_Real[k]; Temp_Imag = Fft_Image[k]; temp = Fft_Real[k+b];
                Fft_Real[k] = Fft_Real[k] + ((Fft_Real[k+b]*COS_TAB[p])>>7) + ((Fft_Image[k+b]*SIN_TAB[p])>>7);
                Fft_Image[k] = Fft_Image[k] - ((Fft_Real[k+b]*SIN_TAB[p])>>7) + ((Fft_Image[k+b]*COS_TAB[p])>>7);
                Fft_Real[k+b] = Temp_Real - ((Fft_Real[k+b]*COS_TAB[p])>>7) - ((Fft_Image[k+b]*SIN_TAB[p])>>7);
                Fft_Image[k+b] = Temp_Imag + ((temp*SIN_TAB[p])>>7) - ((Fft_Image[k+b]*COS_TAB[p])>>7);     
                // 移位.防止溢出. 结果已经是本值的 1/64               
              Fft_Real[k] >>= 1;             
                Fft_Image[k] >>= 1; 
               Fft_Real[k+b]  >>= 1;                 
                Fft_Image[k+b]  >>= 1; 
                                                                               
            }     
        } 
    } 
// X=((((Fft_Real[1]* Fft_Real[1]))+((Fft_Image[1]*Fft_Image[1])))>>7);
  Fft_Real[0]=Fft_Image[0]=0;    //去掉直流分量
//  Fft_Real[63]=Fft_Image[63]=0;
 for(j=0;j<64;j++)            
 {                     
  TEMP1=((((Fft_Real[j]* Fft_Real[j]))+((Fft_Image[j]*Fft_Image[j])))>>1);//求功率
  if(TEMP1>1)TEMP1--;
  else TEMP1=0;
  if(TEMP1>31)TEMP1=31; 
  if(TEMP1>(LED_TAB[j]))LED_TAB[j]=TEMP1; 
  if(TEMP1>(LED_TAB1[j]))
  {   LED_TAB1[j]=TEMP1;
   LED_TAB2[j]=18;      //提顿速度=12
  }
 }       
}[page]

void Init()
{  
 
//-----------------------------------------------------------------------------------
     P1ASF = 0x02;                 //0000,0010, 将 P1.1 置成模拟口    
     AUXR1 &=0xFB;                 //1111,1011, 令 ADRJ=0
  EADC=1;        //AD中断打开
  ADC_CONTR = ADC_POWER | ADC_SPEEDHH | ADC_START | channel;
       //1110 1001   1打开 A/D (ADC_POWER)转换电源;11速度为70周期一次;
          //0中断标志清零;1启动adc(ADC_START);001AD通道打开(这里为P1.1);
//-----------------------------------------------------------------------------------
  P2M0=1;
  P0M0=1;
     TMOD=0X12;        
  TH0=0x30;      //大约20K的采样率(要完整频段需40K以上。但音频中10k以下居多,故本人选择20K采样,美观些)                                 
  TL0=0x30;
  TH1=0xEE;               
  TL1=0XC0;
  ET0=1;            //定时器0 打开
  TR0=0;            //关闭定时器
  ET1=1;
  TR1=1;
  PT1=0;
  PT0=1;
     IPH=PADCH;
  IP=PADC;        //中断优先级
  EA=1;       //总中断打开 
}


 void ADC_Finish() interrupt 5
 {    ADC_CONTR &= !ADC_FLAG; 
     Fft_Real[LIST_TAB[ADC_Count]]=(int)((ADC_RES)<<1)+(ADC_RESL>>1)-256;//-512; //按LIST_TAB表里的顺序,进行存储 采样值,,
 //  ADC_CONTR = ADC_POWER | ADC_SPEEDHH| ADC_START | channel; // 为了采集负电压,采用 偏置采集。电压提高到1/2 vcc,,所以要减去256
   if(ADC_Count<=127)ADC_Count++;
   else {EADC=0;TR0=0;}                 
 }

 void LED_Display() interrupt 3      //中断一次 显示一行。。。
 {    
  TH1=0xF3;             
  TL1=0X00;      
  for (G=0;G<64;G++)        //往点阵屏填充 一行的 数据
  { 
   if(LED_TAB[G]<=LINE+16)SDA_R_TOP=1;
   else SDA_R_TOP=0;
     if(LED_TAB[G]<=LINE)SDA_R=1;
   else SDA_R=0;

   if(LED_TAB1[G]==LINE){SDA_G_TOP=1;SDA_G=0;}
   else if(LED_TAB1[G]==(LINE+16)){SDA_G_TOP=0;SDA_G=1;}
   else SDA_G=SDA_G_TOP=1; 
   SHCP=1;SHCP=0;
  }
  STCP=1;STCP=0;
  P2=15-LINE;
  if(LINE>0)LINE--;
  else LINE=15;
  //////////////////////////
 
   if(LED_TAB[COUNT]>0)LED_TAB[COUNT]--;     //柱状递减,
   COUNT++;
   if(LED_TAB[COUNT]>0)LED_TAB[COUNT]--;
   COUNT++;
   if(LED_TAB[COUNT]>0)LED_TAB[COUNT]--;
   COUNT++;
   if(LED_TAB[COUNT]>0)LED_TAB[COUNT]--;
   COUNT++;
    if(LED_TAB[COUNT]>0)LED_TAB[COUNT]--;     //柱状递减,
   COUNT++;
   if(LED_TAB[COUNT]>0)LED_TAB[COUNT]--;
   COUNT++;
   if(LED_TAB[COUNT]>0)LED_TAB[COUNT]--;
   COUNT++;
   if(LED_TAB[COUNT]>0)LED_TAB[COUNT]--;
   COUNT++;
   if(COUNT>=64)COUNT=0;

                  //漂浮物递减
   if(LED_TAB2[COUNT1]==0)     //判断是否需要停顿    
   {
     if(LED_TAB1[COUNT1]>LED_TAB[COUNT1])LED_TAB1[COUNT1]--;//大于柱状则递减(保持漂浮物在柱状之上) 
   }
   else LED_TAB2[COUNT1]--;
   COUNT1++;
   if(LED_TAB2[COUNT1]==0)
   {
     if(LED_TAB1[COUNT1]>LED_TAB[COUNT1])LED_TAB1[COUNT1]--; 
   }
   else LED_TAB2[COUNT1]--;
   COUNT1++;
   if(LED_TAB2[COUNT1]==0)     //判断是否需要停顿    
   {
     if(LED_TAB1[COUNT1]>LED_TAB[COUNT1])LED_TAB1[COUNT1]--;//大于柱状则递减(保持漂浮物在柱状之上) 
   }
   else LED_TAB2[COUNT1]--;
   COUNT1++;
   if(LED_TAB2[COUNT1]==0)
   {
     if(LED_TAB1[COUNT1]>LED_TAB[COUNT1])LED_TAB1[COUNT1]--; 
   }
   else LED_TAB2[COUNT1]--;
   COUNT1++;
   if(COUNT1>=64)COUNT1=0;
 }

void Ad_Control() interrupt 1      //控制采样率
{
 ADC_CONTR = ADC_POWER | ADC_SPEEDHH| ADC_START | channel;  //开始AD采集

  //==============================================================================================================
 // *******************          main()       *********************************            
 //===============================================================================================================

 void main()
 {
  Init();
 while(1)
 {  
  ADC_Count=0;
  TR0=1;
  EADC=1;            //开启定时器中断0,,开启ADC
  while(ADC_Count<128);
  FFT();
  //FFT运算。并转换为 功率值。。。         
  // TR1=1;   
 }
 }

关键字:51单片机  音频信号  频谱显示 引用地址:用51单片机实现音频信号的频谱显示

上一篇:51单片机控制8位LED数码管做9999累加的程序
下一篇:51单片机心形流水灯源程序

推荐阅读最新更新时间:2024-03-16 13:59

51单片机环境的搭建-1.3指令下载/烧写软件
第一章 51单片机环境的搭建-1.3指令下载/烧写软件 1.3 指令下载/烧写程序下载STC-ISP:http://pan.baidu.com/s/1kU3oXD1 将STC-ISP-v4.80.rar文件解压,即可使用
[单片机]
51单片机 矩阵键盘
uchar code KEY_TABLE = { 0x77,0xB7,0xD7,0xE7 0x7B,0xBB,0xDB,0xEB 0x7D,0xBD,0xDD,0xED 0x7E,0xBE,0xDE,0xEE };//高四位是列,低四位是行 uchar code TABLE = { '1', '2', '3', '4', '5', '6', '7', '8', '9', 'C', '0', 'A', };
[单片机]
<font color='red'>51单片机</font> 矩阵键盘
音频信号自动增益控制系统
音频信号自动增益控制系统
[模拟电子]
<font color='red'>音频信号</font>自动增益控制系统
C51单片机功能模块和Keil C 数据类型
单片机特点:集成度高,价格低廉,功能强,速度快,功耗低,体积小,使用灵活,稳定可靠 使用领域:家用电器,智能仪器仪表,通信系统,网络系统,数控机床,数据采集和处理,自动检测,工业控制,火箭制导 51单片机数据类型 数据类型    位(bit)    字节(byte)     取值范围 bit         1      1/8         0-1 signed char     8      1         -128~+127 unsigned char     8       1         0~255 enum        8/16      1or2     -128~+127or-32768~+3
[单片机]
AT89C51单片机解密原理
AT89C系列单片机擦除操作的时序为:擦除开始---- 擦除操作硬件初始化(10微秒)---- 擦除加密锁定位(50----200微秒)--- 擦除片内程序存储器内的数据(10毫秒)----- 擦除结束。如果用程序监控擦除过程,一旦加密锁定位被擦除就终止擦除操作,停止进一步擦除片内程序存储器,加过密的单片机就变成没加密的单片机了。片内程序可通过总线被读出。 对于AT89C系列单片机有两种不可破解的加密方法。 一、永久性地破坏单片机的加密位的加密方法。简称OTP加密模式。 二、永久性地破坏单片机的数据总线的加密方法。简称烧总线加密模式。AT89C系列单片机OTP加密模式原理这种编程加密算法烧坏加密锁定位(把芯片内的硅片击
[单片机]
51单片机快速入门指南】6.1:LCD1602的八线、四线控制及自定义符号,完美兼容Proteus仿真
普中51-单核-A2 STC89C52 Windows 10 20H2 Proteus 8 Frofessional v8.9 SP2 Keil uVision V5.29.0.0 PK51 Prof.Developers Kit Version:9.60.0.0 硬知识 摘自《通用1602 液晶显示模块使用手册》、《液晶LCD1602(中文资料)—— ball 2010-5-13整理》 显示特性 单5V电源电压,低功耗、长寿命、高可靠性 内置192种字符(160个5x7点阵字符和32个5x10点阵字符) 具有64个字节的自定义字符RAM,可自定义8个5x8点阵字符或四个5x11点阵字符
[单片机]
【<font color='red'>51单片机</font>快速入门指南】6.1:LCD1602的八线、四线控制及自定义符号,完美兼容Proteus仿真
51单片机--设置数码管的闪烁位置,便于输入数字
/* 有四个独立按键为 K1~K4,有四个数码管,可显示 0000~9999。 初始时都显示 0。 四个数码管中,有一个处于闪烁状态。闪烁的频率为 1Hz。 利用 K3、K4 可以使得闪烁的位置左右移动。 只有不停闪烁的位,才可以用 K1、K2 使其数值递增或递减。 数值的范围是 0~9。 题目连接: http://zhidao.baidu.com/question/455550887523150405.html 做而论道编写程序如下: */ #include reg52.h #include intrins.h sbit k_a = P3^1; sbit k_s = P3^3; sbit
[单片机]
<font color='red'>51单片机</font>--设置数码管的闪烁位置,便于输入数字
TI 54xxDSP与51单片机的接口技术
摘要:TI的54xxDSP是一种定点DSP系列芯片,产生应用于各种信号处理系统,特别是语音信号处理系统。在这些系统中,通常由两部分组成。一部分为DSP子系统,这是整个系统的核心,主要完成采样、数字信号处理以及输出等功能;另一部分为单片机子系统,进行交互界面的控制,如键盘和显示。两个子系统不是各自孤立的,需要进行必要的数据交换。本文主要讨论DSP和51单片机之间通过HPI接口进行连接的设计方法,给出硬件连接以及软件编程方法。 关键词:DSP HPI 单片机 TMS320C54xx是TI公司针对音频信号处理领域推出的一种定点DSP系列芯片,已经在很多语音信号处理系统中得到了广泛的应用。在这些系统中,通常包含DSP和单片机两个子系
[应用]
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved