Blame winpr/libwinpr/path/include/PathAllocCombine.c

Packit 1fb8d4
Packit 1fb8d4
/*
Packit 1fb8d4
#define DEFINE_UNICODE		FALSE
Packit 1fb8d4
#define _PATH_SEPARATOR_CHR	'\\'
Packit 1fb8d4
#define _PATH_SEPARATOR_STR	"\\"
Packit 1fb8d4
#define PATH_ALLOC_COMBINE	PathAllocCombineA
Packit 1fb8d4
*/
Packit 1fb8d4
Packit 1fb8d4
/**
Packit 1fb8d4
 * FIXME: These implementations of the PathAllocCombine functions have
Packit 1fb8d4
 * several issues:
Packit 1fb8d4
 * - pszPathIn or pszMore may be NULL (but not both)
Packit 1fb8d4
 * - no check if pszMore is fully qualified (if so, it must be directly
Packit 1fb8d4
 *   copied to the output buffer without being combined with pszPathIn.
Packit 1fb8d4
 * - if pszMore begins with a _single_ backslash it must be combined with
Packit 1fb8d4
 *   only the root of the path pointed to by pszPathIn and there's no code
Packit 1fb8d4
 *   to extract the root of pszPathIn.
Packit 1fb8d4
 * - the function will crash with some short string lengths of the parameters
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#if DEFINE_UNICODE
Packit 1fb8d4
Packit Service 5a9772
HRESULT PATH_ALLOC_COMBINE(PCWSTR pszPathIn, PCWSTR pszMore, unsigned long dwFlags,
Packit Service 5a9772
                           PWSTR* ppszPathOut)
Packit 1fb8d4
{
Packit 1fb8d4
#ifdef _WIN32
Packit 1fb8d4
	PWSTR pszPathOut;
Packit 1fb8d4
	BOOL backslashIn;
Packit 1fb8d4
	BOOL backslashMore;
Packit Service 5a9772
	size_t pszMoreLength;
Packit Service 5a9772
	size_t pszPathInLength;
Packit Service 5a9772
	size_t pszPathOutLength;
Packit 1fb8d4
	WLog_WARN(TAG, "%s: has known bugs and needs fixing.", __FUNCTION__);
Packit 1fb8d4
Packit 1fb8d4
	if (!ppszPathOut)
Packit 1fb8d4
		return E_INVALIDARG;
Packit 1fb8d4
Packit 1fb8d4
	if (!pszPathIn && !pszMore)
Packit 1fb8d4
		return E_INVALIDARG;
Packit 1fb8d4
Packit 1fb8d4
	if (!pszMore)
Packit 1fb8d4
		return E_FAIL; /* valid but not implemented, see top comment */
Packit 1fb8d4
Packit 1fb8d4
	if (!pszPathIn)
Packit Service 5a9772
		return E_FAIL; /* valid but not implemented, see top comment */
Packit 1fb8d4
Packit 1fb8d4
	pszPathInLength = wcslen(pszPathIn);
Packit 1fb8d4
	pszMoreLength = wcslen(pszMore);
Packit 1fb8d4
Packit 1fb8d4
	/* prevent segfaults - the complete implementation below is buggy */
Packit 1fb8d4
	if (pszPathInLength < 3)
Packit 1fb8d4
		return E_FAIL;
Packit 1fb8d4
Packit 1fb8d4
	backslashIn = (pszPathIn[pszPathInLength - 1] == _PATH_SEPARATOR_CHR) ? TRUE : FALSE;
Packit 1fb8d4
	backslashMore = (pszMore[0] == _PATH_SEPARATOR_CHR) ? TRUE : FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (backslashMore)
Packit 1fb8d4
	{
Packit 1fb8d4
		if ((pszPathIn[1] == ':') && (pszPathIn[2] == _PATH_SEPARATOR_CHR))
Packit 1fb8d4
		{
Packit 1fb8d4
			size_t sizeOfBuffer;
Packit 1fb8d4
			pszPathOutLength = 2 + pszMoreLength;
Packit 1fb8d4
			sizeOfBuffer = (pszPathOutLength + 1) * 2;
Packit Service 5a9772
			pszPathOut = (PWSTR)HeapAlloc(GetProcessHeap(), 0, sizeOfBuffer * 2);
Packit 1fb8d4
Packit 1fb8d4
			if (!pszPathOut)
Packit 1fb8d4
				return E_OUTOFMEMORY;
Packit 1fb8d4
Packit 1fb8d4
			swprintf_s(pszPathOut, sizeOfBuffer, L"%c:%s", pszPathIn[0], pszMore);
Packit 1fb8d4
			*ppszPathOut = pszPathOut;
Packit 1fb8d4
			return S_OK;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		size_t sizeOfBuffer;
Packit 1fb8d4
		pszPathOutLength = pszPathInLength + pszMoreLength;
Packit 1fb8d4
		sizeOfBuffer = (pszPathOutLength + 1) * 2;
Packit Service 5a9772
		pszPathOut = (PWSTR)HeapAlloc(GetProcessHeap(), 0, sizeOfBuffer * 2);
Packit 1fb8d4
Packit 1fb8d4
		if (!pszPathOut)
Packit 1fb8d4
			return E_OUTOFMEMORY;
Packit 1fb8d4
Packit 1fb8d4
		if (backslashIn)
Packit 1fb8d4
			swprintf_s(pszPathOut, sizeOfBuffer, L"%s%s", pszPathIn, pszMore);
Packit 1fb8d4
		else
Packit Service 5a9772
			swprintf_s(pszPathOut, sizeOfBuffer, L"%s" _PATH_SEPARATOR_STR L"%s", pszPathIn,
Packit Service 5a9772
			           pszMore);
Packit 1fb8d4
Packit 1fb8d4
		*ppszPathOut = pszPathOut;
Packit 1fb8d4
		return S_OK;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit Service 5a9772
#endif
Packit 1fb8d4
	return E_FAIL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
#else
Packit 1fb8d4
Packit 1fb8d4
HRESULT PATH_ALLOC_COMBINE(PCSTR pszPathIn, PCSTR pszMore, unsigned long dwFlags, PSTR* ppszPathOut)
Packit 1fb8d4
{
Packit 1fb8d4
	PSTR pszPathOut;
Packit 1fb8d4
	BOOL backslashIn;
Packit 1fb8d4
	BOOL backslashMore;
Packit 1fb8d4
	int pszMoreLength;
Packit 1fb8d4
	int pszPathInLength;
Packit 1fb8d4
	int pszPathOutLength;
Packit 1fb8d4
	WLog_WARN(TAG, "%s: has known bugs and needs fixing.", __FUNCTION__);
Packit 1fb8d4
Packit 1fb8d4
	if (!ppszPathOut)
Packit 1fb8d4
		return E_INVALIDARG;
Packit 1fb8d4
Packit 1fb8d4
	if (!pszPathIn && !pszMore)
Packit 1fb8d4
		return E_INVALIDARG;
Packit 1fb8d4
Packit 1fb8d4
	if (!pszMore)
Packit 1fb8d4
		return E_FAIL; /* valid but not implemented, see top comment */
Packit 1fb8d4
Packit 1fb8d4
	if (!pszPathIn)
Packit Service 5a9772
		return E_FAIL; /* valid but not implemented, see top comment */
Packit 1fb8d4
Packit 1fb8d4
	pszPathInLength = lstrlenA(pszPathIn);
Packit 1fb8d4
	pszMoreLength = lstrlenA(pszMore);
Packit 1fb8d4
Packit 1fb8d4
	/* prevent segfaults - the complete implementation below is buggy */
Packit 1fb8d4
	if (pszPathInLength < 3)
Packit 1fb8d4
		return E_FAIL;
Packit 1fb8d4
Packit 1fb8d4
	backslashIn = (pszPathIn[pszPathInLength - 1] == _PATH_SEPARATOR_CHR) ? TRUE : FALSE;
Packit 1fb8d4
	backslashMore = (pszMore[0] == _PATH_SEPARATOR_CHR) ? TRUE : FALSE;
Packit 1fb8d4
Packit 1fb8d4
	if (backslashMore)
Packit 1fb8d4
	{
Packit 1fb8d4
		if ((pszPathIn[1] == ':') && (pszPathIn[2] == _PATH_SEPARATOR_CHR))
Packit 1fb8d4
		{
Packit 1fb8d4
			size_t sizeOfBuffer;
Packit 1fb8d4
			pszPathOutLength = 2 + pszMoreLength;
Packit 1fb8d4
			sizeOfBuffer = (pszPathOutLength + 1) * 2;
Packit Service 5a9772
			pszPathOut = (PSTR)HeapAlloc(GetProcessHeap(), 0, sizeOfBuffer * 2);
Packit 1fb8d4
Packit 1fb8d4
			if (!pszPathOut)
Packit 1fb8d4
				return E_OUTOFMEMORY;
Packit 1fb8d4
Packit 1fb8d4
			sprintf_s(pszPathOut, sizeOfBuffer, "%c:%s", pszPathIn[0], pszMore);
Packit 1fb8d4
			*ppszPathOut = pszPathOut;
Packit 1fb8d4
			return S_OK;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		size_t sizeOfBuffer;
Packit 1fb8d4
		pszPathOutLength = pszPathInLength + pszMoreLength;
Packit 1fb8d4
		sizeOfBuffer = (pszPathOutLength + 1) * 2;
Packit Service 5a9772
		pszPathOut = (PSTR)HeapAlloc(GetProcessHeap(), 0, sizeOfBuffer * 2);
Packit 1fb8d4
Packit 1fb8d4
		if (!pszPathOut)
Packit 1fb8d4
			return E_OUTOFMEMORY;
Packit 1fb8d4
Packit 1fb8d4
		if (backslashIn)
Packit 1fb8d4
			sprintf_s(pszPathOut, sizeOfBuffer, "%s%s", pszPathIn, pszMore);
Packit 1fb8d4
		else
Packit 1fb8d4
			sprintf_s(pszPathOut, sizeOfBuffer, "%s" _PATH_SEPARATOR_STR "%s", pszPathIn, pszMore);
Packit 1fb8d4
Packit 1fb8d4
		*ppszPathOut = pszPathOut;
Packit 1fb8d4
		return S_OK;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return E_FAIL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
/*
Packit 1fb8d4
#undef DEFINE_UNICODE
Packit 1fb8d4
#undef _PATH_SEPARATOR_CHR
Packit 1fb8d4
#undef _PATH_SEPARATOR_STR
Packit 1fb8d4
#undef PATH_ALLOC_COMBINE
Packit 1fb8d4
*/