| 
									
										
										
										
											2022-04-23 04:59:50 -04:00
										 |  |  | // SPDX-FileCopyrightText: Copyright 2018 yuzu Emulator Project
 | 
					
						
							|  |  |  | // SPDX-License-Identifier: GPL-2.0-or-later
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | #include <algorithm>
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-19 22:00:44 -04:00
										 |  |  | #include "common/assert.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  | #include "core/file_sys/vfs_concat.h"
 | 
					
						
							| 
									
										
										
										
											2018-09-25 17:26:09 -04:00
										 |  |  | #include "core/file_sys/vfs_static.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace FileSys { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  | ConcatenatedVfsFile::ConcatenatedVfsFile(ConcatenationMap&& concatenation_map_, std::string&& name_) | 
					
						
							|  |  |  |     : concatenation_map(std::move(concatenation_map_)), name(std::move(name_)) { | 
					
						
							|  |  |  |     DEBUG_ASSERT(this->VerifyContinuity()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ConcatenatedVfsFile::VerifyContinuity() const { | 
					
						
							|  |  |  |     u64 last_offset = 0; | 
					
						
							|  |  |  |     for (auto& entry : concatenation_map) { | 
					
						
							|  |  |  |         if (entry.offset != last_offset) { | 
					
						
							| 
									
										
										
										
											2018-09-19 22:00:44 -04:00
										 |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |         last_offset = entry.offset + entry.file->GetSize(); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2018-09-19 22:00:44 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-09-19 19:19:05 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-23 21:50:16 -04:00
										 |  |  | ConcatenatedVfsFile::~ConcatenatedVfsFile() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  | VirtualFile ConcatenatedVfsFile::MakeConcatenatedFile(const std::vector<VirtualFile>& files, | 
					
						
							|  |  |  |                                                       std::string&& name) { | 
					
						
							|  |  |  |     // Fold trivial cases.
 | 
					
						
							|  |  |  |     if (files.empty()) { | 
					
						
							| 
									
										
										
										
											2018-09-25 17:38:16 -04:00
										 |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (files.size() == 1) { | 
					
						
							|  |  |  |         return files.front(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Make the concatenation map from the input.
 | 
					
						
							|  |  |  |     std::vector<ConcatenationEntry> concatenation_map; | 
					
						
							|  |  |  |     concatenation_map.reserve(files.size()); | 
					
						
							|  |  |  |     u64 last_offset = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (auto& file : files) { | 
					
						
							|  |  |  |         concatenation_map.emplace_back(ConcatenationEntry{ | 
					
						
							|  |  |  |             .offset = last_offset, | 
					
						
							|  |  |  |             .file = file, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         last_offset += file->GetSize(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-25 17:38:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     return VirtualFile(new ConcatenatedVfsFile(std::move(concatenation_map), std::move(name))); | 
					
						
							| 
									
										
										
										
											2018-09-25 17:38:16 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | VirtualFile ConcatenatedVfsFile::MakeConcatenatedFile(u8 filler_byte, | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |                                                       const std::multimap<u64, VirtualFile>& files, | 
					
						
							|  |  |  |                                                       std::string&& name) { | 
					
						
							|  |  |  |     // Fold trivial cases.
 | 
					
						
							|  |  |  |     if (files.empty()) { | 
					
						
							| 
									
										
										
										
											2018-09-25 17:38:16 -04:00
										 |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (files.size() == 1) { | 
					
						
							| 
									
										
										
										
											2018-09-25 17:38:16 -04:00
										 |  |  |         return files.begin()->second; | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-25 17:38:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     // Make the concatenation map from the input.
 | 
					
						
							|  |  |  |     std::vector<ConcatenationEntry> concatenation_map; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     concatenation_map.reserve(files.size()); | 
					
						
							|  |  |  |     u64 last_offset = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Iteration of a multimap is ordered, so offset will be strictly non-decreasing.
 | 
					
						
							|  |  |  |     for (auto& [offset, file] : files) { | 
					
						
							|  |  |  |         if (offset > last_offset) { | 
					
						
							|  |  |  |             concatenation_map.emplace_back(ConcatenationEntry{ | 
					
						
							|  |  |  |                 .offset = last_offset, | 
					
						
							|  |  |  |                 .file = std::make_shared<StaticVfsFile>(filler_byte, offset - last_offset), | 
					
						
							|  |  |  |             }); | 
					
						
							| 
									
										
										
										
											2018-09-25 17:38:16 -04:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |         concatenation_map.emplace_back(ConcatenationEntry{ | 
					
						
							|  |  |  |             .offset = offset, | 
					
						
							|  |  |  |             .file = file, | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         last_offset = offset + file->GetSize(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-09-25 17:38:16 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     return VirtualFile(new ConcatenatedVfsFile(std::move(concatenation_map), std::move(name))); | 
					
						
							| 
									
										
										
										
											2018-09-25 17:38:16 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  | std::string ConcatenatedVfsFile::GetName() const { | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     if (concatenation_map.empty()) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  |         return ""; | 
					
						
							| 
									
										
										
										
											2020-12-10 01:31:58 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (!name.empty()) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  |         return name; | 
					
						
							| 
									
										
										
										
											2020-12-10 01:31:58 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     return concatenation_map.front().file->GetName(); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | std::size_t ConcatenatedVfsFile::GetSize() const { | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     if (concatenation_map.empty()) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  |         return 0; | 
					
						
							| 
									
										
										
										
											2020-12-10 01:31:58 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     return concatenation_map.back().offset + concatenation_map.back().file->GetSize(); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | bool ConcatenatedVfsFile::Resize(std::size_t new_size) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-10 01:31:58 -05:00
										 |  |  | VirtualDir ConcatenatedVfsFile::GetContainingDirectory() const { | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     if (concatenation_map.empty()) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  |         return nullptr; | 
					
						
							| 
									
										
										
										
											2020-12-10 01:31:58 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     return concatenation_map.front().file->GetContainingDirectory(); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ConcatenatedVfsFile::IsWritable() const { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool ConcatenatedVfsFile::IsReadable() const { | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | std::size_t ConcatenatedVfsFile::Read(u8* data, std::size_t length, std::size_t offset) const { | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     const ConcatenationEntry key{ | 
					
						
							|  |  |  |         .offset = offset, | 
					
						
							|  |  |  |         .file = nullptr, | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Read nothing if the map is empty.
 | 
					
						
							|  |  |  |     if (concatenation_map.empty()) { | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Binary search to find the iterator to the first position we can check.
 | 
					
						
							|  |  |  |     // It must exist, since we are not empty and are comparing unsigned integers.
 | 
					
						
							|  |  |  |     auto it = std::prev(std::upper_bound(concatenation_map.begin(), concatenation_map.end(), key)); | 
					
						
							|  |  |  |     u64 cur_length = length; | 
					
						
							|  |  |  |     u64 cur_offset = offset; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (cur_length > 0 && it != concatenation_map.end()) { | 
					
						
							|  |  |  |         // Check if we can read the file at this position.
 | 
					
						
							|  |  |  |         const auto& file = it->file; | 
					
						
							|  |  |  |         const u64 file_offset = it->offset; | 
					
						
							|  |  |  |         const u64 file_size = file->GetSize(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (cur_offset >= file_offset + file_size) { | 
					
						
							|  |  |  |             // Entirely out of bounds read.
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |         // Read the file at this position.
 | 
					
						
							|  |  |  |         const u64 intended_read_size = std::min<u64>(cur_length, file_size); | 
					
						
							|  |  |  |         const u64 actual_read_size = | 
					
						
							|  |  |  |             file->Read(data + (cur_offset - offset), intended_read_size, cur_offset - file_offset); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |         // Update tracking.
 | 
					
						
							|  |  |  |         cur_offset += actual_read_size; | 
					
						
							|  |  |  |         cur_length -= actual_read_size; | 
					
						
							|  |  |  |         it++; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-26 15:41:17 -04:00
										 |  |  |     return cur_offset - offset; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-09-15 15:21:06 +02:00
										 |  |  | std::size_t ConcatenatedVfsFile::Write(const u8* data, std::size_t length, std::size_t offset) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-02 02:34:40 -04:00
										 |  |  | bool ConcatenatedVfsFile::Rename(std::string_view new_name) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2018-09-19 22:00:44 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:46:41 -04:00
										 |  |  | } // namespace FileSys
 |