C中的共享内存问题

时间:2016-11-24 16:05:16

标签: c memory semaphore shared

我在C中遇到共享记忆的问题。

我想我分配了这个共享内存变量,但是我无法将其删除,因此我无法在不重新启动计算机的情况下启动我的程序,因为我遇到了问题:这个共享内存已经存在。

而且,我不确定我的信号量是否正在使用。

我必须使用共享变量,2 processus应该增加它。

void prendre(int semid, int no){
    struct sembuf op[1];
    int er; 
    op[0].sem_num=no;
    op[0].sem_op=-1;
    op[0].sem_flg=0;
    er=semop(semid,op,1);
    if(er==-1) { perror("erreur semop prendre \n"); exit(1);}
}

void rendre(int semid, int no){
    struct sembuf op[1];
    int er; 
    op[0].sem_num=no;
    op[0].sem_op=1;
    op[0].sem_flg=0; 
    er=semop(semid,op,1);
    if(er==-1) { perror("erreur semop prendre \n"); exit(1);}
}

void init_semaphore(int semid){
    int er;
    union semun{
        int val;
        struct semid_ds *buf;
        ushort *array;
    }arg;
    arg.val=1;
    er = semctl(semid,0,SETVAL,arg);
    if(er == -1) {perror("erreur semctl\n");exit(1);}
}

void *runDuprocessus(int *pid, int *mem){
    int k=0;
    for (k=0;k<1e6;k++){
        *mem += 1;
    }
    printf("globale processus %d = %d\n",*pid, *mem);
    exit(1);
}

int creationSegment(int size, char *name, int cle){
    int shmid ; // l'identificateur de la memoire partagee 
    key_t clef ; // la clef associee au segment

    clef = ftok(name,(key_t) cle) ;
    shmid = shmget( clef, size, IPC_CREAT|IPC_EXCL|SHM_R|SHM_W ) ;
    if ( shmid== -1 ) {
        perror("Echec creation segment mémoire partagée!\n") ;
        exit(1) ;
    }
  return shmid ;
}

int main(void){
    int destru;
    pid_t pid; 
    int semid=0;
    int *mem;
    int shmid; // identifiant du segment de la mémoire partagé
    int flag = 0; // flag du segment 
    char *name = (char*) malloc(25*sizeof(char));
    int size = sizeof(int);
    key_t cle;
    int tmp=0;

// Création du segment de mémoire partagé
name = "creationSegmentParatage.c";
shmid = creationSegment(size,name,2);

// Création du sémaphore
cle = ftok("proccessus.c",0);
if(cle==(key_t) -1) {perror("erreur ftok\n");exit(1);}

semid=semget(cle,1,IPC_CREAT|0666);
if(semid == -1) {perror("erreur semget\n");exit(1);}

init_semaphore(semid); //Initialisation du semaphore
// Création du fork
pid = fork();
if(pid < 0){
    printf("Erreur fork creation !\n");
    exit(0);
}else if(pid == 0){
    //printf("Processus fils : \n");
    mem =shmat(shmid,0,flag);//Attachement segment mémoire
    if(mem == (int*)-1){
        perror("Probleme attachement segment mémoire\n");
        exit(1);
    }
    prendre(semid,0); //Prise semaphore
    runDuprocessus(&pid,mem);
    tmp = shmdt(mem);
    if(tmp ==-1){ //Détachement du segment mémoire
        perror("detachement impossible\n") ;
        exit(1) ;
    }   
    rendre(semid,0); //Libération semaphore
}
else {
    //printf("Processus père : \n");
    mem = shmat(shmid,0,flag); //Rattachement segment mémoire
    if(mem == (int*)-1){
        perror("Probleme attachement segment mémoire\n");
        exit(1);
    }
    prendre(semid,0);
    runDuprocessus(&pid,mem);
    tmp = shmdt(mem);
    if(tmp ==-1){ //Détachement du segment mémoire
        perror("detachement impossible\n") ;
        exit(1) ;
    }   
    rendre(semid,0);
}
wait(NULL); //Attente du fils
//Destruction du segment mémoire
destru = shmctl(shmid,IPC_RMID,NULL);
if (destru == -1){
    perror("Erreur lors de la destruction") ;
    exit(1) ;
}
//Destruction du semaphore
//sem_destroy(semid);
return 0;
}

感谢您将来的帮助!

1 个答案:

答案 0 :(得分:1)

  

我想我分配了这个共享内存变量,但是我无法删除它,因此我无法在不重新启动计算机的情况下启动我的程序,因为我遇到了问题:这个共享内存已经存在。

您可以获得共享内存段......

shmid = shmget( clef, size, IPC_CREAT|IPC_EXCL|SHM_R|SHM_W ) ;

标志组合IPC_CREAT|IPC_EXCL坚持要新创建共享内存段。如果指定键的某个段已经存在,则该调用将失败并显示错误,这是您正在观察的错误。共享内存段一直存在,直到它们被明确删除,并且有许多方法可以在不删除段的情况下退出程序。

有两个主要选项:

  1. 如果程序可以使用在上一次运行期间创建的共享内存段 - 或其他同期 - 那么您可以删除IPC_EXCL标志来自shmget()来电。

  2. 如果确实需要确保为每次运行创建新的新段,则必须确保在不再需要时删除该段。特别是,如果在程序退出时段的需要结束,则应确保一旦创建段,程序不会退出 - 至少不正常 - 不再删除段。

  3. 另请注意,如果程序的每次运行都需要自己的共享内存段,那么您必须为不同的运行使用不同的键,或者接受一次只运行一个程序实例,如果它会异常退出,然后您可能需要手动清理 - 例如ipcrm

    如果您有关于信号量使用的特定问题,那么这些问题将更好地作为单独的问题提出。