From: http://blog.csdn.net/snowstart/archive/2005/03/22/326715.aspx
(SnowStart于2005年3月22日)
位图是windows 图形编程中非常重要的一个方面。在进行普通的位图操作中,如GDI 函数BitBlt ,StretchBlt , StretchDIBits ,都会用到一个光栅操作码,即ROP 码,像SRCCOPY ,PATPAINT ,SRCAND 等,由于最近在开发图形驱动,涉及了许多的ROP2 ,ROP3 和ROP4码 的操作,对ROP 码进行了深入的研究,以下详细介绍之以和大家分享。
二元光栅操作: 我们在使用GDI 画线和填充区域时,GDI 使用二元光栅操作码ROP2 组合画笔或画刷像素和目标像素以得到新的目标像素。如SetROP2 函数和GetROP2 函数支持16 种二元光栅操作,如:( 具体见wingdi.h)
#define R2_NOT 6 // Dn
#define R2_XORPEN 7 // DPx
三元光栅操作: 对于图像有同样的光栅操作用于生成各种特殊效果,我们要处理的有三种像素,源图像像素、目标图像像素和画刷像素(模板图像像素),称之为三元光栅操作,使用的是ROP3码 ,如:( 更多的参见wingdi.h)
#define SRCPAINT (DWORD)0x00EE0086 // dest = source OR dest
#define SRCAND (DWORD)0x008800C6 // dest = source AND dest
四元光栅操作: 是混合了源图像像素,目标图像像素和模板画刷像素外,又增加了一个掩码位图,用到4 个变量形成了四元光栅操作,相应的为ROP4码 ,GDI 函数中MaskBlt 函数使用的是ROP4码 ,也是唯一接受四元光栅操作的API 函数
光栅操作的编码:
一个字节可以编码256 种光栅操作,假定P 为画笔或画刷的位,S 为源图像的位,D 为目标图像的位。如果操作的结果和P 一样,编码为0xF0 ,如果操作的结果和S 一样,编码为0xCC ,如果操作的结果和D 一样,编码为0xAA 。
Const BYTE rop_P =0xF0; // 1 1 1 1 0 0 0 0
Const BYTE rop_S =0xCC; // 1 1 0 0 1 1 0 0
Const BYTE rop_D =0xAA; // 1 0 1 0 1 0 1 0
所有其他的光栅操作可以基于这三个常量的布尔操作,如定义源图像S 和画刷P 的逻辑与AND ,计算rop_S&rop_P=0xC0 即可
GDI 中光栅操作实际上使用32 位的DWORD 编码的,而不是简单的0-255 单个字节编码,双字节中高8 位表示上述的256 种单字节光栅操作编码的一种,低字节16 位定义了光栅操作的运算式的编码。
低字节16 位被分成两部分:操作符和操作数,具体如下:
低字节高11 位表示操作符
两位表示一个逻辑操作(00-NOT ,01-XOR ,10-OR ,11-AND ),共5 种逻辑操作,剩下一位表示最后的操作是否为NOT 操作
低字节低5 位表示操作数
前3 位指定分析串,后2 位指定分析串的偏移(8 种分析串如下)
000:SPDDDDDD
001:SPDSPDSP
010:SDPSDPSD
011:DDDDDDDD
100:DDDDDDDD
101:S+SP-DSS
110:S+SP-PDS
111:S+SD-PDS
分析串中的“+ ”和“- ”被称为特殊操作数。在256 种光栅操作中有16 种复杂的操作无法用单个累加器的机制表示,需要临时存储操作数,“+ ”表示压入堆栈,“- ”表示出栈,且是成对出现的,
注意: 分析串读时要从后往前读,移位时也是往前移的
Op5 |
Op4 |
Op3 |
Op2 |
Op1 |
Not |
分析串 |
偏移 |
图1. 编码光栅运算式(光栅编码的低16 位)
举例说明:如光栅操作编码为0x00E20746 ,低位字节为0x0746 ,二进制为0000 0111 0100 0110
显然Op5=NOT,Op4=NOT,Op3=XOR,Op2=AND,Op1=XOR ,而Not=0 表示不需要额外的NOT ;
分析串=001 移位=2 ,实际串为SPDSPDSP 左移位2 个符号即DSPDSPSP ,Op1-Op3 是二元操作符,Op4-Op5 是一元操作符,所以只需4 个操作数,这样分析串被截断为DSPD ,光栅操作的后缀表示为DSPDxaxnm 简化为DSPDxax 和GDI 定义的D^(S&(P^D))) 是一样的
低16 位的这种编码,实际上是早期时为了节省内存和汇编代码量而设计的编码,新的GDI 实现不再使用这种现在看来较慢的机制,因此通常情况下,丢掉三元光栅操作的低位字是安全的,但是很难说是否会有某个图形设备驱动去检查32 位光栅操作码的每一位匹配与否,为了安全起见,当用到新的光栅操作码时,应该检查和使用整个光栅操作编码。
三元光栅操作码只使用32 位光栅码的24 位。光栅操作码的高8 位通常全为位0 。Windows2000 及以上版本,引入了两个新的标志用于控制位图传输操作CAPTUREBLT 和NOMIRRORBITMAP 。
NOMIRRORBITMAP 标志(0x80000000 )防止位图被垂直或水平的镜像,因为源和目标矩形的轴的方向或其他方面有所不同
CAPTUREBLT 标志(0x40000000 )用于重叠窗口操作。
四元光栅编码:
四元光栅操作中的第四个参与者是一个单色屏蔽位图,四元光栅代码由一个前景三元光栅代码和一个背景三元光栅代码组成。掩码像素为1 时,使用前景光栅操作代码,掩码像素为0 时,使用背景光栅操作代码,GDI 定义了宏MAKEROP4 用于将2 个24 位的三元光栅操作码组合成32 位的四元光栅操作码ROP4 。定义如下:
#define MAKEROP4(fore,back) (DWORD)((((back) << 8) & 0xFF000000) | (fore))
背景ROP 索引(8 位) |
前景ROP 索引(8 位) |
前景ROP 公式的编码(ROP 的低16 位) |
该宏取背景光栅操作码的高8 位光栅操作索引,左移8 位,然后和24 位的前景光栅操作代码组成32 位的ROP4 。
注意掩码在四元光栅操作中的地位和其他三个并不平等,它被限制在两个值: 1 (白色)和 0 (黑色)。不能是彩色位图,因为彩色位图可以和画刷、源及目标彩色位图组合。提供四元光栅操作的主要目的是提供简单和有效地实现透明位图的显示方法。