#include "stdafx.h" #include "N64Rom.h" #include "SystemGlobals.h" #include #include #include #include #include #include #ifdef _WIN32 #include #endif CN64Rom::CN64Rom() : m_ROMImage(nullptr), m_ROMImageBase(nullptr), m_RomFileSize(0), m_ErrorMsg(EMPTY_STRING), m_Country(Country_Unknown), m_CicChip(CIC_UNKNOWN) { } CN64Rom::~CN64Rom() { UnallocateRomImage(); } bool CN64Rom::AllocateRomImage(uint32_t RomFileSize) { WriteTrace(TraceN64System, TraceDebug, "Allocating memory for ROM"); std::unique_ptr ImageBase(new uint8_t[RomFileSize + 0x2000]); if (ImageBase.get() == nullptr) { SetError(MSG_MEM_ALLOC_ERROR); WriteTrace(TraceN64System, TraceError, "Failed to allocate memory for ROM (size: 0x%X)", RomFileSize); return false; } uint8_t * Image = (uint8_t *)(((uint64_t)ImageBase.get() + 0xFFF) & ~0xFFF); // Start at beginning of memory page WriteTrace(TraceN64System, TraceDebug, "Allocated ROM memory (%p)", Image); // Save information about the ROM loaded m_ROMImageBase = ImageBase.release(); m_ROMImage = Image; m_RomFileSize = RomFileSize; return true; } bool CN64Rom::AllocateAndLoadN64Image(const char * FileLoc, bool LoadBootCodeOnly) { WriteTrace(TraceN64System, TraceDebug, "Trying to open %s", FileLoc); if (!m_RomFile.Open(FileLoc, CFileBase::modeRead)) { SetError(MSG_ROM_FAILED_TO_OPEN); WriteTrace(TraceN64System, TraceError, "Failed to open %s", FileLoc); return false; } // Read the first 4 bytes and make sure it is a valid N64 image uint8_t Test[4]; m_RomFile.SeekToBegin(); if (m_RomFile.Read(Test, sizeof(Test)) != sizeof(Test)) { m_RomFile.Close(); SetError(MSG_ROM_FAILED_READ_IDENT); WriteTrace(TraceN64System, TraceError, "Failed to read ident bytes"); return false; } if (!IsValidRomImage(Test)) { m_RomFile.Close(); SetError(MSG_ROM_INVALID_IMAGE_FILE); WriteTrace(TraceN64System, TraceError, "Invalid image file %X %X %X %X", Test[0], Test[1], Test[2], Test[3]); return false; } uint32_t RomFileSize = m_RomFile.GetLength(); WriteTrace(TraceN64System, TraceDebug, "Successfully opened, size: 0x%X", RomFileSize); // If loading boot code then just load the first 0x1000 bytes if (LoadBootCodeOnly) { WriteTrace(TraceN64System, TraceDebug, "Loading boot code, so loading the first 0x1000 bytes", RomFileSize); RomFileSize = 0x1000; } if (!AllocateRomImage(RomFileSize)) { m_RomFile.Close(); SetError(MSG_ROM_FAILED_ROM_ALLOCATE); return false; } // Load the N64 ROM into the allocated memory g_Notify->DisplayMessage(5, MSG_LOADING); m_RomFile.SeekToBegin(); uint32_t count, TotalRead = 0; for (count = 0; count < (int)RomFileSize; count += ReadFromRomSection) { uint32_t dwToRead = RomFileSize - count; if (dwToRead > ReadFromRomSection) { dwToRead = ReadFromRomSection; } if (m_RomFile.Read(&m_ROMImage[count], dwToRead) != dwToRead) { m_RomFile.Close(); SetError(MSG_FAIL_IMAGE); WriteTrace(TraceN64System, TraceError, "Failed to read file (TotalRead: 0x%X)", TotalRead); return false; } TotalRead += dwToRead; // Show message of how much of the ROM has been loaded (as a percent) g_Notify->DisplayMessage(0, stdstr_f("%s: %.2f%c", GS(MSG_LOADED), ((float)TotalRead / (float)RomFileSize) * 100.0f, '%').c_str()); } if (RomFileSize != TotalRead) { m_RomFile.Close(); SetError(MSG_FAIL_IMAGE); WriteTrace(TraceN64System, TraceError, "Expected to read: 0x%X, read: 0x%X", TotalRead, RomFileSize); return false; } g_Notify->DisplayMessage(5, MSG_BYTESWAP); ByteSwapRom(); // Protect the memory so that it can't be written to ProtectMemory(m_ROMImage, m_RomFileSize, MEM_READONLY); return true; } bool CN64Rom::AllocateAndLoadZipImage(const char * FileLoc, bool LoadBootCodeOnly) { unzFile file = unzOpen(FileLoc); if (file == nullptr) { return false; } int port = unzGoToFirstFile(file); bool FoundRom = false; // Scan through all files in zip until a suitable file is found while (port == UNZ_OK && !FoundRom) { unz_file_info info; char zname[260]; unzGetCurrentFileInfo(file, &info, zname, sizeof(zname), nullptr, 0, nullptr, 0); if (unzLocateFile(file, zname, 1) != UNZ_OK) { SetError(MSG_FAIL_ZIP); break; } if (unzOpenCurrentFile(file) != UNZ_OK) { SetError(MSG_FAIL_ZIP); break; } // Read the first 4 bytes to check magic number uint8_t Test[4]; unzReadCurrentFile(file, Test, sizeof(Test)); if (IsValidRomImage(Test)) { // Get the size of the ROM and try to allocate the memory needed uint32_t RomFileSize = info.uncompressed_size; if (LoadBootCodeOnly) { RomFileSize = 0x1000; } if (!AllocateRomImage(RomFileSize)) { m_RomFile.Close(); return false; } // Load the N64 ROM into the allocated memory g_Notify->DisplayMessage(5, MSG_LOADING); memcpy(m_ROMImage, Test, 4); uint32_t dwRead, count, TotalRead = 0; for (count = 4; count < (int)RomFileSize; count += ReadFromRomSection) { uint32_t dwToRead = RomFileSize - count; if (dwToRead > ReadFromRomSection) { dwToRead = ReadFromRomSection; } dwRead = unzReadCurrentFile(file, &m_ROMImage[count], dwToRead); if (dwRead == 0) { SetError(MSG_FAIL_ZIP); unzCloseCurrentFile(file); break; } TotalRead += dwRead; // Show message of how much of the ROM has been loaded (as a percent) g_Notify->DisplayMessage(5, stdstr_f("%s: %.2f%c", GS(MSG_LOADED), ((float)TotalRead / (float)RomFileSize) * 100.0f, '%').c_str()); } dwRead = TotalRead + 4; if (RomFileSize != dwRead) { unzCloseCurrentFile(file); SetError(MSG_FAIL_ZIP); g_Notify->DisplayMessage(1, ""); break; } FoundRom = true; g_Notify->DisplayMessage(5, MSG_BYTESWAP); ByteSwapRom(); // Protect the memory so that it can't be written to ProtectMemory(m_ROMImage, m_RomFileSize, MEM_READONLY); } unzCloseCurrentFile(file); if (!FoundRom) { port = unzGoToNextFile(file); } } unzClose(file); return FoundRom; } void CN64Rom::ByteSwapRom() { uint32_t count; switch (*((uint32_t *)&m_ROMImage[0])) { case 0x12408037: for (count = 0; count < m_RomFileSize; count += 4) { m_ROMImage[count] ^= m_ROMImage[count + 2]; m_ROMImage[count + 2] ^= m_ROMImage[count]; m_ROMImage[count] ^= m_ROMImage[count + 2]; m_ROMImage[count + 1] ^= m_ROMImage[count + 3]; m_ROMImage[count + 3] ^= m_ROMImage[count + 1]; m_ROMImage[count + 1] ^= m_ROMImage[count + 3]; } break; case 0x40072780: // 64DD IPL case 0x40123780: for (count = 0; count < m_RomFileSize; count += 4) { m_ROMImage[count] ^= m_ROMImage[count + 3]; m_ROMImage[count + 3] ^= m_ROMImage[count]; m_ROMImage[count] ^= m_ROMImage[count + 3]; m_ROMImage[count + 1] ^= m_ROMImage[count + 2]; m_ROMImage[count + 2] ^= m_ROMImage[count + 1]; m_ROMImage[count + 1] ^= m_ROMImage[count + 2]; } break; case 0x80371240: break; default: g_Notify->DisplayError(stdstr_f("ByteSwapRom: %X", m_ROMImage[0]).c_str()); } } CICChip CN64Rom::GetCicChipID(uint8_t * RomData, uint64_t * CRC) { uint64_t crc = 0; uint64_t crcAleck64 = 0; int32_t count; for (count = 0x40; count < 0x1000; count += 4) { if (count == 0xC00) crcAleck64 = crc; //From 0x40 to 0xC00 (Aleck64) crc += *(uint32_t *)(RomData + count); } if (CRC != nullptr) { *CRC = crc; } switch (crc) { case 0x000000D0027FDF31: return CIC_NUS_6101; case 0x000000CFFB631223: return CIC_NUS_6101; case 0x000000D057C85244: return CIC_NUS_6102; case 0x000000D6497E414B: return CIC_NUS_6103; case 0x0000011A49F60E96: return CIC_NUS_6105; case 0x000000D6D5BE5580: return CIC_NUS_6106; case 0x000001053BC19870: return CIC_NUS_5167; // 64DD conversion CIC case 0x000000D2E53EF008: return CIC_NUS_8303; // 64DD IPL case 0x000000D2E53EF39F: return CIC_NUS_8401; // 64DD IPL tool case 0x000000D2E53E5DDA: return CIC_NUS_DDUS; // 64DD IPL US (different CIC) case 0x0000000AF3A34BC8: return CIC_MINI_IPL3; default: //Aleck64 CIC if (crcAleck64 == 0x000000A5F80BF620) { if (CRC != nullptr) { *CRC = crcAleck64; } return CIC_NUS_5101; } return CIC_UNKNOWN; } } void CN64Rom::CalculateCicChip() { uint64_t CRC = 0; if (m_ROMImage == nullptr) { m_CicChip = CIC_UNKNOWN; return; } m_CicChip = GetCicChipID(m_ROMImage, &CRC); if (m_CicChip == CIC_UNKNOWN) { if (HaveDebugger()) { g_Notify->DisplayError(stdstr_f("Unknown CIC checksum:\n%llX.", CRC).c_str()); } } } void CN64Rom::CalculateRomCrc() { uint32_t t0, t2, t3, t4, t5; uint32_t a0, a1, a2, a3; uint32_t s0; uint32_t v0, v1; uint32_t length = 0x00100000; // CIC_NUS_6101 at=0x5D588B65 , s6=0x3F // CIC_NUS_6102 at=0x5D588B65 , s6=0x3F // CIC_NUS_6103 at=0x6C078965 , s6=0x78 // CIC_NUS_6105 at=0x5d588b65 , s6=0x91 // CIC_NUS_6106 at=0x6C078965 , s6=0x85 // 64DD IPL (JPN) at=0x02E90EDD , s6=0xdd // 64DD IPL (USA) at=0x02E90EDD , s6=0xde // 64DD TOOL IPL at=0x0260BCD5 , s6=0xdd // CIC_NUS_5101 (Aleck64) at=0x6C078965 , s6=0xac //v0 = 0xFFFFFFFF & (s6 * at) + 1; switch (m_CicChip) { case CIC_NUS_6101: case CIC_NUS_6102: v0 = 0xF8CA4DDC; break; case CIC_NUS_6103: v0 = 0xA3886759; break; case CIC_NUS_6105: v0 = 0xDF26F436; break; case CIC_NUS_6106: v0 = 0x1FEA617A; break; case CIC_NUS_DDUS: length = 0x000A0000; v0 = 0x861AE3A7; break; case CIC_NUS_8303: length = 0x000A0000; v0 = 0x8331D4CA; break; case CIC_NUS_8401: length = 0x000A0000; v0 = 0x0D8303E2; break; case CIC_NUS_5101: v0 = 0x95104FDD; break; default: return; } ProtectMemory(m_ROMImage, m_RomFileSize, MEM_READWRITE); if (m_CicChip == CIC_NUS_5101 && (*(uint32_t *)(m_ROMImage + 0x8) == 0x80100400)) length = 0x003FE000; v1 = 0; t0 = 0; t5 = 0x20; a3 = v0; t2 = v0; t3 = v0; s0 = v0; a2 = v0; t4 = v0; for (t0 = 0; t0 < length; t0 += 4) { v0 = *(uint32_t *)(m_ROMImage + t0 + 0x1000); v1 = a3 + v0; a1 = v1; if (v1 < a3) { if (m_CicChip == CIC_NUS_DDUS || m_CicChip == CIC_NUS_8303) { t2 = t2 ^ t3; } else { t2 += 0x1; } } v1 = v0 & 0x001F; a0 = (v0 << v1) | (v0 >> (t5 - v1)); a3 = a1; t3 = t3 ^ v0; s0 = s0 + a0; if (a2 < v0) { a2 = a3 ^ v0 ^ a2; } else { if (m_CicChip == CIC_NUS_8303) a2 = a2 + a0; else a2 = a2 ^ a0; } if (m_CicChip == CIC_NUS_6105) { t4 = (v0 ^ (*(uint32_t *)(m_ROMImage + (0xFF & t0) + 0x750))) + t4; } else t4 = (v0 ^ s0) + t4; } if (m_CicChip == CIC_NUS_6103) { a3 = (a3 ^ t2) + t3; s0 = (s0 ^ a2) + t4; } else if (m_CicChip == CIC_NUS_6106) { a3 = 0xFFFFFFFF & (a3 * t2) + t3; s0 = 0xFFFFFFFF & (s0 * a2) + t4; } else if (m_CicChip == CIC_NUS_5101) { a3 = 0xFFFFFFFF & (a3 ^ t2) + t3; s0 = 0xFFFFFFFF & (s0 ^ a2) + t4; } else { a3 = a3 ^ t2 ^ t3; s0 = s0 ^ a2 ^ t4; } *(uint32_t *)(m_ROMImage + 0x10) = a3; *(uint32_t *)(m_ROMImage + 0x14) = s0; ProtectMemory(m_ROMImage, m_RomFileSize, MEM_READONLY); } CICChip CN64Rom::CicChipID() { return m_CicChip; } bool CN64Rom::IsValidRomImage(uint8_t Test[4]) { if (*((uint32_t *)&Test[0]) == 0x40123780) { return true; } if (*((uint32_t *)&Test[0]) == 0x12408037) { return true; } if (*((uint32_t *)&Test[0]) == 0x80371240) { return true; } if (*((uint32_t *)&Test[0]) == 0x40072780) { return true; } // 64DD IPL return false; } bool CN64Rom::IsLoadedRomDDIPL() { switch (CicChipID()) { case CIC_NUS_8303: case CIC_NUS_DDUS: case CIC_NUS_8401: return true; default: return false; } } void CN64Rom::CleanRomName(char * RomName, bool byteswap) { if (byteswap) { for (int count = 0; count < 20; count += 4) { RomName[count] ^= RomName[count + 3]; RomName[count + 3] ^= RomName[count]; RomName[count] ^= RomName[count + 3]; RomName[count + 1] ^= RomName[count + 2]; RomName[count + 2] ^= RomName[count + 1]; RomName[count + 1] ^= RomName[count + 2]; } } // Truncate all the spaces at the end of the string for (int count = 19; count >= 0; count--) { if (RomName[count] == ' ') { RomName[count] = '\0'; } else if (RomName[count] == '\0') { } else { count = -1; } } RomName[20] = '\0'; // Remove all special characters from the string for (int count = 0; count < (int)strlen(RomName); count++) { switch (RomName[count]) { case '/': case '\\': RomName[count] = '-'; break; case ':': RomName[count] = ';'; break; } } } void CN64Rom::NotificationCB(const char * Status, CN64Rom * /*_this*/) { g_Notify->DisplayMessage(5, stdstr_f("%s", Status).c_str()); } bool CN64Rom::LoadN64Image(const char * FileLoc, bool LoadBootCodeOnly) { WriteTrace(TraceN64System, TraceDebug, "Start (FileLoc: \"%s\" LoadBootCodeOnly: %s)", FileLoc, LoadBootCodeOnly ? "true" : "false"); UnallocateRomImage(); m_ErrorMsg = EMPTY_STRING; stdstr ext = CPath(FileLoc).GetExtension(); bool Loaded7zFile = false; #ifdef _WIN32 if (strstr(FileLoc, "?") != nullptr || _stricmp(ext.c_str(), "7z") == 0) { stdstr FullPath = FileLoc; // This should be a 7-zip file char * SubFile = strstr(const_cast(FullPath.c_str()), "?"); if (SubFile != nullptr) { *SubFile = '\0'; SubFile += 1; } //else load first found file until dialog is implemented //{ // Pop up a dialog and select file // Allocate memory for sub name and copy selected file name to variable //} C7zip ZipFile(FullPath.c_str()); ZipFile.SetNotificationCallback((C7zip::LP7ZNOTIFICATION)NotificationCB, this); for (int i = 0; i < ZipFile.NumFiles(); i++) { CSzFileItem * f = ZipFile.FileItem(i); if (f->IsDir) { continue; } stdstr ZipFileName; ZipFileName.FromUTF16(ZipFile.FileNameIndex(i).c_str()); if (SubFile != nullptr) { if (_stricmp(ZipFileName.c_str(), SubFile) != 0) { continue; } } // Get the size of the ROM and try to allocate the memory needed uint32_t RomFileSize = (uint32_t)f->Size; // If loading boot code then just load the first 0x1000 bytes if (LoadBootCodeOnly) { RomFileSize = 0x1000; } if (!AllocateRomImage(RomFileSize)) { WriteTrace(TraceN64System, TraceDebug, "Done (res: false)"); return false; } // Load the N64 ROM to the allocated memory g_Notify->DisplayMessage(5, MSG_LOADING); if (!ZipFile.GetFile(i, m_ROMImage, RomFileSize)) { SetError(MSG_FAIL_IMAGE); WriteTrace(TraceN64System, TraceDebug, "Done (res: false)"); return false; } if (!IsValidRomImage(m_ROMImage)) { if (i < ZipFile.NumFiles() - 1) { UnallocateRomImage(); continue; } SetError(MSG_FAIL_IMAGE); WriteTrace(TraceN64System, TraceDebug, "Done (res: false)"); return false; } g_Notify->DisplayMessage(5, MSG_BYTESWAP); ByteSwapRom(); // Protect the memory so that it can't be written to ProtectMemory(m_ROMImage, m_RomFileSize, MEM_READONLY); Loaded7zFile = true; break; } if (!Loaded7zFile) { SetError(MSG_7Z_FILE_NOT_FOUND); WriteTrace(TraceN64System, TraceDebug, "Done (res: false)"); return false; } } #endif // Try to open the file as a zip file if (!Loaded7zFile) { if (!AllocateAndLoadZipImage(FileLoc, LoadBootCodeOnly)) { if (m_ErrorMsg != EMPTY_STRING) { WriteTrace(TraceN64System, TraceDebug, "Done (res: false)"); return false; } if (!AllocateAndLoadN64Image(FileLoc, LoadBootCodeOnly)) { WriteTrace(TraceN64System, TraceDebug, "Done (res: false)"); return false; } } } char RomName[260]; // Get the header from the ROM image memcpy(&RomName[0], (void *)(m_ROMImage + 0x20), 20); CN64Rom::CleanRomName(RomName); if (strlen(RomName) == 0) { strcpy(RomName, CPath(FileLoc).GetName().c_str()); CN64Rom::CleanRomName(RomName, false); } WriteTrace(TraceN64System, TraceDebug, "RomName \"%s\"", RomName); m_RomName = RomName; m_FileName = FileLoc; m_MD5 = ""; if (!LoadBootCodeOnly) { // Calculate files MD5 checksum m_MD5 = MD5((const unsigned char *)m_ROMImage, m_RomFileSize).hex_digest(); WriteTrace(TraceN64System, TraceDebug, "MD5: %s", m_MD5.c_str()); } m_Country = (Country)m_ROMImage[0x3D]; CalculateCicChip(); uint32_t CRC1, CRC2; if (IsLoadedRomDDIPL()) { // Handle CRC differently if it is a 64DD IPL CRC1 = (*(uint16_t *)(&m_ROMImage[0x608]) << 16) | *(uint16_t *)(&m_ROMImage[0x60C]); CRC2 = (*(uint16_t *)(&m_ROMImage[0x638]) << 16) | *(uint16_t *)(&m_ROMImage[0x63C]); } else { CRC1 = *(uint32_t *)(&m_ROMImage[0x10]); CRC2 = *(uint32_t *)(&m_ROMImage[0x14]); } m_RomIdent = stdstr_f("%08X-%08X-C:%X", CRC1, CRC2, m_ROMImage[0x3D]); { CIniFileBase::SectionList GameIdentifiers; CIniFile RomDatabase(g_Settings->LoadStringVal(SupportFile_RomDatabase).c_str()); RomDatabase.GetVectorOfSections(GameIdentifiers); if (GameIdentifiers.find(m_RomIdent.c_str()) == GameIdentifiers.end()) { char InternalName[22] = {0}; memcpy(InternalName, (void *)(m_ROMImage + 0x20), 20); CN64Rom::CleanRomName(InternalName); std::string AltIdentifier = stdstr_f("%s-C:%X", stdstr(InternalName).Trim().ToUpper().c_str(), m_Country); AltIdentifier = RomDatabase.GetString(AltIdentifier.c_str(), "Alt Identifier", ""); if (!AltIdentifier.empty()) { m_RomIdent = AltIdentifier; } } } WriteTrace(TraceN64System, TraceDebug, "Ident: %s", m_RomIdent.c_str()); if (!LoadBootCodeOnly && g_Rom == this) { g_Settings->SaveBool(GameRunning_LoadingInProgress, false); if (!g_Disk) { SaveRomSettingID(false); } else if (!IsLoadedRomDDIPL()) { g_Settings->SaveString(Game_GameName, m_RomName.c_str()); // Use base games save file if loaded in combo } } if (g_Settings->LoadBool(Game_CRC_Recalc)) { // Calculate ROM header CRC CalculateRomCrc(); } WriteTrace(TraceN64System, TraceDebug, "Done (res: true)"); return true; } bool CN64Rom::LoadN64ImageIPL(const char * FileLoc, bool LoadBootCodeOnly) { UnallocateRomImage(); m_ErrorMsg = EMPTY_STRING; stdstr ext = CPath(FileLoc).GetExtension(); bool Loaded7zFile = false; #ifdef _WIN32 if (strstr(FileLoc, "?") != nullptr || _stricmp(ext.c_str(), "7z") == 0) { stdstr FullPath = FileLoc; // This should be a 7-zip file char * SubFile = strstr(const_cast(FullPath.c_str()), "?"); if (SubFile != nullptr) { *SubFile = '\0'; SubFile += 1; } //else load first found file until dialog is implemented //{ // Pop up a dialog and select file // Allocate memory for sub name and copy selected file name to variable //} C7zip ZipFile(FullPath.c_str()); ZipFile.SetNotificationCallback((C7zip::LP7ZNOTIFICATION)NotificationCB, this); for (int i = 0; i < ZipFile.NumFiles(); i++) { CSzFileItem * f = ZipFile.FileItem(i); if (f->IsDir) { continue; } stdstr ZipFileName; ZipFileName.FromUTF16(ZipFile.FileNameIndex(i).c_str()); if (SubFile != nullptr) { if (_stricmp(ZipFileName.c_str(), SubFile) != 0) { continue; } } // Get the size of the ROM and try to allocate the memory needed uint32_t RomFileSize = (uint32_t)f->Size; // If loading boot code then just load the first 0x1000 bytes if (LoadBootCodeOnly) { RomFileSize = 0x1000; } if (!AllocateRomImage(RomFileSize)) { WriteTrace(TraceN64System, TraceDebug, "Done (res: false)"); return false; } // Load the N64 ROM to the allocated memory g_Notify->DisplayMessage(5, MSG_LOADING); if (!ZipFile.GetFile(i, m_ROMImage, RomFileSize)) { SetError(MSG_FAIL_IMAGE_IPL); WriteTrace(TraceN64System, TraceDebug, "Done (res: false)"); return false; } if (!IsValidRomImage(m_ROMImage)) { if (i < ZipFile.NumFiles() - 1) { UnallocateRomImage(); continue; } SetError(MSG_FAIL_IMAGE_IPL); WriteTrace(TraceN64System, TraceDebug, "Done (res: false)"); return false; } g_Notify->DisplayMessage(5, MSG_BYTESWAP); ByteSwapRom(); // Protect the memory so that it can't be written to ProtectMemory(m_ROMImage, m_RomFileSize, MEM_READONLY); Loaded7zFile = true; break; } if (!Loaded7zFile) { SetError(MSG_7Z_FILE_NOT_FOUND); WriteTrace(TraceN64System, TraceDebug, "Done (res: false)"); return false; } } #endif // Try to open the file as a zip file if (!Loaded7zFile) { if (!AllocateAndLoadZipImage(FileLoc, LoadBootCodeOnly)) { if (m_ErrorMsg != EMPTY_STRING) { return false; } if (!AllocateAndLoadN64Image(FileLoc, LoadBootCodeOnly)) { return false; } } } char RomName[260]; // Get the header from the ROM image memcpy(&RomName[0], (void *)(m_ROMImage + 0x20), 20); CN64Rom::CleanRomName(RomName); if (strlen(RomName) == 0) { strcpy(RomName, CPath(FileLoc).GetName().c_str()); CN64Rom::CleanRomName(RomName, false); } WriteTrace(TraceN64System, TraceDebug, "RomName %s", RomName); m_RomName = RomName; m_FileName = FileLoc; m_MD5 = ""; if (!LoadBootCodeOnly) { // Calculate files MD5 checksum m_MD5 = MD5((const unsigned char *)m_ROMImage, m_RomFileSize).hex_digest(); WriteTrace(TraceN64System, TraceDebug, "MD5: %s", m_MD5.c_str()); } m_Country = (Country)m_ROMImage[0x3D]; CalculateCicChip(); uint32_t CRC1, CRC2; if (IsLoadedRomDDIPL()) { // Handle CRC differently if it is a 64DD IPL CRC1 = (*(uint16_t *)(&m_ROMImage[0x608]) << 16) | *(uint16_t *)(&m_ROMImage[0x60C]); CRC2 = (*(uint16_t *)(&m_ROMImage[0x638]) << 16) | *(uint16_t *)(&m_ROMImage[0x63C]); } else { CRC1 = *(uint32_t *)(&m_ROMImage[0x10]); CRC2 = *(uint32_t *)(&m_ROMImage[0x14]); } m_RomIdent = stdstr_f("%08X-%08X-C:%X", CRC1, CRC2, m_ROMImage[0x3D]); WriteTrace(TraceN64System, TraceDebug, "Ident: %s", m_RomIdent.c_str()); if (!IsLoadedRomDDIPL()) { SetError(MSG_FAIL_IMAGE_IPL); return false; } if (!LoadBootCodeOnly && g_DDRom == this) { g_Settings->SaveBool(GameRunning_LoadingInProgress, false); } if (g_Settings->LoadBool(Game_CRC_Recalc)) { // Calculate ROM header CRC CalculateRomCrc(); } return true; } // Save the settings of the loaded ROM, so all loaded settings about ROM will be identified with this ROM void CN64Rom::SaveRomSettingID(bool temp) { g_Settings->SaveBool(Game_TempLoaded, temp); g_Settings->SaveString(Game_GameName, m_RomName.c_str()); g_Settings->SaveString(Game_IniKey, m_RomIdent.c_str()); g_Settings->SaveString(Game_UniqueSaveDir, stdstr_f("%s-%s", m_RomName.c_str(), m_MD5.c_str()).c_str()); g_Settings->SaveDword(Game_SystemType, IsPal() ? SYSTEM_PAL : SYSTEM_NTSC); } void CN64Rom::ClearRomSettingID() { g_Settings->SaveString(Game_GameName, ""); g_Settings->SaveString(Game_IniKey, ""); } void CN64Rom::SetError(LanguageStringID ErrorMsg) { m_ErrorMsg = ErrorMsg; } bool CN64Rom::IsPal() { switch (m_Country) { case Country_Germany: case Country_French: case Country_Italian: case Country_Europe: case Country_Spanish: case Country_Australia: case Country_EuropeanX_PAL: case Country_EuropeanY_PAL: return true; } return false; } void CN64Rom::UnallocateRomImage() { m_RomFile.Close(); if (m_ROMImageBase) { ProtectMemory(m_ROMImage, m_RomFileSize, MEM_READWRITE); delete[] m_ROMImageBase; m_ROMImageBase = nullptr; } m_ROMImage = nullptr; }