Fix various typos in FatPkg/EnhancedFatDxe comments. Cc: Ray Ni <ray.ni@intel.com> Signed-off-by: Antoine Coeur <coeur@gmx.fr> Reviewed-by: Philippe Mathieu-Daude <philmd@redhat.com> Acked-by: Liming Gao <liming.gao@intel.com> Signed-off-by: Philippe Mathieu-Daude <philmd@redhat.com> Message-Id: <20200207010831.9046-8-philmd@redhat.com>
		
			
				
	
	
		
			1392 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1392 lines
		
	
	
		
			37 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /** @file
 | |
|   Functions for performing directory entry io.
 | |
| 
 | |
| Copyright (c) 2005 - 2015, Intel Corporation. All rights reserved.<BR>
 | |
| SPDX-License-Identifier: BSD-2-Clause-Patent
 | |
| 
 | |
| **/
 | |
| 
 | |
| #include "Fat.h"
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Get a directory entry from disk for the Ofile.
 | |
| 
 | |
|   @param  Parent                - The parent of the OFile which need to update.
 | |
|   @param  IoMode                - Indicate whether to read directory entry or write directory entry.
 | |
|   @param  EntryPos              - The position of the directory entry to be accessed.
 | |
|   @param  Entry                 - The directory entry read or written.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - Access the directory entry successfully.
 | |
|   @return other                 - An error occurred when reading the directory entry.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| EFI_STATUS
 | |
| FatAccessEntry (
 | |
|   IN     FAT_OFILE            *Parent,
 | |
|   IN     IO_MODE              IoMode,
 | |
|   IN     UINTN                EntryPos,
 | |
|   IN OUT VOID                 *Entry
 | |
|   )
 | |
| {
 | |
|   UINTN Position;
 | |
|   UINTN BufferSize;
 | |
| 
 | |
|   Position = EntryPos * sizeof (FAT_DIRECTORY_ENTRY);
 | |
|   if (Position >= Parent->FileSize) {
 | |
|     //
 | |
|     // End of directory
 | |
|     //
 | |
|     ASSERT (IoMode == ReadData);
 | |
|     ((FAT_DIRECTORY_ENTRY *) Entry)->FileName[0] = EMPTY_ENTRY_MARK;
 | |
|     ((FAT_DIRECTORY_ENTRY *) Entry)->Attributes  = 0;
 | |
|     return EFI_SUCCESS;
 | |
|   }
 | |
| 
 | |
|   BufferSize = sizeof (FAT_DIRECTORY_ENTRY);
 | |
|   return FatAccessOFile (Parent, IoMode, Position, &BufferSize, Entry, NULL);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Save the directory entry to disk.
 | |
| 
 | |
|   @param  OFile                 - The parent OFile which needs to update.
 | |
|   @param  DirEnt                - The directory entry to be saved.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - Store the directory entry successfully.
 | |
|   @return other                 - An error occurred when writing the directory entry.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FatStoreDirEnt (
 | |
|   IN FAT_OFILE            *OFile,
 | |
|   IN FAT_DIRENT           *DirEnt
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS        Status;
 | |
|   FAT_DIRECTORY_LFN LfnEntry;
 | |
|   UINTN             EntryPos;
 | |
|   CHAR16            *LfnBufferPointer;
 | |
|   CHAR16            LfnBuffer[MAX_LFN_ENTRIES * LFN_CHAR_TOTAL + 1];
 | |
|   UINT8             EntryCount;
 | |
|   UINT8             LfnOrdinal;
 | |
| 
 | |
|   EntryPos   = DirEnt->EntryPos;
 | |
|   EntryCount = DirEnt->EntryCount;
 | |
|   //
 | |
|   // Write directory entry
 | |
|   //
 | |
|   Status = FatAccessEntry (OFile, WriteData, EntryPos, &DirEnt->Entry);
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return Status;
 | |
|   }
 | |
| 
 | |
|   if (--EntryCount > 0) {
 | |
|     //
 | |
|     // Write LFN directory entry
 | |
|     //
 | |
|     SetMem (LfnBuffer, sizeof (CHAR16) * LFN_CHAR_TOTAL * EntryCount, 0xff);
 | |
|     Status = StrCpyS (
 | |
|                LfnBuffer,
 | |
|                ARRAY_SIZE (LfnBuffer),
 | |
|                DirEnt->FileString
 | |
|                );
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
| 
 | |
|     LfnBufferPointer    = LfnBuffer;
 | |
|     LfnEntry.Attributes = FAT_ATTRIBUTE_LFN;
 | |
|     LfnEntry.Type       = 0;
 | |
|     LfnEntry.MustBeZero = 0;
 | |
|     LfnEntry.Checksum   = FatCheckSum (DirEnt->Entry.FileName);
 | |
|     for (LfnOrdinal = 1; LfnOrdinal <= EntryCount; LfnOrdinal++) {
 | |
|       LfnEntry.Ordinal = LfnOrdinal;
 | |
|       if (LfnOrdinal == EntryCount) {
 | |
|         LfnEntry.Ordinal |= FAT_LFN_LAST;
 | |
|       }
 | |
| 
 | |
|       CopyMem (LfnEntry.Name1, LfnBufferPointer, sizeof (CHAR16) * LFN_CHAR1_LEN);
 | |
|       LfnBufferPointer += LFN_CHAR1_LEN;
 | |
|       CopyMem (LfnEntry.Name2, LfnBufferPointer, sizeof (CHAR16) * LFN_CHAR2_LEN);
 | |
|       LfnBufferPointer += LFN_CHAR2_LEN;
 | |
|       CopyMem (LfnEntry.Name3, LfnBufferPointer, sizeof (CHAR16) * LFN_CHAR3_LEN);
 | |
|       LfnBufferPointer += LFN_CHAR3_LEN;
 | |
|       EntryPos--;
 | |
|       if (DirEnt->Invalid) {
 | |
|         LfnEntry.Ordinal = DELETE_ENTRY_MARK;
 | |
|       }
 | |
| 
 | |
|       Status = FatAccessEntry (OFile, WriteData, EntryPos, &LfnEntry);
 | |
|       if (EFI_ERROR (Status)) {
 | |
|         return Status;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Determine whether the directory entry is "." or ".." entry.
 | |
| 
 | |
|   @param  DirEnt               - The corresponding directory entry.
 | |
| 
 | |
|   @retval TRUE                 - The directory entry is "." or ".." directory entry
 | |
|   @retval FALSE                - The directory entry is not "." or ".." directory entry
 | |
| 
 | |
| **/
 | |
| BOOLEAN
 | |
| FatIsDotDirEnt (
 | |
|   IN FAT_DIRENT  *DirEnt
 | |
|   )
 | |
| {
 | |
|   CHAR16  *FileString;
 | |
|   FileString = DirEnt->FileString;
 | |
|   if (StrCmp (FileString, L".") == 0 || StrCmp (FileString, L"..") == 0) {
 | |
|     return TRUE;
 | |
|   }
 | |
| 
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Set the OFile's cluster info in its directory entry.
 | |
| 
 | |
|   @param  OFile                 - The corresponding OFile.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| VOID
 | |
| FatSetDirEntCluster (
 | |
|   IN FAT_OFILE    *OFile
 | |
|   )
 | |
| {
 | |
|   UINTN       Cluster;
 | |
|   FAT_DIRENT  *DirEnt;
 | |
| 
 | |
|   DirEnt                        = OFile->DirEnt;
 | |
|   Cluster                       = OFile->FileCluster;
 | |
|   DirEnt->Entry.FileClusterHigh = (UINT16) (Cluster >> 16);
 | |
|   DirEnt->Entry.FileCluster     = (UINT16) Cluster;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Set the OFile's cluster and size info in its directory entry.
 | |
| 
 | |
|   @param  OFile                 - The corresponding OFile.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| FatUpdateDirEntClusterSizeInfo (
 | |
|   IN FAT_OFILE    *OFile
 | |
|   )
 | |
| {
 | |
|   ASSERT (OFile->ODir == NULL);
 | |
|   OFile->DirEnt->Entry.FileSize = (UINT32) OFile->FileSize;
 | |
|   FatSetDirEntCluster (OFile);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Copy all the information of DirEnt2 to DirEnt1 except for 8.3 name.
 | |
| 
 | |
|   @param  DirEnt1               - The destination directory entry.
 | |
|   @param  DirEnt2               - The source directory entry.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| FatCloneDirEnt (
 | |
|   IN  FAT_DIRENT          *DirEnt1,
 | |
|   IN  FAT_DIRENT          *DirEnt2
 | |
|   )
 | |
| {
 | |
|   UINT8 *Entry1;
 | |
|   UINT8 *Entry2;
 | |
|   Entry1  = (UINT8 *) &DirEnt1->Entry;
 | |
|   Entry2  = (UINT8 *) &DirEnt2->Entry;
 | |
|   CopyMem (
 | |
|     Entry1 + FAT_ENTRY_INFO_OFFSET,
 | |
|     Entry2 + FAT_ENTRY_INFO_OFFSET,
 | |
|     sizeof (FAT_DIRECTORY_ENTRY) - FAT_ENTRY_INFO_OFFSET
 | |
|     );
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Get the LFN for the directory entry.
 | |
| 
 | |
|   @param  Parent                - The parent directory.
 | |
|   @param  DirEnt                - The directory entry to get LFN.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| VOID
 | |
| FatLoadLongNameEntry (
 | |
|   IN FAT_OFILE           *Parent,
 | |
|   IN FAT_DIRENT          *DirEnt
 | |
|   )
 | |
| {
 | |
|   CHAR16            LfnBuffer[MAX_LFN_ENTRIES * LFN_CHAR_TOTAL + 1];
 | |
|   CHAR16            *LfnBufferPointer;
 | |
|   CHAR8             *File8Dot3Name;
 | |
|   UINTN             EntryPos;
 | |
|   UINT8             LfnOrdinal;
 | |
|   UINT8             LfnChecksum;
 | |
|   FAT_DIRECTORY_LFN LfnEntry;
 | |
|   EFI_STATUS        Status;
 | |
| 
 | |
|   EntryPos          = DirEnt->EntryPos;
 | |
|   File8Dot3Name     = DirEnt->Entry.FileName;
 | |
|   LfnBufferPointer  = LfnBuffer;
 | |
|   //
 | |
|   // Computes checksum for LFN
 | |
|   //
 | |
|   LfnChecksum = FatCheckSum (File8Dot3Name);
 | |
|   LfnOrdinal  = 1;
 | |
|   do {
 | |
|     if (EntryPos == 0) {
 | |
|       LfnBufferPointer = LfnBuffer;
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     EntryPos--;
 | |
|     Status = FatAccessEntry (Parent, ReadData, EntryPos, &LfnEntry);
 | |
|     if (EFI_ERROR (Status) ||
 | |
|         LfnEntry.Attributes != FAT_ATTRIBUTE_LFN ||
 | |
|         LfnEntry.MustBeZero != 0 ||
 | |
|         LfnEntry.Checksum != LfnChecksum ||
 | |
|         (LfnEntry.Ordinal & (~FAT_LFN_LAST)) != LfnOrdinal ||
 | |
|         LfnOrdinal > MAX_LFN_ENTRIES
 | |
|         ) {
 | |
|       //
 | |
|       // The directory entry does not have a long file name or
 | |
|       // some error occurs when loading long file name for a directory entry,
 | |
|       // and then we load the long name from short name
 | |
|       //
 | |
|       LfnBufferPointer = LfnBuffer;
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     CopyMem (LfnBufferPointer, LfnEntry.Name1, sizeof (CHAR16) * LFN_CHAR1_LEN);
 | |
|     LfnBufferPointer += LFN_CHAR1_LEN;
 | |
|     CopyMem (LfnBufferPointer, LfnEntry.Name2, sizeof (CHAR16) * LFN_CHAR2_LEN);
 | |
|     LfnBufferPointer += LFN_CHAR2_LEN;
 | |
|     CopyMem (LfnBufferPointer, LfnEntry.Name3, sizeof (CHAR16) * LFN_CHAR3_LEN);
 | |
|     LfnBufferPointer += LFN_CHAR3_LEN;
 | |
|     LfnOrdinal++;
 | |
|   } while ((LfnEntry.Ordinal & FAT_LFN_LAST) == 0);
 | |
|   DirEnt->EntryCount = LfnOrdinal;
 | |
|   //
 | |
|   // Terminate current Lfnbuffer
 | |
|   //
 | |
|   *LfnBufferPointer = 0;
 | |
|   if (LfnBufferPointer == LfnBuffer) {
 | |
|     //
 | |
|     // Fail to get the long file name from long file name entry,
 | |
|     // get the file name from short name
 | |
|     //
 | |
|     FatGetFileNameViaCaseFlag (
 | |
|       DirEnt,
 | |
|       LfnBuffer,
 | |
|       ARRAY_SIZE (LfnBuffer)
 | |
|       );
 | |
|   }
 | |
| 
 | |
|   DirEnt->FileString = AllocateCopyPool (StrSize (LfnBuffer), LfnBuffer);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Add this directory entry node to the list of directory entries and hash table.
 | |
| 
 | |
|   @param  ODir                  - The parent OFile which needs to be updated.
 | |
|   @param  DirEnt                - The directory entry to be added.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| VOID
 | |
| FatAddDirEnt (
 | |
|   IN FAT_ODIR             *ODir,
 | |
|   IN FAT_DIRENT           *DirEnt
 | |
|   )
 | |
| {
 | |
|   if (DirEnt->Link.BackLink == NULL) {
 | |
|     DirEnt->Link.BackLink = &ODir->ChildList;
 | |
|   }
 | |
|   InsertTailList (DirEnt->Link.BackLink, &DirEnt->Link);
 | |
|   FatInsertToHashTable (ODir, DirEnt);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Load from disk the next directory entry at current end of directory position.
 | |
| 
 | |
|   @param  OFile                 - The parent OFile.
 | |
|   @param  PtrDirEnt             - The directory entry that is loaded.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - Load the directory entry successfully.
 | |
|   @retval EFI_OUT_OF_RESOURCES  - Out of resource.
 | |
|   @return other                 - An error occurred when reading the directory entries.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| EFI_STATUS
 | |
| FatLoadNextDirEnt (
 | |
|   IN  FAT_OFILE           *OFile,
 | |
|   OUT FAT_DIRENT          **PtrDirEnt
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS          Status;
 | |
|   FAT_DIRENT          *DirEnt;
 | |
|   FAT_ODIR            *ODir;
 | |
|   FAT_DIRECTORY_ENTRY Entry;
 | |
| 
 | |
|   ODir = OFile->ODir;
 | |
|   //
 | |
|   // Make sure the parent's directory has been opened
 | |
|   //
 | |
|   ASSERT (ODir != NULL);
 | |
|   //
 | |
|   // Assert we have not reached the end of directory
 | |
|   //
 | |
|   ASSERT (!ODir->EndOfDir);
 | |
|   DirEnt = NULL;
 | |
| 
 | |
|   for (;;) {
 | |
|     //
 | |
|     // Read the next directory entry until we find a valid directory entry (excluding lfn entry)
 | |
|     //
 | |
|     Status = FatAccessEntry (OFile, ReadData, ODir->CurrentEndPos, &Entry);
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
| 
 | |
|     if (((UINT8) Entry.FileName[0] != DELETE_ENTRY_MARK) && (Entry.Attributes & FAT_ATTRIBUTE_VOLUME_ID) == 0) {
 | |
|       //
 | |
|       // We get a valid directory entry, then handle it
 | |
|       //
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     ODir->CurrentEndPos++;
 | |
|   }
 | |
| 
 | |
|   if (Entry.FileName[0] != EMPTY_ENTRY_MARK) {
 | |
|     //
 | |
|     // Although FAT spec states this field is always 0 for FAT12 & FAT16, some applications
 | |
|     // might use it for some special usage, it is safer to zero it in memory for FAT12 & FAT16.
 | |
|     //
 | |
|     if (OFile->Volume->FatType != Fat32) {
 | |
|       Entry.FileClusterHigh = 0;
 | |
|     }
 | |
| 
 | |
|     //
 | |
|     // This is a valid directory entry
 | |
|     //
 | |
|     DirEnt = AllocateZeroPool (sizeof (FAT_DIRENT));
 | |
|     if (DirEnt == NULL) {
 | |
|       return EFI_OUT_OF_RESOURCES;
 | |
|     }
 | |
| 
 | |
|     DirEnt->Signature = FAT_DIRENT_SIGNATURE;
 | |
|     //
 | |
|     // Remember the directory's entry position on disk
 | |
|     //
 | |
|     DirEnt->EntryPos = (UINT16) ODir->CurrentEndPos;
 | |
|     CopyMem (&DirEnt->Entry, &Entry, sizeof (FAT_DIRECTORY_ENTRY));
 | |
|     FatLoadLongNameEntry (OFile, DirEnt);
 | |
|     if (DirEnt->FileString == NULL) {
 | |
|       Status = EFI_OUT_OF_RESOURCES;
 | |
|       goto Done;
 | |
|     }
 | |
|     //
 | |
|     // Add this directory entry to directory
 | |
|     //
 | |
|     FatAddDirEnt (ODir, DirEnt);
 | |
|     //
 | |
|     // Point to next directory entry
 | |
|     //
 | |
|     ODir->CurrentEndPos++;
 | |
|   } else {
 | |
|     ODir->EndOfDir = TRUE;
 | |
|   }
 | |
| 
 | |
|   *PtrDirEnt = DirEnt;
 | |
|   return EFI_SUCCESS;
 | |
| 
 | |
| Done:
 | |
|   FatFreeDirEnt (DirEnt);
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Get the directory entry's info into Buffer.
 | |
| 
 | |
|   @param  Volume                - FAT file system volume.
 | |
|   @param  DirEnt                - The corresponding directory entry.
 | |
|   @param  BufferSize            - Size of Buffer.
 | |
|   @param  Buffer                - Buffer containing file info.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - Get the file info successfully.
 | |
|   @retval EFI_BUFFER_TOO_SMALL  - The buffer is too small.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FatGetDirEntInfo (
 | |
|   IN     FAT_VOLUME         *Volume,
 | |
|   IN     FAT_DIRENT         *DirEnt,
 | |
|   IN OUT UINTN              *BufferSize,
 | |
|      OUT VOID               *Buffer
 | |
|   )
 | |
| {
 | |
|   UINTN               Size;
 | |
|   UINTN               NameSize;
 | |
|   UINTN               ResultSize;
 | |
|   UINTN               Cluster;
 | |
|   EFI_STATUS          Status;
 | |
|   EFI_FILE_INFO       *Info;
 | |
|   FAT_DIRECTORY_ENTRY *Entry;
 | |
|   FAT_DATE_TIME       FatLastAccess;
 | |
| 
 | |
|   ASSERT_VOLUME_LOCKED (Volume);
 | |
| 
 | |
|   Size        = SIZE_OF_EFI_FILE_INFO;
 | |
|   NameSize    = StrSize (DirEnt->FileString);
 | |
|   ResultSize  = Size + NameSize;
 | |
| 
 | |
|   Status      = EFI_BUFFER_TOO_SMALL;
 | |
|   if (*BufferSize >= ResultSize) {
 | |
|     Status      = EFI_SUCCESS;
 | |
|     Entry       = &DirEnt->Entry;
 | |
|     Info        = Buffer;
 | |
|     Info->Size  = ResultSize;
 | |
|     if ((Entry->Attributes & FAT_ATTRIBUTE_DIRECTORY) != 0) {
 | |
|       Cluster             = (Entry->FileClusterHigh << 16) | Entry->FileCluster;
 | |
|       Info->PhysicalSize  = FatPhysicalDirSize (Volume, Cluster);
 | |
|       Info->FileSize      = Info->PhysicalSize;
 | |
|     } else {
 | |
|       Info->FileSize      = Entry->FileSize;
 | |
|       Info->PhysicalSize  = FatPhysicalFileSize (Volume, Entry->FileSize);
 | |
|     }
 | |
| 
 | |
|     ZeroMem (&FatLastAccess.Time, sizeof (FatLastAccess.Time));
 | |
|     CopyMem (&FatLastAccess.Date, &Entry->FileLastAccess, sizeof (FatLastAccess.Date));
 | |
|     FatFatTimeToEfiTime (&FatLastAccess, &Info->LastAccessTime);
 | |
|     FatFatTimeToEfiTime (&Entry->FileCreateTime, &Info->CreateTime);
 | |
|     FatFatTimeToEfiTime (&Entry->FileModificationTime, &Info->ModificationTime);
 | |
|     Info->Attribute = Entry->Attributes & EFI_FILE_VALID_ATTR;
 | |
|     CopyMem ((CHAR8 *) Buffer + Size, DirEnt->FileString, NameSize);
 | |
|   }
 | |
| 
 | |
|   *BufferSize = ResultSize;
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Search the directory for the directory entry whose filename is FileNameString.
 | |
| 
 | |
|   @param  OFile                 - The parent OFile whose directory is to be searched.
 | |
|   @param  FileNameString        - The filename to be searched.
 | |
|   @param  PtrDirEnt             - pointer to the directory entry if found.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - Find the directory entry or not found.
 | |
|   @return other                 - An error occurred when reading the directory entries.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| EFI_STATUS
 | |
| FatSearchODir (
 | |
|   IN  FAT_OFILE      *OFile,
 | |
|   IN  CHAR16         *FileNameString,
 | |
|   OUT FAT_DIRENT     **PtrDirEnt
 | |
|   )
 | |
| {
 | |
|   BOOLEAN     PossibleShortName;
 | |
|   CHAR8       File8Dot3Name[FAT_NAME_LEN];
 | |
|   FAT_ODIR    *ODir;
 | |
|   FAT_DIRENT  *DirEnt;
 | |
|   EFI_STATUS  Status;
 | |
| 
 | |
|   ODir = OFile->ODir;
 | |
|   ASSERT (ODir != NULL);
 | |
|   //
 | |
|   // Check if the file name is a valid short name
 | |
|   //
 | |
|   PossibleShortName = FatCheckIs8Dot3Name (FileNameString, File8Dot3Name);
 | |
|   //
 | |
|   // Search the hash table first
 | |
|   //
 | |
|   DirEnt = *FatLongNameHashSearch (ODir, FileNameString);
 | |
|   if (DirEnt == NULL && PossibleShortName) {
 | |
|       DirEnt = *FatShortNameHashSearch (ODir, File8Dot3Name);
 | |
|   }
 | |
|   if (DirEnt == NULL) {
 | |
|     //
 | |
|     // We fail to get the directory entry from hash table; we then
 | |
|     // search the rest directory
 | |
|     //
 | |
|     while (!ODir->EndOfDir) {
 | |
|       Status = FatLoadNextDirEnt (OFile, &DirEnt);
 | |
|       if (EFI_ERROR (Status)) {
 | |
|         return Status;
 | |
|       }
 | |
| 
 | |
|       if (DirEnt != NULL) {
 | |
|         if (FatStriCmp (FileNameString, DirEnt->FileString) == 0) {
 | |
|           break;
 | |
|         }
 | |
| 
 | |
|         if (PossibleShortName && CompareMem (File8Dot3Name, DirEnt->Entry.FileName, FAT_NAME_LEN) == 0) {
 | |
|           break;
 | |
|         }
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   *PtrDirEnt = DirEnt;
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Set the OFile's current directory cursor to the list head.
 | |
| 
 | |
|   @param OFile                 - The directory OFile whose directory cursor is reset.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| FatResetODirCursor (
 | |
|   IN FAT_OFILE    *OFile
 | |
|   )
 | |
| {
 | |
|   FAT_ODIR  *ODir;
 | |
| 
 | |
|   ODir = OFile->ODir;
 | |
|   ASSERT (ODir != NULL);
 | |
|   ODir->CurrentCursor = &(ODir->ChildList);
 | |
|   ODir->CurrentPos    = 0;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Set the directory's cursor to the next and get the next directory entry.
 | |
| 
 | |
|   @param  OFile                 - The parent OFile.
 | |
|   @param PtrDirEnt             - The next directory entry.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - We get the next directory entry successfully.
 | |
|   @return other                 - An error occurred when get next directory entry.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FatGetNextDirEnt (
 | |
|   IN  FAT_OFILE     *OFile,
 | |
|   OUT FAT_DIRENT    **PtrDirEnt
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
|   FAT_DIRENT  *DirEnt;
 | |
|   FAT_ODIR    *ODir;
 | |
| 
 | |
|   ODir = OFile->ODir;
 | |
|   ASSERT (ODir != NULL);
 | |
|   if (ODir->CurrentCursor->ForwardLink == &ODir->ChildList) {
 | |
|     //
 | |
|     // End of directory, we will try one more time
 | |
|     //
 | |
|     if (!ODir->EndOfDir) {
 | |
|       //
 | |
|       // Read directory from disk
 | |
|       //
 | |
|       Status = FatLoadNextDirEnt (OFile, &DirEnt);
 | |
|       if (EFI_ERROR (Status)) {
 | |
|         return Status;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (ODir->CurrentCursor->ForwardLink == &ODir->ChildList) {
 | |
|     //
 | |
|     // End of directory, return NULL
 | |
|     //
 | |
|     DirEnt              = NULL;
 | |
|     ODir->CurrentPos    = ODir->CurrentEndPos;
 | |
|   } else {
 | |
|     ODir->CurrentCursor = ODir->CurrentCursor->ForwardLink;
 | |
|     DirEnt              = DIRENT_FROM_LINK (ODir->CurrentCursor);
 | |
|     ODir->CurrentPos    = DirEnt->EntryPos + 1;
 | |
|   }
 | |
| 
 | |
|   *PtrDirEnt = DirEnt;
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Set the directory entry count according to the filename.
 | |
| 
 | |
|   @param  OFile                 - The corresponding OFile.
 | |
|   @param  DirEnt                - The directory entry to be set.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| VOID
 | |
| FatSetEntryCount (
 | |
|   IN FAT_OFILE    *OFile,
 | |
|   IN FAT_DIRENT   *DirEnt
 | |
|   )
 | |
| {
 | |
|   CHAR16  *FileString;
 | |
|   CHAR8   *File8Dot3Name;
 | |
| 
 | |
|   //
 | |
|   // Get new entry count and set the 8.3 name
 | |
|   //
 | |
|   DirEnt->EntryCount  = 1;
 | |
|   FileString          = DirEnt->FileString;
 | |
|   File8Dot3Name       = DirEnt->Entry.FileName;
 | |
|   SetMem (File8Dot3Name, FAT_NAME_LEN, ' ');
 | |
|   if (StrCmp (FileString, L".") == 0) {
 | |
|     //
 | |
|     // "." entry
 | |
|     //
 | |
|     File8Dot3Name[0] = '.';
 | |
|     FatCloneDirEnt (DirEnt, OFile->DirEnt);
 | |
|   } else if (StrCmp (FileString, L"..") == 0) {
 | |
|     //
 | |
|     // ".." entry
 | |
|     //
 | |
|     File8Dot3Name[0]  = '.';
 | |
|     File8Dot3Name[1]  = '.';
 | |
|     FatCloneDirEnt (DirEnt, OFile->Parent->DirEnt);
 | |
|   } else {
 | |
|     //
 | |
|     // Normal name
 | |
|     //
 | |
|     if (FatCheckIs8Dot3Name (FileString, File8Dot3Name)) {
 | |
|       //
 | |
|       // This file name is a valid 8.3 file name, we need to further check its case flag
 | |
|       //
 | |
|       FatSetCaseFlag (DirEnt);
 | |
|     } else {
 | |
|       //
 | |
|       // The file name is not a valid 8.3 name we need to generate an 8.3 name for it
 | |
|       //
 | |
|       FatCreate8Dot3Name (OFile, DirEnt);
 | |
|       DirEnt->EntryCount = (UINT8)(LFN_ENTRY_NUMBER (StrLen (FileString)) + DirEnt->EntryCount);
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Append a zero cluster to the current OFile.
 | |
| 
 | |
|   @param  OFile        - The directory OFile which needs to be updated.
 | |
| 
 | |
|   @retval EFI_SUCCESS  - Append a zero cluster to the OFile successfully.
 | |
|   @return other        - An error occurred when appending the zero cluster.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| EFI_STATUS
 | |
| FatExpandODir (
 | |
|   IN FAT_OFILE  *OFile
 | |
|   )
 | |
| {
 | |
|   return FatExpandOFile (OFile, OFile->FileSize + OFile->Volume->ClusterSize);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Search the Root OFile for the possible volume label.
 | |
| 
 | |
|   @param  Root                  - The Root OFile.
 | |
|   @param  DirEnt                - The returned directory entry of volume label.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - The search process is completed successfully.
 | |
|   @return other                 - An error occurred when searching volume label.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| EFI_STATUS
 | |
| FatSeekVolumeId (
 | |
|   IN  FAT_OFILE            *Root,
 | |
|   OUT FAT_DIRENT           *DirEnt
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS          Status;
 | |
|   UINTN               EntryPos;
 | |
|   FAT_DIRECTORY_ENTRY *Entry;
 | |
| 
 | |
|   EntryPos        = 0;
 | |
|   Entry           = &DirEnt->Entry;
 | |
|   DirEnt->Invalid = TRUE;
 | |
|   do {
 | |
|     Status = FatAccessEntry (Root, ReadData, EntryPos, Entry);
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
| 
 | |
|     if (((UINT8) Entry->FileName[0] != DELETE_ENTRY_MARK) && (((Entry->Attributes) & (~FAT_ATTRIBUTE_ARCHIVE)) == FAT_ATTRIBUTE_VOLUME_ID)) {
 | |
|       DirEnt->EntryPos   = (UINT16) EntryPos;
 | |
|       DirEnt->EntryCount = 1;
 | |
|       DirEnt->Invalid    = FALSE;
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     EntryPos++;
 | |
|   } while (Entry->FileName[0] != EMPTY_ENTRY_MARK);
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Use First Fit Algorithm to insert directory entry.
 | |
|   Only this function will erase "E5" entries in a directory.
 | |
|   In view of safest recovery, this function will only be triggered
 | |
|   when maximum directory entry number has reached.
 | |
| 
 | |
|   @param  OFile                 - The corresponding OFile.
 | |
|   @param  DirEnt                - The directory entry to be inserted.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - The directory entry has been successfully inserted.
 | |
|   @retval EFI_VOLUME_FULL       - The directory can not hold more directory entries.
 | |
|   @return Others                - Some error occurred when inserting new directory entries.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| EFI_STATUS
 | |
| FatFirstFitInsertDirEnt (
 | |
|   IN FAT_OFILE    *OFile,
 | |
|   IN FAT_DIRENT   *DirEnt
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS      Status;
 | |
|   FAT_ODIR        *ODir;
 | |
|   LIST_ENTRY      *CurrentEntry;
 | |
|   FAT_DIRENT      *CurrentDirEnt;
 | |
|   UINT32          CurrentPos;
 | |
|   UINT32          LabelPos;
 | |
|   UINT32          NewEntryPos;
 | |
|   UINT16          EntryCount;
 | |
|   FAT_DIRENT      LabelDirEnt;
 | |
| 
 | |
|   LabelPos = 0;
 | |
|   if (OFile->Parent == NULL) {
 | |
|     Status = FatSeekVolumeId (OFile, &LabelDirEnt);
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
| 
 | |
|     if (!LabelDirEnt.Invalid) {
 | |
|       LabelPos = LabelDirEnt.EntryPos;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   EntryCount  = DirEnt->EntryCount;
 | |
|   NewEntryPos = EntryCount;
 | |
|   CurrentPos  = 0;
 | |
|   ODir        = OFile->ODir;
 | |
|   for (CurrentEntry = ODir->ChildList.ForwardLink;
 | |
|        CurrentEntry != &ODir->ChildList;
 | |
|        CurrentEntry = CurrentEntry->ForwardLink
 | |
|       ) {
 | |
|     CurrentDirEnt = DIRENT_FROM_LINK (CurrentEntry);
 | |
|     if (NewEntryPos + CurrentDirEnt->EntryCount <= CurrentDirEnt->EntryPos) {
 | |
|       if (LabelPos > NewEntryPos || LabelPos <= CurrentPos) {
 | |
|         //
 | |
|         // first fit succeeded
 | |
|         //
 | |
|         goto Done;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     CurrentPos  = CurrentDirEnt->EntryPos;
 | |
|     NewEntryPos = CurrentPos + EntryCount;
 | |
|   }
 | |
| 
 | |
|   if (NewEntryPos >= ODir->CurrentEndPos) {
 | |
|     return EFI_VOLUME_FULL;
 | |
|   }
 | |
| 
 | |
| Done:
 | |
|   DirEnt->EntryPos   = (UINT16) NewEntryPos;
 | |
|   DirEnt->Link.BackLink = CurrentEntry;
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Find the new directory entry position for the directory entry.
 | |
| 
 | |
|   @param  OFile                 - The corresponding OFile.
 | |
|   @param  DirEnt                - The directory entry whose new position is to be set.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - The new directory entry position is successfully found.
 | |
|   @retval EFI_VOLUME_FULL       - The directory has reach its maximum capacity.
 | |
|   @return other                 - An error occurred when reading the directory entry.
 | |
| 
 | |
| **/
 | |
| STATIC
 | |
| EFI_STATUS
 | |
| FatNewEntryPos (
 | |
|   IN FAT_OFILE    *OFile,
 | |
|   IN FAT_DIRENT   *DirEnt
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
|   FAT_ODIR    *ODir;
 | |
|   FAT_DIRENT  *TempDirEnt;
 | |
|   UINT32      NewEndPos;
 | |
| 
 | |
|   ODir = OFile->ODir;
 | |
|   ASSERT (ODir != NULL);
 | |
|   //
 | |
|   // Make sure the whole directory has been loaded
 | |
|   //
 | |
|   while (!ODir->EndOfDir) {
 | |
|     Status = FatLoadNextDirEnt (OFile, &TempDirEnt);
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
|   }
 | |
|   //
 | |
|   // We will append this entry to the end of directory
 | |
|   //
 | |
|   FatGetCurrentFatTime (&DirEnt->Entry.FileCreateTime);
 | |
|   CopyMem (&DirEnt->Entry.FileModificationTime, &DirEnt->Entry.FileCreateTime, sizeof (FAT_DATE_TIME));
 | |
|   CopyMem (&DirEnt->Entry.FileLastAccess, &DirEnt->Entry.FileCreateTime.Date, sizeof (FAT_DATE));
 | |
|   NewEndPos = ODir->CurrentEndPos + DirEnt->EntryCount;
 | |
|   if (NewEndPos * sizeof (FAT_DIRECTORY_ENTRY) > OFile->FileSize) {
 | |
|     if (NewEndPos >= (OFile->IsFixedRootDir ? OFile->Volume->RootEntries : FAT_MAX_DIRENTRY_COUNT)) {
 | |
|       //
 | |
|       // We try to use fist fit algorithm to insert this directory entry
 | |
|       //
 | |
|       return FatFirstFitInsertDirEnt (OFile, DirEnt);
 | |
|     }
 | |
|     //
 | |
|     // We should allocate a new cluster for this directory
 | |
|     //
 | |
|     Status = FatExpandODir (OFile);
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
|   }
 | |
|   //
 | |
|   // We append our directory entry at the end of directory file
 | |
|   //
 | |
|   ODir->CurrentEndPos = NewEndPos;
 | |
|   DirEnt->EntryPos = (UINT16) (ODir->CurrentEndPos - 1);
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Get the directory entry for the volume.
 | |
| 
 | |
|   @param  Volume                - FAT file system volume.
 | |
|   @param  Name                  - The file name of the volume.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - Update the volume with the directory entry successfully.
 | |
|   @return others                - An error occurred when getting volume label.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FatGetVolumeEntry (
 | |
|   IN FAT_VOLUME           *Volume,
 | |
|   IN CHAR16               *Name
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
|   FAT_DIRENT  LabelDirEnt;
 | |
| 
 | |
|   *Name   = 0;
 | |
|   Status  = FatSeekVolumeId (Volume->Root, &LabelDirEnt);
 | |
|   if (!EFI_ERROR (Status)) {
 | |
|     if (!LabelDirEnt.Invalid) {
 | |
|       FatNameToStr (LabelDirEnt.Entry.FileName, FAT_NAME_LEN, FALSE, Name);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Set the relevant directory entry into disk for the volume.
 | |
| 
 | |
|   @param  Volume              - FAT file system volume.
 | |
|   @param  Name                - The new file name of the volume.
 | |
| 
 | |
|   @retval EFI_SUCCESS         - Update the Volume successfully.
 | |
|   @retval EFI_UNSUPPORTED     - The input label is not a valid volume label.
 | |
|   @return other               - An error occurred when setting volume label.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FatSetVolumeEntry (
 | |
|   IN FAT_VOLUME           *Volume,
 | |
|   IN CHAR16               *Name
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
|   FAT_DIRENT  LabelDirEnt;
 | |
|   FAT_OFILE   *Root;
 | |
| 
 | |
|   Root    = Volume->Root;
 | |
|   Status  = FatSeekVolumeId (Volume->Root, &LabelDirEnt);
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return Status;
 | |
|   }
 | |
| 
 | |
|   if (LabelDirEnt.Invalid) {
 | |
|     //
 | |
|     // If there is not the relevant directory entry, create a new one
 | |
|     //
 | |
|     ZeroMem (&LabelDirEnt, sizeof (FAT_DIRENT));
 | |
|     LabelDirEnt.EntryCount = 1;
 | |
|     Status                 = FatNewEntryPos (Root, &LabelDirEnt);
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
| 
 | |
|     LabelDirEnt.Entry.Attributes = FAT_ATTRIBUTE_VOLUME_ID;
 | |
|   }
 | |
| 
 | |
|   SetMem (LabelDirEnt.Entry.FileName, FAT_NAME_LEN, ' ');
 | |
|   if (FatStrToFat (Name, FAT_NAME_LEN, LabelDirEnt.Entry.FileName)) {
 | |
|     return EFI_UNSUPPORTED;
 | |
|   }
 | |
| 
 | |
|   FatGetCurrentFatTime (&LabelDirEnt.Entry.FileModificationTime);
 | |
|   return FatStoreDirEnt (Root, &LabelDirEnt);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Create "." and ".." directory entries in the newly-created parent OFile.
 | |
| 
 | |
|   @param  OFile                 - The parent OFile.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - The dot directory entries are successfully created.
 | |
|   @return other                 - An error occurred when creating the directory entry.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FatCreateDotDirEnts (
 | |
|   IN FAT_OFILE          *OFile
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
|   FAT_DIRENT  *DirEnt;
 | |
| 
 | |
|   Status = FatExpandODir (OFile);
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return Status;
 | |
|   }
 | |
| 
 | |
|   FatSetDirEntCluster (OFile);
 | |
|   //
 | |
|   // Create "."
 | |
|   //
 | |
|   Status = FatCreateDirEnt (OFile, L".", FAT_ATTRIBUTE_DIRECTORY, &DirEnt);
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     return Status;
 | |
|   }
 | |
|   //
 | |
|   // Create ".."
 | |
|   //
 | |
|   Status = FatCreateDirEnt (OFile, L"..", FAT_ATTRIBUTE_DIRECTORY, &DirEnt);
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Create a directory entry in the parent OFile.
 | |
| 
 | |
|   @param  OFile                 - The parent OFile.
 | |
|   @param  FileName              - The filename of the newly-created directory entry.
 | |
|   @param  Attributes            - The attribute of the newly-created directory entry.
 | |
|   @param  PtrDirEnt             - The pointer to the newly-created directory entry.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - The directory entry is successfully created.
 | |
|   @retval EFI_OUT_OF_RESOURCES  - Not enough memory to create the directory entry.
 | |
|   @return other                 - An error occurred when creating the directory entry.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FatCreateDirEnt (
 | |
|   IN  FAT_OFILE         *OFile,
 | |
|   IN  CHAR16            *FileName,
 | |
|   IN  UINT8             Attributes,
 | |
|   OUT FAT_DIRENT        **PtrDirEnt
 | |
|   )
 | |
| {
 | |
|   FAT_DIRENT  *DirEnt;
 | |
|   FAT_ODIR    *ODir;
 | |
|   EFI_STATUS  Status;
 | |
| 
 | |
|   ASSERT (OFile != NULL);
 | |
|   ODir = OFile->ODir;
 | |
|   ASSERT (ODir != NULL);
 | |
|   DirEnt = AllocateZeroPool (sizeof (FAT_DIRENT));
 | |
|   if (DirEnt == NULL) {
 | |
|     return EFI_OUT_OF_RESOURCES;
 | |
|   }
 | |
| 
 | |
|   DirEnt->Signature   = FAT_DIRENT_SIGNATURE;
 | |
|   DirEnt->FileString  = AllocateCopyPool (StrSize (FileName), FileName);
 | |
|   if (DirEnt->FileString == NULL) {
 | |
|     Status = EFI_OUT_OF_RESOURCES;
 | |
|     goto Done;
 | |
|   }
 | |
|   //
 | |
|   // Determine how many directory entries we need
 | |
|   //
 | |
|   FatSetEntryCount (OFile, DirEnt);
 | |
|   //
 | |
|   // Determine the file's directory entry position
 | |
|   //
 | |
|   Status = FatNewEntryPos (OFile, DirEnt);
 | |
|   if (EFI_ERROR (Status)) {
 | |
|     goto Done;
 | |
|   }
 | |
| 
 | |
|   FatAddDirEnt (ODir, DirEnt);
 | |
|   DirEnt->Entry.Attributes = Attributes;
 | |
|   *PtrDirEnt               = DirEnt;
 | |
|   DEBUG ((EFI_D_INFO, "FSOpen: Created new directory entry '%S'\n", DirEnt->FileString));
 | |
|   return FatStoreDirEnt (OFile, DirEnt);
 | |
| 
 | |
| Done:
 | |
|   FatFreeDirEnt (DirEnt);
 | |
|   return Status;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Remove this directory entry node from the list of directory entries and hash table.
 | |
| 
 | |
|   @param  OFile                - The parent OFile.
 | |
|   @param  DirEnt               - The directory entry to be removed.
 | |
| 
 | |
|   @retval EFI_SUCCESS          - The directory entry is successfully removed.
 | |
|   @return other                - An error occurred when removing the directory entry.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FatRemoveDirEnt (
 | |
|   IN FAT_OFILE    *OFile,
 | |
|   IN FAT_DIRENT   *DirEnt
 | |
|   )
 | |
| {
 | |
|   FAT_ODIR  *ODir;
 | |
| 
 | |
|   ODir = OFile->ODir;
 | |
|   if (ODir->CurrentCursor == &DirEnt->Link) {
 | |
|     //
 | |
|     // Move the directory cursor to its previous directory entry
 | |
|     //
 | |
|     ODir->CurrentCursor = ODir->CurrentCursor->BackLink;
 | |
|   }
 | |
|   //
 | |
|   // Remove from directory entry list
 | |
|   //
 | |
|   RemoveEntryList (&DirEnt->Link);
 | |
|   //
 | |
|   // Remove from hash table
 | |
|   //
 | |
|   FatDeleteFromHashTable (ODir, DirEnt);
 | |
|   DirEnt->Entry.FileName[0] = DELETE_ENTRY_MARK;
 | |
|   DirEnt->Invalid           = TRUE;
 | |
|   return FatStoreDirEnt (OFile, DirEnt);
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Open the directory entry to get the OFile.
 | |
| 
 | |
|   @param  Parent                - The parent OFile.
 | |
|   @param  DirEnt                - The directory entry to be opened.
 | |
| 
 | |
|   @retval EFI_SUCCESS           - The directory entry is successfully opened.
 | |
|   @retval EFI_OUT_OF_RESOURCES  - not enough memory to allocate a new OFile.
 | |
|   @return other                 - An error occurred when opening the directory entry.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FatOpenDirEnt (
 | |
|   IN FAT_OFILE         *Parent,
 | |
|   IN FAT_DIRENT        *DirEnt
 | |
|   )
 | |
| {
 | |
|   FAT_OFILE   *OFile;
 | |
|   FAT_VOLUME  *Volume;
 | |
| 
 | |
|   if (DirEnt->OFile == NULL) {
 | |
|     //
 | |
|     // Open the directory entry
 | |
|     //
 | |
|     OFile = AllocateZeroPool (sizeof (FAT_OFILE));
 | |
|     if (OFile == NULL) {
 | |
|       return EFI_OUT_OF_RESOURCES;
 | |
|     }
 | |
| 
 | |
|     OFile->Signature = FAT_OFILE_SIGNATURE;
 | |
|     InitializeListHead (&OFile->Opens);
 | |
|     InitializeListHead (&OFile->ChildHead);
 | |
|     OFile->Parent = Parent;
 | |
|     OFile->DirEnt = DirEnt;
 | |
|     if (Parent != NULL) {
 | |
|       //
 | |
|       // The newly created OFile is not root
 | |
|       //
 | |
|       Volume             = Parent->Volume;
 | |
|       OFile->FullPathLen = Parent->FullPathLen + 1 + StrLen (DirEnt->FileString);
 | |
|       OFile->FileCluster = ((DirEnt->Entry.FileClusterHigh) << 16) | (DirEnt->Entry.FileCluster);
 | |
|       InsertTailList (&Parent->ChildHead, &OFile->ChildLink);
 | |
|     } else {
 | |
|       //
 | |
|       // The newly created OFile is root
 | |
|       //
 | |
|       Volume                = VOLUME_FROM_ROOT_DIRENT (DirEnt);
 | |
|       Volume->Root          = OFile;
 | |
|       OFile->FileCluster    = Volume->RootCluster;
 | |
|       if (Volume->FatType  != Fat32) {
 | |
|         OFile->IsFixedRootDir  = TRUE;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     OFile->FileCurrentCluster  = OFile->FileCluster;
 | |
|     OFile->Volume              = Volume;
 | |
|     InsertHeadList (&Volume->CheckRef, &OFile->CheckLink);
 | |
| 
 | |
|     OFile->FileSize = DirEnt->Entry.FileSize;
 | |
|     if ((DirEnt->Entry.Attributes & FAT_ATTRIBUTE_DIRECTORY) != 0) {
 | |
|       if (OFile->IsFixedRootDir) {
 | |
|         OFile->FileSize = Volume->RootEntries * sizeof (FAT_DIRECTORY_ENTRY);
 | |
|       } else {
 | |
|         OFile->FileSize = FatPhysicalDirSize (Volume, OFile->FileCluster);
 | |
|       }
 | |
| 
 | |
|       FatRequestODir (OFile);
 | |
|       if (OFile->ODir == NULL) {
 | |
|         return EFI_OUT_OF_RESOURCES;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     DirEnt->OFile = OFile;
 | |
|   }
 | |
| 
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Close the directory entry and free the OFile.
 | |
| 
 | |
|   @param  DirEnt               - The directory entry to be closed.
 | |
| 
 | |
| **/
 | |
| VOID
 | |
| FatCloseDirEnt (
 | |
|   IN FAT_DIRENT        *DirEnt
 | |
|   )
 | |
| {
 | |
|   FAT_OFILE   *OFile;
 | |
|   FAT_VOLUME  *Volume;
 | |
| 
 | |
|   OFile   = DirEnt->OFile;
 | |
|   ASSERT (OFile != NULL);
 | |
|   Volume  = OFile->Volume;
 | |
| 
 | |
|   if (OFile->ODir != NULL) {
 | |
|     FatDiscardODir (OFile);
 | |
|   }
 | |
| 
 | |
|   if (OFile->Parent == NULL) {
 | |
|     Volume->Root = NULL;
 | |
|   } else {
 | |
|     RemoveEntryList (&OFile->ChildLink);
 | |
|   }
 | |
| 
 | |
|   FreePool (OFile);
 | |
|   DirEnt->OFile = NULL;
 | |
|   if (DirEnt->Invalid == TRUE) {
 | |
|     //
 | |
|     // Free directory entry itself
 | |
|     //
 | |
|     FatFreeDirEnt (DirEnt);
 | |
|   }
 | |
| }
 | |
| 
 | |
| /**
 | |
| 
 | |
|   Traverse filename and open all OFiles that can be opened.
 | |
|   Update filename pointer to the component that can't be opened.
 | |
|   If more than one name component remains, returns an error;
 | |
|   otherwise, return the remaining name component so that the caller might choose to create it.
 | |
| 
 | |
|   @param  PtrOFile              - As input, the reference OFile; as output, the located OFile.
 | |
|   @param  FileName              - The file name relevant to the OFile.
 | |
|   @param  Attributes            - The attribute of the destination OFile.
 | |
|   @param  NewFileName           - The remaining file name.
 | |
| 
 | |
|   @retval EFI_NOT_FOUND         - The file name can't be opened and there is more than one
 | |
|                           components within the name left (this means the name can
 | |
|                           not be created either).
 | |
|   @retval EFI_INVALID_PARAMETER - The parameter is not valid.
 | |
|   @retval EFI_SUCCESS           - Open the file successfully.
 | |
|   @return other                 - An error occurred when locating the OFile.
 | |
| 
 | |
| **/
 | |
| EFI_STATUS
 | |
| FatLocateOFile (
 | |
|   IN OUT FAT_OFILE        **PtrOFile,
 | |
|   IN     CHAR16           *FileName,
 | |
|   IN     UINT8            Attributes,
 | |
|      OUT CHAR16           *NewFileName
 | |
|   )
 | |
| {
 | |
|   EFI_STATUS  Status;
 | |
|   FAT_VOLUME  *Volume;
 | |
|   CHAR16      ComponentName[EFI_PATH_STRING_LENGTH];
 | |
|   UINTN       FileNameLen;
 | |
|   BOOLEAN     DirIntended;
 | |
|   CHAR16      *Next;
 | |
|   FAT_OFILE   *OFile;
 | |
|   FAT_DIRENT  *DirEnt;
 | |
| 
 | |
|   DirEnt = NULL;
 | |
| 
 | |
|   FileNameLen = StrLen (FileName);
 | |
|   if (FileNameLen == 0) {
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
| 
 | |
|   OFile       = *PtrOFile;
 | |
|   Volume      = OFile->Volume;
 | |
| 
 | |
|   DirIntended = FALSE;
 | |
|   if (FileName[FileNameLen - 1] == PATH_NAME_SEPARATOR) {
 | |
|     DirIntended = TRUE;
 | |
|   }
 | |
|   //
 | |
|   // If name starts with path name separator, then move to root OFile
 | |
|   //
 | |
|   if (*FileName == PATH_NAME_SEPARATOR) {
 | |
|     OFile = Volume->Root;
 | |
|     FileName++;
 | |
|     FileNameLen--;
 | |
|   }
 | |
|   //
 | |
|   // Per FAT Spec the file name should meet the following criteria:
 | |
|   //   C1. Length (FileLongName) <= 255
 | |
|   //   C2. Length (X:FileFullPath<NUL>) <= 260
 | |
|   // Here we check C2 first.
 | |
|   //
 | |
|   if (2 + OFile->FullPathLen + 1 + FileNameLen + 1 > EFI_PATH_STRING_LENGTH) {
 | |
|     //
 | |
|     // Full path length can not surpass 256
 | |
|     //
 | |
|     return EFI_INVALID_PARAMETER;
 | |
|   }
 | |
|   //
 | |
|   // Start at current location
 | |
|   //
 | |
|   Next = FileName;
 | |
|   for (;;) {
 | |
|     //
 | |
|     // Get the next component name
 | |
|     //
 | |
|     FileName = Next;
 | |
|     Next     = FatGetNextNameComponent (FileName, ComponentName);
 | |
| 
 | |
|     //
 | |
|     // If end of the file name, we're done
 | |
|     //
 | |
|     if (ComponentName[0] == 0) {
 | |
|       if (DirIntended && OFile->ODir == NULL) {
 | |
|         return EFI_NOT_FOUND;
 | |
|       }
 | |
| 
 | |
|       NewFileName[0] = 0;
 | |
|       break;
 | |
|     }
 | |
|     //
 | |
|     // If "dot", then current
 | |
|     //
 | |
|     if (StrCmp (ComponentName, L".") == 0) {
 | |
|       continue;
 | |
|     }
 | |
|     //
 | |
|     // If "dot dot", then parent
 | |
|     //
 | |
|     if (StrCmp (ComponentName, L"..") == 0) {
 | |
|       if (OFile->Parent == NULL) {
 | |
|         return EFI_INVALID_PARAMETER;
 | |
|       }
 | |
|       OFile = OFile->Parent;
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     if (!FatFileNameIsValid (ComponentName, NewFileName)) {
 | |
|       return EFI_INVALID_PARAMETER;
 | |
|     }
 | |
|     //
 | |
|     // We have a component name, try to open it
 | |
|     //
 | |
|     if (OFile->ODir == NULL) {
 | |
|       //
 | |
|       // This file isn't a directory, can't open it
 | |
|       //
 | |
|       return EFI_NOT_FOUND;
 | |
|     }
 | |
|     //
 | |
|     // Search the compName in the directory
 | |
|     //
 | |
|     Status = FatSearchODir (OFile, NewFileName, &DirEnt);
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
| 
 | |
|     if (DirEnt == NULL) {
 | |
|       //
 | |
|       // component name is not found in the directory
 | |
|       //
 | |
|       if (*Next != 0) {
 | |
|         return EFI_NOT_FOUND;
 | |
|       }
 | |
| 
 | |
|       if (DirIntended && (Attributes & FAT_ATTRIBUTE_DIRECTORY) == 0) {
 | |
|         return EFI_INVALID_PARAMETER;
 | |
|       }
 | |
|       //
 | |
|       // It's the last component name - return with the open
 | |
|       // path and the remaining name
 | |
|       //
 | |
|       break;
 | |
|     }
 | |
| 
 | |
|     Status = FatOpenDirEnt (OFile, DirEnt);
 | |
|     if (EFI_ERROR (Status)) {
 | |
|       return Status;
 | |
|     }
 | |
| 
 | |
|     OFile = DirEnt->OFile;
 | |
|   }
 | |
| 
 | |
|   *PtrOFile = OFile;
 | |
|   return EFI_SUCCESS;
 | |
| }
 | |
| 
 |