【转】c嵌入汇编

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

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

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


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

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

具体方法是这样的,要知道,汇编代码其实也是一常数,8位CPU中,一句代码是用8位二进制表示,16位CPU中,一句代码是用16位二进制表示,所以,以51单片机(8位)为例,其一系列的汇编程序代码,在C程序中就是一个unsigned char类型的数组。那么,我们可以把想嵌入的汇编代码在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 SRC File”
两项设置成黑体的“√”将“Link Public Only”的“√”去掉,再编译即可。
用此方法可以在c源代码的任意位置用#pragma asm和#pragma endasm嵌入汇编语句。
但要注意的是在直接使用形参时要小心,在不同的优化级别下产生的汇编代码有所不同,
可以察看对应的.lst文件看一看,得到正确的优化级别后,#pragma OT(x,speed)锁定
优化级别(这里的值是0-9)。



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

以下是一个简单的示例程序,使用C语言嵌入汇编语言,将十进制数换为二进制并输出。 ```c #include <stdio.h> int main() { int num; printf("Enter a decimal number: "); scanf("%d", &num); __asm__ ( "mov eax, %0\n\t" // 将 num 存入 eax 中 "mov ecx, 31\n\t" // 初始化 ecx,用于循环移位操作 "mov ebx, 0\n\t" // 初始化 ebx,用于存储二进制结果 "loop:\n\t" "shl eax, 1\n\t" // 将 eax 左移一位 "adc ebx, 0\n\t" // 将进位标志位清零 "jnc skip\n\t" // 如果进位标志位为 0,则跳过后面一步 "inc ebx\n\t" // 如果进位标志位为 1,则在 ebx 中加 1 "skip:\n\t" "loop loop\n\t" // 循环移位和加法操作,直到 ecx 为 0 "mov %0, ebx\n\t" // 将二进制结果存入 num 中 : "=r" (num) // 输出操作数,将修改后的 num 值存回变量中 : "0" (num) // 输入操作数,将 num 的值传递给汇编代码 : "eax", "ebx", "ecx" // 受影响的寄存器 ); printf("Binary equivalent: %d", num); return 0; } ``` 在这个程序中,我们使用了一个嵌入汇编块,它包含了一些汇编代码和一些特殊的指令,用于访问C语言中的变量。 在汇编代码中,我们使用了一些寄存器,如 eax、ebx 和 ecx。我们将 num 的值存入 eax 中,并使用 ecx 对其进行循环移位操作。在每次移位时,我们检查进位标志位是否为 1,如果是,则在 ebx 中加 1。最后,我们将二进制结果存回 num 中,以便在C语言中输出。 注意,嵌入汇编块被包含在 __asm__ 关键字中,并使用了三个冒号分隔的部分。第一个部分是输出操作数,它告诉编译器我们将修改哪个变量,并将其存回变量中。第二个部分是输入操作数,它告诉编译器我们将使用哪个变量,并将其传递给汇编代码。第三个部分是受影响的寄存器,它告诉编译器哪些寄存器的值可能会被改变,并且需要保存它们的值。 这个程序可以将输入的十进制数换为二进制并输出。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值