mirror of
git://sourceware.org/git/newlib-cygwin.git
synced 2025-02-21 00:07:36 +08:00
* fhandler_disk_file.cc (path_conv::isgood_inode): Move to be defined
earlier. (get_ino_by_handle): Take additional path_conv argument, accommodate throughout. Only use FileId if isgood_inode check is true. (fhandler_base::open_fs): Simplify setting ino due to above change. (readdir_get_ino): Make sure to return always a non-zero inode number. (fhandler_disk_file::readdir): Always open file in dir with FILE_OPEN_REPARSE_POINT so as not to open wrong file. Drop call to isgood_inode here. * path.cc (symlink_info::check): Call fs.update in case we're fetching file information from call to NtQueryDirectoryFile.
This commit is contained in:
parent
fc69f1aed5
commit
752c477b42
@ -1,3 +1,17 @@
|
|||||||
|
2009-03-12 Corinna Vinschen <corinna@vinschen.de>
|
||||||
|
|
||||||
|
* fhandler_disk_file.cc (path_conv::isgood_inode): Move to be defined
|
||||||
|
earlier.
|
||||||
|
(get_ino_by_handle): Take additional path_conv argument, accommodate
|
||||||
|
throughout. Only use FileId if isgood_inode check is true.
|
||||||
|
(fhandler_base::open_fs): Simplify setting ino due to above change.
|
||||||
|
(readdir_get_ino): Make sure to return always a non-zero inode number.
|
||||||
|
(fhandler_disk_file::readdir): Always open file in dir with
|
||||||
|
FILE_OPEN_REPARSE_POINT so as not to open wrong file.
|
||||||
|
Drop call to isgood_inode here.
|
||||||
|
* path.cc (symlink_info::check): Call fs.update in case we're fetching
|
||||||
|
file information from call to NtQueryDirectoryFile.
|
||||||
|
|
||||||
2009-03-12 Corinna Vinschen <corinna@vinschen.de>
|
2009-03-12 Corinna Vinschen <corinna@vinschen.de>
|
||||||
|
|
||||||
* flock.cc (fhandler_disk_file::lock): Don't test file open mode in
|
* flock.cc (fhandler_disk_file::lock): Don't test file open mode in
|
||||||
|
@ -139,6 +139,17 @@ public:
|
|||||||
void rewind () { memset (found, 0, sizeof found); }
|
void rewind () { memset (found, 0, sizeof found); }
|
||||||
};
|
};
|
||||||
|
|
||||||
|
inline bool
|
||||||
|
path_conv::isgood_inode (__ino64_t ino) const
|
||||||
|
{
|
||||||
|
/* We can't trust remote inode numbers of only 32 bit. That means,
|
||||||
|
all remote inode numbers when running under NT4, as well as remote NT4
|
||||||
|
NTFS, as well as shares of Samba version < 3.0.
|
||||||
|
The known exception are SFU NFS shares, which return the valid 32 bit
|
||||||
|
inode number from the remote file system unchanged. */
|
||||||
|
return hasgood_inode () && (ino > UINT32_MAX || !isremote () || fs_is_nfs ());
|
||||||
|
}
|
||||||
|
|
||||||
static inline bool
|
static inline bool
|
||||||
is_volume_mountpoint (POBJECT_ATTRIBUTES attr)
|
is_volume_mountpoint (POBJECT_ATTRIBUTES attr)
|
||||||
{
|
{
|
||||||
@ -165,13 +176,14 @@ is_volume_mountpoint (POBJECT_ATTRIBUTES attr)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static inline __ino64_t
|
static inline __ino64_t
|
||||||
get_ino_by_handle (HANDLE hdl)
|
get_ino_by_handle (path_conv &pc, HANDLE hdl)
|
||||||
{
|
{
|
||||||
IO_STATUS_BLOCK io;
|
IO_STATUS_BLOCK io;
|
||||||
FILE_INTERNAL_INFORMATION fai;
|
FILE_INTERNAL_INFORMATION fai;
|
||||||
|
|
||||||
if (NT_SUCCESS (NtQueryInformationFile (hdl, &io, &fai, sizeof fai,
|
if (NT_SUCCESS (NtQueryInformationFile (hdl, &io, &fai, sizeof fai,
|
||||||
FileInternalInformation)))
|
FileInternalInformation))
|
||||||
|
&& pc.isgood_inode (fai.FileId.QuadPart))
|
||||||
return fai.FileId.QuadPart;
|
return fai.FileId.QuadPart;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@ -254,17 +266,6 @@ path_conv::ndisk_links (DWORD nNumberOfLinks)
|
|||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
inline bool
|
|
||||||
path_conv::isgood_inode (__ino64_t ino) const
|
|
||||||
{
|
|
||||||
/* We can't trust remote inode numbers of only 32 bit. That means,
|
|
||||||
all remote inode numbers when running under NT4, as well as remote NT4
|
|
||||||
NTFS, as well as shares of Samba version < 3.0.
|
|
||||||
The known exception are SFU NFS shares, which return the valid 32 bit
|
|
||||||
inode number from the remote file system unchanged. */
|
|
||||||
return hasgood_inode () && (ino > UINT32_MAX || !isremote () || fs_is_nfs ());
|
|
||||||
}
|
|
||||||
|
|
||||||
/* For files on NFS shares, we request an EA of type NfsV3Attributes.
|
/* For files on NFS shares, we request an EA of type NfsV3Attributes.
|
||||||
This returns the content of a struct fattr3 as defined in RFC 1813.
|
This returns the content of a struct fattr3 as defined in RFC 1813.
|
||||||
The content is the NFS equivalent of struct stat. so there's not much
|
The content is the NFS equivalent of struct stat. so there's not much
|
||||||
@ -1331,8 +1332,7 @@ fhandler_base::open_fs (int flags, mode_t mode)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pc.hasgood_inode ())
|
ino = get_ino_by_handle (pc, get_handle ());
|
||||||
ino = get_ino_by_handle (get_handle ());
|
|
||||||
/* A unique ID is necessary to recognize fhandler entries which are
|
/* A unique ID is necessary to recognize fhandler entries which are
|
||||||
duplicated by dup(2) or fork(2). */
|
duplicated by dup(2) or fork(2). */
|
||||||
AllocateLocallyUniqueId ((PLUID) &unique_id);
|
AllocateLocallyUniqueId ((PLUID) &unique_id);
|
||||||
@ -1658,7 +1658,9 @@ readdir_get_ino (const char *path, bool dot_dot)
|
|||||||
| (pc.is_rep_symlink ()
|
| (pc.is_rep_symlink ()
|
||||||
? FILE_OPEN_REPARSE_POINT : 0))))
|
? FILE_OPEN_REPARSE_POINT : 0))))
|
||||||
{
|
{
|
||||||
ino = get_ino_by_handle (hdl);
|
ino = get_ino_by_handle (pc, hdl);
|
||||||
|
if (!ino)
|
||||||
|
ino = hash_path_name (0, pc.get_nt_native_path ());
|
||||||
NtClose (hdl);
|
NtClose (hdl);
|
||||||
}
|
}
|
||||||
return ino;
|
return ino;
|
||||||
@ -1710,7 +1712,7 @@ fhandler_disk_file::readdir_helper (DIR *dir, dirent *de, DWORD w32_err,
|
|||||||
FILE_SHARE_VALID_FLAGS,
|
FILE_SHARE_VALID_FLAGS,
|
||||||
FILE_OPEN_FOR_BACKUP_INTENT))))
|
FILE_OPEN_FOR_BACKUP_INTENT))))
|
||||||
{
|
{
|
||||||
de->d_ino = get_ino_by_handle (reph);
|
de->d_ino = get_ino_by_handle (pc, reph);
|
||||||
NtClose (reph);
|
NtClose (reph);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1891,13 +1893,13 @@ go_ahead:
|
|||||||
|
|
||||||
if (dir->__d_position == 0 && FileNameLength == 2
|
if (dir->__d_position == 0 && FileNameLength == 2
|
||||||
&& FileName[0] == '.')
|
&& FileName[0] == '.')
|
||||||
de->d_ino = get_ino_by_handle (get_handle ());
|
de->d_ino = get_ino_by_handle (pc, get_handle ());
|
||||||
else if (dir->__d_position == 1 && FileNameLength == 4
|
else if (dir->__d_position == 1 && FileNameLength == 4
|
||||||
&& FileName[0] == L'.' && FileName[1] == L'.')
|
&& FileName[0] == L'.' && FileName[1] == L'.')
|
||||||
if (!(dir->__flags & dirent_isroot))
|
if (!(dir->__flags & dirent_isroot))
|
||||||
de->d_ino = readdir_get_ino (get_name (), true);
|
de->d_ino = readdir_get_ino (get_name (), true);
|
||||||
else
|
else
|
||||||
de->d_ino = get_ino_by_handle (get_handle ());
|
de->d_ino = get_ino_by_handle (pc, get_handle ());
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
HANDLE hdl;
|
HANDLE hdl;
|
||||||
@ -1907,18 +1909,16 @@ go_ahead:
|
|||||||
get_handle (), NULL);
|
get_handle (), NULL);
|
||||||
if (NT_SUCCESS (NtOpenFile (&hdl, READ_CONTROL, &attr, &io,
|
if (NT_SUCCESS (NtOpenFile (&hdl, READ_CONTROL, &attr, &io,
|
||||||
FILE_SHARE_VALID_FLAGS,
|
FILE_SHARE_VALID_FLAGS,
|
||||||
FILE_OPEN_FOR_BACKUP_INTENT)))
|
FILE_OPEN_FOR_BACKUP_INTENT
|
||||||
|
| FILE_OPEN_REPARSE_POINT)))
|
||||||
{
|
{
|
||||||
de->d_ino = get_ino_by_handle (hdl);
|
de->d_ino = get_ino_by_handle (pc, hdl);
|
||||||
NtClose (hdl);
|
NtClose (hdl);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/* Enforce namehash as inode number on untrusted file systems. */
|
/* Untrusted file system. Don't try to fetch inode number again. */
|
||||||
if (!pc.isgood_inode (de->d_ino))
|
if (de->d_ino == 0)
|
||||||
{
|
dir->__flags &= ~dirent_set_d_ino;
|
||||||
dir->__flags &= ~dirent_set_d_ino;
|
|
||||||
de->d_ino = 0;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1928,8 +1928,7 @@ go_ahead:
|
|||||||
else if (!(dir->__flags & dirent_saw_dot))
|
else if (!(dir->__flags & dirent_saw_dot))
|
||||||
{
|
{
|
||||||
strcpy (de->d_name , ".");
|
strcpy (de->d_name , ".");
|
||||||
if (pc.isgood_inode (de->d_ino))
|
de->d_ino = get_ino_by_handle (pc, get_handle ());
|
||||||
de->d_ino = get_ino_by_handle (get_handle ());
|
|
||||||
dir->__d_position++;
|
dir->__d_position++;
|
||||||
dir->__flags |= dirent_saw_dot;
|
dir->__flags |= dirent_saw_dot;
|
||||||
res = 0;
|
res = 0;
|
||||||
@ -1940,7 +1939,7 @@ go_ahead:
|
|||||||
if (!(dir->__flags & dirent_isroot))
|
if (!(dir->__flags & dirent_isroot))
|
||||||
de->d_ino = readdir_get_ino (get_name (), true);
|
de->d_ino = readdir_get_ino (get_name (), true);
|
||||||
else
|
else
|
||||||
de->d_ino = get_ino_by_handle (get_handle ());
|
de->d_ino = get_ino_by_handle (pc, get_handle ());
|
||||||
dir->__d_position++;
|
dir->__d_position++;
|
||||||
dir->__flags |= dirent_saw_dot_dot;
|
dir->__flags |= dirent_saw_dot_dot;
|
||||||
res = 0;
|
res = 0;
|
||||||
|
@ -2191,7 +2191,7 @@ symlink_info::check (char *path, const suffix_info *suffixes, unsigned opt,
|
|||||||
&attr, &io, FILE_SHARE_VALID_FLAGS,
|
&attr, &io, FILE_SHARE_VALID_FLAGS,
|
||||||
FILE_OPEN_REPARSE_POINT
|
FILE_OPEN_REPARSE_POINT
|
||||||
| FILE_OPEN_FOR_BACKUP_INTENT);
|
| FILE_OPEN_FOR_BACKUP_INTENT);
|
||||||
attr.Attributes = ci_flag;
|
attr.Attributes = 0;
|
||||||
if (NT_SUCCESS (status))
|
if (NT_SUCCESS (status))
|
||||||
{
|
{
|
||||||
fs.update (&upath, h);
|
fs.update (&upath, h);
|
||||||
@ -2265,6 +2265,9 @@ symlink_info::check (char *path, const suffix_info *suffixes, unsigned opt,
|
|||||||
&fdi_buf, sizeof fdi_buf,
|
&fdi_buf, sizeof fdi_buf,
|
||||||
FileDirectoryInformation,
|
FileDirectoryInformation,
|
||||||
TRUE, &basename, TRUE);
|
TRUE, &basename, TRUE);
|
||||||
|
/* Take the opportunity to check file system while we're
|
||||||
|
having the handle to the parent dir. */
|
||||||
|
fs.update (&upath, h);
|
||||||
NtClose (dir);
|
NtClose (dir);
|
||||||
if (!NT_SUCCESS (status))
|
if (!NT_SUCCESS (status))
|
||||||
{
|
{
|
||||||
|
Loading…
x
Reference in New Issue
Block a user