* path.cc (struct symlink_info): Add bool argument to declaration of

check_reparse_point.
	(symlink_info::check_reparse_point): Add bool argument to indicate
	remote drive.  Handle STATUS_PENDING.  Don't evaluate junctions on
	remote drives.  Fix comments.
	(symlink_info::check): Drop check for is_remote_drive and associated
	comment here.  Add fs.is_remote_drive as second parameter to
	check_reparse_point call.
This commit is contained in:
Corinna Vinschen 2011-12-24 13:11:34 +00:00
parent 9ffe887f6c
commit 855ea3fddd
2 changed files with 40 additions and 17 deletions

View File

@ -1,3 +1,14 @@
2011-12-24 Corinna Vinschen <vinschen@redhat.com>
* path.cc (struct symlink_info): Add bool argument to declaration of
check_reparse_point.
(symlink_info::check_reparse_point): Add bool argument to indicate
remote drive. Handle STATUS_PENDING. Don't evaluate junctions on
remote drives. Fix comments.
(symlink_info::check): Drop check for is_remote_drive and associated
comment here. Add fs.is_remote_drive as second parameter to
check_reparse_point call.
2011-12-23 Corinna Vinschen <vinschen@redhat.com> 2011-12-23 Corinna Vinschen <vinschen@redhat.com>
* pinfo.cc (pinfo_basic::pinfo_basic): Fix formatting. Set uid and gid * pinfo.cc (pinfo_basic::pinfo_basic): Fix formatting. Set uid and gid

View File

@ -102,7 +102,7 @@ struct symlink_info
bool parse_device (const char *); bool parse_device (const char *);
int check_sysfile (HANDLE h); int check_sysfile (HANDLE h);
int check_shortcut (HANDLE h); int check_shortcut (HANDLE h);
int check_reparse_point (HANDLE h); int check_reparse_point (HANDLE h, bool remote);
int check_nfs_symlink (HANDLE h); int check_nfs_symlink (HANDLE h);
int posixify (char *srcbuf); int posixify (char *srcbuf);
bool set_error (int); bool set_error (int);
@ -1958,7 +1958,7 @@ symlink_info::check_sysfile (HANDLE h)
} }
int int
symlink_info::check_reparse_point (HANDLE h) symlink_info::check_reparse_point (HANDLE h, bool remote)
{ {
tmp_pathbuf tp; tmp_pathbuf tp;
NTSTATUS status; NTSTATUS status;
@ -1967,9 +1967,20 @@ symlink_info::check_reparse_point (HANDLE h)
UNICODE_STRING subst; UNICODE_STRING subst;
char srcbuf[SYMLINK_MAX + 7]; char srcbuf[SYMLINK_MAX + 7];
status = NtFsControlFile (h, NULL, NULL, NULL, &io, FSCTL_GET_REPARSE_POINT, /* On remote drives or under heavy load, NtFsControlFile can return with
NULL, 0, (LPVOID) rp, STATUS_PENDING. If so, instead of creating an event object, just set
io.Status to an invalid value and perform a minimal wait until io.Status
changed. */
memset (&io, 0xff, sizeof io);
status = NtFsControlFile (h, NULL, NULL, NULL, &io,
FSCTL_GET_REPARSE_POINT, NULL, 0, (LPVOID) rp,
MAXIMUM_REPARSE_DATA_BUFFER_SIZE); MAXIMUM_REPARSE_DATA_BUFFER_SIZE);
if (status == STATUS_PENDING)
{
while (io.Status == (NTSTATUS) 0xffffffff)
Sleep (1L);
status = io.Status;
}
if (!NT_SUCCESS (status)) if (!NT_SUCCESS (status))
{ {
debug_printf ("NtFsControlFile(FSCTL_GET_REPARSE_POINT) failed, %p", debug_printf ("NtFsControlFile(FSCTL_GET_REPARSE_POINT) failed, %p",
@ -1978,12 +1989,21 @@ symlink_info::check_reparse_point (HANDLE h)
return 0; return 0;
} }
if (rp->ReparseTag == IO_REPARSE_TAG_SYMLINK) if (rp->ReparseTag == IO_REPARSE_TAG_SYMLINK)
/* Windows evaluates native symlink literally. If a remote symlink points
to, say, C:\foo, it will be handled as if the target is the local file
C:\foo. That comes in handy since that's how symlinks are treated under
POSIX as well. */
RtlInitCountedUnicodeString (&subst, RtlInitCountedUnicodeString (&subst,
(WCHAR *)((char *)rp->SymbolicLinkReparseBuffer.PathBuffer (WCHAR *)((char *)rp->SymbolicLinkReparseBuffer.PathBuffer
+ rp->SymbolicLinkReparseBuffer.SubstituteNameOffset), + rp->SymbolicLinkReparseBuffer.SubstituteNameOffset),
rp->SymbolicLinkReparseBuffer.SubstituteNameLength); rp->SymbolicLinkReparseBuffer.SubstituteNameLength);
else if (rp->ReparseTag == IO_REPARSE_TAG_MOUNT_POINT) else if (!remote && rp->ReparseTag == IO_REPARSE_TAG_MOUNT_POINT)
{ {
/* Don't handle junctions on remote filesystems as symlinks. This type
of reparse point is handled transparently by the OS so that the
target of the junction is the remote directory it is supposed to
point to. If we handle it as symlink, it will be mistreated as
pointing to a dir on the local system. */
RtlInitCountedUnicodeString (&subst, RtlInitCountedUnicodeString (&subst,
(WCHAR *)((char *)rp->MountPointReparseBuffer.PathBuffer (WCHAR *)((char *)rp->MountPointReparseBuffer.PathBuffer
+ rp->MountPointReparseBuffer.SubstituteNameOffset), + rp->MountPointReparseBuffer.SubstituteNameOffset),
@ -1998,10 +2018,9 @@ symlink_info::check_reparse_point (HANDLE h)
} }
else else
{ {
/* Maybe it's a reparse point, but it's certainly not one we /* Maybe it's a reparse point, but it's certainly not one we recognize.
recognize. Drop the REPARSE file attribute so we don't even Drop REPARSE attribute so we don't try to use the flag accidentally.
try to use the flag for some special handling. It's just some It's just some arbitrary file or directory for us. */
arbitrary file or directory for us. */
fileattr &= ~FILE_ATTRIBUTE_REPARSE_POINT; fileattr &= ~FILE_ATTRIBUTE_REPARSE_POINT;
return 0; return 0;
} }
@ -2597,14 +2616,7 @@ restart:
with SYSTEM and HIDDEN flags set. */ with SYSTEM and HIDDEN flags set. */
if ((fileattr & FILE_ATTRIBUTE_REPARSE_POINT)) if ((fileattr & FILE_ATTRIBUTE_REPARSE_POINT))
{ {
/* Don't check reparse points on remote filesystems. A reparse point res = check_reparse_point (h, fs.is_remote_drive ());
pointing to another file on the remote system will be mistreated
as pointing to a local file on the local system. This breaks the
way reparse points are transparently handled on remote systems. */
if (fs.is_remote_drive())
res = 0;
else
res = check_reparse_point (h);
if (res == -1) if (res == -1)
{ {
/* Volume mount point. The filesystem information for the top /* Volume mount point. The filesystem information for the top