我正在将一个小型学术操作系统从 TriCore 移植到 ARM Cortex(Thumb-2 指令集)。为了让调度程序工作,我有时需要直接跳转到另一个函数而不修改堆栈或链接寄存器。
在 TriCore(或者更确切地说,在 tricore-g++)上,这个包装模板(对于任何三参数函数)有效:
template< class A1, class A2, class A3 >
inline void __attribute__((always_inline))
JUMP3( void (*func)( A1, A2, A3), A1 a1, A2 a2, A3 a3 ) {
typedef void (* __attribute__((interrupt_handler)) Jump3)( A1, A2, A3);
( (Jump3)func )( a1, a2, a3 );
}
//example for using the template:
JUMP3( superDispatch, this, me, next );
这将生成汇编指令J
(又名 JUMP)而不是CALL
,当跳转到(否则正常)C++ 函数时,堆栈和 CSA 保持不变superDispatch(SchedulerImplementation* obj, Task::Id from, Task::Id to)
。
现在我需要在 ARM Cortex(或者更确切地说,对于 arm-none-linux-gnueabi-g++)上的等效行为,即生成B
(又名 BRANCH)指令而不是BLX
(又名带有链接和交换的 BRANCH)。但是 arm-g++ 没有interrupt_handler
属性,我找不到任何等效的属性。
所以我尝试asm volatile
直接使用并编写 asm 代码:
template< class A1, class A2, class A3 >
inline void __attribute__((always_inline))
JUMP3( void (*func)( A1, A2, A3), A1 a1, A2 a2, A3 a3 ) {
asm volatile (
"mov.w r0, %1;"
"mov.w r1, %2;"
"mov.w r2, %3;"
"b %0;"
:
: "r"(func), "r"(a1), "r"(a2), "r"(a3)
: "r0", "r1", "r2"
);
}
到目前为止,这么好,至少在我的理论中。Thumb-2 需要在寄存器中传递函数参数,即在这种情况下为 r0..r2,因此它应该可以工作。
但是然后链接器死了
undefined reference to `r6'
在 asm 语句的右括号上……我不知道该怎么做。好吧,我不是 C++ 方面的专家,而且 asm 语法也不是很简单……所以有人给我提示吗?提示正确__attribute__
arm-g++ 是一种方法,提示修复 asm 代码将是另一种方法。另一种方法可能是在输入 asm 语句时告诉编译器a1..a3
应该已经在寄存器r0..r2
中(我调查了一下,但没有找到任何提示)。