Blame src/win32/path_w32.c

Packit Service 20376f
/*
Packit Service 20376f
 * Copyright (C) the libgit2 contributors. All rights reserved.
Packit Service 20376f
 *
Packit Service 20376f
 * This file is part of libgit2, distributed under the GNU GPL v2 with
Packit Service 20376f
 * a Linking Exception. For full terms see the included COPYING file.
Packit Service 20376f
 */
Packit Service 20376f
Packit Service 20376f
#include "common.h"
Packit Service 20376f
#include "path.h"
Packit Service 20376f
#include "path_w32.h"
Packit Service 20376f
#include "utf-conv.h"
Packit Service 20376f
#include "posix.h"
Packit Service 20376f
#include "reparse.h"
Packit Service 20376f
#include "dir.h"
Packit Service 20376f
Packit Service 20376f
#define PATH__NT_NAMESPACE     L"\\\\?\\"
Packit Service 20376f
#define PATH__NT_NAMESPACE_LEN 4
Packit Service 20376f
Packit Service 20376f
#define PATH__ABSOLUTE_LEN     3
Packit Service 20376f
Packit Service 20376f
#define path__is_dirsep(p) ((p) == '/' || (p) == '\\')
Packit Service 20376f
Packit Service 20376f
#define path__is_absolute(p) \
Packit Service 20376f
	(git__isalpha((p)[0]) && (p)[1] == ':' && ((p)[2] == '\\' || (p)[2] == '/'))
Packit Service 20376f
Packit Service 20376f
#define path__is_nt_namespace(p) \
Packit Service 20376f
	(((p)[0] == '\\' && (p)[1] == '\\' && (p)[2] == '?' && (p)[3] == '\\') || \
Packit Service 20376f
	 ((p)[0] == '/' && (p)[1] == '/' && (p)[2] == '?' && (p)[3] == '/'))
Packit Service 20376f
Packit Service 20376f
#define path__is_unc(p) \
Packit Service 20376f
	(((p)[0] == '\\' && (p)[1] == '\\') || ((p)[0] == '/' && (p)[1] == '/'))
Packit Service 20376f
Packit Service 20376f
GIT_INLINE(int) path__cwd(wchar_t *path, int size)
Packit Service 20376f
{
Packit Service 20376f
	int len;
Packit Service 20376f
Packit Service 20376f
	if ((len = GetCurrentDirectoryW(size, path)) == 0) {
Packit Service 20376f
		errno = GetLastError() == ERROR_ACCESS_DENIED ? EACCES : ENOENT;
Packit Service 20376f
		return -1;
Packit Service 20376f
	} else if (len > size) {
Packit Service 20376f
		errno = ENAMETOOLONG;
Packit Service 20376f
		return -1;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* The Win32 APIs may return "\\?\" once you've used it first.
Packit Service 20376f
	 * But it may not.  What a gloriously predictible API!
Packit Service 20376f
	 */
Packit Service 20376f
	if (wcsncmp(path, PATH__NT_NAMESPACE, PATH__NT_NAMESPACE_LEN))
Packit Service 20376f
		return len;
Packit Service 20376f
Packit Service 20376f
	len -= PATH__NT_NAMESPACE_LEN;
Packit Service 20376f
Packit Service 20376f
	memmove(path, path + PATH__NT_NAMESPACE_LEN, sizeof(wchar_t) * len);
Packit Service 20376f
	return len;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static wchar_t *path__skip_server(wchar_t *path)
Packit Service 20376f
{
Packit Service 20376f
	wchar_t *c;
Packit Service 20376f
Packit Service 20376f
	for (c = path; *c; c++) {
Packit Service 20376f
		if (path__is_dirsep(*c))
Packit Service 20376f
			return c + 1;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return c;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static wchar_t *path__skip_prefix(wchar_t *path)
Packit Service 20376f
{
Packit Service 20376f
	if (path__is_nt_namespace(path)) {
Packit Service 20376f
		path += PATH__NT_NAMESPACE_LEN;
Packit Service 20376f
Packit Service 20376f
		if (wcsncmp(path, L"UNC\\", 4) == 0)
Packit Service 20376f
			path = path__skip_server(path + 4);
Packit Service 20376f
		else if (path__is_absolute(path))
Packit Service 20376f
			path += PATH__ABSOLUTE_LEN;
Packit Service 20376f
	} else if (path__is_absolute(path)) {
Packit Service 20376f
		path += PATH__ABSOLUTE_LEN;
Packit Service 20376f
	} else if (path__is_unc(path)) {
Packit Service 20376f
		path = path__skip_server(path + 2);
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return path;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_win32_path_canonicalize(git_win32_path path)
Packit Service 20376f
{
Packit Service 20376f
	wchar_t *base, *from, *to, *next;
Packit Service 20376f
	size_t len;
Packit Service 20376f
Packit Service 20376f
	base = to = path__skip_prefix(path);
Packit Service 20376f
Packit Service 20376f
	/* Unposixify if the prefix */
Packit Service 20376f
	for (from = path; from < to; from++) {
Packit Service 20376f
		if (*from == L'/')
Packit Service 20376f
			*from = L'\\';
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	while (*from) {
Packit Service 20376f
		for (next = from; *next; ++next) {
Packit Service 20376f
			if (*next == L'/') {
Packit Service 20376f
				*next = L'\\';
Packit Service 20376f
				break;
Packit Service 20376f
			}
Packit Service 20376f
Packit Service 20376f
			if (*next == L'\\')
Packit Service 20376f
				break;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		len = next - from;
Packit Service 20376f
Packit Service 20376f
		if (len == 1 && from[0] == L'.')
Packit Service 20376f
			/* do nothing with singleton dot */;
Packit Service 20376f
Packit Service 20376f
		else if (len == 2 && from[0] == L'.' && from[1] == L'.') {
Packit Service 20376f
			if (to == base) {
Packit Service 20376f
				/* no more path segments to strip, eat the "../" */
Packit Service 20376f
				if (*next == L'\\')
Packit Service 20376f
					len++;
Packit Service 20376f
Packit Service 20376f
				base = to;
Packit Service 20376f
			} else {
Packit Service 20376f
				/* back up a path segment */
Packit Service 20376f
				while (to > base && to[-1] == L'\\') to--;
Packit Service 20376f
				while (to > base && to[-1] != L'\\') to--;
Packit Service 20376f
			}
Packit Service 20376f
		} else {
Packit Service 20376f
			if (*next == L'\\' && *from != L'\\')
Packit Service 20376f
				len++;
Packit Service 20376f
Packit Service 20376f
			if (to != from)
Packit Service 20376f
				memmove(to, from, sizeof(wchar_t) * len);
Packit Service 20376f
Packit Service 20376f
			to += len;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		from += len;
Packit Service 20376f
Packit Service 20376f
		while (*from == L'\\') from++;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* Strip trailing backslashes */
Packit Service 20376f
	while (to > base && to[-1] == L'\\') to--;
Packit Service 20376f
Packit Service 20376f
	*to = L'\0';
Packit Service 20376f
Packit Service 20376f
	return (to - path);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_win32_path__cwd(wchar_t *out, size_t len)
Packit Service 20376f
{
Packit Service 20376f
	int cwd_len;
Packit Service 20376f
Packit Service 20376f
	if ((cwd_len = path__cwd(out, len)) < 0)
Packit Service 20376f
		return -1;
Packit Service 20376f
Packit Service 20376f
	/* UNC paths */
Packit Service 20376f
	if (wcsncmp(L"\\\\", out, 2) == 0) {
Packit Service 20376f
		/* Our buffer must be at least 5 characters larger than the
Packit Service 20376f
		 * current working directory:  we swallow one of the leading
Packit Service 20376f
		 * '\'s, but we we add a 'UNC' specifier to the path, plus
Packit Service 20376f
		 * a trailing directory separator, plus a NUL.
Packit Service 20376f
		 */
Packit Service 20376f
		if (cwd_len > MAX_PATH - 4) {
Packit Service 20376f
			errno = ENAMETOOLONG;
Packit Service 20376f
			return -1;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		memmove(out+2, out, sizeof(wchar_t) * cwd_len);
Packit Service 20376f
		out[0] = L'U';
Packit Service 20376f
		out[1] = L'N';
Packit Service 20376f
		out[2] = L'C';
Packit Service 20376f
Packit Service 20376f
		cwd_len += 2;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	/* Our buffer must be at least 2 characters larger than the current
Packit Service 20376f
	 * working directory.  (One character for the directory separator,
Packit Service 20376f
	 * one for the null.
Packit Service 20376f
	 */
Packit Service 20376f
	else if (cwd_len > MAX_PATH - 2) {
Packit Service 20376f
		errno = ENAMETOOLONG;
Packit Service 20376f
		return -1;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return cwd_len;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_win32_path_from_utf8(git_win32_path out, const char *src)
Packit Service 20376f
{
Packit Service 20376f
	wchar_t *dest = out;
Packit Service 20376f
Packit Service 20376f
	/* All win32 paths are in NT-prefixed format, beginning with "\\?\". */
Packit Service 20376f
	memcpy(dest, PATH__NT_NAMESPACE, sizeof(wchar_t) * PATH__NT_NAMESPACE_LEN);
Packit Service 20376f
	dest += PATH__NT_NAMESPACE_LEN;
Packit Service 20376f
Packit Service 20376f
	/* See if this is an absolute path (beginning with a drive letter) */
Packit Service 20376f
	if (path__is_absolute(src)) {
Packit Service 20376f
		if (git__utf8_to_16(dest, MAX_PATH, src) < 0)
Packit Service 20376f
			goto on_error;
Packit Service 20376f
	}
Packit Service 20376f
	/* File-prefixed NT-style paths beginning with \\?\ */
Packit Service 20376f
	else if (path__is_nt_namespace(src)) {
Packit Service 20376f
		/* Skip the NT prefix, the destination already contains it */
Packit Service 20376f
		if (git__utf8_to_16(dest, MAX_PATH, src + PATH__NT_NAMESPACE_LEN) < 0)
Packit Service 20376f
			goto on_error;
Packit Service 20376f
	}
Packit Service 20376f
	/* UNC paths */
Packit Service 20376f
	else if (path__is_unc(src)) {
Packit Service 20376f
		memcpy(dest, L"UNC\\", sizeof(wchar_t) * 4);
Packit Service 20376f
		dest += 4;
Packit Service 20376f
Packit Service 20376f
		/* Skip the leading "\\" */
Packit Service 20376f
		if (git__utf8_to_16(dest, MAX_PATH - 2, src + 2) < 0)
Packit Service 20376f
			goto on_error;
Packit Service 20376f
	}
Packit Service 20376f
	/* Absolute paths omitting the drive letter */
Packit Service 20376f
	else if (src[0] == '\\' || src[0] == '/') {
Packit Service 20376f
		if (path__cwd(dest, MAX_PATH) < 0)
Packit Service 20376f
			goto on_error;
Packit Service 20376f
Packit Service 20376f
		if (!path__is_absolute(dest)) {
Packit Service 20376f
			errno = ENOENT;
Packit Service 20376f
			goto on_error;
Packit Service 20376f
		}
Packit Service 20376f
Packit Service 20376f
		/* Skip the drive letter specification ("C:") */	
Packit Service 20376f
		if (git__utf8_to_16(dest + 2, MAX_PATH - 2, src) < 0)
Packit Service 20376f
			goto on_error;
Packit Service 20376f
	}
Packit Service 20376f
	/* Relative paths */
Packit Service 20376f
	else {
Packit Service 20376f
		int cwd_len;
Packit Service 20376f
Packit Service 20376f
		if ((cwd_len = git_win32_path__cwd(dest, MAX_PATH)) < 0)
Packit Service 20376f
			goto on_error;
Packit Service 20376f
Packit Service 20376f
		dest[cwd_len++] = L'\\';
Packit Service 20376f
Packit Service 20376f
		if (git__utf8_to_16(dest + cwd_len, MAX_PATH - cwd_len, src) < 0)
Packit Service 20376f
			goto on_error;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	return git_win32_path_canonicalize(out);
Packit Service 20376f
Packit Service 20376f
on_error:
Packit Service 20376f
	/* set windows error code so we can use its error message */
Packit Service 20376f
	if (errno == ENAMETOOLONG)
Packit Service 20376f
		SetLastError(ERROR_FILENAME_EXCED_RANGE);
Packit Service 20376f
Packit Service 20376f
	return -1;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
int git_win32_path_to_utf8(git_win32_utf8_path dest, const wchar_t *src)
Packit Service 20376f
{
Packit Service 20376f
	char *out = dest;
Packit Service 20376f
	int len;
Packit Service 20376f
Packit Service 20376f
	/* Strip NT namespacing "\\?\" */
Packit Service 20376f
	if (path__is_nt_namespace(src)) {
Packit Service 20376f
		src += 4;
Packit Service 20376f
Packit Service 20376f
		/* "\\?\UNC\server\share" -> "\\server\share" */
Packit Service 20376f
		if (wcsncmp(src, L"UNC\\", 4) == 0) {
Packit Service 20376f
			src += 4;
Packit Service 20376f
Packit Service 20376f
			memcpy(dest, "\\\\", 2);
Packit Service 20376f
			out = dest + 2;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if ((len = git__utf16_to_8(out, GIT_WIN_PATH_UTF8, src)) < 0)
Packit Service 20376f
		return len;
Packit Service 20376f
Packit Service 20376f
	git_path_mkposix(dest);
Packit Service 20376f
Packit Service 20376f
	return len;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
char *git_win32_path_8dot3_name(const char *path)
Packit Service 20376f
{
Packit Service 20376f
	git_win32_path longpath, shortpath;
Packit Service 20376f
	wchar_t *start;
Packit Service 20376f
	char *shortname;
Packit Service 20376f
	int len, namelen = 1;
Packit Service 20376f
Packit Service 20376f
	if (git_win32_path_from_utf8(longpath, path) < 0)
Packit Service 20376f
		return NULL;
Packit Service 20376f
Packit Service 20376f
	len = GetShortPathNameW(longpath, shortpath, GIT_WIN_PATH_UTF16);
Packit Service 20376f
Packit Service 20376f
	while (len && shortpath[len-1] == L'\\')
Packit Service 20376f
		shortpath[--len] = L'\0';
Packit Service 20376f
Packit Service 20376f
	if (len == 0 || len >= GIT_WIN_PATH_UTF16)
Packit Service 20376f
		return NULL;
Packit Service 20376f
Packit Service 20376f
	for (start = shortpath + (len - 1);
Packit Service 20376f
		start > shortpath && *(start-1) != '/' && *(start-1) != '\\';
Packit Service 20376f
		start--)
Packit Service 20376f
		namelen++;
Packit Service 20376f
Packit Service 20376f
	/* We may not have actually been given a short name.  But if we have,
Packit Service 20376f
	 * it will be in the ASCII byte range, so we don't need to worry about
Packit Service 20376f
	 * multi-byte sequences and can allocate naively.
Packit Service 20376f
	 */
Packit Service 20376f
	if (namelen > 12 || (shortname = git__malloc(namelen + 1)) == NULL)
Packit Service 20376f
		return NULL;
Packit Service 20376f
Packit Service 20376f
	if ((len = git__utf16_to_8(shortname, namelen + 1, start)) < 0)
Packit Service 20376f
		return NULL;
Packit Service 20376f
Packit Service 20376f
	return shortname;
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
static bool path_is_volume(wchar_t *target, size_t target_len)
Packit Service 20376f
{
Packit Service 20376f
	return (target_len && wcsncmp(target, L"\\??\\Volume{", 11) == 0);
Packit Service 20376f
}
Packit Service 20376f
Packit Service 20376f
/* On success, returns the length, in characters, of the path stored in dest.
Packit Service 20376f
* On failure, returns a negative value. */
Packit Service 20376f
int git_win32_path_readlink_w(git_win32_path dest, const git_win32_path path)
Packit Service 20376f
{
Packit Service 20376f
	BYTE buf[MAXIMUM_REPARSE_DATA_BUFFER_SIZE];
Packit Service 20376f
	GIT_REPARSE_DATA_BUFFER *reparse_buf = (GIT_REPARSE_DATA_BUFFER *)buf;
Packit Service 20376f
	HANDLE handle = NULL;
Packit Service 20376f
	DWORD ioctl_ret;
Packit Service 20376f
	wchar_t *target;
Packit Service 20376f
	size_t target_len;
Packit Service 20376f
Packit Service 20376f
	int error = -1;
Packit Service 20376f
Packit Service 20376f
	handle = CreateFileW(path, GENERIC_READ,
Packit Service 20376f
		FILE_SHARE_READ | FILE_SHARE_DELETE, NULL, OPEN_EXISTING,
Packit Service 20376f
		FILE_FLAG_OPEN_REPARSE_POINT | FILE_FLAG_BACKUP_SEMANTICS, NULL);
Packit Service 20376f
Packit Service 20376f
	if (handle == INVALID_HANDLE_VALUE) {
Packit Service 20376f
		errno = ENOENT;
Packit Service 20376f
		return -1;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (!DeviceIoControl(handle, FSCTL_GET_REPARSE_POINT, NULL, 0,
Packit Service 20376f
		reparse_buf, sizeof(buf), &ioctl_ret, NULL)) {
Packit Service 20376f
		errno = EINVAL;
Packit Service 20376f
		goto on_error;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	switch (reparse_buf->ReparseTag) {
Packit Service 20376f
	case IO_REPARSE_TAG_SYMLINK:
Packit Service 20376f
		target = reparse_buf->SymbolicLinkReparseBuffer.PathBuffer +
Packit Service 20376f
			(reparse_buf->SymbolicLinkReparseBuffer.SubstituteNameOffset / sizeof(WCHAR));
Packit Service 20376f
		target_len = reparse_buf->SymbolicLinkReparseBuffer.SubstituteNameLength / sizeof(WCHAR);
Packit Service 20376f
	break;
Packit Service 20376f
	case IO_REPARSE_TAG_MOUNT_POINT:
Packit Service 20376f
		target = reparse_buf->MountPointReparseBuffer.PathBuffer +
Packit Service 20376f
			(reparse_buf->MountPointReparseBuffer.SubstituteNameOffset / sizeof(WCHAR));
Packit Service 20376f
		target_len = reparse_buf->MountPointReparseBuffer.SubstituteNameLength / sizeof(WCHAR);
Packit Service 20376f
	break;
Packit Service 20376f
	default:
Packit Service 20376f
		errno = EINVAL;
Packit Service 20376f
		goto on_error;
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
	if (path_is_volume(target, target_len)) {
Packit Service 20376f
		/* This path is a reparse point that represents another volume mounted
Packit Service 20376f
		* at this location, it is not a symbolic link our input was canonical.
Packit Service 20376f
		*/
Packit Service 20376f
		errno = EINVAL;
Packit Service 20376f
		error = -1;
Packit Service 20376f
	} else if (target_len) {
Packit Service 20376f
		/* The path may need to have a prefix removed. */
Packit Service 20376f
		target_len = git_win32__canonicalize_path(target, target_len);
Packit Service 20376f
Packit Service 20376f
		/* Need one additional character in the target buffer
Packit Service 20376f
		* for the terminating NULL. */
Packit Service 20376f
		if (GIT_WIN_PATH_UTF16 > target_len) {
Packit Service 20376f
			wcscpy(dest, target);
Packit Service 20376f
			error = (int)target_len;
Packit Service 20376f
		}
Packit Service 20376f
	}
Packit Service 20376f
Packit Service 20376f
on_error:
Packit Service 20376f
	CloseHandle(handle);
Packit Service 20376f
	return error;
Packit Service 20376f
}