[apue] 作为 daemon, 启动 Unix Domain Socket 侦听失败?
前段时间写一个传递文件句柄的小 demo,有 server 端、有 client 端,之间通过 Unix Domain Socket 通讯。
在普通模式下,双方可以正常建立连接,当server端作为daemon启动时,则第一次启动成功,之后再启动, listen 会连接报 ENOTSUPP 错误,导致启动失败。
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 端的代码:
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 端代码:
一开始怀疑是用于 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 侦听失败?的更多相关文章
- 由一个简单需求到Linux环境下的syslog、unix domain socket
本文记录了因为一个简单的日志需求,继而对linux环境下syslog.rsyslog.unix domain socket的学习.本文关注使用层面,并不涉及rsyslog的实现原理,感兴趣的读者可以参 ...
- 问题解决: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 ...
- Unix domain socket 简介
Unix domain socket 又叫 IPC(inter-process communication 进程间通信) socket,用于实现同一主机上的进程间通信.socket 原本是为网络通讯设 ...
- linux一切皆文件之Unix domain socket描述符(二)
一.知识准备 1.在linux中,一切皆为文件,所有不同种类的类型都被抽象成文件(比如:块设备,socket套接字,pipe队列) 2.操作这些不同的类型就像操作文件一样,比如增删改查等 3.主要用于 ...
- php, hhvm与odp & Unix domain Socket方式
接上一篇,复习一下 启动php或hhvm: php/sbin/php-fpm start hhvm/bin/hhvm_control start 启动nginx或lighttpd: webserver ...
- 问题解决: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 ...
- monitor a local unix domain socket like tcpdump
Can I monitor a local unix domain socket like tcpdump? - Super User https://superuser.com/questions/ ...
- ndk学习16: unix domain socket
一.UNIX Domain Socket 概念: UNIX Domain Socket是在socket架构上发展起来的用于同一台主机的进程间通讯(IPC) 特点: 1. 它不需要经过网络协议栈,不需要 ...
- Unix Domain Socket 域套接字实现
主要注意流程: STREAM SOCKET: Server : socket() ---> bind() ---> listen() ---> accept() Client: ...
随机推荐
- 析构函数 p157
析构函数 确保对象的各部分被正确的清除,及做一些用户指定的其他清理工作. 当对象超出它的作用域时,编译器将自动调用析构函数:手动用new在堆上分配的对象空间,需要调用'delete 对象地址'进行手动 ...
- Python--day41--线程队列
1,普通队列:queue.Queue(),先进先出 import queue q = queue.Queue() #队列 先进先出 q.put(1) q.put(2) q.put(3) q.put(4 ...
- codeforce 378 div 2 F —— Drivers Dissatisfaction (最小生成树,LCA,倍增)
官方题解: If you choose any n - 1 roads then price of reducing overall dissatisfaction is equal to min(c ...
- linux下mysql5.7忘记root密码修改
朋友最近开始学服务器,mysql密码忘了又不会弄,让我帮忙解决一下.重置或修改mysql的root密码这种事平时很少做,还是得google辅助一下,于是弄完了写篇博客记录一下,方便若干月后又有人遇到这 ...
- 思数云hadoop目录
全文检索.数据分析挖掘.推荐系统.广告系统.图像识别.海量存储.快速查询 l Hadoop介绍 n Hadoop来源与历史 n Hadoop版本 n Hadoop开源与商业 l HDFS系统架构 n ...
- Docker 安装nginx 与端口映射
1. 拉取镜像(网易云docker镜像仓库) docker pull hub.c.163.com/library/nginx:latest 2. 运行nignx,并做端口映射 -d 后台运行 -p映 ...
- AbstactFactory模式
AbstractFactory模式就是用来解决这类问题的:要创建一组相关或者相互依赖的对象. AbstractFactory模式关键就是将这一组对象的创建封装到一个用于创建对象的类(ConcreteF ...
- 在Android上为所欲为的一些技术
https://www.jianshu.com/p/87ce6f565d37Android JNI(一)——NDK与JNI基础 https://www.android-doc.com/guide/co ...
- 仿快播APP源码
目录 仿快播系统 一.项目总结三步走 二.项目需求分析 三.搭建框架 四.ORM框架分析 五.功能分析 六.项目开发--仿快播视频 服务端client start.py ---- 启动文件 conf ...
- AutoCad .Net二次开发求两曲线最小距离
测试结果: 主要思路:假设有两条曲线分别是c1和c2,把c1按照1的距离划分我这里用变量jd表示,得到一个曲线集合coll,然后遍历coll,得到coll中每一个曲线的两个端点,再用这两个端点分别求离 ...