gpt4 book ai didi

c - 在 cygwin 中重新实现 mremap,新空间不可写

转载 作者:太空宇宙 更新时间:2023-11-04 04:47:55 26 4
gpt4 key购买 nike

我正在尝试重现 mremap 的工作以增加 mmap 的内存/文件。我大致关注了this example但它是为匿名映射而设计的,而我希望它是一个命名文件(这是一个编程语言解释器,所以这个文件相当于解释器状态的“核心转储”)。

由于#if-blocks 的嵌套,原文很难阅读。所以这是“提取的”,但是真实的代码(对于我正在调试的编译路径)。完整源码可查看here ,但它不包含有问题的代码,因为它尚未提交,因为它不起作用。 :) 那里有 mremap 版本,还有一个 malloc 回退。

unsigned pgsz /*= getpagesize()*/ = 4096;

typedef struct {
int fd;
unsigned char *base;
unsigned used;
unsigned max;
} mfile;

/* reallocate and possibly move mem->base */
mfile *growmem(mfile *mem,
unsigned sz)
{
void *tmp;
int newfd;

printf("growmem: %p %u + %u\n", mem->base, mem->max, sz);
if (sz < pgsz) sz = pgsz;
else sz = (sz/pgsz + 1) * pgsz;
sz += mem->max;
printf("growmem: new size: %u\n", sz);

msync(mem->base, mem->used, MS_SYNC);
newfd = dup(mem->fd);
tmp = mmap(NULL, sz,
PROT_READ|PROT_WRITE,
MAP_SHARED | (mem->fd == -1? MAP_ANONYMOUS : 0),
newfd, 0);

if (tmp == MAP_FAILED)
error(VMerror, "unable to grow memory");

memcpy(tmp, mem->base, mem->used);
munmap(mem->base, mem->max);
mem->fd = newfd;

mem->base = tmp;
mem->max = sz;
return mem;
}

这在调用时似乎有效,但程序稍后崩溃。我没有准确地追踪到哪里,但是基于追踪主 eval 循环,它看起来非常像内存的新部分不可写。失败的运算符分配一个新的字典对象。而且我怀疑我通过重新使用相同的文件描述符做了一些愚蠢的事情。 编辑:新文件描述符,相同的段错误。

Breakpoint 2, eval (ctx=0x80049a70) at src/bin/itp.c:482
482 object t = top(ctx->lo, ctx->es, 0);
(gdb)
Continuing.
evalload <name "dict">
eval
Executing: <operator 24>
Stack: 0:<integer 1>1:<integer 0>2:<integer 0>3:<integer 1>4:<integer 0>5:<integer 0>6:<array L 229 100 78 0>7:<integer 1>8:<dict L 102 16 658 0>9:<integer 8>
Dict Stack: 0:<dict G 358 10 10 0>1:<dict G 358 100 188 0>2:<dict L 102 100 9 0>3:<dict G 486 24 310 0>4:<dict L 102 6 80 0>
Exec Stack: 0:<operator 60>1:<array G 357 2 387 6>2:<boolean false>3:<array G 325 17 416 14>4:<operator 54>5:<operator 63>6:<array G 453 18 414 0>7:<array G 325 1 414 17>8:<array G 325 4 413 5>9:<boolean false>10:<file>11:<file>12:<array L 101 3 103 16>13:<array L 101 1 113 4>14:<operator 24>
2929 [main] itp 613952 exception::handle: Exception: STATUS_ACCESS_VIOLATION
10097 [main] itp 613952 open_stackdumpfile: Dumping stack trace to itp.exe.stackdump





Program received signal SIGSEGV, Segmentation fault.
0x61129ab9 in memset () from /usr/bin/cygwin1.dll
(gdb)
Continuing.
warning: SuspendThread failed. (winerr 6)
[Inferior 1 (process 613952) exited with code 0105400]
(gdb)

追踪到确切的路线。当尝试触及一些新内存时,它在稍后的分配中失败。

unsigned mfalloc(mfile *mem,
unsigned sz)
{
unsigned adr = mem->used;

if (sz) {
if (sz + mem->used >= mem->max)
mem = growmem(mem,sz);
mem->used += sz;
memset(mem->base+adr, 0, sz); //bzero(mem->base+adr, sz);
/* bus error with mremap(SHARED,ANON)! */
}
return adr;
}

我做错了什么?

附言。我在错误调用之前暂停了调试器。如果任何变量值有用,我可以得到它们。 mem->fd 是 12,因此我们不会遇到此代码之前遇到的完全相同的 SHARED/ANON 错误,但在 mmap 调用中需要进行一些调整以避免这种情况组合。

编辑:它似乎可以毫无问题地访问内存/文件的旧内容。这是新的空间,不可用。现有的打开文件会阻止更大的连续 map 吗?唔。或许我应该 msync,然后是 munmap,然后是 fdclose,然后 重新打开和 mmap。 ...这需要重新设计来测试它,因为 mfile 结构需要存储文件名。 ...

最佳答案

如果为 map 请求的大小较大,Cygwin 的 mmap 似乎不会自动增加文件大小。所以首先 ftruncate

关于c - 在 cygwin 中重新实现 mremap,新空间不可写,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19075829/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com