【问题标题】:SIGBUS When Using Placement new?SIGBUS 何时使用 Placement new?
【发布时间】:2018-02-08 07:49:56
【问题描述】:

我有以下代码(已更新):

#include <iostream>
#include <cassert>
#include <errno.h>
#include <string.h>
#include <sys/mman.h>
#include <fcntl.h>
#include <new>

struct S {
uint32_t a;
uint32_t b;
};

int main() {
    const auto fd = open("/tmp/abc.txt", O_RDWR | O_CREAT, S_IRWXU);
    assert(fd > 0);
    void* ptr = mmap(nullptr, sizeof(S), PROT_READ | PROT_WRITE, MAP_SHARED, fd, 0);
    assert(MAP_FAILED != ptr);
    std::cout << "address: " << ptr << std::endl;
    auto tptr = new (ptr) S();
    tptr->a = 99;
    const auto rc = msync(&ptr, sizeof(S), MS_ASYNC);
    std::cout << "msync returned " << rc << std::endl;
    if (rc != 0) {
        std::cout << "error code = " << errno << ": " << strerror(errno) << std::endl;
    }
}

当我在 GDB 中运行它时,我得到了这个:

address: 0x7ffff7ff8000

Program received signal SIGBUS, Bus error.
0x0000000000400adb in main () at msync.cpp:20
20      auto tptr = new (ptr) S();

我看到有人提到内存对齐问题,我检查了 0x7ffff7ff8000 是否可以被 2、8、16、32 和 64 整除。然后,我很困惑它期望什么样的对齐。还是别的什么?

提前致谢。

【问题讨论】:

  • 存在三个主要问题。首先是对齐:您需要确保指针与您尝试放置的类型正确对齐。第二个更正式:为了使用标准库中的“placement”placement new 运算符,您需要包含其标题,即&lt;new&gt;。第三,您应该通过从new 表达式返回的类型化指针访问对象,而不是转换原始指针。
  • 感谢您的快速输入。我已经完成了你建议的#2 和#3。但仍然是同样的问题。对于#1,如何确保内存对齐(以可移植的方式)?我有点依赖 mmap 在页面中运行,因此它返回的地址将与页面对齐。

标签: c++ linux g++


【解决方案1】:

您似乎正在尝试在此处创建文件并写入其中,因此最初它的大小为零并且占用零内存页面。但是mmap 不能写到文件末尾,从而有效地为您分配内存:它首先如何知道要添加到文件中的字节数?您需要确保/tmp/abc.txt 包含一些随后可以被放置new覆盖的字符。它不能追加

在我将一些 8 个随机字节写入 /tmp/abc.txt 后运行您的程序成功并覆盖这些

63 00 00 00 00 00 00 00

在我的 x86-64 上符合预期。然后程序会报告您可能打算生成的msync 错误,并正常退出。

【讨论】:

  • 好收获!我实际上并不打算产生更多错误;因此我在这里创建了另一个帖子:stackoverflow.com/questions/48680640/…
  • @Hel 我做到了,这就是我回答中最后一句话的意思。但是为什么msync 调用会失败确实是一个单独的问题,我并没有真正研究过。
  • 不用担心。它来自我愚蠢的错字。 :((谢谢!
  • 或者你可以在mmap之后使用ftruncate(fd, length)
  • fruncate() 听起来是一个更好的选择。谢谢!
猜你喜欢
  • 2021-08-22
  • 1970-01-01
  • 2021-08-31
  • 1970-01-01
  • 1970-01-01
  • 2023-03-22
  • 2017-10-12
  • 2019-12-23
  • 1970-01-01
相关资源
最近更新 更多