博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
unix平台下I/O聚集和分离的一种方案
阅读量:6124 次
发布时间:2019-06-21

本文共 3334 字,大约阅读时间需要 11 分钟。

最近学习了两个unix平台下两个软件的构架,其中一个就是大名鼎鼎的oracle,首先oracle的体系结构官方又很多资料,

以前的大牛也分析了很多,我就不说了(因为说起了又是一篇文章)。我对它们的I/O处理有一些认识,猜测其使用的技术。

基本是前端有个listener接受请求,让后将“请求”发给server进程(这个里面有复杂的session机制在里面)。

unix下“请求”的本质就是“文件描述符”所以这篇文章的技术本质就是“进程间传输文件描述符”!

以下是我对中间件I/O处理的一个实现:

假设一个系统有大量的不同的I/O请求(这个在中间件开发中尤其常见),我们的怎个平台设计可以类oracle

listener----〉server当然可以是多对多的关系!

listener:这里我们可以做I/O路由,请求缓存,甚至可以是异步处理!

server:当然就是业务逻辑的处理。

实现:

listener创建一个tcp服务,一个map(做I/O路由),unix socket(用于和server之间传输fd),一块共享内存(用来缓存请求的数据)

server创建一个unix socket(用于接收listener发送的文件描述符)。

服务的流程:

client---(tcp)--〉listener---(unix socket)---〉server

server--〉结果返回给client

我实现了一份具体的代码,可以去这里下载:

以下就是进程间传输文件描述符需要的两个功能函数:

/*

* @file    server_utility.c
* @author  Luo ZhiHui <camel.flying@gmail.com>
* @version 1.0
*
* @section LICENSE
*
*
* @section DESCRIPTION
* Unix domain socket transfer multiple file descripers.
*/

#include <sys/types.h>

#include <sys/socket.h>

#define SEND_FD_NUMBER 2

#define    HAVE_MSGHDR_MSG_CONTROL
#define UNIX_SOCKET_PATH "/tmp/transmissionFD"

ssize_t send_fd(int fd, void *ptr, size_t nbytes, int *send_fd_array, size_t send_fd_num)

{
    struct msghdr    msg;
    struct iovec    iov[1];
    int index=0;

#ifdef    HAVE_MSGHDR_MSG_CONTROL

    union {
      struct cmsghdr    cm;
      char                control[CMSG_SPACE(sizeof(int)*send_fd_num)];
    } control_un;
    struct cmsghdr    *cmptr;

    msg.msg_control = control_un.control;

    msg.msg_controllen = sizeof(control_un.control);

    cmptr = CMSG_FIRSTHDR(&msg);

    cmptr->cmsg_len  =CMSG_LEN(sizeof(int)*send_fd_num);
    cmptr->cmsg_level = SOL_SOCKET;
    cmptr->cmsg_type = SCM_RIGHTS;
       
    for(index=0; index<send_fd_num; index++)
    {
        *((int*)(CMSG_DATA(cmptr)+sizeof(int)*index)) = send_fd_array[index];
    }
#else
    msg.msg_accrights = (caddr_t)send_fd_array
    msg.msg_accrightslen = sizeof(int)*send_fd_num;
#endif

    msg.msg_name = NULL;

    msg.msg_namelen = 0;

    iov[0].iov_base = ptr;

    iov[0].iov_len = nbytes;
    msg.msg_iov = iov;
    msg.msg_iovlen = 1;

    return(sendmsg(fd, &msg, 0));

}

ssize_t recv_fd(int fd, void *ptr, size_t nbytes, int *recv_fd_array, size_t recv_fd_num)

{
    struct msghdr    msg;
    struct iovec    iov[1];
    ssize_t            n;
    int index;

#ifdef    HAVE_MSGHDR_MSG_CONTROL

    union
    {
      struct cmsghdr    cm;
      char                control[CMSG_SPACE(sizeof(int)*(recv_fd_num))];
    } control_un;
    struct cmsghdr    *cmptr;

    msg.msg_control = control_un.control;

    msg.msg_controllen = sizeof(control_un.control);
#else
    int    newfd;

    msg.msg_accrights = (caddr_t)recv_fd_array;

    msg.msg_accrightslen = sizeof(int)*recv_fd_num;
#endif

    msg.msg_name = NULL;

    msg.msg_namelen = 0;

    iov[0].iov_base = ptr;

    iov[0].iov_len = nbytes;
    msg.msg_iov = iov;
    msg.msg_iovlen = 1;

    if ((n=recvmsg(fd, &msg, 0)) <= 0)

        return(n);

#ifdef    HAVE_MSGHDR_MSG_CONTROL

    if((cmptr = CMSG_FIRSTHDR(&msg))!=NULL && cmptr->cmsg_len==CMSG_LEN(sizeof(int)*(recv_fd_num)))
    {
        if(cmptr->cmsg_level != SOL_SOCKET)
        {
            return -1;
        }
        if(cmptr->cmsg_type != SCM_RIGHTS)
        {
            return -2;
        }
        for(index=0; index<recv_fd_num; index++)
        {
            recv_fd_array[index] =*((int *)(CMSG_DATA(cmptr)+sizeof(int)*index));
        }
    }
    else
    {
        recv_fd_array[0] = -1;        /* descriptor was not passed */
    }
#else
    if(msg.msg_accrightslen == sizeof(int)*recv_fd_num)
    {
        ;
    }
    else
    {
        recv_fd_array[0] = -1;        /* descriptor was not passed */
    }
#endif

    return(n);

}

申明:以上代码载自unix环境高级编程,参考代码是传输一个fd,本人将其改成一次传输多个fd(不知道是不是一种改进)在linux和aix下测试过

关于以上代码的解释可以参照这里:

参考资料:

steven unix高级环境编程,

unix网络编程I

unix网络编程II

 

转载于:https://www.cnblogs.com/OCaml/archive/2012/02/26/2368481.html

你可能感兴趣的文章
C 函数sscanf()的用法
查看>>
python模块之hashlib: md5和sha算法
查看>>
linux系统安装的引导镜像制作流程分享
查看>>
解决ros建***能登录不能访问内网远程桌面的问题
查看>>
pfsense锁住自己
查看>>
vsftpd 相关总结
查看>>
bash complete -C command
查看>>
解决zabbix 3.0中1151端口不能运行问题
查看>>
售前工程师的成长---一个老员工的经验之谈
查看>>
Get到的优秀博客网址
查看>>
dubbo
查看>>
【Git入门之四】操作项目
查看>>
老男孩教育每日一题-第107天-简述你对***的理解,常见的有哪几种?
查看>>
Python学习--time
查看>>
在OSCHINA上的第一篇博文,以后好好学习吧
查看>>
高利率时代的结局,任重道远,前途叵测
查看>>
Debian 6.05安装后乱码
查看>>
欢迎大家观看本人录制的51CTO精彩视频课程!
查看>>
IntelliJ IDEA中设置忽略@param注释中的参数与方法中的参数列表不一致的检查
查看>>
关于软件开发的一些感悟
查看>>