auto Icarus::megaDriveManifest(string location) -> string { vector buffer; concatenate(buffer, {location, "program.rom"}); concatenate(buffer, {location, "patch.rom" }); return megaDriveManifest(buffer, location); } auto Icarus::megaDriveManifest(vector& buffer, string location) -> string { if(settings["icarus/UseDatabase"].boolean()) { auto digest = Hash::SHA256(buffer).digest(); for(auto game : Database::MegaDrive.find("game")) { if(game["sha256"].text() == digest) return BML::serialize(game); } } if(settings["icarus/UseHeuristics"].boolean()) { Heuristics::MegaDrive game{buffer, location}; if(auto manifest = game.manifest()) return manifest; } return {}; } auto Icarus::megaDriveImport(vector& buffer, string location) -> string { auto name = Location::prefix(location); auto source = Location::path(location); string target{settings["Library/Location"].text(), "Mega Drive/", name, ".md/"}; auto manifest = megaDriveManifest(buffer, location); if(!manifest) return failure("failed to parse ROM image"); if(!create(target)) return failure("library path unwritable"); if(exists({source, name, ".sav"}) && !exists({target, "save.ram"})) { copy({source, name, ".sav"}, {target, "save.ram"}); } if(settings["icarus/CreateManifests"].boolean()) write({target, "manifest.bml"}, manifest); uint offset = 0; auto document = BML::unserialize(manifest); for(auto rom : document.find("game/board/memory(type=ROM)")) { auto name = string{rom["content"].text(), ".rom"}.downcase(); auto size = rom["size"].natural(); if(size > buffer.size() - offset) { missingFiles.append(name); continue; } write({target, name}, buffer.data() + offset, size); offset += size; } if(missingFiles) return failure({"ROM image is missing data: ", missingFiles.merge("; ")}); return success(target); }