- From a4912fefe520ffe275f1ea7e9bd16a584a3038ca Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Fri, 7 Feb 2025 15:16:32 +0100
- Subject: [PATCH 1/3] daemon,nfs41_build_features.h: Add build option to treat
- unresolveable symlinks as symdirs
- Add build option to treat unresolveable symlinks as symlinks to
- directories, i.e. the symlink has |FILE_ATTRIBUTE_DIRECTORY| set.
- The idea is that such symlinks are UNC paths or drives (e.g. T:\),
- and powershell+cmd.exe will only cd into such symlinks if
- the flag |FILE_ATTRIBUTE_DIRECTORY| is set for them.
- ToDo: Maybe we should read the symlink value, and only set
- |FILE_ATTRIBUTE_DIRECTORY| if the symlink value ends with
- "/", "/." or "/.." ...
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/getattr.c | 10 +++++++++-
- daemon/open.c | 12 ++++++++++--
- daemon/readdir.c | 11 ++++++++++-
- nfs41_build_features.h | 14 ++++++++++++++
- 4 files changed, 43 insertions(+), 4 deletions(-)
- diff --git a/daemon/getattr.c b/daemon/getattr.c
- index 02ed9ed..9be46be 100644
- --- a/daemon/getattr.c
- +++ b/daemon/getattr.c
- @@ -165,8 +165,16 @@ static int handle_getattr(void *daemon_context, nfs41_upcall *upcall)
- nfs41_file_info target_info;
- int target_status = nfs41_symlink_follow(upcall->root_ref,
- state->session, &state->file, &target_info);
- - if (target_status == NO_ERROR && target_info.type == NF4DIR)
- + if (target_status == NO_ERROR) {
- + info.symlink_dir = target_info.type == NF4DIR;
- + }
- + else {
- +#ifdef NFS41_DRIVER_TREAT_UNRESOLVEABLE_SYMLINKS_AS_DIRS
- info.symlink_dir = TRUE;
- +#else
- + info.symlink_dir = FALSE;
- +#endif /* NFS41_DRIVER_TREAT_UNRESOLVEABLE_SYMLINKS_AS_DIRS */
- + }
- }
- EASSERT((info.attrmask.count > 0) &&
- diff --git a/daemon/open.c b/daemon/open.c
- index 7d99c6d..3f4fbba 100644
- --- a/daemon/open.c
- +++ b/daemon/open.c
- @@ -709,8 +709,16 @@ static int handle_open(void *daemon_context, nfs41_upcall *upcall)
- nfs41_file_info target_info;
- int target_status = nfs41_symlink_follow(upcall->root_ref,
- state->session, &state->file, &target_info);
- - if (target_status == NO_ERROR && target_info.type == NF4DIR)
- - info.symlink_dir = TRUE;
- + if (target_status == NO_ERROR) {
- + info.symlink_dir = target_info.type == NF4DIR;
- + }
- + else {
- +#ifdef NFS41_DRIVER_TREAT_UNRESOLVEABLE_SYMLINKS_AS_DIRS
- + target_info.type = TRUE;
- +#else
- + target_info.type = FALSE;
- +#endif /* NFS41_DRIVER_TREAT_UNRESOLVEABLE_SYMLINKS_AS_DIRS */
- + }
- } else {
- /* replace the path with the symlink target */
- status = nfs41_symlink_target(state->session,
- diff --git a/daemon/readdir.c b/daemon/readdir.c
- index 14325bf..72e9902 100644
- --- a/daemon/readdir.c
- +++ b/daemon/readdir.c
- @@ -25,6 +25,8 @@
- #include <stdlib.h>
- #include <stdbool.h>
- #include <string.h>
- +
- +#include "nfs41_build_features.h"
- #include "nfs41_driver.h" /* for |FILE_INFO_TIME_NOT_SET| */
- #include "from_kernel.h"
- #include "nfs41_ops.h"
- @@ -498,7 +500,14 @@ static int lookup_symlink(
- last_component(path.path, path.path + path.len, &file.name);
- status = nfs41_symlink_follow(root, session, &file, &info);
- - if (status) goto out;
- + if (status) {
- +#ifdef NFS41_DRIVER_TREAT_UNRESOLVEABLE_SYMLINKS_AS_DIRS
- + info_out->symlink_dir = TRUE;
- +#else
- + info_out->symlink_dir = FALSE;
- +#endif /* NFS41_DRIVER_TREAT_UNRESOLVEABLE_SYMLINKS_AS_DIRS */
- + goto out;
- + }
- info_out->symlink_dir = info.type == NF4DIR;
- out:
- diff --git a/nfs41_build_features.h b/nfs41_build_features.h
- index a127e40..d2fd138 100644
- --- a/nfs41_build_features.h
- +++ b/nfs41_build_features.h
- @@ -181,4 +181,18 @@
- */
- #define NFS41_DRIVER_DEFAULT_NFS4MINORVERSION 2
- +/*
- + * NFS41_DRIVER_TREAT_UNRESOLVEABLE_SYMLINKS_AS_DIRS - treat symlinks
- + * which cannot be resolved within the NFS filesystem as dirs
- + *
- + * The idea is that such symlinks are UNC paths or drives (e.g. T:\),
- + * and powershell+cmd.exe will only cd into such symlinks if
- + * the flag |FILE_ATTRIBUTE_DIRECTORY| is set for them.
- + *
- + * ToDo: Maybe we should read the symlink value, and only set
- + * |FILE_ATTRIBUTE_DIRECTORY| if the symlink value ends with
- + * "/", "/." or "/.." ...
- + */
- +#define NFS41_DRIVER_TREAT_UNRESOLVEABLE_SYMLINKS_AS_DIRS 1
- +
- #endif /* !_NFS41_DRIVER_BUILDFEATURES_ */
- --
- 2.45.1
- From 1ce56f8588a9b1de5b1ccdfc3746194ddc16d8af Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 8 Feb 2025 14:10:10 +0100
- Subject: [PATCH 2/3] daemon,sys,tests: Reparse symlinks to other filesystems
- (devletter, UNC)
- Add support for reparsing symlinks to other filesystems via
- devletter (e.g. "/cygdrive/l/foo") and UNC (e.g.
- "\\lab17@2049\nfs4\home\rmainz\tmp") syntax.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/open.c | 83 +++++++++++++++++++++++++++++++++++++++-
- sys/nfs41sys_openclose.c | 67 ++++++++++++++++++++++++--------
- tests/manual_testing.txt | 17 +++++++-
- 3 files changed, 148 insertions(+), 19 deletions(-)
- diff --git a/daemon/open.c b/daemon/open.c
- index 3f4fbba..ad7bb36 100644
- --- a/daemon/open.c
- +++ b/daemon/open.c
- @@ -1,5 +1,6 @@
- /* NFSv4.1 client for Windows
- - * Copyright (C) 2012 The Regents of the University of Michigan
- + * Copyright (C) 2012 The Regents of the University of Michigan
- + * Copyright (C) 2023-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- @@ -605,6 +606,85 @@ static int create_with_ea(
- || (disposition == FILE_OPEN_IF && lookup_status == NFS4ERR_NOENT);
- }
- +static
- +void symlink2ntpath(
- + IN OUT nfs41_abs_path *restrict symlink)
- +{
- + DPRINTF(1, ("--> symlink2ntpath(symlink='%.*s', len=%d)\n",
- + (int)symlink->len,
- + symlink->path,
- + (int)symlink->len));
- +
- + /*
- + * Path with \cygdrive\<devletter> prefix
- + * (e.g. "\cygdrive\l\path" or "\cygdrive\l" ) ?
- + */
- + if ((symlink->len > 10) &&
- + (!memcmp(symlink->path, "\\cygdrive\\", 10)) &&
- + ((symlink->path[11] == '\\') || (symlink->len == 11))) {
- + char deviceletter;
- + char ntpath_buf[NFS41_MAX_PATH_LEN];
- +
- + deviceletter = symlink->path[10];
- +
- + EASSERT(((deviceletter >= 'a') && (deviceletter <= 'z')) ||
- + ((deviceletter >= 'A') && (deviceletter <= 'Z')));
- +
- + /* Return an "drive absolute" NT path */
- + (void)snprintf(ntpath_buf, sizeof(ntpath_buf),
- + "\\??\\%c:\\%.*s",
- + toupper(deviceletter),
- + (int)(symlink->len-12),
- + &symlink->path[12]);
- + (void)strcpy(symlink->path, ntpath_buf);
- + symlink->len = (unsigned short)strlen(ntpath_buf);
- +
- + DPRINTF(1, ("drive abolute symlink='%.*s', len=%d\n",
- + (int)symlink->len,
- + symlink->path,
- + (int)symlink->len));
- + }
- + /* UNC path (e.g. "\\server\path") ? */
- + else if ((symlink->len > 3) &&
- + (!memcmp(symlink->path, "\\\\", 2)) &&
- + (memchr(&symlink->path[3], '\\', symlink->len-3) != NULL)) {
- + char ntpath_buf[NFS41_MAX_PATH_LEN];
- +
- + /* return an (absolute) UNC NT PATH*/
- + (void)snprintf(ntpath_buf, sizeof(ntpath_buf),
- + "\\??\\UNC\\%.*s",
- + (int)(symlink->len-2),
- + &symlink->path[2]);
- + (void)strcpy(symlink->path, ntpath_buf);
- + symlink->len = (unsigned short)strlen(ntpath_buf);
- +
- + DPRINTF(1, ("UNC symlink='%.*s', len=%d\n",
- + (int)symlink->len,
- + symlink->path,
- + (int)symlink->len));
- + }
- + else {
- + /*
- + * All other paths just return a "drive abolute" path,
- + * e.g. \foo\bar
- + *
- + * FIXME/BUG: What about a directory called "??" in the
- + * base of a filesystem, and access it with a trailing
- + * slash in the symlink - that would result in "\??\",
- + * which the kernel would treat as direct NT path...
- + */
- +
- + /* These asserts should never be triggered... */
- + EASSERT(symlink->len > 0);
- + EASSERT((symlink->len > 0) && (symlink->path[0] == '\\'));
- + }
- +
- + DPRINTF(1, ("<-- symlink2ntpath(symlink='%.*s', len=%d)\n",
- + (int)symlink->len,
- + symlink->path,
- + (int)symlink->len));
- +}
- +
- static int handle_open(void *daemon_context, nfs41_upcall *upcall)
- {
- int status = 0;
- @@ -727,6 +807,7 @@ static int handle_open(void *daemon_context, nfs41_upcall *upcall)
- eprintf("nfs41_symlink_target() for '%s' failed with %d\n",
- args->path, status);
- } else {
- + symlink2ntpath(&args->symlink);
- /* tell the driver to call RxPrepareToReparseSymbolicLink() */
- upcall->last_error = ERROR_REPARSE;
- args->symlink_embedded = FALSE;
- diff --git a/sys/nfs41sys_openclose.c b/sys/nfs41sys_openclose.c
- index 19b2c3d..f28ede5 100644
- --- a/sys/nfs41sys_openclose.c
- +++ b/sys/nfs41sys_openclose.c
- @@ -1,6 +1,6 @@
- /* NFSv4.1 client for Windows
- * Copyright (C) 2012 The Regents of the University of Michigan
- - * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
- + * Copyright (C) 2023-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- @@ -735,11 +735,33 @@ retry_on_link:
- UNICODE_STRING AbsPath;
- PCHAR buf;
- BOOLEAN ReparseRequired;
- + /* symhasntpathprefix - symlink target has "\??\" prefix ? */
- + BOOLEAN symhasntpathprefix;
- +
- + if ((entry->u.Open.symlink.Length > (4*sizeof(wchar_t))) &&
- + (!memcmp(entry->u.Open.symlink.Buffer,
- + L"\\??\\", (4*sizeof(wchar_t))))) {
- + symhasntpathprefix = TRUE;
- + DbgP("symhasntpathprefix = TRUE\n");
- + }
- + else {
- + symhasntpathprefix = FALSE;
- + DbgP("symhasntpathprefix = TRUE\n");
- + }
- - /* allocate the string for RxPrepareToReparseSymbolicLink(), and
- - * format an absolute path "DeviceName+VNetRootName+symlink" */
- - AbsPath.Length = DeviceObject->DeviceName.Length +
- - VNetRootPrefix->Length + entry->u.Open.symlink.Length;
- + /*
- + * Allocate the string for |RxPrepareToReparseSymbolicLink()|,
- + * and format an absolute path
- + * "DeviceName+VNetRootName+symlink" if the symlink is
- + * device-relative, or just "symlink" if the input is an NT path
- + * (which starts with "\??\", see above)
- + */
- + AbsPath.Length = 0;
- + if (symhasntpathprefix == FALSE) {
- + AbsPath.Length += DeviceObject->DeviceName.Length +
- + VNetRootPrefix->Length;
- + }
- + AbsPath.Length += entry->u.Open.symlink.Length;
- AbsPath.MaximumLength = AbsPath.Length + sizeof(UNICODE_NULL);
- AbsPath.Buffer = RxAllocatePoolWithTag(NonPagedPoolNx,
- AbsPath.MaximumLength, NFS41_MM_POOLTAG);
- @@ -749,25 +771,36 @@ retry_on_link:
- }
- buf = (PCHAR)AbsPath.Buffer;
- - RtlCopyMemory(buf, DeviceObject->DeviceName.Buffer,
- - DeviceObject->DeviceName.Length);
- - buf += DeviceObject->DeviceName.Length;
- - RtlCopyMemory(buf, VNetRootPrefix->Buffer, VNetRootPrefix->Length);
- - buf += VNetRootPrefix->Length;
- + if (symhasntpathprefix == FALSE) {
- + RtlCopyMemory(buf, DeviceObject->DeviceName.Buffer,
- + DeviceObject->DeviceName.Length);
- + buf += DeviceObject->DeviceName.Length;
- + RtlCopyMemory(buf, VNetRootPrefix->Buffer,
- + VNetRootPrefix->Length);
- + buf += VNetRootPrefix->Length;
- + }
- +
- RtlCopyMemory(buf, entry->u.Open.symlink.Buffer,
- entry->u.Open.symlink.Length);
- - RxFreePool(entry->u.Open.symlink.Buffer);
- - entry->u.Open.symlink.Buffer = NULL;
- buf += entry->u.Open.symlink.Length;
- *(PWCHAR)buf = UNICODE_NULL;
- + RxFreePool(entry->u.Open.symlink.Buffer);
- + entry->u.Open.symlink.Buffer = NULL;
- +
- status = RxPrepareToReparseSymbolicLink(RxContext,
- entry->u.Open.symlink_embedded, &AbsPath, TRUE, &ReparseRequired);
- -#ifdef DEBUG_OPEN
- - DbgP("RxPrepareToReparseSymbolicLink(%u, '%wZ') returned 0x%08lX, "
- - "FileName is '%wZ'\n", entry->u.Open.symlink_embedded,
- - &AbsPath, status, &RxContext->CurrentIrpSp->FileObject->FileName);
- -#endif
- +
- + DbgP("nfs41_Create: "
- + "RxPrepareToReparseSymbolicLink(%u, '%wZ') returned "
- + "ReparseRequired=%d, status=0x%lx, "
- + "FileName is '%wZ'\n",
- + entry->u.Open.symlink_embedded,
- + &AbsPath,
- + (int)ReparseRequired,
- + (long)status,
- + &RxContext->CurrentIrpSp->FileObject->FileName);
- +
- if (status == STATUS_SUCCESS) {
- /* if a reparse is not required, reopen the link itself. this
- * happens with operations on cygwin symlinks, where the reparse
- diff --git a/tests/manual_testing.txt b/tests/manual_testing.txt
- index 6957e0e..75f2053 100644
- --- a/tests/manual_testing.txt
- +++ b/tests/manual_testing.txt
- @@ -1,5 +1,5 @@
- #
- -# ms-nfs41-client manual testing sequence, 2025-02-03
- +# ms-nfs41-client manual testing sequence, 2025-02-08
- #
- # Draft version, needs to be turned into automated tests
- # if possible
- @@ -185,6 +185,21 @@ cmd /C 'mklink /D targetdir1_sym targetdir1'
- cmd /C 'mklink /D targetdir2_sym .\targetdir1'
- # 6. Cygwin /dev/ symlinks, e.g. foo --> /dev/null
- ln -s /dev/zero foo && ls -l foo && rm foo
- +# 7. cmd.exe follow sublink dir to other filesystem
- +# (this assumes we have a drive 'M:' with a subdir "builds")
- +rm -f symlink1_to_m_builds ; cmd /C 'mklink /D symlink1_to_m_builds M:\builds' ; cmd /C 'cd symlink1_to_m_builds && dir && echo test_OK'
- +# 8. cmd.exe follow sublink dir to other filesystem
- +# (this assumes we have an valid UNC path
- +# \\derfwnb4966_ipv6linklocal@2049\nfs4\bigdisk\ with a subdir "builds")
- +rm -f symlink1_to_unc ; cmd /C 'mklink /D symlink1_to_unc \\derfwnb4966_ipv6linklocal@2049\nfs4\bigdisk\builds\' ; cmd /C 'cd symlink1_to_unc && dir && echo test_OK'
- +# 9a. powershell follow sublink dir to other filesystem
- +# (this assumes we have an valid UNC path
- +# \\derfwnb4966_ipv6linklocal@2049\nfs4\bigdisk\ with a subdir "builds")
- +rm -f symlink1_to_unc ; cmd /C 'mklink /D symlink1_to_unc \\derfwnb4966_ipv6linklocal@2049\nfs4\bigdisk\builds\' ; powershell -Command 'cd symlink1_to_unc ; if ($?) { dir ; if ($?) { echo "test OK" } else { echo "dir failed" } } else { echo "cd failed" }'
- +# 9b. powershell follow sublink dir to other filesystem
- +# (this assumes we have an valid UNC path
- +# \\derfwpc5131_ipv4@2049\nfs4\export\home2\rmainz\ with a subdir "tmp")
- +rm -f symlink1_to_h_tmp ; cmd /C 'mklink /D symlink1_to_h_tmp \\derfwpc5131_ipv4@2049\nfs4\export\home2\rmainz\tmp' ; powershell -Command 'cd symlink1_to_h_tmp ; if ($?) { dir ; if ($?) { echo "test OK" } else { echo "dir failed" } } else { echo "cd failed" }'
- #
- # Tests for groups
- --
- 2.45.1
- From de00557e7298320ce584efcf4ef0b19a5842a9ad Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 8 Feb 2025 14:37:15 +0100
- Subject: [PATCH 3/3] cygwin: Document symlink reparse/translation support
- Document symlink reparse/translation support, including
- powershell/cmd.exe/etc. interoperabiilty.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- cygwin/README.bintarball.txt | 14 ++++++++++++++
- 1 file changed, 14 insertions(+)
- diff --git a/cygwin/README.bintarball.txt b/cygwin/README.bintarball.txt
- index de64241..819a4c7 100644
- --- a/cygwin/README.bintarball.txt
- +++ b/cygwin/README.bintarball.txt
- @@ -48,6 +48,10 @@ NFSv4.2/NFSv4.1 filesystem driver for Windows 10/11&Windows Server 2019
- - /sbin/nfs_mount prints UNC paths in Win32+Cygwin/MSYS2 formats
- - Cygwin/MSYS2 bash+ksh93 support UNC paths, e.g.
- cd //derfwnb4966@2049/nfs4/bigdisk/mysqldb4/
- + - Symlinks on NFS can redirect to other filesystems via UNC
- + syntax and work with Cygwin, MSYS2, cmd.exe, powershell etc.,
- + e.g.:
- + mklink /D symlnk1_to_h_tmp \\lab17@2049\nfs4\export\home\rsm\tmp
- - WSL support
- - Mount Windows NFSv4.2 shares via drive letter or UNC path
- @@ -63,6 +67,16 @@ NFSv4.2/NFSv4.1 filesystem driver for Windows 10/11&Windows Server 2019
- - Cygwin /usr/bin/setfacl+/usr/bin/getfacl
- - Windows Explorer ACL dialog
- +- Symlink reparse and translation support
- + - Translates Win32/NT symlink syntax (e.g.
- + $ mklink /D ... Y:\tmp\ #) to NFS/POSIX syntax (e.g.
- + "/cygdrive/y/tmp/") and back
- + - Translates Cygwin /cygdrive/<devletter> symlinks on NFS to
- + Win32 <devletter>:\ and back
- + - Pass-through for NFS /dev-Symlinks (e.g. /dev/null) to Cygwin
- + - Interoperability for symlinks between Cygwin, powershell,
- + cmd.exe and other POSIX-compatible NFSv4.2/NFSv4.1 clients.
- +
- - Support for NFSv4 public mounts (i.e. use the NFSv4 public file handle
- lookup protocol via $ nfs_mount -o public ... #)
- --
- 2.45.1
msnfs41client: Support for symlink reparse to other filesystems+symlink fixes+misc 2025-02-08
Posted by Anonymous on Sat 8th Feb 2025 13:45
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.