// Copyright 2016 Dolphin Emulator Project // Licensed under GPLv2+ // Refer to the license.txt file included. #include #include #include "Common/CommonFuncs.h" #include "Common/FileUtil.h" #include "DiscIO/TGCBlob.h" namespace { template struct Interval { T start; T length; T End() const { return start + length; } bool IsEmpty() const { return length == 0; } }; template void SplitInterval(T split_point, Interval interval, Interval* out_1, Interval* out_2) { if (interval.start < split_point) *out_1 = {interval.start, std::min(interval.length, split_point - interval.start)}; else *out_1 = {0, 0}; if (interval.End() > split_point) *out_2 = {std::max(interval.start, split_point), std::min(interval.length, interval.End() - split_point)}; else *out_2 = {0, 0}; } u32 SubtractBE32(u32 minuend_be, u32 subtrahend_le) { return Common::swap32(Common::swap32(minuend_be) - subtrahend_le); } void Replace8(u64 offset, u64 nbytes, u8* out_ptr, u64 replace_offset, u8 replace_value) { if (offset <= replace_offset && offset + nbytes > replace_offset) out_ptr[replace_offset - offset] = replace_value; } void Replace32(u64 offset, u64 nbytes, u8* out_ptr, u64 replace_offset, u32 replace_value) { for (size_t i = 0; i < sizeof(u32); ++i) Replace8(offset, nbytes, out_ptr, replace_offset + i, reinterpret_cast(&replace_value)[i]); } } namespace DiscIO { bool IsTGCBlob(const std::string& path) { File::IOFile file(path, "rb"); TGCHeader header; return (file.ReadArray(&header, 1) && header.magic == Common::swap32(0xAE0F38A2)); } std::unique_ptr TGCFileReader::Create(const std::string& path) { if (IsTGCBlob(path)) { File::IOFile file(path, "rb"); return std::unique_ptr(new TGCFileReader(std::move(file))); } return nullptr; } TGCFileReader::TGCFileReader(File::IOFile&& file) : m_file(std::move(file)) { m_file.ReadArray(&m_header, 1); u32 header_size = Common::swap32(m_header.header_size); m_size = m_file.GetSize(); m_file_offset = Common::swap32(m_header.unknown_important_2) - Common::swap32(m_header.unknown_important_1) + header_size; } u64 TGCFileReader::GetDataSize() const { return m_size + Common::swap32(m_header.unknown_important_2) - Common::swap32(m_header.unknown_important_1); } bool TGCFileReader::Read(u64 offset, u64 nbytes, u8* out_ptr) { Interval first_part; Interval empty_part; Interval last_part; u32 header_size = Common::swap32(m_header.header_size); SplitInterval(m_file_offset, Interval{offset, nbytes}, &first_part, &last_part); SplitInterval(m_size - header_size, first_part, &first_part, &empty_part); // Offsets before m_file_offset are read as usual if (!first_part.IsEmpty()) { if (!InternalRead(first_part.start, first_part.length, out_ptr + (first_part.start - offset))) return false; } // If any offset before m_file_offset isn't actually in the file, // treat it as 0x00 bytes (so e.g. MD5 calculation of the whole disc won't fail) if (!empty_part.IsEmpty()) std::fill_n(out_ptr + (empty_part.start - offset), empty_part.length, 0); // Offsets after m_file_offset are read as if they are (offset - m_file_offset) if (!last_part.IsEmpty()) { if (!InternalRead(last_part.start - m_file_offset, last_part.length, out_ptr + (last_part.start - offset))) return false; } return true; } bool TGCFileReader::InternalRead(u64 offset, u64 nbytes, u8* out_ptr) { u32 header_size = Common::swap32(m_header.header_size); if (m_file.Seek(offset + header_size, SEEK_SET) && m_file.ReadBytes(out_ptr, nbytes)) { Replace32(offset, nbytes, out_ptr, 0x420, SubtractBE32(m_header.dol_offset, header_size)); Replace32(offset, nbytes, out_ptr, 0x424, SubtractBE32(m_header.fst_offset, header_size)); return true; } m_file.Clear(); return false; } } // namespace DiscIO