Boot显示logo的方式

首先需要有LCD的驱动支持,XC2440板的LCD驱动是:drivers/video/xc2440_lcd.c

显示LOGO的函数调用流程:
在arch/arm/lib/board.c中的start_armboot调用common/stdio.c中的stdio_init
如果定义了CONFIG_VIDEO和CONFIG_CFB_CONSOLE,调用drivers/video/cfb_console.c中的drv_video_init
video_init调用XC2440的LCD驱动中的video_hw_init
接着video_logo调用logo_plot显示logo

相关阅读:U-Boot源代码下载地址

在config头文件中(xc2440.h)定义和LCD LOGO相关的宏

#define CONFIG_CMD_BMP

#define CONFIG_VIDEO

#define CONFIG_VIDEO_S3C2410

#define CONFIG_VIDEO_LOGO

#define CONFIG_VIDEO_BMP_LOGO

#define VIDEO_FB_16BPP_WORD_SWAP

如果定义了VIDEO_LOGO和VIDEO_BMP_LOGO
编译时会执行tools目录下的bmp_logo程序,读取"tools/logos/denx.bmp"文件,在include下生成bmp_logo.h文件


文件内容如下:

#define BMP_LOGO_WIDTH        160

#define BMP_LOGO_HEIGHT        96

#define BMP_LOGO_COLORS        31

#define BMP_LOGO_OFFSET        16

unsigned short bmp_logo_palette[] = {

.......................

};

unsigned char bmp_logo_bitmap[] = {

.......................

};

显示logo函数就会调用这个数组中的数据,显示到屏上


在cfb_console.c中定义了LOGO的属性

#define VIDEO_LOGO_WIDTH    BMP_LOGO_WIDTH

#define VIDEO_LOGO_HEIGHT    BMP_LOGO_HEIGHT

#define VIDEO_LOGO_LUT_OFFSET    BMP_LOGO_OFFSET

#define VIDEO_LOGO_COLORS    BMP_LOGO_COLORS


这个bmp文件是有要求的,位深度必须为8位,大小也有限制(全屏图片肯定是不行的,读取文件的后半部数据都是0)
自己做个logo图片,jpeg格式就可以,使用linux下的图片工具转成8位的bmp文件

jpegtopnm logo.jpg | ppmquant 31 | ppmtobmp -bpp 8 > denx.bmp
也可以搞个脚本,方便转换

#!/bin/sh
jpegtopnm $1 | ppmquant 31 | ppmtobmp -bpp 8 > $2

用法:脚本 原始文件名 转换后的文件名


三星平台默认使用denx.bmp文件,把自己的logo改成这个文件名,再编译就行了


一个显卡设备的结构体:
static GraphicDevice *pGD; /* Pointer to Graphic array */


GraphicDevice是表示显示设备属性的结构体
pGD代表的就是显示设备


在xc2440_lcd.c驱动中,给pGD赋值

pGD->winSizeX = res_mode->xres;

pGD->winSizeY = res_mode->yres;

pGD->plnSizeX = res_mode->xres;

pGD->plnSizeY = res_mode->yres;

pGD->gdfBytesPP = 2;

pGD->gdfIndex = GDF_16BIT_565RGB;

pGD->frameAdrs = LCD_VIDEO_ADDR;

pGD->memSize = VIDEO_MEM_SIZE;

显存的地址:#define LCD_VIDEO_ADDR 0x33d00000  (in xc2440.h)
显存的大小:#define VIDEO_MEM_SIZE   0x200000     (in xc2440_lcd.c)


在cfb_console.c中得到显卡显存的地址:
#define VIDEO_FB_ADRS        (pGD->frameAdrs)       

video_fb_address = (void *)VIDEO_FB_ADRS
后面的操作就使用video_fb_address


在logo_plot函数中,显示BMP_LOGO的流程 
(dest是指向video_fb_address的)

source = bmp_logo_bitmap;

logo_red = malloc (BMP_LOGO_COLORS);

logo_green = malloc (BMP_LOGO_COLORS);

logo_blue = malloc (BMP_LOGO_COLORS);

for (i = 0; i < VIDEO_LOGO_COLORS; i++) {

logo_red[i] = (bmp_logo_palette[i] & 0x0f00) >> 4;

logo_green[i] = (bmp_logo_palette[i] & 0x00f0);

logo_blue[i] = (bmp_logo_palette[i] & 0x000f) << 4;

}

while (ycount--) {

xcount = VIDEO_LOGO_WIDTH;

while (xcount--) {

r = logo_red[*source - VIDEO_LOGO_LUT_OFFSET];

g = logo_green[*source - VIDEO_LOGO_LUT_OFFSET];

b = logo_blue[*source - VIDEO_LOGO_LUT_OFFSET];

*(unsigned short *) dest = SWAP16 ((unsigned short) (((r >> 3) << 11) | ((g >> 2) << 5) | (b >> 3)));

}

source++;

dest += VIDEO_PIXEL_SIZE;

}

dest += skip;


如果只定义VIDEO_LOGO,没有定义VIDEO_BMP_LOGO
就使用LINUX_LOGO显示,用到include/linux_logo.h中定义的数组,显示的方式也不一样

内容版权声明:除非注明,否则皆为本站原创文章。

转载注明出处:http://www.heiqu.com/pxpzy.html