PS:要转载请注明出处,本人版权所有。

PS: 这个只是基于《我自己》的理解,

如果和你的原则及想法相冲突,请谅解,勿喷。

环境说明

  无

前言


  在《记一次有趣的hwclock写RTC的PermissionDenied错误》(https://www.cnblogs.com/Iflyinsky/p/17841708.html 或者 https://https/flyinskyin2013.github.io/2023/11/19/blog_idx_125/)中,我们提到了关于高通板卡,我们无法通过hwclock来写入rtc时钟。但是我们通过相关调查、测试、分析后发现,高通自己搞了一套新的方式来写入rtc时钟,这种特殊方法用的技术就是abstract socket。这也是我在工作中第一次遇到abstract socket,这里做一个简要的记录。

Abstract sockets


  

说明

  我们先不看代码,我们先看看abstract sockets具体是什么样子的。我们都知道,我们可以通过netstat查看unix socket相关的连接信息。我们来看看ubuntu 18.04里面用netstat查看unix socket信息如下:

  注意图中我们看到的path那一列,除了普通的路径外,我们会看到一些奇怪的名字,每个名字前面都有一个@符号。这种特殊的名字,就是abstract sockets。

简介

  首先,根据文档,对于传统的uds来说,其支持两种类型,它们分别是:有名字的、未命名的。对于linux来说,这里还支持一种独立于文件系统的,且是一种不可移植的扩展,这就是我们要介绍的就是抽象的这种类型。

  对于抽象的uds来说,socket权限对其来说是没有意义的,例如umask、fchown、fchmod等不会影响其访问权限。

  对于抽象的uds来说,抽象的uds将会自动被销毁,当所有的引用打开的socket关闭的时候。

  对于抽象的uds来说,和有名字的uds最大的区别在于sockaddr_un.sun_path[0]是0x00,如果大家对c风格的字符串有印象,这是一个字符串的终止符。abstract uds的名字由sockaddr_un.sun_path的其余字节给出(这个时候的0x00是没有特殊意义的),其长度是addrlen - sizeof(sa_family_t)来定义的。

使用小例子

server

#include <sys/socket.h>
#include <sys/un.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <stdio.h> int main(int argc, char * argv[])
{
int uds = socket(AF_UNIX, SOCK_STREAM, 0);
if (uds < 0) perror("socket:"); struct sockaddr_un addr;
addr.sun_family = AF_UNIX;
memset(addr.sun_path, 0x0, 108);
const char * addr_path = "abstract_test";
memcpy(addr.sun_path + 1, addr_path, strlen(addr_path)); int ret = bind(uds, &addr, strlen(addr_path) + sizeof(sa_family_t) + 1);
if (ret < 0) perror("bind:"); ret = listen(uds, 2);
if (ret < 0) perror("listen:"); struct sockaddr_un r_addr;
socklen_t r_addr_len;
int _new = accept(uds, &r_addr, &r_addr_len);
if (_new < 0) perror("accept:");
char r_msg[256];
memset(r_msg, 0x0, 256);
ret = read(_new, r_msg, 256);
if (ret < 0) perror("read:");
printf("client msg = %s\n", r_msg);
write(_new, "hello client", sizeof("hello client")); sleep(5); close(uds);
return 0;
}

client.c

#include <sys/socket.h>
#include <sys/un.h>
#include <unistd.h>
#include <errno.h>
#include <string.h>
#include <stdio.h> int main(int argc, char * argv[])
{
int uds = socket(AF_UNIX, SOCK_STREAM, 0);
if (uds < 0) perror("socket:"); struct sockaddr_un addr;
addr.sun_family = AF_UNIX;
memset(addr.sun_path, 0x0, 108);
const char* addr_path = "abstract_test";
memcpy(addr.sun_path + 1, addr_path, strlen(addr_path)); int ret = connect(uds, &addr, strlen(addr_path) + sizeof(sa_family_t) + 1);
if (ret < 0) perror("connect:"); write(uds, "hello server", sizeof("hello server")); char r_msg[256];
memset(r_msg, 0x0, 256);
read(uds, r_msg, 256);
printf("server msg %s\n", r_msg); sleep(5);
close(uds);
return 0;
}

  首先我们撸了两个例子,一个是服务端,一个是客户端。

  首先我们运行客户端。然后通过netstat查看我们的socket。如下图:

  当我们运行服务端后,再次运行客户端,就会看到我们传输的消息,如下图:

后记


  通过我们如上的实验,可以看到其和普通的uds用法差不多,唯一的区别就是其不需要和文件系统绑定了。

  其实,这种不和文件系统绑定的特性,可以给我们带来更多有趣的用法(例如:不受文件系统权限影响进行通信),后续有缘分享。

参考文献


打赏、订阅、收藏、丢香蕉、硬币,请关注公众号(攻城狮的搬砖之路)

PS: 请尊重原创,不喜勿喷。

PS: 要转载请注明出处,本人版权所有。

PS: 有问题请留言,看到后我会第一时间回复。

初识uds之abstract socket的更多相关文章

  1. Envoy 基础教程:使用 Unix Domain Socket(UDS) 与上游集群通信

    Envoy Proxy 在大多数情况下都是作为 Sidecar 与应用部署在同一网络环境中,每个应用只需要与 Envoy(localhost)交互,不需要知道其他服务的地址.然而这并不是 Envoy ...

  2. Java Socket:Java-NIO-ServerSocketChannel

    ServerSocketChannel 让我们从最简单的ServerSocketChannel来开始对socket通道类的讨论 ServerSocketChannel是一个基于通道的socket监听器 ...

  3. Unix domain socket IPC

    UNIX Domain socket 虽然网络socket也可用于同一台主机的进程间通讯(通过lo地址127.0.0.1),但是unix domain socket用于IPC更有效率:不需要经过网络协 ...

  4. socket编程相关阐述

    一.socket初识 ①服务端 import socket server = socket.socket() server.bind(('127.0.0.1', 8080)) server.liste ...

  5. PYTHON学习之路_PYTHON基础(8)

    学习内容: Python模块介绍 1.经典类 or 新式类 2.抽象接口 3.静态方法.类方法.属性方法 4.反射 5.异常处理 6.socket编程初识 7.用socket实现get.put文件等功 ...

  6. 分布式计算框架Gearman原理详解

    什么是Gearman? Gearman提供了一个通用的应用程序框架,用于将工作转移到更适合于工作的其他机器或流程.它允许你并行工作,负载平衡处理,并在语言间调用函数.它可用于从高可用性网站到传输数据库 ...

  7. gearman(异步计算)学习

    Gearman是什么? 它是分布式的程序调用框架,可完成跨语言的相互调 用,适合在后台运行工作任务.最初是2005年perl版本,2008年发布C/C++版本.目前大部分源码都是(Gearmand服务 ...

  8. Linux configure关于交叉编译的参数设置【转】

    转自:http://blog.csdn.net/darennet/article/details/9003005 configure的参数众多,一般包括如下 --srcdir=DIR 这个选项对安装没 ...

  9. gearman参数说明

    -b, –backlog=BACKLOG 连接请求队列的最大值 -d, –daemon Daemon 守护进程化 -f, –file-descriptors=FDS 可打开的文件描述符数量 -h, – ...

  10. gearman知识文章

    一篇文章: Gearman介绍.调研.测试与原理分析 gearman是什么? 它是分布式的程序调用框架,可完成跨语言的相互调用,适合在后台运行工作任务.最初是2005年perl版本,2008年发布C/ ...

随机推荐

  1. ZR 七连 Day 1 游记

    ZR 七连 Day 1 游记 游记篇 赛前搞笑事件 今天是第一场正睿,还是要 好好对待 的 $ 17:59:58 $ 还在吃饭 $ 17:59:59 $ 做出重要决定,先打着比赛,有空就吃一口包子 $ ...

  2. phpwind论坛,后台老是有缓存 不及时更新,操作无效等问题的解决方法。

  3. NC20573 [SDOI2011]染色

    题目链接 题目 题目描述 给定一棵有n个节点的无根树和m个操作,操作有2类: 1.将节点a到节点b路径上所有点都染成颜色c: 2.询问节点a到节点b路径上的颜色段数量(连续相同颜色被认为是同一段),如 ...

  4. NC16619 [NOIP2008]传球游戏

    题目链接 题目 题目描述 上体育课的时候,小蛮的老师经常带着同学们一起做游戏.这次,老师带着同学们一起做传球游戏. 游戏规则是这样的:n个同学站成一个圆圈,其中的一个同学手里拿着一个球,当老师吹哨子时 ...

  5. python 中异常类型总结

    异常类型: 异常名称 描述BaseException             所有异常的基类SystemExit                   解释器请求退出KeyboardInterrupt  ...

  6. tensorflow中交叉熵损失函数详解

    1 前言 tensorflow中定义了3个交叉熵损失函数: softmax_cross_entropy_with_logits(logits, labels) softmax_cross_entrop ...

  7. java generic 介绍

    一 介绍: 在Java SE 1.5之前,没有泛型的情况的下,通过对类型Object的引用来实现参数的"任意化","任意化"带来的缺点是要做显式的强制类型转换, ...

  8. Spring Boot图书管理系统项目实战-3.用户登录

    导航: pre:  2.项目搭建 next:4.基础信息管理 只挑重点的讲,具体的请看项目源码. 1.项目源码 需要源码的朋友,请捐赠任意金额后留下邮箱发送:) 2.登录页设计 <!DOCTYP ...

  9. 使用webgl(three.js)创建自动化抽象化3D机房,3D机房模块详细介绍(抽象版一)

    目前市面上有两种机房 一种是普通机房 一种是由微模块组成的机房,本文主要介绍普通机房的抽象化体现模式. 抽象机房模式:机房展示过程中,我们需要对机房进行建模,当遇到大量机房需要建模时,这无疑是巨大工作 ...

  10. FileBeat简单使用

    简介 首先要了解ELK架构 这种结构因为需要在各个服务器上部署 Logstash,而它比较消耗 CPU 和内存资源,所以比较适合计算资源丰富的服务器,否则容易造成服务器性能下降,甚至可能导致无法正常工 ...