用mmap映射的内存,与mprotect一起使用

时间:2012-05-28 20:51:18

标签: c mmap mprotect

我必须为线程提供内存区域并应用内存管理的基本概念。我们的想法是创建一个线程本地存储,并通过写入,读取,克隆和擦除来管理。问题是当我尝试使用mprotect与TLS中的线程关联的页面取消保护时,这会向我发送错误。这是我的结构:

每个页面的分配

int cnt;
for (cnt = 0; cnt < page_num; cnt++) {
    struct page *p;
    p = (struct page *) calloc(1, sizeof(struct page));

    int *map =  mmap(0, page_size, 0, MAP_ANON | MAP_PRIVATE, 0, 0);
    if (map == MAP_FAILED) {
        perror("Error mmapping the file");
        exit(EXIT_FAILURE);
    }

    p->address = (unsigned int)map; 

它是由mmap映射的,但当我尝试使用此代码保护页面时(或者unprotect我有相同的错误,这是我第一次尝试)

for(int i = 0; i < currentTLS->page_num; i++){
    tls_unprotect(currentTLS->pages[i]);
}

方法tls_unprotect:

void tls_protect(struct page *p){
  if (mprotect((void *) p->address,PAGESIZE, PROT_READ | PROT_WRITE)) {
      fprintf(stderr, "tls_unprotect: could not unprotect page\n");
      exit(errno);
  }
} 

保护方法是一样的。

errorno代码为12。

我感谢任何形式的帮助。感谢。

1 个答案:

答案 0 :(得分:3)

p->address = (unsigned int)map; 

这个演员很可疑。在大多数64位系统(包括OS X AFAIK)上,int为32位宽,因此太短而无法容纳指针。
该转换会丢弃组成指针的8个字节。

p->address应声明为void*,并且该作业根本不需要演员表。如果您需要将其存储为无符号整数类型,请使用uintptr_t(在<stdint.h>,C99中)。