2016-11-24 54 views
1

共享內存我有一個問題,在C.問題在C

共享memrory我覺得我這個分配共享內存變量,但我不能刪除它,所以我不能無需重新啓動啓動我PROGRAMM我電腦,因爲我得到了這個問題:這個共享內存已經存在。

此外,我不知道我的信號量使用。

我必須使用一個共享變量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

請注意,您可以使用'ipcs'命令來枚舉打開的sysV IPC對象,'ipcrm'命令可以在不重新啓動計算機的情況下刪除共享內存和其他IPC對象。 –

回答

1

我覺得我這個分配共享內存變量,但我不能刪除它,所以我不能無需重新啓動我的電腦啓動我PROGRAMM因爲我也有這個問題:這個共享內存已經存在。

您獲得共享內存段是這樣的...

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

標誌組合IPC_CREAT|IPC_EXCL堅持認爲,共享內存段被新創建。如果指定的鍵已經存在段,那麼調用將失敗並出現錯誤,這是您似乎正在觀察的內容。共享內存段會一直存在,直到它們被明確刪除,並且程序可以在不刪除段的情況下退出多種方式。

主要有兩種選擇:

  1. 如果它是可以接受的程序使用以前運行時創建的共享內存段 - 或其他同期一個 - 那麼你可以刪除IPC_EXCL國旗從shmget()打電話。

  2. 如果您確實需要確保爲每次運行創建新的新鮮細分,那麼您必須確保細分在不再需要時被移除。特別是,如果程序退出時段的需要結束,那麼您應該確保創建段後,程序不會退出 - 至少不會正常情況下 - 不會再次刪除段。

注意,那就是,如果程序每次運行都需要自己的共享內存段,那麼你必須使用不同的密鑰對不同的運行,或接受只有一個程序的實例可以同時運行,如果它異常退出,則可能需要手動清理 - 例如ipcrm

如果您有特定關於信號量使用的問題,那麼這些會作爲一個單獨的問題提出更好。