[lnkForumImage]
TotalShareware - Download Free Software

Confronta i prezzi di migliaia di prodotti.
Asp Forum
 Home | Login | Register | Search 


 

Forums >

comp.lang.ruby

Pulseaudio c extension callback function

Edwin van Leeuwen

11/5/2007 5:25:00 PM

Hi all,

I started working on a ruby wrapper for pulseaudio. I am a bit out of my
depth here, because my c experience is very limited and very rusty and I
am kinda stuck. The library is basically event driven. You attach some
callback function, start the mainloop and wait for events. In c you
would basically the following:

static void
callback( pa_context *c, void *userdata )
{
}

pa_mainloop *m = pa_mainloop_new();
pa_mainloop_api *api = pa_mainloop_get_api( m );
pa_context *context = pa_context_new(api, "name");
/* Here the callback function is attached */
pa_context_set_state_callback( context, callback, userdata );
pa_context_connect( connect, NULL, 0, NULL);

The interesting part is the callback function, which is defined as
follows:

static void
callback( pa_context *c, void *userdata )
{
}

I now want to define my own callback function and have it change the
status of my context object. Then I can define listeners in ruby on the
status method and will be home free. The problem is to somehow pass the
current context object to the c based callback function. My context
class is just a wrapper around pa_context, so ideally I could use
pa_context *c to "find" my object again, but if I use Data_Wrap_Struct
it will create a new context object won't it? The closest I came was:

static void
callback( pa_context *c, void *klass )
{
VALUE obj;
obj = * (VALUE *) klass;
rb_funcall( obj, rb_intern( "status=" ), 1, INT2NUM(2) );
}

static VALUE
cContext_connect( VALUE klass )
{
pa_context_set_state_callback( DATA_PTR( klass ), callback, &klass );
pa_context_connect(DATA_PTR( klass ), NULL, 0, NULL);
return klass;
}

but then I got the following error:
NoMethodError: undefined method `status=' for
PulseAudio::Mainloop:0xb79ac9ac>
(I have no clue where this mainloop object comes from, except that it is
ofcourse the mainloop that will call the callback function)

Also I have the feeling this could introduce problems with the GC
cleaning up my context object, because officialy it's not referenced
anymore (but I'm reasonably clueless about memory management and
pointers etc).

Does anyone have any pointers for me :) ?

Cheers,

Edwin
--
Posted via http://www.ruby-....

3 Answers

Paul Brannan

11/5/2007 7:12:00 PM

0

On Tue, Nov 06, 2007 at 02:25:29AM +0900, Edwin Van leeuwen wrote:
> static void
> callback( pa_context *c, void *klass )
> {
> VALUE obj;
> obj = * (VALUE *) klass;
> rb_funcall( obj, rb_intern( "status=" ), 1, INT2NUM(2) );
> }
>
> static VALUE
> cContext_connect( VALUE klass )
> {
> pa_context_set_state_callback( DATA_PTR( klass ), callback, &klass );

klass is a temporary variable; you probably don't want to take its
address. Instead you can cast klass to a pointer:

pa_context_set_state_callback( DATA_PTR( klass ), callback, (void *)klass );

then in your callback function, write:

VALUE obj = (VALUE *) klass;

> pa_context_connect(DATA_PTR( klass ), NULL, 0, NULL);
> return klass;
> }
>
> but then I got the following error:
> NoMethodError: undefined method `status=' for
> PulseAudio::Mainloop:0xb79ac9ac>
> (I have no clue where this mainloop object comes from, except that it is
> ofcourse the mainloop that will call the callback function)
>
> Also I have the feeling this could introduce problems with the GC
> cleaning up my context object, because officialy it's not referenced
> anymore (but I'm reasonably clueless about memory management and
> pointers etc).

If I understand your question, when the context object is destroyed, you
need to unregister the callback. That will ensure that the library
doesn't keep a reference to the freed object. You can do this by
passing a `free' function to Data_Wrap_Struct.

> Does anyone have any pointers for me :) ?

Other than the pointers to object you've already created? :)

Paul


Edwin van Leeuwen

11/5/2007 8:08:00 PM

0

Paul Brannan wrote:
> klass is a temporary variable; you probably don't want to take its
> address. Instead you can cast klass to a pointer:
>
> pa_context_set_state_callback( DATA_PTR( klass ), callback, (void
> *)klass );
>
> then in your callback function, write:
>
> VALUE obj = (VALUE *) klass;
>

Thanks, it worked when I changed it to:

VALUE obj = (VALUE) klass;

Otherwise I got a compile error. I really should refresh my c, because
clearly my approach of changing the code until it stops segvaulting is
not the best :) Although it seems to give me decent progress :)

> If I understand your question, when the context object is destroyed, you
> need to unregister the callback. That will ensure that the library
> doesn't keep a reference to the freed object. You can do this by
> passing a `free' function to Data_Wrap_Struct.

Yes, I am already using the unref function for the context itself. I am
not sure if that also unregisters the callback. I guess I should be
testing that later.

Edwin
--
Posted via http://www.ruby-....

Paul Brannan

11/5/2007 8:26:00 PM

0

On Tue, Nov 06, 2007 at 05:08:17AM +0900, Edwin Van leeuwen wrote:
> Thanks, it worked when I changed it to:
>
> VALUE obj = (VALUE) klass;

Oops, typo. :)

> Yes, I am already using the unref function for the context itself. I am
> not sure if that also unregisters the callback. I guess I should be
> testing that later.

That's the behavior I would expect, but I've never used pulseaudio.

Paul