`
huangqinqin
  • 浏览: 367106 次
  • 性别: Icon_minigender_2
  • 来自: 福州
社区版块
存档分类
最新评论

DBUS基础知识

 
阅读更多


1.  进程间使用D-Bus通信

    D-Bus是一种高级的进程间通信机制,它由freedesktop.org项目提供,使用GPL许可证发行。D-Bus最主要的用途是在Linux桌面环境为进程提供通信,同时能将Linux桌面环境和Linux内核事件作为消息传递到进程。D-Bus的主要概率为总线,注册后的进程可通过总线接收或传递消息,进程也可注册后等待内核事件响应,例如等待网络状态的转变或者计算机发出关机指令。目前,D-Bus已被大多数Linux发行版所采用,开发者可使用D-Bus实现各种复杂的进程间通信任务。

2.  D-Bus的基本概念

    D-Bus是一个消息总线系统,其功能已涵盖进程间通信的所有需求,并具备一些特殊的用途。D-Bus是三层架构的进程间通信系统,其中包括:

    接口层:接口层由函数库libdbus提供,进程可通过该库使用D-Bus的能力。

    总线层:总线层实际上是由D-Bus总线守护进程提供的。它在Linux系统启动时运行,负责进程间的消息路由和传递,其中包括Linux内核和Linux桌面环境的消息传递。

    包装层:包装层一系列基于特定应用程序框架的Wrapper库。

    D-Bus具备自身的协议,协议基于二进制数据设计,与数据结构和编码方式无关。该协议无需对数据进行序列化,保证了信息传递的高效性。无论是libdbus,还是D-Bus总线守护进程,均不需要太大的系统开销。

    总线是D-Bus的进程间通信机制,一个系统中通常存在多条总线,这些总线由D-Bus总线守护进程管理。最重要的总线为系统总线(System Bus),Linux内核引导时,该总线就已被装入内存。只有Linux内核、Linux桌面环境和权限较高的程序才能向该总线写入消息,以此保障系统安全性,防止有恶意进程假冒Linux发送消息。

    会话总线(Session Buses)由普通进程创建,可同时存在多条。会话总线属于某个进程私有,它用于进程间传递消息。

    进程必须注册后才能收到总线中的消息,并且可同时连接到多条总线中。D-Bus提供了匹配器(Matchers)使进程可以有选择性的接收消息,另外运行进程注册回调函数,在收到指定消息时进行处理。匹配器的功能等同与路由,用于避免处理无关消息造成进程的性能下降。除此以外,D-Bus机制的重要概念有以下几个。

    对象:对象是封装后的匹配器与回调函数,它以对等(peer-to-peer)协议使每个消息都有一个源地址和一个目的地址。这些地址又称为对象路径,或者称之为总线名称。对象的接口是回调函数,它以类似C++的虚拟函数实现。当一个进程注册到某个总线时,都要创建相应的消息对象。

    消息:D-Bus的消息分为信号(signals)、方法调用(method calls)、方法返回(method returns)和错误(errors)。信号是最基本的消息,注册的进程可简单地发送信号到总线上,其他进程通过总线读取消息。方法调用是通过总线传递参数,执行另一个进程接口函数的机制,用于某个进程控制另一个进程。方法返回是注册的进程在收到相关信息后,自动做出反应的机制,由回调函数实现。错误是信号的一种,是注册进程错误处理机制之一。

    服务:服务(Services)是进程注册的抽象。进程注册某个地址后,即可获得对应总线的服务。D-Bus提供了服务查询接口,进程可通过该接口查询某个服务是否存在。或者在服务结束时自动收到来自系统的消息。

    建立服务的流程:
    ----------------------------------
        建立一个dbus连接之后 -- dbus_bus_get(),为这个dbus连接(DbusConnection)起名 -- dbus_bus_request_name(),这个名字将会成为我们在后续进行远程调用的时候的服务名,然后我们进入监听循环 -- dbus_connection_read_write()。在循环中,我们从总线上取出消息 -- dbus_connection_pop_message(),并通过比对消息中的方法接口名和方法名 -- dbus_message_is_method_call(),如果一致,那么我们跳转到相应的处理中去。在相应的处理中,我们会从消息中取出远程调用的参数。并且建立起回传结果的通路 -- reply_to_method_call()。回传动作本身等同于一次不需要等待结果的远程调用。
    发送信号的流程:
    ----------------------------------
        建立一个dbus连接之后,为这个dbus连接起名,建立一个发送信号的通道,注意,在建立通道的函数中,需要我们填写该信号的接口名和信号名 -- dbus_message_new_signal()。然后我们把信号对应的相关参数压进去 -- dbus_message_iter_init_append(); dbus_message_iter_append_basic()。然后就可以启动发送了 -- dbus_connection_send(); dbus_connection_flush。
    进行一次远程调用的流程:
    ----------------------------------
        建立好dbus连接之后,为这dbus连接命名,申请一个远程调用通道 -- dbus_message_new_method_call(),注意,在申请远程调用通道的时候,需要填写服务器名,本次调用的接口名,和本次调用名(方法名)。压入本次调用的参数 -- dbus_message_iter_init_append(); dbus_message_iter_append_basic(),实际上是申请了一个首地址,我们就是把我们真正要传的参数,往这个首地址里面送(送完之后一般都会判断是否内存越界了)。然后就是启动发送调用并释放发送相关的消息结构 -- dbus_connection_send_with_reply()。这个启动函数中带有一个句柄。我们马上会阻塞等待这个句柄给我们带回总线上回传的消息。当这个句柄回传消息之后,我们从消息结构中分离出参数。用dbus提供的函数提取参数的类型和参数 -- dbus_message_iter_init(); dbus_message_iter_next(); dbus_message_iter_get_arg_type(); dbus_message_iter_get_basic()。也就达成了我们进行本次远程调用的目的了。
    信号接收流程:
    ----------------------------------
        建立一个dbus连接之后,为这个dbus连接起名,为我们将要进行的消息循环添加匹配条件(就是通过信号名和信号接口名来进行匹配控制的) -- dbus_bus_add_match()。我们进入等待循环后,只需要对信号名,信号接口名进行判断就可以分别处理各种信号了。在各个处理分支上。我们可以分离出消息中的参数。对参数类型进行判断和其他的处理。
    dbus_connection_read_write()
    --------------------------------------
        As long as the connection is open, this function will block until it can read or write, then read or write, then return #TRUE.
        If the connection is closed, the function returns #FALSE.
    dbus_connection_pop_message()
    --------------------------------------
        Returns the first-received message from the incoming message queue, removing it from the queue. The caller owns a reference to the returned message. If the queue is empty, returns #NULL.
    dbus_connection_send()
    --------------------------------------
        Adds a message to the outgoing message queue. Does not block to write the message to the network; that happens asynchronously. To force the message to be written, call dbus_connection_flush(). Because this only queues the message, the only reason it can
    fail is lack of memory. Even if the connection is disconnected, no error will be returned.
        @param connection the connection.
        @param message the message to write.
        @param serial return location for message serial, or #NULL if you don't care
        @returns #TRUE on success.
    dbus_connection_send_with_reply()
    --------------------------------------
        Queues a message to send, as with dbus_connection_send(), but also returns a #DBusPendingCall used to receive a reply to the message. If no reply is received in the given timeout_milliseconds, this function expires the pending reply and generates a synthetic error reply (generated in-process, not by the remote application) indicating that a timeout occurred.
        A #DBusPendingCall will see a reply message before any filters or registered object path handlers. See dbus_connection_dispatch() for details on when handlers are run.
        A #DBusPendingCall will always see exactly one reply message, unless it's cancelled with dbus_pending_call_cancel().
        If #NULL is passed for the pending_return, the #DBusPendingCall will still be generated internally, and used to track the message reply timeout. This means a timeout error will occur if no reply arrives, unlike with dbus_connection_send().
        If -1 is passed for the timeout, a sane default timeout is used. -1 is typically the best value for the timeout for this reason, unless you want a very short or very long timeout. There is no way to avoid a timeout entirely, other than passing INT_MAX for the
    timeout to mean "very long timeout." libdbus clamps an INT_MAX timeout down to a few hours timeout though.
        @warning if the connection is disconnected, the #DBusPendingCall will be set to #NULL, so be careful with this.
        @param connection the connection
        @param message the message to send
        @param pending_return return location for a #DBusPendingCall object, or #NULL if connection is disconnected
        @param timeout_milliseconds timeout in milliseconds or -1 for default
        @returns #FALSE if no memory, #TRUE otherwise.
    dbus_message_is_signal()
    --------------------------------------
        Checks whether the message is a signal with the given interface and member fields. If the message is not #DBUS_MESSAGE_TYPE_SIGNAL, or has a different interface or member field, returns #FALSE.
    dbus_message_iter_init()
    --------------------------------------
        Initializes a #DBusMessageIter for reading the arguments of the message passed in.
    dbus_message_iter_next()
    --------------------------------------
        Moves the iterator to the next field, if any. If there's no next field, returns #FALSE. If the iterator moves forward, returns #TRUE.
    dbus_message_iter_get_arg_type()
    --------------------------------------
        Returns the argument type of the argument that the message iterator points to. If the iterator is at the end of the message, returns #DBUS_TYPE_INVALID.
    dbus_message_iter_get_basic()
    --------------------------------------
        Reads a basic-typed value from the message iterator. Basic types are the non-containers such as integer and string.
    dbus_message_new_signal()
    --------------------------------------
        Constructs a new message representing a signal emission. Returns #NULL if memory can't be allocated for the message. A signal is identified by its originating object path, interface, and the name of the signal.
        Path, interface, and signal name must all be valid (the D-Bus specification defines the syntax of these fields).
        @param path the path to the object emitting the signal
        @param interface the interface the signal is emitted from
        @param name name of the signal
        @returns a new DBusMessage, free with dbus_message_unref()
    dbus_message_iter_init_append()
    --------------------------------------
        Initializes a #DBusMessageIter for appending arguments to the end of a message.
        @param message the message
        @param iter pointer to an iterator to initialize
    dbus_message_iter_append_basic()
    --------------------------------------
        Appends a basic-typed value to the message. The basic types are the non-container types such as integer and string.
        @param iter the append iterator
        @param type the type of the value
        @param value the address of the value
        @returns #FALSE if not enough memory
    dbus_message_new_method_call()
    --------------------------------------
        Constructs a new message to invoke a method on a remote object. Returns #NULL if memory can't be allocated for the message. The destination may be #NULL in which case no destination is set; this is appropriate when using D-Bus in a peer-to-peer context (no message bus). The interface may be #NULL, which means that if multiple methods with the given name exist it is undefined which one will be invoked.
        The path and method names may not be #NULL.
        Destination, path, interface, and method name can't contain any invalid characters (see the D-Bus specification).
        @param destination name that the message should be sent to or #NULL
        @param path object path the message should be sent to
        @param interface interface to invoke method on, or #NULL
        @param method method to invoke
        @returns a new DBusMessage, free with dbus_message_unref()
    dbus_bus_get()
    --------------------------------------
        Connects to a bus daemon and registers the client with it. If a connection to the bus already exists, then that connection is returned. The caller of this function owns a reference to the bus.
        @param type bus type
        @param error address where an error can be returned.
        @returns a #DBusConnection with new ref
    dbus_bus_request_name()
    --------------------------------------
        Asks the bus to assign the given name to this connection by invoking the RequestName method on the bus.
        First you should know that for each bus name, the bus stores a queue of connections that would like to own it. Only one owns it at a time - called the primary owner. If the primary owner releases the name or disconnects, then the next owner in the queue atomically takes over.
        So for example if you have an application org.freedesktop.TextEditor and multiple instances of it can be run, you can have all of them sitting in the queue. The first one to start up will receive messages sent to org.freedesktop.TextEditor, but if that one exits another will become the primary owner and receive messages.
        The queue means you don't need to manually watch for the current owner to disappear and then request the name again.
        @param connection the connection
        @param name the name to request
        @param flags flags
        @param error location to store the error
        @returns a result code, -1 if error is set
        给DBusConnection起名字(命名) -- 两个相互通信的连接(connection)不能同名
        命名规则: xxx.xxx (zeng.xiaolong)
    dbus_bus_add_match()
    --------------------------------------
        Adds a match rule to match messages going through the message bus. The "rule" argument is the string form of a match rule.
        @param connection connection to the message bus
        @param rule textual form of match rule
        @param error location to store any errors
    dbus_pending_call_block()
    --------------------------------------
        Block until the pending call is completed. The blocking is as with dbus_connection_send_with_reply_and_block(); it does not enter the main loop or process other messages, it simply waits for the reply in question.
        If the pending call is already completed, this function returns immediately.
        @todo when you start blocking, the timeout is reset, but it should really only use time remaining since the pending call was created. This requires storing timestamps instead of intervals in the timeout
        @param pending the pending call
    dbus_pending_call_steal_reply()
    --------------------------------------
        Gets the reply, or returns #NULL if none has been received yet. Ownership of the reply message passes to the caller. This function can only be called once per pending call, since the reply message is tranferred to the caller.
        @param pending the pending call
        @returns the reply message or #NULL.

    安装D-Bus可在其官方网站下载源码编译,地址为http://dbus.freedesktop.org。或者在终端上输入下列指令:

        yum install dbus dbus-devel dbus-doc

    安装后,头文件位于"/usr/include/dbus-<版本号>/dbus"目录中,编译使用D-Bus的程序时需加入编译指令"`pkg-config --cflags --libs dbus-1`"。
分享到:
评论

相关推荐

    dbus基础知识

    学习 dbus 基础知识,首先需要理解以下关键概念: 1. **消息总线**:了解如何连接到系统总线和会话总线,以及如何监听和发送消息。 2. **XML 配置**:DBus 使用 XML 文件来定义接口,理解 XML 文件结构对于创建...

    Dbus基础知识.pdf

    DBus基础知识详解 DBus是一种高效的进程间通信(IPC)机制,由freedesktop.org项目开发并遵循GPL许可证。它主要用于Linux桌面环境,允许不同进程之间进行通信,并能够将内核事件和桌面环境事件作为消息传递。DBus的...

    Dbus简介和例程

    《Dbus基础知识.doc》可能涵盖了以下内容: 1. **DBus的基本概念**:DBus作为一个中间件,提供了发布/订阅模型,允许应用程序发布信号或调用其他应用的方法。它基于UNIX套接字,支持TCP/IP,可以跨网络工作。 2. *...

    DBUS API说明文档

    以上知识点详细介绍了DBUS API中地址解析与认证机制的核心概念、宏定义、类型定义及函数功能,为开发者提供了全面的理论基础和实践指导,有助于在实际开发中更加熟练地应用DBUS API进行进程间通信的设计与实现。

    dbus调用系统输入法显隐

    1. **DBus基础**: - DBus是Linux内核的一项功能,提供进程间通信(IPC)机制,支持同步和异步通信。 - DBus包含两个主要部分:总线(Bus)和代理(Agent)。总线负责消息传递,代理则是提供服务的应用程序。 - ...

    dbus-introspect:Bourne Shell脚本反省DBus

    在深入探讨这个话题之前,我们首先需要了解DBus的基础知识。 DBus是一种中间件,它为进程间的通信(IPC)提供了一种简单而高效的方式。DBus服务可以发布和接收消息,这些消息包含了调用方法、响应、信号和其他类型...

    dbus-guide:ETS2修改DBus的指南,也称为DBus World

    首先,了解DBus的基础知识至关重要。DBus是一种轻量级的消息传递系统,允许不同的应用程序之间共享服务和数据。它基于一个中央服务器(称为总线),提供两种主要的通信类型:本地(session)总线和系统总线。通过...

    Ruby-一个Ruby的例子

    在这个"Ruby-一个Ruby的例子"中,我们将探讨Ruby的基础知识,以及如何通过具体的代码示例来理解其核心特性。 首先,Ruby的面向对象特性是其核心之一。在Ruby中,一切都是对象,包括基本数据类型如数字、字符串和...

    计算机应用基础统考模拟题单选全.pdf

    内容包括了计算机基础知识点的选择题、部分答案和一些计算机术语。由于文件内容庞大且部分扫描文字可能存在错误,下面将对文档中出现的计算机相关知识点进行详细解释和汇总。 **计算机基础知识** 1. **存储单位**...

    计算机基础培训教材.doc

    在当今信息化的时代背景下,掌握计算机基础知识成为了现代社会成员的基本能力之一。计算机作为一种多功能的电子设备,广泛应用于科学研究、工程技术、教育、医疗等多个领域,其作用不可或缺。为了帮助初学者快速入门...

    基于BLUEZ的低功耗蓝牙开发

    本文将深入探讨基于BLUEZ进行低功耗蓝牙开发的相关知识点。 1. **BLUEZ简介** - BLUEZ是由Haiku, Inc.的Jouni Malinen开发的开源项目,它是Linux平台上的官方蓝牙协议栈。 - 该项目提供了API接口,允许开发者通过...

    GNU C Library Reference

    这个库是Linux和其他类UNIX系统上进行程序开发的基础,为开发者提供了大量的函数和功能,使得编写高效、可靠的系统级和应用程序变得更加便捷。以下是glibc的一些关键知识点: 1. **基本数据类型**:glibc支持C99...

    计算机组成原理(20211010103013).pdf

    综合以上分析,文档内容涵盖了计算机组成原理的一些基础知识点,包括CPU的结构和功能、寄存器的作用、指令集以及总线系统的基本概念。这些知识点是计算机科学与技术专业的重要基础,对于理解计算机硬件的工作原理至...

    实战Linux_Bluetooth编程

    本文将详细介绍Linux下的蓝牙编程基础知识,包括蓝牙基本概念、蓝牙协议栈、Linux下的蓝牙开发等。 一、蓝牙基本概念 蓝牙是一种短距离、低成本的无线传输应用技术,于1998年由爱立信、诺基亚、东芝、IBM和Intel五...

    计算机组成原理实验报告总结寄存器的原理及操作.doc.pdf

    通过实际动手操作,不仅能加深理论知识的理解,还能提升问题解决能力,为后续的计算机系统设计和分析打下坚实基础。在互联网时代,这样的实践能力对于IT专业人士来说尤为重要,因为这有助于他们更好地适应快速发展的...

    实验二系列数据寄存器读写实验.docx

    通过这样的实践,学生能够更好地掌握计算机系统的基础知识,包括地址空间、数据传输和控制信号的作用。此外,这个实验也与互联网技术密切相关,因为互联网的底层数据处理正是基于这样的基本原理。

    Python-python自动化运维技术与最佳实践书中示例及案例

    1. **Python基础知识**:Python以其简洁易读的语法和强大的标准库而备受推崇,是自动化运维的理想选择。书中可能涵盖变量、数据类型、控制流(if语句、for循环、while循环)、函数、模块等基础知识。 2. **文件操作...

    计算机硬件基础实验报告.doc

    首先,报告概述了计算机硬件基础知识,并明确实验的目标是为了让学生能够熟悉计算机内部的核心组件,包括运算器、存储器、数据通路、微程序控制器及硬连线控制器的工作原理。通过实验操作,学生可以验证运算器的功能...

    计算机组成原理实验报告3-数据输出实验 移位门实验.docx

    《计算机组成原理实验报告——数据输出...总结来说,这次实验不仅提供了实践经验,更促进了理论知识的巩固。参与者在实际操作中学习到了寄存器、数据总线、移位门等核心概念,为后续的计算机系统学习打下了坚实的基础。

Global site tag (gtag.js) - Google Analytics