2012-02-23 19:22:05 +04:00
|
|
|
/*
|
2012-10-09 07:02:04 +04:00
|
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
2011-07-28 08:38:25 +04:00
|
|
|
* FreeRDP Core
|
|
|
|
*
|
|
|
|
* Copyright 2011 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2012-08-15 01:09:01 +04:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2011-07-28 08:38:25 +04:00
|
|
|
#include "rdp.h"
|
|
|
|
#include "input.h"
|
|
|
|
#include "update.h"
|
2011-08-28 23:51:49 +04:00
|
|
|
#include "surface.h"
|
2011-07-28 21:46:36 +04:00
|
|
|
#include "transport.h"
|
2011-07-28 19:09:51 +04:00
|
|
|
#include "connection.h"
|
2011-11-10 00:59:03 +04:00
|
|
|
#include "extension.h"
|
2011-07-28 08:38:25 +04:00
|
|
|
|
2012-11-22 04:22:41 +04:00
|
|
|
#include <winpr/crt.h>
|
|
|
|
|
2011-07-28 08:38:25 +04:00
|
|
|
#include <freerdp/freerdp.h>
|
2012-05-22 00:01:24 +04:00
|
|
|
#include <freerdp/errorcodes.h>
|
2012-08-31 03:57:21 +04:00
|
|
|
#include <freerdp/locale/keyboard.h>
|
2011-07-28 08:38:25 +04:00
|
|
|
|
2012-04-13 15:16:08 +04:00
|
|
|
/* connectErrorCode is 'extern' in errorcodes.h. See comment there.*/
|
|
|
|
|
2012-02-23 19:30:04 +04:00
|
|
|
/** Creates a new connection based on the settings found in the "instance" parameter
|
|
|
|
* It will use the callbacks registered on the structure to process the pre/post connect operations
|
|
|
|
* that the caller requires.
|
|
|
|
* @see struct rdp_freerdp in freerdp.h
|
2012-02-23 19:22:05 +04:00
|
|
|
*
|
|
|
|
* @param instance - pointer to a rdp_freerdp structure that contains base information to establish the connection.
|
|
|
|
* On return, this function will be initialized with the new connection's settings.
|
|
|
|
*
|
2012-10-09 10:31:28 +04:00
|
|
|
* @return TRUE if successful. FALSE otherwise.
|
2012-02-23 19:30:04 +04:00
|
|
|
*
|
2012-02-23 19:22:05 +04:00
|
|
|
*/
|
2012-10-09 10:38:39 +04:00
|
|
|
BOOL freerdp_connect(freerdp* instance)
|
2012-02-27 13:16:00 +04:00
|
|
|
{
|
2011-07-28 21:46:36 +04:00
|
|
|
rdpRdp* rdp;
|
2012-09-04 01:08:46 +04:00
|
|
|
rdpSettings* settings;
|
2012-10-09 10:38:39 +04:00
|
|
|
BOOL status = FALSE;
|
2012-05-22 00:01:24 +04:00
|
|
|
|
2012-04-13 15:16:08 +04:00
|
|
|
/* We always set the return code to 0 before we start the connect sequence*/
|
2012-05-22 00:01:24 +04:00
|
|
|
connectErrorCode = 0;
|
2011-07-28 21:46:36 +04:00
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
rdp = instance->context->rdp;
|
2012-09-04 01:08:46 +04:00
|
|
|
settings = instance->settings;
|
2011-07-28 21:46:36 +04:00
|
|
|
|
2012-08-26 19:32:36 +04:00
|
|
|
IFCALLRET(instance->PreConnect, status, instance);
|
|
|
|
|
2012-11-07 19:33:06 +04:00
|
|
|
if (settings->KeyboardLayout == KBD_JAPANESE_INPUT_SYSTEM_MS_IME2002)
|
2012-09-22 22:27:30 +04:00
|
|
|
{
|
2012-11-07 19:33:06 +04:00
|
|
|
settings->KeyboardType = 7;
|
|
|
|
settings->KeyboardSubType = 2;
|
|
|
|
settings->KeyboardFunctionKey = 12;
|
2012-09-22 22:27:30 +04:00
|
|
|
}
|
2012-08-24 20:25:59 +04:00
|
|
|
|
2012-08-03 12:40:05 +04:00
|
|
|
extension_load_and_init_plugins(rdp->extension);
|
2012-06-19 10:49:53 +04:00
|
|
|
extension_pre_connect(rdp->extension);
|
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
if (status != TRUE)
|
2012-02-27 13:13:39 +04:00
|
|
|
{
|
2012-09-04 01:08:46 +04:00
|
|
|
if(!connectErrorCode)
|
|
|
|
{
|
2012-04-13 15:16:08 +04:00
|
|
|
connectErrorCode = PREECONNECTERROR;
|
|
|
|
}
|
2012-06-06 13:46:21 +04:00
|
|
|
fprintf(stderr, "%s:%d: freerdp_pre_connect failed\n", __FILE__, __LINE__);
|
2012-10-09 10:31:28 +04:00
|
|
|
return FALSE;
|
2011-12-09 22:53:19 +04:00
|
|
|
}
|
2011-08-27 04:26:40 +04:00
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
status = rdp_client_connect(rdp);
|
2012-09-04 01:08:46 +04:00
|
|
|
/* --authonly tests the connection without a UI */
|
2012-11-08 03:23:25 +04:00
|
|
|
if (instance->settings->AuthenticationOnly)
|
2012-09-04 01:08:46 +04:00
|
|
|
{
|
2012-06-06 13:46:21 +04:00
|
|
|
fprintf(stderr, "%s:%d: Authentication only, exit status %d\n", __FILE__, __LINE__, !status);
|
2012-06-06 13:19:30 +04:00
|
|
|
return status;
|
2012-06-06 13:46:21 +04:00
|
|
|
}
|
2011-08-27 04:26:40 +04:00
|
|
|
|
2012-02-27 13:13:39 +04:00
|
|
|
if (status)
|
|
|
|
{
|
2012-11-08 00:13:14 +04:00
|
|
|
if (instance->settings->DumpRemoteFx)
|
2012-02-27 13:13:39 +04:00
|
|
|
{
|
2012-11-08 00:13:14 +04:00
|
|
|
instance->update->pcap_rfx = pcap_open(instance->settings->DumpRemoteFxFile, TRUE);
|
2011-10-20 16:25:25 +04:00
|
|
|
if (instance->update->pcap_rfx)
|
2012-10-09 10:31:28 +04:00
|
|
|
instance->update->dump_rfx = TRUE;
|
2011-08-27 04:26:40 +04:00
|
|
|
}
|
|
|
|
|
2011-11-10 00:59:03 +04:00
|
|
|
extension_post_connect(rdp->extension);
|
|
|
|
|
2011-12-09 22:53:19 +04:00
|
|
|
IFCALLRET(instance->PostConnect, status, instance);
|
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
if (status != TRUE)
|
2012-02-27 13:13:39 +04:00
|
|
|
{
|
2011-12-09 22:53:19 +04:00
|
|
|
printf("freerdp_post_connect failed\n");
|
2012-05-22 00:01:24 +04:00
|
|
|
|
|
|
|
if (!connectErrorCode)
|
|
|
|
{
|
2012-04-13 15:16:08 +04:00
|
|
|
connectErrorCode = POSTCONNECTERROR;
|
|
|
|
}
|
2012-05-22 00:01:24 +04:00
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
return FALSE;
|
2011-12-09 22:53:19 +04:00
|
|
|
}
|
2011-08-28 23:51:49 +04:00
|
|
|
|
2012-11-08 00:13:14 +04:00
|
|
|
if (instance->settings->PlayRemoteFx)
|
2012-02-27 13:13:39 +04:00
|
|
|
{
|
2011-08-28 23:51:49 +04:00
|
|
|
STREAM* s;
|
|
|
|
rdpUpdate* update;
|
|
|
|
pcap_record record;
|
|
|
|
|
|
|
|
s = stream_new(1024);
|
2012-11-08 00:13:14 +04:00
|
|
|
instance->update->pcap_rfx = pcap_open(instance->settings->PlayRemoteFxFile, FALSE);
|
2012-05-22 00:01:24 +04:00
|
|
|
|
2011-10-20 16:25:25 +04:00
|
|
|
if (instance->update->pcap_rfx)
|
2012-10-09 10:31:28 +04:00
|
|
|
instance->update->play_rfx = TRUE;
|
2012-05-22 00:01:24 +04:00
|
|
|
|
2011-08-28 23:51:49 +04:00
|
|
|
update = instance->update;
|
|
|
|
|
2012-02-27 13:13:39 +04:00
|
|
|
while (instance->update->play_rfx && pcap_has_next_record(update->pcap_rfx))
|
|
|
|
{
|
2011-08-28 23:51:49 +04:00
|
|
|
pcap_get_next_record_header(update->pcap_rfx, &record);
|
|
|
|
|
2012-10-09 11:01:37 +04:00
|
|
|
s->data = (BYTE*) realloc(s->data, record.length);
|
2011-08-28 23:51:49 +04:00
|
|
|
record.data = s->data;
|
|
|
|
s->size = record.length;
|
|
|
|
|
|
|
|
pcap_get_next_record_content(update->pcap_rfx, &record);
|
|
|
|
stream_set_pos(s, 0);
|
|
|
|
|
2011-11-22 04:41:49 +04:00
|
|
|
update->BeginPaint(update->context);
|
2011-08-28 23:51:49 +04:00
|
|
|
update_recv_surfcmds(update, s->size, s);
|
2011-11-22 04:41:49 +04:00
|
|
|
update->EndPaint(update->context);
|
2011-08-28 23:51:49 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 07:21:26 +04:00
|
|
|
free(s->data);
|
2012-10-09 10:31:28 +04:00
|
|
|
return TRUE;
|
2011-08-28 23:51:49 +04:00
|
|
|
}
|
2011-08-27 04:26:40 +04:00
|
|
|
}
|
2012-05-22 00:01:24 +04:00
|
|
|
|
|
|
|
if (!connectErrorCode)
|
|
|
|
{
|
2012-04-13 15:16:08 +04:00
|
|
|
connectErrorCode = UNDEFINEDCONNECTERROR;
|
|
|
|
}
|
2012-05-22 00:01:24 +04:00
|
|
|
|
2011-07-28 21:46:36 +04:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2012-10-09 10:38:39 +04:00
|
|
|
BOOL freerdp_get_fds(freerdp* instance, void** rfds, int* rcount, void** wfds, int* wcount)
|
2012-02-27 13:13:39 +04:00
|
|
|
{
|
2011-07-28 21:46:36 +04:00
|
|
|
rdpRdp* rdp;
|
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
rdp = instance->context->rdp;
|
2011-08-20 16:30:18 +04:00
|
|
|
transport_get_fds(rdp->transport, rfds, rcount);
|
2011-07-28 21:46:36 +04:00
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
return TRUE;
|
2011-07-28 21:46:36 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 10:38:39 +04:00
|
|
|
BOOL freerdp_check_fds(freerdp* instance)
|
2012-02-27 13:13:39 +04:00
|
|
|
{
|
2011-08-01 08:43:53 +04:00
|
|
|
int status;
|
2011-10-16 08:50:10 +04:00
|
|
|
rdpRdp* rdp;
|
2011-07-28 21:46:36 +04:00
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
rdp = instance->context->rdp;
|
2011-07-28 21:46:36 +04:00
|
|
|
|
2011-08-01 08:43:53 +04:00
|
|
|
status = rdp_check_fds(rdp);
|
2011-10-16 08:50:10 +04:00
|
|
|
|
2011-08-01 08:43:53 +04:00
|
|
|
if (status < 0)
|
2012-10-09 10:31:28 +04:00
|
|
|
return FALSE;
|
2011-07-29 01:44:09 +04:00
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
return TRUE;
|
2011-07-28 19:09:51 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:01:37 +04:00
|
|
|
static int freerdp_send_channel_data(freerdp* instance, int channel_id, BYTE* data, int size)
|
2012-02-27 13:13:39 +04:00
|
|
|
{
|
2011-10-16 08:50:10 +04:00
|
|
|
return rdp_send_channel_data(instance->context->rdp, channel_id, data, size);
|
2011-08-03 07:00:56 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 10:38:39 +04:00
|
|
|
BOOL freerdp_disconnect(freerdp* instance)
|
2012-02-27 13:13:39 +04:00
|
|
|
{
|
2011-08-24 19:07:55 +04:00
|
|
|
rdpRdp* rdp;
|
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
rdp = instance->context->rdp;
|
2011-08-24 19:07:55 +04:00
|
|
|
transport_disconnect(rdp->transport);
|
2011-10-17 05:02:28 +04:00
|
|
|
|
2012-10-09 10:31:28 +04:00
|
|
|
return TRUE;
|
2011-08-24 19:07:55 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 10:38:39 +04:00
|
|
|
BOOL freerdp_shall_disconnect(freerdp* instance)
|
2012-02-27 13:13:39 +04:00
|
|
|
{
|
2012-02-07 01:37:18 +04:00
|
|
|
return instance->context->rdp->disconnect;
|
|
|
|
}
|
|
|
|
|
2012-02-27 13:13:39 +04:00
|
|
|
void freerdp_get_version(int* major, int* minor, int* revision)
|
|
|
|
{
|
2012-01-14 22:33:04 +04:00
|
|
|
if (major != NULL)
|
|
|
|
*major = FREERDP_VERSION_MAJOR;
|
|
|
|
|
|
|
|
if (minor != NULL)
|
|
|
|
*minor = FREERDP_VERSION_MINOR;
|
|
|
|
|
|
|
|
if (revision != NULL)
|
|
|
|
*revision = FREERDP_VERSION_REVISION;
|
|
|
|
}
|
|
|
|
|
2012-02-23 19:22:05 +04:00
|
|
|
/** Allocator function for a rdp context.
|
|
|
|
* The function will allocate a rdpRdp structure using rdp_new(), then copy
|
2012-02-23 19:30:04 +04:00
|
|
|
* its contents to the appropriate fields in the rdp_freerdp structure given in parameters.
|
2012-03-02 01:06:03 +04:00
|
|
|
* It will also initialize the 'context' field in the rdp_freerdp structure as needed.
|
2012-02-23 19:30:04 +04:00
|
|
|
* If the caller has set the ContextNew callback in the 'instance' parameter, it will be called at the end of the function.
|
2012-02-23 19:22:05 +04:00
|
|
|
*
|
|
|
|
* @param instance - Pointer to the rdp_freerdp structure that will be initialized with the new context.
|
|
|
|
*/
|
2012-02-27 13:13:39 +04:00
|
|
|
void freerdp_context_new(freerdp* instance)
|
|
|
|
{
|
2011-10-16 08:50:10 +04:00
|
|
|
rdpRdp* rdp;
|
|
|
|
|
|
|
|
rdp = rdp_new(instance);
|
2012-02-23 19:30:04 +04:00
|
|
|
// FIXME - we're not checking where rdp_new returns NULL, and have no way to report an error to the caller
|
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
instance->input = rdp->input;
|
|
|
|
instance->update = rdp->update;
|
|
|
|
instance->settings = rdp->settings;
|
|
|
|
|
2012-11-22 04:22:41 +04:00
|
|
|
instance->context = (rdpContext*) malloc(instance->context_size);
|
|
|
|
ZeroMemory(instance->context, instance->context_size);
|
|
|
|
|
2011-10-21 01:28:59 +04:00
|
|
|
instance->context->graphics = graphics_new(instance->context);
|
2011-10-16 08:50:10 +04:00
|
|
|
instance->context->instance = instance;
|
|
|
|
instance->context->rdp = rdp;
|
|
|
|
|
|
|
|
instance->update->context = instance->context;
|
2011-11-22 02:48:03 +04:00
|
|
|
instance->update->pointer->context = instance->context;
|
2011-11-22 03:11:43 +04:00
|
|
|
instance->update->primary->context = instance->context;
|
2011-11-22 04:41:49 +04:00
|
|
|
instance->update->secondary->context = instance->context;
|
2011-11-22 04:53:38 +04:00
|
|
|
instance->update->altsec->context = instance->context;
|
2011-11-22 02:48:03 +04:00
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
instance->input->context = instance->context;
|
|
|
|
|
2012-05-26 17:34:09 +04:00
|
|
|
update_register_client_callbacks(rdp->update);
|
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
IFCALL(instance->ContextNew, instance, instance->context);
|
|
|
|
}
|
|
|
|
|
2012-02-23 19:22:05 +04:00
|
|
|
/** Deallocator function for a rdp context.
|
|
|
|
* The function will deallocate the resources from the 'instance' parameter that were allocated from a call
|
|
|
|
* to freerdp_context_new().
|
|
|
|
* If the ContextFree callback is set in the 'instance' parameter, it will be called before deallocation occurs.
|
|
|
|
*
|
|
|
|
* @param instance - Pointer to the rdp_freerdp structure that was initialized by a call to freerdp_context_new().
|
|
|
|
* On return, the fields associated to the context are invalid.
|
|
|
|
*/
|
2012-02-27 13:13:39 +04:00
|
|
|
void freerdp_context_free(freerdp* instance)
|
|
|
|
{
|
2012-02-11 00:35:22 +04:00
|
|
|
if (instance->context == NULL)
|
|
|
|
return;
|
2012-02-11 00:50:29 +04:00
|
|
|
|
2011-10-16 08:50:10 +04:00
|
|
|
IFCALL(instance->ContextFree, instance, instance->context);
|
2012-02-11 00:50:29 +04:00
|
|
|
|
2012-02-02 03:42:20 +04:00
|
|
|
rdp_free(instance->context->rdp);
|
|
|
|
graphics_free(instance->context->graphics);
|
2012-02-11 00:50:29 +04:00
|
|
|
|
2012-10-09 07:21:26 +04:00
|
|
|
free(instance->context);
|
2012-02-11 00:35:22 +04:00
|
|
|
instance->context = NULL;
|
2011-10-16 08:50:10 +04:00
|
|
|
}
|
|
|
|
|
2012-10-09 11:26:39 +04:00
|
|
|
UINT32 freerdp_error_info(freerdp* instance)
|
2012-02-27 13:13:39 +04:00
|
|
|
{
|
2011-10-24 22:37:21 +04:00
|
|
|
return instance->context->rdp->errorInfo;
|
|
|
|
}
|
|
|
|
|
2012-02-23 19:22:05 +04:00
|
|
|
/** Allocator function for the rdp_freerdp structure.
|
|
|
|
* @return an allocated structure filled with 0s. Need to be deallocated using freerdp_free()
|
|
|
|
*/
|
2012-02-27 13:13:39 +04:00
|
|
|
freerdp* freerdp_new()
|
|
|
|
{
|
2011-07-28 08:38:25 +04:00
|
|
|
freerdp* instance;
|
|
|
|
|
2012-11-22 04:22:41 +04:00
|
|
|
instance = (freerdp*) malloc(sizeof(freerdp));
|
|
|
|
ZeroMemory(instance, sizeof(freerdp));
|
2011-07-28 08:38:25 +04:00
|
|
|
|
2012-02-27 13:16:00 +04:00
|
|
|
if (instance != NULL)
|
|
|
|
{
|
2011-10-21 07:15:18 +04:00
|
|
|
instance->context_size = sizeof(rdpContext);
|
2011-08-03 07:00:56 +04:00
|
|
|
instance->SendChannelData = freerdp_send_channel_data;
|
2011-07-28 08:38:25 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
2012-02-23 19:22:05 +04:00
|
|
|
/** Deallocator function for the rdp_freerdp structure.
|
|
|
|
* @param instance - pointer to the rdp_freerdp structure to deallocate.
|
|
|
|
* On return, this pointer is not valid anymore.
|
|
|
|
*/
|
2012-02-27 13:13:39 +04:00
|
|
|
void freerdp_free(freerdp* instance)
|
|
|
|
{
|
|
|
|
if (instance)
|
|
|
|
{
|
2012-10-09 07:21:26 +04:00
|
|
|
free(instance);
|
2011-09-07 10:19:11 +04:00
|
|
|
}
|
2011-07-28 08:38:25 +04:00
|
|
|
}
|