2019-11-16 05:27:57 +00:00
|
|
|
#include "bios.h"
|
2020-01-10 03:31:12 +00:00
|
|
|
#include "common/assert.h"
|
2020-07-31 06:03:32 +00:00
|
|
|
#include "common/file_system.h"
|
2020-04-07 01:54:24 +00:00
|
|
|
#include "common/log.h"
|
2020-01-10 03:31:12 +00:00
|
|
|
#include "common/md5_digest.h"
|
2019-11-16 05:27:57 +00:00
|
|
|
#include "cpu_disasm.h"
|
2020-09-22 13:08:07 +00:00
|
|
|
#include <array>
|
2019-11-28 13:29:03 +00:00
|
|
|
#include <cerrno>
|
2019-11-16 05:27:57 +00:00
|
|
|
Log_SetChannel(BIOS);
|
|
|
|
|
|
|
|
namespace BIOS {
|
|
|
|
static constexpr Hash MakeHashFromString(const char str[])
|
|
|
|
{
|
|
|
|
Hash h{};
|
|
|
|
for (int i = 0; str[i] != '\0'; i++)
|
|
|
|
{
|
|
|
|
u8 nibble = 0;
|
|
|
|
char ch = str[i];
|
|
|
|
if (ch >= '0' && ch <= '9')
|
|
|
|
nibble = str[i] - '0';
|
|
|
|
else if (ch >= 'a' && ch <= 'z')
|
|
|
|
nibble = 0xA + (str[i] - 'a');
|
|
|
|
else if (ch >= 'A' && ch <= 'Z')
|
|
|
|
nibble = 0xA + (str[i] - 'A');
|
|
|
|
|
|
|
|
h.bytes[i / 2] |= nibble << (((i & 1) ^ 1) * 4);
|
|
|
|
}
|
|
|
|
return h;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string Hash::ToString() const
|
|
|
|
{
|
|
|
|
char str[33];
|
|
|
|
std::snprintf(str, sizeof(str), "%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x", bytes[0],
|
|
|
|
bytes[1], bytes[2], bytes[3], bytes[4], bytes[5], bytes[6], bytes[7], bytes[8], bytes[9], bytes[10],
|
|
|
|
bytes[11], bytes[12], bytes[13], bytes[14], bytes[15]);
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
2021-01-05 07:38:42 +00:00
|
|
|
static constexpr std::array<ImageInfo, 27> s_image_infos = {
|
2020-12-28 00:14:28 +00:00
|
|
|
{{"SCPH-1000, DTL-H1000 (v1.0)", ConsoleRegion::NTSC_J, MakeHashFromString("239665b1a3dade1b5a52c06338011044"), true},
|
|
|
|
{"SCPH-1001, 5003, DTL-H1201, H3001 (v2.2 12-04-95 A)", ConsoleRegion::NTSC_U,
|
|
|
|
MakeHashFromString("924e392ed05558ffdb115408c263dccf"), true},
|
|
|
|
{"SCPH-1002, DTL-H1002 (v2.0 05-10-95 E)", ConsoleRegion::PAL,
|
|
|
|
MakeHashFromString("54847e693405ffeb0359c6287434cbef"), true},
|
|
|
|
{"SCPH-1002, DTL-H1102 (v2.1 07-17-95 E)", ConsoleRegion::PAL,
|
|
|
|
MakeHashFromString("417b34706319da7cf001e76e40136c23"), true},
|
|
|
|
{"SCPH-1002, DTL-H1202, H3002 (v2.2 12-04-95 E)", ConsoleRegion::PAL,
|
|
|
|
MakeHashFromString("e2110b8a2b97a8e0b857a45d32f7e187"), true},
|
|
|
|
{"DTL-H1100 (v2.2 03-06-96 D)", ConsoleRegion::NTSC_J, MakeHashFromString("ca5cfc321f916756e3f0effbfaeba13b"), true},
|
|
|
|
{"SCPH-3000, DTL-H1000H (v1.1 01-22-95)", ConsoleRegion::NTSC_J,
|
|
|
|
MakeHashFromString("849515939161e62f6b866f6853006780"), true},
|
|
|
|
{"SCPH-1001, DTL-H1001 (v2.0 05-07-95 A)", ConsoleRegion::NTSC_U,
|
|
|
|
MakeHashFromString("dc2b9bf8da62ec93e868cfd29f0d067d"), true},
|
|
|
|
{"SCPH-3500 (v2.1 07-17-95 J)", ConsoleRegion::NTSC_J, MakeHashFromString("cba733ceeff5aef5c32254f1d617fa62"), true},
|
|
|
|
{"SCPH-1001, DTL-H1101 (v2.1 07-17-95 A)", ConsoleRegion::NTSC_U,
|
|
|
|
MakeHashFromString("da27e8b6dab242d8f91a9b25d80c63b8"), true},
|
|
|
|
{"SCPH-5000, DTL-H1200, H3000 (v2.2 12-04-95 J)", ConsoleRegion::NTSC_J,
|
|
|
|
MakeHashFromString("57a06303dfa9cf9351222dfcbb4a29d9"), true},
|
|
|
|
{"SCPH-5500 (v3.0 09-09-96 J)", ConsoleRegion::NTSC_J, MakeHashFromString("8dd7d5296a650fac7319bce665a6a53c"), true},
|
|
|
|
{"SCPH-5501, 5503, 7003 (v3.0 11-18-96 A)", ConsoleRegion::NTSC_U,
|
|
|
|
MakeHashFromString("490f666e1afb15b7362b406ed1cea246"), true},
|
|
|
|
{"SCPH-5502, 5552 (v3.0 01-06-97 E)", ConsoleRegion::PAL, MakeHashFromString("32736f17079d0b2b7024407c39bd3050"),
|
|
|
|
true},
|
|
|
|
{"SCPH-7000, 7500, 9000 (v4.0 08-18-97 J)", ConsoleRegion::NTSC_J,
|
|
|
|
MakeHashFromString("8e4c14f567745eff2f0408c8129f72a6"), true},
|
|
|
|
{"SCPH-7000W (v4.1 11-14-97 A)", ConsoleRegion::NTSC_J, MakeHashFromString("b84be139db3ee6cbd075630aa20a6553"),
|
|
|
|
true},
|
|
|
|
{"SCPH-7001, 7501, 7503, 9001, 9003, 9903 (v4.1 12-16-97 A)", ConsoleRegion::NTSC_U,
|
|
|
|
MakeHashFromString("1e68c231d0896b7eadcad1d7d8e76129"), true},
|
|
|
|
{"SCPH-7002, 7502, 9002 (v4.1 12-16-97 E)", ConsoleRegion::PAL,
|
|
|
|
MakeHashFromString("b9d9a0286c33dc6b7237bb13cd46fdee"), true},
|
|
|
|
{"SCPH-100 (v4.3 03-11-00 J)", ConsoleRegion::NTSC_J, MakeHashFromString("8abc1b549a4a80954addc48ef02c4521"), true},
|
|
|
|
{"SCPH-101 (v4.4 03-24-00 A)", ConsoleRegion::NTSC_U, MakeHashFromString("9a09ab7e49b422c007e6d54d7c49b965"), true},
|
|
|
|
{"SCPH-101 (v4.5 05-25-00 A)", ConsoleRegion::NTSC_U, MakeHashFromString("6e3735ff4c7dc899ee98981385f6f3d0"), true},
|
|
|
|
{"SCPH-102 (v4.4 03-24-00 E)", ConsoleRegion::PAL, MakeHashFromString("b10f5e0e3d9eb60e5159690680b1e774"), true},
|
|
|
|
{"SCPH-102 (v4.5 05-25-00 E)", ConsoleRegion::PAL, MakeHashFromString("de93caec13d1a141a40a79f5c86168d6"), true},
|
|
|
|
{"PSP, SCPH-1000R (v4.5 05-25-00 J)", ConsoleRegion::Auto, MakeHashFromString("c53ca5908936d412331790f4426c6c33"),
|
|
|
|
true},
|
|
|
|
{"SCPH-1000R (v4.5 05-25-00 J)", ConsoleRegion::NTSC_J, MakeHashFromString("476d68a94ccec3b9c8303bbd1daf2810"),
|
|
|
|
true},
|
2021-01-05 07:38:42 +00:00
|
|
|
{"PS3 (v5.0 06-23-03 A)", ConsoleRegion::Auto, MakeHashFromString("c02a6fbb1b27359f84e92fae8bc21316"), false},
|
2020-12-28 00:14:28 +00:00
|
|
|
{"PS3 (v5.0 06-23-03 A)", ConsoleRegion::Auto, MakeHashFromString("81bbe60ba7a3d1cea1d48c14cbcc647b"), false}}};
|
2019-11-16 05:27:57 +00:00
|
|
|
|
|
|
|
Hash GetHash(const Image& image)
|
|
|
|
{
|
|
|
|
Hash hash;
|
|
|
|
MD5Digest digest;
|
|
|
|
digest.Update(image.data(), static_cast<u32>(image.size()));
|
|
|
|
digest.Final(hash.bytes);
|
|
|
|
return hash;
|
|
|
|
}
|
|
|
|
|
2020-09-22 13:08:07 +00:00
|
|
|
std::optional<Image> LoadImageFromFile(const char* filename)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
|
|
|
Image ret(BIOS_SIZE);
|
2020-09-22 13:08:07 +00:00
|
|
|
auto fp = FileSystem::OpenManagedCFile(filename, "rb");
|
2019-11-16 05:27:57 +00:00
|
|
|
if (!fp)
|
|
|
|
{
|
2020-09-22 13:08:07 +00:00
|
|
|
Log_ErrorPrintf("Failed to open BIOS image '%s', errno=%d", filename, errno);
|
2019-11-16 05:27:57 +00:00
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2020-07-31 06:03:32 +00:00
|
|
|
std::fseek(fp.get(), 0, SEEK_END);
|
|
|
|
const u32 size = static_cast<u32>(std::ftell(fp.get()));
|
|
|
|
std::fseek(fp.get(), 0, SEEK_SET);
|
2019-11-16 05:27:57 +00:00
|
|
|
|
2021-01-05 07:38:42 +00:00
|
|
|
if (size != BIOS_SIZE && size != BIOS_SIZE_PS2 && size != BIOS_SIZE_PS3)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
2021-01-05 07:38:42 +00:00
|
|
|
Log_ErrorPrintf("BIOS image '%s' size mismatch, expecting either %u or %u or %u bytes but got %u bytes", filename,
|
|
|
|
BIOS_SIZE, BIOS_SIZE_PS2, BIOS_SIZE_PS3, size);
|
2019-11-16 05:27:57 +00:00
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
2020-07-31 06:03:32 +00:00
|
|
|
if (std::fread(ret.data(), 1, ret.size(), fp.get()) != ret.size())
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
2020-09-22 13:08:07 +00:00
|
|
|
Log_ErrorPrintf("Failed to read BIOS image '%s'", filename);
|
2019-11-16 05:27:57 +00:00
|
|
|
return std::nullopt;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2020-09-22 13:08:07 +00:00
|
|
|
std::optional<Hash> GetHashForFile(const char* filename)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
|
|
|
auto image = LoadImageFromFile(filename);
|
|
|
|
if (!image)
|
|
|
|
return std::nullopt;
|
|
|
|
|
|
|
|
return GetHash(*image);
|
|
|
|
}
|
|
|
|
|
2020-09-22 13:08:07 +00:00
|
|
|
const ImageInfo* GetImageInfoForHash(const Hash& hash)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
2020-09-22 13:08:07 +00:00
|
|
|
for (const ImageInfo& ii : s_image_infos)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
2020-09-22 13:08:07 +00:00
|
|
|
if (ii.hash == hash)
|
|
|
|
return ⅈ
|
|
|
|
}
|
2019-11-16 05:27:57 +00:00
|
|
|
|
2020-09-22 13:08:07 +00:00
|
|
|
return nullptr;
|
|
|
|
}
|
2019-11-16 05:27:57 +00:00
|
|
|
|
2020-09-22 13:08:07 +00:00
|
|
|
bool IsValidHashForRegion(ConsoleRegion region, const Hash& hash)
|
|
|
|
{
|
|
|
|
const ImageInfo* ii = GetImageInfoForHash(hash);
|
|
|
|
if (!ii)
|
|
|
|
return false;
|
2019-11-16 05:27:57 +00:00
|
|
|
|
2020-09-22 13:08:07 +00:00
|
|
|
return (ii->region == ConsoleRegion::Auto || ii->region == region);
|
2019-11-16 05:27:57 +00:00
|
|
|
}
|
|
|
|
|
2020-10-29 12:05:09 +00:00
|
|
|
void PatchBIOS(u8* image, u32 image_size, u32 address, u32 value, u32 mask /*= UINT32_C(0xFFFFFFFF)*/)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
|
|
|
const u32 phys_address = address & UINT32_C(0x1FFFFFFF);
|
|
|
|
const u32 offset = phys_address - BIOS_BASE;
|
2020-10-29 12:05:09 +00:00
|
|
|
Assert(phys_address >= BIOS_BASE && (offset + sizeof(u32)) <= image_size);
|
2019-11-16 05:27:57 +00:00
|
|
|
|
|
|
|
u32 existing_value;
|
2020-10-29 12:05:09 +00:00
|
|
|
std::memcpy(&existing_value, &image[offset], sizeof(existing_value));
|
2019-11-16 05:27:57 +00:00
|
|
|
u32 new_value = (existing_value & ~mask) | value;
|
2020-10-29 12:05:09 +00:00
|
|
|
std::memcpy(&image[offset], &new_value, sizeof(new_value));
|
2019-11-16 05:27:57 +00:00
|
|
|
|
|
|
|
SmallString old_disasm, new_disasm;
|
|
|
|
CPU::DisassembleInstruction(&old_disasm, address, existing_value);
|
|
|
|
CPU::DisassembleInstruction(&new_disasm, address, new_value);
|
|
|
|
Log_DevPrintf("BIOS-Patch 0x%08X (+0x%X): 0x%08X %s -> %08X %s", address, offset, existing_value,
|
|
|
|
old_disasm.GetCharArray(), new_value, new_disasm.GetCharArray());
|
|
|
|
}
|
|
|
|
|
2020-10-29 12:05:09 +00:00
|
|
|
bool PatchBIOSEnableTTY(u8* image, u32 image_size, const Hash& hash)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
2020-09-22 13:08:07 +00:00
|
|
|
const ImageInfo* ii = GetImageInfoForHash(hash);
|
2020-12-28 00:14:28 +00:00
|
|
|
if (!ii || !ii->patch_compatible)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
|
|
|
Log_WarningPrintf("Incompatible version for TTY patch: %s", hash.ToString().c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Log_InfoPrintf("Patching BIOS to enable TTY/printf");
|
2020-10-29 12:05:09 +00:00
|
|
|
PatchBIOS(image, image_size, 0x1FC06F0C, 0x24010001);
|
|
|
|
PatchBIOS(image, image_size, 0x1FC06F14, 0xAF81A9C0);
|
2019-11-16 05:27:57 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-10-29 12:05:09 +00:00
|
|
|
bool PatchBIOSFastBoot(u8* image, u32 image_size, const Hash& hash)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
2020-09-22 13:08:07 +00:00
|
|
|
const ImageInfo* ii = GetImageInfoForHash(hash);
|
2020-12-28 00:14:28 +00:00
|
|
|
if (!ii || !ii->patch_compatible)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
|
|
|
Log_WarningPrintf("Incompatible version for fast-boot patch: %s", hash.ToString().c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Replace the shell entry point with a return back to the bootstrap.
|
|
|
|
Log_InfoPrintf("Patching BIOS to skip intro");
|
2020-11-10 11:35:03 +00:00
|
|
|
PatchBIOS(image, image_size, 0x1FC18000, 0x3C011F80); // lui at, 1f80
|
|
|
|
PatchBIOS(image, image_size, 0x1FC18004, 0x3C0A0300); // lui t2, 0300h
|
|
|
|
PatchBIOS(image, image_size, 0x1FC18008, 0xAC2A1814); // sw zero, 1814h(at) ; turn the display on
|
|
|
|
PatchBIOS(image, image_size, 0x1FC1800C, 0x03E00008); // jr ra
|
|
|
|
PatchBIOS(image, image_size, 0x1FC18010, 0x00000000); // nop
|
2019-11-16 05:27:57 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2020-10-29 12:05:09 +00:00
|
|
|
bool PatchBIOSForEXE(u8* image, u32 image_size, u32 r_pc, u32 r_gp, u32 r_sp, u32 r_fp)
|
2019-11-16 05:27:57 +00:00
|
|
|
{
|
2020-10-29 12:05:09 +00:00
|
|
|
#define PATCH(offset, value) PatchBIOS(image, image_size, (offset), (value))
|
|
|
|
|
2019-11-16 05:27:57 +00:00
|
|
|
// pc has to be done first because we can't load it in the delay slot
|
2020-10-29 12:05:09 +00:00
|
|
|
PATCH(0xBFC06FF0, UINT32_C(0x3C080000) | r_pc >> 16); // lui $t0, (r_pc >> 16)
|
|
|
|
PATCH(0xBFC06FF4, UINT32_C(0x35080000) | (r_pc & UINT32_C(0xFFFF))); // ori $t0, $t0, (r_pc & 0xFFFF)
|
|
|
|
PATCH(0xBFC06FF8, UINT32_C(0x3C1C0000) | r_gp >> 16); // lui $gp, (r_gp >> 16)
|
|
|
|
PATCH(0xBFC06FFC, UINT32_C(0x379C0000) | (r_gp & UINT32_C(0xFFFF))); // ori $gp, $gp, (r_gp & 0xFFFF)
|
2019-12-11 06:34:56 +00:00
|
|
|
|
|
|
|
if (r_sp != 0)
|
|
|
|
{
|
2020-10-29 12:05:09 +00:00
|
|
|
PATCH(0xBFC07000, UINT32_C(0x3C1D0000) | r_sp >> 16); // lui $sp, (r_sp >> 16)
|
|
|
|
PATCH(0xBFC07004, UINT32_C(0x37BD0000) | (r_sp & UINT32_C(0xFFFF))); // ori $sp, $sp, (r_sp & 0xFFFF)
|
2019-12-11 06:34:56 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-10-29 12:05:09 +00:00
|
|
|
PATCH(0xBFC07000, UINT32_C(0x00000000)); // nop
|
|
|
|
PATCH(0xBFC07004, UINT32_C(0x00000000)); // nop
|
2019-12-11 06:34:56 +00:00
|
|
|
}
|
|
|
|
if (r_fp != 0)
|
|
|
|
{
|
2020-10-29 12:05:09 +00:00
|
|
|
PATCH(0xBFC07008, UINT32_C(0x3C1E0000) | r_fp >> 16); // lui $fp, (r_fp >> 16)
|
|
|
|
PATCH(0xBFC0700C, UINT32_C(0x01000008)); // jr $t0
|
|
|
|
PATCH(0xBFC07010, UINT32_C(0x37DE0000) | (r_fp & UINT32_C(0xFFFF))); // ori $fp, $fp, (r_fp & 0xFFFF)
|
2019-12-11 06:34:56 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-10-29 12:05:09 +00:00
|
|
|
PATCH(0xBFC07008, UINT32_C(0x00000000)); // nop
|
|
|
|
PATCH(0xBFC0700C, UINT32_C(0x01000008)); // jr $t0
|
|
|
|
PATCH(0xBFC07010, UINT32_C(0x00000000)); // nop
|
2019-12-11 06:34:56 +00:00
|
|
|
}
|
|
|
|
|
2020-10-29 12:05:09 +00:00
|
|
|
#undef PATCH
|
|
|
|
|
2019-11-16 05:27:57 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-12-04 11:12:50 +00:00
|
|
|
bool IsValidPSExeHeader(const PSEXEHeader& header, u32 file_size)
|
|
|
|
{
|
|
|
|
static constexpr char expected_id[] = {'P', 'S', '-', 'X', ' ', 'E', 'X', 'E'};
|
|
|
|
if (std::memcmp(header.id, expected_id, sizeof(expected_id)) != 0)
|
|
|
|
return false;
|
|
|
|
|
2021-10-28 17:16:55 +00:00
|
|
|
if ((header.file_size + sizeof(PSEXEHeader)) > file_size)
|
2021-11-06 00:48:23 +00:00
|
|
|
{
|
|
|
|
Log_WarningPrintf("Incorrect file size in PS-EXE header: %u bytes should not be greater than %u bytes",
|
2022-03-27 07:16:21 +00:00
|
|
|
header.file_size, static_cast<unsigned>(file_size - sizeof(PSEXEHeader)));
|
2021-11-06 00:48:23 +00:00
|
|
|
}
|
2019-12-04 11:12:50 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-01-24 03:55:19 +00:00
|
|
|
DiscRegion GetPSExeDiscRegion(const PSEXEHeader& header)
|
|
|
|
{
|
|
|
|
static constexpr char ntsc_u_id[] = "Sony Computer Entertainment Inc. for North America area";
|
|
|
|
static constexpr char ntsc_j_id[] = "Sony Computer Entertainment Inc. for Japan area";
|
|
|
|
static constexpr char pal_id[] = "Sony Computer Entertainment Inc. for Europe area";
|
|
|
|
|
|
|
|
if (std::memcmp(header.marker, ntsc_u_id, sizeof(ntsc_u_id) - 1) == 0)
|
|
|
|
return DiscRegion::NTSC_U;
|
|
|
|
else if (std::memcmp(header.marker, ntsc_j_id, sizeof(ntsc_j_id) - 1) == 0)
|
|
|
|
return DiscRegion::NTSC_J;
|
|
|
|
else if (std::memcmp(header.marker, pal_id, sizeof(pal_id) - 1) == 0)
|
|
|
|
return DiscRegion::PAL;
|
|
|
|
else
|
|
|
|
return DiscRegion::Other;
|
|
|
|
}
|
|
|
|
|
2020-07-21 15:30:07 +00:00
|
|
|
} // namespace BIOS
|