[NES] restore iNES header detection

This commit is contained in:
zeromus 2011-03-19 20:12:06 +00:00
parent c7f0c457ad
commit b041d31ab1
6 changed files with 145 additions and 88 deletions

View File

@ -202,7 +202,7 @@ namespace BizHawk.Emulation.Consoles.Nintendo
public override string ToString()
{
return string.Format("r={0},vr={1},pr={2},cr={3},ba={4},pa={5},{6},brd={7},map={8},sys={9}", prg_size, chr_size, wram_size, vram_size, wram_battery, pad_h, pad_v, board_type, mapper, system);
return string.Format("pr={0},cr={1},wr={2},vr={3},ba={4},pa={5}|{6},brd={7},map={8},sys={9}", prg_size, chr_size, wram_size, vram_size, wram_battery?1:0, pad_h, pad_v, board_type, mapper, system);
}
}
@ -275,6 +275,8 @@ namespace BizHawk.Emulation.Consoles.Nintendo
cart.vram_size = short.Parse(dict["VRAM"]);
if (dict.ContainsKey("WRAM"))
cart.wram_size = short.Parse(dict["WRAM"]);
if (dict.ContainsKey("bad"))
Console.WriteLine("rom is flagged as BAD!");
return cart;
}
@ -416,53 +418,6 @@ namespace BizHawk.Emulation.Consoles.Nintendo
return sha1_table[sha1];
}
}
// static class BoardDetector
// {
// public static string Detect(RomInfo romInfo)
// {
// string key = string.Format("{0} {1} {2} {3}",romInfo.MapperNumber,romInfo.PRG_Size,romInfo.CHR_Size,romInfo.PRAM_Size);
// string board;
// Table.TryGetValue(key, out board);
// return board;
// }
// public static Dictionary<string,string> Table = new Dictionary<string,string>();
// static BoardDetector()
// {
// var sr = new StringReader(ClassifyTable);
// string line;
// while ((line = sr.ReadLine()) != null)
// {
// var parts = line.Split('\t');
// if (parts.Length < 5) continue;
// string key = parts[0] + "\t" + parts[1] + "\t" + parts[2] + "\t" + parts[3];
// string board = line.Replace(key, "");
// board = board.TrimStart('\t');
// if (board.IndexOf(';') != -1)
// board = board.Substring(0, board.IndexOf(';'));
// Table[key] = board;
// }
// }
////MAP PRG CHR PRAM BOARD
// static string ClassifyTable = @"
//0 1 1 0 NROM
//0 2 1 0 NROM
//1 8 0 8 SNROM; this handles zelda,
//2 8 0 0 UNROM
//2 16 0 0 UOROM
//3 2 2 0 CNROM
//3 2 4 0 CNROM
//7 8 0 0 ANROM
//7 16 0 0 AOROM
//11 4 2 0 Discrete_74x377
//11 2 4 0 Discrete_74x377
//13 2 0 0 CPROM
//66 4 2 0 GxROM
//66 8 4 0 GxROM
//";
// }
}
}

View File

@ -42,6 +42,8 @@ namespace BizHawk.Emulation.Consoles.Nintendo
chr_mask = (Cart.chr_size / 8) - 1;
SetMirrorType(Cart.pad_h, Cart.pad_v);
Console.WriteLine("Caution! If this board was inferred from iNES then the mirr.type might be wrong");
return true;
}

View File

@ -17,6 +17,7 @@ namespace BizHawk.Emulation.Consoles.Nintendo
switch (Cart.board_type)
{
case "HVC-NROM-256": //super mario bros.
case "NES-NROM-256": //10 yard fight
AssertPrg(32); AssertChr(8); AssertVram(0); AssertWram(0);
break;

View File

@ -13,6 +13,8 @@ namespace BizHawk.Emulation.Consoles.Nintendo
public partial class NES : IEmulator
{
static readonly bool USE_DATABASE = true;
//Game issues:
//3-D World Runner - UNROM - weird lines in gameplay (scanlines off?)
//JJ - Tobidase Daisakusen Part 2 (J) - same as 3-D World Runner
@ -371,6 +373,11 @@ namespace BizHawk.Emulation.Consoles.Nintendo
public string GameName { get { return game_name; } }
enum EDetectionOrigin
{
None, BootGodDB, GameDB, INES
}
public unsafe void LoadGame(IGame game)
{
byte[] file = game.GetFileData();
@ -379,6 +386,8 @@ namespace BizHawk.Emulation.Consoles.Nintendo
throw new Exception("You've tried to open a UNIF rom. We don't have any UNIF roms to test with. Please consult the developers.");
fixed (byte* bfile = &file[0])
{
var origin = EDetectionOrigin.None;
var header = (iNES_HEADER*)bfile;
if (!header->CheckID()) throw new InvalidOperationException("iNES header not found");
header->Cleanup();
@ -399,17 +408,36 @@ namespace BizHawk.Emulation.Consoles.Nintendo
Console.WriteLine("headerless rom hash: {0}", hash_sha1);
Console.WriteLine("headerless rom hash: {0}", hash_md5);
CartInfo choice = IdentifyFromBootGodDB(hash_sha1);
CartInfo choice = null;
if(USE_DATABASE)
choice = IdentifyFromBootGodDB(hash_sha1);
if(choice == null)
{
choice = IdentifyFromGameDB(hash_md5);
if (choice == null)
choice = IdentifyFromGameDB(hash_sha1);
if (choice == null)
throw new Exception("couldnt identify");
if (USE_DATABASE)
{
choice = IdentifyFromGameDB(hash_md5);
if (choice == null)
choice = IdentifyFromGameDB(hash_sha1);
}
if (choice == null)
{
Console.WriteLine("Attempting inference from iNES header");
choice = header->Analyze();
string iNES_board = iNESBoardDetector.Detect(choice);
if (iNES_board == null)
throw new Exception("couldnt identify NES rom");
Console.WriteLine("trying board " + iNES_board);
choice.board_type = iNES_board;
choice.game.name = game.Name;
origin = EDetectionOrigin.INES;
}
else
{
origin = EDetectionOrigin.GameDB;
Console.WriteLine("Chose board from gamedb: ");
}
}
else origin = EDetectionOrigin.BootGodDB;
Console.WriteLine(choice.game);
Console.WriteLine(choice);
@ -418,11 +446,31 @@ namespace BizHawk.Emulation.Consoles.Nintendo
game_name = choice.game.name;
//find a INESBoard to handle this
Type boardType = FindBoard(choice);
if (boardType == null)
Type boardType = null;
bool iNES_tryAgain = false;
try
{
throw new Exception("No class implements the necessary board type: " + choice.board_type);
boardType = FindBoard(choice);
if (boardType == null)
iNES_tryAgain = true;
}
catch(Exception)
{
if (origin == EDetectionOrigin.INES)
iNES_tryAgain = true;
else throw;
}
if (iNES_tryAgain)
{
//try again with a different wram size.. because iNES sucks that way
choice.wram_size = 8;
Console.WriteLine("Trying classification again with iNES wram adjustment. new parameters:");
Console.WriteLine(choice);
boardType = FindBoard(choice);
}
if (boardType == null)
throw new Exception("No class implements the necessary board type: " + choice.board_type);
board = (INESBoard)Activator.CreateInstance(boardType);
cart = choice;

View File

@ -10,13 +10,61 @@ namespace BizHawk.Emulation.Consoles.Nintendo
{
partial class NES
{
//public class RomHeaderInfo
//{
// public int MapperNo, Mirroring, Num_PRG_Banks, Num_CHR_Banks, Num_PRAM_Banks;
// public bool Battery;
// public byte[] ROM, VROM;
//}
/// <summary>
/// attempts to classify a rom based on iNES header information
/// </summary>
static class iNESBoardDetector
{
public static string Detect(CartInfo cartInfo)
{
string key = string.Format("{0} {1} {2} {3} {4}", cartInfo.mapper, cartInfo.prg_size, cartInfo.chr_size, cartInfo.wram_size, cartInfo.vram_size);
string board;
Table.TryGetValue(key, out board);
if (board == null)
{
//if it didnt work, try again with a different wram size. because iNES is weird that way
key = string.Format("{0} {1} {2} {3} {4}", cartInfo.mapper, cartInfo.prg_size, cartInfo.chr_size, 8, cartInfo.vram_size);
Table.TryGetValue(key, out board);
}
return board;
}
public static Dictionary<string, string> Table = new Dictionary<string, string>();
static iNESBoardDetector()
{
var sr = new StringReader(ClassifyTable);
string line;
while ((line = sr.ReadLine()) != null)
{
var parts = line.Split('\t');
if (parts.Length < 6) continue;
string key = parts[0] + "\t" + parts[1] + "\t" + parts[2] + "\t" + parts[3] + "\t" + parts[4];
string board = line.Replace(key, "");
board = board.TrimStart('\t');
if (board.IndexOf(';') != -1)
board = board.Substring(0, board.IndexOf(';'));
Table[key] = board;
}
}
//MAP PRG CHR WRAM VRAM BOARD
static string ClassifyTable = @"
0 16 8 0 0 NES-NROM-128; balloon fight, but its broken right now
0 32 8 0 0 NES-NROM-256; super mario bros
1 32 32 0 0 NES-SEROM; lolo
1 128 0 8 0 NES-SNROM; zelda
2 128 0 0 0 NES-UNROM; mega man
2 256 0 0 0 NES-UOROM; paperboy 2
3 32 32 0 0 NES-CNROM; adventure island
7 128 0 0 0 NES-ANROM; marble madness
7 256 0 0 8 NES-AOROM; battletoads
11 32 16 0 0 Discrete_74x377
11 16 32 0 0 Discrete_74x377
13 32 0 0 16 NES-CPROM; videomation
66 64 16 0 0 NES-MHROM; super mario bros / duck hunt
66 128 32 0 0 NES-GNROM; gumshoe
";
}
unsafe struct iNES_HEADER
{
@ -58,34 +106,37 @@ namespace BizHawk.Emulation.Consoles.Nintendo
}
}
//public RomInfo Analyze()
//{
// var ret = new RomInfo();
// ret.MapperNumber = (ROM_type >> 4);
// ret.MapperNumber |= (ROM_type2 & 0xF0);
// int mirroring = (ROM_type & 1);
// if ((ROM_type & 8) != 0) mirroring = 2;
// if (mirroring == 0) ret.MirrorType = EMirrorType.Horizontal;
// else if (mirroring == 1) ret.MirrorType = EMirrorType.Vertical;
// else ret.MirrorType = EMirrorType.External;
// ret.PRG_Size = ROM_size;
// if (ret.PRG_Size == 0)
// ret.PRG_Size = 256;
// ret.CHR_Size = VROM_size;
// ret.Battery = (ROM_type & 2) != 0;
public CartInfo Analyze()
{
var ret = new CartInfo();
ret.game = new GameInfo();
int mapper = (ROM_type >> 4);
mapper |= (ROM_type2 & 0xF0);
ret.mapper = (byte)mapper;
int mirroring = (ROM_type & 1);
if ((ROM_type & 8) != 0) mirroring = 2;
if (mirroring == 0) ret.pad_v = 1;
else if (mirroring == 1) ret.pad_h = 1;
ret.prg_size = (short)(ROM_size * 16);
if (ret.prg_size == 0)
ret.prg_size = 256 * 16;
ret.chr_size = (short)(VROM_size * 8);
ret.wram_battery = (ROM_type & 2) != 0;
// fixed (iNES_HEADER* self = &this) ret.PRAM_Size = self->reserve[0] * 8;
// //0 is supposed to mean 1 (for compatibility, as this is an extension to original iNES format)
// if (ret.PRAM_Size == 0) ret.PRAM_Size = 8;
fixed (iNES_HEADER* self = &this) ret.wram_size = (short)(self->reserve[0] * 8);
//0 is supposed to mean 1 (for compatibility, as this is an extension to original iNES format)
//but we'll try using 8 later if it doesn't work with 0
// Console.WriteLine("iNES header: map:{0}, mirror:{1}, PRG:{2}, CHR:{3}, CRAM:{4}, PRAM:{5}, bat:{6}", ret.MapperNumber, ret.MirrorType, ret.PRG_Size, ret.CHR_Size, ret.CRAM_Size, ret.PRAM_Size, ret.Battery ? 1 : 0);
//iNES wants us to assume that no chr -> 8KB vram
if (ret.chr_size == 0) ret.vram_size = 8;
// //fceux calls uppow2(PRG_Banks) here, and also ups the chr size as well
// //then it does something complicated that i don't understand with making sure it doesnt read too much data
// //fceux only allows this condition for mappers in the list "not_power2" which is only 228
//let's not put a lot of hacks in here. that's what the databases are for.
//for example of one not to add: videomation hack to change vram = 8 -> 16
// return ret;
//}
Console.WriteLine("iNES: map:{0}, mirror:{1}, PRG:{2}, CHR:{3}, WRAM:{4}, VRAM:{5}, bat:{6}", ret.mapper, mirroring, ret.prg_size, ret.chr_size, ret.wram_size, ret.vram_size, ret.wram_battery ? 1 : 0);
return ret;
}
}
}

View File

@ -2267,7 +2267,7 @@ sha1:22E6986903141495BA4589AC65982F3FB4D0E37B Adventures of Lolo (U) NES board
sha1:CF655333DCE649A3C7060E9989860F2FC74E473A Demon Sword (U) NES board=NES-SL1ROM;PRG=128;CHR=128
sha1:7786BA1FE8E7E9E542EEB13CF2A6E2A1AD7F696D Metal Gear (U) NES board=KONAMI-UNROM;PRG=128
sha1:894F20405286F5F75133CE4648300E2C67972B40 Solomon's Key (U) NES board=NES-CNROM;PRG=32;CHR=32
sha1:0C53B06E1D13AE917536BB39010914EA3D111FF5 Thunder & Lightning (U) NES board=NES-GNROM;PRG=128;CHR=32
sha1:0C53B06E1D13AE917536BB39010914EA3D111FF5 Thunder & Lightning (U) NES board=NES-GNROM;PRG=128;CHR=32;bad
#include gamedb_neshomebrew.txt
#include gamedb_user.txt