内嵌汇编
HCS08 C 语言提供一个很有用的功能,内嵌汇编。凡是C语言程序段能出现的地方汇编语言源程序都可以出现,但是汇编语言源程序必需位于一个C语言函数中。
语法格式
1) "asm" <汇编指令> ";" ["/*" 注释 "*/"]
双引号中的内容为关键字,方括号中的内容为可选项。
例如:
asm sta COPCTL; /* 喂狗 */
2) "asm" "{"
<汇编指令 1> [";" 注释]
<汇编指令 2> [";" 注释]
"}"
要求:大括号内每条汇编指令占一行;标号以“:”结尾占一行;注释以“;”开始;可以用变量名访问全局变量和C 函数中的局部变量。汇编语言结束前要保证堆栈内容与汇编开始前一致。
例如:用内嵌汇编方法实现统计字符串中字符的个数。
int strlen (char *str)
/*** 'str' 参数由堆栈传递. 函数返回字符串'str'的长度
假定字符串的长度小于256!
*/
{
asm {
LDHX str ; 装入指针
CLRA ; 初始化计数器
BRA test ; 跳到test
loop:
AIX #1 ; 指针加1
INCA ; 计数器加1
test:
TST 0,X ; 字符串是否结束?
BNE loop ; 下一字符
CLRX ; 返回值在 X:A(参见后文)
};
/* 这里可以继续写 C 语言程序段 */
}
2)混和调用
混和调用指的是C语言函数调用汇编语言函数和汇编语言函数调用C语言函数。实现混和调用要解决两个问题1:常量和变量的相互访问;2调用协议
1常量和变量的相互访问
C 语言函数访问汇编语言常量、变量
在汇编语言源程序中采用汇编语法定义变量和常量;在C语言中把这些变量、常量声明为外部常量变量,即可以用变量、常量名来访问。汇编器和编译器敏感段名的大小写。
例如:
汇编源程序中
XDEF ASMData, ASMConst ;声明 ASMData, ASMConst 可以在模块外访问
MyData: SECTION
ASMData: DS.W 1 ; 定义变量
MyConst: SECTION
ASMConst: DC.W $44A6 ; 定义常量
在 C 语言源程序中
#pragma DATA_SEG MyData /* 定义变量段 */
extern int ASMData; /* 变量 ASMData 是外部模块声明的 */
#pragma DATA_SEG DEFAULT /* 返回默认变量段 */
#pragma CONST_SEG MyConst /* 定义常数段 */
extern const int ASMConst; /* 常量 ASMConst 是外部模块声明的*/
#pragma CONST_SEG DEFAULT /*返回默认常数段 */
汇编语言访问C语言变量、常量
在C语言中用标准格式定义常量、变量,在汇编语言中声明这些常量变量是外部定义即可。
例如:
在C语言源文件中:
unsigned int CData; /* 定义变量 */
unsigned const int CConst; /* 定义常量 */
在汇编语言源文件中:
XREF CData ; 外部声明的变量
XREF CConst; 外部声明的常量
使用:
LDD CConst
STD CData
2)调用协议
调用协议,即函数参数传递和返回值的协议。对于HC08和HCS08调用协议是不同的。
HC08 参数传递:调用者函数把所需参数从左到右依次入栈,调用结束后由调用者函数从堆栈中清除传递的参数。在函数传递的是值参数的情况下,如果最后一个参数长度为2字节,则用X:A寄存器对传递;如果最后一个参数长度为1字节且倒数第2个参数长度超过1字节,则最后一个参数用寄存器A传递;如果最后一个参数长度为1字节且倒数第2个参数长度也是1字节,则最后一个参数用寄存器X传递,倒数第2个参数用寄存器A传递;