2008-03-09 14:11:42 +00:00
|
|
|
/*
|
2001-10-10 21:46:13 +00:00
|
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
|
|
* PROJECT: ReactOS kernel
|
|
|
|
* FILE: ntoskrnl/cc/copy.c
|
|
|
|
* PURPOSE: Implements cache managers copy interface
|
2005-01-26 13:58:37 +00:00
|
|
|
*
|
2007-10-19 23:21:45 +00:00
|
|
|
* PROGRAMMERS:
|
2001-10-10 21:46:13 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* INCLUDES ******************************************************************/
|
|
|
|
|
2004-08-15 16:39:12 +00:00
|
|
|
#include <ntoskrnl.h>
|
2001-10-10 21:46:13 +00:00
|
|
|
#define NDEBUG
|
2008-08-30 16:31:06 +00:00
|
|
|
#include <debug.h>
|
2001-10-10 21:46:13 +00:00
|
|
|
|
|
|
|
/* GLOBALS *******************************************************************/
|
|
|
|
|
2010-07-15 22:50:12 +00:00
|
|
|
static PFN_NUMBER CcZeroPage = 0;
|
2002-06-10 21:11:56 +00:00
|
|
|
|
2004-05-22 18:28:18 +00:00
|
|
|
#define MAX_ZERO_LENGTH (256 * 1024)
|
2004-08-25 15:08:29 +00:00
|
|
|
#define MAX_RW_LENGTH (256 * 1024)
|
2004-06-06 08:36:31 +00:00
|
|
|
|
2005-11-22 02:30:18 +00:00
|
|
|
ULONG CcFastMdlReadWait;
|
2007-01-21 04:46:38 +00:00
|
|
|
ULONG CcFastMdlReadNotPossible;
|
2005-11-22 02:30:18 +00:00
|
|
|
ULONG CcFastReadNotPossible;
|
|
|
|
ULONG CcFastReadWait;
|
Implemented:
ExSetResourceOwnerPointer, FsRtlDissectDbcs, FsRtlDoesDbcsContainWildCards, FsRtlAcquireFileExclusive, FsRtlReleaseFile, FsRtlGetNextMcbEntry, FsRtlLookupLastMcbEntry, FsRtlLookupMcbEntry, FsRtlRemoveMcbEntry, FsRtlIncrementCcFastReadResourceMiss, FsRtlIncrementCcFastReadNotPossible, FsRtlIncrementCcFastReadWait, FsRtlIncrementCcFastReadNoWait , FsRtlAreNamesEqual, FsRtlDoesNameContainWildCards, IoCreateDriver, IoDeleteDriver
Thanks to Filip for reviewing some of these.
svn path=/trunk/; revision=12408
2004-12-30 18:30:05 +00:00
|
|
|
ULONG CcFastReadNoWait;
|
|
|
|
ULONG CcFastReadResourceMiss;
|
|
|
|
|
2001-10-10 21:46:13 +00:00
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
|
2005-05-09 01:38:29 +00:00
|
|
|
VOID
|
2005-09-14 01:44:19 +00:00
|
|
|
NTAPI
|
2002-08-08 17:54:16 +00:00
|
|
|
CcInitCacheZeroPage(VOID)
|
2002-06-10 21:11:56 +00:00
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
|
2004-08-01 07:24:59 +00:00
|
|
|
Status = MmRequestPageMemoryConsumer(MC_NPPOOL, TRUE, &CcZeroPage);
|
2002-06-10 21:11:56 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
DbgPrint("Can't allocate CcZeroPage.\n");
|
2008-08-24 15:48:05 +00:00
|
|
|
KeBugCheck(CACHE_MANAGER);
|
2002-06-10 21:11:56 +00:00
|
|
|
}
|
|
|
|
Status = MiZeroPage(CcZeroPage);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
DbgPrint("Can't zero out CcZeroPage.\n");
|
2008-08-24 15:48:05 +00:00
|
|
|
KeBugCheck(CACHE_MANAGER);
|
2002-06-10 21:11:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2002-01-26 21:21:02 +00:00
|
|
|
NTSTATUS
|
2005-09-14 01:44:19 +00:00
|
|
|
NTAPI
|
2002-09-08 10:23:54 +00:00
|
|
|
ReadCacheSegmentChain(PBCB Bcb, ULONG ReadOffset, ULONG Length,
|
2002-01-26 21:21:02 +00:00
|
|
|
PVOID Buffer)
|
|
|
|
{
|
2002-09-08 10:23:54 +00:00
|
|
|
PCACHE_SEGMENT head;
|
|
|
|
PCACHE_SEGMENT current;
|
|
|
|
PCACHE_SEGMENT previous;
|
2002-01-26 21:21:02 +00:00
|
|
|
IO_STATUS_BLOCK Iosb;
|
|
|
|
LARGE_INTEGER SegOffset;
|
|
|
|
NTSTATUS Status;
|
|
|
|
ULONG TempLength;
|
2002-08-28 07:13:04 +00:00
|
|
|
KEVENT Event;
|
2004-06-06 08:36:31 +00:00
|
|
|
PMDL Mdl;
|
|
|
|
|
2009-08-23 15:49:43 +00:00
|
|
|
Mdl = _alloca(MmSizeOfMdl(NULL, MAX_RW_LENGTH));
|
2002-01-26 21:21:02 +00:00
|
|
|
|
|
|
|
Status = CcRosGetCacheSegmentChain(Bcb, ReadOffset, Length, &head);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
return(Status);
|
|
|
|
}
|
|
|
|
current = head;
|
|
|
|
while (current != NULL)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* If the current segment is valid then copy it into the
|
|
|
|
* user buffer.
|
|
|
|
*/
|
|
|
|
if (current->Valid)
|
|
|
|
{
|
|
|
|
TempLength = min(Bcb->CacheSegmentSize, Length);
|
|
|
|
memcpy(Buffer, current->BaseAddress, TempLength);
|
2005-07-05 00:57:47 +00:00
|
|
|
|
|
|
|
Buffer = (PVOID)((ULONG_PTR)Buffer + TempLength);
|
|
|
|
|
2005-05-09 01:38:29 +00:00
|
|
|
Length = Length - TempLength;
|
2002-01-26 21:21:02 +00:00
|
|
|
previous = current;
|
|
|
|
current = current->NextInChain;
|
|
|
|
CcRosReleaseCacheSegment(Bcb, previous, TRUE, FALSE, FALSE);
|
|
|
|
}
|
|
|
|
/*
|
2002-07-17 21:04:57 +00:00
|
|
|
* Otherwise read in as much as we can.
|
2002-01-26 21:21:02 +00:00
|
|
|
*/
|
|
|
|
else
|
|
|
|
{
|
2002-09-08 10:23:54 +00:00
|
|
|
PCACHE_SEGMENT current2;
|
2002-01-26 21:21:02 +00:00
|
|
|
ULONG current_size;
|
|
|
|
ULONG i;
|
2010-07-15 22:50:12 +00:00
|
|
|
PPFN_NUMBER MdlPages;
|
2002-01-26 21:21:02 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Count the maximum number of bytes we could read starting
|
|
|
|
* from the current segment.
|
|
|
|
*/
|
|
|
|
current2 = current;
|
|
|
|
current_size = 0;
|
2004-08-25 15:08:29 +00:00
|
|
|
while (current2 != NULL && !current2->Valid && current_size < MAX_RW_LENGTH)
|
2002-01-26 21:21:02 +00:00
|
|
|
{
|
|
|
|
current2 = current2->NextInChain;
|
|
|
|
current_size += Bcb->CacheSegmentSize;
|
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2002-01-26 21:21:02 +00:00
|
|
|
/*
|
|
|
|
* Create an MDL which contains all their pages.
|
|
|
|
*/
|
2004-06-20 02:10:07 +00:00
|
|
|
MmInitializeMdl(Mdl, NULL, current_size);
|
2002-01-26 21:21:02 +00:00
|
|
|
Mdl->MdlFlags |= (MDL_PAGES_LOCKED | MDL_IO_PAGE_READ);
|
|
|
|
current2 = current;
|
2004-08-25 15:08:29 +00:00
|
|
|
current_size = 0;
|
2010-07-15 22:50:12 +00:00
|
|
|
MdlPages = (PPFN_NUMBER)(Mdl + 1);
|
2004-08-25 15:08:29 +00:00
|
|
|
while (current2 != NULL && !current2->Valid && current_size < MAX_RW_LENGTH)
|
2002-01-26 21:21:02 +00:00
|
|
|
{
|
2004-08-01 07:24:59 +00:00
|
|
|
PVOID address = current2->BaseAddress;
|
2005-06-21 23:42:58 +00:00
|
|
|
for (i = 0; i < (Bcb->CacheSegmentSize / PAGE_SIZE); i++, address = RVA(address, PAGE_SIZE))
|
2002-01-26 21:21:02 +00:00
|
|
|
{
|
2004-08-01 07:24:59 +00:00
|
|
|
*MdlPages++ = MmGetPfnForProcess(NULL, address);
|
2002-01-26 21:21:02 +00:00
|
|
|
}
|
|
|
|
current2 = current2->NextInChain;
|
2004-08-25 15:08:29 +00:00
|
|
|
current_size += Bcb->CacheSegmentSize;
|
2002-01-26 21:21:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Read in the information.
|
|
|
|
*/
|
|
|
|
SegOffset.QuadPart = current->FileOffset;
|
2002-08-28 07:13:04 +00:00
|
|
|
KeInitializeEvent(&Event, NotificationEvent, FALSE);
|
2002-01-26 21:21:02 +00:00
|
|
|
Status = IoPageRead(Bcb->FileObject,
|
|
|
|
Mdl,
|
|
|
|
&SegOffset,
|
2002-08-28 07:13:04 +00:00
|
|
|
&Event,
|
|
|
|
&Iosb);
|
|
|
|
if (Status == STATUS_PENDING)
|
|
|
|
{
|
|
|
|
KeWaitForSingleObject(&Event, Executive, KernelMode, FALSE, NULL);
|
|
|
|
Status = Iosb.Status;
|
|
|
|
}
|
2004-08-25 15:08:29 +00:00
|
|
|
if (Mdl->MdlFlags & MDL_MAPPED_TO_SYSTEM_VA)
|
|
|
|
{
|
|
|
|
MmUnmapLockedPages(Mdl->MappedSystemVa, Mdl);
|
|
|
|
}
|
2002-01-26 21:21:02 +00:00
|
|
|
if (!NT_SUCCESS(Status) && Status != STATUS_END_OF_FILE)
|
|
|
|
{
|
|
|
|
while (current != NULL)
|
|
|
|
{
|
|
|
|
previous = current;
|
|
|
|
current = current->NextInChain;
|
|
|
|
CcRosReleaseCacheSegment(Bcb, previous, FALSE, FALSE, FALSE);
|
|
|
|
}
|
|
|
|
return(Status);
|
|
|
|
}
|
2004-08-25 15:08:29 +00:00
|
|
|
current_size = 0;
|
|
|
|
while (current != NULL && !current->Valid && current_size < MAX_RW_LENGTH)
|
2002-01-26 21:21:02 +00:00
|
|
|
{
|
|
|
|
previous = current;
|
|
|
|
current = current->NextInChain;
|
|
|
|
TempLength = min(Bcb->CacheSegmentSize, Length);
|
|
|
|
memcpy(Buffer, previous->BaseAddress, TempLength);
|
2005-07-05 00:57:47 +00:00
|
|
|
|
|
|
|
Buffer = (PVOID)((ULONG_PTR)Buffer + TempLength);
|
|
|
|
|
2005-05-09 01:38:29 +00:00
|
|
|
Length = Length - TempLength;
|
2002-01-26 21:21:02 +00:00
|
|
|
CcRosReleaseCacheSegment(Bcb, previous, TRUE, FALSE, FALSE);
|
2004-08-25 15:08:29 +00:00
|
|
|
current_size += Bcb->CacheSegmentSize;
|
2002-01-26 21:21:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return(STATUS_SUCCESS);
|
|
|
|
}
|
|
|
|
|
2005-05-09 01:38:29 +00:00
|
|
|
NTSTATUS
|
2005-09-14 01:44:19 +00:00
|
|
|
NTAPI
|
2002-09-08 10:23:54 +00:00
|
|
|
ReadCacheSegment(PCACHE_SEGMENT CacheSeg)
|
2001-10-10 21:46:13 +00:00
|
|
|
{
|
|
|
|
ULONG Size;
|
|
|
|
PMDL Mdl;
|
|
|
|
NTSTATUS Status;
|
|
|
|
LARGE_INTEGER SegOffset;
|
|
|
|
IO_STATUS_BLOCK IoStatus;
|
2002-08-28 07:13:04 +00:00
|
|
|
KEVENT Event;
|
2001-10-10 21:46:13 +00:00
|
|
|
|
|
|
|
SegOffset.QuadPart = CacheSeg->FileOffset;
|
2003-12-30 18:52:06 +00:00
|
|
|
Size = (ULONG)(CacheSeg->Bcb->AllocationSize.QuadPart - CacheSeg->FileOffset);
|
2001-10-10 21:46:13 +00:00
|
|
|
if (Size > CacheSeg->Bcb->CacheSegmentSize)
|
2002-01-26 21:21:02 +00:00
|
|
|
{
|
|
|
|
Size = CacheSeg->Bcb->CacheSegmentSize;
|
|
|
|
}
|
2009-08-23 15:49:43 +00:00
|
|
|
Mdl = _alloca(MmSizeOfMdl(CacheSeg->BaseAddress, Size));
|
2004-06-20 02:10:07 +00:00
|
|
|
MmInitializeMdl(Mdl, CacheSeg->BaseAddress, Size);
|
2001-10-10 21:46:13 +00:00
|
|
|
MmBuildMdlForNonPagedPool(Mdl);
|
2004-06-06 08:36:31 +00:00
|
|
|
Mdl->MdlFlags |= MDL_IO_PAGE_READ;
|
2002-08-28 07:13:04 +00:00
|
|
|
KeInitializeEvent(&Event, NotificationEvent, FALSE);
|
2005-05-09 01:38:29 +00:00
|
|
|
Status = IoPageRead(CacheSeg->Bcb->FileObject, Mdl, &SegOffset, & Event, &IoStatus);
|
2002-08-28 07:13:04 +00:00
|
|
|
if (Status == STATUS_PENDING)
|
|
|
|
{
|
|
|
|
KeWaitForSingleObject(&Event, Executive, KernelMode, FALSE, NULL);
|
|
|
|
Status = IoStatus.Status;
|
|
|
|
}
|
|
|
|
|
2001-10-10 21:46:13 +00:00
|
|
|
if (!NT_SUCCESS(Status) && Status != STATUS_END_OF_FILE)
|
2002-01-26 21:21:02 +00:00
|
|
|
{
|
|
|
|
DPRINT1("IoPageRead failed, Status %x\n", Status);
|
|
|
|
return Status;
|
|
|
|
}
|
2001-10-10 21:46:13 +00:00
|
|
|
if (CacheSeg->Bcb->CacheSegmentSize > Size)
|
2002-01-26 21:21:02 +00:00
|
|
|
{
|
2005-05-09 01:38:29 +00:00
|
|
|
memset ((char*)CacheSeg->BaseAddress + Size, 0,
|
2002-01-26 21:21:02 +00:00
|
|
|
CacheSeg->Bcb->CacheSegmentSize - Size);
|
|
|
|
}
|
2001-10-10 21:46:13 +00:00
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2005-05-09 01:38:29 +00:00
|
|
|
NTSTATUS
|
2005-09-14 01:44:19 +00:00
|
|
|
NTAPI
|
2002-09-08 10:23:54 +00:00
|
|
|
WriteCacheSegment(PCACHE_SEGMENT CacheSeg)
|
2001-10-10 21:46:13 +00:00
|
|
|
{
|
|
|
|
ULONG Size;
|
|
|
|
PMDL Mdl;
|
|
|
|
NTSTATUS Status;
|
|
|
|
IO_STATUS_BLOCK IoStatus;
|
|
|
|
LARGE_INTEGER SegOffset;
|
2002-08-28 07:13:04 +00:00
|
|
|
KEVENT Event;
|
2001-10-10 21:46:13 +00:00
|
|
|
|
2002-07-17 21:04:57 +00:00
|
|
|
CacheSeg->Dirty = FALSE;
|
2001-10-10 21:46:13 +00:00
|
|
|
SegOffset.QuadPart = CacheSeg->FileOffset;
|
2003-12-30 18:52:06 +00:00
|
|
|
Size = (ULONG)(CacheSeg->Bcb->AllocationSize.QuadPart - CacheSeg->FileOffset);
|
2001-10-10 21:46:13 +00:00
|
|
|
if (Size > CacheSeg->Bcb->CacheSegmentSize)
|
2002-07-17 21:04:57 +00:00
|
|
|
{
|
|
|
|
Size = CacheSeg->Bcb->CacheSegmentSize;
|
|
|
|
}
|
- Major rewrite of Memory Descriptor List (MDL) implementation (moving it towards using System PTEs).
- MmCreateMdl, MmSizeOfMdl: No Change.
- MmBuildMdlForNonPagedPool: Do not use MmGetPfnForProcess, just normal PMMPTE manipulation.
- This seems to cause issues in certain scenarios, because in ReactOS, nonpaged pool, a resident and guaranteed resources, does not always have its PDEs mapped!
- By calling MmGetPfnForProcess, this wound up in the annals of ReactOS mm code, which lazy-remapped the PDE. We detected this issue specifically in the cache manager, and fixed it there. It should not appear anywhere else.
- MmAllocatePagesForMdl, MmAllocatePagesForMdlEx, MmFreePagesFromMdl:
- The *Ex function is now implemented.
- Allocating pages now uses MiAllocatePagesForMdl, which is based on the older MmAllocPagesSpecifyRange.
- The code is cleaner, better commented, and better handles partial MDLs.
- Cache flags are still ignored (so the Ex functionality isn't really there).
- MmMapLockedPages, MmMapLockedPagesSpecifyCache, MmUnmapLockedPages:
- These functions now use System PTEs for the mappings, instead of the hacked-up "MDL Mapping Space".
- This frees up 256MB of Kernel Virtual Address Space.
- Takes advantage of all System PTE functionality.
- Once again, optimizations in the System PTE code will be felt here.
- For user-space mappings however, the old code is still kept and used.
- MiMapLockedPagesInUserSpace and MiUnMapLockedPagesInUserSpace are now in virtual.c and provide this.
- MmProbeAndLockPages, MmUnlockPages:
- The pages are actually probed now, in SEH. This did not seem to happen before (did someone misread the function's name?)
- Probe for write is only done for write access to user pages (as documented).
- We do not probe/check for write access for kernel requests (force Operation to be IoReadAccess).
- Proper locking is used now: Address Space lock for user mappings, PFN lock for kernel mappings.
- Faulting in pages (to make them available before locking) is now done outside the address space/PFN lock.
- You don't want to be holding a spinlock/mutex while doing disk I/O!
- For write/modify access, if the PTE is not writable, fail the request since the PTE protection overrides.
- However, if the PTE is writable but also copy on write, then we'll fault the page in for write access, which is a legitimate operation for certain user-mode scenarios.
- The old version always provided the CopyOnWrite behavior, even for non-CopyOnWrite pages!
- Reference and lock every valid page that has a PFN entry (non-I/O Pages).
- The older code did not seem to lock pages that had to be faulted in (weren't already valid).
- Cleanup the cleanup code (no pun intended). Because we now mark the pages as locked early-on, and because of changes in MmUnlockPages, we can simply use MmUnlockPages in case of error, since it will be able to fully back-out and references/locks that we did.
- Previous code attempted to do this on its own, in a pretty inconsistent manner, which would leave page leaks (both in references and lock count).
- In MmUnlockPages, not as many changes, but we now:
- Still make sure that an I/O Mapping MDL doesn't have valid PFN database pages (non-I/O).
- An MDL can cover pages that are both I/O mapped and RAM mapped, so we have to unlock/dereference the latter instead of skipping them as the old code did.
- Use the PFN lock when checking pages and unlocking/dereferencing them.
- Overall, non-complete MDLs are now marked by having a -1 PFN, and the MDL code has been updated to early-break out of page-scanning loops and/or ignore such pages, which can happen in a sparse MDL.
- Implementation has been tested on VMWare and QEMU for a variety of tasks and was found to be reliable and stable.
svn path=/trunk/; revision=41707
2009-06-30 08:29:22 +00:00
|
|
|
//
|
|
|
|
// Nonpaged pool PDEs in ReactOS must actually be synchronized between the
|
|
|
|
// MmGlobalPageDirectory and the real system PDE directory. What a mess...
|
|
|
|
//
|
|
|
|
{
|
|
|
|
int i = 0;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
MmGetPfnForProcess(NULL, (PVOID)((ULONG_PTR)CacheSeg->BaseAddress + (i << PAGE_SHIFT)));
|
|
|
|
} while (++i < (Size >> PAGE_SHIFT));
|
|
|
|
}
|
2009-08-23 15:49:43 +00:00
|
|
|
Mdl = _alloca(MmSizeOfMdl(CacheSeg->BaseAddress, Size));
|
2004-06-20 02:10:07 +00:00
|
|
|
MmInitializeMdl(Mdl, CacheSeg->BaseAddress, Size);
|
2001-10-10 21:46:13 +00:00
|
|
|
MmBuildMdlForNonPagedPool(Mdl);
|
2004-06-06 08:36:31 +00:00
|
|
|
Mdl->MdlFlags |= MDL_IO_PAGE_READ;
|
2002-08-28 07:13:04 +00:00
|
|
|
KeInitializeEvent(&Event, NotificationEvent, FALSE);
|
IO Manager Cleanup continues:
- Removed many extra files that expanded the I/O Manager too much. We usually stick with the standard of
one object/class per file, like io/device.c or io/controller.c, so it was very confusing to have some
objects split up in 5 or 6 different files, some containing only one api. Additionally, even a third
system was used, were objects were bunched up together by class. This mess was so bad that NtCreateFile,
IopCreateFile, IoCreateFile, IopDeleteFile, NtDeleteFile and NtWriteFile were in 5 different files (as an
example).
- Cleaned up some IRP code and fixed a couple of bugs, mainly:
- Write I/O Type in IRP
- Write proper IRP Flags where they shoudl be used (Will help for completing requests when i clean up that code)
- Do *NOT* zero out buffers or data that shouldn't be zeroed. Scsiport actually dependen on this incorrect
behaviour. Code should never depend on a buffer being zeroed!
- Remove a lot of duplicated code and helper/alternate functions that weren't really useful.
- Free MDL and IRP on some failures where we didn't
- Alphabetized some of the large io files for easier lookup of functions. This and the deletions have resulted
in a completely bloated diff file. I will provide a cleaned up diff on request by manually downloading the
old revision and copy/pasting the new code directly above it. The functions which we touched are:
- IoAllocateIrp
- IoBuild[A]SyncronousFsdRequest
- IoBuildDeviceIoControlRequest
- IoInitializeIrp
- IoPageRead, IoSynchronousPageWrite
svn path=/trunk/; revision=14837
2005-04-28 00:54:59 +00:00
|
|
|
Status = IoSynchronousPageWrite(CacheSeg->Bcb->FileObject, Mdl, &SegOffset, &Event, &IoStatus);
|
2002-08-28 07:13:04 +00:00
|
|
|
if (Status == STATUS_PENDING)
|
|
|
|
{
|
|
|
|
KeWaitForSingleObject(&Event, Executive, KernelMode, FALSE, NULL);
|
|
|
|
Status = IoStatus.Status;
|
|
|
|
}
|
2003-02-18 22:06:53 +00:00
|
|
|
if (!NT_SUCCESS(Status) && (Status != STATUS_END_OF_FILE))
|
2002-07-17 21:04:57 +00:00
|
|
|
{
|
|
|
|
DPRINT1("IoPageWrite failed, Status %x\n", Status);
|
|
|
|
CacheSeg->Dirty = TRUE;
|
|
|
|
return(Status);
|
|
|
|
}
|
|
|
|
return(STATUS_SUCCESS);
|
2001-10-10 21:46:13 +00:00
|
|
|
}
|
|
|
|
|
2004-06-19 05:04:33 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* @unimplemented
|
|
|
|
*/
|
2008-11-29 20:47:48 +00:00
|
|
|
BOOLEAN NTAPI
|
2004-06-19 05:04:33 +00:00
|
|
|
CcCanIWrite (
|
|
|
|
IN PFILE_OBJECT FileObject,
|
|
|
|
IN ULONG BytesToWrite,
|
|
|
|
IN BOOLEAN Wait,
|
|
|
|
IN BOOLEAN Retrying)
|
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2003-07-10 06:27:13 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2008-11-29 20:47:48 +00:00
|
|
|
BOOLEAN NTAPI
|
2002-01-26 21:21:02 +00:00
|
|
|
CcCopyRead (IN PFILE_OBJECT FileObject,
|
|
|
|
IN PLARGE_INTEGER FileOffset,
|
|
|
|
IN ULONG Length,
|
|
|
|
IN BOOLEAN Wait,
|
|
|
|
OUT PVOID Buffer,
|
|
|
|
OUT PIO_STATUS_BLOCK IoStatus)
|
2001-10-10 21:46:13 +00:00
|
|
|
{
|
2002-01-26 21:21:02 +00:00
|
|
|
ULONG ReadOffset;
|
|
|
|
ULONG TempLength;
|
|
|
|
NTSTATUS Status = STATUS_SUCCESS;
|
|
|
|
PVOID BaseAddress;
|
2002-09-08 10:23:54 +00:00
|
|
|
PCACHE_SEGMENT CacheSeg;
|
2002-01-26 21:21:02 +00:00
|
|
|
BOOLEAN Valid;
|
|
|
|
ULONG ReadLength = 0;
|
2002-09-08 10:23:54 +00:00
|
|
|
PBCB Bcb;
|
2002-01-26 21:21:02 +00:00
|
|
|
KIRQL oldirql;
|
|
|
|
PLIST_ENTRY current_entry;
|
2002-09-08 10:23:54 +00:00
|
|
|
PCACHE_SEGMENT current;
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-07-04 22:11:00 +00:00
|
|
|
DPRINT("CcCopyRead(FileObject 0x%p, FileOffset %I64x, "
|
|
|
|
"Length %d, Wait %d, Buffer 0x%p, IoStatus 0x%p)\n",
|
|
|
|
FileObject, FileOffset->QuadPart, Length, Wait,
|
2002-01-26 21:21:02 +00:00
|
|
|
Buffer, IoStatus);
|
|
|
|
|
2003-06-07 Casper S. Hornstrup <chorns@users.sourceforge.net>
Changes for compiling with w32api
* include/ddk/cctypes.h (PREACTOS_COMMON_FCB_HEADER): Remove.
(FSRTL_COMMON_FCB_HEADER): Add.
* include/ddk/iotypes.h (FILE_OBJECT): Rename field
SectionObjectPointers to SectionObjectPointer.
* ntoskrnl/cc/copy.c, ntoskrnl/cc/misc.c, ntoskrnl/cc/pin.c,
ntoskrnl/cc/view.c, ntoskrnl/io/rawfs.c, ntoskrnl/mm/section.c,
drivers/fs/cdfs/cleanup.c, drivers/fs/cdfs/fcb.c,
drivers/fs/cdfs/fsctl.c, drivers/fs/ntfs/fcb.c, drivers/fs/ntfs/fsctl.c,
drivers/fs/vfat/close.c, drivers/fs/vfat/create.c,
drivers/fs/vfat/finfo.c, drivers/fs/vfat/fcb.c, drivers/fs/vfat/fsctl.c:
Use new FILE_OBJECT structure.
* drivers/fs/cdfs/cdfs.h, drivers/fs/ntfs/ntfs.h, drivers/fs/vfat/vfat.h:
Use new FSRTL_COMMON_FCB_HEADER structure.
* drivers/net/afd/include/afd.h (FSRTL_COMMON_FCB_HEADER): Remove.
* include/ddk/ketypes.h (KQUEUE): Match w32api structure.
* ntoskrnl/ke/queue.c, ntoskrnl/ke/wait.c: Use new structure.
* ntoskrnl/ke/spinlock.c (KeAcquireSpinLockAtDpcLevel,
KeReleaseSpinLockFromDpcLevel): Undefine before declaring.
svn path=/trunk/; revision=4865
2003-06-07 11:34:36 +00:00
|
|
|
Bcb = FileObject->SectionObjectPointer->SharedCacheMap;
|
2003-12-30 18:52:06 +00:00
|
|
|
ReadOffset = (ULONG)FileOffset->QuadPart;
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2002-01-26 21:21:02 +00:00
|
|
|
DPRINT("AllocationSize %d, FileSize %d\n",
|
2001-10-10 21:46:13 +00:00
|
|
|
(ULONG)Bcb->AllocationSize.QuadPart,
|
|
|
|
(ULONG)Bcb->FileSize.QuadPart);
|
|
|
|
|
2002-01-26 21:21:02 +00:00
|
|
|
/*
|
|
|
|
* Check for the nowait case that all the cache segments that would
|
|
|
|
* cover this read are in memory.
|
|
|
|
*/
|
|
|
|
if (!Wait)
|
|
|
|
{
|
|
|
|
KeAcquireSpinLock(&Bcb->BcbLock, &oldirql);
|
|
|
|
current_entry = Bcb->BcbSegmentListHead.Flink;
|
|
|
|
while (current_entry != &Bcb->BcbSegmentListHead)
|
|
|
|
{
|
2005-05-09 01:38:29 +00:00
|
|
|
current = CONTAINING_RECORD(current_entry, CACHE_SEGMENT,
|
2002-01-26 21:21:02 +00:00
|
|
|
BcbSegmentListEntry);
|
|
|
|
if (!current->Valid && current->FileOffset < ReadOffset + Length
|
|
|
|
&& current->FileOffset + Bcb->CacheSegmentSize > ReadOffset)
|
|
|
|
{
|
|
|
|
KeReleaseSpinLock(&Bcb->BcbLock, oldirql);
|
|
|
|
IoStatus->Status = STATUS_UNSUCCESSFUL;
|
|
|
|
IoStatus->Information = 0;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
current_entry = current_entry->Flink;
|
|
|
|
}
|
|
|
|
KeReleaseSpinLock(&Bcb->BcbLock, oldirql);
|
|
|
|
}
|
|
|
|
|
|
|
|
TempLength = ReadOffset % Bcb->CacheSegmentSize;
|
|
|
|
if (TempLength != 0)
|
|
|
|
{
|
|
|
|
TempLength = min (Length, Bcb->CacheSegmentSize - TempLength);
|
|
|
|
Status = CcRosRequestCacheSegment(Bcb,
|
2005-05-09 01:38:29 +00:00
|
|
|
ROUND_DOWN(ReadOffset,
|
2002-01-26 21:21:02 +00:00
|
|
|
Bcb->CacheSegmentSize),
|
|
|
|
&BaseAddress, &Valid, &CacheSeg);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
IoStatus->Information = 0;
|
|
|
|
IoStatus->Status = Status;
|
|
|
|
DPRINT("CcRosRequestCacheSegment faild, Status %x\n", Status);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
if (!Valid)
|
|
|
|
{
|
|
|
|
Status = ReadCacheSegment(CacheSeg);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
IoStatus->Information = 0;
|
|
|
|
IoStatus->Status = Status;
|
2003-05-25 21:50:18 +00:00
|
|
|
CcRosReleaseCacheSegment(Bcb, CacheSeg, FALSE, FALSE, FALSE);
|
2002-01-26 21:21:02 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
memcpy (Buffer, (char*)BaseAddress + ReadOffset % Bcb->CacheSegmentSize,
|
2002-01-26 21:21:02 +00:00
|
|
|
TempLength);
|
|
|
|
CcRosReleaseCacheSegment(Bcb, CacheSeg, TRUE, FALSE, FALSE);
|
|
|
|
ReadLength += TempLength;
|
|
|
|
Length -= TempLength;
|
|
|
|
ReadOffset += TempLength;
|
2004-05-22 18:28:18 +00:00
|
|
|
Buffer = (PVOID)((char*)Buffer + TempLength);
|
2005-05-09 01:38:29 +00:00
|
|
|
}
|
2002-01-26 21:21:02 +00:00
|
|
|
while (Length > 0)
|
|
|
|
{
|
2004-06-06 08:36:31 +00:00
|
|
|
TempLength = min(max(Bcb->CacheSegmentSize, MAX_RW_LENGTH), Length);
|
2006-12-06 13:29:21 +00:00
|
|
|
Status = ReadCacheSegmentChain(Bcb, ReadOffset, TempLength, Buffer);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
IoStatus->Information = 0;
|
|
|
|
IoStatus->Status = Status;
|
|
|
|
DPRINT1("ReadCacheSegmentChain failed, Status %x\n", Status);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2002-01-26 21:21:02 +00:00
|
|
|
ReadLength += TempLength;
|
|
|
|
Length -= TempLength;
|
|
|
|
ReadOffset += TempLength;
|
2005-07-05 00:57:47 +00:00
|
|
|
|
|
|
|
Buffer = (PVOID)((ULONG_PTR)Buffer + TempLength);
|
2002-01-26 21:21:02 +00:00
|
|
|
}
|
|
|
|
IoStatus->Status = STATUS_SUCCESS;
|
|
|
|
IoStatus->Information = ReadLength;
|
|
|
|
DPRINT("CcCopyRead O.K.\n");
|
|
|
|
return TRUE;
|
2001-10-10 21:46:13 +00:00
|
|
|
}
|
|
|
|
|
2003-07-10 06:27:13 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2008-11-29 20:47:48 +00:00
|
|
|
BOOLEAN NTAPI
|
2002-07-17 21:04:57 +00:00
|
|
|
CcCopyWrite (IN PFILE_OBJECT FileObject,
|
|
|
|
IN PLARGE_INTEGER FileOffset,
|
|
|
|
IN ULONG Length,
|
|
|
|
IN BOOLEAN Wait,
|
|
|
|
IN PVOID Buffer)
|
2001-10-10 21:46:13 +00:00
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
ULONG WriteOffset;
|
|
|
|
KIRQL oldirql;
|
2002-09-08 10:23:54 +00:00
|
|
|
PBCB Bcb;
|
2001-10-10 21:46:13 +00:00
|
|
|
PLIST_ENTRY current_entry;
|
2002-09-08 10:23:54 +00:00
|
|
|
PCACHE_SEGMENT CacheSeg;
|
2001-10-10 21:46:13 +00:00
|
|
|
ULONG TempLength;
|
|
|
|
PVOID BaseAddress;
|
|
|
|
BOOLEAN Valid;
|
|
|
|
|
2005-07-04 22:11:00 +00:00
|
|
|
DPRINT("CcCopyWrite(FileObject 0x%p, FileOffset %I64x, "
|
|
|
|
"Length %d, Wait %d, Buffer 0x%p)\n",
|
|
|
|
FileObject, FileOffset->QuadPart, Length, Wait, Buffer);
|
2001-10-10 21:46:13 +00:00
|
|
|
|
2003-06-07 Casper S. Hornstrup <chorns@users.sourceforge.net>
Changes for compiling with w32api
* include/ddk/cctypes.h (PREACTOS_COMMON_FCB_HEADER): Remove.
(FSRTL_COMMON_FCB_HEADER): Add.
* include/ddk/iotypes.h (FILE_OBJECT): Rename field
SectionObjectPointers to SectionObjectPointer.
* ntoskrnl/cc/copy.c, ntoskrnl/cc/misc.c, ntoskrnl/cc/pin.c,
ntoskrnl/cc/view.c, ntoskrnl/io/rawfs.c, ntoskrnl/mm/section.c,
drivers/fs/cdfs/cleanup.c, drivers/fs/cdfs/fcb.c,
drivers/fs/cdfs/fsctl.c, drivers/fs/ntfs/fcb.c, drivers/fs/ntfs/fsctl.c,
drivers/fs/vfat/close.c, drivers/fs/vfat/create.c,
drivers/fs/vfat/finfo.c, drivers/fs/vfat/fcb.c, drivers/fs/vfat/fsctl.c:
Use new FILE_OBJECT structure.
* drivers/fs/cdfs/cdfs.h, drivers/fs/ntfs/ntfs.h, drivers/fs/vfat/vfat.h:
Use new FSRTL_COMMON_FCB_HEADER structure.
* drivers/net/afd/include/afd.h (FSRTL_COMMON_FCB_HEADER): Remove.
* include/ddk/ketypes.h (KQUEUE): Match w32api structure.
* ntoskrnl/ke/queue.c, ntoskrnl/ke/wait.c: Use new structure.
* ntoskrnl/ke/spinlock.c (KeAcquireSpinLockAtDpcLevel,
KeReleaseSpinLockFromDpcLevel): Undefine before declaring.
svn path=/trunk/; revision=4865
2003-06-07 11:34:36 +00:00
|
|
|
Bcb = FileObject->SectionObjectPointer->SharedCacheMap;
|
2001-10-10 21:46:13 +00:00
|
|
|
WriteOffset = (ULONG)FileOffset->QuadPart;
|
|
|
|
|
|
|
|
if (!Wait)
|
|
|
|
{
|
2002-07-17 21:04:57 +00:00
|
|
|
/* testing, if the requested datas are available */
|
|
|
|
KeAcquireSpinLock(&Bcb->BcbLock, &oldirql);
|
|
|
|
current_entry = Bcb->BcbSegmentListHead.Flink;
|
|
|
|
while (current_entry != &Bcb->BcbSegmentListHead)
|
|
|
|
{
|
2005-05-09 01:38:29 +00:00
|
|
|
CacheSeg = CONTAINING_RECORD(current_entry, CACHE_SEGMENT,
|
2002-07-17 21:04:57 +00:00
|
|
|
BcbSegmentListEntry);
|
|
|
|
if (!CacheSeg->Valid)
|
|
|
|
{
|
2005-05-09 01:38:29 +00:00
|
|
|
if ((WriteOffset >= CacheSeg->FileOffset &&
|
2002-07-17 21:04:57 +00:00
|
|
|
WriteOffset < CacheSeg->FileOffset + Bcb->CacheSegmentSize)
|
2005-05-09 01:38:29 +00:00
|
|
|
|| (WriteOffset + Length > CacheSeg->FileOffset &&
|
|
|
|
WriteOffset + Length <= CacheSeg->FileOffset +
|
2002-07-17 21:04:57 +00:00
|
|
|
Bcb->CacheSegmentSize))
|
|
|
|
{
|
|
|
|
KeReleaseSpinLock(&Bcb->BcbLock, oldirql);
|
|
|
|
/* datas not available */
|
|
|
|
return(FALSE);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
current_entry = current_entry->Flink;
|
|
|
|
}
|
|
|
|
KeReleaseSpinLock(&Bcb->BcbLock, oldirql);
|
2001-10-10 21:46:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
TempLength = WriteOffset % Bcb->CacheSegmentSize;
|
|
|
|
if (TempLength != 0)
|
2002-07-17 21:04:57 +00:00
|
|
|
{
|
|
|
|
ULONG ROffset;
|
|
|
|
ROffset = ROUND_DOWN(WriteOffset, Bcb->CacheSegmentSize);
|
|
|
|
TempLength = min (Length, Bcb->CacheSegmentSize - TempLength);
|
|
|
|
Status = CcRosRequestCacheSegment(Bcb, ROffset,
|
|
|
|
&BaseAddress, &Valid, &CacheSeg);
|
|
|
|
if (!NT_SUCCESS(Status))
|
2002-06-10 21:11:56 +00:00
|
|
|
{
|
2002-07-17 21:04:57 +00:00
|
|
|
return(FALSE);
|
2002-06-10 21:11:56 +00:00
|
|
|
}
|
2002-07-17 21:04:57 +00:00
|
|
|
if (!Valid)
|
2002-06-10 21:11:56 +00:00
|
|
|
{
|
2002-07-17 21:04:57 +00:00
|
|
|
if (!NT_SUCCESS(ReadCacheSegment(CacheSeg)))
|
|
|
|
{
|
|
|
|
return(FALSE);
|
|
|
|
}
|
2002-06-10 21:11:56 +00:00
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
memcpy ((char*)BaseAddress + WriteOffset % Bcb->CacheSegmentSize,
|
2002-07-17 21:04:57 +00:00
|
|
|
Buffer, TempLength);
|
|
|
|
CcRosReleaseCacheSegment(Bcb, CacheSeg, TRUE, TRUE, FALSE);
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2002-07-17 21:04:57 +00:00
|
|
|
Length -= TempLength;
|
|
|
|
WriteOffset += TempLength;
|
2005-07-05 00:57:47 +00:00
|
|
|
|
|
|
|
Buffer = (PVOID)((ULONG_PTR)Buffer + TempLength);
|
2002-07-17 21:04:57 +00:00
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2002-07-17 21:04:57 +00:00
|
|
|
while (Length > 0)
|
|
|
|
{
|
|
|
|
TempLength = min (Bcb->CacheSegmentSize, Length);
|
|
|
|
Status = CcRosRequestCacheSegment(Bcb, WriteOffset,
|
|
|
|
&BaseAddress, &Valid, &CacheSeg);
|
|
|
|
if (!NT_SUCCESS(Status))
|
2002-06-10 21:11:56 +00:00
|
|
|
{
|
2002-07-17 21:04:57 +00:00
|
|
|
return(FALSE);
|
2002-06-10 21:11:56 +00:00
|
|
|
}
|
2002-07-17 21:04:57 +00:00
|
|
|
if (!Valid && TempLength < Bcb->CacheSegmentSize)
|
2002-06-10 21:11:56 +00:00
|
|
|
{
|
2002-07-17 21:04:57 +00:00
|
|
|
if (!NT_SUCCESS(ReadCacheSegment(CacheSeg)))
|
|
|
|
{
|
2003-05-25 21:50:18 +00:00
|
|
|
CcRosReleaseCacheSegment(Bcb, CacheSeg, FALSE, FALSE, FALSE);
|
2002-07-17 21:04:57 +00:00
|
|
|
return FALSE;
|
|
|
|
}
|
2002-06-10 21:11:56 +00:00
|
|
|
}
|
2002-07-17 21:04:57 +00:00
|
|
|
memcpy (BaseAddress, Buffer, TempLength);
|
|
|
|
CcRosReleaseCacheSegment(Bcb, CacheSeg, TRUE, TRUE, FALSE);
|
|
|
|
Length -= TempLength;
|
|
|
|
WriteOffset += TempLength;
|
2005-07-05 00:57:47 +00:00
|
|
|
|
|
|
|
Buffer = (PVOID)((ULONG_PTR)Buffer + TempLength);
|
2002-07-17 21:04:57 +00:00
|
|
|
}
|
|
|
|
return(TRUE);
|
|
|
|
}
|
|
|
|
|
2004-06-21 04:11:44 +00:00
|
|
|
/*
|
|
|
|
* @unimplemented
|
|
|
|
*/
|
|
|
|
VOID
|
2008-11-29 20:47:48 +00:00
|
|
|
NTAPI
|
2004-06-21 04:11:44 +00:00
|
|
|
CcDeferWrite (
|
|
|
|
IN PFILE_OBJECT FileObject,
|
|
|
|
IN PCC_POST_DEFERRED_WRITE PostRoutine,
|
|
|
|
IN PVOID Context1,
|
|
|
|
IN PVOID Context2,
|
|
|
|
IN ULONG BytesToWrite,
|
|
|
|
IN BOOLEAN Retrying
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @unimplemented
|
|
|
|
*/
|
2005-01-01 01:51:05 +00:00
|
|
|
VOID
|
2008-11-29 20:47:48 +00:00
|
|
|
NTAPI
|
2004-06-21 04:11:44 +00:00
|
|
|
CcFastCopyRead (
|
2005-01-01 01:51:05 +00:00
|
|
|
IN PFILE_OBJECT FileObject,
|
|
|
|
IN ULONG FileOffset,
|
|
|
|
IN ULONG Length,
|
|
|
|
IN ULONG PageCount,
|
|
|
|
OUT PVOID Buffer,
|
|
|
|
OUT PIO_STATUS_BLOCK IoStatus
|
2004-06-21 04:11:44 +00:00
|
|
|
)
|
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* @unimplemented
|
|
|
|
*/
|
2005-01-01 01:51:05 +00:00
|
|
|
VOID
|
2008-11-29 20:47:48 +00:00
|
|
|
NTAPI
|
2005-01-01 01:51:05 +00:00
|
|
|
CcFastCopyWrite(
|
|
|
|
IN PFILE_OBJECT FileObject,
|
|
|
|
IN ULONG FileOffset,
|
|
|
|
IN ULONG Length,
|
|
|
|
IN PVOID Buffer)
|
2004-06-21 04:11:44 +00:00
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @unimplemented
|
|
|
|
*/
|
|
|
|
NTSTATUS
|
2008-11-29 20:47:48 +00:00
|
|
|
NTAPI
|
2004-06-21 04:11:44 +00:00
|
|
|
CcWaitForCurrentLazyWriterActivity (
|
|
|
|
VOID
|
|
|
|
)
|
|
|
|
{
|
|
|
|
UNIMPLEMENTED;
|
|
|
|
return STATUS_NOT_IMPLEMENTED;
|
|
|
|
}
|
|
|
|
|
2003-07-10 06:27:13 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2008-11-29 20:47:48 +00:00
|
|
|
BOOLEAN NTAPI
|
2002-07-17 21:04:57 +00:00
|
|
|
CcZeroData (IN PFILE_OBJECT FileObject,
|
|
|
|
IN PLARGE_INTEGER StartOffset,
|
|
|
|
IN PLARGE_INTEGER EndOffset,
|
|
|
|
IN BOOLEAN Wait)
|
|
|
|
{
|
|
|
|
NTSTATUS Status;
|
|
|
|
LARGE_INTEGER WriteOffset;
|
|
|
|
ULONG Length;
|
2004-05-22 18:28:18 +00:00
|
|
|
ULONG CurrentLength;
|
2002-07-17 21:04:57 +00:00
|
|
|
PMDL Mdl;
|
|
|
|
ULONG i;
|
|
|
|
IO_STATUS_BLOCK Iosb;
|
2002-08-28 07:13:04 +00:00
|
|
|
KEVENT Event;
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-07-04 22:11:00 +00:00
|
|
|
DPRINT("CcZeroData(FileObject 0x%p, StartOffset %I64x, EndOffset %I64x, "
|
2005-05-09 01:38:29 +00:00
|
|
|
"Wait %d)\n", FileObject, StartOffset->QuadPart, EndOffset->QuadPart,
|
2002-07-17 21:04:57 +00:00
|
|
|
Wait);
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2002-07-17 21:04:57 +00:00
|
|
|
Length = EndOffset->u.LowPart - StartOffset->u.LowPart;
|
2004-05-22 18:28:18 +00:00
|
|
|
WriteOffset.QuadPart = StartOffset->QuadPart;
|
2002-07-17 21:04:57 +00:00
|
|
|
|
2003-06-07 Casper S. Hornstrup <chorns@users.sourceforge.net>
Changes for compiling with w32api
* include/ddk/cctypes.h (PREACTOS_COMMON_FCB_HEADER): Remove.
(FSRTL_COMMON_FCB_HEADER): Add.
* include/ddk/iotypes.h (FILE_OBJECT): Rename field
SectionObjectPointers to SectionObjectPointer.
* ntoskrnl/cc/copy.c, ntoskrnl/cc/misc.c, ntoskrnl/cc/pin.c,
ntoskrnl/cc/view.c, ntoskrnl/io/rawfs.c, ntoskrnl/mm/section.c,
drivers/fs/cdfs/cleanup.c, drivers/fs/cdfs/fcb.c,
drivers/fs/cdfs/fsctl.c, drivers/fs/ntfs/fcb.c, drivers/fs/ntfs/fsctl.c,
drivers/fs/vfat/close.c, drivers/fs/vfat/create.c,
drivers/fs/vfat/finfo.c, drivers/fs/vfat/fcb.c, drivers/fs/vfat/fsctl.c:
Use new FILE_OBJECT structure.
* drivers/fs/cdfs/cdfs.h, drivers/fs/ntfs/ntfs.h, drivers/fs/vfat/vfat.h:
Use new FSRTL_COMMON_FCB_HEADER structure.
* drivers/net/afd/include/afd.h (FSRTL_COMMON_FCB_HEADER): Remove.
* include/ddk/ketypes.h (KQUEUE): Match w32api structure.
* ntoskrnl/ke/queue.c, ntoskrnl/ke/wait.c: Use new structure.
* ntoskrnl/ke/spinlock.c (KeAcquireSpinLockAtDpcLevel,
KeReleaseSpinLockFromDpcLevel): Undefine before declaring.
svn path=/trunk/; revision=4865
2003-06-07 11:34:36 +00:00
|
|
|
if (FileObject->SectionObjectPointer->SharedCacheMap == NULL)
|
2002-07-17 21:04:57 +00:00
|
|
|
{
|
|
|
|
/* File is not cached */
|
2004-06-06 08:36:31 +00:00
|
|
|
|
2009-08-23 15:49:43 +00:00
|
|
|
Mdl = _alloca(MmSizeOfMdl(NULL, MAX_ZERO_LENGTH));
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2002-07-17 21:04:57 +00:00
|
|
|
while (Length > 0)
|
|
|
|
{
|
2004-05-22 18:28:18 +00:00
|
|
|
if (Length + WriteOffset.u.LowPart % PAGE_SIZE > MAX_ZERO_LENGTH)
|
2002-07-17 21:04:57 +00:00
|
|
|
{
|
2004-05-22 18:28:18 +00:00
|
|
|
CurrentLength = MAX_ZERO_LENGTH - WriteOffset.u.LowPart % PAGE_SIZE;
|
2002-07-17 21:04:57 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2004-05-22 18:28:18 +00:00
|
|
|
CurrentLength = Length;
|
2002-07-17 21:04:57 +00:00
|
|
|
}
|
2010-01-13 22:35:43 +00:00
|
|
|
MmInitializeMdl(Mdl, (PVOID)(ULONG_PTR)WriteOffset.QuadPart, CurrentLength);
|
2002-07-17 21:04:57 +00:00
|
|
|
Mdl->MdlFlags |= (MDL_PAGES_LOCKED | MDL_IO_PAGE_READ);
|
|
|
|
for (i = 0; i < ((Mdl->Size - sizeof(MDL)) / sizeof(ULONG)); i++)
|
|
|
|
{
|
2010-07-15 22:50:12 +00:00
|
|
|
((PPFN_NUMBER)(Mdl + 1))[i] = CcZeroPage;
|
2002-07-17 21:04:57 +00:00
|
|
|
}
|
2002-08-28 07:13:04 +00:00
|
|
|
KeInitializeEvent(&Event, NotificationEvent, FALSE);
|
IO Manager Cleanup continues:
- Removed many extra files that expanded the I/O Manager too much. We usually stick with the standard of
one object/class per file, like io/device.c or io/controller.c, so it was very confusing to have some
objects split up in 5 or 6 different files, some containing only one api. Additionally, even a third
system was used, were objects were bunched up together by class. This mess was so bad that NtCreateFile,
IopCreateFile, IoCreateFile, IopDeleteFile, NtDeleteFile and NtWriteFile were in 5 different files (as an
example).
- Cleaned up some IRP code and fixed a couple of bugs, mainly:
- Write I/O Type in IRP
- Write proper IRP Flags where they shoudl be used (Will help for completing requests when i clean up that code)
- Do *NOT* zero out buffers or data that shouldn't be zeroed. Scsiport actually dependen on this incorrect
behaviour. Code should never depend on a buffer being zeroed!
- Remove a lot of duplicated code and helper/alternate functions that weren't really useful.
- Free MDL and IRP on some failures where we didn't
- Alphabetized some of the large io files for easier lookup of functions. This and the deletions have resulted
in a completely bloated diff file. I will provide a cleaned up diff on request by manually downloading the
old revision and copy/pasting the new code directly above it. The functions which we touched are:
- IoAllocateIrp
- IoBuild[A]SyncronousFsdRequest
- IoBuildDeviceIoControlRequest
- IoInitializeIrp
- IoPageRead, IoSynchronousPageWrite
svn path=/trunk/; revision=14837
2005-04-28 00:54:59 +00:00
|
|
|
Status = IoSynchronousPageWrite(FileObject, Mdl, &WriteOffset, &Event, &Iosb);
|
2002-08-28 07:13:04 +00:00
|
|
|
if (Status == STATUS_PENDING)
|
|
|
|
{
|
|
|
|
KeWaitForSingleObject(&Event, Executive, KernelMode, FALSE, NULL);
|
|
|
|
Status = Iosb.Status;
|
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
MmUnmapLockedPages(Mdl->MappedSystemVa, Mdl);
|
2002-07-17 21:04:57 +00:00
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
return(FALSE);
|
|
|
|
}
|
2004-05-22 18:28:18 +00:00
|
|
|
WriteOffset.QuadPart += CurrentLength;
|
|
|
|
Length -= CurrentLength;
|
2002-07-17 21:04:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* File is cached */
|
2002-06-10 21:11:56 +00:00
|
|
|
KIRQL oldirql;
|
2002-09-08 10:23:54 +00:00
|
|
|
PBCB Bcb;
|
2002-06-10 21:11:56 +00:00
|
|
|
PLIST_ENTRY current_entry;
|
2002-09-08 10:23:54 +00:00
|
|
|
PCACHE_SEGMENT CacheSeg, current, previous;
|
2002-06-10 21:11:56 +00:00
|
|
|
ULONG TempLength;
|
|
|
|
|
2003-06-07 Casper S. Hornstrup <chorns@users.sourceforge.net>
Changes for compiling with w32api
* include/ddk/cctypes.h (PREACTOS_COMMON_FCB_HEADER): Remove.
(FSRTL_COMMON_FCB_HEADER): Add.
* include/ddk/iotypes.h (FILE_OBJECT): Rename field
SectionObjectPointers to SectionObjectPointer.
* ntoskrnl/cc/copy.c, ntoskrnl/cc/misc.c, ntoskrnl/cc/pin.c,
ntoskrnl/cc/view.c, ntoskrnl/io/rawfs.c, ntoskrnl/mm/section.c,
drivers/fs/cdfs/cleanup.c, drivers/fs/cdfs/fcb.c,
drivers/fs/cdfs/fsctl.c, drivers/fs/ntfs/fcb.c, drivers/fs/ntfs/fsctl.c,
drivers/fs/vfat/close.c, drivers/fs/vfat/create.c,
drivers/fs/vfat/finfo.c, drivers/fs/vfat/fcb.c, drivers/fs/vfat/fsctl.c:
Use new FILE_OBJECT structure.
* drivers/fs/cdfs/cdfs.h, drivers/fs/ntfs/ntfs.h, drivers/fs/vfat/vfat.h:
Use new FSRTL_COMMON_FCB_HEADER structure.
* drivers/net/afd/include/afd.h (FSRTL_COMMON_FCB_HEADER): Remove.
* include/ddk/ketypes.h (KQUEUE): Match w32api structure.
* ntoskrnl/ke/queue.c, ntoskrnl/ke/wait.c: Use new structure.
* ntoskrnl/ke/spinlock.c (KeAcquireSpinLockAtDpcLevel,
KeReleaseSpinLockFromDpcLevel): Undefine before declaring.
svn path=/trunk/; revision=4865
2003-06-07 11:34:36 +00:00
|
|
|
Bcb = FileObject->SectionObjectPointer->SharedCacheMap;
|
2002-06-10 21:11:56 +00:00
|
|
|
if (Wait)
|
2002-07-17 21:04:57 +00:00
|
|
|
{
|
|
|
|
/* testing, if the requested datas are available */
|
2002-06-10 21:11:56 +00:00
|
|
|
KeAcquireSpinLock(&Bcb->BcbLock, &oldirql);
|
|
|
|
current_entry = Bcb->BcbSegmentListHead.Flink;
|
|
|
|
while (current_entry != &Bcb->BcbSegmentListHead)
|
2002-07-17 21:04:57 +00:00
|
|
|
{
|
2005-05-09 01:38:29 +00:00
|
|
|
CacheSeg = CONTAINING_RECORD(current_entry, CACHE_SEGMENT,
|
2002-07-17 21:04:57 +00:00
|
|
|
BcbSegmentListEntry);
|
|
|
|
if (!CacheSeg->Valid)
|
2002-06-10 21:11:56 +00:00
|
|
|
{
|
2005-05-09 01:38:29 +00:00
|
|
|
if ((WriteOffset.u.LowPart >= CacheSeg->FileOffset &&
|
2004-05-22 18:28:18 +00:00
|
|
|
WriteOffset.u.LowPart < CacheSeg->FileOffset + Bcb->CacheSegmentSize)
|
2005-05-09 01:38:29 +00:00
|
|
|
|| (WriteOffset.u.LowPart + Length > CacheSeg->FileOffset &&
|
|
|
|
WriteOffset.u.LowPart + Length <=
|
2002-07-17 21:04:57 +00:00
|
|
|
CacheSeg->FileOffset + Bcb->CacheSegmentSize))
|
|
|
|
{
|
|
|
|
KeReleaseSpinLock(&Bcb->BcbLock, oldirql);
|
|
|
|
/* datas not available */
|
|
|
|
return(FALSE);
|
|
|
|
}
|
2002-06-10 21:11:56 +00:00
|
|
|
}
|
2002-07-17 21:04:57 +00:00
|
|
|
current_entry = current_entry->Flink;
|
|
|
|
}
|
2002-06-10 21:11:56 +00:00
|
|
|
KeReleaseSpinLock(&Bcb->BcbLock, oldirql);
|
2002-07-17 21:04:57 +00:00
|
|
|
}
|
2002-06-10 21:11:56 +00:00
|
|
|
while (Length > 0)
|
2002-07-17 21:04:57 +00:00
|
|
|
{
|
2004-05-22 18:28:18 +00:00
|
|
|
ULONG Offset;
|
|
|
|
Offset = WriteOffset.u.LowPart % Bcb->CacheSegmentSize;
|
|
|
|
if (Length + Offset > MAX_ZERO_LENGTH)
|
2002-06-10 21:11:56 +00:00
|
|
|
{
|
2004-05-22 18:28:18 +00:00
|
|
|
CurrentLength = MAX_ZERO_LENGTH - Offset;
|
2002-06-10 21:11:56 +00:00
|
|
|
}
|
2002-07-17 21:04:57 +00:00
|
|
|
else
|
2002-06-10 21:11:56 +00:00
|
|
|
{
|
2004-05-22 18:28:18 +00:00
|
|
|
CurrentLength = Length;
|
|
|
|
}
|
|
|
|
Status = CcRosGetCacheSegmentChain (Bcb, WriteOffset.u.LowPart - Offset,
|
|
|
|
Offset + CurrentLength, &CacheSeg);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
return FALSE;
|
2002-06-10 21:11:56 +00:00
|
|
|
}
|
2002-07-17 21:04:57 +00:00
|
|
|
current = CacheSeg;
|
2004-05-22 18:28:18 +00:00
|
|
|
|
2002-07-17 21:04:57 +00:00
|
|
|
while (current != NULL)
|
|
|
|
{
|
2004-05-22 18:28:18 +00:00
|
|
|
Offset = WriteOffset.u.LowPart % Bcb->CacheSegmentSize;
|
|
|
|
if (Offset != 0 ||
|
|
|
|
Offset + CurrentLength < Bcb->CacheSegmentSize)
|
|
|
|
{
|
2002-07-17 21:04:57 +00:00
|
|
|
if (!current->Valid)
|
|
|
|
{
|
2004-05-22 18:28:18 +00:00
|
|
|
/* read the segment */
|
2002-07-17 21:04:57 +00:00
|
|
|
Status = ReadCacheSegment(current);
|
|
|
|
if (!NT_SUCCESS(Status))
|
|
|
|
{
|
2005-05-09 01:38:29 +00:00
|
|
|
DPRINT1("ReadCacheSegment failed, status %x\n",
|
2002-07-17 21:04:57 +00:00
|
|
|
Status);
|
|
|
|
}
|
|
|
|
}
|
2004-05-22 18:28:18 +00:00
|
|
|
TempLength = min (CurrentLength, Bcb->CacheSegmentSize - Offset);
|
2002-07-17 21:04:57 +00:00
|
|
|
}
|
|
|
|
else
|
2004-05-22 18:28:18 +00:00
|
|
|
{
|
2002-07-17 21:04:57 +00:00
|
|
|
TempLength = Bcb->CacheSegmentSize;
|
|
|
|
}
|
2004-05-22 18:28:18 +00:00
|
|
|
memset ((PUCHAR)current->BaseAddress + Offset, 0, TempLength);
|
|
|
|
|
|
|
|
WriteOffset.QuadPart += TempLength;
|
|
|
|
CurrentLength -= TempLength;
|
2002-07-17 21:04:57 +00:00
|
|
|
Length -= TempLength;
|
2004-05-22 18:28:18 +00:00
|
|
|
|
|
|
|
current = current->NextInChain;
|
2005-05-09 01:38:29 +00:00
|
|
|
}
|
2004-05-22 18:28:18 +00:00
|
|
|
|
|
|
|
current = CacheSeg;
|
2002-07-17 21:04:57 +00:00
|
|
|
while (current != NULL)
|
|
|
|
{
|
|
|
|
previous = current;
|
|
|
|
current = current->NextInChain;
|
2004-05-22 18:28:18 +00:00
|
|
|
CcRosReleaseCacheSegment(Bcb, previous, TRUE, TRUE, FALSE);
|
2002-07-17 21:04:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return(TRUE);
|
2002-06-10 21:11:56 +00:00
|
|
|
}
|