在单独的.S文件中编写内联汇编

时间:2017-11-20 12:47:09

标签: c eclipse assembly arm raspberry-pi3

我正在尝试将我的汇编代码与我的C代码分开,因为有关于另一个问题的建议但是我收到了这个错误:

arm-linux-gnueabihf-gcc -O0 -g3 -Wall -c -fmessage-length=0 -mfpu=neon -o src/ASM.o ../src/ASM.c 
In file included from ../src/ASM.c:1:0:
../src/asm.S:1:12: error: expected ‘=’, ‘,’, ‘;’, ‘asm’ or ‘__attribute__’ before ‘:’ token
 mul64x64asm:
      ^

我尝试在一堆地方放一个分号,但它没有解决我的问题,即使它与这个问题无关,任何帮助都会受到赞赏。 这是我的ASM.c和asm.S文件:

ASM.c

#include "asm.S"

int main(void) {
    extern  void mul64x64asm();
    mul64x64asm();
    return 1;

}

asm.S

mul64x64asm:
      MOVW R0,0x12f4
      MOVT R0,0x5678

更新: 我尝试了建议的答案,最后得到了这个

arm-linux-gnueabihf-as -g --gstabs -o src/asm.o ../src/asm.S 
arm-linux-gnueabihf-gcc -O0 -g3 -Wall -c -fmessage-length=0 -mfpu=neon -o src/ASM.o ../src/ASM.c 
arm-linux-gnueabihf-gcc -o ASM src/ASM.o src/asm.o ../src/asm.S 
src/ASM.o: In function `main':
/home/yunus/eclipse-workspace/ASM/Debug/../src/ASM.c:4: undefined reference to `mul64x64asm'
collect2: error: ld returned 1 exit status

我认为eclipse正在将asm.S和ASM.c编译成对象并在此链接它们。

更新2: 所以我按照@fuz的指示改变了我的asm.S文件 asm.S

.globl mul64x64asm
mul64x64asm:
          MOVW R0,0x12f4
          MOVT R0,0x5678

我想用尽可能少的代码来解决这个问题,所以我删除了.size符号。

../src/asm.S:2: Error: unrecognised symbol type ""

删除了类型符号,我现在收到此错误:

arm-linux-gnueabihf-as -g --gstabs -o src/asm.o ../src/asm.S 
arm-linux-gnueabihf-gcc -o ASM src/ASM.o src/asm.o ../src/asm.S 
/tmp/ccYM9smZ.o: In function `mul64x64asm':
(.text+0x0): multiple definition of `mul64x64asm'
src/asm.o:../src/asm.S:3: first defined here
collect2: error: ld returned 1 exit status

从eclipse中的链接器配置中删除了我的asm.S文件,因此它不会在此行中链接:

arm-linux-gnueabihf-gcc -o ASM src/ASM.o src/asm.o ../src/asm.S

最后让它发挥作用。感谢大家的回答/评论。

最终更新:为了在eclipse中调试我的代码,我必须在asm.S文件中包含更多行,这是我最终得到的最终版本:

.text
.globl mul64x64asm
.type mul64x64asm,%function 
mul64x64asm:
           mov r0, #2
           mov r1, #3
           bx lr
           .size mul64x64asm,.-mul64x64asm

3 个答案:

答案 0 :(得分:5)

你快到了。要将程序集中编写的函数添加到程序中,请将其放在单独的文件中。

确保将您要在程序中其他位置使用的所有符号标记为全局符号,以便链接器在链接时考虑它们:

    .globl mul64x64asm
    .type mul64x64asm,%function
mul64x64asm:
    MOVW R0,0x12f4
    MOVT R0,0x5678
    bx lr                        @ don't forget to return instead of fallthrough
    .size mul64x64asm,.-mul64x64asm

.globl指令将符号可见性调整为全局。 .type指令将符号类型标记为函数,这对您来说非常重要使用动态链接。 .size指令将mul64x64asm的符号大小设置为此处(.)与函数开头之间的差异。这对于调试很有用,但如果你太懒,可以省略。

在非ARM上,将使用.type mul64x64asm,@function,但@是ARM上的注释字符,因此气体使用%代替。

现在组装它并将其链接到您的程序中,就像任何其他目标文件一样。

答案 1 :(得分:2)

预处理器和#include指令有点被误导。它会将包含文件中的代码完全粘贴到执行#include

的文件中

预处理后,您的代码看起来像

mul64x64asm:
      MOVW R0,0x12f4
      MOVT R0,0x5678

int main(void) {
    extern  void mul64x64asm();
    mul64x64asm();
    return 1;

}

这肯定不是有效的C代码。

如果要进行内联汇编,则必须以编译器理解的方式执行此操作。 Find the documentation for your version of GCC,并详细了解它。

另一个可能更好的选择是使用汇编程序将asm.S源文件构建到目标文件中。然后将主程序与此目标文件链接以创建可执行程序。

答案 2 :(得分:2)

您应该从程序集中创建.o。然后用以下代码编译你的c代码:

gcc main.c asm.o