7c6fe21daf
Signed-off-by: Simon McVittie <smcv@debian.org> Part-of: <https://gitlab.gnome.org/GNOME/mutter/-/merge_requests/2049>
293 lines
9.7 KiB
C
293 lines
9.7 KiB
C
/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */
|
|
/*
|
|
* Copyright (C) 2020 Canonical, Ltd.
|
|
*
|
|
* This program is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU General Public License as
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
* License, or (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Author: Marco Trevisan <marco.trevisan@canonical.com>
|
|
*/
|
|
|
|
#include "config.h"
|
|
|
|
#include "orientation-manager-unit-tests.h"
|
|
|
|
#include "tests/meta-sensors-proxy-mock.h"
|
|
|
|
const char *
|
|
orientation_to_string (MetaOrientation orientation)
|
|
{
|
|
switch (orientation)
|
|
{
|
|
case META_ORIENTATION_UNDEFINED:
|
|
return "(undefined)";
|
|
case META_ORIENTATION_NORMAL:
|
|
return "normal";
|
|
case META_ORIENTATION_BOTTOM_UP:
|
|
return "bottom-up";
|
|
case META_ORIENTATION_LEFT_UP:
|
|
return "left-up";
|
|
case META_ORIENTATION_RIGHT_UP:
|
|
return "right-up";
|
|
default:
|
|
return "(invalid)";
|
|
}
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
MetaOrientation expected;
|
|
MetaOrientation orientation;
|
|
gulong connection_id;
|
|
guint timeout_id;
|
|
unsigned int times_signalled;
|
|
} WaitForOrientation;
|
|
|
|
static void
|
|
on_orientation_changed (WaitForOrientation *wfo,
|
|
MetaOrientationManager *orientation_manager)
|
|
{
|
|
wfo->orientation = meta_orientation_manager_get_orientation (orientation_manager);
|
|
wfo->times_signalled++;
|
|
|
|
g_test_message ("wait_for_orientation_changes: Orientation changed to %d: %s",
|
|
wfo->orientation, orientation_to_string (wfo->orientation));
|
|
}
|
|
|
|
static gboolean
|
|
on_max_wait_timeout (gpointer data)
|
|
{
|
|
WaitForOrientation *wfo = data;
|
|
|
|
wfo->timeout_id = 0;
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
/*
|
|
* Assert that the orientation eventually changes to @orientation.
|
|
*/
|
|
void
|
|
wait_for_orientation (MetaOrientationManager *orientation_manager,
|
|
MetaOrientation orientation,
|
|
unsigned int *times_signalled_out)
|
|
{
|
|
WaitForOrientation wfo = {
|
|
.expected = orientation,
|
|
};
|
|
|
|
wfo.orientation = meta_orientation_manager_get_orientation (orientation_manager);
|
|
g_test_message ("%s: Waiting for orientation to change from "
|
|
"%d: %s to %d: %s...",
|
|
G_STRFUNC, wfo.orientation,
|
|
orientation_to_string (wfo.orientation),
|
|
orientation, orientation_to_string (orientation));
|
|
|
|
/* This timeout can be relatively generous because we don't expect to
|
|
* reach it: if we do, that's a test failure. */
|
|
wfo.timeout_id = g_timeout_add_seconds (10, on_max_wait_timeout, &wfo);
|
|
wfo.connection_id = g_signal_connect_swapped (orientation_manager,
|
|
"orientation-changed",
|
|
G_CALLBACK (on_orientation_changed),
|
|
&wfo);
|
|
|
|
while (wfo.orientation != orientation && wfo.timeout_id != 0)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
if (wfo.orientation != orientation)
|
|
{
|
|
g_error ("Timed out waiting for orientation to change from %s to %s "
|
|
"(received %u orientation-changed signal(s) while waiting)",
|
|
orientation_to_string (wfo.orientation),
|
|
orientation_to_string (orientation),
|
|
wfo.times_signalled);
|
|
}
|
|
|
|
g_test_message ("%s: Orientation is now %d: %s",
|
|
G_STRFUNC, orientation, orientation_to_string (orientation));
|
|
|
|
g_clear_handle_id (&wfo.timeout_id, g_source_remove);
|
|
g_signal_handler_disconnect (orientation_manager, wfo.connection_id);
|
|
|
|
if (times_signalled_out != NULL)
|
|
*times_signalled_out = wfo.times_signalled;
|
|
}
|
|
|
|
/*
|
|
* Wait for a possible orientation change, but don't assert that one occurs.
|
|
*/
|
|
void
|
|
wait_for_possible_orientation_change (MetaOrientationManager *orientation_manager,
|
|
unsigned int *times_signalled_out)
|
|
{
|
|
WaitForOrientation wfo = {
|
|
.expected = META_ORIENTATION_UNDEFINED,
|
|
};
|
|
|
|
wfo.orientation = meta_orientation_manager_get_orientation (orientation_manager);
|
|
g_test_message ("%s: Waiting for orientation to maybe change from %d: %s...",
|
|
G_STRFUNC, wfo.orientation,
|
|
orientation_to_string (wfo.orientation));
|
|
|
|
/* This can't be as long as the timeout for wait_for_orientation(),
|
|
* because in the usual case we expect to reach this timeout: we're
|
|
* only waiting so that if the orientation (incorrectly?) changed here,
|
|
* we'd have a chance to detect that. */
|
|
wfo.timeout_id = g_timeout_add (1000, on_max_wait_timeout, &wfo);
|
|
wfo.connection_id = g_signal_connect_swapped (orientation_manager,
|
|
"orientation-changed",
|
|
G_CALLBACK (on_orientation_changed),
|
|
&wfo);
|
|
|
|
while (wfo.times_signalled == 0 && wfo.timeout_id != 0)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
if (wfo.timeout_id == 0)
|
|
{
|
|
g_test_message ("%s: Orientation didn't change", G_STRFUNC);
|
|
}
|
|
else
|
|
{
|
|
g_test_message ("%s: Orientation is now %d: %s",
|
|
G_STRFUNC, wfo.orientation,
|
|
orientation_to_string (wfo.orientation));
|
|
}
|
|
|
|
g_clear_handle_id (&wfo.timeout_id, g_source_remove);
|
|
g_signal_handler_disconnect (orientation_manager, wfo.connection_id);
|
|
|
|
if (times_signalled_out != NULL)
|
|
*times_signalled_out = wfo.times_signalled;
|
|
}
|
|
|
|
static void
|
|
meta_test_orientation_manager_no_daemon (void)
|
|
{
|
|
g_autoptr (MetaOrientationManager) manager = NULL;
|
|
|
|
manager = g_object_new (META_TYPE_ORIENTATION_MANAGER, NULL);
|
|
g_assert_false (meta_orientation_manager_has_accelerometer (manager));
|
|
g_assert_cmpuint (meta_orientation_manager_get_orientation (manager),
|
|
==,
|
|
META_ORIENTATION_UNDEFINED);
|
|
}
|
|
|
|
static void
|
|
meta_test_orientation_manager_no_device (void)
|
|
{
|
|
g_autoptr (MetaOrientationManager) manager = NULL;
|
|
MetaSensorsProxyMock* orientation_mock = NULL;
|
|
|
|
orientation_mock = meta_sensors_proxy_mock_get ();
|
|
manager = g_object_new (META_TYPE_ORIENTATION_MANAGER, NULL);
|
|
g_assert_false (meta_orientation_manager_has_accelerometer (manager));
|
|
g_assert_cmpuint (meta_orientation_manager_get_orientation (manager),
|
|
==,
|
|
META_ORIENTATION_UNDEFINED);
|
|
|
|
g_object_unref (orientation_mock);
|
|
}
|
|
|
|
static gboolean
|
|
on_wait_for_accel_timeout (gpointer data)
|
|
{
|
|
guint *timeout_p = data;
|
|
|
|
*timeout_p = 0;
|
|
return G_SOURCE_REMOVE;
|
|
}
|
|
|
|
static void
|
|
meta_test_orientation_manager_has_accelerometer (void)
|
|
{
|
|
g_autoptr (MetaOrientationManager) manager = NULL;
|
|
g_autoptr (MetaSensorsProxyMock) orientation_mock = NULL;
|
|
guint timeout_id;
|
|
|
|
manager = g_object_new (META_TYPE_ORIENTATION_MANAGER, NULL);
|
|
orientation_mock = meta_sensors_proxy_mock_get ();
|
|
|
|
timeout_id = g_timeout_add_seconds (10, on_wait_for_accel_timeout, &timeout_id);
|
|
meta_sensors_proxy_mock_set_property (orientation_mock,
|
|
"HasAccelerometer",
|
|
g_variant_new_boolean (TRUE));
|
|
|
|
while (!meta_orientation_manager_has_accelerometer (manager) &&
|
|
timeout_id != 0)
|
|
g_main_context_iteration (NULL, TRUE);
|
|
|
|
g_debug ("Checking whether accelerometer is present");
|
|
g_assert_true (meta_orientation_manager_has_accelerometer (manager));
|
|
g_assert_cmpuint (meta_orientation_manager_get_orientation (manager),
|
|
==,
|
|
META_ORIENTATION_UNDEFINED);
|
|
g_clear_handle_id (&timeout_id, g_source_remove);
|
|
}
|
|
|
|
static void
|
|
orientation_changed_cb (MetaOrientationManager *manager,
|
|
gpointer user_data)
|
|
{
|
|
gboolean *changed_called = user_data;
|
|
|
|
*changed_called = TRUE;
|
|
}
|
|
|
|
static void
|
|
meta_test_orientation_manager_accelerometer_orientations (void)
|
|
{
|
|
g_autoptr (MetaOrientationManager) manager = NULL;
|
|
g_autoptr (MetaSensorsProxyMock) orientation_mock = NULL;
|
|
|
|
manager = g_object_new (META_TYPE_ORIENTATION_MANAGER, NULL);
|
|
orientation_mock = meta_sensors_proxy_mock_get ();
|
|
|
|
MetaOrientation initial;
|
|
gboolean changed_called;
|
|
unsigned i;
|
|
|
|
g_signal_connect (manager, "orientation-changed",
|
|
G_CALLBACK (orientation_changed_cb),
|
|
&changed_called);
|
|
|
|
initial = meta_orientation_manager_get_orientation (manager);
|
|
|
|
for (i = initial + 1; i != initial; i = (i + 1) % META_N_ORIENTATIONS)
|
|
{
|
|
unsigned int times_signalled = 0;
|
|
|
|
changed_called = FALSE;
|
|
g_debug ("Checking orientation %d", i);
|
|
meta_sensors_proxy_mock_set_orientation (orientation_mock, i);
|
|
wait_for_orientation (manager, i, ×_signalled);
|
|
g_assert_cmpuint (times_signalled, <=, 1);
|
|
|
|
if (i != META_ORIENTATION_UNDEFINED)
|
|
g_assert_true (changed_called);
|
|
else
|
|
g_assert_false (changed_called);
|
|
}
|
|
}
|
|
|
|
void
|
|
init_orientation_manager_tests (void)
|
|
{
|
|
g_test_add_func ("/backends/orientation-manager/no-daemon",
|
|
meta_test_orientation_manager_no_daemon);
|
|
g_test_add_func ("/backends/orientation-manager/no-device",
|
|
meta_test_orientation_manager_no_device);
|
|
g_test_add_func ("/backends/orientation-manager/has-accelerometer",
|
|
meta_test_orientation_manager_has_accelerometer);
|
|
g_test_add_func ("/backends/orientation-manager/accelerometer-orientations",
|
|
meta_test_orientation_manager_accelerometer_orientations);
|
|
}
|