- From 56803c3980a0ae29b7364e3d3fc3562750fc57d3 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 8 Jan 2025 11:22:50 +0100
- Subject: [PATCH 01/44] daemon: Fix buffer overflow with long filenames in
- |handle_readdir()|
- Fix buffer overflow with long filenames in |handle_readdir()|.
- Found with $ stress-ng --filename 4 #
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/readdir.c | 7 +++++++
- 1 file changed, 7 insertions(+)
- diff --git a/daemon/readdir.c b/daemon/readdir.c
- index a02b70a..39b50fb 100644
- --- a/daemon/readdir.c
- +++ b/daemon/readdir.c
- @@ -767,6 +767,13 @@ fetch_entries:
- nfs41_readdir_entry *entry = (nfs41_readdir_entry*)entry_buf;
- entry->cookie = 0;
- entry->name_len = (uint32_t)strlen(args->filter) + 1;
- + if (entry->name_len >= NFS41_MAX_COMPONENT_LEN) {
- + DPRINTF(1,
- + ("entry->name_len(=%d) >= NFS41_MAX_COMPONENT_LEN\n",
- + (int)entry->name_len));
- + status = ERROR_FILENAME_EXCED_RANGE;
- + goto out_free_cookie;
- + }
- StringCbCopyA(entry->name, entry->name_len, args->filter);
- entry->next_entry_offset = 0;
- --
- 2.45.1
- From 88a87a8da55eecd23d4f13c8afdf59bb8b115792 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 8 Jan 2025 11:56:20 +0100
- Subject: [PATCH 02/44] README.md,cygwin,daemon,mount,sys,tests: Add basic
- NFSv4.2 support
- Add basic NFSv4.2 support, including protocol autonegotiation.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- README.html | 4 +-
- README.md | 2 +-
- cygwin/README.bintarball.txt | 29 +++++----
- cygwin/devel/msnfs41client.bash | 4 +-
- daemon/callback_server.c | 5 +-
- daemon/callback_xdr.c | 2 +
- daemon/lookup.c | 5 +-
- daemon/mount.c | 12 +++-
- daemon/name_cache.c | 3 +-
- daemon/namespace.c | 66 +++++++++++++++++--
- daemon/nfs41.h | 3 +
- daemon/nfs41_callback.h | 4 ++
- daemon/nfs41_client.c | 11 +++-
- daemon/nfs41_compound.c | 3 +-
- daemon/nfs41_compound.h | 1 +
- daemon/nfs41_const.h | 14 ++--
- daemon/nfs41_ops.c | 112 +++++++++++++++++++++-----------
- daemon/nfs41_ops.h | 23 +++++++
- daemon/nfs41_xdr.c | 21 ++++++
- daemon/service.h | 2 +-
- daemon/upcall.h | 1 +
- mount/mount.c | 3 +
- sys/nfs41_driver.h | 8 +++
- sys/nfs41sys_driver.h | 3 +
- sys/nfs41sys_mount.c | 28 +++++++-
- tests/manual_testing.txt | 20 +++---
- tests/nfs_server_setup.txt | 4 +-
- 27 files changed, 299 insertions(+), 94 deletions(-)
- diff --git a/README.html b/README.html
- index c96aa0a..3400e4c 100644
- --- a/README.html
- +++ b/README.html
- @@ -3,7 +3,7 @@
- <html xmlns="http://www.w3.org/1999/xhtml">
- <head>
- <meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
- - <title>Windows NFS 4.1 Client Instructions</title>
- + <title>Windows NFS 4.2 Client Instructions</title>
- <link rel="stylesheet" title="CITI Default" href="http://www.citi.umich.edu/format/citi.css" type="text/css"/>
- <link rel="icon" href="http://www.citi.umich.edu/images/citilogo-16x16.png" type="image/png"/>
- <link rel="shortcut icon" href="http://www.citi.umich.edu/images/citilogo-16x16.png" type="image/png"/>
- @@ -23,7 +23,7 @@
- </head>
- <body>
- <div id="page">
- -<h1>Windows NFS 4.1 Client Instructions</h1>
- +<h1>Windows NFS 4.2 Client Instructions</h1>
- <div id="content">
- <div id="index">
- <ol>
- diff --git a/README.md b/README.md
- index 4568f14..71b0448 100644
- --- a/README.md
- +++ b/README.md
- @@ -1,5 +1,5 @@
- -# Windows NFS 4.1 Client Instructions
- +# Windows NFS 4.2 Client Instructions
- 1. [Building from Source](#build)
- 2. [Installing Binaries](#install)
- diff --git a/cygwin/README.bintarball.txt b/cygwin/README.bintarball.txt
- index 0afaecc..de64241 100644
- --- a/cygwin/README.bintarball.txt
- +++ b/cygwin/README.bintarball.txt
- @@ -10,13 +10,13 @@
- #
- # 1. What is this ?
- #
- -NFSv4.1 filesystem driver for Windows 10/11&Windows Server 2019
- +NFSv4.2/NFSv4.1 filesystem driver for Windows 10/11&Windows Server 2019
- #
- # 2. Features:
- #
- -- Full NFSv4.1 protocol support
- +- Full NFSv4.2/NFSv4.1 protocol support
- - idmapper (mapping usernames and uid/gid values between server and
- client)
- @@ -50,7 +50,7 @@ NFSv4.1 filesystem driver for Windows 10/11&Windows Server 2019
- cd //derfwnb4966@2049/nfs4/bigdisk/mysqldb4/
- - WSL support
- - - Mount Windows NFSv4.1 shares via drive letter or UNC path
- + - Mount Windows NFSv4.2 shares via drive letter or UNC path
- in WSL via mount -t drvfs
- - Supports NFS owner/group to WSL uid/gid mapping
- @@ -80,18 +80,19 @@ NFSv4.1 filesystem driver for Windows 10/11&Windows Server 2019
- - Supports primary group changes in the calling process/thread
- (via |SetTokenInformation(..., TokenPrimaryGroup,...)|), e.g.
- if the calling process/threads switches the primary group
- - in its access token then the NFSv4.1 client will use that
- + in its access token then the NFSv4.2 client will use that
- group as GID for file creation.
- - newgrp(1)/sg(1)-style "winsg" utilty to run cmd.exe with
- different primary group, e.g.
- $ winsg [-] -g group [-c command | /C command] #
- - Software compatibility:
- - - Any NFSv4.1 server (Linux, Solaris, Illumos, FreeBSD, nfs4j,
- - ...)
- + - Any NFSv4.2/NFSv4.1 server (Linux, Solaris, Illumos,
- + FreeBSD, nfs4j, ...)
- - All tools from Cygwin/MSYS2/MinGW
- - Visual Studio
- - - VMware Workstation (can use VMs hosted on NFSv4.1 filesystem)
- + - VMware Workstation (can use VMs hosted on NFSv4.2/NFSv4.1
- + filesystem)
- #
- @@ -189,7 +190,7 @@ echo %PROCESSOR_ARCHITECTURE%
- - Cygwin 64bit can be installed like this:
- # ---- snip ----
- # Install Cygwin 64bit on Windows 64bit with packages required by "ms-nfs41-client"
- -# (Windows NFSv4.1 client):
- +# (Windows NFSv4.2 client):
- # 1. Create subdir
- mkdir download
- cd download
- @@ -203,7 +204,7 @@ setup-x86_64.exe -q --site "https://mirrors.kernel.org/sourceware/cygwin" -P cyg
- - Cygwin 32bit can be installed like this:
- # ---- snip ----
- # Install Cygwin 32bit on Windows 32bit with packages required by "ms-nfs41-client"
- -# (Windows NFSv4.1 client):
- +# (Windows NFSv4.2 client):
- # 1. Create subdir
- mkdir download
- cd download
- @@ -346,7 +347,7 @@ $ cd ~ && /sbin/nfs_umount '\0.49.202.230@2049\nfs4\net_tmpfs2'
- $ cd ~
- $ net use '\\10.49.202.230@2049\nfs4\net_tmpfs2' /delete
- -# List mounted NFSv4.1 filesystems:
- +# List mounted NFSv4.2 filesystems:
- $ /sbin/nfs_mount
- @@ -369,8 +370,8 @@ does not wait until nfsd*.exe is available for accepting mounts.
- # WSL usage:
- -Example 1: Mount Windows NFSv4.1 share via Windows drive letter
- -# Mount NFSv4.1 share in Windows to drive letter 'N':
- +Example 1: Mount Windows NFSv4.2 share via Windows drive letter
- +# Mount NFSv4.2 share in Windows to drive letter 'N':
- ---- snip ----
- $ /sbin/nfs_mount -o rw 'N' nfs://10.49.202.230//bigdisk
- Successfully mounted '10.49.202.230@2049' to drive 'N:'
- @@ -383,8 +384,8 @@ $ mkdir /mnt/n
- $ mount -t drvfs N: /mnt/n
- ---- snip ----
- -Example 2: Mount Windows NFSv4.1 share via UNC path:
- -# Mount NFSv4.1 share in Windows
- +Example 2: Mount Windows NFSv4.2 share via UNC path:
- +# Mount NFSv4.2 share in Windows
- ---- snip ----
- $ /sbin/nfs_mount -o rw nfs://10.49.202.230//bigdisk
- Successfully mounted '10.49.202.230@2049' to drive '\\10.49.202.230@2049\nfs4\bigdisk'
- diff --git a/cygwin/devel/msnfs41client.bash b/cygwin/devel/msnfs41client.bash
- index 6c9a401..111b328 100755
- --- a/cygwin/devel/msnfs41client.bash
- +++ b/cygwin/devel/msnfs41client.bash
- @@ -1,8 +1,8 @@
- #!/bin/bash
- #
- -# msnfs41client.bash - simple Cygwin frontent for the msnfsv41
- -# NFSv4.1 filesystem driver development
- +# msnfs41client.bash - simple Cygwin frontent for the msnfs41client
- +# NFSv4.2/NFSv4.1 filesystem driver development
- #
- #
- diff --git a/daemon/callback_server.c b/daemon/callback_server.c
- index dc3139a..7c41917 100644
- --- a/daemon/callback_server.c
- +++ b/daemon/callback_server.c
- @@ -387,9 +387,10 @@ static void handle_cb_compound(nfs41_rpc_clnt *rpc_clnt, cb_req *req, struct cb_
- }
- DPRINTF(CBSLVL, ("CB_COMPOUND('%s', %u)\n", args.tag.str, args.argarray_count));
- - if (args.minorversion != 1) {
- + if ((args.minorversion != 1) && (args.minorversion != 2)) {
- res->status = NFS4ERR_MINOR_VERS_MISMATCH; //XXXXX
- - eprintf("args.minorversion %u != 1\n", args.minorversion);
- + eprintf("handle_cb_compound: args.minorversion %u != 1/2\n",
- + (unsigned int)args.minorversion);
- goto out;
- }
- diff --git a/daemon/callback_xdr.c b/daemon/callback_xdr.c
- index d2f4e24..c69341a 100644
- --- a/daemon/callback_xdr.c
- +++ b/daemon/callback_xdr.c
- @@ -574,6 +574,7 @@ static const struct xdr_discrim cb_argop_discrim[] = {
- { OP_CB_WANTS_CANCELLED, (xdrproc_t)op_cb_wants_cancelled_args },
- { OP_CB_NOTIFY_LOCK, (xdrproc_t)op_cb_notify_lock_args },
- { OP_CB_NOTIFY_DEVICEID, (xdrproc_t)op_cb_notify_deviceid_args },
- + { OP_CB_OFFLOAD, NULL_xdrproc_t },
- { OP_CB_ILLEGAL, NULL_xdrproc_t },
- };
- @@ -623,6 +624,7 @@ static const struct xdr_discrim cb_resop_discrim[] = {
- { OP_CB_WANTS_CANCELLED, (xdrproc_t)op_cb_wants_cancelled_res },
- { OP_CB_NOTIFY_LOCK, (xdrproc_t)op_cb_notify_lock_res },
- { OP_CB_NOTIFY_DEVICEID, (xdrproc_t)op_cb_notify_deviceid_res },
- + { OP_CB_OFFLOAD, NULL_xdrproc_t },
- { OP_CB_ILLEGAL, NULL_xdrproc_t },
- };
- diff --git a/daemon/lookup.c b/daemon/lookup.c
- index ee883b9..a27e923 100644
- --- a/daemon/lookup.c
- +++ b/daemon/lookup.c
- @@ -40,7 +40,7 @@
- * We use |128| here to pack as much data into one compound
- * to optimise for high latency links (satellite, ssh tunnel etc.)
- *
- - * The real value is negotiated with the NFSv4.1 server
- + * The real value is negotiated with the NFSv4.2/4.1 server
- * at session creation time (e.g. (|min(
- * session->fore_chan_attrs.ca_maxoperations|,
- * MAX_LOOKUP_COMPONENTS)|), see |max_lookup_components()| below.
- @@ -135,7 +135,8 @@ static int lookup_rpc(
- nfs_argop4 argops[4+MAX_LOOKUP_COMPONENTS*3];
- nfs_resop4 resops[4+MAX_LOOKUP_COMPONENTS*3];
- - compound_init(&compound, argops, resops, "lookup");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "lookup");
- compound_add_op(&compound, OP_SEQUENCE, &args->sequence, &res->sequence);
- nfs41_session_sequence(&args->sequence, session, 0);
- diff --git a/daemon/mount.c b/daemon/mount.c
- index 63c5e1f..be8854f 100644
- --- a/daemon/mount.c
- +++ b/daemon/mount.c
- @@ -52,11 +52,15 @@ static int parse_mount(unsigned char *buffer, uint32_t length, nfs41_upcall *upc
- if (status) goto out;
- status = safe_read(&buffer, &length, &args->use_nfspubfh, sizeof(DWORD));
- if (status) goto out;
- + status = safe_read(&buffer, &length, &args->nfsvers, sizeof(DWORD));
- + if (status) goto out;
- DPRINTF(1, ("parsing NFS41_SYSOP_MOUNT: hostport='%s' root='%s' "
- - "sec_flavor='%s' rsize=%d wsize=%d use_nfspubfh=%d\n",
- + "sec_flavor='%s' rsize=%d wsize=%d use_nfspubfh=%d "
- + "nfsvers=%d\n",
- args->hostport, args->path, secflavorop2name(args->sec_flavor),
- - args->rsize, args->wsize, args->use_nfspubfh));
- + args->rsize, args->wsize, args->use_nfspubfh,
- + args->nfsvers));
- return status;
- out:
- DPRINTF(1, ("parsing NFS41_SYSOP_MOUNT: failed %d\n", status));
- @@ -148,7 +152,9 @@ static int handle_mount(void *daemon_context, nfs41_upcall *upcall)
- } else {
- // create root
- status = nfs41_root_create(hostname, port,
- - args->use_nfspubfh?true:false, args->sec_flavor,
- + args->use_nfspubfh?true:false,
- + args->nfsvers,
- + args->sec_flavor,
- args->wsize + WRITE_OVERHEAD, args->rsize + READ_OVERHEAD, &root);
- if (status) {
- eprintf("nfs41_root_create(hostname='%s', port=%d) failed %d\n",
- diff --git a/daemon/name_cache.c b/daemon/name_cache.c
- index a5e5a1e..444512b 100644
- --- a/daemon/name_cache.c
- +++ b/daemon/name_cache.c
- @@ -1356,7 +1356,8 @@ static int rpc_array_putfh(
- *valid_out = 0;
- - compound_init(&compound, argops, resops, "array_putfh");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "array_putfh");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- diff --git a/daemon/namespace.c b/daemon/namespace.c
- index e0faf23..811e715 100644
- --- a/daemon/namespace.c
- +++ b/daemon/namespace.c
- @@ -26,6 +26,9 @@
- #include "nfs41_ops.h"
- #include "util.h"
- #include "daemon_debug.h"
- +/* for |ERROR_NFS_VERSION_MISMATCH|+|NFS_VERSION_AUTONEGOTIATION| */
- +#include "nfs41_driver.h"
- +
- #define NSLVL 2 /* dprintf level for namespace logging */
- @@ -39,6 +42,7 @@ int nfs41_root_create(
- IN const char *name,
- IN uint32_t port,
- IN bool use_nfspubfh,
- + IN DWORD nfsvers,
- IN uint32_t sec_flavor,
- IN uint32_t wsize,
- IN uint32_t rsize,
- @@ -47,7 +51,10 @@ int nfs41_root_create(
- int status = NO_ERROR;
- nfs41_root *root;
- - DPRINTF(NSLVL, ("--> nfs41_root_create(name='%s', port=%d)\n", name, port));
- + DPRINTF(NSLVL,
- + ("--> nfs41_root_create(name='%s', port=%d, "
- + "use_nfspubfh=%d, nfsvers=%d)\n",
- + name, port, (int)use_nfspubfh, (int)nfsvers));
- root = calloc(1, sizeof(nfs41_root));
- if (root == NULL) {
- @@ -57,6 +64,17 @@ int nfs41_root_create(
- list_init(&root->clients);
- root->use_nfspubfh = use_nfspubfh;
- + if (nfsvers == NFS_VERSION_AUTONEGOTIATION) {
- + /*
- + * Use auto negotiation, |nfs41_root_mount_addrs()| will
- + * set |root->nfsminorvers| to the minor version being used
- + */
- + root->nfsminorvers = NFS_VERSION_AUTONEGOTIATION;
- + }
- + else {
- + root->nfsminorvers = nfsvers % 10; /* 41 --> 1, 42 --> 2, ... */
- + EASSERT((root->nfsminorvers >= 1) && (root->nfsminorvers <= 2));
- + }
- root->wsize = wsize;
- root->rsize = rsize;
- InitializeCriticalSection(&root->lock);
- @@ -64,7 +82,8 @@ int nfs41_root_create(
- root->sec_flavor = sec_flavor;
- /* generate a unique client_owner */
- - status = nfs41_client_owner(name, port, use_nfspubfh, sec_flavor, &root->client_owner);
- + status = nfs41_client_owner(name, port, root->nfsminorvers,
- + use_nfspubfh, sec_flavor, &root->client_owner);
- if (status) {
- eprintf("nfs41_client_owner() failed with %d\n", status);
- free(root);
- @@ -368,12 +387,49 @@ int nfs41_root_mount_addrs(
- goto out;
- }
- + bool nfsminorvers_autonegotiate = false;
- +
- + /*
- + * NFSv4 protocol minor version "autonegotiation"
- + * First try with 4.2, and if this fails try 4.1
- + */
- + if (root->nfsminorvers == NFS_VERSION_AUTONEGOTIATION) {
- + root->nfsminorvers = 2;
- + nfsminorvers_autonegotiate = true;
- + }
- +
- +retry_nfs41_exchange_id:
- + if (nfsminorvers_autonegotiate) {
- + DPRINTF(0, ("nfs41_root_mount_addrs: "
- + "Autonegotiating NFS version, "
- + "trying NFSv4.%d\n",
- + (int)root->nfsminorvers));
- + }
- +
- /* get a clientid with exchangeid */
- - status = nfs41_exchange_id(rpc, &root->client_owner,
- + status = nfs41_exchange_id(rpc, root->nfsminorvers,
- + &root->client_owner,
- nfs41_exchange_id_flags(is_data), &exchangeid);
- if (status) {
- - eprintf("nfs41_exchange_id() failed '%s'\n", nfs_error_string(status));
- - status = ERROR_BAD_NET_RESP;
- + if (status == NFS4ERR_MINOR_VERS_MISMATCH) {
- + if (nfsminorvers_autonegotiate &&
- + (root->nfsminorvers >= 1)) {
- + root->nfsminorvers--;
- + goto retry_nfs41_exchange_id;
- + }
- +
- + eprintf("nfs41_root_mount_addrs: "
- + "nfs41_exchange_id() NFS4ERR_MINOR_VERS_MISMATCH,"
- + "nfsminorvers=%d failed\n",
- + (int)root->nfsminorvers);
- + status = ERROR_NFS_VERSION_MISMATCH;
- + }
- + else {
- + eprintf("nfs41_root_mount_addrs: "
- + "nfs41_exchange_id() failed '%s'\n",
- + nfs_error_string(status));
- + status = ERROR_BAD_NET_RESP;
- + }
- goto out_free_rpc;
- }
- diff --git a/daemon/nfs41.h b/daemon/nfs41.h
- index f5607a6..e6e82ad 100644
- --- a/daemon/nfs41.h
- +++ b/daemon/nfs41.h
- @@ -300,6 +300,7 @@ typedef struct __nfs41_root {
- CRITICAL_SECTION lock;
- struct list_entry clients;
- bool use_nfspubfh;
- + DWORD nfsminorvers;
- uint32_t wsize;
- uint32_t rsize;
- #pragma warning( push )
- @@ -318,6 +319,7 @@ int nfs41_root_create(
- IN const char *name,
- IN uint32_t port,
- IN bool use_nfspubfh,
- + IN DWORD nfsvers,
- IN uint32_t sec_flavor,
- IN uint32_t wsize,
- IN uint32_t rsize,
- @@ -436,6 +438,7 @@ void nfs41_server_addrs(
- int nfs41_client_owner(
- IN const char *name,
- IN uint32_t port,
- + IN int nfsminorvers,
- IN bool use_nfspubfh,
- IN uint32_t sec_flavor,
- OUT client_owner4 *owner);
- diff --git a/daemon/nfs41_callback.h b/daemon/nfs41_callback.h
- index 58e5080..c829920 100644
- --- a/daemon/nfs41_callback.h
- +++ b/daemon/nfs41_callback.h
- @@ -35,6 +35,7 @@ enum nfs41_callback_proc {
- enum nfs41_callback_op {
- OP_CB_GETATTR = 3,
- OP_CB_RECALL = 4,
- + /* Callback operations new to NFSv4.1 */
- OP_CB_LAYOUTRECALL = 5,
- OP_CB_NOTIFY = 6,
- OP_CB_PUSH_DELEG = 7,
- @@ -45,6 +46,9 @@ enum nfs41_callback_op {
- OP_CB_WANTS_CANCELLED = 12,
- OP_CB_NOTIFY_LOCK = 13,
- OP_CB_NOTIFY_DEVICEID = 14,
- + /* Callback operations new to NFSv4.2 */
- + OP_CB_OFFLOAD = 15,
- +
- OP_CB_ILLEGAL = 10044
- };
- diff --git a/daemon/nfs41_client.c b/daemon/nfs41_client.c
- index aa2cc25..b33a22d 100644
- --- a/daemon/nfs41_client.c
- +++ b/daemon/nfs41_client.c
- @@ -174,7 +174,8 @@ int nfs41_client_renew(
- nfs41_exchange_id_res exchangeid = { 0 };
- int status;
- - status = nfs41_exchange_id(client->rpc, &client->owner,
- + status = nfs41_exchange_id(client->rpc,
- + client->root->nfsminorvers, &client->owner,
- nfs41_exchange_id_flags(client->is_data), &exchangeid);
- if (status) {
- eprintf("nfs41_exchange_id() failed with %d\n", status);
- @@ -362,6 +363,7 @@ out:
- int nfs41_client_owner(
- IN const char *name,
- IN uint32_t port,
- + IN int nfsminorvers,
- IN bool use_nfspubfh,
- IN uint32_t sec_flavor,
- OUT client_owner4 *owner)
- @@ -421,6 +423,13 @@ int nfs41_client_owner(
- goto out_context;
- }
- + if (!CryptHashData(hash,
- + (const BYTE*)&nfsminorvers, (DWORD)sizeof(int), 0)) {
- + status = GetLastError();
- + eprintf("CryptHashData() failed with %d\n", status);
- + goto out_hash;
- + }
- +
- if (!CryptHashData(hash,
- (const BYTE*)&use_nfspubfh, (DWORD)sizeof(bool), 0)) {
- status = GetLastError();
- diff --git a/daemon/nfs41_compound.c b/daemon/nfs41_compound.c
- index 08faf55..ecee929 100644
- --- a/daemon/nfs41_compound.c
- +++ b/daemon/nfs41_compound.c
- @@ -40,6 +40,7 @@ int compound_error(int status)
- void compound_init(
- nfs41_compound *compound,
- + int minorversion,
- nfs_argop4 *argops,
- nfs_resop4 *resops,
- const char *tag)
- @@ -47,7 +48,7 @@ void compound_init(
- /* initialize args */
- compound->args.tag_len = (uint32_t)strlen(tag);
- memcpy(compound->args.tag, tag, compound->args.tag_len);
- - compound->args.minorversion = 1;
- + compound->args.minorversion = minorversion;
- compound->args.argarray_count = 0;
- compound->args.argarray = argops;
- diff --git a/daemon/nfs41_compound.h b/daemon/nfs41_compound.h
- index d222559..db75c2c 100644
- --- a/daemon/nfs41_compound.h
- +++ b/daemon/nfs41_compound.h
- @@ -62,6 +62,7 @@ int compound_error(int status);
- void compound_init(
- nfs41_compound *compound,
- + int minorversion,
- nfs_argop4 *argops,
- nfs_resop4 *resops,
- const char *tag);
- diff --git a/daemon/nfs41_const.h b/daemon/nfs41_const.h
- index 0738241..a4e408c 100644
- --- a/daemon/nfs41_const.h
- +++ b/daemon/nfs41_const.h
- @@ -222,13 +222,13 @@ enum nfsstat4 {
- NFS4ERR_DELEG_REVOKED = 10087, /* deleg./layout revoked */
- /* NFSv4.2 errors start here... */
- - NFS4ERR_PARTNER_NOTSUPP = 10088,
- - NFS4ERR_PARTNER_NO_AUTH = 10089,
- - NFS4ERR_UNION_NOTSUPP = 10090,
- - NFS4ERR_OFFLOAD_DENIED = 10091,
- - NFS4ERR_WRONG_LFS = 10092,
- - NFS4ERR_BADLABEL = 10093,
- - NFS4ERR_OFFLOAD_NO_REQS = 10094,
- + NFS4ERR_PARTNER_NOTSUPP = 10088, /* s2s not supported */
- + NFS4ERR_PARTNER_NO_AUTH = 10089, /* s2s not authorized */
- + NFS4ERR_UNION_NOTSUPP = 10090, /* arm of union not supp */
- + NFS4ERR_OFFLOAD_DENIED = 10091, /* dest not allowing copy */
- + NFS4ERR_WRONG_LFS = 10092, /* LFS not supported */
- + NFS4ERR_BADLABEL = 10093, /* incorrect label */
- + NFS4ERR_OFFLOAD_NO_REQS = 10094, /* dest not meeting reqs */
- /* NFSv4 xattr (RFC 8276) error codes start here... */
- NFS4ERR_NOXATTR = 10095,
- diff --git a/daemon/nfs41_ops.c b/daemon/nfs41_ops.c
- index 5e28c09..c49cca0 100644
- --- a/daemon/nfs41_ops.c
- +++ b/daemon/nfs41_ops.c
- @@ -50,6 +50,7 @@
- int nfs41_exchange_id(
- IN nfs41_rpc_clnt *rpc,
- + IN int nfsminorvers,
- IN client_owner4 *owner,
- IN uint32_t flags_in,
- OUT nfs41_exchange_id_res *res_out)
- @@ -64,7 +65,8 @@ int nfs41_exchange_id(
- /* fixme: This should be a function argument */
- extern nfs41_daemon_globals nfs41_dg;
- - compound_init(&compound, &argop, &resop, "exchange_id");
- + compound_init(&compound, nfsminorvers,
- + &argop, &resop, "exchange_id");
- compound_add_op(&compound, OP_EXCHANGE_ID, &ex_id, res_out);
- @@ -142,7 +144,8 @@ int nfs41_create_session(nfs41_client *clnt, nfs41_session *session, bool_t try_
- DPRINTF(1, ("--> nfs41_create_session(clnt=0x%p,session=0x%p,try_recovery=%d)\n",
- clnt, session, (int)try_recovery));
- - compound_init(&compound, &argop, &resop, "create_session");
- + compound_init(&compound, clnt->root->nfsminorvers,
- + &argop, &resop, "create_session");
- compound_add_op(&compound, OP_CREATE_SESSION, &req, &reply);
- @@ -256,7 +259,8 @@ enum nfsstat4 nfs41_bind_conn_to_session(
- nfs41_bind_conn_to_session_args bind_args = { 0 };
- nfs41_bind_conn_to_session_res bind_res = { 0 };
- - compound_init(&compound, &argop, &resop, "bind_conn_to_session");
- + compound_init(&compound, rpc->client->root->nfsminorvers,
- + &argop, &resop, "bind_conn_to_session");
- compound_add_op(&compound, OP_BIND_CONN_TO_SESSION, &bind_args, &bind_res);
- bind_args.sessionid = (unsigned char *)sessionid;
- @@ -283,7 +287,8 @@ int nfs41_destroy_session(
- nfs41_destroy_session_args ds_args;
- nfs41_destroy_session_res ds_res;
- - compound_init(&compound, &argop, &resop, "destroy_session");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + &argop, &resop, "destroy_session");
- compound_add_op(&compound, OP_DESTROY_SESSION, &ds_args, &ds_res);
- ds_args.dsa_sessionid = session->session_id;
- @@ -312,7 +317,8 @@ int nfs41_destroy_clientid(
- nfs41_destroy_clientid_args dc_args;
- nfs41_destroy_clientid_res dc_res;
- - compound_init(&compound, &argops, &resops, "destroy_clientid");
- + compound_init(&compound, rpc->client->root->nfsminorvers,
- + &argops, &resops, "destroy_clientid");
- compound_add_op(&compound, OP_DESTROY_CLIENTID, &dc_args, &dc_res);
- dc_args.dca_clientid = clientid;
- @@ -338,7 +344,8 @@ enum nfsstat4 nfs41_reclaim_complete(
- nfs41_sequence_res sequence_res;
- nfs41_reclaim_complete_res reclaim_res;
- - compound_init(&compound, argops, resops, "reclaim_complete");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "reclaim_complete");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -494,7 +501,8 @@ int nfs41_open(
- attr_request.arr[0] |= FATTR4_WORD0_FSID;
- - compound_init(&compound, argops, resops, "open");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "open");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 1);
- @@ -621,7 +629,8 @@ int nfs41_create(
- nfs41_superblock_getattr_mask(parent->fh.superblock, &attr_request);
- - compound_init(&compound, argops, resops, "create");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "create");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 1);
- @@ -709,7 +718,8 @@ int nfs41_close(
- nfs41_superblock_getattr_mask(file->fh.superblock, &attr_request);
- - compound_init(&compound, argops, resops, "close");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "close");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 1);
- @@ -773,7 +783,8 @@ int nfs41_write(
- nfs41_superblock_getattr_mask(file->fh.superblock, &attr_request);
- - compound_init(&compound, argops, resops,
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops,
- stateid->stateid.seqid == 0 ? "ds write" : "write");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- @@ -858,7 +869,8 @@ int nfs41_read(
- nfs41_read_args read_args;
- nfs41_read_res read_res;
- - compound_init(&compound, argops, resops,
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops,
- stateid->stateid.seqid == 0 ? "ds read" : "read");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- @@ -920,7 +932,8 @@ int nfs41_commit(
- bitmap4 attr_request;
- nfs41_file_info info, *pinfo;
- - compound_init(&compound, argops, resops,
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops,
- do_getattr ? "commit" : "ds commit");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- @@ -988,7 +1001,8 @@ int nfs41_lock(
- nfs41_lock_args lock_args;
- nfs41_lock_res lock_res;
- - compound_init(&compound, argops, resops, "lock");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "lock");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1046,7 +1060,8 @@ int nfs41_unlock(
- nfs41_locku_args locku_args;
- nfs41_locku_res locku_res;
- - compound_init(&compound, argops, resops, "unlock");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "unlock");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1092,7 +1107,8 @@ int nfs41_readdir(
- nfs41_readdir_args readdir_args;
- nfs41_readdir_res readdir_res;
- - compound_init(&compound, argops, resops, "readdir");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "readdir");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1142,7 +1158,8 @@ int nfs41_getattr(
- nfs41_getattr_args getattr_args;
- nfs41_getattr_res getattr_res NDSH(= { 0 });
- - compound_init(&compound, argops, resops, "getattr");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "getattr");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1202,7 +1219,8 @@ int nfs41_superblock_getattr(
- nfs41_openattr_args openattr_args;
- nfs41_openattr_res openattr_res;
- - compound_init(&compound, argops, resops, "getfsattr");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "getfsattr");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1269,7 +1287,8 @@ int nfs41_remove(
- nfs41_superblock_getattr_mask(parent->fh.superblock, &attr_request);
- - compound_init(&compound, argops, resops, "remove");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "remove");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 1);
- @@ -1340,7 +1359,8 @@ int nfs41_rename(
- nfs41_superblock_getattr_mask(src_dir->fh.superblock, &attr_request);
- - compound_init(&compound, argops, resops, "rename");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "rename");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 1);
- @@ -1434,7 +1454,8 @@ int nfs41_setattr(
- nfs41_getattr_res getattr_res NDSH(= { 0 });
- bitmap4 attr_request;
- - compound_init(&compound, argops, resops, "setattr");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "setattr");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1530,7 +1551,8 @@ int nfs41_link(
- nfs41_superblock_getattr_mask(dst_dir->fh.superblock, &cinfo->attrmask);
- cinfo->attrmask.arr[0] |= FATTR4_WORD0_FSID;
- - compound_init(&compound, argops, resops, "link");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "link");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 1);
- @@ -1619,7 +1641,8 @@ int nfs41_readlink(
- nfs41_putfh_res putfh_res;
- nfs41_readlink_res readlink_res;
- - compound_init(&compound, argops, resops, "readlink");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "readlink");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1663,7 +1686,8 @@ int nfs41_access(
- nfs41_access_args access_args;
- nfs41_access_res access_res;
- - compound_init(&compound, argops, resops, "access");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "access");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1700,7 +1724,8 @@ int nfs41_send_sequence(
- nfs41_sequence_args sequence_args;
- nfs41_sequence_res sequence_res;
- - compound_init(&compound, argops, resops, "sequence");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "sequence");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1734,7 +1759,8 @@ enum nfsstat4 nfs41_want_delegation(
- nfs41_want_delegation_args wd_args;
- nfs41_want_delegation_res wd_res;
- - compound_init(&compound, argops, resops, "want_delegation");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "want_delegation");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1768,7 +1794,8 @@ int nfs41_delegpurge(
- nfs41_sequence_res sequence_res;
- nfs41_delegpurge_res dp_res;
- - compound_init(&compound, argops, resops, "delegpurge");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "delegpurge");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1801,7 +1828,8 @@ int nfs41_delegreturn(
- nfs41_delegreturn_args dr_args;
- nfs41_delegreturn_res dr_res;
- - compound_init(&compound, argops, resops, "delegreturn");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "delegreturn");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1849,7 +1877,8 @@ enum nfsstat4 nfs41_fs_locations(
- bitmap4 attr_request = { .count=1, .arr[0]=FATTR4_WORD0_FS_LOCATIONS };
- nfs41_file_info info;
- - compound_init(&compound, argops, resops, "fs_locations");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "fs_locations");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1893,7 +1922,8 @@ int nfs41_secinfo(
- nfs41_secinfo_args secinfo_args;
- nfs41_secinfo_no_name_res secinfo_res;
- - compound_init(&compound, argops, resops, "secinfo");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "secinfo");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1941,7 +1971,8 @@ int nfs41_secinfo_noname(
- nfs41_secinfo_no_name_args noname_args;
- nfs41_secinfo_no_name_res noname_res;
- - compound_init(&compound, argops, resops, "secinfo_no_name");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "secinfo_no_name");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -1986,7 +2017,8 @@ enum nfsstat4 nfs41_free_stateid(
- nfs41_free_stateid_args freestateid_args;
- nfs41_free_stateid_res freestateid_res;
- - compound_init(&compound, argops, resops, "free_stateid");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "free_stateid");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -2018,7 +2050,8 @@ enum nfsstat4 nfs41_test_stateid(
- nfs41_test_stateid_args teststateid_args;
- nfs41_test_stateid_res teststateid_res;
- - compound_init(&compound, argops, resops, "test_stateid");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "test_stateid");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -2061,7 +2094,8 @@ enum nfsstat4 pnfs_rpc_layoutget(
- uint32_t i;
- struct list_entry *entry;
- - compound_init(&compound, argops, resops, "layoutget");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "layoutget");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -2128,7 +2162,8 @@ enum nfsstat4 pnfs_rpc_layoutcommit(
- nfs41_superblock_getattr_mask(file->fh.superblock, &attr_request);
- - compound_init(&compound, argops, resops, "layoutcommit");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "layoutcommit");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -2184,7 +2219,8 @@ enum nfsstat4 pnfs_rpc_layoutreturn(
- nfs41_putfh_res putfh_res;
- pnfs_layoutreturn_args layoutreturn_args;
- - compound_init(&compound, argops, resops, "layoutreturn");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "layoutreturn");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -2225,7 +2261,8 @@ enum nfsstat4 pnfs_rpc_getdeviceinfo(
- pnfs_getdeviceinfo_args getdeviceinfo_args;
- pnfs_getdeviceinfo_res getdeviceinfo_res;
- - compound_init(&compound, argops, resops, "get_deviceinfo");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "get_deviceinfo");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- @@ -2265,7 +2302,8 @@ enum nfsstat4 nfs41_rpc_openattr(
- nfs41_openattr_res openattr_res;
- nfs41_getfh_res getfh_res;
- - compound_init(&compound, argops, resops, "openattr");
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops, "openattr");
- compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- nfs41_session_sequence(&sequence_args, session, 0);
- diff --git a/daemon/nfs41_ops.h b/daemon/nfs41_ops.h
- index 8ac81a3..81bad50 100644
- --- a/daemon/nfs41_ops.h
- +++ b/daemon/nfs41_ops.h
- @@ -86,6 +86,28 @@ enum nfs_opnum4 {
- OP_WANT_DELEGATION = 56,
- OP_DESTROY_CLIENTID = 57,
- OP_RECLAIM_COMPLETE = 58,
- +
- + /* new operations for NFSv4.2 */
- + OP_ALLOCATE = 59,
- + OP_COPY = 60,
- + OP_COPY_NOTIFY = 61,
- + OP_DEALLOCATE = 62,
- + OP_IO_ADVISE = 63,
- + OP_LAYOUTERROR = 64,
- + OP_LAYOUTSTATS = 65,
- + OP_OFFLOAD_CANCEL = 66,
- + OP_OFFLOAD_STATUS = 67,
- + OP_READ_PLUS = 68,
- + OP_SEEK = 69,
- + OP_WRITE_SAME = 70,
- + OP_CLONE = 71,
- +
- + /* xattr support (RFC8726) */
- + OP_GETXATTR = 72,
- + OP_SETXATTR = 73,
- + OP_LISTXATTRS = 74,
- + OP_REMOVEXATTR = 75,
- +
- OP_ILLEGAL = 10044
- };
- @@ -1001,6 +1023,7 @@ typedef struct __pnfs_getdeviceinfo_res {
- /* nfs41_ops.c */
- int nfs41_exchange_id(
- IN nfs41_rpc_clnt *rpc,
- + IN int nfsminorvers,
- IN client_owner4 *owner,
- IN uint32_t flags_in,
- OUT nfs41_exchange_id_res *res_out);
- diff --git a/daemon/nfs41_xdr.c b/daemon/nfs41_xdr.c
- index 2385e5d..0004138 100644
- --- a/daemon/nfs41_xdr.c
- +++ b/daemon/nfs41_xdr.c
- @@ -3624,6 +3624,27 @@ static const op_table_entry g_op_table[] = {
- { encode_op_want_delegation, decode_op_want_delegation }, /* OP_WANT_DELEGATION = 56 */
- { encode_op_destroy_clientid, decode_op_destroy_clientid }, /* OP_DESTROY_CLIENTID = 57 */
- { encode_op_reclaim_complete, decode_op_reclaim_complete }, /* OP_RECLAIM_COMPLETE = 58 */
- +
- + /* new operations for NFSv4.2 */
- + { NULL, NULL }, /* OP_ALLOCATE = 59, */
- + { NULL, NULL }, /* OP_COPY = 60, */
- + { NULL, NULL }, /* OP_COPY_NOTIFY = 61, */
- + { NULL, NULL }, /* OP_DEALLOCATE = 62, */
- + { NULL, NULL }, /* OP_IO_ADVISE = 63, */
- + { NULL, NULL }, /* OP_LAYOUTERROR = 64, */
- + { NULL, NULL }, /* OP_LAYOUTSTATS = 65, */
- + { NULL, NULL }, /* OP_OFFLOAD_CANCEL = 66, */
- + { NULL, NULL }, /* OP_OFFLOAD_STATUS = 67, */
- + { NULL, NULL }, /* OP_READ_PLUS = 68, */
- + { NULL, NULL }, /* OP_SEEK = 69, */
- + { NULL, NULL }, /* OP_WRITE_SAME = 70, */
- + { NULL, NULL }, /* OP_CLONE = 71, */
- +
- + /* xattr support (RFC8726) */
- + { NULL, NULL }, /* OP_GETXATTR = 72, */
- + { NULL, NULL }, /* OP_SETXATTR = 73, */
- + { NULL, NULL }, /* OP_LISTXATTRS = 74, */
- + { NULL, NULL }, /* OP_REMOVEXATTR = 75, */
- };
- static const uint32_t g_op_table_size = ARRAYSIZE(g_op_table);
- diff --git a/daemon/service.h b/daemon/service.h
- index ab60e39..c3dc443 100644
- --- a/daemon/service.h
- +++ b/daemon/service.h
- @@ -66,7 +66,7 @@ extern "C" {
- // internal name of the service
- #define SZSERVICENAME "pnfs"
- // displayed name of the service
- -#define SZSERVICEDISPLAYNAME "NFSv4.1 Client"
- +#define SZSERVICEDISPLAYNAME "NFSv4.2 Client"
- // list of service dependencies - "dep1\0dep2\0\0"
- #define SZDEPENDENCIES ""
- //////////////////////////////////////////////////////////////////////////////
- diff --git a/daemon/upcall.h b/daemon/upcall.h
- index f6ac12d..5f44ede 100644
- --- a/daemon/upcall.h
- +++ b/daemon/upcall.h
- @@ -37,6 +37,7 @@ typedef struct __mount_upcall_args {
- DWORD rsize;
- DWORD wsize;
- DWORD use_nfspubfh;
- + DWORD nfsvers;
- DWORD lease_time;
- FILE_FS_ATTRIBUTE_INFORMATION FsAttrs;
- } mount_upcall_args;
- diff --git a/mount/mount.c b/mount/mount.c
- index c9b5637..2ebf8e5 100644
- --- a/mount/mount.c
- +++ b/mount/mount.c
- @@ -129,6 +129,9 @@ void PrintMountUsage(LPWSTR pProcess)
- "\tro\tmount as read-only\n"
- "\trw\tmount as read-write (default)\n"
- "\tport=#\tTCP port to use (defaults to 2049)\n"
- + "\tvers=#\tNFS protocol version, either 4.1 or 4.2\n"
- + "\t\tIf this option is not specified, the client negotiates a\n"
- + "\t\tsuitable version with the server, trying version 4.2 and then 4.1\n"
- "\trsize=#\tread buffer size in bytes\n"
- "\twsize=#\twrite buffer size in bytes\n"
- "\tsec=sys:krb5:krb5i:krb5p\tspecify (gss) security flavor\n"
- diff --git a/sys/nfs41_driver.h b/sys/nfs41_driver.h
- index c08e8fb..7d933f0 100644
- --- a/sys/nfs41_driver.h
- +++ b/sys/nfs41_driver.h
- @@ -104,4 +104,12 @@ typedef enum _nfs41_start_driver_state {
- NFS41_START_DRIVER_STARTED,
- NFS41_START_DRIVER_STOPPED
- } nfs41_start_driver_state;
- +
- +#define NFS_VERSION_AUTONEGOTIATION (0xFFFF)
- +
- +/*
- + * Error/Status codes
- + */
- +#define ERROR_NFS_VERSION_MISMATCH ERROR_REMOTE_FILE_VERSION_MISMATCH
- +#define STATUS_NFS_VERSION_MISMATCH STATUS_REMOTE_FILE_VERSION_MISMATCH
- #endif
- diff --git a/sys/nfs41sys_driver.h b/sys/nfs41sys_driver.h
- index 0e0107c..79ef82c 100644
- --- a/sys/nfs41sys_driver.h
- +++ b/sys/nfs41sys_driver.h
- @@ -208,6 +208,7 @@ typedef struct _updowncall_entry {
- DWORD wsize;
- DWORD lease_time;
- DWORD use_nfspubfh;
- + DWORD nfsvers;
- } Mount;
- struct {
- PMDL MdlAddress;
- @@ -308,6 +309,7 @@ typedef struct _NFS41_MOUNT_CREATEMODE {
- typedef struct _NFS41_MOUNT_CONFIG {
- BOOLEAN use_nfspubfh;
- + DWORD nfsvers;
- DWORD ReadSize;
- DWORD WriteSize;
- BOOLEAN ReadOnly;
- @@ -415,6 +417,7 @@ typedef struct _NFS41_V_NET_ROOT_EXTENSION {
- NFS41_MOUNT_CREATEMODE dir_createmode;
- NFS41_MOUNT_CREATEMODE file_createmode;
- USHORT MountPathLen;
- + DWORD nfsvers;
- BOOLEAN read_only;
- BOOLEAN write_thru;
- BOOLEAN nocache;
- diff --git a/sys/nfs41sys_mount.c b/sys/nfs41sys_mount.c
- index 8cc45f4..93a6d34 100644
- --- a/sys/nfs41sys_mount.c
- +++ b/sys/nfs41sys_mount.c
- @@ -112,7 +112,7 @@ NTSTATUS marshal_nfs41_mount(
- goto out;
- }
- header_len = *len + length_as_utf8(entry->u.Mount.srv_name) +
- - length_as_utf8(entry->u.Mount.root) + 4 * sizeof(DWORD);
- + length_as_utf8(entry->u.Mount.root) + 5 * sizeof(DWORD);
- if (header_len > buf_len) {
- status = STATUS_INSUFFICIENT_RESOURCES;
- goto out;
- @@ -128,16 +128,20 @@ NTSTATUS marshal_nfs41_mount(
- RtlCopyMemory(tmp, &entry->u.Mount.wsize, sizeof(DWORD));
- tmp += sizeof(DWORD);
- RtlCopyMemory(tmp, &entry->u.Mount.use_nfspubfh, sizeof(DWORD));
- + tmp += sizeof(DWORD);
- + RtlCopyMemory(tmp, &entry->u.Mount.nfsvers, sizeof(DWORD));
- *len = header_len;
- #ifdef DEBUG_MARSHAL_DETAIL
- DbgP("marshal_nfs41_mount: server name='%wZ' mount point='%wZ' "
- - "sec_flavor='%s' rsize=%d wsize=%d use_nfspubfh=%d\n",
- + "sec_flavor='%s' rsize=%d wsize=%d use_nfspubfh=%d "
- + "nfsvers=%d\n",
- entry->u.Mount.srv_name, entry->u.Mount.root,
- secflavorop2name(entry->u.Mount.sec_flavor),
- (int)entry->u.Mount.rsize, (int)entry->u.Mount.wsize,
- - (int)entry->u.Mount.use_nfspubfh);
- + (int)entry->u.Mount.use_nfspubfh,
- + (int)entry->u.Mount.nfsvers);
- #endif
- out:
- return status;
- @@ -227,6 +231,7 @@ NTSTATUS map_mount_errors(
- case ERROR_BAD_NET_NAME: return STATUS_BAD_NETWORK_NAME;
- case ERROR_BAD_NETPATH: return STATUS_BAD_NETWORK_PATH;
- case ERROR_NOT_SUPPORTED: return STATUS_NOT_SUPPORTED;
- + case ERROR_NFS_VERSION_MISMATCH: return STATUS_NFS_VERSION_MISMATCH;
- case ERROR_INTERNAL_ERROR: return STATUS_INTERNAL_ERROR;
- default:
- print_error("map_mount_errors: "
- @@ -261,6 +266,7 @@ NTSTATUS nfs41_mount(
- entry->u.Mount.rsize = config->ReadSize;
- entry->u.Mount.wsize = config->WriteSize;
- entry->u.Mount.use_nfspubfh = config->use_nfspubfh;
- + entry->u.Mount.nfsvers = config->nfsvers;
- entry->u.Mount.sec_flavor = sec_flavor;
- entry->u.Mount.FsAttrs = FsAttrs;
- @@ -295,6 +301,7 @@ void nfs41_MountConfig_InitDefaults(
- Config->ReadSize = MOUNT_CONFIG_RW_SIZE_DEFAULT;
- Config->WriteSize = MOUNT_CONFIG_RW_SIZE_DEFAULT;
- Config->use_nfspubfh = FALSE;
- + Config->nfsvers = NFS_VERSION_AUTONEGOTIATION;
- Config->ReadOnly = FALSE;
- Config->write_thru = FALSE;
- Config->nocache = FALSE;
- @@ -484,6 +491,16 @@ NTSTATUS nfs41_MountConfig_ParseOptions(
- &Config->WriteSize, MOUNT_CONFIG_RW_SIZE_MIN,
- MOUNT_CONFIG_RW_SIZE_MAX);
- }
- + else if (wcsncmp(L"vers", Name, NameLen) == 0) {
- + if (wcsncmp(L"4.2", usValue.Buffer, usValue.Length) == 0)
- + Config->nfsvers = 42;
- + else if (wcsncmp(L"4.1", usValue.Buffer, usValue.Length) == 0)
- + Config->nfsvers = 41;
- + else {
- + status = STATUS_INVALID_PARAMETER;
- + print_error("Invalid vers= string\n");
- + }
- + }
- else if (wcsncmp(L"public", Name, NameLen) == 0) {
- /*
- + We ignore this value here, and instead rely on the
- @@ -874,6 +891,8 @@ NTSTATUS nfs41_CreateVNetRoot(
- DbgP("nfs41_MountConfig_ParseOptions() failed\n");
- goto out_free;
- }
- +
- + pVNetRootContext->nfsvers = Config->nfsvers;
- pVNetRootContext->read_only = Config->ReadOnly;
- pVNetRootContext->write_thru = Config->write_thru;
- pVNetRootContext->nocache = Config->nocache;
- @@ -996,6 +1015,7 @@ NTSTATUS nfs41_CreateVNetRoot(
- goto out_free;
- }
- + pVNetRootContext->nfsvers = Config->nfsvers;
- pVNetRootContext->read_only = Config->ReadOnly;
- pVNetRootContext->write_thru = Config->write_thru;
- pVNetRootContext->nocache = Config->nocache;
- @@ -1008,6 +1028,7 @@ NTSTATUS nfs41_CreateVNetRoot(
- "MntPt='%wZ', "
- "SrvName='%wZ', "
- "usenfspubfh=%d, "
- + "nfsvers=%d, "
- "ro=%d, "
- "writethru=%d, "
- "nocache=%d "
- @@ -1019,6 +1040,7 @@ NTSTATUS nfs41_CreateVNetRoot(
- &Config->MntPt,
- &Config->SrvName,
- Config->use_nfspubfh?1:0,
- + (int)Config->nfsvers,
- Config->ReadOnly?1:0,
- Config->write_thru?1:0,
- Config->nocache?1:0,
- diff --git a/tests/manual_testing.txt b/tests/manual_testing.txt
- index 6ee7186..3f5b115 100644
- --- a/tests/manual_testing.txt
- +++ b/tests/manual_testing.txt
- @@ -1,5 +1,5 @@
- #
- -# ms-nfs41-client manual testing sequence, 2024-12-19
- +# ms-nfs41-client manual testing sequence, 2025-01-08
- #
- # Draft version, needs to be turned into automated tests
- # if possible
- @@ -94,8 +94,8 @@
- # regtool -i set '/HKEY_LOCAL_MACHINE/SOFTWARE/Microsoft/Windows/CurrentVersion/Policies/DataCollection/AllowTelemetry' 0
- # ---- snip ----
- #
- -# - A timeserver shoud be running on both Windows (NFSv4.1 client and
- -# NFSv4.1 server).
- +# - A timeserver shoud be running on both Windows (NFSv4.2/NFSv4.1
- +# client and NFSv4.2/NFSv4.1 server).
- # For example on Windows add timeserver 10.49.0.6 like this:
- # ---- snip ----
- # sc config w32time start=auto
- @@ -196,7 +196,7 @@ root@DERFWNB4966:~# usermod -a -G cygwingrp2 roland_mainz
- #
- Testcase:
- -------- snip --------
- -# cd to a NFSv4.1 filesystem
- +# cd to a NFS >= v4.1 filesystem
- $ rm -f test1.txt
- $ touch test1.txt
- $ icacls test1.txt /grant:r 'cygwingrp1:(WDAC)' /t /c
- @@ -220,7 +220,7 @@ other::r--
- or one-liner:
- -------- snip --------
- -# cd to a NFSv4.1 filesystem
- +# cd to a NFS >= v4.1 filesystem
- # getfact output should contain both "cygwingrp1" and "cygwingrp2"
- ksh93 -c 'rm -f test1.txt ; touch test1.txt ; icacls test1.txt /grant:r "cygwingrp1:(WDAC)" /grant:r "cygwingrp2:(WDAC)" /t /c ; getfacl test1.txt | grep -C 20 --colour -E "cygwingrp[12]"'
- -------- snip --------
- @@ -244,7 +244,7 @@ icacls mytestfile1.txt | grep --colour -E 'cygwingrp2.+GR'
- #
- # Compile each of the following package
- -# on a NFSv4.1 share, and run each build in parallel/sequence
- +# on a NFS >= v4.1 share, and run each build in parallel/sequence
- # multiple times on one or multiple mounts
- #
- # ./nfs_mount -p -o sec=sys T derfwnb4966_ipv6:/net_tmpfs2
- @@ -364,7 +364,7 @@ make -j8 all
- #
- -# Run Cygwin installer from NFSv4.1 share
- +# Run Cygwin installer from NFS >= v4.1 share
- #
- wget 'https://www.cygwin.com/setup-x86_64.exe'
- chmod a+rx setup-x86_64.exe
- @@ -440,21 +440,21 @@ svn checkout https://svn.FreeBSD.org/base/head/share/man
- #
- -# Run parallel make job on NFSv4.1 filesystem (/cygdrive/n/xxx/)
- +# Run parallel make job on NFS >= v4.1 filesystem (/cygdrive/n/xxx/)
- #
- cd /cygdrive/n/xxx/
- time ksh93 $msnfs41clientgitroot/tests/fstest_make_numtree1/fstest_make_numtree1.ksh93 all
- #
- -# Run DrMemory with log dir on NFSv4.1 filesystem
- +# Run DrMemory with log dir on NFS >= v4.1 filesystem
- #
- cd /cygdrive/n/xxx/
- drmemory -batch -check_uninit_all -strict_bitops -logdir "$(cygpath -w "$PWD")" -- "$(cygpath -w /sbin/nfs_mount)"
- #
- # Run Windows tar (/cygdrive/c/Windows/system32/tar) tests
- -# on NFSv4.1 filesystem
- +# on NFS >= v4.1 filesystem
- #
- # Notes:
- # - Win10 /cygdrive/c/Windows/system32/tar uses write-only handles
- diff --git a/tests/nfs_server_setup.txt b/tests/nfs_server_setup.txt
- index 68836e8..9f3a2a3 100644
- --- a/tests/nfs_server_setup.txt
- +++ b/tests/nfs_server_setup.txt
- @@ -1,10 +1,10 @@
- #
- -# NFSv4.1 server setup for testing
- +# NFSv4.2/NFSv4.1 server setup for testing
- #
- #
- # TODO:
- -# - Debian Linux NFSv4.1 server setup
- +# - Debian Linux NFSv4.2 server setup
- # - Solaris 11.4 NFSv4.1 server setup
- # - Illumos 11.4 NFSv4.1 server setup
- # - FreeBSD NFSv4.1 server setup
- --
- 2.45.1
- From 1b489cf00e6c04e1edb8968f00cd2db5e66deeb0 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 8 Jan 2025 12:01:29 +0100
- Subject: [PATCH 03/44] daemon: Remove unnecessary |ZeroMemory()| from
- |compound_init()|
- Remove unnecessary |ZeroMemory()| from |compound_init()|
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/nfs41_compound.c | 3 ++-
- 1 file changed, 2 insertions(+), 1 deletion(-)
- diff --git a/daemon/nfs41_compound.c b/daemon/nfs41_compound.c
- index ecee929..cb35663 100644
- --- a/daemon/nfs41_compound.c
- +++ b/daemon/nfs41_compound.c
- @@ -53,7 +53,8 @@ void compound_init(
- compound->args.argarray = argops;
- /* initialize results */
- - ZeroMemory(&compound->res, sizeof(nfs41_compound_res));
- + compound->res.status = 0;
- + compound->res.tag[0] = '\0';
- compound->res.tag_len = NFS4_OPAQUE_LIMIT;
- compound->res.resarray_count = 0;
- compound->res.resarray = resops;
- --
- 2.45.1
- From 1dde35a6521b22633fa8c04c85e43810fe904484 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 8 Jan 2025 12:16:40 +0100
- Subject: [PATCH 04/44] cygwin: msnfs41client "require_file" uses the wrong
- file name
- msnfs41client "require_file" uses the wrong file name for error
- messages.
- Reported-by: Dan Shelton <dan.f.shelton@gmail.com>
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- cygwin/devel/msnfs41client.bash | 2 +-
- 1 file changed, 1 insertion(+), 1 deletion(-)
- diff --git a/cygwin/devel/msnfs41client.bash b/cygwin/devel/msnfs41client.bash
- index 111b328..cff5cc2 100755
- --- a/cygwin/devel/msnfs41client.bash
- +++ b/cygwin/devel/msnfs41client.bash
- @@ -745,7 +745,7 @@ function require_file
- typeset testfile="$1"
- if [[ ! -f "$testfile" ]] ; then
- - printf $"%s: File %q not found in %q\n" "$0" "$cmd" "$PWD" 1>&2
- + printf $"%s: File %q not found in %q\n" "$0" "$testfile" "$PWD" 1>&2
- return 1
- fi
- return 0
- --
- 2.45.1
- From 923fa90cfb5ae8dbed27cb846b2b98c72a476b48 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 8 Jan 2025 13:27:00 +0100
- Subject: [PATCH 05/44] build.vc19,sys,include: Move sys/nfs41_driver.h to
- include/nfs41_driver.h
- Move sys/nfs41_driver.h to include/nfs41_driver.h
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- build.vc19/nfs41_np/nfs41_np.vcxproj | 12 ++++++------
- build.vc19/nfs_mount/nfs_mount.vcxproj | 12 ++++++------
- build.vc19/nfsd/nfsd.vcxproj | 12 ++++++------
- {sys => include}/nfs41_driver.h | 0
- 4 files changed, 18 insertions(+), 18 deletions(-)
- rename {sys => include}/nfs41_driver.h (100%)
- diff --git a/build.vc19/nfs41_np/nfs41_np.vcxproj b/build.vc19/nfs41_np/nfs41_np.vcxproj
- index e07ccfc..c61e65a 100644
- --- a/build.vc19/nfs41_np/nfs41_np.vcxproj
- +++ b/build.vc19/nfs41_np/nfs41_np.vcxproj
- @@ -121,7 +121,7 @@
- <WarningLevel>Level4</WarningLevel>
- <Optimization>Disabled</Optimization>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;_DEBUG;_WINDOWS;_USRDLL;NFS41_NP_EXPORTS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDebugDLL</RuntimeLibrary>
- <AdditionalOptions>/wd4100</AdditionalOptions>
- @@ -140,7 +140,7 @@
- <WarningLevel>Level4</WarningLevel>
- <Optimization>Disabled</Optimization>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;_DEBUG;_WINDOWS;_USRDLL;NFS41_NP_EXPORTS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDebugDLL</RuntimeLibrary>
- <AdditionalOptions>/wd4100</AdditionalOptions>
- @@ -159,7 +159,7 @@
- <WarningLevel>Level4</WarningLevel>
- <Optimization>Disabled</Optimization>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;_DEBUG;_WINDOWS;_USRDLL;NFS41_NP_EXPORTS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDebugDLL</RuntimeLibrary>
- <AdditionalOptions>/wd4100</AdditionalOptions>
- @@ -180,7 +180,7 @@
- <FunctionLevelLinking>true</FunctionLevelLinking>
- <IntrinsicFunctions>true</IntrinsicFunctions>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;NDEBUG;_WINDOWS;_USRDLL;NFS41_NP_EXPORTS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDLL</RuntimeLibrary>
- <AdditionalOptions>/wd4100</AdditionalOptions>
- @@ -202,7 +202,7 @@
- <FunctionLevelLinking>true</FunctionLevelLinking>
- <IntrinsicFunctions>true</IntrinsicFunctions>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;NDEBUG;_WINDOWS;_USRDLL;NFS41_NP_EXPORTS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDLL</RuntimeLibrary>
- <AdditionalOptions>/wd4100</AdditionalOptions>
- @@ -224,7 +224,7 @@
- <FunctionLevelLinking>true</FunctionLevelLinking>
- <IntrinsicFunctions>true</IntrinsicFunctions>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;NDEBUG;_WINDOWS;_USRDLL;NFS41_NP_EXPORTS;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDLL</RuntimeLibrary>
- <AdditionalOptions>/wd4100</AdditionalOptions>
- diff --git a/build.vc19/nfs_mount/nfs_mount.vcxproj b/build.vc19/nfs_mount/nfs_mount.vcxproj
- index c85ab61..399d150 100644
- --- a/build.vc19/nfs_mount/nfs_mount.vcxproj
- +++ b/build.vc19/nfs_mount/nfs_mount.vcxproj
- @@ -137,7 +137,7 @@
- <WarningLevel>Level4</WarningLevel>
- <Optimization>Disabled</Optimization>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDebugDLL</RuntimeLibrary>
- <SupportJustMyCode>false</SupportJustMyCode>
- @@ -155,7 +155,7 @@
- <WarningLevel>Level4</WarningLevel>
- <Optimization>Disabled</Optimization>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDebugDLL</RuntimeLibrary>
- <SupportJustMyCode>false</SupportJustMyCode>
- @@ -173,7 +173,7 @@
- <WarningLevel>Level4</WarningLevel>
- <Optimization>Disabled</Optimization>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDebugDLL</RuntimeLibrary>
- <SupportJustMyCode>false</SupportJustMyCode>
- @@ -193,7 +193,7 @@
- <FunctionLevelLinking>true</FunctionLevelLinking>
- <IntrinsicFunctions>true</IntrinsicFunctions>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDLL</RuntimeLibrary>
- </ClCompile>
- @@ -214,7 +214,7 @@
- <FunctionLevelLinking>true</FunctionLevelLinking>
- <IntrinsicFunctions>true</IntrinsicFunctions>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDLL</RuntimeLibrary>
- </ClCompile>
- @@ -235,7 +235,7 @@
- <FunctionLevelLinking>true</FunctionLevelLinking>
- <IntrinsicFunctions>true</IntrinsicFunctions>
- <PreprocessorDefinitions>WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\sys;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <RuntimeLibrary>MultiThreadedDLL</RuntimeLibrary>
- </ClCompile>
- diff --git a/build.vc19/nfsd/nfsd.vcxproj b/build.vc19/nfsd/nfsd.vcxproj
- index 43d07f7..6c1d743 100644
- --- a/build.vc19/nfsd/nfsd.vcxproj
- +++ b/build.vc19/nfsd/nfsd.vcxproj
- @@ -137,7 +137,7 @@
- <WarningLevel>Level4</WarningLevel>
- <Optimization>Disabled</Optimization>
- <PreprocessorDefinitions>WIN32_LEAN_AND_MEAN;FD_SETSIZE=1024;INET6;NO_CB_4_KRB5P;STANDALONE_NFSD;_CRT_SECURE_NO_WARNINGS;_WINSOCK_DEPRECATED_NO_WARNINGS;WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\sys;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <BasicRuntimeChecks>Default</BasicRuntimeChecks>
- <BufferSecurityCheck>false</BufferSecurityCheck>
- @@ -159,7 +159,7 @@
- <WarningLevel>Level4</WarningLevel>
- <Optimization>Disabled</Optimization>
- <PreprocessorDefinitions>WIN32_LEAN_AND_MEAN;FD_SETSIZE=1024;INET6;NO_CB_4_KRB5P;STANDALONE_NFSD;_CRT_SECURE_NO_WARNINGS;_WINSOCK_DEPRECATED_NO_WARNINGS;WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\sys;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <BasicRuntimeChecks>Default</BasicRuntimeChecks>
- <BufferSecurityCheck>false</BufferSecurityCheck>
- @@ -181,7 +181,7 @@
- <WarningLevel>Level4</WarningLevel>
- <Optimization>Disabled</Optimization>
- <PreprocessorDefinitions>WIN32_LEAN_AND_MEAN;FD_SETSIZE=1024;INET6;NO_CB_4_KRB5P;STANDALONE_NFSD;_CRT_SECURE_NO_WARNINGS;_WINSOCK_DEPRECATED_NO_WARNINGS;WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\sys;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <BasicRuntimeChecks>Default</BasicRuntimeChecks>
- <BufferSecurityCheck>false</BufferSecurityCheck>
- @@ -205,7 +205,7 @@
- <FunctionLevelLinking>true</FunctionLevelLinking>
- <IntrinsicFunctions>true</IntrinsicFunctions>
- <PreprocessorDefinitions>WIN32_LEAN_AND_MEAN;FD_SETSIZE=1024;INET6;NO_CB_4_KRB5P;_CRT_SECURE_NO_WARNINGS;_WINSOCK_DEPRECATED_NO_WARNINGS;WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\sys;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <BufferSecurityCheck>false</BufferSecurityCheck>
- <StringPooling>true</StringPooling>
- @@ -229,7 +229,7 @@
- <FunctionLevelLinking>true</FunctionLevelLinking>
- <IntrinsicFunctions>true</IntrinsicFunctions>
- <PreprocessorDefinitions>WIN32_LEAN_AND_MEAN;FD_SETSIZE=1024;INET6;NO_CB_4_KRB5P;_CRT_SECURE_NO_WARNINGS;_WINSOCK_DEPRECATED_NO_WARNINGS;WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\sys;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <BufferSecurityCheck>false</BufferSecurityCheck>
- <StringPooling>true</StringPooling>
- @@ -253,7 +253,7 @@
- <FunctionLevelLinking>true</FunctionLevelLinking>
- <IntrinsicFunctions>true</IntrinsicFunctions>
- <PreprocessorDefinitions>WIN32_LEAN_AND_MEAN;FD_SETSIZE=1024;INET6;NO_CB_4_KRB5P;_CRT_SECURE_NO_WARNINGS;_WINSOCK_DEPRECATED_NO_WARNINGS;WIN32;_CRT_STDIO_ISO_WIDE_SPECIFIERS=1;UNICODE;_UNICODE;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
- - <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\sys;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- + <AdditionalIncludeDirectories>..\..\include;..\..\libtirpc\tirpc;..\..\dll;..\..;$(IntermediateOutputPath);%(AdditionalIncludeDirectories)</AdditionalIncludeDirectories>
- <LanguageStandard_C>stdc17</LanguageStandard_C>
- <BufferSecurityCheck>false</BufferSecurityCheck>
- <StringPooling>true</StringPooling>
- diff --git a/sys/nfs41_driver.h b/include/nfs41_driver.h
- similarity index 100%
- rename from sys/nfs41_driver.h
- rename to include/nfs41_driver.h
- --
- 2.45.1
- From f256e74641db9dc893431c7d4c2302cefac2b861 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Thu, 9 Jan 2025 11:39:49 +0100
- Subject: [PATCH 06/44] build.vc19: Reformat XML used for generating
- git_version.h in VS19's preferred style
- Reformat XML used for generating git_version.h in VS19's preferred style
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- build.vc19/nfs_mount/nfs_mount.vcxproj | 14 ++------------
- build.vc19/nfsd/nfsd.vcxproj | 14 ++------------
- 2 files changed, 4 insertions(+), 24 deletions(-)
- diff --git a/build.vc19/nfs_mount/nfs_mount.vcxproj b/build.vc19/nfs_mount/nfs_mount.vcxproj
- index 399d150..63ccec7 100644
- --- a/build.vc19/nfs_mount/nfs_mount.vcxproj
- +++ b/build.vc19/nfs_mount/nfs_mount.vcxproj
- @@ -1,20 +1,10 @@
- <?xml version="1.0" encoding="utf-8"?>
- <Project DefaultTargets="Build" ToolsVersion="14.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
- <Target Name="generate_git_version_header" BeforeTargets="ClCompile">
- - <Exec
- - Command="git describe --long --always --dirty --exclude=* --abbrev=8"
- - ConsoleToMSBuild="True"
- - IgnoreExitCode="False">
- + <Exec Command="git describe --long --always --dirty --exclude=* --abbrev=8" ConsoleToMSBuild="True" IgnoreExitCode="False">
- <Output TaskParameter="ConsoleOutput" PropertyName="git_version_string" />
- </Exec>
- -
- - <WriteLinesToFile
- - File="$(IntermediateOutputPath)/git_version.h"
- - Overwrite="True"
- - Lines="
- -/* Generated file, do not edit */
- -#define GIT_COMMIT_ID "$(git_version_string)"
- -" />
- + <WriteLinesToFile File="$(IntermediateOutputPath)/git_version.h" Overwrite="True" Lines="
/* Generated file, do not edit */
#define GIT_COMMIT_ID "$(git_version_string)"
" />
- </Target>
- <ItemGroup Label="ProjectConfigurations">
- <ProjectConfiguration Include="Debug|Win32">
- diff --git a/build.vc19/nfsd/nfsd.vcxproj b/build.vc19/nfsd/nfsd.vcxproj
- index 6c1d743..ed5470a 100644
- --- a/build.vc19/nfsd/nfsd.vcxproj
- +++ b/build.vc19/nfsd/nfsd.vcxproj
- @@ -1,20 +1,10 @@
- <?xml version="1.0" encoding="utf-8"?>
- <Project DefaultTargets="Build" ToolsVersion="14.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
- <Target Name="generate_git_version_header" BeforeTargets="ClCompile">
- - <Exec
- - Command="git describe --long --always --dirty --exclude=* --abbrev=8"
- - ConsoleToMSBuild="True"
- - IgnoreExitCode="False">
- + <Exec Command="git describe --long --always --dirty --exclude=* --abbrev=8" ConsoleToMSBuild="True" IgnoreExitCode="False">
- <Output TaskParameter="ConsoleOutput" PropertyName="git_version_string" />
- </Exec>
- -
- - <WriteLinesToFile
- - File="$(IntermediateOutputPath)/git_version.h"
- - Overwrite="True"
- - Lines="
- -/* Generated file, do not edit */
- -#define GIT_COMMIT_ID "$(git_version_string)"
- -" />
- + <WriteLinesToFile File="$(IntermediateOutputPath)/git_version.h" Overwrite="True" Lines="
/* Generated file, do not edit */
#define GIT_COMMIT_ID "$(git_version_string)"
" />
- </Target>
- <ItemGroup Label="ProjectConfigurations">
- <ProjectConfiguration Include="Debug|Win32">
- --
- 2.45.1
- From 80132695dafcbd0db0137bb185cf08692cda3ab7 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Thu, 9 Jan 2025 14:00:40 +0100
- Subject: [PATCH 07/44] tests: Fix nfs_ea widechar printf format strings
- Fix nfs_ea widechar printf format strings, e.g. use %ls for
- wide strings.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/ea/Makefile | 4 ++--
- tests/ea/main.c | 28 ++++++++++++++++++----------
- 2 files changed, 20 insertions(+), 12 deletions(-)
- diff --git a/tests/ea/Makefile b/tests/ea/Makefile
- index 9f2bc98..480449b 100644
- --- a/tests/ea/Makefile
- +++ b/tests/ea/Makefile
- @@ -7,10 +7,10 @@
- all: nfs_ea.i686.exe nfs_ea.x86_64.exe nfs_ea.exe
- nfs_ea.i686.exe: main.c
- - clang -target i686-pc-windows-gnu -municode -Wall -DUNICODE=1 -D_UNICODE=1 -isystem /usr/include/w32api/ddk -g main.c -lntdll -o nfs_ea.i686.exe
- + clang -target i686-pc-windows-gnu -std=gnu17 -municode -Wall -Wextra -D_CRT_STDIO_ISO_WIDE_SPECIFIERS=1 -DUNICODE=1 -D_UNICODE=1 -isystem /usr/include/w32api/ddk -g main.c -lntdll -o nfs_ea.i686.exe
- nfs_ea.x86_64.exe: main.c
- - clang -target x86_64-pc-windows-gnu -municode -Wall -DUNICODE=1 -D_UNICODE=1 -isystem /usr/include/w32api/ddk -g main.c -lntdll -o nfs_ea.x86_64.exe
- + clang -target x86_64-pc-windows-gnu -std=gnu17 -municode -Wall -Wextra -D_CRT_STDIO_ISO_WIDE_SPECIFIERS=1 -DUNICODE=1 -D_UNICODE=1 -isystem /usr/include/w32api/ddk -g main.c -lntdll -o nfs_ea.x86_64.exe
- nfs_ea.exe: nfs_ea.x86_64.exe
- rm -f nfs_ea.exe
- diff --git a/tests/ea/main.c b/tests/ea/main.c
- index 6a9931c..983b529 100644
- --- a/tests/ea/main.c
- +++ b/tests/ea/main.c
- @@ -20,6 +20,14 @@
- * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
- */
- +#if ((__STDC_VERSION__-0) < 201710L)
- +#error Code requires ISO C17
- +#endif
- +
- +#ifndef _CRT_STDIO_ISO_WIDE_SPECIFIERS
- +#error Code requires ISO wide-char behaviour
- +#endif /* !_CRT_STDIO_ISO_WIDE_SPECIFIERS */
- +
- #include <ntifs.h>
- #include <strsafe.h>
- #include <stdio.h>
- @@ -108,7 +116,7 @@ static NTSTATUS ea_get(
- status = RtlUnicodeToUTF8N(EaQuery->EaName, MAX_EA_VALUE,
- &ActualByteCount, EaName, EaNameLength);
- if (status) {
- - fwprintf(stderr, L"RtlUnicodeToUTF8N('%s') failed with 0x%lx\n", EaName, (long)status);
- + fwprintf(stderr, L"RtlUnicodeToUTF8N('%ls') failed with 0x%lx\n", EaName, (long)status);
- goto out;
- }
- EaQuery->EaNameLength = (UCHAR)ActualByteCount - 1;
- @@ -169,7 +177,7 @@ static NTSTATUS full_ea_init(
- FIELD_OFFSET(FILE_FULL_EA_INFORMATION, EaName),
- &ActualByteCount, EaName, EaNameLength);
- if (status) {
- - fwprintf(stderr, L"RtlUnicodeToUTF8N('%s') failed with 0x%lx\n", EaName, (long)status);
- + fwprintf(stderr, L"RtlUnicodeToUTF8N('%ls') failed with 0x%lx\n", EaName, (long)status);
- goto out;
- }
- EaBuffer->EaNameLength = (UCHAR)ActualByteCount - 1;
- @@ -184,7 +192,7 @@ static NTSTATUS full_ea_init(
- MAX_FULLEA - FIELD_OFFSET(FILE_FULL_EA_INFORMATION, EaName) - EaBuffer->EaNameLength - 1,
- &ActualByteCount, EaValue, EaValueLength);
- if (status) {
- - fwprintf(stderr, L"RtlUnicodeToUTF8N('%s') failed with 0x%lx\n", EaName, (long)status);
- + fwprintf(stderr, L"RtlUnicodeToUTF8N('%ls') failed with 0x%lx\n", EaName, (long)status);
- goto out;
- }
- EaBuffer->EaValueLength = (UCHAR)ActualByteCount - 1;
- @@ -253,7 +261,7 @@ int wmain(int argc, const wchar_t *argv[])
- status = full_ea_init(argv[3], argv[4], EaBuffer, &EaLength);
- if (status)
- goto out;
- - wprintf(L"Creating file %s.\n", argv[1]);
- + wprintf(L"Creating file '%ls'.\n", argv[1]);
- } else if (wcscmp(argv[2], L"set") == 0) {
- if (argc < 4) {
- fwprintf(stderr, L"Usage: nfs_ea <ntobjectpath> set <name> [value]\n");
- @@ -280,7 +288,7 @@ int wmain(int argc, const wchar_t *argv[])
- &IoStatusBlock, NULL, FileAttributes, ShareAccess,
- CreateDisposition, CreateOptions, EaBuffer, EaLength);
- if (status) {
- - fwprintf(stderr, L"NtCreateFile(%s) failed with 0x%lx\n", FileName.Buffer, (long)status);
- + fwprintf(stderr, L"NtCreateFile('%ls') failed with 0x%lx\n", FileName.Buffer, (long)status);
- goto out;
- }
- @@ -291,18 +299,18 @@ int wmain(int argc, const wchar_t *argv[])
- if (status)
- goto out_close;
- - wprintf(L"Setting extended attribute '%s' on file '%s':\n",
- + wprintf(L"Setting extended attribute '%ls' on file '%ls':\n",
- argv[3], FileName.Buffer);
- status = ea_set(FileHandle, EaBuffer, EaLength);
- } else if (wcscmp(argv[2], L"get") == 0) {
- - wprintf(L"Querying extended attribute on file '%s':\n",
- - argv[3], FileName.Buffer);
- + wprintf(L"Querying extended attribute on file '%ls':\n",
- + FileName.Buffer);
- status = ea_get(FileHandle, argv + 3, argc - 3);
- } else if (wcscmp(argv[2], L"list") == 0) {
- - wprintf(L"Listing extended attributes for '%s':\n", FileName.Buffer);
- + wprintf(L"Listing extended attributes for '%ls':\n", FileName.Buffer);
- status = ea_list(FileHandle);
- } else if (wcscmp(argv[2], L"create") == 0) {
- - wprintf(L"File '%s' was created with \n", FileName.Buffer);
- + wprintf(L"File '%ls' was created with \n", FileName.Buffer);
- status = ea_get(FileHandle, argv + 3, 1);
- }
- --
- 2.45.1
- From 0b0240c186cf5086b40aa8e772a0c875929f436e Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Thu, 9 Jan 2025 14:08:03 +0100
- Subject: [PATCH 08/44] tests: Add nfs_ea usage
- Add usage message to nfs_ea
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/ea/main.c | 2 ++
- 1 file changed, 2 insertions(+)
- diff --git a/tests/ea/main.c b/tests/ea/main.c
- index 983b529..cd954f8 100644
- --- a/tests/ea/main.c
- +++ b/tests/ea/main.c
- @@ -247,6 +247,8 @@ int wmain(int argc, const wchar_t *argv[])
- if (argc < 3) {
- fwprintf(stderr, L"Usage: nfs_ea <ntobjectpath> <create|set|get|list> ...\n");
- + fwprintf(stderr, L"Example:\n");
- + fwprintf(stderr, L"\tnfs_ea '\\??\\L:\\builds\\bash_build1' get NfsV3Attributes\n");
- status = STATUS_INVALID_PARAMETER;
- goto out;
- }
- --
- 2.45.1
- From d62b751b6b5791582d2fb37ee6906de339944385 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Thu, 9 Jan 2025 15:05:35 +0100
- Subject: [PATCH 09/44] tests: Add "getnfs3attr" subcmd to nfs_ea
- Add "getnfs3attr" subcmd to nfs_ea, which formats
- the contents of a "NfsV3Attributes" EA.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/ea/main.c | 93 +++++++++++++++++++++++++++++++++++++++++++++++--
- 1 file changed, 91 insertions(+), 2 deletions(-)
- diff --git a/tests/ea/main.c b/tests/ea/main.c
- index cd954f8..507bc12 100644
- --- a/tests/ea/main.c
- +++ b/tests/ea/main.c
- @@ -158,6 +158,85 @@ out:
- return status;
- }
- +typedef struct _nfs3_attrs {
- + DWORD type, mode, nlink, uid, gid, filler1;
- + LARGE_INTEGER size, used;
- + struct {
- + DWORD specdata1;
- + DWORD specdata2;
- + } rdev;
- + LONGLONG fsid, fileid;
- + LONGLONG atime, mtime, ctime;
- +} nfs3_attrs;
- +
- +static NTSTATUS ea_get_nfs3attr(
- + HANDLE FileHandle)
- +{
- + IO_STATUS_BLOCK IoStatusBlock;
- + CHAR GetBuffer[MAX_LIST_LEN] = { 0 };
- + CHAR FullBuffer[MAX_LIST_LEN] = { 0 };
- + PFILE_GET_EA_INFORMATION EaList = (PFILE_GET_EA_INFORMATION)GetBuffer, EaQuery;
- + PFILE_FULL_EA_INFORMATION EaBuffer = (PFILE_FULL_EA_INFORMATION)FullBuffer;
- + ULONG EaListLength;
- + NTSTATUS status;
- +
- + (void)memset(&IoStatusBlock, 0, sizeof(IoStatusBlock));
- +
- + EaQuery = EaList;
- + EaListLength = 0;
- +
- + (void)strcpy(EaQuery->EaName, "NfsV3Attributes");
- + EaQuery->EaNameLength = 15;
- + EaQuery->NextEntryOffset = FIELD_OFFSET(FILE_GET_EA_INFORMATION, EaName) + EaQuery->EaNameLength + 1;
- +
- + EaListLength += EaQuery->NextEntryOffset;
- + EaQuery->NextEntryOffset = 0;
- + EaQuery = (PFILE_GET_EA_INFORMATION)((PCHAR)EaQuery + EaQuery->NextEntryOffset);
- +
- + status = ZwQueryEaFile(FileHandle, &IoStatusBlock,
- + EaBuffer, MAX_FULLEA, FALSE, EaList, EaListLength, NULL, TRUE);
- + switch (status) {
- + case STATUS_SUCCESS:
- + break;
- + case STATUS_NO_EAS_ON_FILE:
- + (void)fprintf(stderr, "No EAs on file, status=0x%lx.\n", (long)status);
- + goto out;
- + default:
- + (void)fprintf(stderr, "ZwQueryEaFile('%s') failed with 0x%lx\n", EaList->EaName, (long)status);
- + goto out;
- + }
- +
- + (void)printf("%s:\n", EaBuffer->EaName);
- + nfs3_attrs *n3a = (void *)(EaBuffer->EaName + EaBuffer->EaNameLength + 1);
- + (void)printf("(\n"
- + "\ttype=%d\n"
- + "\tmode=0%o\n"
- + "\tnlink=%d\n"
- + "\tuid=%d\n\tgid=%d\n"
- + "\tsize=%lld\n\tused=%lld\n"
- + "\trdev=( specdata1=0x%x specdata2=0x%x )\n"
- + "\tfsid=%lld\n\tfileid=%lld\n"
- + "\tatime=%lld\n\tmtime=%lld\n\tctime=%lld\n"
- + ")\n",
- + (int)n3a->type,
- + (int)n3a->mode,
- + (int)n3a->nlink,
- + (int)n3a->uid,
- + (int)n3a->gid,
- + (long long)n3a->size.QuadPart,
- + (long long)n3a->used.QuadPart,
- + (int)n3a->rdev.specdata1,
- + (int)n3a->rdev.specdata2,
- + (long long)n3a->fsid,
- + (long long)n3a->fileid,
- + (long long)n3a->atime,
- + (long long)n3a->mtime,
- + (long long)n3a->ctime);
- +
- +out:
- + return status;
- +}
- +
- static NTSTATUS full_ea_init(
- IN LPCWSTR EaName,
- IN LPCWSTR EaValue,
- @@ -246,9 +325,9 @@ int wmain(int argc, const wchar_t *argv[])
- ULONG EaLength = 0;
- if (argc < 3) {
- - fwprintf(stderr, L"Usage: nfs_ea <ntobjectpath> <create|set|get|list> ...\n");
- + fwprintf(stderr, L"Usage: nfs_ea <ntobjectpath> <create|set|get|getnfs3attr|list> ...\n");
- fwprintf(stderr, L"Example:\n");
- - fwprintf(stderr, L"\tnfs_ea '\\??\\L:\\builds\\bash_build1' get NfsV3Attributes\n");
- + fwprintf(stderr, L"\tnfs_ea '\\??\\L:\\builds\\bash_build1' getnfs3attr\n");
- status = STATUS_INVALID_PARAMETER;
- goto out;
- }
- @@ -277,6 +356,12 @@ int wmain(int argc, const wchar_t *argv[])
- status = STATUS_INVALID_PARAMETER;
- goto out;
- }
- + } else if (wcscmp(argv[2], L"getnfs3attr") == 0) {
- + if (argc < 3) {
- + fwprintf(stderr, L"Usage: nfs_ea <ntobjectpath> getnfs3attr\n");
- + status = STATUS_INVALID_PARAMETER;
- + goto out;
- + }
- } else if (wcscmp(argv[2], L"list") != 0) {
- fwprintf(stderr, L"Usage: nfs_ea <ntobjectpath> <create|set|get|list> ...\n");
- status = STATUS_INVALID_PARAMETER;
- @@ -308,6 +393,10 @@ int wmain(int argc, const wchar_t *argv[])
- wprintf(L"Querying extended attribute on file '%ls':\n",
- FileName.Buffer);
- status = ea_get(FileHandle, argv + 3, argc - 3);
- + } else if (wcscmp(argv[2], L"getnfs3attr") == 0) {
- + wprintf(L"Querying extended attribute 'NfsV3Attributes' on file '%ls':\n",
- + FileName.Buffer);
- + status = ea_get_nfs3attr(FileHandle);
- } else if (wcscmp(argv[2], L"list") == 0) {
- wprintf(L"Listing extended attributes for '%ls':\n", FileName.Buffer);
- status = ea_list(FileHandle);
- --
- 2.45.1
- From 931a784fee9f017e6092d4a6f8c620f289f24071 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Thu, 9 Jan 2025 17:22:51 +0100
- Subject: [PATCH 10/44] tests: winfsinfo1: Add getnfs3attr subcmd
- winfsinfo1: Add getnfs3attr subcmd.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/winfsinfo1/Makefile | 4 +-
- tests/winfsinfo1/winfsinfo.c | 149 ++++++++++++++++++++++++++++++++++-
- 2 files changed, 150 insertions(+), 3 deletions(-)
- diff --git a/tests/winfsinfo1/Makefile b/tests/winfsinfo1/Makefile
- index 47ee557..b0b8d93 100644
- --- a/tests/winfsinfo1/Makefile
- +++ b/tests/winfsinfo1/Makefile
- @@ -7,10 +7,10 @@
- all: winfsinfo.i686.exe winfsinfo.x86_64.exe winfsinfo.exe
- winfsinfo.i686.exe: winfsinfo.c
- - clang -target i686-pc-windows-gnu -Wall -DUNICODE=1 -D_UNICODE=1 -g winfsinfo.c -o winfsinfo.i686.exe
- + clang -target i686-pc-windows-gnu -Wall -Wextra -DUNICODE=1 -D_UNICODE=1 -g winfsinfo.c -lntdll -o winfsinfo.i686.exe
- winfsinfo.x86_64.exe: winfsinfo.c
- - clang -target x86_64-pc-windows-gnu -Wall -DUNICODE=1 -D_UNICODE=1 -g winfsinfo.c -o winfsinfo.x86_64.exe
- + clang -target x86_64-pc-windows-gnu -Wall -Wextra -DUNICODE=1 -D_UNICODE=1 -g winfsinfo.c -lntdll -o winfsinfo.x86_64.exe
- winfsinfo.exe: winfsinfo.x86_64.exe
- ln -s winfsinfo.x86_64.exe winfsinfo.exe
- diff --git a/tests/winfsinfo1/winfsinfo.c b/tests/winfsinfo1/winfsinfo.c
- index 682d089..7b80ea4 100644
- --- a/tests/winfsinfo1/winfsinfo.c
- +++ b/tests/winfsinfo1/winfsinfo.c
- @@ -622,6 +622,149 @@ done:
- return res;
- }
- +typedef struct _nfs3_attrs {
- + DWORD type, mode, nlink, uid, gid, filler1;
- + LARGE_INTEGER size, used;
- + struct {
- + DWORD specdata1;
- + DWORD specdata2;
- + } rdev;
- + LONGLONG fsid, fileid;
- + LONGLONG atime, mtime, ctime;
- +} nfs3_attrs;
- +
- +#define NfsV3Attributes_NAME "NfsV3Attributes"
- +
- +typedef struct _FILE_EA_INFORMATION {
- + ULONG EaSize;
- +} FILE_EA_INFORMATION, *PFILE_EA_INFORMATION;
- +
- +typedef struct _FILE_GET_EA_INFORMATION {
- + ULONG NextEntryOffset;
- + UCHAR EaNameLength;
- + CHAR EaName[1];
- +} FILE_GET_EA_INFORMATION, *PFILE_GET_EA_INFORMATION;
- +
- +typedef struct _FILE_FULL_EA_INFORMATION {
- + ULONG NextEntryOffset;
- + UCHAR Flags;
- + UCHAR EaNameLength;
- + USHORT EaValueLength;
- + CHAR EaName[1];
- +} FILE_FULL_EA_INFORMATION, *PFILE_FULL_EA_INFORMATION;
- +
- +typedef struct _IO_STATUS_BLOCK {
- + union {
- + NTSTATUS Status;
- + PVOID Pointer;
- + } DUMMYUNIONNAME;
- + ULONG_PTR Information;
- +} IO_STATUS_BLOCK, *PIO_STATUS_BLOCK;
- +
- +#define STATUS_SUCCESS ((NTSTATUS)0x00000000)
- +#define STATUS_NO_EAS_ON_FILE ((NTSTATUS)0xC0000052)
- +
- +NTSYSAPI
- +NTSTATUS
- +NTAPI
- +ZwQueryEaFile(
- + IN HANDLE FileHandle,
- + OUT PIO_STATUS_BLOCK IoStatusBlock,
- + OUT PVOID Buffer,
- + IN ULONG Length,
- + IN BOOLEAN ReturnSingleEntry,
- + IN PVOID EaList OPTIONAL,
- + IN ULONG EaListLength,
- + IN PULONG EaIndex OPTIONAL,
- + IN BOOLEAN RestartScan);
- +
- +static
- +bool get_getnfs3attr(const char *progname, const char *filename)
- +{
- + int res = EXIT_FAILURE;
- +
- + HANDLE fileHandle = CreateFileA(filename,
- + GENERIC_READ, FILE_SHARE_READ, NULL, OPEN_EXISTING,
- + FILE_FLAG_BACKUP_SEMANTICS, NULL);
- + if (fileHandle == INVALID_HANDLE_VALUE) {
- + (void)fprintf(stderr,
- + "%s: Error opening file '%s'. Last error was %d.\n",
- + progname,
- + filename,
- + (int)GetLastError());
- + return EXIT_FAILURE;
- + }
- +
- + struct {
- + FILE_FULL_EA_INFORMATION ffeai;
- + char buf[sizeof(NfsV3Attributes_NAME) + sizeof(nfs3_attrs)];
- + } ffeai_buf;
- + struct {
- + FILE_GET_EA_INFORMATION fgeai;
- + char buf[sizeof(NfsV3Attributes_NAME)];
- + } fgeai_buf;
- +
- + NTSTATUS status;
- + IO_STATUS_BLOCK io;
- +
- + fgeai_buf.fgeai.NextEntryOffset = 0;
- + fgeai_buf.fgeai.EaNameLength = 15;
- + (void)strcpy(fgeai_buf.fgeai.EaName, NfsV3Attributes_NAME);
- +
- + status = ZwQueryEaFile(fileHandle, &io,
- + &ffeai_buf.ffeai, sizeof(ffeai_buf), TRUE,
- + &fgeai_buf.fgeai, sizeof(fgeai_buf), NULL, TRUE);
- +
- + switch (status) {
- + case STATUS_SUCCESS:
- + break;
- + case STATUS_NO_EAS_ON_FILE:
- + (void)fprintf(stderr, "No EAs on file, status=0x%lx.\n", (long)status);
- + res = EXIT_FAILURE;
- + goto done;
- + default:
- + (void)fprintf(stderr, "ZwQueryEaFile() failed with 0x%lx\n", (long)status);
- + res = EXIT_FAILURE;
- + goto done;
- + }
- +
- + nfs3_attrs *n3a = (nfs3_attrs *)(ffeai_buf.ffeai.EaName
- + + ffeai_buf.ffeai.EaNameLength + 1);
- +
- + (void)printf("(\n");
- +
- + (void)printf("\tfilename='%s'\n"
- + "\ttype=%d\n"
- + "\tmode=0%o\n"
- + "\tnlink=%d\n"
- + "\tuid=%d\n\tgid=%d\n"
- + "\tsize=%lld\n\tused=%lld\n"
- + "\trdev=( specdata1=0x%x specdata2=0x%x )\n"
- + "\tfsid=%lld\n\tfileid=%lld\n"
- + "\tatime=%lld\n\tmtime=%lld\n\tctime=%lld\n"
- + ")\n",
- + filename,
- + (int)n3a->type,
- + (int)n3a->mode,
- + (int)n3a->nlink,
- + (int)n3a->uid,
- + (int)n3a->gid,
- + (long long)n3a->size.QuadPart,
- + (long long)n3a->used.QuadPart,
- + (int)n3a->rdev.specdata1,
- + (int)n3a->rdev.specdata2,
- + (long long)n3a->fsid,
- + (long long)n3a->fileid,
- + (long long)n3a->atime,
- + (long long)n3a->mtime,
- + (long long)n3a->ctime);
- + res = EXIT_SUCCESS;
- +
- +done:
- + (void)CloseHandle(fileHandle);
- + return res;
- +}
- +
- static
- void usage(void)
- {
- @@ -633,7 +776,8 @@ void usage(void)
- "filenameinfo|"
- "filenormalizednameinfo|"
- "filecasesensitiveinfo|"
- - "getfiletime"
- + "getfiletime|"
- + "getnfs3attr"
- "> path\n");
- }
- @@ -672,6 +816,9 @@ int main(int ac, char *av[])
- else if (!strcmp(subcmd, "filecasesensitiveinfo")) {
- return get_filecasesensitiveinfo(av[0], av[2]);
- }
- + else if (!strcmp(subcmd, "getnfs3attr")) {
- + return get_getnfs3attr(av[0], av[2]);
- + }
- else {
- (void)fprintf(stderr, "%s: Unknown subcmd '%s'\n", av[0], subcmd);
- return EXIT_FAILURE;
- --
- 2.45.1
- From b1d6efedf80badcf08d485b773ad3577cde48485 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 11 Jan 2025 14:18:33 +0100
- Subject: [PATCH 11/44] Backport "update signal and key_call declarations to
- allow compile with gcc-15"
- Backport "update signal and key_call declarations to allow compile with gcc-15"
- ---- snip ----
- From 240ee6c774729c9c24812aa8912f1fcf8996b162 Mon Sep 17 00:00:00 2001
- From: Rudi Heitbaum <rudi@heitbaum.com>
- Date: Thu, 2 Jan 2025 08:46:24 -0500
- Subject: [PATCH] update signal and key_call declarations to allow compile with
- gcc-15
- Follow up patch addressing the following declarations:
- sed -n 75,77p libtirpc-1.3.6/src/key_call.c
- cryptkeyres *(*__key_encryptsession_pk_LOCAL)() = 0;
- cryptkeyres *(*__key_decryptsession_pk_LOCAL)() = 0;
- des_block *(*__key_gendes_LOCAL)() = 0;
- Signed-off-by: Rudi Heitbaum <rudi@heitbaum.com>
- Signed-off-by: Steve Dickson <steved@redhat.com>
- ---- snip ----
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- libtirpc/src/auth_time.c | 4 ++--
- libtirpc/src/key_call.c | 6 +++---
- 2 files changed, 5 insertions(+), 5 deletions(-)
- diff --git a/libtirpc/src/auth_time.c b/libtirpc/src/auth_time.c
- index 854442b..1e72ed4 100644
- --- a/libtirpc/src/auth_time.c
- +++ b/libtirpc/src/auth_time.c
- @@ -250,7 +250,7 @@ __rpc_get_time_offset(td, srv, thost, uaddr, netid)
- endpoint teps[32];
- nis_server tsrv = { 0 };
- #ifndef _WIN32
- - void (*oldsig)() = NULL; /* old alarm handler */
- + void (*oldsig)(int) = NULL; /* old alarm handler */
- #endif
- struct sockaddr_in sin;
- int s = RPC_ANYSOCK;
- @@ -420,7 +420,7 @@ __rpc_get_time_offset(td, srv, thost, uaddr, netid)
- } else {
- int res;
- #ifndef _WIN32
- - oldsig = (void (*)())signal(SIGALRM, alarm_hndler);
- + oldsig = (void (*)(int))signal(SIGALRM, alarm_hndler);
- saw_alarm = 0; /* global tracking the alarm */
- alarm(20); /* only wait 20 seconds */
- #else
- diff --git a/libtirpc/src/key_call.c b/libtirpc/src/key_call.c
- index f65fc6a..5722cb9 100644
- --- a/libtirpc/src/key_call.c
- +++ b/libtirpc/src/key_call.c
- @@ -86,9 +86,9 @@
- * implementations of these functions, and to call those in key_call().
- */
- -cryptkeyres *(*__key_encryptsession_pk_LOCAL)() = 0;
- -cryptkeyres *(*__key_decryptsession_pk_LOCAL)() = 0;
- -des_block *(*__key_gendes_LOCAL)() = 0;
- +cryptkeyres *(*__key_encryptsession_pk_LOCAL)(uid_t, char *) = 0;
- +cryptkeyres *(*__key_decryptsession_pk_LOCAL)(uid_t, char *) = 0;
- +des_block *(*__key_gendes_LOCAL)(uid_t, char *) = 0;
- static int key_call( u_long, xdrproc_t, void *, xdrproc_t, void *);
- --
- 2.45.1
- From b3747552bcf7de67a3b611844e7e006819b002e8 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 11 Jan 2025 14:38:43 +0100
- Subject: [PATCH 12/44] tests: nfsbuildtest bash build requires the "texinfo"
- package
- nfsbuildtest bash build requires the "texinfo" package
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/nfsbuildtest/nfsbuildtest.ksh93 | 1 +
- 1 file changed, 1 insertion(+)
- diff --git a/tests/nfsbuildtest/nfsbuildtest.ksh93 b/tests/nfsbuildtest/nfsbuildtest.ksh93
- index ecda2cd..bd996ab 100644
- --- a/tests/nfsbuildtest/nfsbuildtest.ksh93
- +++ b/tests/nfsbuildtest/nfsbuildtest.ksh93
- @@ -621,6 +621,7 @@ function main
- is_toolkit_pkg_installed itp 'libncurses-devel' || (( errc++ ))
- is_toolkit_pkg_installed itp 'gettext' || (( errc++ ))
- is_toolkit_pkg_installed itp 'gettext-devel' || (( errc++ ))
- + is_toolkit_pkg_installed itp 'texinfo' || (( errc++ ))
- (( errc > 0 )) && return 1
- bash_build
- return $?
- --
- 2.45.1
- From 4f09701d7365ff065230c834933ca41a20a9e2fa Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Tue, 14 Jan 2025 13:52:12 +0100
- Subject: [PATCH 13/44] cygwin: Add secureboot switch to msnfs41client.bash
- install subcmd
- Add secureboot switch to msnfs41client.bash install subcmd
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- cygwin/devel/msnfs41client.bash | 16 ++++++++++------
- 1 file changed, 10 insertions(+), 6 deletions(-)
- diff --git a/cygwin/devel/msnfs41client.bash b/cygwin/devel/msnfs41client.bash
- index cff5cc2..4699afb 100755
- --- a/cygwin/devel/msnfs41client.bash
- +++ b/cygwin/devel/msnfs41client.bash
- @@ -99,6 +99,8 @@ function nfsclient_install
- set -o xtrace
- set -o errexit
- + typeset use_secureboot=false
- +
- # switch to the location where this script is installed,
- # because on Cygwin the script will be installed
- # in /cygdrive/c/cygwin/lib/msnfs41client/ (32bit) or
- @@ -140,13 +142,15 @@ function nfsclient_install
- regtool add '/HKEY_LOCAL_MACHINE/SOFTWARE/Microsoft/Windows NT/CurrentVersion/Image File Execution Options/nfs_mount.exe'
- regtool -i set '/HKEY_LOCAL_MACHINE/SOFTWARE/Microsoft/Windows NT/CurrentVersion/Image File Execution Options/nfs_mount.exe/FrontEndHeapDebugOptions' 0x08
- - # make sure we can load the kernel driver
- - # (does not work with SecureBoot)
- - bcdedit /set testsigning on
- + if ! $use_secureboot ; then
- + # make sure we can load the kernel driver
- + # (does not work with SecureBoot)
- + bcdedit /set testsigning on
- - # enable local kernel debugging
- - bcdedit /debug on
- - bcdedit /dbgsettings local
- + # enable local kernel debugging
- + bcdedit /debug on
- + bcdedit /dbgsettings local
- + fi
- # set domain name
- typeset win_domainname=''
- --
- 2.45.1
- From 2c35e9b0af9581fdd2bc021a2f63662b19910c0e Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Tue, 14 Jan 2025 13:53:13 +0100
- Subject: [PATCH 14/44] cygwin: Add workaround for coreutils 9.5-1 /bin/cp hang
- with compressed files
- Add workaround for coreutils 9.5-1 /bin/cp hang with compressed files.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- cygwin/Makefile.install | 2 ++
- 1 file changed, 2 insertions(+)
- diff --git a/cygwin/Makefile.install b/cygwin/Makefile.install
- index 1ec9884..8a4b878 100644
- --- a/cygwin/Makefile.install
- +++ b/cygwin/Makefile.install
- @@ -58,6 +58,8 @@ installdest:
- cp $(VS_BUILD_DIR32)/../Release/nfs41_np.lib $(DESTDIR)/$(CYGWIN_BASEPATH)/lib/msnfs41client/i686/nfs41_np.lib ; \
- cp $(VS_BUILD_DIR32)/../Release/nfs41_np.pdb $(DESTDIR)/$(CYGWIN_BASEPATH)/lib/msnfs41client/i686/nfs41_np.pdb ; \
- fi
- + # workaround for coreutils 9.5-1 /bin/cp bug stuck in an endless loop with compressed files
- + chattr -V -c $(VS_BUILD_DIR)/nfs41_driver.*
- cp $(VS_BUILD_DIR)/nfs41_driver.* $(DESTDIR)/$(CYGWIN_BASEPATH)/lib/msnfs41client/.
- cp $(PROJECT_BASEDIR_DIR)/nfs41rdr.inf $(DESTDIR)/$(CYGWIN_BASEPATH)/lib/msnfs41client/.
- cp $(PROJECT_BASEDIR_DIR)/etc_netconfig $(DESTDIR)/$(CYGWIN_BASEPATH)/lib/msnfs41client/.
- --
- 2.45.1
- From e7bb534eb0e0c9f6aaaddc474d6e337a6e6a9d8b Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Tue, 14 Jan 2025 14:34:21 +0100
- Subject: [PATCH 15/44] tests: Fix nfsbuildtest bash build for gcc15
- Fix nfsbuildtest bash build for gcc15, by forcing gcc -std=gnu17
- (ISO C17 mode+GNU extensions).
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/nfsbuildtest/nfsbuildtest.ksh93 | 2 +-
- 1 file changed, 1 insertion(+), 1 deletion(-)
- diff --git a/tests/nfsbuildtest/nfsbuildtest.ksh93 b/tests/nfsbuildtest/nfsbuildtest.ksh93
- index bd996ab..85bfcc7 100644
- --- a/tests/nfsbuildtest/nfsbuildtest.ksh93
- +++ b/tests/nfsbuildtest/nfsbuildtest.ksh93
- @@ -295,7 +295,7 @@ function bash_build
- fi
- # workaround for gcc 15.0 used in Cygwin 3.6
- - export CFLAGS="-Wno-error=implicit-function-declaration -Wno-error=implicit-int"
- + export CFLAGS="-std=gnu17 -Wno-error=implicit-function-declaration -Wno-error=implicit-int"
- if $config_use_posix_ksh93_builtins ; then
- CONFIG_SHELL=/usr/bin/ksh93 ksh93 ./configure "${configure_options[@]}"
- --
- 2.45.1
- From 9348daf80483a2c618e8bfb1e6222d2362ddbea0 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Tue, 14 Jan 2025 15:45:13 +0100
- Subject: [PATCH 16/44] daemon: Cleanup Cygwin+MSYS2 silly rename code+support
- newer Cygwin
- Cleanup Cygwin+MSYS2 silly rename code, and add support
- for newer Cygwin (>= commit "Cygwin: try_to_bin: transpose deleted
- file name to valid Unicode chars").
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/setattr.c | 108 ++++++++++++++++++++++++++++++++---------------
- 1 file changed, 75 insertions(+), 33 deletions(-)
- diff --git a/daemon/setattr.c b/daemon/setattr.c
- index 79709da..1d02042 100644
- --- a/daemon/setattr.c
- +++ b/daemon/setattr.c
- @@ -237,6 +237,59 @@ static int is_dst_name_opened(nfs41_abs_path *dst_path, nfs41_session *dst_sessi
- return status;
- }
- +
- +#define CYGWIN_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE 1
- +#define MSYS2_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE 1
- +
- +#if defined(CYGWIN_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE) || \
- + defined(MSYS2_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE)
- +#define STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE_SUPPORT 1
- +#endif
- +
- +#ifdef STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE_SUPPORT
- +typedef struct _silly_rename_seq
- +{
- + const char *name;
- + size_t size;
- + const wchar_t *in_sequence;
- + const wchar_t *out_sequence;
- +} silly_rename_seq;
- +
- +const silly_rename_seq silly_rename_seqlist[] = {
- +#ifdef CYGWIN_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE
- + /* old Cygwin sequence, using valid Unicode characters */
- + {
- + .name="Cygwin1",
- + .size=4,
- + .in_sequence=L".\xdc63\xdc79\xdc67",
- + .out_sequence=L".cyg"
- + },
- + /*
- + * New Cygwin sequence, using valid Unicode characters -
- + * see Cygwin commit "Cygwin: try_to_bin: transpose
- + * deleted file name to valid Unicode chars"
- + */
- + {
- + .name="Cygwin2",
- + .size=4,
- + .in_sequence=L".\xf763\xf779\xf767",
- + .out_sequence=L".cyg"
- + },
- +#endif /* CYGWIN_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE */
- +#ifdef MSYS2_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE
- + {
- + .name="msys2",
- + .size=5,
- + .in_sequence=L".\xdc6d\xdc73\xdc79\xdc73",
- + .out_sequence=L".msys"
- + },
- +#endif /* MSYS2_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE */
- + {
- + .name = NULL
- + }
- +};
- +#endif /* STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE_SUPPORT */
- +
- static int handle_nfs41_rename(void *daemon_context, setattr_upcall_args *args)
- {
- nfs41_open_state *state = args->state;
- @@ -284,14 +337,12 @@ static int handle_nfs41_rename(void *daemon_context, setattr_upcall_args *args)
- EASSERT((rename->FileNameLength%sizeof(WCHAR)) == 0);
- -#define CYGWIN_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE 1
- -#define MSYS2_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE 1
- -
- -#ifdef CYGWIN_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE
- +#ifdef STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE_SUPPORT
- /*
- - * Stomp Cygwin "silly rename" invalid Unicode sequence
- + * Stomp old+new Cygwin+MSYS2 "silly rename" invalid Unicode
- + * sequence
- *
- - * Cygwin has it's own variation of "silly rename" (i.e. if
- + * Cygwin+MSYS2 has it's own variation of "silly rename" (i.e. if
- * someone deletes a file while someone else still has
- * a valid fd to that file it first renames that file with a
- * special prefix, see
- @@ -299,12 +350,15 @@ static int handle_nfs41_rename(void *daemon_context, setattr_upcall_args *args)
- * |try_to_bin()|).
- *
- * Unfortunately on filesystems supporting Unicode
- - * (i.e. |FILE_UNICODE_ON_DISK|) Cygwin adds the prefix
- + * (i.e. |FILE_UNICODE_ON_DISK|) older Cygwin (before Cygwin
- + * commit "Cygwin: try_to_bin: transpose deleted file name to
- + * valid Unicode chars") adds the prefix
- * L".\xdc63\xdc79\xdc67", which is NOT a valid UTF-16 sequence,
- * and will be rejected by a filesystem validating the
- * UTF-16 sequence (e.g. SAMBA, ReFS, OpenZFS, ...; for SAMBA
- * Cygwin uses the ".cyg" prefix used for
- * non-|FILE_UNICODE_ON_DISK| filesystems).
- + *
- * In our case the NFSv4.1 protocol requires valid UTF-8
- * sequences, and the NFS server will reject filenames if either
- * the server or the exported filesystem will validate the UTF-8
- @@ -315,35 +369,23 @@ static int handle_nfs41_rename(void *daemon_context, setattr_upcall_args *args)
- * used for non-|FILE_UNICODE_ON_DISK| filesystems.
- * We ignore the side-effects here, e.g. that Win32 will still
- * "remember" the original filename in the file name cache.
- - */
- - if ((rename->FileNameLength > (4*sizeof(wchar_t))) &&
- - (!memcmp(rename->FileName,
- - L".\xdc63\xdc79\xdc67", (4*sizeof(wchar_t))))) {
- - DPRINTF(1, ("handle_nfs41_rename(args->path='%s'): "
- - "Cygwin sillyrename prefix \".\\xdc63\\xdc79\\xdc67\" "
- - "detected, squishing prefix to \".cyg\"\n",
- - args->path));
- - (void)memcpy(rename->FileName, L".cyg", 4*sizeof(wchar_t));
- - }
- -#endif /* CYGWIN_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE */
- -#ifdef MSYS2_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE
- - /*
- - * Stomp MSYS2 "silly rename" invalid Unicode sequence
- *
- - * Same procedure as Cygwin "silly rename", just with a different
- - * prefix (L".\xdc6d\xdc73\xdc79\xdc73")
- + * For MSYS2+newer Cygwin we do the same.
- */
- - if ((rename->FileNameLength > (5*sizeof(wchar_t))) &&
- - (!memcmp(rename->FileName,
- - L".\xdc6d\xdc73\xdc79\xdc73", (5*sizeof(wchar_t))))) {
- - DPRINTF(1, ("handle_nfs41_rename(args->path='%s'): "
- - "msys2 sillyrename prefix "
- - "\".\\xdc6d\\xdc73\\xdc79\\xdc73\" detected, squishing "
- - "prefix to \".msys\"\n",
- - args->path));
- - (void)memcpy(rename->FileName, L".msys", 5*sizeof(wchar_t));
- + for (const silly_rename_seq *srs = &silly_rename_seqlist[0];
- + srs->name != NULL ; srs++) {
- + if ((rename->FileNameLength > (srs->size*sizeof(wchar_t))) &&
- + (!memcmp(rename->FileName,
- + srs->in_sequence, (srs->size*sizeof(wchar_t))))) {
- + DPRINTF(1, ("handle_nfs41_rename(args->path='%s'): "
- + "'%s' sillyrename prefix "
- + "detected, squishing prefix to '%ls'\n",
- + args->path, srs->name, srs->out_sequence));
- + (void)memcpy(rename->FileName, srs->out_sequence,
- + srs->size*sizeof(wchar_t));
- + }
- }
- -#endif /* MSYS2_STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE */
- +#endif /* STOMP_SILLY_RENAME_INVALID_UTF16_SEQUENCE_SUPPORT */
- dst_path.len = (unsigned short)WideCharToMultiByte(CP_UTF8,
- WC_ERR_INVALID_CHARS|WC_NO_BEST_FIT_CHARS,
- --
- 2.45.1
- From 560a81d7c555dae57d20116ff8b96bc32fad99d0 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 15 Jan 2025 14:39:37 +0100
- Subject: [PATCH 17/44] sys: Allocate memory for |nfs41_fcb_list_entry| via
- |ExAllocateFromNPagedLookasideList()|&co.
- Allocate memory for |nfs41_fcb_list_entry| via
- |ExAllocateFromNPagedLookasideList()|&co.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- sys/nfs41sys_buildconfig.h | 1 +
- sys/nfs41sys_driver.c | 44 +++++++++++++++++++++++++++++++++++---
- sys/nfs41sys_driver.h | 5 +++++
- sys/nfs41sys_openclose.c | 3 +--
- 4 files changed, 48 insertions(+), 5 deletions(-)
- diff --git a/sys/nfs41sys_buildconfig.h b/sys/nfs41sys_buildconfig.h
- index 3774af6..bc0c0d9 100644
- --- a/sys/nfs41sys_buildconfig.h
- +++ b/sys/nfs41sys_buildconfig.h
- @@ -23,6 +23,7 @@
- /* Driver build config */
- #define USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM 1
- +#define USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM 1
- // #define LOOKASIDELISTS_STATS 1
- // #define USE_ENTIRE_PATH_FOR_NETROOT 1
- diff --git a/sys/nfs41sys_driver.c b/sys/nfs41sys_driver.c
- index e2f8ed7..dd7975c 100644
- --- a/sys/nfs41sys_driver.c
- +++ b/sys/nfs41sys_driver.c
- @@ -92,6 +92,9 @@ DECLARE_CONST_ANSI_STRING(NfsActOnLink, EA_NFSACTONLINK);
- NPAGED_LOOKASIDE_LIST updowncall_entry_upcall_lookasidelist;
- NPAGED_LOOKASIDE_LIST updowncall_entry_downcall_lookasidelist;
- #endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
- +#ifdef USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM
- +NPAGED_LOOKASIDE_LIST fcblistentry_lookasidelist;
- +#endif /* USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM */
- #ifdef ENABLE_TIMINGS
- nfs41_timings lookup;
- @@ -140,6 +143,31 @@ LARGE_INTEGER unix_time_diff;
- nfs41_init_driver_state nfs41_init_state = NFS41_INIT_DRIVER_STARTABLE;
- nfs41_start_driver_state nfs41_start_state = NFS41_START_DRIVER_STARTABLE;
- +nfs41_fcb_list_entry *nfs41_allocate_nfs41_fcb_list_entry(void)
- +{
- + nfs41_fcb_list_entry *e;
- +#ifdef USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM
- + e = ExAllocateFromNPagedLookasideList(
- + &fcblistentry_lookasidelist);
- +
- +#else
- + e = RxAllocatePoolWithTag(NonPagedPoolNx,
- + sizeof(nfs41_fcb_list_entry),
- + NFS41_MM_POOLTAG_OPEN);
- +#endif /* USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM */
- +
- + return e;
- +}
- +
- +void nfs41_free_nfs41_fcb_list_entry(nfs41_fcb_list_entry *entry)
- +{
- +#ifdef USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM
- + ExFreeToNPagedLookasideList(&fcblistentry_lookasidelist,
- + entry);
- +#else
- + RxFreePool(entry);
- +#endif /* USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM */
- +}
- NTSTATUS marshall_unicode_as_utf8(
- IN OUT unsigned char **pos,
- @@ -675,7 +703,7 @@ VOID nfs41_remove_fcb_entry(
- DbgP("nfs41_remove_fcb_entry: Found match for fcb=0x%p\n", fcb);
- #endif
- RemoveEntryList(pEntry);
- - RxFreePool(cur);
- + nfs41_free_nfs41_fcb_list_entry(cur);
- break;
- }
- if (pEntry->Flink == &openlist.head) {
- @@ -908,8 +936,7 @@ void enable_caching(
- #ifdef DEBUG_TIME_BASED_COHERENCY
- DbgP("enable_caching: delegation recalled: srv_open=0x%p\n", SrvOpen);
- #endif
- - oentry = RxAllocatePoolWithTag(NonPagedPoolNx,
- - sizeof(nfs41_fcb_list_entry), NFS41_MM_POOLTAG_OPEN);
- + oentry = nfs41_allocate_nfs41_fcb_list_entry();
- if (oentry == NULL)
- goto out_release_fcblistlock;
- oentry->fcb = SrvOpen->pFcb;
- @@ -1361,6 +1388,17 @@ NTSTATUS DriverEntry(
- POOL_NX_ALLOCATION, sizeof(nfs41_updowncall_entry),
- NFS41_MM_POOLTAG_DOWN, 0);
- #endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
- +#ifdef USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM
- + /*
- + * The |Depth| parameter is unfortunately ignored in Win10,
- + * otherwise we could use |MmQuerySystemSize()| to scale the
- + * lookasidelists
- + */
- + ExInitializeNPagedLookasideList(
- + &fcblistentry_lookasidelist, NULL, NULL,
- + POOL_NX_ALLOCATION, sizeof(nfs41_fcb_list_entry),
- + NFS41_MM_POOLTAG_OPEN, 0);
- +#endif /* USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM */
- InitializeObjectAttributes(&oattrs, NULL, OBJ_KERNEL_HANDLE, NULL, NULL);
- status = PsCreateSystemThread(&dev_exts->openlistHandle, mask,
- &oattrs, NULL, NULL, &fcbopen_main, NULL);
- diff --git a/sys/nfs41sys_driver.h b/sys/nfs41sys_driver.h
- index 79ef82c..860e476 100644
- --- a/sys/nfs41sys_driver.h
- +++ b/sys/nfs41sys_driver.h
- @@ -119,6 +119,9 @@ DECLARE_EXTERN_CONST_ANSI_STRING(NfsActOnLink);
- extern NPAGED_LOOKASIDE_LIST updowncall_entry_upcall_lookasidelist;
- extern NPAGED_LOOKASIDE_LIST updowncall_entry_downcall_lookasidelist;
- #endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
- +#ifdef USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM
- +extern NPAGED_LOOKASIDE_LIST fcblistentry_lookasidelist;
- +#endif /* USE_LOOKASIDELISTS_FOR_FCBLISTENTRY_MEM */
- #ifdef ENABLE_TIMINGS
- extern nfs41_timings lookup;
- @@ -608,6 +611,8 @@ NTSTATUS nfs41_QueryDirectory(
- IN OUT PRX_CONTEXT RxContext);
- /* nfs41sys_driver.c */
- +nfs41_fcb_list_entry *nfs41_allocate_nfs41_fcb_list_entry(void);
- +void nfs41_free_nfs41_fcb_list_entry(nfs41_fcb_list_entry *entry);
- NTSTATUS marshall_unicode_as_utf8(
- IN OUT unsigned char **pos,
- IN PCUNICODE_STRING str);
- diff --git a/sys/nfs41sys_openclose.c b/sys/nfs41sys_openclose.c
- index d2624e7..59c8398 100644
- --- a/sys/nfs41sys_openclose.c
- +++ b/sys/nfs41sys_openclose.c
- @@ -911,8 +911,7 @@ retry_on_link:
- DbgP("nfs41_Create: received no delegations: srv_open=0x%p "
- "ctime=%llu\n", SrvOpen, entry->ChangeTime);
- #endif
- - oentry = RxAllocatePoolWithTag(NonPagedPoolNx,
- - sizeof(nfs41_fcb_list_entry), NFS41_MM_POOLTAG_OPEN);
- + oentry = nfs41_allocate_nfs41_fcb_list_entry();
- if (oentry == NULL) {
- status = STATUS_INSUFFICIENT_RESOURCES;
- goto out_free;
- --
- 2.45.1
- From 73fa76e699a7dac16f33c67f208350e93738baab Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 15 Jan 2025 14:44:10 +0100
- Subject: [PATCH 18/44] daemon: Remove unnecessary |ZeroMemory()| from
- |decode_readdir_entry()|
- Remove unnecessary |ZeroMemory()| from |decode_readdir_entry()|.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/nfs41_xdr.c | 7 +++++--
- 1 file changed, 5 insertions(+), 2 deletions(-)
- diff --git a/daemon/nfs41_xdr.c b/daemon/nfs41_xdr.c
- index 0004138..f899b7b 100644
- --- a/daemon/nfs41_xdr.c
- +++ b/daemon/nfs41_xdr.c
- @@ -2328,14 +2328,14 @@ static bool_t decode_readdir_entry(
- readdir_entry_iterator *it)
- {
- uint64_t cookie;
- - unsigned char name[NFS4_OPAQUE_LIMIT];
- + unsigned char name[NFS4_OPAQUE_LIMIT+1];
- unsigned char *nameptr = &name[0];
- uint32_t name_len, entry_len;
- fattr4 attrs = { 0 };
- /* decode into temporaries so we can determine if there's enough
- * room in the buffer for this entry */
- - ZeroMemory(name, NFS4_OPAQUE_LIMIT);
- + name[0] = '\0';
- name_len = NFS4_OPAQUE_LIMIT;
- entry_len = (uint32_t)FIELD_OFFSET(nfs41_readdir_entry, name);
- attrs.attr_vals_len = NFS4_OPAQUE_LIMIT;
- @@ -2356,6 +2356,9 @@ static bool_t decode_readdir_entry(
- return TRUE;
- name_len += 1; /* account for null terminator */
- + name[name_len] = '\0';
- + EASSERT(name_len < NFS4_OPAQUE_LIMIT);
- +
- if (entry_len + name_len <= it->remaining_len)
- {
- XDR fattr_xdr;
- --
- 2.45.1
- From 5a99020f85a8ee0810f4c9c19e6c3d0f59812a96 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 15 Jan 2025 16:08:54 +0100
- Subject: [PATCH 19/44] sys: Set |SRVOPEN_FLAG_NO_BUFFERING_STATE_CHANGE| if we
- only want file attributes
- Set |SRVOPEN_FLAG_NO_BUFFERING_STATE_CHANGE| if we only want file
- attributes.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- sys/nfs41sys_openclose.c | 15 +++++++++++++++
- 1 file changed, 15 insertions(+)
- diff --git a/sys/nfs41sys_openclose.c b/sys/nfs41sys_openclose.c
- index 59c8398..d6b2767 100644
- --- a/sys/nfs41sys_openclose.c
- +++ b/sys/nfs41sys_openclose.c
- @@ -342,6 +342,15 @@ static BOOLEAN isWriteOnlyDesiredAccess(PNT_CREATE_PARAMETERS params)
- return FALSE;
- }
- +static BOOLEAN isAttributeOnlyDesiredAccess(PNT_CREATE_PARAMETERS params)
- +{
- + if ((params->DesiredAccess &
- + ~(FILE_READ_ATTRIBUTES|FILE_WRITE_ATTRIBUTES|SYNCHRONIZE)) == 0) {
- + return TRUE;
- + }
- + return FALSE;
- +}
- +
- static BOOLEAN areOpenParamsValid(NT_CREATE_PARAMETERS *params)
- {
- /* from ms-fsa page 52 */
- @@ -856,6 +865,12 @@ retry_on_link:
- #endif
- RxChangeBufferingState((PSRV_OPEN)SrvOpen, ULongToPtr(flag), 1);
- }
- +
- + if (!nfs41_fcb->StandardInfo.Directory &&
- + isAttributeOnlyDesiredAccess(params)) {
- + SrvOpen->Flags |= SRVOPEN_FLAG_NO_BUFFERING_STATE_CHANGE;
- + }
- +
- if (!nfs41_fcb->StandardInfo.Directory &&
- isDataAccess(params->DesiredAccess)) {
- nfs41_fobx->deleg_type = entry->u.Open.deleg_type;
- --
- 2.45.1
- From 106ecd4283f9a1274dc24bbc14d0bf7885ee6ce9 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Thu, 16 Jan 2025 15:41:49 +0100
- Subject: [PATCH 20/44] daemon: |bitmap4_cpy()| should only write |dst| fields
- which are used by the |src|
- |bitmap4_cpy()| should only write |dst| fields which are used by the
- |src|.
- That should fix drmemory hits.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/util.h | 6 +++++-
- 1 file changed, 5 insertions(+), 1 deletion(-)
- diff --git a/daemon/util.h b/daemon/util.h
- index 7a4e12a..d17dc45 100644
- --- a/daemon/util.h
- +++ b/daemon/util.h
- @@ -131,7 +131,11 @@ static __inline void bitmap4_cpy(
- OUT bitmap4 *restrict dst,
- IN const bitmap4 *restrict src)
- {
- - (void)memcpy(dst, src, sizeof(bitmap4));
- + uint32_t i;
- + for (i = 0; i < src->count; i++) {
- + dst->arr[i] = src->arr[i];
- + }
- + dst->count = src->count;
- }
- static __inline void bitmap4_clear(
- --
- 2.45.1
- From f13f053b83400d0b09e4fbb922a6cccdddd5112d Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Thu, 16 Jan 2025 15:47:16 +0100
- Subject: [PATCH 21/44] daemon: Fix drmemory hits in |nfs41_file_info_cpy()|
- Fix drmemory hits in |nfs41_file_info_cpy()|, which were caused
- by using |memcpy()| over only partially initialised
- |nfs41_file_info| structures.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/delegation.c | 2 +-
- daemon/fileinfoutil.c | 146 ++++++++++++++++++++++++++++++++++++++----
- daemon/lookup.c | 4 +-
- daemon/util.h | 6 +-
- 4 files changed, 143 insertions(+), 15 deletions(-)
- diff --git a/daemon/delegation.c b/daemon/delegation.c
- index 97d63f5..22f1009 100644
- --- a/daemon/delegation.c
- +++ b/daemon/delegation.c
- @@ -515,7 +515,7 @@ int nfs41_delegate_open(
- if (create == OPEN4_CREATE) {
- EASSERT(createattrs != NULL);
- - nfs41_file_info_cpy(info, createattrs);
- + nfs41_file_info_cpy(info, createattrs, 0);
- /* write delegations allow us to simulate OPEN4_CREATE with SETATTR */
- status = delegation_truncate(deleg, client, &stateid, info);
- diff --git a/daemon/fileinfoutil.c b/daemon/fileinfoutil.c
- index a5269fe..8278a90 100644
- --- a/daemon/fileinfoutil.c
- +++ b/daemon/fileinfoutil.c
- @@ -445,16 +445,140 @@ void nfs_to_stat_lx_info(
- /* copy |nfs41_file_info| */
- void nfs41_file_info_cpy(
- OUT nfs41_file_info *dest,
- - IN const nfs41_file_info *src)
- + IN const nfs41_file_info *src,
- + IN int flags)
- {
- - /*
- - * FIXME: Using |memcpy()| here over the whole struct
- - * |nfs41_file_info| will trigger DrMemory uninitialized
- - * variable hits if |*src| was not completely initialized
- - */
- - (void)memcpy(dest, src, sizeof(nfs41_file_info));
- - if (src->owner != NULL)
- - dest->owner = dest->owner_buf;
- - if (src->owner_group != NULL)
- - dest->owner_group = dest->owner_group_buf;
- + const bitmap4 *attrmask = &src->attrmask;
- + bitmap4_cpy(&dest->attrmask, &src->attrmask);
- +
- + if (attrmask->count > 0) {
- + if (attrmask->arr[0] & FATTR4_WORD0_SUPPORTED_ATTRS) {
- + dest->supported_attrs = src->supported_attrs;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_TYPE) {
- + dest->type = src->type;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_CHANGE) {
- + dest->change = src->change;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_SIZE) {
- + dest->size = src->size;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_LINK_SUPPORT) {
- + dest->link_support = src->link_support;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_SYMLINK_SUPPORT) {
- + dest->symlink_support = src->symlink_support;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_FSID) {
- + dest->fsid = src->fsid;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_LEASE_TIME) {
- + dest->lease_time = src->lease_time;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_RDATTR_ERROR) {
- + dest->rdattr_error = src->rdattr_error;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_ACL) {
- + /* fixme: we should copy the contents! */
- + dest->acl = src->acl;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_ACLSUPPORT) {
- + dest->aclsupport = src->aclsupport;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_ARCHIVE) {
- + dest->archive = src->archive;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_CANSETTIME) {
- + dest->cansettime = src->cansettime;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_CASE_INSENSITIVE) {
- + dest->case_insensitive = src->case_insensitive;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_CASE_PRESERVING) {
- + dest->case_preserving = src->case_preserving;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_FILEID) {
- + dest->fileid = src->fileid;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_FS_LOCATIONS) {
- + /* fixme: we should copy the contents, not the pointer! */
- + dest->fs_locations = src->fs_locations;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_HIDDEN) {
- + dest->hidden = src->hidden;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_MAXREAD) {
- + dest->maxread = src->maxread;
- + }
- + if (attrmask->arr[0] & FATTR4_WORD0_MAXWRITE) {
- + dest->maxwrite = src->maxwrite;
- + }
- + }
- + if (attrmask->count > 1) {
- + if (attrmask->arr[1] & FATTR4_WORD1_MODE) {
- + dest->mode = src->mode;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_NUMLINKS) {
- + dest->numlinks = src->numlinks;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_OWNER) {
- + EASSERT(src->owner != NULL);
- + EASSERT(src->owner[0] != '\0');
- + dest->owner = dest->owner_buf;
- + (void)strcpy(dest->owner, src->owner);
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_OWNER_GROUP) {
- + EASSERT(src->owner_group != NULL);
- + EASSERT(src->owner_group[0] != '\0');
- + dest->owner_group = dest->owner_group_buf;
- + (void)strcpy(dest->owner_group_buf, src->owner_group_buf);
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_SPACE_AVAIL) {
- + dest->space_avail = src->space_avail;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_SPACE_FREE) {
- + dest->space_free = src->space_free;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_SPACE_TOTAL) {
- + dest->space_total = src->space_total;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_SYSTEM) {
- + dest->system = src->system;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_TIME_ACCESS) {
- + dest->time_access = src->time_access;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_TIME_CREATE) {
- + dest->time_create = src->time_create;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_TIME_DELTA) {
- + dest->time_delta = src->time_delta;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_TIME_MODIFY) {
- + dest->time_modify = src->time_modify;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_DACL) {
- + /* fixme: we should copy the contents! */
- + dest->acl = src->acl;
- + }
- + if (attrmask->arr[1] & FATTR4_WORD1_FS_LAYOUT_TYPE) {
- + dest->fs_layout_types = src->fs_layout_types;
- + }
- + }
- + if (attrmask->count > 2) {
- + if (attrmask->arr[2] & FATTR4_WORD2_MODE_SET_MASKED) {
- + dest->mode_mask = src->mode_mask;
- + }
- + if (attrmask->arr[2] & FATTR4_WORD2_MDSTHRESHOLD) {
- + dest->mdsthreshold = src->mdsthreshold;
- + }
- + if (attrmask->arr[2] & FATTR4_WORD2_SUPPATTR_EXCLCREAT) {
- + dest->suppattr_exclcreat = src->suppattr_exclcreat;
- + }
- + }
- +
- + if (flags & NFS41FILEINFOCPY_COPY_SYMLINK_DIR) {
- + dest->symlink_dir = src->symlink_dir;
- + }
- }
- diff --git a/daemon/lookup.c b/daemon/lookup.c
- index a27e923..f98edcc 100644
- --- a/daemon/lookup.c
- +++ b/daemon/lookup.c
- @@ -236,7 +236,7 @@ static int server_lookup(
- if (target_out)
- *target_out = dir;
- if (info_out) {
- - nfs41_file_info_cpy(info_out, res->getrootattr.info);
- + nfs41_file_info_cpy(info_out, res->getrootattr.info, 0);
- }
- } else if (count == 1) {
- if (parent_out)
- @@ -285,7 +285,7 @@ static int server_lookup(
- if (target_out)
- *target_out = file;
- if (info_out) {
- - nfs41_file_info_cpy(info_out, res->getattr[i].info);
- + nfs41_file_info_cpy(info_out, res->getattr[i].info, 0);
- }
- } else if (i == count-2) {
- if (parent_out)
- diff --git a/daemon/util.h b/daemon/util.h
- index d17dc45..8460c2a 100644
- --- a/daemon/util.h
- +++ b/daemon/util.h
- @@ -205,9 +205,13 @@ void nfs_to_stat_lx_info(
- IN const nfs41_file_info *info,
- OUT PFILE_STAT_LX_INFORMATION stat_lx_out);
- #endif /* NFS41_DRIVER_WSL_SUPPORT */
- +
- +/* Copy |info->symlink_dir| */
- +#define NFS41FILEINFOCPY_COPY_SYMLINK_DIR (1 << 0)
- void nfs41_file_info_cpy(
- OUT nfs41_file_info *dest,
- - IN const nfs41_file_info *src);
- + IN const nfs41_file_info *src,
- + IN int flags);
- /* http://msdn.microsoft.com/en-us/library/ms724290%28VS.85%29.aspx:
- * A file time is a 64-bit value that represents the number of
- --
- 2.45.1
- From fdc35492fceffcd005c8cd50c86e7341ba16d811 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Thu, 16 Jan 2025 17:49:32 +0100
- Subject: [PATCH 22/44] tests: Document how to set up a Solaris 11.4 NFSv4.1
- nfsd
- Document how to set up a Solaris 11.4 NFSv4.1 nfsd.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/nfs_server_setup.txt | 27 +++++++++++++++++++++++++++
- 1 file changed, 27 insertions(+)
- diff --git a/tests/nfs_server_setup.txt b/tests/nfs_server_setup.txt
- index 9f3a2a3..3577dd7 100644
- --- a/tests/nfs_server_setup.txt
- +++ b/tests/nfs_server_setup.txt
- @@ -40,4 +40,31 @@ Start-Service -Name ms-nfs41-client-service
- C:\cygwin64\sbin\nfs_mount -o rw N nfs://192.168.209.129//
- +
- +#
- +# Solaris 11.4 NFSv4.1 server setup
- +#
- +
- +# 1. Server setup
- +svcadm enable network/nfs/server
- +sharectl set -p nfsmapid_domain=global.loc nfs
- +sharectl set -p server_delegation=on nfs
- +mkdir /nfsdata
- +chmod a+rwx /nfsdata
- +share -F nfs -o rw /nfsdata/
- +svcs svc:/network/nfs/server:default
- +
- +# 2. Windows ms-nfs41-client setup:
- +# Add entries for groups "sys" and "nobody"
- +echo "sys:S-1-0-3:3:" >>/etc/groups
- +echo "nobody:S-1-0-65534:65534:" >>/etc/group
- +
- +# 3. Misc commands:
- +ls -v filename # list ACLs
- +chmod A... # to edit ACLs
- +
- +# 4. Troubleshooting:
- +See https://docs.oracle.com/en/operating-systems/solaris/oracle-solaris/11.4/manage-nfs/troubleshooting-network-file-systems1.html
- +
- +
- # EOF.
- --
- 2.45.1
- From f10c221750d6345f68537ab933c69db0881f84ff Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Fri, 17 Jan 2025 15:29:01 +0100
- Subject: [PATCH 23/44] tests: Fix nfsbuildtest.ksh93 msnfs41client build
- Fix nfsbuildtest.ksh93 msnfs41client build, which stopped working
- after adding the --version support (commit
- "build.vc19,daemon,mount: Add -V/--version options to
- nfs_mount+nfsd, including commit id").
- Reported-by: Cedric Blancher <cedric.blancher@gmail.com>
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/nfsbuildtest/nfsbuildtest.ksh93 | 14 ++++++++++++--
- 1 file changed, 12 insertions(+), 2 deletions(-)
- diff --git a/tests/nfsbuildtest/nfsbuildtest.ksh93 b/tests/nfsbuildtest/nfsbuildtest.ksh93
- index 85bfcc7..e86a1f1 100644
- --- a/tests/nfsbuildtest/nfsbuildtest.ksh93
- +++ b/tests/nfsbuildtest/nfsbuildtest.ksh93
- @@ -430,6 +430,9 @@ function msnfs41client_build
- cd "$PWD/ms-nfs41-client/"
- + # make sure git commands (like "git describe ...") work
- + git config --global --add safe.directory "$PWD"
- +
- #
- # patch sources and configure build
- #
- @@ -446,8 +449,15 @@ function msnfs41client_build
- #
- # build ms-nfs41-client
- #
- - export PATH+=":/cygdrive/c/Program Files (x86)/Microsoft Visual Studio/2019/Community/MSBuild/Current/Bin/"
- - time make -j1 -f cygwin/Makefile bintarball
- + (
- + # Visual Studio 19 64bit+32bit kernel codepath
- + # fixme: Add support for VS2022 64bit-only codepath
- + export PATH+=":/cygdrive/c/Program Files (x86)/Microsoft Visual Studio/2019/Community/MSBuild/Current/Bin/"
- + set -o errexit
- + time make -j1 -f cygwin/Makefile build
- + time make -j1 -f cygwin/Makefile installdest
- + time make -j1 -f cygwin/Makefile bintarball
- + )
- echo $?
- echo "#Done."
- --
- 2.45.1
- From 3e9870c8c98fba0ec164c93a8a1a8ef64ad0c439 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Fri, 17 Jan 2025 15:32:00 +0100
- Subject: [PATCH 24/44] daemon: Increase |NFS4_EASIZE| to deal with larger EAs
- Increase |NFS4_EASIZE| to deal with larger EAs, like
- Solaris "SUNWattr_ro", "SUNWattr_rw" etc.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/nfs41_const.h | 2 +-
- 1 file changed, 1 insertion(+), 1 deletion(-)
- diff --git a/daemon/nfs41_const.h b/daemon/nfs41_const.h
- index a4e408c..44209d2 100644
- --- a/daemon/nfs41_const.h
- +++ b/daemon/nfs41_const.h
- @@ -32,7 +32,7 @@
- #define NFS4_OPAQUE_LIMIT 1024
- #define NFS4_SESSIONID_SIZE 16
- #define NFS4_STATEID_OTHER 12
- -#define NFS4_EASIZE 256
- +#define NFS4_EASIZE 2048
- #define NFS4_EANAME_SIZE 128
- /*
- --
- 2.45.1
- From f75dd94b8621cb8d45591383e085873efc51bd87 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 18 Jan 2025 15:35:49 +0100
- Subject: [PATCH 25/44] daemon: Fix DrMemory hit in |nfs41_setattr()|
- Fix DrMemory hit in |nfs41_setattr()|:
- ---- snip ----
- Error #3: UNINITIALIZED READ: reading 4 byte(s)
- 0 nfs41_setattr [daemon\nfs41_ops.c:1514]
- 1 handle_setacl [daemon\acl.c:1468]
- 2 upcall_handle [daemon\upcall.c:220]
- 3 nfsd_worker_thread_main [daemon\nfs41_daemon.c:207]
- 4 nfsd_thread_main [daemon\nfs41_daemon.c:245]
- ---- snip ----
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/nfs41_ops.c | 3 ++-
- 1 file changed, 2 insertions(+), 1 deletion(-)
- diff --git a/daemon/nfs41_ops.c b/daemon/nfs41_ops.c
- index c49cca0..a4e059d 100644
- --- a/daemon/nfs41_ops.c
- +++ b/daemon/nfs41_ops.c
- @@ -1510,7 +1510,8 @@ int nfs41_setattr(
- nfs41_attr_cache_update(session_name_cache(session),
- file->fh.fileid, info);
- - if (setattr_res.attrsset.arr[0] & FATTR4_WORD0_SIZE)
- + if ((setattr_res.attrsset.count > 0) &&
- + (setattr_res.attrsset.arr[0] & FATTR4_WORD0_SIZE))
- nfs41_superblock_space_changed(file->fh.superblock);
- out:
- return status;
- --
- 2.45.1
- From 36b51c29b81a310114fd165871a3388c64cc7f79 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 18 Jan 2025 15:38:08 +0100
- Subject: [PATCH 26/44] daemon: |idmap_lookup_user()|+|idmap_lookup_group()|
- should not insert invalid data into cache
- idmapper |idmap_lookup_user()|+|idmap_lookup_group()| should not
- insert invalid data into cache on lookup failure.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/idmap.c | 8 ++++----
- 1 file changed, 4 insertions(+), 4 deletions(-)
- diff --git a/daemon/idmap.c b/daemon/idmap.c
- index 4f68dd0..37b2889 100644
- --- a/daemon/idmap.c
- +++ b/daemon/idmap.c
- @@ -813,15 +813,15 @@ static int idmap_lookup_user(
- (unsigned int)user->gid));
- }
- #endif /* !NFS41_DRIVER_FEATURE_IDMAPPER_CYGWIN */
- - if (context->config.cache_ttl) {
- + if ((status == 0) && context->config.cache_ttl) {
- /* insert the entry into the cache */
- cache_insert(&context->users, lookup, &user->entry);
- }
- #ifndef NFS41_DRIVER_FEATURE_IDMAPPER_CYGWIN
- out_free_values:
- -#endif
- for (i = 0; i < NUM_ATTRIBUTES; i++)
- ldap_value_freeA(values[i]);
- +#endif
- out:
- return status;
- }
- @@ -918,15 +918,15 @@ static int idmap_lookup_group(
- (unsigned int)group->gid));
- }
- #endif /* !NFS41_DRIVER_FEATURE_IDMAPPER_CYGWIN */
- - if (context->config.cache_ttl) {
- + if ((status == 0) && context->config.cache_ttl) {
- /* insert the entry into the cache */
- cache_insert(&context->groups, lookup, &group->entry);
- }
- #ifndef NFS41_DRIVER_FEATURE_IDMAPPER_CYGWIN
- out_free_values:
- -#endif
- for (i = 0; i < NUM_ATTRIBUTES; i++)
- ldap_value_freeA(values[i]);
- +#endif
- out:
- return status;
- }
- --
- 2.45.1
- From 2420813a9a59520723a4d67256f5c2e4980fca4f Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 18 Jan 2025 16:00:28 +0100
- Subject: [PATCH 27/44] daemon: |idmap_lookup_user()|+|idmap_lookup_group()|
- should not insert invalid data into cache [fix compiler warnings]
- Fix "Unused variables" compiler warnings triggered by commit
- "|idmap_lookup_user()|+|idmap_lookup_group()| should not insert
- invalid data into cache".
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/idmap.c | 10 ++++++----
- 1 file changed, 6 insertions(+), 4 deletions(-)
- diff --git a/daemon/idmap.c b/daemon/idmap.c
- index 37b2889..7f090f8 100644
- --- a/daemon/idmap.c
- +++ b/daemon/idmap.c
- @@ -660,16 +660,17 @@ static int idmap_lookup_user(
- const struct idmap_lookup *lookup,
- struct idmap_user *user)
- {
- - PCHAR* values[NUM_ATTRIBUTES] = { NULL };
- #ifndef NFS41_DRIVER_FEATURE_IDMAPPER_CYGWIN
- + PCHAR* values[NUM_ATTRIBUTES] = { NULL };
- const unsigned attributes = ATTR_FLAG(ATTR_USER_NAME)
- | ATTR_FLAG(ATTR_PRINCIPAL)
- | ATTR_FLAG(ATTR_UID)
- | ATTR_FLAG(ATTR_GID);
- /* principal is optional; we'll cache it if we have it */
- const unsigned optional = ATTR_FLAG(ATTR_PRINCIPAL);
- + int i;
- #endif /* !NFS41_DRIVER_FEATURE_IDMAPPER_CYGWIN */
- - int i, status;
- + int status;
- /* check the user cache for an existing entry */
- status = cache_lookup(&context->users, lookup, &user->entry);
- @@ -831,12 +832,13 @@ static int idmap_lookup_group(
- const struct idmap_lookup *lookup,
- struct idmap_group *group)
- {
- - PCHAR* values[NUM_ATTRIBUTES] = { NULL };
- #ifndef NFS41_DRIVER_FEATURE_IDMAPPER_CYGWIN
- + PCHAR* values[NUM_ATTRIBUTES] = { NULL };
- const unsigned attributes = ATTR_FLAG(ATTR_GROUP_NAME)
- | ATTR_FLAG(ATTR_GID);
- + int i;
- #endif
- - int i, status;
- + int status;
- /* check the group cache for an existing entry */
- status = cache_lookup(&context->groups, lookup, &group->entry);
- --
- 2.45.1
- From 15d1de4ee1e7ef4b17e1c5f0137cbbc60716ff53 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Sat, 18 Jan 2025 17:04:09 +0100
- Subject: [PATCH 28/44] daemon: Set list of AUP_GIDs for |AUTH_UNIX|
- Set list of AUP_GIDs for |AUTH_UNIX|, based on the list of groups
- in the impersonation token's |TOKEN_GROUPS| data.
- This fixes use of Cygwin /bin/chgrp with Solaris 11.4 NFSv4.1 nfsd,
- which correctly checks any NFSv4 SETATTR request with an
- |FATTR4_WORD1_OWNER_GROUP| attribute against the RPC's AUP_GID list.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/accesstoken.c | 146 ++++++++++++++++++++++++++++++++++++++++
- daemon/accesstoken.h | 6 ++
- daemon/nfs41_compound.c | 19 +++++-
- daemon/nfs41_const.h | 3 +
- daemon/nfs41_rpc.c | 19 +++++-
- 5 files changed, 188 insertions(+), 5 deletions(-)
- diff --git a/daemon/accesstoken.c b/daemon/accesstoken.c
- index c212c91..e9a8d14 100644
- --- a/daemon/accesstoken.c
- +++ b/daemon/accesstoken.c
- @@ -23,6 +23,7 @@
- #include "accesstoken.h"
- #include "sid.h"
- #include "daemon_debug.h"
- +#include "nfs41_daemon.h"
- #include <Lmcons.h>
- #ifndef _NFS41_DRIVER_BUILDFEATURES_
- @@ -130,6 +131,151 @@ bool get_token_primarygroup_name(HANDLE tok, char *out_buffer)
- return true;
- }
- +bool fill_auth_unix_aup_gids(HANDLE tok,
- + gid_t *aup_gids, int *num_aup_gids)
- +{
- + char group_names_buff[RPC_AUTHUNIX_AUP_MAX_NUM_GIDS*(GNLEN+1)];
- + char *group_names[RPC_AUTHUNIX_AUP_MAX_NUM_GIDS];
- + char *s;
- + int i;
- + int num_groups;
- +
- + /* fixme: This should be a function argument */
- + extern nfs41_daemon_globals nfs41_dg;
- +
- + /*
- + * VS2019 |_alloca()| cannot be used in a loop, so we use multiple
- + * pointers into one buffer instead
- + */
- + for (s=group_names_buff,i=0 ; i < RPC_AUTHUNIX_AUP_MAX_NUM_GIDS ; i++) {
- + group_names[i] = s;
- + s += GNLEN+1;
- + }
- +
- + if (!get_token_groups_names(tok,
- + RPC_AUTHUNIX_AUP_MAX_NUM_GIDS, group_names, &num_groups)) {
- + eprintf("fill_auth_unix_aup_gids: "
- + "get_token_groups_names() failed\n");
- + *num_aup_gids = 0;
- + return false;
- + }
- +
- + gid_t map_gid;
- + *num_aup_gids = 0;
- +
- + for (i=0 ; i < num_groups ; i++) {
- + if (nfs41_idmap_group_to_gid(
- + nfs41_dg.idmapper,
- + group_names[i],
- + &map_gid) == 0) {
- + aup_gids[(*num_aup_gids)++] = map_gid;
- + }
- + else {
- + eprintf("fill_auth_unix_aup_gids: "
- + "no group mapping for '%s'\n",
- + group_names[i]);
- + }
- + }
- +
- + return true;
- +}
- +
- +bool get_token_groups_names(HANDLE tok,
- + int num_out_buffers, char *out_buffers[],
- + int *out_buffers_count)
- +{
- + DWORD tokdatalen;
- + PTOKEN_GROUPS ptgroups;
- + char namebuffer[GNLEN+1];
- + DWORD namesize;
- + char domainbuffer[UNLEN+1];
- + DWORD domainbuffer_size;
- + SID_NAME_USE name_use;
- + bool retval = false;
- +
- + DPRINTF(1,
- + ("--> get_token_groups_names"
- + "(tok=0x%p,num_out_buffers=%d,out_buffers=0x%p)\n",
- + (void *)tok, num_out_buffers, out_buffers));
- +
- + tokdatalen = sizeof(TOKEN_GROUPS)+GETTOKINFO_EXTRA_BUFFER;
- + ptgroups = _alloca(tokdatalen);
- + if (!GetTokenInformation(tok, TokenGroups, ptgroups,
- + tokdatalen, &tokdatalen)) {
- + DPRINTF(0, ("get_token_groups_names: "
- + "GetTokenInformation(tok=0x%p, TokenGroups) failed, "
- + "status=%d.\n",
- + (void *)tok, (int)GetLastError()));
- + retval = false;
- + goto done;
- + }
- +
- + DWORD i;
- + int iob = 0; /* index in |out_buffers| */
- +
- + DPRINTF(1, ("get_token_groups_names: got %d groups\n",
- + (int)ptgroups->GroupCount));
- +
- + for (i = 0 ; i < ptgroups->GroupCount ; i++) {
- + if (!(ptgroups->Groups[i].Attributes & SE_GROUP_ENABLED)) {
- + continue;
- + }
- +
- + namesize = sizeof(namebuffer)-1;
- + domainbuffer_size = sizeof(domainbuffer)-1;
- +
- + if (!LookupAccountSidA(NULL, ptgroups->Groups[i].Sid,
- + namebuffer, &namesize, domainbuffer, &domainbuffer_size, &name_use)) {
- + DPRINTF(0, ("get_token_groups_names: "
- + "LookupAccountSidA() failed, status=%d.\n",
- + (int)GetLastError()));
- + continue;
- + }
- +
- + if (iob < num_out_buffers) {
- + DPRINTF(1,
- + ("get_token_groups_names: adding group='%s', domain='%s'\n",
- + namebuffer, domainbuffer));
- + (void)strcpy(out_buffers[iob], namebuffer);
- + iob++;
- + }
- + else {
- + DPRINTF(0,
- + ("get_token_groups_names: "
- + "buffer full, skip group='%s', domain='%s'\n",
- + namebuffer, domainbuffer));
- + }
- + }
- +
- + *out_buffers_count = iob;
- + retval = true;
- +
- +done:
- + if (retval) {
- + /* success */
- + DPRINTF(1,
- + ("<-- get_token_groups_names"
- + "(tok=0x%p,num_out_buffers=%d,out_buffers=0x%p,*out_buffers_count=%d), retval=%d\n",
- + (void *)tok,
- + num_out_buffers,
- + out_buffers,
- + *out_buffers_count,
- + (int)retval));
- + }
- + else {
- + /* failure */
- + DPRINTF(1,
- + ("<-- get_token_groups_names"
- + "(tok=0x%p,num_out_buffers=%d,out_buffers=0x%p) failed, retval=%d\n",
- + (void *)tok,
- + num_out_buffers,
- + out_buffers,
- + (int)retval));
- + }
- +
- + return retval;
- +}
- +
- bool get_token_authenticationid(HANDLE tok, LUID *out_authenticationid)
- {
- DWORD tokdatalen;
- diff --git a/daemon/accesstoken.h b/daemon/accesstoken.h
- index 67f49cd..3208528 100644
- --- a/daemon/accesstoken.h
- +++ b/daemon/accesstoken.h
- @@ -24,10 +24,16 @@
- #include <Windows.h>
- #include <stdbool.h>
- +#include "nfs41_types.h" /* for |gid_t| */
- bool get_token_user_name(HANDLE tok, char *out_buffer);
- bool get_token_primarygroup_name(HANDLE tok, char *out_buffer);
- bool get_token_authenticationid(HANDLE tok, LUID *out_authenticationid);
- bool set_token_privilege(HANDLE tok, const char *seprivname, bool enable_priv);
- +bool fill_auth_unix_aup_gids(HANDLE tok,
- + gid_t *, int *num_aup_gids);
- +bool get_token_groups_names(HANDLE tok,
- + int num_out_buffers, char *out_buffers[],
- + int *out_buffers_count);
- #endif /* !__NFS41_DAEMON_ACCESSTOKEN_H__ */
- diff --git a/daemon/nfs41_compound.c b/daemon/nfs41_compound.c
- index cb35663..8b1f192 100644
- --- a/daemon/nfs41_compound.c
- +++ b/daemon/nfs41_compound.c
- @@ -3,6 +3,7 @@
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- + * Roland Mainz <roland.mainz@nrubsig.org>
- *
- * This library is free software; you can redistribute it and/or modify it
- * under the terms of the GNU Lesser General Public License as published by
- @@ -22,6 +23,7 @@
- #include <stdio.h>
- #include <stdlib.h>
- +#include "accesstoken.h"
- #include "nfs41_compound.h"
- #include "nfs41_xdr.h"
- #include "nfs41_ops.h"
- @@ -111,14 +113,25 @@ static int create_new_rpc_auth(nfs41_session *session, uint32_t op,
- sec_flavor = secinfo[i].type;
- } else {
- char machname[MAXHOSTNAMELEN + 1];
- - gid_t gids[1];
- + gid_t aup_gids[RPC_AUTHUNIX_AUP_MAX_NUM_GIDS];
- + int num_aup_gids = 0;
- +
- + if (!fill_auth_unix_aup_gids(GetCurrentThreadToken(),
- + aup_gids, &num_aup_gids)) {
- + eprintf("create_new_rpc_auth: "
- + "fill_auth_unix_aup_gids() failed\n");
- + continue;
- + }
- +
- if (gethostname(machname, sizeof(machname)) == -1) {
- eprintf("nfs41_rpc_clnt_create: gethostname failed\n");
- continue;
- }
- machname[sizeof(machname) - 1] = '\0';
- - auth = authsys_create(machname, session->client->rpc->uid,
- - session->client->rpc->gid, 0, gids);
- + auth = authsys_create(machname,
- + session->client->rpc->uid,
- + session->client->rpc->gid,
- + num_aup_gids, aup_gids);
- if (auth == NULL) {
- eprintf("handle_wrongsecinfo_noname: authsys_create failed\n");
- continue;
- diff --git a/daemon/nfs41_const.h b/daemon/nfs41_const.h
- index 44209d2..c785df6 100644
- --- a/daemon/nfs41_const.h
- +++ b/daemon/nfs41_const.h
- @@ -35,6 +35,9 @@
- #define NFS4_EASIZE 2048
- #define NFS4_EANAME_SIZE 128
- +/* Maximum number of AUP GIDs for |AUTH_UNIX| */
- +#define RPC_AUTHUNIX_AUP_MAX_NUM_GIDS 16
- +
- /*
- * |NFS4_FATTR4_OWNER_LIMIT| - limits for
- * |fattr4_owner|+|fattr4_owner_group|
- diff --git a/daemon/nfs41_rpc.c b/daemon/nfs41_rpc.c
- index f92f313..45cb3bb 100644
- --- a/daemon/nfs41_rpc.c
- +++ b/daemon/nfs41_rpc.c
- @@ -3,6 +3,7 @@
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- + * Roland Mainz <roland.mainz@nrubsig.org>
- *
- * This library is free software; you can redistribute it and/or modify it
- * under the terms of the GNU Lesser General Public License as published by
- @@ -19,6 +20,7 @@
- * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
- */
- +#include "accesstoken.h"
- #include "nfs41_ops.h"
- #include "daemon_debug.h"
- #include "nfs41_xdr.h"
- @@ -157,7 +159,6 @@ int nfs41_rpc_clnt_create(
- uint32_t addr_index;
- int status;
- char machname[MAXHOSTNAMELEN + 1];
- - gid_t gids[1];
- bool_t needcb = 1;
- rpc = calloc(1, sizeof(nfs41_rpc_clnt));
- @@ -191,12 +192,26 @@ int nfs41_rpc_clnt_create(
- rpc->sec_flavor = sec_flavor;
- if (sec_flavor == RPCSEC_AUTH_SYS) {
- + gid_t aup_gids[RPC_AUTHUNIX_AUP_MAX_NUM_GIDS];
- + int num_aup_gids = 0;
- +
- if (gethostname(machname, sizeof(machname)) == -1) {
- eprintf("nfs41_rpc_clnt_create: gethostname failed\n");
- goto out_err_client;
- }
- machname[sizeof(machname) - 1] = '\0';
- - client->cl_auth = authsys_create(machname, uid, gid, 0, gids);
- +
- + if (!fill_auth_unix_aup_gids(GetCurrentThreadToken(),
- + aup_gids, &num_aup_gids)) {
- + eprintf("nfs41_rpc_clnt_create: "
- + "fill_auth_unix_aup_gids() failed\n");
- + status = ERROR_NETWORK_UNREACHABLE;
- + goto out_err_client;
- + }
- +
- + client->cl_auth = authsys_create(machname,
- + uid, gid,
- + num_aup_gids, aup_gids);
- if (client->cl_auth == NULL) {
- eprintf("nfs41_rpc_clnt_create: failed to create rpc authsys\n");
- status = ERROR_NETWORK_UNREACHABLE;
- --
- 2.45.1
- From 2da86679ae2c1d0bfcb5985d12d59ef2ecd78dcd Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Mon, 20 Jan 2025 11:05:26 +0100
- Subject: [PATCH 29/44] tests: Update Solaris nfsd instructions
- Update Solaris nfsd instructions, add NTP client+disable ZFS sync/ZIL
- information.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/nfs_server_setup.txt | 22 +++++++++++++++-------
- 1 file changed, 15 insertions(+), 7 deletions(-)
- diff --git a/tests/nfs_server_setup.txt b/tests/nfs_server_setup.txt
- index 3577dd7..4d81277 100644
- --- a/tests/nfs_server_setup.txt
- +++ b/tests/nfs_server_setup.txt
- @@ -45,7 +45,12 @@ C:\cygwin64\sbin\nfs_mount -o rw N nfs://192.168.209.129//
- # Solaris 11.4 NFSv4.1 server setup
- #
- -# 1. Server setup
- +##### 1. Server setup
- +# enable ntp server "10.49.0.5"
- +printf "server 10.49.0.5\n" >/etc/inet/ntp.conf
- +svcadm enable ntp
- +
- +# configure&start NFS server
- svcadm enable network/nfs/server
- sharectl set -p nfsmapid_domain=global.loc nfs
- sharectl set -p server_delegation=on nfs
- @@ -54,16 +59,19 @@ chmod a+rwx /nfsdata
- share -F nfs -o rw /nfsdata/
- svcs svc:/network/nfs/server:default
- -# 2. Windows ms-nfs41-client setup:
- -# Add entries for groups "sys" and "nobody"
- -echo "sys:S-1-0-3:3:" >>/etc/groups
- -echo "nobody:S-1-0-65534:65534:" >>/etc/group
- +# performance: disable sync/ZIL on ZFS pool which exports NFS files
- +zfs set sync=disabled rpool
- +
- +##### 2. Windows ms-nfs41-client setup:
- +# Add entries for groups "sys" and "nobody" used by Solaris nfsd
- +printf "sys:S-1-0-3:3:\n" >>/etc/groups
- +printf "nobody:S-1-0-65534:65534:\n" >>/etc/group
- -# 3. Misc commands:
- +##### 3. Misc commands:
- ls -v filename # list ACLs
- chmod A... # to edit ACLs
- -# 4. Troubleshooting:
- +##### 4. Troubleshooting:
- See https://docs.oracle.com/en/operating-systems/solaris/oracle-solaris/11.4/manage-nfs/troubleshooting-network-file-systems1.html
- --
- 2.45.1
- From b225d27cb0eb27bcf17273d79da8fc91908c40c5 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Mon, 20 Jan 2025 13:25:47 +0100
- Subject: [PATCH 30/44] daemon,sys: |nfs3_attrs| EA |fileid| field should
- contain valid information
- |nfs3_attrs| extended attribute |fileid| field should contain valid
- information obtained from the NFSv4.1 server, and not "0".
- Reported-by: Lionel Cons <lionelcons1972@gmail.com>
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/open.c | 17 +++++++++++++++--
- daemon/upcall.h | 1 +
- sys/nfs41sys_driver.h | 4 +++-
- sys/nfs41sys_ea.c | 2 ++
- sys/nfs41sys_openclose.c | 9 +++++++--
- 5 files changed, 28 insertions(+), 5 deletions(-)
- diff --git a/daemon/open.c b/daemon/open.c
- index 1499cf9..b51883c 100644
- --- a/daemon/open.c
- +++ b/daemon/open.c
- @@ -780,6 +780,9 @@ static int handle_open(void *daemon_context, nfs41_upcall *upcall)
- }
- nfs_to_basic_info(state->file.name.name, &info, &args->basic_info);
- nfs_to_standard_info(&info, &args->std_info);
- + EASSERT((info.attrmask.count > 0) &&
- + (info.attrmask.arr[0] & FATTR4_WORD0_FILEID));
- + args->fileid = info.fileid;
- EASSERT((info.attrmask.count > 1) &&
- (info.attrmask.arr[1] & FATTR4_WORD1_MODE));
- args->mode = info.mode;
- @@ -795,6 +798,9 @@ static int handle_open(void *daemon_context, nfs41_upcall *upcall)
- nfs_to_basic_info(state->file.name.name, &info, &args->basic_info);
- nfs_to_standard_info(&info, &args->std_info);
- + EASSERT((info.attrmask.count > 0) &&
- + (info.attrmask.arr[0] & FATTR4_WORD0_FILEID));
- + args->fileid = info.fileid;
- EASSERT((info.attrmask.count > 1) &&
- (info.attrmask.arr[1] & FATTR4_WORD1_MODE));
- args->mode = info.mode;
- @@ -1027,6 +1033,9 @@ create_chgrp_out:
- nfs_to_basic_info(state->file.name.name, &info, &args->basic_info);
- nfs_to_standard_info(&info, &args->std_info);
- + EASSERT((info.attrmask.count > 0) &&
- + (info.attrmask.arr[0] & FATTR4_WORD0_FILEID));
- + args->fileid = info.fileid;
- EASSERT((info.attrmask.count > 1) &&
- (info.attrmask.arr[1] & FATTR4_WORD1_MODE));
- args->mode = info.mode;
- @@ -1060,6 +1069,8 @@ static int marshall_open(unsigned char *buffer, uint32_t *length, nfs41_upcall *
- if (status) goto out;
- status = safe_write(&buffer, length, &args->std_info, sizeof(args->std_info));
- if (status) goto out;
- + status = safe_write(&buffer, length, &args->fileid, sizeof(args->fileid));
- + if (status) goto out;
- status = safe_write(&buffer, length, &upcall->state_ref, sizeof(HANDLE));
- if (status) goto out;
- status = safe_write(&buffer, length, &args->mode, sizeof(args->mode));
- @@ -1089,8 +1100,10 @@ static int marshall_open(unsigned char *buffer, uint32_t *length, nfs41_upcall *
- goto out;
- }
- }
- - DPRINTF(2, ("NFS41_SYSOP_OPEN: downcall open_state=0x%p mode %o changeattr 0x%llu\n",
- - upcall->state_ref, args->mode, args->changeattr));
- + DPRINTF(2, ("NFS41_SYSOP_OPEN: downcall "
- + "open_state=0x%p fileid=0x%llx mode=0o%o changeattr=0x%llu\n",
- + upcall->state_ref, (unsigned long long)args->fileid,
- + args->mode, args->changeattr));
- out:
- return status;
- }
- diff --git a/daemon/upcall.h b/daemon/upcall.h
- index 5f44ede..99e7962 100644
- --- a/daemon/upcall.h
- +++ b/daemon/upcall.h
- @@ -46,6 +46,7 @@ typedef struct __open_upcall_args {
- nfs41_abs_path symlink;
- FILE_BASIC_INFO basic_info;
- FILE_STANDARD_INFO std_info;
- + ULONGLONG fileid;
- const char *path;
- ULONG access_mask;
- ULONG access_mode;
- diff --git a/sys/nfs41sys_driver.h b/sys/nfs41sys_driver.h
- index 860e476..225335e 100644
- --- a/sys/nfs41sys_driver.h
- +++ b/sys/nfs41sys_driver.h
- @@ -154,7 +154,7 @@ typedef struct _nfs3_attrs {
- DWORD specdata1;
- DWORD specdata2;
- } rdev;
- - LONGLONG fsid, fileid;
- + ULONGLONG fsid, fileid;
- LONGLONG atime, mtime, ctime;
- } nfs3_attrs;
- @@ -231,6 +231,7 @@ typedef struct _updowncall_entry {
- struct {
- FILE_BASIC_INFORMATION binfo;
- FILE_STANDARD_INFORMATION sinfo;
- + ULONGLONG fileid;
- UNICODE_STRING symlink;
- ULONG access_mask;
- ULONG access_mode;
- @@ -435,6 +436,7 @@ typedef struct _NFS41_FCB {
- NODE_BYTE_SIZE NodeByteSize;
- FILE_BASIC_INFORMATION BasicInfo;
- FILE_STANDARD_INFORMATION StandardInfo;
- + ULONGLONG fileid;
- BOOLEAN Renamed;
- BOOLEAN DeletePending;
- DWORD mode;
- diff --git a/sys/nfs41sys_ea.c b/sys/nfs41sys_ea.c
- index fbd61cb..1f8fdfc 100644
- --- a/sys/nfs41sys_ea.c
- +++ b/sys/nfs41sys_ea.c
- @@ -211,6 +211,8 @@ static void create_nfs3_attrs(
- attrs->nlink = nfs41_fcb->StandardInfo.NumberOfLinks;
- attrs->size.QuadPart = attrs->used.QuadPart =
- nfs41_fcb->StandardInfo.EndOfFile.QuadPart;
- + attrs->fsid = 0xBABAFACEBABAFACE;
- + attrs->fileid = nfs41_fcb->fileid;
- file_time_to_nfs_time(&nfs41_fcb->BasicInfo.LastAccessTime, &attrs->atime);
- file_time_to_nfs_time(&nfs41_fcb->BasicInfo.ChangeTime, &attrs->mtime);
- file_time_to_nfs_time(&nfs41_fcb->BasicInfo.CreationTime, &attrs->ctime);
- diff --git a/sys/nfs41sys_openclose.c b/sys/nfs41sys_openclose.c
- index d6b2767..3e8a428 100644
- --- a/sys/nfs41sys_openclose.c
- +++ b/sys/nfs41sys_openclose.c
- @@ -265,6 +265,8 @@ NTSTATUS unmarshal_nfs41_open(
- *buf += sizeof(FILE_BASIC_INFORMATION);
- RtlCopyMemory(&cur->u.Open.sinfo, *buf, sizeof(FILE_STANDARD_INFORMATION));
- *buf += sizeof(FILE_STANDARD_INFORMATION);
- + RtlCopyMemory(&cur->u.Open.fileid, *buf, sizeof(ULONGLONG));
- + *buf += sizeof(ULONGLONG);
- RtlCopyMemory(&cur->open_state, *buf, sizeof(HANDLE));
- *buf += sizeof(HANDLE);
- RtlCopyMemory(&cur->u.Open.mode, *buf, sizeof(DWORD));
- @@ -301,12 +303,14 @@ NTSTATUS unmarshal_nfs41_open(
- #endif
- }
- #ifdef DEBUG_MARSHAL_DETAIL
- - DbgP("unmarshal_nfs41_open: open_state 0x%x mode 0%o "
- + DbgP("unmarshal_nfs41_open: "
- + "open_state 0x%x fileid=0x%llx mode 0%o "
- #ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
- "owner_local_uid %u owner_group_local_gid %u "
- #endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
- "changeattr %llu "
- - "deleg_type %d\n", cur->open_state, cur->u.Open.mode,
- + "deleg_type %d\n",
- + cur->open_state, cur->u.Open.fileid, cur->u.Open.mode,
- #ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
- cur->u.Open.owner_local_uid, cur->u.Open.owner_group_local_gid,
- #endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
- @@ -812,6 +816,7 @@ retry_on_link:
- sizeof(entry->u.Open.binfo));
- RtlCopyMemory(&nfs41_fcb->StandardInfo, &entry->u.Open.sinfo,
- sizeof(entry->u.Open.sinfo));
- + nfs41_fcb->fileid = entry->u.Open.fileid;
- nfs41_fcb->mode = entry->u.Open.mode;
- #ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
- nfs41_fcb->owner_local_uid = entry->u.Open.owner_local_uid;
- --
- 2.45.1
- From 7bd426c8475a52b68b5630569235251a87ee63dc Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Mon, 20 Jan 2025 16:10:56 +0100
- Subject: [PATCH 31/44] daemon,sys: |nfs3_attrs| EA |fsid| field should contain
- valid information
- |nfs3_attrs| extended attribute |fsid| field should contain valid
- information obtained from the NFSv4.1 server, and not "0".
- Reported-by: Lionel Cons <lionelcons1972@gmail.com>
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/name_cache.c | 10 +++++++++-
- daemon/nfs41_superblock.c | 4 ++--
- daemon/open.c | 25 +++++++++++++++++++++++--
- daemon/upcall.h | 1 +
- sys/nfs41sys_driver.h | 2 ++
- sys/nfs41sys_ea.c | 12 +++++++++++-
- sys/nfs41sys_openclose.c | 12 ++++++++++--
- 7 files changed, 58 insertions(+), 8 deletions(-)
- diff --git a/daemon/name_cache.c b/daemon/name_cache.c
- index 444512b..d2b46af 100644
- --- a/daemon/name_cache.c
- +++ b/daemon/name_cache.c
- @@ -82,6 +82,8 @@ struct attr_cache_entry {
- uint64_t change;
- uint64_t size;
- uint64_t fileid;
- + uint64_t fsid_major;
- + uint64_t fsid_minor;
- int64_t time_access_s;
- int64_t time_create_s;
- int64_t time_modify_s;
- @@ -295,6 +297,10 @@ static void attr_cache_update(
- entry->invalidated = 0;
- entry->expiration = UTIL_GETRELTIME() + NAME_CACHE_EXPIRATION;
- }
- + if (info->attrmask.arr[0] & FATTR4_WORD0_FSID) {
- + entry->fsid_major = info->fsid.major;
- + entry->fsid_minor = info->fsid.minor;
- + }
- if (info->attrmask.arr[0] & FATTR4_WORD0_SIZE)
- entry->size = info->size;
- if (info->attrmask.arr[0] & FATTR4_WORD0_HIDDEN)
- @@ -341,7 +347,7 @@ static void copy_attrs(
- {
- dst->attrmask.count = 2;
- dst->attrmask.arr[0] = FATTR4_WORD0_TYPE | FATTR4_WORD0_CHANGE
- - | FATTR4_WORD0_SIZE | FATTR4_WORD0_FILEID
- + | FATTR4_WORD0_SIZE | FATTR4_WORD0_FSID | FATTR4_WORD0_FILEID
- | FATTR4_WORD0_HIDDEN | FATTR4_WORD0_ARCHIVE;
- dst->attrmask.arr[1] = FATTR4_WORD1_MODE
- | FATTR4_WORD1_NUMLINKS
- @@ -386,6 +392,8 @@ static void copy_attrs(
- dst->owner_group = NULL;
- }
- dst->fileid = src->fileid;
- + dst->fsid.major = src->fsid_major;
- + dst->fsid.minor = src->fsid_minor;
- dst->hidden = src->hidden;
- dst->system = src->system;
- dst->archive = src->archive;
- diff --git a/daemon/nfs41_superblock.c b/daemon/nfs41_superblock.c
- index cfbac4d..6005dc6 100644
- --- a/daemon/nfs41_superblock.c
- +++ b/daemon/nfs41_superblock.c
- @@ -136,8 +136,8 @@ static int get_superblock_attrs(
- superblock->default_getattr.count = 2;
- superblock->default_getattr.arr[0] = FATTR4_WORD0_TYPE
- | FATTR4_WORD0_CHANGE | FATTR4_WORD0_SIZE
- - | FATTR4_WORD0_FILEID | FATTR4_WORD0_HIDDEN
- - | FATTR4_WORD0_ARCHIVE;
- + | FATTR4_WORD0_FSID | FATTR4_WORD0_FILEID
- + | FATTR4_WORD0_HIDDEN | FATTR4_WORD0_ARCHIVE;
- superblock->default_getattr.arr[1] = FATTR4_WORD1_MODE
- | FATTR4_WORD1_NUMLINKS | FATTR4_WORD1_SYSTEM
- | FATTR4_WORD1_TIME_ACCESS | FATTR4_WORD1_TIME_CREATE
- diff --git a/daemon/open.c b/daemon/open.c
- index b51883c..4ecad9f 100644
- --- a/daemon/open.c
- +++ b/daemon/open.c
- @@ -783,6 +783,10 @@ static int handle_open(void *daemon_context, nfs41_upcall *upcall)
- EASSERT((info.attrmask.count > 0) &&
- (info.attrmask.arr[0] & FATTR4_WORD0_FILEID));
- args->fileid = info.fileid;
- + EASSERT((info.attrmask.count > 0) &&
- + (info.attrmask.arr[0] & FATTR4_WORD0_FSID));
- + args->fsid_major = info.fsid.major;
- + args->fsid_minor = info.fsid.minor;
- EASSERT((info.attrmask.count > 1) &&
- (info.attrmask.arr[1] & FATTR4_WORD1_MODE));
- args->mode = info.mode;
- @@ -801,6 +805,10 @@ static int handle_open(void *daemon_context, nfs41_upcall *upcall)
- EASSERT((info.attrmask.count > 0) &&
- (info.attrmask.arr[0] & FATTR4_WORD0_FILEID));
- args->fileid = info.fileid;
- + EASSERT((info.attrmask.count > 0) &&
- + (info.attrmask.arr[0] & FATTR4_WORD0_FSID));
- + args->fsid_major = info.fsid.major;
- + args->fsid_minor = info.fsid.minor;
- EASSERT((info.attrmask.count > 1) &&
- (info.attrmask.arr[1] & FATTR4_WORD1_MODE));
- args->mode = info.mode;
- @@ -1036,6 +1044,10 @@ create_chgrp_out:
- EASSERT((info.attrmask.count > 0) &&
- (info.attrmask.arr[0] & FATTR4_WORD0_FILEID));
- args->fileid = info.fileid;
- + EASSERT((info.attrmask.count > 0) &&
- + (info.attrmask.arr[0] & FATTR4_WORD0_FSID));
- + args->fsid_major = info.fsid.major;
- + args->fsid_minor = info.fsid.minor;
- EASSERT((info.attrmask.count > 1) &&
- (info.attrmask.arr[1] & FATTR4_WORD1_MODE));
- args->mode = info.mode;
- @@ -1071,6 +1083,10 @@ static int marshall_open(unsigned char *buffer, uint32_t *length, nfs41_upcall *
- if (status) goto out;
- status = safe_write(&buffer, length, &args->fileid, sizeof(args->fileid));
- if (status) goto out;
- + status = safe_write(&buffer, length, &args->fsid_major, sizeof(args->fsid_major));
- + if (status) goto out;
- + status = safe_write(&buffer, length, &args->fsid_minor, sizeof(args->fsid_minor));
- + if (status) goto out;
- status = safe_write(&buffer, length, &upcall->state_ref, sizeof(HANDLE));
- if (status) goto out;
- status = safe_write(&buffer, length, &args->mode, sizeof(args->mode));
- @@ -1101,8 +1117,13 @@ static int marshall_open(unsigned char *buffer, uint32_t *length, nfs41_upcall *
- }
- }
- DPRINTF(2, ("NFS41_SYSOP_OPEN: downcall "
- - "open_state=0x%p fileid=0x%llx mode=0o%o changeattr=0x%llu\n",
- - upcall->state_ref, (unsigned long long)args->fileid,
- + "open_state=0x%p "
- + "fileid=0x%llx fsid=(0x%llx.0x%llx) "
- + "mode=0o%o changeattr=0x%llu\n",
- + upcall->state_ref,
- + (unsigned long long)args->fileid,
- + (unsigned long long)args->fsid_major,
- + (unsigned long long)args->fsid_minor,
- args->mode, args->changeattr));
- out:
- return status;
- diff --git a/daemon/upcall.h b/daemon/upcall.h
- index 99e7962..fc0ab74 100644
- --- a/daemon/upcall.h
- +++ b/daemon/upcall.h
- @@ -47,6 +47,7 @@ typedef struct __open_upcall_args {
- FILE_BASIC_INFO basic_info;
- FILE_STANDARD_INFO std_info;
- ULONGLONG fileid;
- + ULONGLONG fsid_major, fsid_minor;
- const char *path;
- ULONG access_mask;
- ULONG access_mode;
- diff --git a/sys/nfs41sys_driver.h b/sys/nfs41sys_driver.h
- index 225335e..911e1cc 100644
- --- a/sys/nfs41sys_driver.h
- +++ b/sys/nfs41sys_driver.h
- @@ -232,6 +232,7 @@ typedef struct _updowncall_entry {
- FILE_BASIC_INFORMATION binfo;
- FILE_STANDARD_INFORMATION sinfo;
- ULONGLONG fileid;
- + ULONGLONG fsid_major, fsid_minor;
- UNICODE_STRING symlink;
- ULONG access_mask;
- ULONG access_mode;
- @@ -437,6 +438,7 @@ typedef struct _NFS41_FCB {
- FILE_BASIC_INFORMATION BasicInfo;
- FILE_STANDARD_INFORMATION StandardInfo;
- ULONGLONG fileid;
- + ULONGLONG fsid_major, fsid_minor;
- BOOLEAN Renamed;
- BOOLEAN DeletePending;
- DWORD mode;
- diff --git a/sys/nfs41sys_ea.c b/sys/nfs41sys_ea.c
- index 1f8fdfc..cb3e708 100644
- --- a/sys/nfs41sys_ea.c
- +++ b/sys/nfs41sys_ea.c
- @@ -211,7 +211,17 @@ static void create_nfs3_attrs(
- attrs->nlink = nfs41_fcb->StandardInfo.NumberOfLinks;
- attrs->size.QuadPart = attrs->used.QuadPart =
- nfs41_fcb->StandardInfo.EndOfFile.QuadPart;
- - attrs->fsid = 0xBABAFACEBABAFACE;
- + /*
- + * NFSv4.1 |nfs41_fsid| contains two 64bit fields (|major|,
- + * |minor|), but the |nfs3_attrs.fsid| field is only one 64bit
- + * value.
- + *
- + * For now we XOR both |nfs41_fsid.major|^|nfs41_fsid.minor|
- + * to avoid loosing data and to deal with NFSv4.1 filesystems
- + * which might have |0| in either |nfs41_fsid.major| or
- + * |nfs41_fsid.minor|.
- + */
- + attrs->fsid = nfs41_fcb->fsid_major ^ nfs41_fcb->fsid_minor;
- attrs->fileid = nfs41_fcb->fileid;
- file_time_to_nfs_time(&nfs41_fcb->BasicInfo.LastAccessTime, &attrs->atime);
- file_time_to_nfs_time(&nfs41_fcb->BasicInfo.ChangeTime, &attrs->mtime);
- diff --git a/sys/nfs41sys_openclose.c b/sys/nfs41sys_openclose.c
- index 3e8a428..4398d2a 100644
- --- a/sys/nfs41sys_openclose.c
- +++ b/sys/nfs41sys_openclose.c
- @@ -267,6 +267,10 @@ NTSTATUS unmarshal_nfs41_open(
- *buf += sizeof(FILE_STANDARD_INFORMATION);
- RtlCopyMemory(&cur->u.Open.fileid, *buf, sizeof(ULONGLONG));
- *buf += sizeof(ULONGLONG);
- + RtlCopyMemory(&cur->u.Open.fsid_major, *buf, sizeof(ULONGLONG));
- + *buf += sizeof(ULONGLONG);
- + RtlCopyMemory(&cur->u.Open.fsid_minor, *buf, sizeof(ULONGLONG));
- + *buf += sizeof(ULONGLONG);
- RtlCopyMemory(&cur->open_state, *buf, sizeof(HANDLE));
- *buf += sizeof(HANDLE);
- RtlCopyMemory(&cur->u.Open.mode, *buf, sizeof(DWORD));
- @@ -304,13 +308,15 @@ NTSTATUS unmarshal_nfs41_open(
- }
- #ifdef DEBUG_MARSHAL_DETAIL
- DbgP("unmarshal_nfs41_open: "
- - "open_state 0x%x fileid=0x%llx mode 0%o "
- + "open_state 0x%x fileid=0x%llx fsid=(0x%llx.0x%llx) mode 0%o "
- #ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
- "owner_local_uid %u owner_group_local_gid %u "
- #endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
- "changeattr %llu "
- "deleg_type %d\n",
- - cur->open_state, cur->u.Open.fileid, cur->u.Open.mode,
- + cur->open_state, cur->u.Open.fileid,
- + cur->u.Open.fsid_major, cur->u.Open.fsid_minor,
- + cur->u.Open.mode,
- #ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
- cur->u.Open.owner_local_uid, cur->u.Open.owner_group_local_gid,
- #endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
- @@ -817,6 +823,8 @@ retry_on_link:
- RtlCopyMemory(&nfs41_fcb->StandardInfo, &entry->u.Open.sinfo,
- sizeof(entry->u.Open.sinfo));
- nfs41_fcb->fileid = entry->u.Open.fileid;
- + nfs41_fcb->fsid_major = entry->u.Open.fsid_major;
- + nfs41_fcb->fsid_minor = entry->u.Open.fsid_minor;
- nfs41_fcb->mode = entry->u.Open.mode;
- #ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
- nfs41_fcb->owner_local_uid = entry->u.Open.owner_local_uid;
- --
- 2.45.1
- From bc755c67c160d504707304ffb87adfb81c03fbd6 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Tue, 21 Jan 2025 13:58:13 +0100
- Subject: [PATCH 32/44] include,sys,tests: Move |nfs3_attrs| to
- include/nfs_ea.h
- Move |nfs3_attrs| to include/nfs_ea.h and switch all users of
- |nfs3_attr| over to use that header.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- include/nfs_ea.h | 24 +++++++++++++++++++++-
- sys/nfs41sys_driver.h | 21 -------------------
- tests/ea/Makefile | 4 ++--
- tests/ea/main.c | 13 ++----------
- tests/winfsinfo1/Makefile | 4 ++--
- tests/winfsinfo1/winfsinfo.c | 40 ++++++++++++++++++------------------
- 6 files changed, 49 insertions(+), 57 deletions(-)
- diff --git a/include/nfs_ea.h b/include/nfs_ea.h
- index aaf36c0..38965d6 100644
- --- a/include/nfs_ea.h
- +++ b/include/nfs_ea.h
- @@ -1,6 +1,6 @@
- /*
- * NFSv4.1 client for Windows
- - * Copyright (C) 2024 Roland Mainz <roland.mainz@nrubsig.org>
- + * Copyright (C) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Roland Mainz <roland.mainz@nrubsig.org>
- *
- @@ -35,4 +35,26 @@
- #define EA_NFSACTONLINK "NfsActOnLink"
- #define EA_NFSACTONLINK_LEN (11)
- +/* NfsV3Attributes uses |nfs3_attrs| as content */
- +typedef struct _nfs3_attrs {
- + DWORD type, mode, nlink, uid, gid, filler1;
- + LARGE_INTEGER size, used;
- + struct {
- + DWORD specdata1;
- + DWORD specdata2;
- + } rdev;
- + ULONGLONG fsid, fileid;
- + LONGLONG atime, mtime, ctime;
- +} nfs3_attrs;
- +
- +enum ftype3 {
- + NF3REG = 1,
- + NF3DIR,
- + NF3BLK,
- + NF3CHR,
- + NF3LNK,
- + NF3SOCK,
- + NF3FIFO
- +};
- +
- #endif /* !__NFS41_DAEMON_NFSV4_EA */
- diff --git a/sys/nfs41sys_driver.h b/sys/nfs41sys_driver.h
- index 911e1cc..8bb88bc 100644
- --- a/sys/nfs41sys_driver.h
- +++ b/sys/nfs41sys_driver.h
- @@ -147,27 +147,6 @@ extern nfs41_timings getexattr;
- #define MILLISECONDS(milli) (((signed __int64)(milli)) * MICROSECONDS(1000L))
- #define SECONDS(seconds) (((signed __int64)(seconds)) * MILLISECONDS(1000L))
- -typedef struct _nfs3_attrs {
- - DWORD type, mode, nlink, uid, gid, filler1;
- - LARGE_INTEGER size, used;
- - struct {
- - DWORD specdata1;
- - DWORD specdata2;
- - } rdev;
- - ULONGLONG fsid, fileid;
- - LONGLONG atime, mtime, ctime;
- -} nfs3_attrs;
- -
- -enum ftype3 {
- - NF3REG = 1,
- - NF3DIR,
- - NF3BLK,
- - NF3CHR,
- - NF3LNK,
- - NF3SOCK,
- - NF3FIFO
- -};
- -
- typedef enum _nfs41_updowncall_state {
- NFS41_WAITING_FOR_UPCALL,
- NFS41_WAITING_FOR_DOWNCALL,
- diff --git a/tests/ea/Makefile b/tests/ea/Makefile
- index 480449b..65e58a3 100644
- --- a/tests/ea/Makefile
- +++ b/tests/ea/Makefile
- @@ -7,10 +7,10 @@
- all: nfs_ea.i686.exe nfs_ea.x86_64.exe nfs_ea.exe
- nfs_ea.i686.exe: main.c
- - clang -target i686-pc-windows-gnu -std=gnu17 -municode -Wall -Wextra -D_CRT_STDIO_ISO_WIDE_SPECIFIERS=1 -DUNICODE=1 -D_UNICODE=1 -isystem /usr/include/w32api/ddk -g main.c -lntdll -o nfs_ea.i686.exe
- + clang -target i686-pc-windows-gnu -std=gnu17 -municode -Wall -Wextra -D_CRT_STDIO_ISO_WIDE_SPECIFIERS=1 -DUNICODE=1 -D_UNICODE=1 -isystem /usr/include/w32api/ddk -I../../include -g main.c -lntdll -o nfs_ea.i686.exe
- nfs_ea.x86_64.exe: main.c
- - clang -target x86_64-pc-windows-gnu -std=gnu17 -municode -Wall -Wextra -D_CRT_STDIO_ISO_WIDE_SPECIFIERS=1 -DUNICODE=1 -D_UNICODE=1 -isystem /usr/include/w32api/ddk -g main.c -lntdll -o nfs_ea.x86_64.exe
- + clang -target x86_64-pc-windows-gnu -std=gnu17 -municode -Wall -Wextra -D_CRT_STDIO_ISO_WIDE_SPECIFIERS=1 -DUNICODE=1 -D_UNICODE=1 -isystem /usr/include/w32api/ddk -I../../include -g main.c -lntdll -o nfs_ea.x86_64.exe
- nfs_ea.exe: nfs_ea.x86_64.exe
- rm -f nfs_ea.exe
- diff --git a/tests/ea/main.c b/tests/ea/main.c
- index 507bc12..2f1596f 100644
- --- a/tests/ea/main.c
- +++ b/tests/ea/main.c
- @@ -36,6 +36,8 @@
- typedef unsigned long DWORD, *PDWORD, *LPDWORD;
- #endif
- +#include "nfs_ea.h"
- +
- #define MAX_LIST_LEN 4096
- #define MAX_EA_VALUE 256
- @@ -158,17 +160,6 @@ out:
- return status;
- }
- -typedef struct _nfs3_attrs {
- - DWORD type, mode, nlink, uid, gid, filler1;
- - LARGE_INTEGER size, used;
- - struct {
- - DWORD specdata1;
- - DWORD specdata2;
- - } rdev;
- - LONGLONG fsid, fileid;
- - LONGLONG atime, mtime, ctime;
- -} nfs3_attrs;
- -
- static NTSTATUS ea_get_nfs3attr(
- HANDLE FileHandle)
- {
- diff --git a/tests/winfsinfo1/Makefile b/tests/winfsinfo1/Makefile
- index b0b8d93..6ef7039 100644
- --- a/tests/winfsinfo1/Makefile
- +++ b/tests/winfsinfo1/Makefile
- @@ -7,10 +7,10 @@
- all: winfsinfo.i686.exe winfsinfo.x86_64.exe winfsinfo.exe
- winfsinfo.i686.exe: winfsinfo.c
- - clang -target i686-pc-windows-gnu -Wall -Wextra -DUNICODE=1 -D_UNICODE=1 -g winfsinfo.c -lntdll -o winfsinfo.i686.exe
- + clang -target i686-pc-windows-gnu -Wall -Wextra -DUNICODE=1 -D_UNICODE=1 -I../../include -g winfsinfo.c -lntdll -o winfsinfo.i686.exe
- winfsinfo.x86_64.exe: winfsinfo.c
- - clang -target x86_64-pc-windows-gnu -Wall -Wextra -DUNICODE=1 -D_UNICODE=1 -g winfsinfo.c -lntdll -o winfsinfo.x86_64.exe
- + clang -target x86_64-pc-windows-gnu -Wall -Wextra -DUNICODE=1 -D_UNICODE=1 -I../../include -g winfsinfo.c -lntdll -o winfsinfo.x86_64.exe
- winfsinfo.exe: winfsinfo.x86_64.exe
- ln -s winfsinfo.x86_64.exe winfsinfo.exe
- diff --git a/tests/winfsinfo1/winfsinfo.c b/tests/winfsinfo1/winfsinfo.c
- index 7b80ea4..1756b6f 100644
- --- a/tests/winfsinfo1/winfsinfo.c
- +++ b/tests/winfsinfo1/winfsinfo.c
- @@ -1,7 +1,7 @@
- /*
- * MIT License
- *
- - * Copyright (c) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
- + * Copyright (c) 2023-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Permission is hereby granted, free of charge, to any person obtaining a copy
- * of this software and associated documentation files (the "Software"), to deal
- @@ -38,6 +38,8 @@
- #include <stdint.h>
- #include <stdbool.h>
- +#include "nfs_ea.h"
- +
- static
- bool filetime2localsystemtime(const FILETIME *ft, SYSTEMTIME *st)
- {
- @@ -622,19 +624,6 @@ done:
- return res;
- }
- -typedef struct _nfs3_attrs {
- - DWORD type, mode, nlink, uid, gid, filler1;
- - LARGE_INTEGER size, used;
- - struct {
- - DWORD specdata1;
- - DWORD specdata2;
- - } rdev;
- - LONGLONG fsid, fileid;
- - LONGLONG atime, mtime, ctime;
- -} nfs3_attrs;
- -
- -#define NfsV3Attributes_NAME "NfsV3Attributes"
- -
- typedef struct _FILE_EA_INFORMATION {
- ULONG EaSize;
- } FILE_EA_INFORMATION, *PFILE_EA_INFORMATION;
- @@ -697,11 +686,11 @@ bool get_getnfs3attr(const char *progname, const char *filename)
- struct {
- FILE_FULL_EA_INFORMATION ffeai;
- - char buf[sizeof(NfsV3Attributes_NAME) + sizeof(nfs3_attrs)];
- + char buf[sizeof(EA_NFSV3ATTRIBUTES) + sizeof(nfs3_attrs)];
- } ffeai_buf;
- struct {
- FILE_GET_EA_INFORMATION fgeai;
- - char buf[sizeof(NfsV3Attributes_NAME)];
- + char buf[sizeof(EA_NFSV3ATTRIBUTES)];
- } fgeai_buf;
- NTSTATUS status;
- @@ -709,7 +698,7 @@ bool get_getnfs3attr(const char *progname, const char *filename)
- fgeai_buf.fgeai.NextEntryOffset = 0;
- fgeai_buf.fgeai.EaNameLength = 15;
- - (void)strcpy(fgeai_buf.fgeai.EaName, NfsV3Attributes_NAME);
- + (void)strcpy(fgeai_buf.fgeai.EaName, EA_NFSV3ATTRIBUTES);
- status = ZwQueryEaFile(fileHandle, &io,
- &ffeai_buf.ffeai, sizeof(ffeai_buf), TRUE,
- @@ -728,6 +717,17 @@ bool get_getnfs3attr(const char *progname, const char *filename)
- goto done;
- }
- + if (ffeai_buf.ffeai.EaValueLength < sizeof(nfs3_attrs)) {
- + (void)fprintf(stderr,
- + "EA '%s' size too small (%ld bytes), "
- + "expected at least %ld bytes for nfs3_attrs\n",
- + EA_NFSV3ATTRIBUTES,
- + (long)ffeai_buf.ffeai.EaValueLength,
- + (long)sizeof(nfs3_attrs));
- + res = EXIT_FAILURE;
- + goto done;
- + }
- +
- nfs3_attrs *n3a = (nfs3_attrs *)(ffeai_buf.ffeai.EaName
- + ffeai_buf.ffeai.EaNameLength + 1);
- @@ -740,7 +740,7 @@ bool get_getnfs3attr(const char *progname, const char *filename)
- "\tuid=%d\n\tgid=%d\n"
- "\tsize=%lld\n\tused=%lld\n"
- "\trdev=( specdata1=0x%x specdata2=0x%x )\n"
- - "\tfsid=%lld\n\tfileid=%lld\n"
- + "\tfsid=0x%llx\n\tfileid=0x%llx\n"
- "\tatime=%lld\n\tmtime=%lld\n\tctime=%lld\n"
- ")\n",
- filename,
- @@ -753,8 +753,8 @@ bool get_getnfs3attr(const char *progname, const char *filename)
- (long long)n3a->used.QuadPart,
- (int)n3a->rdev.specdata1,
- (int)n3a->rdev.specdata2,
- - (long long)n3a->fsid,
- - (long long)n3a->fileid,
- + (unsigned long long)n3a->fsid,
- + (unsigned long long)n3a->fileid,
- (long long)n3a->atime,
- (long long)n3a->mtime,
- (long long)n3a->ctime);
- --
- 2.45.1
- From c4c4c6c7796d39378f60ebb1ae56e69eca2c833d Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Tue, 21 Jan 2025 14:04:20 +0100
- Subject: [PATCH 33/44] tests: Rename winfsinfo getnfs3attr to nfs3attr
- Rename winfsinfo getnfs3attr to nfs3attr to be consistent with
- other command names.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- tests/winfsinfo1/winfsinfo.c | 8 ++++----
- 1 file changed, 4 insertions(+), 4 deletions(-)
- diff --git a/tests/winfsinfo1/winfsinfo.c b/tests/winfsinfo1/winfsinfo.c
- index 1756b6f..f60a89b 100644
- --- a/tests/winfsinfo1/winfsinfo.c
- +++ b/tests/winfsinfo1/winfsinfo.c
- @@ -668,7 +668,7 @@ ZwQueryEaFile(
- IN BOOLEAN RestartScan);
- static
- -bool get_getnfs3attr(const char *progname, const char *filename)
- +bool get_nfs3attr(const char *progname, const char *filename)
- {
- int res = EXIT_FAILURE;
- @@ -777,7 +777,7 @@ void usage(void)
- "filenormalizednameinfo|"
- "filecasesensitiveinfo|"
- "getfiletime|"
- - "getnfs3attr"
- + "nfs3attr"
- "> path\n");
- }
- @@ -816,8 +816,8 @@ int main(int ac, char *av[])
- else if (!strcmp(subcmd, "filecasesensitiveinfo")) {
- return get_filecasesensitiveinfo(av[0], av[2]);
- }
- - else if (!strcmp(subcmd, "getnfs3attr")) {
- - return get_getnfs3attr(av[0], av[2]);
- + else if (!strcmp(subcmd, "nfs3attr")) {
- + return get_nfs3attr(av[0], av[2]);
- }
- else {
- (void)fprintf(stderr, "%s: Unknown subcmd '%s'\n", av[0], subcmd);
- --
- 2.45.1
- From 975726a8f52d4a60da366bb2e286a78a8344b17f Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Tue, 21 Jan 2025 16:13:49 +0100
- Subject: [PATCH 34/44] include,sys,tests: |nfs3_attrs| should use fixed-size,
- simple integer datatypes
- |nfs3_attrs| should use fixed-size, simple integer datatypes.
- Note that <stdint.h> cannot be used because the Windows kernel API
- is incompatible with the VS19 version of that header, so we use simple
- Windows Types for this.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- include/nfs_ea.h | 21 ++++++++++++++-------
- sys/nfs41sys_ea.c | 4 ++--
- tests/ea/main.c | 4 ++--
- tests/winfsinfo1/winfsinfo.c | 4 ++--
- 4 files changed, 20 insertions(+), 13 deletions(-)
- diff --git a/include/nfs_ea.h b/include/nfs_ea.h
- index 38965d6..e546e79 100644
- --- a/include/nfs_ea.h
- +++ b/include/nfs_ea.h
- @@ -35,16 +35,23 @@
- #define EA_NFSACTONLINK "NfsActOnLink"
- #define EA_NFSACTONLINK_LEN (11)
- -/* NfsV3Attributes uses |nfs3_attrs| as content */
- +/*
- + * "NfsV3Attributes" uses |nfs3_attrs| as content
- + */
- +/*
- + * Note that we cannot use <stdint.h> in the Windows kernel, so we
- + * use Windows Types here
- + */
- +
- typedef struct _nfs3_attrs {
- - DWORD type, mode, nlink, uid, gid, filler1;
- - LARGE_INTEGER size, used;
- + UINT32 type, mode, nlink, uid, gid, filler1;
- + UINT64 size, used;
- struct {
- - DWORD specdata1;
- - DWORD specdata2;
- + UINT32 specdata1;
- + UINT32 specdata2;
- } rdev;
- - ULONGLONG fsid, fileid;
- - LONGLONG atime, mtime, ctime;
- + UINT64 fsid, fileid;
- + INT64 atime, mtime, ctime;
- } nfs3_attrs;
- enum ftype3 {
- diff --git a/sys/nfs41sys_ea.c b/sys/nfs41sys_ea.c
- index cb3e708..cbc8ca2 100644
- --- a/sys/nfs41sys_ea.c
- +++ b/sys/nfs41sys_ea.c
- @@ -177,7 +177,7 @@ static void print_nfs3_attrs(
- DbgP("type=%d mode=0%o nlink=%d size=%lld "
- "atime=0x%llx mtime=0x%llx ctime=0x%llx\n",
- attrs->type, attrs->mode, attrs->nlink,
- - (long long)attrs->size.QuadPart,
- + (long long)attrs->size,
- (long long)attrs->atime,
- (long long)attrs->mtime,
- (long long)attrs->ctime);
- @@ -209,7 +209,7 @@ static void create_nfs3_attrs(
- attrs->gid = nfs41_fcb->owner_group_local_gid;
- #endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
- attrs->nlink = nfs41_fcb->StandardInfo.NumberOfLinks;
- - attrs->size.QuadPart = attrs->used.QuadPart =
- + attrs->size = attrs->used =
- nfs41_fcb->StandardInfo.EndOfFile.QuadPart;
- /*
- * NFSv4.1 |nfs41_fsid| contains two 64bit fields (|major|,
- diff --git a/tests/ea/main.c b/tests/ea/main.c
- index 2f1596f..ade7816 100644
- --- a/tests/ea/main.c
- +++ b/tests/ea/main.c
- @@ -214,8 +214,8 @@ static NTSTATUS ea_get_nfs3attr(
- (int)n3a->nlink,
- (int)n3a->uid,
- (int)n3a->gid,
- - (long long)n3a->size.QuadPart,
- - (long long)n3a->used.QuadPart,
- + (long long)n3a->size,
- + (long long)n3a->used,
- (int)n3a->rdev.specdata1,
- (int)n3a->rdev.specdata2,
- (long long)n3a->fsid,
- diff --git a/tests/winfsinfo1/winfsinfo.c b/tests/winfsinfo1/winfsinfo.c
- index f60a89b..569dcd5 100644
- --- a/tests/winfsinfo1/winfsinfo.c
- +++ b/tests/winfsinfo1/winfsinfo.c
- @@ -749,8 +749,8 @@ bool get_nfs3attr(const char *progname, const char *filename)
- (int)n3a->nlink,
- (int)n3a->uid,
- (int)n3a->gid,
- - (long long)n3a->size.QuadPart,
- - (long long)n3a->used.QuadPart,
- + (long long)n3a->size,
- + (long long)n3a->used,
- (int)n3a->rdev.specdata1,
- (int)n3a->rdev.specdata2,
- (unsigned long long)n3a->fsid,
- --
- 2.45.1
- From b725f2009b39bd7feae744b454befdbadb88c9b2 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Tue, 21 Jan 2025 16:42:45 +0100
- Subject: [PATCH 35/44] include,sys,tests: Add support for sub-second
- timestamps in |nfs3_attrs| EA
- Add support for sub-second (NFSv4 uses nanosecond resolution,
- but the Win32 API only supports 100ns intervals) timestamps in
- |nfs3_attrs| EA.
- Reported-by: Lionel Cons <lionelcons1972@gmail.com>
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- include/nfs_ea.h | 6 +++++-
- sys/nfs41sys_ea.c | 21 +++++++++++++++------
- tests/ea/main.c | 10 ++++++----
- tests/winfsinfo1/winfsinfo.c | 10 ++++++----
- 4 files changed, 32 insertions(+), 15 deletions(-)
- diff --git a/include/nfs_ea.h b/include/nfs_ea.h
- index e546e79..e582a92 100644
- --- a/include/nfs_ea.h
- +++ b/include/nfs_ea.h
- @@ -42,6 +42,10 @@
- * Note that we cannot use <stdint.h> in the Windows kernel, so we
- * use Windows Types here
- */
- +typedef struct _nfs3_attrs_timestruc_t {
- + INT32 tv_sec;
- + UINT32 tv_nsec;
- +} nfs3_attrs_timestruc_t;
- typedef struct _nfs3_attrs {
- UINT32 type, mode, nlink, uid, gid, filler1;
- @@ -51,7 +55,7 @@ typedef struct _nfs3_attrs {
- UINT32 specdata2;
- } rdev;
- UINT64 fsid, fileid;
- - INT64 atime, mtime, ctime;
- + nfs3_attrs_timestruc_t atime, mtime, ctime;
- } nfs3_attrs;
- enum ftype3 {
- diff --git a/sys/nfs41sys_ea.c b/sys/nfs41sys_ea.c
- index cbc8ca2..51e1bf0 100644
- --- a/sys/nfs41sys_ea.c
- +++ b/sys/nfs41sys_ea.c
- @@ -175,21 +175,30 @@ static void print_nfs3_attrs(
- nfs3_attrs *attrs)
- {
- DbgP("type=%d mode=0%o nlink=%d size=%lld "
- - "atime=0x%llx mtime=0x%llx ctime=0x%llx\n",
- + "atime=(tv_sec=%ld,tv_nsec=%lu) "
- + "mtime=(tv_sec=%ld,tv_nsec=%lu) "
- + "ctime=(tv_sec=%ld,tv_nsec=%lu)\n",
- attrs->type, attrs->mode, attrs->nlink,
- (long long)attrs->size,
- - (long long)attrs->atime,
- - (long long)attrs->mtime,
- - (long long)attrs->ctime);
- + (long)attrs->atime.tv_sec, (unsigned long)attrs->atime.tv_nsec,
- + (long)attrs->mtime.tv_sec, (unsigned long)attrs->mtime.tv_nsec,
- + (long)attrs->ctime.tv_sec, (unsigned long)attrs->ctime.tv_nsec);
- }
- static void file_time_to_nfs_time(
- IN const PLARGE_INTEGER file_time,
- - OUT LONGLONG *nfs_time)
- + OUT nfs3_attrs_timestruc_t *nfs_time)
- {
- + /*
- + * Win32 timestamps (|time_file|) use 100-nanosecond intervals
- + * (10000000 intervals == one second) since January 1, 1601 (UTC),
- + * while "old UNIX" timestamps count in seconds since 00:00:00 UTC
- + * on 1 January 1970
- + */
- LARGE_INTEGER diff = unix_time_diff;
- diff.QuadPart = file_time->QuadPart - diff.QuadPart;
- - *nfs_time = diff.QuadPart / 10000000;
- + nfs_time->tv_sec = (INT32)(diff.QuadPart / 10000000LL);
- + nfs_time->tv_nsec = (UINT32)((diff.QuadPart % 10000000LL) * 100LL);
- }
- static void create_nfs3_attrs(
- diff --git a/tests/ea/main.c b/tests/ea/main.c
- index ade7816..7492edb 100644
- --- a/tests/ea/main.c
- +++ b/tests/ea/main.c
- @@ -207,7 +207,9 @@ static NTSTATUS ea_get_nfs3attr(
- "\tsize=%lld\n\tused=%lld\n"
- "\trdev=( specdata1=0x%x specdata2=0x%x )\n"
- "\tfsid=%lld\n\tfileid=%lld\n"
- - "\tatime=%lld\n\tmtime=%lld\n\tctime=%lld\n"
- + "\tatime=(tv_sec=%ld,tv_nsec=%lu)\n"
- + "\tmtime=(tv_sec=%ld,tv_nsec=%lu)\n"
- + "\tctime=(tv_sec=%ld,tv_nsec=%lu)\n"
- ")\n",
- (int)n3a->type,
- (int)n3a->mode,
- @@ -220,9 +222,9 @@ static NTSTATUS ea_get_nfs3attr(
- (int)n3a->rdev.specdata2,
- (long long)n3a->fsid,
- (long long)n3a->fileid,
- - (long long)n3a->atime,
- - (long long)n3a->mtime,
- - (long long)n3a->ctime);
- + (long)n3a->atime.tv_sec, (unsigned long)n3a->atime.tv_nsec,
- + (long)n3a->mtime.tv_sec, (unsigned long)n3a->mtime.tv_nsec,
- + (long)n3a->ctime.tv_sec, (unsigned long)n3a->ctime.tv_nsec);
- out:
- return status;
- diff --git a/tests/winfsinfo1/winfsinfo.c b/tests/winfsinfo1/winfsinfo.c
- index 569dcd5..06cbc2e 100644
- --- a/tests/winfsinfo1/winfsinfo.c
- +++ b/tests/winfsinfo1/winfsinfo.c
- @@ -741,7 +741,9 @@ bool get_nfs3attr(const char *progname, const char *filename)
- "\tsize=%lld\n\tused=%lld\n"
- "\trdev=( specdata1=0x%x specdata2=0x%x )\n"
- "\tfsid=0x%llx\n\tfileid=0x%llx\n"
- - "\tatime=%lld\n\tmtime=%lld\n\tctime=%lld\n"
- + "\tatime=(tv_sec=%ld,tv_nsec=%lu)\n"
- + "\tmtime=(tv_sec=%ld,tv_nsec=%lu)\n"
- + "\tctime=(tv_sec=%ld,tv_nsec=%lu)\n"
- ")\n",
- filename,
- (int)n3a->type,
- @@ -755,9 +757,9 @@ bool get_nfs3attr(const char *progname, const char *filename)
- (int)n3a->rdev.specdata2,
- (unsigned long long)n3a->fsid,
- (unsigned long long)n3a->fileid,
- - (long long)n3a->atime,
- - (long long)n3a->mtime,
- - (long long)n3a->ctime);
- + (long)n3a->atime.tv_sec, (unsigned long)n3a->atime.tv_nsec,
- + (long)n3a->mtime.tv_sec, (unsigned long)n3a->mtime.tv_nsec,
- + (long)n3a->ctime.tv_sec, (unsigned long)n3a->ctime.tv_nsec);
- res = EXIT_SUCCESS;
- done:
- --
- 2.45.1
- From 7e1c21df3355e8b06fb3527cffbb02f30c6b4256 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Tue, 21 Jan 2025 16:58:24 +0100
- Subject: [PATCH 36/44] daemon,include,sys: Unsupported timestamps in
- |nfs3_attrs| EA should return |tv_sec==-1|
- Unsupported timestamps in |nfs3_attrs| EA should return |tv_sec==-1|.
- Reported-by: Lionel Cons <lionelcons1972@gmail.com>
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/fileinfoutil.c | 1 +
- daemon/readdir.c | 1 +
- daemon/util.h | 8 --------
- include/nfs41_driver.h | 8 ++++++++
- sys/nfs41sys_ea.c | 10 ++++++++++
- 5 files changed, 20 insertions(+), 8 deletions(-)
- diff --git a/daemon/fileinfoutil.c b/daemon/fileinfoutil.c
- index 8278a90..bddbfd3 100644
- --- a/daemon/fileinfoutil.c
- +++ b/daemon/fileinfoutil.c
- @@ -31,6 +31,7 @@
- #include "nfs41_daemon.h"
- #include "util.h"
- #include "nfs41_ops.h"
- +#include "nfs41_driver.h" /* for |FILE_INFO_TIME_NOT_SET| */
- ULONG nfs_file_info_to_attributes(
- diff --git a/daemon/readdir.c b/daemon/readdir.c
- index 39b50fb..70cbbed 100644
- --- a/daemon/readdir.c
- +++ b/daemon/readdir.c
- @@ -25,6 +25,7 @@
- #include <stdlib.h>
- #include <stdbool.h>
- #include <string.h>
- +#include "nfs41_driver.h" /* for |FILE_INFO_TIME_NOT_SET| */
- #include "from_kernel.h"
- #include "nfs41_ops.h"
- #include "daemon_debug.h"
- diff --git a/daemon/util.h b/daemon/util.h
- index 8460c2a..b536f48 100644
- --- a/daemon/util.h
- +++ b/daemon/util.h
- @@ -49,14 +49,6 @@ enum stable_how4;
- (((signed long long)(t1))-((signed long long)(t2)))
- typedef ULONGLONG util_reltimestamp;
- -/*
- - * LargeInteger.QuadPart value to indicate a time value was not
- - * available
- - *
- - * gisburn: FIXME: We need a better header for this
- - */
- -#define FILE_INFO_TIME_NOT_SET (0LL)
- -
- #define PTR2PTRDIFF_T(p) ((ptrdiff_t)((char *)((void *)(p)) - ((char *)0)))
- #define PTRDIFF_T2PTR(d) ((void *)(((char *)0) + (d)))
- diff --git a/include/nfs41_driver.h b/include/nfs41_driver.h
- index 7d933f0..0d9b8b3 100644
- --- a/include/nfs41_driver.h
- +++ b/include/nfs41_driver.h
- @@ -107,6 +107,14 @@ typedef enum _nfs41_start_driver_state {
- #define NFS_VERSION_AUTONEGOTIATION (0xFFFF)
- +/*
- + * LargeInteger.QuadPart value to indicate a time value was not
- + * available
- + *
- + * gisburn: FIXME: We need a better header for this
- + */
- +#define FILE_INFO_TIME_NOT_SET (0LL)
- +
- /*
- * Error/Status codes
- */
- diff --git a/sys/nfs41sys_ea.c b/sys/nfs41sys_ea.c
- index 51e1bf0..dc14598 100644
- --- a/sys/nfs41sys_ea.c
- +++ b/sys/nfs41sys_ea.c
- @@ -189,6 +189,16 @@ static void file_time_to_nfs_time(
- IN const PLARGE_INTEGER file_time,
- OUT nfs3_attrs_timestruc_t *nfs_time)
- {
- + if (file_time->QuadPart == FILE_INFO_TIME_NOT_SET) {
- + /*
- + * Return tv_sec==-1 to indicate that this
- + * value is not supported
- + */
- + nfs_time->tv_sec = -1L;
- + nfs_time->tv_nsec = ~0UL;
- + return;
- + }
- +
- /*
- * Win32 timestamps (|time_file|) use 100-nanosecond intervals
- * (10000000 intervals == one second) since January 1, 1601 (UTC),
- --
- 2.45.1
- From ac6ea385d9912623fb95d36cf3febbc670bec959 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Tue, 21 Jan 2025 17:26:05 +0100
- Subject: [PATCH 37/44] sys: |nfs3_attrs|'s |mtime|+|ctime| uses wrong values
- |nfs3_attrs| |mtime|+|ctime| uses wrong values.
- Per Cygwin source code the mapping should be:
- |FILE_BASIC_INFO.LastWriteTime| <---> |nfs3_attrs.mtime|
- |FILE_BASIC_INFO.ChangeTime| <---> |nfs3_attrs.ctime|
- |FILE_BASIC_INFO.LastAccessTime| <---> |nfs3_attrs.atime|
- Reported-by: Lionel Cons <lionelcons1972@gmail.com>
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- sys/nfs41sys_ea.c | 4 ++--
- 1 file changed, 2 insertions(+), 2 deletions(-)
- diff --git a/sys/nfs41sys_ea.c b/sys/nfs41sys_ea.c
- index dc14598..629f3d8 100644
- --- a/sys/nfs41sys_ea.c
- +++ b/sys/nfs41sys_ea.c
- @@ -243,8 +243,8 @@ static void create_nfs3_attrs(
- attrs->fsid = nfs41_fcb->fsid_major ^ nfs41_fcb->fsid_minor;
- attrs->fileid = nfs41_fcb->fileid;
- file_time_to_nfs_time(&nfs41_fcb->BasicInfo.LastAccessTime, &attrs->atime);
- - file_time_to_nfs_time(&nfs41_fcb->BasicInfo.ChangeTime, &attrs->mtime);
- - file_time_to_nfs_time(&nfs41_fcb->BasicInfo.CreationTime, &attrs->ctime);
- + file_time_to_nfs_time(&nfs41_fcb->BasicInfo.LastWriteTime, &attrs->mtime);
- + file_time_to_nfs_time(&nfs41_fcb->BasicInfo.ChangeTime, &attrs->ctime);
- }
- --
- 2.45.1
- From 6c1de151174b6a9b843341756b1b200a94d26fb3 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 22 Jan 2025 11:43:36 +0100
- Subject: [PATCH 38/44] daemon: Cleanup |nfs41_file_info|
- Cleanup |nfs41_file_info|.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/nfs41_types.h | 36 ++++++++++++++++++++----------------
- 1 file changed, 20 insertions(+), 16 deletions(-)
- diff --git a/daemon/nfs41_types.h b/daemon/nfs41_types.h
- index 93375db..8472f04 100644
- --- a/daemon/nfs41_types.h
- +++ b/daemon/nfs41_types.h
- @@ -198,44 +198,48 @@ typedef struct __mdsthreshold4 {
- } mdsthreshold4;
- typedef struct __nfs41_file_info {
- + bitmap4 attrmask;
- + uint64_t fileid;
- nfs41_fsid fsid;
- mdsthreshold4 mdsthreshold;
- nfstime4 time_access;
- nfstime4 time_create;
- nfstime4 time_modify;
- nfsacl41 *acl;
- - nfstime4 *time_delta; /* XXX: per-fs */
- - bitmap4 attrmask;
- - bitmap4 *supported_attrs; /* XXX: per-fs */
- - bitmap4 *suppattr_exclcreat; /* XXX: per-fs */
- - uint64_t maxread; /* XXX: per-fs */
- - uint64_t maxwrite; /* XXX: per-fs */
- uint64_t change;
- uint64_t size;
- - uint64_t fileid;
- - uint64_t space_avail; /* XXX: per-fs */
- - uint64_t space_free; /* XXX: per-fs */
- - uint64_t space_total; /* XXX: per-fs */
- uint32_t type;
- uint32_t numlinks;
- uint32_t rdattr_error;
- uint32_t mode;
- uint32_t mode_mask;
- - fs_locations4 *fs_locations; /* XXX: per-fs */
- - uint32_t lease_time; /* XXX: per-server */
- - uint32_t fs_layout_types; /* pnfs, XXX: per-fs */
- + char *owner;
- + char *owner_group;
- bool_t hidden;
- bool_t system;
- bool_t archive;
- - bool_t cansettime; /* XXX: per-fs */
- bool_t case_insensitive;
- bool_t case_preserving;
- bool_t symlink_dir;
- bool_t symlink_support;
- bool_t link_support;
- - char *owner;
- - char *owner_group;
- +
- + /* per-fs attributes */
- uint32_t aclsupport;
- + uint64_t space_avail;
- + uint64_t space_free;
- + uint64_t space_total;
- + nfstime4 *time_delta;
- + bitmap4 *supported_attrs;
- + bitmap4 *suppattr_exclcreat;
- + uint64_t maxread;
- + uint64_t maxwrite;
- + fs_locations4 *fs_locations;
- + bool_t cansettime;
- + uint32_t fs_layout_types; /* pnfs, per-fs */
- +
- + /* per-server attributes */
- + uint32_t lease_time;
- /* Buffers */
- char owner_buf[NFS4_FATTR4_OWNER_LIMIT+1];
- --
- 2.45.1
- From 230fd28b25221e0c754c3bd83b9a79bc009b37d6 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 22 Jan 2025 11:45:07 +0100
- Subject: [PATCH 39/44] daemon: name cache should only return values which were
- really set
- name cache should only return values which were really set via
- storing the information which fields are valid in a flags field,
- and not rely on "magic values".
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/name_cache.c | 156 +++++++++++++++++++++++++++++++-------------
- 1 file changed, 110 insertions(+), 46 deletions(-)
- diff --git a/daemon/name_cache.c b/daemon/name_cache.c
- index d2b46af..1faaa14 100644
- --- a/daemon/name_cache.c
- +++ b/daemon/name_cache.c
- @@ -75,10 +75,26 @@ static __inline bool_t is_delegation(
- return type == OPEN_DELEGATE_READ || type == OPEN_DELEGATE_WRITE;
- }
- +#define NC_ATTR_TYPE (1 << 0)
- +#define NC_ATTR_CHANGE (1 << 1)
- +#define NC_ATTR_FSID (1 << 2)
- +#define NC_ATTR_SIZE (1 << 3)
- +#define NC_ATTR_HIDDEN (1 << 4)
- +#define NC_ATTR_ARCHIVE (1 << 5)
- +#define NC_ATTR_MODE (1 << 6)
- +#define NC_ATTR_NUMLINKS (1 << 7)
- +#define NC_ATTR_OWNER (1 << 8)
- +#define NC_ATTR_OWNER_GROUP (1 << 9)
- +#define NC_ATTR_TIME_ACCESS (1 << 10)
- +#define NC_ATTR_TIME_CREATE (1 << 11)
- +#define NC_ATTR_TIME_MODIFY (1 << 12)
- +#define NC_ATTR_SYSTEM (1 << 13)
- +
- /* attribute cache */
- struct attr_cache_entry {
- RB_ENTRY(attr_cache_entry) rbnode;
- struct list_entry free_entry;
- + uint32_t nc_attrs;
- uint64_t change;
- uint64_t size;
- uint64_t fileid;
- @@ -139,6 +155,7 @@ static int attr_cache_entry_create(
- entry = attr_entry(cache->free_entries.next);
- list_remove(&entry->free_entry);
- + entry->nc_attrs = 0;
- entry->fileid = fileid;
- entry->invalidated = FALSE;
- entry->delegated = FALSE;
- @@ -289,52 +306,73 @@ static void attr_cache_update(
- {
- /* update the attributes present in mask */
- if (info->attrmask.count > 0) {
- - if (info->attrmask.arr[0] & FATTR4_WORD0_TYPE)
- + if (info->attrmask.arr[0] & FATTR4_WORD0_TYPE) {
- + entry->nc_attrs |= NC_ATTR_TYPE;
- entry->type = (unsigned char)(info->type & NFS_FTYPE_MASK);
- + }
- if (info->attrmask.arr[0] & FATTR4_WORD0_CHANGE) {
- + entry->nc_attrs |= NC_ATTR_CHANGE;
- entry->change = info->change;
- /* revalidate whenever we get a change attribute */
- entry->invalidated = 0;
- entry->expiration = UTIL_GETRELTIME() + NAME_CACHE_EXPIRATION;
- }
- if (info->attrmask.arr[0] & FATTR4_WORD0_FSID) {
- + entry->nc_attrs |= NC_ATTR_FSID;
- entry->fsid_major = info->fsid.major;
- entry->fsid_minor = info->fsid.minor;
- }
- - if (info->attrmask.arr[0] & FATTR4_WORD0_SIZE)
- + if (info->attrmask.arr[0] & FATTR4_WORD0_SIZE) {
- + entry->nc_attrs |= NC_ATTR_SIZE;
- entry->size = info->size;
- - if (info->attrmask.arr[0] & FATTR4_WORD0_HIDDEN)
- + }
- + if (info->attrmask.arr[0] & FATTR4_WORD0_HIDDEN) {
- + entry->nc_attrs |= NC_ATTR_HIDDEN;
- entry->hidden = info->hidden;
- - if (info->attrmask.arr[0] & FATTR4_WORD0_ARCHIVE)
- + }
- + if (info->attrmask.arr[0] & FATTR4_WORD0_ARCHIVE) {
- + entry->nc_attrs |= NC_ATTR_ARCHIVE;
- entry->archive = info->archive;
- + }
- }
- if (info->attrmask.count > 1) {
- - if (info->attrmask.arr[1] & FATTR4_WORD1_MODE)
- + if (info->attrmask.arr[1] & FATTR4_WORD1_MODE) {
- + entry->nc_attrs |= NC_ATTR_MODE;
- entry->mode = info->mode;
- + }
- if (info->attrmask.arr[1] & FATTR4_WORD1_OWNER) {
- + entry->nc_attrs |= NC_ATTR_OWNER;
- EASSERT(info->owner != NULL);
- (void)strcpy(entry->owner, info->owner);
- }
- if (info->attrmask.arr[1] & FATTR4_WORD1_OWNER_GROUP) {
- + entry->nc_attrs |= NC_ATTR_OWNER_GROUP;
- EASSERT(info->owner_group != NULL);
- (void)strcpy(entry->owner_group, info->owner_group);
- }
- - if (info->attrmask.arr[1] & FATTR4_WORD1_NUMLINKS)
- + if (info->attrmask.arr[1] & FATTR4_WORD1_NUMLINKS) {
- + entry->nc_attrs |= NC_ATTR_NUMLINKS;
- entry->numlinks = info->numlinks;
- + }
- if (info->attrmask.arr[1] & FATTR4_WORD1_TIME_ACCESS) {
- - entry->time_access_s = info->time_access.seconds;
- + entry->nc_attrs |= NC_ATTR_TIME_ACCESS;
- + entry->time_access_s = info->time_access.seconds;
- entry->time_access_ns = info->time_access.nseconds;
- }
- if (info->attrmask.arr[1] & FATTR4_WORD1_TIME_CREATE) {
- - entry->time_create_s = info->time_create.seconds;
- + entry->nc_attrs |= NC_ATTR_TIME_CREATE;
- + entry->time_create_s = info->time_create.seconds;
- entry->time_create_ns = info->time_create.nseconds;
- }
- if (info->attrmask.arr[1] & FATTR4_WORD1_TIME_MODIFY) {
- - entry->time_modify_s = info->time_modify.seconds;
- + entry->nc_attrs |= NC_ATTR_TIME_MODIFY;
- + entry->time_modify_s = info->time_modify.seconds;
- entry->time_modify_ns = info->time_modify.nseconds;
- }
- - if (info->attrmask.arr[1] & FATTR4_WORD1_SYSTEM)
- + if (info->attrmask.arr[1] & FATTR4_WORD1_SYSTEM) {
- + entry->nc_attrs |= NC_ATTR_SYSTEM;
- entry->system = info->system;
- + }
- }
- if (is_delegation(delegation))
- @@ -345,36 +383,49 @@ static void copy_attrs(
- OUT nfs41_file_info *dst,
- IN const struct attr_cache_entry *src)
- {
- - dst->attrmask.count = 2;
- - dst->attrmask.arr[0] = FATTR4_WORD0_TYPE | FATTR4_WORD0_CHANGE
- - | FATTR4_WORD0_SIZE | FATTR4_WORD0_FSID | FATTR4_WORD0_FILEID
- - | FATTR4_WORD0_HIDDEN | FATTR4_WORD0_ARCHIVE;
- - dst->attrmask.arr[1] = FATTR4_WORD1_MODE
- - | FATTR4_WORD1_NUMLINKS
- - | FATTR4_WORD1_SYSTEM;
- -
- - dst->change = src->change;
- - dst->size = src->size;
- - if (!((src->time_access_s == 0) && (src->time_access_ns == 0))) {
- + dst->attrmask.arr[0] = 0;
- + dst->attrmask.arr[1] = 0;
- +
- + dst->attrmask.arr[0] |= FATTR4_WORD0_FILEID;
- + dst->fileid = src->fileid;
- +
- + if (src->nc_attrs & NC_ATTR_CHANGE) {
- + dst->attrmask.arr[0] |= FATTR4_WORD0_CHANGE;
- + dst->change = src->change;
- + }
- + if (src->nc_attrs & NC_ATTR_SIZE) {
- + dst->attrmask.arr[0] |= FATTR4_WORD0_SIZE;
- + dst->size = src->size;
- + }
- + if (src->nc_attrs & NC_ATTR_TIME_ACCESS) {
- dst->attrmask.arr[1] |= FATTR4_WORD1_TIME_ACCESS;
- - dst->time_access.seconds = src->time_access_s;
- - dst->time_access.nseconds = src->time_access_ns;
- + dst->time_access.seconds = src->time_access_s;
- + dst->time_access.nseconds = src->time_access_ns;
- }
- - if (!((src->time_create_s == 0) && (src->time_create_ns == 0))) {
- + if (src->nc_attrs & NC_ATTR_TIME_CREATE) {
- dst->attrmask.arr[1] |= FATTR4_WORD1_TIME_CREATE;
- - dst->time_create.seconds = src->time_create_s;
- - dst->time_create.nseconds = src->time_create_ns;
- + dst->time_create.seconds = src->time_create_s;
- + dst->time_create.nseconds = src->time_create_ns;
- }
- - if (!((src->time_modify_s == 0) && (src->time_modify_ns == 0))) {
- + if (src->nc_attrs & NC_ATTR_TIME_MODIFY) {
- dst->attrmask.arr[1] |= FATTR4_WORD1_TIME_MODIFY;
- - dst->time_modify.seconds = src->time_modify_s;
- - dst->time_modify.nseconds = src->time_modify_ns;
- + dst->time_modify.seconds = src->time_modify_s;
- + dst->time_modify.nseconds = src->time_modify_ns;
- }
- - dst->type = src->type;
- - dst->numlinks = src->numlinks;
- - dst->mode = src->mode;
- -
- - if (src->owner[0] != '\0') {
- + if (src->nc_attrs & NC_ATTR_TYPE) {
- + dst->attrmask.arr[0] |= FATTR4_WORD0_TYPE;
- + dst->type = src->type;
- + }
- + if (src->nc_attrs & NC_ATTR_NUMLINKS) {
- + dst->attrmask.arr[1] |= FATTR4_WORD1_NUMLINKS;
- + dst->numlinks = src->numlinks;
- + }
- + if (src->nc_attrs & NC_ATTR_MODE) {
- + dst->attrmask.arr[1] |= FATTR4_WORD1_MODE;
- + dst->mode = src->mode;
- + }
- + if (src->nc_attrs & NC_ATTR_OWNER) {
- + dst->attrmask.arr[1] |= FATTR4_WORD1_OWNER;
- dst->owner = dst->owner_buf;
- (void)strcpy(dst->owner, src->owner);
- }
- @@ -382,8 +433,8 @@ static void copy_attrs(
- /* this should only happen for newly created files/dirs */
- dst->owner = NULL;
- }
- -
- - if (src->owner_group[0] != '\0') {
- + if (src->nc_attrs & NC_ATTR_OWNER_GROUP) {
- + dst->attrmask.arr[1] |= FATTR4_WORD1_OWNER_GROUP;
- dst->owner_group = dst->owner_group_buf;
- (void)strcpy(dst->owner_group, src->owner_group);
- }
- @@ -391,17 +442,30 @@ static void copy_attrs(
- /* this should only happen for newly created files/dirs */
- dst->owner_group = NULL;
- }
- - dst->fileid = src->fileid;
- - dst->fsid.major = src->fsid_major;
- - dst->fsid.minor = src->fsid_minor;
- - dst->hidden = src->hidden;
- - dst->system = src->system;
- - dst->archive = src->archive;
- + if (src->nc_attrs & NC_ATTR_FSID) {
- + dst->attrmask.arr[0] |= FATTR4_WORD0_FSID;
- + dst->fsid.major = src->fsid_major;
- + dst->fsid.minor = src->fsid_minor;
- + }
- + if (src->nc_attrs & NC_ATTR_HIDDEN) {
- + dst->attrmask.arr[0] |= FATTR4_WORD0_HIDDEN;
- + dst->hidden = src->hidden;
- + }
- + if (src->nc_attrs & NC_ATTR_ARCHIVE) {
- + dst->attrmask.arr[0] |= FATTR4_WORD0_ARCHIVE;
- + dst->archive = src->archive;
- + }
- + if (src->nc_attrs & NC_ATTR_SYSTEM) {
- + dst->attrmask.arr[1] |= FATTR4_WORD1_SYSTEM;
- + dst->system = src->system;
- + }
- - if (dst->owner)
- - dst->attrmask.arr[1] |= FATTR4_WORD1_OWNER;
- - if (dst->owner_group)
- - dst->attrmask.arr[1] |= FATTR4_WORD1_OWNER_GROUP;
- + if (dst->attrmask.arr[1] != 0) {
- + dst->attrmask.count = 2;
- + }
- + else {
- + dst->attrmask.count = 1;
- + }
- }
- --
- 2.45.1
- From ddfbbc8c53076df23dc642ce92a179d7de3d08f7 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 22 Jan 2025 11:48:51 +0100
- Subject: [PATCH 40/44] cygwin: Update drmemory flags for better test coverage
- and performance
- Update drmemory flags for better test coverage and performance.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- cygwin/devel/msnfs41client.bash | 6 ++++++
- 1 file changed, 6 insertions(+)
- diff --git a/cygwin/devel/msnfs41client.bash b/cygwin/devel/msnfs41client.bash
- index 4699afb..92171fd 100755
- --- a/cygwin/devel/msnfs41client.bash
- +++ b/cygwin/devel/msnfs41client.bash
- @@ -465,12 +465,15 @@ function nfsclient_rundeamon
- '-delay_frees_maxsz' $((64*1024*1024))
- '-redzone_size' '4096'
- '-check_uninitialized'
- + '-check_uninit_non_moves'
- '-check_uninit_all'
- '-strict_bitops'
- '-gen_suppress_syms'
- '-preload_symbols'
- # no symbol cache, user "SYSTEM" cannot write data to cache
- '-no_use_symcache'
- + # disable leak checking for performance
- + '-no_check_leaks'
- '--'
- "$(cygpath -w "$(which "${nfsd_args[0]}")")"
- "${nfsd_args[@]:1}"
- @@ -597,12 +600,15 @@ function nfsclient_system_rundeamon
- '-delay_frees_maxsz' $((64*1024*1024))
- '-redzone_size' '4096'
- '-check_uninitialized'
- + '-check_uninit_non_moves'
- '-check_uninit_all'
- '-strict_bitops'
- '-gen_suppress_syms'
- '-preload_symbols'
- # no symbol cache, user "SYSTEM" cannot write data to cache
- '-no_use_symcache'
- + # disable leak checking for performance
- + '-no_check_leaks'
- '--'
- "$(cygpath -w "$(which "${nfsd_args[0]}")")"
- "${nfsd_args[@]:1}"
- --
- 2.45.1
- From c79eb1992013052f352a0bb9f5b5cb3c71853f14 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 22 Jan 2025 12:30:23 +0100
- Subject: [PATCH 41/44] daemon: |bitmap_isset()| should return |true|/|false|
- |bitmap_isset()| should return |true|/|false|, and not
- an integer value based on |mask->arr[word] & flag| or |0|.
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/util.h | 4 ++--
- 1 file changed, 2 insertions(+), 2 deletions(-)
- diff --git a/daemon/util.h b/daemon/util.h
- index b536f48..829cbba 100644
- --- a/daemon/util.h
- +++ b/daemon/util.h
- @@ -76,12 +76,12 @@ bool_t verify_commit(
- IN nfs41_write_verf *verf);
- /* bitmap4 */
- -static __inline bool_t bitmap_isset(
- +static __inline bool bitmap_isset(
- IN const bitmap4 *mask,
- IN uint32_t word,
- IN uint32_t flag)
- {
- - return mask->count > word && mask->arr[word] & flag;
- + return ((mask->count > word) && (mask->arr[word] & flag))?true:false;
- }
- static __inline void bitmap_set(
- IN bitmap4 *mask,
- --
- 2.45.1
- From 1ee25090cee142291e3b1c874f844f5d1d93e931 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 22 Jan 2025 12:36:08 +0100
- Subject: [PATCH 42/44] daemon: |bitmap_*()| functions should use |restrict|
- |bitmap_*()| functions should use |restrict|
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/util.h | 10 +++++-----
- 1 file changed, 5 insertions(+), 5 deletions(-)
- diff --git a/daemon/util.h b/daemon/util.h
- index 829cbba..1e73110 100644
- --- a/daemon/util.h
- +++ b/daemon/util.h
- @@ -77,14 +77,14 @@ bool_t verify_commit(
- /* bitmap4 */
- static __inline bool bitmap_isset(
- - IN const bitmap4 *mask,
- + IN const bitmap4 *restrict mask,
- IN uint32_t word,
- IN uint32_t flag)
- {
- return ((mask->count > word) && (mask->arr[word] & flag))?true:false;
- }
- static __inline void bitmap_set(
- - IN bitmap4 *mask,
- + IN bitmap4 *restrict mask,
- IN uint32_t word,
- IN uint32_t flag)
- {
- @@ -96,7 +96,7 @@ static __inline void bitmap_set(
- }
- }
- static __inline void bitmap_unset(
- - IN bitmap4 *mask,
- + IN bitmap4 *restrict mask,
- IN uint32_t word,
- IN uint32_t flag)
- {
- @@ -107,8 +107,8 @@ static __inline void bitmap_unset(
- }
- }
- static __inline void bitmap_intersect(
- - IN bitmap4 *dst,
- - IN const bitmap4 *src)
- + IN bitmap4 *restrict dst,
- + IN const bitmap4 *restrict src)
- {
- uint32_t i, count = 0;
- for (i = 0; i < 3; i++) {
- --
- 2.45.1
- From 6bb3cdf7c6b43ed775bb488acda79690c1a3c6d4 Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Wed, 22 Jan 2025 12:49:37 +0100
- Subject: [PATCH 43/44] daemon: fileinfoutil&co should use |restrict|
- fileinfoutil&co should use |restrict|
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- daemon/fileinfoutil.c | 32 ++++++++++++-------------
- daemon/util.h | 56 +++++++++++++++++++++----------------------
- 2 files changed, 44 insertions(+), 44 deletions(-)
- diff --git a/daemon/fileinfoutil.c b/daemon/fileinfoutil.c
- index bddbfd3..f86aa5a 100644
- --- a/daemon/fileinfoutil.c
- +++ b/daemon/fileinfoutil.c
- @@ -35,7 +35,7 @@
- ULONG nfs_file_info_to_attributes(
- - IN const nfs41_file_info *info)
- + IN const nfs41_file_info *restrict info)
- {
- ULONG attrs = 0;
- @@ -74,8 +74,8 @@ ULONG nfs_file_info_to_attributes(
- }
- void nfs_to_basic_info(
- - IN const char *name,
- - IN const nfs41_file_info *info,
- + IN const char *restrict name,
- + IN const nfs41_file_info *restrict info,
- OUT PFILE_BASIC_INFO basic_out)
- {
- EASSERT(info->attrmask.count > 0);
- @@ -121,8 +121,8 @@ void nfs_to_basic_info(
- }
- void nfs_to_standard_info(
- - IN const nfs41_file_info *info,
- - OUT PFILE_STANDARD_INFO std_out)
- + IN const nfs41_file_info *restrict info,
- + OUT PFILE_STANDARD_INFO restrict std_out)
- {
- const ULONG FileAttributes = nfs_file_info_to_attributes(info);
- @@ -139,9 +139,9 @@ void nfs_to_standard_info(
- }
- void nfs_to_network_openinfo(
- - IN const char *name,
- - IN const nfs41_file_info *info,
- - OUT PFILE_NETWORK_OPEN_INFORMATION net_out)
- + IN const char *restrict name,
- + IN const nfs41_file_info *restrict info,
- + OUT PFILE_NETWORK_OPEN_INFORMATION restrict net_out)
- {
- EASSERT(info->attrmask.count > 0);
- @@ -189,9 +189,9 @@ void nfs_to_network_openinfo(
- #ifdef NFS41_DRIVER_WSL_SUPPORT
- void nfs_to_stat_info(
- - IN const char *name,
- - IN const nfs41_file_info *info,
- - OUT PFILE_STAT_INFORMATION stat_out)
- + IN const char *restrict name,
- + IN const nfs41_file_info *restrict info,
- + OUT PFILE_STAT_INFORMATION restrict stat_out)
- {
- EASSERT(info->attrmask.count > 0);
- @@ -249,9 +249,9 @@ void nfs_to_stat_info(
- void nfs_to_stat_lx_info(
- IN void *daemon_context,
- - IN const char *name,
- - IN const nfs41_file_info *info,
- - OUT PFILE_STAT_LX_INFORMATION stat_lx_out)
- + IN const char *restrict name,
- + IN const nfs41_file_info *restrict info,
- + OUT PFILE_STAT_LX_INFORMATION restrict stat_lx_out)
- {
- nfs41_daemon_globals *nfs41_dg =
- (nfs41_daemon_globals *)daemon_context;
- @@ -445,8 +445,8 @@ void nfs_to_stat_lx_info(
- /* copy |nfs41_file_info| */
- void nfs41_file_info_cpy(
- - OUT nfs41_file_info *dest,
- - IN const nfs41_file_info *src,
- + OUT nfs41_file_info *restrict dest,
- + IN const nfs41_file_info *restrict src,
- IN int flags)
- {
- const bitmap4 *attrmask = &src->attrmask;
- diff --git a/daemon/util.h b/daemon/util.h
- index 1e73110..c5bbb1b 100644
- --- a/daemon/util.h
- +++ b/daemon/util.h
- @@ -174,35 +174,35 @@ static __inline void open_delegation4_cpy(
- }
- ULONG nfs_file_info_to_attributes(
- - IN const nfs41_file_info *info);
- + IN const nfs41_file_info *restrict info);
- void nfs_to_basic_info(
- - IN const char *name,
- - IN const nfs41_file_info *info,
- - OUT PFILE_BASIC_INFO basic_out);
- + IN const char *restrict name,
- + IN const nfs41_file_info *restrict info,
- + OUT PFILE_BASIC_INFO restrict basic_out);
- void nfs_to_standard_info(
- - IN const nfs41_file_info *info,
- - OUT PFILE_STANDARD_INFO std_out);
- + IN const nfs41_file_info *restrict info,
- + OUT PFILE_STANDARD_INFO restrict std_out);
- void nfs_to_network_openinfo(
- - IN const char *name,
- - IN const nfs41_file_info *info,
- - OUT PFILE_NETWORK_OPEN_INFORMATION std_out);
- + IN const char *restrict name,
- + IN const nfs41_file_info *restrict info,
- + OUT PFILE_NETWORK_OPEN_INFORMATION restrict std_out);
- #ifdef NFS41_DRIVER_WSL_SUPPORT
- void nfs_to_stat_info(
- - IN const char *name,
- - IN const nfs41_file_info *info,
- - OUT PFILE_STAT_INFORMATION stat_out);
- + IN const char *restrict name,
- + IN const nfs41_file_info *restrict info,
- + OUT PFILE_STAT_INFORMATION restrict stat_out);
- void nfs_to_stat_lx_info(
- IN void *daemon_context,
- - IN const char *name,
- - IN const nfs41_file_info *info,
- - OUT PFILE_STAT_LX_INFORMATION stat_lx_out);
- + IN const char *restrict name,
- + IN const nfs41_file_info *restrict info,
- + OUT PFILE_STAT_LX_INFORMATION restrict stat_lx_out);
- #endif /* NFS41_DRIVER_WSL_SUPPORT */
- /* Copy |info->symlink_dir| */
- #define NFS41FILEINFOCPY_COPY_SYMLINK_DIR (1 << 0)
- void nfs41_file_info_cpy(
- - OUT nfs41_file_info *dest,
- - IN const nfs41_file_info *src,
- + OUT nfs41_file_info *restrict dest,
- + IN const nfs41_file_info *restrict src,
- IN int flags);
- /* http://msdn.microsoft.com/en-us/library/ms724290%28VS.85%29.aspx:
- @@ -212,8 +212,8 @@ void nfs41_file_info_cpy(
- #define FILETIME_EPOCH 116444736000000000LL
- static __inline void file_time_to_nfs_time(
- - IN const PLARGE_INTEGER file_time,
- - OUT nfstime4 *nfs_time)
- + IN const PLARGE_INTEGER restrict file_time,
- + OUT nfstime4 *restrict nfs_time)
- {
- LONGLONG diff = file_time->QuadPart - FILETIME_EPOCH;
- nfs_time->seconds = diff / 10000000;
- @@ -221,8 +221,8 @@ static __inline void file_time_to_nfs_time(
- }
- static __inline void nfs_time_to_file_time(
- - IN const nfstime4 *nfs_time,
- - OUT PLARGE_INTEGER file_time)
- + IN const nfstime4 *restrict nfs_time,
- + OUT PLARGE_INTEGER restrict file_time)
- {
- file_time->QuadPart = FILETIME_EPOCH +
- nfs_time->seconds * 10000000 +
- @@ -230,12 +230,12 @@ static __inline void nfs_time_to_file_time(
- }
- void get_file_time(
- - OUT PLARGE_INTEGER file_time);
- + OUT PLARGE_INTEGER restrict file_time);
- void get_nfs_time(
- - OUT nfstime4 *nfs_time);
- + OUT nfstime4 *restrict nfs_time);
- static __inline void nfstime_normalize(
- - IN OUT nfstime4 *nfstime)
- + IN OUT nfstime4 *restrict nfstime)
- {
- /* return time in normalized form (0 <= nsec < 1s) */
- while ((int32_t)nfstime->nseconds < 0) {
- @@ -244,8 +244,8 @@ static __inline void nfstime_normalize(
- }
- }
- static __inline void nfstime_diff(
- - IN const nfstime4 *lhs,
- - IN const nfstime4 *rhs,
- + IN const nfstime4 *restrict lhs,
- + IN const nfstime4 *restrict rhs,
- OUT nfstime4 *result)
- {
- /* result = lhs - rhs */
- @@ -254,8 +254,8 @@ static __inline void nfstime_diff(
- nfstime_normalize(result);
- }
- static __inline void nfstime_abs(
- - IN const nfstime4 *nt,
- - OUT nfstime4 *result)
- + IN const nfstime4 *restrict nt,
- + OUT nfstime4 *restrict result)
- {
- if (nt->seconds < 0) {
- const nfstime4 zero = { .seconds=0LL, .nseconds=0UL };
- --
- 2.45.1
- From 1f92cdde986948d79320d5a22c031cebc273c2da Mon Sep 17 00:00:00 2001
- From: Roland Mainz <roland.mainz@nrubsig.org>
- Date: Mon, 27 Jan 2025 13:33:10 +0100
- Subject: [PATCH 44/44] build.vc19,daemon: RFE: Implement NFSv4.2 READ_PLUS
- support (read sparse files)
- RFE: Implement NFSv4.2 READ_PLUS support (read sparse files)
- Reported-by: Lionel Cons <lionelcons1972@gmail.com>
- Reported-by: Cedric Blancher <cedric.blancher@gmail.com>
- Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
- ---
- build.vc19/nfsd/nfsd.vcxproj | 2 +
- build.vc19/nfsd/nfsd.vcxproj.filters | 6 +
- daemon/namespace.c | 14 +-
- daemon/nfs41.h | 4 +-
- daemon/nfs41_ops.c | 4 +-
- daemon/nfs41_ops.h | 73 +++++++++-
- daemon/nfs41_xdr.c | 30 ++++-
- daemon/nfs41_xdr.h | 10 +-
- daemon/nfs42_ops.c | 96 ++++++++++++++
- daemon/nfs42_xdr.c | 192 +++++++++++++++++++++++++++
- daemon/readwrite.c | 18 ++-
- daemon/recovery.c | 6 +-
- 12 files changed, 443 insertions(+), 12 deletions(-)
- create mode 100644 daemon/nfs42_ops.c
- create mode 100644 daemon/nfs42_xdr.c
- diff --git a/build.vc19/nfsd/nfsd.vcxproj b/build.vc19/nfsd/nfsd.vcxproj
- index ed5470a..1d9a09f 100644
- --- a/build.vc19/nfsd/nfsd.vcxproj
- +++ b/build.vc19/nfsd/nfsd.vcxproj
- @@ -285,6 +285,8 @@
- <ClCompile Include="..\..\daemon\nfs41_session.c" />
- <ClCompile Include="..\..\daemon\nfs41_superblock.c" />
- <ClCompile Include="..\..\daemon\nfs41_xdr.c" />
- + <ClCompile Include="..\..\daemon\nfs42_ops.c" />
- + <ClCompile Include="..\..\daemon\nfs42_xdr.c" />
- <ClCompile Include="..\..\daemon\open.c" />
- <ClCompile Include="..\..\daemon\pnfs_debug.c" />
- <ClCompile Include="..\..\daemon\pnfs_device.c" />
- diff --git a/build.vc19/nfsd/nfsd.vcxproj.filters b/build.vc19/nfsd/nfsd.vcxproj.filters
- index ffc0e6e..5f6939f 100644
- --- a/build.vc19/nfsd/nfsd.vcxproj.filters
- +++ b/build.vc19/nfsd/nfsd.vcxproj.filters
- @@ -90,6 +90,12 @@
- <ClCompile Include="..\..\daemon\nfs41_xdr.c">
- <Filter>Source Files</Filter>
- </ClCompile>
- + <ClCompile Include="..\..\daemon\nfs42_ops.c">
- + <Filter>Source Files</Filter>
- + </ClCompile>
- + <ClCompile Include="..\..\daemon\nfs42_xdr.c">
- + <Filter>Source Files</Filter>
- + </ClCompile>
- <ClCompile Include="..\..\daemon\open.c">
- <Filter>Source Files</Filter>
- </ClCompile>
- diff --git a/daemon/namespace.c b/daemon/namespace.c
- index 811e715..b30e0eb 100644
- --- a/daemon/namespace.c
- +++ b/daemon/namespace.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) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- @@ -64,6 +65,11 @@ int nfs41_root_create(
- list_init(&root->clients);
- root->use_nfspubfh = use_nfspubfh;
- + /*
- + * nfs41_root_mount_addrs() will enable NFSv4.2 features (like
- + * |OP_READ_PLUS|) after NFSv4.x minor version autonegitiation
- + */
- + root->supports_nfs42_read_plus = false;
- if (nfsvers == NFS_VERSION_AUTONEGOTIATION) {
- /*
- * Use auto negotiation, |nfs41_root_mount_addrs()| will
- @@ -433,6 +439,12 @@ retry_nfs41_exchange_id:
- goto out_free_rpc;
- }
- + /* Enable NFS features after NFSv4.x minor version negotiation */
- + if (root->nfsminorvers >= 2) {
- + DPRINTF(0, ("nfs41_root_mount_addrs: Enabling OP_READ_PLUS\n"));
- + root->supports_nfs42_read_plus = true;
- + }
- +
- /* attempt to match existing clients by the exchangeid response */
- EnterCriticalSection(&root->lock);
- status = root_client_find(root, &exchangeid, is_data, &client);
- diff --git a/daemon/nfs41.h b/daemon/nfs41.h
- index e6e82ad..666c4ae 100644
- --- a/daemon/nfs41.h
- +++ b/daemon/nfs41.h
- @@ -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) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- @@ -300,6 +301,7 @@ typedef struct __nfs41_root {
- CRITICAL_SECTION lock;
- struct list_entry clients;
- bool use_nfspubfh;
- + bool supports_nfs42_read_plus;
- DWORD nfsminorvers;
- uint32_t wsize;
- uint32_t rsize;
- diff --git a/daemon/nfs41_ops.c b/daemon/nfs41_ops.c
- index a4e059d..a384175 100644
- --- a/daemon/nfs41_ops.c
- +++ b/daemon/nfs41_ops.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) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- @@ -908,6 +909,7 @@ out:
- return status;
- }
- +
- int nfs41_commit(
- IN nfs41_session *session,
- IN nfs41_path_fh *file,
- diff --git a/daemon/nfs41_ops.h b/daemon/nfs41_ops.h
- index 81bad50..2e2e49a 100644
- --- a/daemon/nfs41_ops.h
- +++ b/daemon/nfs41_ops.h
- @@ -1,8 +1,10 @@
- /* 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) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- + * Roland Mainz <roland.mainz@nrubsig.org>
- *
- * This library is free software; you can redistribute it and/or modify it
- * under the terms of the GNU Lesser General Public License as published by
- @@ -726,6 +728,65 @@ typedef struct __nfs41_read_res {
- nfs41_read_res_ok resok4;
- } nfs41_read_res;
- +typedef struct __data4 {
- + uint64_t offset;
- + uint32_t count;
- + uint32_t data_len;
- + unsigned char *data;
- +} data4;
- +
- +typedef struct __data_info4 {
- + uint64_t offset;
- + uint64_t length;
- +} data_info4;
- +
- +typedef enum __data_content4 {
- + NFS4_CONTENT_DATA = 0,
- + NFS4_CONTENT_HOLE = 1
- +} data_content4;
- +
- +/* OP_READ_PLUS */
- +typedef struct __nfs42_read_plus_args {
- + stateid_arg *stateid; /* -> nfs41_op_open_res_ok.stateid */
- + uint64_t offset;
- + uint32_t count;
- +} nfs42_read_plus_args;
- +
- +typedef union __nfs42_read_plus_content {
- + /* switch (data_content4 rpc_content) */
- + data_content4 content;
- + /* case NFS4_CONTENT_DATA: */
- + data4 data;
- + /* case NFS4_CONTENT_HOLE: */
- + data_info4 hole;
- + /* default: */
- +} nfs42_read_plus_content;
- +
- +typedef struct __nfs42_read_plus_res_ok {
- + /*
- + * HACK: Copy of |nfs42_read_plus_args.offset| - needed because
- + * |nfs_op_decode_proc()| only has a |nfs_resop4| argument but
- + * no |nfs_argop4| argument (yet)
- + */
- + uint64_t args_offset;
- +
- + bool_t eof;
- + uint32_t count;
- + /*
- + * We don't define a |nfs42_read_plus_content *contents| member
- + * here, because |decode_read_plus_res_ok()| only uses this
- + * internally and does not have an external consumer, saving a
- + * |malloc()|+|free()| in this case
- + */
- + unsigned char *data; /* caller-allocated */
- + uint32_t data_len;
- +} nfs42_read_plus_res_ok;
- +
- +typedef struct __nfs42_read_plus_res {
- + uint32_t status;
- + /* case NFS4_OK: */
- + nfs42_read_plus_res_ok resok4;
- +} nfs42_read_plus_res;
- /* OP_READDIR */
- typedef struct __nfs41_readdir_args {
- @@ -1112,6 +1173,16 @@ int nfs41_read(
- OUT uint32_t *data_len_out,
- OUT bool_t *eof_out);
- +int nfs42_read_plus(
- + IN nfs41_session *session,
- + IN nfs41_path_fh *file,
- + IN stateid_arg *stateid,
- + IN uint64_t offset,
- + IN uint32_t count,
- + OUT unsigned char *data_out,
- + OUT uint32_t *data_len_out,
- + OUT bool_t *eof_out);
- +
- int nfs41_commit(
- IN nfs41_session *session,
- IN nfs41_path_fh *file,
- diff --git a/daemon/nfs41_xdr.c b/daemon/nfs41_xdr.c
- index f899b7b..1108950 100644
- --- a/daemon/nfs41_xdr.c
- +++ b/daemon/nfs41_xdr.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) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- @@ -131,7 +132,7 @@ static bool_t xdr_settime4(
- }
- /* stateid4 */
- -static bool_t xdr_stateid4(
- +bool_t xdr_stateid4(
- XDR *xdr,
- stateid4 *si)
- {
- @@ -2262,11 +2263,32 @@ static bool_t decode_read_res_ok(
- nfs41_read_res_ok *res)
- {
- unsigned char *data = res->data;
- + uint32_t data_len = res->data_len;
- + uint32_t count;
- if (!xdr_bool(xdr, &res->eof))
- return FALSE;
- - return xdr_bytes(xdr, (char **)&data, &res->data_len, NFS41_MAX_FILEIO_SIZE);
- + if (!xdr_u_int32_t(xdr, &count)) {
- + DPRINTF(0, ("decode_read_res_ok: decoding 'count' failed\n"));
- + return FALSE;
- + }
- +
- + EASSERT(count <= data_len);
- + /*
- + * If a buggy server erroneously sends more data then we
- + * requested we'll clamp this via |__min()| to avoid an buffer
- + * overflow (but will still get an RPC error later).
- + */
- + count = __min(data_len, count);
- + if (!xdr_opaque(xdr, (char *)data, count)) {
- + DPRINTF(0, ("decode_read_res_ok_ decoding 'bytes' failed\n"));
- + return FALSE;
- + }
- +
- + res->data_len = count;
- +
- + return TRUE;
- }
- static bool_t decode_op_read(
- @@ -3638,7 +3660,7 @@ static const op_table_entry g_op_table[] = {
- { NULL, NULL }, /* OP_LAYOUTSTATS = 65, */
- { NULL, NULL }, /* OP_OFFLOAD_CANCEL = 66, */
- { NULL, NULL }, /* OP_OFFLOAD_STATUS = 67, */
- - { NULL, NULL }, /* OP_READ_PLUS = 68, */
- + { encode_op_read_plus, decode_op_read_plus }, /* OP_READ_PLUS = 68, */
- { NULL, NULL }, /* OP_SEEK = 69, */
- { NULL, NULL }, /* OP_WRITE_SAME = 70, */
- { NULL, NULL }, /* OP_CLONE = 71, */
- diff --git a/daemon/nfs41_xdr.h b/daemon/nfs41_xdr.h
- index 9a3fc61..0b320bd 100644
- --- a/daemon/nfs41_xdr.h
- +++ b/daemon/nfs41_xdr.h
- @@ -1,8 +1,10 @@
- /* 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) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- + * Roland Mainz <roland.mainz@nrubsig.org>
- *
- * This library is free software; you can redistribute it and/or modify it
- * under the terms of the GNU Lesser General Public License as published by
- @@ -23,10 +25,16 @@
- #define __NFS41_NFS_XDR_H__
- #include "nfs41_types.h"
- +#include "nfs41_compound.h"
- bool_t nfs_encode_compound(XDR *xdr, caddr_t *args);
- bool_t nfs_decode_compound(XDR *xdr, caddr_t *res);
- void nfsacl41_free(nfsacl41 *acl);
- +bool_t xdr_stateid4(XDR *xdr, stateid4 *si);
- +
- +/* NFSv4.2 ops */
- +bool_t encode_op_read_plus(XDR *xdr, nfs_argop4 *argop);
- +bool_t decode_op_read_plus(XDR *xdr, nfs_resop4 *resop);
- #endif /* !__NFS41_NFS_XDR_H__ */
- diff --git a/daemon/nfs42_ops.c b/daemon/nfs42_ops.c
- new file mode 100644
- index 0000000..287fd2e
- --- /dev/null
- +++ b/daemon/nfs42_ops.c
- @@ -0,0 +1,96 @@
- +/* NFSv4.1 client for Windows
- + * Copyright (C) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- + *
- + * Roland Mainz <roland.mainz@nrubsig.org>
- + *
- + * This library is free software; you can redistribute it and/or modify it
- + * under the terms of the GNU Lesser General Public License as published by
- + * the Free Software Foundation; either version 2.1 of the License, or (at
- + * your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful, but
- + * without any warranty; without even the implied warranty of merchantability
- + * or fitness for a particular purpose. See the GNU Lesser General Public
- + * License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public License
- + * along with this library; if not, write to the Free Software Foundation,
- + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
- + */
- +
- +#include <Windows.h>
- +#include <strsafe.h>
- +#include <stdio.h>
- +#include <stdlib.h>
- +#include <time.h>
- +
- +#include "nfs41_build_features.h"
- +#include "nfs41_daemon.h"
- +#include "nfs41_ops.h"
- +#include "nfs41_compound.h"
- +#include "nfs41_xdr.h"
- +#include "name_cache.h"
- +#include "delegation.h"
- +#include "daemon_debug.h"
- +#include "util.h"
- +
- +int nfs42_read_plus(
- + IN nfs41_session *session,
- + IN nfs41_path_fh *file,
- + IN stateid_arg *stateid,
- + IN uint64_t offset,
- + IN uint32_t count,
- + OUT unsigned char *data_out,
- + OUT uint32_t *data_len_out,
- + OUT bool_t *eof_out)
- +{
- + int status;
- + nfs41_compound compound;
- + nfs_argop4 argops[4];
- + nfs_resop4 resops[4];
- + nfs41_sequence_args sequence_args;
- + nfs41_sequence_res sequence_res;
- + nfs41_putfh_args putfh_args;
- + nfs41_putfh_res putfh_res;
- + nfs42_read_plus_args read_plus_args;
- + nfs42_read_plus_res read_plus_res;
- +
- + compound_init(&compound, session->client->root->nfsminorvers,
- + argops, resops,
- + stateid->stateid.seqid == 0 ? "ds read_plus" : "read_plus");
- +
- + compound_add_op(&compound, OP_SEQUENCE, &sequence_args, &sequence_res);
- + nfs41_session_sequence(&sequence_args, session, 0);
- +
- + compound_add_op(&compound, OP_PUTFH, &putfh_args, &putfh_res);
- + putfh_args.file = file;
- + putfh_args.in_recovery = 0;
- +
- + compound_add_op(&compound, OP_READ_PLUS, &read_plus_args, &read_plus_res);
- + read_plus_args.stateid = stateid;
- + read_plus_args.offset = offset;
- + read_plus_args.count = count;
- + read_plus_res.resok4.args_offset = offset; /* hack */
- + read_plus_res.resok4.data_len = count;
- + read_plus_res.resok4.data = data_out;
- +
- + status = compound_encode_send_decode(session, &compound, TRUE);
- + if (status)
- + goto out;
- +
- + if (compound_error(status = compound.res.status))
- + goto out;
- +
- + *data_len_out = read_plus_res.resok4.data_len;
- + *eof_out = read_plus_res.resok4.eof;
- +
- + /* we shouldn't ever see this, but a buggy server could
- + * send us into an infinite loop. return NFS4ERR_IO */
- + if (!read_plus_res.resok4.data_len && !read_plus_res.resok4.eof) {
- + status = NFS4ERR_IO;
- + eprintf("READ_PLUS succeeded with len=0 and eof=0; returning '%s'\n",
- + nfs_error_string(status));
- + }
- +out:
- + return status;
- +}
- diff --git a/daemon/nfs42_xdr.c b/daemon/nfs42_xdr.c
- new file mode 100644
- index 0000000..ab56c19
- --- /dev/null
- +++ b/daemon/nfs42_xdr.c
- @@ -0,0 +1,192 @@
- +/* NFSv4.1 client for Windows
- + * Copyright (C) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- + *
- + * Roland Mainz <roland.mainz@nrubsig.org>
- + *
- + * This library is free software; you can redistribute it and/or modify it
- + * under the terms of the GNU Lesser General Public License as published by
- + * the Free Software Foundation; either version 2.1 of the License, or (at
- + * your option) any later version.
- + *
- + * This library is distributed in the hope that it will be useful, but
- + * without any warranty; without even the implied warranty of merchantability
- + * or fitness for a particular purpose. See the GNU Lesser General Public
- + * License for more details.
- + *
- + * You should have received a copy of the GNU Lesser General Public License
- + * along with this library; if not, write to the Free Software Foundation,
- + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
- + */
- +
- +#include <Windows.h>
- +#include <strsafe.h>
- +
- +#include "nfs41_compound.h"
- +#include "nfs41_ops.h"
- +#include "nfs41_xdr.h"
- +#include "util.h"
- +#include "daemon_debug.h"
- +#include "rpc/rpc.h"
- +
- +/* fixme: copy from nfs41_xdr.c */
- +static __inline int unexpected_op(uint32_t op, uint32_t expected)
- +{
- + if (op == expected)
- + return 0;
- +
- + eprintf("Op table mismatch. Got '%s' (%d), expected '%s' (%d).\n",
- + nfs_opnum_to_string(op), op,
- + nfs_opnum_to_string(expected), expected);
- + return 1;
- +}
- +
- +/*
- + * OP_READ_PLUS
- + */
- +bool_t encode_op_read_plus(
- + XDR *xdr,
- + nfs_argop4 *argop)
- +{
- + nfs42_read_plus_args *args = (nfs42_read_plus_args *)argop->arg;
- +
- + if (unexpected_op(argop->op, OP_READ_PLUS))
- + return FALSE;
- +
- + if (!xdr_stateid4(xdr, &args->stateid->stateid))
- + return FALSE;
- +
- + if (!xdr_u_hyper(xdr, &args->offset))
- + return FALSE;
- +
- + return xdr_u_int32_t(xdr, &args->count);
- +}
- +
- +static bool_t decode_read_plus_res_ok(
- + XDR *xdr,
- + nfs42_read_plus_res_ok *res)
- +{
- + nfs42_read_plus_content *contents = NULL;
- + uint64_t read_data_len = 0ULL;
- +
- + if (!xdr_bool(xdr, &res->eof)) {
- + DPRINTF(0, ("decode eof failed\n"));
- + return FALSE;
- + }
- +
- + if (!xdr_u_int32_t(xdr, &res->count)) {
- + DPRINTF(0, ("decode count failed\n"));
- + return FALSE;
- + }
- +
- + /*
- + * Note that |res->count==0| is a valid value for "READ_PLUS"
- + * replies
- + */
- + if (res->count == 0) {
- + res->data_len = 0L;
- + return TRUE;
- + }
- +
- + contents = _alloca(res->count * sizeof(nfs42_read_plus_content));
- +
- + uint32_t i, co;
- +
- + for (i = 0 ; i < res->count ; i++) {
- + if (!xdr_u_int32_t(xdr, &co)) {
- + DPRINTF(0, ("i=%d, decode co failed\n", (int)i));
- + return FALSE;
- + }
- + contents[i].content = co;
- +
- + switch(co) {
- + case NFS4_CONTENT_DATA:
- + if (!xdr_u_hyper(xdr, &contents[i].data.offset)) {
- + DPRINTF(0,
- + ("i=%d, decoding 'offset' failed\n", (int)i));
- + return FALSE;
- + }
- + if (!xdr_u_int32_t(xdr, &contents[i].data.count)) {
- + DPRINTF(0,
- + ("i=%d, decoding 'count' failed\n", (int)i));
- + return FALSE;
- + }
- +
- + contents[i].data.data = res->data +
- + (contents[i].data.offset - res->args_offset);
- + contents[i].data.data_len = contents[i].data.count;
- +
- + EASSERT(((contents[i].data.data - res->data) +
- + contents[i].data.data_len) <= res->data_len);
- + if (!xdr_opaque(xdr,
- + (char *)contents[i].data.data,
- + contents[i].data.data_len)) {
- + DPRINTF(0,
- + ("i=%d, decoding 'bytes' failed\n", (int)i));
- + return FALSE;
- + }
- + read_data_len = __max(read_data_len,
- + ((size_t)(contents[i].data.data - res->data) +
- + contents[i].data.data_len));
- + break;
- + case NFS4_CONTENT_HOLE:
- + unsigned char *hole_buff;
- + uint64_t hole_length;
- +
- + DPRINTF(0,
- + ("i=%d, 'NFS4_CONTENT_HOLE' content\n", (int)i));
- + if (!xdr_u_hyper(xdr, &contents[i].hole.offset))
- + return FALSE;
- + if (!xdr_u_hyper(xdr, &contents[i].hole.length))
- + return FALSE;
- +
- +
- + hole_buff = res->data +
- + (contents[i].hole.offset - res->args_offset);
- + hole_length = contents[i].hole.length;
- +
- + /*
- + * NFSv4.2 "READ_PLUS" is required to return the
- + * whole hole even if |hole.length| is bigger than
- + * the requested size
- + */
- + if (((hole_buff - res->data) + hole_length) >
- + res->data_len) {
- + hole_length = res->data_len;
- + }
- +
- + EASSERT(hole_length < UINT_MAX);
- + (void)memset(hole_buff, 0, (size_t)hole_length);
- +
- + read_data_len = __max(read_data_len,
- + ((hole_buff - res->data) + hole_length));
- + break;
- + default:
- + eprintf("decode_read_plus_res_ok: unknown co=%d\n",
- + (int)co);
- + return FALSE;
- + }
- + }
- +
- + EASSERT(read_data_len < UINT_MAX);
- + res->data_len = (uint32_t)read_data_len;
- + return TRUE;
- +}
- +
- +bool_t decode_op_read_plus(
- + XDR *xdr,
- + nfs_resop4 *resop)
- +{
- + nfs42_read_plus_res *res = (nfs42_read_plus_res *)resop->res;
- +
- + if (unexpected_op(resop->op, OP_READ_PLUS))
- + return FALSE;
- +
- + if (!xdr_u_int32_t(xdr, &res->status))
- + return FALSE;
- +
- + if (res->status == NFS4_OK)
- + return decode_read_plus_res_ok(xdr, &res->resok4);
- +
- + return TRUE;
- +}
- +
- diff --git a/daemon/readwrite.c b/daemon/readwrite.c
- index 355336a..7a079b5 100644
- --- a/daemon/readwrite.c
- +++ b/daemon/readwrite.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) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- @@ -77,8 +78,21 @@ static int read_from_mds(
- while(to_rcv > 0) {
- uint32_t bytes_read = 0, chunk = min(to_rcv, maxreadsize);
- - status = nfs41_read(session, file, stateid, args->offset + reloffset, chunk,
- + if (session->client->root->supports_nfs42_read_plus) {
- + status = nfs42_read_plus(session, file, stateid,
- + args->offset + reloffset, chunk,
- p, &bytes_read, &eof);
- + if (status) {
- + DPRINTF(0, ("nfs42_read_plus() failed, status=%d\n", status));
- + session->client->root->supports_nfs42_read_plus = false;
- + }
- + }
- + else {
- + status = nfs41_read(session, file, stateid,
- + args->offset + reloffset, chunk,
- + p, &bytes_read, &eof);
- + }
- +
- if (status == NFS4ERR_OPENMODE && !len) {
- stateid->type = STATEID_SPECIAL;
- stateid4_cpy(&stateid->stateid, &special_read_stateid);
- diff --git a/daemon/recovery.c b/daemon/recovery.c
- index eee4031..44b996e 100644
- --- a/daemon/recovery.c
- +++ b/daemon/recovery.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) 2024-2025 Roland Mainz <roland.mainz@nrubsig.org>
- *
- * Olga Kornievskaia <aglo@umich.edu>
- * Casey Bodley <cbodley@umich.edu>
- @@ -805,6 +806,9 @@ bool_t nfs41_recover_stateid(
- } else if (argop->op == OP_READ) {
- nfs41_read_args *read = (nfs41_read_args*)argop->arg;
- stateid = read->stateid;
- + } else if (argop->op == OP_READ_PLUS) {
- + nfs42_read_plus_args *read_plus = (nfs42_read_plus_args *)argop->arg;
- + stateid = read_plus->stateid;
- } else if (argop->op == OP_WRITE) {
- nfs41_write_args *write = (nfs41_write_args*)argop->arg;
- stateid = write->stateid;
- --
- 2.45.1
msnfs41client: Jan2025/001 patches
Posted by Anonymous on Mon 27th Jan 2025 13:08
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.