背景
在使用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
的文档中发现:
如果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;
}
网友评论