前段时间写一个传递文件句柄的小 demo,有 server 端、有 client 端,之间通过 Unix Domain Socket 通讯。

在普通模式下,双方可以正常建立连接,当server端作为daemon启动时,则第一次启动成功,之后再启动, listen 会连接报 ENOTSUPP 错误,导致启动失败。

spipe.c

 int cli_conn(const char *name)
{
int fd, len, err, rval;
struct sockaddr_un un; if ((fd = socket (AF_UNIX, SOCK_STREAM, )) < ) {
printf ("create socket failed\n");
return -;
} printf ("create socket ok\n");
memset (&un, , sizeof (un));
un.sun_family = AF_UNIX;
strcpy (un.sun_path, name);
len = offsetof (struct sockaddr_un, sun_path) + strlen (name);
if (connect (fd, (struct sockaddr *)&un, len) < ) {
err = errno;
printf ("connect failed\n");
rval = -;
goto errout;
} printf ("connect to server ok\n");
return fd;
errout:
close (fd);
errno = err;
return rval;
} int serv_listen (const char *name)
{
int fd, len, err, rval;
struct sockaddr_un un; if ((fd = socket (AF_UNIX, SOCK_STREAM, )) < ) {
printf ("socket failed\n");
return -;
} printf ("create socket ok\n");
unlink (name);
memset (&un, , sizeof(un));
un.sun_family = AF_UNIX;
strcpy (un.sun_path, name);
len = offsetof (struct sockaddr_un, sun_path) + strlen (name); if (bind (fd, (struct sockaddr *)&un, len) < ) {
err = errno;
printf ("bind failed\n");
rval = -;
goto errout;
} printf ("bind socket to path ok\n");
if (listen (fd, QLEN) < ) {
err = errno;
printf ("listen failed, errno %d\n", errno);
rval = -;
goto errout;
} printf ("start listen on socket ok\n");
return fd;
errout:
close (fd);
errno = err;
return rval;
} int serv_accept (int listenfd, uid_t *uidptr)
{
int clifd, err, rval;
time_t staletime;
struct sockaddr_un un;
struct stat statbuf; size_t len = sizeof (un);
if ((clifd = accept (listenfd, (struct sockaddr *)&un, &len)) < ) {
printf ("accept failed\n");
return -;
} len -= offsetof (struct sockaddr_un, sun_path);
un.sun_path[len] = ;
printf ("accept %s ok\n", un.sun_path); unlink (un.sun_path);
return clifd; errout:
close (clifd);
errno = err;
return rval;
}

出错的位置在 serv_listen (line 57) 处,出错时的 server 端输出为:

Jan 17 00:24:44 localhost opend: create socket ok
Jan 17 00:24:44 localhost opend: bind socket to path ok
Jan 17 00:24:44 localhost opend: listen failed, errno 95
Jan 17 00:24:44 localhost opend: serv_listen error: Operation not supported

errno 95 为 ENOTSUPP。不以 daemon 运行时正常的输出如下:

create socket ok
bind socket to path ok
start listen on socket ok
accept ok
new connection: uid 0, fd 4

可能细心的读者会觉得,以 daemon 方式运行 printf 怎么还可以输出呢,是有以下宏定义做了处理:

 #ifdef USE_APUE
#include "../apue.h"
#define printf log_msg
#endif

以 daemon 运行时会定义 USE_APUE 宏,从而将 printf 重定义为 log_msg 输出到 syslog。

下面是 server 端的代码:

csopend2.c

 int main (int argc, char *argv[])
{
int c = ;
log_open ("open.serv", LOG_PID, LOG_USER); opterr = ; // don't want getopt() writting to stderr !
while ((c = getopt (argc, argv, "d")) != EOF) {
switch (c) {
case 'd':
debug = log_to_stderr = ;
break;
case '?':
err_quit ("unrecongnized option: -%c", optopt);
}
} if (debug == )
{
log_to_stderr = ;
daemonize ("opend");
} loop ();
return ;
}

不使用 -d 时表示 daemon 运行(与常识相反?),上面标黄的代码就是。

对应的 client 端代码:

csopenc.c

一开始怀疑是用于 listen 的本地 socket 文件已经存在,于是去 /tmp 目录看了下,果然有 opend 这个文件,删除之,再运行,不行;

然后怀疑是没有复用端口(?)所致,于是在 listen 之前添加了以下代码段:

     int opt = ;
if (setsockopt(fd, SOL_SOCKET, SO_REUSEADDR, (const void *)&opt, sizeof(opt)) < ) {
err = errno;
printf ("setsockopt failed\n");
rval = -;
goto errout;
}

设置端口复用。编译、运行,输出如下:

Jan 17 00:43:11 localhost opend: create socket ok
Jan 17 00:43:11 localhost opend: bind socket to path ok
Jan 17 00:43:11 localhost opend: set socket option ok
Jan 17 00:43:11 localhost opend: listen failed, errno 95
Jan 17 00:43:11 localhost opend: serv_listen error: Operation not supported

设置成功了,但还是不行

难道 daemon 与普通进程使用 Unix 域套接字还有什么区别么?

暂时存疑……

[apue] 作为 daemon, 启动 Unix Domain Socket 侦听失败?的更多相关文章

  1. 由一个简单需求到Linux环境下的syslog、unix domain socket

    本文记录了因为一个简单的日志需求,继而对linux环境下syslog.rsyslog.unix domain socket的学习.本文关注使用层面,并不涉及rsyslog的实现原理,感兴趣的读者可以参 ...

  2. 问题解决:psql: could not connect to server: No such file or directory Is the server running locally and accepting connections on Unix domain socket "/var/run/postgresql/.s.PGSQL.5432"?

    错误提示: psql: could not connect to server: No such file or directory Is the server running locally and ...

  3. Unix domain socket 简介

    Unix domain socket 又叫 IPC(inter-process communication 进程间通信) socket,用于实现同一主机上的进程间通信.socket 原本是为网络通讯设 ...

  4. linux一切皆文件之Unix domain socket描述符(二)

    一.知识准备 1.在linux中,一切皆为文件,所有不同种类的类型都被抽象成文件(比如:块设备,socket套接字,pipe队列) 2.操作这些不同的类型就像操作文件一样,比如增删改查等 3.主要用于 ...

  5. php, hhvm与odp & Unix domain Socket方式

    接上一篇,复习一下 启动php或hhvm: php/sbin/php-fpm start hhvm/bin/hhvm_control start 启动nginx或lighttpd: webserver ...

  6. 问题解决:psql: could not connect to server: No such file or directory Is the server running locally and accepting connections on Unix domain socket "/var/run/postgresql/.s.PGSQL.5432"?

    错误提示: psql: could not connect to server: No such file or directory Is the server running locally and ...

  7. monitor a local unix domain socket like tcpdump

    Can I monitor a local unix domain socket like tcpdump? - Super User https://superuser.com/questions/ ...

  8. ndk学习16: unix domain socket

    一.UNIX Domain Socket 概念: UNIX Domain Socket是在socket架构上发展起来的用于同一台主机的进程间通讯(IPC) 特点: 1. 它不需要经过网络协议栈,不需要 ...

  9. Unix Domain Socket 域套接字实现

    主要注意流程: STREAM SOCKET: Server :  socket() --->  bind() ---> listen()  ---> accept() Client: ...

随机推荐

  1. 2018-2-13-C#-枚举转字符串

    title author date CreateTime categories C# 枚举转字符串 lindexi 2018-2-13 17:23:3 +0800 2018-2-13 17:23:3 ...

  2. no_expand优化案例

    bond 来看一个烂语句: select a.*,b.dn from temp_allcrmuser a, phs_smc_user b  where a.USERNUMBER=b.dn  and ( ...

  3. 【a403】遍历树问题

    Time Limit: 1 second Memory Limit: 32 MB [问题描述] 我们都很熟悉二叉树的前序.中序.后序遍历,在数据结构中常提出这样的问题:已知一棵二叉树的前序和中序遍历, ...

  4. 初次使用pycharm 的interpreter option为空解决办法。

    第一步:进入Setting. 第二步:进入Project中的Project Interpreter.选择添加即可.

  5. Git 删除大文件的方法

    git 仓库中删除历史大文件   git 仓库中删除历史大文件 在git中增加了一个很大的文件,而且被保存在历史提交记录中,每次拉取代码都很大,速度很慢.而且用删除 提交历史记录的方式不是很实际. 以 ...

  6. Codeforces Round #524 (Div. 2) codeforces 1080A~1080F

    目录 codeforces1080A codeforces 1080B codeforces 1080C codeforces 1080D codeforces 1080E codeforces 10 ...

  7. 第二阶段:2.商业需求分析及BRD:4.产品需求分析总结

    产品的需求筛选 战略定位要考虑公司的战略问题.产品定位要分阶段,各个阶段的需求不同. 其实现在需求分析跟筛选都是非常快的. 不把需要当成需求,意思就是不要用户说需要什么就是什么,用户需要引导. 先分类 ...

  8. Spring in action读书笔记(一) 自动化装配bean

    需要引入的ja包 <dependency> <groupId>org.springframework</groupId> <artifactId>spr ...

  9. 【题解】Comet OJ Round 70 简要题解

    [题解]Comet OJ Round 70 简要题解 A 将放在地上的书按照从小到大排序后,问题的本质就变成了合并两个序列使得字典序最小.可以直接模拟归并排序.直接用循环和std::merge实现这个 ...

  10. 洛谷$P1390$ 公约数的和 欧拉函数

    正解:欧拉函数 解题报告: 传送门$QwQ$ 首先显然十分套路地变下形是趴 $\begin{align*}&=\sum_{i=1}^n\sum_{j=1}^n gcd(i,j)\\&= ...