// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // Filename : File.cs // Author(s) : Natalia Portillo // // Component : FATX filesystem plugin. // // --[ Description ] ---------------------------------------------------------- // // Methods to handle files. // // --[ License ] -------------------------------------------------------------- // // This library is free software; you can redistribute it and/or modify // it under the terms of the GNU Lesser General Public License as // published by the Free Software Foundation; either version 2.1 of the // License, or (at your option) any later version. // // This library is distributed in the hope that it will be useful, but // WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU // Lesser General Public License for more details. // // You should have received a copy of the GNU Lesser General Public // License along with this library; if not, see . // // ---------------------------------------------------------------------------- // Copyright © 2011-2019 Natalia Portillo // ****************************************************************************/ using System; using System.Collections.Generic; using System.IO; using System.Linq; using DiscImageChef.CommonTypes.Structs; using FileAttributes = DiscImageChef.CommonTypes.Structs.FileAttributes; namespace DiscImageChef.Filesystems.FATX { public partial class XboxFatPlugin { public Errno MapBlock(string path, long fileBlock, out long deviceBlock) { deviceBlock = 0; if(!mounted) return Errno.AccessDenied; Errno err = Stat(path, out FileEntryInfo stat); if(err != Errno.NoError) return err; if(stat.Attributes.HasFlag(FileAttributes.Directory) && !debug) return Errno.IsDirectory; uint[] clusters = GetClusters((uint)stat.Inode); if(fileBlock >= clusters.Length) return Errno.InvalidArgument; deviceBlock = (long)(firstClusterSector + (clusters[fileBlock] - 1) * sectorsPerCluster); return Errno.NoError; } public Errno GetAttributes(string path, out FileAttributes attributes) { attributes = new FileAttributes(); if(!mounted) return Errno.AccessDenied; Errno err = Stat(path, out FileEntryInfo stat); if(err != Errno.NoError) return err; attributes = stat.Attributes; return Errno.NoError; } public Errno Read(string path, long offset, long size, ref byte[] buf) { if(!mounted) return Errno.AccessDenied; Errno err = Stat(path, out FileEntryInfo stat); if(err != Errno.NoError) return err; if(stat.Attributes.HasFlag(FileAttributes.Directory) && !debug) return Errno.IsDirectory; if(offset >= stat.Length) return Errno.InvalidArgument; if(size + offset >= stat.Length) size = stat.Length - offset; uint[] clusters = GetClusters((uint)stat.Inode); long firstCluster = offset / bytesPerCluster; long offsetInCluster = offset % bytesPerCluster; long sizeInClusters = (size + offsetInCluster) / bytesPerCluster; if((size + offsetInCluster) % bytesPerCluster > 0) sizeInClusters++; MemoryStream ms = new MemoryStream(); for(int i = 0; i < sizeInClusters; i++) { if(i + firstCluster >= clusters.Length) return Errno.InvalidArgument; byte[] buffer = imagePlugin.ReadSectors(firstClusterSector + (clusters[i + firstCluster] - 1) * sectorsPerCluster, sectorsPerCluster); ms.Write(buffer, 0, buffer.Length); } ms.Position = offsetInCluster; buf = new byte[size]; ms.Read(buf, 0, (int)size); return Errno.NoError; } public Errno Stat(string path, out FileEntryInfo stat) { stat = null; if(!mounted) return Errno.AccessDenied; if(debug && (string.IsNullOrEmpty(path) || path == "$" || path == "/")) { stat = new FileEntryInfo { Attributes = FileAttributes.Directory | FileAttributes.System | FileAttributes.Hidden, Blocks = GetClusters(superblock.rootDirectoryCluster).Length, BlockSize = bytesPerCluster, Length = GetClusters(superblock.rootDirectoryCluster).Length * bytesPerCluster, Inode = superblock.rootDirectoryCluster, Links = 1 }; return Errno.NoError; } Errno err = GetFileEntry(path, out DirectoryEntry entry); if(err != Errno.NoError) return err; stat = new FileEntryInfo { Attributes = new FileAttributes(), Blocks = entry.length / bytesPerCluster, BlockSize = bytesPerCluster, Length = entry.length, Inode = entry.firstCluster, Links = 1, CreationTime = littleEndian ? DateHandlers.DosToDateTime(entry.creationDate, entry.creationTime).AddYears(20) : DateHandlers.DosToDateTime(entry.creationTime, entry.creationDate), AccessTime = littleEndian ? DateHandlers.DosToDateTime(entry.lastAccessDate, entry.lastAccessTime).AddYears(20) : DateHandlers.DosToDateTime(entry.lastAccessTime, entry.lastAccessDate), LastWriteTime = littleEndian ? DateHandlers .DosToDateTime(entry.lastWrittenDate, entry.lastWrittenTime).AddYears(20) : DateHandlers.DosToDateTime(entry.lastWrittenTime, entry.lastWrittenDate) }; if(entry.length % bytesPerCluster > 0) stat.Blocks++; if(entry.attributes.HasFlag(Attributes.Directory)) { stat.Attributes |= FileAttributes.Directory; stat.Blocks = GetClusters(entry.firstCluster).Length; stat.Length = stat.Blocks * stat.BlockSize; } if(entry.attributes.HasFlag(Attributes.ReadOnly)) stat.Attributes |= FileAttributes.ReadOnly; if(entry.attributes.HasFlag(Attributes.Hidden)) stat.Attributes |= FileAttributes.Hidden; if(entry.attributes.HasFlag(Attributes.System)) stat.Attributes |= FileAttributes.System; if(entry.attributes.HasFlag(Attributes.Archive)) stat.Attributes |= FileAttributes.Archive; return Errno.NoError; } 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 clusters = new List(); uint nextCluster = startCluster; if(fat16 is null) while((nextCluster & FAT32_MASK) > 0 && (nextCluster & FAT32_MASK) <= FAT32_FORMATTED) { clusters.Add(nextCluster); nextCluster = fat32[nextCluster]; } else while(nextCluster > 0 && nextCluster <= FAT16_FORMATTED) { clusters.Add(nextCluster); nextCluster = fat16[nextCluster]; } return clusters.ToArray(); } Errno GetFileEntry(string path, out DirectoryEntry entry) { entry = new DirectoryEntry(); string cutPath = path.StartsWith("/") ? path.Substring(1).ToLower(cultureInfo) : path.ToLower(cultureInfo); string[] pieces = cutPath.Split(new[] {'/'}, StringSplitOptions.RemoveEmptyEntries); if(pieces.Length == 0) return Errno.InvalidArgument; string parentPath = string.Join("/", pieces, 0, pieces.Length - 1); Errno err = ReadDir(parentPath, out _); if(err != Errno.NoError) return err; Dictionary parent; if(pieces.Length == 1) parent = rootDirectory; else if(!directoryCache.TryGetValue(parentPath, out parent)) return Errno.InvalidArgument; KeyValuePair dirent = parent.FirstOrDefault(t => t.Key.ToLower(cultureInfo) == pieces[pieces.Length - 1]); if(string.IsNullOrEmpty(dirent.Key)) return Errno.NoSuchFile; entry = dirent.Value; return Errno.NoError; } } }