471 lines
12 KiB
C
471 lines
12 KiB
C
/**
|
|
* FreeRDP: A Remote Desktop Protocol Implementation
|
|
* Channel Addins
|
|
*
|
|
* Copyright 2012 Marc-Andre Moreau <marcandre.moreau@gmail.com>
|
|
* Copyright 2015 Thincast Technologies GmbH
|
|
* Copyright 2015 DI (FH) Martin Haimberger <martin.haimberger@thincast.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.
|
|
*/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#endif
|
|
|
|
#include <winpr/crt.h>
|
|
#include <winpr/path.h>
|
|
#include <winpr/file.h>
|
|
#include <winpr/synch.h>
|
|
#include <winpr/library.h>
|
|
#include <winpr/collections.h>
|
|
|
|
#include <freerdp/addin.h>
|
|
#include <freerdp/build-config.h>
|
|
#include <freerdp/client/channels.h>
|
|
|
|
#include "tables.h"
|
|
|
|
#include "addin.h"
|
|
|
|
#include <freerdp/channels/log.h>
|
|
#define TAG CHANNELS_TAG("addin")
|
|
|
|
extern const STATIC_ENTRY_TABLE CLIENT_STATIC_ENTRY_TABLES[];
|
|
|
|
static void* freerdp_channels_find_static_entry_in_table(const STATIC_ENTRY_TABLE* table,
|
|
const char* identifier)
|
|
{
|
|
size_t index = 0;
|
|
STATIC_ENTRY* pEntry;
|
|
pEntry = (STATIC_ENTRY*)&table->table[index++];
|
|
|
|
while (pEntry->entry != NULL)
|
|
{
|
|
if (strcmp(pEntry->name, identifier) == 0)
|
|
{
|
|
return (void*)pEntry->entry;
|
|
}
|
|
|
|
pEntry = (STATIC_ENTRY*)&table->table[index++];
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void* freerdp_channels_client_find_static_entry(const char* name, const char* identifier)
|
|
{
|
|
size_t index = 0;
|
|
STATIC_ENTRY_TABLE* pEntry;
|
|
pEntry = (STATIC_ENTRY_TABLE*)&CLIENT_STATIC_ENTRY_TABLES[index++];
|
|
|
|
while (pEntry->table != NULL)
|
|
{
|
|
if (strcmp(pEntry->name, name) == 0)
|
|
{
|
|
return freerdp_channels_find_static_entry_in_table(pEntry, identifier);
|
|
}
|
|
|
|
pEntry = (STATIC_ENTRY_TABLE*)&CLIENT_STATIC_ENTRY_TABLES[index++];
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
extern const STATIC_ADDIN_TABLE CLIENT_STATIC_ADDIN_TABLE[];
|
|
|
|
static FREERDP_ADDIN** freerdp_channels_list_client_static_addins(LPCSTR pszName,
|
|
LPCSTR pszSubsystem,
|
|
LPCSTR pszType, DWORD dwFlags)
|
|
{
|
|
size_t i, j;
|
|
DWORD nAddins;
|
|
FREERDP_ADDIN** ppAddins = NULL;
|
|
STATIC_SUBSYSTEM_ENTRY* subsystems;
|
|
nAddins = 0;
|
|
ppAddins = (FREERDP_ADDIN**)calloc(128, sizeof(FREERDP_ADDIN*));
|
|
|
|
if (!ppAddins)
|
|
{
|
|
WLog_ERR(TAG, "calloc failed!");
|
|
return NULL;
|
|
}
|
|
|
|
ppAddins[nAddins] = NULL;
|
|
|
|
for (i = 0; CLIENT_STATIC_ADDIN_TABLE[i].name != NULL; i++)
|
|
{
|
|
FREERDP_ADDIN* pAddin = (FREERDP_ADDIN*)calloc(1, sizeof(FREERDP_ADDIN));
|
|
|
|
if (!pAddin)
|
|
{
|
|
WLog_ERR(TAG, "calloc failed!");
|
|
goto error_out;
|
|
}
|
|
|
|
sprintf_s(pAddin->cName, ARRAYSIZE(pAddin->cName), "%s", CLIENT_STATIC_ADDIN_TABLE[i].name);
|
|
pAddin->dwFlags = FREERDP_ADDIN_CLIENT;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_STATIC;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_NAME;
|
|
ppAddins[nAddins++] = pAddin;
|
|
subsystems = (STATIC_SUBSYSTEM_ENTRY*)CLIENT_STATIC_ADDIN_TABLE[i].table;
|
|
|
|
for (j = 0; subsystems[j].name != NULL; j++)
|
|
{
|
|
pAddin = (FREERDP_ADDIN*)calloc(1, sizeof(FREERDP_ADDIN));
|
|
|
|
if (!pAddin)
|
|
{
|
|
WLog_ERR(TAG, "calloc failed!");
|
|
goto error_out;
|
|
}
|
|
|
|
sprintf_s(pAddin->cName, ARRAYSIZE(pAddin->cName), "%s",
|
|
CLIENT_STATIC_ADDIN_TABLE[i].name);
|
|
sprintf_s(pAddin->cSubsystem, ARRAYSIZE(pAddin->cSubsystem), "%s", subsystems[j].name);
|
|
pAddin->dwFlags = FREERDP_ADDIN_CLIENT;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_STATIC;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_NAME;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_SUBSYSTEM;
|
|
ppAddins[nAddins++] = pAddin;
|
|
}
|
|
}
|
|
|
|
return ppAddins;
|
|
error_out:
|
|
freerdp_channels_addin_list_free(ppAddins);
|
|
return NULL;
|
|
}
|
|
|
|
static FREERDP_ADDIN** freerdp_channels_list_dynamic_addins(LPCSTR pszName, LPCSTR pszSubsystem,
|
|
LPCSTR pszType, DWORD dwFlags)
|
|
{
|
|
int index;
|
|
int nDashes;
|
|
HANDLE hFind;
|
|
DWORD nAddins;
|
|
LPSTR pszPattern;
|
|
size_t cchPattern;
|
|
LPCSTR pszAddinPath = FREERDP_ADDIN_PATH;
|
|
LPCSTR pszInstallPrefix = FREERDP_INSTALL_PREFIX;
|
|
LPCSTR pszExtension;
|
|
LPSTR pszSearchPath;
|
|
size_t cchSearchPath;
|
|
size_t cchAddinPath;
|
|
size_t cchInstallPrefix;
|
|
FREERDP_ADDIN** ppAddins;
|
|
WIN32_FIND_DATAA FindData;
|
|
cchAddinPath = strnlen(pszAddinPath, sizeof(FREERDP_ADDIN_PATH));
|
|
cchInstallPrefix = strnlen(pszInstallPrefix, sizeof(FREERDP_INSTALL_PREFIX));
|
|
pszExtension = PathGetSharedLibraryExtensionA(0);
|
|
cchPattern = 128 + strnlen(pszExtension, MAX_PATH) + 2;
|
|
pszPattern = (LPSTR)malloc(cchPattern + 1);
|
|
|
|
if (!pszPattern)
|
|
{
|
|
WLog_ERR(TAG, "malloc failed!");
|
|
return NULL;
|
|
}
|
|
|
|
if (pszName && pszSubsystem && pszType)
|
|
{
|
|
sprintf_s(pszPattern, cchPattern, FREERDP_SHARED_LIBRARY_PREFIX "%s-client-%s-%s.%s",
|
|
pszName, pszSubsystem, pszType, pszExtension);
|
|
}
|
|
else if (pszName && pszType)
|
|
{
|
|
sprintf_s(pszPattern, cchPattern, FREERDP_SHARED_LIBRARY_PREFIX "%s-client-?-%s.%s",
|
|
pszName, pszType, pszExtension);
|
|
}
|
|
else if (pszName)
|
|
{
|
|
sprintf_s(pszPattern, cchPattern, FREERDP_SHARED_LIBRARY_PREFIX "%s-client*.%s", pszName,
|
|
pszExtension);
|
|
}
|
|
else
|
|
{
|
|
sprintf_s(pszPattern, cchPattern, FREERDP_SHARED_LIBRARY_PREFIX "?-client*.%s",
|
|
pszExtension);
|
|
}
|
|
|
|
cchPattern = strnlen(pszPattern, cchPattern);
|
|
cchSearchPath = cchInstallPrefix + cchAddinPath + cchPattern + 3;
|
|
pszSearchPath = (LPSTR)malloc(cchSearchPath + 1);
|
|
|
|
if (!pszSearchPath)
|
|
{
|
|
WLog_ERR(TAG, "malloc failed!");
|
|
free(pszPattern);
|
|
return NULL;
|
|
}
|
|
|
|
CopyMemory(pszSearchPath, pszInstallPrefix, cchInstallPrefix);
|
|
pszSearchPath[cchInstallPrefix] = '\0';
|
|
NativePathCchAppendA(pszSearchPath, cchSearchPath + 1, pszAddinPath);
|
|
NativePathCchAppendA(pszSearchPath, cchSearchPath + 1, pszPattern);
|
|
free(pszPattern);
|
|
hFind = FindFirstFileA(pszSearchPath, &FindData);
|
|
free(pszSearchPath);
|
|
nAddins = 0;
|
|
ppAddins = (FREERDP_ADDIN**)calloc(128, sizeof(FREERDP_ADDIN*));
|
|
|
|
if (!ppAddins)
|
|
{
|
|
FindClose(hFind);
|
|
WLog_ERR(TAG, "calloc failed!");
|
|
return NULL;
|
|
}
|
|
|
|
if (hFind == INVALID_HANDLE_VALUE)
|
|
return ppAddins;
|
|
|
|
do
|
|
{
|
|
BOOL used = FALSE;
|
|
FREERDP_ADDIN* pAddin = (FREERDP_ADDIN*)calloc(1, sizeof(FREERDP_ADDIN));
|
|
|
|
if (!pAddin)
|
|
{
|
|
WLog_ERR(TAG, "calloc failed!");
|
|
goto error_out;
|
|
}
|
|
|
|
nDashes = 0;
|
|
for (index = 0; FindData.cFileName[index]; index++)
|
|
nDashes += (FindData.cFileName[index] == '-') ? 1 : 0;
|
|
|
|
if (nDashes == 1)
|
|
{
|
|
size_t len;
|
|
char* p[2] = { 0 };
|
|
/* <name>-client.<extension> */
|
|
p[0] = FindData.cFileName;
|
|
p[1] = strchr(p[0], '-') + 1;
|
|
|
|
len = p[1] - p[0];
|
|
if (len < 1)
|
|
{
|
|
WLog_WARN(TAG, "Skipping file '%s', invalid format", FindData.cFileName);
|
|
goto skip;
|
|
}
|
|
strncpy(pAddin->cName, p[0], MIN(ARRAYSIZE(pAddin->cName), len - 1));
|
|
|
|
pAddin->dwFlags = FREERDP_ADDIN_CLIENT;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_DYNAMIC;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_NAME;
|
|
ppAddins[nAddins++] = pAddin;
|
|
|
|
used = TRUE;
|
|
}
|
|
else if (nDashes == 2)
|
|
{
|
|
size_t len;
|
|
char* p[4] = { 0 };
|
|
/* <name>-client-<subsystem>.<extension> */
|
|
p[0] = FindData.cFileName;
|
|
p[1] = strchr(p[0], '-') + 1;
|
|
p[2] = strchr(p[1], '-') + 1;
|
|
p[3] = strchr(p[2], '.') + 1;
|
|
|
|
len = p[1] - p[0];
|
|
if (len < 1)
|
|
{
|
|
WLog_WARN(TAG, "Skipping file '%s', invalid format", FindData.cFileName);
|
|
goto skip;
|
|
}
|
|
strncpy(pAddin->cName, p[0], MIN(ARRAYSIZE(pAddin->cName), len - 1));
|
|
|
|
len = p[3] - p[2];
|
|
if (len < 1)
|
|
{
|
|
WLog_WARN(TAG, "Skipping file '%s', invalid format", FindData.cFileName);
|
|
goto skip;
|
|
}
|
|
strncpy(pAddin->cSubsystem, p[2], MIN(ARRAYSIZE(pAddin->cSubsystem), len - 1));
|
|
|
|
pAddin->dwFlags = FREERDP_ADDIN_CLIENT;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_DYNAMIC;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_NAME;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_SUBSYSTEM;
|
|
ppAddins[nAddins++] = pAddin;
|
|
|
|
used = TRUE;
|
|
}
|
|
else if (nDashes == 3)
|
|
{
|
|
size_t len;
|
|
char* p[5] = { 0 };
|
|
/* <name>-client-<subsystem>-<type>.<extension> */
|
|
p[0] = FindData.cFileName;
|
|
p[1] = strchr(p[0], '-') + 1;
|
|
p[2] = strchr(p[1], '-') + 1;
|
|
p[3] = strchr(p[2], '-') + 1;
|
|
p[4] = strchr(p[3], '.') + 1;
|
|
|
|
len = p[1] - p[0];
|
|
if (len < 1)
|
|
{
|
|
WLog_WARN(TAG, "Skipping file '%s', invalid format", FindData.cFileName);
|
|
goto skip;
|
|
}
|
|
strncpy(pAddin->cName, p[0], MIN(ARRAYSIZE(pAddin->cName), len - 1));
|
|
|
|
len = p[3] - p[2];
|
|
if (len < 1)
|
|
{
|
|
WLog_WARN(TAG, "Skipping file '%s', invalid format", FindData.cFileName);
|
|
goto skip;
|
|
}
|
|
strncpy(pAddin->cSubsystem, p[2], MIN(ARRAYSIZE(pAddin->cSubsystem), len - 1));
|
|
|
|
len = p[4] - p[3];
|
|
if (len < 1)
|
|
{
|
|
WLog_WARN(TAG, "Skipping file '%s', invalid format", FindData.cFileName);
|
|
goto skip;
|
|
}
|
|
strncpy(pAddin->cType, p[3], MIN(ARRAYSIZE(pAddin->cType), len - 1));
|
|
|
|
pAddin->dwFlags = FREERDP_ADDIN_CLIENT;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_DYNAMIC;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_NAME;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_SUBSYSTEM;
|
|
pAddin->dwFlags |= FREERDP_ADDIN_TYPE;
|
|
ppAddins[nAddins++] = pAddin;
|
|
|
|
used = TRUE;
|
|
}
|
|
|
|
skip:
|
|
if (!used)
|
|
free(pAddin);
|
|
|
|
} while (FindNextFileA(hFind, &FindData));
|
|
|
|
FindClose(hFind);
|
|
ppAddins[nAddins] = NULL;
|
|
return ppAddins;
|
|
error_out:
|
|
FindClose(hFind);
|
|
freerdp_channels_addin_list_free(ppAddins);
|
|
return NULL;
|
|
}
|
|
|
|
FREERDP_ADDIN** freerdp_channels_list_addins(LPCSTR pszName, LPCSTR pszSubsystem, LPCSTR pszType,
|
|
DWORD dwFlags)
|
|
{
|
|
if (dwFlags & FREERDP_ADDIN_STATIC)
|
|
return freerdp_channels_list_client_static_addins(pszName, pszSubsystem, pszType, dwFlags);
|
|
else if (dwFlags & FREERDP_ADDIN_DYNAMIC)
|
|
return freerdp_channels_list_dynamic_addins(pszName, pszSubsystem, pszType, dwFlags);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
void freerdp_channels_addin_list_free(FREERDP_ADDIN** ppAddins)
|
|
{
|
|
size_t index;
|
|
|
|
if (!ppAddins)
|
|
return;
|
|
|
|
for (index = 0; ppAddins[index] != NULL; index++)
|
|
free(ppAddins[index]);
|
|
|
|
free(ppAddins);
|
|
}
|
|
|
|
extern const STATIC_ENTRY CLIENT_VirtualChannelEntryEx_TABLE[];
|
|
|
|
static BOOL freerdp_channels_is_virtual_channel_entry_ex(LPCSTR pszName)
|
|
{
|
|
size_t i;
|
|
|
|
for (i = 0; CLIENT_VirtualChannelEntryEx_TABLE[i].name != NULL; i++)
|
|
{
|
|
const STATIC_ENTRY* entry = &CLIENT_VirtualChannelEntryEx_TABLE[i];
|
|
|
|
if (!strncmp(entry->name, pszName, MAX_PATH))
|
|
return TRUE;
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
PVIRTUALCHANNELENTRY freerdp_channels_load_static_addin_entry(LPCSTR pszName, LPCSTR pszSubsystem,
|
|
LPCSTR pszType, DWORD dwFlags)
|
|
{
|
|
const STATIC_ADDIN_TABLE* table = CLIENT_STATIC_ADDIN_TABLE;
|
|
const char* type = NULL;
|
|
|
|
if (!pszName)
|
|
return NULL;
|
|
|
|
if (dwFlags & FREERDP_ADDIN_CHANNEL_DYNAMIC)
|
|
type = "DVCPluginEntry";
|
|
else if (dwFlags & FREERDP_ADDIN_CHANNEL_DEVICE)
|
|
type = "DeviceServiceEntry";
|
|
else if (dwFlags & FREERDP_ADDIN_CHANNEL_STATIC)
|
|
{
|
|
if (dwFlags & FREERDP_ADDIN_CHANNEL_ENTRYEX)
|
|
type = "VirtualChannelEntryEx";
|
|
else
|
|
type = "VirtualChannelEntry";
|
|
}
|
|
|
|
for (; table->name != NULL; table++)
|
|
{
|
|
if (strncmp(table->name, pszName, MAX_PATH) == 0)
|
|
{
|
|
if (type && strncmp(table->type, type, MAX_PATH))
|
|
continue;
|
|
|
|
if (pszSubsystem != NULL)
|
|
{
|
|
const STATIC_SUBSYSTEM_ENTRY* subsystems = table->table;
|
|
|
|
for (; subsystems->name != NULL; subsystems++)
|
|
{
|
|
/* If the pszSubsystem is an empty string use the default backend. */
|
|
if ((strnlen(pszSubsystem, 1) ==
|
|
0) || /* we only want to know if strnlen is > 0 */
|
|
(strncmp(subsystems->name, pszSubsystem, MAX_PATH) == 0))
|
|
{
|
|
if (pszType)
|
|
{
|
|
if (strncmp(subsystems->type, pszType, MAX_PATH) == 0)
|
|
return (PVIRTUALCHANNELENTRY)subsystems->entry;
|
|
}
|
|
else
|
|
{
|
|
return (PVIRTUALCHANNELENTRY)subsystems->entry;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (dwFlags & FREERDP_ADDIN_CHANNEL_ENTRYEX)
|
|
{
|
|
if (!freerdp_channels_is_virtual_channel_entry_ex(pszName))
|
|
return NULL;
|
|
}
|
|
|
|
return (PVIRTUALCHANNELENTRY)table->entry;
|
|
}
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|