Commit b2e85f98 by Edward Thomson

win32: rename `git_win32__canonicalize_path`

The internal API `git_win32__canonicalize_path` is far, far too easily
confused with the internal API `git_win32_path_canonicalize`.  The
former removes the namespace prefix from a path (eg, given
`\\?\C:\Temp\foo`, it returns `C:\Temp\foo`, and given
`\\?\UNC\server\share`, it returns `\\server\share`).  As such, rename
it to `git_win32_path_remove_namespace`.

`git_win32_path_canonicalize` remains unchanged.
parent 3f096ca5
...@@ -220,7 +220,7 @@ int git_win32_path_from_utf8(git_win32_path out, const char *src) ...@@ -220,7 +220,7 @@ int git_win32_path_from_utf8(git_win32_path out, const char *src)
goto on_error; goto on_error;
} }
/* Skip the drive letter specification ("C:") */ /* Skip the drive letter specification ("C:") */
if (git__utf8_to_16(dest + 2, MAX_PATH - 2, src) < 0) if (git__utf8_to_16(dest + 2, MAX_PATH - 2, src) < 0)
goto on_error; goto on_error;
} }
...@@ -315,7 +315,7 @@ static bool path_is_volume(wchar_t *target, size_t target_len) ...@@ -315,7 +315,7 @@ static bool path_is_volume(wchar_t *target, size_t target_len)
} }
/* On success, returns the length, in characters, of the path stored in dest. /* On success, returns the length, in characters, of the path stored in dest.
* On failure, returns a negative value. */ * On failure, returns a negative value. */
int git_win32_path_readlink_w(git_win32_path dest, const git_win32_path path) int git_win32_path_readlink_w(git_win32_path dest, const git_win32_path path)
{ {
BYTE buf[MAXIMUM_REPARSE_DATA_BUFFER_SIZE]; BYTE buf[MAXIMUM_REPARSE_DATA_BUFFER_SIZE];
...@@ -360,16 +360,16 @@ int git_win32_path_readlink_w(git_win32_path dest, const git_win32_path path) ...@@ -360,16 +360,16 @@ int git_win32_path_readlink_w(git_win32_path dest, const git_win32_path path)
if (path_is_volume(target, target_len)) { if (path_is_volume(target, target_len)) {
/* This path is a reparse point that represents another volume mounted /* This path is a reparse point that represents another volume mounted
* at this location, it is not a symbolic link our input was canonical. * at this location, it is not a symbolic link our input was canonical.
*/ */
errno = EINVAL; errno = EINVAL;
error = -1; error = -1;
} else if (target_len) { } else if (target_len) {
/* The path may need to have a prefix removed. */ /* The path may need to have a namespace prefix removed. */
target_len = git_win32__canonicalize_path(target, target_len); target_len = git_win32_path_remove_namespace(target, target_len);
/* Need one additional character in the target buffer /* Need one additional character in the target buffer
* for the terminating NULL. */ * for the terminating NULL. */
if (GIT_WIN_PATH_UTF16 > target_len) { if (GIT_WIN_PATH_UTF16 > target_len) {
wcscpy(dest, target); wcscpy(dest, target);
error = (int)target_len; error = (int)target_len;
...@@ -380,3 +380,86 @@ on_error: ...@@ -380,3 +380,86 @@ on_error:
CloseHandle(handle); CloseHandle(handle);
return error; return error;
} }
/**
* Removes any trailing backslashes from a path, except in the case of a drive
* letter path (C:\, D:\, etc.). This function cannot fail.
*
* @param path The path which should be trimmed.
* @return The length of the modified string (<= the input length)
*/
size_t git_win32_path_trim_end(wchar_t *str, size_t len)
{
while (1) {
if (!len || str[len - 1] != L'\\')
break;
/*
* Don't trim backslashes from drive letter paths, which
* are 3 characters long and of the form C:\, D:\, etc.
*/
if (len == 3 && git_win32__isalpha(str[0]) && str[1] == ':')
break;
len--;
}
str[len] = L'\0';
return len;
}
/**
* Removes any of the following namespace prefixes from a path,
* if found: "\??\", "\\?\", "\\?\UNC\". This function cannot fail.
*
* @param path The path which should be converted.
* @return The length of the modified string (<= the input length)
*/
size_t git_win32_path_remove_namespace(wchar_t *str, size_t len)
{
static const wchar_t dosdevices_prefix[] = L"\\\?\?\\";
static const wchar_t nt_prefix[] = L"\\\\?\\";
static const wchar_t unc_prefix[] = L"UNC\\";
static const wchar_t unc_canonicalized_prefix[] = L"\\\\";
size_t to_advance = 0;
/* "\??\" -- DOS Devices prefix */
if (len >= CONST_STRLEN(dosdevices_prefix) &&
!wcsncmp(str, dosdevices_prefix, CONST_STRLEN(dosdevices_prefix))) {
to_advance += CONST_STRLEN(dosdevices_prefix);
len -= CONST_STRLEN(dosdevices_prefix);
}
/* "\\?\" -- NT namespace prefix */
else if (len >= CONST_STRLEN(nt_prefix) &&
!wcsncmp(str, nt_prefix, CONST_STRLEN(nt_prefix))) {
to_advance += CONST_STRLEN(nt_prefix);
len -= CONST_STRLEN(nt_prefix);
}
/* "\??\UNC\", "\\?\UNC\" -- UNC prefix */
if (to_advance && len >= CONST_STRLEN(unc_prefix) &&
!wcsncmp(str + to_advance, unc_prefix, CONST_STRLEN(unc_prefix))) {
/*
* The proper Win32 path for a UNC share has "\\" at beginning of it
* and looks like "\\server\share\<folderStructure>".
* So, remove the UNC prefix, but leave room for a "\\"
*/
to_advance += (CONST_STRLEN(unc_prefix) - CONST_STRLEN(unc_canonicalized_prefix));
len -= (CONST_STRLEN(unc_prefix) - CONST_STRLEN(unc_canonicalized_prefix));
/**
* Place a "\\" in the string so the result is "\\server\\share\<folderStructure>"
*/
memmove(str + to_advance, unc_canonicalized_prefix, CONST_STRLEN(unc_canonicalized_prefix) * sizeof(wchar_t));
}
if (to_advance) {
memmove(str, str + to_advance, len * sizeof(wchar_t));
str[len] = L'\0';
}
return git_win32_path_trim_end(str, len);
}
...@@ -83,4 +83,22 @@ extern char *git_win32_path_8dot3_name(const char *path); ...@@ -83,4 +83,22 @@ extern char *git_win32_path_8dot3_name(const char *path);
extern int git_win32_path_readlink_w(git_win32_path dest, const git_win32_path path); extern int git_win32_path_readlink_w(git_win32_path dest, const git_win32_path path);
/**
* Removes any trailing backslashes from a path, except in the case of a drive
* letter path (C:\, D:\, etc.). This function cannot fail.
*
* @param path The path which should be trimmed.
* @return The length of the modified string (<= the input length)
*/
size_t git_win32_path_trim_end(wchar_t *str, size_t len);
/**
* Removes any of the following namespace prefixes from a path,
* if found: "\??\", "\\?\", "\\?\UNC\". This function cannot fail.
*
* @param path The path which should be converted.
* @return The length of the modified string (<= the input length)
*/
size_t git_win32_path_remove_namespace(wchar_t *str, size_t len);
#endif #endif
...@@ -354,7 +354,7 @@ static int do_lstat(const char *path, struct stat *buf, bool posixly_correct) ...@@ -354,7 +354,7 @@ static int do_lstat(const char *path, struct stat *buf, bool posixly_correct)
if ((len = git_win32_path_from_utf8(path_w, path)) < 0) if ((len = git_win32_path_from_utf8(path_w, path)) < 0)
return -1; return -1;
git_win32__path_trim_end(path_w, len); git_win32_path_trim_end(path_w, len);
return lstat_w(path_w, buf, posixly_correct); return lstat_w(path_w, buf, posixly_correct);
} }
...@@ -648,8 +648,8 @@ static int getfinalpath_w( ...@@ -648,8 +648,8 @@ static int getfinalpath_w(
if (!dwChars || dwChars >= GIT_WIN_PATH_UTF16) if (!dwChars || dwChars >= GIT_WIN_PATH_UTF16)
return -1; return -1;
/* The path may be delivered to us with a prefix; canonicalize */ /* The path may be delivered to us with a namespace prefix; remove */
return (int)git_win32__canonicalize_path(dest, dwChars); return (int)git_win32_path_remove_namespace(dest, dwChars);
} }
static int follow_and_lstat_link(git_win32_path path, struct stat* buf) static int follow_and_lstat_link(git_win32_path path, struct stat* buf)
......
...@@ -93,83 +93,3 @@ int git_win32__hidden(bool *out, const char *path) ...@@ -93,83 +93,3 @@ int git_win32__hidden(bool *out, const char *path)
*out = (attrs & FILE_ATTRIBUTE_HIDDEN) ? true : false; *out = (attrs & FILE_ATTRIBUTE_HIDDEN) ? true : false;
return 0; return 0;
} }
/**
* Removes any trailing backslashes from a path, except in the case of a drive
* letter path (C:\, D:\, etc.). This function cannot fail.
*
* @param path The path which should be trimmed.
* @return The length of the modified string (<= the input length)
*/
size_t git_win32__path_trim_end(wchar_t *str, size_t len)
{
while (1) {
if (!len || str[len - 1] != L'\\')
break;
/* Don't trim backslashes from drive letter paths, which
* are 3 characters long and of the form C:\, D:\, etc. */
if (len == 3 && git_win32__isalpha(str[0]) && str[1] == ':')
break;
len--;
}
str[len] = L'\0';
return len;
}
/**
* Removes any of the following namespace prefixes from a path,
* if found: "\??\", "\\?\", "\\?\UNC\". This function cannot fail.
*
* @param path The path which should be converted.
* @return The length of the modified string (<= the input length)
*/
size_t git_win32__canonicalize_path(wchar_t *str, size_t len)
{
static const wchar_t dosdevices_prefix[] = L"\\\?\?\\";
static const wchar_t nt_prefix[] = L"\\\\?\\";
static const wchar_t unc_prefix[] = L"UNC\\";
static const wchar_t unc_canonicalized_prefix[] = L"\\\\";
size_t to_advance = 0;
/* "\??\" -- DOS Devices prefix */
if (len >= CONST_STRLEN(dosdevices_prefix) &&
!wcsncmp(str, dosdevices_prefix, CONST_STRLEN(dosdevices_prefix))) {
to_advance += CONST_STRLEN(dosdevices_prefix);
len -= CONST_STRLEN(dosdevices_prefix);
}
/* "\\?\" -- NT namespace prefix */
else if (len >= CONST_STRLEN(nt_prefix) &&
!wcsncmp(str, nt_prefix, CONST_STRLEN(nt_prefix))) {
to_advance += CONST_STRLEN(nt_prefix);
len -= CONST_STRLEN(nt_prefix);
}
/* "\??\UNC\", "\\?\UNC\" -- UNC prefix */
if (to_advance && len >= CONST_STRLEN(unc_prefix) &&
!wcsncmp(str + to_advance, unc_prefix, CONST_STRLEN(unc_prefix))) {
/**
* The proper Win32 path for a UNC share has "\\" at beginning of it
* and looks like "\\server\share\<folderStructure>".
* So, remove th UNC prefix, but leave room for a "\\"
*/
to_advance += (CONST_STRLEN(unc_prefix) - CONST_STRLEN(unc_canonicalized_prefix));
len -= (CONST_STRLEN(unc_prefix) - CONST_STRLEN(unc_canonicalized_prefix));
/**
* Place a "\\" in the string so the result is "\\server\\share\<folderStructure>"
*/
memmove(str + to_advance, unc_canonicalized_prefix, CONST_STRLEN(unc_canonicalized_prefix) * sizeof(wchar_t));
}
if (to_advance) {
memmove(str, str + to_advance, len * sizeof(wchar_t));
str[len] = L'\0';
}
return git_win32__path_trim_end(str, len);
}
...@@ -60,24 +60,6 @@ extern int git_win32__set_hidden(const char *path, bool hidden); ...@@ -60,24 +60,6 @@ extern int git_win32__set_hidden(const char *path, bool hidden);
extern int git_win32__hidden(bool *hidden, const char *path); extern int git_win32__hidden(bool *hidden, const char *path);
/** /**
* Removes any trailing backslashes from a path, except in the case of a drive
* letter path (C:\, D:\, etc.). This function cannot fail.
*
* @param path The path which should be trimmed.
* @return The length of the modified string (<= the input length)
*/
size_t git_win32__path_trim_end(wchar_t *str, size_t len);
/**
* Removes any of the following namespace prefixes from a path,
* if found: "\??\", "\\?\", "\\?\UNC\". This function cannot fail.
*
* @param path The path which should be converted.
* @return The length of the modified string (<= the input length)
*/
size_t git_win32__canonicalize_path(wchar_t *str, size_t len);
/**
* Converts a FILETIME structure to a struct timespec. * Converts a FILETIME structure to a struct timespec.
* *
* @param FILETIME A pointer to a FILETIME * @param FILETIME A pointer to a FILETIME
......
...@@ -129,7 +129,7 @@ void test_path_win32__absolute_from_relative(void) ...@@ -129,7 +129,7 @@ void test_path_win32__absolute_from_relative(void)
#endif #endif
} }
void static test_canonicalize(const wchar_t *in, const wchar_t *expected) static void test_canonicalize(const wchar_t *in, const wchar_t *expected)
{ {
#ifdef GIT_WIN32 #ifdef GIT_WIN32
git_win32_path canonical; git_win32_path canonical;
...@@ -145,7 +145,7 @@ void static test_canonicalize(const wchar_t *in, const wchar_t *expected) ...@@ -145,7 +145,7 @@ void static test_canonicalize(const wchar_t *in, const wchar_t *expected)
#endif #endif
} }
void static test_canonicalize_path(const wchar_t *in, const wchar_t *expected) static void test_remove_namespace(const wchar_t *in, const wchar_t *expected)
{ {
#ifdef GIT_WIN32 #ifdef GIT_WIN32
git_win32_path canonical; git_win32_path canonical;
...@@ -153,7 +153,7 @@ void static test_canonicalize_path(const wchar_t *in, const wchar_t *expected) ...@@ -153,7 +153,7 @@ void static test_canonicalize_path(const wchar_t *in, const wchar_t *expected)
cl_assert(wcslen(in) < MAX_PATH); cl_assert(wcslen(in) < MAX_PATH);
wcscpy(canonical, in); wcscpy(canonical, in);
cl_must_pass(git_win32__canonicalize_path(canonical, wcslen(in))); cl_must_pass(git_win32_path_remove_namespace(canonical, wcslen(in)));
cl_assert_equal_wcs(expected, canonical); cl_assert_equal_wcs(expected, canonical);
#else #else
GIT_UNUSED(in); GIT_UNUSED(in);
...@@ -161,9 +161,9 @@ void static test_canonicalize_path(const wchar_t *in, const wchar_t *expected) ...@@ -161,9 +161,9 @@ void static test_canonicalize_path(const wchar_t *in, const wchar_t *expected)
#endif #endif
} }
void test_path_win32__canonicalize_path(void) void test_path_win32__remove_namespace(void)
{ {
test_canonicalize_path(L"\\\\?\\UNC\\server\\C$\\folder", L"\\\\server\\C$\\folder"); test_remove_namespace(L"\\\\?\\UNC\\server\\C$\\folder", L"\\\\server\\C$\\folder");
} }
void test_path_win32__canonicalize(void) void test_path_win32__canonicalize(void)
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment