mirror of
				https://git.zaroz.cloud/nintendo-back-up/yuzu/yuzu-mainline.git
				synced 2025-03-21 01:53:15 +00:00 
			
		
		
		
	ips_layer: Add support for escape sequences and midline comments
More accurately follows IPSwitch specification.
This commit is contained in:
		
							parent
							
								
									8886f2e55e
								
							
						
					
					
						commit
						9669cdb710
					
				@ -17,6 +17,11 @@ enum class IPSFileType {
 | 
				
			|||||||
    Error,
 | 
					    Error,
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const std::map<const char*, const char*> ESCAPE_CHARACTER_MAP{
 | 
				
			||||||
 | 
					    {"\\a", "\a"}, {"\\b", "\b"},  {"\\f", "\f"},  {"\\n", "\n"},  {"\\r", "\r"},  {"\\t", "\t"},
 | 
				
			||||||
 | 
					    {"\\v", "\v"}, {"\\\\", "\\"}, {"\\\'", "\'"}, {"\\\"", "\""}, {"\\\?", "\?"},
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static IPSFileType IdentifyMagic(const std::vector<u8>& magic) {
 | 
					static IPSFileType IdentifyMagic(const std::vector<u8>& magic) {
 | 
				
			||||||
    if (magic.size() != 5)
 | 
					    if (magic.size() != 5)
 | 
				
			||||||
        return IPSFileType::Error;
 | 
					        return IPSFileType::Error;
 | 
				
			||||||
@ -89,7 +94,7 @@ VirtualFile PatchIPS(const VirtualFile& in, const VirtualFile& ips) {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
IPSwitchCompiler::IPSwitchCompiler(VirtualFile patch_text_)
 | 
					IPSwitchCompiler::IPSwitchCompiler(VirtualFile patch_text_)
 | 
				
			||||||
    : valid(false), patch_text(std::move(patch_text_)), nso_build_id{}, is_little_endian(false),
 | 
					    : valid(false), patch_text(std::move(patch_text_)), nso_build_id{}, is_little_endian(false),
 | 
				
			||||||
      offset_shift(0), print_values(false) {
 | 
					      offset_shift(0), print_values(false), last_comment("") {
 | 
				
			||||||
    Parse();
 | 
					    Parse();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -105,6 +110,18 @@ static bool StartsWith(const std::string& base, const std::string& check) {
 | 
				
			|||||||
    return base.size() >= check.size() && base.substr(0, check.size()) == check;
 | 
					    return base.size() >= check.size() && base.substr(0, check.size()) == check;
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static std::string EscapeStringSequences(std::string in) {
 | 
				
			||||||
 | 
					    for (const auto& seq : ESCAPE_CHARACTER_MAP) {
 | 
				
			||||||
 | 
					        for (auto index = in.find(seq.first); index != std::string::npos;
 | 
				
			||||||
 | 
					             index = in.find(seq.first, index)) {
 | 
				
			||||||
 | 
					            in.replace(index, std::strlen(seq.first), seq.second);
 | 
				
			||||||
 | 
					            index += std::strlen(seq.second);
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    return in;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void IPSwitchCompiler::Parse() {
 | 
					void IPSwitchCompiler::Parse() {
 | 
				
			||||||
    const auto bytes = patch_text->ReadAllBytes();
 | 
					    const auto bytes = patch_text->ReadAllBytes();
 | 
				
			||||||
    std::stringstream s;
 | 
					    std::stringstream s;
 | 
				
			||||||
@ -121,6 +138,13 @@ void IPSwitchCompiler::Parse() {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    for (std::size_t i = 0; i < lines.size(); ++i) {
 | 
					    for (std::size_t i = 0; i < lines.size(); ++i) {
 | 
				
			||||||
        auto line = lines[i];
 | 
					        auto line = lines[i];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					        // Remove midline comments
 | 
				
			||||||
 | 
					        if (!StartsWith(line, "//") && line.find("//") != std::string::npos) {
 | 
				
			||||||
 | 
					            last_comment = line.substr(line.find("//") + 2);
 | 
				
			||||||
 | 
					            line = line.substr(0, line.find("//"));
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
        if (StartsWith(line, "@stop")) {
 | 
					        if (StartsWith(line, "@stop")) {
 | 
				
			||||||
            // Force stop
 | 
					            // Force stop
 | 
				
			||||||
            break;
 | 
					            break;
 | 
				
			||||||
@ -132,11 +156,18 @@ void IPSwitchCompiler::Parse() {
 | 
				
			|||||||
            nso_build_id = Common::HexStringToArray<0x20>(raw_build_id);
 | 
					            nso_build_id = Common::HexStringToArray<0x20>(raw_build_id);
 | 
				
			||||||
        } else if (StartsWith(line, "@flag offset_shift ")) {
 | 
					        } else if (StartsWith(line, "@flag offset_shift ")) {
 | 
				
			||||||
            // Offset Shift Flag
 | 
					            // Offset Shift Flag
 | 
				
			||||||
            offset_shift = std::stoull(line.substr(19), nullptr, 0);
 | 
					            offset_shift = std::stoll(line.substr(19), nullptr, 0);
 | 
				
			||||||
        } else if (StartsWith(line, "#")) {
 | 
					        } else if (StartsWith(line, "#")) {
 | 
				
			||||||
            // Mandatory Comment
 | 
					            // Mandatory Comment
 | 
				
			||||||
            LOG_INFO(Loader, "[IPSwitchCompiler ('{}')] Forced output comment: {}",
 | 
					            LOG_INFO(Loader, "[IPSwitchCompiler ('{}')] Forced output comment: {}",
 | 
				
			||||||
                     patch_text->GetName(), line.substr(1));
 | 
					                     patch_text->GetName(), line.substr(1));
 | 
				
			||||||
 | 
					        } else if (StartsWith(line, "//")) {
 | 
				
			||||||
 | 
					            // Normal Comment
 | 
				
			||||||
 | 
					            last_comment = line.substr(2);
 | 
				
			||||||
 | 
					            if (last_comment.find_first_not_of(' ') == std::string::npos)
 | 
				
			||||||
 | 
					                continue;
 | 
				
			||||||
 | 
					            if (last_comment.find_first_not_of(' ') != 0)
 | 
				
			||||||
 | 
					                last_comment = last_comment.substr(last_comment.find_first_not_of(' '));
 | 
				
			||||||
        } else if (StartsWith(line, "@little-endian")) {
 | 
					        } else if (StartsWith(line, "@little-endian")) {
 | 
				
			||||||
            // Set values to read as little endian
 | 
					            // Set values to read as little endian
 | 
				
			||||||
            is_little_endian = true;
 | 
					            is_little_endian = true;
 | 
				
			||||||
@ -151,11 +182,10 @@ void IPSwitchCompiler::Parse() {
 | 
				
			|||||||
            const auto enabled = StartsWith(line, "@enabled");
 | 
					            const auto enabled = StartsWith(line, "@enabled");
 | 
				
			||||||
            if (i == 0)
 | 
					            if (i == 0)
 | 
				
			||||||
                return;
 | 
					                return;
 | 
				
			||||||
            const auto name = lines[i - 1].substr(3);
 | 
					 | 
				
			||||||
            LOG_INFO(Loader, "[IPSwitchCompiler ('{}')] Parsing patch '{}' ({})",
 | 
					            LOG_INFO(Loader, "[IPSwitchCompiler ('{}')] Parsing patch '{}' ({})",
 | 
				
			||||||
                     patch_text->GetName(), name, line.substr(1));
 | 
					                     patch_text->GetName(), last_comment, line.substr(1));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            IPSwitchPatch patch{name, enabled, {}};
 | 
					            IPSwitchPatch patch{last_comment, enabled, {}};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
            // Read rest of patch
 | 
					            // Read rest of patch
 | 
				
			||||||
            while (true) {
 | 
					            while (true) {
 | 
				
			||||||
@ -173,10 +203,11 @@ void IPSwitchCompiler::Parse() {
 | 
				
			|||||||
                // 9 - first char of replacement val
 | 
					                // 9 - first char of replacement val
 | 
				
			||||||
                if (line[9] == '\"') {
 | 
					                if (line[9] == '\"') {
 | 
				
			||||||
                    // string replacement
 | 
					                    // string replacement
 | 
				
			||||||
                    const auto end_index = line.find_last_of('\"');
 | 
					                    const auto end_index = line.find('\"', 10);
 | 
				
			||||||
                    if (end_index == std::string::npos || end_index < 10)
 | 
					                    if (end_index == std::string::npos || end_index < 10)
 | 
				
			||||||
                        return;
 | 
					                        return;
 | 
				
			||||||
                    const auto value = line.substr(10, end_index - 10);
 | 
					                    auto value = line.substr(10, end_index - 10);
 | 
				
			||||||
 | 
					                    value = EscapeStringSequences(value);
 | 
				
			||||||
                    replace.reserve(value.size());
 | 
					                    replace.reserve(value.size());
 | 
				
			||||||
                    std::copy(value.begin(), value.end(), std::back_inserter(replace));
 | 
					                    std::copy(value.begin(), value.end(), std::back_inserter(replace));
 | 
				
			||||||
                } else {
 | 
					                } else {
 | 
				
			||||||
 | 
				
			|||||||
@ -34,8 +34,9 @@ private:
 | 
				
			|||||||
    std::vector<IPSwitchPatch> patches;
 | 
					    std::vector<IPSwitchPatch> patches;
 | 
				
			||||||
    std::array<u8, 0x20> nso_build_id;
 | 
					    std::array<u8, 0x20> nso_build_id;
 | 
				
			||||||
    bool is_little_endian;
 | 
					    bool is_little_endian;
 | 
				
			||||||
    u64 offset_shift;
 | 
					    s64 offset_shift;
 | 
				
			||||||
    bool print_values;
 | 
					    bool print_values;
 | 
				
			||||||
 | 
					    std::string last_comment;
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
 | 
					
 | 
				
			||||||
} // namespace FileSys
 | 
					} // namespace FileSys
 | 
				
			||||||
 | 
				
			|||||||
@ -36,6 +36,7 @@ public:
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
    // Currently tracked NSO patches:
 | 
					    // Currently tracked NSO patches:
 | 
				
			||||||
    // - IPS
 | 
					    // - IPS
 | 
				
			||||||
 | 
					    // - IPSwitch
 | 
				
			||||||
    std::vector<u8> PatchNSO(const std::vector<u8>& nso) const;
 | 
					    std::vector<u8> PatchNSO(const std::vector<u8>& nso) const;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    // Checks to see if PatchNSO() will have any effect given the NSO's build ID.
 | 
					    // Checks to see if PatchNSO() will have any effect given the NSO's build ID.
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
		Reference in New Issue
	
	Block a user