ipc.h

#include <sys/types.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/sem.h>
#include <sys/shm.h>
#include <errno.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h> #ifndef _IPC_H_
#define _IPC_H_ union semun {
int val;
struct semid_ds *buf;
unsigned short *array; struct seminfo *_buf;
}; int sem_create(key_t key);
int sem_open(key_t key); int sem_p(int semid);
int sem_v(int semid);
int sem_d(int semid);
int sem_setval(int semid, int val);
int sem_getval(int semid);
int sem_getmode(int semid);
int sem_setmode(int semid, char* mode); #endif //_IPC_H_

ipc.c

#include "ipc.h"

int sem_create(key_t key)
{
int semid = semget(key,, | IPC_CREAT | IPC_EXCL);
if (- == semid)
{
printf("sem create faild\n");
exit();
}
return semid;
}
int sem_open(key_t key)
{
int semid = semget(key,,);
if (- == semid)
{
printf("sem open faild\n");
exit();
}
return semid;
} int sem_p(int semid)
{
struct sembuf sb = {,-,};
int ret = semop(semid,&sb,);
if(- == ret)
{
printf("sem p faild\n");
exit();
}
return ret;
}
int sem_v(int semid)
{
struct sembuf sb = {,,};
int ret = semop(semid,&sb,);
if(- == ret)
{
printf("sem v faild\n");
exit();
}
return ret;
}
int sem_d(int semid)
{
int ret = semctl(semid, , IPC_RMID, );
return ret;
}
int sem_setval(int semid, int val)
{
union semun su;
su.val = val;
int ret = semctl(semid, , SETVAL,su);
if (- == ret)
{
printf("sem setval faild\n");
exit();
}
return ret;
}
int sem_getval(int semid)
{
int ret = semctl(semid, , GETVAL,);
if (- == ret)
{
printf("sem getval faild\n");
exit();
}
return ret;
}
int sem_getmode(int semid)
{
union semun su;
struct semid_ds sem;
su.buf = &sem;
int ret = semctl(semid , , IPC_STAT, su);
if (- == ret)
{
printf("sem getmode failed\n");
exit();
}
printf("current permissions is %o\n",su.buf->sem_perm.mode);
return ret;
}
int sem_setmode(int semid, char* mode)
{
union semun su;
struct semid_ds sem;
su.buf = &sem;
int ret = semctl(semid , , IPC_STAT, su);
if (- == ret)
{
printf("sem getmode failed\n");
exit();
}
printf("current permissions is %o\n",su.buf->sem_perm.mode);
sscanf(mode ,"%o",(unsigned int*)&su.buf->sem_perm.mode);
ret = semctl(semid , , IPC_STAT, su);
if (- == ret)
{
printf("sem getmode failed\n");
exit();
}
printf("permissios update...\n"); return ret;
}

shmfifo.h

#include "ipc.h"

#ifndef _SHM_FIFO_H_
#define _SHM_FIFO_H_ typedef struct shmfifo shmfifo_t;
typedef struct shmhead shmhead_t; typedef struct stu
{
char name[];
int age;
}STU; struct shmhead
{
unsigned int blksize; // 块大小
unsigned int blocks; // 总块数
unsigned int rd_index; // 读索引
unsigned int wr_index; // 写索引
}; struct shmfifo
{
shmhead_t *p_shm; // 共享内存头部指针
char *p_payload; // 有效负载的起始地址 int shmid; // 共享内存ID
int sem_mutex; // 用来互斥用的信号量
int sem_full; // 用来控制共享内存是否满的信号量
int sem_empty; // 用来控制共享内存是否空的信号量
}; shmfifo_t* shmfifo_init(int key, int blksize, int blocks);//初始化
void shmfifo_put(shmfifo_t *fifo, const void *buf);//添加数据到环形缓冲区
void shmfifo_get(shmfifo_t *fifo, void *buf);//从缓冲区中取数据
void shmfifo_destroy(shmfifo_t *fifo);//释放共享内存的环形缓冲区 #endif /* _SHM_FIFO_H_ */

shmfifo.c

#include "shmfifo.h"
#include <assert.h> shmfifo_t* shmfifo_init(int key, int blksize, int blocks)
{
//分配内存空间
shmfifo_t *fifo = (shmfifo_t *)malloc(sizeof(shmfifo_t));
assert(fifo != NULL);
memset(fifo, , sizeof(shmfifo_t)); int shmid;
shmid = shmget(key, , );
int size = sizeof(shmhead_t) + blksize*blocks;
if (shmid == -)
{//创建共享内存
fifo->shmid = shmget(key, size, IPC_CREAT | );
if (fifo->shmid == -)
exit(); fifo->p_shm = (shmhead_t*)shmat(fifo->shmid, NULL, );
if (fifo->p_shm == (shmhead_t*)-)
exit(); fifo->p_payload = (char*)(fifo->p_shm + ); //进行字段初始化
fifo->p_shm->blksize = blksize;
fifo->p_shm->blocks = blocks;
fifo->p_shm->rd_index = ;
fifo->p_shm->wr_index = ; fifo->sem_mutex = sem_create(key);
fifo->sem_full = sem_create(key+);
fifo->sem_empty = sem_create(key+); sem_setval(fifo->sem_mutex, );
sem_setval(fifo->sem_full, blocks);
sem_setval(fifo->sem_empty, );
}
else
{//打开共享内存
fifo->shmid = shmid;
fifo->p_shm = (shmhead_t*)shmat(fifo->shmid, NULL, );
if (fifo->p_shm == (shmhead_t*)-)
exit(); fifo->p_payload = (char*)(fifo->p_shm + ); fifo->sem_mutex = sem_open(key);
fifo->sem_full = sem_open(key+);
fifo->sem_empty = sem_open(key+);
} return fifo;
} void shmfifo_put(shmfifo_t *fifo, const void *buf)
{
sem_p(fifo->sem_full);
sem_p(fifo->sem_mutex); //生产产品
memcpy(fifo->p_payload+fifo->p_shm->blksize*fifo->p_shm->wr_index,
buf, fifo->p_shm->blksize);
fifo->p_shm->wr_index = (fifo->p_shm->wr_index + ) % fifo->p_shm->blocks; sem_v(fifo->sem_mutex);
sem_v(fifo->sem_empty);
} void shmfifo_get(shmfifo_t *fifo, void *buf)
{
sem_p(fifo->sem_empty);
sem_p(fifo->sem_mutex); memcpy(buf, fifo->p_payload+fifo->p_shm->blksize*fifo->p_shm->rd_index,
fifo->p_shm->blksize);
fifo->p_shm->rd_index = (fifo->p_shm->rd_index + ) % fifo->p_shm->blocks;
sem_v(fifo->sem_mutex);
sem_v(fifo->sem_full);
} void shmfifo_destroy(shmfifo_t *fifo)
{
//删除创建的信息量集
sem_d(fifo->sem_mutex);
sem_d(fifo->sem_full);
sem_d(fifo->sem_empty); //删除共享内存
shmdt(fifo->p_shm);//删除共享内存头部
shmctl(fifo->shmid, IPC_RMID, );//删除整个共享内存 //释放fifo的内存
free(fifo);
}

shmfifo_send.c

#include "shmfifo.h"
/*
typedef struct stu
{
char name[32];
int age;
}STU;
*/
int main(void)
{
shmfifo_t *fifo = shmfifo_init(,sizeof(STU),); STU s;
memset(&s, , sizeof(STU));
s.name[] = 'A';
int i;
for(i=;i<;i++)
{
s.age = + i;
shmfifo_put(fifo,&s);
s.name[] = s.name[] + ;
printf("send ok\n");
}
return ;
}

shmfifo_recv.c

#include "shmfifo.h"
/*
typedef struct stu
{
char name[32];
int age;
}STU;
*/
int main(void)
{
shmfifo_t *fifo = shmfifo_init(,sizeof(STU),); STU s;
memset(&s, , sizeof(STU)); int i;
for(i=;i<;i++)
{
shmfifo_get(fifo,&s);
printf("name = %s, age = %d\n",s.name,s.age);
}
return ;
}

shmfifo_free.c

#include "shmfifo.h"

int main(void)
{
shmfifo_t *fifo = shmfifo_init(,sizeof(STU),);
shmfifo_destroy(fifo);
return ;
}

Makefile

.PHONY:clean all
CC=gcc
CFLAGS=-Wall -g
BIN= shmfifo_send shmfifo_recv shmfifo_free
OBJS1=shmfifo_send.o shmfifo.o ipc.o
OBJS2=shmfifo_recv.o shmfifo.o ipc.o
OBJS3=shmfifo_free.o shmfifo.o ipc.o
all:$(BIN)
%.o:%.c
$(CC) $(CFLAGS) -c $< -o $@
shmfifo_send:$(OBJS1)
$(CC) $(CFLAGS) $^ -o $@
shmfifo_recv:$(OBJS2)
$(CC) $(CFLAGS) $^ -o $@
shmfifo_free:$(OBJS3)
$(CC) $(CFLAGS) $^ -o $@ clean:
rm -f *.o $(BIN)

system v共享内存与信号量综合的更多相关文章

  1. 第三十三章 System V共享内存与信号量综合

    用信号量解决生产者.消费者问题 实现shmfifo ip.h #ifndef _IPC_H #define _IPC_H #include <unistd.h> #include < ...

  2. Linux进程通信之System V共享内存

    前面已经介绍过了POSIX共享内存区,System V共享内存区在概念上类似POSIX共享内存区,POSIX共享内存区的使用是调用shm_open创建共享内存区后调用mmap进行内存区的映射,而Sys ...

  3. 阐述linux IPC(五岁以下儿童):system V共享内存

    [版权声明:尊重原创.转载请保留源:blog.csdn.net/shallnet 要么 .../gentleliu,文章学习交流,不用于商业用途]         system V共享内存和posix ...

  4. php进程(线程)通信基础--System V共享内存

    PHP默认情况没有开启功能,要支持该功能在编译PHP的时候要加入下面几个选项  System V消息,--enable-sysvmsg   System V信号量支持,--enable-sysvsem ...

  5. System V共享内存介绍

    (一)简单概念 共享内存作为一种进程间通信的方式,其相较于其他进程间通信方式而言最大的优点就是数据传输速率快.其内部实现的方式采用了Linux进程地址空间中的mmap文件映射区,将文件内容直接映射到各 ...

  6. System V共享内存

    目录 1. 概述 2. System V共享内存API shmget shmat shmdt shmctl 3. 简单的程序 代码实现 common.h shmcreate.c shmrmid.c s ...

  7. UNIX环境高级编程——System V 共享内存区

    共享内存区域是被多个进程共享的一部分物理内存.如果多个进程都把该内存区域映射到自己的虚拟地址空间,则这些进程就都可以直接访问该共享内存区域,从而可以通过该区域进行通信.共享内存是进程间共享数据的一种最 ...

  8. Linux IPC实践(9) --System V共享内存

    共享内存API #include <sys/ipc.h> #include <sys/shm.h> int shmget(key_t key, size_t size, int ...

  9. System V 共享内存区

    1.概述 系统调用mmap通过映射一个普通文件实现共享内存.System V 则是通过映射特殊文件系统shm中的文件实现进程间的共享内存通信.也就是说,每个共享内存区域对应特殊文件系统shm中的一个文 ...

随机推荐

  1. SpringMvc4.2.5 零配置出现 No mapping found for HTTP request with URI(转)

    原文地址:SpringMvc4.2.5 零配置出现 No mapping found for HTTP request with URI 采用 spring 零配置,参考 http://hanqunf ...

  2. 【转】JQuery插件定义

    一:导言 有些WEB开发者,会引用一个JQuery类库,然后在网页上写一写("#"),("."),写了几年就对别人说非常熟悉JQuery.我曾经也是这样的人,直 ...

  3. [LeetCode] 96. Unique Binary Search Trees(给定一个数字n,有多少个唯一二叉搜索树) ☆☆☆

    [Leetcode] Unique binary search trees 唯一二叉搜索树 Unique Binary Search Trees leetcode java 描述 Given n, h ...

  4. JavaScript 上万条数据 导出Excel文件(改装版)

    最近项目要js实现将数据导出excel文件,网上很多插件实现~~那个开心呀,谁知道后面数据量达到上万条时出问题:浏览器不仅卡死,导出的excel文件一直提示网络失败.... debug调试发现var  ...

  5. PyCharm+Scrapy爬取安居客楼盘信息

    一.说明 1.1 开发环境说明 开发环境--PyCharm 爬虫框架--Scrapy 开发语言--Python 3.6 安装第三方库--Scrapy.pymysql.matplotlib 数据库--M ...

  6. weblogic控制台用户名密码修改

    1.记得用户名密码但想修改密码修改方法 保存后立即生效,即你退出后即能以新密码登录:但由于启动的用户名密码和登录的用户名密码是同一个,所以我们需要去修改DOMAIN_HOME/servers/serv ...

  7. servlet/和/*匹配的区别

    两者真正的区别是,两者的长度不同,根据最长路径匹配的优先级,/*比/更容易被选中,而/的真正含义是,缺省匹配.既所有的URL都无法被选中的时候,就一定会选中/,可见它的优先级是最低的,这就两者的区别.

  8. 把旧系统迁移到.Net Core 2.0 日记(6) MapRoute/Area/ViewPath

    我想实现 http://localhost:5000/{moduleName}/{controller}/{action}/{id?} 这样的url. 有2个方法 方法1: 在路由里设置多个modul ...

  9. 从零开始学习Vue(四)

    这里引入一个概念SPA(single Page Application), 接着上次的例子,我们在页面底部做了一个Tab的菜单,点击不同的按钮应该是显示不同的内容. 按传统的MVC的思维,我要在Con ...

  10. flex入门----基础知识

    传统的页面布局   在flex出现之前,双列布局,三列布局,动态盒居中,绝对居中布局等常见的布局均是采用dispaly+float+定位来布局的,一般包括以下几种布局策略: normal flow(文 ...