关于DBus (signal)的使用

**

DBUS

**
DBUS有四种类型的消息:
1.方法调用(method call) 在对象上执行一个方法
2.方法返回(method return)返回方法执行的结果
3.错误(error)调用方法产生的异常
4.信号(signal)通知指定的信号发生了,可以想象成“事件”。

客户端:

发送信号的流程:

1.建立与DBus后台的连接

 	DBusError err;
    DBusConnection *connection;
    DBusMessage *msg;
    DBusMessageIter arg;
    dbus_uint32_t serial = 0;
    int ret;

    dbus_error_init(&err);
    //步骤1:建立与D-Bus后台的连接
    connection = dbus_bus_get(DBUS_BUS_SESSION, &err);
    if (dbus_error_is_set(&err))
    {
        fprintf(stderr, "ConnectionErr : %s\n", err.message);
        dbus_error_free(&err);
    }
    if (connection == NULL)
        return -1;

**

2.给连接名分配一个well-knmown的名字作为Bus name

**

        //步骤2:给连接名分配一个well-known的名字作为Bus name,这个步骤不是必须的,可以用if 0来注释着一段代码,我们可以用这个名字来检查,是否已经开启了这个应用的另外的进程。
#if 1
    ret = dbus_bus_request_name(connection, "test.singal.source", DBUS_NAME_FLAG_REPLACE_EXISTING, &err);
    if (dbus_error_is_set(&err))
    {
        fprintf(stderr, "Name Err :%s\n", err.message);
        dbus_error_free(&err);
    }
    if (ret != DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER)
        return -1;
#endif

3.发送一个信号(给出信号的路径,接口,以及信号名,创建一个Message)

  	dbus_message_iter_init_append(msg, &arg);
    if (!dbus_message_iter_append_basic(&arg, DBUS_TYPE_STRING, &sigvalue))
    {
        fprintf(stderr, "Out OfMemory!\n");
        return -1;
    }
    printf("Signal Send: %s\n", sigvalue);

4.将信号从连接中发送

if (!dbus_connection_send(connection, msg, &serial))
    {
        fprintf(stderr, "Out of Memory!\n");
        return -1;
    }
    dbus_connection_flush(connection);
    printf("--------Success Signal Send----------\n");

5.释放相关的内存分配。

 dbus_message_unref(msg);

6.主函数实现

// 客户端代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
//#include <dbus/dbus-glib.h>
#include <dbus/dbus.h>
#include <unistd.h>

int send_a_signal(char *sigvalue)
{
    DBusError err;
    DBusConnection *connection;
    DBusMessage *msg;
    DBusMessageIter arg;
    dbus_uint32_t serial = 0;
    int ret;

    dbus_error_init(&err);

    //步骤1:建立与D-Bus后台的连接
    //
    connection = dbus_bus_get(DBUS_BUS_SESSION, &err);
    if (dbus_error_is_set(&err))
    {
        fprintf(stderr, "ConnectionErr : %s\n", err.message);
        dbus_error_free(&err);
    }
    if (connection == NULL)
        return -1;

        //步骤2:给连接名分配一个well-known的名字作为Bus name,这个步骤不是必须的,可以用if 0来注释着一段代码,我们可以用这个名字来检查,是否已经开启了这个应用的另外的进程。
#if 1
    ret = dbus_bus_request_name(connection, "test.singal.source", DBUS_NAME_FLAG_REPLACE_EXISTING, &err);
    if (dbus_error_is_set(&err))
    {
        fprintf(stderr, "Name Err :%s\n", err.message);
        dbus_error_free(&err);
    }
    if (ret != DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER)
        return -1;
#endif

    //步骤3:发送一个信号
    //根据图,我们给出这个信号的路径(即可以指向对象),接口,以及信号名,创建一个Message
    if ((msg = dbus_message_new_signal("/test/signal/Object", "test.signal.Type", "Test")) == NULL)
    {
        fprintf(stderr, "MessageNULL\n");
        return -1;
    }
    //给这个信号(messge)具体的内容
    dbus_message_iter_init_append(msg, &arg);
    if (!dbus_message_iter_append_basic(&arg, DBUS_TYPE_STRING, &sigvalue))
    {
        fprintf(stderr, "Out OfMemory!\n");
        return -1;
    }

    printf("Signal Send: %s\n", sigvalue);
    //步骤4: 将信号从连接中发送
    if (!dbus_connection_send(connection, msg, &serial))
    {
        fprintf(stderr, "Out of Memory!\n");
        return -1;
    }
    dbus_connection_flush(connection);
    printf("--------Success Signal Send----------\n");

    //步骤5: 释放相关的分配的内存。
    dbus_message_unref(msg);
    return 0;
}
int main(void)
{
    printf("------Start Send_a_signal!-------\n");
    send_a_signal("hello ");
    return 0;
}

服务端:

建立一个dbus连接之后,为这个dbus连接起名,为我们将要进行的消息循环添加匹配条件(就是通过信号名和信号接口名来进行匹配控制的) – dbus_bus_add_match()。我们进入等待循环后,只需要对信号名,信号接口名进行判断就可以分别处理各种信号了。在各个处理分支上。我们 可以分离出消息中的参数。对参数类型进行判断和其他的处理。

1.获取连接


    connection = dbus_bus_get(DBUS_BUS_SESSION, &err);
    if(dbus_error_is_set(&err))
    {
        fprintf(stderr,"Connection error %s\n",err.message);
    }

    if(NULL == connection)
    {
        return;
    }

2.设置感兴趣消息

dbus_bus_add_match(connection,"type='signal',interface='test.signal.type'",&err);
    dbus_connection_flush(connection);
    if(dbus_error_is_set(&err))
    {
        fprintf(stderr,"Match Error%s\n",err.message);
        dbus_error_free(&err);
    }

3.等待消息:

 while(1)
    {
        dbus_connection_read_write(connection,0);
        msg = dbus_connection_pop_message(connection);
        if(NULL == msg)
        {
                sleep(1);
                continue;
        }
        if(dbus_message_is_signal(msg,"test.signal.type","Test"))
        {
                if(!dbus_message_iter_init(msg,&arg))
                {
                        fprintf(stderr,"MessageHas no pragrom!\n");
                }else if(dbus_message_iter_get_arg_type(&arg) != DBUS_TYPE_STRING)
                {
                        fprintf(stderr,"Param isnot string!\n");
                }else
                {
                    dbus_message_iter_get_arg_basic(&arg,&sigvalue);
                    printf("Recetive Signal Value:%s\n",sigvalue);
                }

        }
        dbus_message_unref(msg);
    }
}

4.服务段代码:

// 服务端代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <dbus/dbus.h>
#include <unistd.h>

void listen_signal()
{
    DBusMessage * msg;
    DBusMessageIter arg;
    DBusConnection * connection;
    DBusError err;
    int ret;
    char * sigvalue;

     //步骤1:建立与D-Bus后台的连接
    dbus_error_init(&err);
    connection =dbus_bus_get(DBUS_BUS_SESSION, &err);
    if(dbus_error_is_set(&err)){
        fprintf(stderr,"ConnectionError %s\n",err.message);
        dbus_error_free(&err);
    }
    if(connection == NULL)
        return;

   //步骤2:给连接名分配一个可记忆名字test.singal.dest作为Bus name,这个步骤不是必须的,但推荐这样处理
    ret =dbus_bus_request_name(connection,"test.singal.dest",DBUS_NAME_FLAG_REPLACE_EXISTING,&err);
    if(dbus_error_is_set(&err)){
        fprintf(stderr,"Name Error%s\n",err.message);
        dbus_error_free(&err);
    }
    if(ret !=DBUS_REQUEST_NAME_REPLY_PRIMARY_OWNER)
        return;

    //步骤3:通知D-Bus daemon,希望监听来行接口test.signal.Type的信号
    dbus_bus_add_match(connection,"type='signal',interface='test.signal.Type'",&err);
    //实际需要发送东西给daemon来通知希望监听的内容,所以需要flush
    dbus_connection_flush(connection);
    if(dbus_error_is_set(&err)){
        fprintf(stderr,"Match Error%s\n",err.message);
        dbus_error_free(&err);
    }

    //步骤4:在循环中监听,每隔开1秒,就去试图自己的连接中获取这个信号。这里给出的是中连接中获取任何消息的方式,所以获取后去检查一下这个消息是否我们期望的信号,并获取内容。我们也可以通过这个方式来获取method call消息。
    while(1){
        dbus_connection_read_write(connection,0);
        msg =dbus_connection_pop_message (connection);
        if(msg == NULL){
            sleep(1);
            continue;
        }

        if(dbus_message_is_signal(msg,"test.signal.Type","Test")){
            if(!dbus_message_iter_init(msg,&arg))
                fprintf(stderr,"MessageHas no Param");
            else if(dbus_message_iter_get_arg_type(&arg)!= DBUS_TYPE_STRING)
                fprintf(stderr,"Param isnot string");
            else
                dbus_message_iter_get_basic(&arg,&sigvalue);
            printf("Recetive Singal Value : %s\n",sigvalue);
        }
        dbus_message_unref(msg);
    }//End of while

}
int main(void){

    printf("------Start Listen_signal!-------\n");
    listen_signal();
    return 0;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值