配置dbus和dbus-daemon

上一级页面:index-wsl

前言

D-Bus daemon是D-Bus的非常重要的一个服务,类似于IP网络中的路由器。

初始化dbus

"machine ID"通常在系统安装或首次启动时从一个随机数源生成, 并且之后无论经过多少次开关机,也一直保持不变。

ubuntu一般不用安装dbus,它自带这个包,ubuntu与ui交互需要D-Bus

1
2
sudo apt install dbus dbus-x11
sudo apt-mark auto dbus dbus-x11

生成dbus机器码(machine-id)

1
sudo systemd-machine-id-setup

配置systemd

wsl2的systemd是使用init实现的,可以更换为更完整的systemd实现,解决方案见wsl2开启systemctl命令简单方法

dbus不能正常连接

1
# Failed to connect to socket /var/run/dbus/system_bus_socket: No such file or directory

dbus是否启动?如果没启动,则:

1
sudo /etc/init.d/dbus start

dbus启动了,守护进程dbus-daemon是否启动?如果没启动,则:

1
sudo dbus-daemon --system

如何配置Dbus

1
sudo dpkg-reconfigure dbus

重启wsl

1
sudo systemctl reboot

dbus-daemon介绍

D-Bus daemon是D-Bus的非常重要的一个服务,类似于IP网络中的路由器。

跟dbus-daemon这个后台服务有关的应用程序包括:

dbus-daemon: D-Bus的后台进程,作为D-Bus的消息中转枢纽,可分成system和session两种。

dbus-launch: 启动一个dbus-daemon,后面有不同的参数。一般而言,dbus-daemon启动后需要将其地址告诉给所有需要使用该bus的applications。

当系统启动时,需要使用dbus-launch来启动dbus-daemon,一般而言,

一般采用下面的命令启动dbus daemon以及dbus application

(1)

1
eval `dbus-launch --auto-syntax`

(2)

1
./yourapp

第一行代码,采用eval来执行两次,第一次执行dbus-lauch --auto-syntax,除了启动dbus daemon之外,还输出了下面的内容:

1
2
3
4
5
DBUS_SESSION_BUS_ADDRESS='unix:path=/tmp/dbus-6Z62FMmwf3,guid=5dbd92e4865a3f56880d2120000000d6';

export DBUS_SESSION_BUS_ADDRESS;

DBUS_SESSION_BUS_PID=998;

第二次执行时就将环境变量DBUS_SESSION_BUS_ADDRESS暴露出去了。所有的dbus application在注册DBUS服务时,必须知道这个DBUS_SESSION_BUS_ADDRESS的数据。

在第二行执行自己的dbus application的时候,根据DBUS_SESSION_BUS_ADDRESS环境变量,能够找到session bus进行注册和通讯。

最终的DBUS_SESSION_BUS_ADDRESS配置脚本见wsl2-x11-环境变量配置

理解

dbus-daemon是一个后台进程,负责消息的转发。它就像个路由器。最常见的基于dbus的程序也是符合C/S结构的。

比如我们自己写了两个程序,A和B,其中A是客户,B是服务。假设A要调用B的一个函数C,那么实际的消息流动方向是:A告诉dbus-daemon我要调用B的C函数,然后dbus-daemon则去调用B的C函数,如果C有返回值的话,B会把返回值告诉dbus-daemon,然后dbus-daemon再把返回值告诉A。由此可以看出,dbus-daemon是很关键的一个后台进程。

以ubuntu为例,通常情况下,会有两个dbus-daemon进程,一个是属于system的,一个是属于session的,这两个进程,都是在用户登录的时候由dbus-launch启动的。

大多数普通程序,都是使用session的dbus-daemon,默认情况下,A就是将消息发给了这个属于session的dbus-daemon。

但是在调试过程中,有时想看到一些更详细的信息,因此可以自己再启动一个dbus-daemon,现在就有一个问题,A的消息是流向了哪一个dbus-daemon?

其实dbus-daemon是有地址的,而且有一个环境变量来表示它–DBUS_SESSION_BUS_ADDRESS,可以用命令env查看到。我们的程序,也就就是依靠这个环境变量来确认使用哪一个dbus-daemon的。

当我们登录进桌面环境的时候,系统启动脚本会调用到dbus-launch来启动一个dbus-daemon,同时会把这个dbus-daemon的地址赋予环境变量DBUS_SESSION_BUS_ADDRESS。

一般情况下,我们是不需要考虑DBUS_SESSION_BUS_ADDRESS的,但是,有些时候,单独启动一个dbus-daemon,则有助于程序的调试。

比如,还是假设我们写了两个程序,A和B,其中A是客户,B是服务。同时还使用dbus的自启动功能(简要说明一下,什么是dbus的自启动----A要调用B的函数,如果B进程还没有启动,则dbus-daemon会自动的先把B进程启动起来,这就是自启动)。

由于利用dbus-daemon自启动机制运行的服务进程,都是后台进程,标准输出设备已经被重定向了,如果B进程有一些调试用的打印信息,则很难直接查看。这个时候,就可以单独启动一个dbus-daemon,让A和B都使用我们自己启动的这个dbus-daemon,此时,这个dbus-daemon还能把B的打印信息显示出来。

先在终端下启动一个dbus-daemon,命令如下形式如下:

1
DBUS_VERBOSE=1 dbus-daemon --session --print-address 

这样的话,该dbus-daemon会前台执行,并且打印出它的地址,地址形式类似于unix:abstract=/tmp/dbus-YcjSNNPJHg,guid=18b385acdbd58611ffd3196b4beb69f0

然后,在执行我们的A程序的时候,设置环境变量DBUS_SESSION_BUS_ADDRESS为刚才得到的地址值,例如这样执行A程序:

1
DBUS_SESSION_BUS_ADDRESS=unix:abstract=/tmp/dbus-YcjSNNPJHg,guid=18b385acdbd58611ffd3196b4beb69f0  ./A

这样的话,我们的程序A和B,就是使用我们自己启动的这个dbus-daemon来转发消息了。同时,刚才启动dbus-daemon的那个终端,还会把B的打印信息显示出来。

理解上面说的这个后,还可以做的更多。

dbus-launch是一个用来启动dbus-daemon的脚本,它也会打印出它启动的那个dbus-daemon的地址。

所有使用到dbus技术的程序,都会用到DBUS_SESSION_BUS_ADDRESS环境变量,所以,我们也可以用这个环境变量来控制dbus-monitor的行为。

参考、引用、致谢

linux进程之dbus-daemon_海斌的技术博客的技术博客_51CTO博客

©著作权归作者所有:来自51CTO博客作者kasonbin的原创作品,请联系作者获取转载授权,否则将追究法律责任

linux进程之dbus-daemon

https://blog.51cto.com/kasonbin/1352140