mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2024-11-15 05:47:56 -07:00
Improve error handling in DiscIO::CompressFileToBlob and DecompressBlobToFile
This commit is contained in:
parent
4b7748f3c0
commit
9387603fcf
@ -108,7 +108,7 @@ void CompressedBlobReader::GetBlock(u64 block_num, u8 *out_ptr)
|
|||||||
u32 block_hash = HashAdler32(source, comp_block_size);
|
u32 block_hash = HashAdler32(source, comp_block_size);
|
||||||
if (block_hash != m_hashes[block_num])
|
if (block_hash != m_hashes[block_num])
|
||||||
PanicAlert("Hash of block %" PRIu64 " is %08x instead of %08x.\n"
|
PanicAlert("Hash of block %" PRIu64 " is %08x instead of %08x.\n"
|
||||||
"Your ISO, %s, is corrupt.",
|
"Your ISO, \"%s\", is corrupt.",
|
||||||
block_num, block_hash, m_hashes[block_num],
|
block_num, block_hash, m_hashes[block_num],
|
||||||
m_file_name.c_str());
|
m_file_name.c_str());
|
||||||
|
|
||||||
@ -150,7 +150,23 @@ bool CompressFileToBlob(const std::string& infile, const std::string& outfile, u
|
|||||||
|
|
||||||
if (IsCompressedBlob(infile))
|
if (IsCompressedBlob(infile))
|
||||||
{
|
{
|
||||||
PanicAlertT("%s is already compressed! Cannot compress it further.", infile.c_str());
|
PanicAlertT("\"%s\" is already compressed! Cannot compress it further.", infile.c_str());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
File::IOFile inf(infile, "rb");
|
||||||
|
if (!inf)
|
||||||
|
{
|
||||||
|
PanicAlertT("Failed to open the input file \"%s\".", infile.c_str());
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
File::IOFile f(outfile, "wb");
|
||||||
|
if (!f)
|
||||||
|
{
|
||||||
|
PanicAlertT("Failed to open the output file \"%s\".\n"
|
||||||
|
"Check that you have permissions to write the target folder and that the media can be written.",
|
||||||
|
outfile.c_str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -158,7 +174,7 @@ bool CompressFileToBlob(const std::string& infile, const std::string& outfile, u
|
|||||||
{
|
{
|
||||||
if (!DiscScrubber::SetupScrub(infile, block_size))
|
if (!DiscScrubber::SetupScrub(infile, block_size))
|
||||||
{
|
{
|
||||||
PanicAlertT("%s failed to be scrubbed. Probably the image is corrupt.", infile.c_str());
|
PanicAlertT("\"%s\" failed to be scrubbed. Probably the image is corrupt.", infile.c_str());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -167,14 +183,8 @@ bool CompressFileToBlob(const std::string& infile, const std::string& outfile, u
|
|||||||
|
|
||||||
z_stream z = {};
|
z_stream z = {};
|
||||||
if (deflateInit(&z, 9) != Z_OK)
|
if (deflateInit(&z, 9) != Z_OK)
|
||||||
return false;
|
|
||||||
|
|
||||||
File::IOFile inf(infile, "rb");
|
|
||||||
File::IOFile f(outfile, "wb");
|
|
||||||
|
|
||||||
if (!f || !inf)
|
|
||||||
{
|
{
|
||||||
deflateEnd(&z);
|
DiscScrubber::Cleanup();
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -204,7 +214,7 @@ bool CompressFileToBlob(const std::string& infile, const std::string& outfile, u
|
|||||||
int num_compressed = 0;
|
int num_compressed = 0;
|
||||||
int num_stored = 0;
|
int num_stored = 0;
|
||||||
int progress_monitor = std::max<int>(1, header.num_blocks / 1000);
|
int progress_monitor = std::max<int>(1, header.num_blocks / 1000);
|
||||||
bool was_cancelled = false;
|
bool success = true;
|
||||||
|
|
||||||
for (u32 i = 0; i < header.num_blocks; i++)
|
for (u32 i = 0; i < header.num_blocks; i++)
|
||||||
{
|
{
|
||||||
@ -216,9 +226,12 @@ bool CompressFileToBlob(const std::string& infile, const std::string& outfile, u
|
|||||||
ratio = (int)(100 * position / inpos);
|
ratio = (int)(100 * position / inpos);
|
||||||
|
|
||||||
std::string temp = StringFromFormat("%i of %i blocks. Compression ratio %i%%", i, header.num_blocks, ratio);
|
std::string temp = StringFromFormat("%i of %i blocks. Compression ratio %i%%", i, header.num_blocks, ratio);
|
||||||
was_cancelled = !callback(temp, (float)i / (float)header.num_blocks, arg);
|
bool was_cancelled = !callback(temp, (float)i / (float)header.num_blocks, arg);
|
||||||
if (was_cancelled)
|
if (was_cancelled)
|
||||||
|
{
|
||||||
|
success = false;
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
offsets[i] = position;
|
offsets[i] = position;
|
||||||
@ -240,35 +253,51 @@ bool CompressFileToBlob(const std::string& infile, const std::string& outfile, u
|
|||||||
if (retval != Z_OK)
|
if (retval != Z_OK)
|
||||||
{
|
{
|
||||||
ERROR_LOG(DISCIO, "Deflate failed");
|
ERROR_LOG(DISCIO, "Deflate failed");
|
||||||
goto cleanup;
|
success = false;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
int status = deflate(&z, Z_FINISH);
|
int status = deflate(&z, Z_FINISH);
|
||||||
int comp_size = block_size - z.avail_out;
|
int comp_size = block_size - z.avail_out;
|
||||||
|
|
||||||
|
u8* write_buf;
|
||||||
|
int write_size;
|
||||||
if ((status != Z_STREAM_END) || (z.avail_out < 10))
|
if ((status != Z_STREAM_END) || (z.avail_out < 10))
|
||||||
{
|
{
|
||||||
//PanicAlert("%i %i Store %i", i*block_size, position, comp_size);
|
//PanicAlert("%i %i Store %i", i*block_size, position, comp_size);
|
||||||
// let's store uncompressed
|
// let's store uncompressed
|
||||||
|
write_buf = in_buf;
|
||||||
offsets[i] |= 0x8000000000000000ULL;
|
offsets[i] |= 0x8000000000000000ULL;
|
||||||
f.WriteBytes(in_buf, block_size);
|
write_size = block_size;
|
||||||
hashes[i] = HashAdler32(in_buf, block_size);
|
|
||||||
position += block_size;
|
|
||||||
num_stored++;
|
num_stored++;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// let's store compressed
|
// let's store compressed
|
||||||
//PanicAlert("Comp %i to %i", block_size, comp_size);
|
//PanicAlert("Comp %i to %i", block_size, comp_size);
|
||||||
f.WriteBytes(out_buf, comp_size);
|
write_buf = out_buf;
|
||||||
hashes[i] = HashAdler32(out_buf, comp_size);
|
write_size = comp_size;
|
||||||
position += comp_size;
|
|
||||||
num_compressed++;
|
num_compressed++;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!f.WriteBytes(write_buf, write_size))
|
||||||
|
{
|
||||||
|
PanicAlertT(
|
||||||
|
"Failed to write the output file \"%s\".\n"
|
||||||
|
"Check that you have enough space available on the target drive.",
|
||||||
|
outfile.c_str());
|
||||||
|
success = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
position += write_size;
|
||||||
|
|
||||||
|
hashes[i] = HashAdler32(write_buf, write_size);
|
||||||
}
|
}
|
||||||
|
|
||||||
header.compressed_data_size = position;
|
header.compressed_data_size = position;
|
||||||
|
|
||||||
if (was_cancelled)
|
if (!success)
|
||||||
{
|
{
|
||||||
// Remove the incomplete output file.
|
// Remove the incomplete output file.
|
||||||
f.Close();
|
f.Close();
|
||||||
@ -283,7 +312,6 @@ bool CompressFileToBlob(const std::string& infile, const std::string& outfile, u
|
|||||||
f.WriteArray(hashes, header.num_blocks);
|
f.WriteArray(hashes, header.num_blocks);
|
||||||
}
|
}
|
||||||
|
|
||||||
cleanup:
|
|
||||||
// Cleanup
|
// Cleanup
|
||||||
delete[] in_buf;
|
delete[] in_buf;
|
||||||
delete[] out_buf;
|
delete[] out_buf;
|
||||||
@ -291,10 +319,13 @@ cleanup:
|
|||||||
delete[] hashes;
|
delete[] hashes;
|
||||||
|
|
||||||
deflateEnd(&z);
|
deflateEnd(&z);
|
||||||
|
|
||||||
DiscScrubber::Cleanup();
|
DiscScrubber::Cleanup();
|
||||||
callback("Done compressing disc image.", 1.0f, arg);
|
|
||||||
return true;
|
if (success)
|
||||||
|
{
|
||||||
|
callback("Done compressing disc image.", 1.0f, arg);
|
||||||
|
}
|
||||||
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool DecompressBlobToFile(const std::string& infile, const std::string& outfile, CompressCB callback, void* arg)
|
bool DecompressBlobToFile(const std::string& infile, const std::string& outfile, CompressCB callback, void* arg)
|
||||||
@ -307,11 +338,20 @@ bool DecompressBlobToFile(const std::string& infile, const std::string& outfile,
|
|||||||
|
|
||||||
std::unique_ptr<CompressedBlobReader> reader(CompressedBlobReader::Create(infile));
|
std::unique_ptr<CompressedBlobReader> reader(CompressedBlobReader::Create(infile));
|
||||||
if (!reader)
|
if (!reader)
|
||||||
|
{
|
||||||
|
PanicAlertT("Failed to open the input file \"%s\".", infile.c_str());
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
File::IOFile f(outfile, "wb");
|
File::IOFile f(outfile, "wb");
|
||||||
if (!f)
|
if (!f)
|
||||||
|
{
|
||||||
|
PanicAlertT(
|
||||||
|
"Failed to open the output file \"%s\".\n"
|
||||||
|
"Check that you have permissions to write the target folder and that the media can be written.",
|
||||||
|
outfile.c_str());
|
||||||
return false;
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
const CompressedBlobHeader &header = reader->GetHeader();
|
const CompressedBlobHeader &header = reader->GetHeader();
|
||||||
static const size_t BUFFER_BLOCKS = 32;
|
static const size_t BUFFER_BLOCKS = 32;
|
||||||
@ -320,22 +360,33 @@ bool DecompressBlobToFile(const std::string& infile, const std::string& outfile,
|
|||||||
std::vector<u8> buffer(buffer_size);
|
std::vector<u8> buffer(buffer_size);
|
||||||
u32 num_buffers = (header.num_blocks + BUFFER_BLOCKS - 1) / BUFFER_BLOCKS;
|
u32 num_buffers = (header.num_blocks + BUFFER_BLOCKS - 1) / BUFFER_BLOCKS;
|
||||||
int progress_monitor = std::max<int>(1, num_buffers / 100);
|
int progress_monitor = std::max<int>(1, num_buffers / 100);
|
||||||
bool was_cancelled = false;
|
bool success = true;
|
||||||
|
|
||||||
for (u64 i = 0; i < num_buffers; i++)
|
for (u64 i = 0; i < num_buffers; i++)
|
||||||
{
|
{
|
||||||
if (i % progress_monitor == 0)
|
if (i % progress_monitor == 0)
|
||||||
{
|
{
|
||||||
was_cancelled = !callback("Unpacking", (float)i / (float)num_buffers, arg);
|
bool was_cancelled = !callback("Unpacking", (float)i / (float)num_buffers, arg);
|
||||||
if (was_cancelled)
|
if (was_cancelled)
|
||||||
|
{
|
||||||
|
success = false;
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
const size_t sz = i == num_buffers - 1 ? last_buffer_size : buffer_size;
|
const size_t sz = i == num_buffers - 1 ? last_buffer_size : buffer_size;
|
||||||
reader->Read(i * buffer_size, sz, buffer.data());
|
reader->Read(i * buffer_size, sz, buffer.data());
|
||||||
f.WriteBytes(buffer.data(), sz);
|
if (!f.WriteBytes(buffer.data(), sz))
|
||||||
|
{
|
||||||
|
PanicAlertT(
|
||||||
|
"Failed to write the output file \"%s\".\n"
|
||||||
|
"Check that you have enough space available on the target drive.",
|
||||||
|
outfile.c_str());
|
||||||
|
success = false;
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (was_cancelled)
|
if (!success)
|
||||||
{
|
{
|
||||||
// Remove the incomplete output file.
|
// Remove the incomplete output file.
|
||||||
f.Close();
|
f.Close();
|
||||||
|
Loading…
Reference in New Issue
Block a user