在调试联盛德W800芯片代码时,看到了下面的代码实现,记录下C语言内嵌汇编语言的实现方式。部分代码如下:
该函数实现功能:将入参newMask的值写入到psr寄存器,用于恢复现场。
代码下载路径:http://www.winnermicro.com/html/1/156/158/558.html
__asm__ (
汇编语句部分
:输出部分
:输入部分
:破坏描述部分
)
1)C内嵌汇编以关键字 __asm__ 或 asm 开始。如果使用 volatile 关键字,则表示告诉编译器不优化后续的代码。
2) 汇编语句部分:可以包含多条汇编语句,每条语句之间用“ ”或“;”,例:
注:汇编语句中的操作数可以使用占位符引用C语言变量,名称如下:%0,%1,…。
2) 输出部分:在汇编语言中被修改的C变量
3) 输入部分:作为参数输入到汇编语言中的C变量
注:输出和输入部分是针对汇编语句部分的输入和输出参数而言。例:如果汇编语言使用变量设置寄存器,则用输入部分;如果汇编语言从寄存器读取值,保存到变量,则用输出部分。
4) 破坏描述部分:通知编译器使用了哪些寄存器或内存。
5) 每个部分之间用”:”隔开。汇编语句部分必不可少,其他部分可以省略。例: asm("nop") ,实现空操作。
今天有点时间,重新改下了下,为避免因编译器和平台实现而出现的问题,我写了三个版本,分别是windows下vc6.0,windows下mingw和cygwin和linux下的gcc/g++。vc6.0:
#include <stdio.h>
const char* input = "%d"
const char* output = "%d\n"
int n
int main()
{
__asm
{
lea eax, n
push eax
push input
loopx:
call scanf
cmp eax, 1
jne end
mov ecx, n
jecxz end
dec ecx
push ecx
push output
call printf
add esp, 8
jmp loopx
end:
add esp, 8
}
return 0
}
mingw/cygwin:
#include <stdio.h>
const char* input= "%d"
const char* output = "%d\n"
int n
int main()
{
__asm__
(
"loop: \n"
"pushl $_n \n"
"pushl _input \n"
"call _scanf \n"
"addl $8, %esp\n"
"cmpl $1, %eax\n"
"jneend \n"
"movl _n, %ecx\n"
"jecxz end \n"
"decl %ecx\n"
"pushl %ecx\n"
"pushl _output \n"
"call _printf \n"
"addl $8, %esp\n"
"jmploop\n"
"end:"
)
return 0
}
linux gcc/g++:
#include <stdio.h>
const char* input = "%d"
const char* output = "%d\n"
int n
int main()
{
__asm__
(
"pushl $n \n"
"pushl input \n"
"loop: \n"
"call scanf \n"
"cmp $1, %eax\n"
"jne end \n"
"movl n, %ecx \n"
"jecxz end \n"
"decl %ecx\n"
"pushl %ecx\n"
"pushl output \n"
"call printf \n"
"addl $8, %esp\n"
"jmp loop\n"
"end: \n"
"addl $8, %esp\n")
return 0
}
用宏指令asm就可以实现C中嵌入汇编了。比如:
void func()
{
C语言代码……
#pragma asm
MOV R6,#23
DELAY2:
MOV R7,#191
DELAY1:
DJNZ R7,DELAY1
DJNZ R6,DELAY2
RET
#pragma endasm
C语言代码……
}