Duckstation/src/util/cd_image_hasher.cpp

143 lines
4.3 KiB
C++
Raw Normal View History

// SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
// SPDX-License-Identifier: (GPL-3.0 OR CC-BY-NC-ND-4.0)
2020-06-07 15:19:35 +00:00
#include "cd_image_hasher.h"
#include "cd_image.h"
#include "util/host.h"
#include "common/md5_digest.h"
#include "common/string_util.h"
2020-06-07 15:19:35 +00:00
namespace CDImageHasher {
static bool ReadIndex(CDImage* image, u8 track, u8 index, MD5Digest* digest, ProgressCallback* progress_callback);
static bool ReadTrack(CDImage* image, u8 track, MD5Digest* digest, ProgressCallback* progress_callback);
} // namespace CDImageHasher
bool CDImageHasher::ReadIndex(CDImage* image, u8 track, u8 index, MD5Digest* digest,
ProgressCallback* progress_callback)
2020-06-07 15:19:35 +00:00
{
const CDImage::LBA index_start = image->GetTrackIndexPosition(track, index);
const u32 index_length = image->GetTrackIndexLength(track, index);
const u32 update_interval = std::max<u32>(index_length / 100u, 1u);
progress_callback->SetStatusText(
fmt::format(TRANSLATE_FS("CDImageHasher", "Computing hash for Track {}/Index {}..."), track, index).c_str());
2020-06-07 15:19:35 +00:00
progress_callback->SetProgressRange(index_length);
if (!image->Seek(index_start))
{
progress_callback->FormatModalError("Failed to seek to sector {} for track {} index {}", index_start, track, index);
2020-06-07 15:19:35 +00:00
return false;
}
std::array<u8, CDImage::RAW_SECTOR_SIZE> sector;
for (u32 lba = 0; lba < index_length; lba++)
{
if ((lba % update_interval) == 0)
progress_callback->SetProgressValue(lba);
if (!image->ReadRawSector(sector.data(), nullptr))
2020-06-07 15:19:35 +00:00
{
progress_callback->FormatModalError("Failed to read sector {} from image", image->GetPositionOnDisc());
2020-06-07 15:19:35 +00:00
return false;
}
2024-07-18 07:33:15 +00:00
digest->Update(sector);
2020-06-07 15:19:35 +00:00
}
progress_callback->SetProgressValue(index_length);
return true;
}
bool CDImageHasher::ReadTrack(CDImage* image, u8 track, MD5Digest* digest, ProgressCallback* progress_callback)
2020-06-07 15:19:35 +00:00
{
static constexpr u8 INDICES_TO_READ = 2;
progress_callback->PushState();
const bool dataTrack = track == 1;
progress_callback->SetProgressRange(dataTrack ? 1 : 2);
u8 progress = 0;
2020-06-07 15:19:35 +00:00
for (u8 index = 0; index < INDICES_TO_READ; index++)
{
progress_callback->SetProgressValue(progress);
2020-06-07 15:19:35 +00:00
// skip index 0 if data track
if (dataTrack && index == 0)
2020-06-07 15:19:35 +00:00
continue;
progress++;
2020-06-07 15:19:35 +00:00
progress_callback->PushState();
if (!ReadIndex(image, track, index, digest, progress_callback))
{
progress_callback->PopState();
progress_callback->PopState();
return false;
}
progress_callback->PopState();
}
progress_callback->SetProgressValue(progress);
2020-06-07 15:19:35 +00:00
progress_callback->PopState();
return true;
}
std::string CDImageHasher::HashToString(const Hash& hash)
2020-06-07 15:19:35 +00:00
{
return fmt::format("{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}{:02x}",
hash[0], hash[1], hash[2], hash[3], hash[4], hash[5], hash[6], hash[7], hash[8], hash[9], hash[10],
hash[11], hash[12], hash[13], hash[14], hash[15]);
2020-06-07 15:19:35 +00:00
}
2024-05-05 10:21:54 +00:00
std::optional<CDImageHasher::Hash> CDImageHasher::HashFromString(std::string_view str)
{
auto decoded = StringUtil::DecodeHex(str);
if (decoded && decoded->size() == std::tuple_size_v<Hash>)
{
Hash result;
std::copy(decoded->begin(), decoded->end(), result.begin());
return result;
}
return std::nullopt;
}
2020-06-07 15:19:35 +00:00
bool CDImageHasher::GetImageHash(CDImage* image, Hash* out_hash,
ProgressCallback* progress_callback /*= ProgressCallback::NullProgressCallback*/)
2020-06-07 15:19:35 +00:00
{
MD5Digest digest;
progress_callback->SetProgressRange(image->GetTrackCount());
progress_callback->SetProgressValue(0);
progress_callback->PushState();
for (u32 i = 1; i <= image->GetTrackCount(); i++)
{
progress_callback->SetProgressValue(i - 1);
2021-06-26 05:05:21 +00:00
if (!ReadTrack(image, static_cast<u8>(i), &digest, progress_callback))
2020-06-07 15:19:35 +00:00
{
progress_callback->PopState();
return false;
}
}
progress_callback->SetProgressValue(image->GetTrackCount());
2024-07-18 07:33:15 +00:00
digest.Final(*out_hash);
2020-06-07 15:19:35 +00:00
return true;
}
bool CDImageHasher::GetTrackHash(CDImage* image, u8 track, Hash* out_hash,
ProgressCallback* progress_callback /*= ProgressCallback::NullProgressCallback*/)
2020-06-07 15:19:35 +00:00
{
MD5Digest digest;
if (!ReadTrack(image, track, &digest, progress_callback))
return false;
2024-07-18 07:33:15 +00:00
digest.Final(*out_hash);
2020-06-07 15:19:35 +00:00
return true;
}