// SPDX-FileCopyrightText: 2019-2022 Connor McLaughlin // SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0) #include "iso_reader.h" #include "cd_image.h" #include "common/log.h" #include Log_SetChannel(ISOReader); static bool FilenamesEqual(const char* a, const char* b, u32 length) { u32 pos = 0; for (; pos < length && *a != '\0' && *b != '\0'; pos++) { if (std::tolower(*(a++)) != std::tolower(*(b++))) return false; } return true; } ISOReader::ISOReader() = default; ISOReader::~ISOReader() = default; bool ISOReader::Open(CDImage* image, u32 track_number) { m_image = image; m_track_number = track_number; if (!ReadPVD()) return false; return true; } bool ISOReader::ReadPVD() { // volume descriptor start at sector 16 if (!m_image->Seek(m_track_number, 16)) return false; // try only a maximum of 256 volume descriptors for (u32 i = 0; i < 256; i++) { u8 buffer[SECTOR_SIZE]; if (m_image->Read(CDImage::ReadMode::DataOnly, 1, buffer) != 1) return false; const ISOVolumeDescriptorHeader* header = reinterpret_cast(buffer); if (header->type_code != 1) continue; else if (header->type_code == 255) break; std::memcpy(&m_pvd, buffer, sizeof(ISOPrimaryVolumeDescriptor)); Log_DebugPrintf("PVD found at index %u", i); return true; } Log_ErrorPrint("PVD not found"); return false; } std::optional ISOReader::LocateFile(const char* path) { u8 sector_buffer[SECTOR_SIZE]; const ISODirectoryEntry* root_de = reinterpret_cast(m_pvd.root_directory_entry); if (*path == '\0' || std::strcmp(path, "/") == 0) { // locating the root directory return *root_de; } // start at the root directory return LocateFile(path, sector_buffer, root_de->location_le, root_de->length_le); } std::optional ISOReader::LocateFile(const char* path, u8* sector_buffer, u32 directory_record_lba, u32 directory_record_size) { if (directory_record_size == 0) { Log_ErrorPrintf("Directory entry record size 0 while looking for '%s'", path); return std::nullopt; } // strip any leading slashes const char* path_component_start = path; while (*path_component_start == '/' || *path_component_start == '\\') path_component_start++; u32 path_component_length = 0; const char* path_component_end = path_component_start; while (*path_component_end != '\0' && *path_component_end != '/' && *path_component_end != '\\') { path_component_length++; path_component_end++; } // start reading directory entries const u32 num_sectors = (directory_record_size + (SECTOR_SIZE - 1)) / SECTOR_SIZE; if (!m_image->Seek(m_track_number, directory_record_lba)) { Log_ErrorPrintf("Seek to LBA %u failed", directory_record_lba); return std::nullopt; } for (u32 i = 0; i < num_sectors; i++) { if (m_image->Read(CDImage::ReadMode::DataOnly, 1, sector_buffer) != 1) { Log_ErrorPrintf("Failed to read LBA %u", directory_record_lba + i); return std::nullopt; } u32 sector_offset = 0; while ((sector_offset + sizeof(ISODirectoryEntry)) < SECTOR_SIZE) { const ISODirectoryEntry* de = reinterpret_cast(§or_buffer[sector_offset]); const char* de_filename = reinterpret_cast(§or_buffer[sector_offset + sizeof(ISODirectoryEntry)]); if ((sector_offset + de->entry_length) > SECTOR_SIZE || de->filename_length > de->entry_length || de->entry_length < sizeof(ISODirectoryEntry)) { break; } sector_offset += de->entry_length; // skip current/parent directory if (de->filename_length == 1 && (*de_filename == '\x0' || *de_filename == '\x1')) continue; // check filename length if (de->filename_length < path_component_length) continue; if (de->flags & ISODirectoryEntryFlag_Directory) { // directories don't have the version? so check the length instead if (de->filename_length != path_component_length || !FilenamesEqual(de_filename, path_component_start, path_component_length)) { continue; } } else { // compare filename if (!FilenamesEqual(de_filename, path_component_start, path_component_length) || de_filename[path_component_length] != ';') { continue; } } // found it. is this the file we're looking for? if (*path_component_end == '\0') return *de; // if it is a directory, recurse into it if (de->flags & ISODirectoryEntryFlag_Directory) return LocateFile(path_component_end, sector_buffer, de->location_le, de->length_le); // we're looking for a directory but got a file Log_ErrorPrintf("Looking for directory but got file"); return std::nullopt; } } std::string temp(path_component_start, path_component_length); Log_ErrorPrintf("Path component '%s' not found", temp.c_str()); return std::nullopt; } std::vector ISOReader::GetFilesInDirectory(const char* path) { std::string base_path = path; u32 directory_record_lba; u32 directory_record_length; if (base_path.empty()) { // root directory const ISODirectoryEntry* root_de = reinterpret_cast(m_pvd.root_directory_entry); directory_record_lba = root_de->location_le; directory_record_length = root_de->length_le; } else { auto directory_de = LocateFile(base_path.c_str()); if (!directory_de) { Log_ErrorPrintf("Directory entry not found for '%s'", path); return {}; } if ((directory_de->flags & ISODirectoryEntryFlag_Directory) == 0) { Log_ErrorPrintf("Path '%s' is not a directory, can't list", path); return {}; } directory_record_lba = directory_de->location_le; directory_record_length = directory_de->length_le; if (base_path[base_path.size() - 1] != '/') base_path += '/'; } // start reading directory entries const u32 num_sectors = (directory_record_length + (SECTOR_SIZE - 1)) / SECTOR_SIZE; if (!m_image->Seek(m_track_number, directory_record_lba)) { Log_ErrorPrintf("Seek to LBA %u failed", directory_record_lba); return {}; } std::vector files; u8 sector_buffer[SECTOR_SIZE]; for (u32 i = 0; i < num_sectors; i++) { if (m_image->Read(CDImage::ReadMode::DataOnly, 1, sector_buffer) != 1) { Log_ErrorPrintf("Failed to read LBA %u", directory_record_lba + i); break; } u32 sector_offset = 0; while ((sector_offset + sizeof(ISODirectoryEntry)) < SECTOR_SIZE) { const ISODirectoryEntry* de = reinterpret_cast(§or_buffer[sector_offset]); const char* de_filename = reinterpret_cast(§or_buffer[sector_offset + sizeof(ISODirectoryEntry)]); if ((sector_offset + de->entry_length) > SECTOR_SIZE || de->filename_length > de->entry_length || de->entry_length < sizeof(ISODirectoryEntry)) { break; } sector_offset += de->entry_length; // skip current/parent directory if (de->filename_length == 1 && (*de_filename == '\x0' || *de_filename == '\x1')) continue; // strip off terminator/file version std::string filename(de_filename, de->filename_length); std::string::size_type pos = filename.rfind(';'); if (pos == std::string::npos) { Log_ErrorPrintf("Invalid filename '%s'", filename.c_str()); continue; } filename.erase(pos); if (!filename.empty()) files.push_back(base_path + filename); } } return files; } bool ISOReader::ReadFile(const char* path, std::vector* data) { auto de = LocateFile(path); if (!de) { Log_ErrorPrintf("File not found: '%s'", path); return false; } if (de->flags & ISODirectoryEntryFlag_Directory) { Log_ErrorPrintf("File is a directory: '%s'", path); return false; } if (!m_image->Seek(m_track_number, de->location_le)) return false; if (de->length_le == 0) { data->clear(); return true; } const u32 num_sectors = (de->length_le + (SECTOR_SIZE - 1)) / SECTOR_SIZE; data->resize(num_sectors * u64(SECTOR_SIZE)); if (m_image->Read(CDImage::ReadMode::DataOnly, num_sectors, data->data()) != num_sectors) return false; data->resize(de->length_le); return true; } bool ISOReader::FileExists(const char* path) { auto de = LocateFile(path); return de.has_value(); }