mirror of
https://github.com/reactos/reactos.git
synced 2024-11-09 08:08:38 +00:00
6afbc8f483
svn path=/branches/reactos-yarotows/; revision=45219
250 lines
5.5 KiB
C
250 lines
5.5 KiB
C
/*
|
|
* COPYRIGHT: See COPYING in the top level directory
|
|
* PROJECT: ReactOS kernel
|
|
* FILE: ntoskrnl/cc/fs.c
|
|
* PURPOSE: Implements cache managers functions useful for File Systems
|
|
*
|
|
* PROGRAMMERS: Alex Ionescu
|
|
*/
|
|
|
|
/* INCLUDES ******************************************************************/
|
|
|
|
#include <ntoskrnl.h>
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
#ifndef VACB_MAPPING_GRANULARITY
|
|
#define VACB_MAPPING_GRANULARITY (256 * 1024)
|
|
#endif
|
|
|
|
/* GLOBALS *****************************************************************/
|
|
|
|
extern KGUARDED_MUTEX ViewLock;
|
|
extern ULONG DirtyPageCount;
|
|
|
|
NTSTATUS CcRosInternalFreeCacheSegment(PCACHE_SEGMENT CacheSeg);
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
/*
|
|
* @unimplemented
|
|
*/
|
|
LARGE_INTEGER
|
|
NTAPI
|
|
CcGetDirtyPages (
|
|
IN PVOID LogHandle,
|
|
IN PDIRTY_PAGE_ROUTINE DirtyPageRoutine,
|
|
IN PVOID Context1,
|
|
IN PVOID Context2
|
|
)
|
|
{
|
|
LARGE_INTEGER i;
|
|
UNIMPLEMENTED;
|
|
i.QuadPart = 0;
|
|
return i;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
PFILE_OBJECT
|
|
NTAPI
|
|
CcGetFileObjectFromBcb (
|
|
IN PVOID Bcb
|
|
)
|
|
{
|
|
PINTERNAL_BCB iBcb = (PINTERNAL_BCB)Bcb;
|
|
return iBcb->CacheSegment->Bcb->FileObject;
|
|
}
|
|
|
|
/*
|
|
* @unimplemented
|
|
*/
|
|
LARGE_INTEGER
|
|
NTAPI
|
|
CcGetLsnForFileObject (
|
|
IN PFILE_OBJECT FileObject,
|
|
OUT PLARGE_INTEGER OldestLsn OPTIONAL
|
|
)
|
|
{
|
|
LARGE_INTEGER i;
|
|
UNIMPLEMENTED;
|
|
i.QuadPart = 0;
|
|
return i;
|
|
}
|
|
|
|
/*
|
|
* @unimplemented
|
|
*/
|
|
VOID
|
|
NTAPI
|
|
CcInitializeCacheMap (
|
|
IN PFILE_OBJECT FileObject,
|
|
IN PCC_FILE_SIZES FileSizes,
|
|
IN BOOLEAN PinAccess,
|
|
IN PCACHE_MANAGER_CALLBACKS CallBacks,
|
|
IN PVOID LazyWriterContext
|
|
)
|
|
{
|
|
ASSERT(FileObject);
|
|
ASSERT(FileSizes);
|
|
|
|
/* Call old ROS cache init function */
|
|
CcRosInitializeFileCache(FileObject,
|
|
/*PAGE_SIZE*/ VACB_MAPPING_GRANULARITY, CallBacks,
|
|
LazyWriterContext);
|
|
}
|
|
|
|
/*
|
|
* @unimplemented
|
|
*/
|
|
BOOLEAN
|
|
NTAPI
|
|
CcIsThereDirtyData (
|
|
IN PVPB Vpb
|
|
)
|
|
{
|
|
UNIMPLEMENTED;
|
|
return FALSE;
|
|
}
|
|
|
|
/*
|
|
* @unimplemented
|
|
*/
|
|
BOOLEAN
|
|
NTAPI
|
|
CcPurgeCacheSection (
|
|
IN PSECTION_OBJECT_POINTERS SectionObjectPointer,
|
|
IN PLARGE_INTEGER FileOffset OPTIONAL,
|
|
IN ULONG Length,
|
|
IN BOOLEAN UninitializeCacheMaps
|
|
)
|
|
{
|
|
//UNIMPLEMENTED;
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
VOID NTAPI
|
|
CcSetFileSizes (IN PFILE_OBJECT FileObject,
|
|
IN PCC_FILE_SIZES FileSizes)
|
|
{
|
|
KIRQL oldirql;
|
|
PBCB Bcb;
|
|
PLIST_ENTRY current_entry;
|
|
PCACHE_SEGMENT current;
|
|
LIST_ENTRY FreeListHead;
|
|
NTSTATUS Status;
|
|
|
|
DPRINT("CcSetFileSizes(FileObject 0x%p, FileSizes 0x%p)\n",
|
|
FileObject, FileSizes);
|
|
DPRINT("AllocationSize %d, FileSize %d, ValidDataLength %d\n",
|
|
(ULONG)FileSizes->AllocationSize.QuadPart,
|
|
(ULONG)FileSizes->FileSize.QuadPart,
|
|
(ULONG)FileSizes->ValidDataLength.QuadPart);
|
|
|
|
Bcb = FileObject->SectionObjectPointer->SharedCacheMap;
|
|
|
|
/*
|
|
* It is valid to call this function on file objects that weren't
|
|
* initialized for caching. In this case it's simple no-op.
|
|
*/
|
|
if (Bcb == NULL)
|
|
return;
|
|
|
|
if (FileSizes->AllocationSize.QuadPart < Bcb->AllocationSize.QuadPart)
|
|
{
|
|
InitializeListHead(&FreeListHead);
|
|
KeAcquireGuardedMutex(&ViewLock);
|
|
KeAcquireSpinLock(&Bcb->BcbLock, &oldirql);
|
|
|
|
current_entry = Bcb->BcbSegmentListHead.Flink;
|
|
while (current_entry != &Bcb->BcbSegmentListHead)
|
|
{
|
|
current = CONTAINING_RECORD(current_entry, CACHE_SEGMENT, BcbSegmentListEntry);
|
|
current_entry = current_entry->Flink;
|
|
if (current->FileOffset > FileSizes->AllocationSize.QuadPart ||
|
|
(current->FileOffset == 0 && FileSizes->AllocationSize.QuadPart == 0))
|
|
{
|
|
if (current->ReferenceCount == 0 || (current->ReferenceCount == 1 && current->Dirty))
|
|
{
|
|
RemoveEntryList(¤t->BcbSegmentListEntry);
|
|
RemoveEntryList(¤t->CacheSegmentListEntry);
|
|
RemoveEntryList(¤t->CacheSegmentLRUListEntry);
|
|
if (current->Dirty)
|
|
{
|
|
RemoveEntryList(¤t->DirtySegmentListEntry);
|
|
DirtyPageCount -= Bcb->CacheSegmentSize / PAGE_SIZE;
|
|
}
|
|
InsertHeadList(&FreeListHead, ¤t->BcbSegmentListEntry);
|
|
}
|
|
else
|
|
{
|
|
DPRINT1("Anyone has referenced a cache segment behind the new size.\n");
|
|
KeBugCheck(CACHE_MANAGER);
|
|
}
|
|
}
|
|
}
|
|
|
|
Bcb->AllocationSize = FileSizes->AllocationSize;
|
|
Bcb->FileSize = FileSizes->FileSize;
|
|
KeReleaseSpinLock(&Bcb->BcbLock, oldirql);
|
|
KeReleaseGuardedMutex(&ViewLock);
|
|
|
|
current_entry = FreeListHead.Flink;
|
|
while(current_entry != &FreeListHead)
|
|
{
|
|
current = CONTAINING_RECORD(current_entry, CACHE_SEGMENT, BcbSegmentListEntry);
|
|
current_entry = current_entry->Flink;
|
|
Status = CcRosInternalFreeCacheSegment(current);
|
|
if (!NT_SUCCESS(Status))
|
|
{
|
|
DPRINT1("CcRosInternalFreeCacheSegment failed, status = %x\n", Status);
|
|
KeBugCheck(CACHE_MANAGER);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
KeAcquireSpinLock(&Bcb->BcbLock, &oldirql);
|
|
Bcb->AllocationSize = FileSizes->AllocationSize;
|
|
Bcb->FileSize = FileSizes->FileSize;
|
|
KeReleaseSpinLock(&Bcb->BcbLock, oldirql);
|
|
}
|
|
}
|
|
|
|
/*
|
|
* @unimplemented
|
|
*/
|
|
VOID
|
|
NTAPI
|
|
CcSetLogHandleForFile (
|
|
IN PFILE_OBJECT FileObject,
|
|
IN PVOID LogHandle,
|
|
IN PFLUSH_TO_LSN FlushToLsnRoutine
|
|
)
|
|
{
|
|
UNIMPLEMENTED;
|
|
}
|
|
|
|
/*
|
|
* @unimplemented
|
|
*/
|
|
BOOLEAN
|
|
NTAPI
|
|
CcUninitializeCacheMap (
|
|
IN PFILE_OBJECT FileObject,
|
|
IN PLARGE_INTEGER TruncateSize OPTIONAL,
|
|
IN PCACHE_UNINITIALIZE_EVENT UninitializeCompleteEvent OPTIONAL
|
|
)
|
|
{
|
|
#if 0
|
|
UNIMPLEMENTED;
|
|
return FALSE;
|
|
#else
|
|
return NT_SUCCESS(CcRosReleaseFileCache(FileObject));
|
|
#endif
|
|
}
|