* path.cc (struct symlink_info): Change size of contents member to
be able to keep SYMLINK_MAX sized strings. (symlink_worker): Rework for long path names. When writing windows shortcuts, store pathname additionally "hidden" after the actual shortcut data to workaround size limitations of the description field. (symlink_info::check_shortcut): Drop file name parameter. Drop max file size check. Use NT functions. Use appended full path if available, description otherwise. Check symlink string length for not exceeding SYMLINK_MAX. Don't close file here. (symlink_info::check_sysfile): Drop file name parameter. Use NT functions. Check symlink string length for not exceeding SYMLINK_MAX. Don't close file here. (symlink_info::check_reparse_point): Drop file name parameter. Drop useless length checks. Allow SYMLINK_MAX length symlink strings. Don't close file here. (symlink_info::posixify): Allow SYMLINK_MAX length symlink strings. (symlink_info::check): Turn around checking for symlink file attributes. Use NT functions. Close symlink file here. * include/limits.h (PATH_MAX): Define as 32760. Change comment. (SYMLINK_MAX): Define as PATH_MAX - 1.
This commit is contained in:
parent
4797f5bca3
commit
044b62c767
@ -1,3 +1,26 @@
|
|||||||
|
2007-10-10 Corinna Vinschen <corinna@vinschen.de>
|
||||||
|
|
||||||
|
* path.cc (struct symlink_info): Change size of contents member to
|
||||||
|
be able to keep SYMLINK_MAX sized strings.
|
||||||
|
(symlink_worker): Rework for long path names. When writing windows
|
||||||
|
shortcuts, store pathname additionally "hidden" after the actual
|
||||||
|
shortcut data to workaround size limitations of the description field.
|
||||||
|
(symlink_info::check_shortcut): Drop file name parameter. Drop max
|
||||||
|
file size check. Use NT functions. Use appended full path if
|
||||||
|
available, description otherwise. Check symlink string length for
|
||||||
|
not exceeding SYMLINK_MAX. Don't close file here.
|
||||||
|
(symlink_info::check_sysfile): Drop file name parameter. Use NT
|
||||||
|
functions. Check symlink string length for not exceeding SYMLINK_MAX.
|
||||||
|
Don't close file here.
|
||||||
|
(symlink_info::check_reparse_point): Drop file name parameter. Drop
|
||||||
|
useless length checks. Allow SYMLINK_MAX length symlink strings.
|
||||||
|
Don't close file here.
|
||||||
|
(symlink_info::posixify): Allow SYMLINK_MAX length symlink strings.
|
||||||
|
(symlink_info::check): Turn around checking for symlink file attributes.
|
||||||
|
Use NT functions. Close symlink file here.
|
||||||
|
* include/limits.h (PATH_MAX): Define as 32760. Change comment.
|
||||||
|
(SYMLINK_MAX): Define as PATH_MAX - 1.
|
||||||
|
|
||||||
2007-10-10 Corinna Vinschen <corinna@vinschen.de>
|
2007-10-10 Corinna Vinschen <corinna@vinschen.de>
|
||||||
|
|
||||||
* fhandler_socket.cc (fhandler_socket::bind): Open file for deletion,
|
* fhandler_socket.cc (fhandler_socket::bind): Open file for deletion,
|
||||||
|
@ -318,9 +318,10 @@ details. */
|
|||||||
#undef NAME_MAX
|
#undef NAME_MAX
|
||||||
#define NAME_MAX 255
|
#define NAME_MAX 255
|
||||||
|
|
||||||
/* Maximum length of a path */
|
/* Maximum length of a path including trailing NUL.
|
||||||
|
(32767 - max. native NT path prefix) */
|
||||||
#undef PATH_MAX
|
#undef PATH_MAX
|
||||||
#define PATH_MAX 260
|
#define PATH_MAX 32760
|
||||||
|
|
||||||
/* # of bytes in a pipe buf. This is the max # of bytes which can be
|
/* # of bytes in a pipe buf. This is the max # of bytes which can be
|
||||||
written to a pipe in one atomic operation. */
|
written to a pipe in one atomic operation. */
|
||||||
@ -352,7 +353,7 @@ details. */
|
|||||||
|
|
||||||
/* Maximum number of bytes in a symbolic link. */
|
/* Maximum number of bytes in a symbolic link. */
|
||||||
#undef SYMLINK_MAX
|
#undef SYMLINK_MAX
|
||||||
#define SYMLINK_MAX PATH_MAX
|
#define SYMLINK_MAX (PATH_MAX - 1)
|
||||||
|
|
||||||
|
|
||||||
/* Runtime Increasable Values */
|
/* Runtime Increasable Values */
|
||||||
|
@ -89,7 +89,7 @@ static void backslashify (const char *, char *, int);
|
|||||||
|
|
||||||
struct symlink_info
|
struct symlink_info
|
||||||
{
|
{
|
||||||
char contents[CYG_MAX_PATH + 4];
|
char contents[SYMLINK_MAX + 1];
|
||||||
char *ext_here;
|
char *ext_here;
|
||||||
int extn;
|
int extn;
|
||||||
unsigned pflags;
|
unsigned pflags;
|
||||||
@ -106,9 +106,9 @@ struct symlink_info
|
|||||||
int set (char *path);
|
int set (char *path);
|
||||||
bool parse_device (const char *);
|
bool parse_device (const char *);
|
||||||
bool case_check (char *path);
|
bool case_check (char *path);
|
||||||
int check_sysfile (const char *path, HANDLE h);
|
int check_sysfile (HANDLE h);
|
||||||
int check_shortcut (const char *path, HANDLE h);
|
int check_shortcut (HANDLE h);
|
||||||
int check_reparse_point (const char *path, HANDLE h);
|
int check_reparse_point (HANDLE h);
|
||||||
int posixify (char *srcbuf);
|
int posixify (char *srcbuf);
|
||||||
bool set_error (int);
|
bool set_error (int);
|
||||||
};
|
};
|
||||||
@ -2874,16 +2874,17 @@ int
|
|||||||
symlink_worker (const char *oldpath, const char *newpath, bool use_winsym,
|
symlink_worker (const char *oldpath, const char *newpath, bool use_winsym,
|
||||||
bool isdevice)
|
bool isdevice)
|
||||||
{
|
{
|
||||||
HANDLE h;
|
|
||||||
int res = -1;
|
int res = -1;
|
||||||
path_conv win32_path, win32_oldpath;
|
size_t len;
|
||||||
char from[CYG_MAX_PATH + 5];
|
path_conv win32_newpath, win32_oldpath;
|
||||||
char cwd[CYG_MAX_PATH], *cp = NULL, c = 0;
|
char *buf, *cp;
|
||||||
char w32oldpath[CYG_MAX_PATH];
|
|
||||||
char reloldpath[CYG_MAX_PATH] = { 0 };
|
|
||||||
DWORD written;
|
|
||||||
SECURITY_ATTRIBUTES sa = sec_none_nih;
|
SECURITY_ATTRIBUTES sa = sec_none_nih;
|
||||||
security_descriptor sd;
|
security_descriptor sd;
|
||||||
|
OBJECT_ATTRIBUTES attr;
|
||||||
|
IO_STATUS_BLOCK io;
|
||||||
|
NTSTATUS status;
|
||||||
|
HANDLE fh;
|
||||||
|
FILE_BASIC_INFORMATION fbi;
|
||||||
|
|
||||||
/* POSIX says that empty 'newpath' is invalid input while empty
|
/* POSIX says that empty 'newpath' is invalid input while empty
|
||||||
'oldpath' is valid -- it's symlink resolver job to verify if
|
'oldpath' is valid -- it's symlink resolver job to verify if
|
||||||
@ -2897,192 +2898,239 @@ symlink_worker (const char *oldpath, const char *newpath, bool use_winsym,
|
|||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strlen (oldpath) >= CYG_MAX_PATH)
|
if (strlen (oldpath) > SYMLINK_MAX)
|
||||||
{
|
{
|
||||||
set_errno (ENAMETOOLONG);
|
set_errno (ENAMETOOLONG);
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
win32_path.check (newpath, PC_SYM_NOFOLLOW,
|
len = strlen (newpath);
|
||||||
transparent_exe ? stat_suffixes : NULL);
|
/* Trailing dirsep is a no-no. */
|
||||||
if (use_winsym && !win32_path.exists ())
|
if (isdirsep (newpath[len - 1]))
|
||||||
{
|
{
|
||||||
strcpy (from, newpath);
|
set_errno (ENOENT);
|
||||||
strcat (from, ".lnk");
|
goto done;
|
||||||
win32_path.check (from, PC_SYM_NOFOLLOW);
|
}
|
||||||
|
/* We need the normalized full path below. */
|
||||||
|
win32_newpath.check (newpath, PC_SYM_NOFOLLOW | PC_POSIX, stat_suffixes);
|
||||||
|
if (use_winsym && !win32_newpath.exists ())
|
||||||
|
{
|
||||||
|
char *newplnk = (char *) alloca (len + 5);
|
||||||
|
stpcpy (stpcpy (newplnk, newpath), ".lnk");
|
||||||
|
win32_newpath.check (newplnk, PC_SYM_NOFOLLOW | PC_POSIX);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (win32_path.error)
|
if (win32_newpath.error)
|
||||||
{
|
{
|
||||||
set_errno (win32_path.case_clash ? ECASECLASH : win32_path.error);
|
set_errno (win32_newpath.case_clash ? ECASECLASH : win32_newpath.error);
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
syscall_printf ("symlink (%s, %s)", oldpath, win32_path.get_win32 ());
|
syscall_printf ("symlink (%s, %S)", oldpath,
|
||||||
|
win32_newpath.get_nt_native_path ());
|
||||||
|
|
||||||
if ((!isdevice && win32_path.exists ())
|
if ((!isdevice && win32_newpath.exists ())
|
||||||
|| win32_path.is_auto_device ())
|
|| win32_newpath.is_auto_device ())
|
||||||
{
|
{
|
||||||
set_errno (EEXIST);
|
set_errno (EEXIST);
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
|
||||||
DWORD create_how;
|
if (use_winsym)
|
||||||
if (!use_winsym)
|
|
||||||
create_how = CREATE_NEW;
|
|
||||||
else if (isdevice)
|
|
||||||
{
|
{
|
||||||
strcpy (w32oldpath, oldpath);
|
ITEMIDLIST *pidl = NULL;
|
||||||
create_how = CREATE_ALWAYS;
|
size_t full_len = 0;
|
||||||
SetFileAttributes (win32_path.get_win32 (), FILE_ATTRIBUTE_NORMAL);
|
unsigned short oldpath_len, desc_len, relpath_len, pidl_len = 0;
|
||||||
}
|
char desc[MAX_PATH + 1], *relpath;
|
||||||
else
|
|
||||||
{
|
|
||||||
if (!isabspath (oldpath))
|
|
||||||
{
|
|
||||||
getcwd (cwd, CYG_MAX_PATH);
|
|
||||||
if ((cp = strrchr (from, '/')) || (cp = strrchr (from, '\\')))
|
|
||||||
{
|
|
||||||
c = *cp;
|
|
||||||
*cp = '\0';
|
|
||||||
chdir (from);
|
|
||||||
}
|
|
||||||
backslashify (oldpath, reloldpath, 0);
|
|
||||||
/* Creating an ITEMIDLIST requires an absolute path. So if we
|
|
||||||
create a shortcut file, we create relative and absolute Win32
|
|
||||||
paths, the first for the relpath field and the latter for the
|
|
||||||
ITEMIDLIST field. */
|
|
||||||
if (GetFileAttributes (reloldpath) == INVALID_FILE_ATTRIBUTES)
|
|
||||||
{
|
|
||||||
win32_oldpath.check (oldpath, PC_SYM_NOFOLLOW,
|
|
||||||
transparent_exe ? stat_suffixes : NULL);
|
|
||||||
if (win32_oldpath.error != ENOENT)
|
|
||||||
strcpy (use_winsym ? reloldpath : w32oldpath,
|
|
||||||
win32_oldpath.get_win32 ());
|
|
||||||
}
|
|
||||||
else if (!use_winsym)
|
|
||||||
strcpy (w32oldpath, reloldpath);
|
|
||||||
if (use_winsym)
|
|
||||||
{
|
|
||||||
win32_oldpath.check (oldpath, PC_SYM_NOFOLLOW,
|
|
||||||
transparent_exe ? stat_suffixes : NULL);
|
|
||||||
strcpy (w32oldpath, win32_oldpath.get_win32 ());
|
|
||||||
}
|
|
||||||
if (cp)
|
|
||||||
{
|
|
||||||
*cp = c;
|
|
||||||
chdir (cwd);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
win32_oldpath.check (oldpath, PC_SYM_NOFOLLOW,
|
|
||||||
transparent_exe ? stat_suffixes : NULL);
|
|
||||||
strcpy (w32oldpath, win32_oldpath.get_win32 ());
|
|
||||||
}
|
|
||||||
create_how = CREATE_NEW;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (allow_ntsec && win32_path.has_acls ())
|
if (!isdevice)
|
||||||
set_security_attribute (S_IFLNK | STD_RBITS | STD_WBITS,
|
{
|
||||||
&sa, sd);
|
/* First create an IDLIST to learn how big our shortcut is
|
||||||
|
going to be. */
|
||||||
h = CreateFile (win32_path.get_win32 (), GENERIC_WRITE, 0, &sa, create_how,
|
|
||||||
FILE_ATTRIBUTE_NORMAL, 0);
|
|
||||||
if (h == INVALID_HANDLE_VALUE)
|
|
||||||
__seterrno ();
|
|
||||||
else
|
|
||||||
{
|
|
||||||
bool success = false;
|
|
||||||
|
|
||||||
if (use_winsym)
|
|
||||||
{
|
|
||||||
/* A path of 240 chars with 120 one character directories in it
|
|
||||||
can result in a 6K shortcut. */
|
|
||||||
char *buf = (char *) alloca (8192);
|
|
||||||
win_shortcut_hdr *shortcut_header = (win_shortcut_hdr *) buf;
|
|
||||||
HRESULT hres;
|
|
||||||
IShellFolder *psl;
|
IShellFolder *psl;
|
||||||
WCHAR wc_path[CYG_MAX_PATH];
|
|
||||||
ITEMIDLIST *pidl = NULL, *p;
|
|
||||||
unsigned short len;
|
|
||||||
|
|
||||||
memset (shortcut_header, 0, sizeof *shortcut_header);
|
/* The symlink target is relative to the directory in which
|
||||||
shortcut_header->size = sizeof *shortcut_header;
|
the symlink gets created, not relative to the cwd. Therefore
|
||||||
shortcut_header->magic = GUID_shortcut;
|
we have to mangle the path quite a bit before calling path_conv. */
|
||||||
shortcut_header->flags = (WSH_FLAG_DESC | WSH_FLAG_RELPATH);
|
if (!isabspath (oldpath))
|
||||||
shortcut_header->run = SW_NORMAL;
|
|
||||||
cp = buf + sizeof (win_shortcut_hdr);
|
|
||||||
/* Creating an IDLIST */
|
|
||||||
hres = SHGetDesktopFolder (&psl);
|
|
||||||
if (SUCCEEDED (hres))
|
|
||||||
{
|
{
|
||||||
MultiByteToWideChar (CP_ACP, 0, w32oldpath, -1, wc_path,
|
len = strrchr (win32_newpath.normalized_path, '/')
|
||||||
CYG_MAX_PATH);
|
- win32_newpath.normalized_path + 1;
|
||||||
hres = psl->ParseDisplayName (NULL, NULL, wc_path, NULL,
|
char *absoldpath = (char *) alloca (len + strlen (oldpath) + 1);
|
||||||
&pidl, NULL);
|
stpcpy (stpncpy (absoldpath, win32_newpath.normalized_path, len),
|
||||||
if (SUCCEEDED (hres))
|
oldpath);
|
||||||
|
win32_oldpath.check (absoldpath, PC_SYM_NOFOLLOW, stat_suffixes);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
win32_oldpath.check (oldpath, PC_SYM_NOFOLLOW, stat_suffixes);
|
||||||
|
if (SUCCEEDED (SHGetDesktopFolder (&psl)))
|
||||||
|
{
|
||||||
|
WCHAR wc_path[win32_oldpath.get_wide_win32_path_len () + 1];
|
||||||
|
win32_oldpath.get_wide_win32_path (wc_path);
|
||||||
|
/* Amazing but true: Even though the ParseDisplayName method
|
||||||
|
takes a wide char path name, it does not understand the
|
||||||
|
Win32 prefix for long pathnames! So we have to tack off
|
||||||
|
the prefix and convert tyhe path to the "normal" syntax
|
||||||
|
for ParseDisplayName. I have no idea if it's able to take
|
||||||
|
long path names at all since I can't test it right now. */
|
||||||
|
WCHAR *wc = wc_path + 4;
|
||||||
|
if (!wcscmp (wc, L"UNC\\"))
|
||||||
|
*(wc += 2) = L'\\';
|
||||||
|
HRESULT res;
|
||||||
|
if (SUCCEEDED (res = psl->ParseDisplayName (NULL, NULL, wc, NULL,
|
||||||
|
&pidl, NULL)))
|
||||||
{
|
{
|
||||||
shortcut_header->flags |= WSH_FLAG_IDLIST;
|
ITEMIDLIST *p;
|
||||||
|
|
||||||
for (p = pidl; p->mkid.cb > 0;
|
for (p = pidl; p->mkid.cb > 0;
|
||||||
p = (ITEMIDLIST *)((char *) p + p->mkid.cb))
|
p = (ITEMIDLIST *)((char *) p + p->mkid.cb))
|
||||||
;
|
;
|
||||||
len = (char *) p - (char *) pidl + 2;
|
pidl_len = (char *) p - (char *) pidl + 2;
|
||||||
*(unsigned short *)cp = len;
|
|
||||||
memcpy (cp += 2, pidl, len);
|
|
||||||
cp += len;
|
|
||||||
CoTaskMemFree (pidl);
|
|
||||||
}
|
}
|
||||||
psl->Release ();
|
psl->Release ();
|
||||||
}
|
}
|
||||||
/* Creating a description */
|
}
|
||||||
*(unsigned short *)cp = len = strlen (oldpath);
|
/* Compute size of shortcut file. */
|
||||||
memcpy (cp += 2, oldpath, len);
|
full_len = sizeof (win_shortcut_hdr);
|
||||||
cp += len;
|
if (pidl_len)
|
||||||
/* Creating a relpath */
|
full_len += sizeof (unsigned short) + pidl_len;
|
||||||
if (reloldpath[0])
|
oldpath_len = strlen (oldpath);
|
||||||
{
|
/* Unfortunately the length of the description is restricted to a
|
||||||
*(unsigned short *)cp = len = strlen (reloldpath);
|
length of MAX_PATH up to NT4, and to a length of 2000 bytes
|
||||||
memcpy (cp += 2, reloldpath, len);
|
since W2K. We don't want to add considerations for the different
|
||||||
}
|
lengths and even 2000 bytes is not enough for long path names.
|
||||||
else
|
So what we do here is to set the description to the POSIX path
|
||||||
{
|
only if the path is not longer than MAX_PATH characters. We
|
||||||
*(unsigned short *)cp = len = strlen (w32oldpath);
|
append the full path name after the regular shortcut data
|
||||||
memcpy (cp += 2, w32oldpath, len);
|
(see below), which works fine with Windows Explorer as well
|
||||||
}
|
as older Cygwin versions (as long as the whole file isn't bigger
|
||||||
cp += len;
|
than 8K). The description field is only used for backward
|
||||||
success = WriteFile (h, buf, cp - buf, &written, NULL)
|
compatibility to older Cygwin versions and those versions are
|
||||||
&& written == (DWORD) (cp - buf);
|
not capable of handling long path names anyway. */
|
||||||
|
desc_len = stpcpy (desc, oldpath_len > MAX_PATH
|
||||||
|
? "[path too long]" : oldpath) - desc;
|
||||||
|
full_len += sizeof (unsigned short) + desc_len;
|
||||||
|
/* Devices get the oldpath string unchanged as relative path. */
|
||||||
|
if (isdevice)
|
||||||
|
{
|
||||||
|
relpath_len = oldpath_len;
|
||||||
|
stpcpy (relpath = (char *) alloca (relpath_len + 1), oldpath);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
/* This is the old technique creating a symlink. */
|
relpath_len = strlen (win32_oldpath.get_win32 ());
|
||||||
char buf[sizeof (SYMLINK_COOKIE) + CYG_MAX_PATH + 10];
|
stpcpy (relpath = (char *) alloca (relpath_len + 1),
|
||||||
|
win32_oldpath.get_win32 ());
|
||||||
__small_sprintf (buf, "%s%s", SYMLINK_COOKIE, oldpath);
|
|
||||||
DWORD len = strlen (buf) + 1;
|
|
||||||
|
|
||||||
/* Note that the terminating nul is written. */
|
|
||||||
success = WriteFile (h, buf, len, &written, NULL)
|
|
||||||
|| written != len;
|
|
||||||
|
|
||||||
}
|
}
|
||||||
if (success)
|
full_len += sizeof (unsigned short) + relpath_len;
|
||||||
{
|
full_len += sizeof (unsigned short) + oldpath_len;
|
||||||
CloseHandle (h);
|
/* 1 byte more for trailing 0 written by stpcpy. */
|
||||||
DWORD attr = use_winsym ? FILE_ATTRIBUTE_READONLY
|
buf = (char *) alloca (full_len + 1);
|
||||||
: FILE_ATTRIBUTE_SYSTEM;
|
|
||||||
SetFileAttributes (win32_path.get_win32 (), attr);
|
|
||||||
|
|
||||||
res = 0;
|
/* Create shortcut header */
|
||||||
}
|
win_shortcut_hdr *shortcut_header = (win_shortcut_hdr *) buf;
|
||||||
else
|
memset (shortcut_header, 0, sizeof *shortcut_header);
|
||||||
|
shortcut_header->size = sizeof *shortcut_header;
|
||||||
|
shortcut_header->magic = GUID_shortcut;
|
||||||
|
shortcut_header->flags = (WSH_FLAG_DESC | WSH_FLAG_RELPATH);
|
||||||
|
if (pidl)
|
||||||
|
shortcut_header->flags |= WSH_FLAG_IDLIST;
|
||||||
|
shortcut_header->run = SW_NORMAL;
|
||||||
|
cp = buf + sizeof (win_shortcut_hdr);
|
||||||
|
|
||||||
|
/* Create IDLIST */
|
||||||
|
if (pidl)
|
||||||
{
|
{
|
||||||
__seterrno ();
|
*(unsigned short *)cp = pidl_len;
|
||||||
CloseHandle (h);
|
memcpy (cp += 2, pidl, pidl_len);
|
||||||
DeleteFileA (win32_path.get_win32 ());
|
cp += pidl_len;
|
||||||
|
CoTaskMemFree (pidl);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Create description */
|
||||||
|
*(unsigned short *)cp = desc_len;
|
||||||
|
cp = stpcpy (cp += 2, desc);
|
||||||
|
|
||||||
|
/* Create relpath */
|
||||||
|
*(unsigned short *)cp = relpath_len;
|
||||||
|
cp = stpcpy (cp += 2, relpath);
|
||||||
|
|
||||||
|
/* Append the POSIX path after the regular shortcut data for
|
||||||
|
the long path support. */
|
||||||
|
*(unsigned short *)cp = oldpath_len;
|
||||||
|
cp = stpcpy (cp += 2, oldpath);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
/* This is the old technique creating a symlink. */
|
||||||
|
unsigned short oldpath_len = strlen (oldpath);
|
||||||
|
buf = (char *) alloca (sizeof (SYMLINK_COOKIE) + oldpath_len + 1);
|
||||||
|
/* Note that the terminating nul is written. */
|
||||||
|
cp = stpcpy (stpcpy (buf, SYMLINK_COOKIE), oldpath) + 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isdevice && win32_newpath.exists ())
|
||||||
|
{
|
||||||
|
status = NtOpenFile (&fh, FILE_WRITE_ATTRIBUTES,
|
||||||
|
win32_newpath.get_object_attr (attr, sa),
|
||||||
|
&io, 0, FILE_OPEN_FOR_BACKUP_INTENT);
|
||||||
|
if (!NT_SUCCESS (status))
|
||||||
|
{
|
||||||
|
__seterrno_from_nt_status (status);
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
fbi.CreationTime.QuadPart = fbi.LastAccessTime.QuadPart
|
||||||
|
= fbi.LastWriteTime.QuadPart = fbi.ChangeTime.QuadPart = 0LL;
|
||||||
|
fbi.FileAttributes = FILE_ATTRIBUTE_NORMAL;
|
||||||
|
status = NtSetInformationFile (fh, &io, &fbi, sizeof fbi,
|
||||||
|
FileBasicInformation);
|
||||||
|
NtClose (fh);
|
||||||
|
if (!NT_SUCCESS (status))
|
||||||
|
{
|
||||||
|
__seterrno_from_nt_status (status);
|
||||||
|
goto done;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
if (allow_ntsec && win32_newpath.has_acls ())
|
||||||
|
set_security_attribute (S_IFLNK | STD_RBITS | STD_WBITS,
|
||||||
|
&sa, sd);
|
||||||
|
status = NtCreateFile (&fh, DELETE | FILE_GENERIC_WRITE,
|
||||||
|
win32_newpath.get_object_attr (attr, sa),
|
||||||
|
&io, NULL, FILE_ATTRIBUTE_NORMAL,
|
||||||
|
FILE_SHARE_VALID_FLAGS,
|
||||||
|
isdevice ? FILE_OVERWRITE_IF : FILE_CREATE,
|
||||||
|
FILE_SYNCHRONOUS_IO_NONALERT
|
||||||
|
| FILE_NON_DIRECTORY_FILE
|
||||||
|
| FILE_OPEN_FOR_BACKUP_INTENT,
|
||||||
|
NULL, 0);
|
||||||
|
if (!NT_SUCCESS (status))
|
||||||
|
{
|
||||||
|
__seterrno_from_nt_status (status);
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
status = NtWriteFile (fh, NULL, NULL, NULL, &io, buf, cp - buf, NULL, NULL);
|
||||||
|
if (NT_SUCCESS (status) && io.Information == (ULONG) (cp - buf))
|
||||||
|
{
|
||||||
|
fbi.CreationTime.QuadPart = fbi.LastAccessTime.QuadPart
|
||||||
|
= fbi.LastWriteTime.QuadPart = fbi.ChangeTime.QuadPart = 0LL;
|
||||||
|
fbi.FileAttributes = use_winsym ? FILE_ATTRIBUTE_READONLY
|
||||||
|
: FILE_ATTRIBUTE_SYSTEM;
|
||||||
|
status = NtSetInformationFile (fh, &io, &fbi, sizeof fbi,
|
||||||
|
FileBasicInformation);
|
||||||
|
if (!NT_SUCCESS (status))
|
||||||
|
debug_printf ("Setting attributes failed, status = %p", status);
|
||||||
|
res = 0;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
__seterrno_from_nt_status (status);
|
||||||
|
FILE_DISPOSITION_INFORMATION fdi = { TRUE };
|
||||||
|
status = NtSetInformationFile (fh, &io, &fdi, sizeof fdi,
|
||||||
|
FileDispositionInformation);
|
||||||
|
if (!NT_SUCCESS (status))
|
||||||
|
debug_printf ("Setting delete dispostion failed, status = %p", status);
|
||||||
|
}
|
||||||
|
NtClose (fh);
|
||||||
|
|
||||||
done:
|
done:
|
||||||
syscall_printf ("%d = symlink_worker (%s, %s, %d, %d)", res, oldpath,
|
syscall_printf ("%d = symlink_worker (%s, %s, %d, %d)", res, oldpath,
|
||||||
@ -3104,80 +3152,98 @@ cmp_shortcut_header (win_shortcut_hdr *file_header)
|
|||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
symlink_info::check_shortcut (const char *path, HANDLE h)
|
symlink_info::check_shortcut (HANDLE h)
|
||||||
{
|
{
|
||||||
win_shortcut_hdr *file_header;
|
win_shortcut_hdr *file_header;
|
||||||
char *buf, *cp;
|
char *buf, *cp;
|
||||||
unsigned short len;
|
unsigned short len;
|
||||||
int res = 0;
|
int res = 0;
|
||||||
DWORD size, got = 0;
|
DWORD size;
|
||||||
|
IO_STATUS_BLOCK io;
|
||||||
|
|
||||||
if ((size = GetFileSize (h, NULL)) > 8192) /* Not a Cygwin symlink. */
|
size = GetFileSize (h, NULL);
|
||||||
goto file_not_symlink;
|
|
||||||
buf = (char *) alloca (size + 1);
|
buf = (char *) alloca (size + 1);
|
||||||
if (!ReadFile (h, buf, size, &got, 0))
|
if (!NT_SUCCESS (NtReadFile (h, NULL, NULL, NULL,
|
||||||
|
&io, buf, size, NULL, NULL)))
|
||||||
{
|
{
|
||||||
set_error (EIO);
|
set_error (EIO);
|
||||||
goto close_it;
|
goto close_it;
|
||||||
}
|
}
|
||||||
file_header = (win_shortcut_hdr *) buf;
|
file_header = (win_shortcut_hdr *) buf;
|
||||||
if (got != size || !cmp_shortcut_header (file_header))
|
if (io.Information != size || !cmp_shortcut_header (file_header))
|
||||||
goto file_not_symlink;
|
goto file_not_symlink;
|
||||||
cp = buf + sizeof (win_shortcut_hdr);
|
cp = buf + sizeof (win_shortcut_hdr);
|
||||||
if (file_header->flags & WSH_FLAG_IDLIST) /* Skip ITEMIDLIST */
|
if (file_header->flags & WSH_FLAG_IDLIST) /* Skip ITEMIDLIST */
|
||||||
cp += *(unsigned short *) cp + 2;
|
cp += *(unsigned short *) cp + 2;
|
||||||
if ((len = *(unsigned short *) cp) == 0 || len >= CYG_MAX_PATH)
|
if (!(len = *(unsigned short *) cp))
|
||||||
goto file_not_symlink;
|
goto file_not_symlink;
|
||||||
cp += 2;
|
cp += 2;
|
||||||
cp[len] = '\0';
|
|
||||||
/* Check if this is a device file - these start with the sequence :\\ */
|
/* Check if this is a device file - these start with the sequence :\\ */
|
||||||
if (strncmp (cp, ":\\", 2) == 0)
|
if (strncmp (cp, ":\\", 2) == 0)
|
||||||
res = strlen (strcpy (contents, cp)); /* Don't try to mess with device files */
|
res = strlen (strcpy (contents, cp)); /* Don't mess with device files */
|
||||||
else
|
else
|
||||||
res = posixify (cp);
|
{
|
||||||
|
/* Has appended full path? If so, use it instead of description. */
|
||||||
|
unsigned short relpath_len = *(unsigned short *) (cp + len);
|
||||||
|
if (cp + len + 2 + relpath_len < buf + size)
|
||||||
|
{
|
||||||
|
cp += len + 2 + relpath_len;
|
||||||
|
len = *(unsigned short *) cp;
|
||||||
|
cp += 2;
|
||||||
|
}
|
||||||
|
if (len > SYMLINK_MAX)
|
||||||
|
goto file_not_symlink;
|
||||||
|
cp[len] = '\0';
|
||||||
|
res = posixify (cp);
|
||||||
|
}
|
||||||
if (res) /* It's a symlink. */
|
if (res) /* It's a symlink. */
|
||||||
pflags = PATH_SYMLINK | PATH_LNK;
|
pflags = PATH_SYMLINK | PATH_LNK;
|
||||||
goto close_it;
|
goto close_it;
|
||||||
|
|
||||||
file_not_symlink:
|
file_not_symlink:
|
||||||
/* Not a symlink, see if executable. */
|
/* Not a symlink, see if executable. */
|
||||||
if (!(pflags & PATH_ALL_EXEC) && has_exec_chars ((const char *) &file_header, got))
|
if (!(pflags & PATH_ALL_EXEC) && has_exec_chars ((const char *) &file_header, io.Information))
|
||||||
pflags |= PATH_EXEC;
|
pflags |= PATH_EXEC;
|
||||||
|
|
||||||
close_it:
|
close_it:
|
||||||
CloseHandle (h);
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
symlink_info::check_sysfile (const char *path, HANDLE h)
|
symlink_info::check_sysfile (HANDLE h)
|
||||||
{
|
{
|
||||||
char cookie_buf[sizeof (SYMLINK_COOKIE) - 1];
|
char cookie_buf[sizeof (SYMLINK_COOKIE) - 1];
|
||||||
char srcbuf[CYG_MAX_PATH];
|
char srcbuf[SYMLINK_MAX + 2];
|
||||||
DWORD got;
|
IO_STATUS_BLOCK io;
|
||||||
int res = 0;
|
int res = 0;
|
||||||
|
|
||||||
if (!ReadFile (h, cookie_buf, sizeof (cookie_buf), &got, 0))
|
if (!NT_SUCCESS (NtReadFile (h, NULL, NULL, NULL, &io,
|
||||||
|
cookie_buf, sizeof (cookie_buf), NULL, NULL)))
|
||||||
{
|
{
|
||||||
debug_printf ("ReadFile1 failed");
|
debug_printf ("ReadFile1 failed");
|
||||||
set_error (EIO);
|
set_error (EIO);
|
||||||
}
|
}
|
||||||
else if (got == sizeof (cookie_buf)
|
else if (io.Information == sizeof (cookie_buf)
|
||||||
&& memcmp (cookie_buf, SYMLINK_COOKIE, sizeof (cookie_buf)) == 0)
|
&& memcmp (cookie_buf, SYMLINK_COOKIE, sizeof (cookie_buf)) == 0)
|
||||||
{
|
{
|
||||||
/* It's a symlink. */
|
/* It's a symlink. */
|
||||||
pflags = PATH_SYMLINK;
|
pflags = PATH_SYMLINK;
|
||||||
|
|
||||||
res = ReadFile (h, srcbuf, CYG_MAX_PATH, &got, 0);
|
if (!NT_SUCCESS (NtReadFile (h, NULL, NULL, NULL, &io,
|
||||||
if (!res)
|
srcbuf, SYMLINK_MAX + 2, NULL, NULL)))
|
||||||
{
|
{
|
||||||
debug_printf ("ReadFile2 failed");
|
debug_printf ("ReadFile2 failed");
|
||||||
set_error (EIO);
|
set_error (EIO);
|
||||||
}
|
}
|
||||||
|
else if (io.Information > SYMLINK_MAX + 1)
|
||||||
|
{
|
||||||
|
debug_printf ("symlink string too long");
|
||||||
|
set_error (EIO);
|
||||||
|
}
|
||||||
else
|
else
|
||||||
res = posixify (srcbuf);
|
res = posixify (srcbuf);
|
||||||
}
|
}
|
||||||
else if (got == sizeof (cookie_buf)
|
else if (io.Information == sizeof (cookie_buf)
|
||||||
&& memcmp (cookie_buf, SOCKET_COOKIE, sizeof (cookie_buf)) == 0)
|
&& memcmp (cookie_buf, SOCKET_COOKIE, sizeof (cookie_buf)) == 0)
|
||||||
pflags |= PATH_SOCKET;
|
pflags |= PATH_SOCKET;
|
||||||
else
|
else
|
||||||
@ -3185,44 +3251,32 @@ symlink_info::check_sysfile (const char *path, HANDLE h)
|
|||||||
/* Not a symlink, see if executable. */
|
/* Not a symlink, see if executable. */
|
||||||
if (pflags & PATH_ALL_EXEC)
|
if (pflags & PATH_ALL_EXEC)
|
||||||
/* Nothing to do */;
|
/* Nothing to do */;
|
||||||
else if (has_exec_chars (cookie_buf, got))
|
else if (has_exec_chars (cookie_buf, io.Information))
|
||||||
pflags |= PATH_EXEC;
|
pflags |= PATH_EXEC;
|
||||||
else
|
else
|
||||||
pflags |= PATH_NOTEXEC;
|
pflags |= PATH_NOTEXEC;
|
||||||
}
|
}
|
||||||
syscall_printf ("%d = symlink.check_sysfile (%s, %s) (%p)",
|
|
||||||
res, path, contents, pflags);
|
|
||||||
|
|
||||||
CloseHandle (h);
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
symlink_info::check_reparse_point (const char *path, HANDLE h)
|
symlink_info::check_reparse_point (HANDLE h)
|
||||||
{
|
{
|
||||||
int res = 0;
|
|
||||||
PREPARSE_DATA_BUFFER rp = (PREPARSE_DATA_BUFFER)
|
PREPARSE_DATA_BUFFER rp = (PREPARSE_DATA_BUFFER)
|
||||||
alloca (MAXIMUM_REPARSE_DATA_BUFFER_SIZE);
|
alloca (MAXIMUM_REPARSE_DATA_BUFFER_SIZE);
|
||||||
DWORD size;
|
DWORD size;
|
||||||
char srcbuf[CYG_MAX_PATH + 6];
|
char srcbuf[SYMLINK_MAX + 7];
|
||||||
|
|
||||||
if (!DeviceIoControl (h, FSCTL_GET_REPARSE_POINT, NULL, 0, (LPVOID) rp,
|
if (!DeviceIoControl (h, FSCTL_GET_REPARSE_POINT, NULL, 0, (LPVOID) rp,
|
||||||
MAXIMUM_REPARSE_DATA_BUFFER_SIZE, &size, NULL))
|
MAXIMUM_REPARSE_DATA_BUFFER_SIZE, &size, NULL))
|
||||||
{
|
{
|
||||||
debug_printf ("DeviceIoControl(FSCTL_GET_REPARSE_POINT) failed, %E");
|
debug_printf ("DeviceIoControl(FSCTL_GET_REPARSE_POINT) failed, %E");
|
||||||
set_error (EIO);
|
set_error (EIO);
|
||||||
goto close_it;
|
return 0;
|
||||||
}
|
}
|
||||||
if (rp->ReparseTag == IO_REPARSE_TAG_SYMLINK)
|
if (rp->ReparseTag == IO_REPARSE_TAG_SYMLINK)
|
||||||
{
|
{
|
||||||
if (rp->SymbolicLinkReparseBuffer.SubstituteNameLength
|
sys_wcstombs (srcbuf, SYMLINK_MAX + 1,
|
||||||
> 2 * (CYG_MAX_PATH + 6))
|
|
||||||
{
|
|
||||||
debug_printf ("Symlink name too long");
|
|
||||||
set_error (EIO);
|
|
||||||
goto close_it;
|
|
||||||
}
|
|
||||||
sys_wcstombs (srcbuf, CYG_MAX_PATH,
|
|
||||||
(WCHAR *)((char *)rp->SymbolicLinkReparseBuffer.PathBuffer
|
(WCHAR *)((char *)rp->SymbolicLinkReparseBuffer.PathBuffer
|
||||||
+ rp->SymbolicLinkReparseBuffer.SubstituteNameOffset),
|
+ rp->SymbolicLinkReparseBuffer.SubstituteNameOffset),
|
||||||
rp->SymbolicLinkReparseBuffer.SubstituteNameLength / 2);
|
rp->SymbolicLinkReparseBuffer.SubstituteNameLength / 2);
|
||||||
@ -3234,26 +3288,16 @@ symlink_info::check_reparse_point (const char *path, HANDLE h)
|
|||||||
if (rp->SymbolicLinkReparseBuffer.PrintNameLength == 0)
|
if (rp->SymbolicLinkReparseBuffer.PrintNameLength == 0)
|
||||||
{
|
{
|
||||||
/* Likely a volume mount point. Not treated as symlink. */
|
/* Likely a volume mount point. Not treated as symlink. */
|
||||||
goto close_it;
|
return 0;
|
||||||
}
|
}
|
||||||
if (rp->MountPointReparseBuffer.SubstituteNameLength
|
sys_wcstombs (srcbuf, SYMLINK_MAX + 1,
|
||||||
> 2 * (CYG_MAX_PATH + 6))
|
|
||||||
{
|
|
||||||
debug_printf ("Symlink name too long");
|
|
||||||
set_error (EIO);
|
|
||||||
goto close_it;
|
|
||||||
}
|
|
||||||
sys_wcstombs (srcbuf, CYG_MAX_PATH,
|
|
||||||
(WCHAR *)((char *)rp->MountPointReparseBuffer.PathBuffer
|
(WCHAR *)((char *)rp->MountPointReparseBuffer.PathBuffer
|
||||||
+ rp->MountPointReparseBuffer.SubstituteNameOffset),
|
+ rp->MountPointReparseBuffer.SubstituteNameOffset),
|
||||||
rp->MountPointReparseBuffer.SubstituteNameLength / 2);
|
rp->MountPointReparseBuffer.SubstituteNameLength / 2);
|
||||||
pflags = PATH_SYMLINK | PATH_REP;
|
pflags = PATH_SYMLINK | PATH_REP;
|
||||||
fileattr &= ~FILE_ATTRIBUTE_DIRECTORY;
|
fileattr &= ~FILE_ATTRIBUTE_DIRECTORY;
|
||||||
}
|
}
|
||||||
res = posixify (srcbuf);
|
return posixify (srcbuf);
|
||||||
close_it:
|
|
||||||
CloseHandle (h);
|
|
||||||
return res;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int
|
int
|
||||||
@ -3297,7 +3341,7 @@ symlink_info::posixify (char *srcbuf)
|
|||||||
slashify (srcbuf, contents, 0);
|
slashify (srcbuf, contents, 0);
|
||||||
else /* Paths starting with \ are current drive relative. */
|
else /* Paths starting with \ are current drive relative. */
|
||||||
{
|
{
|
||||||
char cvtbuf[CYG_MAX_PATH + 6];
|
char cvtbuf[SYMLINK_MAX + 1];
|
||||||
|
|
||||||
strncpy (cvtbuf, cygheap->cwd.win32, 2);
|
strncpy (cvtbuf, cygheap->cwd.win32, 2);
|
||||||
strcpy (cvtbuf + 2, srcbuf);
|
strcpy (cvtbuf + 2, srcbuf);
|
||||||
@ -3524,6 +3568,7 @@ symlink_info::check (char *path, const suffix_info *suffixes, unsigned opt)
|
|||||||
{
|
{
|
||||||
FILE_BASIC_INFORMATION fbi;
|
FILE_BASIC_INFORMATION fbi;
|
||||||
NTSTATUS status;
|
NTSTATUS status;
|
||||||
|
IO_STATUS_BLOCK io;
|
||||||
|
|
||||||
error = 0;
|
error = 0;
|
||||||
get_nt_native_path (suffix.path, upath);
|
get_nt_native_path (suffix.path, upath);
|
||||||
@ -3561,7 +3606,6 @@ symlink_info::check (char *path, const suffix_info *suffixes, unsigned opt)
|
|||||||
UNICODE_STRING dirname, basename;
|
UNICODE_STRING dirname, basename;
|
||||||
OBJECT_ATTRIBUTES dattr;
|
OBJECT_ATTRIBUTES dattr;
|
||||||
HANDLE dir;
|
HANDLE dir;
|
||||||
IO_STATUS_BLOCK io;
|
|
||||||
FILE_DIRECTORY_INFORMATION fdi;
|
FILE_DIRECTORY_INFORMATION fdi;
|
||||||
|
|
||||||
RtlSplitUnicodePath (&upath, &dirname, &basename);
|
RtlSplitUnicodePath (&upath, &dirname, &basename);
|
||||||
@ -3624,9 +3668,10 @@ symlink_info::check (char *path, const suffix_info *suffixes, unsigned opt)
|
|||||||
== FILE_ATTRIBUTE_DIRECTORY)
|
== FILE_ATTRIBUTE_DIRECTORY)
|
||||||
goto file_not_symlink;
|
goto file_not_symlink;
|
||||||
|
|
||||||
/* Reparse points are potentially symlinks. */
|
/* Windows shortcuts are potentially treated as symlinks. */
|
||||||
if (fileattr & FILE_ATTRIBUTE_REPARSE_POINT)
|
/* Valid Cygwin & U/WIN shortcuts are R/O. */
|
||||||
sym_check = 3;
|
if ((fileattr & FILE_ATTRIBUTE_READONLY) && suffix.lnk_match ())
|
||||||
|
sym_check = 1;
|
||||||
|
|
||||||
/* This is the old Cygwin method creating symlinks: */
|
/* This is the old Cygwin method creating symlinks: */
|
||||||
/* A symlink will have the `system' file attribute. */
|
/* A symlink will have the `system' file attribute. */
|
||||||
@ -3634,29 +3679,29 @@ symlink_info::check (char *path, const suffix_info *suffixes, unsigned opt)
|
|||||||
else if (fileattr & FILE_ATTRIBUTE_SYSTEM)
|
else if (fileattr & FILE_ATTRIBUTE_SYSTEM)
|
||||||
sym_check = 2;
|
sym_check = 2;
|
||||||
|
|
||||||
/* Windows shortcuts are potentially treated as symlinks. */
|
/* Reparse points are potentially symlinks. */
|
||||||
/* Valid Cygwin & U/WIN shortcuts are R/O. */
|
else if (fileattr & FILE_ATTRIBUTE_REPARSE_POINT)
|
||||||
else if ((fileattr & FILE_ATTRIBUTE_READONLY) && suffix.lnk_match ())
|
sym_check = 3;
|
||||||
sym_check = 1;
|
|
||||||
|
|
||||||
if (!sym_check)
|
if (!sym_check)
|
||||||
goto file_not_symlink;
|
goto file_not_symlink;
|
||||||
|
|
||||||
/* Open the file. */
|
|
||||||
|
|
||||||
h = CreateFile (suffix.path, GENERIC_READ, FILE_SHARE_READ,
|
|
||||||
&sec_none_nih, OPEN_EXISTING,
|
|
||||||
sym_check == 3 ? FILE_FLAG_OPEN_REPARSE_POINT
|
|
||||||
| FILE_FLAG_BACKUP_SEMANTICS
|
|
||||||
: FILE_ATTRIBUTE_NORMAL, 0);
|
|
||||||
res = -1;
|
res = -1;
|
||||||
if (h == INVALID_HANDLE_VALUE)
|
|
||||||
|
/* Open the file. */
|
||||||
|
status = NtOpenFile (&h, FILE_GENERIC_READ, &attr, &io,
|
||||||
|
FILE_SHARE_VALID_FLAGS,
|
||||||
|
FILE_SYNCHRONOUS_IO_NONALERT
|
||||||
|
| FILE_OPEN_FOR_BACKUP_INTENT
|
||||||
|
| (sym_check == 3 ? FILE_OPEN_REPARSE_POINT : 0));
|
||||||
|
if (!NT_SUCCESS (status))
|
||||||
goto file_not_symlink;
|
goto file_not_symlink;
|
||||||
|
|
||||||
switch (sym_check)
|
switch (sym_check)
|
||||||
{
|
{
|
||||||
case 1:
|
case 1:
|
||||||
res = check_shortcut (suffix.path, h);
|
res = check_shortcut (h);
|
||||||
|
NtClose (h);
|
||||||
if (!res)
|
if (!res)
|
||||||
/* check more below */;
|
/* check more below */;
|
||||||
else if (contents[0] == ':' && contents[1] == '\\' && parse_device (contents))
|
else if (contents[0] == ':' && contents[1] == '\\' && parse_device (contents))
|
||||||
@ -3671,12 +3716,14 @@ symlink_info::check (char *path, const suffix_info *suffixes, unsigned opt)
|
|||||||
fileattr = INVALID_FILE_ATTRIBUTES;
|
fileattr = INVALID_FILE_ATTRIBUTES;
|
||||||
continue; /* in case we're going to tack *another* .lnk on this filename. */
|
continue; /* in case we're going to tack *another* .lnk on this filename. */
|
||||||
case 2:
|
case 2:
|
||||||
res = check_sysfile (suffix.path, h);
|
res = check_sysfile (h);
|
||||||
|
NtClose (h);
|
||||||
if (!res)
|
if (!res)
|
||||||
goto file_not_symlink;
|
goto file_not_symlink;
|
||||||
break;
|
break;
|
||||||
case 3:
|
case 3:
|
||||||
res = check_reparse_point (suffix.path, h);
|
res = check_reparse_point (h);
|
||||||
|
NtClose (h);
|
||||||
if (!res)
|
if (!res)
|
||||||
goto file_not_symlink;
|
goto file_not_symlink;
|
||||||
break;
|
break;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user