- From c5db6fc20df4954573244f28af58195c83c6989d Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Fri, 28 Mar 2025 15:37:13 +0100
- Subject: [PATCH 1/3] daemon: |readdir_copy_full_dir_info()| should set
- |EaSize| to a non-zero value
- |readdir_copy_full_dir_info()| should set |EaSize| to a non-zero value
- to indicate that we *MAY* have EAs.
- The real size used by our EAs does not matter, as there is a possible
- race-condition anyway and applications have to deal with that since ages.
- We now return 64k, which is the maximum EA size on NTFS.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/readdir.c | 25 ++++++++++++++++++-------
- 1 file changed, 18 insertions(+), 7 deletions(-)
- diff --git a/daemon/readdir.c b/daemon/readdir.c
- index 9711dba..1561ea2 100644
- --- a/daemon/readdir.c
- +++ b/daemon/readdir.c
- @@ -401,13 +401,24 @@ static void readdir_copy_full_dir_info(
- IN PFILE_DIR_INFO_UNION info)
- {
- readdir_copy_dir_info(entry, superblock, info);
- - /* for files with the FILE_ATTRIBUTE_REPARSE_POINT attribute,
- - * EaSize is used instead to specify its reparse tag. this makes
- - * the 'dir' command to show files as <SYMLINK>, and triggers a
- - * FSCTL_GET_REPARSE_POINT to query the symlink target
- - */
- - info->fifdi.EaSize = entry->attr_info.type == NF4LNK ?
- - IO_REPARSE_TAG_SYMLINK : 0;
- + if (entry->attr_info.type == NF4LNK) {
- + /*
- + * For files with the |FILE_ATTRIBUTE_REPARSE_POINT|
- + * attribute, |EaSize| is used instead to specify its
- + * reparse tag. This makes the cmd.exe 'dir' command to
- + * show files as <SYMLINK>/<SYMLINKD>, and triggers a
- + * |FSCTL_GET_REPARSE_POINT| to query the symlink target
- + */
- + info->fifdi.EaSize = IO_REPARSE_TAG_SYMLINK;
- + }
- + else {
- + /*
- + * Always return the maximum EA size (64k), so
- + + applications will look for EAs (a value of |0| would
- + * mean "no EAs here")
- + */
- + info->fifdi.EaSize = (64*1024)-1;
- + }
- }
- static void readdir_copy_both_dir_info(
- --
- 2.45.1
- From 014b31742b29107a4a87b4ffdd0a305f5c47fefa Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Fri, 28 Mar 2025 16:48:43 +0100
- Subject: [PATCH 2/3] include,daemon: ReactOS: Use WDK/ntifs.h names for
- directory info types
- ReactOS: Use WDK/ntifs.h names for directory info types.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/daemon_debug.c | 2 +-
- daemon/daemon_debug.h | 3 ++-
- daemon/readdir.c | 16 ++++++++--------
- include/from_kernel.h | 32 ++++++++++++++++++++++++--------
- 4 files changed, 35 insertions(+), 18 deletions(-)
- diff --git a/daemon/daemon_debug.c b/daemon/daemon_debug.c
- index 083fe2f..50856e9 100644
- --- a/daemon/daemon_debug.c
- +++ b/daemon/daemon_debug.c
- @@ -369,7 +369,7 @@ void print_share_mode(int level, DWORD mode)
- fprintf(dlog_file, "\n");
- }
- -void print_file_id_both_dir_info(int level, const FILE_ID_BOTH_DIR_INFO *pboth_dir_info)
- +void print_file_id_both_dir_info(int level, const FILE_ID_BOTH_DIR_INFORMATION *pboth_dir_info)
- {
- /* printf %zd is for |size_t| */
- diff --git a/daemon/daemon_debug.h b/daemon/daemon_debug.h
- index 4b0a336..a4408f0 100644
- --- a/daemon/daemon_debug.h
- +++ b/daemon/daemon_debug.h
- @@ -137,7 +137,8 @@ void print_create_attributes(int level, DWORD create_opts);
- void print_disposition(int level, DWORD disposition);
- void print_access_mask(int level, DWORD access_mask);
- void print_share_mode(int level, DWORD mode);
- -void print_file_id_both_dir_info(int level, const FILE_ID_BOTH_DIR_INFO *p);
- +typedef struct _FILE_ID_BOTH_DIR_INFORMATION FILE_ID_BOTH_DIR_INFORMATION, *PFILE_ID_BOTH_DIR_INFORMATION;
- +void print_file_id_both_dir_info(int level, const FILE_ID_BOTH_DIR_INFORMATION *p);
- void print_sid(const char *label, PSID sid);
- typedef enum _nfs41_opcodes nfs41_opcodes;
- const char* opcode2string(nfs41_opcodes opcode);
- diff --git a/daemon/readdir.c b/daemon/readdir.c
- index 1561ea2..385534f 100644
- --- a/daemon/readdir.c
- +++ b/daemon/readdir.c
- @@ -251,11 +251,11 @@ int main(int ac, char *av[])
- typedef union _FILE_DIR_INFO_UNION {
- ULONG NextEntryOffset;
- FILE_NAMES_INFORMATION fni;
- - FILE_DIRECTORY_INFO fdi;
- - FILE_FULL_DIR_INFO ffdi;
- - FILE_ID_FULL_DIR_INFO fifdi;
- + FILE_DIRECTORY_INFORMATION fdi;
- + FILE_FULL_DIR_INFORMATION ffdi;
- + FILE_ID_FULL_DIR_INFORMATION fifdi;
- FILE_BOTH_DIR_INFORMATION fbdi;
- - FILE_ID_BOTH_DIR_INFO fibdi;
- + FILE_ID_BOTH_DIR_INFORMATION fibdi;
- } FILE_DIR_INFO_UNION, *PFILE_DIR_INFO_UNION;
- @@ -299,16 +299,16 @@ static uint32_t readdir_size_for_entry(
- switch (query_class)
- {
- case FileDirectoryInformation:
- - needed += FIELD_OFFSET(FILE_DIRECTORY_INFO, FileName);
- + needed += FIELD_OFFSET(FILE_DIRECTORY_INFORMATION, FileName);
- break;
- case FileIdFullDirectoryInformation:
- - needed += FIELD_OFFSET(FILE_ID_FULL_DIR_INFO, FileName);
- + needed += FIELD_OFFSET(FILE_ID_FULL_DIR_INFORMATION, FileName);
- break;
- case FileFullDirectoryInformation:
- - needed += FIELD_OFFSET(FILE_FULL_DIR_INFO, FileName);
- + needed += FIELD_OFFSET(FILE_FULL_DIR_INFORMATION, FileName);
- break;
- case FileIdBothDirectoryInformation:
- - needed += FIELD_OFFSET(FILE_ID_BOTH_DIR_INFO, FileName);
- + needed += FIELD_OFFSET(FILE_ID_BOTH_DIR_INFORMATION, FileName);
- break;
- case FileBothDirectoryInformation:
- needed += FIELD_OFFSET(FILE_BOTH_DIR_INFORMATION, FileName);
- diff --git a/include/from_kernel.h b/include/from_kernel.h
- index 753f31c..66d0cec 100644
- --- a/include/from_kernel.h
- +++ b/include/from_kernel.h
- @@ -153,7 +153,7 @@ typedef struct _FILE_NAMES_INFORMATION {
- WCHAR FileName[1];
- } FILE_NAMES_INFORMATION, *PFILE_NAMES_INFORMATION;
- -typedef struct _FILE_DIRECTORY_INFO {
- +typedef struct _FILE_DIRECTORY_INFORMATION {
- ULONG NextEntryOffset;
- ULONG FileIndex;
- LARGE_INTEGER CreationTime;
- @@ -165,7 +165,7 @@ typedef struct _FILE_DIRECTORY_INFO {
- ULONG FileAttributes;
- ULONG FileNameLength;
- WCHAR FileName[1];
- -} FILE_DIRECTORY_INFO, *PFILE_DIRECTORY_INFO;
- +} FILE_DIRECTORY_INFORMATION, *PFILE_DIRECTORY_INFORMATION;
- typedef struct _FILE_BOTH_DIR_INFORMATION {
- ULONG NextEntryOffset;
- @@ -184,8 +184,7 @@ typedef struct _FILE_BOTH_DIR_INFORMATION {
- WCHAR FileName[1];
- } FILE_BOTH_DIR_INFORMATION, *PFILE_BOTH_DIR_INFORMATION;
- -#ifdef FIXME_OLD_DDK
- -typedef struct _FILE_FULL_DIR_INFO {
- +typedef struct _FILE_FULL_DIR_INFORMATION {
- ULONG NextEntryOffset;
- ULONG FileIndex;
- LARGE_INTEGER CreationTime;
- @@ -198,10 +197,9 @@ typedef struct _FILE_FULL_DIR_INFO {
- ULONG FileNameLength;
- ULONG EaSize;
- WCHAR FileName[1];
- -} FILE_FULL_DIR_INFO, *PFILE_FULL_DIR_INFO;
- -#endif /* FIXME_OLD_DDK */
- +} FILE_FULL_DIR_INFORMATION, *PFILE_FULL_DIR_INFORMATION;
- -typedef struct _FILE_ID_FULL_DIR_INFO {
- +typedef struct _FILE_ID_FULL_DIR_INFORMATION {
- ULONG NextEntryOffset;
- ULONG FileIndex;
- LARGE_INTEGER CreationTime;
- @@ -215,7 +213,25 @@ typedef struct _FILE_ID_FULL_DIR_INFO {
- ULONG EaSize;
- LARGE_INTEGER FileId;
- WCHAR FileName[1];
- -} FILE_ID_FULL_DIR_INFO, *PFILE_ID_FULL_DIR_INFO;
- +} FILE_ID_FULL_DIR_INFORMATION, *PFILE_ID_FULL_DIR_INFORMATION;
- +
- +typedef struct _FILE_ID_BOTH_DIR_INFORMATION {
- + ULONG NextEntryOffset;
- + ULONG FileIndex;
- + LARGE_INTEGER CreationTime;
- + LARGE_INTEGER LastAccessTime;
- + LARGE_INTEGER LastWriteTime;
- + LARGE_INTEGER ChangeTime;
- + LARGE_INTEGER EndOfFile;
- + LARGE_INTEGER AllocationSize;
- + ULONG FileAttributes;
- + ULONG FileNameLength;
- + ULONG EaSize;
- + CCHAR ShortNameLength;
- + WCHAR ShortName[12];
- + LARGE_INTEGER FileId;
- + WCHAR FileName[1];
- +} FILE_ID_BOTH_DIR_INFORMATION, *PFILE_ID_BOTH_DIR_INFORMATION;
- typedef struct _FILE_LINK_INFORMATION {
- BOOLEAN ReplaceIfExists;
- --
- 2.45.1
- From 5fde0e7180ac0fc44c618eacd278a74a674effd0 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Fri, 28 Mar 2025 18:16:10 +0100
- Subject: [PATCH 3/3] daemon,include,sys: Implement
- |FileIdExtdDirectoryInformation|+|FileIdExtdBothDirectoryInformation|
- Implement |FileIdExtdDirectoryInformation| and
- |FileIdExtdBothDirectoryInformation| dir query classes.
- Needed for (better) WSL support.
- Reported-by: Cedric Blancher <cedric.blancher@gmail.com>
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/fileinfoutil.c | 11 +++++++++
- daemon/getattr.c | 6 +----
- daemon/readdir.c | 54 ++++++++++++++++++++++++++++++++++++++-----
- daemon/util.h | 4 ++++
- include/from_kernel.h | 36 +++++++++++++++++++++++++++++
- sys/nfs41sys_dir.c | 2 ++
- 6 files changed, 102 insertions(+), 11 deletions(-)
- diff --git a/daemon/fileinfoutil.c b/daemon/fileinfoutil.c
- index be684fc..85a36ae 100644
- --- a/daemon/fileinfoutil.c
- +++ b/daemon/fileinfoutil.c
- @@ -33,6 +33,17 @@
- #include "nfs41_ops.h"
- #include "nfs41_driver.h" /* for |FILE_INFO_TIME_NOT_SET| */
- +void nfs41_file_info_to_FILE_ID_128(
- + IN const nfs41_file_info *restrict info,
- + OUT FILE_ID_128 *restrict out_fid128)
- +{
- + uint64_t fileid128[2] = {
- + (info->fsid.minor ^ info->fsid.major),
- + info->fileid
- + };
- + (void)memcpy(out_fid128, &fileid128[0], 16);
- +}
- +
- ULONG nfs_file_info_to_attributes(
- IN const nfs41_superblock *restrict superblock,
- diff --git a/daemon/getattr.c b/daemon/getattr.c
- index 0eb7869..c9a0904 100644
- --- a/daemon/getattr.c
- +++ b/daemon/getattr.c
- @@ -219,12 +219,8 @@ static int handle_getattr(void *daemon_context, nfs41_upcall *upcall)
- &args->remote_protocol_info);
- break;
- case FileIdInformation:
- - uint64_t fileid128[2] = {
- - (info.fsid.minor ^ info.fsid.major),
- - info.fileid
- - };
- + nfs41_file_info_to_FILE_ID_128(&info, &args->id_info.FileId);
- args->id_info.VolumeSerialNumber = 0xBABAFACE; /* 64bit! */
- - (void)memcpy(&args->id_info.FileId, &fileid128[0], 16);
- break;
- #ifdef NFS41_DRIVER_WSL_SUPPORT
- case FileStatInformation:
- diff --git a/daemon/readdir.c b/daemon/readdir.c
- index 385534f..0dc1190 100644
- --- a/daemon/readdir.c
- +++ b/daemon/readdir.c
- @@ -256,6 +256,8 @@ typedef union _FILE_DIR_INFO_UNION {
- FILE_ID_FULL_DIR_INFORMATION fifdi;
- FILE_BOTH_DIR_INFORMATION fbdi;
- FILE_ID_BOTH_DIR_INFORMATION fibdi;
- + FILE_ID_EXTD_DIR_INFORMATION fiedi;
- + FILE_ID_EXTD_BOTH_DIR_INFORMATION fiebdi;
- } FILE_DIR_INFO_UNION, *PFILE_DIR_INFO_UNION;
- @@ -304,6 +306,13 @@ static uint32_t readdir_size_for_entry(
- case FileIdFullDirectoryInformation:
- needed += FIELD_OFFSET(FILE_ID_FULL_DIR_INFORMATION, FileName);
- break;
- + case FileIdExtdDirectoryInformation:
- + needed += FIELD_OFFSET(FILE_ID_EXTD_DIR_INFORMATION, FileName);
- + break;
- + case FileIdExtdBothDirectoryInformation:
- + needed += FIELD_OFFSET(FILE_ID_EXTD_BOTH_DIR_INFORMATION,
- + FileName);
- + break;
- case FileFullDirectoryInformation:
- needed += FIELD_OFFSET(FILE_FULL_DIR_INFORMATION, FileName);
- break;
- @@ -395,6 +404,17 @@ static void readdir_copy_shortname(
- }
- #endif /* !NFS41_DRIVER_DISABLE_8DOT3_SHORTNAME_GENERATION */
- +static
- +ULONG get_ea_size(void)
- +{
- + /*
- + * Always return the maximum EA size (64k), so
- + + applications will look for EAs (a value of |0| would
- + * mean "no EAs here")
- + */
- + return (64*1024UL)-1;
- +}
- +
- static void readdir_copy_full_dir_info(
- IN nfs41_readdir_entry *entry,
- IN const nfs41_superblock *restrict superblock,
- @@ -412,12 +432,7 @@ static void readdir_copy_full_dir_info(
- info->fifdi.EaSize = IO_REPARSE_TAG_SYMLINK;
- }
- else {
- - /*
- - * Always return the maximum EA size (64k), so
- - + applications will look for EAs (a value of |0| would
- - * mean "no EAs here")
- - */
- - info->fifdi.EaSize = (64*1024)-1;
- + info->fifdi.EaSize = get_ea_size();
- }
- }
- @@ -592,6 +607,33 @@ static int readdir_copy_entry(
- readdir_copy_filename(wname, wname_size,
- info->fifdi.FileName, &info->fifdi.FileNameLength);
- break;
- + case FileIdExtdDirectoryInformation:
- + readdir_copy_dir_info(entry, superblock, info);
- + info->fiedi.EaSize = get_ea_size();
- + info->fiedi.ReparsePointTag =
- + (entry->attr_info.type == NF4LNK)?
- + IO_REPARSE_TAG_SYMLINK : 0;
- + nfs41_file_info_to_FILE_ID_128(&entry->attr_info, &info->fiedi.FileId);
- + readdir_copy_filename(wname, wname_size,
- + info->fiedi.FileName, &info->fiedi.FileNameLength);
- + break;
- + case FileIdExtdBothDirectoryInformation:
- + readdir_copy_dir_info(entry, superblock, info);
- + info->fiebdi.EaSize = get_ea_size();
- + info->fiebdi.ReparsePointTag =
- + (entry->attr_info.type == NF4LNK)?
- + IO_REPARSE_TAG_SYMLINK : 0;
- + nfs41_file_info_to_FILE_ID_128(&entry->attr_info, &info->fiebdi.FileId);
- +#ifdef NFS41_DRIVER_DISABLE_8DOT3_SHORTNAME_GENERATION
- + info->fiebdi.ShortName[0] = L'\0';
- + info->fiebdi.ShortNameLength = 0;
- +#else
- + readdir_copy_shortname(wname, info->fiebdi.ShortName,
- + &info->fiebdi.ShortNameLength);
- +#endif /* NFS41_DRIVER_DISABLE_8DOT3_SHORTNAME_GENERATION */
- + readdir_copy_filename(wname, wname_size,
- + info->fiebdi.FileName, &info->fiebdi.FileNameLength);
- + break;
- case FileBothDirectoryInformation:
- readdir_copy_both_dir_info(entry, wname, superblock, info);
- readdir_copy_filename(wname, wname_size,
- diff --git a/daemon/util.h b/daemon/util.h
- index 65a1eee..fa01c7e 100644
- --- a/daemon/util.h
- +++ b/daemon/util.h
- @@ -187,6 +187,10 @@ static __inline void open_delegation4_cpy(
- (void)memcpy(dst, src, sizeof(open_delegation4));
- }
- +typedef struct _FILE_ID_128 FILE_ID_128, *PFILE_ID_128;
- +void nfs41_file_info_to_FILE_ID_128(
- + IN const nfs41_file_info *restrict info,
- + OUT FILE_ID_128 *restrict out_fid128);
- ULONG nfs_file_info_to_attributes(
- IN const nfs41_superblock *restrict superblock,
- IN const nfs41_file_info *restrict info);
- diff --git a/include/from_kernel.h b/include/from_kernel.h
- index 66d0cec..db8817c 100644
- --- a/include/from_kernel.h
- +++ b/include/from_kernel.h
- @@ -233,6 +233,42 @@ typedef struct _FILE_ID_BOTH_DIR_INFORMATION {
- WCHAR FileName[1];
- } FILE_ID_BOTH_DIR_INFORMATION, *PFILE_ID_BOTH_DIR_INFORMATION;
- +typedef struct _FILE_ID_EXTD_DIR_INFORMATION {
- + ULONG NextEntryOffset;
- + ULONG FileIndex;
- + LARGE_INTEGER CreationTime;
- + LARGE_INTEGER LastAccessTime;
- + LARGE_INTEGER LastWriteTime;
- + LARGE_INTEGER ChangeTime;
- + LARGE_INTEGER EndOfFile;
- + LARGE_INTEGER AllocationSize;
- + ULONG FileAttributes;
- + ULONG FileNameLength;
- + ULONG EaSize;
- + ULONG ReparsePointTag;
- + FILE_ID_128 FileId;
- + WCHAR FileName[1];
- +} FILE_ID_EXTD_DIR_INFORMATION, *PFILE_ID_EXTD_DIR_INFORMATION;
- +
- +typedef struct _FILE_ID_EXTD_BOTH_DIR_INFORMATION {
- + ULONG NextEntryOffset;
- + ULONG FileIndex;
- + LARGE_INTEGER CreationTime;
- + LARGE_INTEGER LastAccessTime;
- + LARGE_INTEGER LastWriteTime;
- + LARGE_INTEGER ChangeTime;
- + LARGE_INTEGER EndOfFile;
- + LARGE_INTEGER AllocationSize;
- + ULONG FileAttributes;
- + ULONG FileNameLength;
- + ULONG EaSize;
- + ULONG ReparsePointTag;
- + FILE_ID_128 FileId;
- + CCHAR ShortNameLength;
- + WCHAR ShortName[12];
- + WCHAR FileName[1];
- +} FILE_ID_EXTD_BOTH_DIR_INFORMATION, *PFILE_ID_EXTD_BOTH_DIR_INFORMATION;
- +
- typedef struct _FILE_LINK_INFORMATION {
- BOOLEAN ReplaceIfExists;
- HANDLE RootDirectory;
- diff --git a/sys/nfs41sys_dir.c b/sys/nfs41sys_dir.c
- index fee2847..4d0681d 100644
- --- a/sys/nfs41sys_dir.c
- +++ b/sys/nfs41sys_dir.c
- @@ -249,6 +249,8 @@ NTSTATUS nfs41_QueryDirectory(
- case FileIdFullDirectoryInformation:
- case FileBothDirectoryInformation:
- case FileIdBothDirectoryInformation:
- + case FileIdExtdDirectoryInformation:
- + case FileIdExtdBothDirectoryInformation:
- break;
- default:
- print_error("nfs41_QueryDirectory: unhandled dir query class %d\n",
- --
- 2.45.1
msnfs41client: Patch for EaSize in dir listing, implementing |FileIdExtdDirectoryInfo|&co for WSL+misc, 2025-03-28
Posted by Anonymous on Fri 28th Mar 2025 17:27
raw | new post
Submit a correction or amendment below (click here to make a fresh posting)
After submitting an amendment, you'll be able to view the differences between the old and new posts easily.