Blame libfreerdp/core/freerdp.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * FreeRDP Core
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2011 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 "rdp.h"
Packit 1fb8d4
#include "input.h"
Packit 1fb8d4
#include "update.h"
Packit 1fb8d4
#include "surface.h"
Packit 1fb8d4
#include "transport.h"
Packit 1fb8d4
#include "connection.h"
Packit 1fb8d4
#include "message.h"
Packit 1fb8d4
#include "buildflags.h"
Packit 1fb8d4
Packit 1fb8d4
#include <assert.h>
Packit 1fb8d4
Packit 1fb8d4
#include <winpr/crt.h>
Packit 1fb8d4
#include <winpr/string.h>
Packit 1fb8d4
#include <winpr/stream.h>
Packit 1fb8d4
#include <winpr/wtsapi.h>
Packit 1fb8d4
#include <winpr/ssl.h>
Packit 1fb8d4
#include <winpr/debug.h>
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/freerdp.h>
Packit 1fb8d4
#include <freerdp/error.h>
Packit 1fb8d4
#include <freerdp/event.h>
Packit 1fb8d4
#include <freerdp/locale/keyboard.h>
Packit 1fb8d4
#include <freerdp/channels/channels.h>
Packit 1fb8d4
#include <freerdp/version.h>
Packit 1fb8d4
#include <freerdp/log.h>
Packit 1fb8d4
#include <freerdp/cache/pointer.h>
Packit 1fb8d4
Packit 1fb8d4
#define TAG FREERDP_TAG("core")
Packit 1fb8d4
Packit 1fb8d4
/* connectErrorCode is 'extern' in error.h. See comment there.*/
Packit 1fb8d4
Packit 1fb8d4
UINT freerdp_channel_add_init_handle_data(rdpChannelHandles* handles, void* pInitHandle,
Packit 1fb8d4
        void* pUserData)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!handles->init)
Packit 1fb8d4
		handles->init = ListDictionary_New(TRUE);
Packit 1fb8d4
Packit 1fb8d4
	if (!handles->init)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "ListDictionary_New failed!");
Packit 1fb8d4
		return ERROR_NOT_ENOUGH_MEMORY;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (!ListDictionary_Add(handles->init, pInitHandle, pUserData))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "ListDictionary_Add failed!");
Packit 1fb8d4
		return ERROR_INTERNAL_ERROR;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void* freerdp_channel_get_init_handle_data(rdpChannelHandles* handles, void* pInitHandle)
Packit 1fb8d4
{
Packit 1fb8d4
	void* pUserData = NULL;
Packit 1fb8d4
	pUserData = ListDictionary_GetItemValue(handles->init, pInitHandle);
Packit 1fb8d4
	return pUserData;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void freerdp_channel_remove_init_handle_data(rdpChannelHandles* handles, void* pInitHandle)
Packit 1fb8d4
{
Packit 1fb8d4
	ListDictionary_Remove(handles->init, pInitHandle);
Packit 1fb8d4
Packit 1fb8d4
	if (ListDictionary_Count(handles->init) < 1)
Packit 1fb8d4
	{
Packit 1fb8d4
		ListDictionary_Free(handles->init);
Packit 1fb8d4
		handles->init = NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
UINT freerdp_channel_add_open_handle_data(rdpChannelHandles* handles, DWORD openHandle,
Packit 1fb8d4
        void* pUserData)
Packit 1fb8d4
{
Packit 1fb8d4
	void* pOpenHandle = (void*)(size_t) openHandle;
Packit 1fb8d4
Packit 1fb8d4
	if (!handles->open)
Packit 1fb8d4
		handles->open = ListDictionary_New(TRUE);
Packit 1fb8d4
Packit 1fb8d4
	if (!handles->open)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "ListDictionary_New failed!");
Packit 1fb8d4
		return ERROR_NOT_ENOUGH_MEMORY;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (!ListDictionary_Add(handles->open, pOpenHandle, pUserData))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "ListDictionary_Add failed!");
Packit 1fb8d4
		return ERROR_INTERNAL_ERROR;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return CHANNEL_RC_OK;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void* freerdp_channel_get_open_handle_data(rdpChannelHandles* handles, DWORD openHandle)
Packit 1fb8d4
{
Packit 1fb8d4
	void* pUserData = NULL;
Packit 1fb8d4
	void* pOpenHandle = (void*)(size_t) openHandle;
Packit 1fb8d4
	pUserData = ListDictionary_GetItemValue(handles->open, pOpenHandle);
Packit 1fb8d4
	return pUserData;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void freerdp_channel_remove_open_handle_data(rdpChannelHandles* handles, DWORD openHandle)
Packit 1fb8d4
{
Packit 1fb8d4
	void* pOpenHandle = (void*)(size_t) openHandle;
Packit 1fb8d4
	ListDictionary_Remove(handles->open, pOpenHandle);
Packit 1fb8d4
Packit 1fb8d4
	if (ListDictionary_Count(handles->open) < 1)
Packit 1fb8d4
	{
Packit 1fb8d4
		ListDictionary_Free(handles->open);
Packit 1fb8d4
		handles->open = NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/** Creates a new connection based on the settings found in the "instance" parameter
Packit 1fb8d4
 *  It will use the callbacks registered on the structure to process the pre/post connect operations
Packit 1fb8d4
 *  that the caller requires.
Packit 1fb8d4
 *  @see struct rdp_freerdp in freerdp.h
Packit 1fb8d4
 *
Packit 1fb8d4
 *  @param instance - pointer to a rdp_freerdp structure that contains base information to establish the connection.
Packit 1fb8d4
 *  				  On return, this function will be initialized with the new connection's settings.
Packit 1fb8d4
 *
Packit 1fb8d4
 *  @return TRUE if successful. FALSE otherwise.
Packit 1fb8d4
 *
Packit 1fb8d4
 */
Packit 1fb8d4
BOOL freerdp_connect(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT status2 = CHANNEL_RC_OK;
Packit 1fb8d4
	rdpRdp* rdp;
Packit 1fb8d4
	BOOL status = TRUE;
Packit 1fb8d4
	rdpSettings* settings;
Packit 1fb8d4
	ConnectionResultEventArgs e;
Packit 1fb8d4
Packit 1fb8d4
	if (!instance)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	/* We always set the return code to 0 before we start the connect sequence*/
Packit 1fb8d4
	instance->ConnectionCallbackState = CLIENT_STATE_INITIAL;
Packit 1fb8d4
	connectErrorCode = 0;
Packit 1fb8d4
	instance->context->LastError = FREERDP_ERROR_SUCCESS;
Packit 1fb8d4
	clearChannelError(instance->context);
Packit 1fb8d4
	ResetEvent(instance->context->abortEvent);
Packit 1fb8d4
	rdp = instance->context->rdp;
Packit 1fb8d4
	settings = instance->settings;
Packit 1fb8d4
	IFCALLRET(instance->PreConnect, status, instance);
Packit 1fb8d4
	instance->ConnectionCallbackState = CLIENT_STATE_PRECONNECT_PASSED;
Packit 1fb8d4
Packit 1fb8d4
	if (status)
Packit 1fb8d4
		status2 = freerdp_channels_pre_connect(instance->context->channels,
Packit 1fb8d4
		                                       instance);
Packit 1fb8d4
Packit 1fb8d4
	if (settings->KeyboardLayout == KBD_JAPANESE_INPUT_SYSTEM_MS_IME2002)
Packit 1fb8d4
	{
Packit 1fb8d4
		settings->KeyboardType = 7;
Packit 1fb8d4
		settings->KeyboardSubType = 2;
Packit 1fb8d4
		settings->KeyboardFunctionKey = 12;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (!status || (status2 != CHANNEL_RC_OK))
Packit 1fb8d4
	{
Packit 1fb8d4
		if (!freerdp_get_last_error(rdp->context))
Packit 1fb8d4
			freerdp_set_last_error(instance->context, FREERDP_ERROR_PRE_CONNECT_FAILED);
Packit 1fb8d4
Packit 1fb8d4
		WLog_ERR(TAG, "freerdp_pre_connect failed");
Packit 1fb8d4
		goto freerdp_connect_finally;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	status = rdp_client_connect(rdp);
Packit 1fb8d4
Packit 1fb8d4
	/* --authonly tests the connection without a UI */
Packit 1fb8d4
	if (instance->settings->AuthenticationOnly)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "Authentication only, exit status %"PRId32"", !status);
Packit 1fb8d4
		goto freerdp_connect_finally;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (instance->settings->DumpRemoteFx)
Packit 1fb8d4
	{
Packit 1fb8d4
		instance->update->pcap_rfx = pcap_open(instance->settings->DumpRemoteFxFile,
Packit 1fb8d4
		                                       TRUE);
Packit 1fb8d4
Packit 1fb8d4
		if (instance->update->pcap_rfx)
Packit 1fb8d4
			instance->update->dump_rfx = TRUE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (status)
Packit 1fb8d4
	{
Packit 1fb8d4
		pointer_cache_register_callbacks(instance->context->update);
Packit 1fb8d4
		IFCALLRET(instance->PostConnect, status, instance);
Packit 1fb8d4
		instance->ConnectionCallbackState = CLIENT_STATE_POSTCONNECT_PASSED;
Packit 1fb8d4
Packit 1fb8d4
		if (status)
Packit 1fb8d4
			status2 = freerdp_channels_post_connect(instance->context->channels, instance);
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		if (freerdp_get_last_error(instance->context) == FREERDP_ERROR_CONNECT_TRANSPORT_FAILED)
Packit 1fb8d4
			status = freerdp_reconnect(instance);
Packit 1fb8d4
		else
Packit 1fb8d4
			goto freerdp_connect_finally;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (!status || (status2 != CHANNEL_RC_OK)
Packit 1fb8d4
	    || !update_post_connect(instance->update))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "freerdp_post_connect failed");
Packit 1fb8d4
Packit 1fb8d4
		if (!freerdp_get_last_error(rdp->context))
Packit 1fb8d4
			freerdp_set_last_error(instance->context, FREERDP_ERROR_POST_CONNECT_FAILED);
Packit 1fb8d4
Packit 1fb8d4
		status = FALSE;
Packit 1fb8d4
		goto freerdp_connect_finally;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (instance->settings->PlayRemoteFx)
Packit 1fb8d4
	{
Packit 1fb8d4
		wStream* s;
Packit 1fb8d4
		rdpUpdate* update;
Packit 1fb8d4
		pcap_record record;
Packit 1fb8d4
		update = instance->update;
Packit 1fb8d4
		update->pcap_rfx = pcap_open(settings->PlayRemoteFxFile, FALSE);
Packit 1fb8d4
		status = FALSE;
Packit 1fb8d4
Packit 1fb8d4
		if (!update->pcap_rfx)
Packit 1fb8d4
			goto freerdp_connect_finally;
Packit 1fb8d4
		else
Packit 1fb8d4
			update->play_rfx = TRUE;
Packit 1fb8d4
Packit 1fb8d4
		status = TRUE;
Packit 1fb8d4
Packit 1fb8d4
		while (pcap_has_next_record(update->pcap_rfx) && status)
Packit 1fb8d4
		{
Packit 1fb8d4
			pcap_get_next_record_header(update->pcap_rfx, &record);
Packit 1fb8d4
Packit 1fb8d4
			if (!(s = StreamPool_Take(rdp->transport->ReceivePool, record.length)))
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			record.data = Stream_Buffer(s);
Packit 1fb8d4
			pcap_get_next_record_content(update->pcap_rfx, &record);
Packit 1fb8d4
			Stream_SetLength(s, record.length);
Packit 1fb8d4
			Stream_SetPosition(s, 0);
Packit 1fb8d4
Packit 1fb8d4
			if (!update->BeginPaint(update->context))
Packit 1fb8d4
				status = FALSE;
Packit 1fb8d4
			else if (update_recv_surfcmds(update, s) < 0)
Packit 1fb8d4
				status = FALSE;
Packit 1fb8d4
			else if (!update->EndPaint(update->context))
Packit 1fb8d4
				status = FALSE;
Packit 1fb8d4
Packit 1fb8d4
			Stream_Release(s);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		pcap_close(update->pcap_rfx);
Packit 1fb8d4
		update->pcap_rfx = NULL;
Packit 1fb8d4
		goto freerdp_connect_finally;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (rdp->errorInfo == ERRINFO_SERVER_INSUFFICIENT_PRIVILEGES)
Packit 1fb8d4
		freerdp_set_last_error(instance->context,
Packit 1fb8d4
		                       FREERDP_ERROR_INSUFFICIENT_PRIVILEGES);
Packit 1fb8d4
Packit 1fb8d4
	SetEvent(rdp->transport->connectedEvent);
Packit 1fb8d4
freerdp_connect_finally:
Packit 1fb8d4
	EventArgsInit(&e, "freerdp");
Packit 1fb8d4
	e.result = status ? 0 : -1;
Packit 1fb8d4
	PubSub_OnConnectionResult(instance->context->pubSub, instance->context, &e);
Packit 1fb8d4
Packit 1fb8d4
	if (!status)
Packit 1fb8d4
		freerdp_disconnect(instance);
Packit 1fb8d4
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_abort_connect(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!instance || !instance->context)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return SetEvent(instance->context->abortEvent);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_get_fds(freerdp* instance, void** rfds, int* rcount, void** wfds,
Packit 1fb8d4
                     int* wcount)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpRdp* rdp = instance->context->rdp;
Packit 1fb8d4
	transport_get_fds(rdp->transport, rfds, rcount);
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_check_fds(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	int status;
Packit 1fb8d4
	rdpRdp* rdp;
Packit 1fb8d4
Packit 1fb8d4
	if (!instance)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (!instance->context)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (!instance->context->rdp)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	rdp = instance->context->rdp;
Packit 1fb8d4
	status = rdp_check_fds(rdp);
Packit 1fb8d4
Packit 1fb8d4
	if (status < 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		TerminateEventArgs e;
Packit 1fb8d4
		rdpContext* context = instance->context;
Packit 1fb8d4
		WLog_DBG(TAG, "rdp_check_fds() - %i", status);
Packit 1fb8d4
		EventArgsInit(&e, "freerdp");
Packit 1fb8d4
		e.code = 0;
Packit 1fb8d4
		PubSub_OnTerminate(context->pubSub, context, &e);
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
DWORD freerdp_get_event_handles(rdpContext* context, HANDLE* events,
Packit 1fb8d4
                                DWORD count)
Packit 1fb8d4
{
Packit 1fb8d4
	DWORD nCount = 0;
Packit 1fb8d4
	nCount += transport_get_event_handles(context->rdp->transport, events, count);
Packit 1fb8d4
Packit 1fb8d4
	if (nCount == 0)
Packit 1fb8d4
		return 0;
Packit 1fb8d4
Packit 1fb8d4
	if (events && (nCount < count + 2))
Packit 1fb8d4
	{
Packit 1fb8d4
		events[nCount++] = freerdp_channels_get_event_handle(context->instance);
Packit 1fb8d4
		events[nCount++] = getChannelErrorEventHandle(context);
Packit 1fb8d4
		events[nCount++] = context->abortEvent;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
		return 0;
Packit 1fb8d4
Packit 1fb8d4
	if (context->settings->AsyncInput)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (nCount >= count)
Packit 1fb8d4
			return 0;
Packit 1fb8d4
Packit 1fb8d4
		events[nCount++] = freerdp_get_message_queue_event_handle(
Packit 1fb8d4
		                       context->instance, FREERDP_INPUT_MESSAGE_QUEUE);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return nCount;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_check_event_handles(rdpContext* context)
Packit 1fb8d4
{
Packit 1fb8d4
	BOOL status;
Packit 1fb8d4
	status = freerdp_check_fds(context->instance);
Packit 1fb8d4
Packit 1fb8d4
	if (!status)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (freerdp_get_last_error(context) == FREERDP_ERROR_SUCCESS)
Packit 1fb8d4
			WLog_ERR(TAG, "freerdp_check_fds() failed - %"PRIi32"", status);
Packit 1fb8d4
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	status = freerdp_channels_check_fds(context->channels, context->instance);
Packit 1fb8d4
Packit 1fb8d4
	if (!status)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (freerdp_get_last_error(context) == FREERDP_ERROR_SUCCESS)
Packit 1fb8d4
			WLog_ERR(TAG, "freerdp_channels_check_fds() failed - %"PRIi32"", status);
Packit 1fb8d4
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (!status)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	status = checkChannelErrorEvent(context);
Packit 1fb8d4
Packit 1fb8d4
	if (!status)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (freerdp_get_last_error(context) == FREERDP_ERROR_SUCCESS)
Packit 1fb8d4
			WLog_ERR(TAG, "checkChannelErrorEvent() failed - %"PRIi32"", status);
Packit 1fb8d4
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (context->settings->AsyncInput)
Packit 1fb8d4
	{
Packit 1fb8d4
		int rc = freerdp_message_queue_process_pending_messages(
Packit 1fb8d4
		             context->instance, FREERDP_INPUT_MESSAGE_QUEUE);
Packit 1fb8d4
Packit 1fb8d4
		if (rc < 0)
Packit 1fb8d4
			return FALSE;
Packit 1fb8d4
		else
Packit 1fb8d4
			status = TRUE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
wMessageQueue* freerdp_get_message_queue(freerdp* instance, DWORD id)
Packit 1fb8d4
{
Packit 1fb8d4
	wMessageQueue* queue = NULL;
Packit 1fb8d4
Packit 1fb8d4
	switch (id)
Packit 1fb8d4
	{
Packit 1fb8d4
		case FREERDP_UPDATE_MESSAGE_QUEUE:
Packit 1fb8d4
			queue = instance->update->queue;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_INPUT_MESSAGE_QUEUE:
Packit 1fb8d4
			queue = instance->input->queue;
Packit 1fb8d4
			break;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return queue;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
HANDLE freerdp_get_message_queue_event_handle(freerdp* instance, DWORD id)
Packit 1fb8d4
{
Packit 1fb8d4
	HANDLE event = NULL;
Packit 1fb8d4
	wMessageQueue* queue = NULL;
Packit 1fb8d4
	queue = freerdp_get_message_queue(instance, id);
Packit 1fb8d4
Packit 1fb8d4
	if (queue)
Packit 1fb8d4
		event = MessageQueue_Event(queue);
Packit 1fb8d4
Packit 1fb8d4
	return event;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
int freerdp_message_queue_process_message(freerdp* instance, DWORD id,
Packit 1fb8d4
        wMessage* message)
Packit 1fb8d4
{
Packit 1fb8d4
	int status = -1;
Packit 1fb8d4
Packit 1fb8d4
	switch (id)
Packit 1fb8d4
	{
Packit 1fb8d4
		case FREERDP_UPDATE_MESSAGE_QUEUE:
Packit 1fb8d4
			status = update_message_queue_process_message(instance->update, message);
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_INPUT_MESSAGE_QUEUE:
Packit 1fb8d4
			status = input_message_queue_process_message(instance->input, message);
Packit 1fb8d4
			break;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
int freerdp_message_queue_process_pending_messages(freerdp* instance, DWORD id)
Packit 1fb8d4
{
Packit 1fb8d4
	int status = -1;
Packit 1fb8d4
Packit 1fb8d4
	switch (id)
Packit 1fb8d4
	{
Packit 1fb8d4
		case FREERDP_UPDATE_MESSAGE_QUEUE:
Packit 1fb8d4
			status = update_message_queue_process_pending_messages(instance->update);
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_INPUT_MESSAGE_QUEUE:
Packit 1fb8d4
			status = input_message_queue_process_pending_messages(instance->input);
Packit 1fb8d4
			break;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static int freerdp_send_channel_data(freerdp* instance, UINT16 channelId,
Packit 1fb8d4
                                     BYTE* data, int size)
Packit 1fb8d4
{
Packit 1fb8d4
	return rdp_send_channel_data(instance->context->rdp, channelId, data, size);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_disconnect(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	BOOL rc = TRUE;
Packit 1fb8d4
	rdpRdp* rdp;
Packit 1fb8d4
Packit 1fb8d4
	if (!instance || !instance->context || !instance->context->rdp)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	rdp = instance->context->rdp;
Packit 1fb8d4
Packit 1fb8d4
	if (!rdp_client_disconnect(rdp))
Packit 1fb8d4
		rc = FALSE;
Packit 1fb8d4
Packit 1fb8d4
	update_post_disconnect(instance->update);
Packit 1fb8d4
Packit 1fb8d4
	if (instance->settings->AsyncInput)
Packit 1fb8d4
	{
Packit 1fb8d4
		wMessageQueue* inputQueue = freerdp_get_message_queue(instance,
Packit 1fb8d4
		                            FREERDP_INPUT_MESSAGE_QUEUE);
Packit 1fb8d4
		MessageQueue_PostQuit(inputQueue, 0);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	IFCALL(instance->PostDisconnect, instance);
Packit 1fb8d4
Packit 1fb8d4
	if (instance->update->pcap_rfx)
Packit 1fb8d4
	{
Packit 1fb8d4
		instance->update->dump_rfx = FALSE;
Packit 1fb8d4
		pcap_close(instance->update->pcap_rfx);
Packit 1fb8d4
		instance->update->pcap_rfx = NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	freerdp_channels_close(instance->context->channels, instance);
Packit 1fb8d4
	return rc;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_disconnect_before_reconnect(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpRdp* rdp = instance->context->rdp;
Packit 1fb8d4
	return rdp_client_disconnect_and_clear(rdp);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_reconnect(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	BOOL status;
Packit 1fb8d4
	rdpRdp* rdp = instance->context->rdp;
Packit 1fb8d4
	ResetEvent(instance->context->abortEvent);
Packit 1fb8d4
	status = rdp_client_reconnect(rdp);
Packit 1fb8d4
	return status;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_shall_disconnect(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!instance || !instance->context)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (WaitForSingleObject(instance->context->abortEvent, 0) != WAIT_OBJECT_0)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL freerdp_focus_required(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpRdp* rdp;
Packit 1fb8d4
	BOOL bRetCode = FALSE;
Packit 1fb8d4
	rdp = instance->context->rdp;
Packit 1fb8d4
Packit 1fb8d4
	if (rdp->resendFocus)
Packit 1fb8d4
	{
Packit 1fb8d4
		bRetCode = TRUE;
Packit 1fb8d4
		rdp->resendFocus = FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return bRetCode;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void freerdp_set_focus(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpRdp* rdp;
Packit 1fb8d4
	rdp = instance->context->rdp;
Packit 1fb8d4
	rdp->resendFocus = TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void freerdp_get_version(int* major, int* minor, int* revision)
Packit 1fb8d4
{
Packit 1fb8d4
	if (major != NULL)
Packit 1fb8d4
		*major = FREERDP_VERSION_MAJOR;
Packit 1fb8d4
Packit 1fb8d4
	if (minor != NULL)
Packit 1fb8d4
		*minor = FREERDP_VERSION_MINOR;
Packit 1fb8d4
Packit 1fb8d4
	if (revision != NULL)
Packit 1fb8d4
		*revision = FREERDP_VERSION_REVISION;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* freerdp_get_version_string(void)
Packit 1fb8d4
{
Packit 1fb8d4
	return FREERDP_VERSION_FULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* freerdp_get_build_date(void)
Packit 1fb8d4
{
Packit 1fb8d4
	static char build_date[] = __DATE__ " " __TIME__;
Packit 1fb8d4
	return build_date;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* freerdp_get_build_config(void)
Packit 1fb8d4
{
Packit 1fb8d4
	static const char build_config[] =
Packit 1fb8d4
	    "Build configuration: " BUILD_CONFIG "\n"
Packit 1fb8d4
	    "Build type:          " BUILD_TYPE "\n"
Packit 1fb8d4
	    "CFLAGS:              " CFLAGS "\n"
Packit 1fb8d4
	    "Compiler:            " COMPILER_ID ", " COMPILER_VERSION "\n"
Packit 1fb8d4
	    "Target architecture: " TARGET_ARCH "\n";
Packit 1fb8d4
	return build_config;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* freerdp_get_build_revision(void)
Packit 1fb8d4
{
Packit 1fb8d4
	return GIT_REVISION;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static wEventType FreeRDP_Events[] =
Packit 1fb8d4
{
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(WindowStateChange)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(ResizeWindow)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(LocalResizeWindow)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(EmbedWindow)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(PanningChange)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(ZoomingChange)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(ErrorInfo)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(Terminate)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(ConnectionResult)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(ChannelConnected)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(ChannelDisconnected)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(MouseEvent)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(Activated)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(Timer)
Packit 1fb8d4
	DEFINE_EVENT_ENTRY(GraphicsReset)
Packit 1fb8d4
};
Packit 1fb8d4
Packit 1fb8d4
/** Allocator function for a rdp context.
Packit 1fb8d4
 *  The function will allocate a rdpRdp structure using rdp_new(), then copy
Packit 1fb8d4
 *  its contents to the appropriate fields in the rdp_freerdp structure given in parameters.
Packit 1fb8d4
 *  It will also initialize the 'context' field in the rdp_freerdp structure as needed.
Packit 1fb8d4
 *  If the caller has set the ContextNew callback in the 'instance' parameter, it will be called at the end of the function.
Packit 1fb8d4
 *
Packit 1fb8d4
 *  @param instance - Pointer to the rdp_freerdp structure that will be initialized with the new context.
Packit 1fb8d4
 */
Packit 1fb8d4
BOOL freerdp_context_new(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	rdpRdp* rdp;
Packit 1fb8d4
	rdpContext* context;
Packit 1fb8d4
	BOOL ret = TRUE;
Packit 1fb8d4
	instance->context = (rdpContext*) calloc(1, instance->ContextSize);
Packit 1fb8d4
Packit 1fb8d4
	if (!instance->context)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	context = instance->context;
Packit 1fb8d4
	context->instance = instance;
Packit 1fb8d4
	context->ServerMode = FALSE;
Packit 1fb8d4
	context->settings = instance->settings;
Packit 1fb8d4
	context->disconnectUltimatum = 0;
Packit 1fb8d4
	context->pubSub = PubSub_New(TRUE);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->pubSub)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	PubSub_AddEventTypes(context->pubSub, FreeRDP_Events,
Packit 1fb8d4
	                     ARRAYSIZE(FreeRDP_Events));
Packit 1fb8d4
	context->metrics = metrics_new(context);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->metrics)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	rdp = rdp_new(context);
Packit 1fb8d4
Packit 1fb8d4
	if (!rdp)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	instance->input = rdp->input;
Packit 1fb8d4
	instance->update = rdp->update;
Packit 1fb8d4
	instance->settings = rdp->settings;
Packit 1fb8d4
	instance->autodetect = rdp->autodetect;
Packit 1fb8d4
	context->graphics = graphics_new(context);
Packit 1fb8d4
Packit 1fb8d4
	if (!context->graphics)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	context->rdp = rdp;
Packit 1fb8d4
	context->input = instance->input;
Packit 1fb8d4
	context->update = instance->update;
Packit 1fb8d4
	context->settings = instance->settings;
Packit 1fb8d4
	context->autodetect = instance->autodetect;
Packit 1fb8d4
	instance->update->context = instance->context;
Packit 1fb8d4
	instance->update->pointer->context = instance->context;
Packit 1fb8d4
	instance->update->primary->context = instance->context;
Packit 1fb8d4
	instance->update->secondary->context = instance->context;
Packit 1fb8d4
	instance->update->altsec->context = instance->context;
Packit 1fb8d4
	instance->input->context = context;
Packit 1fb8d4
	instance->autodetect->context = context;
Packit 1fb8d4
Packit 1fb8d4
	if (!(context->errorDescription = calloc(1, 500)))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "calloc failed!");
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (!(context->channelErrorEvent = CreateEvent(NULL, TRUE, FALSE, NULL)))
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "CreateEvent failed!");
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	update_register_client_callbacks(rdp->update);
Packit 1fb8d4
	instance->context->abortEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
Packit 1fb8d4
Packit 1fb8d4
	if (!instance->context->abortEvent)
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	if (!(context->channels = freerdp_channels_new(instance)))
Packit 1fb8d4
		goto fail;
Packit 1fb8d4
Packit 1fb8d4
	IFCALLRET(instance->ContextNew, ret, instance, instance->context);
Packit 1fb8d4
Packit 1fb8d4
	if (ret)
Packit 1fb8d4
		return TRUE;
Packit 1fb8d4
Packit 1fb8d4
fail:
Packit 1fb8d4
	freerdp_context_free(instance);
Packit 1fb8d4
	return FALSE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/** Deallocator function for a rdp context.
Packit 1fb8d4
 *  The function will deallocate the resources from the 'instance' parameter that were allocated from a call
Packit 1fb8d4
 *  to freerdp_context_new().
Packit 1fb8d4
 *  If the ContextFree callback is set in the 'instance' parameter, it will be called before deallocation occurs.
Packit 1fb8d4
 *
Packit 1fb8d4
 *  @param instance - Pointer to the rdp_freerdp structure that was initialized by a call to freerdp_context_new().
Packit 1fb8d4
 *  				  On return, the fields associated to the context are invalid.
Packit 1fb8d4
 */
Packit 1fb8d4
void freerdp_context_free(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!instance)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	if (!instance->context)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	IFCALL(instance->ContextFree, instance, instance->context);
Packit 1fb8d4
	rdp_free(instance->context->rdp);
Packit 1fb8d4
	instance->context->rdp = NULL;
Packit 1fb8d4
	graphics_free(instance->context->graphics);
Packit 1fb8d4
	instance->context->graphics = NULL;
Packit 1fb8d4
	PubSub_Free(instance->context->pubSub);
Packit 1fb8d4
	metrics_free(instance->context->metrics);
Packit 1fb8d4
	CloseHandle(instance->context->channelErrorEvent);
Packit 1fb8d4
	free(instance->context->errorDescription);
Packit 1fb8d4
	CloseHandle(instance->context->abortEvent);
Packit 1fb8d4
	instance->context->abortEvent = NULL;
Packit 1fb8d4
	freerdp_channels_free(instance->context->channels);
Packit 1fb8d4
	free(instance->context);
Packit 1fb8d4
	instance->context = NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
int freerdp_get_disconnect_ultimatum(rdpContext* context)
Packit 1fb8d4
{
Packit 1fb8d4
	return context->disconnectUltimatum;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
UINT32 freerdp_error_info(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	return instance->context->rdp->errorInfo;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void freerdp_set_error_info(rdpRdp* rdp, UINT32 error)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!rdp)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	rdp_set_error_info(rdp, error);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
UINT32 freerdp_get_last_error(rdpContext* context)
Packit 1fb8d4
{
Packit 1fb8d4
	return context->LastError;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* freerdp_get_last_error_name(UINT32 code)
Packit 1fb8d4
{
Packit 1fb8d4
	const char* name = NULL;
Packit 1fb8d4
	const UINT32 cls = GET_FREERDP_ERROR_CLASS(code);
Packit 1fb8d4
	const UINT32 type = GET_FREERDP_ERROR_TYPE(code);
Packit 1fb8d4
Packit 1fb8d4
	switch (cls)
Packit 1fb8d4
	{
Packit 1fb8d4
		case FREERDP_ERROR_ERRBASE_CLASS:
Packit 1fb8d4
			name = freerdp_get_error_base_name(type);
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_ERRINFO_CLASS:
Packit 1fb8d4
			name = freerdp_get_error_info_name(type);
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_CONNECT_CLASS:
Packit 1fb8d4
			name = freerdp_get_error_connect_name(type);
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			name = "Unknown error class";
Packit 1fb8d4
			break;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return name;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* freerdp_get_last_error_string(UINT32 code)
Packit 1fb8d4
{
Packit 1fb8d4
	const char* string = NULL;
Packit 1fb8d4
	const UINT32 cls = GET_FREERDP_ERROR_CLASS(code);
Packit 1fb8d4
	const UINT32 type = GET_FREERDP_ERROR_TYPE(code);
Packit 1fb8d4
Packit 1fb8d4
	switch (cls)
Packit 1fb8d4
	{
Packit 1fb8d4
		case FREERDP_ERROR_ERRBASE_CLASS:
Packit 1fb8d4
			string = freerdp_get_error_base_string(type);
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_ERRINFO_CLASS:
Packit 1fb8d4
			string = freerdp_get_error_info_string(type);
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_CONNECT_CLASS:
Packit 1fb8d4
			string = freerdp_get_error_connect_string(type);
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			string = "Unknown error class";
Packit 1fb8d4
			break;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return string;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void freerdp_set_last_error(rdpContext* context, UINT32 lastError)
Packit 1fb8d4
{
Packit 1fb8d4
	if (lastError)
Packit 1fb8d4
		WLog_ERR(TAG, "freerdp_set_last_error %s [0x%08"PRIX32"]",
Packit 1fb8d4
		         freerdp_get_last_error_name(lastError), lastError);
Packit 1fb8d4
Packit 1fb8d4
	if (context->LastError != 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "TODO: Trying to set error code %s, but %s already set!",
Packit 1fb8d4
		         freerdp_get_last_error_name(lastError),
Packit 1fb8d4
		         freerdp_get_last_error_name(context->LastError));
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	context->LastError = lastError;
Packit 1fb8d4
Packit 1fb8d4
	switch (lastError)
Packit 1fb8d4
	{
Packit 1fb8d4
		case FREERDP_ERROR_PRE_CONNECT_FAILED:
Packit 1fb8d4
			connectErrorCode = PREECONNECTERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_CONNECT_UNDEFINED:
Packit 1fb8d4
			connectErrorCode = UNDEFINEDCONNECTERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_POST_CONNECT_FAILED:
Packit 1fb8d4
			connectErrorCode = POSTCONNECTERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_DNS_ERROR:
Packit 1fb8d4
			connectErrorCode = DNSERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_DNS_NAME_NOT_FOUND:
Packit 1fb8d4
			connectErrorCode = DNSNAMENOTFOUND;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_CONNECT_FAILED:
Packit 1fb8d4
			connectErrorCode = CONNECTERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_MCS_CONNECT_INITIAL_ERROR:
Packit 1fb8d4
			connectErrorCode = MCSCONNECTINITIALERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_TLS_CONNECT_FAILED:
Packit 1fb8d4
			connectErrorCode = TLSCONNECTERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_AUTHENTICATION_FAILED:
Packit 1fb8d4
			connectErrorCode = AUTHENTICATIONERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_INSUFFICIENT_PRIVILEGES:
Packit 1fb8d4
			connectErrorCode = INSUFFICIENTPRIVILEGESERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_CONNECT_CANCELLED:
Packit 1fb8d4
			connectErrorCode = CANCELEDBYUSER;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_SECURITY_NEGO_CONNECT_FAILED:
Packit 1fb8d4
			connectErrorCode = CONNECTERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		case FREERDP_ERROR_CONNECT_TRANSPORT_FAILED:
Packit 1fb8d4
			connectErrorCode = CONNECTERROR;
Packit 1fb8d4
			break;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* freerdp_get_logon_error_info_type(UINT32 type)
Packit 1fb8d4
{
Packit 1fb8d4
	switch (type)
Packit 1fb8d4
	{
Packit 1fb8d4
		case LOGON_MSG_DISCONNECT_REFUSED:
Packit 1fb8d4
			return "LOGON_MSG_DISCONNECT_REFUSED";
Packit 1fb8d4
Packit 1fb8d4
		case LOGON_MSG_NO_PERMISSION:
Packit 1fb8d4
			return "LOGON_MSG_NO_PERMISSION";
Packit 1fb8d4
Packit 1fb8d4
		case LOGON_MSG_BUMP_OPTIONS:
Packit 1fb8d4
			return "LOGON_MSG_BUMP_OPTIONS";
Packit 1fb8d4
Packit 1fb8d4
		case LOGON_MSG_RECONNECT_OPTIONS:
Packit 1fb8d4
			return "LOGON_MSG_RECONNECT_OPTIONS";
Packit 1fb8d4
Packit 1fb8d4
		case LOGON_MSG_SESSION_TERMINATE:
Packit 1fb8d4
			return "LOGON_MSG_SESSION_TERMINATE";
Packit 1fb8d4
Packit 1fb8d4
		case LOGON_MSG_SESSION_CONTINUE:
Packit 1fb8d4
			return "LOGON_MSG_SESSION_CONTINUE";
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return "UNKNOWN";
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* freerdp_get_logon_error_info_data(UINT32 data)
Packit 1fb8d4
{
Packit 1fb8d4
	switch (data)
Packit 1fb8d4
	{
Packit 1fb8d4
		case LOGON_FAILED_BAD_PASSWORD:
Packit 1fb8d4
			return "LOGON_FAILED_BAD_PASSWORD";
Packit 1fb8d4
Packit 1fb8d4
		case LOGON_FAILED_UPDATE_PASSWORD:
Packit 1fb8d4
			return "LOGON_FAILED_UPDATE_PASSWORD";
Packit 1fb8d4
Packit 1fb8d4
		case LOGON_FAILED_OTHER:
Packit 1fb8d4
			return "LOGON_FAILED_OTHER";
Packit 1fb8d4
Packit 1fb8d4
		case LOGON_WARNING:
Packit 1fb8d4
			return "LOGON_WARNING";
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return "SESSION_ID";
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/** Allocator function for the rdp_freerdp structure.
Packit 1fb8d4
 *  @return an allocated structure filled with 0s. Need to be deallocated using freerdp_free()
Packit 1fb8d4
 */
Packit 1fb8d4
freerdp* freerdp_new()
Packit 1fb8d4
{
Packit 1fb8d4
	freerdp* instance;
Packit 1fb8d4
	instance = (freerdp*) calloc(1, sizeof(freerdp));
Packit 1fb8d4
Packit 1fb8d4
	if (!instance)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	instance->ContextSize = sizeof(rdpContext);
Packit 1fb8d4
	instance->SendChannelData = freerdp_send_channel_data;
Packit 1fb8d4
	instance->ReceiveChannelData = freerdp_channels_data;
Packit 1fb8d4
	return instance;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/** Deallocator function for the rdp_freerdp structure.
Packit 1fb8d4
 *  @param instance - pointer to the rdp_freerdp structure to deallocate.
Packit 1fb8d4
 *                    On return, this pointer is not valid anymore.
Packit 1fb8d4
 */
Packit 1fb8d4
void freerdp_free(freerdp* instance)
Packit 1fb8d4
{
Packit 1fb8d4
	free(instance);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
ULONG freerdp_get_transport_sent(rdpContext* context, BOOL resetCount)
Packit 1fb8d4
{
Packit 1fb8d4
	ULONG written = context->rdp->transport->written;
Packit 1fb8d4
Packit 1fb8d4
	if (resetCount)
Packit 1fb8d4
		context->rdp->transport->written = 0;
Packit 1fb8d4
Packit 1fb8d4
	return written;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
HANDLE getChannelErrorEventHandle(rdpContext* context)
Packit 1fb8d4
{
Packit 1fb8d4
	return context->channelErrorEvent;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL checkChannelErrorEvent(rdpContext* context)
Packit 1fb8d4
{
Packit 1fb8d4
	if (WaitForSingleObject(context->channelErrorEvent, 0) == WAIT_OBJECT_0)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "%s. Error was %"PRIu32"", context->errorDescription,
Packit 1fb8d4
		         context->channelErrorNum);
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
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 1fb8d4
UINT getChannelError(rdpContext* context)
Packit 1fb8d4
{
Packit 1fb8d4
	return context->channelErrorNum;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
const char* getChannelErrorDescription(rdpContext* context)
Packit 1fb8d4
{
Packit 1fb8d4
	return context->errorDescription;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void clearChannelError(rdpContext* context)
Packit 1fb8d4
{
Packit 1fb8d4
	context->channelErrorNum = 0;
Packit 1fb8d4
	memset(context->errorDescription, 0, 500);
Packit 1fb8d4
	ResetEvent(context->channelErrorEvent);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void setChannelError(rdpContext* context, UINT errorNum, char* description)
Packit 1fb8d4
{
Packit 1fb8d4
	context->channelErrorNum = errorNum;
Packit 1fb8d4
	strncpy(context->errorDescription, description, 499);
Packit 1fb8d4
	SetEvent(context->channelErrorEvent);
Packit 1fb8d4
}