Blame client/Sample/tf_freerdp.c

Packit Service b1ea74
/**
Packit Service b1ea74
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit Service b1ea74
 * FreeRDP Test UI
Packit Service b1ea74
 *
Packit Service b1ea74
 * Copyright 2011 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit Service b1ea74
 * Copyright 2016,2018 Armin Novak <armin.novak@thincast.com>
Packit Service b1ea74
 * Copyright 2016,2018 Thincast Technologies GmbH
Packit Service b1ea74
 *
Packit Service b1ea74
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit Service b1ea74
 * you may not use this file except in compliance with the License.
Packit Service b1ea74
 * You may obtain a copy of the License at
Packit Service b1ea74
 *
Packit Service b1ea74
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit Service b1ea74
 *
Packit Service b1ea74
 * Unless required by applicable law or agreed to in writing, software
Packit Service b1ea74
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit Service b1ea74
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit Service b1ea74
 * See the License for the specific language governing permissions and
Packit Service b1ea74
 * limitations under the License.
Packit Service b1ea74
 */
Packit Service b1ea74
Packit Service b1ea74
#ifdef HAVE_CONFIG_H
Packit Service b1ea74
#include "config.h"
Packit Service b1ea74
#endif
Packit Service b1ea74
Packit Service b1ea74
#include <errno.h>
Packit Service b1ea74
#include <stdio.h>
Packit Service b1ea74
#include <string.h>
Packit Service b1ea74
Packit Service b1ea74
#include <freerdp/freerdp.h>
Packit Service b1ea74
#include <freerdp/constants.h>
Packit Service b1ea74
#include <freerdp/gdi/gdi.h>
Packit Service b1ea74
#include <freerdp/utils/signal.h>
Packit Service b1ea74
Packit Service b1ea74
#include <freerdp/client/file.h>
Packit Service b1ea74
#include <freerdp/client/cmdline.h>
Packit Service b1ea74
#include <freerdp/client/cliprdr.h>
Packit Service b1ea74
#include <freerdp/client/channels.h>
Packit Service b1ea74
#include <freerdp/channels/channels.h>
Packit Service b1ea74
Packit Service b1ea74
#include <winpr/crt.h>
Packit Service b1ea74
#include <winpr/synch.h>
Packit Service b1ea74
#include <freerdp/log.h>
Packit Service b1ea74
Packit Service b1ea74
#include "tf_channels.h"
Packit Service b1ea74
#include "tf_freerdp.h"
Packit Service b1ea74
Packit Service b1ea74
#define TAG CLIENT_TAG("sample")
Packit Service b1ea74
Packit Service b1ea74
/* This function is called whenever a new frame starts.
Packit Service b1ea74
 * It can be used to reset invalidated areas. */
Packit Service b1ea74
static BOOL tf_begin_paint(rdpContext* context)
Packit Service b1ea74
{
Packit Service b1ea74
	rdpGdi* gdi = context->gdi;
Packit Service b1ea74
	gdi->primary->hdc->hwnd->invalid->null = TRUE;
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* This function is called when the library completed composing a new
Packit Service b1ea74
 * frame. Read out the changed areas and blit them to your output device.
Packit Service b1ea74
 * The image buffer will have the format specified by gdi_init
Packit Service b1ea74
 */
Packit Service b1ea74
static BOOL tf_end_paint(rdpContext* context)
Packit Service b1ea74
{
Packit Service b1ea74
	rdpGdi* gdi = context->gdi;
Packit Service b1ea74
Packit Service b1ea74
	if (gdi->primary->hdc->hwnd->invalid->null)
Packit Service b1ea74
		return TRUE;
Packit Service b1ea74
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* This function is called to output a System BEEP */
Packit Service b1ea74
static BOOL tf_play_sound(rdpContext* context, const PLAY_SOUND_UPDATE* play_sound)
Packit Service b1ea74
{
Packit Service b1ea74
	/* TODO: Implement */
Packit Service b1ea74
	WINPR_UNUSED(context);
Packit Service b1ea74
	WINPR_UNUSED(play_sound);
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* This function is called to update the keyboard indocator LED */
Packit Service b1ea74
static BOOL tf_keyboard_set_indicators(rdpContext* context, UINT16 led_flags)
Packit Service b1ea74
{
Packit Service b1ea74
	/* TODO: Set local keyboard indicator LED status */
Packit Service b1ea74
	WINPR_UNUSED(context);
Packit Service b1ea74
	WINPR_UNUSED(led_flags);
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* This function is called to set the IME state */
Packit Service b1ea74
static BOOL tf_keyboard_set_ime_status(rdpContext* context, UINT16 imeId, UINT32 imeState,
Packit Service b1ea74
                                       UINT32 imeConvMode)
Packit Service b1ea74
{
Packit Service b1ea74
	if (!context)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	WLog_WARN(TAG,
Packit Service b1ea74
	          "KeyboardSetImeStatus(unitId=%04" PRIx16 ", imeState=%08" PRIx32
Packit Service b1ea74
	          ", imeConvMode=%08" PRIx32 ") ignored",
Packit Service b1ea74
	          imeId, imeState, imeConvMode);
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* Called before a connection is established.
Packit Service b1ea74
 * Set all configuration options to support and load channels here. */
Packit Service b1ea74
static BOOL tf_pre_connect(freerdp* instance)
Packit Service b1ea74
{
Packit Service b1ea74
	rdpSettings* settings;
Packit Service b1ea74
	settings = instance->settings;
Packit Service b1ea74
	/* Optional OS identifier sent to server */
Packit Service b1ea74
	settings->OsMajorType = OSMAJORTYPE_UNIX;
Packit Service b1ea74
	settings->OsMinorType = OSMINORTYPE_NATIVE_XSERVER;
Packit Service b1ea74
	/* settings->OrderSupport is initialized at this point.
Packit Service b1ea74
	 * Only override it if you plan to implement custom order
Packit Service b1ea74
	 * callbacks or deactiveate certain features. */
Packit Service b1ea74
	/* Register the channel listeners.
Packit Service b1ea74
	 * They are required to set up / tear down channels if they are loaded. */
Packit Service b1ea74
	PubSub_SubscribeChannelConnected(instance->context->pubSub, tf_OnChannelConnectedEventHandler);
Packit Service b1ea74
	PubSub_SubscribeChannelDisconnected(instance->context->pubSub,
Packit Service b1ea74
	                                    tf_OnChannelDisconnectedEventHandler);
Packit Service b1ea74
Packit Service b1ea74
	/* Load all required plugins / channels / libraries specified by current
Packit Service b1ea74
	 * settings. */
Packit Service b1ea74
	if (!freerdp_client_load_addins(instance->context->channels, instance->settings))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	/* TODO: Any code your client requires */
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* Called after a RDP connection was successfully established.
Packit Service b1ea74
 * Settings might have changed during negociation of client / server feature
Packit Service b1ea74
 * support.
Packit Service b1ea74
 *
Packit Service b1ea74
 * Set up local framebuffers and paing callbacks.
Packit Service b1ea74
 * If required, register pointer callbacks to change the local mouse cursor
Packit Service b1ea74
 * when hovering over the RDP window
Packit Service b1ea74
 */
Packit Service b1ea74
static BOOL tf_post_connect(freerdp* instance)
Packit Service b1ea74
{
Packit Service b1ea74
	if (!gdi_init(instance, PIXEL_FORMAT_XRGB32))
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	instance->update->BeginPaint = tf_begin_paint;
Packit Service b1ea74
	instance->update->EndPaint = tf_end_paint;
Packit Service b1ea74
	instance->update->PlaySound = tf_play_sound;
Packit Service b1ea74
	instance->update->SetKeyboardIndicators = tf_keyboard_set_indicators;
Packit Service b1ea74
	instance->update->SetKeyboardImeStatus = tf_keyboard_set_ime_status;
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* This function is called whether a session ends by failure or success.
Packit Service b1ea74
 * Clean up everything allocated by pre_connect and post_connect.
Packit Service b1ea74
 */
Packit Service b1ea74
static void tf_post_disconnect(freerdp* instance)
Packit Service b1ea74
{
Packit Service b1ea74
	tfContext* context;
Packit Service b1ea74
Packit Service b1ea74
	if (!instance)
Packit Service b1ea74
		return;
Packit Service b1ea74
Packit Service b1ea74
	if (!instance->context)
Packit Service b1ea74
		return;
Packit Service b1ea74
Packit Service b1ea74
	context = (tfContext*)instance->context;
Packit Service b1ea74
	PubSub_UnsubscribeChannelConnected(instance->context->pubSub,
Packit Service b1ea74
	                                   tf_OnChannelConnectedEventHandler);
Packit Service b1ea74
	PubSub_UnsubscribeChannelDisconnected(instance->context->pubSub,
Packit Service b1ea74
	                                      tf_OnChannelDisconnectedEventHandler);
Packit Service b1ea74
	gdi_free(instance);
Packit Service b1ea74
	/* TODO : Clean up custom stuff */
Packit Service b1ea74
	WINPR_UNUSED(context);
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* RDP main loop.
Packit Service b1ea74
 * Connects RDP, loops while running and handles event and dispatch, cleans up
Packit Service b1ea74
 * after the connection ends. */
Packit Service b1ea74
static DWORD WINAPI tf_client_thread_proc(LPVOID arg)
Packit Service b1ea74
{
Packit Service b1ea74
	freerdp* instance = (freerdp*)arg;
Packit Service b1ea74
	DWORD nCount;
Packit Service b1ea74
	DWORD status;
Packit Service b1ea74
	DWORD result = 0;
Packit Service b1ea74
	HANDLE handles[64];
Packit Service b1ea74
	BOOL rc = freerdp_connect(instance);
Packit Service b1ea74
Packit Service b1ea74
	if (instance->settings->AuthenticationOnly)
Packit Service b1ea74
	{
Packit Service b1ea74
		result = freerdp_get_last_error(instance->context);
Packit Service b1ea74
		freerdp_abort_connect(instance);
Packit Service b1ea74
		WLog_ERR(TAG, "Authentication only, exit status 0x%08" PRIx32 "", result);
Packit Service b1ea74
		goto disconnect;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (!rc)
Packit Service b1ea74
	{
Packit Service b1ea74
		result = freerdp_get_last_error(instance->context);
Packit Service b1ea74
		WLog_ERR(TAG, "connection failure 0x%08" PRIx32, result);
Packit Service b1ea74
		return result;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	while (!freerdp_shall_disconnect(instance))
Packit Service b1ea74
	{
Packit Service b1ea74
		nCount = freerdp_get_event_handles(instance->context, &handles[0], 64);
Packit Service b1ea74
Packit Service b1ea74
		if (nCount == 0)
Packit Service b1ea74
		{
Packit Service b1ea74
			WLog_ERR(TAG, "%s: freerdp_get_event_handles failed", __FUNCTION__);
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		status = WaitForMultipleObjects(nCount, handles, FALSE, 100);
Packit Service b1ea74
Packit Service b1ea74
		if (status == WAIT_FAILED)
Packit Service b1ea74
		{
Packit Service b1ea74
			WLog_ERR(TAG, "%s: WaitForMultipleObjects failed with %" PRIu32 "", __FUNCTION__,
Packit Service b1ea74
			         status);
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
Packit Service b1ea74
		if (!freerdp_check_event_handles(instance->context))
Packit Service b1ea74
		{
Packit Service b1ea74
			if (freerdp_get_last_error(instance->context) == FREERDP_ERROR_SUCCESS)
Packit Service b1ea74
				WLog_ERR(TAG, "Failed to check FreeRDP event handles");
Packit Service b1ea74
Packit Service b1ea74
			break;
Packit Service b1ea74
		}
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
disconnect:
Packit Service b1ea74
	freerdp_disconnect(instance);
Packit Service b1ea74
	return result;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* Optional global initializer.
Packit Service b1ea74
 * Here we just register a signal handler to print out stack traces
Packit Service b1ea74
 * if available. */
Packit Service b1ea74
static BOOL tf_client_global_init(void)
Packit Service b1ea74
{
Packit Service b1ea74
	if (freerdp_handle_signals() != 0)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
/* Optional global tear down */
Packit Service b1ea74
static void tf_client_global_uninit(void)
Packit Service b1ea74
{
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static int tf_logon_error_info(freerdp* instance, UINT32 data, UINT32 type)
Packit Service b1ea74
{
Packit Service b1ea74
	tfContext* tf;
Packit Service b1ea74
	const char* str_data = freerdp_get_logon_error_info_data(data);
Packit Service b1ea74
	const char* str_type = freerdp_get_logon_error_info_type(type);
Packit Service b1ea74
Packit Service b1ea74
	if (!instance || !instance->context)
Packit Service b1ea74
		return -1;
Packit Service b1ea74
Packit Service b1ea74
	tf = (tfContext*)instance->context;
Packit Service b1ea74
	WLog_INFO(TAG, "Logon Error Info %s [%s]", str_data, str_type);
Packit Service b1ea74
	WINPR_UNUSED(tf);
Packit Service b1ea74
Packit Service b1ea74
	return 1;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static BOOL tf_client_new(freerdp* instance, rdpContext* context)
Packit Service b1ea74
{
Packit Service b1ea74
	tfContext* tf = (tfContext*)context;
Packit Service b1ea74
Packit Service b1ea74
	if (!instance || !context)
Packit Service b1ea74
		return FALSE;
Packit Service b1ea74
Packit Service b1ea74
	instance->PreConnect = tf_pre_connect;
Packit Service b1ea74
	instance->PostConnect = tf_post_connect;
Packit Service b1ea74
	instance->PostDisconnect = tf_post_disconnect;
Packit Service b1ea74
	instance->Authenticate = client_cli_authenticate;
Packit Service b1ea74
	instance->GatewayAuthenticate = client_cli_gw_authenticate;
Packit Service b1ea74
	instance->VerifyCertificateEx = client_cli_verify_certificate_ex;
Packit Service b1ea74
	instance->VerifyChangedCertificateEx = client_cli_verify_changed_certificate_ex;
Packit Service b1ea74
	instance->LogonErrorInfo = tf_logon_error_info;
Packit Service b1ea74
	/* TODO: Client display set up */
Packit Service b1ea74
	WINPR_UNUSED(tf);
Packit Service b1ea74
	return TRUE;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static void tf_client_free(freerdp* instance, rdpContext* context)
Packit Service b1ea74
{
Packit Service b1ea74
	tfContext* tf = (tfContext*)instance->context;
Packit Service b1ea74
Packit Service b1ea74
	if (!context)
Packit Service b1ea74
		return;
Packit Service b1ea74
Packit Service b1ea74
	/* TODO: Client display tear down */
Packit Service b1ea74
	WINPR_UNUSED(tf);
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static int tf_client_start(rdpContext* context)
Packit Service b1ea74
{
Packit Service b1ea74
	/* TODO: Start client related stuff */
Packit Service b1ea74
	WINPR_UNUSED(context);
Packit Service b1ea74
	return 0;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static int tf_client_stop(rdpContext* context)
Packit Service b1ea74
{
Packit Service b1ea74
	/* TODO: Stop client related stuff */
Packit Service b1ea74
	WINPR_UNUSED(context);
Packit Service b1ea74
	return 0;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
static int RdpClientEntry(RDP_CLIENT_ENTRY_POINTS* pEntryPoints)
Packit Service b1ea74
{
Packit Service b1ea74
	ZeroMemory(pEntryPoints, sizeof(RDP_CLIENT_ENTRY_POINTS));
Packit Service b1ea74
	pEntryPoints->Version = RDP_CLIENT_INTERFACE_VERSION;
Packit Service b1ea74
	pEntryPoints->Size = sizeof(RDP_CLIENT_ENTRY_POINTS_V1);
Packit Service b1ea74
	pEntryPoints->GlobalInit = tf_client_global_init;
Packit Service b1ea74
	pEntryPoints->GlobalUninit = tf_client_global_uninit;
Packit Service b1ea74
	pEntryPoints->ContextSize = sizeof(tfContext);
Packit Service b1ea74
	pEntryPoints->ClientNew = tf_client_new;
Packit Service b1ea74
	pEntryPoints->ClientFree = tf_client_free;
Packit Service b1ea74
	pEntryPoints->ClientStart = tf_client_start;
Packit Service b1ea74
	pEntryPoints->ClientStop = tf_client_stop;
Packit Service b1ea74
	return 0;
Packit Service b1ea74
}
Packit Service b1ea74
Packit Service b1ea74
int main(int argc, char* argv[])
Packit Service b1ea74
{
Packit Service b1ea74
	int rc = -1;
Packit Service b1ea74
	DWORD status;
Packit Service b1ea74
	RDP_CLIENT_ENTRY_POINTS clientEntryPoints;
Packit Service b1ea74
	rdpContext* context;
Packit Service b1ea74
	RdpClientEntry(&clientEntryPoints);
Packit Service b1ea74
	context = freerdp_client_context_new(&clientEntryPoints);
Packit Service b1ea74
Packit Service b1ea74
	if (!context)
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	status = freerdp_client_settings_parse_command_line(context->settings, argc, argv, FALSE);
Packit Service b1ea74
	status =
Packit Service b1ea74
	    freerdp_client_settings_command_line_status_print(context->settings, status, argc, argv);
Packit Service b1ea74
Packit Service b1ea74
	if (status)
Packit Service b1ea74
	{
Packit Service b1ea74
		rc = 0;
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
	}
Packit Service b1ea74
Packit Service b1ea74
	if (freerdp_client_start(context) != 0)
Packit Service b1ea74
		goto fail;
Packit Service b1ea74
Packit Service b1ea74
	rc = tf_client_thread_proc(context->instance);
Packit Service b1ea74
Packit Service b1ea74
	if (freerdp_client_stop(context) != 0)
Packit Service b1ea74
		rc = -1;
Packit Service b1ea74
Packit Service b1ea74
fail:
Packit Service b1ea74
	freerdp_client_context_free(context);
Packit Service b1ea74
	return rc;
Packit Service b1ea74
}