首先,需要了解wayland的工作原理,需要了解几个其他的内容。
众所周知,子进程会继承父进程已经打开的文件描述符fd,但是fork之后的是不会被继承的,这个时候是否无能无力了?答应是NO。Linux提供了一个系统调用sendmsg,借助它,可以实现进程间传递文件描述符fd,而且不仅限于父进程到子进程。sendmsg函数的原型如下:
#include
ssize_t sendmsg(int socket, const struct msghdr *message, int flags);
ssize_t recvmsg(int socket, struct msghdr *message, int flags);
recvmsg函数用来接收fd,这里的socket必须为UnixSocket(AF_UNIX),在Linux上执行man 7 unix,并搜索SCM_RIGHTS,即看到有关说明:Send or receive a set of open file descriptors from another process。
通过sendmsg发送的fd,并不是将fd值传递给目标进程,而是活生生地在目标进程空间里复制指向同一个file结构体的fd,所以不要期望在两个进程中,fd值相同。
具体的使用示例,请baidu或google关键词:sendmsg fd,即可找到,这里就不多说了。
int epoll_create(int size) // 创建epoll fd描述符
int epoll_ctl(int epfd, int op, int fd, struct epoll_event *event) // 监听其他的文件描述符
int epoll_wait(int epfd, struct epoll_event * events, int maxevents, int timeout) // 启动监听
https://baike.baidu.com/item/epoll%E5%87%BD%E6%95%B0/10792875
int bind(SOCKET socket, const struct sockaddr* address, socklen_t address_len);
// 在上一个函数使用AF_UNIX参数创建时, bind函数指定一个文件路径来实现进程间通信, 使用结构体(sockaddr_un)保存文件路径
int listen( int sockfd, int backlog); // 启动监听网络套接字
int accept( int fd, struct socketaddr* addr, socklen_t* len); // 接受连接进来的客户端
performed using descriptor FD.*/
extern int signalfd (int __fd, const sigset_t *__mask, int __flags) // 产生信号监听fd
/* Clear all signals from SET. */
extern int sigemptyset (sigset_t *__set) __THROW __nonnull ((1));
/* Set all signals in SET. */
extern int sigfillset (sigset_t *__set) __THROW __nonnull ((1));
/* Add SIGNO to SET. */
extern int sigaddset (sigset_t *__set, int __signo) __THROW __nonnull ((1));
// 让信号在被处理之前程序都处于阻塞状态
extern int sigprocmask (int __how, const sigset_t *__restrict __set,
sigset_t *__restrict __oset)
// 以上API 创建一个信号的fd
我来描述一下,wayland是怎么把上面的知识点给联系起来的。
首先,我们必须要清楚wayland是一个CS架构的协议,所以肯定涉及到通信,但是wayland的CS局限于同一个操作系统,因为底层会利用操作系统的特性来传递进程中的文件描述符(文件句柄),接下来我们分为服务器端和客户端来分析:
以上就是简单的描述wayland的工作原理,下面开始更详细的描述wayland的工作原理。不过,在进行更详细的原理介绍之前,我必须要给大家把wayland通过的接口介绍一遍,其实wayland接口明白之后,读者应该就能明白大概的原理了.