mini2440裸机音乐播放器(非常久曾经的笔记)
【这是好久曾经写的。有点乱,没时间整理。当做记录用的。】
图片粘贴失效。没上传图,想要的直接下载文档吧。
项目目的:通过IIS,触摸屏,LCD模块实现音乐播放器功能(button上一首、下一首、播放、暂停的音频控制功能,并实现播放歌曲时显示对应的歌曲图片,不播放时显示hello music图片)
项目设备:windows7(32位),mini2440。uboot(nandflash)。ADS1.2开发环境,jlink v8。耳机;
功能模块LCD、触摸屏、IIS 分别介绍:
Main函数中主频设置:
U32 mpll_val = 0; mpll_val = (92<<12)|(1<<4)|(1); //init FCLK=400M, ChangeMPllValue((mpll_val>>12)&0xff, (mpll_val>>4)&0x3f, mpll_val&3); ChangeClockDivider(14, 12); //the result of rCLKDIVN [0:1:0:1] 3-0 bit cal_cpu_bus_clk(); //HCLK=100M PCLK=50M |
一、LCD(显示对应的图片)
我的LCD是统宝3.5 TFT屏,就是类似由240 x 320个会产生不同颜色的LED组成的。我的目的就是让这些点产生与图形对应的颜色。
LCD彩色图片转换工具转换bmp图片过程:将图片的第一个点转换为两个常量。然后一个点一个点转换就转换完一行。一行一行就完毕一张图片(一帧)的数组数据转换。所产生的产量就是我们要的bmp图片数组了。这个转换和LCD显示时序是同样的。
LCD的时序显示图解:
从图中也可知道整个屏幕的点亮流程:是从一行的第一个点阵点亮,再一行里一点挨一点点亮,点亮完一行跳转下一行点亮,一行一行点亮完后。就完毕了一张图片在LCD里的显示。
1、绘图函数主要分析(原型见附录2):
c = bmp[p+1] | (bmp[p]<<8) ; // bmp数组是8位char类型,我採用16bpp[ 5 : 6 : 5 ]来显示,超过8位
LCD_BUFFER[y0+y][x0+x]= c ; // LCD_BUFFER数组存放图片显示数据
p= p + 2 ;
2、调用图片函数实參分析:
Pait_Bmp( X0, Y0, X1, Y1, hellomusic);
X0, Y0各自是图片显示的LCD横、纵坐标,
X1。 Y1各自是图片横、纵向的点阵数。 hellomusic是一个图片数组的首地址
3、将LCD_BUFFER图片数组写入LCD帧缓冲数据寄存器
#define M5D(n) ((n)&0x1fffff) #define LCD_ADDR ((U32)LCD_BUFFER) //宏定义LCD_ADDR为LCD_BUFFER图片数组首地址 // rLCDSADDR1中[20:0]存储的是LCD_ADDR图片数组的起始位置。但在LCD_ADDR中是[21:1],所以採用(M5D(LCD_ADDR>> 1) rLCDSADDR1 = ((LCD_ADDR>> 22) << 21) | (M5D(LCD_ADDR>> 1)); // rLCDSADDR2中[20:0]存储的是LCD_ADDR图片数组的结束位置 rLCDSADDR2 = M5D((LCD_ADDR+ LCD_WIDTH |
4、LCD寄存器初始化:
/************************************************************** TFT LCD功能模块初始化//此函数没开启屏蔽功能 **************************************************************/ void LCD_Init(void) { #define M5D(n) ((n)&0x1fffff) #define LCD_ADDR ((U32)LCD_BUFFER) //宏定义LCD_ADDR为LCD_BUFFER图片数组首地址 //Con1 控制寄存器命令在以下解释 //Con2 //Con3 //Con4 //Con5 rLCDSADDR1 = ((LCD_ADDR>> 22) << 21) | ((M5D(LCD_ADDR>> 1)) << 0); rLCDSADDR2 = M5D((LCD_ADDR+ LCD_WIDTH rLCDSADDR3 = LCD_WIDTH; rLCDINTMSK |= 3; //中断屏蔽 rTCONSEL &= (~7); rTPAL = 0x0; rTCONSEL &= ~((1<<4) | 1); } |
Con1:
#define LCD_PIXCLOCK 4 //con1[17:8]CLKVAL设置为4,依据TFT:VCLK= HCLK / [(CLKVAL + 1) × 2] 。VCLK=10MHz,VCLK就是显示速率 [6:5]显示模式选择为 TFT LCD面板 模式 [4:1]选择为 TFT的16bpp 像素模式 rLCDCON1 = (LCD_PIXCLOCK << 8) | (3 << 5) | (12 << 1); |
Con2
#define LCD_UPPER_MARGIN 1 #define LCD_HEIGHT 320 #define LCD_LOWER_MARGIN 4 //con2[23:14]设置高度320 VSPW[5:0]决定对应脉冲的宽度; VFPD[13:6]、VBPD[31:24]决定了延时时间; rLCDCON2 |
Con3
#define LCD_RIGHT_MARGIN 25 #define LCD_WIDTH 240 #define LCD_LEFT_MARGIN 0 ,[18:8]设置高度240。HFPD、HBPD决定了延时时间; rLCDCON3 = (LCD_RIGHT_MARGIN << 19) | ((LCD_WIDTH - 1) << 8) | (LCD_LEFT_MARGIN << 0); |
Con4
#define LCD_HSYNC_LEN 4 //con4 HSPW[7:0]决定对应脉冲的宽度; rLCDCON4 = (13 |
Con5
//con5[11] 5:6:5输出格式,[9]、[8]脉冲翻转,[3]同意PWREN信号,[0]半字节交换使能; rLCDCON5 = ((1<<11) | (1 << 9) | (1 << 8) | (1 << 3) | (1 << 0)); |
5、LCD视频输出控制函数与开启电源函数
/************************************************************** LCD视频和控制信号输出或者停止,1开启视频输出 **************************************************************/ void Lcd_EnvidOnOff(int onoff){ ) rLCDCON1;// ENVID=ON else rLCDCON1 =rLCDCON1& } /************************************************************** )就开启了LCD电源 **************************************************************/ void Lcd_PowerEnable(int invpwren,int rGPGUP );//禁止上拉使能 rGPGCON );//GPG4=LCD_PWREN //开启LCD电源 rLCDCON5 ); // rLCDCON5 ); // } |
LCD总结:
通过调用上面的函数:
LCD_Init(); // TFT LCD功能模块初始化
Lcd_PowerEnable); //
TFT LCD 电源控制引脚使能
Lcd_EnvidOnOff开启视频输出
就能够使用Pait_Bmp( X0, Y0, X1, Y1, hellomusic);来实现LCD显示图片了
二、触摸屏(点击屏幕时产生中断)
中断是当硬件支持中断时。设置对应的控制寄存器开启中断。其中断发生时CPU跳转到中断程序完毕中断操作。以提高操作效率。
这个解释非常短,可是在项目试验中才发现中断真的非常奇妙,并且非常省事。不用再写别的逻辑函数以实现同样的功能,也不会有这么高的处理效率。并且写出来的程序也会变非常大。
触摸屏就支持中断操作。
1、寄存器设置
当使用触摸屏功能时,需将XP,XM,YP,YM设置为触摸屏引脚使能,寄存器设置为:
void Touch_Init(void) { rADCCON)); //A/D分频时钟有效,其值为9 rADCTSC=0xd3; //光标按下中断信号,YM有效。YP无效,XM有效。XP无效。XP上拉电阻,普通ADC转换。等待中断模式 rADCDLY;//正常转换模式转换延时大约为(1/3.6864M)*50000=13.56ms rINTSUBMSK ); //TC中断使能 rINTMSK ); //ADC总中断使能 pISR_ADC=(U32)Adc_Tc_Handler; //指向中断向量表 } |
二、中断处理函数
触摸屏中断是有sub寄存器的。通过图14-1能够看出中断发生是:
中断请求中断服务>>次级源中断挂起>>SRCPND>>仲裁>>INTPND>>irq中断>>中断函数入口
中断函数:
/************************************************************** TFT LCD *触摸屏中断函数* **************************************************************/ static { rADCTSC);//XP上拉电阻无效,自己主动连续測量X坐标和Y坐标. rADCCON);//ADC转换開始 ));//检測ADC转换是否開始且ADCCON[0]自己主动清0 )));//检測ADCCON[15]是否为1,ADC转换是否结束,(必须) )));//检測ADC中断是否已请求 //这里就是将转换的坐标值存储在 xdata、ydata。 xdata=rADCDAT0&0x3ff;//读x坐标 ydata=rADCDAT1&0x3ff;//读y坐标 rSUBSRCPND); //清除中断 rSRCPND); rINTPND); rADCTSC =0xd3; //ADC等待中断模式 rADCTSC); //ADCTSC[8]=1,设置抬起中断信号 ))); //检測触屏抬起中断是否已请求 rADCTSC );//ADCTSC[8]=0光标按下中断信号 rSUBSRCPND); rSRCPND); rINTPND); } |
触摸屏总结:在开启LCD电源时 Lcd_PowerEnable); //
TFT LCD 电源控制引脚使能
初始化触摸屏中断 Touch_Init();
每次中断产生的坐标值都会存储在xdata、ydata;
三、IIS(实现将音频文件按一定频率传输)
1、IIS预分频值的确定
IIS我设置DMA无效,FIFO输出模式,设置了预分频值,保证传输的速率和採样频率一样。
IISSCLK为串行时钟,每个时钟信号传送一位音频信号。因此IISSCLK的频率=声道数×採样频率×採样位数。如採样频率fs为44.1kHz。採样的位数为16位。声道数2个(左、右两个声道),则IISSCLK的频率=32fs=1411.2kHz。
1、Fs 2、採样的位数 3、声道数
使用 wav数据提取器 查看
採样频率
右键音乐文件à属性à具体信息
PCLK(50MHZ)经过两个预分频器处理后分别得到IISSCLK、IISLRCK和CDCLK,寄存器IISPSR是IIS预分频器寄存器,5~9位是预分频器A,0~4位是预分频器B。一般来说。这两个预分频器的值N相等。即仅仅要知道一个,还有一个也就知道,而这里我们是通过CDCLK来计算预分频器B的值N的,即CDCLK=PCLK
/ (N+1)。
我设置主时钟频率( CODECLK )选择384fs,由下表。因fs=44.1kHZ,CODECLK=16.9344。
可知N= PCLK(50MHZ)/ CODECLK(16.9344MHZ) -1
= 2 ;
|
IIS初始化
/************************************************************** IIS初始化 **************************************************************/ void iis_init(void){ //配置IIS接口 rGPEUP = rGPEUP& rGPECON = rGPECON& //配置s3c2440的IIS寄存器 ,所以CDCLK=PCLK/(2+1)=16.66666kHz rIISPSR ; //无效DMA,输入空暇。预分频器有效。开启IIS rIISCON = 位,CODECLK=384fs。SCLK=32fs rIISMOD = rIISFCON = } |
UDA1341
在我看来。IIS就是实现将音频(WAV)数据按一定频率发送一定大小的数据给FIFO或者DMA,UDA1341就是将这些音频数据转换成电信号通过音频接口发送出去。而关于uda1341的配置是參考来的
//通过io口模拟L3总线写数据 为数据模式 //关于地址模式和数据模式以及传输时序注意參考数据手冊 static { int i,j; if(address== rGPBDAT = rGPBDAT&(~(L3D| L3M } rGPBDAT = rGPBDAT& } Delay); //数据传输 ;i++) { if(data& { rGPBDAT &=~L3C; //L3C=L rGPBDAT |= L3D; //L3D=H ;j++) ; //等待一段时间 rGPBDAT |= L3C; //L3C=H rGPBDAT |= L3D; //L3D=H ;j++) ; //等待一段时间 } else // L { rGPBDAT &=~L3C; //L3C=L rGPBDAT &=~L3D; //L3D=L ;j++) ; //等待一段时间 rGPBDAT |= L3C; //L3C=H rGPBDAT &=~L3D; //L3D=L ;j++) ; //等待一段时间 } data ; } rGPBDAT = rGPBDAT& } |
UDA1341初始化
//UDA1341初始化 //配置L3接口总线。GPB2:L3MODE, GPB3:L3DATA, GPB4:L3CLOCK rGPBCON = rGPBUP = rGPBDAT = rGPBDAT = rGPBDAT& 页中的操作顺序。首先在地址模式下。 //选择操作地址000101xx +10(STATUS)=0X16 write_UA1341); write_UA1341);// write_UA1341); write_UA1341); //0,0,01, write_UA1341); write_UA1341); //1,0,0,0, //Gain of DAC 6 dB,Gain of ADC 0dB,ADC non-inverting, //DAC non-inverting,Single speed playback,ADC-Off DAC-On |
IIS总结:
初始化IIS后 iis_init(); // IIS初始化
UDA1341_init();
就能够通过下面语句实现歌曲的播放
下面语句是在while循环下 buffer=music1; //初始化buffer指向music1音乐数组地址 if((rIISCON& { 位数据 ;i++){ rIISFIFO位数据于FIFO } count次循环。每次循环指向两个不同字节的字节总和 |
四、界面素材
第一排图片为button图片,较暗的那张为点击后短暂显示的图片
背景图:
五、WAV音频文件的制作
1、 网上下载无损音乐(我下载了WAV格式和APE格式的)
2、 打开格式工厂à音频à->WAVà加入文件à截取片段(无损音乐太大了,须要截取一小部分。20秒一首就能够了)à确定à输出配置(例如以下图)à确定à点击開始。
3、 打开截取WAV目录
4、 使用DataToHex将WAV文件下的WAV转换成音频数据数组文件。改动数组文件名称Array[],
5、 将wav音频数据文件改为c文件。请不要改为h,不然debug要好久。
6、 制作完毕!
主函数实现功能:
/************************************************************** music *子main函数* **************************************************************/ void music_player(void) { static U32 save_A, save_C, save_CP, save_D, save_E, save_EP, save_B, save_LCD1, save_LCD2, save_SR, save_S, save_SM, save_SF, save_LCDINT, //**********************保护现场************************** save_SR = rIISPSR; save_S = rIISCON; save_SM = rIISMOD; save_SF = rIISFCON; save_A = rADCCON; save_B = rGPBCON; save_BD = rGPBDAT; save_EP = rGPEUP; save_E = rGPECON; save_G = rGPGCON; save_GP = rGPGUP; save_C = rGPCCON; save_CP = rGPCUP; save_D = rGPDCON; save_DP = rGPDUP; save_LCD1 = rLCDCON1; save_LCD2 = rLCDCON2; save_LCD3 = rLCDCON3; save_LCD4 = rLCDCON4; save_LCD5 = rLCDCON5; save_LCDINT = rLCDINTMSK; save_INTS = rINTSUBMSK; save_INT = rINTMSK; //**********************开启LCD显示功能********************************** Lcd_Port_Init();//port初始化 LCD_Init(); //TFT LCD功能模块初始化 Lcd_PowerEnable); // Lcd_EnvidOnOff开启视频输出 //******************************************************** 显示图片界面 //******************************************************** Pait_Bmp, music_interface); Pait_Bmp, hellomusic); Pait_Bmp, button2); //上一首 Pait_Bmp, Pait_Bmp, button4); //播放 //******************触摸屏中断初始化,IIS数据流设置************ Touch_Init(); //开触摸屏中断 iis_init(); // IIS初始化 UDA1341_init(); //UDA1341初始化 buffer=music1; //初始化buffer指向music1音乐数组地址 Delay); Uart_Printf("\n\nq:退出\n"); while(Uart_GetKey()!= { // button1是 播放 图片函数 ,button1_1是 播放 阴影图片函数。 // button2是 上一首 图片函数 ,button2_2是 上一首 阴影图片函数。 // button3是 下一首 图片函数 ,button3_3是 下一首 阴影图片函数; // button4是 暂停 图片函数 。button4_4是 暂停 阴影图片函数。 // flag 为 1 播放状态 。flag 为 0 暂停状态 ; // song_num 歌曲序号标志。song_num1 歌曲跳转标志,仅仅有按上一首、下一首song_num1才不为0,一般为0。 ){ rIISCON|= //******************播放对应歌曲显示对应的图片 ************ , , , //***************新一首歌曲处理语句,song_num1一般为0,当为详细数时就为新一首歌曲序号 //*************** buffer音乐播放地址赋初值 length音乐长度又一次赋值 并显示音乐图片 //*************** 下一FIFO字节位置count初始化 ){ ){ ){ //****************播放音频数据语句*************************** if((rIISCON&
位数据 ;i++){ rIISFIFO); // } count次循环x每次循环指向两个不同字节)的字节总和 //**************音乐播放完毕后 音乐标志song_num1指向下一首歌 if(count>length){ song_num2; ;} song_num1=song_num2; } } } //**************暂停时关闭IIS 显示欢迎图片 ){ } Lcd_ClearScr( //**********************还原现场************************** rIISPSR = save_SR; rIISCON = save_S; rIISMOD = save_SM; rIISFCON = save_SF; rADCCON = save_A; rGPBCON = save_B; rGPBDAT = save_BD; rGPEUP = save_EP; rGPECON = save_E; rGPGCON = save_G; rGPGUP = save_GP; rGPCCON = save_C; rGPCUP = save_CP; rGPDCON = save_D; rGPDUP = save_DP; rLCDCON1 = save_LCD1; rLCDCON2 = save_LCD2; rLCDCON3 = save_LCD3; rLCDCON4 = save_LCD4; rLCDCON5 = save_LCD5; rLCDINTMSK = save_LCDINT; rINTSUBMSK = save_INTS; rINTMSK = save_INT; //******************************************************** } |
遇到的问题与解决
做这个项目的时候遇到过非常多问题。LCD是怎么点亮的。颜色是怎么产生的?触摸屏的是怎么实现存储我的坐标地址的?还有IIS的FIFO传输问题问题,IIS的音频文件是什么格式的?怎样用代码设计出自己想要的功能?
1、当时音频文件困扰我好久,使用友善之臂提供的音频文件能够播放,可是自己制作的却不行,网上也没找到原因。mp3、wav这些常见的格式音乐一个一个试。都不行,结果发现如今的压缩技术已经将这些音频文件压缩的非常小。方便存储并且不失真。可是这些都不是这个项目须要的无损音频文件,找到无损音频文件后,才把我的音频文件问题攻克了;
2、而关于IIS的FIFO传输,FIFO一个数据块是有64个字节的。当时我使用过分64次循环,一次传输一个字节出来,结果是沙哑的音质,后面再将FIFO设置成32次,一次传输两个字节,结果奇迹般的放出了自己爱听的歌曲,一次传输两个字节,也就是实现双声道的效果吧
3、还记得当时有过LCD显示图片移位的现象,这是由于自己对 Pait_Bmp(intx0,int y0,int h,int l,const unsigned char *bmp)函数里的參数没设置好
以上都是项目中出现的一些比較困扰的问题,平时还会遇到非常多调试问题,这些问题一般博客里都会有人做记录的,自己也就能够学习和解决。可是当遇到一些中断、地址映射、系统载入过程、nandflash和norflash差别、uboot烧写到nandflash后裸机怎么执行的、GPBDAT写入控制命令怎么能控制模块等等问题时,就须要多次使用和执行后才明确。
1、中断:经过多次中断实验最终明确了中断时。CPU的跳转与所带来的方便。
2、地址映射:内存就这么大。为什么载入一块大的存储器时还能可以读取到那么高的地址?这就是地址映射强大;
3、系统载入:裸机下的2440init.S文件,写驱动时须要烧写系统时的bootloader。这些都是为了初始化CPU、
硬件平台,2440init.S还为裸机提供物理上的引脚定义(初始化)。bootloader则还还提供Linux所需的接口,载入完Linux内核,之后就是文件系统了
裸机:2440init.S --àMain函数--à 详细函数功能
Linux:bootloader --àlinux内核 à 文件系统
4、nandflash和norflash差别:norflash里面的数据是能够直接运行的,可是容量小;nandflash容量更大。可是须要前4k映射片选空间。
5、uboot烧写到nandflash后裸机的执行。 nandflash内的文件不能直接执行。要将前4k映射片选空间才可使用。并且uboot烧写到nandflash后,执行裸机文件并非在SDRAM的起始位置,被uboot先使用了SDRAM前面的空间。而在编译器下设置的裸机的SDRAM起始位置为0x30000000,所以须要将裸机执行的起始地址在编译器里初始化,初始化文件见附录1,不然裸机会跑飞。
6、而关于GPBDAT写入控制命令能控制模块,是由于这个宏定义:#define GPBDAT (*(volatile unsigned long*)0x56000014),GPBDAT宏定义为0x56000014寄存器地址内的值,而寄存器地址的初始化是在头文件2440init.S文件
非常多东西都是自己做项目再做总结, 总结中可能会发现新问题或者找到以前迷惑的问题,可是寻常遇到的问题主要还是通过查博客解决这个问题,做项目、看文档、查手冊、问同学老师、读博客、写博客。这是我学习进步的方法。我也会坚持下去的。
————————————————————项目介绍完—————————————————————
附录
1、
Setmem 0x53000000 0x00000000 32
Setmem 0x4A000008 0xFFFFFFFF 32
Setmem 0x4A00001C 0x000007FF 32
Setmem 0x53000000 0x00000000 32
Setmem 0x56000050 0x000055AA 32
Setmem 0x4C000014 0x00000007 32
Setmem 0x4C000000 0x00FFFFFF 32
Setmem 0x4C000004 0x00061012 32
Setmem 0x4C000008 0x00040042 32
Setmem 0x48000000 0x22111120 32
Setmem 0x48000004 0x00002F50 32
Setmem 0x48000008 0x00000700 32
Setmem 0x4800000C 0x00000700 32
Setmem 0x48000010 0x00000700 32
Setmem 0x48000014 0x00000700 32
Setmem 0x48000018 0x0007FFFC 32
Setmem 0x4800001C 0x00018005 32
Setmem 0x48000020 0x00018005 32
Setmem 0x48000024 0x008E0459 32
Setmem 0x48000028 0x00000032 32
Setmem 0x4800002C 0x00000030 32
Setmem 0x48000030 0x00000030 32
2、
/************************************************************** TFT LCD *在屏幕上绘图* **************************************************************/ void Pait_Bmp(int x0,int y0,int h,int l,const unsigned char *bmp) { int x,y; U32 c; int p = 0; for( y = 0 ; y < l ; y++ ) { for( x = 0 ; x < h ; x++ ) { c = bmp[p+1] | (bmp[p]<<8) ; if ( ( (x0+x) < SCR_XSIZE) && ( (y0+y) < SCR_YSIZE) ) LCD_BUFFER[y0+y][x0+x] = c ; p = p + 2 ; } } } |
3、项目整个代码
/*********************************** 实现功能 music播放器 ***********************************/ #include "2440lib.h" #include "2440slib.h" #include "LCD_init.h" #include "2440addr.h" #define L3C (1<<4) #define L3D (1<<3) #define L3M (1<<2) #define rIISFIFO (*(volatile unsigned long*)0x55000010) extern extern extern extern extern extern extern extern extern extern extern extern extern extern extern extern //xdata, ydata用于存储屏幕坐标(不是屏幕像素点阵) volatile volatile volatile volatile /************************************************************** 键值处理函数 **************************************************************/ static // button1是 播放 图片函数 ,button1_1是 播放 阴影图片函数; // button2是 上一首 图片函数 。button2_2是 上一首 阴影图片函数。 // button3是 下一首 图片函数 ,button3_3是 下一首 阴影图片函数; // button4是 暂停 图片函数 。button4_4是 暂停 阴影图片函数; // flag 为 1 播放状态 ,flag 为 0 暂停状态 ; // song_num 歌曲序号标志。song_num1 歌曲跳转标志。仅仅有按上一首、下一首song_num1才不为0,一般为0; if( xdata>= ); ){ Pait_Bmp, Pait_Bmp, if( xdata>= );
){ Pait_Bmp, Pait_Bmp, if( xdata>= ); //下一段非常凑巧,当点击(上一首/下一首) 后。 //刚好实现了我的功能 ){botten1= Pait_Bmp, else Pait_Bmp, } } /************************************************************** TFT LCD *触摸屏中断函数* **************************************************************/ static { rADCTSC);//XP上拉电阻无效,自己主动连续測量X坐标和Y坐标. rADCCON);//ADC转换開始 ));//检測ADC转换是否開始且ADCCON[0]自己主动清0 )));//检測ADCCON[15]是否为1,ADC转换是否结束,(必须) )));//检測ADC中断是否已请求 xdata=rADCDAT0&0x3ff;//读x坐标 ydata=rADCDAT1&0x3ff;//读y坐标 //Beep(2000, 100); Button_Handle(); //button处理函数 Uart_Printf("\n Xdata=%04d, Ydata=%04d\n", xdata, Uart_Printf("\n flag=%d, 1:播放 0:暂停", Uart_Printf("\n botten1=%d, 1:上一首 2:下一首",botten1); Uart_Printf("\n **正在播放第%d首歌曲** rSUBSRCPND); //清除中断 rSRCPND); rINTPND); rADCTSC =0xd3; //ADC等待中断模式 rADCTSC); //ADCTSC[8]=1,设置抬起中断信号 ))); //检測触屏抬起中断是否已请求 rADCTSC );//ADCTSC[8]=0光标按下中断信号 // 因为以下这段代码和上面这段代码是看到大神们都会加上去的,调试得到 // 结论:上面的清除中断实现触屏中断,而以下这段则是加快中断后的响应 // (我的实验结果是:没以下的时候。触屏后的数值显示时间变长,蜂鸣器的 // 响声也变长,有种慢一拍的感觉) rSUBSRCPND); rSRCPND); rINTPND); } static { rADCCON)); //A/D分频时钟有效,其值为9 rADCTSC=0xd3; //光标按下中断信号,YM有效,YP无效,XM有效。XP无效。XP上拉电阻。普通ADC转换,等待中断模式 rADCDLY;//正常转换模式转换延时大约为(1/3.6864M)*50000=13.56ms rINTSUBMSK );//TC中断使能 rINTMSK );//ADC总中断使能 pISR_ADC=(U32)Adc_Tc_Handler;//指向中断向量表 } /************************************************************** IIS初始化 **************************************************************/ void iis_init(void){ //配置IIS接口 rGPEUP = rGPEUP& rGPECON = rGPECON& //配置s3c2440的IIS寄存器 ,所以CDCLK=PCLK/(2+1)=16.66666kHz rIISPSR ; //无效DMA,输入空暇,预分频器有效。开启IIS rIISCON = 位。CODECLK=384fs,SCLK=32fs rIISMOD = rIISFCON = } /************************************************************** UA1341函数 **************************************************************/ //通过io口模拟L3总线写数据 为数据模式 //关于地址模式和数据模式以及传输时序注意參考数据手冊 static { int i,j; if(address== { rGPBDAT = rGPBDAT&(~(L3D| L3M } rGPBDAT = rGPBDAT& } Delay); //数据传输 ;i++) { if(data& { rGPBDAT &=~L3C; //L3C=L rGPBDAT |= L3D; //L3D=H ;j++) ; //等待一段时间 rGPBDAT |= L3C; //L3C=H rGPBDAT |= L3D; //L3D=H ;j++) ; //等待一段时间 } else // L { rGPBDAT &=~L3C; //L3C=L rGPBDAT &=~L3D; //L3D=L ;j++) ; //等待一段时间 rGPBDAT |= L3C; //L3C=H rGPBDAT &=~L3D; //L3D=L ;j++) ; //等待一段时间 } data ; } rGPBDAT = rGPBDAT& } /************************************************************** UDA1341初始化 **************************************************************/ void UDA1341_init(void){ //UDA1341初始化 //配置L3接口总线。GPB2:L3MODE, GPB3:L3DATA, GPB4:L3CLOCK rGPBCON = rGPBUP = rGPBDAT = rGPBDAT = rGPBDAT& 页中的操作顺序。首先在地址模式下。 //选择操作地址000101xx +10(STATUS)=0X16 write_UA1341); write_UA1341);// write_UA1341); write_UA1341); //0,0,01, write_UA1341); write_UA1341); //1,0,0,0, //Gain of DAC 6 dB,Gain of ADC 0dB,ADC non-inverting, //DAC non-inverting,Single speed playback,ADC-Off DAC-On } /************************************************************** music *子main函数* **************************************************************/ void music_player(void) { static U32 save_A, save_C, save_CP, save_D, save_E, save_EP, save_B, save_LCD1, save_LCD2, save_SR, save_S, save_SM, save_SF, save_LCDINT, //**********************保护现场************************** save_SR = rIISPSR; save_S = rIISCON; save_SM = rIISMOD; save_SF = rIISFCON; save_A = rADCCON; save_B = rGPBCON; save_BD = rGPBDAT; save_EP = rGPEUP; save_E = rGPECON; save_G = rGPGCON; save_GP = rGPGUP; save_C = rGPCCON; save_CP = rGPCUP; save_D = rGPDCON; save_DP = rGPDUP; save_LCD1 = rLCDCON1; save_LCD2 = rLCDCON2; save_LCD3 = rLCDCON3; save_LCD4 = rLCDCON4; save_LCD5 = rLCDCON5; save_LCDINT = rLCDINTMSK; save_INTS = rINTSUBMSK; save_INT = rINTMSK; //******************************************************** Lcd_Port_Init();//port初始化 LCD_Init(); //TFT LCD功能模块初始化 Lcd_PowerEnable); // Lcd_EnvidOnOff开启视频输出 /*红(255:0:0);绿(0:255:0);蓝(0:0:255);黑(0:0:0);白(255,255,255)*/ /*在屏幕上显示三基色*/ Lcd_ClearScr)| Delay); Lcd_ClearScr)| Delay); Lcd_ClearScr)| Delay); Lcd_ClearScr( Delay); Pait_Bmp, music_interface); Pait_Bmp, hellomusic); Pait_Bmp, button2); //上一首 Pait_Bmp, Pait_Bmp, button4); //播放 Touch_Init(); //开触摸屏中断 iis_init(); // IIS初始化 UDA1341_init(); //UDA1341初始化 buffer=music1; //初始化buffer指向music1音乐数组地址 //while(1) Delay);Uart_Printf("\n\nq:退出\n"); while(Uart_GetKey()!= { ){ rIISCON|= //处理点击播放时音乐图片的显示(点击播放时 flag1=1) , , , // 音乐标志song_num1不为零时(歌曲播放完song_num1赋值或者通过键值(上一首/下一首)给song_num1赋值) // buffer音乐播放地址赋初值 length音乐长度又一次赋值并显示音乐图片 下一FIFO字节位置count初始化 ){ ){ ){ if((rIISCON&
位数据 ;i++){ rIISFIFO); // } count次循环。每次循环指向两个不同字节的字节总和 //音乐播放完毕后 音乐标志song_num1指向下一首歌 if(count>length){ song_num2; ;} song_num1=song_num2; } } } //暂停时关闭IIS 显示欢迎图片 ){ } Lcd_ClearScr( //**********************还原现场************************** rIISPSR = save_SR; rIISCON = save_S; rIISMOD = save_SM; rIISFCON = save_SF; rADCCON = save_A; rGPBCON = save_B; rGPBDAT = save_BD; rGPEUP = save_EP; rGPECON = save_E; rGPGCON = save_G; rGPGUP = save_GP; rGPCCON = save_C; rGPCUP = save_CP; rGPDCON = save_D; rGPDUP = save_DP; rLCDCON1 = save_LCD1; rLCDCON2 = save_LCD2; rLCDCON3 = save_LCD3; rLCDCON4 = save_LCD4; rLCDCON5 = save_LCD5; rLCDINTMSK = save_LCDINT; rINTSUBMSK = save_INTS; rINTMSK = save_INT; //******************************************************** } |
附录
wav工具下载地址
视频地址
mini2440裸机音乐播放器(非常久曾经的笔记)的更多相关文章
- Android应用--简、美音乐播放器增加音量控制
Android应用--简.美音乐播放器增加音量控制 2013年6月26日简.美音乐播放器继续完善中.. 题外话:上一篇博客是在6月11号发的,那篇博客似乎有点问题,可能是因为代码结构有点乱的原因,很难 ...
- QT5:C++实现基于multimedia的音乐播放器(二)
今天接着上一篇来实现播放器的槽函数. 先来实现播放模式,槽函数如下: //播放模式 void Music::musicPlayPattern() { //z=++z%3; ) { //顺序播放 pla ...
- iOS之基于FreeStreamer的简单音乐播放器(模仿QQ音乐)
代码地址如下:http://www.demodashi.com/demo/11944.html 天道酬勤 前言 作为一名iOS开发者,每当使用APP的时候,总难免会情不自禁的去想想,这个怎么做的?该怎 ...
- H5音乐播放器
前段时间无聊用JavaScript基于H5的audio写一个音乐播放器.误喷,技术有限,文笔不好之处希望各位大神海涵. 1.HTML代码: <div id="music" c ...
- 微信小程序音乐播放器
写在前面 1.入门几天小白的作品,希望为您有帮助,有好的意见或简易烦请赐教 2.微信小程序审核音乐类别已经下架,想要发布选题需慎重.附一个参考链接,感谢https://www.hishop.com.c ...
- SE Springer小组之《Spring音乐播放器》可行性研究报告三、四
3 对现有系统的分析 由于本次可行性分析主要是建立在团队自行实现一个音乐软件的目标上,并不是在一个现有系统的基础上开发改进的新系统.因此这里将分析一款市面上已经存在的音乐软件(以下称为W音乐),并为之 ...
- 卡拉OK效果的实现-iOS音乐播放器
自己编写的音乐播放器偶然用到这个模块,发现没有思路,而且上网搜了搜,关于这方面的文章不是很多,没找到满意的结果,然后自己也是想了想,最终实现了这种效果,想通了发现其实很简单. 直接上原理: 第一种: ...
- 小菜学习Winform(二)WMPLib实现音乐播放器
前言 现在网上有很多的音乐播放器,但好像都不是.net平台做的,在.net中实现音乐文件的播放功能很简单,下面就简单实现下. SoundPlayer类 在.net提供了音乐文件的类:SoundPlay ...
- 【大结局】《从案例中学习JavaScript》之酷炫音乐播放器(四)
这是之前写的用H5制作的音乐播放器,前三节其实已经做得差不多了,音轨的制作原理已经在上一节说明,不过一直还没有和音乐对接. 本章作为该系列的一个完结篇,我会专门把动态音轨的实现代码贴出来,demo地址 ...
随机推荐
- JavaScript 之 parseInt
首先还是从很热门的实例parseInt("09")==0说起. parseInt(numString, [radix])这个函数后面如果不跟第2个参数来表示进制的话,默认是10进制 ...
- webpack打包过滤console.log
在webpack.prod.conf.js里面的plugins里面加上 drop_debugger: true, drop_console: true new webpack.optimize.Ugl ...
- java创建线程的三种方式及其对比
第一种方法:继承Thread类,重写run()方法,run()方法代表线程要执行的任务.第二种方法:实现Runnable接口,重写run()方法,run()方法代表线程要执行的任务.第三种方法:实现c ...
- git查看各个branch之间的关系图
两种方法: 一. 使用Git log命令 git log --graph --decorate --oneline --simplify-by-decoration --all 说明: --deco ...
- Windows下如何配置tomcat环境变量
Apache Tomcat 是一款 Java Servlet 和 JavaServer Pages 技术的开源软件实现,可以作为测试 Servlet 的独立服务器,而且可以集成到 Apache Web ...
- Tomcat自带log的配置详解
最近几天被日志搞得烦死了,不详细了解不行呀! Tomcat根目录文件作用说明 根目录下有bin,conf,lib,logs,temp,webapps,work 7个文件夹 bin目录 主要是用来存放t ...
- 搭建Weblogic服务器
安妮,我的小熊熊在ne....... 01.安全设置 service iptables stop chkconfig iptables off #关闭防火墙,只是建议,为了简便操作 setenf ...
- esp8266烧录Html文件,实现内置网页控制设备!
代码地址如下:http://www.demodashi.com/demo/14321.html 一.前言: 这个月也快结束了,时间真快,我服务器知识自学依然在路途中,这几天听到热点网页配置esp826 ...
- linux配置server笔记
设置防火墙开放80port -A INPUT -m state --state NEW -m tcp -p tcp --dport 80 -j ACCEPT 尽管看不懂是什么,可是这个是用于开放80p ...
- hdu 4031(树状数组+辅助数组)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4031 Attack Time Limit: 5000/3000 MS (Java/Others) ...