2 * FreeRDP: A Remote Desktop Protocol client.
3 * Dynamic Virtual Channel Manager
5 * Copyright 2010-2011 Vic Lee
7 * Licensed under the Apache License, Version 2.0 (the "License");
8 * you may not use this file except in compliance with the License.
9 * You may obtain a copy of the License at
11 * http://www.apache.org/licenses/LICENSE-2.0
13 * Unless required by applicable law or agreed to in writing, software
14 * distributed under the License is distributed on an "AS IS" BASIS,
15 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16 * See the License for the specific language governing permissions and
17 * limitations under the License.
23 #include <freerdp/utils/memory.h>
24 #include <freerdp/utils/stream.h>
25 #include <freerdp/utils/list.h>
26 #include <freerdp/utils/load_plugin.h>
28 #include "drdynvc_types.h"
31 #define MAX_PLUGINS 10
33 typedef struct _DVCMAN DVCMAN;
36 IWTSVirtualChannelManager iface;
38 drdynvcPlugin* drdynvc;
40 const char* plugin_names[MAX_PLUGINS];
41 IWTSPlugin* plugins[MAX_PLUGINS];
44 IWTSListener* listeners[MAX_PLUGINS];
50 typedef struct _DVCMAN_LISTENER DVCMAN_LISTENER;
51 struct _DVCMAN_LISTENER
58 IWTSListenerCallback* listener_callback;
61 typedef struct _DVCMAN_ENTRY_POINTS DVCMAN_ENTRY_POINTS;
62 struct _DVCMAN_ENTRY_POINTS
64 IDRDYNVC_ENTRY_POINTS iface;
67 RDP_PLUGIN_DATA* plugin_data;
70 typedef struct _DVCMAN_CHANNEL DVCMAN_CHANNEL;
71 struct _DVCMAN_CHANNEL
73 IWTSVirtualChannel iface;
78 IWTSVirtualChannelCallback* channel_callback;
83 static int dvcman_get_configuration(IWTSListener* pListener, void** ppPropertyBag)
85 *ppPropertyBag = NULL;
89 static int dvcman_create_listener(IWTSVirtualChannelManager* pChannelMgr,
90 const char* pszChannelName, uint32 ulFlags,
91 IWTSListenerCallback* pListenerCallback, IWTSListener** ppListener)
93 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
94 DVCMAN_LISTENER* listener;
96 if (dvcman->num_listeners < MAX_PLUGINS)
98 DEBUG_DVC("%d.%s.", dvcman->num_listeners, pszChannelName);
99 listener = xnew(DVCMAN_LISTENER);
100 listener->iface.GetConfiguration = dvcman_get_configuration;
101 listener->dvcman = dvcman;
102 listener->channel_name = xstrdup(pszChannelName);
103 listener->flags = ulFlags;
104 listener->listener_callback = pListenerCallback;
107 *ppListener = (IWTSListener*)listener;
108 dvcman->listeners[dvcman->num_listeners++] = (IWTSListener*)listener;
113 DEBUG_WARN("Maximum DVC listener number reached.");
118 static int dvcman_push_event(IWTSVirtualChannelManager* pChannelMgr, RDP_EVENT* pEvent)
121 DVCMAN* dvcman = (DVCMAN*) pChannelMgr;
123 error = drdynvc_push_event(dvcman->drdynvc, pEvent);
127 DEBUG_DVC("event_type %d pushed.", pEvent->event_type);
131 DEBUG_WARN("event_type %d push failed.", pEvent->event_type);
137 static int dvcman_register_plugin(IDRDYNVC_ENTRY_POINTS* pEntryPoints, const char* name, IWTSPlugin* pPlugin)
139 DVCMAN* dvcman = ((DVCMAN_ENTRY_POINTS*) pEntryPoints)->dvcman;
141 if (dvcman->num_plugins < MAX_PLUGINS)
143 DEBUG_DVC("num_plugins %d", dvcman->num_plugins);
144 dvcman->plugin_names[dvcman->num_plugins] = name;
145 dvcman->plugins[dvcman->num_plugins++] = pPlugin;
150 DEBUG_WARN("Maximum DVC plugin number reached.");
155 IWTSPlugin* dvcman_get_plugin(IDRDYNVC_ENTRY_POINTS* pEntryPoints, const char* name)
158 DVCMAN* dvcman = ((DVCMAN_ENTRY_POINTS*) pEntryPoints)->dvcman;
160 for (i = 0; i < dvcman->num_plugins; i++)
162 if (dvcman->plugin_names[i] == name ||
163 strcmp(dvcman->plugin_names[i], name) == 0)
165 return dvcman->plugins[i];
172 RDP_PLUGIN_DATA* dvcman_get_plugin_data(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
174 return ((DVCMAN_ENTRY_POINTS*) pEntryPoints)->plugin_data;
177 IWTSVirtualChannelManager* dvcman_new(drdynvcPlugin* plugin)
181 dvcman = xnew(DVCMAN);
182 dvcman->iface.CreateListener = dvcman_create_listener;
183 dvcman->iface.PushEvent = dvcman_push_event;
184 dvcman->drdynvc = plugin;
185 dvcman->channels = list_new();
187 return (IWTSVirtualChannelManager*) dvcman;
190 int dvcman_load_plugin(IWTSVirtualChannelManager* pChannelMgr, RDP_PLUGIN_DATA* data)
192 DVCMAN_ENTRY_POINTS entryPoints;
193 PDVC_PLUGIN_ENTRY pDVCPluginEntry = NULL;
195 while (data && data->size > 0)
197 pDVCPluginEntry = freerdp_load_plugin((char*) data->data[0], "DVCPluginEntry");
199 if (pDVCPluginEntry != NULL)
201 entryPoints.iface.RegisterPlugin = dvcman_register_plugin;
202 entryPoints.iface.GetPlugin = dvcman_get_plugin;
203 entryPoints.iface.GetPluginData = dvcman_get_plugin_data;
204 entryPoints.dvcman = (DVCMAN*) pChannelMgr;
205 entryPoints.plugin_data = data;
206 pDVCPluginEntry((IDRDYNVC_ENTRY_POINTS*) &entryPoints);
209 data = (RDP_PLUGIN_DATA*)(((void*) data) + data->size);
215 static void dvcman_channel_free(DVCMAN_CHANNEL* channel)
217 if (channel->channel_callback)
218 channel->channel_callback->OnClose(channel->channel_callback);
223 void dvcman_free(IWTSVirtualChannelManager* pChannelMgr)
227 DVCMAN_LISTENER* listener;
228 DVCMAN_CHANNEL* channel;
229 DVCMAN* dvcman = (DVCMAN*) pChannelMgr;
231 while ((channel = (DVCMAN_CHANNEL*) list_dequeue(dvcman->channels)) != NULL)
232 dvcman_channel_free(channel);
234 list_free(dvcman->channels);
236 for (i = 0; i < dvcman->num_listeners; i++)
238 listener = (DVCMAN_LISTENER*) dvcman->listeners[i];
239 xfree(listener->channel_name);
243 for (i = 0; i < dvcman->num_plugins; i++)
245 pPlugin = dvcman->plugins[i];
247 if (pPlugin->Terminated)
248 pPlugin->Terminated(pPlugin);
254 int dvcman_init(IWTSVirtualChannelManager* pChannelMgr)
258 DVCMAN* dvcman = (DVCMAN*) pChannelMgr;
260 for (i = 0; i < dvcman->num_plugins; i++)
262 pPlugin = dvcman->plugins[i];
264 if (pPlugin->Initialize)
265 pPlugin->Initialize(pPlugin, pChannelMgr);
271 static int dvcman_write_channel(IWTSVirtualChannel* pChannel, uint32 cbSize, uint8* pBuffer, void* pReserved)
273 DVCMAN_CHANNEL* channel = (DVCMAN_CHANNEL*) pChannel;
275 return drdynvc_write_data(channel->dvcman->drdynvc, channel->channel_id, pBuffer, cbSize);
278 static int dvcman_close_channel_iface(IWTSVirtualChannel* pChannel)
280 DVCMAN_CHANNEL* channel = (DVCMAN_CHANNEL*) pChannel;
281 DVCMAN* dvcman = channel->dvcman;
283 DEBUG_DVC("id=%d", channel->channel_id);
285 if (list_remove(dvcman->channels, channel) == NULL)
286 DEBUG_WARN("channel not found");
288 dvcman_channel_free(channel);
293 int dvcman_create_channel(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId, const char* ChannelName)
297 DVCMAN_LISTENER* listener;
298 DVCMAN_CHANNEL* channel;
299 IWTSVirtualChannelCallback* pCallback;
300 DVCMAN* dvcman = (DVCMAN*) pChannelMgr;
302 for (i = 0; i < dvcman->num_listeners; i++)
304 listener = (DVCMAN_LISTENER*)dvcman->listeners[i];
306 if (strcmp(listener->channel_name, ChannelName) == 0)
308 channel = xnew(DVCMAN_CHANNEL);
309 channel->iface.Write = dvcman_write_channel;
310 channel->iface.Close = dvcman_close_channel_iface;
311 channel->dvcman = dvcman;
312 channel->channel_id = ChannelId;
317 if (listener->listener_callback->OnNewChannelConnection(listener->listener_callback,
318 (IWTSVirtualChannel*) channel, NULL, &bAccept, &pCallback) == 0 && bAccept == 1)
320 DEBUG_DVC("listener %s created new channel %d",
321 listener->channel_name, channel->channel_id);
322 channel->channel_callback = pCallback;
323 list_add(dvcman->channels, channel);
329 DEBUG_WARN("channel rejected by plugin");
330 dvcman_channel_free(channel);
339 static DVCMAN_CHANNEL* dvcman_find_channel_by_id(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId)
342 DVCMAN* dvcman = (DVCMAN*) pChannelMgr;
344 for (curr = dvcman->channels->head; curr; curr = curr->next)
346 if (((DVCMAN_CHANNEL*) curr->data)->channel_id == ChannelId)
348 return (DVCMAN_CHANNEL*)curr->data;
355 int dvcman_close_channel(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId)
357 DVCMAN_CHANNEL* channel;
358 IWTSVirtualChannel* ichannel;
360 channel = dvcman_find_channel_by_id(pChannelMgr, ChannelId);
364 DEBUG_WARN("ChannelId %d not found!", ChannelId);
368 if (channel->dvc_data)
370 stream_free(channel->dvc_data);
371 channel->dvc_data = NULL;
374 DEBUG_DVC("dvcman_close_channel: channel %d closed", ChannelId);
375 ichannel = (IWTSVirtualChannel*)channel;
376 ichannel->Close(ichannel);
381 int dvcman_receive_channel_data_first(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId, uint32 length)
383 DVCMAN_CHANNEL* channel;
385 channel = dvcman_find_channel_by_id(pChannelMgr, ChannelId);
389 DEBUG_WARN("ChannelId %d not found!", ChannelId);
393 if (channel->dvc_data)
394 stream_free(channel->dvc_data);
396 channel->dvc_data = stream_new(length);
401 int dvcman_receive_channel_data(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId, uint8* data, uint32 data_size)
404 DVCMAN_CHANNEL* channel;
406 channel = dvcman_find_channel_by_id(pChannelMgr, ChannelId);
410 DEBUG_WARN("ChannelId %d not found!", ChannelId);
414 if (channel->dvc_data)
416 /* Fragmented data */
417 if (stream_get_length(channel->dvc_data) + data_size > stream_get_size(channel->dvc_data))
419 DEBUG_WARN("data exceeding declared length!");
420 stream_free(channel->dvc_data);
421 channel->dvc_data = NULL;
425 stream_write(channel->dvc_data, data, data_size);
427 if (stream_get_length(channel->dvc_data) >= stream_get_size(channel->dvc_data))
429 error = channel->channel_callback->OnDataReceived(channel->channel_callback,
430 stream_get_size(channel->dvc_data), stream_get_data(channel->dvc_data));
431 stream_free(channel->dvc_data);
432 channel->dvc_data = NULL;
437 error = channel->channel_callback->OnDataReceived(channel->channel_callback, data_size, data);