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];
47 struct dvcman_channel_list* channels;
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 FRDP_PLUGIN_DATA* plugin_data;
70 typedef struct _DVCMAN_CHANNEL DVCMAN_CHANNEL;
71 struct _DVCMAN_CHANNEL
73 IWTSVirtualChannel iface;
78 IWTSVirtualChannelCallback* channel_callback;
83 struct dvcman_channel_list_item
85 DVCMAN_CHANNEL channel;
88 DEFINE_LIST_TYPE(dvcman_channel_list, dvcman_channel_list_item)
90 static int dvcman_get_configuration(IWTSListener* pListener,
93 *ppPropertyBag = NULL;
97 static int dvcman_create_listener(IWTSVirtualChannelManager* pChannelMgr,
98 const char* pszChannelName,
100 IWTSListenerCallback* pListenerCallback,
101 IWTSListener** ppListener)
103 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
104 DVCMAN_LISTENER* listener;
106 if (dvcman->num_listeners < MAX_PLUGINS)
108 DEBUG_DVC("%d.%s.", dvcman->num_listeners, pszChannelName);
109 listener = xnew(DVCMAN_LISTENER);
110 listener->iface.GetConfiguration = dvcman_get_configuration;
111 listener->dvcman = dvcman;
112 listener->channel_name = xstrdup(pszChannelName);
113 listener->flags = ulFlags;
114 listener->listener_callback = pListenerCallback;
117 *ppListener = (IWTSListener*)listener;
118 dvcman->listeners[dvcman->num_listeners++] = (IWTSListener*)listener;
123 DEBUG_WARN("Maximum DVC listener number reached.");
128 static int dvcman_push_event(IWTSVirtualChannelManager* pChannelMgr,
131 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
134 error = drdynvc_push_event(dvcman->drdynvc, pEvent);
137 DEBUG_DVC("event_type %d pushed.", pEvent->event_type);
141 DEBUG_WARN("event_type %d push failed.", pEvent->event_type);
146 static int dvcman_register_plugin(IDRDYNVC_ENTRY_POINTS* pEntryPoints,
147 const char* name, IWTSPlugin* pPlugin)
149 DVCMAN* dvcman = ((DVCMAN_ENTRY_POINTS*)pEntryPoints)->dvcman;
151 if (dvcman->num_plugins < MAX_PLUGINS)
153 DEBUG_DVC("num_plugins %d", dvcman->num_plugins);
154 dvcman->plugin_names[dvcman->num_plugins] = name;
155 dvcman->plugins[dvcman->num_plugins++] = pPlugin;
160 DEBUG_WARN("Maximum DVC plugin number reached.");
165 IWTSPlugin* dvcman_get_plugin(IDRDYNVC_ENTRY_POINTS* pEntryPoints,
168 DVCMAN* dvcman = ((DVCMAN_ENTRY_POINTS*)pEntryPoints)->dvcman;
171 for (i = 0; i < dvcman->num_plugins; i++)
173 if (dvcman->plugin_names[i] == name ||
174 strcmp(dvcman->plugin_names[i], name) == 0)
176 return dvcman->plugins[i];
182 FRDP_PLUGIN_DATA* dvcman_get_plugin_data(IDRDYNVC_ENTRY_POINTS* pEntryPoints)
184 return ((DVCMAN_ENTRY_POINTS*)pEntryPoints)->plugin_data;
187 IWTSVirtualChannelManager* dvcman_new(drdynvcPlugin* plugin)
191 dvcman = xnew(DVCMAN);
192 dvcman->iface.CreateListener = dvcman_create_listener;
193 dvcman->iface.PushEvent = dvcman_push_event;
194 dvcman->drdynvc = plugin;
195 dvcman->channels = dvcman_channel_list_new();
197 return (IWTSVirtualChannelManager*)dvcman;
200 int dvcman_load_plugin(IWTSVirtualChannelManager* pChannelMgr, FRDP_PLUGIN_DATA* data)
202 DVCMAN_ENTRY_POINTS entryPoints;
203 PDVC_PLUGIN_ENTRY pDVCPluginEntry = NULL;
207 pDVCPluginEntry = freerdp_load_plugin((char*)data->data[0], "DVCPluginEntry");
209 if(pDVCPluginEntry != NULL)
211 entryPoints.iface.RegisterPlugin = dvcman_register_plugin;
212 entryPoints.iface.GetPlugin = dvcman_get_plugin;
213 entryPoints.iface.GetPluginData = dvcman_get_plugin_data;
214 entryPoints.dvcman = (DVCMAN*)pChannelMgr;
215 entryPoints.plugin_data = data;
216 pDVCPluginEntry((IDRDYNVC_ENTRY_POINTS*)&entryPoints);
223 void dvcman_free(IWTSVirtualChannelManager* pChannelMgr)
225 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
228 DVCMAN_LISTENER* listener;
230 dvcman_channel_list_free(dvcman->channels);
231 for (i = 0; i < dvcman->num_listeners; i++)
233 listener = (DVCMAN_LISTENER*)dvcman->listeners[i];
234 xfree(listener->channel_name);
237 for (i = 0; i < dvcman->num_plugins; i++)
239 pPlugin = dvcman->plugins[i];
240 if (pPlugin->Terminated)
241 pPlugin->Terminated(pPlugin);
246 int dvcman_init(IWTSVirtualChannelManager* pChannelMgr)
248 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
252 for (i = 0; i < dvcman->num_plugins; i++)
254 pPlugin = dvcman->plugins[i];
255 if (pPlugin->Initialize)
256 pPlugin->Initialize(pPlugin, pChannelMgr);
261 static int dvcman_write_channel(IWTSVirtualChannel* pChannel,
266 DVCMAN_CHANNEL* channel = (DVCMAN_CHANNEL*)pChannel;
268 return drdynvc_write_data(channel->dvcman->drdynvc, channel->channel_id, pBuffer, cbSize);
271 static void dvcman_channel_list_item_free(struct dvcman_channel_list_item* item)
273 DVCMAN_CHANNEL* channel = (DVCMAN_CHANNEL*)item;
275 if (channel->channel_callback)
276 channel->channel_callback->OnClose(channel->channel_callback);
280 static int dvcman_close_channel_iface(IWTSVirtualChannel* pChannel)
282 DVCMAN_CHANNEL* channel = (DVCMAN_CHANNEL*)pChannel;
283 DVCMAN* dvcman = channel->dvcman;
285 DEBUG_DVC("id=%d", channel->channel_id);
287 if (dvcman_channel_list_remove(dvcman->channels, (struct dvcman_channel_list_item*)channel) == NULL)
288 DEBUG_WARN("channel not found");
289 dvcman_channel_list_item_free((struct dvcman_channel_list_item*)channel);
294 int dvcman_create_channel(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId, const char* ChannelName)
296 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
298 DVCMAN_LISTENER* listener;
299 DVCMAN_CHANNEL* channel;
300 struct dvcman_channel_list_item* item;
302 IWTSVirtualChannelCallback* pCallback;
304 for (i = 0; i < dvcman->num_listeners; i++)
306 listener = (DVCMAN_LISTENER*)dvcman->listeners[i];
307 if (strcmp(listener->channel_name, ChannelName) == 0)
309 item = dvcman_channel_list_item_new();
310 channel = (DVCMAN_CHANNEL*)item;
311 channel->iface.Write = dvcman_write_channel;
312 channel->iface.Close = dvcman_close_channel_iface;
313 channel->dvcman = dvcman;
314 channel->channel_id = ChannelId;
318 if (listener->listener_callback->OnNewChannelConnection(listener->listener_callback,
319 (IWTSVirtualChannel*)channel, NULL, &bAccept, &pCallback) == 0 && bAccept == 1)
321 DEBUG_DVC("listener %s created new channel %d",
322 listener->channel_name, channel->channel_id);
323 channel->channel_callback = pCallback;
324 dvcman_channel_list_add(dvcman->channels, item);
329 DEBUG_WARN("channel rejected by plugin");
330 dvcman_channel_list_item_free(item);
338 static DVCMAN_CHANNEL* dvcman_find_channel_by_id(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId)
340 DVCMAN* dvcman = (DVCMAN*)pChannelMgr;
341 struct dvcman_channel_list_item* curr;
343 for (curr = dvcman->channels->head; curr; curr = dvcman_channel_list_item_next(curr))
345 if (((DVCMAN_CHANNEL*)curr)->channel_id == ChannelId)
347 return (DVCMAN_CHANNEL*)curr;
353 int dvcman_close_channel(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId)
355 DVCMAN_CHANNEL* channel;
356 IWTSVirtualChannel* ichannel;
358 channel = dvcman_find_channel_by_id(pChannelMgr, ChannelId);
361 DEBUG_WARN("ChannelId %d not found!", ChannelId);
364 if (channel->dvc_data)
366 stream_free(channel->dvc_data);
367 channel->dvc_data = NULL;
369 DEBUG_DVC("dvcman_close_channel: channel %d closed", ChannelId);
370 ichannel = (IWTSVirtualChannel*)channel;
371 ichannel->Close(ichannel);
375 int dvcman_receive_channel_data_first(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId, uint32 length)
377 DVCMAN_CHANNEL* channel;
379 channel = dvcman_find_channel_by_id(pChannelMgr, ChannelId);
382 DEBUG_WARN("ChannelId %d not found!", ChannelId);
385 if (channel->dvc_data)
386 stream_free(channel->dvc_data);
387 channel->dvc_data = stream_new(length);
392 int dvcman_receive_channel_data(IWTSVirtualChannelManager* pChannelMgr, uint32 ChannelId, uint8* data, uint32 data_size)
394 DVCMAN_CHANNEL* channel;
397 channel = dvcman_find_channel_by_id(pChannelMgr, ChannelId);
400 DEBUG_WARN("ChannelId %d not found!", ChannelId);
404 if (channel->dvc_data)
406 /* Fragmented data */
407 if (stream_get_length(channel->dvc_data) + data_size > stream_get_size(channel->dvc_data))
409 DEBUG_WARN("data exceeding declared length!");
410 stream_free(channel->dvc_data);
411 channel->dvc_data = NULL;
414 stream_write(channel->dvc_data, data, data_size);
415 if (stream_get_length(channel->dvc_data) >= stream_get_size(channel->dvc_data))
417 error = channel->channel_callback->OnDataReceived(channel->channel_callback,
418 stream_get_size(channel->dvc_data), stream_get_data(channel->dvc_data));
419 stream_free(channel->dvc_data);
420 channel->dvc_data = NULL;
425 error = channel->channel_callback->OnDataReceived(channel->channel_callback,