2022-12-04 11:03:45 +00:00
|
|
|
// SPDX-FileCopyrightText: 2019-2022 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"
|
2022-07-08 12:43:38 +00:00
|
|
|
#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)
|
|
|
|
{
|
|
|
|
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->SetFormattedStatusText("Computing hash for track %u/index %u...", track, index);
|
|
|
|
progress_callback->SetProgressRange(index_length);
|
|
|
|
|
|
|
|
if (!image->Seek(index_start))
|
|
|
|
{
|
|
|
|
progress_callback->DisplayFormattedModalError("Failed to seek to sector %u for track %u index %u", index_start,
|
|
|
|
track, index);
|
|
|
|
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);
|
|
|
|
|
2021-05-26 17:46:33 +00:00
|
|
|
if (!image->ReadRawSector(sector.data(), nullptr))
|
2020-06-07 15:19:35 +00:00
|
|
|
{
|
|
|
|
progress_callback->DisplayFormattedModalError("Failed to read sector %u from image", image->GetPositionOnDisc());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
digest->Update(sector.data(), static_cast<u32>(sector.size()));
|
|
|
|
}
|
|
|
|
|
|
|
|
progress_callback->SetProgressValue(index_length);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool ReadTrack(CDImage* image, u8 track, MD5Digest* digest, ProgressCallback* progress_callback)
|
|
|
|
{
|
|
|
|
static constexpr u8 INDICES_TO_READ = 2;
|
|
|
|
|
|
|
|
progress_callback->PushState();
|
|
|
|
|
2021-10-21 22:36:51 +00:00
|
|
|
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++)
|
|
|
|
{
|
2021-10-21 22:36:51 +00:00
|
|
|
progress_callback->SetProgressValue(progress);
|
2020-06-07 15:19:35 +00:00
|
|
|
|
|
|
|
// skip index 0 if data track
|
2021-10-21 22:36:51 +00:00
|
|
|
if (dataTrack && index == 0)
|
2020-06-07 15:19:35 +00:00
|
|
|
continue;
|
|
|
|
|
2021-10-21 22:36:51 +00:00
|
|
|
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();
|
|
|
|
}
|
|
|
|
|
2021-10-21 22:36:51 +00:00
|
|
|
progress_callback->SetProgressValue(progress);
|
2020-06-07 15:19:35 +00:00
|
|
|
progress_callback->PopState();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string HashToString(const Hash& hash)
|
|
|
|
{
|
2023-09-20 14:32:39 +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
|
|
|
}
|
2021-10-23 14:56:54 +00:00
|
|
|
|
2023-09-20 14:32:39 +00:00
|
|
|
std::optional<Hash> HashFromString(const 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;
|
2021-10-23 14:56:54 +00:00
|
|
|
}
|
2020-06-07 15:19:35 +00:00
|
|
|
|
|
|
|
bool GetImageHash(CDImage* image, Hash* out_hash,
|
|
|
|
ProgressCallback* progress_callback /*= ProgressCallback::NullProgressCallback*/)
|
|
|
|
{
|
|
|
|
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());
|
|
|
|
digest.Final(out_hash->data());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool GetTrackHash(CDImage* image, u8 track, Hash* out_hash,
|
|
|
|
ProgressCallback* progress_callback /*= ProgressCallback::NullProgressCallback*/)
|
|
|
|
{
|
|
|
|
MD5Digest digest;
|
|
|
|
if (!ReadTrack(image, track, &digest, progress_callback))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
digest.Final(out_hash->data());
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace CDImageHasher
|