1 /* 2 * ReactOS kernel 3 * Copyright (C) 2002, 2014 ReactOS Team 4 * 5 * This program is free software; you can redistribute it and/or modify 6 * it under the terms of the GNU General Public License as published by 7 * the Free Software Foundation; either version 2 of the License, or 8 * (at your option) any later version. 9 * 10 * This program is distributed in the hope that it will be useful, 11 * but WITHOUT ANY WARRANTY; without even the implied warranty of 12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 13 * GNU General Public License for more details. 14 * 15 * You should have received a copy of the GNU General Public License 16 * along with this program; if not, write to the Free Software 17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA. 18 * 19 * COPYRIGHT: See COPYING in the top level directory 20 * PROJECT: ReactOS kernel 21 * FILE: drivers/filesystem/ntfs/rw.c 22 * PURPOSE: NTFS filesystem driver 23 * PROGRAMMERS: Art Yerkes 24 * Pierre Schweitzer (pierre@reactos.org) 25 * Trevor Thompson 26 */ 27 28 /* INCLUDES *****************************************************************/ 29 30 #include <ntddk.h> 31 #include "ntfs.h" 32 33 #define NDEBUG 34 #include <debug.h> 35 36 /* FUNCTIONS ****************************************************************/ 37 38 /* 39 * FUNCTION: Reads data from a file 40 */ 41 static 42 NTSTATUS 43 NtfsReadFile(PDEVICE_EXTENSION DeviceExt, 44 PFILE_OBJECT FileObject, 45 PUCHAR Buffer, 46 ULONG Length, 47 ULONG ReadOffset, 48 ULONG IrpFlags, 49 PULONG LengthRead) 50 { 51 NTSTATUS Status = STATUS_SUCCESS; 52 PNTFS_FCB Fcb; 53 PFILE_RECORD_HEADER FileRecord; 54 PNTFS_ATTR_CONTEXT DataContext; 55 ULONG RealLength; 56 ULONG RealReadOffset; 57 ULONG RealLengthRead; 58 ULONG ToRead; 59 BOOLEAN AllocatedBuffer = FALSE; 60 PCHAR ReadBuffer = (PCHAR)Buffer; 61 ULONGLONG StreamSize; 62 63 DPRINT1("NtfsReadFile(%p, %p, %p, %lu, %lu, %lx, %p)\n", DeviceExt, FileObject, Buffer, Length, ReadOffset, IrpFlags, LengthRead); 64 65 *LengthRead = 0; 66 67 if (Length == 0) 68 { 69 DPRINT1("Null read!\n"); 70 return STATUS_SUCCESS; 71 } 72 73 Fcb = (PNTFS_FCB)FileObject->FsContext; 74 75 if (NtfsFCBIsCompressed(Fcb)) 76 { 77 DPRINT1("Compressed file!\n"); 78 UNIMPLEMENTED; 79 return STATUS_NOT_IMPLEMENTED; 80 } 81 82 FileRecord = ExAllocateFromNPagedLookasideList(&DeviceExt->FileRecLookasideList); 83 if (FileRecord == NULL) 84 { 85 DPRINT1("Not enough memory!\n"); 86 return STATUS_INSUFFICIENT_RESOURCES; 87 } 88 89 Status = ReadFileRecord(DeviceExt, Fcb->MFTIndex, FileRecord); 90 if (!NT_SUCCESS(Status)) 91 { 92 DPRINT1("Can't find record!\n"); 93 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 94 return Status; 95 } 96 97 98 Status = FindAttribute(DeviceExt, FileRecord, AttributeData, Fcb->Stream, wcslen(Fcb->Stream), &DataContext, NULL); 99 if (!NT_SUCCESS(Status)) 100 { 101 NTSTATUS BrowseStatus; 102 FIND_ATTR_CONTXT Context; 103 PNTFS_ATTR_RECORD Attribute; 104 105 DPRINT1("No '%S' data stream associated with file!\n", Fcb->Stream); 106 107 BrowseStatus = FindFirstAttribute(&Context, DeviceExt, FileRecord, FALSE, &Attribute); 108 while (NT_SUCCESS(BrowseStatus)) 109 { 110 if (Attribute->Type == AttributeData) 111 { 112 UNICODE_STRING Name; 113 114 Name.Length = Attribute->NameLength * sizeof(WCHAR); 115 Name.MaximumLength = Name.Length; 116 Name.Buffer = (PWCHAR)((ULONG_PTR)Attribute + Attribute->NameOffset); 117 DPRINT1("Data stream: '%wZ' available\n", &Name); 118 } 119 120 BrowseStatus = FindNextAttribute(&Context, &Attribute); 121 } 122 FindCloseAttribute(&Context); 123 124 ReleaseAttributeContext(DataContext); 125 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 126 return Status; 127 } 128 129 StreamSize = AttributeDataLength(DataContext->pRecord); 130 if (ReadOffset >= StreamSize) 131 { 132 DPRINT1("Reading beyond stream end!\n"); 133 ReleaseAttributeContext(DataContext); 134 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 135 return STATUS_END_OF_FILE; 136 } 137 138 ToRead = Length; 139 if (ReadOffset + Length > StreamSize) 140 ToRead = StreamSize - ReadOffset; 141 142 RealReadOffset = ReadOffset; 143 RealLength = ToRead; 144 145 if ((ReadOffset % DeviceExt->NtfsInfo.BytesPerSector) != 0 || (ToRead % DeviceExt->NtfsInfo.BytesPerSector) != 0) 146 { 147 RealReadOffset = ROUND_DOWN(ReadOffset, DeviceExt->NtfsInfo.BytesPerSector); 148 RealLength = ROUND_UP(ToRead, DeviceExt->NtfsInfo.BytesPerSector); 149 /* do we need to extend RealLength by one sector? */ 150 if (RealLength + RealReadOffset < ReadOffset + Length) 151 { 152 if (RealReadOffset + RealLength + DeviceExt->NtfsInfo.BytesPerSector <= AttributeAllocatedLength(DataContext->pRecord)) 153 RealLength += DeviceExt->NtfsInfo.BytesPerSector; 154 } 155 156 157 ReadBuffer = ExAllocatePoolWithTag(NonPagedPool, RealLength, TAG_NTFS); 158 if (ReadBuffer == NULL) 159 { 160 DPRINT1("Not enough memory!\n"); 161 ReleaseAttributeContext(DataContext); 162 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 163 return STATUS_INSUFFICIENT_RESOURCES; 164 } 165 AllocatedBuffer = TRUE; 166 } 167 168 DPRINT("Effective read: %lu at %lu for stream '%S'\n", RealLength, RealReadOffset, Fcb->Stream); 169 RealLengthRead = ReadAttribute(DeviceExt, DataContext, RealReadOffset, (PCHAR)ReadBuffer, RealLength); 170 if (RealLengthRead == 0) 171 { 172 DPRINT1("Read failure!\n"); 173 ReleaseAttributeContext(DataContext); 174 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 175 if (AllocatedBuffer) 176 { 177 ExFreePoolWithTag(ReadBuffer, TAG_NTFS); 178 } 179 return Status; 180 } 181 182 ReleaseAttributeContext(DataContext); 183 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 184 185 *LengthRead = ToRead; 186 187 DPRINT("%lu got read\n", *LengthRead); 188 189 if (AllocatedBuffer) 190 { 191 RtlCopyMemory(Buffer, ReadBuffer + (ReadOffset - RealReadOffset), ToRead); 192 } 193 194 if (ToRead != Length) 195 { 196 RtlZeroMemory(Buffer + ToRead, Length - ToRead); 197 } 198 199 if (AllocatedBuffer) 200 { 201 ExFreePoolWithTag(ReadBuffer, TAG_NTFS); 202 } 203 204 return STATUS_SUCCESS; 205 } 206 207 208 NTSTATUS 209 NtfsRead(PNTFS_IRP_CONTEXT IrpContext) 210 { 211 PDEVICE_EXTENSION DeviceExt; 212 PIO_STACK_LOCATION Stack; 213 PFILE_OBJECT FileObject; 214 PVOID Buffer; 215 ULONG ReadLength; 216 LARGE_INTEGER ReadOffset; 217 ULONG ReturnedReadLength = 0; 218 NTSTATUS Status = STATUS_SUCCESS; 219 PIRP Irp; 220 PDEVICE_OBJECT DeviceObject; 221 222 DPRINT("NtfsRead(IrpContext %p)\n", IrpContext); 223 224 DeviceObject = IrpContext->DeviceObject; 225 Irp = IrpContext->Irp; 226 Stack = IrpContext->Stack; 227 FileObject = IrpContext->FileObject; 228 229 DeviceExt = DeviceObject->DeviceExtension; 230 ReadLength = Stack->Parameters.Read.Length; 231 ReadOffset = Stack->Parameters.Read.ByteOffset; 232 Buffer = NtfsGetUserBuffer(Irp, BooleanFlagOn(Irp->Flags, IRP_PAGING_IO)); 233 234 Status = NtfsReadFile(DeviceExt, 235 FileObject, 236 Buffer, 237 ReadLength, 238 ReadOffset.u.LowPart, 239 Irp->Flags, 240 &ReturnedReadLength); 241 if (NT_SUCCESS(Status)) 242 { 243 if (FileObject->Flags & FO_SYNCHRONOUS_IO) 244 { 245 FileObject->CurrentByteOffset.QuadPart = 246 ReadOffset.QuadPart + ReturnedReadLength; 247 } 248 249 Irp->IoStatus.Information = ReturnedReadLength; 250 } 251 else 252 { 253 Irp->IoStatus.Information = 0; 254 } 255 256 return Status; 257 } 258 259 /** 260 * @name NtfsWriteFile 261 * @implemented 262 * 263 * Writes a file to the disk. It presently borrows a lot of code from NtfsReadFile() and 264 * VFatWriteFileData(). It needs some more work before it will be complete; it won't handle 265 * page files, asnyc io, cached writes, etc. 266 * 267 * @param DeviceExt 268 * Points to the target disk's DEVICE_EXTENSION 269 * 270 * @param FileObject 271 * Pointer to a FILE_OBJECT describing the target file 272 * 273 * @param Buffer 274 * The data that's being written to the file 275 * 276 * @Param Length 277 * The size of the data buffer being written, in bytes 278 * 279 * @param WriteOffset 280 * Offset, in bytes, from the beginning of the file. Indicates where to start 281 * writing data. 282 * 283 * @param IrpFlags 284 * TODO: flags are presently ignored in code. 285 * 286 * @param CaseSensitive 287 * Boolean indicating if the function should operate in case-sensitive mode. This will be TRUE 288 * if an application opened the file with the FILE_FLAG_POSIX_SEMANTICS flag. 289 * 290 * @param LengthWritten 291 * Pointer to a ULONG. This ULONG will be set to the number of bytes successfully written. 292 * 293 * @return 294 * STATUS_SUCCESS if successful, STATUS_NOT_IMPLEMENTED if a required feature isn't implemented, 295 * STATUS_INSUFFICIENT_RESOURCES if an allocation failed, STATUS_ACCESS_DENIED if the write itself fails, 296 * STATUS_PARTIAL_COPY or STATUS_UNSUCCESSFUL if ReadFileRecord() fails, or 297 * STATUS_OBJECT_NAME_NOT_FOUND if the file's data stream could not be found. 298 * 299 * @remarks Called by NtfsWrite(). It may perform a read-modify-write operation if the requested write is 300 * not sector-aligned. LengthWritten only refers to how much of the requested data has been written; 301 * extra data that needs to be written to make the write sector-aligned will not affect it. 302 * 303 */ 304 NTSTATUS NtfsWriteFile(PDEVICE_EXTENSION DeviceExt, 305 PFILE_OBJECT FileObject, 306 const PUCHAR Buffer, 307 ULONG Length, 308 ULONG WriteOffset, 309 ULONG IrpFlags, 310 BOOLEAN CaseSensitive, 311 PULONG LengthWritten) 312 { 313 NTSTATUS Status = STATUS_NOT_IMPLEMENTED; 314 PNTFS_FCB Fcb; 315 PFILE_RECORD_HEADER FileRecord; 316 PNTFS_ATTR_CONTEXT DataContext; 317 ULONG AttributeOffset; 318 ULONGLONG StreamSize; 319 320 DPRINT("NtfsWriteFile(%p, %p, %p, %lu, %lu, %x, %s, %p)\n", 321 DeviceExt, 322 FileObject, 323 Buffer, 324 Length, 325 WriteOffset, 326 IrpFlags, 327 (CaseSensitive ? "TRUE" : "FALSE"), 328 LengthWritten); 329 330 *LengthWritten = 0; 331 332 ASSERT(DeviceExt); 333 334 if (Length == 0) 335 { 336 if (Buffer == NULL) 337 return STATUS_SUCCESS; 338 else 339 return STATUS_INVALID_PARAMETER; 340 } 341 342 // get the File control block 343 Fcb = (PNTFS_FCB)FileObject->FsContext; 344 ASSERT(Fcb); 345 346 DPRINT("Fcb->PathName: %wS\n", Fcb->PathName); 347 DPRINT("Fcb->ObjectName: %wS\n", Fcb->ObjectName); 348 349 // we don't yet handle compression 350 if (NtfsFCBIsCompressed(Fcb)) 351 { 352 DPRINT("Compressed file!\n"); 353 UNIMPLEMENTED; 354 return STATUS_NOT_IMPLEMENTED; 355 } 356 357 // allocate non-paged memory for the FILE_RECORD_HEADER 358 FileRecord = ExAllocateFromNPagedLookasideList(&DeviceExt->FileRecLookasideList); 359 if (FileRecord == NULL) 360 { 361 DPRINT1("Not enough memory! Can't write %wS!\n", Fcb->PathName); 362 return STATUS_INSUFFICIENT_RESOURCES; 363 } 364 365 // read the FILE_RECORD_HEADER from the drive (or cache) 366 DPRINT("Reading file record...\n"); 367 Status = ReadFileRecord(DeviceExt, Fcb->MFTIndex, FileRecord); 368 if (!NT_SUCCESS(Status)) 369 { 370 // We couldn't get the file's record. Free the memory and return the error 371 DPRINT1("Can't find record for %wS!\n", Fcb->ObjectName); 372 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 373 return Status; 374 } 375 376 DPRINT("Found record for %wS\n", Fcb->ObjectName); 377 378 // Find the attribute with the data stream for our file 379 DPRINT("Finding Data Attribute...\n"); 380 Status = FindAttribute(DeviceExt, FileRecord, AttributeData, Fcb->Stream, wcslen(Fcb->Stream), &DataContext, 381 &AttributeOffset); 382 383 // Did we fail to find the attribute? 384 if (!NT_SUCCESS(Status)) 385 { 386 NTSTATUS BrowseStatus; 387 FIND_ATTR_CONTXT Context; 388 PNTFS_ATTR_RECORD Attribute; 389 390 DPRINT1("No '%S' data stream associated with file!\n", Fcb->Stream); 391 392 // Couldn't find the requested data stream; print a list of streams available 393 BrowseStatus = FindFirstAttribute(&Context, DeviceExt, FileRecord, FALSE, &Attribute); 394 while (NT_SUCCESS(BrowseStatus)) 395 { 396 if (Attribute->Type == AttributeData) 397 { 398 UNICODE_STRING Name; 399 400 Name.Length = Attribute->NameLength * sizeof(WCHAR); 401 Name.MaximumLength = Name.Length; 402 Name.Buffer = (PWCHAR)((ULONG_PTR)Attribute + Attribute->NameOffset); 403 DPRINT1("Data stream: '%wZ' available\n", &Name); 404 } 405 406 BrowseStatus = FindNextAttribute(&Context, &Attribute); 407 } 408 FindCloseAttribute(&Context); 409 410 ReleaseAttributeContext(DataContext); 411 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 412 return Status; 413 } 414 415 // Get the size of the stream on disk 416 StreamSize = AttributeDataLength(DataContext->pRecord); 417 418 DPRINT("WriteOffset: %lu\tStreamSize: %I64u\n", WriteOffset, StreamSize); 419 420 // Are we trying to write beyond the end of the stream? 421 if (WriteOffset + Length > StreamSize) 422 { 423 // is increasing the stream size allowed? 424 if (!(Fcb->Flags & FCB_IS_VOLUME) && 425 !(IrpFlags & IRP_PAGING_IO)) 426 { 427 LARGE_INTEGER DataSize; 428 ULONGLONG AllocationSize; 429 PFILENAME_ATTRIBUTE fileNameAttribute; 430 ULONGLONG ParentMFTId; 431 UNICODE_STRING filename; 432 433 DataSize.QuadPart = WriteOffset + Length; 434 435 // set the attribute data length 436 Status = SetAttributeDataLength(FileObject, Fcb, DataContext, AttributeOffset, FileRecord, &DataSize); 437 if (!NT_SUCCESS(Status)) 438 { 439 ReleaseAttributeContext(DataContext); 440 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 441 *LengthWritten = 0; 442 return Status; 443 } 444 445 AllocationSize = AttributeAllocatedLength(DataContext->pRecord); 446 447 // now we need to update this file's size in every directory index entry that references it 448 // TODO: put this code in its own function and adapt it to work with every filename / hardlink 449 // stored in the file record. 450 fileNameAttribute = GetBestFileNameFromRecord(Fcb->Vcb, FileRecord); 451 ASSERT(fileNameAttribute); 452 453 ParentMFTId = fileNameAttribute->DirectoryFileReferenceNumber & NTFS_MFT_MASK; 454 455 filename.Buffer = fileNameAttribute->Name; 456 filename.Length = fileNameAttribute->NameLength * sizeof(WCHAR); 457 filename.MaximumLength = filename.Length; 458 459 Status = UpdateFileNameRecord(Fcb->Vcb, 460 ParentMFTId, 461 &filename, 462 FALSE, 463 DataSize.QuadPart, 464 AllocationSize, 465 CaseSensitive); 466 467 } 468 else 469 { 470 // TODO - just fail for now 471 ReleaseAttributeContext(DataContext); 472 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 473 *LengthWritten = 0; 474 return STATUS_ACCESS_DENIED; 475 } 476 } 477 478 DPRINT("Length: %lu\tWriteOffset: %lu\tStreamSize: %I64u\n", Length, WriteOffset, StreamSize); 479 480 // Write the data to the attribute 481 Status = WriteAttribute(DeviceExt, DataContext, WriteOffset, Buffer, Length, LengthWritten, FileRecord); 482 483 // Did the write fail? 484 if (!NT_SUCCESS(Status)) 485 { 486 DPRINT1("Write failure!\n"); 487 ReleaseAttributeContext(DataContext); 488 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 489 490 return Status; 491 } 492 493 // This should never happen: 494 if (*LengthWritten != Length) 495 { 496 DPRINT1("\a\tNTFS DRIVER ERROR: length written (%lu) differs from requested (%lu), but no error was indicated!\n", 497 *LengthWritten, Length); 498 Status = STATUS_UNEXPECTED_IO_ERROR; 499 } 500 501 ReleaseAttributeContext(DataContext); 502 ExFreeToNPagedLookasideList(&DeviceExt->FileRecLookasideList, FileRecord); 503 504 return Status; 505 } 506 507 /** 508 * @name NtfsWrite 509 * @implemented 510 * 511 * Handles IRP_MJ_WRITE I/O Request Packets for NTFS. This code borrows a lot from 512 * VfatWrite, and needs a lot of cleaning up. It also needs a lot more of the code 513 * from VfatWrite integrated. 514 * 515 * @param IrpContext 516 * Points to an NTFS_IRP_CONTEXT which describes the write 517 * 518 * @return 519 * STATUS_SUCCESS if successful, 520 * STATUS_INSUFFICIENT_RESOURCES if an allocation failed, 521 * STATUS_INVALID_DEVICE_REQUEST if called on the main device object, 522 * STATUS_NOT_IMPLEMENTED or STATUS_ACCESS_DENIED if a required feature isn't implemented. 523 * STATUS_PARTIAL_COPY, STATUS_UNSUCCESSFUL, or STATUS_OBJECT_NAME_NOT_FOUND if NtfsWriteFile() fails. 524 * 525 * @remarks Called by NtfsDispatch() in response to an IRP_MJ_WRITE request. Page files are not implemented. 526 * Support for large files (>4gb) is not implemented. Cached writes, file locks, transactions, etc - not implemented. 527 * 528 */ 529 NTSTATUS 530 NtfsWrite(PNTFS_IRP_CONTEXT IrpContext) 531 { 532 PNTFS_FCB Fcb; 533 PERESOURCE Resource = NULL; 534 LARGE_INTEGER ByteOffset; 535 PUCHAR Buffer; 536 NTSTATUS Status = STATUS_SUCCESS; 537 ULONG Length = 0; 538 ULONG ReturnedWriteLength = 0; 539 PDEVICE_OBJECT DeviceObject = NULL; 540 PDEVICE_EXTENSION DeviceExt = NULL; 541 PFILE_OBJECT FileObject = NULL; 542 PIRP Irp = NULL; 543 ULONG BytesPerSector; 544 545 DPRINT("NtfsWrite(IrpContext %p)\n", IrpContext); 546 ASSERT(IrpContext); 547 548 // get the I/O request packet 549 Irp = IrpContext->Irp; 550 551 // This request is not allowed on the main device object 552 if (IrpContext->DeviceObject == NtfsGlobalData->DeviceObject) 553 { 554 DPRINT1("\t\t\t\tNtfsWrite is called with the main device object.\n"); 555 556 Irp->IoStatus.Information = 0; 557 return STATUS_INVALID_DEVICE_REQUEST; 558 } 559 560 // get the File control block 561 Fcb = (PNTFS_FCB)IrpContext->FileObject->FsContext; 562 ASSERT(Fcb); 563 564 DPRINT("About to write %wS\n", Fcb->ObjectName); 565 DPRINT("NTFS Version: %d.%d\n", Fcb->Vcb->NtfsInfo.MajorVersion, Fcb->Vcb->NtfsInfo.MinorVersion); 566 567 // setup some more locals 568 FileObject = IrpContext->FileObject; 569 DeviceObject = IrpContext->DeviceObject; 570 DeviceExt = DeviceObject->DeviceExtension; 571 BytesPerSector = DeviceExt->StorageDevice->SectorSize; 572 Length = IrpContext->Stack->Parameters.Write.Length; 573 574 // get the file offset we'll be writing to 575 ByteOffset = IrpContext->Stack->Parameters.Write.ByteOffset; 576 if (ByteOffset.u.LowPart == FILE_WRITE_TO_END_OF_FILE && 577 ByteOffset.u.HighPart == -1) 578 { 579 ByteOffset.QuadPart = Fcb->RFCB.FileSize.QuadPart; 580 } 581 582 DPRINT("ByteOffset: %I64u\tLength: %lu\tBytes per sector: %lu\n", ByteOffset.QuadPart, 583 Length, BytesPerSector); 584 585 if (ByteOffset.u.HighPart && !(Fcb->Flags & FCB_IS_VOLUME)) 586 { 587 // TODO: Support large files 588 DPRINT1("FIXME: Writing to large files is not yet supported at this time.\n"); 589 return STATUS_INVALID_PARAMETER; 590 } 591 592 // Is this a non-cached write? A non-buffered write? 593 if (IrpContext->Irp->Flags & (IRP_PAGING_IO | IRP_NOCACHE) || (Fcb->Flags & FCB_IS_VOLUME) || 594 IrpContext->FileObject->Flags & FILE_NO_INTERMEDIATE_BUFFERING) 595 { 596 // non-cached and non-buffered writes must be sector aligned 597 if (ByteOffset.u.LowPart % BytesPerSector != 0 || Length % BytesPerSector != 0) 598 { 599 DPRINT1("Non-cached writes and non-buffered writes must be sector aligned!\n"); 600 return STATUS_INVALID_PARAMETER; 601 } 602 } 603 604 if (Length == 0) 605 { 606 DPRINT1("Null write!\n"); 607 608 IrpContext->Irp->IoStatus.Information = 0; 609 610 // FIXME: Doesn't accurately detect when a user passes NULL to WriteFile() for the buffer 611 if (Irp->UserBuffer == NULL && Irp->MdlAddress == NULL) 612 { 613 // FIXME: Update last write time 614 return STATUS_SUCCESS; 615 } 616 617 return STATUS_INVALID_PARAMETER; 618 } 619 620 // get the Resource 621 if (Fcb->Flags & FCB_IS_VOLUME) 622 { 623 Resource = &DeviceExt->DirResource; 624 } 625 else if (IrpContext->Irp->Flags & IRP_PAGING_IO) 626 { 627 Resource = &Fcb->PagingIoResource; 628 } 629 else 630 { 631 Resource = &Fcb->MainResource; 632 } 633 634 // acquire exclusive access to the Resource 635 if (!ExAcquireResourceExclusiveLite(Resource, BooleanFlagOn(IrpContext->Flags, IRPCONTEXT_CANWAIT))) 636 { 637 return STATUS_CANT_WAIT; 638 } 639 640 /* From VfatWrite(). Todo: Handle file locks 641 if (!(IrpContext->Irp->Flags & IRP_PAGING_IO) && 642 FsRtlAreThereCurrentFileLocks(&Fcb->FileLock)) 643 { 644 if (!FsRtlCheckLockForWriteAccess(&Fcb->FileLock, IrpContext->Irp)) 645 { 646 Status = STATUS_FILE_LOCK_CONFLICT; 647 goto ByeBye; 648 } 649 }*/ 650 651 // Is this an async request to a file? 652 if (!(IrpContext->Flags & IRPCONTEXT_CANWAIT) && !(Fcb->Flags & FCB_IS_VOLUME)) 653 { 654 DPRINT1("FIXME: Async writes not supported in NTFS!\n"); 655 656 ExReleaseResourceLite(Resource); 657 return STATUS_NOT_IMPLEMENTED; 658 } 659 660 // get the buffer of data the user is trying to write 661 Buffer = NtfsGetUserBuffer(Irp, BooleanFlagOn(Irp->Flags, IRP_PAGING_IO)); 662 ASSERT(Buffer); 663 664 // lock the buffer 665 Status = NtfsLockUserBuffer(Irp, Length, IoReadAccess); 666 667 // were we unable to lock the buffer? 668 if (!NT_SUCCESS(Status)) 669 { 670 DPRINT1("Unable to lock user buffer!\n"); 671 672 ExReleaseResourceLite(Resource); 673 return Status; 674 } 675 676 DPRINT("Existing File Size(Fcb->RFCB.FileSize.QuadPart): %I64u\n", Fcb->RFCB.FileSize.QuadPart); 677 DPRINT("About to write the data. Length: %lu\n", Length); 678 679 // TODO: handle HighPart of ByteOffset (large files) 680 681 // write the file 682 Status = NtfsWriteFile(DeviceExt, 683 FileObject, 684 Buffer, 685 Length, 686 ByteOffset.LowPart, 687 Irp->Flags, 688 BooleanFlagOn(IrpContext->Stack->Flags, SL_CASE_SENSITIVE), 689 &ReturnedWriteLength); 690 691 IrpContext->Irp->IoStatus.Status = Status; 692 693 // was the write successful? 694 if (NT_SUCCESS(Status)) 695 { 696 // TODO: Update timestamps 697 698 if (FileObject->Flags & FO_SYNCHRONOUS_IO) 699 { 700 // advance the file pointer 701 FileObject->CurrentByteOffset.QuadPart = ByteOffset.QuadPart + ReturnedWriteLength; 702 } 703 704 IrpContext->PriorityBoost = IO_DISK_INCREMENT; 705 } 706 else 707 { 708 DPRINT1("Write not Succesful!\tReturned length: %lu\n", ReturnedWriteLength); 709 } 710 711 Irp->IoStatus.Information = ReturnedWriteLength; 712 713 // Note: We leave the user buffer that we locked alone, it's up to the I/O manager to unlock and free it 714 715 ExReleaseResourceLite(Resource); 716 717 return Status; 718 } 719 720 /* EOF */ 721