2 * Copyright (C) 2010-2011 Robert Ancell.
3 * Author: Robert Ancell <robert.ancell@canonical.com>
5 * This program is free software: you can redistribute it and/or modify it under
6 * the terms of the GNU General Public License as published by the Free Software
7 * Foundation, either version 3 of the License, or (at your option) any later
8 * version. See http://www.gnu.org/copyleft/gpl.html the full text of the
18 #include "display-manager.h"
19 #include "configuration.h"
20 #include "seat-xlocal.h"
21 #include "seat-xremote.h"
22 #include "seat-unity.h"
23 #include "seat-surfaceflinger.h"
32 static guint signals[LAST_SIGNAL] = { 0 };
34 struct DisplayManagerPrivate
36 /* The seats available */
39 /* TRUE if stopping the display manager (waiting for seats to stop) */
46 G_DEFINE_TYPE (DisplayManager, display_manager, G_TYPE_OBJECT);
49 display_manager_new (void)
51 return g_object_new (DISPLAY_MANAGER_TYPE, NULL);
55 display_manager_get_seats (DisplayManager *manager)
57 return manager->priv->seats;
61 check_stopped (DisplayManager *manager)
63 if (manager->priv->stopping &&
64 !manager->priv->stopped &&
65 g_list_length (manager->priv->seats) == 0)
67 manager->priv->stopped = TRUE;
68 g_debug ("Display manager stopped");
69 g_signal_emit (manager, signals[STOPPED], 0);
74 seat_stopped_cb (Seat *seat, DisplayManager *manager)
76 manager->priv->seats = g_list_remove (manager->priv->seats, seat);
77 g_signal_handlers_disconnect_matched (seat, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, manager);
79 if (!manager->priv->stopping)
80 g_signal_emit (manager, signals[SEAT_REMOVED], 0, seat);
82 g_object_unref (seat);
84 check_stopped (manager);
88 display_manager_add_seat (DisplayManager *manager, Seat *seat)
92 g_return_val_if_fail (!manager->priv->stopping, FALSE);
94 result = seat_start (SEAT (seat));
98 manager->priv->seats = g_list_append (manager->priv->seats, g_object_ref (seat));
99 g_signal_connect (seat, "stopped", G_CALLBACK (seat_stopped_cb), manager);
100 g_signal_emit (manager, signals[SEAT_ADDED], 0, seat);
106 display_manager_start (DisplayManager *manager)
108 g_return_if_fail (manager != NULL);
110 /* Disable Plymouth if no X servers are replacing it */
111 if (plymouth_get_is_active ())
113 g_debug ("Stopping Plymouth, no displays replace it");
114 plymouth_quit (FALSE);
119 display_manager_stop (DisplayManager *manager)
123 g_return_if_fail (manager != NULL);
125 if (manager->priv->stopping)
128 g_debug ("Stopping display manager");
130 manager->priv->stopping = TRUE;
132 /* Stop all the seats. Copy the list as it might be modified if a seat stops during this loop */
133 seats = g_list_copy (manager->priv->seats);
134 for (link = seats; link; link = link->next)
136 Seat *seat = link->data;
141 check_stopped (manager);
145 display_manager_init (DisplayManager *manager)
147 manager->priv = G_TYPE_INSTANCE_GET_PRIVATE (manager, DISPLAY_MANAGER_TYPE, DisplayManagerPrivate);
149 /* Load the seat modules */
150 seat_register_module ("xlocal", SEAT_XLOCAL_TYPE);
151 seat_register_module ("xremote", SEAT_XREMOTE_TYPE);
152 seat_register_module ("unity", SEAT_UNITY_TYPE);
153 seat_register_module ("surfaceflinger", SEAT_SURFACEFLINGER_TYPE);
157 display_manager_finalize (GObject *object)
159 DisplayManager *self;
162 self = DISPLAY_MANAGER (object);
164 for (link = self->priv->seats; link; link = link->next)
166 Seat *seat = link->data;
167 g_signal_handlers_disconnect_matched (seat, G_SIGNAL_MATCH_DATA, 0, 0, NULL, NULL, self);
169 g_list_free_full (self->priv->seats, g_object_unref);
171 G_OBJECT_CLASS (display_manager_parent_class)->finalize (object);
175 display_manager_class_init (DisplayManagerClass *klass)
177 GObjectClass *object_class = G_OBJECT_CLASS (klass);
179 object_class->finalize = display_manager_finalize;
181 g_type_class_add_private (klass, sizeof (DisplayManagerPrivate));
183 signals[SEAT_ADDED] =
184 g_signal_new ("seat-added",
185 G_TYPE_FROM_CLASS (klass),
187 G_STRUCT_OFFSET (DisplayManagerClass, seat_added),
190 G_TYPE_NONE, 1, SEAT_TYPE);
191 signals[SEAT_REMOVED] =
192 g_signal_new ("seat-removed",
193 G_TYPE_FROM_CLASS (klass),
195 G_STRUCT_OFFSET (DisplayManagerClass, seat_removed),
198 G_TYPE_NONE, 1, SEAT_TYPE);
200 g_signal_new ("stopped",
201 G_TYPE_FROM_CLASS (klass),
203 G_STRUCT_OFFSET (DisplayManagerClass, stopped),