Blame server/Windows/wf_interface.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * FreeRDP Windows Server
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit 1fb8d4
 * Copyright 2012 Corey Clayton <can.of.tuna@gmail.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 <winpr/tchar.h>
Packit 1fb8d4
#include <winpr/windows.h>
Packit 1fb8d4
#include <winpr/winsock.h>
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/freerdp.h>
Packit 1fb8d4
#include <freerdp/listener.h>
Packit 1fb8d4
#include <freerdp/constants.h>
Packit 1fb8d4
#include <freerdp/channels/wtsvc.h>
Packit 1fb8d4
#include <freerdp/channels/channels.h>
Packit 1fb8d4
#include <freerdp/build-config.h>
Packit 1fb8d4
Packit 1fb8d4
#include "wf_peer.h"
Packit 1fb8d4
#include "wf_settings.h"
Packit 1fb8d4
#include "wf_info.h"
Packit 1fb8d4
Packit 1fb8d4
#include "wf_interface.h"
Packit 1fb8d4
Packit 1fb8d4
#define TAG SERVER_TAG("windows")
Packit 1fb8d4
Packit 1fb8d4
#define SERVER_KEY "Software\\"FREERDP_VENDOR_STRING"\\" \
Packit 1fb8d4
		FREERDP_PRODUCT_STRING"\\Server"
Packit 1fb8d4
Packit 1fb8d4
cbCallback cbEvent;
Packit 1fb8d4
Packit 1fb8d4
int get_screen_info(int id, _TCHAR* name, int* width, int* height, int* bpp)
Packit 1fb8d4
{
Packit 1fb8d4
	DISPLAY_DEVICE dd;
Packit 1fb8d4
Packit 1fb8d4
	memset(&dd, 0, sizeof(DISPLAY_DEVICE));
Packit 1fb8d4
	dd.cb = sizeof(DISPLAY_DEVICE);
Packit 1fb8d4
Packit 1fb8d4
	if (EnumDisplayDevices(NULL, id, &dd, 0) != 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		HDC dc;
Packit 1fb8d4
Packit 1fb8d4
		if (name != NULL)
Packit 1fb8d4
			_stprintf(name, _T("%s (%s)"), dd.DeviceName, dd.DeviceString);
Packit 1fb8d4
Packit 1fb8d4
		dc = CreateDC(dd.DeviceName, NULL, NULL, NULL);
Packit 1fb8d4
		*width = GetDeviceCaps(dc, HORZRES);
Packit 1fb8d4
		*height = GetDeviceCaps(dc, VERTRES);
Packit 1fb8d4
		*bpp = GetDeviceCaps(dc, BITSPIXEL);
Packit 1fb8d4
		//ReleaseDC(NULL, dc);
Packit 1fb8d4
		DeleteDC(dc);
Packit 1fb8d4
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		return 0;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void set_screen_id(int id)
Packit 1fb8d4
{
Packit 1fb8d4
	wfInfo* wfi;
Packit 1fb8d4
Packit 1fb8d4
	wfi = wf_info_get_instance();
Packit 1fb8d4
	if (!wfi)
Packit 1fb8d4
		return;
Packit 1fb8d4
	wfi->screenID = id;
Packit 1fb8d4
Packit 1fb8d4
	return;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static DWORD WINAPI wf_server_main_loop(LPVOID lpParam)
Packit 1fb8d4
{
Packit 1fb8d4
	int i, fds;
Packit 1fb8d4
	int rcount;
Packit 1fb8d4
	int max_fds;
Packit 1fb8d4
	void* rfds[32];
Packit 1fb8d4
	fd_set rfds_set;
Packit 1fb8d4
	freerdp_listener* instance;
Packit 1fb8d4
	wfInfo* wfi;
Packit 1fb8d4
Packit 1fb8d4
	wfi = wf_info_get_instance();
Packit 1fb8d4
	if (!wfi)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "Failed to get instance");
Packit 1fb8d4
		return -1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	wfi->force_all_disconnect = FALSE;
Packit 1fb8d4
Packit 1fb8d4
	ZeroMemory(rfds, sizeof(rfds));
Packit 1fb8d4
	instance = (freerdp_listener*) lpParam;
Packit 1fb8d4
Packit 1fb8d4
	while(wfi->force_all_disconnect == FALSE)
Packit 1fb8d4
	{
Packit 1fb8d4
		rcount = 0;
Packit 1fb8d4
Packit 1fb8d4
		if (instance->GetFileDescriptor(instance, rfds, &rcount) != TRUE)
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_ERR(TAG, "Failed to get FreeRDP file descriptor");
Packit 1fb8d4
			break;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		max_fds = 0;
Packit 1fb8d4
		FD_ZERO(&rfds_set);
Packit 1fb8d4
Packit 1fb8d4
		for (i = 0; i < rcount; i++)
Packit 1fb8d4
		{
Packit 1fb8d4
			fds = (int)(long)(rfds[i]);
Packit 1fb8d4
Packit 1fb8d4
			if (fds > max_fds)
Packit 1fb8d4
				max_fds = fds;
Packit 1fb8d4
Packit 1fb8d4
			FD_SET(fds, &rfds_set);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (max_fds == 0)
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
Packit 1fb8d4
		select(max_fds + 1, &rfds_set, NULL, NULL, NULL);
Packit 1fb8d4
Packit 1fb8d4
		if (instance->CheckFileDescriptor(instance) != TRUE)
Packit 1fb8d4
		{
Packit 1fb8d4
			WLog_ERR(TAG, "Failed to check FreeRDP file descriptor");
Packit 1fb8d4
			break;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	WLog_INFO(TAG, "wf_server_main_loop terminating");
Packit 1fb8d4
	instance->Close(instance);
Packit 1fb8d4
Packit 1fb8d4
	return 0;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL wfreerdp_server_start(wfServer* server)
Packit 1fb8d4
{
Packit 1fb8d4
	freerdp_listener* instance;
Packit 1fb8d4
Packit 1fb8d4
	server->instance = freerdp_listener_new();
Packit 1fb8d4
	server->instance->PeerAccepted = wf_peer_accepted;
Packit 1fb8d4
	instance = server->instance;
Packit 1fb8d4
Packit 1fb8d4
	wf_settings_read_dword(HKEY_LOCAL_MACHINE, SERVER_KEY,
Packit 1fb8d4
				_T("DefaultPort"), &server->port);
Packit 1fb8d4
Packit 1fb8d4
	if (!instance->Open(instance, NULL, (UINT16) server->port))
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (!(server->thread = CreateThread(NULL, 0, wf_server_main_loop, (void*) instance, 0, NULL)))
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL wfreerdp_server_stop(wfServer* server)
Packit 1fb8d4
{
Packit 1fb8d4
	wfInfo* wfi;
Packit 1fb8d4
Packit 1fb8d4
	wfi = wf_info_get_instance();
Packit 1fb8d4
	if (!wfi)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	WLog_INFO(TAG, "Stopping server");
Packit 1fb8d4
	wfi->force_all_disconnect = TRUE;
Packit 1fb8d4
	server->instance->Close(server->instance);
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
wfServer* wfreerdp_server_new()
Packit 1fb8d4
{
Packit 1fb8d4
	WSADATA wsaData;
Packit 1fb8d4
	wfServer* server;
Packit 1fb8d4
Packit 1fb8d4
	if (WSAStartup(MAKEWORD(2, 2), &wsaData) != 0)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	server = (wfServer*) calloc(1, sizeof(wfServer));
Packit 1fb8d4
Packit 1fb8d4
	if (server)
Packit 1fb8d4
	{
Packit 1fb8d4
		server->port = 3389;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	WTSRegisterWtsApiFunctionTable(FreeRDP_InitWtsApi());
Packit 1fb8d4
Packit 1fb8d4
	cbEvent = NULL;
Packit 1fb8d4
Packit 1fb8d4
	return server;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void wfreerdp_server_free(wfServer* server)
Packit 1fb8d4
{
Packit 1fb8d4
	free(server);
Packit 1fb8d4
Packit 1fb8d4
	WSACleanup();
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL wfreerdp_server_is_running(wfServer* server)
Packit 1fb8d4
{
Packit 1fb8d4
	DWORD tStatus;
Packit 1fb8d4
	BOOL bRet;
Packit 1fb8d4
Packit 1fb8d4
	bRet = GetExitCodeThread(server->thread, &tStatus);
Packit 1fb8d4
	if (bRet == 0)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_ERR(TAG, "Error in call to GetExitCodeThread");
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (tStatus == STILL_ACTIVE)
Packit 1fb8d4
		return TRUE;
Packit 1fb8d4
	return FALSE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
UINT32 wfreerdp_server_num_peers()
Packit 1fb8d4
{
Packit 1fb8d4
	wfInfo* wfi;
Packit 1fb8d4
Packit 1fb8d4
	wfi = wf_info_get_instance();
Packit 1fb8d4
	if (!wfi)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
	return wfi->peerCount;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
UINT32 wfreerdp_server_get_peer_hostname(int pId, wchar_t * dstStr)
Packit 1fb8d4
{
Packit 1fb8d4
	wfInfo* wfi;
Packit 1fb8d4
	freerdp_peer* peer;
Packit 1fb8d4
Packit 1fb8d4
	wfi = wf_info_get_instance();
Packit 1fb8d4
	if (!wfi)
Packit 1fb8d4
		return 0;
Packit 1fb8d4
	peer = wfi->peers[pId];
Packit 1fb8d4
Packit 1fb8d4
	if (peer)
Packit 1fb8d4
	{
Packit 1fb8d4
		UINT32 sLen;
Packit 1fb8d4
Packit 1fb8d4
		sLen = strnlen_s(peer->hostname, 50);
Packit 1fb8d4
		swprintf(dstStr, 50, L"%hs", peer->hostname);
Packit 1fb8d4
		return sLen;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_WARN(TAG, "nonexistent peer id=%d", pId);
Packit 1fb8d4
		return 0;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL wfreerdp_server_peer_is_local(int pId)
Packit 1fb8d4
{
Packit 1fb8d4
	wfInfo* wfi;
Packit 1fb8d4
	freerdp_peer* peer;
Packit 1fb8d4
Packit 1fb8d4
	wfi = wf_info_get_instance();
Packit 1fb8d4
	if (!wfi)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	peer = wfi->peers[pId];
Packit 1fb8d4
Packit 1fb8d4
	if (peer)
Packit 1fb8d4
	{
Packit 1fb8d4
		return peer->local;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL wfreerdp_server_peer_is_connected(int pId)
Packit 1fb8d4
{
Packit 1fb8d4
	wfInfo* wfi;
Packit 1fb8d4
	freerdp_peer* peer;
Packit 1fb8d4
Packit 1fb8d4
	wfi = wf_info_get_instance();
Packit 1fb8d4
	if (!wfi)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	peer = wfi->peers[pId];
Packit 1fb8d4
Packit 1fb8d4
Packit 1fb8d4
	if (peer)
Packit 1fb8d4
	{
Packit 1fb8d4
		return peer->connected;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL wfreerdp_server_peer_is_activated(int pId)
Packit 1fb8d4
{
Packit 1fb8d4
	wfInfo* wfi;
Packit 1fb8d4
	freerdp_peer* peer;
Packit 1fb8d4
Packit 1fb8d4
	wfi = wf_info_get_instance();
Packit 1fb8d4
	if (!wfi)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	peer = wfi->peers[pId];
Packit 1fb8d4
Packit 1fb8d4
	if (peer)
Packit 1fb8d4
	{
Packit 1fb8d4
		return peer->activated;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL wfreerdp_server_peer_is_authenticated(int pId)
Packit 1fb8d4
{
Packit 1fb8d4
	wfInfo* wfi;
Packit 1fb8d4
	freerdp_peer* peer;
Packit 1fb8d4
Packit 1fb8d4
	wfi = wf_info_get_instance();
Packit 1fb8d4
	if (!wfi)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	peer = wfi->peers[pId];
Packit 1fb8d4
Packit 1fb8d4
	if (peer)
Packit 1fb8d4
	{
Packit 1fb8d4
		return peer->authenticated;
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void wfreerdp_server_register_callback_event(cbCallback cb)
Packit 1fb8d4
{
Packit 1fb8d4
	cbEvent = cb;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void wfreerdp_server_peer_callback_event(int pId, UINT32 eType)
Packit 1fb8d4
{
Packit 1fb8d4
	if (cbEvent)
Packit 1fb8d4
		cbEvent(pId, eType);
Packit 1fb8d4
}
Packit 1fb8d4