···
+
use crate::enums::CartridgeHeaderAddress::OldLicenseeCode;
+
CGBFlag, CartridgeHeaderAddress, CartridgeType, DestinationCode, Error, RamSize, RomSize,
+
// https://github.com/ISSOtm/gb-bootroms/blob/2dce25910043ce2ad1d1d3691436f2c7aabbda00/src/dmg.asm#L259-L269
+
// Each tile is encoded using 2 (!) bytes
+
// The tiles are represented below
+
// XX.. .XX. XX.. .... .... .... .... .... .... ...X X... ....
+
// XXX. .XX. XX.. .... ..XX .... .... .... .... ...X X... ....
+
// XXX. .XX. .... .... .XXX X... .... .... .... ...X X... ....
+
// XX.X .XX. XX.X X.XX ..XX ..XX XX.. XX.X X... XXXX X..X XXX.
+
// XX.X .XX. XX.X XX.X X.XX .XX. .XX. XXX. XX.X X..X X.XX ..XX
+
// XX.. XXX. XX.X X..X X.XX .XXX XXX. XX.. XX.X X..X X.XX ..XX
+
// XX.. XXX. XX.X X..X X.XX .XX. .... XX.. XX.X X..X X.XX ..XX
+
// XX.. .XX. XX.X X..X X.XX ..XX XXX. XX.. XX.. XXXX X..X XXX.
+
const NINTENDO_LOGO: [u8; 48] = [
+
0x0CE, 0x0ED, 0x066, 0x066, 0x0CC, 0x00D, 0x000, 0x00B, 0x003, 0x073, 0x000, 0x083, 0x000,
+
0x00C, 0x000, 0x00D, 0x000, 0x008, 0x011, 0x01F, 0x088, 0x089, 0x000, 0x00E, 0x0DC, 0x0CC,
+
0x06E, 0x0E6, 0x0DD, 0x0DD, 0x0D9, 0x099, 0x0BB, 0x0BB, 0x067, 0x063, 0x06E, 0x00E, 0x0EC,
+
0x0CC, 0x0DD, 0x0DC, 0x099, 0x09F, 0x0BB, 0x0B9, 0x033, 0x03E,
+
// const ROM_NAME: &str = "OtherLegallyObtainedRom.gb";
const ROM_NAME: &str = "LegallyObtainedRom.gb";
//Should be 80 bytes (0x014F(335) - 0x0100(256)) + 1 to include the last address
+
manufacturer_code: [char; 4],
+
// https://gbdev.io/pandocs/The_Cartridge_Header.html#01440145--new-licensee-code
+
license_code: [char; 2],
+
cart_type: CartridgeType,
+
old_licensee_code: Option<u8>,
+
destination_code: DestinationCode,
+
fn bytes_to_chars<const N: usize>(bytes: &[u8], break_on_null: bool) -> [char; N] {
+
let mut chars = [0x000 as char; N];
+
for (i, byte) in bytes.iter().enumerate() {
+
if break_on_null && *byte == 0x00 {
+
chars[i] = *byte as char;
+
fn parse(rom_bytes: &[u8]) -> Result<Self, Error> {
+
let start = CartridgeHeaderAddress::CartridgeHeaderStart as usize;
+
let header_buffer: &[u8] = rom_bytes[start..end]
+
.map_err(|_| Error::CartridgeReadError)?;
+
//Checks if the Nintendo logo matches the device's version. Early anti piracy feature. Neat to take note of
+
let nintendo_logo_from_rom = &rom_bytes[CartridgeHeaderAddress::NintendoLogoStart as usize
+
..CartridgeHeaderAddress::NintendoLogoEnd as usize + 1];
+
for (i, true_logo_byte) in NINTENDO_LOGO.iter().enumerate() {
+
let rom_byte = nintendo_logo_from_rom[i];
+
if rom_byte != *true_logo_byte {
+
return Err(Error::CartridgeReadError);
+
let title = &rom_bytes[CartridgeHeaderAddress::TitleStart as usize
+
..CartridgeHeaderAddress::TitleEnd as usize + 1];
+
let title_chars = Self::bytes_to_chars::<16>(title, true);
+
let manufacturer_code = &rom_bytes[CartridgeHeaderAddress::ManufacturerCodeStart as usize
+
..CartridgeHeaderAddress::ManufacturerCodeEnd as usize + 1];
+
let man_code_chars = Self::bytes_to_chars::<4>(manufacturer_code, false);
+
let cgb_flag_byte = rom_bytes[0x0143];
+
let cgb_flag = match cgb_flag_byte {
+
0x80 => CGBFlag::SupportsColorBackwardCompatiable,
+
0xC0 => CGBFlag::ColorOnly,
+
let license_code = &rom_bytes[CartridgeHeaderAddress::NewLicenseStart as usize
+
..CartridgeHeaderAddress::NewLicenseEnd as usize + 1];
+
let license_code_chars = Self::bytes_to_chars::<2>(license_code, false);
+
// https://gbdev.io/pandocs/The_Cartridge_Header.html#0146--sgb-flag
+
let sgb_flag = rom_bytes[CartridgeHeaderAddress::SgbFlag as usize] == 0x03;
+
CartridgeType::from_byte(rom_bytes[CartridgeHeaderAddress::CartType as usize]);
+
let rom_size = RomSize::from_byte(rom_bytes[CartridgeHeaderAddress::RomSize as usize]);
+
let ram_size = RamSize::from_byte(rom_bytes[CartridgeHeaderAddress::RamSize as usize]);
+
let mut old_licensee_code = None;
+
old_licensee_code = Some(rom_bytes[CartridgeHeaderAddress::OldLicenseeCode as usize])
+
DestinationCode::from_byte(rom_bytes[CartridgeHeaderAddress::DestinationCode as usize]);
+
let version = rom_bytes[CartridgeHeaderAddress::MaskRomVersion as usize];
+
let header_checksum = rom_bytes[CartridgeHeaderAddress::HeaderChecksum as usize];
+
let global_checksum = u16::from_le_bytes([
+
rom_bytes[CartridgeHeaderAddress::GlobalChecksumStart as usize],
+
rom_bytes[CartridgeHeaderAddress::CartridgeHeaderEnd as usize],
+
.map_err(|_| Error::CartridgeReadError)?,
+
manufacturer_code: man_code_chars,
+
license_code: license_code_chars,
···
fn main() -> std::io::Result<()> {
let mut rom_file = File::open(ROM_NAME)?;
let mut rom_buffer: Vec<u8> = Vec::new();
rom_file.read_to_end(&mut rom_buffer)?;
+
let cart_header = match CartridgeHeader::parse(&*rom_buffer) {
+
return Err(std::io::Error::new(
+
std::io::ErrorKind::Other,
+
// cart_header.print_test();
+
let title: String = String::from_iter(cart_header.title);
+
println!("Title: {}", title);
+
let manufacturer_code = String::from_iter(cart_header.manufacturer_code);
+
println!("Manufacturer Code: {}", manufacturer_code);
+
match cart_header.old_licensee_code {
+
Some(code) => println!("Uses Old Licensee Code: {:#X}", code),
+
"Uses New Licensee Code: {}",
+
String::from_iter(cart_header.license_code)
+
println!("CGB Flag: {:?}", cart_header.cgb_flag);
+
println!("Supports SGB: {:?}", cart_header.support_gb);
+
println!("Cartridge Type: {:?}", cart_header.cart_type);
+
println!("ROM Size: {:?}", cart_header.rom_size);
+
println!("RAM Size: {:?}", cart_header.ram_size);
+
match cart_header.destination_code {
+
DestinationCode::Japan => println!("Destination Code: Japan"),
+
DestinationCode::NotJapan => println!("Destination Code: Not Japan"),
+
println!("Version: {:?}", cart_header.version);
+
println!("Header Checksum: {:#X}", cart_header.header_checksum);
+
println!("Global Checksum: {:#X}", cart_header.global_checksum);