2017-07-19 16:31:08 +01:00
|
|
|
|
// /***************************************************************************
|
2016-07-28 18:13:49 +01:00
|
|
|
|
// The Disc Image Chef
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
|
|
//
|
2016-08-01 00:41:02 +01:00
|
|
|
|
// Filename : Info.cs
|
2016-07-28 18:13:49 +01:00
|
|
|
|
// Author(s) : Natalia Portillo <claunia@claunia.com>
|
|
|
|
|
|
//
|
2016-08-01 00:41:02 +01:00
|
|
|
|
// Component : Apple Macintosh File System plugin.
|
2016-07-28 18:13:49 +01:00
|
|
|
|
//
|
|
|
|
|
|
// --[ Description ] ----------------------------------------------------------
|
|
|
|
|
|
//
|
|
|
|
|
|
// Identifies the Apple Macintosh File System 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 <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
//
|
|
|
|
|
|
// ----------------------------------------------------------------------------
|
2017-12-19 03:50:57 +00:00
|
|
|
|
// Copyright © 2011-2018 Natalia Portillo
|
2016-07-28 18:13:49 +01:00
|
|
|
|
// ****************************************************************************/
|
2014-04-17 19:58:14 +00:00
|
|
|
|
|
2011-03-03 18:34:33 +00:00
|
|
|
|
using System;
|
|
|
|
|
|
using System.Text;
|
2017-07-19 16:31:08 +01:00
|
|
|
|
using DiscImageChef.CommonTypes;
|
2017-12-21 14:30:38 +00:00
|
|
|
|
using DiscImageChef.DiscImages;
|
|
|
|
|
|
using Schemas;
|
2011-03-03 18:34:33 +00:00
|
|
|
|
|
2016-08-01 00:41:02 +01:00
|
|
|
|
namespace DiscImageChef.Filesystems.AppleMFS
|
2011-03-03 18:34:33 +00:00
|
|
|
|
{
|
2016-08-01 00:41:02 +01:00
|
|
|
|
// Information from Inside Macintosh Volume II
|
2017-12-21 16:27:09 +00:00
|
|
|
|
public partial class AppleMFS
|
2014-04-14 02:29:13 +00:00
|
|
|
|
{
|
2017-12-21 14:30:38 +00:00
|
|
|
|
public override bool Identify(ImagePlugin imagePlugin, Partition partition)
|
2014-04-14 02:29:13 +00:00
|
|
|
|
{
|
2016-07-28 22:25:26 +01:00
|
|
|
|
ushort drSigWord;
|
2014-04-14 02:29:13 +00:00
|
|
|
|
|
2017-12-20 17:26:28 +00:00
|
|
|
|
if(2 + partition.Start >= partition.End) return false;
|
2014-07-09 19:49:14 +01:00
|
|
|
|
|
2017-07-19 16:37:11 +01:00
|
|
|
|
byte[] mdb_sector = imagePlugin.ReadSector(2 + partition.Start);
|
2014-04-14 02:29:13 +00:00
|
|
|
|
|
2016-08-01 17:59:22 +01:00
|
|
|
|
BigEndianBitConverter.IsLittleEndian = BitConverter.IsLittleEndian;
|
|
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
drSigWord = BigEndianBitConverter.ToUInt16(mdb_sector, 0x000);
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
return drSigWord == MFS_MAGIC;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2017-12-21 14:30:38 +00:00
|
|
|
|
public override void GetInformation(ImagePlugin imagePlugin, Partition partition,
|
2017-12-19 20:33:03 +00:00
|
|
|
|
out string information)
|
2014-04-14 02:29:13 +00:00
|
|
|
|
{
|
|
|
|
|
|
information = "";
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
StringBuilder sb = new StringBuilder();
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
MFS_MasterDirectoryBlock MDB = new MFS_MasterDirectoryBlock();
|
|
|
|
|
|
MFS_BootBlock BB = new MFS_BootBlock();
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
byte[] pString = new byte[16];
|
|
|
|
|
|
byte[] variable_size;
|
|
|
|
|
|
|
2017-07-19 16:37:11 +01:00
|
|
|
|
byte[] mdb_sector = imagePlugin.ReadSector(2 + partition.Start);
|
|
|
|
|
|
byte[] bb_sector = imagePlugin.ReadSector(0 + partition.Start);
|
2012-08-05 00:43:49 +00:00
|
|
|
|
|
2016-08-01 17:59:22 +01:00
|
|
|
|
BigEndianBitConverter.IsLittleEndian = BitConverter.IsLittleEndian;
|
|
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
MDB.drSigWord = BigEndianBitConverter.ToUInt16(mdb_sector, 0x000);
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(MDB.drSigWord != MFS_MAGIC) return;
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
MDB.drCrDate = BigEndianBitConverter.ToUInt32(mdb_sector, 0x002);
|
2014-04-17 02:58:08 +00:00
|
|
|
|
MDB.drLsBkUp = BigEndianBitConverter.ToUInt32(mdb_sector, 0x006);
|
|
|
|
|
|
MDB.drAtrb = BigEndianBitConverter.ToUInt16(mdb_sector, 0x00A);
|
|
|
|
|
|
MDB.drNmFls = BigEndianBitConverter.ToUInt16(mdb_sector, 0x00C);
|
|
|
|
|
|
MDB.drDirSt = BigEndianBitConverter.ToUInt16(mdb_sector, 0x00E);
|
|
|
|
|
|
MDB.drBlLen = BigEndianBitConverter.ToUInt16(mdb_sector, 0x010);
|
|
|
|
|
|
MDB.drNmAlBlks = BigEndianBitConverter.ToUInt16(mdb_sector, 0x012);
|
|
|
|
|
|
MDB.drAlBlkSiz = BigEndianBitConverter.ToUInt32(mdb_sector, 0x014);
|
|
|
|
|
|
MDB.drClpSiz = BigEndianBitConverter.ToUInt32(mdb_sector, 0x018);
|
|
|
|
|
|
MDB.drAlBlSt = BigEndianBitConverter.ToUInt16(mdb_sector, 0x01C);
|
|
|
|
|
|
MDB.drNxtFNum = BigEndianBitConverter.ToUInt32(mdb_sector, 0x01E);
|
|
|
|
|
|
MDB.drFreeBks = BigEndianBitConverter.ToUInt16(mdb_sector, 0x022);
|
|
|
|
|
|
MDB.drVNSiz = mdb_sector[0x024];
|
2017-12-19 20:33:03 +00:00
|
|
|
|
variable_size = new byte[MDB.drVNSiz + 1];
|
|
|
|
|
|
Array.Copy(mdb_sector, 0x024, variable_size, 0, MDB.drVNSiz + 1);
|
2017-06-06 21:23:20 +01:00
|
|
|
|
MDB.drVN = StringHandlers.PascalToString(variable_size, CurrentEncoding);
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
BB.signature = BigEndianBitConverter.ToUInt16(bb_sector, 0x000);
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
|
|
|
|
|
if(BB.signature == MFSBB_MAGIC)
|
2014-04-14 02:29:13 +00:00
|
|
|
|
{
|
|
|
|
|
|
BB.branch = BigEndianBitConverter.ToUInt32(bb_sector, 0x002);
|
|
|
|
|
|
BB.boot_flags = bb_sector[0x006];
|
|
|
|
|
|
BB.boot_version = bb_sector[0x007];
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
BB.sec_sv_pages = BigEndianBitConverter.ToInt16(bb_sector, 0x008);
|
2012-08-05 00:43:49 +00:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
Array.Copy(mdb_sector, 0x00A, pString, 0, 16);
|
2017-06-06 21:23:20 +01:00
|
|
|
|
BB.system_name = StringHandlers.PascalToString(pString, CurrentEncoding);
|
2014-04-14 02:29:13 +00:00
|
|
|
|
Array.Copy(mdb_sector, 0x01A, pString, 0, 16);
|
2017-06-06 21:23:20 +01:00
|
|
|
|
BB.finder_name = StringHandlers.PascalToString(pString, CurrentEncoding);
|
2014-04-14 02:29:13 +00:00
|
|
|
|
Array.Copy(mdb_sector, 0x02A, pString, 0, 16);
|
2017-06-06 21:23:20 +01:00
|
|
|
|
BB.debug_name = StringHandlers.PascalToString(pString, CurrentEncoding);
|
2014-04-14 02:29:13 +00:00
|
|
|
|
Array.Copy(mdb_sector, 0x03A, pString, 0, 16);
|
2017-06-06 21:23:20 +01:00
|
|
|
|
BB.disasm_name = StringHandlers.PascalToString(pString, CurrentEncoding);
|
2014-04-14 02:29:13 +00:00
|
|
|
|
Array.Copy(mdb_sector, 0x04A, pString, 0, 16);
|
2017-06-06 21:23:20 +01:00
|
|
|
|
BB.stupscr_name = StringHandlers.PascalToString(pString, CurrentEncoding);
|
2014-04-14 02:29:13 +00:00
|
|
|
|
Array.Copy(mdb_sector, 0x05A, pString, 0, 16);
|
2017-06-06 21:23:20 +01:00
|
|
|
|
BB.bootup_name = StringHandlers.PascalToString(pString, CurrentEncoding);
|
2014-04-14 02:29:13 +00:00
|
|
|
|
Array.Copy(mdb_sector, 0x06A, pString, 0, 16);
|
2017-06-06 21:23:20 +01:00
|
|
|
|
BB.clipbrd_name = StringHandlers.PascalToString(pString, CurrentEncoding);
|
2012-08-05 00:43:49 +00:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
BB.max_files = BigEndianBitConverter.ToUInt16(bb_sector, 0x07A);
|
|
|
|
|
|
BB.queue_size = BigEndianBitConverter.ToUInt16(bb_sector, 0x07C);
|
|
|
|
|
|
BB.heap_128k = BigEndianBitConverter.ToUInt32(bb_sector, 0x07E);
|
|
|
|
|
|
BB.heap_256k = BigEndianBitConverter.ToUInt32(bb_sector, 0x082);
|
|
|
|
|
|
BB.heap_512k = BigEndianBitConverter.ToUInt32(bb_sector, 0x086);
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else BB.signature = 0x0000;
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
sb.AppendLine("Apple Macintosh File System");
|
|
|
|
|
|
sb.AppendLine();
|
|
|
|
|
|
sb.AppendLine("Master Directory Block:");
|
|
|
|
|
|
sb.AppendFormat("Creation date: {0}", DateHandlers.MacToDateTime(MDB.drCrDate)).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Last backup date: {0}", DateHandlers.MacToDateTime(MDB.drLsBkUp)).AppendLine();
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if((MDB.drAtrb & 0x80) == 0x80) sb.AppendLine("Volume is locked by hardware.");
|
|
|
|
|
|
if((MDB.drAtrb & 0x8000) == 0x8000) sb.AppendLine("Volume is locked by software.");
|
2014-04-14 02:29:13 +00:00
|
|
|
|
sb.AppendFormat("{0} files on volume", MDB.drNmFls).AppendLine();
|
2016-08-01 17:59:22 +01:00
|
|
|
|
sb.AppendFormat("First directory sector: {0}", MDB.drDirSt).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("{0} sectors in directory.", MDB.drBlLen).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("{0} volume allocation blocks.", MDB.drNmAlBlks + 1).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Size of allocation blocks: {0} bytes", MDB.drAlBlkSiz).AppendLine();
|
2014-04-14 02:29:13 +00:00
|
|
|
|
sb.AppendFormat("{0} bytes to allocate.", MDB.drClpSiz).AppendLine();
|
2016-08-01 17:59:22 +01:00
|
|
|
|
sb.AppendFormat("First allocation block (#2) starts in sector {0}.", MDB.drAlBlSt).AppendLine();
|
2014-04-14 02:29:13 +00:00
|
|
|
|
sb.AppendFormat("Next unused file number: {0}", MDB.drNxtFNum).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("{0} unused allocation blocks.", MDB.drFreeBks).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Volume name: {0}", MDB.drVN).AppendLine();
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
|
|
|
|
|
if(BB.signature == MFSBB_MAGIC)
|
2014-04-14 02:29:13 +00:00
|
|
|
|
{
|
|
|
|
|
|
sb.AppendLine("Volume is bootable.");
|
|
|
|
|
|
sb.AppendLine();
|
|
|
|
|
|
sb.AppendLine("Boot Block:");
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if((BB.boot_flags & 0x40) == 0x40) sb.AppendLine("Boot block should be executed.");
|
2017-12-20 23:07:46 +00:00
|
|
|
|
if((BB.boot_flags & 0x80) == 0x80) sb.AppendLine("Boot block is in new unknown format.");
|
2014-04-14 02:29:13 +00:00
|
|
|
|
else
|
|
|
|
|
|
{
|
2017-12-19 20:33:03 +00:00
|
|
|
|
if(BB.sec_sv_pages > 0) sb.AppendLine("Allocate secondary sound buffer at boot.");
|
|
|
|
|
|
else if(BB.sec_sv_pages < 0) sb.AppendLine("Allocate secondary sound and video buffers at boot.");
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
sb.AppendFormat("System filename: {0}", BB.system_name).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Finder filename: {0}", BB.finder_name).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Debugger filename: {0}", BB.debug_name).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Disassembler filename: {0}", BB.disasm_name).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Startup screen filename: {0}", BB.stupscr_name).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("First program to execute at boot: {0}", BB.bootup_name).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Clipboard filename: {0}", BB.clipbrd_name).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Maximum opened files: {0}", BB.max_files * 4).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Event queue size: {0}", BB.queue_size).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Heap size with 128KiB of RAM: {0} bytes", BB.heap_128k).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Heap size with 256KiB of RAM: {0} bytes", BB.heap_256k).AppendLine();
|
|
|
|
|
|
sb.AppendFormat("Heap size with 512KiB of RAM or more: {0} bytes", BB.heap_512k).AppendLine();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
else sb.AppendLine("Volume is not bootable.");
|
2016-04-19 02:11:47 +01:00
|
|
|
|
|
2014-04-14 02:29:13 +00:00
|
|
|
|
information = sb.ToString();
|
2015-12-05 17:10:27 +00:00
|
|
|
|
|
2017-12-21 14:30:38 +00:00
|
|
|
|
xmlFSType = new FileSystemType();
|
2016-04-19 02:11:47 +01:00
|
|
|
|
if(MDB.drLsBkUp > 0)
|
2015-12-06 05:09:31 +00:00
|
|
|
|
{
|
|
|
|
|
|
xmlFSType.BackupDate = DateHandlers.MacToDateTime(MDB.drLsBkUp);
|
|
|
|
|
|
xmlFSType.BackupDateSpecified = true;
|
|
|
|
|
|
}
|
2015-12-05 17:10:27 +00:00
|
|
|
|
xmlFSType.Bootable = BB.signature == MFSBB_MAGIC;
|
|
|
|
|
|
xmlFSType.Clusters = MDB.drNmAlBlks;
|
|
|
|
|
|
xmlFSType.ClusterSize = (int)MDB.drAlBlkSiz;
|
2016-04-19 02:11:47 +01:00
|
|
|
|
if(MDB.drCrDate > 0)
|
2015-12-06 05:09:31 +00:00
|
|
|
|
{
|
|
|
|
|
|
xmlFSType.CreationDate = DateHandlers.MacToDateTime(MDB.drCrDate);
|
|
|
|
|
|
xmlFSType.CreationDateSpecified = true;
|
|
|
|
|
|
}
|
2015-12-05 17:10:27 +00:00
|
|
|
|
xmlFSType.Files = MDB.drNmFls;
|
2015-12-06 05:09:31 +00:00
|
|
|
|
xmlFSType.FilesSpecified = true;
|
2015-12-05 17:10:27 +00:00
|
|
|
|
xmlFSType.FreeClusters = MDB.drFreeBks;
|
2015-12-06 05:09:31 +00:00
|
|
|
|
xmlFSType.FreeClustersSpecified = true;
|
2015-12-05 17:10:27 +00:00
|
|
|
|
xmlFSType.Type = "MFS";
|
|
|
|
|
|
xmlFSType.VolumeName = MDB.drVN;
|
2014-04-14 02:29:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2017-12-19 20:33:03 +00:00
|
|
|
|
}
|