我正在尝试创建一个小程序,它接收物理内存位置并打印存储在该位置的数据。我将两个参数传递给程序 - 我要打印的内存的地址和大小(以字节为单位)。
我遇到的问题是当我传入的地址超过某个值时,strtol()函数会传出一个无意义的值。代码如下:
#include <stdio.h>
#include <stdlib.h>
#include <sys/mman.h>
#include <asm-generic/fcntl.h>
#include <unistd.h>
int main(int argc, char** argv)
{
unsigned int mem_address,mem_size;
int loop, i;
int *ptr, *mem_address_current;
printf("mem_addr: %s\n",argv[1]);
printf("mem_size: %s\n",argv[2]);
mem_address = strtol(argv[1], NULL, 16);
mem_size = strtol(argv[2], NULL, 16);
printf("mem_addr: %x\n",mem_address);
printf("mem_size: %x\n",mem_size);
int mem_dev = open("/dev/mem", O_RDWR);
if(mem_dev == -1)
{
printf("No worky\n");
exit(1);
}
int alloc_mem_size, page_mask, page_size;
void *mem_pointer, *virt_addr;
page_size = sysconf(_SC_PAGESIZE);
alloc_mem_size = (((mem_size / page_size) + 1) * page_size);
page_mask = (page_size - 1);
mem_pointer = mmap(NULL,
alloc_mem_size,
PROT_READ | PROT_WRITE,
MAP_SHARED,
mem_dev,
(mem_address & ~page_mask)
);
if(mem_pointer == MAP_FAILED)
{
printf("no_worky2\n");
exit(1) ;
}
virt_addr = (mem_pointer + (mem_address & page_mask));
ptr = mem_pointer;
loop = (mem_size/16) + 1;
for(i = 0;i < loop;i++) {
printf("%#x: %08x %08x %08x %08x\n", ptr, *ptr, *(ptr + 1), *(ptr + 2), *(ptr + 3));
ptr = ptr + 4;
}
return 0;
}
如果我运行以下命令,我会得到预期值。
root@p1025:~# ./test_prog_ppc 0100000 16
mem_addr: 0100000
mem_size: 16
mem_addr: 100000
mem_size: 16
0x48001000: 38210020 4e800020 9421ffe0 7c0802a6
0x48001010: 2c050000 bf410008 7c7e1b78 90010024
但是,如果我更改输入地址,则值会从预期变化:
root@p1025:~# ./test_prog_ppc ffee0000 16
mem_addr: ffee0000
mem_size: 16
mem_addr: 7fffffff
mem_size: 16
关于为什么会发生这种情况的任何想法?
感谢您的帮助。
答案 0 :(得分:5)
strtol()
返回(已签名)long
。根据您的输入,您尝试解析无符号的32位数字,因此如果系统上long
为32位,strtol()
将超出范围。在这种情况下,它会返回LONG_MAX
,在您的系统上显示为0x7FFFFFFF
。
您应该使用strtoul()
,并将结果存储在unsigned long
中,而不是现在存储在unsigned int
中,因为long
保证在int
至少32位,strtol()
不是。
另外,你真的应该将真实指针的地址作为argv[1]
和朋友的第二个参数传递,并使用它来检查错误。现在,如果argv[2]
或{{1}}中的任何一个没有以数字开头,您将无法区分输入是否错误,或者输入是否是合法的零。
答案 1 :(得分:3)
记录了此行为,请参阅strtol
返回值
...
如果正确的值超出可表示值的范围,则应返回{LONG_MIN},{LONG_MAX},{LLONG_MIN}或{LLONG_MAX}(根据值的符号),并将errno设置为[ERANGE]。< / p>
您的返回值为LONG_MAX
,当您测试errno
时,它应为ERANGE
。
您可以使用strtoul
,strtoll
或strtoull
中的一个,它们都可以处理更大的值。虽然strtoul
仅限于unsigned long
,但如果您使用的是64位系统,则应使用其中一个strto*ll
函数。