1 /* NFSv4.1 client for Windows 2 * Copyright © 2012 The Regents of the University of Michigan 3 * 4 * Olga Kornievskaia <aglo@umich.edu> 5 * Casey Bodley <cbodley@umich.edu> 6 * 7 * This library is free software; you can redistribute it and/or modify it 8 * under the terms of the GNU Lesser General Public License as published by 9 * the Free Software Foundation; either version 2.1 of the License, or (at 10 * your option) any later version. 11 * 12 * This library is distributed in the hope that it will be useful, but 13 * without any warranty; without even the implied warranty of merchantability 14 * or fitness for a particular purpose. See the GNU Lesser General Public 15 * License for more details. 16 * 17 * You should have received a copy of the GNU Lesser General Public License 18 * along with this library; if not, write to the Free Software Foundation, 19 * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 20 */ 21 22 #define MINIRDR__NAME "Value is ignored, only fact of definition" 23 #include <rx.h> 24 #include <windef.h> 25 #include <winerror.h> 26 27 #include <ntstrsafe.h> 28 29 #ifdef __REACTOS__ 30 #include <pseh/pseh2.h> 31 #endif 32 33 #include "nfs41_driver.h" 34 #include "nfs41_np.h" 35 #include "nfs41_debug.h" 36 37 #if defined(__REACTOS__) && (NTDDI_VERSION < NTDDI_WIN7) 38 NTSTATUS NTAPI RtlUnicodeToUTF8N(CHAR *utf8_dest, ULONG utf8_bytes_max, 39 ULONG *utf8_bytes_written, 40 const WCHAR *uni_src, ULONG uni_bytes); 41 NTSTATUS NTAPI RtlUTF8ToUnicodeN(WCHAR *uni_dest, ULONG uni_bytes_max, 42 ULONG *uni_bytes_written, 43 const CHAR *utf8_src, ULONG utf8_bytes); 44 #endif /* defined(__REACTOS__) && (NTDDI_VERSION < NTDDI_WIN7) */ 45 46 #define USE_MOUNT_SEC_CONTEXT 47 48 /* debugging printout defines */ 49 //#define DEBUG_FSDDISPATCH 50 #define DEBUG_MARSHAL_HEADER 51 #define DEBUG_MARSHAL_DETAIL 52 //#define DEBUG_OPEN 53 //#define DEBUG_CLOSE 54 //#define DEBUG_CACHE 55 #define DEBUG_INVALIDATE_CACHE 56 //#define DEBUG_READ 57 //#define DEBUG_WRITE 58 //#define DEBUG_DIR_QUERY 59 //#define DEBUG_FILE_QUERY 60 //#define DEBUG_FILE_SET 61 //#define DEBUG_ACL_QUERY 62 //#define DEBUG_ACL_SET 63 //#define DEBUG_EA_QUERY 64 //#define DEBUG_EA_SET 65 //#define DEBUG_LOCK 66 //#define DEBUG_MISC 67 #define DEBUG_TIME_BASED_COHERENCY 68 #define DEBUG_MOUNT 69 //#define DEBUG_VOLUME_QUERY 70 71 //#define ENABLE_TIMINGS 72 //#define ENABLE_INDV_TIMINGS 73 #ifdef ENABLE_TIMINGS 74 typedef struct __nfs41_timings { 75 LONG tops, sops; 76 LONGLONG ticks, size; 77 } nfs41_timings; 78 79 nfs41_timings lookup, readdir, open, close, getattr, setattr, getacl, setacl, volume, 80 read, write, lock, unlock, setexattr, getexattr; 81 #endif 82 DRIVER_INITIALIZE DriverEntry; 83 DRIVER_UNLOAD nfs41_driver_unload; 84 DRIVER_DISPATCH ( nfs41_FsdDispatch ); 85 86 struct _MINIRDR_DISPATCH nfs41_ops; 87 PRDBSS_DEVICE_OBJECT nfs41_dev; 88 89 #define DISABLE_CACHING 0 90 #define ENABLE_READ_CACHING 1 91 #define ENABLE_WRITE_CACHING 2 92 #define ENABLE_READWRITE_CACHING 3 93 94 #define NFS41_MM_POOLTAG ('nfs4') 95 #define NFS41_MM_POOLTAG_ACL ('acls') 96 #define NFS41_MM_POOLTAG_MOUNT ('mnts') 97 #define NFS41_MM_POOLTAG_OPEN ('open') 98 #define NFS41_MM_POOLTAG_UP ('upca') 99 #define NFS41_MM_POOLTAG_DOWN ('down') 100 101 KEVENT upcallEvent; 102 FAST_MUTEX upcallLock, downcallLock, fcblistLock; 103 FAST_MUTEX xidLock; 104 FAST_MUTEX openOwnerLock; 105 106 LONGLONG xid = 0; 107 LONG open_owner_id = 1; 108 109 #define DECLARE_CONST_ANSI_STRING(_var, _string) \ 110 const CHAR _var ## _buffer[] = _string; \ 111 const ANSI_STRING _var = { sizeof(_string) - sizeof(CHAR), \ 112 sizeof(_string), (PCH) _var ## _buffer } 113 #define RELATIVE(wait) (-(wait)) 114 #define NANOSECONDS(nanos) (((signed __int64)(nanos)) / 100L) 115 #define MICROSECONDS(micros) (((signed __int64)(micros)) * NANOSECONDS(1000L)) 116 #define MILLISECONDS(milli) (((signed __int64)(milli)) * MICROSECONDS(1000L)) 117 #define SECONDS(seconds) (((signed __int64)(seconds)) * MILLISECONDS(1000L)) 118 119 DECLARE_CONST_ANSI_STRING(NfsV3Attributes, "NfsV3Attributes"); 120 DECLARE_CONST_ANSI_STRING(NfsSymlinkTargetName, "NfsSymlinkTargetName"); 121 DECLARE_CONST_ANSI_STRING(NfsActOnLink, "NfsActOnLink"); 122 123 INLINE BOOL AnsiStrEq( 124 IN const ANSI_STRING *lhs, 125 IN const CHAR *rhs, 126 IN const UCHAR rhs_len) 127 { 128 return lhs->Length == rhs_len && 129 RtlCompareMemory(lhs->Buffer, rhs, rhs_len) == rhs_len; 130 } 131 132 typedef struct _nfs3_attrs { 133 DWORD type, mode, nlink, uid, gid, filler1; 134 LARGE_INTEGER size, used; 135 struct { 136 DWORD specdata1; 137 DWORD specdata2; 138 } rdev; 139 LONGLONG fsid, fileid; 140 LONGLONG atime, mtime, ctime; 141 } nfs3_attrs; 142 LARGE_INTEGER unix_time_diff; //needed to convert windows time to unix 143 144 enum ftype3 { 145 NF3REG = 1, 146 NF3DIR, 147 NF3BLK, 148 NF3CHR, 149 NF3LNK, 150 NF3SOCK, 151 NF3FIFO 152 }; 153 154 typedef enum _nfs41_updowncall_state { 155 NFS41_WAITING_FOR_UPCALL, 156 NFS41_WAITING_FOR_DOWNCALL, 157 NFS41_DONE_PROCESSING, 158 NFS41_NOT_WAITING 159 } nfs41_updowncall_state; 160 161 #ifdef __REACTOS__ 162 #undef _errno 163 #undef errno 164 #endif 165 166 typedef struct _updowncall_entry { 167 DWORD version; 168 LONGLONG xid; 169 DWORD opcode; 170 NTSTATUS status; 171 nfs41_updowncall_state state; 172 FAST_MUTEX lock; 173 LIST_ENTRY next; 174 KEVENT cond; 175 DWORD errno; 176 BOOLEAN async_op; 177 SECURITY_CLIENT_CONTEXT sec_ctx; 178 PSECURITY_CLIENT_CONTEXT psec_ctx; 179 HANDLE open_state; 180 HANDLE session; 181 PUNICODE_STRING filename; 182 PVOID buf; 183 ULONG buf_len; 184 ULONGLONG ChangeTime; 185 union { 186 struct { 187 PUNICODE_STRING srv_name; 188 PUNICODE_STRING root; 189 PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs; 190 DWORD sec_flavor; 191 DWORD rsize; 192 DWORD wsize; 193 DWORD lease_time; 194 } Mount; 195 struct { 196 PMDL MdlAddress; 197 ULONGLONG offset; 198 PRX_CONTEXT rxcontext; 199 } ReadWrite; 200 struct { 201 LONGLONG offset; 202 LONGLONG length; 203 BOOLEAN exclusive; 204 BOOLEAN blocking; 205 } Lock; 206 struct { 207 ULONG count; 208 LOWIO_LOCK_LIST locks; 209 } Unlock; 210 struct { 211 FILE_BASIC_INFORMATION binfo; 212 FILE_STANDARD_INFORMATION sinfo; 213 UNICODE_STRING symlink; 214 ULONG access_mask; 215 ULONG access_mode; 216 ULONG attrs; 217 ULONG copts; 218 ULONG disp; 219 ULONG cattrs; 220 LONG open_owner_id; 221 DWORD mode; 222 HANDLE srv_open; 223 DWORD deleg_type; 224 BOOLEAN symlink_embedded; 225 PMDL EaMdl; 226 PVOID EaBuffer; 227 } Open; 228 struct { 229 HANDLE srv_open; 230 BOOLEAN remove; 231 BOOLEAN renamed; 232 } Close; 233 struct { 234 PUNICODE_STRING filter; 235 FILE_INFORMATION_CLASS InfoClass; 236 BOOLEAN restart_scan; 237 BOOLEAN return_single; 238 BOOLEAN initial_query; 239 PMDL mdl; 240 PVOID mdl_buf; 241 } QueryFile; 242 struct { 243 FILE_INFORMATION_CLASS InfoClass; 244 } SetFile; 245 struct { 246 DWORD mode; 247 } SetEa; 248 struct { 249 PVOID EaList; 250 ULONG EaListLength; 251 ULONG Overflow; 252 ULONG EaIndex; 253 BOOLEAN ReturnSingleEntry; 254 BOOLEAN RestartScan; 255 } QueryEa; 256 struct { 257 PUNICODE_STRING target; 258 BOOLEAN set; 259 } Symlink; 260 struct { 261 FS_INFORMATION_CLASS query; 262 } Volume; 263 struct { 264 SECURITY_INFORMATION query; 265 } Acl; 266 } u; 267 268 } nfs41_updowncall_entry; 269 270 typedef struct _updowncall_list { 271 LIST_ENTRY head; 272 } nfs41_updowncall_list; 273 nfs41_updowncall_list upcall, downcall; 274 275 typedef struct _nfs41_mount_entry { 276 LIST_ENTRY next; 277 LUID login_id; 278 HANDLE authsys_session; 279 HANDLE gss_session; 280 HANDLE gssi_session; 281 HANDLE gssp_session; 282 } nfs41_mount_entry; 283 284 typedef struct _nfs41_mount_list { 285 LIST_ENTRY head; 286 } nfs41_mount_list; 287 288 #define nfs41_AddEntry(lock,list,pEntry) \ 289 ExAcquireFastMutex(&lock); \ 290 InsertTailList(&(list).head, &(pEntry)->next); \ 291 ExReleaseFastMutex(&lock); 292 #define nfs41_RemoveFirst(lock,list,pEntry) \ 293 ExAcquireFastMutex(&lock); \ 294 pEntry = (IsListEmpty(&(list).head) \ 295 ? NULL \ 296 : RemoveHeadList(&(list).head)); \ 297 ExReleaseFastMutex(&lock); 298 #define nfs41_RemoveEntry(lock,pEntry) \ 299 ExAcquireFastMutex(&lock); \ 300 RemoveEntryList(&pEntry->next); \ 301 ExReleaseFastMutex(&lock); 302 #define nfs41_IsListEmpty(lock,list,flag) \ 303 ExAcquireFastMutex(&lock); \ 304 flag = IsListEmpty(&(list).head); \ 305 ExReleaseFastMutex(&lock); 306 #define nfs41_GetFirstEntry(lock,list,pEntry) \ 307 ExAcquireFastMutex(&lock); \ 308 pEntry = (IsListEmpty(&(list).head) \ 309 ? NULL \ 310 : (nfs41_updowncall_entry *) \ 311 (CONTAINING_RECORD((list).head.Flink, \ 312 nfs41_updowncall_entry, \ 313 next))); \ 314 ExReleaseFastMutex(&lock); 315 #define nfs41_GetFirstMountEntry(lock,list,pEntry) \ 316 ExAcquireFastMutex(&lock); \ 317 pEntry = (IsListEmpty(&(list).head) \ 318 ? NULL \ 319 : (nfs41_mount_entry *) \ 320 (CONTAINING_RECORD((list).head.Flink, \ 321 nfs41_mount_entry, \ 322 next))); \ 323 ExReleaseFastMutex(&lock); 324 325 /* In order to cooperate with other network providers, 326 * we only claim paths of the format '\\server\nfs4\path' */ 327 DECLARE_CONST_UNICODE_STRING(NfsPrefix, L"\\nfs4"); 328 DECLARE_CONST_UNICODE_STRING(AUTH_SYS_NAME, L"sys"); 329 DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5_NAME, L"krb5"); 330 DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5I_NAME, L"krb5i"); 331 DECLARE_CONST_UNICODE_STRING(AUTHGSS_KRB5P_NAME, L"krb5p"); 332 DECLARE_CONST_UNICODE_STRING(SLASH, L"\\"); 333 DECLARE_CONST_UNICODE_STRING(EMPTY_STRING, L""); 334 335 #define SERVER_NAME_BUFFER_SIZE 1024 336 #define MOUNT_CONFIG_RW_SIZE_MIN 1024 337 #define MOUNT_CONFIG_RW_SIZE_DEFAULT 1048576 338 #define MOUNT_CONFIG_RW_SIZE_MAX 1048576 339 #define MAX_SEC_FLAVOR_LEN 12 340 #define UPCALL_TIMEOUT_DEFAULT 50 /* in seconds */ 341 342 typedef struct _NFS41_MOUNT_CONFIG { 343 DWORD ReadSize; 344 DWORD WriteSize; 345 BOOLEAN ReadOnly; 346 BOOLEAN write_thru; 347 BOOLEAN nocache; 348 WCHAR srv_buffer[SERVER_NAME_BUFFER_SIZE]; 349 UNICODE_STRING SrvName; 350 WCHAR mntpt_buffer[MAX_PATH]; 351 UNICODE_STRING MntPt; 352 WCHAR sec_flavor[MAX_SEC_FLAVOR_LEN]; 353 UNICODE_STRING SecFlavor; 354 DWORD timeout; 355 } NFS41_MOUNT_CONFIG, *PNFS41_MOUNT_CONFIG; 356 357 typedef struct _NFS41_NETROOT_EXTENSION { 358 NODE_TYPE_CODE NodeTypeCode; 359 NODE_BYTE_SIZE NodeByteSize; 360 DWORD nfs41d_version; 361 BOOLEAN mounts_init; 362 FAST_MUTEX mountLock; 363 nfs41_mount_list mounts; 364 } NFS41_NETROOT_EXTENSION, *PNFS41_NETROOT_EXTENSION; 365 #define NFS41GetNetRootExtension(pNetRoot) \ 366 (((pNetRoot) == NULL) ? NULL : \ 367 (PNFS41_NETROOT_EXTENSION)((pNetRoot)->Context)) 368 369 /* FileSystemName as reported by FileFsAttributeInfo query */ 370 #define FS_NAME L"NFS" 371 #define FS_NAME_LEN (sizeof(FS_NAME) - sizeof(WCHAR)) 372 #define FS_ATTR_LEN (sizeof(FILE_FS_ATTRIBUTE_INFORMATION) + FS_NAME_LEN) 373 374 /* FileSystemName as reported by FileFsAttributeInfo query */ 375 #define VOL_NAME L"PnfsVolume" 376 #define VOL_NAME_LEN (sizeof(VOL_NAME) - sizeof(WCHAR)) 377 #define VOL_ATTR_LEN (sizeof(FILE_FS_VOLUME_INFORMATION) + VOL_NAME_LEN) 378 379 typedef struct _NFS41_V_NET_ROOT_EXTENSION { 380 NODE_TYPE_CODE NodeTypeCode; 381 NODE_BYTE_SIZE NodeByteSize; 382 HANDLE session; 383 FILE_FS_ATTRIBUTE_INFORMATION FsAttrs; 384 DWORD sec_flavor; 385 DWORD timeout; 386 USHORT MountPathLen; 387 BOOLEAN read_only; 388 BOOLEAN write_thru; 389 BOOLEAN nocache; 390 #define STORE_MOUNT_SEC_CONTEXT 391 #ifdef STORE_MOUNT_SEC_CONTEXT 392 SECURITY_CLIENT_CONTEXT mount_sec_ctx; 393 #endif 394 } NFS41_V_NET_ROOT_EXTENSION, *PNFS41_V_NET_ROOT_EXTENSION; 395 #define NFS41GetVNetRootExtension(pVNetRoot) \ 396 (((pVNetRoot) == NULL) ? NULL : \ 397 (PNFS41_V_NET_ROOT_EXTENSION)((pVNetRoot)->Context)) 398 399 typedef struct _NFS41_FCB { 400 NODE_TYPE_CODE NodeTypeCode; 401 NODE_BYTE_SIZE NodeByteSize; 402 FILE_BASIC_INFORMATION BasicInfo; 403 FILE_STANDARD_INFORMATION StandardInfo; 404 BOOLEAN Renamed; 405 BOOLEAN DeletePending; 406 DWORD mode; 407 ULONGLONG changeattr; 408 } NFS41_FCB, *PNFS41_FCB; 409 #define NFS41GetFcbExtension(pFcb) \ 410 (((pFcb) == NULL) ? NULL : (PNFS41_FCB)((pFcb)->Context)) 411 412 typedef struct _NFS41_FOBX { 413 NODE_TYPE_CODE NodeTypeCode; 414 NODE_BYTE_SIZE NodeByteSize; 415 416 HANDLE nfs41_open_state; 417 SECURITY_CLIENT_CONTEXT sec_ctx; 418 PVOID acl; 419 DWORD acl_len; 420 LARGE_INTEGER time; 421 DWORD deleg_type; 422 BOOLEAN write_thru; 423 BOOLEAN nocache; 424 } NFS41_FOBX, *PNFS41_FOBX; 425 #define NFS41GetFobxExtension(pFobx) \ 426 (((pFobx) == NULL) ? NULL : (PNFS41_FOBX)((pFobx)->Context)) 427 428 typedef struct _NFS41_SERVER_ENTRY { 429 PMRX_SRV_CALL pRdbssSrvCall; 430 WCHAR NameBuffer[SERVER_NAME_BUFFER_SIZE]; 431 UNICODE_STRING Name; // the server name. 432 } NFS41_SERVER_ENTRY, *PNFS41_SERVER_ENTRY; 433 434 typedef struct _NFS41_DEVICE_EXTENSION { 435 NODE_TYPE_CODE NodeTypeCode; 436 NODE_BYTE_SIZE NodeByteSize; 437 PRDBSS_DEVICE_OBJECT DeviceObject; 438 ULONG ActiveNodes; 439 HANDLE SharedMemorySection; 440 DWORD nfs41d_version; 441 BYTE VolAttrs[VOL_ATTR_LEN]; 442 DWORD VolAttrsLen; 443 HANDLE openlistHandle; 444 } NFS41_DEVICE_EXTENSION, *PNFS41_DEVICE_EXTENSION; 445 446 #define NFS41GetDeviceExtension(RxContext,pExt) \ 447 PNFS41_DEVICE_EXTENSION pExt = (PNFS41_DEVICE_EXTENSION) \ 448 ((PBYTE)(RxContext->RxDeviceObject) + sizeof(RDBSS_DEVICE_OBJECT)) 449 450 typedef struct _nfs41_fcb_list_entry { 451 LIST_ENTRY next; 452 PMRX_FCB fcb; 453 HANDLE session; 454 PNFS41_FOBX nfs41_fobx; 455 ULONGLONG ChangeTime; 456 BOOLEAN skip; 457 } nfs41_fcb_list_entry; 458 459 typedef struct _nfs41_fcb_list { 460 LIST_ENTRY head; 461 } nfs41_fcb_list; 462 nfs41_fcb_list openlist; 463 464 typedef enum _NULMRX_STORAGE_TYPE_CODES { 465 NTC_NFS41_DEVICE_EXTENSION = (NODE_TYPE_CODE)0xFC00, 466 } NFS41_STORAGE_TYPE_CODES; 467 #define RxDefineNode( node, type ) \ 468 node->NodeTypeCode = NTC_##type; \ 469 node->NodeByteSize = sizeof(type); 470 471 #define RDR_NULL_STATE 0 472 #define RDR_UNLOADED 1 473 #define RDR_UNLOADING 2 474 #define RDR_LOADING 3 475 #define RDR_LOADED 4 476 #define RDR_STOPPED 5 477 #define RDR_STOPPING 6 478 #define RDR_STARTING 7 479 #define RDR_STARTED 8 480 481 nfs41_init_driver_state nfs41_init_state = NFS41_INIT_DRIVER_STARTABLE; 482 nfs41_start_driver_state nfs41_start_state = NFS41_START_DRIVER_STARTABLE; 483 484 NTSTATUS map_readwrite_errors(DWORD status); 485 486 void print_debug_header( 487 PRX_CONTEXT RxContext) 488 { 489 490 PIO_STACK_LOCATION IrpSp = RxContext->CurrentIrpSp; 491 492 if (IrpSp) { 493 DbgP("FileOject %p name %wZ access r=%d,w=%d,d=%d share r=%d,w=%d,d=%d\n", 494 IrpSp->FileObject, &IrpSp->FileObject->FileName, 495 IrpSp->FileObject->ReadAccess, IrpSp->FileObject->WriteAccess, 496 IrpSp->FileObject->DeleteAccess, IrpSp->FileObject->SharedRead, 497 IrpSp->FileObject->SharedWrite, IrpSp->FileObject->SharedDelete); 498 print_file_object(0, IrpSp->FileObject); 499 print_irps_flags(0, RxContext->CurrentIrpSp); 500 } else 501 DbgP("Couldn't print FileObject IrpSp is NULL\n"); 502 503 print_fo_all(1, RxContext); 504 if (RxContext->CurrentIrp) 505 print_irp_flags(0, RxContext->CurrentIrp); 506 } 507 508 /* convert strings from unicode -> ansi during marshalling to 509 * save space in the upcall buffers and avoid extra copies */ 510 INLINE ULONG length_as_utf8( 511 PCUNICODE_STRING str) 512 { 513 ULONG ActualCount = 0; 514 RtlUnicodeToUTF8N(NULL, 0xffff, &ActualCount, str->Buffer, str->Length); 515 return sizeof(str->MaximumLength) + ActualCount + sizeof(UNICODE_NULL); 516 } 517 518 NTSTATUS marshall_unicode_as_utf8( 519 IN OUT unsigned char **pos, 520 IN PCUNICODE_STRING str) 521 { 522 ANSI_STRING ansi; 523 ULONG ActualCount; 524 NTSTATUS status; 525 526 if (str->Length == 0) { 527 status = STATUS_SUCCESS; 528 ActualCount = 0; 529 ansi.MaximumLength = 1; 530 goto out_copy; 531 } 532 533 /* query the number of bytes required for the utf8 encoding */ 534 status = RtlUnicodeToUTF8N(NULL, 0xffff, 535 &ActualCount, str->Buffer, str->Length); 536 if (status) { 537 print_error("RtlUnicodeToUTF8N('%wZ') failed with 0x%08X\n", 538 str, status); 539 goto out; 540 } 541 542 /* convert the string directly into the upcall buffer */ 543 ansi.Buffer = (PCHAR)*pos + sizeof(ansi.MaximumLength); 544 ansi.MaximumLength = (USHORT)ActualCount + sizeof(UNICODE_NULL); 545 status = RtlUnicodeToUTF8N(ansi.Buffer, ansi.MaximumLength, 546 &ActualCount, str->Buffer, str->Length); 547 if (status) { 548 print_error("RtlUnicodeToUTF8N(%hu, '%wZ', %hu) failed with 0x%08X\n", 549 ansi.MaximumLength, str, str->Length, status); 550 goto out; 551 } 552 553 out_copy: 554 RtlCopyMemory(*pos, &ansi.MaximumLength, sizeof(ansi.MaximumLength)); 555 *pos += sizeof(ansi.MaximumLength); 556 (*pos)[ActualCount] = '\0'; 557 *pos += ansi.MaximumLength; 558 out: 559 return status; 560 } 561 562 NTSTATUS marshal_nfs41_header( 563 nfs41_updowncall_entry *entry, 564 unsigned char *buf, 565 ULONG buf_len, 566 ULONG *len) 567 { 568 NTSTATUS status = STATUS_SUCCESS; 569 ULONG header_len = 0; 570 unsigned char *tmp = buf; 571 572 header_len = sizeof(entry->version) + sizeof(entry->xid) + 573 sizeof(entry->opcode) + 2 * sizeof(HANDLE); 574 if (header_len > buf_len) { 575 status = STATUS_INSUFFICIENT_RESOURCES; 576 goto out; 577 } 578 else 579 *len = header_len; 580 RtlCopyMemory(tmp, &entry->version, sizeof(entry->version)); 581 tmp += sizeof(entry->version); 582 RtlCopyMemory(tmp, &entry->xid, sizeof(entry->xid)); 583 tmp += sizeof(entry->xid); 584 RtlCopyMemory(tmp, &entry->opcode, sizeof(entry->opcode)); 585 tmp += sizeof(entry->opcode); 586 RtlCopyMemory(tmp, &entry->session, sizeof(HANDLE)); 587 tmp += sizeof(HANDLE); 588 RtlCopyMemory(tmp, &entry->open_state, sizeof(HANDLE)); 589 tmp += sizeof(HANDLE); 590 591 #ifdef DEBUG_MARSHAL_HEADER 592 if (MmIsAddressValid(entry->filename)) 593 DbgP("[upcall header] xid=%lld opcode=%s filename=%wZ version=%d " 594 "session=0x%x open_state=0x%x\n", entry->xid, 595 opcode2string(entry->opcode), entry->filename, 596 entry->version, entry->session, entry->open_state); 597 else 598 status = STATUS_INTERNAL_ERROR; 599 #endif 600 out: 601 return status; 602 } 603 604 const char* secflavorop2name( 605 DWORD sec_flavor) 606 { 607 switch(sec_flavor) { 608 case RPCSEC_AUTH_SYS: return "AUTH_SYS"; 609 case RPCSEC_AUTHGSS_KRB5: return "AUTHGSS_KRB5"; 610 case RPCSEC_AUTHGSS_KRB5I: return "AUTHGSS_KRB5I"; 611 case RPCSEC_AUTHGSS_KRB5P: return "AUTHGSS_KRB5P"; 612 } 613 614 return "UNKNOWN FLAVOR"; 615 } 616 NTSTATUS marshal_nfs41_mount( 617 nfs41_updowncall_entry *entry, 618 unsigned char *buf, 619 ULONG buf_len, 620 ULONG *len) 621 { 622 NTSTATUS status = STATUS_SUCCESS; 623 ULONG header_len = 0; 624 unsigned char *tmp = buf; 625 626 status = marshal_nfs41_header(entry, tmp, buf_len, len); 627 if (status) goto out; 628 else tmp += *len; 629 630 /* 03/25/2011: Kernel crash to nfsd not running but mount upcall cued up */ 631 if (!MmIsAddressValid(entry->u.Mount.srv_name) || 632 !MmIsAddressValid(entry->u.Mount.root)) { 633 status = STATUS_INTERNAL_ERROR; 634 goto out; 635 } 636 header_len = *len + length_as_utf8(entry->u.Mount.srv_name) + 637 length_as_utf8(entry->u.Mount.root) + 3 * sizeof(DWORD); 638 if (header_len > buf_len) { 639 status = STATUS_INSUFFICIENT_RESOURCES; 640 goto out; 641 } 642 status = marshall_unicode_as_utf8(&tmp, entry->u.Mount.srv_name); 643 if (status) goto out; 644 status = marshall_unicode_as_utf8(&tmp, entry->u.Mount.root); 645 if (status) goto out; 646 RtlCopyMemory(tmp, &entry->u.Mount.sec_flavor, sizeof(DWORD)); 647 tmp += sizeof(DWORD); 648 RtlCopyMemory(tmp, &entry->u.Mount.rsize, sizeof(DWORD)); 649 tmp += sizeof(DWORD); 650 RtlCopyMemory(tmp, &entry->u.Mount.wsize, sizeof(DWORD)); 651 652 *len = header_len; 653 654 #ifdef DEBUG_MARSHAL_DETAIL 655 DbgP("marshal_nfs41_mount: server name=%wZ mount point=%wZ sec_flavor=%s " 656 "rsize=%d wsize=%d\n", entry->u.Mount.srv_name, entry->u.Mount.root, 657 secflavorop2name(entry->u.Mount.sec_flavor), entry->u.Mount.rsize, 658 entry->u.Mount.wsize); 659 #endif 660 out: 661 return status; 662 } 663 664 NTSTATUS marshal_nfs41_unmount( 665 nfs41_updowncall_entry *entry, 666 unsigned char *buf, 667 ULONG buf_len, 668 ULONG *len) 669 { 670 return marshal_nfs41_header(entry, buf, buf_len, len); 671 } 672 673 NTSTATUS marshal_nfs41_open( 674 nfs41_updowncall_entry *entry, 675 unsigned char *buf, 676 ULONG buf_len, 677 ULONG *len) 678 { 679 NTSTATUS status = STATUS_SUCCESS; 680 ULONG header_len = 0; 681 unsigned char *tmp = buf; 682 683 status = marshal_nfs41_header(entry, tmp, buf_len, len); 684 if (status) goto out; 685 else tmp += *len; 686 687 header_len = *len + length_as_utf8(entry->filename) + 688 7 * sizeof(ULONG) + 2 * sizeof(HANDLE) + 689 length_as_utf8(&entry->u.Open.symlink); 690 if (header_len > buf_len) { 691 status = STATUS_INSUFFICIENT_RESOURCES; 692 goto out; 693 } 694 status = marshall_unicode_as_utf8(&tmp, entry->filename); 695 if (status) goto out; 696 RtlCopyMemory(tmp, &entry->u.Open.access_mask, 697 sizeof(entry->u.Open.access_mask)); 698 tmp += sizeof(entry->u.Open.access_mask); 699 RtlCopyMemory(tmp, &entry->u.Open.access_mode, 700 sizeof(entry->u.Open.access_mode)); 701 tmp += sizeof(entry->u.Open.access_mode); 702 RtlCopyMemory(tmp, &entry->u.Open.attrs, sizeof(entry->u.Open.attrs)); 703 tmp += sizeof(entry->u.Open.attrs); 704 RtlCopyMemory(tmp, &entry->u.Open.copts, sizeof(entry->u.Open.copts)); 705 tmp += sizeof(entry->u.Open.copts); 706 RtlCopyMemory(tmp, &entry->u.Open.disp, sizeof(entry->u.Open.disp)); 707 tmp += sizeof(entry->u.Open.disp); 708 RtlCopyMemory(tmp, &entry->u.Open.open_owner_id, 709 sizeof(entry->u.Open.open_owner_id)); 710 tmp += sizeof(entry->u.Open.open_owner_id); 711 RtlCopyMemory(tmp, &entry->u.Open.mode, sizeof(DWORD)); 712 tmp += sizeof(DWORD); 713 RtlCopyMemory(tmp, &entry->u.Open.srv_open, sizeof(HANDLE)); 714 tmp += sizeof(HANDLE); 715 status = marshall_unicode_as_utf8(&tmp, &entry->u.Open.symlink); 716 if (status) goto out; 717 718 _SEH2_TRY { 719 if (entry->u.Open.EaMdl) { 720 entry->u.Open.EaBuffer = 721 MmMapLockedPagesSpecifyCache(entry->u.Open.EaMdl, 722 #ifndef __REACTOS__ 723 UserMode, MmNonCached, NULL, TRUE, NormalPagePriority); 724 #else 725 UserMode, MmCached, NULL, TRUE, NormalPagePriority); 726 #endif 727 if (entry->u.Open.EaBuffer == NULL) { 728 print_error("MmMapLockedPagesSpecifyCache failed to map pages\n"); 729 status = STATUS_INSUFFICIENT_RESOURCES; 730 goto out; 731 } 732 } 733 } _SEH2_EXCEPT (EXCEPTION_EXECUTE_HANDLER) { 734 print_error("Call to MmMapLocked failed due to exception 0x%x\n", _SEH2_GetExceptionCode()); 735 status = STATUS_ACCESS_DENIED; 736 goto out; 737 } _SEH2_END; 738 RtlCopyMemory(tmp, &entry->u.Open.EaBuffer, sizeof(HANDLE)); 739 *len = header_len; 740 741 #ifdef DEBUG_MARSHAL_DETAIL 742 DbgP("marshal_nfs41_open: name=%wZ mask=0x%x access=0x%x attrs=0x%x " 743 "opts=0x%x dispo=0x%x open_owner_id=0x%x mode=%o srv_open=%p ea=%p\n", 744 entry->filename, entry->u.Open.access_mask, 745 entry->u.Open.access_mode, entry->u.Open.attrs, entry->u.Open.copts, 746 entry->u.Open.disp, entry->u.Open.open_owner_id, entry->u.Open.mode, 747 entry->u.Open.srv_open, entry->u.Open.EaBuffer); 748 #endif 749 out: 750 return status; 751 } 752 753 NTSTATUS marshal_nfs41_rw( 754 nfs41_updowncall_entry *entry, 755 unsigned char *buf, 756 ULONG buf_len, 757 ULONG *len) 758 { 759 NTSTATUS status = STATUS_SUCCESS; 760 ULONG header_len = 0; 761 unsigned char *tmp = buf; 762 763 status = marshal_nfs41_header(entry, tmp, buf_len, len); 764 if (status) goto out; 765 else tmp += *len; 766 767 header_len = *len + sizeof(entry->buf_len) + 768 sizeof(entry->u.ReadWrite.offset) + sizeof(HANDLE); 769 if (header_len > buf_len) { 770 status = STATUS_INSUFFICIENT_RESOURCES; 771 goto out; 772 } 773 774 RtlCopyMemory(tmp, &entry->buf_len, sizeof(entry->buf_len)); 775 tmp += sizeof(entry->buf_len); 776 RtlCopyMemory(tmp, &entry->u.ReadWrite.offset, 777 sizeof(entry->u.ReadWrite.offset)); 778 tmp += sizeof(entry->u.ReadWrite.offset); 779 _SEH2_TRY { 780 entry->u.ReadWrite.MdlAddress->MdlFlags |= MDL_MAPPING_CAN_FAIL; 781 entry->buf = 782 MmMapLockedPagesSpecifyCache(entry->u.ReadWrite.MdlAddress, 783 #ifndef __REACTOS__ 784 UserMode, MmNonCached, NULL, TRUE, NormalPagePriority); 785 #else 786 UserMode, MmCached, NULL, TRUE, NormalPagePriority); 787 #endif 788 if (entry->buf == NULL) { 789 print_error("MmMapLockedPagesSpecifyCache failed to map pages\n"); 790 status = STATUS_INSUFFICIENT_RESOURCES; 791 goto out; 792 } 793 } _SEH2_EXCEPT (EXCEPTION_EXECUTE_HANDLER) { 794 NTSTATUS code; 795 code = _SEH2_GetExceptionCode(); 796 print_error("Call to MmMapLocked failed due to exception 0x%x\n", code); 797 status = STATUS_ACCESS_DENIED; 798 goto out; 799 } _SEH2_END; 800 RtlCopyMemory(tmp, &entry->buf, sizeof(HANDLE)); 801 *len = header_len; 802 803 #ifdef DEBUG_MARSHAL_DETAIL 804 DbgP("marshal_nfs41_rw: len=%lu offset=%llu MdlAddress=%p Userspace=%p\n", 805 entry->buf_len, entry->u.ReadWrite.offset, 806 entry->u.ReadWrite.MdlAddress, entry->buf); 807 #endif 808 out: 809 return status; 810 } 811 812 NTSTATUS marshal_nfs41_lock( 813 nfs41_updowncall_entry *entry, 814 unsigned char *buf, 815 ULONG buf_len, 816 ULONG *len) 817 { 818 NTSTATUS status = STATUS_SUCCESS; 819 ULONG header_len = 0; 820 unsigned char *tmp = buf; 821 822 status = marshal_nfs41_header(entry, tmp, buf_len, len); 823 if (status) goto out; 824 else tmp += *len; 825 826 header_len = *len + 2 * sizeof(LONGLONG) + 2 * sizeof(BOOLEAN); 827 if (header_len > buf_len) { 828 status = STATUS_INSUFFICIENT_RESOURCES; 829 goto out; 830 } 831 RtlCopyMemory(tmp, &entry->u.Lock.offset, sizeof(LONGLONG)); 832 tmp += sizeof(LONGLONG); 833 RtlCopyMemory(tmp, &entry->u.Lock.length, sizeof(LONGLONG)); 834 tmp += sizeof(LONGLONG); 835 RtlCopyMemory(tmp, &entry->u.Lock.exclusive, sizeof(BOOLEAN)); 836 tmp += sizeof(BOOLEAN); 837 RtlCopyMemory(tmp, &entry->u.Lock.blocking, sizeof(BOOLEAN)); 838 *len = header_len; 839 840 #ifdef DEBUG_MARSHAL_DETAIL 841 DbgP("marshal_nfs41_lock: offset=%llx length=%llx exclusive=%u " 842 "blocking=%u\n", entry->u.Lock.offset, entry->u.Lock.length, 843 entry->u.Lock.exclusive, entry->u.Lock.blocking); 844 #endif 845 out: 846 return status; 847 } 848 849 NTSTATUS marshal_nfs41_unlock( 850 nfs41_updowncall_entry *entry, 851 unsigned char *buf, 852 ULONG buf_len, 853 ULONG *len) 854 { 855 NTSTATUS status = STATUS_SUCCESS; 856 ULONG header_len = 0; 857 unsigned char *tmp = buf; 858 PLOWIO_LOCK_LIST lock; 859 860 status = marshal_nfs41_header(entry, tmp, buf_len, len); 861 if (status) goto out; 862 else tmp += *len; 863 864 header_len = *len + sizeof(ULONG) + 865 entry->u.Unlock.count * 2 * sizeof(LONGLONG); 866 if (header_len > buf_len) { 867 status = STATUS_INSUFFICIENT_RESOURCES; 868 goto out; 869 } 870 RtlCopyMemory(tmp, &entry->u.Unlock.count, sizeof(ULONG)); 871 tmp += sizeof(ULONG); 872 873 lock = &entry->u.Unlock.locks; 874 while (lock) { 875 RtlCopyMemory(tmp, &lock->ByteOffset, sizeof(LONGLONG)); 876 tmp += sizeof(LONGLONG); 877 RtlCopyMemory(tmp, &lock->Length, sizeof(LONGLONG)); 878 tmp += sizeof(LONGLONG); 879 lock = lock->Next; 880 } 881 *len = header_len; 882 883 #ifdef DEBUG_MARSHAL_DETAIL 884 DbgP("marshal_nfs41_unlock: count=%u\n", entry->u.Unlock.count); 885 #endif 886 out: 887 return status; 888 } 889 890 NTSTATUS marshal_nfs41_close( 891 nfs41_updowncall_entry *entry, 892 unsigned char *buf, 893 ULONG buf_len, 894 ULONG *len) 895 { 896 NTSTATUS status = STATUS_SUCCESS; 897 ULONG header_len = 0; 898 unsigned char *tmp = buf; 899 900 status = marshal_nfs41_header(entry, tmp, buf_len, len); 901 if (status) goto out; 902 else tmp += *len; 903 904 header_len = *len + sizeof(BOOLEAN) + sizeof(HANDLE); 905 if (entry->u.Close.remove) 906 header_len += length_as_utf8(entry->filename) + 907 sizeof(BOOLEAN); 908 909 if (header_len > buf_len) { 910 status = STATUS_INSUFFICIENT_RESOURCES; 911 goto out; 912 } 913 RtlCopyMemory(tmp, &entry->u.Close.remove, sizeof(BOOLEAN)); 914 tmp += sizeof(BOOLEAN); 915 RtlCopyMemory(tmp, &entry->u.Close.srv_open, sizeof(HANDLE)); 916 if (entry->u.Close.remove) { 917 tmp += sizeof(HANDLE); 918 status = marshall_unicode_as_utf8(&tmp, entry->filename); 919 if (status) goto out; 920 RtlCopyMemory(tmp, &entry->u.Close.renamed, sizeof(BOOLEAN)); 921 } 922 *len = header_len; 923 924 #ifdef DEBUG_MARSHAL_DETAIL 925 DbgP("marshal_nfs41_close: name=%wZ remove=%d srv_open=%p renamed=%d\n", 926 entry->filename->Length?entry->filename:&SLASH, 927 entry->u.Close.remove, entry->u.Close.srv_open, entry->u.Close.renamed); 928 #endif 929 out: 930 return status; 931 } 932 933 NTSTATUS marshal_nfs41_dirquery( 934 nfs41_updowncall_entry *entry, 935 unsigned char *buf, 936 ULONG buf_len, 937 ULONG *len) 938 { 939 NTSTATUS status = STATUS_SUCCESS; 940 ULONG header_len = 0; 941 unsigned char *tmp = buf; 942 943 status = marshal_nfs41_header(entry, tmp, buf_len, len); 944 if (status) goto out; 945 else tmp += *len; 946 947 header_len = *len + 2 * sizeof(ULONG) + sizeof(HANDLE) + 948 length_as_utf8(entry->u.QueryFile.filter) + 3 * sizeof(BOOLEAN); 949 if (header_len > buf_len) { 950 status = STATUS_INSUFFICIENT_RESOURCES; 951 goto out; 952 } 953 954 RtlCopyMemory(tmp, &entry->u.QueryFile.InfoClass, sizeof(ULONG)); 955 tmp += sizeof(ULONG); 956 RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG)); 957 tmp += sizeof(ULONG); 958 status = marshall_unicode_as_utf8(&tmp, entry->u.QueryFile.filter); 959 if (status) goto out; 960 RtlCopyMemory(tmp, &entry->u.QueryFile.initial_query, sizeof(BOOLEAN)); 961 tmp += sizeof(BOOLEAN); 962 RtlCopyMemory(tmp, &entry->u.QueryFile.restart_scan, sizeof(BOOLEAN)); 963 tmp += sizeof(BOOLEAN); 964 RtlCopyMemory(tmp, &entry->u.QueryFile.return_single, sizeof(BOOLEAN)); 965 tmp += sizeof(BOOLEAN); 966 _SEH2_TRY { 967 entry->u.QueryFile.mdl_buf = 968 MmMapLockedPagesSpecifyCache(entry->u.QueryFile.mdl, 969 #ifndef __REACTOS__ 970 UserMode, MmNonCached, NULL, TRUE, NormalPagePriority); 971 #else 972 UserMode, MmCached, NULL, TRUE, NormalPagePriority); 973 #endif 974 if (entry->u.QueryFile.mdl_buf == NULL) { 975 print_error("MmMapLockedPagesSpecifyCache failed to map pages\n"); 976 status = STATUS_INSUFFICIENT_RESOURCES; 977 goto out; 978 } 979 } _SEH2_EXCEPT (EXCEPTION_EXECUTE_HANDLER) { 980 NTSTATUS code; 981 code = _SEH2_GetExceptionCode(); 982 print_error("Call to MmMapLocked failed due to exception 0x%x\n", code); 983 status = STATUS_ACCESS_DENIED; 984 goto out; 985 } _SEH2_END; 986 RtlCopyMemory(tmp, &entry->u.QueryFile.mdl_buf, sizeof(HANDLE)); 987 *len = header_len; 988 989 #ifdef DEBUG_MARSHAL_DETAIL 990 DbgP("marshal_nfs41_dirquery: filter='%wZ'class=%d len=%d " 991 "1st\\restart\\single=%d\\%d\\%d\n", entry->u.QueryFile.filter, 992 entry->u.QueryFile.InfoClass, entry->buf_len, 993 entry->u.QueryFile.initial_query, entry->u.QueryFile.restart_scan, 994 entry->u.QueryFile.return_single); 995 #endif 996 out: 997 return status; 998 } 999 1000 NTSTATUS marshal_nfs41_filequery( 1001 nfs41_updowncall_entry *entry, 1002 unsigned char *buf, 1003 ULONG buf_len, 1004 ULONG *len) 1005 { 1006 NTSTATUS status = STATUS_SUCCESS; 1007 ULONG header_len = 0; 1008 unsigned char *tmp = buf; 1009 1010 status = marshal_nfs41_header(entry, tmp, buf_len, len); 1011 if (status) goto out; 1012 else tmp += *len; 1013 1014 header_len = *len + 2 * sizeof(ULONG); 1015 if (header_len > buf_len) { 1016 status = STATUS_INSUFFICIENT_RESOURCES; 1017 goto out; 1018 } 1019 RtlCopyMemory(tmp, &entry->u.QueryFile.InfoClass, sizeof(ULONG)); 1020 tmp += sizeof(ULONG); 1021 RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG)); 1022 tmp += sizeof(ULONG); 1023 RtlCopyMemory(tmp, &entry->session, sizeof(HANDLE)); 1024 tmp += sizeof(HANDLE); 1025 RtlCopyMemory(tmp, &entry->open_state, sizeof(HANDLE)); 1026 *len = header_len; 1027 1028 #ifdef DEBUG_MARSHAL_DETAIL 1029 DbgP("marshal_nfs41_filequery: class=%d\n", entry->u.QueryFile.InfoClass); 1030 #endif 1031 out: 1032 return status; 1033 } 1034 1035 NTSTATUS marshal_nfs41_fileset( 1036 nfs41_updowncall_entry *entry, 1037 unsigned char *buf, 1038 ULONG buf_len, 1039 ULONG *len) 1040 { 1041 NTSTATUS status = STATUS_SUCCESS; 1042 ULONG header_len = 0; 1043 unsigned char *tmp = buf; 1044 1045 status = marshal_nfs41_header(entry, tmp, buf_len, len); 1046 if (status) goto out; 1047 else tmp += *len; 1048 1049 header_len = *len + length_as_utf8(entry->filename) + 1050 2 * sizeof(ULONG) + entry->buf_len; 1051 if (header_len > buf_len) { 1052 status = STATUS_INSUFFICIENT_RESOURCES; 1053 goto out; 1054 } 1055 status = marshall_unicode_as_utf8(&tmp, entry->filename); 1056 if (status) goto out; 1057 RtlCopyMemory(tmp, &entry->u.SetFile.InfoClass, sizeof(ULONG)); 1058 tmp += sizeof(ULONG); 1059 RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG)); 1060 tmp += sizeof(ULONG); 1061 RtlCopyMemory(tmp, entry->buf, entry->buf_len); 1062 *len = header_len; 1063 1064 #ifdef DEBUG_MARSHAL_DETAIL 1065 DbgP("marshal_nfs41_fileset: filename='%wZ' class=%d\n", 1066 entry->filename, entry->u.SetFile.InfoClass); 1067 #endif 1068 out: 1069 return status; 1070 } 1071 1072 NTSTATUS marshal_nfs41_easet( 1073 nfs41_updowncall_entry *entry, 1074 unsigned char *buf, 1075 ULONG buf_len, 1076 ULONG *len) 1077 { 1078 NTSTATUS status = STATUS_SUCCESS; 1079 ULONG header_len = 0; 1080 unsigned char *tmp = buf; 1081 1082 status = marshal_nfs41_header(entry, tmp, buf_len, len); 1083 if (status) goto out; 1084 else tmp += *len; 1085 1086 header_len = *len + length_as_utf8(entry->filename) + 1087 sizeof(ULONG) + entry->buf_len + sizeof(DWORD); 1088 if (header_len > buf_len) { 1089 status = STATUS_INSUFFICIENT_RESOURCES; 1090 goto out; 1091 } 1092 1093 status = marshall_unicode_as_utf8(&tmp, entry->filename); 1094 if (status) goto out; 1095 RtlCopyMemory(tmp, &entry->u.SetEa.mode, sizeof(DWORD)); 1096 tmp += sizeof(DWORD); 1097 RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG)); 1098 tmp += sizeof(ULONG); 1099 RtlCopyMemory(tmp, entry->buf, entry->buf_len); 1100 *len = header_len; 1101 1102 #ifdef DEBUG_MARSHAL_DETAIL 1103 DbgP("marshal_nfs41_easet: filename=%wZ, buflen=%d mode=0x%x\n", 1104 entry->filename, entry->buf_len, entry->u.SetEa.mode); 1105 #endif 1106 out: 1107 return status; 1108 } 1109 1110 NTSTATUS marshal_nfs41_eaget( 1111 nfs41_updowncall_entry *entry, 1112 unsigned char *buf, 1113 ULONG buf_len, 1114 ULONG *len) 1115 { 1116 NTSTATUS status = STATUS_SUCCESS; 1117 ULONG header_len = 0; 1118 unsigned char *tmp = buf; 1119 1120 status = marshal_nfs41_header(entry, tmp, buf_len, len); 1121 if (status) goto out; 1122 else tmp += *len; 1123 1124 header_len = *len + length_as_utf8(entry->filename) + 1125 3 * sizeof(ULONG) + entry->u.QueryEa.EaListLength + 2 * sizeof(BOOLEAN); 1126 1127 if (header_len > buf_len) { 1128 status = STATUS_INSUFFICIENT_RESOURCES; 1129 goto out; 1130 } 1131 1132 status = marshall_unicode_as_utf8(&tmp, entry->filename); 1133 if (status) goto out; 1134 RtlCopyMemory(tmp, &entry->u.QueryEa.EaIndex, sizeof(ULONG)); 1135 tmp += sizeof(ULONG); 1136 RtlCopyMemory(tmp, &entry->u.QueryEa.RestartScan, sizeof(BOOLEAN)); 1137 tmp += sizeof(BOOLEAN); 1138 RtlCopyMemory(tmp, &entry->u.QueryEa.ReturnSingleEntry, sizeof(BOOLEAN)); 1139 tmp += sizeof(BOOLEAN); 1140 RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG)); 1141 tmp += sizeof(ULONG); 1142 RtlCopyMemory(tmp, &entry->u.QueryEa.EaListLength, sizeof(ULONG)); 1143 tmp += sizeof(ULONG); 1144 if (entry->u.QueryEa.EaList && entry->u.QueryEa.EaListLength) 1145 RtlCopyMemory(tmp, entry->u.QueryEa.EaList, 1146 entry->u.QueryEa.EaListLength); 1147 *len = header_len; 1148 1149 #ifdef DEBUG_MARSHAL_DETAIL 1150 DbgP("marshal_nfs41_eaget: filename=%wZ, index=%d list_len=%d " 1151 "rescan=%d single=%d\n", entry->filename, 1152 entry->u.QueryEa.EaIndex, entry->u.QueryEa.EaListLength, 1153 entry->u.QueryEa.RestartScan, entry->u.QueryEa.ReturnSingleEntry); 1154 #endif 1155 out: 1156 return status; 1157 } 1158 1159 NTSTATUS marshal_nfs41_symlink( 1160 nfs41_updowncall_entry *entry, 1161 unsigned char *buf, 1162 ULONG buf_len, 1163 ULONG *len) 1164 { 1165 NTSTATUS status = STATUS_SUCCESS; 1166 ULONG header_len = 0; 1167 unsigned char *tmp = buf; 1168 1169 status = marshal_nfs41_header(entry, tmp, buf_len, len); 1170 if (status) goto out; 1171 else tmp += *len; 1172 1173 header_len = *len + sizeof(BOOLEAN) + length_as_utf8(entry->filename); 1174 if (entry->u.Symlink.set) 1175 header_len += length_as_utf8(entry->u.Symlink.target); 1176 if (header_len > buf_len) { 1177 status = STATUS_INSUFFICIENT_RESOURCES; 1178 goto out; 1179 } 1180 1181 status = marshall_unicode_as_utf8(&tmp, entry->filename); 1182 if (status) goto out; 1183 RtlCopyMemory(tmp, &entry->u.Symlink.set, sizeof(BOOLEAN)); 1184 tmp += sizeof(BOOLEAN); 1185 if (entry->u.Symlink.set) { 1186 status = marshall_unicode_as_utf8(&tmp, entry->u.Symlink.target); 1187 if (status) goto out; 1188 } 1189 *len = header_len; 1190 1191 #ifdef DEBUG_MARSHAL_DETAIL 1192 DbgP("marshal_nfs41_symlink: name %wZ symlink target %wZ\n", 1193 entry->filename, 1194 entry->u.Symlink.set?entry->u.Symlink.target : NULL); 1195 #endif 1196 out: 1197 return status; 1198 } 1199 1200 NTSTATUS marshal_nfs41_volume( 1201 nfs41_updowncall_entry *entry, 1202 unsigned char *buf, 1203 ULONG buf_len, 1204 ULONG *len) 1205 { 1206 NTSTATUS status = STATUS_SUCCESS; 1207 ULONG header_len = 0; 1208 unsigned char *tmp = buf; 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(FS_INFORMATION_CLASS); 1215 if (header_len > buf_len) { 1216 status = STATUS_INSUFFICIENT_RESOURCES; 1217 goto out; 1218 } 1219 1220 RtlCopyMemory(tmp, &entry->u.Volume.query, sizeof(FS_INFORMATION_CLASS)); 1221 *len = header_len; 1222 1223 #ifdef DEBUG_MARSHAL_DETAIL 1224 DbgP("marshal_nfs41_volume: class=%d\n", entry->u.Volume.query); 1225 #endif 1226 out: 1227 return status; 1228 } 1229 1230 NTSTATUS marshal_nfs41_getacl( 1231 nfs41_updowncall_entry *entry, 1232 unsigned char *buf, 1233 ULONG buf_len, 1234 ULONG *len) 1235 { 1236 NTSTATUS status = STATUS_SUCCESS; 1237 ULONG header_len = 0; 1238 unsigned char *tmp = buf; 1239 1240 status = marshal_nfs41_header(entry, tmp, buf_len, len); 1241 if (status) goto out; 1242 else tmp += *len; 1243 1244 header_len = *len + sizeof(SECURITY_INFORMATION); 1245 if (header_len > buf_len) { 1246 status = STATUS_INSUFFICIENT_RESOURCES; 1247 goto out; 1248 } 1249 1250 RtlCopyMemory(tmp, &entry->u.Acl.query, sizeof(SECURITY_INFORMATION)); 1251 *len = header_len; 1252 1253 #ifdef DEBUG_MARSHAL_DETAIL 1254 DbgP("marshal_nfs41_getacl: class=0x%x\n", entry->u.Acl.query); 1255 #endif 1256 out: 1257 return status; 1258 } 1259 1260 NTSTATUS marshal_nfs41_setacl( 1261 nfs41_updowncall_entry *entry, 1262 unsigned char *buf, 1263 ULONG buf_len, 1264 ULONG *len) 1265 { 1266 NTSTATUS status = STATUS_SUCCESS; 1267 ULONG header_len = 0; 1268 unsigned char *tmp = buf; 1269 1270 status = marshal_nfs41_header(entry, tmp, buf_len, len); 1271 if (status) goto out; 1272 else tmp += *len; 1273 1274 header_len = *len + sizeof(SECURITY_INFORMATION) + 1275 sizeof(ULONG) + entry->buf_len; 1276 if (header_len > buf_len) { 1277 status = STATUS_INSUFFICIENT_RESOURCES; 1278 goto out; 1279 } 1280 1281 RtlCopyMemory(tmp, &entry->u.Acl.query, sizeof(SECURITY_INFORMATION)); 1282 tmp += sizeof(SECURITY_INFORMATION); 1283 RtlCopyMemory(tmp, &entry->buf_len, sizeof(ULONG)); 1284 tmp += sizeof(ULONG); 1285 RtlCopyMemory(tmp, entry->buf, entry->buf_len); 1286 *len = header_len; 1287 1288 #ifdef DEBUG_MARSHAL_DETAIL 1289 DbgP("marshal_nfs41_setacl: class=0x%x sec_desc_len=%lu\n", 1290 entry->u.Acl.query, entry->buf_len); 1291 #endif 1292 out: 1293 return status; 1294 } 1295 1296 NTSTATUS marshal_nfs41_shutdown( 1297 nfs41_updowncall_entry *entry, 1298 unsigned char *buf, 1299 ULONG buf_len, 1300 ULONG *len) 1301 { 1302 return marshal_nfs41_header(entry, buf, buf_len, len); 1303 } 1304 1305 void nfs41_invalidate_cache ( 1306 IN PRX_CONTEXT RxContext) 1307 { 1308 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 1309 unsigned char *buf = LowIoContext->ParamsFor.IoCtl.pInputBuffer; 1310 ULONG flag = DISABLE_CACHING; 1311 PMRX_SRV_OPEN srv_open; 1312 1313 RtlCopyMemory(&srv_open, buf, sizeof(HANDLE)); 1314 #ifdef DEBUG_INVALIDATE_CACHE 1315 DbgP("nfs41_invalidate_cache: received srv_open=%p %wZ\n", 1316 srv_open, srv_open->pAlreadyPrefixedName); 1317 #endif 1318 if (MmIsAddressValid(srv_open)) 1319 RxIndicateChangeOfBufferingStateForSrvOpen( 1320 srv_open->pFcb->pNetRoot->pSrvCall, srv_open, 1321 srv_open->Key, ULongToPtr(flag)); 1322 } 1323 1324 NTSTATUS handle_upcall( 1325 IN PRX_CONTEXT RxContext, 1326 IN nfs41_updowncall_entry *entry, 1327 OUT ULONG *len) 1328 { 1329 NTSTATUS status = STATUS_SUCCESS; 1330 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 1331 ULONG cbOut = LowIoContext->ParamsFor.IoCtl.OutputBufferLength; 1332 unsigned char *pbOut = LowIoContext->ParamsFor.IoCtl.pOutputBuffer; 1333 1334 status = SeImpersonateClientEx(entry->psec_ctx, NULL); 1335 if (status != STATUS_SUCCESS) { 1336 print_error("SeImpersonateClientEx failed %x\n", status); 1337 goto out; 1338 } 1339 1340 switch(entry->opcode) { 1341 case NFS41_SHUTDOWN: 1342 status = marshal_nfs41_shutdown(entry, pbOut, cbOut, len); 1343 KeSetEvent(&entry->cond, 0, FALSE); 1344 break; 1345 case NFS41_MOUNT: 1346 status = marshal_nfs41_mount(entry, pbOut, cbOut, len); 1347 break; 1348 case NFS41_UNMOUNT: 1349 status = marshal_nfs41_unmount(entry, pbOut, cbOut, len); 1350 break; 1351 case NFS41_OPEN: 1352 status = marshal_nfs41_open(entry, pbOut, cbOut, len); 1353 break; 1354 case NFS41_READ: 1355 status = marshal_nfs41_rw(entry, pbOut, cbOut, len); 1356 break; 1357 case NFS41_WRITE: 1358 status = marshal_nfs41_rw(entry, pbOut, cbOut, len); 1359 break; 1360 case NFS41_LOCK: 1361 status = marshal_nfs41_lock(entry, pbOut, cbOut, len); 1362 break; 1363 case NFS41_UNLOCK: 1364 status = marshal_nfs41_unlock(entry, pbOut, cbOut, len); 1365 break; 1366 case NFS41_CLOSE: 1367 status = marshal_nfs41_close(entry, pbOut, cbOut, len); 1368 break; 1369 case NFS41_DIR_QUERY: 1370 status = marshal_nfs41_dirquery(entry, pbOut, cbOut, len); 1371 break; 1372 case NFS41_FILE_QUERY: 1373 status = marshal_nfs41_filequery(entry, pbOut, cbOut, len); 1374 break; 1375 case NFS41_FILE_SET: 1376 status = marshal_nfs41_fileset(entry, pbOut, cbOut, len); 1377 break; 1378 case NFS41_EA_SET: 1379 status = marshal_nfs41_easet(entry, pbOut, cbOut, len); 1380 break; 1381 case NFS41_EA_GET: 1382 status = marshal_nfs41_eaget(entry, pbOut, cbOut, len); 1383 break; 1384 case NFS41_SYMLINK: 1385 status = marshal_nfs41_symlink(entry, pbOut, cbOut, len); 1386 break; 1387 case NFS41_VOLUME_QUERY: 1388 status = marshal_nfs41_volume(entry, pbOut, cbOut, len); 1389 break; 1390 case NFS41_ACL_QUERY: 1391 status = marshal_nfs41_getacl(entry, pbOut, cbOut, len); 1392 break; 1393 case NFS41_ACL_SET: 1394 status = marshal_nfs41_setacl(entry, pbOut, cbOut, len); 1395 break; 1396 default: 1397 status = STATUS_INVALID_PARAMETER; 1398 print_error("Unknown nfs41 ops %d\n", entry->opcode); 1399 } 1400 1401 if (status == STATUS_SUCCESS) 1402 print_hexbuf(0, (unsigned char *)"upcall buffer", pbOut, *len); 1403 1404 out: 1405 return status; 1406 } 1407 1408 NTSTATUS nfs41_UpcallCreate( 1409 IN DWORD opcode, 1410 IN PSECURITY_CLIENT_CONTEXT clnt_sec_ctx, 1411 IN HANDLE session, 1412 IN HANDLE open_state, 1413 IN DWORD version, 1414 IN PUNICODE_STRING filename, 1415 OUT nfs41_updowncall_entry **entry_out) 1416 { 1417 NTSTATUS status = STATUS_SUCCESS; 1418 nfs41_updowncall_entry *entry; 1419 SECURITY_SUBJECT_CONTEXT sec_ctx; 1420 SECURITY_QUALITY_OF_SERVICE sec_qos; 1421 1422 entry = RxAllocatePoolWithTag(NonPagedPool, sizeof(nfs41_updowncall_entry), 1423 NFS41_MM_POOLTAG_UP); 1424 if (entry == NULL) { 1425 status = STATUS_INSUFFICIENT_RESOURCES; 1426 goto out; 1427 } 1428 1429 RtlZeroMemory(entry, sizeof(nfs41_updowncall_entry)); 1430 entry->xid = InterlockedIncrement64(&xid); 1431 entry->opcode = opcode; 1432 entry->state = NFS41_WAITING_FOR_UPCALL; 1433 entry->session = session; 1434 entry->open_state = open_state; 1435 entry->version = version; 1436 if (filename && filename->Length) entry->filename = filename; 1437 else if (filename && !filename->Length) entry->filename = (PUNICODE_STRING)&SLASH; 1438 else entry->filename = (PUNICODE_STRING)&EMPTY_STRING; 1439 /*XXX KeInitializeEvent will bugcheck under verifier if allocated 1440 * from PagedPool? */ 1441 KeInitializeEvent(&entry->cond, SynchronizationEvent, FALSE); 1442 ExInitializeFastMutex(&entry->lock); 1443 1444 if (clnt_sec_ctx == NULL) { 1445 SeCaptureSubjectContext(&sec_ctx); 1446 sec_qos.ContextTrackingMode = SECURITY_DYNAMIC_TRACKING; 1447 sec_qos.ImpersonationLevel = SecurityImpersonation; 1448 sec_qos.Length = sizeof(SECURITY_QUALITY_OF_SERVICE); 1449 sec_qos.EffectiveOnly = 0; 1450 status = SeCreateClientSecurityFromSubjectContext(&sec_ctx, &sec_qos, 1451 1, &entry->sec_ctx); 1452 if (status != STATUS_SUCCESS) { 1453 print_error("nfs41_UpcallCreate: " 1454 "SeCreateClientSecurityFromSubjectContext failed with %x\n", 1455 status); 1456 RxFreePool(entry); 1457 } else 1458 entry->psec_ctx = &entry->sec_ctx; 1459 SeReleaseSubjectContext(&sec_ctx); 1460 } else 1461 entry->psec_ctx = clnt_sec_ctx; 1462 1463 *entry_out = entry; 1464 out: 1465 return status; 1466 } 1467 1468 NTSTATUS nfs41_UpcallWaitForReply( 1469 IN nfs41_updowncall_entry *entry, 1470 IN DWORD secs) 1471 { 1472 NTSTATUS status = STATUS_SUCCESS; 1473 1474 nfs41_AddEntry(upcallLock, upcall, entry); 1475 KeSetEvent(&upcallEvent, 0, FALSE); 1476 if (!entry->async_op) { 1477 LARGE_INTEGER timeout; 1478 timeout.QuadPart = RELATIVE(SECONDS(secs)); 1479 /* 02/03/2011 AGLO: it is not clear what the "right" waiting design 1480 * should be. Having non-interruptable waiting seems to be the right 1481 * approach. However, when things go wrong, the only wait to proceed 1482 * is a reboot (since "waits" are not interruptable we can't stop a 1483 * hung task. Having interruptable wait causes issues with security 1484 * context. For now, I'm making CLOSE non-interruptable but keeping 1485 * the rest interruptable so that we don't have to reboot all the time 1486 */ 1487 /* 02/15/2011 cbodley: added NFS41_UNLOCK for the same reason. locking 1488 * tests were triggering an interrupted unlock, which led to a bugcheck 1489 * in CloseSrvOpen() */ 1490 #define MAKE_WAITONCLOSE_NONITERRUPTABLE 1491 #ifdef MAKE_WAITONCLOSE_NONITERRUPTABLE 1492 if (entry->opcode == NFS41_CLOSE || entry->opcode == NFS41_UNLOCK) 1493 status = KeWaitForSingleObject(&entry->cond, Executive, 1494 KernelMode, FALSE, &timeout); 1495 else { 1496 status = KeWaitForSingleObject(&entry->cond, Executive, 1497 UserMode, TRUE, &timeout); 1498 } 1499 if (status != STATUS_SUCCESS) { 1500 print_wait_status(1, "[downcall]", status, 1501 opcode2string(entry->opcode), entry, entry->xid); 1502 if (status == STATUS_TIMEOUT) 1503 status = STATUS_NETWORK_UNREACHABLE; 1504 } 1505 #else 1506 1507 status = KeWaitForSingleObject(&entry->cond, Executive, KernelMode, FALSE, NULL); 1508 #endif 1509 print_wait_status(0, "[downcall]", status, opcode2string(entry->opcode), 1510 entry, entry->xid); 1511 } else 1512 goto out; 1513 1514 switch(status) { 1515 case STATUS_SUCCESS: break; 1516 case STATUS_USER_APC: 1517 case STATUS_ALERTED: 1518 default: 1519 ExAcquireFastMutex(&entry->lock); 1520 if (entry->state == NFS41_DONE_PROCESSING) { 1521 ExReleaseFastMutex(&entry->lock); 1522 break; 1523 } 1524 DbgP("[upcall] abandoning %s entry=%p xid=%lld\n", 1525 opcode2string(entry->opcode), entry, entry->xid); 1526 entry->state = NFS41_NOT_WAITING; 1527 ExReleaseFastMutex(&entry->lock); 1528 goto out; 1529 } 1530 nfs41_RemoveEntry(downcallLock, entry); 1531 out: 1532 return status; 1533 } 1534 1535 NTSTATUS nfs41_upcall( 1536 IN PRX_CONTEXT RxContext) 1537 { 1538 NTSTATUS status = STATUS_SUCCESS; 1539 nfs41_updowncall_entry *entry = NULL; 1540 ULONG len = 0; 1541 PLIST_ENTRY pEntry; 1542 1543 process_upcall: 1544 nfs41_RemoveFirst(upcallLock, upcall, pEntry); 1545 if (pEntry) { 1546 entry = (nfs41_updowncall_entry *)CONTAINING_RECORD(pEntry, 1547 nfs41_updowncall_entry, next); 1548 ExAcquireFastMutex(&entry->lock); 1549 nfs41_AddEntry(downcallLock, downcall, entry); 1550 status = handle_upcall(RxContext, entry, &len); 1551 if (status == STATUS_SUCCESS && 1552 entry->state == NFS41_WAITING_FOR_UPCALL) 1553 entry->state = NFS41_WAITING_FOR_DOWNCALL; 1554 ExReleaseFastMutex(&entry->lock); 1555 if (status) { 1556 entry->status = status; 1557 KeSetEvent(&entry->cond, 0, FALSE); 1558 RxContext->InformationToReturn = 0; 1559 } else 1560 RxContext->InformationToReturn = len; 1561 } 1562 else { 1563 status = KeWaitForSingleObject(&upcallEvent, Executive, UserMode, TRUE, 1564 (PLARGE_INTEGER) NULL); 1565 print_wait_status(0, "[upcall]", status, NULL, NULL, 0); 1566 switch (status) { 1567 case STATUS_SUCCESS: goto process_upcall; 1568 case STATUS_USER_APC: 1569 case STATUS_ALERTED: 1570 default: goto out; 1571 } 1572 } 1573 out: 1574 return status; 1575 } 1576 1577 void unmarshal_nfs41_header( 1578 nfs41_updowncall_entry *tmp, 1579 unsigned char **buf) 1580 { 1581 RtlZeroMemory(tmp, sizeof(nfs41_updowncall_entry)); 1582 1583 RtlCopyMemory(&tmp->xid, *buf, sizeof(tmp->xid)); 1584 *buf += sizeof(tmp->xid); 1585 RtlCopyMemory(&tmp->opcode, *buf, sizeof(tmp->opcode)); 1586 *buf += sizeof(tmp->opcode); 1587 RtlCopyMemory(&tmp->status, *buf, sizeof(tmp->status)); 1588 *buf += sizeof(tmp->status); 1589 RtlCopyMemory(&tmp->errno, *buf, sizeof(tmp->errno)); 1590 *buf += sizeof(tmp->errno); 1591 #ifdef DEBUG_MARSHAL_HEADER 1592 DbgP("[downcall header] xid=%lld opcode=%s status=%d errno=%d\n", tmp->xid, 1593 opcode2string(tmp->opcode), tmp->status, tmp->errno); 1594 #endif 1595 } 1596 1597 void unmarshal_nfs41_mount( 1598 nfs41_updowncall_entry *cur, 1599 unsigned char **buf) 1600 { 1601 RtlCopyMemory(&cur->session, *buf, sizeof(HANDLE)); 1602 *buf += sizeof(HANDLE); 1603 RtlCopyMemory(&cur->version, *buf, sizeof(DWORD)); 1604 *buf += sizeof(DWORD); 1605 RtlCopyMemory(&cur->u.Mount.lease_time, *buf, sizeof(DWORD)); 1606 *buf += sizeof(DWORD); 1607 RtlCopyMemory(cur->u.Mount.FsAttrs, *buf, sizeof(FILE_FS_ATTRIBUTE_INFORMATION)); 1608 #ifdef DEBUG_MARSHAL_DETAIL 1609 DbgP("unmarshal_nfs41_mount: session pointer 0x%x version %d lease_time " 1610 "%d\n", cur->session, cur->version, cur->u.Mount.lease_time); 1611 #endif 1612 } 1613 1614 VOID unmarshal_nfs41_setattr( 1615 nfs41_updowncall_entry *cur, 1616 PULONGLONG dest_buf, 1617 unsigned char **buf) 1618 { 1619 RtlCopyMemory(dest_buf, *buf, sizeof(ULONGLONG)); 1620 #ifdef DEBUG_MARSHAL_DETAIL 1621 DbgP("unmarshal_nfs41_setattr: returned ChangeTime %llu\n", *dest_buf); 1622 #endif 1623 } 1624 1625 NTSTATUS unmarshal_nfs41_rw( 1626 nfs41_updowncall_entry *cur, 1627 unsigned char **buf) 1628 { 1629 NTSTATUS status = STATUS_SUCCESS; 1630 1631 RtlCopyMemory(&cur->buf_len, *buf, sizeof(cur->buf_len)); 1632 *buf += sizeof(cur->buf_len); 1633 RtlCopyMemory(&cur->ChangeTime, *buf, sizeof(ULONGLONG)); 1634 #ifdef DEBUG_MARSHAL_DETAIL 1635 DbgP("unmarshal_nfs41_rw: returned len %lu ChangeTime %llu\n", 1636 cur->buf_len, cur->ChangeTime); 1637 #endif 1638 #if 1 1639 /* 08/27/2010: it looks like we really don't need to call 1640 * MmUnmapLockedPages() eventhough we called 1641 * MmMapLockedPagesSpecifyCache() as the MDL passed to us 1642 * is already locked. 1643 */ 1644 _SEH2_TRY { 1645 MmUnmapLockedPages(cur->buf, cur->u.ReadWrite.MdlAddress); 1646 } _SEH2_EXCEPT (EXCEPTION_EXECUTE_HANDLER) { 1647 NTSTATUS code; 1648 code = _SEH2_GetExceptionCode(); 1649 print_error("Call to MmUnmapLockedPages failed due to" 1650 " exception 0x%0x\n", code); 1651 status = STATUS_ACCESS_DENIED; 1652 } _SEH2_END; 1653 #endif 1654 return status; 1655 } 1656 1657 NTSTATUS unmarshal_nfs41_open( 1658 nfs41_updowncall_entry *cur, 1659 unsigned char **buf) 1660 { 1661 NTSTATUS status = STATUS_SUCCESS; 1662 1663 _SEH2_TRY { 1664 if (cur->u.Open.EaBuffer) 1665 MmUnmapLockedPages(cur->u.Open.EaBuffer, cur->u.Open.EaMdl); 1666 } _SEH2_EXCEPT (EXCEPTION_EXECUTE_HANDLER) { 1667 print_error("MmUnmapLockedPages thrown exception=0x%0x\n", _SEH2_GetExceptionCode()); 1668 status = cur->status = STATUS_ACCESS_DENIED; 1669 goto out; 1670 } _SEH2_END; 1671 1672 RtlCopyMemory(&cur->u.Open.binfo, *buf, sizeof(FILE_BASIC_INFORMATION)); 1673 *buf += sizeof(FILE_BASIC_INFORMATION); 1674 RtlCopyMemory(&cur->u.Open.sinfo, *buf, sizeof(FILE_STANDARD_INFORMATION)); 1675 *buf += sizeof(FILE_STANDARD_INFORMATION); 1676 RtlCopyMemory(&cur->open_state, *buf, sizeof(HANDLE)); 1677 *buf += sizeof(HANDLE); 1678 RtlCopyMemory(&cur->u.Open.mode, *buf, sizeof(DWORD)); 1679 *buf += sizeof(DWORD); 1680 RtlCopyMemory(&cur->ChangeTime, *buf, sizeof(ULONGLONG)); 1681 *buf += sizeof(ULONGLONG); 1682 RtlCopyMemory(&cur->u.Open.deleg_type, *buf, sizeof(DWORD)); 1683 *buf += sizeof(DWORD); 1684 if (cur->errno == ERROR_REPARSE) { 1685 RtlCopyMemory(&cur->u.Open.symlink_embedded, *buf, sizeof(BOOLEAN)); 1686 *buf += sizeof(BOOLEAN); 1687 RtlCopyMemory(&cur->u.Open.symlink.MaximumLength, *buf, 1688 sizeof(USHORT)); 1689 *buf += sizeof(USHORT); 1690 cur->u.Open.symlink.Length = cur->u.Open.symlink.MaximumLength - 1691 sizeof(WCHAR); 1692 cur->u.Open.symlink.Buffer = RxAllocatePoolWithTag(NonPagedPool, 1693 cur->u.Open.symlink.MaximumLength, NFS41_MM_POOLTAG); 1694 if (cur->u.Open.symlink.Buffer == NULL) { 1695 cur->status = STATUS_INSUFFICIENT_RESOURCES; 1696 status = STATUS_UNSUCCESSFUL; 1697 goto out; 1698 } 1699 RtlCopyMemory(cur->u.Open.symlink.Buffer, *buf, 1700 cur->u.Open.symlink.MaximumLength); 1701 #ifdef DEBUG_MARSHAL_DETAIL 1702 DbgP("unmarshal_nfs41_open: ERROR_REPARSE -> '%wZ'\n", &cur->u.Open.symlink); 1703 #endif 1704 } 1705 #ifdef DEBUG_MARSHAL_DETAIL 1706 DbgP("unmarshal_nfs41_open: open_state 0x%x mode %o changeattr %llu " 1707 "deleg_type %d\n", cur->open_state, cur->u.Open.mode, 1708 cur->ChangeTime, cur->u.Open.deleg_type); 1709 #endif 1710 out: 1711 return status; 1712 } 1713 1714 NTSTATUS unmarshal_nfs41_dirquery( 1715 nfs41_updowncall_entry *cur, 1716 unsigned char **buf) 1717 { 1718 NTSTATUS status = STATUS_SUCCESS; 1719 ULONG buf_len; 1720 1721 RtlCopyMemory(&buf_len, *buf, sizeof(ULONG)); 1722 #ifdef DEBUG_MARSHAL_DETAIL 1723 DbgP("unmarshal_nfs41_dirquery: reply size %d\n", buf_len); 1724 #endif 1725 *buf += sizeof(ULONG); 1726 _SEH2_TRY { 1727 MmUnmapLockedPages(cur->u.QueryFile.mdl_buf, cur->u.QueryFile.mdl); 1728 } _SEH2_EXCEPT (EXCEPTION_EXECUTE_HANDLER) { 1729 NTSTATUS code; 1730 code = _SEH2_GetExceptionCode(); 1731 print_error("MmUnmapLockedPages thrown exception=0x%0x\n", code); 1732 status = STATUS_ACCESS_DENIED; 1733 } _SEH2_END; 1734 if (buf_len > cur->buf_len) 1735 cur->status = STATUS_BUFFER_TOO_SMALL; 1736 cur->buf_len = buf_len; 1737 1738 return status; 1739 } 1740 1741 void unmarshal_nfs41_attrget( 1742 nfs41_updowncall_entry *cur, 1743 PVOID attr_value, 1744 ULONG *attr_len, 1745 unsigned char **buf) 1746 { 1747 ULONG buf_len; 1748 RtlCopyMemory(&buf_len, *buf, sizeof(ULONG)); 1749 if (buf_len > *attr_len) { 1750 cur->status = STATUS_BUFFER_TOO_SMALL; 1751 return; 1752 } 1753 *buf += sizeof(ULONG); 1754 *attr_len = buf_len; 1755 RtlCopyMemory(attr_value, *buf, buf_len); 1756 *buf += buf_len; 1757 } 1758 1759 void unmarshal_nfs41_eaget( 1760 nfs41_updowncall_entry *cur, 1761 unsigned char **buf) 1762 { 1763 RtlCopyMemory(&cur->u.QueryEa.Overflow, *buf, sizeof(ULONG)); 1764 *buf += sizeof(ULONG); 1765 RtlCopyMemory(&cur->buf_len, *buf, sizeof(ULONG)); 1766 *buf += sizeof(ULONG); 1767 if (cur->u.QueryEa.Overflow != ERROR_INSUFFICIENT_BUFFER) { 1768 RtlCopyMemory(cur->buf, *buf, cur->buf_len); 1769 *buf += cur->buf_len; 1770 } 1771 } 1772 1773 void unmarshal_nfs41_getattr( 1774 nfs41_updowncall_entry *cur, 1775 unsigned char **buf) 1776 { 1777 unmarshal_nfs41_attrget(cur, cur->buf, &cur->buf_len, buf); 1778 RtlCopyMemory(&cur->ChangeTime, *buf, sizeof(LONGLONG)); 1779 #ifdef DEBUG_MARSHAL_DETAIL 1780 if (cur->u.QueryFile.InfoClass == FileBasicInformation) 1781 DbgP("[unmarshal_nfs41_getattr] ChangeTime %llu\n", cur->ChangeTime); 1782 #endif 1783 } 1784 1785 NTSTATUS unmarshal_nfs41_getacl( 1786 nfs41_updowncall_entry *cur, 1787 unsigned char **buf) 1788 { 1789 NTSTATUS status = STATUS_SUCCESS; 1790 DWORD buf_len; 1791 1792 RtlCopyMemory(&buf_len, *buf, sizeof(DWORD)); 1793 *buf += sizeof(DWORD); 1794 cur->buf = RxAllocatePoolWithTag(NonPagedPool, 1795 buf_len, NFS41_MM_POOLTAG_ACL); 1796 if (cur->buf == NULL) { 1797 cur->status = status = STATUS_INSUFFICIENT_RESOURCES; 1798 goto out; 1799 } 1800 RtlCopyMemory(cur->buf, *buf, buf_len); 1801 if (buf_len > cur->buf_len) 1802 cur->status = STATUS_BUFFER_TOO_SMALL; 1803 cur->buf_len = buf_len; 1804 1805 out: 1806 return status; 1807 } 1808 1809 void unmarshal_nfs41_symlink( 1810 nfs41_updowncall_entry *cur, 1811 unsigned char **buf) 1812 { 1813 if (cur->u.Symlink.set) return; 1814 1815 RtlCopyMemory(&cur->u.Symlink.target->Length, *buf, sizeof(USHORT)); 1816 *buf += sizeof(USHORT); 1817 if (cur->u.Symlink.target->Length > 1818 cur->u.Symlink.target->MaximumLength) { 1819 cur->status = STATUS_BUFFER_TOO_SMALL; 1820 return; 1821 } 1822 RtlCopyMemory(cur->u.Symlink.target->Buffer, *buf, 1823 cur->u.Symlink.target->Length); 1824 cur->u.Symlink.target->Length -= sizeof(UNICODE_NULL); 1825 } 1826 1827 NTSTATUS nfs41_downcall( 1828 IN PRX_CONTEXT RxContext) 1829 { 1830 NTSTATUS status = STATUS_SUCCESS; 1831 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 1832 ULONG in_len = LowIoContext->ParamsFor.IoCtl.InputBufferLength; 1833 unsigned char *buf = LowIoContext->ParamsFor.IoCtl.pInputBuffer; 1834 PLIST_ENTRY pEntry; 1835 nfs41_updowncall_entry *tmp, *cur= NULL; 1836 BOOLEAN found = 0; 1837 1838 print_hexbuf(0, (unsigned char *)"downcall buffer", buf, in_len); 1839 1840 tmp = RxAllocatePoolWithTag(NonPagedPool, sizeof(nfs41_updowncall_entry), 1841 NFS41_MM_POOLTAG_DOWN); 1842 if (tmp == NULL) goto out; 1843 1844 unmarshal_nfs41_header(tmp, &buf); 1845 1846 ExAcquireFastMutex(&downcallLock); 1847 pEntry = &downcall.head; 1848 pEntry = pEntry->Flink; 1849 while (pEntry != NULL) { 1850 cur = (nfs41_updowncall_entry *)CONTAINING_RECORD(pEntry, 1851 nfs41_updowncall_entry, next); 1852 if (cur->xid == tmp->xid) { 1853 found = 1; 1854 break; 1855 } 1856 if (pEntry->Flink == &downcall.head) 1857 break; 1858 pEntry = pEntry->Flink; 1859 } 1860 ExReleaseFastMutex(&downcallLock); 1861 SeStopImpersonatingClient(); 1862 if (!found) { 1863 print_error("Didn't find xid=%lld entry\n", tmp->xid); 1864 goto out_free; 1865 } 1866 1867 ExAcquireFastMutex(&cur->lock); 1868 if (cur->state == NFS41_NOT_WAITING) { 1869 DbgP("[downcall] Nobody is waiting for this request!!!\n"); 1870 switch(cur->opcode) { 1871 case NFS41_WRITE: 1872 case NFS41_READ: 1873 MmUnmapLockedPages(cur->buf, cur->u.ReadWrite.MdlAddress); 1874 break; 1875 case NFS41_DIR_QUERY: 1876 MmUnmapLockedPages(cur->u.QueryFile.mdl_buf, 1877 cur->u.QueryFile.mdl); 1878 IoFreeMdl(cur->u.QueryFile.mdl); 1879 break; 1880 case NFS41_OPEN: 1881 if (cur->u.Open.EaMdl) { 1882 MmUnmapLockedPages(cur->u.Open.EaBuffer, 1883 cur->u.Open.EaMdl); 1884 IoFreeMdl(cur->u.Open.EaMdl); 1885 } 1886 break; 1887 } 1888 ExReleaseFastMutex(&cur->lock); 1889 nfs41_RemoveEntry(downcallLock, cur); 1890 RxFreePool(cur); 1891 status = STATUS_UNSUCCESSFUL; 1892 goto out_free; 1893 } 1894 cur->state = NFS41_DONE_PROCESSING; 1895 cur->status = tmp->status; 1896 cur->errno = tmp->errno; 1897 status = STATUS_SUCCESS; 1898 1899 if (!tmp->status) { 1900 switch (tmp->opcode) { 1901 case NFS41_MOUNT: 1902 unmarshal_nfs41_mount(cur, &buf); 1903 break; 1904 case NFS41_WRITE: 1905 case NFS41_READ: 1906 status = unmarshal_nfs41_rw(cur, &buf); 1907 break; 1908 case NFS41_OPEN: 1909 status = unmarshal_nfs41_open(cur, &buf); 1910 break; 1911 case NFS41_DIR_QUERY: 1912 status = unmarshal_nfs41_dirquery(cur, &buf); 1913 break; 1914 case NFS41_FILE_QUERY: 1915 unmarshal_nfs41_getattr(cur, &buf); 1916 break; 1917 case NFS41_EA_GET: 1918 unmarshal_nfs41_eaget(cur, &buf); 1919 break; 1920 case NFS41_SYMLINK: 1921 unmarshal_nfs41_symlink(cur, &buf); 1922 break; 1923 case NFS41_VOLUME_QUERY: 1924 unmarshal_nfs41_attrget(cur, cur->buf, &cur->buf_len, &buf); 1925 break; 1926 case NFS41_ACL_QUERY: 1927 status = unmarshal_nfs41_getacl(cur, &buf); 1928 break; 1929 case NFS41_FILE_SET: 1930 unmarshal_nfs41_setattr(cur, &cur->ChangeTime, &buf); 1931 break; 1932 case NFS41_EA_SET: 1933 unmarshal_nfs41_setattr(cur, &cur->ChangeTime, &buf); 1934 break; 1935 case NFS41_ACL_SET: 1936 unmarshal_nfs41_setattr(cur, &cur->ChangeTime, &buf); 1937 break; 1938 } 1939 } 1940 ExReleaseFastMutex(&cur->lock); 1941 if (cur->async_op) { 1942 if (cur->status == STATUS_SUCCESS) { 1943 cur->u.ReadWrite.rxcontext->StoredStatus = STATUS_SUCCESS; 1944 cur->u.ReadWrite.rxcontext->InformationToReturn = 1945 cur->buf_len; 1946 } else { 1947 cur->u.ReadWrite.rxcontext->StoredStatus = 1948 map_readwrite_errors(cur->status); 1949 cur->u.ReadWrite.rxcontext->InformationToReturn = 0; 1950 } 1951 nfs41_RemoveEntry(downcallLock, cur); 1952 RxLowIoCompletion(cur->u.ReadWrite.rxcontext); 1953 RxFreePool(cur); 1954 } else 1955 KeSetEvent(&cur->cond, 0, FALSE); 1956 1957 out_free: 1958 RxFreePool(tmp); 1959 out: 1960 return status; 1961 } 1962 1963 NTSTATUS nfs41_shutdown_daemon( 1964 DWORD version) 1965 { 1966 NTSTATUS status = STATUS_SUCCESS; 1967 nfs41_updowncall_entry *entry = NULL; 1968 1969 DbgEn(); 1970 status = nfs41_UpcallCreate(NFS41_SHUTDOWN, NULL, INVALID_HANDLE_VALUE, 1971 INVALID_HANDLE_VALUE, version, NULL, &entry); 1972 if (status) goto out; 1973 1974 status = nfs41_UpcallWaitForReply(entry, UPCALL_TIMEOUT_DEFAULT); 1975 SeDeleteClientSecurity(&entry->sec_ctx); 1976 if (status) goto out; 1977 1978 RxFreePool(entry); 1979 out: 1980 DbgEx(); 1981 return status; 1982 } 1983 1984 NTSTATUS SharedMemoryInit( 1985 OUT PHANDLE phSection) 1986 { 1987 NTSTATUS status; 1988 HANDLE hSection; 1989 UNICODE_STRING SectionName; 1990 SECURITY_DESCRIPTOR SecurityDesc; 1991 OBJECT_ATTRIBUTES SectionAttrs; 1992 LARGE_INTEGER nSectionSize; 1993 1994 DbgEn(); 1995 1996 RtlInitUnicodeString(&SectionName, NFS41_SHARED_MEMORY_NAME); 1997 1998 /* XXX: setting dacl=NULL grants access to everyone */ 1999 status = RtlCreateSecurityDescriptor(&SecurityDesc, 2000 SECURITY_DESCRIPTOR_REVISION); 2001 if (status) { 2002 print_error("RtlCreateSecurityDescriptor() failed with %08X\n", status); 2003 goto out; 2004 } 2005 status = RtlSetDaclSecurityDescriptor(&SecurityDesc, TRUE, NULL, FALSE); 2006 if (status) { 2007 print_error("RtlSetDaclSecurityDescriptor() failed with %08X\n", status); 2008 goto out; 2009 } 2010 2011 InitializeObjectAttributes(&SectionAttrs, &SectionName, 2012 0, NULL, &SecurityDesc); 2013 2014 nSectionSize.QuadPart = sizeof(NFS41NP_SHARED_MEMORY); 2015 2016 status = ZwCreateSection(&hSection, SECTION_MAP_READ | SECTION_MAP_WRITE, 2017 &SectionAttrs, &nSectionSize, PAGE_READWRITE, SEC_COMMIT, NULL); 2018 switch (status) { 2019 case STATUS_SUCCESS: 2020 break; 2021 case STATUS_OBJECT_NAME_COLLISION: 2022 DbgP("section already created; returning success\n"); 2023 status = STATUS_SUCCESS; 2024 goto out; 2025 default: 2026 DbgP("ZwCreateSection failed with %08X\n", status); 2027 goto out; 2028 } 2029 out: 2030 DbgEx(); 2031 return status; 2032 } 2033 2034 NTSTATUS SharedMemoryFree( 2035 IN HANDLE hSection) 2036 { 2037 NTSTATUS status; 2038 DbgEn(); 2039 status = ZwClose(hSection); 2040 DbgEx(); 2041 return status; 2042 } 2043 2044 #ifdef __REACTOS__ 2045 NTSTATUS NTAPI nfs41_Start( 2046 #else 2047 NTSTATUS nfs41_Start( 2048 #endif 2049 IN OUT PRX_CONTEXT RxContext, 2050 IN OUT PRDBSS_DEVICE_OBJECT dev) 2051 { 2052 NTSTATUS status; 2053 NFS41GetDeviceExtension(RxContext, DevExt); 2054 2055 DbgEn(); 2056 2057 status = SharedMemoryInit(&DevExt->SharedMemorySection); 2058 if (status) { 2059 print_error("InitSharedMemory failed with %08X\n", status); 2060 status = STATUS_INSUFFICIENT_RESOURCES; 2061 goto out; 2062 } 2063 2064 InterlockedCompareExchange((PLONG)&nfs41_start_state, 2065 NFS41_START_DRIVER_STARTED, 2066 NFS41_START_DRIVER_START_IN_PROGRESS); 2067 out: 2068 DbgEx(); 2069 return status; 2070 } 2071 2072 #ifdef __REACTOS__ 2073 NTSTATUS NTAPI nfs41_Stop( 2074 #else 2075 NTSTATUS nfs41_Stop( 2076 #endif 2077 IN OUT PRX_CONTEXT RxContext, 2078 IN OUT PRDBSS_DEVICE_OBJECT dev) 2079 { 2080 NTSTATUS status; 2081 NFS41GetDeviceExtension(RxContext, DevExt); 2082 DbgEn(); 2083 status = SharedMemoryFree(DevExt->SharedMemorySection); 2084 DbgEx(); 2085 return status; 2086 } 2087 2088 NTSTATUS GetConnectionHandle( 2089 IN PUNICODE_STRING ConnectionName, 2090 IN PVOID EaBuffer, 2091 IN ULONG EaLength, 2092 OUT PHANDLE Handle) 2093 { 2094 NTSTATUS status; 2095 IO_STATUS_BLOCK IoStatusBlock; 2096 OBJECT_ATTRIBUTES ObjectAttributes; 2097 2098 #ifdef DEBUG_MOUNT 2099 DbgEn(); 2100 #endif 2101 InitializeObjectAttributes(&ObjectAttributes, ConnectionName, 2102 OBJ_CASE_INSENSITIVE|OBJ_KERNEL_HANDLE, NULL, NULL); 2103 2104 print_error("Len %d Buf %p\n", EaLength, EaBuffer); 2105 2106 status = ZwCreateFile(Handle, SYNCHRONIZE, &ObjectAttributes, 2107 &IoStatusBlock, NULL, FILE_ATTRIBUTE_NORMAL, 2108 FILE_SHARE_READ | FILE_SHARE_WRITE | FILE_SHARE_DELETE, 2109 FILE_OPEN_IF, 2110 FILE_CREATE_TREE_CONNECTION | FILE_SYNCHRONOUS_IO_NONALERT, 2111 EaBuffer, EaLength); 2112 2113 #ifdef DEBUG_MOUNT 2114 DbgEx(); 2115 #endif 2116 return status; 2117 } 2118 2119 NTSTATUS nfs41_GetConnectionInfoFromBuffer( 2120 IN PVOID Buffer, 2121 IN ULONG BufferLen, 2122 OUT PUNICODE_STRING pConnectionName, 2123 OUT PVOID *ppEaBuffer, 2124 OUT PULONG pEaLength) 2125 { 2126 NTSTATUS status = STATUS_SUCCESS; 2127 USHORT NameLength, EaPadding; 2128 ULONG EaLength, BufferLenExpected; 2129 PBYTE ptr; 2130 2131 /* make sure buffer is at least big enough for header */ 2132 if (BufferLen < sizeof(USHORT) + sizeof(USHORT) + sizeof(ULONG)) { 2133 status = STATUS_BAD_NETWORK_NAME; 2134 print_error("Invalid input buffer.\n"); 2135 pConnectionName->Length = pConnectionName->MaximumLength = 0; 2136 *ppEaBuffer = NULL; 2137 *pEaLength = 0; 2138 goto out; 2139 } 2140 2141 ptr = Buffer; 2142 NameLength = *(PUSHORT)ptr; 2143 ptr += sizeof(USHORT); 2144 EaPadding = *(PUSHORT)ptr; 2145 ptr += sizeof(USHORT); 2146 EaLength = *(PULONG)ptr; 2147 ptr += sizeof(ULONG); 2148 2149 /* validate buffer length */ 2150 BufferLenExpected = sizeof(USHORT) + sizeof(USHORT) + sizeof(ULONG) + 2151 NameLength + EaPadding + EaLength; 2152 if (BufferLen != BufferLenExpected) { 2153 status = STATUS_BAD_NETWORK_NAME; 2154 print_error("Received buffer of length %lu, but expected %lu bytes.\n", 2155 BufferLen, BufferLenExpected); 2156 pConnectionName->Length = pConnectionName->MaximumLength = 0; 2157 *ppEaBuffer = NULL; 2158 *pEaLength = 0; 2159 goto out; 2160 } 2161 2162 pConnectionName->Buffer = (PWCH)ptr; 2163 pConnectionName->Length = NameLength - sizeof(WCHAR); 2164 pConnectionName->MaximumLength = NameLength; 2165 2166 if (EaLength) 2167 *ppEaBuffer = ptr + NameLength + EaPadding; 2168 else 2169 *ppEaBuffer = NULL; 2170 *pEaLength = EaLength; 2171 2172 out: 2173 return status; 2174 } 2175 2176 NTSTATUS nfs41_CreateConnection( 2177 IN PRX_CONTEXT RxContext, 2178 OUT PBOOLEAN PostToFsp) 2179 { 2180 NTSTATUS status = STATUS_SUCCESS; 2181 HANDLE Handle = INVALID_HANDLE_VALUE; 2182 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 2183 PVOID Buffer = LowIoContext->ParamsFor.IoCtl.pInputBuffer, EaBuffer; 2184 ULONG BufferLen = LowIoContext->ParamsFor.IoCtl.InputBufferLength, EaLength; 2185 UNICODE_STRING FileName; 2186 BOOLEAN Wait = BooleanFlagOn(RxContext->Flags, RX_CONTEXT_FLAG_WAIT); 2187 2188 #ifdef DEBUG_MOUNT 2189 DbgEn(); 2190 #endif 2191 2192 if (!Wait) { 2193 //just post right now! 2194 DbgP("returning STATUS_PENDING\n"); 2195 *PostToFsp = TRUE; 2196 status = STATUS_PENDING; 2197 goto out; 2198 } 2199 2200 status = nfs41_GetConnectionInfoFromBuffer(Buffer, BufferLen, 2201 &FileName, &EaBuffer, &EaLength); 2202 if (status != STATUS_SUCCESS) 2203 goto out; 2204 2205 status = GetConnectionHandle(&FileName, EaBuffer, EaLength, &Handle); 2206 if (!status && Handle != INVALID_HANDLE_VALUE) 2207 ZwClose(Handle); 2208 out: 2209 #ifdef DEBUG_MOUNT 2210 DbgEx(); 2211 #endif 2212 return status; 2213 } 2214 2215 #ifdef ENABLE_TIMINGS 2216 void print_op_stat( 2217 const char *op_str, 2218 nfs41_timings *time, BOOLEAN clear) 2219 { 2220 DbgP("%-9s: num_ops=%-10d delta_ticks=%-10d size=%-10d\n", op_str, 2221 time->tops, time->tops ? time->ticks/time->tops : 0, 2222 time->sops ? time->size/time->sops : 0); 2223 if (clear) { 2224 time->tops = 0; 2225 time->ticks = 0; 2226 time->size = 0; 2227 time->sops = 0; 2228 } 2229 } 2230 #endif 2231 NTSTATUS nfs41_unmount( 2232 HANDLE session, 2233 DWORD version, 2234 DWORD timeout) 2235 { 2236 NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES; 2237 nfs41_updowncall_entry *entry; 2238 2239 #ifdef DEBUG_MOUNT 2240 DbgEn(); 2241 #endif 2242 status = nfs41_UpcallCreate(NFS41_UNMOUNT, NULL, session, 2243 INVALID_HANDLE_VALUE, version, NULL, &entry); 2244 SeDeleteClientSecurity(&entry->sec_ctx); 2245 if (status) goto out; 2246 2247 nfs41_UpcallWaitForReply(entry, timeout); 2248 RxFreePool(entry); 2249 out: 2250 #ifdef ENABLE_TIMINGS 2251 print_op_stat("lookup", &lookup, 1); 2252 print_op_stat("open", &open, 1); 2253 print_op_stat("close", &close, 1); 2254 print_op_stat("volume", &volume, 1); 2255 print_op_stat("getattr", &getattr, 1); 2256 print_op_stat("setattr", &setattr, 1); 2257 print_op_stat("getexattr", &getexattr, 1); 2258 print_op_stat("setexattr", &setexattr, 1); 2259 print_op_stat("readdir", &readdir, 1); 2260 print_op_stat("getacl", &getacl, 1); 2261 print_op_stat("setacl", &setacl, 1); 2262 print_op_stat("read", &read, 1); 2263 print_op_stat("write", &write, 1); 2264 print_op_stat("lock", &lock, 1); 2265 print_op_stat("unlock", &unlock, 1); 2266 #endif 2267 #ifdef DEBUG_MOUNT 2268 DbgEx(); 2269 #endif 2270 return status; 2271 } 2272 2273 NTSTATUS nfs41_DeleteConnection ( 2274 IN PRX_CONTEXT RxContext, 2275 OUT PBOOLEAN PostToFsp) 2276 { 2277 NTSTATUS status = STATUS_INVALID_PARAMETER; 2278 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 2279 PWCHAR ConnectName = LowIoContext->ParamsFor.IoCtl.pInputBuffer; 2280 ULONG ConnectNameLen = LowIoContext->ParamsFor.IoCtl.InputBufferLength; 2281 HANDLE Handle; 2282 UNICODE_STRING FileName; 2283 PFILE_OBJECT pFileObject; 2284 BOOLEAN Wait = BooleanFlagOn(RxContext->Flags, RX_CONTEXT_FLAG_WAIT); 2285 2286 #ifdef DEBUG_MOUNT 2287 DbgEn(); 2288 #endif 2289 2290 if (!Wait) { 2291 //just post right now! 2292 *PostToFsp = TRUE; 2293 DbgP("returning STATUS_PENDING\n"); 2294 status = STATUS_PENDING; 2295 goto out; 2296 } 2297 2298 FileName.Buffer = ConnectName; 2299 FileName.Length = (USHORT) ConnectNameLen - sizeof(WCHAR); 2300 FileName.MaximumLength = (USHORT) ConnectNameLen; 2301 2302 status = GetConnectionHandle(&FileName, NULL, 0, &Handle); 2303 if (status != STATUS_SUCCESS) 2304 goto out; 2305 2306 status = ObReferenceObjectByHandle(Handle, 0L, NULL, KernelMode, 2307 (PVOID *)&pFileObject, NULL); 2308 if (NT_SUCCESS(status)) { 2309 PV_NET_ROOT VNetRoot; 2310 2311 // VNetRoot exists as FOBx in the FsContext2 2312 VNetRoot = (PV_NET_ROOT) pFileObject->FsContext2; 2313 // make sure the node looks right 2314 if (NodeType(VNetRoot) == RDBSS_NTC_V_NETROOT) 2315 { 2316 #ifdef DEBUG_MOUNT 2317 DbgP("Calling RxFinalizeConnection for NetRoot %p from VNetRoot %p\n", 2318 VNetRoot->NetRoot, VNetRoot); 2319 #endif 2320 status = RxFinalizeConnection(VNetRoot->NetRoot, VNetRoot, TRUE); 2321 } 2322 else 2323 status = STATUS_BAD_NETWORK_NAME; 2324 2325 ObDereferenceObject(pFileObject); 2326 } 2327 ZwClose(Handle); 2328 out: 2329 #ifdef DEBUG_MOUNT 2330 DbgEx(); 2331 #endif 2332 return status; 2333 } 2334 2335 #ifdef __REACTOS__ 2336 NTSTATUS NTAPI nfs41_DevFcbXXXControlFile( 2337 #else 2338 NTSTATUS nfs41_DevFcbXXXControlFile( 2339 #endif 2340 IN OUT PRX_CONTEXT RxContext) 2341 { 2342 NTSTATUS status = STATUS_INVALID_DEVICE_REQUEST; 2343 UCHAR op = RxContext->MajorFunction; 2344 PLOWIO_CONTEXT io_ctx = &RxContext->LowIoContext; 2345 ULONG fsop = io_ctx->ParamsFor.FsCtl.FsControlCode, state; 2346 ULONG in_len = io_ctx->ParamsFor.IoCtl.InputBufferLength; 2347 DWORD *buf = io_ctx->ParamsFor.IoCtl.pInputBuffer; 2348 NFS41GetDeviceExtension(RxContext, DevExt); 2349 DWORD nfs41d_version = 0; 2350 2351 //DbgEn(); 2352 2353 print_ioctl(0, op); 2354 switch(op) { 2355 case IRP_MJ_FILE_SYSTEM_CONTROL: 2356 status = STATUS_INVALID_DEVICE_REQUEST; 2357 break; 2358 case IRP_MJ_DEVICE_CONTROL: 2359 case IRP_MJ_INTERNAL_DEVICE_CONTROL: 2360 print_fs_ioctl(0, fsop); 2361 switch (fsop) { 2362 case IOCTL_NFS41_INVALCACHE: 2363 nfs41_invalidate_cache(RxContext); 2364 status = STATUS_SUCCESS; 2365 break; 2366 case IOCTL_NFS41_READ: 2367 status = nfs41_upcall(RxContext); 2368 break; 2369 case IOCTL_NFS41_WRITE: 2370 status = nfs41_downcall(RxContext); 2371 break; 2372 case IOCTL_NFS41_ADDCONN: 2373 status = nfs41_CreateConnection(RxContext, &RxContext->PostRequest); 2374 break; 2375 case IOCTL_NFS41_DELCONN: 2376 if (RxContext->RxDeviceObject->NumberOfActiveFcbs > 0) { 2377 DbgP("device has open handles %d\n", 2378 RxContext->RxDeviceObject->NumberOfActiveFcbs); 2379 #ifdef __REACTOS__ 2380 if (RxContext->RxDeviceObject->pRxNetNameTable != NULL) 2381 { 2382 #define DUMP_FCB_TABLE_FROM_NETROOT(N) \ 2383 { \ 2384 USHORT Bucket2; \ 2385 BOOLEAN Release2 = FALSE; \ 2386 if (!RxIsFcbTableLockAcquired(&(N)->FcbTable)) \ 2387 { \ 2388 RxAcquireFcbTableLockExclusive(&(N)->FcbTable, TRUE); \ 2389 Release2 = TRUE; \ 2390 } \ 2391 for (Bucket2 = 0; Bucket2 < (N)->FcbTable.NumberOfBuckets; ++Bucket2) \ 2392 { \ 2393 PLIST_ENTRY Entry2; \ 2394 for (Entry2 = (N)->FcbTable.HashBuckets[Bucket2].Flink; \ 2395 Entry2 != &(N)->FcbTable.HashBuckets[Bucket2]; \ 2396 Entry2 = Entry2->Flink) \ 2397 { \ 2398 PFCB Fcb; \ 2399 Fcb = CONTAINING_RECORD(Entry2, FCB, FcbTableEntry.HashLinks); \ 2400 DbgP("Fcb: %p still has %d references\n", Fcb, Fcb->NodeReferenceCount); \ 2401 DbgP("It is for: %wZ\n", &Fcb->FcbTableEntry.Path); \ 2402 } \ 2403 } \ 2404 if (Release2) \ 2405 { \ 2406 RxReleaseFcbTableLock(&(N)->FcbTable); \ 2407 } \ 2408 } 2409 USHORT Bucket; 2410 BOOLEAN Release = FALSE; 2411 2412 if (!RxIsPrefixTableLockAcquired(RxContext->RxDeviceObject->pRxNetNameTable)) 2413 { 2414 RxAcquirePrefixTableLockExclusive(RxContext->RxDeviceObject->pRxNetNameTable, TRUE); 2415 Release = TRUE; 2416 } 2417 2418 for (Bucket = 0; Bucket < RxContext->RxDeviceObject->pRxNetNameTable->TableSize; ++Bucket) 2419 { 2420 PLIST_ENTRY Entry; 2421 2422 for (Entry = RxContext->RxDeviceObject->pRxNetNameTable->HashBuckets[Bucket].Flink; 2423 Entry != &RxContext->RxDeviceObject->pRxNetNameTable->HashBuckets[Bucket]; 2424 Entry = Entry->Flink) 2425 { 2426 PVOID Container; 2427 2428 Container = CONTAINING_RECORD(Entry, RX_PREFIX_ENTRY, HashLinks)->ContainingRecord; 2429 switch (NodeType(Container) & ~RX_SCAVENGER_MASK) 2430 { 2431 case RDBSS_NTC_NETROOT: 2432 { 2433 PNET_ROOT NetRoot; 2434 2435 NetRoot = Container; 2436 DUMP_FCB_TABLE_FROM_NETROOT(NetRoot); 2437 break; 2438 } 2439 2440 case RDBSS_NTC_V_NETROOT: 2441 { 2442 PV_NET_ROOT VNetRoot; 2443 2444 VNetRoot = Container; 2445 if (VNetRoot->NetRoot != NULL) 2446 { 2447 PNET_ROOT NetRoot; 2448 2449 NetRoot = VNetRoot->NetRoot; 2450 DUMP_FCB_TABLE_FROM_NETROOT(NetRoot); 2451 } 2452 break; 2453 } 2454 2455 default: 2456 { 2457 DbgP("Other node found: %x\n", NodeType(Container)); 2458 break; 2459 } 2460 } 2461 } 2462 } 2463 2464 if (Release) 2465 { 2466 RxReleasePrefixTableLock(RxContext->RxDeviceObject->pRxNetNameTable); 2467 } 2468 #undef DUMP_FCB_TABLE_FROM_NETROOT 2469 } 2470 else 2471 { 2472 DbgP("RxNetNameTable is NULL for: %p\n", RxContext->RxDeviceObject); 2473 } 2474 #endif 2475 status = STATUS_REDIRECTOR_HAS_OPEN_HANDLES; 2476 break; 2477 } 2478 status = nfs41_DeleteConnection(RxContext, &RxContext->PostRequest); 2479 break; 2480 case IOCTL_NFS41_GETSTATE: 2481 state = RDR_NULL_STATE; 2482 2483 if (io_ctx->ParamsFor.IoCtl.OutputBufferLength >= sizeof(ULONG)) { 2484 // map the states to control app's equivalents 2485 print_driver_state(nfs41_start_state); 2486 switch (nfs41_start_state) { 2487 case NFS41_START_DRIVER_STARTABLE: 2488 case NFS41_START_DRIVER_STOPPED: 2489 state = RDR_STOPPED; 2490 break; 2491 case NFS41_START_DRIVER_START_IN_PROGRESS: 2492 state = RDR_STARTING; 2493 break; 2494 case NFS41_START_DRIVER_STARTED: 2495 state = RDR_STARTED; 2496 break; 2497 } 2498 *(ULONG *)io_ctx->ParamsFor.IoCtl.pOutputBuffer = state; 2499 RxContext->InformationToReturn = sizeof(ULONG); 2500 status = STATUS_SUCCESS; 2501 } else 2502 status = STATUS_INVALID_PARAMETER; 2503 break; 2504 case IOCTL_NFS41_START: 2505 print_driver_state(nfs41_start_state); 2506 if (in_len >= sizeof(DWORD)) { 2507 RtlCopyMemory(&nfs41d_version, buf, sizeof(DWORD)); 2508 DbgP("NFS41 Daemon sent start request with version %d\n", 2509 nfs41d_version); 2510 DbgP("Currently used NFS41 Daemon version is %d\n", 2511 DevExt->nfs41d_version); 2512 DevExt->nfs41d_version = nfs41d_version; 2513 } 2514 switch(nfs41_start_state) { 2515 case NFS41_START_DRIVER_STARTABLE: 2516 (nfs41_start_driver_state)InterlockedCompareExchange( 2517 (PLONG)&nfs41_start_state, 2518 NFS41_START_DRIVER_START_IN_PROGRESS, 2519 NFS41_START_DRIVER_STARTABLE); 2520 //lack of break is intentional 2521 case NFS41_START_DRIVER_START_IN_PROGRESS: 2522 status = RxStartMinirdr(RxContext, &RxContext->PostRequest); 2523 if (status == STATUS_REDIRECTOR_STARTED) { 2524 DbgP("redirector started\n"); 2525 status = STATUS_SUCCESS; 2526 } else if (status == STATUS_PENDING && 2527 RxContext->PostRequest == TRUE) { 2528 DbgP("RxStartMinirdr pending %08lx\n", status); 2529 status = STATUS_MORE_PROCESSING_REQUIRED; 2530 } 2531 break; 2532 case NFS41_START_DRIVER_STARTED: 2533 status = STATUS_SUCCESS; 2534 break; 2535 default: 2536 status = STATUS_INVALID_PARAMETER; 2537 } 2538 break; 2539 case IOCTL_NFS41_STOP: 2540 if (nfs41_start_state == NFS41_START_DRIVER_STARTED) 2541 nfs41_shutdown_daemon(DevExt->nfs41d_version); 2542 if (RxContext->RxDeviceObject->NumberOfActiveFcbs > 0) { 2543 DbgP("device has open handles %d\n", 2544 RxContext->RxDeviceObject->NumberOfActiveFcbs); 2545 status = STATUS_REDIRECTOR_HAS_OPEN_HANDLES; 2546 break; 2547 } 2548 2549 state = (nfs41_start_driver_state)InterlockedCompareExchange( 2550 (PLONG)&nfs41_start_state, 2551 NFS41_START_DRIVER_STARTABLE, 2552 NFS41_START_DRIVER_STARTED); 2553 2554 status = RxStopMinirdr(RxContext, &RxContext->PostRequest); 2555 DbgP("RxStopMinirdr status %08lx\n", status); 2556 if (status == STATUS_PENDING && RxContext->PostRequest == TRUE ) 2557 status = STATUS_MORE_PROCESSING_REQUIRED; 2558 break; 2559 default: 2560 status = STATUS_INVALID_DEVICE_REQUEST; 2561 }; 2562 break; 2563 default: 2564 status = STATUS_INVALID_DEVICE_REQUEST; 2565 }; 2566 2567 //DbgEx(); 2568 return status; 2569 } 2570 2571 #ifndef __REACTOS__ 2572 NTSTATUS _nfs41_CreateSrvCall( 2573 PMRX_SRVCALL_CALLBACK_CONTEXT pCallbackContext) 2574 { 2575 #else 2576 NTSTATUS NTAPI _nfs41_CreateSrvCall( 2577 PVOID pContext) 2578 { 2579 PMRX_SRVCALL_CALLBACK_CONTEXT pCallbackContext = pContext; 2580 #endif 2581 NTSTATUS status = STATUS_SUCCESS; 2582 PMRX_SRVCALL_CALLBACK_CONTEXT SCCBC = pCallbackContext; 2583 PMRX_SRV_CALL pSrvCall; 2584 PMRX_SRVCALLDOWN_STRUCTURE SrvCalldownStructure = 2585 (PMRX_SRVCALLDOWN_STRUCTURE)(SCCBC->SrvCalldownStructure); 2586 PNFS41_SERVER_ENTRY pServerEntry = NULL; 2587 2588 #ifdef DEBUG_MOUNT 2589 DbgEn(); 2590 #endif 2591 2592 pSrvCall = SrvCalldownStructure->SrvCall; 2593 2594 ASSERT( pSrvCall ); 2595 ASSERT( NodeType(pSrvCall) == RDBSS_NTC_SRVCALL ); 2596 print_srv_call(0, pSrvCall); 2597 2598 // validate the server name with the test name of 'pnfs' 2599 #ifdef DEBUG_MOUNT 2600 DbgP("SrvCall: Connection Name Length: %d %wZ\n", 2601 pSrvCall->pSrvCallName->Length, pSrvCall->pSrvCallName); 2602 #endif 2603 2604 if (pSrvCall->pSrvCallName->Length > SERVER_NAME_BUFFER_SIZE) { 2605 print_error("Server name '%wZ' too long for server entry (max %u)\n", 2606 pSrvCall->pSrvCallName, SERVER_NAME_BUFFER_SIZE); 2607 status = STATUS_NAME_TOO_LONG; 2608 goto out; 2609 } 2610 2611 /* Let's create our own representation of the server */ 2612 pServerEntry = (PNFS41_SERVER_ENTRY)RxAllocatePoolWithTag(PagedPool, 2613 sizeof(NFS41_SERVER_ENTRY), NFS41_MM_POOLTAG); 2614 if (pServerEntry == NULL) { 2615 status = STATUS_INSUFFICIENT_RESOURCES; 2616 goto out; 2617 } 2618 RtlZeroMemory(pServerEntry, sizeof(NFS41_SERVER_ENTRY)); 2619 2620 pServerEntry->Name.Buffer = pServerEntry->NameBuffer; 2621 pServerEntry->Name.Length = pSrvCall->pSrvCallName->Length; 2622 pServerEntry->Name.MaximumLength = SERVER_NAME_BUFFER_SIZE; 2623 RtlCopyMemory(pServerEntry->Name.Buffer, pSrvCall->pSrvCallName->Buffer, 2624 pServerEntry->Name.Length); 2625 2626 pCallbackContext->RecommunicateContext = pServerEntry; 2627 #ifdef __REACTOS__ 2628 InterlockedExchangePointer((void * volatile *)&pServerEntry->pRdbssSrvCall, pSrvCall); 2629 #else 2630 InterlockedExchangePointer(&pServerEntry->pRdbssSrvCall, pSrvCall); 2631 #endif 2632 2633 out: 2634 SCCBC->Status = status; 2635 SrvCalldownStructure->CallBack(SCCBC); 2636 2637 #ifdef DEBUG_MOUNT 2638 DbgEx(); 2639 #endif 2640 return status; 2641 } 2642 2643 #ifdef __REACTOS__ 2644 VOID NTAPI _nfs41_CreateSrvCall_v( 2645 PVOID pCallbackContext) 2646 { 2647 _nfs41_CreateSrvCall(pCallbackContext); 2648 } 2649 #endif 2650 2651 #ifdef __REACTOS__ 2652 NTSTATUS NTAPI nfs41_CreateSrvCall( 2653 #else 2654 NTSTATUS nfs41_CreateSrvCall( 2655 #endif 2656 PMRX_SRV_CALL pSrvCall, 2657 PMRX_SRVCALL_CALLBACK_CONTEXT pCallbackContext) 2658 { 2659 NTSTATUS status; 2660 2661 ASSERT( pSrvCall ); 2662 ASSERT( NodeType(pSrvCall) == RDBSS_NTC_SRVCALL ); 2663 2664 if (IoGetCurrentProcess() == RxGetRDBSSProcess()) { 2665 DbgP("executing with RDBSS context\n"); 2666 status = _nfs41_CreateSrvCall(pCallbackContext); 2667 } else { 2668 status = RxDispatchToWorkerThread(nfs41_dev, DelayedWorkQueue, 2669 #ifdef __REACTOS__ 2670 _nfs41_CreateSrvCall_v, pCallbackContext); 2671 #else 2672 _nfs41_CreateSrvCall, pCallbackContext); 2673 #endif 2674 if (status != STATUS_SUCCESS) { 2675 print_error("RxDispatchToWorkerThread returned status %08lx\n", 2676 status); 2677 pCallbackContext->Status = status; 2678 pCallbackContext->SrvCalldownStructure->CallBack(pCallbackContext); 2679 status = STATUS_PENDING; 2680 } 2681 } 2682 /* RDBSS expects MRxCreateSrvCall to return STATUS_PENDING */ 2683 if (status == STATUS_SUCCESS) 2684 status = STATUS_PENDING; 2685 2686 return status; 2687 } 2688 2689 #ifdef __REACTOS__ 2690 NTSTATUS NTAPI nfs41_SrvCallWinnerNotify( 2691 #else 2692 NTSTATUS nfs41_SrvCallWinnerNotify( 2693 #endif 2694 IN OUT PMRX_SRV_CALL pSrvCall, 2695 IN BOOLEAN ThisMinirdrIsTheWinner, 2696 IN OUT PVOID pSrvCallContext) 2697 { 2698 NTSTATUS status = STATUS_SUCCESS; 2699 PNFS41_SERVER_ENTRY pServerEntry; 2700 2701 pServerEntry = (PNFS41_SERVER_ENTRY)pSrvCallContext; 2702 2703 if (!ThisMinirdrIsTheWinner) { 2704 ASSERT(1); 2705 goto out; 2706 } 2707 2708 pSrvCall->Context = pServerEntry; 2709 out: 2710 return status; 2711 } 2712 2713 NTSTATUS map_mount_errors( 2714 DWORD status) 2715 { 2716 switch (status) { 2717 case NO_ERROR: return STATUS_SUCCESS; 2718 case ERROR_NETWORK_UNREACHABLE: return STATUS_NETWORK_UNREACHABLE; 2719 case ERROR_BAD_NET_RESP: return STATUS_UNEXPECTED_NETWORK_ERROR; 2720 case ERROR_BAD_NET_NAME: return STATUS_BAD_NETWORK_NAME; 2721 case ERROR_BAD_NETPATH: return STATUS_BAD_NETWORK_PATH; 2722 default: 2723 print_error("failed to map windows error %d to NTSTATUS; " 2724 "defaulting to STATUS_INSUFFICIENT_RESOURCES\n", status); 2725 return STATUS_INSUFFICIENT_RESOURCES; 2726 } 2727 } 2728 2729 NTSTATUS nfs41_mount( 2730 PNFS41_MOUNT_CONFIG config, 2731 DWORD sec_flavor, 2732 PHANDLE session, 2733 DWORD *version, 2734 PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs) 2735 { 2736 NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES; 2737 nfs41_updowncall_entry *entry; 2738 2739 #ifdef DEBUG_MOUNT 2740 DbgEn(); 2741 DbgP("Server Name %wZ Mount Point %wZ SecFlavor %d\n", 2742 &config->SrvName, &config->MntPt, sec_flavor); 2743 #endif 2744 status = nfs41_UpcallCreate(NFS41_MOUNT, NULL, *session, 2745 INVALID_HANDLE_VALUE, *version, &config->MntPt, &entry); 2746 if (status) goto out; 2747 2748 entry->u.Mount.srv_name = &config->SrvName; 2749 entry->u.Mount.root = &config->MntPt; 2750 entry->u.Mount.rsize = config->ReadSize; 2751 entry->u.Mount.wsize = config->WriteSize; 2752 entry->u.Mount.sec_flavor = sec_flavor; 2753 entry->u.Mount.FsAttrs = FsAttrs; 2754 2755 status = nfs41_UpcallWaitForReply(entry, config->timeout); 2756 SeDeleteClientSecurity(&entry->sec_ctx); 2757 if (status) goto out; 2758 *session = entry->session; 2759 if (entry->u.Mount.lease_time > config->timeout) 2760 config->timeout = entry->u.Mount.lease_time; 2761 2762 /* map windows ERRORs to NTSTATUS */ 2763 status = map_mount_errors(entry->status); 2764 if (status == STATUS_SUCCESS) 2765 *version = entry->version; 2766 RxFreePool(entry); 2767 out: 2768 #ifdef DEBUG_MOUNT 2769 DbgEx(); 2770 #endif 2771 return status; 2772 } 2773 2774 /* TODO: move mount config stuff to another file -cbodley */ 2775 2776 void nfs41_MountConfig_InitDefaults( 2777 OUT PNFS41_MOUNT_CONFIG Config) 2778 { 2779 RtlZeroMemory(Config, sizeof(NFS41_MOUNT_CONFIG)); 2780 2781 Config->ReadSize = MOUNT_CONFIG_RW_SIZE_DEFAULT; 2782 Config->WriteSize = MOUNT_CONFIG_RW_SIZE_DEFAULT; 2783 Config->ReadOnly = FALSE; 2784 Config->write_thru = FALSE; 2785 Config->nocache = FALSE; 2786 Config->SrvName.Length = 0; 2787 Config->SrvName.MaximumLength = SERVER_NAME_BUFFER_SIZE; 2788 Config->SrvName.Buffer = Config->srv_buffer; 2789 Config->MntPt.Length = 0; 2790 Config->MntPt.MaximumLength = MAX_PATH; 2791 Config->MntPt.Buffer = Config->mntpt_buffer; 2792 Config->SecFlavor.Length = 0; 2793 Config->SecFlavor.MaximumLength = MAX_SEC_FLAVOR_LEN; 2794 Config->SecFlavor.Buffer = Config->sec_flavor; 2795 RtlCopyUnicodeString(&Config->SecFlavor, &AUTH_SYS_NAME); 2796 Config->timeout = UPCALL_TIMEOUT_DEFAULT; 2797 } 2798 2799 NTSTATUS nfs41_MountConfig_ParseBoolean( 2800 IN PFILE_FULL_EA_INFORMATION Option, 2801 IN PUNICODE_STRING usValue, 2802 OUT PBOOLEAN Value) 2803 { 2804 NTSTATUS status = STATUS_SUCCESS; 2805 2806 /* if no value is specified, assume TRUE 2807 * if a value is specified, it must be a '1' */ 2808 if (Option->EaValueLength == 0 || *usValue->Buffer == L'1') 2809 *Value = TRUE; 2810 else 2811 *Value = FALSE; 2812 2813 DbgP(" '%ls' -> '%wZ' -> %u\n", 2814 (LPWSTR)Option->EaName, usValue, *Value); 2815 return status; 2816 } 2817 2818 NTSTATUS nfs41_MountConfig_ParseDword( 2819 IN PFILE_FULL_EA_INFORMATION Option, 2820 IN PUNICODE_STRING usValue, 2821 OUT PDWORD Value, 2822 IN DWORD Minimum, 2823 IN DWORD Maximum) 2824 { 2825 NTSTATUS status = STATUS_INVALID_PARAMETER; 2826 LPWSTR Name = (LPWSTR)Option->EaName; 2827 2828 if (Option->EaValueLength) { 2829 status = RtlUnicodeStringToInteger(usValue, 0, Value); 2830 if (status == STATUS_SUCCESS) { 2831 #ifdef IMPOSE_MINMAX_RWSIZES 2832 if (*Value < Minimum) 2833 *Value = Minimum; 2834 if (*Value > Maximum) 2835 *Value = Maximum; 2836 DbgP(" '%ls' -> '%wZ' -> %lu\n", Name, usValue, *Value); 2837 #endif 2838 } 2839 else 2840 print_error("Failed to convert %s='%wZ' to unsigned long.\n", 2841 Name, usValue); 2842 } 2843 2844 return status; 2845 } 2846 2847 NTSTATUS nfs41_MountConfig_ParseOptions( 2848 IN PFILE_FULL_EA_INFORMATION EaBuffer, 2849 IN ULONG EaLength, 2850 IN OUT PNFS41_MOUNT_CONFIG Config) 2851 { 2852 NTSTATUS status = STATUS_SUCCESS; 2853 PFILE_FULL_EA_INFORMATION Option; 2854 LPWSTR Name; 2855 size_t NameLen; 2856 UNICODE_STRING usValue; 2857 Option = EaBuffer; 2858 while (status == STATUS_SUCCESS) { 2859 Name = (LPWSTR)Option->EaName; 2860 NameLen = Option->EaNameLength/sizeof(WCHAR); 2861 2862 usValue.Length = usValue.MaximumLength = Option->EaValueLength; 2863 usValue.Buffer = (PWCH)(Option->EaName + 2864 Option->EaNameLength + sizeof(WCHAR)); 2865 2866 if (wcsncmp(L"ro", Name, NameLen) == 0) { 2867 status = nfs41_MountConfig_ParseBoolean(Option, &usValue, 2868 &Config->ReadOnly); 2869 } 2870 else if (wcsncmp(L"writethru", Name, NameLen) == 0) { 2871 status = nfs41_MountConfig_ParseBoolean(Option, &usValue, 2872 &Config->write_thru); 2873 } 2874 else if (wcsncmp(L"nocache", Name, NameLen) == 0) { 2875 status = nfs41_MountConfig_ParseBoolean(Option, &usValue, 2876 &Config->nocache); 2877 } 2878 else if (wcsncmp(L"timeout", Name, NameLen) == 0) { 2879 status = nfs41_MountConfig_ParseDword(Option, &usValue, 2880 &Config->timeout, UPCALL_TIMEOUT_DEFAULT, 2881 UPCALL_TIMEOUT_DEFAULT); 2882 } 2883 else if (wcsncmp(L"rsize", Name, NameLen) == 0) { 2884 status = nfs41_MountConfig_ParseDword(Option, &usValue, 2885 &Config->ReadSize, MOUNT_CONFIG_RW_SIZE_MIN, 2886 MOUNT_CONFIG_RW_SIZE_MAX); 2887 } 2888 else if (wcsncmp(L"wsize", Name, NameLen) == 0) { 2889 status = nfs41_MountConfig_ParseDword(Option, &usValue, 2890 &Config->WriteSize, MOUNT_CONFIG_RW_SIZE_MIN, 2891 MOUNT_CONFIG_RW_SIZE_MAX); 2892 } 2893 else if (wcsncmp(L"srvname", Name, NameLen) == 0) { 2894 if (usValue.Length > Config->SrvName.MaximumLength) 2895 status = STATUS_NAME_TOO_LONG; 2896 else 2897 RtlCopyUnicodeString(&Config->SrvName, &usValue); 2898 } 2899 else if (wcsncmp(L"mntpt", Name, NameLen) == 0) { 2900 if (usValue.Length > Config->MntPt.MaximumLength) 2901 status = STATUS_NAME_TOO_LONG; 2902 else 2903 RtlCopyUnicodeString(&Config->MntPt, &usValue); 2904 } 2905 else if (wcsncmp(L"sec", Name, NameLen) == 0) { 2906 if (usValue.Length > Config->SecFlavor.MaximumLength) 2907 status = STATUS_NAME_TOO_LONG; 2908 else 2909 RtlCopyUnicodeString(&Config->SecFlavor, &usValue); 2910 } 2911 else { 2912 status = STATUS_INVALID_PARAMETER; 2913 print_error("Unrecognized option '%ls' -> '%wZ'\n", 2914 Name, usValue); 2915 } 2916 2917 if (Option->NextEntryOffset == 0) 2918 break; 2919 2920 Option = (PFILE_FULL_EA_INFORMATION) 2921 ((PBYTE)Option + Option->NextEntryOffset); 2922 } 2923 2924 return status; 2925 } 2926 2927 NTSTATUS has_nfs_prefix( 2928 IN PUNICODE_STRING SrvCallName, 2929 IN PUNICODE_STRING NetRootName) 2930 { 2931 NTSTATUS status = STATUS_BAD_NETWORK_NAME; 2932 2933 if (NetRootName->Length == SrvCallName->Length + NfsPrefix.Length) { 2934 const UNICODE_STRING NetRootPrefix = { 2935 NfsPrefix.Length, 2936 NetRootName->MaximumLength - SrvCallName->Length, 2937 &NetRootName->Buffer[SrvCallName->Length/2] 2938 }; 2939 if (RtlCompareUnicodeString(&NetRootPrefix, &NfsPrefix, FALSE) == 0) 2940 status = STATUS_SUCCESS; 2941 } 2942 return status; 2943 } 2944 2945 NTSTATUS map_sec_flavor( 2946 IN PUNICODE_STRING sec_flavor_name, 2947 OUT PDWORD sec_flavor) 2948 { 2949 if (RtlCompareUnicodeString(sec_flavor_name, &AUTH_SYS_NAME, FALSE) == 0) 2950 *sec_flavor = RPCSEC_AUTH_SYS; 2951 else if (RtlCompareUnicodeString(sec_flavor_name, &AUTHGSS_KRB5_NAME, FALSE) == 0) 2952 *sec_flavor = RPCSEC_AUTHGSS_KRB5; 2953 else if (RtlCompareUnicodeString(sec_flavor_name, &AUTHGSS_KRB5I_NAME, FALSE) == 0) 2954 *sec_flavor = RPCSEC_AUTHGSS_KRB5I; 2955 else if (RtlCompareUnicodeString(sec_flavor_name, &AUTHGSS_KRB5P_NAME, FALSE) == 0) 2956 *sec_flavor = RPCSEC_AUTHGSS_KRB5P; 2957 else return STATUS_INVALID_PARAMETER; 2958 return STATUS_SUCCESS; 2959 } 2960 2961 NTSTATUS nfs41_GetLUID( 2962 PLUID id) 2963 { 2964 NTSTATUS status = STATUS_SUCCESS; 2965 SECURITY_SUBJECT_CONTEXT sec_ctx; 2966 SECURITY_QUALITY_OF_SERVICE sec_qos; 2967 SECURITY_CLIENT_CONTEXT clnt_sec_ctx; 2968 2969 SeCaptureSubjectContext(&sec_ctx); 2970 sec_qos.ContextTrackingMode = SECURITY_DYNAMIC_TRACKING; 2971 sec_qos.ImpersonationLevel = SecurityIdentification; 2972 sec_qos.Length = sizeof(SECURITY_QUALITY_OF_SERVICE); 2973 sec_qos.EffectiveOnly = 0; 2974 status = SeCreateClientSecurityFromSubjectContext(&sec_ctx, &sec_qos, 1, 2975 &clnt_sec_ctx); 2976 if (status) { 2977 print_error("nfs41_GetLUID: SeCreateClientSecurityFromSubjectContext " 2978 "failed %x\n", status); 2979 goto release_sec_ctx; 2980 } 2981 status = SeQueryAuthenticationIdToken(clnt_sec_ctx.ClientToken, id); 2982 if (status) { 2983 print_error("SeQueryAuthenticationIdToken failed %x\n", status); 2984 goto release_clnt_sec_ctx; 2985 } 2986 release_clnt_sec_ctx: 2987 SeDeleteClientSecurity(&clnt_sec_ctx); 2988 release_sec_ctx: 2989 SeReleaseSubjectContext(&sec_ctx); 2990 2991 return status; 2992 } 2993 2994 NTSTATUS nfs41_get_sec_ctx( 2995 IN enum _SECURITY_IMPERSONATION_LEVEL level, 2996 OUT PSECURITY_CLIENT_CONTEXT out_ctx) 2997 { 2998 NTSTATUS status; 2999 SECURITY_SUBJECT_CONTEXT ctx; 3000 SECURITY_QUALITY_OF_SERVICE sec_qos; 3001 3002 SeCaptureSubjectContext(&ctx); 3003 sec_qos.ContextTrackingMode = SECURITY_STATIC_TRACKING; 3004 sec_qos.ImpersonationLevel = level; 3005 sec_qos.Length = sizeof(SECURITY_QUALITY_OF_SERVICE); 3006 sec_qos.EffectiveOnly = 0; 3007 status = SeCreateClientSecurityFromSubjectContext(&ctx, &sec_qos, 1, out_ctx); 3008 if (status != STATUS_SUCCESS) { 3009 print_error("SeCreateClientSecurityFromSubjectContext " 3010 "failed with %x\n", status); 3011 } 3012 #ifdef DEBUG_MOUNT 3013 DbgP("Created client security token %p\n", out_ctx->ClientToken); 3014 #endif 3015 SeReleaseSubjectContext(&ctx); 3016 3017 return status; 3018 } 3019 3020 #ifdef __REACTOS__ 3021 NTSTATUS NTAPI nfs41_CreateVNetRoot( 3022 #else 3023 NTSTATUS nfs41_CreateVNetRoot( 3024 #endif 3025 IN OUT PMRX_CREATENETROOT_CONTEXT pCreateNetRootContext) 3026 { 3027 NTSTATUS status = STATUS_SUCCESS; 3028 NFS41_MOUNT_CONFIG *Config; 3029 __notnull PMRX_V_NET_ROOT pVNetRoot = (PMRX_V_NET_ROOT) 3030 pCreateNetRootContext->pVNetRoot; 3031 __notnull PMRX_NET_ROOT pNetRoot = pVNetRoot->pNetRoot; 3032 __notnull PMRX_SRV_CALL pSrvCall = pNetRoot->pSrvCall; 3033 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 3034 NFS41GetVNetRootExtension(pVNetRoot); 3035 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 3036 NFS41GetNetRootExtension(pNetRoot); 3037 NFS41GetDeviceExtension(pCreateNetRootContext->RxContext,DevExt); 3038 DWORD nfs41d_version = DevExt->nfs41d_version; 3039 nfs41_mount_entry *existing_mount = NULL; 3040 LUID luid; 3041 BOOLEAN found_existing_mount = FALSE, found_matching_flavor = FALSE; 3042 3043 ASSERT((NodeType(pNetRoot) == RDBSS_NTC_NETROOT) && 3044 (NodeType(pNetRoot->pSrvCall) == RDBSS_NTC_SRVCALL)); 3045 3046 #ifdef DEBUG_MOUNT 3047 DbgEn(); 3048 print_srv_call(0, pSrvCall); 3049 print_net_root(0, pNetRoot); 3050 print_v_net_root(0, pVNetRoot); 3051 3052 DbgP("pVNetRoot=%p pNetRoot=%p pSrvCall=%p\n", pVNetRoot, pNetRoot, pSrvCall); 3053 DbgP("pNetRoot=%wZ Type=%d pSrvCallName=%wZ VirtualNetRootStatus=0x%x " 3054 "NetRootStatus=0x%x\n", pNetRoot->pNetRootName, 3055 pNetRoot->Type, pSrvCall->pSrvCallName, 3056 pCreateNetRootContext->VirtualNetRootStatus, 3057 pCreateNetRootContext->NetRootStatus); 3058 #endif 3059 3060 if (pNetRoot->Type != NET_ROOT_DISK && pNetRoot->Type != NET_ROOT_WILD) { 3061 print_error("nfs41_CreateVNetRoot: Unsupported NetRoot Type %u\n", 3062 pNetRoot->Type); 3063 status = STATUS_NOT_SUPPORTED; 3064 goto out; 3065 } 3066 3067 pVNetRootContext->session = INVALID_HANDLE_VALUE; 3068 3069 /* In order to cooperate with other network providers, we must 3070 * only claim paths of the form '\\server\nfs4\path' */ 3071 status = has_nfs_prefix(pSrvCall->pSrvCallName, pNetRoot->pNetRootName); 3072 if (status) { 3073 print_error("nfs41_CreateVNetRoot: NetRootName %wZ doesn't match " 3074 "'\\nfs4'!\n", pNetRoot->pNetRootName); 3075 goto out; 3076 } 3077 pNetRoot->MRxNetRootState = MRX_NET_ROOT_STATE_GOOD; 3078 pNetRoot->DeviceType = FILE_DEVICE_DISK; 3079 3080 Config = RxAllocatePoolWithTag(NonPagedPool, 3081 sizeof(NFS41_MOUNT_CONFIG), NFS41_MM_POOLTAG); 3082 if (Config == NULL) { 3083 status = STATUS_INSUFFICIENT_RESOURCES; 3084 goto out; 3085 } 3086 nfs41_MountConfig_InitDefaults(Config); 3087 3088 if (pCreateNetRootContext->RxContext->Create.EaLength) { 3089 /* parse the extended attributes for mount options */ 3090 status = nfs41_MountConfig_ParseOptions( 3091 pCreateNetRootContext->RxContext->Create.EaBuffer, 3092 pCreateNetRootContext->RxContext->Create.EaLength, 3093 Config); 3094 if (status != STATUS_SUCCESS) 3095 goto out_free; 3096 pVNetRootContext->read_only = Config->ReadOnly; 3097 pVNetRootContext->write_thru = Config->write_thru; 3098 pVNetRootContext->nocache = Config->nocache; 3099 } else { 3100 /* use the SRV_CALL name (without leading \) as the hostname */ 3101 Config->SrvName.Buffer = pSrvCall->pSrvCallName->Buffer + 1; 3102 Config->SrvName.Length = 3103 pSrvCall->pSrvCallName->Length - sizeof(WCHAR); 3104 Config->SrvName.MaximumLength = 3105 pSrvCall->pSrvCallName->MaximumLength - sizeof(WCHAR); 3106 } 3107 pVNetRootContext->MountPathLen = Config->MntPt.Length; 3108 pVNetRootContext->timeout = Config->timeout; 3109 3110 status = map_sec_flavor(&Config->SecFlavor, &pVNetRootContext->sec_flavor); 3111 if (status != STATUS_SUCCESS) { 3112 DbgP("Invalid rpcsec security flavor %wZ\n", &Config->SecFlavor); 3113 goto out_free; 3114 } 3115 3116 status = nfs41_GetLUID(&luid); 3117 if (status) 3118 goto out_free; 3119 3120 if (!pNetRootContext->mounts_init) { 3121 #ifdef DEBUG_MOUNT 3122 DbgP("Initializing mount array\n"); 3123 #endif 3124 ExInitializeFastMutex(&pNetRootContext->mountLock); 3125 InitializeListHead(&pNetRootContext->mounts.head); 3126 pNetRootContext->mounts_init = TRUE; 3127 } else { 3128 PLIST_ENTRY pEntry; 3129 3130 ExAcquireFastMutex(&pNetRootContext->mountLock); 3131 pEntry = &pNetRootContext->mounts.head; 3132 pEntry = pEntry->Flink; 3133 while (pEntry != NULL) { 3134 existing_mount = (nfs41_mount_entry *)CONTAINING_RECORD(pEntry, 3135 nfs41_mount_entry, next); 3136 #ifdef DEBUG_MOUNT 3137 DbgP("comparing %x.%x with %x.%x\n", luid.HighPart, luid.LowPart, 3138 existing_mount->login_id.HighPart, existing_mount->login_id.LowPart); 3139 #endif 3140 if (RtlEqualLuid(&luid, &existing_mount->login_id)) { 3141 #ifdef DEBUG_MOUNT 3142 DbgP("Found a matching LUID entry\n"); 3143 #endif 3144 found_existing_mount = TRUE; 3145 switch(pVNetRootContext->sec_flavor) { 3146 case RPCSEC_AUTH_SYS: 3147 if (existing_mount->authsys_session != INVALID_HANDLE_VALUE) 3148 pVNetRootContext->session = 3149 existing_mount->authsys_session; 3150 break; 3151 case RPCSEC_AUTHGSS_KRB5: 3152 if (existing_mount->gssi_session != INVALID_HANDLE_VALUE) 3153 pVNetRootContext->session = existing_mount->gss_session; 3154 break; 3155 case RPCSEC_AUTHGSS_KRB5I: 3156 if (existing_mount->gss_session != INVALID_HANDLE_VALUE) 3157 pVNetRootContext->session = existing_mount->gssi_session; 3158 break; 3159 case RPCSEC_AUTHGSS_KRB5P: 3160 if (existing_mount->gssp_session != INVALID_HANDLE_VALUE) 3161 pVNetRootContext->session = existing_mount->gssp_session; 3162 break; 3163 } 3164 if (pVNetRootContext->session && 3165 pVNetRootContext->session != INVALID_HANDLE_VALUE) 3166 found_matching_flavor = 1; 3167 break; 3168 } 3169 if (pEntry->Flink == &pNetRootContext->mounts.head) 3170 break; 3171 pEntry = pEntry->Flink; 3172 } 3173 ExReleaseFastMutex(&pNetRootContext->mountLock); 3174 #ifdef DEBUG_MOUNT 3175 if (!found_matching_flavor) 3176 DbgP("Didn't find matching security flavor\n"); 3177 #endif 3178 } 3179 3180 /* send the mount upcall */ 3181 status = nfs41_mount(Config, pVNetRootContext->sec_flavor, 3182 &pVNetRootContext->session, &nfs41d_version, 3183 &pVNetRootContext->FsAttrs); 3184 if (status != STATUS_SUCCESS) { 3185 BOOLEAN MountsEmpty; 3186 nfs41_IsListEmpty(pNetRootContext->mountLock, 3187 pNetRootContext->mounts, MountsEmpty); 3188 if (!found_existing_mount && MountsEmpty) 3189 pNetRootContext->mounts_init = FALSE; 3190 pVNetRootContext->session = INVALID_HANDLE_VALUE; 3191 goto out_free; 3192 } 3193 pVNetRootContext->timeout = Config->timeout; 3194 3195 if (!found_existing_mount) { 3196 /* create a new mount entry and add it to the list */ 3197 nfs41_mount_entry *entry; 3198 entry = RxAllocatePoolWithTag(NonPagedPool, sizeof(nfs41_mount_entry), 3199 NFS41_MM_POOLTAG_MOUNT); 3200 if (entry == NULL) { 3201 status = STATUS_INSUFFICIENT_RESOURCES; 3202 goto out_free; 3203 } 3204 entry->authsys_session = entry->gss_session = 3205 entry->gssi_session = entry->gssp_session = INVALID_HANDLE_VALUE; 3206 switch (pVNetRootContext->sec_flavor) { 3207 case RPCSEC_AUTH_SYS: 3208 entry->authsys_session = pVNetRootContext->session; break; 3209 case RPCSEC_AUTHGSS_KRB5: 3210 entry->gss_session = pVNetRootContext->session; break; 3211 case RPCSEC_AUTHGSS_KRB5I: 3212 entry->gssi_session = pVNetRootContext->session; break; 3213 case RPCSEC_AUTHGSS_KRB5P: 3214 entry->gssp_session = pVNetRootContext->session; break; 3215 } 3216 RtlCopyLuid(&entry->login_id, &luid); 3217 nfs41_AddEntry(pNetRootContext->mountLock, 3218 pNetRootContext->mounts, entry); 3219 } else if (!found_matching_flavor) { 3220 ASSERT(existing_mount != NULL); 3221 /* modify existing mount entry */ 3222 #ifdef DEBUG_MOUNT 3223 DbgP("Using existing %d flavor session 0x%x\n", 3224 pVNetRootContext->sec_flavor); 3225 #endif 3226 switch (pVNetRootContext->sec_flavor) { 3227 case RPCSEC_AUTH_SYS: 3228 existing_mount->authsys_session = pVNetRootContext->session; break; 3229 case RPCSEC_AUTHGSS_KRB5: 3230 existing_mount->gss_session = pVNetRootContext->session; break; 3231 case RPCSEC_AUTHGSS_KRB5I: 3232 existing_mount->gssi_session = pVNetRootContext->session; break; 3233 case RPCSEC_AUTHGSS_KRB5P: 3234 existing_mount->gssp_session = pVNetRootContext->session; break; 3235 } 3236 } 3237 pNetRootContext->nfs41d_version = nfs41d_version; 3238 #ifdef DEBUG_MOUNT 3239 DbgP("Saving new session 0x%x\n", pVNetRootContext->session); 3240 #endif 3241 #ifdef STORE_MOUNT_SEC_CONTEXT 3242 status = nfs41_get_sec_ctx(SecurityImpersonation, 3243 &pVNetRootContext->mount_sec_ctx); 3244 #endif 3245 3246 out_free: 3247 RxFreePool(Config); 3248 out: 3249 pCreateNetRootContext->VirtualNetRootStatus = status; 3250 if (pNetRoot->Context == NULL) 3251 pCreateNetRootContext->NetRootStatus = status; 3252 pCreateNetRootContext->Callback(pCreateNetRootContext); 3253 3254 /* RDBSS expects that MRxCreateVNetRoot returns STATUS_PENDING 3255 * on success or failure */ 3256 status = STATUS_PENDING; 3257 #ifdef DEBUG_MOUNT 3258 DbgEx(); 3259 #endif 3260 return status; 3261 } 3262 3263 #ifdef __REACTOS__ 3264 VOID NTAPI nfs41_ExtractNetRootName( 3265 #else 3266 VOID nfs41_ExtractNetRootName( 3267 #endif 3268 IN PUNICODE_STRING FilePathName, 3269 IN PMRX_SRV_CALL SrvCall, 3270 OUT PUNICODE_STRING NetRootName, 3271 OUT PUNICODE_STRING RestOfName OPTIONAL) 3272 { 3273 ULONG length = FilePathName->Length; 3274 PWCH w = FilePathName->Buffer; 3275 PWCH wlimit = (PWCH)(((PCHAR)w)+length); 3276 PWCH wlow; 3277 3278 w += (SrvCall->pSrvCallName->Length/sizeof(WCHAR)); 3279 NetRootName->Buffer = wlow = w; 3280 /* parse the entire path into NetRootName */ 3281 #if USE_ENTIRE_PATH 3282 w = wlimit; 3283 #else 3284 for (;;) { 3285 if (w >= wlimit) 3286 break; 3287 if ((*w == OBJ_NAME_PATH_SEPARATOR) && (w != wlow)) 3288 break; 3289 w++; 3290 } 3291 #endif 3292 NetRootName->Length = NetRootName->MaximumLength 3293 = (USHORT)((PCHAR)w - (PCHAR)wlow); 3294 #ifdef DEBUG_MOUNT 3295 DbgP("In: pSrvCall %p PathName=%wZ SrvCallName=%wZ Out: NetRootName=%wZ\n", 3296 SrvCall, FilePathName, SrvCall->pSrvCallName, NetRootName); 3297 #endif 3298 return; 3299 3300 } 3301 3302 #ifdef __REACTOS__ 3303 NTSTATUS NTAPI nfs41_FinalizeSrvCall( 3304 #else 3305 NTSTATUS nfs41_FinalizeSrvCall( 3306 #endif 3307 PMRX_SRV_CALL pSrvCall, 3308 BOOLEAN Force) 3309 { 3310 NTSTATUS status = STATUS_SUCCESS; 3311 PNFS41_SERVER_ENTRY pServerEntry = (PNFS41_SERVER_ENTRY)(pSrvCall->Context); 3312 3313 #ifdef DEBUG_MOUNT 3314 DbgEn(); 3315 #endif 3316 print_srv_call(0, pSrvCall); 3317 3318 if (pSrvCall->Context == NULL) 3319 goto out; 3320 3321 #ifndef __REACTOS__ 3322 InterlockedCompareExchangePointer(&pServerEntry->pRdbssSrvCall, 3323 NULL, pSrvCall); 3324 #else 3325 InterlockedCompareExchangePointer((void * volatile *)&pServerEntry->pRdbssSrvCall, 3326 NULL, pSrvCall); 3327 #endif 3328 RxFreePool(pServerEntry); 3329 3330 pSrvCall->Context = NULL; 3331 out: 3332 #ifdef DEBUG_MOUNT 3333 DbgEx(); 3334 #endif 3335 return status; 3336 } 3337 3338 #ifdef __REACTOS__ 3339 NTSTATUS NTAPI nfs41_FinalizeNetRoot( 3340 #else 3341 NTSTATUS nfs41_FinalizeNetRoot( 3342 #endif 3343 IN OUT PMRX_NET_ROOT pNetRoot, 3344 IN PBOOLEAN ForceDisconnect) 3345 { 3346 NTSTATUS status = STATUS_SUCCESS; 3347 PNFS41_NETROOT_EXTENSION pNetRootContext = 3348 NFS41GetNetRootExtension((PMRX_NET_ROOT)pNetRoot); 3349 nfs41_updowncall_entry *tmp; 3350 nfs41_mount_entry *mount_tmp; 3351 3352 #ifdef DEBUG_MOUNT 3353 DbgEn(); 3354 print_net_root(1, pNetRoot); 3355 #endif 3356 3357 if (pNetRoot->Type != NET_ROOT_DISK && pNetRoot->Type != NET_ROOT_WILD) { 3358 status = STATUS_NOT_SUPPORTED; 3359 goto out; 3360 } 3361 3362 if (pNetRootContext == NULL || !pNetRootContext->mounts_init) { 3363 print_error("nfs41_FinalizeNetRoot: No valid session established\n"); 3364 goto out; 3365 } 3366 3367 if (pNetRoot->NumberOfFcbs > 0 || pNetRoot->NumberOfSrvOpens > 0) { 3368 print_error("%d open Fcbs %d open SrvOpens\n", pNetRoot->NumberOfFcbs, 3369 pNetRoot->NumberOfSrvOpens); 3370 goto out; 3371 } 3372 3373 do { 3374 nfs41_GetFirstMountEntry(pNetRootContext->mountLock, 3375 pNetRootContext->mounts, mount_tmp); 3376 if (mount_tmp == NULL) 3377 break; 3378 #ifdef DEBUG_MOUNT 3379 DbgP("Removing entry luid %x.%x from mount list\n", 3380 mount_tmp->login_id.HighPart, mount_tmp->login_id.LowPart); 3381 #endif 3382 if (mount_tmp->authsys_session != INVALID_HANDLE_VALUE) { 3383 status = nfs41_unmount(mount_tmp->authsys_session, 3384 pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT); 3385 if (status) 3386 print_error("nfs41_unmount AUTH_SYS failed with %d\n", status); 3387 } 3388 if (mount_tmp->gss_session != INVALID_HANDLE_VALUE) { 3389 status = nfs41_unmount(mount_tmp->gss_session, 3390 pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT); 3391 if (status) 3392 print_error("nfs41_unmount RPCSEC_GSS_KRB5 failed with %d\n", 3393 status); 3394 } 3395 if (mount_tmp->gssi_session != INVALID_HANDLE_VALUE) { 3396 status = nfs41_unmount(mount_tmp->gssi_session, 3397 pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT); 3398 if (status) 3399 print_error("nfs41_unmount RPCSEC_GSS_KRB5I failed with %d\n", 3400 status); 3401 } 3402 if (mount_tmp->gssp_session != INVALID_HANDLE_VALUE) { 3403 status = nfs41_unmount(mount_tmp->gssp_session, 3404 pNetRootContext->nfs41d_version, UPCALL_TIMEOUT_DEFAULT); 3405 if (status) 3406 print_error("nfs41_unmount RPCSEC_GSS_KRB5P failed with %d\n", 3407 status); 3408 } 3409 nfs41_RemoveEntry(pNetRootContext->mountLock, mount_tmp); 3410 RxFreePool(mount_tmp); 3411 } while (1); 3412 /* ignore any errors from unmount */ 3413 status = STATUS_SUCCESS; 3414 3415 // check if there is anything waiting in the upcall or downcall queue 3416 do { 3417 nfs41_GetFirstEntry(upcallLock, upcall, tmp); 3418 if (tmp != NULL) { 3419 DbgP("Removing entry from upcall list\n"); 3420 nfs41_RemoveEntry(upcallLock, tmp); 3421 tmp->status = STATUS_INSUFFICIENT_RESOURCES; 3422 KeSetEvent(&tmp->cond, 0, FALSE); 3423 } else 3424 break; 3425 } while (1); 3426 3427 do { 3428 nfs41_GetFirstEntry(downcallLock, downcall, tmp); 3429 if (tmp != NULL) { 3430 DbgP("Removing entry from downcall list\n"); 3431 nfs41_RemoveEntry(downcallLock, tmp); 3432 tmp->status = STATUS_INSUFFICIENT_RESOURCES; 3433 KeSetEvent(&tmp->cond, 0, FALSE); 3434 } else 3435 break; 3436 } while (1); 3437 out: 3438 #ifdef DEBUG_MOUNT 3439 DbgEx(); 3440 #endif 3441 return status; 3442 } 3443 3444 #ifdef __REACTOS__ 3445 NTSTATUS NTAPI nfs41_FinalizeVNetRoot( 3446 #else 3447 NTSTATUS nfs41_FinalizeVNetRoot( 3448 #endif 3449 IN OUT PMRX_V_NET_ROOT pVNetRoot, 3450 IN PBOOLEAN ForceDisconnect) 3451 { 3452 NTSTATUS status = STATUS_SUCCESS; 3453 PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 3454 NFS41GetVNetRootExtension(pVNetRoot); 3455 #ifdef DEBUG_MOUNT 3456 DbgEn(); 3457 print_v_net_root(1, pVNetRoot); 3458 #endif 3459 if (pVNetRoot->pNetRoot->Type != NET_ROOT_DISK && 3460 pVNetRoot->pNetRoot->Type != NET_ROOT_WILD) 3461 status = STATUS_NOT_SUPPORTED; 3462 #ifdef STORE_MOUNT_SEC_CONTEXT 3463 else if (pVNetRootContext->session != INVALID_HANDLE_VALUE) { 3464 #ifdef DEBUG_MOUNT 3465 DbgP("nfs41_FinalizeVNetRoot: deleting security context: %p\n", 3466 pVNetRootContext->mount_sec_ctx.ClientToken); 3467 #endif 3468 SeDeleteClientSecurity(&pVNetRootContext->mount_sec_ctx); 3469 } 3470 #endif 3471 #ifdef DEBUG_MOUNT 3472 DbgEx(); 3473 #endif 3474 return status; 3475 } 3476 3477 BOOLEAN isDataAccess( 3478 ACCESS_MASK mask) 3479 { 3480 if (mask & (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA)) 3481 return TRUE; 3482 return FALSE; 3483 } 3484 3485 BOOLEAN isOpen2Create( 3486 ULONG disposition) 3487 { 3488 if (disposition == FILE_CREATE || disposition == FILE_OPEN_IF || 3489 disposition == FILE_OVERWRITE_IF || disposition == FILE_SUPERSEDE) 3490 return TRUE; 3491 return FALSE; 3492 } 3493 3494 BOOLEAN isFilenameTooLong( 3495 PUNICODE_STRING name, 3496 PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext) 3497 { 3498 PFILE_FS_ATTRIBUTE_INFORMATION attrs = &pVNetRootContext->FsAttrs; 3499 LONG len = attrs->MaximumComponentNameLength, count = 1, i; 3500 PWCH p = name->Buffer; 3501 for (i = 0; i < name->Length / 2; i++) { 3502 if (p[0] == L'\\') count = 1; 3503 else { 3504 if (p[0] == L'\0') return FALSE; 3505 if (count > len) return TRUE; 3506 count++; 3507 } 3508 p++; 3509 } 3510 return FALSE; 3511 } 3512 3513 BOOLEAN isStream( 3514 PUNICODE_STRING name) 3515 { 3516 LONG i; 3517 PWCH p = name->Buffer; 3518 for (i = 0; i < name->Length / 2; i++) { 3519 if (p[0] == L':') return TRUE; 3520 else if (p[0] == L'\0') return FALSE; 3521 p++; 3522 } 3523 return FALSE; 3524 } 3525 3526 BOOLEAN areOpenParamsValid(NT_CREATE_PARAMETERS *params) 3527 { 3528 /* from ms-fsa page 52 */ 3529 if ((params->CreateOptions & FILE_DELETE_ON_CLOSE) && 3530 !(params->DesiredAccess & DELETE)) 3531 return FALSE; 3532 if ((params->CreateOptions & FILE_DIRECTORY_FILE) && 3533 (params->Disposition == FILE_SUPERSEDE || 3534 params->Disposition == FILE_OVERWRITE || 3535 params->Disposition == FILE_OVERWRITE_IF)) 3536 return FALSE; 3537 if ((params->CreateOptions & FILE_NO_INTERMEDIATE_BUFFERING) && 3538 (params->DesiredAccess & FILE_APPEND_DATA) && 3539 !(params->DesiredAccess & FILE_WRITE_DATA)) 3540 return FALSE; 3541 /* from ms-fsa 3.1.5.1.1 page 56 */ 3542 if ((params->CreateOptions & FILE_DIRECTORY_FILE) && 3543 (params->FileAttributes & FILE_ATTRIBUTE_TEMPORARY)) 3544 return FALSE; 3545 return TRUE; 3546 } 3547 3548 NTSTATUS map_open_errors( 3549 DWORD status, 3550 USHORT len) 3551 { 3552 switch (status) { 3553 case NO_ERROR: return STATUS_SUCCESS; 3554 case ERROR_ACCESS_DENIED: 3555 if (len > 0) return STATUS_ACCESS_DENIED; 3556 else return STATUS_SUCCESS; 3557 case ERROR_INVALID_REPARSE_DATA: 3558 case ERROR_INVALID_NAME: return STATUS_OBJECT_NAME_INVALID; 3559 case ERROR_FILE_EXISTS: return STATUS_OBJECT_NAME_COLLISION; 3560 case ERROR_FILE_INVALID: return STATUS_FILE_INVALID; 3561 case ERROR_FILE_NOT_FOUND: return STATUS_OBJECT_NAME_NOT_FOUND; 3562 case ERROR_FILENAME_EXCED_RANGE: return STATUS_NAME_TOO_LONG; 3563 case ERROR_NETWORK_ACCESS_DENIED: return STATUS_NETWORK_ACCESS_DENIED; 3564 case ERROR_PATH_NOT_FOUND: return STATUS_OBJECT_PATH_NOT_FOUND; 3565 case ERROR_BAD_NETPATH: return STATUS_BAD_NETWORK_PATH; 3566 case ERROR_SHARING_VIOLATION: return STATUS_SHARING_VIOLATION; 3567 case ERROR_REPARSE: return STATUS_REPARSE; 3568 case ERROR_TOO_MANY_LINKS: return STATUS_TOO_MANY_LINKS; 3569 case ERROR_DIRECTORY: return STATUS_FILE_IS_A_DIRECTORY; 3570 case ERROR_BAD_FILE_TYPE: return STATUS_NOT_A_DIRECTORY; 3571 default: 3572 print_error("[ERROR] nfs41_Create: upcall returned %d returning " 3573 "STATUS_INSUFFICIENT_RESOURCES\n", status); 3574 case ERROR_OUTOFMEMORY: return STATUS_INSUFFICIENT_RESOURCES; 3575 } 3576 } 3577 3578 DWORD map_disposition_to_create_retval( 3579 DWORD disposition, 3580 DWORD errno) 3581 { 3582 switch(disposition) { 3583 case FILE_SUPERSEDE: 3584 if (errno == ERROR_FILE_NOT_FOUND) return FILE_CREATED; 3585 else return FILE_SUPERSEDED; 3586 case FILE_CREATE: return FILE_CREATED; 3587 case FILE_OPEN: return FILE_OPENED; 3588 case FILE_OPEN_IF: 3589 if (errno == ERROR_FILE_NOT_FOUND) return FILE_CREATED; 3590 else return FILE_OPENED; 3591 case FILE_OVERWRITE: return FILE_OVERWRITTEN; 3592 case FILE_OVERWRITE_IF: 3593 if (errno == ERROR_FILE_NOT_FOUND) return FILE_CREATED; 3594 else return FILE_OVERWRITTEN; 3595 default: 3596 print_error("unknown disposition %d\n", disposition); 3597 return FILE_OPENED; 3598 } 3599 } 3600 3601 static BOOLEAN create_should_pass_ea( 3602 IN PFILE_FULL_EA_INFORMATION ea, 3603 IN ULONG disposition) 3604 { 3605 /* don't pass cygwin EAs */ 3606 if (AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength) 3607 || AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength) 3608 || AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) 3609 return FALSE; 3610 /* only set EAs on file creation */ 3611 return disposition == FILE_SUPERSEDE || disposition == FILE_CREATE 3612 || disposition == FILE_OPEN_IF || disposition == FILE_OVERWRITE 3613 || disposition == FILE_OVERWRITE_IF; 3614 } 3615 3616 NTSTATUS check_nfs41_create_args( 3617 IN PRX_CONTEXT RxContext) 3618 { 3619 NTSTATUS status = STATUS_SUCCESS; 3620 PNT_CREATE_PARAMETERS params = &RxContext->Create.NtCreateParameters; 3621 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 3622 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 3623 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 3624 __notnull PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs = 3625 &pVNetRootContext->FsAttrs; 3626 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 3627 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 3628 __notnull PMRX_FCB Fcb = RxContext->pFcb; 3629 __notnull PNFS41_FCB nfs41_fcb = (PNFS41_FCB)Fcb->Context; 3630 PFILE_FULL_EA_INFORMATION ea = (PFILE_FULL_EA_INFORMATION) 3631 RxContext->CurrentIrp->AssociatedIrp.SystemBuffer; 3632 3633 if (Fcb->pNetRoot->Type != NET_ROOT_DISK && 3634 Fcb->pNetRoot->Type != NET_ROOT_WILD) { 3635 print_error("nfs41_Create: Unsupported NetRoot Type %u\n", 3636 Fcb->pNetRoot->Type); 3637 status = STATUS_NOT_SUPPORTED; 3638 goto out; 3639 } 3640 3641 if (FlagOn(Fcb->FcbState, FCB_STATE_PAGING_FILE )) { 3642 print_error("FCB_STATE_PAGING_FILE not implemented\n"); 3643 status = STATUS_NOT_IMPLEMENTED; 3644 goto out; 3645 } 3646 3647 if (!pNetRootContext->mounts_init) { 3648 print_error("nfs41_Create: No valid session established\n"); 3649 status = STATUS_INSUFFICIENT_RESOURCES; 3650 goto out; 3651 } 3652 3653 if (isStream(SrvOpen->pAlreadyPrefixedName)) { 3654 status = STATUS_NOT_SUPPORTED; 3655 goto out; 3656 } 3657 3658 if (pVNetRootContext->read_only && 3659 (params->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA))) { 3660 status = STATUS_NETWORK_ACCESS_DENIED; 3661 goto out; 3662 } 3663 3664 /* if FCB was marked for deletion and opened multiple times, as soon 3665 * as first close happen, FCB transitions into delete_pending state 3666 * no more opens allowed 3667 */ 3668 if (Fcb->OpenCount && nfs41_fcb->DeletePending) { 3669 status = STATUS_DELETE_PENDING; 3670 goto out; 3671 } 3672 3673 /* ms-fsa: 3.1.5.1.2.1 page 68 */ 3674 if (Fcb->OpenCount && nfs41_fcb->StandardInfo.DeletePending && 3675 !(params->ShareAccess & FILE_SHARE_DELETE) && 3676 (params->DesiredAccess & (FILE_EXECUTE | FILE_READ_DATA | 3677 FILE_WRITE_DATA | FILE_APPEND_DATA))) { 3678 status = STATUS_SHARING_VIOLATION; 3679 goto out; 3680 } 3681 3682 /* rdbss seems miss this sharing_violation check */ 3683 if (Fcb->OpenCount && params->Disposition == FILE_SUPERSEDE) { 3684 #ifdef __REACTOS__ 3685 if ((!RxContext->CurrentIrpSp->FileObject->SharedRead && 3686 (params->DesiredAccess & FILE_READ_DATA)) || 3687 ((!RxContext->CurrentIrpSp->FileObject->SharedWrite && 3688 (params->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA | 3689 FILE_WRITE_ATTRIBUTES))) || 3690 (!RxContext->CurrentIrpSp->FileObject->SharedDelete && 3691 (params->DesiredAccess & DELETE)))) { 3692 #else 3693 if ((!RxContext->CurrentIrpSp->FileObject->SharedRead && 3694 (params->DesiredAccess & FILE_READ_DATA)) || 3695 (!RxContext->CurrentIrpSp->FileObject->SharedWrite && 3696 (params->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA | 3697 FILE_WRITE_ATTRIBUTES)) || 3698 (!RxContext->CurrentIrpSp->FileObject->SharedDelete && 3699 (params->DesiredAccess & DELETE)))) { 3700 #endif 3701 status = STATUS_SHARING_VIOLATION; 3702 goto out; 3703 } 3704 } 3705 if (isFilenameTooLong(SrvOpen->pAlreadyPrefixedName, pVNetRootContext)) { 3706 status = STATUS_OBJECT_NAME_INVALID; 3707 goto out; 3708 } 3709 3710 if (!areOpenParamsValid(params)) { 3711 status = STATUS_INVALID_PARAMETER; 3712 goto out; 3713 } 3714 3715 /* from ms-fsa 3.1.5.1.1 page 56 */ 3716 if ((params->CreateOptions & FILE_DELETE_ON_CLOSE) && 3717 (params->FileAttributes & FILE_ATTRIBUTE_READONLY)) { 3718 status = STATUS_CANNOT_DELETE; 3719 goto out; 3720 } 3721 3722 if (ea) { 3723 /* ignore cygwin EAs when checking support and access */ 3724 if (!AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength) && 3725 !AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength) && 3726 !AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) { 3727 if (!(FsAttrs->FileSystemAttributes & FILE_SUPPORTS_EXTENDED_ATTRIBUTES)) { 3728 status = STATUS_EAS_NOT_SUPPORTED; 3729 goto out; 3730 } 3731 } 3732 } else if (RxContext->CurrentIrpSp->Parameters.Create.EaLength) { 3733 status = STATUS_INVALID_PARAMETER; 3734 goto out; 3735 } 3736 3737 out: 3738 return status; 3739 } 3740 3741 #ifdef __REACTOS__ 3742 NTSTATUS NTAPI nfs41_Create( 3743 #else 3744 NTSTATUS nfs41_Create( 3745 #endif 3746 IN OUT PRX_CONTEXT RxContext) 3747 { 3748 NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES; 3749 nfs41_updowncall_entry *entry = NULL; 3750 PNT_CREATE_PARAMETERS params = &RxContext->Create.NtCreateParameters; 3751 PFILE_FULL_EA_INFORMATION ea = (PFILE_FULL_EA_INFORMATION) 3752 RxContext->CurrentIrp->AssociatedIrp.SystemBuffer; 3753 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 3754 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 3755 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 3756 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 3757 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 3758 __notnull PMRX_FCB Fcb = RxContext->pFcb; 3759 __notnull PNFS41_FCB nfs41_fcb = (PNFS41_FCB)Fcb->Context; 3760 PNFS41_FOBX nfs41_fobx = NULL; 3761 BOOLEAN oldDeletePending = nfs41_fcb->StandardInfo.DeletePending; 3762 #ifdef ENABLE_TIMINGS 3763 LARGE_INTEGER t1, t2; 3764 t1 = KeQueryPerformanceCounter(NULL); 3765 #endif 3766 3767 ASSERT( NodeType(SrvOpen) == RDBSS_NTC_SRVOPEN ); 3768 3769 #ifdef DEBUG_OPEN 3770 DbgEn(); 3771 print_debug_header(RxContext); 3772 print_nt_create_params(1, RxContext->Create.NtCreateParameters); 3773 if (ea) print_ea_info(0, ea); 3774 #endif 3775 3776 status = check_nfs41_create_args(RxContext); 3777 if (status) goto out; 3778 3779 #if defined(STORE_MOUNT_SEC_CONTEXT) && defined (USE_MOUNT_SEC_CONTEXT) 3780 status = nfs41_UpcallCreate(NFS41_OPEN, &pVNetRootContext->mount_sec_ctx, 3781 #else 3782 status = nfs41_UpcallCreate(NFS41_OPEN, NULL, 3783 #endif 3784 pVNetRootContext->session, INVALID_HANDLE_VALUE, 3785 pNetRootContext->nfs41d_version, 3786 SrvOpen->pAlreadyPrefixedName, &entry); 3787 if (status) goto out; 3788 3789 entry->u.Open.access_mask = params->DesiredAccess; 3790 entry->u.Open.access_mode = params->ShareAccess; 3791 entry->u.Open.attrs = params->FileAttributes; 3792 if (!(params->CreateOptions & FILE_DIRECTORY_FILE)) 3793 entry->u.Open.attrs |= FILE_ATTRIBUTE_ARCHIVE; 3794 entry->u.Open.disp = params->Disposition; 3795 entry->u.Open.copts = params->CreateOptions; 3796 entry->u.Open.srv_open = SrvOpen; 3797 /* treat the NfsActOnLink ea as FILE_OPEN_REPARSE_POINT */ 3798 if ((ea && AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength)) || 3799 (entry->u.Open.access_mask & DELETE)) 3800 entry->u.Open.copts |= FILE_OPEN_REPARSE_POINT; 3801 if (isDataAccess(params->DesiredAccess) || isOpen2Create(params->Disposition)) 3802 entry->u.Open.open_owner_id = InterlockedIncrement(&open_owner_id); 3803 // if we are creating a file check if nfsv3attributes were passed in 3804 if (params->Disposition != FILE_OPEN && params->Disposition != FILE_OVERWRITE) { 3805 entry->u.Open.mode = 0777; 3806 if (ea && AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength)) { 3807 nfs3_attrs *attrs = (nfs3_attrs *)(ea->EaName + ea->EaNameLength + 1); 3808 #ifdef DEBUG_OPEN 3809 DbgP("creating file with mode %o\n", attrs->mode); 3810 #endif 3811 entry->u.Open.mode = attrs->mode; 3812 } 3813 if (params->FileAttributes & FILE_ATTRIBUTE_READONLY) 3814 entry->u.Open.mode = 0444; 3815 } 3816 if (entry->u.Open.disp == FILE_CREATE && ea && 3817 AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) { 3818 /* for a cygwin symlink, given as a unicode string */ 3819 entry->u.Open.symlink.Buffer = (PWCH)(ea->EaName + ea->EaNameLength + 1); 3820 entry->u.Open.symlink.MaximumLength = entry->u.Open.symlink.Length = ea->EaValueLength; 3821 } 3822 retry_on_link: 3823 if (ea && create_should_pass_ea(ea, params->Disposition)) { 3824 /* lock the extended attribute buffer for read access in user space */ 3825 entry->u.Open.EaMdl = IoAllocateMdl(ea, 3826 RxContext->CurrentIrpSp->Parameters.Create.EaLength, 3827 FALSE, FALSE, NULL); 3828 if (entry->u.Open.EaMdl == NULL) { 3829 status = STATUS_INTERNAL_ERROR; 3830 RxFreePool(entry); 3831 goto out; 3832 } 3833 entry->u.Open.EaMdl->MdlFlags |= MDL_MAPPING_CAN_FAIL; 3834 MmProbeAndLockPages(entry->u.Open.EaMdl, KernelMode, IoModifyAccess); 3835 } 3836 3837 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 3838 #ifndef USE_MOUNT_SEC_CONTEXT 3839 SeDeleteClientSecurity(&entry->sec_ctx); 3840 #endif 3841 if (status) goto out; 3842 3843 if (entry->u.Open.EaMdl) { 3844 MmUnlockPages(entry->u.Open.EaMdl); 3845 IoFreeMdl(entry->u.Open.EaMdl); 3846 } 3847 3848 if (entry->status == NO_ERROR && entry->errno == ERROR_REPARSE) { 3849 /* symbolic link handling. when attempting to open a symlink when the 3850 * FILE_OPEN_REPARSE_POINT flag is not set, replace the filename with 3851 * the symlink target's by calling RxPrepareToReparseSymbolicLink() 3852 * and returning STATUS_REPARSE. the object manager will attempt to 3853 * open the new path, and return its handle for the original open */ 3854 PRDBSS_DEVICE_OBJECT DeviceObject = RxContext->RxDeviceObject; 3855 PV_NET_ROOT VNetRoot = (PV_NET_ROOT) 3856 RxContext->pRelevantSrvOpen->pVNetRoot; 3857 PUNICODE_STRING VNetRootPrefix = &VNetRoot->PrefixEntry.Prefix; 3858 UNICODE_STRING AbsPath; 3859 PCHAR buf; 3860 BOOLEAN ReparseRequired; 3861 3862 /* allocate the string for RxPrepareToReparseSymbolicLink(), and 3863 * format an absolute path "DeviceName+VNetRootName+symlink" */ 3864 AbsPath.Length = DeviceObject->DeviceName.Length + 3865 VNetRootPrefix->Length + entry->u.Open.symlink.Length; 3866 AbsPath.MaximumLength = AbsPath.Length + sizeof(UNICODE_NULL); 3867 AbsPath.Buffer = RxAllocatePoolWithTag(NonPagedPool, 3868 AbsPath.MaximumLength, NFS41_MM_POOLTAG); 3869 if (AbsPath.Buffer == NULL) { 3870 status = STATUS_INSUFFICIENT_RESOURCES; 3871 goto out_free; 3872 } 3873 3874 buf = (PCHAR)AbsPath.Buffer; 3875 RtlCopyMemory(buf, DeviceObject->DeviceName.Buffer, 3876 DeviceObject->DeviceName.Length); 3877 buf += DeviceObject->DeviceName.Length; 3878 RtlCopyMemory(buf, VNetRootPrefix->Buffer, VNetRootPrefix->Length); 3879 buf += VNetRootPrefix->Length; 3880 RtlCopyMemory(buf, entry->u.Open.symlink.Buffer, 3881 entry->u.Open.symlink.Length); 3882 RxFreePool(entry->u.Open.symlink.Buffer); 3883 buf += entry->u.Open.symlink.Length; 3884 *(PWCHAR)buf = UNICODE_NULL; 3885 3886 status = RxPrepareToReparseSymbolicLink(RxContext, 3887 entry->u.Open.symlink_embedded, &AbsPath, TRUE, &ReparseRequired); 3888 #ifdef DEBUG_OPEN 3889 DbgP("RxPrepareToReparseSymbolicLink(%u, '%wZ') returned %08lX, " 3890 "FileName is '%wZ'\n", entry->u.Open.symlink_embedded, 3891 &AbsPath, status, &RxContext->CurrentIrpSp->FileObject->FileName); 3892 #endif 3893 if (status == STATUS_SUCCESS) { 3894 /* if a reparse is not required, reopen the link itself. this 3895 * happens with operations on cygwin symlinks, where the reparse 3896 * flag is not set */ 3897 if (!ReparseRequired) { 3898 entry->u.Open.symlink.Length = 0; 3899 entry->u.Open.copts |= FILE_OPEN_REPARSE_POINT; 3900 goto retry_on_link; 3901 } 3902 status = STATUS_REPARSE; 3903 } 3904 goto out_free; 3905 } 3906 3907 status = map_open_errors(entry->status, 3908 SrvOpen->pAlreadyPrefixedName->Length); 3909 if (status) { 3910 #ifdef DEBUG_OPEN 3911 print_open_error(1, status); 3912 #endif 3913 goto out_free; 3914 } 3915 3916 if (!RxIsFcbAcquiredExclusive(Fcb)) { 3917 ASSERT(!RxIsFcbAcquiredShared(Fcb)); 3918 RxAcquireExclusiveFcbResourceInMRx(Fcb); 3919 } 3920 3921 RxContext->pFobx = RxCreateNetFobx(RxContext, SrvOpen); 3922 if (RxContext->pFobx == NULL) { 3923 status = STATUS_INSUFFICIENT_RESOURCES; 3924 goto out_free; 3925 } 3926 #ifdef DEBUG_OPEN 3927 DbgP("nfs41_Create: created FOBX %p\n", RxContext->pFobx); 3928 #endif 3929 nfs41_fobx = (PNFS41_FOBX)(RxContext->pFobx)->Context; 3930 nfs41_fobx->nfs41_open_state = entry->open_state; 3931 #ifndef USE_MOUNT_SEC_CONTEXT 3932 status = nfs41_get_sec_ctx(SecurityImpersonation, &nfs41_fobx->sec_ctx); 3933 if (status) 3934 goto out_free; 3935 #else 3936 RtlCopyMemory(&nfs41_fobx->sec_ctx, &pVNetRootContext->mount_sec_ctx, 3937 sizeof(nfs41_fobx->sec_ctx)); 3938 #endif 3939 3940 // we get attributes only for data access and file (not directories) 3941 if (Fcb->OpenCount == 0 || 3942 (Fcb->OpenCount > 0 && 3943 nfs41_fcb->changeattr != entry->ChangeTime)) { 3944 FCB_INIT_PACKET InitPacket; 3945 RX_FILE_TYPE StorageType = FileTypeNotYetKnown; 3946 RtlCopyMemory(&nfs41_fcb->BasicInfo, &entry->u.Open.binfo, 3947 sizeof(entry->u.Open.binfo)); 3948 RtlCopyMemory(&nfs41_fcb->StandardInfo, &entry->u.Open.sinfo, 3949 sizeof(entry->u.Open.sinfo)); 3950 nfs41_fcb->mode = entry->u.Open.mode; 3951 nfs41_fcb->changeattr = entry->ChangeTime; 3952 if (((params->CreateOptions & FILE_DELETE_ON_CLOSE) && 3953 !pVNetRootContext->read_only) || oldDeletePending) 3954 nfs41_fcb->StandardInfo.DeletePending = TRUE; 3955 3956 RxFormInitPacket(InitPacket, 3957 &entry->u.Open.binfo.FileAttributes, 3958 &entry->u.Open.sinfo.NumberOfLinks, 3959 &entry->u.Open.binfo.CreationTime, 3960 &entry->u.Open.binfo.LastAccessTime, 3961 &entry->u.Open.binfo.LastWriteTime, 3962 &entry->u.Open.binfo.ChangeTime, 3963 &entry->u.Open.sinfo.AllocationSize, 3964 &entry->u.Open.sinfo.EndOfFile, 3965 &entry->u.Open.sinfo.EndOfFile); 3966 3967 if (entry->u.Open.sinfo.Directory) 3968 StorageType = FileTypeDirectory; 3969 else 3970 StorageType = FileTypeFile; 3971 3972 RxFinishFcbInitialization(Fcb, RDBSS_STORAGE_NTC(StorageType), 3973 &InitPacket); 3974 } 3975 #ifdef DEBUG_OPEN 3976 else 3977 DbgP("$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$$\n"); 3978 3979 print_basic_info(1, &nfs41_fcb->BasicInfo); 3980 print_std_info(1, &nfs41_fcb->StandardInfo); 3981 #endif 3982 3983 /* aglo: 05/10/2012. it seems like always have to invalid the cache if the 3984 * file has been opened before and being opened again for data access. 3985 * If the file was opened before, RDBSS might have cached (unflushed) data 3986 * and by opening it again, we will not have the correct representation of 3987 * the file size and data content. fileio tests 208, 219, 221. 3988 */ 3989 if (Fcb->OpenCount > 0 && (isDataAccess(params->DesiredAccess) || 3990 nfs41_fcb->changeattr != entry->ChangeTime) && 3991 !nfs41_fcb->StandardInfo.Directory) { 3992 ULONG flag = DISABLE_CACHING; 3993 #ifdef DEBUG_OPEN 3994 DbgP("nfs41_Create: reopening (changed) file %wZ\n", 3995 SrvOpen->pAlreadyPrefixedName); 3996 #endif 3997 RxChangeBufferingState((PSRV_OPEN)SrvOpen, ULongToPtr(flag), 1); 3998 } 3999 if (!nfs41_fcb->StandardInfo.Directory && 4000 isDataAccess(params->DesiredAccess)) { 4001 nfs41_fobx->deleg_type = entry->u.Open.deleg_type; 4002 #ifdef DEBUG_OPEN 4003 DbgP("nfs41_Create: received delegation %d\n", entry->u.Open.deleg_type); 4004 #endif 4005 if (!(params->CreateOptions & FILE_WRITE_THROUGH) && 4006 !pVNetRootContext->write_thru && 4007 (entry->u.Open.deleg_type == 2 || 4008 (params->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA)))) { 4009 #ifdef DEBUG_OPEN 4010 DbgP("nfs41_Create: enabling write buffering\n"); 4011 #endif 4012 SrvOpen->BufferingFlags |= 4013 (FCB_STATE_WRITECACHING_ENABLED | 4014 FCB_STATE_WRITEBUFFERING_ENABLED); 4015 } else if (params->CreateOptions & FILE_WRITE_THROUGH || 4016 pVNetRootContext->write_thru) 4017 nfs41_fobx->write_thru = TRUE; 4018 if (entry->u.Open.deleg_type >= 1 || 4019 params->DesiredAccess & FILE_READ_DATA) { 4020 #ifdef DEBUG_OPEN 4021 DbgP("nfs41_Create: enabling read buffering\n"); 4022 #endif 4023 SrvOpen->BufferingFlags |= 4024 (FCB_STATE_READBUFFERING_ENABLED | 4025 FCB_STATE_READCACHING_ENABLED); 4026 } 4027 if (pVNetRootContext->nocache || 4028 (params->CreateOptions & FILE_NO_INTERMEDIATE_BUFFERING)) { 4029 #ifdef DEBUG_OPEN 4030 DbgP("nfs41_Create: disabling buffering\n"); 4031 #endif 4032 SrvOpen->BufferingFlags = FCB_STATE_DISABLE_LOCAL_BUFFERING; 4033 nfs41_fobx->nocache = TRUE; 4034 } else if (!entry->u.Open.deleg_type && !Fcb->OpenCount) { 4035 nfs41_fcb_list_entry *oentry; 4036 #ifdef DEBUG_OPEN 4037 DbgP("nfs41_Create: received no delegations: srv_open=%p " 4038 "ctime=%llu\n", SrvOpen, entry->ChangeTime); 4039 #endif 4040 oentry = RxAllocatePoolWithTag(NonPagedPool, 4041 sizeof(nfs41_fcb_list_entry), NFS41_MM_POOLTAG_OPEN); 4042 if (oentry == NULL) { 4043 status = STATUS_INSUFFICIENT_RESOURCES; 4044 goto out_free; 4045 } 4046 oentry->fcb = RxContext->pFcb; 4047 oentry->nfs41_fobx = nfs41_fobx; 4048 oentry->session = pVNetRootContext->session; 4049 oentry->ChangeTime = entry->ChangeTime; 4050 oentry->skip = FALSE; 4051 nfs41_AddEntry(fcblistLock, openlist, oentry); 4052 } 4053 } 4054 4055 if ((params->CreateOptions & FILE_DELETE_ON_CLOSE) && 4056 !pVNetRootContext->read_only) 4057 nfs41_fcb->StandardInfo.DeletePending = TRUE; 4058 4059 RxContext->Create.ReturnedCreateInformation = 4060 map_disposition_to_create_retval(params->Disposition, entry->errno); 4061 4062 RxContext->pFobx->OffsetOfNextEaToReturn = 1; 4063 #ifndef __REACTOS__ 4064 RxContext->CurrentIrp->IoStatus.Information = 4065 RxContext->Create.ReturnedCreateInformation; 4066 #endif 4067 status = RxContext->CurrentIrp->IoStatus.Status = STATUS_SUCCESS; 4068 4069 out_free: 4070 if (entry) 4071 RxFreePool(entry); 4072 out: 4073 #ifdef ENABLE_TIMINGS 4074 t2 = KeQueryPerformanceCounter(NULL); 4075 if ((params->DesiredAccess & FILE_READ_DATA) || 4076 (params->DesiredAccess & FILE_WRITE_DATA) || 4077 (params->DesiredAccess & FILE_APPEND_DATA) || 4078 (params->DesiredAccess & FILE_EXECUTE)) { 4079 InterlockedIncrement(&open.tops); 4080 InterlockedAdd64(&open.ticks, t2.QuadPart - t1.QuadPart); 4081 #ifdef ENABLE_INDV_TIMINGS 4082 DbgP("nfs41_Create open delta = %d op=%d sum=%d\n", 4083 t2.QuadPart - t1.QuadPart, open.tops, open.ticks); 4084 #endif 4085 } else { 4086 InterlockedIncrement(&lookup.tops); 4087 InterlockedAdd64(&lookup.ticks, t2.QuadPart - t1.QuadPart); 4088 #ifdef ENABLE_INDV_TIMINGS 4089 DbgP("nfs41_Create lookup delta = %d op=%d sum=%d\n", 4090 t2.QuadPart - t1.QuadPart, lookup.tops, lookup.ticks); 4091 #endif 4092 } 4093 #endif 4094 #ifdef DEBUG_OPEN 4095 DbgEx(); 4096 #endif 4097 return status; 4098 } 4099 4100 #ifdef __REACTOS__ 4101 NTSTATUS NTAPI nfs41_CollapseOpen( 4102 #else 4103 NTSTATUS nfs41_CollapseOpen( 4104 #endif 4105 IN OUT PRX_CONTEXT RxContext) 4106 { 4107 NTSTATUS status = STATUS_MORE_PROCESSING_REQUIRED; 4108 DbgEn(); 4109 DbgEx(); 4110 return status; 4111 } 4112 4113 #ifdef __REACTOS__ 4114 NTSTATUS NTAPI nfs41_ShouldTryToCollapseThisOpen( 4115 #else 4116 NTSTATUS nfs41_ShouldTryToCollapseThisOpen( 4117 #endif 4118 IN OUT PRX_CONTEXT RxContext) 4119 { 4120 if (RxContext->pRelevantSrvOpen == NULL) 4121 return STATUS_SUCCESS; 4122 else return STATUS_MORE_PROCESSING_REQUIRED; 4123 } 4124 4125 #ifdef __REACTOS__ 4126 ULONG NTAPI nfs41_ExtendForCache( 4127 #else 4128 ULONG nfs41_ExtendForCache( 4129 #endif 4130 IN OUT PRX_CONTEXT RxContext, 4131 IN PLARGE_INTEGER pNewFileSize, 4132 OUT PLARGE_INTEGER pNewAllocationSize) 4133 { 4134 NTSTATUS status = STATUS_SUCCESS; 4135 __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb); 4136 #ifdef DEBUG_CACHE 4137 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 4138 DbgEn(); 4139 print_debug_header(RxContext); 4140 DbgP("input: byte count 0x%x filesize 0x%x alloc size 0x%x\n", 4141 LowIoContext->ParamsFor.ReadWrite.ByteCount, *pNewFileSize, 4142 *pNewAllocationSize); 4143 #endif 4144 pNewAllocationSize->QuadPart = pNewFileSize->QuadPart + 8192; 4145 nfs41_fcb->StandardInfo.AllocationSize.QuadPart = 4146 pNewAllocationSize->QuadPart; 4147 nfs41_fcb->StandardInfo.EndOfFile.QuadPart = pNewFileSize->QuadPart; 4148 #ifdef DEBUG_CACHE 4149 DbgP("new filesize 0x%x new allocation size 0x%x\n", *pNewFileSize, 4150 *pNewAllocationSize); 4151 #endif 4152 #ifdef DEBUG_CACHE 4153 DbgEx(); 4154 #endif 4155 return status; 4156 } 4157 4158 VOID nfs41_remove_fcb_entry( 4159 PMRX_FCB fcb) 4160 { 4161 PLIST_ENTRY pEntry; 4162 nfs41_fcb_list_entry *cur; 4163 ExAcquireFastMutex(&fcblistLock); 4164 4165 pEntry = openlist.head.Flink; 4166 while (!IsListEmpty(&openlist.head)) { 4167 cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry, 4168 nfs41_fcb_list_entry, next); 4169 if (cur->fcb == fcb) { 4170 #ifdef DEBUG_CLOSE 4171 DbgP("nfs41_remove_srvopen_entry: Found match for fcb=%p\n", fcb); 4172 #endif 4173 RemoveEntryList(pEntry); 4174 RxFreePool(cur); 4175 break; 4176 } 4177 if (pEntry->Flink == &openlist.head) { 4178 #ifdef DEBUG_CLOSE 4179 DbgP("nfs41_remove_srvopen_entry: reached EOL looking for fcb " 4180 "%p\n", fcb); 4181 #endif 4182 break; 4183 } 4184 pEntry = pEntry->Flink; 4185 } 4186 ExReleaseFastMutex(&fcblistLock); 4187 } 4188 4189 NTSTATUS map_close_errors( 4190 DWORD status) 4191 { 4192 switch (status) { 4193 case NO_ERROR: return STATUS_SUCCESS; 4194 case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED; 4195 case ERROR_NOT_EMPTY: return STATUS_DIRECTORY_NOT_EMPTY; 4196 case ERROR_FILE_INVALID: return STATUS_FILE_INVALID; 4197 default: 4198 print_error("failed to map windows error %d to NTSTATUS; " 4199 "defaulting to STATUS_INTERNAL_ERROR\n", status); 4200 case ERROR_INTERNAL_ERROR: return STATUS_INTERNAL_ERROR; 4201 } 4202 } 4203 4204 #ifdef __REACTOS__ 4205 NTSTATUS NTAPI nfs41_CloseSrvOpen( 4206 #else 4207 NTSTATUS nfs41_CloseSrvOpen( 4208 #endif 4209 IN OUT PRX_CONTEXT RxContext) 4210 { 4211 NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES; 4212 nfs41_updowncall_entry *entry; 4213 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 4214 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 4215 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 4216 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 4217 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 4218 __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb); 4219 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 4220 #ifdef ENABLE_TIMINGS 4221 LARGE_INTEGER t1, t2; 4222 t1 = KeQueryPerformanceCounter(NULL); 4223 #endif 4224 4225 #ifdef DEBUG_CLOSE 4226 DbgEn(); 4227 print_debug_header(RxContext); 4228 #endif 4229 4230 if (!nfs41_fobx->deleg_type && !nfs41_fcb->StandardInfo.Directory && 4231 !RxContext->pFcb->OpenCount) { 4232 nfs41_remove_fcb_entry(RxContext->pFcb); 4233 } 4234 4235 status = nfs41_UpcallCreate(NFS41_CLOSE, &nfs41_fobx->sec_ctx, 4236 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 4237 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 4238 if (status) goto out; 4239 4240 entry->u.Close.srv_open = SrvOpen; 4241 if (nfs41_fcb->StandardInfo.DeletePending) 4242 nfs41_fcb->DeletePending = TRUE; 4243 if (!RxContext->pFcb->OpenCount || 4244 (nfs41_fcb->StandardInfo.DeletePending && 4245 nfs41_fcb->StandardInfo.Directory)) 4246 entry->u.Close.remove = nfs41_fcb->StandardInfo.DeletePending; 4247 if (!RxContext->pFcb->OpenCount) 4248 entry->u.Close.renamed = nfs41_fcb->Renamed; 4249 4250 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 4251 #ifndef USE_MOUNT_SEC_CONTEXT 4252 SeDeleteClientSecurity(&nfs41_fobx->sec_ctx); 4253 #endif 4254 if (status) goto out; 4255 4256 /* map windows ERRORs to NTSTATUS */ 4257 status = map_close_errors(entry->status); 4258 RxFreePool(entry); 4259 out: 4260 #ifdef ENABLE_TIMINGS 4261 t2 = KeQueryPerformanceCounter(NULL); 4262 InterlockedIncrement(&close.tops); 4263 InterlockedAdd64(&close.ticks, t2.QuadPart - t1.QuadPart); 4264 #ifdef ENABLE_INDV_TIMINGS 4265 DbgP("nfs41_CloseSrvOpen delta = %d op=%d sum=%d\n", 4266 t2.QuadPart - t1.QuadPart, close.tops, close.ticks); 4267 #endif 4268 #endif 4269 #ifdef DEBUG_CLOSE 4270 DbgEx(); 4271 #endif 4272 return status; 4273 } 4274 4275 #ifdef __REACTOS__ 4276 NTSTATUS NTAPI nfs41_Flush( 4277 #else 4278 NTSTATUS nfs41_Flush( 4279 #endif 4280 IN OUT PRX_CONTEXT RxContext) 4281 { 4282 return STATUS_SUCCESS; 4283 } 4284 4285 #ifdef __REACTOS__ 4286 NTSTATUS NTAPI nfs41_DeallocateForFcb( 4287 #else 4288 NTSTATUS nfs41_DeallocateForFcb( 4289 #endif 4290 IN OUT PMRX_FCB pFcb) 4291 { 4292 return STATUS_SUCCESS; 4293 } 4294 4295 #ifdef __REACTOS__ 4296 NTSTATUS NTAPI nfs41_DeallocateForFobx( 4297 #else 4298 NTSTATUS nfs41_DeallocateForFobx( 4299 #endif 4300 IN OUT PMRX_FOBX pFobx) 4301 { 4302 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(pFobx); 4303 if (nfs41_fobx->acl) 4304 RxFreePool(nfs41_fobx->acl); 4305 return STATUS_SUCCESS; 4306 } 4307 4308 void print_debug_filedirquery_header( 4309 PRX_CONTEXT RxContext) 4310 { 4311 print_debug_header(RxContext); 4312 DbgP("FileName='%wZ', InfoClass = %s\n", 4313 GET_ALREADY_PREFIXED_NAME_FROM_CONTEXT(RxContext), 4314 print_file_information_class(RxContext->Info.FileInformationClass)); 4315 } 4316 4317 void print_querydir_args( 4318 PRX_CONTEXT RxContext) 4319 { 4320 print_debug_filedirquery_header(RxContext); 4321 DbgP("Filter='%wZ', Index=%d, Restart/Single/Specified/Init=%d/%d/%d/%d\n", 4322 &RxContext->pFobx->UnicodeQueryTemplate, 4323 RxContext->QueryDirectory.FileIndex, 4324 RxContext->QueryDirectory.RestartScan, 4325 RxContext->QueryDirectory.ReturnSingleEntry, 4326 RxContext->QueryDirectory.IndexSpecified, 4327 RxContext->QueryDirectory.InitialQuery); 4328 } 4329 4330 NTSTATUS map_querydir_errors( 4331 DWORD status) 4332 { 4333 switch (status) { 4334 case ERROR_ACCESS_DENIED: return STATUS_ACCESS_DENIED; 4335 case ERROR_BUFFER_OVERFLOW: return STATUS_BUFFER_OVERFLOW; 4336 case ERROR_FILE_NOT_FOUND: return STATUS_NO_SUCH_FILE; 4337 case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED; 4338 case ERROR_INVALID_PARAMETER: return STATUS_INVALID_PARAMETER; 4339 case ERROR_NO_MORE_FILES: return STATUS_NO_MORE_FILES; 4340 case ERROR_OUTOFMEMORY: return STATUS_INSUFFICIENT_RESOURCES; 4341 case ERROR_FILENAME_EXCED_RANGE: return STATUS_NAME_TOO_LONG; 4342 default: 4343 print_error("failed to map windows error %d to NTSTATUS; " 4344 "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status); 4345 case ERROR_BAD_NET_RESP: return STATUS_INVALID_NETWORK_RESPONSE; 4346 } 4347 } 4348 4349 NTSTATUS check_nfs41_dirquery_args( 4350 IN PRX_CONTEXT RxContext) 4351 { 4352 if (RxContext->Info.Buffer == NULL) 4353 return STATUS_INVALID_USER_BUFFER; 4354 return STATUS_SUCCESS; 4355 } 4356 4357 #ifdef __REACTOS__ 4358 NTSTATUS NTAPI nfs41_QueryDirectory( 4359 #else 4360 NTSTATUS nfs41_QueryDirectory( 4361 #endif 4362 IN OUT PRX_CONTEXT RxContext) 4363 { 4364 NTSTATUS status = STATUS_INVALID_PARAMETER; 4365 nfs41_updowncall_entry *entry; 4366 FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass; 4367 PUNICODE_STRING Filter = &RxContext->pFobx->UnicodeQueryTemplate; 4368 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 4369 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 4370 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 4371 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 4372 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 4373 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 4374 #ifdef ENABLE_TIMINGS 4375 LARGE_INTEGER t1, t2; 4376 t1 = KeQueryPerformanceCounter(NULL); 4377 #endif 4378 4379 #ifdef DEBUG_DIR_QUERY 4380 DbgEn(); 4381 print_querydir_args(RxContext); 4382 #endif 4383 4384 status = check_nfs41_dirquery_args(RxContext); 4385 if (status) goto out; 4386 4387 switch (InfoClass) { 4388 /* classes handled in readdir_copy_entry() and readdir_size_for_entry() */ 4389 case FileNamesInformation: 4390 case FileDirectoryInformation: 4391 case FileFullDirectoryInformation: 4392 case FileIdFullDirectoryInformation: 4393 case FileBothDirectoryInformation: 4394 case FileIdBothDirectoryInformation: 4395 break; 4396 default: 4397 print_error("nfs41_QueryDirectory: unhandled dir query class %d\n", 4398 InfoClass); 4399 status = STATUS_NOT_SUPPORTED; 4400 goto out; 4401 } 4402 status = nfs41_UpcallCreate(NFS41_DIR_QUERY, &nfs41_fobx->sec_ctx, 4403 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 4404 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 4405 if (status) goto out; 4406 4407 entry->u.QueryFile.InfoClass = InfoClass; 4408 entry->buf_len = RxContext->Info.LengthRemaining; 4409 entry->buf = RxContext->Info.Buffer; 4410 entry->u.QueryFile.mdl = IoAllocateMdl(RxContext->Info.Buffer, 4411 RxContext->Info.LengthRemaining, FALSE, FALSE, NULL); 4412 if (entry->u.QueryFile.mdl == NULL) { 4413 status = STATUS_INTERNAL_ERROR; 4414 RxFreePool(entry); 4415 goto out; 4416 } 4417 entry->u.QueryFile.mdl->MdlFlags |= MDL_MAPPING_CAN_FAIL; 4418 MmProbeAndLockPages(entry->u.QueryFile.mdl, KernelMode, IoModifyAccess); 4419 4420 entry->u.QueryFile.filter = Filter; 4421 entry->u.QueryFile.initial_query = RxContext->QueryDirectory.InitialQuery; 4422 entry->u.QueryFile.restart_scan = RxContext->QueryDirectory.RestartScan; 4423 entry->u.QueryFile.return_single = RxContext->QueryDirectory.ReturnSingleEntry; 4424 4425 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 4426 if (status) goto out; 4427 MmUnlockPages(entry->u.QueryFile.mdl); 4428 4429 if (entry->status == STATUS_BUFFER_TOO_SMALL) { 4430 DbgP("nfs41_QueryDirectory: buffer too small provided %d need %lu\n", 4431 RxContext->Info.LengthRemaining, entry->buf_len); 4432 RxContext->InformationToReturn = entry->buf_len; 4433 status = STATUS_BUFFER_TOO_SMALL; 4434 } else if (entry->status == STATUS_SUCCESS) { 4435 #ifdef ENABLE_TIMINGS 4436 InterlockedIncrement(&readdir.sops); 4437 InterlockedAdd64(&readdir.size, entry->u.QueryFile.buf_len); 4438 #endif 4439 RxContext->Info.LengthRemaining -= entry->buf_len; 4440 status = STATUS_SUCCESS; 4441 } else { 4442 /* map windows ERRORs to NTSTATUS */ 4443 status = map_querydir_errors(entry->status); 4444 } 4445 IoFreeMdl(entry->u.QueryFile.mdl); 4446 RxFreePool(entry); 4447 out: 4448 #ifdef ENABLE_TIMINGS 4449 t2 = KeQueryPerformanceCounter(NULL); 4450 InterlockedIncrement(&readdir.tops); 4451 InterlockedAdd64(&readdir.ticks, t2.QuadPart - t1.QuadPart); 4452 #ifdef ENABLE_INDV_TIMINGS 4453 DbgP("nfs41_QueryDirectory delta = %d ops=%d sum=%d\n", 4454 t2.QuadPart - t1.QuadPart, readdir.tops, readdir.ticks); 4455 #endif 4456 #endif 4457 #ifdef DEBUG_DIR_QUERY 4458 DbgEx(); 4459 #endif 4460 return status; 4461 } 4462 4463 void print_queryvolume_args( 4464 PRX_CONTEXT RxContext) 4465 { 4466 print_debug_header(RxContext); 4467 DbgP("FileName='%wZ', InfoClass = %s BufferLen = %d\n", 4468 GET_ALREADY_PREFIXED_NAME_FROM_CONTEXT(RxContext), 4469 print_fs_information_class(RxContext->Info.FileInformationClass), 4470 RxContext->Info.LengthRemaining); 4471 } 4472 4473 NTSTATUS map_volume_errors( 4474 DWORD status) 4475 { 4476 switch (status) { 4477 case ERROR_ACCESS_DENIED: return STATUS_ACCESS_DENIED; 4478 case ERROR_VC_DISCONNECTED: return STATUS_CONNECTION_DISCONNECTED; 4479 case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED; 4480 case ERROR_INVALID_PARAMETER: return STATUS_INVALID_PARAMETER; 4481 case ERROR_OUTOFMEMORY: return STATUS_INSUFFICIENT_RESOURCES; 4482 default: 4483 print_error("failed to map windows error %d to NTSTATUS; " 4484 "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status); 4485 case ERROR_BAD_NET_RESP: return STATUS_INVALID_NETWORK_RESPONSE; 4486 } 4487 } 4488 4489 void nfs41_create_volume_info(PFILE_FS_VOLUME_INFORMATION pVolInfo, DWORD *len) 4490 { 4491 DECLARE_CONST_UNICODE_STRING(VolName, VOL_NAME); 4492 4493 RtlZeroMemory(pVolInfo, sizeof(FILE_FS_VOLUME_INFORMATION)); 4494 pVolInfo->VolumeSerialNumber = 0xBABAFACE; 4495 pVolInfo->VolumeLabelLength = VolName.Length; 4496 RtlCopyMemory(&pVolInfo->VolumeLabel[0], (PVOID)VolName.Buffer, 4497 VolName.MaximumLength); 4498 *len = sizeof(FILE_FS_VOLUME_INFORMATION) + VolName.Length; 4499 } 4500 4501 static BOOLEAN is_root_directory( 4502 PRX_CONTEXT RxContext) 4503 { 4504 __notnull PV_NET_ROOT VNetRoot = (PV_NET_ROOT) 4505 RxContext->pRelevantSrvOpen->pVNetRoot; 4506 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 4507 NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot); 4508 4509 /* calculate the root directory's length, including vnetroot prefix, 4510 * mount path, and a trailing \ */ 4511 const USHORT RootPathLen = VNetRoot->PrefixEntry.Prefix.Length + 4512 pVNetRootContext->MountPathLen + sizeof(WCHAR); 4513 4514 return RxContext->CurrentIrpSp->FileObject->FileName.Length <= RootPathLen; 4515 } 4516 4517 #ifdef __REACTOS__ 4518 NTSTATUS NTAPI nfs41_QueryVolumeInformation( 4519 #else 4520 NTSTATUS nfs41_QueryVolumeInformation( 4521 #endif 4522 IN OUT PRX_CONTEXT RxContext) 4523 { 4524 NTSTATUS status = STATUS_INVALID_PARAMETER; 4525 nfs41_updowncall_entry *entry; 4526 ULONG RemainingLength = RxContext->Info.LengthRemaining, SizeUsed; 4527 FS_INFORMATION_CLASS InfoClass = RxContext->Info.FsInformationClass; 4528 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 4529 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 4530 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 4531 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 4532 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 4533 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 4534 NFS41GetDeviceExtension(RxContext, DevExt); 4535 4536 #ifdef ENABLE_TIMINGS 4537 LARGE_INTEGER t1, t2; 4538 t1 = KeQueryPerformanceCounter(NULL); 4539 #endif 4540 4541 #ifdef DEBUG_VOLUME_QUERY 4542 DbgEn(); 4543 print_queryvolume_args(RxContext); 4544 #endif 4545 4546 status = check_nfs41_dirquery_args(RxContext); 4547 if (status) goto out; 4548 4549 #ifdef __REACTOS__ 4550 RtlZeroMemory(RxContext->Info.Buffer, RxContext->Info.LengthRemaining); 4551 #endif // __REACTOS__ 4552 4553 switch (InfoClass) { 4554 case FileFsVolumeInformation: 4555 if ((ULONG)RxContext->Info.LengthRemaining >= DevExt->VolAttrsLen) { 4556 RtlCopyMemory(RxContext->Info.Buffer, DevExt->VolAttrs, 4557 DevExt->VolAttrsLen); 4558 RxContext->Info.LengthRemaining -= DevExt->VolAttrsLen; 4559 status = STATUS_SUCCESS; 4560 } else { 4561 RtlCopyMemory(RxContext->Info.Buffer, DevExt->VolAttrs, 4562 RxContext->Info.LengthRemaining); 4563 status = STATUS_BUFFER_OVERFLOW; 4564 } 4565 goto out; 4566 case FileFsDeviceInformation: 4567 { 4568 PFILE_FS_DEVICE_INFORMATION pDevInfo = RxContext->Info.Buffer; 4569 4570 SizeUsed = sizeof(FILE_FS_DEVICE_INFORMATION); 4571 if (RemainingLength < SizeUsed) { 4572 status = STATUS_BUFFER_TOO_SMALL; 4573 RxContext->InformationToReturn = SizeUsed; 4574 goto out; 4575 } 4576 pDevInfo->DeviceType = RxContext->pFcb->pNetRoot->DeviceType; 4577 pDevInfo->Characteristics = FILE_REMOTE_DEVICE | FILE_DEVICE_IS_MOUNTED; 4578 RxContext->Info.LengthRemaining -= SizeUsed; 4579 status = STATUS_SUCCESS; 4580 goto out; 4581 } 4582 case FileAccessInformation: 4583 status = STATUS_NOT_SUPPORTED; 4584 goto out; 4585 4586 case FileFsAttributeInformation: 4587 if (RxContext->Info.LengthRemaining < FS_ATTR_LEN) { 4588 RxContext->InformationToReturn = FS_ATTR_LEN; 4589 status = STATUS_BUFFER_TOO_SMALL; 4590 goto out; 4591 } 4592 4593 /* on attribute queries for the root directory, 4594 * use cached volume attributes from mount */ 4595 if (is_root_directory(RxContext)) { 4596 PFILE_FS_ATTRIBUTE_INFORMATION attrs = 4597 (PFILE_FS_ATTRIBUTE_INFORMATION)RxContext->Info.Buffer; 4598 DECLARE_CONST_UNICODE_STRING(FsName, FS_NAME); 4599 4600 RtlCopyMemory(attrs, &pVNetRootContext->FsAttrs, 4601 sizeof(pVNetRootContext->FsAttrs)); 4602 4603 /* fill in the FileSystemName */ 4604 RtlCopyMemory(attrs->FileSystemName, FsName.Buffer, 4605 FsName.MaximumLength); /* 'MaximumLength' to include null */ 4606 attrs->FileSystemNameLength = FsName.Length; 4607 4608 RxContext->Info.LengthRemaining -= FS_ATTR_LEN; 4609 goto out; 4610 } 4611 /* else fall through and send the upcall */ 4612 case FileFsSizeInformation: 4613 case FileFsFullSizeInformation: 4614 break; 4615 4616 default: 4617 print_error("nfs41_QueryVolumeInformation: unhandled class %d\n", InfoClass); 4618 status = STATUS_NOT_SUPPORTED; 4619 goto out; 4620 } 4621 status = nfs41_UpcallCreate(NFS41_VOLUME_QUERY, &nfs41_fobx->sec_ctx, 4622 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 4623 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 4624 if (status) goto out; 4625 4626 entry->u.Volume.query = InfoClass; 4627 entry->buf = RxContext->Info.Buffer; 4628 entry->buf_len = RxContext->Info.LengthRemaining; 4629 4630 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 4631 if (status) goto out; 4632 4633 if (entry->status == STATUS_BUFFER_TOO_SMALL) { 4634 RxContext->InformationToReturn = entry->buf_len; 4635 status = STATUS_BUFFER_TOO_SMALL; 4636 } else if (entry->status == STATUS_SUCCESS) { 4637 if (InfoClass == FileFsAttributeInformation) { 4638 /* fill in the FileSystemName */ 4639 PFILE_FS_ATTRIBUTE_INFORMATION attrs = 4640 (PFILE_FS_ATTRIBUTE_INFORMATION)RxContext->Info.Buffer; 4641 DECLARE_CONST_UNICODE_STRING(FsName, FS_NAME); 4642 4643 RtlCopyMemory(attrs->FileSystemName, FsName.Buffer, 4644 FsName.MaximumLength); /* 'MaximumLength' to include null */ 4645 attrs->FileSystemNameLength = FsName.Length; 4646 4647 entry->buf_len = FS_ATTR_LEN; 4648 } 4649 #ifdef ENABLE_TIMINGS 4650 InterlockedIncrement(&volume.sops); 4651 InterlockedAdd64(&volume.size, entry->u.Volume.buf_len); 4652 #endif 4653 RxContext->Info.LengthRemaining -= entry->buf_len; 4654 status = STATUS_SUCCESS; 4655 } else { 4656 status = map_volume_errors(entry->status); 4657 } 4658 RxFreePool(entry); 4659 out: 4660 #ifdef ENABLE_TIMINGS 4661 t2 = KeQueryPerformanceCounter(NULL); 4662 InterlockedIncrement(&volume.tops); 4663 InterlockedAdd64(&volume.ticks, t2.QuadPart - t1.QuadPart); 4664 #ifdef ENABLE_INDV_TIMINGS 4665 DbgP("nfs41_QueryVolumeInformation delta = %d op=%d sum=%d\n", 4666 t2.QuadPart - t1.QuadPart, volume.tops, volume.ticks); 4667 #endif 4668 #endif 4669 #ifdef DEBUG_VOLUME_QUERY 4670 DbgEx(); 4671 #endif 4672 return status; 4673 } 4674 4675 VOID nfs41_update_fcb_list( 4676 PMRX_FCB fcb, 4677 ULONGLONG ChangeTime) 4678 { 4679 PLIST_ENTRY pEntry; 4680 nfs41_fcb_list_entry *cur; 4681 ExAcquireFastMutex(&fcblistLock); 4682 pEntry = openlist.head.Flink; 4683 while (!IsListEmpty(&openlist.head)) { 4684 cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry, 4685 nfs41_fcb_list_entry, next); 4686 if (cur->fcb == fcb && 4687 cur->ChangeTime != ChangeTime) { 4688 #if defined(DEBUG_FILE_SET) || defined(DEBUG_ACL_SET) || \ 4689 defined(DEBUG_WRITE) || defined(DEBUG_EA_SET) 4690 DbgP("nfs41_update_fcb_list: Found match for fcb %p: updating " 4691 "%llu to %llu\n", fcb, cur->ChangeTime, ChangeTime); 4692 #endif 4693 cur->ChangeTime = ChangeTime; 4694 break; 4695 } 4696 /* place an upcall for this srv_open */ 4697 if (pEntry->Flink == &openlist.head) { 4698 #if defined(DEBUG_FILE_SET) || defined(DEBUG_ACL_SET) || \ 4699 defined(DEBUG_WRITE) || defined(DEBUG_EA_SET) 4700 DbgP("nfs41_update_fcb_list: reached EOL loooking for " 4701 "fcb=%p\n", fcb); 4702 #endif 4703 break; 4704 } 4705 pEntry = pEntry->Flink; 4706 } 4707 ExReleaseFastMutex(&fcblistLock); 4708 } 4709 4710 void print_nfs3_attrs( 4711 nfs3_attrs *attrs) 4712 { 4713 DbgP("type=%d mode=%o nlink=%d size=%d atime=%x mtime=%x ctime=%x\n", 4714 attrs->type, attrs->mode, attrs->nlink, attrs->size, attrs->atime, 4715 attrs->mtime, attrs->ctime); 4716 } 4717 4718 void file_time_to_nfs_time( 4719 IN const PLARGE_INTEGER file_time, 4720 OUT LONGLONG *nfs_time) 4721 { 4722 LARGE_INTEGER diff = unix_time_diff; 4723 diff.QuadPart = file_time->QuadPart - diff.QuadPart; 4724 *nfs_time = diff.QuadPart / 10000000; 4725 } 4726 4727 void create_nfs3_attrs( 4728 nfs3_attrs *attrs, 4729 PNFS41_FCB nfs41_fcb) 4730 { 4731 RtlZeroMemory(attrs, sizeof(nfs3_attrs)); 4732 if (nfs41_fcb->BasicInfo.FileAttributes & FILE_ATTRIBUTE_REPARSE_POINT) 4733 attrs->type = NF3LNK; 4734 else if (nfs41_fcb->StandardInfo.Directory) 4735 attrs->type = NF3DIR; 4736 else 4737 attrs->type = NF3REG; 4738 attrs->mode = nfs41_fcb->mode; 4739 attrs->nlink = nfs41_fcb->StandardInfo.NumberOfLinks; 4740 attrs->size.QuadPart = attrs->used.QuadPart = 4741 nfs41_fcb->StandardInfo.EndOfFile.QuadPart; 4742 file_time_to_nfs_time(&nfs41_fcb->BasicInfo.LastAccessTime, &attrs->atime); 4743 file_time_to_nfs_time(&nfs41_fcb->BasicInfo.ChangeTime, &attrs->mtime); 4744 file_time_to_nfs_time(&nfs41_fcb->BasicInfo.CreationTime, &attrs->ctime); 4745 } 4746 4747 4748 NTSTATUS map_setea_error( 4749 DWORD error) 4750 { 4751 switch (error) { 4752 case NO_ERROR: return STATUS_SUCCESS; 4753 case ERROR_FILE_NOT_FOUND: return STATUS_NO_EAS_ON_FILE; 4754 case ERROR_ACCESS_DENIED: return STATUS_ACCESS_DENIED; 4755 case ERROR_NETWORK_ACCESS_DENIED: return STATUS_NETWORK_ACCESS_DENIED; 4756 case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED; 4757 case ERROR_FILE_TOO_LARGE: return STATUS_EA_TOO_LARGE; 4758 case ERROR_BUFFER_OVERFLOW: return STATUS_BUFFER_OVERFLOW; 4759 case STATUS_BUFFER_TOO_SMALL: 4760 case ERROR_INSUFFICIENT_BUFFER: return STATUS_BUFFER_TOO_SMALL; 4761 case ERROR_INVALID_EA_HANDLE: return STATUS_NONEXISTENT_EA_ENTRY; 4762 case ERROR_NO_MORE_FILES: return STATUS_NO_MORE_EAS; 4763 case ERROR_EA_FILE_CORRUPT: return STATUS_EA_CORRUPT_ERROR; 4764 default: 4765 print_error("failed to map windows error %d to NTSTATUS; " 4766 "defaulting to STATUS_INVALID_PARAMETER\n", error); 4767 case ERROR_INVALID_PARAMETER: return STATUS_INVALID_PARAMETER; 4768 } 4769 } 4770 4771 NTSTATUS check_nfs41_setea_args( 4772 IN PRX_CONTEXT RxContext) 4773 { 4774 NTSTATUS status; 4775 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 4776 NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot); 4777 __notnull PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs = 4778 &pVNetRootContext->FsAttrs; 4779 __notnull PFILE_FULL_EA_INFORMATION ea = 4780 (PFILE_FULL_EA_INFORMATION)RxContext->Info.Buffer; 4781 4782 status = check_nfs41_dirquery_args(RxContext); 4783 if (status) goto out; 4784 4785 if (ea == NULL) { 4786 status = STATUS_INVALID_PARAMETER; 4787 goto out; 4788 } 4789 if (AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength) || 4790 AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) { 4791 status = STATUS_INVALID_PARAMETER; /* only allowed on create */ 4792 goto out; 4793 } 4794 /* ignore cygwin EAs when checking support */ 4795 if (!(FsAttrs->FileSystemAttributes & FILE_SUPPORTS_EXTENDED_ATTRIBUTES) 4796 && !AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength)) { 4797 status = STATUS_EAS_NOT_SUPPORTED; 4798 goto out; 4799 } 4800 if ((RxContext->pRelevantSrvOpen->DesiredAccess & FILE_WRITE_EA) == 0) { 4801 status = STATUS_ACCESS_DENIED; 4802 goto out; 4803 } 4804 if (pVNetRootContext->read_only) { 4805 print_error("check_nfs41_setattr_args: Read-only mount\n"); 4806 status = STATUS_ACCESS_DENIED; 4807 goto out; 4808 } 4809 out: 4810 return status; 4811 } 4812 4813 #ifdef __REACTOS__ 4814 NTSTATUS NTAPI nfs41_SetEaInformation( 4815 #else 4816 NTSTATUS nfs41_SetEaInformation( 4817 #endif 4818 IN OUT PRX_CONTEXT RxContext) 4819 { 4820 NTSTATUS status = STATUS_EAS_NOT_SUPPORTED; 4821 nfs41_updowncall_entry *entry; 4822 __notnull PFILE_FULL_EA_INFORMATION eainfo = 4823 (PFILE_FULL_EA_INFORMATION)RxContext->Info.Buffer; 4824 nfs3_attrs *attrs = NULL; 4825 ULONG buflen = RxContext->CurrentIrpSp->Parameters.SetEa.Length, error_offset; 4826 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 4827 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 4828 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 4829 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 4830 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 4831 __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb); 4832 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 4833 #ifdef ENABLE_TIMINGS 4834 LARGE_INTEGER t1, t2; 4835 t1 = KeQueryPerformanceCounter(NULL); 4836 #endif 4837 4838 #ifdef DEBUG_EA_SET 4839 DbgEn(); 4840 print_debug_header(RxContext); 4841 print_ea_info(1, eainfo); 4842 #endif 4843 4844 status = check_nfs41_setea_args(RxContext); 4845 if (status) goto out; 4846 4847 status = nfs41_UpcallCreate(NFS41_EA_SET, &nfs41_fobx->sec_ctx, 4848 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 4849 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 4850 if (status) goto out; 4851 4852 if (AnsiStrEq(&NfsV3Attributes, eainfo->EaName, eainfo->EaNameLength)) { 4853 attrs = (nfs3_attrs *)(eainfo->EaName + eainfo->EaNameLength + 1); 4854 #ifdef DEBUG_EA_SET 4855 print_nfs3_attrs(attrs); 4856 DbgP("old mode is %o new mode is %o\n", nfs41_fcb->mode, attrs->mode); 4857 #endif 4858 entry->u.SetEa.mode = attrs->mode; 4859 } else { 4860 entry->u.SetEa.mode = 0; 4861 status = IoCheckEaBufferValidity(eainfo, buflen, &error_offset); 4862 if (status) { 4863 RxFreePool(entry); 4864 goto out; 4865 } 4866 } 4867 entry->buf = eainfo; 4868 entry->buf_len = buflen; 4869 4870 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 4871 if (status) goto out; 4872 #ifdef ENABLE_TIMINGS 4873 if (entry->status == STATUS_SUCCESS) { 4874 InterlockedIncrement(&setexattr.sops); 4875 InterlockedAdd64(&setexattr.size, entry->u.SetEa.buf_len); 4876 } 4877 #endif 4878 status = map_setea_error(entry->status); 4879 if (!status) { 4880 if (!nfs41_fobx->deleg_type && entry->ChangeTime && 4881 (SrvOpen->DesiredAccess & 4882 (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA))) 4883 nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime); 4884 nfs41_fcb->changeattr = entry->ChangeTime; 4885 nfs41_fcb->mode = entry->u.SetEa.mode; 4886 } 4887 RxFreePool(entry); 4888 out: 4889 #ifdef ENABLE_TIMINGS 4890 t2 = KeQueryPerformanceCounter(NULL); 4891 InterlockedIncrement(&setexattr.tops); 4892 InterlockedAdd64(&setexattr.ticks, t2.QuadPart - t1.QuadPart); 4893 #ifdef ENABLE_INDV_TIMINGS 4894 DbgP("nfs41_SetEaInformation delta = %d op=%d sum=%d\n", 4895 t2.QuadPart - t1.QuadPart, setexattr.tops, setexattr.ticks); 4896 #endif 4897 #endif 4898 #ifdef DEBUG_EA_SET 4899 DbgEx(); 4900 #endif 4901 return status; 4902 } 4903 4904 NTSTATUS check_nfs41_queryea_args( 4905 IN PRX_CONTEXT RxContext) 4906 { 4907 NTSTATUS status; 4908 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 4909 NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot); 4910 __notnull PFILE_FS_ATTRIBUTE_INFORMATION FsAttrs = 4911 &pVNetRootContext->FsAttrs; 4912 PFILE_GET_EA_INFORMATION ea = (PFILE_GET_EA_INFORMATION) 4913 RxContext->CurrentIrpSp->Parameters.QueryEa.EaList; 4914 4915 status = check_nfs41_dirquery_args(RxContext); 4916 if (status) goto out; 4917 4918 if (!(FsAttrs->FileSystemAttributes & FILE_SUPPORTS_EXTENDED_ATTRIBUTES)) { 4919 if (ea == NULL) { 4920 status = STATUS_EAS_NOT_SUPPORTED; 4921 goto out; 4922 } 4923 /* ignore cygwin EAs when checking support */ 4924 if (!AnsiStrEq(&NfsV3Attributes, ea->EaName, ea->EaNameLength) && 4925 !AnsiStrEq(&NfsActOnLink, ea->EaName, ea->EaNameLength) && 4926 !AnsiStrEq(&NfsSymlinkTargetName, ea->EaName, ea->EaNameLength)) { 4927 status = STATUS_EAS_NOT_SUPPORTED; 4928 goto out; 4929 } 4930 } 4931 if ((RxContext->pRelevantSrvOpen->DesiredAccess & FILE_READ_EA) == 0) { 4932 status = STATUS_ACCESS_DENIED; 4933 goto out; 4934 } 4935 out: 4936 return status; 4937 } 4938 4939 static NTSTATUS QueryCygwinSymlink( 4940 IN OUT PRX_CONTEXT RxContext, 4941 IN PFILE_GET_EA_INFORMATION query, 4942 OUT PFILE_FULL_EA_INFORMATION info) 4943 { 4944 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 4945 __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext = 4946 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 4947 __notnull PNFS41_NETROOT_EXTENSION NetRootContext = 4948 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 4949 __notnull PNFS41_FOBX Fobx = NFS41GetFobxExtension(RxContext->pFobx); 4950 nfs41_updowncall_entry *entry; 4951 UNICODE_STRING TargetName; 4952 const USHORT HeaderLen = FIELD_OFFSET(FILE_FULL_EA_INFORMATION, EaName) + 4953 query->EaNameLength + 1; 4954 NTSTATUS status; 4955 4956 if (RxContext->Info.LengthRemaining < HeaderLen) { 4957 status = STATUS_BUFFER_TOO_SMALL; 4958 RxContext->InformationToReturn = HeaderLen; 4959 goto out; 4960 } 4961 4962 TargetName.Buffer = (PWCH)(info->EaName + query->EaNameLength + 1); 4963 TargetName.MaximumLength = (USHORT)min(RxContext->Info.LengthRemaining - 4964 HeaderLen, 0xFFFF); 4965 4966 status = nfs41_UpcallCreate(NFS41_SYMLINK, &Fobx->sec_ctx, 4967 VNetRootContext->session, Fobx->nfs41_open_state, 4968 NetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 4969 if (status) goto out; 4970 4971 entry->u.Symlink.target = &TargetName; 4972 entry->u.Symlink.set = FALSE; 4973 4974 status = nfs41_UpcallWaitForReply(entry, VNetRootContext->timeout); 4975 if (status) goto out; 4976 4977 status = map_setea_error(entry->status); 4978 if (status == STATUS_SUCCESS) { 4979 info->NextEntryOffset = 0; 4980 info->Flags = 0; 4981 info->EaNameLength = query->EaNameLength; 4982 info->EaValueLength = TargetName.Length - sizeof(UNICODE_NULL); 4983 TargetName.Buffer[TargetName.Length/sizeof(WCHAR)] = UNICODE_NULL; 4984 RtlCopyMemory(info->EaName, query->EaName, query->EaNameLength); 4985 RxContext->Info.LengthRemaining = HeaderLen + info->EaValueLength; 4986 } else if (status == STATUS_BUFFER_TOO_SMALL) { 4987 RxContext->InformationToReturn = HeaderLen + 4988 entry->u.Symlink.target->Length; 4989 } 4990 RxFreePool(entry); 4991 out: 4992 return status; 4993 } 4994 4995 static NTSTATUS QueryCygwinEA( 4996 IN OUT PRX_CONTEXT RxContext, 4997 IN PFILE_GET_EA_INFORMATION query, 4998 OUT PFILE_FULL_EA_INFORMATION info) 4999 { 5000 NTSTATUS status = STATUS_NONEXISTENT_EA_ENTRY; 5001 5002 if (query == NULL) 5003 goto out; 5004 5005 if (AnsiStrEq(&NfsSymlinkTargetName, query->EaName, query->EaNameLength)) { 5006 __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb); 5007 if (nfs41_fcb->BasicInfo.FileAttributes & FILE_ATTRIBUTE_REPARSE_POINT) { 5008 status = QueryCygwinSymlink(RxContext, query, info); 5009 goto out; 5010 } else { 5011 const LONG LengthRequired = sizeof(FILE_FULL_EA_INFORMATION) + 5012 NfsSymlinkTargetName.Length - sizeof(CHAR); 5013 if (LengthRequired > RxContext->Info.LengthRemaining) { 5014 status = STATUS_BUFFER_TOO_SMALL; 5015 RxContext->InformationToReturn = LengthRequired; 5016 goto out; 5017 } 5018 info->NextEntryOffset = 0; 5019 info->Flags = 0; 5020 info->EaValueLength = 0; 5021 info->EaNameLength = (UCHAR)NfsActOnLink.Length; 5022 RtlCopyMemory(info->EaName, NfsSymlinkTargetName.Buffer, 5023 NfsSymlinkTargetName.Length); 5024 RxContext->Info.LengthRemaining = LengthRequired; 5025 status = STATUS_SUCCESS; 5026 goto out; 5027 } 5028 } 5029 5030 if (AnsiStrEq(&NfsV3Attributes, query->EaName, query->EaNameLength)) { 5031 nfs3_attrs attrs; 5032 5033 const LONG LengthRequired = sizeof(FILE_FULL_EA_INFORMATION) + 5034 NfsV3Attributes.Length + sizeof(nfs3_attrs) - sizeof(CHAR); 5035 if (LengthRequired > RxContext->Info.LengthRemaining) { 5036 status = STATUS_BUFFER_TOO_SMALL; 5037 RxContext->InformationToReturn = LengthRequired; 5038 goto out; 5039 } 5040 5041 create_nfs3_attrs(&attrs, NFS41GetFcbExtension(RxContext->pFcb)); 5042 #ifdef DEBUG_EA_QUERY 5043 print_nfs3_attrs(&attrs); 5044 #endif 5045 5046 info->NextEntryOffset = 0; 5047 info->Flags = 0; 5048 info->EaNameLength = (UCHAR)NfsV3Attributes.Length; 5049 info->EaValueLength = sizeof(nfs3_attrs); 5050 RtlCopyMemory(info->EaName, NfsV3Attributes.Buffer, 5051 NfsV3Attributes.Length); 5052 RtlCopyMemory(info->EaName + info->EaNameLength + 1, &attrs, 5053 sizeof(nfs3_attrs)); 5054 RxContext->Info.LengthRemaining = LengthRequired; 5055 status = STATUS_SUCCESS; 5056 goto out; 5057 } 5058 5059 if (AnsiStrEq(&NfsActOnLink, query->EaName, query->EaNameLength)) { 5060 5061 const LONG LengthRequired = sizeof(FILE_FULL_EA_INFORMATION) + 5062 query->EaNameLength - sizeof(CHAR); 5063 if (LengthRequired > RxContext->Info.LengthRemaining) { 5064 status = STATUS_BUFFER_TOO_SMALL; 5065 RxContext->InformationToReturn = LengthRequired; 5066 goto out; 5067 } 5068 5069 info->NextEntryOffset = 0; 5070 info->Flags = 0; 5071 info->EaNameLength = query->EaNameLength; 5072 info->EaValueLength = 0; 5073 RtlCopyMemory(info->EaName, query->EaName, query->EaNameLength); 5074 RxContext->Info.LengthRemaining = LengthRequired; 5075 status = STATUS_SUCCESS; 5076 goto out; 5077 } 5078 out: 5079 return status; 5080 } 5081 5082 #ifdef __REACTOS__ 5083 NTSTATUS NTAPI nfs41_QueryEaInformation( 5084 #else 5085 NTSTATUS nfs41_QueryEaInformation( 5086 #endif 5087 IN OUT PRX_CONTEXT RxContext) 5088 { 5089 NTSTATUS status = STATUS_EAS_NOT_SUPPORTED; 5090 nfs41_updowncall_entry *entry; 5091 PFILE_GET_EA_INFORMATION query = (PFILE_GET_EA_INFORMATION) 5092 RxContext->CurrentIrpSp->Parameters.QueryEa.EaList; 5093 ULONG buflen = RxContext->CurrentIrpSp->Parameters.QueryEa.Length; 5094 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 5095 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 5096 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 5097 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 5098 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 5099 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 5100 #ifdef ENABLE_TIMINGS 5101 LARGE_INTEGER t1, t2; 5102 t1 = KeQueryPerformanceCounter(NULL); 5103 #endif 5104 5105 #ifdef DEBUG_EA_QUERY 5106 DbgEn(); 5107 print_debug_header(RxContext); 5108 print_get_ea(1, query); 5109 #endif 5110 status = check_nfs41_queryea_args(RxContext); 5111 if (status) goto out; 5112 5113 /* handle queries for cygwin EAs */ 5114 status = QueryCygwinEA(RxContext, query, 5115 (PFILE_FULL_EA_INFORMATION)RxContext->Info.Buffer); 5116 if (status != STATUS_NONEXISTENT_EA_ENTRY) 5117 goto out; 5118 5119 status = nfs41_UpcallCreate(NFS41_EA_GET, &nfs41_fobx->sec_ctx, 5120 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 5121 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 5122 if (status) goto out; 5123 5124 entry->buf_len = buflen; 5125 entry->buf = RxContext->Info.Buffer; 5126 entry->u.QueryEa.EaList = query; 5127 entry->u.QueryEa.EaListLength = query == NULL ? 0 : 5128 RxContext->QueryEa.UserEaListLength; 5129 entry->u.QueryEa.EaIndex = RxContext->QueryEa.IndexSpecified ? 5130 RxContext->QueryEa.UserEaIndex : 0; 5131 entry->u.QueryEa.RestartScan = RxContext->QueryEa.RestartScan; 5132 entry->u.QueryEa.ReturnSingleEntry = RxContext->QueryEa.ReturnSingleEntry; 5133 5134 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 5135 if (status) goto out; 5136 5137 if (entry->status == STATUS_SUCCESS) { 5138 switch (entry->u.QueryEa.Overflow) { 5139 case ERROR_INSUFFICIENT_BUFFER: 5140 status = STATUS_BUFFER_TOO_SMALL; 5141 break; 5142 case ERROR_BUFFER_OVERFLOW: 5143 status = RxContext->IoStatusBlock.Status = STATUS_BUFFER_OVERFLOW; 5144 break; 5145 default: 5146 RxContext->IoStatusBlock.Status = STATUS_SUCCESS; 5147 break; 5148 } 5149 RxContext->InformationToReturn = entry->buf_len; 5150 #ifdef ENABLE_TIMINGS 5151 InterlockedIncrement(&getexattr.sops); 5152 InterlockedAdd64(&getexattr.size, entry->u.QueryEa.buf_len); 5153 #endif 5154 } else { 5155 status = map_setea_error(entry->status); 5156 } 5157 RxFreePool(entry); 5158 out: 5159 #ifdef ENABLE_TIMINGS 5160 t2 = KeQueryPerformanceCounter(NULL); 5161 InterlockedIncrement(&getexattr.tops); 5162 InterlockedAdd64(&getexattr.ticks, t2.QuadPart - t1.QuadPart); 5163 #ifdef ENABLE_INDV_TIMINGS 5164 DbgP("nfs41_QueryEaInformation delta = %d op=%d sum=%d\n", 5165 t2.QuadPart - t1.QuadPart, getexattr.tops, getexattr.ticks); 5166 #endif 5167 #endif 5168 #ifdef DEBUG_EA_QUERY 5169 DbgEx(); 5170 #endif 5171 return status; 5172 } 5173 5174 NTSTATUS map_query_acl_error( 5175 DWORD error) 5176 { 5177 switch (error) { 5178 case NO_ERROR: return STATUS_SUCCESS; 5179 case ERROR_NOT_SUPPORTED: return STATUS_NOT_SUPPORTED; 5180 case ERROR_ACCESS_DENIED: return STATUS_ACCESS_DENIED; 5181 case ERROR_FILE_NOT_FOUND: return STATUS_OBJECT_NAME_NOT_FOUND; 5182 case ERROR_INVALID_PARAMETER: return STATUS_INVALID_PARAMETER; 5183 default: 5184 print_error("failed to map windows error %d to NTSTATUS; " 5185 "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", error); 5186 case ERROR_BAD_NET_RESP: return STATUS_INVALID_NETWORK_RESPONSE; 5187 } 5188 } 5189 5190 NTSTATUS check_nfs41_getacl_args( 5191 PRX_CONTEXT RxContext) 5192 { 5193 NTSTATUS status = STATUS_SUCCESS; 5194 SECURITY_INFORMATION info_class = 5195 RxContext->CurrentIrpSp->Parameters.QuerySecurity.SecurityInformation; 5196 5197 /* we don't support sacls */ 5198 if (info_class == SACL_SECURITY_INFORMATION || 5199 info_class == LABEL_SECURITY_INFORMATION) { 5200 status = STATUS_NOT_SUPPORTED; 5201 goto out; 5202 } 5203 if (RxContext->CurrentIrp->UserBuffer == NULL && 5204 RxContext->CurrentIrpSp->Parameters.QuerySecurity.Length) 5205 status = STATUS_INVALID_USER_BUFFER; 5206 out: 5207 return status; 5208 } 5209 5210 #ifdef __REACTOS__ 5211 NTSTATUS NTAPI nfs41_QuerySecurityInformation( 5212 #else 5213 NTSTATUS nfs41_QuerySecurityInformation( 5214 #endif 5215 IN OUT PRX_CONTEXT RxContext) 5216 { 5217 NTSTATUS status = STATUS_NOT_SUPPORTED; 5218 nfs41_updowncall_entry *entry; 5219 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 5220 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 5221 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 5222 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 5223 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 5224 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 5225 SECURITY_INFORMATION info_class = 5226 RxContext->CurrentIrpSp->Parameters.QuerySecurity.SecurityInformation; 5227 #ifdef ENABLE_TIMINGS 5228 LARGE_INTEGER t1, t2; 5229 t1 = KeQueryPerformanceCounter(NULL); 5230 #endif 5231 5232 #ifdef DEBUG_ACL_QUERY 5233 DbgEn(); 5234 print_debug_header(RxContext); 5235 print_acl_args(info_class); 5236 #endif 5237 5238 status = check_nfs41_getacl_args(RxContext); 5239 if (status) goto out; 5240 5241 if (nfs41_fobx->acl && nfs41_fobx->acl_len) { 5242 LARGE_INTEGER current_time; 5243 KeQuerySystemTime(¤t_time); 5244 #ifdef DEBUG_ACL_QUERY 5245 DbgP("CurrentTime %lx Saved Acl time %lx\n", 5246 current_time.QuadPart, nfs41_fobx->time.QuadPart); 5247 #endif 5248 if (current_time.QuadPart - nfs41_fobx->time.QuadPart <= 20*1000) { 5249 PSECURITY_DESCRIPTOR sec_desc = (PSECURITY_DESCRIPTOR) 5250 RxContext->CurrentIrp->UserBuffer; 5251 RtlCopyMemory(sec_desc, nfs41_fobx->acl, nfs41_fobx->acl_len); 5252 RxContext->IoStatusBlock.Information = 5253 RxContext->InformationToReturn = nfs41_fobx->acl_len; 5254 RxContext->IoStatusBlock.Status = status = STATUS_SUCCESS; 5255 #ifdef ENABLE_TIMINGS 5256 InterlockedIncrement(&getacl.sops); 5257 InterlockedAdd64(&getacl.size, nfs41_fobx->acl_len); 5258 #endif 5259 } else status = 1; 5260 RxFreePool(nfs41_fobx->acl); 5261 nfs41_fobx->acl = NULL; 5262 nfs41_fobx->acl_len = 0; 5263 if (!status) 5264 goto out; 5265 } 5266 5267 status = nfs41_UpcallCreate(NFS41_ACL_QUERY, &nfs41_fobx->sec_ctx, 5268 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 5269 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 5270 if (status) goto out; 5271 5272 entry->u.Acl.query = info_class; 5273 /* we can't provide RxContext->CurrentIrp->UserBuffer to the upcall thread 5274 * because it becomes an invalid pointer with that execution context 5275 */ 5276 entry->buf_len = RxContext->CurrentIrpSp->Parameters.QuerySecurity.Length; 5277 5278 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 5279 if (status) goto out; 5280 5281 if (entry->status == STATUS_BUFFER_TOO_SMALL) { 5282 #ifdef DEBUG_ACL_QUERY 5283 DbgP("nfs41_QuerySecurityInformation: provided buffer size=%d but we " 5284 "need %lu\n", 5285 RxContext->CurrentIrpSp->Parameters.QuerySecurity.Length, 5286 entry->buf_len); 5287 #endif 5288 status = STATUS_BUFFER_OVERFLOW; 5289 RxContext->InformationToReturn = entry->buf_len; 5290 5291 /* Save ACL buffer */ 5292 nfs41_fobx->acl = entry->buf; 5293 nfs41_fobx->acl_len = entry->buf_len; 5294 KeQuerySystemTime(&nfs41_fobx->time); 5295 } else if (entry->status == STATUS_SUCCESS) { 5296 PSECURITY_DESCRIPTOR sec_desc = (PSECURITY_DESCRIPTOR) 5297 RxContext->CurrentIrp->UserBuffer; 5298 RtlCopyMemory(sec_desc, entry->buf, entry->buf_len); 5299 #ifdef ENABLE_TIMINGS 5300 InterlockedIncrement(&getacl.sops); 5301 InterlockedAdd64(&getacl.size, entry->u.Acl.buf_len); 5302 #endif 5303 RxFreePool(entry->buf); 5304 nfs41_fobx->acl = NULL; 5305 nfs41_fobx->acl_len = 0; 5306 RxContext->IoStatusBlock.Information = RxContext->InformationToReturn = 5307 entry->buf_len; 5308 RxContext->IoStatusBlock.Status = status = STATUS_SUCCESS; 5309 } else { 5310 status = map_query_acl_error(entry->status); 5311 } 5312 RxFreePool(entry); 5313 out: 5314 #ifdef ENABLE_TIMINGS 5315 t2 = KeQueryPerformanceCounter(NULL); 5316 /* only count getacl that we made an upcall for */ 5317 if (status == STATUS_BUFFER_OVERFLOW) { 5318 InterlockedIncrement(&getacl.tops); 5319 InterlockedAdd64(&getacl.ticks, t2.QuadPart - t1.QuadPart); 5320 } 5321 #ifdef ENABLE_INDV_TIMINGS 5322 DbgP("nfs41_QuerySecurityInformation: delta = %d op=%d sum=%d\n", 5323 t2.QuadPart - t1.QuadPart, getacl.tops, getacl.ticks); 5324 #endif 5325 #endif 5326 #ifdef DEBUG_ACL_QUERY 5327 DbgEx(); 5328 #endif 5329 return status; 5330 } 5331 5332 NTSTATUS check_nfs41_setacl_args( 5333 PRX_CONTEXT RxContext) 5334 { 5335 NTSTATUS status = STATUS_SUCCESS; 5336 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 5337 NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot); 5338 SECURITY_INFORMATION info_class = 5339 RxContext->CurrentIrpSp->Parameters.SetSecurity.SecurityInformation; 5340 5341 if (pVNetRootContext->read_only) { 5342 print_error("check_nfs41_setacl_args: Read-only mount\n"); 5343 status = STATUS_ACCESS_DENIED; 5344 goto out; 5345 } 5346 /* we don't support sacls */ 5347 if (info_class == SACL_SECURITY_INFORMATION || 5348 info_class == LABEL_SECURITY_INFORMATION) { 5349 status = STATUS_NOT_SUPPORTED; 5350 goto out; 5351 } 5352 out: 5353 return status; 5354 } 5355 5356 #ifdef __REACTOS__ 5357 NTSTATUS NTAPI nfs41_SetSecurityInformation( 5358 #else 5359 NTSTATUS nfs41_SetSecurityInformation( 5360 #endif 5361 IN OUT PRX_CONTEXT RxContext) 5362 { 5363 NTSTATUS status = STATUS_NOT_SUPPORTED; 5364 nfs41_updowncall_entry *entry; 5365 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 5366 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 5367 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 5368 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 5369 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 5370 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 5371 __notnull PSECURITY_DESCRIPTOR sec_desc = 5372 RxContext->CurrentIrpSp->Parameters.SetSecurity.SecurityDescriptor; 5373 __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb); 5374 SECURITY_INFORMATION info_class = 5375 RxContext->CurrentIrpSp->Parameters.SetSecurity.SecurityInformation; 5376 #ifdef ENABLE_TIMINGS 5377 LARGE_INTEGER t1, t2; 5378 t1 = KeQueryPerformanceCounter(NULL); 5379 #endif 5380 5381 #ifdef DEBUG_ACL_SET 5382 DbgEn(); 5383 print_debug_header(RxContext); 5384 print_acl_args(info_class); 5385 #endif 5386 5387 status = check_nfs41_setacl_args(RxContext); 5388 if (status) goto out; 5389 5390 /* check that ACL is present */ 5391 if (info_class & DACL_SECURITY_INFORMATION) { 5392 PACL acl; 5393 BOOLEAN present, dacl_default; 5394 status = RtlGetDaclSecurityDescriptor(sec_desc, &present, &acl, 5395 &dacl_default); 5396 if (status) { 5397 DbgP("RtlGetDaclSecurityDescriptor failed %x\n", status); 5398 goto out; 5399 } 5400 if (present == FALSE) { 5401 DbgP("NO ACL present\n"); 5402 goto out; 5403 } 5404 } 5405 5406 status = nfs41_UpcallCreate(NFS41_ACL_SET, &nfs41_fobx->sec_ctx, 5407 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 5408 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 5409 if (status) goto out; 5410 5411 entry->u.Acl.query = info_class; 5412 entry->buf = sec_desc; 5413 entry->buf_len = RtlLengthSecurityDescriptor(sec_desc); 5414 #ifdef ENABLE_TIMINGS 5415 InterlockedIncrement(&setacl.sops); 5416 InterlockedAdd64(&setacl.size, entry->u.Acl.buf_len); 5417 #endif 5418 5419 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 5420 if (status) goto out; 5421 5422 status = map_query_acl_error(entry->status); 5423 if (!status) { 5424 if (!nfs41_fobx->deleg_type && entry->ChangeTime && 5425 (SrvOpen->DesiredAccess & 5426 (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA))) 5427 nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime); 5428 nfs41_fcb->changeattr = entry->ChangeTime; 5429 } 5430 RxFreePool(entry); 5431 out: 5432 #ifdef ENABLE_TIMINGS 5433 t2 = KeQueryPerformanceCounter(NULL); 5434 InterlockedIncrement(&setacl.tops); 5435 InterlockedAdd64(&setacl.ticks, t2.QuadPart - t1.QuadPart); 5436 #ifdef ENABLE_INDV_TIMINGS 5437 DbgP("nfs41_SetSecurityInformation delta = %d op=%d sum=%d\n", 5438 t2.QuadPart - t1.QuadPart, setacl.tops, setacl.ticks); 5439 #endif 5440 #endif 5441 #ifdef DEBUG_ACL_SET 5442 DbgEx(); 5443 #endif 5444 return status; 5445 } 5446 5447 NTSTATUS map_queryfile_error( 5448 DWORD error) 5449 { 5450 switch (error) { 5451 case ERROR_ACCESS_DENIED: return STATUS_ACCESS_DENIED; 5452 case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED; 5453 case ERROR_INVALID_PARAMETER: return STATUS_INVALID_PARAMETER; 5454 default: 5455 print_error("failed to map windows error %d to NTSTATUS; " 5456 "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", error); 5457 case ERROR_BAD_NET_RESP: return STATUS_INVALID_NETWORK_RESPONSE; 5458 } 5459 } 5460 5461 #ifdef __REACTOS__ 5462 NTSTATUS NTAPI nfs41_QueryFileInformation( 5463 #else 5464 NTSTATUS nfs41_QueryFileInformation( 5465 #endif 5466 IN OUT PRX_CONTEXT RxContext) 5467 { 5468 NTSTATUS status = STATUS_OBJECT_NAME_NOT_FOUND; 5469 FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass; 5470 nfs41_updowncall_entry *entry; 5471 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 5472 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 5473 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 5474 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 5475 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 5476 __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb); 5477 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 5478 #ifdef ENABLE_TIMINGS 5479 LARGE_INTEGER t1, t2; 5480 t1 = KeQueryPerformanceCounter(NULL); 5481 #endif 5482 5483 #ifdef DEBUG_FILE_QUERY 5484 DbgEn(); 5485 print_debug_filedirquery_header(RxContext); 5486 #endif 5487 5488 status = check_nfs41_dirquery_args(RxContext); 5489 if (status) goto out; 5490 5491 switch (InfoClass) { 5492 case FileEaInformation: 5493 { 5494 PFILE_EA_INFORMATION info = 5495 (PFILE_EA_INFORMATION)RxContext->Info.Buffer; 5496 info->EaSize = 0; 5497 RxContext->Info.LengthRemaining -= sizeof(FILE_EA_INFORMATION); 5498 status = STATUS_SUCCESS; 5499 goto out; 5500 } 5501 case FileBasicInformation: 5502 case FileStandardInformation: 5503 case FileInternalInformation: 5504 case FileAttributeTagInformation: 5505 case FileNetworkOpenInformation: 5506 break; 5507 default: 5508 print_error("nfs41_QueryFileInformation: unhandled class %d\n", InfoClass); 5509 status = STATUS_NOT_SUPPORTED; 5510 goto out; 5511 } 5512 5513 status = nfs41_UpcallCreate(NFS41_FILE_QUERY, &nfs41_fobx->sec_ctx, 5514 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 5515 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 5516 if (status) goto out; 5517 5518 entry->u.QueryFile.InfoClass = InfoClass; 5519 entry->buf = RxContext->Info.Buffer; 5520 entry->buf_len = RxContext->Info.LengthRemaining; 5521 5522 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 5523 if (status) goto out; 5524 5525 if (entry->status == STATUS_BUFFER_TOO_SMALL) { 5526 RxContext->InformationToReturn = entry->buf_len; 5527 status = STATUS_BUFFER_TOO_SMALL; 5528 } else if (entry->status == STATUS_SUCCESS) { 5529 BOOLEAN DeletePending = FALSE; 5530 #ifdef ENABLE_TIMINGS 5531 InterlockedIncrement(&getattr.sops); 5532 InterlockedAdd64(&getattr.size, entry->u.QueryFile.buf_len); 5533 #endif 5534 RxContext->Info.LengthRemaining -= entry->buf_len; 5535 status = STATUS_SUCCESS; 5536 5537 switch (InfoClass) { 5538 case FileBasicInformation: 5539 RtlCopyMemory(&nfs41_fcb->BasicInfo, RxContext->Info.Buffer, 5540 sizeof(nfs41_fcb->BasicInfo)); 5541 #ifdef DEBUG_FILE_QUERY 5542 print_basic_info(1, &nfs41_fcb->BasicInfo); 5543 #endif 5544 break; 5545 case FileStandardInformation: 5546 /* this a fix for RDBSS behaviour when it first calls ExtendForCache, 5547 * then it sends a file query irp for standard attributes and 5548 * expects to receive EndOfFile of value set by the ExtendForCache. 5549 * It seems to cache the filesize based on that instead of sending 5550 * a file size query for after doing the write. 5551 */ 5552 { 5553 PFILE_STANDARD_INFORMATION std_info; 5554 std_info = (PFILE_STANDARD_INFORMATION)RxContext->Info.Buffer; 5555 if (nfs41_fcb->StandardInfo.AllocationSize.QuadPart > 5556 std_info->AllocationSize.QuadPart) { 5557 #ifdef DEBUG_FILE_QUERY 5558 DbgP("Old AllocationSize is bigger: saving %x\n", 5559 nfs41_fcb->StandardInfo.AllocationSize.QuadPart); 5560 #endif 5561 std_info->AllocationSize.QuadPart = 5562 nfs41_fcb->StandardInfo.AllocationSize.QuadPart; 5563 } 5564 if (nfs41_fcb->StandardInfo.EndOfFile.QuadPart > 5565 std_info->EndOfFile.QuadPart) { 5566 #ifdef DEBUG_FILE_QUERY 5567 DbgP("Old EndOfFile is bigger: saving %x\n", 5568 nfs41_fcb->StandardInfo.EndOfFile); 5569 #endif 5570 std_info->EndOfFile.QuadPart = 5571 nfs41_fcb->StandardInfo.EndOfFile.QuadPart; 5572 } 5573 std_info->DeletePending = nfs41_fcb->DeletePending; 5574 } 5575 if (nfs41_fcb->StandardInfo.DeletePending) 5576 DeletePending = TRUE; 5577 RtlCopyMemory(&nfs41_fcb->StandardInfo, RxContext->Info.Buffer, 5578 sizeof(nfs41_fcb->StandardInfo)); 5579 nfs41_fcb->StandardInfo.DeletePending = DeletePending; 5580 #ifdef DEBUG_FILE_QUERY 5581 print_std_info(1, &nfs41_fcb->StandardInfo); 5582 #endif 5583 break; 5584 } 5585 } else { 5586 status = map_queryfile_error(entry->status); 5587 } 5588 RxFreePool(entry); 5589 out: 5590 #ifdef ENABLE_TIMINGS 5591 t2 = KeQueryPerformanceCounter(NULL); 5592 InterlockedIncrement(&getattr.tops); 5593 InterlockedAdd64(&getattr.ticks, t2.QuadPart - t1.QuadPart); 5594 #ifdef ENABLE_INDV_TIMINGS 5595 DbgP("nfs41_QueryFileInformation delta = %d op=%d sum=%d\n", 5596 t2.QuadPart - t1.QuadPart, getattr.tops, getattr.ticks); 5597 #endif 5598 #endif 5599 #ifdef DEBUG_FILE_QUERY 5600 DbgEx(); 5601 #endif 5602 return status; 5603 } 5604 5605 NTSTATUS map_setfile_error( 5606 DWORD error) 5607 { 5608 switch (error) { 5609 case NO_ERROR: return STATUS_SUCCESS; 5610 case ERROR_NOT_EMPTY: return STATUS_DIRECTORY_NOT_EMPTY; 5611 case ERROR_FILE_EXISTS: return STATUS_OBJECT_NAME_COLLISION; 5612 case ERROR_FILE_NOT_FOUND: return STATUS_OBJECT_NAME_NOT_FOUND; 5613 case ERROR_PATH_NOT_FOUND: return STATUS_OBJECT_PATH_NOT_FOUND; 5614 case ERROR_ACCESS_DENIED: return STATUS_ACCESS_DENIED; 5615 case ERROR_FILE_INVALID: return STATUS_FILE_INVALID; 5616 case ERROR_NOT_SAME_DEVICE: return STATUS_NOT_SAME_DEVICE; 5617 case ERROR_NOT_SUPPORTED: return STATUS_NOT_IMPLEMENTED; 5618 case ERROR_NETWORK_ACCESS_DENIED: return STATUS_NETWORK_ACCESS_DENIED; 5619 case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED; 5620 case ERROR_BUFFER_OVERFLOW: return STATUS_INSUFFICIENT_RESOURCES; 5621 default: 5622 print_error("failed to map windows error %d to NTSTATUS; " 5623 "defaulting to STATUS_INVALID_PARAMETER\n", error); 5624 case ERROR_INVALID_PARAMETER: return STATUS_INVALID_PARAMETER; 5625 } 5626 } 5627 5628 NTSTATUS check_nfs41_setattr_args( 5629 IN PRX_CONTEXT RxContext) 5630 { 5631 NTSTATUS status = STATUS_SUCCESS; 5632 FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass; 5633 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 5634 NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot); 5635 5636 if (pVNetRootContext->read_only) { 5637 print_error("check_nfs41_setattr_args: Read-only mount\n"); 5638 status = STATUS_ACCESS_DENIED; 5639 goto out; 5640 } 5641 5642 /* http://msdn.microsoft.com/en-us/library/ff469355(v=PROT.10).aspx 5643 * http://msdn.microsoft.com/en-us/library/ff469424(v=PROT.10).aspx 5644 * If Open.GrantedAccess does not contain FILE_WRITE_DATA, the operation 5645 * MUST be failed with STATUS_ACCESS_DENIED. 5646 */ 5647 if (InfoClass == FileAllocationInformation || 5648 InfoClass == FileEndOfFileInformation) { 5649 if (!(RxContext->pRelevantSrvOpen->DesiredAccess & FILE_WRITE_DATA)) { 5650 status = STATUS_ACCESS_DENIED; 5651 goto out; 5652 } 5653 } 5654 status = check_nfs41_dirquery_args(RxContext); 5655 if (status) goto out; 5656 5657 switch (InfoClass) { 5658 case FileRenameInformation: 5659 { 5660 PFILE_RENAME_INFORMATION rinfo = 5661 (PFILE_RENAME_INFORMATION)RxContext->Info.Buffer; 5662 UNICODE_STRING dst = { (USHORT)rinfo->FileNameLength, 5663 (USHORT)rinfo->FileNameLength, rinfo->FileName }; 5664 #ifdef DEBUG_FILE_SET 5665 DbgP("Attempting to rename to '%wZ'\n", &dst); 5666 #endif 5667 if (isFilenameTooLong(&dst, pVNetRootContext)) { 5668 status = STATUS_OBJECT_NAME_INVALID; 5669 goto out; 5670 } 5671 if (rinfo->RootDirectory) { 5672 status = STATUS_INVALID_PARAMETER; 5673 goto out; 5674 } 5675 break; 5676 } 5677 case FileLinkInformation: 5678 { 5679 PFILE_LINK_INFORMATION linfo = 5680 (PFILE_LINK_INFORMATION)RxContext->Info.Buffer; 5681 UNICODE_STRING dst = { (USHORT)linfo->FileNameLength, 5682 (USHORT)linfo->FileNameLength, linfo->FileName }; 5683 #ifdef DEBUG_FILE_SET 5684 DbgP("Attempting to add link as '%wZ'\n", &dst); 5685 #endif 5686 if (isFilenameTooLong(&dst, pVNetRootContext)) { 5687 status = STATUS_OBJECT_NAME_INVALID; 5688 goto out; 5689 } 5690 if (linfo->RootDirectory) { 5691 status = STATUS_INVALID_PARAMETER; 5692 goto out; 5693 } 5694 break; 5695 } 5696 case FileDispositionInformation: 5697 { 5698 PFILE_DISPOSITION_INFORMATION dinfo = 5699 (PFILE_DISPOSITION_INFORMATION)RxContext->Info.Buffer; 5700 __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb); 5701 if (dinfo->DeleteFile && nfs41_fcb->DeletePending) { 5702 status = STATUS_DELETE_PENDING; 5703 goto out; 5704 } 5705 break; 5706 } 5707 case FileBasicInformation: 5708 case FileAllocationInformation: 5709 case FileEndOfFileInformation: 5710 break; 5711 default: 5712 print_error("nfs41_SetFileInformation: unhandled class %d\n", InfoClass); 5713 status = STATUS_NOT_SUPPORTED; 5714 } 5715 5716 out: 5717 return status; 5718 } 5719 5720 #ifdef __REACTOS__ 5721 NTSTATUS NTAPI nfs41_SetFileInformation( 5722 #else 5723 NTSTATUS nfs41_SetFileInformation( 5724 #endif 5725 IN OUT PRX_CONTEXT RxContext) 5726 { 5727 NTSTATUS status = STATUS_INVALID_PARAMETER; 5728 nfs41_updowncall_entry *entry; 5729 FILE_INFORMATION_CLASS InfoClass = RxContext->Info.FileInformationClass; 5730 FILE_RENAME_INFORMATION rinfo; 5731 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 5732 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 5733 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 5734 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 5735 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 5736 __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb); 5737 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 5738 #ifdef ENABLE_TIMINGS 5739 LARGE_INTEGER t1, t2; 5740 t1 = KeQueryPerformanceCounter(NULL); 5741 #endif 5742 5743 #ifdef DEBUG_FILE_SET 5744 DbgEn(); 5745 print_debug_filedirquery_header(RxContext); 5746 #endif 5747 5748 status = check_nfs41_setattr_args(RxContext); 5749 if (status) goto out; 5750 5751 switch (InfoClass) { 5752 case FileDispositionInformation: 5753 { 5754 PFILE_DISPOSITION_INFORMATION dinfo = 5755 (PFILE_DISPOSITION_INFORMATION)RxContext->Info.Buffer; 5756 if (dinfo->DeleteFile) { 5757 nfs41_fcb->DeletePending = TRUE; 5758 // we can delete directories right away 5759 if (nfs41_fcb->StandardInfo.Directory) 5760 break; 5761 nfs41_fcb->StandardInfo.DeletePending = TRUE; 5762 if (RxContext->pFcb->OpenCount > 1) { 5763 rinfo.ReplaceIfExists = 0; 5764 rinfo.RootDirectory = INVALID_HANDLE_VALUE; 5765 rinfo.FileNameLength = 0; 5766 rinfo.FileName[0] = L'\0'; 5767 InfoClass = FileRenameInformation; 5768 nfs41_fcb->Renamed = TRUE; 5769 break; 5770 } 5771 } else { 5772 /* section 4.3.3 of [FSBO] 5773 * "file system behavior in the microsoft windows environment" 5774 */ 5775 if (nfs41_fcb->DeletePending) { 5776 nfs41_fcb->DeletePending = 0; 5777 nfs41_fcb->StandardInfo.DeletePending = 0; 5778 } 5779 } 5780 status = STATUS_SUCCESS; 5781 goto out; 5782 } 5783 case FileEndOfFileInformation: 5784 { 5785 PFILE_END_OF_FILE_INFORMATION info = 5786 (PFILE_END_OF_FILE_INFORMATION)RxContext->Info.Buffer; 5787 nfs41_fcb->StandardInfo.AllocationSize = 5788 nfs41_fcb->StandardInfo.EndOfFile = info->EndOfFile; 5789 break; 5790 } 5791 case FileRenameInformation: 5792 { 5793 /* noop if filename and destination are the same */ 5794 PFILE_RENAME_INFORMATION prinfo = 5795 (PFILE_RENAME_INFORMATION)RxContext->Info.Buffer; 5796 const UNICODE_STRING dst = { (USHORT)prinfo->FileNameLength, 5797 (USHORT)prinfo->FileNameLength, prinfo->FileName }; 5798 if (RtlCompareUnicodeString(&dst, 5799 SrvOpen->pAlreadyPrefixedName, FALSE) == 0) { 5800 status = STATUS_SUCCESS; 5801 goto out; 5802 } 5803 } 5804 } 5805 5806 status = nfs41_UpcallCreate(NFS41_FILE_SET, &nfs41_fobx->sec_ctx, 5807 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 5808 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 5809 if (status) goto out; 5810 5811 entry->u.SetFile.InfoClass = InfoClass; 5812 5813 /* original irp has infoclass for remove but we need to rename instead, 5814 * thus we changed the local variable infoclass */ 5815 if (RxContext->Info.FileInformationClass == FileDispositionInformation && 5816 InfoClass == FileRenameInformation) { 5817 entry->buf = &rinfo; 5818 entry->buf_len = sizeof(rinfo); 5819 } else { 5820 entry->buf = RxContext->Info.Buffer; 5821 entry->buf_len = RxContext->Info.Length; 5822 } 5823 #ifdef ENABLE_TIMINGS 5824 InterlockedIncrement(&setattr.sops); 5825 InterlockedAdd64(&setattr.size, entry->u.SetFile.buf_len); 5826 #endif 5827 5828 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 5829 if (status) goto out; 5830 5831 status = map_setfile_error(entry->status); 5832 if (!status) { 5833 if (!nfs41_fobx->deleg_type && entry->ChangeTime && 5834 (SrvOpen->DesiredAccess & 5835 (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA))) 5836 nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime); 5837 nfs41_fcb->changeattr = entry->ChangeTime; 5838 } 5839 RxFreePool(entry); 5840 out: 5841 #ifdef ENABLE_TIMINGS 5842 t2 = KeQueryPerformanceCounter(NULL); 5843 InterlockedIncrement(&setattr.tops); 5844 InterlockedAdd64(&setattr.ticks, t2.QuadPart - t1.QuadPart); 5845 #ifdef ENABLE_INDV_TIMINGS 5846 DbgP("nfs41_SetFileInformation delta = %d op=%d sum=%d\n", 5847 t2.QuadPart - t1.QuadPart, setattr.tops, setattr.ticks); 5848 #endif 5849 #endif 5850 #ifdef DEBUG_FILE_SET 5851 DbgEx(); 5852 #endif 5853 return status; 5854 } 5855 5856 NTSTATUS nfs41_SetFileInformationAtCleanup( 5857 IN OUT PRX_CONTEXT RxContext) 5858 { 5859 NTSTATUS status; 5860 DbgEn(); 5861 status = nfs41_SetFileInformation(RxContext); 5862 DbgEx(); 5863 return status; 5864 } 5865 5866 #ifdef __REACTOS__ 5867 NTSTATUS NTAPI nfs41_IsValidDirectory ( 5868 #else 5869 NTSTATUS nfs41_IsValidDirectory ( 5870 #endif 5871 IN OUT PRX_CONTEXT RxContext, 5872 IN PUNICODE_STRING DirectoryName) 5873 { 5874 return STATUS_SUCCESS; 5875 } 5876 5877 #ifdef __REACTOS__ 5878 NTSTATUS NTAPI nfs41_ComputeNewBufferingState( 5879 #else 5880 NTSTATUS nfs41_ComputeNewBufferingState( 5881 #endif 5882 IN OUT PMRX_SRV_OPEN pSrvOpen, 5883 IN PVOID pMRxContext, 5884 OUT ULONG *pNewBufferingState) 5885 { 5886 NTSTATUS status = STATUS_SUCCESS; 5887 ULONG flag = PtrToUlong(pMRxContext), oldFlags = pSrvOpen->BufferingFlags; 5888 5889 switch(flag) { 5890 case DISABLE_CACHING: 5891 if (pSrvOpen->BufferingFlags & 5892 (FCB_STATE_READBUFFERING_ENABLED | FCB_STATE_READCACHING_ENABLED)) 5893 pSrvOpen->BufferingFlags &= 5894 ~(FCB_STATE_READBUFFERING_ENABLED | 5895 FCB_STATE_READCACHING_ENABLED); 5896 if (pSrvOpen->BufferingFlags & 5897 (FCB_STATE_WRITECACHING_ENABLED | FCB_STATE_WRITEBUFFERING_ENABLED)) 5898 pSrvOpen->BufferingFlags &= 5899 ~(FCB_STATE_WRITECACHING_ENABLED | 5900 FCB_STATE_WRITEBUFFERING_ENABLED); 5901 pSrvOpen->BufferingFlags |= FCB_STATE_DISABLE_LOCAL_BUFFERING; 5902 break; 5903 case ENABLE_READ_CACHING: 5904 pSrvOpen->BufferingFlags |= 5905 (FCB_STATE_READBUFFERING_ENABLED | FCB_STATE_READCACHING_ENABLED); 5906 break; 5907 case ENABLE_WRITE_CACHING: 5908 pSrvOpen->BufferingFlags |= 5909 (FCB_STATE_WRITECACHING_ENABLED | FCB_STATE_WRITEBUFFERING_ENABLED); 5910 break; 5911 case ENABLE_READWRITE_CACHING: 5912 pSrvOpen->BufferingFlags = 5913 (FCB_STATE_READBUFFERING_ENABLED | FCB_STATE_READCACHING_ENABLED | 5914 FCB_STATE_WRITECACHING_ENABLED | FCB_STATE_WRITEBUFFERING_ENABLED); 5915 } 5916 #ifdef DEBUG_TIME_BASED_COHERENCY 5917 DbgP("nfs41_ComputeNewBufferingState: %wZ pSrvOpen %p Old %08x New %08x\n", 5918 pSrvOpen->pAlreadyPrefixedName, pSrvOpen, oldFlags, 5919 pSrvOpen->BufferingFlags); 5920 *pNewBufferingState = pSrvOpen->BufferingFlags; 5921 #endif 5922 return status; 5923 } 5924 5925 void print_readwrite_args( 5926 PRX_CONTEXT RxContext) 5927 { 5928 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 5929 5930 print_debug_header(RxContext); 5931 DbgP("Bytecount 0x%x Byteoffset 0x%x Buffer %p\n", 5932 LowIoContext->ParamsFor.ReadWrite.ByteCount, 5933 LowIoContext->ParamsFor.ReadWrite.ByteOffset, 5934 LowIoContext->ParamsFor.ReadWrite.Buffer); 5935 } 5936 5937 void enable_caching( 5938 PMRX_SRV_OPEN SrvOpen, 5939 PNFS41_FOBX nfs41_fobx, 5940 ULONGLONG ChangeTime, 5941 HANDLE session) 5942 { 5943 ULONG flag = 0; 5944 PLIST_ENTRY pEntry; 5945 nfs41_fcb_list_entry *cur; 5946 BOOLEAN found = FALSE; 5947 5948 if (SrvOpen->DesiredAccess & FILE_READ_DATA) 5949 flag = ENABLE_READ_CACHING; 5950 if ((SrvOpen->DesiredAccess & FILE_WRITE_DATA) && 5951 !nfs41_fobx->write_thru) 5952 flag = ENABLE_WRITE_CACHING; 5953 if ((SrvOpen->DesiredAccess & FILE_READ_DATA) && 5954 (SrvOpen->DesiredAccess & FILE_WRITE_DATA) && 5955 !nfs41_fobx->write_thru) 5956 flag = ENABLE_READWRITE_CACHING; 5957 5958 #if defined(DEBUG_TIME_BASED_COHERENCY) || \ 5959 defined(DEBUG_WRITE) || defined(DEBUG_READ) 5960 print_caching_level(1, flag, SrvOpen->pAlreadyPrefixedName); 5961 #endif 5962 5963 if (!flag) 5964 return; 5965 5966 RxChangeBufferingState((PSRV_OPEN)SrvOpen, ULongToPtr(flag), 1); 5967 5968 ExAcquireFastMutex(&fcblistLock); 5969 pEntry = openlist.head.Flink; 5970 while (!IsListEmpty(&openlist.head)) { 5971 cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry, 5972 nfs41_fcb_list_entry, next); 5973 if (cur->fcb == SrvOpen->pFcb) { 5974 #ifdef DEBUG_TIME_BASED_COHERENCY 5975 DbgP("enable_caching: Looked&Found match for fcb=%p %wZ\n", 5976 SrvOpen->pFcb, SrvOpen->pAlreadyPrefixedName); 5977 #endif 5978 cur->skip = FALSE; 5979 found = TRUE; 5980 break; 5981 } 5982 if (pEntry->Flink == &openlist.head) { 5983 #ifdef DEBUG_TIME_BASED_COHERENCY 5984 DbgP("enable_caching: reached EOL looking for fcb=%p %wZ\n", 5985 SrvOpen->pFcb, SrvOpen->pAlreadyPrefixedName); 5986 #endif 5987 break; 5988 } 5989 pEntry = pEntry->Flink; 5990 } 5991 if (!found && nfs41_fobx->deleg_type) { 5992 nfs41_fcb_list_entry *oentry; 5993 #ifdef DEBUG_TIME_BASED_COHERENCY 5994 DbgP("enable_caching: delegation recalled: srv_open=%p\n", SrvOpen); 5995 #endif 5996 oentry = RxAllocatePoolWithTag(NonPagedPool, 5997 sizeof(nfs41_fcb_list_entry), NFS41_MM_POOLTAG_OPEN); 5998 if (oentry == NULL) return; 5999 oentry->fcb = SrvOpen->pFcb; 6000 oentry->session = session; 6001 oentry->nfs41_fobx = nfs41_fobx; 6002 oentry->ChangeTime = ChangeTime; 6003 oentry->skip = FALSE; 6004 InsertTailList(&openlist.head, &oentry->next); 6005 nfs41_fobx->deleg_type = 0; 6006 } 6007 ExReleaseFastMutex(&fcblistLock); 6008 } 6009 6010 NTSTATUS map_readwrite_errors( 6011 DWORD status) 6012 { 6013 switch (status) { 6014 case ERROR_ACCESS_DENIED: return STATUS_ACCESS_DENIED; 6015 case ERROR_HANDLE_EOF: return STATUS_END_OF_FILE; 6016 case ERROR_FILE_INVALID: return STATUS_FILE_INVALID; 6017 case ERROR_INVALID_PARAMETER: return STATUS_INVALID_PARAMETER; 6018 case ERROR_LOCK_VIOLATION: return STATUS_FILE_LOCK_CONFLICT; 6019 case ERROR_NETWORK_ACCESS_DENIED: return STATUS_NETWORK_ACCESS_DENIED; 6020 case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED; 6021 default: 6022 print_error("failed to map windows error %d to NTSTATUS; " 6023 "defaulting to STATUS_NET_WRITE_FAULT\n", status); 6024 case ERROR_NET_WRITE_FAULT: return STATUS_NET_WRITE_FAULT; 6025 } 6026 } 6027 6028 NTSTATUS check_nfs41_read_args( 6029 IN PRX_CONTEXT RxContext) 6030 { 6031 if (!RxContext->LowIoContext.ParamsFor.ReadWrite.Buffer) 6032 return STATUS_INVALID_USER_BUFFER; 6033 return STATUS_SUCCESS; 6034 } 6035 6036 #ifdef __REACTOS__ 6037 NTSTATUS NTAPI nfs41_Read( 6038 #else 6039 NTSTATUS nfs41_Read( 6040 #endif 6041 IN OUT PRX_CONTEXT RxContext) 6042 { 6043 NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES; 6044 nfs41_updowncall_entry *entry; 6045 BOOLEAN async = FALSE; 6046 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 6047 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 6048 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 6049 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 6050 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 6051 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 6052 __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb); 6053 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 6054 DWORD io_delay; 6055 #ifdef ENABLE_TIMINGS 6056 LARGE_INTEGER t1, t2; 6057 t1 = KeQueryPerformanceCounter(NULL); 6058 #endif 6059 6060 #ifdef DEBUG_READ 6061 DbgEn(); 6062 print_readwrite_args(RxContext); 6063 #endif 6064 status = check_nfs41_read_args(RxContext); 6065 if (status) goto out; 6066 6067 status = nfs41_UpcallCreate(NFS41_READ, &nfs41_fobx->sec_ctx, 6068 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 6069 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 6070 if (status) goto out; 6071 6072 entry->u.ReadWrite.MdlAddress = LowIoContext->ParamsFor.ReadWrite.Buffer; 6073 entry->buf_len = LowIoContext->ParamsFor.ReadWrite.ByteCount; 6074 entry->u.ReadWrite.offset = LowIoContext->ParamsFor.ReadWrite.ByteOffset; 6075 if (FlagOn(RxContext->CurrentIrpSp->FileObject->Flags, 6076 FO_SYNCHRONOUS_IO) == FALSE) { 6077 entry->u.ReadWrite.rxcontext = RxContext; 6078 async = entry->async_op = TRUE; 6079 } 6080 6081 /* assume network speed is 100MB/s and disk speed is 100MB/s so add 6082 * time to transfer requested bytes over the network and read from disk 6083 */ 6084 io_delay = pVNetRootContext->timeout + 2 * entry->buf_len / 104857600; 6085 status = nfs41_UpcallWaitForReply(entry, io_delay); 6086 if (status) goto out; 6087 6088 if (async) { 6089 #ifdef DEBUG_READ 6090 DbgP("This is asynchronous read, returning control back to the user\n"); 6091 #endif 6092 status = STATUS_PENDING; 6093 goto out; 6094 } 6095 6096 if (entry->status == NO_ERROR) { 6097 #ifdef ENABLE_TIMINGS 6098 InterlockedIncrement(&read.sops); 6099 InterlockedAdd64(&read.size, entry->u.ReadWrite.len); 6100 #endif 6101 status = RxContext->CurrentIrp->IoStatus.Status = STATUS_SUCCESS; 6102 RxContext->IoStatusBlock.Information = entry->buf_len; 6103 6104 if ((!BooleanFlagOn(LowIoContext->ParamsFor.ReadWrite.Flags, 6105 LOWIO_READWRITEFLAG_PAGING_IO) && 6106 (SrvOpen->DesiredAccess & FILE_READ_DATA) && 6107 !pVNetRootContext->nocache && !nfs41_fobx->nocache && 6108 !(SrvOpen->BufferingFlags & 6109 (FCB_STATE_READBUFFERING_ENABLED | 6110 FCB_STATE_READCACHING_ENABLED)))) { 6111 enable_caching(SrvOpen, nfs41_fobx, nfs41_fcb->changeattr, 6112 pVNetRootContext->session); 6113 } 6114 } else { 6115 status = map_readwrite_errors(entry->status); 6116 RxContext->CurrentIrp->IoStatus.Status = status; 6117 RxContext->IoStatusBlock.Information = 0; 6118 } 6119 RxFreePool(entry); 6120 out: 6121 #ifdef ENABLE_TIMINGS 6122 t2 = KeQueryPerformanceCounter(NULL); 6123 InterlockedIncrement(&read.tops); 6124 InterlockedAdd64(&read.ticks, t2.QuadPart - t1.QuadPart); 6125 #ifdef ENABLE_INDV_TIMINGS 6126 DbgP("nfs41_Read delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart, 6127 read.tops, read.ticks); 6128 #endif 6129 #endif 6130 #ifdef DEBUG_READ 6131 DbgEx(); 6132 #endif 6133 return status; 6134 } 6135 6136 NTSTATUS check_nfs41_write_args( 6137 IN PRX_CONTEXT RxContext) 6138 { 6139 NTSTATUS status = STATUS_SUCCESS; 6140 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 6141 NFS41GetVNetRootExtension(RxContext->pRelevantSrvOpen->pVNetRoot); 6142 6143 if (!RxContext->LowIoContext.ParamsFor.ReadWrite.Buffer) { 6144 status = STATUS_INVALID_USER_BUFFER; 6145 goto out; 6146 } 6147 6148 if (pVNetRootContext->read_only) { 6149 print_error("check_nfs41_write_args: Read-only mount\n"); 6150 status = STATUS_ACCESS_DENIED; 6151 goto out; 6152 } 6153 out: 6154 return status; 6155 } 6156 6157 #ifdef __REACTOS__ 6158 NTSTATUS NTAPI nfs41_Write( 6159 #else 6160 NTSTATUS nfs41_Write( 6161 #endif 6162 IN OUT PRX_CONTEXT RxContext) 6163 { 6164 NTSTATUS status = STATUS_INSUFFICIENT_RESOURCES; 6165 nfs41_updowncall_entry *entry; 6166 BOOLEAN async = FALSE; 6167 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 6168 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 6169 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 6170 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 6171 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 6172 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 6173 __notnull PNFS41_FCB nfs41_fcb = NFS41GetFcbExtension(RxContext->pFcb); 6174 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 6175 DWORD io_delay; 6176 #ifdef ENABLE_TIMINGS 6177 LARGE_INTEGER t1, t2; 6178 t1 = KeQueryPerformanceCounter(NULL); 6179 #endif 6180 6181 #ifdef DEBUG_WRITE 6182 DbgEn(); 6183 print_readwrite_args(RxContext); 6184 #endif 6185 6186 status = check_nfs41_write_args(RxContext); 6187 if (status) goto out; 6188 6189 status = nfs41_UpcallCreate(NFS41_WRITE, &nfs41_fobx->sec_ctx, 6190 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 6191 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 6192 if (status) goto out; 6193 6194 entry->u.ReadWrite.MdlAddress = LowIoContext->ParamsFor.ReadWrite.Buffer; 6195 entry->buf_len = LowIoContext->ParamsFor.ReadWrite.ByteCount; 6196 entry->u.ReadWrite.offset = LowIoContext->ParamsFor.ReadWrite.ByteOffset; 6197 6198 if (FlagOn(RxContext->CurrentIrpSp->FileObject->Flags, 6199 FO_SYNCHRONOUS_IO) == FALSE) { 6200 entry->u.ReadWrite.rxcontext = RxContext; 6201 async = entry->async_op = TRUE; 6202 } 6203 6204 /* assume network speed is 100MB/s and disk speed is 100MB/s so add 6205 * time to transfer requested bytes over the network and write to disk 6206 */ 6207 io_delay = pVNetRootContext->timeout + 2 * entry->buf_len / 104857600; 6208 status = nfs41_UpcallWaitForReply(entry, io_delay); 6209 if (status) goto out; 6210 6211 if (async) { 6212 #ifdef DEBUG_WRITE 6213 DbgP("This is asynchronous write, returning control back to the user\n"); 6214 #endif 6215 status = STATUS_PENDING; 6216 goto out; 6217 } 6218 6219 if (entry->status == NO_ERROR) { 6220 //update cached file attributes 6221 #ifdef ENABLE_TIMINGS 6222 InterlockedIncrement(&write.sops); 6223 InterlockedAdd64(&write.size, entry->u.ReadWrite.len); 6224 #endif 6225 nfs41_fcb->StandardInfo.EndOfFile.QuadPart = entry->buf_len + 6226 entry->u.ReadWrite.offset; 6227 status = RxContext->CurrentIrp->IoStatus.Status = STATUS_SUCCESS; 6228 RxContext->IoStatusBlock.Information = entry->buf_len; 6229 nfs41_fcb->changeattr = entry->ChangeTime; 6230 6231 //re-enable write buffering 6232 if (!BooleanFlagOn(LowIoContext->ParamsFor.ReadWrite.Flags, 6233 LOWIO_READWRITEFLAG_PAGING_IO) && 6234 (SrvOpen->DesiredAccess & (FILE_WRITE_DATA | FILE_APPEND_DATA)) && 6235 !pVNetRootContext->write_thru && 6236 !pVNetRootContext->nocache && 6237 !nfs41_fobx->write_thru && !nfs41_fobx->nocache && 6238 !(SrvOpen->BufferingFlags & 6239 (FCB_STATE_WRITEBUFFERING_ENABLED | 6240 FCB_STATE_WRITECACHING_ENABLED))) { 6241 enable_caching(SrvOpen, nfs41_fobx, nfs41_fcb->changeattr, 6242 pVNetRootContext->session); 6243 } else if (!nfs41_fobx->deleg_type) 6244 nfs41_update_fcb_list(RxContext->pFcb, entry->ChangeTime); 6245 6246 } else { 6247 status = map_readwrite_errors(entry->status); 6248 RxContext->CurrentIrp->IoStatus.Status = status; 6249 RxContext->IoStatusBlock.Information = 0; 6250 } 6251 RxFreePool(entry); 6252 out: 6253 #ifdef ENABLE_TIMINGS 6254 t2 = KeQueryPerformanceCounter(NULL); 6255 InterlockedIncrement(&write.tops); 6256 InterlockedAdd64(&write.ticks, t2.QuadPart - t1.QuadPart); 6257 #ifdef ENABLE_INDV_TIMINGS 6258 DbgP("nfs41_Write delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart, 6259 write.tops, write.ticks); 6260 #endif 6261 #endif 6262 #ifdef DEBUG_WRITE 6263 DbgEx(); 6264 #endif 6265 return status; 6266 } 6267 6268 #ifdef __REACTOS__ 6269 NTSTATUS NTAPI nfs41_IsLockRealizable( 6270 #else 6271 NTSTATUS nfs41_IsLockRealizable( 6272 #endif 6273 IN OUT PMRX_FCB pFcb, 6274 IN PLARGE_INTEGER ByteOffset, 6275 IN PLARGE_INTEGER Length, 6276 IN ULONG LowIoLockFlags) 6277 { 6278 NTSTATUS status = STATUS_SUCCESS; 6279 #ifdef DEBUG_LOCK 6280 DbgEn(); 6281 DbgP("offset 0x%llx, length 0x%llx, exclusive=%u, blocking=%u\n", 6282 ByteOffset->QuadPart,Length->QuadPart, 6283 BooleanFlagOn(LowIoLockFlags, SL_EXCLUSIVE_LOCK), 6284 !BooleanFlagOn(LowIoLockFlags, SL_FAIL_IMMEDIATELY)); 6285 #endif 6286 6287 /* NFS lock operations with length=0 MUST fail with NFS4ERR_INVAL */ 6288 if (Length->QuadPart == 0) 6289 status = STATUS_NOT_SUPPORTED; 6290 6291 #ifdef DEBUG_LOCK 6292 DbgEx(); 6293 #endif 6294 return status; 6295 } 6296 6297 NTSTATUS map_lock_errors( 6298 DWORD status) 6299 { 6300 switch (status) { 6301 case NO_ERROR: return STATUS_SUCCESS; 6302 case ERROR_NETNAME_DELETED: return STATUS_NETWORK_NAME_DELETED; 6303 case ERROR_LOCK_FAILED: return STATUS_LOCK_NOT_GRANTED; 6304 case ERROR_NOT_LOCKED: return STATUS_RANGE_NOT_LOCKED; 6305 case ERROR_ATOMIC_LOCKS_NOT_SUPPORTED: return STATUS_UNSUCCESSFUL; 6306 case ERROR_OUTOFMEMORY: return STATUS_INSUFFICIENT_RESOURCES; 6307 case ERROR_SHARING_VIOLATION: return STATUS_SHARING_VIOLATION; 6308 case ERROR_FILE_INVALID: return STATUS_FILE_INVALID; 6309 /* if we return ERROR_INVALID_PARAMETER, Windows translates that to 6310 * success!! */ 6311 case ERROR_INVALID_PARAMETER: return STATUS_LOCK_NOT_GRANTED; 6312 default: 6313 print_error("failed to map windows error %d to NTSTATUS; " 6314 "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status); 6315 case ERROR_BAD_NET_RESP: return STATUS_INVALID_NETWORK_RESPONSE; 6316 } 6317 } 6318 6319 void print_lock_args( 6320 PRX_CONTEXT RxContext) 6321 { 6322 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 6323 const ULONG flags = LowIoContext->ParamsFor.Locks.Flags; 6324 print_debug_header(RxContext); 6325 DbgP("offset 0x%llx, length 0x%llx, exclusive=%u, blocking=%u\n", 6326 LowIoContext->ParamsFor.Locks.ByteOffset, 6327 LowIoContext->ParamsFor.Locks.Length, 6328 BooleanFlagOn(flags, SL_EXCLUSIVE_LOCK), 6329 !BooleanFlagOn(flags, SL_FAIL_IMMEDIATELY)); 6330 } 6331 6332 6333 /* use exponential backoff between polls for blocking locks */ 6334 #define MSEC_TO_RELATIVE_WAIT (-10000) 6335 #define MIN_LOCK_POLL_WAIT (500 * MSEC_TO_RELATIVE_WAIT) /* 500ms */ 6336 #define MAX_LOCK_POLL_WAIT (30000 * MSEC_TO_RELATIVE_WAIT) /* 30s */ 6337 6338 void denied_lock_backoff( 6339 IN OUT PLARGE_INTEGER delay) 6340 { 6341 if (delay->QuadPart == 0) 6342 delay->QuadPart = MIN_LOCK_POLL_WAIT; 6343 else 6344 delay->QuadPart <<= 1; 6345 6346 if (delay->QuadPart < MAX_LOCK_POLL_WAIT) 6347 delay->QuadPart = MAX_LOCK_POLL_WAIT; 6348 } 6349 6350 #ifdef __REACTOS__ 6351 NTSTATUS NTAPI nfs41_Lock( 6352 #else 6353 NTSTATUS nfs41_Lock( 6354 #endif 6355 IN OUT PRX_CONTEXT RxContext) 6356 { 6357 NTSTATUS status = STATUS_SUCCESS; 6358 nfs41_updowncall_entry *entry; 6359 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 6360 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 6361 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 6362 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 6363 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 6364 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 6365 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 6366 const ULONG flags = LowIoContext->ParamsFor.Locks.Flags; 6367 #ifdef _MSC_VER 6368 LARGE_INTEGER poll_delay = {0}; 6369 #else 6370 LARGE_INTEGER poll_delay; 6371 #endif 6372 #ifdef ENABLE_TIMINGS 6373 LARGE_INTEGER t1, t2; 6374 t1 = KeQueryPerformanceCounter(NULL); 6375 #endif 6376 6377 #ifndef _MSC_VER 6378 poll_delay.QuadPart = 0; 6379 #endif 6380 6381 #ifdef DEBUG_LOCK 6382 DbgEn(); 6383 print_lock_args(RxContext); 6384 #endif 6385 6386 /* RxReleaseFcbResourceForThreadInMRx(RxContext, RxContext->pFcb, 6387 LowIoContext->ResourceThreadId); */ 6388 6389 status = nfs41_UpcallCreate(NFS41_LOCK, &nfs41_fobx->sec_ctx, 6390 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 6391 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 6392 if (status) goto out; 6393 6394 entry->u.Lock.offset = LowIoContext->ParamsFor.Locks.ByteOffset; 6395 entry->u.Lock.length = LowIoContext->ParamsFor.Locks.Length; 6396 entry->u.Lock.exclusive = BooleanFlagOn(flags, SL_EXCLUSIVE_LOCK); 6397 entry->u.Lock.blocking = !BooleanFlagOn(flags, SL_FAIL_IMMEDIATELY); 6398 6399 retry_upcall: 6400 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 6401 if (status) goto out; 6402 6403 /* blocking locks keep trying until it succeeds */ 6404 if (entry->status == ERROR_LOCK_FAILED && entry->u.Lock.blocking) { 6405 denied_lock_backoff(&poll_delay); 6406 DbgP("returned ERROR_LOCK_FAILED; retrying in %llums\n", 6407 poll_delay.QuadPart / MSEC_TO_RELATIVE_WAIT); 6408 KeDelayExecutionThread(KernelMode, FALSE, &poll_delay); 6409 entry->state = NFS41_WAITING_FOR_UPCALL; 6410 goto retry_upcall; 6411 } 6412 6413 status = map_lock_errors(entry->status); 6414 RxContext->CurrentIrp->IoStatus.Status = status; 6415 6416 RxFreePool(entry); 6417 out: 6418 #ifdef ENABLE_TIMINGS 6419 t2 = KeQueryPerformanceCounter(NULL); 6420 InterlockedIncrement(&lock.tops); 6421 InterlockedAdd64(&lock.ticks, t2.QuadPart - t1.QuadPart); 6422 #ifdef ENABLE_INDV_TIMINGS 6423 DbgP("nfs41_Lock delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart, 6424 lock.tops, lock.ticks); 6425 #endif 6426 #endif 6427 #ifdef DEBUG_LOCK 6428 DbgEx(); 6429 #endif 6430 return status; 6431 } 6432 6433 void print_unlock_args( 6434 PRX_CONTEXT RxContext) 6435 { 6436 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 6437 print_debug_header(RxContext); 6438 if (LowIoContext->Operation == LOWIO_OP_UNLOCK_MULTIPLE) { 6439 PLOWIO_LOCK_LIST lock = LowIoContext->ParamsFor.Locks.LockList; 6440 DbgP("LOWIO_OP_UNLOCK_MULTIPLE:"); 6441 while (lock) { 6442 DbgP(" (offset=%llu, length=%llu)", lock->ByteOffset, lock->Length); 6443 lock = lock->Next; 6444 } 6445 DbgP("\n"); 6446 } else { 6447 DbgP("LOWIO_OP_UNLOCK: offset=%llu, length=%llu\n", 6448 LowIoContext->ParamsFor.Locks.ByteOffset, 6449 LowIoContext->ParamsFor.Locks.Length); 6450 } 6451 } 6452 6453 __inline ULONG unlock_list_count( 6454 PLOWIO_LOCK_LIST lock) 6455 { 6456 ULONG count = 0; 6457 while (lock) { 6458 count++; 6459 lock = lock->Next; 6460 } 6461 return count; 6462 } 6463 6464 #ifdef __REACTOS__ 6465 NTSTATUS NTAPI nfs41_Unlock( 6466 #else 6467 NTSTATUS nfs41_Unlock( 6468 #endif 6469 IN OUT PRX_CONTEXT RxContext) 6470 { 6471 NTSTATUS status = STATUS_SUCCESS; 6472 nfs41_updowncall_entry *entry; 6473 PLOWIO_CONTEXT LowIoContext = &RxContext->LowIoContext; 6474 __notnull PNFS41_FOBX nfs41_fobx = NFS41GetFobxExtension(RxContext->pFobx); 6475 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 6476 __notnull PNFS41_V_NET_ROOT_EXTENSION pVNetRootContext = 6477 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 6478 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 6479 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 6480 #ifdef ENABLE_TIMINGS 6481 LARGE_INTEGER t1, t2; 6482 t1 = KeQueryPerformanceCounter(NULL); 6483 #endif 6484 #ifdef DEBUG_LOCK 6485 DbgEn(); 6486 print_lock_args(RxContext); 6487 #endif 6488 6489 /* RxReleaseFcbResourceForThreadInMRx(RxContext, RxContext->pFcb, 6490 LowIoContext->ResourceThreadId); */ 6491 6492 status = nfs41_UpcallCreate(NFS41_UNLOCK, &nfs41_fobx->sec_ctx, 6493 pVNetRootContext->session, nfs41_fobx->nfs41_open_state, 6494 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 6495 if (status) goto out; 6496 6497 if (LowIoContext->Operation == LOWIO_OP_UNLOCK_MULTIPLE) { 6498 entry->u.Unlock.count = unlock_list_count( 6499 LowIoContext->ParamsFor.Locks.LockList); 6500 RtlCopyMemory(&entry->u.Unlock.locks, 6501 LowIoContext->ParamsFor.Locks.LockList, 6502 sizeof(LOWIO_LOCK_LIST)); 6503 } else { 6504 entry->u.Unlock.count = 1; 6505 entry->u.Unlock.locks.ByteOffset = 6506 LowIoContext->ParamsFor.Locks.ByteOffset; 6507 entry->u.Unlock.locks.Length = 6508 LowIoContext->ParamsFor.Locks.Length; 6509 } 6510 6511 status = nfs41_UpcallWaitForReply(entry, pVNetRootContext->timeout); 6512 if (status) goto out; 6513 6514 status = map_lock_errors(entry->status); 6515 RxContext->CurrentIrp->IoStatus.Status = status; 6516 RxFreePool(entry); 6517 out: 6518 #ifdef ENABLE_TIMINGS 6519 t2 = KeQueryPerformanceCounter(NULL); 6520 InterlockedIncrement(&unlock.tops); 6521 InterlockedAdd64(&unlock.ticks, t2.QuadPart - t1.QuadPart); 6522 #ifdef ENABLE_INDV_TIMINGS 6523 DbgP("nfs41_Unlock delta = %d op=%d sum=%d\n", t2.QuadPart - t1.QuadPart, 6524 unlock.tops, unlock.ticks); 6525 #endif 6526 #endif 6527 #ifdef DEBUG_LOCK 6528 DbgEx(); 6529 #endif 6530 return status; 6531 } 6532 6533 NTSTATUS map_symlink_errors( 6534 NTSTATUS status) 6535 { 6536 switch (status) { 6537 case NO_ERROR: return STATUS_SUCCESS; 6538 case ERROR_INVALID_REPARSE_DATA: return STATUS_IO_REPARSE_DATA_INVALID; 6539 case ERROR_NOT_A_REPARSE_POINT: return STATUS_NOT_A_REPARSE_POINT; 6540 case ERROR_ACCESS_DENIED: return STATUS_ACCESS_DENIED; 6541 case ERROR_NOT_EMPTY: return STATUS_DIRECTORY_NOT_EMPTY; 6542 case ERROR_OUTOFMEMORY: return STATUS_INSUFFICIENT_RESOURCES; 6543 case ERROR_INSUFFICIENT_BUFFER: return STATUS_BUFFER_TOO_SMALL; 6544 case STATUS_BUFFER_TOO_SMALL: 6545 case ERROR_BUFFER_OVERFLOW: return STATUS_BUFFER_OVERFLOW; 6546 default: 6547 print_error("failed to map windows error %d to NTSTATUS; " 6548 "defaulting to STATUS_INVALID_NETWORK_RESPONSE\n", status); 6549 case ERROR_BAD_NET_RESP: return STATUS_INVALID_NETWORK_RESPONSE; 6550 } 6551 } 6552 6553 void print_reparse_buffer( 6554 PREPARSE_DATA_BUFFER Reparse) 6555 { 6556 UNICODE_STRING name; 6557 DbgP("ReparseTag: %08X\n", Reparse->ReparseTag); 6558 DbgP("ReparseDataLength: %8u\n", Reparse->ReparseDataLength); 6559 DbgP("Reserved: %8u\n", Reparse->Reserved); 6560 DbgP("SubstituteNameOffset: %8u\n", 6561 Reparse->SymbolicLinkReparseBuffer.SubstituteNameOffset); 6562 DbgP("SubstituteNameLength: %8u\n", 6563 Reparse->SymbolicLinkReparseBuffer.SubstituteNameLength); 6564 DbgP("PrintNameOffset: %8u\n", 6565 Reparse->SymbolicLinkReparseBuffer.PrintNameOffset); 6566 DbgP("PrintNameLength: %8u\n", 6567 Reparse->SymbolicLinkReparseBuffer.PrintNameLength); 6568 DbgP("Flags: %08X\n", 6569 Reparse->SymbolicLinkReparseBuffer.Flags); 6570 6571 name.Buffer = &Reparse->SymbolicLinkReparseBuffer.PathBuffer[ 6572 Reparse->SymbolicLinkReparseBuffer.SubstituteNameOffset/sizeof(WCHAR)]; 6573 name.MaximumLength = name.Length = 6574 Reparse->SymbolicLinkReparseBuffer.SubstituteNameLength; 6575 DbgP("SubstituteName: %wZ\n", &name); 6576 6577 name.Buffer = &Reparse->SymbolicLinkReparseBuffer.PathBuffer[ 6578 Reparse->SymbolicLinkReparseBuffer.PrintNameOffset/sizeof(WCHAR)]; 6579 name.MaximumLength = name.Length = 6580 Reparse->SymbolicLinkReparseBuffer.PrintNameLength; 6581 DbgP("PrintName: %wZ\n", &name); 6582 } 6583 6584 NTSTATUS check_nfs41_setreparse_args( 6585 IN PRX_CONTEXT RxContext) 6586 { 6587 NTSTATUS status = STATUS_SUCCESS; 6588 __notnull XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl; 6589 __notnull PREPARSE_DATA_BUFFER Reparse = (PREPARSE_DATA_BUFFER)FsCtl->pInputBuffer; 6590 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 6591 __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext = 6592 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 6593 const ULONG HeaderLen = REPARSE_DATA_BUFFER_HEADER_SIZE; 6594 6595 /* access checks */ 6596 if (VNetRootContext->read_only) { 6597 status = STATUS_MEDIA_WRITE_PROTECTED; 6598 goto out; 6599 } 6600 if (!(SrvOpen->DesiredAccess & (FILE_WRITE_DATA|FILE_WRITE_ATTRIBUTES))) { 6601 status = STATUS_ACCESS_DENIED; 6602 goto out; 6603 } 6604 6605 /* must have a filename longer than vnetroot name, 6606 * or it's trying to operate on the volume itself */ 6607 if (is_root_directory(RxContext)) { 6608 status = STATUS_INVALID_PARAMETER; 6609 goto out; 6610 } 6611 if (FsCtl->pOutputBuffer != NULL) { 6612 status = STATUS_INVALID_PARAMETER; 6613 goto out; 6614 } 6615 6616 /* validate input buffer and length */ 6617 if (!Reparse) { 6618 status = STATUS_INVALID_BUFFER_SIZE; 6619 goto out; 6620 } 6621 6622 if (FsCtl->InputBufferLength < HeaderLen || 6623 FsCtl->InputBufferLength > MAXIMUM_REPARSE_DATA_BUFFER_SIZE) { 6624 status = STATUS_IO_REPARSE_DATA_INVALID; 6625 goto out; 6626 } 6627 if (FsCtl->InputBufferLength != HeaderLen + Reparse->ReparseDataLength) { 6628 status = STATUS_IO_REPARSE_DATA_INVALID; 6629 goto out; 6630 } 6631 6632 /* validate reparse tag */ 6633 if (!IsReparseTagValid(Reparse->ReparseTag)) { 6634 status = STATUS_IO_REPARSE_TAG_INVALID; 6635 goto out; 6636 } 6637 if (Reparse->ReparseTag != IO_REPARSE_TAG_SYMLINK) { 6638 status = STATUS_IO_REPARSE_TAG_MISMATCH; 6639 goto out; 6640 } 6641 out: 6642 return status; 6643 } 6644 6645 NTSTATUS nfs41_SetReparsePoint( 6646 IN OUT PRX_CONTEXT RxContext) 6647 { 6648 NTSTATUS status; 6649 UNICODE_STRING TargetName; 6650 __notnull XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl; 6651 __notnull PREPARSE_DATA_BUFFER Reparse = (PREPARSE_DATA_BUFFER)FsCtl->pInputBuffer; 6652 __notnull PNFS41_FOBX Fobx = NFS41GetFobxExtension(RxContext->pFobx); 6653 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 6654 __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext = 6655 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 6656 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 6657 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 6658 nfs41_updowncall_entry *entry; 6659 6660 #ifdef DEBUG_SYMLINK 6661 DbgEn(); 6662 print_debug_header(RxContext); 6663 print_reparse_buffer(Reparse); 6664 #endif 6665 status = check_nfs41_setreparse_args(RxContext); 6666 if (status) goto out; 6667 6668 TargetName.MaximumLength = TargetName.Length = 6669 Reparse->SymbolicLinkReparseBuffer.PrintNameLength; 6670 TargetName.Buffer = &Reparse->SymbolicLinkReparseBuffer.PathBuffer[ 6671 Reparse->SymbolicLinkReparseBuffer.PrintNameOffset/sizeof(WCHAR)]; 6672 6673 status = nfs41_UpcallCreate(NFS41_SYMLINK, &Fobx->sec_ctx, 6674 VNetRootContext->session, Fobx->nfs41_open_state, 6675 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 6676 if (status) goto out; 6677 6678 entry->u.Symlink.target = &TargetName; 6679 entry->u.Symlink.set = TRUE; 6680 6681 status = nfs41_UpcallWaitForReply(entry, VNetRootContext->timeout); 6682 if (status) goto out; 6683 6684 status = map_symlink_errors(entry->status); 6685 RxFreePool(entry); 6686 out: 6687 #ifdef DEBUG_SYMLINK 6688 DbgEx(); 6689 #endif 6690 return status; 6691 } 6692 6693 NTSTATUS check_nfs41_getreparse_args( 6694 PRX_CONTEXT RxContext) 6695 { 6696 NTSTATUS status = STATUS_SUCCESS; 6697 XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl; 6698 const USHORT HeaderLen = FIELD_OFFSET(REPARSE_DATA_BUFFER, 6699 SymbolicLinkReparseBuffer.PathBuffer); 6700 6701 /* must have a filename longer than vnetroot name, 6702 * or it's trying to operate on the volume itself */ 6703 if (is_root_directory(RxContext)) { 6704 status = STATUS_INVALID_PARAMETER; 6705 goto out; 6706 } 6707 /* ifs reparse tests expect STATUS_INVALID_PARAMETER, 6708 * but 'dir' passes a buffer here when querying symlinks 6709 if (FsCtl->pInputBuffer != NULL) { 6710 status = STATUS_INVALID_PARAMETER; 6711 goto out; 6712 } */ 6713 if (!FsCtl->pOutputBuffer) { 6714 status = STATUS_INVALID_USER_BUFFER; 6715 goto out; 6716 } 6717 if (!BooleanFlagOn(RxContext->pFcb->Attributes, 6718 FILE_ATTRIBUTE_REPARSE_POINT)) { 6719 status = STATUS_NOT_A_REPARSE_POINT; 6720 DbgP("FILE_ATTRIBUTE_REPARSE_POINT is not set!\n"); 6721 goto out; 6722 } 6723 6724 if (FsCtl->OutputBufferLength < HeaderLen) { 6725 RxContext->InformationToReturn = HeaderLen; 6726 status = STATUS_BUFFER_TOO_SMALL; 6727 goto out; 6728 } 6729 out: 6730 return status; 6731 } 6732 6733 NTSTATUS nfs41_GetReparsePoint( 6734 IN OUT PRX_CONTEXT RxContext) 6735 { 6736 NTSTATUS status; 6737 UNICODE_STRING TargetName; 6738 XXCTL_LOWIO_COMPONENT *FsCtl = &RxContext->LowIoContext.ParamsFor.FsCtl; 6739 __notnull PNFS41_FOBX Fobx = NFS41GetFobxExtension(RxContext->pFobx); 6740 __notnull PMRX_SRV_OPEN SrvOpen = RxContext->pRelevantSrvOpen; 6741 __notnull PNFS41_V_NET_ROOT_EXTENSION VNetRootContext = 6742 NFS41GetVNetRootExtension(SrvOpen->pVNetRoot); 6743 __notnull PNFS41_NETROOT_EXTENSION pNetRootContext = 6744 NFS41GetNetRootExtension(SrvOpen->pVNetRoot->pNetRoot); 6745 nfs41_updowncall_entry *entry; 6746 const USHORT HeaderLen = FIELD_OFFSET(REPARSE_DATA_BUFFER, 6747 SymbolicLinkReparseBuffer.PathBuffer); 6748 6749 #ifdef DEBUG_SYMLINK 6750 DbgEn(); 6751 #endif 6752 status = check_nfs41_getreparse_args(RxContext); 6753 if (status) goto out; 6754 6755 TargetName.Buffer = (PWCH)((PBYTE)FsCtl->pOutputBuffer + HeaderLen); 6756 TargetName.MaximumLength = (USHORT)min(FsCtl->OutputBufferLength - 6757 HeaderLen, 0xFFFF); 6758 6759 status = nfs41_UpcallCreate(NFS41_SYMLINK, &Fobx->sec_ctx, 6760 VNetRootContext->session, Fobx->nfs41_open_state, 6761 pNetRootContext->nfs41d_version, SrvOpen->pAlreadyPrefixedName, &entry); 6762 if (status) goto out; 6763 6764 entry->u.Symlink.target = &TargetName; 6765 entry->u.Symlink.set = FALSE; 6766 6767 status = nfs41_UpcallWaitForReply(entry, VNetRootContext->timeout); 6768 if (status) goto out; 6769 6770 status = map_symlink_errors(entry->status); 6771 if (status == STATUS_SUCCESS) { 6772 /* fill in the output buffer */ 6773 PREPARSE_DATA_BUFFER Reparse = (PREPARSE_DATA_BUFFER) 6774 FsCtl->pOutputBuffer; 6775 Reparse->ReparseTag = IO_REPARSE_TAG_SYMLINK; 6776 Reparse->ReparseDataLength = HeaderLen + TargetName.Length - 6777 REPARSE_DATA_BUFFER_HEADER_SIZE; 6778 Reparse->Reserved = 0; 6779 Reparse->SymbolicLinkReparseBuffer.Flags = SYMLINK_FLAG_RELATIVE; 6780 /* PrintName and SubstituteName point to the same string */ 6781 Reparse->SymbolicLinkReparseBuffer.SubstituteNameOffset = 0; 6782 Reparse->SymbolicLinkReparseBuffer.SubstituteNameLength = 6783 TargetName.Length; 6784 Reparse->SymbolicLinkReparseBuffer.PrintNameOffset = 0; 6785 Reparse->SymbolicLinkReparseBuffer.PrintNameLength = TargetName.Length; 6786 print_reparse_buffer(Reparse); 6787 6788 RxContext->IoStatusBlock.Information = HeaderLen + TargetName.Length; 6789 } else if (status == STATUS_BUFFER_TOO_SMALL) { 6790 RxContext->InformationToReturn = HeaderLen + TargetName.Length; 6791 } 6792 RxFreePool(entry); 6793 out: 6794 #ifdef DEBUG_SYMLINK 6795 DbgEx(); 6796 #endif 6797 return status; 6798 } 6799 6800 #ifdef __REACTOS__ 6801 NTSTATUS NTAPI nfs41_FsCtl( 6802 #else 6803 NTSTATUS nfs41_FsCtl( 6804 #endif 6805 IN OUT PRX_CONTEXT RxContext) 6806 { 6807 NTSTATUS status = STATUS_INVALID_DEVICE_REQUEST; 6808 #ifdef DEBUG_MISC 6809 DbgEn(); 6810 print_debug_header(RxContext); 6811 #endif 6812 switch (RxContext->LowIoContext.ParamsFor.FsCtl.FsControlCode) { 6813 case FSCTL_SET_REPARSE_POINT: 6814 status = nfs41_SetReparsePoint(RxContext); 6815 break; 6816 6817 case FSCTL_GET_REPARSE_POINT: 6818 status = nfs41_GetReparsePoint(RxContext); 6819 break; 6820 #ifdef DEBUG_MISC 6821 default: 6822 DbgP("FsControlCode: %d\n", 6823 RxContext->LowIoContext.ParamsFor.FsCtl.FsControlCode); 6824 #endif 6825 } 6826 #ifdef DEBUG_MISC 6827 DbgEx(); 6828 #endif 6829 return status; 6830 } 6831 6832 #ifdef __REACTOS__ 6833 NTSTATUS NTAPI nfs41_CompleteBufferingStateChangeRequest( 6834 #else 6835 NTSTATUS nfs41_CompleteBufferingStateChangeRequest( 6836 #endif 6837 IN OUT PRX_CONTEXT RxContext, 6838 IN OUT PMRX_SRV_OPEN SrvOpen, 6839 IN PVOID pContext) 6840 { 6841 return STATUS_SUCCESS; 6842 } 6843 6844 #ifdef __REACTOS__ 6845 NTSTATUS NTAPI nfs41_FsdDispatch ( 6846 #else 6847 NTSTATUS nfs41_FsdDispatch ( 6848 #endif 6849 IN PDEVICE_OBJECT dev, 6850 IN PIRP Irp) 6851 { 6852 #ifdef DEBUG_FSDDISPATCH 6853 PIO_STACK_LOCATION IrpSp = IoGetCurrentIrpStackLocation( Irp ); 6854 #endif 6855 NTSTATUS status; 6856 6857 #ifdef DEBUG_FSDDISPATCH 6858 DbgEn(); 6859 DbgP("CURRENT IRP = %d.%d\n", IrpSp->MajorFunction, IrpSp->MinorFunction); 6860 if(IrpSp->FileObject) 6861 DbgP("FileOject %p Filename %wZ\n", IrpSp->FileObject, 6862 &IrpSp->FileObject->FileName); 6863 #endif 6864 6865 if (dev != (PDEVICE_OBJECT)nfs41_dev) { 6866 print_error("*** not ours ***\n"); 6867 Irp->IoStatus.Status = STATUS_INVALID_DEVICE_REQUEST; 6868 Irp->IoStatus.Information = 0; 6869 IoCompleteRequest(Irp, IO_NO_INCREMENT ); 6870 status = STATUS_INVALID_DEVICE_REQUEST; 6871 goto out; 6872 } 6873 6874 status = RxFsdDispatch((PRDBSS_DEVICE_OBJECT)dev,Irp); 6875 /* AGLO: 08/05/2009 - looks like RxFsdDispatch frees IrpSp */ 6876 6877 out: 6878 #ifdef DEBUG_FSDDISPATCH 6879 DbgP("IoStatus status = 0x%x info = 0x%x\n", Irp->IoStatus.Status, 6880 Irp->IoStatus.Information); 6881 DbgEx(); 6882 #endif 6883 return status; 6884 } 6885 6886 #ifdef __REACTOS__ 6887 NTSTATUS NTAPI nfs41_Unimplemented( 6888 #else 6889 NTSTATUS nfs41_Unimplemented( 6890 #endif 6891 PRX_CONTEXT RxContext) 6892 { 6893 return STATUS_NOT_IMPLEMENTED; 6894 } 6895 6896 #ifdef __REACTOS__ 6897 NTSTATUS NTAPI nfs41_AreFilesAliased( 6898 #else 6899 NTSTATUS nfs41_AreFilesAliased( 6900 #endif 6901 PFCB a, 6902 PFCB b) 6903 { 6904 return STATUS_NOT_IMPLEMENTED; 6905 } 6906 6907 NTSTATUS nfs41_init_ops() 6908 { 6909 DbgEn(); 6910 6911 ZeroAndInitializeNodeType(&nfs41_ops, RDBSS_NTC_MINIRDR_DISPATCH, 6912 sizeof(MINIRDR_DISPATCH)); 6913 6914 nfs41_ops.MRxFlags = (RDBSS_MANAGE_NET_ROOT_EXTENSION | 6915 RDBSS_MANAGE_V_NET_ROOT_EXTENSION | 6916 RDBSS_MANAGE_FCB_EXTENSION | 6917 RDBSS_MANAGE_FOBX_EXTENSION); 6918 6919 nfs41_ops.MRxSrvCallSize = 0; // srvcall extension is not handled in rdbss 6920 nfs41_ops.MRxNetRootSize = sizeof(NFS41_NETROOT_EXTENSION); 6921 nfs41_ops.MRxVNetRootSize = sizeof(NFS41_V_NET_ROOT_EXTENSION); 6922 nfs41_ops.MRxFcbSize = sizeof(NFS41_FCB); 6923 nfs41_ops.MRxFobxSize = sizeof(NFS41_FOBX); 6924 6925 // Mini redirector cancel routine .. 6926 6927 nfs41_ops.MRxCancel = NULL; 6928 6929 // 6930 // Mini redirector Start/Stop. Each mini-rdr can be started or stopped 6931 // while the others continue to operate. 6932 // 6933 6934 nfs41_ops.MRxStart = nfs41_Start; 6935 nfs41_ops.MRxStop = nfs41_Stop; 6936 nfs41_ops.MRxDevFcbXXXControlFile = nfs41_DevFcbXXXControlFile; 6937 6938 // 6939 // Mini redirector name resolution. 6940 // 6941 6942 nfs41_ops.MRxCreateSrvCall = nfs41_CreateSrvCall; 6943 nfs41_ops.MRxSrvCallWinnerNotify = nfs41_SrvCallWinnerNotify; 6944 nfs41_ops.MRxCreateVNetRoot = nfs41_CreateVNetRoot; 6945 nfs41_ops.MRxExtractNetRootName = nfs41_ExtractNetRootName; 6946 nfs41_ops.MRxFinalizeSrvCall = nfs41_FinalizeSrvCall; 6947 nfs41_ops.MRxFinalizeNetRoot = nfs41_FinalizeNetRoot; 6948 nfs41_ops.MRxFinalizeVNetRoot = nfs41_FinalizeVNetRoot; 6949 6950 // 6951 // File System Object Creation/Deletion. 6952 // 6953 6954 nfs41_ops.MRxCreate = nfs41_Create; 6955 nfs41_ops.MRxCollapseOpen = nfs41_CollapseOpen; 6956 nfs41_ops.MRxShouldTryToCollapseThisOpen = nfs41_ShouldTryToCollapseThisOpen; 6957 nfs41_ops.MRxExtendForCache = nfs41_ExtendForCache; 6958 nfs41_ops.MRxExtendForNonCache = nfs41_ExtendForCache; 6959 nfs41_ops.MRxCloseSrvOpen = nfs41_CloseSrvOpen; 6960 nfs41_ops.MRxFlush = nfs41_Flush; 6961 nfs41_ops.MRxDeallocateForFcb = nfs41_DeallocateForFcb; 6962 nfs41_ops.MRxDeallocateForFobx = nfs41_DeallocateForFobx; 6963 nfs41_ops.MRxIsLockRealizable = nfs41_IsLockRealizable; 6964 6965 // 6966 // File System Objects query/Set 6967 // 6968 6969 nfs41_ops.MRxQueryDirectory = nfs41_QueryDirectory; 6970 nfs41_ops.MRxQueryVolumeInfo = nfs41_QueryVolumeInformation; 6971 nfs41_ops.MRxQueryEaInfo = nfs41_QueryEaInformation; 6972 nfs41_ops.MRxSetEaInfo = nfs41_SetEaInformation; 6973 nfs41_ops.MRxQuerySdInfo = nfs41_QuerySecurityInformation; 6974 nfs41_ops.MRxSetSdInfo = nfs41_SetSecurityInformation; 6975 nfs41_ops.MRxQueryFileInfo = nfs41_QueryFileInformation; 6976 nfs41_ops.MRxSetFileInfo = nfs41_SetFileInformation; 6977 6978 // 6979 // Buffering state change 6980 // 6981 6982 nfs41_ops.MRxComputeNewBufferingState = nfs41_ComputeNewBufferingState; 6983 6984 // 6985 // File System Object I/O 6986 // 6987 6988 nfs41_ops.MRxLowIOSubmit[LOWIO_OP_READ] = nfs41_Read; 6989 nfs41_ops.MRxLowIOSubmit[LOWIO_OP_WRITE] = nfs41_Write; 6990 nfs41_ops.MRxLowIOSubmit[LOWIO_OP_SHAREDLOCK] = nfs41_Lock; 6991 nfs41_ops.MRxLowIOSubmit[LOWIO_OP_EXCLUSIVELOCK] = nfs41_Lock; 6992 nfs41_ops.MRxLowIOSubmit[LOWIO_OP_UNLOCK] = nfs41_Unlock; 6993 nfs41_ops.MRxLowIOSubmit[LOWIO_OP_UNLOCK_MULTIPLE] = nfs41_Unlock; 6994 nfs41_ops.MRxLowIOSubmit[LOWIO_OP_FSCTL] = nfs41_FsCtl; 6995 6996 // 6997 // Miscellanous 6998 // 6999 7000 nfs41_ops.MRxCompleteBufferingStateChangeRequest = 7001 nfs41_CompleteBufferingStateChangeRequest; 7002 nfs41_ops.MRxIsValidDirectory = nfs41_IsValidDirectory; 7003 7004 nfs41_ops.MRxTruncate = nfs41_Unimplemented; 7005 nfs41_ops.MRxZeroExtend = nfs41_Unimplemented; 7006 nfs41_ops.MRxAreFilesAliased = nfs41_AreFilesAliased; 7007 nfs41_ops.MRxQueryQuotaInfo = nfs41_Unimplemented; 7008 nfs41_ops.MRxSetQuotaInfo = nfs41_Unimplemented; 7009 nfs41_ops.MRxSetVolumeInfo = nfs41_Unimplemented; 7010 7011 DbgR(); 7012 return(STATUS_SUCCESS); 7013 } 7014 7015 KSTART_ROUTINE fcbopen_main; 7016 #ifdef __REACTOS__ 7017 VOID NTAPI fcbopen_main(PVOID ctx) 7018 #else 7019 VOID fcbopen_main(PVOID ctx) 7020 #endif 7021 { 7022 NTSTATUS status; 7023 LARGE_INTEGER timeout; 7024 7025 DbgEn(); 7026 timeout.QuadPart = RELATIVE(SECONDS(30)); 7027 while(1) { 7028 PLIST_ENTRY pEntry; 7029 nfs41_fcb_list_entry *cur; 7030 status = KeDelayExecutionThread(KernelMode, TRUE, &timeout); 7031 ExAcquireFastMutex(&fcblistLock); 7032 pEntry = openlist.head.Flink; 7033 while (!IsListEmpty(&openlist.head)) { 7034 PNFS41_NETROOT_EXTENSION pNetRootContext; 7035 nfs41_updowncall_entry *entry; 7036 FILE_BASIC_INFORMATION binfo; 7037 PNFS41_FCB nfs41_fcb; 7038 cur = (nfs41_fcb_list_entry *)CONTAINING_RECORD(pEntry, 7039 nfs41_fcb_list_entry, next); 7040 7041 #ifdef DEBUG_TIME_BASED_COHERENCY 7042 DbgP("fcbopen_main: Checking attributes for fcb=%p " 7043 "change_time=%llu skipping=%d\n", cur->fcb, 7044 cur->ChangeTime, cur->skip); 7045 #endif 7046 if (cur->skip) goto out; 7047 pNetRootContext = 7048 NFS41GetNetRootExtension(cur->fcb->pNetRoot); 7049 /* place an upcall for this srv_open */ 7050 status = nfs41_UpcallCreate(NFS41_FILE_QUERY, 7051 &cur->nfs41_fobx->sec_ctx, cur->session, 7052 cur->nfs41_fobx->nfs41_open_state, 7053 pNetRootContext->nfs41d_version, NULL, &entry); 7054 if (status) goto out; 7055 7056 entry->u.QueryFile.InfoClass = FileBasicInformation; 7057 entry->buf = &binfo; 7058 entry->buf_len = sizeof(binfo); 7059 7060 status = nfs41_UpcallWaitForReply(entry, UPCALL_TIMEOUT_DEFAULT); 7061 if (status) goto out; 7062 7063 if (cur->ChangeTime != entry->ChangeTime) { 7064 ULONG flag = DISABLE_CACHING; 7065 PMRX_SRV_OPEN srv_open; 7066 PLIST_ENTRY psrvEntry; 7067 #ifdef DEBUG_TIME_BASED_COHERENCY 7068 DbgP("fcbopen_main: old ctime=%llu new_ctime=%llu\n", 7069 cur->ChangeTime, entry->ChangeTime); 7070 #endif 7071 cur->ChangeTime = entry->ChangeTime; 7072 cur->skip = TRUE; 7073 psrvEntry = &cur->fcb->SrvOpenList; 7074 psrvEntry = psrvEntry->Flink; 7075 while (!IsListEmpty(&cur->fcb->SrvOpenList)) { 7076 srv_open = (PMRX_SRV_OPEN)CONTAINING_RECORD(psrvEntry, 7077 MRX_SRV_OPEN, SrvOpenQLinks); 7078 if (srv_open->DesiredAccess & 7079 (FILE_READ_DATA | FILE_WRITE_DATA | FILE_APPEND_DATA)) { 7080 #ifdef DEBUG_TIME_BASED_COHERENCY 7081 DbgP("fcbopen_main: ************ Invalidate the cache %wZ" 7082 "************\n", srv_open->pAlreadyPrefixedName); 7083 #endif 7084 RxIndicateChangeOfBufferingStateForSrvOpen( 7085 cur->fcb->pNetRoot->pSrvCall, srv_open, 7086 srv_open->Key, ULongToPtr(flag)); 7087 } 7088 if (psrvEntry->Flink == &cur->fcb->SrvOpenList) { 7089 #ifdef DEBUG_TIME_BASED_COHERENCY 7090 DbgP("fcbopen_main: reached end of srvopen for fcb %p\n", 7091 cur->fcb); 7092 #endif 7093 break; 7094 } 7095 psrvEntry = psrvEntry->Flink; 7096 }; 7097 } 7098 nfs41_fcb = (PNFS41_FCB)cur->fcb->Context; 7099 nfs41_fcb->changeattr = entry->ChangeTime; 7100 RxFreePool(entry); 7101 out: 7102 if (pEntry->Flink == &openlist.head) { 7103 #ifdef DEBUG_TIME_BASED_COHERENCY 7104 DbgP("fcbopen_main: reached end of the fcb list\n"); 7105 #endif 7106 break; 7107 } 7108 pEntry = pEntry->Flink; 7109 } 7110 ExReleaseFastMutex(&fcblistLock); 7111 } 7112 DbgEx(); 7113 } 7114 7115 #ifdef __REACTOS__ 7116 NTSTATUS NTAPI DriverEntry( 7117 #else 7118 NTSTATUS DriverEntry( 7119 #endif 7120 IN PDRIVER_OBJECT drv, 7121 IN PUNICODE_STRING path) 7122 { 7123 NTSTATUS status; 7124 ULONG flags = 0, i; 7125 UNICODE_STRING dev_name, user_dev_name; 7126 PNFS41_DEVICE_EXTENSION dev_exts; 7127 TIME_FIELDS jan_1_1970 = {1970, 1, 1, 0, 0, 0, 0, 0}; 7128 ACCESS_MASK mask = 0; 7129 OBJECT_ATTRIBUTES oattrs; 7130 7131 DbgEn(); 7132 7133 status = RxDriverEntry(drv, path); 7134 if (status != STATUS_SUCCESS) { 7135 print_error("RxDriverEntry failed: %08lx\n", status); 7136 goto out; 7137 } 7138 7139 RtlInitUnicodeString(&dev_name, NFS41_DEVICE_NAME); 7140 SetFlag(flags, RX_REGISTERMINI_FLAG_DONT_PROVIDE_MAILSLOTS); 7141 7142 status = nfs41_init_ops(); 7143 if (status != STATUS_SUCCESS) { 7144 print_error("nfs41_init_ops failed to initialize dispatch table\n"); 7145 goto out; 7146 } 7147 7148 DbgP("calling RxRegisterMinirdr\n"); 7149 status = RxRegisterMinirdr(&nfs41_dev, drv, &nfs41_ops, flags, &dev_name, 7150 sizeof(NFS41_DEVICE_EXTENSION), 7151 FILE_DEVICE_NETWORK_FILE_SYSTEM, FILE_REMOTE_DEVICE); 7152 if (status != STATUS_SUCCESS) { 7153 print_error("RxRegisterMinirdr failed: %08lx\n", status); 7154 goto out; 7155 } 7156 #ifndef __REACTOS__ 7157 nfs41_dev->Flags |= DO_BUFFERED_IO; 7158 #endif 7159 7160 dev_exts = (PNFS41_DEVICE_EXTENSION) 7161 ((PBYTE)(nfs41_dev) + sizeof(RDBSS_DEVICE_OBJECT)); 7162 7163 RxDefineNode(dev_exts, NFS41_DEVICE_EXTENSION); 7164 dev_exts->DeviceObject = nfs41_dev; 7165 nfs41_create_volume_info((PFILE_FS_VOLUME_INFORMATION)dev_exts->VolAttrs, 7166 &dev_exts->VolAttrsLen); 7167 7168 RtlInitUnicodeString(&user_dev_name, NFS41_SHADOW_DEVICE_NAME); 7169 DbgP("calling IoCreateSymbolicLink %wZ %wZ\n", &user_dev_name, &dev_name); 7170 status = IoCreateSymbolicLink(&user_dev_name, &dev_name); 7171 if (status != STATUS_SUCCESS) { 7172 print_error("Device name IoCreateSymbolicLink failed: %08lx\n", status); 7173 goto out_unregister; 7174 } 7175 7176 KeInitializeEvent(&upcallEvent, SynchronizationEvent, FALSE ); 7177 ExInitializeFastMutex(&upcallLock); 7178 ExInitializeFastMutex(&downcallLock); 7179 ExInitializeFastMutex(&xidLock); 7180 ExInitializeFastMutex(&openOwnerLock); 7181 ExInitializeFastMutex(&fcblistLock); 7182 InitializeListHead(&upcall.head); 7183 InitializeListHead(&downcall.head); 7184 InitializeListHead(&openlist.head); 7185 InitializeObjectAttributes(&oattrs, NULL, OBJ_KERNEL_HANDLE, NULL, NULL); 7186 status = PsCreateSystemThread(&dev_exts->openlistHandle, mask, 7187 &oattrs, NULL, NULL, &fcbopen_main, NULL); 7188 if (status != STATUS_SUCCESS) 7189 goto out_unregister; 7190 7191 drv->DriverUnload = nfs41_driver_unload; 7192 7193 for (i = 0; i <= IRP_MJ_MAXIMUM_FUNCTION; i++) 7194 drv->MajorFunction[i] = (PDRIVER_DISPATCH)nfs41_FsdDispatch; 7195 7196 RtlTimeFieldsToTime(&jan_1_1970, &unix_time_diff); 7197 7198 out_unregister: 7199 if (status != STATUS_SUCCESS) 7200 RxUnregisterMinirdr(nfs41_dev); 7201 out: 7202 DbgEx(); 7203 return status; 7204 } 7205 7206 #ifdef __REACTOS__ 7207 VOID NTAPI nfs41_driver_unload(IN PDRIVER_OBJECT drv) 7208 #else 7209 VOID nfs41_driver_unload(IN PDRIVER_OBJECT drv) 7210 #endif 7211 { 7212 PRX_CONTEXT RxContext; 7213 NTSTATUS status; 7214 UNICODE_STRING dev_name, pipe_name; 7215 7216 DbgEn(); 7217 7218 RxContext = RxCreateRxContext(NULL, nfs41_dev, RX_CONTEXT_FLAG_IN_FSP); 7219 if (RxContext == NULL) { 7220 status = STATUS_INSUFFICIENT_RESOURCES; 7221 goto unload; 7222 } 7223 status = RxStopMinirdr(RxContext, &RxContext->PostRequest); 7224 RxDereferenceAndDeleteRxContext(RxContext); 7225 7226 unload: 7227 RtlInitUnicodeString(&dev_name, NFS41_SHADOW_DEVICE_NAME); 7228 status = IoDeleteSymbolicLink(&dev_name); 7229 if (status != STATUS_SUCCESS) { 7230 print_error("couldn't delete device symbolic link\n"); 7231 } 7232 RtlInitUnicodeString(&pipe_name, NFS41_SHADOW_PIPE_NAME); 7233 status = IoDeleteSymbolicLink(&pipe_name); 7234 if (status != STATUS_SUCCESS) { 7235 print_error("couldn't delete pipe symbolic link\n"); 7236 } 7237 RxUnload(drv); 7238 7239 DbgP("driver unloaded %p\n", drv); 7240 DbgR(); 7241 } 7242