| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  | // Copyright 2018 yuzu emulator team
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  | #include <cstring>
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  | #include "common/common_funcs.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  | #include "common/swap.h"
 | 
					
						
							|  |  |  | #include "content_archive.h"
 | 
					
						
							|  |  |  | #include "core/file_sys/nca_metadata.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace FileSys { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  | bool operator>=(TitleType lhs, TitleType rhs) { | 
					
						
							|  |  |  |     return static_cast<size_t>(lhs) >= static_cast<size_t>(rhs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool operator<=(TitleType lhs, TitleType rhs) { | 
					
						
							|  |  |  |     return static_cast<size_t>(lhs) <= static_cast<size_t>(rhs); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CNMT::CNMT(VirtualFile file) { | 
					
						
							|  |  |  |     if (file->ReadObject(&header) != sizeof(CNMTHeader)) | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // If type is {Application, Update, AOC} has opt-header.
 | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     if (header.type >= TitleType::Application && header.type <= TitleType::AOC) { | 
					
						
							|  |  |  |         if (file->ReadObject(&opt_header, sizeof(CNMTHeader)) != sizeof(OptionalHeader)) { | 
					
						
							|  |  |  |             LOG_WARNING(Loader, "Failed to read optional header."); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     for (u16 i = 0; i < header.number_content_entries; ++i) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |         auto& next = content_records.emplace_back(ContentRecord{}); | 
					
						
							|  |  |  |         if (file->ReadObject(&next, sizeof(CNMTHeader) + i * sizeof(ContentRecord) + | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |                                         header.table_offset) != sizeof(ContentRecord)) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |             content_records.erase(content_records.end() - 1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     for (u16 i = 0; i < header.number_meta_entries; ++i) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |         auto& next = meta_records.emplace_back(MetaRecord{}); | 
					
						
							|  |  |  |         if (file->ReadObject(&next, sizeof(CNMTHeader) + i * sizeof(MetaRecord) + | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |                                         header.table_offset) != sizeof(MetaRecord)) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |             meta_records.erase(meta_records.end() - 1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CNMT::CNMT(CNMTHeader header, OptionalHeader opt_header, std::vector<ContentRecord> content_records, | 
					
						
							|  |  |  |            std::vector<MetaRecord> meta_records) | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     : header(std::move(header)), opt_header(std::move(opt_header)), | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |       content_records(std::move(content_records)), meta_records(std::move(meta_records)) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u64 CNMT::GetTitleID() const { | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     return header.title_id; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | u32 CNMT::GetTitleVersion() const { | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     return header.title_version; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | TitleType CNMT::GetType() const { | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     return header.type; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const std::vector<ContentRecord>& CNMT::GetContentRecords() const { | 
					
						
							|  |  |  |     return content_records; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | const std::vector<MetaRecord>& CNMT::GetMetaRecords() const { | 
					
						
							|  |  |  |     return meta_records; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CNMT::UnionRecords(const CNMT& other) { | 
					
						
							|  |  |  |     bool change = false; | 
					
						
							|  |  |  |     for (const auto& rec : other.content_records) { | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |         const auto iter = std::find_if(content_records.begin(), content_records.end(), | 
					
						
							|  |  |  |                                        [&rec](const ContentRecord& r) { | 
					
						
							|  |  |  |                                            return r.nca_id == rec.nca_id && r.type == rec.type; | 
					
						
							|  |  |  |                                        }); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |         if (iter == content_records.end()) { | 
					
						
							|  |  |  |             content_records.emplace_back(rec); | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |             ++header.number_content_entries; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |             change = true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     for (const auto& rec : other.meta_records) { | 
					
						
							|  |  |  |         const auto iter = | 
					
						
							| 
									
										
										
										
											2018-08-09 23:10:32 -04:00
										 |  |  |             std::find_if(meta_records.begin(), meta_records.end(), [&rec](const MetaRecord& r) { | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |                 return r.title_id == rec.title_id && r.title_version == rec.title_version && | 
					
						
							|  |  |  |                        r.type == rec.type; | 
					
						
							|  |  |  |             }); | 
					
						
							|  |  |  |         if (iter == meta_records.end()) { | 
					
						
							|  |  |  |             meta_records.emplace_back(rec); | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |             ++header.number_meta_entries; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |             change = true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return change; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::vector<u8> CNMT::Serialize() const { | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     const bool has_opt_header = | 
					
						
							|  |  |  |         header.type >= TitleType::Application && header.type <= TitleType::AOC; | 
					
						
							| 
									
										
										
										
											2018-08-12 15:55:44 -04:00
										 |  |  |     const auto dead_zone = header.table_offset + sizeof(CNMTHeader); | 
					
						
							|  |  |  |     std::vector<u8> out( | 
					
						
							|  |  |  |         std::max(sizeof(CNMTHeader) + (has_opt_header ? sizeof(OptionalHeader) : 0), dead_zone) + | 
					
						
							|  |  |  |         content_records.size() * sizeof(ContentRecord) + meta_records.size() * sizeof(MetaRecord)); | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     memcpy(out.data(), &header, sizeof(CNMTHeader)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Optional Header
 | 
					
						
							|  |  |  |     if (has_opt_header) { | 
					
						
							|  |  |  |         memcpy(out.data() + sizeof(CNMTHeader), &opt_header, sizeof(OptionalHeader)); | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-08-11 15:39:09 -04:00
										 |  |  |     auto offset = header.table_offset; | 
					
						
							| 
									
										
										
										
											2018-08-09 20:51:52 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (const auto& rec : content_records) { | 
					
						
							|  |  |  |         memcpy(out.data() + offset + sizeof(CNMTHeader), &rec, sizeof(ContentRecord)); | 
					
						
							|  |  |  |         offset += sizeof(ContentRecord); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (const auto& rec : meta_records) { | 
					
						
							|  |  |  |         memcpy(out.data() + offset + sizeof(CNMTHeader), &rec, sizeof(MetaRecord)); | 
					
						
							|  |  |  |         offset += sizeof(MetaRecord); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return out; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | } // namespace FileSys
 |