linux ipc通信 - 信號(hào)量

函數(shù)原型

/* Semaphore control operation.  */
extern int semctl (int __semid, int __semnum, int __cmd, ...) __THROW;

/* Get semaphore.  */
extern int semget (key_t __key, int __nsems, int __semflg) __THROW;

/* Operate on semaphore.  */
extern int semop (int __semid, struct sembuf *__sops, size_t __nsops) __THROW;
semget 函數(shù)介紹

int semget (key_t __key, int __nsems, int __semflg)绍撞;
如果key不存在藕筋,且?guī)в衏teate標(biāo)志則創(chuàng)建,如果已經(jīng)存在則直接返回id

  • key:ipc的鍵骚露,用于與semid匹配
  • nsems:信號(hào)集中信號(hào)的數(shù)量
  • flag:創(chuàng)建的控制選項(xiàng)
semctl 函數(shù)介紹

int semctl (int __semid, int __semnum, int __cmd, ...);

  • semid 通過(guò)semget 返回的id
  • semnum 要操作第幾個(gè)信號(hào)量
  • cmd 命令,支持的命名如下
命令 說(shuō)明
IPC_STAT 從信號(hào)量集上檢索semid_ds結(jié)構(gòu),并存到semun聯(lián)合體參數(shù)的成員buf的地址中
IPC_SET 設(shè)置一個(gè)信號(hào)量集合的semid_ds結(jié)構(gòu)中ipc_perm域的值默勾,并從semun的buf中取出值
IPC_RMID 從內(nèi)核中刪除信號(hào)量集合
GETALL 從信號(hào)量集合中獲得所有信號(hào)量的值,并把其整數(shù)值存到semun聯(lián)合體成員的一個(gè)指針數(shù)組中
GETNCNT 返回當(dāng)前等待資源的進(jìn)程個(gè)數(shù)
GETPID 返回最后一個(gè)執(zhí)行系統(tǒng)調(diào)用semop()進(jìn)程的PID
GETVAL 返回信號(hào)量集合內(nèi)單個(gè)信號(hào)量的值
GETZCNT 返回當(dāng)前等待100%資源利用的進(jìn)程個(gè)數(shù)
SETALL 與GETALL正好相反
SETVAL 用聯(lián)合體中val成員的值設(shè)置信號(hào)量集合中單個(gè)信號(hào)量的值
semop 函數(shù)介紹

int semop (int __semid, struct sembuf *__sops, size_t __nsops)
對(duì)信號(hào)量集中的信號(hào)量進(jìn)行加減操作

  • semid: semphore 句柄
  • ops 操作選項(xiàng)
struct sembuf {
    unsigned short sem_num; /* 需要操作的信號(hào)量編號(hào) */
    short sem_op; /* 對(duì)信號(hào)量值進(jìn)行加減的值 */
    short sem_flg; /* 操作選項(xiàng) */
}

如果sem_op 為正值聚谁,表示對(duì)信號(hào)量進(jìn)行v操作母剥,會(huì)將信號(hào)量的值加上sem_op的值
如果sem_op 為負(fù)值,表示對(duì)信號(hào)量進(jìn)行p操作垦巴,會(huì)將信號(hào)量的值減上sem_op絕對(duì)值,
如果減上之后小于0媳搪,如果指定了IPC_NOWAIT,則semop返回出錯(cuò)返回EAGAIN骤宣,如果沒(méi)有指定則會(huì)阻塞

/* Mode bits for `msgget', `semget', and `shmget'.  */
#define IPC_CREAT   01000       /* Create key if key does not exist. */
#define IPC_EXCL    02000       /* Fail if key exists.  */
#define IPC_NOWAIT  04000       /* Return error on wait.  */
后面3位是用戶(hù)秦爆、用戶(hù)組、其他用戶(hù)的讀寫(xiě)執(zhí)行權(quán)限憔披,如 0777
創(chuàng)建的時(shí)候要帶上IPC_CREAT 和IPC_EXCL標(biāo)志
/**
 * @file sem_lsw.c
 * @author lisiwen (lisiwen945@163.com)
 * @brief 實(shí)現(xiàn)兩個(gè)進(jìn)程的同步讀寫(xiě)
 * @version 0.1
 * @date 2021-01-28
 *
 * @copyright Copyright (c) 2021
 *
 */
#include <apue.h>
#include <sys/sem.h>

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

#define SEM_SEND 0
#define SEM_RECV 1

key_t key = 128;

// 創(chuàng)建信號(hào)量集等限,初始化semums個(gè)信號(hào)燈 和信號(hào)燈的值
int SemCreate(int value, int semnums) {
    // 創(chuàng)建信號(hào)箱集
    int semid = semget(key, semnums, IPC_CREAT | IPC_EXCL | 0777);
    if (semid < 0) {
        return -1;
    }
    union semun un;
    unsigned short *array = (unsigned short *)calloc(semnums, sizeof(unsigned short));
    for (int i = 0; i < semnums; i++) {
        array[i] = value;
    }
    un.array = array;
    // 初始化信號(hào)量集中所有信號(hào)量燈的值
    if (semctl(semid, 0, SETALL, un) < 0) {
        perror("semctl error");
        return -1;
    }
    free(array);
    printf("create semaphore %d success\n", semid);
    return semid;
}

// 對(duì)信號(hào)量集中的具體的信號(hào)燈作PV操作, semnum 是要操作的第幾個(gè)信號(hào)
// p 操作對(duì)信號(hào)的value -1
// v 操作對(duì)信號(hào)的value +1
void SemP(int semnum) {
    int semid = semget(key, 2, 0777);
    if (semid < 0) {
        perror("semget error\n");
        return;
    }
    // 定義sembuf類(lèi)型的結(jié)構(gòu)體數(shù)組,放置若干個(gè)結(jié)構(gòu)體變量
    // SEM_UNDO表示進(jìn)程退出如果沒(méi)有釋放芬膝,則代為釋放
    struct sembuf ops[] = {{semnum, -1, SEM_UNDO}};
    if (semop(semid, ops, sizeof(ops) / sizeof(struct sembuf)) < 0) {
        perror("semop error");
    }

}
void SemV(int semnum) {
    int semid = semget(key, 2, 0777);
    if (semid < 0) {
        perror("semget error\n");
        return;
    }
    struct sembuf ops[] = {{semnum, 1, SEM_UNDO}};
    if (semop(semid, ops, sizeof(ops) / sizeof(struct sembuf)) < 0) {
        perror("semop error");
    }
}

// 銷(xiāo)毀信號(hào)量集
void SemDel() {
    int semid = semget(key, 2, 0777);
    if (semid < 0) {
        perror("semget error\n");
        return;
    }
    // 0 表示對(duì)所有信號(hào)量進(jìn)行操作
    if (semctl(semid, 0, IPC_RMID, NULL) < 0) {
        perror("semop error");
    }
    printf("del semaphore %d success\n", semid);
}

int main(int argc, char* argv[])
{
    if (argc < 2) {
        printf("%s arg", argv[0]);
    }

    if (strcmp(argv[1], "send") == 0) {
        for (int i = 0; i < 10; i++) {
            sleep(1);
            printf("send num %d\n", i);
            SemV(SEM_SEND);
            SemP(SEM_RECV);
        }
    } else if (strcmp(argv[1], "recv") == 0) {
        for (int i = 0; i < 10; i++) {
            SemP(SEM_SEND);
            printf("recv num %d\n", i);
            SemV(SEM_RECV);
        }
    } else if (strcmp(argv[1], "create") == 0) {
        int value = 0; // 初始value為0
        int semnums = 2; // 構(gòu)建兩個(gè)信號(hào)量
        SemCreate(value, semnums);
    } else if (strcmp(argv[1], "del") == 0) {
        SemDel();
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末望门,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子锰霜,更是在濱河造成了極大的恐慌筹误,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,123評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件癣缅,死亡現(xiàn)場(chǎng)離奇詭異厨剪,居然都是意外死亡哄酝,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,031評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門(mén)祷膳,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)陶衅,“玉大人,你說(shuō)我怎么就攤上這事直晨〔缶” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,723評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵勇皇,是天一觀的道長(zhǎng)罩句。 經(jīng)常有香客問(wèn)我,道長(zhǎng)儒士,這世上最難降的妖魔是什么的止? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,357評(píng)論 1 283
  • 正文 為了忘掉前任,我火速辦了婚禮着撩,結(jié)果婚禮上诅福,老公的妹妹穿的比我還像新娘。我一直安慰自己拖叙,他們只是感情好氓润,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,412評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著薯鳍,像睡著了一般咖气。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上挖滤,一...
    開(kāi)封第一講書(shū)人閱讀 49,760評(píng)論 1 289
  • 那天崩溪,我揣著相機(jī)與錄音,去河邊找鬼斩松。 笑死伶唯,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的惧盹。 我是一名探鬼主播乳幸,決...
    沈念sama閱讀 38,904評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼钧椰!你這毒婦竟也來(lái)了粹断?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,672評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤嫡霞,失蹤者是張志新(化名)和其女友劉穎瓶埋,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,118評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡悬赏,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,456評(píng)論 2 325
  • 正文 我和宋清朗相戀三年狡汉,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了娄徊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片闽颇。...
    茶點(diǎn)故事閱讀 38,599評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖寄锐,靈堂內(nèi)的尸體忽然破棺而出兵多,到底是詐尸還是另有隱情,我是刑警寧澤橄仆,帶...
    沈念sama閱讀 34,264評(píng)論 4 328
  • 正文 年R本政府宣布剩膘,位于F島的核電站,受9級(jí)特大地震影響盆顾,放射性物質(zhì)發(fā)生泄漏怠褐。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,857評(píng)論 3 312
  • 文/蒙蒙 一您宪、第九天 我趴在偏房一處隱蔽的房頂上張望奈懒。 院中可真熱鬧,春花似錦宪巨、人聲如沸磷杏。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,731評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)极祸。三九已至,卻和暖如春怠晴,著一層夾襖步出監(jiān)牢的瞬間遥金,已是汗流浹背。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,956評(píng)論 1 264
  • 我被黑心中介騙來(lái)泰國(guó)打工蒜田, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留稿械,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,286評(píng)論 2 360
  • 正文 我出身青樓物邑,卻偏偏與公主長(zhǎng)得像溜哮,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子色解,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,465評(píng)論 2 348

推薦閱讀更多精彩內(nèi)容