From 341ee445094e4744c3ad5e21605e832e764e1614 Mon Sep 17 00:00:00 2001 From: zeromus Date: Mon, 7 Mar 2011 01:07:49 +0000 Subject: [PATCH] re-engineer HawkFile to be aware of archives. its a little more complex to use now (not only do you have to open it, you have to call one of the Bind() methods on it to choose an interior file), but its more powerful. --- .../Consoles/Nintendo/NES/BoardDetector.cs | 3 +- BizHawk.MultiClient/HawkFile.cs | 270 +++++++++++++----- BizHawk.MultiClient/MainForm.cs | 171 +++++------ BizHawk.MultiClient/RomGame.cs | 57 ++-- BizHawk.Util/7z/FileSignatureChecker.cs | 32 ++- BizHawk.Util/7z/Formats.cs | 14 +- 6 files changed, 355 insertions(+), 192 deletions(-) diff --git a/BizHawk.Emulation/Consoles/Nintendo/NES/BoardDetector.cs b/BizHawk.Emulation/Consoles/Nintendo/NES/BoardDetector.cs index c8b0998706..9546b76f7d 100644 --- a/BizHawk.Emulation/Consoles/Nintendo/NES/BoardDetector.cs +++ b/BizHawk.Emulation/Consoles/Nintendo/NES/BoardDetector.cs @@ -1,12 +1,11 @@ //http://nesdev.parodius.com/bbs/viewtopic.php?p=4571&sid=db4c7e35316cc5d734606dd02f11dccb using System; +using System.Xml; using System.Diagnostics; using System.Globalization; using System.IO; using System.Collections.Generic; -using BizHawk.Emulation.CPUs.M6502; - namespace BizHawk.Emulation.Consoles.Nintendo { diff --git a/BizHawk.MultiClient/HawkFile.cs b/BizHawk.MultiClient/HawkFile.cs index 7764d5245e..fa80f19e32 100644 --- a/BizHawk.MultiClient/HawkFile.cs +++ b/BizHawk.MultiClient/HawkFile.cs @@ -3,92 +3,230 @@ using System.IO; namespace BizHawk.MultiClient { + //todo: + //split into "bind" and "open (the bound thing)" + //scan archive to flatten interior directories down to a path (maintain our own archive item list) + public class HawkFile : IDisposable { - private bool zipped; - public bool Zipped { get { return zipped; } } + /// + /// returns whether a bound file exists. if there is no bound file, it can't exist + /// + public bool Exists { get { if (!rootExists) return false; return boundStream != null; } } - private bool exists; - public bool Exists { get { return exists; } } - - private string extension; - public string Extension { get { return extension; } } + /// + /// returns whether the root exists (the actual physical file) + /// + public bool RootExists { get { return rootExists; } } - public string Directory { get { return Path.GetDirectoryName(rawFileName); } } + /// + /// gets the directory containing the root + /// + public string Directory { get { return Path.GetDirectoryName(rootPath); } } - private string rawFileName; - private string name; - public string Name { get { return name; } } - public string FullName { get { return name + "." + extension; } } + /// + /// returns a stream for the currently bound file + /// + public Stream GetStream() + { + if (boundStream == null) + throw new InvalidOperationException("HawkFil: Can't call GetStream() before youve successfully bound something!"); + return boundStream; + } - private IDisposable thingToDispose; - private Stream zippedStream; + /// + /// indicates whether this instance is bound + /// + public bool IsBound { get { return boundStream != null; } } - public HawkFile(string path) : this(path,"SMS","PCE","SGX","GG","SG","BIN","SMD","GB","IPS") {} - - public HawkFile(string path, params string[] recognizedExtensions) + /// + /// returns the complete canonical name ("archive|member") of the bound file + /// + public string CanonicalName { get { return MakeCanonicalName(rootPath,memberPath); } } + + /// + /// returns the virtual name of the bound file (disregarding the archive) + /// + public string Name { get { return GetBoundNameFromCanonical(MakeCanonicalName(rootPath,memberPath)); } } + + /// + /// returns the extension of Name + /// + public string Extension { get { return Path.GetExtension(Name); } } + + //--- + bool rootExists; + string rootPath; + string memberPath; + Stream rootStream, boundStream; + SevenZip.SevenZipExtractor extractor; + + public static bool PathExists(string path) + { + using (var hf = new HawkFile(path)) + return hf.Exists; + } + + public HawkFile(string path) { - var file = new FileInfo(path); + string autobind = null; + if (IsCanonicalArchivePath(path)) + { + string[] parts = path.Split('|'); + path = parts[0]; + autobind = parts[1]; + } - exists = file.Exists; - if (file.Exists == false) + var fi = new FileInfo(path); + + rootExists = fi.Exists; + if (fi.Exists == false) return; - if (file.Extension.ToLower().In(".zip",".rar",".7z")) - { - LoadZipFile(path, recognizedExtensions); - return; - } + rootPath = path; - zipped = false; - extension = file.Extension.Substring(1).ToUpperInvariant(); - rawFileName = path; - name = Path.GetFileNameWithoutExtension(path); + AnalyzeArchive(path); + if (extractor == null) + { + rootStream = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.Read); + } + + if (autobind != null) + { + autobind = autobind.ToUpperInvariant(); + for (int i = 0; i < extractor.ArchiveFileData.Count; i++) + { + if (extractor.ArchiveFileNames[i].ToUpperInvariant() == autobind) + { + BindArchiveMember(i); + return; + } + } + } } - private void LoadZipFile(string path, string[] recognizedExtensions) - { - zipped = true; - rawFileName = path; + /// + /// is the supplied path a canonical name including an archive? + /// + bool IsCanonicalArchivePath(string path) + { + return (path.IndexOf('|') != -1); + } - using (var extractor = new SevenZip.SevenZipExtractor(path)) - { - thingToDispose = extractor; - foreach (var e in extractor.ArchiveFileData) - { - extension = Path.GetExtension(e.FileName).Substring(1).ToUpperInvariant(); + /// + /// converts a canonical name to a bound name (the bound part, whether or not it is an archive) + /// + string GetBoundNameFromCanonical(string canonical) + { + string[] parts = canonical.Split('|'); + return parts[parts.Length - 1]; + } - if (extension.In(recognizedExtensions)) - { - // we found our match. - name = Path.GetFileNameWithoutExtension(e.FileName); - zippedStream = new MemoryStream(); - //e.Extract(zippedStream); - extractor.ExtractFile(e.Index,zippedStream); - thingToDispose = zippedStream; - return; - } - } - exists = false; - } - } - - public Stream GetStream() + /// + /// makes a canonical name from two parts + /// + string MakeCanonicalName(string root, string member) + { + if (member == null) return root; + else return string.Format("{0}|{1}", root, member); + } + + void BindArchiveMember(int index) + { + boundStream = new MemoryStream(); + extractor.ExtractFile(index, boundStream); + boundStream.Position = 0; + memberPath = extractor.ArchiveFileNames[index]; + Console.WriteLine("bound " + CanonicalName); + } + + /// + /// Removes any existing binding + /// + public void Unbind() + { + if (boundStream != null && boundStream != rootStream) boundStream.Close(); + boundStream = null; + memberPath = null; + } + + void BindRoot() + { + boundStream = rootStream; + Console.WriteLine("bound " + CanonicalName); + } + + /// + /// Binds the first item in the archive (or the file itself). Supposing that there is anything in the archive. + /// + public HawkFile BindFirst() + { + BindFirstOf(); + return this; + } + + /// + /// Binds the first item in the archive (or the file itself) if the extension matches one of the supplied templates + /// + public HawkFile BindFirstOf(params string[] extensions) + { + if (!rootExists) return this; + if (boundStream != null) throw new InvalidOperationException("stream already bound!"); + + if (extractor == null) + { + //open uncompressed file + string extension = Path.GetExtension(rootPath).Substring(1).ToUpperInvariant(); + if (extensions.Length==0 || extension.In(extensions)) + { + BindRoot(); + } + return this; + } + + for(int i=0;i("config.ini"); @@ -386,93 +391,95 @@ namespace BizHawk.MultiClient private bool LoadRom(string path) { - var file = new FileInfo(path); - if (file.Exists == false) return false; - - CloseGame(); - - var game = new RomGame(path); - Global.Game = game; - - switch (game.System) + using (var file = new HawkFile(path)) { - case "SG": - case "SMS": - Global.Emulator = new SMS(); - Global.Emulator.Controller = Global.SMSControls; - if (Global.Config.SmsEnableFM) game.AddOptions("UseFM"); - if (Global.Config.SmsAllowOverlock) game.AddOptions("AllowOverclock"); - if (Global.Config.SmsForceStereoSeparation) game.AddOptions("ForceStereo"); - break; - case "GG": - Global.Emulator = new SMS { IsGameGear = true }; - Global.Emulator.Controller = Global.SMSControls; - if (Global.Config.SmsAllowOverlock) game.AddOptions("AllowOverclock"); - break; - case "PCE": - Global.Emulator = new PCEngine(NecSystemType.TurboGrafx); - Global.Emulator.Controller = Global.PCEControls; - break; - case "SGX": - Global.Emulator = new PCEngine(NecSystemType.SuperGrafx); - Global.Emulator.Controller = Global.PCEControls; - break; - case "GEN": - Global.Emulator = new Genesis(false);//TODO - Global.Emulator.Controller = Global.GenControls; - break; - case "TI83": - Global.Emulator = new TI83(); - Global.Emulator.Controller = Global.TI83Controls; - break; - case "NES": - Global.Emulator = new NES(); - Global.Emulator.Controller = Global.NESControls; - break; - case "GB": - Global.Emulator = new Gameboy(); - break; - } + if (!file.RootExists) return false; - if (Global.Emulator is NullEmulator) - { - throw new Exception(); - } - - HandlePlatformMenus(Global.Game.System); - Global.Emulator.LoadGame(game); - Text = DisplayNameForSystem(game.System) + " - " + game.Name; - ResetRewindBuffer(); - Global.Config.RecentRoms.Add(file.FullName); - if (File.Exists(game.SaveRamPath)) - LoadSaveRam(); + CloseGame(); - if (game.System == "GB") - { - new BizHawk.Emulation.Consoles.Gameboy.Debugger(Global.Emulator as Gameboy).Show(); - } - - if (InputLog.GetMovieMode() == MOVIEMODE.RECORD) - InputLog.StartNewRecording(); //TODO: Uncomment and check for a user movie selected? - else if (InputLog.GetMovieMode() == MOVIEMODE.PLAY) - { - InputLog.LoadMovie(); //TODO: Debug - InputLog.StartPlayback(); //TODO: Debug - } + var game = new RomGame(file); + Global.Game = game; - //setup the throttle based on platform's specifications - //(one day later for some systems we will need to modify it at runtime as the display mode changes) - { - object o = Global.Emulator.Query(EmulatorQuery.VsyncRate); - if (o is double) - throttle.SetCoreFps((double)o); - else throttle.SetCoreFps(60); - SetSpeedPercent(Global.Config.SpeedPercent); + switch (game.System) + { + case "SG": + case "SMS": + Global.Emulator = new SMS(); + Global.Emulator.Controller = Global.SMSControls; + if (Global.Config.SmsEnableFM) game.AddOptions("UseFM"); + if (Global.Config.SmsAllowOverlock) game.AddOptions("AllowOverclock"); + if (Global.Config.SmsForceStereoSeparation) game.AddOptions("ForceStereo"); + break; + case "GG": + Global.Emulator = new SMS { IsGameGear = true }; + Global.Emulator.Controller = Global.SMSControls; + if (Global.Config.SmsAllowOverlock) game.AddOptions("AllowOverclock"); + break; + case "PCE": + Global.Emulator = new PCEngine(NecSystemType.TurboGrafx); + Global.Emulator.Controller = Global.PCEControls; + break; + case "SGX": + Global.Emulator = new PCEngine(NecSystemType.SuperGrafx); + Global.Emulator.Controller = Global.PCEControls; + break; + case "GEN": + Global.Emulator = new Genesis(false);//TODO + Global.Emulator.Controller = Global.GenControls; + break; + case "TI83": + Global.Emulator = new TI83(); + Global.Emulator.Controller = Global.TI83Controls; + break; + case "NES": + Global.Emulator = new NES(); + Global.Emulator.Controller = Global.NESControls; + break; + case "GB": + Global.Emulator = new Gameboy(); + break; + } + + if (Global.Emulator is NullEmulator) + { + throw new Exception(); + } + + HandlePlatformMenus(Global.Game.System); + Global.Emulator.LoadGame(game); + Text = DisplayNameForSystem(game.System) + " - " + game.Name; + ResetRewindBuffer(); + Global.Config.RecentRoms.Add(file.CanonicalName); + if (File.Exists(game.SaveRamPath)) + LoadSaveRam(); + + if (game.System == "GB") + { + new BizHawk.Emulation.Consoles.Gameboy.Debugger(Global.Emulator as Gameboy).Show(); + } + + if (InputLog.GetMovieMode() == MOVIEMODE.RECORD) + InputLog.StartNewRecording(); //TODO: Uncomment and check for a user movie selected? + else if (InputLog.GetMovieMode() == MOVIEMODE.PLAY) + { + InputLog.LoadMovie(); //TODO: Debug + InputLog.StartPlayback(); //TODO: Debug + } + + //setup the throttle based on platform's specifications + //(one day later for some systems we will need to modify it at runtime as the display mode changes) + { + object o = Global.Emulator.Query(EmulatorQuery.VsyncRate); + if (o is double) + throttle.SetCoreFps((double)o); + else throttle.SetCoreFps(60); + SetSpeedPercent(Global.Config.SpeedPercent); + } + RamSearch1.Restart(); + HexEditor1.Restart(); + CurrentlyOpenRom = path; + return true; } - RamSearch1.Restart(); - HexEditor1.Restart(); - CurrentlyOpenRom = path; - return true; } private void LoadSaveRam() diff --git a/BizHawk.MultiClient/RomGame.cs b/BizHawk.MultiClient/RomGame.cs index 67a7306e6f..5b6d2326ab 100644 --- a/BizHawk.MultiClient/RomGame.cs +++ b/BizHawk.MultiClient/RomGame.cs @@ -14,43 +14,44 @@ namespace BizHawk.MultiClient private List options; private const int BankSize = 4096; - public RomGame(string path) : this(path, null){} + public RomGame(HawkFile file) : this(file, null){} - public RomGame(string path, string patch) + public RomGame(HawkFile file, string patch) { - using (var file = new HawkFile(path)) - { - if (!file.Exists) - throw new Exception("The file needs to exist, yo."); + if(!file.IsBound) + file.BindFirstOf("SMS", "PCE", "SGX", "GG", "SG", "BIN", "SMD", "GB", "NES"); + if (!file.Exists) + throw new Exception("The file needs to exist, yo."); - var stream = file.GetStream(); + var stream = file.GetStream(); - FileData = Util.ReadAllBytes(stream); + FileData = Util.ReadAllBytes(stream); - int header = (int)(stream.Length % BankSize); - stream.Position = header; - int length = (int)stream.Length - header; + int header = (int)(stream.Length % BankSize); + stream.Position = header; + int length = (int)stream.Length - header; - RomData = new byte[length]; - stream.Read(RomData, 0, length); + RomData = new byte[length]; + stream.Read(RomData, 0, length); - if (file.Extension == "SMD") - RomData = DeInterleaveSMD(RomData); + if (file.Extension == "SMD") + RomData = DeInterleaveSMD(RomData); - var info = Database.GetGameInfo(RomData, file.FullName); - name = info.Name; - System = info.System; - options = new List(info.GetOptions()); - CheckForPatchOptions(); - } + var info = Database.GetGameInfo(RomData, file.Name); + name = info.Name; + System = info.System; + options = new List(info.GetOptions()); + CheckForPatchOptions(); - if (patch != null) - { - using (var stream = new HawkFile(patch).GetStream()) - { - IPS.Patch(RomData, stream); - } - } + if (patch != null) + { + using (var patchFile = new HawkFile(patch)) + { + patchFile.BindFirstOf("IPS"); + if(patchFile.Exists) + IPS.Patch(RomData, patchFile.GetStream()); + } + } } public void AddOptions(params string[] options) diff --git a/BizHawk.Util/7z/FileSignatureChecker.cs b/BizHawk.Util/7z/FileSignatureChecker.cs index 294cdc95d6..7f89935697 100644 --- a/BizHawk.Util/7z/FileSignatureChecker.cs +++ b/BizHawk.Util/7z/FileSignatureChecker.cs @@ -24,8 +24,10 @@ namespace SevenZip /// The signature checker class. Original code by Siddharth Uppal, adapted by Markhor. /// /// Based on the code at http://blog.somecreativity.com/2008/04/08/how-to-check-if-a-file-is-compressed-in-c/# - internal static class FileChecker + public static class FileChecker { + public static bool ThrowExceptions = true; + private const int SIGNATURE_SIZE = 16; private const int SFX_SCAN_LENGTH = 256 * 1024; @@ -69,13 +71,19 @@ namespace SevenZip public static InArchiveFormat CheckSignature(Stream stream, out int offset, out bool isExecutable) { offset = 0; + isExecutable = false; + if (!stream.CanRead) { - throw new ArgumentException("The stream must be readable."); + if (ThrowExceptions) + throw new ArgumentException("The stream must be readable."); + else return InArchiveFormat.None; } if (stream.Length < SIGNATURE_SIZE) { - throw new ArgumentException("The stream is invalid."); + if (ThrowExceptions) + throw new ArgumentException("The stream is invalid."); + else return InArchiveFormat.None; } #region Get file signature @@ -207,8 +215,10 @@ namespace SevenZip } } #endregion - - throw new ArgumentException("The stream is invalid or no corresponding signature was found."); + + if (ThrowExceptions) + throw new ArgumentException("The stream is invalid or no corresponding signature was found."); + else return InArchiveFormat.None; } /// @@ -225,14 +235,16 @@ namespace SevenZip { try { - return CheckSignature(fs, out offset, out isExecutable); + InArchiveFormat format = CheckSignature(fs, out offset, out isExecutable); + if (format != InArchiveFormat.None) return format; } catch (ArgumentException) { - offset = 0; - isExecutable = false; - return Formats.FormatByFileName(fileName, true); - } + } + + offset = 0; + isExecutable = false; + return Formats.FormatByFileName(fileName, true); } } } diff --git a/BizHawk.Util/7z/Formats.cs b/BizHawk.Util/7z/Formats.cs index 3c7acb859e..8c35af13fe 100644 --- a/BizHawk.Util/7z/Formats.cs +++ b/BizHawk.Util/7z/Formats.cs @@ -186,7 +186,11 @@ namespace SevenZip /// Microsoft virtual hard disk file format. /// /// Wikipedia information - Vhd + Vhd, + /// + /// Not an archive + /// + None } #if COMPRESS @@ -514,13 +518,15 @@ namespace SevenZip { if (String.IsNullOrEmpty(fileName) && reportErrors) { - throw new ArgumentException("File name is null or empty string!"); + throw new ArgumentException("File name is null or empty string!"); } string extension = Path.GetExtension(fileName).Substring(1); if (!InExtensionFormats.ContainsKey(extension) && reportErrors) { - throw new ArgumentException("Extension \"" + extension + - "\" is not a supported archive file name extension."); + if (FileChecker.ThrowExceptions) + throw new ArgumentException("Extension \"" + extension + "\" is not a supported archive file name extension."); + else return InArchiveFormat.None; + } return InExtensionFormats[extension]; }