【转】c嵌入汇编

来源:互联网 发布:ubuntu安装 闪下划线 编辑:程序博客网 时间:2024/04/27 15:56

         由于C语言在定时方面的缺陷,在某些场合下,C程序中需要嵌入汇编代码,以弥补C的缺陷。通常,在C语言程序中嵌入汇编的方法,无外乎两种

   · 在C程序中填入#asm和#endasm的编译命令,在之中填入汇编代码

· 编写汇编函数,通过链接定位器和特殊的编译方式链接方式加入到所有程序的目标代码中


但这两种方法都存在实现困难的问题,对刚起步经验不足的程序员来说,基本上很难弄懂和编译成功。

而我想到的办法是:将汇编代码定义成一数组,然后使C程序以代码方式运行数组。其实,这就是我们所知的蠕虫病毒的原理。

具体方法是这样的,要知道,汇编代码其实也是一常数,8位CPU中,一句代码是用8位二进制表示,16位CPU中,一句代码是用16位二进制表示,所以,以51单片机(8位)为例,其一系列的汇编程序代码,在C程序中就是一个unsignedchar类型的数组。那么,我们可以把想嵌入的汇编代码在C程序中定义成一个数组:

unsigned char code huibian[n]={……};

然后的问题就是如何启动它了。

我们知道,C语言中,无返回类型函数指针的定义为

void (*name_of_function )();

另外C语言中强制类型转换为

(type)variable;

所以,基于上面两个语句,我们就可以把huibian这个数组指针(数组名就是一指针)转化为函数指针

(void(*)())huibian;

那么,既然都是函数指针了,直接运行就行了

(*(void(*)())huibian)();

这样就大功告成了。

再有,这方法的缺点就是,一开始得先把汇编代码译制成数组,这工作代码短还凑活,长了就不好办了,可以的话,编一个译码的工具程序。

 
//<asm.h>
#ifdef ASM
unsigned long shiftR1(register unsigned long);
#else
extern unsigned long shiftR1(register unsigned long);
#endif
//end of asm.h
//<asm.c>
#define ASM
#include <asm.h>
#include <reg52.h>
#pragma OT(4,speed)
unsigned long shiftR1(register unsigned long x)
{
#pragma asm
clr c
mov a,r4
rrc a
mov r4,a

mov a,r5
rrc a
mov r5,a

mov a,r6
rrc a
mov r6,a

mov a,r7
rrc a
mov r7,a

#pragma endasm
return(x);
}
//end of asm.c
将此源文件加入要编译的工程文件,
将光标指向此文件,选择右键菜单“option for file 'asm.c'”,
将属性单“properties”中的“Generate Assembler SRC File”“Assemble SRCFile”
两项设置成黑体的“√”将“Link Public Only”的“√”去掉,再编译即可。
用此方法可以在c源代码的任意位置用#pragma asm和#pragma endasm嵌入汇编语句。
但要注意的是在直接使用形参时要小心,在不同的优化级别下产生的汇编代码有所不同,
可以察看对应的.lst文件看一看,得到正确的优化级别后,#pragma OT(x,speed)锁定
优化级别(这里的值是0-9)。



转自:http://blog.sina.com.cn/s/blog_51c18d570100ewag.html

原创粉丝点击