将共享内存与矩阵一起使用

Posted

技术标签:

【中文标题】将共享内存与矩阵一起使用【英文标题】:Using shared memory with matrices 【发布时间】:2013-05-04 15:26:36 【问题描述】:

我想在共享内存段中创建一个矩阵。在我的第二个程序中,我可以读取tailleXtailleY,但我的矩阵的值只有“0”。我在我的函数initialiserSegMem 中创建了我的矩阵,并且值没问题...像往常一样我对指针有一些问题...

我的结构 mem_share:

typedef struct mem_partage
    int** carte;
    int tailleY;
    int tailleX;
mem_share;

mem_share initialiserDonneeMem(grille* g)
    mem_share mem_share_carte;
    int x = g->tailleX;
    int y = g->tailleY;
    int i,j;
    mem_share_carte.carte = malloc(y*sizeof(int*));
    for(i=0;i < y;i++)
        mem_share_carte.carte[i] = malloc(x*sizeof(int));
     
    mem_share_carte.carte = g->carte;
    mem_share_carte.tailleY = y;
    mem_share_carte.tailleX = x;

    return mem_share_carte;


void initialiserSegMem(mem_share *mem_share_carte)
    int shmid,id_memoire,i,j;   
    int test = 100;  

    int *adresse_mem;
    key_t cle;
    cle = 9999;
    /* Creation of the shared memory segment */
    if(shmget(cle,(int) sizeof(mem_share) ,IPC_CREAT | IPC_EXCL | 0777)== -1) 
        printf("Erreur : Segment de mémoire partagée existant");
    
    /*on attache le segment a notre espace memoire */
    id_memoire = shmget(cle,0,0);
    adresse_mem = shmat(id_memoire,NULL,0);
    /* les 2premiers int seront les tailles Y et X*/
    printf("tailleY %d zzzz",mem_share_carte->tailleY);
    memcpy(adresse_mem,&mem_share_carte->tailleY,sizeof(int));
    adresse_mem+=sizeof(int);
    memcpy(adresse_mem,&mem_share_carte->tailleX,sizeof(int));
    adresse_mem+=sizeof(int);

    for(i=0;i<mem_share_carte->tailleY;i++)
        printf("\n");
            for(j=0;j<mem_share_carte->tailleX;j++)
                memcpy(adresse_mem,&(mem_share_carte->carte[i][j]),sizeof(int));
            adresse_mem+=sizeof(int);
            
    

在我的第二个程序中,我试图读取共享记忆:

int shmid,tailleY,tailleX,y,x;

int *adresse_mem;
key_t cle;
/* on recupere la cle du segment */
cle = 9999;
if ((shmid = shmget(cle, 0, 0666)) < 0) 
    perror("shmget");
    exit(1);
    
/*
    * on attache le segment a notre espace memoire
*/
if ((adresse_mem = shmat(shmid, NULL, 0)) == (char *) -1) 
    perror("shmat");
    exit(1);

/*
* on lit le segment :
*  - on recupe la taille y
*  - puis la taille x
*  - puis la carte
*/
memcpy(&tailleY,adresse_mem,sizeof(int));
adresse_mem+=sizeof(int);
printf("tailleY %d",tailleY);
memcpy(&tailleX,adresse_mem,sizeof(int));
printf("tailleX %d",tailleX);
adresse_mem+=sizeof(int);

int** carte;
carte = malloc(tailleY*sizeof(int*));

for(y=0;y<tailleY;y++)
    carte[y] = malloc(tailleX*sizeof(int));
    for(x=0;x>tailleX;x++)
        memcpy(&carte[y][x],adresse_mem,sizeof(int));
        adresse_mem+=sizeof(int);
    

/*
    affichage de la carte
*/
for(y = 0;y<tailleY;y++)
    printf("\n");
    for(x = 0;x<tailleX;x++)
        printf("%d ",carte[y][x]);
    

顺便说一句,在我的第一个程序中,我必须更新我共享记忆中的矩阵,所以我可以用我的新矩阵调用initialiserSegMem 吗?

【问题讨论】:

对于初学者,不要使用 "malloc()" :) 一切 - 矩阵变量,以及所有矩阵内容 - 必须驻留在您使用 "shmget()/shmattach 分配的共享内存中()”。优秀链接:Beej's Guide. for(x=0;x>tailleX;x++) O M G 这是 x 【参考方案1】:

一个简单的示例代码。子和父使用 System V 共享内存段共享一个 2D 向量。 sizeElement 用于数据类型更改时的偏移量估计。这里使用 double。不是函数返回代码的错误处理。

#include <stdio.h>
#include <unistd.h>
#include <sys/shm.h>
#include <sys/stat.h>
#include <wait.h>

unsigned int sizeof_dm(int rows, int cols, size_t sizeElement)
    size_t size = rows * (sizeof(void *) + (cols * sizeElement));
    return size;


void create_index(void **m, int rows, int cols, size_t sizeElement)
    int i;  
    size_t sizeRow = cols * sizeElement;
    m[0] = m+rows;
    for(i=1; i<rows; i++)      
        m[i] = (m[i-1]+sizeRow);
    


void print_matriz(double **matrix, int Rows, int Cols)
    printf("\n");
        for(int i=0; i<Rows; i++)
            for(int j=0; j<Cols; j++)
                printf("%.2f\t",matrix[i][j]);
            printf("\n");
        


int main(int argc, char **argv)

    double **matrix;
    int Cols = 10, Rows = 5, shmId;

    size_t sizeMatrix = sizeof_dm(Rows,Cols,sizeof(double));
    shmId = shmget(IPC_PRIVATE, sizeMatrix, IPC_CREAT|0600);    
    matrix = shmat(shmId, NULL, 0); 
    create_index((void*)matrix, Rows, Cols, sizeof(double));


    if(fork())
        int pos=0;
        for(int i=0; i<Rows; i++)
            for(int j=0; j<Cols; j++)
                matrix[i][j] = pos++;
               
        wait(NULL); 
        print_matriz(matrix, Rows, Cols); //verify child's write
        shmdt(matrix);
        shmctl(shmId, IPC_RMID, 0);
    
    else   
        sleep(3); //avoid race condition
        print_matriz(matrix, Rows, Cols);
        matrix[2][2] = -99.99; //writing test from child
        shmdt(matrix);
    
    return 0;

【讨论】:

以上是关于将共享内存与矩阵一起使用的主要内容,如果未能解决你的问题,请参考以下文章

我可以将 MPI 与共享内存一起使用吗

与 Dask 共享内存

与 GPU 共享 RAM

涉及多个进程的矩阵乘法中的共享内存

如何在 Ada 中使用 Linux 将任意字符串写入并读取到共享内存?

使用信号量进行共享内存同步