Files
grd/grd-rdp-dvc-display-control.c
2026-02-13 13:06:50 +09:00

220 lines
7.3 KiB
C

/*
* Copyright (C) 2021-2023 Pascal Nowack
*
* 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.
*/
#include "config.h"
#include "grd-rdp-dvc-display-control.h"
#include <freerdp/server/disp.h>
#include "grd-rdp-layout-manager.h"
#include "grd-rdp-monitor-config.h"
#include "grd-session-rdp.h"
struct _GrdRdpDvcDisplayControl
{
GrdRdpDvc parent;
DispServerContext *disp_context;
gboolean channel_opened;
gboolean pending_caps;
GrdRdpLayoutManager *layout_manager;
GrdSessionRdp *session_rdp;
};
G_DEFINE_TYPE (GrdRdpDvcDisplayControl, grd_rdp_dvc_display_control,
GRD_TYPE_RDP_DVC)
static void
grd_rdp_dvc_display_control_maybe_init (GrdRdpDvc *dvc)
{
GrdRdpDvcDisplayControl *display_control = GRD_RDP_DVC_DISPLAY_CONTROL (dvc);
DispServerContext *disp_context;
if (display_control->channel_opened)
return;
disp_context = display_control->disp_context;
if (disp_context->Open (disp_context))
{
g_warning ("[RDP.DISP] Failed to open channel. Terminating protocol");
grd_rdp_dvc_queue_channel_tear_down (GRD_RDP_DVC (display_control));
return;
}
display_control->channel_opened = TRUE;
}
static void
dvc_creation_status (gpointer user_data,
int32_t creation_status)
{
GrdRdpDvcDisplayControl *display_control = user_data;
DispServerContext *disp_context = display_control->disp_context;
if (creation_status < 0)
{
g_debug ("[RDP.DISP] Failed to open channel (CreationStatus %i). "
"Terminating protocol", creation_status);
grd_rdp_dvc_queue_channel_tear_down (GRD_RDP_DVC (display_control));
return;
}
g_debug ("[RDP.DISP] Channel opened successfully. "
"Ready for receiving new monitor layouts");
display_control->pending_caps = FALSE;
disp_context->DisplayControlCaps (disp_context);
}
static BOOL
disp_channel_id_assigned (DispServerContext *disp_context,
uint32_t channel_id)
{
GrdRdpDvcDisplayControl *display_control = disp_context->custom;
GrdRdpDvc *dvc = GRD_RDP_DVC (display_control);
g_debug ("[RDP.DISP] DVC channel id assigned to id %u", channel_id);
grd_rdp_dvc_subscribe_creation_status (dvc, channel_id,
dvc_creation_status,
display_control);
return TRUE;
}
static uint32_t
disp_monitor_layout (DispServerContext *disp_context,
const DISPLAY_CONTROL_MONITOR_LAYOUT_PDU *monitor_layout_pdu)
{
GrdRdpDvcDisplayControl *display_control = disp_context->custom;
GrdSessionRdp *session_rdp = display_control->session_rdp;
GrdRdpLayoutManager *layout_manager = display_control->layout_manager;
GrdRdpMonitorConfig *monitor_config;
g_autoptr (GError) error = NULL;
if (display_control->pending_caps)
{
g_warning ("[RDP.DISP] Protocol violation: Received monitor layout PDU "
"before being able to send capabilities. Terminating session");
grd_session_rdp_notify_error (session_rdp,
GRD_SESSION_RDP_ERROR_BAD_MONITOR_DATA);
return CHANNEL_RC_OK;
}
if (monitor_layout_pdu->NumMonitors > disp_context->MaxNumMonitors)
{
g_warning ("[RDP.DISP] Protocol violation: Invalid monitor layout: "
"(exceeded maximum monitor count). Terminating session");
grd_session_rdp_notify_error (session_rdp,
GRD_SESSION_RDP_ERROR_BAD_MONITOR_DATA);
return CHANNEL_RC_OK;
}
monitor_config =
grd_rdp_monitor_config_new_from_disp_monitor_layout (monitor_layout_pdu,
&error);
if (!monitor_config)
{
g_warning ("[RDP.DISP] Received invalid monitor layout from client: %s. "
"Terminating session", error->message);
grd_session_rdp_notify_error (session_rdp,
GRD_SESSION_RDP_ERROR_BAD_MONITOR_DATA);
return CHANNEL_RC_OK;
}
g_debug ("[RDP.DISP] Received new monitor layout. PDU contains %u monitor%s",
monitor_layout_pdu->NumMonitors,
monitor_layout_pdu->NumMonitors == 1 ? "" : "s");
grd_rdp_layout_manager_submit_new_monitor_config (layout_manager,
monitor_config);
return CHANNEL_RC_OK;
}
GrdRdpDvcDisplayControl *
grd_rdp_dvc_display_control_new (GrdRdpLayoutManager *layout_manager,
GrdSessionRdp *session_rdp,
GrdRdpDvcHandler *dvc_handler,
HANDLE vcm,
uint32_t max_monitor_count)
{
GrdRdpDvcDisplayControl *display_control;
DispServerContext *disp_context;
display_control = g_object_new (GRD_TYPE_RDP_DVC_DISPLAY_CONTROL, NULL);
disp_context = disp_server_context_new (vcm);
if (!disp_context)
g_error ("[RDP.DISP] Failed to create server context");
display_control->disp_context = disp_context;
display_control->layout_manager = layout_manager;
display_control->session_rdp = session_rdp;
grd_rdp_dvc_initialize_base (GRD_RDP_DVC (display_control),
dvc_handler, session_rdp,
GRD_RDP_CHANNEL_DISPLAY_CONTROL);
disp_context->MaxNumMonitors = max_monitor_count;
disp_context->MaxMonitorAreaFactorA = 8192;
disp_context->MaxMonitorAreaFactorB = 8192;
disp_context->ChannelIdAssigned = disp_channel_id_assigned;
disp_context->DispMonitorLayout = disp_monitor_layout;
disp_context->custom = display_control;
return display_control;
}
static void
grd_rdp_dvc_display_control_dispose (GObject *object)
{
GrdRdpDvcDisplayControl *display_control =
GRD_RDP_DVC_DISPLAY_CONTROL (object);
GrdRdpDvc *dvc = GRD_RDP_DVC (display_control);
if (display_control->channel_opened)
{
display_control->disp_context->Close (display_control->disp_context);
display_control->channel_opened = FALSE;
}
grd_rdp_dvc_maybe_unsubscribe_creation_status (dvc);
g_clear_pointer (&display_control->disp_context, disp_server_context_free);
G_OBJECT_CLASS (grd_rdp_dvc_display_control_parent_class)->dispose (object);
}
static void
grd_rdp_dvc_display_control_init (GrdRdpDvcDisplayControl *display_control)
{
display_control->pending_caps = TRUE;
}
static void
grd_rdp_dvc_display_control_class_init (GrdRdpDvcDisplayControlClass *klass)
{
GObjectClass *object_class = G_OBJECT_CLASS (klass);
GrdRdpDvcClass *dvc_class = GRD_RDP_DVC_CLASS (klass);
object_class->dispose = grd_rdp_dvc_display_control_dispose;
dvc_class->maybe_init = grd_rdp_dvc_display_control_maybe_init;
}