再谈MMX优化

类别:软件工程 点击:0 评论:0 推荐:

这几天帮别人做了彩色空间转换的MMX优化,包括YUV420<->RGB,YUV420<->YUV422,UYVY->YUV420,感觉收获不少.MMX指令的确强劲,使用内联的MMX指令后转换速度比优化前平均提高了3-4倍.下面承接原来的MMX优化的入门篇稍微深入点讲解MMX指令的用法.

首先在VC的内联汇编中使用MMX指令的确时相当方便,而且效率也非常高.推荐使用.

一般有两种传送方式:数组和指针.使用数组比较方便,但使用指针更为普遍.
数组传送方式:
int a[6];
...
_asm{
movq mm0,[a];        //movq传送64位,mm0内容为:a[3]a[2]a[1]a[0]
movd mm1,[a+8];   //movd传送低32位,注意这个地方加8而不是4
}
指针传送方式:
int *p;
...
_asm{
mov eax,[p];
movq mm0,[eax];
movq mm1,[eax+8]
}

MMX中的加减使用就比较简单了,唯一值得注意的是有的是那些是带符号的运算,那些是带饱和的运算.
PMADDWD是两对16位数相乘然后再将32位的结果相加的运算.

MMX中乘法比较是分高低位的,因为要保证乘后的结果和乘数的位数一致,而且往往运算的结果都是高位为0,所以用低位乘法就够了.但是如果想得到一个完整的结果要算两次.
移动指令中要注意的是psllw,psrld等都是逻辑位移,是需要移动符号位的,只有psraw,psrad才是算术位移不移动符号位,MMX中没有除法运算,所以要想办法把被除数变换成2的N次方,然后在用psraw mm0,n来实现除法运算.注意移出的空位都用0填充.

MMX中最复杂就要属紧缩指令了.它们负责重新排列MMX寄存器中数,使之能够按照规定顺序输入输出数据.
packsswb mm0,mm1   // mm0中存放4个16位数:ABCD  mm1:EFGH packsswb后mm0:ABCDEFGH
packuswb 跟packsswb的唯一区别是就是不带符号位
punpckhbw 展开高阶紧缩数据 punpcklbw展开低阶紧缩数据
例如:mm0和mm1分别存放8个8位数.
mm0:A7,A6,A5,A4,A3,A2,A1,A0 mm1:B7,B6,B5,B4,B3,B2,B1,B0
punpckhbw mm0,mm1;    //结果  mm0:B7,A7,B6,A6,B5,A5,B4,A4
punpcklbw mm0,mm1;     //结果  mm0:B3,A3,B2,A2,B1,A1,B0,A0

下面具体举例:
比如src数组中存放着UYVYUYVY....的序列(Y,U,V为8位数),我们要把它按顺序转换成YYYYY,UUUU,VVVV进行存放
_asm{
movq            mm0,[src];        //mm0:Y3 V1 Y2 U1 Y1 V0 Y0 U0
movq            mm1,[src+8];     //mm1:Y7 V3 Y6 U3 Y5 V2 Y4 U2
movq            mm2,mm0;        //mm2做中转,mm0即将改变
punpcklbw    mm0,mm1;       //mm0:Y5 Y1 V2 V0 Y4 Y0 U2 U0
punpckhbw   mm2,mm1;       //mm2:Y7 Y3 V3 V1 Y6 Y2 U3 U1
movq            mm3,mm0;     
punpcklbw    mm0,mm2;       //mm0:Y6 Y4 Y2 Y0 U3 U2 U1 U0
movd            [U],mm0;         //传低32位的U
punpckhbw   mm3,mm2;      //mm3:Y7 Y5 Y3 Y1 V3 V2 V1 V0
movd            [V],mm3;        //传低32位的V
punpckhbw   mm0,mm3;     //mm0:Y7 Y6 Y5 Y4 Y3 Y2 Y1 Y0
movq            [y],mm0;         //传8个Y
emms;                                 //凡是用MMX指令兄弟门别忘拉
}

 

本文地址:http://com.8s8s.com/it/it35391.htm