From bd8003c725873d834590c0db16075de0e8c91689 Mon Sep 17 00:00:00 2001 From: zeromus Date: Sun, 7 Jun 2020 16:43:50 -0500 Subject: [PATCH] thread gamedb load --- src/BizHawk.Client.EmuHawk/MainForm.cs | 4 +- .../Database/Database.cs | 87 ++++++++++++------- 2 files changed, 59 insertions(+), 32 deletions(-) diff --git a/src/BizHawk.Client.EmuHawk/MainForm.cs b/src/BizHawk.Client.EmuHawk/MainForm.cs index bd66fdaa59..d91fb64cf7 100644 --- a/src/BizHawk.Client.EmuHawk/MainForm.cs +++ b/src/BizHawk.Client.EmuHawk/MainForm.cs @@ -275,6 +275,8 @@ namespace BizHawk.Client.EmuHawk CloseRomContextMenuItem.Image = Properties.Resources.Close; } + Database.InitializeDatabase(Path.Combine(PathUtils.ExeDirectoryPath, "gamedb", "gamedb.txt")); + GlobalWin.MainForm = this; Rewinder = new Rewinder(); @@ -327,8 +329,6 @@ namespace BizHawk.Client.EmuHawk MessageBox.Show(e.Message); } - Database.LoadDatabase(Path.Combine(PathUtils.ExeDirectoryPath, "gamedb", "gamedb.txt")); - // TODO GL - a lot of disorganized wiring-up here // installed separately on Unix (via package manager or from https://developer.nvidia.com/cg-toolkit-download), look in $PATH CGC.CGCBinPath = OSTailoredCode.IsUnixHost ? "cgc" : Path.Combine(PathUtils.DllDirectoryPath, "cgc.exe"); diff --git a/src/BizHawk.Emulation.Common/Database/Database.cs b/src/BizHawk.Emulation.Common/Database/Database.cs index e528d35896..5bee5fed82 100644 --- a/src/BizHawk.Emulation.Common/Database/Database.cs +++ b/src/BizHawk.Emulation.Common/Database/Database.cs @@ -13,7 +13,9 @@ namespace BizHawk.Emulation.Common { public static class Database { - private static readonly Dictionary DB = new Dictionary(); + private static Dictionary DB; + + static EventWaitHandle acquire = new EventWaitHandle(false, EventResetMode.ManualReset); private static string RemoveHashType(string hash) { @@ -31,19 +33,6 @@ namespace BizHawk.Emulation.Common return hash; } - public static GameInfo CheckDatabase(string hash) - { - var hashNoType = RemoveHashType(hash); - DB.TryGetValue(hashNoType, out var cgi); - if (cgi == null) - { - Console.WriteLine($"DB: hash {hash} not in game database."); - return null; - } - - return new GameInfo(cgi); - } - private static void LoadDatabase_Escape(string line, string path) { if (!line.ToUpperInvariant().StartsWith("#INCLUDE")) @@ -56,7 +45,7 @@ namespace BizHawk.Emulation.Common if (File.Exists(filename)) { Debug.WriteLine("loading external game database {0}", line); - LoadDatabase(filename); + InitializeDatabase(filename); } else { @@ -97,9 +86,15 @@ namespace BizHawk.Emulation.Common File.AppendAllText(path, sb.ToString()); } - public static void LoadDatabase(string path) + static bool initialized = false; + + static void initializeWork(string path) { - using var reader = new StreamReader(new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.ReadWrite)); + //reminder: this COULD be done on several threads, if it takes even longer + Dictionary db = new Dictionary(); + var stopwatch = Stopwatch.StartNew(); + + using var reader = new StreamReader(new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.Read)); while (reader.EndOfStream == false) { var line = reader.ReadLine() ?? ""; @@ -129,17 +124,17 @@ namespace BizHawk.Emulation.Common // remove a hash type identifier. well don't really need them for indexing (they're just there for human purposes) Status = items[1].Trim() switch - { - "B" => RomStatus.BadDump, - "V" => RomStatus.BadDump, - "T" => RomStatus.TranslatedRom, - "O" => RomStatus.Overdump, - "I" => RomStatus.Bios, - "D" => RomStatus.Homebrew, - "H" => RomStatus.Hack, - "U" => RomStatus.Unknown, - _ => RomStatus.GoodDump - }, + { + "B" => RomStatus.BadDump, + "V" => RomStatus.BadDump, + "T" => RomStatus.TranslatedRom, + "O" => RomStatus.Overdump, + "I" => RomStatus.Bios, + "D" => RomStatus.Homebrew, + "H" => RomStatus.Hack, + "U" => RomStatus.Unknown, + _ => RomStatus.GoodDump + }, Name = items[2], System = items[3], MetaData = items.Length >= 6 ? items[5] : null, @@ -148,23 +143,55 @@ namespace BizHawk.Emulation.Common }; #if DEBUG - if (DB.ContainsKey(game.Hash)) + if (db.ContainsKey(game.Hash)) { Console.WriteLine("gamedb: Multiple hash entries {0}, duplicate detected on \"{1}\" and \"{2}\"", game.Hash, game.Name, DB[game.Hash].Name); } #endif - DB[game.Hash] = game; + db[game.Hash] = game; } catch { Debug.WriteLine($"Error parsing database entry: {line}"); } } + + //commit the finished database load + //it's left as null until now to help catch mistakes in using the resource + DB = db; + acquire.Set(); + + Console.WriteLine("GameDB load: " + stopwatch.Elapsed + " sec"); + } + + public static void InitializeDatabase(string path) + { + if (initialized) throw new InvalidOperationException("Did not expect re-initialize of game Database"); + initialized = true; + + ThreadPool.QueueUserWorkItem(_=>initializeWork(path)); + } + + public static GameInfo CheckDatabase(string hash) + { + acquire.WaitOne(); + + var hashNoType = RemoveHashType(hash); + DB.TryGetValue(hashNoType, out var cgi); + if (cgi == null) + { + Console.WriteLine($"DB: hash {hash} not in game database."); + return null; + } + + return new GameInfo(cgi); } public static GameInfo GetGameInfo(byte[] romData, string fileName) { + acquire.WaitOne(); + var hash = $"{CRC32.Calculate(romData):X8}"; if (DB.TryGetValue(hash, out var cgi)) {