mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-25 23:41:19 +01:00
7c45afecb2
Some callers already have the file info, making the relatively slow FindFileInfo calls unnecessary. Callers that didn't have the file info will now need to call FindFileInfo on their own.
268 lines
6.4 KiB
C++
268 lines
6.4 KiB
C++
// Copyright 2008 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <cstddef>
|
|
#include <map>
|
|
#include <memory>
|
|
#include <optional>
|
|
#include <string>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "Common/Assert.h"
|
|
#include "Common/ColorUtil.h"
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/Logging/Log.h"
|
|
#include "Common/MsgHandler.h"
|
|
#include "Common/StringUtil.h"
|
|
#include "DiscIO/Blob.h"
|
|
#include "DiscIO/Enums.h"
|
|
#include "DiscIO/Filesystem.h"
|
|
#include "DiscIO/Volume.h"
|
|
#include "DiscIO/VolumeGC.h"
|
|
|
|
namespace DiscIO
|
|
{
|
|
VolumeGC::VolumeGC(std::unique_ptr<BlobReader> reader) : m_pReader(std::move(reader))
|
|
{
|
|
_assert_(m_pReader);
|
|
}
|
|
|
|
VolumeGC::~VolumeGC()
|
|
{
|
|
}
|
|
|
|
bool VolumeGC::Read(u64 _Offset, u64 _Length, u8* _pBuffer, const Partition& partition) const
|
|
{
|
|
if (partition != PARTITION_NONE)
|
|
return false;
|
|
|
|
return m_pReader->Read(_Offset, _Length, _pBuffer);
|
|
}
|
|
|
|
std::string VolumeGC::GetGameID(const Partition& partition) const
|
|
{
|
|
static const std::string NO_UID("NO_UID");
|
|
|
|
char ID[6];
|
|
|
|
if (!Read(0, sizeof(ID), reinterpret_cast<u8*>(ID), partition))
|
|
{
|
|
PanicAlertT("Failed to read unique ID from disc image");
|
|
return NO_UID;
|
|
}
|
|
|
|
return DecodeString(ID);
|
|
}
|
|
|
|
Region VolumeGC::GetRegion() const
|
|
{
|
|
const std::optional<u8> country_code = ReadSwapped<u8>(3, PARTITION_NONE);
|
|
return country_code ? RegionSwitchGC(*country_code) : Region::UNKNOWN_REGION;
|
|
}
|
|
|
|
Country VolumeGC::GetCountry(const Partition& partition) const
|
|
{
|
|
const std::optional<u8> country_code = ReadSwapped<u8>(3, partition);
|
|
return country_code ? CountrySwitch(*country_code) : Country::COUNTRY_UNKNOWN;
|
|
}
|
|
|
|
std::string VolumeGC::GetMakerID(const Partition& partition) const
|
|
{
|
|
char makerID[2];
|
|
if (!Read(0x4, 0x2, (u8*)&makerID, partition))
|
|
return std::string();
|
|
|
|
return DecodeString(makerID);
|
|
}
|
|
|
|
std::optional<u16> VolumeGC::GetRevision(const Partition& partition) const
|
|
{
|
|
std::optional<u8> revision = ReadSwapped<u8>(7, partition);
|
|
return revision ? *revision : std::optional<u16>();
|
|
}
|
|
|
|
std::string VolumeGC::GetInternalName(const Partition& partition) const
|
|
{
|
|
char name[0x60];
|
|
if (Read(0x20, 0x60, (u8*)name, partition))
|
|
return DecodeString(name);
|
|
|
|
return "";
|
|
}
|
|
|
|
std::map<Language, std::string> VolumeGC::GetShortNames() const
|
|
{
|
|
LoadBannerFile();
|
|
return m_short_names;
|
|
}
|
|
|
|
std::map<Language, std::string> VolumeGC::GetLongNames() const
|
|
{
|
|
LoadBannerFile();
|
|
return m_long_names;
|
|
}
|
|
|
|
std::map<Language, std::string> VolumeGC::GetShortMakers() const
|
|
{
|
|
LoadBannerFile();
|
|
return m_short_makers;
|
|
}
|
|
|
|
std::map<Language, std::string> VolumeGC::GetLongMakers() const
|
|
{
|
|
LoadBannerFile();
|
|
return m_long_makers;
|
|
}
|
|
|
|
std::map<Language, std::string> VolumeGC::GetDescriptions() const
|
|
{
|
|
LoadBannerFile();
|
|
return m_descriptions;
|
|
}
|
|
|
|
std::vector<u32> VolumeGC::GetBanner(int* width, int* height) const
|
|
{
|
|
LoadBannerFile();
|
|
*width = m_image_width;
|
|
*height = m_image_height;
|
|
return m_image_buffer;
|
|
}
|
|
|
|
std::string VolumeGC::GetApploaderDate(const Partition& partition) const
|
|
{
|
|
char date[16];
|
|
if (!Read(0x2440, 0x10, (u8*)&date, partition))
|
|
return std::string();
|
|
|
|
return DecodeString(date);
|
|
}
|
|
|
|
BlobType VolumeGC::GetBlobType() const
|
|
{
|
|
return m_pReader->GetBlobType();
|
|
}
|
|
|
|
u64 VolumeGC::GetSize() const
|
|
{
|
|
return m_pReader->GetDataSize();
|
|
}
|
|
|
|
u64 VolumeGC::GetRawSize() const
|
|
{
|
|
return m_pReader->GetRawSize();
|
|
}
|
|
|
|
std::optional<u8> VolumeGC::GetDiscNumber(const Partition& partition) const
|
|
{
|
|
return ReadSwapped<u8>(6, partition);
|
|
}
|
|
|
|
Platform VolumeGC::GetVolumeType() const
|
|
{
|
|
return Platform::GAMECUBE_DISC;
|
|
}
|
|
|
|
void VolumeGC::LoadBannerFile() const
|
|
{
|
|
// If opening.bnr has been loaded already, return immediately
|
|
if (m_banner_loaded)
|
|
return;
|
|
|
|
m_banner_loaded = true;
|
|
|
|
GCBanner banner_file;
|
|
std::unique_ptr<FileSystem> file_system(CreateFileSystem(this, PARTITION_NONE));
|
|
if (!file_system)
|
|
return;
|
|
|
|
const FileInfo* file_info = file_system->FindFileInfo("opening.bnr");
|
|
if (!file_info)
|
|
return;
|
|
|
|
size_t file_size = static_cast<size_t>(file_info->GetSize());
|
|
constexpr int BNR1_MAGIC = 0x31524e42;
|
|
constexpr int BNR2_MAGIC = 0x32524e42;
|
|
if (file_size != BNR1_SIZE && file_size != BNR2_SIZE)
|
|
{
|
|
WARN_LOG(DISCIO, "Invalid opening.bnr. Size: %0zx", file_size);
|
|
return;
|
|
}
|
|
|
|
if (file_size != file_system->ReadFile(file_info, reinterpret_cast<u8*>(&banner_file), file_size))
|
|
{
|
|
WARN_LOG(DISCIO, "Could not read opening.bnr.");
|
|
return;
|
|
}
|
|
|
|
bool is_bnr1;
|
|
if (banner_file.id == BNR1_MAGIC && file_size == BNR1_SIZE)
|
|
{
|
|
is_bnr1 = true;
|
|
}
|
|
else if (banner_file.id == BNR2_MAGIC && file_size == BNR2_SIZE)
|
|
{
|
|
is_bnr1 = false;
|
|
}
|
|
else
|
|
{
|
|
WARN_LOG(DISCIO, "Invalid opening.bnr. Type: %0x Size: %0zx", banner_file.id, file_size);
|
|
return;
|
|
}
|
|
|
|
ExtractBannerInformation(banner_file, is_bnr1);
|
|
}
|
|
|
|
void VolumeGC::ExtractBannerInformation(const GCBanner& banner_file, bool is_bnr1) const
|
|
{
|
|
u32 number_of_languages = 0;
|
|
Language start_language = Language::LANGUAGE_UNKNOWN;
|
|
|
|
if (is_bnr1) // NTSC
|
|
{
|
|
bool is_japanese = GetRegion() == Region::NTSC_J;
|
|
number_of_languages = 1;
|
|
start_language = is_japanese ? Language::LANGUAGE_JAPANESE : Language::LANGUAGE_ENGLISH;
|
|
}
|
|
else // PAL
|
|
{
|
|
number_of_languages = 6;
|
|
start_language = Language::LANGUAGE_ENGLISH;
|
|
}
|
|
|
|
m_image_width = GC_BANNER_WIDTH;
|
|
m_image_height = GC_BANNER_HEIGHT;
|
|
m_image_buffer = std::vector<u32>(m_image_width * m_image_height);
|
|
ColorUtil::decode5A3image(m_image_buffer.data(), banner_file.image, m_image_width,
|
|
m_image_height);
|
|
|
|
for (u32 i = 0; i < number_of_languages; ++i)
|
|
{
|
|
const GCBannerInformation& info = banner_file.information[i];
|
|
Language language = static_cast<Language>(static_cast<int>(start_language) + i);
|
|
|
|
std::string description = DecodeString(info.description);
|
|
if (!description.empty())
|
|
m_descriptions[language] = description;
|
|
|
|
std::string short_name = DecodeString(info.short_name);
|
|
if (!short_name.empty())
|
|
m_short_names[language] = short_name;
|
|
|
|
std::string long_name = DecodeString(info.long_name);
|
|
if (!long_name.empty())
|
|
m_long_names[language] = long_name;
|
|
|
|
std::string short_maker = DecodeString(info.short_maker);
|
|
if (!short_maker.empty())
|
|
m_short_makers[language] = short_maker;
|
|
|
|
std::string long_maker = DecodeString(info.long_maker);
|
|
if (!long_maker.empty())
|
|
m_long_makers[language] = long_maker;
|
|
}
|
|
}
|
|
|
|
} // namespace
|