大佬教程收集整理的这篇文章主要介绍了Linux内核模块中x87内联汇编中的操作数类型不匹配,大佬教程大佬觉得挺不错的,现在分享给大家,也给大家做个参考。
float sqroot(float arg){ float returnValue; asm( "fld %1\n" "fsqrt\n" "fst %0" :"=r"(returnvalue) : "r"(arg) ); return returnValue; }
这失败了,产生了以下错误:
Error: operand type mismatch for `fld' Error: operand type mismatch for `fst'
任何和所有帮助将不胜感激.
你需要kernel_fpu_begin()/ kernel_fpu_end()来保证安全.
而不是从内联asm加载/存储,请求输入并在x87寄存器堆栈的顶部生成输出,并让编译器在需要时发出加载/存储指令.编译器已经知道如何做到这一点,你只需要为sqrt指令本身使用内联asm,你可以用这种方式向编译器描述:
static inline float sqroot(float arg) { asm("fsqrt" : "+t"(arg) ); return arg; }
(有关此on the Godbolt compiler explorer,请参阅编译器生成的asm)
寄存器约束必须告诉块使用浮点寄存器.
或者完全避免内联asm,使用可以内联的GNU C内置函数
您需要使用-fno-math-errno将内置实际内联为fsqrt或sqrtss,而不使用回退来调用sqrtf以获取将导致NaN的输入.
static inline float sqrooT_BuilTin(float arg) { return __builTin_sqrtf(arg); }
对于x86-64,我们得到sqrtss%xmm0,%xmm0 / ret,而对于i386,我们得到fld / fsqrt / ret. (参见上面的Godbolt链接).恒定传播通过__builTin_sqrt和其他优化工作.
编辑:纳入@ iwillnotexist-idontexist的观点(重新加载).
另外,如果是我,我会在声明中添加静态内联并将其放在头文件中.这将允许编译器更智能地管理寄存器并避免堆栈帧开销.
(我也很想将float改为double.否则,你将丢弃实际浮点指令中使用的额外精度.虽然如果你最终经常将值存储为float,那么将会有一个额外的cvtpd2ps指令.oTOH,如果你将参数传递给printf,例如,这实际上避免了cvtps2pd.)
但Linux内核kprintf无论如何都没有double的转换.
如果使用-mfpmath = 387(32位代码的默认值)进行编译,则内联后值将保留在80位x87寄存器中.但是,对于使用64位默认值-mfpmath = sse的64位代码,这将导致在加载回XMM寄存器时舍入为浮点数.
kernel_fpu_begin()保存完整的FPU状态,并且避免使用SSE寄存器,只使用X87不会使它或最终的FPU恢复时更便宜地返回用户空间.
以上是大佬教程为你收集整理的Linux内核模块中x87内联汇编中的操作数类型不匹配全部内容,希望文章能够帮你解决Linux内核模块中x87内联汇编中的操作数类型不匹配所遇到的程序开发问题。
如果觉得大佬教程网站内容还不错,欢迎将大佬教程推荐给程序员好友。
本图文内容来源于网友网络收集整理提供,作为学习参考使用,版权属于原作者。
如您有任何意见或建议可联系处理。小编QQ:384754419,请注明来意。