先上代码,最后讲解。

draw.c

  1 #include <stdio.h>
  2 #include <linux/fb.h>
  3 #include <sys/types.h>
  4 #include <sys/stat.h>
  5 #include <fcntl.h>
  6 #include <sys/mman.h>
  7 #include "draw.h"
  8 
  9 static struct fb_var_screeninfo __g_vinfo;    /* 显示信息 */
 10 static color_t *__gp_frame;                    /* 虚拟屏幕首地址 */
 11 static color_t __g_pen_color;                /* 画笔颜色 */
 12 static color_t __g_bk_color;                /* 背景颜色 */
 13 static font_size_t __g_font_size;            /* 当前字体大小 */
 14 
 15 /* 定义绘制不同字体的函数和字体大小 */
 16 extern int draw_one_char_fs16 (unsigned int x, unsigned int y, char dat);
 17 extern int draw_one_char_fs24 (unsigned int x, unsigned int y, char dat);
 18 extern int draw_one_char_fs32 (unsigned int x, unsigned int y, char dat);
 19 extern int draw_one_char_fs48 (unsigned int x, unsigned int y, char dat);
 20 extern int draw_one_char_fs64 (unsigned int x, unsigned int y, char dat);
 21     
 22 static struct struct_draw_font __g_draw_font_tab[] = {
 23     {draw_one_char_fs16, FONT_SIZE_16},
 24     {draw_one_char_fs24, FONT_SIZE_24},
 25     {draw_one_char_fs32, FONT_SIZE_32},
 26     {draw_one_char_fs48, FONT_SIZE_48},
 27     {draw_one_char_fs64, FONT_SIZE_64},
 28 };
 29 
 30 /**
 31  * \brief framebuffer初始化
 32  */
 33 int framebuffer_init (void)
 34 {
 35     int fd = 0;
 36 
 37     fd = open("/dev/fb0", O_RDWR);
 38     if (fd == -1) {
 39         perror("fail to open /dev/fb0\n");
 40         return -1;
 41     }
 42 
 43     ioctl(fd, FBIOGET_VSCREENINFO, &__g_vinfo);                    /* 获取显示信息 */
 44     printf("bits_per_pixel = %d\n", __g_vinfo.bits_per_pixel);    /* 得到一个像素点对应的位数 */
 45 
 46     __gp_frame = mmap(NULL,                             /* 映射区的开始地址,为NULL表示由系统决定映射区的起始地址 */
 47                       __g_vinfo.xres_virtual * __g_vinfo.yres_virtual * __g_vinfo.bits_per_pixel / 8,    /* 映射区大小 */
 48                       PROT_WRITE | PROT_READ,            /* 内存保护标志(可读可写) */
 49                       MAP_SHARED,                        /* 映射对象类型(与其他进程共享) */
 50                       fd,                                /* 有效的文件描述符 */
 51                       0);                                /* 被映射内容的偏移量 */
 52     if (__gp_frame == NULL) {
 53         perror("fail to mmap\n");
 54         return -1;
 55     }
 56 
 57     return 0;
 58 }
 59 
 60 /**
 61  * \brief 填充整屏
 62  */
 63 void full_screen (color_t color)
 64 {
 65     int i;
 66     color_t *p = __gp_frame;
 67 
 68     for (i = 0; i < __g_vinfo.xres_virtual * __g_vinfo.yres_virtual; i++) {
 69         *p++ = color;
 70     }
 71 }
 72 
 73 /**
 74  * \brief 设置画笔颜色
 75  */
 76 void set_pen_color (color_t color)
 77 {
 78     __g_pen_color = color;
 79 }
 80 
 81 /**
 82  * \brief 获得画笔颜色
 83  */
 84 color_t get_pen_color (void)
 85 {
 86     return __g_pen_color;
 87 }
 88 
 89 /**
 90  * \brief 设置背景颜色
 91  */
 92 void set_bk_color (color_t color)
 93 {
 94     __g_bk_color = color;
 95 }
 96 
 97 /**
 98  * \brief 获得画笔颜色
 99  */
100 color_t get_bk_color (void)
101 {
102     return __g_bk_color;
103 }
104 
105 /**
106  * \brief 绘制一个点
107  *
108  * \retval 成功返回0,失败返回-1
109  */
110 int draw_point (unsigned int x, unsigned int y)
111 {
112     color_t *p_color;
113 
114     if ((x > __g_vinfo.xres_virtual) || (y > __g_vinfo.yres_virtual)) {
115         return -1;
116     }
117 
118     p_color = __gp_frame + y * __g_vinfo.xres_virtual + x;
119     *p_color = __g_pen_color;
120 
121     return 0;
122 }
123 
124 /**
125  * \brief 绘制一个背景点
126  *
127  * \retval 成功返回0,失败返回-1
128  */
129 int draw_bk_point (unsigned int x, unsigned int y)
130 {
131     color_t *p_color;
132 
133     if ((x > __g_vinfo.xres_virtual) || (y > __g_vinfo.yres_virtual)) {
134         return -1;
135     }
136 
137     p_color = __gp_frame + y * __g_vinfo.xres_virtual + x;
138     *p_color = __g_bk_color;
139 
140     return 0;
141 }
142 
143 /**
144  * \brief 设置字体大小
145  *
146  * \retval 成功返回0,失败返回-1
147  */
148 int set_font_size (font_size_t font)
149 {
150     if (font >= MAX_FONT_SIZE) {
151         return -1;
152     }
153 
154     __g_font_size = font;
155 
156     return 0;
157 }
158 
159 /**
160  * \brief 获取字体大小
161  *
162  * \retval 字体大小
163  */
164 font_size_t get_font_size (void)
165 {
166     return __g_font_size;    
167 }
168 
169 /**
170  * \brief 绘制一个英文字符
171  *
172  * \retval 成功返回0,失败返回-1
173  */
174 int draw_one_char (unsigned int x, unsigned int y, char dat)
175 {
176     int i;
177 
178     for (i = 0; i < sizeof(__g_draw_font_tab) / sizeof(__g_draw_font_tab[0]); i++) {
179         if (__g_draw_font_tab[i].font_size == __g_font_size) {
180             return __g_draw_font_tab[i].draw_one_char_fn(x, y, dat);
181         }
182     }
183 
184     return -1;
185 }
186 
187 /**
188  * \brief 绘制一个字符串
189  *
190  * \retval 返回成功绘制的字符个数
191  */
192 int draw_str (unsigned int x, unsigned int y, char *p_str)
193 {
194     int ret = 0;
195     int char_num = 0;
196     font_size_t font_size;
197 
198     font_size = get_font_size();
199 
200     while (*p_str) {
201         if (0 == draw_one_char(x + char_num * font_size / 2, y, *p_str)) {
202             ret++;
203         }
204         char_num++;
205         p_str++;
206     }
207 
208     return ret;
209 }

draw.h

  1 typedef unsigned short color_t;                /* 定义颜色类型,根据实际情况修改,此处为unsigned short是565的屏 */
  2 
  3 /**< \brief 定义常用颜色 */
  4 #define LCD_BLACK    0x0000                    /* 黑色 */
  5 #define LCD_BLUE    0x001F                    /* 蓝色 */
  6 #define LCD_GREEN    0x07E0                    /* 绿色 */
  7 #define LCD_RED        0xF800                    /* 红色 */
  8 #define LCD_CYAN    (LCD_BLUE | LCD_GREEN)    /* 青色 */
  9 #define LCD_MAGENTA    (LCD_BLUE | LCD_RED)    /* 洋红 */
 10 #define LCD_YELLOW    (LCD_GREEN | LCD_RED)    /* 黄色 */
 11 #define LCD_WHITE    (LCD_CYAN | LCD_RED)    /* 白色 */
 12 
 13 /**< \brief 定义支持的字体类型 */
 14 typedef enum enum_font_size {
 15     FONT_SIZE_16 = 16,
 16     FONT_SIZE_24 = 24,
 17     FONT_SIZE_32 = 32,
 18     FONT_SIZE_48 = 48,
 19     FONT_SIZE_64 = 64,
 20 
 21     MAX_FONT_SIZE    /* 最大字体号,该项放在最后即可 */
 22 } font_size_t;
 23 
 24 /* 定义绘制字符的函数指针类型 */
 25 typedef int (* draw_one_char_t) (unsigned int x, unsigned int y, char dat);
 26 
 27 /* 定义绘制字体的函数和字体大小结构 */
 28 struct struct_draw_font {
 29     draw_one_char_t draw_one_char_fn;    /* 绘制函数 */
 30     font_size_t font_size;                /* 字体大小 */
 31 };
 32 
 33 /**
 34  * \brief framebuffer初始化
 35  */
 36 extern int framebuffer_init (void);
 37 
 38 /**
 39  * \brief 填充整屏
 40  */
 41 extern void full_screen (color_t color);
 42 
 43 /**
 44  * \brief 设置画笔颜色
 45  */
 46 extern void set_pen_color (color_t color);
 47 
 48 /**
 49  * \brief 获得画笔颜色
 50  */
 51 extern color_t get_pen_color (void);
 52 
 53 /**
 54  * \brief 设置背景颜色
 55  */
 56 extern void set_bk_color (color_t color);
 57 
 58 /**
 59  * \brief 获得背景颜色
 60  */
 61 extern color_t get_bk_color (void);
 62 
 63 /**
 64  * \brief 绘制一个点
 65  *
 66  * \retval 成功返回0,失败返回-1
 67  */
 68 extern int draw_point (unsigned int x, unsigned int y);
 69 
 70 /**
 71  * \brief 绘制一个背景点
 72  *
 73  * \retval 成功返回0,失败返回-1
 74  */
 75 extern int draw_bk_point (unsigned int x, unsigned int y);
 76 
 77 /**
 78  * \brief 设置字体大小
 79  *
 80  * \retval 成功返回0,失败返回-1
 81  */
 82 extern int set_font_size (font_size_t font);
 83 
 84 /**
 85  * \brief 获取字体大小
 86  *
 87  * \retval 字体大小
 88  */
 89 extern font_size_t get_font_size (void);
 90 
 91 /**
 92  * \brief 绘制一个英文字符
 93  *
 94  * \retval 成功返回0,失败返回-1
 95  */
 96 extern int draw_one_char (unsigned int x, unsigned int y, char dat);
 97 
 98 /**
 99  * \brief 绘制一个字符串
100  *
101  * \retval 返回成功绘制的字符个数
102  */
103 extern int draw_str (unsigned int x, unsigned int y, char *p_str);

font_size_48.c

注:编译时除了48号字体外还有16,24,32,64等字体,但是我使用的是48号字体

  1 /* 支持的字符 */
  2 static const char char_fs48[] = {"0123456789:- "};
  3 
  4 /* 字模 */
  5 static const char font_size_48[][16 * 9] = {
  6 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
  7  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7E,0x00,0x80,0xC3,0x01,0xC0,0x01,
  8  0x03,0xC0,0x00,0x07,0x60,0x00,0x06,0x70,0x00,0x0E,0x30,0x00,0x0C,0x38,0x00,0x1C,
  9  0x38,0x00,0x1C,0x38,0x00,0x1C,0x18,0x00,0x38,0x1C,0x00,0x38,0x1C,0x00,0x38,0x1C,
 10  0x00,0x38,0x1C,0x00,0x38,0x1C,0x00,0x38,0x1C,0x00,0x38,0x1C,0x00,0x38,0x1C,0x00,
 11  0x38,0x1C,0x00,0x38,0x1C,0x00,0x38,0x1C,0x00,0x38,0x18,0x00,0x18,0x38,0x00,0x1C,
 12  0x38,0x00,0x1C,0x38,0x00,0x1C,0x30,0x00,0x0C,0x70,0x00,0x0E,0x60,0x00,0x06,0xC0,
 13  0x00,0x07,0xC0,0x01,0x03,0x00,0xC3,0x01,0x00,0x7E,0x00,0x00,0x00,0x00,0x00,0x00,
 14  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"0",0*/
 15 
 16 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
 17  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x20,0x00,0x00,0x30,0x00,0x00,0x38,
 18  0x00,0xC0,0x3F,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,
 19  0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,
 20  0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,
 21  0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,
 22  0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,0x38,0x00,0x00,
 23  0x38,0x00,0x00,0x38,0x00,0x00,0x7C,0x00,0xC0,0xFF,0x07,0x00,0x00,0x00,0x00,0x00,
 24  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"1",1*/
 25 
 26 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
 27  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFE,0x00,0x80,0xC1,0x03,0x60,0x00,
 28  0x07,0x30,0x00,0x0E,0x30,0x00,0x0E,0x38,0x00,0x1C,0x38,0x00,0x1C,0x38,0x00,0x1C,
 29  0x78,0x00,0x1C,0x78,0x00,0x1C,0x70,0x00,0x1C,0x00,0x00,0x0E,0x00,0x00,0x0E,0x00,
 30  0x00,0x07,0x00,0x00,0x07,0x00,0x80,0x03,0x00,0xC0,0x01,0x00,0xE0,0x00,0x00,0x70,
 31  0x00,0x00,0x38,0x00,0x00,0x1C,0x00,0x00,0x0E,0x00,0x00,0x07,0x00,0x80,0x03,0x00,
 32  0xC0,0x01,0x00,0xE0,0x00,0x10,0x60,0x00,0x10,0x30,0x00,0x18,0x38,0x00,0x08,0x1C,
 33  0x00,0x0E,0xFC,0xFF,0x0F,0xFC,0xFF,0x0F,0xFC,0xFF,0x0F,0x00,0x00,0x00,0x00,0x00,
 34  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"2",2*/
 35 
 36 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
 37  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x7E,0x00,0x80,0xC1,0x01,0x40,0x80,
 38  0x03,0x20,0x00,0x07,0x30,0x00,0x07,0x30,0x00,0x0E,0x70,0x00,0x0E,0x70,0x00,0x0E,
 39  0x60,0x00,0x0E,0x00,0x00,0x0E,0x00,0x00,0x06,0x00,0x00,0x07,0x00,0x00,0x03,0x00,
 40  0x80,0x01,0x00,0xE0,0x00,0x00,0x3E,0x00,0x00,0xE0,0x00,0x00,0x80,0x03,0x00,0x00,
 41  0x07,0x00,0x00,0x0E,0x00,0x00,0x0C,0x00,0x00,0x1C,0x00,0x00,0x1C,0x00,0x00,0x1C,
 42  0x30,0x00,0x1C,0x78,0x00,0x1C,0x78,0x00,0x1C,0x78,0x00,0x0C,0x38,0x00,0x0E,0x30,
 43  0x00,0x06,0x60,0x00,0x03,0xC0,0xC0,0x01,0x00,0x7F,0x00,0x00,0x00,0x00,0x00,0x00,
 44  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"3",3*/
 45 
 46 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
 47  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80,0x01,0x00,0xC0,0x01,0x00,0xC0,
 48  0x01,0x00,0xE0,0x01,0x00,0xF0,0x01,0x00,0xD0,0x01,0x00,0xC8,0x01,0x00,0xC8,0x01,
 49  0x00,0xC4,0x01,0x00,0xC6,0x01,0x00,0xC2,0x01,0x00,0xC3,0x01,0x00,0xC1,0x01,0x80,
 50  0xC0,0x01,0xC0,0xC0,0x01,0x40,0xC0,0x01,0x60,0xC0,0x01,0x20,0xC0,0x01,0x30,0xC0,
 51  0x01,0x18,0xC0,0x01,0x18,0xC0,0x01,0x0C,0xC0,0x01,0xFC,0xFF,0x3F,0x00,0xC0,0x01,
 52  0x00,0xC0,0x01,0x00,0xC0,0x01,0x00,0xC0,0x01,0x00,0xC0,0x01,0x00,0xC0,0x01,0x00,
 53  0xC0,0x01,0x00,0xC0,0x01,0x00,0xE0,0x03,0x00,0xFE,0x3F,0x00,0x00,0x00,0x00,0x00,
 54  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"4",4*/
 55 
 56 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
 57  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xE0,0xFF,0x0F,0xE0,0xFF,0x0F,0xE0,0xFF,
 58  0x0F,0x20,0x00,0x00,0x20,0x00,0x00,0x20,0x00,0x00,0x20,0x00,0x00,0x20,0x00,0x00,
 59  0x30,0x00,0x00,0x30,0x00,0x00,0x30,0x00,0x00,0x30,0x00,0x00,0x30,0xFE,0x00,0x30,
 60  0xFF,0x01,0xB0,0x81,0x07,0xF0,0x00,0x07,0x70,0x00,0x0E,0x30,0x00,0x0E,0x00,0x00,
 61  0x1E,0x00,0x00,0x1C,0x00,0x00,0x1C,0x00,0x00,0x1C,0x00,0x00,0x1C,0x30,0x00,0x1C,
 62  0x78,0x00,0x1C,0x78,0x00,0x1C,0x78,0x00,0x0C,0x38,0x00,0x0E,0x30,0x00,0x0E,0x30,
 63  0x00,0x07,0x60,0x00,0x03,0xC0,0xC1,0x01,0x00,0x7F,0x00,0x00,0x00,0x00,0x00,0x00,
 64  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"5",5*/
 65 
 66 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
 67  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xF8,0x01,0x00,0x07,0x02,0x80,0x01,
 68  0x06,0xC0,0x00,0x0E,0x60,0x00,0x0E,0x60,0x00,0x0E,0x30,0x00,0x00,0x30,0x00,0x00,
 69  0x38,0x00,0x00,0x38,0x00,0x00,0x18,0x00,0x00,0x18,0x00,0x00,0x1C,0xFC,0x00,0x1C,
 70  0xFF,0x03,0x9C,0x83,0x07,0xDC,0x00,0x0E,0x7C,0x00,0x1C,0x3C,0x00,0x1C,0x3C,0x00,
 71  0x38,0x1C,0x00,0x38,0x1C,0x00,0x38,0x1C,0x00,0x38,0x1C,0x00,0x38,0x1C,0x00,0x38,
 72  0x18,0x00,0x38,0x38,0x00,0x38,0x38,0x00,0x18,0x30,0x00,0x1C,0x70,0x00,0x1C,0xE0,
 73  0x00,0x0C,0xC0,0x01,0x06,0x80,0x83,0x03,0x00,0xFE,0x00,0x00,0x00,0x00,0x00,0x00,
 74  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"6",6*/
 75 
 76 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
 77  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xF0,0xFF,0x1F,0xF0,0xFF,0x1F,0xF0,0xFF,
 78  0x0F,0x70,0x00,0x0C,0x30,0x00,0x04,0x18,0x00,0x02,0x18,0x00,0x03,0x08,0x00,0x01,
 79  0x00,0x80,0x01,0x00,0x80,0x00,0x00,0xC0,0x00,0x00,0x40,0x00,0x00,0x60,0x00,0x00,
 80  0x60,0x00,0x00,0x30,0x00,0x00,0x30,0x00,0x00,0x18,0x00,0x00,0x18,0x00,0x00,0x18,
 81  0x00,0x00,0x1C,0x00,0x00,0x0C,0x00,0x00,0x0E,0x00,0x00,0x0E,0x00,0x00,0x0E,0x00,
 82  0x00,0x0E,0x00,0x00,0x0F,0x00,0x00,0x0F,0x00,0x00,0x0F,0x00,0x00,0x0F,0x00,0x00,
 83  0x0F,0x00,0x00,0x0F,0x00,0x00,0x0F,0x00,0x00,0x06,0x00,0x00,0x00,0x00,0x00,0x00,
 84  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"7",7*/
 85 
 86 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
 87  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0xFE,0x00,0xC0,0x81,0x03,0xE0,0x00,
 88  0x07,0x70,0x00,0x0E,0x70,0x00,0x0E,0x38,0x00,0x1C,0x38,0x00,0x1C,0x38,0x00,0x1C,
 89  0x38,0x00,0x1C,0x78,0x00,0x1C,0x78,0x00,0x0C,0xF0,0x00,0x0E,0xE0,0x03,0x07,0xE0,
 90  0x87,0x03,0x80,0xDF,0x01,0x00,0x7F,0x00,0xC0,0xFD,0x00,0xE0,0xF0,0x03,0x70,0xC0,
 91  0x07,0x30,0x80,0x07,0x38,0x00,0x0F,0x38,0x00,0x0E,0x1C,0x00,0x1E,0x1C,0x00,0x1C,
 92  0x1C,0x00,0x1C,0x1C,0x00,0x1C,0x1C,0x00,0x1C,0x18,0x00,0x0C,0x38,0x00,0x0E,0x30,
 93  0x00,0x06,0x60,0x00,0x03,0xC0,0x81,0x01,0x00,0x7F,0x00,0x00,0x00,0x00,0x00,0x00,
 94  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"8",8*/
 95 
 96 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
 97  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x3F,0x00,0xC0,0xC1,0x00,0x60,0x80,
 98  0x01,0x70,0x00,0x03,0x38,0x00,0x06,0x38,0x00,0x06,0x18,0x00,0x0E,0x1C,0x00,0x0C,
 99  0x1C,0x00,0x0C,0x1C,0x00,0x1C,0x1C,0x00,0x1C,0x1C,0x00,0x1C,0x1C,0x00,0x1C,0x1C,
100  0x00,0x1C,0x3C,0x00,0x1E,0x38,0x00,0x1F,0x38,0x00,0x1D,0x78,0x80,0x1D,0xF0,0xE1,
101  0x1C,0xE0,0x7F,0x1C,0x80,0x1F,0x1C,0x00,0x00,0x0E,0x00,0x00,0x0E,0x00,0x00,0x0E,
102  0x00,0x00,0x0E,0x00,0x00,0x06,0x00,0x00,0x07,0x70,0x00,0x03,0x70,0x80,0x03,0x70,
103  0xC0,0x01,0x70,0xE0,0x00,0x60,0x70,0x00,0x80,0x1F,0x00,0x00,0x00,0x00,0x00,0x00,
104  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"9",9*/
105 
106 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
107  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
108  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
109  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1C,0x00,0x00,0x3E,0x00,0x00,0x3E,0x00,0x00,
110  0x3E,0x00,0x00,0x1C,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
111  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
112  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x1C,0x00,0x00,
113  0x3E,0x00,0x00,0x3E,0x00,0x00,0x3E,0x00,0x00,0x1C,0x00,0x00,0x00,0x00,0x00,0x00,
114  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*":",10*/
115 
116 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
117  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
118  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
119  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
120  0x00,0x00,0x00,0x00,0x00,0xFC,0xFF,0x3F,0xFC,0xFF,0x3F,0x00,0x00,0x00,0x00,0x00,
121  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
122  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
123  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
124  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*"-",11*/
125 
126 {0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
127  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
128  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
129  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
130  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
131  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
132  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
133  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,
134  0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00},/*" ",12*/
135 };
136 
137 /**
138  * \brief 通过输入的字符获取在字模表的偏移量
139  *
140  * \retval 字母表中有该字符则返回偏移量,否则返回-1
141  */
142 static int __get_tab_offset (char dat)
143 {
144     int i;
145 
146     for (i = 0; i < sizeof(char_fs48) - 1; i++) {
147         if (dat == char_fs48[i]) {
148             return i;
149         }
150     }
151 
152     return -1;
153 }
154 
155 /**
156  * \brief 绘制一个英文字符,48号字体
157  *
158  * \retval 成功返回0,失败返回-1
159  */
160 int draw_one_char_fs48 (unsigned int x, unsigned int y, char dat)
161 {
162     int offset = 0;
163     int i, j;
164     char tmp = 0;
165     
166     offset = __get_tab_offset(dat);
167     if (offset == -1) {
168         return -1;
169     }
170 
171     for (i = 0; i < 48; i++) {
172         tmp = font_size_48[offset][i * 3];
173         for (j = 0 ; j < 8; j++) {
174             if (tmp & 0x01) {
175                 draw_point(x + j, y + i);
176             } else {
177                 draw_bk_point(x + j, y + i);
178             }
179     
180             tmp    >>= 1;
181         }
182         
183         tmp = font_size_48[offset][i * 3 + 1];
184         for (; j < 16; j++) {
185             if (tmp & 0x01) {
186                 draw_point(x + j, y + i);
187             } else {
188                 draw_bk_point(x + j, y + i);
189             }
190     
191             tmp    >>= 1;
192         }
193         
194         tmp = font_size_48[offset][i * 3 + 2];
195         for (; j < 24; j++) {
196             if (tmp & 0x01) {
197                 draw_point(x + j, y + i);
198             } else {
199                 draw_bk_point(x + j, y + i);
200             }
201     
202             tmp    >>= 1;
203         }
204     }
205     
206     return 0;
207 }

main.c

  1 #include <stdio.h>
  2 #include <time.h>
  3 #include <sys/time.h>
  4 #include <pthread.h>
  5 #include <sys/types.h>
  6 #include <sys/stat.h>
  7 #include <fcntl.h>
  8 #include <stdlib.h>
  9 #include "draw.h"
 10 #include "figure.h"
 11 
 12 /* 定义按键值 */
 13 #define KEY_OK        0
 14 #define KEY_LEFT    2
 15 #define KEY_RIGHT    3
 16 #define KEY_UP        4
 17 #define KEY_DOWN    5
 18 
 19 typedef struct struct_cursor {
 20     unsigned int x1;
 21     unsigned int y1;
 22     unsigned int x2;
 23     unsigned int y2;
 24     
 25     unsigned int offset_pix;    /* 一个字符所占的像素点 */
 26     int last_offset;            /* 上次偏移量 */
 27 } cursor_t;
 28 cursor_t __g_cursor;
 29 
 30 unsigned int __g_show_x, __g_show_y;    /* 时钟显示的位置 */
 31 
 32 #define MAX_CURSOR_OFFSET    5    /* 最大光标偏移量 */
 33  
 34 /**
 35  * \brief 初始化光标
 36  *
 37  * \param[in] (x1, y1), (x2, y2):第一个游标的对角线坐标
 38  */
 39 void init_cursor (unsigned int x1, 
 40                   unsigned int y1, 
 41                   unsigned int x2, 
 42                   unsigned int y2)
 43 {
 44     __g_cursor.x1 = x1;
 45     __g_cursor.x2 = x2;
 46     __g_cursor.y1 = y1;
 47     __g_cursor.y2 = y2;
 48 
 49     if (x2 > x1) {
 50         __g_cursor.offset_pix = x2 - x1 + 1;
 51     } else {
 52         __g_cursor.offset_pix = x1 - x2 + 1;
 53     }
 54 
 55     //__g_cursor.last_offset = MAX_CURSOR_OFFSET;    /* 光标初始在最右 */
 56     __g_cursor.last_offset = 0;    /* 光标初始在最左 */
 57 }
 58 
 59 /**
 60  * \brief 绘制光标
 61  *
 62  * \param[in] offset:光标偏移量,以字符为单位
 63  */
 64 void draw_cursor (unsigned int offset)
 65 {
 66     if (offset == 0) {
 67         draw_rectangle(__g_cursor.x1,
 68                        __g_cursor.y1,
 69                        __g_cursor.x2 + __g_cursor.offset_pix * 3,
 70                        __g_cursor.y2);
 71         draw_rectangle(__g_cursor.x1 + 1,
 72                        __g_cursor.y1 + 1,
 73                        __g_cursor.x2 + __g_cursor.offset_pix * 3 - 1,
 74                        __g_cursor.y2 - 1);
 75     } else {
 76         draw_rectangle(__g_cursor.x1 + __g_cursor.offset_pix * (2 + 3 * offset),
 77                        __g_cursor.y1,
 78                        __g_cursor.x1 + __g_cursor.offset_pix * (4 + 3 * offset),
 79                        __g_cursor.y2);
 80         draw_rectangle(__g_cursor.x1 + __g_cursor.offset_pix * (2 + 3 * offset) + 1,
 81                        __g_cursor.y1,
 82                        __g_cursor.x1 + __g_cursor.offset_pix * (4 + 3 * offset) - 1,
 83                        __g_cursor.y2);
 84     }
 85 }
 86 
 87 /**
 88  * \brief 删除光标
 89  *
 90  * \param[in] offset:光标偏移量,以字符为单位
 91  */
 92 void delete_cursor (unsigned int offset)
 93 {
 94     color_t old_pen_color;
 95 
 96     old_pen_color = get_pen_color();
 97     set_pen_color(get_bk_color());
 98 
 99     draw_cursor(offset);
100     set_pen_color(old_pen_color);
101 }
102 
103 /**
104  * \brief 移动光标
105  *
106  * \param[in] dir:光标移动方向,-1左移一位,1右移一位,其他值无效
107  */
108 void move_cursor (int dir)
109 {
110     if ((dir == -1) && (__g_cursor.last_offset != 0)) {
111         delete_cursor(__g_cursor.last_offset);
112         __g_cursor.last_offset--;
113         draw_cursor(__g_cursor.last_offset);
114     } else if ((dir == 1) && (__g_cursor.last_offset != MAX_CURSOR_OFFSET)) {
115         delete_cursor(__g_cursor.last_offset);
116         __g_cursor.last_offset++;
117         draw_cursor(__g_cursor.last_offset);
118     }
119 }
120 
121 /**
122  * \brief 调整时间
123  *
124  * \param[in] adj:为1表示加1,为-1表示减1,其他值无效
125  */
126 void adj_time (int adj)
127 {
128     struct timeval  tv;
129     struct timezone tz;
130     struct tm *p_tm = NULL;
131 
132     if ((adj != 1) && (adj != -1)) {
133         return;
134     }
135 
136     gettimeofday(&tv, &tz);
137     p_tm = localtime(&tv.tv_sec);
138     if (p_tm == NULL) {
139         return;
140     }
141 
142     switch (__g_cursor.last_offset) {
143     case 5:     /**/
144         p_tm->tm_sec += adj;
145     break;
146 
147     case 4:     /**/
148         p_tm->tm_min += adj;
149     break;
150         
151     case 3:     /**/
152         p_tm->tm_hour += adj;
153     break;
154 
155     case 2:        /**/
156         p_tm->tm_mday += adj;
157     break;
158 
159     case 1:        /**/
160         p_tm->tm_mon += adj;
161     break;
162 
163     case 0:        /**/
164         p_tm->tm_year += adj;
165     break;
166     }
167 
168     tv.tv_sec = mktime(p_tm);
169 
170     settimeofday(&tv, &tz);
171 }
172 
173 /**
174  * \brief 显示时间
175  */
176 void show_time (void)
177 {
178     char show_buf[32] = { 0 };
179     time_t total_sec;
180     struct tm *p_tm = NULL;
181     static int last_sec = 60;    /* 赋一个不可能达到的秒数值,确保第一秒能显示 */
182     
183     time(&total_sec);
184     p_tm = localtime(&total_sec);
185     if (p_tm == NULL) {
186         return;
187     }
188         
189     if (last_sec != p_tm->tm_sec) {
190         last_sec = p_tm->tm_sec;
191 
192         sprintf(show_buf,
193                 "%4d-%02d-%02d %02d:%02d:%02d",
194                 p_tm->tm_year + 1900,                
195                 p_tm->tm_mon  + 1,
196                 p_tm->tm_mday,
197                 p_tm->tm_hour,                
198                 p_tm->tm_min,
199                 p_tm->tm_sec);
200         
201         draw_str(__g_show_x, __g_show_y, show_buf);
202 
203         draw_cursor(__g_cursor.last_offset);
204     }
205 }
206 
207 /**
208  * \brief 按键处理函数
209  */
210 void key_handler (int key_val)
211 {
212     switch (key_val) {
213         case KEY_LEFT:
214             move_cursor(-1);
215         break;
216 
217         case KEY_RIGHT:
218             move_cursor(1);
219         break;
220 
221         case KEY_UP:
222             adj_time(1);
223             show_time();
224         break;
225 
226         case KEY_DOWN:
227             adj_time(-1);
228             show_time();
229         break;
230 
231         case KEY_OK:
232             exit(1);
233         break;
234     }
235 }
236 
237 /**
238  * \brief 该线程负责检测按键动作并移动光标,同时记录光标位置
239  */
240 void * cursor_pthread (void *p_arg)
241 {
242     int fd;
243     int key_val;
244     int ret;
245 
246     fd = open("/dev/mykey", O_RDWR);
247     if (fd == -1) {
248         printf("fail to open\n");
249     }
250 
251     while (1) {
252         read(fd, &key_val, 4);    /* 阻塞 */
253         key_handler(key_val);
254     }
255 }
256 
257 
258 int main(int argc, const char *argv[])
259 {
260     //int i = 0;
261     font_size_t font_size = FONT_SIZE_48;
262     int x, y;
263     pthread_t id;
264 
265     if (framebuffer_init()) {
266         return 0;
267     }
268 
269     full_screen(LCD_YELLOW);
270     set_pen_color(LCD_RED);
271     set_bk_color(LCD_WHITE);
272 
273     set_font_size(font_size);
274 
275     x = (480 - 19 * font_size / 2) / 2;    /* 此处的19为要显示的19个字符 */
276     if (x < 0) {
277         x = 0;
278     }
279     y = (272 - font_size) / 2;
280     __g_show_x = x;
281     __g_show_y = y;
282 
283     init_cursor(x, y, x + font_size / 2 - 1, y + font_size - 1);
284 
285     pthread_create(&id, NULL, cursor_pthread, NULL);
286 
287     while (1) {
288         show_time();
289 
290         //usleep(200 * 1000);
291         sleep(1);
292     }
293 
294     return 0;
295 }

Makefile

 1 CC = arm-none-linux-gnueabi-
 2 OBJS = main.o draw.o font_size_16.o font_size_24.o font_size_32.o \
 3        font_size_48.o font_size_64.o figure.o
 4 OBJ = a.out
 5 $(OBJ):$(OBJS)
 6     $(CC)gcc $^ -lpthread -o $@
 7 
 8 %.o:%.c
 9     $(CC)gcc -c $< -o $@
10 
11 .PHONY: 
12 clean:
13     rm $(OBJ) *.o
14 
15 cp:
16     sudo cp $(OBJ) ../../rootfs/home/test

figure.c

  1 #include "draw.h"
  2 
  3 /**
  4  * \brief 指定两点画一条直线
  5  */
  6 void draw_line (unsigned int x1, unsigned int y1, unsigned int x2, unsigned int y2)
  7 {
  8     int t; 
  9     int xerr=0, yerr=0, delta_x, delta_y, distance; 
 10     int incx, incy, urow, ucol; 
 11 
 12     delta_x = x2 - x1;                             /* 计算坐标增量 */
 13     delta_y = y2 - y1; 
 14     urow = x1;
 15     ucol = y1;
 16     
 17     if(delta_x > 0) {
 18         incx = 1;                     /* 设置单步方向 */
 19     } else if (delta_x == 0) {
 20         incx = 0;                    /* 垂直线 */
 21     } else {
 22         incx = -1; 
 23         delta_x = -delta_x;
 24     }
 25     
 26     if (delta_y > 0) {
 27         incy = 1; 
 28     } else if (delta_y == 0) {
 29         incy = 0;                    /* 水平线 */
 30     } else {
 31         incy = -1; 
 32         delta_y = -delta_y;
 33     } 
 34     
 35     if (delta_x > delta_y) {
 36         distance = delta_x;         /* 选取基本增量坐标轴 */
 37     } else {
 38         distance = delta_y; 
 39     }
 40     
 41     for (t = 0; t <= distance + 1; t++)    {        /* 画线输出 */
 42         draw_point(urow , ucol);                /* 画点 */
 43         
 44         xerr += delta_x ; 
 45         yerr += delta_y ; 
 46         if (xerr > distance) { 
 47             xerr -= distance; 
 48             urow += incx; 
 49         } 
 50         if (yerr > distance) {
 51             yerr -= distance; 
 52             ucol += incy; 
 53         }
 54     }
 55 }
 56 
 57 /**
 58  * \brief 绘制一个矩形
 59  *
 60  * \param[in] (x1, y1),(x2, y2):对角坐标
 61  */
 62 void draw_rectangle (unsigned int x1, unsigned int y1, unsigned int x2, unsigned int y2)
 63 {
 64     int i , temp;
 65 
 66     if (x1 > x2)    {
 67         temp = x1;    
 68         x1 = x2; 
 69         x2 = temp;
 70     }
 71     if (y1 > y2) {
 72         temp = y1;
 73         y1 = y2; 
 74         y2 = temp;
 75     }
 76 
 77     for (i = x1; i <= x2; i++) {
 78         draw_point(i , y1);
 79         draw_point(i , y2);
 80     }
 81     for(i = y1 + 1; i < y2; i++) {
 82         draw_point(x1 , i);
 83         draw_point(x2 , i);
 84     }
 85 }
 86 
 87 
 88 /**
 89  * \brief 绘制实心一个三角形
 90  *
 91  * \param[in] x, y:三角形顶点的坐标
 92  * \param[in] layers:组成三角形的横线数目
 93  */
 94 void draw_triangle (unsigned int x, unsigned int y, unsigned int layers)
 95 {
 96     int line_x1, line_x2, line_y;
 97     int i;
 98     
 99     if (layers > 1) {
100         draw_point(x, y);
101         layers++;
102         line_x1 = x - 1;
103         line_x2 = x + 1;
104         line_y = y + 1;
105         
106         for (i = 0; i < layers; i++) {
107             draw_line(line_x1, line_y, line_x2, line_y);
108             
109             line_x1--;
110             line_x2++;
111             line_y++;
112         }
113     }
114 }
115 
116 /**
117  * \brief 绘制一个实心倒三角形
118  *
119  * \param[in] x, y:三角形顶点的坐标
120  * \param[in] layers:组成三角形的横线数目
121  */
122 void draw_inverted_triangle (unsigned int x, unsigned int y, unsigned int layers)
123 {
124     int line_x1, line_x2, line_y;
125     int i;
126     
127     if (layers > 1) {
128         draw_point(x, y);
129         layers--;
130         line_x1 = x - 1;
131         line_x2 = x + 1;
132         line_y = y - 1;
133         
134         for (i = 0; i < layers; i++) {
135             draw_line(line_x1, line_y, line_x2, line_y);
136             
137             line_x1--;
138             line_x2++;
139             line_y--;
140         }
141     }
142 }

figure.h

 1 /**
 2  * \brief 指定两点画一条直线
 3  */
 4 extern void draw_line (unsigned int x1, unsigned int y1, unsigned int x2, unsigned int y2);
 5 
 6 /**
 7  * \brief 绘制一个矩形
 8  *
 9  * \param[in] (x1, y1),(x2, y2):对角坐标
10  */
11 extern void draw_rectangle (unsigned int x1, unsigned int y1, unsigned int x2, unsigned int y2);
12 
13 /**
14  * \brief 绘制实心一个三角形
15  *
16  * \param[in] x, y:三角形顶点的坐标
17  * \param[in] layers:组成三角形的横线数目
18  */
19 extern void draw_triangle (unsigned int x, unsigned int y, unsigned int layers);
20 
21 /**
22  * \brief 绘制一个实心倒三角形
23  *
24  * \param[in] x, y:三角形顶点的坐标
25  * \param[in] layers:组成三角形的横线数目
26  */
27 extern void draw_inverted_triangle (unsigned int x, unsigned int y, unsigned int layers);

在使用时要先将用 insmod key.ko将第十二章编写好的按键驱动加载到内核中,然后运行a.out,通过按键就可以调整时间了。调整时间的关键是main.c中第126行的adj_time函数,该函数在switch语句里面直接对p_tm里的值进行加一或减一操作,无需检查数据是否符合要求。比如平年的2月是没有29号的,将2月29号用mktime函数转换得到的自动就是3月1号的值。