aLoNe IT Develop Studio

  • 首页
  • IT前沿
  • Web技术
  • 小姿势
  • 运维管理
  • 随笔
欢迎来到aLoNe.Adams.K的个人博客
  1. 首页
  2. 小姿势
  3. 正文

常见颜色深度的 bitmap 之间的相互转换

2016年9月23日 3631点热度 0人点赞 0条评论
作者: 刘鹏
日期: 2009-08-14
本文介绍了常见色深的格式转换原理和算法。

http://www.linuxgraphics.cn/graphics/image_processing_color_depth_convert.html

简介

在很多场合中会遇到不同颜色深度的 surface 之间的转化问题,如笔者在开发 一个虚拟 framebuffer 时就需要将客户 GUI 系统的颜色深度转换成虚拟 framebuffer 给定的颜色深度,根据工作中的实际经验,笔者总结了在几对典型 的颜色深度之间进行转换的思路和实现方法,读者在此基础可以举一反三,处理 更多的转换。

16 位色转成 24 位色

16位色有两种格式,一种是565, 另一种是555,本文以 565 格式为例。

对于 16 位色的 surface,一个像素由两个字节表示,其中 R 颜色分量占 5 位, G 颜色分量占6 位,B 颜色分量占 5 位。

24 位色的 surface,一个像素由三个字节表示,R、G、B 分量各占 8 位。

要将 16 位色转换成 24 位色,需要将565布局的 R、B、G 分量取出来,分别填 充到三个字节中去,注意要填充时放在高位,末位补 0 ,否则误差会放大,分 拆情况如下图所示:

图1
图1

实现转换的代码如下所示:

for (row = 0;row < src_buffer_height; row++) {
        sp = src_buf + src_buf_pitch * row;
	dp = dst_buf + dst_buf_pitch * row;

        for (col = 0; col < src_buffer_width; col++) {
                word = *((Uint16*)sp);
                sp+=2;

                *dp++ = ((word & 0xF800) >> 11) << 3; /* red */
                *dp++ = ((word & 0x07E0) >> 5) << 2;  /* green */
                *dp++ = (word & 0x001F) << 3;         /* blue */
        }
}

4 位色转为 8 位色

对于 4 位色 surface,每个像素由 4 位表示,每个字节可表示两个像素;

对于 8 位色 surface,每个像素由 8 位表示,每个字节表示一个像素;

要将 4 位色转换成 8 位色,需要一个字节的高 4 位和低 4 位分别转成一个字 节,如下图所示:

图2
图2

完成转化的示例代码如下所示:

for (row = 0;row < src_buffer_height; row++) {
        sp = src_buf + src_buf_pitch * row;
	dp = dst_buf + dst_buf_pitch * row;

        for (col = 0; col < src_buffer_width; col++) {
                bit = (col% 2) << 2;
                if (bit == 0)
                c = *sp++;
                *dp++ = (c >> bit) & 0x0f;
        }
}

2 位色转为 8 位色

对于 2 位色 surface,每个像素由 2 位表示,每个字节可表示四个像素;

对于 8 位色 surface,每个像素由 8 位表示,每个字节表示一个像素;

要将 2 位色转换成 8 位色,需要一个字节按两位为一个单元分拆,然后将每个 单元填充成一个字节,如下图所示:

图3
图3

转换的实现代码如下所示:

for (row = 0;row < src_buffer_height; row++) {
        sp = src_buf + src_buf_pitch * row;
	dp = dst_buf + dst_buf_pitch * row;

        for (col = 0; col < src_buffer_width; col++) {
                bit = (col% 4) << 1;
                if (bit == 0)
                c = *sp++;
                *dp++ = (c >> bit) & 0x03;
        }
}

1 位色转为 8 位色

对于 1 位色 surface,每个像素由 1 位表示,每个字节可表示八个像素;

对于 8 位色 surface,每个像素由 8 位表示,每个字节表示一个像素;

要将 1 位色转换成 8 位色,需要将一个字节逐位分拆并将其填充成一个字节,如下图所示:

图4
图4

实现代码如下所示:

for (row = 0;row < src_buffer_height; row++) {
        sp = src_buf + src_buf_pitch * row;
	dp = dst_buf + dst_buf_pitch * row;

        for (col = 0; col < src_buffer_width; col++) {
                bit = col %8;
                if (bit == 0)
                c = *sp++;
                *dp++ = (c >> bit) & 0x01;
        }
}
标签: 暂无
最后更新:2016年9月23日

aLoNe.Adams.K

一只胖菜鸟!

点赞
< 上一篇
下一篇 >

文章评论

razz evil exclaim smile redface biggrin eek confused idea lol mad twisted rolleyes wink cool arrow neutral cry mrgreen drooling persevering
取消回复
文章分类
  • IT前沿 / 3篇
  • Web技术 / 2篇
  • 小姿势 / 24篇
  • 运维管理 / 5篇
  • 随笔 / 3篇
标签聚合
google CentOS 反向代理 debian 虚表 Yum sqlite Orange Pi
最新 热点 随机
最新 热点 随机
团队开发规范培训(代码管理篇) 团队开发规范培训(序) 忆 我的座右铭 DevilBox全程操作记录 香橙派Orange Pi Zero基于Debian进行反向代理配置
Git下获取更新文件方法 忆 关于Framaroot的PC端控制方法 团队开发规范培训(代码管理篇) Avalon2高德地图组件 DevilBox全程操作记录
友情连接
  • 李建伟博客
  • 不忘初心的简书
归档
  • 2022年12月 / 5篇
  • 2022年6月 / 1篇
  • 2022年2月 / 3篇
  • 2020年11月 / 1篇
  • 2018年12月 / 1篇
  • 2018年5月 / 1篇
  • 2018年3月 / 1篇
  • 2017年11月 / 1篇
  • 2017年8月 / 1篇
  • 2017年3月 / 2篇
  • 2017年1月 / 1篇
  • 2016年9月 / 16篇

COPYRIGHT © 2022 aLoNe IT Develop Studio. ALL RIGHTS RESERVED.

Theme Kratos Made By Seaton Jiang

陕ICP备19021656号-2