After an evaluation, GNOME has moved from Bugzilla to GitLab. Learn more about GitLab.
No new issues can be reported in GNOME Bugzilla anymore.
To report an issue in a GNOME project, go to GNOME GitLab.
Do not go to GNOME Gitlab for: Bluefish, Doxygen, GnuCash, GStreamer, java-gnome, LDTP, NetworkManager, Tomboy.
Bug 704176 - Accessing "this" from a closure causes assertion self != NULL failure in some situations
Accessing "this" from a closure causes assertion self != NULL failure in some...
Status: RESOLVED FIXED
Product: vala
Classification: Core
Component: Bindings
unspecified
Other Linux
: Normal normal
: ---
Assigned To: Vala maintainers
Vala maintainers
Depends on:
Blocks:
 
 
Reported: 2013-07-13 23:03 UTC by Simon Kågedal
Modified: 2013-08-26 19:43 UTC
See Also:
GNOME target: ---
GNOME version: ---


Attachments
Code that triggers error. (1.08 KB, text/x-vala)
2013-07-13 23:03 UTC, Simon Kågedal
  Details
gir parser: support scope=async parameters. (5.78 KB, patch)
2013-07-14 18:36 UTC, Evan Nemerson
needs-work Details | Review
gir parser: support scope=async syntax (14.79 KB, patch)
2013-07-26 12:33 UTC, Timm Bäder
none Details | Review
girparser: Support scope=async parameters. (5.39 KB, patch)
2013-08-24 11:48 UTC, Luca Bruno
none Details | Review

Description Simon Kågedal 2013-07-13 23:03:24 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
Comment 1 Luca Bruno 2013-07-14 08:06:41 UTC
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.
Comment 2 Evan Nemerson 2013-07-14 18:36:14 UTC
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.
Comment 3 Simon Kågedal 2013-07-14 19:55:24 UTC
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.
Comment 4 Luca Bruno 2013-07-15 08:13:33 UTC
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.
Comment 5 Luca Bruno 2013-07-15 08:13:34 UTC
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.
Comment 6 Luca Bruno 2013-07-15 08:21:44 UTC
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.
Comment 7 Evan Nemerson 2013-07-15 08:49:11 UTC
(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.
Comment 8 Luca Bruno 2013-07-15 08:58:53 UTC
> 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.
Comment 9 Timm Bäder 2013-07-26 12:33:20 UTC
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).
Comment 10 Simon Kågedal 2013-07-31 17:08:43 UTC
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
Comment 11 Evan Nemerson 2013-08-01 04:23:38 UTC
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.
Comment 12 Simon Kågedal 2013-08-01 06:43:57 UTC
I'm using Vala 0.20.1 from the PPA.  What/where is d74c5b06? Can't find any such git revision of vala.
Comment 13 Evan Nemerson 2013-08-01 06:56:29 UTC
that is too old.  sorry, typo.  should be d74c5fb0


https://git.gnome.org/browse/vala/commit/?id=d74c5fb0654ef5985d763a7847c8013de781aac3
Comment 14 Simon Kågedal 2013-08-01 08:11:22 UTC
Allright! With git master vala and libsoup-2.4.vapi from your patch, the test case now compiles and runs fine.
Comment 15 Timm Bäder 2013-08-01 13:28:13 UTC
(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'?
Comment 16 Luca Bruno 2013-08-24 11:48:22 UTC
Created attachment 253008 [details] [review]
girparser: Support scope=async parameters.

Based on patch by Evan.

Fixes bug 704176.
Comment 17 Luca Bruno 2013-08-26 19:43:33 UTC
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.