clients(PV操作共享内核内存进行输入输出分屏) - server(进程间通信)模型实现

时间:2020-12-07 02:04:41

1、拓扑结构

clients(PV操作共享内核内存进行输入输出分屏) - server(进程间通信)模型实现

2、PV操作共享内核内存进行输入输出分屏

(1)

 int semop(int semid,struct sembuf *sops,size_t nsops);

功能描述

操作一个或一组信号。
 
 
semid:
信号集的识别码,可通过semget获取。
 
sops:
指向存储信号操作结构的数组指针,信号操作结构的原型如下
 struct sembuf
{
unsigned short sem_num; /* semaphore number */
short sem_op; /* semaphore operation */
short sem_flg; /* operation flags */
};

这三个字段的意义分别为:

sem_num:
操作信号在信号集中的编号,第一个信号的编号是0。
 
sem_op:
如果其值为正数,该值会加到现有的信号内含值中。通常用于释放所控资源的使用权;如果sem_op的值为负数,而其绝对值又大于信号的现值,操作将会阻塞,直到信号值大于或等于sem_op的绝对值。通常用于获取资源的使用权;如果sem_op的值为0,则操作将暂时阻塞,直到信号的值变为0。
 
sem_flg:
信号操作标志,可能的选择有两种
IPC_NOWAIT //对信号的操作不能满足时,semop()不会阻塞,并立即返回,同时设定错误提示。
SEM_UNDO //程序结束时(不论正常或不正常),保证信号值会被重设为semop()调用前的值。这样做的目的在于避免程序在异常情况下结束时未将锁定的资源解锁,造成该资源永远锁定。
 
nsops:信号操作结构的数量,恒大于或等于1。
 void P(int semid)
{
struct sembuf my_buf ;
memset(&my_buf, , sizeof(my_buf) );
my_buf.sem_num = ;
my_buf.sem_op = - ;
my_buf.sem_flg = SEM_UNDO ;
semop(semid, &my_buf, );
}
void V(int semid)
{
struct sembuf my_buf ;
memset(&my_buf, , sizeof(my_buf) );
my_buf.sem_num = ;
my_buf.sem_op = ;
my_buf.sem_flg = SEM_UNDO ;
semop(semid, &my_buf, ); }

(2)

shmget(得到一个共享内存标识符或创建一个共享内存对象)
所需头文件
#include <sys/ipc.h>
#include <sys/shm.h>
函数说明
得到一个共享内存标识符或创建一个共享内存对象并返回共享内存标识符
函数原型
int shmget(key_t key, size_t size, int shmflg)
函数传入值
key
0(IPC_PRIVATE):会建立新共享内存对象
大于0的32位整数:视参数shmflg来确定操作。通常要求此值来源于ftok返回的IPC键值
size
大于0的整数:新建的共享内存大小,以字节为单位
0:只获取共享内存时指定为0
shmflg
0:取共享内存标识符,若不存在则函数会报错
IPC_CREAT:当shmflg&IPC_CREAT为真时,如果内核中不存在键值与key相等的共享内存,则新建一个共享内存;如果存在这样的共享内存,返回此共享内存的标识符
IPC_CREAT|IPC_EXCL:如果内核中不存在键值 与key相等的共享内存,则新建一个共享内存;如果存在这样的共享内存则报错
函数返回值
成功:返回共享内存的标识符
出错:-1,错误原因存于error中

(3)

int semget(key_t key, int nsems, int semflg);
key:
所创建或打开信号量集的键值。
nsems:
创建的信号量集中的信号量的个数,该参数只在创建信号量集时有效。
semflg:
调用函数的操作类型,也可用于设置信号量集的访问权限,两者通过or表示
     shm_key = (key_t)atoi(argv[]);
sem_key = (key_t)atoi(argv[]); my_shm = shmget(shm_key, sizeof(MBUF), |IPC_CREAT); my_sem = semget(sem_key, , | IPC_CREAT);

(4)

int semctl(int semid,int semnum,int cmd, /*union semun arg*/);

SETVAL设置信号量集中的一个单独的信号量的值。

 semctl(my_sem, , SETVAL, );

(5)

shmat(把共享内存区对象映射到调用进程的地址空间)
所需头文件
#include <sys/types.h>
#include <sys/shm.h>
函数说明
连接共享内存标识符为shmid的共享内存,连接成功后把共享内存区对象映射到调用进程的地址空间,随后可像本地空间一样访问
函数原型
void *shmat(int shmid, const void *shmaddr, int shmflg)
函数传入值
shmid
共享内存标识符
shmaddr
指定共享内存出现在进程内存地址的什么位置,直接指定为NULL让内核自己决定一个合适的地址位置
shmflg
SHM_RDONLY:为只读模式,其他为读写模式
函数返回值
成功:附加好的共享内存地址
出错:-1,错误原因存于error中
 p = (pMBUF)shmat(my_shm, NULL, );

(6)

client_in PV操作:

 while( P(my_sem), p -> m_flag == )
{
V(my_sem);
sleep();
}
strcpy(p ->m_buf, line);
p ->m_flag = ;
V(my_sem);

(7)

shmdt(断开共享内存连接)
所需头文件
#include <sys/types.h>
#include <sys/shm.h>
函数说明
与shmat函数相反,是用来断开与共享内存附加点的地址,禁止本进程访问此片共享内存
函数原型
int shmdt(const void *shmaddr)
函数传入值
shmaddr:连接的共享内存的起始地址
函数返回值
成功:0
出错:-1,错误原因存于error中
 shmdt(p);

(8)

删除内存 删除 信号量

shmctl(共享内存管理)
所需头文件
#include <sys/types.h>
#include <sys/shm.h>
函数说明
完成对共享内存的控制
函数原型
int shmctl(int shmid, int cmd, struct shmid_ds *buf)
函数传入值
shmid
共享内存标识符
cmd
IPC_STAT:得到共享内存的状态,把共享内存的shmid_ds结构复制到buf中
IPC_SET:改变共享内存的状态,把buf所指的shmid_ds结构中的uid、gid、mode复制到共享内存的shmid_ds结构内
IPC_RMID:删除这片共享内存
buf
共享内存管理结构体。具体说明参见共享内存内核结构定义部分
函数返回值
成功:0
出错:-1,错误原因存于error中
 shmctl(my_shm, IPC_RMID, NULL) ;
semctl(my_sem, , IPC_RMID);//0为信号量编号

3、clients - server 通信

(1)上下线信息管道

server:

 char file_name[]  = "";
char client_r[], client_w[];
char line[] ;
int fd_read ;
int client_id ;
int fd_cr , fd_cw ;
FILE* fp ;
sprintf(file_name, "%s/%s", PATH, NAME);
mkfifo(file_name, ) ;
fd_read = open(file_name, O_RDONLY);
open(file_name, O_WRONLY); fp = fdopen(fd_read, "r");

client_in:

 char server_name[]= "" ;
char read_file[], write_file[] ;
char msg[] ="" ;
int fd_r, fd_w ;
sprintf(server_name, "%s/%s", PATH, SERVER);
int fd_server = open(server_name, O_WRONLY);
sprintf(msg, "%d\n", getpid());
write(fd_server, msg, strlen(msg));

(2) 读写管道

client_in:

 memset(read_file, , );
memset(write_file, , ); sprintf(read_file, "%s/%d_r.fifo", PATH, getpid());
sprintf(write_file, "%s/%d_w.fifo", PATH, getpid()); mkfifo(read_file, );
mkfifo(write_file, ); fd_r = open(read_file, O_RDONLY);
fd_w = open(write_file, O_WRONLY);

server:

while(memset(line,  , ), fgets(line, , fp) != NULL)// "pid\n"
{// cr cw pid_r.fifo pid_w.fifo
sscanf(line, "%d", &client_id);
printf("client: %d request !\n", client_id) ;
memset(client_r, , );
memset(client_w, , ); sprintf(client_r, "%s/%d_r.fifo", PATH, client_id); sprintf(client_w, "%s/%d_w.fifo", PATH, client_id); fd_cw = open(client_r, O_WRONLY);
fd_cr = open(client_w, O_RDONLY);
/*
****************
*/
close(fd_cr);
close(fd_cw);
}

(3)信息传递

client_in:

 while(memset(line, , ), fgets(line, , stdin) != NULL)
{
write(fd_w, line, strlen(line)); memset(line, , ); read(fd_r, line, );
/**
************
*/ }

server:

 write(fd_wr, buf, strlen(buf));

4、server fork 子进程对客户端传来的信息进行处理

(1)fork

 if(fork() == )
{
child_main(fd_cr, fd_cw);
close(fd_cr);
close(fd_cw);
exit();
}

(2)reserve 函数

 void reverse(char* str)
{
int bg, end ;
char tmp ;
bg = ;
end = strlen(str) - ;
while(bg < end)
{
tmp = str[bg] ;
str[bg] = str[end] ;
str[end] = tmp ;
bg ++ ;
end -- ;
} }

(3) child_main  read and write

 void child_main(int fd_rd, int fd_wr)
{
char buf[] ;
while(memset(buf, , ), read(fd_rd, buf, ) != )
{
reverse(buf);
write(fd_wr, buf, strlen(buf));
} }

(4)防止僵尸进程

  如果子进程先退出,系统不会自动清理掉子进程的环境,而必须由父进程调用wait或waitpid函数来完成清理工作,如果父进程不做清理工作,则已经退出的子进程将成为僵尸进程(defunct)。

 signal(, child_handle);

下次收到17号信号(子进程退出信号) 就 调用 child_handle 函数

 void child_handle(int sig_num)
{
printf("a child exit!\n");
wait(NULL);
}

5、详细代码:

server:

 #include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/stat.h>
#include<sys/types.h>
#include<fcntl.h>
#include<signal.h>
#define PATH "/home/comst/pipe"
#define NAME "server.fifo" void child_handle(int sig_num)
{
printf("a child exit!\n");
wait(NULL);
}
void reverse(char* str)
{
int bg, end ;
char tmp ;
bg = ;
end = strlen(str) - ;
while(bg < end)
{
tmp = str[bg] ;
str[bg] = str[end] ;
str[end] = tmp ;
bg ++ ;
end -- ;
} }
void child_main(int fd_rd, int fd_wr)
{
char buf[] ;
while(memset(buf, , ), read(fd_rd, buf, ) != )
{
reverse(buf);
write(fd_wr, buf, strlen(buf));
} }
int main(int argc, char* argv[])
{
signal(, child_handle);
char file_name[] = "";
char client_r[], client_w[];
char line[] ;
int fd_read ;
int client_id ;
int fd_cr , fd_cw ;
FILE* fp ;
sprintf(file_name, "%s/%s", PATH, NAME);
mkfifo(file_name, ) ;
fd_read = open(file_name, O_RDONLY);
open(file_name, O_WRONLY); fp = fdopen(fd_read, "r"); while(memset(line, , ), fgets(line, , fp) != NULL)// "pid\n"
{// cr cw pid_r.fifo pid_w.fifo
sscanf(line, "%d", &client_id);
printf("client: %d request !\n", client_id) ;
memset(client_r, , );
memset(client_w, , ); sprintf(client_r, "%s/%d_r.fifo", PATH, client_id); sprintf(client_w, "%s/%d_w.fifo", PATH, client_id); fd_cw = open(client_r, O_WRONLY);
fd_cr = open(client_w, O_RDONLY);
if(fork() == )
{
child_main(fd_cr, fd_cw);
close(fd_cr);
close(fd_cw);
exit();
}
close(fd_cr);
close(fd_cw);
} memset(file_name, , );
sprintf(file_name, "%s/%s", PATH, NAME);
unlink(file_name);
return ;
}

client.h:

 #ifndef __CLINET_H__
#define __CLINET_H__ #include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/shm.h>
#include<sys/sem.h>
typedef struct tag
{
int m_flag ;
char m_buf[] ;
}MBUF, *pMBUF;
void P(int semid) ;
void V(int semid);
#endif

client_in:

 #include "client.h"
#include<sys/stat.h>
#include<sys/types.h>
#include<sys/fcntl.h>
#define PATH "/home/comst/pipe"
#define SERVER "server.fifo"
int main(int argc, char* argv[])//shm_key sem_key
{ char server_name[]= "" ;
char read_file[], write_file[] ;
char msg[] ="" ;
int fd_r, fd_w ;
sprintf(server_name, "%s/%s", PATH, SERVER);
int fd_server = open(server_name, O_WRONLY);
sprintf(msg, "%d\n", getpid());
write(fd_server, msg, strlen(msg)); memset(read_file, , );
memset(write_file, , ); sprintf(read_file, "%s/%d_r.fifo", PATH, getpid());
sprintf(write_file, "%s/%d_w.fifo", PATH, getpid()); mkfifo(read_file, );
mkfifo(write_file, ); fd_r = open(read_file, O_RDONLY);
fd_w = open(write_file, O_WRONLY); key_t shm_key, sem_key ;
int my_shm, my_sem ; char line[] ; pMBUF p ;
shm_key = (key_t)atoi(argv[]);
sem_key = (key_t)atoi(argv[]); my_shm = shmget(shm_key, sizeof(MBUF), |IPC_CREAT); my_sem = semget(sem_key, , | IPC_CREAT);
semctl(my_sem, , SETVAL, ); p = (pMBUF)shmat(my_shm, NULL, );
memset(p, , sizeof(MBUF)); while(memset(line, , ), fgets(line, , stdin) != NULL)
{
write(fd_w, line, strlen(line)); memset(line, , ); read(fd_r, line, );
while( P(my_sem), p -> m_flag == )
{
V(my_sem);
sleep();
}
strcpy(p ->m_buf, line);
p ->m_flag = ;
V(my_sem); } while( P(my_sem), p -> m_flag == )
{
V(my_sem);
sleep();
}
strcpy(p ->m_buf, "over");
p ->m_flag = ;
V(my_sem); sleep(); shmdt(p);
shmctl(my_shm, IPC_RMID, NULL) ; semctl(my_sem, , IPC_RMID); }

client_out:

 #include "client.h"
int main(int argc, char* argv[])//shm_key sem_key
{
key_t shm_key, sem_key ;
int my_shm, my_sem ; char line[] ; pMBUF p ;
shm_key = atoi(argv[]);
sem_key = atoi(argv[]); my_shm = shmget(shm_key, sizeof(MBUF), ); my_sem = semget(sem_key, , );
semctl(my_sem, , SETVAL, ); p = (pMBUF)shmat(my_shm, NULL, );
memset(p, , sizeof(MBUF)); while()
{
while(P(my_sem), p -> m_flag == )
{
V(my_sem);
sleep();
}
printf("%d : %s\n", getpid(), p -> m_buf);
if(strcmp(p ->m_buf, "over") == )
{ V(my_sem);
break ;
}
p -> m_flag = ;
V(my_sem); } shmdt(p); }

P_V func

 #include "client.h"
void P(int semid)
{
struct sembuf my_buf ;
memset(&my_buf, , sizeof(my_buf) );
my_buf.sem_num = ;
my_buf.sem_op = - ;
my_buf.sem_flg = SEM_UNDO ;
semop(semid, &my_buf, );
}
void V(int semid)
{
struct sembuf my_buf ;
memset(&my_buf, , sizeof(my_buf) );
my_buf.sem_num = ;
my_buf.sem_op = ;
my_buf.sem_flg = SEM_UNDO ;
semop(semid, &my_buf, ); }