Improve error handling in DiscIO::CompressFileToBlob and DecompressBlobToFile

This commit is contained in:
Michael Ehrenreich 2015-03-01 00:49:58 +01:00
parent 4b7748f3c0
commit 9387603fcf

View File

@ -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();