1. 增加Nandflash读取代码
因为要显示图片,而图片明显是放在Nandflash中比较合适,因此需要有能够操作Nandflash的函数。在U-boot中已经有能操作Nandflash的函数了,但是我找了半天也不知道该怎么调用,因此还是上自己的代码吧。
1 #define NFCONF *(volatile unsigned int *)0x70200000 2 #define NFCONT *(volatile unsigned int *)0x70200004 3 #define NFCMMD *(volatile unsigned char *)0x70200008 4 #define NFADDR *(volatile unsigned int *)0x7020000c 5 #define NFDATA *(volatile unsigned char *)0x70200010 6 #define NFSTAT *(volatile unsigned int *)0x70200028 7 #define NAND_SECTOR_SIZE 2048 8 #define NAND_BLOCK_MASK (NAND_SECTOR_SIZE - 1) 9 10 #define TACLS 7 11 #define TWRPH0 7 12 #define TWRPH1 7 13 14 /** 15 * \brief Nandflash初始化 16 */ 17 static void __nand_init(void) 18 { 19 NFCONF = (TACLS<<12) | (TWRPH0<<8) | (TWRPH1<<4); 20 NFCONT = (1<<4) | (1<<1) | (1); 21 } 22 23 static void __nand_select(void) 24 { 25 NFCONT &= ~(1<<1); 26 } 27 28 static void __nand_deselect(void) 29 { 30 NFCONT |= (1<<1); 31 } 32 33 static void __nand_cmd(unsigned char cmd) 34 { 35 NFCMMD = cmd; 36 } 37 38 static void __nand_addr (unsigned int addr) 39 { 40 unsigned int col = addr&NAND_BLOCK_MASK; 41 unsigned int page = addr/NAND_SECTOR_SIZE; 42 43 NFADDR = col & 0xff; 44 NFADDR = (col>>8) & 0xff; 45 NFADDR = page & 0xff; 46 NFADDR = (page >> 8) & 0xff; 47 NFADDR = (page >> 16) & 0x3; 48 } 49 50 static void __nand_wait_teady(void) 51 { 52 int i; 53 while (!(NFSTAT & 1)) 54 for (i = 0; i < 10; i++); 55 } 56 57 static unsigned char __nand_data(void) 58 { 59 unsigned char p; 60 p = NFDATA; 61 return p; 62 } 63 64 /** 65 * \brief 从Nandflash中读取数据至内存中 66 */ 67 static void __nand_read (unsigned int addr, unsigned char *buf, unsigned int len) 68 { 69 int col = addr % 2048; 70 int i = 0; 71 __nand_select(); 72 while (i < len) { 73 __nand_cmd(0); 74 __nand_addr(addr); 75 __nand_cmd(0x30); 76 __nand_wait_teady(); 77 78 for (; col < 2048&&(i < len); col++) { 79 buf[i] = __nand_data(); 80 i++; 81 addr++; 82 } 83 col = 0; 84 } 85 __nand_deselect(); 86 } 87 88 static void __nand_reset(void) 89 { 90 __nand_select(); 91 __nand_cmd(0xff); 92 __nand_wait_teady(); 93 __nand_deselect(); 94 } 95 96 /** 97 * \brief 将Nandflash中的内容读取至内存 98 */ 99 void nand_copy2ram(unsigned int addr, unsigned char *buff, unsigned int len) 100 { 101 //__nand_init(); 102 __nand_reset(); 103 __nand_read(addr, buff, len); 104 }
注意第101行我把__nand_init注释掉了,因为U-boot起来之后肯定已经对Nandflash初始化过了。
2. 修改LCD代码
lcd.c
1 #include "lcd.h" 2 3 static int PEN_COLOR = LCD_RED; /* 定义画笔(前景)颜色 */ 4 static int BK_COLOR = LCD_BLACK; /* 定义背景颜色 */ 5 6 /** 7 * \brief LCD初始化 8 */ 9 void lcd_init (void) 10 { 11 GPICON = 0; 12 GPICON |= 0xaaaaaaaa; 13 GPJCON = 0; 14 GPJCON |= 0xaaaaaa; 15 16 MIFPCON &= ~(0x1<<3); 17 SPCON &= ~0x3; 18 SPCON |= 0x1; 19 20 VIDCON0 = (14<<6) | (0x1<<4) | (0x3); 21 VIDCON1 = (0x1<6) | (0x1<<5); 22 23 VIDTCON0 = (1<<16) | (1<<8) | (9); 24 VIDTCON1 = (1<<16) | (1<<8) | (40); 25 26 VIDTCON2 = ((LCD_HEIGHT - 1) << 11) | (LCD_WIDTH - 1); 27 WINCON0 = (0xb<<2) | (0x1); 28 29 VIDOSD0A = 0; 30 VIDOSD0B = ((LCD_WIDTH - 1) << 11) | ((LCD_HEIGHT - 1)); 31 32 VIDOSD0C = LCD_WIDTH * LCD_HEIGHT; 33 34 VIDW00ADD0B0 = SHOW_BUF; 35 } 36 37 /** 38 * \brief 设置画笔颜色 39 */ 40 void lcd_set_pen_color (int color) 41 { 42 PEN_COLOR = color; 43 } 44 45 /** 46 * \brief 设置背景颜色 47 */ 48 void lcd_set_bk_color (int color) 49 { 50 BK_COLOR = color; 51 } 52 53 /** 54 * \brief 绘制一个点 55 */ 56 void lcd_draw_point (int x, int y) 57 { 58 *((int *)SHOW_BUF + x + y * LCD_WIDTH) = PEN_COLOR; 59 } 60 61 void lcd_draw_bk (int x, int y) 62 { 63 *((int *)SHOW_BUF + x + y * LCD_WIDTH) = BK_COLOR; 64 } 65 66 /** 67 * \brief 清屏(填充背景色) 68 */ 69 void lcd_clean (void) 70 { 71 int i, j; 72 for(i=0; i<LCD_HEIGHT; i++) { 73 for(j=0; j<LCD_WIDTH; j++) { 74 lcd_draw_bk(j, i); 75 } 76 } 77 } 78 79 /** 80 * \brief 显示一张BMP图片 81 */ 82 void lcd_show_bmp (int x , int y , unsigned char *pic) 83 { 84 col_t c; 85 BITMAPFILEHEADER filehead; 86 BITMAPINFOHEADER infohead; 87 int i, j; 88 unsigned char *p = 0; 89 int width_error = 0; 90 int old_pen_color; 91 92 memcpy(&filehead , pic, sizeof(filehead)); 93 memcpy(&infohead , pic + sizeof(filehead), sizeof(infohead)); 94 95 width_error = (4 - infohead.biWidth * 3 % 4) % 4; 96 97 p = pic + sizeof(filehead) + sizeof(infohead); 98 99 old_pen_color = PEN_COLOR; 100 101 c.c.non = 0; 102 for (j = infohead.biHeight-1; j>=0; j--) { 103 for (i = 0; i < infohead.biWidth; i++) { 104 c.c.b = *p++; 105 c.c.g = *p++; 106 c.c.r = *p++; 107 108 PEN_COLOR = c.l; 109 lcd_draw_point(x + i, y + j); 110 } 111 p += width_error; 112 } 113 PEN_COLOR = old_pen_color; 114 }
lcd.h
1 #ifndef __LCD_H 2 #define __LCD_H 3 4 /* 定义寄存器地址 */ 5 #define GPICON (*(unsigned int *)0x7F008100) 6 #define SPCON (*(unsigned int *)0x7F0081A0) 7 #define MIFPCON (*(unsigned int *)0x7410800c) 8 #define GPJCON (*(unsigned int *)0x7F008120) 9 10 #define VIDCON0 (*(unsigned int *)0x77100000) 11 #define VIDCON1 (*(unsigned int *)0x77100004) 12 #define VIDCON2 (*(unsigned int *)0x77100008) 13 #define VIDTCON0 (*(unsigned int *)0x77100010) 14 #define VIDTCON1 (*(unsigned int *)0x77100014) 15 #define VIDTCON2 (*(unsigned int *)0x77100018) 16 #define WINCON0 (*(unsigned int *)0x77100020) 17 #define WINCON1 (*(unsigned int *)0x77100024) 18 #define WINCON2 (*(unsigned int *)0x77100028) 19 #define WINCON3 (*(unsigned int *)0x7710002C) 20 #define WINCON4 (*(unsigned int *)0x77100030) 21 #define VIDOSD0A (*(unsigned int *)0x77100040) 22 #define VIDOSD0B (*(unsigned int *)0x77100044) 23 #define VIDOSD0C (*(unsigned int *)0x77100048) 24 #define VIDOSD1A (*(unsigned int *)0x77100050) 25 #define VIDOSD1B (*(unsigned int *)0x77100054) 26 #define VIDOSD1C (*(unsigned int *)0x77100058) 27 #define VIDW00ADD0B0 (*(unsigned int *)0x771000A0) 28 #define GPECON (*(unsigned int *)0x7f008080) 29 #define GPEDAT (*(unsigned int *)0x7f008084) 30 31 #define LCD_WIDTH 480 /* 定义LCD宽度 */ 32 #define LCD_HEIGHT 272 /* 定义LCD高度 */ 33 34 /* 定义显存地址 */ 35 #define SHOW_BUF 0x54000000 36 37 /* 定义常用颜色 */ 38 #define COL(R,G,B) ((R<<16) | (G<<8) | (B)) 39 #define LCD_RED COL(0xFF, 0, 0) 40 #define LCD_GREEN COL(0, 0xFF, 0) 41 #define LCD_BLUE COL(0, 0, 0xFF) 42 #define LCD_WHITE COL(0xFF, 0xFF, 0xFF) 43 #define LCD_BLACK COL(0, 0, 0) 44 45 typedef unsigned short WORD; 46 typedef unsigned char BYTE; 47 typedef unsigned int DWORD; 48 typedef int LONG; 49 50 #pragma pack(2) 51 typedef struct tagBITMAPFILEHEADER 52 { 53 WORD bfType; // 位图文件的类型,必须为BM 54 DWORD bfSize; // 位图文件的大小,以字节为单位 55 WORD bfReserved1; // 位图文件保留字,必须为0 56 WORD bfReserved2; // 位图文件保留字,必须为0 57 DWORD bfOffBits; // 位图数据的起始位置,以相对于位图 58 // 文件头的偏移量表示,以字节为单位 59 } BITMAPFILEHEADER; 60 61 typedef struct tagBITMAPINFOHEADER 62 { 63 DWORD biSize; // 本结构所占用字节数 64 LONG biWidth; // 位图的宽度,以像素为单位 65 LONG biHeight; // 位图的高度,以像素为单位 66 WORD biPlanes; // 目标设备的级别,必须为1 67 WORD biBitCount; // 每个像素所需的位数,必须是1(双色), 68 // 4(16色),8(256色)或24(真彩色)之一 69 DWORD biCompression; // 位图压缩类型,必须是 0(不压缩), 70 // 1(BI_RLE8压缩类型)或2(BI_RLE4压缩类型)之一 71 DWORD biSizeImage; // 位图的大小,以字节为单位 72 LONG biXPelsPerMeter; // 位图水平分辨率,每米像素数 73 LONG biYPelsPerMeter; // 位图垂直分辨率,每米像素数 74 DWORD biClrUsed; // 位图实际使用的颜色表中的颜色数 75 DWORD biClrImportant; // 位图显示过程中重要的颜色数 76 } BITMAPINFOHEADER; 77 #pragma pack(4) 78 79 struct _rgb_ 80 { 81 unsigned char b; 82 unsigned char g; 83 unsigned char r; 84 unsigned char non; 85 }; 86 87 typedef union _color_ 88 { 89 struct _rgb_ c; 90 unsigned int l; 91 } col_t; 92 93 /******************************* 函数声明 **************************/ 94 /** 95 * \brief LCD初始化 96 */ 97 extern void lcd_init (void); 98 99 /** 100 * \brief 设置画笔颜色 101 */ 102 extern void lcd_set_pen_color (int color); 103 104 /** 105 * \brief 设置背景颜色 106 */ 107 extern void lcd_set_bk_color (int color); 108 109 /** 110 * \brief 绘制一个点 111 */ 112 extern void lcd_draw_point (int x, int y); 113 114 /** 115 * \brief 清屏(填充背景色) 116 */ 117 extern void lcd_clean (void); 118 119 /** 120 * \brief 显示一张BMP图片 121 */ 122 extern void lcd_show_bmp (int x , int y , unsigned char *pic); 123 124 #endif
3. 修改board.c和Makefile
去掉原先填充纯色的代码,改为显示图片的代码:
nand_copy2ram第一个参数为Nandflash中要读取的地址,在这里我写的是0x100000,因此运行程序前需要保证该地址有我想要的图片;第二个参数为内存地址,需要与保证其他程序不会使用该地址,且与lcd_show_bmp传入的地址相同;第三个参数为拷贝数据大小,一张480*272的24位图大小不会超过400K,这里传入0x80000足够了。
此外还要修改lib_arm下的Makefile,在COBJS后面加上nand.o后编译。
4. 下载程序
编译成功后在U-boot中将新编译出的U-boot下载进Nandflash的0地址,将BMP图片下载到0x100000地址。重启开发板,能看到LCD上显示出图片。