1 /***********************************************************************
2 * U-Boot_bmp_logo_hacking
3 * :
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 }