Files
grd/grd-vnc-server.c
2026-02-13 13:06:50 +09:00

302 lines
8.4 KiB
C

/*
* Copyright (C) 2015 Red Hat Inc.
*
* 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, write to the Free Software
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
* 02111-1307, USA.
*
* Written by:
* Jonas Ådahl <jadahl@gmail.com>
*/
#include "config.h"
#include "grd-vnc-server.h"
#include <gio/gio.h>
#include <rfb/rfb.h>
#include "grd-context.h"
#include "grd-debug.h"
#include "grd-session-vnc.h"
#include "grd-throttler.h"
#include "grd-utils.h"
enum
{
PROP_0,
PROP_CONTEXT,
};
struct _GrdVncServer
{
GSocketService parent;
GrdThrottler *throttler;
GList *sessions;
GList *stopped_sessions;
guint cleanup_sessions_idle_id;
GrdContext *context;
};
G_DEFINE_TYPE (GrdVncServer, grd_vnc_server, G_TYPE_SOCKET_SERVICE)
static void
allow_connection_cb (GrdThrottler *throttler,
GSocketConnection *connection,
gpointer user_data);
GrdContext *
grd_vnc_server_get_context (GrdVncServer *vnc_server)
{
return vnc_server->context;
}
GrdVncServer *
grd_vnc_server_new (GrdContext *context)
{
GrdVncServer *vnc_server;
vnc_server = g_object_new (GRD_TYPE_VNC_SERVER,
"context", context,
NULL);
return vnc_server;
}
static void
grd_vnc_server_cleanup_stopped_sessions (GrdVncServer *vnc_server)
{
g_list_free_full (vnc_server->stopped_sessions, g_object_unref);
vnc_server->stopped_sessions = NULL;
}
static gboolean
cleanup_stopped_sessions_idle (GrdVncServer *vnc_server)
{
grd_vnc_server_cleanup_stopped_sessions (vnc_server);
vnc_server->cleanup_sessions_idle_id = 0;
return G_SOURCE_REMOVE;
}
static void
on_session_stopped (GrdSession *session, GrdVncServer *vnc_server)
{
g_debug ("VNC session stopped");
vnc_server->stopped_sessions = g_list_append (vnc_server->stopped_sessions,
session);
vnc_server->sessions = g_list_remove (vnc_server->sessions, session);
if (!vnc_server->cleanup_sessions_idle_id)
{
vnc_server->cleanup_sessions_idle_id =
g_idle_add ((GSourceFunc) cleanup_stopped_sessions_idle,
vnc_server);
}
}
static void
allow_connection_cb (GrdThrottler *throttler,
GSocketConnection *connection,
gpointer user_data)
{
GrdVncServer *vnc_server = GRD_VNC_SERVER (user_data);
GrdSessionVnc *session_vnc;
g_debug ("Creating new VNC session");
session_vnc = grd_session_vnc_new (vnc_server, connection);
vnc_server->sessions = g_list_append (vnc_server->sessions, session_vnc);
g_signal_connect (session_vnc, "stopped",
G_CALLBACK (on_session_stopped),
vnc_server);
}
static gboolean
on_incoming (GSocketService *service,
GSocketConnection *connection)
{
GrdVncServer *vnc_server = GRD_VNC_SERVER (service);
grd_throttler_handle_connection (vnc_server->throttler,
connection);
return TRUE;
}
gboolean
grd_vnc_server_start (GrdVncServer *vnc_server,
GError **error)
{
GrdSettings *settings = grd_context_get_settings (vnc_server->context);
int vnc_port;
uint16_t selected_vnc_port = 0;
gboolean negotiate_port;
GrdDBusRemoteDesktopVncServer *vnc_server_iface;
g_object_get (G_OBJECT (settings),
"vnc-port", &vnc_port,
"vnc-negotiate-port", &negotiate_port,
NULL);
g_debug ("[VNC] Trying to bind to TCP socket:");
if (!grd_bind_socket (G_SOCKET_LISTENER (vnc_server),
vnc_port,
&selected_vnc_port,
negotiate_port,
error))
return FALSE;
g_assert (selected_vnc_port != 0);
g_signal_connect (vnc_server, "incoming", G_CALLBACK (on_incoming), NULL);
vnc_server_iface = grd_context_get_vnc_server_interface (vnc_server->context);
grd_dbus_remote_desktop_vnc_server_set_enabled (vnc_server_iface, 1);
grd_dbus_remote_desktop_vnc_server_set_port (vnc_server_iface,
selected_vnc_port);
return TRUE;
}
void
grd_vnc_server_stop (GrdVncServer *vnc_server)
{
GrdDBusRemoteDesktopVncServer *vnc_server_iface;
g_socket_service_stop (G_SOCKET_SERVICE (vnc_server));
g_socket_listener_close (G_SOCKET_LISTENER (vnc_server));
vnc_server_iface = grd_context_get_vnc_server_interface (vnc_server->context);
grd_dbus_remote_desktop_vnc_server_set_enabled (vnc_server_iface, FALSE);
grd_dbus_remote_desktop_vnc_server_set_port (vnc_server_iface, -1);
while (vnc_server->sessions)
{
GrdSession *session = vnc_server->sessions->data;
grd_session_stop (session);
}
grd_vnc_server_cleanup_stopped_sessions (vnc_server);
g_clear_handle_id (&vnc_server->cleanup_sessions_idle_id,
g_source_remove);
g_clear_object (&vnc_server->throttler);
}
static void
grd_vnc_server_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
GrdVncServer *vnc_server = GRD_VNC_SERVER (object);
switch (prop_id)
{
case PROP_CONTEXT:
vnc_server->context = g_value_get_object (value);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
static void
grd_vnc_server_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
GrdVncServer *vnc_server = GRD_VNC_SERVER (object);
switch (prop_id)
{
case PROP_CONTEXT:
g_value_set_object (value, vnc_server->context);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
}
}
static void
grd_vnc_server_dispose (GObject *object)
{
GrdVncServer *vnc_server = GRD_VNC_SERVER (object);
g_assert (!vnc_server->sessions);
g_assert (!vnc_server->stopped_sessions);
g_assert (!vnc_server->cleanup_sessions_idle_id);
g_assert (!vnc_server->throttler);
G_OBJECT_CLASS (grd_vnc_server_parent_class)->dispose (object);
}
static void
grd_vnc_server_constructed (GObject *object)
{
GrdVncServer *vnc_server = GRD_VNC_SERVER (object);
GrdThrottlerLimits *limits;
if (grd_get_debug_flags () & GRD_DEBUG_VNC)
rfbLogEnable (1);
else
rfbLogEnable (0);
limits = grd_throttler_limits_new (vnc_server->context);
/* TODO: Add the rfbScreen instance to GrdVncServer to support multiple
* sessions. */
grd_throttler_limits_set_max_global_connections (limits, 1);
vnc_server->throttler = grd_throttler_new (limits,
allow_connection_cb,
vnc_server);
G_OBJECT_CLASS (grd_vnc_server_parent_class)->constructed (object);
}
static void
grd_vnc_server_init (GrdVncServer *vnc_server)
{
}
static void
grd_vnc_server_class_init (GrdVncServerClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
object_class->set_property = grd_vnc_server_set_property;
object_class->get_property = grd_vnc_server_get_property;
object_class->dispose = grd_vnc_server_dispose;
object_class->constructed = grd_vnc_server_constructed;
g_object_class_install_property (object_class,
PROP_CONTEXT,
g_param_spec_object ("context",
"GrdContext",
"The GrdContext instance",
GRD_TYPE_CONTEXT,
G_PARAM_READWRITE |
G_PARAM_CONSTRUCT_ONLY |
G_PARAM_STATIC_STRINGS));
}