Suppose I have a method like this
<interface name="org.Test.ChildTest">
<!-- set_age(guint32 new_age): sets new age -->
<method name="set_age">
<arg type="u" name="new_age" direction="in"/>
</method>
In my table of methods I have:
{ (GCallback) child_test_set_age, dbus_glib_marshal_child_test_BOOLEAN__UINT_POINTER, 0 }
and the right GObject method signature is:
gboolean
child_test_set_age (ChildTest *childTest, guint ageIn, GError** error)
Why does my method, child_test_set_age()
, still get called on DBus even if the callback arguments don't match the one specified in my XML? For example if I add another argument after guint ageIn
, like a char*
or guint
or some other random type?
I've noticed that this won't work if the DBus function includes members with direction OUT. It seems that any argument of type IN that's unnecessary gets discarded and the call is done as usual.
Although I believe this does not make any difference, I'm using D-BUS Binding Tool 0.94, glib-2.30.0 and dbus-glib 0.94.
You've hit on an interesting detail that exists because of the C language. Functions in C are strongly typed, and so strictly speaking you'd have to have functions to deal with every single possible type of callback, something like the following nightmare:
Luckily two features of the C language make it possible to avoid that mess.
Since function pointers are all the same size, it's OK to cast them all to
void (*callback)(void)
, which is whatGCallback
is a typedef for.GCallback
is used in all GLib-platform API for callbacks that can have variable numbers and types of arguments. That's why you have to castchild_test_set_age
toGCallback
in your code sample above.But even if you can pass function pointers around as if they're all the same, how do you make sure the functions actually get their arguments? That's what the C calling convention is for. The compiler generates code such that the caller pushes the function's arguments on to the stack, the function reads the arguments from the stack but doesn't pop them, and when it returns, the caller pops the arguments back off the stack. So the caller can push a different number of arguments than the function expects, as long as the function can find all the arguments it tries to access!
Let's illustrate that with your example: calling the method
child_test_set_age(ChildTest *childTest, guint ageIn, GError **error)
. Let's assume pointers and integers are the same size on your platform, and I'll skip over some details in order to get the general idea across.The caller puts the arguments onto the stack:
...and calls the function. The function gets that stack and looks for its arguments there:
Everything is fine. Then the function returns and the caller pops the arguments off the stack.
Now, however, if you give your function a different type than the DBus signature in the XML, let's say
child_test_set_age(ChildTest *childTest, guint ageIn, guint otherNumberIn, GError **error)
, let's say the same arguments get pushed onto the stack, but your function interprets them differently:The first two parameters are fine. The third one, since DBus doesn't know you're expecting another
guint
, is going to be theGError **
cast to aguint
. If you're lucky enough that that pointer wasNULL
, thenotherNumberIn
will be equal to 0, but otherwise it'll be a memory location cast to an integer: garbage.The fourth parameter is especially dangerous, because it will try to read something off the stack, and you have no idea what's on there. So if you try to access the
error
pointer, you'll probably crash your program with a segfault.However, if you manage to get through the function without a segfault, then the caller will neatly pop three arguments off the stack after the function returns and everything will be back to normal. So that's why your program still seems to work.
The "out" parameters are implemented using pointer arguments, and so that's why they don't work; the function is almost guaranteed to write to an invalid memory address because the pointers are garbage.
In summary, your function can have a different signature if the following conditions hold: