2016-12-09 03:07:49 +01:00
|
|
|
/** @file
|
|
|
|
Cache implementation for EFI FAT File system driver.
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2013-10-30 04:13:16 +01:00
|
|
|
Copyright (c) 2005 - 2013, Intel Corporation. All rights reserved.<BR>
|
2019-04-04 01:03:46 +02:00
|
|
|
SPDX-License-Identifier: BSD-2-Clause-Patent
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
**/
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
#include "Fat.h"
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
/**
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
This function is used by the Data Cache.
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
When this function is called by write command, all entries in this range
|
|
|
|
are older than the contents in disk, so they are invalid; just mark them invalid.
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
When this function is called by read command, if any entry in this range
|
2019-10-09 07:38:15 +02:00
|
|
|
is dirty, it means that the relative info directly read from media is older than
|
2016-12-09 03:07:49 +01:00
|
|
|
than the info in the cache; So need to update the relative info in the Buffer.
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
@param Volume - FAT file system volume.
|
|
|
|
@param IoMode - This function is called by read command or write command
|
|
|
|
@param StartPageNo - First PageNo to be checked in the cache.
|
|
|
|
@param EndPageNo - Last PageNo to be checked in the cache.
|
|
|
|
@param Buffer - The user buffer need to update. Only when doing the read command
|
|
|
|
and there is dirty cache in the cache range, this parameter will be used.
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
**/
|
2007-12-20 10:16:45 +01:00
|
|
|
STATIC
|
|
|
|
VOID
|
|
|
|
FatFlushDataCacheRange (
|
2021-12-05 23:53:58 +01:00
|
|
|
IN FAT_VOLUME *Volume,
|
|
|
|
IN IO_MODE IoMode,
|
|
|
|
IN UINTN StartPageNo,
|
|
|
|
IN UINTN EndPageNo,
|
|
|
|
OUT UINT8 *Buffer
|
2007-12-20 10:16:45 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
UINTN PageNo;
|
|
|
|
UINTN GroupNo;
|
|
|
|
UINTN GroupMask;
|
|
|
|
UINTN PageSize;
|
|
|
|
UINT8 PageAlignment;
|
|
|
|
DISK_CACHE *DiskCache;
|
|
|
|
CACHE_TAG *CacheTag;
|
|
|
|
UINT8 *BaseAddress;
|
|
|
|
|
2016-12-08 06:18:08 +01:00
|
|
|
DiskCache = &Volume->DiskCache[CacheData];
|
2007-12-20 10:16:45 +01:00
|
|
|
BaseAddress = DiskCache->CacheBase;
|
|
|
|
GroupMask = DiskCache->GroupMask;
|
|
|
|
PageAlignment = DiskCache->PageAlignment;
|
|
|
|
PageSize = (UINTN)1 << PageAlignment;
|
|
|
|
|
|
|
|
for (PageNo = StartPageNo; PageNo < EndPageNo; PageNo++) {
|
2021-12-05 23:53:58 +01:00
|
|
|
GroupNo = PageNo & GroupMask;
|
|
|
|
CacheTag = &DiskCache->CacheTag[GroupNo];
|
|
|
|
if ((CacheTag->RealSize > 0) && (CacheTag->PageNo == PageNo)) {
|
2007-12-20 10:16:45 +01:00
|
|
|
//
|
|
|
|
// When reading data form disk directly, if some dirty data
|
|
|
|
// in cache is in this rang, this data in the Buffer need to
|
|
|
|
// be updated with the cache's dirty data.
|
|
|
|
//
|
2016-12-08 06:18:08 +01:00
|
|
|
if (IoMode == ReadDisk) {
|
2007-12-20 10:16:45 +01:00
|
|
|
if (CacheTag->Dirty) {
|
|
|
|
CopyMem (
|
|
|
|
Buffer + ((PageNo - StartPageNo) << PageAlignment),
|
|
|
|
BaseAddress + (GroupNo << PageAlignment),
|
|
|
|
PageSize
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
//
|
|
|
|
// Make all valid entries in this range invalid.
|
|
|
|
//
|
|
|
|
CacheTag->RealSize = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
/**
|
|
|
|
|
|
|
|
Exchange the cache page with the image on the disk
|
|
|
|
|
|
|
|
@param Volume - FAT file system volume.
|
|
|
|
@param DataType - Indicate the cache type.
|
|
|
|
@param IoMode - Indicate whether to load this page from disk or store this page to disk.
|
|
|
|
@param CacheTag - The Cache Tag for the current cache page.
|
|
|
|
@param Task point to task instance.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS - Cache page exchanged successfully.
|
|
|
|
@return Others - An error occurred when exchanging cache page.
|
|
|
|
|
|
|
|
**/
|
2007-12-20 10:16:45 +01:00
|
|
|
STATIC
|
|
|
|
EFI_STATUS
|
|
|
|
FatExchangeCachePage (
|
2021-12-05 23:53:58 +01:00
|
|
|
IN FAT_VOLUME *Volume,
|
|
|
|
IN CACHE_DATA_TYPE DataType,
|
|
|
|
IN IO_MODE IoMode,
|
|
|
|
IN CACHE_TAG *CacheTag,
|
|
|
|
IN FAT_TASK *Task
|
2007-12-20 10:16:45 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN GroupNo;
|
|
|
|
UINTN PageNo;
|
|
|
|
UINTN WriteCount;
|
|
|
|
UINTN RealSize;
|
|
|
|
UINT64 EntryPos;
|
|
|
|
UINT64 MaxSize;
|
|
|
|
DISK_CACHE *DiskCache;
|
|
|
|
VOID *PageAddress;
|
|
|
|
UINT8 PageAlignment;
|
|
|
|
|
|
|
|
DiskCache = &Volume->DiskCache[DataType];
|
|
|
|
PageNo = CacheTag->PageNo;
|
|
|
|
GroupNo = PageNo & DiskCache->GroupMask;
|
|
|
|
PageAlignment = DiskCache->PageAlignment;
|
|
|
|
PageAddress = DiskCache->CacheBase + (GroupNo << PageAlignment);
|
|
|
|
EntryPos = DiskCache->BaseAddress + LShiftU64 (PageNo, PageAlignment);
|
|
|
|
RealSize = CacheTag->RealSize;
|
2016-12-08 06:18:08 +01:00
|
|
|
if (IoMode == ReadDisk) {
|
2021-12-05 23:53:58 +01:00
|
|
|
RealSize = (UINTN)1 << PageAlignment;
|
|
|
|
MaxSize = DiskCache->LimitAddress - EntryPos;
|
2007-12-20 10:16:45 +01:00
|
|
|
if (MaxSize < RealSize) {
|
2021-11-17 04:21:22 +01:00
|
|
|
DEBUG ((DEBUG_INFO, "FatDiskIo: Cache Page OutBound occurred! \n"));
|
2021-12-05 23:53:58 +01:00
|
|
|
RealSize = (UINTN)MaxSize;
|
2007-12-20 10:16:45 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
WriteCount = 1;
|
2021-12-05 23:53:58 +01:00
|
|
|
if ((DataType == CacheFat) && (IoMode == WriteDisk)) {
|
2007-12-20 10:16:45 +01:00
|
|
|
WriteCount = Volume->NumFats;
|
|
|
|
}
|
|
|
|
|
|
|
|
do {
|
|
|
|
//
|
|
|
|
// Only fat table writing will execute more than once
|
|
|
|
//
|
2013-10-30 04:13:16 +01:00
|
|
|
Status = FatDiskIo (Volume, IoMode, EntryPos, RealSize, PageAddress, Task);
|
2007-12-20 10:16:45 +01:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
EntryPos += Volume->FatSize;
|
|
|
|
} while (--WriteCount > 0);
|
|
|
|
|
2021-12-05 23:53:58 +01:00
|
|
|
CacheTag->Dirty = FALSE;
|
|
|
|
CacheTag->RealSize = RealSize;
|
2007-12-20 10:16:45 +01:00
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
/**
|
|
|
|
|
|
|
|
Get one cache page by specified PageNo.
|
|
|
|
|
|
|
|
@param Volume - FAT file system volume.
|
|
|
|
@param CacheDataType - The cache type: CACHE_FAT or CACHE_DATA.
|
|
|
|
@param PageNo - PageNo to match with the cache.
|
|
|
|
@param CacheTag - The Cache Tag for the current cache page.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS - Get the cache page successfully.
|
|
|
|
@return other - An error occurred when accessing data.
|
|
|
|
|
|
|
|
**/
|
2007-12-20 10:16:45 +01:00
|
|
|
STATIC
|
|
|
|
EFI_STATUS
|
|
|
|
FatGetCachePage (
|
2021-12-05 23:53:58 +01:00
|
|
|
IN FAT_VOLUME *Volume,
|
|
|
|
IN CACHE_DATA_TYPE CacheDataType,
|
|
|
|
IN UINTN PageNo,
|
|
|
|
IN CACHE_TAG *CacheTag
|
2007-12-20 10:16:45 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN OldPageNo;
|
|
|
|
|
|
|
|
OldPageNo = CacheTag->PageNo;
|
2021-12-05 23:53:58 +01:00
|
|
|
if ((CacheTag->RealSize > 0) && (OldPageNo == PageNo)) {
|
2007-12-20 10:16:45 +01:00
|
|
|
//
|
|
|
|
// Cache Hit occurred
|
|
|
|
//
|
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// Write dirty cache page back to disk
|
|
|
|
//
|
2021-12-05 23:53:58 +01:00
|
|
|
if ((CacheTag->RealSize > 0) && CacheTag->Dirty) {
|
2016-12-08 06:18:08 +01:00
|
|
|
Status = FatExchangeCachePage (Volume, CacheDataType, WriteDisk, CacheTag, NULL);
|
2007-12-20 10:16:45 +01:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
}
|
2021-12-05 23:53:58 +01:00
|
|
|
|
2007-12-20 10:16:45 +01:00
|
|
|
//
|
|
|
|
// Load new data from disk;
|
|
|
|
//
|
2021-12-05 23:53:58 +01:00
|
|
|
CacheTag->PageNo = PageNo;
|
|
|
|
Status = FatExchangeCachePage (Volume, CacheDataType, ReadDisk, CacheTag, NULL);
|
2007-12-20 10:16:45 +01:00
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
/**
|
|
|
|
|
|
|
|
Read Length bytes from the position of Offset into Buffer, or
|
|
|
|
write Length bytes from Buffer into the position of Offset.
|
|
|
|
|
|
|
|
@param Volume - FAT file system volume.
|
|
|
|
@param CacheDataType - The type of cache: CACHE_DATA or CACHE_FAT.
|
|
|
|
@param IoMode - Indicate the type of disk access.
|
|
|
|
@param PageNo - The number of unaligned cache page.
|
|
|
|
@param Offset - The starting byte of cache page.
|
|
|
|
@param Length - The number of bytes that is read or written
|
|
|
|
@param Buffer - Buffer containing cache data.
|
|
|
|
|
|
|
|
@retval EFI_SUCCESS - The data was accessed correctly.
|
|
|
|
@return Others - An error occurred when accessing unaligned cache page.
|
|
|
|
|
|
|
|
**/
|
2007-12-20 10:16:45 +01:00
|
|
|
STATIC
|
|
|
|
EFI_STATUS
|
|
|
|
FatAccessUnalignedCachePage (
|
2021-12-05 23:53:58 +01:00
|
|
|
IN FAT_VOLUME *Volume,
|
|
|
|
IN CACHE_DATA_TYPE CacheDataType,
|
|
|
|
IN IO_MODE IoMode,
|
|
|
|
IN UINTN PageNo,
|
|
|
|
IN UINTN Offset,
|
|
|
|
IN UINTN Length,
|
|
|
|
IN OUT VOID *Buffer
|
2007-12-20 10:16:45 +01:00
|
|
|
)
|
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
VOID *Source;
|
|
|
|
VOID *Destination;
|
|
|
|
DISK_CACHE *DiskCache;
|
|
|
|
CACHE_TAG *CacheTag;
|
|
|
|
UINTN GroupNo;
|
|
|
|
|
|
|
|
DiskCache = &Volume->DiskCache[CacheDataType];
|
|
|
|
GroupNo = PageNo & DiskCache->GroupMask;
|
|
|
|
CacheTag = &DiskCache->CacheTag[GroupNo];
|
|
|
|
Status = FatGetCachePage (Volume, CacheDataType, PageNo, CacheTag);
|
|
|
|
if (!EFI_ERROR (Status)) {
|
|
|
|
Source = DiskCache->CacheBase + (GroupNo << DiskCache->PageAlignment) + Offset;
|
|
|
|
Destination = Buffer;
|
2016-12-08 06:18:08 +01:00
|
|
|
if (IoMode != ReadDisk) {
|
2021-12-05 23:53:58 +01:00
|
|
|
CacheTag->Dirty = TRUE;
|
|
|
|
DiskCache->Dirty = TRUE;
|
|
|
|
Destination = Source;
|
|
|
|
Source = Buffer;
|
2007-12-20 10:16:45 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
CopyMem (Destination, Source, Length);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
/**
|
2007-12-20 10:16:45 +01:00
|
|
|
|
|
|
|
Read BufferSize bytes from the position of Offset into Buffer,
|
|
|
|
or write BufferSize bytes from Buffer into the position of Offset.
|
|
|
|
|
|
|
|
Base on the parameter of CACHE_DATA_TYPE, the data access will be divided into
|
|
|
|
the access of FAT cache (CACHE_FAT) and the access of Data cache (CACHE_DATA):
|
|
|
|
|
|
|
|
1. Access of FAT cache (CACHE_FAT): Access the data in the FAT cache, if there is cache
|
|
|
|
page hit, just return the cache page; else update the related cache page and return
|
|
|
|
the right cache page.
|
|
|
|
2. Access of Data cache (CACHE_DATA):
|
|
|
|
The access data will be divided into UnderRun data, Aligned data and OverRun data;
|
|
|
|
The UnderRun data and OverRun data will be accessed by the Data cache,
|
|
|
|
but the Aligned data will be accessed with disk directly.
|
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
@param Volume - FAT file system volume.
|
|
|
|
@param CacheDataType - The type of cache: CACHE_DATA or CACHE_FAT.
|
|
|
|
@param IoMode - Indicate the type of disk access.
|
|
|
|
@param Offset - The starting byte offset to read from.
|
|
|
|
@param BufferSize - Size of Buffer.
|
|
|
|
@param Buffer - Buffer containing cache data.
|
|
|
|
@param Task point to task instance.
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
@retval EFI_SUCCESS - The data was accessed correctly.
|
|
|
|
@retval EFI_MEDIA_CHANGED - The MediaId does not match the current device.
|
|
|
|
@return Others - An error occurred when accessing cache.
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
FatAccessCache (
|
2021-12-05 23:53:58 +01:00
|
|
|
IN FAT_VOLUME *Volume,
|
|
|
|
IN CACHE_DATA_TYPE CacheDataType,
|
|
|
|
IN IO_MODE IoMode,
|
|
|
|
IN UINT64 Offset,
|
|
|
|
IN UINTN BufferSize,
|
|
|
|
IN OUT UINT8 *Buffer,
|
|
|
|
IN FAT_TASK *Task
|
2016-12-09 03:07:49 +01:00
|
|
|
)
|
2007-12-20 10:16:45 +01:00
|
|
|
{
|
|
|
|
EFI_STATUS Status;
|
|
|
|
UINTN PageSize;
|
|
|
|
UINTN UnderRun;
|
|
|
|
UINTN OverRun;
|
|
|
|
UINTN AlignedSize;
|
|
|
|
UINTN Length;
|
|
|
|
UINTN PageNo;
|
|
|
|
UINTN AlignedPageCount;
|
|
|
|
UINTN OverRunPageNo;
|
|
|
|
DISK_CACHE *DiskCache;
|
|
|
|
UINT64 EntryPos;
|
|
|
|
UINT8 PageAlignment;
|
|
|
|
|
|
|
|
ASSERT (Volume->CacheBuffer != NULL);
|
|
|
|
|
|
|
|
Status = EFI_SUCCESS;
|
|
|
|
DiskCache = &Volume->DiskCache[CacheDataType];
|
|
|
|
EntryPos = Offset - DiskCache->BaseAddress;
|
|
|
|
PageAlignment = DiskCache->PageAlignment;
|
|
|
|
PageSize = (UINTN)1 << PageAlignment;
|
2021-12-05 23:53:58 +01:00
|
|
|
PageNo = (UINTN)RShiftU64 (EntryPos, PageAlignment);
|
|
|
|
UnderRun = ((UINTN)EntryPos) & (PageSize - 1);
|
2007-12-20 10:16:45 +01:00
|
|
|
|
|
|
|
if (UnderRun > 0) {
|
|
|
|
Length = PageSize - UnderRun;
|
|
|
|
if (Length > BufferSize) {
|
|
|
|
Length = BufferSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
Status = FatAccessUnalignedCachePage (Volume, CacheDataType, IoMode, PageNo, UnderRun, Length, Buffer);
|
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
Buffer += Length;
|
|
|
|
BufferSize -= Length;
|
|
|
|
PageNo++;
|
|
|
|
}
|
|
|
|
|
2021-12-05 23:53:58 +01:00
|
|
|
AlignedPageCount = BufferSize >> PageAlignment;
|
|
|
|
OverRunPageNo = PageNo + AlignedPageCount;
|
2007-12-20 10:16:45 +01:00
|
|
|
//
|
|
|
|
// The access of the Aligned data
|
|
|
|
//
|
|
|
|
if (AlignedPageCount > 0) {
|
|
|
|
//
|
|
|
|
// Accessing fat table cannot have alignment data
|
|
|
|
//
|
2016-12-08 06:18:08 +01:00
|
|
|
ASSERT (CacheDataType == CacheData);
|
2007-12-20 10:16:45 +01:00
|
|
|
|
|
|
|
EntryPos = Volume->RootPos + LShiftU64 (PageNo, PageAlignment);
|
|
|
|
AlignedSize = AlignedPageCount << PageAlignment;
|
2013-10-30 04:13:16 +01:00
|
|
|
Status = FatDiskIo (Volume, IoMode, EntryPos, AlignedSize, Buffer, Task);
|
2007-12-20 10:16:45 +01:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
2021-12-05 23:53:58 +01:00
|
|
|
|
2007-12-20 10:16:45 +01:00
|
|
|
//
|
|
|
|
// If these access data over laps the relative cache range, these cache pages need
|
|
|
|
// to be updated.
|
|
|
|
//
|
|
|
|
FatFlushDataCacheRange (Volume, IoMode, PageNo, OverRunPageNo, Buffer);
|
2021-12-05 23:53:58 +01:00
|
|
|
Buffer += AlignedSize;
|
|
|
|
BufferSize -= AlignedSize;
|
2007-12-20 10:16:45 +01:00
|
|
|
}
|
2021-12-05 23:53:58 +01:00
|
|
|
|
2007-12-20 10:16:45 +01:00
|
|
|
//
|
|
|
|
// The access of the OverRun data
|
|
|
|
//
|
|
|
|
OverRun = BufferSize;
|
|
|
|
if (OverRun > 0) {
|
|
|
|
//
|
|
|
|
// Last read is not a complete page
|
|
|
|
//
|
|
|
|
Status = FatAccessUnalignedCachePage (Volume, CacheDataType, IoMode, OverRunPageNo, 0, OverRun, Buffer);
|
|
|
|
}
|
|
|
|
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
/**
|
2007-12-20 10:16:45 +01:00
|
|
|
|
|
|
|
Flush all the dirty cache back, include the FAT cache and the Data cache.
|
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
@param Volume - FAT file system volume.
|
|
|
|
@param Task point to task instance.
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
@retval EFI_SUCCESS - Flush all the dirty cache back successfully
|
|
|
|
@return other - An error occurred when writing the data into the disk
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
FatVolumeFlushCache (
|
2021-12-05 23:53:58 +01:00
|
|
|
IN FAT_VOLUME *Volume,
|
|
|
|
IN FAT_TASK *Task
|
2016-12-09 03:07:49 +01:00
|
|
|
)
|
2007-12-20 10:16:45 +01:00
|
|
|
{
|
2021-12-05 23:53:58 +01:00
|
|
|
EFI_STATUS Status;
|
|
|
|
CACHE_DATA_TYPE CacheDataType;
|
|
|
|
UINTN GroupIndex;
|
|
|
|
UINTN GroupMask;
|
|
|
|
DISK_CACHE *DiskCache;
|
|
|
|
CACHE_TAG *CacheTag;
|
|
|
|
|
|
|
|
for (CacheDataType = (CACHE_DATA_TYPE)0; CacheDataType < CacheMaxType; CacheDataType++) {
|
2007-12-20 10:16:45 +01:00
|
|
|
DiskCache = &Volume->DiskCache[CacheDataType];
|
|
|
|
if (DiskCache->Dirty) {
|
|
|
|
//
|
|
|
|
// Data cache or fat cache is dirty, write the dirty data back
|
|
|
|
//
|
|
|
|
GroupMask = DiskCache->GroupMask;
|
|
|
|
for (GroupIndex = 0; GroupIndex <= GroupMask; GroupIndex++) {
|
|
|
|
CacheTag = &DiskCache->CacheTag[GroupIndex];
|
2021-12-05 23:53:58 +01:00
|
|
|
if ((CacheTag->RealSize > 0) && CacheTag->Dirty) {
|
2007-12-20 10:16:45 +01:00
|
|
|
//
|
|
|
|
// Write back all Dirty Data Cache Page to disk
|
|
|
|
//
|
2016-12-08 06:18:08 +01:00
|
|
|
Status = FatExchangeCachePage (Volume, CacheDataType, WriteDisk, CacheTag, Task);
|
2007-12-20 10:16:45 +01:00
|
|
|
if (EFI_ERROR (Status)) {
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
DiskCache->Dirty = FALSE;
|
|
|
|
}
|
|
|
|
}
|
2021-12-05 23:53:58 +01:00
|
|
|
|
2007-12-20 10:16:45 +01:00
|
|
|
//
|
|
|
|
// Flush the block device.
|
|
|
|
//
|
|
|
|
Status = Volume->BlockIo->FlushBlocks (Volume->BlockIo);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
/**
|
2007-12-20 10:16:45 +01:00
|
|
|
|
|
|
|
Initialize the disk cache according to Volume's FatType.
|
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
@param Volume - FAT file system volume.
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
@retval EFI_SUCCESS - The disk cache is successfully initialized.
|
|
|
|
@retval EFI_OUT_OF_RESOURCES - Not enough memory to allocate disk cache.
|
2007-12-20 10:16:45 +01:00
|
|
|
|
2016-12-09 03:07:49 +01:00
|
|
|
**/
|
|
|
|
EFI_STATUS
|
|
|
|
FatInitializeDiskCache (
|
2021-12-05 23:53:58 +01:00
|
|
|
IN FAT_VOLUME *Volume
|
2016-12-09 03:07:49 +01:00
|
|
|
)
|
2007-12-20 10:16:45 +01:00
|
|
|
{
|
|
|
|
DISK_CACHE *DiskCache;
|
|
|
|
UINTN FatCacheGroupCount;
|
|
|
|
UINTN DataCacheSize;
|
|
|
|
UINTN FatCacheSize;
|
|
|
|
UINT8 *CacheBuffer;
|
|
|
|
|
|
|
|
DiskCache = Volume->DiskCache;
|
|
|
|
//
|
|
|
|
// Configure the parameters of disk cache
|
|
|
|
//
|
2016-12-08 06:18:08 +01:00
|
|
|
if (Volume->FatType == Fat12) {
|
2021-12-05 23:53:58 +01:00
|
|
|
FatCacheGroupCount = FAT_FATCACHE_GROUP_MIN_COUNT;
|
2016-12-08 06:18:08 +01:00
|
|
|
DiskCache[CacheFat].PageAlignment = FAT_FATCACHE_PAGE_MIN_ALIGNMENT;
|
|
|
|
DiskCache[CacheData].PageAlignment = FAT_DATACACHE_PAGE_MIN_ALIGNMENT;
|
2007-12-20 10:16:45 +01:00
|
|
|
} else {
|
2021-12-05 23:53:58 +01:00
|
|
|
FatCacheGroupCount = FAT_FATCACHE_GROUP_MAX_COUNT;
|
2016-12-08 06:18:08 +01:00
|
|
|
DiskCache[CacheFat].PageAlignment = FAT_FATCACHE_PAGE_MAX_ALIGNMENT;
|
|
|
|
DiskCache[CacheData].PageAlignment = FAT_DATACACHE_PAGE_MAX_ALIGNMENT;
|
2007-12-20 10:16:45 +01:00
|
|
|
}
|
|
|
|
|
2021-12-05 23:53:58 +01:00
|
|
|
DiskCache[CacheData].GroupMask = FAT_DATACACHE_GROUP_COUNT - 1;
|
|
|
|
DiskCache[CacheData].BaseAddress = Volume->RootPos;
|
|
|
|
DiskCache[CacheData].LimitAddress = Volume->VolumeSize;
|
|
|
|
DiskCache[CacheFat].GroupMask = FatCacheGroupCount - 1;
|
|
|
|
DiskCache[CacheFat].BaseAddress = Volume->FatPos;
|
|
|
|
DiskCache[CacheFat].LimitAddress = Volume->FatPos + Volume->FatSize;
|
|
|
|
FatCacheSize = FatCacheGroupCount << DiskCache[CacheFat].PageAlignment;
|
|
|
|
DataCacheSize = FAT_DATACACHE_GROUP_COUNT << DiskCache[CacheData].PageAlignment;
|
2007-12-20 10:16:45 +01:00
|
|
|
//
|
|
|
|
// Allocate the Fat Cache buffer
|
|
|
|
//
|
2012-12-24 03:42:01 +01:00
|
|
|
CacheBuffer = AllocateZeroPool (FatCacheSize + DataCacheSize);
|
2007-12-20 10:16:45 +01:00
|
|
|
if (CacheBuffer == NULL) {
|
|
|
|
return EFI_OUT_OF_RESOURCES;
|
|
|
|
}
|
|
|
|
|
2021-12-05 23:53:58 +01:00
|
|
|
Volume->CacheBuffer = CacheBuffer;
|
2016-12-08 06:18:08 +01:00
|
|
|
DiskCache[CacheFat].CacheBase = CacheBuffer;
|
|
|
|
DiskCache[CacheData].CacheBase = CacheBuffer + FatCacheSize;
|
2007-12-20 10:16:45 +01:00
|
|
|
return EFI_SUCCESS;
|
|
|
|
}
|