2023-11-18 06:21:51 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2023 Connor McLaughlin <stenzek@gmail.com>
|
2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
|
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
#include "cd_image.h"
|
|
|
|
#include "cd_subchannel_replacement.h"
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2022-07-08 12:43:38 +00:00
|
|
|
#include "common/assert.h"
|
|
|
|
#include "common/error.h"
|
|
|
|
#include "common/file_system.h"
|
|
|
|
#include "common/log.h"
|
|
|
|
#include "common/path.h"
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
#include <algorithm>
|
|
|
|
#include <cerrno>
|
|
|
|
#include <map>
|
2022-07-08 11:57:06 +00:00
|
|
|
#include <sstream>
|
2023-11-18 06:21:51 +00:00
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
Log_SetChannel(CDImageMemory);
|
|
|
|
|
2023-11-18 06:21:51 +00:00
|
|
|
namespace {
|
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
class CDImageM3u : public CDImage
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
CDImageM3u();
|
|
|
|
~CDImageM3u() override;
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
bool Open(const char* path, bool apply_patches, Error* Error);
|
2021-03-26 16:19:23 +00:00
|
|
|
|
|
|
|
bool ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index) override;
|
|
|
|
bool HasNonStandardSubchannel() const override;
|
|
|
|
|
|
|
|
bool HasSubImages() const override;
|
|
|
|
u32 GetSubImageCount() const override;
|
|
|
|
u32 GetCurrentSubImage() const override;
|
|
|
|
std::string GetSubImageMetadata(u32 index, const std::string_view& type) const override;
|
2023-08-19 13:40:36 +00:00
|
|
|
bool SwitchSubImage(u32 index, Error* error) override;
|
2021-03-26 16:19:23 +00:00
|
|
|
|
|
|
|
protected:
|
|
|
|
bool ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index) override;
|
|
|
|
|
|
|
|
private:
|
|
|
|
struct Entry
|
|
|
|
{
|
|
|
|
// TODO: Worth storing any other data?
|
|
|
|
std::string filename;
|
|
|
|
std::string title;
|
|
|
|
};
|
|
|
|
|
|
|
|
std::vector<Entry> m_entries;
|
|
|
|
std::unique_ptr<CDImage> m_current_image;
|
|
|
|
u32 m_current_image_index = UINT32_C(0xFFFFFFFF);
|
2022-08-27 06:52:24 +00:00
|
|
|
bool m_apply_patches = false;
|
2021-03-26 16:19:23 +00:00
|
|
|
};
|
|
|
|
|
2023-11-18 06:21:51 +00:00
|
|
|
} // namespace
|
|
|
|
|
2021-03-26 16:19:23 +00:00
|
|
|
CDImageM3u::CDImageM3u() = default;
|
|
|
|
|
|
|
|
CDImageM3u::~CDImageM3u() = default;
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
bool CDImageM3u::Open(const char* path, bool apply_patches, Error* error)
|
2021-03-26 16:19:23 +00:00
|
|
|
{
|
2021-04-16 17:47:40 +00:00
|
|
|
std::FILE* fp = FileSystem::OpenCFile(path, "rb");
|
|
|
|
if (!fp)
|
|
|
|
return false;
|
|
|
|
|
2021-05-12 06:35:18 +00:00
|
|
|
std::optional<std::string> m3u_file(FileSystem::ReadFileToString(fp));
|
2021-04-16 17:47:40 +00:00
|
|
|
std::fclose(fp);
|
2021-05-12 06:35:18 +00:00
|
|
|
if (!m3u_file.has_value() || m3u_file->empty())
|
2021-03-26 16:19:23 +00:00
|
|
|
{
|
2023-08-19 13:40:36 +00:00
|
|
|
Error::SetString(error, "Failed to read M3u file");
|
2021-03-26 16:19:23 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2021-05-12 06:35:18 +00:00
|
|
|
std::istringstream ifs(m3u_file.value());
|
2021-03-26 16:19:23 +00:00
|
|
|
m_filename = path;
|
2022-08-27 06:52:24 +00:00
|
|
|
m_apply_patches = apply_patches;
|
2021-03-26 16:19:23 +00:00
|
|
|
|
|
|
|
std::vector<std::string> entries;
|
|
|
|
std::string line;
|
|
|
|
while (std::getline(ifs, line))
|
|
|
|
{
|
|
|
|
u32 start_offset = 0;
|
|
|
|
while (start_offset < line.size() && std::isspace(line[start_offset]))
|
|
|
|
start_offset++;
|
|
|
|
|
|
|
|
// skip comments
|
|
|
|
if (start_offset == line.size() || line[start_offset] == '#')
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// strip ending whitespace
|
|
|
|
u32 end_offset = static_cast<u32>(line.size()) - 1;
|
|
|
|
while (std::isspace(line[end_offset]) && end_offset > start_offset)
|
|
|
|
end_offset--;
|
|
|
|
|
|
|
|
// anything?
|
|
|
|
if (start_offset == end_offset)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Entry entry;
|
2021-04-16 17:47:40 +00:00
|
|
|
std::string entry_filename(line.begin() + start_offset, line.begin() + end_offset + 1);
|
2022-07-08 11:57:06 +00:00
|
|
|
entry.title = Path::GetFileTitle(entry_filename);
|
|
|
|
if (!Path::IsAbsolute(entry_filename))
|
|
|
|
entry.filename = Path::BuildRelativePath(path, entry_filename);
|
2021-04-16 17:47:40 +00:00
|
|
|
else
|
|
|
|
entry.filename = std::move(entry_filename);
|
2021-03-26 16:19:23 +00:00
|
|
|
|
|
|
|
Log_DevPrintf("Read path from m3u: '%s'", entry.filename.c_str());
|
|
|
|
m_entries.push_back(std::move(entry));
|
|
|
|
}
|
|
|
|
|
|
|
|
Log_InfoPrintf("Loaded %zu paths from m3u '%s'", m_entries.size(), path);
|
|
|
|
return !m_entries.empty() && SwitchSubImage(0, error);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageM3u::HasNonStandardSubchannel() const
|
|
|
|
{
|
|
|
|
return m_current_image->HasNonStandardSubchannel();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageM3u::HasSubImages() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 CDImageM3u::GetSubImageCount() const
|
|
|
|
{
|
|
|
|
return static_cast<u32>(m_entries.size());
|
|
|
|
}
|
|
|
|
|
|
|
|
u32 CDImageM3u::GetCurrentSubImage() const
|
|
|
|
{
|
|
|
|
return m_current_image_index;
|
|
|
|
}
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
bool CDImageM3u::SwitchSubImage(u32 index, Error* error)
|
2021-03-26 16:19:23 +00:00
|
|
|
{
|
|
|
|
if (index >= m_entries.size())
|
|
|
|
return false;
|
|
|
|
else if (index == m_current_image_index)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
const Entry& entry = m_entries[index];
|
2022-08-27 06:52:24 +00:00
|
|
|
std::unique_ptr<CDImage> new_image = CDImage::Open(entry.filename.c_str(), m_apply_patches, error);
|
2021-03-26 16:19:23 +00:00
|
|
|
if (!new_image)
|
|
|
|
{
|
|
|
|
Log_ErrorPrintf("Failed to load subimage %u (%s)", index, entry.filename.c_str());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
CopyTOC(new_image.get());
|
|
|
|
m_current_image = std::move(new_image);
|
|
|
|
m_current_image_index = index;
|
|
|
|
if (!Seek(1, Position{0, 0, 0}))
|
|
|
|
Panic("Failed to seek to start after sub-image change.");
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string CDImageM3u::GetSubImageMetadata(u32 index, const std::string_view& type) const
|
|
|
|
{
|
|
|
|
if (index > m_entries.size())
|
|
|
|
return {};
|
|
|
|
|
|
|
|
if (type == "title")
|
|
|
|
return m_entries[index].title;
|
|
|
|
else if (type == "file_title")
|
2022-07-08 11:57:06 +00:00
|
|
|
return std::string(Path::GetFileTitle(m_entries[index].filename));
|
2021-03-26 16:19:23 +00:00
|
|
|
|
|
|
|
return CDImage::GetSubImageMetadata(index, type);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageM3u::ReadSectorFromIndex(void* buffer, const Index& index, LBA lba_in_index)
|
|
|
|
{
|
|
|
|
return m_current_image->ReadSectorFromIndex(buffer, index, lba_in_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDImageM3u::ReadSubChannelQ(SubChannelQ* subq, const Index& index, LBA lba_in_index)
|
|
|
|
{
|
|
|
|
return m_current_image->ReadSubChannelQ(subq, index, lba_in_index);
|
|
|
|
}
|
|
|
|
|
2023-08-19 13:40:36 +00:00
|
|
|
std::unique_ptr<CDImage> CDImage::OpenM3uImage(const char* filename, bool apply_patches, Error* error)
|
2021-03-26 16:19:23 +00:00
|
|
|
{
|
|
|
|
std::unique_ptr<CDImageM3u> image = std::make_unique<CDImageM3u>();
|
2022-08-27 06:52:24 +00:00
|
|
|
if (!image->Open(filename, apply_patches, error))
|
2021-03-26 16:19:23 +00:00
|
|
|
return {};
|
|
|
|
|
|
|
|
return image;
|
|
|
|
}
|