C语言中嵌入汇编,究竟有何意义?
我实在想不透,这种行为的意义,难道仅仅是为了快吗?
另外,汇编语言依赖于特定的机器,在C语言中嵌入的汇编语言,会不会出现移植性的问题?
[解决办法]
比如可以直接可以操作寄存器的值
会出现移植性的问题
[解决办法]
这样加快操作速度。。看你移植的对象了!!
[解决办法]
在C语言中嵌入的汇编语言,会不会出现移植性的问题?肯定会呀。
使用嵌入汇编,主要两个目的:一是为了控制方便(比如生成位置无关的代码、使用特权指令、使用特定架构的指令实现算法等等),其次才是提高性能。
[解决办法]
//=====================================================================================/* CPUID指令是intel IA32架构下获得CPU信息的汇编指令, 可以得到CPU类型,型号,制造商信息,商标信息,序列号, 缓存等一系列CPU相关的东西。*/#include <windows.h>#include <iostream>#include <string>using namespace std;//用来存储eax,ebx,ecx,edx四个寄存器的信息DWORD deax;DWORD debx;DWORD decx;DWORD dedx;void ExeCPUID(DWORD veax) //初始化CPU{__asm{ mov eax,veax cpuid mov deax,eax mov debx,ebx mov decx,ecx mov dedx,edx}}/* 在Intel Pentium以上级别的CPU中,有一个称为“时间戳(Time Stamp)”的部件, 它以64位无符号整型数的格式,记录了自CPU上电以来所经过的时钟周期数。 由于目前的CPU主频都非常高,因此这个部件可以达到纳秒级的计时精度。 这个精确性是上述两种方法所无法比拟的。 在Pentium以上的CPU中,提供了一条机器指令RDTSC(Read Time Stamp Counter) 来读取这个时间戳的数字,并将其保存在EDX:EAX寄存器对中*/long GetCPUFreq() //获取CPU频率,单位: MHZ{ int start,over; _asm { RDTSC mov start,eax } Sleep(50); _asm { RDTSC mov over,eax } return (over-start)/50000;}/* 把eax = 0作为输入参数,可以得到CPU的制造商信息。 cpuid指令执行以后,会返回一个12字符的制造商信息, 前四个字符的ASC码按低位到高位放在ebx,中间四个放在edx,最后四个字符放在ecx。*/string GetManID() //获取制造商信息{ char ID[25]; memset(ID,0,sizeof(ID)); ExeCPUID(0); //初始化 memcpy(ID+0,&debx,4); //制造商信息复制到数组 memcpy(ID+4,&dedx,4); memcpy(ID+8,&decx,4); return string(ID);}/* 在我的电脑上点击右键,选择属性,可以在窗口的下面看到一条CPU的信息, 这就是CPU的商标字符串。CPU的商标字符串也是通过cpuid得到的。 由于商标的字符串很长(48个字符),所以不能在一次cpuid指令执行时全部得到, 所以intel把它分成了3个操作,eax的输入参数分别是0x80000002,0x80000003,0x80000004, 每次返回的16个字符,按照从低位到高位的顺序依次放在eax, ebx, ecx, edx。 因此,可以用循环的方式,每次执行完以后保存结果,然后执行下一次cpuid。*/string GetCPUType(){ const DWORD id = 0x80000002; //从0x80000002开始,到0x80000004结束 char CPUType[49];//用来存储CPU型号信息 memset(CPUType,0,sizeof(CPUType));//初始化数组 for(DWORD t = 0 ; t < 3 ; t++ ) { ExeCPUID(id+t); //每次循环结束,保存信息到数组 memcpy(CPUType+16*t+ 0,&deax,4); memcpy(CPUType+16*t+ 4,&debx,4); memcpy(CPUType+16*t+ 8,&decx,4); memcpy(CPUType+16*t+12,&dedx,4); } return string(CPUType);}void main() { cout<<"本机CPU信息如下:"<<endl; cout<<"CPU 主 频: "<<GetCPUFreq()<<" MHZ"<<endl; cout<<"CPU 制造商: "<<GetManID()<<endl; cout<<"CPU 型 号: "<<GetCPUType()<<endl; cin.get();}
[解决办法]
比如
//有符号整形a和b,如何判断a+b是否溢出#include <stdio.h>int ifo_add(int a,int b) { __asm { mov eax,a add eax,b jo overflowed xor eax,eax jmp no_overflowedoverflowed: mov eax,1no_overflowed: }}int main() { int a,b; a= 1;b= 2;printf("%11d+(%2d) %d\n",a,b,ifo_add(a,b)); a= -1;b=-2;printf("%11d+(%2d) %d\n",a,b,ifo_add(a,b)); a= 2147483647;b= 1;printf("%11d+(%2d) %d\n",a,b,ifo_add(a,b)); a=-2147483647;b=-1;printf("%11d+(%2d) %d\n",a,b,ifo_add(a,b)); a=-2147483647;b=-2;printf("%11d+(%2d) %d\n",a,b,ifo_add(a,b));}// 1+( 2) 0// -1+(-2) 0// 2147483647+( 1) 1//-2147483647+(-1) 0//-2147483647+(-2) 1
[解决办法]
_asm 和 __asm 有啥区别
[解决办法]
1. 不同“架构”的cpu,汇编不同
2. 为了“快”,而且有些功能 “高级”语言不提供
3. compiler 手册会提供一定的嵌入格式,规避了“污染”其他程序中的重要数据的风险--比如先 push stack...
[解决办法]
[解决办法]
汇编一个用途是针对不同的CPU做优化,可想而知是不能移植的,不过一般用宏定义来区分,比如:
#if HAVE_AMD3DNOW
__asm__ volatile("femms");
#elif HAVE_MMX
__asm__ volatile("emms");
#endif
可以看ffmpeg代码,里面用到了大量的汇编。
[解决办法]