From: Rico Tzschichholz Date: Sun, 21 Oct 2018 08:34:28 +0000 (+0200) Subject: vala: Async methods don't allow out-parameters before in-parameters X-Git-Tag: 0.43.5~35 X-Git-Url: http://git.ipfire.org/cgi-bin/gitweb.cgi?a=commitdiff_plain;h=dd304f4d8ace88491a7c93e6939bc577fd2f7eaf;p=thirdparty%2Fvala.git vala: Async methods don't allow out-parameters before in-parameters out-parameters are always handled in the *_finish implementation and therefore are asynchronous. Report an error for occurances in source and a warning for GIR sources where convert them to pointer-types to create a usuable signature for bindings. Fixes https://gitlab.gnome.org/GNOME/vala/issues/636 --- diff --git a/tests/Makefile.am b/tests/Makefile.am index b3d16c732..87afd9070 100644 --- a/tests/Makefile.am +++ b/tests/Makefile.am @@ -410,6 +410,7 @@ TESTS = \ asynchronous/bug793158.vala \ asynchronous/closures.vala \ asynchronous/generator.vala \ + asynchronous/out-parameter-invalid.test \ asynchronous/result-pos.vala \ asynchronous/yield.vala \ generics/bug640330.test \ @@ -443,6 +444,7 @@ TESTS = \ gir/bug792998.test \ gir/array-fixed-length.test \ gir/async-result-pos.test \ + gir/async-sync-out.test \ gir/class.test \ gir/delegate-alias-without-target.test \ gir/delegate-closure-destroy-index-conflict.test \ diff --git a/tests/asynchronous/out-parameter-invalid.test b/tests/asynchronous/out-parameter-invalid.test new file mode 100644 index 000000000..070218315 --- /dev/null +++ b/tests/asynchronous/out-parameter-invalid.test @@ -0,0 +1,7 @@ +Invalid Code + +async void foo (out int i, int j) { +} + +void main () { +} diff --git a/tests/gir/async-sync-out.test b/tests/gir/async-sync-out.test new file mode 100644 index 000000000..48956f61d --- /dev/null +++ b/tests/gir/async-sync-out.test @@ -0,0 +1,58 @@ +GIR + +Input: + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + +Output: + +[CCode (cheader_filename = "test.h", type_id = "test_foo_get_type ()")] +public class Foo : GLib.Object { + [CCode (has_construct_function = false)] + public Foo (); + public async void method_async (string input, string* sync_output, out string output) throws GLib.Error; +} diff --git a/vala/valagirparser.vala b/vala/valagirparser.vala index 061eb17c5..e522d3029 100644 --- a/vala/valagirparser.vala +++ b/vala/valagirparser.vala @@ -4073,6 +4073,20 @@ public class Vala.GirParser : CodeVisitor { void process_async_method (Node node) { var m = (Method) node.symbol; + + // TODO: async methods with out-parameters before in-parameters are not supported + bool requires_pointer = false; + foreach (var param in m.get_parameters ()) { + if (param.direction == ParameterDirection.IN) { + requires_pointer = true; + } else if (requires_pointer) { + param.direction = ParameterDirection.IN; + param.variable_type.nullable = false; + param.variable_type = new PointerType (param.variable_type); + Report.warning (param.source_reference, "Synchronous out-parameters are not supported in async methods"); + } + } + string finish_method_base; if (m.name == null) { assert (m is CreationMethod); diff --git a/vala/valamethod.vala b/vala/valamethod.vala index bd203ab36..22980085f 100644 --- a/vala/valamethod.vala +++ b/vala/valamethod.vala @@ -815,6 +815,20 @@ public class Vala.Method : Subroutine, Callable { } } + if (coroutine) { + // TODO: async methods with out-parameters before in-parameters are not supported + bool requires_pointer = false; + for (int i = parameters.size - 1; i >= 0; i--) { + var param = parameters[i]; + if (param.direction == ParameterDirection.IN) { + requires_pointer = true; + } else if (requires_pointer) { + error = true; + Report.error (param.source_reference, "Synchronous out-parameters are not supported in async methods"); + } + } + } + if (error_types != null) { foreach (DataType error_type in error_types) { error_type.check (context);