// /*************************************************************************** // The Disc Image Chef // ---------------------------------------------------------------------------- // // Filename : MicroDOS.cs // Author(s) : Natalia Portillo // // Component : MicroDOS filesystem plugin // // --[ Description ] ---------------------------------------------------------- // // Identifies the MicroDOS filesystem and shows information. // // --[ 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.Runtime.InteropServices; using System.Text; using DiscImageChef.CommonTypes; using DiscImageChef.CommonTypes.Interfaces; using Schemas; using Marshal = DiscImageChef.Helpers.Marshal; namespace DiscImageChef.Filesystems { // Information from http://www.owg.ru/mkt/BK/MKDOS.TXT // Thanks to tarlabnor for translating it public class MicroDOS : IFilesystem { const ushort MAGIC = 0xA72E; const ushort MAGIC2 = 0x530C; public FileSystemType XmlFsType { get; private set; } public Encoding Encoding { get; private set; } public string Name => "MicroDOS file system"; public Guid Id => new Guid("9F9A364A-1A27-48A3-B730-7A7122000324"); public string Author => "Natalia Portillo"; public bool Identify(IMediaImage imagePlugin, Partition partition) { if(1 + partition.Start >= partition.End) return false; if(imagePlugin.Info.SectorSize < 512) return false; byte[] bk0 = imagePlugin.ReadSector(0 + partition.Start); MicroDosBlock0 block0 = Marshal.ByteArrayToStructureLittleEndian(bk0); return block0.label == MAGIC && block0.mklabel == MAGIC2; } public void GetInformation(IMediaImage imagePlugin, Partition partition, out string information, Encoding encoding) { Encoding = encoding ?? Encoding.GetEncoding("koi8-r"); information = ""; StringBuilder sb = new StringBuilder(); byte[] bk0 = imagePlugin.ReadSector(0 + partition.Start); MicroDosBlock0 block0 = Marshal.ByteArrayToStructureLittleEndian(bk0); sb.AppendLine("MicroDOS filesystem"); sb.AppendFormat("Volume has {0} blocks ({1} bytes)", block0.blocks, block0.blocks * 512).AppendLine(); sb.AppendFormat("Volume has {0} blocks used ({1} bytes)", block0.usedBlocks, block0.usedBlocks * 512) .AppendLine(); sb.AppendFormat("Volume contains {0} files", block0.files).AppendLine(); sb.AppendFormat("First used block is {0}", block0.firstUsedBlock).AppendLine(); XmlFsType = new FileSystemType { Type = "MicroDOS", ClusterSize = 512, Clusters = block0.blocks, Files = block0.files, FilesSpecified = true, FreeClusters = block0.blocks - block0.usedBlocks, FreeClustersSpecified = true }; information = sb.ToString(); } // Followed by directory entries [StructLayout(LayoutKind.Sequential, Pack = 1)] struct MicroDosBlock0 { /// BK starts booting here [MarshalAs(UnmanagedType.ByValArray, SizeConst = 24)] public byte[] bootCode; /// Number of files in directory public ushort files; /// Total number of blocks in files of the directory public ushort usedBlocks; /// Unknown [MarshalAs(UnmanagedType.ByValArray, SizeConst = 228)] public byte[] unknown; /// Ownership label (label that shows it belongs to Micro DOS format) public ushort label; /// MK-DOS directory format label public ushort mklabel; /// Unknown [MarshalAs(UnmanagedType.ByValArray, SizeConst = 50)] public byte[] unknown2; /// /// Disk size in blocks (absolute value for the system unlike NORD, NORTON etc.) that /// doesn't use two fixed values 40 or 80 tracks, but i.e. if you drive works with 76 tracks /// this field will contain an appropriate number of blocks /// public ushort blocks; /// Number of the first file's block. Value is changable public ushort firstUsedBlock; /// Unknown [MarshalAs(UnmanagedType.ByValArray, SizeConst = 6)] public byte[] unknown3; } [StructLayout(LayoutKind.Sequential, Pack = 1)] struct DirectoryEntry { /// File status public byte status; /// Directory number (0 - root) public byte directory; /// File name 14. symbols in ASCII KOI8 [MarshalAs(UnmanagedType.ByValArray, SizeConst = 12)] public byte[] filename; /// Block number public ushort blockNo; /// Length in blocks public ushort blocks; /// Address public ushort address; /// Length public ushort length; } enum FileStatus : byte { CommonFile = 0, Protected = 1, LogicalDisk = 2, BadFile = 0x80, Deleted = 0xFF } } }