GNOME Bugzilla – Bug 704176
Accessing "this" from a closure causes assertion self != NULL failure in some situations
Last modified: 2013-08-26 19:43:33 UTC
Created attachment 249078 [details] Code that triggers error. A closure sent to Soup.Session.queue_message() that contains access to "this" seems to sometimes not set this correctly as I get assertion errors: ** (process:6878): CRITICAL **: soup_test_handle_response1: assertion `self != NULL' failed I've tried to narrow it down as far as I could, but I can't reproduce it with a simpler test case that doesn't use Soup. It happens with code like this: session.queue_message (msg1, (s, m) => { stdout.printf (@"Got response: $(m.status_code)\n"); this.handle_response1 (some_object, (string) m.response_body.data); }); But this works: session.queue_message (msg2, (s, m) => { stdout.printf (@"2: Got response: $(m.status_code)\n"); this.handle_response2 ((string) m.response_body.data); }); I.e., it's triggered if the method called this takes an object and a string, not just a string. Attaching a file "test-update.vala", compiling and running it looks like this for me: simon@struts:~/projects/fido/server$ valac-0.20 test-update.vala --pkg libsoup-2.4 --target-glib=2.36 /home/simon/projects/fido/server/test-update.vala.c: In function ‘__lambda2_’: /home/simon/projects/fido/server/test-update.vala.c:308:10: warning: assignment discards ‘const’ qualifier from pointer target type [enabled by default] /home/simon/projects/fido/server/test-update.vala.c: In function ‘__lambda3_’: /home/simon/projects/fido/server/test-update.vala.c:347:10: warning: assignment discards ‘const’ qualifier from pointer target type [enabled by default] simon@struts:~/projects/fido/server$ ./test-update 1: Got response: 200 ** (process:6878): CRITICAL **: soup_test_handle_response1: assertion `self != NULL' failed 2: Got response: 200 2: Parsing string beginning with <!doctype html><html ^C I run Vala 0.20 from the Ubuntu PPA on a Ubuntu 13.04 system. Regards, Simon Kågedal Reimer
At the moment, the callback passed to queue_message is unowned and is called async. That means you shouldn't use it as a closure: you shouldn't use anything outside of the callback scope. This will be fixed soon by fixing the bindings.
Created attachment 249132 [details] [review] gir parser: support scope=async parameters. I wrote a patch for this issue recently. I just need to do some testing and rebuild all the gir bindings.
Thanks for the help! That will help me get my work done for now, glad to hear there's work on a fix. Might be able to help test the patch later.
Review of attachment 249132 [details] [review]: Thanks for the patch. ::: vala/valagirparser.vala @@ +3280,3 @@ List<ParameterInfo> parameters = node.parameters; + foreach (ParameterInfo info in parameters) { There's a similar loop in this method, can we move this near to the other one? @@ +3294,3 @@ + if (info.param.variable_type is DelegateType) { + var d = ((DelegateType) info.param.variable_type).delegate_symbol; + dt.value_owned = info.param.variable_type.value_owned; Why is DestroyNotify being marked is_async? @@ +3296,3 @@ + if (!(d.name == "DestroyNotify" && d.parent_symbol.name == "GLib")) { + info.param.set_attribute_string ("CCode", "scope", "async"); + dt.nullable = info.param.variable_type.nullable; This shouldn't be necessary as it's done by semantic analyzer later on.
Eek the patch review tool is very confusing, it didn't highlight the line, rather removed it :-(. The second comment is about you checking GLib.DestroyNotify, but I don't see why it's marked is_async. The last comment is about the "is_called_once = true" thing, that should be done automatically.
(In reply to comment #5) > There's a similar loop in this method, can we move this near to the other one? Not sure, but it's worth a try. I seem to remember starting out there and having to move to where it is now for some reason, but I don't remember what that reason was. > Why is DestroyNotify being marked is_async? is_async in the ParameterInfo basically just means the parameter was annotated with scope="async" in the GIR. Why DestroyNotify arguments are annotated as async in the GIR is irrelevant (although it does make sense). They are, so we have to deal with it. It would be awesome to just have vala notice that scope=async doesn't make sense for arguments with a delegate type without a target, drop the annotation and set is_called_once to false. That would eliminate the need to special-case DestroyNotify, and correct the output for other arguments which match that criteria and are marked as async (if there are any). That would need to happen somewhere else, though. > This shouldn't be necessary as it's done by semantic analyzer later on. Hm, I know I tried setting is_called_once and the VAPI was still generated without the annotation. Not sure if I tried going the other way.
> is_async in the ParameterInfo basically just means the parameter was annotated > with scope="async" in the GIR. Why DestroyNotify arguments are annotated as > async in the GIR is irrelevant (although it does make sense). They are, so we > have to deal with it. > > It would be awesome to just have vala notice that scope=async doesn't make > sense for arguments with a delegate type without a target, drop the annotation > and set is_called_once to false. That would eliminate the need to special-case > DestroyNotify, and correct the output for other arguments which match that > criteria and are marked as async (if there are any). That would need to happen > somewhere else, though. That would be an error from vala to the user in the analyzer, it's the girparser that must do it. Ok, given the gir annotates DestroyNotify with scope="async" your code makes sense.
Created attachment 250195 [details] [review] gir parser: support scope=async syntax The second loop was only being executed for void functions so in order to make that work I had to restructure the code a bit. The 'info.param.variable_type.value_owned = true' from the previous patch is commented out here because I wasn't entirely sure that the change is correct(i.e. previously all the now scope=async annotated callbacks were 'owned', now they aren't).
Hi, I tried now to compile the test code attached above with the libsoup-2.4.vapi from Evan's and Timm's patches. With Evan's, the generated code becomes: soup_session_queue_message ((SoupSession*) _tmp2_, _tmp3_, ___lambda2__soup_session_callback, block1_data_ref (_data1_), block1_data_unref); Which fails when gcc'ing since there's too many arguments. With Timm's patch things work as expected! Regards, Simon Kågedal Reimer
AFAIK this shouldn't be an issue with d74c5b06 applied. Perhaps you're using an older valac? It should be owned. scope=async + unowned doesn't make sense.
I'm using Vala 0.20.1 from the PPA. What/where is d74c5b06? Can't find any such git revision of vala.
that is too old. sorry, typo. should be d74c5fb0 https://git.gnome.org/browse/vala/commit/?id=d74c5fb0654ef5985d763a7847c8013de781aac3
Allright! With git master vala and libsoup-2.4.vapi from your patch, the test case now compiles and runs fine.
(In reply to comment #11) > It should be owned. scope=async + unowned doesn't make sense. So uncomment the 'info.param.variable_type.value_owned = true'?
Created attachment 253008 [details] [review] girparser: Support scope=async parameters. Based on patch by Evan. Fixes bug 704176.
commit 3e20fd82c05daa60005aaff11411cd1814beb778 Author: Luca Bruno <lucabru@src.gnome.org> Date: Mon Jul 8 23:03:58 2013 -0700 girparser: Support scope=async parameters. Based on patch by Evan. Fixes bug 704176. This problem has been fixed in the development version. The fix will be available in the next major software release. Thank you for your bug report.