Linux编程--strtol与strtoll

背景

在使用strtol的过程中,发现在部分手机上读取字符串中的地址出现问题,导致在使用So的过程中出现Crash。

环境

// 比较/proc/self/maps中每一行中是否有lib_so的名称
 if (strstr(maps_line, lib_so) != nullptr) {
        // 如果该行找到了Lib的基址,需要解析出基址位置,16进制
        long addr_base = strtol(maps_line, nullptr, 16);
        LOGE(TAG,"get base maps_line :%s   addr:%x",maps_line,addr_base);
        return (void *) addr_base;
}

而对应读取出来的maps_line为:

e8c9b000-e90f3000 r-xp 00000000 fd:00 3064     /system/lib/libart.so

而打印出来的结果为:

addr_base = 7fffffff;

原因

由于有符号的long最大值为7fffffff,所以如果读取出来的值超出7fffffff的话,那么就会溢出,返回的也就会是7fffffff

尝试使用过unsigned long来声明addr_base,仍然没有解决该问题。

最后,在man strtol的文档中发现:

man strtol

如果strtox的函数超过或者低于最大最小值的话,就会返回对应的最大最小值,即使声明unsigned也没用。

方案

最后,通过使用strtoll将地址转换成long long型的地址,再转换成指针即可。

if (strstr(maps_line, lib_so) != nullptr) {
        // 如果该行找到了Lib的基址,需要解析出基址位置,16进制
        // 如果基址超过7fffffff的话,就会溢    
        long long addr_base = strtoll(maps_line, nullptr, 16);
        LOGE(TAG,"get base maps_line :%s   addr:%x",maps_line,addr_base);  
        return (void *) addr_base;
}

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

扫码关注云+社区

领取腾讯云代金券