2018-03-02 12:14:54 +01:00
|
|
|
/*
|
|
|
|
* Copyright 2018 Nikolay Sivov
|
2018-11-20 12:05:46 +01:00
|
|
|
* Copyright 2018 Zhiyi Zhang
|
2018-03-02 12:14:54 +01:00
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2.1 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdarg.h>
|
|
|
|
|
|
|
|
#include "windef.h"
|
|
|
|
#include "winbase.h"
|
|
|
|
#include "pathcch.h"
|
|
|
|
#include "strsafe.h"
|
|
|
|
|
|
|
|
#include "wine/debug.h"
|
|
|
|
#include "wine/unicode.h"
|
|
|
|
|
|
|
|
WINE_DEFAULT_DEBUG_CHANNEL(path);
|
|
|
|
|
2018-11-20 12:06:11 +01:00
|
|
|
static SIZE_T strnlenW(const WCHAR *string, SIZE_T maxlen)
|
|
|
|
{
|
|
|
|
SIZE_T i;
|
|
|
|
|
|
|
|
for (i = 0; i < maxlen; i++)
|
|
|
|
if (!string[i]) break;
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
|
2018-03-02 12:14:54 +01:00
|
|
|
HRESULT WINAPI PathCchAddBackslash(WCHAR *path, SIZE_T size)
|
|
|
|
{
|
|
|
|
return PathCchAddBackslashEx(path, size, NULL, NULL);
|
|
|
|
}
|
|
|
|
|
|
|
|
HRESULT WINAPI PathCchAddBackslashEx(WCHAR *path, SIZE_T size, WCHAR **endptr, SIZE_T *remaining)
|
|
|
|
{
|
|
|
|
BOOL needs_termination;
|
|
|
|
SIZE_T length;
|
|
|
|
|
|
|
|
TRACE("%s, %lu, %p, %p\n", debugstr_w(path), size, endptr, remaining);
|
|
|
|
|
|
|
|
length = strlenW(path);
|
|
|
|
needs_termination = size && length && path[length - 1] != '\\';
|
|
|
|
|
|
|
|
if (length >= (needs_termination ? size - 1 : size))
|
|
|
|
{
|
|
|
|
if (endptr) *endptr = NULL;
|
|
|
|
if (remaining) *remaining = 0;
|
|
|
|
return STRSAFE_E_INSUFFICIENT_BUFFER;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!needs_termination)
|
|
|
|
{
|
|
|
|
if (endptr) *endptr = path + length;
|
|
|
|
if (remaining) *remaining = size - length;
|
|
|
|
return S_FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
path[length++] = '\\';
|
|
|
|
path[length] = 0;
|
|
|
|
|
|
|
|
if (endptr) *endptr = path + length;
|
|
|
|
if (remaining) *remaining = size - length;
|
|
|
|
|
|
|
|
return S_OK;
|
|
|
|
}
|
2018-11-20 12:05:46 +01:00
|
|
|
|
2018-11-20 12:06:11 +01:00
|
|
|
HRESULT WINAPI PathCchAddExtension(WCHAR *path, SIZE_T size, const WCHAR *extension)
|
|
|
|
{
|
|
|
|
const WCHAR *existing_extension, *next;
|
|
|
|
SIZE_T path_length, extension_length, dot_length;
|
|
|
|
BOOL has_dot;
|
|
|
|
HRESULT hr;
|
|
|
|
|
|
|
|
TRACE("%s %lu %s\n", wine_dbgstr_w(path), size, wine_dbgstr_w(extension));
|
|
|
|
|
|
|
|
if (!path || !size || size > PATHCCH_MAX_CCH || !extension) return E_INVALIDARG;
|
|
|
|
|
|
|
|
next = extension;
|
|
|
|
while (*next)
|
|
|
|
{
|
|
|
|
if ((*next == '.' && next > extension) || *next == ' ' || *next == '\\') return E_INVALIDARG;
|
|
|
|
next++;
|
|
|
|
}
|
|
|
|
|
|
|
|
has_dot = extension[0] == '.' ? TRUE : FALSE;
|
|
|
|
|
|
|
|
hr = PathCchFindExtension(path, size, &existing_extension);
|
|
|
|
if (FAILED(hr)) return hr;
|
|
|
|
if (*existing_extension) return S_FALSE;
|
|
|
|
|
|
|
|
path_length = strnlenW(path, size);
|
|
|
|
dot_length = has_dot ? 0 : 1;
|
|
|
|
extension_length = strlenW(extension);
|
|
|
|
|
|
|
|
if (path_length + dot_length + extension_length + 1 > size) return STRSAFE_E_INSUFFICIENT_BUFFER;
|
|
|
|
|
|
|
|
/* If extension is empty or only dot, return S_OK with path unchanged */
|
|
|
|
if (!extension[0] || (extension[0] == '.' && !extension[1])) return S_OK;
|
|
|
|
|
|
|
|
if (!has_dot)
|
|
|
|
{
|
|
|
|
path[path_length] = '.';
|
|
|
|
path_length++;
|
|
|
|
}
|
|
|
|
|
|
|
|
strcpyW(path + path_length, extension);
|
|
|
|
return S_OK;
|
|
|
|
}
|
|
|
|
|
2018-11-20 12:05:46 +01:00
|
|
|
HRESULT WINAPI PathCchFindExtension(const WCHAR *path, SIZE_T size, const WCHAR **extension)
|
|
|
|
{
|
|
|
|
const WCHAR *lastpoint = NULL;
|
|
|
|
SIZE_T counter = 0;
|
|
|
|
|
|
|
|
TRACE("%s %lu %p\n", wine_dbgstr_w(path), size, extension);
|
|
|
|
|
|
|
|
if (!path || !size || size > PATHCCH_MAX_CCH)
|
|
|
|
{
|
|
|
|
*extension = NULL;
|
|
|
|
return E_INVALIDARG;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (*path)
|
|
|
|
{
|
|
|
|
if (*path == '\\' || *path == ' ')
|
|
|
|
lastpoint = NULL;
|
|
|
|
else if (*path == '.')
|
|
|
|
lastpoint = path;
|
|
|
|
|
|
|
|
path++;
|
|
|
|
counter++;
|
|
|
|
if (counter == size || counter == PATHCCH_MAX_CCH)
|
|
|
|
{
|
|
|
|
*extension = NULL;
|
|
|
|
return E_INVALIDARG;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*extension = lastpoint ? lastpoint : path;
|
|
|
|
return S_OK;
|
|
|
|
}
|