0
#include <stdio.h>
#include <fcntl.h>
#include <string.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/shm.h>
#include <sys/sem.h>

int main()
{
  key_t key, k, shmid;
  int ret, semid, i = 0;
  struct sembuf op[1];
  void *shmptr = (void *) 0;

  union semun {
    int val;
    struct senid_ds *buf;
    unsigned short *array;
  };

  union semun arg;

  /* Creating a shared memory */
  k = ftok(".", 1);

  if (-1 == k) {
    perror("\n ftok \n");
    return 1;
  }

  shmid = shmget(k, 1000, IPC_CREAT | 0744);

  if (-1 == shmid) {
    perror("\n Shared memory creation failed \n");
    return 2;
  }

  shmptr = shmat(shmid, (void *) 0, 0);

  if (NULL == shmptr) {
    perror("\n Process cant attach to shared memory segment! \n");
    return 3;
  }

  /* Creating semaphore */
  key = ftok(".", 4);

  if (-1 == key) {
    perror("\n ftok \n");
    return 4;
  }

#if 0
  semid = semget(key, 1, IPC_CREAT);
#else
  semid = semget(key, 1, 0);
#endif

  if (-1 == semid) {
    perror("\n Cannot create semaphore \n");
    return 5;
  }

#if 0
  arg.val = 1;
  ret = semctl(semid, 0, SETVAL, arg);
  printf("\n ret : %d \n", ret);
#endif

  op[0].sem_num = 0;
  op[0].sem_op = -1;
  op[0].sem_flg = SEM_UNDO;

  ret = semop(semid, op, 1);

  // printf("\n ret : %d semop : %d\n",ret,op[0].sem);
  printf("\n Process 1 locked sem1 \n");

  /* Do Operation on shared resource */
  while (i < 4) {
    strcpy((char*) shmptr, "BDC");
    printf("\n %s \n", (char *) shmptr);
    sleep(2);
    printf("\n After sleep \n");
    i++;
    shmptr++;
  }

  op[0].sem_op = 1;
  ret = semop(semid, op, 1);
  printf("\n Process A unlocked sem1 \n");

  return 0;
}

共有メモリ セグメントをロックするサンプル プログラムを試してみました。同じプログラムの 2 つのインスタンスを試しました。セマフォが適切にロックされているかどうかを確認します。しかし、上記のプログラムは、プロセス 1 が既にロックされている場合、他のプロセスが共有メモリを使用することを常に許可します。間違いがどこにあるかを特定できませんでした。

4

1 に答える 1

1

ステートメントを有効にIPC_CREATし、3 番目の引数に必要だが不足しているアクセス許可 (少なくとも S_IRWXU) をビット単位または論理和で指定する必要があります。

セマフォを 1 に初期化するステートメントを有効にする必要がありますがSETVAL、外部入力に基づいてそのステートメントのみを実行する必要があります (つまり、プログラムは、セマフォを作成して初期化する引数を使用して呼び出されます)。それ以外の場合、プログラムはセマフォが既に存在すると想定する必要があります。

呼び出しからの戻り値を確認する必要がありsemop()ます。

于 2013-09-23T21:33:23.440 に答える