2011-08-10 01:42:10 +04:00
|
|
|
/**
|
2012-10-09 07:02:04 +04:00
|
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
2011-08-10 01:42:10 +04:00
|
|
|
* RAIL Virtual Channel Plugin
|
|
|
|
*
|
|
|
|
* Copyright 2011 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
|
|
* Copyright 2011 Roman Barabanov <romanbarabanov@gmail.com>
|
|
|
|
* Copyright 2011 Vic Lee
|
|
|
|
*
|
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
|
|
|
*
|
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
*
|
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2012-08-15 01:09:01 +04:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2011-08-10 01:42:10 +04:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2012-08-15 01:09:01 +04:00
|
|
|
|
2012-10-09 04:26:11 +04:00
|
|
|
#include <winpr/crt.h>
|
|
|
|
|
2011-08-10 01:42:10 +04:00
|
|
|
#include <freerdp/types.h>
|
2012-10-09 04:26:11 +04:00
|
|
|
#include <freerdp/constants.h>
|
2011-08-10 01:42:10 +04:00
|
|
|
#include <freerdp/utils/svc_plugin.h>
|
2011-08-22 03:04:21 +04:00
|
|
|
#include <freerdp/utils/rail.h>
|
2011-08-10 01:42:10 +04:00
|
|
|
#include <freerdp/rail.h>
|
|
|
|
|
2011-08-12 00:27:16 +04:00
|
|
|
#include "rail_orders.h"
|
2011-08-10 01:42:10 +04:00
|
|
|
#include "rail_main.h"
|
|
|
|
|
2011-08-19 16:42:14 +04:00
|
|
|
void rail_send_channel_data(void* rail_object, void* data, size_t length)
|
2011-08-10 01:42:10 +04:00
|
|
|
{
|
2013-03-21 23:19:33 +04:00
|
|
|
wStream* s = NULL;
|
2011-08-12 12:26:57 +04:00
|
|
|
railPlugin* plugin = (railPlugin*) rail_object;
|
2011-08-10 01:42:10 +04:00
|
|
|
|
2013-05-09 01:48:30 +04:00
|
|
|
s = Stream_New(NULL, length);
|
2013-05-09 00:09:16 +04:00
|
|
|
Stream_Write(s, data, length);
|
2011-08-10 01:42:10 +04:00
|
|
|
|
|
|
|
svc_plugin_send((rdpSvcPlugin*) plugin, s);
|
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void on_free_rail_channel_event(wMessage* event)
|
2011-08-10 01:42:10 +04:00
|
|
|
{
|
2013-03-29 05:13:56 +04:00
|
|
|
rail_free_cloned_order(GetMessageType(event->id), event->wParam);
|
2011-08-10 01:42:10 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:01:37 +04:00
|
|
|
void rail_send_channel_event(void* rail_object, UINT16 event_type, void* param)
|
2011-08-10 01:42:10 +04:00
|
|
|
{
|
2011-10-21 22:08:31 +04:00
|
|
|
void * payload = NULL;
|
2013-03-29 05:13:56 +04:00
|
|
|
wMessage* out_event = NULL;
|
2011-08-18 05:33:22 +04:00
|
|
|
railPlugin* plugin = (railPlugin*) rail_object;
|
2011-08-22 03:04:21 +04:00
|
|
|
|
|
|
|
payload = rail_clone_order(event_type, param);
|
2011-10-21 22:08:31 +04:00
|
|
|
|
2011-08-22 03:04:21 +04:00
|
|
|
if (payload != NULL)
|
|
|
|
{
|
2013-03-29 03:33:31 +04:00
|
|
|
out_event = freerdp_event_new(RailChannel_Class, event_type,
|
2011-08-22 03:04:21 +04:00
|
|
|
on_free_rail_channel_event, payload);
|
2011-10-21 22:08:31 +04:00
|
|
|
|
2011-08-22 03:04:21 +04:00
|
|
|
svc_plugin_send_event((rdpSvcPlugin*) plugin, out_event);
|
|
|
|
}
|
2011-08-10 01:42:10 +04:00
|
|
|
}
|
|
|
|
|
2011-08-12 12:26:57 +04:00
|
|
|
static void rail_process_connect(rdpSvcPlugin* plugin)
|
2011-08-10 01:42:10 +04:00
|
|
|
{
|
2011-08-12 12:26:57 +04:00
|
|
|
railPlugin* rail = (railPlugin*) plugin;
|
2011-08-10 01:42:10 +04:00
|
|
|
|
2011-08-12 12:26:57 +04:00
|
|
|
rail->rail_order = rail_order_new();
|
2012-11-19 22:26:56 +04:00
|
|
|
rail->rail_order->settings = (rdpSettings*) plugin->channel_entry_points.pExtendedData;
|
2011-08-19 16:42:14 +04:00
|
|
|
rail->rail_order->plugin = rail;
|
2011-08-10 01:42:10 +04:00
|
|
|
}
|
|
|
|
|
2011-08-12 12:26:57 +04:00
|
|
|
static void rail_process_terminate(rdpSvcPlugin* plugin)
|
2011-08-10 01:42:10 +04:00
|
|
|
{
|
2011-08-15 01:05:34 +04:00
|
|
|
|
2011-08-12 12:26:57 +04:00
|
|
|
}
|
2011-08-10 01:42:10 +04:00
|
|
|
|
2013-03-21 23:19:33 +04:00
|
|
|
static void rail_process_receive(rdpSvcPlugin* plugin, wStream* s)
|
2011-08-12 12:26:57 +04:00
|
|
|
{
|
|
|
|
railPlugin* rail = (railPlugin*) plugin;
|
|
|
|
rail_order_recv(rail->rail_order, s);
|
2013-05-09 01:48:30 +04:00
|
|
|
Stream_Free(s, TRUE);
|
2011-08-12 12:26:57 +04:00
|
|
|
}
|
2011-08-10 01:42:10 +04:00
|
|
|
|
2012-11-19 22:26:56 +04:00
|
|
|
static void rail_process_addin_args(rdpRailOrder* rail_order, rdpSettings* settings)
|
2011-08-21 05:12:17 +04:00
|
|
|
{
|
2011-10-21 22:08:31 +04:00
|
|
|
char* exeOrFile;
|
|
|
|
|
2012-11-19 22:26:56 +04:00
|
|
|
exeOrFile = settings->RemoteApplicationProgram;
|
2011-10-21 22:08:31 +04:00
|
|
|
|
|
|
|
if (strlen(exeOrFile) >= 2)
|
|
|
|
{
|
|
|
|
if (strncmp(exeOrFile, "||", 2) != 0)
|
|
|
|
rail_order->exec.flags |= RAIL_EXEC_FLAG_FILE;
|
|
|
|
}
|
|
|
|
|
2013-08-29 01:47:37 +04:00
|
|
|
rail_string_to_unicode_string(settings->RemoteApplicationProgram, &rail_order->exec.exeOrFile);
|
|
|
|
rail_string_to_unicode_string(settings->ShellWorkingDirectory, &rail_order->exec.workingDir);
|
|
|
|
rail_string_to_unicode_string(settings->RemoteApplicationCmdLine, &rail_order->exec.arguments);
|
2011-08-21 05:12:17 +04:00
|
|
|
|
|
|
|
rail_send_client_exec_order(rail_order);
|
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void rail_recv_set_sysparams_event(rdpRailOrder* rail_order, wMessage* event)
|
2011-08-18 05:33:22 +04:00
|
|
|
{
|
2011-08-21 05:12:17 +04:00
|
|
|
RAIL_SYSPARAM_ORDER* sysparam;
|
2011-08-18 05:33:22 +04:00
|
|
|
|
|
|
|
/* Send System Parameters */
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
sysparam = (RAIL_SYSPARAM_ORDER*) event->wParam;
|
2011-08-21 05:12:17 +04:00
|
|
|
memmove(&rail_order->sysparam, sysparam, sizeof(RAIL_SYSPARAM_ORDER));
|
|
|
|
|
2011-08-18 05:33:22 +04:00
|
|
|
rail_send_client_sysparams_order(rail_order);
|
|
|
|
|
|
|
|
/* execute */
|
|
|
|
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_order->exec.flags = RAIL_EXEC_FLAG_EXPAND_ARGUMENTS;
|
2011-08-18 05:33:22 +04:00
|
|
|
|
2012-11-19 22:26:56 +04:00
|
|
|
rail_process_addin_args(rail_order, rail_order->settings);
|
2011-08-18 05:33:22 +04:00
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void rail_recv_exec_remote_app_event(rdpRailOrder* rail_order, wMessage* event)
|
2011-08-21 05:12:17 +04:00
|
|
|
{
|
2012-11-19 22:26:56 +04:00
|
|
|
/**
|
|
|
|
* TODO: replace event system by an API to allow the execution
|
|
|
|
* of multiple remote apps over the same connection. RAIL is
|
|
|
|
* always built-in, so clients can safely link to it.
|
|
|
|
*/
|
2011-08-21 05:12:17 +04:00
|
|
|
|
2012-11-19 22:26:56 +04:00
|
|
|
//rail_process_addin_args(rail_order, data);
|
2011-08-21 05:12:17 +04:00
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void rail_recv_activate_event(rdpRailOrder* rail_order, wMessage* event)
|
2011-08-21 05:12:17 +04:00
|
|
|
{
|
2013-03-29 05:13:56 +04:00
|
|
|
RAIL_ACTIVATE_ORDER* activate = (RAIL_ACTIVATE_ORDER*) event->wParam;
|
2011-08-21 05:12:17 +04:00
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
CopyMemory(&rail_order->activate, activate, sizeof(RAIL_ACTIVATE_ORDER));
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_send_client_activate_order(rail_order);
|
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void rail_recv_sysmenu_event(rdpRailOrder* rail_order, wMessage* event)
|
2011-08-21 05:12:17 +04:00
|
|
|
{
|
2013-03-29 05:13:56 +04:00
|
|
|
RAIL_SYSMENU_ORDER* sysmenu = (RAIL_SYSMENU_ORDER*) event->wParam;
|
2011-08-21 05:12:17 +04:00
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
CopyMemory(&rail_order->sysmenu, sysmenu, sizeof(RAIL_SYSMENU_ORDER));
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_send_client_sysmenu_order(rail_order);
|
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void rail_recv_syscommand_event(rdpRailOrder* rail_order, wMessage* event)
|
2011-08-21 05:12:17 +04:00
|
|
|
{
|
2013-03-29 05:13:56 +04:00
|
|
|
RAIL_SYSCOMMAND_ORDER* syscommand = (RAIL_SYSCOMMAND_ORDER*) event->wParam;
|
2011-08-21 05:12:17 +04:00
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
CopyMemory(&rail_order->syscommand, syscommand, sizeof(RAIL_SYSCOMMAND_ORDER));
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_send_client_syscommand_order(rail_order);
|
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void rail_recv_notify_event(rdpRailOrder* rail_order, wMessage* event)
|
2011-08-21 05:12:17 +04:00
|
|
|
{
|
2013-03-29 05:13:56 +04:00
|
|
|
RAIL_NOTIFY_EVENT_ORDER* notify = (RAIL_NOTIFY_EVENT_ORDER*) event->wParam;
|
2011-08-21 05:12:17 +04:00
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
CopyMemory(&rail_order->notify_event, notify, sizeof(RAIL_NOTIFY_EVENT_ORDER));
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_send_client_notify_event_order(rail_order);
|
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void rail_recv_window_move_event(rdpRailOrder* rail_order, wMessage* event)
|
2011-08-21 05:12:17 +04:00
|
|
|
{
|
2013-03-29 05:13:56 +04:00
|
|
|
RAIL_WINDOW_MOVE_ORDER* window_move = (RAIL_WINDOW_MOVE_ORDER*) event->wParam;
|
2011-08-21 05:12:17 +04:00
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
CopyMemory(&rail_order->window_move, window_move, sizeof(RAIL_WINDOW_MOVE_ORDER));
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_send_client_window_move_order(rail_order);
|
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void rail_recv_app_req_event(rdpRailOrder* rail_order, wMessage* event)
|
2011-08-21 05:12:17 +04:00
|
|
|
{
|
2013-03-29 05:13:56 +04:00
|
|
|
RAIL_GET_APPID_REQ_ORDER* get_appid_req = (RAIL_GET_APPID_REQ_ORDER*) event->wParam;
|
2011-08-21 05:12:17 +04:00
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
CopyMemory(&rail_order->get_appid_req, get_appid_req, sizeof(RAIL_GET_APPID_REQ_ORDER));
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_send_client_get_appid_req_order(rail_order);
|
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void rail_recv_langbarinfo_event(rdpRailOrder* rail_order, wMessage* event)
|
2011-08-21 05:12:17 +04:00
|
|
|
{
|
2013-03-29 05:13:56 +04:00
|
|
|
RAIL_LANGBAR_INFO_ORDER* langbar_info = (RAIL_LANGBAR_INFO_ORDER*) event->wParam;
|
2011-08-21 05:12:17 +04:00
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
CopyMemory(&rail_order->langbar_info, langbar_info, sizeof(RAIL_LANGBAR_INFO_ORDER));
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_send_client_langbar_info_order(rail_order);
|
|
|
|
}
|
|
|
|
|
2013-03-29 05:13:56 +04:00
|
|
|
static void rail_process_event(rdpSvcPlugin* plugin, wMessage* event)
|
2011-08-12 12:26:57 +04:00
|
|
|
{
|
|
|
|
railPlugin* rail = NULL;
|
2011-08-18 05:33:22 +04:00
|
|
|
rail = (railPlugin*) plugin;
|
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
switch (GetMessageType(event->id))
|
2011-08-18 05:33:22 +04:00
|
|
|
{
|
2013-03-29 04:23:16 +04:00
|
|
|
case RailChannel_ClientSystemParam:
|
2011-08-18 05:33:22 +04:00
|
|
|
rail_recv_set_sysparams_event(rail->rail_order, event);
|
|
|
|
break;
|
2011-08-10 01:42:10 +04:00
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
case RailChannel_ClientExecute:
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_recv_exec_remote_app_event(rail->rail_order, event);
|
|
|
|
break;
|
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
case RailChannel_ClientActivate:
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_recv_activate_event(rail->rail_order, event);
|
|
|
|
break;
|
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
case RailChannel_ClientSystemMenu:
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_recv_sysmenu_event(rail->rail_order, event);
|
|
|
|
break;
|
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
case RailChannel_ClientSystemCommand:
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_recv_syscommand_event(rail->rail_order, event);
|
|
|
|
break;
|
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
case RailChannel_ClientNotifyEvent:
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_recv_notify_event(rail->rail_order, event);
|
|
|
|
break;
|
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
case RailChannel_ClientWindowMove:
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_recv_window_move_event(rail->rail_order, event);
|
|
|
|
break;
|
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
case RailChannel_ClientGetAppIdRequest:
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_recv_app_req_event(rail->rail_order, event);
|
|
|
|
break;
|
|
|
|
|
2013-03-29 04:23:16 +04:00
|
|
|
case RailChannel_ClientLanguageBarInfo:
|
2011-08-21 05:12:17 +04:00
|
|
|
rail_recv_langbarinfo_event(rail->rail_order, event);
|
|
|
|
break;
|
|
|
|
|
2011-08-18 05:33:22 +04:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2011-08-10 01:42:10 +04:00
|
|
|
|
2011-08-12 12:26:57 +04:00
|
|
|
freerdp_event_free(event);
|
2011-08-10 01:42:10 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 06:48:17 +04:00
|
|
|
/* rail is always built-in */
|
2012-10-09 04:26:11 +04:00
|
|
|
#define VirtualChannelEntry rail_VirtualChannelEntry
|
|
|
|
|
2012-11-01 07:04:31 +04:00
|
|
|
int VirtualChannelEntry(PCHANNEL_ENTRY_POINTS pEntryPoints)
|
2012-10-09 04:26:11 +04:00
|
|
|
{
|
|
|
|
railPlugin* _p;
|
|
|
|
|
|
|
|
_p = (railPlugin*) malloc(sizeof(railPlugin));
|
|
|
|
ZeroMemory(_p, sizeof(railPlugin));
|
|
|
|
|
|
|
|
_p->plugin.channel_def.options =
|
|
|
|
CHANNEL_OPTION_INITIALIZED |
|
|
|
|
CHANNEL_OPTION_ENCRYPT_RDP |
|
|
|
|
CHANNEL_OPTION_COMPRESS_RDP |
|
|
|
|
CHANNEL_OPTION_SHOW_PROTOCOL;
|
|
|
|
|
|
|
|
strcpy(_p->plugin.channel_def.name, "rail");
|
|
|
|
|
|
|
|
_p->plugin.connect_callback = rail_process_connect;
|
|
|
|
_p->plugin.receive_callback = rail_process_receive;
|
|
|
|
_p->plugin.event_callback = rail_process_event;
|
|
|
|
_p->plugin.terminate_callback = rail_process_terminate;
|
|
|
|
|
|
|
|
svc_plugin_init((rdpSvcPlugin*) _p, pEntryPoints);
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|