[PATCH wayland 2/7] Add a default dispatcher function and related test

Jason Ekstrand jason at jlekstrand.net
Mon Feb 18 08:03:23 PST 2013


Signed-off-by: Jason Ekstrand <jason at jlekstrand.net>
---
 src/connection.c        | 82 ++++++++++++++++++++++++++++++++++++++++++
 src/wayland-private.h   |  5 +++
 src/wayland-util.h      | 12 +++++++
 tests/.gitignore        |  1 +
 tests/Makefile.am       |  2 ++
 tests/dispatcher-test.c | 95 +++++++++++++++++++++++++++++++++++++++++++++++++
 6 files changed, 197 insertions(+)
 create mode 100644 tests/dispatcher-test.c

diff --git a/src/connection.c b/src/connection.c
index 141875e..7be6295 100644
--- a/src/connection.c
+++ b/src/connection.c
@@ -1,5 +1,6 @@
 /*
  * Copyright © 2008 Kristian Høgsberg
+ * Copyright © 2013 Jason Ekstrand
  *
  * Permission to use, copy, modify, distribute, and sell this software and its
  * documentation for any purpose is hereby granted without fee, provided that
@@ -1039,3 +1040,84 @@ wl_closure_destroy(struct wl_closure *closure)
 {
 	free(closure);
 }
+
+static void
+convert_arguments_to_ffi(const char *signature, union wl_argument *args,
+			 int count, ffi_type **ffi_types, void** ffi_args)
+{
+	int i;
+	const char *sig_iter;
+	struct argument_details arg;
+
+	sig_iter = signature;
+	for (i = 0; i < count; i++) {
+		sig_iter = get_next_argument(sig_iter, &arg);
+
+		switch(arg.type) {
+		case 'i':
+			ffi_types[i] = &ffi_type_sint32;
+			ffi_args[i] = &args[i].i;
+			break;
+		case 'u':
+			ffi_types[i] = &ffi_type_uint32;
+			ffi_args[i] = &args[i].u;
+			break;
+		case 'f':
+			ffi_types[i] = &ffi_type_sint32;
+			ffi_args[i] = &args[i].f;
+			break;
+		case 's':
+			ffi_types[i] = &ffi_type_pointer;
+			ffi_args[i] = &args[i].s;
+			break;
+		case 'o':
+			ffi_types[i] = &ffi_type_pointer;
+			ffi_args[i] = &args[i].o;
+			break;
+		case 'n':
+			ffi_types[i] = &ffi_type_uint32;
+			ffi_args[i] = &args[i].n;
+			break;
+		case 'a':
+			ffi_types[i] = &ffi_type_pointer;
+			ffi_args[i] = &args[i].a;
+			break;
+		case 'h':
+			ffi_types[i] = &ffi_type_sint32;
+			ffi_args[i] = &args[i].h;
+			break;
+		default:
+			printf("unknown type\n");
+			assert(0);
+			break;
+		}
+	}
+}
+
+void
+wl_interface_default_dispatcher(struct wl_object *target, uint32_t opcode,
+				const struct wl_message *message, void *data,
+				union wl_argument *args)
+{
+	int count;
+	ffi_cif cif;
+	ffi_type *ffi_types[WL_CLOSURE_MAX_ARGS + 2];
+	void * ffi_args[WL_CLOSURE_MAX_ARGS + 2];
+	void (* const * implementation)(void);
+
+	count = arg_count_for_signature(message->signature);
+
+	ffi_types[0] = &ffi_type_pointer;
+	ffi_args[0] = &data;
+	ffi_types[1] = &ffi_type_pointer;
+	ffi_args[1] = ⌖
+
+	convert_arguments_to_ffi(message->signature, args, count,
+				 ffi_types + 2, ffi_args + 2);
+
+	ffi_prep_cif(&cif, FFI_DEFAULT_ABI,
+		     count + 2, &ffi_type_void, ffi_types);
+
+	implementation = target->implementation;
+	ffi_call(&cif, implementation[opcode], NULL, ffi_args);
+}
diff --git a/src/wayland-private.h b/src/wayland-private.h
index ecd7f17..18f280e 100644
--- a/src/wayland-private.h
+++ b/src/wayland-private.h
@@ -123,6 +123,11 @@ wl_closure_print(struct wl_closure *closure, struct wl_object *target, int send)
 void
 wl_closure_destroy(struct wl_closure *closure);
 
+void
+wl_interface_default_dispatcher(struct wl_object *target, uint32_t opcode,
+				const struct wl_message *message,
+				void *data, union wl_argument *args);
+
 extern wl_log_func_t wl_log_handler;
 
 void wl_log(const char *fmt, ...);
diff --git a/src/wayland-util.h b/src/wayland-util.h
index 257a5d1..df7b384 100644
--- a/src/wayland-util.h
+++ b/src/wayland-util.h
@@ -1,5 +1,6 @@
 /*
  * Copyright © 2008 Kristian Høgsberg
+ * Copyright © 2013 Jason Ekstrand
  *
  * Permission to use, copy, modify, distribute, and sell this software and its
  * documentation for any purpose is hereby granted without fee, provided that
@@ -198,6 +199,17 @@ static inline wl_fixed_t wl_fixed_from_int(int i)
 	return i * 256;
 }
 
+union wl_argument {
+	int32_t i;
+	uint32_t u;
+	wl_fixed_t f;
+	const char *s;
+	struct wl_object *o;
+	uint32_t n;
+	struct wl_array *a;
+	int32_t h;
+};
+
 typedef void (*wl_log_func_t)(const char *, va_list);
 
 #ifdef  __cplusplus
diff --git a/tests/.gitignore b/tests/.gitignore
index ccd440a..0e5a973 100644
--- a/tests/.gitignore
+++ b/tests/.gitignore
@@ -4,6 +4,7 @@
 array-test
 client-test
 connection-test
+dispatcher-test
 display-test
 event-loop-test
 exec-fd-leak-checker
diff --git a/tests/Makefile.am b/tests/Makefile.am
index 54157bc..f9ed1ae 100644
--- a/tests/Makefile.am
+++ b/tests/Makefile.am
@@ -3,6 +3,7 @@ TESTS =						\
 	client-test				\
 	display-test				\
 	connection-test				\
+	dispatcher-test				\
 	event-loop-test				\
 	fixed-test				\
 	list-test				\
@@ -25,6 +26,7 @@ array_test_SOURCES = array-test.c $(test_runner_src)
 client_test_SOURCES = client-test.c $(test_runner_src)
 display_test_SOURCES = display-test.c $(test_runner_src)
 connection_test_SOURCES = connection-test.c $(test_runner_src)
+dispatcher_test_SOURCES = dispatcher-test.c $(test_runner_src)
 event_loop_test_SOURCES = event-loop-test.c $(test_runner_src)
 fixed_test_SOURCES = fixed-test.c $(test_runner_src)
 list_test_SOURCES = list-test.c $(test_runner_src)
diff --git a/tests/dispatcher-test.c b/tests/dispatcher-test.c
new file mode 100644
index 0000000..d38e7cb
--- /dev/null
+++ b/tests/dispatcher-test.c
@@ -0,0 +1,95 @@
+/*
+ * Copyright © 2013 Jason Ekstrand
+ *
+ * Permission to use, copy, modify, distribute, and sell this software and its
+ * documentation for any purpose is hereby granted without fee, provided that
+ * the above copyright notice appear in all copies and that both that copyright
+ * notice and this permission notice appear in supporting documentation, and
+ * that the name of the copyright holders not be used in advertising or
+ * publicity pertaining to distribution of the software without specific,
+ * written prior permission.  The copyright holders make no representations
+ * about the suitability of this software for any purpose.  It is provided "as
+ * is" without express or implied warranty.
+ *
+ * THE COPYRIGHT HOLDERS DISCLAIM ALL WARRANTIES WITH REGARD TO THIS SOFTWARE,
+ * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO
+ * EVENT SHALL THE COPYRIGHT HOLDERS BE LIABLE FOR ANY SPECIAL, INDIRECT OR
+ * CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE,
+ * DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
+ * TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE
+ * OF THIS SOFTWARE.
+ */
+
+#include <stdlib.h>
+#include <string.h>
+#include <assert.h>
+
+#include "wayland-private.h"
+#include "test-runner.h"
+
+static void
+test_function(void *data, struct wl_object *target, int32_t i, uint32_t u,
+	      wl_fixed_t f, const char *s, struct wl_object *o, uint32_t n,
+	      struct wl_array *a, int32_t h)
+{
+	union wl_argument *args;
+
+	args = (union wl_argument *)data;
+	args[0].i = i;
+	args[1].u = u;
+	args[2].f = f;
+	args[3].s = s;
+	args[4].o = o;
+	args[5].n = n;
+	args[6].a = a;
+	args[7].h = h;
+	args[8].o = target;
+}
+
+const struct wl_interface *null_interfaces[8];
+
+struct wl_message test_message = {
+	"test_message", "iuf?sonah", null_interfaces
+};
+
+struct wl_interface test_interface = {
+	"test_interface",
+	1,
+	1, &test_message,
+	1, &test_message,
+};
+
+void (* test_function_ptr)(void) = (void (*)(void))&test_function;
+
+TEST(default_dispatcher)
+{
+	struct wl_object target, dummy_obj;
+	struct wl_array arr;
+	union wl_argument args_in[8], args_out[9];
+
+	target.interface = &test_interface;
+	target.implementation = &test_function_ptr;
+	target.id = 7;
+
+	args_in[0].i = -356;
+	args_in[1].u = 42;
+	args_in[2].f = 1007 * 256 + 128; /* 1007.5 */
+	args_in[3].s = "Test String";
+	args_in[4].o = &dummy_obj;
+	args_in[5].n = 75284;
+	args_in[6].a = &arr;
+	args_in[7].h = 37;
+
+	wl_interface_default_dispatcher(&target, 0, &test_message,
+					args_out, args_in);
+
+	assert(args_in[0].i == args_out[0].i);
+	assert(args_in[1].u == args_out[1].u);
+	assert(args_in[2].f == args_out[2].f);
+	assert(args_in[3].s == args_out[3].s);
+	assert(args_in[4].o == args_out[4].o);
+	assert(args_in[5].n == args_out[5].n);
+	assert(args_in[6].a == args_out[6].a);
+	assert(args_in[7].h == args_out[7].h);
+	assert(args_out[8].o == &target);
+}
-- 
1.8.1.2



More information about the wayland-devel mailing list