ppsspp/Core/Loaders.cpp

135 lines
3.3 KiB
C++
Raw Normal View History

2012-11-01 15:19:01 +00:00
// Copyright (c) 2012- PPSSPP Project.
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, version 2.0 or later versions.
2012-11-01 15:19:01 +00:00
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License 2.0 for more details.
// A copy of the GPL 2.0 should have been included with the program.
// If not, see http://www.gnu.org/licenses/
// Official git repository and contact information can be found at
// https://github.com/hrydgard/ppsspp and http://www.ppsspp.org/.
#include "file/file_util.h"
#include "MIPS/MIPS.h"
#include "MIPS/MIPSCodeUtils.h"
#include "HLE/HLE.h"
#include "HLE/sceKernelModule.h"
#include "PSPLoaders.h"
#include "MemMap.h"
#include "Loaders.h"
#include "System.h"
// TODO : improve, look in the file more
EmuFileType Identify_File(const char *filename)
{
//then: easy bulletproof IDs.
FILE *f = fopen(filename, "rb");
if (!f)
{
//File does not exists
return FILETYPE_ERROR;
}
u32 id;
2013-05-08 15:36:57 +00:00
u32 psar_offset, psar_id;
2013-05-08 15:36:57 +00:00
size_t readSize = fread(&id,4,1,f);
if(readSize != 1)
2013-05-30 15:40:35 +00:00
{
fclose(f);
return FILETYPE_ERROR;
2013-05-30 15:40:35 +00:00
}
2013-05-08 15:36:57 +00:00
psar_id = 0;
fseek(f, 0x24, SEEK_SET);
fread(&psar_offset, 4, 1, f);
fseek(f, psar_offset, SEEK_SET);
fread(&psar_id, 4, 1, f);
fclose(f);
if (strlen(filename) < 5) {
ERROR_LOG(LOADER, "invalid filename %s", filename);
}
const char *extension = filename + strlen(filename) - 4;
2012-11-01 15:19:01 +00:00
if (id == 'FLE\x7F')
{
if (!strcasecmp(extension, ".plf") || strstr(filename,"BOOT.BIN") ||
!strcasecmp(extension, ".elf") || !strcasecmp(extension,".prx") )
2012-11-01 15:19:01 +00:00
{
return FILETYPE_PSP_ELF;
}
else
return FILETYPE_UNKNOWN_ELF;
2012-11-01 15:19:01 +00:00
}
else if (id == 'PBP\x00')
{
2013-05-08 15:36:57 +00:00
if(psar_id == 'MUPN')
return FILETYPE_PSP_ISO_NP;
else
return FILETYPE_PSP_PBP;
2012-11-01 15:19:01 +00:00
}
else
{
if (!strcasecmp(extension,".pbp"))
2012-11-01 15:19:01 +00:00
{
return FILETYPE_PSP_PBP;
}
else if (!strcasecmp(extension,".iso"))
2012-11-01 15:19:01 +00:00
{
return FILETYPE_PSP_ISO;
}
else if (!strcasecmp(extension,".cso"))
2012-11-01 15:19:01 +00:00
{
return FILETYPE_PSP_ISO;
}
else if (!strcasecmp(extension,".bin"))
2012-11-01 15:19:01 +00:00
{
return FILETYPE_UNKNOWN_BIN;
}
}
return FILETYPE_UNKNOWN;
}
bool LoadFile(const char *filename, std::string *error_string)
{
2013-01-07 08:10:05 +00:00
INFO_LOG(LOADER,"Identifying file...");
2012-11-01 15:19:01 +00:00
switch (Identify_File(filename))
{
case FILETYPE_PSP_PBP:
case FILETYPE_PSP_ELF:
{
2013-01-07 08:10:05 +00:00
INFO_LOG(LOADER,"File is an ELF!");
2012-11-01 15:19:01 +00:00
std::string path = getDir(filename);
// If loading from memstick...
size_t pos = path.find("/PSP/GAME/");
if (pos != std::string::npos)
2013-01-10 11:27:10 +00:00
pspFileSystem.SetStartingDirectory("ms0:" + path.substr(pos));
2012-11-01 15:19:01 +00:00
return Load_PSP_ELF_PBP(filename, error_string);
}
case FILETYPE_PSP_ISO:
2013-05-08 15:36:57 +00:00
case FILETYPE_PSP_ISO_NP:
2013-01-10 11:27:10 +00:00
pspFileSystem.SetStartingDirectory("disc0:/PSP_GAME/USRDIR");
2012-11-01 15:19:01 +00:00
return Load_PSP_ISO(filename, error_string);
case FILETYPE_ERROR:
2013-03-30 08:11:53 +00:00
ERROR_LOG(LOADER, "Could not read file");
2012-11-01 15:19:01 +00:00
*error_string = "Error reading file";
break;
case FILETYPE_UNKNOWN_BIN:
case FILETYPE_UNKNOWN_ELF:
case FILETYPE_UNKNOWN:
default:
2013-01-07 08:10:05 +00:00
ERROR_LOG(LOADER, "Failed to identify file");
*error_string = "Failed to identify file";
2012-11-01 15:19:01 +00:00
break;
}
return false;
}