U-Boot_bmp_logo_hacking

  1 /***********************************************************************
  2  *                    U-Boot_bmp_logo_hacking 
  3  *  声明:
  4  *      1. 该源代码来自myzr_android4_2_2_1_1_0.tar.bz2中的:
  5  *          bootable/bootloader/uboot-imx/tools/bmp_logo.c
  6  *      2. 通过阅读该源码可以知道大致如何解析bmp图片,以及一些自动生成
  7  *          的文件是如何做到的,如一些自动生成.h和.c文件;
  8  *      3. 阅读该源码技能需求:
  9  *          1. bmp图片的格式的一些基本信息;
 10  *          2. 类Unix系统编程;
 11  *          3. C语言;
 12  *      4. 本源程序的阅读技巧:
 13  *          1. 本人是用了vim + ctags;
 14  *          2. 如果您是在windows下,传说中是可以是用Source Insight;
 15  *          3. 找main函数开始阅读;
 16  *
 17  *                      2015-4-19 周日 晴 深圳 南山 西丽平山村 曾剑锋
 18  **********************************************************************/
 19 
 20 
 21 
 22 /**
 23  * 源程序中仅仅是用了#include "compiler.h",由于我们仅仅需要本文件,
 24  * 所以本人注释了那一行,添加以下本文件需要用到的头文件
 25  */
 26 //#include "compiler.h"
 27 #include <errno.h>
 28 #include <stdlib.h>
 29 #include <stdint.h>
 30 #include <stdio.h>
 31 #include <string.h>
 32 
 33 /**
 34  * 用于大致描述一个bmp图片的结构体
 35  */
 36 typedef struct bitmap_s {        /* bitmap description */
 37     uint16_t width;
 38     uint16_t height;
 39     uint8_t    palette[256*3];
 40     uint8_t    *data;
 41 } bitmap_t;
 42 
 43 /**
 44  * 默认的颜色映射的大小
 45  */
 46 #define DEFAULT_CMAP_SIZE    16    /* size of default color map    */
 47 
 48 /*
 49  * Neutralize little endians.
 50  * bmp图片用的好象是小端的存储方式
 51  */
 52 uint16_t le_short(uint16_t x)
 53 {
 54     uint16_t val;
 55     uint8_t *p = (uint8_t *)(&x);
 56 
 57     val =  (*p++ & 0xff) << 0;
 58     val |= (*p & 0xff) << 8;
 59 
 60     return val;
 61 }
 62 
 63 /**
 64  * 在文件的当前位置,偏移多少个字节
 65  */
 66 void skip_bytes (FILE *fp, int n)
 67 {
 68     while (n-- > 0)
 69         fgetc (fp);
 70 }
 71 
 72 /**
 73  * 错误输出函数,输出到标准错误输出
 74  */
 75 __attribute__ ((__noreturn__))
 76 int error (char * msg, FILE *fp)
 77 {
 78     fprintf (stderr, "ERROR: %s\n", msg);
 79 
 80     fclose (fp);
 81 
 82     exit (EXIT_FAILURE);
 83 }
 84 
 85 int main (int argc, char *argv[])
 86 {
 87     /**
 88      * 局部变量说明:
 89      *     1. i           : for循环计数;
 90      *     2. x           : 字符暂存变量,for循环计数;
 91      *     3. fp          : 打开的bmp文件指针;
 92      *     4. bmp         : 用于存储bmp一些数据的数据结构;
 93      *     5. b           : 指向上面bmp数据结构的指针;
 94      *     6. data_offset : bmp数据区相对文件头的偏移;
 95      *     7. n_colors    : 实际使用了多少种颜色
 96      */
 97     int    i, x;
 98     FILE    *fp;
 99     bitmap_t bmp;
100     bitmap_t *b = &bmp;
101     uint16_t data_offset, n_colors;
102 
103     /**
104      * 命令行参数个数检查
105      */
106     if (argc < 2) {
107         fprintf (stderr, "Usage: %s file\n", argv[0]);
108         exit (EXIT_FAILURE);
109     }
110 
111     /**
112      * 以二进制只读的方式打开bmp文件
113      */
114     if ((fp = fopen (argv[1], "rb")) == NULL) {
115         perror (argv[1]);
116         exit (EXIT_FAILURE);
117     }
118 
119     /**
120      * 检查是否是bmp图片
121      */
122     if (fgetc (fp) != 'B' || fgetc (fp) != 'M')
123         error ("Input file is not a bitmap", fp);
124 
125     /*
126      * read width and height of the image, and the number of colors used;
127      * ignore the rest
128      */
129     /**
130      * 前面的'B','M'占用了2个字节,
131      * 2字节 + 8字节 = 10字节,
132      * 这时文件指针正好指向11字节(保存bmp数据偏移的位置)
133      */
134     skip_bytes (fp, 8);
135     if (fread (&data_offset, sizeof (uint16_t), 1, fp) != 1)
136         error ("Couldn't read bitmap data offset", fp);
137     skip_bytes (fp, 6);
138 
139     /**
140      * 前面的'B','M'占用了2个字节,
141      * 2字节 + 8字节 = 10字节,
142      * 10字节 + 2字节 = 12字节,
143      * 12字节 + 6字节 = 18字节,
144      * 这时文件指针正好指向19字节(保存bmp宽的位置)
145      */
146     if (fread (&b->width,   sizeof (uint16_t), 1, fp) != 1)
147         error ("Couldn't read bitmap width", fp);
148     skip_bytes (fp, 2);
149 
150     /**
151      * 前面的'B','M'占用了2个字节,
152      * 2字节 + 8字节 = 10字节,
153      * 10字节 + 2字节 = 12字节,
154      * 12字节 + 6字节 = 18字节,
155      * 18字节 + 2字节 = 20字节,
156      * 20字节 + 2字节 = 22字节,
157      * 这时文件指针正好指向23字节(保存bmp高的位置)
158      */
159     if (fread (&b->height,  sizeof (uint16_t), 1, fp) != 1)
160         error ("Couldn't read bitmap height", fp);
161     skip_bytes (fp, 22);
162 
163     /**
164      * 前面的'B','M'占用了2个字节,
165      * 2字节 + 8字节 = 10字节,
166      * 10字节 + 2字节 = 12字节,
167      * 12字节 + 6字节 = 18字节,
168      * 18字节 + 2字节 = 20字节,
169      * 20字节 + 2字节 = 22字节,
170      * 22字节 + 2字节 = 24字节,
171      * 24字节 + 22字节 = 46字节,
172      * 这时文件指针正好指向47字节(保存bmp图实际是用的颜色数)
173      * skip_bytes (fp, 6);  --> 跳出位图信息头
174      */
175     if (fread (&n_colors, sizeof (uint16_t), 1, fp) != 1)
176         error ("Couldn't read bitmap colors", fp);
177     skip_bytes (fp, 6);
178 
179     /*
180      * Repair endianess.
181      * 防止数据出现大小不兼容的问题
182      */
183     data_offset = le_short(data_offset);
184     b->width = le_short(b->width);
185     b->height = le_short(b->height);
186     n_colors = le_short(n_colors);
187 
188     /* assume we are working with an 8-bit file */
189     /**
190      * 防止颜色数太小,或太大
191      */
192     if ((n_colors == 0) || (n_colors > 256 - DEFAULT_CMAP_SIZE)) {
193         /* reserve DEFAULT_CMAP_SIZE color map entries for default map */
194         n_colors = 256 - DEFAULT_CMAP_SIZE;
195     }
196 
197     /**
198      * 打印出一些注释信息和宏定义数据
199      */
200     printf ("/*\n"
201         " * Automatically generated by \"tools/bmp_logo\"\n"
202         " *\n"
203         " * DO NOT EDIT\n"
204         " *\n"
205         " */\n\n\n"
206         "#ifndef __BMP_LOGO_H__\n"
207         "#define __BMP_LOGO_H__\n\n"
208         "#define BMP_LOGO_WIDTH\t\t%d\n"
209         "#define BMP_LOGO_HEIGHT\t\t%d\n"
210         "#define BMP_LOGO_COLORS\t\t%d\n"
211         "#define BMP_LOGO_OFFSET\t\t%d\n"
212         "\n",
213         b->width, b->height, n_colors,
214         DEFAULT_CMAP_SIZE);
215 
216     /* allocate memory */
217     /**
218      * 采用内存分配的方式,获取data的存储空间
219      */
220     if ((b->data = (uint8_t *)malloc(b->width * b->height)) == NULL)
221         error ("Error allocating memory for file", fp);
222 
223     /* read and print the palette information */
224     /**
225      *  以下是一个输出结果示例:
226      *      unsigned short bmp_logo_palette[] = {
227      *          0x0FFF,  0x0DDE,  0x026B,  0x026B,  0x0FFF,  0x0FFF,  0x048C,  0x026B,
228      *          0x026B,  0x0BDE,  0x047C,  0x027B,  0x09BE,  0x026B,  0x0EEF,  0x037B,
229      *          0x08AD,  0x0DEF,  0x027B,  0x069D,  0x0CDE,  0x0ACE,  0x08BD,  0x07AD,
230      *          0x027B,  0x058C,  0x037B,  0x0CDE,  0x06AD,  0x037C,
231      *      };
232      */
233     printf ("unsigned short bmp_logo_palette[] = {\n");
234 
235     for (i=0; i<n_colors; ++i) {
236         b->palette[(int)(i*3+2)] = fgetc(fp);   //个人查资料认为是blue
237         b->palette[(int)(i*3+1)] = fgetc(fp);   //个人查资料认为是green
238         b->palette[(int)(i*3+0)] = fgetc(fp);   //个人查资料认为是red
239         x=fgetc(fp);
240 
241         /**
242          * 输出的结果正好和读出来的结果相反,主要是因为
243          * 读取时,后面的高位,输出时先输出的是高位
244          * 另外这里还考虑到格式化对齐的问题,主要是
245          * 方便阅读输出的数据.
246          */
247         printf ("%s0x0%X%X%X,%s",   
248             ((i%8) == 0) ? "\t" : "  ",
249             (b->palette[(int)(i*3+0)] >> 4) & 0x0F,
250             (b->palette[(int)(i*3+1)] >> 4) & 0x0F,
251             (b->palette[(int)(i*3+2)] >> 4) & 0x0F,
252             ((i%8) == 7) ? "\n" : ""    
253         );
254     }
255 
256     /* seek to offset indicated by file header */
257     /**
258      * 感觉这行代码不应该放这里,应该放到下面2行后面去比较合理
259      */
260     fseek(fp, (long)data_offset, SEEK_SET);
261 
262     /* read the bitmap; leave room for default color map */
263     printf ("\n");
264     printf ("};\n");
265 
266     printf ("\n");
267 
268     /**
269      * 1. 以下是输出结果示例:
270      *     unsigned char bmp_logo_bitmap[] = {
271      *         0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10,
272      *         0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10, 0x10,
273      *         ......
274      *         0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
275      *         0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13, 0x13,
276      *     }
277      * 2. 位图数据记录顺序是在扫描行内是从左到右,扫描行之间是从下到上,
278      *      Windows规定一个扫描行所占的字节数必须是4的倍数,不足以0填充;
279      * 3. i = (b->height-1)*b->width : 相当于跳到数组的左下脚
280      */
281     printf ("unsigned char bmp_logo_bitmap[] = {\n");
282     for (i=(b->height-1)*b->width; i>=0; i-=b->width) {
283         for (x = 0; x < b->width; x++) {
284             b->data[(uint16_t) i + x] = (uint8_t) fgetc (fp) \
285                         + DEFAULT_CMAP_SIZE;    //不知道这里为什么需要加这个参数
286         }
287     }
288     fclose (fp);
289 
290     /**
291      * 输出bmp数据
292      */
293     for (i=0; i<(b->height*b->width); ++i) {
294         if ((i%8) == 0)
295             putchar ('\t');
296         printf ("0x%02X,%c",
297             b->data[i],
298             ((i%8) == 7) ? '\n' : ' '
299         );
300     }
301     printf ("\n"
302         "};\n\n"
303         "#endif /* __BMP_LOGO_H__ */\n"
304     );
305 
306     return (0);
307 }

 

posted on 2015-04-20 14:56  zengjf  阅读(774)  评论(0编辑  收藏  举报

导航