2

Glibを​​内部的に使用して他のモジュールに接続するlibに取り組んでいます。

このlibには、dbus接続をセットアップするinitializeメソッドと、すべての内部リソース(dbus接続を含む)をファイナライズするterminateメソッドがあります。少なくともこれを行う必要があります。

ただし、terminateメソッドが呼び出された後はdbus呼び出しを再度行うことはできません(もちろん、dbusメソッドを再度呼び出す前にinitializeメソッドを再度呼び出しました)。

私は主にC++を使用しています(いくつかの使用済みライブラリはCにあります)。

これが私の内部dbus初期化子です:

#ifdef __cplusplus
extern "C" {
#endif

#include <dbus/dbus-glib.h>
#include <stdio.h>

#define _DBUS_SERVICE       "Removed due to restrictions"
#define _DBUS_PATH      "/"
#define _DBUS_INTERFACE "Removed due to restrictions"

static DBusGConnection * my_dbus_conn = NULL;
static DBusGProxy * my_proxy = NULL;

gboolean dbus_init() {
    GError *error = NULL;
    g_type_init();
    my_dbus_conn = dbus_g_bus_get(DBUS_BUS_SESSION, &error);
    if (my_dbus_conn == NULL) {
        g_printerr("DBUS Connection Error (%s)\n", error->message);
        g_error_free(error);
        return false;
    }
    my_proxy = dbus_g_proxy_new_for_name(my_dbus_conn, _DBUS_SERVICE, _DBUS_PATH, _DBUS_INTERFACE);
    if (my_proxy == NULL) {
        g_printerr("ERROR: DBUS Proxy creation Error (%s)\n", error->message);
        g_error_free(error);
        return false;
    }
    return true;
}

DBusGProxy * dbus_get_proxy() {
    if (my_proxy == NULL) {
        dbus_init();
    }
    return my_proxy;
}

void dbus_term() {
    g_object_unref(G_OBJECT (my_proxy));
    dbus_g_connection_unref(my_dbus_conn);
    my_dbus_conn = NULL;
}

#ifdef __cplusplus
}
#endif

これが私の呼び出しコードです:

MY_G_DEBUG("Manager", "Creating Server IPC stub...");
this->m_pDbusServer = get_proxy();

if (this->m_pDbusServer == NULL) {
    return;
}

MY_G_DEBUG("Manager", "CheckPoint 1!");
gboolean success;
GError *gerror = NULL;
char **l_pHelloWorldReply;

success = dbus_server_helloworld(this->m_pDbusServer, "Hello World from Manager", &l_pHelloWorldReply, &gerror);

MY_G_DEBUG("Manager", "CheckPoint 2!");

if (success == FALSE) {
    MY_G_DEBUG("Manager", "CheckPoint 3!");
    if (gerror == NULL) {
        MY_G_DEBUG("Manager", "CRITICAL: Hello World call Failed (error not set)");
    } else {
        MY_G_DEBUG("Manager", "CRITICAL: Hello World call Failed (%s)", gerror->message);
        g_error_free(gerror);
        gerror = NULL;
    }
    return;
} else {
    MY_G_DEBUG("Manager", "CheckPoint 4!");
    for (guint ii = 0; l_pHelloWorldReply[ii] != NULL; ii++) {
        MY_G_DEBUG("Manager", "Hello World response %d: %s", ii, l_pHelloWorldReply[ii]);
    }
}

MY_G_DEBUG("Manager", "CheckPoint 5!");

ロギング出力は次のとおりです。

(process:24498): Manager-DEBUG: Checkpoint: 1!
(process:24498): Manager-DEBUG: Checkpoint: 2!
(process:24498): Manager-DEBUG: Checkpoint: 3!
(process:24498): Manager-DEBUG: CRITICAL: Hello World call Failed (error not set)

dbus_term関数を呼び出すたびにエラーメッセージが表示されましたが、修正できませんでした。

(process:24498): GLib-GObject-WARNING **: invalid uninstantiatable type `<invalid>' in cast to `GObject'
(process:24498): GLib-GObject-CRITICAL **: g_object_unref: assertion `G_IS_OBJECT (object)' failed

初めてメソッドを呼び出そうとすると、すべてがうまく機能します...これはdbus_termの問題が原因であると強く信じています。

なぜそれが起こっているのか誰かが知っていますか?

どうもありがとう。

4

2 に答える 2

1

まだ開いているdbus接続への最後の参照を削除するのはバグです。使用しているバインディングの問題の1つは、接続が実際にいつ閉じられるかを知ることが難しいことです。やや不満ですが、セッションバスのような有名なバスへの接続については、1つの参照を残すのが私の習慣です。

于 2011-03-17T07:50:44.527 に答える
-2

新しいコードではdbus-glibを使用せず、代わりにGDBusを使用してください。

GDBusは、glib2.26の新しいdbusAPIです。dbus-glibに対するその利点には、GIO非同期パターン(GAsync *、GCancellable)の使用、バリアントタイプへのGVariantの使用、およびより優れたスレッドサポートが含まれます。

GDBusへの移行については、 http: //library.gnome.org/devel/gio/2.28/ch29.htmlを参照 してください。

于 2011-03-17T08:40:16.337 に答える