mirror of
https://github.com/aaru-dps/Aaru.git
synced 2025-12-16 19:24:25 +00:00
Read Xbox FAT root directory.
This commit is contained in:
@@ -40,11 +40,20 @@ namespace DiscImageChef.Filesystems.FATX
|
|||||||
const uint FATX_CIGAM = 0x46415458;
|
const uint FATX_CIGAM = 0x46415458;
|
||||||
const byte UNUSED_DIRENTRY = 0x00;
|
const byte UNUSED_DIRENTRY = 0x00;
|
||||||
const byte DELETED_DIRENTRY = 0xE5;
|
const byte DELETED_DIRENTRY = 0xE5;
|
||||||
|
const byte FINISHED_DIRENTRY = 0xFF;
|
||||||
const byte MAX_FILENAME = 42;
|
const byte MAX_FILENAME = 42;
|
||||||
const int MAX_XFAT16_CLUSTERS = 65525;
|
const int MAX_XFAT16_CLUSTERS = 65525;
|
||||||
const int FAT_START = 4096;
|
const int FAT_START = 4096;
|
||||||
const uint FATX32_ID = 0xFFFFFFF8;
|
const uint FATX32_ID = 0xFFFFFFF8;
|
||||||
const ushort FATX16_ID = 0xFFF8;
|
const ushort FATX16_ID = 0xFFF8;
|
||||||
|
const uint FAT32_MASK = 0x0FFFFFFF;
|
||||||
|
const uint FAT32_END_MASK = 0xFFFFFF8;
|
||||||
|
const uint FAT32_FORMATTED = 0xFFFFFF6;
|
||||||
|
const uint FAT32_BAD = 0xFFFFFF7;
|
||||||
|
const ushort FAT16_END_MASK = 0xFFF8;
|
||||||
|
const ushort FAT16_FORMATTED = 0xFFF6;
|
||||||
|
const ushort FAT16_BAD = 0xFFF7;
|
||||||
|
const ushort FAT_RESERVED = 1;
|
||||||
|
|
||||||
[Flags]
|
[Flags]
|
||||||
enum Attributes : byte
|
enum Attributes : byte
|
||||||
|
|||||||
@@ -42,16 +42,18 @@ namespace DiscImageChef.Filesystems.FATX
|
|||||||
{
|
{
|
||||||
public partial class XboxFatPlugin : IReadOnlyFilesystem
|
public partial class XboxFatPlugin : IReadOnlyFilesystem
|
||||||
{
|
{
|
||||||
ushort[] fat16;
|
uint bytesPerCluster;
|
||||||
uint[] fat32;
|
ushort[] fat16;
|
||||||
ulong fatStartSector;
|
uint[] fat32;
|
||||||
ulong firstClusterSector;
|
ulong fatStartSector;
|
||||||
IMediaImage imagePlugin;
|
ulong firstClusterSector;
|
||||||
bool littleEndian;
|
IMediaImage imagePlugin;
|
||||||
bool mounted;
|
bool littleEndian;
|
||||||
Partition partition;
|
bool mounted;
|
||||||
uint sectorsPerCluster;
|
Partition partition;
|
||||||
FileSystemInfo stat;
|
Dictionary<string, DirectoryEntry> rootDirectory;
|
||||||
|
uint sectorsPerCluster;
|
||||||
|
FileSystemInfo stat;
|
||||||
|
|
||||||
Superblock superblock;
|
Superblock superblock;
|
||||||
public FileSystemType XmlFsType { get; private set; }
|
public FileSystemType XmlFsType { get; private set; }
|
||||||
|
|||||||
@@ -31,6 +31,7 @@
|
|||||||
// ****************************************************************************/
|
// ****************************************************************************/
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using DiscImageChef.CommonTypes.Structs;
|
using DiscImageChef.CommonTypes.Structs;
|
||||||
|
|
||||||
namespace DiscImageChef.Filesystems.FATX
|
namespace DiscImageChef.Filesystems.FATX
|
||||||
@@ -67,5 +68,35 @@ namespace DiscImageChef.Filesystems.FATX
|
|||||||
|
|
||||||
throw new NotImplementedException();
|
throw new NotImplementedException();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint[] GetClusters(uint startCluster)
|
||||||
|
{
|
||||||
|
if(startCluster == 0) return null;
|
||||||
|
|
||||||
|
if(fat16 is null)
|
||||||
|
{
|
||||||
|
if(startCluster >= fat32.Length) return null;
|
||||||
|
}
|
||||||
|
else if(startCluster >= fat16.Length) return null;
|
||||||
|
|
||||||
|
List<uint> clusters = new List<uint>();
|
||||||
|
|
||||||
|
uint nextCluster = startCluster;
|
||||||
|
|
||||||
|
if(fat16 is null)
|
||||||
|
while((nextCluster & FAT32_MASK) > 0 && (nextCluster & FAT32_MASK) <= FAT32_BAD)
|
||||||
|
{
|
||||||
|
clusters.Add(nextCluster);
|
||||||
|
nextCluster = fat32[nextCluster];
|
||||||
|
}
|
||||||
|
else
|
||||||
|
while(nextCluster > 0 && nextCluster <= FAT16_BAD)
|
||||||
|
{
|
||||||
|
clusters.Add(nextCluster);
|
||||||
|
nextCluster = fat16[nextCluster];
|
||||||
|
}
|
||||||
|
|
||||||
|
return clusters.ToArray();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -159,11 +159,55 @@ namespace DiscImageChef.Filesystems.FATX
|
|||||||
this.partition = partition;
|
this.partition = partition;
|
||||||
this.imagePlugin = imagePlugin;
|
this.imagePlugin = imagePlugin;
|
||||||
firstClusterSector = fatStartSector + fatSize;
|
firstClusterSector = fatStartSector + fatSize;
|
||||||
|
bytesPerCluster = sectorsPerCluster * imagePlugin.Info.SectorSize;
|
||||||
|
|
||||||
DicConsole.DebugWriteLine("Xbox FAT plugin", "sectorsPerCluster = {0}", sectorsPerCluster);
|
DicConsole.DebugWriteLine("Xbox FAT plugin", "sectorsPerCluster = {0}", sectorsPerCluster);
|
||||||
|
DicConsole.DebugWriteLine("Xbox FAT plugin", "bytesPerCluster = {0}", bytesPerCluster);
|
||||||
DicConsole.DebugWriteLine("Xbox FAT plugin", "firstClusterSector = {0}", firstClusterSector);
|
DicConsole.DebugWriteLine("Xbox FAT plugin", "firstClusterSector = {0}", firstClusterSector);
|
||||||
|
|
||||||
throw new NotImplementedException();
|
uint[] rootDirectoryClusters = GetClusters(superblock.rootDirectoryCluster);
|
||||||
|
|
||||||
|
if(rootDirectoryClusters is null) return Errno.InvalidArgument;
|
||||||
|
|
||||||
|
byte[] rootDirectoryBuffer = new byte[bytesPerCluster * rootDirectoryClusters.Length];
|
||||||
|
|
||||||
|
DicConsole.DebugWriteLine("Xbox FAT plugin", "Reading root directory");
|
||||||
|
for(int i = 0; i < rootDirectoryClusters.Length; i++)
|
||||||
|
{
|
||||||
|
buffer =
|
||||||
|
imagePlugin.ReadSectors(firstClusterSector + (rootDirectoryClusters[i] - 1) * sectorsPerCluster,
|
||||||
|
sectorsPerCluster);
|
||||||
|
Array.Copy(buffer, 0, rootDirectoryBuffer, i * bytesPerCluster, bytesPerCluster);
|
||||||
|
}
|
||||||
|
|
||||||
|
rootDirectory = new Dictionary<string, DirectoryEntry>();
|
||||||
|
|
||||||
|
int pos = 0;
|
||||||
|
while(pos < rootDirectoryBuffer.Length)
|
||||||
|
{
|
||||||
|
DirectoryEntry entry = littleEndian
|
||||||
|
? Marshal
|
||||||
|
.ByteArrayToStructureLittleEndian<DirectoryEntry
|
||||||
|
>(rootDirectoryBuffer, pos, Marshal.SizeOf<DirectoryEntry>())
|
||||||
|
: Marshal.ByteArrayToStructureBigEndian<DirectoryEntry>(rootDirectoryBuffer,
|
||||||
|
pos,
|
||||||
|
Marshal
|
||||||
|
.SizeOf<
|
||||||
|
DirectoryEntry
|
||||||
|
>());
|
||||||
|
|
||||||
|
pos += Marshal.SizeOf<DirectoryEntry>();
|
||||||
|
|
||||||
|
if(entry.filenameSize == UNUSED_DIRENTRY || entry.filenameSize == FINISHED_DIRENTRY) break;
|
||||||
|
|
||||||
|
if(entry.filenameSize == DELETED_DIRENTRY || entry.filenameSize > MAX_FILENAME) continue;
|
||||||
|
|
||||||
|
string filename = Encoding.GetString(entry.filename, 0, entry.filenameSize);
|
||||||
|
|
||||||
|
rootDirectory.Add(filename, entry);
|
||||||
|
}
|
||||||
|
|
||||||
|
return Errno.NoError;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Errno Unmount()
|
public Errno Unmount()
|
||||||
|
|||||||
Reference in New Issue
Block a user