private Fat16RootDirectory(Fat16BootSector bs, bool readOnly) : base(bs.GetRootDirEntryCount(), readOnly, true) { if (bs.GetRootDirEntryCount() <= 0) { throw new ArgumentException( "root directory size is " + bs.GetRootDirEntryCount()); } deviceOffset = FatUtils.GetRootDirOffset(bs); device = bs.GetDevice(); }
/// <summary> /// Constructor for FatFileSystem in specified readOnly mode /// </summary> /// <param name="device">the <see cref="IBlockDevice"/> holding the file system</param> /// <param name="readOnly"></param> /// <param name="ignoreFatDifferences"></param> /// <exception cref="IOException">IOException on read error</exception> private FatFileSystem(IBlockDevice device, bool readOnly, bool ignoreFatDifferences) : base(readOnly) { bs = BootSector.Read(device); if (bs.GetNrFats() <= 0) { throw new IOException( "boot sector says there are no FATs"); } filesOffset = FatUtils.GetFilesOffset(bs); fatType = bs.GetFatType(); fat = Fat.Read(bs, 0); if (!ignoreFatDifferences) { for (var i = 1; i < bs.GetNrFats(); i++) { var tmpFat = Fat.Read(bs, i); if (!fat.Equals(tmpFat)) { throw new IOException("FAT " + i + " differs from FAT 0"); } } } if (fatType == FatType.BaseFat32) { var f32Bs = (Fat32BootSector)bs; var rootDirFile = new ClusterChain(fat, f32Bs.GetRootDirFirstCluster(), IsReadOnly()); rootDirStore = ClusterChainDirectory.ReadRoot(rootDirFile); fsiSector = FsInfoSector.Read(f32Bs); if (fsiSector.GetFreeClusterCount() != fat.GetFreeClusterCount()) { throw new IOException("free cluster count mismatch - fat: " + fat.GetFreeClusterCount() + " - fsinfo: " + fsiSector.GetFreeClusterCount()); } } else { rootDirStore = Fat16RootDirectory.Read((Fat16BootSector)bs, readOnly); fsiSector = null; } rootDir = new FatLfnDirectory(rootDirStore, fat, IsReadOnly()); }
/// <summary> /// Reads a <see cref="Fat"/> as specified by a <see cref="BootSector"/>. /// </summary> /// <param name="bs">the boot sector specifying the <see cref="Fat"/> layout</param> /// <param name="fatNr">the number of the <see cref="Fat"/> to read</param> /// <returns>the <see cref="Fat"/> that was read</returns> /// <exception cref="IOException">IOException on read error</exception> /// <exception cref="ArgumentException">ArgumentException if fatNr is greater than /// <see cref="BootSector.GetNrFats"/></exception> public static Fat Read(BootSector bs, int fatNr) { if (fatNr > bs.GetNrFats()) { throw new ArgumentException( "boot sector says there are only " + bs.GetNrFats() + " FATs when reading FAT #" + fatNr); } var fatOffset = FatUtils.GetFatOffset(bs, fatNr); var result = new Fat(bs, fatOffset); result.Read(); return(result); }
public ClusterChain(Fat fat, long startCluster, bool readOnly) : base(readOnly) { this.fat = fat; if (startCluster != 0) { this.fat.TestCluster(startCluster); if (this.fat.IsFreeCluster(startCluster)) { throw new ArgumentException( "cluster " + startCluster + " is free"); } } device = fat.GetDevice(); dataOffset = FatUtils.GetFilesOffset(fat.GetBootSector()); this.startCluster = startCluster; clusterSize = fat.GetBootSector().GetBytesPerCluster(); }
/// <summary> /// Creates a new <see cref="Fat"/> as specified by a <see cref="BootSector"/>. /// </summary> /// <param name="bs">the boot sector specifying the <see cref="Fat"/> layout</param> /// <param name="fatNr">the number of the <see cref="Fat"/> to create</param> /// <returns>the <see cref="Fat"/> that was created</returns> /// <exception cref="IOException">IOException on write error</exception> /// <exception cref="ArgumentException">ArgumentException if fatNr is greater than /// <see cref="BootSector.GetNrFats"/></exception> public static Fat Create(BootSector bs, int fatNr) { if (fatNr > bs.GetNrFats()) { throw new ArgumentException( "boot sector says there are only " + bs.GetNrFats() + " FATs when creating FAT #" + fatNr); } var fatOffset = FatUtils.GetFatOffset(bs, fatNr); var result = new Fat(bs, fatOffset); if (bs.GetDataClusterCount() > result.entries.Length) { throw new IOException("FAT too small for device"); } result.Init(bs.GetMediumDescriptor()); result.Write(); return(result); }
/// <summary> /// Flush all changed structures to the device. /// </summary> /// <exception cref="IOException">IOException on write error</exception> public override void Flush() { CheckClosed(); if (bs.IsDirty()) { bs.Write(); } for (var i = 0; i < bs.GetNrFats(); i++) { fat.WriteCopy(FatUtils.GetFatOffset(bs, i)); } rootDir.Flush(); if (fsiSector != null) { fsiSector.SetFreeClusterCount(fat.GetFreeClusterCount()); fsiSector.SetLastAllocatedCluster(fat.GetLastAllocatedCluster()); fsiSector.Write(); } }
/** * Returns the size of the data-containing portion of the file system. * * @return the number of bytes usable for storing user data */ private long GetDataSize() { return((GetSectorCount() * GetBytesPerSector()) - FatUtils.GetFilesOffset(this)); }
/// <summary> /// Initializes the boot sector and file system for the device. The file /// system created by this method will always be in read-write mode. /// </summary> /// <returns>the file system that was created</returns> /// <exception cref="System.IO.IOException">IOException on write error</exception> public FatFileSystem Format() { var sectorSize = device.GetSectorSize(); var totalSectors = (int)(device.GetSize() / sectorSize); FsInfoSector fsi; BootSector bs; if (sectorsPerCluster == 0) { throw new Exception(); } if (fatType == FatType.BaseFat32) { bs = new Fat32BootSector(device); InitBootSector(bs); var f32Bs = (Fat32BootSector)bs; f32Bs.SetFsInfoSectorNr(1); f32Bs.SetSectorsPerFat(SectorsPerFat(0, totalSectors)); var rnd = new Random(); f32Bs.SetFileSystemId(rnd.Next()); f32Bs.SetVolumeLabel(label); /* create FS info sector */ fsi = FsInfoSector.Create(f32Bs); } else { bs = new Fat16BootSector(device); InitBootSector(bs); var f16Bs = (Fat16BootSector)bs; var rootDirEntries = RootDirectorySize( device.GetSectorSize(), totalSectors); f16Bs.SetRootDirEntryCount(rootDirEntries); f16Bs.SetSectorsPerFat(SectorsPerFat(rootDirEntries, totalSectors)); if (label != null) { f16Bs.SetVolumeLabel(label); } fsi = null; } // bs.write(); if (fatType == FatType.BaseFat32) { var f32Bs = (Fat32BootSector)bs; /* possibly writes the boot sector copy */ f32Bs.WriteCopy(device); } var fat = Fat.Create(bs, 0); AbstractDirectory rootDirStore; if (fatType == FatType.BaseFat32) { rootDirStore = ClusterChainDirectory.CreateRoot(fat); fsi.SetFreeClusterCount(fat.GetFreeClusterCount()); fsi.SetLastAllocatedCluster(fat.GetLastAllocatedCluster()); fsi.Write(); } else { rootDirStore = Fat16RootDirectory.Create((Fat16BootSector)bs); } var rootDir = new FatLfnDirectory(rootDirStore, fat, false); rootDir.Flush(); for (var i = 0; i < bs.GetNrFats(); i++) { fat.WriteCopy(FatUtils.GetFatOffset(bs, i)); } bs.Write(); var fs = FatFileSystem.Read(device, false); if (label != null) { fs.SetVolumeLabel(label); } fs.Flush(); return(fs); }