semctl

Autres langues

Langue: ja

Version: 2004-11-10 (mandriva - 01/05/08)

Section: 2 (Appels système)

名前

semctl - セマフォの制御操作を行なう

書式

 #include <sys/types.h>
 #include <sys/ipc.h>
 #include <sys/sem.h>
 
 int semctl(int semid, int semnum, int cmd, ...);
 

説明

semctl() は、 semid で指定されたセマフォ集合 (semaphore set) またはセマフォ集合の semnun 番目のセマフォに対して、 cmd で指定された制御操作を行なう (集合内のセマフォの番号は 0 から始まる)。

この関数は、 cmd の値に依存して、3 個または 4 個の引き数を持つ。 引き数が 4 個の場合、第 4 引き数の型は union semun である。 呼び出し元プログラムは、 この共用体 (union) を以下のように定義しなければならない。

 union semun {
     int              val;    /* SETVAL の値 */
     struct semid_ds *buf;    /* IPC_STAT, IPC_SET 用のバッファ */
     unsigned short  *array;  /* GETALL, SETALL 用の配列 */
     struct seminfo  *__buf;  /* IPC_INFO 用のバッファ
                                 (Linux 固有) */
 };
 

semid_ds データ構造体は <sys/sem.h> で以下のように定義されている:

 
 struct semid_ds {
     struct ipc_perm sem_perm;  /* 所有権と許可 */
     time_t          sem_otime; /* 最後の semop の時刻 */
     time_t          sem_ctime; /* 最後に変更が行われた時刻 */
     unsigned short  sem_nsems; /* 集合内のセマフォの数 */
 };
 

ipc_perm 構造体は <sys/ipc.h> で以下のように定義されている (強調されたフィールドは IPC_SET を使って設定可能である):

 struct ipc_perm {
     key_t key;            /* semget(2) に与えられるキー */
     uid_t uid;            /* 所有者 (owner) の実効 UID */
     gid_t gid;            /* 所有者の実効 GID */
     uid_t cuid;           /* 作成者 (creator) の実効 UID */
     gid_t cgid;           /* 作成者の実効 GID */
     unsigned short mode;  /* 許可 */
     unsigned short seq;   /* シーケンス番号 */
 };
 

cmd として有効な値は

IPC_STAT
semid に関連づけられたカーネルデータ構造体の情報を arg.buf で指された semid_ds 構造体へコピーする。 semnum 引き数は無視される。 呼び出したプロセスはそのセマフォ集合に対する 読み込み許可を持たなければならない。
IPC_SET
arg.buf で指定された semid_ds 構造体のメンバーのいくつかの値を、 このセマフォに関連づけられたカーネルデータ構造体に書き込み、 sem_ctime メンバーの値も更新する。 構造体の以下のメンバーが更新される: sem_perm.uid, sem_perm.gid, sem_perm.mode (の最下位 9 ビット)。 呼び出したプロセスの実効 UID が所有者 (sem_perm.uid) または作成者 (sem_perm.cuid) と一致するか、呼び出した人が特権を持たなければならない。 semnum 引き数は無視される。
IPC_RMID
セマフォ集合をただちに削除し、その集合上の semop(2) コールでブロックされている全てのプロセスを目覚めさせる (エラー値が返されて、 errnoEIDRM が設定される)。 呼び出したプロセスの実効ユーザ ID が そのセマフォ集合の作成者または所有者と一致するか、 呼び出した人が特権を持たなければならない。 semnum 引き数は無視される。
IPC_INFO (Linux 固有)
システム全体でのセマフォの制限とパラメータに関する情報を、 arg.__buf が指す構造体に入れて返す。 この構造体は seminfo 型である。 seminfo は _GNU_SOURCE 機能検査マクロが定義された場合に <sys/sem.h> で以下のように定義される:
 
 struct  seminfo {
     int semmap;  /* セマフォ・マップの最大エントリ数;
                     未使用 */
     int semmni;  /* セマフォ集合の最大数 */
     int semmns;  /* 全セマフォ集合中のセマフォの
                     最大数 */
     int semmnu;  /* アンドゥ構造体のシステム全体での
                     最大数; 未使用 */
     int semmsl;  /* 一つのセマフォ集合の最大セマフォ数 */
     int semopm;  /* semop(2) に渡す操作の最大数 */
     int semume;  /* プロセスあたりのアンドゥ・エントリ
                     の最大数; 未使用 */
     int semusz;  /* 構造体 sem_undo のサイズ */
     int semvmx;  /* セマフォの最大値 */
     int semaem;  /* セマフォの調整 (semaphore adjustment;
                      SEM_UNDO) のために記録される最大値 */
 };
 
 
設定 semmsl, semmns, semopm, semmni/proc/sys/kernel/sem 経由で変更可能である。 詳しくは proc(5) を参照。
SEM_INFO (Linux 固有)
IPC_INFO のときと同じ情報を格納した seminfo 構造体を返す。 但し、以下のフィールドにはセマフォが消費しているシステム資源に 関する情報が格納される点が異なる。 semusz フィールドは現在システム上に存在するセマフォ集合の数を返す。 semaem フィールドはシステム上の全てのセマフォ集合に含まれる セマフォの総数を返す。
SEM_STAT (Linux 固有)
IPC_STAT と同じく semid_ds 構造体を返す。 但し、 semid 引き数は、セマフォ識別子ではなく、システム上の全てのセマフォ集合 に関する情報を管理するカーネルの内部配列へのインデックスである。
GETALL
集合の全てのセマフォの semval の値 (現在の値) を arg.array に返す。 semnum 引き数は無視される。 呼び出したプロセスはそのセマフォ集合に読み込み許可を持たなければならない。
GETNCNT
システムコールは集合の semnum 番目のセマフォの semncnt の値 (このセマフォの値が増加するのを待っているプロセス数) を返す (集合の semnum 番目のセマフォの semval の増加を待っているプロセスの数を返す)。 呼び出したプロセスはそのセマフォ集合に読み込み許可を持たなければならない。
GETPID
システムコールは集合の semnum 番目のセマフォの sempid の値 (集合の semnum 番目のセマフォに最後に semop(2) コールを実行したプロセスの PID) を返す。 呼び出したプロセスはそのセマフォ集合に読み込み許可を持たなければならない。
GETVAL
システムコールは集合の semnum 番目のセマフォの semval の値を返す。 呼び出したプロセスはそのセマフォ集合に読み込み許可を持たなければならない。
GETZCNT
システムコールは集合の semnum 番目のセマフォの semzcnt の値 (このセマフォの値が 0 になるのを待っているプロセスの数) を返す (集合の semnum 番目のセマフォの semval の値が 0 になるのを待っているプロセスの数を返す)。 呼び出したプロセスはそのセマフォ集合に読み込み許可を持たなければならない。
SETALL
集合の全てのセマフォの semvalarg.array で指定された値を設定する。 その集合に関連する semid_ds 構造体の sem_ctime メンバーの値も更新する。 全てのプロセスのセマフォの変更についてのアンドゥ・エントリ (semop(2) を参照) は消去 (clear) される。 セマフォの値の変更により、他のプロセス内でブロックされている semop(2) コールの続行が許可されると、それらのプロセスは起こされる (wake up)。 semnum 引き数は無視される。 呼び出したプロセスはそのセマフォ集合に 変更 (書き込み) 許可を持たなければならない。
SETVAL
集合の semnum 番目のセマフォの semvalarg.val の値を設定する。その集合に関連する semid_ds 構造体の sem_ctime メンバーの値も更新する。 全てのプロセスのセマフォの変更についてのアンドゥ・エントリは消去される。 セマフォの値の変更により、他のプロセス内でブロックされている semop(2) コールの続行が許可されると、それらのプロセスは起こされる (wake up)。 呼び出したプロセスはそのセマフォ集合に 変更 (書き込み) 許可を持たなければならない。

返り値

失敗した場合、 semctl() は -1 を返し、 errno にそのエラーを示す。

そうでなければシステムコールは cmd によって以下の負でない値を返す:

GETNCNT
semncnt の値
GETPID
sempid の値
GETVAL
semval の値
GETZCNT
semzcnt の値
IPC_INFO
全てのセマフォ集合に関する情報を管理しているカーネルの内部配列の使用中 エントリのインデックスの最大値 (この情報は、システムの全てのセマフォ集合に関する情報を取得するために SEM_STAT 操作を繰り返し実行する際に使用できる)
SEM_INFO
IPC_INFO と同じ
SEM_STAT
semid で指定されたインデックスを持つセマフォ集合の識別子

cmd の値がそれ以外の場合、成功すると 0 が返される。

エラー

失敗した場合は errno には以下の値のどれかが設定される:
EACCES
cmd 引き数が GETALL, GETPID, GETVAL, GETNCNT, GETZCNT, IPC_STAT, SEM_STAT, SETALL, SETVAL のうちの何れかの値を持ち、 呼び出したプロセスがセマフォに対して必要とされる許可と CAP_IPC_OWNER ケーパビリティ (capability) を持っていない。
EFAULT
arg.buf または arg.array で指されているアドレスにアクセスすることができない。
EIDRM
セマフォ集合が削除された。
EINVAL
cmd または semid に無効な値が指定された。 もしくは、 SEM_STAT 操作の場合に、 semid で指定されたインデックス値が現在未使用の配列のスロットを参照いていた。
EPERM
cmd 引き数に IPC_SET または IPC_RMID が指定され、呼び出したプロセスの実効ユーザ ID がセマフォの (sem_perm.cuid で見つかる) 作成者または (sem_perm.uid で見つかる) 所有者でもなく、 プロセスが CAP_SYS_ADMIN ケーパビリティを持たない。
ERANGE
cmd 引き数に SETALL または SETVAL が指定され、(集合のセマフォのどれかの) semval に設定される値が 0 より小さいか、実装の制限 SEMVMX よりも大きい。

準拠

SVr4, POSIX.1-2001.

注意

IPC_INFO, SEM_STAT, SEM_INFO 操作は ipcs(8) プログラムによって割当られた資源について情報を提供するために使用される。 将来的にはこれらは変更されるか、 /proc ファイル・システム・インタフェースに移動されるかもしれない。

構造体 semid_ds 内の多くのフィールドは、 Linux 2.2 では短かったが、Linux 2.4 では長くなった。 この利点を生かすには、glibc-2.1.91 以降の環境下で 再コンパイルすれば十分である。 カーネルは新しい形式の呼び出しと古い形式の呼び出しを cmd 内の IPC_64 フラグで区別する。

初期のバージョンの glibc では、 semun 共用体は <sys/sem.h> で定義されていたが、 POSIX.1-2001 では呼び出し側がこの共用体を定義する必要がある。 この共用体が定義されていない glibc のバージョンでは、 マクロ _SEM_SEMUN_UNDEFINED が <sys/sem.h> で定義されている。

以下は semctl() コールに影響するセマフォ集合のシステム制限:

SEMVMX
semval の最大値 : 実装依存 (32767)。

移植性を高めるための一番良い方法は、常に 4 個の引き数で semctl() を呼び出すことである。

Linux において、 semctl() はシステムコールではないが、システムコール ipc(2) を使って実装されている。

関連項目

ipc(2), semget(2), semop(2), capabilities(7), sem_overview(7), svipc(7)