undefined reference to `__udivdi3'

本文详细解释了Linux中处理64位数除法时出现的编译错误,并提供了解决方法,包括引入合适的头文件、使用特定的函数(如do_div)以及如何在代码中正确应用这些函数,避免编译时的未知符号错误。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

【问题】

编译Linux下面的代码,经常会遇到这种错误:

undefined reference to `__udivdi3'

【解决过程】

之前遇到过几次了,都是类似的原因导致此问题的。后来才了解,其根本原因:

嵌入式中,32位系统中(目前多数系统都是,比如ARM的片子),对于普通的a除以b(b为32位):

(1)当a为32位,Linux 内核中,常用uint32_t 类型,可以直接写为 a/b

(2)但是,对于a是64位,uint64_t的时候,就要用到专门的除操作相关的函数,linux内核里面一般为

do_div(n, base),注意,此处do_div得到的结果是余数而真正的a/b的结果,是用a来保存的

do_div(n,base)的具体定义,和当前体系结构有关,对于arm平台,在

arch/arm/include\asm\div64.h

其实现很复杂,感兴趣的自己去代码里看吧,这里不多说了。

因此,如果你当前写代码,a/b,如果a是uint64_t类型,那么一定要利用do_div(a,b),而得到结果a,

而不能简单的用a/b,否则编译可以正常编译,但是最后链接最后出错,会提示上面的那个错误:

undefined reference to "__udivdi3"

【解决方法】

知道原因,就好办了。办法就是,去你代码里面找到对应的用到除法的地方,即类似于a/b的地方,其中被除数a为64位,Linux中一般用用uint64_t,将a/b用do_div(a,b)得到的a去代替(注意,不是直接用do_div()得到真正a除b后的结果,因为do_div(a,b)得到的是余数,囧。。。),即可,而具体写其他,就显得很麻烦。此处,我们可以借鉴Linux中\fs\yaffs2\yaffs_fs.c中的宏:

static uint32_t YCALCBLOCKS(uint64_t partition_size, uint32_t block_size)
{
uint64_t result = partition_size;
do_div(result, block_size);
return (uint32_t)result;
}

来自己也去封装一个支持64位数的除法的函数,不过,Linux内核就是好,早已经帮我们实现了对应的64位的unsingned和signed两个函数:

static inline u64 div_u64(u64 dividend, u32 divisor);

static inline s64 div_s64(s64 dividend, s32 divisor);
我们可以直接拿过来用了,注意用此函数时,要包含对应头文件:

#include <linux/math64.h>

总结一下就是:

1.先包含头文件:

 

2.然后用(a,b)得到a/b的结果即可。

【提示】

如果需要在进行64位除数的时候,同时得到余数remainder,可以直接用

static inline u64 div_u64_rem(u64 dividend, u32 divisor, u32 *remainder);
static inline s64 div_s64_rem(s64 dividend, s32 divisor, s32 *remainder);

【引用】

1. - 64 bit division in linux

If you've encountered an error message like this

Unknown symbol __udivdi3Unknown symbol __umoddi3Unresolved symbol __udivdi3Unresolved symbol __umoddi3

you most likely want to make a 64 bit division, which is not supported by default in linux kernel space. 
To solve this problem, you need to use the do_div macro available in asm/div64.h:

#include <asm/div64.h>unsigned long long x, y, result;unsigned long mod;mod = do_div(x, y);result = x;

If you want to calculate x / y with do_div(x, y), the result of the division is in x, the remainder is returned from the do_div function
Since do_div is just an asm (assembler) macro, it doesn't break real time determinism, so it's also suitable for use in RTAI classic, RTAI fusion and ADEOS/ADEOS-IPIPE applications.

根据提供的引用内容,"undefined reference to __ashldi3"错误通常是由于缺少对应的库文件或链接错误导致的。这个错误通常与编译器和链接器有关。 解决这个问题的方法有以下几种: 1. 确保链接器能够找到正确的库文件。你可以尝试在编译命令中添加对应的库文件路径和库文件名。例如,如果你使用的是gcc编译器,可以使用"-L"选项指定库文件路径,使用"-l"选项指定库文件名。例如: ```shell gcc -o output_file source_file.c -L/path/to/library -lmylibrary ``` 2. 检查库文件是否正确安装。如果你使用的是第三方库,确保你已经正确安装了该库,并且库文件的路径正确。 3. 检查编译选项是否正确。有时候,编译选项可能会导致链接错误。你可以尝试修改编译选项,例如添加或删除一些选项,以解决链接错误。 4. 检查代码中是否存在语法错误或其他错误。有时候,链接错误可能是由于代码中的错误导致的。确保你的代码没有语法错误,并且所有的函数和变量都正确定义和引用。 5. 如果以上方法都无效,你可以尝试重新编译和链接整个项目。有时候,重新编译和链接可以解决一些链接错误。 请注意,以上方法只是一些常见的解决链接错误的方法,具体的解决方法可能因情况而异。如果你能提供更多的上下文信息,例如你使用的编译器和链接器版本,以及你的代码和编译命令,我可以给出更具体的建议。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值