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,
86 *ppPropertyBag = NULL;
90 static int dvcman_create_listener(IWTSVirtualChannelManager* pChannelMgr,
91 const char* pszChannelName,
93 IWTSListenerCallback* pListenerCallback,
94 IWTSListener** ppListener)
96 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
97 DVCMAN_LISTENER* listener;
99 if (dvcman->num_listeners < MAX_PLUGINS)
101 DEBUG_DVC("%d.%s.", dvcman->num_listeners, pszChannelName);
102 listener = xnew(DVCMAN_LISTENER);
103 listener->iface.GetConfiguration = dvcman_get_configuration;
104 listener->dvcman = dvcman;
105 listener->channel_name = xstrdup(pszChannelName);
106 listener->flags = ulFlags;
107 listener->listener_callback = pListenerCallback;
110 *ppListener = (IWTSListener*)listener;
111 dvcman->listeners[dvcman->num_listeners++] = (IWTSListener*)listener;
116 DEBUG_WARN("Maximum DVC listener number reached.");
121 static int dvcman_push_event(IWTSVirtualChannelManager* pChannelMgr,
124 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
127 error = drdynvc_push_event(dvcman->drdynvc, pEvent);
130 DEBUG_DVC("event_type %d pushed.", pEvent->event_type);
134 DEBUG_WARN("event_type %d push failed.", pEvent->event_type);
139 static int dvcman_register_plugin(IDRDYNVC_ENTRY_POINTS* pEntryPoints,
140 const char* name, IWTSPlugin* pPlugin)
142 DVCMAN* dvcman = ((DVCMAN_ENTRY_POINTS*)pEntryPoints)->dvcman;
144 if (dvcman->num_plugins < MAX_PLUGINS)
146 DEBUG_DVC("num_plugins %d", dvcman->num_plugins);
147 dvcman->plugin_names[dvcman->num_plugins] = name;
148 dvcman->plugins[dvcman->num_plugins++] = pPlugin;
153 DEBUG_WARN("Maximum DVC plugin number reached.");
158 IWTSPlugin* dvcman_get_plugin(IDRDYNVC_ENTRY_POINTS* pEntryPoints,
161 DVCMAN* dvcman = ((DVCMAN_ENTRY_POINTS*)pEntryPoints)->dvcman;
164 for (i = 0; i < dvcman->num_plugins; i++)
166 if (dvcman->plugin_names[i] == name ||
167 strcmp(dvcman->plugin_names[i], name) == 0)
169 return dvcman->plugins[i];
175 RDP_PLUGIN_DATA* dvcman_get_plugin_data(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
177 return ((DVCMAN_ENTRY_POINTS*)pEntryPoints)->plugin_data;
180 IWTSVirtualChannelManager* dvcman_new(drdynvcPlugin* plugin)
184 dvcman = xnew(DVCMAN);
185 dvcman->iface.CreateListener = dvcman_create_listener;
186 dvcman->iface.PushEvent = dvcman_push_event;
187 dvcman->drdynvc = plugin;
188 dvcman->channels = list_new();
190 return (IWTSVirtualChannelManager*)dvcman;
193 int dvcman_load_plugin(IWTSVirtualChannelManager* pChannelMgr, RDP_PLUGIN_DATA* data)
195 DVCMAN_ENTRY_POINTS entryPoints;
196 PDVC_PLUGIN_ENTRY pDVCPluginEntry = NULL;
200 pDVCPluginEntry = freerdp_load_plugin((char*)data->data[0], "DVCPluginEntry");
202 if(pDVCPluginEntry != NULL)
204 entryPoints.iface.RegisterPlugin = dvcman_register_plugin;
205 entryPoints.iface.GetPlugin = dvcman_get_plugin;
206 entryPoints.iface.GetPluginData = dvcman_get_plugin_data;
207 entryPoints.dvcman = (DVCMAN*)pChannelMgr;
208 entryPoints.plugin_data = data;
209 pDVCPluginEntry((IDRDYNVC_ENTRY_POINTS*)&entryPoints);
216 static void dvcman_channel_free(DVCMAN_CHANNEL* channel)
218 if (channel->channel_callback)
219 channel->channel_callback->OnClose(channel->channel_callback);
223 void dvcman_free(IWTSVirtualChannelManager* pChannelMgr)
225 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
228 DVCMAN_LISTENER* listener;
229 DVCMAN_CHANNEL* channel;
231 while ((channel = (DVCMAN_CHANNEL*)list_dequeue(dvcman->channels)) != NULL)
232 dvcman_channel_free(channel);
233 list_free(dvcman->channels);
234 for (i = 0; i < dvcman->num_listeners; i++)
236 listener = (DVCMAN_LISTENER*)dvcman->listeners[i];
237 xfree(listener->channel_name);
240 for (i = 0; i < dvcman->num_plugins; i++)
242 pPlugin = dvcman->plugins[i];
243 if (pPlugin->Terminated)
244 pPlugin->Terminated(pPlugin);
249 int dvcman_init(IWTSVirtualChannelManager* pChannelMgr)
251 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
255 for (i = 0; i < dvcman->num_plugins; i++)
257 pPlugin = dvcman->plugins[i];
258 if (pPlugin->Initialize)
259 pPlugin->Initialize(pPlugin, pChannelMgr);
264 static int dvcman_write_channel(IWTSVirtualChannel* pChannel,
269 DVCMAN_CHANNEL* channel = (DVCMAN_CHANNEL*)pChannel;
271 return drdynvc_write_data(channel->dvcman->drdynvc, channel->channel_id, pBuffer, cbSize);
274 static int dvcman_close_channel_iface(IWTSVirtualChannel* pChannel)
276 DVCMAN_CHANNEL* channel = (DVCMAN_CHANNEL*)pChannel;
277 DVCMAN* dvcman = channel->dvcman;
279 DEBUG_DVC("id=%d", channel->channel_id);
281 if (list_remove(dvcman->channels, channel) == NULL)
282 DEBUG_WARN("channel not found");
283 dvcman_channel_free(channel);
288 int dvcman_create_channel(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId, const char* ChannelName)
290 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
292 DVCMAN_LISTENER* listener;
293 DVCMAN_CHANNEL* channel;
295 IWTSVirtualChannelCallback* pCallback;
297 for (i = 0; i < dvcman->num_listeners; i++)
299 listener = (DVCMAN_LISTENER*)dvcman->listeners[i];
300 if (strcmp(listener->channel_name, ChannelName) == 0)
302 channel = xnew(DVCMAN_CHANNEL);
303 channel->iface.Write = dvcman_write_channel;
304 channel->iface.Close = dvcman_close_channel_iface;
305 channel->dvcman = dvcman;
306 channel->channel_id = ChannelId;
310 if (listener->listener_callback->OnNewChannelConnection(listener->listener_callback,
311 (IWTSVirtualChannel*)channel, NULL, &bAccept, &pCallback) == 0 && bAccept == 1)
313 DEBUG_DVC("listener %s created new channel %d",
314 listener->channel_name, channel->channel_id);
315 channel->channel_callback = pCallback;
316 list_add(dvcman->channels, channel);
321 DEBUG_WARN("channel rejected by plugin");
322 dvcman_channel_free(channel);
330 static DVCMAN_CHANNEL* dvcman_find_channel_by_id(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId)
332 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
335 for (curr = dvcman->channels->head; curr; curr = curr->next)
337 if (((DVCMAN_CHANNEL*)curr->data)->channel_id == ChannelId)
339 return (DVCMAN_CHANNEL*)curr->data;
345 int dvcman_close_channel(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId)
347 DVCMAN_CHANNEL* channel;
348 IWTSVirtualChannel* ichannel;
350 channel = dvcman_find_channel_by_id(pChannelMgr, ChannelId);
353 DEBUG_WARN("ChannelId %d not found!", ChannelId);
356 if (channel->dvc_data)
358 stream_free(channel->dvc_data);
359 channel->dvc_data = NULL;
361 DEBUG_DVC("dvcman_close_channel: channel %d closed", ChannelId);
362 ichannel = (IWTSVirtualChannel*)channel;
363 ichannel->Close(ichannel);
367 int dvcman_receive_channel_data_first(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId, uint32 length)
369 DVCMAN_CHANNEL* channel;
371 channel = dvcman_find_channel_by_id(pChannelMgr, ChannelId);
374 DEBUG_WARN("ChannelId %d not found!", ChannelId);
377 if (channel->dvc_data)
378 stream_free(channel->dvc_data);
379 channel->dvc_data = stream_new(length);
384 int dvcman_receive_channel_data(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId, uint8* data, uint32 data_size)
386 DVCMAN_CHANNEL* channel;
389 channel = dvcman_find_channel_by_id(pChannelMgr, ChannelId);
392 DEBUG_WARN("ChannelId %d not found!", ChannelId);
396 if (channel->dvc_data)
398 /* Fragmented data */
399 if (stream_get_length(channel->dvc_data) + data_size > stream_get_size(channel->dvc_data))
401 DEBUG_WARN("data exceeding declared length!");
402 stream_free(channel->dvc_data);
403 channel->dvc_data = NULL;
406 stream_write(channel->dvc_data, data, data_size);
407 if (stream_get_length(channel->dvc_data) >= stream_get_size(channel->dvc_data))
409 error = channel->channel_callback->OnDataReceived(channel->channel_callback,
410 stream_get_size(channel->dvc_data), stream_get_data(channel->dvc_data));
411 stream_free(channel->dvc_data);
412 channel->dvc_data = NULL;
417 error = channel->channel_callback->OnDataReceived(channel->channel_callback,