pastebin - collaborative debugging tool
rovema.kpaste.net RSS


msnfs41client: Split kernel driver into smaller source files, 2024-10-12
Posted by Anonymous on Sat 12th Oct 2024 16:13
raw | new post

  1. From 54e5d30339ad151a351cb16ce2d2892333b7a121 Mon Sep 17 00:00:00 2001
  2. From: Roland Mainz <roland.mainz@nrubsig.org>
  3. Date: Sat, 12 Oct 2024 17:03:46 +0200
  4. Subject: [PATCH] sys: Split nfs41_driver.c into smaller source files
  5.  
  6. Split nfs41_driver.c into smaller source files
  7.  
  8. Signed-off-by: Cedric Blancher <cedric.blancher@gmail.com>
  9. ---
  10. build.vc19/nfs41_driver/nfs41_driver.vcxproj |   18 +-
  11.  sys/copysup.c                                |    2 +-
  12.  sys/nfs41_driver.c                           | 8124 ------------------
  13.  sys/nfs41_driver.h                           |    7 +-
  14.  sys/nfs41sys_acl.c                           |  431 +
  15.  sys/nfs41sys_buildconfig.h                   |   66 +
  16.  sys/{nfs41_debug.c => nfs41sys_debug.c}      |  841 +-
  17.  sys/{nfs41_debug.h => nfs41sys_debug.h}      |   99 +-
  18.  sys/nfs41sys_dir.c                           |  330 +
  19.  sys/nfs41sys_driver.c                        | 1444 ++++
  20.  sys/nfs41sys_driver.h                        |  825 ++
  21.  sys/nfs41sys_ea.c                            |  642 ++
  22.  sys/nfs41sys_fileinfo.c                      |  706 ++
  23.  sys/nfs41sys_fsctl.c                         |  111 +
  24.  sys/nfs41sys_ioctl.c                         |   90 +
  25.  sys/nfs41sys_lock.c                          |  398 +
  26.  sys/nfs41sys_mount.c                         | 1499 ++++
  27.  sys/nfs41sys_openclose.c                     | 1026 +++
  28.  sys/nfs41sys_readwrite.c                     |  435 +
  29.  sys/nfs41sys_symlink.c                       |  406 +
  30.  sys/nfs41sys_updowncall.c                    |  668 ++
  31.  sys/nfs41sys_util.c                          |  118 +
  32.  sys/nfs41sys_util.h                          |   55 +
  33.  sys/nfs41sys_volinfo.c                       |  310 +
  34.  24 files changed, 10064 insertions(+), 8587 deletions(-)
  35.  delete mode 100644 sys/nfs41_driver.c
  36.  create mode 100644 sys/nfs41sys_acl.c
  37.  create mode 100644 sys/nfs41sys_buildconfig.h
  38.  rename sys/{nfs41_debug.c => nfs41sys_debug.c} (92%)
  39.  rename sys/{nfs41_debug.h => nfs41sys_debug.h} (79%)
  40.  create mode 100644 sys/nfs41sys_dir.c
  41.  create mode 100644 sys/nfs41sys_driver.c
  42.  create mode 100644 sys/nfs41sys_driver.h
  43.  create mode 100644 sys/nfs41sys_ea.c
  44.  create mode 100644 sys/nfs41sys_fileinfo.c
  45.  create mode 100644 sys/nfs41sys_fsctl.c
  46.  create mode 100644 sys/nfs41sys_ioctl.c
  47.  create mode 100644 sys/nfs41sys_lock.c
  48.  create mode 100644 sys/nfs41sys_mount.c
  49.  create mode 100644 sys/nfs41sys_openclose.c
  50.  create mode 100644 sys/nfs41sys_readwrite.c
  51.  create mode 100644 sys/nfs41sys_symlink.c
  52.  create mode 100644 sys/nfs41sys_updowncall.c
  53.  create mode 100644 sys/nfs41sys_util.c
  54.  create mode 100644 sys/nfs41sys_util.h
  55.  create mode 100644 sys/nfs41sys_volinfo.c
  56.  
  57. diff --git a/build.vc19/nfs41_driver/nfs41_driver.vcxproj b/build.vc19/nfs41_driver/nfs41_driver.vcxproj
  58. index c05a30f..56e4026 100644
  59. --- a/build.vc19/nfs41_driver/nfs41_driver.vcxproj
  60. +++ b/build.vc19/nfs41_driver/nfs41_driver.vcxproj
  61. @@ -250,8 +250,22 @@
  62.    </ItemGroup>
  63.    <ItemGroup>
  64.      <ClCompile Include="..\..\sys\copysup.c" />
  65. -    <ClCompile Include="..\..\sys\nfs41_debug.c" />
  66. -    <ClCompile Include="..\..\sys\nfs41_driver.c" />
  67. +    <ClCompile Include="..\..\sys\nfs41sys_acl.c" />
  68. +    <ClCompile Include="..\..\sys\nfs41sys_debug.c" />
  69. +    <ClCompile Include="..\..\sys\nfs41sys_dir.c" />
  70. +    <ClCompile Include="..\..\sys\nfs41sys_driver.c" />
  71. +    <ClCompile Include="..\..\sys\nfs41sys_ea.c" />
  72. +    <ClCompile Include="..\..\sys\nfs41sys_fileinfo.c" />
  73. +    <ClCompile Include="..\..\sys\nfs41sys_fsctl.c" />
  74. +    <ClCompile Include="..\..\sys\nfs41sys_ioctl.c" />
  75. +    <ClCompile Include="..\..\sys\nfs41sys_lock.c" />
  76. +    <ClCompile Include="..\..\sys\nfs41sys_mount.c" />
  77. +    <ClCompile Include="..\..\sys\nfs41sys_openclose.c" />
  78. +    <ClCompile Include="..\..\sys\nfs41sys_readwrite.c" />
  79. +    <ClCompile Include="..\..\sys\nfs41sys_symlink.c" />
  80. +    <ClCompile Include="..\..\sys\nfs41sys_util.c" />
  81. +    <ClCompile Include="..\..\sys\nfs41sys_updowncall.c" />
  82. +    <ClCompile Include="..\..\sys\nfs41sys_volinfo.c" />
  83.      <ClCompile Include="..\..\sys\wmlkm.c" />
  84.    </ItemGroup>
  85.    <ItemGroup>
  86. diff --git a/sys/copysup.c b/sys/copysup.c
  87. index 515e465..a14f778 100644
  88. --- a/sys/copysup.c
  89. +++ b/sys/copysup.c
  90. @@ -56,7 +56,7 @@
  91.  #include <Ntstrsafe.h>
  92.  #include <stdbool.h>
  93.  
  94. -#include "nfs41_debug.h"
  95. +#include "nfs41sys_debug.h"
  96.  #include "nfs41_build_features.h"
  97.  
  98.  #define COPYSUP_MAX_HOLE_SIZE (2*4096LL)
  99. diff --git a/sys/nfs41_driver.c b/sys/nfs41_driver.c
  100. deleted file mode 100644
  101. index 09645b1..0000000
  102. --- a/sys/nfs41_driver.c
  103. +++ /dev/null
  104. @@ -1,8124 +0,0 @@
  105. -/* NFSv4.1 client for Windows
  106. - * Copyright (C) 2012 The Regents of the University of Michigan
  107. - *
  108. - * Olga Kornievskaia <aglo@umich.edu>
  109. - * Casey Bodley <cbodley@umich.edu>
  110. - * Roland Mainz <roland.mainz@nrubsig.org>
  111. - *
  112. - * This library is free software; you can redistribute it and/or modify it
  113. - * under the terms of the GNU Lesser General Public License as published by
  114. - * the Free Software Foundation; either version 2.1 of the License, or (at
  115. - * your option) any later version.
  116. - *
  117. - * This library is distributed in the hope that it will be useful, but
  118. - * without any warranty; without even the implied warranty of merchantability
  119. - * or fitness for a particular purpose.  See the GNU Lesser General Public
  120. - * License for more details.
  121. - *
  122. - * You should have received a copy of the GNU Lesser General Public License
  123. - * along with this library; if not, write to the Free Software Foundation,
  124. - * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  125. - */
  126. -
  127. -#ifndef _KERNEL_MODE
  128. -#error module requires kernel mode
  129. -#endif
  130. -
  131. -#if ((__STDC_VERSION__-0) < 201710L)
  132. -#error Code requires ISO C17
  133. -#endif
  134. -
  135. -/* FIXME: Why does VS22 need this, but not VC19 ? */
  136. -#if _MSC_VER >= 1900
  137. -#if defined(_WIN64) && defined(_M_X64)
  138. -#ifndef _AMD64_
  139. -#define _AMD64_
  140. -#endif
  141. -#elif defined(_WIN32) && defined(_M_IX86)
  142. -#ifndef _X86_
  143. -#define _X86_
  144. -#endif
  145. -#elif defined(_WIN64) && defined(_M_ARM64)
  146. -#ifndef _ARM64_
  147. -#define _ARM64_
  148. -#endif
  149. -#elif defined(_WIN32) && defined(_M_ARM)
  150. -#ifndef _ARM_
  151. -#define _ARM_
  152. -#endif
  153. -#else
  154. -#error Unsupported arch
  155. -#endif
  156. -#endif /* _MSC_VER >= 1900 */
  157. -
  158. -/* Driver build config */
  159. -#define USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM 1
  160. -// #define LOOKASIDELISTS_STATS 1
  161. -
  162. -#define MINIRDR__NAME "Value is ignored, only fact of definition"
  163. -#include <rx.h>
  164. -#include <windef.h>
  165. -#include <winerror.h>
  166. -
  167. -#include <Ntstrsafe.h>
  168. -
  169. -#include "nfs41_driver.h"
  170. -#include "nfs41_np.h"
  171. -#include "nfs41_debug.h"
  172. -#include "nfs41_build_features.h"
  173. -#include "nfs_ea.h"
  174. -
  175. -// #define USE_ENTIRE_PATH_FOR_NETROOT 1
  176. -
  177. -/* debugging printout defines */
  178. -#if defined(_DEBUG)
  179. -/* Debug build defines follow... */
  180. -#define DEBUG_MARSHAL_HEADER
  181. -#define DEBUG_MARSHAL_DETAIL
  182. -//#define DEBUG_MARSHAL_DETAIL_RW
  183. -//#define DEBUG_SECURITY_TOKEN
  184. -#define DEBUG_MOUNTCONFIG
  185. -//#define DEBUG_OPEN
  186. -//#define DEBUG_CLOSE
  187. -//#define DEBUG_CACHE
  188. -#define DEBUG_INVALIDATE_CACHE
  189. -//#define DEBUG_READ
  190. -//#define DEBUG_WRITE
  191. -//#define DEBUG_DIR_QUERY
  192. -//#define DEBUG_FILE_QUERY
  193. -//#define DEBUG_FILE_SET
  194. -//#define DEBUG_ACL_QUERY
  195. -//#define DEBUG_ACL_SET
  196. -//#define DEBUG_EA_QUERY
  197. -//#define DEBUG_EA_SET
  198. -//#define DEBUG_LOCK
  199. -#define DEBUG_FSCTL
  200. -#define DEBUG_IOCTL
  201. -#define DEBUG_TIME_BASED_COHERENCY
  202. -#define DEBUG_MOUNT
  203. -//#define DEBUG_VOLUME_QUERY
  204. -
  205. -//#define ENABLE_TIMINGS
  206. -//#define ENABLE_INDV_TIMINGS
  207. -#elif defined(NDEBUG)
  208. -/* Release build defines follow... */
  209. -#else
  210. -#error Neither _DEBUG NOR _NDEBUG defined
  211. -#endif
  212. -
  213. -#ifdef ENABLE_TIMINGS
  214. -typedef struct __nfs41_timings {
  215. -    LONG tops, sops;
  216. -    LONGLONG ticks, size;
  217. -} nfs41_timings;
  218. -
  219. -nfs41_timings lookup, readdir, open, close, getattr, setattr, getacl, setacl, volume,
  220. -    read, write, lock, unlock, setexattr, getexattr;
  221. -#endif
  222. -DRIVER_INITIALIZE DriverEntry;
  223. -DRIVER_UNLOAD nfs41_driver_unload;
  224. -_Dispatch_type_(IRP_MJ_CREATE) \
  225. -    _Dispatch_type_(IRP_MJ_CREATE_NAMED_PIPE) \
  226. -    DRIVER_DISPATCH(nfs41_FsdDispatch);
  227. -
  228. -struct _MINIRDR_DISPATCH nfs41_ops;
  229. -PRDBSS_DEVICE_OBJECT nfs41_dev;
  230. -
  231. -#define DISABLE_CACHING 0
  232. -#define ENABLE_READ_CACHING 1
  233. -#define ENABLE_WRITE_CACHING 2
  234. -#define ENABLE_READWRITE_CACHING 3
  235. -
  236. -#define NFS41_MM_POOLTAG        ('nfs4')
  237. -#define NFS41_MM_POOLTAG_ACL    ('acls')
  238. -#define NFS41_MM_POOLTAG_MOUNT  ('mnts')
  239. -#define NFS41_MM_POOLTAG_OPEN   ('open')
  240. -#define NFS41_MM_POOLTAG_UP     ('upca')
  241. -#define NFS41_MM_POOLTAG_DOWN   ('down')
  242. -
  243. -KEVENT upcallEvent;
  244. -FAST_MUTEX upcallLock, downcallLock, fcblistLock;
  245. -FAST_MUTEX openOwnerLock;
  246. -
  247. -LONGLONG xid = 0;
  248. -LONG open_owner_id = 1;
  249. -
  250. -#define DECLARE_CONST_ANSI_STRING(_var, _string) \
  251. -    const CHAR _var ## _buffer[] = _string; \
  252. -    const ANSI_STRING _var = { sizeof(_string) - sizeof(CHAR), \
  253. -        sizeof(_string), (PCH) _var ## _buffer }
  254. -#define RELATIVE(wait) (-(wait))
  255. -#define NANOSECONDS(nanos) (((signed __int64)(nanos)) / 100L)
  256. -#define MICROSECONDS(micros) (((signed __int64)(micros)) * NANOSECONDS(1000L))
  257. -#define MILLISECONDS(milli) (((signed __int64)(milli)) * MICROSECONDS(1000L))
  258. -#define SECONDS(seconds) (((signed __int64)(seconds)) * MILLISECONDS(1000L))
  259. -
  260. -DECLARE_CONST_ANSI_STRING(NfsV3Attributes, EA_NFSV3ATTRIBUTES);
  261. -DECLARE_CONST_ANSI_STRING(NfsSymlinkTargetName, EA_NFSSYMLINKTARGETNAME);
  262. -DECLARE_CONST_ANSI_STRING(NfsActOnLink, EA_NFSACTONLINK);
  263. -
  264. -#ifdef NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL
  265. -const LUID SystemLuid = SYSTEM_LUID;
  266. -#endif /* NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL */
  267. -
  268. -INLINE BOOL AnsiStrEq(
  269. -    IN const ANSI_STRING *lhs,
  270. -    IN const CHAR *rhs,
  271. -    IN const UCHAR rhs_len)
  272. -{
  273. -    return lhs->Length == rhs_len &&
  274. -        RtlCompareMemory(lhs->Buffer, rhs, rhs_len) == rhs_len;
  275. -}
  276. -
  277. -typedef struct _nfs3_attrs {
  278. -    DWORD type, mode, nlink, uid, gid, filler1;
  279. -    LARGE_INTEGER size, used;
  280. -    struct {
  281. -        DWORD specdata1;
  282. -        DWORD specdata2;
  283. -    } rdev;
  284. -    LONGLONG fsid, fileid;
  285. -    LONGLONG atime, mtime, ctime;
  286. -} nfs3_attrs;
  287. -LARGE_INTEGER unix_time_diff; //needed to convert windows time to unix
  288. -
  289. -enum ftype3 {
  290. -    NF3REG = 1,
  291. -    NF3DIR,
  292. -    NF3BLK,
  293. -    NF3CHR,
  294. -    NF3LNK,
  295. -    NF3SOCK,
  296. -    NF3FIFO
  297. -};
  298. -
  299. -typedef enum _nfs41_updowncall_state {
  300. -    NFS41_WAITING_FOR_UPCALL,
  301. -    NFS41_WAITING_FOR_DOWNCALL,
  302. -    NFS41_DONE_PROCESSING,
  303. -    NFS41_NOT_WAITING
  304. -} nfs41_updowncall_state;
  305. -
  306. -typedef struct _updowncall_entry {
  307. -    DWORD version;
  308. -    LONGLONG xid;
  309. -    nfs41_opcodes opcode;
  310. -    NTSTATUS status;
  311. -    nfs41_updowncall_state state;
  312. -    FAST_MUTEX lock;
  313. -    LIST_ENTRY next;
  314. -    KEVENT cond;
  315. -#undef errno
  316. -    DWORD errno;
  317. -    BOOLEAN async_op;
  318. -    SECURITY_CLIENT_CONTEXT sec_ctx;
  319. -    PSECURITY_CLIENT_CONTEXT psec_ctx;
  320. -    /*
  321. -     * Refcount client token during lifetime of this |updowncall_entry|
  322. -     * to avoid crashes during |SeImpersonateClientEx()| if the
  323. -     * calling thread disappears.
  324. -     */
  325. -    PVOID psec_ctx_clienttoken;
  326. -    HANDLE open_state;
  327. -    HANDLE session;
  328. -    PUNICODE_STRING filename;
  329. -    PVOID buf;
  330. -    ULONG buf_len;
  331. -    ULONGLONG ChangeTime;
  332. -    union {
  333. -        struct {
  334. -            PUNICODE_STRING srv_name; /* hostname, or hostname@port */
  335. -            PUNICODE_STRING root;
  336. -            PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs;
  337. -            DWORD sec_flavor;
  338. -            DWORD rsize;
  339. -            DWORD wsize;
  340. -            DWORD lease_time;
  341. -            DWORD use_nfspubfh;
  342. -        } Mount;
  343. -        struct {
  344. -            PMDL MdlAddress;
  345. -            ULONGLONG offset;
  346. -            PRX_CONTEXT rxcontext;
  347. -        } ReadWrite;
  348. -        struct {
  349. -            LONGLONG offset;
  350. -            LONGLONG length;
  351. -            BOOLEAN exclusive;
  352. -            BOOLEAN blocking;
  353. -        } Lock;
  354. -        struct {
  355. -            ULONG count;
  356. -            LOWIO_LOCK_LIST locks;
  357. -        } Unlock;
  358. -        struct {
  359. -            FILE_BASIC_INFORMATION binfo;
  360. -            FILE_STANDARD_INFORMATION sinfo;
  361. -            UNICODE_STRING symlink;
  362. -            ULONG access_mask;
  363. -            ULONG access_mode;
  364. -            ULONG attrs;
  365. -            ULONG copts;
  366. -            ULONG disp;
  367. -            ULONG cattrs;
  368. -            LONG open_owner_id;
  369. -            DWORD mode;
  370. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  371. -            DWORD owner_local_uid;
  372. -            DWORD owner_group_local_gid;
  373. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  374. -            HANDLE srv_open;
  375. -            DWORD deleg_type;
  376. -            BOOLEAN symlink_embedded;
  377. -            PMDL EaMdl;
  378. -            PVOID EaBuffer;
  379. -        } Open;
  380. -        struct {
  381. -            HANDLE srv_open;
  382. -            BOOLEAN remove;
  383. -            BOOLEAN renamed;
  384. -        } Close;
  385. -        struct {
  386. -            PUNICODE_STRING filter;
  387. -            FILE_INFORMATION_CLASS InfoClass;
  388. -            BOOLEAN restart_scan;
  389. -            BOOLEAN return_single;
  390. -            BOOLEAN initial_query;
  391. -            PMDL mdl;
  392. -            PVOID mdl_buf;
  393. -        } QueryFile;
  394. -        struct {
  395. -            FILE_INFORMATION_CLASS InfoClass;
  396. -        } SetFile;
  397. -        struct {
  398. -            DWORD mode;
  399. -        } SetEa;
  400. -        struct {
  401. -            PVOID EaList;
  402. -            ULONG EaListLength;
  403. -            ULONG Overflow;
  404. -            ULONG EaIndex;
  405. -            BOOLEAN ReturnSingleEntry;
  406. -            BOOLEAN RestartScan;
  407. -        } QueryEa;
  408. -        struct {
  409. -            PUNICODE_STRING target;
  410. -            BOOLEAN set;
  411. -        } Symlink;
  412. -        struct {
  413. -            FS_INFORMATION_CLASS query;
  414. -        } Volume;
  415. -        struct {
  416. -            SECURITY_INFORMATION query;
  417. -        } Acl;
  418. -    } u;
  419. -
  420. -} nfs41_updowncall_entry;
  421. -
  422. -typedef struct _updowncall_list {
  423. -    LIST_ENTRY head;
  424. -} nfs41_updowncall_list;
  425. -nfs41_updowncall_list upcall, downcall;
  426. -
  427. -
  428. -#if _MSC_VER >= 1900
  429. -/*
  430. - * gisburn: VS22 chokes on the original define for
  431. - * |DECLARE_CONST_UNICODE_STRING|, so we use one
  432. - * without the offending stuff
  433. - */
  434. -#undef DECLARE_CONST_UNICODE_STRING
  435. -#define DECLARE_CONST_UNICODE_STRING(_var, _string) \
  436. -       const WCHAR _var ## _buffer[] = _string; \
  437. -       const UNICODE_STRING _var = { sizeof(_string) - sizeof(WCHAR), sizeof(_string), (PWCH) _var ## _buffer }
  438. -#endif /* _MSC_VER >= 1900 */
  439. -
  440. -
  441. -/*
  442. - * In order to cooperate with other network providers,
  443. - * we only claim paths of the format '\\server\nfs4\path' or
  444. - * '\\server\pubnfs4\path'
  445. - */
  446. -DECLARE_CONST_UNICODE_STRING(NfsPrefix, L"\\nfs4");
  447. -DECLARE_CONST_UNICODE_STRING(PubNfsPrefix, L"\\pubnfs4");
  448. -DECLARE_CONST_UNICODE_STRING(AUTH_SYS_NAME, L"sys");
  449. -DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5_NAME, L"krb5");
  450. -DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5I_NAME, L"krb5i");
  451. -DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5P_NAME, L"krb5p");
  452. -DECLARE_CONST_UNICODE_STRING(SLASH, L"\\");
  453. -DECLARE_CONST_UNICODE_STRING(EMPTY_STRING, L"");
  454. -
  455. -#define SERVER_NAME_BUFFER_SIZE         1024
  456. -#define MOUNT_CONFIG_RW_SIZE_MIN        1024
  457. -#define MOUNT_CONFIG_RW_SIZE_DEFAULT    1048576
  458. -#define MOUNT_CONFIG_RW_SIZE_MAX        1048576
  459. -#define MAX_SEC_FLAVOR_LEN              12
  460. -#define UPCALL_TIMEOUT_DEFAULT          50  /* in seconds */
  461. -
  462. -typedef struct _NFS41_MOUNT_CONFIG {
  463. -    BOOLEAN use_nfspubfh;
  464. -    DWORD ReadSize;
  465. -    DWORD WriteSize;
  466. -    BOOLEAN ReadOnly;
  467. -    BOOLEAN write_thru;
  468. -    BOOLEAN nocache;
  469. -    BOOLEAN timebasedcoherency;
  470. -    WCHAR srv_buffer[SERVER_NAME_BUFFER_SIZE];
  471. -    UNICODE_STRING SrvName; /* hostname, or hostname@port */
  472. -    WCHAR mntpt_buffer[NFS41_SYS_MAX_PATH_LEN];
  473. -    UNICODE_STRING MntPt;
  474. -    WCHAR sec_flavor_buffer[MAX_SEC_FLAVOR_LEN];
  475. -    UNICODE_STRING SecFlavor;
  476. -    DWORD timeout;
  477. -    struct {
  478. -        BOOLEAN use_nfsv3attrsea_mode;
  479. -        DWORD mode;
  480. -    } createmode;
  481. -} NFS41_MOUNT_CONFIG, *PNFS41_MOUNT_CONFIG;
  482. -
  483. -typedef struct _nfs41_mount_entry {
  484. -    LIST_ENTRY next;
  485. -    LUID login_id;
  486. -    HANDLE authsys_session;
  487. -    HANDLE gss_session;
  488. -    HANDLE gssi_session;
  489. -    HANDLE gssp_session;
  490. -    NFS41_MOUNT_CONFIG Config;
  491. -} nfs41_mount_entry;
  492. -
  493. -typedef struct _nfs41_mount_list {
  494. -    LIST_ENTRY head;
  495. -} nfs41_mount_list;
  496. -
  497. -#define nfs41_AddEntry(lock,list,pEntry)                    \
  498. -            ExAcquireFastMutex(&lock);                      \
  499. -            InsertTailList(&(list).head, &(pEntry)->next);  \
  500. -            ExReleaseFastMutex(&lock);
  501. -#define nfs41_RemoveFirst(lock,list,pEntry)                 \
  502. -            ExAcquireFastMutex(&lock);                      \
  503. -            pEntry = (IsListEmpty(&(list).head)             \
  504. -            ? NULL                                          \
  505. -            : RemoveHeadList(&(list).head));                \
  506. -            ExReleaseFastMutex(&lock);
  507. -#define nfs41_RemoveEntry(lock,pEntry)                      \
  508. -            ExAcquireFastMutex(&lock);                      \
  509. -            RemoveEntryList(&pEntry->next);                 \
  510. -            ExReleaseFastMutex(&lock);
  511. -#define nfs41_IsListEmpty(lock,list,flag)                   \
  512. -            ExAcquireFastMutex(&lock);                      \
  513. -            flag = IsListEmpty(&(list).head);               \
  514. -            ExReleaseFastMutex(&lock);
  515. -#define nfs41_GetFirstEntry(lock,list,pEntry)               \
  516. -            ExAcquireFastMutex(&lock);                      \
  517. -            pEntry = (IsListEmpty(&(list).head)             \
  518. -             ? NULL                                         \
  519. -             : (nfs41_updowncall_entry *)                   \
  520. -               (CONTAINING_RECORD((list).head.Flink,        \
  521. -                                  nfs41_updowncall_entry,   \
  522. -                                  next)));                  \
  523. -            ExReleaseFastMutex(&lock);
  524. -#define nfs41_GetFirstMountEntry(lock,list,pEntry)          \
  525. -            ExAcquireFastMutex(&lock);                      \
  526. -            pEntry = (IsListEmpty(&(list).head)             \
  527. -             ? NULL                                         \
  528. -             : (nfs41_mount_entry *)                        \
  529. -               (CONTAINING_RECORD((list).head.Flink,        \
  530. -                                  nfs41_mount_entry,        \
  531. -                                  next)));                  \
  532. -            ExReleaseFastMutex(&lock);
  533. -
  534. -
  535. -typedef struct _NFS41_NETROOT_EXTENSION {
  536. -    NODE_TYPE_CODE          NodeTypeCode;
  537. -    NODE_BYTE_SIZE          NodeByteSize;
  538. -    DWORD                   nfs41d_version;
  539. -    BOOLEAN                 mounts_init;
  540. -    FAST_MUTEX              mountLock;
  541. -    nfs41_mount_list        mounts;
  542. -} NFS41_NETROOT_EXTENSION, *PNFS41_NETROOT_EXTENSION;
  543. -#define NFS41GetNetRootExtension(pNetRoot)      \
  544. -        (((pNetRoot) == NULL) ? NULL :          \
  545. -        (PNFS41_NETROOT_EXTENSION)((pNetRoot)->Context))
  546. -
  547. -/* FileSystemName as reported by FileFsAttributeInfo query */
  548. -#if ((NFS41_DRIVER_DEBUG_FS_NAME) == 1)
  549. -#define FS_NAME     L"NFS"
  550. -#elif  ((NFS41_DRIVER_DEBUG_FS_NAME) == 2)
  551. -#define FS_NAME     L"DEBUG-NFS41"
  552. -#else
  553. -#error NFS41_DRIVER_DEBUG_FS_NAME not defined
  554. -#endif
  555. -#define FS_NAME_LEN (sizeof(FS_NAME) - sizeof(WCHAR))
  556. -#define FS_ATTR_LEN (sizeof(FILE_FS_ATTRIBUTE_INFORMATION) + FS_NAME_LEN)
  557. -
  558. -/* FileSystemName as reported by FileFsAttributeInfo query */
  559. -#define VOL_NAME     L"PnfsVolume"
  560. -#define VOL_NAME_LEN (sizeof(VOL_NAME) - sizeof(WCHAR))
  561. -#define VOL_ATTR_LEN (sizeof(FILE_FS_VOLUME_INFORMATION) + VOL_NAME_LEN)
  562. -
  563. -typedef struct _NFS41_V_NET_ROOT_EXTENSION {
  564. -    NODE_TYPE_CODE          NodeTypeCode;
  565. -    NODE_BYTE_SIZE          NodeByteSize;
  566. -    HANDLE                  session;
  567. -    FILE_FS_ATTRIBUTE_INFORMATION FsAttrs;
  568. -    DWORD                   sec_flavor;
  569. -    DWORD                   timeout;
  570. -    struct {
  571. -        BOOLEAN use_nfsv3attrsea_mode;
  572. -        DWORD mode;
  573. -    } createmode;
  574. -    USHORT                  MountPathLen;
  575. -    BOOLEAN                 read_only;
  576. -    BOOLEAN                 write_thru;
  577. -    BOOLEAN                 nocache;
  578. -    BOOLEAN                 timebasedcoherency;
  579. -} NFS41_V_NET_ROOT_EXTENSION, *PNFS41_V_NET_ROOT_EXTENSION;
  580. -#define NFS41GetVNetRootExtension(pVNetRoot)      \
  581. -        (((pVNetRoot) == NULL) ? NULL :           \
  582. -        (PNFS41_V_NET_ROOT_EXTENSION)((pVNetRoot)->Context))
  583. -
  584. -typedef struct _NFS41_FCB {
  585. -    NODE_TYPE_CODE          NodeTypeCode;
  586. -    NODE_BYTE_SIZE          NodeByteSize;
  587. -    FILE_BASIC_INFORMATION  BasicInfo;
  588. -    FILE_STANDARD_INFORMATION StandardInfo;
  589. -    BOOLEAN                 Renamed;
  590. -    BOOLEAN                 DeletePending;
  591. -    DWORD                   mode;
  592. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  593. -    DWORD                   owner_local_uid;       /* owner mapped into local uid */
  594. -    DWORD                   owner_group_local_gid; /* owner group mapped into local gid */
  595. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  596. -    ULONGLONG               changeattr;
  597. -} NFS41_FCB, *PNFS41_FCB;
  598. -#define NFS41GetFcbExtension(pFcb)      \
  599. -        (((pFcb) == NULL) ? NULL : (PNFS41_FCB)((pFcb)->Context))
  600. -
  601. -typedef struct _NFS41_FOBX {
  602. -    NODE_TYPE_CODE          NodeTypeCode;
  603. -    NODE_BYTE_SIZE          NodeByteSize;
  604. -
  605. -    HANDLE nfs41_open_state;
  606. -    SECURITY_CLIENT_CONTEXT sec_ctx;
  607. -    PVOID acl;
  608. -    DWORD acl_len;
  609. -    LARGE_INTEGER time;
  610. -    DWORD deleg_type;
  611. -    BOOLEAN write_thru;
  612. -    BOOLEAN nocache;
  613. -    BOOLEAN timebasedcoherency;
  614. -} NFS41_FOBX, *PNFS41_FOBX;
  615. -#define NFS41GetFobxExtension(pFobx)  \
  616. -        (((pFobx) == NULL) ? NULL : (PNFS41_FOBX)((pFobx)->Context))
  617. -
  618. -typedef struct _NFS41_SERVER_ENTRY {
  619. -    PMRX_SRV_CALL                 pRdbssSrvCall;
  620. -    WCHAR                         NameBuffer[SERVER_NAME_BUFFER_SIZE];
  621. -    UNICODE_STRING                Name;             // the server name.
  622. -} NFS41_SERVER_ENTRY, *PNFS41_SERVER_ENTRY;
  623. -
  624. -typedef struct _NFS41_DEVICE_EXTENSION {
  625. -    NODE_TYPE_CODE          NodeTypeCode;
  626. -    NODE_BYTE_SIZE          NodeByteSize;
  627. -    PRDBSS_DEVICE_OBJECT    DeviceObject;
  628. -    ULONG                   ActiveNodes;
  629. -    HANDLE                  SharedMemorySection;
  630. -    DWORD                   nfs41d_version;
  631. -    BYTE                    VolAttrs[VOL_ATTR_LEN];
  632. -    DWORD                   VolAttrsLen;
  633. -    HANDLE                  openlistHandle;
  634. -} NFS41_DEVICE_EXTENSION, *PNFS41_DEVICE_EXTENSION;
  635. -
  636. -#define NFS41GetDeviceExtension(RxContext,pExt)        \
  637. -        PNFS41_DEVICE_EXTENSION pExt = (PNFS41_DEVICE_EXTENSION) \
  638. -        ((PBYTE)(RxContext->RxDeviceObject) + sizeof(RDBSS_DEVICE_OBJECT))
  639. -
  640. -typedef struct _nfs41_fcb_list_entry {
  641. -    LIST_ENTRY next;
  642. -    PMRX_FCB fcb;
  643. -    HANDLE session;
  644. -    PNFS41_FOBX nfs41_fobx;
  645. -    ULONGLONG ChangeTime;
  646. -    BOOLEAN skip;
  647. -} nfs41_fcb_list_entry;
  648. -
  649. -typedef struct _nfs41_fcb_list {
  650. -    LIST_ENTRY head;
  651. -} nfs41_fcb_list;
  652. -nfs41_fcb_list openlist;
  653. -
  654. -typedef enum _NULMRX_STORAGE_TYPE_CODES {
  655. -    NTC_NFS41_DEVICE_EXTENSION      =   (NODE_TYPE_CODE)0xFC00,    
  656. -} NFS41_STORAGE_TYPE_CODES;
  657. -#define RxDefineNode( node, type )          \
  658. -        node->NodeTypeCode = NTC_##type;    \
  659. -        node->NodeByteSize = sizeof(type);
  660. -
  661. -#define RDR_NULL_STATE  0
  662. -#define RDR_UNLOADED    1
  663. -#define RDR_UNLOADING   2
  664. -#define RDR_LOADING     3
  665. -#define RDR_LOADED      4
  666. -#define RDR_STOPPED     5
  667. -#define RDR_STOPPING    6
  668. -#define RDR_STARTING    7
  669. -#define RDR_STARTED     8
  670. -
  671. -/*
  672. - * Assume network speed is 10MB/s (100base-T ethernet, lowest common
  673. - * denominator which we support) plus disk speed is 10MB/s so add
  674. - * time to transfer requested bytes over the network and read from
  675. - * disk.
  676. - * FIXME: What about ssh-tunneled NFSv4 mounts - should this be a
  677. - * tuneable/mount option ?
  678. - */
  679. -#define EXTRA_TIMEOUT_PER_BYTE(size)  ((2LL * (size)) / (10*1024*1024LL))
  680. -
  681. -nfs41_init_driver_state nfs41_init_state = NFS41_INIT_DRIVER_STARTABLE;
  682. -nfs41_start_driver_state nfs41_start_state = NFS41_START_DRIVER_STARTABLE;
  683. -
  684. -#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  685. -NPAGED_LOOKASIDE_LIST updowncall_entry_upcall_lookasidelist;
  686. -NPAGED_LOOKASIDE_LIST updowncall_entry_downcall_lookasidelist;
  687. -#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  688. -
  689. -static
  690. -nfs41_updowncall_entry *nfs41_upcall_allocate_updowncall_entry(void)
  691. -{
  692. -    nfs41_updowncall_entry *e;
  693. -#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  694. -    e = ExAllocateFromNPagedLookasideList(
  695. -        &updowncall_entry_upcall_lookasidelist);
  696. -
  697. -#ifdef LOOKASIDELISTS_STATS
  698. -    volatile static long cnt = 0;
  699. -    if ((cnt++ % 100) == 0) {
  700. -        print_lookasidelist_stat("updowncall_entry_upcall",
  701. -            &updowncall_entry_upcall_lookasidelist);
  702. -    }
  703. -#endif /* LOOKASIDELISTS_STATS */
  704. -#else
  705. -    e = RxAllocatePoolWithTag(NonPagedPoolNx,
  706. -        sizeof(nfs41_updowncall_entry),
  707. -        NFS41_MM_POOLTAG_UP);
  708. -#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  709. -
  710. -    return e;
  711. -}
  712. -
  713. -static
  714. -void nfs41_upcall_free_updowncall_entry(nfs41_updowncall_entry *entry)
  715. -{
  716. -#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  717. -    ExFreeToNPagedLookasideList(&updowncall_entry_upcall_lookasidelist,
  718. -        entry);
  719. -#else
  720. -    RxFreePool(entry);
  721. -#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  722. -}
  723. -
  724. -static
  725. -nfs41_updowncall_entry *nfs41_downcall_allocate_updowncall_entry(void)
  726. -{
  727. -    nfs41_updowncall_entry *e;
  728. -#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  729. -    e = ExAllocateFromNPagedLookasideList(
  730. -        &updowncall_entry_downcall_lookasidelist);
  731. -
  732. -#ifdef LOOKASIDELISTS_STATS
  733. -    volatile static long cnt = 0;
  734. -    if ((cnt++ % 100) == 0) {
  735. -        print_lookasidelist_stat("updowncall_entry_downcall",
  736. -            &updowncall_entry_downcall_lookasidelist);
  737. -    }
  738. -#endif /* LOOKASIDELISTS_STATS */
  739. -#else
  740. -    e = RxAllocatePoolWithTag(NonPagedPoolNx,
  741. -        sizeof(nfs41_updowncall_entry),
  742. -        NFS41_MM_POOLTAG_DOWN);
  743. -#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  744. -    return e;
  745. -}
  746. -
  747. -static
  748. -void nfs41_downcall_free_updowncall_entry(nfs41_updowncall_entry *entry)
  749. -{
  750. -#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  751. -    ExFreeToNPagedLookasideList(&updowncall_entry_downcall_lookasidelist,
  752. -        entry);
  753. -#else
  754. -    RxFreePool(entry);
  755. -#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  756. -}
  757. -
  758. -/* Local prototypes */
  759. -static NTSTATUS map_mount_errors(
  760. -    DWORD status);
  761. -static NTSTATUS map_sec_flavor(
  762. -    IN PUNICODE_STRING sec_flavor_name,
  763. -    OUT PDWORD sec_flavor);
  764. -static NTSTATUS map_open_errors(
  765. -    DWORD status,
  766. -    USHORT len);
  767. -static NTSTATUS map_close_errors(
  768. -    DWORD status);
  769. -static NTSTATUS map_querydir_errors(
  770. -    DWORD status);
  771. -static NTSTATUS map_volume_errors(
  772. -    DWORD status);
  773. -static NTSTATUS map_setea_error(
  774. -    DWORD error);
  775. -static NTSTATUS map_query_acl_error(
  776. -    DWORD error);
  777. -static NTSTATUS map_queryfile_error(
  778. -    DWORD error);
  779. -static NTSTATUS map_setfile_error(
  780. -    DWORD error);
  781. -static NTSTATUS map_readwrite_errors(DWORD status);
  782. -static NTSTATUS map_lock_errors(
  783. -    DWORD status);
  784. -static NTSTATUS map_symlink_errors(
  785. -    NTSTATUS status);
  786. -
  787. -static void copy_nfs41_mount_config(NFS41_MOUNT_CONFIG *dest, NFS41_MOUNT_CONFIG *src)
  788. -{
  789. -    RtlCopyMemory(dest, src, sizeof(NFS41_MOUNT_CONFIG));
  790. -    dest->SrvName.Buffer = dest->srv_buffer;
  791. -    dest->MntPt.Buffer = dest->mntpt_buffer;
  792. -    dest->SecFlavor.Buffer = dest->sec_flavor_buffer;
  793. -}
  794. -
  795. -static void print_debug_header(
  796. -    PRX_CONTEXT RxContext)
  797. -{
  798. -
  799. -    PIO_STACK_LOCATION IrpSp = RxContext->CurrentIrpSp;
  800. -
  801. -    if (IrpSp) {
  802. -        DbgP("FileOject 0x%p name '%wZ' access r=%d,w=%d,d=%d share r=%d,w=%d,d=%d\n",
  803. -            IrpSp->FileObject, &IrpSp->FileObject->FileName,
  804. -            IrpSp->FileObject->ReadAccess, IrpSp->FileObject->WriteAccess,
  805. -            IrpSp->FileObject->DeleteAccess, IrpSp->FileObject->SharedRead,
  806. -            IrpSp->FileObject->SharedWrite, IrpSp->FileObject->SharedDelete);
  807. -        print_file_object(0, IrpSp->FileObject);
  808. -        print_irps_flags(0, RxContext->CurrentIrpSp);
  809. -    } else
  810. -        DbgP("Couldn't print FileObject IrpSp is NULL\n");
  811. -
  812. -    print_fo_all(1, RxContext);
  813. -    if (RxContext->CurrentIrp)
  814. -        print_irp_flags(0, RxContext->CurrentIrp);
  815. -}
  816. -
  817. -/* convert strings from unicode -> ansi during marshalling to
  818. - * save space in the upcall buffers and avoid extra copies */
  819. -static INLINE ULONG length_as_utf8(
  820. -    PCUNICODE_STRING str)
  821. -{
  822. -    ULONG ActualCount = 0;
  823. -    RtlUnicodeToUTF8N(NULL, 0xffff, &ActualCount, str->Buffer, str->Length);
  824. -    return sizeof(str->MaximumLength) + ActualCount + sizeof(UNICODE_NULL);
  825. -}
  826. -
  827. -static NTSTATUS marshall_unicode_as_utf8(
  828. -    IN OUT unsigned char **pos,
  829. -    IN PCUNICODE_STRING str)
  830. -{
  831. -    ANSI_STRING ansi;
  832. -    ULONG ActualCount;
  833. -    NTSTATUS status;
  834. -
  835. -    if (str->Length == 0) {
  836. -        status = STATUS_SUCCESS;
  837. -        ActualCount = 0;
  838. -        ansi.MaximumLength = 1;
  839. -        goto out_copy;
  840. -    }
  841. -
  842. -    /* query the number of bytes required for the utf8 encoding */
  843. -    status = RtlUnicodeToUTF8N(NULL, 0xffff,
  844. -        &ActualCount, str->Buffer, str->Length);
  845. -    if (status) {
  846. -        print_error("RtlUnicodeToUTF8N('%wZ') failed with 0x%08X\n",
  847. -            str, status);
  848. -        goto out;
  849. -    }
  850. -
  851. -    /* convert the string directly into the upcall buffer */
  852. -    ansi.Buffer = (PCHAR)*pos + sizeof(ansi.MaximumLength);
  853. -    ansi.MaximumLength = (USHORT)ActualCount + sizeof(UNICODE_NULL);
  854. -    status = RtlUnicodeToUTF8N(ansi.Buffer, ansi.MaximumLength,
  855. -        &ActualCount, str->Buffer, str->Length);
  856. -    if (status) {
  857. -        print_error("RtlUnicodeToUTF8N(%hu, '%wZ', %hu) failed with 0x%08X\n",
  858. -            ansi.MaximumLength, str, str->Length, status);
  859. -        goto out;
  860. -    }
  861. -
  862. -out_copy:
  863. -    RtlCopyMemory(*pos, &ansi.MaximumLength, sizeof(ansi.MaximumLength));
  864. -    *pos += sizeof(ansi.MaximumLength);
  865. -    (*pos)[ActualCount] = '\0';
  866. -    *pos += ansi.MaximumLength;
  867. -out:
  868. -    return status;
  869. -}
  870. -
  871. -static NTSTATUS marshal_nfs41_header(
  872. -    nfs41_updowncall_entry *entry,
  873. -    unsigned char *buf,
  874. -    ULONG buf_len,
  875. -    ULONG *len)
  876. -{
  877. -    NTSTATUS status = STATUS_SUCCESS;
  878. -    ULONG header_len = 0;
  879. -    unsigned char *tmp = buf;
  880. -
  881. -    header_len = sizeof(entry->version) + sizeof(entry->xid) +
  882. -        sizeof(entry->opcode) + 2 * sizeof(HANDLE);
  883. -    if (header_len > buf_len) {
  884. -        status = STATUS_INSUFFICIENT_RESOURCES;
  885. -        goto out;
  886. -    }
  887. -    else
  888. -        *len = header_len;
  889. -    RtlCopyMemory(tmp, &entry->version, sizeof(entry->version));
  890. -    tmp += sizeof(entry->version);
  891. -    RtlCopyMemory(tmp, &entry->xid, sizeof(entry->xid));
  892. -    tmp += sizeof(entry->xid);
  893. -    RtlCopyMemory(tmp, &entry->opcode, sizeof(entry->opcode));
  894. -    tmp += sizeof(entry->opcode);
  895. -    RtlCopyMemory(tmp, &entry->session, sizeof(HANDLE));
  896. -    tmp += sizeof(HANDLE);
  897. -    RtlCopyMemory(tmp, &entry->open_state, sizeof(HANDLE));
  898. -    tmp += sizeof(HANDLE);
  899. -
  900. -    /*
  901. -     * gisburn: FIXME: For currently unknown reasons we need to
  902. -     * validate |entry->filename|+it's contents, because a heavily
  903. -     * stressed system somehow sometimes causes garbage there
  904. -     */
  905. -    if (MmIsAddressValid(entry->filename) &&
  906. -        (entry->filename != NULL) &&
  907. -        MmIsAddressValid(entry->filename->Buffer)) {
  908. -#ifdef DEBUG_MARSHAL_HEADER
  909. -        DbgP("[upcall header] xid=%lld opcode='%s' filename='%wZ' version=%d "
  910. -            "session=0x%x open_state=0x%x\n", entry->xid,
  911. -            ENTRY_OPCODE2STRING(entry), entry->filename,
  912. -            entry->version, entry->session, entry->open_state);
  913. -#endif /* DEBUG_MARSHAL_HEADER */
  914. -    }
  915. -    else {
  916. -        DbgP("[upcall header] Invalid filename 0x%p\n", entry);
  917. -        status = STATUS_INTERNAL_ERROR;
  918. -    }
  919. -out:
  920. -    return status;
  921. -}
  922. -
  923. -static const char* secflavorop2name(
  924. -    DWORD sec_flavor)
  925. -{
  926. -    switch(sec_flavor) {
  927. -    case RPCSEC_AUTH_SYS:      return "AUTH_SYS";
  928. -    case RPCSEC_AUTHGSS_KRB5:  return "AUTHGSS_KRB5";
  929. -    case RPCSEC_AUTHGSS_KRB5I: return "AUTHGSS_KRB5I";
  930. -    case RPCSEC_AUTHGSS_KRB5P: return "AUTHGSS_KRB5P";
  931. -    }
  932. -
  933. -    return "UNKNOWN FLAVOR";
  934. -}
  935. -
  936. -static NTSTATUS marshal_nfs41_mount(
  937. -    nfs41_updowncall_entry *entry,
  938. -    unsigned char *buf,
  939. -    ULONG buf_len,
  940. -    ULONG *len)
  941. -{
  942. -    NTSTATUS status = STATUS_SUCCESS;
  943. -    ULONG header_len = 0;
  944. -    unsigned char *tmp = buf;
  945. -
  946. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  947. -    if (status) goto out;
  948. -    else tmp += *len;
  949. -
  950. -    /* 03/25/2011: Kernel crash to nfsd not running but mount upcall cued up */
  951. -    if (!MmIsAddressValid(entry->u.Mount.srv_name) ||
  952. -            !MmIsAddressValid(entry->u.Mount.root)) {
  953. -        status = STATUS_INTERNAL_ERROR;
  954. -        goto out;
  955. -    }
  956. -    header_len = *len + length_as_utf8(entry->u.Mount.srv_name) +
  957. -        length_as_utf8(entry->u.Mount.root) + 4 * sizeof(DWORD);
  958. -    if (header_len > buf_len) {
  959. -        status = STATUS_INSUFFICIENT_RESOURCES;
  960. -        goto out;
  961. -    }
  962. -    status = marshall_unicode_as_utf8(&tmp, entry->u.Mount.srv_name);
  963. -    if (status) goto out;
  964. -    status = marshall_unicode_as_utf8(&tmp, entry->u.Mount.root);
  965. -    if (status) goto out;
  966. -    RtlCopyMemory(tmp, &entry->u.Mount.sec_flavor, sizeof(DWORD));
  967. -    tmp += sizeof(DWORD);
  968. -    RtlCopyMemory(tmp, &entry->u.Mount.rsize, sizeof(DWORD));
  969. -    tmp += sizeof(DWORD);
  970. -    RtlCopyMemory(tmp, &entry->u.Mount.wsize, sizeof(DWORD));
  971. -    tmp += sizeof(DWORD);
  972. -    RtlCopyMemory(tmp, &entry->u.Mount.use_nfspubfh, sizeof(DWORD));
  973. -
  974. -    *len = header_len;
  975. -
  976. -#ifdef DEBUG_MARSHAL_DETAIL
  977. -    DbgP("marshal_nfs41_mount: server name='%wZ' mount point='%wZ' "
  978. -         "sec_flavor='%s' rsize=%d wsize=%d use_nfspubfh=%d\n",
  979. -        entry->u.Mount.srv_name, entry->u.Mount.root,
  980. -         secflavorop2name(entry->u.Mount.sec_flavor),
  981. -         (int)entry->u.Mount.rsize, (int)entry->u.Mount.wsize,
  982. -         (int)entry->u.Mount.use_nfspubfh);
  983. -#endif
  984. -out:
  985. -    return status;
  986. -}
  987. -
  988. -static NTSTATUS marshal_nfs41_unmount(
  989. -    nfs41_updowncall_entry *entry,
  990. -    unsigned char *buf,
  991. -    ULONG buf_len,
  992. -    ULONG *len)
  993. -{
  994. -    return marshal_nfs41_header(entry, buf, buf_len, len);
  995. -}
  996. -
  997. -static NTSTATUS marshal_nfs41_open(
  998. -    nfs41_updowncall_entry *entry,
  999. -    unsigned char *buf,
  1000. -    ULONG buf_len,
  1001. -    ULONG *len)
  1002. -{
  1003. -    NTSTATUS status = STATUS_SUCCESS;
  1004. -    ULONG header_len = 0;
  1005. -    unsigned char *tmp = buf;
  1006. -
  1007. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1008. -    if (status) goto out;
  1009. -    else tmp += *len;
  1010. -
  1011. -    header_len = *len + length_as_utf8(entry->filename) +
  1012. -        7 * sizeof(ULONG) +
  1013. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  1014. -        2 * sizeof(DWORD) +
  1015. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  1016. -        2 * sizeof(HANDLE) +
  1017. -        length_as_utf8(&entry->u.Open.symlink);
  1018. -    if (header_len > buf_len) {
  1019. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1020. -        goto out;
  1021. -    }
  1022. -    status = marshall_unicode_as_utf8(&tmp, entry->filename);
  1023. -    if (status) goto out;
  1024. -    RtlCopyMemory(tmp, &entry->u.Open.access_mask,
  1025. -        sizeof(entry->u.Open.access_mask));
  1026. -    tmp += sizeof(entry->u.Open.access_mask);
  1027. -    RtlCopyMemory(tmp, &entry->u.Open.access_mode,
  1028. -        sizeof(entry->u.Open.access_mode));
  1029. -    tmp += sizeof(entry->u.Open.access_mode);
  1030. -    RtlCopyMemory(tmp, &entry->u.Open.attrs, sizeof(entry->u.Open.attrs));
  1031. -    tmp += sizeof(entry->u.Open.attrs);
  1032. -    RtlCopyMemory(tmp, &entry->u.Open.copts, sizeof(entry->u.Open.copts));
  1033. -    tmp += sizeof(entry->u.Open.copts);
  1034. -    RtlCopyMemory(tmp, &entry->u.Open.disp, sizeof(entry->u.Open.disp));
  1035. -    tmp += sizeof(entry->u.Open.disp);
  1036. -    RtlCopyMemory(tmp, &entry->u.Open.open_owner_id,
  1037. -        sizeof(entry->u.Open.open_owner_id));
  1038. -    tmp += sizeof(entry->u.Open.open_owner_id);
  1039. -    RtlCopyMemory(tmp, &entry->u.Open.mode, sizeof(DWORD));
  1040. -    tmp += sizeof(DWORD);
  1041. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  1042. -    RtlCopyMemory(tmp, &entry->u.Open.owner_local_uid, sizeof(DWORD));
  1043. -    tmp += sizeof(DWORD);
  1044. -    RtlCopyMemory(tmp, &entry->u.Open.owner_group_local_gid, sizeof(DWORD));
  1045. -    tmp += sizeof(DWORD);
  1046. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  1047. -    RtlCopyMemory(tmp, &entry->u.Open.srv_open, sizeof(HANDLE));
  1048. -    tmp += sizeof(HANDLE);
  1049. -    status = marshall_unicode_as_utf8(&tmp, &entry->u.Open.symlink);
  1050. -    if (status) goto out;
  1051. -
  1052. -    __try {
  1053. -        if (entry->u.Open.EaMdl) {
  1054. -            entry->u.Open.EaBuffer =
  1055. -                MmMapLockedPagesSpecifyCache(entry->u.Open.EaMdl,
  1056. -                    UserMode, MmCached, NULL, TRUE,
  1057. -                    NormalPagePriority|MdlMappingNoExecute);
  1058. -            if (entry->u.Open.EaBuffer == NULL) {
  1059. -                print_error("marshal_nfs41_open: "
  1060. -                    "MmMapLockedPagesSpecifyCache() failed to "
  1061. -                    "map pages\n");
  1062. -                status = STATUS_INSUFFICIENT_RESOURCES;
  1063. -                goto out;
  1064. -            }
  1065. -        }
  1066. -    } __except(EXCEPTION_EXECUTE_HANDLER) {
  1067. -        print_error("marshal_nfs41_open: Call to "
  1068. -            "MmMapLockedPagesSpecifyCache() failed "
  1069. -            "due to exception 0x%x\n", (int)GetExceptionCode());
  1070. -        status = STATUS_ACCESS_VIOLATION;
  1071. -        goto out;
  1072. -    }
  1073. -    RtlCopyMemory(tmp, &entry->u.Open.EaBuffer, sizeof(HANDLE));
  1074. -    *len = header_len;
  1075. -
  1076. -#ifdef DEBUG_MARSHAL_DETAIL
  1077. -    DbgP("marshal_nfs41_open: name='%wZ' mask=0x%x access=0x%x attrs=0x%x "
  1078. -         "opts=0x%x dispo=0x%x open_owner_id=0x%x mode=0%o "
  1079. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  1080. -         "owner_local_uid=%lu owner_group_local_gid=%lu "
  1081. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  1082. -         "srv_open=0x%p ea=0x%p\n",
  1083. -         entry->filename, entry->u.Open.access_mask,
  1084. -         entry->u.Open.access_mode, entry->u.Open.attrs, entry->u.Open.copts,
  1085. -         entry->u.Open.disp, entry->u.Open.open_owner_id, entry->u.Open.mode,
  1086. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  1087. -         entry->u.Open.owner_local_uid,entry->u.Open.owner_group_local_gid,
  1088. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  1089. -         entry->u.Open.srv_open, entry->u.Open.EaBuffer);
  1090. -#endif
  1091. -out:
  1092. -    return status;
  1093. -}
  1094. -
  1095. -static NTSTATUS marshal_nfs41_rw(
  1096. -    nfs41_updowncall_entry *entry,
  1097. -    unsigned char *buf,
  1098. -    ULONG buf_len,
  1099. -    ULONG *len)
  1100. -{
  1101. -    NTSTATUS status = STATUS_SUCCESS;
  1102. -    ULONG header_len = 0;
  1103. -    unsigned char *tmp = buf;
  1104. -
  1105. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1106. -    if (status) goto out;
  1107. -    else tmp += *len;
  1108. -
  1109. -    header_len = *len + sizeof(entry->buf_len) +
  1110. -        sizeof(entry->u.ReadWrite.offset) + sizeof(HANDLE);
  1111. -    if (header_len > buf_len) {
  1112. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1113. -        goto out;
  1114. -    }
  1115. -
  1116. -    RtlCopyMemory(tmp, &entry->buf_len, sizeof(entry->buf_len));
  1117. -    tmp += sizeof(entry->buf_len);
  1118. -    RtlCopyMemory(tmp, &entry->u.ReadWrite.offset,
  1119. -        sizeof(entry->u.ReadWrite.offset));
  1120. -    tmp += sizeof(entry->u.ReadWrite.offset);
  1121. -    __try {
  1122. -#pragma warning( push )
  1123. -/*
  1124. - * C28145: "The opaque MDL structure should not be modified by a
  1125. - * driver.", |MDL_MAPPING_CAN_FAIL| is the exception
  1126. - */
  1127. -#pragma warning (disable : 28145)
  1128. -        entry->u.ReadWrite.MdlAddress->MdlFlags |= MDL_MAPPING_CAN_FAIL;
  1129. -#pragma warning( pop )
  1130. -        entry->buf =
  1131. -            MmMapLockedPagesSpecifyCache(entry->u.ReadWrite.MdlAddress,
  1132. -                UserMode, MmCached, NULL, TRUE, NormalPagePriority);
  1133. -        if (entry->buf == NULL) {
  1134. -            print_error("marshal_nfs41_rw: "
  1135. -                "MmMapLockedPagesSpecifyCache() failed to map pages\n");
  1136. -            status = STATUS_INSUFFICIENT_RESOURCES;
  1137. -            goto out;
  1138. -        }
  1139. -    } __except(EXCEPTION_EXECUTE_HANDLER) {
  1140. -        NTSTATUS code;
  1141. -        code = GetExceptionCode();
  1142. -        print_error("marshal_nfs41_rw: Call to "
  1143. -            "MmMapLockedPagesSpecifyCache() failed due to "
  1144. -            "exception 0x%x\n", (int)code);
  1145. -        status = STATUS_ACCESS_VIOLATION;
  1146. -        goto out;
  1147. -    }
  1148. -    RtlCopyMemory(tmp, &entry->buf, sizeof(HANDLE));
  1149. -    *len = header_len;
  1150. -
  1151. -#ifdef DEBUG_MARSHAL_DETAIL_RW
  1152. -    DbgP("marshal_nfs41_rw: len=%lu offset=%llu "
  1153. -        "MdlAddress=0x%p Userspace=0x%p\n",
  1154. -        entry->buf_len, entry->u.ReadWrite.offset,
  1155. -        entry->u.ReadWrite.MdlAddress, entry->buf);
  1156. -#endif
  1157. -out:
  1158. -    return status;
  1159. -}
  1160. -
  1161. -static NTSTATUS marshal_nfs41_lock(
  1162. -    nfs41_updowncall_entry *entry,
  1163. -    unsigned char *buf,
  1164. -    ULONG buf_len,
  1165. -    ULONG *len)
  1166. -{
  1167. -    NTSTATUS status = STATUS_SUCCESS;
  1168. -    ULONG header_len = 0;
  1169. -    unsigned char *tmp = buf;
  1170. -
  1171. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1172. -    if (status) goto out;
  1173. -    else tmp += *len;
  1174. -
  1175. -    header_len = *len + 2 * sizeof(LONGLONG) + 2 * sizeof(BOOLEAN);
  1176. -    if (header_len > buf_len) {
  1177. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1178. -        goto out;
  1179. -    }
  1180. -    RtlCopyMemory(tmp, &entry->u.Lock.offset, sizeof(LONGLONG));
  1181. -    tmp += sizeof(LONGLONG);
  1182. -    RtlCopyMemory(tmp, &entry->u.Lock.length, sizeof(LONGLONG));
  1183. -    tmp += sizeof(LONGLONG);
  1184. -    RtlCopyMemory(tmp, &entry->u.Lock.exclusive, sizeof(BOOLEAN));
  1185. -    tmp += sizeof(BOOLEAN);
  1186. -    RtlCopyMemory(tmp, &entry->u.Lock.blocking, sizeof(BOOLEAN));
  1187. -    *len = header_len;
  1188. -
  1189. -#ifdef DEBUG_MARSHAL_DETAIL
  1190. -    DbgP("marshal_nfs41_lock: "
  1191. -        "offset=0x%llx length=0x%llx exclusive=%u "
  1192. -        "blocking=%u\n", entry->u.Lock.offset, entry->u.Lock.length,
  1193. -        entry->u.Lock.exclusive, entry->u.Lock.blocking);
  1194. -#endif
  1195. -out:
  1196. -    return status;
  1197. -}
  1198. -
  1199. -static NTSTATUS marshal_nfs41_unlock(
  1200. -    nfs41_updowncall_entry *entry,
  1201. -    unsigned char *buf,
  1202. -    ULONG buf_len,
  1203. -    ULONG *len)
  1204. -{
  1205. -    NTSTATUS status = STATUS_SUCCESS;
  1206. -    ULONG header_len = 0;
  1207. -    unsigned char *tmp = buf;
  1208. -    PLOWIO_LOCK_LIST lock;
  1209. -
  1210. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1211. -    if (status) goto out;
  1212. -    else tmp += *len;
  1213. -
  1214. -    header_len = *len + sizeof(ULONG) +
  1215. -        (size_t)entry->u.Unlock.count * 2 * sizeof(LONGLONG);
  1216. -    if (header_len > buf_len) {
  1217. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1218. -        goto out;
  1219. -    }
  1220. -    RtlCopyMemory(tmp, &entry->u.Unlock.count, sizeof(ULONG));
  1221. -    tmp += sizeof(ULONG);
  1222. -
  1223. -    lock = &entry->u.Unlock.locks;
  1224. -    while (lock) {
  1225. -        RtlCopyMemory(tmp, &lock->ByteOffset, sizeof(LONGLONG));
  1226. -        tmp += sizeof(LONGLONG);
  1227. -        RtlCopyMemory(tmp, &lock->Length, sizeof(LONGLONG));
  1228. -        tmp += sizeof(LONGLONG);
  1229. -        lock = lock->Next;
  1230. -    }
  1231. -    *len = header_len;
  1232. -
  1233. -#ifdef DEBUG_MARSHAL_DETAIL
  1234. -    DbgP("marshal_nfs41_unlock: count=%u\n", entry->u.Unlock.count);
  1235. -#endif
  1236. -out:
  1237. -    return status;
  1238. -}
  1239. -
  1240. -static NTSTATUS marshal_nfs41_close(
  1241. -    nfs41_updowncall_entry *entry,
  1242. -    unsigned char *buf,
  1243. -    ULONG buf_len,
  1244. -    ULONG *len)
  1245. -{
  1246. -    NTSTATUS status = STATUS_SUCCESS;
  1247. -    ULONG header_len = 0;
  1248. -    unsigned char *tmp = buf;
  1249. -
  1250. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1251. -    if (status) goto out;
  1252. -    else tmp += *len;
  1253. -
  1254. -    header_len = *len + sizeof(BOOLEAN) + sizeof(HANDLE);
  1255. -    if (entry->u.Close.remove)
  1256. -        header_len += length_as_utf8(entry->filename) +
  1257. -            sizeof(BOOLEAN);
  1258. -
  1259. -    if (header_len > buf_len) {
  1260. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1261. -        goto out;
  1262. -    }
  1263. -    RtlCopyMemory(tmp, &entry->u.Close.remove, sizeof(BOOLEAN));
  1264. -    tmp += sizeof(BOOLEAN);
  1265. -    RtlCopyMemory(tmp, &entry->u.Close.srv_open, sizeof(HANDLE));
  1266. -    if (entry->u.Close.remove) {
  1267. -        tmp += sizeof(HANDLE);
  1268. -        status = marshall_unicode_as_utf8(&tmp, entry->filename);
  1269. -        if (status) goto out;
  1270. -        RtlCopyMemory(tmp, &entry->u.Close.renamed, sizeof(BOOLEAN));
  1271. -    }
  1272. -    *len = header_len;
  1273. -
  1274. -#ifdef DEBUG_MARSHAL_DETAIL
  1275. -    DbgP("marshal_nfs41_close: name='%wZ' remove=%d srv_open=0x%p renamed=%d\n",
  1276. -        entry->filename->Length?entry->filename:&SLASH,
  1277. -        entry->u.Close.remove, entry->u.Close.srv_open, entry->u.Close.renamed);
  1278. -#endif
  1279. -out:
  1280. -    return status;
  1281. -}
  1282. -
  1283. -static NTSTATUS marshal_nfs41_dirquery(
  1284. -    nfs41_updowncall_entry *entry,
  1285. -    unsigned char *buf,
  1286. -    ULONG buf_len,
  1287. -    ULONG *len)
  1288. -{
  1289. -    NTSTATUS status = STATUS_SUCCESS;
  1290. -    ULONG header_len = 0;
  1291. -    unsigned char *tmp = buf;
  1292. -
  1293. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1294. -    if (status) goto out;
  1295. -    else tmp += *len;
  1296. -
  1297. -    header_len = *len + 2 * sizeof(ULONG) + sizeof(HANDLE) +
  1298. -        length_as_utf8(entry->u.QueryFile.filter) + 3 * sizeof(BOOLEAN);
  1299. -    if (header_len > buf_len) {
  1300. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1301. -        goto out;
  1302. -    }
  1303. -
  1304. -    RtlCopyMemory(tmp, &entry->u.QueryFile.InfoClass, sizeof(ULONG));
  1305. -    tmp += sizeof(ULONG);
  1306. -    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  1307. -    tmp += sizeof(ULONG);
  1308. -    status = marshall_unicode_as_utf8(&tmp, entry->u.QueryFile.filter);
  1309. -    if (status) goto out;
  1310. -    RtlCopyMemory(tmp, &entry->u.QueryFile.initial_query, sizeof(BOOLEAN));
  1311. -    tmp += sizeof(BOOLEAN);
  1312. -    RtlCopyMemory(tmp, &entry->u.QueryFile.restart_scan, sizeof(BOOLEAN));
  1313. -    tmp += sizeof(BOOLEAN);
  1314. -    RtlCopyMemory(tmp, &entry->u.QueryFile.return_single, sizeof(BOOLEAN));
  1315. -    tmp += sizeof(BOOLEAN);
  1316. -    __try {
  1317. -        entry->u.QueryFile.mdl_buf =
  1318. -            MmMapLockedPagesSpecifyCache(entry->u.QueryFile.mdl,
  1319. -                UserMode, MmCached, NULL, TRUE,
  1320. -                NormalPagePriority|MdlMappingNoExecute);
  1321. -        if (entry->u.QueryFile.mdl_buf == NULL) {
  1322. -            print_error("marshal_nfs41_dirquery: "
  1323. -                "MmMapLockedPagesSpecifyCache() failed to map pages\n");
  1324. -            status = STATUS_INSUFFICIENT_RESOURCES;
  1325. -            goto out;
  1326. -        }
  1327. -    } __except(EXCEPTION_EXECUTE_HANDLER) {
  1328. -        NTSTATUS code;
  1329. -        code = GetExceptionCode();
  1330. -        print_error("marshal_nfs41_dirquery: Call to "
  1331. -            "MmMapLockedPagesSpecifyCache() failed "
  1332. -            "due to exception 0x%x\n", (int)code);
  1333. -        status = STATUS_ACCESS_VIOLATION;
  1334. -        goto out;
  1335. -    }
  1336. -    RtlCopyMemory(tmp, &entry->u.QueryFile.mdl_buf, sizeof(HANDLE));
  1337. -    *len = header_len;
  1338. -
  1339. -#ifdef DEBUG_MARSHAL_DETAIL
  1340. -    DbgP("marshal_nfs41_dirquery: filter='%wZ' class=%d len=%d "
  1341. -         "1st\\restart\\single=%d\\%d\\%d\n", entry->u.QueryFile.filter,
  1342. -         entry->u.QueryFile.InfoClass, entry->buf_len,
  1343. -         entry->u.QueryFile.initial_query, entry->u.QueryFile.restart_scan,
  1344. -         entry->u.QueryFile.return_single);
  1345. -#endif
  1346. -out:
  1347. -    return status;
  1348. -}
  1349. -
  1350. -static NTSTATUS marshal_nfs41_filequery(
  1351. -    nfs41_updowncall_entry *entry,
  1352. -    unsigned char *buf,
  1353. -    ULONG buf_len,
  1354. -    ULONG *len)
  1355. -{
  1356. -    NTSTATUS status = STATUS_SUCCESS;
  1357. -    ULONG header_len = 0;
  1358. -    unsigned char *tmp = buf;
  1359. -
  1360. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1361. -    if (status) goto out;
  1362. -    else tmp += *len;
  1363. -
  1364. -    header_len = *len + 2 * sizeof(ULONG) + 2*sizeof(HANDLE);
  1365. -    if (header_len > buf_len) {
  1366. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1367. -        goto out;
  1368. -    }
  1369. -    RtlCopyMemory(tmp, &entry->u.QueryFile.InfoClass, sizeof(ULONG));
  1370. -    tmp += sizeof(ULONG);
  1371. -    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  1372. -    tmp += sizeof(ULONG);
  1373. -    RtlCopyMemory(tmp, &entry->session, sizeof(HANDLE));
  1374. -    tmp += sizeof(HANDLE);
  1375. -    RtlCopyMemory(tmp, &entry->open_state, sizeof(HANDLE));
  1376. -    *len = header_len;
  1377. -
  1378. -#ifdef DEBUG_MARSHAL_DETAIL
  1379. -    DbgP("marshal_nfs41_filequery: class=%d\n", entry->u.QueryFile.InfoClass);
  1380. -#endif
  1381. -out:
  1382. -    return status;
  1383. -}
  1384. -
  1385. -static NTSTATUS marshal_nfs41_fileset(
  1386. -    nfs41_updowncall_entry *entry,
  1387. -    unsigned char *buf,
  1388. -    ULONG buf_len,
  1389. -    ULONG *len)
  1390. -{
  1391. -    NTSTATUS status = STATUS_SUCCESS;
  1392. -    ULONG header_len = 0;
  1393. -    unsigned char *tmp = buf;
  1394. -
  1395. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1396. -    if (status) goto out;
  1397. -    else tmp += *len;
  1398. -
  1399. -    header_len = *len + length_as_utf8(entry->filename) +
  1400. -        2 * sizeof(ULONG) + entry->buf_len;
  1401. -    if (header_len > buf_len) {
  1402. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1403. -        goto out;
  1404. -    }
  1405. -    status = marshall_unicode_as_utf8(&tmp, entry->filename);
  1406. -    if (status) goto out;
  1407. -    RtlCopyMemory(tmp, &entry->u.SetFile.InfoClass, sizeof(ULONG));
  1408. -    tmp += sizeof(ULONG);
  1409. -    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  1410. -    tmp += sizeof(ULONG);
  1411. -    RtlCopyMemory(tmp, entry->buf, entry->buf_len);
  1412. -    *len = header_len;
  1413. -
  1414. -#ifdef DEBUG_MARSHAL_DETAIL
  1415. -    DbgP("marshal_nfs41_fileset: filename='%wZ' class=%d\n",
  1416. -        entry->filename, entry->u.SetFile.InfoClass);
  1417. -#endif
  1418. -out:
  1419. -    return status;
  1420. -}
  1421. -
  1422. -static NTSTATUS marshal_nfs41_easet(
  1423. -    nfs41_updowncall_entry *entry,
  1424. -    unsigned char *buf,
  1425. -    ULONG buf_len,
  1426. -    ULONG *len)
  1427. -{
  1428. -    NTSTATUS status = STATUS_SUCCESS;
  1429. -    ULONG header_len = 0;
  1430. -    unsigned char *tmp = buf;
  1431. -
  1432. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1433. -    if (status) goto out;
  1434. -    else tmp += *len;
  1435. -
  1436. -    header_len = *len + length_as_utf8(entry->filename) +
  1437. -        sizeof(ULONG) + entry->buf_len  + sizeof(DWORD);
  1438. -    if (header_len > buf_len) {
  1439. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1440. -        goto out;
  1441. -    }
  1442. -
  1443. -    status = marshall_unicode_as_utf8(&tmp, entry->filename);
  1444. -    if (status) goto out;
  1445. -    RtlCopyMemory(tmp, &entry->u.SetEa.mode, sizeof(DWORD));
  1446. -    tmp += sizeof(DWORD);
  1447. -    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  1448. -    tmp += sizeof(ULONG);
  1449. -    RtlCopyMemory(tmp, entry->buf, entry->buf_len);    
  1450. -    *len = header_len;
  1451. -
  1452. -#ifdef DEBUG_MARSHAL_DETAIL
  1453. -    DbgP("marshal_nfs41_easet: filename='%wZ', buflen=%d mode=0x%x\n",
  1454. -        entry->filename, entry->buf_len, entry->u.SetEa.mode);
  1455. -#endif
  1456. -out:
  1457. -    return status;
  1458. -}
  1459. -
  1460. -static NTSTATUS marshal_nfs41_eaget(
  1461. -    nfs41_updowncall_entry *entry,
  1462. -    unsigned char *buf,
  1463. -    ULONG buf_len,
  1464. -    ULONG *len)
  1465. -{
  1466. -    NTSTATUS status = STATUS_SUCCESS;
  1467. -    ULONG header_len = 0;
  1468. -    unsigned char *tmp = buf;
  1469. -
  1470. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1471. -    if (status) goto out;
  1472. -    else tmp += *len;
  1473. -
  1474. -    header_len = *len + length_as_utf8(entry->filename) +
  1475. -        3 * sizeof(ULONG) + entry->u.QueryEa.EaListLength + 2 * sizeof(BOOLEAN);
  1476. -
  1477. -    if (header_len > buf_len) {
  1478. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1479. -        goto out;
  1480. -    }
  1481. -
  1482. -    status = marshall_unicode_as_utf8(&tmp, entry->filename);
  1483. -    if (status) goto out;
  1484. -    RtlCopyMemory(tmp, &entry->u.QueryEa.EaIndex, sizeof(ULONG));
  1485. -    tmp += sizeof(ULONG);
  1486. -    RtlCopyMemory(tmp, &entry->u.QueryEa.RestartScan, sizeof(BOOLEAN));
  1487. -    tmp += sizeof(BOOLEAN);
  1488. -    RtlCopyMemory(tmp, &entry->u.QueryEa.ReturnSingleEntry, sizeof(BOOLEAN));
  1489. -    tmp += sizeof(BOOLEAN);
  1490. -    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  1491. -    tmp += sizeof(ULONG);
  1492. -    RtlCopyMemory(tmp, &entry->u.QueryEa.EaListLength, sizeof(ULONG));
  1493. -    tmp += sizeof(ULONG);
  1494. -    if (entry->u.QueryEa.EaList && entry->u.QueryEa.EaListLength)
  1495. -        RtlCopyMemory(tmp, entry->u.QueryEa.EaList,
  1496. -            entry->u.QueryEa.EaListLength);
  1497. -    *len = header_len;
  1498. -
  1499. -#ifdef DEBUG_MARSHAL_DETAIL
  1500. -    DbgP("marshal_nfs41_eaget: filename='%wZ', index=%d list_len=%d "
  1501. -        "rescan=%d single=%d\n", entry->filename,
  1502. -        entry->u.QueryEa.EaIndex, entry->u.QueryEa.EaListLength,
  1503. -        entry->u.QueryEa.RestartScan, entry->u.QueryEa.ReturnSingleEntry);
  1504. -#endif
  1505. -out:
  1506. -    return status;
  1507. -}
  1508. -
  1509. -static NTSTATUS marshal_nfs41_symlink(
  1510. -    nfs41_updowncall_entry *entry,
  1511. -    unsigned char *buf,
  1512. -    ULONG buf_len,
  1513. -    ULONG *len)
  1514. -{
  1515. -    NTSTATUS status = STATUS_SUCCESS;
  1516. -    ULONG header_len = 0;
  1517. -    unsigned char *tmp = buf;
  1518. -
  1519. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1520. -    if (status) goto out;
  1521. -    else tmp += *len;
  1522. -
  1523. -    header_len = *len + sizeof(BOOLEAN) + length_as_utf8(entry->filename);
  1524. -    if (entry->u.Symlink.set)
  1525. -        header_len += length_as_utf8(entry->u.Symlink.target);
  1526. -    if (header_len > buf_len) {
  1527. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1528. -        goto out;
  1529. -    }
  1530. -
  1531. -    status = marshall_unicode_as_utf8(&tmp, entry->filename);
  1532. -    if (status) goto out;
  1533. -    RtlCopyMemory(tmp, &entry->u.Symlink.set, sizeof(BOOLEAN));
  1534. -    tmp += sizeof(BOOLEAN);
  1535. -    if (entry->u.Symlink.set) {
  1536. -        status = marshall_unicode_as_utf8(&tmp, entry->u.Symlink.target);
  1537. -        if (status) goto out;
  1538. -    }
  1539. -    *len = header_len;
  1540. -
  1541. -#ifdef DEBUG_MARSHAL_DETAIL
  1542. -    DbgP("marshal_nfs41_symlink: name '%wZ' symlink target '%wZ'\n",
  1543. -         entry->filename,
  1544. -         entry->u.Symlink.set?entry->u.Symlink.target : NULL);
  1545. -#endif
  1546. -out:
  1547. -    return status;
  1548. -}
  1549. -
  1550. -static NTSTATUS marshal_nfs41_volume(
  1551. -    nfs41_updowncall_entry *entry,
  1552. -    unsigned char *buf,
  1553. -    ULONG buf_len,
  1554. -    ULONG *len)
  1555. -{
  1556. -    NTSTATUS status = STATUS_SUCCESS;
  1557. -    ULONG header_len = 0;
  1558. -    unsigned char *tmp = buf;
  1559. -
  1560. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1561. -    if (status) goto out;
  1562. -    else tmp += *len;
  1563. -
  1564. -    header_len = *len + sizeof(FS_INFORMATION_CLASS);
  1565. -    if (header_len > buf_len) {
  1566. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1567. -        goto out;
  1568. -    }
  1569. -
  1570. -    RtlCopyMemory(tmp, &entry->u.Volume.query, sizeof(FS_INFORMATION_CLASS));
  1571. -    *len = header_len;
  1572. -
  1573. -#ifdef DEBUG_MARSHAL_DETAIL
  1574. -    DbgP("marshal_nfs41_volume: class=%d\n", entry->u.Volume.query);
  1575. -#endif
  1576. -out:
  1577. -    return status;
  1578. -}
  1579. -
  1580. -static NTSTATUS marshal_nfs41_getacl(
  1581. -    nfs41_updowncall_entry *entry,
  1582. -    unsigned char *buf,
  1583. -    ULONG buf_len,
  1584. -    ULONG *len)
  1585. -{
  1586. -    NTSTATUS status = STATUS_SUCCESS;
  1587. -    ULONG header_len = 0;
  1588. -    unsigned char *tmp = buf;
  1589. -
  1590. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1591. -    if (status) goto out;
  1592. -    else tmp += *len;
  1593. -
  1594. -    header_len = *len + sizeof(SECURITY_INFORMATION);
  1595. -    if (header_len > buf_len) {
  1596. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1597. -        goto out;
  1598. -    }
  1599. -
  1600. -    RtlCopyMemory(tmp, &entry->u.Acl.query, sizeof(SECURITY_INFORMATION));
  1601. -    *len = header_len;
  1602. -
  1603. -#ifdef DEBUG_MARSHAL_DETAIL
  1604. -    DbgP("marshal_nfs41_getacl: class=0x%x\n", entry->u.Acl.query);
  1605. -#endif
  1606. -out:
  1607. -    return status;
  1608. -}
  1609. -
  1610. -static NTSTATUS marshal_nfs41_setacl(
  1611. -    nfs41_updowncall_entry *entry,
  1612. -    unsigned char *buf,
  1613. -    ULONG buf_len,
  1614. -    ULONG *len)
  1615. -{
  1616. -    NTSTATUS status = STATUS_SUCCESS;
  1617. -    ULONG header_len = 0;
  1618. -    unsigned char *tmp = buf;
  1619. -
  1620. -    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  1621. -    if (status) goto out;
  1622. -    else tmp += *len;
  1623. -
  1624. -    header_len = *len + sizeof(SECURITY_INFORMATION) +
  1625. -        sizeof(ULONG) + entry->buf_len;
  1626. -    if (header_len > buf_len) {
  1627. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1628. -        goto out;
  1629. -    }
  1630. -
  1631. -    RtlCopyMemory(tmp, &entry->u.Acl.query, sizeof(SECURITY_INFORMATION));
  1632. -    tmp += sizeof(SECURITY_INFORMATION);
  1633. -    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  1634. -    tmp += sizeof(ULONG);
  1635. -    RtlCopyMemory(tmp, entry->buf, entry->buf_len);
  1636. -    *len = header_len;
  1637. -
  1638. -#ifdef DEBUG_MARSHAL_DETAIL
  1639. -    DbgP("marshal_nfs41_setacl: class=0x%x sec_desc_len=%lu\n",
  1640. -         entry->u.Acl.query, entry->buf_len);
  1641. -#endif
  1642. -out:
  1643. -    return status;
  1644. -}
  1645. -
  1646. -static NTSTATUS marshal_nfs41_shutdown(
  1647. -    nfs41_updowncall_entry *entry,
  1648. -    unsigned char *buf,
  1649. -    ULONG buf_len,
  1650. -    ULONG *len)
  1651. -{
  1652. -    return marshal_nfs41_header(entry, buf, buf_len, len);
  1653. -}
  1654. -
  1655. -static NTSTATUS nfs41_invalidate_cache(
  1656. -    IN PRX_CONTEXT RxContext)
  1657. -{
  1658. -    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  1659. -    unsigned char *buf = LowIoContext->ParamsFor.IoCtl.pInputBuffer;
  1660. -    ULONG flag = DISABLE_CACHING;
  1661. -    PMRX_SRV_OPEN srv_open;
  1662. -    NTSTATUS status;
  1663. -
  1664. -    RtlCopyMemory(&srv_open, buf, sizeof(HANDLE));
  1665. -#ifdef DEBUG_INVALIDATE_CACHE
  1666. -    DbgP("nfs41_invalidate_cache: received srv_open=0x%p '%wZ'\n",
  1667. -        srv_open, srv_open->pAlreadyPrefixedName);
  1668. -#endif
  1669. -    if (MmIsAddressValid(srv_open)) {
  1670. -        RxIndicateChangeOfBufferingStateForSrvOpen(
  1671. -            srv_open->pFcb->pNetRoot->pSrvCall, srv_open,
  1672. -            srv_open->Key, ULongToPtr(flag));
  1673. -        status = STATUS_SUCCESS;
  1674. -    }
  1675. -    else {
  1676. -        print_error("nfs41_invalidate_cache: "
  1677. -            "invalid ptr srv_open=0x%p file='%wZ'\n",
  1678. -            srv_open, srv_open->pAlreadyPrefixedName);
  1679. -        status = STATUS_INVALID_HANDLE;
  1680. -    }
  1681. -
  1682. -    return status;
  1683. -}
  1684. -
  1685. -static NTSTATUS handle_upcall(
  1686. -    IN PRX_CONTEXT RxContext,
  1687. -    IN nfs41_updowncall_entry *entry,
  1688. -    OUT ULONG *len)
  1689. -{
  1690. -    NTSTATUS status = STATUS_SUCCESS;
  1691. -    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  1692. -    ULONG cbOut = LowIoContext->ParamsFor.IoCtl.OutputBufferLength;
  1693. -    unsigned char *pbOut = LowIoContext->ParamsFor.IoCtl.pOutputBuffer;
  1694. -
  1695. -#ifdef NFS41_DRIVER_STABILITY_HACKS
  1696. -    /*
  1697. -     * Workaround for random crashes like this while compiling
  1698. -     * the "gcc" compiler with a highly-parallel build.
  1699. -     * Stack trace usually looks like this:
  1700. -     * ---- snip ----
  1701. -     * nt!SeTokenCanImpersonate+0x47
  1702. -     * nt!PsImpersonateClient+0x126
  1703. -     * nt!SeImpersonateClientEx+0x35
  1704. -     * nfs41_driver!handle_upcall+0x59 [C:\cygwin64\home\roland_mainz\work\msnfs41_uidmapping\ms-nfs41-client\sys\nfs41_driver.c @ 1367]
  1705. -     * nfs41_driver!nfs41_upcall+0xe7 [C:\cygwin64\home\roland_mainz\work\msnfs41_uidmapping\ms-nfs41-client\sys\nfs41_driver.c @ 1578]
  1706. -     * nfs41_driver!nfs41_DevFcbXXXControlFile+0x128 [C:\cygwin64\home\roland_mainz\work\msnfs41_uidmapping\ms-nfs41-client\sys\nfs41_driver.c @ 2418]
  1707. -     * nfs41_driver!RxXXXControlFileCallthru+0x76 [base\fs\rdr2\rdbss\ntdevfcb.c @ 130]
  1708. -     * nfs41_driver!RxCommonDevFCBIoCtl+0x58 [base\fs\rdr2\rdbss\ntdevfcb.c @ 491]
  1709. -     * nfs41_driver!RxFsdCommonDispatch+0x442 [base\fs\rdr2\rdbss\ntfsd.c @ 848]
  1710. -     * nfs41_driver!RxFsdDispatch+0xfd [base\fs\rdr2\rdbss\ntfsd.c @ 442]
  1711. -     * nfs41_driver!nfs41_FsdDispatch+0x67 [C:\cygwin64\home\roland_mainz\work\msnfs41_uidmapping\ms-nfs41-client\sys\nfs41_driver.c @ 6863]
  1712. -     * nt!IofCallDriver+0x55
  1713. -     * mup!MupiCallUncProvider+0xb8
  1714. -     * mup!MupStateMachine+0x59
  1715. -     * mup!MupFsdIrpPassThrough+0x17e
  1716. -     * nt!IofCallDriver+0x55
  1717. -     * FLTMGR!FltpDispatch+0xd6
  1718. -     * nt!IofCallDriver+0x55
  1719. -     * nt!IopSynchronousServiceTail+0x34c
  1720. -     * nt!IopXxxControlFile+0xd13
  1721. -     * nt!NtDeviceIoControlFile+0x56
  1722. -     * nt!KiSystemServiceCopyEnd+0x25
  1723. -     * ntdll!NtDeviceIoControlFile+0x14
  1724. -     * KERNELBASE!DeviceIoControl+0x6b
  1725. -     * KERNEL32!DeviceIoControlImplementation+0x81
  1726. -     * nfsd_debug+0xc7b14
  1727. -     * nfsd_debug+0xc79fb
  1728. -     * nfsd_debug+0x171e80
  1729. -     * KERNEL32!BaseThreadInitThunk+0x14
  1730. -     * ntdll!RtlUserThreadStart+0x21
  1731. -     * ---- snip ----
  1732. -     */
  1733. -    __try {
  1734. -        status = SeImpersonateClientEx(entry->psec_ctx, NULL);
  1735. -    } __except(EXCEPTION_EXECUTE_HANDLER) {
  1736. -        NTSTATUS code;
  1737. -        code = GetExceptionCode();
  1738. -        print_error("handle_upcall: Call to SeImpersonateClientEx() "
  1739. -            "failed due to exception 0x%0x\n", (int)code);
  1740. -        status = STATUS_INTERNAL_ERROR;
  1741. -    }
  1742. -#else
  1743. -    status = SeImpersonateClientEx(entry->psec_ctx, NULL);
  1744. -#endif /* NFS41_DRIVER_STABILITY_HACKS */
  1745. -    if (status != STATUS_SUCCESS) {
  1746. -        print_error("handle_upcall: "
  1747. -            "SeImpersonateClientEx() failed 0x%x\n", status);
  1748. -        goto out;
  1749. -    }
  1750. -
  1751. -    switch(entry->opcode) {
  1752. -    case NFS41_SHUTDOWN:
  1753. -        status = marshal_nfs41_shutdown(entry, pbOut, cbOut, len);
  1754. -        KeSetEvent(&entry->cond, 0, FALSE);
  1755. -        break;
  1756. -    case NFS41_MOUNT:
  1757. -        status = marshal_nfs41_mount(entry, pbOut, cbOut, len);
  1758. -        break;
  1759. -    case NFS41_UNMOUNT:
  1760. -        status = marshal_nfs41_unmount(entry, pbOut, cbOut, len);
  1761. -        break;
  1762. -    case NFS41_OPEN:
  1763. -        status = marshal_nfs41_open(entry, pbOut, cbOut, len);
  1764. -        break;
  1765. -    case NFS41_READ:
  1766. -        status = marshal_nfs41_rw(entry, pbOut, cbOut, len);
  1767. -        break;
  1768. -    case NFS41_WRITE:
  1769. -        status = marshal_nfs41_rw(entry, pbOut, cbOut, len);
  1770. -        break;
  1771. -    case NFS41_LOCK:
  1772. -        status = marshal_nfs41_lock(entry, pbOut, cbOut, len);
  1773. -        break;
  1774. -    case NFS41_UNLOCK:
  1775. -        status = marshal_nfs41_unlock(entry, pbOut, cbOut, len);
  1776. -        break;
  1777. -    case NFS41_CLOSE:
  1778. -        status = marshal_nfs41_close(entry, pbOut, cbOut, len);
  1779. -        break;
  1780. -    case NFS41_DIR_QUERY:
  1781. -        status = marshal_nfs41_dirquery(entry, pbOut, cbOut, len);
  1782. -        break;
  1783. -    case NFS41_FILE_QUERY:
  1784. -    case NFS41_FILE_QUERY_TIME_BASED_COHERENCY:
  1785. -        status = marshal_nfs41_filequery(entry, pbOut, cbOut, len);
  1786. -        break;
  1787. -    case NFS41_FILE_SET:
  1788. -        status = marshal_nfs41_fileset(entry, pbOut, cbOut, len);
  1789. -        break;
  1790. -    case NFS41_EA_SET:
  1791. -        status = marshal_nfs41_easet(entry, pbOut, cbOut, len);
  1792. -        break;
  1793. -    case NFS41_EA_GET:
  1794. -        status = marshal_nfs41_eaget(entry, pbOut, cbOut, len);
  1795. -        break;
  1796. -    case NFS41_SYMLINK:
  1797. -        status = marshal_nfs41_symlink(entry, pbOut, cbOut, len);
  1798. -        break;
  1799. -    case NFS41_VOLUME_QUERY:
  1800. -        status = marshal_nfs41_volume(entry, pbOut, cbOut, len);
  1801. -        break;
  1802. -    case NFS41_ACL_QUERY:
  1803. -        status = marshal_nfs41_getacl(entry, pbOut, cbOut, len);
  1804. -        break;
  1805. -    case NFS41_ACL_SET:
  1806. -        status = marshal_nfs41_setacl(entry, pbOut, cbOut, len);
  1807. -        break;
  1808. -    default:
  1809. -        status = STATUS_INVALID_PARAMETER;
  1810. -        print_error("Unknown nfs41 ops %d\n", entry->opcode);
  1811. -    }
  1812. -
  1813. -    // if (status == STATUS_SUCCESS)
  1814. -    //     print_hexbuf("upcall buffer", pbOut, *len);
  1815. -
  1816. -out:
  1817. -    return status;
  1818. -}
  1819. -
  1820. -static NTSTATUS nfs41_UpcallCreate(
  1821. -    IN DWORD opcode,
  1822. -    IN PSECURITY_CLIENT_CONTEXT clnt_sec_ctx,
  1823. -    IN HANDLE session,
  1824. -    IN HANDLE open_state,
  1825. -    IN DWORD version,
  1826. -    IN PUNICODE_STRING filename,
  1827. -    OUT nfs41_updowncall_entry **entry_out)
  1828. -{
  1829. -    NTSTATUS status = STATUS_SUCCESS;
  1830. -    nfs41_updowncall_entry *entry;
  1831. -    SECURITY_SUBJECT_CONTEXT sec_ctx;
  1832. -    SECURITY_QUALITY_OF_SERVICE sec_qos;
  1833. -
  1834. -    entry = nfs41_upcall_allocate_updowncall_entry();
  1835. -    if (entry == NULL) {
  1836. -        status = STATUS_INSUFFICIENT_RESOURCES;
  1837. -        goto out;
  1838. -    }
  1839. -
  1840. -    RtlZeroMemory(entry, sizeof(nfs41_updowncall_entry));
  1841. -    entry->xid = InterlockedIncrement64(&xid);
  1842. -    entry->opcode = opcode;
  1843. -    entry->state = NFS41_WAITING_FOR_UPCALL;
  1844. -    entry->session = session;
  1845. -    entry->open_state = open_state;
  1846. -    entry->version = version;
  1847. -    if (filename && filename->Length) entry->filename = filename;
  1848. -    else if (filename && !filename->Length) entry->filename = (PUNICODE_STRING)&SLASH;
  1849. -    else entry->filename = (PUNICODE_STRING)&EMPTY_STRING;
  1850. -    /*XXX KeInitializeEvent will bugcheck under verifier if allocated
  1851. -     * from PagedPool? */
  1852. -    KeInitializeEvent(&entry->cond, SynchronizationEvent, FALSE);
  1853. -    ExInitializeFastMutex(&entry->lock);
  1854. -
  1855. -    if (clnt_sec_ctx == NULL) {
  1856. -        SeCaptureSubjectContext(&sec_ctx);
  1857. -        sec_qos.ContextTrackingMode = SECURITY_STATIC_TRACKING;
  1858. -        sec_qos.ImpersonationLevel = SecurityImpersonation;
  1859. -        sec_qos.Length = sizeof(SECURITY_QUALITY_OF_SERVICE);
  1860. -        sec_qos.EffectiveOnly = 0;
  1861. -        entry->psec_ctx = &entry->sec_ctx;
  1862. -        /*
  1863. -         * Arg |ServerIsRemote| must be |FALSE|, otherwise processes
  1864. -         * like Cygwin setup-x86_64.exe can fail during "Activation
  1865. -         * Context" creation in
  1866. -         * |SeCreateClientSecurityFromSubjectContext()| with
  1867. -         * |STATUS_BAD_IMPERSONATION_LEVEL|
  1868. -         */
  1869. -        status = SeCreateClientSecurityFromSubjectContext(&sec_ctx, &sec_qos,
  1870. -                    FALSE, entry->psec_ctx);
  1871. -        if (status != STATUS_SUCCESS) {
  1872. -            print_error("nfs41_UpcallCreate: "
  1873. -                "SeCreateClientSecurityFromSubjectContext() "
  1874. -                "failed with 0x%x\n",
  1875. -                status);
  1876. -            nfs41_upcall_free_updowncall_entry(entry);
  1877. -            entry = NULL;
  1878. -        }
  1879. -
  1880. -        SeReleaseSubjectContext(&sec_ctx);
  1881. -    } else {
  1882. -        entry->psec_ctx = clnt_sec_ctx;
  1883. -    }
  1884. -
  1885. -    if (entry && entry->psec_ctx) {
  1886. -        /*
  1887. -         * Refcount client token (as |entry->psec_ctx_clienttoken|)
  1888. -         * during lifetime of this |updowncall_entry| to avoid
  1889. -         * crashes during |SeImpersonateClientEx()| if the
  1890. -         * calling client thread exits.
  1891. -         */
  1892. -        entry->psec_ctx_clienttoken = entry->psec_ctx->ClientToken;
  1893. -        ObReferenceObject(entry->psec_ctx_clienttoken);
  1894. -    }
  1895. -
  1896. -    *entry_out = entry;
  1897. -out:
  1898. -    return status;
  1899. -}
  1900. -
  1901. -static void nfs41_UpcallDestroy(nfs41_updowncall_entry *entry)
  1902. -{
  1903. -    if (!entry)
  1904. -        return;
  1905. -
  1906. -    if (entry->psec_ctx_clienttoken) {
  1907. -        ObDereferenceObject(entry->psec_ctx_clienttoken);
  1908. -    }
  1909. -
  1910. -    nfs41_upcall_free_updowncall_entry(entry);
  1911. -}
  1912. -
  1913. -
  1914. -static NTSTATUS nfs41_UpcallWaitForReply(
  1915. -    IN nfs41_updowncall_entry *entry,
  1916. -    IN DWORD secs)
  1917. -{
  1918. -    NTSTATUS status = STATUS_SUCCESS;
  1919. -
  1920. -    nfs41_AddEntry(upcallLock, upcall, entry);
  1921. -    KeSetEvent(&upcallEvent, 0, FALSE);
  1922. -
  1923. -    if (entry->async_op)
  1924. -        goto out;
  1925. -
  1926. -    LARGE_INTEGER timeout;
  1927. -    timeout.QuadPart = RELATIVE(SECONDS(secs));
  1928. -retry_wait:
  1929. -    status = KeWaitForSingleObject(&entry->cond, Executive,
  1930. -                UserMode, FALSE, &timeout);
  1931. -
  1932. -    if (status == STATUS_TIMEOUT)
  1933. -            status = STATUS_NETWORK_UNREACHABLE;
  1934. -
  1935. -    print_wait_status(0, "[downcall]", status,
  1936. -        ENTRY_OPCODE2STRING(entry), entry,
  1937. -        (entry?entry->xid:-1LL));
  1938. -
  1939. -    switch(status) {
  1940. -    case STATUS_SUCCESS:
  1941. -        break;
  1942. -    case STATUS_USER_APC:
  1943. -    case STATUS_ALERTED:
  1944. -        DbgP("nfs41_UpcallWaitForReply: KeWaitForSingleObject() "
  1945. -            "returned status(=0x%lx), "
  1946. -            "retry waiting for '%s' entry=0x%p xid=%lld\n",
  1947. -            (long)status,
  1948. -            ENTRY_OPCODE2STRING(entry),
  1949. -            entry,
  1950. -            (entry?entry->xid:-1LL));
  1951. -        if (entry) {
  1952. -            goto retry_wait;
  1953. -        }
  1954. -        /* fall-through */
  1955. -    default:
  1956. -        ExAcquireFastMutex(&entry->lock);
  1957. -        if (entry->state == NFS41_DONE_PROCESSING) {
  1958. -            ExReleaseFastMutex(&entry->lock);
  1959. -            break;
  1960. -        }
  1961. -        DbgP("[upcall] abandoning '%s' entry=0x%p xid=%lld\n",
  1962. -            ENTRY_OPCODE2STRING(entry),
  1963. -            entry,
  1964. -            (entry?entry->xid:-1LL));
  1965. -        entry->state = NFS41_NOT_WAITING;
  1966. -        ExReleaseFastMutex(&entry->lock);
  1967. -        goto out;
  1968. -    }
  1969. -    nfs41_RemoveEntry(downcallLock, entry);
  1970. -out:
  1971. -    return status;
  1972. -}
  1973. -
  1974. -static NTSTATUS nfs41_upcall(
  1975. -    IN PRX_CONTEXT RxContext)
  1976. -{
  1977. -    NTSTATUS status = STATUS_SUCCESS;
  1978. -    ULONG len = 0;
  1979. -    PLIST_ENTRY pEntry = NULL;
  1980. -
  1981. -process_upcall:
  1982. -    nfs41_RemoveFirst(upcallLock, upcall, pEntry);
  1983. -    if (pEntry) {
  1984. -        nfs41_updowncall_entry *entry;
  1985. -
  1986. -        entry = (nfs41_updowncall_entry *)CONTAINING_RECORD(pEntry,
  1987. -                    nfs41_updowncall_entry, next);
  1988. -        ExAcquireFastMutex(&entry->lock);
  1989. -        nfs41_AddEntry(downcallLock, downcall, entry);
  1990. -        status = handle_upcall(RxContext, entry, &len);
  1991. -        if (status == STATUS_SUCCESS &&
  1992. -                entry->state == NFS41_WAITING_FOR_UPCALL)
  1993. -            entry->state = NFS41_WAITING_FOR_DOWNCALL;
  1994. -        ExReleaseFastMutex(&entry->lock);
  1995. -        if (status) {
  1996. -            entry->status = status;
  1997. -            KeSetEvent(&entry->cond, 0, FALSE);
  1998. -            RxContext->InformationToReturn = 0;
  1999. -        } else
  2000. -            RxContext->InformationToReturn = len;
  2001. -    }
  2002. -    else {
  2003. -/*
  2004. - * gisburn: |NFSV41_UPCALL_RETRY_WAIT| disabled for now because it
  2005. - * causes nfsd_debug.exe to hang on <CTRL-C>
  2006. - */
  2007. -#ifdef NFSV41_UPCALL_RETRY_WAIT
  2008. -retry_wait:
  2009. -#endif /* NFSV41_UPCALL_RETRY_WAIT */
  2010. -        status = KeWaitForSingleObject(&upcallEvent, Executive, UserMode, TRUE,
  2011. -            (PLARGE_INTEGER) NULL);
  2012. -        print_wait_status(0, "[upcall]", status, NULL, NULL, 0);
  2013. -        switch (status) {
  2014. -            case STATUS_SUCCESS:
  2015. -                goto process_upcall;
  2016. -            case STATUS_USER_APC:
  2017. -            case STATUS_ALERTED:
  2018. -                DbgP("nfs41_upcall: KeWaitForSingleObject() "
  2019. -                    "returned status(=0x%lx)"
  2020. -#ifdef NFSV41_UPCALL_RETRY_WAIT
  2021. -                    ", retry waiting"
  2022. -#endif /* NFSV41_UPCALL_RETRY_WAIT */
  2023. -                    "\n",
  2024. -                    (long)status);
  2025. -#ifdef NFSV41_UPCALL_RETRY_WAIT
  2026. -                goto retry_wait;
  2027. -#else
  2028. -                /* fall-through */
  2029. -#endif /* NFSV41_UPCALL_RETRY_WAIT */
  2030. -            default:
  2031. -                DbgP("nfs41_upcall: KeWaitForSingleObject() "
  2032. -                    "returned UNEXPECTED status(=0x%lx)\n",
  2033. -                    (long)status);
  2034. -                goto out;
  2035. -        }
  2036. -    }
  2037. -out:
  2038. -    return status;
  2039. -}
  2040. -
  2041. -static void unmarshal_nfs41_header(
  2042. -    nfs41_updowncall_entry *tmp,
  2043. -    unsigned char **buf)
  2044. -{
  2045. -    RtlZeroMemory(tmp, sizeof(nfs41_updowncall_entry));
  2046. -
  2047. -    RtlCopyMemory(&tmp->xid, *buf, sizeof(tmp->xid));
  2048. -    *buf += sizeof(tmp->xid);
  2049. -    RtlCopyMemory(&tmp->opcode, *buf, sizeof(tmp->opcode));
  2050. -    *buf += sizeof(tmp->opcode);
  2051. -    RtlCopyMemory(&tmp->status, *buf, sizeof(tmp->status));
  2052. -    *buf += sizeof(tmp->status);
  2053. -    RtlCopyMemory(&tmp->errno, *buf, sizeof(tmp->errno));
  2054. -    *buf += sizeof(tmp->errno);
  2055. -#ifdef DEBUG_MARSHAL_HEADER
  2056. -    DbgP("[downcall header] "
  2057. -        "xid=%lld opcode='%s' status=0x%lx errno=%d\n",
  2058. -        tmp->xid,
  2059. -        ENTRY_OPCODE2STRING(tmp), (long)tmp->status, tmp->errno);
  2060. -#endif
  2061. -}
  2062. -
  2063. -static void unmarshal_nfs41_mount(
  2064. -    nfs41_updowncall_entry *cur,
  2065. -    unsigned char **buf)
  2066. -{
  2067. -    RtlCopyMemory(&cur->session, *buf, sizeof(HANDLE));
  2068. -    *buf += sizeof(HANDLE);
  2069. -    RtlCopyMemory(&cur->version, *buf, sizeof(DWORD));
  2070. -    *buf += sizeof(DWORD);
  2071. -    RtlCopyMemory(&cur->u.Mount.lease_time, *buf, sizeof(DWORD));
  2072. -    *buf += sizeof(DWORD);
  2073. -    RtlCopyMemory(cur->u.Mount.FsAttrs, *buf, sizeof(FILE_FS_ATTRIBUTE_INFORMATION));
  2074. -#ifdef DEBUG_MARSHAL_DETAIL
  2075. -    DbgP("unmarshal_nfs41_mount: session pointer 0x%x version %d lease_time "
  2076. -         "%d\n", cur->session, cur->version, cur->u.Mount.lease_time);
  2077. -#endif
  2078. -}
  2079. -
  2080. -static void unmarshal_nfs41_setattr(
  2081. -    nfs41_updowncall_entry *cur,
  2082. -    PULONGLONG dest_buf,
  2083. -    unsigned char **buf)
  2084. -{
  2085. -    RtlCopyMemory(dest_buf, *buf, sizeof(ULONGLONG));
  2086. -#ifdef DEBUG_MARSHAL_DETAIL
  2087. -    DbgP("unmarshal_nfs41_setattr: returned ChangeTime %llu\n", *dest_buf);
  2088. -#endif
  2089. -}
  2090. -
  2091. -static NTSTATUS unmarshal_nfs41_rw(
  2092. -    nfs41_updowncall_entry *cur,
  2093. -    unsigned char **buf)
  2094. -{
  2095. -    NTSTATUS status = STATUS_SUCCESS;
  2096. -
  2097. -    RtlCopyMemory(&cur->buf_len, *buf, sizeof(cur->buf_len));
  2098. -    *buf += sizeof(cur->buf_len);
  2099. -    RtlCopyMemory(&cur->ChangeTime, *buf, sizeof(ULONGLONG));
  2100. -#ifdef DEBUG_MARSHAL_DETAIL_RW
  2101. -    DbgP("unmarshal_nfs41_rw: returned len %lu ChangeTime %llu\n",
  2102. -        cur->buf_len, cur->ChangeTime);
  2103. -#endif
  2104. -#if 1
  2105. -    /* 08/27/2010: it looks like we really don't need to call
  2106. -        * MmUnmapLockedPages() eventhough we called
  2107. -        * MmMapLockedPagesSpecifyCache() as the MDL passed to us
  2108. -        * is already locked.
  2109. -        */
  2110. -    __try {
  2111. -        MmUnmapLockedPages(cur->buf, cur->u.ReadWrite.MdlAddress);
  2112. -    } __except(EXCEPTION_EXECUTE_HANDLER) {
  2113. -        NTSTATUS code;
  2114. -        code = GetExceptionCode();
  2115. -        print_error("unmarshal_nfs41_rw: Call to MmUnmapLockedPages() "
  2116. -            "failed due to exception 0x%0x\n", (int)code);
  2117. -        status = STATUS_ACCESS_VIOLATION;
  2118. -    }
  2119. -#endif
  2120. -    return status;
  2121. -}
  2122. -
  2123. -static NTSTATUS unmarshal_nfs41_open(
  2124. -    nfs41_updowncall_entry *cur,
  2125. -    unsigned char **buf)
  2126. -{
  2127. -    NTSTATUS status = STATUS_SUCCESS;
  2128. -
  2129. -    __try {
  2130. -        if (cur->u.Open.EaBuffer)
  2131. -            MmUnmapLockedPages(cur->u.Open.EaBuffer, cur->u.Open.EaMdl);
  2132. -    } __except(EXCEPTION_EXECUTE_HANDLER) {
  2133. -        print_error("MmUnmapLockedPages thrown exception=0x%0x\n", GetExceptionCode());
  2134. -        status = cur->status = STATUS_ACCESS_VIOLATION;
  2135. -        goto out;
  2136. -    }
  2137. -
  2138. -    RtlCopyMemory(&cur->u.Open.binfo, *buf, sizeof(FILE_BASIC_INFORMATION));
  2139. -    *buf += sizeof(FILE_BASIC_INFORMATION);
  2140. -    RtlCopyMemory(&cur->u.Open.sinfo, *buf, sizeof(FILE_STANDARD_INFORMATION));
  2141. -    *buf += sizeof(FILE_STANDARD_INFORMATION);
  2142. -    RtlCopyMemory(&cur->open_state, *buf, sizeof(HANDLE));
  2143. -    *buf += sizeof(HANDLE);
  2144. -    RtlCopyMemory(&cur->u.Open.mode, *buf, sizeof(DWORD));
  2145. -    *buf += sizeof(DWORD);
  2146. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  2147. -    RtlCopyMemory(&cur->u.Open.owner_local_uid, *buf, sizeof(DWORD));
  2148. -    *buf += sizeof(DWORD);
  2149. -    RtlCopyMemory(&cur->u.Open.owner_group_local_gid, *buf, sizeof(DWORD));
  2150. -    *buf += sizeof(DWORD);
  2151. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  2152. -    RtlCopyMemory(&cur->ChangeTime, *buf, sizeof(ULONGLONG));
  2153. -    *buf += sizeof(ULONGLONG);
  2154. -    RtlCopyMemory(&cur->u.Open.deleg_type, *buf, sizeof(DWORD));
  2155. -    *buf += sizeof(DWORD);
  2156. -    if (cur->errno == ERROR_REPARSE) {
  2157. -        RtlCopyMemory(&cur->u.Open.symlink_embedded, *buf, sizeof(BOOLEAN));
  2158. -        *buf += sizeof(BOOLEAN);
  2159. -        RtlCopyMemory(&cur->u.Open.symlink.MaximumLength, *buf,
  2160. -            sizeof(USHORT));
  2161. -        *buf += sizeof(USHORT);
  2162. -        cur->u.Open.symlink.Length = cur->u.Open.symlink.MaximumLength -
  2163. -            sizeof(WCHAR);
  2164. -        cur->u.Open.symlink.Buffer = RxAllocatePoolWithTag(NonPagedPoolNx,
  2165. -            cur->u.Open.symlink.MaximumLength, NFS41_MM_POOLTAG);
  2166. -        if (cur->u.Open.symlink.Buffer == NULL) {
  2167. -            cur->status = STATUS_INSUFFICIENT_RESOURCES;
  2168. -            status = STATUS_UNSUCCESSFUL;
  2169. -            goto out;
  2170. -        }
  2171. -        RtlCopyMemory(cur->u.Open.symlink.Buffer, *buf,
  2172. -            cur->u.Open.symlink.MaximumLength);
  2173. -#ifdef DEBUG_MARSHAL_DETAIL
  2174. -        DbgP("unmarshal_nfs41_open: ERROR_REPARSE -> '%wZ'\n", &cur->u.Open.symlink);
  2175. -#endif
  2176. -    }
  2177. -#ifdef DEBUG_MARSHAL_DETAIL
  2178. -    DbgP("unmarshal_nfs41_open: open_state 0x%x mode 0%o "
  2179. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  2180. -        "owner_local_uid %u owner_group_local_gid %u "
  2181. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  2182. -        "changeattr %llu "
  2183. -        "deleg_type %d\n", cur->open_state, cur->u.Open.mode,
  2184. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  2185. -        cur->u.Open.owner_local_uid, cur->u.Open.owner_group_local_gid,
  2186. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  2187. -        cur->ChangeTime, cur->u.Open.deleg_type);
  2188. -#endif /* DEBUG_MARSHAL_DETAIL */
  2189. -out:
  2190. -    return status;
  2191. -}
  2192. -
  2193. -static NTSTATUS unmarshal_nfs41_dirquery(
  2194. -    nfs41_updowncall_entry *cur,
  2195. -    unsigned char **buf)
  2196. -{
  2197. -    NTSTATUS status = STATUS_SUCCESS;
  2198. -    ULONG buf_len;
  2199. -    
  2200. -    RtlCopyMemory(&buf_len, *buf, sizeof(ULONG));
  2201. -#ifdef DEBUG_MARSHAL_DETAIL
  2202. -    DbgP("unmarshal_nfs41_dirquery: reply size %d\n", buf_len);
  2203. -#endif
  2204. -    *buf += sizeof(ULONG);
  2205. -    __try {
  2206. -        MmUnmapLockedPages(cur->u.QueryFile.mdl_buf, cur->u.QueryFile.mdl);
  2207. -    } __except(EXCEPTION_EXECUTE_HANDLER) {
  2208. -        NTSTATUS code;
  2209. -        code = GetExceptionCode();
  2210. -        print_error("MmUnmapLockedPages thrown exception=0x%0x\n", code);
  2211. -        status = STATUS_ACCESS_VIOLATION;
  2212. -    }
  2213. -    if (buf_len > cur->buf_len)
  2214. -        cur->status = STATUS_BUFFER_TOO_SMALL;
  2215. -    cur->buf_len = buf_len;
  2216. -
  2217. -    return status;
  2218. -}
  2219. -
  2220. -static void unmarshal_nfs41_attrget(
  2221. -    nfs41_updowncall_entry *cur,
  2222. -    PVOID attr_value,
  2223. -    ULONG *attr_len,
  2224. -    unsigned char **buf)
  2225. -{
  2226. -    ULONG buf_len;
  2227. -    RtlCopyMemory(&buf_len, *buf, sizeof(ULONG));
  2228. -    if (buf_len > *attr_len) {
  2229. -        cur->status = STATUS_BUFFER_TOO_SMALL;        
  2230. -        return;
  2231. -    }
  2232. -    *buf += sizeof(ULONG);
  2233. -    *attr_len = buf_len;
  2234. -    RtlCopyMemory(attr_value, *buf, buf_len);
  2235. -    *buf += buf_len;
  2236. -}
  2237. -
  2238. -static void unmarshal_nfs41_eaget(
  2239. -    nfs41_updowncall_entry *cur,
  2240. -    unsigned char **buf)
  2241. -{
  2242. -    RtlCopyMemory(&cur->u.QueryEa.Overflow, *buf, sizeof(ULONG));
  2243. -    *buf += sizeof(ULONG);
  2244. -    RtlCopyMemory(&cur->buf_len, *buf, sizeof(ULONG));
  2245. -    *buf += sizeof(ULONG);
  2246. -    if (cur->u.QueryEa.Overflow != ERROR_INSUFFICIENT_BUFFER) {
  2247. -        RtlCopyMemory(cur->buf, *buf, cur->buf_len);
  2248. -        *buf += cur->buf_len;
  2249. -    }
  2250. -}
  2251. -
  2252. -static void unmarshal_nfs41_getattr(
  2253. -    nfs41_updowncall_entry *cur,
  2254. -    unsigned char **buf)
  2255. -{
  2256. -    unmarshal_nfs41_attrget(cur, cur->buf, &cur->buf_len, buf);
  2257. -    RtlCopyMemory(&cur->ChangeTime, *buf, sizeof(LONGLONG));
  2258. -#ifdef DEBUG_MARSHAL_DETAIL
  2259. -    if (cur->u.QueryFile.InfoClass == FileBasicInformation)
  2260. -        DbgP("[unmarshal_nfs41_getattr] ChangeTime %llu\n", cur->ChangeTime);
  2261. -#endif
  2262. -}
  2263. -
  2264. -static NTSTATUS unmarshal_nfs41_getacl(
  2265. -    nfs41_updowncall_entry *cur,
  2266. -    unsigned char **buf)
  2267. -{
  2268. -    NTSTATUS status = STATUS_SUCCESS;
  2269. -    DWORD buf_len;
  2270. -
  2271. -    RtlCopyMemory(&buf_len, *buf, sizeof(DWORD));
  2272. -    *buf += sizeof(DWORD);
  2273. -    cur->buf = RxAllocatePoolWithTag(NonPagedPoolNx,
  2274. -        buf_len, NFS41_MM_POOLTAG_ACL);
  2275. -    if (cur->buf == NULL) {
  2276. -        cur->status = status = STATUS_INSUFFICIENT_RESOURCES;
  2277. -        goto out;
  2278. -    }
  2279. -    RtlCopyMemory(cur->buf, *buf, buf_len);
  2280. -    if (buf_len > cur->buf_len)
  2281. -        cur->status = STATUS_BUFFER_TOO_SMALL;
  2282. -    cur->buf_len = buf_len;
  2283. -
  2284. -out:
  2285. -    return status;
  2286. -}
  2287. -
  2288. -static void unmarshal_nfs41_symlink(
  2289. -    nfs41_updowncall_entry *cur,
  2290. -    unsigned char **buf)
  2291. -{
  2292. -    if (cur->u.Symlink.set) return;
  2293. -
  2294. -    RtlCopyMemory(&cur->u.Symlink.target->Length, *buf, sizeof(USHORT));
  2295. -    *buf += sizeof(USHORT);
  2296. -    if (cur->u.Symlink.target->Length >
  2297. -            cur->u.Symlink.target->MaximumLength) {
  2298. -        cur->status = STATUS_BUFFER_TOO_SMALL;
  2299. -        return;
  2300. -    }
  2301. -    RtlCopyMemory(cur->u.Symlink.target->Buffer, *buf,
  2302. -        cur->u.Symlink.target->Length);
  2303. -    cur->u.Symlink.target->Length -= sizeof(UNICODE_NULL);
  2304. -}
  2305. -
  2306. -static NTSTATUS nfs41_downcall(
  2307. -    IN PRX_CONTEXT RxContext)
  2308. -{
  2309. -    NTSTATUS status = STATUS_SUCCESS;
  2310. -    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  2311. -#ifdef DEBUG_PRINT_DOWNCALL_HEXBUF
  2312. -    ULONG in_len = LowIoContext->ParamsFor.IoCtl.InputBufferLength;
  2313. -#endif /* DEBUG_PRINT_DOWNCALL_HEXBUF */
  2314. -    unsigned char *buf = LowIoContext->ParamsFor.IoCtl.pInputBuffer;
  2315. -    PLIST_ENTRY pEntry;
  2316. -    nfs41_updowncall_entry *tmp, *cur= NULL;
  2317. -    BOOLEAN found = 0;
  2318. -
  2319. -#ifdef DEBUG_PRINT_DOWNCALL_HEXBUF
  2320. -    print_hexbuf("downcall buffer", buf, in_len);
  2321. -#endif /* DEBUG_PRINT_DOWNCALL_HEXBUF */
  2322. -
  2323. -    tmp = nfs41_downcall_allocate_updowncall_entry();
  2324. -    if (tmp == NULL) goto out;
  2325. -
  2326. -    unmarshal_nfs41_header(tmp, &buf);
  2327. -
  2328. -    ExAcquireFastMutex(&downcallLock);
  2329. -    pEntry = &downcall.head;
  2330. -    pEntry = pEntry->Flink;
  2331. -    while (pEntry != NULL) {
  2332. -        cur = (nfs41_updowncall_entry *)CONTAINING_RECORD(pEntry,
  2333. -                nfs41_updowncall_entry, next);
  2334. -        if (cur->xid == tmp->xid) {
  2335. -            found = 1;
  2336. -            break;
  2337. -        }
  2338. -        if (pEntry->Flink == &downcall.head)
  2339. -            break;
  2340. -        pEntry = pEntry->Flink;
  2341. -    }
  2342. -    ExReleaseFastMutex(&downcallLock);
  2343. -    SeStopImpersonatingClient();
  2344. -    if (!found) {
  2345. -        print_error("Didn't find xid=%lld entry\n", tmp->xid);
  2346. -        goto out_free;
  2347. -    }
  2348. -
  2349. -    ExAcquireFastMutex(&cur->lock);    
  2350. -    if (cur->state == NFS41_NOT_WAITING) {
  2351. -        DbgP("[downcall] Nobody is waiting for this request!!!\n");
  2352. -        switch(cur->opcode) {
  2353. -        case NFS41_WRITE:
  2354. -        case NFS41_READ:
  2355. -            MmUnmapLockedPages(cur->buf, cur->u.ReadWrite.MdlAddress);
  2356. -            break;
  2357. -        case NFS41_DIR_QUERY:
  2358. -            MmUnmapLockedPages(cur->u.QueryFile.mdl_buf,
  2359. -                    cur->u.QueryFile.mdl);
  2360. -            IoFreeMdl(cur->u.QueryFile.mdl);
  2361. -            break;
  2362. -        case NFS41_OPEN:
  2363. -            if (cur->u.Open.EaMdl) {
  2364. -                MmUnmapLockedPages(cur->u.Open.EaBuffer,
  2365. -                        cur->u.Open.EaMdl);
  2366. -                IoFreeMdl(cur->u.Open.EaMdl);
  2367. -            }
  2368. -            break;
  2369. -        }
  2370. -        ExReleaseFastMutex(&cur->lock);
  2371. -        nfs41_RemoveEntry(downcallLock, cur);
  2372. -        nfs41_UpcallDestroy(cur);
  2373. -        status = STATUS_UNSUCCESSFUL;
  2374. -        goto out_free;
  2375. -    }
  2376. -    cur->state = NFS41_DONE_PROCESSING;
  2377. -    cur->status = tmp->status;
  2378. -    cur->errno = tmp->errno;
  2379. -    status = STATUS_SUCCESS;
  2380. -
  2381. -    if (!tmp->status) {
  2382. -        switch (tmp->opcode) {
  2383. -        case NFS41_MOUNT:
  2384. -            unmarshal_nfs41_mount(cur, &buf);
  2385. -            break;
  2386. -        case NFS41_WRITE:
  2387. -        case NFS41_READ:
  2388. -            status = unmarshal_nfs41_rw(cur, &buf);
  2389. -            break;
  2390. -        case NFS41_OPEN:
  2391. -            status = unmarshal_nfs41_open(cur, &buf);
  2392. -            break;
  2393. -        case NFS41_DIR_QUERY:
  2394. -            status = unmarshal_nfs41_dirquery(cur, &buf);
  2395. -            break;
  2396. -        case NFS41_FILE_QUERY:
  2397. -        case NFS41_FILE_QUERY_TIME_BASED_COHERENCY:
  2398. -            unmarshal_nfs41_getattr(cur, &buf);
  2399. -            break;
  2400. -        case NFS41_EA_GET:
  2401. -            unmarshal_nfs41_eaget(cur, &buf);
  2402. -            break;
  2403. -        case NFS41_SYMLINK:
  2404. -            unmarshal_nfs41_symlink(cur, &buf);
  2405. -            break;
  2406. -        case NFS41_VOLUME_QUERY:
  2407. -            unmarshal_nfs41_attrget(cur, cur->buf, &cur->buf_len, &buf);
  2408. -            break;
  2409. -        case NFS41_ACL_QUERY:
  2410. -            status = unmarshal_nfs41_getacl(cur, &buf);
  2411. -            break;
  2412. -        case NFS41_FILE_SET:
  2413. -            unmarshal_nfs41_setattr(cur, &cur->ChangeTime, &buf);
  2414. -            break;
  2415. -        case NFS41_EA_SET:
  2416. -            unmarshal_nfs41_setattr(cur, &cur->ChangeTime, &buf);
  2417. -            break;
  2418. -        case NFS41_ACL_SET:
  2419. -            unmarshal_nfs41_setattr(cur, &cur->ChangeTime, &buf);
  2420. -            break;
  2421. -        }
  2422. -    }
  2423. -    ExReleaseFastMutex(&cur->lock);
  2424. -    if (cur->async_op) {
  2425. -        switch (cur->opcode) {
  2426. -            case NFS41_WRITE:
  2427. -            case NFS41_READ:
  2428. -                if (cur->status == STATUS_SUCCESS) {
  2429. -                    cur->u.ReadWrite.rxcontext->StoredStatus =
  2430. -                        STATUS_SUCCESS;
  2431. -                    cur->u.ReadWrite.rxcontext->InformationToReturn =
  2432. -                        cur->buf_len;
  2433. -                } else {
  2434. -                    cur->u.ReadWrite.rxcontext->StoredStatus =
  2435. -                        map_readwrite_errors(cur->status);
  2436. -                    cur->u.ReadWrite.rxcontext->InformationToReturn = 0;
  2437. -                }
  2438. -                nfs41_RemoveEntry(downcallLock, cur);
  2439. -                RxLowIoCompletion(cur->u.ReadWrite.rxcontext);
  2440. -                nfs41_UpcallDestroy(cur);
  2441. -                break;
  2442. -            default:
  2443. -                print_error("##### nfs41_downcall: "
  2444. -                    "unknown async opcode=%d ####\n",
  2445. -                    (int)cur->opcode);
  2446. -                break;
  2447. -        }
  2448. -    } else
  2449. -        KeSetEvent(&cur->cond, 0, FALSE);
  2450. -
  2451. -out_free:
  2452. -    nfs41_downcall_free_updowncall_entry(tmp);
  2453. -out:
  2454. -    return status;
  2455. -}
  2456. -
  2457. -static NTSTATUS nfs41_shutdown_daemon(
  2458. -    DWORD version)
  2459. -{
  2460. -    NTSTATUS status = STATUS_SUCCESS;
  2461. -    nfs41_updowncall_entry *entry = NULL;
  2462. -
  2463. -    DbgEn();
  2464. -    status = nfs41_UpcallCreate(NFS41_SHUTDOWN, NULL, INVALID_HANDLE_VALUE,
  2465. -        INVALID_HANDLE_VALUE, version, NULL, &entry);
  2466. -    if (status) goto out;
  2467. -
  2468. -    status = nfs41_UpcallWaitForReply(entry, UPCALL_TIMEOUT_DEFAULT);
  2469. -    if (entry->psec_ctx == &entry->sec_ctx) {
  2470. -        SeDeleteClientSecurity(entry->psec_ctx);
  2471. -    }
  2472. -    entry->psec_ctx = NULL;
  2473. -    if (status) goto out;
  2474. -
  2475. -    nfs41_UpcallDestroy(entry);
  2476. -out:
  2477. -    DbgEx();
  2478. -    return status;
  2479. -}
  2480. -
  2481. -static NTSTATUS SharedMemoryInit(
  2482. -    OUT PHANDLE phSection)
  2483. -{
  2484. -    NTSTATUS status;
  2485. -    HANDLE hSection;
  2486. -    UNICODE_STRING SectionName;
  2487. -    SECURITY_DESCRIPTOR SecurityDesc;
  2488. -    OBJECT_ATTRIBUTES SectionAttrs;
  2489. -    LARGE_INTEGER nSectionSize;
  2490. -
  2491. -    DbgEn();
  2492. -
  2493. -    RtlInitUnicodeString(&SectionName, NFS41_SHARED_MEMORY_NAME);
  2494. -
  2495. -    /* XXX: setting dacl=NULL grants access to everyone */
  2496. -    status = RtlCreateSecurityDescriptor(&SecurityDesc,
  2497. -        SECURITY_DESCRIPTOR_REVISION);
  2498. -    if (status) {
  2499. -        print_error("RtlCreateSecurityDescriptor() failed with %08X\n", status);
  2500. -        goto out;
  2501. -    }
  2502. -    status = RtlSetDaclSecurityDescriptor(&SecurityDesc, TRUE, NULL, FALSE);
  2503. -    if (status) {
  2504. -        print_error("RtlSetDaclSecurityDescriptor() failed with %08X\n", status);
  2505. -        goto out;
  2506. -    }
  2507. -
  2508. -    InitializeObjectAttributes(&SectionAttrs, &SectionName,
  2509. -        0, NULL, &SecurityDesc);
  2510. -
  2511. -    nSectionSize.QuadPart = sizeof(NFS41NP_SHARED_MEMORY);
  2512. -
  2513. -    status = ZwCreateSection(&hSection, SECTION_MAP_READ | SECTION_MAP_WRITE,
  2514. -        &SectionAttrs, &nSectionSize, PAGE_READWRITE, SEC_COMMIT, NULL);
  2515. -    switch (status) {
  2516. -    case STATUS_SUCCESS:
  2517. -        break;
  2518. -    case STATUS_OBJECT_NAME_COLLISION:
  2519. -        DbgP("section already created; returning success\n");
  2520. -        status = STATUS_SUCCESS;
  2521. -        goto out;
  2522. -    default:
  2523. -        DbgP("ZwCreateSection failed with %08X\n", status);
  2524. -        goto out;
  2525. -    }
  2526. -out:
  2527. -    DbgEx();
  2528. -    return status;
  2529. -}
  2530. -
  2531. -static NTSTATUS SharedMemoryFree(
  2532. -    IN HANDLE hSection)
  2533. -{
  2534. -    NTSTATUS status;
  2535. -    DbgEn();
  2536. -    status = ZwClose(hSection);
  2537. -    DbgEx();
  2538. -    return status;
  2539. -}
  2540. -
  2541. -static NTSTATUS nfs41_Start(
  2542. -    IN OUT PRX_CONTEXT RxContext,
  2543. -    IN OUT PRDBSS_DEVICE_OBJECT dev)
  2544. -{
  2545. -    NTSTATUS status;
  2546. -    NFS41GetDeviceExtension(RxContext, DevExt);
  2547. -
  2548. -    DbgEn();
  2549. -
  2550. -    status = SharedMemoryInit(&DevExt->SharedMemorySection);
  2551. -    if (status) {
  2552. -        print_error("InitSharedMemory failed with %08X\n", status);
  2553. -        status = STATUS_INSUFFICIENT_RESOURCES;
  2554. -        goto out;
  2555. -    }
  2556. -
  2557. -    InterlockedCompareExchange((PLONG)&nfs41_start_state,
  2558. -        NFS41_START_DRIVER_STARTED,
  2559. -        NFS41_START_DRIVER_START_IN_PROGRESS);
  2560. -out:
  2561. -    DbgEx();
  2562. -    return status;
  2563. -}
  2564. -
  2565. -static NTSTATUS nfs41_Stop(
  2566. -    IN OUT PRX_CONTEXT RxContext,
  2567. -    IN OUT PRDBSS_DEVICE_OBJECT dev)
  2568. -{
  2569. -    NTSTATUS status;
  2570. -    NFS41GetDeviceExtension(RxContext, DevExt);
  2571. -    DbgEn();
  2572. -    status = SharedMemoryFree(DevExt->SharedMemorySection);
  2573. -    DbgEx();
  2574. -    return status;
  2575. -}
  2576. -
  2577. -static NTSTATUS GetConnectionHandle(
  2578. -    IN PUNICODE_STRING ConnectionName,
  2579. -    IN PVOID EaBuffer,
  2580. -    IN ULONG EaLength,
  2581. -    OUT PHANDLE Handle)
  2582. -{
  2583. -    NTSTATUS status;
  2584. -    IO_STATUS_BLOCK IoStatusBlock;
  2585. -    OBJECT_ATTRIBUTES ObjectAttributes;
  2586. -
  2587. -#ifdef DEBUG_MOUNT
  2588. -    DbgEn();
  2589. -#endif
  2590. -    InitializeObjectAttributes(&ObjectAttributes, ConnectionName,
  2591. -        OBJ_CASE_INSENSITIVE|OBJ_KERNEL_HANDLE, NULL, NULL);
  2592. -
  2593. -    status = ZwCreateFile(Handle, SYNCHRONIZE, &ObjectAttributes,
  2594. -        &IoStatusBlock, NULL, FILE_ATTRIBUTE_NORMAL,
  2595. -        FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
  2596. -        FILE_OPEN_IF,
  2597. -        FILE_CREATE_TREE_CONNECTION | FILE_SYNCHRONOUS_IO_NONALERT,
  2598. -        EaBuffer, EaLength);
  2599. -
  2600. -#ifdef DEBUG_MOUNT
  2601. -    DbgEx();
  2602. -#endif
  2603. -    return status;
  2604. -}
  2605. -
  2606. -static NTSTATUS nfs41_GetConnectionInfoFromBuffer(
  2607. -    IN PVOID Buffer,
  2608. -    IN ULONG BufferLen,
  2609. -    OUT PUNICODE_STRING pConnectionName,
  2610. -    OUT PVOID *ppEaBuffer,
  2611. -    OUT PULONG pEaLength)
  2612. -{
  2613. -    NTSTATUS status = STATUS_SUCCESS;
  2614. -    USHORT NameLength, EaPadding;
  2615. -    ULONG EaLength, BufferLenExpected;
  2616. -    PBYTE ptr;
  2617. -
  2618. -    /* make sure buffer is at least big enough for header */
  2619. -    if (BufferLen < sizeof(USHORT) + sizeof(USHORT) + sizeof(ULONG)) {
  2620. -        status = STATUS_BAD_NETWORK_NAME;
  2621. -        print_error("Invalid input buffer.\n");
  2622. -        pConnectionName->Length = pConnectionName->MaximumLength = 0;
  2623. -        *ppEaBuffer = NULL;
  2624. -        *pEaLength = 0;
  2625. -        goto out;
  2626. -    }
  2627. -
  2628. -    ptr = Buffer;
  2629. -    NameLength = *(PUSHORT)ptr;
  2630. -    ptr += sizeof(USHORT);
  2631. -    EaPadding = *(PUSHORT)ptr;
  2632. -    ptr += sizeof(USHORT);
  2633. -    EaLength = *(PULONG)ptr;
  2634. -    ptr += sizeof(ULONG);
  2635. -
  2636. -    /* validate buffer length */
  2637. -    BufferLenExpected = sizeof(USHORT) + sizeof(USHORT) + sizeof(ULONG) +
  2638. -        NameLength + EaPadding + EaLength;
  2639. -    if (BufferLen != BufferLenExpected) {
  2640. -        status = STATUS_BAD_NETWORK_NAME;
  2641. -        print_error("Received buffer of length %lu, but expected %lu bytes.\n",
  2642. -            BufferLen, BufferLenExpected);
  2643. -        pConnectionName->Length = pConnectionName->MaximumLength = 0;
  2644. -        *ppEaBuffer = NULL;
  2645. -        *pEaLength = 0;
  2646. -        goto out;
  2647. -    }
  2648. -
  2649. -    pConnectionName->Buffer = (PWCH)ptr;
  2650. -    pConnectionName->Length = NameLength - sizeof(WCHAR);
  2651. -    pConnectionName->MaximumLength = NameLength;
  2652. -
  2653. -    if (EaLength)
  2654. -        *ppEaBuffer = ptr + NameLength + EaPadding;
  2655. -    else
  2656. -        *ppEaBuffer = NULL;
  2657. -    *pEaLength = EaLength;
  2658. -
  2659. -out:
  2660. -    return status;
  2661. -}
  2662. -
  2663. -static NTSTATUS nfs41_CreateConnection(
  2664. -    IN PRX_CONTEXT RxContext,
  2665. -    OUT PBOOLEAN PostToFsp)
  2666. -{
  2667. -    NTSTATUS status = STATUS_SUCCESS;
  2668. -    HANDLE Handle = INVALID_HANDLE_VALUE;
  2669. -    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  2670. -    PVOID Buffer = LowIoContext->ParamsFor.IoCtl.pInputBuffer, EaBuffer;
  2671. -    ULONG BufferLen = LowIoContext->ParamsFor.IoCtl.InputBufferLength, EaLength;
  2672. -    UNICODE_STRING FileName;
  2673. -    BOOLEAN Wait = BooleanFlagOn(RxContext->Flags, RX_CONTEXT_FLAG_WAIT);
  2674. -
  2675. -#ifdef DEBUG_MOUNT
  2676. -    DbgEn();
  2677. -#endif
  2678. -
  2679. -    if (!Wait) {
  2680. -        //just post right now!
  2681. -        DbgP("returning STATUS_PENDING\n");
  2682. -        *PostToFsp = TRUE;
  2683. -        status = STATUS_PENDING;
  2684. -        goto out;
  2685. -    }
  2686. -
  2687. -    status = nfs41_GetConnectionInfoFromBuffer(Buffer, BufferLen,
  2688. -        &FileName, &EaBuffer, &EaLength);
  2689. -    if (status != STATUS_SUCCESS)
  2690. -        goto out;
  2691. -
  2692. -    status = GetConnectionHandle(&FileName, EaBuffer, EaLength, &Handle);
  2693. -    if (!status && Handle != INVALID_HANDLE_VALUE)
  2694. -        ZwClose(Handle);
  2695. -out:
  2696. -#ifdef DEBUG_MOUNT
  2697. -    DbgEx();
  2698. -#endif
  2699. -    return status;
  2700. -}
  2701. -
  2702. -#ifdef ENABLE_TIMINGS
  2703. -static void print_op_stat(
  2704. -    const char *op_str,
  2705. -    nfs41_timings *time, BOOLEAN clear)
  2706. -{
  2707. -    DbgP("%-9s: num_ops=%-10d delta_ticks=%-10d size=%-10d\n", op_str,
  2708. -        time->tops, time->tops ? time->ticks/time->tops : 0,
  2709. -        time->sops ? time->size/time->sops : 0);
  2710. -    if (clear) {
  2711. -        time->tops = 0;
  2712. -        time->ticks = 0;
  2713. -        time->size = 0;
  2714. -        time->sops = 0;
  2715. -    }
  2716. -}
  2717. -#endif
  2718. -static NTSTATUS nfs41_unmount(
  2719. -    HANDLE session,
  2720. -    DWORD version,
  2721. -    DWORD timeout)
  2722. -{
  2723. -    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  2724. -    nfs41_updowncall_entry *entry;
  2725. -
  2726. -#ifdef DEBUG_MOUNT
  2727. -    DbgEn();
  2728. -#endif
  2729. -    status = nfs41_UpcallCreate(NFS41_UNMOUNT, NULL, session,
  2730. -        INVALID_HANDLE_VALUE, version, NULL, &entry);
  2731. -    if (status) goto out;
  2732. -
  2733. -    nfs41_UpcallWaitForReply(entry, timeout);
  2734. -
  2735. -    if (entry->psec_ctx == &entry->sec_ctx) {
  2736. -        SeDeleteClientSecurity(entry->psec_ctx);
  2737. -    }
  2738. -    entry->psec_ctx = NULL;
  2739. -    nfs41_UpcallDestroy(entry);
  2740. -out:
  2741. -#ifdef ENABLE_TIMINGS
  2742. -    print_op_stat("lookup", &lookup, 1);
  2743. -    print_op_stat("open", &open, 1);
  2744. -    print_op_stat("close", &close, 1);
  2745. -    print_op_stat("volume", &volume, 1);
  2746. -    print_op_stat("getattr", &getattr, 1);
  2747. -    print_op_stat("setattr", &setattr, 1);
  2748. -    print_op_stat("getexattr", &getexattr, 1);
  2749. -    print_op_stat("setexattr", &setexattr, 1);
  2750. -    print_op_stat("readdir", &readdir, 1);
  2751. -    print_op_stat("getacl", &getacl, 1);
  2752. -    print_op_stat("setacl", &setacl, 1);
  2753. -    print_op_stat("read", &read, 1);
  2754. -    print_op_stat("write", &write, 1);
  2755. -    print_op_stat("lock", &lock, 1);
  2756. -    print_op_stat("unlock", &unlock, 1);
  2757. -#endif
  2758. -#ifdef DEBUG_MOUNT
  2759. -    DbgEx();
  2760. -#endif
  2761. -    return status;
  2762. -}
  2763. -
  2764. -static NTSTATUS nfs41_DeleteConnection (
  2765. -    IN PRX_CONTEXT RxContext,
  2766. -    OUT PBOOLEAN PostToFsp)
  2767. -{
  2768. -    NTSTATUS status = STATUS_INVALID_PARAMETER;
  2769. -    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  2770. -    PWCHAR ConnectName = LowIoContext->ParamsFor.IoCtl.pInputBuffer;
  2771. -    ULONG ConnectNameLen = LowIoContext->ParamsFor.IoCtl.InputBufferLength;
  2772. -    HANDLE Handle;
  2773. -    UNICODE_STRING FileName;
  2774. -    PFILE_OBJECT pFileObject;
  2775. -    BOOLEAN Wait = BooleanFlagOn(RxContext->Flags, RX_CONTEXT_FLAG_WAIT);
  2776. -
  2777. -#ifdef DEBUG_MOUNT
  2778. -    DbgEn();
  2779. -#endif
  2780. -
  2781. -    if (!Wait) {
  2782. -        //just post right now!
  2783. -        *PostToFsp = TRUE;
  2784. -        DbgP("returning STATUS_PENDING\n");
  2785. -        status = STATUS_PENDING;
  2786. -        goto out;
  2787. -    }
  2788. -
  2789. -    FileName.Buffer = ConnectName;
  2790. -    FileName.Length = (USHORT) ConnectNameLen - sizeof(WCHAR);
  2791. -    FileName.MaximumLength = (USHORT) ConnectNameLen;
  2792. -
  2793. -    status = GetConnectionHandle(&FileName, NULL, 0, &Handle);
  2794. -    if (status != STATUS_SUCCESS)
  2795. -        goto out;
  2796. -
  2797. -    status = ObReferenceObjectByHandle(Handle, 0L, NULL, KernelMode,
  2798. -                (PVOID *)&pFileObject, NULL);
  2799. -    if (NT_SUCCESS(status)) {
  2800. -        PV_NET_ROOT VNetRoot;
  2801. -
  2802. -        // VNetRoot exists as FOBx in the FsContext2
  2803. -        VNetRoot = (PV_NET_ROOT) pFileObject->FsContext2;
  2804. -        // make sure the node looks right
  2805. -        if (NodeType(VNetRoot) == RDBSS_NTC_V_NETROOT)
  2806. -        {
  2807. -#ifdef DEBUG_MOUNT
  2808. -            DbgP("Calling RxFinalizeConnection for NetRoot 0x%p from VNetRoot 0x%p\n",
  2809. -                VNetRoot->NetRoot, VNetRoot);
  2810. -#endif
  2811. -            status = RxFinalizeConnection(VNetRoot->NetRoot, VNetRoot, TRUE);
  2812. -        }
  2813. -        else
  2814. -            status = STATUS_BAD_NETWORK_NAME;
  2815. -
  2816. -        ObDereferenceObject(pFileObject);
  2817. -    }
  2818. -    ZwClose(Handle);
  2819. -out:
  2820. -#ifdef DEBUG_MOUNT
  2821. -    DbgEx();
  2822. -#endif
  2823. -    return status;
  2824. -}
  2825. -
  2826. -static NTSTATUS nfs41_DevFcbXXXControlFile(
  2827. -    IN OUT PRX_CONTEXT RxContext)
  2828. -{
  2829. -    NTSTATUS status = STATUS_INVALID_DEVICE_REQUEST;
  2830. -    UCHAR op = RxContext->MajorFunction;
  2831. -    PLOWIO_CONTEXT io_ctx = &RxContext->LowIoContext;
  2832. -    ULONG fsop = io_ctx->ParamsFor.FsCtl.FsControlCode, state;
  2833. -    ULONG in_len = io_ctx->ParamsFor.IoCtl.InputBufferLength;
  2834. -    DWORD *buf = io_ctx->ParamsFor.IoCtl.pInputBuffer;
  2835. -    NFS41GetDeviceExtension(RxContext, DevExt);
  2836. -    DWORD nfs41d_version = 0;
  2837. -
  2838. -    //DbgEn();
  2839. -
  2840. -    //print_ioctl(op);
  2841. -    switch(op) {
  2842. -    case IRP_MJ_FILE_SYSTEM_CONTROL:
  2843. -        status = STATUS_INVALID_DEVICE_REQUEST;
  2844. -        break;
  2845. -    case IRP_MJ_DEVICE_CONTROL:
  2846. -    case IRP_MJ_INTERNAL_DEVICE_CONTROL:
  2847. -        //print_fs_ioctl(fsop);
  2848. -        switch (fsop) {
  2849. -        case IOCTL_NFS41_INVALCACHE:
  2850. -            status = nfs41_invalidate_cache(RxContext);
  2851. -            break;
  2852. -        case IOCTL_NFS41_READ:
  2853. -            status = nfs41_upcall(RxContext);
  2854. -            break;
  2855. -        case IOCTL_NFS41_WRITE:
  2856. -            status = nfs41_downcall(RxContext);
  2857. -            break;
  2858. -        case IOCTL_NFS41_ADDCONN:
  2859. -            status = nfs41_CreateConnection(RxContext, &RxContext->PostRequest);
  2860. -            break;
  2861. -        case IOCTL_NFS41_DELCONN:
  2862. -            if (RxContext->RxDeviceObject->NumberOfActiveFcbs > 0) {
  2863. -                DbgP("device has open handles %d\n",
  2864. -                    RxContext->RxDeviceObject->NumberOfActiveFcbs);
  2865. -                status = STATUS_REDIRECTOR_HAS_OPEN_HANDLES;
  2866. -                break;
  2867. -            }
  2868. -            status = nfs41_DeleteConnection(RxContext, &RxContext->PostRequest);
  2869. -            break;
  2870. -        case IOCTL_NFS41_GETSTATE:
  2871. -            state = RDR_NULL_STATE;
  2872. -
  2873. -            if (io_ctx->ParamsFor.IoCtl.OutputBufferLength >= sizeof(ULONG)) {
  2874. -                // map the states to control app's equivalents
  2875. -                print_driver_state(nfs41_start_state);
  2876. -                switch (nfs41_start_state) {
  2877. -                case NFS41_START_DRIVER_STARTABLE:
  2878. -                case NFS41_START_DRIVER_STOPPED:
  2879. -                    state = RDR_STOPPED;
  2880. -                    break;
  2881. -                case NFS41_START_DRIVER_START_IN_PROGRESS:
  2882. -                    state = RDR_STARTING;
  2883. -                    break;
  2884. -                case NFS41_START_DRIVER_STARTED:
  2885. -                    state = RDR_STARTED;
  2886. -                    break;
  2887. -                }
  2888. -                *(ULONG *)io_ctx->ParamsFor.IoCtl.pOutputBuffer = state;
  2889. -                RxContext->InformationToReturn = sizeof(ULONG);
  2890. -                status = STATUS_SUCCESS;
  2891. -            } else
  2892. -                status = STATUS_INVALID_PARAMETER;
  2893. -            break;
  2894. -        case IOCTL_NFS41_START:
  2895. -            print_driver_state(nfs41_start_state);
  2896. -            if (in_len >= sizeof(DWORD)) {
  2897. -                RtlCopyMemory(&nfs41d_version, buf, sizeof(DWORD));
  2898. -                DbgP("NFS41 Daemon sent start request with version %d\n",
  2899. -                    nfs41d_version);
  2900. -                DbgP("Currently used NFS41 Daemon version is %d\n",
  2901. -                    DevExt->nfs41d_version);
  2902. -                DevExt->nfs41d_version = nfs41d_version;
  2903. -            }
  2904. -            switch(nfs41_start_state) {
  2905. -            case NFS41_START_DRIVER_STARTABLE:
  2906. -                (nfs41_start_driver_state)InterlockedCompareExchange(
  2907. -                              (PLONG)&nfs41_start_state,
  2908. -                              NFS41_START_DRIVER_START_IN_PROGRESS,
  2909. -                              NFS41_START_DRIVER_STARTABLE);
  2910. -                    //lack of break is intentional
  2911. -            case NFS41_START_DRIVER_START_IN_PROGRESS:
  2912. -                status = RxStartMinirdr(RxContext, &RxContext->PostRequest);
  2913. -                if (status == STATUS_REDIRECTOR_STARTED) {
  2914. -                    DbgP("redirector started\n");
  2915. -                    status = STATUS_SUCCESS;
  2916. -                } else if (status == STATUS_PENDING &&
  2917. -                            RxContext->PostRequest == TRUE) {
  2918. -                    DbgP("RxStartMinirdr pending 0x%08lx\n", status);
  2919. -                    status = STATUS_MORE_PROCESSING_REQUIRED;
  2920. -                }
  2921. -                break;
  2922. -            case NFS41_START_DRIVER_STARTED:
  2923. -                status = STATUS_SUCCESS;
  2924. -                break;
  2925. -            default:
  2926. -                status = STATUS_INVALID_PARAMETER;
  2927. -            }
  2928. -            break;
  2929. -        case IOCTL_NFS41_STOP:
  2930. -            if (nfs41_start_state == NFS41_START_DRIVER_STARTED)
  2931. -                nfs41_shutdown_daemon(DevExt->nfs41d_version);
  2932. -            if (RxContext->RxDeviceObject->NumberOfActiveFcbs > 0) {
  2933. -                DbgP("device has open handles %d\n",
  2934. -                    RxContext->RxDeviceObject->NumberOfActiveFcbs);
  2935. -                status = STATUS_REDIRECTOR_HAS_OPEN_HANDLES;
  2936. -                break;
  2937. -            }
  2938. -
  2939. -            state = (nfs41_start_driver_state)InterlockedCompareExchange(
  2940. -                        (PLONG)&nfs41_start_state,
  2941. -                        NFS41_START_DRIVER_STARTABLE,
  2942. -                        NFS41_START_DRIVER_STARTED);
  2943. -
  2944. -            status = RxStopMinirdr(RxContext, &RxContext->PostRequest);
  2945. -            DbgP("RxStopMinirdr status 0x%08lx\n", status);
  2946. -            if (status == STATUS_PENDING && RxContext->PostRequest == TRUE )
  2947. -                status = STATUS_MORE_PROCESSING_REQUIRED;
  2948. -            break;
  2949. -        default:
  2950. -            status = STATUS_INVALID_DEVICE_REQUEST;
  2951. -        };
  2952. -        break;
  2953. -    default:
  2954. -        status = STATUS_INVALID_DEVICE_REQUEST;
  2955. -    };
  2956. -
  2957. -    //DbgEx();
  2958. -    return status;
  2959. -}
  2960. -
  2961. -static NTSTATUS _nfs41_CreateSrvCall(
  2962. -    PMRX_SRVCALL_CALLBACK_CONTEXT pCallbackContext)
  2963. -{
  2964. -    NTSTATUS status = STATUS_SUCCESS;
  2965. -    PMRX_SRVCALL_CALLBACK_CONTEXT SCCBC = pCallbackContext;
  2966. -    PMRX_SRV_CALL pSrvCall;
  2967. -    PMRX_SRVCALLDOWN_STRUCTURE SrvCalldownStructure =
  2968. -        (PMRX_SRVCALLDOWN_STRUCTURE)(SCCBC->SrvCalldownStructure);
  2969. -    PNFS41_SERVER_ENTRY pServerEntry = NULL;
  2970. -
  2971. -#ifdef DEBUG_MOUNT
  2972. -    DbgEn();
  2973. -#endif
  2974. -
  2975. -    pSrvCall = SrvCalldownStructure->SrvCall;
  2976. -
  2977. -    ASSERT( pSrvCall );
  2978. -    ASSERT( NodeType(pSrvCall) == RDBSS_NTC_SRVCALL );
  2979. -    // print_srv_call(pSrvCall);
  2980. -
  2981. -    // validate the server name with the test name of 'pnfs'
  2982. -#ifdef DEBUG_MOUNT
  2983. -    DbgP("SrvCall: Connection Name Length: %d '%wZ'\n",
  2984. -        pSrvCall->pSrvCallName->Length, pSrvCall->pSrvCallName);
  2985. -#endif
  2986. -
  2987. -    if (pSrvCall->pSrvCallName->Length > SERVER_NAME_BUFFER_SIZE) {
  2988. -        print_error("Server name '%wZ' too long for server entry (max %u)\n",
  2989. -            pSrvCall->pSrvCallName, SERVER_NAME_BUFFER_SIZE);
  2990. -        status = STATUS_NAME_TOO_LONG;
  2991. -        goto out;
  2992. -    }
  2993. -
  2994. -    /* Let's create our own representation of the server */
  2995. -    pServerEntry = (PNFS41_SERVER_ENTRY)RxAllocatePoolWithTag(NonPagedPoolNx,
  2996. -        sizeof(NFS41_SERVER_ENTRY), NFS41_MM_POOLTAG);
  2997. -    if (pServerEntry == NULL) {
  2998. -        status = STATUS_INSUFFICIENT_RESOURCES;
  2999. -        goto out;
  3000. -    }
  3001. -    RtlZeroMemory(pServerEntry, sizeof(NFS41_SERVER_ENTRY));
  3002. -
  3003. -    pServerEntry->Name.Buffer = pServerEntry->NameBuffer;
  3004. -    pServerEntry->Name.Length = pSrvCall->pSrvCallName->Length;
  3005. -    pServerEntry->Name.MaximumLength = SERVER_NAME_BUFFER_SIZE;
  3006. -    RtlCopyMemory(pServerEntry->Name.Buffer, pSrvCall->pSrvCallName->Buffer,
  3007. -        pServerEntry->Name.Length);
  3008. -
  3009. -    pCallbackContext->RecommunicateContext = pServerEntry;
  3010. -    InterlockedExchangePointer(&pServerEntry->pRdbssSrvCall, pSrvCall);
  3011. -
  3012. -out:
  3013. -    SCCBC->Status = status;
  3014. -    SrvCalldownStructure->CallBack(SCCBC);
  3015. -
  3016. -#ifdef DEBUG_MOUNT
  3017. -    DbgEx();
  3018. -#endif
  3019. -    return status;
  3020. -}
  3021. -
  3022. -static NTSTATUS nfs41_CreateSrvCall(
  3023. -    PMRX_SRV_CALL pSrvCall,
  3024. -    PMRX_SRVCALL_CALLBACK_CONTEXT pCallbackContext)
  3025. -{
  3026. -    NTSTATUS status;
  3027. -
  3028. -    ASSERT( pSrvCall );
  3029. -    ASSERT( NodeType(pSrvCall) == RDBSS_NTC_SRVCALL );
  3030. -
  3031. -    if (IoGetCurrentProcess() == RxGetRDBSSProcess()) {
  3032. -        DbgP("executing with RDBSS context\n");
  3033. -        status = _nfs41_CreateSrvCall(pCallbackContext);
  3034. -    } else {
  3035. -        status = RxDispatchToWorkerThread(nfs41_dev, DelayedWorkQueue,
  3036. -           (PRX_WORKERTHREAD_ROUTINE)_nfs41_CreateSrvCall, pCallbackContext);
  3037. -        if (status != STATUS_SUCCESS) {
  3038. -            print_error("RxDispatchToWorkerThread returned status 0x%08lx\n",
  3039. -                status);
  3040. -            pCallbackContext->Status = status;
  3041. -            pCallbackContext->SrvCalldownStructure->CallBack(pCallbackContext);
  3042. -            status = STATUS_PENDING;
  3043. -        }
  3044. -    }
  3045. -    /* RDBSS expects MRxCreateSrvCall to return STATUS_PENDING */
  3046. -    if (status == STATUS_SUCCESS)
  3047. -        status = STATUS_PENDING;
  3048. -
  3049. -    return status;
  3050. -}
  3051. -
  3052. -static NTSTATUS nfs41_SrvCallWinnerNotify(
  3053. -    IN OUT PMRX_SRV_CALL pSrvCall,
  3054. -    IN BOOLEAN ThisMinirdrIsTheWinner,
  3055. -    IN OUT PVOID pSrvCallContext)
  3056. -{
  3057. -    NTSTATUS status = STATUS_SUCCESS;
  3058. -    PNFS41_SERVER_ENTRY pServerEntry;
  3059. -
  3060. -    pServerEntry = (PNFS41_SERVER_ENTRY)pSrvCallContext;
  3061. -
  3062. -    if (!ThisMinirdrIsTheWinner) {
  3063. -        ASSERT(1);
  3064. -        goto out;
  3065. -    }
  3066. -
  3067. -    pSrvCall->Context = pServerEntry;
  3068. -out:
  3069. -    return status;
  3070. -}
  3071. -
  3072. -static NTSTATUS map_mount_errors(
  3073. -    DWORD status)
  3074. -{
  3075. -    switch (status) {
  3076. -    case NO_ERROR:              return STATUS_SUCCESS;
  3077. -    case ERROR_ACCESS_DENIED:   return STATUS_ACCESS_DENIED;
  3078. -    case ERROR_NETWORK_UNREACHABLE: return STATUS_NETWORK_UNREACHABLE;
  3079. -    case ERROR_BAD_NET_RESP:    return STATUS_UNEXPECTED_NETWORK_ERROR;
  3080. -    case ERROR_BAD_NET_NAME:    return STATUS_BAD_NETWORK_NAME;
  3081. -    case ERROR_BAD_NETPATH:     return STATUS_BAD_NETWORK_PATH;
  3082. -    case ERROR_NOT_SUPPORTED:   return STATUS_NOT_SUPPORTED;
  3083. -    case ERROR_INTERNAL_ERROR:  return STATUS_INTERNAL_ERROR;
  3084. -    default:
  3085. -        print_error("map_mount_errors: "
  3086. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  3087. -            "defaulting to STATUS_INSUFFICIENT_RESOURCES\n", status);
  3088. -        return STATUS_INSUFFICIENT_RESOURCES;
  3089. -    }
  3090. -}
  3091. -
  3092. -static NTSTATUS nfs41_mount(
  3093. -    PNFS41_MOUNT_CONFIG config,
  3094. -    DWORD sec_flavor,
  3095. -    PHANDLE session,
  3096. -    DWORD *version,
  3097. -    PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs)
  3098. -{
  3099. -    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  3100. -    nfs41_updowncall_entry *entry;
  3101. -
  3102. -#ifdef DEBUG_MOUNT
  3103. -    DbgEn();
  3104. -    DbgP("Server Name '%wZ' Mount Point '%wZ' SecFlavor %d\n",
  3105. -        &config->SrvName, &config->MntPt, sec_flavor);
  3106. -#endif
  3107. -    status = nfs41_UpcallCreate(NFS41_MOUNT, NULL, *session,
  3108. -        INVALID_HANDLE_VALUE, *version, &config->MntPt, &entry);
  3109. -    if (status) goto out;
  3110. -
  3111. -    entry->u.Mount.srv_name = &config->SrvName;
  3112. -    entry->u.Mount.root = &config->MntPt;
  3113. -    entry->u.Mount.rsize = config->ReadSize;
  3114. -    entry->u.Mount.wsize = config->WriteSize;
  3115. -    entry->u.Mount.use_nfspubfh = config->use_nfspubfh;
  3116. -    entry->u.Mount.sec_flavor = sec_flavor;
  3117. -    entry->u.Mount.FsAttrs = FsAttrs;
  3118. -
  3119. -    status = nfs41_UpcallWaitForReply(entry, config->timeout);
  3120. -    if (entry->psec_ctx == &entry->sec_ctx) {
  3121. -        SeDeleteClientSecurity(entry->psec_ctx);
  3122. -    }
  3123. -    entry->psec_ctx = NULL;
  3124. -    if (status) goto out;
  3125. -    *session = entry->session;
  3126. -    if (entry->u.Mount.lease_time > config->timeout)
  3127. -        config->timeout = entry->u.Mount.lease_time;
  3128. -
  3129. -    /* map windows ERRORs to NTSTATUS */
  3130. -    status = map_mount_errors(entry->status);
  3131. -    if (status == STATUS_SUCCESS)
  3132. -        *version = entry->version;
  3133. -    nfs41_UpcallDestroy(entry);
  3134. -out:
  3135. -#ifdef DEBUG_MOUNT
  3136. -    DbgEx();
  3137. -#endif
  3138. -    return status;
  3139. -}
  3140. -
  3141. -/* TODO: move mount config stuff to another file -cbodley */
  3142. -
  3143. -static void nfs41_MountConfig_InitDefaults(
  3144. -    OUT PNFS41_MOUNT_CONFIG Config)
  3145. -{
  3146. -    RtlZeroMemory(Config, sizeof(NFS41_MOUNT_CONFIG));
  3147. -
  3148. -    Config->ReadSize = MOUNT_CONFIG_RW_SIZE_DEFAULT;
  3149. -    Config->WriteSize = MOUNT_CONFIG_RW_SIZE_DEFAULT;
  3150. -    Config->use_nfspubfh = FALSE;
  3151. -    Config->ReadOnly = FALSE;
  3152. -    Config->write_thru = FALSE;
  3153. -    Config->nocache = FALSE;
  3154. -    Config->timebasedcoherency = FALSE; /* disabled by default because of bugs */
  3155. -    Config->SrvName.Length = 0;
  3156. -    Config->SrvName.MaximumLength = SERVER_NAME_BUFFER_SIZE;
  3157. -    Config->SrvName.Buffer = Config->srv_buffer;
  3158. -    Config->MntPt.Length = 0;
  3159. -    Config->MntPt.MaximumLength = NFS41_SYS_MAX_PATH_LEN;
  3160. -    Config->MntPt.Buffer = Config->mntpt_buffer;
  3161. -    Config->SecFlavor.Length = 0;
  3162. -    Config->SecFlavor.MaximumLength = MAX_SEC_FLAVOR_LEN;
  3163. -    Config->SecFlavor.Buffer = Config->sec_flavor_buffer;
  3164. -    RtlCopyUnicodeString(&Config->SecFlavor, &AUTH_SYS_NAME);
  3165. -    Config->timeout = UPCALL_TIMEOUT_DEFAULT;
  3166. -    Config->createmode.use_nfsv3attrsea_mode = TRUE;
  3167. -    Config->createmode.mode = NFS41_DRIVER_DEFAULT_CREATE_MODE;
  3168. -}
  3169. -
  3170. -static NTSTATUS nfs41_MountConfig_ParseBoolean(
  3171. -    IN PFILE_FULL_EA_INFORMATION Option,
  3172. -    IN PUNICODE_STRING usValue,
  3173. -    IN BOOLEAN negate_val,
  3174. -    OUT PBOOLEAN Value)
  3175. -{
  3176. -    NTSTATUS status = STATUS_SUCCESS;
  3177. -
  3178. -    /* if no value is specified, assume TRUE
  3179. -     * if a value is specified, it must be a '1' */
  3180. -    if (Option->EaValueLength == 0 || *usValue->Buffer == L'1')
  3181. -        *Value = negate_val?FALSE:TRUE;
  3182. -    else
  3183. -        *Value = negate_val?TRUE:FALSE;
  3184. -
  3185. -    DbgP("    '%ls' -> '%wZ' -> %u\n",
  3186. -        (LPWSTR)Option->EaName, usValue, *Value);
  3187. -    return status;
  3188. -}
  3189. -
  3190. -
  3191. -/* Parse |signed| integer value */
  3192. -static NTSTATUS nfs41_MountConfig_ParseINT64(
  3193. -    IN PFILE_FULL_EA_INFORMATION Option,
  3194. -    IN PUNICODE_STRING usValue,
  3195. -    OUT INT64 *outValue,
  3196. -    IN INT64 Minimum,
  3197. -    IN INT64 Maximum)
  3198. -{
  3199. -    NTSTATUS status;
  3200. -    LONG64 Value = 0;
  3201. -    LPWSTR Name = (LPWSTR)Option->EaName;
  3202. -
  3203. -    if (!Option->EaValueLength)
  3204. -        return STATUS_INVALID_PARAMETER;
  3205. -
  3206. -    status = RtlUnicodeStringToInt64(usValue, 0, &Value, NULL);
  3207. -    if (status == STATUS_SUCCESS) {
  3208. -        if ((Value < Minimum) || (Value > Maximum))
  3209. -            status = STATUS_INVALID_PARAMETER;
  3210. -
  3211. -        if (status == STATUS_SUCCESS) {
  3212. -            *outValue = Value;
  3213. -        }
  3214. -    }
  3215. -    else {
  3216. -        print_error("nfs41_MountConfig_ParseINT64: "
  3217. -            "Failed to convert '%s'='%wZ' to unsigned long.\n",
  3218. -            Name, usValue);
  3219. -    }
  3220. -
  3221. -    return status;
  3222. -}
  3223. -
  3224. -/* Parse |unsigned| integer value */
  3225. -static NTSTATUS nfs41_MountConfig_ParseDword(
  3226. -    IN PFILE_FULL_EA_INFORMATION Option,
  3227. -    IN PUNICODE_STRING usValue,
  3228. -    OUT PDWORD outValue,
  3229. -    IN DWORD Minimum,
  3230. -    IN DWORD Maximum)
  3231. -{
  3232. -    INT64 tmpValue;
  3233. -    NTSTATUS status;
  3234. -
  3235. -    status = nfs41_MountConfig_ParseINT64(
  3236. -        Option, usValue,
  3237. -        &tmpValue, Minimum, Maximum);
  3238. -
  3239. -    if (status == STATUS_SUCCESS) {
  3240. -        *outValue = (DWORD)tmpValue;
  3241. -    }
  3242. -
  3243. -    return status;
  3244. -}
  3245. -
  3246. -static NTSTATUS nfs41_MountConfig_ParseOptions(
  3247. -    IN PFILE_FULL_EA_INFORMATION EaBuffer,
  3248. -    IN ULONG EaLength,
  3249. -    IN OUT PNFS41_MOUNT_CONFIG Config)
  3250. -{
  3251. -    DbgP("--> nfs41_MountConfig_ParseOptions(EaBuffer=0x%p,EaLength=%ld)\n",
  3252. -        (void *)EaBuffer,
  3253. -        (long)EaLength);
  3254. -    NTSTATUS  status = STATUS_SUCCESS;
  3255. -    PFILE_FULL_EA_INFORMATION Option;
  3256. -    LPWSTR Name;
  3257. -    size_t NameLen;
  3258. -    UNICODE_STRING  usValue;
  3259. -    ULONG error_offset;
  3260. -
  3261. -    status = IoCheckEaBufferValidity(EaBuffer, EaLength, &error_offset);
  3262. -    if (status) {
  3263. -        DbgP("status(=0x%lx)=IoCheckEaBufferValidity"
  3264. -            "(eainfo=0x%p, buflen=%lu, &(error_offset=%d)) failed\n",
  3265. -            (long)status, (void *)EaBuffer, EaLength,
  3266. -            (int)error_offset);
  3267. -        goto out;
  3268. -    }
  3269. -
  3270. -    Option = EaBuffer;
  3271. -    while (status == STATUS_SUCCESS) {
  3272. -        DbgP("Option=0x%p\n", (void *)Option);
  3273. -        Name = (LPWSTR)Option->EaName;
  3274. -        NameLen = Option->EaNameLength/sizeof(WCHAR);
  3275. -
  3276. -        DbgP("nfs41_MountConfig_ParseOptions: Name='%*S'/NameLen=%d\n",
  3277. -            (int)NameLen, Name, (int)NameLen);
  3278. -
  3279. -        usValue.Length = usValue.MaximumLength = Option->EaValueLength;
  3280. -        usValue.Buffer = (PWCH)(Option->EaName +
  3281. -            Option->EaNameLength + sizeof(WCHAR));
  3282. -
  3283. -        DbgP("nfs41_MountConfig_ParseOptions: option/usValue='%wZ'/%ld\n",
  3284. -            &usValue, (long)usValue.Length);
  3285. -
  3286. -        if (wcsncmp(L"ro", Name, NameLen) == 0) {
  3287. -            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  3288. -                FALSE, &Config->ReadOnly);
  3289. -        } else if (wcsncmp(L"rw", Name, NameLen) == 0) {
  3290. -            /* opposite of "ro", so negate */
  3291. -            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  3292. -                TRUE, &Config->ReadOnly);
  3293. -        }
  3294. -        else if (wcsncmp(L"writethru", Name, NameLen) == 0) {
  3295. -            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  3296. -                FALSE, &Config->write_thru);
  3297. -        }
  3298. -        else if (wcsncmp(L"nowritethru", Name, NameLen) == 0) {
  3299. -            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  3300. -                TRUE, &Config->write_thru);
  3301. -        }
  3302. -        else if (wcsncmp(L"cache", Name, NameLen) == 0) {
  3303. -            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  3304. -                TRUE, &Config->nocache);
  3305. -        }
  3306. -        else if (wcsncmp(L"nocache", Name, NameLen) == 0) {
  3307. -            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  3308. -                FALSE, &Config->nocache);
  3309. -        }
  3310. -        else if (wcsncmp(L"timebasedcoherency", Name, NameLen) == 0) {
  3311. -            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  3312. -                FALSE, &Config->timebasedcoherency);
  3313. -        }
  3314. -        else if (wcsncmp(L"notimebasedcoherency", Name, NameLen) == 0) {
  3315. -            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  3316. -                TRUE, &Config->timebasedcoherency);
  3317. -        }
  3318. -        else if (wcsncmp(L"timeout", Name, NameLen) == 0) {
  3319. -            status = nfs41_MountConfig_ParseDword(Option, &usValue,
  3320. -                &Config->timeout, UPCALL_TIMEOUT_DEFAULT,
  3321. -                UPCALL_TIMEOUT_DEFAULT);
  3322. -        }
  3323. -        else if (wcsncmp(L"rsize", Name, NameLen) == 0) {
  3324. -            status = nfs41_MountConfig_ParseDword(Option, &usValue,
  3325. -                &Config->ReadSize, MOUNT_CONFIG_RW_SIZE_MIN,
  3326. -                MOUNT_CONFIG_RW_SIZE_MAX);
  3327. -        }
  3328. -        else if (wcsncmp(L"wsize", Name, NameLen) == 0) {
  3329. -            status = nfs41_MountConfig_ParseDword(Option, &usValue,
  3330. -                &Config->WriteSize, MOUNT_CONFIG_RW_SIZE_MIN,
  3331. -                MOUNT_CONFIG_RW_SIZE_MAX);
  3332. -        }
  3333. -        else if (wcsncmp(L"public", Name, NameLen) == 0) {
  3334. -            /*
  3335. -             + We ignore this value here, and instead rely on the
  3336. -             * /pubnfs4 prefix
  3337. -             */
  3338. -            BOOLEAN dummy;
  3339. -            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  3340. -                FALSE, &dummy);
  3341. -        }
  3342. -        else if (wcsncmp(L"srvname", Name, NameLen) == 0) {
  3343. -            if (usValue.Length > Config->SrvName.MaximumLength)
  3344. -                status = STATUS_NAME_TOO_LONG;
  3345. -            else
  3346. -                RtlCopyUnicodeString(&Config->SrvName, &usValue);
  3347. -        }
  3348. -        else if (wcsncmp(L"mntpt", Name, NameLen) == 0) {
  3349. -            if (usValue.Length > Config->MntPt.MaximumLength)
  3350. -                status = STATUS_NAME_TOO_LONG;
  3351. -            else
  3352. -                RtlCopyUnicodeString(&Config->MntPt, &usValue);
  3353. -        }
  3354. -        else if (wcsncmp(L"sec", Name, NameLen) == 0) {
  3355. -            if (usValue.Length > Config->SecFlavor.MaximumLength)
  3356. -                status = STATUS_NAME_TOO_LONG;
  3357. -            else
  3358. -                RtlCopyUnicodeString(&Config->SecFlavor, &usValue);
  3359. -        }
  3360. -        else if (wcsncmp(L"createmode", Name, NameLen) == 0) {
  3361. -#define NFSV3ATTRMODE_WSTR L"nfsv3attrmode+"
  3362. -#define NFSV3ATTRMODE_WCSLEN (14)
  3363. -#define NFSV3ATTRMODE_BYTELEN (NFSV3ATTRMODE_WCSLEN*sizeof(WCHAR))
  3364. -            if ((usValue.Length >= NFSV3ATTRMODE_BYTELEN) &&
  3365. -                (!wcsncmp(NFSV3ATTRMODE_WSTR,
  3366. -                    usValue.Buffer,
  3367. -                    min(NFSV3ATTRMODE_WCSLEN,
  3368. -                        usValue.Length/sizeof(WCHAR))))) {
  3369. -                usValue.Buffer += NFSV3ATTRMODE_WCSLEN;
  3370. -                usValue.Length = usValue.MaximumLength =
  3371. -                    usValue.Length - NFSV3ATTRMODE_BYTELEN;
  3372. -#ifdef DEBUG_MOUNTCONFIG
  3373. -                DbgP("nfs41_MountConfig_ParseOptions: createmode "
  3374. -                    "nfs4attr "
  3375. -                    "leftover option/usValue='%wZ'/%ld\n",
  3376. -                    &usValue, (long)usValue.Length);
  3377. -#endif /* DEBUG_MOUNTCONFIG */
  3378. -
  3379. -                Config->createmode.use_nfsv3attrsea_mode = TRUE;
  3380. -            }
  3381. -            else {
  3382. -#ifdef DEBUG_MOUNTCONFIG
  3383. -                DbgP("nfs41_MountConfig_ParseOptions: createmode "
  3384. -                    "leftover option/usValue='%wZ'/%ld\n",
  3385. -                    &usValue, (long)usValue.Length);
  3386. -#endif /* DEBUG_MOUNTCONFIG */
  3387. -                Config->createmode.use_nfsv3attrsea_mode = FALSE;
  3388. -            }
  3389. -
  3390. -            /*
  3391. -             * Reject mode values not prefixed with "0o", as
  3392. -             * |RtlUnicodeStringToInteger()| uses
  3393. -             * 0o (e.g. "0o123") as prefix for octal values,
  3394. -             * and does not understand the traditional
  3395. -             * UNIX/POSIX/ISO C "0" (e.g. "0123") prefix
  3396. -             */
  3397. -            if ((usValue.Length >= (3*sizeof(WCHAR))) &&
  3398. -                (usValue.Buffer[0] == L'0') &&
  3399. -                (usValue.Buffer[1] == L'o')) {
  3400. -                status = nfs41_MountConfig_ParseDword(Option,
  3401. -                    &usValue,
  3402. -                    &Config->createmode.mode, 0,
  3403. -                    0777);
  3404. -                if (status == STATUS_SUCCESS) {
  3405. -                    if (Config->createmode.mode > 0777) {
  3406. -                        status = STATUS_INVALID_PARAMETER;
  3407. -                        print_error("mode 0%o out of bounds\n",
  3408. -                            (int)Config->createmode.mode);
  3409. -                    }
  3410. -                }
  3411. -            }
  3412. -            else {
  3413. -                status = STATUS_INVALID_PARAMETER;
  3414. -                print_error("Invalid createmode '%wZ'\n",
  3415. -                    usValue);
  3416. -            }
  3417. -
  3418. -            DbgP("nfs41_MountConfig_ParseOptions: createmode: "
  3419. -                "status=0x%lx, "
  3420. -                "createmode=(use_nfsv3attrsea_mode=%d, mode=0%o\n",
  3421. -                (long)status,
  3422. -                (int)Config->createmode.use_nfsv3attrsea_mode,
  3423. -                (int)Config->createmode.mode);
  3424. -        }
  3425. -        else {
  3426. -            status = STATUS_INVALID_PARAMETER;
  3427. -            print_error("Unrecognized option '%ls' -> '%wZ'\n",
  3428. -                Name, usValue);
  3429. -        }
  3430. -
  3431. -        if (Option->NextEntryOffset == 0)
  3432. -            break;
  3433. -
  3434. -        Option = (PFILE_FULL_EA_INFORMATION)
  3435. -            ((PBYTE)Option + Option->NextEntryOffset);
  3436. -    }
  3437. -
  3438. -out:
  3439. -    DbgP("<-- nfs41_MountConfig_ParseOptions, status=0x%lx\n",
  3440. -        (long)status);
  3441. -    return status;
  3442. -}
  3443. -
  3444. -static NTSTATUS has_nfs_prefix(
  3445. -    IN PUNICODE_STRING SrvCallName,
  3446. -    IN PUNICODE_STRING NetRootName,
  3447. -    OUT BOOLEAN *pubfh_prefix)
  3448. -{
  3449. -    NTSTATUS status = STATUS_BAD_NETWORK_NAME;
  3450. -
  3451. -#ifdef USE_ENTIRE_PATH_FOR_NETROOT
  3452. -    if (NetRootName->Length >=
  3453. -        (SrvCallName->Length + NfsPrefix.Length)) {
  3454. -        size_t len = NetRootName->Length / 2;
  3455. -        size_t i;
  3456. -        int state = 0;
  3457. -
  3458. -        /* Scan \hostname@port\nfs4 */
  3459. -        for (i = 0 ; i < len ; i++) {
  3460. -            wchar_t ch = NetRootName->Buffer[i];
  3461. -
  3462. -            if ((ch == L'\\') && (state == 0)) {
  3463. -                state = 1;
  3464. -                continue;
  3465. -            }
  3466. -            else if ((ch == L'@') && (state == 1)) {
  3467. -                state = 2;
  3468. -                continue;
  3469. -            }
  3470. -            else if ((ch == L'\\') && (state == 2)) {
  3471. -                state = 3;
  3472. -                break;
  3473. -            }
  3474. -            else if (ch == L'\\') {
  3475. -                /* Abort, '\\' with wrong state */
  3476. -                break;
  3477. -            }
  3478. -        }
  3479. -
  3480. -        if (state == 3) {
  3481. -            if (!memcmp(&NetRootName->Buffer[i], L"\\nfs4",
  3482. -                (4*sizeof(wchar_t))))) {
  3483. -                *pubfh_prefix = FALSE;
  3484. -                status = STATUS_SUCCESS;
  3485. -            }
  3486. -            if ((NetRootName->Length >=
  3487. -                (SrvCallName->Length + PubNfsPrefix.Length)) &&
  3488. -                (!memcmp(&NetRootName->Buffer[i], L"\\pubnfs4",
  3489. -                    (4*sizeof(wchar_t))))) {
  3490. -                *pubfh_prefix = TRUE;
  3491. -                status = STATUS_SUCCESS;
  3492. -            }
  3493. -        }
  3494. -    }
  3495. -#else
  3496. -    if (NetRootName->Length ==
  3497. -        (SrvCallName->Length + NfsPrefix.Length)) {
  3498. -        const UNICODE_STRING NetRootPrefix = {
  3499. -            NfsPrefix.Length,
  3500. -            NetRootName->MaximumLength - SrvCallName->Length,
  3501. -            &NetRootName->Buffer[SrvCallName->Length/2]
  3502. -        };
  3503. -        if (!RtlCompareUnicodeString(&NetRootPrefix, &NfsPrefix, FALSE))
  3504. -            *pubfh_prefix = FALSE;
  3505. -            status = STATUS_SUCCESS;
  3506. -    }
  3507. -    else if (NetRootName->Length ==
  3508. -        (SrvCallName->Length + PubNfsPrefix.Length)) {
  3509. -        const UNICODE_STRING PubNetRootPrefix = {
  3510. -            PubNfsPrefix.Length,
  3511. -            NetRootName->MaximumLength - SrvCallName->Length,
  3512. -            &NetRootName->Buffer[SrvCallName->Length/2]
  3513. -        };
  3514. -        if (!RtlCompareUnicodeString(&PubNetRootPrefix, &PubNfsPrefix, FALSE))
  3515. -            *pubfh_prefix = TRUE;
  3516. -            status = STATUS_SUCCESS;
  3517. -    }
  3518. -#endif
  3519. -    return status;
  3520. -}
  3521. -
  3522. -static NTSTATUS map_sec_flavor(
  3523. -    IN PUNICODE_STRING sec_flavor_name,
  3524. -    OUT PDWORD sec_flavor)
  3525. -{
  3526. -    if (RtlCompareUnicodeString(sec_flavor_name, &AUTH_SYS_NAME, FALSE) == 0)
  3527. -        *sec_flavor = RPCSEC_AUTH_SYS;
  3528. -    else if (RtlCompareUnicodeString(sec_flavor_name, &AUTHGSS_KRB5_NAME, FALSE) == 0)
  3529. -        *sec_flavor = RPCSEC_AUTHGSS_KRB5;
  3530. -    else if (RtlCompareUnicodeString(sec_flavor_name, &AUTHGSS_KRB5I_NAME, FALSE) == 0)
  3531. -        *sec_flavor = RPCSEC_AUTHGSS_KRB5I;
  3532. -    else if (RtlCompareUnicodeString(sec_flavor_name, &AUTHGSS_KRB5P_NAME, FALSE) == 0)
  3533. -        *sec_flavor = RPCSEC_AUTHGSS_KRB5P;
  3534. -    else return STATUS_INVALID_PARAMETER;
  3535. -    return STATUS_SUCCESS;
  3536. -}
  3537. -
  3538. -static NTSTATUS nfs41_GetLUID(
  3539. -    PLUID id)
  3540. -{
  3541. -    NTSTATUS status = STATUS_SUCCESS;
  3542. -    SECURITY_SUBJECT_CONTEXT sec_ctx;
  3543. -    SECURITY_QUALITY_OF_SERVICE sec_qos;
  3544. -    SECURITY_CLIENT_CONTEXT clnt_sec_ctx;
  3545. -
  3546. -    SeCaptureSubjectContext(&sec_ctx);
  3547. -    sec_qos.ContextTrackingMode = SECURITY_STATIC_TRACKING;
  3548. -    sec_qos.ImpersonationLevel = SecurityIdentification;
  3549. -    sec_qos.Length = sizeof(SECURITY_QUALITY_OF_SERVICE);
  3550. -    sec_qos.EffectiveOnly = 0;
  3551. -    /*
  3552. -     * Arg |ServerIsRemote| must be |FALSE|, otherwise processes
  3553. -     * like Cygwin setup-x86_64.exe can fail during "Activation
  3554. -     * Context" creation in
  3555. -     * |SeCreateClientSecurityFromSubjectContext()| with
  3556. -     * |STATUS_BAD_IMPERSONATION_LEVEL|
  3557. -     */
  3558. -    status = SeCreateClientSecurityFromSubjectContext(&sec_ctx, &sec_qos,
  3559. -        FALSE, &clnt_sec_ctx);
  3560. -    if (status) {
  3561. -        print_error("nfs41_GetLUID: SeCreateClientSecurityFromSubjectContext "
  3562. -             "failed 0x%x\n", status);
  3563. -        goto release_sec_ctx;
  3564. -    }
  3565. -    status = SeQueryAuthenticationIdToken(clnt_sec_ctx.ClientToken, id);
  3566. -    if (status) {
  3567. -        print_error("nfs41_GetLUID: "
  3568. -            "SeQueryAuthenticationIdToken() failed 0x%x\n", status);
  3569. -        goto release_clnt_sec_ctx;
  3570. -    }
  3571. -release_clnt_sec_ctx:
  3572. -    SeDeleteClientSecurity(&clnt_sec_ctx);
  3573. -release_sec_ctx:
  3574. -    SeReleaseSubjectContext(&sec_ctx);
  3575. -
  3576. -    return status;
  3577. -}
  3578. -
  3579. -static NTSTATUS nfs41_get_sec_ctx(
  3580. -    IN enum _SECURITY_IMPERSONATION_LEVEL level,
  3581. -    OUT PSECURITY_CLIENT_CONTEXT out_ctx)
  3582. -{
  3583. -    NTSTATUS status;
  3584. -    SECURITY_SUBJECT_CONTEXT ctx;
  3585. -    SECURITY_QUALITY_OF_SERVICE sec_qos;
  3586. -
  3587. -    SeCaptureSubjectContext(&ctx);
  3588. -    sec_qos.ContextTrackingMode = SECURITY_STATIC_TRACKING;
  3589. -    sec_qos.ImpersonationLevel = level;
  3590. -    sec_qos.Length = sizeof(SECURITY_QUALITY_OF_SERVICE);
  3591. -    sec_qos.EffectiveOnly = 0;
  3592. -    /*
  3593. -     * Arg |ServerIsRemote| must be |FALSE|, otherwise processes
  3594. -     * like Cygwin setup-x86_64.exe can fail during "Activation
  3595. -     * Context" creation in
  3596. -     * |SeCreateClientSecurityFromSubjectContext()| with
  3597. -     * |STATUS_BAD_IMPERSONATION_LEVEL|
  3598. -     */
  3599. -    status = SeCreateClientSecurityFromSubjectContext(&ctx, &sec_qos,
  3600. -        FALSE, out_ctx);
  3601. -    if (status != STATUS_SUCCESS) {
  3602. -        print_error("SeCreateClientSecurityFromSubjectContext "
  3603. -            "failed with 0x%x\n", status);
  3604. -    }
  3605. -#ifdef DEBUG_SECURITY_TOKEN
  3606. -    DbgP("Created client security token 0x%p\n", out_ctx->ClientToken);
  3607. -#endif
  3608. -    SeReleaseSubjectContext(&ctx);
  3609. -
  3610. -    return status;
  3611. -}
  3612. -
  3613. -static NTSTATUS nfs41_CreateVNetRoot(
  3614. -    IN OUT PMRX_CREATENETROOT_CONTEXT pCreateNetRootContext)
  3615. -{
  3616. -    NTSTATUS status = STATUS_SUCCESS;
  3617. -    NFS41_MOUNT_CONFIG *Config;
  3618. -    __notnull PMRX_V_NET_ROOT pVNetRoot = (PMRX_V_NET_ROOT)
  3619. -        pCreateNetRootContext->pVNetRoot;
  3620. -    __notnull PMRX_NET_ROOT pNetRoot = pVNetRoot->pNetRoot;
  3621. -    __notnull PMRX_SRV_CALL pSrvCall = pNetRoot->pSrvCall;
  3622. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  3623. -        NFS41GetVNetRootExtension(pVNetRoot);
  3624. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  3625. -        NFS41GetNetRootExtension(pNetRoot);
  3626. -    NFS41GetDeviceExtension(pCreateNetRootContext->RxContext,DevExt);
  3627. -    DWORD nfs41d_version = DevExt->nfs41d_version;
  3628. -    nfs41_mount_entry *existing_mount = NULL;
  3629. -    LUID luid;
  3630. -    BOOLEAN found_existing_mount = FALSE, found_matching_flavor = FALSE;
  3631. -
  3632. -    ASSERT((NodeType(pNetRoot) == RDBSS_NTC_NETROOT) &&
  3633. -        (NodeType(pNetRoot->pSrvCall) == RDBSS_NTC_SRVCALL));
  3634. -
  3635. -#ifdef DEBUG_MOUNT
  3636. -    DbgEn();
  3637. -    // print_srv_call(pSrvCall);
  3638. -    // print_net_root(pNetRoot);
  3639. -    // print_v_net_root(pVNetRoot);
  3640. -
  3641. -    DbgP("pVNetRoot=0x%p pNetRoot=0x%p pSrvCall=0x%p\n", pVNetRoot, pNetRoot, pSrvCall);
  3642. -    DbgP("pNetRoot='%wZ' Type=%d pSrvCallName='%wZ' VirtualNetRootStatus=0x%x "
  3643. -        "NetRootStatus=0x%x\n", pNetRoot->pNetRootName,
  3644. -        pNetRoot->Type, pSrvCall->pSrvCallName,
  3645. -        pCreateNetRootContext->VirtualNetRootStatus,
  3646. -        pCreateNetRootContext->NetRootStatus);
  3647. -#endif
  3648. -
  3649. -    if (pNetRoot->Type != NET_ROOT_DISK && pNetRoot->Type != NET_ROOT_WILD) {
  3650. -        print_error("nfs41_CreateVNetRoot: Unsupported NetRoot Type %u\n",
  3651. -            pNetRoot->Type);
  3652. -        status = STATUS_NOT_SUPPORTED;
  3653. -        goto out;
  3654. -    }
  3655. -
  3656. -    pVNetRootContext->session = INVALID_HANDLE_VALUE;
  3657. -
  3658. -    /*
  3659. -     * In order to cooperate with other network providers, we
  3660. -     * must only claim paths of the form '\\server\nfs4\path'
  3661. -     * or '\\server\pubnfs4\path'
  3662. -     */
  3663. -    BOOLEAN pubfh_prefix = FALSE;
  3664. -    status = has_nfs_prefix(pSrvCall->pSrvCallName, pNetRoot->pNetRootName, &pubfh_prefix);
  3665. -    if (status) {
  3666. -        print_error("nfs41_CreateVNetRoot: NetRootName '%wZ' doesn't match "
  3667. -            "'\\nfs4' or '\\pubnfs4'!\n", pNetRoot->pNetRootName);
  3668. -        goto out;
  3669. -    }
  3670. -    pNetRoot->MRxNetRootState = MRX_NET_ROOT_STATE_GOOD;
  3671. -    pNetRoot->DeviceType = FILE_DEVICE_DISK;
  3672. -
  3673. -    Config = RxAllocatePoolWithTag(NonPagedPoolNx,
  3674. -            sizeof(NFS41_MOUNT_CONFIG), NFS41_MM_POOLTAG);
  3675. -    if (Config == NULL) {
  3676. -        status = STATUS_INSUFFICIENT_RESOURCES;
  3677. -        goto out;
  3678. -    }
  3679. -    nfs41_MountConfig_InitDefaults(Config);
  3680. -
  3681. -    if (pCreateNetRootContext->RxContext->Create.EaLength) {
  3682. -        /* Codepath for nfs_mount.exe */
  3683. -        DbgP("Codepath for nfs_mount.exe, "
  3684. -            "Create->{ EaBuffer=0x%p, EaLength=%ld }\n",
  3685. -            pCreateNetRootContext->RxContext->Create.EaBuffer,
  3686. -            (long)pCreateNetRootContext->RxContext->Create.EaLength);
  3687. -
  3688. -        /* parse the extended attributes for mount options */
  3689. -        status = nfs41_MountConfig_ParseOptions(
  3690. -            pCreateNetRootContext->RxContext->Create.EaBuffer,
  3691. -            pCreateNetRootContext->RxContext->Create.EaLength,
  3692. -            Config);
  3693. -        if (status != STATUS_SUCCESS) {
  3694. -            DbgP("nfs41_MountConfig_ParseOptions() failed\n");
  3695. -            goto out_free;
  3696. -        }
  3697. -        pVNetRootContext->read_only = Config->ReadOnly;
  3698. -        pVNetRootContext->write_thru = Config->write_thru;
  3699. -        pVNetRootContext->nocache = Config->nocache;
  3700. -        pVNetRootContext->timebasedcoherency = Config->timebasedcoherency;
  3701. -    } else {
  3702. -        /*
  3703. -         * Codepath for \\server@port\nfs4\path or
  3704. -         * \\server@port\pubnfs4\path
  3705. -         */
  3706. -        DbgP("Codepath for \\\\server@port\\@(pubnfs4|nfs4)\\path\n");
  3707. -
  3708. -        /*
  3709. -         * STATUS_NFS_SHARE_NOT_MOUNTED - status code for the case
  3710. -         * when a NFS filesystem is accessed via UNC path, but no
  3711. -         * nfs_mount.exe was done for that filesystem
  3712. -         */
  3713. -#define STATUS_NFS_SHARE_NOT_MOUNTED STATUS_BAD_NETWORK_PATH
  3714. -        if (!pNetRootContext->mounts_init) {
  3715. -            /*
  3716. -             * We can only support UNC paths when we got valid
  3717. -             * mount options via nfs_mount.exe before this point.
  3718. -             */
  3719. -            DbgP("pNetRootContext(=0x%p) not initalised yet\n",
  3720. -                pNetRootContext);
  3721. -            status = STATUS_NFS_SHARE_NOT_MOUNTED;
  3722. -            goto out_free;
  3723. -        }
  3724. -
  3725. -        /*
  3726. -         * gisburn: Fixme: Originally the code was using the
  3727. -         * SRV_CALL name (without leading \) as the hostname
  3728. -         * like this:
  3729. -         * ---- snip ----
  3730. -         * Config->SrvName.Buffer = pSrvCall->pSrvCallName->Buffer+1;
  3731. -         * Config->SrvName.Length =
  3732. -         *     pSrvCall->pSrvCallName->Length - sizeof(WCHAR);
  3733. -         * Config->SrvName.MaximumLength =
  3734. -         *     pSrvCall->pSrvCallName->MaximumLength - sizeof(WCHAR);
  3735. -         * ---- snip ----
  3736. -         * IMHO we should validate that the hostname in
  3737. -         * |existing_mount->Config| below matches
  3738. -         * |pSrvCall->pSrvCallName->Buffer|
  3739. -         */
  3740. -
  3741. -        status = nfs41_GetLUID(&luid);
  3742. -        if (status)
  3743. -            goto out_free;
  3744. -
  3745. -#ifdef DEBUG_MOUNT
  3746. -        DbgP("UNC path LUID 0x%lx.0x%lx\n",
  3747. -            (long)luid.HighPart, (long)luid.LowPart);
  3748. -#endif
  3749. -
  3750. -        PLIST_ENTRY pEntry;
  3751. -        nfs41_mount_entry *found_mount_entry = NULL;
  3752. -#ifdef NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL
  3753. -        nfs41_mount_entry *found_system_mount_entry = NULL;
  3754. -#endif /* NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL */
  3755. -
  3756. -        status = STATUS_NFS_SHARE_NOT_MOUNTED;
  3757. -
  3758. -        ExAcquireFastMutex(&pNetRootContext->mountLock);
  3759. -        pEntry = &pNetRootContext->mounts.head;
  3760. -        pEntry = pEntry->Flink;
  3761. -        while (pEntry != NULL) {
  3762. -            existing_mount = (nfs41_mount_entry *)CONTAINING_RECORD(pEntry,
  3763. -                    nfs41_mount_entry, next);
  3764. -
  3765. -#ifdef DEBUG_MOUNT
  3766. -            DbgP("finding mount config: "
  3767. -                "comparing luid=(0x%lx.0x%lx) with "
  3768. -                "existing_mount->login_id=(0x%lx.0x%lx)\n",
  3769. -                (long)luid.HighPart, (long)luid.LowPart,
  3770. -                (long)existing_mount->login_id.HighPart,
  3771. -                (long)existing_mount->login_id.LowPart);
  3772. -#endif
  3773. -
  3774. -            if (RtlEqualLuid(&luid, &existing_mount->login_id)) {
  3775. -                /* found existing mount with exact LUID match */
  3776. -                found_mount_entry = existing_mount;
  3777. -                break;
  3778. -            }
  3779. -#ifdef NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL
  3780. -            else if (RtlEqualLuid(&SystemLuid,
  3781. -                &existing_mount->login_id)) {
  3782. -                /*
  3783. -                 * found existing mount for user "SYSTEM"
  3784. -                 * We continue searching the |pNetRootContext->mounts|
  3785. -                 * list for an exact match ...
  3786. -                 */
  3787. -                found_system_mount_entry = existing_mount;
  3788. -            }
  3789. -#endif /* NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL */
  3790. -            if (pEntry->Flink == &pNetRootContext->mounts.head)
  3791. -                break;
  3792. -            pEntry = pEntry->Flink;
  3793. -        }
  3794. -
  3795. -        if (found_mount_entry) {
  3796. -            copy_nfs41_mount_config(Config, &found_mount_entry->Config);
  3797. -            DbgP("Found existing mount: LUID=(0x%lx.0x%lx) Entry Config->MntPt='%wZ'\n",
  3798. -                (long)found_mount_entry->login_id.HighPart,
  3799. -                (long)found_mount_entry->login_id.LowPart,
  3800. -                &Config->MntPt);
  3801. -            status = STATUS_SUCCESS;
  3802. -        }
  3803. -#ifdef NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL
  3804. -        else if (found_system_mount_entry) {
  3805. -            copy_nfs41_mount_config(Config, &found_system_mount_entry->Config);
  3806. -            DbgP("Found existing SYSTEM mount: Entry Config->MntPt='%wZ'\n",
  3807. -                &Config->MntPt);
  3808. -            status = STATUS_SUCCESS;
  3809. -        }
  3810. -#endif /* NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL */
  3811. -        ExReleaseFastMutex(&pNetRootContext->mountLock);
  3812. -
  3813. -        if (status != STATUS_SUCCESS) {
  3814. -            DbgP("No existing mount found, "
  3815. -                "status==STATUS_NFS_SHARE_NOT_MOUNTED\n");
  3816. -            goto out_free;
  3817. -        }
  3818. -
  3819. -        pVNetRootContext->read_only = Config->ReadOnly;
  3820. -        pVNetRootContext->write_thru = Config->write_thru;
  3821. -        pVNetRootContext->nocache = Config->nocache;
  3822. -        pVNetRootContext->timebasedcoherency = Config->timebasedcoherency;
  3823. -    }
  3824. -
  3825. -    Config->use_nfspubfh = pubfh_prefix;
  3826. -
  3827. -    DbgP("Config->{ "
  3828. -        "MntPt='%wZ', "
  3829. -        "SrvName='%wZ', "
  3830. -        "use_nfspubfh=%d, "
  3831. -        "ReadOnly=%d, "
  3832. -        "write_thru=%d, "
  3833. -        "nocache=%d "
  3834. -        "timebasedcoherency=%d "
  3835. -        "timeout=%d "
  3836. -        "createmode.use_nfsv3attrsea_mode=%d "
  3837. -        "Config->createmode.mode=0%o "
  3838. -        "}\n",
  3839. -        &Config->MntPt,
  3840. -        &Config->SrvName,
  3841. -        Config->use_nfspubfh?1:0,
  3842. -        Config->ReadOnly?1:0,
  3843. -        Config->write_thru?1:0,
  3844. -        Config->nocache?1:0,
  3845. -        Config->timebasedcoherency?1:0,
  3846. -        Config->timeout,
  3847. -        Config->createmode.use_nfsv3attrsea_mode?1:0,
  3848. -        Config->createmode.mode);
  3849. -
  3850. -    pVNetRootContext->MountPathLen = Config->MntPt.Length;
  3851. -    pVNetRootContext->timeout = Config->timeout;
  3852. -    pVNetRootContext->createmode.use_nfsv3attrsea_mode =
  3853. -        Config->createmode.use_nfsv3attrsea_mode;
  3854. -    pVNetRootContext->createmode.mode =
  3855. -        Config->createmode.mode;
  3856. -
  3857. -    status = map_sec_flavor(&Config->SecFlavor, &pVNetRootContext->sec_flavor);
  3858. -    if (status != STATUS_SUCCESS) {
  3859. -        DbgP("Invalid rpcsec security flavor '%wZ'\n", &Config->SecFlavor);
  3860. -        goto out_free;
  3861. -    }
  3862. -
  3863. -    status = nfs41_GetLUID(&luid);
  3864. -    if (status)
  3865. -        goto out_free;
  3866. -
  3867. -    if (!pNetRootContext->mounts_init) {
  3868. -#ifdef DEBUG_MOUNT
  3869. -        DbgP("Initializing mount array\n");
  3870. -#endif
  3871. -        ExInitializeFastMutex(&pNetRootContext->mountLock);
  3872. -        InitializeListHead(&pNetRootContext->mounts.head);
  3873. -        pNetRootContext->mounts_init = TRUE;
  3874. -    } else {
  3875. -        PLIST_ENTRY pEntry;
  3876. -
  3877. -        ExAcquireFastMutex(&pNetRootContext->mountLock);
  3878. -        pEntry = &pNetRootContext->mounts.head;
  3879. -        pEntry = pEntry->Flink;
  3880. -        while (pEntry != NULL) {
  3881. -            existing_mount = (nfs41_mount_entry *)CONTAINING_RECORD(pEntry,
  3882. -                    nfs41_mount_entry, next);
  3883. -#ifdef DEBUG_MOUNT
  3884. -            DbgP("comparing 0x%lx.0x%lx with 0x%lx.0x%lx\n",
  3885. -                (long)luid.HighPart, (long)luid.LowPart,
  3886. -                (long)existing_mount->login_id.HighPart,
  3887. -                (long)existing_mount->login_id.LowPart);
  3888. -#endif
  3889. -            if (RtlEqualLuid(&luid, &existing_mount->login_id)) {
  3890. -#ifdef DEBUG_MOUNT
  3891. -                DbgP("Found a matching LUID entry\n");
  3892. -#endif
  3893. -                found_existing_mount = TRUE;
  3894. -                switch(pVNetRootContext->sec_flavor) {
  3895. -                case RPCSEC_AUTH_SYS:
  3896. -                    if (existing_mount->authsys_session != INVALID_HANDLE_VALUE)
  3897. -                        pVNetRootContext->session =
  3898. -                            existing_mount->authsys_session;
  3899. -                    break;
  3900. -                case RPCSEC_AUTHGSS_KRB5:
  3901. -                    if (existing_mount->gssi_session != INVALID_HANDLE_VALUE)
  3902. -                        pVNetRootContext->session = existing_mount->gss_session;
  3903. -                    break;
  3904. -                case RPCSEC_AUTHGSS_KRB5I:
  3905. -                    if (existing_mount->gss_session != INVALID_HANDLE_VALUE)
  3906. -                        pVNetRootContext->session = existing_mount->gssi_session;
  3907. -                    break;
  3908. -                case RPCSEC_AUTHGSS_KRB5P:
  3909. -                    if (existing_mount->gssp_session != INVALID_HANDLE_VALUE)
  3910. -                        pVNetRootContext->session = existing_mount->gssp_session;
  3911. -                    break;
  3912. -                }
  3913. -                if (pVNetRootContext->session &&
  3914. -                        pVNetRootContext->session != INVALID_HANDLE_VALUE)
  3915. -                    found_matching_flavor = 1;
  3916. -                break;
  3917. -            }
  3918. -            if (pEntry->Flink == &pNetRootContext->mounts.head)
  3919. -                break;
  3920. -            pEntry = pEntry->Flink;
  3921. -        }
  3922. -        ExReleaseFastMutex(&pNetRootContext->mountLock);
  3923. -#ifdef DEBUG_MOUNT
  3924. -        if (!found_matching_flavor)
  3925. -            DbgP("Didn't find matching security flavor\n");
  3926. -#endif
  3927. -    }
  3928. -
  3929. -    /* send the mount upcall */
  3930. -    status = nfs41_mount(Config, pVNetRootContext->sec_flavor,
  3931. -        &pVNetRootContext->session, &nfs41d_version,
  3932. -        &pVNetRootContext->FsAttrs);
  3933. -    if (status != STATUS_SUCCESS) {
  3934. -        BOOLEAN MountsEmpty;
  3935. -        nfs41_IsListEmpty(pNetRootContext->mountLock,
  3936. -            pNetRootContext->mounts, MountsEmpty);
  3937. -        if (!found_existing_mount && MountsEmpty)
  3938. -            pNetRootContext->mounts_init = FALSE;
  3939. -        pVNetRootContext->session = INVALID_HANDLE_VALUE;
  3940. -        goto out_free;
  3941. -    }
  3942. -    pVNetRootContext->timeout = Config->timeout;
  3943. -
  3944. -    if (!found_existing_mount) {
  3945. -        /* create a new mount entry and add it to the list */
  3946. -        nfs41_mount_entry *entry;
  3947. -        entry = RxAllocatePoolWithTag(NonPagedPoolNx, sizeof(nfs41_mount_entry),
  3948. -            NFS41_MM_POOLTAG_MOUNT);
  3949. -        if (entry == NULL) {
  3950. -            status = STATUS_INSUFFICIENT_RESOURCES;
  3951. -            goto out_free;
  3952. -        }
  3953. -        entry->authsys_session = entry->gss_session =
  3954. -            entry->gssi_session = entry->gssp_session = INVALID_HANDLE_VALUE;
  3955. -        switch (pVNetRootContext->sec_flavor) {
  3956. -        case RPCSEC_AUTH_SYS:
  3957. -            entry->authsys_session = pVNetRootContext->session; break;
  3958. -        case RPCSEC_AUTHGSS_KRB5:
  3959. -            entry->gss_session = pVNetRootContext->session; break;
  3960. -        case RPCSEC_AUTHGSS_KRB5I:
  3961. -            entry->gssi_session = pVNetRootContext->session; break;
  3962. -        case RPCSEC_AUTHGSS_KRB5P:
  3963. -            entry->gssp_session = pVNetRootContext->session; break;
  3964. -        }
  3965. -        RtlCopyLuid(&entry->login_id, &luid);
  3966. -        /*
  3967. -         * Save mount config so we can use it for
  3968. -         * \\server@port\@(pubnfs4|nfs4)\path mounts later
  3969. -         */
  3970. -        copy_nfs41_mount_config(&entry->Config, Config);
  3971. -        nfs41_AddEntry(pNetRootContext->mountLock,
  3972. -            pNetRootContext->mounts, entry);
  3973. -    } else if (!found_matching_flavor) {
  3974. -        ASSERT(existing_mount != NULL);
  3975. -        /* modify existing mount entry */
  3976. -#ifdef DEBUG_MOUNT
  3977. -        DbgP("Using existing %d flavor session 0x%x\n",
  3978. -            pVNetRootContext->sec_flavor);
  3979. -#endif
  3980. -        switch (pVNetRootContext->sec_flavor) {
  3981. -        case RPCSEC_AUTH_SYS:
  3982. -            existing_mount->authsys_session = pVNetRootContext->session; break;
  3983. -        case RPCSEC_AUTHGSS_KRB5:
  3984. -            existing_mount->gss_session = pVNetRootContext->session; break;
  3985. -        case RPCSEC_AUTHGSS_KRB5I:
  3986. -            existing_mount->gssi_session = pVNetRootContext->session; break;
  3987. -        case RPCSEC_AUTHGSS_KRB5P:
  3988. -            existing_mount->gssp_session = pVNetRootContext->session; break;
  3989. -        }
  3990. -    }
  3991. -    pNetRootContext->nfs41d_version = nfs41d_version;
  3992. -#ifdef DEBUG_MOUNT
  3993. -    DbgP("Saving new session 0x%x\n", pVNetRootContext->session);
  3994. -#endif
  3995. -
  3996. -out_free:
  3997. -    RxFreePool(Config);
  3998. -out:
  3999. -    pCreateNetRootContext->VirtualNetRootStatus = status;
  4000. -    if (pNetRoot->Context == NULL)
  4001. -        pCreateNetRootContext->NetRootStatus = status;
  4002. -    pCreateNetRootContext->Callback(pCreateNetRootContext);
  4003. -
  4004. -    /* RDBSS expects that MRxCreateVNetRoot returns STATUS_PENDING
  4005. -     * on success or failure */
  4006. -    status = STATUS_PENDING;
  4007. -#ifdef DEBUG_MOUNT
  4008. -    DbgEx();
  4009. -#endif
  4010. -    return status;
  4011. -}
  4012. -
  4013. -static VOID nfs41_ExtractNetRootName(
  4014. -    IN PUNICODE_STRING FilePathName,
  4015. -    IN PMRX_SRV_CALL SrvCall,
  4016. -    OUT PUNICODE_STRING NetRootName,
  4017. -    OUT PUNICODE_STRING RestOfName OPTIONAL)
  4018. -{
  4019. -    ULONG length = FilePathName->Length;
  4020. -    PWCH w = FilePathName->Buffer;
  4021. -    PWCH wlimit = (PWCH)(((PCHAR)w)+length);
  4022. -    PWCH wlow;
  4023. -
  4024. -    w += (SrvCall->pSrvCallName->Length/sizeof(WCHAR));
  4025. -    NetRootName->Buffer = wlow = w;
  4026. -    /* parse the entire path into NetRootName */
  4027. -#if USE_ENTIRE_PATH_FOR_NETROOT
  4028. -    w = wlimit;
  4029. -#else
  4030. -    for (;;) {
  4031. -        if (w >= wlimit)
  4032. -            break;
  4033. -        if ((*w == OBJ_NAME_PATH_SEPARATOR) && (w != wlow))
  4034. -            break;
  4035. -        w++;
  4036. -    }
  4037. -#endif
  4038. -    NetRootName->Length = NetRootName->MaximumLength
  4039. -                = (USHORT)((PCHAR)w - (PCHAR)wlow);
  4040. -#ifdef DEBUG_MOUNT
  4041. -    DbgP("nfs41_ExtractNetRootName: "
  4042. -        "In: pSrvCall 0x%p PathName='%wZ' SrvCallName='%wZ' "
  4043. -        "Out: NetRootName='%wZ'\n",
  4044. -        SrvCall, FilePathName, SrvCall->pSrvCallName, NetRootName);
  4045. -#endif
  4046. -    return;
  4047. -
  4048. -}
  4049. -
  4050. -static NTSTATUS nfs41_FinalizeSrvCall(
  4051. -    PMRX_SRV_CALL pSrvCall,
  4052. -    BOOLEAN Force)
  4053. -{
  4054. -    NTSTATUS status = STATUS_SUCCESS;
  4055. -    PNFS41_SERVER_ENTRY pServerEntry = (PNFS41_SERVER_ENTRY)(pSrvCall->Context);
  4056. -
  4057. -#ifdef DEBUG_MOUNT
  4058. -    DbgEn();
  4059. -#endif
  4060. -    // print_srv_call(pSrvCall);
  4061. -
  4062. -    if (pSrvCall->Context == NULL)
  4063. -        goto out;
  4064. -
  4065. -    InterlockedCompareExchangePointer(&pServerEntry->pRdbssSrvCall,
  4066. -        NULL, pSrvCall);
  4067. -    RxFreePool(pServerEntry);
  4068. -
  4069. -    pSrvCall->Context = NULL;
  4070. -out:
  4071. -#ifdef DEBUG_MOUNT
  4072. -    DbgEx();
  4073. -#endif
  4074. -    return status;
  4075. -}
  4076. -
  4077. -static NTSTATUS nfs41_FinalizeNetRoot(
  4078. -    IN OUT PMRX_NET_ROOT pNetRoot,
  4079. -    IN PBOOLEAN ForceDisconnect)
  4080. -{
  4081. -    NTSTATUS status = STATUS_SUCCESS;
  4082. -    PNFS41_NETROOT_EXTENSION pNetRootContext =
  4083. -        NFS41GetNetRootExtension((PMRX_NET_ROOT)pNetRoot);
  4084. -    nfs41_updowncall_entry *tmp;
  4085. -    nfs41_mount_entry *mount_tmp;
  4086. -
  4087. -#ifdef DEBUG_MOUNT
  4088. -    DbgEn();
  4089. -    print_net_root(pNetRoot);
  4090. -#endif
  4091. -
  4092. -    if (pNetRoot->Type != NET_ROOT_DISK && pNetRoot->Type != NET_ROOT_WILD) {
  4093. -        status = STATUS_NOT_SUPPORTED;
  4094. -        goto out;
  4095. -    }
  4096. -
  4097. -    if (pNetRootContext == NULL || !pNetRootContext->mounts_init) {
  4098. -        print_error("nfs41_FinalizeNetRoot: No valid session established\n");
  4099. -        goto out;
  4100. -    }
  4101. -
  4102. -    if (pNetRoot->NumberOfFcbs > 0 || pNetRoot->NumberOfSrvOpens > 0) {
  4103. -        print_error("%d open Fcbs %d open SrvOpens\n", pNetRoot->NumberOfFcbs,
  4104. -            pNetRoot->NumberOfSrvOpens);
  4105. -        goto out;
  4106. -    }
  4107. -
  4108. -    do {
  4109. -        nfs41_GetFirstMountEntry(pNetRootContext->mountLock,
  4110. -            pNetRootContext->mounts, mount_tmp);
  4111. -        if (mount_tmp == NULL)
  4112. -            break;
  4113. -#ifdef DEBUG_MOUNT
  4114. -        DbgP("Removing entry luid 0x%lx.0x%lx from mount list\n",
  4115. -            (long)mount_tmp->login_id.HighPart,
  4116. -            (long)mount_tmp->login_id.LowPart);
  4117. -#endif
  4118. -        if (mount_tmp->authsys_session != INVALID_HANDLE_VALUE) {
  4119. -            status = nfs41_unmount(mount_tmp->authsys_session,
  4120. -                pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT);
  4121. -            if (status)
  4122. -                print_error("nfs41_unmount AUTH_SYS failed with %d\n", status);
  4123. -        }
  4124. -        if (mount_tmp->gss_session != INVALID_HANDLE_VALUE) {
  4125. -            status = nfs41_unmount(mount_tmp->gss_session,
  4126. -                pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT);
  4127. -            if (status)
  4128. -                print_error("nfs41_unmount RPCSEC_GSS_KRB5 failed with %d\n",
  4129. -                            status);
  4130. -        }
  4131. -        if (mount_tmp->gssi_session != INVALID_HANDLE_VALUE) {
  4132. -            status = nfs41_unmount(mount_tmp->gssi_session,
  4133. -                pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT);
  4134. -            if (status)
  4135. -                print_error("nfs41_unmount RPCSEC_GSS_KRB5I failed with %d\n",
  4136. -                            status);
  4137. -        }
  4138. -        if (mount_tmp->gssp_session != INVALID_HANDLE_VALUE) {
  4139. -            status = nfs41_unmount(mount_tmp->gssp_session,
  4140. -                pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT);
  4141. -            if (status)
  4142. -                print_error("nfs41_unmount RPCSEC_GSS_KRB5P failed with %d\n",
  4143. -                            status);
  4144. -        }
  4145. -        nfs41_RemoveEntry(pNetRootContext->mountLock, mount_tmp);
  4146. -        RxFreePool(mount_tmp);
  4147. -        mount_tmp = NULL;
  4148. -    } while (1);
  4149. -    /* ignore any errors from unmount */
  4150. -    status = STATUS_SUCCESS;
  4151. -
  4152. -    // check if there is anything waiting in the upcall or downcall queue
  4153. -    do {
  4154. -        nfs41_GetFirstEntry(upcallLock, upcall, tmp);
  4155. -        if (tmp != NULL) {
  4156. -            DbgP("Removing entry from upcall list\n");
  4157. -            nfs41_RemoveEntry(upcallLock, tmp);
  4158. -            tmp->status = STATUS_INSUFFICIENT_RESOURCES;
  4159. -            KeSetEvent(&tmp->cond, 0, FALSE);
  4160. -        } else
  4161. -            break;
  4162. -    } while (1);
  4163. -
  4164. -    do {
  4165. -        nfs41_GetFirstEntry(downcallLock, downcall, tmp);
  4166. -        if (tmp != NULL) {
  4167. -            DbgP("Removing entry from downcall list\n");
  4168. -            nfs41_RemoveEntry(downcallLock, tmp);
  4169. -            tmp->status = STATUS_INSUFFICIENT_RESOURCES;
  4170. -            KeSetEvent(&tmp->cond, 0, FALSE);
  4171. -        } else
  4172. -            break;
  4173. -    } while (1);
  4174. -out:
  4175. -#ifdef DEBUG_MOUNT
  4176. -    DbgEx();
  4177. -#endif
  4178. -    return status;
  4179. -}
  4180. -
  4181. -
  4182. -static NTSTATUS nfs41_FinalizeVNetRoot(
  4183. -    IN OUT PMRX_V_NET_ROOT pVNetRoot,
  4184. -    IN PBOOLEAN ForceDisconnect)
  4185. -{
  4186. -    NTSTATUS status = STATUS_SUCCESS;
  4187. -#ifdef DEBUG_MOUNT
  4188. -    DbgEn();
  4189. -    print_v_net_root(pVNetRoot);
  4190. -#endif
  4191. -    if (pVNetRoot->pNetRoot->Type != NET_ROOT_DISK &&
  4192. -            pVNetRoot->pNetRoot->Type != NET_ROOT_WILD)
  4193. -        status = STATUS_NOT_SUPPORTED;
  4194. -#ifdef DEBUG_MOUNT
  4195. -    DbgEx();
  4196. -#endif
  4197. -    return status;
  4198. -}
  4199. -
  4200. -static BOOLEAN isDataAccess(
  4201. -    ACCESS_MASK mask)
  4202. -{
  4203. -    if (mask & (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA))
  4204. -        return TRUE;
  4205. -    return FALSE;
  4206. -}
  4207. -
  4208. -static BOOLEAN isOpen2Create(
  4209. -    ULONG disposition)
  4210. -{
  4211. -    if (disposition == FILE_CREATE || disposition == FILE_OPEN_IF ||
  4212. -            disposition == FILE_OVERWRITE_IF || disposition == FILE_SUPERSEDE)
  4213. -        return TRUE;
  4214. -    return FALSE;
  4215. -}
  4216. -
  4217. -static BOOLEAN isFilenameTooLong(
  4218. -    PUNICODE_STRING name,
  4219. -    PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext)
  4220. -{
  4221. -    PFILE_FS_ATTRIBUTE_INFORMATION attrs = &pVNetRootContext->FsAttrs;
  4222. -    LONG len = attrs->MaximumComponentNameLength, count = 1, i;
  4223. -    PWCH p = name->Buffer;
  4224. -    for (i = 0; i < name->Length / 2; i++) {
  4225. -        if (p[0] == L'\\') count = 1;
  4226. -        else {
  4227. -            if (p[0] == L'\0') return FALSE;
  4228. -            if (count > len) return TRUE;
  4229. -            count++;
  4230. -        }
  4231. -        p++;
  4232. -    }
  4233. -    return FALSE;
  4234. -}
  4235. -
  4236. -static BOOLEAN isStream(
  4237. -    PUNICODE_STRING name)
  4238. -{
  4239. -    LONG i;
  4240. -    PWCH p = name->Buffer;
  4241. -    for (i = 0; i < name->Length / 2; i++) {
  4242. -        if (p[0] == L':') return TRUE;
  4243. -        else if (p[0] == L'\0') return FALSE;
  4244. -        p++;
  4245. -    }
  4246. -    return FALSE;
  4247. -}
  4248. -
  4249. -static BOOLEAN areOpenParamsValid(NT_CREATE_PARAMETERS *params)
  4250. -{
  4251. -    /* from ms-fsa page 52 */
  4252. -    if ((params->CreateOptions & FILE_DELETE_ON_CLOSE) &&
  4253. -            !(params->DesiredAccess & DELETE))
  4254. -        return FALSE;
  4255. -    if ((params->CreateOptions & FILE_DIRECTORY_FILE) &&
  4256. -            (params->Disposition == FILE_SUPERSEDE ||
  4257. -                params->Disposition == FILE_OVERWRITE ||
  4258. -                params->Disposition == FILE_OVERWRITE_IF))
  4259. -        return FALSE;
  4260. -    if ((params->CreateOptions & FILE_NO_INTERMEDIATE_BUFFERING) &&
  4261. -            (params->DesiredAccess & FILE_APPEND_DATA) &&
  4262. -            !(params->DesiredAccess & FILE_WRITE_DATA))
  4263. -        return FALSE;
  4264. -    /* from ms-fsa 3.1.5.1.1 page 56 */
  4265. -    if ((params->CreateOptions & FILE_DIRECTORY_FILE) &&
  4266. -            (params->FileAttributes & FILE_ATTRIBUTE_TEMPORARY))
  4267. -        return FALSE;
  4268. -    return TRUE;
  4269. -}
  4270. -
  4271. -static NTSTATUS map_open_errors(
  4272. -    DWORD status,
  4273. -    USHORT len)
  4274. -{
  4275. -    switch (status) {
  4276. -    case NO_ERROR:                      return STATUS_SUCCESS;
  4277. -    case ERROR_ACCESS_DENIED:
  4278. -        if (len > 0)                    return STATUS_ACCESS_DENIED;
  4279. -        else                            return STATUS_SUCCESS;
  4280. -    case ERROR_INVALID_REPARSE_DATA:
  4281. -    case ERROR_INVALID_NAME:            return STATUS_OBJECT_NAME_INVALID;
  4282. -    case ERROR_FILE_EXISTS:             return STATUS_OBJECT_NAME_COLLISION;
  4283. -    case ERROR_FILE_INVALID:            return STATUS_FILE_INVALID;
  4284. -    case ERROR_FILE_NOT_FOUND:          return STATUS_OBJECT_NAME_NOT_FOUND;
  4285. -    case ERROR_FILENAME_EXCED_RANGE:    return STATUS_NAME_TOO_LONG;
  4286. -    case ERROR_NETWORK_ACCESS_DENIED:   return STATUS_NETWORK_ACCESS_DENIED;
  4287. -    case ERROR_PATH_NOT_FOUND:          return STATUS_OBJECT_PATH_NOT_FOUND;
  4288. -    case ERROR_BAD_NETPATH:             return STATUS_BAD_NETWORK_PATH;
  4289. -    case ERROR_SHARING_VIOLATION:       return STATUS_SHARING_VIOLATION;
  4290. -    case ERROR_REPARSE:                 return STATUS_REPARSE;
  4291. -    case ERROR_TOO_MANY_LINKS:          return STATUS_TOO_MANY_LINKS;
  4292. -    case ERROR_DIRECTORY:               return STATUS_FILE_IS_A_DIRECTORY;
  4293. -    case ERROR_BAD_FILE_TYPE:           return STATUS_NOT_A_DIRECTORY;
  4294. -    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  4295. -    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  4296. -    case ERROR_FILE_TOO_LARGE:          return STATUS_FILE_TOO_LARGE;
  4297. -    case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  4298. -    default:
  4299. -        print_error("[ERROR] nfs41_Create: upcall returned ERROR_0x%x "
  4300. -            "returning STATUS_INSUFFICIENT_RESOURCES\n", status);
  4301. -    case ERROR_OUTOFMEMORY:             return STATUS_INSUFFICIENT_RESOURCES;
  4302. -    }
  4303. -}
  4304. -
  4305. -static DWORD map_disposition_to_create_retval(
  4306. -    DWORD disposition,
  4307. -    DWORD errno)
  4308. -{
  4309. -    switch(disposition) {
  4310. -    case FILE_SUPERSEDE:
  4311. -        if (errno == ERROR_FILE_NOT_FOUND)  return FILE_CREATED;
  4312. -        else                                return FILE_SUPERSEDED;
  4313. -    case FILE_CREATE:                       return FILE_CREATED;
  4314. -    case FILE_OPEN:                         return FILE_OPENED;
  4315. -    case FILE_OPEN_IF:
  4316. -        if (errno == ERROR_FILE_NOT_FOUND)  return FILE_CREATED;
  4317. -        else                                return FILE_OPENED;
  4318. -    case FILE_OVERWRITE:                    return FILE_OVERWRITTEN;
  4319. -    case FILE_OVERWRITE_IF:
  4320. -        if (errno == ERROR_FILE_NOT_FOUND)  return FILE_CREATED;
  4321. -        else                                return FILE_OVERWRITTEN;
  4322. -    default:
  4323. -        print_error("unknown disposition %d\n", disposition);
  4324. -        return FILE_OPENED;
  4325. -    }
  4326. -}
  4327. -
  4328. -static BOOLEAN create_should_pass_ea(
  4329. -    IN PFILE_FULL_EA_INFORMATION ea,
  4330. -    IN ULONG disposition)
  4331. -{
  4332. -    /* don't pass cygwin EAs */
  4333. -    if (AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength)
  4334. -        || AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength)
  4335. -        || AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength))
  4336. -        return FALSE;
  4337. -    /* only set EAs on file creation */
  4338. -    return disposition == FILE_SUPERSEDE || disposition == FILE_CREATE
  4339. -        || disposition == FILE_OPEN_IF || disposition == FILE_OVERWRITE
  4340. -        || disposition == FILE_OVERWRITE_IF;
  4341. -}
  4342. -
  4343. -static NTSTATUS check_nfs41_create_args(
  4344. -    IN PRX_CONTEXT RxContext)
  4345. -{
  4346. -    NTSTATUS status = STATUS_SUCCESS;
  4347. -    PNT_CREATE_PARAMETERS params = &RxContext->Create.NtCreateParameters;
  4348. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  4349. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  4350. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  4351. -    __notnull PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs =
  4352. -        &pVNetRootContext->FsAttrs;
  4353. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  4354. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  4355. -    __notnull PMRX_FCB Fcb = RxContext->pFcb;
  4356. -    __notnull PNFS41_FCB nfs41_fcb = (PNFS41_FCB)Fcb->Context;
  4357. -    PFILE_FULL_EA_INFORMATION ea = (PFILE_FULL_EA_INFORMATION)
  4358. -        RxContext->CurrentIrp->AssociatedIrp.SystemBuffer;
  4359. -
  4360. -    if (Fcb->pNetRoot->Type != NET_ROOT_DISK &&
  4361. -            Fcb->pNetRoot->Type != NET_ROOT_WILD) {
  4362. -        print_error("nfs41_Create: Unsupported NetRoot Type %u\n",
  4363. -            Fcb->pNetRoot->Type);
  4364. -        status = STATUS_NOT_SUPPORTED;
  4365. -        goto out;
  4366. -    }
  4367. -
  4368. -    if (FlagOn(Fcb->FcbState, FCB_STATE_PAGING_FILE )) {
  4369. -        print_error("FCB_STATE_PAGING_FILE not implemented\n");
  4370. -        status = STATUS_NOT_IMPLEMENTED;
  4371. -        goto out;
  4372. -    }
  4373. -    
  4374. -    if (!pNetRootContext->mounts_init) {
  4375. -        print_error("nfs41_Create: No valid session established\n");
  4376. -        status = STATUS_INSUFFICIENT_RESOURCES;
  4377. -        goto out;
  4378. -    }
  4379. -
  4380. -    if (isStream(SrvOpen->pAlreadyPrefixedName)) {
  4381. -        status = STATUS_NOT_SUPPORTED;
  4382. -        goto out;
  4383. -    }
  4384. -
  4385. -    if (pVNetRootContext->read_only &&
  4386. -            (params->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA))) {
  4387. -        status = STATUS_MEDIA_WRITE_PROTECTED;
  4388. -        goto out;
  4389. -    }
  4390. -
  4391. -    /* if FCB was marked for deletion and opened multiple times, as soon
  4392. -     * as first close happen, FCB transitions into delete_pending state
  4393. -     * no more opens allowed
  4394. -     */
  4395. -    if (Fcb->OpenCount && nfs41_fcb->DeletePending) {
  4396. -        status = STATUS_DELETE_PENDING;
  4397. -        goto out;
  4398. -    }
  4399. -
  4400. -    /* ms-fsa: 3.1.5.1.2.1 page 68 */
  4401. -    if (Fcb->OpenCount && nfs41_fcb->StandardInfo.DeletePending &&
  4402. -            !(params->ShareAccess & FILE_SHARE_DELETE) &&
  4403. -                (params->DesiredAccess & (FILE_EXECUTE | FILE_READ_DATA |
  4404. -                    FILE_WRITE_DATA | FILE_APPEND_DATA))) {
  4405. -        status = STATUS_SHARING_VIOLATION;
  4406. -        goto out;
  4407. -    }
  4408. -
  4409. -    /* rdbss seems miss this sharing_violation check */
  4410. -    if (Fcb->OpenCount && params->Disposition == FILE_SUPERSEDE) {
  4411. -        if ((!RxContext->CurrentIrpSp->FileObject->SharedRead &&
  4412. -                (params->DesiredAccess & FILE_READ_DATA)) ||
  4413. -            ((!RxContext->CurrentIrpSp->FileObject->SharedWrite &&
  4414. -                (params->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA |
  4415. -                    FILE_WRITE_ATTRIBUTES))) ||
  4416. -            (!RxContext->CurrentIrpSp->FileObject->SharedDelete &&
  4417. -                (params->DesiredAccess & DELETE)))) {
  4418. -            status = STATUS_SHARING_VIOLATION;
  4419. -            goto out;
  4420. -        }
  4421. -    }
  4422. -    if (isFilenameTooLong(SrvOpen->pAlreadyPrefixedName, pVNetRootContext)) {
  4423. -        status = STATUS_OBJECT_NAME_INVALID;
  4424. -        goto out;
  4425. -    }
  4426. -
  4427. -    /* We do not support oplocks (yet) */
  4428. -    if (params->CreateOptions & FILE_OPEN_REQUIRING_OPLOCK) {
  4429. -        status = STATUS_INVALID_PARAMETER;
  4430. -        goto out;
  4431. -    }
  4432. -
  4433. -    if (!areOpenParamsValid(params)) {
  4434. -        status = STATUS_INVALID_PARAMETER;
  4435. -        goto out;
  4436. -    }
  4437. -
  4438. -    /* from ms-fsa 3.1.5.1.1 page 56 */
  4439. -    if ((params->CreateOptions & FILE_DELETE_ON_CLOSE) &&
  4440. -            (params->FileAttributes & FILE_ATTRIBUTE_READONLY)) {
  4441. -        status = STATUS_CANNOT_DELETE;
  4442. -        goto out;
  4443. -    }
  4444. -
  4445. -    if (ea) {
  4446. -        /* ignore cygwin EAs when checking support and access */
  4447. -        if (!AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength) &&
  4448. -            !AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength) &&
  4449. -            !AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) {
  4450. -            if (!(FsAttrs->FileSystemAttributes & FILE_SUPPORTS_EXTENDED_ATTRIBUTES)) {
  4451. -                status = STATUS_EAS_NOT_SUPPORTED;
  4452. -                goto out;
  4453. -            }
  4454. -        }
  4455. -    } else if (RxContext->CurrentIrpSp->Parameters.Create.EaLength) {
  4456. -        status = STATUS_INVALID_PARAMETER;
  4457. -        goto out;
  4458. -    }
  4459. -
  4460. -out:
  4461. -    return status;
  4462. -}
  4463. -
  4464. -static NTSTATUS nfs41_Create(
  4465. -    IN OUT PRX_CONTEXT RxContext)
  4466. -{
  4467. -    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  4468. -    nfs41_updowncall_entry *entry = NULL;
  4469. -    PNT_CREATE_PARAMETERS params = &RxContext->Create.NtCreateParameters;
  4470. -    PFILE_FULL_EA_INFORMATION ea = (PFILE_FULL_EA_INFORMATION)
  4471. -        RxContext->CurrentIrp->AssociatedIrp.SystemBuffer;
  4472. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  4473. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  4474. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  4475. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  4476. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  4477. -    __notnull PMRX_FCB Fcb = RxContext->pFcb;
  4478. -    __notnull PNFS41_FCB nfs41_fcb = (PNFS41_FCB)Fcb->Context;
  4479. -    PNFS41_FOBX nfs41_fobx = NULL;
  4480. -    BOOLEAN oldDeletePending = nfs41_fcb->StandardInfo.DeletePending;
  4481. -#ifdef ENABLE_TIMINGS
  4482. -    LARGE_INTEGER t1, t2;
  4483. -    t1 = KeQueryPerformanceCounter(NULL);
  4484. -#endif
  4485. -
  4486. -    ASSERT( NodeType(SrvOpen) == RDBSS_NTC_SRVOPEN );
  4487. -
  4488. -#ifdef DEBUG_OPEN
  4489. -    DbgEn();
  4490. -    print_debug_header(RxContext);
  4491. -    print_nt_create_params(1, RxContext->Create.NtCreateParameters);
  4492. -    // if (ea) print_ea_info(ea);
  4493. -#endif
  4494. -
  4495. -    status = check_nfs41_create_args(RxContext);
  4496. -    if (status) goto out;
  4497. -
  4498. -    status = nfs41_UpcallCreate(NFS41_OPEN, NULL,
  4499. -        pVNetRootContext->session, INVALID_HANDLE_VALUE,
  4500. -        pNetRootContext->nfs41d_version,
  4501. -        SrvOpen->pAlreadyPrefixedName, &entry);
  4502. -    if (status) goto out;
  4503. -
  4504. -    entry->u.Open.access_mask = params->DesiredAccess;
  4505. -    entry->u.Open.access_mode = params->ShareAccess;
  4506. -    entry->u.Open.attrs = params->FileAttributes;
  4507. -    if (!(params->CreateOptions & FILE_DIRECTORY_FILE))
  4508. -        entry->u.Open.attrs |= FILE_ATTRIBUTE_ARCHIVE;
  4509. -    entry->u.Open.disp = params->Disposition;
  4510. -    entry->u.Open.copts = params->CreateOptions;
  4511. -    entry->u.Open.srv_open = SrvOpen;
  4512. -    /* treat the NfsActOnLink ea as FILE_OPEN_REPARSE_POINT */
  4513. -    if ((ea && AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength)) ||
  4514. -            (entry->u.Open.access_mask & DELETE))
  4515. -        entry->u.Open.copts |= FILE_OPEN_REPARSE_POINT;
  4516. -    if (isDataAccess(params->DesiredAccess) || isOpen2Create(params->Disposition))
  4517. -        entry->u.Open.open_owner_id = InterlockedIncrement(&open_owner_id);
  4518. -    // if we are creating a file check if nfsv3attributes were passed in
  4519. -    if (params->Disposition != FILE_OPEN && params->Disposition != FILE_OVERWRITE) {
  4520. -        /* Get default mode */
  4521. -        entry->u.Open.mode = pVNetRootContext->createmode.mode;
  4522. -
  4523. -        /* Use mode from NfsV3Attributes */
  4524. -        if (pVNetRootContext->createmode.use_nfsv3attrsea_mode &&
  4525. -            ea && AnsiStrEq(&NfsV3Attributes,
  4526. -            ea->EaName, ea->EaNameLength)) {
  4527. -            nfs3_attrs *attrs =
  4528. -                (nfs3_attrs *)(ea->EaName + ea->EaNameLength + 1);
  4529. -
  4530. -            entry->u.Open.mode = attrs->mode;
  4531. -#ifdef DEBUG_OPEN
  4532. -            DbgP("creating file with EA mode 0%o\n",
  4533. -                entry->u.Open.mode);
  4534. -#endif
  4535. -        }
  4536. -        else {
  4537. -#ifdef DEBUG_OPEN
  4538. -            DbgP("creating file with default mode 0%o\n",
  4539. -                entry->u.Open.mode);
  4540. -#endif
  4541. -        }
  4542. -
  4543. -        if (params->FileAttributes & FILE_ATTRIBUTE_READONLY) {
  4544. -            entry->u.Open.mode &= ~0222;
  4545. -            DbgP("FILE_ATTRIBUTE_READONLY set, using mode 0%o\n",
  4546. -                entry->u.Open.mode);
  4547. -        }
  4548. -    }
  4549. -    if (entry->u.Open.disp == FILE_CREATE && ea &&
  4550. -            AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) {
  4551. -        /* for a cygwin symlink, given as a unicode string */
  4552. -        entry->u.Open.symlink.Buffer = (PWCH)(ea->EaName + ea->EaNameLength + 1);
  4553. -        entry->u.Open.symlink.MaximumLength = entry->u.Open.symlink.Length = ea->EaValueLength;
  4554. -    }
  4555. -retry_on_link:
  4556. -    if (ea && create_should_pass_ea(ea, params->Disposition)) {
  4557. -        /* lock the extended attribute buffer for read access in user space */
  4558. -        entry->u.Open.EaMdl = IoAllocateMdl(ea,
  4559. -            RxContext->CurrentIrpSp->Parameters.Create.EaLength,
  4560. -            FALSE, FALSE, NULL);
  4561. -        if (entry->u.Open.EaMdl == NULL) {
  4562. -            status = STATUS_INTERNAL_ERROR;
  4563. -            nfs41_UpcallDestroy(entry);
  4564. -            entry = NULL;
  4565. -            goto out;
  4566. -        }
  4567. -#pragma warning( push )
  4568. -/*
  4569. - * C28145: "The opaque MDL structure should not be modified by a
  4570. - * driver.", |MDL_MAPPING_CAN_FAIL| is the exception
  4571. - */
  4572. -#pragma warning (disable : 28145)
  4573. -        entry->u.Open.EaMdl->MdlFlags |= MDL_MAPPING_CAN_FAIL;
  4574. -#pragma warning( pop )
  4575. -        MmProbeAndLockPages(entry->u.Open.EaMdl, KernelMode, IoModifyAccess);
  4576. -    }
  4577. -
  4578. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  4579. -    if (entry->psec_ctx == &entry->sec_ctx) {
  4580. -        SeDeleteClientSecurity(entry->psec_ctx);
  4581. -    }
  4582. -    entry->psec_ctx = NULL;
  4583. -    if (status) goto out;
  4584. -
  4585. -    if (entry->u.Open.EaMdl) {
  4586. -        MmUnlockPages(entry->u.Open.EaMdl);
  4587. -        IoFreeMdl(entry->u.Open.EaMdl);
  4588. -    }
  4589. -
  4590. -    if (entry->status == NO_ERROR && entry->errno == ERROR_REPARSE) {
  4591. -        /* symbolic link handling. when attempting to open a symlink when the
  4592. -         * FILE_OPEN_REPARSE_POINT flag is not set, replace the filename with
  4593. -         * the symlink target's by calling RxPrepareToReparseSymbolicLink()
  4594. -         * and returning STATUS_REPARSE. the object manager will attempt to
  4595. -         * open the new path, and return its handle for the original open */
  4596. -        PRDBSS_DEVICE_OBJECT DeviceObject = RxContext->RxDeviceObject;
  4597. -        PV_NET_ROOT VNetRoot = (PV_NET_ROOT)
  4598. -            RxContext->pRelevantSrvOpen->pVNetRoot;
  4599. -        PUNICODE_STRING VNetRootPrefix = &VNetRoot->PrefixEntry.Prefix;
  4600. -        UNICODE_STRING AbsPath;
  4601. -        PCHAR buf;
  4602. -        BOOLEAN ReparseRequired;
  4603. -
  4604. -        /* allocate the string for RxPrepareToReparseSymbolicLink(), and
  4605. -         * format an absolute path "DeviceName+VNetRootName+symlink" */
  4606. -        AbsPath.Length = DeviceObject->DeviceName.Length +
  4607. -            VNetRootPrefix->Length + entry->u.Open.symlink.Length;
  4608. -        AbsPath.MaximumLength = AbsPath.Length + sizeof(UNICODE_NULL);
  4609. -        AbsPath.Buffer = RxAllocatePoolWithTag(NonPagedPoolNx,
  4610. -            AbsPath.MaximumLength, NFS41_MM_POOLTAG);
  4611. -        if (AbsPath.Buffer == NULL) {
  4612. -            status = STATUS_INSUFFICIENT_RESOURCES;
  4613. -            goto out_free;
  4614. -        }
  4615. -
  4616. -        buf = (PCHAR)AbsPath.Buffer;
  4617. -        RtlCopyMemory(buf, DeviceObject->DeviceName.Buffer,
  4618. -            DeviceObject->DeviceName.Length);
  4619. -        buf += DeviceObject->DeviceName.Length;
  4620. -        RtlCopyMemory(buf, VNetRootPrefix->Buffer, VNetRootPrefix->Length);
  4621. -        buf += VNetRootPrefix->Length;
  4622. -        RtlCopyMemory(buf, entry->u.Open.symlink.Buffer,
  4623. -            entry->u.Open.symlink.Length);
  4624. -        RxFreePool(entry->u.Open.symlink.Buffer);
  4625. -        entry->u.Open.symlink.Buffer = NULL;
  4626. -        buf += entry->u.Open.symlink.Length;
  4627. -        *(PWCHAR)buf = UNICODE_NULL;
  4628. -
  4629. -        status = RxPrepareToReparseSymbolicLink(RxContext,
  4630. -            entry->u.Open.symlink_embedded, &AbsPath, TRUE, &ReparseRequired);
  4631. -#ifdef DEBUG_OPEN
  4632. -        DbgP("RxPrepareToReparseSymbolicLink(%u, '%wZ') returned 0x%08lX, "
  4633. -            "FileName is '%wZ'\n", entry->u.Open.symlink_embedded,
  4634. -            &AbsPath, status, &RxContext->CurrentIrpSp->FileObject->FileName);
  4635. -#endif
  4636. -        if (status == STATUS_SUCCESS) {
  4637. -            /* if a reparse is not required, reopen the link itself.  this
  4638. -             * happens with operations on cygwin symlinks, where the reparse
  4639. -             * flag is not set */
  4640. -            if (!ReparseRequired) {
  4641. -                entry->u.Open.symlink.Length = 0;
  4642. -                entry->u.Open.copts |= FILE_OPEN_REPARSE_POINT;
  4643. -                goto retry_on_link;
  4644. -            }
  4645. -            status = STATUS_REPARSE;
  4646. -        }
  4647. -        goto out_free;
  4648. -    }
  4649. -
  4650. -    status = map_open_errors(entry->status,
  4651. -                SrvOpen->pAlreadyPrefixedName->Length);
  4652. -    if (status) {
  4653. -#ifdef DEBUG_OPEN
  4654. -        print_open_error(1, status);
  4655. -#endif
  4656. -        goto out_free;
  4657. -    }
  4658. -
  4659. -    if (!RxIsFcbAcquiredExclusive(Fcb)) {
  4660. -        ASSERT(!RxIsFcbAcquiredShared(Fcb));
  4661. -        RxAcquireExclusiveFcbResourceInMRx(Fcb);
  4662. -    }
  4663. -
  4664. -    RxContext->pFobx = RxCreateNetFobx(RxContext, SrvOpen);
  4665. -    if (RxContext->pFobx == NULL) {
  4666. -        status = STATUS_INSUFFICIENT_RESOURCES;
  4667. -        goto out_free;
  4668. -    }
  4669. -#ifdef DEBUG_OPEN
  4670. -    DbgP("nfs41_Create: created FOBX 0x%p\n", RxContext->pFobx);
  4671. -#endif
  4672. -    nfs41_fobx = (PNFS41_FOBX)(RxContext->pFobx)->Context;
  4673. -    nfs41_fobx->nfs41_open_state = entry->open_state;
  4674. -    if (nfs41_fobx->sec_ctx.ClientToken == NULL) {
  4675. -        status = nfs41_get_sec_ctx(SecurityImpersonation, &nfs41_fobx->sec_ctx);
  4676. -        if (status)
  4677. -            goto out_free;
  4678. -    }
  4679. -
  4680. -    // we get attributes only for data access and file (not directories)
  4681. -    if (Fcb->OpenCount == 0 ||
  4682. -            (Fcb->OpenCount > 0 &&
  4683. -                nfs41_fcb->changeattr != entry->ChangeTime)) {
  4684. -        FCB_INIT_PACKET InitPacket;
  4685. -        RX_FILE_TYPE StorageType = FileTypeNotYetKnown;
  4686. -        RtlCopyMemory(&nfs41_fcb->BasicInfo, &entry->u.Open.binfo,
  4687. -            sizeof(entry->u.Open.binfo));
  4688. -        RtlCopyMemory(&nfs41_fcb->StandardInfo, &entry->u.Open.sinfo,
  4689. -            sizeof(entry->u.Open.sinfo));
  4690. -        nfs41_fcb->mode = entry->u.Open.mode;
  4691. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  4692. -        nfs41_fcb->owner_local_uid = entry->u.Open.owner_local_uid;
  4693. -        nfs41_fcb->owner_group_local_gid = entry->u.Open.owner_group_local_gid;
  4694. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  4695. -        nfs41_fcb->changeattr = entry->ChangeTime;
  4696. -        if (((params->CreateOptions & FILE_DELETE_ON_CLOSE) &&
  4697. -                !pVNetRootContext->read_only) || oldDeletePending)
  4698. -            nfs41_fcb->StandardInfo.DeletePending = TRUE;
  4699. -
  4700. -        RxFormInitPacket(InitPacket,
  4701. -            &entry->u.Open.binfo.FileAttributes,
  4702. -            &entry->u.Open.sinfo.NumberOfLinks,
  4703. -            &entry->u.Open.binfo.CreationTime,
  4704. -            &entry->u.Open.binfo.LastAccessTime,
  4705. -            &entry->u.Open.binfo.LastWriteTime,
  4706. -            &entry->u.Open.binfo.ChangeTime,
  4707. -            &entry->u.Open.sinfo.AllocationSize,
  4708. -            &entry->u.Open.sinfo.EndOfFile,
  4709. -            &entry->u.Open.sinfo.EndOfFile);
  4710. -
  4711. -        if (entry->u.Open.sinfo.Directory)
  4712. -            StorageType = FileTypeDirectory;
  4713. -        else
  4714. -            StorageType = FileTypeFile;
  4715. -
  4716. -        RxFinishFcbInitialization(Fcb, RDBSS_STORAGE_NTC(StorageType),
  4717. -                                    &InitPacket);
  4718. -    }
  4719. -#ifdef DEBUG_OPEN
  4720. -    else
  4721. -        DbgP("$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$\n");
  4722. -
  4723. -    print_basic_info(1, &nfs41_fcb->BasicInfo);
  4724. -    print_std_info(1, &nfs41_fcb->StandardInfo);
  4725. -#endif
  4726. -
  4727. -    /* aglo: 05/10/2012. it seems like always have to invalid the cache if the
  4728. -     * file has been opened before and being opened again for data access.
  4729. -     * If the file was opened before, RDBSS might have cached (unflushed) data
  4730. -     * and by opening it again, we will not have the correct representation of
  4731. -     * the file size and data content. fileio tests 208, 219, 221.
  4732. -     */
  4733. -    if (Fcb->OpenCount > 0 && (isDataAccess(params->DesiredAccess) ||
  4734. -            nfs41_fcb->changeattr != entry->ChangeTime) &&
  4735. -                !nfs41_fcb->StandardInfo.Directory) {
  4736. -        ULONG flag = DISABLE_CACHING;
  4737. -#ifdef DEBUG_OPEN
  4738. -        DbgP("nfs41_Create: reopening (changed) file '%wZ'\n",
  4739. -            SrvOpen->pAlreadyPrefixedName);
  4740. -#endif
  4741. -        RxChangeBufferingState((PSRV_OPEN)SrvOpen, ULongToPtr(flag), 1);
  4742. -    }
  4743. -    if (!nfs41_fcb->StandardInfo.Directory &&
  4744. -            isDataAccess(params->DesiredAccess)) {
  4745. -        nfs41_fobx->deleg_type = entry->u.Open.deleg_type;
  4746. -#ifdef DEBUG_OPEN
  4747. -        DbgP("nfs41_Create: received delegation %d\n", entry->u.Open.deleg_type);
  4748. -#endif
  4749. -        if (!(params->CreateOptions & FILE_WRITE_THROUGH) &&
  4750. -                !pVNetRootContext->write_thru &&
  4751. -                (entry->u.Open.deleg_type == 2 ||
  4752. -                (params->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA)))) {
  4753. -#ifdef DEBUG_OPEN
  4754. -            DbgP("nfs41_Create: enabling write buffering\n");
  4755. -#endif
  4756. -            SrvOpen->BufferingFlags |=
  4757. -                (FCB_STATE_WRITECACHING_ENABLED |
  4758. -                FCB_STATE_WRITEBUFFERING_ENABLED);
  4759. -        } else if (params->CreateOptions & FILE_WRITE_THROUGH ||
  4760. -                    pVNetRootContext->write_thru)
  4761. -            nfs41_fobx->write_thru = TRUE;
  4762. -        if (entry->u.Open.deleg_type >= 1 ||
  4763. -                params->DesiredAccess & FILE_READ_DATA) {
  4764. -#ifdef DEBUG_OPEN
  4765. -            DbgP("nfs41_Create: enabling read buffering\n");
  4766. -#endif
  4767. -            SrvOpen->BufferingFlags |=
  4768. -                (FCB_STATE_READBUFFERING_ENABLED |
  4769. -                FCB_STATE_READCACHING_ENABLED);
  4770. -        }
  4771. -        nfs41_fobx->timebasedcoherency = pVNetRootContext->timebasedcoherency;
  4772. -        if (pVNetRootContext->nocache ||
  4773. -                (params->CreateOptions & FILE_NO_INTERMEDIATE_BUFFERING)) {
  4774. -#ifdef DEBUG_OPEN
  4775. -            DbgP("nfs41_Create: disabling buffering\n");
  4776. -#endif
  4777. -            SrvOpen->BufferingFlags = FCB_STATE_DISABLE_LOCAL_BUFFERING;
  4778. -            nfs41_fobx->nocache = TRUE;
  4779. -        } else if (!entry->u.Open.deleg_type && !Fcb->OpenCount) {
  4780. -            nfs41_fcb_list_entry *oentry;
  4781. -#ifdef DEBUG_OPEN
  4782. -            DbgP("nfs41_Create: received no delegations: srv_open=0x%p "
  4783. -                "ctime=%llu\n", SrvOpen, entry->ChangeTime);
  4784. -#endif
  4785. -            oentry = RxAllocatePoolWithTag(NonPagedPoolNx,
  4786. -                sizeof(nfs41_fcb_list_entry), NFS41_MM_POOLTAG_OPEN);
  4787. -            if (oentry == NULL) {
  4788. -                status = STATUS_INSUFFICIENT_RESOURCES;
  4789. -                goto out_free;
  4790. -            }
  4791. -            oentry->fcb = RxContext->pFcb;
  4792. -            oentry->nfs41_fobx = nfs41_fobx;
  4793. -            oentry->session = pVNetRootContext->session;
  4794. -            oentry->ChangeTime = entry->ChangeTime;
  4795. -            oentry->skip = FALSE;
  4796. -            nfs41_AddEntry(fcblistLock, openlist, oentry);
  4797. -        }
  4798. -    }
  4799. -
  4800. -    if ((params->CreateOptions & FILE_DELETE_ON_CLOSE) &&
  4801. -            !pVNetRootContext->read_only)
  4802. -        nfs41_fcb->StandardInfo.DeletePending = TRUE;
  4803. -
  4804. -    RxContext->Create.ReturnedCreateInformation =
  4805. -        map_disposition_to_create_retval(params->Disposition, entry->errno);
  4806. -
  4807. -    RxContext->pFobx->OffsetOfNextEaToReturn = 1;
  4808. -    RxContext->CurrentIrp->IoStatus.Information =
  4809. -        RxContext->Create.ReturnedCreateInformation;
  4810. -    status = RxContext->CurrentIrp->IoStatus.Status = STATUS_SUCCESS;
  4811. -
  4812. -out_free:
  4813. -    if (entry)
  4814. -        nfs41_UpcallDestroy(entry);
  4815. -out:
  4816. -#ifdef ENABLE_TIMINGS
  4817. -    t2 = KeQueryPerformanceCounter(NULL);
  4818. -    if ((params->DesiredAccess & FILE_READ_DATA) ||
  4819. -            (params->DesiredAccess & FILE_WRITE_DATA) ||
  4820. -            (params->DesiredAccess & FILE_APPEND_DATA) ||
  4821. -            (params->DesiredAccess & FILE_EXECUTE)) {
  4822. -        InterlockedIncrement(&open.tops);
  4823. -        InterlockedAdd64(&open.ticks, t2.QuadPart - t1.QuadPart);
  4824. -#ifdef ENABLE_INDV_TIMINGS
  4825. -    DbgP("nfs41_Create open delta = %d op=%d sum=%d\n",
  4826. -        t2.QuadPart - t1.QuadPart, open.tops, open.ticks);
  4827. -#endif
  4828. -    } else {
  4829. -        InterlockedIncrement(&lookup.tops);
  4830. -        InterlockedAdd64(&lookup.ticks, t2.QuadPart - t1.QuadPart);
  4831. -#ifdef ENABLE_INDV_TIMINGS
  4832. -    DbgP("nfs41_Create lookup delta = %d op=%d sum=%d\n",
  4833. -        t2.QuadPart - t1.QuadPart, lookup.tops, lookup.ticks);
  4834. -#endif
  4835. -    }
  4836. -#endif
  4837. -#ifdef DEBUG_OPEN
  4838. -    DbgEx();
  4839. -#endif
  4840. -    return status;
  4841. -}
  4842. -
  4843. -static NTSTATUS nfs41_CollapseOpen(
  4844. -    IN OUT PRX_CONTEXT RxContext)
  4845. -{
  4846. -    NTSTATUS status = STATUS_MORE_PROCESSING_REQUIRED;
  4847. -    DbgEn();
  4848. -    DbgEx();
  4849. -    return status;
  4850. -}
  4851. -
  4852. -static NTSTATUS nfs41_ShouldTryToCollapseThisOpen(
  4853. -    IN OUT PRX_CONTEXT RxContext)
  4854. -{
  4855. -    if (RxContext->pRelevantSrvOpen == NULL)
  4856. -        return STATUS_SUCCESS;
  4857. -    else return STATUS_MORE_PROCESSING_REQUIRED;
  4858. -}
  4859. -
  4860. -static ULONG nfs41_ExtendForCache(
  4861. -    IN OUT PRX_CONTEXT RxContext,
  4862. -    IN PLARGE_INTEGER pNewFileSize,
  4863. -    OUT PLARGE_INTEGER pNewAllocationSize)
  4864. -{
  4865. -    NTSTATUS status = STATUS_SUCCESS;
  4866. -    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  4867. -#ifdef DEBUG_CACHE
  4868. -    PLOWIO_CONTEXT LowIoContext  = &RxContext->LowIoContext;
  4869. -    DbgEn();
  4870. -    print_debug_header(RxContext);
  4871. -    DbgP("input: byte count 0x%x filesize 0x%x alloc size 0x%x\n",
  4872. -        LowIoContext->ParamsFor.ReadWrite.ByteCount, *pNewFileSize,
  4873. -        *pNewAllocationSize);
  4874. -#endif
  4875. -    pNewAllocationSize->QuadPart = pNewFileSize->QuadPart + 8192;
  4876. -    nfs41_fcb->StandardInfo.AllocationSize.QuadPart =
  4877. -        pNewAllocationSize->QuadPart;
  4878. -    nfs41_fcb->StandardInfo.EndOfFile.QuadPart = pNewFileSize->QuadPart;
  4879. -#ifdef DEBUG_CACHE
  4880. -    DbgP("new filesize 0x%x new allocation size 0x%x\n",
  4881. -        *pNewFileSize, *pNewAllocationSize);
  4882. -#endif
  4883. -#ifdef DEBUG_CACHE
  4884. -    DbgEx();
  4885. -#endif
  4886. -    return status;
  4887. -}
  4888. -
  4889. -static VOID nfs41_remove_fcb_entry(
  4890. -    PMRX_FCB fcb)
  4891. -{
  4892. -    PLIST_ENTRY pEntry;
  4893. -    nfs41_fcb_list_entry *cur;
  4894. -    ExAcquireFastMutex(&fcblistLock);
  4895. -
  4896. -    pEntry = openlist.head.Flink;
  4897. -    while (!IsListEmpty(&openlist.head)) {
  4898. -        cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry,
  4899. -                nfs41_fcb_list_entry, next);
  4900. -        if (cur->fcb == fcb) {
  4901. -#ifdef DEBUG_CLOSE
  4902. -            DbgP("nfs41_remove_fcb_entry: Found match for fcb=0x%p\n", fcb);
  4903. -#endif
  4904. -            RemoveEntryList(pEntry);
  4905. -            RxFreePool(cur);
  4906. -            break;
  4907. -        }
  4908. -        if (pEntry->Flink == &openlist.head) {
  4909. -#ifdef DEBUG_CLOSE
  4910. -            DbgP("nfs41_remove_fcb_entry: reached EOL looking "
  4911. -                "for fcb 0x%p\n", fcb);
  4912. -#endif
  4913. -            break;
  4914. -        }
  4915. -        pEntry = pEntry->Flink;
  4916. -    }
  4917. -    ExReleaseFastMutex(&fcblistLock);
  4918. -}
  4919. -
  4920. -static VOID nfs41_invalidate_fobx_entry(
  4921. -    IN OUT PMRX_FOBX pFobx)
  4922. -{
  4923. -    PLIST_ENTRY pEntry;
  4924. -    nfs41_fcb_list_entry *cur;
  4925. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(pFobx);
  4926. -
  4927. -    ExAcquireFastMutex(&fcblistLock);
  4928. -
  4929. -    pEntry = openlist.head.Flink;
  4930. -    while (!IsListEmpty(&openlist.head)) {
  4931. -        cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry,
  4932. -                nfs41_fcb_list_entry, next);
  4933. -        if (cur->nfs41_fobx == nfs41_fobx) {
  4934. -#ifdef DEBUG_CLOSE
  4935. -            DbgP("nfs41_invalidate_fobx_entry: Found match for fobx=0x%p\n", fobx);
  4936. -#endif
  4937. -            cur->nfs41_fobx = NULL;
  4938. -            break;
  4939. -        }
  4940. -        if (pEntry->Flink == &openlist.head) {
  4941. -#ifdef DEBUG_CLOSE
  4942. -            DbgP("nfs41_invalidate_fobx_entry: reached EOL looking "
  4943. -                "for fobx 0x%p\n", fobx);
  4944. -#endif
  4945. -            break;
  4946. -        }
  4947. -        pEntry = pEntry->Flink;
  4948. -    }
  4949. -    ExReleaseFastMutex(&fcblistLock);
  4950. -}
  4951. -
  4952. -static NTSTATUS map_close_errors(
  4953. -    DWORD status)
  4954. -{
  4955. -    switch (status) {
  4956. -    case NO_ERROR:              return STATUS_SUCCESS;
  4957. -    case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED;
  4958. -    case ERROR_NOT_EMPTY:       return STATUS_DIRECTORY_NOT_EMPTY;
  4959. -    case ERROR_FILE_INVALID:    return STATUS_FILE_INVALID;
  4960. -    case ERROR_DISK_FULL:       return STATUS_DISK_FULL;
  4961. -    case ERROR_DISK_QUOTA_EXCEEDED: return STATUS_DISK_QUOTA_EXCEEDED;
  4962. -    case ERROR_FILE_TOO_LARGE:  return STATUS_FILE_TOO_LARGE;
  4963. -    default:
  4964. -        print_error("map_close_errors: "
  4965. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  4966. -            "defaulting to STATUS_INTERNAL_ERROR\n", status);
  4967. -    case ERROR_INTERNAL_ERROR:  return STATUS_INTERNAL_ERROR;
  4968. -    }
  4969. -}
  4970. -
  4971. -static NTSTATUS nfs41_CloseSrvOpen(
  4972. -    IN OUT PRX_CONTEXT RxContext)
  4973. -{
  4974. -    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  4975. -    nfs41_updowncall_entry *entry;
  4976. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  4977. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  4978. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  4979. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  4980. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  4981. -    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  4982. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  4983. -#ifdef ENABLE_TIMINGS
  4984. -    LARGE_INTEGER t1, t2;
  4985. -    t1 = KeQueryPerformanceCounter(NULL);
  4986. -#endif
  4987. -
  4988. -#ifdef DEBUG_CLOSE
  4989. -    DbgEn();
  4990. -    print_debug_header(RxContext);
  4991. -#endif
  4992. -
  4993. -    if (!nfs41_fobx->deleg_type && !nfs41_fcb->StandardInfo.Directory &&
  4994. -            !RxContext->pFcb->OpenCount) {
  4995. -        nfs41_remove_fcb_entry(RxContext->pFcb);
  4996. -    }
  4997. -
  4998. -    status = nfs41_UpcallCreate(NFS41_CLOSE, &nfs41_fobx->sec_ctx,
  4999. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  5000. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  5001. -    if (status) goto out;
  5002. -
  5003. -    entry->u.Close.srv_open = SrvOpen;
  5004. -    if (nfs41_fcb->StandardInfo.DeletePending)
  5005. -        nfs41_fcb->DeletePending = TRUE;
  5006. -    if (!RxContext->pFcb->OpenCount ||
  5007. -            (nfs41_fcb->StandardInfo.DeletePending &&
  5008. -                nfs41_fcb->StandardInfo.Directory))
  5009. -        entry->u.Close.remove = nfs41_fcb->StandardInfo.DeletePending;
  5010. -    if (!RxContext->pFcb->OpenCount)
  5011. -        entry->u.Close.renamed = nfs41_fcb->Renamed;
  5012. -
  5013. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  5014. -    if (status) goto out;
  5015. -
  5016. -    /* map windows ERRORs to NTSTATUS */
  5017. -    status = map_close_errors(entry->status);
  5018. -    nfs41_UpcallDestroy(entry);
  5019. -out:
  5020. -#ifdef ENABLE_TIMINGS
  5021. -    t2 = KeQueryPerformanceCounter(NULL);
  5022. -    InterlockedIncrement(&close.tops);
  5023. -    InterlockedAdd64(&close.ticks, t2.QuadPart - t1.QuadPart);
  5024. -#ifdef ENABLE_INDV_TIMINGS
  5025. -    DbgP("nfs41_CloseSrvOpen delta = %d op=%d sum=%d\n",
  5026. -        t2.QuadPart - t1.QuadPart, close.tops, close.ticks);
  5027. -#endif
  5028. -#endif
  5029. -#ifdef DEBUG_CLOSE
  5030. -    DbgEx();
  5031. -#endif
  5032. -    return status;
  5033. -}
  5034. -
  5035. -static NTSTATUS nfs41_Flush(
  5036. -    IN OUT PRX_CONTEXT RxContext)
  5037. -{
  5038. -    DbgP("nfs41_Flush: FileName='%wZ'\n",
  5039. -        GET_ALREADY_PREFIXED_NAME_FROM_CONTEXT(RxContext));
  5040. -
  5041. -    return STATUS_SUCCESS;
  5042. -}
  5043. -
  5044. -static NTSTATUS nfs41_DeallocateForFcb(
  5045. -    IN OUT PMRX_FCB pFcb)
  5046. -{
  5047. -    nfs41_remove_fcb_entry(pFcb);
  5048. -    return STATUS_SUCCESS;
  5049. -}
  5050. -
  5051. -static NTSTATUS nfs41_DeallocateForFobx(
  5052. -    IN OUT PMRX_FOBX pFobx)
  5053. -{
  5054. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(pFobx);
  5055. -
  5056. -    nfs41_invalidate_fobx_entry(pFobx);
  5057. -
  5058. -    if (nfs41_fobx->acl) {
  5059. -        RxFreePool(nfs41_fobx->acl);
  5060. -        nfs41_fobx->acl = NULL;
  5061. -    }
  5062. -
  5063. -    if (nfs41_fobx->sec_ctx.ClientToken) {
  5064. -        SeDeleteClientSecurity(&nfs41_fobx->sec_ctx);
  5065. -        nfs41_fobx->sec_ctx.ClientToken = NULL;
  5066. -    }
  5067. -
  5068. -    return STATUS_SUCCESS;
  5069. -}
  5070. -
  5071. -static void print_debug_filedirquery_header(
  5072. -    PRX_CONTEXT RxContext)
  5073. -{
  5074. -    print_debug_header(RxContext);
  5075. -    DbgP("FileName='%wZ', InfoClass = '%s'\n",
  5076. -        GET_ALREADY_PREFIXED_NAME_FROM_CONTEXT(RxContext),
  5077. -        print_file_information_class(RxContext->Info.FileInformationClass));
  5078. -}
  5079. -
  5080. -static void print_querydir_args(
  5081. -    PRX_CONTEXT RxContext)
  5082. -{
  5083. -    print_debug_filedirquery_header(RxContext);
  5084. -    DbgP("Filter='%wZ', Index=%d, Restart/Single/Specified/Init=%d/%d/%d/%d\n",
  5085. -        &RxContext->pFobx->UnicodeQueryTemplate,
  5086. -        RxContext->QueryDirectory.FileIndex,
  5087. -        RxContext->QueryDirectory.RestartScan,
  5088. -        RxContext->QueryDirectory.ReturnSingleEntry,
  5089. -        RxContext->QueryDirectory.IndexSpecified,
  5090. -        RxContext->QueryDirectory.InitialQuery);
  5091. -}
  5092. -
  5093. -static NTSTATUS map_querydir_errors(
  5094. -    DWORD status)
  5095. -{
  5096. -    switch (status) {
  5097. -    case ERROR_ACCESS_DENIED:       return STATUS_ACCESS_DENIED;
  5098. -    case ERROR_BUFFER_OVERFLOW:     return STATUS_BUFFER_OVERFLOW;
  5099. -    case ERROR_FILE_NOT_FOUND:      return STATUS_NO_SUCH_FILE;
  5100. -    case ERROR_NETNAME_DELETED:     return STATUS_NETWORK_NAME_DELETED;
  5101. -    case ERROR_INVALID_PARAMETER:   return STATUS_INVALID_PARAMETER;
  5102. -    case ERROR_NO_MORE_FILES:       return STATUS_NO_MORE_FILES;
  5103. -    case ERROR_OUTOFMEMORY:         return STATUS_INSUFFICIENT_RESOURCES;
  5104. -    case ERROR_FILENAME_EXCED_RANGE: return STATUS_NAME_TOO_LONG;
  5105. -    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  5106. -    default:
  5107. -        print_error("map_querydir_errors: "
  5108. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  5109. -            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status);
  5110. -    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  5111. -    }
  5112. -}
  5113. -
  5114. -static NTSTATUS check_nfs41_dirquery_args(
  5115. -    IN PRX_CONTEXT RxContext)
  5116. -{
  5117. -    if (RxContext->Info.Buffer == NULL)
  5118. -        return STATUS_INVALID_USER_BUFFER;
  5119. -    return STATUS_SUCCESS;
  5120. -}
  5121. -
  5122. -static NTSTATUS nfs41_QueryDirectory(
  5123. -    IN OUT PRX_CONTEXT RxContext)
  5124. -{
  5125. -    NTSTATUS status = STATUS_INVALID_PARAMETER;
  5126. -    nfs41_updowncall_entry *entry;
  5127. -    FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass;
  5128. -    PUNICODE_STRING Filter = &RxContext->pFobx->UnicodeQueryTemplate;
  5129. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  5130. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  5131. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  5132. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  5133. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  5134. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  5135. -#ifdef ENABLE_TIMINGS
  5136. -    LARGE_INTEGER t1, t2;
  5137. -    t1 = KeQueryPerformanceCounter(NULL);
  5138. -#endif
  5139. -
  5140. -#ifdef DEBUG_DIR_QUERY
  5141. -    DbgEn();
  5142. -    print_querydir_args(RxContext);
  5143. -#endif
  5144. -
  5145. -    status = check_nfs41_dirquery_args(RxContext);
  5146. -    if (status) goto out;
  5147. -
  5148. -    switch (InfoClass) {
  5149. -    /* classes handled in readdir_copy_entry() and readdir_size_for_entry() */
  5150. -    case FileNamesInformation:
  5151. -    case FileDirectoryInformation:
  5152. -    case FileFullDirectoryInformation:
  5153. -    case FileIdFullDirectoryInformation:
  5154. -    case FileBothDirectoryInformation:
  5155. -    case FileIdBothDirectoryInformation:
  5156. -        break;
  5157. -    default:
  5158. -        print_error("nfs41_QueryDirectory: unhandled dir query class %d\n",
  5159. -            InfoClass);
  5160. -        status = STATUS_NOT_SUPPORTED;
  5161. -        goto out;
  5162. -    }
  5163. -    status = nfs41_UpcallCreate(NFS41_DIR_QUERY, &nfs41_fobx->sec_ctx,
  5164. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  5165. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  5166. -    if (status) goto out;
  5167. -
  5168. -    entry->u.QueryFile.InfoClass = InfoClass;
  5169. -    entry->buf_len = RxContext->Info.LengthRemaining;
  5170. -    entry->buf = RxContext->Info.Buffer;
  5171. -    entry->u.QueryFile.mdl = IoAllocateMdl(RxContext->Info.Buffer,
  5172. -        RxContext->Info.LengthRemaining, FALSE, FALSE, NULL);
  5173. -    if (entry->u.QueryFile.mdl == NULL) {
  5174. -        status = STATUS_INTERNAL_ERROR;
  5175. -        nfs41_UpcallDestroy(entry);
  5176. -        goto out;
  5177. -    }
  5178. -#pragma warning( push )
  5179. -/*
  5180. - * C28145: "The opaque MDL structure should not be modified by a
  5181. - * driver.", |MDL_MAPPING_CAN_FAIL| is the exception
  5182. - */
  5183. -#pragma warning (disable : 28145)
  5184. -    entry->u.QueryFile.mdl->MdlFlags |= MDL_MAPPING_CAN_FAIL;
  5185. -#pragma warning( pop )
  5186. -
  5187. -    MmProbeAndLockPages(entry->u.QueryFile.mdl, KernelMode, IoModifyAccess);
  5188. -
  5189. -    entry->u.QueryFile.filter = Filter;
  5190. -    entry->u.QueryFile.initial_query = RxContext->QueryDirectory.InitialQuery;
  5191. -    entry->u.QueryFile.restart_scan = RxContext->QueryDirectory.RestartScan;
  5192. -    entry->u.QueryFile.return_single = RxContext->QueryDirectory.ReturnSingleEntry;
  5193. -
  5194. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  5195. -    if (status) goto out;
  5196. -    MmUnlockPages(entry->u.QueryFile.mdl);
  5197. -
  5198. -    if (entry->status == STATUS_BUFFER_TOO_SMALL) {
  5199. -        DbgP("nfs41_QueryDirectory: buffer too small provided %d need %lu\n",
  5200. -            RxContext->Info.LengthRemaining, entry->buf_len);
  5201. -        RxContext->InformationToReturn = entry->buf_len;
  5202. -        status = STATUS_BUFFER_TOO_SMALL;
  5203. -    } else if (entry->status == STATUS_SUCCESS) {
  5204. -#ifdef ENABLE_TIMINGS
  5205. -        InterlockedIncrement(&readdir.sops);
  5206. -        InterlockedAdd64(&readdir.size, entry->u.QueryFile.buf_len);
  5207. -#endif
  5208. -        RxContext->Info.LengthRemaining -= entry->buf_len;
  5209. -        status = STATUS_SUCCESS;
  5210. -    } else if ((entry->status == STATUS_ACCESS_VIOLATION) ||
  5211. -        (entry->status == STATUS_INSUFFICIENT_RESOURCES)) {
  5212. -        DbgP("nfs41_QueryDirectory: internal error: entry->status=0x%x\n",
  5213. -            (int)entry->status);
  5214. -        status = STATUS_INSUFFICIENT_RESOURCES;
  5215. -    } else {
  5216. -        /* map windows ERRORs to NTSTATUS */
  5217. -        status = map_querydir_errors(entry->status);
  5218. -    }
  5219. -    IoFreeMdl(entry->u.QueryFile.mdl);
  5220. -    nfs41_UpcallDestroy(entry);
  5221. -out:
  5222. -#ifdef ENABLE_TIMINGS
  5223. -    t2 = KeQueryPerformanceCounter(NULL);
  5224. -    InterlockedIncrement(&readdir.tops);
  5225. -    InterlockedAdd64(&readdir.ticks, t2.QuadPart - t1.QuadPart);
  5226. -#ifdef ENABLE_INDV_TIMINGS
  5227. -    DbgP("nfs41_QueryDirectory delta = %d ops=%d sum=%d\n",
  5228. -        t2.QuadPart - t1.QuadPart, readdir.tops, readdir.ticks);
  5229. -#endif
  5230. -#endif
  5231. -#ifdef DEBUG_DIR_QUERY
  5232. -    DbgEx();
  5233. -#endif
  5234. -    return status;
  5235. -}
  5236. -
  5237. -static void print_queryvolume_args(
  5238. -    PRX_CONTEXT RxContext)
  5239. -{
  5240. -    print_debug_header(RxContext);
  5241. -    DbgP("FileName='%wZ', InfoClass = '%s' BufferLen = %d\n",
  5242. -        GET_ALREADY_PREFIXED_NAME_FROM_CONTEXT(RxContext),
  5243. -        print_fs_information_class(RxContext->Info.FileInformationClass),
  5244. -        RxContext->Info.LengthRemaining);
  5245. -}
  5246. -
  5247. -static NTSTATUS map_volume_errors(
  5248. -    DWORD status)
  5249. -{
  5250. -    switch (status) {
  5251. -    case ERROR_ACCESS_DENIED:       return STATUS_ACCESS_DENIED;
  5252. -    case ERROR_VC_DISCONNECTED:     return STATUS_CONNECTION_DISCONNECTED;
  5253. -    case ERROR_NETNAME_DELETED:     return STATUS_NETWORK_NAME_DELETED;
  5254. -    case ERROR_INVALID_PARAMETER:   return STATUS_INVALID_PARAMETER;
  5255. -    case ERROR_OUTOFMEMORY:         return STATUS_INSUFFICIENT_RESOURCES;
  5256. -    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  5257. -    default:
  5258. -        print_error("map_volume_errors: "
  5259. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  5260. -            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status);
  5261. -    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  5262. -    }
  5263. -}
  5264. -
  5265. -static void nfs41_create_volume_info(PFILE_FS_VOLUME_INFORMATION pVolInfo, DWORD *len)
  5266. -{
  5267. -    DECLARE_CONST_UNICODE_STRING(VolName, VOL_NAME);
  5268. -
  5269. -    RtlZeroMemory(pVolInfo, sizeof(FILE_FS_VOLUME_INFORMATION));
  5270. -    pVolInfo->VolumeSerialNumber = 0xBABAFACE;
  5271. -    pVolInfo->VolumeLabelLength = VolName.Length;
  5272. -    RtlCopyMemory(&pVolInfo->VolumeLabel[0], (PVOID)VolName.Buffer,
  5273. -        VolName.MaximumLength);
  5274. -    *len = sizeof(FILE_FS_VOLUME_INFORMATION) + VolName.Length;
  5275. -}
  5276. -
  5277. -static BOOLEAN is_root_directory(
  5278. -    PRX_CONTEXT RxContext)
  5279. -{
  5280. -    __notnull PV_NET_ROOT VNetRoot = (PV_NET_ROOT)
  5281. -        RxContext->pRelevantSrvOpen->pVNetRoot;
  5282. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  5283. -        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  5284. -
  5285. -    /* calculate the root directory's length, including vnetroot prefix,
  5286. -     * mount path, and a trailing \ */
  5287. -    const USHORT RootPathLen = VNetRoot->PrefixEntry.Prefix.Length +
  5288. -            pVNetRootContext->MountPathLen + sizeof(WCHAR);
  5289. -
  5290. -    return RxContext->CurrentIrpSp->FileObject->FileName.Length <= RootPathLen;
  5291. -}
  5292. -
  5293. -static NTSTATUS nfs41_QueryVolumeInformation(
  5294. -    IN OUT PRX_CONTEXT RxContext)
  5295. -{
  5296. -    NTSTATUS status = STATUS_INVALID_PARAMETER;
  5297. -    nfs41_updowncall_entry *entry;
  5298. -    ULONG RemainingLength = RxContext->Info.LengthRemaining, SizeUsed;
  5299. -    FS_INFORMATION_CLASS InfoClass = RxContext->Info.FsInformationClass;
  5300. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  5301. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  5302. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  5303. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  5304. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  5305. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  5306. -    NFS41GetDeviceExtension(RxContext, DevExt);
  5307. -
  5308. -#ifdef ENABLE_TIMINGS
  5309. -    LARGE_INTEGER t1, t2;
  5310. -    t1 = KeQueryPerformanceCounter(NULL);
  5311. -#endif
  5312. -
  5313. -#ifdef DEBUG_VOLUME_QUERY
  5314. -    DbgEn();
  5315. -    print_queryvolume_args(RxContext);
  5316. -#endif
  5317. -
  5318. -    status = check_nfs41_dirquery_args(RxContext);
  5319. -    if (status) goto out;
  5320. -
  5321. -    RtlZeroMemory(RxContext->Info.Buffer, RxContext->Info.LengthRemaining);
  5322. -
  5323. -    switch (InfoClass) {
  5324. -    case FileFsVolumeInformation:
  5325. -        if ((ULONG)RxContext->Info.LengthRemaining >= DevExt->VolAttrsLen) {
  5326. -            RtlCopyMemory(RxContext->Info.Buffer, DevExt->VolAttrs,
  5327. -                DevExt->VolAttrsLen);
  5328. -            RxContext->Info.LengthRemaining -= DevExt->VolAttrsLen;
  5329. -            status = STATUS_SUCCESS;
  5330. -        } else {
  5331. -            RtlCopyMemory(RxContext->Info.Buffer, DevExt->VolAttrs,
  5332. -                RxContext->Info.LengthRemaining);
  5333. -            status = STATUS_BUFFER_OVERFLOW;
  5334. -        }
  5335. -        goto out;
  5336. -    case FileFsDeviceInformation:
  5337. -    {
  5338. -        PFILE_FS_DEVICE_INFORMATION pDevInfo = RxContext->Info.Buffer;
  5339. -
  5340. -        SizeUsed = sizeof(FILE_FS_DEVICE_INFORMATION);
  5341. -        if (RemainingLength < SizeUsed) {
  5342. -            status = STATUS_BUFFER_TOO_SMALL;
  5343. -            RxContext->InformationToReturn = SizeUsed;
  5344. -            goto out;
  5345. -        }
  5346. -        pDevInfo->DeviceType = RxContext->pFcb->pNetRoot->DeviceType;
  5347. -        pDevInfo->Characteristics = FILE_REMOTE_DEVICE | FILE_DEVICE_IS_MOUNTED;
  5348. -        RxContext->Info.LengthRemaining -= SizeUsed;
  5349. -        status = STATUS_SUCCESS;
  5350. -        goto out;
  5351. -    }
  5352. -    case FileAccessInformation:
  5353. -        status = STATUS_NOT_SUPPORTED;
  5354. -        goto out;
  5355. -
  5356. -    case FileFsAttributeInformation:
  5357. -        if (RxContext->Info.LengthRemaining < FS_ATTR_LEN) {
  5358. -            RxContext->InformationToReturn = FS_ATTR_LEN;
  5359. -            status = STATUS_BUFFER_TOO_SMALL;
  5360. -            goto out;
  5361. -        }
  5362. -
  5363. -        /* on attribute queries for the root directory,
  5364. -         * use cached volume attributes from mount */
  5365. -        if (is_root_directory(RxContext)) {
  5366. -            PFILE_FS_ATTRIBUTE_INFORMATION attrs =
  5367. -                (PFILE_FS_ATTRIBUTE_INFORMATION)RxContext->Info.Buffer;
  5368. -            DECLARE_CONST_UNICODE_STRING(FsName, FS_NAME);
  5369. -
  5370. -            RtlCopyMemory(attrs, &pVNetRootContext->FsAttrs,
  5371. -                sizeof(pVNetRootContext->FsAttrs));
  5372. -
  5373. -            /* fill in the FileSystemName */
  5374. -            RtlCopyMemory(attrs->FileSystemName, FsName.Buffer,
  5375. -                FsName.MaximumLength); /* 'MaximumLength' to include null */
  5376. -            attrs->FileSystemNameLength = FsName.Length;
  5377. -
  5378. -            RxContext->Info.LengthRemaining -= FS_ATTR_LEN;
  5379. -            goto out;
  5380. -        }
  5381. -        /* else fall through and send the upcall */
  5382. -    case FileFsSizeInformation:
  5383. -    case FileFsFullSizeInformation:
  5384. -        break;
  5385. -
  5386. -    default:
  5387. -        print_error("nfs41_QueryVolumeInformation: unhandled class %d\n", InfoClass);
  5388. -        status = STATUS_NOT_SUPPORTED;
  5389. -        goto out;
  5390. -    }
  5391. -    status = nfs41_UpcallCreate(NFS41_VOLUME_QUERY, &nfs41_fobx->sec_ctx,
  5392. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  5393. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  5394. -    if (status) goto out;
  5395. -
  5396. -    entry->u.Volume.query = InfoClass;
  5397. -    entry->buf = RxContext->Info.Buffer;
  5398. -    entry->buf_len = RxContext->Info.LengthRemaining;
  5399. -
  5400. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  5401. -    if (status) goto out;
  5402. -
  5403. -    if (entry->status == STATUS_BUFFER_TOO_SMALL) {
  5404. -        RxContext->InformationToReturn = entry->buf_len;
  5405. -        status = STATUS_BUFFER_TOO_SMALL;
  5406. -    } else if (entry->status == STATUS_SUCCESS) {
  5407. -        if (InfoClass == FileFsAttributeInformation) {
  5408. -            /* fill in the FileSystemName */
  5409. -            PFILE_FS_ATTRIBUTE_INFORMATION attrs =
  5410. -                (PFILE_FS_ATTRIBUTE_INFORMATION)RxContext->Info.Buffer;
  5411. -            DECLARE_CONST_UNICODE_STRING(FsName, FS_NAME);
  5412. -
  5413. -            RtlCopyMemory(attrs->FileSystemName, FsName.Buffer,
  5414. -                FsName.MaximumLength); /* 'MaximumLength' to include null */
  5415. -            attrs->FileSystemNameLength = FsName.Length;
  5416. -
  5417. -            entry->buf_len = FS_ATTR_LEN;
  5418. -        }
  5419. -#ifdef ENABLE_TIMINGS
  5420. -        InterlockedIncrement(&volume.sops);
  5421. -        InterlockedAdd64(&volume.size, entry->u.Volume.buf_len);
  5422. -#endif
  5423. -        RxContext->Info.LengthRemaining -= entry->buf_len;
  5424. -        status = STATUS_SUCCESS;
  5425. -    } else {
  5426. -        status = map_volume_errors(entry->status);
  5427. -    }
  5428. -    nfs41_UpcallDestroy(entry);
  5429. -out:
  5430. -#ifdef ENABLE_TIMINGS
  5431. -    t2 = KeQueryPerformanceCounter(NULL);
  5432. -    InterlockedIncrement(&volume.tops);
  5433. -    InterlockedAdd64(&volume.ticks, t2.QuadPart - t1.QuadPart);
  5434. -#ifdef ENABLE_INDV_TIMINGS
  5435. -    DbgP("nfs41_QueryVolumeInformation delta = %d op=%d sum=%d\n",
  5436. -        t2.QuadPart - t1.QuadPart, volume.tops, volume.ticks);
  5437. -#endif
  5438. -#endif
  5439. -#ifdef DEBUG_VOLUME_QUERY
  5440. -    DbgEx();
  5441. -#endif
  5442. -    return status;
  5443. -}
  5444. -
  5445. -static VOID nfs41_update_fcb_list(
  5446. -    PMRX_FCB fcb,
  5447. -    ULONGLONG ChangeTime)
  5448. -{
  5449. -    PLIST_ENTRY pEntry;
  5450. -    nfs41_fcb_list_entry *cur;
  5451. -    ExAcquireFastMutex(&fcblistLock);
  5452. -    pEntry = openlist.head.Flink;
  5453. -    while (!IsListEmpty(&openlist.head)) {
  5454. -        cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry,
  5455. -                nfs41_fcb_list_entry, next);
  5456. -        if (cur->fcb == fcb &&
  5457. -                cur->ChangeTime != ChangeTime) {
  5458. -#if defined(DEBUG_FILE_SET) || defined(DEBUG_ACL_SET) || \
  5459. -    defined(DEBUG_WRITE) || defined(DEBUG_EA_SET)
  5460. -            DbgP("nfs41_update_fcb_list: Found match for fcb 0x%p: "
  5461. -                "updating %llu to %llu\n",
  5462. -                fcb, cur->ChangeTime, ChangeTime);
  5463. -#endif
  5464. -            cur->ChangeTime = ChangeTime;
  5465. -            break;
  5466. -        }
  5467. -        /* place an upcall for this srv_open */
  5468. -        if (pEntry->Flink == &openlist.head) {
  5469. -#if defined(DEBUG_FILE_SET) || defined(DEBUG_ACL_SET) || \
  5470. -    defined(DEBUG_WRITE) || defined(DEBUG_EA_SET)
  5471. -            DbgP("nfs41_update_fcb_list: reached EOL loooking for "
  5472. -                "fcb=0x%p\n", fcb);
  5473. -#endif
  5474. -            break;
  5475. -        }
  5476. -        pEntry = pEntry->Flink;
  5477. -    }
  5478. -    ExReleaseFastMutex(&fcblistLock);
  5479. -}
  5480. -
  5481. -static void print_nfs3_attrs(
  5482. -    nfs3_attrs *attrs)
  5483. -{
  5484. -    DbgP("type=%d mode=0%o nlink=%d size=%d "
  5485. -        "atime=0x%x mtime=0x%x ctime=0x%x\n",
  5486. -        attrs->type, attrs->mode, attrs->nlink, attrs->size, attrs->atime,
  5487. -        attrs->mtime, attrs->ctime);
  5488. -}
  5489. -
  5490. -static void file_time_to_nfs_time(
  5491. -    IN const PLARGE_INTEGER file_time,
  5492. -    OUT LONGLONG *nfs_time)
  5493. -{
  5494. -    LARGE_INTEGER diff = unix_time_diff;
  5495. -    diff.QuadPart = file_time->QuadPart - diff.QuadPart;
  5496. -    *nfs_time = diff.QuadPart / 10000000;
  5497. -}
  5498. -
  5499. -static void create_nfs3_attrs(
  5500. -    nfs3_attrs *attrs,
  5501. -    PNFS41_FCB nfs41_fcb)
  5502. -{
  5503. -    RtlZeroMemory(attrs, sizeof(nfs3_attrs));
  5504. -    if (nfs41_fcb->BasicInfo.FileAttributes & FILE_ATTRIBUTE_REPARSE_POINT)
  5505. -        attrs->type = NF3LNK;
  5506. -    else if (nfs41_fcb->StandardInfo.Directory)
  5507. -        attrs->type = NF3DIR;
  5508. -    else
  5509. -        attrs->type = NF3REG;
  5510. -    attrs->mode = nfs41_fcb->mode;
  5511. -#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  5512. -    attrs->uid = nfs41_fcb->owner_local_uid;
  5513. -    attrs->gid = nfs41_fcb->owner_group_local_gid;
  5514. -#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  5515. -    attrs->nlink = nfs41_fcb->StandardInfo.NumberOfLinks;
  5516. -    attrs->size.QuadPart = attrs->used.QuadPart =
  5517. -        nfs41_fcb->StandardInfo.EndOfFile.QuadPart;
  5518. -    file_time_to_nfs_time(&nfs41_fcb->BasicInfo.LastAccessTime, &attrs->atime);
  5519. -    file_time_to_nfs_time(&nfs41_fcb->BasicInfo.ChangeTime, &attrs->mtime);
  5520. -    file_time_to_nfs_time(&nfs41_fcb->BasicInfo.CreationTime, &attrs->ctime);
  5521. -}
  5522. -
  5523. -
  5524. -static NTSTATUS map_setea_error(
  5525. -    DWORD error)
  5526. -{
  5527. -    switch (error) {
  5528. -    case NO_ERROR:                      return STATUS_SUCCESS;
  5529. -    case ERROR_FILE_NOT_FOUND:          return STATUS_NO_EAS_ON_FILE;
  5530. -    case ERROR_ACCESS_DENIED:           return STATUS_ACCESS_DENIED;
  5531. -    case ERROR_NETWORK_ACCESS_DENIED:   return STATUS_NETWORK_ACCESS_DENIED;
  5532. -    case ERROR_NETNAME_DELETED:         return STATUS_NETWORK_NAME_DELETED;
  5533. -    case ERROR_FILE_TOO_LARGE:          return STATUS_EA_TOO_LARGE;
  5534. -    case ERROR_BUFFER_OVERFLOW:         return STATUS_BUFFER_OVERFLOW;
  5535. -    case STATUS_BUFFER_TOO_SMALL:
  5536. -    case ERROR_INSUFFICIENT_BUFFER:     return STATUS_BUFFER_TOO_SMALL;
  5537. -    case ERROR_INVALID_EA_HANDLE:       return STATUS_NONEXISTENT_EA_ENTRY;
  5538. -    case ERROR_NO_MORE_FILES:           return STATUS_NO_MORE_EAS;
  5539. -    case ERROR_EA_FILE_CORRUPT:         return STATUS_EA_CORRUPT_ERROR;
  5540. -    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  5541. -    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  5542. -    case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  5543. -    default:
  5544. -        print_error("map_setea_error: "
  5545. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  5546. -            "defaulting to STATUS_INVALID_PARAMETER\n", error);
  5547. -    case ERROR_INVALID_PARAMETER:       return STATUS_INVALID_PARAMETER;
  5548. -    }
  5549. -}
  5550. -
  5551. -static NTSTATUS check_nfs41_setea_args(
  5552. -    IN PRX_CONTEXT RxContext)
  5553. -{
  5554. -    NTSTATUS status;
  5555. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  5556. -        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  5557. -    __notnull PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs =
  5558. -        &pVNetRootContext->FsAttrs;
  5559. -    __notnull PFILE_FULL_EA_INFORMATION ea =
  5560. -        (PFILE_FULL_EA_INFORMATION)RxContext->Info.Buffer;
  5561. -
  5562. -    status = check_nfs41_dirquery_args(RxContext);
  5563. -    if (status) goto out;
  5564. -
  5565. -    if (ea == NULL) {
  5566. -        status = STATUS_INVALID_PARAMETER;
  5567. -        goto out;
  5568. -    }
  5569. -    if (AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength) ||
  5570. -        AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) {
  5571. -        status = STATUS_INVALID_PARAMETER; /* only allowed on create */
  5572. -        goto out;
  5573. -    }
  5574. -    /* ignore cygwin EAs when checking support */
  5575. -    if (!(FsAttrs->FileSystemAttributes & FILE_SUPPORTS_EXTENDED_ATTRIBUTES)
  5576. -        && !AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength)) {
  5577. -        status = STATUS_EAS_NOT_SUPPORTED;
  5578. -        goto out;
  5579. -    }
  5580. -    if ((RxContext->pRelevantSrvOpen->DesiredAccess & FILE_WRITE_EA) == 0) {
  5581. -        status = STATUS_ACCESS_DENIED;
  5582. -        goto out;
  5583. -    }
  5584. -    if (pVNetRootContext->read_only) {
  5585. -        print_error("check_nfs41_setattr_args: Read-only mount\n");
  5586. -        status = STATUS_MEDIA_WRITE_PROTECTED;
  5587. -        goto out;
  5588. -    }
  5589. -out:
  5590. -    return status;
  5591. -}
  5592. -
  5593. -static NTSTATUS nfs41_SetEaInformation(
  5594. -    IN OUT PRX_CONTEXT RxContext)
  5595. -{
  5596. -    NTSTATUS status = STATUS_EAS_NOT_SUPPORTED;
  5597. -    nfs41_updowncall_entry *entry;
  5598. -    __notnull PFILE_FULL_EA_INFORMATION eainfo =
  5599. -        (PFILE_FULL_EA_INFORMATION)RxContext->Info.Buffer;        
  5600. -    nfs3_attrs *attrs = NULL;
  5601. -    ULONG buflen = RxContext->CurrentIrpSp->Parameters.SetEa.Length, error_offset;
  5602. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  5603. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  5604. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  5605. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  5606. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  5607. -    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  5608. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  5609. -#ifdef ENABLE_TIMINGS
  5610. -    LARGE_INTEGER t1, t2;
  5611. -    t1 = KeQueryPerformanceCounter(NULL);
  5612. -#endif
  5613. -
  5614. -#ifdef DEBUG_EA_SET
  5615. -    DbgEn();
  5616. -    print_debug_header(RxContext);
  5617. -    print_ea_info(eainfo);
  5618. -#endif
  5619. -
  5620. -    status = check_nfs41_setea_args(RxContext);
  5621. -    if (status) goto out;
  5622. -
  5623. -    status = nfs41_UpcallCreate(NFS41_EA_SET, &nfs41_fobx->sec_ctx,
  5624. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  5625. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  5626. -    if (status) goto out;
  5627. -
  5628. -    if (AnsiStrEq(&NfsV3Attributes, eainfo->EaName, eainfo->EaNameLength)) {
  5629. -        attrs = (nfs3_attrs *)(eainfo->EaName + eainfo->EaNameLength + 1);
  5630. -#ifdef DEBUG_EA_SET
  5631. -        print_nfs3_attrs(attrs);
  5632. -        DbgP("old mode is 0%o new mode is 0%o\n", nfs41_fcb->mode, attrs->mode);
  5633. -#endif
  5634. -        entry->u.SetEa.mode = attrs->mode;
  5635. -    } else {
  5636. -        entry->u.SetEa.mode = 0;
  5637. -        status = IoCheckEaBufferValidity(eainfo, buflen, &error_offset);
  5638. -        if (status) {
  5639. -            DbgP("nfs41_SetEaInformation: "
  5640. -                "status(=0x%lx)=IoCheckEaBufferValidity"
  5641. -                "(eainfo=0x%p, buflen=%lu, &(error_offset=%d))\n",
  5642. -                (long)status, (void *)eainfo, buflen,
  5643. -                (int)error_offset);
  5644. -            nfs41_UpcallDestroy(entry);
  5645. -            entry = NULL;
  5646. -            goto out;
  5647. -        }
  5648. -    }
  5649. -    entry->buf = eainfo;
  5650. -    entry->buf_len = buflen;
  5651. -    
  5652. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  5653. -    if (status) goto out;
  5654. -#ifdef ENABLE_TIMINGS
  5655. -    if (entry->status == STATUS_SUCCESS) {
  5656. -        InterlockedIncrement(&setexattr.sops);
  5657. -        InterlockedAdd64(&setexattr.size, entry->u.SetEa.buf_len);
  5658. -    }
  5659. -#endif
  5660. -    status = map_setea_error(entry->status);
  5661. -    if (!status) {
  5662. -        if (!nfs41_fobx->deleg_type && entry->ChangeTime &&
  5663. -                (SrvOpen->DesiredAccess &
  5664. -                (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA)))
  5665. -            nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime);
  5666. -        nfs41_fcb->changeattr = entry->ChangeTime;
  5667. -        nfs41_fcb->mode = entry->u.SetEa.mode;
  5668. -    }
  5669. -    nfs41_UpcallDestroy(entry);
  5670. -out:
  5671. -#ifdef ENABLE_TIMINGS
  5672. -    t2 = KeQueryPerformanceCounter(NULL);
  5673. -    InterlockedIncrement(&setexattr.tops);
  5674. -    InterlockedAdd64(&setexattr.ticks, t2.QuadPart - t1.QuadPart);
  5675. -#ifdef ENABLE_INDV_TIMINGS
  5676. -    DbgP("nfs41_SetEaInformation delta = %d op=%d sum=%d\n",
  5677. -        t2.QuadPart - t1.QuadPart, setexattr.tops, setexattr.ticks);
  5678. -#endif
  5679. -#endif
  5680. -#ifdef DEBUG_EA_SET
  5681. -    DbgEx();
  5682. -#endif
  5683. -    return status;
  5684. -}
  5685. -
  5686. -static NTSTATUS check_nfs41_queryea_args(
  5687. -    IN PRX_CONTEXT RxContext)
  5688. -{
  5689. -    NTSTATUS status;
  5690. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  5691. -        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  5692. -    __notnull PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs =
  5693. -        &pVNetRootContext->FsAttrs;
  5694. -    PFILE_GET_EA_INFORMATION ea = (PFILE_GET_EA_INFORMATION)
  5695. -            RxContext->CurrentIrpSp->Parameters.QueryEa.EaList;
  5696. -
  5697. -    status = check_nfs41_dirquery_args(RxContext);
  5698. -    if (status) goto out;
  5699. -
  5700. -    if (!(FsAttrs->FileSystemAttributes & FILE_SUPPORTS_EXTENDED_ATTRIBUTES)) {
  5701. -        if (ea == NULL) {
  5702. -            status = STATUS_EAS_NOT_SUPPORTED;
  5703. -            goto out;
  5704. -        }
  5705. -        /* ignore cygwin EAs when checking support */
  5706. -        if (!AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength) &&
  5707. -            !AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength) &&
  5708. -            !AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) {
  5709. -            status = STATUS_EAS_NOT_SUPPORTED;
  5710. -            goto out;
  5711. -        }
  5712. -    }
  5713. -    if ((RxContext->pRelevantSrvOpen->DesiredAccess & FILE_READ_EA) == 0) {
  5714. -        status = STATUS_ACCESS_DENIED;
  5715. -        goto out;
  5716. -    }
  5717. -out:
  5718. -    return status;
  5719. -}
  5720. -
  5721. -static NTSTATUS QueryCygwinSymlink(
  5722. -    IN OUT PRX_CONTEXT RxContext,
  5723. -    IN PFILE_GET_EA_INFORMATION query,
  5724. -    OUT PFILE_FULL_EA_INFORMATION info)
  5725. -{
  5726. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  5727. -    __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext =
  5728. -            NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  5729. -    __notnull PNFS41_NETROOT_EXTENSION NetRootContext =
  5730. -            NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  5731. -    __notnull PNFS41_FOBX Fobx = NFS41GetFobxExtension(RxContext->pFobx);
  5732. -    nfs41_updowncall_entry *entry;
  5733. -    UNICODE_STRING TargetName;
  5734. -    const USHORT HeaderLen = FIELD_OFFSET(FILE_FULL_EA_INFORMATION, EaName) +
  5735. -        query->EaNameLength + 1;
  5736. -    NTSTATUS status;
  5737. -
  5738. -    if (RxContext->Info.LengthRemaining < HeaderLen) {
  5739. -        status = STATUS_BUFFER_TOO_SMALL;
  5740. -        RxContext->InformationToReturn = HeaderLen;
  5741. -        goto out;
  5742. -    }
  5743. -
  5744. -    TargetName.Buffer = (PWCH)(info->EaName + query->EaNameLength + 1);
  5745. -    TargetName.MaximumLength = (USHORT)min(RxContext->Info.LengthRemaining -
  5746. -        HeaderLen, 0xFFFF);
  5747. -
  5748. -    status = nfs41_UpcallCreate(NFS41_SYMLINK, &Fobx->sec_ctx,
  5749. -        VNetRootContext->session, Fobx->nfs41_open_state,
  5750. -        NetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  5751. -    if (status) goto out;
  5752. -
  5753. -    entry->u.Symlink.target = &TargetName;
  5754. -    entry->u.Symlink.set = FALSE;
  5755. -
  5756. -    status = nfs41_UpcallWaitForReply(entry, VNetRootContext->timeout);
  5757. -    if (status) goto out;
  5758. -
  5759. -    status = map_setea_error(entry->status);
  5760. -    if (status == STATUS_SUCCESS) {
  5761. -        info->NextEntryOffset = 0;
  5762. -        info->Flags = 0;
  5763. -        info->EaNameLength = query->EaNameLength;
  5764. -        info->EaValueLength = TargetName.Length - sizeof(UNICODE_NULL);
  5765. -        TargetName.Buffer[TargetName.Length/sizeof(WCHAR)] = UNICODE_NULL;
  5766. -        RtlCopyMemory(info->EaName, query->EaName, query->EaNameLength);
  5767. -        RxContext->Info.LengthRemaining = HeaderLen + info->EaValueLength;
  5768. -    } else if (status == STATUS_BUFFER_TOO_SMALL) {
  5769. -        RxContext->InformationToReturn = (ULONG_PTR)HeaderLen +
  5770. -            entry->u.Symlink.target->Length;
  5771. -    }
  5772. -    nfs41_UpcallDestroy(entry);
  5773. -out:
  5774. -    return status;
  5775. -}
  5776. -
  5777. -static NTSTATUS QueryCygwinEA(
  5778. -    IN OUT PRX_CONTEXT RxContext,
  5779. -    IN PFILE_GET_EA_INFORMATION query,
  5780. -    OUT PFILE_FULL_EA_INFORMATION info)
  5781. -{
  5782. -    NTSTATUS status = STATUS_NONEXISTENT_EA_ENTRY;
  5783. -
  5784. -    if (query == NULL)
  5785. -        goto out;
  5786. -
  5787. -    if (AnsiStrEq(&NfsSymlinkTargetName, query->EaName, query->EaNameLength)) {
  5788. -        __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  5789. -        if (nfs41_fcb->BasicInfo.FileAttributes & FILE_ATTRIBUTE_REPARSE_POINT) {
  5790. -            status = QueryCygwinSymlink(RxContext, query, info);
  5791. -            goto out;
  5792. -        } else {
  5793. -            const LONG LengthRequired = sizeof(FILE_FULL_EA_INFORMATION) +
  5794. -                NfsSymlinkTargetName.Length - sizeof(CHAR);
  5795. -            if (LengthRequired > RxContext->Info.LengthRemaining) {
  5796. -                status = STATUS_BUFFER_TOO_SMALL;
  5797. -                RxContext->InformationToReturn = LengthRequired;
  5798. -                goto out;
  5799. -            }
  5800. -            info->NextEntryOffset = 0;
  5801. -            info->Flags = 0;
  5802. -            info->EaValueLength = 0;
  5803. -            info->EaNameLength = (UCHAR)NfsActOnLink.Length;
  5804. -            RtlCopyMemory(info->EaName, NfsSymlinkTargetName.Buffer,
  5805. -                NfsSymlinkTargetName.Length);
  5806. -            RxContext->Info.LengthRemaining = LengthRequired;
  5807. -            status = STATUS_SUCCESS;
  5808. -            goto out;
  5809. -        }
  5810. -    }
  5811. -
  5812. -    if (AnsiStrEq(&NfsV3Attributes, query->EaName, query->EaNameLength)) {
  5813. -        nfs3_attrs attrs;
  5814. -
  5815. -        const LONG LengthRequired = sizeof(FILE_FULL_EA_INFORMATION) +
  5816. -            NfsV3Attributes.Length + sizeof(nfs3_attrs) - sizeof(CHAR);
  5817. -        if (LengthRequired > RxContext->Info.LengthRemaining) {
  5818. -            status = STATUS_BUFFER_TOO_SMALL;
  5819. -            RxContext->InformationToReturn = LengthRequired;
  5820. -            goto out;
  5821. -        }
  5822. -
  5823. -        create_nfs3_attrs(&attrs, NFS41GetFcbExtension(RxContext->pFcb));
  5824. -#ifdef DEBUG_EA_QUERY
  5825. -        print_nfs3_attrs(&attrs);
  5826. -#endif
  5827. -
  5828. -        info->NextEntryOffset = 0;
  5829. -        info->Flags = 0;
  5830. -        info->EaNameLength = (UCHAR)NfsV3Attributes.Length;
  5831. -        info->EaValueLength = sizeof(nfs3_attrs);
  5832. -        RtlCopyMemory(info->EaName, NfsV3Attributes.Buffer,
  5833. -            NfsV3Attributes.Length);
  5834. -        RtlCopyMemory(info->EaName + info->EaNameLength + 1, &attrs,
  5835. -            sizeof(nfs3_attrs));
  5836. -        RxContext->Info.LengthRemaining = LengthRequired;
  5837. -        status = STATUS_SUCCESS;
  5838. -        goto out;
  5839. -    }
  5840. -
  5841. -    if (AnsiStrEq(&NfsActOnLink, query->EaName, query->EaNameLength)) {
  5842. -
  5843. -        const LONG LengthRequired = sizeof(FILE_FULL_EA_INFORMATION) +
  5844. -            query->EaNameLength - sizeof(CHAR);
  5845. -        if (LengthRequired > RxContext->Info.LengthRemaining) {
  5846. -            status = STATUS_BUFFER_TOO_SMALL;
  5847. -            RxContext->InformationToReturn = LengthRequired;
  5848. -            goto out;
  5849. -        }
  5850. -
  5851. -        info->NextEntryOffset = 0;
  5852. -        info->Flags = 0;
  5853. -        info->EaNameLength = query->EaNameLength;
  5854. -        info->EaValueLength = 0;
  5855. -        RtlCopyMemory(info->EaName, query->EaName, query->EaNameLength);
  5856. -        RxContext->Info.LengthRemaining = LengthRequired;
  5857. -        status = STATUS_SUCCESS;
  5858. -        goto out;
  5859. -    }
  5860. -out:
  5861. -    return status;
  5862. -}
  5863. -
  5864. -static NTSTATUS nfs41_QueryEaInformation(
  5865. -    IN OUT PRX_CONTEXT RxContext)
  5866. -{
  5867. -    NTSTATUS status = STATUS_EAS_NOT_SUPPORTED;
  5868. -    nfs41_updowncall_entry *entry;
  5869. -    PFILE_GET_EA_INFORMATION query = (PFILE_GET_EA_INFORMATION)
  5870. -            RxContext->CurrentIrpSp->Parameters.QueryEa.EaList;
  5871. -    ULONG buflen = RxContext->CurrentIrpSp->Parameters.QueryEa.Length;
  5872. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  5873. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  5874. -            NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  5875. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  5876. -            NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  5877. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  5878. -#ifdef ENABLE_TIMINGS
  5879. -    LARGE_INTEGER t1, t2;
  5880. -    t1 = KeQueryPerformanceCounter(NULL);
  5881. -#endif
  5882. -
  5883. -#ifdef DEBUG_EA_QUERY
  5884. -    DbgEn();
  5885. -    print_debug_header(RxContext);
  5886. -    print_get_ea(1, query);
  5887. -#endif
  5888. -    status = check_nfs41_queryea_args(RxContext);
  5889. -    if (status) goto out;
  5890. -
  5891. -    /* handle queries for cygwin EAs */
  5892. -    status = QueryCygwinEA(RxContext, query,
  5893. -        (PFILE_FULL_EA_INFORMATION)RxContext->Info.Buffer);
  5894. -    if (status != STATUS_NONEXISTENT_EA_ENTRY)
  5895. -        goto out;
  5896. -
  5897. -    status = nfs41_UpcallCreate(NFS41_EA_GET, &nfs41_fobx->sec_ctx,
  5898. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  5899. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  5900. -    if (status) goto out;
  5901. -
  5902. -    entry->buf_len = buflen;
  5903. -    entry->buf = RxContext->Info.Buffer;
  5904. -    entry->u.QueryEa.EaList = query;
  5905. -    entry->u.QueryEa.EaListLength = query == NULL ? 0 :
  5906. -        RxContext->QueryEa.UserEaListLength;
  5907. -    entry->u.QueryEa.EaIndex = RxContext->QueryEa.IndexSpecified ?
  5908. -        RxContext->QueryEa.UserEaIndex : 0;
  5909. -    entry->u.QueryEa.RestartScan = RxContext->QueryEa.RestartScan;
  5910. -    entry->u.QueryEa.ReturnSingleEntry = RxContext->QueryEa.ReturnSingleEntry;
  5911. -
  5912. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  5913. -    if (status) goto out;
  5914. -
  5915. -    if (entry->status == STATUS_SUCCESS) {
  5916. -        switch (entry->u.QueryEa.Overflow) {
  5917. -        case ERROR_INSUFFICIENT_BUFFER:
  5918. -            status = STATUS_BUFFER_TOO_SMALL;
  5919. -            break;
  5920. -        case ERROR_BUFFER_OVERFLOW:
  5921. -            status = RxContext->IoStatusBlock.Status = STATUS_BUFFER_OVERFLOW;
  5922. -            break;
  5923. -        default:
  5924. -            RxContext->IoStatusBlock.Status = STATUS_SUCCESS;
  5925. -            break;
  5926. -        }
  5927. -        RxContext->InformationToReturn = entry->buf_len;
  5928. -#ifdef ENABLE_TIMINGS
  5929. -        InterlockedIncrement(&getexattr.sops);
  5930. -        InterlockedAdd64(&getexattr.size, entry->u.QueryEa.buf_len);
  5931. -#endif
  5932. -    } else {
  5933. -        status = map_setea_error(entry->status);
  5934. -    }
  5935. -    nfs41_UpcallDestroy(entry);
  5936. -out:
  5937. -#ifdef ENABLE_TIMINGS
  5938. -    t2 = KeQueryPerformanceCounter(NULL);
  5939. -    InterlockedIncrement(&getexattr.tops);
  5940. -    InterlockedAdd64(&getexattr.ticks, t2.QuadPart - t1.QuadPart);
  5941. -#ifdef ENABLE_INDV_TIMINGS
  5942. -    DbgP("nfs41_QueryEaInformation delta = %d op=%d sum=%d\n",
  5943. -        t2.QuadPart - t1.QuadPart, getexattr.tops, getexattr.ticks);
  5944. -#endif
  5945. -#endif
  5946. -#ifdef DEBUG_EA_QUERY
  5947. -    DbgEx();
  5948. -#endif
  5949. -    return status;
  5950. -}
  5951. -
  5952. -static NTSTATUS map_query_acl_error(
  5953. -    DWORD error)
  5954. -{
  5955. -    switch (error) {
  5956. -    case NO_ERROR:                  return STATUS_SUCCESS;
  5957. -    case ERROR_NOT_SUPPORTED:       return STATUS_NOT_SUPPORTED;
  5958. -    case ERROR_NONE_MAPPED:         return STATUS_NONE_MAPPED;
  5959. -    case ERROR_ACCESS_DENIED:       return STATUS_ACCESS_DENIED;
  5960. -    case ERROR_FILE_NOT_FOUND:      return STATUS_OBJECT_NAME_NOT_FOUND;
  5961. -    case ERROR_INVALID_PARAMETER:   return STATUS_INVALID_PARAMETER;
  5962. -    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  5963. -    default:
  5964. -        print_error("map_query_acl_error: "
  5965. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  5966. -            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", error);
  5967. -    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  5968. -    }
  5969. -}
  5970. -
  5971. -static NTSTATUS check_nfs41_getacl_args(
  5972. -    PRX_CONTEXT RxContext)
  5973. -{
  5974. -    NTSTATUS status = STATUS_SUCCESS;
  5975. -    SECURITY_INFORMATION info_class =
  5976. -        RxContext->CurrentIrpSp->Parameters.QuerySecurity.SecurityInformation;
  5977. -
  5978. -    /* we don't support sacls */
  5979. -    if (info_class == SACL_SECURITY_INFORMATION ||
  5980. -            info_class == LABEL_SECURITY_INFORMATION) {
  5981. -        status = STATUS_NOT_SUPPORTED;
  5982. -        goto out;
  5983. -    }
  5984. -    if (RxContext->CurrentIrp->UserBuffer == NULL &&
  5985. -            RxContext->CurrentIrpSp->Parameters.QuerySecurity.Length)
  5986. -        status = STATUS_INVALID_USER_BUFFER;
  5987. -out:
  5988. -    return status;
  5989. -}
  5990. -
  5991. -static NTSTATUS nfs41_QuerySecurityInformation(
  5992. -    IN OUT PRX_CONTEXT RxContext)
  5993. -{
  5994. -    NTSTATUS status = STATUS_NOT_SUPPORTED;
  5995. -    nfs41_updowncall_entry *entry;
  5996. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  5997. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  5998. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  5999. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  6000. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  6001. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  6002. -    SECURITY_INFORMATION info_class =
  6003. -        RxContext->CurrentIrpSp->Parameters.QuerySecurity.SecurityInformation;
  6004. -#ifdef ENABLE_TIMINGS
  6005. -    LARGE_INTEGER t1, t2;
  6006. -    t1 = KeQueryPerformanceCounter(NULL);
  6007. -#endif
  6008. -
  6009. -#ifdef DEBUG_ACL_QUERY
  6010. -    DbgEn();
  6011. -    print_debug_header(RxContext);
  6012. -    print_acl_args(info_class);
  6013. -#endif
  6014. -
  6015. -    status = check_nfs41_getacl_args(RxContext);
  6016. -    if (status) goto out;
  6017. -
  6018. -    if (nfs41_fobx->acl && nfs41_fobx->acl_len) {
  6019. -        LARGE_INTEGER current_time;
  6020. -        KeQuerySystemTime(&current_time);
  6021. -#ifdef DEBUG_ACL_QUERY
  6022. -        DbgP("CurrentTime 0x%lx Saved Acl time 0x%lx\n",
  6023. -            current_time.QuadPart, nfs41_fobx->time.QuadPart);
  6024. -#endif
  6025. -        if (current_time.QuadPart - nfs41_fobx->time.QuadPart <= 20*1000) {
  6026. -            PSECURITY_DESCRIPTOR sec_desc = (PSECURITY_DESCRIPTOR)
  6027. -                RxContext->CurrentIrp->UserBuffer;
  6028. -            RtlCopyMemory(sec_desc, nfs41_fobx->acl, nfs41_fobx->acl_len);
  6029. -            RxContext->IoStatusBlock.Information =
  6030. -                RxContext->InformationToReturn = nfs41_fobx->acl_len;
  6031. -            RxContext->IoStatusBlock.Status = status = STATUS_SUCCESS;
  6032. -#ifdef ENABLE_TIMINGS
  6033. -            InterlockedIncrement(&getacl.sops);
  6034. -            InterlockedAdd64(&getacl.size, nfs41_fobx->acl_len);
  6035. -#endif
  6036. -        } else status = 1;
  6037. -        RxFreePool(nfs41_fobx->acl);
  6038. -        nfs41_fobx->acl = NULL;
  6039. -        nfs41_fobx->acl_len = 0;
  6040. -        if (!status)
  6041. -            goto out;
  6042. -    }
  6043. -
  6044. -    status = nfs41_UpcallCreate(NFS41_ACL_QUERY, &nfs41_fobx->sec_ctx,
  6045. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  6046. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  6047. -    if (status) goto out;
  6048. -
  6049. -    entry->u.Acl.query = info_class;
  6050. -    /* we can't provide RxContext->CurrentIrp->UserBuffer to the upcall thread
  6051. -     * because it becomes an invalid pointer with that execution context
  6052. -     */
  6053. -    entry->buf_len = RxContext->CurrentIrpSp->Parameters.QuerySecurity.Length;
  6054. -
  6055. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  6056. -    if (status) goto out;
  6057. -
  6058. -    if (entry->status == STATUS_BUFFER_TOO_SMALL) {
  6059. -#ifdef DEBUG_ACL_QUERY
  6060. -        DbgP("nfs41_QuerySecurityInformation: provided buffer size=%d but we "
  6061. -             "need %lu\n",
  6062. -             RxContext->CurrentIrpSp->Parameters.QuerySecurity.Length,
  6063. -             entry->buf_len);
  6064. -#endif
  6065. -        status = STATUS_BUFFER_OVERFLOW;
  6066. -        RxContext->InformationToReturn = entry->buf_len;
  6067. -
  6068. -        /* Save ACL buffer */
  6069. -        nfs41_fobx->acl = entry->buf;
  6070. -        nfs41_fobx->acl_len = entry->buf_len;
  6071. -        KeQuerySystemTime(&nfs41_fobx->time);
  6072. -    } else if (entry->status == STATUS_SUCCESS) {
  6073. -        PSECURITY_DESCRIPTOR sec_desc = (PSECURITY_DESCRIPTOR)
  6074. -            RxContext->CurrentIrp->UserBuffer;
  6075. -        RtlCopyMemory(sec_desc, entry->buf, entry->buf_len);
  6076. -#ifdef ENABLE_TIMINGS
  6077. -        InterlockedIncrement(&getacl.sops);
  6078. -        InterlockedAdd64(&getacl.size, entry->u.Acl.buf_len);
  6079. -#endif
  6080. -        RxFreePool(entry->buf);
  6081. -        entry->buf = NULL;
  6082. -        nfs41_fobx->acl = NULL;
  6083. -        nfs41_fobx->acl_len = 0;
  6084. -        RxContext->IoStatusBlock.Information = RxContext->InformationToReturn =
  6085. -            entry->buf_len;
  6086. -        RxContext->IoStatusBlock.Status = status = STATUS_SUCCESS;
  6087. -    } else {
  6088. -        status = map_query_acl_error(entry->status);
  6089. -    }
  6090. -    nfs41_UpcallDestroy(entry);
  6091. -out:
  6092. -#ifdef ENABLE_TIMINGS
  6093. -    t2 = KeQueryPerformanceCounter(NULL);
  6094. -    /* only count getacl that we made an upcall for */
  6095. -    if (status == STATUS_BUFFER_OVERFLOW) {
  6096. -        InterlockedIncrement(&getacl.tops);
  6097. -        InterlockedAdd64(&getacl.ticks, t2.QuadPart - t1.QuadPart);
  6098. -    }
  6099. -#ifdef ENABLE_INDV_TIMINGS
  6100. -    DbgP("nfs41_QuerySecurityInformation: delta = %d op=%d sum=%d\n",
  6101. -        t2.QuadPart - t1.QuadPart, getacl.tops, getacl.ticks);
  6102. -#endif
  6103. -#endif
  6104. -#ifdef DEBUG_ACL_QUERY
  6105. -    DbgEx();
  6106. -#endif
  6107. -    return status;
  6108. -}
  6109. -
  6110. -static NTSTATUS check_nfs41_setacl_args(
  6111. -    PRX_CONTEXT RxContext)
  6112. -{
  6113. -    NTSTATUS status = STATUS_SUCCESS;
  6114. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  6115. -        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  6116. -    SECURITY_INFORMATION info_class =
  6117. -        RxContext->CurrentIrpSp->Parameters.SetSecurity.SecurityInformation;
  6118. -
  6119. -    if (pVNetRootContext->read_only) {
  6120. -        print_error("check_nfs41_setacl_args: Read-only mount\n");
  6121. -        status = STATUS_MEDIA_WRITE_PROTECTED;
  6122. -        goto out;
  6123. -    }
  6124. -    /* we don't support sacls */
  6125. -    if (info_class == SACL_SECURITY_INFORMATION  ||
  6126. -            info_class == LABEL_SECURITY_INFORMATION) {
  6127. -        status = STATUS_NOT_SUPPORTED;      
  6128. -        goto out;
  6129. -    }
  6130. -out:
  6131. -    return status;
  6132. -}
  6133. -
  6134. -static NTSTATUS nfs41_SetSecurityInformation(
  6135. -    IN OUT PRX_CONTEXT RxContext)
  6136. -{
  6137. -    NTSTATUS status = STATUS_NOT_SUPPORTED;
  6138. -    nfs41_updowncall_entry *entry;
  6139. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  6140. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  6141. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  6142. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  6143. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  6144. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  6145. -    __notnull PSECURITY_DESCRIPTOR sec_desc =
  6146. -        RxContext->CurrentIrpSp->Parameters.SetSecurity.SecurityDescriptor;
  6147. -    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  6148. -    SECURITY_INFORMATION info_class =
  6149. -        RxContext->CurrentIrpSp->Parameters.SetSecurity.SecurityInformation;
  6150. -#ifdef ENABLE_TIMINGS
  6151. -    LARGE_INTEGER t1, t2;
  6152. -    t1 = KeQueryPerformanceCounter(NULL);
  6153. -#endif
  6154. -
  6155. -#ifdef DEBUG_ACL_SET
  6156. -    DbgEn();
  6157. -    print_debug_header(RxContext);
  6158. -    print_acl_args(info_class);
  6159. -#endif
  6160. -
  6161. -    status = check_nfs41_setacl_args(RxContext);
  6162. -    if (status) goto out;
  6163. -
  6164. -    /* check that ACL is present */
  6165. -    if (info_class & DACL_SECURITY_INFORMATION) {
  6166. -        PACL acl;
  6167. -        BOOLEAN present, dacl_default;
  6168. -        status = RtlGetDaclSecurityDescriptor(sec_desc, &present, &acl,
  6169. -                    &dacl_default);
  6170. -        if (status) {
  6171. -            DbgP("RtlGetDaclSecurityDescriptor failed 0x%x\n", status);
  6172. -            goto out;
  6173. -        }
  6174. -        if (present == FALSE) {
  6175. -            DbgP("NO ACL present\n");
  6176. -            goto out;
  6177. -        }
  6178. -    }
  6179. -
  6180. -    status = nfs41_UpcallCreate(NFS41_ACL_SET, &nfs41_fobx->sec_ctx,
  6181. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  6182. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  6183. -    if (status) goto out;
  6184. -
  6185. -    entry->u.Acl.query = info_class;
  6186. -    entry->buf = sec_desc;
  6187. -    entry->buf_len = RtlLengthSecurityDescriptor(sec_desc);
  6188. -#ifdef ENABLE_TIMINGS
  6189. -    InterlockedIncrement(&setacl.sops);
  6190. -    InterlockedAdd64(&setacl.size, entry->u.Acl.buf_len);    
  6191. -#endif
  6192. -
  6193. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  6194. -    if (status) goto out;
  6195. -
  6196. -    status = map_query_acl_error(entry->status);
  6197. -    if (!status) {
  6198. -        if (!nfs41_fobx->deleg_type && entry->ChangeTime &&
  6199. -                (SrvOpen->DesiredAccess &
  6200. -                (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA)))
  6201. -            nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime);
  6202. -        nfs41_fcb->changeattr = entry->ChangeTime;
  6203. -    }
  6204. -    nfs41_UpcallDestroy(entry);
  6205. -out:
  6206. -#ifdef ENABLE_TIMINGS
  6207. -    t2 = KeQueryPerformanceCounter(NULL);
  6208. -    InterlockedIncrement(&setacl.tops);
  6209. -    InterlockedAdd64(&setacl.ticks, t2.QuadPart - t1.QuadPart);
  6210. -#ifdef ENABLE_INDV_TIMINGS
  6211. -    DbgP("nfs41_SetSecurityInformation delta = %d op=%d sum=%d\n",
  6212. -        t2.QuadPart - t1.QuadPart, setacl.tops, setacl.ticks);
  6213. -#endif
  6214. -#endif
  6215. -#ifdef DEBUG_ACL_SET
  6216. -    DbgEx();
  6217. -#endif
  6218. -    return status;
  6219. -}
  6220. -
  6221. -static NTSTATUS map_queryfile_error(
  6222. -    DWORD error)
  6223. -{
  6224. -    switch (error) {
  6225. -    case ERROR_ACCESS_DENIED:       return STATUS_ACCESS_DENIED;
  6226. -    case ERROR_NETNAME_DELETED:     return STATUS_NETWORK_NAME_DELETED;
  6227. -    case ERROR_INVALID_PARAMETER:   return STATUS_INVALID_PARAMETER;
  6228. -    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  6229. -    default:
  6230. -        print_error("map_queryfile_error: "
  6231. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  6232. -            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", error);
  6233. -    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  6234. -    }
  6235. -}
  6236. -
  6237. -static NTSTATUS nfs41_QueryFileInformation(
  6238. -    IN OUT PRX_CONTEXT RxContext)
  6239. -{
  6240. -    NTSTATUS status = STATUS_OBJECT_NAME_NOT_FOUND;
  6241. -    FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass;
  6242. -    nfs41_updowncall_entry *entry;
  6243. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  6244. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  6245. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  6246. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  6247. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  6248. -    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  6249. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  6250. -#ifdef ENABLE_TIMINGS
  6251. -    LARGE_INTEGER t1, t2;
  6252. -    t1 = KeQueryPerformanceCounter(NULL);
  6253. -#endif
  6254. -
  6255. -#ifdef DEBUG_FILE_QUERY
  6256. -    DbgEn();
  6257. -    print_debug_filedirquery_header(RxContext);
  6258. -    DbgP("--> nfs41_QueryFileInformation, RxContext->Info.LengthRemaining=%ld\n",
  6259. -        (long)RxContext->Info.LengthRemaining);
  6260. -#endif
  6261. -
  6262. -    status = check_nfs41_dirquery_args(RxContext);
  6263. -    if (status) {
  6264. -        print_error("check_nfs41_dirquery_args failed.\n");
  6265. -        goto out;
  6266. -    }
  6267. -
  6268. -    RtlZeroMemory(RxContext->Info.Buffer, RxContext->Info.LengthRemaining);
  6269. -
  6270. -#ifdef DEBUG_FILE_QUERY
  6271. -    DbgP("nfs41_QueryFileInformation, RxContext->Info.LengthRemaining=%ld\n",
  6272. -        (long)RxContext->Info.LengthRemaining);
  6273. -#endif
  6274. -
  6275. -    switch (InfoClass) {
  6276. -    case FileEaInformation:
  6277. -    {
  6278. -        if (RxContext->Info.LengthRemaining <
  6279. -            sizeof(FILE_EA_INFORMATION)) {
  6280. -            print_error("nfs41_QueryFileInformation: "
  6281. -                "FILE_EA_INFORMATION buffer too small\n");
  6282. -            status = STATUS_BUFFER_TOO_SMALL;
  6283. -            goto out;
  6284. -        }
  6285. -
  6286. -        PFILE_EA_INFORMATION info =
  6287. -            (PFILE_EA_INFORMATION)RxContext->Info.Buffer;
  6288. -        info->EaSize = 0;
  6289. -        RxContext->Info.LengthRemaining -= sizeof(FILE_EA_INFORMATION);
  6290. -        status = STATUS_SUCCESS;
  6291. -        goto out;
  6292. -    }
  6293. -    case FileRemoteProtocolInformation:
  6294. -    {
  6295. -        if (RxContext->Info.LengthRemaining <
  6296. -            sizeof(FILE_REMOTE_PROTOCOL_INFORMATION)) {
  6297. -            print_error("nfs41_QueryFileInformation: "
  6298. -                "FILE_REMOTE_PROTOCOL_INFORMATION buffer too small\n");
  6299. -            status = STATUS_BUFFER_TOO_SMALL;
  6300. -            goto out;
  6301. -        }
  6302. -
  6303. -        PFILE_REMOTE_PROTOCOL_INFORMATION info =
  6304. -            (PFILE_REMOTE_PROTOCOL_INFORMATION)RxContext->Info.Buffer;
  6305. -
  6306. -        (void)RtlZeroMemory(info,
  6307. -            sizeof(FILE_REMOTE_PROTOCOL_INFORMATION));
  6308. -        info->StructureVersion = 1;
  6309. -        info->StructureSize = sizeof(FILE_REMOTE_PROTOCOL_INFORMATION);
  6310. -        info->Protocol = WNNC_NET_RDR2SAMPLE; /* FIXME! */
  6311. -        /*
  6312. -         * ToDo: If we add NFSv4.1/NFSv4.2 protocol negotiation, then
  6313. -         * we need to call the userland daemon to return the correct
  6314. -         * protocol minor version
  6315. -         */
  6316. -        info->ProtocolMajorVersion = 4;
  6317. -        info->ProtocolMinorVersion = 1;
  6318. -        info->ProtocolRevision = 0;
  6319. -        RxContext->Info.LengthRemaining -=
  6320. -            sizeof(FILE_REMOTE_PROTOCOL_INFORMATION);
  6321. -        status = STATUS_SUCCESS;
  6322. -        goto out;
  6323. -    }
  6324. -    case FileCaseSensitiveInformation:
  6325. -    {
  6326. -        if (RxContext->Info.LengthRemaining <
  6327. -            sizeof(FILE_CASE_SENSITIVE_INFORMATION)) {
  6328. -            print_error("nfs41_QueryFileInformation: "
  6329. -                "FILE_CASE_SENSITIVE_INFORMATION buffer too small\n");
  6330. -            status = STATUS_BUFFER_TOO_SMALL;
  6331. -            goto out;
  6332. -        }
  6333. -
  6334. -        PFILE_CASE_SENSITIVE_INFORMATION info =
  6335. -            (PFILE_CASE_SENSITIVE_INFORMATION)RxContext->Info.Buffer;
  6336. -
  6337. -        ULONG fsattrs = pVNetRootContext->FsAttrs.FileSystemAttributes;
  6338. -
  6339. -        /*
  6340. -         * For NFSv4.1 |FATTR4_WORD0_CASE_INSENSITIVE| used
  6341. -         * to fill |FsAttrs.FileSystemAttributes| is per
  6342. -         * filesystem.
  6343. -         * FIXME: Future NFSv4.x standards should make this a
  6344. -         * per-filesystem, per-directory and
  6345. -         * per-extended-attribute-dir attribute to support
  6346. -         * Win32
  6347. -         */
  6348. -        if (fsattrs & FILE_CASE_SENSITIVE_SEARCH) {
  6349. -            info->Flags = FILE_CS_FLAG_CASE_SENSITIVE_DIR;
  6350. -        }
  6351. -
  6352. -        RxContext->Info.LengthRemaining -=
  6353. -            sizeof(FILE_CASE_SENSITIVE_INFORMATION);
  6354. -        status = STATUS_SUCCESS;
  6355. -        goto out;
  6356. -    }
  6357. -    case FileBasicInformation:
  6358. -    case FileStandardInformation:
  6359. -    case FileInternalInformation:
  6360. -    case FileAttributeTagInformation:
  6361. -    case FileNetworkOpenInformation:
  6362. -        break;
  6363. -    default:
  6364. -        print_error("nfs41_QueryFileInformation: unhandled class %d\n", InfoClass);
  6365. -        status = STATUS_NOT_SUPPORTED;
  6366. -        goto out;
  6367. -    }
  6368. -
  6369. -    status = nfs41_UpcallCreate(NFS41_FILE_QUERY, &nfs41_fobx->sec_ctx,
  6370. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  6371. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  6372. -    if (status) {
  6373. -        print_error("nfs41_UpcallCreate() failed, status=0x%lx\n",
  6374. -            (long)status);
  6375. -        goto out;
  6376. -    }
  6377. -
  6378. -    entry->u.QueryFile.InfoClass = InfoClass;
  6379. -    entry->buf = RxContext->Info.Buffer;
  6380. -    entry->buf_len = RxContext->Info.LengthRemaining;
  6381. -
  6382. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  6383. -    if (status) {
  6384. -        print_error("nfs41_UpcallWaitForReply() failed, status=0x%lx\n",
  6385. -            (long)status);
  6386. -        goto out;
  6387. -    }
  6388. -
  6389. -    if (entry->status == STATUS_BUFFER_TOO_SMALL) {
  6390. -        RxContext->InformationToReturn = entry->buf_len;
  6391. -        print_error("entry->status == STATUS_BUFFER_TOO_SMALL\n");
  6392. -        status = STATUS_BUFFER_TOO_SMALL;
  6393. -    } else if (entry->status == STATUS_SUCCESS) {
  6394. -#ifdef DEBUG_FILE_QUERY
  6395. -        print_error("entry->status == STATUS_SUCCESS\n");
  6396. -#endif
  6397. -        BOOLEAN DeletePending = FALSE;
  6398. -#ifdef ENABLE_TIMINGS
  6399. -        InterlockedIncrement(&getattr.sops);
  6400. -        InterlockedAdd64(&getattr.size, entry->u.QueryFile.buf_len);
  6401. -#endif
  6402. -        RxContext->Info.LengthRemaining -= entry->buf_len;
  6403. -        status = STATUS_SUCCESS;
  6404. -
  6405. -        switch (InfoClass) {
  6406. -        case FileBasicInformation:
  6407. -            RtlCopyMemory(&nfs41_fcb->BasicInfo, RxContext->Info.Buffer,
  6408. -                sizeof(nfs41_fcb->BasicInfo));
  6409. -#ifdef DEBUG_FILE_QUERY
  6410. -            print_basic_info(1, &nfs41_fcb->BasicInfo);
  6411. -#endif
  6412. -            break;
  6413. -        case FileStandardInformation:
  6414. -            /* this a fix for RDBSS behaviour when it first calls ExtendForCache,
  6415. -             * then it sends a file query irp for standard attributes and
  6416. -             * expects to receive EndOfFile of value set by the ExtendForCache.
  6417. -             * It seems to cache the filesize based on that instead of sending
  6418. -             * a file size query for after doing the write.
  6419. -             */
  6420. -        {
  6421. -            PFILE_STANDARD_INFORMATION std_info;
  6422. -            std_info = (PFILE_STANDARD_INFORMATION)RxContext->Info.Buffer;
  6423. -            if (nfs41_fcb->StandardInfo.AllocationSize.QuadPart >
  6424. -                    std_info->AllocationSize.QuadPart) {
  6425. -#ifdef DEBUG_FILE_QUERY
  6426. -                DbgP("Old AllocationSize is bigger: saving 0x%x\n",
  6427. -                    nfs41_fcb->StandardInfo.AllocationSize.QuadPart);
  6428. -#endif
  6429. -                std_info->AllocationSize.QuadPart =
  6430. -                    nfs41_fcb->StandardInfo.AllocationSize.QuadPart;
  6431. -            }
  6432. -            if (nfs41_fcb->StandardInfo.EndOfFile.QuadPart >
  6433. -                    std_info->EndOfFile.QuadPart) {
  6434. -#ifdef DEBUG_FILE_QUERY
  6435. -                DbgP("Old EndOfFile is bigger: saving 0x%x\n",
  6436. -                    nfs41_fcb->StandardInfo.EndOfFile);
  6437. -#endif
  6438. -                std_info->EndOfFile.QuadPart =
  6439. -                    nfs41_fcb->StandardInfo.EndOfFile.QuadPart;
  6440. -            }
  6441. -            std_info->DeletePending = nfs41_fcb->DeletePending;
  6442. -        }
  6443. -            if (nfs41_fcb->StandardInfo.DeletePending)
  6444. -                DeletePending = TRUE;
  6445. -            RtlCopyMemory(&nfs41_fcb->StandardInfo, RxContext->Info.Buffer,
  6446. -                sizeof(nfs41_fcb->StandardInfo));
  6447. -            nfs41_fcb->StandardInfo.DeletePending = DeletePending;
  6448. -#ifdef DEBUG_FILE_QUERY
  6449. -            print_std_info(1, &nfs41_fcb->StandardInfo);
  6450. -#endif
  6451. -            break;
  6452. -        case FileNetworkOpenInformation:
  6453. -        case FileInternalInformation:
  6454. -        case FileAttributeTagInformation:
  6455. -            break;
  6456. -        default:
  6457. -            print_error("Unhandled/unsupported InfoClass(%d)\n", (int)InfoClass);
  6458. -        }
  6459. -    } else {
  6460. -        status = map_queryfile_error(entry->status);
  6461. -        print_error("status(0x%lx) = map_queryfile_error(entry->status(0x%lx));\n",
  6462. -            (long)status, (long)entry->status);
  6463. -    }
  6464. -    nfs41_UpcallDestroy(entry);
  6465. -out:
  6466. -#ifdef ENABLE_TIMINGS
  6467. -    t2 = KeQueryPerformanceCounter(NULL);
  6468. -    InterlockedIncrement(&getattr.tops);
  6469. -    InterlockedAdd64(&getattr.ticks, t2.QuadPart - t1.QuadPart);
  6470. -#ifdef ENABLE_INDV_TIMINGS
  6471. -    DbgP("nfs41_QueryFileInformation delta = %d op=%d sum=%d\n",
  6472. -        t2.QuadPart - t1.QuadPart, getattr.tops, getattr.ticks);
  6473. -#endif
  6474. -#endif
  6475. -#ifdef DEBUG_FILE_QUERY
  6476. -    DbgEx();
  6477. -    DbgP("<-- nfs41_QueryFileInformation, status=0x%lx\n", (long)status);
  6478. -#endif
  6479. -    return status;
  6480. -}
  6481. -
  6482. -static NTSTATUS map_setfile_error(
  6483. -    DWORD error)
  6484. -{
  6485. -    switch (error) {
  6486. -    case NO_ERROR:                      return STATUS_SUCCESS;
  6487. -    case ERROR_NOT_EMPTY:               return STATUS_DIRECTORY_NOT_EMPTY;
  6488. -    case ERROR_FILE_EXISTS:             return STATUS_OBJECT_NAME_COLLISION;
  6489. -    case ERROR_FILE_NOT_FOUND:          return STATUS_OBJECT_NAME_NOT_FOUND;
  6490. -    case ERROR_PATH_NOT_FOUND:          return STATUS_OBJECT_PATH_NOT_FOUND;
  6491. -    case ERROR_ACCESS_DENIED:           return STATUS_ACCESS_DENIED;
  6492. -    case ERROR_FILE_INVALID:            return STATUS_FILE_INVALID;
  6493. -    case ERROR_NOT_SAME_DEVICE:         return STATUS_NOT_SAME_DEVICE;
  6494. -    case ERROR_NOT_SUPPORTED:           return STATUS_NOT_IMPLEMENTED;
  6495. -    case ERROR_NETWORK_ACCESS_DENIED:   return STATUS_NETWORK_ACCESS_DENIED;
  6496. -    case ERROR_NETNAME_DELETED:         return STATUS_NETWORK_NAME_DELETED;
  6497. -    case ERROR_BUFFER_OVERFLOW:         return STATUS_INSUFFICIENT_RESOURCES;
  6498. -    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  6499. -    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  6500. -    case ERROR_FILE_TOO_LARGE:          return STATUS_FILE_TOO_LARGE;
  6501. -    case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  6502. -    default:
  6503. -        print_error("map_setfile_error: "
  6504. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  6505. -            "defaulting to STATUS_INVALID_PARAMETER\n", error);
  6506. -    case ERROR_INVALID_PARAMETER:       return STATUS_INVALID_PARAMETER;
  6507. -    }
  6508. -}
  6509. -
  6510. -static NTSTATUS check_nfs41_setattr_args(
  6511. -    IN PRX_CONTEXT RxContext)
  6512. -{
  6513. -    NTSTATUS status = STATUS_SUCCESS;
  6514. -    FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass;
  6515. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  6516. -        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  6517. -
  6518. -    if (pVNetRootContext->read_only) {
  6519. -        print_error("check_nfs41_setattr_args: Read-only mount\n");
  6520. -        status = STATUS_MEDIA_WRITE_PROTECTED;
  6521. -        goto out;
  6522. -    }
  6523. -
  6524. -    /* http://msdn.microsoft.com/en-us/library/ff469355(v=PROT.10).aspx
  6525. -     * http://msdn.microsoft.com/en-us/library/ff469424(v=PROT.10).aspx
  6526. -     * If Open.GrantedAccess does not contain FILE_WRITE_DATA, the operation
  6527. -     * MUST be failed with STATUS_ACCESS_DENIED.
  6528. -     */
  6529. -    if (InfoClass == FileAllocationInformation ||
  6530. -            InfoClass == FileEndOfFileInformation) {
  6531. -        if (!(RxContext->pRelevantSrvOpen->DesiredAccess & FILE_WRITE_DATA)) {
  6532. -            status = STATUS_ACCESS_DENIED;
  6533. -            goto out;
  6534. -        }
  6535. -    }
  6536. -    status = check_nfs41_dirquery_args(RxContext);
  6537. -    if (status) goto out;
  6538. -
  6539. -    switch (InfoClass) {
  6540. -    case FileRenameInformation:
  6541. -    {
  6542. -        PFILE_RENAME_INFORMATION rinfo =
  6543. -            (PFILE_RENAME_INFORMATION)RxContext->Info.Buffer;
  6544. -        UNICODE_STRING dst = { (USHORT)rinfo->FileNameLength,
  6545. -            (USHORT)rinfo->FileNameLength, rinfo->FileName };
  6546. -#ifdef DEBUG_FILE_SET
  6547. -        DbgP("Attempting to rename to '%wZ'\n", &dst);
  6548. -#endif
  6549. -        if (isFilenameTooLong(&dst, pVNetRootContext)) {
  6550. -            status = STATUS_OBJECT_NAME_INVALID;
  6551. -            goto out;
  6552. -        }
  6553. -        if (rinfo->RootDirectory) {
  6554. -            status = STATUS_INVALID_PARAMETER;
  6555. -            goto out;
  6556. -        }
  6557. -        break;
  6558. -    }
  6559. -    case FileLinkInformation:
  6560. -    {
  6561. -        PFILE_LINK_INFORMATION linfo =
  6562. -            (PFILE_LINK_INFORMATION)RxContext->Info.Buffer;
  6563. -        UNICODE_STRING dst = { (USHORT)linfo->FileNameLength,
  6564. -            (USHORT)linfo->FileNameLength, linfo->FileName };
  6565. -#ifdef DEBUG_FILE_SET
  6566. -        DbgP("Attempting to add link as '%wZ'\n", &dst);
  6567. -#endif
  6568. -        if (isFilenameTooLong(&dst, pVNetRootContext)) {
  6569. -            status = STATUS_OBJECT_NAME_INVALID;
  6570. -            goto out;
  6571. -        }
  6572. -        if (linfo->RootDirectory) {
  6573. -            status = STATUS_INVALID_PARAMETER;
  6574. -            goto out;
  6575. -        }
  6576. -        break;
  6577. -    }
  6578. -    case FileDispositionInformation:
  6579. -    {
  6580. -        PFILE_DISPOSITION_INFORMATION dinfo =
  6581. -            (PFILE_DISPOSITION_INFORMATION)RxContext->Info.Buffer;
  6582. -        __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  6583. -        if (dinfo->DeleteFile && nfs41_fcb->DeletePending) {
  6584. -            status = STATUS_DELETE_PENDING;
  6585. -            goto out;
  6586. -        }
  6587. -        break;
  6588. -    }
  6589. -    case FileBasicInformation:
  6590. -    case FileAllocationInformation:
  6591. -    case FileEndOfFileInformation:
  6592. -        break;
  6593. -    default:
  6594. -        print_error("nfs41_SetFileInformation: unhandled class %d\n", InfoClass);
  6595. -        status = STATUS_NOT_SUPPORTED;
  6596. -    }
  6597. -
  6598. -out:
  6599. -    return status;
  6600. -}
  6601. -
  6602. -static NTSTATUS nfs41_SetFileInformation(
  6603. -    IN OUT PRX_CONTEXT RxContext)
  6604. -{
  6605. -    NTSTATUS status = STATUS_INVALID_PARAMETER;
  6606. -    nfs41_updowncall_entry *entry;
  6607. -    FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass;
  6608. -    FILE_RENAME_INFORMATION rinfo;
  6609. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  6610. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  6611. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  6612. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  6613. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  6614. -    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  6615. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  6616. -#ifdef ENABLE_TIMINGS
  6617. -    LARGE_INTEGER t1, t2;
  6618. -    t1 = KeQueryPerformanceCounter(NULL);
  6619. -#endif
  6620. -
  6621. -#ifdef DEBUG_FILE_SET
  6622. -    DbgEn();
  6623. -    print_debug_filedirquery_header(RxContext);
  6624. -#endif
  6625. -
  6626. -    status = check_nfs41_setattr_args(RxContext);
  6627. -    if (status) goto out;
  6628. -
  6629. -    switch (InfoClass) {
  6630. -    case FileDispositionInformation:
  6631. -        {
  6632. -            PFILE_DISPOSITION_INFORMATION dinfo =
  6633. -                (PFILE_DISPOSITION_INFORMATION)RxContext->Info.Buffer;
  6634. -            if (dinfo->DeleteFile) {
  6635. -                nfs41_fcb->DeletePending = TRUE;
  6636. -                // we can delete directories right away
  6637. -                if (nfs41_fcb->StandardInfo.Directory)
  6638. -                    break;
  6639. -                nfs41_fcb->StandardInfo.DeletePending = TRUE;
  6640. -                if (RxContext->pFcb->OpenCount > 1) {
  6641. -                    rinfo.ReplaceIfExists = 0;
  6642. -                    rinfo.RootDirectory = INVALID_HANDLE_VALUE;
  6643. -                    rinfo.FileNameLength = 0;
  6644. -                    rinfo.FileName[0] = L'\0';
  6645. -                    InfoClass = FileRenameInformation;
  6646. -                    nfs41_fcb->Renamed = TRUE;
  6647. -                    break;
  6648. -                }
  6649. -            } else {
  6650. -                /* section 4.3.3 of [FSBO]
  6651. -                 * "file system behavior in the microsoft windows environment"
  6652. -                 */
  6653. -                if (nfs41_fcb->DeletePending) {
  6654. -                    nfs41_fcb->DeletePending = 0;
  6655. -                    nfs41_fcb->StandardInfo.DeletePending = 0;
  6656. -                }
  6657. -            }
  6658. -            status = STATUS_SUCCESS;
  6659. -            goto out;
  6660. -        }
  6661. -    case FileAllocationInformation:
  6662. -        {
  6663. -            PFILE_ALLOCATION_INFORMATION info =
  6664. -                (PFILE_ALLOCATION_INFORMATION)RxContext->Info.Buffer;
  6665. -
  6666. -            nfs41_fcb->StandardInfo.AllocationSize.QuadPart = info->AllocationSize.QuadPart;
  6667. -            if (nfs41_fcb->StandardInfo.EndOfFile.QuadPart > info->AllocationSize.QuadPart) {
  6668. -                nfs41_fcb->StandardInfo.EndOfFile.QuadPart = info->AllocationSize.QuadPart;
  6669. -            }
  6670. -            break;
  6671. -        }
  6672. -    case FileEndOfFileInformation:
  6673. -        {
  6674. -            PFILE_END_OF_FILE_INFORMATION info =
  6675. -                (PFILE_END_OF_FILE_INFORMATION)RxContext->Info.Buffer;
  6676. -
  6677. -            if (info->EndOfFile.QuadPart > nfs41_fcb->StandardInfo.AllocationSize.QuadPart) {
  6678. -                nfs41_fcb->StandardInfo.AllocationSize.QuadPart =
  6679. -                    nfs41_fcb->StandardInfo.EndOfFile.QuadPart = info->EndOfFile.QuadPart;
  6680. -            }
  6681. -            else {
  6682. -                nfs41_fcb->StandardInfo.EndOfFile.QuadPart = info->EndOfFile.QuadPart;
  6683. -            }
  6684. -            break;
  6685. -        }
  6686. -    case FileRenameInformation:
  6687. -        {
  6688. -            /* noop if filename and destination are the same */
  6689. -            PFILE_RENAME_INFORMATION prinfo =
  6690. -                (PFILE_RENAME_INFORMATION)RxContext->Info.Buffer;
  6691. -            const UNICODE_STRING dst = { (USHORT)prinfo->FileNameLength,
  6692. -                (USHORT)prinfo->FileNameLength, prinfo->FileName };
  6693. -            if (RtlCompareUnicodeString(&dst,
  6694. -                    SrvOpen->pAlreadyPrefixedName, FALSE) == 0) {
  6695. -                status = STATUS_SUCCESS;
  6696. -                goto out;
  6697. -            }
  6698. -        }
  6699. -    }
  6700. -
  6701. -    status = nfs41_UpcallCreate(NFS41_FILE_SET, &nfs41_fobx->sec_ctx,
  6702. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  6703. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  6704. -    if (status) goto out;
  6705. -
  6706. -    entry->u.SetFile.InfoClass = InfoClass;
  6707. -
  6708. -    /* original irp has infoclass for remove but we need to rename instead,
  6709. -     * thus we changed the local variable infoclass */
  6710. -    if (RxContext->Info.FileInformationClass == FileDispositionInformation &&
  6711. -            InfoClass == FileRenameInformation) {
  6712. -        entry->buf = &rinfo;
  6713. -        entry->buf_len = sizeof(rinfo);
  6714. -    } else {
  6715. -        entry->buf = RxContext->Info.Buffer;
  6716. -        entry->buf_len = RxContext->Info.Length;
  6717. -    }
  6718. -#ifdef ENABLE_TIMINGS
  6719. -    InterlockedIncrement(&setattr.sops);
  6720. -    InterlockedAdd64(&setattr.size, entry->u.SetFile.buf_len);
  6721. -#endif
  6722. -
  6723. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  6724. -    if (status) goto out;
  6725. -
  6726. -    status = map_setfile_error(entry->status);
  6727. -    if (!status) {
  6728. -        if (!nfs41_fobx->deleg_type && entry->ChangeTime &&
  6729. -                (SrvOpen->DesiredAccess &
  6730. -                (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA)))
  6731. -            nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime);
  6732. -        nfs41_fcb->changeattr = entry->ChangeTime;
  6733. -    }
  6734. -    nfs41_UpcallDestroy(entry);
  6735. -out:
  6736. -#ifdef ENABLE_TIMINGS
  6737. -    t2 = KeQueryPerformanceCounter(NULL);
  6738. -    InterlockedIncrement(&setattr.tops);
  6739. -    InterlockedAdd64(&setattr.ticks, t2.QuadPart - t1.QuadPart);
  6740. -#ifdef ENABLE_INDV_TIMINGS
  6741. -    DbgP("nfs41_SetFileInformation delta = %d op=%d sum=%d\n",
  6742. -        t2.QuadPart - t1.QuadPart, setattr.tops, setattr.ticks);
  6743. -#endif
  6744. -#endif
  6745. -#ifdef DEBUG_FILE_SET
  6746. -    DbgEx();
  6747. -#endif
  6748. -    return status;
  6749. -}
  6750. -
  6751. -static NTSTATUS nfs41_SetFileInformationAtCleanup(
  6752. -      IN OUT PRX_CONTEXT RxContext)
  6753. -{
  6754. -    NTSTATUS status;
  6755. -    DbgEn();
  6756. -    status = nfs41_SetFileInformation(RxContext);
  6757. -    DbgEx();
  6758. -    return status;
  6759. -}
  6760. -
  6761. -static NTSTATUS nfs41_IsValidDirectory (
  6762. -    IN OUT PRX_CONTEXT RxContext,
  6763. -    IN PUNICODE_STRING DirectoryName)
  6764. -{
  6765. -    return STATUS_SUCCESS;
  6766. -}
  6767. -
  6768. -static NTSTATUS nfs41_ComputeNewBufferingState(
  6769. -    IN OUT PMRX_SRV_OPEN pSrvOpen,
  6770. -    IN PVOID pMRxContext,
  6771. -    OUT ULONG *pNewBufferingState)
  6772. -{
  6773. -    NTSTATUS status = STATUS_SUCCESS;
  6774. -    ULONG flag = PtrToUlong(pMRxContext);
  6775. -#ifdef DEBUG_TIME_BASED_COHERENCY
  6776. -    ULONG oldFlags = pSrvOpen->BufferingFlags;
  6777. -#endif
  6778. -
  6779. -    switch(flag) {
  6780. -    case DISABLE_CACHING:
  6781. -        if (pSrvOpen->BufferingFlags &
  6782. -            (FCB_STATE_READBUFFERING_ENABLED | FCB_STATE_READCACHING_ENABLED))
  6783. -            pSrvOpen->BufferingFlags &=
  6784. -                ~(FCB_STATE_READBUFFERING_ENABLED |
  6785. -                  FCB_STATE_READCACHING_ENABLED);
  6786. -        if (pSrvOpen->BufferingFlags &
  6787. -            (FCB_STATE_WRITECACHING_ENABLED | FCB_STATE_WRITEBUFFERING_ENABLED))
  6788. -            pSrvOpen->BufferingFlags &=
  6789. -                ~(FCB_STATE_WRITECACHING_ENABLED |
  6790. -                  FCB_STATE_WRITEBUFFERING_ENABLED);
  6791. -        pSrvOpen->BufferingFlags |= FCB_STATE_DISABLE_LOCAL_BUFFERING;
  6792. -        break;
  6793. -    case ENABLE_READ_CACHING:
  6794. -        pSrvOpen->BufferingFlags |=
  6795. -            (FCB_STATE_READBUFFERING_ENABLED | FCB_STATE_READCACHING_ENABLED);
  6796. -        break;
  6797. -    case ENABLE_WRITE_CACHING:
  6798. -        pSrvOpen->BufferingFlags |=
  6799. -            (FCB_STATE_WRITECACHING_ENABLED | FCB_STATE_WRITEBUFFERING_ENABLED);
  6800. -        break;
  6801. -    case ENABLE_READWRITE_CACHING:
  6802. -        pSrvOpen->BufferingFlags =
  6803. -            (FCB_STATE_READBUFFERING_ENABLED | FCB_STATE_READCACHING_ENABLED |
  6804. -            FCB_STATE_WRITECACHING_ENABLED | FCB_STATE_WRITEBUFFERING_ENABLED);
  6805. -    }
  6806. -#ifdef DEBUG_TIME_BASED_COHERENCY
  6807. -    DbgP("nfs41_ComputeNewBufferingState: '%wZ' pSrvOpen 0x%p Old %08x New %08x\n",
  6808. -         pSrvOpen->pAlreadyPrefixedName, pSrvOpen, oldFlags,
  6809. -         pSrvOpen->BufferingFlags);
  6810. -    *pNewBufferingState = pSrvOpen->BufferingFlags;
  6811. -#endif
  6812. -    return status;
  6813. -}
  6814. -
  6815. -static void print_readwrite_args(
  6816. -    PRX_CONTEXT RxContext)
  6817. -{
  6818. -    PLOWIO_CONTEXT LowIoContext  = &RxContext->LowIoContext;
  6819. -
  6820. -    print_debug_header(RxContext);
  6821. -    DbgP("Bytecount 0x%x Byteoffset 0x%x Buffer 0x%p\n",
  6822. -        LowIoContext->ParamsFor.ReadWrite.ByteCount,
  6823. -        LowIoContext->ParamsFor.ReadWrite.ByteOffset,
  6824. -        LowIoContext->ParamsFor.ReadWrite.Buffer);
  6825. -}
  6826. -
  6827. -static void enable_caching(
  6828. -    PMRX_SRV_OPEN SrvOpen,
  6829. -    PNFS41_FOBX nfs41_fobx,
  6830. -    ULONGLONG ChangeTime,
  6831. -    HANDLE session)
  6832. -{
  6833. -    ULONG flag = 0;
  6834. -    PLIST_ENTRY pEntry;
  6835. -    nfs41_fcb_list_entry *cur;
  6836. -    BOOLEAN found = FALSE;
  6837. -
  6838. -    if (SrvOpen->DesiredAccess & FILE_READ_DATA)
  6839. -        flag = ENABLE_READ_CACHING;
  6840. -    if ((SrvOpen->DesiredAccess & FILE_WRITE_DATA) &&
  6841. -            !nfs41_fobx->write_thru)
  6842. -        flag = ENABLE_WRITE_CACHING;
  6843. -    if ((SrvOpen->DesiredAccess & FILE_READ_DATA) &&
  6844. -            (SrvOpen->DesiredAccess & FILE_WRITE_DATA) &&
  6845. -            !nfs41_fobx->write_thru)
  6846. -        flag = ENABLE_READWRITE_CACHING;
  6847. -
  6848. -#if defined(DEBUG_TIME_BASED_COHERENCY) || \
  6849. -        defined(DEBUG_WRITE) || defined(DEBUG_READ)
  6850. -    print_caching_level(1, flag, SrvOpen->pAlreadyPrefixedName);
  6851. -#endif
  6852. -
  6853. -    if (!flag)
  6854. -        return;
  6855. -
  6856. -    RxChangeBufferingState((PSRV_OPEN)SrvOpen, ULongToPtr(flag), 1);
  6857. -
  6858. -    ExAcquireFastMutex(&fcblistLock);
  6859. -    pEntry = openlist.head.Flink;
  6860. -    while (!IsListEmpty(&openlist.head)) {
  6861. -        cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry,
  6862. -                nfs41_fcb_list_entry, next);
  6863. -        if (cur->fcb == SrvOpen->pFcb) {
  6864. -#ifdef DEBUG_TIME_BASED_COHERENCY
  6865. -            DbgP("enable_caching: Looked&Found match for fcb=0x%p '%wZ'\n",
  6866. -                SrvOpen->pFcb, SrvOpen->pAlreadyPrefixedName);
  6867. -#endif
  6868. -            cur->skip = FALSE;
  6869. -            found = TRUE;
  6870. -            break;
  6871. -        }
  6872. -        if (pEntry->Flink == &openlist.head) {
  6873. -#ifdef DEBUG_TIME_BASED_COHERENCY
  6874. -            DbgP("enable_caching: reached EOL looking for fcb=0x%p '%wZ'\n",
  6875. -                SrvOpen->pFcb, SrvOpen->pAlreadyPrefixedName);
  6876. -#endif
  6877. -            break;
  6878. -        }
  6879. -        pEntry = pEntry->Flink;
  6880. -    }
  6881. -    if (!found && nfs41_fobx->deleg_type) {
  6882. -        nfs41_fcb_list_entry *oentry;
  6883. -#ifdef DEBUG_TIME_BASED_COHERENCY
  6884. -        DbgP("enable_caching: delegation recalled: srv_open=0x%p\n", SrvOpen);
  6885. -#endif
  6886. -        oentry = RxAllocatePoolWithTag(NonPagedPoolNx,
  6887. -            sizeof(nfs41_fcb_list_entry), NFS41_MM_POOLTAG_OPEN);
  6888. -        if (oentry == NULL) return;
  6889. -        oentry->fcb = SrvOpen->pFcb;
  6890. -        oentry->session = session;
  6891. -        oentry->nfs41_fobx = nfs41_fobx;
  6892. -        oentry->ChangeTime = ChangeTime;
  6893. -        oentry->skip = FALSE;
  6894. -        InsertTailList(&openlist.head, &oentry->next);
  6895. -        nfs41_fobx->deleg_type = 0;
  6896. -    }
  6897. -    ExReleaseFastMutex(&fcblistLock);
  6898. -}
  6899. -
  6900. -static NTSTATUS map_readwrite_errors(
  6901. -    DWORD status)
  6902. -{
  6903. -    switch (status) {
  6904. -    case ERROR_ACCESS_DENIED:           return STATUS_ACCESS_DENIED;
  6905. -    case ERROR_HANDLE_EOF:              return STATUS_END_OF_FILE;
  6906. -    case ERROR_FILE_INVALID:            return STATUS_FILE_INVALID;
  6907. -    case ERROR_INVALID_PARAMETER:       return STATUS_INVALID_PARAMETER;
  6908. -    case ERROR_LOCK_VIOLATION:          return STATUS_FILE_LOCK_CONFLICT;
  6909. -    case ERROR_NETWORK_ACCESS_DENIED:   return STATUS_NETWORK_ACCESS_DENIED;
  6910. -    case ERROR_NETNAME_DELETED:         return STATUS_NETWORK_NAME_DELETED;
  6911. -    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  6912. -    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  6913. -    case ERROR_FILE_TOO_LARGE:          return STATUS_FILE_TOO_LARGE;
  6914. -    case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  6915. -    default:
  6916. -        print_error("map_readwrite_errors: "
  6917. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  6918. -            "defaulting to STATUS_NET_WRITE_FAULT\n", status);
  6919. -    case ERROR_NET_WRITE_FAULT:         return STATUS_NET_WRITE_FAULT;
  6920. -    }
  6921. -}
  6922. -
  6923. -static NTSTATUS check_nfs41_read_args(
  6924. -    IN PRX_CONTEXT RxContext)
  6925. -{
  6926. -    if (!RxContext->LowIoContext.ParamsFor.ReadWrite.Buffer)
  6927. -        return STATUS_INVALID_USER_BUFFER;
  6928. -    return STATUS_SUCCESS;
  6929. -}
  6930. -
  6931. -static NTSTATUS nfs41_Read(
  6932. -    IN OUT PRX_CONTEXT RxContext)
  6933. -{
  6934. -    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  6935. -    nfs41_updowncall_entry *entry;
  6936. -    BOOLEAN async = FALSE;
  6937. -    PLOWIO_CONTEXT LowIoContext  = &RxContext->LowIoContext;
  6938. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  6939. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  6940. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  6941. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  6942. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  6943. -    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  6944. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  6945. -    DWORD io_delay;
  6946. -#ifdef ENABLE_TIMINGS
  6947. -    LARGE_INTEGER t1, t2;
  6948. -    t1 = KeQueryPerformanceCounter(NULL);
  6949. -#endif
  6950. -
  6951. -#ifdef DEBUG_READ
  6952. -    DbgEn();
  6953. -    print_readwrite_args(RxContext);
  6954. -#endif
  6955. -    status = check_nfs41_read_args(RxContext);
  6956. -    if (status) goto out;
  6957. -
  6958. -    status = nfs41_UpcallCreate(NFS41_READ, &nfs41_fobx->sec_ctx,
  6959. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  6960. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  6961. -    if (status) goto out;
  6962. -
  6963. -    entry->u.ReadWrite.MdlAddress = LowIoContext->ParamsFor.ReadWrite.Buffer;
  6964. -    entry->buf_len = LowIoContext->ParamsFor.ReadWrite.ByteCount;
  6965. -    entry->u.ReadWrite.offset = LowIoContext->ParamsFor.ReadWrite.ByteOffset;
  6966. -    if (FlagOn(RxContext->CurrentIrpSp->FileObject->Flags,
  6967. -            FO_SYNCHRONOUS_IO) == FALSE) {
  6968. -        entry->u.ReadWrite.rxcontext = RxContext;
  6969. -        async = entry->async_op = TRUE;
  6970. -    }
  6971. -
  6972. -    /* Add extra timeout depending on buffer size */
  6973. -    io_delay = pVNetRootContext->timeout +
  6974. -        EXTRA_TIMEOUT_PER_BYTE(entry->buf_len);
  6975. -    status = nfs41_UpcallWaitForReply(entry, io_delay);
  6976. -    if (status) goto out;
  6977. -
  6978. -    if (async) {
  6979. -#ifdef DEBUG_READ
  6980. -        DbgP("This is asynchronous read, returning control back to the user\n");
  6981. -#endif
  6982. -        status = STATUS_PENDING;
  6983. -        goto out;
  6984. -    }
  6985. -
  6986. -    if (entry->status == NO_ERROR) {
  6987. -#ifdef ENABLE_TIMINGS
  6988. -        InterlockedIncrement(&read.sops);
  6989. -        InterlockedAdd64(&read.size, entry->u.ReadWrite.len);
  6990. -#endif
  6991. -        status = RxContext->CurrentIrp->IoStatus.Status = STATUS_SUCCESS;
  6992. -        RxContext->IoStatusBlock.Information = entry->buf_len;
  6993. -
  6994. -        if ((!BooleanFlagOn(LowIoContext->ParamsFor.ReadWrite.Flags,
  6995. -                LOWIO_READWRITEFLAG_PAGING_IO) &&
  6996. -                (SrvOpen->DesiredAccess & FILE_READ_DATA) &&
  6997. -                !pVNetRootContext->nocache && !nfs41_fobx->nocache &&
  6998. -                !(SrvOpen->BufferingFlags &
  6999. -                (FCB_STATE_READBUFFERING_ENABLED |
  7000. -                 FCB_STATE_READCACHING_ENABLED)))) {
  7001. -            enable_caching(SrvOpen, nfs41_fobx, nfs41_fcb->changeattr,
  7002. -                pVNetRootContext->session);
  7003. -        }
  7004. -    } else {
  7005. -        status = map_readwrite_errors(entry->status);
  7006. -        RxContext->CurrentIrp->IoStatus.Status = status;
  7007. -        RxContext->IoStatusBlock.Information = 0;
  7008. -    }
  7009. -    nfs41_UpcallDestroy(entry);
  7010. -out:
  7011. -#ifdef ENABLE_TIMINGS
  7012. -    t2 = KeQueryPerformanceCounter(NULL);
  7013. -    InterlockedIncrement(&read.tops);
  7014. -    InterlockedAdd64(&read.ticks, t2.QuadPart - t1.QuadPart);
  7015. -#ifdef ENABLE_INDV_TIMINGS
  7016. -    DbgP("nfs41_Read delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart,
  7017. -        read.tops, read.ticks);
  7018. -#endif
  7019. -#endif
  7020. -#ifdef DEBUG_READ
  7021. -    DbgEx();
  7022. -#endif
  7023. -    return status;
  7024. -}
  7025. -
  7026. -static NTSTATUS check_nfs41_write_args(
  7027. -    IN PRX_CONTEXT RxContext)
  7028. -{
  7029. -    NTSTATUS status = STATUS_SUCCESS;
  7030. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  7031. -        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  7032. -
  7033. -    if (!RxContext->LowIoContext.ParamsFor.ReadWrite.Buffer) {
  7034. -        status = STATUS_INVALID_USER_BUFFER;
  7035. -        goto out;
  7036. -    }
  7037. -
  7038. -    if (pVNetRootContext->read_only) {
  7039. -        print_error("check_nfs41_write_args: Read-only mount\n");
  7040. -        status = STATUS_MEDIA_WRITE_PROTECTED;
  7041. -        goto out;
  7042. -    }
  7043. -out:
  7044. -    return status;
  7045. -}
  7046. -
  7047. -static NTSTATUS nfs41_Write(
  7048. -    IN OUT PRX_CONTEXT RxContext)
  7049. -{
  7050. -    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  7051. -    nfs41_updowncall_entry *entry;
  7052. -    BOOLEAN async = FALSE;
  7053. -    PLOWIO_CONTEXT LowIoContext  = &RxContext->LowIoContext;
  7054. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  7055. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  7056. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  7057. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  7058. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  7059. -    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  7060. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  7061. -    DWORD io_delay;
  7062. -#ifdef ENABLE_TIMINGS
  7063. -    LARGE_INTEGER t1, t2;
  7064. -    t1 = KeQueryPerformanceCounter(NULL);
  7065. -#endif
  7066. -
  7067. -#ifdef DEBUG_WRITE
  7068. -    DbgEn();
  7069. -    print_readwrite_args(RxContext);
  7070. -#endif
  7071. -
  7072. -    status = check_nfs41_write_args(RxContext);
  7073. -    if (status) goto out;
  7074. -
  7075. -    status = nfs41_UpcallCreate(NFS41_WRITE, &nfs41_fobx->sec_ctx,
  7076. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  7077. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  7078. -    if (status) goto out;
  7079. -
  7080. -    entry->u.ReadWrite.MdlAddress = LowIoContext->ParamsFor.ReadWrite.Buffer;
  7081. -    entry->buf_len = LowIoContext->ParamsFor.ReadWrite.ByteCount;
  7082. -    entry->u.ReadWrite.offset = LowIoContext->ParamsFor.ReadWrite.ByteOffset;
  7083. -
  7084. -    if (FlagOn(RxContext->CurrentIrpSp->FileObject->Flags,
  7085. -            FO_SYNCHRONOUS_IO) == FALSE) {
  7086. -        entry->u.ReadWrite.rxcontext = RxContext;
  7087. -        async = entry->async_op = TRUE;
  7088. -    }
  7089. -
  7090. -    /* Add extra timeout depending on buffer size */
  7091. -    io_delay = pVNetRootContext->timeout +
  7092. -        EXTRA_TIMEOUT_PER_BYTE(entry->buf_len);
  7093. -    status = nfs41_UpcallWaitForReply(entry, io_delay);
  7094. -    if (status) goto out;
  7095. -
  7096. -    if (async) {
  7097. -#ifdef DEBUG_WRITE
  7098. -        DbgP("This is asynchronous write, returning control back to the user\n");
  7099. -#endif
  7100. -        status = STATUS_PENDING;
  7101. -        goto out;
  7102. -    }
  7103. -    
  7104. -    if (entry->status == NO_ERROR) {
  7105. -        //update cached file attributes
  7106. -#ifdef ENABLE_TIMINGS
  7107. -        InterlockedIncrement(&write.sops);
  7108. -        InterlockedAdd64(&write.size, entry->u.ReadWrite.len);
  7109. -#endif
  7110. -        nfs41_fcb->StandardInfo.EndOfFile.QuadPart = entry->buf_len +
  7111. -            entry->u.ReadWrite.offset;
  7112. -        status = RxContext->CurrentIrp->IoStatus.Status = STATUS_SUCCESS;
  7113. -        RxContext->IoStatusBlock.Information = entry->buf_len;
  7114. -        nfs41_fcb->changeattr = entry->ChangeTime;
  7115. -
  7116. -        //re-enable write buffering
  7117. -        if (!BooleanFlagOn(LowIoContext->ParamsFor.ReadWrite.Flags,
  7118. -                LOWIO_READWRITEFLAG_PAGING_IO) &&
  7119. -                (SrvOpen->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA)) &&
  7120. -                !pVNetRootContext->write_thru &&
  7121. -                !pVNetRootContext->nocache &&
  7122. -                !nfs41_fobx->write_thru && !nfs41_fobx->nocache &&
  7123. -                !(SrvOpen->BufferingFlags &
  7124. -                (FCB_STATE_WRITEBUFFERING_ENABLED |
  7125. -                 FCB_STATE_WRITECACHING_ENABLED))) {
  7126. -            enable_caching(SrvOpen, nfs41_fobx, nfs41_fcb->changeattr,
  7127. -                pVNetRootContext->session);
  7128. -        } else if (!nfs41_fobx->deleg_type)
  7129. -            nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime);
  7130. -
  7131. -    } else {
  7132. -        status = map_readwrite_errors(entry->status);
  7133. -        RxContext->CurrentIrp->IoStatus.Status = status;
  7134. -        RxContext->IoStatusBlock.Information = 0;
  7135. -    }
  7136. -    nfs41_UpcallDestroy(entry);
  7137. -out:
  7138. -#ifdef ENABLE_TIMINGS
  7139. -    t2 = KeQueryPerformanceCounter(NULL);
  7140. -    InterlockedIncrement(&write.tops);
  7141. -    InterlockedAdd64(&write.ticks, t2.QuadPart - t1.QuadPart);
  7142. -#ifdef ENABLE_INDV_TIMINGS
  7143. -    DbgP("nfs41_Write delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart,
  7144. -        write.tops, write.ticks);
  7145. -#endif
  7146. -#endif
  7147. -#ifdef DEBUG_WRITE
  7148. -    DbgEx();
  7149. -#endif
  7150. -    return status;
  7151. -}
  7152. -
  7153. -static NTSTATUS nfs41_IsLockRealizable(
  7154. -    IN OUT PMRX_FCB pFcb,
  7155. -    IN PLARGE_INTEGER  ByteOffset,
  7156. -    IN PLARGE_INTEGER  Length,
  7157. -    IN ULONG  LowIoLockFlags)
  7158. -{
  7159. -    NTSTATUS status = STATUS_SUCCESS;
  7160. -#ifdef DEBUG_LOCK
  7161. -    DbgEn();
  7162. -    DbgP("offset 0x%llx, length 0x%llx, exclusive=%u, blocking=%u\n",
  7163. -        ByteOffset->QuadPart,Length->QuadPart,
  7164. -        BooleanFlagOn(LowIoLockFlags, SL_EXCLUSIVE_LOCK),
  7165. -        !BooleanFlagOn(LowIoLockFlags, SL_FAIL_IMMEDIATELY));
  7166. -#endif
  7167. -
  7168. -    /* NFS lock operations with length=0 MUST fail with NFS4ERR_INVAL */
  7169. -    if (Length->QuadPart == 0)
  7170. -        status = STATUS_NOT_SUPPORTED;
  7171. -
  7172. -#ifdef DEBUG_LOCK
  7173. -    DbgEx();
  7174. -#endif
  7175. -    return status;
  7176. -}
  7177. -
  7178. -static NTSTATUS map_lock_errors(
  7179. -    DWORD status)
  7180. -{
  7181. -    switch (status) {
  7182. -    case NO_ERROR:                  return STATUS_SUCCESS;
  7183. -    case ERROR_NETNAME_DELETED:     return STATUS_NETWORK_NAME_DELETED;
  7184. -    case ERROR_LOCK_FAILED:         return STATUS_LOCK_NOT_GRANTED;
  7185. -    case ERROR_NOT_LOCKED:          return STATUS_RANGE_NOT_LOCKED;
  7186. -    case ERROR_ATOMIC_LOCKS_NOT_SUPPORTED: return STATUS_UNSUCCESSFUL;
  7187. -    case ERROR_OUTOFMEMORY:         return STATUS_INSUFFICIENT_RESOURCES;
  7188. -    case ERROR_SHARING_VIOLATION:   return STATUS_SHARING_VIOLATION;
  7189. -    case ERROR_FILE_INVALID:        return STATUS_FILE_INVALID;
  7190. -    /* if we return ERROR_INVALID_PARAMETER, Windows translates that to
  7191. -     * success!! */
  7192. -    case ERROR_INVALID_PARAMETER:   return STATUS_LOCK_NOT_GRANTED;
  7193. -    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  7194. -    default:
  7195. -        print_error("map_lock_errors: "
  7196. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  7197. -            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status);
  7198. -    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  7199. -    }
  7200. -}
  7201. -
  7202. -static void print_lock_args(
  7203. -    PRX_CONTEXT RxContext)
  7204. -{
  7205. -    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  7206. -    const ULONG flags = LowIoContext->ParamsFor.Locks.Flags;
  7207. -    print_debug_header(RxContext);
  7208. -    DbgP("offset 0x%llx, length 0x%llx, exclusive=%u, blocking=%u\n",
  7209. -        LowIoContext->ParamsFor.Locks.ByteOffset,
  7210. -        LowIoContext->ParamsFor.Locks.Length,
  7211. -        BooleanFlagOn(flags, SL_EXCLUSIVE_LOCK),
  7212. -        !BooleanFlagOn(flags, SL_FAIL_IMMEDIATELY));
  7213. -}
  7214. -
  7215. -
  7216. -/* use exponential backoff between polls for blocking locks */
  7217. -#define MSEC_TO_RELATIVE_WAIT   (-10000)
  7218. -#define MIN_LOCK_POLL_WAIT      (500 * MSEC_TO_RELATIVE_WAIT) /* 500ms */
  7219. -#define MAX_LOCK_POLL_WAIT      (30000 * MSEC_TO_RELATIVE_WAIT) /* 30s */
  7220. -
  7221. -static void denied_lock_backoff(
  7222. -    IN OUT PLARGE_INTEGER delay)
  7223. -{
  7224. -    if (delay->QuadPart == 0)
  7225. -        delay->QuadPart = MIN_LOCK_POLL_WAIT;
  7226. -    else
  7227. -        delay->QuadPart <<= 1;
  7228. -
  7229. -    if (delay->QuadPart < MAX_LOCK_POLL_WAIT)
  7230. -        delay->QuadPart = MAX_LOCK_POLL_WAIT;
  7231. -}
  7232. -
  7233. -static NTSTATUS nfs41_Lock(
  7234. -    IN OUT PRX_CONTEXT RxContext)
  7235. -{
  7236. -    NTSTATUS status = STATUS_SUCCESS;
  7237. -    nfs41_updowncall_entry *entry;
  7238. -    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  7239. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  7240. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  7241. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  7242. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  7243. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  7244. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  7245. -    const ULONG flags = LowIoContext->ParamsFor.Locks.Flags;
  7246. -    LARGE_INTEGER poll_delay = {0};
  7247. -#ifdef ENABLE_TIMINGS
  7248. -    LARGE_INTEGER t1, t2;
  7249. -    t1 = KeQueryPerformanceCounter(NULL);
  7250. -#endif
  7251. -
  7252. -    poll_delay.QuadPart = 0;
  7253. -
  7254. -#ifdef DEBUG_LOCK
  7255. -    DbgEn();
  7256. -    print_lock_args(RxContext);
  7257. -#endif
  7258. -
  7259. -/*  RxReleaseFcbResourceForThreadInMRx(RxContext, RxContext->pFcb,
  7260. -        LowIoContext->ResourceThreadId); */
  7261. -
  7262. -    status = nfs41_UpcallCreate(NFS41_LOCK, &nfs41_fobx->sec_ctx,
  7263. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  7264. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  7265. -    if (status) goto out;
  7266. -
  7267. -    entry->u.Lock.offset = LowIoContext->ParamsFor.Locks.ByteOffset;
  7268. -    entry->u.Lock.length = LowIoContext->ParamsFor.Locks.Length;
  7269. -    entry->u.Lock.exclusive = BooleanFlagOn(flags, SL_EXCLUSIVE_LOCK);
  7270. -    entry->u.Lock.blocking = !BooleanFlagOn(flags, SL_FAIL_IMMEDIATELY);
  7271. -
  7272. -retry_upcall:
  7273. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  7274. -    if (status) goto out;
  7275. -
  7276. -    /* blocking locks keep trying until it succeeds */
  7277. -    if (entry->status == ERROR_LOCK_FAILED && entry->u.Lock.blocking) {
  7278. -        denied_lock_backoff(&poll_delay);
  7279. -        DbgP("returned ERROR_LOCK_FAILED; retrying in %llums\n",
  7280. -            poll_delay.QuadPart / MSEC_TO_RELATIVE_WAIT);
  7281. -        KeDelayExecutionThread(KernelMode, FALSE, &poll_delay);
  7282. -        entry->state = NFS41_WAITING_FOR_UPCALL;
  7283. -        goto retry_upcall;
  7284. -    }
  7285. -
  7286. -    status = map_lock_errors(entry->status);
  7287. -    RxContext->CurrentIrp->IoStatus.Status = status;
  7288. -
  7289. -    nfs41_UpcallDestroy(entry);
  7290. -out:
  7291. -#ifdef ENABLE_TIMINGS
  7292. -    t2 = KeQueryPerformanceCounter(NULL);
  7293. -    InterlockedIncrement(&lock.tops);
  7294. -    InterlockedAdd64(&lock.ticks, t2.QuadPart - t1.QuadPart);
  7295. -#ifdef ENABLE_INDV_TIMINGS
  7296. -    DbgP("nfs41_Lock delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart,
  7297. -        lock.tops, lock.ticks);
  7298. -#endif
  7299. -#endif
  7300. -#ifdef DEBUG_LOCK
  7301. -    DbgEx();
  7302. -#endif
  7303. -    return status;
  7304. -}
  7305. -
  7306. -static void print_unlock_args(
  7307. -    PRX_CONTEXT RxContext)
  7308. -{
  7309. -    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  7310. -    print_debug_header(RxContext);
  7311. -    if (LowIoContext->Operation == LOWIO_OP_UNLOCK_MULTIPLE) {
  7312. -        PLOWIO_LOCK_LIST lock = LowIoContext->ParamsFor.Locks.LockList;
  7313. -        DbgP("LOWIO_OP_UNLOCK_MULTIPLE:");
  7314. -        while (lock) {
  7315. -            DbgP(" (offset=%llu, length=%llu)", lock->ByteOffset, lock->Length);
  7316. -            lock = lock->Next;
  7317. -        }
  7318. -        DbgP("\n");
  7319. -    } else {
  7320. -        DbgP("LOWIO_OP_UNLOCK: offset=%llu, length=%llu\n",
  7321. -            LowIoContext->ParamsFor.Locks.ByteOffset,
  7322. -            LowIoContext->ParamsFor.Locks.Length);
  7323. -    }
  7324. -}
  7325. -
  7326. -__inline ULONG unlock_list_count(
  7327. -    PLOWIO_LOCK_LIST lock)
  7328. -{
  7329. -    ULONG count = 0;
  7330. -    while (lock) {
  7331. -        count++;
  7332. -        lock = lock->Next;
  7333. -    }
  7334. -    return count;
  7335. -}
  7336. -
  7337. -static NTSTATUS nfs41_Unlock(
  7338. -    IN OUT PRX_CONTEXT RxContext)
  7339. -{
  7340. -    NTSTATUS status = STATUS_SUCCESS;
  7341. -    nfs41_updowncall_entry *entry;
  7342. -    PLOWIO_CONTEXT LowIoContext  = &RxContext->LowIoContext;
  7343. -    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  7344. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  7345. -    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  7346. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  7347. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  7348. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  7349. -#ifdef ENABLE_TIMINGS
  7350. -    LARGE_INTEGER t1, t2;
  7351. -    t1 = KeQueryPerformanceCounter(NULL);
  7352. -#endif
  7353. -#ifdef DEBUG_LOCK
  7354. -    DbgEn();
  7355. -    print_lock_args(RxContext);
  7356. -#endif
  7357. -
  7358. -/*  RxReleaseFcbResourceForThreadInMRx(RxContext, RxContext->pFcb,
  7359. -        LowIoContext->ResourceThreadId); */
  7360. -
  7361. -    status = nfs41_UpcallCreate(NFS41_UNLOCK, &nfs41_fobx->sec_ctx,
  7362. -        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  7363. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  7364. -    if (status) goto out;
  7365. -
  7366. -    if (LowIoContext->Operation == LOWIO_OP_UNLOCK_MULTIPLE) {
  7367. -        entry->u.Unlock.count = unlock_list_count(
  7368. -            LowIoContext->ParamsFor.Locks.LockList);
  7369. -        RtlCopyMemory(&entry->u.Unlock.locks,
  7370. -            LowIoContext->ParamsFor.Locks.LockList,
  7371. -            sizeof(LOWIO_LOCK_LIST));
  7372. -    } else {
  7373. -        entry->u.Unlock.count = 1;
  7374. -        entry->u.Unlock.locks.ByteOffset =
  7375. -            LowIoContext->ParamsFor.Locks.ByteOffset;
  7376. -        entry->u.Unlock.locks.Length =
  7377. -            LowIoContext->ParamsFor.Locks.Length;
  7378. -    }
  7379. -
  7380. -    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  7381. -    if (status) goto out;
  7382. -
  7383. -    status = map_lock_errors(entry->status);
  7384. -    RxContext->CurrentIrp->IoStatus.Status = status;
  7385. -    nfs41_UpcallDestroy(entry);
  7386. -out:
  7387. -#ifdef ENABLE_TIMINGS
  7388. -    t2 = KeQueryPerformanceCounter(NULL);
  7389. -    InterlockedIncrement(&unlock.tops);
  7390. -    InterlockedAdd64(&unlock.ticks, t2.QuadPart - t1.QuadPart);
  7391. -#ifdef ENABLE_INDV_TIMINGS
  7392. -    DbgP("nfs41_Unlock delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart,
  7393. -        unlock.tops, unlock.ticks);
  7394. -#endif
  7395. -#endif
  7396. -#ifdef DEBUG_LOCK
  7397. -    DbgEx();
  7398. -#endif
  7399. -    return status;
  7400. -}
  7401. -
  7402. -static NTSTATUS map_symlink_errors(
  7403. -    NTSTATUS status)
  7404. -{
  7405. -    switch (status) {
  7406. -    case NO_ERROR:                  return STATUS_SUCCESS;
  7407. -    case ERROR_INVALID_REPARSE_DATA: return STATUS_IO_REPARSE_DATA_INVALID;
  7408. -    case ERROR_NOT_A_REPARSE_POINT: return STATUS_NOT_A_REPARSE_POINT;
  7409. -    case ERROR_ACCESS_DENIED:       return STATUS_ACCESS_DENIED;
  7410. -    case ERROR_NOT_EMPTY:           return STATUS_DIRECTORY_NOT_EMPTY;
  7411. -    case ERROR_OUTOFMEMORY:         return STATUS_INSUFFICIENT_RESOURCES;
  7412. -    case ERROR_INSUFFICIENT_BUFFER: return STATUS_BUFFER_TOO_SMALL;
  7413. -    case STATUS_BUFFER_TOO_SMALL:
  7414. -    case ERROR_BUFFER_OVERFLOW:     return STATUS_BUFFER_OVERFLOW;
  7415. -    case ERROR_DISK_FULL:           return STATUS_DISK_FULL;
  7416. -    case ERROR_DISK_QUOTA_EXCEEDED: return STATUS_DISK_QUOTA_EXCEEDED;
  7417. -    case ERROR_FILE_TOO_LARGE:      return STATUS_FILE_TOO_LARGE;
  7418. -    case ERROR_TOO_MANY_LINKS:      return STATUS_TOO_MANY_LINKS;
  7419. -    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  7420. -    default:
  7421. -        print_error("map_symlink_errors: "
  7422. -            "failed to map windows ERROR_0x%x to NTSTATUS; "
  7423. -            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status);
  7424. -    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  7425. -    }
  7426. -}
  7427. -
  7428. -static void print_reparse_buffer(
  7429. -    PREPARSE_DATA_BUFFER Reparse)
  7430. -{
  7431. -    UNICODE_STRING name;
  7432. -    DbgP("ReparseTag:           %08X\n", Reparse->ReparseTag);
  7433. -    DbgP("ReparseDataLength:    %8u\n", Reparse->ReparseDataLength);
  7434. -    DbgP("Reserved:             %8u\n", Reparse->Reserved);
  7435. -    DbgP("SubstituteNameOffset: %8u\n",
  7436. -         Reparse->SymbolicLinkReparseBuffer.SubstituteNameOffset);
  7437. -    DbgP("SubstituteNameLength: %8u\n",
  7438. -         Reparse->SymbolicLinkReparseBuffer.SubstituteNameLength);
  7439. -    DbgP("PrintNameOffset:      %8u\n",
  7440. -         Reparse->SymbolicLinkReparseBuffer.PrintNameOffset);
  7441. -    DbgP("PrintNameLength:      %8u\n",
  7442. -         Reparse->SymbolicLinkReparseBuffer.PrintNameLength);
  7443. -    DbgP("Flags:                %08X\n",
  7444. -         Reparse->SymbolicLinkReparseBuffer.Flags);
  7445. -
  7446. -    name.Buffer = &Reparse->SymbolicLinkReparseBuffer.PathBuffer[
  7447. -        Reparse->SymbolicLinkReparseBuffer.SubstituteNameOffset/sizeof(WCHAR)];
  7448. -    name.MaximumLength = name.Length =
  7449. -        Reparse->SymbolicLinkReparseBuffer.SubstituteNameLength;
  7450. -    DbgP("SubstituteName:       '%wZ'\n", &name);
  7451. -
  7452. -    name.Buffer = &Reparse->SymbolicLinkReparseBuffer.PathBuffer[
  7453. -        Reparse->SymbolicLinkReparseBuffer.PrintNameOffset/sizeof(WCHAR)];
  7454. -    name.MaximumLength = name.Length =
  7455. -        Reparse->SymbolicLinkReparseBuffer.PrintNameLength;
  7456. -    DbgP("PrintName:            '%wZ'\n", &name);
  7457. -}
  7458. -
  7459. -static NTSTATUS check_nfs41_setreparse_args(
  7460. -    IN PRX_CONTEXT RxContext)
  7461. -{
  7462. -    NTSTATUS status = STATUS_SUCCESS;
  7463. -    __notnull XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl;
  7464. -    __notnull PREPARSE_DATA_BUFFER Reparse = (PREPARSE_DATA_BUFFER)FsCtl->pInputBuffer;
  7465. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  7466. -    __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext =
  7467. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  7468. -    const ULONG HeaderLen = REPARSE_DATA_BUFFER_HEADER_SIZE;
  7469. -
  7470. -    /* access checks */
  7471. -    if (VNetRootContext->read_only) {
  7472. -        status = STATUS_MEDIA_WRITE_PROTECTED;
  7473. -        goto out;
  7474. -    }
  7475. -    if (!(SrvOpen->DesiredAccess & (FILE_WRITE_DATA|FILE_WRITE_ATTRIBUTES))) {
  7476. -        status = STATUS_ACCESS_DENIED;
  7477. -        goto out;
  7478. -    }
  7479. -
  7480. -    /* must have a filename longer than vnetroot name,
  7481. -     * or it's trying to operate on the volume itself */
  7482. -    if (is_root_directory(RxContext)) {
  7483. -        status = STATUS_INVALID_PARAMETER;
  7484. -        goto out;
  7485. -    }
  7486. -    if (FsCtl->pOutputBuffer != NULL) {
  7487. -        status = STATUS_INVALID_PARAMETER;
  7488. -        goto out;
  7489. -    }
  7490. -
  7491. -    /* validate input buffer and length */
  7492. -    if (!Reparse) {
  7493. -        status = STATUS_INVALID_BUFFER_SIZE;
  7494. -        goto out;
  7495. -    }
  7496. -
  7497. -    if (FsCtl->InputBufferLength < HeaderLen ||
  7498. -            FsCtl->InputBufferLength > MAXIMUM_REPARSE_DATA_BUFFER_SIZE) {
  7499. -        status = STATUS_IO_REPARSE_DATA_INVALID;
  7500. -        goto out;
  7501. -    }
  7502. -    if (FsCtl->InputBufferLength != HeaderLen + Reparse->ReparseDataLength) {
  7503. -        status = STATUS_IO_REPARSE_DATA_INVALID;
  7504. -        goto out;
  7505. -    }
  7506. -
  7507. -    /* validate reparse tag */
  7508. -    if (!IsReparseTagValid(Reparse->ReparseTag)) {
  7509. -        status = STATUS_IO_REPARSE_TAG_INVALID;
  7510. -        goto out;
  7511. -    }
  7512. -    if (Reparse->ReparseTag != IO_REPARSE_TAG_SYMLINK) {
  7513. -        status = STATUS_IO_REPARSE_TAG_MISMATCH;
  7514. -        goto out;
  7515. -    }
  7516. -out:
  7517. -    return status;
  7518. -}
  7519. -
  7520. -static NTSTATUS nfs41_SetReparsePoint(
  7521. -    IN OUT PRX_CONTEXT RxContext)
  7522. -{
  7523. -    NTSTATUS status;
  7524. -    UNICODE_STRING TargetName;
  7525. -    __notnull XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl;
  7526. -    __notnull PREPARSE_DATA_BUFFER Reparse = (PREPARSE_DATA_BUFFER)FsCtl->pInputBuffer;
  7527. -    __notnull PNFS41_FOBX Fobx = NFS41GetFobxExtension(RxContext->pFobx);
  7528. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  7529. -    __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext =
  7530. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  7531. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  7532. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  7533. -    nfs41_updowncall_entry *entry;
  7534. -
  7535. -#ifdef DEBUG_SYMLINK
  7536. -    DbgEn();
  7537. -    print_debug_header(RxContext);
  7538. -    print_reparse_buffer(Reparse);
  7539. -#endif
  7540. -    status = check_nfs41_setreparse_args(RxContext);
  7541. -    if (status) goto out;
  7542. -
  7543. -    TargetName.MaximumLength = TargetName.Length =
  7544. -        Reparse->SymbolicLinkReparseBuffer.PrintNameLength;
  7545. -    TargetName.Buffer = &Reparse->SymbolicLinkReparseBuffer.PathBuffer[
  7546. -        Reparse->SymbolicLinkReparseBuffer.PrintNameOffset/sizeof(WCHAR)];
  7547. -
  7548. -    status = nfs41_UpcallCreate(NFS41_SYMLINK, &Fobx->sec_ctx,
  7549. -        VNetRootContext->session, Fobx->nfs41_open_state,
  7550. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  7551. -    if (status) goto out;
  7552. -
  7553. -    entry->u.Symlink.target = &TargetName;
  7554. -    entry->u.Symlink.set = TRUE;
  7555. -
  7556. -    status = nfs41_UpcallWaitForReply(entry, VNetRootContext->timeout);
  7557. -    if (status) goto out;
  7558. -
  7559. -    status = map_symlink_errors(entry->status);
  7560. -    nfs41_UpcallDestroy(entry);
  7561. -out:
  7562. -#ifdef DEBUG_SYMLINK
  7563. -    DbgEx();
  7564. -#endif
  7565. -    return status;
  7566. -}
  7567. -
  7568. -static NTSTATUS check_nfs41_getreparse_args(
  7569. -    PRX_CONTEXT RxContext)
  7570. -{
  7571. -    NTSTATUS status = STATUS_SUCCESS;
  7572. -    XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl;
  7573. -    const USHORT HeaderLen = FIELD_OFFSET(REPARSE_DATA_BUFFER,
  7574. -        SymbolicLinkReparseBuffer.PathBuffer);
  7575. -
  7576. -    /* must have a filename longer than vnetroot name,
  7577. -     * or it's trying to operate on the volume itself */
  7578. -    if (is_root_directory(RxContext)) {
  7579. -        status = STATUS_INVALID_PARAMETER;
  7580. -        goto out;
  7581. -    }
  7582. -    /* ifs reparse tests expect STATUS_INVALID_PARAMETER,
  7583. -     * but 'dir' passes a buffer here when querying symlinks
  7584. -    if (FsCtl->pInputBuffer != NULL) {
  7585. -        status = STATUS_INVALID_PARAMETER;
  7586. -        goto out;
  7587. -    } */
  7588. -    if (!FsCtl->pOutputBuffer) {
  7589. -        status = STATUS_INVALID_USER_BUFFER;
  7590. -        goto out;
  7591. -    }
  7592. -    if (!BooleanFlagOn(RxContext->pFcb->Attributes,
  7593. -            FILE_ATTRIBUTE_REPARSE_POINT)) {
  7594. -        status = STATUS_NOT_A_REPARSE_POINT;
  7595. -        DbgP("FILE_ATTRIBUTE_REPARSE_POINT is not set!\n");
  7596. -        goto out;
  7597. -    }
  7598. -
  7599. -    if (FsCtl->OutputBufferLength < HeaderLen) {
  7600. -        RxContext->InformationToReturn = HeaderLen;
  7601. -        status = STATUS_BUFFER_TOO_SMALL;
  7602. -        goto out;
  7603. -    }
  7604. -out:
  7605. -    return status;
  7606. -}
  7607. -
  7608. -static NTSTATUS nfs41_GetReparsePoint(
  7609. -    IN OUT PRX_CONTEXT RxContext)
  7610. -{
  7611. -    NTSTATUS status;
  7612. -    UNICODE_STRING TargetName;
  7613. -    XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl;
  7614. -    __notnull PNFS41_FOBX Fobx = NFS41GetFobxExtension(RxContext->pFobx);
  7615. -    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  7616. -    __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext =
  7617. -        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  7618. -    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  7619. -        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  7620. -    nfs41_updowncall_entry *entry;
  7621. -    const USHORT HeaderLen = FIELD_OFFSET(REPARSE_DATA_BUFFER,
  7622. -        SymbolicLinkReparseBuffer.PathBuffer);
  7623. -
  7624. -#ifdef DEBUG_SYMLINK
  7625. -    DbgEn();
  7626. -#endif
  7627. -    status = check_nfs41_getreparse_args(RxContext);
  7628. -    if (status) goto out;
  7629. -
  7630. -    TargetName.Buffer = (PWCH)((PBYTE)FsCtl->pOutputBuffer + HeaderLen);
  7631. -    TargetName.MaximumLength = (USHORT)min(FsCtl->OutputBufferLength -
  7632. -        HeaderLen, 0xFFFF);
  7633. -
  7634. -    status = nfs41_UpcallCreate(NFS41_SYMLINK, &Fobx->sec_ctx,
  7635. -        VNetRootContext->session, Fobx->nfs41_open_state,
  7636. -        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  7637. -    if (status) goto out;
  7638. -
  7639. -    entry->u.Symlink.target = &TargetName;
  7640. -    entry->u.Symlink.set = FALSE;
  7641. -
  7642. -    status = nfs41_UpcallWaitForReply(entry, VNetRootContext->timeout);
  7643. -    if (status) goto out;
  7644. -
  7645. -    status = map_symlink_errors(entry->status);
  7646. -    if (status == STATUS_SUCCESS) {
  7647. -        /* fill in the output buffer */
  7648. -        PREPARSE_DATA_BUFFER Reparse = (PREPARSE_DATA_BUFFER)
  7649. -            FsCtl->pOutputBuffer;
  7650. -        Reparse->ReparseTag = IO_REPARSE_TAG_SYMLINK;
  7651. -        Reparse->ReparseDataLength = HeaderLen + TargetName.Length -
  7652. -            REPARSE_DATA_BUFFER_HEADER_SIZE;
  7653. -        Reparse->Reserved = 0;
  7654. -        Reparse->SymbolicLinkReparseBuffer.Flags = SYMLINK_FLAG_RELATIVE;
  7655. -        /* PrintName and SubstituteName point to the same string */
  7656. -        Reparse->SymbolicLinkReparseBuffer.SubstituteNameOffset = 0;
  7657. -        Reparse->SymbolicLinkReparseBuffer.SubstituteNameLength =
  7658. -            TargetName.Length;
  7659. -        Reparse->SymbolicLinkReparseBuffer.PrintNameOffset = 0;
  7660. -        Reparse->SymbolicLinkReparseBuffer.PrintNameLength = TargetName.Length;
  7661. -        print_reparse_buffer(Reparse);
  7662. -
  7663. -        RxContext->IoStatusBlock.Information =
  7664. -            (ULONG_PTR)HeaderLen + TargetName.Length;
  7665. -    } else if (status == STATUS_BUFFER_TOO_SMALL) {
  7666. -        RxContext->InformationToReturn =
  7667. -            (ULONG_PTR)HeaderLen + TargetName.Length;
  7668. -    }
  7669. -    nfs41_UpcallDestroy(entry);
  7670. -out:
  7671. -#ifdef DEBUG_SYMLINK
  7672. -    DbgEx();
  7673. -#endif
  7674. -    return status;
  7675. -}
  7676. -
  7677. -static NTSTATUS nfs41_FsCtl(
  7678. -    IN OUT PRX_CONTEXT RxContext)
  7679. -{
  7680. -    NTSTATUS status = STATUS_INVALID_DEVICE_REQUEST;
  7681. -#ifdef DEBUG_FSCTL
  7682. -    DbgEn();
  7683. -    print_debug_header(RxContext);
  7684. -#endif
  7685. -    const ULONG fscontrolcode =
  7686. -        RxContext->LowIoContext.ParamsFor.FsCtl.FsControlCode;
  7687. -
  7688. -    switch (fscontrolcode) {
  7689. -    case FSCTL_SET_REPARSE_POINT:
  7690. -        status = nfs41_SetReparsePoint(RxContext);
  7691. -        break;
  7692. -    case FSCTL_GET_REPARSE_POINT:
  7693. -        status = nfs41_GetReparsePoint(RxContext);
  7694. -        break;
  7695. -    default:
  7696. -        break;
  7697. -    }
  7698. -
  7699. -#ifdef DEBUG_FSCTL
  7700. -    const char *fsctl_str = fsctl2string(fscontrolcode);
  7701. -
  7702. -    if (fsctl_str) {
  7703. -        DbgP("nfs41_FsCtl: FsControlCode='%s', status=0x%lx\n",
  7704. -            fsctl_str, (long)status);
  7705. -    }
  7706. -    else {
  7707. -        DbgP("nfs41_FsCtl: FsControlCode=0x%lx, status=0x%lx\n",
  7708. -            (unsigned long)fscontrolcode, (long)status);
  7709. -    }
  7710. -#endif /* DEBUG_FSCTL */
  7711. -
  7712. -#ifdef DEBUG_FSCTL
  7713. -    DbgEx();
  7714. -#endif
  7715. -    return status;
  7716. -}
  7717. -
  7718. -static NTSTATUS nfs41_IoCtl(
  7719. -    IN OUT PRX_CONTEXT RxContext)
  7720. -{
  7721. -    NTSTATUS status = STATUS_INVALID_DEVICE_REQUEST;
  7722. -#ifdef DEBUG_IOCTL
  7723. -    DbgEn();
  7724. -    print_debug_header(RxContext);
  7725. -#endif /* DEBUG_IOCTL */
  7726. -    const ULONG iocontrolcode =
  7727. -        RxContext->LowIoContext.ParamsFor.IoCtl.IoControlCode;
  7728. -
  7729. -    DbgP("nfs41_IoCtl: IoControlCode=0x%lx, status=0x%lx\n",
  7730. -        (unsigned long)iocontrolcode, (long)status);
  7731. -
  7732. -#ifdef DEBUG_IOCTL
  7733. -    DbgEx();
  7734. -#endif
  7735. -    return status;
  7736. -}
  7737. -
  7738. -static NTSTATUS nfs41_CompleteBufferingStateChangeRequest(
  7739. -    IN OUT PRX_CONTEXT RxContext,
  7740. -    IN OUT PMRX_SRV_OPEN SrvOpen,
  7741. -    IN PVOID pContext)
  7742. -{
  7743. -    return STATUS_SUCCESS;
  7744. -}
  7745. -
  7746. -/* nfs41_FsdDispatch() - must be public symbol */
  7747. -NTSTATUS nfs41_FsdDispatch (
  7748. -    IN PDEVICE_OBJECT dev,
  7749. -    IN PIRP Irp)
  7750. -{
  7751. -#ifdef DEBUG_FSDDISPATCH
  7752. -    PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp );
  7753. -#endif
  7754. -    NTSTATUS status;
  7755. -
  7756. -#ifdef DEBUG_FSDDISPATCH
  7757. -    DbgEn();
  7758. -    DbgP("CURRENT IRP = %d.%d\n", IrpSp->MajorFunction, IrpSp->MinorFunction);
  7759. -    if(IrpSp->FileObject)
  7760. -        DbgP("FileOject 0x%p Filename '%wZ'\n", IrpSp->FileObject,
  7761. -                &IrpSp->FileObject->FileName);
  7762. -#endif
  7763. -
  7764. -    if (dev != (PDEVICE_OBJECT)nfs41_dev) {
  7765. -        print_error("*** not ours ***\n");
  7766. -        Irp->IoStatus.Status = STATUS_INVALID_DEVICE_REQUEST;
  7767. -        Irp->IoStatus.Information = 0;
  7768. -        IoCompleteRequest(Irp, IO_NO_INCREMENT );
  7769. -        status = STATUS_INVALID_DEVICE_REQUEST;
  7770. -        goto out;
  7771. -    }
  7772. -
  7773. -    status = RxFsdDispatch((PRDBSS_DEVICE_OBJECT)dev,Irp);
  7774. -    /* AGLO: 08/05/2009 - looks like RxFsdDispatch frees IrpSp */
  7775. -
  7776. -out:
  7777. -#ifdef DEBUG_FSDDISPATCH
  7778. -    DbgP("IoStatus status = 0x%lx info = 0x%x\n",
  7779. -        (long)Irp->IoStatus.Status,
  7780. -        Irp->IoStatus.Information);
  7781. -    DbgEx();
  7782. -#endif
  7783. -    return status;
  7784. -}
  7785. -
  7786. -static NTSTATUS nfs41_Unimplemented(
  7787. -    PRX_CONTEXT RxContext)
  7788. -{
  7789. -    return STATUS_NOT_IMPLEMENTED;
  7790. -}
  7791. -
  7792. -static NTSTATUS nfs41_AreFilesAliased(
  7793. -    PFCB a,
  7794. -    PFCB b)
  7795. -{
  7796. -    DbgP("nfs41_AreFilesAliased: a=0x%p b=%0x%p\n",
  7797. -        (void *)a, (void *)b);
  7798. -    return STATUS_NOT_IMPLEMENTED;
  7799. -}
  7800. -
  7801. -static NTSTATUS nfs41_init_ops()
  7802. -{
  7803. -    DbgEn();
  7804. -
  7805. -    ZeroAndInitializeNodeType(&nfs41_ops, RDBSS_NTC_MINIRDR_DISPATCH,
  7806. -        sizeof(MINIRDR_DISPATCH));
  7807. -
  7808. -#define FIXME_WORKAROUND_FOR_WIN10_SCAVENGER_CRASH 1
  7809. -#ifdef FIXME_WORKAROUND_FOR_WIN10_SCAVENGER_CRASH
  7810. -    /*
  7811. -     * gisburn: Ugly workaround for crash in Win10 scavenger code
  7812. -     * with a stack trace like this:
  7813. -     * -- snip --
  7814. -     * nt!KeBugCheckEx
  7815. -     * nt!KiBugCheckDispatch+0x69
  7816. -     * nt!KiFastFailDispatch+0xd0
  7817. -     * nt!KiRaiseSecurityCheckFailure+0x31d (TrapFrame @ fffffe0b`41ca0900)
  7818. -     * nfs41_driver!RtlFailFast(void)+0x5 (Inline Function @ fffff801`41ba47dd) [onecore\external\ifskit\inc\wdm.h @ 11545]
  7819. -     * nfs41_driver!FatalListEntryError(void)+0x5 (Inline Function @ fffff801`41ba47dd) [onecore\external\ifskit\inc\wdm.h @ 11778]
  7820. -     * nfs41_driver!RemoveEntryList(void)+0x33 (Inline Function @ fffff801`41ba47dd) [onecore\external\ifskit\inc\wdm.h @ 11811]
  7821. -     * nfs41_driver!RxpUndoScavengerFinalizationMarking(void * Instance = 0xffffca8f`b8f537d0)+0xad [base\fs\rdr2\rxce\scavengr.c @ 1154]
  7822. -     * nfs41_driver!RxScavengerFinalizeEntries(struct _RDBSS_DEVICE_OBJECT * RxDeviceObject = <Value unavailable error>)+0x407 [base\fs\rdr2\rxce\scavengr.c @ 1710]
  7823. -     * nfs41_driver!RxScavengerTimerRoutine(void * Context = 0xffffca8f`bb0d4060)+0x87 [base\fs\rdr2\rxce\scavengr.c @ 1826]
  7824. -     * nfs41_driver!RxpWorkerThreadDispatcher(struct _RX_WORK_QUEUE_ * pWorkQueue = 0xfffff801`41b99240, union _LARGE_INTEGER * pWaitInterval = 0x00000000`00000000)+0xbb [base\fs\rdr2\rxce\rxworkq.c @ 1343]
  7825. -     * nfs41_driver!RxBootstrapWorkerThreadDispatcher(struct _RX_WORK_QUEUE_ * pWorkQueue = <Value unavailable error>)+0xb [base\fs\rdr2\rxce\rxworkq.c @ 1469]
  7826. -     * nt!PspSystemThreadStartup+0x55
  7827. -     * nt!KiStartSystemThread+0x28
  7828. -     * -- snip --
  7829. -     *
  7830. -     * As workaround we "disable" the scavenger by only running it
  7831. -     * every 128 years, until then we should have found a fix.
  7832. -     */
  7833. -    nfs41_ops.ScavengerTimeout = 3600UL*24*365*128;
  7834. -#endif /* FIXME_WORKAROUND_FOR_WIN10_SCAVENGER_CRASH */
  7835. -
  7836. -    nfs41_ops.MRxFlags = (RDBSS_MANAGE_NET_ROOT_EXTENSION |
  7837. -                            RDBSS_MANAGE_V_NET_ROOT_EXTENSION |
  7838. -                            RDBSS_MANAGE_FCB_EXTENSION |
  7839. -                            RDBSS_MANAGE_FOBX_EXTENSION);
  7840. -
  7841. -    nfs41_ops.MRxSrvCallSize  = 0; // srvcall extension is not handled in rdbss
  7842. -    nfs41_ops.MRxNetRootSize  = sizeof(NFS41_NETROOT_EXTENSION);
  7843. -    nfs41_ops.MRxVNetRootSize = sizeof(NFS41_V_NET_ROOT_EXTENSION);
  7844. -    nfs41_ops.MRxFcbSize      = sizeof(NFS41_FCB);
  7845. -    nfs41_ops.MRxFobxSize     = sizeof(NFS41_FOBX);
  7846. -
  7847. -    // Mini redirector cancel routine ..
  7848. -    
  7849. -    nfs41_ops.MRxCancel = NULL;
  7850. -
  7851. -    //
  7852. -    // Mini redirector Start/Stop. Each mini-rdr can be started or stopped
  7853. -    // while the others continue to operate.
  7854. -    //
  7855. -
  7856. -    nfs41_ops.MRxStart                = (PMRX_CALLDOWN_CTX)nfs41_Start;
  7857. -    nfs41_ops.MRxStop                 = (PMRX_CALLDOWN_CTX)nfs41_Stop;
  7858. -    nfs41_ops.MRxDevFcbXXXControlFile =
  7859. -        (PMRX_CALLDOWN)nfs41_DevFcbXXXControlFile;
  7860. -
  7861. -    //
  7862. -    // Mini redirector name resolution.
  7863. -    //
  7864. -
  7865. -    nfs41_ops.MRxCreateSrvCall       =
  7866. -        (PMRX_CREATE_SRVCALL)nfs41_CreateSrvCall;
  7867. -    nfs41_ops.MRxSrvCallWinnerNotify =
  7868. -        (PMRX_SRVCALL_WINNER_NOTIFY)nfs41_SrvCallWinnerNotify;
  7869. -    nfs41_ops.MRxCreateVNetRoot      =
  7870. -        (PMRX_CREATE_V_NET_ROOT)nfs41_CreateVNetRoot;
  7871. -    nfs41_ops.MRxExtractNetRootName  =
  7872. -        (PMRX_EXTRACT_NETROOT_NAME)nfs41_ExtractNetRootName;
  7873. -    nfs41_ops.MRxFinalizeSrvCall     =
  7874. -        (PMRX_FINALIZE_SRVCALL_CALLDOWN)nfs41_FinalizeSrvCall;
  7875. -    nfs41_ops.MRxFinalizeNetRoot     =
  7876. -        (PMRX_FINALIZE_NET_ROOT_CALLDOWN)nfs41_FinalizeNetRoot;
  7877. -    nfs41_ops.MRxFinalizeVNetRoot    =
  7878. -        (PMRX_FINALIZE_V_NET_ROOT_CALLDOWN)nfs41_FinalizeVNetRoot;
  7879. -
  7880. -    //
  7881. -    // File System Object Creation/Deletion.
  7882. -    //
  7883. -
  7884. -    nfs41_ops.MRxCreate            =
  7885. -        (PMRX_CALLDOWN)nfs41_Create;
  7886. -    nfs41_ops.MRxCollapseOpen      =
  7887. -        (PMRX_CALLDOWN)nfs41_CollapseOpen;
  7888. -    nfs41_ops.MRxShouldTryToCollapseThisOpen =
  7889. -        (PMRX_CALLDOWN)nfs41_ShouldTryToCollapseThisOpen;
  7890. -    nfs41_ops.MRxExtendForCache    =
  7891. -        (PMRX_EXTENDFILE_CALLDOWN)nfs41_ExtendForCache;
  7892. -    nfs41_ops.MRxExtendForNonCache =
  7893. -        (PMRX_EXTENDFILE_CALLDOWN)nfs41_ExtendForCache;
  7894. -    nfs41_ops.MRxCloseSrvOpen      =
  7895. -        (PMRX_CALLDOWN)nfs41_CloseSrvOpen;
  7896. -    nfs41_ops.MRxFlush             =
  7897. -        (PMRX_CALLDOWN)nfs41_Flush;
  7898. -    nfs41_ops.MRxDeallocateForFcb  =
  7899. -        (PMRX_DEALLOCATE_FOR_FCB)nfs41_DeallocateForFcb;
  7900. -    nfs41_ops.MRxDeallocateForFobx =
  7901. -        (PMRX_DEALLOCATE_FOR_FOBX)nfs41_DeallocateForFobx;
  7902. -    nfs41_ops.MRxIsLockRealizable  =
  7903. -        (PMRX_IS_LOCK_REALIZABLE)nfs41_IsLockRealizable;
  7904. -
  7905. -    //
  7906. -    // File System Objects query/Set
  7907. -    //
  7908. -
  7909. -    nfs41_ops.MRxQueryDirectory       =
  7910. -        (PMRX_CALLDOWN)nfs41_QueryDirectory;
  7911. -    nfs41_ops.MRxQueryVolumeInfo      =
  7912. -        (PMRX_CALLDOWN)nfs41_QueryVolumeInformation;
  7913. -    nfs41_ops.MRxSetVolumeInfo        =
  7914. -        (PMRX_CALLDOWN)nfs41_Unimplemented;
  7915. -    nfs41_ops.MRxQueryEaInfo          =
  7916. -        (PMRX_CALLDOWN)nfs41_QueryEaInformation;
  7917. -    nfs41_ops.MRxSetEaInfo            =
  7918. -        (PMRX_CALLDOWN)nfs41_SetEaInformation;
  7919. -    nfs41_ops.MRxQuerySdInfo          =
  7920. -        (PMRX_CALLDOWN)nfs41_QuerySecurityInformation;
  7921. -    nfs41_ops.MRxSetSdInfo            =
  7922. -        (PMRX_CALLDOWN)nfs41_SetSecurityInformation;
  7923. -    nfs41_ops.MRxQueryFileInfo        =
  7924. -        (PMRX_CALLDOWN)nfs41_QueryFileInformation;
  7925. -    nfs41_ops.MRxSetFileInfo          =
  7926. -        (PMRX_CALLDOWN)nfs41_SetFileInformation;
  7927. -    nfs41_ops.MRxQueryQuotaInfo       =
  7928. -        (PMRX_CALLDOWN)nfs41_Unimplemented;
  7929. -    nfs41_ops.MRxSetQuotaInfo         =
  7930. -        (PMRX_CALLDOWN)nfs41_Unimplemented;
  7931. -
  7932. -    //
  7933. -    // Buffering state change
  7934. -    //
  7935. -
  7936. -    nfs41_ops.MRxComputeNewBufferingState =
  7937. -        (PMRX_COMPUTE_NEW_BUFFERING_STATE)nfs41_ComputeNewBufferingState;
  7938. -
  7939. -    //
  7940. -    // File System Object I/O
  7941. -    //
  7942. -
  7943. -    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_READ]            =
  7944. -        (PMRX_CALLDOWN)nfs41_Read;
  7945. -    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_WRITE]           =
  7946. -        (PMRX_CALLDOWN)nfs41_Write;
  7947. -    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_SHAREDLOCK]      =
  7948. -        (PMRX_CALLDOWN)nfs41_Lock;
  7949. -    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_EXCLUSIVELOCK]   =
  7950. -        (PMRX_CALLDOWN)nfs41_Lock;
  7951. -    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_UNLOCK]          =
  7952. -        (PMRX_CALLDOWN)nfs41_Unlock;
  7953. -    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_UNLOCK_MULTIPLE] =
  7954. -        (PMRX_CALLDOWN)nfs41_Unlock;
  7955. -    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_FSCTL]           =
  7956. -        (PMRX_CALLDOWN)nfs41_FsCtl;
  7957. -    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_IOCTL]           =
  7958. -        (PMRX_CALLDOWN)nfs41_IoCtl;
  7959. -
  7960. -    //
  7961. -    // Miscellanous
  7962. -    //
  7963. -
  7964. -    nfs41_ops.MRxCompleteBufferingStateChangeRequest =
  7965. -        (PMRX_CHANGE_BUFFERING_STATE_CALLDOWN)nfs41_CompleteBufferingStateChangeRequest;
  7966. -    nfs41_ops.MRxIsValidDirectory =
  7967. -        (PMRX_CHKDIR_CALLDOWN)nfs41_IsValidDirectory;
  7968. -
  7969. -    nfs41_ops.MRxTruncate =
  7970. -        (PMRX_CALLDOWN)nfs41_Unimplemented;
  7971. -    nfs41_ops.MRxZeroExtend =
  7972. -        (PMRX_CALLDOWN)nfs41_Unimplemented;
  7973. -    nfs41_ops.MRxAreFilesAliased =
  7974. -        (PMRX_CHKFCB_CALLDOWN)nfs41_AreFilesAliased;
  7975. -
  7976. -    DbgR();
  7977. -    return(STATUS_SUCCESS);
  7978. -}
  7979. -
  7980. -KSTART_ROUTINE fcbopen_main;
  7981. -VOID fcbopen_main(PVOID ctx)
  7982. -{
  7983. -    NTSTATUS status;
  7984. -    LARGE_INTEGER timeout;
  7985. -
  7986. -//    DbgEn();
  7987. -    timeout.QuadPart = RELATIVE(SECONDS(30));
  7988. -    while(1) {
  7989. -        PLIST_ENTRY pEntry;
  7990. -        nfs41_fcb_list_entry *cur;
  7991. -        status = KeDelayExecutionThread(KernelMode, TRUE, &timeout);
  7992. -        ExAcquireFastMutex(&fcblistLock);
  7993. -        pEntry = openlist.head.Flink;
  7994. -        while (!IsListEmpty(&openlist.head)) {
  7995. -            PNFS41_NETROOT_EXTENSION pNetRootContext;
  7996. -            nfs41_updowncall_entry *entry = NULL;
  7997. -            FILE_BASIC_INFORMATION binfo;
  7998. -            PNFS41_FCB nfs41_fcb;
  7999. -            cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry,
  8000. -                    nfs41_fcb_list_entry, next);
  8001. -
  8002. -#ifdef DEBUG_TIME_BASED_COHERENCY
  8003. -            DbgP("fcbopen_main: Checking attributes for fcb=0x%p "
  8004. -                "change_time=%llu skipping=%d\n", cur->fcb,
  8005. -                cur->ChangeTime, cur->skip);
  8006. -#endif
  8007. -            if (cur->skip) goto out;
  8008. -
  8009. -            /*
  8010. -             * This can only happen if |nfs41_DeallocateForFobx()|
  8011. -             * was called
  8012. -             */
  8013. -            if ((!cur->nfs41_fobx) || (!cur->nfs41_fobx->sec_ctx.ClientToken))
  8014. -                goto out;
  8015. -
  8016. -            if (!cur->nfs41_fobx->timebasedcoherency) {
  8017. -#ifdef DEBUG_TIME_BASED_COHERENCY
  8018. -                DbgP("fcbopen_main: timebasedcoherency disabled for "
  8019. -                    "fcb=0x%p, nfs41_fobx=0x%p\n", cur->fcb, cur->nfs41_fobx);
  8020. -#endif
  8021. -                goto out;
  8022. -            }
  8023. -
  8024. -            pNetRootContext =
  8025. -                NFS41GetNetRootExtension(cur->fcb->pNetRoot);
  8026. -            /* place an upcall for this srv_open */
  8027. -            status = nfs41_UpcallCreate(
  8028. -                NFS41_FILE_QUERY_TIME_BASED_COHERENCY,
  8029. -                &cur->nfs41_fobx->sec_ctx, cur->session,
  8030. -                cur->nfs41_fobx->nfs41_open_state,
  8031. -                pNetRootContext->nfs41d_version, NULL, &entry);
  8032. -            if (status) goto out;
  8033. -
  8034. -            entry->u.QueryFile.InfoClass = FileBasicInformation;
  8035. -            entry->buf = &binfo;
  8036. -            entry->buf_len = sizeof(binfo);
  8037. -
  8038. -            status = nfs41_UpcallWaitForReply(entry, UPCALL_TIMEOUT_DEFAULT);
  8039. -            if (status) goto out;
  8040. -
  8041. -            if (cur->ChangeTime != entry->ChangeTime) {
  8042. -                ULONG flag = DISABLE_CACHING;
  8043. -                PMRX_SRV_OPEN srv_open;
  8044. -                PLIST_ENTRY psrvEntry;
  8045. -#ifdef DEBUG_TIME_BASED_COHERENCY
  8046. -                DbgP("fcbopen_main: old ctime=%llu new_ctime=%llu\n",
  8047. -                    cur->ChangeTime, entry->ChangeTime);
  8048. -#endif
  8049. -                cur->ChangeTime = entry->ChangeTime;
  8050. -                cur->skip = TRUE;
  8051. -                psrvEntry = &cur->fcb->SrvOpenList;
  8052. -                psrvEntry = psrvEntry->Flink;
  8053. -                while (!IsListEmpty(&cur->fcb->SrvOpenList)) {
  8054. -                    srv_open = (PMRX_SRV_OPEN)CONTAINING_RECORD(psrvEntry,
  8055. -                            MRX_SRV_OPEN, SrvOpenQLinks);
  8056. -                    if (srv_open->DesiredAccess &
  8057. -                            (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA)) {
  8058. -#ifdef DEBUG_TIME_BASED_COHERENCY
  8059. -                        DbgP("fcbopen_main: ************ Invalidate the cache '%wZ'"
  8060. -                             "************\n", srv_open->pAlreadyPrefixedName);
  8061. -#endif
  8062. -                        RxIndicateChangeOfBufferingStateForSrvOpen(
  8063. -                            cur->fcb->pNetRoot->pSrvCall, srv_open,
  8064. -                            srv_open->Key, ULongToPtr(flag));
  8065. -                    }
  8066. -                    if (psrvEntry->Flink == &cur->fcb->SrvOpenList) {
  8067. -#ifdef DEBUG_TIME_BASED_COHERENCY
  8068. -                        DbgP("fcbopen_main: reached end of srvopen for fcb 0x%p\n",
  8069. -                            cur->fcb);
  8070. -#endif
  8071. -                        break;
  8072. -                    }
  8073. -                    psrvEntry = psrvEntry->Flink;
  8074. -                };
  8075. -            }
  8076. -            nfs41_fcb = (PNFS41_FCB)cur->fcb->Context;
  8077. -            nfs41_fcb->changeattr = entry->ChangeTime;
  8078. -out:
  8079. -            nfs41_UpcallDestroy(entry);
  8080. -            entry = NULL;
  8081. -            if (pEntry->Flink == &openlist.head) {
  8082. -#ifdef DEBUG_TIME_BASED_COHERENCY
  8083. -                DbgP("fcbopen_main: reached end of the fcb list\n");
  8084. -#endif
  8085. -                break;
  8086. -            }
  8087. -            pEntry = pEntry->Flink;
  8088. -        }
  8089. -        ExReleaseFastMutex(&fcblistLock);
  8090. -    }
  8091. -//    DbgEx();
  8092. -}
  8093. -
  8094. -
  8095. -/* Main driver entry point, must be public symbol */
  8096. -NTSTATUS DriverEntry(
  8097. -    IN PDRIVER_OBJECT drv,
  8098. -    IN PUNICODE_STRING path)
  8099. -{
  8100. -    NTSTATUS status;
  8101. -    ULONG flags = 0, i;
  8102. -    UNICODE_STRING dev_name, user_dev_name;
  8103. -    PNFS41_DEVICE_EXTENSION dev_exts;
  8104. -    TIME_FIELDS jan_1_1970 = {1970, 1, 1, 0, 0, 0, 0, 0};
  8105. -    ACCESS_MASK mask = 0;
  8106. -    OBJECT_ATTRIBUTES oattrs;
  8107. -
  8108. -    DbgEn();
  8109. -
  8110. -    status = RxDriverEntry(drv, path);
  8111. -    if (status != STATUS_SUCCESS) {
  8112. -        print_error("RxDriverEntry failed: 0x%08lx\n", status);
  8113. -        goto out;
  8114. -    }
  8115. -
  8116. -    RtlInitUnicodeString(&dev_name, NFS41_DEVICE_NAME);
  8117. -    SetFlag(flags, RX_REGISTERMINI_FLAG_DONT_PROVIDE_MAILSLOTS);
  8118. -
  8119. -    status = nfs41_init_ops();
  8120. -    if (status != STATUS_SUCCESS) {
  8121. -        print_error("nfs41_init_ops failed to initialize dispatch table\n");
  8122. -        goto out;
  8123. -    }
  8124. -
  8125. -    DbgP("calling RxRegisterMinirdr\n");
  8126. -    status = RxRegisterMinirdr(&nfs41_dev, drv, &nfs41_ops, flags, &dev_name,
  8127. -                sizeof(NFS41_DEVICE_EXTENSION),
  8128. -                FILE_DEVICE_NETWORK_FILE_SYSTEM, FILE_REMOTE_DEVICE);
  8129. -    if (status != STATUS_SUCCESS) {
  8130. -        print_error("RxRegisterMinirdr failed: 0x%08lx\n", status);
  8131. -        goto out;
  8132. -    }
  8133. -    nfs41_dev->Flags |= DO_BUFFERED_IO;
  8134. -
  8135. -    dev_exts = (PNFS41_DEVICE_EXTENSION)
  8136. -        ((PBYTE)(nfs41_dev) + sizeof(RDBSS_DEVICE_OBJECT));
  8137. -
  8138. -    RxDefineNode(dev_exts, NFS41_DEVICE_EXTENSION);
  8139. -    dev_exts->DeviceObject = nfs41_dev;
  8140. -    nfs41_create_volume_info((PFILE_FS_VOLUME_INFORMATION)dev_exts->VolAttrs,
  8141. -        &dev_exts->VolAttrsLen);
  8142. -
  8143. -    RtlInitUnicodeString(&user_dev_name, NFS41_SHADOW_DEVICE_NAME);
  8144. -    DbgP("calling IoCreateSymbolicLink '%wZ' '%wZ'\n", &user_dev_name, &dev_name);
  8145. -    status = IoCreateSymbolicLink(&user_dev_name, &dev_name);
  8146. -    if (status != STATUS_SUCCESS) {
  8147. -        print_error("Device name IoCreateSymbolicLink failed: 0x%08lx\n", status);
  8148. -        goto out_unregister;
  8149. -    }
  8150. -
  8151. -    KeInitializeEvent(&upcallEvent, SynchronizationEvent, FALSE );
  8152. -    ExInitializeFastMutex(&upcallLock);
  8153. -    ExInitializeFastMutex(&downcallLock);
  8154. -    ExInitializeFastMutex(&openOwnerLock);
  8155. -    ExInitializeFastMutex(&fcblistLock);
  8156. -    InitializeListHead(&upcall.head);
  8157. -    InitializeListHead(&downcall.head);
  8158. -    InitializeListHead(&openlist.head);
  8159. -#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  8160. -    /*
  8161. -     * The |Depth| parameter is unfortunately ignored in Win10,
  8162. -     * otherwise we could use |MmQuerySystemSize()| to scale the
  8163. -     * lookasidelists
  8164. -     */
  8165. -    ExInitializeNPagedLookasideList(
  8166. -        &updowncall_entry_upcall_lookasidelist, NULL, NULL,
  8167. -        POOL_NX_ALLOCATION, sizeof(nfs41_updowncall_entry),
  8168. -        NFS41_MM_POOLTAG_UP, 0);
  8169. -    ExInitializeNPagedLookasideList(
  8170. -        &updowncall_entry_downcall_lookasidelist, NULL, NULL,
  8171. -        POOL_NX_ALLOCATION, sizeof(nfs41_updowncall_entry),
  8172. -        NFS41_MM_POOLTAG_DOWN, 0);
  8173. -#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  8174. -    InitializeObjectAttributes(&oattrs, NULL, OBJ_KERNEL_HANDLE, NULL, NULL);
  8175. -    status = PsCreateSystemThread(&dev_exts->openlistHandle, mask,
  8176. -        &oattrs, NULL, NULL, &fcbopen_main, NULL);
  8177. -    if (status != STATUS_SUCCESS)
  8178. -        goto out_unregister;
  8179. -
  8180. -    drv->DriverUnload = nfs41_driver_unload;
  8181. -
  8182. -    for (i = 0; i <= IRP_MJ_MAXIMUM_FUNCTION; i++)
  8183. -        drv->MajorFunction[i] = (PDRIVER_DISPATCH)nfs41_FsdDispatch;
  8184. -
  8185. -    RtlTimeFieldsToTime(&jan_1_1970, &unix_time_diff);
  8186. -
  8187. -out_unregister:
  8188. -    if (status != STATUS_SUCCESS)
  8189. -        RxUnregisterMinirdr(nfs41_dev);
  8190. -out:
  8191. -    DbgEx();
  8192. -    return status;
  8193. -}
  8194. -
  8195. -
  8196. -/* nfs41_driver_unload() - must be public symbol */
  8197. -VOID nfs41_driver_unload(IN PDRIVER_OBJECT drv)
  8198. -{
  8199. -    PRX_CONTEXT RxContext;
  8200. -    NTSTATUS    status;
  8201. -    UNICODE_STRING dev_name, pipe_name;
  8202. -
  8203. -    DbgEn();
  8204. -
  8205. -    RxContext = RxCreateRxContext(NULL, nfs41_dev, RX_CONTEXT_FLAG_IN_FSP);
  8206. -    if (RxContext == NULL) {
  8207. -        status = STATUS_INSUFFICIENT_RESOURCES;
  8208. -        goto unload;
  8209. -    }
  8210. -    status = RxStopMinirdr(RxContext, &RxContext->PostRequest);
  8211. -    RxDereferenceAndDeleteRxContext(RxContext);
  8212. -
  8213. -unload:
  8214. -    RtlInitUnicodeString(&dev_name, NFS41_SHADOW_DEVICE_NAME);
  8215. -    status = IoDeleteSymbolicLink(&dev_name);
  8216. -    if (status != STATUS_SUCCESS) {
  8217. -        print_error("couldn't delete device symbolic link\n");
  8218. -    }
  8219. -    RtlInitUnicodeString(&pipe_name, NFS41_SHADOW_PIPE_NAME);
  8220. -    status = IoDeleteSymbolicLink(&pipe_name);
  8221. -    if (status != STATUS_SUCCESS) {
  8222. -        print_error("couldn't delete pipe symbolic link\n");
  8223. -    }
  8224. -    RxUnload(drv);
  8225. -
  8226. -    DbgP("driver unloaded 0x%p\n", drv);
  8227. -    DbgR();
  8228. -}
  8229. diff --git a/sys/nfs41_driver.h b/sys/nfs41_driver.h
  8230. index cee3bf5..9d4ad15 100644
  8231. --- a/sys/nfs41_driver.h
  8232. +++ b/sys/nfs41_driver.h
  8233. @@ -1,8 +1,10 @@
  8234.  /* NFSv4.1 client for Windows
  8235. - * Copyright (C) 2012 The Regents of the University of Michigan
  8236. + * Copyright (C) 2012 The Regents of the University of Michigan
  8237. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  8238.   *
  8239.   * Olga Kornievskaia <aglo@umich.edu>
  8240.   * Casey Bodley <cbodley@umich.edu>
  8241. + * Roland Mainz <roland.mainz@nrubsig.org>
  8242.   *
  8243.   * This library is free software; you can redistribute it and/or modify it
  8244.   * under the terms of the GNU Lesser General Public License as published by
  8245. @@ -18,9 +20,8 @@
  8246.   * along with this library; if not, write to the Free Software Foundation,
  8247.   * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  8248.   */
  8249. -
  8250.  #ifndef _NFS41_DRIVER_
  8251. -#define _NFS41_DRIVER_
  8252. +#define _NFS41_DRIVER_ 1
  8253.  
  8254.  #define NFS41_DEVICE_NAME L"\\Device\\nfs41_driver"
  8255.  #define NFS41_SHADOW_DEVICE_NAME L"\\??\\nfs41_driver"
  8256. diff --git a/sys/nfs41sys_acl.c b/sys/nfs41sys_acl.c
  8257. new file mode 100644
  8258. index 0000000..02e7324
  8259. --- /dev/null
  8260. +++ b/sys/nfs41sys_acl.c
  8261. @@ -0,0 +1,431 @@
  8262. +/* NFSv4.1 client for Windows
  8263. + * Copyright (C) 2012 The Regents of the University of Michigan
  8264. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  8265. + *
  8266. + * Olga Kornievskaia <aglo@umich.edu>
  8267. + * Casey Bodley <cbodley@umich.edu>
  8268. + * Roland Mainz <roland.mainz@nrubsig.org>
  8269. + *
  8270. + * This library is free software; you can redistribute it and/or modify it
  8271. + * under the terms of the GNU Lesser General Public License as published by
  8272. + * the Free Software Foundation; either version 2.1 of the License, or (at
  8273. + * your option) any later version.
  8274. + *
  8275. + * This library is distributed in the hope that it will be useful, but
  8276. + * without any warranty; without even the implied warranty of merchantability
  8277. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  8278. + * License for more details.
  8279. + *
  8280. + * You should have received a copy of the GNU Lesser General Public License
  8281. + * along with this library; if not, write to the Free Software Foundation,
  8282. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  8283. + */
  8284. +
  8285. +#ifndef _KERNEL_MODE
  8286. +#error module requires kernel mode
  8287. +#endif
  8288. +
  8289. +#if ((__STDC_VERSION__-0) < 201710L)
  8290. +#error Code requires ISO C17
  8291. +#endif
  8292. +
  8293. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  8294. +#if _MSC_VER >= 1900
  8295. +#if defined(_WIN64) && defined(_M_X64)
  8296. +#ifndef _AMD64_
  8297. +#define _AMD64_
  8298. +#endif
  8299. +#elif defined(_WIN32) && defined(_M_IX86)
  8300. +#ifndef _X86_
  8301. +#define _X86_
  8302. +#endif
  8303. +#elif defined(_WIN64) && defined(_M_ARM64)
  8304. +#ifndef _ARM64_
  8305. +#define _ARM64_
  8306. +#endif
  8307. +#elif defined(_WIN32) && defined(_M_ARM)
  8308. +#ifndef _ARM_
  8309. +#define _ARM_
  8310. +#endif
  8311. +#else
  8312. +#error Unsupported arch
  8313. +#endif
  8314. +#endif /* _MSC_VER >= 1900 */
  8315. +
  8316. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  8317. +#include <rx.h>
  8318. +#include <windef.h>
  8319. +#include <winerror.h>
  8320. +
  8321. +#include <Ntstrsafe.h>
  8322. +
  8323. +#include "nfs41sys_buildconfig.h"
  8324. +
  8325. +#include "nfs41_driver.h"
  8326. +#include "nfs41sys_debug.h"
  8327. +#include "nfs41_build_features.h"
  8328. +
  8329. +#include "nfs41sys_driver.h"
  8330. +#include "nfs41sys_util.h"
  8331. +
  8332. +
  8333. +NTSTATUS marshal_nfs41_getacl(
  8334. +    nfs41_updowncall_entry *entry,
  8335. +    unsigned char *buf,
  8336. +    ULONG buf_len,
  8337. +    ULONG *len)
  8338. +{
  8339. +    NTSTATUS status = STATUS_SUCCESS;
  8340. +    ULONG header_len = 0;
  8341. +    unsigned char *tmp = buf;
  8342. +
  8343. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  8344. +    if (status) goto out;
  8345. +    else tmp += *len;
  8346. +
  8347. +    header_len = *len + sizeof(SECURITY_INFORMATION);
  8348. +    if (header_len > buf_len) {
  8349. +        status = STATUS_INSUFFICIENT_RESOURCES;
  8350. +        goto out;
  8351. +    }
  8352. +
  8353. +    RtlCopyMemory(tmp, &entry->u.Acl.query, sizeof(SECURITY_INFORMATION));
  8354. +    *len = header_len;
  8355. +
  8356. +#ifdef DEBUG_MARSHAL_DETAIL
  8357. +    DbgP("marshal_nfs41_getacl: class=0x%x\n", entry->u.Acl.query);
  8358. +#endif
  8359. +out:
  8360. +    return status;
  8361. +}
  8362. +
  8363. +NTSTATUS marshal_nfs41_setacl(
  8364. +    nfs41_updowncall_entry *entry,
  8365. +    unsigned char *buf,
  8366. +    ULONG buf_len,
  8367. +    ULONG *len)
  8368. +{
  8369. +    NTSTATUS status = STATUS_SUCCESS;
  8370. +    ULONG header_len = 0;
  8371. +    unsigned char *tmp = buf;
  8372. +
  8373. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  8374. +    if (status) goto out;
  8375. +    else tmp += *len;
  8376. +
  8377. +    header_len = *len + sizeof(SECURITY_INFORMATION) +
  8378. +        sizeof(ULONG) + entry->buf_len;
  8379. +    if (header_len > buf_len) {
  8380. +        status = STATUS_INSUFFICIENT_RESOURCES;
  8381. +        goto out;
  8382. +    }
  8383. +
  8384. +    RtlCopyMemory(tmp, &entry->u.Acl.query, sizeof(SECURITY_INFORMATION));
  8385. +    tmp += sizeof(SECURITY_INFORMATION);
  8386. +    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  8387. +    tmp += sizeof(ULONG);
  8388. +    RtlCopyMemory(tmp, entry->buf, entry->buf_len);
  8389. +    *len = header_len;
  8390. +
  8391. +#ifdef DEBUG_MARSHAL_DETAIL
  8392. +    DbgP("marshal_nfs41_setacl: class=0x%x sec_desc_len=%lu\n",
  8393. +         entry->u.Acl.query, entry->buf_len);
  8394. +#endif
  8395. +out:
  8396. +    return status;
  8397. +}
  8398. +
  8399. +NTSTATUS unmarshal_nfs41_getacl(
  8400. +    nfs41_updowncall_entry *cur,
  8401. +    unsigned char **buf)
  8402. +{
  8403. +    NTSTATUS status = STATUS_SUCCESS;
  8404. +    DWORD buf_len;
  8405. +
  8406. +    RtlCopyMemory(&buf_len, *buf, sizeof(DWORD));
  8407. +    *buf += sizeof(DWORD);
  8408. +    cur->buf = RxAllocatePoolWithTag(NonPagedPoolNx,
  8409. +        buf_len, NFS41_MM_POOLTAG_ACL);
  8410. +    if (cur->buf == NULL) {
  8411. +        cur->status = status = STATUS_INSUFFICIENT_RESOURCES;
  8412. +        goto out;
  8413. +    }
  8414. +    RtlCopyMemory(cur->buf, *buf, buf_len);
  8415. +    if (buf_len > cur->buf_len)
  8416. +        cur->status = STATUS_BUFFER_TOO_SMALL;
  8417. +    cur->buf_len = buf_len;
  8418. +
  8419. +out:
  8420. +    return status;
  8421. +}
  8422. +
  8423. +NTSTATUS map_query_acl_error(
  8424. +    DWORD error)
  8425. +{
  8426. +    switch (error) {
  8427. +    case NO_ERROR:                  return STATUS_SUCCESS;
  8428. +    case ERROR_NOT_SUPPORTED:       return STATUS_NOT_SUPPORTED;
  8429. +    case ERROR_NONE_MAPPED:         return STATUS_NONE_MAPPED;
  8430. +    case ERROR_ACCESS_DENIED:       return STATUS_ACCESS_DENIED;
  8431. +    case ERROR_FILE_NOT_FOUND:      return STATUS_OBJECT_NAME_NOT_FOUND;
  8432. +    case ERROR_INVALID_PARAMETER:   return STATUS_INVALID_PARAMETER;
  8433. +    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  8434. +    default:
  8435. +        print_error("map_query_acl_error: "
  8436. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  8437. +            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", error);
  8438. +    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  8439. +    }
  8440. +}
  8441. +
  8442. +static
  8443. +NTSTATUS check_nfs41_getacl_args(
  8444. +    PRX_CONTEXT RxContext)
  8445. +{
  8446. +    NTSTATUS status = STATUS_SUCCESS;
  8447. +    SECURITY_INFORMATION info_class =
  8448. +        RxContext->CurrentIrpSp->Parameters.QuerySecurity.SecurityInformation;
  8449. +
  8450. +    /* we don't support sacls */
  8451. +    if (info_class == SACL_SECURITY_INFORMATION ||
  8452. +            info_class == LABEL_SECURITY_INFORMATION) {
  8453. +        status = STATUS_NOT_SUPPORTED;
  8454. +        goto out;
  8455. +    }
  8456. +    if (RxContext->CurrentIrp->UserBuffer == NULL &&
  8457. +            RxContext->CurrentIrpSp->Parameters.QuerySecurity.Length)
  8458. +        status = STATUS_INVALID_USER_BUFFER;
  8459. +out:
  8460. +    return status;
  8461. +}
  8462. +
  8463. +NTSTATUS nfs41_QuerySecurityInformation(
  8464. +    IN OUT PRX_CONTEXT RxContext)
  8465. +{
  8466. +    NTSTATUS status = STATUS_NOT_SUPPORTED;
  8467. +    nfs41_updowncall_entry *entry;
  8468. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  8469. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  8470. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  8471. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  8472. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  8473. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  8474. +    SECURITY_INFORMATION info_class =
  8475. +        RxContext->CurrentIrpSp->Parameters.QuerySecurity.SecurityInformation;
  8476. +#ifdef ENABLE_TIMINGS
  8477. +    LARGE_INTEGER t1, t2;
  8478. +    t1 = KeQueryPerformanceCounter(NULL);
  8479. +#endif
  8480. +
  8481. +#ifdef DEBUG_ACL_QUERY
  8482. +    DbgEn();
  8483. +    print_debug_header(RxContext);
  8484. +    print_acl_args(info_class);
  8485. +#endif
  8486. +
  8487. +    status = check_nfs41_getacl_args(RxContext);
  8488. +    if (status) goto out;
  8489. +
  8490. +    if (nfs41_fobx->acl && nfs41_fobx->acl_len) {
  8491. +        LARGE_INTEGER current_time;
  8492. +        KeQuerySystemTime(&current_time);
  8493. +#ifdef DEBUG_ACL_QUERY
  8494. +        DbgP("CurrentTime 0x%lx Saved Acl time 0x%lx\n",
  8495. +            current_time.QuadPart, nfs41_fobx->time.QuadPart);
  8496. +#endif
  8497. +        if (current_time.QuadPart - nfs41_fobx->time.QuadPart <= 20*1000) {
  8498. +            PSECURITY_DESCRIPTOR sec_desc = (PSECURITY_DESCRIPTOR)
  8499. +                RxContext->CurrentIrp->UserBuffer;
  8500. +            RtlCopyMemory(sec_desc, nfs41_fobx->acl, nfs41_fobx->acl_len);
  8501. +            RxContext->IoStatusBlock.Information =
  8502. +                RxContext->InformationToReturn = nfs41_fobx->acl_len;
  8503. +            RxContext->IoStatusBlock.Status = status = STATUS_SUCCESS;
  8504. +#ifdef ENABLE_TIMINGS
  8505. +            InterlockedIncrement(&getacl.sops);
  8506. +            InterlockedAdd64(&getacl.size, nfs41_fobx->acl_len);
  8507. +#endif
  8508. +        } else status = 1;
  8509. +        RxFreePool(nfs41_fobx->acl);
  8510. +        nfs41_fobx->acl = NULL;
  8511. +        nfs41_fobx->acl_len = 0;
  8512. +        if (!status)
  8513. +            goto out;
  8514. +    }
  8515. +
  8516. +    status = nfs41_UpcallCreate(NFS41_ACL_QUERY, &nfs41_fobx->sec_ctx,
  8517. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  8518. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  8519. +    if (status) goto out;
  8520. +
  8521. +    entry->u.Acl.query = info_class;
  8522. +    /* we can't provide RxContext->CurrentIrp->UserBuffer to the upcall thread
  8523. +     * because it becomes an invalid pointer with that execution context
  8524. +     */
  8525. +    entry->buf_len = RxContext->CurrentIrpSp->Parameters.QuerySecurity.Length;
  8526. +
  8527. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  8528. +    if (status) goto out;
  8529. +
  8530. +    if (entry->status == STATUS_BUFFER_TOO_SMALL) {
  8531. +#ifdef DEBUG_ACL_QUERY
  8532. +        DbgP("nfs41_QuerySecurityInformation: provided buffer size=%d but we "
  8533. +             "need %lu\n",
  8534. +             RxContext->CurrentIrpSp->Parameters.QuerySecurity.Length,
  8535. +             entry->buf_len);
  8536. +#endif
  8537. +        status = STATUS_BUFFER_OVERFLOW;
  8538. +        RxContext->InformationToReturn = entry->buf_len;
  8539. +
  8540. +        /* Save ACL buffer */
  8541. +        nfs41_fobx->acl = entry->buf;
  8542. +        nfs41_fobx->acl_len = entry->buf_len;
  8543. +        KeQuerySystemTime(&nfs41_fobx->time);
  8544. +    } else if (entry->status == STATUS_SUCCESS) {
  8545. +        PSECURITY_DESCRIPTOR sec_desc = (PSECURITY_DESCRIPTOR)
  8546. +            RxContext->CurrentIrp->UserBuffer;
  8547. +        RtlCopyMemory(sec_desc, entry->buf, entry->buf_len);
  8548. +#ifdef ENABLE_TIMINGS
  8549. +        InterlockedIncrement(&getacl.sops);
  8550. +        InterlockedAdd64(&getacl.size, entry->u.Acl.buf_len);
  8551. +#endif
  8552. +        RxFreePool(entry->buf);
  8553. +        entry->buf = NULL;
  8554. +        nfs41_fobx->acl = NULL;
  8555. +        nfs41_fobx->acl_len = 0;
  8556. +        RxContext->IoStatusBlock.Information = RxContext->InformationToReturn =
  8557. +            entry->buf_len;
  8558. +        RxContext->IoStatusBlock.Status = status = STATUS_SUCCESS;
  8559. +    } else {
  8560. +        status = map_query_acl_error(entry->status);
  8561. +    }
  8562. +    nfs41_UpcallDestroy(entry);
  8563. +out:
  8564. +#ifdef ENABLE_TIMINGS
  8565. +    t2 = KeQueryPerformanceCounter(NULL);
  8566. +    /* only count getacl that we made an upcall for */
  8567. +    if (status == STATUS_BUFFER_OVERFLOW) {
  8568. +        InterlockedIncrement(&getacl.tops);
  8569. +        InterlockedAdd64(&getacl.ticks, t2.QuadPart - t1.QuadPart);
  8570. +    }
  8571. +#ifdef ENABLE_INDV_TIMINGS
  8572. +    DbgP("nfs41_QuerySecurityInformation: delta = %d op=%d sum=%d\n",
  8573. +        t2.QuadPart - t1.QuadPart, getacl.tops, getacl.ticks);
  8574. +#endif
  8575. +#endif
  8576. +#ifdef DEBUG_ACL_QUERY
  8577. +    DbgEx();
  8578. +#endif
  8579. +    return status;
  8580. +}
  8581. +
  8582. +static
  8583. +NTSTATUS check_nfs41_setacl_args(
  8584. +    PRX_CONTEXT RxContext)
  8585. +{
  8586. +    NTSTATUS status = STATUS_SUCCESS;
  8587. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  8588. +        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  8589. +    SECURITY_INFORMATION info_class =
  8590. +        RxContext->CurrentIrpSp->Parameters.SetSecurity.SecurityInformation;
  8591. +
  8592. +    if (pVNetRootContext->read_only) {
  8593. +        print_error("check_nfs41_setacl_args: Read-only mount\n");
  8594. +        status = STATUS_MEDIA_WRITE_PROTECTED;
  8595. +        goto out;
  8596. +    }
  8597. +    /* we don't support sacls */
  8598. +    if (info_class == SACL_SECURITY_INFORMATION  ||
  8599. +            info_class == LABEL_SECURITY_INFORMATION) {
  8600. +        status = STATUS_NOT_SUPPORTED;
  8601. +        goto out;
  8602. +    }
  8603. +out:
  8604. +    return status;
  8605. +}
  8606. +
  8607. +NTSTATUS nfs41_SetSecurityInformation(
  8608. +    IN OUT PRX_CONTEXT RxContext)
  8609. +{
  8610. +    NTSTATUS status = STATUS_NOT_SUPPORTED;
  8611. +    nfs41_updowncall_entry *entry;
  8612. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  8613. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  8614. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  8615. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  8616. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  8617. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  8618. +    __notnull PSECURITY_DESCRIPTOR sec_desc =
  8619. +        RxContext->CurrentIrpSp->Parameters.SetSecurity.SecurityDescriptor;
  8620. +    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  8621. +    SECURITY_INFORMATION info_class =
  8622. +        RxContext->CurrentIrpSp->Parameters.SetSecurity.SecurityInformation;
  8623. +#ifdef ENABLE_TIMINGS
  8624. +    LARGE_INTEGER t1, t2;
  8625. +    t1 = KeQueryPerformanceCounter(NULL);
  8626. +#endif
  8627. +
  8628. +#ifdef DEBUG_ACL_SET
  8629. +    DbgEn();
  8630. +    print_debug_header(RxContext);
  8631. +    print_acl_args(info_class);
  8632. +#endif
  8633. +
  8634. +    status = check_nfs41_setacl_args(RxContext);
  8635. +    if (status) goto out;
  8636. +
  8637. +    /* check that ACL is present */
  8638. +    if (info_class & DACL_SECURITY_INFORMATION) {
  8639. +        PACL acl;
  8640. +        BOOLEAN present, dacl_default;
  8641. +        status = RtlGetDaclSecurityDescriptor(sec_desc, &present, &acl,
  8642. +                    &dacl_default);
  8643. +        if (status) {
  8644. +            DbgP("RtlGetDaclSecurityDescriptor failed 0x%x\n", status);
  8645. +            goto out;
  8646. +        }
  8647. +        if (present == FALSE) {
  8648. +            DbgP("NO ACL present\n");
  8649. +            goto out;
  8650. +        }
  8651. +    }
  8652. +
  8653. +    status = nfs41_UpcallCreate(NFS41_ACL_SET, &nfs41_fobx->sec_ctx,
  8654. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  8655. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  8656. +    if (status) goto out;
  8657. +
  8658. +    entry->u.Acl.query = info_class;
  8659. +    entry->buf = sec_desc;
  8660. +    entry->buf_len = RtlLengthSecurityDescriptor(sec_desc);
  8661. +#ifdef ENABLE_TIMINGS
  8662. +    InterlockedIncrement(&setacl.sops);
  8663. +    InterlockedAdd64(&setacl.size, entry->u.Acl.buf_len);
  8664. +#endif
  8665. +
  8666. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  8667. +    if (status) goto out;
  8668. +
  8669. +    status = map_query_acl_error(entry->status);
  8670. +    if (!status) {
  8671. +        if (!nfs41_fobx->deleg_type && entry->ChangeTime &&
  8672. +                (SrvOpen->DesiredAccess &
  8673. +                (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA)))
  8674. +            nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime);
  8675. +        nfs41_fcb->changeattr = entry->ChangeTime;
  8676. +    }
  8677. +    nfs41_UpcallDestroy(entry);
  8678. +out:
  8679. +#ifdef ENABLE_TIMINGS
  8680. +    t2 = KeQueryPerformanceCounter(NULL);
  8681. +    InterlockedIncrement(&setacl.tops);
  8682. +    InterlockedAdd64(&setacl.ticks, t2.QuadPart - t1.QuadPart);
  8683. +#ifdef ENABLE_INDV_TIMINGS
  8684. +    DbgP("nfs41_SetSecurityInformation delta = %d op=%d sum=%d\n",
  8685. +        t2.QuadPart - t1.QuadPart, setacl.tops, setacl.ticks);
  8686. +#endif
  8687. +#endif
  8688. +#ifdef DEBUG_ACL_SET
  8689. +    DbgEx();
  8690. +#endif
  8691. +    return status;
  8692. +}
  8693. diff --git a/sys/nfs41sys_buildconfig.h b/sys/nfs41sys_buildconfig.h
  8694. new file mode 100644
  8695. index 0000000..3774af6
  8696. --- /dev/null
  8697. +++ b/sys/nfs41sys_buildconfig.h
  8698. @@ -0,0 +1,66 @@
  8699. +/* NFSv4.1 client for Windows
  8700. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  8701. + *
  8702. + * Roland Mainz <roland.mainz@nrubsig.org>
  8703. + *
  8704. + * This library is free software; you can redistribute it and/or modify it
  8705. + * under the terms of the GNU Lesser General Public License as published by
  8706. + * the Free Software Foundation; either version 2.1 of the License, or (at
  8707. + * your option) any later version.
  8708. + *
  8709. + * This library is distributed in the hope that it will be useful, but
  8710. + * without any warranty; without even the implied warranty of merchantability
  8711. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  8712. + * License for more details.
  8713. + *
  8714. + * You should have received a copy of the GNU Lesser General Public License
  8715. + * along with this library; if not, write to the Free Software Foundation,
  8716. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  8717. + */
  8718. +
  8719. +#ifndef _NFS41SYS_BUILDCONFIG_H_
  8720. +#define _NFS41SYS_BUILDCONFIG_H_ 1
  8721. +
  8722. +/* Driver build config */
  8723. +#define USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM 1
  8724. +// #define LOOKASIDELISTS_STATS 1
  8725. +
  8726. +// #define USE_ENTIRE_PATH_FOR_NETROOT 1
  8727. +
  8728. +/* debugging printout defines */
  8729. +#if defined(_DEBUG)
  8730. +/* Debug build defines follow... */
  8731. +#define DEBUG_MARSHAL_HEADER
  8732. +#define DEBUG_MARSHAL_DETAIL
  8733. +//#define DEBUG_MARSHAL_DETAIL_RW
  8734. +//#define DEBUG_SECURITY_TOKEN
  8735. +#define DEBUG_MOUNTCONFIG
  8736. +//#define DEBUG_OPEN
  8737. +//#define DEBUG_CLOSE
  8738. +//#define DEBUG_CACHE
  8739. +#define DEBUG_INVALIDATE_CACHE
  8740. +//#define DEBUG_READ
  8741. +//#define DEBUG_WRITE
  8742. +//#define DEBUG_DIR_QUERY
  8743. +//#define DEBUG_FILE_QUERY
  8744. +//#define DEBUG_FILE_SET
  8745. +//#define DEBUG_ACL_QUERY
  8746. +//#define DEBUG_ACL_SET
  8747. +//#define DEBUG_EA_QUERY
  8748. +//#define DEBUG_EA_SET
  8749. +//#define DEBUG_LOCK
  8750. +#define DEBUG_FSCTL
  8751. +#define DEBUG_IOCTL
  8752. +#define DEBUG_TIME_BASED_COHERENCY
  8753. +#define DEBUG_MOUNT
  8754. +//#define DEBUG_VOLUME_QUERY
  8755. +
  8756. +//#define ENABLE_TIMINGS
  8757. +//#define ENABLE_INDV_TIMINGS
  8758. +#elif defined(NDEBUG)
  8759. +/* Release build defines follow... */
  8760. +#else
  8761. +#error Neither _DEBUG NOR _NDEBUG defined
  8762. +#endif
  8763. +
  8764. +#endif /* !_NFS41SYS_BUILDCONFIG_H_ */
  8765. diff --git a/sys/nfs41_debug.c b/sys/nfs41sys_debug.c
  8766. similarity index 92%
  8767. rename from sys/nfs41_debug.c
  8768. rename to sys/nfs41sys_debug.c
  8769. index 23673bf..7bb0f33 100644
  8770. --- a/sys/nfs41_debug.c
  8771. +++ b/sys/nfs41sys_debug.c
  8772. @@ -1,5 +1,5 @@
  8773. -/* NFSv4.1 client for Windows
  8774. - * Copyright (C) 2012 The Regents of the University of Michigan
  8775. +/* NFSv4.1 client for Windows
  8776. + * Copyright (C) 2012 The Regents of the University of Michigan
  8777.   *
  8778.   * Olga Kornievskaia <aglo@umich.edu>
  8779.   * Casey Bodley <cbodley@umich.edu>
  8780. @@ -7,17 +7,17 @@
  8781.   *
  8782.   * This library is free software; you can redistribute it and/or modify it
  8783.   * under the terms of the GNU Lesser General Public License as published by
  8784. - * the Free Software Foundation; either version 2.1 of the License, or (at
  8785. - * your option) any later version.
  8786. - *
  8787. - * This library is distributed in the hope that it will be useful, but
  8788. - * without any warranty; without even the implied warranty of merchantability
  8789. - * or fitness for a particular purpose.  See the GNU Lesser General Public
  8790. - * License for more details.
  8791. - *
  8792. - * You should have received a copy of the GNU Lesser General Public License
  8793. - * along with this library; if not, write to the Free Software Foundation,
  8794. - * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  8795. + * the Free Software Foundation; either version 2.1 of the License, or (at
  8796. + * your option) any later version.
  8797. + *
  8798. + * This library is distributed in the hope that it will be useful, but
  8799. + * without any warranty; without even the implied warranty of merchantability
  8800. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  8801. + * License for more details.
  8802. + *
  8803. + * You should have received a copy of the GNU Lesser General Public License
  8804. + * along with this library; if not, write to the Free Software Foundation,
  8805. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  8806.   */
  8807.  
  8808.  #define MINIRDR__NAME "Value is ignored, only fact of definition"
  8809. @@ -49,30 +49,30 @@
  8810.  #include <rx.h>
  8811.  
  8812.  #include "nfs41_driver.h"
  8813. -#include "nfs41_debug.h"
  8814. -#include <stdio.h>
  8815. -#include <stdarg.h>
  8816. -#include <ntstrsafe.h>
  8817. -#include <winerror.h>
  8818. -
  8819. -//#define INCLUDE_TIMESTAMPS
  8820. -
  8821. -ULONG __cdecl DbgP(IN PCCH fmt, ...)
  8822. -{
  8823. -    CHAR msg[512];
  8824. -    va_list args;
  8825. -    NTSTATUS status;
  8826. -
  8827. -    va_start(args, fmt);
  8828. -    ASSERT(fmt != NULL);
  8829. -    status = RtlStringCbVPrintfA(msg, sizeof(msg), fmt, args);
  8830. -    if (NT_SUCCESS(status)) {
  8831. -#ifdef INCLUDE_TIMESTAMPS
  8832. -        LARGE_INTEGER timestamp, local_time;
  8833. -        TIME_FIELDS time_fields;
  8834. -
  8835. -        KeQuerySystemTime(&timestamp);
  8836. -        ExSystemTimeToLocalTime(&timestamp,&local_time);
  8837. +#include "nfs41sys_debug.h"
  8838. +#include <stdio.h>
  8839. +#include <stdarg.h>
  8840. +#include <ntstrsafe.h>
  8841. +#include <winerror.h>
  8842. +
  8843. +//#define INCLUDE_TIMESTAMPS
  8844. +
  8845. +ULONG __cdecl DbgP(IN PCCH fmt, ...)
  8846. +{
  8847. +    CHAR msg[512];
  8848. +    va_list args;
  8849. +    NTSTATUS status;
  8850. +
  8851. +    va_start(args, fmt);
  8852. +    ASSERT(fmt != NULL);
  8853. +    status = RtlStringCbVPrintfA(msg, sizeof(msg), fmt, args);
  8854. +    if (NT_SUCCESS(status)) {
  8855. +#ifdef INCLUDE_TIMESTAMPS
  8856. +        LARGE_INTEGER timestamp, local_time;
  8857. +        TIME_FIELDS time_fields;
  8858. +
  8859. +        KeQuerySystemTime(&timestamp);
  8860. +        ExSystemTimeToLocalTime(&timestamp,&local_time);
  8861.          RtlTimeToTimeFields(&local_time, &time_fields);
  8862.  
  8863.          DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL,
  8864. @@ -85,39 +85,39 @@ ULONG __cdecl DbgP(IN PCCH fmt, ...)
  8865.  #endif
  8866.      }
  8867.      va_end(args);
  8868. -
  8869. -    return 0;
  8870. -}
  8871. -
  8872. -ULONG __cdecl print_error(IN PCCH fmt, ...)
  8873. -{
  8874. -    CHAR msg[512];
  8875. -    va_list args;
  8876. -    NTSTATUS status;
  8877. -
  8878. -    va_start(args, fmt);
  8879. -    ASSERT(fmt != NULL);
  8880. -    status = RtlStringCbVPrintfA(msg, sizeof(msg), fmt, args);
  8881. -    if (NT_SUCCESS(status)) {
  8882. -#ifdef INCLUDE_TIMESTAMPS
  8883. -        LARGE_INTEGER timestamp, local_time;
  8884. -        TIME_FIELDS time_fields;
  8885. -
  8886. -        KeQuerySystemTime(&timestamp);
  8887. -        ExSystemTimeToLocalTime(&timestamp,&local_time);
  8888. -        RtlTimeToTimeFields(&local_time, &time_fields);
  8889. -
  8890. -        DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL,
  8891. -            "[%ld].[%02u:%02u:%02u.%u] %s", IoGetCurrentProcess(),
  8892. -            time_fields.Hour, time_fields.Minute, time_fields.Second,
  8893. +
  8894. +    return 0;
  8895. +}
  8896. +
  8897. +ULONG __cdecl print_error(IN PCCH fmt, ...)
  8898. +{
  8899. +    CHAR msg[512];
  8900. +    va_list args;
  8901. +    NTSTATUS status;
  8902. +
  8903. +    va_start(args, fmt);
  8904. +    ASSERT(fmt != NULL);
  8905. +    status = RtlStringCbVPrintfA(msg, sizeof(msg), fmt, args);
  8906. +    if (NT_SUCCESS(status)) {
  8907. +#ifdef INCLUDE_TIMESTAMPS
  8908. +        LARGE_INTEGER timestamp, local_time;
  8909. +        TIME_FIELDS time_fields;
  8910. +
  8911. +        KeQuerySystemTime(&timestamp);
  8912. +        ExSystemTimeToLocalTime(&timestamp,&local_time);
  8913. +        RtlTimeToTimeFields(&local_time, &time_fields);
  8914. +
  8915. +        DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL,
  8916. +            "[%ld].[%02u:%02u:%02u.%u] %s", IoGetCurrentProcess(),
  8917. +            time_fields.Hour, time_fields.Minute, time_fields.Second,
  8918.              time_fields.Milliseconds, msg);
  8919.  #else
  8920.          DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL,
  8921.              "[%04x] %s", PsGetCurrentProcessShortDebugId(), msg);
  8922. -#endif
  8923. +#endif /* INCLUDE_TIMESTAMPS */
  8924.      }
  8925.      va_end(args);
  8926. -
  8927. +
  8928.      return 0;
  8929.  }
  8930.  
  8931. @@ -130,17 +130,17 @@ void print_hexbuf(const char *title, unsigned char *buf, int len)
  8932.      KeQuerySystemTime(&timestamp);
  8933.      ExSystemTimeToLocalTime(&timestamp,&local_time);
  8934.      RtlTimeToTimeFields(&local_time, &time_fields);
  8935. -
  8936. -    DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL,
  8937. -        "[%ld].[%02u:%02u:%02u.%u] %s\n", IoGetCurrentProcess(),
  8938. -        time_fields.Hour, time_fields.Minute, time_fields.Second,
  8939. -        time_fields.Milliseconds, title);
  8940. -    for(j = 0, k = 0; j < len; j++, k++) {
  8941. -        DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL,
  8942. -            "%02x ", buf[j]);
  8943. -        if (((k+1) % 30 == 0 && k > 0))
  8944. -            DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL, "\n");
  8945. -    }
  8946. +
  8947. +    DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL,
  8948. +        "[%ld].[%02u:%02u:%02u.%u] %s\n", IoGetCurrentProcess(),
  8949. +        time_fields.Hour, time_fields.Minute, time_fields.Second,
  8950. +        time_fields.Milliseconds, title);
  8951. +    for(j = 0, k = 0; j < len; j++, k++) {
  8952. +        DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL,
  8953. +            "%02x ", buf[j]);
  8954. +        if (((k+1) % 30 == 0 && k > 0))
  8955. +            DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL, "\n");
  8956. +    }
  8957.      DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL, "\n");
  8958.  }
  8959.  
  8960. @@ -149,15 +149,15 @@ void print_ioctl(int op)
  8961.      switch(op) {
  8962.          case IRP_MJ_FILE_SYSTEM_CONTROL:
  8963.              DbgP("IRP_MJ_FILE_SYSTEM_CONTROL\n");
  8964. -            break;
  8965. -        case IRP_MJ_DEVICE_CONTROL:
  8966. -            DbgP("IRP_MJ_DEVICE_CONTROL\n");
  8967. -            break;
  8968. -        case IRP_MJ_INTERNAL_DEVICE_CONTROL:
  8969. -            DbgP("IRP_MJ_INTERNAL_DEVICE_CONTROL\n");
  8970. -            break;
  8971. -        default:
  8972. -            DbgP("UNKNOWN MJ IRP %d\n", op);
  8973. +            break;
  8974. +        case IRP_MJ_DEVICE_CONTROL:
  8975. +            DbgP("IRP_MJ_DEVICE_CONTROL\n");
  8976. +            break;
  8977. +        case IRP_MJ_INTERNAL_DEVICE_CONTROL:
  8978. +            DbgP("IRP_MJ_INTERNAL_DEVICE_CONTROL\n");
  8979. +            break;
  8980. +        default:
  8981. +            DbgP("UNKNOWN MJ IRP %d\n", op);
  8982.      };
  8983.  }
  8984.  
  8985. @@ -166,54 +166,54 @@ void print_fs_ioctl(int op)
  8986.      switch(op) {
  8987.          case IOCTL_NFS41_INVALCACHE:
  8988.              DbgP("IOCTL_NFS41_INVALCACHE\n");
  8989. -            break;
  8990. -        case IOCTL_NFS41_READ:
  8991. -            DbgP("IOCTL_NFS41_UPCALL\n");
  8992. -            break;
  8993. -        case IOCTL_NFS41_WRITE:
  8994. -            DbgP("IOCTL_NFS41_DOWNCALL\n");
  8995. -            break;
  8996. -        case IOCTL_NFS41_ADDCONN:
  8997. -            DbgP("IOCTL_NFS41_ADDCONN\n");
  8998. -            break;
  8999. -        case IOCTL_NFS41_DELCONN:
  9000. -            DbgP("IOCTL_NFS41_DELCONN\n");
  9001. -            break;
  9002. -        case IOCTL_NFS41_GETSTATE:
  9003. -            DbgP("IOCTL_NFS41_GETSTATE\n");
  9004. -            break;
  9005. -        case IOCTL_NFS41_START:
  9006. -            DbgP("IOCTL_NFS41_START\n");
  9007. -            break;
  9008. -        case IOCTL_NFS41_STOP:
  9009. -            DbgP("IOCTL_NFS41_STOP\n");
  9010. -            break;
  9011. -        default:
  9012. -            DbgP("UNKNOWN FS IOCTL %d\n", op);
  9013. -    };
  9014. -}
  9015. -
  9016. -void print_driver_state(int state)
  9017. -{
  9018. -    switch (state) {
  9019. -        case NFS41_START_DRIVER_STARTABLE:
  9020. -            DbgP("NFS41_START_DRIVER_STARTABLE\n");
  9021. -            break;
  9022. -        case NFS41_START_DRIVER_STOPPED:
  9023. -            DbgP("NFS41_START_DRIVER_STOPPED\n");
  9024. -            break;
  9025. -        case NFS41_START_DRIVER_START_IN_PROGRESS:
  9026. -            DbgP("NFS41_START_DRIVER_START_IN_PROGRESS\n");
  9027. -            break;
  9028. -        case NFS41_START_DRIVER_STARTED:
  9029. -            DbgP("NFS41_START_DRIVER_STARTED\n");
  9030. -            break;
  9031. -        default:
  9032. -            DbgP("UNKNOWN DRIVER STATE %d\n", state);
  9033. -    };
  9034. -
  9035. -}
  9036. -
  9037. +            break;
  9038. +        case IOCTL_NFS41_READ:
  9039. +            DbgP("IOCTL_NFS41_UPCALL\n");
  9040. +            break;
  9041. +        case IOCTL_NFS41_WRITE:
  9042. +            DbgP("IOCTL_NFS41_DOWNCALL\n");
  9043. +            break;
  9044. +        case IOCTL_NFS41_ADDCONN:
  9045. +            DbgP("IOCTL_NFS41_ADDCONN\n");
  9046. +            break;
  9047. +        case IOCTL_NFS41_DELCONN:
  9048. +            DbgP("IOCTL_NFS41_DELCONN\n");
  9049. +            break;
  9050. +        case IOCTL_NFS41_GETSTATE:
  9051. +            DbgP("IOCTL_NFS41_GETSTATE\n");
  9052. +            break;
  9053. +        case IOCTL_NFS41_START:
  9054. +            DbgP("IOCTL_NFS41_START\n");
  9055. +            break;
  9056. +        case IOCTL_NFS41_STOP:
  9057. +            DbgP("IOCTL_NFS41_STOP\n");
  9058. +            break;
  9059. +        default:
  9060. +            DbgP("UNKNOWN FS IOCTL %d\n", op);
  9061. +    };
  9062. +}
  9063. +
  9064. +void print_driver_state(int state)
  9065. +{
  9066. +    switch (state) {
  9067. +        case NFS41_START_DRIVER_STARTABLE:
  9068. +            DbgP("NFS41_START_DRIVER_STARTABLE\n");
  9069. +            break;
  9070. +        case NFS41_START_DRIVER_STOPPED:
  9071. +            DbgP("NFS41_START_DRIVER_STOPPED\n");
  9072. +            break;
  9073. +        case NFS41_START_DRIVER_START_IN_PROGRESS:
  9074. +            DbgP("NFS41_START_DRIVER_START_IN_PROGRESS\n");
  9075. +            break;
  9076. +        case NFS41_START_DRIVER_STARTED:
  9077. +            DbgP("NFS41_START_DRIVER_STARTED\n");
  9078. +            break;
  9079. +        default:
  9080. +            DbgP("UNKNOWN DRIVER STATE %d\n", state);
  9081. +    };
  9082. +
  9083. +}
  9084. +
  9085.  void print_basic_info(int on, PFILE_BASIC_INFORMATION info)
  9086.  {
  9087.      if (!on) return;
  9088. @@ -222,7 +222,7 @@ void print_basic_info(int on, PFILE_BASIC_INFORMATION info)
  9089.          info->CreationTime.QuadPart, info->LastAccessTime.QuadPart,
  9090.          info->LastWriteTime.QuadPart, info->ChangeTime.QuadPart,
  9091.          info->FileAttributes);
  9092. -}
  9093. +}
  9094.  void print_std_info(int on, PFILE_STANDARD_INFORMATION info)
  9095.  {
  9096.      if (!on) return;
  9097. @@ -312,8 +312,8 @@ VOID print_v_net_root(IN PMRX_V_NET_ROOT p)
  9098.      //DbgP("IsExplicitConnection %d\n", p->IsExplicitConnection);
  9099.      DbgP("*****************\n");
  9100.  #endif
  9101. -}
  9102. -
  9103. +}
  9104. +
  9105.  void print_file_object(int on, PFILE_OBJECT file)
  9106.  {
  9107.      if (!on) return;
  9108. @@ -334,7 +334,7 @@ void print_fo_all(int on, PRX_CONTEXT c)
  9109.              c->pFcb->OpenCount, c->pFcb, c->pRelevantSrvOpen, c->pFobx,
  9110.              c->pRelevantSrvOpen->pVNetRoot, c->pFcb->pNetRoot);
  9111.  }
  9112. -
  9113. +
  9114.  VOID print_fcb(int on, IN PMRX_FCB p)
  9115.  {
  9116.      if (!on) return;
  9117. @@ -353,15 +353,15 @@ VOID print_fcb(int on, IN PMRX_FCB p)
  9118.      //DbgP("ActualAllocationLength %ull\n", p->ActualAllocationLength);
  9119.      //DbgP("Attributes %ld\n", p->Attributes);
  9120.      //DbgP("IsFileWritten %d\n", p->IsFileWritten);
  9121. -    //DbgP("fShouldBeOrphaned %d\n", p->fShouldBeOrphaned);
  9122. -    //DbgP("fMiniInited %ld\n", p->fMiniInited);
  9123. -    //DbgP("CachedNetRootType %c\n", p->CachedNetRootType);
  9124. -    //DbgP("SrvOpenList\n");
  9125. -    //DbgP("SrvOpenListVersion %ld\n", p->SrvOpenListVersion);
  9126. -    DbgP("*****************\n");
  9127. -#endif
  9128. -}
  9129. -
  9130. +    //DbgP("fShouldBeOrphaned %d\n", p->fShouldBeOrphaned);
  9131. +    //DbgP("fMiniInited %ld\n", p->fMiniInited);
  9132. +    //DbgP("CachedNetRootType %c\n", p->CachedNetRootType);
  9133. +    //DbgP("SrvOpenList\n");
  9134. +    //DbgP("SrvOpenListVersion %ld\n", p->SrvOpenListVersion);
  9135. +    DbgP("*****************\n");
  9136. +#endif
  9137. +}
  9138. +
  9139.  VOID print_srv_open(int on, IN PMRX_SRV_OPEN p)
  9140.  {
  9141.      if (!on) return;
  9142. @@ -380,13 +380,13 @@ VOID print_srv_open(int on, IN PMRX_SRV_OPEN p)
  9143.      //DbgP("DesiredAccess\n");
  9144.      //DbgP("ShareAccess %ld\n", p->ShareAccess);
  9145.      //DbgP("CreateOptions %ld\n", p->CreateOptions);
  9146. -    //DbgP("BufferingFlags %ld\n", p->BufferingFlags);
  9147. -    //DbgP("ulFileSizeVersion %ld\n", p->ulFileSizeVersion);
  9148. -    //DbgP("SrvOpenQLinks\n");
  9149. -    DbgP("*****************\n");
  9150. -#endif
  9151. -}
  9152. -
  9153. +    //DbgP("BufferingFlags %ld\n", p->BufferingFlags);
  9154. +    //DbgP("ulFileSizeVersion %ld\n", p->ulFileSizeVersion);
  9155. +    //DbgP("SrvOpenQLinks\n");
  9156. +    DbgP("*****************\n");
  9157. +#endif
  9158. +}
  9159. +
  9160.  VOID print_fobx(int on, IN PMRX_FOBX p)
  9161.  {
  9162.      if (!on) return;
  9163. @@ -401,8 +401,8 @@ VOID print_fobx(int on, IN PMRX_FOBX p)
  9164.      DbgP("*****************\n");
  9165.  #endif
  9166.  }
  9167. -
  9168. -VOID print_irp_flags(int on, PIRP irp)
  9169. +
  9170. +VOID print_irp_flags(int on, PIRP irp)
  9171.  {
  9172.      if (!on) return;
  9173.      if (irp->Flags)
  9174. @@ -414,21 +414,21 @@ VOID print_irp_flags(int on, PIRP irp)
  9175.              irp->Flags,
  9176.              (irp->Flags & IRP_NOCACHE)?"NOCACHE":"",
  9177.              (irp->Flags & IRP_PAGING_IO)?"PAGING_IO":"",
  9178. -            (irp->Flags & IRP_MOUNT_COMPLETION)?"MOUNT":"",
  9179. -            (irp->Flags & IRP_SYNCHRONOUS_API)?"SYNC":"",
  9180. -            (irp->Flags & IRP_ASSOCIATED_IRP)?"ASSOC_IPR":"",
  9181. -            (irp->Flags & IRP_BUFFERED_IO)?"BUFFERED":"",
  9182. -            (irp->Flags & IRP_DEALLOCATE_BUFFER)?"DEALLOC_BUF":"",
  9183. -            (irp->Flags & IRP_INPUT_OPERATION)?"INPUT_OP":"",
  9184. -            (irp->Flags & IRP_SYNCHRONOUS_PAGING_IO)?"SYNC_PAGIN_IO":"",
  9185. -            (irp->Flags & IRP_CREATE_OPERATION)?"CREATE_OP":"",
  9186. -            (irp->Flags & IRP_READ_OPERATION)?"READ_OP":"",
  9187. -            (irp->Flags & IRP_WRITE_OPERATION)?"WRITE_OP":"",
  9188. -            (irp->Flags & IRP_CLOSE_OPERATION)?"CLOSE_OP":"",
  9189. -            (irp->Flags & IRP_DEFER_IO_COMPLETION)?"DEFER_IO":"");
  9190. -}
  9191. -
  9192. -void print_irps_flags(int on, PIO_STACK_LOCATION irps)
  9193. +            (irp->Flags & IRP_MOUNT_COMPLETION)?"MOUNT":"",
  9194. +            (irp->Flags & IRP_SYNCHRONOUS_API)?"SYNC":"",
  9195. +            (irp->Flags & IRP_ASSOCIATED_IRP)?"ASSOC_IPR":"",
  9196. +            (irp->Flags & IRP_BUFFERED_IO)?"BUFFERED":"",
  9197. +            (irp->Flags & IRP_DEALLOCATE_BUFFER)?"DEALLOC_BUF":"",
  9198. +            (irp->Flags & IRP_INPUT_OPERATION)?"INPUT_OP":"",
  9199. +            (irp->Flags & IRP_SYNCHRONOUS_PAGING_IO)?"SYNC_PAGIN_IO":"",
  9200. +            (irp->Flags & IRP_CREATE_OPERATION)?"CREATE_OP":"",
  9201. +            (irp->Flags & IRP_READ_OPERATION)?"READ_OP":"",
  9202. +            (irp->Flags & IRP_WRITE_OPERATION)?"WRITE_OP":"",
  9203. +            (irp->Flags & IRP_CLOSE_OPERATION)?"CLOSE_OP":"",
  9204. +            (irp->Flags & IRP_DEFER_IO_COMPLETION)?"DEFER_IO":"");
  9205. +}
  9206. +
  9207. +void print_irps_flags(int on, PIO_STACK_LOCATION irps)
  9208.  {
  9209.      if (!on) return;
  9210.      if (irps->Flags)
  9211. @@ -436,9 +436,9 @@ void print_irps_flags(int on, PIO_STACK_LOCATION irps)
  9212.              (irps->Flags & SL_CASE_SENSITIVE)?"CASE_SENSITIVE":"",
  9213.              (irps->Flags & SL_OPEN_PAGING_FILE)?"PAGING_FILE":"",
  9214.              (irps->Flags & SL_FORCE_ACCESS_CHECK)?"ACCESS_CHECK":"",
  9215. -            (irps->Flags & SL_OPEN_TARGET_DIRECTORY)?"TARGET_DIR":"");
  9216. -}
  9217. -void print_nt_create_params(int on, NT_CREATE_PARAMETERS params)
  9218. +            (irps->Flags & SL_OPEN_TARGET_DIRECTORY)?"TARGET_DIR":"");
  9219. +}
  9220. +void print_nt_create_params(int on, NT_CREATE_PARAMETERS params)
  9221.  {
  9222.      if (!on) return;
  9223.      if (params.FileAttributes)
  9224. @@ -448,29 +448,29 @@ void print_nt_create_params(int on, NT_CREATE_PARAMETERS params)
  9225.              params.FileAttributes,
  9226.              (params.FileAttributes & FILE_ATTRIBUTE_TEMPORARY)?"TEMPFILE ":"",
  9227.              (params.FileAttributes & FILE_ATTRIBUTE_READONLY)?"READONLY ":"",
  9228. -            (params.FileAttributes & FILE_ATTRIBUTE_HIDDEN)?"HIDDEN ":"",
  9229. -            (params.FileAttributes & FILE_ATTRIBUTE_SYSTEM)?"SYSTEM ":"",
  9230. -            (params.FileAttributes & FILE_ATTRIBUTE_ARCHIVE)?"ARCHIVE ":"",
  9231. -            (params.FileAttributes & FILE_ATTRIBUTE_DIRECTORY)?"DIR ":"",
  9232. -            (params.FileAttributes & FILE_ATTRIBUTE_DEVICE)?"DEVICE ":"",
  9233. -            (params.FileAttributes & FILE_ATTRIBUTE_NORMAL)?"NORMAL ":"",
  9234. -            (params.FileAttributes & FILE_ATTRIBUTE_SPARSE_FILE)?"SPARSE_FILE ":"",
  9235. -            (params.FileAttributes & FILE_ATTRIBUTE_REPARSE_POINT)?"REPARSE_POINT ":"",
  9236. -            (params.FileAttributes & FILE_ATTRIBUTE_COMPRESSED)?"COMPRESSED ":"",
  9237. -            (params.FileAttributes & FILE_ATTRIBUTE_NOT_CONTENT_INDEXED)?"NOT INDEXED ":"",
  9238. -            (params.FileAttributes & FILE_ATTRIBUTE_ENCRYPTED)?"ENCRYPTED ":"",
  9239. -            (params.FileAttributes & FILE_ATTRIBUTE_VIRTUAL)?"VIRTUAL":"");
  9240. -
  9241. -    if (params.Disposition  == FILE_SUPERSEDE)
  9242. -        DbgP("Create Dispositions: FILE_SUPERSEDE\n");
  9243. -    if (params.Disposition == FILE_CREATE)
  9244. -        DbgP("Create Dispositions: FILE_CREATE\n");
  9245. -    if (params.Disposition == FILE_OPEN)
  9246. -        DbgP("Create Dispositions: FILE_OPEN\n");
  9247. -    if (params.Disposition == FILE_OPEN_IF)
  9248. -        DbgP("Create Dispositions: FILE_OPEN_IF\n");
  9249. -    if (params.Disposition == FILE_OVERWRITE)
  9250. -        DbgP("Create Dispositions: FILE_OVERWRITE\n");
  9251. +            (params.FileAttributes & FILE_ATTRIBUTE_HIDDEN)?"HIDDEN ":"",
  9252. +            (params.FileAttributes & FILE_ATTRIBUTE_SYSTEM)?"SYSTEM ":"",
  9253. +            (params.FileAttributes & FILE_ATTRIBUTE_ARCHIVE)?"ARCHIVE ":"",
  9254. +            (params.FileAttributes & FILE_ATTRIBUTE_DIRECTORY)?"DIR ":"",
  9255. +            (params.FileAttributes & FILE_ATTRIBUTE_DEVICE)?"DEVICE ":"",
  9256. +            (params.FileAttributes & FILE_ATTRIBUTE_NORMAL)?"NORMAL ":"",
  9257. +            (params.FileAttributes & FILE_ATTRIBUTE_SPARSE_FILE)?"SPARSE_FILE ":"",
  9258. +            (params.FileAttributes & FILE_ATTRIBUTE_REPARSE_POINT)?"REPARSE_POINT ":"",
  9259. +            (params.FileAttributes & FILE_ATTRIBUTE_COMPRESSED)?"COMPRESSED ":"",
  9260. +            (params.FileAttributes & FILE_ATTRIBUTE_NOT_CONTENT_INDEXED)?"NOT INDEXED ":"",
  9261. +            (params.FileAttributes & FILE_ATTRIBUTE_ENCRYPTED)?"ENCRYPTED ":"",
  9262. +            (params.FileAttributes & FILE_ATTRIBUTE_VIRTUAL)?"VIRTUAL":"");
  9263. +
  9264. +    if (params.Disposition  == FILE_SUPERSEDE)
  9265. +        DbgP("Create Dispositions: FILE_SUPERSEDE\n");
  9266. +    if (params.Disposition == FILE_CREATE)
  9267. +        DbgP("Create Dispositions: FILE_CREATE\n");
  9268. +    if (params.Disposition == FILE_OPEN)
  9269. +        DbgP("Create Dispositions: FILE_OPEN\n");
  9270. +    if (params.Disposition == FILE_OPEN_IF)
  9271. +        DbgP("Create Dispositions: FILE_OPEN_IF\n");
  9272. +    if (params.Disposition == FILE_OVERWRITE)
  9273. +        DbgP("Create Dispositions: FILE_OVERWRITE\n");
  9274.      if (params.Disposition == FILE_OVERWRITE_IF)
  9275.          DbgP("Create Dispositions: FILE_OVERWRITE_IF\n");
  9276.  
  9277. @@ -482,15 +482,15 @@ void print_nt_create_params(int on, NT_CREATE_PARAMETERS params)
  9278.          (params.CreateOptions & FILE_DIRECTORY_FILE)?"DIRFILE":"",
  9279.          (params.CreateOptions & FILE_NON_DIRECTORY_FILE)?"FILE":"",
  9280.          (params.CreateOptions & FILE_DELETE_ON_CLOSE)?"DELETE_ON_CLOSE":"",
  9281. -        (params.CreateOptions & FILE_WRITE_THROUGH)?"WRITE_THROUGH":"",
  9282. -        (params.CreateOptions & FILE_SEQUENTIAL_ONLY)?"SEQUENTIAL":"",
  9283. -        (params.CreateOptions & FILE_RANDOM_ACCESS)?"RANDOM":"",
  9284. -        (params.CreateOptions & FILE_NO_INTERMEDIATE_BUFFERING)?"NO_BUFFERING":"",
  9285. -        (params.CreateOptions & FILE_SYNCHRONOUS_IO_ALERT)?"SYNC_ALERT":"",
  9286. -        (params.CreateOptions & FILE_SYNCHRONOUS_IO_NONALERT)?"SYNC_NOALERT":"",
  9287. -        (params.CreateOptions & FILE_CREATE_TREE_CONNECTION)?"CREATE_TREE_CONN":"",
  9288. -        (params.CreateOptions & FILE_COMPLETE_IF_OPLOCKED)?"OPLOCKED":"",
  9289. -        (params.CreateOptions & FILE_NO_EA_KNOWLEDGE)?"NO_EA":"",
  9290. +        (params.CreateOptions & FILE_WRITE_THROUGH)?"WRITE_THROUGH":"",
  9291. +        (params.CreateOptions & FILE_SEQUENTIAL_ONLY)?"SEQUENTIAL":"",
  9292. +        (params.CreateOptions & FILE_RANDOM_ACCESS)?"RANDOM":"",
  9293. +        (params.CreateOptions & FILE_NO_INTERMEDIATE_BUFFERING)?"NO_BUFFERING":"",
  9294. +        (params.CreateOptions & FILE_SYNCHRONOUS_IO_ALERT)?"SYNC_ALERT":"",
  9295. +        (params.CreateOptions & FILE_SYNCHRONOUS_IO_NONALERT)?"SYNC_NOALERT":"",
  9296. +        (params.CreateOptions & FILE_CREATE_TREE_CONNECTION)?"CREATE_TREE_CONN":"",
  9297. +        (params.CreateOptions & FILE_COMPLETE_IF_OPLOCKED)?"OPLOCKED":"",
  9298. +        (params.CreateOptions & FILE_NO_EA_KNOWLEDGE)?"NO_EA":"",
  9299.          (params.CreateOptions & FILE_OPEN_REPARSE_POINT)?"OPEN_REPARSE":"",
  9300.          (params.CreateOptions & FILE_OPEN_BY_FILE_ID)?"BY_ID":"",
  9301.          (params.CreateOptions & FILE_OPEN_FOR_BACKUP_INTENT)?"4_BACKUP":"",
  9302. @@ -508,108 +508,108 @@ void print_nt_create_params(int on, NT_CREATE_PARAMETERS params)
  9303.          params.DesiredAccess,
  9304.          (params.DesiredAccess & FILE_READ_DATA)?"READ":"",
  9305.          (params.DesiredAccess & STANDARD_RIGHTS_READ)?"READ_ACL":"",
  9306. -        (params.DesiredAccess & FILE_READ_ATTRIBUTES)?"GETATTR":"",
  9307. -        (params.DesiredAccess & FILE_READ_EA)?"READ_EA":"",
  9308. -        (params.DesiredAccess & FILE_WRITE_DATA)?"WRITE":"",
  9309. -        (params.DesiredAccess & FILE_WRITE_ATTRIBUTES)?"SETATTR":"",
  9310. -        (params.DesiredAccess & FILE_WRITE_EA)?"WRITE_EA":"",
  9311. -        (params.DesiredAccess & FILE_APPEND_DATA)?"APPEND":"",
  9312. -        (params.DesiredAccess & FILE_EXECUTE)?"EXEC":"",
  9313. -        (params.DesiredAccess & FILE_LIST_DIRECTORY)?"LSDIR":"",
  9314. -        (params.DesiredAccess & FILE_TRAVERSE)?"TRAVERSE":"",
  9315. -        (params.DesiredAccess & FILE_LIST_DIRECTORY)?"LSDIR":"",
  9316. -        (params.DesiredAccess & DELETE)?"DELETE":"",
  9317. -        (params.DesiredAccess & READ_CONTROL)?"READ_CONTROL":"",
  9318. -        (params.DesiredAccess & WRITE_DAC)?"WRITE_DAC":"",
  9319. -        (params.DesiredAccess & WRITE_OWNER)?"WRITE_OWNER":"",
  9320. -        (params.DesiredAccess & SYNCHRONIZE)?"SYNCHRONIZE":"");
  9321. -}
  9322. -
  9323. -unsigned char * print_file_information_class(int InfoClass)
  9324. -{
  9325. -    switch(InfoClass) {
  9326. -        case FileBothDirectoryInformation:
  9327. -            return (unsigned char *)"FileBothDirectoryInformation";
  9328. -        case FileDirectoryInformation:
  9329. -            return (unsigned char *)"FileDirectoryInformation";
  9330. -        case FileFullDirectoryInformation:
  9331. -            return (unsigned char *)"FileFullDirectoryInformation";
  9332. -        case FileIdBothDirectoryInformation:
  9333. -            return (unsigned char *)"FileIdBothDirectoryInformation";
  9334. -        case FileIdFullDirectoryInformation:
  9335. -            return (unsigned char *)"FileIdFullDirectoryInformation";
  9336. -        case FileNamesInformation:
  9337. -            return (unsigned char *)"FileNamesInformation";
  9338. -        case FileObjectIdInformation:
  9339. -            return (unsigned char *)"FileObjectIdInformation";
  9340. -        case FileQuotaInformation:
  9341. -            return (unsigned char *)"FileQuotaInformation";
  9342. -        case FileReparsePointInformation:
  9343. -            return (unsigned char *)"FileReparsePointInformation";
  9344. -        case FileAllInformation:
  9345. -            return (unsigned char *)"FileAllInformation";
  9346. -        case FileAttributeTagInformation:
  9347. -            return (unsigned char *)"FileAttributeTagInformation";
  9348. -        case FileBasicInformation:
  9349. -            return (unsigned char *)"FileBasicInformation";
  9350. -        case FileCompressionInformation:
  9351. -            return (unsigned char *)"FileCompressionInformation";
  9352. -        case FileEaInformation:
  9353. -            return (unsigned char *)"FileEaInformation";
  9354. -        case FileInternalInformation:
  9355. -            return (unsigned char *)"FileInternalInformation";
  9356. -        case FileNameInformation:
  9357. -            return (unsigned char *)"FileNameInformation";
  9358. -        case FileNetworkOpenInformation:
  9359. -            return (unsigned char *)"FileNetworkOpenInformation";
  9360. -        case FilePositionInformation:
  9361. -            return (unsigned char *)"FilePositionInformation";
  9362. -        case FileStandardInformation:
  9363. -            return (unsigned char *)"FileStandardInformation";
  9364. -        case FileStreamInformation:
  9365. -            return (unsigned char *)"FileStreamInformation";
  9366. -        case FileAllocationInformation:
  9367. -            return (unsigned char *)"FileAllocationInformation";
  9368. -        case FileDispositionInformation:
  9369. -            return (unsigned char *)"FileDispositionInformation";
  9370. -        case FileEndOfFileInformation:
  9371. -            return (unsigned char *)"FileEndOfFileInformation";
  9372. -        case FileLinkInformation:
  9373. -            return (unsigned char *)"FileLinkInformation";
  9374. -        case FileRenameInformation:
  9375. -            return (unsigned char *)"FileRenameInformation";
  9376. -        case FileValidDataLengthInformation:
  9377. -            return (unsigned char *)"FileValidDataLengthInformation";
  9378. -        default:
  9379. -            return (unsigned char *)"UNKNOWN";
  9380. -    }
  9381. -}
  9382. -
  9383. -unsigned char *print_fs_information_class(int InfoClass)
  9384. -{
  9385. -    switch (InfoClass) {
  9386. -        case FileFsAttributeInformation:
  9387. -            return (unsigned char *)"FileFsAttributeInformation";
  9388. -        case FileFsControlInformation:
  9389. -            return (unsigned char *)"FileFsControlInformation";
  9390. -        case FileFsDeviceInformation:
  9391. -            return (unsigned char *)"FileFsDeviceInformation";
  9392. -        case FileFsDriverPathInformation:
  9393. -            return (unsigned char *)"FileFsDriverPathInformation";
  9394. -        case FileFsFullSizeInformation:
  9395. -            return (unsigned char *)"FileFsFullSizeInformation";
  9396. -        case FileFsObjectIdInformation:
  9397. -            return (unsigned char *)"FileFsObjectIdInformation";
  9398. -        case FileFsSizeInformation:
  9399. -            return (unsigned char *)"FileFsSizeInformation";
  9400. -        case FileFsVolumeInformation:
  9401. -            return (unsigned char *)"FileFsVolumeInformation";
  9402. -        default:
  9403. -            return (unsigned char *)"UNKNOWN";
  9404. -    }
  9405. -}
  9406. -
  9407. -void print_caching_level(int on, ULONG flag, PUNICODE_STRING name)
  9408. +        (params.DesiredAccess & FILE_READ_ATTRIBUTES)?"GETATTR":"",
  9409. +        (params.DesiredAccess & FILE_READ_EA)?"READ_EA":"",
  9410. +        (params.DesiredAccess & FILE_WRITE_DATA)?"WRITE":"",
  9411. +        (params.DesiredAccess & FILE_WRITE_ATTRIBUTES)?"SETATTR":"",
  9412. +        (params.DesiredAccess & FILE_WRITE_EA)?"WRITE_EA":"",
  9413. +        (params.DesiredAccess & FILE_APPEND_DATA)?"APPEND":"",
  9414. +        (params.DesiredAccess & FILE_EXECUTE)?"EXEC":"",
  9415. +        (params.DesiredAccess & FILE_LIST_DIRECTORY)?"LSDIR":"",
  9416. +        (params.DesiredAccess & FILE_TRAVERSE)?"TRAVERSE":"",
  9417. +        (params.DesiredAccess & FILE_LIST_DIRECTORY)?"LSDIR":"",
  9418. +        (params.DesiredAccess & DELETE)?"DELETE":"",
  9419. +        (params.DesiredAccess & READ_CONTROL)?"READ_CONTROL":"",
  9420. +        (params.DesiredAccess & WRITE_DAC)?"WRITE_DAC":"",
  9421. +        (params.DesiredAccess & WRITE_OWNER)?"WRITE_OWNER":"",
  9422. +        (params.DesiredAccess & SYNCHRONIZE)?"SYNCHRONIZE":"");
  9423. +}
  9424. +
  9425. +const char *print_file_information_class(int InfoClass)
  9426. +{
  9427. +    switch(InfoClass) {
  9428. +        case FileBothDirectoryInformation:
  9429. +            return "FileBothDirectoryInformation";
  9430. +        case FileDirectoryInformation:
  9431. +            return "FileDirectoryInformation";
  9432. +        case FileFullDirectoryInformation:
  9433. +            return "FileFullDirectoryInformation";
  9434. +        case FileIdBothDirectoryInformation:
  9435. +            return "FileIdBothDirectoryInformation";
  9436. +        case FileIdFullDirectoryInformation:
  9437. +            return "FileIdFullDirectoryInformation";
  9438. +        case FileNamesInformation:
  9439. +            return "FileNamesInformation";
  9440. +        case FileObjectIdInformation:
  9441. +            return "FileObjectIdInformation";
  9442. +        case FileQuotaInformation:
  9443. +            return "FileQuotaInformation";
  9444. +        case FileReparsePointInformation:
  9445. +            return "FileReparsePointInformation";
  9446. +        case FileAllInformation:
  9447. +            return "FileAllInformation";
  9448. +        case FileAttributeTagInformation:
  9449. +            return "FileAttributeTagInformation";
  9450. +        case FileBasicInformation:
  9451. +            return "FileBasicInformation";
  9452. +        case FileCompressionInformation:
  9453. +            return "FileCompressionInformation";
  9454. +        case FileEaInformation:
  9455. +            return "FileEaInformation";
  9456. +        case FileInternalInformation:
  9457. +            return "FileInternalInformation";
  9458. +        case FileNameInformation:
  9459. +            return "FileNameInformation";
  9460. +        case FileNetworkOpenInformation:
  9461. +            return "FileNetworkOpenInformation";
  9462. +        case FilePositionInformation:
  9463. +            return "FilePositionInformation";
  9464. +        case FileStandardInformation:
  9465. +            return "FileStandardInformation";
  9466. +        case FileStreamInformation:
  9467. +            return "FileStreamInformation";
  9468. +        case FileAllocationInformation:
  9469. +            return "FileAllocationInformation";
  9470. +        case FileDispositionInformation:
  9471. +            return "FileDispositionInformation";
  9472. +        case FileEndOfFileInformation:
  9473. +            return "FileEndOfFileInformation";
  9474. +        case FileLinkInformation:
  9475. +            return "FileLinkInformation";
  9476. +        case FileRenameInformation:
  9477. +            return "FileRenameInformation";
  9478. +        case FileValidDataLengthInformation:
  9479. +            return "FileValidDataLengthInformation";
  9480. +        default:
  9481. +            return "UNKNOWN_InfoClass";
  9482. +    }
  9483. +}
  9484. +
  9485. +const char *print_fs_information_class(int InfoClass)
  9486. +{
  9487. +    switch (InfoClass) {
  9488. +        case FileFsAttributeInformation:
  9489. +            return "FileFsAttributeInformation";
  9490. +        case FileFsControlInformation:
  9491. +            return "FileFsControlInformation";
  9492. +        case FileFsDeviceInformation:
  9493. +            return "FileFsDeviceInformation";
  9494. +        case FileFsDriverPathInformation:
  9495. +            return "FileFsDriverPathInformation";
  9496. +        case FileFsFullSizeInformation:
  9497. +            return "FileFsFullSizeInformation";
  9498. +        case FileFsObjectIdInformation:
  9499. +            return "FileFsObjectIdInformation";
  9500. +        case FileFsSizeInformation:
  9501. +            return "FileFsSizeInformation";
  9502. +        case FileFsVolumeInformation:
  9503. +            return "FileFsVolumeInformation";
  9504. +        default:
  9505. +            return "UNKNOWN_FsInfoClass";
  9506. +    }
  9507. +}
  9508. +
  9509. +void print_caching_level(int on, ULONG flag, PUNICODE_STRING name)
  9510.  {
  9511.      if (!on) return;
  9512.      switch(flag) {
  9513. @@ -628,31 +628,31 @@ void print_caching_level(int on, ULONG flag, PUNICODE_STRING name)
  9514.      }
  9515.  }
  9516.  
  9517. -const char *opcode2string(int opcode)
  9518. -{
  9519. -    switch(opcode) {
  9520. -    case NFS41_SHUTDOWN: return "NFS41_SHUTDOWN";
  9521. -    case NFS41_MOUNT: return "NFS41_MOUNT";
  9522. -    case NFS41_UNMOUNT: return "NFS41_UNMOUNT";
  9523. -    case NFS41_OPEN: return "NFS41_OPEN";
  9524. -    case NFS41_CLOSE: return "NFS41_CLOSE";
  9525. -    case NFS41_READ: return "NFS41_READ";
  9526. -    case NFS41_WRITE: return "NFS41_WRITE";
  9527. -    case NFS41_LOCK: return "NFS41_LOCK";
  9528. -    case NFS41_UNLOCK: return "NFS41_UNLOCK";
  9529. -    case NFS41_DIR_QUERY: return "NFS41_DIR_QUERY";
  9530. -    case NFS41_FILE_QUERY: return "NFS41_FILE_QUERY";
  9531. -    case NFS41_FILE_SET: return "NFS41_FILE_SET";
  9532. -    case NFS41_EA_SET: return "NFS41_EA_SET";
  9533. -    case NFS41_EA_GET: return "NFS41_EA_GET";
  9534. -    case NFS41_SYMLINK: return "NFS41_SYMLINK";
  9535. -    case NFS41_VOLUME_QUERY: return "NFS41_VOLUME_QUERY";
  9536. -    case NFS41_ACL_QUERY: return "NFS41_ACL_QUERY";
  9537. -    case NFS41_ACL_SET: return "NFS41_ACL_SET";
  9538. -    default: return "UNKNOWN";
  9539. -    }
  9540. -}
  9541. -
  9542. +const char *opcode2string(int opcode)
  9543. +{
  9544. +    switch(opcode) {
  9545. +    case NFS41_SHUTDOWN: return "NFS41_SHUTDOWN";
  9546. +    case NFS41_MOUNT: return "NFS41_MOUNT";
  9547. +    case NFS41_UNMOUNT: return "NFS41_UNMOUNT";
  9548. +    case NFS41_OPEN: return "NFS41_OPEN";
  9549. +    case NFS41_CLOSE: return "NFS41_CLOSE";
  9550. +    case NFS41_READ: return "NFS41_READ";
  9551. +    case NFS41_WRITE: return "NFS41_WRITE";
  9552. +    case NFS41_LOCK: return "NFS41_LOCK";
  9553. +    case NFS41_UNLOCK: return "NFS41_UNLOCK";
  9554. +    case NFS41_DIR_QUERY: return "NFS41_DIR_QUERY";
  9555. +    case NFS41_FILE_QUERY: return "NFS41_FILE_QUERY";
  9556. +    case NFS41_FILE_SET: return "NFS41_FILE_SET";
  9557. +    case NFS41_EA_SET: return "NFS41_EA_SET";
  9558. +    case NFS41_EA_GET: return "NFS41_EA_GET";
  9559. +    case NFS41_SYMLINK: return "NFS41_SYMLINK";
  9560. +    case NFS41_VOLUME_QUERY: return "NFS41_VOLUME_QUERY";
  9561. +    case NFS41_ACL_QUERY: return "NFS41_ACL_QUERY";
  9562. +    case NFS41_ACL_SET: return "NFS41_ACL_SET";
  9563. +    default: return "UNKNOWN";
  9564. +    }
  9565. +}
  9566. +
  9567.  void print_acl_args(
  9568.      SECURITY_INFORMATION info)
  9569.  {
  9570. @@ -660,58 +660,58 @@ void print_acl_args(
  9571.          (info & OWNER_SECURITY_INFORMATION)?"OWNER":"",
  9572.          (info & GROUP_SECURITY_INFORMATION)?"GROUP":"",
  9573.          (info & DACL_SECURITY_INFORMATION)?"DACL":"",
  9574. -        (info & SACL_SECURITY_INFORMATION)?"SACL":"");
  9575. -}
  9576. -
  9577. -void print_open_error(int on, int status)
  9578. -{
  9579. -    if (!on) return;
  9580. -    switch (status) {
  9581. -    case STATUS_ACCESS_DENIED:
  9582. -        DbgP("[ERROR] nfs41_Create: STATUS_ACCESS_DENIED\n");
  9583. -        break;
  9584. -    case STATUS_NETWORK_ACCESS_DENIED:
  9585. -        DbgP("[ERROR] nfs41_Create: STATUS_NETWORK_ACCESS_DENIED\n");
  9586. -        break;
  9587. -    case STATUS_OBJECT_NAME_INVALID:
  9588. -        DbgP("[ERROR] nfs41_Create: STATUS_OBJECT_NAME_INVALID\n");
  9589. -        break;
  9590. -    case STATUS_OBJECT_NAME_COLLISION:
  9591. -        DbgP("[ERROR] nfs41_Create: STATUS_OBJECT_NAME_COLLISION\n");
  9592. -        break;
  9593. -    case STATUS_FILE_INVALID:
  9594. -        DbgP("[ERROR] nfs41_Create: STATUS_FILE_INVALID\n");
  9595. -        break;
  9596. -    case STATUS_OBJECT_NAME_NOT_FOUND:
  9597. -        DbgP("[ERROR] nfs41_Create: STATUS_OBJECT_NAME_NOT_FOUND\n");
  9598. -        break;
  9599. -    case STATUS_NAME_TOO_LONG:
  9600. -        DbgP("[ERROR] nfs41_Create: STATUS_NAME_TOO_LONG\n");
  9601. -        break;
  9602. -    case STATUS_OBJECT_PATH_NOT_FOUND:
  9603. -        DbgP("[ERROR] nfs41_Create: STATUS_OBJECT_PATH_NOT_FOUND\n");
  9604. -        break;
  9605. -    case STATUS_BAD_NETWORK_PATH:
  9606. -        DbgP("[ERROR] nfs41_Create: STATUS_BAD_NETWORK_PATH\n");
  9607. -        break;
  9608. -    case STATUS_SHARING_VIOLATION:
  9609. -        DbgP("[ERROR] nfs41_Create: STATUS_SHARING_VIOLATION\n");
  9610. -        break;
  9611. -    case ERROR_REPARSE:
  9612. -        DbgP("[ERROR] nfs41_Create: STATUS_REPARSE\n");
  9613. -        break;
  9614. -    case ERROR_TOO_MANY_LINKS:
  9615. -        DbgP("[ERROR] nfs41_Create: STATUS_TOO_MANY_LINKS\n");
  9616. -        break;
  9617. -    case ERROR_DIRECTORY:
  9618. -        DbgP("[ERROR] nfs41_Create: STATUS_FILE_IS_A_DIRECTORY\n");
  9619. -        break;
  9620. -    case ERROR_BAD_FILE_TYPE:
  9621. -        DbgP("[ERROR] nfs41_Create: STATUS_NOT_A_DIRECTORY\n");
  9622. -        break;
  9623. -    default:
  9624. -        DbgP("[ERROR] nfs41_Create: STATUS_INSUFFICIENT_RESOURCES\n");
  9625. -        break;
  9626. +        (info & SACL_SECURITY_INFORMATION)?"SACL":"");
  9627. +}
  9628. +
  9629. +void print_open_error(int on, int status)
  9630. +{
  9631. +    if (!on) return;
  9632. +    switch (status) {
  9633. +    case STATUS_ACCESS_DENIED:
  9634. +        DbgP("[ERROR] nfs41_Create: STATUS_ACCESS_DENIED\n");
  9635. +        break;
  9636. +    case STATUS_NETWORK_ACCESS_DENIED:
  9637. +        DbgP("[ERROR] nfs41_Create: STATUS_NETWORK_ACCESS_DENIED\n");
  9638. +        break;
  9639. +    case STATUS_OBJECT_NAME_INVALID:
  9640. +        DbgP("[ERROR] nfs41_Create: STATUS_OBJECT_NAME_INVALID\n");
  9641. +        break;
  9642. +    case STATUS_OBJECT_NAME_COLLISION:
  9643. +        DbgP("[ERROR] nfs41_Create: STATUS_OBJECT_NAME_COLLISION\n");
  9644. +        break;
  9645. +    case STATUS_FILE_INVALID:
  9646. +        DbgP("[ERROR] nfs41_Create: STATUS_FILE_INVALID\n");
  9647. +        break;
  9648. +    case STATUS_OBJECT_NAME_NOT_FOUND:
  9649. +        DbgP("[ERROR] nfs41_Create: STATUS_OBJECT_NAME_NOT_FOUND\n");
  9650. +        break;
  9651. +    case STATUS_NAME_TOO_LONG:
  9652. +        DbgP("[ERROR] nfs41_Create: STATUS_NAME_TOO_LONG\n");
  9653. +        break;
  9654. +    case STATUS_OBJECT_PATH_NOT_FOUND:
  9655. +        DbgP("[ERROR] nfs41_Create: STATUS_OBJECT_PATH_NOT_FOUND\n");
  9656. +        break;
  9657. +    case STATUS_BAD_NETWORK_PATH:
  9658. +        DbgP("[ERROR] nfs41_Create: STATUS_BAD_NETWORK_PATH\n");
  9659. +        break;
  9660. +    case STATUS_SHARING_VIOLATION:
  9661. +        DbgP("[ERROR] nfs41_Create: STATUS_SHARING_VIOLATION\n");
  9662. +        break;
  9663. +    case ERROR_REPARSE:
  9664. +        DbgP("[ERROR] nfs41_Create: STATUS_REPARSE\n");
  9665. +        break;
  9666. +    case ERROR_TOO_MANY_LINKS:
  9667. +        DbgP("[ERROR] nfs41_Create: STATUS_TOO_MANY_LINKS\n");
  9668. +        break;
  9669. +    case ERROR_DIRECTORY:
  9670. +        DbgP("[ERROR] nfs41_Create: STATUS_FILE_IS_A_DIRECTORY\n");
  9671. +        break;
  9672. +    case ERROR_BAD_FILE_TYPE:
  9673. +        DbgP("[ERROR] nfs41_Create: STATUS_NOT_A_DIRECTORY\n");
  9674. +        break;
  9675. +    default:
  9676. +        DbgP("[ERROR] nfs41_Create: STATUS_INSUFFICIENT_RESOURCES\n");
  9677. +        break;
  9678.      }
  9679.  }
  9680.  
  9681. @@ -738,32 +738,32 @@ void print_wait_status(int on, const char *prefix, NTSTATUS status,
  9682.      }
  9683.  }
  9684.  /* This is taken from toaster/func.  Rumor says this should be replaced
  9685. - * with a WMI interface???
  9686. - */
  9687. -ULONG
  9688. -dprintk(
  9689. -    IN PCHAR func,
  9690. -    IN ULONG flags,
  9691. -    IN PCHAR format,
  9692. -    ...)
  9693. -{
  9694. -    #define     TEMP_BUFFER_SIZE        1024
  9695. -    va_list    list;
  9696. -    CHAR      debugMessageBuffer[TEMP_BUFFER_SIZE];
  9697. -    NTSTATUS status, rv = STATUS_SUCCESS;
  9698. -
  9699. -    va_start(list, format);
  9700. -
  9701. -    if (format)
  9702. -    {
  9703. -        //
  9704. -        // Use the safe string function, RtlStringCbVPrintfA, instead of _vsnprintf.
  9705. -        // RtlStringCbVPrintfA NULL terminates the output buffer even if the message
  9706. -        // is longer than the buffer. This prevents malicious code from compromising
  9707. -        // the security of the system.
  9708. -        //
  9709. -        status = RtlStringCbVPrintfA(debugMessageBuffer, sizeof(debugMessageBuffer),
  9710. -                                    format, list);
  9711. + * with a WMI interface???
  9712. + */
  9713. +ULONG
  9714. +dprintk(
  9715. +    IN PCHAR func,
  9716. +    IN ULONG flags,
  9717. +    IN PCHAR format,
  9718. +    ...)
  9719. +{
  9720. +    #define     TEMP_BUFFER_SIZE        1024
  9721. +    va_list    list;
  9722. +    CHAR      debugMessageBuffer[TEMP_BUFFER_SIZE];
  9723. +    NTSTATUS status, rv = STATUS_SUCCESS;
  9724. +
  9725. +    va_start(list, format);
  9726. +
  9727. +    if (format)
  9728. +    {
  9729. +        //
  9730. +        // Use the safe string function, RtlStringCbVPrintfA, instead of _vsnprintf.
  9731. +        // RtlStringCbVPrintfA NULL terminates the output buffer even if the message
  9732. +        // is longer than the buffer. This prevents malicious code from compromising
  9733. +        // the security of the system.
  9734. +        //
  9735. +        status = RtlStringCbVPrintfA(debugMessageBuffer, sizeof(debugMessageBuffer),
  9736. +                                    format, list);
  9737.  
  9738.          if (!NT_SUCCESS(status))
  9739.              rv = DbgPrintEx(PNFS_FLTR_ID, DPFLTR_MASK | flags,
  9740. @@ -773,7 +773,7 @@ dprintk(
  9741.                      PNFS_TRACE_TAG, func, debugMessageBuffer);
  9742.      }
  9743.      va_end(list);
  9744. -
  9745. +
  9746.      return rv;
  9747.  }
  9748.  
  9749. @@ -1016,3 +1016,24 @@ void print_lookasidelist_stat(const char *label, PNPAGED_LOOKASIDE_LIST ll)
  9750.          (long)ll->L.MaximumDepth);
  9751.  }
  9752.  #endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  9753. +
  9754. +void print_debug_header(
  9755. +    PRX_CONTEXT RxContext)
  9756. +{
  9757. +    PIO_STACK_LOCATION IrpSp = RxContext->CurrentIrpSp;
  9758. +
  9759. +    if (IrpSp) {
  9760. +        DbgP("FileOject=0x%p name='%wZ' access r=%d,w=%d,d=%d share r=%d,w=%d,d=%d\n",
  9761. +            IrpSp->FileObject, &IrpSp->FileObject->FileName,
  9762. +            IrpSp->FileObject->ReadAccess, IrpSp->FileObject->WriteAccess,
  9763. +            IrpSp->FileObject->DeleteAccess, IrpSp->FileObject->SharedRead,
  9764. +            IrpSp->FileObject->SharedWrite, IrpSp->FileObject->SharedDelete);
  9765. +        print_file_object(0, IrpSp->FileObject);
  9766. +        print_irps_flags(0, RxContext->CurrentIrpSp);
  9767. +    } else
  9768. +        DbgP("Couldn't print FileObject IrpSp is NULL\n");
  9769. +
  9770. +    print_fo_all(1, RxContext);
  9771. +    if (RxContext->CurrentIrp)
  9772. +        print_irp_flags(0, RxContext->CurrentIrp);
  9773. +}
  9774. diff --git a/sys/nfs41_debug.h b/sys/nfs41sys_debug.h
  9775. similarity index 79%
  9776. rename from sys/nfs41_debug.h
  9777. rename to sys/nfs41sys_debug.h
  9778. index 5799929..d5e0f03 100644
  9779. --- a/sys/nfs41_debug.h
  9780. +++ b/sys/nfs41sys_debug.h
  9781. @@ -1,5 +1,5 @@
  9782. -/* NFSv4.1 client for Windows
  9783. - * Copyright (C) 2012 The Regents of the University of Michigan
  9784. +/* NFSv4.1 client for Windows
  9785. + * Copyright (C) 2012 The Regents of the University of Michigan
  9786.   *
  9787.   * Olga Kornievskaia <aglo@umich.edu>
  9788.   * Casey Bodley <cbodley@umich.edu>
  9789. @@ -7,24 +7,24 @@
  9790.   *
  9791.   * This library is free software; you can redistribute it and/or modify it
  9792.   * under the terms of the GNU Lesser General Public License as published by
  9793. - * the Free Software Foundation; either version 2.1 of the License, or (at
  9794. - * your option) any later version.
  9795. - *
  9796. - * This library is distributed in the hope that it will be useful, but
  9797. - * without any warranty; without even the implied warranty of merchantability
  9798. - * or fitness for a particular purpose.  See the GNU Lesser General Public
  9799. - * License for more details.
  9800. - *
  9801. - * You should have received a copy of the GNU Lesser General Public License
  9802. - * along with this library; if not, write to the Free Software Foundation,
  9803. - * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  9804. - */
  9805. -
  9806. -#ifndef _NFS41_DEBUG_
  9807. -#define _NFS41_DEBUG_
  9808. -
  9809. -#define _DRIVER_NAME_ "NFS4.1 Driver"
  9810. -
  9811. + * the Free Software Foundation; either version 2.1 of the License, or (at
  9812. + * your option) any later version.
  9813. + *
  9814. + * This library is distributed in the hope that it will be useful, but
  9815. + * without any warranty; without even the implied warranty of merchantability
  9816. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  9817. + * License for more details.
  9818. + *
  9819. + * You should have received a copy of the GNU Lesser General Public License
  9820. + * along with this library; if not, write to the Free Software Foundation,
  9821. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  9822. + */
  9823. +
  9824. +#ifndef NFS41SYS_DEBUG_H
  9825. +#define NFS41SYS_DEBUG_H 1
  9826. +
  9827. +#define _DRIVER_NAME_ "NFS4.1 Driver"
  9828. +
  9829.  ULONG __cdecl DbgP(IN PCCH fmt, ...);
  9830.  ULONG __cdecl print_error(IN PCCH fmt, ...);
  9831.  VOID print_fo_all(int on, IN PRX_CONTEXT c);
  9832. @@ -34,11 +34,11 @@ VOID print_v_net_root(IN PMRX_V_NET_ROOT p);
  9833.  VOID print_fcb(int on, IN PMRX_FCB p);
  9834.  VOID print_srv_open(int on, IN PMRX_SRV_OPEN p);
  9835.  VOID print_fobx(int on, IN PMRX_FOBX p);
  9836. -VOID print_irp_flags(int on, PIRP irp);
  9837. -VOID print_irps_flags(int on, PIO_STACK_LOCATION irps);
  9838. +VOID print_irp_flags(int on, PIRP irp);
  9839. +VOID print_irps_flags(int on, PIO_STACK_LOCATION irps);
  9840.  void print_nt_create_params(int on, NT_CREATE_PARAMETERS params);
  9841. -unsigned char *print_file_information_class(int InfoClass);
  9842. -unsigned char *print_fs_information_class(int InfoClass);
  9843. +const char *print_file_information_class(int InfoClass);
  9844. +const char *print_fs_information_class(int InfoClass);
  9845.  void print_hexbuf(const char *title, unsigned char *buf, int len);
  9846.  void print_ioctl(int op);
  9847.  void print_fs_ioctl(int op);
  9848. @@ -59,6 +59,7 @@ const char *fsctl2string(ULONG fsctl);
  9849.  #ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  9850.  void print_lookasidelist_stat(const char *label, PNPAGED_LOOKASIDE_LIST ll);
  9851.  #endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  9852. +void print_debug_header(PRX_CONTEXT RxContext);
  9853.  
  9854.  #define PTR2PTRDIFF_T(p) (((char *)(p))-((char *)0))
  9855.  #define PsGetCurrentProcessShortDebugId() ((int)PTR2PTRDIFF_T(PsGetCurrentProcessId()))
  9856. @@ -69,7 +70,7 @@ void print_lookasidelist_stat(const char *label, PNPAGED_LOOKASIDE_LIST ll);
  9857.          __func__); __try {
  9858.  
  9859.  #define DbgEx() DbgPrintEx(DPFLTR_IHVNETWORK_ID, DPFLTR_ERROR_LEVEL, \
  9860. -        "<-- [%s] [%04x] %s status = %08lx\n", _DRIVER_NAME_, PsGetCurrentProcessShortDebugId(), \
  9861. +        "<-- [%s] [%04x] %s status = 0x%08lx\n", _DRIVER_NAME_, PsGetCurrentProcessShortDebugId(), \
  9862.          __func__, status); \
  9863.          } __except (EXCEPTION_EXECUTE_HANDLER) { \
  9864.              status = GetExceptionCode() ; \
  9865. @@ -80,33 +81,37 @@ void print_lookasidelist_stat(const char *label, PNPAGED_LOOKASIDE_LIST ll);
  9866.          } __except (EXCEPTION_EXECUTE_HANDLER) { \
  9867.              NTSTATUS exc_status; \
  9868.              exc_status = GetExceptionCode() ; \
  9869. -            DbgP("Exception encountered with value = Ox%x\n", (int)exc_status); \
  9870. +            DbgP("Exception encountered with value = 0x%x\n", (int)exc_status); \
  9871.          }
  9872.  
  9873.  /* These are for ToasterDebugPrint */
  9874. -
  9875. -#define     DBG_ERROR       0x00000001
  9876. -#define     DBG_WARN        0x00000002
  9877. -#define     DBG_TRACE       0x00000004
  9878. -#define     DBG_INFO        0x00000008
  9879. -#define     DBG_DISP_IN     0x00000010 /* Marks entry into dispatch functions */
  9880. -#define     DBG_DISP_OUT    0x00000020 /* Marks exit from dispatch functions */
  9881. -
  9882. -/* I want to do:
  9883. - * #define dprintk(flags, args...) DbgPrintEx(DPFLTR_IHVDRIVER_ID, DPFLTR_MASK | flags, ## args)
  9884. - * but the ... is gcc specific, can't seem to do it here.
  9885. - */
  9886. -#define PNFS_TRACE_TAG      "PNFSMRX: "
  9887. +
  9888. +#define DBG_ERROR    0x00000001
  9889. +#define DBG_WARN     0x00000002
  9890. +#define DBG_TRACE    0x00000004
  9891. +#define DBG_INFO     0x00000008
  9892. +#define DBG_DISP_IN  0x00000010 /* Marks entry into dispatch functions */
  9893. +#define DBG_DISP_OUT 0x00000020 /* Marks exit from dispatch functions */
  9894. +
  9895. +/* I want to do:
  9896. + * #define dprintk(flags, args...) DbgPrintEx(DPFLTR_IHVDRIVER_ID, DPFLTR_MASK | flags, ## args)
  9897. + * but the ... is gcc specific, can't seem to do it here.
  9898. + */
  9899. +#define PNFS_TRACE_TAG      "PNFSMRX: "
  9900.  #define PNFS_FLTR_ID        DPFLTR_IHVDRIVER_ID
  9901.  
  9902. -#define DbgEnter()      DbgPrintEx(PNFS_FLTR_ID, DPFLTR_MASK | DBG_DISP_IN, "%s*** %s ***\n", \
  9903. -                                PNFS_TRACE_TAG, __func__);
  9904. -#define DbgExit(status) DbgPrintEx(PNFS_FLTR_ID, DPFLTR_MASK | DBG_DISP_OUT, "%s<-- %s <-- 0x%08lx\n", \
  9905. -                                PNFS_TRACE_TAG, __func__, status);
  9906. +#define DbgEnter() \
  9907. +    DbgPrintEx(PNFS_FLTR_ID, DPFLTR_MASK | DBG_DISP_IN, "%s*** %s ***\n", \
  9908. +        PNFS_TRACE_TAG, __func__);
  9909. +#define DbgExit(status) \
  9910. +    DbgPrintEx(PNFS_FLTR_ID, DPFLTR_MASK | DBG_DISP_OUT, "%s<-- %s <-- 0x%08lx\n", \
  9911. +        PNFS_TRACE_TAG, __func__, (status));
  9912. +
  9913.  ULONG
  9914.  dprintk(
  9915.      IN PCHAR func,
  9916. -    IN ULONG flags,
  9917. -    IN PCHAR format,
  9918. -    ...);
  9919. -#endif
  9920. \ No newline at end of file
  9921. +    IN ULONG flags,
  9922. +    IN PCHAR format,
  9923. +    ...);
  9924. +
  9925. +#endif /* !NFS41SYS_DEBUG_H */
  9926. diff --git a/sys/nfs41sys_dir.c b/sys/nfs41sys_dir.c
  9927. new file mode 100644
  9928. index 0000000..4dc0bf5
  9929. --- /dev/null
  9930. +++ b/sys/nfs41sys_dir.c
  9931. @@ -0,0 +1,330 @@
  9932. +/* NFSv4.1 client for Windows
  9933. + * Copyright (C) 2012 The Regents of the University of Michigan
  9934. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  9935. + *
  9936. + * Olga Kornievskaia <aglo@umich.edu>
  9937. + * Casey Bodley <cbodley@umich.edu>
  9938. + * Roland Mainz <roland.mainz@nrubsig.org>
  9939. + *
  9940. + * This library is free software; you can redistribute it and/or modify it
  9941. + * under the terms of the GNU Lesser General Public License as published by
  9942. + * the Free Software Foundation; either version 2.1 of the License, or (at
  9943. + * your option) any later version.
  9944. + *
  9945. + * This library is distributed in the hope that it will be useful, but
  9946. + * without any warranty; without even the implied warranty of merchantability
  9947. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  9948. + * License for more details.
  9949. + *
  9950. + * You should have received a copy of the GNU Lesser General Public License
  9951. + * along with this library; if not, write to the Free Software Foundation,
  9952. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  9953. + */
  9954. +
  9955. +#ifndef _KERNEL_MODE
  9956. +#error module requires kernel mode
  9957. +#endif
  9958. +
  9959. +#if ((__STDC_VERSION__-0) < 201710L)
  9960. +#error Code requires ISO C17
  9961. +#endif
  9962. +
  9963. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  9964. +#if _MSC_VER >= 1900
  9965. +#if defined(_WIN64) && defined(_M_X64)
  9966. +#ifndef _AMD64_
  9967. +#define _AMD64_
  9968. +#endif
  9969. +#elif defined(_WIN32) && defined(_M_IX86)
  9970. +#ifndef _X86_
  9971. +#define _X86_
  9972. +#endif
  9973. +#elif defined(_WIN64) && defined(_M_ARM64)
  9974. +#ifndef _ARM64_
  9975. +#define _ARM64_
  9976. +#endif
  9977. +#elif defined(_WIN32) && defined(_M_ARM)
  9978. +#ifndef _ARM_
  9979. +#define _ARM_
  9980. +#endif
  9981. +#else
  9982. +#error Unsupported arch
  9983. +#endif
  9984. +#endif /* _MSC_VER >= 1900 */
  9985. +
  9986. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  9987. +#include <rx.h>
  9988. +#include <windef.h>
  9989. +#include <winerror.h>
  9990. +
  9991. +#include <Ntstrsafe.h>
  9992. +
  9993. +#include "nfs41sys_buildconfig.h"
  9994. +
  9995. +#include "nfs41_driver.h"
  9996. +#include "nfs41sys_debug.h"
  9997. +#include "nfs41_build_features.h"
  9998. +
  9999. +#include "nfs41sys_driver.h"
  10000. +#include "nfs41sys_util.h"
  10001. +
  10002. +NTSTATUS marshal_nfs41_dirquery(
  10003. +    nfs41_updowncall_entry *entry,
  10004. +    unsigned char *buf,
  10005. +    ULONG buf_len,
  10006. +    ULONG *len)
  10007. +{
  10008. +    NTSTATUS status = STATUS_SUCCESS;
  10009. +    ULONG header_len = 0;
  10010. +    unsigned char *tmp = buf;
  10011. +
  10012. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  10013. +    if (status) goto out;
  10014. +    else tmp += *len;
  10015. +
  10016. +    header_len = *len + 2 * sizeof(ULONG) + sizeof(HANDLE) +
  10017. +        length_as_utf8(entry->u.QueryFile.filter) + 3 * sizeof(BOOLEAN);
  10018. +    if (header_len > buf_len) {
  10019. +        status = STATUS_INSUFFICIENT_RESOURCES;
  10020. +        goto out;
  10021. +    }
  10022. +
  10023. +    RtlCopyMemory(tmp, &entry->u.QueryFile.InfoClass, sizeof(ULONG));
  10024. +    tmp += sizeof(ULONG);
  10025. +    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  10026. +    tmp += sizeof(ULONG);
  10027. +    status = marshall_unicode_as_utf8(&tmp, entry->u.QueryFile.filter);
  10028. +    if (status) goto out;
  10029. +    RtlCopyMemory(tmp, &entry->u.QueryFile.initial_query, sizeof(BOOLEAN));
  10030. +    tmp += sizeof(BOOLEAN);
  10031. +    RtlCopyMemory(tmp, &entry->u.QueryFile.restart_scan, sizeof(BOOLEAN));
  10032. +    tmp += sizeof(BOOLEAN);
  10033. +    RtlCopyMemory(tmp, &entry->u.QueryFile.return_single, sizeof(BOOLEAN));
  10034. +    tmp += sizeof(BOOLEAN);
  10035. +    __try {
  10036. +        entry->u.QueryFile.mdl_buf =
  10037. +            MmMapLockedPagesSpecifyCache(entry->u.QueryFile.mdl,
  10038. +                UserMode, MmCached, NULL, TRUE,
  10039. +                NormalPagePriority|MdlMappingNoExecute);
  10040. +        if (entry->u.QueryFile.mdl_buf == NULL) {
  10041. +            print_error("marshal_nfs41_dirquery: "
  10042. +                "MmMapLockedPagesSpecifyCache() failed to map pages\n");
  10043. +            status = STATUS_INSUFFICIENT_RESOURCES;
  10044. +            goto out;
  10045. +        }
  10046. +    } __except(EXCEPTION_EXECUTE_HANDLER) {
  10047. +        NTSTATUS code;
  10048. +        code = GetExceptionCode();
  10049. +        print_error("marshal_nfs41_dirquery: Call to "
  10050. +            "MmMapLockedPagesSpecifyCache() failed "
  10051. +            "due to exception 0x%x\n", (int)code);
  10052. +        status = STATUS_ACCESS_VIOLATION;
  10053. +        goto out;
  10054. +    }
  10055. +    RtlCopyMemory(tmp, &entry->u.QueryFile.mdl_buf, sizeof(HANDLE));
  10056. +    *len = header_len;
  10057. +
  10058. +#ifdef DEBUG_MARSHAL_DETAIL
  10059. +    DbgP("marshal_nfs41_dirquery: filter='%wZ' class=%d len=%d "
  10060. +         "1st\\restart\\single=%d\\%d\\%d\n", entry->u.QueryFile.filter,
  10061. +         entry->u.QueryFile.InfoClass, entry->buf_len,
  10062. +         entry->u.QueryFile.initial_query, entry->u.QueryFile.restart_scan,
  10063. +         entry->u.QueryFile.return_single);
  10064. +#endif
  10065. +out:
  10066. +    return status;
  10067. +}
  10068. +
  10069. +NTSTATUS unmarshal_nfs41_dirquery(
  10070. +    nfs41_updowncall_entry *cur,
  10071. +    unsigned char **buf)
  10072. +{
  10073. +    NTSTATUS status = STATUS_SUCCESS;
  10074. +    ULONG buf_len;
  10075. +
  10076. +    RtlCopyMemory(&buf_len, *buf, sizeof(ULONG));
  10077. +#ifdef DEBUG_MARSHAL_DETAIL
  10078. +    DbgP("unmarshal_nfs41_dirquery: reply size %d\n", buf_len);
  10079. +#endif
  10080. +    *buf += sizeof(ULONG);
  10081. +    __try {
  10082. +        MmUnmapLockedPages(cur->u.QueryFile.mdl_buf, cur->u.QueryFile.mdl);
  10083. +    } __except(EXCEPTION_EXECUTE_HANDLER) {
  10084. +        NTSTATUS code;
  10085. +        code = GetExceptionCode();
  10086. +        print_error("MmUnmapLockedPages thrown exception=0x%0x\n", code);
  10087. +        status = STATUS_ACCESS_VIOLATION;
  10088. +    }
  10089. +    if (buf_len > cur->buf_len)
  10090. +        cur->status = STATUS_BUFFER_TOO_SMALL;
  10091. +    cur->buf_len = buf_len;
  10092. +
  10093. +    return status;
  10094. +}
  10095. +
  10096. +static void print_debug_filedirquery_header(
  10097. +    PRX_CONTEXT RxContext)
  10098. +{
  10099. +    print_debug_header(RxContext);
  10100. +    DbgP("FileName='%wZ', InfoClass = '%s'\n",
  10101. +        GET_ALREADY_PREFIXED_NAME_FROM_CONTEXT(RxContext),
  10102. +        print_file_information_class(RxContext->Info.FileInformationClass));
  10103. +}
  10104. +
  10105. +static void print_querydir_args(
  10106. +    PRX_CONTEXT RxContext)
  10107. +{
  10108. +    print_debug_filedirquery_header(RxContext);
  10109. +    DbgP("Filter='%wZ', Index=%d, Restart/Single/Specified/Init=%d/%d/%d/%d\n",
  10110. +        &RxContext->pFobx->UnicodeQueryTemplate,
  10111. +        RxContext->QueryDirectory.FileIndex,
  10112. +        RxContext->QueryDirectory.RestartScan,
  10113. +        RxContext->QueryDirectory.ReturnSingleEntry,
  10114. +        RxContext->QueryDirectory.IndexSpecified,
  10115. +        RxContext->QueryDirectory.InitialQuery);
  10116. +}
  10117. +
  10118. +NTSTATUS map_querydir_errors(
  10119. +    DWORD status)
  10120. +{
  10121. +    switch (status) {
  10122. +    case ERROR_ACCESS_DENIED:       return STATUS_ACCESS_DENIED;
  10123. +    case ERROR_BUFFER_OVERFLOW:     return STATUS_BUFFER_OVERFLOW;
  10124. +    case ERROR_FILE_NOT_FOUND:      return STATUS_NO_SUCH_FILE;
  10125. +    case ERROR_NETNAME_DELETED:     return STATUS_NETWORK_NAME_DELETED;
  10126. +    case ERROR_INVALID_PARAMETER:   return STATUS_INVALID_PARAMETER;
  10127. +    case ERROR_NO_MORE_FILES:       return STATUS_NO_MORE_FILES;
  10128. +    case ERROR_OUTOFMEMORY:         return STATUS_INSUFFICIENT_RESOURCES;
  10129. +    case ERROR_FILENAME_EXCED_RANGE: return STATUS_NAME_TOO_LONG;
  10130. +    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  10131. +    default:
  10132. +        print_error("map_querydir_errors: "
  10133. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  10134. +            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status);
  10135. +    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  10136. +    }
  10137. +}
  10138. +
  10139. +NTSTATUS check_nfs41_dirquery_args(
  10140. +    IN PRX_CONTEXT RxContext)
  10141. +{
  10142. +    if (RxContext->Info.Buffer == NULL)
  10143. +        return STATUS_INVALID_USER_BUFFER;
  10144. +    return STATUS_SUCCESS;
  10145. +}
  10146. +
  10147. +NTSTATUS nfs41_QueryDirectory(
  10148. +    IN OUT PRX_CONTEXT RxContext)
  10149. +{
  10150. +    NTSTATUS status = STATUS_INVALID_PARAMETER;
  10151. +    nfs41_updowncall_entry *entry;
  10152. +    FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass;
  10153. +    PUNICODE_STRING Filter = &RxContext->pFobx->UnicodeQueryTemplate;
  10154. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  10155. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  10156. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  10157. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  10158. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  10159. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  10160. +#ifdef ENABLE_TIMINGS
  10161. +    LARGE_INTEGER t1, t2;
  10162. +    t1 = KeQueryPerformanceCounter(NULL);
  10163. +#endif
  10164. +
  10165. +#ifdef DEBUG_DIR_QUERY
  10166. +    DbgEn();
  10167. +    print_querydir_args(RxContext);
  10168. +#endif
  10169. +
  10170. +    status = check_nfs41_dirquery_args(RxContext);
  10171. +    if (status) goto out;
  10172. +
  10173. +    switch (InfoClass) {
  10174. +    /* classes handled in readdir_copy_entry() and readdir_size_for_entry() */
  10175. +    case FileNamesInformation:
  10176. +    case FileDirectoryInformation:
  10177. +    case FileFullDirectoryInformation:
  10178. +    case FileIdFullDirectoryInformation:
  10179. +    case FileBothDirectoryInformation:
  10180. +    case FileIdBothDirectoryInformation:
  10181. +        break;
  10182. +    default:
  10183. +        print_error("nfs41_QueryDirectory: unhandled dir query class %d\n",
  10184. +            InfoClass);
  10185. +        status = STATUS_NOT_SUPPORTED;
  10186. +        goto out;
  10187. +    }
  10188. +    status = nfs41_UpcallCreate(NFS41_DIR_QUERY, &nfs41_fobx->sec_ctx,
  10189. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  10190. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  10191. +    if (status) goto out;
  10192. +
  10193. +    entry->u.QueryFile.InfoClass = InfoClass;
  10194. +    entry->buf_len = RxContext->Info.LengthRemaining;
  10195. +    entry->buf = RxContext->Info.Buffer;
  10196. +    entry->u.QueryFile.mdl = IoAllocateMdl(RxContext->Info.Buffer,
  10197. +        RxContext->Info.LengthRemaining, FALSE, FALSE, NULL);
  10198. +    if (entry->u.QueryFile.mdl == NULL) {
  10199. +        status = STATUS_INTERNAL_ERROR;
  10200. +        nfs41_UpcallDestroy(entry);
  10201. +        goto out;
  10202. +    }
  10203. +#pragma warning( push )
  10204. +/*
  10205. + * C28145: "The opaque MDL structure should not be modified by a
  10206. + * driver.", |MDL_MAPPING_CAN_FAIL| is the exception
  10207. + */
  10208. +#pragma warning (disable : 28145)
  10209. +    entry->u.QueryFile.mdl->MdlFlags |= MDL_MAPPING_CAN_FAIL;
  10210. +#pragma warning( pop )
  10211. +
  10212. +    MmProbeAndLockPages(entry->u.QueryFile.mdl, KernelMode, IoModifyAccess);
  10213. +
  10214. +    entry->u.QueryFile.filter = Filter;
  10215. +    entry->u.QueryFile.initial_query = RxContext->QueryDirectory.InitialQuery;
  10216. +    entry->u.QueryFile.restart_scan = RxContext->QueryDirectory.RestartScan;
  10217. +    entry->u.QueryFile.return_single = RxContext->QueryDirectory.ReturnSingleEntry;
  10218. +
  10219. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  10220. +    if (status) goto out;
  10221. +    MmUnlockPages(entry->u.QueryFile.mdl);
  10222. +
  10223. +    if (entry->status == STATUS_BUFFER_TOO_SMALL) {
  10224. +        DbgP("nfs41_QueryDirectory: buffer too small provided %d need %lu\n",
  10225. +            RxContext->Info.LengthRemaining, entry->buf_len);
  10226. +        RxContext->InformationToReturn = entry->buf_len;
  10227. +        status = STATUS_BUFFER_TOO_SMALL;
  10228. +    } else if (entry->status == STATUS_SUCCESS) {
  10229. +#ifdef ENABLE_TIMINGS
  10230. +        InterlockedIncrement(&readdir.sops);
  10231. +        InterlockedAdd64(&readdir.size, entry->u.QueryFile.buf_len);
  10232. +#endif
  10233. +        RxContext->Info.LengthRemaining -= entry->buf_len;
  10234. +        status = STATUS_SUCCESS;
  10235. +    } else if ((entry->status == STATUS_ACCESS_VIOLATION) ||
  10236. +        (entry->status == STATUS_INSUFFICIENT_RESOURCES)) {
  10237. +        DbgP("nfs41_QueryDirectory: internal error: entry->status=0x%x\n",
  10238. +            (int)entry->status);
  10239. +        status = STATUS_INSUFFICIENT_RESOURCES;
  10240. +    } else {
  10241. +        /* map windows ERRORs to NTSTATUS */
  10242. +        status = map_querydir_errors(entry->status);
  10243. +    }
  10244. +    IoFreeMdl(entry->u.QueryFile.mdl);
  10245. +    nfs41_UpcallDestroy(entry);
  10246. +out:
  10247. +#ifdef ENABLE_TIMINGS
  10248. +    t2 = KeQueryPerformanceCounter(NULL);
  10249. +    InterlockedIncrement(&readdir.tops);
  10250. +    InterlockedAdd64(&readdir.ticks, t2.QuadPart - t1.QuadPart);
  10251. +#ifdef ENABLE_INDV_TIMINGS
  10252. +    DbgP("nfs41_QueryDirectory delta = %d ops=%d sum=%d\n",
  10253. +        t2.QuadPart - t1.QuadPart, readdir.tops, readdir.ticks);
  10254. +#endif
  10255. +#endif
  10256. +#ifdef DEBUG_DIR_QUERY
  10257. +    DbgEx();
  10258. +#endif
  10259. +    return status;
  10260. +}
  10261. +
  10262. diff --git a/sys/nfs41sys_driver.c b/sys/nfs41sys_driver.c
  10263. new file mode 100644
  10264. index 0000000..193ab28
  10265. --- /dev/null
  10266. +++ b/sys/nfs41sys_driver.c
  10267. @@ -0,0 +1,1444 @@
  10268. +/* NFSv4.1 client for Windows
  10269. + * Copyright (C) 2012 The Regents of the University of Michigan
  10270. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  10271. + *
  10272. + * Olga Kornievskaia <aglo@umich.edu>
  10273. + * Casey Bodley <cbodley@umich.edu>
  10274. + * Roland Mainz <roland.mainz@nrubsig.org>
  10275. + *
  10276. + * This library is free software; you can redistribute it and/or modify it
  10277. + * under the terms of the GNU Lesser General Public License as published by
  10278. + * the Free Software Foundation; either version 2.1 of the License, or (at
  10279. + * your option) any later version.
  10280. + *
  10281. + * This library is distributed in the hope that it will be useful, but
  10282. + * without any warranty; without even the implied warranty of merchantability
  10283. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  10284. + * License for more details.
  10285. + *
  10286. + * You should have received a copy of the GNU Lesser General Public License
  10287. + * along with this library; if not, write to the Free Software Foundation,
  10288. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  10289. + */
  10290. +
  10291. +#ifndef _KERNEL_MODE
  10292. +#error module requires kernel mode
  10293. +#endif
  10294. +
  10295. +#if ((__STDC_VERSION__-0) < 201710L)
  10296. +#error Code requires ISO C17
  10297. +#endif
  10298. +
  10299. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  10300. +#if _MSC_VER >= 1900
  10301. +#if defined(_WIN64) && defined(_M_X64)
  10302. +#ifndef _AMD64_
  10303. +#define _AMD64_
  10304. +#endif
  10305. +#elif defined(_WIN32) && defined(_M_IX86)
  10306. +#ifndef _X86_
  10307. +#define _X86_
  10308. +#endif
  10309. +#elif defined(_WIN64) && defined(_M_ARM64)
  10310. +#ifndef _ARM64_
  10311. +#define _ARM64_
  10312. +#endif
  10313. +#elif defined(_WIN32) && defined(_M_ARM)
  10314. +#ifndef _ARM_
  10315. +#define _ARM_
  10316. +#endif
  10317. +#else
  10318. +#error Unsupported arch
  10319. +#endif
  10320. +#endif /* _MSC_VER >= 1900 */
  10321. +
  10322. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  10323. +#include <rx.h>
  10324. +#include <windef.h>
  10325. +#include <winerror.h>
  10326. +
  10327. +#include <Ntstrsafe.h>
  10328. +
  10329. +#include "nfs41sys_buildconfig.h"
  10330. +
  10331. +#include "nfs41_driver.h"
  10332. +#include "nfs41_np.h"
  10333. +#include "nfs41sys_debug.h"
  10334. +#include "nfs41_build_features.h"
  10335. +#include "nfs_ea.h"
  10336. +
  10337. +#include "nfs41sys_driver.h"
  10338. +#include "nfs41sys_util.h"
  10339. +
  10340. +/*
  10341. + * In order to cooperate with other network providers,
  10342. + * we only claim paths of the format '\\server\nfs4\path' or
  10343. + * '\\server\pubnfs4\path'
  10344. + */
  10345. +DECLARE_CONST_UNICODE_STRING(NfsPrefix, L"\\nfs4");
  10346. +DECLARE_CONST_UNICODE_STRING(PubNfsPrefix, L"\\pubnfs4");
  10347. +DECLARE_CONST_UNICODE_STRING(AUTH_SYS_NAME, L"sys");
  10348. +DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5_NAME, L"krb5");
  10349. +DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5I_NAME, L"krb5i");
  10350. +DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5P_NAME, L"krb5p");
  10351. +DECLARE_CONST_UNICODE_STRING(SLASH, L"\\");
  10352. +DECLARE_CONST_UNICODE_STRING(EMPTY_STRING, L"");
  10353. +
  10354. +DECLARE_CONST_ANSI_STRING(NfsV3Attributes, EA_NFSV3ATTRIBUTES);
  10355. +DECLARE_CONST_ANSI_STRING(NfsSymlinkTargetName, EA_NFSSYMLINKTARGETNAME);
  10356. +DECLARE_CONST_ANSI_STRING(NfsActOnLink, EA_NFSACTONLINK);
  10357. +
  10358. +#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  10359. +NPAGED_LOOKASIDE_LIST updowncall_entry_upcall_lookasidelist;
  10360. +NPAGED_LOOKASIDE_LIST updowncall_entry_downcall_lookasidelist;
  10361. +#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  10362. +
  10363. +#ifdef ENABLE_TIMINGS
  10364. +nfs41_timings lookup;
  10365. +nfs41_timings readdir;
  10366. +nfs41_timings open;
  10367. +nfs41_timings close;
  10368. +nfs41_timings getattr;
  10369. +nfs41_timings setattr;
  10370. +nfs41_timings getacl;
  10371. +nfs41_timings setacl;
  10372. +nfs41_timings volume;
  10373. +nfs41_timings read;
  10374. +nfs41_timings write;
  10375. +nfs41_timings lock;
  10376. +nfs41_timings unlock;
  10377. +nfs41_timings setexattr;
  10378. +nfs41_timings getexattr;
  10379. +#endif /* ENABLE_TIMINGS */
  10380. +
  10381. +DRIVER_INITIALIZE DriverEntry;
  10382. +DRIVER_UNLOAD nfs41_driver_unload;
  10383. +_Dispatch_type_(IRP_MJ_CREATE) \
  10384. +    _Dispatch_type_(IRP_MJ_CREATE_NAMED_PIPE) \
  10385. +    DRIVER_DISPATCH(nfs41_FsdDispatch);
  10386. +
  10387. +struct _MINIRDR_DISPATCH nfs41_ops;
  10388. +PRDBSS_DEVICE_OBJECT nfs41_dev;
  10389. +
  10390. +
  10391. +KEVENT upcallEvent;
  10392. +FAST_MUTEX upcallLock, downcallLock, fcblistLock;
  10393. +FAST_MUTEX openOwnerLock;
  10394. +
  10395. +LONGLONG xid = 0;
  10396. +LONG open_owner_id = 1;
  10397. +
  10398. +
  10399. +#ifdef NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL
  10400. +const LUID SystemLuid = SYSTEM_LUID;
  10401. +#endif /* NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL */
  10402. +
  10403. +/* |unix_time_diff| - needed to convert windows time to unix */
  10404. +LARGE_INTEGER unix_time_diff;
  10405. +
  10406. +
  10407. +nfs41_init_driver_state nfs41_init_state = NFS41_INIT_DRIVER_STARTABLE;
  10408. +nfs41_start_driver_state nfs41_start_state = NFS41_START_DRIVER_STARTABLE;
  10409. +
  10410. +
  10411. +NTSTATUS marshall_unicode_as_utf8(
  10412. +    IN OUT unsigned char **pos,
  10413. +    IN PCUNICODE_STRING str)
  10414. +{
  10415. +    ANSI_STRING ansi;
  10416. +    ULONG ActualCount;
  10417. +    NTSTATUS status;
  10418. +
  10419. +    if (str->Length == 0) {
  10420. +        status = STATUS_SUCCESS;
  10421. +        ActualCount = 0;
  10422. +        ansi.MaximumLength = 1;
  10423. +        goto out_copy;
  10424. +    }
  10425. +
  10426. +    /* query the number of bytes required for the utf8 encoding */
  10427. +    status = RtlUnicodeToUTF8N(NULL, 0xffff,
  10428. +        &ActualCount, str->Buffer, str->Length);
  10429. +    if (status) {
  10430. +        print_error("RtlUnicodeToUTF8N('%wZ') failed with 0x%08X\n",
  10431. +            str, status);
  10432. +        goto out;
  10433. +    }
  10434. +
  10435. +    /* convert the string directly into the upcall buffer */
  10436. +    ansi.Buffer = (PCHAR)*pos + sizeof(ansi.MaximumLength);
  10437. +    ansi.MaximumLength = (USHORT)ActualCount + sizeof(UNICODE_NULL);
  10438. +    status = RtlUnicodeToUTF8N(ansi.Buffer, ansi.MaximumLength,
  10439. +        &ActualCount, str->Buffer, str->Length);
  10440. +    if (status) {
  10441. +        print_error("RtlUnicodeToUTF8N(%hu, '%wZ', %hu) failed with 0x%08X\n",
  10442. +            ansi.MaximumLength, str, str->Length, status);
  10443. +        goto out;
  10444. +    }
  10445. +
  10446. +out_copy:
  10447. +    RtlCopyMemory(*pos, &ansi.MaximumLength, sizeof(ansi.MaximumLength));
  10448. +    *pos += sizeof(ansi.MaximumLength);
  10449. +    (*pos)[ActualCount] = '\0';
  10450. +    *pos += ansi.MaximumLength;
  10451. +out:
  10452. +    return status;
  10453. +}
  10454. +
  10455. +NTSTATUS marshal_nfs41_header(
  10456. +    nfs41_updowncall_entry *entry,
  10457. +    unsigned char *buf,
  10458. +    ULONG buf_len,
  10459. +    ULONG *len)
  10460. +{
  10461. +    NTSTATUS status = STATUS_SUCCESS;
  10462. +    ULONG header_len = 0;
  10463. +    unsigned char *tmp = buf;
  10464. +
  10465. +    header_len = sizeof(entry->version) + sizeof(entry->xid) +
  10466. +        sizeof(entry->opcode) + 2 * sizeof(HANDLE);
  10467. +    if (header_len > buf_len) {
  10468. +        status = STATUS_INSUFFICIENT_RESOURCES;
  10469. +        goto out;
  10470. +    }
  10471. +    else
  10472. +        *len = header_len;
  10473. +    RtlCopyMemory(tmp, &entry->version, sizeof(entry->version));
  10474. +    tmp += sizeof(entry->version);
  10475. +    RtlCopyMemory(tmp, &entry->xid, sizeof(entry->xid));
  10476. +    tmp += sizeof(entry->xid);
  10477. +    RtlCopyMemory(tmp, &entry->opcode, sizeof(entry->opcode));
  10478. +    tmp += sizeof(entry->opcode);
  10479. +    RtlCopyMemory(tmp, &entry->session, sizeof(HANDLE));
  10480. +    tmp += sizeof(HANDLE);
  10481. +    RtlCopyMemory(tmp, &entry->open_state, sizeof(HANDLE));
  10482. +    tmp += sizeof(HANDLE);
  10483. +
  10484. +    /*
  10485. +     * gisburn: FIXME: For currently unknown reasons we need to
  10486. +     * validate |entry->filename|+it's contents, because a heavily
  10487. +     * stressed system somehow sometimes causes garbage there
  10488. +     */
  10489. +    if (MmIsAddressValid(entry->filename) &&
  10490. +        (entry->filename != NULL) &&
  10491. +        MmIsAddressValid(entry->filename->Buffer)) {
  10492. +#ifdef DEBUG_MARSHAL_HEADER
  10493. +        DbgP("[upcall header] xid=%lld opcode='%s' filename='%wZ' version=%d "
  10494. +            "session=0x%x open_state=0x%x\n", entry->xid,
  10495. +            ENTRY_OPCODE2STRING(entry), entry->filename,
  10496. +            entry->version, entry->session, entry->open_state);
  10497. +#endif /* DEBUG_MARSHAL_HEADER */
  10498. +    }
  10499. +    else {
  10500. +        DbgP("[upcall header] Invalid filename 0x%p\n", entry);
  10501. +        status = STATUS_INTERNAL_ERROR;
  10502. +    }
  10503. +out:
  10504. +    return status;
  10505. +}
  10506. +
  10507. +NTSTATUS marshal_nfs41_shutdown(
  10508. +    nfs41_updowncall_entry *entry,
  10509. +    unsigned char *buf,
  10510. +    ULONG buf_len,
  10511. +    ULONG *len)
  10512. +{
  10513. +    return marshal_nfs41_header(entry, buf, buf_len, len);
  10514. +}
  10515. +
  10516. +NTSTATUS nfs41_invalidate_cache(
  10517. +    IN PRX_CONTEXT RxContext)
  10518. +{
  10519. +    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  10520. +    unsigned char *buf = LowIoContext->ParamsFor.IoCtl.pInputBuffer;
  10521. +    ULONG flag = DISABLE_CACHING;
  10522. +    PMRX_SRV_OPEN srv_open;
  10523. +    NTSTATUS status;
  10524. +
  10525. +    RtlCopyMemory(&srv_open, buf, sizeof(HANDLE));
  10526. +#ifdef DEBUG_INVALIDATE_CACHE
  10527. +    DbgP("nfs41_invalidate_cache: received srv_open=0x%p '%wZ'\n",
  10528. +        srv_open, srv_open->pAlreadyPrefixedName);
  10529. +#endif
  10530. +    if (MmIsAddressValid(srv_open)) {
  10531. +        RxIndicateChangeOfBufferingStateForSrvOpen(
  10532. +            srv_open->pFcb->pNetRoot->pSrvCall, srv_open,
  10533. +            srv_open->Key, ULongToPtr(flag));
  10534. +        status = STATUS_SUCCESS;
  10535. +    }
  10536. +    else {
  10537. +        print_error("nfs41_invalidate_cache: "
  10538. +            "invalid ptr srv_open=0x%p file='%wZ'\n",
  10539. +            srv_open, srv_open->pAlreadyPrefixedName);
  10540. +        status = STATUS_INVALID_HANDLE;
  10541. +    }
  10542. +
  10543. +    return status;
  10544. +}
  10545. +
  10546. +NTSTATUS nfs41_shutdown_daemon(
  10547. +    DWORD version)
  10548. +{
  10549. +    NTSTATUS status = STATUS_SUCCESS;
  10550. +    nfs41_updowncall_entry *entry = NULL;
  10551. +
  10552. +    DbgEn();
  10553. +    status = nfs41_UpcallCreate(NFS41_SHUTDOWN, NULL, INVALID_HANDLE_VALUE,
  10554. +        INVALID_HANDLE_VALUE, version, NULL, &entry);
  10555. +    if (status) goto out;
  10556. +
  10557. +    status = nfs41_UpcallWaitForReply(entry, UPCALL_TIMEOUT_DEFAULT);
  10558. +    if (entry->psec_ctx == &entry->sec_ctx) {
  10559. +        SeDeleteClientSecurity(entry->psec_ctx);
  10560. +    }
  10561. +    entry->psec_ctx = NULL;
  10562. +    if (status) goto out;
  10563. +
  10564. +    nfs41_UpcallDestroy(entry);
  10565. +out:
  10566. +    DbgEx();
  10567. +    return status;
  10568. +}
  10569. +
  10570. +NTSTATUS SharedMemoryInit(
  10571. +    OUT PHANDLE phSection)
  10572. +{
  10573. +    NTSTATUS status;
  10574. +    HANDLE hSection;
  10575. +    UNICODE_STRING SectionName;
  10576. +    SECURITY_DESCRIPTOR SecurityDesc;
  10577. +    OBJECT_ATTRIBUTES SectionAttrs;
  10578. +    LARGE_INTEGER nSectionSize;
  10579. +
  10580. +    DbgEn();
  10581. +
  10582. +    RtlInitUnicodeString(&SectionName, NFS41_SHARED_MEMORY_NAME);
  10583. +
  10584. +    /* XXX: setting dacl=NULL grants access to everyone */
  10585. +    status = RtlCreateSecurityDescriptor(&SecurityDesc,
  10586. +        SECURITY_DESCRIPTOR_REVISION);
  10587. +    if (status) {
  10588. +        print_error("RtlCreateSecurityDescriptor() failed with %08X\n", status);
  10589. +        goto out;
  10590. +    }
  10591. +    status = RtlSetDaclSecurityDescriptor(&SecurityDesc, TRUE, NULL, FALSE);
  10592. +    if (status) {
  10593. +        print_error("RtlSetDaclSecurityDescriptor() failed with %08X\n", status);
  10594. +        goto out;
  10595. +    }
  10596. +
  10597. +    InitializeObjectAttributes(&SectionAttrs, &SectionName,
  10598. +        0, NULL, &SecurityDesc);
  10599. +
  10600. +    nSectionSize.QuadPart = sizeof(NFS41NP_SHARED_MEMORY);
  10601. +
  10602. +    status = ZwCreateSection(&hSection, SECTION_MAP_READ | SECTION_MAP_WRITE,
  10603. +        &SectionAttrs, &nSectionSize, PAGE_READWRITE, SEC_COMMIT, NULL);
  10604. +    switch (status) {
  10605. +    case STATUS_SUCCESS:
  10606. +        break;
  10607. +    case STATUS_OBJECT_NAME_COLLISION:
  10608. +        DbgP("section already created; returning success\n");
  10609. +        status = STATUS_SUCCESS;
  10610. +        goto out;
  10611. +    default:
  10612. +        DbgP("ZwCreateSection failed with %08X\n", status);
  10613. +        goto out;
  10614. +    }
  10615. +out:
  10616. +    DbgEx();
  10617. +    return status;
  10618. +}
  10619. +
  10620. +NTSTATUS SharedMemoryFree(
  10621. +    IN HANDLE hSection)
  10622. +{
  10623. +    NTSTATUS status;
  10624. +    DbgEn();
  10625. +    status = ZwClose(hSection);
  10626. +    DbgEx();
  10627. +    return status;
  10628. +}
  10629. +
  10630. +NTSTATUS nfs41_Start(
  10631. +    IN OUT PRX_CONTEXT RxContext,
  10632. +    IN OUT PRDBSS_DEVICE_OBJECT dev)
  10633. +{
  10634. +    NTSTATUS status;
  10635. +    NFS41GetDeviceExtension(RxContext, DevExt);
  10636. +
  10637. +    DbgEn();
  10638. +
  10639. +    status = SharedMemoryInit(&DevExt->SharedMemorySection);
  10640. +    if (status) {
  10641. +        print_error("InitSharedMemory failed with %08X\n", status);
  10642. +        status = STATUS_INSUFFICIENT_RESOURCES;
  10643. +        goto out;
  10644. +    }
  10645. +
  10646. +    InterlockedCompareExchange((PLONG)&nfs41_start_state,
  10647. +        NFS41_START_DRIVER_STARTED,
  10648. +        NFS41_START_DRIVER_START_IN_PROGRESS);
  10649. +out:
  10650. +    DbgEx();
  10651. +    return status;
  10652. +}
  10653. +
  10654. +NTSTATUS nfs41_Stop(
  10655. +    IN OUT PRX_CONTEXT RxContext,
  10656. +    IN OUT PRDBSS_DEVICE_OBJECT dev)
  10657. +{
  10658. +    NTSTATUS status;
  10659. +    NFS41GetDeviceExtension(RxContext, DevExt);
  10660. +    DbgEn();
  10661. +    status = SharedMemoryFree(DevExt->SharedMemorySection);
  10662. +    DbgEx();
  10663. +    return status;
  10664. +}
  10665. +
  10666. +#ifdef ENABLE_TIMINGS
  10667. +static void print_op_stat(
  10668. +    const char *op_str,
  10669. +    nfs41_timings *time, BOOLEAN clear)
  10670. +{
  10671. +    DbgP("%-9s: num_ops=%-10d delta_ticks=%-10d size=%-10d\n", op_str,
  10672. +        time->tops, time->tops ? time->ticks/time->tops : 0,
  10673. +        time->sops ? time->size/time->sops : 0);
  10674. +    if (clear) {
  10675. +        time->tops = 0;
  10676. +        time->ticks = 0;
  10677. +        time->size = 0;
  10678. +        time->sops = 0;
  10679. +    }
  10680. +}
  10681. +#endif
  10682. +
  10683. +NTSTATUS nfs41_DevFcbXXXControlFile(
  10684. +    IN OUT PRX_CONTEXT RxContext)
  10685. +{
  10686. +    NTSTATUS status = STATUS_INVALID_DEVICE_REQUEST;
  10687. +    UCHAR op = RxContext->MajorFunction;
  10688. +    PLOWIO_CONTEXT io_ctx = &RxContext->LowIoContext;
  10689. +    ULONG fsop = io_ctx->ParamsFor.FsCtl.FsControlCode, state;
  10690. +    ULONG in_len = io_ctx->ParamsFor.IoCtl.InputBufferLength;
  10691. +    DWORD *buf = io_ctx->ParamsFor.IoCtl.pInputBuffer;
  10692. +    NFS41GetDeviceExtension(RxContext, DevExt);
  10693. +    DWORD nfs41d_version = 0;
  10694. +
  10695. +    //DbgEn();
  10696. +
  10697. +    //print_ioctl(op);
  10698. +    switch(op) {
  10699. +    case IRP_MJ_FILE_SYSTEM_CONTROL:
  10700. +        status = STATUS_INVALID_DEVICE_REQUEST;
  10701. +        break;
  10702. +    case IRP_MJ_DEVICE_CONTROL:
  10703. +    case IRP_MJ_INTERNAL_DEVICE_CONTROL:
  10704. +        //print_fs_ioctl(fsop);
  10705. +        switch (fsop) {
  10706. +        case IOCTL_NFS41_INVALCACHE:
  10707. +            status = nfs41_invalidate_cache(RxContext);
  10708. +            break;
  10709. +        case IOCTL_NFS41_READ:
  10710. +            status = nfs41_upcall(RxContext);
  10711. +            break;
  10712. +        case IOCTL_NFS41_WRITE:
  10713. +            status = nfs41_downcall(RxContext);
  10714. +            break;
  10715. +        case IOCTL_NFS41_ADDCONN:
  10716. +            status = nfs41_CreateConnection(RxContext, &RxContext->PostRequest);
  10717. +            break;
  10718. +        case IOCTL_NFS41_DELCONN:
  10719. +            if (RxContext->RxDeviceObject->NumberOfActiveFcbs > 0) {
  10720. +                DbgP("device has open handles %d\n",
  10721. +                    RxContext->RxDeviceObject->NumberOfActiveFcbs);
  10722. +                status = STATUS_REDIRECTOR_HAS_OPEN_HANDLES;
  10723. +                break;
  10724. +            }
  10725. +            status = nfs41_DeleteConnection(RxContext, &RxContext->PostRequest);
  10726. +            break;
  10727. +        case IOCTL_NFS41_GETSTATE:
  10728. +            state = RDR_NULL_STATE;
  10729. +
  10730. +            if (io_ctx->ParamsFor.IoCtl.OutputBufferLength >= sizeof(ULONG)) {
  10731. +                // map the states to control app's equivalents
  10732. +                print_driver_state(nfs41_start_state);
  10733. +                switch (nfs41_start_state) {
  10734. +                case NFS41_START_DRIVER_STARTABLE:
  10735. +                case NFS41_START_DRIVER_STOPPED:
  10736. +                    state = RDR_STOPPED;
  10737. +                    break;
  10738. +                case NFS41_START_DRIVER_START_IN_PROGRESS:
  10739. +                    state = RDR_STARTING;
  10740. +                    break;
  10741. +                case NFS41_START_DRIVER_STARTED:
  10742. +                    state = RDR_STARTED;
  10743. +                    break;
  10744. +                }
  10745. +                *(ULONG *)io_ctx->ParamsFor.IoCtl.pOutputBuffer = state;
  10746. +                RxContext->InformationToReturn = sizeof(ULONG);
  10747. +                status = STATUS_SUCCESS;
  10748. +            } else
  10749. +                status = STATUS_INVALID_PARAMETER;
  10750. +            break;
  10751. +        case IOCTL_NFS41_START:
  10752. +            print_driver_state(nfs41_start_state);
  10753. +            if (in_len >= sizeof(DWORD)) {
  10754. +                RtlCopyMemory(&nfs41d_version, buf, sizeof(DWORD));
  10755. +                DbgP("NFS41 Daemon sent start request with version %d\n",
  10756. +                    nfs41d_version);
  10757. +                DbgP("Currently used NFS41 Daemon version is %d\n",
  10758. +                    DevExt->nfs41d_version);
  10759. +                DevExt->nfs41d_version = nfs41d_version;
  10760. +            }
  10761. +            switch(nfs41_start_state) {
  10762. +            case NFS41_START_DRIVER_STARTABLE:
  10763. +                (nfs41_start_driver_state)InterlockedCompareExchange(
  10764. +                              (PLONG)&nfs41_start_state,
  10765. +                              NFS41_START_DRIVER_START_IN_PROGRESS,
  10766. +                              NFS41_START_DRIVER_STARTABLE);
  10767. +                    //lack of break is intentional
  10768. +            case NFS41_START_DRIVER_START_IN_PROGRESS:
  10769. +                status = RxStartMinirdr(RxContext, &RxContext->PostRequest);
  10770. +                if (status == STATUS_REDIRECTOR_STARTED) {
  10771. +                    DbgP("redirector started\n");
  10772. +                    status = STATUS_SUCCESS;
  10773. +                } else if (status == STATUS_PENDING &&
  10774. +                            RxContext->PostRequest == TRUE) {
  10775. +                    DbgP("RxStartMinirdr pending 0x%08lx\n", status);
  10776. +                    status = STATUS_MORE_PROCESSING_REQUIRED;
  10777. +                }
  10778. +                break;
  10779. +            case NFS41_START_DRIVER_STARTED:
  10780. +                status = STATUS_SUCCESS;
  10781. +                break;
  10782. +            default:
  10783. +                status = STATUS_INVALID_PARAMETER;
  10784. +            }
  10785. +            break;
  10786. +        case IOCTL_NFS41_STOP:
  10787. +            if (nfs41_start_state == NFS41_START_DRIVER_STARTED)
  10788. +                nfs41_shutdown_daemon(DevExt->nfs41d_version);
  10789. +            if (RxContext->RxDeviceObject->NumberOfActiveFcbs > 0) {
  10790. +                DbgP("device has open handles %d\n",
  10791. +                    RxContext->RxDeviceObject->NumberOfActiveFcbs);
  10792. +                status = STATUS_REDIRECTOR_HAS_OPEN_HANDLES;
  10793. +                break;
  10794. +            }
  10795. +
  10796. +            state = (nfs41_start_driver_state)InterlockedCompareExchange(
  10797. +                        (PLONG)&nfs41_start_state,
  10798. +                        NFS41_START_DRIVER_STARTABLE,
  10799. +                        NFS41_START_DRIVER_STARTED);
  10800. +
  10801. +            status = RxStopMinirdr(RxContext, &RxContext->PostRequest);
  10802. +            DbgP("RxStopMinirdr status 0x%08lx\n", status);
  10803. +            if (status == STATUS_PENDING && RxContext->PostRequest == TRUE )
  10804. +                status = STATUS_MORE_PROCESSING_REQUIRED;
  10805. +            break;
  10806. +        default:
  10807. +            status = STATUS_INVALID_DEVICE_REQUEST;
  10808. +        };
  10809. +        break;
  10810. +    default:
  10811. +        status = STATUS_INVALID_DEVICE_REQUEST;
  10812. +    };
  10813. +
  10814. +    //DbgEx();
  10815. +    return status;
  10816. +}
  10817. +
  10818. +NTSTATUS _nfs41_CreateSrvCall(
  10819. +    PMRX_SRVCALL_CALLBACK_CONTEXT pCallbackContext)
  10820. +{
  10821. +    NTSTATUS status = STATUS_SUCCESS;
  10822. +    PMRX_SRVCALL_CALLBACK_CONTEXT SCCBC = pCallbackContext;
  10823. +    PMRX_SRV_CALL pSrvCall;
  10824. +    PMRX_SRVCALLDOWN_STRUCTURE SrvCalldownStructure =
  10825. +        (PMRX_SRVCALLDOWN_STRUCTURE)(SCCBC->SrvCalldownStructure);
  10826. +    PNFS41_SERVER_ENTRY pServerEntry = NULL;
  10827. +
  10828. +#ifdef DEBUG_MOUNT
  10829. +    DbgEn();
  10830. +#endif
  10831. +
  10832. +    pSrvCall = SrvCalldownStructure->SrvCall;
  10833. +
  10834. +    ASSERT( pSrvCall );
  10835. +    ASSERT( NodeType(pSrvCall) == RDBSS_NTC_SRVCALL );
  10836. +    // print_srv_call(pSrvCall);
  10837. +
  10838. +    // validate the server name with the test name of 'pnfs'
  10839. +#ifdef DEBUG_MOUNT
  10840. +    DbgP("SrvCall: Connection Name Length: %d '%wZ'\n",
  10841. +        pSrvCall->pSrvCallName->Length, pSrvCall->pSrvCallName);
  10842. +#endif
  10843. +
  10844. +    if (pSrvCall->pSrvCallName->Length > SERVER_NAME_BUFFER_SIZE) {
  10845. +        print_error("Server name '%wZ' too long for server entry (max %u)\n",
  10846. +            pSrvCall->pSrvCallName, SERVER_NAME_BUFFER_SIZE);
  10847. +        status = STATUS_NAME_TOO_LONG;
  10848. +        goto out;
  10849. +    }
  10850. +
  10851. +    /* Let's create our own representation of the server */
  10852. +    pServerEntry = (PNFS41_SERVER_ENTRY)RxAllocatePoolWithTag(NonPagedPoolNx,
  10853. +        sizeof(NFS41_SERVER_ENTRY), NFS41_MM_POOLTAG);
  10854. +    if (pServerEntry == NULL) {
  10855. +        status = STATUS_INSUFFICIENT_RESOURCES;
  10856. +        goto out;
  10857. +    }
  10858. +    RtlZeroMemory(pServerEntry, sizeof(NFS41_SERVER_ENTRY));
  10859. +
  10860. +    pServerEntry->Name.Buffer = pServerEntry->NameBuffer;
  10861. +    pServerEntry->Name.Length = pSrvCall->pSrvCallName->Length;
  10862. +    pServerEntry->Name.MaximumLength = SERVER_NAME_BUFFER_SIZE;
  10863. +    RtlCopyMemory(pServerEntry->Name.Buffer, pSrvCall->pSrvCallName->Buffer,
  10864. +        pServerEntry->Name.Length);
  10865. +
  10866. +    pCallbackContext->RecommunicateContext = pServerEntry;
  10867. +    InterlockedExchangePointer(&pServerEntry->pRdbssSrvCall, pSrvCall);
  10868. +
  10869. +out:
  10870. +    SCCBC->Status = status;
  10871. +    SrvCalldownStructure->CallBack(SCCBC);
  10872. +
  10873. +#ifdef DEBUG_MOUNT
  10874. +    DbgEx();
  10875. +#endif
  10876. +    return status;
  10877. +}
  10878. +
  10879. +NTSTATUS nfs41_CreateSrvCall(
  10880. +    PMRX_SRV_CALL pSrvCall,
  10881. +    PMRX_SRVCALL_CALLBACK_CONTEXT pCallbackContext)
  10882. +{
  10883. +    NTSTATUS status;
  10884. +
  10885. +    ASSERT( pSrvCall );
  10886. +    ASSERT( NodeType(pSrvCall) == RDBSS_NTC_SRVCALL );
  10887. +
  10888. +    if (IoGetCurrentProcess() == RxGetRDBSSProcess()) {
  10889. +        DbgP("executing with RDBSS context\n");
  10890. +        status = _nfs41_CreateSrvCall(pCallbackContext);
  10891. +    } else {
  10892. +        status = RxDispatchToWorkerThread(nfs41_dev, DelayedWorkQueue,
  10893. +           (PRX_WORKERTHREAD_ROUTINE)_nfs41_CreateSrvCall, pCallbackContext);
  10894. +        if (status != STATUS_SUCCESS) {
  10895. +            print_error("RxDispatchToWorkerThread returned status 0x%08lx\n",
  10896. +                status);
  10897. +            pCallbackContext->Status = status;
  10898. +            pCallbackContext->SrvCalldownStructure->CallBack(pCallbackContext);
  10899. +            status = STATUS_PENDING;
  10900. +        }
  10901. +    }
  10902. +    /* RDBSS expects MRxCreateSrvCall to return STATUS_PENDING */
  10903. +    if (status == STATUS_SUCCESS)
  10904. +        status = STATUS_PENDING;
  10905. +
  10906. +    return status;
  10907. +}
  10908. +
  10909. +NTSTATUS nfs41_SrvCallWinnerNotify(
  10910. +    IN OUT PMRX_SRV_CALL pSrvCall,
  10911. +    IN BOOLEAN ThisMinirdrIsTheWinner,
  10912. +    IN OUT PVOID pSrvCallContext)
  10913. +{
  10914. +    NTSTATUS status = STATUS_SUCCESS;
  10915. +    PNFS41_SERVER_ENTRY pServerEntry;
  10916. +
  10917. +    pServerEntry = (PNFS41_SERVER_ENTRY)pSrvCallContext;
  10918. +
  10919. +    if (!ThisMinirdrIsTheWinner) {
  10920. +        ASSERT(1);
  10921. +        goto out;
  10922. +    }
  10923. +
  10924. +    pSrvCall->Context = pServerEntry;
  10925. +out:
  10926. +    return status;
  10927. +}
  10928. +
  10929. +static ULONG nfs41_ExtendForCache(
  10930. +    IN OUT PRX_CONTEXT RxContext,
  10931. +    IN PLARGE_INTEGER pNewFileSize,
  10932. +    OUT PLARGE_INTEGER pNewAllocationSize)
  10933. +{
  10934. +    NTSTATUS status = STATUS_SUCCESS;
  10935. +    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  10936. +#ifdef DEBUG_CACHE
  10937. +    PLOWIO_CONTEXT LowIoContext  = &RxContext->LowIoContext;
  10938. +    DbgEn();
  10939. +    print_debug_header(RxContext);
  10940. +    DbgP("input: byte count 0x%x filesize 0x%x alloc size 0x%x\n",
  10941. +        LowIoContext->ParamsFor.ReadWrite.ByteCount, *pNewFileSize,
  10942. +        *pNewAllocationSize);
  10943. +#endif
  10944. +    pNewAllocationSize->QuadPart = pNewFileSize->QuadPart + 8192;
  10945. +    nfs41_fcb->StandardInfo.AllocationSize.QuadPart =
  10946. +        pNewAllocationSize->QuadPart;
  10947. +    nfs41_fcb->StandardInfo.EndOfFile.QuadPart = pNewFileSize->QuadPart;
  10948. +#ifdef DEBUG_CACHE
  10949. +    DbgP("new filesize 0x%x new allocation size 0x%x\n",
  10950. +        *pNewFileSize, *pNewAllocationSize);
  10951. +#endif
  10952. +#ifdef DEBUG_CACHE
  10953. +    DbgEx();
  10954. +#endif
  10955. +    return status;
  10956. +}
  10957. +
  10958. +VOID nfs41_remove_fcb_entry(
  10959. +    PMRX_FCB fcb)
  10960. +{
  10961. +    PLIST_ENTRY pEntry;
  10962. +    nfs41_fcb_list_entry *cur;
  10963. +    ExAcquireFastMutex(&fcblistLock);
  10964. +
  10965. +    pEntry = openlist.head.Flink;
  10966. +    while (!IsListEmpty(&openlist.head)) {
  10967. +        cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry,
  10968. +                nfs41_fcb_list_entry, next);
  10969. +        if (cur->fcb == fcb) {
  10970. +#ifdef DEBUG_CLOSE
  10971. +            DbgP("nfs41_remove_fcb_entry: Found match for fcb=0x%p\n", fcb);
  10972. +#endif
  10973. +            RemoveEntryList(pEntry);
  10974. +            RxFreePool(cur);
  10975. +            break;
  10976. +        }
  10977. +        if (pEntry->Flink == &openlist.head) {
  10978. +#ifdef DEBUG_CLOSE
  10979. +            DbgP("nfs41_remove_fcb_entry: reached EOL looking "
  10980. +                "for fcb 0x%p\n", fcb);
  10981. +#endif
  10982. +            break;
  10983. +        }
  10984. +        pEntry = pEntry->Flink;
  10985. +    }
  10986. +    ExReleaseFastMutex(&fcblistLock);
  10987. +}
  10988. +
  10989. +static
  10990. +VOID nfs41_invalidate_fobx_entry(
  10991. +    IN OUT PMRX_FOBX pFobx)
  10992. +{
  10993. +    PLIST_ENTRY pEntry;
  10994. +    nfs41_fcb_list_entry *cur;
  10995. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(pFobx);
  10996. +
  10997. +    ExAcquireFastMutex(&fcblistLock);
  10998. +
  10999. +    pEntry = openlist.head.Flink;
  11000. +    while (!IsListEmpty(&openlist.head)) {
  11001. +        cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry,
  11002. +                nfs41_fcb_list_entry, next);
  11003. +        if (cur->nfs41_fobx == nfs41_fobx) {
  11004. +#ifdef DEBUG_CLOSE
  11005. +            DbgP("nfs41_invalidate_fobx_entry: Found match for fobx=0x%p\n", fobx);
  11006. +#endif
  11007. +            cur->nfs41_fobx = NULL;
  11008. +            break;
  11009. +        }
  11010. +        if (pEntry->Flink == &openlist.head) {
  11011. +#ifdef DEBUG_CLOSE
  11012. +            DbgP("nfs41_invalidate_fobx_entry: reached EOL looking "
  11013. +                "for fobx 0x%p\n", fobx);
  11014. +#endif
  11015. +            break;
  11016. +        }
  11017. +        pEntry = pEntry->Flink;
  11018. +    }
  11019. +    ExReleaseFastMutex(&fcblistLock);
  11020. +}
  11021. +
  11022. +NTSTATUS nfs41_Flush(
  11023. +    IN OUT PRX_CONTEXT RxContext)
  11024. +{
  11025. +    DbgP("nfs41_Flush: FileName='%wZ'\n",
  11026. +        GET_ALREADY_PREFIXED_NAME_FROM_CONTEXT(RxContext));
  11027. +
  11028. +    return STATUS_SUCCESS;
  11029. +}
  11030. +
  11031. +NTSTATUS nfs41_DeallocateForFcb(
  11032. +    IN OUT PMRX_FCB pFcb)
  11033. +{
  11034. +    nfs41_remove_fcb_entry(pFcb);
  11035. +    return STATUS_SUCCESS;
  11036. +}
  11037. +
  11038. +NTSTATUS nfs41_DeallocateForFobx(
  11039. +    IN OUT PMRX_FOBX pFobx)
  11040. +{
  11041. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(pFobx);
  11042. +
  11043. +    nfs41_invalidate_fobx_entry(pFobx);
  11044. +
  11045. +    if (nfs41_fobx->acl) {
  11046. +        RxFreePool(nfs41_fobx->acl);
  11047. +        nfs41_fobx->acl = NULL;
  11048. +    }
  11049. +
  11050. +    if (nfs41_fobx->sec_ctx.ClientToken) {
  11051. +        SeDeleteClientSecurity(&nfs41_fobx->sec_ctx);
  11052. +        nfs41_fobx->sec_ctx.ClientToken = NULL;
  11053. +    }
  11054. +
  11055. +    return STATUS_SUCCESS;
  11056. +}
  11057. +
  11058. +VOID nfs41_update_fcb_list(
  11059. +    PMRX_FCB fcb,
  11060. +    ULONGLONG ChangeTime)
  11061. +{
  11062. +    PLIST_ENTRY pEntry;
  11063. +    nfs41_fcb_list_entry *cur;
  11064. +    ExAcquireFastMutex(&fcblistLock);
  11065. +    pEntry = openlist.head.Flink;
  11066. +    while (!IsListEmpty(&openlist.head)) {
  11067. +        cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry,
  11068. +                nfs41_fcb_list_entry, next);
  11069. +        if (cur->fcb == fcb &&
  11070. +                cur->ChangeTime != ChangeTime) {
  11071. +#if defined(DEBUG_FILE_SET) || defined(DEBUG_ACL_SET) || \
  11072. +    defined(DEBUG_WRITE) || defined(DEBUG_EA_SET)
  11073. +            DbgP("nfs41_update_fcb_list: Found match for fcb 0x%p: "
  11074. +                "updating %llu to %llu\n",
  11075. +                fcb, cur->ChangeTime, ChangeTime);
  11076. +#endif
  11077. +            cur->ChangeTime = ChangeTime;
  11078. +            break;
  11079. +        }
  11080. +        /* place an upcall for this srv_open */
  11081. +        if (pEntry->Flink == &openlist.head) {
  11082. +#if defined(DEBUG_FILE_SET) || defined(DEBUG_ACL_SET) || \
  11083. +    defined(DEBUG_WRITE) || defined(DEBUG_EA_SET)
  11084. +            DbgP("nfs41_update_fcb_list: reached EOL loooking for "
  11085. +                "fcb=0x%p\n", fcb);
  11086. +#endif
  11087. +            break;
  11088. +        }
  11089. +        pEntry = pEntry->Flink;
  11090. +    }
  11091. +    ExReleaseFastMutex(&fcblistLock);
  11092. +}
  11093. +
  11094. +NTSTATUS nfs41_IsValidDirectory (
  11095. +    IN OUT PRX_CONTEXT RxContext,
  11096. +    IN PUNICODE_STRING DirectoryName)
  11097. +{
  11098. +    return STATUS_SUCCESS;
  11099. +}
  11100. +
  11101. +NTSTATUS nfs41_ComputeNewBufferingState(
  11102. +    IN OUT PMRX_SRV_OPEN pSrvOpen,
  11103. +    IN PVOID pMRxContext,
  11104. +    OUT ULONG *pNewBufferingState)
  11105. +{
  11106. +    NTSTATUS status = STATUS_SUCCESS;
  11107. +    ULONG flag = PtrToUlong(pMRxContext);
  11108. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11109. +    ULONG oldFlags = pSrvOpen->BufferingFlags;
  11110. +#endif
  11111. +
  11112. +    switch(flag) {
  11113. +    case DISABLE_CACHING:
  11114. +        if (pSrvOpen->BufferingFlags &
  11115. +            (FCB_STATE_READBUFFERING_ENABLED | FCB_STATE_READCACHING_ENABLED))
  11116. +            pSrvOpen->BufferingFlags &=
  11117. +                ~(FCB_STATE_READBUFFERING_ENABLED |
  11118. +                  FCB_STATE_READCACHING_ENABLED);
  11119. +        if (pSrvOpen->BufferingFlags &
  11120. +            (FCB_STATE_WRITECACHING_ENABLED | FCB_STATE_WRITEBUFFERING_ENABLED))
  11121. +            pSrvOpen->BufferingFlags &=
  11122. +                ~(FCB_STATE_WRITECACHING_ENABLED |
  11123. +                  FCB_STATE_WRITEBUFFERING_ENABLED);
  11124. +        pSrvOpen->BufferingFlags |= FCB_STATE_DISABLE_LOCAL_BUFFERING;
  11125. +        break;
  11126. +    case ENABLE_READ_CACHING:
  11127. +        pSrvOpen->BufferingFlags |=
  11128. +            (FCB_STATE_READBUFFERING_ENABLED | FCB_STATE_READCACHING_ENABLED);
  11129. +        break;
  11130. +    case ENABLE_WRITE_CACHING:
  11131. +        pSrvOpen->BufferingFlags |=
  11132. +            (FCB_STATE_WRITECACHING_ENABLED | FCB_STATE_WRITEBUFFERING_ENABLED);
  11133. +        break;
  11134. +    case ENABLE_READWRITE_CACHING:
  11135. +        pSrvOpen->BufferingFlags =
  11136. +            (FCB_STATE_READBUFFERING_ENABLED | FCB_STATE_READCACHING_ENABLED |
  11137. +            FCB_STATE_WRITECACHING_ENABLED | FCB_STATE_WRITEBUFFERING_ENABLED);
  11138. +    }
  11139. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11140. +    DbgP("nfs41_ComputeNewBufferingState: '%wZ' pSrvOpen 0x%p Old %08x New %08x\n",
  11141. +         pSrvOpen->pAlreadyPrefixedName, pSrvOpen, oldFlags,
  11142. +         pSrvOpen->BufferingFlags);
  11143. +    *pNewBufferingState = pSrvOpen->BufferingFlags;
  11144. +#endif
  11145. +    return status;
  11146. +}
  11147. +
  11148. +void enable_caching(
  11149. +    PMRX_SRV_OPEN SrvOpen,
  11150. +    PNFS41_FOBX nfs41_fobx,
  11151. +    ULONGLONG ChangeTime,
  11152. +    HANDLE session)
  11153. +{
  11154. +    ULONG flag = 0;
  11155. +    PLIST_ENTRY pEntry;
  11156. +    nfs41_fcb_list_entry *cur;
  11157. +    BOOLEAN found = FALSE;
  11158. +
  11159. +    if (SrvOpen->DesiredAccess & FILE_READ_DATA)
  11160. +        flag = ENABLE_READ_CACHING;
  11161. +    if ((SrvOpen->DesiredAccess & FILE_WRITE_DATA) &&
  11162. +            !nfs41_fobx->write_thru)
  11163. +        flag = ENABLE_WRITE_CACHING;
  11164. +    if ((SrvOpen->DesiredAccess & FILE_READ_DATA) &&
  11165. +            (SrvOpen->DesiredAccess & FILE_WRITE_DATA) &&
  11166. +            !nfs41_fobx->write_thru)
  11167. +        flag = ENABLE_READWRITE_CACHING;
  11168. +
  11169. +#if defined(DEBUG_TIME_BASED_COHERENCY) || \
  11170. +        defined(DEBUG_WRITE) || defined(DEBUG_READ)
  11171. +    print_caching_level(1, flag, SrvOpen->pAlreadyPrefixedName);
  11172. +#endif
  11173. +
  11174. +    if (!flag)
  11175. +        return;
  11176. +
  11177. +    RxChangeBufferingState((PSRV_OPEN)SrvOpen, ULongToPtr(flag), 1);
  11178. +
  11179. +    ExAcquireFastMutex(&fcblistLock);
  11180. +    pEntry = openlist.head.Flink;
  11181. +    while (!IsListEmpty(&openlist.head)) {
  11182. +        cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry,
  11183. +                nfs41_fcb_list_entry, next);
  11184. +        if (cur->fcb == SrvOpen->pFcb) {
  11185. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11186. +            DbgP("enable_caching: Looked&Found match for fcb=0x%p '%wZ'\n",
  11187. +                SrvOpen->pFcb, SrvOpen->pAlreadyPrefixedName);
  11188. +#endif
  11189. +            cur->skip = FALSE;
  11190. +            found = TRUE;
  11191. +            break;
  11192. +        }
  11193. +        if (pEntry->Flink == &openlist.head) {
  11194. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11195. +            DbgP("enable_caching: reached EOL looking for fcb=0x%p '%wZ'\n",
  11196. +                SrvOpen->pFcb, SrvOpen->pAlreadyPrefixedName);
  11197. +#endif
  11198. +            break;
  11199. +        }
  11200. +        pEntry = pEntry->Flink;
  11201. +    }
  11202. +    if (!found && nfs41_fobx->deleg_type) {
  11203. +        nfs41_fcb_list_entry *oentry;
  11204. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11205. +        DbgP("enable_caching: delegation recalled: srv_open=0x%p\n", SrvOpen);
  11206. +#endif
  11207. +        oentry = RxAllocatePoolWithTag(NonPagedPoolNx,
  11208. +            sizeof(nfs41_fcb_list_entry), NFS41_MM_POOLTAG_OPEN);
  11209. +        if (oentry == NULL) return;
  11210. +        oentry->fcb = SrvOpen->pFcb;
  11211. +        oentry->session = session;
  11212. +        oentry->nfs41_fobx = nfs41_fobx;
  11213. +        oentry->ChangeTime = ChangeTime;
  11214. +        oentry->skip = FALSE;
  11215. +        InsertTailList(&openlist.head, &oentry->next);
  11216. +        nfs41_fobx->deleg_type = 0;
  11217. +    }
  11218. +    ExReleaseFastMutex(&fcblistLock);
  11219. +}
  11220. +
  11221. +NTSTATUS nfs41_CompleteBufferingStateChangeRequest(
  11222. +    IN OUT PRX_CONTEXT RxContext,
  11223. +    IN OUT PMRX_SRV_OPEN SrvOpen,
  11224. +    IN PVOID pContext)
  11225. +{
  11226. +    return STATUS_SUCCESS;
  11227. +}
  11228. +
  11229. +/* |nfs41_FsdDispatch()| - must be public symbol */
  11230. +NTSTATUS nfs41_FsdDispatch(
  11231. +    IN PDEVICE_OBJECT dev,
  11232. +    IN PIRP Irp)
  11233. +{
  11234. +#ifdef DEBUG_FSDDISPATCH
  11235. +    PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation(Irp);
  11236. +#endif
  11237. +    NTSTATUS status;
  11238. +
  11239. +#ifdef DEBUG_FSDDISPATCH
  11240. +    DbgEn();
  11241. +    DbgP("CURRENT IRP = %d.%d\n", IrpSp->MajorFunction, IrpSp->MinorFunction);
  11242. +    if(IrpSp->FileObject)
  11243. +        DbgP("FileOject 0x%p Filename '%wZ'\n", IrpSp->FileObject,
  11244. +                &IrpSp->FileObject->FileName);
  11245. +#endif
  11246. +
  11247. +    if (dev != (PDEVICE_OBJECT)nfs41_dev) {
  11248. +        print_error("*** not ours ***\n");
  11249. +        Irp->IoStatus.Status = STATUS_INVALID_DEVICE_REQUEST;
  11250. +        Irp->IoStatus.Information = 0;
  11251. +        IoCompleteRequest(Irp, IO_NO_INCREMENT );
  11252. +        status = STATUS_INVALID_DEVICE_REQUEST;
  11253. +        goto out;
  11254. +    }
  11255. +
  11256. +    status = RxFsdDispatch((PRDBSS_DEVICE_OBJECT)dev,Irp);
  11257. +    /* AGLO: 08/05/2009 - looks like RxFsdDispatch frees IrpSp */
  11258. +
  11259. +out:
  11260. +#ifdef DEBUG_FSDDISPATCH
  11261. +    DbgP("IoStatus status = 0x%lx info = 0x%x\n",
  11262. +        (long)Irp->IoStatus.Status,
  11263. +        Irp->IoStatus.Information);
  11264. +    DbgEx();
  11265. +#endif
  11266. +    return status;
  11267. +}
  11268. +
  11269. +NTSTATUS nfs41_Unimplemented(
  11270. +    PRX_CONTEXT RxContext)
  11271. +{
  11272. +    return STATUS_NOT_IMPLEMENTED;
  11273. +}
  11274. +
  11275. +NTSTATUS nfs41_AreFilesAliased(
  11276. +    PFCB a,
  11277. +    PFCB b)
  11278. +{
  11279. +    DbgP("nfs41_AreFilesAliased: a=0x%p b=%0x%p\n",
  11280. +        (void *)a, (void *)b);
  11281. +    return STATUS_NOT_IMPLEMENTED;
  11282. +}
  11283. +
  11284. +static
  11285. +NTSTATUS nfs41_init_ops(void)
  11286. +{
  11287. +    DbgEn();
  11288. +
  11289. +    ZeroAndInitializeNodeType(&nfs41_ops, RDBSS_NTC_MINIRDR_DISPATCH,
  11290. +        sizeof(MINIRDR_DISPATCH));
  11291. +
  11292. +#define FIXME_WORKAROUND_FOR_WIN10_SCAVENGER_CRASH 1
  11293. +#ifdef FIXME_WORKAROUND_FOR_WIN10_SCAVENGER_CRASH
  11294. +    /*
  11295. +     * gisburn: Ugly workaround for crash in Win10 scavenger code
  11296. +     * with a stack trace like this:
  11297. +     * -- snip --
  11298. +     * nt!KeBugCheckEx
  11299. +     * nt!KiBugCheckDispatch+0x69
  11300. +     * nt!KiFastFailDispatch+0xd0
  11301. +     * nt!KiRaiseSecurityCheckFailure+0x31d (TrapFrame @ fffffe0b`41ca0900)
  11302. +     * nfs41_driver!RtlFailFast(void)+0x5 (Inline Function @ fffff801`41ba47dd) [onecore\external\ifskit\inc\wdm.h @ 11545]
  11303. +     * nfs41_driver!FatalListEntryError(void)+0x5 (Inline Function @ fffff801`41ba47dd) [onecore\external\ifskit\inc\wdm.h @ 11778]
  11304. +     * nfs41_driver!RemoveEntryList(void)+0x33 (Inline Function @ fffff801`41ba47dd) [onecore\external\ifskit\inc\wdm.h @ 11811]
  11305. +     * nfs41_driver!RxpUndoScavengerFinalizationMarking(void * Instance = 0xffffca8f`b8f537d0)+0xad [base\fs\rdr2\rxce\scavengr.c @ 1154]
  11306. +     * nfs41_driver!RxScavengerFinalizeEntries(struct _RDBSS_DEVICE_OBJECT * RxDeviceObject = <Value unavailable error>)+0x407 [base\fs\rdr2\rxce\scavengr.c @ 1710]
  11307. +     * nfs41_driver!RxScavengerTimerRoutine(void * Context = 0xffffca8f`bb0d4060)+0x87 [base\fs\rdr2\rxce\scavengr.c @ 1826]
  11308. +     * nfs41_driver!RxpWorkerThreadDispatcher(struct _RX_WORK_QUEUE_ * pWorkQueue = 0xfffff801`41b99240, union _LARGE_INTEGER * pWaitInterval = 0x00000000`00000000)+0xbb [base\fs\rdr2\rxce\rxworkq.c @ 1343]
  11309. +     * nfs41_driver!RxBootstrapWorkerThreadDispatcher(struct _RX_WORK_QUEUE_ * pWorkQueue = <Value unavailable error>)+0xb [base\fs\rdr2\rxce\rxworkq.c @ 1469]
  11310. +     * nt!PspSystemThreadStartup+0x55
  11311. +     * nt!KiStartSystemThread+0x28
  11312. +     * -- snip --
  11313. +     *
  11314. +     * As workaround we "disable" the scavenger by only running it
  11315. +     * every 128 years, until then we should have found a fix.
  11316. +     */
  11317. +    nfs41_ops.ScavengerTimeout = 3600UL*24*365*128;
  11318. +#endif /* FIXME_WORKAROUND_FOR_WIN10_SCAVENGER_CRASH */
  11319. +
  11320. +    nfs41_ops.MRxFlags = (RDBSS_MANAGE_NET_ROOT_EXTENSION |
  11321. +                            RDBSS_MANAGE_V_NET_ROOT_EXTENSION |
  11322. +                            RDBSS_MANAGE_FCB_EXTENSION |
  11323. +                            RDBSS_MANAGE_FOBX_EXTENSION);
  11324. +
  11325. +    nfs41_ops.MRxSrvCallSize  = 0; // srvcall extension is not handled in rdbss
  11326. +    nfs41_ops.MRxNetRootSize  = sizeof(NFS41_NETROOT_EXTENSION);
  11327. +    nfs41_ops.MRxVNetRootSize = sizeof(NFS41_V_NET_ROOT_EXTENSION);
  11328. +    nfs41_ops.MRxFcbSize      = sizeof(NFS41_FCB);
  11329. +    nfs41_ops.MRxFobxSize     = sizeof(NFS41_FOBX);
  11330. +
  11331. +    // Mini redirector cancel routine
  11332. +
  11333. +    nfs41_ops.MRxCancel = NULL;
  11334. +
  11335. +    //
  11336. +    // Mini redirector Start/Stop. Each mini-rdr can be started or stopped
  11337. +    // while the others continue to operate.
  11338. +    //
  11339. +
  11340. +    nfs41_ops.MRxStart                = (PMRX_CALLDOWN_CTX)nfs41_Start;
  11341. +    nfs41_ops.MRxStop                 = (PMRX_CALLDOWN_CTX)nfs41_Stop;
  11342. +    nfs41_ops.MRxDevFcbXXXControlFile =
  11343. +        (PMRX_CALLDOWN)nfs41_DevFcbXXXControlFile;
  11344. +
  11345. +    //
  11346. +    // Mini redirector name resolution
  11347. +    //
  11348. +
  11349. +    nfs41_ops.MRxCreateSrvCall       =
  11350. +        (PMRX_CREATE_SRVCALL)nfs41_CreateSrvCall;
  11351. +    nfs41_ops.MRxSrvCallWinnerNotify =
  11352. +        (PMRX_SRVCALL_WINNER_NOTIFY)nfs41_SrvCallWinnerNotify;
  11353. +    nfs41_ops.MRxCreateVNetRoot      =
  11354. +        (PMRX_CREATE_V_NET_ROOT)nfs41_CreateVNetRoot;
  11355. +    nfs41_ops.MRxExtractNetRootName  =
  11356. +        (PMRX_EXTRACT_NETROOT_NAME)nfs41_ExtractNetRootName;
  11357. +    nfs41_ops.MRxFinalizeSrvCall     =
  11358. +        (PMRX_FINALIZE_SRVCALL_CALLDOWN)nfs41_FinalizeSrvCall;
  11359. +    nfs41_ops.MRxFinalizeNetRoot     =
  11360. +        (PMRX_FINALIZE_NET_ROOT_CALLDOWN)nfs41_FinalizeNetRoot;
  11361. +    nfs41_ops.MRxFinalizeVNetRoot    =
  11362. +        (PMRX_FINALIZE_V_NET_ROOT_CALLDOWN)nfs41_FinalizeVNetRoot;
  11363. +
  11364. +    //
  11365. +    // File System Object Creation/Deletion
  11366. +    //
  11367. +
  11368. +    nfs41_ops.MRxCreate            =
  11369. +        (PMRX_CALLDOWN)nfs41_Create;
  11370. +    nfs41_ops.MRxCollapseOpen      =
  11371. +        (PMRX_CALLDOWN)nfs41_CollapseOpen;
  11372. +    nfs41_ops.MRxShouldTryToCollapseThisOpen =
  11373. +        (PMRX_CALLDOWN)nfs41_ShouldTryToCollapseThisOpen;
  11374. +    nfs41_ops.MRxExtendForCache    =
  11375. +        (PMRX_EXTENDFILE_CALLDOWN)nfs41_ExtendForCache;
  11376. +    nfs41_ops.MRxExtendForNonCache =
  11377. +        (PMRX_EXTENDFILE_CALLDOWN)nfs41_ExtendForCache;
  11378. +    nfs41_ops.MRxCloseSrvOpen      =
  11379. +        (PMRX_CALLDOWN)nfs41_CloseSrvOpen;
  11380. +    nfs41_ops.MRxFlush             =
  11381. +        (PMRX_CALLDOWN)nfs41_Flush;
  11382. +    nfs41_ops.MRxDeallocateForFcb  =
  11383. +        (PMRX_DEALLOCATE_FOR_FCB)nfs41_DeallocateForFcb;
  11384. +    nfs41_ops.MRxDeallocateForFobx =
  11385. +        (PMRX_DEALLOCATE_FOR_FOBX)nfs41_DeallocateForFobx;
  11386. +    nfs41_ops.MRxIsLockRealizable  =
  11387. +        (PMRX_IS_LOCK_REALIZABLE)nfs41_IsLockRealizable;
  11388. +
  11389. +    //
  11390. +    // File System Objects query/Set
  11391. +    //
  11392. +
  11393. +    nfs41_ops.MRxQueryDirectory       =
  11394. +        (PMRX_CALLDOWN)nfs41_QueryDirectory;
  11395. +    nfs41_ops.MRxQueryVolumeInfo      =
  11396. +        (PMRX_CALLDOWN)nfs41_QueryVolumeInformation;
  11397. +    nfs41_ops.MRxSetVolumeInfo        =
  11398. +        (PMRX_CALLDOWN)nfs41_Unimplemented;
  11399. +    nfs41_ops.MRxQueryEaInfo          =
  11400. +        (PMRX_CALLDOWN)nfs41_QueryEaInformation;
  11401. +    nfs41_ops.MRxSetEaInfo            =
  11402. +        (PMRX_CALLDOWN)nfs41_SetEaInformation;
  11403. +    nfs41_ops.MRxQuerySdInfo          =
  11404. +        (PMRX_CALLDOWN)nfs41_QuerySecurityInformation;
  11405. +    nfs41_ops.MRxSetSdInfo            =
  11406. +        (PMRX_CALLDOWN)nfs41_SetSecurityInformation;
  11407. +    nfs41_ops.MRxQueryFileInfo        =
  11408. +        (PMRX_CALLDOWN)nfs41_QueryFileInformation;
  11409. +    nfs41_ops.MRxSetFileInfo          =
  11410. +        (PMRX_CALLDOWN)nfs41_SetFileInformation;
  11411. +    nfs41_ops.MRxQueryQuotaInfo       =
  11412. +        (PMRX_CALLDOWN)nfs41_Unimplemented;
  11413. +    nfs41_ops.MRxSetQuotaInfo         =
  11414. +        (PMRX_CALLDOWN)nfs41_Unimplemented;
  11415. +
  11416. +    //
  11417. +    // Buffering state change
  11418. +    //
  11419. +
  11420. +    nfs41_ops.MRxComputeNewBufferingState =
  11421. +        (PMRX_COMPUTE_NEW_BUFFERING_STATE)nfs41_ComputeNewBufferingState;
  11422. +
  11423. +    //
  11424. +    // File System Object I/O
  11425. +    //
  11426. +
  11427. +    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_READ]            =
  11428. +        (PMRX_CALLDOWN)nfs41_Read;
  11429. +    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_WRITE]           =
  11430. +        (PMRX_CALLDOWN)nfs41_Write;
  11431. +    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_SHAREDLOCK]      =
  11432. +        (PMRX_CALLDOWN)nfs41_Lock;
  11433. +    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_EXCLUSIVELOCK]   =
  11434. +        (PMRX_CALLDOWN)nfs41_Lock;
  11435. +    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_UNLOCK]          =
  11436. +        (PMRX_CALLDOWN)nfs41_Unlock;
  11437. +    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_UNLOCK_MULTIPLE] =
  11438. +        (PMRX_CALLDOWN)nfs41_Unlock;
  11439. +    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_FSCTL]           =
  11440. +        (PMRX_CALLDOWN)nfs41_FsCtl;
  11441. +    nfs41_ops.MRxLowIOSubmit[LOWIO_OP_IOCTL]           =
  11442. +        (PMRX_CALLDOWN)nfs41_IoCtl;
  11443. +
  11444. +    //
  11445. +    // Miscellanous
  11446. +    //
  11447. +
  11448. +    nfs41_ops.MRxCompleteBufferingStateChangeRequest =
  11449. +        (PMRX_CHANGE_BUFFERING_STATE_CALLDOWN)nfs41_CompleteBufferingStateChangeRequest;
  11450. +    nfs41_ops.MRxIsValidDirectory =
  11451. +        (PMRX_CHKDIR_CALLDOWN)nfs41_IsValidDirectory;
  11452. +
  11453. +    nfs41_ops.MRxTruncate =
  11454. +        (PMRX_CALLDOWN)nfs41_Unimplemented;
  11455. +    nfs41_ops.MRxZeroExtend =
  11456. +        (PMRX_CALLDOWN)nfs41_Unimplemented;
  11457. +    nfs41_ops.MRxAreFilesAliased =
  11458. +        (PMRX_CHKFCB_CALLDOWN)nfs41_AreFilesAliased;
  11459. +
  11460. +    DbgR();
  11461. +    return(STATUS_SUCCESS);
  11462. +}
  11463. +
  11464. +KSTART_ROUTINE fcbopen_main;
  11465. +
  11466. +VOID fcbopen_main(PVOID ctx)
  11467. +{
  11468. +    NTSTATUS status;
  11469. +    LARGE_INTEGER timeout;
  11470. +
  11471. +//    DbgEn();
  11472. +    timeout.QuadPart = RELATIVE(SECONDS(30));
  11473. +    while(1) {
  11474. +        PLIST_ENTRY pEntry;
  11475. +        nfs41_fcb_list_entry *cur;
  11476. +        status = KeDelayExecutionThread(KernelMode, TRUE, &timeout);
  11477. +        ExAcquireFastMutex(&fcblistLock);
  11478. +        pEntry = openlist.head.Flink;
  11479. +        while (!IsListEmpty(&openlist.head)) {
  11480. +            PNFS41_NETROOT_EXTENSION pNetRootContext;
  11481. +            nfs41_updowncall_entry *entry = NULL;
  11482. +            FILE_BASIC_INFORMATION binfo;
  11483. +            PNFS41_FCB nfs41_fcb;
  11484. +            cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry,
  11485. +                    nfs41_fcb_list_entry, next);
  11486. +
  11487. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11488. +            DbgP("fcbopen_main: Checking attributes for fcb=0x%p "
  11489. +                "change_time=%llu skipping=%d\n", cur->fcb,
  11490. +                cur->ChangeTime, cur->skip);
  11491. +#endif
  11492. +            if (cur->skip) goto out;
  11493. +
  11494. +            /*
  11495. +             * This can only happen if |nfs41_DeallocateForFobx()|
  11496. +             * was called
  11497. +             */
  11498. +            if ((!cur->nfs41_fobx) || (!cur->nfs41_fobx->sec_ctx.ClientToken))
  11499. +                goto out;
  11500. +
  11501. +            if (!cur->nfs41_fobx->timebasedcoherency) {
  11502. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11503. +                DbgP("fcbopen_main: timebasedcoherency disabled for "
  11504. +                    "fcb=0x%p, nfs41_fobx=0x%p\n", cur->fcb, cur->nfs41_fobx);
  11505. +#endif
  11506. +                goto out;
  11507. +            }
  11508. +
  11509. +            pNetRootContext =
  11510. +                NFS41GetNetRootExtension(cur->fcb->pNetRoot);
  11511. +            /* place an upcall for this srv_open */
  11512. +            status = nfs41_UpcallCreate(
  11513. +                NFS41_FILE_QUERY_TIME_BASED_COHERENCY,
  11514. +                &cur->nfs41_fobx->sec_ctx, cur->session,
  11515. +                cur->nfs41_fobx->nfs41_open_state,
  11516. +                pNetRootContext->nfs41d_version, NULL, &entry);
  11517. +            if (status) goto out;
  11518. +
  11519. +            entry->u.QueryFile.InfoClass = FileBasicInformation;
  11520. +            entry->buf = &binfo;
  11521. +            entry->buf_len = sizeof(binfo);
  11522. +
  11523. +            status = nfs41_UpcallWaitForReply(entry, UPCALL_TIMEOUT_DEFAULT);
  11524. +            if (status) goto out;
  11525. +
  11526. +            if (cur->ChangeTime != entry->ChangeTime) {
  11527. +                ULONG flag = DISABLE_CACHING;
  11528. +                PMRX_SRV_OPEN srv_open;
  11529. +                PLIST_ENTRY psrvEntry;
  11530. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11531. +                DbgP("fcbopen_main: old ctime=%llu new_ctime=%llu\n",
  11532. +                    cur->ChangeTime, entry->ChangeTime);
  11533. +#endif
  11534. +                cur->ChangeTime = entry->ChangeTime;
  11535. +                cur->skip = TRUE;
  11536. +                psrvEntry = &cur->fcb->SrvOpenList;
  11537. +                psrvEntry = psrvEntry->Flink;
  11538. +                while (!IsListEmpty(&cur->fcb->SrvOpenList)) {
  11539. +                    srv_open = (PMRX_SRV_OPEN)CONTAINING_RECORD(psrvEntry,
  11540. +                            MRX_SRV_OPEN, SrvOpenQLinks);
  11541. +                    if (srv_open->DesiredAccess &
  11542. +                            (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA)) {
  11543. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11544. +                        DbgP("fcbopen_main: ************ Invalidate the cache '%wZ'"
  11545. +                             "************\n", srv_open->pAlreadyPrefixedName);
  11546. +#endif
  11547. +                        RxIndicateChangeOfBufferingStateForSrvOpen(
  11548. +                            cur->fcb->pNetRoot->pSrvCall, srv_open,
  11549. +                            srv_open->Key, ULongToPtr(flag));
  11550. +                    }
  11551. +                    if (psrvEntry->Flink == &cur->fcb->SrvOpenList) {
  11552. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11553. +                        DbgP("fcbopen_main: reached end of srvopen for fcb 0x%p\n",
  11554. +                            cur->fcb);
  11555. +#endif
  11556. +                        break;
  11557. +                    }
  11558. +                    psrvEntry = psrvEntry->Flink;
  11559. +                };
  11560. +            }
  11561. +            nfs41_fcb = (PNFS41_FCB)cur->fcb->Context;
  11562. +            nfs41_fcb->changeattr = entry->ChangeTime;
  11563. +out:
  11564. +            nfs41_UpcallDestroy(entry);
  11565. +            entry = NULL;
  11566. +            if (pEntry->Flink == &openlist.head) {
  11567. +#ifdef DEBUG_TIME_BASED_COHERENCY
  11568. +                DbgP("fcbopen_main: reached end of the fcb list\n");
  11569. +#endif
  11570. +                break;
  11571. +            }
  11572. +            pEntry = pEntry->Flink;
  11573. +        }
  11574. +        ExReleaseFastMutex(&fcblistLock);
  11575. +    }
  11576. +//    DbgEx();
  11577. +}
  11578. +
  11579. +/* Main driver entry point, must be public symbol */
  11580. +NTSTATUS DriverEntry(
  11581. +    IN PDRIVER_OBJECT drv,
  11582. +    IN PUNICODE_STRING path)
  11583. +{
  11584. +    NTSTATUS status;
  11585. +    ULONG flags = 0, i;
  11586. +    UNICODE_STRING dev_name, user_dev_name;
  11587. +    PNFS41_DEVICE_EXTENSION dev_exts;
  11588. +    TIME_FIELDS jan_1_1970 = {1970, 1, 1, 0, 0, 0, 0, 0};
  11589. +    ACCESS_MASK mask = 0;
  11590. +    OBJECT_ATTRIBUTES oattrs;
  11591. +
  11592. +    DbgEn();
  11593. +
  11594. +    status = RxDriverEntry(drv, path);
  11595. +    if (status != STATUS_SUCCESS) {
  11596. +        print_error("RxDriverEntry failed: 0x%08lx\n", status);
  11597. +        goto out;
  11598. +    }
  11599. +
  11600. +    RtlInitUnicodeString(&dev_name, NFS41_DEVICE_NAME);
  11601. +    SetFlag(flags, RX_REGISTERMINI_FLAG_DONT_PROVIDE_MAILSLOTS);
  11602. +
  11603. +    status = nfs41_init_ops();
  11604. +    if (status != STATUS_SUCCESS) {
  11605. +        print_error("nfs41_init_ops failed to initialize dispatch table\n");
  11606. +        goto out;
  11607. +    }
  11608. +
  11609. +    DbgP("calling RxRegisterMinirdr\n");
  11610. +    status = RxRegisterMinirdr(&nfs41_dev, drv, &nfs41_ops, flags, &dev_name,
  11611. +                sizeof(NFS41_DEVICE_EXTENSION),
  11612. +                FILE_DEVICE_NETWORK_FILE_SYSTEM, FILE_REMOTE_DEVICE);
  11613. +    if (status != STATUS_SUCCESS) {
  11614. +        print_error("RxRegisterMinirdr failed: 0x%08lx\n", status);
  11615. +        goto out;
  11616. +    }
  11617. +    nfs41_dev->Flags |= DO_BUFFERED_IO;
  11618. +
  11619. +    dev_exts = (PNFS41_DEVICE_EXTENSION)
  11620. +        ((PBYTE)(nfs41_dev) + sizeof(RDBSS_DEVICE_OBJECT));
  11621. +
  11622. +    RxDefineNode(dev_exts, NFS41_DEVICE_EXTENSION);
  11623. +    dev_exts->DeviceObject = nfs41_dev;
  11624. +    nfs41_create_volume_info((PFILE_FS_VOLUME_INFORMATION)dev_exts->VolAttrs,
  11625. +        &dev_exts->VolAttrsLen);
  11626. +
  11627. +    RtlInitUnicodeString(&user_dev_name, NFS41_SHADOW_DEVICE_NAME);
  11628. +    DbgP("calling IoCreateSymbolicLink '%wZ' '%wZ'\n", &user_dev_name, &dev_name);
  11629. +    status = IoCreateSymbolicLink(&user_dev_name, &dev_name);
  11630. +    if (status != STATUS_SUCCESS) {
  11631. +        print_error("Device name IoCreateSymbolicLink failed: 0x%08lx\n", status);
  11632. +        goto out_unregister;
  11633. +    }
  11634. +
  11635. +    KeInitializeEvent(&upcallEvent, SynchronizationEvent, FALSE );
  11636. +    ExInitializeFastMutex(&upcallLock);
  11637. +    ExInitializeFastMutex(&downcallLock);
  11638. +    ExInitializeFastMutex(&openOwnerLock);
  11639. +    ExInitializeFastMutex(&fcblistLock);
  11640. +    InitializeListHead(&upcall.head);
  11641. +    InitializeListHead(&downcall.head);
  11642. +    InitializeListHead(&openlist.head);
  11643. +#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  11644. +    /*
  11645. +     * The |Depth| parameter is unfortunately ignored in Win10,
  11646. +     * otherwise we could use |MmQuerySystemSize()| to scale the
  11647. +     * lookasidelists
  11648. +     */
  11649. +    ExInitializeNPagedLookasideList(
  11650. +        &updowncall_entry_upcall_lookasidelist, NULL, NULL,
  11651. +        POOL_NX_ALLOCATION, sizeof(nfs41_updowncall_entry),
  11652. +        NFS41_MM_POOLTAG_UP, 0);
  11653. +    ExInitializeNPagedLookasideList(
  11654. +        &updowncall_entry_downcall_lookasidelist, NULL, NULL,
  11655. +        POOL_NX_ALLOCATION, sizeof(nfs41_updowncall_entry),
  11656. +        NFS41_MM_POOLTAG_DOWN, 0);
  11657. +#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  11658. +    InitializeObjectAttributes(&oattrs, NULL, OBJ_KERNEL_HANDLE, NULL, NULL);
  11659. +    status = PsCreateSystemThread(&dev_exts->openlistHandle, mask,
  11660. +        &oattrs, NULL, NULL, &fcbopen_main, NULL);
  11661. +    if (status != STATUS_SUCCESS)
  11662. +        goto out_unregister;
  11663. +
  11664. +    drv->DriverUnload = nfs41_driver_unload;
  11665. +
  11666. +    for (i = 0; i <= IRP_MJ_MAXIMUM_FUNCTION; i++)
  11667. +        drv->MajorFunction[i] = (PDRIVER_DISPATCH)nfs41_FsdDispatch;
  11668. +
  11669. +    RtlTimeFieldsToTime(&jan_1_1970, &unix_time_diff);
  11670. +
  11671. +out_unregister:
  11672. +    if (status != STATUS_SUCCESS)
  11673. +        RxUnregisterMinirdr(nfs41_dev);
  11674. +out:
  11675. +    DbgEx();
  11676. +    return status;
  11677. +}
  11678. +
  11679. +/* |nfs41_driver_unload()| - must be public symbol */
  11680. +VOID nfs41_driver_unload(IN PDRIVER_OBJECT drv)
  11681. +{
  11682. +    PRX_CONTEXT RxContext;
  11683. +    NTSTATUS    status;
  11684. +    UNICODE_STRING dev_name, pipe_name;
  11685. +
  11686. +    DbgEn();
  11687. +
  11688. +    RxContext = RxCreateRxContext(NULL, nfs41_dev, RX_CONTEXT_FLAG_IN_FSP);
  11689. +    if (RxContext == NULL) {
  11690. +        status = STATUS_INSUFFICIENT_RESOURCES;
  11691. +        goto unload;
  11692. +    }
  11693. +    status = RxStopMinirdr(RxContext, &RxContext->PostRequest);
  11694. +    RxDereferenceAndDeleteRxContext(RxContext);
  11695. +
  11696. +unload:
  11697. +    RtlInitUnicodeString(&dev_name, NFS41_SHADOW_DEVICE_NAME);
  11698. +    status = IoDeleteSymbolicLink(&dev_name);
  11699. +    if (status != STATUS_SUCCESS) {
  11700. +        print_error("couldn't delete device symbolic link\n");
  11701. +    }
  11702. +    RtlInitUnicodeString(&pipe_name, NFS41_SHADOW_PIPE_NAME);
  11703. +    status = IoDeleteSymbolicLink(&pipe_name);
  11704. +    if (status != STATUS_SUCCESS) {
  11705. +        print_error("couldn't delete pipe symbolic link\n");
  11706. +    }
  11707. +    RxUnload(drv);
  11708. +
  11709. +    DbgP("driver unloaded 0x%p\n", drv);
  11710. +    DbgR();
  11711. +}
  11712. diff --git a/sys/nfs41sys_driver.h b/sys/nfs41sys_driver.h
  11713. new file mode 100644
  11714. index 0000000..67d5a29
  11715. --- /dev/null
  11716. +++ b/sys/nfs41sys_driver.h
  11717. @@ -0,0 +1,825 @@
  11718. +/* NFSv4.1 client for Windows
  11719. + * Copyright (C) 2012 The Regents of the University of Michigan
  11720. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  11721. + *
  11722. + * Olga Kornievskaia <aglo@umich.edu>
  11723. + * Casey Bodley <cbodley@umich.edu>
  11724. + * Roland Mainz <roland.mainz@nrubsig.org>
  11725. + *
  11726. + * This library is free software; you can redistribute it and/or modify it
  11727. + * under the terms of the GNU Lesser General Public License as published by
  11728. + * the Free Software Foundation; either version 2.1 of the License, or (at
  11729. + * your option) any later version.
  11730. + *
  11731. + * This library is distributed in the hope that it will be useful, but
  11732. + * without any warranty; without even the implied warranty of merchantability
  11733. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  11734. + * License for more details.
  11735. + *
  11736. + * You should have received a copy of the GNU Lesser General Public License
  11737. + * along with this library; if not, write to the Free Software Foundation,
  11738. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  11739. + */
  11740. +
  11741. +#ifndef _KERNEL_MODE
  11742. +#error module requires kernel mode
  11743. +#endif
  11744. +
  11745. +#if ((__STDC_VERSION__-0) < 201710L)
  11746. +#error Code requires ISO C17
  11747. +#endif
  11748. +#ifndef _NFS41SYS_DRIVER_H_
  11749. +#define _NFS41SYS_DRIVER_H_ 1
  11750. +
  11751. +#include "nfs_ea.h"
  11752. +
  11753. +#define DECLARE_CONST_ANSI_STRING(_var, _string) \
  11754. +    const CHAR _var ## _buffer[] = _string; \
  11755. +    const ANSI_STRING _var = { sizeof(_string) - sizeof(CHAR), \
  11756. +        sizeof(_string), (PCH) _var ## _buffer }
  11757. +
  11758. +#define DECLARE_EXTERN_CONST_ANSI_STRING(_var) \
  11759. +    extern const CHAR _var ## _buffer[]; \
  11760. +    extern const ANSI_STRING _var;
  11761. +
  11762. +#if _MSC_VER >= 1900
  11763. +/*
  11764. + * gisburn: VS22 chokes on the original define for
  11765. + * |DECLARE_CONST_UNICODE_STRING|, so we use one
  11766. + * without the offending stuff
  11767. + */
  11768. +#undef DECLARE_CONST_UNICODE_STRING
  11769. +#define DECLARE_CONST_UNICODE_STRING(_var, _string) \
  11770. +       const WCHAR _var ## _buffer[] = _string; \
  11771. +       const UNICODE_STRING _var = { sizeof(_string) - sizeof(WCHAR), sizeof(_string), (PWCH) _var ## _buffer }
  11772. +#endif /* _MSC_VER >= 1900 */
  11773. +
  11774. +#define DECLARE_EXTERN_DECLARE_CONST_UNICODE_STRING(_var) \
  11775. +    extern const WCHAR _var ## _buffer[]; \
  11776. +    extern const UNICODE_STRING _var;
  11777. +
  11778. +
  11779. +#ifdef ENABLE_TIMINGS
  11780. +typedef struct __nfs41_timings {
  11781. +    LONG tops, sops;
  11782. +    LONGLONG ticks, size;
  11783. +} nfs41_timings;
  11784. +#endif /* ENABLE_TIMINGS */
  11785. +
  11786. +#define DISABLE_CACHING 0
  11787. +#define ENABLE_READ_CACHING 1
  11788. +#define ENABLE_WRITE_CACHING 2
  11789. +#define ENABLE_READWRITE_CACHING 3
  11790. +
  11791. +#define NFS41_MM_POOLTAG        ('nfs4')
  11792. +#define NFS41_MM_POOLTAG_ACL    ('acls')
  11793. +#define NFS41_MM_POOLTAG_MOUNT  ('mnts')
  11794. +#define NFS41_MM_POOLTAG_OPEN   ('open')
  11795. +#define NFS41_MM_POOLTAG_UP     ('upca')
  11796. +#define NFS41_MM_POOLTAG_DOWN   ('down')
  11797. +
  11798. +
  11799. +DECLARE_EXTERN_DECLARE_CONST_UNICODE_STRING(NfsPrefix);
  11800. +DECLARE_EXTERN_DECLARE_CONST_UNICODE_STRING(PubNfsPrefix);
  11801. +DECLARE_EXTERN_DECLARE_CONST_UNICODE_STRING(AUTH_SYS_NAME);
  11802. +DECLARE_EXTERN_DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5_NAME);
  11803. +DECLARE_EXTERN_DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5I_NAME);
  11804. +DECLARE_EXTERN_DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5P_NAME);
  11805. +DECLARE_EXTERN_DECLARE_CONST_UNICODE_STRING(SLASH);
  11806. +DECLARE_EXTERN_DECLARE_CONST_UNICODE_STRING(EMPTY_STRING);
  11807. +
  11808. +DECLARE_EXTERN_CONST_ANSI_STRING(NfsV3Attributes);
  11809. +DECLARE_EXTERN_CONST_ANSI_STRING(NfsSymlinkTargetName);
  11810. +DECLARE_EXTERN_CONST_ANSI_STRING(NfsActOnLink);
  11811. +
  11812. +#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  11813. +extern NPAGED_LOOKASIDE_LIST updowncall_entry_upcall_lookasidelist;
  11814. +extern NPAGED_LOOKASIDE_LIST updowncall_entry_downcall_lookasidelist;
  11815. +#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  11816. +
  11817. +#ifdef ENABLE_TIMINGS
  11818. +extern nfs41_timings lookup;
  11819. +extern nfs41_timings readdir;
  11820. +extern nfs41_timings open;
  11821. +extern nfs41_timings close;
  11822. +extern nfs41_timings getattr;
  11823. +extern nfs41_timings setattr;
  11824. +extern nfs41_timings getacl;
  11825. +extern nfs41_timings setacl;
  11826. +extern nfs41_timings volume;
  11827. +extern nfs41_timings read;
  11828. +extern nfs41_timings write;
  11829. +extern nfs41_timings lock;
  11830. +extern nfs41_timings unlock;
  11831. +extern nfs41_timings setexattr;
  11832. +extern nfs41_timings getexattr;
  11833. +#endif /* ENABLE_TIMINGS */
  11834. +
  11835. +#define RELATIVE(wait) (-(wait))
  11836. +#define NANOSECONDS(nanos) (((signed __int64)(nanos)) / 100L)
  11837. +#define MICROSECONDS(micros) (((signed __int64)(micros)) * NANOSECONDS(1000L))
  11838. +#define MILLISECONDS(milli) (((signed __int64)(milli)) * MICROSECONDS(1000L))
  11839. +#define SECONDS(seconds) (((signed __int64)(seconds)) * MILLISECONDS(1000L))
  11840. +
  11841. +typedef struct _nfs3_attrs {
  11842. +    DWORD type, mode, nlink, uid, gid, filler1;
  11843. +    LARGE_INTEGER size, used;
  11844. +    struct {
  11845. +        DWORD specdata1;
  11846. +        DWORD specdata2;
  11847. +    } rdev;
  11848. +    LONGLONG fsid, fileid;
  11849. +    LONGLONG atime, mtime, ctime;
  11850. +} nfs3_attrs;
  11851. +
  11852. +enum ftype3 {
  11853. +    NF3REG = 1,
  11854. +    NF3DIR,
  11855. +    NF3BLK,
  11856. +    NF3CHR,
  11857. +    NF3LNK,
  11858. +    NF3SOCK,
  11859. +    NF3FIFO
  11860. +};
  11861. +
  11862. +typedef enum _nfs41_updowncall_state {
  11863. +    NFS41_WAITING_FOR_UPCALL,
  11864. +    NFS41_WAITING_FOR_DOWNCALL,
  11865. +    NFS41_DONE_PROCESSING,
  11866. +    NFS41_NOT_WAITING
  11867. +} nfs41_updowncall_state;
  11868. +
  11869. +typedef struct _updowncall_entry {
  11870. +    DWORD version;
  11871. +    LONGLONG xid;
  11872. +    nfs41_opcodes opcode;
  11873. +    NTSTATUS status;
  11874. +    nfs41_updowncall_state state;
  11875. +    FAST_MUTEX lock;
  11876. +    LIST_ENTRY next;
  11877. +    KEVENT cond;
  11878. +#undef errno
  11879. +    DWORD errno;
  11880. +    BOOLEAN async_op;
  11881. +    SECURITY_CLIENT_CONTEXT sec_ctx;
  11882. +    PSECURITY_CLIENT_CONTEXT psec_ctx;
  11883. +    /*
  11884. +     * Refcount client token during lifetime of this |updowncall_entry|
  11885. +     * to avoid crashes during |SeImpersonateClientEx()| if the
  11886. +     * calling thread disappears.
  11887. +     */
  11888. +    PVOID psec_ctx_clienttoken;
  11889. +    HANDLE open_state;
  11890. +    HANDLE session;
  11891. +    PUNICODE_STRING filename;
  11892. +    PVOID buf;
  11893. +    ULONG buf_len;
  11894. +    ULONGLONG ChangeTime;
  11895. +    union {
  11896. +        struct {
  11897. +            PUNICODE_STRING srv_name; /* hostname, or hostname@port */
  11898. +            PUNICODE_STRING root;
  11899. +            PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs;
  11900. +            DWORD sec_flavor;
  11901. +            DWORD rsize;
  11902. +            DWORD wsize;
  11903. +            DWORD lease_time;
  11904. +            DWORD use_nfspubfh;
  11905. +        } Mount;
  11906. +        struct {
  11907. +            PMDL MdlAddress;
  11908. +            ULONGLONG offset;
  11909. +            PRX_CONTEXT rxcontext;
  11910. +        } ReadWrite;
  11911. +        struct {
  11912. +            LONGLONG offset;
  11913. +            LONGLONG length;
  11914. +            BOOLEAN exclusive;
  11915. +            BOOLEAN blocking;
  11916. +        } Lock;
  11917. +        struct {
  11918. +            ULONG count;
  11919. +            LOWIO_LOCK_LIST locks;
  11920. +        } Unlock;
  11921. +        struct {
  11922. +            FILE_BASIC_INFORMATION binfo;
  11923. +            FILE_STANDARD_INFORMATION sinfo;
  11924. +            UNICODE_STRING symlink;
  11925. +            ULONG access_mask;
  11926. +            ULONG access_mode;
  11927. +            ULONG attrs;
  11928. +            ULONG copts;
  11929. +            ULONG disp;
  11930. +            ULONG cattrs;
  11931. +            LONG open_owner_id;
  11932. +            DWORD mode;
  11933. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  11934. +            DWORD owner_local_uid;
  11935. +            DWORD owner_group_local_gid;
  11936. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  11937. +            HANDLE srv_open;
  11938. +            DWORD deleg_type;
  11939. +            BOOLEAN symlink_embedded;
  11940. +            PMDL EaMdl;
  11941. +            PVOID EaBuffer;
  11942. +        } Open;
  11943. +        struct {
  11944. +            HANDLE srv_open;
  11945. +            BOOLEAN remove;
  11946. +            BOOLEAN renamed;
  11947. +        } Close;
  11948. +        struct {
  11949. +            PUNICODE_STRING filter;
  11950. +            FILE_INFORMATION_CLASS InfoClass;
  11951. +            BOOLEAN restart_scan;
  11952. +            BOOLEAN return_single;
  11953. +            BOOLEAN initial_query;
  11954. +            PMDL mdl;
  11955. +            PVOID mdl_buf;
  11956. +        } QueryFile;
  11957. +        struct {
  11958. +            FILE_INFORMATION_CLASS InfoClass;
  11959. +        } SetFile;
  11960. +        struct {
  11961. +            DWORD mode;
  11962. +        } SetEa;
  11963. +        struct {
  11964. +            PVOID EaList;
  11965. +            ULONG EaListLength;
  11966. +            ULONG Overflow;
  11967. +            ULONG EaIndex;
  11968. +            BOOLEAN ReturnSingleEntry;
  11969. +            BOOLEAN RestartScan;
  11970. +        } QueryEa;
  11971. +        struct {
  11972. +            PUNICODE_STRING target;
  11973. +            BOOLEAN set;
  11974. +        } Symlink;
  11975. +        struct {
  11976. +            FS_INFORMATION_CLASS query;
  11977. +        } Volume;
  11978. +        struct {
  11979. +            SECURITY_INFORMATION query;
  11980. +        } Acl;
  11981. +    } u;
  11982. +
  11983. +} nfs41_updowncall_entry;
  11984. +
  11985. +typedef struct _updowncall_list {
  11986. +    LIST_ENTRY head;
  11987. +} nfs41_updowncall_list;
  11988. +nfs41_updowncall_list upcall, downcall;
  11989. +
  11990. +
  11991. +#define SERVER_NAME_BUFFER_SIZE         1024
  11992. +#define MOUNT_CONFIG_RW_SIZE_MIN        1024
  11993. +#define MOUNT_CONFIG_RW_SIZE_DEFAULT    1048576
  11994. +#define MOUNT_CONFIG_RW_SIZE_MAX        1048576
  11995. +#define MAX_SEC_FLAVOR_LEN              12
  11996. +#define UPCALL_TIMEOUT_DEFAULT          50  /* in seconds */
  11997. +
  11998. +typedef struct _NFS41_MOUNT_CONFIG {
  11999. +    BOOLEAN use_nfspubfh;
  12000. +    DWORD ReadSize;
  12001. +    DWORD WriteSize;
  12002. +    BOOLEAN ReadOnly;
  12003. +    BOOLEAN write_thru;
  12004. +    BOOLEAN nocache;
  12005. +    BOOLEAN timebasedcoherency;
  12006. +    WCHAR srv_buffer[SERVER_NAME_BUFFER_SIZE];
  12007. +    UNICODE_STRING SrvName; /* hostname, or hostname@port */
  12008. +    WCHAR mntpt_buffer[NFS41_SYS_MAX_PATH_LEN];
  12009. +    UNICODE_STRING MntPt;
  12010. +    WCHAR sec_flavor_buffer[MAX_SEC_FLAVOR_LEN];
  12011. +    UNICODE_STRING SecFlavor;
  12012. +    DWORD timeout;
  12013. +    struct {
  12014. +        BOOLEAN use_nfsv3attrsea_mode;
  12015. +        DWORD mode;
  12016. +    } createmode;
  12017. +} NFS41_MOUNT_CONFIG, *PNFS41_MOUNT_CONFIG;
  12018. +
  12019. +typedef struct _nfs41_mount_entry {
  12020. +    LIST_ENTRY next;
  12021. +    LUID login_id;
  12022. +    HANDLE authsys_session;
  12023. +    HANDLE gss_session;
  12024. +    HANDLE gssi_session;
  12025. +    HANDLE gssp_session;
  12026. +    NFS41_MOUNT_CONFIG Config;
  12027. +} nfs41_mount_entry;
  12028. +
  12029. +typedef struct _nfs41_mount_list {
  12030. +    LIST_ENTRY head;
  12031. +} nfs41_mount_list;
  12032. +
  12033. +#define nfs41_AddEntry(lock,list,pEntry)                    \
  12034. +            ExAcquireFastMutex(&lock);                      \
  12035. +            InsertTailList(&(list).head, &(pEntry)->next);  \
  12036. +            ExReleaseFastMutex(&lock);
  12037. +#define nfs41_RemoveFirst(lock,list,pEntry)                 \
  12038. +            ExAcquireFastMutex(&lock);                      \
  12039. +            pEntry = (IsListEmpty(&(list).head)             \
  12040. +            ? NULL                                          \
  12041. +            : RemoveHeadList(&(list).head));                \
  12042. +            ExReleaseFastMutex(&lock);
  12043. +#define nfs41_RemoveEntry(lock,pEntry)                      \
  12044. +            ExAcquireFastMutex(&lock);                      \
  12045. +            RemoveEntryList(&pEntry->next);                 \
  12046. +            ExReleaseFastMutex(&lock);
  12047. +#define nfs41_IsListEmpty(lock,list,flag)                   \
  12048. +            ExAcquireFastMutex(&lock);                      \
  12049. +            flag = IsListEmpty(&(list).head);               \
  12050. +            ExReleaseFastMutex(&lock);
  12051. +#define nfs41_GetFirstEntry(lock,list,pEntry)               \
  12052. +            ExAcquireFastMutex(&lock);                      \
  12053. +            pEntry = (IsListEmpty(&(list).head)             \
  12054. +             ? NULL                                         \
  12055. +             : (nfs41_updowncall_entry *)                   \
  12056. +               (CONTAINING_RECORD((list).head.Flink,        \
  12057. +                                  nfs41_updowncall_entry,   \
  12058. +                                  next)));                  \
  12059. +            ExReleaseFastMutex(&lock);
  12060. +#define nfs41_GetFirstMountEntry(lock,list,pEntry)          \
  12061. +            ExAcquireFastMutex(&lock);                      \
  12062. +            pEntry = (IsListEmpty(&(list).head)             \
  12063. +             ? NULL                                         \
  12064. +             : (nfs41_mount_entry *)                        \
  12065. +               (CONTAINING_RECORD((list).head.Flink,        \
  12066. +                                  nfs41_mount_entry,        \
  12067. +                                  next)));                  \
  12068. +            ExReleaseFastMutex(&lock);
  12069. +
  12070. +
  12071. +typedef struct _NFS41_NETROOT_EXTENSION {
  12072. +    NODE_TYPE_CODE          NodeTypeCode;
  12073. +    NODE_BYTE_SIZE          NodeByteSize;
  12074. +    DWORD                   nfs41d_version;
  12075. +    BOOLEAN                 mounts_init;
  12076. +    FAST_MUTEX              mountLock;
  12077. +    nfs41_mount_list        mounts;
  12078. +} NFS41_NETROOT_EXTENSION, *PNFS41_NETROOT_EXTENSION;
  12079. +#define NFS41GetNetRootExtension(pNetRoot)      \
  12080. +        (((pNetRoot) == NULL) ? NULL :          \
  12081. +        (PNFS41_NETROOT_EXTENSION)((pNetRoot)->Context))
  12082. +
  12083. +/* FileSystemName as reported by FileFsAttributeInfo query */
  12084. +#if ((NFS41_DRIVER_DEBUG_FS_NAME) == 1)
  12085. +#define FS_NAME     L"NFS"
  12086. +#elif  ((NFS41_DRIVER_DEBUG_FS_NAME) == 2)
  12087. +#define FS_NAME     L"DEBUG-NFS41"
  12088. +#else
  12089. +#error NFS41_DRIVER_DEBUG_FS_NAME not defined
  12090. +#endif
  12091. +#define FS_NAME_LEN (sizeof(FS_NAME) - sizeof(WCHAR))
  12092. +#define FS_ATTR_LEN (sizeof(FILE_FS_ATTRIBUTE_INFORMATION) + FS_NAME_LEN)
  12093. +
  12094. +/* FileSystemName as reported by FileFsAttributeInfo query */
  12095. +#define VOL_NAME     L"PnfsVolume"
  12096. +#define VOL_NAME_LEN (sizeof(VOL_NAME) - sizeof(WCHAR))
  12097. +#define VOL_ATTR_LEN (sizeof(FILE_FS_VOLUME_INFORMATION) + VOL_NAME_LEN)
  12098. +
  12099. +typedef struct _NFS41_V_NET_ROOT_EXTENSION {
  12100. +    NODE_TYPE_CODE          NodeTypeCode;
  12101. +    NODE_BYTE_SIZE          NodeByteSize;
  12102. +    HANDLE                  session;
  12103. +    FILE_FS_ATTRIBUTE_INFORMATION FsAttrs;
  12104. +    DWORD                   sec_flavor;
  12105. +    DWORD                   timeout;
  12106. +    struct {
  12107. +        BOOLEAN use_nfsv3attrsea_mode;
  12108. +        DWORD mode;
  12109. +    } createmode;
  12110. +    USHORT                  MountPathLen;
  12111. +    BOOLEAN                 read_only;
  12112. +    BOOLEAN                 write_thru;
  12113. +    BOOLEAN                 nocache;
  12114. +    BOOLEAN                 timebasedcoherency;
  12115. +} NFS41_V_NET_ROOT_EXTENSION, *PNFS41_V_NET_ROOT_EXTENSION;
  12116. +#define NFS41GetVNetRootExtension(pVNetRoot)      \
  12117. +        (((pVNetRoot) == NULL) ? NULL :           \
  12118. +        (PNFS41_V_NET_ROOT_EXTENSION)((pVNetRoot)->Context))
  12119. +
  12120. +typedef struct _NFS41_FCB {
  12121. +    NODE_TYPE_CODE          NodeTypeCode;
  12122. +    NODE_BYTE_SIZE          NodeByteSize;
  12123. +    FILE_BASIC_INFORMATION  BasicInfo;
  12124. +    FILE_STANDARD_INFORMATION StandardInfo;
  12125. +    BOOLEAN                 Renamed;
  12126. +    BOOLEAN                 DeletePending;
  12127. +    DWORD                   mode;
  12128. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  12129. +    DWORD                   owner_local_uid;       /* owner mapped into local uid */
  12130. +    DWORD                   owner_group_local_gid; /* owner group mapped into local gid */
  12131. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  12132. +    ULONGLONG               changeattr;
  12133. +} NFS41_FCB, *PNFS41_FCB;
  12134. +#define NFS41GetFcbExtension(pFcb)      \
  12135. +        (((pFcb) == NULL) ? NULL : (PNFS41_FCB)((pFcb)->Context))
  12136. +
  12137. +typedef struct _NFS41_FOBX {
  12138. +    NODE_TYPE_CODE          NodeTypeCode;
  12139. +    NODE_BYTE_SIZE          NodeByteSize;
  12140. +
  12141. +    HANDLE nfs41_open_state;
  12142. +    SECURITY_CLIENT_CONTEXT sec_ctx;
  12143. +    PVOID acl;
  12144. +    DWORD acl_len;
  12145. +    LARGE_INTEGER time;
  12146. +    DWORD deleg_type;
  12147. +    BOOLEAN write_thru;
  12148. +    BOOLEAN nocache;
  12149. +    BOOLEAN timebasedcoherency;
  12150. +} NFS41_FOBX, *PNFS41_FOBX;
  12151. +#define NFS41GetFobxExtension(pFobx)  \
  12152. +        (((pFobx) == NULL) ? NULL : (PNFS41_FOBX)((pFobx)->Context))
  12153. +
  12154. +typedef struct _NFS41_SERVER_ENTRY {
  12155. +    PMRX_SRV_CALL                 pRdbssSrvCall;
  12156. +    WCHAR                         NameBuffer[SERVER_NAME_BUFFER_SIZE];
  12157. +    UNICODE_STRING                Name;             // the server name.
  12158. +} NFS41_SERVER_ENTRY, *PNFS41_SERVER_ENTRY;
  12159. +
  12160. +typedef struct _NFS41_DEVICE_EXTENSION {
  12161. +    NODE_TYPE_CODE          NodeTypeCode;
  12162. +    NODE_BYTE_SIZE          NodeByteSize;
  12163. +    PRDBSS_DEVICE_OBJECT    DeviceObject;
  12164. +    ULONG                   ActiveNodes;
  12165. +    HANDLE                  SharedMemorySection;
  12166. +    DWORD                   nfs41d_version;
  12167. +    BYTE                    VolAttrs[VOL_ATTR_LEN];
  12168. +    DWORD                   VolAttrsLen;
  12169. +    HANDLE                  openlistHandle;
  12170. +} NFS41_DEVICE_EXTENSION, *PNFS41_DEVICE_EXTENSION;
  12171. +
  12172. +#define NFS41GetDeviceExtension(RxContext,pExt)        \
  12173. +        PNFS41_DEVICE_EXTENSION pExt = (PNFS41_DEVICE_EXTENSION) \
  12174. +        ((PBYTE)(RxContext->RxDeviceObject) + sizeof(RDBSS_DEVICE_OBJECT))
  12175. +
  12176. +typedef struct _nfs41_fcb_list_entry {
  12177. +    LIST_ENTRY next;
  12178. +    PMRX_FCB fcb;
  12179. +    HANDLE session;
  12180. +    PNFS41_FOBX nfs41_fobx;
  12181. +    ULONGLONG ChangeTime;
  12182. +    BOOLEAN skip;
  12183. +} nfs41_fcb_list_entry;
  12184. +
  12185. +typedef struct _nfs41_fcb_list {
  12186. +    LIST_ENTRY head;
  12187. +} nfs41_fcb_list;
  12188. +nfs41_fcb_list openlist;
  12189. +
  12190. +typedef enum _NULMRX_STORAGE_TYPE_CODES {
  12191. +    NTC_NFS41_DEVICE_EXTENSION      =   (NODE_TYPE_CODE)0xFC00,
  12192. +} NFS41_STORAGE_TYPE_CODES;
  12193. +#define RxDefineNode( node, type )          \
  12194. +        node->NodeTypeCode = NTC_##type;    \
  12195. +        node->NodeByteSize = sizeof(type);
  12196. +
  12197. +#define RDR_NULL_STATE  0
  12198. +#define RDR_UNLOADED    1
  12199. +#define RDR_UNLOADING   2
  12200. +#define RDR_LOADING     3
  12201. +#define RDR_LOADED      4
  12202. +#define RDR_STOPPED     5
  12203. +#define RDR_STOPPING    6
  12204. +#define RDR_STARTING    7
  12205. +#define RDR_STARTED     8
  12206. +
  12207. +/*
  12208. + * Assume network speed is 10MB/s (100base-T ethernet, lowest common
  12209. + * denominator which we support) plus disk speed is 10MB/s so add
  12210. + * time to transfer requested bytes over the network and read from
  12211. + * disk.
  12212. + * FIXME: What about ssh-tunneled NFSv4 mounts - should this be a
  12213. + * tuneable/mount option ?
  12214. + */
  12215. +#define EXTRA_TIMEOUT_PER_BYTE(size) ((2LL * (size)) / (10*1024*1024LL))
  12216. +
  12217. +/* Globals */
  12218. +extern KEVENT upcallEvent;
  12219. +extern FAST_MUTEX upcallLock;
  12220. +extern FAST_MUTEX downcallLock;
  12221. +extern FAST_MUTEX fcblistLock;
  12222. +extern FAST_MUTEX openOwnerLock;
  12223. +
  12224. +extern LONGLONG xid;
  12225. +extern LONG open_owner_id;
  12226. +
  12227. +#ifdef NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL
  12228. +extern const LUID SystemLuid;
  12229. +#endif /* NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL */
  12230. +
  12231. +/* |unix_time_diff| - needed to convert windows time to unix */
  12232. +extern LARGE_INTEGER unix_time_diff;
  12233. +
  12234. +
  12235. +/* Prototypes */
  12236. +NTSTATUS map_mount_errors(
  12237. +    DWORD status);
  12238. +NTSTATUS map_sec_flavor(
  12239. +    IN PUNICODE_STRING sec_flavor_name,
  12240. +    OUT PDWORD sec_flavor);
  12241. +NTSTATUS map_open_errors(
  12242. +    DWORD status,
  12243. +    USHORT len);
  12244. +NTSTATUS map_close_errors(
  12245. +    DWORD status);
  12246. +NTSTATUS map_querydir_errors(
  12247. +    DWORD status);
  12248. +NTSTATUS map_volume_errors(
  12249. +    DWORD status);
  12250. +NTSTATUS map_setea_error(
  12251. +    DWORD error);
  12252. +NTSTATUS map_query_acl_error(
  12253. +    DWORD error);
  12254. +NTSTATUS map_queryfile_error(
  12255. +    DWORD error);
  12256. +NTSTATUS map_setfile_error(
  12257. +    DWORD error);
  12258. +NTSTATUS map_readwrite_errors(DWORD status);
  12259. +NTSTATUS map_lock_errors(
  12260. +    DWORD status);
  12261. +NTSTATUS map_symlink_errors(
  12262. +    NTSTATUS status);
  12263. +
  12264. +VOID nfs41_remove_fcb_entry(
  12265. +    PMRX_FCB fcb);
  12266. +
  12267. +/* nfs41sys_acl.c */
  12268. +NTSTATUS marshal_nfs41_getacl(
  12269. +    nfs41_updowncall_entry *entry,
  12270. +    unsigned char *buf,
  12271. +    ULONG buf_len,
  12272. +    ULONG *len);
  12273. +NTSTATUS marshal_nfs41_setacl(
  12274. +    nfs41_updowncall_entry *entry,
  12275. +    unsigned char *buf,
  12276. +    ULONG buf_len,
  12277. +    ULONG *len);
  12278. +NTSTATUS unmarshal_nfs41_getacl(
  12279. +    nfs41_updowncall_entry *cur,
  12280. +    unsigned char **buf);
  12281. +NTSTATUS nfs41_QuerySecurityInformation(
  12282. +    IN OUT PRX_CONTEXT RxContext);
  12283. +NTSTATUS nfs41_SetSecurityInformation(
  12284. +    IN OUT PRX_CONTEXT RxContext);
  12285. +
  12286. +/* nfs41sys_dir.c */
  12287. +NTSTATUS marshal_nfs41_dirquery(
  12288. +    nfs41_updowncall_entry *entry,
  12289. +    unsigned char *buf,
  12290. +    ULONG buf_len,
  12291. +    ULONG *len);
  12292. +NTSTATUS unmarshal_nfs41_dirquery(
  12293. +    nfs41_updowncall_entry *cur,
  12294. +    unsigned char **buf);
  12295. +NTSTATUS check_nfs41_dirquery_args(
  12296. +    IN PRX_CONTEXT RxContext);
  12297. +NTSTATUS nfs41_QueryDirectory(
  12298. +    IN OUT PRX_CONTEXT RxContext);
  12299. +
  12300. +/* nfs41sys_driver.c */
  12301. +NTSTATUS marshall_unicode_as_utf8(
  12302. +    IN OUT unsigned char **pos,
  12303. +    IN PCUNICODE_STRING str);
  12304. +NTSTATUS marshal_nfs41_shutdown(
  12305. +    nfs41_updowncall_entry *entry,
  12306. +    unsigned char *buf,
  12307. +    ULONG buf_len,
  12308. +    ULONG *len);
  12309. +void enable_caching(
  12310. +    PMRX_SRV_OPEN SrvOpen,
  12311. +    PNFS41_FOBX nfs41_fobx,
  12312. +    ULONGLONG ChangeTime,
  12313. +    HANDLE session);
  12314. +VOID nfs41_update_fcb_list(
  12315. +    PMRX_FCB fcb,
  12316. +    ULONGLONG ChangeTime);
  12317. +
  12318. +/* nfs41sys_ea.c */
  12319. +NTSTATUS marshal_nfs41_easet(
  12320. +    nfs41_updowncall_entry *entry,
  12321. +    unsigned char *buf,
  12322. +    ULONG buf_len,
  12323. +    ULONG *len);
  12324. +NTSTATUS marshal_nfs41_eaget(
  12325. +    nfs41_updowncall_entry *entry,
  12326. +    unsigned char *buf,
  12327. +    ULONG buf_len,
  12328. +    ULONG *len);
  12329. +void unmarshal_nfs41_eaget(
  12330. +    nfs41_updowncall_entry *cur,
  12331. +    unsigned char **buf);
  12332. +NTSTATUS nfs41_SetEaInformation(
  12333. +    IN OUT PRX_CONTEXT RxContext);
  12334. +NTSTATUS nfs41_QueryEaInformation(
  12335. +    IN OUT PRX_CONTEXT RxContext);
  12336. +
  12337. +/* nfs41sys_fsctl.c */
  12338. +NTSTATUS nfs41_FsCtl(
  12339. +    IN OUT PRX_CONTEXT RxContext);
  12340. +
  12341. +/* nfs41sys_ioctl.c */
  12342. +NTSTATUS nfs41_IoCtl(
  12343. +    IN OUT PRX_CONTEXT RxContext);
  12344. +
  12345. +/* nfs41sys_lock.c */
  12346. +NTSTATUS marshal_nfs41_lock(
  12347. +    nfs41_updowncall_entry *entry,
  12348. +    unsigned char *buf,
  12349. +    ULONG buf_len,
  12350. +    ULONG *len);
  12351. +NTSTATUS marshal_nfs41_unlock(
  12352. +    nfs41_updowncall_entry *entry,
  12353. +    unsigned char *buf,
  12354. +    ULONG buf_len,
  12355. +    ULONG *len);
  12356. +NTSTATUS nfs41_IsLockRealizable(
  12357. +    IN OUT PMRX_FCB pFcb,
  12358. +    IN PLARGE_INTEGER  ByteOffset,
  12359. +    IN PLARGE_INTEGER  Length,
  12360. +    IN ULONG  LowIoLockFlags);
  12361. +NTSTATUS nfs41_Lock(
  12362. +    IN OUT PRX_CONTEXT RxContext);
  12363. +NTSTATUS nfs41_Unlock(
  12364. +    IN OUT PRX_CONTEXT RxContext);
  12365. +
  12366. +/* nfs41sys_mount.c */
  12367. +void copy_nfs41_mount_config(NFS41_MOUNT_CONFIG *dest,
  12368. +    NFS41_MOUNT_CONFIG *src);
  12369. +NTSTATUS marshal_nfs41_mount(
  12370. +    nfs41_updowncall_entry *entry,
  12371. +    unsigned char *buf,
  12372. +    ULONG buf_len,
  12373. +    ULONG *len);
  12374. +NTSTATUS marshal_nfs41_unmount(
  12375. +    nfs41_updowncall_entry *entry,
  12376. +    unsigned char *buf,
  12377. +    ULONG buf_len,
  12378. +    ULONG *len);
  12379. +void unmarshal_nfs41_mount(
  12380. +    nfs41_updowncall_entry *cur,
  12381. +    unsigned char **buf);
  12382. +NTSTATUS nfs41_unmount(
  12383. +    HANDLE session,
  12384. +    DWORD version,
  12385. +    DWORD timeout);
  12386. +NTSTATUS nfs41_mount(
  12387. +    PNFS41_MOUNT_CONFIG config,
  12388. +    DWORD sec_flavor,
  12389. +    PHANDLE session,
  12390. +    DWORD *version,
  12391. +    PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs);
  12392. +void nfs41_MountConfig_InitDefaults(
  12393. +    OUT PNFS41_MOUNT_CONFIG Config);
  12394. +NTSTATUS nfs41_MountConfig_ParseOptions(
  12395. +    IN PFILE_FULL_EA_INFORMATION EaBuffer,
  12396. +    IN ULONG EaLength,
  12397. +    IN OUT PNFS41_MOUNT_CONFIG Config);
  12398. +NTSTATUS nfs41_CreateVNetRoot(
  12399. +    IN OUT PMRX_CREATENETROOT_CONTEXT pCreateNetRootContext);
  12400. +VOID nfs41_ExtractNetRootName(
  12401. +    IN PUNICODE_STRING FilePathName,
  12402. +    IN PMRX_SRV_CALL SrvCall,
  12403. +    OUT PUNICODE_STRING NetRootName,
  12404. +    OUT PUNICODE_STRING RestOfName OPTIONAL);
  12405. +NTSTATUS nfs41_FinalizeSrvCall(
  12406. +    PMRX_SRV_CALL pSrvCall,
  12407. +    BOOLEAN Force);
  12408. +NTSTATUS nfs41_FinalizeSrvCall(
  12409. +    PMRX_SRV_CALL pSrvCall,
  12410. +    BOOLEAN Force);
  12411. +NTSTATUS nfs41_FinalizeNetRoot(
  12412. +    IN OUT PMRX_NET_ROOT pNetRoot,
  12413. +    IN PBOOLEAN ForceDisconnect);
  12414. +NTSTATUS nfs41_FinalizeVNetRoot(
  12415. +    IN OUT PMRX_V_NET_ROOT pVNetRoot,
  12416. +    IN PBOOLEAN ForceDisconnect);
  12417. +NTSTATUS GetConnectionHandle(
  12418. +    IN PUNICODE_STRING ConnectionName,
  12419. +    IN PVOID EaBuffer,
  12420. +    IN ULONG EaLength,
  12421. +    OUT PHANDLE Handle);
  12422. +NTSTATUS nfs41_CreateConnection(
  12423. +    IN PRX_CONTEXT RxContext,
  12424. +    OUT PBOOLEAN PostToFsp);
  12425. +NTSTATUS nfs41_DeleteConnection(
  12426. +    IN PRX_CONTEXT RxContext,
  12427. +    OUT PBOOLEAN PostToFsp);
  12428. +
  12429. +/* nfs41sys_openclose.c */
  12430. +NTSTATUS marshal_nfs41_open(
  12431. +    nfs41_updowncall_entry *entry,
  12432. +    unsigned char *buf,
  12433. +    ULONG buf_len,
  12434. +    ULONG *len);
  12435. +NTSTATUS marshal_nfs41_close(
  12436. +    nfs41_updowncall_entry *entry,
  12437. +    unsigned char *buf,
  12438. +    ULONG buf_len,
  12439. +    ULONG *len);
  12440. +NTSTATUS unmarshal_nfs41_open(
  12441. +    nfs41_updowncall_entry *cur,
  12442. +    unsigned char **buf);
  12443. +NTSTATUS nfs41_Create(
  12444. +    IN OUT PRX_CONTEXT RxContext);
  12445. +NTSTATUS nfs41_CollapseOpen(
  12446. +    IN OUT PRX_CONTEXT RxContext);
  12447. +NTSTATUS nfs41_ShouldTryToCollapseThisOpen(
  12448. +    IN OUT PRX_CONTEXT RxContext);
  12449. +NTSTATUS nfs41_CloseSrvOpen(
  12450. +    IN OUT PRX_CONTEXT RxContext);
  12451. +
  12452. +/* nfs41sys_readwrite.c */
  12453. +NTSTATUS marshal_nfs41_rw(
  12454. +    nfs41_updowncall_entry *entry,
  12455. +    unsigned char *buf,
  12456. +    ULONG buf_len,
  12457. +    ULONG *len);
  12458. +NTSTATUS unmarshal_nfs41_rw(
  12459. +    nfs41_updowncall_entry *cur,
  12460. +    unsigned char **buf);
  12461. +NTSTATUS nfs41_Read(
  12462. +    IN OUT PRX_CONTEXT RxContext);
  12463. +NTSTATUS nfs41_Write(
  12464. +    IN OUT PRX_CONTEXT RxContext);
  12465. +
  12466. +/* nfs41sys_symlink.c */
  12467. +NTSTATUS marshal_nfs41_symlink(
  12468. +    nfs41_updowncall_entry *entry,
  12469. +    unsigned char *buf,
  12470. +    ULONG buf_len,
  12471. +    ULONG *len);
  12472. +void unmarshal_nfs41_symlink(
  12473. +    nfs41_updowncall_entry *cur,
  12474. +    unsigned char **buf);
  12475. +NTSTATUS nfs41_SetReparsePoint(
  12476. +    IN OUT PRX_CONTEXT RxContext);
  12477. +NTSTATUS nfs41_GetReparsePoint(
  12478. +    IN OUT PRX_CONTEXT RxContext);
  12479. +
  12480. +/* nfs41_updowncall.c */
  12481. +NTSTATUS marshal_nfs41_header(
  12482. +    nfs41_updowncall_entry *entry,
  12483. +    unsigned char *buf,
  12484. +    ULONG buf_len,
  12485. +    ULONG *len);
  12486. +NTSTATUS nfs41_UpcallCreate(
  12487. +    IN DWORD opcode,
  12488. +    IN PSECURITY_CLIENT_CONTEXT clnt_sec_ctx,
  12489. +    IN HANDLE session,
  12490. +    IN HANDLE open_state,
  12491. +    IN DWORD version,
  12492. +    IN PUNICODE_STRING filename,
  12493. +    OUT nfs41_updowncall_entry **entry_out);
  12494. +void nfs41_UpcallDestroy(nfs41_updowncall_entry *entry);
  12495. +NTSTATUS nfs41_UpcallWaitForReply(
  12496. +    IN nfs41_updowncall_entry *entry,
  12497. +    IN DWORD secs);
  12498. +NTSTATUS nfs41_upcall(
  12499. +    IN PRX_CONTEXT RxContext);
  12500. +NTSTATUS nfs41_downcall(
  12501. +    IN PRX_CONTEXT RxContext);
  12502. +
  12503. +/* nfs41sys_fileinfo.c */
  12504. +NTSTATUS marshal_nfs41_filequery(
  12505. +    nfs41_updowncall_entry *entry,
  12506. +    unsigned char *buf,
  12507. +    ULONG buf_len,
  12508. +    ULONG *len);
  12509. +NTSTATUS marshal_nfs41_fileset(
  12510. +    nfs41_updowncall_entry *entry,
  12511. +    unsigned char *buf,
  12512. +    ULONG buf_len,
  12513. +    ULONG *len);
  12514. +void unmarshal_nfs41_setattr(
  12515. +    nfs41_updowncall_entry *cur,
  12516. +    PULONGLONG dest_buf,
  12517. +    unsigned char **buf);
  12518. +void unmarshal_nfs41_getattr(
  12519. +    nfs41_updowncall_entry *cur,
  12520. +    unsigned char **buf);
  12521. +NTSTATUS nfs41_QueryFileInformation(
  12522. +    IN OUT PRX_CONTEXT RxContext);
  12523. +NTSTATUS nfs41_SetFileInformation(
  12524. +    IN OUT PRX_CONTEXT RxContext);
  12525. +
  12526. +/* nfs41sys_volinfo.c */
  12527. +NTSTATUS marshal_nfs41_volume(
  12528. +    nfs41_updowncall_entry *entry,
  12529. +    unsigned char *buf,
  12530. +    ULONG buf_len,
  12531. +    ULONG *len);
  12532. +void unmarshal_nfs41_attrget(
  12533. +    nfs41_updowncall_entry *cur,
  12534. +    PVOID attr_value,
  12535. +    ULONG *attr_len,
  12536. +    unsigned char **buf);
  12537. +NTSTATUS nfs41_QueryVolumeInformation(
  12538. +    IN OUT PRX_CONTEXT RxContext);
  12539. +void nfs41_create_volume_info(
  12540. +    PFILE_FS_VOLUME_INFORMATION pVolInfo, DWORD *len);
  12541. +
  12542. +#endif /* !_NFS41SYS_DRIVER_H_ */
  12543. diff --git a/sys/nfs41sys_ea.c b/sys/nfs41sys_ea.c
  12544. new file mode 100644
  12545. index 0000000..db63399
  12546. --- /dev/null
  12547. +++ b/sys/nfs41sys_ea.c
  12548. @@ -0,0 +1,642 @@
  12549. +/* NFSv4.1 client for Windows
  12550. + * Copyright (C) 2012 The Regents of the University of Michigan
  12551. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  12552. + *
  12553. + * Olga Kornievskaia <aglo@umich.edu>
  12554. + * Casey Bodley <cbodley@umich.edu>
  12555. + * Roland Mainz <roland.mainz@nrubsig.org>
  12556. + *
  12557. + * This library is free software; you can redistribute it and/or modify it
  12558. + * under the terms of the GNU Lesser General Public License as published by
  12559. + * the Free Software Foundation; either version 2.1 of the License, or (at
  12560. + * your option) any later version.
  12561. + *
  12562. + * This library is distributed in the hope that it will be useful, but
  12563. + * without any warranty; without even the implied warranty of merchantability
  12564. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  12565. + * License for more details.
  12566. + *
  12567. + * You should have received a copy of the GNU Lesser General Public License
  12568. + * along with this library; if not, write to the Free Software Foundation,
  12569. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  12570. + */
  12571. +
  12572. +#ifndef _KERNEL_MODE
  12573. +#error module requires kernel mode
  12574. +#endif
  12575. +
  12576. +#if ((__STDC_VERSION__-0) < 201710L)
  12577. +#error Code requires ISO C17
  12578. +#endif
  12579. +
  12580. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  12581. +#if _MSC_VER >= 1900
  12582. +#if defined(_WIN64) && defined(_M_X64)
  12583. +#ifndef _AMD64_
  12584. +#define _AMD64_
  12585. +#endif
  12586. +#elif defined(_WIN32) && defined(_M_IX86)
  12587. +#ifndef _X86_
  12588. +#define _X86_
  12589. +#endif
  12590. +#elif defined(_WIN64) && defined(_M_ARM64)
  12591. +#ifndef _ARM64_
  12592. +#define _ARM64_
  12593. +#endif
  12594. +#elif defined(_WIN32) && defined(_M_ARM)
  12595. +#ifndef _ARM_
  12596. +#define _ARM_
  12597. +#endif
  12598. +#else
  12599. +#error Unsupported arch
  12600. +#endif
  12601. +#endif /* _MSC_VER >= 1900 */
  12602. +
  12603. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  12604. +#include <rx.h>
  12605. +#include <windef.h>
  12606. +#include <winerror.h>
  12607. +
  12608. +#include <Ntstrsafe.h>
  12609. +
  12610. +#include "nfs41sys_buildconfig.h"
  12611. +
  12612. +#include "nfs41_driver.h"
  12613. +#include "nfs41sys_debug.h"
  12614. +#include "nfs41_build_features.h"
  12615. +
  12616. +#include "nfs41sys_driver.h"
  12617. +#include "nfs41sys_util.h"
  12618. +
  12619. +NTSTATUS marshal_nfs41_easet(
  12620. +    nfs41_updowncall_entry *entry,
  12621. +    unsigned char *buf,
  12622. +    ULONG buf_len,
  12623. +    ULONG *len)
  12624. +{
  12625. +    NTSTATUS status = STATUS_SUCCESS;
  12626. +    ULONG header_len = 0;
  12627. +    unsigned char *tmp = buf;
  12628. +
  12629. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  12630. +    if (status) goto out;
  12631. +    else tmp += *len;
  12632. +
  12633. +    header_len = *len + length_as_utf8(entry->filename) +
  12634. +        sizeof(ULONG) + entry->buf_len  + sizeof(DWORD);
  12635. +    if (header_len > buf_len) {
  12636. +        status = STATUS_INSUFFICIENT_RESOURCES;
  12637. +        goto out;
  12638. +    }
  12639. +
  12640. +    status = marshall_unicode_as_utf8(&tmp, entry->filename);
  12641. +    if (status) goto out;
  12642. +    RtlCopyMemory(tmp, &entry->u.SetEa.mode, sizeof(DWORD));
  12643. +    tmp += sizeof(DWORD);
  12644. +    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  12645. +    tmp += sizeof(ULONG);
  12646. +    RtlCopyMemory(tmp, entry->buf, entry->buf_len);
  12647. +    *len = header_len;
  12648. +
  12649. +#ifdef DEBUG_MARSHAL_DETAIL
  12650. +    DbgP("marshal_nfs41_easet: filename='%wZ', buflen=%d mode=0x%x\n",
  12651. +        entry->filename, entry->buf_len, entry->u.SetEa.mode);
  12652. +#endif
  12653. +out:
  12654. +    return status;
  12655. +}
  12656. +
  12657. +NTSTATUS marshal_nfs41_eaget(
  12658. +    nfs41_updowncall_entry *entry,
  12659. +    unsigned char *buf,
  12660. +    ULONG buf_len,
  12661. +    ULONG *len)
  12662. +{
  12663. +    NTSTATUS status = STATUS_SUCCESS;
  12664. +    ULONG header_len = 0;
  12665. +    unsigned char *tmp = buf;
  12666. +
  12667. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  12668. +    if (status) goto out;
  12669. +    else tmp += *len;
  12670. +
  12671. +    header_len = *len + length_as_utf8(entry->filename) +
  12672. +        3 * sizeof(ULONG) + entry->u.QueryEa.EaListLength + 2 * sizeof(BOOLEAN);
  12673. +
  12674. +    if (header_len > buf_len) {
  12675. +        status = STATUS_INSUFFICIENT_RESOURCES;
  12676. +        goto out;
  12677. +    }
  12678. +
  12679. +    status = marshall_unicode_as_utf8(&tmp, entry->filename);
  12680. +    if (status) goto out;
  12681. +    RtlCopyMemory(tmp, &entry->u.QueryEa.EaIndex, sizeof(ULONG));
  12682. +    tmp += sizeof(ULONG);
  12683. +    RtlCopyMemory(tmp, &entry->u.QueryEa.RestartScan, sizeof(BOOLEAN));
  12684. +    tmp += sizeof(BOOLEAN);
  12685. +    RtlCopyMemory(tmp, &entry->u.QueryEa.ReturnSingleEntry, sizeof(BOOLEAN));
  12686. +    tmp += sizeof(BOOLEAN);
  12687. +    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  12688. +    tmp += sizeof(ULONG);
  12689. +    RtlCopyMemory(tmp, &entry->u.QueryEa.EaListLength, sizeof(ULONG));
  12690. +    tmp += sizeof(ULONG);
  12691. +    if (entry->u.QueryEa.EaList && entry->u.QueryEa.EaListLength)
  12692. +        RtlCopyMemory(tmp, entry->u.QueryEa.EaList,
  12693. +            entry->u.QueryEa.EaListLength);
  12694. +    *len = header_len;
  12695. +
  12696. +#ifdef DEBUG_MARSHAL_DETAIL
  12697. +    DbgP("marshal_nfs41_eaget: filename='%wZ', index=%d list_len=%d "
  12698. +        "rescan=%d single=%d\n", entry->filename,
  12699. +        entry->u.QueryEa.EaIndex, entry->u.QueryEa.EaListLength,
  12700. +        entry->u.QueryEa.RestartScan, entry->u.QueryEa.ReturnSingleEntry);
  12701. +#endif
  12702. +out:
  12703. +    return status;
  12704. +}
  12705. +
  12706. +void unmarshal_nfs41_eaget(
  12707. +    nfs41_updowncall_entry *cur,
  12708. +    unsigned char **buf)
  12709. +{
  12710. +    RtlCopyMemory(&cur->u.QueryEa.Overflow, *buf, sizeof(ULONG));
  12711. +    *buf += sizeof(ULONG);
  12712. +    RtlCopyMemory(&cur->buf_len, *buf, sizeof(ULONG));
  12713. +    *buf += sizeof(ULONG);
  12714. +    if (cur->u.QueryEa.Overflow != ERROR_INSUFFICIENT_BUFFER) {
  12715. +        RtlCopyMemory(cur->buf, *buf, cur->buf_len);
  12716. +        *buf += cur->buf_len;
  12717. +    }
  12718. +}
  12719. +
  12720. +static void print_nfs3_attrs(
  12721. +    nfs3_attrs *attrs)
  12722. +{
  12723. +    DbgP("type=%d mode=0%o nlink=%d size=%d "
  12724. +        "atime=0x%x mtime=0x%x ctime=0x%x\n",
  12725. +        attrs->type, attrs->mode, attrs->nlink, attrs->size, attrs->atime,
  12726. +        attrs->mtime, attrs->ctime);
  12727. +}
  12728. +
  12729. +static void file_time_to_nfs_time(
  12730. +    IN const PLARGE_INTEGER file_time,
  12731. +    OUT LONGLONG *nfs_time)
  12732. +{
  12733. +    LARGE_INTEGER diff = unix_time_diff;
  12734. +    diff.QuadPart = file_time->QuadPart - diff.QuadPart;
  12735. +    *nfs_time = diff.QuadPart / 10000000;
  12736. +}
  12737. +
  12738. +static void create_nfs3_attrs(
  12739. +    nfs3_attrs *attrs,
  12740. +    PNFS41_FCB nfs41_fcb)
  12741. +{
  12742. +    RtlZeroMemory(attrs, sizeof(nfs3_attrs));
  12743. +    if (nfs41_fcb->BasicInfo.FileAttributes & FILE_ATTRIBUTE_REPARSE_POINT)
  12744. +        attrs->type = NF3LNK;
  12745. +    else if (nfs41_fcb->StandardInfo.Directory)
  12746. +        attrs->type = NF3DIR;
  12747. +    else
  12748. +        attrs->type = NF3REG;
  12749. +    attrs->mode = nfs41_fcb->mode;
  12750. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  12751. +    attrs->uid = nfs41_fcb->owner_local_uid;
  12752. +    attrs->gid = nfs41_fcb->owner_group_local_gid;
  12753. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  12754. +    attrs->nlink = nfs41_fcb->StandardInfo.NumberOfLinks;
  12755. +    attrs->size.QuadPart = attrs->used.QuadPart =
  12756. +        nfs41_fcb->StandardInfo.EndOfFile.QuadPart;
  12757. +    file_time_to_nfs_time(&nfs41_fcb->BasicInfo.LastAccessTime, &attrs->atime);
  12758. +    file_time_to_nfs_time(&nfs41_fcb->BasicInfo.ChangeTime, &attrs->mtime);
  12759. +    file_time_to_nfs_time(&nfs41_fcb->BasicInfo.CreationTime, &attrs->ctime);
  12760. +}
  12761. +
  12762. +
  12763. +NTSTATUS map_setea_error(
  12764. +    DWORD error)
  12765. +{
  12766. +    switch (error) {
  12767. +    case NO_ERROR:                      return STATUS_SUCCESS;
  12768. +    case ERROR_FILE_NOT_FOUND:          return STATUS_NO_EAS_ON_FILE;
  12769. +    case ERROR_ACCESS_DENIED:           return STATUS_ACCESS_DENIED;
  12770. +    case ERROR_NETWORK_ACCESS_DENIED:   return STATUS_NETWORK_ACCESS_DENIED;
  12771. +    case ERROR_NETNAME_DELETED:         return STATUS_NETWORK_NAME_DELETED;
  12772. +    case ERROR_FILE_TOO_LARGE:          return STATUS_EA_TOO_LARGE;
  12773. +    case ERROR_BUFFER_OVERFLOW:         return STATUS_BUFFER_OVERFLOW;
  12774. +    case STATUS_BUFFER_TOO_SMALL:
  12775. +    case ERROR_INSUFFICIENT_BUFFER:     return STATUS_BUFFER_TOO_SMALL;
  12776. +    case ERROR_INVALID_EA_HANDLE:       return STATUS_NONEXISTENT_EA_ENTRY;
  12777. +    case ERROR_NO_MORE_FILES:           return STATUS_NO_MORE_EAS;
  12778. +    case ERROR_EA_FILE_CORRUPT:         return STATUS_EA_CORRUPT_ERROR;
  12779. +    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  12780. +    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  12781. +    case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  12782. +    default:
  12783. +        print_error("map_setea_error: "
  12784. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  12785. +            "defaulting to STATUS_INVALID_PARAMETER\n", error);
  12786. +    case ERROR_INVALID_PARAMETER:       return STATUS_INVALID_PARAMETER;
  12787. +    }
  12788. +}
  12789. +
  12790. +NTSTATUS check_nfs41_setea_args(
  12791. +    IN PRX_CONTEXT RxContext)
  12792. +{
  12793. +    NTSTATUS status;
  12794. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  12795. +        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  12796. +    __notnull PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs =
  12797. +        &pVNetRootContext->FsAttrs;
  12798. +    __notnull PFILE_FULL_EA_INFORMATION ea =
  12799. +        (PFILE_FULL_EA_INFORMATION)RxContext->Info.Buffer;
  12800. +
  12801. +    status = check_nfs41_dirquery_args(RxContext);
  12802. +    if (status) goto out;
  12803. +
  12804. +    if (ea == NULL) {
  12805. +        status = STATUS_INVALID_PARAMETER;
  12806. +        goto out;
  12807. +    }
  12808. +    if (AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength) ||
  12809. +        AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) {
  12810. +        status = STATUS_INVALID_PARAMETER; /* only allowed on create */
  12811. +        goto out;
  12812. +    }
  12813. +    /* ignore cygwin EAs when checking support */
  12814. +    if (!(FsAttrs->FileSystemAttributes & FILE_SUPPORTS_EXTENDED_ATTRIBUTES)
  12815. +        && !AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength)) {
  12816. +        status = STATUS_EAS_NOT_SUPPORTED;
  12817. +        goto out;
  12818. +    }
  12819. +    if ((RxContext->pRelevantSrvOpen->DesiredAccess & FILE_WRITE_EA) == 0) {
  12820. +        status = STATUS_ACCESS_DENIED;
  12821. +        goto out;
  12822. +    }
  12823. +    if (pVNetRootContext->read_only) {
  12824. +        print_error("check_nfs41_setattr_args: Read-only mount\n");
  12825. +        status = STATUS_MEDIA_WRITE_PROTECTED;
  12826. +        goto out;
  12827. +    }
  12828. +out:
  12829. +    return status;
  12830. +}
  12831. +
  12832. +NTSTATUS nfs41_SetEaInformation(
  12833. +    IN OUT PRX_CONTEXT RxContext)
  12834. +{
  12835. +    NTSTATUS status = STATUS_EAS_NOT_SUPPORTED;
  12836. +    nfs41_updowncall_entry *entry;
  12837. +    __notnull PFILE_FULL_EA_INFORMATION eainfo =
  12838. +        (PFILE_FULL_EA_INFORMATION)RxContext->Info.Buffer;
  12839. +    nfs3_attrs *attrs = NULL;
  12840. +    ULONG buflen = RxContext->CurrentIrpSp->Parameters.SetEa.Length, error_offset;
  12841. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  12842. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  12843. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  12844. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  12845. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  12846. +    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  12847. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  12848. +#ifdef ENABLE_TIMINGS
  12849. +    LARGE_INTEGER t1, t2;
  12850. +    t1 = KeQueryPerformanceCounter(NULL);
  12851. +#endif
  12852. +
  12853. +#ifdef DEBUG_EA_SET
  12854. +    DbgEn();
  12855. +    print_debug_header(RxContext);
  12856. +    print_ea_info(eainfo);
  12857. +#endif
  12858. +
  12859. +    status = check_nfs41_setea_args(RxContext);
  12860. +    if (status) goto out;
  12861. +
  12862. +    status = nfs41_UpcallCreate(NFS41_EA_SET, &nfs41_fobx->sec_ctx,
  12863. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  12864. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  12865. +    if (status) goto out;
  12866. +
  12867. +    if (AnsiStrEq(&NfsV3Attributes, eainfo->EaName, eainfo->EaNameLength)) {
  12868. +        attrs = (nfs3_attrs *)(eainfo->EaName + eainfo->EaNameLength + 1);
  12869. +#ifdef DEBUG_EA_SET
  12870. +        print_nfs3_attrs(attrs);
  12871. +        DbgP("old mode is 0%o new mode is 0%o\n", nfs41_fcb->mode, attrs->mode);
  12872. +#endif
  12873. +        entry->u.SetEa.mode = attrs->mode;
  12874. +    } else {
  12875. +        entry->u.SetEa.mode = 0;
  12876. +        status = IoCheckEaBufferValidity(eainfo, buflen, &error_offset);
  12877. +        if (status) {
  12878. +            DbgP("nfs41_SetEaInformation: "
  12879. +                "status(=0x%lx)=IoCheckEaBufferValidity"
  12880. +                "(eainfo=0x%p, buflen=%lu, &(error_offset=%d))\n",
  12881. +                (long)status, (void *)eainfo, buflen,
  12882. +                (int)error_offset);
  12883. +            nfs41_UpcallDestroy(entry);
  12884. +            entry = NULL;
  12885. +            goto out;
  12886. +        }
  12887. +    }
  12888. +    entry->buf = eainfo;
  12889. +    entry->buf_len = buflen;
  12890. +
  12891. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  12892. +    if (status) goto out;
  12893. +#ifdef ENABLE_TIMINGS
  12894. +    if (entry->status == STATUS_SUCCESS) {
  12895. +        InterlockedIncrement(&setexattr.sops);
  12896. +        InterlockedAdd64(&setexattr.size, entry->u.SetEa.buf_len);
  12897. +    }
  12898. +#endif
  12899. +    status = map_setea_error(entry->status);
  12900. +    if (!status) {
  12901. +        if (!nfs41_fobx->deleg_type && entry->ChangeTime &&
  12902. +                (SrvOpen->DesiredAccess &
  12903. +                (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA)))
  12904. +            nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime);
  12905. +        nfs41_fcb->changeattr = entry->ChangeTime;
  12906. +        nfs41_fcb->mode = entry->u.SetEa.mode;
  12907. +    }
  12908. +    nfs41_UpcallDestroy(entry);
  12909. +out:
  12910. +#ifdef ENABLE_TIMINGS
  12911. +    t2 = KeQueryPerformanceCounter(NULL);
  12912. +    InterlockedIncrement(&setexattr.tops);
  12913. +    InterlockedAdd64(&setexattr.ticks, t2.QuadPart - t1.QuadPart);
  12914. +#ifdef ENABLE_INDV_TIMINGS
  12915. +    DbgP("nfs41_SetEaInformation delta = %d op=%d sum=%d\n",
  12916. +        t2.QuadPart - t1.QuadPart, setexattr.tops, setexattr.ticks);
  12917. +#endif
  12918. +#endif
  12919. +#ifdef DEBUG_EA_SET
  12920. +    DbgEx();
  12921. +#endif
  12922. +    return status;
  12923. +}
  12924. +
  12925. +NTSTATUS check_nfs41_queryea_args(
  12926. +    IN PRX_CONTEXT RxContext)
  12927. +{
  12928. +    NTSTATUS status;
  12929. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  12930. +        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  12931. +    __notnull PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs =
  12932. +        &pVNetRootContext->FsAttrs;
  12933. +    PFILE_GET_EA_INFORMATION ea = (PFILE_GET_EA_INFORMATION)
  12934. +            RxContext->CurrentIrpSp->Parameters.QueryEa.EaList;
  12935. +
  12936. +    status = check_nfs41_dirquery_args(RxContext);
  12937. +    if (status) goto out;
  12938. +
  12939. +    if (!(FsAttrs->FileSystemAttributes & FILE_SUPPORTS_EXTENDED_ATTRIBUTES)) {
  12940. +        if (ea == NULL) {
  12941. +            status = STATUS_EAS_NOT_SUPPORTED;
  12942. +            goto out;
  12943. +        }
  12944. +        /* ignore cygwin EAs when checking support */
  12945. +        if (!AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength) &&
  12946. +            !AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength) &&
  12947. +            !AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) {
  12948. +            status = STATUS_EAS_NOT_SUPPORTED;
  12949. +            goto out;
  12950. +        }
  12951. +    }
  12952. +    if ((RxContext->pRelevantSrvOpen->DesiredAccess & FILE_READ_EA) == 0) {
  12953. +        status = STATUS_ACCESS_DENIED;
  12954. +        goto out;
  12955. +    }
  12956. +out:
  12957. +    return status;
  12958. +}
  12959. +
  12960. +NTSTATUS QueryCygwinSymlink(
  12961. +    IN OUT PRX_CONTEXT RxContext,
  12962. +    IN PFILE_GET_EA_INFORMATION query,
  12963. +    OUT PFILE_FULL_EA_INFORMATION info)
  12964. +{
  12965. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  12966. +    __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext =
  12967. +            NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  12968. +    __notnull PNFS41_NETROOT_EXTENSION NetRootContext =
  12969. +            NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  12970. +    __notnull PNFS41_FOBX Fobx = NFS41GetFobxExtension(RxContext->pFobx);
  12971. +    nfs41_updowncall_entry *entry;
  12972. +    UNICODE_STRING TargetName;
  12973. +    const USHORT HeaderLen = FIELD_OFFSET(FILE_FULL_EA_INFORMATION, EaName) +
  12974. +        query->EaNameLength + 1;
  12975. +    NTSTATUS status;
  12976. +
  12977. +    if (RxContext->Info.LengthRemaining < HeaderLen) {
  12978. +        status = STATUS_BUFFER_TOO_SMALL;
  12979. +        RxContext->InformationToReturn = HeaderLen;
  12980. +        goto out;
  12981. +    }
  12982. +
  12983. +    TargetName.Buffer = (PWCH)(info->EaName + query->EaNameLength + 1);
  12984. +    TargetName.MaximumLength = (USHORT)min(RxContext->Info.LengthRemaining -
  12985. +        HeaderLen, 0xFFFF);
  12986. +
  12987. +    status = nfs41_UpcallCreate(NFS41_SYMLINK, &Fobx->sec_ctx,
  12988. +        VNetRootContext->session, Fobx->nfs41_open_state,
  12989. +        NetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  12990. +    if (status) goto out;
  12991. +
  12992. +    entry->u.Symlink.target = &TargetName;
  12993. +    entry->u.Symlink.set = FALSE;
  12994. +
  12995. +    status = nfs41_UpcallWaitForReply(entry, VNetRootContext->timeout);
  12996. +    if (status) goto out;
  12997. +
  12998. +    status = map_setea_error(entry->status);
  12999. +    if (status == STATUS_SUCCESS) {
  13000. +        info->NextEntryOffset = 0;
  13001. +        info->Flags = 0;
  13002. +        info->EaNameLength = query->EaNameLength;
  13003. +        info->EaValueLength = TargetName.Length - sizeof(UNICODE_NULL);
  13004. +        TargetName.Buffer[TargetName.Length/sizeof(WCHAR)] = UNICODE_NULL;
  13005. +        RtlCopyMemory(info->EaName, query->EaName, query->EaNameLength);
  13006. +        RxContext->Info.LengthRemaining = HeaderLen + info->EaValueLength;
  13007. +    } else if (status == STATUS_BUFFER_TOO_SMALL) {
  13008. +        RxContext->InformationToReturn = (ULONG_PTR)HeaderLen +
  13009. +            entry->u.Symlink.target->Length;
  13010. +    }
  13011. +    nfs41_UpcallDestroy(entry);
  13012. +out:
  13013. +    return status;
  13014. +}
  13015. +
  13016. +NTSTATUS QueryCygwinEA(
  13017. +    IN OUT PRX_CONTEXT RxContext,
  13018. +    IN PFILE_GET_EA_INFORMATION query,
  13019. +    OUT PFILE_FULL_EA_INFORMATION info)
  13020. +{
  13021. +    NTSTATUS status = STATUS_NONEXISTENT_EA_ENTRY;
  13022. +
  13023. +    if (query == NULL)
  13024. +        goto out;
  13025. +
  13026. +    if (AnsiStrEq(&NfsSymlinkTargetName, query->EaName, query->EaNameLength)) {
  13027. +        __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  13028. +        if (nfs41_fcb->BasicInfo.FileAttributes & FILE_ATTRIBUTE_REPARSE_POINT) {
  13029. +            status = QueryCygwinSymlink(RxContext, query, info);
  13030. +            goto out;
  13031. +        } else {
  13032. +            const LONG LengthRequired = sizeof(FILE_FULL_EA_INFORMATION) +
  13033. +                NfsSymlinkTargetName.Length - sizeof(CHAR);
  13034. +            if (LengthRequired > RxContext->Info.LengthRemaining) {
  13035. +                status = STATUS_BUFFER_TOO_SMALL;
  13036. +                RxContext->InformationToReturn = LengthRequired;
  13037. +                goto out;
  13038. +            }
  13039. +            info->NextEntryOffset = 0;
  13040. +            info->Flags = 0;
  13041. +            info->EaValueLength = 0;
  13042. +            info->EaNameLength = (UCHAR)NfsActOnLink.Length;
  13043. +            RtlCopyMemory(info->EaName, NfsSymlinkTargetName.Buffer,
  13044. +                NfsSymlinkTargetName.Length);
  13045. +            RxContext->Info.LengthRemaining = LengthRequired;
  13046. +            status = STATUS_SUCCESS;
  13047. +            goto out;
  13048. +        }
  13049. +    }
  13050. +
  13051. +    if (AnsiStrEq(&NfsV3Attributes, query->EaName, query->EaNameLength)) {
  13052. +        nfs3_attrs attrs;
  13053. +
  13054. +        const LONG LengthRequired = sizeof(FILE_FULL_EA_INFORMATION) +
  13055. +            NfsV3Attributes.Length + sizeof(nfs3_attrs) - sizeof(CHAR);
  13056. +        if (LengthRequired > RxContext->Info.LengthRemaining) {
  13057. +            status = STATUS_BUFFER_TOO_SMALL;
  13058. +            RxContext->InformationToReturn = LengthRequired;
  13059. +            goto out;
  13060. +        }
  13061. +
  13062. +        create_nfs3_attrs(&attrs, NFS41GetFcbExtension(RxContext->pFcb));
  13063. +#ifdef DEBUG_EA_QUERY
  13064. +        print_nfs3_attrs(&attrs);
  13065. +#endif
  13066. +
  13067. +        info->NextEntryOffset = 0;
  13068. +        info->Flags = 0;
  13069. +        info->EaNameLength = (UCHAR)NfsV3Attributes.Length;
  13070. +        info->EaValueLength = sizeof(nfs3_attrs);
  13071. +        RtlCopyMemory(info->EaName, NfsV3Attributes.Buffer,
  13072. +            NfsV3Attributes.Length);
  13073. +        RtlCopyMemory(info->EaName + info->EaNameLength + 1, &attrs,
  13074. +            sizeof(nfs3_attrs));
  13075. +        RxContext->Info.LengthRemaining = LengthRequired;
  13076. +        status = STATUS_SUCCESS;
  13077. +        goto out;
  13078. +    }
  13079. +
  13080. +    if (AnsiStrEq(&NfsActOnLink, query->EaName, query->EaNameLength)) {
  13081. +
  13082. +        const LONG LengthRequired = sizeof(FILE_FULL_EA_INFORMATION) +
  13083. +            query->EaNameLength - sizeof(CHAR);
  13084. +        if (LengthRequired > RxContext->Info.LengthRemaining) {
  13085. +            status = STATUS_BUFFER_TOO_SMALL;
  13086. +            RxContext->InformationToReturn = LengthRequired;
  13087. +            goto out;
  13088. +        }
  13089. +
  13090. +        info->NextEntryOffset = 0;
  13091. +        info->Flags = 0;
  13092. +        info->EaNameLength = query->EaNameLength;
  13093. +        info->EaValueLength = 0;
  13094. +        RtlCopyMemory(info->EaName, query->EaName, query->EaNameLength);
  13095. +        RxContext->Info.LengthRemaining = LengthRequired;
  13096. +        status = STATUS_SUCCESS;
  13097. +        goto out;
  13098. +    }
  13099. +out:
  13100. +    return status;
  13101. +}
  13102. +
  13103. +NTSTATUS nfs41_QueryEaInformation(
  13104. +    IN OUT PRX_CONTEXT RxContext)
  13105. +{
  13106. +    NTSTATUS status = STATUS_EAS_NOT_SUPPORTED;
  13107. +    nfs41_updowncall_entry *entry;
  13108. +    PFILE_GET_EA_INFORMATION query = (PFILE_GET_EA_INFORMATION)
  13109. +            RxContext->CurrentIrpSp->Parameters.QueryEa.EaList;
  13110. +    ULONG buflen = RxContext->CurrentIrpSp->Parameters.QueryEa.Length;
  13111. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  13112. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  13113. +            NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  13114. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  13115. +            NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  13116. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  13117. +#ifdef ENABLE_TIMINGS
  13118. +    LARGE_INTEGER t1, t2;
  13119. +    t1 = KeQueryPerformanceCounter(NULL);
  13120. +#endif
  13121. +
  13122. +#ifdef DEBUG_EA_QUERY
  13123. +    DbgEn();
  13124. +    print_debug_header(RxContext);
  13125. +    print_get_ea(1, query);
  13126. +#endif
  13127. +    status = check_nfs41_queryea_args(RxContext);
  13128. +    if (status) goto out;
  13129. +
  13130. +    /* handle queries for cygwin EAs */
  13131. +    status = QueryCygwinEA(RxContext, query,
  13132. +        (PFILE_FULL_EA_INFORMATION)RxContext->Info.Buffer);
  13133. +    if (status != STATUS_NONEXISTENT_EA_ENTRY)
  13134. +        goto out;
  13135. +
  13136. +    status = nfs41_UpcallCreate(NFS41_EA_GET, &nfs41_fobx->sec_ctx,
  13137. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  13138. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  13139. +    if (status) goto out;
  13140. +
  13141. +    entry->buf_len = buflen;
  13142. +    entry->buf = RxContext->Info.Buffer;
  13143. +    entry->u.QueryEa.EaList = query;
  13144. +    entry->u.QueryEa.EaListLength = query == NULL ? 0 :
  13145. +        RxContext->QueryEa.UserEaListLength;
  13146. +    entry->u.QueryEa.EaIndex = RxContext->QueryEa.IndexSpecified ?
  13147. +        RxContext->QueryEa.UserEaIndex : 0;
  13148. +    entry->u.QueryEa.RestartScan = RxContext->QueryEa.RestartScan;
  13149. +    entry->u.QueryEa.ReturnSingleEntry = RxContext->QueryEa.ReturnSingleEntry;
  13150. +
  13151. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  13152. +    if (status) goto out;
  13153. +
  13154. +    if (entry->status == STATUS_SUCCESS) {
  13155. +        switch (entry->u.QueryEa.Overflow) {
  13156. +        case ERROR_INSUFFICIENT_BUFFER:
  13157. +            status = STATUS_BUFFER_TOO_SMALL;
  13158. +            break;
  13159. +        case ERROR_BUFFER_OVERFLOW:
  13160. +            status = RxContext->IoStatusBlock.Status = STATUS_BUFFER_OVERFLOW;
  13161. +            break;
  13162. +        default:
  13163. +            RxContext->IoStatusBlock.Status = STATUS_SUCCESS;
  13164. +            break;
  13165. +        }
  13166. +        RxContext->InformationToReturn = entry->buf_len;
  13167. +#ifdef ENABLE_TIMINGS
  13168. +        InterlockedIncrement(&getexattr.sops);
  13169. +        InterlockedAdd64(&getexattr.size, entry->u.QueryEa.buf_len);
  13170. +#endif
  13171. +    } else {
  13172. +        status = map_setea_error(entry->status);
  13173. +    }
  13174. +    nfs41_UpcallDestroy(entry);
  13175. +out:
  13176. +#ifdef ENABLE_TIMINGS
  13177. +    t2 = KeQueryPerformanceCounter(NULL);
  13178. +    InterlockedIncrement(&getexattr.tops);
  13179. +    InterlockedAdd64(&getexattr.ticks, t2.QuadPart - t1.QuadPart);
  13180. +#ifdef ENABLE_INDV_TIMINGS
  13181. +    DbgP("nfs41_QueryEaInformation delta = %d op=%d sum=%d\n",
  13182. +        t2.QuadPart - t1.QuadPart, getexattr.tops, getexattr.ticks);
  13183. +#endif
  13184. +#endif
  13185. +#ifdef DEBUG_EA_QUERY
  13186. +    DbgEx();
  13187. +#endif
  13188. +    return status;
  13189. +}
  13190. +
  13191. diff --git a/sys/nfs41sys_fileinfo.c b/sys/nfs41sys_fileinfo.c
  13192. new file mode 100644
  13193. index 0000000..0839bc2
  13194. --- /dev/null
  13195. +++ b/sys/nfs41sys_fileinfo.c
  13196. @@ -0,0 +1,706 @@
  13197. +/* NFSv4.1 client for Windows
  13198. + * Copyright (C) 2012 The Regents of the University of Michigan
  13199. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  13200. + *
  13201. + * Olga Kornievskaia <aglo@umich.edu>
  13202. + * Casey Bodley <cbodley@umich.edu>
  13203. + * Roland Mainz <roland.mainz@nrubsig.org>
  13204. + *
  13205. + * This library is free software; you can redistribute it and/or modify it
  13206. + * under the terms of the GNU Lesser General Public License as published by
  13207. + * the Free Software Foundation; either version 2.1 of the License, or (at
  13208. + * your option) any later version.
  13209. + *
  13210. + * This library is distributed in the hope that it will be useful, but
  13211. + * without any warranty; without even the implied warranty of merchantability
  13212. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  13213. + * License for more details.
  13214. + *
  13215. + * You should have received a copy of the GNU Lesser General Public License
  13216. + * along with this library; if not, write to the Free Software Foundation,
  13217. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  13218. + */
  13219. +
  13220. +#ifndef _KERNEL_MODE
  13221. +#error module requires kernel mode
  13222. +#endif
  13223. +
  13224. +#if ((__STDC_VERSION__-0) < 201710L)
  13225. +#error Code requires ISO C17
  13226. +#endif
  13227. +
  13228. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  13229. +#if _MSC_VER >= 1900
  13230. +#if defined(_WIN64) && defined(_M_X64)
  13231. +#ifndef _AMD64_
  13232. +#define _AMD64_
  13233. +#endif
  13234. +#elif defined(_WIN32) && defined(_M_IX86)
  13235. +#ifndef _X86_
  13236. +#define _X86_
  13237. +#endif
  13238. +#elif defined(_WIN64) && defined(_M_ARM64)
  13239. +#ifndef _ARM64_
  13240. +#define _ARM64_
  13241. +#endif
  13242. +#elif defined(_WIN32) && defined(_M_ARM)
  13243. +#ifndef _ARM_
  13244. +#define _ARM_
  13245. +#endif
  13246. +#else
  13247. +#error Unsupported arch
  13248. +#endif
  13249. +#endif /* _MSC_VER >= 1900 */
  13250. +
  13251. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  13252. +#include <rx.h>
  13253. +#include <windef.h>
  13254. +#include <winerror.h>
  13255. +
  13256. +#include <Ntstrsafe.h>
  13257. +
  13258. +#include "nfs41sys_buildconfig.h"
  13259. +
  13260. +#include "nfs41_driver.h"
  13261. +#include "nfs41sys_debug.h"
  13262. +#include "nfs41_build_features.h"
  13263. +
  13264. +#include "nfs41sys_driver.h"
  13265. +#include "nfs41sys_util.h"
  13266. +
  13267. +
  13268. +NTSTATUS marshal_nfs41_filequery(
  13269. +    nfs41_updowncall_entry *entry,
  13270. +    unsigned char *buf,
  13271. +    ULONG buf_len,
  13272. +    ULONG *len)
  13273. +{
  13274. +    NTSTATUS status = STATUS_SUCCESS;
  13275. +    ULONG header_len = 0;
  13276. +    unsigned char *tmp = buf;
  13277. +
  13278. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  13279. +    if (status) goto out;
  13280. +    else tmp += *len;
  13281. +
  13282. +    header_len = *len + 2 * sizeof(ULONG) + 2*sizeof(HANDLE);
  13283. +    if (header_len > buf_len) {
  13284. +        status = STATUS_INSUFFICIENT_RESOURCES;
  13285. +        goto out;
  13286. +    }
  13287. +    RtlCopyMemory(tmp, &entry->u.QueryFile.InfoClass, sizeof(ULONG));
  13288. +    tmp += sizeof(ULONG);
  13289. +    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  13290. +    tmp += sizeof(ULONG);
  13291. +    RtlCopyMemory(tmp, &entry->session, sizeof(HANDLE));
  13292. +    tmp += sizeof(HANDLE);
  13293. +    RtlCopyMemory(tmp, &entry->open_state, sizeof(HANDLE));
  13294. +    *len = header_len;
  13295. +
  13296. +#ifdef DEBUG_MARSHAL_DETAIL
  13297. +    DbgP("marshal_nfs41_filequery: class=%d\n", entry->u.QueryFile.InfoClass);
  13298. +#endif
  13299. +out:
  13300. +    return status;
  13301. +}
  13302. +
  13303. +NTSTATUS marshal_nfs41_fileset(
  13304. +    nfs41_updowncall_entry *entry,
  13305. +    unsigned char *buf,
  13306. +    ULONG buf_len,
  13307. +    ULONG *len)
  13308. +{
  13309. +    NTSTATUS status = STATUS_SUCCESS;
  13310. +    ULONG header_len = 0;
  13311. +    unsigned char *tmp = buf;
  13312. +
  13313. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  13314. +    if (status) goto out;
  13315. +    else tmp += *len;
  13316. +
  13317. +    header_len = *len + length_as_utf8(entry->filename) +
  13318. +        2 * sizeof(ULONG) + entry->buf_len;
  13319. +    if (header_len > buf_len) {
  13320. +        status = STATUS_INSUFFICIENT_RESOURCES;
  13321. +        goto out;
  13322. +    }
  13323. +    status = marshall_unicode_as_utf8(&tmp, entry->filename);
  13324. +    if (status) goto out;
  13325. +    RtlCopyMemory(tmp, &entry->u.SetFile.InfoClass, sizeof(ULONG));
  13326. +    tmp += sizeof(ULONG);
  13327. +    RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG));
  13328. +    tmp += sizeof(ULONG);
  13329. +    RtlCopyMemory(tmp, entry->buf, entry->buf_len);
  13330. +    *len = header_len;
  13331. +
  13332. +#ifdef DEBUG_MARSHAL_DETAIL
  13333. +    DbgP("marshal_nfs41_fileset: filename='%wZ' class=%d\n",
  13334. +        entry->filename, entry->u.SetFile.InfoClass);
  13335. +#endif
  13336. +out:
  13337. +    return status;
  13338. +}
  13339. +
  13340. +void unmarshal_nfs41_setattr(
  13341. +    nfs41_updowncall_entry *cur,
  13342. +    PULONGLONG dest_buf,
  13343. +    unsigned char **buf)
  13344. +{
  13345. +    RtlCopyMemory(dest_buf, *buf, sizeof(ULONGLONG));
  13346. +#ifdef DEBUG_MARSHAL_DETAIL
  13347. +    DbgP("unmarshal_nfs41_setattr: returned ChangeTime %llu\n", *dest_buf);
  13348. +#endif
  13349. +}
  13350. +
  13351. +void unmarshal_nfs41_getattr(
  13352. +    nfs41_updowncall_entry *cur,
  13353. +    unsigned char **buf)
  13354. +{
  13355. +    unmarshal_nfs41_attrget(cur, cur->buf, &cur->buf_len, buf);
  13356. +    RtlCopyMemory(&cur->ChangeTime, *buf, sizeof(LONGLONG));
  13357. +#ifdef DEBUG_MARSHAL_DETAIL
  13358. +    if (cur->u.QueryFile.InfoClass == FileBasicInformation)
  13359. +        DbgP("[unmarshal_nfs41_getattr] ChangeTime %llu\n", cur->ChangeTime);
  13360. +#endif
  13361. +}
  13362. +
  13363. +NTSTATUS map_queryfile_error(
  13364. +    DWORD error)
  13365. +{
  13366. +    switch (error) {
  13367. +    case ERROR_ACCESS_DENIED:       return STATUS_ACCESS_DENIED;
  13368. +    case ERROR_NETNAME_DELETED:     return STATUS_NETWORK_NAME_DELETED;
  13369. +    case ERROR_INVALID_PARAMETER:   return STATUS_INVALID_PARAMETER;
  13370. +    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  13371. +    default:
  13372. +        print_error("map_queryfile_error: "
  13373. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  13374. +            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", error);
  13375. +    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  13376. +    }
  13377. +}
  13378. +
  13379. +NTSTATUS nfs41_QueryFileInformation(
  13380. +    IN OUT PRX_CONTEXT RxContext)
  13381. +{
  13382. +    NTSTATUS status = STATUS_OBJECT_NAME_NOT_FOUND;
  13383. +    FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass;
  13384. +    nfs41_updowncall_entry *entry;
  13385. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  13386. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  13387. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  13388. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  13389. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  13390. +    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  13391. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  13392. +#ifdef ENABLE_TIMINGS
  13393. +    LARGE_INTEGER t1, t2;
  13394. +    t1 = KeQueryPerformanceCounter(NULL);
  13395. +#endif
  13396. +
  13397. +#ifdef DEBUG_FILE_QUERY
  13398. +    DbgEn();
  13399. +    print_debug_filedirquery_header(RxContext);
  13400. +    DbgP("--> nfs41_QueryFileInformation, RxContext->Info.LengthRemaining=%ld\n",
  13401. +        (long)RxContext->Info.LengthRemaining);
  13402. +#endif
  13403. +
  13404. +    status = check_nfs41_dirquery_args(RxContext);
  13405. +    if (status) {
  13406. +        print_error("check_nfs41_dirquery_args failed.\n");
  13407. +        goto out;
  13408. +    }
  13409. +
  13410. +    RtlZeroMemory(RxContext->Info.Buffer, RxContext->Info.LengthRemaining);
  13411. +
  13412. +#ifdef DEBUG_FILE_QUERY
  13413. +    DbgP("nfs41_QueryFileInformation, RxContext->Info.LengthRemaining=%ld\n",
  13414. +        (long)RxContext->Info.LengthRemaining);
  13415. +#endif
  13416. +
  13417. +    switch (InfoClass) {
  13418. +    case FileEaInformation:
  13419. +    {
  13420. +        if (RxContext->Info.LengthRemaining <
  13421. +            sizeof(FILE_EA_INFORMATION)) {
  13422. +            print_error("nfs41_QueryFileInformation: "
  13423. +                "FILE_EA_INFORMATION buffer too small\n");
  13424. +            status = STATUS_BUFFER_TOO_SMALL;
  13425. +            goto out;
  13426. +        }
  13427. +
  13428. +        PFILE_EA_INFORMATION info =
  13429. +            (PFILE_EA_INFORMATION)RxContext->Info.Buffer;
  13430. +        info->EaSize = 0;
  13431. +        RxContext->Info.LengthRemaining -= sizeof(FILE_EA_INFORMATION);
  13432. +        status = STATUS_SUCCESS;
  13433. +        goto out;
  13434. +    }
  13435. +    case FileRemoteProtocolInformation:
  13436. +    {
  13437. +        if (RxContext->Info.LengthRemaining <
  13438. +            sizeof(FILE_REMOTE_PROTOCOL_INFORMATION)) {
  13439. +            print_error("nfs41_QueryFileInformation: "
  13440. +                "FILE_REMOTE_PROTOCOL_INFORMATION buffer too small\n");
  13441. +            status = STATUS_BUFFER_TOO_SMALL;
  13442. +            goto out;
  13443. +        }
  13444. +
  13445. +        PFILE_REMOTE_PROTOCOL_INFORMATION info =
  13446. +            (PFILE_REMOTE_PROTOCOL_INFORMATION)RxContext->Info.Buffer;
  13447. +
  13448. +        (void)RtlZeroMemory(info,
  13449. +            sizeof(FILE_REMOTE_PROTOCOL_INFORMATION));
  13450. +        info->StructureVersion = 1;
  13451. +        info->StructureSize = sizeof(FILE_REMOTE_PROTOCOL_INFORMATION);
  13452. +        info->Protocol = WNNC_NET_RDR2SAMPLE; /* FIXME! */
  13453. +        /*
  13454. +         * ToDo: If we add NFSv4.1/NFSv4.2 protocol negotiation, then
  13455. +         * we need to call the userland daemon to return the correct
  13456. +         * protocol minor version
  13457. +         */
  13458. +        info->ProtocolMajorVersion = 4;
  13459. +        info->ProtocolMinorVersion = 1;
  13460. +        info->ProtocolRevision = 0;
  13461. +        RxContext->Info.LengthRemaining -=
  13462. +            sizeof(FILE_REMOTE_PROTOCOL_INFORMATION);
  13463. +        status = STATUS_SUCCESS;
  13464. +        goto out;
  13465. +    }
  13466. +    case FileCaseSensitiveInformation:
  13467. +    {
  13468. +        if (RxContext->Info.LengthRemaining <
  13469. +            sizeof(FILE_CASE_SENSITIVE_INFORMATION)) {
  13470. +            print_error("nfs41_QueryFileInformation: "
  13471. +                "FILE_CASE_SENSITIVE_INFORMATION buffer too small\n");
  13472. +            status = STATUS_BUFFER_TOO_SMALL;
  13473. +            goto out;
  13474. +        }
  13475. +
  13476. +        PFILE_CASE_SENSITIVE_INFORMATION info =
  13477. +            (PFILE_CASE_SENSITIVE_INFORMATION)RxContext->Info.Buffer;
  13478. +
  13479. +        ULONG fsattrs = pVNetRootContext->FsAttrs.FileSystemAttributes;
  13480. +
  13481. +        /*
  13482. +         * For NFSv4.1 |FATTR4_WORD0_CASE_INSENSITIVE| used
  13483. +         * to fill |FsAttrs.FileSystemAttributes| is per
  13484. +         * filesystem.
  13485. +         * FIXME: Future NFSv4.x standards should make this a
  13486. +         * per-filesystem, per-directory and
  13487. +         * per-extended-attribute-dir attribute to support
  13488. +         * Win32
  13489. +         */
  13490. +        if (fsattrs & FILE_CASE_SENSITIVE_SEARCH) {
  13491. +            info->Flags = FILE_CS_FLAG_CASE_SENSITIVE_DIR;
  13492. +        }
  13493. +
  13494. +        RxContext->Info.LengthRemaining -=
  13495. +            sizeof(FILE_CASE_SENSITIVE_INFORMATION);
  13496. +        status = STATUS_SUCCESS;
  13497. +        goto out;
  13498. +    }
  13499. +    case FileBasicInformation:
  13500. +    case FileStandardInformation:
  13501. +    case FileInternalInformation:
  13502. +    case FileAttributeTagInformation:
  13503. +    case FileNetworkOpenInformation:
  13504. +        break;
  13505. +    default:
  13506. +        print_error("nfs41_QueryFileInformation: unhandled class %d\n", InfoClass);
  13507. +        status = STATUS_NOT_SUPPORTED;
  13508. +        goto out;
  13509. +    }
  13510. +
  13511. +    status = nfs41_UpcallCreate(NFS41_FILE_QUERY, &nfs41_fobx->sec_ctx,
  13512. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  13513. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  13514. +    if (status) {
  13515. +        print_error("nfs41_UpcallCreate() failed, status=0x%lx\n",
  13516. +            (long)status);
  13517. +        goto out;
  13518. +    }
  13519. +
  13520. +    entry->u.QueryFile.InfoClass = InfoClass;
  13521. +    entry->buf = RxContext->Info.Buffer;
  13522. +    entry->buf_len = RxContext->Info.LengthRemaining;
  13523. +
  13524. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  13525. +    if (status) {
  13526. +        print_error("nfs41_UpcallWaitForReply() failed, status=0x%lx\n",
  13527. +            (long)status);
  13528. +        goto out;
  13529. +    }
  13530. +
  13531. +    if (entry->status == STATUS_BUFFER_TOO_SMALL) {
  13532. +        RxContext->InformationToReturn = entry->buf_len;
  13533. +        print_error("entry->status == STATUS_BUFFER_TOO_SMALL\n");
  13534. +        status = STATUS_BUFFER_TOO_SMALL;
  13535. +    } else if (entry->status == STATUS_SUCCESS) {
  13536. +#ifdef DEBUG_FILE_QUERY
  13537. +        print_error("entry->status == STATUS_SUCCESS\n");
  13538. +#endif
  13539. +        BOOLEAN DeletePending = FALSE;
  13540. +#ifdef ENABLE_TIMINGS
  13541. +        InterlockedIncrement(&getattr.sops);
  13542. +        InterlockedAdd64(&getattr.size, entry->u.QueryFile.buf_len);
  13543. +#endif
  13544. +        RxContext->Info.LengthRemaining -= entry->buf_len;
  13545. +        status = STATUS_SUCCESS;
  13546. +
  13547. +        switch (InfoClass) {
  13548. +        case FileBasicInformation:
  13549. +            RtlCopyMemory(&nfs41_fcb->BasicInfo, RxContext->Info.Buffer,
  13550. +                sizeof(nfs41_fcb->BasicInfo));
  13551. +#ifdef DEBUG_FILE_QUERY
  13552. +            print_basic_info(1, &nfs41_fcb->BasicInfo);
  13553. +#endif
  13554. +            break;
  13555. +        case FileStandardInformation:
  13556. +            /* this a fix for RDBSS behaviour when it first calls ExtendForCache,
  13557. +             * then it sends a file query irp for standard attributes and
  13558. +             * expects to receive EndOfFile of value set by the ExtendForCache.
  13559. +             * It seems to cache the filesize based on that instead of sending
  13560. +             * a file size query for after doing the write.
  13561. +             */
  13562. +        {
  13563. +            PFILE_STANDARD_INFORMATION std_info;
  13564. +            std_info = (PFILE_STANDARD_INFORMATION)RxContext->Info.Buffer;
  13565. +            if (nfs41_fcb->StandardInfo.AllocationSize.QuadPart >
  13566. +                    std_info->AllocationSize.QuadPart) {
  13567. +#ifdef DEBUG_FILE_QUERY
  13568. +                DbgP("Old AllocationSize is bigger: saving 0x%x\n",
  13569. +                    nfs41_fcb->StandardInfo.AllocationSize.QuadPart);
  13570. +#endif
  13571. +                std_info->AllocationSize.QuadPart =
  13572. +                    nfs41_fcb->StandardInfo.AllocationSize.QuadPart;
  13573. +            }
  13574. +            if (nfs41_fcb->StandardInfo.EndOfFile.QuadPart >
  13575. +                    std_info->EndOfFile.QuadPart) {
  13576. +#ifdef DEBUG_FILE_QUERY
  13577. +                DbgP("Old EndOfFile is bigger: saving 0x%x\n",
  13578. +                    nfs41_fcb->StandardInfo.EndOfFile);
  13579. +#endif
  13580. +                std_info->EndOfFile.QuadPart =
  13581. +                    nfs41_fcb->StandardInfo.EndOfFile.QuadPart;
  13582. +            }
  13583. +            std_info->DeletePending = nfs41_fcb->DeletePending;
  13584. +        }
  13585. +            if (nfs41_fcb->StandardInfo.DeletePending)
  13586. +                DeletePending = TRUE;
  13587. +            RtlCopyMemory(&nfs41_fcb->StandardInfo, RxContext->Info.Buffer,
  13588. +                sizeof(nfs41_fcb->StandardInfo));
  13589. +            nfs41_fcb->StandardInfo.DeletePending = DeletePending;
  13590. +#ifdef DEBUG_FILE_QUERY
  13591. +            print_std_info(1, &nfs41_fcb->StandardInfo);
  13592. +#endif
  13593. +            break;
  13594. +        case FileNetworkOpenInformation:
  13595. +        case FileInternalInformation:
  13596. +        case FileAttributeTagInformation:
  13597. +            break;
  13598. +        default:
  13599. +            print_error("Unhandled/unsupported InfoClass(%d)\n", (int)InfoClass);
  13600. +        }
  13601. +    } else {
  13602. +        status = map_queryfile_error(entry->status);
  13603. +        print_error("status(0x%lx) = map_queryfile_error(entry->status(0x%lx));\n",
  13604. +            (long)status, (long)entry->status);
  13605. +    }
  13606. +    nfs41_UpcallDestroy(entry);
  13607. +out:
  13608. +#ifdef ENABLE_TIMINGS
  13609. +    t2 = KeQueryPerformanceCounter(NULL);
  13610. +    InterlockedIncrement(&getattr.tops);
  13611. +    InterlockedAdd64(&getattr.ticks, t2.QuadPart - t1.QuadPart);
  13612. +#ifdef ENABLE_INDV_TIMINGS
  13613. +    DbgP("nfs41_QueryFileInformation delta = %d op=%d sum=%d\n",
  13614. +        t2.QuadPart - t1.QuadPart, getattr.tops, getattr.ticks);
  13615. +#endif
  13616. +#endif
  13617. +#ifdef DEBUG_FILE_QUERY
  13618. +    DbgEx();
  13619. +    DbgP("<-- nfs41_QueryFileInformation, status=0x%lx\n", (long)status);
  13620. +#endif
  13621. +    return status;
  13622. +}
  13623. +
  13624. +NTSTATUS map_setfile_error(
  13625. +    DWORD error)
  13626. +{
  13627. +    switch (error) {
  13628. +    case NO_ERROR:                      return STATUS_SUCCESS;
  13629. +    case ERROR_NOT_EMPTY:               return STATUS_DIRECTORY_NOT_EMPTY;
  13630. +    case ERROR_FILE_EXISTS:             return STATUS_OBJECT_NAME_COLLISION;
  13631. +    case ERROR_FILE_NOT_FOUND:          return STATUS_OBJECT_NAME_NOT_FOUND;
  13632. +    case ERROR_PATH_NOT_FOUND:          return STATUS_OBJECT_PATH_NOT_FOUND;
  13633. +    case ERROR_ACCESS_DENIED:           return STATUS_ACCESS_DENIED;
  13634. +    case ERROR_FILE_INVALID:            return STATUS_FILE_INVALID;
  13635. +    case ERROR_NOT_SAME_DEVICE:         return STATUS_NOT_SAME_DEVICE;
  13636. +    case ERROR_NOT_SUPPORTED:           return STATUS_NOT_IMPLEMENTED;
  13637. +    case ERROR_NETWORK_ACCESS_DENIED:   return STATUS_NETWORK_ACCESS_DENIED;
  13638. +    case ERROR_NETNAME_DELETED:         return STATUS_NETWORK_NAME_DELETED;
  13639. +    case ERROR_BUFFER_OVERFLOW:         return STATUS_INSUFFICIENT_RESOURCES;
  13640. +    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  13641. +    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  13642. +    case ERROR_FILE_TOO_LARGE:          return STATUS_FILE_TOO_LARGE;
  13643. +    case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  13644. +    default:
  13645. +        print_error("map_setfile_error: "
  13646. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  13647. +            "defaulting to STATUS_INVALID_PARAMETER\n", error);
  13648. +    case ERROR_INVALID_PARAMETER:       return STATUS_INVALID_PARAMETER;
  13649. +    }
  13650. +}
  13651. +
  13652. +static
  13653. +NTSTATUS check_nfs41_setattr_args(
  13654. +    IN PRX_CONTEXT RxContext)
  13655. +{
  13656. +    NTSTATUS status = STATUS_SUCCESS;
  13657. +    FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass;
  13658. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  13659. +        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  13660. +
  13661. +    if (pVNetRootContext->read_only) {
  13662. +        print_error("check_nfs41_setattr_args: Read-only mount\n");
  13663. +        status = STATUS_MEDIA_WRITE_PROTECTED;
  13664. +        goto out;
  13665. +    }
  13666. +
  13667. +    /* http://msdn.microsoft.com/en-us/library/ff469355(v=PROT.10).aspx
  13668. +     * http://msdn.microsoft.com/en-us/library/ff469424(v=PROT.10).aspx
  13669. +     * If Open.GrantedAccess does not contain FILE_WRITE_DATA, the operation
  13670. +     * MUST be failed with STATUS_ACCESS_DENIED.
  13671. +     */
  13672. +    if (InfoClass == FileAllocationInformation ||
  13673. +            InfoClass == FileEndOfFileInformation) {
  13674. +        if (!(RxContext->pRelevantSrvOpen->DesiredAccess & FILE_WRITE_DATA)) {
  13675. +            status = STATUS_ACCESS_DENIED;
  13676. +            goto out;
  13677. +        }
  13678. +    }
  13679. +    status = check_nfs41_dirquery_args(RxContext);
  13680. +    if (status) goto out;
  13681. +
  13682. +    switch (InfoClass) {
  13683. +    case FileRenameInformation:
  13684. +    {
  13685. +        PFILE_RENAME_INFORMATION rinfo =
  13686. +            (PFILE_RENAME_INFORMATION)RxContext->Info.Buffer;
  13687. +        UNICODE_STRING dst = { (USHORT)rinfo->FileNameLength,
  13688. +            (USHORT)rinfo->FileNameLength, rinfo->FileName };
  13689. +#ifdef DEBUG_FILE_SET
  13690. +        DbgP("Attempting to rename to '%wZ'\n", &dst);
  13691. +#endif
  13692. +        if (isFilenameTooLong(&dst, pVNetRootContext)) {
  13693. +            status = STATUS_OBJECT_NAME_INVALID;
  13694. +            goto out;
  13695. +        }
  13696. +        if (rinfo->RootDirectory) {
  13697. +            status = STATUS_INVALID_PARAMETER;
  13698. +            goto out;
  13699. +        }
  13700. +        break;
  13701. +    }
  13702. +    case FileLinkInformation:
  13703. +    {
  13704. +        PFILE_LINK_INFORMATION linfo =
  13705. +            (PFILE_LINK_INFORMATION)RxContext->Info.Buffer;
  13706. +        UNICODE_STRING dst = { (USHORT)linfo->FileNameLength,
  13707. +            (USHORT)linfo->FileNameLength, linfo->FileName };
  13708. +#ifdef DEBUG_FILE_SET
  13709. +        DbgP("Attempting to add link as '%wZ'\n", &dst);
  13710. +#endif
  13711. +        if (isFilenameTooLong(&dst, pVNetRootContext)) {
  13712. +            status = STATUS_OBJECT_NAME_INVALID;
  13713. +            goto out;
  13714. +        }
  13715. +        if (linfo->RootDirectory) {
  13716. +            status = STATUS_INVALID_PARAMETER;
  13717. +            goto out;
  13718. +        }
  13719. +        break;
  13720. +    }
  13721. +    case FileDispositionInformation:
  13722. +    {
  13723. +        PFILE_DISPOSITION_INFORMATION dinfo =
  13724. +            (PFILE_DISPOSITION_INFORMATION)RxContext->Info.Buffer;
  13725. +        __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  13726. +        if (dinfo->DeleteFile && nfs41_fcb->DeletePending) {
  13727. +            status = STATUS_DELETE_PENDING;
  13728. +            goto out;
  13729. +        }
  13730. +        break;
  13731. +    }
  13732. +    case FileBasicInformation:
  13733. +    case FileAllocationInformation:
  13734. +    case FileEndOfFileInformation:
  13735. +        break;
  13736. +    default:
  13737. +        print_error("nfs41_SetFileInformation: unhandled class %d\n", InfoClass);
  13738. +        status = STATUS_NOT_SUPPORTED;
  13739. +    }
  13740. +
  13741. +out:
  13742. +    return status;
  13743. +}
  13744. +
  13745. +NTSTATUS nfs41_SetFileInformation(
  13746. +    IN OUT PRX_CONTEXT RxContext)
  13747. +{
  13748. +    NTSTATUS status = STATUS_INVALID_PARAMETER;
  13749. +    nfs41_updowncall_entry *entry;
  13750. +    FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass;
  13751. +    FILE_RENAME_INFORMATION rinfo;
  13752. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  13753. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  13754. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  13755. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  13756. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  13757. +    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  13758. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  13759. +#ifdef ENABLE_TIMINGS
  13760. +    LARGE_INTEGER t1, t2;
  13761. +    t1 = KeQueryPerformanceCounter(NULL);
  13762. +#endif
  13763. +
  13764. +#ifdef DEBUG_FILE_SET
  13765. +    DbgEn();
  13766. +    print_debug_filedirquery_header(RxContext);
  13767. +#endif
  13768. +
  13769. +    status = check_nfs41_setattr_args(RxContext);
  13770. +    if (status) goto out;
  13771. +
  13772. +    switch (InfoClass) {
  13773. +    case FileDispositionInformation:
  13774. +        {
  13775. +            PFILE_DISPOSITION_INFORMATION dinfo =
  13776. +                (PFILE_DISPOSITION_INFORMATION)RxContext->Info.Buffer;
  13777. +            if (dinfo->DeleteFile) {
  13778. +                nfs41_fcb->DeletePending = TRUE;
  13779. +                // we can delete directories right away
  13780. +                if (nfs41_fcb->StandardInfo.Directory)
  13781. +                    break;
  13782. +                nfs41_fcb->StandardInfo.DeletePending = TRUE;
  13783. +                if (RxContext->pFcb->OpenCount > 1) {
  13784. +                    rinfo.ReplaceIfExists = 0;
  13785. +                    rinfo.RootDirectory = INVALID_HANDLE_VALUE;
  13786. +                    rinfo.FileNameLength = 0;
  13787. +                    rinfo.FileName[0] = L'\0';
  13788. +                    InfoClass = FileRenameInformation;
  13789. +                    nfs41_fcb->Renamed = TRUE;
  13790. +                    break;
  13791. +                }
  13792. +            } else {
  13793. +                /* section 4.3.3 of [FSBO]
  13794. +                 * "file system behavior in the microsoft windows environment"
  13795. +                 */
  13796. +                if (nfs41_fcb->DeletePending) {
  13797. +                    nfs41_fcb->DeletePending = 0;
  13798. +                    nfs41_fcb->StandardInfo.DeletePending = 0;
  13799. +                }
  13800. +            }
  13801. +            status = STATUS_SUCCESS;
  13802. +            goto out;
  13803. +        }
  13804. +    case FileAllocationInformation:
  13805. +        {
  13806. +            PFILE_ALLOCATION_INFORMATION info =
  13807. +                (PFILE_ALLOCATION_INFORMATION)RxContext->Info.Buffer;
  13808. +
  13809. +            nfs41_fcb->StandardInfo.AllocationSize.QuadPart = info->AllocationSize.QuadPart;
  13810. +            if (nfs41_fcb->StandardInfo.EndOfFile.QuadPart > info->AllocationSize.QuadPart) {
  13811. +                nfs41_fcb->StandardInfo.EndOfFile.QuadPart = info->AllocationSize.QuadPart;
  13812. +            }
  13813. +            break;
  13814. +        }
  13815. +    case FileEndOfFileInformation:
  13816. +        {
  13817. +            PFILE_END_OF_FILE_INFORMATION info =
  13818. +                (PFILE_END_OF_FILE_INFORMATION)RxContext->Info.Buffer;
  13819. +
  13820. +            if (info->EndOfFile.QuadPart > nfs41_fcb->StandardInfo.AllocationSize.QuadPart) {
  13821. +                nfs41_fcb->StandardInfo.AllocationSize.QuadPart =
  13822. +                    nfs41_fcb->StandardInfo.EndOfFile.QuadPart = info->EndOfFile.QuadPart;
  13823. +            }
  13824. +            else {
  13825. +                nfs41_fcb->StandardInfo.EndOfFile.QuadPart = info->EndOfFile.QuadPart;
  13826. +            }
  13827. +            break;
  13828. +        }
  13829. +    case FileRenameInformation:
  13830. +        {
  13831. +            /* noop if filename and destination are the same */
  13832. +            PFILE_RENAME_INFORMATION prinfo =
  13833. +                (PFILE_RENAME_INFORMATION)RxContext->Info.Buffer;
  13834. +            const UNICODE_STRING dst = { (USHORT)prinfo->FileNameLength,
  13835. +                (USHORT)prinfo->FileNameLength, prinfo->FileName };
  13836. +            if (RtlCompareUnicodeString(&dst,
  13837. +                    SrvOpen->pAlreadyPrefixedName, FALSE) == 0) {
  13838. +                status = STATUS_SUCCESS;
  13839. +                goto out;
  13840. +            }
  13841. +        }
  13842. +    }
  13843. +
  13844. +    status = nfs41_UpcallCreate(NFS41_FILE_SET, &nfs41_fobx->sec_ctx,
  13845. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  13846. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  13847. +    if (status) goto out;
  13848. +
  13849. +    entry->u.SetFile.InfoClass = InfoClass;
  13850. +
  13851. +    /* original irp has infoclass for remove but we need to rename instead,
  13852. +     * thus we changed the local variable infoclass */
  13853. +    if (RxContext->Info.FileInformationClass == FileDispositionInformation &&
  13854. +            InfoClass == FileRenameInformation) {
  13855. +        entry->buf = &rinfo;
  13856. +        entry->buf_len = sizeof(rinfo);
  13857. +    } else {
  13858. +        entry->buf = RxContext->Info.Buffer;
  13859. +        entry->buf_len = RxContext->Info.Length;
  13860. +    }
  13861. +#ifdef ENABLE_TIMINGS
  13862. +    InterlockedIncrement(&setattr.sops);
  13863. +    InterlockedAdd64(&setattr.size, entry->u.SetFile.buf_len);
  13864. +#endif
  13865. +
  13866. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  13867. +    if (status) goto out;
  13868. +
  13869. +    status = map_setfile_error(entry->status);
  13870. +    if (!status) {
  13871. +        if (!nfs41_fobx->deleg_type && entry->ChangeTime &&
  13872. +                (SrvOpen->DesiredAccess &
  13873. +                (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA)))
  13874. +            nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime);
  13875. +        nfs41_fcb->changeattr = entry->ChangeTime;
  13876. +    }
  13877. +    nfs41_UpcallDestroy(entry);
  13878. +out:
  13879. +#ifdef ENABLE_TIMINGS
  13880. +    t2 = KeQueryPerformanceCounter(NULL);
  13881. +    InterlockedIncrement(&setattr.tops);
  13882. +    InterlockedAdd64(&setattr.ticks, t2.QuadPart - t1.QuadPart);
  13883. +#ifdef ENABLE_INDV_TIMINGS
  13884. +    DbgP("nfs41_SetFileInformation delta = %d op=%d sum=%d\n",
  13885. +        t2.QuadPart - t1.QuadPart, setattr.tops, setattr.ticks);
  13886. +#endif
  13887. +#endif
  13888. +#ifdef DEBUG_FILE_SET
  13889. +    DbgEx();
  13890. +#endif
  13891. +    return status;
  13892. +}
  13893. +
  13894. +NTSTATUS nfs41_SetFileInformationAtCleanup(
  13895. +      IN OUT PRX_CONTEXT RxContext)
  13896. +{
  13897. +    NTSTATUS status;
  13898. +    DbgEn();
  13899. +    status = nfs41_SetFileInformation(RxContext);
  13900. +    DbgEx();
  13901. +    return status;
  13902. +}
  13903. diff --git a/sys/nfs41sys_fsctl.c b/sys/nfs41sys_fsctl.c
  13904. new file mode 100644
  13905. index 0000000..adb8837
  13906. --- /dev/null
  13907. +++ b/sys/nfs41sys_fsctl.c
  13908. @@ -0,0 +1,111 @@
  13909. +/* NFSv4.1 client for Windows
  13910. + * Copyright (C) 2012 The Regents of the University of Michigan
  13911. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  13912. + *
  13913. + * Olga Kornievskaia <aglo@umich.edu>
  13914. + * Casey Bodley <cbodley@umich.edu>
  13915. + * Roland Mainz <roland.mainz@nrubsig.org>
  13916. + *
  13917. + * This library is free software; you can redistribute it and/or modify it
  13918. + * under the terms of the GNU Lesser General Public License as published by
  13919. + * the Free Software Foundation; either version 2.1 of the License, or (at
  13920. + * your option) any later version.
  13921. + *
  13922. + * This library is distributed in the hope that it will be useful, but
  13923. + * without any warranty; without even the implied warranty of merchantability
  13924. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  13925. + * License for more details.
  13926. + *
  13927. + * You should have received a copy of the GNU Lesser General Public License
  13928. + * along with this library; if not, write to the Free Software Foundation,
  13929. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  13930. + */
  13931. +
  13932. +#ifndef _KERNEL_MODE
  13933. +#error module requires kernel mode
  13934. +#endif
  13935. +
  13936. +#if ((__STDC_VERSION__-0) < 201710L)
  13937. +#error Code requires ISO C17
  13938. +#endif
  13939. +
  13940. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  13941. +#if _MSC_VER >= 1900
  13942. +#if defined(_WIN64) && defined(_M_X64)
  13943. +#ifndef _AMD64_
  13944. +#define _AMD64_
  13945. +#endif
  13946. +#elif defined(_WIN32) && defined(_M_IX86)
  13947. +#ifndef _X86_
  13948. +#define _X86_
  13949. +#endif
  13950. +#elif defined(_WIN64) && defined(_M_ARM64)
  13951. +#ifndef _ARM64_
  13952. +#define _ARM64_
  13953. +#endif
  13954. +#elif defined(_WIN32) && defined(_M_ARM)
  13955. +#ifndef _ARM_
  13956. +#define _ARM_
  13957. +#endif
  13958. +#else
  13959. +#error Unsupported arch
  13960. +#endif
  13961. +#endif /* _MSC_VER >= 1900 */
  13962. +
  13963. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  13964. +#include <rx.h>
  13965. +#include <windef.h>
  13966. +#include <winerror.h>
  13967. +
  13968. +#include <Ntstrsafe.h>
  13969. +
  13970. +#include "nfs41sys_buildconfig.h"
  13971. +
  13972. +#include "nfs41_driver.h"
  13973. +#include "nfs41sys_debug.h"
  13974. +#include "nfs41_build_features.h"
  13975. +
  13976. +#include "nfs41sys_driver.h"
  13977. +#include "nfs41sys_util.h"
  13978. +
  13979. +
  13980. +NTSTATUS nfs41_FsCtl(
  13981. +    IN OUT PRX_CONTEXT RxContext)
  13982. +{
  13983. +    NTSTATUS status = STATUS_INVALID_DEVICE_REQUEST;
  13984. +#ifdef DEBUG_FSCTL
  13985. +    DbgEn();
  13986. +    print_debug_header(RxContext);
  13987. +#endif
  13988. +    const ULONG fscontrolcode =
  13989. +        RxContext->LowIoContext.ParamsFor.FsCtl.FsControlCode;
  13990. +
  13991. +    switch (fscontrolcode) {
  13992. +    case FSCTL_SET_REPARSE_POINT:
  13993. +        status = nfs41_SetReparsePoint(RxContext);
  13994. +        break;
  13995. +    case FSCTL_GET_REPARSE_POINT:
  13996. +        status = nfs41_GetReparsePoint(RxContext);
  13997. +        break;
  13998. +    default:
  13999. +        break;
  14000. +    }
  14001. +
  14002. +#ifdef DEBUG_FSCTL
  14003. +    const char *fsctl_str = fsctl2string(fscontrolcode);
  14004. +
  14005. +    if (fsctl_str) {
  14006. +        DbgP("nfs41_FsCtl: FsControlCode='%s', status=0x%lx\n",
  14007. +            fsctl_str, (long)status);
  14008. +    }
  14009. +    else {
  14010. +        DbgP("nfs41_FsCtl: FsControlCode=0x%lx, status=0x%lx\n",
  14011. +            (unsigned long)fscontrolcode, (long)status);
  14012. +    }
  14013. +#endif /* DEBUG_FSCTL */
  14014. +
  14015. +#ifdef DEBUG_FSCTL
  14016. +    DbgEx();
  14017. +#endif
  14018. +    return status;
  14019. +}
  14020. diff --git a/sys/nfs41sys_ioctl.c b/sys/nfs41sys_ioctl.c
  14021. new file mode 100644
  14022. index 0000000..56da370
  14023. --- /dev/null
  14024. +++ b/sys/nfs41sys_ioctl.c
  14025. @@ -0,0 +1,90 @@
  14026. +/* NFSv4.1 client for Windows
  14027. + * Copyright (C) 2012 The Regents of the University of Michigan
  14028. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  14029. + *
  14030. + * Olga Kornievskaia <aglo@umich.edu>
  14031. + * Casey Bodley <cbodley@umich.edu>
  14032. + * Roland Mainz <roland.mainz@nrubsig.org>
  14033. + *
  14034. + * This library is free software; you can redistribute it and/or modify it
  14035. + * under the terms of the GNU Lesser General Public License as published by
  14036. + * the Free Software Foundation; either version 2.1 of the License, or (at
  14037. + * your option) any later version.
  14038. + *
  14039. + * This library is distributed in the hope that it will be useful, but
  14040. + * without any warranty; without even the implied warranty of merchantability
  14041. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  14042. + * License for more details.
  14043. + *
  14044. + * You should have received a copy of the GNU Lesser General Public License
  14045. + * along with this library; if not, write to the Free Software Foundation,
  14046. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  14047. + */
  14048. +
  14049. +#ifndef _KERNEL_MODE
  14050. +#error module requires kernel mode
  14051. +#endif
  14052. +
  14053. +#if ((__STDC_VERSION__-0) < 201710L)
  14054. +#error Code requires ISO C17
  14055. +#endif
  14056. +
  14057. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  14058. +#if _MSC_VER >= 1900
  14059. +#if defined(_WIN64) && defined(_M_X64)
  14060. +#ifndef _AMD64_
  14061. +#define _AMD64_
  14062. +#endif
  14063. +#elif defined(_WIN32) && defined(_M_IX86)
  14064. +#ifndef _X86_
  14065. +#define _X86_
  14066. +#endif
  14067. +#elif defined(_WIN64) && defined(_M_ARM64)
  14068. +#ifndef _ARM64_
  14069. +#define _ARM64_
  14070. +#endif
  14071. +#elif defined(_WIN32) && defined(_M_ARM)
  14072. +#ifndef _ARM_
  14073. +#define _ARM_
  14074. +#endif
  14075. +#else
  14076. +#error Unsupported arch
  14077. +#endif
  14078. +#endif /* _MSC_VER >= 1900 */
  14079. +
  14080. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  14081. +#include <rx.h>
  14082. +#include <windef.h>
  14083. +#include <winerror.h>
  14084. +
  14085. +#include <Ntstrsafe.h>
  14086. +
  14087. +#include "nfs41sys_buildconfig.h"
  14088. +
  14089. +#include "nfs41_driver.h"
  14090. +#include "nfs41sys_debug.h"
  14091. +#include "nfs41_build_features.h"
  14092. +
  14093. +#include "nfs41sys_driver.h"
  14094. +#include "nfs41sys_util.h"
  14095. +
  14096. +
  14097. +NTSTATUS nfs41_IoCtl(
  14098. +    IN OUT PRX_CONTEXT RxContext)
  14099. +{
  14100. +    NTSTATUS status = STATUS_INVALID_DEVICE_REQUEST;
  14101. +#ifdef DEBUG_IOCTL
  14102. +    DbgEn();
  14103. +    print_debug_header(RxContext);
  14104. +#endif /* DEBUG_IOCTL */
  14105. +    const ULONG iocontrolcode =
  14106. +        RxContext->LowIoContext.ParamsFor.IoCtl.IoControlCode;
  14107. +
  14108. +    DbgP("nfs41_IoCtl: IoControlCode=0x%lx, status=0x%lx\n",
  14109. +        (unsigned long)iocontrolcode, (long)status);
  14110. +
  14111. +#ifdef DEBUG_IOCTL
  14112. +    DbgEx();
  14113. +#endif
  14114. +    return status;
  14115. +}
  14116. diff --git a/sys/nfs41sys_lock.c b/sys/nfs41sys_lock.c
  14117. new file mode 100644
  14118. index 0000000..208fc6d
  14119. --- /dev/null
  14120. +++ b/sys/nfs41sys_lock.c
  14121. @@ -0,0 +1,398 @@
  14122. +/* NFSv4.1 client for Windows
  14123. + * Copyright (C) 2012 The Regents of the University of Michigan
  14124. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  14125. + *
  14126. + * Olga Kornievskaia <aglo@umich.edu>
  14127. + * Casey Bodley <cbodley@umich.edu>
  14128. + * Roland Mainz <roland.mainz@nrubsig.org>
  14129. + *
  14130. + * This library is free software; you can redistribute it and/or modify it
  14131. + * under the terms of the GNU Lesser General Public License as published by
  14132. + * the Free Software Foundation; either version 2.1 of the License, or (at
  14133. + * your option) any later version.
  14134. + *
  14135. + * This library is distributed in the hope that it will be useful, but
  14136. + * without any warranty; without even the implied warranty of merchantability
  14137. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  14138. + * License for more details.
  14139. + *
  14140. + * You should have received a copy of the GNU Lesser General Public License
  14141. + * along with this library; if not, write to the Free Software Foundation,
  14142. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  14143. + */
  14144. +
  14145. +#ifndef _KERNEL_MODE
  14146. +#error module requires kernel mode
  14147. +#endif
  14148. +
  14149. +#if ((__STDC_VERSION__-0) < 201710L)
  14150. +#error Code requires ISO C17
  14151. +#endif
  14152. +
  14153. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  14154. +#if _MSC_VER >= 1900
  14155. +#if defined(_WIN64) && defined(_M_X64)
  14156. +#ifndef _AMD64_
  14157. +#define _AMD64_
  14158. +#endif
  14159. +#elif defined(_WIN32) && defined(_M_IX86)
  14160. +#ifndef _X86_
  14161. +#define _X86_
  14162. +#endif
  14163. +#elif defined(_WIN64) && defined(_M_ARM64)
  14164. +#ifndef _ARM64_
  14165. +#define _ARM64_
  14166. +#endif
  14167. +#elif defined(_WIN32) && defined(_M_ARM)
  14168. +#ifndef _ARM_
  14169. +#define _ARM_
  14170. +#endif
  14171. +#else
  14172. +#error Unsupported arch
  14173. +#endif
  14174. +#endif /* _MSC_VER >= 1900 */
  14175. +
  14176. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  14177. +#include <rx.h>
  14178. +#include <windef.h>
  14179. +#include <winerror.h>
  14180. +
  14181. +#include <Ntstrsafe.h>
  14182. +
  14183. +#include "nfs41sys_buildconfig.h"
  14184. +
  14185. +#include "nfs41_driver.h"
  14186. +#include "nfs41sys_debug.h"
  14187. +#include "nfs41_build_features.h"
  14188. +
  14189. +#include "nfs41sys_driver.h"
  14190. +#include "nfs41sys_util.h"
  14191. +
  14192. +
  14193. +NTSTATUS marshal_nfs41_lock(
  14194. +    nfs41_updowncall_entry *entry,
  14195. +    unsigned char *buf,
  14196. +    ULONG buf_len,
  14197. +    ULONG *len)
  14198. +{
  14199. +    NTSTATUS status = STATUS_SUCCESS;
  14200. +    ULONG header_len = 0;
  14201. +    unsigned char *tmp = buf;
  14202. +
  14203. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  14204. +    if (status) goto out;
  14205. +    else tmp += *len;
  14206. +
  14207. +    header_len = *len + 2 * sizeof(LONGLONG) + 2 * sizeof(BOOLEAN);
  14208. +    if (header_len > buf_len) {
  14209. +        status = STATUS_INSUFFICIENT_RESOURCES;
  14210. +        goto out;
  14211. +    }
  14212. +    RtlCopyMemory(tmp, &entry->u.Lock.offset, sizeof(LONGLONG));
  14213. +    tmp += sizeof(LONGLONG);
  14214. +    RtlCopyMemory(tmp, &entry->u.Lock.length, sizeof(LONGLONG));
  14215. +    tmp += sizeof(LONGLONG);
  14216. +    RtlCopyMemory(tmp, &entry->u.Lock.exclusive, sizeof(BOOLEAN));
  14217. +    tmp += sizeof(BOOLEAN);
  14218. +    RtlCopyMemory(tmp, &entry->u.Lock.blocking, sizeof(BOOLEAN));
  14219. +    *len = header_len;
  14220. +
  14221. +#ifdef DEBUG_MARSHAL_DETAIL
  14222. +    DbgP("marshal_nfs41_lock: "
  14223. +        "offset=0x%llx length=0x%llx exclusive=%u "
  14224. +        "blocking=%u\n", entry->u.Lock.offset, entry->u.Lock.length,
  14225. +        entry->u.Lock.exclusive, entry->u.Lock.blocking);
  14226. +#endif
  14227. +out:
  14228. +    return status;
  14229. +}
  14230. +
  14231. +NTSTATUS marshal_nfs41_unlock(
  14232. +    nfs41_updowncall_entry *entry,
  14233. +    unsigned char *buf,
  14234. +    ULONG buf_len,
  14235. +    ULONG *len)
  14236. +{
  14237. +    NTSTATUS status = STATUS_SUCCESS;
  14238. +    ULONG header_len = 0;
  14239. +    unsigned char *tmp = buf;
  14240. +    PLOWIO_LOCK_LIST lock;
  14241. +
  14242. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  14243. +    if (status) goto out;
  14244. +    else tmp += *len;
  14245. +
  14246. +    header_len = *len + sizeof(ULONG) +
  14247. +        (size_t)entry->u.Unlock.count * 2 * sizeof(LONGLONG);
  14248. +    if (header_len > buf_len) {
  14249. +        status = STATUS_INSUFFICIENT_RESOURCES;
  14250. +        goto out;
  14251. +    }
  14252. +    RtlCopyMemory(tmp, &entry->u.Unlock.count, sizeof(ULONG));
  14253. +    tmp += sizeof(ULONG);
  14254. +
  14255. +    lock = &entry->u.Unlock.locks;
  14256. +    while (lock) {
  14257. +        RtlCopyMemory(tmp, &lock->ByteOffset, sizeof(LONGLONG));
  14258. +        tmp += sizeof(LONGLONG);
  14259. +        RtlCopyMemory(tmp, &lock->Length, sizeof(LONGLONG));
  14260. +        tmp += sizeof(LONGLONG);
  14261. +        lock = lock->Next;
  14262. +    }
  14263. +    *len = header_len;
  14264. +
  14265. +#ifdef DEBUG_MARSHAL_DETAIL
  14266. +    DbgP("marshal_nfs41_unlock: count=%u\n", entry->u.Unlock.count);
  14267. +#endif
  14268. +out:
  14269. +    return status;
  14270. +}
  14271. +
  14272. +NTSTATUS nfs41_IsLockRealizable(
  14273. +    IN OUT PMRX_FCB pFcb,
  14274. +    IN PLARGE_INTEGER  ByteOffset,
  14275. +    IN PLARGE_INTEGER  Length,
  14276. +    IN ULONG  LowIoLockFlags)
  14277. +{
  14278. +    NTSTATUS status = STATUS_SUCCESS;
  14279. +#ifdef DEBUG_LOCK
  14280. +    DbgEn();
  14281. +    DbgP("offset 0x%llx, length 0x%llx, exclusive=%u, blocking=%u\n",
  14282. +        ByteOffset->QuadPart,Length->QuadPart,
  14283. +        BooleanFlagOn(LowIoLockFlags, SL_EXCLUSIVE_LOCK),
  14284. +        !BooleanFlagOn(LowIoLockFlags, SL_FAIL_IMMEDIATELY));
  14285. +#endif
  14286. +
  14287. +    /* NFS lock operations with length=0 MUST fail with NFS4ERR_INVAL */
  14288. +    if (Length->QuadPart == 0)
  14289. +        status = STATUS_NOT_SUPPORTED;
  14290. +
  14291. +#ifdef DEBUG_LOCK
  14292. +    DbgEx();
  14293. +#endif
  14294. +    return status;
  14295. +}
  14296. +
  14297. +NTSTATUS map_lock_errors(
  14298. +    DWORD status)
  14299. +{
  14300. +    switch (status) {
  14301. +    case NO_ERROR:                  return STATUS_SUCCESS;
  14302. +    case ERROR_NETNAME_DELETED:     return STATUS_NETWORK_NAME_DELETED;
  14303. +    case ERROR_LOCK_FAILED:         return STATUS_LOCK_NOT_GRANTED;
  14304. +    case ERROR_NOT_LOCKED:          return STATUS_RANGE_NOT_LOCKED;
  14305. +    case ERROR_ATOMIC_LOCKS_NOT_SUPPORTED: return STATUS_UNSUCCESSFUL;
  14306. +    case ERROR_OUTOFMEMORY:         return STATUS_INSUFFICIENT_RESOURCES;
  14307. +    case ERROR_SHARING_VIOLATION:   return STATUS_SHARING_VIOLATION;
  14308. +    case ERROR_FILE_INVALID:        return STATUS_FILE_INVALID;
  14309. +    /* if we return ERROR_INVALID_PARAMETER, Windows translates that to
  14310. +     * success!! */
  14311. +    case ERROR_INVALID_PARAMETER:   return STATUS_LOCK_NOT_GRANTED;
  14312. +    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  14313. +    default:
  14314. +        print_error("map_lock_errors: "
  14315. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  14316. +            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status);
  14317. +    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  14318. +    }
  14319. +}
  14320. +
  14321. +static void print_lock_args(
  14322. +    PRX_CONTEXT RxContext)
  14323. +{
  14324. +    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  14325. +    const ULONG flags = LowIoContext->ParamsFor.Locks.Flags;
  14326. +    print_debug_header(RxContext);
  14327. +    DbgP("offset 0x%llx, length 0x%llx, exclusive=%u, blocking=%u\n",
  14328. +        LowIoContext->ParamsFor.Locks.ByteOffset,
  14329. +        LowIoContext->ParamsFor.Locks.Length,
  14330. +        BooleanFlagOn(flags, SL_EXCLUSIVE_LOCK),
  14331. +        !BooleanFlagOn(flags, SL_FAIL_IMMEDIATELY));
  14332. +}
  14333. +
  14334. +
  14335. +/* use exponential backoff between polls for blocking locks */
  14336. +#define MSEC_TO_RELATIVE_WAIT   (-10000)
  14337. +#define MIN_LOCK_POLL_WAIT      (500 * MSEC_TO_RELATIVE_WAIT) /* 500ms */
  14338. +#define MAX_LOCK_POLL_WAIT      (30000 * MSEC_TO_RELATIVE_WAIT) /* 30s */
  14339. +
  14340. +static void denied_lock_backoff(
  14341. +    IN OUT PLARGE_INTEGER delay)
  14342. +{
  14343. +    if (delay->QuadPart == 0)
  14344. +        delay->QuadPart = MIN_LOCK_POLL_WAIT;
  14345. +    else
  14346. +        delay->QuadPart <<= 1;
  14347. +
  14348. +    if (delay->QuadPart < MAX_LOCK_POLL_WAIT)
  14349. +        delay->QuadPart = MAX_LOCK_POLL_WAIT;
  14350. +}
  14351. +
  14352. +NTSTATUS nfs41_Lock(
  14353. +    IN OUT PRX_CONTEXT RxContext)
  14354. +{
  14355. +    NTSTATUS status = STATUS_SUCCESS;
  14356. +    nfs41_updowncall_entry *entry;
  14357. +    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  14358. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  14359. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  14360. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  14361. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  14362. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  14363. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  14364. +    const ULONG flags = LowIoContext->ParamsFor.Locks.Flags;
  14365. +    LARGE_INTEGER poll_delay = {0};
  14366. +#ifdef ENABLE_TIMINGS
  14367. +    LARGE_INTEGER t1, t2;
  14368. +    t1 = KeQueryPerformanceCounter(NULL);
  14369. +#endif
  14370. +
  14371. +    poll_delay.QuadPart = 0;
  14372. +
  14373. +#ifdef DEBUG_LOCK
  14374. +    DbgEn();
  14375. +    print_lock_args(RxContext);
  14376. +#endif
  14377. +
  14378. +/*  RxReleaseFcbResourceForThreadInMRx(RxContext, RxContext->pFcb,
  14379. +        LowIoContext->ResourceThreadId); */
  14380. +
  14381. +    status = nfs41_UpcallCreate(NFS41_LOCK, &nfs41_fobx->sec_ctx,
  14382. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  14383. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  14384. +    if (status) goto out;
  14385. +
  14386. +    entry->u.Lock.offset = LowIoContext->ParamsFor.Locks.ByteOffset;
  14387. +    entry->u.Lock.length = LowIoContext->ParamsFor.Locks.Length;
  14388. +    entry->u.Lock.exclusive = BooleanFlagOn(flags, SL_EXCLUSIVE_LOCK);
  14389. +    entry->u.Lock.blocking = !BooleanFlagOn(flags, SL_FAIL_IMMEDIATELY);
  14390. +
  14391. +retry_upcall:
  14392. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  14393. +    if (status) goto out;
  14394. +
  14395. +    /* blocking locks keep trying until it succeeds */
  14396. +    if (entry->status == ERROR_LOCK_FAILED && entry->u.Lock.blocking) {
  14397. +        denied_lock_backoff(&poll_delay);
  14398. +        DbgP("returned ERROR_LOCK_FAILED; retrying in %llums\n",
  14399. +            poll_delay.QuadPart / MSEC_TO_RELATIVE_WAIT);
  14400. +        KeDelayExecutionThread(KernelMode, FALSE, &poll_delay);
  14401. +        entry->state = NFS41_WAITING_FOR_UPCALL;
  14402. +        goto retry_upcall;
  14403. +    }
  14404. +
  14405. +    status = map_lock_errors(entry->status);
  14406. +    RxContext->CurrentIrp->IoStatus.Status = status;
  14407. +
  14408. +    nfs41_UpcallDestroy(entry);
  14409. +out:
  14410. +#ifdef ENABLE_TIMINGS
  14411. +    t2 = KeQueryPerformanceCounter(NULL);
  14412. +    InterlockedIncrement(&lock.tops);
  14413. +    InterlockedAdd64(&lock.ticks, t2.QuadPart - t1.QuadPart);
  14414. +#ifdef ENABLE_INDV_TIMINGS
  14415. +    DbgP("nfs41_Lock delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart,
  14416. +        lock.tops, lock.ticks);
  14417. +#endif
  14418. +#endif
  14419. +#ifdef DEBUG_LOCK
  14420. +    DbgEx();
  14421. +#endif
  14422. +    return status;
  14423. +}
  14424. +
  14425. +static void print_unlock_args(
  14426. +    PRX_CONTEXT RxContext)
  14427. +{
  14428. +    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  14429. +    print_debug_header(RxContext);
  14430. +    if (LowIoContext->Operation == LOWIO_OP_UNLOCK_MULTIPLE) {
  14431. +        PLOWIO_LOCK_LIST lock = LowIoContext->ParamsFor.Locks.LockList;
  14432. +        DbgP("LOWIO_OP_UNLOCK_MULTIPLE:");
  14433. +        while (lock) {
  14434. +            DbgP(" (offset=%llu, length=%llu)", lock->ByteOffset, lock->Length);
  14435. +            lock = lock->Next;
  14436. +        }
  14437. +        DbgP("\n");
  14438. +    } else {
  14439. +        DbgP("LOWIO_OP_UNLOCK: offset=%llu, length=%llu\n",
  14440. +            LowIoContext->ParamsFor.Locks.ByteOffset,
  14441. +            LowIoContext->ParamsFor.Locks.Length);
  14442. +    }
  14443. +}
  14444. +
  14445. +__inline ULONG unlock_list_count(
  14446. +    PLOWIO_LOCK_LIST lock)
  14447. +{
  14448. +    ULONG count = 0;
  14449. +    while (lock) {
  14450. +        count++;
  14451. +        lock = lock->Next;
  14452. +    }
  14453. +    return count;
  14454. +}
  14455. +
  14456. +NTSTATUS nfs41_Unlock(
  14457. +    IN OUT PRX_CONTEXT RxContext)
  14458. +{
  14459. +    NTSTATUS status = STATUS_SUCCESS;
  14460. +    nfs41_updowncall_entry *entry;
  14461. +    PLOWIO_CONTEXT LowIoContext  = &RxContext->LowIoContext;
  14462. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  14463. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  14464. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  14465. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  14466. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  14467. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  14468. +#ifdef ENABLE_TIMINGS
  14469. +    LARGE_INTEGER t1, t2;
  14470. +    t1 = KeQueryPerformanceCounter(NULL);
  14471. +#endif
  14472. +#ifdef DEBUG_LOCK
  14473. +    DbgEn();
  14474. +    print_lock_args(RxContext);
  14475. +#endif
  14476. +
  14477. +/*  RxReleaseFcbResourceForThreadInMRx(RxContext, RxContext->pFcb,
  14478. +        LowIoContext->ResourceThreadId); */
  14479. +
  14480. +    status = nfs41_UpcallCreate(NFS41_UNLOCK, &nfs41_fobx->sec_ctx,
  14481. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  14482. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  14483. +    if (status) goto out;
  14484. +
  14485. +    if (LowIoContext->Operation == LOWIO_OP_UNLOCK_MULTIPLE) {
  14486. +        entry->u.Unlock.count = unlock_list_count(
  14487. +            LowIoContext->ParamsFor.Locks.LockList);
  14488. +        RtlCopyMemory(&entry->u.Unlock.locks,
  14489. +            LowIoContext->ParamsFor.Locks.LockList,
  14490. +            sizeof(LOWIO_LOCK_LIST));
  14491. +    } else {
  14492. +        entry->u.Unlock.count = 1;
  14493. +        entry->u.Unlock.locks.ByteOffset =
  14494. +            LowIoContext->ParamsFor.Locks.ByteOffset;
  14495. +        entry->u.Unlock.locks.Length =
  14496. +            LowIoContext->ParamsFor.Locks.Length;
  14497. +    }
  14498. +
  14499. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  14500. +    if (status) goto out;
  14501. +
  14502. +    status = map_lock_errors(entry->status);
  14503. +    RxContext->CurrentIrp->IoStatus.Status = status;
  14504. +    nfs41_UpcallDestroy(entry);
  14505. +out:
  14506. +#ifdef ENABLE_TIMINGS
  14507. +    t2 = KeQueryPerformanceCounter(NULL);
  14508. +    InterlockedIncrement(&unlock.tops);
  14509. +    InterlockedAdd64(&unlock.ticks, t2.QuadPart - t1.QuadPart);
  14510. +#ifdef ENABLE_INDV_TIMINGS
  14511. +    DbgP("nfs41_Unlock delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart,
  14512. +        unlock.tops, unlock.ticks);
  14513. +#endif
  14514. +#endif
  14515. +#ifdef DEBUG_LOCK
  14516. +    DbgEx();
  14517. +#endif
  14518. +    return status;
  14519. +}
  14520. diff --git a/sys/nfs41sys_mount.c b/sys/nfs41sys_mount.c
  14521. new file mode 100644
  14522. index 0000000..e062df6
  14523. --- /dev/null
  14524. +++ b/sys/nfs41sys_mount.c
  14525. @@ -0,0 +1,1499 @@
  14526. +/* NFSv4.1 client for Windows
  14527. + * Copyright (C) 2012 The Regents of the University of Michigan
  14528. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  14529. + *
  14530. + * Olga Kornievskaia <aglo@umich.edu>
  14531. + * Casey Bodley <cbodley@umich.edu>
  14532. + * Roland Mainz <roland.mainz@nrubsig.org>
  14533. + *
  14534. + * This library is free software; you can redistribute it and/or modify it
  14535. + * under the terms of the GNU Lesser General Public License as published by
  14536. + * the Free Software Foundation; either version 2.1 of the License, or (at
  14537. + * your option) any later version.
  14538. + *
  14539. + * This library is distributed in the hope that it will be useful, but
  14540. + * without any warranty; without even the implied warranty of merchantability
  14541. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  14542. + * License for more details.
  14543. + *
  14544. + * You should have received a copy of the GNU Lesser General Public License
  14545. + * along with this library; if not, write to the Free Software Foundation,
  14546. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  14547. + */
  14548. +
  14549. +#ifndef _KERNEL_MODE
  14550. +#error module requires kernel mode
  14551. +#endif
  14552. +
  14553. +#if ((__STDC_VERSION__-0) < 201710L)
  14554. +#error Code requires ISO C17
  14555. +#endif
  14556. +
  14557. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  14558. +#if _MSC_VER >= 1900
  14559. +#if defined(_WIN64) && defined(_M_X64)
  14560. +#ifndef _AMD64_
  14561. +#define _AMD64_
  14562. +#endif
  14563. +#elif defined(_WIN32) && defined(_M_IX86)
  14564. +#ifndef _X86_
  14565. +#define _X86_
  14566. +#endif
  14567. +#elif defined(_WIN64) && defined(_M_ARM64)
  14568. +#ifndef _ARM64_
  14569. +#define _ARM64_
  14570. +#endif
  14571. +#elif defined(_WIN32) && defined(_M_ARM)
  14572. +#ifndef _ARM_
  14573. +#define _ARM_
  14574. +#endif
  14575. +#else
  14576. +#error Unsupported arch
  14577. +#endif
  14578. +#endif /* _MSC_VER >= 1900 */
  14579. +
  14580. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  14581. +#include <rx.h>
  14582. +#include <windef.h>
  14583. +#include <winerror.h>
  14584. +
  14585. +#include <Ntstrsafe.h>
  14586. +
  14587. +#include "nfs41sys_buildconfig.h"
  14588. +
  14589. +#include "nfs41_driver.h"
  14590. +#include "nfs41sys_debug.h"
  14591. +#include "nfs41_build_features.h"
  14592. +
  14593. +#include "nfs41sys_driver.h"
  14594. +#include "nfs41sys_util.h"
  14595. +
  14596. +void copy_nfs41_mount_config(NFS41_MOUNT_CONFIG *dest,
  14597. +    NFS41_MOUNT_CONFIG *src)
  14598. +{
  14599. +    RtlCopyMemory(dest, src, sizeof(NFS41_MOUNT_CONFIG));
  14600. +    dest->SrvName.Buffer = dest->srv_buffer;
  14601. +    dest->MntPt.Buffer = dest->mntpt_buffer;
  14602. +    dest->SecFlavor.Buffer = dest->sec_flavor_buffer;
  14603. +}
  14604. +
  14605. +static const char *secflavorop2name(
  14606. +    DWORD sec_flavor)
  14607. +{
  14608. +    switch(sec_flavor) {
  14609. +    case RPCSEC_AUTH_SYS:      return "AUTH_SYS";
  14610. +    case RPCSEC_AUTHGSS_KRB5:  return "AUTHGSS_KRB5";
  14611. +    case RPCSEC_AUTHGSS_KRB5I: return "AUTHGSS_KRB5I";
  14612. +    case RPCSEC_AUTHGSS_KRB5P: return "AUTHGSS_KRB5P";
  14613. +    }
  14614. +
  14615. +    return "UNKNOWN FLAVOR";
  14616. +}
  14617. +
  14618. +NTSTATUS marshal_nfs41_mount(
  14619. +    nfs41_updowncall_entry *entry,
  14620. +    unsigned char *buf,
  14621. +    ULONG buf_len,
  14622. +    ULONG *len)
  14623. +{
  14624. +    NTSTATUS status = STATUS_SUCCESS;
  14625. +    ULONG header_len = 0;
  14626. +    unsigned char *tmp = buf;
  14627. +
  14628. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  14629. +    if (status) goto out;
  14630. +    else tmp += *len;
  14631. +
  14632. +    /* 03/25/2011: Kernel crash to nfsd not running but mount upcall cued up */
  14633. +    if (!MmIsAddressValid(entry->u.Mount.srv_name) ||
  14634. +            !MmIsAddressValid(entry->u.Mount.root)) {
  14635. +        status = STATUS_INTERNAL_ERROR;
  14636. +        goto out;
  14637. +    }
  14638. +    header_len = *len + length_as_utf8(entry->u.Mount.srv_name) +
  14639. +        length_as_utf8(entry->u.Mount.root) + 4 * sizeof(DWORD);
  14640. +    if (header_len > buf_len) {
  14641. +        status = STATUS_INSUFFICIENT_RESOURCES;
  14642. +        goto out;
  14643. +    }
  14644. +    status = marshall_unicode_as_utf8(&tmp, entry->u.Mount.srv_name);
  14645. +    if (status) goto out;
  14646. +    status = marshall_unicode_as_utf8(&tmp, entry->u.Mount.root);
  14647. +    if (status) goto out;
  14648. +    RtlCopyMemory(tmp, &entry->u.Mount.sec_flavor, sizeof(DWORD));
  14649. +    tmp += sizeof(DWORD);
  14650. +    RtlCopyMemory(tmp, &entry->u.Mount.rsize, sizeof(DWORD));
  14651. +    tmp += sizeof(DWORD);
  14652. +    RtlCopyMemory(tmp, &entry->u.Mount.wsize, sizeof(DWORD));
  14653. +    tmp += sizeof(DWORD);
  14654. +    RtlCopyMemory(tmp, &entry->u.Mount.use_nfspubfh, sizeof(DWORD));
  14655. +
  14656. +    *len = header_len;
  14657. +
  14658. +#ifdef DEBUG_MARSHAL_DETAIL
  14659. +    DbgP("marshal_nfs41_mount: server name='%wZ' mount point='%wZ' "
  14660. +         "sec_flavor='%s' rsize=%d wsize=%d use_nfspubfh=%d\n",
  14661. +        entry->u.Mount.srv_name, entry->u.Mount.root,
  14662. +         secflavorop2name(entry->u.Mount.sec_flavor),
  14663. +         (int)entry->u.Mount.rsize, (int)entry->u.Mount.wsize,
  14664. +         (int)entry->u.Mount.use_nfspubfh);
  14665. +#endif
  14666. +out:
  14667. +    return status;
  14668. +}
  14669. +
  14670. +NTSTATUS marshal_nfs41_unmount(
  14671. +    nfs41_updowncall_entry *entry,
  14672. +    unsigned char *buf,
  14673. +    ULONG buf_len,
  14674. +    ULONG *len)
  14675. +{
  14676. +    return marshal_nfs41_header(entry, buf, buf_len, len);
  14677. +}
  14678. +
  14679. +void unmarshal_nfs41_mount(
  14680. +    nfs41_updowncall_entry *cur,
  14681. +    unsigned char **buf)
  14682. +{
  14683. +    RtlCopyMemory(&cur->session, *buf, sizeof(HANDLE));
  14684. +    *buf += sizeof(HANDLE);
  14685. +    RtlCopyMemory(&cur->version, *buf, sizeof(DWORD));
  14686. +    *buf += sizeof(DWORD);
  14687. +    RtlCopyMemory(&cur->u.Mount.lease_time, *buf, sizeof(DWORD));
  14688. +    *buf += sizeof(DWORD);
  14689. +    RtlCopyMemory(cur->u.Mount.FsAttrs, *buf, sizeof(FILE_FS_ATTRIBUTE_INFORMATION));
  14690. +#ifdef DEBUG_MARSHAL_DETAIL
  14691. +    DbgP("unmarshal_nfs41_mount: session pointer 0x%x version %d lease_time "
  14692. +         "%d\n", cur->session, cur->version, cur->u.Mount.lease_time);
  14693. +#endif
  14694. +}
  14695. +
  14696. +NTSTATUS nfs41_unmount(
  14697. +    HANDLE session,
  14698. +    DWORD version,
  14699. +    DWORD timeout)
  14700. +{
  14701. +    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  14702. +    nfs41_updowncall_entry *entry;
  14703. +
  14704. +#ifdef DEBUG_MOUNT
  14705. +    DbgEn();
  14706. +#endif
  14707. +    status = nfs41_UpcallCreate(NFS41_UNMOUNT, NULL, session,
  14708. +        INVALID_HANDLE_VALUE, version, NULL, &entry);
  14709. +    if (status) goto out;
  14710. +
  14711. +    nfs41_UpcallWaitForReply(entry, timeout);
  14712. +
  14713. +    if (entry->psec_ctx == &entry->sec_ctx) {
  14714. +        SeDeleteClientSecurity(entry->psec_ctx);
  14715. +    }
  14716. +    entry->psec_ctx = NULL;
  14717. +    nfs41_UpcallDestroy(entry);
  14718. +out:
  14719. +#ifdef ENABLE_TIMINGS
  14720. +    print_op_stat("lookup", &lookup, 1);
  14721. +    print_op_stat("open", &open, 1);
  14722. +    print_op_stat("close", &close, 1);
  14723. +    print_op_stat("volume", &volume, 1);
  14724. +    print_op_stat("getattr", &getattr, 1);
  14725. +    print_op_stat("setattr", &setattr, 1);
  14726. +    print_op_stat("getexattr", &getexattr, 1);
  14727. +    print_op_stat("setexattr", &setexattr, 1);
  14728. +    print_op_stat("readdir", &readdir, 1);
  14729. +    print_op_stat("getacl", &getacl, 1);
  14730. +    print_op_stat("setacl", &setacl, 1);
  14731. +    print_op_stat("read", &read, 1);
  14732. +    print_op_stat("write", &write, 1);
  14733. +    print_op_stat("lock", &lock, 1);
  14734. +    print_op_stat("unlock", &unlock, 1);
  14735. +#endif
  14736. +#ifdef DEBUG_MOUNT
  14737. +    DbgEx();
  14738. +#endif
  14739. +    return status;
  14740. +}
  14741. +
  14742. +NTSTATUS map_mount_errors(
  14743. +    DWORD status)
  14744. +{
  14745. +    switch (status) {
  14746. +    case NO_ERROR:              return STATUS_SUCCESS;
  14747. +    case ERROR_ACCESS_DENIED:   return STATUS_ACCESS_DENIED;
  14748. +    case ERROR_NETWORK_UNREACHABLE: return STATUS_NETWORK_UNREACHABLE;
  14749. +    case ERROR_BAD_NET_RESP:    return STATUS_UNEXPECTED_NETWORK_ERROR;
  14750. +    case ERROR_BAD_NET_NAME:    return STATUS_BAD_NETWORK_NAME;
  14751. +    case ERROR_BAD_NETPATH:     return STATUS_BAD_NETWORK_PATH;
  14752. +    case ERROR_NOT_SUPPORTED:   return STATUS_NOT_SUPPORTED;
  14753. +    case ERROR_INTERNAL_ERROR:  return STATUS_INTERNAL_ERROR;
  14754. +    default:
  14755. +        print_error("map_mount_errors: "
  14756. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  14757. +            "defaulting to STATUS_INSUFFICIENT_RESOURCES\n", status);
  14758. +        return STATUS_INSUFFICIENT_RESOURCES;
  14759. +    }
  14760. +}
  14761. +
  14762. +NTSTATUS nfs41_mount(
  14763. +    PNFS41_MOUNT_CONFIG config,
  14764. +    DWORD sec_flavor,
  14765. +    PHANDLE session,
  14766. +    DWORD *version,
  14767. +    PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs)
  14768. +{
  14769. +    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  14770. +    nfs41_updowncall_entry *entry;
  14771. +
  14772. +#ifdef DEBUG_MOUNT
  14773. +    DbgEn();
  14774. +    DbgP("Server Name '%wZ' Mount Point '%wZ' SecFlavor %d\n",
  14775. +        &config->SrvName, &config->MntPt, sec_flavor);
  14776. +#endif
  14777. +    status = nfs41_UpcallCreate(NFS41_MOUNT, NULL, *session,
  14778. +        INVALID_HANDLE_VALUE, *version, &config->MntPt, &entry);
  14779. +    if (status) goto out;
  14780. +
  14781. +    entry->u.Mount.srv_name = &config->SrvName;
  14782. +    entry->u.Mount.root = &config->MntPt;
  14783. +    entry->u.Mount.rsize = config->ReadSize;
  14784. +    entry->u.Mount.wsize = config->WriteSize;
  14785. +    entry->u.Mount.use_nfspubfh = config->use_nfspubfh;
  14786. +    entry->u.Mount.sec_flavor = sec_flavor;
  14787. +    entry->u.Mount.FsAttrs = FsAttrs;
  14788. +
  14789. +    status = nfs41_UpcallWaitForReply(entry, config->timeout);
  14790. +    if (entry->psec_ctx == &entry->sec_ctx) {
  14791. +        SeDeleteClientSecurity(entry->psec_ctx);
  14792. +    }
  14793. +    entry->psec_ctx = NULL;
  14794. +    if (status) goto out;
  14795. +    *session = entry->session;
  14796. +    if (entry->u.Mount.lease_time > config->timeout)
  14797. +        config->timeout = entry->u.Mount.lease_time;
  14798. +
  14799. +    /* map windows ERRORs to NTSTATUS */
  14800. +    status = map_mount_errors(entry->status);
  14801. +    if (status == STATUS_SUCCESS)
  14802. +        *version = entry->version;
  14803. +    nfs41_UpcallDestroy(entry);
  14804. +out:
  14805. +#ifdef DEBUG_MOUNT
  14806. +    DbgEx();
  14807. +#endif
  14808. +    return status;
  14809. +}
  14810. +
  14811. +
  14812. +void nfs41_MountConfig_InitDefaults(
  14813. +    OUT PNFS41_MOUNT_CONFIG Config)
  14814. +{
  14815. +    RtlZeroMemory(Config, sizeof(NFS41_MOUNT_CONFIG));
  14816. +
  14817. +    Config->ReadSize = MOUNT_CONFIG_RW_SIZE_DEFAULT;
  14818. +    Config->WriteSize = MOUNT_CONFIG_RW_SIZE_DEFAULT;
  14819. +    Config->use_nfspubfh = FALSE;
  14820. +    Config->ReadOnly = FALSE;
  14821. +    Config->write_thru = FALSE;
  14822. +    Config->nocache = FALSE;
  14823. +    Config->timebasedcoherency = FALSE; /* disabled by default because of bugs */
  14824. +    Config->SrvName.Length = 0;
  14825. +    Config->SrvName.MaximumLength = SERVER_NAME_BUFFER_SIZE;
  14826. +    Config->SrvName.Buffer = Config->srv_buffer;
  14827. +    Config->MntPt.Length = 0;
  14828. +    Config->MntPt.MaximumLength = NFS41_SYS_MAX_PATH_LEN;
  14829. +    Config->MntPt.Buffer = Config->mntpt_buffer;
  14830. +    Config->SecFlavor.Length = 0;
  14831. +    Config->SecFlavor.MaximumLength = MAX_SEC_FLAVOR_LEN;
  14832. +    Config->SecFlavor.Buffer = Config->sec_flavor_buffer;
  14833. +    RtlCopyUnicodeString(&Config->SecFlavor, &AUTH_SYS_NAME);
  14834. +    Config->timeout = UPCALL_TIMEOUT_DEFAULT;
  14835. +    Config->createmode.use_nfsv3attrsea_mode = TRUE;
  14836. +    Config->createmode.mode = NFS41_DRIVER_DEFAULT_CREATE_MODE;
  14837. +}
  14838. +
  14839. +static
  14840. +NTSTATUS nfs41_MountConfig_ParseBoolean(
  14841. +    IN PFILE_FULL_EA_INFORMATION Option,
  14842. +    IN PUNICODE_STRING usValue,
  14843. +    IN BOOLEAN negate_val,
  14844. +    OUT PBOOLEAN Value)
  14845. +{
  14846. +    NTSTATUS status = STATUS_SUCCESS;
  14847. +
  14848. +    /* if no value is specified, assume TRUE
  14849. +     * if a value is specified, it must be a '1' */
  14850. +    if (Option->EaValueLength == 0 || *usValue->Buffer == L'1')
  14851. +        *Value = negate_val?FALSE:TRUE;
  14852. +    else
  14853. +        *Value = negate_val?TRUE:FALSE;
  14854. +
  14855. +    DbgP("    '%ls' -> '%wZ' -> %u\n",
  14856. +        (LPWSTR)Option->EaName, usValue, *Value);
  14857. +    return status;
  14858. +}
  14859. +
  14860. +
  14861. +/* Parse |signed| integer value */
  14862. +static
  14863. +NTSTATUS nfs41_MountConfig_ParseINT64(
  14864. +    IN PFILE_FULL_EA_INFORMATION Option,
  14865. +    IN PUNICODE_STRING usValue,
  14866. +    OUT INT64 *outValue,
  14867. +    IN INT64 Minimum,
  14868. +    IN INT64 Maximum)
  14869. +{
  14870. +    NTSTATUS status;
  14871. +    LONG64 Value = 0;
  14872. +    LPWSTR Name = (LPWSTR)Option->EaName;
  14873. +
  14874. +    if (!Option->EaValueLength)
  14875. +        return STATUS_INVALID_PARAMETER;
  14876. +
  14877. +    status = RtlUnicodeStringToInt64(usValue, 0, &Value, NULL);
  14878. +    if (status == STATUS_SUCCESS) {
  14879. +        if ((Value < Minimum) || (Value > Maximum))
  14880. +            status = STATUS_INVALID_PARAMETER;
  14881. +
  14882. +        if (status == STATUS_SUCCESS) {
  14883. +            *outValue = Value;
  14884. +        }
  14885. +    }
  14886. +    else {
  14887. +        print_error("nfs41_MountConfig_ParseINT64: "
  14888. +            "Failed to convert '%s'='%wZ' to unsigned long.\n",
  14889. +            Name, usValue);
  14890. +    }
  14891. +
  14892. +    return status;
  14893. +}
  14894. +
  14895. +/* Parse |unsigned| integer value */
  14896. +static
  14897. +NTSTATUS nfs41_MountConfig_ParseDword(
  14898. +    IN PFILE_FULL_EA_INFORMATION Option,
  14899. +    IN PUNICODE_STRING usValue,
  14900. +    OUT PDWORD outValue,
  14901. +    IN DWORD Minimum,
  14902. +    IN DWORD Maximum)
  14903. +{
  14904. +    INT64 tmpValue;
  14905. +    NTSTATUS status;
  14906. +
  14907. +    status = nfs41_MountConfig_ParseINT64(
  14908. +        Option, usValue,
  14909. +        &tmpValue, Minimum, Maximum);
  14910. +
  14911. +    if (status == STATUS_SUCCESS) {
  14912. +        *outValue = (DWORD)tmpValue;
  14913. +    }
  14914. +
  14915. +    return status;
  14916. +}
  14917. +
  14918. +NTSTATUS nfs41_MountConfig_ParseOptions(
  14919. +    IN PFILE_FULL_EA_INFORMATION EaBuffer,
  14920. +    IN ULONG EaLength,
  14921. +    IN OUT PNFS41_MOUNT_CONFIG Config)
  14922. +{
  14923. +    DbgP("--> nfs41_MountConfig_ParseOptions(EaBuffer=0x%p,EaLength=%ld)\n",
  14924. +        (void *)EaBuffer,
  14925. +        (long)EaLength);
  14926. +    NTSTATUS  status = STATUS_SUCCESS;
  14927. +    PFILE_FULL_EA_INFORMATION Option;
  14928. +    LPWSTR Name;
  14929. +    size_t NameLen;
  14930. +    UNICODE_STRING  usValue;
  14931. +    ULONG error_offset;
  14932. +
  14933. +    status = IoCheckEaBufferValidity(EaBuffer, EaLength, &error_offset);
  14934. +    if (status) {
  14935. +        DbgP("status(=0x%lx)=IoCheckEaBufferValidity"
  14936. +            "(eainfo=0x%p, buflen=%lu, &(error_offset=%d)) failed\n",
  14937. +            (long)status, (void *)EaBuffer, EaLength,
  14938. +            (int)error_offset);
  14939. +        goto out;
  14940. +    }
  14941. +
  14942. +    Option = EaBuffer;
  14943. +    while (status == STATUS_SUCCESS) {
  14944. +        DbgP("Option=0x%p\n", (void *)Option);
  14945. +        Name = (LPWSTR)Option->EaName;
  14946. +        NameLen = Option->EaNameLength/sizeof(WCHAR);
  14947. +
  14948. +        DbgP("nfs41_MountConfig_ParseOptions: Name='%*S'/NameLen=%d\n",
  14949. +            (int)NameLen, Name, (int)NameLen);
  14950. +
  14951. +        usValue.Length = usValue.MaximumLength = Option->EaValueLength;
  14952. +        usValue.Buffer = (PWCH)(Option->EaName +
  14953. +            Option->EaNameLength + sizeof(WCHAR));
  14954. +
  14955. +        DbgP("nfs41_MountConfig_ParseOptions: option/usValue='%wZ'/%ld\n",
  14956. +            &usValue, (long)usValue.Length);
  14957. +
  14958. +        if (wcsncmp(L"ro", Name, NameLen) == 0) {
  14959. +            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  14960. +                FALSE, &Config->ReadOnly);
  14961. +        } else if (wcsncmp(L"rw", Name, NameLen) == 0) {
  14962. +            /* opposite of "ro", so negate */
  14963. +            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  14964. +                TRUE, &Config->ReadOnly);
  14965. +        }
  14966. +        else if (wcsncmp(L"writethru", Name, NameLen) == 0) {
  14967. +            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  14968. +                FALSE, &Config->write_thru);
  14969. +        }
  14970. +        else if (wcsncmp(L"nowritethru", Name, NameLen) == 0) {
  14971. +            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  14972. +                TRUE, &Config->write_thru);
  14973. +        }
  14974. +        else if (wcsncmp(L"cache", Name, NameLen) == 0) {
  14975. +            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  14976. +                TRUE, &Config->nocache);
  14977. +        }
  14978. +        else if (wcsncmp(L"nocache", Name, NameLen) == 0) {
  14979. +            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  14980. +                FALSE, &Config->nocache);
  14981. +        }
  14982. +        else if (wcsncmp(L"timebasedcoherency", Name, NameLen) == 0) {
  14983. +            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  14984. +                FALSE, &Config->timebasedcoherency);
  14985. +        }
  14986. +        else if (wcsncmp(L"notimebasedcoherency", Name, NameLen) == 0) {
  14987. +            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  14988. +                TRUE, &Config->timebasedcoherency);
  14989. +        }
  14990. +        else if (wcsncmp(L"timeout", Name, NameLen) == 0) {
  14991. +            status = nfs41_MountConfig_ParseDword(Option, &usValue,
  14992. +                &Config->timeout, UPCALL_TIMEOUT_DEFAULT,
  14993. +                UPCALL_TIMEOUT_DEFAULT);
  14994. +        }
  14995. +        else if (wcsncmp(L"rsize", Name, NameLen) == 0) {
  14996. +            status = nfs41_MountConfig_ParseDword(Option, &usValue,
  14997. +                &Config->ReadSize, MOUNT_CONFIG_RW_SIZE_MIN,
  14998. +                MOUNT_CONFIG_RW_SIZE_MAX);
  14999. +        }
  15000. +        else if (wcsncmp(L"wsize", Name, NameLen) == 0) {
  15001. +            status = nfs41_MountConfig_ParseDword(Option, &usValue,
  15002. +                &Config->WriteSize, MOUNT_CONFIG_RW_SIZE_MIN,
  15003. +                MOUNT_CONFIG_RW_SIZE_MAX);
  15004. +        }
  15005. +        else if (wcsncmp(L"public", Name, NameLen) == 0) {
  15006. +            /*
  15007. +             + We ignore this value here, and instead rely on the
  15008. +             * /pubnfs4 prefix
  15009. +             */
  15010. +            BOOLEAN dummy;
  15011. +            status = nfs41_MountConfig_ParseBoolean(Option, &usValue,
  15012. +                FALSE, &dummy);
  15013. +        }
  15014. +        else if (wcsncmp(L"srvname", Name, NameLen) == 0) {
  15015. +            if (usValue.Length > Config->SrvName.MaximumLength)
  15016. +                status = STATUS_NAME_TOO_LONG;
  15017. +            else
  15018. +                RtlCopyUnicodeString(&Config->SrvName, &usValue);
  15019. +        }
  15020. +        else if (wcsncmp(L"mntpt", Name, NameLen) == 0) {
  15021. +            if (usValue.Length > Config->MntPt.MaximumLength)
  15022. +                status = STATUS_NAME_TOO_LONG;
  15023. +            else
  15024. +                RtlCopyUnicodeString(&Config->MntPt, &usValue);
  15025. +        }
  15026. +        else if (wcsncmp(L"sec", Name, NameLen) == 0) {
  15027. +            if (usValue.Length > Config->SecFlavor.MaximumLength)
  15028. +                status = STATUS_NAME_TOO_LONG;
  15029. +            else
  15030. +                RtlCopyUnicodeString(&Config->SecFlavor, &usValue);
  15031. +        }
  15032. +        else if (wcsncmp(L"createmode", Name, NameLen) == 0) {
  15033. +#define NFSV3ATTRMODE_WSTR L"nfsv3attrmode+"
  15034. +#define NFSV3ATTRMODE_WCSLEN (14)
  15035. +#define NFSV3ATTRMODE_BYTELEN (NFSV3ATTRMODE_WCSLEN*sizeof(WCHAR))
  15036. +            if ((usValue.Length >= NFSV3ATTRMODE_BYTELEN) &&
  15037. +                (!wcsncmp(NFSV3ATTRMODE_WSTR,
  15038. +                    usValue.Buffer,
  15039. +                    min(NFSV3ATTRMODE_WCSLEN,
  15040. +                        usValue.Length/sizeof(WCHAR))))) {
  15041. +                usValue.Buffer += NFSV3ATTRMODE_WCSLEN;
  15042. +                usValue.Length = usValue.MaximumLength =
  15043. +                    usValue.Length - NFSV3ATTRMODE_BYTELEN;
  15044. +#ifdef DEBUG_MOUNTCONFIG
  15045. +                DbgP("nfs41_MountConfig_ParseOptions: createmode "
  15046. +                    "nfs4attr "
  15047. +                    "leftover option/usValue='%wZ'/%ld\n",
  15048. +                    &usValue, (long)usValue.Length);
  15049. +#endif /* DEBUG_MOUNTCONFIG */
  15050. +
  15051. +                Config->createmode.use_nfsv3attrsea_mode = TRUE;
  15052. +            }
  15053. +            else {
  15054. +#ifdef DEBUG_MOUNTCONFIG
  15055. +                DbgP("nfs41_MountConfig_ParseOptions: createmode "
  15056. +                    "leftover option/usValue='%wZ'/%ld\n",
  15057. +                    &usValue, (long)usValue.Length);
  15058. +#endif /* DEBUG_MOUNTCONFIG */
  15059. +                Config->createmode.use_nfsv3attrsea_mode = FALSE;
  15060. +            }
  15061. +
  15062. +            /*
  15063. +             * Reject mode values not prefixed with "0o", as
  15064. +             * |RtlUnicodeStringToInteger()| uses
  15065. +             * 0o (e.g. "0o123") as prefix for octal values,
  15066. +             * and does not understand the traditional
  15067. +             * UNIX/POSIX/ISO C "0" (e.g. "0123") prefix
  15068. +             */
  15069. +            if ((usValue.Length >= (3*sizeof(WCHAR))) &&
  15070. +                (usValue.Buffer[0] == L'0') &&
  15071. +                (usValue.Buffer[1] == L'o')) {
  15072. +                status = nfs41_MountConfig_ParseDword(Option,
  15073. +                    &usValue,
  15074. +                    &Config->createmode.mode, 0,
  15075. +                    0777);
  15076. +                if (status == STATUS_SUCCESS) {
  15077. +                    if (Config->createmode.mode > 0777) {
  15078. +                        status = STATUS_INVALID_PARAMETER;
  15079. +                        print_error("mode 0%o out of bounds\n",
  15080. +                            (int)Config->createmode.mode);
  15081. +                    }
  15082. +                }
  15083. +            }
  15084. +            else {
  15085. +                status = STATUS_INVALID_PARAMETER;
  15086. +                print_error("Invalid createmode '%wZ'\n",
  15087. +                    usValue);
  15088. +            }
  15089. +
  15090. +            DbgP("nfs41_MountConfig_ParseOptions: createmode: "
  15091. +                "status=0x%lx, "
  15092. +                "createmode=(use_nfsv3attrsea_mode=%d, mode=0%o\n",
  15093. +                (long)status,
  15094. +                (int)Config->createmode.use_nfsv3attrsea_mode,
  15095. +                (int)Config->createmode.mode);
  15096. +        }
  15097. +        else {
  15098. +            status = STATUS_INVALID_PARAMETER;
  15099. +            print_error("Unrecognized option '%ls' -> '%wZ'\n",
  15100. +                Name, usValue);
  15101. +        }
  15102. +
  15103. +        if (Option->NextEntryOffset == 0)
  15104. +            break;
  15105. +
  15106. +        Option = (PFILE_FULL_EA_INFORMATION)
  15107. +            ((PBYTE)Option + Option->NextEntryOffset);
  15108. +    }
  15109. +
  15110. +out:
  15111. +    DbgP("<-- nfs41_MountConfig_ParseOptions, status=0x%lx\n",
  15112. +        (long)status);
  15113. +    return status;
  15114. +}
  15115. +
  15116. +NTSTATUS map_sec_flavor(
  15117. +    IN PUNICODE_STRING sec_flavor_name,
  15118. +    OUT PDWORD sec_flavor)
  15119. +{
  15120. +    if (RtlCompareUnicodeString(sec_flavor_name, &AUTH_SYS_NAME, FALSE) == 0)
  15121. +        *sec_flavor = RPCSEC_AUTH_SYS;
  15122. +    else if (RtlCompareUnicodeString(sec_flavor_name, &AUTHGSS_KRB5_NAME, FALSE) == 0)
  15123. +        *sec_flavor = RPCSEC_AUTHGSS_KRB5;
  15124. +    else if (RtlCompareUnicodeString(sec_flavor_name, &AUTHGSS_KRB5I_NAME, FALSE) == 0)
  15125. +        *sec_flavor = RPCSEC_AUTHGSS_KRB5I;
  15126. +    else if (RtlCompareUnicodeString(sec_flavor_name, &AUTHGSS_KRB5P_NAME, FALSE) == 0)
  15127. +        *sec_flavor = RPCSEC_AUTHGSS_KRB5P;
  15128. +    else return STATUS_INVALID_PARAMETER;
  15129. +    return STATUS_SUCCESS;
  15130. +}
  15131. +
  15132. +static
  15133. +NTSTATUS nfs41_GetLUID(
  15134. +    PLUID id)
  15135. +{
  15136. +    NTSTATUS status = STATUS_SUCCESS;
  15137. +    SECURITY_SUBJECT_CONTEXT sec_ctx;
  15138. +    SECURITY_QUALITY_OF_SERVICE sec_qos;
  15139. +    SECURITY_CLIENT_CONTEXT clnt_sec_ctx;
  15140. +
  15141. +    SeCaptureSubjectContext(&sec_ctx);
  15142. +    sec_qos.ContextTrackingMode = SECURITY_STATIC_TRACKING;
  15143. +    sec_qos.ImpersonationLevel = SecurityIdentification;
  15144. +    sec_qos.Length = sizeof(SECURITY_QUALITY_OF_SERVICE);
  15145. +    sec_qos.EffectiveOnly = 0;
  15146. +    /*
  15147. +     * Arg |ServerIsRemote| must be |FALSE|, otherwise processes
  15148. +     * like Cygwin setup-x86_64.exe can fail during "Activation
  15149. +     * Context" creation in
  15150. +     * |SeCreateClientSecurityFromSubjectContext()| with
  15151. +     * |STATUS_BAD_IMPERSONATION_LEVEL|
  15152. +     */
  15153. +    status = SeCreateClientSecurityFromSubjectContext(&sec_ctx, &sec_qos,
  15154. +        FALSE, &clnt_sec_ctx);
  15155. +    if (status) {
  15156. +        print_error("nfs41_GetLUID: SeCreateClientSecurityFromSubjectContext "
  15157. +             "failed 0x%x\n", status);
  15158. +        goto release_sec_ctx;
  15159. +    }
  15160. +    status = SeQueryAuthenticationIdToken(clnt_sec_ctx.ClientToken, id);
  15161. +    if (status) {
  15162. +        print_error("nfs41_GetLUID: "
  15163. +            "SeQueryAuthenticationIdToken() failed 0x%x\n", status);
  15164. +        goto release_clnt_sec_ctx;
  15165. +    }
  15166. +release_clnt_sec_ctx:
  15167. +    SeDeleteClientSecurity(&clnt_sec_ctx);
  15168. +release_sec_ctx:
  15169. +    SeReleaseSubjectContext(&sec_ctx);
  15170. +
  15171. +    return status;
  15172. +}
  15173. +
  15174. +static
  15175. +NTSTATUS has_nfs_prefix(
  15176. +    IN PUNICODE_STRING SrvCallName,
  15177. +    IN PUNICODE_STRING NetRootName,
  15178. +    OUT BOOLEAN *pubfh_prefix)
  15179. +{
  15180. +    NTSTATUS status = STATUS_BAD_NETWORK_NAME;
  15181. +
  15182. +#ifdef USE_ENTIRE_PATH_FOR_NETROOT
  15183. +    if (NetRootName->Length >=
  15184. +        (SrvCallName->Length + NfsPrefix.Length)) {
  15185. +        size_t len = NetRootName->Length / 2;
  15186. +        size_t i;
  15187. +        int state = 0;
  15188. +
  15189. +        /* Scan \hostname@port\nfs4 */
  15190. +        for (i = 0 ; i < len ; i++) {
  15191. +            wchar_t ch = NetRootName->Buffer[i];
  15192. +
  15193. +            if ((ch == L'\\') && (state == 0)) {
  15194. +                state = 1;
  15195. +                continue;
  15196. +            }
  15197. +            else if ((ch == L'@') && (state == 1)) {
  15198. +                state = 2;
  15199. +                continue;
  15200. +            }
  15201. +            else if ((ch == L'\\') && (state == 2)) {
  15202. +                state = 3;
  15203. +                break;
  15204. +            }
  15205. +            else if (ch == L'\\') {
  15206. +                /* Abort, '\\' with wrong state */
  15207. +                break;
  15208. +            }
  15209. +        }
  15210. +
  15211. +        if (state == 3) {
  15212. +            if (!memcmp(&NetRootName->Buffer[i], L"\\nfs4",
  15213. +                (4*sizeof(wchar_t))))) {
  15214. +                *pubfh_prefix = FALSE;
  15215. +                status = STATUS_SUCCESS;
  15216. +            }
  15217. +            if ((NetRootName->Length >=
  15218. +                (SrvCallName->Length + PubNfsPrefix.Length)) &&
  15219. +                (!memcmp(&NetRootName->Buffer[i], L"\\pubnfs4",
  15220. +                    (4*sizeof(wchar_t))))) {
  15221. +                *pubfh_prefix = TRUE;
  15222. +                status = STATUS_SUCCESS;
  15223. +            }
  15224. +        }
  15225. +    }
  15226. +#else
  15227. +    if (NetRootName->Length ==
  15228. +        (SrvCallName->Length + NfsPrefix.Length)) {
  15229. +        const UNICODE_STRING NetRootPrefix = {
  15230. +            NfsPrefix.Length,
  15231. +            NetRootName->MaximumLength - SrvCallName->Length,
  15232. +            &NetRootName->Buffer[SrvCallName->Length/2]
  15233. +        };
  15234. +        if (!RtlCompareUnicodeString(&NetRootPrefix, &NfsPrefix, FALSE))
  15235. +            *pubfh_prefix = FALSE;
  15236. +            status = STATUS_SUCCESS;
  15237. +    }
  15238. +    else if (NetRootName->Length ==
  15239. +        (SrvCallName->Length + PubNfsPrefix.Length)) {
  15240. +        const UNICODE_STRING PubNetRootPrefix = {
  15241. +            PubNfsPrefix.Length,
  15242. +            NetRootName->MaximumLength - SrvCallName->Length,
  15243. +            &NetRootName->Buffer[SrvCallName->Length/2]
  15244. +        };
  15245. +        if (!RtlCompareUnicodeString(&PubNetRootPrefix, &PubNfsPrefix, FALSE))
  15246. +            *pubfh_prefix = TRUE;
  15247. +            status = STATUS_SUCCESS;
  15248. +    }
  15249. +#endif /* USE_ENTIRE_PATH_FOR_NETROOT */
  15250. +    return status;
  15251. +}
  15252. +
  15253. +NTSTATUS nfs41_CreateVNetRoot(
  15254. +    IN OUT PMRX_CREATENETROOT_CONTEXT pCreateNetRootContext)
  15255. +{
  15256. +    NTSTATUS status = STATUS_SUCCESS;
  15257. +    NFS41_MOUNT_CONFIG *Config;
  15258. +    __notnull PMRX_V_NET_ROOT pVNetRoot = (PMRX_V_NET_ROOT)
  15259. +        pCreateNetRootContext->pVNetRoot;
  15260. +    __notnull PMRX_NET_ROOT pNetRoot = pVNetRoot->pNetRoot;
  15261. +    __notnull PMRX_SRV_CALL pSrvCall = pNetRoot->pSrvCall;
  15262. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  15263. +        NFS41GetVNetRootExtension(pVNetRoot);
  15264. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  15265. +        NFS41GetNetRootExtension(pNetRoot);
  15266. +    NFS41GetDeviceExtension(pCreateNetRootContext->RxContext,DevExt);
  15267. +    DWORD nfs41d_version = DevExt->nfs41d_version;
  15268. +    nfs41_mount_entry *existing_mount = NULL;
  15269. +    LUID luid;
  15270. +    BOOLEAN found_existing_mount = FALSE, found_matching_flavor = FALSE;
  15271. +
  15272. +    ASSERT((NodeType(pNetRoot) == RDBSS_NTC_NETROOT) &&
  15273. +        (NodeType(pNetRoot->pSrvCall) == RDBSS_NTC_SRVCALL));
  15274. +
  15275. +#ifdef DEBUG_MOUNT
  15276. +    DbgEn();
  15277. +    // print_srv_call(pSrvCall);
  15278. +    // print_net_root(pNetRoot);
  15279. +    // print_v_net_root(pVNetRoot);
  15280. +
  15281. +    DbgP("pVNetRoot=0x%p pNetRoot=0x%p pSrvCall=0x%p\n", pVNetRoot, pNetRoot, pSrvCall);
  15282. +    DbgP("pNetRoot='%wZ' Type=%d pSrvCallName='%wZ' VirtualNetRootStatus=0x%x "
  15283. +        "NetRootStatus=0x%x\n", pNetRoot->pNetRootName,
  15284. +        pNetRoot->Type, pSrvCall->pSrvCallName,
  15285. +        pCreateNetRootContext->VirtualNetRootStatus,
  15286. +        pCreateNetRootContext->NetRootStatus);
  15287. +#endif
  15288. +
  15289. +    if (pNetRoot->Type != NET_ROOT_DISK && pNetRoot->Type != NET_ROOT_WILD) {
  15290. +        print_error("nfs41_CreateVNetRoot: Unsupported NetRoot Type %u\n",
  15291. +            pNetRoot->Type);
  15292. +        status = STATUS_NOT_SUPPORTED;
  15293. +        goto out;
  15294. +    }
  15295. +
  15296. +    pVNetRootContext->session = INVALID_HANDLE_VALUE;
  15297. +
  15298. +    /*
  15299. +     * In order to cooperate with other network providers, we
  15300. +     * must only claim paths of the form '\\server\nfs4\path'
  15301. +     * or '\\server\pubnfs4\path'
  15302. +     */
  15303. +    BOOLEAN pubfh_prefix = FALSE;
  15304. +    status = has_nfs_prefix(pSrvCall->pSrvCallName, pNetRoot->pNetRootName, &pubfh_prefix);
  15305. +    if (status) {
  15306. +        print_error("nfs41_CreateVNetRoot: NetRootName '%wZ' doesn't match "
  15307. +            "'\\nfs4' or '\\pubnfs4'!\n", pNetRoot->pNetRootName);
  15308. +        goto out;
  15309. +    }
  15310. +    pNetRoot->MRxNetRootState = MRX_NET_ROOT_STATE_GOOD;
  15311. +    pNetRoot->DeviceType = FILE_DEVICE_DISK;
  15312. +
  15313. +    Config = RxAllocatePoolWithTag(NonPagedPoolNx,
  15314. +            sizeof(NFS41_MOUNT_CONFIG), NFS41_MM_POOLTAG);
  15315. +    if (Config == NULL) {
  15316. +        status = STATUS_INSUFFICIENT_RESOURCES;
  15317. +        goto out;
  15318. +    }
  15319. +    nfs41_MountConfig_InitDefaults(Config);
  15320. +
  15321. +    if (pCreateNetRootContext->RxContext->Create.EaLength) {
  15322. +        /* Codepath for nfs_mount.exe */
  15323. +        DbgP("Codepath for nfs_mount.exe, "
  15324. +            "Create->{ EaBuffer=0x%p, EaLength=%ld }\n",
  15325. +            pCreateNetRootContext->RxContext->Create.EaBuffer,
  15326. +            (long)pCreateNetRootContext->RxContext->Create.EaLength);
  15327. +
  15328. +        /* parse the extended attributes for mount options */
  15329. +        status = nfs41_MountConfig_ParseOptions(
  15330. +            pCreateNetRootContext->RxContext->Create.EaBuffer,
  15331. +            pCreateNetRootContext->RxContext->Create.EaLength,
  15332. +            Config);
  15333. +        if (status != STATUS_SUCCESS) {
  15334. +            DbgP("nfs41_MountConfig_ParseOptions() failed\n");
  15335. +            goto out_free;
  15336. +        }
  15337. +        pVNetRootContext->read_only = Config->ReadOnly;
  15338. +        pVNetRootContext->write_thru = Config->write_thru;
  15339. +        pVNetRootContext->nocache = Config->nocache;
  15340. +        pVNetRootContext->timebasedcoherency = Config->timebasedcoherency;
  15341. +    } else {
  15342. +        /*
  15343. +         * Codepath for \\server@port\nfs4\path or
  15344. +         * \\server@port\pubnfs4\path
  15345. +         */
  15346. +        DbgP("Codepath for \\\\server@port\\@(pubnfs4|nfs4)\\path\n");
  15347. +
  15348. +        /*
  15349. +         * STATUS_NFS_SHARE_NOT_MOUNTED - status code for the case
  15350. +         * when a NFS filesystem is accessed via UNC path, but no
  15351. +         * nfs_mount.exe was done for that filesystem
  15352. +         */
  15353. +#define STATUS_NFS_SHARE_NOT_MOUNTED STATUS_BAD_NETWORK_PATH
  15354. +        if (!pNetRootContext->mounts_init) {
  15355. +            /*
  15356. +             * We can only support UNC paths when we got valid
  15357. +             * mount options via nfs_mount.exe before this point.
  15358. +             */
  15359. +            DbgP("pNetRootContext(=0x%p) not initalised yet\n",
  15360. +                pNetRootContext);
  15361. +            status = STATUS_NFS_SHARE_NOT_MOUNTED;
  15362. +            goto out_free;
  15363. +        }
  15364. +
  15365. +        /*
  15366. +         * gisburn: Fixme: Originally the code was using the
  15367. +         * SRV_CALL name (without leading \) as the hostname
  15368. +         * like this:
  15369. +         * ---- snip ----
  15370. +         * Config->SrvName.Buffer = pSrvCall->pSrvCallName->Buffer+1;
  15371. +         * Config->SrvName.Length =
  15372. +         *     pSrvCall->pSrvCallName->Length - sizeof(WCHAR);
  15373. +         * Config->SrvName.MaximumLength =
  15374. +         *     pSrvCall->pSrvCallName->MaximumLength - sizeof(WCHAR);
  15375. +         * ---- snip ----
  15376. +         * IMHO we should validate that the hostname in
  15377. +         * |existing_mount->Config| below matches
  15378. +         * |pSrvCall->pSrvCallName->Buffer|
  15379. +         */
  15380. +
  15381. +        status = nfs41_GetLUID(&luid);
  15382. +        if (status)
  15383. +            goto out_free;
  15384. +
  15385. +#ifdef DEBUG_MOUNT
  15386. +        DbgP("UNC path LUID 0x%lx.0x%lx\n",
  15387. +            (long)luid.HighPart, (long)luid.LowPart);
  15388. +#endif
  15389. +
  15390. +        PLIST_ENTRY pEntry;
  15391. +        nfs41_mount_entry *found_mount_entry = NULL;
  15392. +#ifdef NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL
  15393. +        nfs41_mount_entry *found_system_mount_entry = NULL;
  15394. +#endif /* NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL */
  15395. +
  15396. +        status = STATUS_NFS_SHARE_NOT_MOUNTED;
  15397. +
  15398. +        ExAcquireFastMutex(&pNetRootContext->mountLock);
  15399. +        pEntry = &pNetRootContext->mounts.head;
  15400. +        pEntry = pEntry->Flink;
  15401. +        while (pEntry != NULL) {
  15402. +            existing_mount = (nfs41_mount_entry *)CONTAINING_RECORD(pEntry,
  15403. +                    nfs41_mount_entry, next);
  15404. +
  15405. +#ifdef DEBUG_MOUNT
  15406. +            DbgP("finding mount config: "
  15407. +                "comparing luid=(0x%lx.0x%lx) with "
  15408. +                "existing_mount->login_id=(0x%lx.0x%lx)\n",
  15409. +                (long)luid.HighPart, (long)luid.LowPart,
  15410. +                (long)existing_mount->login_id.HighPart,
  15411. +                (long)existing_mount->login_id.LowPart);
  15412. +#endif
  15413. +
  15414. +            if (RtlEqualLuid(&luid, &existing_mount->login_id)) {
  15415. +                /* found existing mount with exact LUID match */
  15416. +                found_mount_entry = existing_mount;
  15417. +                break;
  15418. +            }
  15419. +#ifdef NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL
  15420. +            else if (RtlEqualLuid(&SystemLuid,
  15421. +                &existing_mount->login_id)) {
  15422. +                /*
  15423. +                 * found existing mount for user "SYSTEM"
  15424. +                 * We continue searching the |pNetRootContext->mounts|
  15425. +                 * list for an exact match ...
  15426. +                 */
  15427. +                found_system_mount_entry = existing_mount;
  15428. +            }
  15429. +#endif /* NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL */
  15430. +            if (pEntry->Flink == &pNetRootContext->mounts.head)
  15431. +                break;
  15432. +            pEntry = pEntry->Flink;
  15433. +        }
  15434. +
  15435. +        if (found_mount_entry) {
  15436. +            copy_nfs41_mount_config(Config, &found_mount_entry->Config);
  15437. +            DbgP("Found existing mount: LUID=(0x%lx.0x%lx) Entry Config->MntPt='%wZ'\n",
  15438. +                (long)found_mount_entry->login_id.HighPart,
  15439. +                (long)found_mount_entry->login_id.LowPart,
  15440. +                &Config->MntPt);
  15441. +            status = STATUS_SUCCESS;
  15442. +        }
  15443. +#ifdef NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL
  15444. +        else if (found_system_mount_entry) {
  15445. +            copy_nfs41_mount_config(Config, &found_system_mount_entry->Config);
  15446. +            DbgP("Found existing SYSTEM mount: Entry Config->MntPt='%wZ'\n",
  15447. +                &Config->MntPt);
  15448. +            status = STATUS_SUCCESS;
  15449. +        }
  15450. +#endif /* NFS41_DRIVER_SYSTEM_LUID_MOUNTS_ARE_GLOBAL */
  15451. +        ExReleaseFastMutex(&pNetRootContext->mountLock);
  15452. +
  15453. +        if (status != STATUS_SUCCESS) {
  15454. +            DbgP("No existing mount found, "
  15455. +                "status==STATUS_NFS_SHARE_NOT_MOUNTED\n");
  15456. +            goto out_free;
  15457. +        }
  15458. +
  15459. +        pVNetRootContext->read_only = Config->ReadOnly;
  15460. +        pVNetRootContext->write_thru = Config->write_thru;
  15461. +        pVNetRootContext->nocache = Config->nocache;
  15462. +        pVNetRootContext->timebasedcoherency = Config->timebasedcoherency;
  15463. +    }
  15464. +
  15465. +    Config->use_nfspubfh = pubfh_prefix;
  15466. +
  15467. +    DbgP("Config->{ "
  15468. +        "MntPt='%wZ', "
  15469. +        "SrvName='%wZ', "
  15470. +        "use_nfspubfh=%d, "
  15471. +        "ReadOnly=%d, "
  15472. +        "write_thru=%d, "
  15473. +        "nocache=%d "
  15474. +        "timebasedcoherency=%d "
  15475. +        "timeout=%d "
  15476. +        "createmode.use_nfsv3attrsea_mode=%d "
  15477. +        "Config->createmode.mode=0%o "
  15478. +        "}\n",
  15479. +        &Config->MntPt,
  15480. +        &Config->SrvName,
  15481. +        Config->use_nfspubfh?1:0,
  15482. +        Config->ReadOnly?1:0,
  15483. +        Config->write_thru?1:0,
  15484. +        Config->nocache?1:0,
  15485. +        Config->timebasedcoherency?1:0,
  15486. +        Config->timeout,
  15487. +        Config->createmode.use_nfsv3attrsea_mode?1:0,
  15488. +        Config->createmode.mode);
  15489. +
  15490. +    pVNetRootContext->MountPathLen = Config->MntPt.Length;
  15491. +    pVNetRootContext->timeout = Config->timeout;
  15492. +    pVNetRootContext->createmode.use_nfsv3attrsea_mode =
  15493. +        Config->createmode.use_nfsv3attrsea_mode;
  15494. +    pVNetRootContext->createmode.mode =
  15495. +        Config->createmode.mode;
  15496. +
  15497. +    status = map_sec_flavor(&Config->SecFlavor, &pVNetRootContext->sec_flavor);
  15498. +    if (status != STATUS_SUCCESS) {
  15499. +        DbgP("Invalid rpcsec security flavor '%wZ'\n", &Config->SecFlavor);
  15500. +        goto out_free;
  15501. +    }
  15502. +
  15503. +    status = nfs41_GetLUID(&luid);
  15504. +    if (status)
  15505. +        goto out_free;
  15506. +
  15507. +    if (!pNetRootContext->mounts_init) {
  15508. +#ifdef DEBUG_MOUNT
  15509. +        DbgP("Initializing mount array\n");
  15510. +#endif
  15511. +        ExInitializeFastMutex(&pNetRootContext->mountLock);
  15512. +        InitializeListHead(&pNetRootContext->mounts.head);
  15513. +        pNetRootContext->mounts_init = TRUE;
  15514. +    } else {
  15515. +        PLIST_ENTRY pEntry;
  15516. +
  15517. +        ExAcquireFastMutex(&pNetRootContext->mountLock);
  15518. +        pEntry = &pNetRootContext->mounts.head;
  15519. +        pEntry = pEntry->Flink;
  15520. +        while (pEntry != NULL) {
  15521. +            existing_mount = (nfs41_mount_entry *)CONTAINING_RECORD(pEntry,
  15522. +                    nfs41_mount_entry, next);
  15523. +#ifdef DEBUG_MOUNT
  15524. +            DbgP("comparing 0x%lx.0x%lx with 0x%lx.0x%lx\n",
  15525. +                (long)luid.HighPart, (long)luid.LowPart,
  15526. +                (long)existing_mount->login_id.HighPart,
  15527. +                (long)existing_mount->login_id.LowPart);
  15528. +#endif
  15529. +            if (RtlEqualLuid(&luid, &existing_mount->login_id)) {
  15530. +#ifdef DEBUG_MOUNT
  15531. +                DbgP("Found a matching LUID entry\n");
  15532. +#endif
  15533. +                found_existing_mount = TRUE;
  15534. +                switch(pVNetRootContext->sec_flavor) {
  15535. +                case RPCSEC_AUTH_SYS:
  15536. +                    if (existing_mount->authsys_session != INVALID_HANDLE_VALUE)
  15537. +                        pVNetRootContext->session =
  15538. +                            existing_mount->authsys_session;
  15539. +                    break;
  15540. +                case RPCSEC_AUTHGSS_KRB5:
  15541. +                    if (existing_mount->gssi_session != INVALID_HANDLE_VALUE)
  15542. +                        pVNetRootContext->session = existing_mount->gss_session;
  15543. +                    break;
  15544. +                case RPCSEC_AUTHGSS_KRB5I:
  15545. +                    if (existing_mount->gss_session != INVALID_HANDLE_VALUE)
  15546. +                        pVNetRootContext->session = existing_mount->gssi_session;
  15547. +                    break;
  15548. +                case RPCSEC_AUTHGSS_KRB5P:
  15549. +                    if (existing_mount->gssp_session != INVALID_HANDLE_VALUE)
  15550. +                        pVNetRootContext->session = existing_mount->gssp_session;
  15551. +                    break;
  15552. +                }
  15553. +                if (pVNetRootContext->session &&
  15554. +                        pVNetRootContext->session != INVALID_HANDLE_VALUE)
  15555. +                    found_matching_flavor = 1;
  15556. +                break;
  15557. +            }
  15558. +            if (pEntry->Flink == &pNetRootContext->mounts.head)
  15559. +                break;
  15560. +            pEntry = pEntry->Flink;
  15561. +        }
  15562. +        ExReleaseFastMutex(&pNetRootContext->mountLock);
  15563. +#ifdef DEBUG_MOUNT
  15564. +        if (!found_matching_flavor)
  15565. +            DbgP("Didn't find matching security flavor\n");
  15566. +#endif
  15567. +    }
  15568. +
  15569. +    /* send the mount upcall */
  15570. +    status = nfs41_mount(Config, pVNetRootContext->sec_flavor,
  15571. +        &pVNetRootContext->session, &nfs41d_version,
  15572. +        &pVNetRootContext->FsAttrs);
  15573. +    if (status != STATUS_SUCCESS) {
  15574. +        BOOLEAN MountsEmpty;
  15575. +        nfs41_IsListEmpty(pNetRootContext->mountLock,
  15576. +            pNetRootContext->mounts, MountsEmpty);
  15577. +        if (!found_existing_mount && MountsEmpty)
  15578. +            pNetRootContext->mounts_init = FALSE;
  15579. +        pVNetRootContext->session = INVALID_HANDLE_VALUE;
  15580. +        goto out_free;
  15581. +    }
  15582. +    pVNetRootContext->timeout = Config->timeout;
  15583. +
  15584. +    if (!found_existing_mount) {
  15585. +        /* create a new mount entry and add it to the list */
  15586. +        nfs41_mount_entry *entry;
  15587. +        entry = RxAllocatePoolWithTag(NonPagedPoolNx, sizeof(nfs41_mount_entry),
  15588. +            NFS41_MM_POOLTAG_MOUNT);
  15589. +        if (entry == NULL) {
  15590. +            status = STATUS_INSUFFICIENT_RESOURCES;
  15591. +            goto out_free;
  15592. +        }
  15593. +        entry->authsys_session = entry->gss_session =
  15594. +            entry->gssi_session = entry->gssp_session = INVALID_HANDLE_VALUE;
  15595. +        switch (pVNetRootContext->sec_flavor) {
  15596. +        case RPCSEC_AUTH_SYS:
  15597. +            entry->authsys_session = pVNetRootContext->session; break;
  15598. +        case RPCSEC_AUTHGSS_KRB5:
  15599. +            entry->gss_session = pVNetRootContext->session; break;
  15600. +        case RPCSEC_AUTHGSS_KRB5I:
  15601. +            entry->gssi_session = pVNetRootContext->session; break;
  15602. +        case RPCSEC_AUTHGSS_KRB5P:
  15603. +            entry->gssp_session = pVNetRootContext->session; break;
  15604. +        }
  15605. +        RtlCopyLuid(&entry->login_id, &luid);
  15606. +        /*
  15607. +         * Save mount config so we can use it for
  15608. +         * \\server@port\@(pubnfs4|nfs4)\path mounts later
  15609. +         */
  15610. +        copy_nfs41_mount_config(&entry->Config, Config);
  15611. +        nfs41_AddEntry(pNetRootContext->mountLock,
  15612. +            pNetRootContext->mounts, entry);
  15613. +    } else if (!found_matching_flavor) {
  15614. +        ASSERT(existing_mount != NULL);
  15615. +        /* modify existing mount entry */
  15616. +#ifdef DEBUG_MOUNT
  15617. +        DbgP("Using existing %d flavor session 0x%x\n",
  15618. +            pVNetRootContext->sec_flavor);
  15619. +#endif
  15620. +        switch (pVNetRootContext->sec_flavor) {
  15621. +        case RPCSEC_AUTH_SYS:
  15622. +            existing_mount->authsys_session = pVNetRootContext->session; break;
  15623. +        case RPCSEC_AUTHGSS_KRB5:
  15624. +            existing_mount->gss_session = pVNetRootContext->session; break;
  15625. +        case RPCSEC_AUTHGSS_KRB5I:
  15626. +            existing_mount->gssi_session = pVNetRootContext->session; break;
  15627. +        case RPCSEC_AUTHGSS_KRB5P:
  15628. +            existing_mount->gssp_session = pVNetRootContext->session; break;
  15629. +        }
  15630. +    }
  15631. +    pNetRootContext->nfs41d_version = nfs41d_version;
  15632. +#ifdef DEBUG_MOUNT
  15633. +    DbgP("Saving new session 0x%x\n", pVNetRootContext->session);
  15634. +#endif
  15635. +
  15636. +out_free:
  15637. +    RxFreePool(Config);
  15638. +out:
  15639. +    pCreateNetRootContext->VirtualNetRootStatus = status;
  15640. +    if (pNetRoot->Context == NULL)
  15641. +        pCreateNetRootContext->NetRootStatus = status;
  15642. +    pCreateNetRootContext->Callback(pCreateNetRootContext);
  15643. +
  15644. +    /* RDBSS expects that MRxCreateVNetRoot returns STATUS_PENDING
  15645. +     * on success or failure */
  15646. +    status = STATUS_PENDING;
  15647. +#ifdef DEBUG_MOUNT
  15648. +    DbgEx();
  15649. +#endif
  15650. +    return status;
  15651. +}
  15652. +
  15653. +VOID nfs41_ExtractNetRootName(
  15654. +    IN PUNICODE_STRING FilePathName,
  15655. +    IN PMRX_SRV_CALL SrvCall,
  15656. +    OUT PUNICODE_STRING NetRootName,
  15657. +    OUT PUNICODE_STRING RestOfName OPTIONAL)
  15658. +{
  15659. +    ULONG length = FilePathName->Length;
  15660. +    PWCH w = FilePathName->Buffer;
  15661. +    PWCH wlimit = (PWCH)(((PCHAR)w)+length);
  15662. +    PWCH wlow;
  15663. +
  15664. +    w += (SrvCall->pSrvCallName->Length/sizeof(WCHAR));
  15665. +    NetRootName->Buffer = wlow = w;
  15666. +    /* parse the entire path into NetRootName */
  15667. +#if USE_ENTIRE_PATH_FOR_NETROOT
  15668. +    w = wlimit;
  15669. +#else
  15670. +    for (;;) {
  15671. +        if (w >= wlimit)
  15672. +            break;
  15673. +        if ((*w == OBJ_NAME_PATH_SEPARATOR) && (w != wlow))
  15674. +            break;
  15675. +        w++;
  15676. +    }
  15677. +#endif
  15678. +    NetRootName->Length = NetRootName->MaximumLength
  15679. +                = (USHORT)((PCHAR)w - (PCHAR)wlow);
  15680. +#ifdef DEBUG_MOUNT
  15681. +    DbgP("nfs41_ExtractNetRootName: "
  15682. +        "In: pSrvCall 0x%p PathName='%wZ' SrvCallName='%wZ' "
  15683. +        "Out: NetRootName='%wZ'\n",
  15684. +        SrvCall, FilePathName, SrvCall->pSrvCallName, NetRootName);
  15685. +#endif
  15686. +    return;
  15687. +
  15688. +}
  15689. +
  15690. +NTSTATUS nfs41_FinalizeSrvCall(
  15691. +    PMRX_SRV_CALL pSrvCall,
  15692. +    BOOLEAN Force)
  15693. +{
  15694. +    NTSTATUS status = STATUS_SUCCESS;
  15695. +    PNFS41_SERVER_ENTRY pServerEntry = (PNFS41_SERVER_ENTRY)(pSrvCall->Context);
  15696. +
  15697. +#ifdef DEBUG_MOUNT
  15698. +    DbgEn();
  15699. +#endif
  15700. +    // print_srv_call(pSrvCall);
  15701. +
  15702. +    if (pSrvCall->Context == NULL)
  15703. +        goto out;
  15704. +
  15705. +    InterlockedCompareExchangePointer(&pServerEntry->pRdbssSrvCall,
  15706. +        NULL, pSrvCall);
  15707. +    RxFreePool(pServerEntry);
  15708. +
  15709. +    pSrvCall->Context = NULL;
  15710. +out:
  15711. +#ifdef DEBUG_MOUNT
  15712. +    DbgEx();
  15713. +#endif
  15714. +    return status;
  15715. +}
  15716. +
  15717. +NTSTATUS nfs41_FinalizeNetRoot(
  15718. +    IN OUT PMRX_NET_ROOT pNetRoot,
  15719. +    IN PBOOLEAN ForceDisconnect)
  15720. +{
  15721. +    NTSTATUS status = STATUS_SUCCESS;
  15722. +    PNFS41_NETROOT_EXTENSION pNetRootContext =
  15723. +        NFS41GetNetRootExtension((PMRX_NET_ROOT)pNetRoot);
  15724. +    nfs41_updowncall_entry *tmp;
  15725. +    nfs41_mount_entry *mount_tmp;
  15726. +
  15727. +#ifdef DEBUG_MOUNT
  15728. +    DbgEn();
  15729. +    print_net_root(pNetRoot);
  15730. +#endif
  15731. +
  15732. +    if (pNetRoot->Type != NET_ROOT_DISK && pNetRoot->Type != NET_ROOT_WILD) {
  15733. +        status = STATUS_NOT_SUPPORTED;
  15734. +        goto out;
  15735. +    }
  15736. +
  15737. +    if (pNetRootContext == NULL || !pNetRootContext->mounts_init) {
  15738. +        print_error("nfs41_FinalizeNetRoot: No valid session established\n");
  15739. +        goto out;
  15740. +    }
  15741. +
  15742. +    if (pNetRoot->NumberOfFcbs > 0 || pNetRoot->NumberOfSrvOpens > 0) {
  15743. +        print_error("%d open Fcbs %d open SrvOpens\n", pNetRoot->NumberOfFcbs,
  15744. +            pNetRoot->NumberOfSrvOpens);
  15745. +        goto out;
  15746. +    }
  15747. +
  15748. +    do {
  15749. +        nfs41_GetFirstMountEntry(pNetRootContext->mountLock,
  15750. +            pNetRootContext->mounts, mount_tmp);
  15751. +        if (mount_tmp == NULL)
  15752. +            break;
  15753. +#ifdef DEBUG_MOUNT
  15754. +        DbgP("Removing entry luid 0x%lx.0x%lx from mount list\n",
  15755. +            (long)mount_tmp->login_id.HighPart,
  15756. +            (long)mount_tmp->login_id.LowPart);
  15757. +#endif
  15758. +        if (mount_tmp->authsys_session != INVALID_HANDLE_VALUE) {
  15759. +            status = nfs41_unmount(mount_tmp->authsys_session,
  15760. +                pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT);
  15761. +            if (status)
  15762. +                print_error("nfs41_unmount AUTH_SYS failed with %d\n", status);
  15763. +        }
  15764. +        if (mount_tmp->gss_session != INVALID_HANDLE_VALUE) {
  15765. +            status = nfs41_unmount(mount_tmp->gss_session,
  15766. +                pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT);
  15767. +            if (status)
  15768. +                print_error("nfs41_unmount RPCSEC_GSS_KRB5 failed with %d\n",
  15769. +                            status);
  15770. +        }
  15771. +        if (mount_tmp->gssi_session != INVALID_HANDLE_VALUE) {
  15772. +            status = nfs41_unmount(mount_tmp->gssi_session,
  15773. +                pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT);
  15774. +            if (status)
  15775. +                print_error("nfs41_unmount RPCSEC_GSS_KRB5I failed with %d\n",
  15776. +                            status);
  15777. +        }
  15778. +        if (mount_tmp->gssp_session != INVALID_HANDLE_VALUE) {
  15779. +            status = nfs41_unmount(mount_tmp->gssp_session,
  15780. +                pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT);
  15781. +            if (status)
  15782. +                print_error("nfs41_unmount RPCSEC_GSS_KRB5P failed with %d\n",
  15783. +                            status);
  15784. +        }
  15785. +        nfs41_RemoveEntry(pNetRootContext->mountLock, mount_tmp);
  15786. +        RxFreePool(mount_tmp);
  15787. +        mount_tmp = NULL;
  15788. +    } while (1);
  15789. +    /* ignore any errors from unmount */
  15790. +    status = STATUS_SUCCESS;
  15791. +
  15792. +    // check if there is anything waiting in the upcall or downcall queue
  15793. +    do {
  15794. +        nfs41_GetFirstEntry(upcallLock, upcall, tmp);
  15795. +        if (tmp != NULL) {
  15796. +            DbgP("Removing entry from upcall list\n");
  15797. +            nfs41_RemoveEntry(upcallLock, tmp);
  15798. +            tmp->status = STATUS_INSUFFICIENT_RESOURCES;
  15799. +            KeSetEvent(&tmp->cond, 0, FALSE);
  15800. +        } else
  15801. +            break;
  15802. +    } while (1);
  15803. +
  15804. +    do {
  15805. +        nfs41_GetFirstEntry(downcallLock, downcall, tmp);
  15806. +        if (tmp != NULL) {
  15807. +            DbgP("Removing entry from downcall list\n");
  15808. +            nfs41_RemoveEntry(downcallLock, tmp);
  15809. +            tmp->status = STATUS_INSUFFICIENT_RESOURCES;
  15810. +            KeSetEvent(&tmp->cond, 0, FALSE);
  15811. +        } else
  15812. +            break;
  15813. +    } while (1);
  15814. +out:
  15815. +#ifdef DEBUG_MOUNT
  15816. +    DbgEx();
  15817. +#endif
  15818. +    return status;
  15819. +}
  15820. +
  15821. +NTSTATUS nfs41_FinalizeVNetRoot(
  15822. +    IN OUT PMRX_V_NET_ROOT pVNetRoot,
  15823. +    IN PBOOLEAN ForceDisconnect)
  15824. +{
  15825. +    NTSTATUS status = STATUS_SUCCESS;
  15826. +#ifdef DEBUG_MOUNT
  15827. +    DbgEn();
  15828. +    print_v_net_root(pVNetRoot);
  15829. +#endif
  15830. +    if (pVNetRoot->pNetRoot->Type != NET_ROOT_DISK &&
  15831. +            pVNetRoot->pNetRoot->Type != NET_ROOT_WILD)
  15832. +        status = STATUS_NOT_SUPPORTED;
  15833. +#ifdef DEBUG_MOUNT
  15834. +    DbgEx();
  15835. +#endif
  15836. +    return status;
  15837. +}
  15838. +
  15839. +NTSTATUS GetConnectionHandle(
  15840. +    IN PUNICODE_STRING ConnectionName,
  15841. +    IN PVOID EaBuffer,
  15842. +    IN ULONG EaLength,
  15843. +    OUT PHANDLE Handle)
  15844. +{
  15845. +    NTSTATUS status;
  15846. +    IO_STATUS_BLOCK IoStatusBlock;
  15847. +    OBJECT_ATTRIBUTES ObjectAttributes;
  15848. +
  15849. +#ifdef DEBUG_MOUNT
  15850. +    DbgEn();
  15851. +#endif
  15852. +    InitializeObjectAttributes(&ObjectAttributes, ConnectionName,
  15853. +        OBJ_CASE_INSENSITIVE|OBJ_KERNEL_HANDLE, NULL, NULL);
  15854. +
  15855. +    status = ZwCreateFile(Handle, SYNCHRONIZE, &ObjectAttributes,
  15856. +        &IoStatusBlock, NULL, FILE_ATTRIBUTE_NORMAL,
  15857. +        FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE,
  15858. +        FILE_OPEN_IF,
  15859. +        FILE_CREATE_TREE_CONNECTION | FILE_SYNCHRONOUS_IO_NONALERT,
  15860. +        EaBuffer, EaLength);
  15861. +
  15862. +#ifdef DEBUG_MOUNT
  15863. +    DbgEx();
  15864. +#endif
  15865. +    return status;
  15866. +}
  15867. +
  15868. +NTSTATUS nfs41_GetConnectionInfoFromBuffer(
  15869. +    IN PVOID Buffer,
  15870. +    IN ULONG BufferLen,
  15871. +    OUT PUNICODE_STRING pConnectionName,
  15872. +    OUT PVOID *ppEaBuffer,
  15873. +    OUT PULONG pEaLength)
  15874. +{
  15875. +    NTSTATUS status = STATUS_SUCCESS;
  15876. +    USHORT NameLength, EaPadding;
  15877. +    ULONG EaLength, BufferLenExpected;
  15878. +    PBYTE ptr;
  15879. +
  15880. +    /* make sure buffer is at least big enough for header */
  15881. +    if (BufferLen < sizeof(USHORT) + sizeof(USHORT) + sizeof(ULONG)) {
  15882. +        status = STATUS_BAD_NETWORK_NAME;
  15883. +        print_error("Invalid input buffer.\n");
  15884. +        pConnectionName->Length = pConnectionName->MaximumLength = 0;
  15885. +        *ppEaBuffer = NULL;
  15886. +        *pEaLength = 0;
  15887. +        goto out;
  15888. +    }
  15889. +
  15890. +    ptr = Buffer;
  15891. +    NameLength = *(PUSHORT)ptr;
  15892. +    ptr += sizeof(USHORT);
  15893. +    EaPadding = *(PUSHORT)ptr;
  15894. +    ptr += sizeof(USHORT);
  15895. +    EaLength = *(PULONG)ptr;
  15896. +    ptr += sizeof(ULONG);
  15897. +
  15898. +    /* validate buffer length */
  15899. +    BufferLenExpected = sizeof(USHORT) + sizeof(USHORT) + sizeof(ULONG) +
  15900. +        NameLength + EaPadding + EaLength;
  15901. +    if (BufferLen != BufferLenExpected) {
  15902. +        status = STATUS_BAD_NETWORK_NAME;
  15903. +        print_error("Received buffer of length %lu, but expected %lu bytes.\n",
  15904. +            BufferLen, BufferLenExpected);
  15905. +        pConnectionName->Length = pConnectionName->MaximumLength = 0;
  15906. +        *ppEaBuffer = NULL;
  15907. +        *pEaLength = 0;
  15908. +        goto out;
  15909. +    }
  15910. +
  15911. +    pConnectionName->Buffer = (PWCH)ptr;
  15912. +    pConnectionName->Length = NameLength - sizeof(WCHAR);
  15913. +    pConnectionName->MaximumLength = NameLength;
  15914. +
  15915. +    if (EaLength)
  15916. +        *ppEaBuffer = ptr + NameLength + EaPadding;
  15917. +    else
  15918. +        *ppEaBuffer = NULL;
  15919. +    *pEaLength = EaLength;
  15920. +
  15921. +out:
  15922. +    return status;
  15923. +}
  15924. +
  15925. +NTSTATUS nfs41_CreateConnection(
  15926. +    IN PRX_CONTEXT RxContext,
  15927. +    OUT PBOOLEAN PostToFsp)
  15928. +{
  15929. +    NTSTATUS status = STATUS_SUCCESS;
  15930. +    HANDLE Handle = INVALID_HANDLE_VALUE;
  15931. +    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  15932. +    PVOID Buffer = LowIoContext->ParamsFor.IoCtl.pInputBuffer, EaBuffer;
  15933. +    ULONG BufferLen = LowIoContext->ParamsFor.IoCtl.InputBufferLength, EaLength;
  15934. +    UNICODE_STRING FileName;
  15935. +    BOOLEAN Wait = BooleanFlagOn(RxContext->Flags, RX_CONTEXT_FLAG_WAIT);
  15936. +
  15937. +#ifdef DEBUG_MOUNT
  15938. +    DbgEn();
  15939. +#endif
  15940. +
  15941. +    if (!Wait) {
  15942. +        //just post right now!
  15943. +        DbgP("returning STATUS_PENDING\n");
  15944. +        *PostToFsp = TRUE;
  15945. +        status = STATUS_PENDING;
  15946. +        goto out;
  15947. +    }
  15948. +
  15949. +    status = nfs41_GetConnectionInfoFromBuffer(Buffer, BufferLen,
  15950. +        &FileName, &EaBuffer, &EaLength);
  15951. +    if (status != STATUS_SUCCESS)
  15952. +        goto out;
  15953. +
  15954. +    status = GetConnectionHandle(&FileName, EaBuffer, EaLength, &Handle);
  15955. +    if (!status && Handle != INVALID_HANDLE_VALUE)
  15956. +        ZwClose(Handle);
  15957. +out:
  15958. +#ifdef DEBUG_MOUNT
  15959. +    DbgEx();
  15960. +#endif
  15961. +    return status;
  15962. +}
  15963. +
  15964. +NTSTATUS nfs41_DeleteConnection(
  15965. +    IN PRX_CONTEXT RxContext,
  15966. +    OUT PBOOLEAN PostToFsp)
  15967. +{
  15968. +    NTSTATUS status = STATUS_INVALID_PARAMETER;
  15969. +    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  15970. +    PWCHAR ConnectName = LowIoContext->ParamsFor.IoCtl.pInputBuffer;
  15971. +    ULONG ConnectNameLen = LowIoContext->ParamsFor.IoCtl.InputBufferLength;
  15972. +    HANDLE Handle;
  15973. +    UNICODE_STRING FileName;
  15974. +    PFILE_OBJECT pFileObject;
  15975. +    BOOLEAN Wait = BooleanFlagOn(RxContext->Flags, RX_CONTEXT_FLAG_WAIT);
  15976. +
  15977. +#ifdef DEBUG_MOUNT
  15978. +    DbgEn();
  15979. +#endif
  15980. +
  15981. +    if (!Wait) {
  15982. +        //just post right now!
  15983. +        *PostToFsp = TRUE;
  15984. +        DbgP("returning STATUS_PENDING\n");
  15985. +        status = STATUS_PENDING;
  15986. +        goto out;
  15987. +    }
  15988. +
  15989. +    FileName.Buffer = ConnectName;
  15990. +    FileName.Length = (USHORT) ConnectNameLen - sizeof(WCHAR);
  15991. +    FileName.MaximumLength = (USHORT) ConnectNameLen;
  15992. +
  15993. +    status = GetConnectionHandle(&FileName, NULL, 0, &Handle);
  15994. +    if (status != STATUS_SUCCESS)
  15995. +        goto out;
  15996. +
  15997. +    status = ObReferenceObjectByHandle(Handle, 0L, NULL, KernelMode,
  15998. +                (PVOID *)&pFileObject, NULL);
  15999. +    if (NT_SUCCESS(status)) {
  16000. +        PV_NET_ROOT VNetRoot;
  16001. +
  16002. +        // VNetRoot exists as FOBx in the FsContext2
  16003. +        VNetRoot = (PV_NET_ROOT) pFileObject->FsContext2;
  16004. +        // make sure the node looks right
  16005. +        if (NodeType(VNetRoot) == RDBSS_NTC_V_NETROOT)
  16006. +        {
  16007. +#ifdef DEBUG_MOUNT
  16008. +            DbgP("Calling RxFinalizeConnection for NetRoot 0x%p from VNetRoot 0x%p\n",
  16009. +                VNetRoot->NetRoot, VNetRoot);
  16010. +#endif
  16011. +            status = RxFinalizeConnection(VNetRoot->NetRoot, VNetRoot, TRUE);
  16012. +        }
  16013. +        else
  16014. +            status = STATUS_BAD_NETWORK_NAME;
  16015. +
  16016. +        ObDereferenceObject(pFileObject);
  16017. +    }
  16018. +    ZwClose(Handle);
  16019. +out:
  16020. +#ifdef DEBUG_MOUNT
  16021. +    DbgEx();
  16022. +#endif
  16023. +    return status;
  16024. +}
  16025. diff --git a/sys/nfs41sys_openclose.c b/sys/nfs41sys_openclose.c
  16026. new file mode 100644
  16027. index 0000000..14f3471
  16028. --- /dev/null
  16029. +++ b/sys/nfs41sys_openclose.c
  16030. @@ -0,0 +1,1026 @@
  16031. +/* NFSv4.1 client for Windows
  16032. + * Copyright (C) 2012 The Regents of the University of Michigan
  16033. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  16034. + *
  16035. + * Olga Kornievskaia <aglo@umich.edu>
  16036. + * Casey Bodley <cbodley@umich.edu>
  16037. + * Roland Mainz <roland.mainz@nrubsig.org>
  16038. + *
  16039. + * This library is free software; you can redistribute it and/or modify it
  16040. + * under the terms of the GNU Lesser General Public License as published by
  16041. + * the Free Software Foundation; either version 2.1 of the License, or (at
  16042. + * your option) any later version.
  16043. + *
  16044. + * This library is distributed in the hope that it will be useful, but
  16045. + * without any warranty; without even the implied warranty of merchantability
  16046. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  16047. + * License for more details.
  16048. + *
  16049. + * You should have received a copy of the GNU Lesser General Public License
  16050. + * along with this library; if not, write to the Free Software Foundation,
  16051. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  16052. + */
  16053. +
  16054. +#ifndef _KERNEL_MODE
  16055. +#error module requires kernel mode
  16056. +#endif
  16057. +
  16058. +#if ((__STDC_VERSION__-0) < 201710L)
  16059. +#error Code requires ISO C17
  16060. +#endif
  16061. +
  16062. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  16063. +#if _MSC_VER >= 1900
  16064. +#if defined(_WIN64) && defined(_M_X64)
  16065. +#ifndef _AMD64_
  16066. +#define _AMD64_
  16067. +#endif
  16068. +#elif defined(_WIN32) && defined(_M_IX86)
  16069. +#ifndef _X86_
  16070. +#define _X86_
  16071. +#endif
  16072. +#elif defined(_WIN64) && defined(_M_ARM64)
  16073. +#ifndef _ARM64_
  16074. +#define _ARM64_
  16075. +#endif
  16076. +#elif defined(_WIN32) && defined(_M_ARM)
  16077. +#ifndef _ARM_
  16078. +#define _ARM_
  16079. +#endif
  16080. +#else
  16081. +#error Unsupported arch
  16082. +#endif
  16083. +#endif /* _MSC_VER >= 1900 */
  16084. +
  16085. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  16086. +#include <rx.h>
  16087. +#include <windef.h>
  16088. +#include <winerror.h>
  16089. +
  16090. +#include <Ntstrsafe.h>
  16091. +
  16092. +#include "nfs41sys_buildconfig.h"
  16093. +
  16094. +#include "nfs41_driver.h"
  16095. +#include "nfs41sys_debug.h"
  16096. +#include "nfs41_build_features.h"
  16097. +
  16098. +#include "nfs41sys_driver.h"
  16099. +#include "nfs41sys_util.h"
  16100. +
  16101. +static
  16102. +NTSTATUS nfs41_get_sec_ctx(
  16103. +    IN enum _SECURITY_IMPERSONATION_LEVEL level,
  16104. +    OUT PSECURITY_CLIENT_CONTEXT out_ctx)
  16105. +{
  16106. +    NTSTATUS status;
  16107. +    SECURITY_SUBJECT_CONTEXT ctx;
  16108. +    SECURITY_QUALITY_OF_SERVICE sec_qos;
  16109. +
  16110. +    SeCaptureSubjectContext(&ctx);
  16111. +    sec_qos.ContextTrackingMode = SECURITY_STATIC_TRACKING;
  16112. +    sec_qos.ImpersonationLevel = level;
  16113. +    sec_qos.Length = sizeof(SECURITY_QUALITY_OF_SERVICE);
  16114. +    sec_qos.EffectiveOnly = 0;
  16115. +    /*
  16116. +     * Arg |ServerIsRemote| must be |FALSE|, otherwise processes
  16117. +     * like Cygwin setup-x86_64.exe can fail during "Activation
  16118. +     * Context" creation in
  16119. +     * |SeCreateClientSecurityFromSubjectContext()| with
  16120. +     * |STATUS_BAD_IMPERSONATION_LEVEL|
  16121. +     */
  16122. +    status = SeCreateClientSecurityFromSubjectContext(&ctx, &sec_qos,
  16123. +        FALSE, out_ctx);
  16124. +    if (status != STATUS_SUCCESS) {
  16125. +        print_error("SeCreateClientSecurityFromSubjectContext "
  16126. +            "failed with 0x%x\n", status);
  16127. +    }
  16128. +#ifdef DEBUG_SECURITY_TOKEN
  16129. +    DbgP("Created client security token 0x%p\n", out_ctx->ClientToken);
  16130. +#endif
  16131. +    SeReleaseSubjectContext(&ctx);
  16132. +
  16133. +    return status;
  16134. +}
  16135. +
  16136. +NTSTATUS marshal_nfs41_open(
  16137. +    nfs41_updowncall_entry *entry,
  16138. +    unsigned char *buf,
  16139. +    ULONG buf_len,
  16140. +    ULONG *len)
  16141. +{
  16142. +    NTSTATUS status = STATUS_SUCCESS;
  16143. +    ULONG header_len = 0;
  16144. +    unsigned char *tmp = buf;
  16145. +
  16146. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  16147. +    if (status) goto out;
  16148. +    else tmp += *len;
  16149. +
  16150. +    header_len = *len + length_as_utf8(entry->filename) +
  16151. +        7 * sizeof(ULONG) +
  16152. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  16153. +        2 * sizeof(DWORD) +
  16154. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  16155. +        2 * sizeof(HANDLE) +
  16156. +        length_as_utf8(&entry->u.Open.symlink);
  16157. +    if (header_len > buf_len) {
  16158. +        status = STATUS_INSUFFICIENT_RESOURCES;
  16159. +        goto out;
  16160. +    }
  16161. +    status = marshall_unicode_as_utf8(&tmp, entry->filename);
  16162. +    if (status) goto out;
  16163. +    RtlCopyMemory(tmp, &entry->u.Open.access_mask,
  16164. +        sizeof(entry->u.Open.access_mask));
  16165. +    tmp += sizeof(entry->u.Open.access_mask);
  16166. +    RtlCopyMemory(tmp, &entry->u.Open.access_mode,
  16167. +        sizeof(entry->u.Open.access_mode));
  16168. +    tmp += sizeof(entry->u.Open.access_mode);
  16169. +    RtlCopyMemory(tmp, &entry->u.Open.attrs, sizeof(entry->u.Open.attrs));
  16170. +    tmp += sizeof(entry->u.Open.attrs);
  16171. +    RtlCopyMemory(tmp, &entry->u.Open.copts, sizeof(entry->u.Open.copts));
  16172. +    tmp += sizeof(entry->u.Open.copts);
  16173. +    RtlCopyMemory(tmp, &entry->u.Open.disp, sizeof(entry->u.Open.disp));
  16174. +    tmp += sizeof(entry->u.Open.disp);
  16175. +    RtlCopyMemory(tmp, &entry->u.Open.open_owner_id,
  16176. +        sizeof(entry->u.Open.open_owner_id));
  16177. +    tmp += sizeof(entry->u.Open.open_owner_id);
  16178. +    RtlCopyMemory(tmp, &entry->u.Open.mode, sizeof(DWORD));
  16179. +    tmp += sizeof(DWORD);
  16180. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  16181. +    RtlCopyMemory(tmp, &entry->u.Open.owner_local_uid, sizeof(DWORD));
  16182. +    tmp += sizeof(DWORD);
  16183. +    RtlCopyMemory(tmp, &entry->u.Open.owner_group_local_gid, sizeof(DWORD));
  16184. +    tmp += sizeof(DWORD);
  16185. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  16186. +    RtlCopyMemory(tmp, &entry->u.Open.srv_open, sizeof(HANDLE));
  16187. +    tmp += sizeof(HANDLE);
  16188. +    status = marshall_unicode_as_utf8(&tmp, &entry->u.Open.symlink);
  16189. +    if (status) goto out;
  16190. +
  16191. +    __try {
  16192. +        if (entry->u.Open.EaMdl) {
  16193. +            entry->u.Open.EaBuffer =
  16194. +                MmMapLockedPagesSpecifyCache(entry->u.Open.EaMdl,
  16195. +                    UserMode, MmCached, NULL, TRUE,
  16196. +                    NormalPagePriority|MdlMappingNoExecute);
  16197. +            if (entry->u.Open.EaBuffer == NULL) {
  16198. +                print_error("marshal_nfs41_open: "
  16199. +                    "MmMapLockedPagesSpecifyCache() failed to "
  16200. +                    "map pages\n");
  16201. +                status = STATUS_INSUFFICIENT_RESOURCES;
  16202. +                goto out;
  16203. +            }
  16204. +        }
  16205. +    } __except(EXCEPTION_EXECUTE_HANDLER) {
  16206. +        print_error("marshal_nfs41_open: Call to "
  16207. +            "MmMapLockedPagesSpecifyCache() failed "
  16208. +            "due to exception 0x%x\n", (int)GetExceptionCode());
  16209. +        status = STATUS_ACCESS_VIOLATION;
  16210. +        goto out;
  16211. +    }
  16212. +    RtlCopyMemory(tmp, &entry->u.Open.EaBuffer, sizeof(HANDLE));
  16213. +    *len = header_len;
  16214. +
  16215. +#ifdef DEBUG_MARSHAL_DETAIL
  16216. +    DbgP("marshal_nfs41_open: name='%wZ' mask=0x%x access=0x%x attrs=0x%x "
  16217. +         "opts=0x%x dispo=0x%x open_owner_id=0x%x mode=0%o "
  16218. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  16219. +         "owner_local_uid=%lu owner_group_local_gid=%lu "
  16220. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  16221. +         "srv_open=0x%p ea=0x%p\n",
  16222. +         entry->filename, entry->u.Open.access_mask,
  16223. +         entry->u.Open.access_mode, entry->u.Open.attrs, entry->u.Open.copts,
  16224. +         entry->u.Open.disp, entry->u.Open.open_owner_id, entry->u.Open.mode,
  16225. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  16226. +         entry->u.Open.owner_local_uid,entry->u.Open.owner_group_local_gid,
  16227. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  16228. +         entry->u.Open.srv_open, entry->u.Open.EaBuffer);
  16229. +#endif
  16230. +out:
  16231. +    return status;
  16232. +}
  16233. +
  16234. +NTSTATUS marshal_nfs41_close(
  16235. +    nfs41_updowncall_entry *entry,
  16236. +    unsigned char *buf,
  16237. +    ULONG buf_len,
  16238. +    ULONG *len)
  16239. +{
  16240. +    NTSTATUS status = STATUS_SUCCESS;
  16241. +    ULONG header_len = 0;
  16242. +    unsigned char *tmp = buf;
  16243. +
  16244. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  16245. +    if (status) goto out;
  16246. +    else tmp += *len;
  16247. +
  16248. +    header_len = *len + sizeof(BOOLEAN) + sizeof(HANDLE);
  16249. +    if (entry->u.Close.remove)
  16250. +        header_len += length_as_utf8(entry->filename) +
  16251. +            sizeof(BOOLEAN);
  16252. +
  16253. +    if (header_len > buf_len) {
  16254. +        status = STATUS_INSUFFICIENT_RESOURCES;
  16255. +        goto out;
  16256. +    }
  16257. +    RtlCopyMemory(tmp, &entry->u.Close.remove, sizeof(BOOLEAN));
  16258. +    tmp += sizeof(BOOLEAN);
  16259. +    RtlCopyMemory(tmp, &entry->u.Close.srv_open, sizeof(HANDLE));
  16260. +    if (entry->u.Close.remove) {
  16261. +        tmp += sizeof(HANDLE);
  16262. +        status = marshall_unicode_as_utf8(&tmp, entry->filename);
  16263. +        if (status) goto out;
  16264. +        RtlCopyMemory(tmp, &entry->u.Close.renamed, sizeof(BOOLEAN));
  16265. +    }
  16266. +    *len = header_len;
  16267. +
  16268. +#ifdef DEBUG_MARSHAL_DETAIL
  16269. +    DbgP("marshal_nfs41_close: name='%wZ' remove=%d srv_open=0x%p renamed=%d\n",
  16270. +        entry->filename->Length?entry->filename:&SLASH,
  16271. +        entry->u.Close.remove, entry->u.Close.srv_open, entry->u.Close.renamed);
  16272. +#endif
  16273. +out:
  16274. +    return status;
  16275. +}
  16276. +
  16277. +NTSTATUS unmarshal_nfs41_open(
  16278. +    nfs41_updowncall_entry *cur,
  16279. +    unsigned char **buf)
  16280. +{
  16281. +    NTSTATUS status = STATUS_SUCCESS;
  16282. +
  16283. +    __try {
  16284. +        if (cur->u.Open.EaBuffer)
  16285. +            MmUnmapLockedPages(cur->u.Open.EaBuffer, cur->u.Open.EaMdl);
  16286. +    } __except(EXCEPTION_EXECUTE_HANDLER) {
  16287. +        print_error("MmUnmapLockedPages thrown exception=0x%0x\n", GetExceptionCode());
  16288. +        status = cur->status = STATUS_ACCESS_VIOLATION;
  16289. +        goto out;
  16290. +    }
  16291. +
  16292. +    RtlCopyMemory(&cur->u.Open.binfo, *buf, sizeof(FILE_BASIC_INFORMATION));
  16293. +    *buf += sizeof(FILE_BASIC_INFORMATION);
  16294. +    RtlCopyMemory(&cur->u.Open.sinfo, *buf, sizeof(FILE_STANDARD_INFORMATION));
  16295. +    *buf += sizeof(FILE_STANDARD_INFORMATION);
  16296. +    RtlCopyMemory(&cur->open_state, *buf, sizeof(HANDLE));
  16297. +    *buf += sizeof(HANDLE);
  16298. +    RtlCopyMemory(&cur->u.Open.mode, *buf, sizeof(DWORD));
  16299. +    *buf += sizeof(DWORD);
  16300. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  16301. +    RtlCopyMemory(&cur->u.Open.owner_local_uid, *buf, sizeof(DWORD));
  16302. +    *buf += sizeof(DWORD);
  16303. +    RtlCopyMemory(&cur->u.Open.owner_group_local_gid, *buf, sizeof(DWORD));
  16304. +    *buf += sizeof(DWORD);
  16305. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  16306. +    RtlCopyMemory(&cur->ChangeTime, *buf, sizeof(ULONGLONG));
  16307. +    *buf += sizeof(ULONGLONG);
  16308. +    RtlCopyMemory(&cur->u.Open.deleg_type, *buf, sizeof(DWORD));
  16309. +    *buf += sizeof(DWORD);
  16310. +    if (cur->errno == ERROR_REPARSE) {
  16311. +        RtlCopyMemory(&cur->u.Open.symlink_embedded, *buf, sizeof(BOOLEAN));
  16312. +        *buf += sizeof(BOOLEAN);
  16313. +        RtlCopyMemory(&cur->u.Open.symlink.MaximumLength, *buf,
  16314. +            sizeof(USHORT));
  16315. +        *buf += sizeof(USHORT);
  16316. +        cur->u.Open.symlink.Length = cur->u.Open.symlink.MaximumLength -
  16317. +            sizeof(WCHAR);
  16318. +        cur->u.Open.symlink.Buffer = RxAllocatePoolWithTag(NonPagedPoolNx,
  16319. +            cur->u.Open.symlink.MaximumLength, NFS41_MM_POOLTAG);
  16320. +        if (cur->u.Open.symlink.Buffer == NULL) {
  16321. +            cur->status = STATUS_INSUFFICIENT_RESOURCES;
  16322. +            status = STATUS_UNSUCCESSFUL;
  16323. +            goto out;
  16324. +        }
  16325. +        RtlCopyMemory(cur->u.Open.symlink.Buffer, *buf,
  16326. +            cur->u.Open.symlink.MaximumLength);
  16327. +#ifdef DEBUG_MARSHAL_DETAIL
  16328. +        DbgP("unmarshal_nfs41_open: ERROR_REPARSE -> '%wZ'\n", &cur->u.Open.symlink);
  16329. +#endif
  16330. +    }
  16331. +#ifdef DEBUG_MARSHAL_DETAIL
  16332. +    DbgP("unmarshal_nfs41_open: open_state 0x%x mode 0%o "
  16333. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  16334. +        "owner_local_uid %u owner_group_local_gid %u "
  16335. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  16336. +        "changeattr %llu "
  16337. +        "deleg_type %d\n", cur->open_state, cur->u.Open.mode,
  16338. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  16339. +        cur->u.Open.owner_local_uid, cur->u.Open.owner_group_local_gid,
  16340. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  16341. +        cur->ChangeTime, cur->u.Open.deleg_type);
  16342. +#endif /* DEBUG_MARSHAL_DETAIL */
  16343. +out:
  16344. +    return status;
  16345. +}
  16346. +
  16347. +static BOOLEAN isDataAccess(
  16348. +    ACCESS_MASK mask)
  16349. +{
  16350. +    if (mask & (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA))
  16351. +        return TRUE;
  16352. +    return FALSE;
  16353. +}
  16354. +
  16355. +static BOOLEAN isOpen2Create(
  16356. +    ULONG disposition)
  16357. +{
  16358. +    if (disposition == FILE_CREATE || disposition == FILE_OPEN_IF ||
  16359. +            disposition == FILE_OVERWRITE_IF || disposition == FILE_SUPERSEDE)
  16360. +        return TRUE;
  16361. +    return FALSE;
  16362. +}
  16363. +
  16364. +static BOOLEAN areOpenParamsValid(NT_CREATE_PARAMETERS *params)
  16365. +{
  16366. +    /* from ms-fsa page 52 */
  16367. +    if ((params->CreateOptions & FILE_DELETE_ON_CLOSE) &&
  16368. +            !(params->DesiredAccess & DELETE))
  16369. +        return FALSE;
  16370. +    if ((params->CreateOptions & FILE_DIRECTORY_FILE) &&
  16371. +            (params->Disposition == FILE_SUPERSEDE ||
  16372. +                params->Disposition == FILE_OVERWRITE ||
  16373. +                params->Disposition == FILE_OVERWRITE_IF))
  16374. +        return FALSE;
  16375. +    if ((params->CreateOptions & FILE_NO_INTERMEDIATE_BUFFERING) &&
  16376. +            (params->DesiredAccess & FILE_APPEND_DATA) &&
  16377. +            !(params->DesiredAccess & FILE_WRITE_DATA))
  16378. +        return FALSE;
  16379. +    /* from ms-fsa 3.1.5.1.1 page 56 */
  16380. +    if ((params->CreateOptions & FILE_DIRECTORY_FILE) &&
  16381. +            (params->FileAttributes & FILE_ATTRIBUTE_TEMPORARY))
  16382. +        return FALSE;
  16383. +    return TRUE;
  16384. +}
  16385. +
  16386. +NTSTATUS map_open_errors(
  16387. +    DWORD status,
  16388. +    USHORT len)
  16389. +{
  16390. +    switch (status) {
  16391. +    case NO_ERROR:                      return STATUS_SUCCESS;
  16392. +    case ERROR_ACCESS_DENIED:
  16393. +        if (len > 0)                    return STATUS_ACCESS_DENIED;
  16394. +        else                            return STATUS_SUCCESS;
  16395. +    case ERROR_INVALID_REPARSE_DATA:
  16396. +    case ERROR_INVALID_NAME:            return STATUS_OBJECT_NAME_INVALID;
  16397. +    case ERROR_FILE_EXISTS:             return STATUS_OBJECT_NAME_COLLISION;
  16398. +    case ERROR_FILE_INVALID:            return STATUS_FILE_INVALID;
  16399. +    case ERROR_FILE_NOT_FOUND:          return STATUS_OBJECT_NAME_NOT_FOUND;
  16400. +    case ERROR_FILENAME_EXCED_RANGE:    return STATUS_NAME_TOO_LONG;
  16401. +    case ERROR_NETWORK_ACCESS_DENIED:   return STATUS_NETWORK_ACCESS_DENIED;
  16402. +    case ERROR_PATH_NOT_FOUND:          return STATUS_OBJECT_PATH_NOT_FOUND;
  16403. +    case ERROR_BAD_NETPATH:             return STATUS_BAD_NETWORK_PATH;
  16404. +    case ERROR_SHARING_VIOLATION:       return STATUS_SHARING_VIOLATION;
  16405. +    case ERROR_REPARSE:                 return STATUS_REPARSE;
  16406. +    case ERROR_TOO_MANY_LINKS:          return STATUS_TOO_MANY_LINKS;
  16407. +    case ERROR_DIRECTORY:               return STATUS_FILE_IS_A_DIRECTORY;
  16408. +    case ERROR_BAD_FILE_TYPE:           return STATUS_NOT_A_DIRECTORY;
  16409. +    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  16410. +    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  16411. +    case ERROR_FILE_TOO_LARGE:          return STATUS_FILE_TOO_LARGE;
  16412. +    case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  16413. +    default:
  16414. +        print_error("[ERROR] nfs41_Create: upcall returned ERROR_0x%x "
  16415. +            "returning STATUS_INSUFFICIENT_RESOURCES\n", status);
  16416. +    case ERROR_OUTOFMEMORY:             return STATUS_INSUFFICIENT_RESOURCES;
  16417. +    }
  16418. +}
  16419. +
  16420. +static DWORD map_disposition_to_create_retval(
  16421. +    DWORD disposition,
  16422. +    DWORD errno)
  16423. +{
  16424. +    switch(disposition) {
  16425. +    case FILE_SUPERSEDE:
  16426. +        if (errno == ERROR_FILE_NOT_FOUND)  return FILE_CREATED;
  16427. +        else                                return FILE_SUPERSEDED;
  16428. +    case FILE_CREATE:                       return FILE_CREATED;
  16429. +    case FILE_OPEN:                         return FILE_OPENED;
  16430. +    case FILE_OPEN_IF:
  16431. +        if (errno == ERROR_FILE_NOT_FOUND)  return FILE_CREATED;
  16432. +        else                                return FILE_OPENED;
  16433. +    case FILE_OVERWRITE:                    return FILE_OVERWRITTEN;
  16434. +    case FILE_OVERWRITE_IF:
  16435. +        if (errno == ERROR_FILE_NOT_FOUND)  return FILE_CREATED;
  16436. +        else                                return FILE_OVERWRITTEN;
  16437. +    default:
  16438. +        print_error("unknown disposition %d\n", disposition);
  16439. +        return FILE_OPENED;
  16440. +    }
  16441. +}
  16442. +
  16443. +static BOOLEAN create_should_pass_ea(
  16444. +    IN PFILE_FULL_EA_INFORMATION ea,
  16445. +    IN ULONG disposition)
  16446. +{
  16447. +    /* don't pass cygwin EAs */
  16448. +    if (AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength)
  16449. +        || AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength)
  16450. +        || AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength))
  16451. +        return FALSE;
  16452. +    /* only set EAs on file creation */
  16453. +    return disposition == FILE_SUPERSEDE || disposition == FILE_CREATE
  16454. +        || disposition == FILE_OPEN_IF || disposition == FILE_OVERWRITE
  16455. +        || disposition == FILE_OVERWRITE_IF;
  16456. +}
  16457. +
  16458. +NTSTATUS check_nfs41_create_args(
  16459. +    IN PRX_CONTEXT RxContext)
  16460. +{
  16461. +    NTSTATUS status = STATUS_SUCCESS;
  16462. +    PNT_CREATE_PARAMETERS params = &RxContext->Create.NtCreateParameters;
  16463. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  16464. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  16465. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  16466. +    __notnull PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs =
  16467. +        &pVNetRootContext->FsAttrs;
  16468. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  16469. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  16470. +    __notnull PMRX_FCB Fcb = RxContext->pFcb;
  16471. +    __notnull PNFS41_FCB nfs41_fcb = (PNFS41_FCB)Fcb->Context;
  16472. +    PFILE_FULL_EA_INFORMATION ea = (PFILE_FULL_EA_INFORMATION)
  16473. +        RxContext->CurrentIrp->AssociatedIrp.SystemBuffer;
  16474. +
  16475. +    if (Fcb->pNetRoot->Type != NET_ROOT_DISK &&
  16476. +            Fcb->pNetRoot->Type != NET_ROOT_WILD) {
  16477. +        print_error("nfs41_Create: Unsupported NetRoot Type %u\n",
  16478. +            Fcb->pNetRoot->Type);
  16479. +        status = STATUS_NOT_SUPPORTED;
  16480. +        goto out;
  16481. +    }
  16482. +
  16483. +    if (FlagOn(Fcb->FcbState, FCB_STATE_PAGING_FILE )) {
  16484. +        print_error("FCB_STATE_PAGING_FILE not implemented\n");
  16485. +        status = STATUS_NOT_IMPLEMENTED;
  16486. +        goto out;
  16487. +    }
  16488. +
  16489. +    if (!pNetRootContext->mounts_init) {
  16490. +        print_error("nfs41_Create: No valid session established\n");
  16491. +        status = STATUS_INSUFFICIENT_RESOURCES;
  16492. +        goto out;
  16493. +    }
  16494. +
  16495. +    if (isStream(SrvOpen->pAlreadyPrefixedName)) {
  16496. +        status = STATUS_NOT_SUPPORTED;
  16497. +        goto out;
  16498. +    }
  16499. +
  16500. +    if (pVNetRootContext->read_only &&
  16501. +            (params->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA))) {
  16502. +        status = STATUS_MEDIA_WRITE_PROTECTED;
  16503. +        goto out;
  16504. +    }
  16505. +
  16506. +    /* if FCB was marked for deletion and opened multiple times, as soon
  16507. +     * as first close happen, FCB transitions into delete_pending state
  16508. +     * no more opens allowed
  16509. +     */
  16510. +    if (Fcb->OpenCount && nfs41_fcb->DeletePending) {
  16511. +        status = STATUS_DELETE_PENDING;
  16512. +        goto out;
  16513. +    }
  16514. +
  16515. +    /* ms-fsa: 3.1.5.1.2.1 page 68 */
  16516. +    if (Fcb->OpenCount && nfs41_fcb->StandardInfo.DeletePending &&
  16517. +            !(params->ShareAccess & FILE_SHARE_DELETE) &&
  16518. +                (params->DesiredAccess & (FILE_EXECUTE | FILE_READ_DATA |
  16519. +                    FILE_WRITE_DATA | FILE_APPEND_DATA))) {
  16520. +        status = STATUS_SHARING_VIOLATION;
  16521. +        goto out;
  16522. +    }
  16523. +
  16524. +    /* rdbss seems miss this sharing_violation check */
  16525. +    if (Fcb->OpenCount && params->Disposition == FILE_SUPERSEDE) {
  16526. +        if ((!RxContext->CurrentIrpSp->FileObject->SharedRead &&
  16527. +                (params->DesiredAccess & FILE_READ_DATA)) ||
  16528. +            ((!RxContext->CurrentIrpSp->FileObject->SharedWrite &&
  16529. +                (params->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA |
  16530. +                    FILE_WRITE_ATTRIBUTES))) ||
  16531. +            (!RxContext->CurrentIrpSp->FileObject->SharedDelete &&
  16532. +                (params->DesiredAccess & DELETE)))) {
  16533. +            status = STATUS_SHARING_VIOLATION;
  16534. +            goto out;
  16535. +        }
  16536. +    }
  16537. +    if (isFilenameTooLong(SrvOpen->pAlreadyPrefixedName, pVNetRootContext)) {
  16538. +        status = STATUS_OBJECT_NAME_INVALID;
  16539. +        goto out;
  16540. +    }
  16541. +
  16542. +    /* We do not support oplocks (yet) */
  16543. +    if (params->CreateOptions & FILE_OPEN_REQUIRING_OPLOCK) {
  16544. +        status = STATUS_INVALID_PARAMETER;
  16545. +        goto out;
  16546. +    }
  16547. +
  16548. +    if (!areOpenParamsValid(params)) {
  16549. +        status = STATUS_INVALID_PARAMETER;
  16550. +        goto out;
  16551. +    }
  16552. +
  16553. +    /* from ms-fsa 3.1.5.1.1 page 56 */
  16554. +    if ((params->CreateOptions & FILE_DELETE_ON_CLOSE) &&
  16555. +            (params->FileAttributes & FILE_ATTRIBUTE_READONLY)) {
  16556. +        status = STATUS_CANNOT_DELETE;
  16557. +        goto out;
  16558. +    }
  16559. +
  16560. +    if (ea) {
  16561. +        /* ignore cygwin EAs when checking support and access */
  16562. +        if (!AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength) &&
  16563. +            !AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength) &&
  16564. +            !AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) {
  16565. +            if (!(FsAttrs->FileSystemAttributes & FILE_SUPPORTS_EXTENDED_ATTRIBUTES)) {
  16566. +                status = STATUS_EAS_NOT_SUPPORTED;
  16567. +                goto out;
  16568. +            }
  16569. +        }
  16570. +    } else if (RxContext->CurrentIrpSp->Parameters.Create.EaLength) {
  16571. +        status = STATUS_INVALID_PARAMETER;
  16572. +        goto out;
  16573. +    }
  16574. +
  16575. +out:
  16576. +    return status;
  16577. +}
  16578. +
  16579. +NTSTATUS nfs41_Create(
  16580. +    IN OUT PRX_CONTEXT RxContext)
  16581. +{
  16582. +    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  16583. +    nfs41_updowncall_entry *entry = NULL;
  16584. +    PNT_CREATE_PARAMETERS params = &RxContext->Create.NtCreateParameters;
  16585. +    PFILE_FULL_EA_INFORMATION ea = (PFILE_FULL_EA_INFORMATION)
  16586. +        RxContext->CurrentIrp->AssociatedIrp.SystemBuffer;
  16587. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  16588. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  16589. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  16590. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  16591. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  16592. +    __notnull PMRX_FCB Fcb = RxContext->pFcb;
  16593. +    __notnull PNFS41_FCB nfs41_fcb = (PNFS41_FCB)Fcb->Context;
  16594. +    PNFS41_FOBX nfs41_fobx = NULL;
  16595. +    BOOLEAN oldDeletePending = nfs41_fcb->StandardInfo.DeletePending;
  16596. +#ifdef ENABLE_TIMINGS
  16597. +    LARGE_INTEGER t1, t2;
  16598. +    t1 = KeQueryPerformanceCounter(NULL);
  16599. +#endif
  16600. +
  16601. +    ASSERT( NodeType(SrvOpen) == RDBSS_NTC_SRVOPEN );
  16602. +
  16603. +#ifdef DEBUG_OPEN
  16604. +    DbgEn();
  16605. +    print_debug_header(RxContext);
  16606. +    print_nt_create_params(1, RxContext->Create.NtCreateParameters);
  16607. +    // if (ea) print_ea_info(ea);
  16608. +#endif
  16609. +
  16610. +    status = check_nfs41_create_args(RxContext);
  16611. +    if (status) goto out;
  16612. +
  16613. +    status = nfs41_UpcallCreate(NFS41_OPEN, NULL,
  16614. +        pVNetRootContext->session, INVALID_HANDLE_VALUE,
  16615. +        pNetRootContext->nfs41d_version,
  16616. +        SrvOpen->pAlreadyPrefixedName, &entry);
  16617. +    if (status) goto out;
  16618. +
  16619. +    entry->u.Open.access_mask = params->DesiredAccess;
  16620. +    entry->u.Open.access_mode = params->ShareAccess;
  16621. +    entry->u.Open.attrs = params->FileAttributes;
  16622. +    if (!(params->CreateOptions & FILE_DIRECTORY_FILE))
  16623. +        entry->u.Open.attrs |= FILE_ATTRIBUTE_ARCHIVE;
  16624. +    entry->u.Open.disp = params->Disposition;
  16625. +    entry->u.Open.copts = params->CreateOptions;
  16626. +    entry->u.Open.srv_open = SrvOpen;
  16627. +    /* treat the NfsActOnLink ea as FILE_OPEN_REPARSE_POINT */
  16628. +    if ((ea && AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength)) ||
  16629. +            (entry->u.Open.access_mask & DELETE))
  16630. +        entry->u.Open.copts |= FILE_OPEN_REPARSE_POINT;
  16631. +    if (isDataAccess(params->DesiredAccess) || isOpen2Create(params->Disposition))
  16632. +        entry->u.Open.open_owner_id = InterlockedIncrement(&open_owner_id);
  16633. +    // if we are creating a file check if nfsv3attributes were passed in
  16634. +    if (params->Disposition != FILE_OPEN && params->Disposition != FILE_OVERWRITE) {
  16635. +        /* Get default mode */
  16636. +        entry->u.Open.mode = pVNetRootContext->createmode.mode;
  16637. +
  16638. +        /* Use mode from NfsV3Attributes */
  16639. +        if (pVNetRootContext->createmode.use_nfsv3attrsea_mode &&
  16640. +            ea && AnsiStrEq(&NfsV3Attributes,
  16641. +            ea->EaName, ea->EaNameLength)) {
  16642. +            nfs3_attrs *attrs =
  16643. +                (nfs3_attrs *)(ea->EaName + ea->EaNameLength + 1);
  16644. +
  16645. +            entry->u.Open.mode = attrs->mode;
  16646. +#ifdef DEBUG_OPEN
  16647. +            DbgP("creating file with EA mode 0%o\n",
  16648. +                entry->u.Open.mode);
  16649. +#endif
  16650. +        }
  16651. +        else {
  16652. +#ifdef DEBUG_OPEN
  16653. +            DbgP("creating file with default mode 0%o\n",
  16654. +                entry->u.Open.mode);
  16655. +#endif
  16656. +        }
  16657. +
  16658. +        if (params->FileAttributes & FILE_ATTRIBUTE_READONLY) {
  16659. +            entry->u.Open.mode &= ~0222;
  16660. +            DbgP("FILE_ATTRIBUTE_READONLY set, using mode 0%o\n",
  16661. +                entry->u.Open.mode);
  16662. +        }
  16663. +    }
  16664. +    if (entry->u.Open.disp == FILE_CREATE && ea &&
  16665. +            AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) {
  16666. +        /* for a cygwin symlink, given as a unicode string */
  16667. +        entry->u.Open.symlink.Buffer = (PWCH)(ea->EaName + ea->EaNameLength + 1);
  16668. +        entry->u.Open.symlink.MaximumLength = entry->u.Open.symlink.Length = ea->EaValueLength;
  16669. +    }
  16670. +retry_on_link:
  16671. +    if (ea && create_should_pass_ea(ea, params->Disposition)) {
  16672. +        /* lock the extended attribute buffer for read access in user space */
  16673. +        entry->u.Open.EaMdl = IoAllocateMdl(ea,
  16674. +            RxContext->CurrentIrpSp->Parameters.Create.EaLength,
  16675. +            FALSE, FALSE, NULL);
  16676. +        if (entry->u.Open.EaMdl == NULL) {
  16677. +            status = STATUS_INTERNAL_ERROR;
  16678. +            nfs41_UpcallDestroy(entry);
  16679. +            entry = NULL;
  16680. +            goto out;
  16681. +        }
  16682. +#pragma warning( push )
  16683. +/*
  16684. + * C28145: "The opaque MDL structure should not be modified by a
  16685. + * driver.", |MDL_MAPPING_CAN_FAIL| is the exception
  16686. + */
  16687. +#pragma warning (disable : 28145)
  16688. +        entry->u.Open.EaMdl->MdlFlags |= MDL_MAPPING_CAN_FAIL;
  16689. +#pragma warning( pop )
  16690. +        MmProbeAndLockPages(entry->u.Open.EaMdl, KernelMode, IoModifyAccess);
  16691. +    }
  16692. +
  16693. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  16694. +    if (entry->psec_ctx == &entry->sec_ctx) {
  16695. +        SeDeleteClientSecurity(entry->psec_ctx);
  16696. +    }
  16697. +    entry->psec_ctx = NULL;
  16698. +    if (status) goto out;
  16699. +
  16700. +    if (entry->u.Open.EaMdl) {
  16701. +        MmUnlockPages(entry->u.Open.EaMdl);
  16702. +        IoFreeMdl(entry->u.Open.EaMdl);
  16703. +    }
  16704. +
  16705. +    if (entry->status == NO_ERROR && entry->errno == ERROR_REPARSE) {
  16706. +        /* symbolic link handling. when attempting to open a symlink when the
  16707. +         * FILE_OPEN_REPARSE_POINT flag is not set, replace the filename with
  16708. +         * the symlink target's by calling RxPrepareToReparseSymbolicLink()
  16709. +         * and returning STATUS_REPARSE. the object manager will attempt to
  16710. +         * open the new path, and return its handle for the original open */
  16711. +        PRDBSS_DEVICE_OBJECT DeviceObject = RxContext->RxDeviceObject;
  16712. +        PV_NET_ROOT VNetRoot = (PV_NET_ROOT)
  16713. +            RxContext->pRelevantSrvOpen->pVNetRoot;
  16714. +        PUNICODE_STRING VNetRootPrefix = &VNetRoot->PrefixEntry.Prefix;
  16715. +        UNICODE_STRING AbsPath;
  16716. +        PCHAR buf;
  16717. +        BOOLEAN ReparseRequired;
  16718. +
  16719. +        /* allocate the string for RxPrepareToReparseSymbolicLink(), and
  16720. +         * format an absolute path "DeviceName+VNetRootName+symlink" */
  16721. +        AbsPath.Length = DeviceObject->DeviceName.Length +
  16722. +            VNetRootPrefix->Length + entry->u.Open.symlink.Length;
  16723. +        AbsPath.MaximumLength = AbsPath.Length + sizeof(UNICODE_NULL);
  16724. +        AbsPath.Buffer = RxAllocatePoolWithTag(NonPagedPoolNx,
  16725. +            AbsPath.MaximumLength, NFS41_MM_POOLTAG);
  16726. +        if (AbsPath.Buffer == NULL) {
  16727. +            status = STATUS_INSUFFICIENT_RESOURCES;
  16728. +            goto out_free;
  16729. +        }
  16730. +
  16731. +        buf = (PCHAR)AbsPath.Buffer;
  16732. +        RtlCopyMemory(buf, DeviceObject->DeviceName.Buffer,
  16733. +            DeviceObject->DeviceName.Length);
  16734. +        buf += DeviceObject->DeviceName.Length;
  16735. +        RtlCopyMemory(buf, VNetRootPrefix->Buffer, VNetRootPrefix->Length);
  16736. +        buf += VNetRootPrefix->Length;
  16737. +        RtlCopyMemory(buf, entry->u.Open.symlink.Buffer,
  16738. +            entry->u.Open.symlink.Length);
  16739. +        RxFreePool(entry->u.Open.symlink.Buffer);
  16740. +        entry->u.Open.symlink.Buffer = NULL;
  16741. +        buf += entry->u.Open.symlink.Length;
  16742. +        *(PWCHAR)buf = UNICODE_NULL;
  16743. +
  16744. +        status = RxPrepareToReparseSymbolicLink(RxContext,
  16745. +            entry->u.Open.symlink_embedded, &AbsPath, TRUE, &ReparseRequired);
  16746. +#ifdef DEBUG_OPEN
  16747. +        DbgP("RxPrepareToReparseSymbolicLink(%u, '%wZ') returned 0x%08lX, "
  16748. +            "FileName is '%wZ'\n", entry->u.Open.symlink_embedded,
  16749. +            &AbsPath, status, &RxContext->CurrentIrpSp->FileObject->FileName);
  16750. +#endif
  16751. +        if (status == STATUS_SUCCESS) {
  16752. +            /* if a reparse is not required, reopen the link itself.  this
  16753. +             * happens with operations on cygwin symlinks, where the reparse
  16754. +             * flag is not set */
  16755. +            if (!ReparseRequired) {
  16756. +                entry->u.Open.symlink.Length = 0;
  16757. +                entry->u.Open.copts |= FILE_OPEN_REPARSE_POINT;
  16758. +                goto retry_on_link;
  16759. +            }
  16760. +            status = STATUS_REPARSE;
  16761. +        }
  16762. +        goto out_free;
  16763. +    }
  16764. +
  16765. +    status = map_open_errors(entry->status,
  16766. +                SrvOpen->pAlreadyPrefixedName->Length);
  16767. +    if (status) {
  16768. +#ifdef DEBUG_OPEN
  16769. +        print_open_error(1, status);
  16770. +#endif
  16771. +        goto out_free;
  16772. +    }
  16773. +
  16774. +    if (!RxIsFcbAcquiredExclusive(Fcb)) {
  16775. +        ASSERT(!RxIsFcbAcquiredShared(Fcb));
  16776. +        RxAcquireExclusiveFcbResourceInMRx(Fcb);
  16777. +    }
  16778. +
  16779. +    RxContext->pFobx = RxCreateNetFobx(RxContext, SrvOpen);
  16780. +    if (RxContext->pFobx == NULL) {
  16781. +        status = STATUS_INSUFFICIENT_RESOURCES;
  16782. +        goto out_free;
  16783. +    }
  16784. +#ifdef DEBUG_OPEN
  16785. +    DbgP("nfs41_Create: created FOBX 0x%p\n", RxContext->pFobx);
  16786. +#endif
  16787. +    nfs41_fobx = (PNFS41_FOBX)(RxContext->pFobx)->Context;
  16788. +    nfs41_fobx->nfs41_open_state = entry->open_state;
  16789. +    if (nfs41_fobx->sec_ctx.ClientToken == NULL) {
  16790. +        status = nfs41_get_sec_ctx(SecurityImpersonation, &nfs41_fobx->sec_ctx);
  16791. +        if (status)
  16792. +            goto out_free;
  16793. +    }
  16794. +
  16795. +    // we get attributes only for data access and file (not directories)
  16796. +    if (Fcb->OpenCount == 0 ||
  16797. +            (Fcb->OpenCount > 0 &&
  16798. +                nfs41_fcb->changeattr != entry->ChangeTime)) {
  16799. +        FCB_INIT_PACKET InitPacket;
  16800. +        RX_FILE_TYPE StorageType = FileTypeNotYetKnown;
  16801. +        RtlCopyMemory(&nfs41_fcb->BasicInfo, &entry->u.Open.binfo,
  16802. +            sizeof(entry->u.Open.binfo));
  16803. +        RtlCopyMemory(&nfs41_fcb->StandardInfo, &entry->u.Open.sinfo,
  16804. +            sizeof(entry->u.Open.sinfo));
  16805. +        nfs41_fcb->mode = entry->u.Open.mode;
  16806. +#ifdef NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES
  16807. +        nfs41_fcb->owner_local_uid = entry->u.Open.owner_local_uid;
  16808. +        nfs41_fcb->owner_group_local_gid = entry->u.Open.owner_group_local_gid;
  16809. +#endif /* NFS41_DRIVER_FEATURE_LOCAL_UIDGID_IN_NFSV3ATTRIBUTES */
  16810. +        nfs41_fcb->changeattr = entry->ChangeTime;
  16811. +        if (((params->CreateOptions & FILE_DELETE_ON_CLOSE) &&
  16812. +                !pVNetRootContext->read_only) || oldDeletePending)
  16813. +            nfs41_fcb->StandardInfo.DeletePending = TRUE;
  16814. +
  16815. +        RxFormInitPacket(InitPacket,
  16816. +            &entry->u.Open.binfo.FileAttributes,
  16817. +            &entry->u.Open.sinfo.NumberOfLinks,
  16818. +            &entry->u.Open.binfo.CreationTime,
  16819. +            &entry->u.Open.binfo.LastAccessTime,
  16820. +            &entry->u.Open.binfo.LastWriteTime,
  16821. +            &entry->u.Open.binfo.ChangeTime,
  16822. +            &entry->u.Open.sinfo.AllocationSize,
  16823. +            &entry->u.Open.sinfo.EndOfFile,
  16824. +            &entry->u.Open.sinfo.EndOfFile);
  16825. +
  16826. +        if (entry->u.Open.sinfo.Directory)
  16827. +            StorageType = FileTypeDirectory;
  16828. +        else
  16829. +            StorageType = FileTypeFile;
  16830. +
  16831. +        RxFinishFcbInitialization(Fcb, RDBSS_STORAGE_NTC(StorageType),
  16832. +                                    &InitPacket);
  16833. +    }
  16834. +#ifdef DEBUG_OPEN
  16835. +    else
  16836. +        DbgP("$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$\n");
  16837. +
  16838. +    print_basic_info(1, &nfs41_fcb->BasicInfo);
  16839. +    print_std_info(1, &nfs41_fcb->StandardInfo);
  16840. +#endif
  16841. +
  16842. +    /* aglo: 05/10/2012. it seems like always have to invalid the cache if the
  16843. +     * file has been opened before and being opened again for data access.
  16844. +     * If the file was opened before, RDBSS might have cached (unflushed) data
  16845. +     * and by opening it again, we will not have the correct representation of
  16846. +     * the file size and data content. fileio tests 208, 219, 221.
  16847. +     */
  16848. +    if (Fcb->OpenCount > 0 && (isDataAccess(params->DesiredAccess) ||
  16849. +            nfs41_fcb->changeattr != entry->ChangeTime) &&
  16850. +                !nfs41_fcb->StandardInfo.Directory) {
  16851. +        ULONG flag = DISABLE_CACHING;
  16852. +#ifdef DEBUG_OPEN
  16853. +        DbgP("nfs41_Create: reopening (changed) file '%wZ'\n",
  16854. +            SrvOpen->pAlreadyPrefixedName);
  16855. +#endif
  16856. +        RxChangeBufferingState((PSRV_OPEN)SrvOpen, ULongToPtr(flag), 1);
  16857. +    }
  16858. +    if (!nfs41_fcb->StandardInfo.Directory &&
  16859. +            isDataAccess(params->DesiredAccess)) {
  16860. +        nfs41_fobx->deleg_type = entry->u.Open.deleg_type;
  16861. +#ifdef DEBUG_OPEN
  16862. +        DbgP("nfs41_Create: received delegation %d\n", entry->u.Open.deleg_type);
  16863. +#endif
  16864. +        if (!(params->CreateOptions & FILE_WRITE_THROUGH) &&
  16865. +                !pVNetRootContext->write_thru &&
  16866. +                (entry->u.Open.deleg_type == 2 ||
  16867. +                (params->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA)))) {
  16868. +#ifdef DEBUG_OPEN
  16869. +            DbgP("nfs41_Create: enabling write buffering\n");
  16870. +#endif
  16871. +            SrvOpen->BufferingFlags |=
  16872. +                (FCB_STATE_WRITECACHING_ENABLED |
  16873. +                FCB_STATE_WRITEBUFFERING_ENABLED);
  16874. +        } else if (params->CreateOptions & FILE_WRITE_THROUGH ||
  16875. +                    pVNetRootContext->write_thru)
  16876. +            nfs41_fobx->write_thru = TRUE;
  16877. +        if (entry->u.Open.deleg_type >= 1 ||
  16878. +                params->DesiredAccess & FILE_READ_DATA) {
  16879. +#ifdef DEBUG_OPEN
  16880. +            DbgP("nfs41_Create: enabling read buffering\n");
  16881. +#endif
  16882. +            SrvOpen->BufferingFlags |=
  16883. +                (FCB_STATE_READBUFFERING_ENABLED |
  16884. +                FCB_STATE_READCACHING_ENABLED);
  16885. +        }
  16886. +        nfs41_fobx->timebasedcoherency = pVNetRootContext->timebasedcoherency;
  16887. +        if (pVNetRootContext->nocache ||
  16888. +                (params->CreateOptions & FILE_NO_INTERMEDIATE_BUFFERING)) {
  16889. +#ifdef DEBUG_OPEN
  16890. +            DbgP("nfs41_Create: disabling buffering\n");
  16891. +#endif
  16892. +            SrvOpen->BufferingFlags = FCB_STATE_DISABLE_LOCAL_BUFFERING;
  16893. +            nfs41_fobx->nocache = TRUE;
  16894. +        } else if (!entry->u.Open.deleg_type && !Fcb->OpenCount) {
  16895. +            nfs41_fcb_list_entry *oentry;
  16896. +#ifdef DEBUG_OPEN
  16897. +            DbgP("nfs41_Create: received no delegations: srv_open=0x%p "
  16898. +                "ctime=%llu\n", SrvOpen, entry->ChangeTime);
  16899. +#endif
  16900. +            oentry = RxAllocatePoolWithTag(NonPagedPoolNx,
  16901. +                sizeof(nfs41_fcb_list_entry), NFS41_MM_POOLTAG_OPEN);
  16902. +            if (oentry == NULL) {
  16903. +                status = STATUS_INSUFFICIENT_RESOURCES;
  16904. +                goto out_free;
  16905. +            }
  16906. +            oentry->fcb = RxContext->pFcb;
  16907. +            oentry->nfs41_fobx = nfs41_fobx;
  16908. +            oentry->session = pVNetRootContext->session;
  16909. +            oentry->ChangeTime = entry->ChangeTime;
  16910. +            oentry->skip = FALSE;
  16911. +            nfs41_AddEntry(fcblistLock, openlist, oentry);
  16912. +        }
  16913. +    }
  16914. +
  16915. +    if ((params->CreateOptions & FILE_DELETE_ON_CLOSE) &&
  16916. +            !pVNetRootContext->read_only)
  16917. +        nfs41_fcb->StandardInfo.DeletePending = TRUE;
  16918. +
  16919. +    RxContext->Create.ReturnedCreateInformation =
  16920. +        map_disposition_to_create_retval(params->Disposition, entry->errno);
  16921. +
  16922. +    RxContext->pFobx->OffsetOfNextEaToReturn = 1;
  16923. +    RxContext->CurrentIrp->IoStatus.Information =
  16924. +        RxContext->Create.ReturnedCreateInformation;
  16925. +    status = RxContext->CurrentIrp->IoStatus.Status = STATUS_SUCCESS;
  16926. +
  16927. +out_free:
  16928. +    if (entry)
  16929. +        nfs41_UpcallDestroy(entry);
  16930. +out:
  16931. +#ifdef ENABLE_TIMINGS
  16932. +    t2 = KeQueryPerformanceCounter(NULL);
  16933. +    if ((params->DesiredAccess & FILE_READ_DATA) ||
  16934. +            (params->DesiredAccess & FILE_WRITE_DATA) ||
  16935. +            (params->DesiredAccess & FILE_APPEND_DATA) ||
  16936. +            (params->DesiredAccess & FILE_EXECUTE)) {
  16937. +        InterlockedIncrement(&open.tops);
  16938. +        InterlockedAdd64(&open.ticks, t2.QuadPart - t1.QuadPart);
  16939. +#ifdef ENABLE_INDV_TIMINGS
  16940. +    DbgP("nfs41_Create open delta = %d op=%d sum=%d\n",
  16941. +        t2.QuadPart - t1.QuadPart, open.tops, open.ticks);
  16942. +#endif
  16943. +    } else {
  16944. +        InterlockedIncrement(&lookup.tops);
  16945. +        InterlockedAdd64(&lookup.ticks, t2.QuadPart - t1.QuadPart);
  16946. +#ifdef ENABLE_INDV_TIMINGS
  16947. +    DbgP("nfs41_Create lookup delta = %d op=%d sum=%d\n",
  16948. +        t2.QuadPart - t1.QuadPart, lookup.tops, lookup.ticks);
  16949. +#endif
  16950. +    }
  16951. +#endif
  16952. +#ifdef DEBUG_OPEN
  16953. +    DbgEx();
  16954. +#endif
  16955. +    return status;
  16956. +}
  16957. +
  16958. +NTSTATUS nfs41_CollapseOpen(
  16959. +    IN OUT PRX_CONTEXT RxContext)
  16960. +{
  16961. +    NTSTATUS status = STATUS_MORE_PROCESSING_REQUIRED;
  16962. +    DbgEn();
  16963. +    DbgEx();
  16964. +    return status;
  16965. +}
  16966. +
  16967. +NTSTATUS nfs41_ShouldTryToCollapseThisOpen(
  16968. +    IN OUT PRX_CONTEXT RxContext)
  16969. +{
  16970. +    if (RxContext->pRelevantSrvOpen == NULL)
  16971. +        return STATUS_SUCCESS;
  16972. +    else return STATUS_MORE_PROCESSING_REQUIRED;
  16973. +}
  16974. +
  16975. +NTSTATUS map_close_errors(
  16976. +    DWORD status)
  16977. +{
  16978. +    switch (status) {
  16979. +    case NO_ERROR:              return STATUS_SUCCESS;
  16980. +    case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED;
  16981. +    case ERROR_NOT_EMPTY:       return STATUS_DIRECTORY_NOT_EMPTY;
  16982. +    case ERROR_FILE_INVALID:    return STATUS_FILE_INVALID;
  16983. +    case ERROR_DISK_FULL:       return STATUS_DISK_FULL;
  16984. +    case ERROR_DISK_QUOTA_EXCEEDED: return STATUS_DISK_QUOTA_EXCEEDED;
  16985. +    case ERROR_FILE_TOO_LARGE:  return STATUS_FILE_TOO_LARGE;
  16986. +    default:
  16987. +        print_error("map_close_errors: "
  16988. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  16989. +            "defaulting to STATUS_INTERNAL_ERROR\n", status);
  16990. +    case ERROR_INTERNAL_ERROR:  return STATUS_INTERNAL_ERROR;
  16991. +    }
  16992. +}
  16993. +
  16994. +NTSTATUS nfs41_CloseSrvOpen(
  16995. +    IN OUT PRX_CONTEXT RxContext)
  16996. +{
  16997. +    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  16998. +    nfs41_updowncall_entry *entry;
  16999. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  17000. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  17001. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  17002. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  17003. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  17004. +    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  17005. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  17006. +#ifdef ENABLE_TIMINGS
  17007. +    LARGE_INTEGER t1, t2;
  17008. +    t1 = KeQueryPerformanceCounter(NULL);
  17009. +#endif
  17010. +
  17011. +#ifdef DEBUG_CLOSE
  17012. +    DbgEn();
  17013. +    print_debug_header(RxContext);
  17014. +#endif
  17015. +
  17016. +    if (!nfs41_fobx->deleg_type && !nfs41_fcb->StandardInfo.Directory &&
  17017. +            !RxContext->pFcb->OpenCount) {
  17018. +        nfs41_remove_fcb_entry(RxContext->pFcb);
  17019. +    }
  17020. +
  17021. +    status = nfs41_UpcallCreate(NFS41_CLOSE, &nfs41_fobx->sec_ctx,
  17022. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  17023. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  17024. +    if (status) goto out;
  17025. +
  17026. +    entry->u.Close.srv_open = SrvOpen;
  17027. +    if (nfs41_fcb->StandardInfo.DeletePending)
  17028. +        nfs41_fcb->DeletePending = TRUE;
  17029. +    if (!RxContext->pFcb->OpenCount ||
  17030. +            (nfs41_fcb->StandardInfo.DeletePending &&
  17031. +                nfs41_fcb->StandardInfo.Directory))
  17032. +        entry->u.Close.remove = nfs41_fcb->StandardInfo.DeletePending;
  17033. +    if (!RxContext->pFcb->OpenCount)
  17034. +        entry->u.Close.renamed = nfs41_fcb->Renamed;
  17035. +
  17036. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  17037. +    if (status) goto out;
  17038. +
  17039. +    /* map windows ERRORs to NTSTATUS */
  17040. +    status = map_close_errors(entry->status);
  17041. +    nfs41_UpcallDestroy(entry);
  17042. +out:
  17043. +#ifdef ENABLE_TIMINGS
  17044. +    t2 = KeQueryPerformanceCounter(NULL);
  17045. +    InterlockedIncrement(&close.tops);
  17046. +    InterlockedAdd64(&close.ticks, t2.QuadPart - t1.QuadPart);
  17047. +#ifdef ENABLE_INDV_TIMINGS
  17048. +    DbgP("nfs41_CloseSrvOpen delta = %d op=%d sum=%d\n",
  17049. +        t2.QuadPart - t1.QuadPart, close.tops, close.ticks);
  17050. +#endif
  17051. +#endif
  17052. +#ifdef DEBUG_CLOSE
  17053. +    DbgEx();
  17054. +#endif
  17055. +    return status;
  17056. +}
  17057. diff --git a/sys/nfs41sys_readwrite.c b/sys/nfs41sys_readwrite.c
  17058. new file mode 100644
  17059. index 0000000..d567824
  17060. --- /dev/null
  17061. +++ b/sys/nfs41sys_readwrite.c
  17062. @@ -0,0 +1,435 @@
  17063. +/* NFSv4.1 client for Windows
  17064. + * Copyright (C) 2012 The Regents of the University of Michigan
  17065. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  17066. + *
  17067. + * Olga Kornievskaia <aglo@umich.edu>
  17068. + * Casey Bodley <cbodley@umich.edu>
  17069. + * Roland Mainz <roland.mainz@nrubsig.org>
  17070. + *
  17071. + * This library is free software; you can redistribute it and/or modify it
  17072. + * under the terms of the GNU Lesser General Public License as published by
  17073. + * the Free Software Foundation; either version 2.1 of the License, or (at
  17074. + * your option) any later version.
  17075. + *
  17076. + * This library is distributed in the hope that it will be useful, but
  17077. + * without any warranty; without even the implied warranty of merchantability
  17078. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  17079. + * License for more details.
  17080. + *
  17081. + * You should have received a copy of the GNU Lesser General Public License
  17082. + * along with this library; if not, write to the Free Software Foundation,
  17083. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  17084. + */
  17085. +
  17086. +#ifndef _KERNEL_MODE
  17087. +#error module requires kernel mode
  17088. +#endif
  17089. +
  17090. +#if ((__STDC_VERSION__-0) < 201710L)
  17091. +#error Code requires ISO C17
  17092. +#endif
  17093. +
  17094. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  17095. +#if _MSC_VER >= 1900
  17096. +#if defined(_WIN64) && defined(_M_X64)
  17097. +#ifndef _AMD64_
  17098. +#define _AMD64_
  17099. +#endif
  17100. +#elif defined(_WIN32) && defined(_M_IX86)
  17101. +#ifndef _X86_
  17102. +#define _X86_
  17103. +#endif
  17104. +#elif defined(_WIN64) && defined(_M_ARM64)
  17105. +#ifndef _ARM64_
  17106. +#define _ARM64_
  17107. +#endif
  17108. +#elif defined(_WIN32) && defined(_M_ARM)
  17109. +#ifndef _ARM_
  17110. +#define _ARM_
  17111. +#endif
  17112. +#else
  17113. +#error Unsupported arch
  17114. +#endif
  17115. +#endif /* _MSC_VER >= 1900 */
  17116. +
  17117. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  17118. +#include <rx.h>
  17119. +#include <windef.h>
  17120. +#include <winerror.h>
  17121. +
  17122. +#include <Ntstrsafe.h>
  17123. +
  17124. +#include "nfs41sys_buildconfig.h"
  17125. +
  17126. +#include "nfs41_driver.h"
  17127. +#include "nfs41_np.h"
  17128. +#include "nfs41sys_debug.h"
  17129. +#include "nfs41_build_features.h"
  17130. +#include "nfs_ea.h"
  17131. +
  17132. +#include "nfs41sys_driver.h"
  17133. +#include "nfs41sys_util.h"
  17134. +
  17135. +
  17136. +static void print_readwrite_args(
  17137. +    PRX_CONTEXT RxContext)
  17138. +{
  17139. +    PLOWIO_CONTEXT LowIoContext  = &RxContext->LowIoContext;
  17140. +
  17141. +    print_debug_header(RxContext);
  17142. +    DbgP("Bytecount 0x%x Byteoffset 0x%x Buffer 0x%p\n",
  17143. +        LowIoContext->ParamsFor.ReadWrite.ByteCount,
  17144. +        LowIoContext->ParamsFor.ReadWrite.ByteOffset,
  17145. +        LowIoContext->ParamsFor.ReadWrite.Buffer);
  17146. +}
  17147. +
  17148. +NTSTATUS marshal_nfs41_rw(
  17149. +    nfs41_updowncall_entry *entry,
  17150. +    unsigned char *buf,
  17151. +    ULONG buf_len,
  17152. +    ULONG *len)
  17153. +{
  17154. +    NTSTATUS status = STATUS_SUCCESS;
  17155. +    ULONG header_len = 0;
  17156. +    unsigned char *tmp = buf;
  17157. +
  17158. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  17159. +    if (status) goto out;
  17160. +    else tmp += *len;
  17161. +
  17162. +    header_len = *len + sizeof(entry->buf_len) +
  17163. +        sizeof(entry->u.ReadWrite.offset) + sizeof(HANDLE);
  17164. +    if (header_len > buf_len) {
  17165. +        status = STATUS_INSUFFICIENT_RESOURCES;
  17166. +        goto out;
  17167. +    }
  17168. +
  17169. +    RtlCopyMemory(tmp, &entry->buf_len, sizeof(entry->buf_len));
  17170. +    tmp += sizeof(entry->buf_len);
  17171. +    RtlCopyMemory(tmp, &entry->u.ReadWrite.offset,
  17172. +        sizeof(entry->u.ReadWrite.offset));
  17173. +    tmp += sizeof(entry->u.ReadWrite.offset);
  17174. +    __try {
  17175. +#pragma warning( push )
  17176. +/*
  17177. + * C28145: "The opaque MDL structure should not be modified by a
  17178. + * driver.", |MDL_MAPPING_CAN_FAIL| is the exception
  17179. + */
  17180. +#pragma warning (disable : 28145)
  17181. +        entry->u.ReadWrite.MdlAddress->MdlFlags |= MDL_MAPPING_CAN_FAIL;
  17182. +#pragma warning( pop )
  17183. +        entry->buf =
  17184. +            MmMapLockedPagesSpecifyCache(entry->u.ReadWrite.MdlAddress,
  17185. +                UserMode, MmCached, NULL, TRUE, NormalPagePriority);
  17186. +        if (entry->buf == NULL) {
  17187. +            print_error("marshal_nfs41_rw: "
  17188. +                "MmMapLockedPagesSpecifyCache() failed to map pages\n");
  17189. +            status = STATUS_INSUFFICIENT_RESOURCES;
  17190. +            goto out;
  17191. +        }
  17192. +    } __except(EXCEPTION_EXECUTE_HANDLER) {
  17193. +        NTSTATUS code;
  17194. +        code = GetExceptionCode();
  17195. +        print_error("marshal_nfs41_rw: Call to "
  17196. +            "MmMapLockedPagesSpecifyCache() failed due to "
  17197. +            "exception 0x%x\n", (int)code);
  17198. +        status = STATUS_ACCESS_VIOLATION;
  17199. +        goto out;
  17200. +    }
  17201. +    RtlCopyMemory(tmp, &entry->buf, sizeof(HANDLE));
  17202. +    *len = header_len;
  17203. +
  17204. +#ifdef DEBUG_MARSHAL_DETAIL_RW
  17205. +    DbgP("marshal_nfs41_rw: len=%lu offset=%llu "
  17206. +        "MdlAddress=0x%p Userspace=0x%p\n",
  17207. +        entry->buf_len, entry->u.ReadWrite.offset,
  17208. +        entry->u.ReadWrite.MdlAddress, entry->buf);
  17209. +#endif
  17210. +out:
  17211. +    return status;
  17212. +}
  17213. +
  17214. +NTSTATUS unmarshal_nfs41_rw(
  17215. +    nfs41_updowncall_entry *cur,
  17216. +    unsigned char **buf)
  17217. +{
  17218. +    NTSTATUS status = STATUS_SUCCESS;
  17219. +
  17220. +    RtlCopyMemory(&cur->buf_len, *buf, sizeof(cur->buf_len));
  17221. +    *buf += sizeof(cur->buf_len);
  17222. +    RtlCopyMemory(&cur->ChangeTime, *buf, sizeof(ULONGLONG));
  17223. +#ifdef DEBUG_MARSHAL_DETAIL_RW
  17224. +    DbgP("unmarshal_nfs41_rw: returned len %lu ChangeTime %llu\n",
  17225. +        cur->buf_len, cur->ChangeTime);
  17226. +#endif
  17227. +#if 1
  17228. +    /* 08/27/2010: it looks like we really don't need to call
  17229. +        * MmUnmapLockedPages() eventhough we called
  17230. +        * MmMapLockedPagesSpecifyCache() as the MDL passed to us
  17231. +        * is already locked.
  17232. +        */
  17233. +    __try {
  17234. +        MmUnmapLockedPages(cur->buf, cur->u.ReadWrite.MdlAddress);
  17235. +    } __except(EXCEPTION_EXECUTE_HANDLER) {
  17236. +        NTSTATUS code;
  17237. +        code = GetExceptionCode();
  17238. +        print_error("unmarshal_nfs41_rw: Call to MmUnmapLockedPages() "
  17239. +            "failed due to exception 0x%0x\n", (int)code);
  17240. +        status = STATUS_ACCESS_VIOLATION;
  17241. +    }
  17242. +#endif
  17243. +    return status;
  17244. +}
  17245. +
  17246. +NTSTATUS map_readwrite_errors(
  17247. +    DWORD status)
  17248. +{
  17249. +    switch (status) {
  17250. +    case ERROR_ACCESS_DENIED:           return STATUS_ACCESS_DENIED;
  17251. +    case ERROR_HANDLE_EOF:              return STATUS_END_OF_FILE;
  17252. +    case ERROR_FILE_INVALID:            return STATUS_FILE_INVALID;
  17253. +    case ERROR_INVALID_PARAMETER:       return STATUS_INVALID_PARAMETER;
  17254. +    case ERROR_LOCK_VIOLATION:          return STATUS_FILE_LOCK_CONFLICT;
  17255. +    case ERROR_NETWORK_ACCESS_DENIED:   return STATUS_NETWORK_ACCESS_DENIED;
  17256. +    case ERROR_NETNAME_DELETED:         return STATUS_NETWORK_NAME_DELETED;
  17257. +    case ERROR_DISK_FULL:               return STATUS_DISK_FULL;
  17258. +    case ERROR_DISK_QUOTA_EXCEEDED:     return STATUS_DISK_QUOTA_EXCEEDED;
  17259. +    case ERROR_FILE_TOO_LARGE:          return STATUS_FILE_TOO_LARGE;
  17260. +    case ERROR_INTERNAL_ERROR:          return STATUS_INTERNAL_ERROR;
  17261. +    default:
  17262. +        print_error("map_readwrite_errors: "
  17263. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  17264. +            "defaulting to STATUS_NET_WRITE_FAULT\n", status);
  17265. +    case ERROR_NET_WRITE_FAULT:         return STATUS_NET_WRITE_FAULT;
  17266. +    }
  17267. +}
  17268. +
  17269. +static NTSTATUS check_nfs41_read_args(
  17270. +    IN PRX_CONTEXT RxContext)
  17271. +{
  17272. +    if (!RxContext->LowIoContext.ParamsFor.ReadWrite.Buffer)
  17273. +        return STATUS_INVALID_USER_BUFFER;
  17274. +    return STATUS_SUCCESS;
  17275. +}
  17276. +
  17277. +NTSTATUS nfs41_Read(
  17278. +    IN OUT PRX_CONTEXT RxContext)
  17279. +{
  17280. +    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  17281. +    nfs41_updowncall_entry *entry;
  17282. +    BOOLEAN async = FALSE;
  17283. +    PLOWIO_CONTEXT LowIoContext  = &RxContext->LowIoContext;
  17284. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  17285. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  17286. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  17287. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  17288. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  17289. +    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  17290. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  17291. +    DWORD io_delay;
  17292. +#ifdef ENABLE_TIMINGS
  17293. +    LARGE_INTEGER t1, t2;
  17294. +    t1 = KeQueryPerformanceCounter(NULL);
  17295. +#endif
  17296. +
  17297. +#ifdef DEBUG_READ
  17298. +    DbgEn();
  17299. +    print_readwrite_args(RxContext);
  17300. +#endif
  17301. +    status = check_nfs41_read_args(RxContext);
  17302. +    if (status) goto out;
  17303. +
  17304. +    status = nfs41_UpcallCreate(NFS41_READ, &nfs41_fobx->sec_ctx,
  17305. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  17306. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  17307. +    if (status) goto out;
  17308. +
  17309. +    entry->u.ReadWrite.MdlAddress = LowIoContext->ParamsFor.ReadWrite.Buffer;
  17310. +    entry->buf_len = LowIoContext->ParamsFor.ReadWrite.ByteCount;
  17311. +    entry->u.ReadWrite.offset = LowIoContext->ParamsFor.ReadWrite.ByteOffset;
  17312. +    if (FlagOn(RxContext->CurrentIrpSp->FileObject->Flags,
  17313. +            FO_SYNCHRONOUS_IO) == FALSE) {
  17314. +        entry->u.ReadWrite.rxcontext = RxContext;
  17315. +        async = entry->async_op = TRUE;
  17316. +    }
  17317. +
  17318. +    /* Add extra timeout depending on buffer size */
  17319. +    io_delay = pVNetRootContext->timeout +
  17320. +        EXTRA_TIMEOUT_PER_BYTE(entry->buf_len);
  17321. +    status = nfs41_UpcallWaitForReply(entry, io_delay);
  17322. +    if (status) goto out;
  17323. +
  17324. +    if (async) {
  17325. +#ifdef DEBUG_READ
  17326. +        DbgP("This is asynchronous read, returning control back to the user\n");
  17327. +#endif
  17328. +        status = STATUS_PENDING;
  17329. +        goto out;
  17330. +    }
  17331. +
  17332. +    if (entry->status == NO_ERROR) {
  17333. +#ifdef ENABLE_TIMINGS
  17334. +        InterlockedIncrement(&read.sops);
  17335. +        InterlockedAdd64(&read.size, entry->u.ReadWrite.len);
  17336. +#endif
  17337. +        status = RxContext->CurrentIrp->IoStatus.Status = STATUS_SUCCESS;
  17338. +        RxContext->IoStatusBlock.Information = entry->buf_len;
  17339. +
  17340. +        if ((!BooleanFlagOn(LowIoContext->ParamsFor.ReadWrite.Flags,
  17341. +                LOWIO_READWRITEFLAG_PAGING_IO) &&
  17342. +                (SrvOpen->DesiredAccess & FILE_READ_DATA) &&
  17343. +                !pVNetRootContext->nocache && !nfs41_fobx->nocache &&
  17344. +                !(SrvOpen->BufferingFlags &
  17345. +                (FCB_STATE_READBUFFERING_ENABLED |
  17346. +                 FCB_STATE_READCACHING_ENABLED)))) {
  17347. +            enable_caching(SrvOpen, nfs41_fobx, nfs41_fcb->changeattr,
  17348. +                pVNetRootContext->session);
  17349. +        }
  17350. +    } else {
  17351. +        status = map_readwrite_errors(entry->status);
  17352. +        RxContext->CurrentIrp->IoStatus.Status = status;
  17353. +        RxContext->IoStatusBlock.Information = 0;
  17354. +    }
  17355. +    nfs41_UpcallDestroy(entry);
  17356. +out:
  17357. +#ifdef ENABLE_TIMINGS
  17358. +    t2 = KeQueryPerformanceCounter(NULL);
  17359. +    InterlockedIncrement(&read.tops);
  17360. +    InterlockedAdd64(&read.ticks, t2.QuadPart - t1.QuadPart);
  17361. +#ifdef ENABLE_INDV_TIMINGS
  17362. +    DbgP("nfs41_Read delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart,
  17363. +        read.tops, read.ticks);
  17364. +#endif
  17365. +#endif
  17366. +#ifdef DEBUG_READ
  17367. +    DbgEx();
  17368. +#endif
  17369. +    return status;
  17370. +}
  17371. +
  17372. +static NTSTATUS check_nfs41_write_args(
  17373. +    IN PRX_CONTEXT RxContext)
  17374. +{
  17375. +    NTSTATUS status = STATUS_SUCCESS;
  17376. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  17377. +        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  17378. +
  17379. +    if (!RxContext->LowIoContext.ParamsFor.ReadWrite.Buffer) {
  17380. +        status = STATUS_INVALID_USER_BUFFER;
  17381. +        goto out;
  17382. +    }
  17383. +
  17384. +    if (pVNetRootContext->read_only) {
  17385. +        print_error("check_nfs41_write_args: Read-only mount\n");
  17386. +        status = STATUS_MEDIA_WRITE_PROTECTED;
  17387. +        goto out;
  17388. +    }
  17389. +out:
  17390. +    return status;
  17391. +}
  17392. +
  17393. +NTSTATUS nfs41_Write(
  17394. +    IN OUT PRX_CONTEXT RxContext)
  17395. +{
  17396. +    NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES;
  17397. +    nfs41_updowncall_entry *entry;
  17398. +    BOOLEAN async = FALSE;
  17399. +    PLOWIO_CONTEXT LowIoContext  = &RxContext->LowIoContext;
  17400. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  17401. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  17402. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  17403. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  17404. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  17405. +    __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb);
  17406. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  17407. +    DWORD io_delay;
  17408. +#ifdef ENABLE_TIMINGS
  17409. +    LARGE_INTEGER t1, t2;
  17410. +    t1 = KeQueryPerformanceCounter(NULL);
  17411. +#endif
  17412. +
  17413. +#ifdef DEBUG_WRITE
  17414. +    DbgEn();
  17415. +    print_readwrite_args(RxContext);
  17416. +#endif
  17417. +
  17418. +    status = check_nfs41_write_args(RxContext);
  17419. +    if (status) goto out;
  17420. +
  17421. +    status = nfs41_UpcallCreate(NFS41_WRITE, &nfs41_fobx->sec_ctx,
  17422. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  17423. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  17424. +    if (status) goto out;
  17425. +
  17426. +    entry->u.ReadWrite.MdlAddress = LowIoContext->ParamsFor.ReadWrite.Buffer;
  17427. +    entry->buf_len = LowIoContext->ParamsFor.ReadWrite.ByteCount;
  17428. +    entry->u.ReadWrite.offset = LowIoContext->ParamsFor.ReadWrite.ByteOffset;
  17429. +
  17430. +    if (FlagOn(RxContext->CurrentIrpSp->FileObject->Flags,
  17431. +            FO_SYNCHRONOUS_IO) == FALSE) {
  17432. +        entry->u.ReadWrite.rxcontext = RxContext;
  17433. +        async = entry->async_op = TRUE;
  17434. +    }
  17435. +
  17436. +    /* Add extra timeout depending on buffer size */
  17437. +    io_delay = pVNetRootContext->timeout +
  17438. +        EXTRA_TIMEOUT_PER_BYTE(entry->buf_len);
  17439. +    status = nfs41_UpcallWaitForReply(entry, io_delay);
  17440. +    if (status) goto out;
  17441. +
  17442. +    if (async) {
  17443. +#ifdef DEBUG_WRITE
  17444. +        DbgP("This is asynchronous write, returning control back to the user\n");
  17445. +#endif
  17446. +        status = STATUS_PENDING;
  17447. +        goto out;
  17448. +    }
  17449. +
  17450. +    if (entry->status == NO_ERROR) {
  17451. +        //update cached file attributes
  17452. +#ifdef ENABLE_TIMINGS
  17453. +        InterlockedIncrement(&write.sops);
  17454. +        InterlockedAdd64(&write.size, entry->u.ReadWrite.len);
  17455. +#endif
  17456. +        nfs41_fcb->StandardInfo.EndOfFile.QuadPart = entry->buf_len +
  17457. +            entry->u.ReadWrite.offset;
  17458. +        status = RxContext->CurrentIrp->IoStatus.Status = STATUS_SUCCESS;
  17459. +        RxContext->IoStatusBlock.Information = entry->buf_len;
  17460. +        nfs41_fcb->changeattr = entry->ChangeTime;
  17461. +
  17462. +        //re-enable write buffering
  17463. +        if (!BooleanFlagOn(LowIoContext->ParamsFor.ReadWrite.Flags,
  17464. +                LOWIO_READWRITEFLAG_PAGING_IO) &&
  17465. +                (SrvOpen->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA)) &&
  17466. +                !pVNetRootContext->write_thru &&
  17467. +                !pVNetRootContext->nocache &&
  17468. +                !nfs41_fobx->write_thru && !nfs41_fobx->nocache &&
  17469. +                !(SrvOpen->BufferingFlags &
  17470. +                (FCB_STATE_WRITEBUFFERING_ENABLED |
  17471. +                 FCB_STATE_WRITECACHING_ENABLED))) {
  17472. +            enable_caching(SrvOpen, nfs41_fobx, nfs41_fcb->changeattr,
  17473. +                pVNetRootContext->session);
  17474. +        } else if (!nfs41_fobx->deleg_type)
  17475. +            nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime);
  17476. +
  17477. +    } else {
  17478. +        status = map_readwrite_errors(entry->status);
  17479. +        RxContext->CurrentIrp->IoStatus.Status = status;
  17480. +        RxContext->IoStatusBlock.Information = 0;
  17481. +    }
  17482. +    nfs41_UpcallDestroy(entry);
  17483. +out:
  17484. +#ifdef ENABLE_TIMINGS
  17485. +    t2 = KeQueryPerformanceCounter(NULL);
  17486. +    InterlockedIncrement(&write.tops);
  17487. +    InterlockedAdd64(&write.ticks, t2.QuadPart - t1.QuadPart);
  17488. +#ifdef ENABLE_INDV_TIMINGS
  17489. +    DbgP("nfs41_Write delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart,
  17490. +        write.tops, write.ticks);
  17491. +#endif
  17492. +#endif
  17493. +#ifdef DEBUG_WRITE
  17494. +    DbgEx();
  17495. +#endif
  17496. +    return status;
  17497. +}
  17498. diff --git a/sys/nfs41sys_symlink.c b/sys/nfs41sys_symlink.c
  17499. new file mode 100644
  17500. index 0000000..ad5d968
  17501. --- /dev/null
  17502. +++ b/sys/nfs41sys_symlink.c
  17503. @@ -0,0 +1,406 @@
  17504. +/* NFSv4.1 client for Windows
  17505. + * Copyright (C) 2012 The Regents of the University of Michigan
  17506. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  17507. + *
  17508. + * Olga Kornievskaia <aglo@umich.edu>
  17509. + * Casey Bodley <cbodley@umich.edu>
  17510. + * Roland Mainz <roland.mainz@nrubsig.org>
  17511. + *
  17512. + * This library is free software; you can redistribute it and/or modify it
  17513. + * under the terms of the GNU Lesser General Public License as published by
  17514. + * the Free Software Foundation; either version 2.1 of the License, or (at
  17515. + * your option) any later version.
  17516. + *
  17517. + * This library is distributed in the hope that it will be useful, but
  17518. + * without any warranty; without even the implied warranty of merchantability
  17519. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  17520. + * License for more details.
  17521. + *
  17522. + * You should have received a copy of the GNU Lesser General Public License
  17523. + * along with this library; if not, write to the Free Software Foundation,
  17524. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  17525. + */
  17526. +
  17527. +#ifndef _KERNEL_MODE
  17528. +#error module requires kernel mode
  17529. +#endif
  17530. +
  17531. +#if ((__STDC_VERSION__-0) < 201710L)
  17532. +#error Code requires ISO C17
  17533. +#endif
  17534. +
  17535. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  17536. +#if _MSC_VER >= 1900
  17537. +#if defined(_WIN64) && defined(_M_X64)
  17538. +#ifndef _AMD64_
  17539. +#define _AMD64_
  17540. +#endif
  17541. +#elif defined(_WIN32) && defined(_M_IX86)
  17542. +#ifndef _X86_
  17543. +#define _X86_
  17544. +#endif
  17545. +#elif defined(_WIN64) && defined(_M_ARM64)
  17546. +#ifndef _ARM64_
  17547. +#define _ARM64_
  17548. +#endif
  17549. +#elif defined(_WIN32) && defined(_M_ARM)
  17550. +#ifndef _ARM_
  17551. +#define _ARM_
  17552. +#endif
  17553. +#else
  17554. +#error Unsupported arch
  17555. +#endif
  17556. +#endif /* _MSC_VER >= 1900 */
  17557. +
  17558. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  17559. +#include <rx.h>
  17560. +#include <windef.h>
  17561. +#include <winerror.h>
  17562. +
  17563. +#include <Ntstrsafe.h>
  17564. +
  17565. +#include "nfs41sys_buildconfig.h"
  17566. +
  17567. +#include "nfs41_driver.h"
  17568. +#include "nfs41sys_debug.h"
  17569. +#include "nfs41_build_features.h"
  17570. +
  17571. +#include "nfs41sys_driver.h"
  17572. +#include "nfs41sys_util.h"
  17573. +
  17574. +
  17575. +NTSTATUS marshal_nfs41_symlink(
  17576. +    nfs41_updowncall_entry *entry,
  17577. +    unsigned char *buf,
  17578. +    ULONG buf_len,
  17579. +    ULONG *len)
  17580. +{
  17581. +    NTSTATUS status = STATUS_SUCCESS;
  17582. +    ULONG header_len = 0;
  17583. +    unsigned char *tmp = buf;
  17584. +
  17585. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  17586. +    if (status) goto out;
  17587. +    else tmp += *len;
  17588. +
  17589. +    header_len = *len + sizeof(BOOLEAN) + length_as_utf8(entry->filename);
  17590. +    if (entry->u.Symlink.set)
  17591. +        header_len += length_as_utf8(entry->u.Symlink.target);
  17592. +    if (header_len > buf_len) {
  17593. +        status = STATUS_INSUFFICIENT_RESOURCES;
  17594. +        goto out;
  17595. +    }
  17596. +
  17597. +    status = marshall_unicode_as_utf8(&tmp, entry->filename);
  17598. +    if (status) goto out;
  17599. +    RtlCopyMemory(tmp, &entry->u.Symlink.set, sizeof(BOOLEAN));
  17600. +    tmp += sizeof(BOOLEAN);
  17601. +    if (entry->u.Symlink.set) {
  17602. +        status = marshall_unicode_as_utf8(&tmp, entry->u.Symlink.target);
  17603. +        if (status) goto out;
  17604. +    }
  17605. +    *len = header_len;
  17606. +
  17607. +#ifdef DEBUG_MARSHAL_DETAIL
  17608. +    DbgP("marshal_nfs41_symlink: name '%wZ' symlink target '%wZ'\n",
  17609. +         entry->filename,
  17610. +         entry->u.Symlink.set?entry->u.Symlink.target : NULL);
  17611. +#endif
  17612. +out:
  17613. +    return status;
  17614. +}
  17615. +
  17616. +void unmarshal_nfs41_symlink(
  17617. +    nfs41_updowncall_entry *cur,
  17618. +    unsigned char **buf)
  17619. +{
  17620. +    if (cur->u.Symlink.set) return;
  17621. +
  17622. +    RtlCopyMemory(&cur->u.Symlink.target->Length, *buf, sizeof(USHORT));
  17623. +    *buf += sizeof(USHORT);
  17624. +    if (cur->u.Symlink.target->Length >
  17625. +            cur->u.Symlink.target->MaximumLength) {
  17626. +        cur->status = STATUS_BUFFER_TOO_SMALL;
  17627. +        return;
  17628. +    }
  17629. +    RtlCopyMemory(cur->u.Symlink.target->Buffer, *buf,
  17630. +        cur->u.Symlink.target->Length);
  17631. +    cur->u.Symlink.target->Length -= sizeof(UNICODE_NULL);
  17632. +}
  17633. +
  17634. +NTSTATUS map_symlink_errors(
  17635. +    NTSTATUS status)
  17636. +{
  17637. +    switch (status) {
  17638. +    case NO_ERROR:                  return STATUS_SUCCESS;
  17639. +    case ERROR_INVALID_REPARSE_DATA: return STATUS_IO_REPARSE_DATA_INVALID;
  17640. +    case ERROR_NOT_A_REPARSE_POINT: return STATUS_NOT_A_REPARSE_POINT;
  17641. +    case ERROR_ACCESS_DENIED:       return STATUS_ACCESS_DENIED;
  17642. +    case ERROR_NOT_EMPTY:           return STATUS_DIRECTORY_NOT_EMPTY;
  17643. +    case ERROR_OUTOFMEMORY:         return STATUS_INSUFFICIENT_RESOURCES;
  17644. +    case ERROR_INSUFFICIENT_BUFFER: return STATUS_BUFFER_TOO_SMALL;
  17645. +    case STATUS_BUFFER_TOO_SMALL:
  17646. +    case ERROR_BUFFER_OVERFLOW:     return STATUS_BUFFER_OVERFLOW;
  17647. +    case ERROR_DISK_FULL:           return STATUS_DISK_FULL;
  17648. +    case ERROR_DISK_QUOTA_EXCEEDED: return STATUS_DISK_QUOTA_EXCEEDED;
  17649. +    case ERROR_FILE_TOO_LARGE:      return STATUS_FILE_TOO_LARGE;
  17650. +    case ERROR_TOO_MANY_LINKS:      return STATUS_TOO_MANY_LINKS;
  17651. +    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  17652. +    default:
  17653. +        print_error("map_symlink_errors: "
  17654. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  17655. +            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status);
  17656. +    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  17657. +    }
  17658. +}
  17659. +
  17660. +static void print_reparse_buffer(
  17661. +    PREPARSE_DATA_BUFFER Reparse)
  17662. +{
  17663. +    UNICODE_STRING name;
  17664. +    DbgP("ReparseTag:           %08X\n", Reparse->ReparseTag);
  17665. +    DbgP("ReparseDataLength:    %8u\n", Reparse->ReparseDataLength);
  17666. +    DbgP("Reserved:             %8u\n", Reparse->Reserved);
  17667. +    DbgP("SubstituteNameOffset: %8u\n",
  17668. +         Reparse->SymbolicLinkReparseBuffer.SubstituteNameOffset);
  17669. +    DbgP("SubstituteNameLength: %8u\n",
  17670. +         Reparse->SymbolicLinkReparseBuffer.SubstituteNameLength);
  17671. +    DbgP("PrintNameOffset:      %8u\n",
  17672. +         Reparse->SymbolicLinkReparseBuffer.PrintNameOffset);
  17673. +    DbgP("PrintNameLength:      %8u\n",
  17674. +         Reparse->SymbolicLinkReparseBuffer.PrintNameLength);
  17675. +    DbgP("Flags:                %08X\n",
  17676. +         Reparse->SymbolicLinkReparseBuffer.Flags);
  17677. +
  17678. +    name.Buffer = &Reparse->SymbolicLinkReparseBuffer.PathBuffer[
  17679. +        Reparse->SymbolicLinkReparseBuffer.SubstituteNameOffset/sizeof(WCHAR)];
  17680. +    name.MaximumLength = name.Length =
  17681. +        Reparse->SymbolicLinkReparseBuffer.SubstituteNameLength;
  17682. +    DbgP("SubstituteName:       '%wZ'\n", &name);
  17683. +
  17684. +    name.Buffer = &Reparse->SymbolicLinkReparseBuffer.PathBuffer[
  17685. +        Reparse->SymbolicLinkReparseBuffer.PrintNameOffset/sizeof(WCHAR)];
  17686. +    name.MaximumLength = name.Length =
  17687. +        Reparse->SymbolicLinkReparseBuffer.PrintNameLength;
  17688. +    DbgP("PrintName:            '%wZ'\n", &name);
  17689. +}
  17690. +
  17691. +static
  17692. +NTSTATUS check_nfs41_setreparse_args(
  17693. +    IN PRX_CONTEXT RxContext)
  17694. +{
  17695. +    NTSTATUS status = STATUS_SUCCESS;
  17696. +    __notnull XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl;
  17697. +    __notnull PREPARSE_DATA_BUFFER Reparse = (PREPARSE_DATA_BUFFER)FsCtl->pInputBuffer;
  17698. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  17699. +    __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext =
  17700. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  17701. +    const ULONG HeaderLen = REPARSE_DATA_BUFFER_HEADER_SIZE;
  17702. +
  17703. +    /* access checks */
  17704. +    if (VNetRootContext->read_only) {
  17705. +        status = STATUS_MEDIA_WRITE_PROTECTED;
  17706. +        goto out;
  17707. +    }
  17708. +    if (!(SrvOpen->DesiredAccess & (FILE_WRITE_DATA|FILE_WRITE_ATTRIBUTES))) {
  17709. +        status = STATUS_ACCESS_DENIED;
  17710. +        goto out;
  17711. +    }
  17712. +
  17713. +    /* must have a filename longer than vnetroot name,
  17714. +     * or it's trying to operate on the volume itself */
  17715. +    if (is_root_directory(RxContext)) {
  17716. +        status = STATUS_INVALID_PARAMETER;
  17717. +        goto out;
  17718. +    }
  17719. +    if (FsCtl->pOutputBuffer != NULL) {
  17720. +        status = STATUS_INVALID_PARAMETER;
  17721. +        goto out;
  17722. +    }
  17723. +
  17724. +    /* validate input buffer and length */
  17725. +    if (!Reparse) {
  17726. +        status = STATUS_INVALID_BUFFER_SIZE;
  17727. +        goto out;
  17728. +    }
  17729. +
  17730. +    if (FsCtl->InputBufferLength < HeaderLen ||
  17731. +            FsCtl->InputBufferLength > MAXIMUM_REPARSE_DATA_BUFFER_SIZE) {
  17732. +        status = STATUS_IO_REPARSE_DATA_INVALID;
  17733. +        goto out;
  17734. +    }
  17735. +    if (FsCtl->InputBufferLength != HeaderLen + Reparse->ReparseDataLength) {
  17736. +        status = STATUS_IO_REPARSE_DATA_INVALID;
  17737. +        goto out;
  17738. +    }
  17739. +
  17740. +    /* validate reparse tag */
  17741. +    if (!IsReparseTagValid(Reparse->ReparseTag)) {
  17742. +        status = STATUS_IO_REPARSE_TAG_INVALID;
  17743. +        goto out;
  17744. +    }
  17745. +    if (Reparse->ReparseTag != IO_REPARSE_TAG_SYMLINK) {
  17746. +        status = STATUS_IO_REPARSE_TAG_MISMATCH;
  17747. +        goto out;
  17748. +    }
  17749. +out:
  17750. +    return status;
  17751. +}
  17752. +
  17753. +NTSTATUS nfs41_SetReparsePoint(
  17754. +    IN OUT PRX_CONTEXT RxContext)
  17755. +{
  17756. +    NTSTATUS status;
  17757. +    UNICODE_STRING TargetName;
  17758. +    __notnull XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl;
  17759. +    __notnull PREPARSE_DATA_BUFFER Reparse = (PREPARSE_DATA_BUFFER)FsCtl->pInputBuffer;
  17760. +    __notnull PNFS41_FOBX Fobx = NFS41GetFobxExtension(RxContext->pFobx);
  17761. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  17762. +    __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext =
  17763. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  17764. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  17765. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  17766. +    nfs41_updowncall_entry *entry;
  17767. +
  17768. +#ifdef DEBUG_SYMLINK
  17769. +    DbgEn();
  17770. +    print_debug_header(RxContext);
  17771. +    print_reparse_buffer(Reparse);
  17772. +#endif
  17773. +    status = check_nfs41_setreparse_args(RxContext);
  17774. +    if (status) goto out;
  17775. +
  17776. +    TargetName.MaximumLength = TargetName.Length =
  17777. +        Reparse->SymbolicLinkReparseBuffer.PrintNameLength;
  17778. +    TargetName.Buffer = &Reparse->SymbolicLinkReparseBuffer.PathBuffer[
  17779. +        Reparse->SymbolicLinkReparseBuffer.PrintNameOffset/sizeof(WCHAR)];
  17780. +
  17781. +    status = nfs41_UpcallCreate(NFS41_SYMLINK, &Fobx->sec_ctx,
  17782. +        VNetRootContext->session, Fobx->nfs41_open_state,
  17783. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  17784. +    if (status) goto out;
  17785. +
  17786. +    entry->u.Symlink.target = &TargetName;
  17787. +    entry->u.Symlink.set = TRUE;
  17788. +
  17789. +    status = nfs41_UpcallWaitForReply(entry, VNetRootContext->timeout);
  17790. +    if (status) goto out;
  17791. +
  17792. +    status = map_symlink_errors(entry->status);
  17793. +    nfs41_UpcallDestroy(entry);
  17794. +out:
  17795. +#ifdef DEBUG_SYMLINK
  17796. +    DbgEx();
  17797. +#endif
  17798. +    return status;
  17799. +}
  17800. +
  17801. +static
  17802. +NTSTATUS check_nfs41_getreparse_args(
  17803. +    PRX_CONTEXT RxContext)
  17804. +{
  17805. +    NTSTATUS status = STATUS_SUCCESS;
  17806. +    XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl;
  17807. +    const USHORT HeaderLen = FIELD_OFFSET(REPARSE_DATA_BUFFER,
  17808. +        SymbolicLinkReparseBuffer.PathBuffer);
  17809. +
  17810. +    /* must have a filename longer than vnetroot name,
  17811. +     * or it's trying to operate on the volume itself */
  17812. +    if (is_root_directory(RxContext)) {
  17813. +        status = STATUS_INVALID_PARAMETER;
  17814. +        goto out;
  17815. +    }
  17816. +    /* ifs reparse tests expect STATUS_INVALID_PARAMETER,
  17817. +     * but 'dir' passes a buffer here when querying symlinks
  17818. +    if (FsCtl->pInputBuffer != NULL) {
  17819. +        status = STATUS_INVALID_PARAMETER;
  17820. +        goto out;
  17821. +    } */
  17822. +    if (!FsCtl->pOutputBuffer) {
  17823. +        status = STATUS_INVALID_USER_BUFFER;
  17824. +        goto out;
  17825. +    }
  17826. +    if (!BooleanFlagOn(RxContext->pFcb->Attributes,
  17827. +            FILE_ATTRIBUTE_REPARSE_POINT)) {
  17828. +        status = STATUS_NOT_A_REPARSE_POINT;
  17829. +        DbgP("FILE_ATTRIBUTE_REPARSE_POINT is not set!\n");
  17830. +        goto out;
  17831. +    }
  17832. +
  17833. +    if (FsCtl->OutputBufferLength < HeaderLen) {
  17834. +        RxContext->InformationToReturn = HeaderLen;
  17835. +        status = STATUS_BUFFER_TOO_SMALL;
  17836. +        goto out;
  17837. +    }
  17838. +out:
  17839. +    return status;
  17840. +}
  17841. +
  17842. +NTSTATUS nfs41_GetReparsePoint(
  17843. +    IN OUT PRX_CONTEXT RxContext)
  17844. +{
  17845. +    NTSTATUS status;
  17846. +    UNICODE_STRING TargetName;
  17847. +    XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl;
  17848. +    __notnull PNFS41_FOBX Fobx = NFS41GetFobxExtension(RxContext->pFobx);
  17849. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  17850. +    __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext =
  17851. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  17852. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  17853. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  17854. +    nfs41_updowncall_entry *entry;
  17855. +    const USHORT HeaderLen = FIELD_OFFSET(REPARSE_DATA_BUFFER,
  17856. +        SymbolicLinkReparseBuffer.PathBuffer);
  17857. +
  17858. +#ifdef DEBUG_SYMLINK
  17859. +    DbgEn();
  17860. +#endif
  17861. +    status = check_nfs41_getreparse_args(RxContext);
  17862. +    if (status) goto out;
  17863. +
  17864. +    TargetName.Buffer = (PWCH)((PBYTE)FsCtl->pOutputBuffer + HeaderLen);
  17865. +    TargetName.MaximumLength = (USHORT)min(FsCtl->OutputBufferLength -
  17866. +        HeaderLen, 0xFFFF);
  17867. +
  17868. +    status = nfs41_UpcallCreate(NFS41_SYMLINK, &Fobx->sec_ctx,
  17869. +        VNetRootContext->session, Fobx->nfs41_open_state,
  17870. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  17871. +    if (status) goto out;
  17872. +
  17873. +    entry->u.Symlink.target = &TargetName;
  17874. +    entry->u.Symlink.set = FALSE;
  17875. +
  17876. +    status = nfs41_UpcallWaitForReply(entry, VNetRootContext->timeout);
  17877. +    if (status) goto out;
  17878. +
  17879. +    status = map_symlink_errors(entry->status);
  17880. +    if (status == STATUS_SUCCESS) {
  17881. +        /* fill in the output buffer */
  17882. +        PREPARSE_DATA_BUFFER Reparse = (PREPARSE_DATA_BUFFER)
  17883. +            FsCtl->pOutputBuffer;
  17884. +        Reparse->ReparseTag = IO_REPARSE_TAG_SYMLINK;
  17885. +        Reparse->ReparseDataLength = HeaderLen + TargetName.Length -
  17886. +            REPARSE_DATA_BUFFER_HEADER_SIZE;
  17887. +        Reparse->Reserved = 0;
  17888. +        Reparse->SymbolicLinkReparseBuffer.Flags = SYMLINK_FLAG_RELATIVE;
  17889. +        /* PrintName and SubstituteName point to the same string */
  17890. +        Reparse->SymbolicLinkReparseBuffer.SubstituteNameOffset = 0;
  17891. +        Reparse->SymbolicLinkReparseBuffer.SubstituteNameLength =
  17892. +            TargetName.Length;
  17893. +        Reparse->SymbolicLinkReparseBuffer.PrintNameOffset = 0;
  17894. +        Reparse->SymbolicLinkReparseBuffer.PrintNameLength = TargetName.Length;
  17895. +        print_reparse_buffer(Reparse);
  17896. +
  17897. +        RxContext->IoStatusBlock.Information =
  17898. +            (ULONG_PTR)HeaderLen + TargetName.Length;
  17899. +    } else if (status == STATUS_BUFFER_TOO_SMALL) {
  17900. +        RxContext->InformationToReturn =
  17901. +            (ULONG_PTR)HeaderLen + TargetName.Length;
  17902. +    }
  17903. +    nfs41_UpcallDestroy(entry);
  17904. +out:
  17905. +#ifdef DEBUG_SYMLINK
  17906. +    DbgEx();
  17907. +#endif
  17908. +    return status;
  17909. +}
  17910. diff --git a/sys/nfs41sys_updowncall.c b/sys/nfs41sys_updowncall.c
  17911. new file mode 100644
  17912. index 0000000..aff160d
  17913. --- /dev/null
  17914. +++ b/sys/nfs41sys_updowncall.c
  17915. @@ -0,0 +1,668 @@
  17916. +/* NFSv4.1 client for Windows
  17917. + * Copyright (C) 2012 The Regents of the University of Michigan
  17918. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  17919. + *
  17920. + * Olga Kornievskaia <aglo@umich.edu>
  17921. + * Casey Bodley <cbodley@umich.edu>
  17922. + * Roland Mainz <roland.mainz@nrubsig.org>
  17923. + *
  17924. + * This library is free software; you can redistribute it and/or modify it
  17925. + * under the terms of the GNU Lesser General Public License as published by
  17926. + * the Free Software Foundation; either version 2.1 of the License, or (at
  17927. + * your option) any later version.
  17928. + *
  17929. + * This library is distributed in the hope that it will be useful, but
  17930. + * without any warranty; without even the implied warranty of merchantability
  17931. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  17932. + * License for more details.
  17933. + *
  17934. + * You should have received a copy of the GNU Lesser General Public License
  17935. + * along with this library; if not, write to the Free Software Foundation,
  17936. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  17937. + */
  17938. +
  17939. +#ifndef _KERNEL_MODE
  17940. +#error module requires kernel mode
  17941. +#endif
  17942. +
  17943. +#if ((__STDC_VERSION__-0) < 201710L)
  17944. +#error Code requires ISO C17
  17945. +#endif
  17946. +
  17947. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  17948. +#if _MSC_VER >= 1900
  17949. +#if defined(_WIN64) && defined(_M_X64)
  17950. +#ifndef _AMD64_
  17951. +#define _AMD64_
  17952. +#endif
  17953. +#elif defined(_WIN32) && defined(_M_IX86)
  17954. +#ifndef _X86_
  17955. +#define _X86_
  17956. +#endif
  17957. +#elif defined(_WIN64) && defined(_M_ARM64)
  17958. +#ifndef _ARM64_
  17959. +#define _ARM64_
  17960. +#endif
  17961. +#elif defined(_WIN32) && defined(_M_ARM)
  17962. +#ifndef _ARM_
  17963. +#define _ARM_
  17964. +#endif
  17965. +#else
  17966. +#error Unsupported arch
  17967. +#endif
  17968. +#endif /* _MSC_VER >= 1900 */
  17969. +
  17970. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  17971. +#include <rx.h>
  17972. +#include <windef.h>
  17973. +#include <winerror.h>
  17974. +
  17975. +#include <Ntstrsafe.h>
  17976. +
  17977. +#include "nfs41sys_buildconfig.h"
  17978. +
  17979. +#include "nfs41_driver.h"
  17980. +#include "nfs41sys_debug.h"
  17981. +#include "nfs41_build_features.h"
  17982. +
  17983. +#include "nfs41sys_driver.h"
  17984. +#include "nfs41sys_util.h"
  17985. +
  17986. +
  17987. +static
  17988. +nfs41_updowncall_entry *nfs41_upcall_allocate_updowncall_entry(void)
  17989. +{
  17990. +    nfs41_updowncall_entry *e;
  17991. +#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  17992. +    e = ExAllocateFromNPagedLookasideList(
  17993. +        &updowncall_entry_upcall_lookasidelist);
  17994. +
  17995. +#ifdef LOOKASIDELISTS_STATS
  17996. +    volatile static long cnt = 0;
  17997. +    if ((cnt++ % 100) == 0) {
  17998. +        print_lookasidelist_stat("updowncall_entry_upcall",
  17999. +            &updowncall_entry_upcall_lookasidelist);
  18000. +    }
  18001. +#endif /* LOOKASIDELISTS_STATS */
  18002. +#else
  18003. +    e = RxAllocatePoolWithTag(NonPagedPoolNx,
  18004. +        sizeof(nfs41_updowncall_entry),
  18005. +        NFS41_MM_POOLTAG_UP);
  18006. +#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  18007. +
  18008. +    return e;
  18009. +}
  18010. +
  18011. +static
  18012. +void nfs41_upcall_free_updowncall_entry(nfs41_updowncall_entry *entry)
  18013. +{
  18014. +#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  18015. +    ExFreeToNPagedLookasideList(&updowncall_entry_upcall_lookasidelist,
  18016. +        entry);
  18017. +#else
  18018. +    RxFreePool(entry);
  18019. +#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  18020. +}
  18021. +
  18022. +static
  18023. +nfs41_updowncall_entry *nfs41_downcall_allocate_updowncall_entry(void)
  18024. +{
  18025. +    nfs41_updowncall_entry *e;
  18026. +#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  18027. +    e = ExAllocateFromNPagedLookasideList(
  18028. +        &updowncall_entry_downcall_lookasidelist);
  18029. +
  18030. +#ifdef LOOKASIDELISTS_STATS
  18031. +    volatile static long cnt = 0;
  18032. +    if ((cnt++ % 100) == 0) {
  18033. +        print_lookasidelist_stat("updowncall_entry_downcall",
  18034. +            &updowncall_entry_downcall_lookasidelist);
  18035. +    }
  18036. +#endif /* LOOKASIDELISTS_STATS */
  18037. +#else
  18038. +    e = RxAllocatePoolWithTag(NonPagedPoolNx,
  18039. +        sizeof(nfs41_updowncall_entry),
  18040. +        NFS41_MM_POOLTAG_DOWN);
  18041. +#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  18042. +    return e;
  18043. +}
  18044. +
  18045. +static
  18046. +void nfs41_downcall_free_updowncall_entry(nfs41_updowncall_entry *entry)
  18047. +{
  18048. +#ifdef USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM
  18049. +    ExFreeToNPagedLookasideList(&updowncall_entry_downcall_lookasidelist,
  18050. +        entry);
  18051. +#else
  18052. +    RxFreePool(entry);
  18053. +#endif /* USE_LOOKASIDELISTS_FOR_UPDOWNCALLENTRY_MEM */
  18054. +}
  18055. +
  18056. +static void unmarshal_nfs41_header(
  18057. +    nfs41_updowncall_entry *tmp,
  18058. +    unsigned char **buf)
  18059. +{
  18060. +    RtlZeroMemory(tmp, sizeof(nfs41_updowncall_entry));
  18061. +
  18062. +    RtlCopyMemory(&tmp->xid, *buf, sizeof(tmp->xid));
  18063. +    *buf += sizeof(tmp->xid);
  18064. +    RtlCopyMemory(&tmp->opcode, *buf, sizeof(tmp->opcode));
  18065. +    *buf += sizeof(tmp->opcode);
  18066. +    RtlCopyMemory(&tmp->status, *buf, sizeof(tmp->status));
  18067. +    *buf += sizeof(tmp->status);
  18068. +    RtlCopyMemory(&tmp->errno, *buf, sizeof(tmp->errno));
  18069. +    *buf += sizeof(tmp->errno);
  18070. +#ifdef DEBUG_MARSHAL_HEADER
  18071. +    DbgP("[downcall header] "
  18072. +        "xid=%lld opcode='%s' status=0x%lx errno=%d\n",
  18073. +        tmp->xid,
  18074. +        ENTRY_OPCODE2STRING(tmp), (long)tmp->status, tmp->errno);
  18075. +#endif
  18076. +}
  18077. +
  18078. +NTSTATUS handle_upcall(
  18079. +    IN PRX_CONTEXT RxContext,
  18080. +    IN nfs41_updowncall_entry *entry,
  18081. +    OUT ULONG *len)
  18082. +{
  18083. +    NTSTATUS status = STATUS_SUCCESS;
  18084. +    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  18085. +    ULONG cbOut = LowIoContext->ParamsFor.IoCtl.OutputBufferLength;
  18086. +    unsigned char *pbOut = LowIoContext->ParamsFor.IoCtl.pOutputBuffer;
  18087. +
  18088. +#ifdef NFS41_DRIVER_STABILITY_HACKS
  18089. +    /*
  18090. +     * Workaround for random crashes like this while compiling
  18091. +     * the "gcc" compiler with a highly-parallel build.
  18092. +     * Stack trace usually looks like this:
  18093. +     * ---- snip ----
  18094. +     * nt!SeTokenCanImpersonate+0x47
  18095. +     * nt!PsImpersonateClient+0x126
  18096. +     * nt!SeImpersonateClientEx+0x35
  18097. +     * nfs41_driver!handle_upcall+0x59 [C:\cygwin64\home\roland_mainz\work\msnfs41_uidmapping\ms-nfs41-client\sys\nfs41_driver.c @ 1367]
  18098. +     * nfs41_driver!nfs41_upcall+0xe7 [C:\cygwin64\home\roland_mainz\work\msnfs41_uidmapping\ms-nfs41-client\sys\nfs41_driver.c @ 1578]
  18099. +     * nfs41_driver!nfs41_DevFcbXXXControlFile+0x128 [C:\cygwin64\home\roland_mainz\work\msnfs41_uidmapping\ms-nfs41-client\sys\nfs41_driver.c @ 2418]
  18100. +     * nfs41_driver!RxXXXControlFileCallthru+0x76 [base\fs\rdr2\rdbss\ntdevfcb.c @ 130]
  18101. +     * nfs41_driver!RxCommonDevFCBIoCtl+0x58 [base\fs\rdr2\rdbss\ntdevfcb.c @ 491]
  18102. +     * nfs41_driver!RxFsdCommonDispatch+0x442 [base\fs\rdr2\rdbss\ntfsd.c @ 848]
  18103. +     * nfs41_driver!RxFsdDispatch+0xfd [base\fs\rdr2\rdbss\ntfsd.c @ 442]
  18104. +     * nfs41_driver!nfs41_FsdDispatch+0x67 [C:\cygwin64\home\roland_mainz\work\msnfs41_uidmapping\ms-nfs41-client\sys\nfs41_driver.c @ 6863]
  18105. +     * nt!IofCallDriver+0x55
  18106. +     * mup!MupiCallUncProvider+0xb8
  18107. +     * mup!MupStateMachine+0x59
  18108. +     * mup!MupFsdIrpPassThrough+0x17e
  18109. +     * nt!IofCallDriver+0x55
  18110. +     * FLTMGR!FltpDispatch+0xd6
  18111. +     * nt!IofCallDriver+0x55
  18112. +     * nt!IopSynchronousServiceTail+0x34c
  18113. +     * nt!IopXxxControlFile+0xd13
  18114. +     * nt!NtDeviceIoControlFile+0x56
  18115. +     * nt!KiSystemServiceCopyEnd+0x25
  18116. +     * ntdll!NtDeviceIoControlFile+0x14
  18117. +     * KERNELBASE!DeviceIoControl+0x6b
  18118. +     * KERNEL32!DeviceIoControlImplementation+0x81
  18119. +     * nfsd_debug+0xc7b14
  18120. +     * nfsd_debug+0xc79fb
  18121. +     * nfsd_debug+0x171e80
  18122. +     * KERNEL32!BaseThreadInitThunk+0x14
  18123. +     * ntdll!RtlUserThreadStart+0x21
  18124. +     * ---- snip ----
  18125. +     */
  18126. +    __try {
  18127. +        status = SeImpersonateClientEx(entry->psec_ctx, NULL);
  18128. +    } __except(EXCEPTION_EXECUTE_HANDLER) {
  18129. +        NTSTATUS code;
  18130. +        code = GetExceptionCode();
  18131. +        print_error("handle_upcall: Call to SeImpersonateClientEx() "
  18132. +            "failed due to exception 0x%0x\n", (int)code);
  18133. +        status = STATUS_INTERNAL_ERROR;
  18134. +    }
  18135. +#else
  18136. +    status = SeImpersonateClientEx(entry->psec_ctx, NULL);
  18137. +#endif /* NFS41_DRIVER_STABILITY_HACKS */
  18138. +    if (status != STATUS_SUCCESS) {
  18139. +        print_error("handle_upcall: "
  18140. +            "SeImpersonateClientEx() failed 0x%x\n", status);
  18141. +        goto out;
  18142. +    }
  18143. +
  18144. +    switch(entry->opcode) {
  18145. +    case NFS41_SHUTDOWN:
  18146. +        status = marshal_nfs41_shutdown(entry, pbOut, cbOut, len);
  18147. +        KeSetEvent(&entry->cond, 0, FALSE);
  18148. +        break;
  18149. +    case NFS41_MOUNT:
  18150. +        status = marshal_nfs41_mount(entry, pbOut, cbOut, len);
  18151. +        break;
  18152. +    case NFS41_UNMOUNT:
  18153. +        status = marshal_nfs41_unmount(entry, pbOut, cbOut, len);
  18154. +        break;
  18155. +    case NFS41_OPEN:
  18156. +        status = marshal_nfs41_open(entry, pbOut, cbOut, len);
  18157. +        break;
  18158. +    case NFS41_READ:
  18159. +        status = marshal_nfs41_rw(entry, pbOut, cbOut, len);
  18160. +        break;
  18161. +    case NFS41_WRITE:
  18162. +        status = marshal_nfs41_rw(entry, pbOut, cbOut, len);
  18163. +        break;
  18164. +    case NFS41_LOCK:
  18165. +        status = marshal_nfs41_lock(entry, pbOut, cbOut, len);
  18166. +        break;
  18167. +    case NFS41_UNLOCK:
  18168. +        status = marshal_nfs41_unlock(entry, pbOut, cbOut, len);
  18169. +        break;
  18170. +    case NFS41_CLOSE:
  18171. +        status = marshal_nfs41_close(entry, pbOut, cbOut, len);
  18172. +        break;
  18173. +    case NFS41_DIR_QUERY:
  18174. +        status = marshal_nfs41_dirquery(entry, pbOut, cbOut, len);
  18175. +        break;
  18176. +    case NFS41_FILE_QUERY:
  18177. +    case NFS41_FILE_QUERY_TIME_BASED_COHERENCY:
  18178. +        status = marshal_nfs41_filequery(entry, pbOut, cbOut, len);
  18179. +        break;
  18180. +    case NFS41_FILE_SET:
  18181. +        status = marshal_nfs41_fileset(entry, pbOut, cbOut, len);
  18182. +        break;
  18183. +    case NFS41_EA_SET:
  18184. +        status = marshal_nfs41_easet(entry, pbOut, cbOut, len);
  18185. +        break;
  18186. +    case NFS41_EA_GET:
  18187. +        status = marshal_nfs41_eaget(entry, pbOut, cbOut, len);
  18188. +        break;
  18189. +    case NFS41_SYMLINK:
  18190. +        status = marshal_nfs41_symlink(entry, pbOut, cbOut, len);
  18191. +        break;
  18192. +    case NFS41_VOLUME_QUERY:
  18193. +        status = marshal_nfs41_volume(entry, pbOut, cbOut, len);
  18194. +        break;
  18195. +    case NFS41_ACL_QUERY:
  18196. +        status = marshal_nfs41_getacl(entry, pbOut, cbOut, len);
  18197. +        break;
  18198. +    case NFS41_ACL_SET:
  18199. +        status = marshal_nfs41_setacl(entry, pbOut, cbOut, len);
  18200. +        break;
  18201. +    default:
  18202. +        status = STATUS_INVALID_PARAMETER;
  18203. +        print_error("Unknown nfs41 ops %d\n", entry->opcode);
  18204. +    }
  18205. +
  18206. +    // if (status == STATUS_SUCCESS)
  18207. +    //     print_hexbuf("upcall buffer", pbOut, *len);
  18208. +
  18209. +out:
  18210. +    return status;
  18211. +}
  18212. +
  18213. +NTSTATUS nfs41_UpcallCreate(
  18214. +    IN DWORD opcode,
  18215. +    IN PSECURITY_CLIENT_CONTEXT clnt_sec_ctx,
  18216. +    IN HANDLE session,
  18217. +    IN HANDLE open_state,
  18218. +    IN DWORD version,
  18219. +    IN PUNICODE_STRING filename,
  18220. +    OUT nfs41_updowncall_entry **entry_out)
  18221. +{
  18222. +    NTSTATUS status = STATUS_SUCCESS;
  18223. +    nfs41_updowncall_entry *entry;
  18224. +    SECURITY_SUBJECT_CONTEXT sec_ctx;
  18225. +    SECURITY_QUALITY_OF_SERVICE sec_qos;
  18226. +
  18227. +    entry = nfs41_upcall_allocate_updowncall_entry();
  18228. +    if (entry == NULL) {
  18229. +        status = STATUS_INSUFFICIENT_RESOURCES;
  18230. +        goto out;
  18231. +    }
  18232. +
  18233. +    RtlZeroMemory(entry, sizeof(nfs41_updowncall_entry));
  18234. +    entry->xid = InterlockedIncrement64(&xid);
  18235. +    entry->opcode = opcode;
  18236. +    entry->state = NFS41_WAITING_FOR_UPCALL;
  18237. +    entry->session = session;
  18238. +    entry->open_state = open_state;
  18239. +    entry->version = version;
  18240. +    if (filename && filename->Length) entry->filename = filename;
  18241. +    else if (filename && !filename->Length) entry->filename = (PUNICODE_STRING)&SLASH;
  18242. +    else entry->filename = (PUNICODE_STRING)&EMPTY_STRING;
  18243. +    /*XXX KeInitializeEvent will bugcheck under verifier if allocated
  18244. +     * from PagedPool? */
  18245. +    KeInitializeEvent(&entry->cond, SynchronizationEvent, FALSE);
  18246. +    ExInitializeFastMutex(&entry->lock);
  18247. +
  18248. +    if (clnt_sec_ctx == NULL) {
  18249. +        SeCaptureSubjectContext(&sec_ctx);
  18250. +        sec_qos.ContextTrackingMode = SECURITY_STATIC_TRACKING;
  18251. +        sec_qos.ImpersonationLevel = SecurityImpersonation;
  18252. +        sec_qos.Length = sizeof(SECURITY_QUALITY_OF_SERVICE);
  18253. +        sec_qos.EffectiveOnly = 0;
  18254. +        entry->psec_ctx = &entry->sec_ctx;
  18255. +        /*
  18256. +         * Arg |ServerIsRemote| must be |FALSE|, otherwise processes
  18257. +         * like Cygwin setup-x86_64.exe can fail during "Activation
  18258. +         * Context" creation in
  18259. +         * |SeCreateClientSecurityFromSubjectContext()| with
  18260. +         * |STATUS_BAD_IMPERSONATION_LEVEL|
  18261. +         */
  18262. +        status = SeCreateClientSecurityFromSubjectContext(&sec_ctx, &sec_qos,
  18263. +                    FALSE, entry->psec_ctx);
  18264. +        if (status != STATUS_SUCCESS) {
  18265. +            print_error("nfs41_UpcallCreate: "
  18266. +                "SeCreateClientSecurityFromSubjectContext() "
  18267. +                "failed with 0x%x\n",
  18268. +                status);
  18269. +            nfs41_upcall_free_updowncall_entry(entry);
  18270. +            entry = NULL;
  18271. +        }
  18272. +
  18273. +        SeReleaseSubjectContext(&sec_ctx);
  18274. +    } else {
  18275. +        entry->psec_ctx = clnt_sec_ctx;
  18276. +    }
  18277. +
  18278. +    if (entry && entry->psec_ctx) {
  18279. +        /*
  18280. +         * Refcount client token (as |entry->psec_ctx_clienttoken|)
  18281. +         * during lifetime of this |updowncall_entry| to avoid
  18282. +         * crashes during |SeImpersonateClientEx()| if the
  18283. +         * calling client thread exits.
  18284. +         */
  18285. +        entry->psec_ctx_clienttoken = entry->psec_ctx->ClientToken;
  18286. +        ObReferenceObject(entry->psec_ctx_clienttoken);
  18287. +    }
  18288. +
  18289. +    *entry_out = entry;
  18290. +out:
  18291. +    return status;
  18292. +}
  18293. +
  18294. +void nfs41_UpcallDestroy(nfs41_updowncall_entry *entry)
  18295. +{
  18296. +    if (!entry)
  18297. +        return;
  18298. +
  18299. +    if (entry->psec_ctx_clienttoken) {
  18300. +        ObDereferenceObject(entry->psec_ctx_clienttoken);
  18301. +    }
  18302. +
  18303. +    nfs41_upcall_free_updowncall_entry(entry);
  18304. +}
  18305. +
  18306. +
  18307. +NTSTATUS nfs41_UpcallWaitForReply(
  18308. +    IN nfs41_updowncall_entry *entry,
  18309. +    IN DWORD secs)
  18310. +{
  18311. +    NTSTATUS status = STATUS_SUCCESS;
  18312. +
  18313. +    nfs41_AddEntry(upcallLock, upcall, entry);
  18314. +    KeSetEvent(&upcallEvent, 0, FALSE);
  18315. +
  18316. +    if (entry->async_op)
  18317. +        goto out;
  18318. +
  18319. +    LARGE_INTEGER timeout;
  18320. +    timeout.QuadPart = RELATIVE(SECONDS(secs));
  18321. +retry_wait:
  18322. +    status = KeWaitForSingleObject(&entry->cond, Executive,
  18323. +                UserMode, FALSE, &timeout);
  18324. +
  18325. +    if (status == STATUS_TIMEOUT)
  18326. +            status = STATUS_NETWORK_UNREACHABLE;
  18327. +
  18328. +    print_wait_status(0, "[downcall]", status,
  18329. +        ENTRY_OPCODE2STRING(entry), entry,
  18330. +        (entry?entry->xid:-1LL));
  18331. +
  18332. +    switch(status) {
  18333. +    case STATUS_SUCCESS:
  18334. +        break;
  18335. +    case STATUS_USER_APC:
  18336. +    case STATUS_ALERTED:
  18337. +        DbgP("nfs41_UpcallWaitForReply: KeWaitForSingleObject() "
  18338. +            "returned status(=0x%lx), "
  18339. +            "retry waiting for '%s' entry=0x%p xid=%lld\n",
  18340. +            (long)status,
  18341. +            ENTRY_OPCODE2STRING(entry),
  18342. +            entry,
  18343. +            (entry?entry->xid:-1LL));
  18344. +        if (entry) {
  18345. +            goto retry_wait;
  18346. +        }
  18347. +        /* fall-through */
  18348. +    default:
  18349. +        ExAcquireFastMutex(&entry->lock);
  18350. +        if (entry->state == NFS41_DONE_PROCESSING) {
  18351. +            ExReleaseFastMutex(&entry->lock);
  18352. +            break;
  18353. +        }
  18354. +        DbgP("[upcall] abandoning '%s' entry=0x%p xid=%lld\n",
  18355. +            ENTRY_OPCODE2STRING(entry),
  18356. +            entry,
  18357. +            (entry?entry->xid:-1LL));
  18358. +        entry->state = NFS41_NOT_WAITING;
  18359. +        ExReleaseFastMutex(&entry->lock);
  18360. +        goto out;
  18361. +    }
  18362. +    nfs41_RemoveEntry(downcallLock, entry);
  18363. +out:
  18364. +    return status;
  18365. +}
  18366. +
  18367. +NTSTATUS nfs41_upcall(
  18368. +    IN PRX_CONTEXT RxContext)
  18369. +{
  18370. +    NTSTATUS status = STATUS_SUCCESS;
  18371. +    ULONG len = 0;
  18372. +    PLIST_ENTRY pEntry = NULL;
  18373. +
  18374. +process_upcall:
  18375. +    nfs41_RemoveFirst(upcallLock, upcall, pEntry);
  18376. +    if (pEntry) {
  18377. +        nfs41_updowncall_entry *entry;
  18378. +
  18379. +        entry = (nfs41_updowncall_entry *)CONTAINING_RECORD(pEntry,
  18380. +                    nfs41_updowncall_entry, next);
  18381. +        ExAcquireFastMutex(&entry->lock);
  18382. +        nfs41_AddEntry(downcallLock, downcall, entry);
  18383. +        status = handle_upcall(RxContext, entry, &len);
  18384. +        if (status == STATUS_SUCCESS &&
  18385. +                entry->state == NFS41_WAITING_FOR_UPCALL)
  18386. +            entry->state = NFS41_WAITING_FOR_DOWNCALL;
  18387. +        ExReleaseFastMutex(&entry->lock);
  18388. +        if (status) {
  18389. +            entry->status = status;
  18390. +            KeSetEvent(&entry->cond, 0, FALSE);
  18391. +            RxContext->InformationToReturn = 0;
  18392. +        } else
  18393. +            RxContext->InformationToReturn = len;
  18394. +    }
  18395. +    else {
  18396. +/*
  18397. + * gisburn: |NFSV41_UPCALL_RETRY_WAIT| disabled for now because it
  18398. + * causes nfsd_debug.exe to hang on <CTRL-C>
  18399. + */
  18400. +#ifdef NFSV41_UPCALL_RETRY_WAIT
  18401. +retry_wait:
  18402. +#endif /* NFSV41_UPCALL_RETRY_WAIT */
  18403. +        status = KeWaitForSingleObject(&upcallEvent, Executive, UserMode, TRUE,
  18404. +            (PLARGE_INTEGER) NULL);
  18405. +        print_wait_status(0, "[upcall]", status, NULL, NULL, 0);
  18406. +        switch (status) {
  18407. +            case STATUS_SUCCESS:
  18408. +                goto process_upcall;
  18409. +            case STATUS_USER_APC:
  18410. +            case STATUS_ALERTED:
  18411. +                DbgP("nfs41_upcall: KeWaitForSingleObject() "
  18412. +                    "returned status(=0x%lx)"
  18413. +#ifdef NFSV41_UPCALL_RETRY_WAIT
  18414. +                    ", retry waiting"
  18415. +#endif /* NFSV41_UPCALL_RETRY_WAIT */
  18416. +                    "\n",
  18417. +                    (long)status);
  18418. +#ifdef NFSV41_UPCALL_RETRY_WAIT
  18419. +                goto retry_wait;
  18420. +#else
  18421. +                /* fall-through */
  18422. +#endif /* NFSV41_UPCALL_RETRY_WAIT */
  18423. +            default:
  18424. +                DbgP("nfs41_upcall: KeWaitForSingleObject() "
  18425. +                    "returned UNEXPECTED status(=0x%lx)\n",
  18426. +                    (long)status);
  18427. +                goto out;
  18428. +        }
  18429. +    }
  18430. +out:
  18431. +    return status;
  18432. +}
  18433. +
  18434. +NTSTATUS nfs41_downcall(
  18435. +    IN PRX_CONTEXT RxContext)
  18436. +{
  18437. +    NTSTATUS status = STATUS_SUCCESS;
  18438. +    PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext;
  18439. +#ifdef DEBUG_PRINT_DOWNCALL_HEXBUF
  18440. +    ULONG in_len = LowIoContext->ParamsFor.IoCtl.InputBufferLength;
  18441. +#endif /* DEBUG_PRINT_DOWNCALL_HEXBUF */
  18442. +    unsigned char *buf = LowIoContext->ParamsFor.IoCtl.pInputBuffer;
  18443. +    PLIST_ENTRY pEntry;
  18444. +    nfs41_updowncall_entry *tmp, *cur= NULL;
  18445. +    BOOLEAN found = 0;
  18446. +
  18447. +#ifdef DEBUG_PRINT_DOWNCALL_HEXBUF
  18448. +    print_hexbuf("downcall buffer", buf, in_len);
  18449. +#endif /* DEBUG_PRINT_DOWNCALL_HEXBUF */
  18450. +
  18451. +    tmp = nfs41_downcall_allocate_updowncall_entry();
  18452. +    if (tmp == NULL) goto out;
  18453. +
  18454. +    unmarshal_nfs41_header(tmp, &buf);
  18455. +
  18456. +    ExAcquireFastMutex(&downcallLock);
  18457. +    pEntry = &downcall.head;
  18458. +    pEntry = pEntry->Flink;
  18459. +    while (pEntry != NULL) {
  18460. +        cur = (nfs41_updowncall_entry *)CONTAINING_RECORD(pEntry,
  18461. +                nfs41_updowncall_entry, next);
  18462. +        if (cur->xid == tmp->xid) {
  18463. +            found = 1;
  18464. +            break;
  18465. +        }
  18466. +        if (pEntry->Flink == &downcall.head)
  18467. +            break;
  18468. +        pEntry = pEntry->Flink;
  18469. +    }
  18470. +    ExReleaseFastMutex(&downcallLock);
  18471. +    SeStopImpersonatingClient();
  18472. +    if (!found) {
  18473. +        print_error("Didn't find xid=%lld entry\n", tmp->xid);
  18474. +        goto out_free;
  18475. +    }
  18476. +
  18477. +    ExAcquireFastMutex(&cur->lock);
  18478. +    if (cur->state == NFS41_NOT_WAITING) {
  18479. +        DbgP("[downcall] Nobody is waiting for this request!!!\n");
  18480. +        switch(cur->opcode) {
  18481. +        case NFS41_WRITE:
  18482. +        case NFS41_READ:
  18483. +            MmUnmapLockedPages(cur->buf, cur->u.ReadWrite.MdlAddress);
  18484. +            break;
  18485. +        case NFS41_DIR_QUERY:
  18486. +            MmUnmapLockedPages(cur->u.QueryFile.mdl_buf,
  18487. +                    cur->u.QueryFile.mdl);
  18488. +            IoFreeMdl(cur->u.QueryFile.mdl);
  18489. +            break;
  18490. +        case NFS41_OPEN:
  18491. +            if (cur->u.Open.EaMdl) {
  18492. +                MmUnmapLockedPages(cur->u.Open.EaBuffer,
  18493. +                        cur->u.Open.EaMdl);
  18494. +                IoFreeMdl(cur->u.Open.EaMdl);
  18495. +            }
  18496. +            break;
  18497. +        }
  18498. +        ExReleaseFastMutex(&cur->lock);
  18499. +        nfs41_RemoveEntry(downcallLock, cur);
  18500. +        nfs41_UpcallDestroy(cur);
  18501. +        status = STATUS_UNSUCCESSFUL;
  18502. +        goto out_free;
  18503. +    }
  18504. +    cur->state = NFS41_DONE_PROCESSING;
  18505. +    cur->status = tmp->status;
  18506. +    cur->errno = tmp->errno;
  18507. +    status = STATUS_SUCCESS;
  18508. +
  18509. +    if (!tmp->status) {
  18510. +        switch (tmp->opcode) {
  18511. +        case NFS41_MOUNT:
  18512. +            unmarshal_nfs41_mount(cur, &buf);
  18513. +            break;
  18514. +        case NFS41_WRITE:
  18515. +        case NFS41_READ:
  18516. +            status = unmarshal_nfs41_rw(cur, &buf);
  18517. +            break;
  18518. +        case NFS41_OPEN:
  18519. +            status = unmarshal_nfs41_open(cur, &buf);
  18520. +            break;
  18521. +        case NFS41_DIR_QUERY:
  18522. +            status = unmarshal_nfs41_dirquery(cur, &buf);
  18523. +            break;
  18524. +        case NFS41_FILE_QUERY:
  18525. +        case NFS41_FILE_QUERY_TIME_BASED_COHERENCY:
  18526. +            unmarshal_nfs41_getattr(cur, &buf);
  18527. +            break;
  18528. +        case NFS41_EA_GET:
  18529. +            unmarshal_nfs41_eaget(cur, &buf);
  18530. +            break;
  18531. +        case NFS41_SYMLINK:
  18532. +            unmarshal_nfs41_symlink(cur, &buf);
  18533. +            break;
  18534. +        case NFS41_VOLUME_QUERY:
  18535. +            unmarshal_nfs41_attrget(cur, cur->buf, &cur->buf_len, &buf);
  18536. +            break;
  18537. +        case NFS41_ACL_QUERY:
  18538. +            status = unmarshal_nfs41_getacl(cur, &buf);
  18539. +            break;
  18540. +        case NFS41_FILE_SET:
  18541. +            unmarshal_nfs41_setattr(cur, &cur->ChangeTime, &buf);
  18542. +            break;
  18543. +        case NFS41_EA_SET:
  18544. +            unmarshal_nfs41_setattr(cur, &cur->ChangeTime, &buf);
  18545. +            break;
  18546. +        case NFS41_ACL_SET:
  18547. +            unmarshal_nfs41_setattr(cur, &cur->ChangeTime, &buf);
  18548. +            break;
  18549. +        }
  18550. +    }
  18551. +    ExReleaseFastMutex(&cur->lock);
  18552. +    if (cur->async_op) {
  18553. +        switch (cur->opcode) {
  18554. +            case NFS41_WRITE:
  18555. +            case NFS41_READ:
  18556. +                if (cur->status == STATUS_SUCCESS) {
  18557. +                    cur->u.ReadWrite.rxcontext->StoredStatus =
  18558. +                        STATUS_SUCCESS;
  18559. +                    cur->u.ReadWrite.rxcontext->InformationToReturn =
  18560. +                        cur->buf_len;
  18561. +                } else {
  18562. +                    cur->u.ReadWrite.rxcontext->StoredStatus =
  18563. +                        map_readwrite_errors(cur->status);
  18564. +                    cur->u.ReadWrite.rxcontext->InformationToReturn = 0;
  18565. +                }
  18566. +                nfs41_RemoveEntry(downcallLock, cur);
  18567. +                RxLowIoCompletion(cur->u.ReadWrite.rxcontext);
  18568. +                nfs41_UpcallDestroy(cur);
  18569. +                break;
  18570. +            default:
  18571. +                print_error("##### nfs41_downcall: "
  18572. +                    "unknown async opcode=%d ####\n",
  18573. +                    (int)cur->opcode);
  18574. +                break;
  18575. +        }
  18576. +    } else
  18577. +        KeSetEvent(&cur->cond, 0, FALSE);
  18578. +
  18579. +out_free:
  18580. +    nfs41_downcall_free_updowncall_entry(tmp);
  18581. +out:
  18582. +    return status;
  18583. +}
  18584. diff --git a/sys/nfs41sys_util.c b/sys/nfs41sys_util.c
  18585. new file mode 100644
  18586. index 0000000..e0e4e8f
  18587. --- /dev/null
  18588. +++ b/sys/nfs41sys_util.c
  18589. @@ -0,0 +1,118 @@
  18590. +/* NFSv4.1 client for Windows
  18591. + * Copyright (C) 2012 The Regents of the University of Michigan
  18592. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  18593. + *
  18594. + * Olga Kornievskaia <aglo@umich.edu>
  18595. + * Casey Bodley <cbodley@umich.edu>
  18596. + * Roland Mainz <roland.mainz@nrubsig.org>
  18597. + *
  18598. + * This library is free software; you can redistribute it and/or modify it
  18599. + * under the terms of the GNU Lesser General Public License as published by
  18600. + * the Free Software Foundation; either version 2.1 of the License, or (at
  18601. + * your option) any later version.
  18602. + *
  18603. + * This library is distributed in the hope that it will be useful, but
  18604. + * without any warranty; without even the implied warranty of merchantability
  18605. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  18606. + * License for more details.
  18607. + *
  18608. + * You should have received a copy of the GNU Lesser General Public License
  18609. + * along with this library; if not, write to the Free Software Foundation,
  18610. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  18611. + */
  18612. +
  18613. +#ifndef _KERNEL_MODE
  18614. +#error module requires kernel mode
  18615. +#endif
  18616. +
  18617. +#if ((__STDC_VERSION__-0) < 201710L)
  18618. +#error Code requires ISO C17
  18619. +#endif
  18620. +
  18621. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  18622. +#if _MSC_VER >= 1900
  18623. +#if defined(_WIN64) && defined(_M_X64)
  18624. +#ifndef _AMD64_
  18625. +#define _AMD64_
  18626. +#endif
  18627. +#elif defined(_WIN32) && defined(_M_IX86)
  18628. +#ifndef _X86_
  18629. +#define _X86_
  18630. +#endif
  18631. +#elif defined(_WIN64) && defined(_M_ARM64)
  18632. +#ifndef _ARM64_
  18633. +#define _ARM64_
  18634. +#endif
  18635. +#elif defined(_WIN32) && defined(_M_ARM)
  18636. +#ifndef _ARM_
  18637. +#define _ARM_
  18638. +#endif
  18639. +#else
  18640. +#error Unsupported arch
  18641. +#endif
  18642. +#endif /* _MSC_VER >= 1900 */
  18643. +
  18644. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  18645. +#include <rx.h>
  18646. +#include <windef.h>
  18647. +#include <winerror.h>
  18648. +
  18649. +#include <Ntstrsafe.h>
  18650. +
  18651. +#include "nfs41sys_buildconfig.h"
  18652. +
  18653. +#include "nfs41_driver.h"
  18654. +#include "nfs41sys_debug.h"
  18655. +#include "nfs41_build_features.h"
  18656. +
  18657. +#include "nfs41sys_driver.h"
  18658. +#include "nfs41sys_util.h"
  18659. +
  18660. +
  18661. +BOOLEAN isFilenameTooLong(
  18662. +    PUNICODE_STRING name,
  18663. +    PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext)
  18664. +{
  18665. +    PFILE_FS_ATTRIBUTE_INFORMATION attrs = &pVNetRootContext->FsAttrs;
  18666. +    LONG len = attrs->MaximumComponentNameLength, count = 1, i;
  18667. +    PWCH p = name->Buffer;
  18668. +    for (i = 0; i < name->Length / 2; i++) {
  18669. +        if (p[0] == L'\\') count = 1;
  18670. +        else {
  18671. +            if (p[0] == L'\0') return FALSE;
  18672. +            if (count > len) return TRUE;
  18673. +            count++;
  18674. +        }
  18675. +        p++;
  18676. +    }
  18677. +    return FALSE;
  18678. +}
  18679. +
  18680. +BOOLEAN isStream(
  18681. +    PUNICODE_STRING name)
  18682. +{
  18683. +    LONG i;
  18684. +    PWCH p = name->Buffer;
  18685. +    for (i = 0; i < name->Length / 2; i++) {
  18686. +        if (p[0] == L':') return TRUE;
  18687. +        else if (p[0] == L'\0') return FALSE;
  18688. +        p++;
  18689. +    }
  18690. +    return FALSE;
  18691. +}
  18692. +
  18693. +BOOLEAN is_root_directory(
  18694. +    PRX_CONTEXT RxContext)
  18695. +{
  18696. +    __notnull PV_NET_ROOT VNetRoot = (PV_NET_ROOT)
  18697. +        RxContext->pRelevantSrvOpen->pVNetRoot;
  18698. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  18699. +        NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot);
  18700. +
  18701. +    /* calculate the root directory's length, including vnetroot prefix,
  18702. +     * mount path, and a trailing \ */
  18703. +    const USHORT RootPathLen = VNetRoot->PrefixEntry.Prefix.Length +
  18704. +            pVNetRootContext->MountPathLen + sizeof(WCHAR);
  18705. +
  18706. +    return RxContext->CurrentIrpSp->FileObject->FileName.Length <= RootPathLen;
  18707. +}
  18708. diff --git a/sys/nfs41sys_util.h b/sys/nfs41sys_util.h
  18709. new file mode 100644
  18710. index 0000000..f0f6791
  18711. --- /dev/null
  18712. +++ b/sys/nfs41sys_util.h
  18713. @@ -0,0 +1,55 @@
  18714. +/* NFSv4.1 client for Windows
  18715. + * Copyright (C) 2012 The Regents of the University of Michigan
  18716. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  18717. + *
  18718. + * Olga Kornievskaia <aglo@umich.edu>
  18719. + * Casey Bodley <cbodley@umich.edu>
  18720. + * Roland Mainz <roland.mainz@nrubsig.org>
  18721. + *
  18722. + * This library is free software; you can redistribute it and/or modify it
  18723. + * under the terms of the GNU Lesser General Public License as published by
  18724. + * the Free Software Foundation; either version 2.1 of the License, or (at
  18725. + * your option) any later version.
  18726. + *
  18727. + * This library is distributed in the hope that it will be useful, but
  18728. + * without any warranty; without even the implied warranty of merchantability
  18729. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  18730. + * License for more details.
  18731. + *
  18732. + * You should have received a copy of the GNU Lesser General Public License
  18733. + * along with this library; if not, write to the Free Software Foundation,
  18734. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  18735. + */
  18736. +
  18737. +#ifndef _NFS41SYS_UTIL_H_
  18738. +#define _NFS41SYS_UTIL_H_ 1
  18739. +
  18740. +static INLINE BOOL AnsiStrEq(
  18741. +    IN const ANSI_STRING *lhs,
  18742. +    IN const CHAR *rhs,
  18743. +    IN const UCHAR rhs_len)
  18744. +{
  18745. +    return lhs->Length == rhs_len &&
  18746. +        RtlCompareMemory(lhs->Buffer, rhs, rhs_len) == rhs_len;
  18747. +}
  18748. +
  18749. +/* convert strings from unicode -> ansi during marshalling to
  18750. + * save space in the upcall buffers and avoid extra copies */
  18751. +static INLINE ULONG length_as_utf8(
  18752. +    PCUNICODE_STRING str)
  18753. +{
  18754. +    ULONG ActualCount = 0;
  18755. +    RtlUnicodeToUTF8N(NULL, 0xffff, &ActualCount, str->Buffer, str->Length);
  18756. +    return sizeof(str->MaximumLength) + ActualCount + sizeof(UNICODE_NULL);
  18757. +}
  18758. +
  18759. +/* Prototypes */
  18760. +BOOLEAN isFilenameTooLong(
  18761. +    PUNICODE_STRING name,
  18762. +    PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext);
  18763. +BOOLEAN isStream(
  18764. +    PUNICODE_STRING name);
  18765. +BOOLEAN is_root_directory(
  18766. +    PRX_CONTEXT RxContext);
  18767. +
  18768. +#endif /* !_NFS41SYS_UTIL_H_ */
  18769. diff --git a/sys/nfs41sys_volinfo.c b/sys/nfs41sys_volinfo.c
  18770. new file mode 100644
  18771. index 0000000..253e050
  18772. --- /dev/null
  18773. +++ b/sys/nfs41sys_volinfo.c
  18774. @@ -0,0 +1,310 @@
  18775. +/* NFSv4.1 client for Windows
  18776. + * Copyright (C) 2012 The Regents of the University of Michigan
  18777. + * Copyright (C) 2023-2024 Roland Mainz <roland.mainz@nrubsig.org>
  18778. + *
  18779. + * Olga Kornievskaia <aglo@umich.edu>
  18780. + * Casey Bodley <cbodley@umich.edu>
  18781. + * Roland Mainz <roland.mainz@nrubsig.org>
  18782. + *
  18783. + * This library is free software; you can redistribute it and/or modify it
  18784. + * under the terms of the GNU Lesser General Public License as published by
  18785. + * the Free Software Foundation; either version 2.1 of the License, or (at
  18786. + * your option) any later version.
  18787. + *
  18788. + * This library is distributed in the hope that it will be useful, but
  18789. + * without any warranty; without even the implied warranty of merchantability
  18790. + * or fitness for a particular purpose.  See the GNU Lesser General Public
  18791. + * License for more details.
  18792. + *
  18793. + * You should have received a copy of the GNU Lesser General Public License
  18794. + * along with this library; if not, write to the Free Software Foundation,
  18795. + * Inc., 51 Franklin Street, Fifth Floor, Boston, MA
  18796. + */
  18797. +
  18798. +#ifndef _KERNEL_MODE
  18799. +#error module requires kernel mode
  18800. +#endif
  18801. +
  18802. +#if ((__STDC_VERSION__-0) < 201710L)
  18803. +#error Code requires ISO C17
  18804. +#endif
  18805. +
  18806. +/* FIXME: Why does VS22 need this, but not VC19 ? */
  18807. +#if _MSC_VER >= 1900
  18808. +#if defined(_WIN64) && defined(_M_X64)
  18809. +#ifndef _AMD64_
  18810. +#define _AMD64_
  18811. +#endif
  18812. +#elif defined(_WIN32) && defined(_M_IX86)
  18813. +#ifndef _X86_
  18814. +#define _X86_
  18815. +#endif
  18816. +#elif defined(_WIN64) && defined(_M_ARM64)
  18817. +#ifndef _ARM64_
  18818. +#define _ARM64_
  18819. +#endif
  18820. +#elif defined(_WIN32) && defined(_M_ARM)
  18821. +#ifndef _ARM_
  18822. +#define _ARM_
  18823. +#endif
  18824. +#else
  18825. +#error Unsupported arch
  18826. +#endif
  18827. +#endif /* _MSC_VER >= 1900 */
  18828. +
  18829. +#define MINIRDR__NAME "Value is ignored, only fact of definition"
  18830. +#include <rx.h>
  18831. +#include <windef.h>
  18832. +#include <winerror.h>
  18833. +
  18834. +#include <Ntstrsafe.h>
  18835. +
  18836. +#include "nfs41sys_buildconfig.h"
  18837. +
  18838. +#include "nfs41_driver.h"
  18839. +#include "nfs41sys_debug.h"
  18840. +#include "nfs41_build_features.h"
  18841. +
  18842. +#include "nfs41sys_driver.h"
  18843. +#include "nfs41sys_util.h"
  18844. +
  18845. +
  18846. +NTSTATUS marshal_nfs41_volume(
  18847. +    nfs41_updowncall_entry *entry,
  18848. +    unsigned char *buf,
  18849. +    ULONG buf_len,
  18850. +    ULONG *len)
  18851. +{
  18852. +    NTSTATUS status = STATUS_SUCCESS;
  18853. +    ULONG header_len = 0;
  18854. +    unsigned char *tmp = buf;
  18855. +
  18856. +    status = marshal_nfs41_header(entry, tmp, buf_len, len);
  18857. +    if (status) goto out;
  18858. +    else tmp += *len;
  18859. +
  18860. +    header_len = *len + sizeof(FS_INFORMATION_CLASS);
  18861. +    if (header_len > buf_len) {
  18862. +        status = STATUS_INSUFFICIENT_RESOURCES;
  18863. +        goto out;
  18864. +    }
  18865. +
  18866. +    RtlCopyMemory(tmp, &entry->u.Volume.query, sizeof(FS_INFORMATION_CLASS));
  18867. +    *len = header_len;
  18868. +
  18869. +#ifdef DEBUG_MARSHAL_DETAIL
  18870. +    DbgP("marshal_nfs41_volume: class=%d\n", entry->u.Volume.query);
  18871. +#endif
  18872. +out:
  18873. +    return status;
  18874. +}
  18875. +
  18876. +void unmarshal_nfs41_attrget(
  18877. +    nfs41_updowncall_entry *cur,
  18878. +    PVOID attr_value,
  18879. +    ULONG *attr_len,
  18880. +    unsigned char **buf)
  18881. +{
  18882. +    ULONG buf_len;
  18883. +    RtlCopyMemory(&buf_len, *buf, sizeof(ULONG));
  18884. +    if (buf_len > *attr_len) {
  18885. +        cur->status = STATUS_BUFFER_TOO_SMALL;
  18886. +        return;
  18887. +    }
  18888. +    *buf += sizeof(ULONG);
  18889. +    *attr_len = buf_len;
  18890. +    RtlCopyMemory(attr_value, *buf, buf_len);
  18891. +    *buf += buf_len;
  18892. +}
  18893. +
  18894. +static void print_queryvolume_args(
  18895. +    PRX_CONTEXT RxContext)
  18896. +{
  18897. +    print_debug_header(RxContext);
  18898. +    DbgP("FileName='%wZ', InfoClass = '%s' BufferLen = %d\n",
  18899. +        GET_ALREADY_PREFIXED_NAME_FROM_CONTEXT(RxContext),
  18900. +        print_fs_information_class(RxContext->Info.FileInformationClass),
  18901. +        RxContext->Info.LengthRemaining);
  18902. +}
  18903. +
  18904. +NTSTATUS map_volume_errors(
  18905. +    DWORD status)
  18906. +{
  18907. +    switch (status) {
  18908. +    case ERROR_ACCESS_DENIED:       return STATUS_ACCESS_DENIED;
  18909. +    case ERROR_VC_DISCONNECTED:     return STATUS_CONNECTION_DISCONNECTED;
  18910. +    case ERROR_NETNAME_DELETED:     return STATUS_NETWORK_NAME_DELETED;
  18911. +    case ERROR_INVALID_PARAMETER:   return STATUS_INVALID_PARAMETER;
  18912. +    case ERROR_OUTOFMEMORY:         return STATUS_INSUFFICIENT_RESOURCES;
  18913. +    case ERROR_INTERNAL_ERROR:      return STATUS_INTERNAL_ERROR;
  18914. +    default:
  18915. +        print_error("map_volume_errors: "
  18916. +            "failed to map windows ERROR_0x%x to NTSTATUS; "
  18917. +            "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status);
  18918. +    case ERROR_BAD_NET_RESP:        return STATUS_INVALID_NETWORK_RESPONSE;
  18919. +    }
  18920. +}
  18921. +
  18922. +void nfs41_create_volume_info(PFILE_FS_VOLUME_INFORMATION pVolInfo, DWORD *len)
  18923. +{
  18924. +    DECLARE_CONST_UNICODE_STRING(VolName, VOL_NAME);
  18925. +
  18926. +    RtlZeroMemory(pVolInfo, sizeof(FILE_FS_VOLUME_INFORMATION));
  18927. +    pVolInfo->VolumeSerialNumber = 0xBABAFACE;
  18928. +    pVolInfo->VolumeLabelLength = VolName.Length;
  18929. +    RtlCopyMemory(&pVolInfo->VolumeLabel[0], (PVOID)VolName.Buffer,
  18930. +        VolName.MaximumLength);
  18931. +    *len = sizeof(FILE_FS_VOLUME_INFORMATION) + VolName.Length;
  18932. +}
  18933. +
  18934. +NTSTATUS nfs41_QueryVolumeInformation(
  18935. +    IN OUT PRX_CONTEXT RxContext)
  18936. +{
  18937. +    NTSTATUS status = STATUS_INVALID_PARAMETER;
  18938. +    nfs41_updowncall_entry *entry;
  18939. +    ULONG RemainingLength = RxContext->Info.LengthRemaining, SizeUsed;
  18940. +    FS_INFORMATION_CLASS InfoClass = RxContext->Info.FsInformationClass;
  18941. +    __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen;
  18942. +    __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext =
  18943. +        NFS41GetVNetRootExtension(SrvOpen->pVNetRoot);
  18944. +    __notnull PNFS41_NETROOT_EXTENSION pNetRootContext =
  18945. +        NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot);
  18946. +    __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx);
  18947. +    NFS41GetDeviceExtension(RxContext, DevExt);
  18948. +
  18949. +#ifdef ENABLE_TIMINGS
  18950. +    LARGE_INTEGER t1, t2;
  18951. +    t1 = KeQueryPerformanceCounter(NULL);
  18952. +#endif
  18953. +
  18954. +#ifdef DEBUG_VOLUME_QUERY
  18955. +    DbgEn();
  18956. +    print_queryvolume_args(RxContext);
  18957. +#endif
  18958. +
  18959. +    status = check_nfs41_dirquery_args(RxContext);
  18960. +    if (status) goto out;
  18961. +
  18962. +    RtlZeroMemory(RxContext->Info.Buffer, RxContext->Info.LengthRemaining);
  18963. +
  18964. +    switch (InfoClass) {
  18965. +    case FileFsVolumeInformation:
  18966. +        if ((ULONG)RxContext->Info.LengthRemaining >= DevExt->VolAttrsLen) {
  18967. +            RtlCopyMemory(RxContext->Info.Buffer, DevExt->VolAttrs,
  18968. +                DevExt->VolAttrsLen);
  18969. +            RxContext->Info.LengthRemaining -= DevExt->VolAttrsLen;
  18970. +            status = STATUS_SUCCESS;
  18971. +        } else {
  18972. +            RtlCopyMemory(RxContext->Info.Buffer, DevExt->VolAttrs,
  18973. +                RxContext->Info.LengthRemaining);
  18974. +            status = STATUS_BUFFER_OVERFLOW;
  18975. +        }
  18976. +        goto out;
  18977. +    case FileFsDeviceInformation:
  18978. +    {
  18979. +        PFILE_FS_DEVICE_INFORMATION pDevInfo = RxContext->Info.Buffer;
  18980. +
  18981. +        SizeUsed = sizeof(FILE_FS_DEVICE_INFORMATION);
  18982. +        if (RemainingLength < SizeUsed) {
  18983. +            status = STATUS_BUFFER_TOO_SMALL;
  18984. +            RxContext->InformationToReturn = SizeUsed;
  18985. +            goto out;
  18986. +        }
  18987. +        pDevInfo->DeviceType = RxContext->pFcb->pNetRoot->DeviceType;
  18988. +        pDevInfo->Characteristics = FILE_REMOTE_DEVICE | FILE_DEVICE_IS_MOUNTED;
  18989. +        RxContext->Info.LengthRemaining -= SizeUsed;
  18990. +        status = STATUS_SUCCESS;
  18991. +        goto out;
  18992. +    }
  18993. +    case FileAccessInformation:
  18994. +        status = STATUS_NOT_SUPPORTED;
  18995. +        goto out;
  18996. +
  18997. +    case FileFsAttributeInformation:
  18998. +        if (RxContext->Info.LengthRemaining < FS_ATTR_LEN) {
  18999. +            RxContext->InformationToReturn = FS_ATTR_LEN;
  19000. +            status = STATUS_BUFFER_TOO_SMALL;
  19001. +            goto out;
  19002. +        }
  19003. +
  19004. +        /* on attribute queries for the root directory,
  19005. +         * use cached volume attributes from mount */
  19006. +        if (is_root_directory(RxContext)) {
  19007. +            PFILE_FS_ATTRIBUTE_INFORMATION attrs =
  19008. +                (PFILE_FS_ATTRIBUTE_INFORMATION)RxContext->Info.Buffer;
  19009. +            DECLARE_CONST_UNICODE_STRING(FsName, FS_NAME);
  19010. +
  19011. +            RtlCopyMemory(attrs, &pVNetRootContext->FsAttrs,
  19012. +                sizeof(pVNetRootContext->FsAttrs));
  19013. +
  19014. +            /* fill in the FileSystemName */
  19015. +            RtlCopyMemory(attrs->FileSystemName, FsName.Buffer,
  19016. +                FsName.MaximumLength); /* 'MaximumLength' to include null */
  19017. +            attrs->FileSystemNameLength = FsName.Length;
  19018. +
  19019. +            RxContext->Info.LengthRemaining -= FS_ATTR_LEN;
  19020. +            goto out;
  19021. +        }
  19022. +        /* else fall through and send the upcall */
  19023. +    case FileFsSizeInformation:
  19024. +    case FileFsFullSizeInformation:
  19025. +        break;
  19026. +
  19027. +    default:
  19028. +        print_error("nfs41_QueryVolumeInformation: unhandled class %d\n", InfoClass);
  19029. +        status = STATUS_NOT_SUPPORTED;
  19030. +        goto out;
  19031. +    }
  19032. +    status = nfs41_UpcallCreate(NFS41_VOLUME_QUERY, &nfs41_fobx->sec_ctx,
  19033. +        pVNetRootContext->session, nfs41_fobx->nfs41_open_state,
  19034. +        pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry);
  19035. +    if (status) goto out;
  19036. +
  19037. +    entry->u.Volume.query = InfoClass;
  19038. +    entry->buf = RxContext->Info.Buffer;
  19039. +    entry->buf_len = RxContext->Info.LengthRemaining;
  19040. +
  19041. +    status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout);
  19042. +    if (status) goto out;
  19043. +
  19044. +    if (entry->status == STATUS_BUFFER_TOO_SMALL) {
  19045. +        RxContext->InformationToReturn = entry->buf_len;
  19046. +        status = STATUS_BUFFER_TOO_SMALL;
  19047. +    } else if (entry->status == STATUS_SUCCESS) {
  19048. +        if (InfoClass == FileFsAttributeInformation) {
  19049. +            /* fill in the FileSystemName */
  19050. +            PFILE_FS_ATTRIBUTE_INFORMATION attrs =
  19051. +                (PFILE_FS_ATTRIBUTE_INFORMATION)RxContext->Info.Buffer;
  19052. +            DECLARE_CONST_UNICODE_STRING(FsName, FS_NAME);
  19053. +
  19054. +            RtlCopyMemory(attrs->FileSystemName, FsName.Buffer,
  19055. +                FsName.MaximumLength); /* 'MaximumLength' to include null */
  19056. +            attrs->FileSystemNameLength = FsName.Length;
  19057. +
  19058. +            entry->buf_len = FS_ATTR_LEN;
  19059. +        }
  19060. +#ifdef ENABLE_TIMINGS
  19061. +        InterlockedIncrement(&volume.sops);
  19062. +        InterlockedAdd64(&volume.size, entry->u.Volume.buf_len);
  19063. +#endif
  19064. +        RxContext->Info.LengthRemaining -= entry->buf_len;
  19065. +        status = STATUS_SUCCESS;
  19066. +    } else {
  19067. +        status = map_volume_errors(entry->status);
  19068. +    }
  19069. +    nfs41_UpcallDestroy(entry);
  19070. +out:
  19071. +#ifdef ENABLE_TIMINGS
  19072. +    t2 = KeQueryPerformanceCounter(NULL);
  19073. +    InterlockedIncrement(&volume.tops);
  19074. +    InterlockedAdd64(&volume.ticks, t2.QuadPart - t1.QuadPart);
  19075. +#ifdef ENABLE_INDV_TIMINGS
  19076. +    DbgP("nfs41_QueryVolumeInformation delta = %d op=%d sum=%d\n",
  19077. +        t2.QuadPart - t1.QuadPart, volume.tops, volume.ticks);
  19078. +#endif
  19079. +#endif
  19080. +#ifdef DEBUG_VOLUME_QUERY
  19081. +    DbgEx();
  19082. +#endif
  19083. +    return status;
  19084. +}
  19085. --
  19086. 2.45.1

Submit a correction or amendment below (click here to make a fresh posting)
After submitting an amendment, you'll be able to view the differences between the old and new posts easily.

Syntax highlighting:

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




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