pastebin - collaborative debugging tool
rovema.kpaste.net RSS


msnfs41client: Patches for Linux nfsd SEEK bug, |FSCTL_QUERY_ALLOCATED_RANGES |+misc, 2025-01-30
Posted by Anonymous on Thu 30th Jan 2025 16:18
raw | new post

  1. From 610744d3c1f21c63bc45714f85db5fe55c0f017f Mon Sep 17 00:00:00 2001
  2. From: Roland Mainz <roland.mainz@nrubsig.org>
  3. Date: Thu, 30 Jan 2025 11:53:58 +0100
  4. Subject: [PATCH 1/6] daemon: Add comment about Linux nfsd SEEK bug, which
  5.  returns |NFS4ERR_NXIO| instead of |sr_eof = TRUE|
  6.  
  7. Add comment about Linux nfsd SEEK bug, which returns |NFS4ERR_NXIO|
  8. instead of |sr_eof = TRUE|.
  9.  
  10. Reported-by: Tigran Mkrtchyan <tigran.mkrtchyan@desy.de>
  11. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  12. ---
  13. daemon/daemon_debug.c | 31 ++++++++++++++++++++++++++++---
  14.  1 file changed, 28 insertions(+), 3 deletions(-)
  15.  
  16. diff --git a/daemon/daemon_debug.c b/daemon/daemon_debug.c
  17. index 16efbd9..cb56e92 100644
  18. --- a/daemon/daemon_debug.c
  19. +++ b/daemon/daemon_debug.c
  20. @@ -1165,7 +1165,29 @@ void debug_list_sparsefile_holes(nfs41_open_state *state)
  21.          NFS4_CONTENT_DATA,
  22.          &seek_sr_eof,
  23.          &seek_sr_offset);
  24. -    if (seek_status && (seek_status != NFS4ERR_NXIO)) {
  25. +
  26. +    /*
  27. +     * 1. Note that Linux returns |NFS4ERR_NXIO| if it cannot find
  28. +     * a data block, but
  29. +     * https://datatracker.ietf.org/doc/html/rfc7862#section-15.11.3
  30. +     * says "... If the server cannot find a corresponding sa_what,
  31. +     * then the status will still be NFS4_OK, but sr_eof would be
  32. +     * TRUE. ..."
  33. +     * 2. NFSv4.2 spec bug:
  34. +     * https://datatracker.ietf.org/doc/html/rfc7862#section-11.2
  35. +     * section "SEEK" does not list |NFS4ERR_NXIO| as valid error
  36. +     * for SEEK, but
  37. +     * https://datatracker.ietf.org/doc/html/rfc7862#section-15.11.3
  38. +     * states "If the sa_offset is beyond the end of the file, then
  39. +     * SEEK MUST return NFS4ERR_NXIO."
  40. +     */
  41. +#define LINUX_NFSD_SEEK_NXIO_BUG_WORKAROUND 1
  42. +
  43. +    if ((seek_status)
  44. +#ifdef LINUX_NFSD_SEEK_NXIO_BUG_WORKAROUND
  45. +        && (seek_status != NFS4ERR_NXIO)
  46. +#endif
  47. +    ) {
  48.          dprintf_out("initial SEEL_DATA failed "
  49.          "OP_SEEK(sa_offset=%llu,sa_what=SEEK_DATA) "
  50.          "failed with %d(='%s')\n",
  51. @@ -1175,8 +1197,11 @@ void debug_list_sparsefile_holes(nfs41_open_state *state)
  52.          goto out;
  53.      }
  54.  
  55. -
  56. -    if (seek_status == NFS4ERR_NXIO) {
  57. +    if (((seek_status == 0) && (seek_sr_eof != FALSE))
  58. +#ifdef LINUX_NFSD_SEEK_NXIO_BUG_WORKAROUND
  59. +        || (seek_status == NFS4ERR_NXIO)
  60. +#endif
  61. +        ) {
  62.          file_has_data_blocks = false;
  63.          offset_of_first_data = ~0ULL;
  64.      }
  65. --
  66. 2.45.1
  67.  
  68. From 5eba533fae76cb06d2776b5d4b7f3fe816ff7f43 Mon Sep 17 00:00:00 2001
  69. From: Dan Shelton <dan.f.shelton@gmail.com>
  70. Date: Thu, 30 Jan 2025 12:21:41 +0100
  71. Subject: [PATCH 2/6] sys: NFS41_SYSOP_WRITE requests should map kernel buffer
  72.  read-only into daemon address space
  73.  
  74. NFS41_SYSOP_WRITE requests should map kernel buffer read-only into daemon
  75. address space.
  76.  
  77. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  78. ---
  79. sys/nfs41sys_readwrite.c | 12 +++++++++++-
  80.  1 file changed, 11 insertions(+), 1 deletion(-)
  81.  
  82. diff --git a/sys/nfs41sys_readwrite.c b/sys/nfs41sys_readwrite.c
  83. index 050819c..4afa8e4 100644
  84. --- a/sys/nfs41sys_readwrite.c
  85. +++ b/sys/nfs41sys_readwrite.c
  86. @@ -118,9 +118,19 @@ NTSTATUS marshal_nfs41_rw(
  87.  #pragma warning (disable : 28145)
  88.          entry->u.ReadWrite.MdlAddress->MdlFlags |= MDL_MAPPING_CAN_FAIL;
  89.  #pragma warning( pop )
  90. +        ULONG prio_writeflags = 0;
  91. +
  92. +        /*
  93. +         * The userland daemon will only read from this memory for
  94. +         * "write" requests, so make it read-only
  95. +         */
  96. +        if (entry->opcode == NFS41_SYSOP_WRITE)
  97. +            prio_writeflags |= MdlMappingNoWrite;
  98. +
  99.          entry->buf =
  100.              MmMapLockedPagesSpecifyCache(entry->u.ReadWrite.MdlAddress,
  101. -                UserMode, MmCached, NULL, FALSE, NormalPagePriority);
  102. +                UserMode, MmCached, NULL, FALSE,
  103. +                (NormalPagePriority|prio_writeflags));
  104.          if (entry->buf == NULL) {
  105.              print_error("marshal_nfs41_rw: "
  106.                  "MmMapLockedPagesSpecifyCache() failed to map pages\n");
  107. --
  108. 2.45.1
  109.  
  110. From ccf5f01d6b7916aeb979155383a9f37bc3a701f8 Mon Sep 17 00:00:00 2001
  111. From: Roland Mainz <roland.mainz@nrubsig.org>
  112. Date: Thu, 30 Jan 2025 14:21:48 +0100
  113. Subject: [PATCH 3/6] sys: Implement dummy version of
  114.  |FSCTL_QUERY_ALLOCATED_RANGES| for testing
  115.  
  116. Implement dummy version of |FSCTL_QUERY_ALLOCATED_RANGES| for
  117. testing, which just returns { 0, filesize } so we can do testing
  118. with Cygwin >= 3.6.x |lseek(..., SEEK_HOLE/SEEK_DATA, ...)| and
  119. Windows $ fsutil sparse queryrange mysparsefile.txt #.
  120.  
  121. ToDo: Full support will require a new upcall into the userland
  122. daemon, which should fill the |FILE_ALLOCATED_RANGE_BUFFER| array
  123. with data from NFSv4.2 SEEK.
  124.  
  125. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  126. ---
  127. sys/nfs41sys_fsctl.c | 107 +++++++++++++++++++++++++++++++++++++++++++
  128.  1 file changed, 107 insertions(+)
  129.  
  130. diff --git a/sys/nfs41sys_fsctl.c b/sys/nfs41sys_fsctl.c
  131. index adb8837..2937d8c 100644
  132. --- a/sys/nfs41sys_fsctl.c
  133. +++ b/sys/nfs41sys_fsctl.c
  134. @@ -68,6 +68,110 @@
  135.  #include "nfs41sys_driver.h"
  136.  #include "nfs41sys_util.h"
  137.  
  138. +static
  139. +NTSTATUS check_nfs41_queryallocatedranges_args(
  140. +    PRX_CONTEXT RxContext)
  141. +{
  142. +    NTSTATUS status = STATUS_SUCCESS;
  143. +    XXCTL_LOWIO_COMPONENT *FsCtl =
  144. +        &RxContext->LowIoContext.ParamsFor.FsCtl;
  145. +    const USHORT HeaderLen = sizeof(FILE_ALLOCATED_RANGE_BUFFER);
  146. +
  147. +    /*
  148. +     * Must have a filename longer than vnetroot name,
  149. +     * or it's trying to operate on the volume itself
  150. +     */
  151. +    if (is_root_directory(RxContext)) {
  152. +        status = STATUS_INVALID_PARAMETER;
  153. +        goto out;
  154. +    }
  155. +
  156. +    if (!FsCtl->pOutputBuffer) {
  157. +        status = STATUS_INVALID_USER_BUFFER;
  158. +        goto out;
  159. +    }
  160. +
  161. +    if (FsCtl->OutputBufferLength < HeaderLen) {
  162. +        RxContext->InformationToReturn = HeaderLen;
  163. +        status = STATUS_BUFFER_TOO_SMALL;
  164. +        goto out;
  165. +    }
  166. +out:
  167. +    return status;
  168. +}
  169. +
  170. +static
  171. +NTSTATUS nfs41_QueryAllocatedRanges(
  172. +    IN OUT PRX_CONTEXT RxContext)
  173. +{
  174. +    NTSTATUS status = STATUS_INVALID_DEVICE_REQUEST;
  175. +    __notnull XXCTL_LOWIO_COMPONENT *FsCtl =
  176. +        &RxContext->LowIoContext.ParamsFor.FsCtl;
  177. +    __notnull PFILE_ALLOCATED_RANGE_BUFFER in_range_buffer =
  178. +        (PFILE_ALLOCATED_RANGE_BUFFER)FsCtl->pInputBuffer;
  179. +    __notnull PFILE_ALLOCATED_RANGE_BUFFER out_range_buffer =
  180. +        (PFILE_ALLOCATED_RANGE_BUFFER)FsCtl->pOutputBuffer;
  181. +    __notnull PNFS41_FCB nfs41_fcb =
  182. +        NFS41GetFcbExtension(RxContext->pFcb);
  183. +
  184. +    DbgEn();
  185. +
  186. +    RxContext->IoStatusBlock.Information = 0;
  187. +
  188. +    status = check_nfs41_queryallocatedranges_args(RxContext);
  189. +    if (status)
  190. +        goto out;
  191. +
  192. +    if (FsCtl->InputBufferLength <
  193. +        sizeof(FILE_ALLOCATED_RANGE_BUFFER)) {
  194. +        DbgP("nfs41_QueryAllocatedRanges: "
  195. +            "in_range_buffer to small\n");
  196. +        status = STATUS_BUFFER_TOO_SMALL;
  197. +        goto out;
  198. +    }
  199. +
  200. +/*
  201. + * FIXME: For now we implement |FSCTL_QUERY_ALLOCATED_RANGES| using
  202. + * a dummy implementation which just returns { 0, filesize }
  203. + * so we can do testing with Cygwin >= 3.6.x
  204. + * |lseek(..., SEEK_HOLE/SEEK_DATA, ...)| and
  205. + * Windows $ fsutil sparse queryrange mysparsefile.txt #.
  206. + *
  207. + * We really need an upcall which issues NFSv4.2 SEEK to enumerate the
  208. + * data/hole sections and fill an array of
  209. + * |FILE_ALLOCATED_RANGE_BUFFER|s with the positions of tha SEEK_DATA
  210. + * results.
  211. + */
  212. +#define NFS41SYS_FSCTL_QUERY_ALLOCATED_RANGES_PLACEHOLDER_DUMMY_IMPL 1
  213. +
  214. +#ifdef NFS41SYS_FSCTL_QUERY_ALLOCATED_RANGES_PLACEHOLDER_DUMMY_IMPL
  215. +    DbgP("nfs41_QueryAllocatedRanges: "
  216. +        "in_range_buffer=(FileOffset=%lld,Length=%lld)\n",
  217. +        (long long)in_range_buffer->FileOffset.QuadPart,
  218. +        (long long)in_range_buffer->Length.QuadPart);
  219. +
  220. +    if (FsCtl->OutputBufferLength <
  221. +        (1*sizeof(FILE_ALLOCATED_RANGE_BUFFER))) {
  222. +        DbgP("nfs41_QueryAllocatedRanges: "
  223. +            "FsCtl->OutputBufferLength too small\n");
  224. +        status = STATUS_BUFFER_TOO_SMALL;
  225. +        goto out;
  226. +    }
  227. +
  228. +    out_range_buffer->FileOffset.QuadPart = 0;
  229. +    out_range_buffer->Length.QuadPart =
  230. +        nfs41_fcb->StandardInfo.EndOfFile.QuadPart;
  231. +
  232. +    RxContext->IoStatusBlock.Information =
  233. +        (ULONG_PTR)1*sizeof(FILE_ALLOCATED_RANGE_BUFFER);
  234. +
  235. +    status = STATUS_SUCCESS;
  236. +#endif /* NFS41SYS_FSCTL_QUERY_ALLOCATED_RANGES_PLACEHOLDER_DUMMY_IMPL */
  237. +
  238. +out:
  239. +    DbgEx();
  240. +    return status;
  241. +}
  242.  
  243.  NTSTATUS nfs41_FsCtl(
  244.      IN OUT PRX_CONTEXT RxContext)
  245. @@ -87,6 +191,9 @@ NTSTATUS nfs41_FsCtl(
  246.      case FSCTL_GET_REPARSE_POINT:
  247.          status = nfs41_GetReparsePoint(RxContext);
  248.          break;
  249. +    case FSCTL_QUERY_ALLOCATED_RANGES:
  250. +        status = nfs41_QueryAllocatedRanges(RxContext);
  251. +        break;
  252.      default:
  253.          break;
  254.      }
  255. --
  256. 2.45.1
  257.  
  258. From 0875aae17b1fb79b4596fb3e047101551372cb56 Mon Sep 17 00:00:00 2001
  259. From: Roland Mainz <roland.mainz@nrubsig.org>
  260. Date: Thu, 30 Jan 2025 16:19:18 +0100
  261. Subject: [PATCH 4/6] sys: |map_close_errors()| should map
  262.  |ERROR_FILE_NOT_FOUND| to |STATUS_NO_SUCH_FILE|
  263.  
  264. |map_close_errors()| should map |ERROR_FILE_NOT_FOUND| to
  265. |STATUS_NO_SUCH_FILE| - this can happen in cases like someone deletes
  266. files on the native filesystem exported by the NFS server.
  267.  
  268. Reported-by: Josh Hurst <joshhurst@gmail.com>
  269. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  270. ---
  271. sys/nfs41sys_openclose.c | 1 +
  272.  1 file changed, 1 insertion(+)
  273.  
  274. diff --git a/sys/nfs41sys_openclose.c b/sys/nfs41sys_openclose.c
  275. index 4398d2a..19b2c3d 100644
  276. --- a/sys/nfs41sys_openclose.c
  277. +++ b/sys/nfs41sys_openclose.c
  278. @@ -1018,6 +1018,7 @@ NTSTATUS map_close_errors(
  279.  {
  280.      switch (status) {
  281.      case NO_ERROR:              return STATUS_SUCCESS;
  282. +    case ERROR_FILE_NOT_FOUND:  return STATUS_NO_SUCH_FILE;
  283.      case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED;
  284.      case ERROR_NOT_EMPTY:       return STATUS_DIRECTORY_NOT_EMPTY;
  285.      case ERROR_FILE_INVALID:    return STATUS_FILE_INVALID;
  286. --
  287. 2.45.1
  288.  
  289. From f00fc3456f6709dd7ee632043f76163b146e15f3 Mon Sep 17 00:00:00 2001
  290. From: Roland Mainz <roland.mainz@nrubsig.org>
  291. Date: Thu, 30 Jan 2025 16:22:18 +0100
  292. Subject: [PATCH 5/6] daemon: Improve error messages in |handle_getattr()|&co.
  293.  
  294. Improve error messages in |handle_getattr()|&co.
  295.  
  296. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  297. ---
  298. daemon/getattr.c | 16 ++++++++++++----
  299.  1 file changed, 12 insertions(+), 4 deletions(-)
  300.  
  301. diff --git a/daemon/getattr.c b/daemon/getattr.c
  302. index 93a23e4..e4388f3 100644
  303. --- a/daemon/getattr.c
  304. +++ b/daemon/getattr.c
  305. @@ -50,7 +50,8 @@ int nfs41_cached_getattr(
  306.  
  307.          status = nfs41_getattr(session, file, &attr_request, info);
  308.          if (status) {
  309. -            eprintf("nfs41_getattr() failed with '%s'\n",
  310. +            eprintf("nfs41_cached_getattr: "
  311. +                "nfs41_getattr() failed with '%s'\n",
  312.                  nfs_error_string(status));
  313.              status = nfs_to_windows_error(status, ERROR_BAD_NET_RESP);
  314.          }
  315. @@ -153,7 +154,10 @@ static int handle_getattr(void *daemon_context, nfs41_upcall *upcall)
  316.  
  317.      status = nfs41_cached_getattr(state->session, &state->file, &info);
  318.      if (status) {
  319. -        eprintf("nfs41_cached_getattr() failed with %d\n", status);
  320. +        eprintf("handle_getattr(state->path.path='%s'): "
  321. +            "nfs41_cached_getattr() failed with %d\n",
  322. +            state->path.path,
  323. +            status);
  324.          goto out;
  325.      }
  326.  
  327. @@ -213,7 +217,10 @@ static int handle_getattr(void *daemon_context, nfs41_upcall *upcall)
  328.          break;
  329.  #endif /* NFS41_DRIVER_WSL_SUPPORT */
  330.      default:
  331. -        eprintf("unhandled file query class %d\n", args->query_class);
  332. +        eprintf("handle_getattr(state->path.path='%s'): "
  333. +            "unhandled file query class %d\n",
  334. +            state->path.path,
  335. +            args->query_class);
  336.          status = ERROR_INVALID_PARAMETER;
  337.          break;
  338.      }
  339. @@ -280,7 +287,8 @@ static int marshall_getattr(unsigned char *buffer, uint32_t *length, nfs41_upcal
  340.          break;
  341.  #endif /* NFS41_DRIVER_WSL_SUPPORT */
  342.      default:
  343. -        eprintf("unknown file query class %d\n", args->query_class);
  344. +        eprintf("marshall_getattr: unknown file query class %d\n",
  345. +            args->query_class);
  346.          status = 103;
  347.          goto out;
  348.      }
  349. --
  350. 2.45.1
  351.  
  352. From af75078ef9db90a9963ac106e4f29bf476c47587 Mon Sep 17 00:00:00 2001
  353. From: Roland Mainz <roland.mainz@nrubsig.org>
  354. Date: Thu, 30 Jan 2025 16:53:46 +0100
  355. Subject: [PATCH 6/6] daemon,nfs41_build_features.h: Add build config to set
  356.  the default NFSv4.x protocol minor version
  357.  
  358. Add build config (|NFS41_DRIVER_DEFAULT_NFS4MINORVERSION|) to set the
  359. default NFSv4.x protocol minor version.
  360.  
  361. Requested for site-specific builds.
  362.  
  363. Reported-by: Lionel Cons <lionelcons1972@gmail.com>
  364. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  365. ---
  366. daemon/namespace.c     | 6 +++++-
  367.  nfs41_build_features.h | 8 ++++++++
  368.  2 files changed, 13 insertions(+), 1 deletion(-)
  369.  
  370. diff --git a/daemon/namespace.c b/daemon/namespace.c
  371. index e034211..25cb94f 100644
  372. --- a/daemon/namespace.c
  373. +++ b/daemon/namespace.c
  374. @@ -400,7 +400,11 @@ int nfs41_root_mount_addrs(
  375.       * First try with 4.2, and if this fails try 4.1
  376.       */
  377.      if (root->nfsminorvers == NFS_VERSION_AUTONEGOTIATION) {
  378. -        root->nfsminorvers = 2;
  379. +#if ((NFS41_DRIVER_DEFAULT_NFS4MINORVERSION > 2) || \
  380. +    (NFS41_DRIVER_DEFAULT_NFS4MINORVERSION < 1))
  381. +#error Illegal NFS41_DRIVER_DEFAULT_NFS4MINORVERSION
  382. +#endif
  383. +        root->nfsminorvers = NFS41_DRIVER_DEFAULT_NFS4MINORVERSION;
  384.          nfsminorvers_autonegotiate = true;
  385.      }
  386.  
  387. diff --git a/nfs41_build_features.h b/nfs41_build_features.h
  388. index bcc765c..a127e40 100644
  389. --- a/nfs41_build_features.h
  390. +++ b/nfs41_build_features.h
  391. @@ -173,4 +173,12 @@
  392.   */
  393.  #define NFS41_DRIVER_WS2022_HACKS 1
  394.  
  395. +/*
  396. + * NFS41_DRIVER_DEFAULT_NFS4MINORVERSION - set default NFSv4.x
  397. + * protocol minor version used by protocol autonegotiation if no
  398. + * minor version was given via $ nfs_mount -o vers= ... #
  399. + * Value can be |1| or |2|
  400. + */
  401. +#define NFS41_DRIVER_DEFAULT_NFS4MINORVERSION 2
  402. +
  403.  #endif /* !_NFS41_DRIVER_BUILDFEATURES_ */
  404. --
  405. 2.45.1

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.

Syntax highlighting:

To highlight particular lines, prefix each line with {%HIGHLIGHT}




All content is user-submitted.
The administrators of this site (kpaste.net) are not responsible for their content.
Abuse reports should be emailed to us at