summaryrefslogtreecommitdiff
path: root/test/test-totem.c
diff options
context:
space:
mode:
Diffstat (limited to 'test/test-totem.c')
-rw-r--r--test/test-totem.c605
1 files changed, 605 insertions, 0 deletions
diff --git a/test/test-totem.c b/test/test-totem.c
new file mode 100644
index 00000000..bafa8d23
--- /dev/null
+++ b/test/test-totem.c
@@ -0,0 +1,605 @@
+/*
+ * Copyright © 2018 Red Hat, Inc.
+ *
+ * Permission is hereby granted, free of charge, to any person obtaining a
+ * copy of this software and associated documentation files (the "Software"),
+ * to deal in the Software without restriction, including without limitation
+ * the rights to use, copy, modify, merge, publish, distribute, sublicense,
+ * and/or sell copies of the Software, and to permit persons to whom the
+ * Software is furnished to do so, subject to the following conditions:
+ *
+ * The above copyright notice and this permission notice (including the next
+ * paragraph) shall be included in all copies or substantial portions of the
+ * Software.
+ *
+ * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+ * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+ * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
+ * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+ * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
+ * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
+ * DEALINGS IN THE SOFTWARE.
+ */
+
+#include <config.h>
+
+#include <check.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <libinput.h>
+#include <unistd.h>
+#include <stdbool.h>
+#include <stdarg.h>
+
+#include "libinput-util.h"
+#include "evdev-tablet.h"
+#include "litest.h"
+
+START_TEST(totem_type)
+{
+ struct litest_device *dev = litest_current_device();
+ struct libinput *li = dev->libinput;
+ struct libinput_event *event;
+ struct libinput_event_tablet_tool *t;
+ struct libinput_tablet_tool *tool;
+
+ litest_drain_events(li);
+
+ litest_tablet_proximity_in(dev, 50, 50, NULL);
+ libinput_dispatch(li);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_PROXIMITY);
+ tool = libinput_event_tablet_tool_get_tool(t);
+
+ ck_assert_int_eq(libinput_tablet_tool_get_type(tool),
+ LIBINPUT_TABLET_TOOL_TYPE_TOTEM);
+ libinput_event_destroy(event);
+}
+END_TEST
+
+START_TEST(totem_axes)
+{
+ struct litest_device *dev = litest_current_device();
+ struct libinput *li = dev->libinput;
+ struct libinput_event *event;
+ struct libinput_event_tablet_tool *t;
+ struct libinput_tablet_tool *tool;
+
+ litest_drain_events(li);
+
+ litest_tablet_proximity_in(dev, 50, 50, NULL);
+ libinput_dispatch(li);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_PROXIMITY);
+ tool = libinput_event_tablet_tool_get_tool(t);
+
+ ck_assert(libinput_tablet_tool_has_rotation(tool));
+ ck_assert(libinput_tablet_tool_has_size(tool));
+ ck_assert(libinput_tablet_tool_has_button(tool, BTN_0));
+
+ libinput_event_destroy(event);
+}
+END_TEST
+
+START_TEST(totem_proximity_in_out)
+{
+ struct litest_device *dev = litest_current_device();
+ struct libinput *li = dev->libinput;
+ struct libinput_event *event;
+ struct libinput_event_tablet_tool *t;
+
+ litest_drain_events(li);
+
+ litest_tablet_proximity_in(dev, 50, 50, NULL);
+ libinput_dispatch(li);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_PROXIMITY);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_proximity_state(t),
+ LIBINPUT_TABLET_TOOL_PROXIMITY_STATE_IN);
+ libinput_event_destroy(event);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_TIP);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_tip_state(t),
+ LIBINPUT_TABLET_TOOL_TIP_DOWN);
+ libinput_event_destroy(event);
+
+ litest_assert_empty_queue(li);
+ litest_tablet_proximity_out(dev);
+ libinput_dispatch(li);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_TIP);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_tip_state(t),
+ LIBINPUT_TABLET_TOOL_TIP_UP);
+ libinput_event_destroy(event);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_PROXIMITY);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_proximity_state(t),
+ LIBINPUT_TABLET_TOOL_PROXIMITY_STATE_OUT);
+ libinput_event_destroy(event);
+}
+END_TEST
+
+START_TEST(totem_proximity_in_on_init)
+{
+ struct litest_device *dev = litest_current_device();
+ struct libinput *li;
+ struct libinput_event *event;
+ struct libinput_event_tablet_tool *t;
+ const char *devnode;
+ double x, y;
+ double w, h;
+ const struct input_absinfo *abs;
+
+ abs = libevdev_get_abs_info(dev->evdev, ABS_MT_POSITION_X);
+ w = (abs->maximum - abs->minimum + 1)/abs->resolution;
+ abs = libevdev_get_abs_info(dev->evdev, ABS_MT_POSITION_Y);
+ h = (abs->maximum - abs->minimum + 1)/abs->resolution;
+
+ litest_tablet_proximity_in(dev, 50, 50, NULL);
+
+ /* for simplicity, we create a new litest context */
+ devnode = libevdev_uinput_get_devnode(dev->uinput);
+ li = litest_create_context();
+ libinput_path_add_device(li, devnode);
+ libinput_dispatch(li);
+
+ litest_wait_for_event_of_type(li,
+ LIBINPUT_EVENT_DEVICE_ADDED,
+ -1);
+ event = libinput_get_event(li);
+ libinput_event_destroy(event);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_PROXIMITY);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_proximity_state(t),
+ LIBINPUT_TABLET_TOOL_PROXIMITY_STATE_IN);
+ x = libinput_event_tablet_tool_get_x(t);
+ y = libinput_event_tablet_tool_get_y(t);
+
+ ck_assert_double_gt(x, w/2 - 1);
+ ck_assert_double_lt(x, w/2 + 1);
+ ck_assert_double_gt(y, h/2 - 1);
+ ck_assert_double_lt(y, h/2 + 1);
+
+ libinput_event_destroy(event);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_TIP);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_tip_state(t),
+ LIBINPUT_TABLET_TOOL_TIP_DOWN);
+ x = libinput_event_tablet_tool_get_x(t);
+ y = libinput_event_tablet_tool_get_y(t);
+
+ ck_assert_double_gt(x, w/2 - 1);
+ ck_assert_double_lt(x, w/2 + 1);
+ ck_assert_double_gt(y, h/2 - 1);
+ ck_assert_double_lt(y, h/2 + 1);
+
+ libinput_event_destroy(event);
+
+ litest_assert_empty_queue(li);
+
+ libinput_unref(li);
+}
+END_TEST
+
+START_TEST(totem_proximity_out_on_suspend)
+{
+ struct litest_device *dev = litest_current_device();
+ struct libinput *li;
+ struct libinput_event *event;
+ struct libinput_event_tablet_tool *t;
+ const char *devnode;
+
+ /* for simplicity, we create a new litest context */
+ devnode = libevdev_uinput_get_devnode(dev->uinput);
+ li = litest_create_context();
+ libinput_path_add_device(li, devnode);
+
+ litest_tablet_proximity_in(dev, 50, 50, NULL);
+ litest_drain_events(li);
+
+ libinput_suspend(li);
+
+ libinput_dispatch(li);
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_TIP);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_tip_state(t),
+ LIBINPUT_TABLET_TOOL_TIP_UP);
+ libinput_event_destroy(event);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_PROXIMITY);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_proximity_state(t),
+ LIBINPUT_TABLET_TOOL_PROXIMITY_STATE_OUT);
+ libinput_event_destroy(event);
+
+ litest_assert_only_typed_events(li, LIBINPUT_EVENT_DEVICE_REMOVED);
+ libinput_unref(li);
+}
+END_TEST
+
+START_TEST(totem_motion)
+{
+ struct litest_device *dev = litest_current_device();
+ struct libinput *li = dev->libinput;
+ struct libinput_event *event;
+ double x = 50, y = 50;
+ double current_x, current_y, old_x, old_y;
+
+ litest_tablet_proximity_in(dev, x, y, NULL);
+ litest_drain_events(li);
+
+ for (int i = 0; i < 30; i++, x++, y--) {
+ struct libinput_event_tablet_tool *t;
+
+ litest_tablet_motion(dev, x + 1, y + 1, NULL);
+ libinput_dispatch(li);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event, LIBINPUT_EVENT_TABLET_TOOL_AXIS);
+
+ ck_assert(libinput_event_tablet_tool_x_has_changed(t));
+ ck_assert(libinput_event_tablet_tool_y_has_changed(t));
+
+ current_x = libinput_event_tablet_tool_get_x(t);
+ current_y = libinput_event_tablet_tool_get_y(t);
+ if (i != 0) {
+ ck_assert_double_gt(current_x, old_x);
+ ck_assert_double_lt(current_y, old_y);
+ }
+ old_x = current_x;
+ old_y = current_y;
+
+ libinput_event_destroy(event);
+ }
+}
+END_TEST
+
+START_TEST(totem_rotation)
+{
+ struct litest_device *dev = litest_current_device();
+ struct libinput *li = dev->libinput;
+ struct libinput_event *event;
+ double r, old_r;
+ struct axis_replacement axes[] = {
+ { ABS_MT_ORIENTATION, 50 }, /* mid-point is 0 */
+ { -1, -1 }
+ };
+
+ litest_tablet_proximity_in(dev, 50, 50, axes);
+ litest_drain_events(li);
+
+ old_r = 360;
+
+ for (int i = 1; i < 30; i++) {
+ struct libinput_event_tablet_tool *t;
+
+
+ litest_axis_set_value(axes, ABS_MT_ORIENTATION, 50 + i);
+ litest_tablet_motion(dev, 50, 50, axes);
+ libinput_dispatch(li);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event, LIBINPUT_EVENT_TABLET_TOOL_AXIS);
+
+ ck_assert(!libinput_event_tablet_tool_x_has_changed(t));
+ ck_assert(!libinput_event_tablet_tool_y_has_changed(t));
+ ck_assert(libinput_event_tablet_tool_rotation_has_changed(t));
+
+ r = libinput_event_tablet_tool_get_rotation(t);
+ ck_assert_double_lt(r, old_r);
+ old_r = r;
+
+ libinput_event_destroy(event);
+ }
+
+ old_r = 0;
+
+ for (int i = 1; i < 30; i++) {
+ struct libinput_event_tablet_tool *t;
+
+
+ litest_axis_set_value(axes, ABS_MT_ORIENTATION, 50 - i);
+ litest_tablet_motion(dev, 50, 50, axes);
+ libinput_dispatch(li);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event, LIBINPUT_EVENT_TABLET_TOOL_AXIS);
+
+ ck_assert(!libinput_event_tablet_tool_x_has_changed(t));
+ ck_assert(!libinput_event_tablet_tool_y_has_changed(t));
+ ck_assert(libinput_event_tablet_tool_rotation_has_changed(t));
+
+ r = libinput_event_tablet_tool_get_rotation(t);
+ ck_assert_double_gt(r, old_r);
+ old_r = r;
+
+ libinput_event_destroy(event);
+ }
+}
+END_TEST
+
+START_TEST(totem_size)
+{
+ struct litest_device *dev = litest_current_device();
+ struct libinput *li = dev->libinput;
+ struct libinput_event *event;
+ struct libinput_event_tablet_tool *t;
+ double smin, smaj;
+
+ litest_drain_events(li);
+
+ litest_tablet_proximity_in(dev, 50, 50, NULL);
+ libinput_dispatch(li);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event, LIBINPUT_EVENT_TABLET_TOOL_PROXIMITY);
+ ck_assert(libinput_event_tablet_tool_size_major_has_changed(t));
+ ck_assert(libinput_event_tablet_tool_size_minor_has_changed(t));
+ smaj = libinput_event_tablet_tool_get_size_major(t);
+ smin = libinput_event_tablet_tool_get_size_minor(t);
+ libinput_event_destroy(event);
+
+ ck_assert_double_eq(smaj, 71.8);
+ ck_assert_double_eq(smin, 71.8);
+
+ litest_drain_events(li);
+}
+END_TEST
+
+START_TEST(totem_button)
+{
+ struct litest_device *dev = litest_current_device();
+ struct libinput *li = dev->libinput;
+ struct libinput_event *event;
+ struct libinput_event_tablet_tool *t;
+
+ litest_tablet_proximity_in(dev, 30, 40, NULL);
+ litest_drain_events(li);
+
+ litest_button_click(dev, BTN_0, true);
+ libinput_dispatch(li);
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event, LIBINPUT_EVENT_TABLET_TOOL_BUTTON);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_button(t), BTN_0);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_button_state(t),
+ LIBINPUT_BUTTON_STATE_PRESSED);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_tip_state(t),
+ LIBINPUT_TABLET_TOOL_TIP_DOWN);
+ libinput_event_destroy(event);
+
+ litest_button_click(dev, BTN_0, false);
+ libinput_dispatch(li);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event, LIBINPUT_EVENT_TABLET_TOOL_BUTTON);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_button(t), BTN_0);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_button_state(t),
+ LIBINPUT_BUTTON_STATE_RELEASED);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_tip_state(t),
+ LIBINPUT_TABLET_TOOL_TIP_DOWN);
+ libinput_event_destroy(event);
+}
+END_TEST
+
+START_TEST(totem_button_down_on_init)
+{
+ struct litest_device *dev = litest_current_device();
+ struct libinput *li;
+ struct libinput_event *event;
+ struct libinput_event_tablet_tool *t;
+ const char *devnode;
+
+ litest_tablet_proximity_in(dev, 50, 50, NULL);
+ litest_button_click(dev, BTN_0, true);
+
+ /* for simplicity, we create a new litest context */
+ devnode = libevdev_uinput_get_devnode(dev->uinput);
+ li = litest_create_context();
+ libinput_path_add_device(li, devnode);
+ libinput_dispatch(li);
+
+ litest_wait_for_event_of_type(li,
+ LIBINPUT_EVENT_DEVICE_ADDED,
+ -1);
+ event = libinput_get_event(li);
+ libinput_event_destroy(event);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_PROXIMITY);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_proximity_state(t),
+ LIBINPUT_TABLET_TOOL_PROXIMITY_STATE_IN);
+
+ libinput_event_destroy(event);
+
+ event = libinput_get_event(li);
+ t = litest_is_tablet_event(event,
+ LIBINPUT_EVENT_TABLET_TOOL_TIP);
+ ck_assert_int_eq(libinput_event_tablet_tool_get_tip_state(t),
+ LIBINPUT_TABLET_TOOL_TIP_DOWN);
+
+ libinput_event_destroy(event);
+
+ /* The button is down on init but we don't expect an event */
+ litest_assert_empty_queue(li);
+
+ litest_button_click(dev, BTN_0, false);
+ libinput_dispatch(li);
+ litest_assert_empty_queue(li);
+
+ /* but buttons after this should be sent */
+ litest_button_click(dev, BTN_0, true);
+ libinput_dispatch(li);
+ litest_assert_tablet_button_event(li, BTN_0, LIBINPUT_BUTTON_STATE_PRESSED);
+ litest_button_click(dev, BTN_0, false);
+ libinput_dispatch(li);
+ litest_assert_tablet_button_event(li, BTN_0, LIBINPUT_BUTTON_STATE_RELEASED);
+
+ libinput_unref(li);
+}
+END_TEST
+
+START_TEST(totem_button_up_on_delete)
+{
+ struct libinput *li = litest_create_context();
+ struct litest_device *dev = litest_add_device(li, LITEST_DELL_CANVAS_TOTEM);
+ struct libevdev *evdev = libevdev_new();
+
+ litest_tablet_proximity_in(dev, 10, 10, NULL);
+ litest_drain_events(li);
+
+ litest_button_click(dev, BTN_0, true);
+ litest_drain_events(li);
+
+ litest_delete_device(dev);
+ libinput_dispatch(li);
+
+ litest_assert_tablet_button_event(li,
+ BTN_0,
+ LIBINPUT_BUTTON_STATE_RELEASED);
+
+ litest_assert_tablet_tip_event(li, LIBINPUT_TABLET_TOOL_TIP_UP);
+ litest_assert_tablet_proximity_event(li,
+ LIBINPUT_TABLET_TOOL_PROXIMITY_STATE_OUT);
+ libevdev_free(evdev);
+ libinput_unref(li);
+}
+END_TEST
+
+START_TEST(totem_arbitration_below)
+{
+ struct litest_device *totem = litest_current_device();
+ struct litest_device *touch;
+ struct libinput *li = totem->libinput;
+
+ touch = litest_add_device(li, LITEST_DELL_CANVAS_TOTEM_TOUCH);
+ litest_drain_events(li);
+
+ /* touches below the totem, cancelled once the totem is down */
+ litest_touch_down(touch, 0, 50, 50);
+ libinput_dispatch(li);
+ litest_assert_touch_down_frame(li);
+ litest_touch_move_to(touch, 0, 50, 50, 50, 70, 10);
+ libinput_dispatch(li);
+ while (libinput_next_event_type(li)) {
+ litest_assert_touch_motion_frame(li);
+ }
+
+ litest_tablet_proximity_in(totem, 50, 70, NULL);
+ libinput_dispatch(li);
+
+ litest_assert_tablet_proximity_event(li, LIBINPUT_TABLET_TOOL_PROXIMITY_STATE_IN);
+ litest_assert_tablet_tip_event(li, LIBINPUT_TABLET_TOOL_TIP_DOWN);
+ litest_assert_touch_cancel(li);
+
+ litest_touch_move_to(touch, 0, 50, 70, 20, 50, 10);
+ litest_assert_empty_queue(li);
+
+ litest_tablet_motion(totem, 20, 50, NULL);
+ litest_assert_only_typed_events(li, LIBINPUT_EVENT_TABLET_TOOL_AXIS);
+
+ litest_touch_up(touch, 0);
+ litest_assert_empty_queue(li);
+
+ litest_delete_device(touch);
+}
+END_TEST
+
+START_TEST(totem_arbitration_during)
+{
+ struct litest_device *totem = litest_current_device();
+ struct litest_device *touch;
+ struct libinput *li = totem->libinput;
+
+ touch = litest_add_device(li, LITEST_DELL_CANVAS_TOTEM_TOUCH);
+ litest_drain_events(li);
+
+ litest_tablet_proximity_in(totem, 50, 50, NULL);
+ libinput_dispatch(li);
+
+ litest_drain_events(li);
+
+ for (int i = 0; i < 3; i++) {
+ litest_touch_down(touch, 0, 51, 51);
+ litest_touch_move_to(touch, 0, 51, 50, 90, 80, 10);
+ litest_touch_up(touch, 0);
+
+ litest_assert_empty_queue(li);
+ }
+
+ litest_delete_device(touch);
+}
+END_TEST
+
+START_TEST(totem_arbitration_outside_rect)
+{
+ struct litest_device *totem = litest_current_device();
+ struct litest_device *touch;
+ struct libinput *li = totem->libinput;
+
+ touch = litest_add_device(li, LITEST_DELL_CANVAS_TOTEM_TOUCH);
+ litest_drain_events(li);
+
+ litest_tablet_proximity_in(totem, 50, 50, NULL);
+ libinput_dispatch(li);
+
+ litest_drain_events(li);
+
+ for (int i = 0; i < 3; i++) {
+ litest_touch_down(touch, 0, 81, 51);
+ litest_touch_move_to(touch, 0, 81, 50, 90, 80, 10);
+ litest_touch_up(touch, 0);
+ libinput_dispatch(li);
+
+ litest_assert_touch_sequence(li);
+ }
+
+ /* moving onto the totem is fine */
+ litest_touch_down(touch, 0, 81, 51);
+ litest_touch_move_to(touch, 0, 81, 50, 50, 50, 10);
+ litest_touch_up(touch, 0);
+ libinput_dispatch(li);
+
+ litest_assert_touch_sequence(li);
+
+ litest_delete_device(touch);
+}
+END_TEST
+
+TEST_COLLECTION(totem)
+{
+ litest_add("totem:tool", totem_type, LITEST_TOTEM, LITEST_ANY);
+ litest_add("totem:tool", totem_axes, LITEST_TOTEM, LITEST_ANY);
+ litest_add("totem:proximity", totem_proximity_in_out, LITEST_TOTEM, LITEST_ANY);
+ litest_add("totem:proximity", totem_proximity_in_on_init, LITEST_TOTEM, LITEST_ANY);
+ litest_add("totem:proximity", totem_proximity_out_on_suspend, LITEST_TOTEM, LITEST_ANY);
+
+ litest_add("totem:axes", totem_motion, LITEST_TOTEM, LITEST_ANY);
+ litest_add("totem:axes", totem_rotation, LITEST_TOTEM, LITEST_ANY);
+ litest_add("totem:axes", totem_size, LITEST_TOTEM, LITEST_ANY);
+ litest_add("totem:button", totem_button, LITEST_TOTEM, LITEST_ANY);
+ litest_add("totem:button", totem_button_down_on_init, LITEST_TOTEM, LITEST_ANY);
+ litest_add_no_device("totem:button", totem_button_up_on_delete);
+
+ litest_add("totem:arbitration", totem_arbitration_below, LITEST_TOTEM, LITEST_ANY);
+ litest_add("totem:arbitration", totem_arbitration_during, LITEST_TOTEM, LITEST_ANY);
+ litest_add("totem:arbitration", totem_arbitration_outside_rect, LITEST_TOTEM, LITEST_ANY);
+}