mirror of
				https://github.com/Atmosphere-NX/Atmosphere-libs.git
				synced 2025-10-25 09:55:49 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			123 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			123 lines
		
	
	
		
			4.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) Atmosphère-NX
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms and conditions of the GNU General Public License,
 | |
|  * version 2, as published by the Free Software Foundation.
 | |
|  *
 | |
|  * This program is distributed in the hope it will be useful, but WITHOUT
 | |
|  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 | |
|  * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License for
 | |
|  * more details.
 | |
|  *
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program.  If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| #pragma once
 | |
| #include <stratosphere.hpp>
 | |
| 
 | |
| namespace ams::ncm {
 | |
| 
 | |
|     class FileMapperFile {
 | |
|         public:
 | |
|             enum class OpenMode {
 | |
|                 Read,
 | |
|                 ReadWrite,
 | |
|                 ReadWriteAppend,
 | |
|             };
 | |
|         private:
 | |
|             const char *m_path;
 | |
|             OpenMode m_mode;
 | |
|             util::optional<fs::FileHandle> m_file;
 | |
|             size_t m_file_size;
 | |
|             size_t m_max_size;
 | |
|         public:
 | |
|             FileMapperFile() : m_file(util::nullopt) { /* ... */ }
 | |
| 
 | |
|             ~FileMapperFile() {
 | |
|                 this->Finalize();
 | |
|             }
 | |
| 
 | |
|             Result Initialize(const char *path, OpenMode mode) {
 | |
|                 /* Set our path/mode. */
 | |
|                 m_path = path;
 | |
|                 m_mode = mode;
 | |
| 
 | |
|                 /* Ensure we're open. */
 | |
|                 R_TRY(this->EnsureOpen());
 | |
| 
 | |
|                 /* Get the file size. */
 | |
|                 s64 size;
 | |
|                 R_TRY(fs::GetFileSize(std::addressof(size), m_file.value()));
 | |
| 
 | |
|                 /* Set our file size/loaded size. */
 | |
|                 m_file_size = static_cast<size_t>(size);
 | |
|                 m_max_size  = static_cast<size_t>(size);
 | |
| 
 | |
|                 R_SUCCEED();
 | |
|             }
 | |
| 
 | |
|             void Finalize() {
 | |
|                 /* If we have a file, close (and flush) it. */
 | |
|                 if (m_file.has_value()) {
 | |
|                     if (m_mode != OpenMode::Read) {
 | |
|                         R_ABORT_UNLESS(fs::FlushFile(m_file.value()));
 | |
|                     }
 | |
|                     fs::CloseFile(m_file.value());
 | |
|                     m_file = util::nullopt;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             size_t GetFileSize() const { return m_file_size; }
 | |
|             size_t GetMaxSize() const { return m_max_size; }
 | |
| 
 | |
|             Result Read(size_t offset, void *dst, size_t size) {
 | |
|                 /* Determine the end offset. */
 | |
|                 const size_t end_offset = offset + size;
 | |
| 
 | |
|                 /* Unless we're allowed to append, we need to have a big enough file. */
 | |
|                 if (m_mode != OpenMode::ReadWriteAppend) {
 | |
|                     R_UNLESS(end_offset <= m_file_size, ncm::ResultMapperInvalidArgument());
 | |
|                 }
 | |
| 
 | |
|                 /* Clear the output. */
 | |
|                 std::memset(dst, 0, size);
 | |
| 
 | |
|                 /* Check that our offset is valid. */
 | |
|                 R_UNLESS(offset <= m_file_size, ncm::ResultMapperInvalidArgument());
 | |
| 
 | |
|                 /* Ensure we're open. */
 | |
|                 R_TRY(this->EnsureOpen());
 | |
| 
 | |
|                 /* Read what we can. */
 | |
|                 const size_t read_size = (offset + size >= m_file_size) ? (m_file_size - offset) : size;
 | |
|                 AMS_ASSERT(read_size >= size);
 | |
| 
 | |
|                 R_TRY(fs::ReadFile(m_file.value(), offset, dst, read_size));
 | |
| 
 | |
|                 /* Update our max size. */
 | |
|                 m_max_size = std::max<size_t>(m_max_size, offset + read_size);
 | |
| 
 | |
|                 R_SUCCEED();
 | |
|             }
 | |
|         private:
 | |
|             Result EnsureOpen() {
 | |
|                 /* If we've opened, we're done. */
 | |
|                 R_SUCCEED_IF(m_file.has_value());
 | |
| 
 | |
|                 /* Open based on our mode. */
 | |
|                 fs::FileHandle file;
 | |
|                 switch (m_mode) {
 | |
|                     case OpenMode::Read:            R_TRY(fs::OpenFile(std::addressof(file), m_path, fs::OpenMode_Read));      break;
 | |
|                     case OpenMode::ReadWrite:       R_TRY(fs::OpenFile(std::addressof(file), m_path, fs::OpenMode_ReadWrite)); break;
 | |
|                     case OpenMode::ReadWriteAppend: R_TRY(fs::OpenFile(std::addressof(file), m_path, fs::OpenMode_All));       break;
 | |
|                     AMS_UNREACHABLE_DEFAULT_CASE();
 | |
|                 }
 | |
| 
 | |
|                 /* Set our file. */
 | |
|                 m_file = file;
 | |
|                 R_SUCCEED();
 | |
|             }
 | |
|     };
 | |
| 
 | |
| } |