MIPSGCC嵌入式汇编(龙芯适用).pdfVIP

  1. 1、本文档共4页,可阅读全部内容。
  2. 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
  3. 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载
  4. 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
  5. 5、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
  6. 6、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们
  7. 7、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
  8. 8、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
MIPSGCC嵌入式汇编(龙芯适用)

当前版本: 0.3 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 可能与一些变量、函数名冲突。 内嵌汇编如何与 C 变量交换数据? 2. GCC 内嵌汇编扩展格式 asm ( assembly code : output_operand : input_operand : clobbered_operand ); 以一个例子来说明: 如果我们要读取CP0 25 号硬件计数寄存器的值,并返回之,可以这样: int get_counter() { int rst; asm( mfc0 %0, $25\t\n : =r (rst) ); return rst; } =r 中,= 为修饰符,表示该操作对象只写,一般用于修饰输出参数列表中。 r 表示任意 一个通用寄存器。 由于我们只要取得一个值,故而只用到了输出列表。代码中也没修改一些寄存器 的值gcc 不知道, (输出、输入列表中的寄存器gcc 是知道的)故而被改变的操作对象列表亦可省 去。 如果我们要重设CP0 24 号硬件计数器之控制寄存器的值,则: unsigned int op = 0x80f; asm volatile( mtc0 %0, $24 : :r(op) ); volatile 关键字表示让GCC 优化生成代码时,不要移动、删除我们的汇编码。 另外 __volatile__与其含义相同,引入的目的与__asm__是一样的。 如果我们重设后,立即读取CP0 24 号寄存器的值,则: unsigned int rst; unsigned int op = 0x80f; asm volatile( mtc0 %1, $24\t\n mfc0 %0, $25\t\n : =r (rst) : r (op) ); 如果我们要操作的对象位于存储器中,我们可以使用 m 来修饰输入输出参数, 如: 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 误判。 因为代码中我们改变了内存中数据的值,如

文档评论(0)

ctuorn0371 + 关注
实名认证
文档贡献者

该用户很懒,什么也没介绍

1亿VIP精品文档

相关文档