簡體   English   中英

如何從子進程寫入共享內存

[英]How to write into shared memory from a child process

我正在試驗子進程,因為這似乎是使不能在線程中使用的舊 C 庫並行運行的唯一方法。

我的最小示例將三個整數值寫入映射內存。 在子進程中,我想更改這些值。 更改最后一個(第三個)值有效,但僅更改第二個值時,此輸出出現錯誤:

initial mapped memory in parent process:                      000000000000000000000000000000010000000000000000000000000000001000000000000000000000000000000011
mapped memory in child process before writing:                000000000000000000000000000000010000000000000000000000000000001000000000000000000000000000000011
mapped memory in child process after writing:                 0000000000000000000000000000000100000000000000000000000000011011
mapped memory in parent process after child process finished: 0000000000000000000000000000000100000000000000000000000000011011
1
27
terminate called after throwing an instance of 'std::invalid_argument'
  what():  bitset::_M_copy_from_ptr

Process finished with exit code 134 (interrupted by signal 6: SIGABRT)

最小的代碼示例是這樣的:

#include <iostream>
#include <vector>
#include <unistd.h>
#include <sys/wait.h>
#include <sys/shm.h>
#include <sys/stat.h>
#include <sys/mman.h>
#include <fcntl.h>
#include <string.h>
#include <stdlib.h>
#include <stdio.h>
#include <bitset>
#include <semaphore.h>

using namespace std;

int main(int /*argc*/, char* /*argv*/[])
{
    const char * shm_name = "/a_shm_name";
    const int SIZE = 4096;
    std::vector<int> v{1, 2, 3};

    int shm_fd = shm_open(shm_name, O_CREAT | O_RDWR, 0666);

    ftruncate(shm_fd, 32 * v.size());

    void * ptr0 = mmap(0, SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, shm_fd, 0);

    //write into the memory segment
    void * ptr = ptr0;
    for(int i = 0; i < v.size(); i++)
    {
        std::string s = std::bitset<32>(v[i]).to_string();
        int count = sprintf((char*)ptr, "%s", s.c_str());
        ptr = ptr + count;
    }

    cout << "initial mapped memory in parent process:                      " << (char*)ptr0 << endl << flush;

    //fork
    int n1 = fork();
    if(n1 == 0)
    {
        //client process
        shm_fd = shm_open(shm_name, O_CREAT | O_RDWR, 0666);

        ptr0 = mmap(0, SIZE, PROT_READ | PROT_WRITE, MAP_SHARED, shm_fd, 0);

        cout << "mapped memory in child process before writing:                " << (char*)ptr0 << endl << flush;
        //write into the memory segment
        //sem_wait(semId);
        ptr = ptr0 + 32 * 1;
        sprintf((char*)ptr, "%s", std::bitset<32>(27).to_string().c_str());
        cout << "mapped memory in child process after writing:                 " << (char*)ptr0 << endl << flush;

        exit(0);
    }
    else
    {
        //parent process
        wait(nullptr);
    }

    //child process finished, back in the parent process
    //client process
    shm_fd = shm_open(shm_name, O_RDONLY, 0666);

    ptr = mmap(0, SIZE, PROT_READ, MAP_SHARED, shm_fd, 0);

    cout << "mapped memory in parent process after child process finished: " << (char*)ptr0 << endl << flush;
    for(int i = 0; i < v.size(); i++)
    {
        cout << std::bitset<32>(std::string((char*)ptr, 32)).to_ulong() << endl;
        ptr = ptr + 32;
    }

    munmap(ptr0, SIZE);

    shm_unlink(shm_name);

    return 0;
}

我有兩個問題:

  • 為什么我不能只更改第二個整數值?
  • 是否有更好的技術在不同的子進程之間共享數據?

這根本與進程或mmap無關:具有相似結果的更簡單的程序

#include <vector>
#include <bitset>
#include <iostream>

int main() {
    using std::cout, std::endl, std::flush;

    const int SIZE = 4096;
    std::vector<int> v{1, 2, 3};

    char buffer[SIZE];
    char * ptr0 = buffer;

    //write into the memory segment
    char * ptr = ptr0;
    for(unsigned int i = 0; i < v.size(); i++)
    {
        std::string s = std::bitset<32>(v[i]).to_string();
        int count = sprintf(ptr, "%s", s.c_str());
        ptr = ptr + count;
    }

    cout << "initial memory:                                " << ptr0 << endl << flush;

    //write into the memory segment
    ptr = ptr0 + 32 * 1;
    sprintf(ptr, "%s", std::bitset<32>(27).to_string().c_str());
    cout << "mapped memory in child process after writing:  " << ptr0 << endl << flush;
}

(我將所有void*更改為char*因為向void*添加數字不是標准 C++,盡管一些編譯器有一個擴展來將其視為添加到char* 。)

sprintf在實際輸出結束后寫入一個空終止符。 因此,當您使用它在子級中寫入“第二個值”時,它會將'\\0'放入“第三個值”的第一個字節中。 此外,您實際上一直依賴((char*)ptr0)[3*32]std::cout “第三個值”之后的'\\0'來知道何時停止打印。

而不是sprintfoperator<< with const char* ,它們都用於以空字符結尾的字符串,我建議只復制字節的函數:

std::string data = std::bitset<32>(value).to_string();
std::copy(data.begin(), data.end(), (char*) ptr);

cout.write((char*) ptr, 32);

暫無
暫無

聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM