Blame libfreerdp/core/client.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * Client Channels
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2014 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit 1fb8d4
 * Copyright 2015 Thincast Technologies GmbH
Packit 1fb8d4
 * Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.com>
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_CONFIG_H
Packit 1fb8d4
#include "config.h"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/log.h>
Packit 1fb8d4
Packit 1fb8d4
#include "rdp.h"
Packit 1fb8d4
Packit 1fb8d4
#include "client.h"
Packit 1fb8d4
Packit 1fb8d4
#define TAG FREERDP_TAG("core.client")
Packit 1fb8d4
Packit 1fb8d4
static WINPR_TLS void* g_pInterface = NULL;
Packit 1fb8d4
static WINPR_TLS rdpChannels* g_channels = NULL; /* use only for VirtualChannelInit hack */
Packit 1fb8d4
Packit 1fb8d4
static volatile LONG g_OpenHandleSeq =
Packit 1fb8d4
    1; /* use global counter to ensure uniqueness across channel manager instances */
Packit 1fb8d4
static WINPR_TLS rdpChannelHandles g_ChannelHandles = { NULL, NULL };
Packit 1fb8d4
Packit Service 5a9772
static BOOL freerdp_channels_process_message_free(wMessage* message, DWORD type);
Packit Service 5a9772
Packit Service 5a9772
static CHANNEL_OPEN_DATA* freerdp_channels_find_channel_open_data_by_name(rdpChannels* channels,
Packit Service 5a9772
                                                                          const char* name)
Packit 1fb8d4
{
Packit 1fb8d4
	int index;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < channels->openDataCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		pChannelOpenData = &channels->openDataList[index];
Packit 1fb8d4
Packit 1fb8d4
		if (strncmp(name, pChannelOpenData->name, CHANNEL_NAME_LEN) == 0)
Packit 1fb8d4
			return pChannelOpenData;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/* returns rdpChannel for the channel name passed in */
Packit Service 5a9772
static rdpMcsChannel* freerdp_channels_find_channel_by_name(rdpRdp* rdp, const char* name)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 index;
Packit Service 5a9772
	rdpMcsChannel* channel = NULL;
Packit Service 5a9772
	rdpMcs* mcs = NULL;
Packit Service 5a9772
Packit Service 5a9772
	if (!rdp)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	mcs = rdp->mcs;
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < mcs->channelCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		channel = &mcs->channels[index];
Packit 1fb8d4
Packit 1fb8d4
		if (strncmp(name, channel->Name, CHANNEL_NAME_LEN) == 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			return channel;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static rdpMcsChannel* freerdp_channels_find_channel_by_id(rdpRdp* rdp, UINT16 channel_id)
Packit 1fb8d4
{
Packit Service 5a9772
	UINT32 index;
Packit Service 5a9772
	rdpMcsChannel* channel = NULL;
Packit Service 5a9772
	rdpMcs* mcs = NULL;
Packit 1fb8d4
Packit Service 5a9772
	if (!rdp)
Packit Service 5a9772
		return NULL;
Packit 1fb8d4
Packit Service 5a9772
	mcs = rdp->mcs;
Packit 1fb8d4
Packit Service 5a9772
	for (index = 0; index < mcs->channelCount; index++)
Packit 1fb8d4
	{
Packit Service 5a9772
		channel = &mcs->channels[index];
Packit Service 5a9772
Packit Service 5a9772
		if (channel->ChannelId == channel_id)
Packit 1fb8d4
		{
Packit Service 5a9772
			return channel;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit Service 5a9772
Packit Service 5a9772
	return NULL;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
static void channel_queue_message_free(wMessage* msg)
Packit Service 5a9772
{
Packit Service 5a9772
	CHANNEL_OPEN_EVENT* ev;
Packit Service 5a9772
Packit Service 5a9772
	if (!msg || (msg->id != 0))
Packit Service 5a9772
		return;
Packit Service 5a9772
Packit Service 5a9772
	ev = (CHANNEL_OPEN_EVENT*)msg->wParam;
Packit Service 5a9772
	free(ev);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static void channel_queue_free(void* obj)
Packit 1fb8d4
{
Packit 1fb8d4
	wMessage* msg = (wMessage*)obj;
Packit Service 5a9772
	freerdp_channels_process_message_free(msg, CHANNEL_EVENT_WRITE_CANCELLED);
Packit 1fb8d4
	channel_queue_message_free(msg);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
rdpChannels* freerdp_channels_new(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpChannels* channels;
Packit Service 5a9772
	channels = (rdpChannels*)calloc(1, sizeof(rdpChannels));
Packit 1fb8d4
Packit 1fb8d4
	if (!channels)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	if (!InitializeCriticalSectionAndSpinCount(&channels->channelsLock, 4000))
Packit 1fb8d4
		goto error;
Packit 1fb8d4
Packit 1fb8d4
	channels->instance = instance;
Packit 1fb8d4
	channels->queue = MessageQueue_New(NULL);
Packit 1fb8d4
Packit 1fb8d4
	if (!channels->queue)
Packit 1fb8d4
		goto error;
Packit 1fb8d4
Packit 1fb8d4
	channels->queue->object.fnObjectFree = channel_queue_free;
Packit 1fb8d4
	channels->openHandles = HashTable_New(TRUE);
Packit 1fb8d4
Packit 1fb8d4
	if (!channels->openHandles)
Packit 1fb8d4
		goto error;
Packit 1fb8d4
Packit 1fb8d4
	return channels;
Packit 1fb8d4
error:
Packit 1fb8d4
	freerdp_channels_free(channels);
Packit 1fb8d4
	return NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void freerdp_channels_free(rdpChannels* channels)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!channels)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	DeleteCriticalSection(&channels->channelsLock);
Packit 1fb8d4
Packit 1fb8d4
	if (channels->queue)
Packit 1fb8d4
	{
Packit 1fb8d4
		MessageQueue_Free(channels->queue);
Packit 1fb8d4
		channels->queue = NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (channels->openHandles)
Packit 1fb8d4
		HashTable_Free(channels->openHandles);
Packit 1fb8d4
Packit 1fb8d4
	free(channels);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * Function description
Packit 1fb8d4
 *
Packit 1fb8d4
 * @return 0 on success, otherwise a Win32 error code
Packit 1fb8d4
 */
Packit Service 5a9772
static UINT freerdp_drdynvc_on_channel_connected(DrdynvcClientContext* context, const char* name,
Packit Service 5a9772
                                                 void* pInterface)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT status = CHANNEL_RC_OK;
Packit 1fb8d4
	ChannelConnectedEventArgs e;
Packit Service 5a9772
	rdpChannels* channels = (rdpChannels*)context->custom;
Packit 1fb8d4
	freerdp* instance = channels->instance;
Packit 1fb8d4
	EventArgsInit(&e, "freerdp");
Packit 1fb8d4
	e.name = name;
Packit 1fb8d4
	e.pInterface = pInterface;
Packit 1fb8d4
	PubSub_OnChannelConnected(instance->context->pubSub, instance->context, &e);
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * Function description
Packit 1fb8d4
 *
Packit 1fb8d4
 * @return 0 on success, otherwise a Win32 error code
Packit 1fb8d4
 */
Packit Service 5a9772
static UINT freerdp_drdynvc_on_channel_disconnected(DrdynvcClientContext* context, const char* name,
Packit Service 5a9772
                                                    void* pInterface)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT status = CHANNEL_RC_OK;
Packit 1fb8d4
	ChannelDisconnectedEventArgs e;
Packit Service 5a9772
	rdpChannels* channels = (rdpChannels*)context->custom;
Packit 1fb8d4
	freerdp* instance = channels->instance;
Packit 1fb8d4
	EventArgsInit(&e, "freerdp");
Packit 1fb8d4
	e.name = name;
Packit 1fb8d4
	e.pInterface = pInterface;
Packit 1fb8d4
	PubSub_OnChannelDisconnected(instance->context->pubSub, instance->context, &e);
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static UINT freerdp_drdynvc_on_channel_attached(DrdynvcClientContext* context, const char* name,
Packit Service 5a9772
                                                void* pInterface)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT status = CHANNEL_RC_OK;
Packit 1fb8d4
	ChannelAttachedEventArgs e;
Packit Service 5a9772
	rdpChannels* channels = (rdpChannels*)context->custom;
Packit 1fb8d4
	freerdp* instance = channels->instance;
Packit 1fb8d4
	EventArgsInit(&e, "freerdp");
Packit 1fb8d4
	e.name = name;
Packit 1fb8d4
	e.pInterface = pInterface;
Packit 1fb8d4
	PubSub_OnChannelAttached(instance->context->pubSub, instance->context, &e);
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static UINT freerdp_drdynvc_on_channel_detached(DrdynvcClientContext* context, const char* name,
Packit Service 5a9772
                                                void* pInterface)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT status = CHANNEL_RC_OK;
Packit 1fb8d4
	ChannelDetachedEventArgs e;
Packit Service 5a9772
	rdpChannels* channels = (rdpChannels*)context->custom;
Packit 1fb8d4
	freerdp* instance = channels->instance;
Packit 1fb8d4
	EventArgsInit(&e, "freerdp");
Packit 1fb8d4
	e.name = name;
Packit 1fb8d4
	e.pInterface = pInterface;
Packit 1fb8d4
	PubSub_OnChannelDetached(instance->context->pubSub, instance->context, &e);
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * go through and inform all the libraries that we are initialized
Packit 1fb8d4
 * called only from main thread
Packit 1fb8d4
 */
Packit 1fb8d4
UINT freerdp_channels_pre_connect(rdpChannels* channels, freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT error = CHANNEL_RC_OK;
Packit 1fb8d4
	int index;
Packit 1fb8d4
	CHANNEL_CLIENT_DATA* pChannelClientData;
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < channels->clientDataCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		pChannelClientData = &channels->clientDataList[index];
Packit 1fb8d4
Packit 1fb8d4
		if (pChannelClientData->pChannelInitEventProc)
Packit 1fb8d4
		{
Packit Service 5a9772
			pChannelClientData->pChannelInitEventProc(pChannelClientData->pInitHandle,
Packit Service 5a9772
			                                          CHANNEL_EVENT_INITIALIZED, 0, 0);
Packit 1fb8d4
		}
Packit 1fb8d4
		else if (pChannelClientData->pChannelInitEventProcEx)
Packit 1fb8d4
		{
Packit 1fb8d4
			pChannelClientData->pChannelInitEventProcEx(pChannelClientData->lpUserParam,
Packit Service 5a9772
			                                            pChannelClientData->pInitHandle,
Packit Service 5a9772
			                                            CHANNEL_EVENT_INITIALIZED, 0, 0);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (CHANNEL_RC_OK != getChannelError(instance->context))
Packit 1fb8d4
			break;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return error;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
UINT freerdp_channels_attach(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT error = CHANNEL_RC_OK;
Packit 1fb8d4
	int index;
Packit 1fb8d4
	char* hostname;
Packit Service 5a9772
	size_t hostnameLength;
Packit 1fb8d4
	rdpChannels* channels;
Packit 1fb8d4
	CHANNEL_CLIENT_DATA* pChannelClientData;
Packit 1fb8d4
	channels = instance->context->channels;
Packit 1fb8d4
	hostname = instance->settings->ServerHostname;
Packit Service 5a9772
	hostnameLength = strlen(hostname);
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < channels->clientDataCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		ChannelAttachedEventArgs e;
Packit 1fb8d4
		CHANNEL_OPEN_DATA* pChannelOpenData = NULL;
Packit 1fb8d4
		pChannelClientData = &channels->clientDataList[index];
Packit 1fb8d4
Packit 1fb8d4
		if (pChannelClientData->pChannelInitEventProc)
Packit 1fb8d4
		{
Packit 1fb8d4
			pChannelClientData->pChannelInitEventProc(
Packit 1fb8d4
			    pChannelClientData->pInitHandle, CHANNEL_EVENT_ATTACHED, hostname, hostnameLength);
Packit 1fb8d4
		}
Packit 1fb8d4
		else if (pChannelClientData->pChannelInitEventProcEx)
Packit 1fb8d4
		{
Packit Service 5a9772
			pChannelClientData->pChannelInitEventProcEx(
Packit Service 5a9772
			    pChannelClientData->lpUserParam, pChannelClientData->pInitHandle,
Packit Service 5a9772
			    CHANNEL_EVENT_ATTACHED, hostname, hostnameLength);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (getChannelError(instance->context) != CHANNEL_RC_OK)
Packit 1fb8d4
			goto fail;
Packit 1fb8d4
Packit 1fb8d4
		pChannelOpenData = &channels->openDataList[index];
Packit 1fb8d4
		EventArgsInit(&e, "freerdp");
Packit Service 5a9772
		e.name = pChannelOpenData->name;
Packit 1fb8d4
		e.pInterface = pChannelOpenData->pInterface;
Packit 1fb8d4
		PubSub_OnChannelAttached(instance->context->pubSub, instance->context, &e);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
fail:
Packit 1fb8d4
	return error;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
UINT freerdp_channels_detach(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT error = CHANNEL_RC_OK;
Packit 1fb8d4
	int index;
Packit 1fb8d4
	char* hostname;
Packit Service 5a9772
	size_t hostnameLength;
Packit 1fb8d4
	rdpChannels* channels;
Packit 1fb8d4
	CHANNEL_CLIENT_DATA* pChannelClientData;
Packit 1fb8d4
	channels = instance->context->channels;
Packit 1fb8d4
	hostname = instance->settings->ServerHostname;
Packit Service 5a9772
	hostnameLength = strlen(hostname);
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < channels->clientDataCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		ChannelDetachedEventArgs e;
Packit 1fb8d4
		CHANNEL_OPEN_DATA* pChannelOpenData;
Packit 1fb8d4
		pChannelClientData = &channels->clientDataList[index];
Packit 1fb8d4
Packit 1fb8d4
		if (pChannelClientData->pChannelInitEventProc)
Packit 1fb8d4
		{
Packit 1fb8d4
			pChannelClientData->pChannelInitEventProc(
Packit 1fb8d4
			    pChannelClientData->pInitHandle, CHANNEL_EVENT_DETACHED, hostname, hostnameLength);
Packit 1fb8d4
		}
Packit 1fb8d4
		else if (pChannelClientData->pChannelInitEventProcEx)
Packit 1fb8d4
		{
Packit Service 5a9772
			pChannelClientData->pChannelInitEventProcEx(
Packit Service 5a9772
			    pChannelClientData->lpUserParam, pChannelClientData->pInitHandle,
Packit Service 5a9772
			    CHANNEL_EVENT_DETACHED, hostname, hostnameLength);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (getChannelError(instance->context) != CHANNEL_RC_OK)
Packit 1fb8d4
			goto fail;
Packit 1fb8d4
Packit 1fb8d4
		pChannelOpenData = &channels->openDataList[index];
Packit 1fb8d4
		EventArgsInit(&e, "freerdp");
Packit Service 5a9772
		e.name = pChannelOpenData->name;
Packit 1fb8d4
		e.pInterface = pChannelOpenData->pInterface;
Packit 1fb8d4
		PubSub_OnChannelDetached(instance->context->pubSub, instance->context, &e);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
fail:
Packit 1fb8d4
	return error;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * go through and inform all the libraries that we are connected
Packit 1fb8d4
 * this will tell the libraries that its ok to call MyVirtualChannelOpen
Packit 1fb8d4
 * called only from main thread
Packit 1fb8d4
 */
Packit 1fb8d4
UINT freerdp_channels_post_connect(rdpChannels* channels, freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT error = CHANNEL_RC_OK;
Packit Service 5a9772
	int index;
Packit 1fb8d4
	char* hostname;
Packit 1fb8d4
	size_t hostnameLength;
Packit 1fb8d4
	CHANNEL_CLIENT_DATA* pChannelClientData;
Packit 1fb8d4
	channels->connected = TRUE;
Packit 1fb8d4
	hostname = instance->settings->ServerHostname;
Packit 1fb8d4
	hostnameLength = strlen(hostname);
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < channels->clientDataCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		ChannelConnectedEventArgs e;
Packit 1fb8d4
		CHANNEL_OPEN_DATA* pChannelOpenData;
Packit 1fb8d4
		pChannelClientData = &channels->clientDataList[index];
Packit 1fb8d4
Packit 1fb8d4
		if (pChannelClientData->pChannelInitEventProc)
Packit 1fb8d4
		{
Packit 1fb8d4
			pChannelClientData->pChannelInitEventProc(
Packit 1fb8d4
			    pChannelClientData->pInitHandle, CHANNEL_EVENT_CONNECTED, hostname, hostnameLength);
Packit 1fb8d4
		}
Packit 1fb8d4
		else if (pChannelClientData->pChannelInitEventProcEx)
Packit 1fb8d4
		{
Packit Service 5a9772
			pChannelClientData->pChannelInitEventProcEx(
Packit Service 5a9772
			    pChannelClientData->lpUserParam, pChannelClientData->pInitHandle,
Packit Service 5a9772
			    CHANNEL_EVENT_CONNECTED, hostname, hostnameLength);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (getChannelError(instance->context) != CHANNEL_RC_OK)
Packit 1fb8d4
			goto fail;
Packit 1fb8d4
Packit 1fb8d4
		pChannelOpenData = &channels->openDataList[index];
Packit 1fb8d4
		EventArgsInit(&e, "freerdp");
Packit 1fb8d4
		e.name = pChannelOpenData->name;
Packit 1fb8d4
		e.pInterface = pChannelOpenData->pInterface;
Packit 1fb8d4
		PubSub_OnChannelConnected(instance->context->pubSub, instance->context, &e);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit Service 5a9772
	channels->drdynvc =
Packit Service 5a9772
	    (DrdynvcClientContext*)freerdp_channels_get_static_channel_interface(channels, "drdynvc");
Packit 1fb8d4
Packit 1fb8d4
	if (channels->drdynvc)
Packit 1fb8d4
	{
Packit Service 5a9772
		channels->drdynvc->custom = (void*)channels;
Packit 1fb8d4
		channels->drdynvc->OnChannelConnected = freerdp_drdynvc_on_channel_connected;
Packit Service 5a9772
		channels->drdynvc->OnChannelDisconnected = freerdp_drdynvc_on_channel_disconnected;
Packit 1fb8d4
		channels->drdynvc->OnChannelAttached = freerdp_drdynvc_on_channel_attached;
Packit 1fb8d4
		channels->drdynvc->OnChannelDetached = freerdp_drdynvc_on_channel_detached;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
fail:
Packit 1fb8d4
	return error;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
BOOL freerdp_channels_data(freerdp* instance, UINT16 channelId, const BYTE* cdata, size_t dataSize,
Packit Service 5a9772
                           UINT32 flags, size_t totalSize)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 index;
Packit 1fb8d4
	rdpMcs* mcs;
Packit 1fb8d4
	rdpChannels* channels;
Packit 1fb8d4
	rdpMcsChannel* channel = NULL;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit Service 5a9772
	union {
Packit Service 5a9772
		const BYTE* pcb;
Packit Service 5a9772
		BYTE* pb;
Packit Service 5a9772
	} data;
Packit Service 5a9772
Packit Service 5a9772
	data.pcb = cdata;
Packit Service 5a9772
	if (!instance || !data.pcb)
Packit Service 5a9772
	{
Packit Service 5a9772
		WLog_ERR(TAG, "%s(%p, %" PRIu16 ", %p, 0x%08x): Invalid arguments", __FUNCTION__, instance,
Packit Service 5a9772
		         channelId, data, flags);
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit 1fb8d4
	mcs = instance->context->rdp->mcs;
Packit 1fb8d4
	channels = instance->context->channels;
Packit 1fb8d4
Packit 1fb8d4
	if (!channels || !mcs)
Packit 1fb8d4
	{
Packit Service 5a9772
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < mcs->channelCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (mcs->channels[index].ChannelId == channelId)
Packit 1fb8d4
		{
Packit 1fb8d4
			channel = &mcs->channels[index];
Packit 1fb8d4
			break;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (!channel)
Packit 1fb8d4
	{
Packit Service 5a9772
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	pChannelOpenData = freerdp_channels_find_channel_open_data_by_name(channels, channel->Name);
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenData)
Packit 1fb8d4
	{
Packit Service 5a9772
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (pChannelOpenData->pChannelOpenEventProc)
Packit 1fb8d4
	{
Packit Service 5a9772
		pChannelOpenData->pChannelOpenEventProc(pChannelOpenData->OpenHandle,
Packit Service 5a9772
		                                        CHANNEL_EVENT_DATA_RECEIVED, data.pb, dataSize,
Packit Service 5a9772
		                                        totalSize, flags);
Packit 1fb8d4
	}
Packit 1fb8d4
	else if (pChannelOpenData->pChannelOpenEventProcEx)
Packit 1fb8d4
	{
Packit Service 5a9772
		pChannelOpenData->pChannelOpenEventProcEx(
Packit Service 5a9772
		    pChannelOpenData->lpUserParam, pChannelOpenData->OpenHandle,
Packit Service 5a9772
		    CHANNEL_EVENT_DATA_RECEIVED, data.pb, dataSize, totalSize, flags);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit Service 5a9772
	return TRUE;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
UINT16 freerdp_channels_get_id_by_name(freerdp* instance, const char* channel_name)
Packit Service 5a9772
{
Packit Service 5a9772
	rdpMcsChannel* mcsChannel;
Packit Service 5a9772
	if (!instance || !channel_name)
Packit Service 5a9772
		return -1;
Packit Service 5a9772
Packit Service 5a9772
	mcsChannel = freerdp_channels_find_channel_by_name(instance->context->rdp, channel_name);
Packit Service 5a9772
	if (!mcsChannel)
Packit Service 5a9772
		return -1;
Packit Service 5a9772
Packit Service 5a9772
	return mcsChannel->ChannelId;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
const char* freerdp_channels_get_name_by_id(freerdp* instance, UINT16 channelId)
Packit Service 5a9772
{
Packit Service 5a9772
	rdpMcsChannel* mcsChannel;
Packit Service 5a9772
	if (!instance)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	mcsChannel = freerdp_channels_find_channel_by_id(instance->context->rdp, channelId);
Packit Service 5a9772
	if (!mcsChannel)
Packit Service 5a9772
		return NULL;
Packit Service 5a9772
Packit Service 5a9772
	return (const char*)mcsChannel->Name;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
BOOL freerdp_channels_process_message_free(wMessage* message, DWORD type)
Packit Service 5a9772
{
Packit Service 5a9772
	if (message->id == WMQ_QUIT)
Packit Service 5a9772
	{
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (message->id == 0)
Packit Service 5a9772
	{
Packit Service 5a9772
		CHANNEL_OPEN_DATA* pChannelOpenData;
Packit Service 5a9772
		CHANNEL_OPEN_EVENT* item = (CHANNEL_OPEN_EVENT*)message->wParam;
Packit Service 5a9772
Packit Service 5a9772
		if (!item)
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
Packit Service 5a9772
		pChannelOpenData = item->pChannelOpenData;
Packit Service 5a9772
Packit Service 5a9772
		if (pChannelOpenData->pChannelOpenEventProc)
Packit Service 5a9772
		{
Packit Service 5a9772
			pChannelOpenData->pChannelOpenEventProc(pChannelOpenData->OpenHandle, type,
Packit Service 5a9772
			                                        item->UserData, item->DataLength,
Packit Service 5a9772
			                                        item->DataLength, 0);
Packit Service 5a9772
		}
Packit Service 5a9772
		else if (pChannelOpenData->pChannelOpenEventProcEx)
Packit Service 5a9772
		{
Packit Service 5a9772
			pChannelOpenData->pChannelOpenEventProcEx(
Packit Service 5a9772
			    pChannelOpenData->lpUserParam, pChannelOpenData->OpenHandle, type, item->UserData,
Packit Service 5a9772
			    item->DataLength, item->DataLength, 0);
Packit Service 5a9772
		}
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	return TRUE;
Packit Service 5a9772
}
Packit Service 5a9772
Packit Service 5a9772
static BOOL freerdp_channels_process_message(freerdp* instance, wMessage* message)
Packit Service 5a9772
{
Packit Service 5a9772
	if (message->id == WMQ_QUIT)
Packit Service 5a9772
	{
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (message->id == 0)
Packit Service 5a9772
	{
Packit Service 5a9772
		rdpMcsChannel* channel;
Packit Service 5a9772
		CHANNEL_OPEN_DATA* pChannelOpenData;
Packit Service 5a9772
		CHANNEL_OPEN_EVENT* item = (CHANNEL_OPEN_EVENT*)message->wParam;
Packit Service 5a9772
Packit Service 5a9772
		if (!item)
Packit Service 5a9772
			return FALSE;
Packit Service 5a9772
Packit Service 5a9772
		pChannelOpenData = item->pChannelOpenData;
Packit Service 5a9772
		channel =
Packit Service 5a9772
		    freerdp_channels_find_channel_by_name(instance->context->rdp, pChannelOpenData->name);
Packit Service 5a9772
Packit Service 5a9772
		if (channel)
Packit Service 5a9772
			instance->SendChannelData(instance, channel->ChannelId, item->Data, item->DataLength);
Packit Service 5a9772
	}
Packit Service 5a9772
Packit Service 5a9772
	if (!freerdp_channels_process_message_free(message, CHANNEL_EVENT_WRITE_COMPLETE))
Packit Service 5a9772
		return FALSE;
Packit Service 5a9772
Packit Service 5a9772
	IFCALL(message->Free, message);
Packit Service 5a9772
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * called only from main thread
Packit 1fb8d4
 */
Packit Service 5a9772
static int freerdp_channels_process_sync(rdpChannels* channels, freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	int status = TRUE;
Packit 1fb8d4
	wMessage message;
Packit 1fb8d4
Packit 1fb8d4
	while (MessageQueue_Peek(channels->queue, &message, TRUE))
Packit 1fb8d4
	{
Packit Service 5a9772
		freerdp_channels_process_message(instance, &message);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * called only from main thread
Packit 1fb8d4
 */
Packit Service 5a9772
BOOL freerdp_channels_get_fds(rdpChannels* channels, freerdp* instance, void** read_fds,
Packit 1fb8d4
                              int* read_count, void** write_fds, int* write_count)
Packit 1fb8d4
{
Packit 1fb8d4
	void* pfd;
Packit 1fb8d4
	pfd = GetEventWaitObject(MessageQueue_Event(channels->queue));
Packit 1fb8d4
Packit 1fb8d4
	if (pfd)
Packit 1fb8d4
	{
Packit 1fb8d4
		read_fds[*read_count] = pfd;
Packit 1fb8d4
		(*read_count)++;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
void* freerdp_channels_get_static_channel_interface(rdpChannels* channels, const char* name)
Packit 1fb8d4
{
Packit 1fb8d4
	void* pInterface = NULL;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit Service 5a9772
	pChannelOpenData = freerdp_channels_find_channel_open_data_by_name(channels, name);
Packit 1fb8d4
Packit 1fb8d4
	if (pChannelOpenData)
Packit 1fb8d4
		pInterface = pChannelOpenData->pInterface;
Packit 1fb8d4
Packit 1fb8d4
	return pInterface;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
HANDLE freerdp_channels_get_event_handle(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	HANDLE event = NULL;
Packit 1fb8d4
	rdpChannels* channels;
Packit 1fb8d4
	channels = instance->context->channels;
Packit 1fb8d4
	event = MessageQueue_Event(channels->queue);
Packit 1fb8d4
	return event;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
int freerdp_channels_process_pending_messages(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpChannels* channels;
Packit 1fb8d4
	channels = instance->context->channels;
Packit 1fb8d4
Packit Service 5a9772
	if (WaitForSingleObject(MessageQueue_Event(channels->queue), 0) == WAIT_OBJECT_0)
Packit 1fb8d4
	{
Packit 1fb8d4
		return freerdp_channels_process_sync(channels, instance);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * called only from main thread
Packit 1fb8d4
 */
Packit 1fb8d4
BOOL freerdp_channels_check_fds(rdpChannels* channels, freerdp* instance)
Packit 1fb8d4
{
Packit Service 5a9772
	if (WaitForSingleObject(MessageQueue_Event(channels->queue), 0) == WAIT_OBJECT_0)
Packit 1fb8d4
	{
Packit 1fb8d4
		freerdp_channels_process_sync(channels, instance);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
UINT freerdp_channels_disconnect(rdpChannels* channels, freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT error = CHANNEL_RC_OK;
Packit 1fb8d4
	int index;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit 1fb8d4
	CHANNEL_CLIENT_DATA* pChannelClientData;
Packit 1fb8d4
Packit 1fb8d4
	if (!channels->connected)
Packit 1fb8d4
		return 0;
Packit 1fb8d4
Packit 1fb8d4
	freerdp_channels_check_fds(channels, instance);
Packit 1fb8d4
Packit 1fb8d4
	/* tell all libraries we are shutting down */
Packit 1fb8d4
	for (index = 0; index < channels->clientDataCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		ChannelDisconnectedEventArgs e;
Packit 1fb8d4
		pChannelClientData = &channels->clientDataList[index];
Packit 1fb8d4
Packit 1fb8d4
		if (pChannelClientData->pChannelInitEventProc)
Packit 1fb8d4
		{
Packit Service 5a9772
			pChannelClientData->pChannelInitEventProc(pChannelClientData->pInitHandle,
Packit Service 5a9772
			                                          CHANNEL_EVENT_DISCONNECTED, 0, 0);
Packit 1fb8d4
		}
Packit 1fb8d4
		else if (pChannelClientData->pChannelInitEventProcEx)
Packit 1fb8d4
		{
Packit 1fb8d4
			pChannelClientData->pChannelInitEventProcEx(pChannelClientData->lpUserParam,
Packit Service 5a9772
			                                            pChannelClientData->pInitHandle,
Packit Service 5a9772
			                                            CHANNEL_EVENT_DISCONNECTED, 0, 0);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (getChannelError(instance->context) != CHANNEL_RC_OK)
Packit 1fb8d4
			continue;
Packit 1fb8d4
Packit 1fb8d4
		pChannelOpenData = &channels->openDataList[index];
Packit 1fb8d4
		EventArgsInit(&e, "freerdp");
Packit 1fb8d4
		e.name = pChannelOpenData->name;
Packit 1fb8d4
		e.pInterface = pChannelOpenData->pInterface;
Packit 1fb8d4
		PubSub_OnChannelDisconnected(instance->context->pubSub, instance->context, &e);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	channels->connected = FALSE;
Packit 1fb8d4
	return error;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void freerdp_channels_close(rdpChannels* channels, freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	int index;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit 1fb8d4
	CHANNEL_CLIENT_DATA* pChannelClientData;
Packit 1fb8d4
	freerdp_channels_check_fds(channels, instance);
Packit 1fb8d4
Packit 1fb8d4
	/* tell all libraries we are shutting down */
Packit 1fb8d4
	for (index = 0; index < channels->clientDataCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		pChannelClientData = &channels->clientDataList[index];
Packit 1fb8d4
Packit 1fb8d4
		if (pChannelClientData->pChannelInitEventProc)
Packit 1fb8d4
		{
Packit Service 5a9772
			pChannelClientData->pChannelInitEventProc(pChannelClientData->pInitHandle,
Packit Service 5a9772
			                                          CHANNEL_EVENT_TERMINATED, 0, 0);
Packit 1fb8d4
		}
Packit 1fb8d4
		else if (pChannelClientData->pChannelInitEventProcEx)
Packit 1fb8d4
		{
Packit 1fb8d4
			pChannelClientData->pChannelInitEventProcEx(pChannelClientData->lpUserParam,
Packit Service 5a9772
			                                            pChannelClientData->pInitHandle,
Packit Service 5a9772
			                                            CHANNEL_EVENT_TERMINATED, 0, 0);
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	channels->clientDataCount = 0;
Packit 1fb8d4
	MessageQueue_PostQuit(channels->queue, 0);
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < channels->openDataCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		pChannelOpenData = &channels->openDataList[index];
Packit 1fb8d4
		freerdp_channel_remove_open_handle_data(&g_ChannelHandles, pChannelOpenData->OpenHandle);
Packit 1fb8d4
Packit 1fb8d4
		if (channels->openHandles)
Packit Service 5a9772
			HashTable_Remove(channels->openHandles, (void*)(UINT_PTR)pChannelOpenData->OpenHandle);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	channels->openDataCount = 0;
Packit 1fb8d4
	channels->initDataCount = 0;
Packit 1fb8d4
	instance->settings->ChannelCount = 0;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static UINT VCAPITYPE FreeRDP_VirtualChannelInitEx(
Packit Service 5a9772
    LPVOID lpUserParam, LPVOID clientContext, LPVOID pInitHandle, PCHANNEL_DEF pChannel,
Packit Service 5a9772
    INT channelCount, ULONG versionRequested, PCHANNEL_INIT_EVENT_EX_FN pChannelInitEventProcEx)
Packit 1fb8d4
{
Packit 1fb8d4
	INT index;
Packit 1fb8d4
	CHANNEL_DEF* channel;
Packit 1fb8d4
	rdpSettings* settings;
Packit 1fb8d4
	PCHANNEL_DEF pChannelDef;
Packit 1fb8d4
	CHANNEL_INIT_DATA* pChannelInitData;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit 1fb8d4
	CHANNEL_CLIENT_DATA* pChannelClientData;
Packit Service 5a9772
	rdpChannels* channels = (rdpChannels*)pInitHandle;
Packit 1fb8d4
Packit 1fb8d4
	if (!pInitHandle)
Packit 1fb8d4
		return CHANNEL_RC_BAD_INIT_HANDLE;
Packit 1fb8d4
Packit Service 5a9772
	if (!pChannel)
Packit Service 5a9772
		return CHANNEL_RC_BAD_CHANNEL;
Packit Service 5a9772
Packit Service 5a9772
	if ((channelCount <= 0) || !pChannelInitEventProcEx)
Packit 1fb8d4
		return CHANNEL_RC_INITIALIZATION_ERROR;
Packit 1fb8d4
Packit Service 5a9772
	pChannelInitData = (CHANNEL_INIT_DATA*)pInitHandle;
Packit 1fb8d4
	channels = pChannelInitData->channels;
Packit 1fb8d4
	pChannelInitData->pInterface = clientContext;
Packit 1fb8d4
Packit 1fb8d4
	if (!channels->can_call_init)
Packit 1fb8d4
		return CHANNEL_RC_NOT_IN_VIRTUALCHANNELENTRY;
Packit 1fb8d4
Packit 1fb8d4
	if ((channels->openDataCount + channelCount) > CHANNEL_MAX_COUNT)
Packit 1fb8d4
		return CHANNEL_RC_TOO_MANY_CHANNELS;
Packit 1fb8d4
Packit 1fb8d4
	if (channels->connected)
Packit 1fb8d4
		return CHANNEL_RC_ALREADY_CONNECTED;
Packit 1fb8d4
Packit 1fb8d4
	if (versionRequested != VIRTUAL_CHANNEL_VERSION_WIN2000)
Packit 1fb8d4
	{
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < channelCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		pChannelDef = &pChannel[index];
Packit 1fb8d4
Packit 1fb8d4
		if (freerdp_channels_find_channel_open_data_by_name(channels, pChannelDef->name) != 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			return CHANNEL_RC_BAD_CHANNEL;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	pChannelClientData = &channels->clientDataList[channels->clientDataCount];
Packit 1fb8d4
	pChannelClientData->pChannelInitEventProcEx = pChannelInitEventProcEx;
Packit 1fb8d4
	pChannelClientData->pInitHandle = pInitHandle;
Packit 1fb8d4
	pChannelClientData->lpUserParam = lpUserParam;
Packit 1fb8d4
	channels->clientDataCount++;
Packit 1fb8d4
	settings = channels->instance->context->settings;
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < channelCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		pChannelDef = &pChannel[index];
Packit 1fb8d4
		pChannelOpenData = &channels->openDataList[channels->openDataCount];
Packit 1fb8d4
		pChannelOpenData->OpenHandle = InterlockedIncrement(&g_OpenHandleSeq);
Packit 1fb8d4
		pChannelOpenData->channels = channels;
Packit 1fb8d4
		pChannelOpenData->lpUserParam = lpUserParam;
Packit Service 5a9772
		HashTable_Add(channels->openHandles, (void*)(UINT_PTR)pChannelOpenData->OpenHandle,
Packit Service 5a9772
		              (void*)pChannelOpenData);
Packit 1fb8d4
		pChannelOpenData->flags = 1; /* init */
Packit 1fb8d4
		strncpy(pChannelOpenData->name, pChannelDef->name, CHANNEL_NAME_LEN);
Packit 1fb8d4
		pChannelOpenData->options = pChannelDef->options;
Packit 1fb8d4
Packit 1fb8d4
		if (settings->ChannelCount < CHANNEL_MAX_COUNT)
Packit 1fb8d4
		{
Packit 1fb8d4
			channel = &settings->ChannelDefArray[settings->ChannelCount];
Packit 1fb8d4
			strncpy(channel->name, pChannelDef->name, 7);
Packit 1fb8d4
			channel->options = pChannelDef->options;
Packit 1fb8d4
			settings->ChannelCount++;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		channels->openDataCount++;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static UINT VCAPITYPE FreeRDP_VirtualChannelInit(LPVOID* ppInitHandle, PCHANNEL_DEF pChannel,
Packit Service 5a9772
                                                 INT channelCount, ULONG versionRequested,
Packit Service 5a9772
                                                 PCHANNEL_INIT_EVENT_FN pChannelInitEventProc)
Packit 1fb8d4
{
Packit 1fb8d4
	INT index;
Packit 1fb8d4
	void* pInterface;
Packit 1fb8d4
	CHANNEL_DEF* channel;
Packit 1fb8d4
	rdpSettings* settings;
Packit 1fb8d4
	PCHANNEL_DEF pChannelDef;
Packit 1fb8d4
	CHANNEL_INIT_DATA* pChannelInitData;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit 1fb8d4
	CHANNEL_CLIENT_DATA* pChannelClientData;
Packit 1fb8d4
	rdpChannels* channels = g_channels;
Packit 1fb8d4
Packit 1fb8d4
	if (!ppInitHandle || !channels)
Packit 1fb8d4
		return CHANNEL_RC_BAD_INIT_HANDLE;
Packit 1fb8d4
Packit Service 5a9772
	if (!pChannel)
Packit Service 5a9772
		return CHANNEL_RC_BAD_CHANNEL;
Packit Service 5a9772
Packit Service 5a9772
	if ((channelCount <= 0) || !pChannelInitEventProc)
Packit 1fb8d4
		return CHANNEL_RC_INITIALIZATION_ERROR;
Packit 1fb8d4
Packit 1fb8d4
	pInterface = g_pInterface;
Packit 1fb8d4
	pChannelInitData = &(channels->initDataList[channels->initDataCount]);
Packit 1fb8d4
	*ppInitHandle = pChannelInitData;
Packit 1fb8d4
	channels->initDataCount++;
Packit 1fb8d4
	pChannelInitData->channels = channels;
Packit 1fb8d4
	pChannelInitData->pInterface = pInterface;
Packit 1fb8d4
Packit 1fb8d4
	if (!channels->can_call_init)
Packit 1fb8d4
		return CHANNEL_RC_NOT_IN_VIRTUALCHANNELENTRY;
Packit 1fb8d4
Packit 1fb8d4
	if (channels->openDataCount + channelCount > CHANNEL_MAX_COUNT)
Packit 1fb8d4
		return CHANNEL_RC_TOO_MANY_CHANNELS;
Packit 1fb8d4
Packit 1fb8d4
	if (channels->connected)
Packit 1fb8d4
		return CHANNEL_RC_ALREADY_CONNECTED;
Packit 1fb8d4
Packit 1fb8d4
	if (versionRequested != VIRTUAL_CHANNEL_VERSION_WIN2000)
Packit 1fb8d4
	{
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < channelCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		pChannelDef = &pChannel[index];
Packit 1fb8d4
Packit Service 5a9772
		if (freerdp_channels_find_channel_open_data_by_name(channels, pChannelDef->name) != 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			return CHANNEL_RC_BAD_CHANNEL;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	pChannelClientData = &channels->clientDataList[channels->clientDataCount];
Packit 1fb8d4
	pChannelClientData->pChannelInitEventProc = pChannelInitEventProc;
Packit 1fb8d4
	pChannelClientData->pInitHandle = *ppInitHandle;
Packit 1fb8d4
	channels->clientDataCount++;
Packit 1fb8d4
	settings = channels->instance->context->settings;
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < channelCount; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		pChannelDef = &pChannel[index];
Packit 1fb8d4
		pChannelOpenData = &channels->openDataList[channels->openDataCount];
Packit 1fb8d4
		pChannelOpenData->OpenHandle = InterlockedIncrement(&g_OpenHandleSeq);
Packit 1fb8d4
		pChannelOpenData->channels = channels;
Packit 1fb8d4
		freerdp_channel_add_open_handle_data(&g_ChannelHandles, pChannelOpenData->OpenHandle,
Packit Service 5a9772
		                                     (void*)channels);
Packit Service 5a9772
		HashTable_Add(channels->openHandles, (void*)(UINT_PTR)pChannelOpenData->OpenHandle,
Packit Service 5a9772
		              (void*)pChannelOpenData);
Packit 1fb8d4
		pChannelOpenData->flags = 1; /* init */
Packit 1fb8d4
		strncpy(pChannelOpenData->name, pChannelDef->name, CHANNEL_NAME_LEN);
Packit 1fb8d4
		pChannelOpenData->options = pChannelDef->options;
Packit 1fb8d4
Packit 1fb8d4
		if (settings->ChannelCount < CHANNEL_MAX_COUNT)
Packit 1fb8d4
		{
Packit 1fb8d4
			channel = &settings->ChannelDefArray[settings->ChannelCount];
Packit 1fb8d4
			strncpy(channel->name, pChannelDef->name, 7);
Packit 1fb8d4
			channel->options = pChannelDef->options;
Packit 1fb8d4
			settings->ChannelCount++;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		channels->openDataCount++;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static UINT VCAPITYPE
Packit Service 5a9772
FreeRDP_VirtualChannelOpenEx(LPVOID pInitHandle, LPDWORD pOpenHandle, PCHAR pChannelName,
Packit Service 5a9772
                             PCHANNEL_OPEN_EVENT_EX_FN pChannelOpenEventProcEx)
Packit 1fb8d4
{
Packit 1fb8d4
	void* pInterface;
Packit 1fb8d4
	rdpChannels* channels;
Packit 1fb8d4
	CHANNEL_INIT_DATA* pChannelInitData;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit Service 5a9772
	pChannelInitData = (CHANNEL_INIT_DATA*)pInitHandle;
Packit 1fb8d4
	channels = pChannelInitData->channels;
Packit 1fb8d4
	pInterface = pChannelInitData->pInterface;
Packit 1fb8d4
Packit 1fb8d4
	if (!pOpenHandle)
Packit 1fb8d4
		return CHANNEL_RC_BAD_CHANNEL_HANDLE;
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenEventProcEx)
Packit 1fb8d4
		return CHANNEL_RC_BAD_PROC;
Packit 1fb8d4
Packit 1fb8d4
	if (!channels->connected)
Packit 1fb8d4
		return CHANNEL_RC_NOT_CONNECTED;
Packit 1fb8d4
Packit 1fb8d4
	pChannelOpenData = freerdp_channels_find_channel_open_data_by_name(channels, pChannelName);
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenData)
Packit 1fb8d4
		return CHANNEL_RC_UNKNOWN_CHANNEL_NAME;
Packit 1fb8d4
Packit 1fb8d4
	if (pChannelOpenData->flags == 2)
Packit 1fb8d4
		return CHANNEL_RC_ALREADY_OPEN;
Packit 1fb8d4
Packit 1fb8d4
	pChannelOpenData->flags = 2; /* open */
Packit 1fb8d4
	pChannelOpenData->pInterface = pInterface;
Packit 1fb8d4
	pChannelOpenData->pChannelOpenEventProcEx = pChannelOpenEventProcEx;
Packit 1fb8d4
	*pOpenHandle = pChannelOpenData->OpenHandle;
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static UINT VCAPITYPE FreeRDP_VirtualChannelOpen(LPVOID pInitHandle, LPDWORD pOpenHandle,
Packit Service 5a9772
                                                 PCHAR pChannelName,
Packit Service 5a9772
                                                 PCHANNEL_OPEN_EVENT_FN pChannelOpenEventProc)
Packit 1fb8d4
{
Packit 1fb8d4
	void* pInterface;
Packit 1fb8d4
	rdpChannels* channels;
Packit 1fb8d4
	CHANNEL_INIT_DATA* pChannelInitData;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit Service 5a9772
	pChannelInitData = (CHANNEL_INIT_DATA*)pInitHandle;
Packit 1fb8d4
	channels = pChannelInitData->channels;
Packit 1fb8d4
	pInterface = pChannelInitData->pInterface;
Packit 1fb8d4
Packit 1fb8d4
	if (!pOpenHandle)
Packit 1fb8d4
		return CHANNEL_RC_BAD_CHANNEL_HANDLE;
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenEventProc)
Packit 1fb8d4
		return CHANNEL_RC_BAD_PROC;
Packit 1fb8d4
Packit 1fb8d4
	if (!channels->connected)
Packit 1fb8d4
		return CHANNEL_RC_NOT_CONNECTED;
Packit 1fb8d4
Packit 1fb8d4
	pChannelOpenData = freerdp_channels_find_channel_open_data_by_name(channels, pChannelName);
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenData)
Packit 1fb8d4
		return CHANNEL_RC_UNKNOWN_CHANNEL_NAME;
Packit 1fb8d4
Packit 1fb8d4
	if (pChannelOpenData->flags == 2)
Packit 1fb8d4
		return CHANNEL_RC_ALREADY_OPEN;
Packit 1fb8d4
Packit 1fb8d4
	pChannelOpenData->flags = 2; /* open */
Packit 1fb8d4
	pChannelOpenData->pInterface = pInterface;
Packit 1fb8d4
	pChannelOpenData->pChannelOpenEventProc = pChannelOpenEventProc;
Packit 1fb8d4
	*pOpenHandle = pChannelOpenData->OpenHandle;
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static UINT VCAPITYPE FreeRDP_VirtualChannelCloseEx(LPVOID pInitHandle, DWORD openHandle)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpChannels* channels = NULL;
Packit 1fb8d4
	CHANNEL_INIT_DATA* pChannelInitData = NULL;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData = NULL;
Packit 1fb8d4
Packit 1fb8d4
	if (!pInitHandle)
Packit 1fb8d4
		return CHANNEL_RC_BAD_INIT_HANDLE;
Packit 1fb8d4
Packit Service 5a9772
	pChannelInitData = (CHANNEL_INIT_DATA*)pInitHandle;
Packit 1fb8d4
	channels = pChannelInitData->channels;
Packit 1fb8d4
Packit 1fb8d4
	if (!channels)
Packit 1fb8d4
		return CHANNEL_RC_BAD_CHANNEL_HANDLE;
Packit 1fb8d4
Packit Service 5a9772
	pChannelOpenData = HashTable_GetItemValue(channels->openHandles, (void*)(UINT_PTR)openHandle);
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenData)
Packit 1fb8d4
		return CHANNEL_RC_BAD_CHANNEL_HANDLE;
Packit 1fb8d4
Packit 1fb8d4
	if (pChannelOpenData->flags != 2)
Packit 1fb8d4
		return CHANNEL_RC_NOT_OPEN;
Packit 1fb8d4
Packit 1fb8d4
	pChannelOpenData->flags = 0;
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static UINT VCAPITYPE FreeRDP_VirtualChannelClose(DWORD openHandle)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpChannels* channels;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit Service 5a9772
	channels = (rdpChannels*)freerdp_channel_get_open_handle_data(&g_ChannelHandles, openHandle);
Packit 1fb8d4
Packit 1fb8d4
	if (!channels)
Packit 1fb8d4
		return CHANNEL_RC_BAD_CHANNEL_HANDLE;
Packit 1fb8d4
Packit Service 5a9772
	pChannelOpenData = HashTable_GetItemValue(channels->openHandles, (void*)(UINT_PTR)openHandle);
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenData)
Packit 1fb8d4
		return CHANNEL_RC_BAD_CHANNEL_HANDLE;
Packit 1fb8d4
Packit 1fb8d4
	if (pChannelOpenData->flags != 2)
Packit 1fb8d4
		return CHANNEL_RC_NOT_OPEN;
Packit 1fb8d4
Packit 1fb8d4
	pChannelOpenData->flags = 0;
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static UINT VCAPITYPE FreeRDP_VirtualChannelWriteEx(LPVOID pInitHandle, DWORD openHandle,
Packit Service 5a9772
                                                    LPVOID pData, ULONG dataLength,
Packit Service 5a9772
                                                    LPVOID pUserData)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpChannels* channels = NULL;
Packit 1fb8d4
	CHANNEL_INIT_DATA* pChannelInitData = NULL;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData = NULL;
Packit 1fb8d4
	CHANNEL_OPEN_EVENT* pChannelOpenEvent = NULL;
Packit 1fb8d4
	wMessage message;
Packit 1fb8d4
Packit 1fb8d4
	if (!pInitHandle)
Packit 1fb8d4
		return CHANNEL_RC_BAD_INIT_HANDLE;
Packit 1fb8d4
Packit Service 5a9772
	pChannelInitData = (CHANNEL_INIT_DATA*)pInitHandle;
Packit 1fb8d4
	channels = pChannelInitData->channels;
Packit 1fb8d4
Packit 1fb8d4
	if (!channels)
Packit 1fb8d4
		return CHANNEL_RC_BAD_CHANNEL_HANDLE;
Packit 1fb8d4
Packit Service 5a9772
	pChannelOpenData = HashTable_GetItemValue(channels->openHandles, (void*)(UINT_PTR)openHandle);
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenData)
Packit 1fb8d4
		return CHANNEL_RC_BAD_CHANNEL_HANDLE;
Packit 1fb8d4
Packit 1fb8d4
	if (!channels->connected)
Packit 1fb8d4
		return CHANNEL_RC_NOT_CONNECTED;
Packit 1fb8d4
Packit 1fb8d4
	if (!pData)
Packit 1fb8d4
		return CHANNEL_RC_NULL_DATA;
Packit 1fb8d4
Packit 1fb8d4
	if (!dataLength)
Packit 1fb8d4
		return CHANNEL_RC_ZERO_LENGTH;
Packit 1fb8d4
Packit 1fb8d4
	if (pChannelOpenData->flags != 2)
Packit 1fb8d4
		return CHANNEL_RC_NOT_OPEN;
Packit 1fb8d4
Packit Service 5a9772
	pChannelOpenEvent = (CHANNEL_OPEN_EVENT*)malloc(sizeof(CHANNEL_OPEN_EVENT));
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenEvent)
Packit 1fb8d4
		return CHANNEL_RC_NO_MEMORY;
Packit 1fb8d4
Packit 1fb8d4
	pChannelOpenEvent->Data = pData;
Packit 1fb8d4
	pChannelOpenEvent->DataLength = dataLength;
Packit 1fb8d4
	pChannelOpenEvent->UserData = pUserData;
Packit 1fb8d4
	pChannelOpenEvent->pChannelOpenData = pChannelOpenData;
Packit 1fb8d4
	message.context = channels;
Packit 1fb8d4
	message.id = 0;
Packit 1fb8d4
	message.wParam = pChannelOpenEvent;
Packit 1fb8d4
	message.lParam = NULL;
Packit 1fb8d4
	message.Free = channel_queue_message_free;
Packit 1fb8d4
Packit 1fb8d4
	if (!MessageQueue_Dispatch(channels->queue, &message))
Packit 1fb8d4
	{
Packit Service 5a9772
		free(pChannelOpenEvent);
Packit 1fb8d4
		return CHANNEL_RC_NO_MEMORY;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit Service 5a9772
static UINT VCAPITYPE FreeRDP_VirtualChannelWrite(DWORD openHandle, LPVOID pData, ULONG dataLength,
Packit Service 5a9772
                                                  LPVOID pUserData)
Packit 1fb8d4
{
Packit 1fb8d4
	wMessage message;
Packit 1fb8d4
	CHANNEL_OPEN_DATA* pChannelOpenData;
Packit 1fb8d4
	CHANNEL_OPEN_EVENT* pChannelOpenEvent;
Packit Service 5a9772
	rdpChannels* channels =
Packit Service 5a9772
	    (rdpChannels*)freerdp_channel_get_open_handle_data(&g_ChannelHandles, openHandle);
Packit 1fb8d4
Packit 1fb8d4
	if (!channels)
Packit 1fb8d4
		return CHANNEL_RC_BAD_CHANNEL_HANDLE;
Packit 1fb8d4
Packit Service 5a9772
	pChannelOpenData = HashTable_GetItemValue(channels->openHandles, (void*)(UINT_PTR)openHandle);
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenData)
Packit 1fb8d4
		return CHANNEL_RC_BAD_CHANNEL_HANDLE;
Packit 1fb8d4
Packit 1fb8d4
	if (!channels->connected)
Packit 1fb8d4
		return CHANNEL_RC_NOT_CONNECTED;
Packit 1fb8d4
Packit 1fb8d4
	if (!pData)
Packit 1fb8d4
		return CHANNEL_RC_NULL_DATA;
Packit 1fb8d4
Packit 1fb8d4
	if (!dataLength)
Packit 1fb8d4
		return CHANNEL_RC_ZERO_LENGTH;
Packit 1fb8d4
Packit 1fb8d4
	if (pChannelOpenData->flags != 2)
Packit 1fb8d4
		return CHANNEL_RC_NOT_OPEN;
Packit 1fb8d4
Packit Service 5a9772
	pChannelOpenEvent = (CHANNEL_OPEN_EVENT*)malloc(sizeof(CHANNEL_OPEN_EVENT));
Packit 1fb8d4
Packit 1fb8d4
	if (!pChannelOpenEvent)
Packit 1fb8d4
		return CHANNEL_RC_NO_MEMORY;
Packit 1fb8d4
Packit 1fb8d4
	pChannelOpenEvent->Data = pData;
Packit 1fb8d4
	pChannelOpenEvent->DataLength = dataLength;
Packit 1fb8d4
	pChannelOpenEvent->UserData = pUserData;
Packit 1fb8d4
	pChannelOpenEvent->pChannelOpenData = pChannelOpenData;
Packit 1fb8d4
	message.context = channels;
Packit 1fb8d4
	message.id = 0;
Packit 1fb8d4
	message.wParam = pChannelOpenEvent;
Packit 1fb8d4
	message.lParam = NULL;
Packit 1fb8d4
	message.Free = channel_queue_message_free;
Packit 1fb8d4
Packit 1fb8d4
	if (!MessageQueue_Dispatch(channels->queue, &message))
Packit 1fb8d4
	{
Packit Service 5a9772
		free(pChannelOpenEvent);
Packit 1fb8d4
		return CHANNEL_RC_NO_MEMORY;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL freerdp_channels_is_loaded(rdpChannels* channels, PVIRTUALCHANNELENTRY entry)
Packit 1fb8d4
{
Packit 1fb8d4
	int i;
Packit 1fb8d4
Packit 1fb8d4
	for (i = 0; i < channels->clientDataCount; i++)
Packit 1fb8d4
	{
Packit 1fb8d4
		CHANNEL_CLIENT_DATA* pChannelClientData = &channels->clientDataList[i];
Packit 1fb8d4
Packit 1fb8d4
		if (pChannelClientData->entry == entry)
Packit 1fb8d4
			return TRUE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return FALSE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static BOOL freerdp_channels_is_loaded_ex(rdpChannels* channels, PVIRTUALCHANNELENTRYEX entryEx)
Packit 1fb8d4
{
Packit 1fb8d4
	int i;
Packit 1fb8d4
Packit 1fb8d4
	for (i = 0; i < channels->clientDataCount; i++)
Packit 1fb8d4
	{
Packit 1fb8d4
		CHANNEL_CLIENT_DATA* pChannelClientData = &channels->clientDataList[i];
Packit 1fb8d4
Packit 1fb8d4
		if (pChannelClientData->entryEx == entryEx)
Packit 1fb8d4
			return TRUE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return FALSE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
int freerdp_channels_client_load(rdpChannels* channels, rdpSettings* settings,
Packit 1fb8d4
                                 PVIRTUALCHANNELENTRY entry, void* data)
Packit 1fb8d4
{
Packit 1fb8d4
	int status;
Packit 1fb8d4
	CHANNEL_ENTRY_POINTS_FREERDP EntryPoints;
Packit 1fb8d4
	CHANNEL_CLIENT_DATA* pChannelClientData;
Packit 1fb8d4
Packit 1fb8d4
	if (channels->clientDataCount + 1 > CHANNEL_MAX_COUNT)
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_ERR(TAG, "error: too many channels");
Packit 1fb8d4
		return 1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (freerdp_channels_is_loaded(channels, entry))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_WARN(TAG, "Skipping, channel already loaded");
Packit 1fb8d4
		return 0;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	pChannelClientData = &channels->clientDataList[channels->clientDataCount];
Packit 1fb8d4
	pChannelClientData->entry = entry;
Packit 1fb8d4
	ZeroMemory(&EntryPoints, sizeof(CHANNEL_ENTRY_POINTS_FREERDP));
Packit 1fb8d4
	EntryPoints.cbSize = sizeof(EntryPoints);
Packit 1fb8d4
	EntryPoints.protocolVersion = VIRTUAL_CHANNEL_VERSION_WIN2000;
Packit 1fb8d4
	EntryPoints.pVirtualChannelInit = FreeRDP_VirtualChannelInit;
Packit 1fb8d4
	EntryPoints.pVirtualChannelOpen = FreeRDP_VirtualChannelOpen;
Packit 1fb8d4
	EntryPoints.pVirtualChannelClose = FreeRDP_VirtualChannelClose;
Packit 1fb8d4
	EntryPoints.pVirtualChannelWrite = FreeRDP_VirtualChannelWrite;
Packit 1fb8d4
	EntryPoints.MagicNumber = FREERDP_CHANNEL_MAGIC_NUMBER;
Packit 1fb8d4
	EntryPoints.ppInterface = &g_pInterface;
Packit 1fb8d4
	EntryPoints.pExtendedData = data;
Packit 1fb8d4
	EntryPoints.context = ((freerdp*)settings->instance)->context;
Packit 1fb8d4
	/* enable VirtualChannelInit */
Packit 1fb8d4
	channels->can_call_init = TRUE;
Packit 1fb8d4
	EnterCriticalSection(&channels->channelsLock);
Packit 1fb8d4
	g_pInterface = NULL;
Packit 1fb8d4
	g_channels = channels;
Packit Service 5a9772
	status = pChannelClientData->entry((PCHANNEL_ENTRY_POINTS)&EntryPoints);
Packit 1fb8d4
	LeaveCriticalSection(&channels->channelsLock);
Packit 1fb8d4
	/* disable MyVirtualChannelInit */
Packit 1fb8d4
	channels->can_call_init = FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (!status)
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_ERR(TAG, "error: channel export function call failed");
Packit 1fb8d4
		return 1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return 0;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
int freerdp_channels_client_load_ex(rdpChannels* channels, rdpSettings* settings,
Packit 1fb8d4
                                    PVIRTUALCHANNELENTRYEX entryEx, void* data)
Packit 1fb8d4
{
Packit 1fb8d4
	int status;
Packit 1fb8d4
	void* pInitHandle = NULL;
Packit 1fb8d4
	CHANNEL_ENTRY_POINTS_FREERDP_EX EntryPointsEx;
Packit 1fb8d4
	CHANNEL_INIT_DATA* pChannelInitData = NULL;
Packit 1fb8d4
	CHANNEL_CLIENT_DATA* pChannelClientData = NULL;
Packit 1fb8d4
Packit 1fb8d4
	if (channels->clientDataCount + 1 > CHANNEL_MAX_COUNT)
Packit 1fb8d4
	{
Packit Service 5a9772
		WLog_ERR(TAG, "error: too many channels");
Packit 1fb8d4
		return 1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (freerdp_channels_is_loaded_ex(channels, entryEx))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_WARN(TAG, "Skipping, channel already loaded");
Packit 1fb8d4
		return 0;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	pChannelClientData = &channels->clientDataList[channels->clientDataCount];
Packit 1fb8d4
	pChannelClientData->entryEx = entryEx;
Packit 1fb8d4
	pChannelInitData = &(channels->initDataList[channels->initDataCount++]);
Packit 1fb8d4
	pInitHandle = pChannelInitData;
Packit 1fb8d4
	pChannelInitData->channels = channels;
Packit 1fb8d4
	ZeroMemory(&EntryPointsEx, sizeof(CHANNEL_ENTRY_POINTS_FREERDP_EX));
Packit 1fb8d4
	EntryPointsEx.cbSize = sizeof(EntryPointsEx);
Packit 1fb8d4
	EntryPointsEx.protocolVersion = VIRTUAL_CHANNEL_VERSION_WIN2000;
Packit 1fb8d4
	EntryPointsEx.pVirtualChannelInitEx = FreeRDP_VirtualChannelInitEx;
Packit 1fb8d4
	EntryPointsEx.pVirtualChannelOpenEx = FreeRDP_VirtualChannelOpenEx;
Packit 1fb8d4
	EntryPointsEx.pVirtualChannelCloseEx = FreeRDP_VirtualChannelCloseEx;
Packit 1fb8d4
	EntryPointsEx.pVirtualChannelWriteEx = FreeRDP_VirtualChannelWriteEx;
Packit 1fb8d4
	EntryPointsEx.MagicNumber = FREERDP_CHANNEL_MAGIC_NUMBER;
Packit 1fb8d4
	EntryPointsEx.pExtendedData = data;
Packit Service 5a9772
	EntryPointsEx.context = ((freerdp*)settings->instance)->context;
Packit 1fb8d4
	/* enable VirtualChannelInit */
Packit 1fb8d4
	channels->can_call_init = TRUE;
Packit 1fb8d4
	EnterCriticalSection(&channels->channelsLock);
Packit Service 5a9772
	status = pChannelClientData->entryEx((PCHANNEL_ENTRY_POINTS_EX)&EntryPointsEx, pInitHandle);
Packit 1fb8d4
	LeaveCriticalSection(&channels->channelsLock);
Packit 1fb8d4
	/* disable MyVirtualChannelInit */
Packit 1fb8d4
	channels->can_call_init = FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (!status)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "error: channel export function call failed");
Packit 1fb8d4
		return 1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return 0;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * this is called when processing the command line parameters
Packit 1fb8d4
 * called only from main thread
Packit 1fb8d4
 */
Packit 1fb8d4
int freerdp_channels_load_plugin(rdpChannels* channels, rdpSettings* settings, const char* name,
Packit 1fb8d4
                                 void* data)
Packit 1fb8d4
{
Packit 1fb8d4
	PVIRTUALCHANNELENTRY entry;
Packit 1fb8d4
	entry = freerdp_load_channel_addin_entry(name, NULL, NULL, FREERDP_ADDIN_CHANNEL_STATIC);
Packit 1fb8d4
Packit 1fb8d4
	if (!entry)
Packit 1fb8d4
		return 1;
Packit 1fb8d4
Packit 1fb8d4
	return freerdp_channels_client_load(channels, settings, entry, data);
Packit 1fb8d4
}