mirror of
				https://github.com/dolphin-emu/dolphin.git
				synced 2025-10-25 01:19:19 +00:00 
			
		
		
		
	Previously, we had WBFS and CISO which both returned an upper bound of the size, and other formats which returned an accurate size. But now we also have NFS, which returns a lower bound of the size. To allow VolumeVerifier to make better informed decisions for NFS, let's use an enum instead of a bool for the type of data size a blob has.
		
			
				
	
	
		
			117 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			117 lines
		
	
	
	
		
			2.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // Copyright 2008 Dolphin Emulator Project
 | |
| // SPDX-License-Identifier: GPL-2.0-or-later
 | |
| 
 | |
| #include "DiscIO/FileBlob.h"
 | |
| 
 | |
| #include <algorithm>
 | |
| #include <memory>
 | |
| #include <string>
 | |
| #include <utility>
 | |
| #include <vector>
 | |
| 
 | |
| #include "Common/Assert.h"
 | |
| #include "Common/FileUtil.h"
 | |
| #include "Common/MsgHandler.h"
 | |
| 
 | |
| namespace DiscIO
 | |
| {
 | |
| PlainFileReader::PlainFileReader(File::IOFile file) : m_file(std::move(file))
 | |
| {
 | |
|   m_size = m_file.GetSize();
 | |
| }
 | |
| 
 | |
| std::unique_ptr<PlainFileReader> PlainFileReader::Create(File::IOFile file)
 | |
| {
 | |
|   if (file)
 | |
|     return std::unique_ptr<PlainFileReader>(new PlainFileReader(std::move(file)));
 | |
| 
 | |
|   return nullptr;
 | |
| }
 | |
| 
 | |
| bool PlainFileReader::Read(u64 offset, u64 nbytes, u8* out_ptr)
 | |
| {
 | |
|   if (m_file.Seek(offset, File::SeekOrigin::Begin) && m_file.ReadBytes(out_ptr, nbytes))
 | |
|   {
 | |
|     return true;
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     m_file.ClearError();
 | |
|     return false;
 | |
|   }
 | |
| }
 | |
| 
 | |
| bool ConvertToPlain(BlobReader* infile, const std::string& infile_path,
 | |
|                     const std::string& outfile_path, CompressCB callback)
 | |
| {
 | |
|   ASSERT(infile->GetDataSizeType() == DataSizeType::Accurate);
 | |
| 
 | |
|   File::IOFile outfile(outfile_path, "wb");
 | |
|   if (!outfile)
 | |
|   {
 | |
|     PanicAlertFmtT(
 | |
|         "Failed to open the output file \"{0}\".\n"
 | |
|         "Check that you have permissions to write the target folder and that the media can "
 | |
|         "be written.",
 | |
|         outfile_path);
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   constexpr size_t DESIRED_BUFFER_SIZE = 0x80000;
 | |
|   u64 buffer_size = infile->GetBlockSize();
 | |
|   if (buffer_size == 0)
 | |
|   {
 | |
|     buffer_size = DESIRED_BUFFER_SIZE;
 | |
|   }
 | |
|   else
 | |
|   {
 | |
|     while (buffer_size < DESIRED_BUFFER_SIZE)
 | |
|       buffer_size *= 2;
 | |
|   }
 | |
| 
 | |
|   std::vector<u8> buffer(buffer_size);
 | |
|   const u64 num_buffers = (infile->GetDataSize() + buffer_size - 1) / buffer_size;
 | |
|   int progress_monitor = std::max<int>(1, num_buffers / 100);
 | |
|   bool success = true;
 | |
| 
 | |
|   for (u64 i = 0; i < num_buffers; i++)
 | |
|   {
 | |
|     if (i % progress_monitor == 0)
 | |
|     {
 | |
|       const bool was_cancelled =
 | |
|           !callback(Common::GetStringT("Unpacking"), (float)i / (float)num_buffers);
 | |
|       if (was_cancelled)
 | |
|       {
 | |
|         success = false;
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|     const u64 inpos = i * buffer_size;
 | |
|     const u64 sz = std::min(buffer_size, infile->GetDataSize() - inpos);
 | |
|     if (!infile->Read(inpos, sz, buffer.data()))
 | |
|     {
 | |
|       PanicAlertFmtT("Failed to read from the input file \"{0}\".", infile_path);
 | |
|       success = false;
 | |
|       break;
 | |
|     }
 | |
|     if (!outfile.WriteBytes(buffer.data(), sz))
 | |
|     {
 | |
|       PanicAlertFmtT("Failed to write the output file \"{0}\".\n"
 | |
|                      "Check that you have enough space available on the target drive.",
 | |
|                      outfile_path);
 | |
|       success = false;
 | |
|       break;
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!success)
 | |
|   {
 | |
|     // Remove the incomplete output file.
 | |
|     outfile.Close();
 | |
|     File::Delete(outfile_path);
 | |
|   }
 | |
| 
 | |
|   return success;
 | |
| }
 | |
| 
 | |
| }  // namespace DiscIO
 |