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