http://share.eepw.com.cn/share/download/id/79366
安装好后,下一步就是像普通驱动寻一样就可以使用了
如果还遇到了串口号冲突的问题,还可以参考campozeng的技术问题贴:
http://forum.eepw.com.cn/thread/223263/1
希望能够帮到你
共同学习,一起进步!
【语音平台开发进程】:SPI读操作最高速测试--1.5MHz
最近一直在进行SPI的调试工作,在调试的过程中不免有些单调,就在VS1053B库调试之余,想测一下我搭建的这款硬件系统的SPI到底能跑到多高的速度,这在SPI读写参数设置中也算是一个参考吧!
以下是我按照有快到慢的速度逐步对SPI进行测试的:
当SPI.setClockDivider(SPI_CLOCK_DIV4)时,即SPI的时钟速度为12MHz时SPI读操作失败!如下图所示:
当SPI.setClockDivider(SPI_CLOCK_DIV8)时,即SPI的时钟速度为6MHz时SPI读操作失败!如下图所示:
当SPI.setClockDivider(SPI_CLOCK_DIV16)时,即SPI的时钟速度为3MHz时SPI读操作部分成功!如下图所示:
当SPI.setClockDivider(SPI_CLOCK_DIV32)时,即SPI的时钟速度为1.5MHz时SPI读操作完全成功!如下图所示:
如以上图中所示,目前已将两个VS1053B库函数调通,这两个库函数分别为:
void VS_Init(void);
用于SPI的初始化;
unsigned int VS_RD_Reg(unsigned char address);
用于读取特定地址的寄存器的值。
OK,共同学习,一起进步!
经过近一周的编写和调试,昨天,正式将VS1053B芯片的SPI驱动程序编写和调试完毕,在编写和调试的过程中也吸取了许多经验和教训,现总结如下:
1、首先,用最原始的方法在主函数里先将SPI对VS1053B芯片的简单控制实现。
2、在最原始的方法上对上述控制进行头文件的简单改写,采用边改写,边调试的方法,这一步目的不再于要实现多么强大的驱动功能,而在于将整个驱动的整体框架搭建起来,以便于在下一步中在向其中添加功能的同时,而又不破坏整个架构的完整性。
3、一步步按照预先的规划逐步完善程序的功能,同样也是边编写边调试。
4、最后,整理出正个驱动程序,写出Demo主程序测试函数,以便于演示和测试。
下面将编写好的驱动程序贴于下面,以便于大家学习和提出宝贵的意见和建议:
VS1053B.h头文件程序:
#ifndef __VS10XX_H__
#define __VS10XX_H__
#include <rxduino.h>
#define SPI_MODE 0x00
#define SPI_STATUS 0x01
#define SPI_BASS 0x02
#define SPI_CLOCKF 0x03
#define SPI_DECODE_TIME 0x04
#define SPI_AUDATA 0x05
#define SPI_WRAM 0x06
#define SPI_WRAMADDR 0x07
#define SPI_HDAT0 0x08
#define SPI_HDAT1 0x09
#define SPI_AIADDR 0x0a
#define SPI_VOL 0x0b
#define SPI_AICTRL0 0x0c
#define SPI_AICTRL1 0x0d
#define SPI_AICTRL2 0x0e
#define SPI_AICTRL3 0x0f
#define VS_DQ PIN_P23 //DREQ
#define VS_RST PIN_P22 //RST
#define VS_XCS PIN_PC4 //XCS--SPI_CS0--IO10
#define VS_XDCS PIN_PC1 //XDCS--SPI_CS2--IO23
unsigned int VS_RD_Reg(unsigned char address);
void VS_WR_Data(unsigned long data);
void VS_WR_Data_Test(unsigned long data);
void VS_WR_Cmd(unsigned char address,unsigned int data);
void VS_Init(void);
void VS_HD_Reset(void);
void VS_Soft_Reset(void);
void VS_Ram_Test(void);
void VS_Sine_Test(void);
void Set1053B(void);
unsigned int GetDecodeTime(void);
void ResetDecodeTime(void);
unsigned int GetHeadInfo(void);
#endif
VS1053B.cpp,作为VS1053B.h的功能实现函数库
#include "VS1053B.h"
#include <spi.h>
//#include <rxduino.h>
unsigned char VS1053Bram[5]={2,10,2,6,250};
unsigned long SPItemp;
const unsigned int bitrate[2][16]=
{
{0,8,16,24,32,40,48,56,64,80,96,112,128,144,160,0},
{0,32,40,48,56,64,80,96,112,128,160,192,224,256,320,0}
};
unsigned int GetHeadInfo(void)
{
unsigned int HEAD0;
unsigned int HEAD1;
HEAD0=VS_RD_Reg(SPI_HDAT0);
HEAD1=VS_RD_Reg(SPI_HDAT1);
//printf("(H0,H1):%x,%x\n",HEAD0,HEAD1);
switch(HEAD1)
{
case 0x7665:return 0;//WAV格式
case 0X4D54:return 1;//MIDI格式
case 0X574D://WMA格式
{
HEAD1=HEAD0*2/25;
if((HEAD1%10)>5)return HEAD1/10+1;
else return HEAD1/10;
}
default://MP3格式
{
HEAD1>>=3;
HEAD1=HEAD1&0x03;
if(HEAD1==3)HEAD1=1;
else HEAD1=0;
return bitrate[HEAD1][HEAD0>>12];
}
}
}
void VS_WR_Cmd(unsigned char address,unsigned int data)
{
unsigned long WRCmdreg;
while(digitalRead(VS_DQ)==0);
SPI.setClockDivider(SPI_CLOCK_DIV128);
SPI.port=SPI_PORT_CS0_DUINO;
//SPI.setClockDivider(SPI_CLOCK_DIV64);
WRCmdreg=(0x02<<24)+(address<<16)+data;
SPItemp=SPI.transfer(WRCmdreg);
}
unsigned int VS_RD_Reg(unsigned char address)
{
unsigned long RDreg;
unsigned int RDregreturn;
while(digitalRead(VS_DQ)==0);
SPI.setClockDivider(SPI_CLOCK_DIV128);
SPI.port=SPI_PORT_CS0_DUINO;
RDreg=(0x03<<24)+(address<<16);
SPItemp=SPI.transfer(RDreg);
RDregreturn = (unsigned int) SPItemp;
return RDregreturn;
}
unsigned int GetDecodeTime(void)
{
return VS_RD_Reg(SPI_DECODE_TIME);
}
void ResetDecodeTime(void)
{
VS_WR_Cmd(SPI_DECODE_TIME,0x0000);
VS_WR_Cmd(SPI_DECODE_TIME,0x0000);
}
void VS_WR_Data(unsigned long data)
{
while(digitalRead(VS_DQ)==0);
SPI.setClockDivider(SPI_CLOCK_DIV64);
SPI.port=SPI_PORT_CS2_MARY2;
SPItemp=SPI.transfer(data);
}
void VS_WR_Data_Test(unsigned long data)
{
while(digitalRead(VS_DQ)==0);
SPI.setClockDivider(SPI_CLOCK_DIV128);
SPI.port=SPI_PORT_CS2_MARY2;
SPItemp=SPI.transfer(data);
}
void VS_Sine_Test(void)
{
VS_HD_Reset();
VS_WR_Cmd(SPI_VOL,0X0505);
VS_WR_Cmd(SPI_MODE,0x0820);
while(VS_DQ== 0);
VS_WR_Data_Test(0x53ef6e2400000000);
delay(500);
VS_WR_Data_Test(0x4578697400000000);
delay(500);
VS_WR_Data_Test(0x53ef6e4400000000);
delay(500);
VS_WR_Data_Test(0x4578697400000000);
delay(500);
}
void VS_Ram_Test(void)
{
unsigned int regvalue;
VS_HD_Reset();
VS_WR_Cmd(SPI_MODE,0x0820);
while (VS_DQ==0); // 等待DREQ为高
VS_WR_Data_Test(0x4dea6d5400000000);
delay(500);
regvalue=VS_RD_Reg(SPI_HDAT0);
Serial.println(regvalue, BIN); //output the result in HEX
delay(500);
regvalue=VS_RD_Reg(SPI_MODE);
Serial.println(regvalue, BIN); //output the result in HEX
}
void VS_Init(void)
{
pinMode(VS_XCS,OUTPUT);
pinMode(VS_RST,OUTPUT);
pinMode(VS_DQ,INPUT);
digitalWrite(VS_RST,0);
delay(500);
digitalWrite(VS_RST,1);
delay(500);
SPI.begin();
//Serial.begin(9600); //set baudrate 9600bps
SPI.port=SPI_PORT_CS0_DUINO;
SPI.setBitOrder(MSBFIRST);
SPI.setBitLength(32);
SPI.setClockDivider(SPI_CLOCK_DIV128);
SPI.setDataMode(SPI_MODE0);
}
void VS_Soft_Reset(void)
{
while(digitalRead(VS_DQ)==0);
SPI.setClockDivider(SPI_CLOCK_DIV128);
SPI.port=SPI_PORT_CS0_DUINO;
VS_WR_Cmd(SPI_MODE,0X0804);
delay(3);
while(digitalRead(VS_DQ)==0);
VS_WR_Cmd(SPI_CLOCKF,0x7800);
VS_WR_Cmd(SPI_AUDATA,0xBB81);
}
void VS_HD_Reset(void)
{
digitalWrite(2,0);
delay(2000);
digitalWrite(VS_XDCS,1);
digitalWrite(VS_XCS,1);
digitalWrite(VS_RST,1);
}
void Set1053B(void)
{
unsigned char t;
unsigned int bass=0;
unsigned int volt=0;
unsigned char vset=0;
vset=255-VS1053Bram[4];
volt=vset;
volt<<=8;
volt+=vset;
//0,henh.1,hfreq.2,lenh.3,lfreq
for(t=0;t<4;t++)
{
bass<<=4;
bass+=VS1053Bram[t];
}
VS_WR_Cmd(SPI_BASS,bass);//BASS
//delay(50);
VS_WR_Cmd(SPI_VOL,volt);
}
gr_sketch.cpp,主函数,Demo示例程序,测试功能用
#include "VS1053B.h"
//#include <spi.h>
unsigned int res;
void setup()
{
pinMode(PIN_LED0,OUTPUT);
VS_Init();
Serial.begin(9600); //set baudrate 9600bps
}
void loop()
{
digitalWrite(PIN_LED0,1); //set led0 on
while(!Serial.available()); //wait command from upper computer
char c=Serial.read();
if(c=='r') //if receive command 'r'
{
VS_HD_Reset();
VS_Soft_Reset();
Set1053B();
res= VS_RD_Reg(SPI_MODE);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_STATUS);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_BASS);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_CLOCKF);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_DECODE_TIME);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_AUDATA);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_WRAM);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_WRAMADDR);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_HDAT0);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_HDAT1);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_AIADDR);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_VOL);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_AICTRL0);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_AICTRL1);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_AICTRL2);
Serial.println(res, BIN); //output the result in BIN
delay(500);
res= VS_RD_Reg(SPI_AICTRL3);
Serial.println(res, BIN); //output the result in BIN
delay(500);
VS_Sine_Test();
VS_Ram_Test();
}
else if(c=='t')
{
VS_HD_Reset();
}
else
{
Serial.println("Please send the right SPI command!");
}
digitalWrite(PIN_LED0,0); //turn off the led0
}
最近,实习公司那边要我们实习生参加年会节目表演,业余时间就要用来练习交谊舞,这不忙里偷闲,整理出一篇文档:“Ardiuno开发板上的mini-SD卡读写”
文中首先介绍了SAKURA开发板上的mini-SD卡读写所用的库函数;其次,介绍了对SD卡进行操作所必须的一些库函数:例如:begin open read write close等;最后,給出一个对SD卡读写的例子。
文档下载链接:
http://share.eepw.com.cn/share/download/id/83421
由于文档较大,就不直接贴在帖子里,帖子留着让大家给我指点迷津,相互交流之用吧!
对于准备播放MP3格式的音乐来说,解析MP3格式的声音文件是必须的,在参考了“MP3文件解析”一文后得知了传说中的MP3文件的帧头格式:
帧头长4字节,对于固定位率的MP3文件,所有帧的帧头格式一样其数据结构如下:
为了更详细地说明MP3文件的格式,如下是在VC++中打开一个名为test.mp3文件,其内容如下:
000000 FF FB 52 8C 00 00 01 49 09 C5 05 24 60 00 2A C1
000010 19 40 A6 00 00 05 96 41 34 18 20 80 08 26 48 29
000020 83 04 00 01 61 41 40 50 10 04 00 C1 21 41 50 64
……
0000D0 FE FF FB 52 8C 11 80 01 EE 90 65 6E 08 20 02 30
0000E0 32 0C CD C0 04 00 46 16 41 89 B8 01 00 08 36 48
0000F0 33 B7 00 00 01 02 FF FF FF F4 E1 2F FF FF FF FF
……
0001A0 DF FF FF FB 52 8C 12 00 01 FE 90 58 6E 09 A0 02
0001B0 33 B0 CA 85 E1 50 01 45 F6 19 61 BC 26 80 28 7C
0001C0 05 AC B4 20 28 94 FF FF FF FF FF FF FF FF FF FF
……
001390 7F FF FF FF FD 4E 00 54 41 47 54 45 53 54 00 00
0013A0 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
……
0013F0 00 00 00 00 04 19 14 03 00 00 00 00 00 00 00 00
001400 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00 00
001410 00 00 00 00 00 00 4E
该文件长度1416H(5.142K),帧头为:FF FB 52 8C,转换成二进制为:
11111111 11111011
01010010 10001100
对照MP3帧头帧格式可知,test.mp3帧头信息见下表:
第1397H开始的三个字节是54 41 47,存放的是字符“TAG”,表示此文件有ID3 V1.0信息。
139AH开始的30个字节存放歌名,前4个非00字节是54 45 53 54,表示“TEST”;
13F4H开始的4个字节是04 19 14 03,存放年份“04/25/2003”;
最后1个字节是4E,表示音乐类别,代号为78,即“Rock&Roll”;
其它字节均为00,未存储信息。
在了解了mp3文件的格式之后,将逐步在SAKURA开发板上实现对mini-SD卡上的mp3文件数据的读取。。。
回复
有奖活动 | |
---|---|
【有奖活动——B站互动赢积分】活动开启啦! | |
【有奖活动】分享技术经验,兑换京东卡 | |
话不多说,快进群! | |
请大声喊出:我要开发板! | |
【有奖活动】EEPW网站征稿正在进行时,欢迎踊跃投稿啦 | |
奖!发布技术笔记,技术评测贴换取您心仪的礼品 | |
打赏了!打赏了!打赏了! |