一、前言
png图片是无损压缩格式,它有一个比较重要的特点是可以让特定区域透明。要显示png图片需要调用png库留出的函数接口,png库留出的函数接口很多,但是帮助手册里没有详细的说明各个函数的用法,网上也没找到太多资料,因此在这里只介绍一种能用的方法。有些函数用法在代码注释里写着。
二、代码
1 /** 2 * filename: png.c 3 * author: Suzkfly 4 * date: 2021-08-14 5 * platform: S3C2416或Ubuntu 6 * 需要提前准备好lv.png这张图片,编译时要加-lpng参数,并且程序运行平台需要有png的库 7 */ 8 #include <stdio.h> 9 #include <stdlib.h> 10 #include <string.h> 11 #include <sys/stat.h> 12 #include <sys/types.h> 13 #include <unistd.h> 14 #include <fcntl.h> 15 #include <errno.h> 16 #include <linux/fb.h> 17 #include <sys/mman.h> 18 #include <png.h> 19 20 /**< \brief 定义每个像素点对应的位数,如果是565的屏则为16,如果是888的屏则为32 */ 21 #define BITS_PER_PIXEL 32 22 //#define BITS_PER_PIXEL 16 23 24 /**< \brief 根据实际情况修改,此处为unsigned short是565的屏,根据程序打印出的 25 bits_per_pixel的值可以判断出输出格式是565还是888 */ 26 #if (BITS_PER_PIXEL == 32) 27 typedef unsigned int color_t; 28 #elif (BITS_PER_PIXEL == 16) 29 typedef unsigned short color_t; 30 #endif 31 32 static struct fb_var_screeninfo __g_vinfo; /* 显示信息 */ 33 color_t *__gp_frame; /* 虚拟屏幕首地址 */ 34 35 /* framebuffer初始化 */ 36 int framebuffer_init (void) 37 { 38 int fd = 0; 39 40 fd = open("/dev/fb0", O_RDWR); 41 if (fd == -1) { 42 perror("fail to open /dev/fb0\n"); 43 return -1; 44 } 45 46 /* 获取显示信息 */ 47 ioctl(fd, FBIOGET_VSCREENINFO, &__g_vinfo); /* 获取显示信息 */ 48 printf("bits_per_pixel = %d\n", __g_vinfo.bits_per_pixel); /* 得到一个像素点对应的位数 */ 49 printf("xres_virtual = %d\n", __g_vinfo.xres_virtual); /* 打印虚拟屏幕列数 */ 50 printf("yres_virtual = %d\n", __g_vinfo.yres_virtual); /* 打印虚拟屏幕行数 */ 51 printf("xres = %d\n", __g_vinfo.xres); /* 打印屏幕列数 */ 52 printf("yres = %d\n", __g_vinfo.yres); /* 打印屏幕行数 */ 53 54 /* 设置显示信息 */ 55 /* 56 __g_vinfo.xres_virtual = 2048; 57 __g_vinfo.yres_virtual = 1920; 58 __g_vinfo.xres = 800; 59 __g_vinfo.yres = 600; 60 if (ioctl(fd, FBIOPUT_VSCREENINFO, &__g_vinfo)) { 61 perror ("Get VScreen Info Error\n"); 62 exit (1); 63 } 64 */ 65 66 /* 获取显示信息 */ 67 //ioctl(fd, FBIOGET_VSCREENINFO, &__g_vinfo); /* 获取显示信息 */ 68 //printf("bits_per_pixel = %d\n", __g_vinfo.bits_per_pixel); /* 得到一个像素点对应的位数 */ 69 //printf("xres_virtual = %d\n", __g_vinfo.xres_virtual); /* 打印虚拟屏幕列数 */ 70 //printf("yres_virtual = %d\n", __g_vinfo.yres_virtual); /* 打印虚拟屏幕行数 */ 71 //printf("xres = %d\n", __g_vinfo.xres); /* 打印屏幕列数 */ 72 //printf("yres = %d\n", __g_vinfo.yres); /* 打印屏幕行数 */ 73 74 __gp_frame = mmap(NULL, /* 映射区的开始地址,为NULL表示由系统决定映射区的起始地址 */ 75 __g_vinfo.xres_virtual * __g_vinfo.yres_virtual * __g_vinfo.bits_per_pixel / 8, /* 映射区大小 */ 76 PROT_WRITE | PROT_READ, /* 内存保护标志(可读可写) */ 77 MAP_SHARED, /* 映射对象类型(与其他进程共享) */ 78 fd, /* 有效的文件描述符 */ 79 0); /* 被映射内容的偏移量 */ 80 if (__gp_frame == NULL) { 81 perror("fail to mmap\n"); 82 return -1; 83 } 84 85 return 0; 86 } 87 88 /** 89 * \brief 填充整屏 90 */ 91 void full_screen (color_t color) 92 { 93 int i; 94 color_t *p = __gp_frame; 95 96 for (i = 0; i < __g_vinfo.xres_virtual * __g_vinfo.yres_virtual; i++) { 97 *p++ = color; 98 } 99 } 100 101 /** 102 * \brief 清屏 103 */ 104 void clear() 105 { 106 full_screen(0); 107 } 108 109 /** 110 * \brief 画点 111 */ 112 void draw_point(int x, int y, color_t color) 113 { 114 color_t *p = __gp_frame; 115 116 p += __g_vinfo.xres_virtual * y + x; 117 *p = color; 118 } 119 120 /** 121 * \brief 在指定位置显示PNG图片 122 * 123 * \param[in] x:x坐标 124 * \param[in] y:y坐标 125 * \param[in] file_path:file_path图片文件路径 126 * 127 * \retval 成功返回0,失败返回-1 128 */ 129 int Show_PNG(int x, int y, const char *file_path) 130 { 131 png_structp png_ptr; /* 该结构可通过png_create_read_struct函数得到,作为内部结构,会作为参数传给其他函数,一般不会被用户使用 */ 132 png_infop info_ptr; /* 用于保存png图片的信息,直接访问容易导致程序出问题,因此如果想要获取png图片信息最好调用库提供的函数接口 */ 133 int color_type = 0; 134 int image_width, image_height; 135 char sig[8] = { 0 }; /* 8个字节用来存储png文件头 */ 136 int i, j; 137 int R, G, B, A = 0; 138 FILE *p_fp; 139 png_bytep *row_pointers; /* 指针数组,数组中的每一个指针都指向一行图像数据 */ 140 int bytes_per_pix = 0; /* 一个像素点占用的字节数 */ 141 color_t color; 142 143 /* 打开png文件 */ 144 if ((p_fp = fopen(file_path, "rb")) == NULL) { 145 fprintf(stderr, "%s open fail!!!\n", file_path); 146 return -1; 147 } 148 149 /* 把文件前8个字节读出来,用来判断该文件是不是png文件,这8个字节必须是: 150 89 50 4e 47 0d 0a 1a 0a。如果使用了png_sig_cmp函数来判断文件头,那么后面 151 要读取数据之前要先调用png_set_sig_bytes,以告知文件头信息已经读取过了。 152 png_sig_cmp传递的第1个参数为需要检查的文件头数据指针,检查范围为 153 sig[第2个参数]~sig[第2个参数+第3个参数-1] */ 154 fread(sig, 1, sizeof(sig), p_fp); 155 if (0 != png_sig_cmp(sig, 0, sizeof(sig))) { 156 fprintf(stderr, "The file is not PNG\n"); 157 return -1; 158 } 159 160 /* 创建一个png_structp结构体,第1个参数为png库的版本,直接传入 161 PNG_LIBPNG_VER_STRING宏即可,第2个参数为给出错处理函数(即第3个参数)传递 162 的指针,第3个参数为出错处理函数的指针,第4个参数为警告处理函数的指针。 163 如果后3个参数传NULL,则表示使用系统默认的处理函数 */ 164 png_ptr = png_create_read_struct(PNG_LIBPNG_VER_STRING, NULL, NULL, NULL); 165 if (png_ptr == NULL) { 166 fclose(p_fp); 167 return -1; 168 } 169 170 /* 创建一个info_ptr,必须的 */ 171 info_ptr = png_create_info_struct(png_ptr); 172 if (info_ptr == NULL) { 173 fclose(p_fp); 174 png_destroy_read_struct(&png_ptr, NULL, NULL); 175 return -1; 176 } 177 178 /* 如果上面png_create_read_struct没有自定义错误处理,这里是必须的 */ 179 if (setjmp(png_jmpbuf(png_ptr))) { 180 png_destroy_read_struct(&png_ptr, &info_ptr, NULL); 181 fclose(p_fp); 182 183 return -1; 184 } 185 186 /* 初始化文件IO,相当于将文件流指针和png_ptr绑定了 */ 187 png_init_io(png_ptr, p_fp); 188 189 /* 如果前面调用了png_sig_cmp,那么就要调用本函数,否则程序在运行时会报错 */ 190 png_set_sig_bytes(png_ptr, sizeof(sig)); 191 192 /* 读取png图片信息。该函数相当于调用了png_read_info(),png_read_image()和 193 png_read_end()。如果调用了该函数,那么就不该再调用任何png_set_transform() 194 函数。 195 PNG_TRANSFORM_IDENTITY No transformation 196 PNG_TRANSFORM_STRIP_16 Strip 16-bit samples to 197 8 bits 198 PNG_TRANSFORM_STRIP_ALPHA Discard the alpha channel(丢弃透明度通道) 199 PNG_TRANSFORM_PACKING Expand 1, 2 and 4-bit 200 samples to bytes 201 PNG_TRANSFORM_PACKSWAP Change order of packed 202 pixels to LSB first 203 PNG_TRANSFORM_EXPAND Perform set_expand() 204 PNG_TRANSFORM_INVERT_MONO Invert monochrome images 205 PNG_TRANSFORM_SHIFT Normalize pixels to the 206 sBIT depth 207 PNG_TRANSFORM_BGR Flip RGB to BGR, RGBA 208 to BGRA 209 PNG_TRANSFORM_SWAP_ALPHA Flip RGBA to ARGB or GA 210 to AG 211 PNG_TRANSFORM_INVERT_ALPHA Change alpha from opacity 212 to transparency 213 PNG_TRANSFORM_SWAP_ENDIAN Byte-swap 16-bit samples 214 215 最后一个参数没有使用,传NULL即可*/ 216 png_read_png(png_ptr, info_ptr, PNG_TRANSFORM_EXPAND, NULL); 217 //png_read_png(png_ptr, info_ptr, PNG_TRANSFORM_EXPAND | PNG_TRANSFORM_STRIP_ALPHA, NULL); 218 219 if ((png_ptr != NULL) && (info_ptr != NULL)) { 220 /* 获取图像的宽高 */ 221 image_width = png_get_image_width(png_ptr, info_ptr); 222 image_height = png_get_image_height(png_ptr, info_ptr); 223 printf("width:%d,height:%d\n", image_width, image_height); 224 225 /* 获取图像颜色类型。如果在png_read_png中选择丢弃透明度通道,那么 226 color_type的值就是PNG_COLOR_TYPE_RGB,否则是PNG_COLOR_TYPE_RGB_ALPHA */ 227 color_type = png_get_color_type(png_ptr, info_ptr); 228 printf("color type:%d\n", color_type); 229 } 230 231 /* 获取所有的图像数据 */ 232 row_pointers = png_get_rows(png_ptr, info_ptr); 233 234 /* RGB格式 */ 235 if ((color_type == PNG_COLOR_TYPE_RGB) || (color_type == PNG_COLOR_TYPE_RGB_ALPHA)) { 236 if (color_type == PNG_COLOR_TYPE_RGB) { 237 bytes_per_pix = 3; 238 } else if (color_type == PNG_COLOR_TYPE_RGB_ALPHA) { 239 bytes_per_pix = 4; 240 } 241 242 for(i = 0; i < image_height; i++) { 243 for(j = 0; j < (image_width * bytes_per_pix); j += bytes_per_pix) { 244 R = row_pointers[i][j + 0]; 245 G = row_pointers[i][j + 1]; 246 B = row_pointers[i][j + 2]; 247 #if (BITS_PER_PIXEL == 32) 248 if (color_type == PNG_COLOR_TYPE_RGB_ALPHA) { 249 A = row_pointers[i][j + 3]; 250 } 251 /* 对于RGB888的显示屏来说,如果支持透明度,那么传入的颜色数据的高 252 8位表示的就是透明度 */ 253 color = (A << 24) | (R << 16) | (G << 8) | B; 254 #elif (BITS_PER_PIXEL == 16) 255 R >>= 3; 256 G >>= 2; 257 B >>= 3; 258 color = (R << 11) | (G << 5) | B; 259 #endif 260 draw_point(j / bytes_per_pix + x, i + y, color); 261 } 262 } 263 } 264 265 /* 清理图像,释放内存 */ 266 png_destroy_read_struct(&png_ptr, &info_ptr, 0); 267 268 fclose(p_fp); 269 270 return 0; 271 } 272 273 274 int main(int argc, const char *argv[]) 275 { 276 if (framebuffer_init()) { 277 return 0; 278 } 279 280 /* 清屏 */ 281 clear(); 282 283 Show_PNG(100, 100, "lv.png"); 284 285 return 0; 286 }
说明:lv.png这张图片需要提前准备好,编译时要加-lpng参数。
三、问题和总结
1. 如果程序运行时报错:
error while loading shared libraries: libpng.so.3: cannot open shared object file: No such file or directory
说明运行程序的平台上缺少png库,需要去下载,并将下载好的库复制到/lib目录下。
我下载的库在这里:libpng12.tar.gz
2. 如果显示的图片有透明区域,那么图片显示出来会有一些黑色区域(不一定是框框,反正就是大片的黑色区域),如下图(右),左边是我在电脑上打开后显示的样子。
为了解决这个问题,我在网上搜了很多资料,发现确实有很多网友遇到过这样的问题,但他们都是用别的平台或者语言。在linux下有很多显示png图片的资料,但是却没有人提到过这个问题。
有意思的是,我将这张图片通过微信发出去,在聊天窗口中显示的也是带黑框的(同上图右相同),但是将图片点开,并点击查看原图,黑框就会消失。这至少说明这是一个普遍存在并且可以解决的问题,希望以后的某一天我也能找到解决办法。