/** * FreeRDP: A Remote Desktop Protocol Implementation * rdp2tcp Virtual Channel Extension * * Copyright 2017 Artur Zaprzala * * 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. */ #include #include #include #include #include #include #include #include #include #define TAG CLIENT_TAG(RDP2TCP_DVC_CHANNEL_NAME) static int const debug = 0; typedef struct { HANDLE hStdOutputRead; HANDLE hStdInputWrite; HANDLE hProcess; HANDLE copyThread; HANDLE writeComplete; DWORD openHandle; void* initHandle; CHANNEL_ENTRY_POINTS_FREERDP_EX channelEntryPoints; char buffer[16 * 1024]; char* commandline; } Plugin; static int init_external_addin(Plugin* plugin) { SECURITY_ATTRIBUTES saAttr; STARTUPINFOA siStartInfo; /* Using ANSI type to match CreateProcessA */ PROCESS_INFORMATION procInfo; saAttr.nLength = sizeof(SECURITY_ATTRIBUTES); saAttr.bInheritHandle = TRUE; saAttr.lpSecurityDescriptor = NULL; siStartInfo.cb = sizeof(STARTUPINFO); siStartInfo.hStdError = GetStdHandle(STD_ERROR_HANDLE); siStartInfo.dwFlags = STARTF_USESTDHANDLES; // Create pipes if (!CreatePipe(&plugin->hStdOutputRead, &siStartInfo.hStdOutput, &saAttr, 0)) { WLog_ERR(TAG, "stdout CreatePipe"); return -1; } if (!SetHandleInformation(plugin->hStdOutputRead, HANDLE_FLAG_INHERIT, 0)) { WLog_ERR(TAG, "stdout SetHandleInformation"); return -1; } if (!CreatePipe(&siStartInfo.hStdInput, &plugin->hStdInputWrite, &saAttr, 0)) { WLog_ERR(TAG, "stdin CreatePipe"); return -1; } if (!SetHandleInformation(plugin->hStdInputWrite, HANDLE_FLAG_INHERIT, 0)) { WLog_ERR(TAG, "stdin SetHandleInformation"); return -1; } // Execute plugin plugin->commandline = _strdup(plugin->channelEntryPoints.pExtendedData); if (!CreateProcessA(NULL, plugin->commandline, // command line NULL, // process security attributes NULL, // primary thread security attributes TRUE, // handles are inherited 0, // creation flags NULL, // use parent's environment NULL, // use parent's current directory &siStartInfo, // STARTUPINFO pointer &procInfo // receives PROCESS_INFORMATION )) { WLog_ERR(TAG, "fork for addin"); return -1; } plugin->hProcess = procInfo.hProcess; (void)CloseHandle(procInfo.hThread); (void)CloseHandle(siStartInfo.hStdOutput); (void)CloseHandle(siStartInfo.hStdInput); return 0; } static void dumpData(char* data, unsigned length) { unsigned const limit = 98; unsigned l = length > limit ? limit / 2 : length; for (unsigned i = 0; i < l; ++i) { printf("%02hhx", data[i]); } if (length > limit) { printf("..."); for (unsigned i = length - l; i < length; ++i) printf("%02hhx", data[i]); } puts(""); } static DWORD WINAPI copyThread(void* data) { DWORD status = WAIT_OBJECT_0; Plugin* plugin = (Plugin*)data; size_t const bufsize = 16ULL * 1024ULL; while (status == WAIT_OBJECT_0) { HANDLE handles[MAXIMUM_WAIT_OBJECTS] = { 0 }; DWORD dwRead = 0; char* buffer = malloc(bufsize); if (!buffer) { (void)fprintf(stderr, "rdp2tcp copyThread: malloc failed\n"); goto fail; } // if (!ReadFile(plugin->hStdOutputRead, plugin->buffer, sizeof plugin->buffer, &dwRead, // NULL)) if (!ReadFile(plugin->hStdOutputRead, buffer, bufsize, &dwRead, NULL)) { free(buffer); goto fail; } if (debug > 1) { printf(">%8u ", (unsigned)dwRead); dumpData(buffer, dwRead); } if (plugin->channelEntryPoints.pVirtualChannelWriteEx( plugin->initHandle, plugin->openHandle, buffer, dwRead, buffer) != CHANNEL_RC_OK) { free(buffer); (void)fprintf(stderr, "rdp2tcp copyThread failed %i\n", (int)dwRead); goto fail; } handles[0] = plugin->writeComplete; handles[1] = freerdp_abort_event(plugin->channelEntryPoints.context); status = WaitForMultipleObjects(2, handles, FALSE, INFINITE); if (status == WAIT_OBJECT_0) (void)ResetEvent(plugin->writeComplete); } fail: ExitThread(0); return 0; } static void closeChannel(Plugin* plugin) { if (debug) puts("rdp2tcp closing channel"); WINPR_ASSERT(plugin); WINPR_ASSERT(plugin->channelEntryPoints.pVirtualChannelCloseEx); plugin->channelEntryPoints.pVirtualChannelCloseEx(plugin->initHandle, plugin->openHandle); } static void dataReceived(Plugin* plugin, void* pData, UINT32 dataLength, UINT32 totalLength, UINT32 dataFlags) { DWORD dwWritten = 0; if (dataFlags & CHANNEL_FLAG_SUSPEND) { if (debug) puts("rdp2tcp Channel Suspend"); return; } if (dataFlags & CHANNEL_FLAG_RESUME) { if (debug) puts("rdp2tcp Channel Resume"); return; } if (debug > 1) { printf("<%c%3u/%3u ", dataFlags & CHANNEL_FLAG_FIRST ? ' ' : '+', totalLength, dataLength); dumpData(pData, dataLength); } if (dataFlags & CHANNEL_FLAG_FIRST) { if (!WriteFile(plugin->hStdInputWrite, &totalLength, sizeof(totalLength), &dwWritten, NULL)) closeChannel(plugin); } if (!WriteFile(plugin->hStdInputWrite, pData, dataLength, &dwWritten, NULL)) closeChannel(plugin); } static void VCAPITYPE VirtualChannelOpenEventEx(LPVOID lpUserParam, DWORD openHandle, UINT event, LPVOID pData, UINT32 dataLength, UINT32 totalLength, UINT32 dataFlags) { Plugin* plugin = (Plugin*)lpUserParam; switch (event) { case CHANNEL_EVENT_DATA_RECEIVED: dataReceived(plugin, pData, dataLength, totalLength, dataFlags); break; case CHANNEL_EVENT_WRITE_CANCELLED: free(pData); break; case CHANNEL_EVENT_WRITE_COMPLETE: (void)SetEvent(plugin->writeComplete); free(pData); break; default: break; } } static void channel_terminated(Plugin* plugin) { if (debug) puts("rdp2tcp terminated"); if (!plugin) return; if (plugin->copyThread) (void)TerminateThread(plugin->copyThread, 0); if (plugin->writeComplete) (void)CloseHandle(plugin->writeComplete); (void)CloseHandle(plugin->hStdInputWrite); (void)CloseHandle(plugin->hStdOutputRead); TerminateProcess(plugin->hProcess, 0); (void)CloseHandle(plugin->hProcess); free(plugin->commandline); free(plugin); } static void channel_initialized(Plugin* plugin) { plugin->writeComplete = CreateEvent(NULL, TRUE, FALSE, NULL); plugin->copyThread = CreateThread(NULL, 0, copyThread, plugin, 0, NULL); } static VOID VCAPITYPE VirtualChannelInitEventEx(LPVOID lpUserParam, LPVOID pInitHandle, UINT event, LPVOID pData, UINT dataLength) { Plugin* plugin = (Plugin*)lpUserParam; switch (event) { case CHANNEL_EVENT_INITIALIZED: channel_initialized(plugin); break; case CHANNEL_EVENT_CONNECTED: if (debug) puts("rdp2tcp connected"); WINPR_ASSERT(plugin); WINPR_ASSERT(plugin->channelEntryPoints.pVirtualChannelOpenEx); if (plugin->channelEntryPoints.pVirtualChannelOpenEx( pInitHandle, &plugin->openHandle, RDP2TCP_DVC_CHANNEL_NAME, VirtualChannelOpenEventEx) != CHANNEL_RC_OK) return; break; case CHANNEL_EVENT_DISCONNECTED: if (debug) puts("rdp2tcp disconnected"); break; case CHANNEL_EVENT_TERMINATED: channel_terminated(plugin); break; default: break; } } #if 1 #define VirtualChannelEntryEx rdp2tcp_VirtualChannelEntryEx #else #define VirtualChannelEntryEx FREERDP_API VirtualChannelEntryEx #endif FREERDP_ENTRY_POINT(BOOL VCAPITYPE VirtualChannelEntryEx(PCHANNEL_ENTRY_POINTS pEntryPoints, PVOID pInitHandle)) { CHANNEL_ENTRY_POINTS_FREERDP_EX* pEntryPointsEx = NULL; CHANNEL_DEF channelDef; Plugin* plugin = (Plugin*)calloc(1, sizeof(Plugin)); if (!plugin) return FALSE; pEntryPointsEx = (CHANNEL_ENTRY_POINTS_FREERDP_EX*)pEntryPoints; WINPR_ASSERT(pEntryPointsEx->cbSize >= sizeof(CHANNEL_ENTRY_POINTS_FREERDP_EX) && pEntryPointsEx->MagicNumber == FREERDP_CHANNEL_MAGIC_NUMBER); plugin->initHandle = pInitHandle; plugin->channelEntryPoints = *pEntryPointsEx; if (init_external_addin(plugin) < 0) { free(plugin); return FALSE; } strncpy(channelDef.name, RDP2TCP_DVC_CHANNEL_NAME, sizeof(channelDef.name)); channelDef.options = CHANNEL_OPTION_INITIALIZED | CHANNEL_OPTION_ENCRYPT_RDP | CHANNEL_OPTION_COMPRESS_RDP; if (pEntryPointsEx->pVirtualChannelInitEx(plugin, NULL, pInitHandle, &channelDef, 1, VIRTUAL_CHANNEL_VERSION_WIN2000, VirtualChannelInitEventEx) != CHANNEL_RC_OK) return FALSE; return TRUE; } // vim:ts=4