mips 内嵌汇编

原文网址:http://blog.chinaunix.net/uid-26817832-id-3146404.html

1. GCC 内嵌汇编的基本格式


asm("assembly code");

如:

asm("syscall"); //触发一个系统调用


如果有多条指令,则需在指令尾部添加'\t'和'\n',如:

asm("li v0, 4011\t\n" "syscall");


括号里的字符串 GCC 前端不作分析,直接传给汇编器 as ,故而相联指令间需插入换行符。

'\t' 加入只为排版对齐一些而已,可以使用 gcc -S tst.c -o tst.s 查看生成的 tst.s

因为 GCC 并不对 asm 后括号中的指令作分析,故而如果指令修改一些的寄存器的值,GCC是
不知道的,这个会引入一些问题。

另外 asm 可以替换为 __asm__ ,效果等价。__asm__ 一般用于头文件中,防止关键字 asm
可能与一些变量、函数名冲突。


2. GCC 内嵌汇编扩展格式

asm (
"assembly code"
: output_operand
: input_operand
: clobbered_operand
);

3. 修饰符

= 只写,常用于修饰所有输出操作数
+ 只读
& 只用于输出,一般和'='一起用,如:"=&r" (val)


4. 具体实例


int get_counter()
{
int rst;

asm(
"mfc0 %0, $25\t\n"
: "=r" (rst)
);

return rst;
}

"=r" 中,'=' 为修饰符,表示该操作对象只写,一般用于修饰输出参数列表中。'r' 表示任意
一个通用寄存器。

unsigned short data[] = {
0x0, 0x0, 0x0, 0x0,
0x1, 0x3, 0x5, 0x7,
0x1, 0x3, 0x5, 0x7,
};

void pmullh()
{
asm volatile
(
".set mips3\n\t"
".set noreorder\n\t"

"ldc1 $f0, %1\n\t"
"ldc1 $f2, %2\n\t"



"pmullh $f2, $f2, $f0\n\t"

"sdc1 $f2, %0\n\t"

".set reorder\n\t"
".set mips0\n\t"

: "=m"(*data)
: "m"(*(data+4)), "m"(*(data+8))
: "$f0", "$f2", "memory"
);
}


注意到使用'm'修饰的操作数,后面括号里跟的不是指针,而是开始的第一个元素值。
%0,%1, %2 依次对应输出列表的一个,输入列表的两个操作数,编译后会被gcc替换
成类似 0($12),8($12),16($12)的形式,其中$12置数组首地址,即: %0等价于0($12)

由于我们嵌入的代码改变了 $f0, $f2 的值,而他们不在输出、输入列表中,故而要将其陈列
于被改变操作数列表(clobbered operand list)中,以告诉gcc 我们改变了他们的值,以免gcc 误判。

因为代码中我们改变了内存中数据的值,如果此前gcc生成的代码读取了该内存处的值,并保
存于寄存器中的话,由于我们更新了这段数据,所以需要告诉gcc在后面要重新加载数据,这
个需要在被改变操作数列表中(clobbered operand list)写入 "memory"。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值