From 20994f07d041826858fab82b94fd426da06a1321 Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Fri, 4 Oct 2024 21:52:00 +0800 Subject: [PATCH 01/11] Add rough check for removal of code locals from FUNC in 2024.8 --- UndertaleModLib/UndertaleChunks.cs | 45 +++++++++++++++++++++++++++--- 1 file changed, 41 insertions(+), 4 deletions(-) diff --git a/UndertaleModLib/UndertaleChunks.cs b/UndertaleModLib/UndertaleChunks.cs index 55466bc2d..0fa26f315 100644 --- a/UndertaleModLib/UndertaleChunks.cs +++ b/UndertaleModLib/UndertaleChunks.cs @@ -1367,9 +1367,35 @@ public class UndertaleChunkFUNC : UndertaleChunk public UndertaleSimpleList Functions = new UndertaleSimpleList(); public UndertaleSimpleList CodeLocals = new UndertaleSimpleList(); + private static bool checkedFor2024_8; + + private void CheckFor2024_8(UndertaleReader reader) + { + if (reader.undertaleData.IsVersionAtLeast(2024, 8) + || reader.Bytecode14OrLower || Length == 0) // Irrelevant or non-deductible + { + checkedFor2024_8 = true; + return; + } + + long returnPos = reader.Position; + + // The CodeLocals list was removed in 2024.8, so we check for its absence here + + uint funcCount = reader.ReadUInt32(); + // If this is 2024.8, the chunk should only contain the Functions list. + // Considering Functions is a SimpleList, this would make the chunk size + // sizeof funcCount + sizeof UndertaleFunction * funcCount + if (Length == 4 + 12 * funcCount) + reader.undertaleData.SetGMS2Version(2024, 8); + + reader.Position = returnPos; + checkedFor2024_8 = true; + } + internal override void SerializeChunk(UndertaleWriter writer) { - if (Functions == null && CodeLocals == null) + if (Functions is null && (writer.undertaleData.IsVersionAtLeast(2024, 8) || CodeLocals is null)) return; UndertaleInstruction.Reference.SerializeReferenceChain(writer, writer.undertaleData.Code, Functions); @@ -1382,12 +1408,16 @@ internal override void SerializeChunk(UndertaleWriter writer) else { writer.WriteUndertaleObject(Functions); - writer.WriteUndertaleObject(CodeLocals); + if (!writer.undertaleData.IsVersionAtLeast(2024, 8)) + writer.WriteUndertaleObject(CodeLocals); } } internal override void UnserializeChunk(UndertaleReader reader) { + if (!checkedFor2024_8) + CheckFor2024_8(reader); + if (Length == 0 && reader.undertaleData.GeneralInfo?.BytecodeVersion > 14) // YYC, 14 < bytecode <= 16, chunk is empty but exists { Functions = null; @@ -1408,12 +1438,18 @@ internal override void UnserializeChunk(UndertaleReader reader) else { Functions = reader.ReadUndertaleObject>(); - CodeLocals = reader.ReadUndertaleObject>(); + if (!reader.undertaleData.IsVersionAtLeast(2024, 8)) + CodeLocals = reader.ReadUndertaleObject>(); + else + CodeLocals = null; } } internal override uint UnserializeObjectCount(UndertaleReader reader) { + checkedFor2024_8 = false; + CheckFor2024_8(reader); + if (Length == 0 && reader.undertaleData.GeneralInfo?.BytecodeVersion > 14) return 0; @@ -1422,7 +1458,8 @@ internal override uint UnserializeObjectCount(UndertaleReader reader) if (!reader.Bytecode14OrLower) { count += 1 + UndertaleSimpleList.UnserializeChildObjectCount(reader); - count += 1 + UndertaleSimpleList.UnserializeChildObjectCount(reader); + if (!reader.undertaleData.IsVersionAtLeast(2024, 8)) + count += 1 + UndertaleSimpleList.UnserializeChildObjectCount(reader); } else count = Length / 12; From 2c152b47cb014fe5065d96e27d28ad76d32c578b Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Fri, 4 Oct 2024 22:26:52 +0800 Subject: [PATCH 02/11] Make CodeLocals null when it's not supported, and add checks --- UndertaleModCli/Program.UMTLibInherited.cs | 4 ++-- UndertaleModCli/Program.cs | 3 ++- UndertaleModLib/Compiler/AssemblyWriter.cs | 2 +- UndertaleModLib/Models/UndertaleGameObject.cs | 21 +++++++++++-------- UndertaleModLib/UndertaleChunks.cs | 1 + UndertaleModTests/GameLoadingTests.cs | 2 +- .../Editors/UndertaleCodeEditor.xaml.cs | 4 ++-- UndertaleModTool/ImportCodeSystem.cs | 2 +- UndertaleModTool/MainWindow.xaml.cs | 16 ++++++++------ UndertaleModTool/ProfileSystem.cs | 2 +- .../Builtin Scripts/HeCanBeEverywhere.csx | 2 +- .../Scripts/Builtin Scripts/SearchASM.csx | 2 +- .../Resource Repackers/GameObjectCopy.csx | 2 +- .../GameObjectCopyInternal.csx | 2 +- .../Resource Repackers/ImportASM_2_3.csx | 2 +- .../Resource Repackers/ImportGML_2_3.csx | 2 +- .../Scripts/Resource Unpackers/ExportASM.csx | 2 +- .../ConvertFrom17to16_for_2.3.csx | 2 ++ .../ExportAndConvert_2_3_ASM.csx | 8 +++---- .../RestoreMissingCodeLocals.csx | 5 +++-- .../Technical Scripts/TestExportAllCode.csx | 2 +- .../UndertaleResourceReferenceMethodsMap.cs | 2 +- 22 files changed, 51 insertions(+), 39 deletions(-) diff --git a/UndertaleModCli/Program.UMTLibInherited.cs b/UndertaleModCli/Program.UMTLibInherited.cs index 24c407bb1..8da953b49 100644 --- a/UndertaleModCli/Program.UMTLibInherited.cs +++ b/UndertaleModCli/Program.UMTLibInherited.cs @@ -449,7 +449,7 @@ public string GetDisassemblyText(UndertaleCode code) try { - return code != null ? code.Disassemble(Data.Variables, Data.CodeLocals.For(code)) : ""; + return code != null ? code.Disassemble(Data.Variables, Data.CodeLocals?.For(code)) : ""; } catch (Exception e) { @@ -752,7 +752,7 @@ void ImportCode(string codeName, string gmlCode, bool isGML = true, bool doParse else if (code.ParentEntry is not null) return; - if (Data?.GeneralInfo.BytecodeVersion > 14 && Data.CodeLocals.ByName(codeName) == null) + if (Data.CodeLocals is not null && Data.CodeLocals.ByName(codeName) is null) { UndertaleCodeLocals locals = new UndertaleCodeLocals(); locals.Name = code.Name; diff --git a/UndertaleModCli/Program.cs b/UndertaleModCli/Program.cs index 50dd39ec4..841b53b8e 100644 --- a/UndertaleModCli/Program.cs +++ b/UndertaleModCli/Program.cs @@ -585,7 +585,8 @@ private void CliQuickInfo() if (!Data.IsYYC()) { Console.WriteLine($"{Data.Code.Count} Code Entries, {Data.Variables.Count} Variables, {Data.Functions.Count} Functions"); - Console.WriteLine($"{Data.CodeLocals.Count} Code locals, {Data.Strings.Count} Strings, {Data.EmbeddedTextures.Count} Embedded Textures"); + var codeLocalsInfo = Data.CodeLocals is not null ? $"{Data.CodeLocals.Count} Code locals, " : ""; + Console.WriteLine($"{codeLocalsInfo}{Data.Strings.Count} Strings, {Data.EmbeddedTextures.Count} Embedded Textures"); } else { diff --git a/UndertaleModLib/Compiler/AssemblyWriter.cs b/UndertaleModLib/Compiler/AssemblyWriter.cs index 9667c9738..7d5ffad8d 100644 --- a/UndertaleModLib/Compiler/AssemblyWriter.cs +++ b/UndertaleModLib/Compiler/AssemblyWriter.cs @@ -107,7 +107,7 @@ public List Finish() bool defineArguments = true; if (compileContext.OriginalCode != null) { - UndertaleCodeLocals locals = compileContext.Data?.CodeLocals.For(compileContext.OriginalCode); + UndertaleCodeLocals locals = compileContext.Data?.CodeLocals?.For(compileContext.OriginalCode); if (locals != null) { // Update the code locals of the UndertaleCode diff --git a/UndertaleModLib/Models/UndertaleGameObject.cs b/UndertaleModLib/Models/UndertaleGameObject.cs index 8e21f3ed1..756a186e9 100644 --- a/UndertaleModLib/Models/UndertaleGameObject.cs +++ b/UndertaleModLib/Models/UndertaleGameObject.cs @@ -302,16 +302,19 @@ public UndertaleCode EventHandlerFor(EventType type, uint subtype, UndertaleData action.CodeId = code; data.Code.Add(code); - UndertaleCodeLocals.LocalVar argsLocal = new UndertaleCodeLocals.LocalVar(); - argsLocal.Name = data.Strings.MakeString("arguments"); - argsLocal.Index = 0; - - var locals = new UndertaleCodeLocals() + if (data.CodeLocals is not null) { - Name = name - }; - locals.Locals.Add(argsLocal); - data.CodeLocals.Add(locals); + UndertaleCodeLocals.LocalVar argsLocal = new UndertaleCodeLocals.LocalVar(); + argsLocal.Name = data.Strings.MakeString("arguments"); + argsLocal.Index = 0; + + var locals = new UndertaleCodeLocals() + { + Name = name + }; + locals.Locals.Add(argsLocal); + data.CodeLocals.Add(locals); + } } return code; } diff --git a/UndertaleModLib/UndertaleChunks.cs b/UndertaleModLib/UndertaleChunks.cs index 0fa26f315..99a7ceebd 100644 --- a/UndertaleModLib/UndertaleChunks.cs +++ b/UndertaleModLib/UndertaleChunks.cs @@ -1434,6 +1434,7 @@ internal override void UnserializeChunk(UndertaleReader reader) Functions.SetCapacity(Length / 12); while (reader.Position + 12 <= startPosition + Length) Functions.Add(reader.ReadUndertaleObject()); + CodeLocals = null; } else { diff --git a/UndertaleModTests/GameLoadingTests.cs b/UndertaleModTests/GameLoadingTests.cs index 766c7d072..9c4a159f2 100644 --- a/UndertaleModTests/GameLoadingTests.cs +++ b/UndertaleModTests/GameLoadingTests.cs @@ -75,7 +75,7 @@ public void DisassembleAndReassembleAllScripts() string disasm; try { - disasm = code.Disassemble(data.Variables, data.CodeLocals.For(code)); + disasm = code.Disassemble(data.Variables, data.CodeLocals?.For(code)); } catch (Exception e) { diff --git a/UndertaleModTool/Editors/UndertaleCodeEditor.xaml.cs b/UndertaleModTool/Editors/UndertaleCodeEditor.xaml.cs index adf0be972..5b834ee8e 100644 --- a/UndertaleModTool/Editors/UndertaleCodeEditor.xaml.cs +++ b/UndertaleModTool/Editors/UndertaleCodeEditor.xaml.cs @@ -529,7 +529,7 @@ private void DisassembleCode(UndertaleCode code, bool first) try { var data = mainWindow.Data; - text = code.Disassemble(data.Variables, data.CodeLocals.For(code)); + text = code.Disassemble(data.Variables, data.CodeLocals?.For(code)); CurrentLocals.Clear(); } @@ -843,7 +843,7 @@ private void PopulateCurrentLocals(UndertaleData data, UndertaleCode code) CurrentLocals.Clear(); // Look up locals for given code entry's name, for syntax highlighting - var locals = data.CodeLocals.ByName(code.Name.Content); + var locals = data.CodeLocals?.ByName(code.Name.Content); if (locals != null) { foreach (var local in locals.Locals) diff --git a/UndertaleModTool/ImportCodeSystem.cs b/UndertaleModTool/ImportCodeSystem.cs index 7e8a55a8b..d7965c38d 100644 --- a/UndertaleModTool/ImportCodeSystem.cs +++ b/UndertaleModTool/ImportCodeSystem.cs @@ -225,7 +225,7 @@ void ImportCode(string codeName, string gmlCode, bool IsGML = true, bool doParse else if (code.ParentEntry is not null) return; - if (Data?.GeneralInfo.BytecodeVersion > 14 && Data.CodeLocals.ByName(codeName) == null) + if (Data.CodeLocals is not null && Data.CodeLocals.ByName(codeName) is null) { UndertaleCodeLocals locals = new UndertaleCodeLocals(); locals.Name = code.Name; diff --git a/UndertaleModTool/MainWindow.xaml.cs b/UndertaleModTool/MainWindow.xaml.cs index 4c1e07a6d..b05ccd15c 100644 --- a/UndertaleModTool/MainWindow.xaml.cs +++ b/UndertaleModTool/MainWindow.xaml.cs @@ -1200,12 +1200,16 @@ private async Task SaveFile(string filename, bool suppressDebug = false) { debugData.SourceCode.Add(new UndertaleScriptSource() { SourceCode = debugData.Strings.MakeString(outputs[i]) }); debugData.DebugInfo.Add(outputsOffsets[i]); - debugData.LocalVars.Add(Data.CodeLocals[i]); - if (debugData.Strings.IndexOf(Data.CodeLocals[i].Name) < 0) - debugData.Strings.Add(Data.CodeLocals[i].Name); - foreach (var local in Data.CodeLocals[i].Locals) - if (debugData.Strings.IndexOf(local.Name) < 0) - debugData.Strings.Add(local.Name); + // FIXME: Probably should write something regardless. + if (Data.CodeLocals is not null) + { + debugData.LocalVars.Add(Data.CodeLocals[i]); + if (debugData.Strings.IndexOf(Data.CodeLocals[i].Name) < 0) + debugData.Strings.Add(Data.CodeLocals[i].Name); + foreach (var local in Data.CodeLocals[i].Locals) + if (debugData.Strings.IndexOf(local.Name) < 0) + debugData.Strings.Add(local.Name); + } } using (UndertaleWriter writer = new UndertaleWriter(new FileStream(Path.ChangeExtension(FilePath, ".yydebug"), FileMode.Create, FileAccess.Write))) diff --git a/UndertaleModTool/ProfileSystem.cs b/UndertaleModTool/ProfileSystem.cs index 60cdd261e..3bc7cca1c 100644 --- a/UndertaleModTool/ProfileSystem.cs +++ b/UndertaleModTool/ProfileSystem.cs @@ -47,7 +47,7 @@ public string GetDisassemblyText(UndertaleCode code) try { - return code != null ? code.Disassemble(Data.Variables, Data.CodeLocals.For(code)) : ""; + return code != null ? code.Disassemble(Data.Variables, Data.CodeLocals?.For(code)) : ""; } catch (Exception e) { diff --git a/UndertaleModTool/Scripts/Builtin Scripts/HeCanBeEverywhere.csx b/UndertaleModTool/Scripts/Builtin Scripts/HeCanBeEverywhere.csx index 35d3ed454..678e5e6d7 100644 --- a/UndertaleModTool/Scripts/Builtin Scripts/HeCanBeEverywhere.csx +++ b/UndertaleModTool/Scripts/Builtin Scripts/HeCanBeEverywhere.csx @@ -260,7 +260,7 @@ for (int i = 0; i < obj_shop1_Draw.Instructions.Count; i++) } } obj_shop1_Draw.UpdateAddresses(); -obj_shop1_Draw.Replace(Assembler.Assemble(obj_shop1_Draw.Disassemble(Data.Variables, Data.CodeLocals.For(obj_shop1_Draw)), Data)); // TODO: no idea why this is needed +obj_shop1_Draw.Replace(Assembler.Assemble(obj_shop1_Draw.Disassemble(Data.Variables, Data.CodeLocals?.For(obj_shop1_Draw)), Data)); // TODO: no idea why this is needed Data.GameObjects.ByName("obj_time").EventHandlerFor(EventType.KeyPress, EventSubtypeKey.vk_f6, Data).ReplaceGML(@" if (global.debug == 1) diff --git a/UndertaleModTool/Scripts/Builtin Scripts/SearchASM.csx b/UndertaleModTool/Scripts/Builtin Scripts/SearchASM.csx index dbe9c21c8..ced1a839a 100644 --- a/UndertaleModTool/Scripts/Builtin Scripts/SearchASM.csx +++ b/UndertaleModTool/Scripts/Builtin Scripts/SearchASM.csx @@ -81,7 +81,7 @@ void DumpCode(UndertaleCode code) try { var lineNumber = 1; - StringReader assemblyText = new(code != null ? code.Disassemble(Data.Variables, Data.CodeLocals.For(code)) : ""); + StringReader assemblyText = new(code != null ? code.Disassemble(Data.Variables, Data.CodeLocals?.For(code)) : ""); bool nameWritten = false; string lineInt; while ((lineInt = assemblyText.ReadLine()) is not null) diff --git a/UndertaleModTool/Scripts/Resource Repackers/GameObjectCopy.csx b/UndertaleModTool/Scripts/Resource Repackers/GameObjectCopy.csx index 58f1090a5..035a08c0a 100644 --- a/UndertaleModTool/Scripts/Resource Repackers/GameObjectCopy.csx +++ b/UndertaleModTool/Scripts/Resource Repackers/GameObjectCopy.csx @@ -161,7 +161,7 @@ for (var j = 0; j < splitStringsList.Count; j++) nativeACT.CodeId.ArgumentsCount = donorACT.CodeId.ArgumentsCount; nativeACT.CodeId.Offset = donorACT.CodeId.Offset; nativeACT.CodeId.WeirdLocalFlag = donorACT.CodeId.WeirdLocalFlag; - if (Data?.GeneralInfo.BytecodeVersion > 14) + if (Data.CodeLocals is not null) { UndertaleCodeLocals nativelocals = Data.CodeLocals.ByName(donorACT.CodeId?.Name?.Content); UndertaleCodeLocals donorlocals = DonorData.CodeLocals.ByName(donorACT.CodeId?.Name?.Content); diff --git a/UndertaleModTool/Scripts/Resource Repackers/GameObjectCopyInternal.csx b/UndertaleModTool/Scripts/Resource Repackers/GameObjectCopyInternal.csx index 8e7519509..c2fcfee1e 100644 --- a/UndertaleModTool/Scripts/Resource Repackers/GameObjectCopyInternal.csx +++ b/UndertaleModTool/Scripts/Resource Repackers/GameObjectCopyInternal.csx @@ -114,7 +114,7 @@ for (var j = 0; j < splitStringsList.Count; j++) nativeACT.CodeId.ArgumentsCount = donorACT.CodeId.ArgumentsCount; nativeACT.CodeId.Offset = donorACT.CodeId.Offset; nativeACT.CodeId.WeirdLocalFlag = donorACT.CodeId.WeirdLocalFlag; - if (Data?.GeneralInfo.BytecodeVersion > 14) + if (Data.CodeLocals is not null) { UndertaleCodeLocals donorlocals = Data.CodeLocals.ByName(donorACT.CodeId?.Name?.Content); UndertaleCodeLocals nativelocals = new UndertaleCodeLocals(); diff --git a/UndertaleModTool/Scripts/Resource Repackers/ImportASM_2_3.csx b/UndertaleModTool/Scripts/Resource Repackers/ImportASM_2_3.csx index 1a78c246c..0429b8490 100644 --- a/UndertaleModTool/Scripts/Resource Repackers/ImportASM_2_3.csx +++ b/UndertaleModTool/Scripts/Resource Repackers/ImportASM_2_3.csx @@ -117,7 +117,7 @@ await Task.Run(() => { code.Name = Data.Strings.MakeString(codeName); Data.Code.Add(code); } - if ((Data?.GeneralInfo.BytecodeVersion > 14) && (Data.CodeLocals.ByName(codeName) == null)) + if (Data.CodeLocals is not null && Data.CodeLocals.ByName(codeName) is null) { UndertaleCodeLocals locals = new UndertaleCodeLocals(); locals.Name = code.Name; diff --git a/UndertaleModTool/Scripts/Resource Repackers/ImportGML_2_3.csx b/UndertaleModTool/Scripts/Resource Repackers/ImportGML_2_3.csx index cdb4a1761..a7a2ba660 100644 --- a/UndertaleModTool/Scripts/Resource Repackers/ImportGML_2_3.csx +++ b/UndertaleModTool/Scripts/Resource Repackers/ImportGML_2_3.csx @@ -118,7 +118,7 @@ await Task.Run(() => { code.Name = Data.Strings.MakeString(codeName); Data.Code.Add(code); } - if ((Data?.GeneralInfo.BytecodeVersion > 14) && (Data.CodeLocals.ByName(codeName) == null)) + if (Data.CodeLocals is not null && Data.CodeLocals.ByName(codeName) is null) { UndertaleCodeLocals locals = new UndertaleCodeLocals(); locals.Name = code.Name; diff --git a/UndertaleModTool/Scripts/Resource Unpackers/ExportASM.csx b/UndertaleModTool/Scripts/Resource Unpackers/ExportASM.csx index dc07e2b88..d25e605bb 100644 --- a/UndertaleModTool/Scripts/Resource Unpackers/ExportASM.csx +++ b/UndertaleModTool/Scripts/Resource Unpackers/ExportASM.csx @@ -39,7 +39,7 @@ void DumpCode(UndertaleCode code) string path = Path.Combine(codeFolder, code.Name.Content + ".asm"); try { - File.WriteAllText(path, (code != null ? code.Disassemble(Data.Variables, Data.CodeLocals.For(code)) : "")); + File.WriteAllText(path, (code != null ? code.Disassemble(Data.Variables, Data.CodeLocals?.For(code)) : "")); } catch (Exception e) { diff --git a/UndertaleModTool/Scripts/Technical Scripts/ConvertFrom17to16_for_2.3.csx b/UndertaleModTool/Scripts/Technical Scripts/ConvertFrom17to16_for_2.3.csx index 792917279..42420b99f 100644 --- a/UndertaleModTool/Scripts/Technical Scripts/ConvertFrom17to16_for_2.3.csx +++ b/UndertaleModTool/Scripts/Technical Scripts/ConvertFrom17to16_for_2.3.csx @@ -77,6 +77,8 @@ applying any changes to the game."; Data.FORM.Chunks.Remove("TAGS"); if (Data.FORM.Chunks.ContainsKey("EMBI")) Data.FORM.Chunks.Remove("EMBI"); + if (Data.FORM.FUNC.CodeLocals is null) + Data.FORM.FUNC.CodeLocals = new UndertaleSimpleList(); Data.SetGMS2Version(2); //Data.IsTPAG4ByteAligned = false; for (int i = 0; i < Data.Code.Count; i++) diff --git a/UndertaleModTool/Scripts/Technical Scripts/ExportAndConvert_2_3_ASM.csx b/UndertaleModTool/Scripts/Technical Scripts/ExportAndConvert_2_3_ASM.csx index 16078c9dd..0009b0503 100644 --- a/UndertaleModTool/Scripts/Technical Scripts/ExportAndConvert_2_3_ASM.csx +++ b/UndertaleModTool/Scripts/Technical Scripts/ExportAndConvert_2_3_ASM.csx @@ -52,7 +52,7 @@ void DumpCode() foreach (UndertaleCode code_orig in Data.Code) { code_orig.Offset = 0; - if (Data.CodeLocals.ByName(code_orig.Name.Content) == null) + if (Data.CodeLocals is not null && Data.CodeLocals.ByName(code_orig.Name.Content) == null) { UndertaleCodeLocals locals = new UndertaleCodeLocals(); locals.Name = code_orig.Name; @@ -90,7 +90,7 @@ void DumpCode() string x = ""; try { - string disasm_code = code_orig.Disassemble(Data.Variables, Data.CodeLocals.For(code_orig)); + string disasm_code = code_orig.Disassemble(Data.Variables, Data.CodeLocals?.For(code_orig)); //ScriptMessage(code_orig.Name.Content); //ScriptMessage("1 " + disasm_code); int ix = -1; @@ -120,7 +120,7 @@ void DumpCode() string str_path_to_use = Path.Combine(codeFolder, code_orig.Name.Content + ".asm"); string code_output = ""; if (code_orig != null) - code_output = code_orig.Disassemble(Data.Variables, Data.CodeLocals.For(code_orig)); + code_output = code_orig.Disassemble(Data.Variables, Data.CodeLocals?.For(code_orig)); File.WriteAllText(str_path_to_use, code_output); } catch (Exception e) @@ -144,7 +144,7 @@ void DumpCode() string str_path_to_use = Path.Combine(codeFolder, "Duplicates", code_orig.Name.Content + ".asm"); string code_output = ""; if (code_orig != null) - code_output = code_orig.Disassemble(Data.Variables, Data.CodeLocals.For(code_orig)); + code_output = code_orig.Disassemble(Data.Variables, Data.CodeLocals?.For(code_orig)); File.WriteAllText(str_path_to_use, code_output); } catch (Exception e) diff --git a/UndertaleModTool/Scripts/Technical Scripts/RestoreMissingCodeLocals.csx b/UndertaleModTool/Scripts/Technical Scripts/RestoreMissingCodeLocals.csx index 1dbb5e1bf..1f00a6672 100644 --- a/UndertaleModTool/Scripts/Technical Scripts/RestoreMissingCodeLocals.csx +++ b/UndertaleModTool/Scripts/Technical Scripts/RestoreMissingCodeLocals.csx @@ -1,6 +1,7 @@ -if (Data?.GeneralInfo.BytecodeVersion < 15) +EnsureDataLoaded(); +if (Data.CodeLocals is null) { - ScriptMessage("Cannot run on this game, bytecode >= 15 required!"); + ScriptMessage("Cannot run on this game, bytecode >= 15 and GM <= 2024.8 required!"); return; } int newCount = 0; diff --git a/UndertaleModTool/Scripts/Technical Scripts/TestExportAllCode.csx b/UndertaleModTool/Scripts/Technical Scripts/TestExportAllCode.csx index ed70e8522..a4e5dbd0f 100644 --- a/UndertaleModTool/Scripts/Technical Scripts/TestExportAllCode.csx +++ b/UndertaleModTool/Scripts/Technical Scripts/TestExportAllCode.csx @@ -170,7 +170,7 @@ if (File.Exists(path_error2)) string asmPath = Path.Combine(asmFolder, code.Name.Content + ".asm"); try { - File.WriteAllText(asmPath, (code != null ? code.Disassemble(Data.Variables, Data.CodeLocals.For(code)) : "")); + File.WriteAllText(asmPath, (code != null ? code.Disassemble(Data.Variables, Data.CodeLocals?.For(code)) : "")); } catch (Exception e) { diff --git a/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs b/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs index 0a2e959e1..fe5fb491f 100644 --- a/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs +++ b/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs @@ -615,7 +615,7 @@ IEnumerable GetExtnFunctions() if (objSrc is not UndertaleString obj) return null; - var codeLocals = data.CodeLocals.Where(x => x.Name == obj || x.Locals.Any(l => l.Name == obj)); + var codeLocals = data.CodeLocals?.Where(x => x.Name == obj || x.Locals.Any(l => l.Name == obj)); if (codeLocals.Any()) return new() { { "Code locals", checkOne ? codeLocals.ToEmptyArray() : codeLocals.ToArray() } }; else From 034735e291eb268e81d762d5959d55eaa18782eb Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Fri, 4 Oct 2024 22:39:10 +0800 Subject: [PATCH 03/11] Hide code locals searching in Find all references for 2024.8+ --- .../UndertaleResourceReferenceMap.cs | 10 +++++++++- .../UndertaleResourceReferenceMethodsMap.cs | 10 +++++++++- 2 files changed, 18 insertions(+), 2 deletions(-) diff --git a/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMap.cs b/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMap.cs index 7f4a1ddd3..f6190144b 100644 --- a/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMap.cs +++ b/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMap.cs @@ -10,6 +10,7 @@ namespace UndertaleModTool.Windows public class TypesForVersion { public (uint Major, uint Minor, uint Release) Version { get; set; } + public (uint Major, uint Minor, uint Release) BeforeVersion { get; set; } = (uint.MaxValue, uint.MaxValue, uint.MaxValue); public (Type, string)[] Types { get; set; } } @@ -172,6 +173,7 @@ public static class UndertaleResourceReferenceMap { // Bytecode version 15 Version = (15, uint.MaxValue, uint.MaxValue), + BeforeVersion = (2024, 8, 0), Types = new[] { (typeof(UndertaleCodeLocals), "Code locals") @@ -439,7 +441,13 @@ public static (Type, string)[] GetTypeMapForVersion(Type type, UndertaleData dat else isAtLeast = typeForVer.Version.CompareTo(version) <= 0; - if (isAtLeast) + bool isAboveMost = false; + if (typeForVer.BeforeVersion.Minor == uint.MaxValue) + isAboveMost = typeForVer.BeforeVersion.Major <= bytecodeVersion; + else + isAboveMost = typeForVer.BeforeVersion.CompareTo(version) <= 0; + + if (isAtLeast && !isAboveMost) outTypes = typeForVer.Types.UnionBy(outTypes, x => x.Item1); } diff --git a/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs b/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs index fe5fb491f..e0de674a1 100644 --- a/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs +++ b/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs @@ -32,6 +32,7 @@ public HashSetTypesOverride(bool containsEverything = false, bool isYYC = false) public class PredicateForVersion { public (uint Major, uint Minor, uint Release) Version { get; set; } + public (uint Major, uint Minor, uint Release) BeforeVersion { get; set; } = (uint.MaxValue, uint.MaxValue, uint.MaxValue); public Func> Predicate { get; set; } } @@ -607,6 +608,7 @@ IEnumerable GetExtnFunctions() { // Bytecode version 15 Version = (15, uint.MaxValue, uint.MaxValue), + BeforeVersion = (2024, 8, 0), Predicate = (objSrc, types, checkOne) => { if (!types.Contains(typeof(UndertaleCodeLocals))) @@ -1434,7 +1436,13 @@ public static Dictionary> GetReferencesOfObject(object obj, else isAtLeast = predicateForVer.Version.CompareTo(ver) <= 0; - if (isAtLeast) + bool isAboveMost = false; + if (predicateForVer.BeforeVersion.Minor == uint.MaxValue) + isAboveMost = predicateForVer.BeforeVersion.Major <= data.GeneralInfo.BytecodeVersion; + else + isAboveMost = predicateForVer.BeforeVersion.CompareTo(ver) <= 0; + + if (isAtLeast && !isAboveMost) { var result = predicateForVer.Predicate(obj, types, checkOne); if (result is null) From af3ad262658c5cf51873da70ebec69ab9228ef37 Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Fri, 4 Oct 2024 22:49:01 +0800 Subject: [PATCH 04/11] Remove pointless 2024.8 check in FUNC Serialize --- UndertaleModLib/UndertaleChunks.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/UndertaleModLib/UndertaleChunks.cs b/UndertaleModLib/UndertaleChunks.cs index 99a7ceebd..4b49062df 100644 --- a/UndertaleModLib/UndertaleChunks.cs +++ b/UndertaleModLib/UndertaleChunks.cs @@ -1395,7 +1395,7 @@ private void CheckFor2024_8(UndertaleReader reader) internal override void SerializeChunk(UndertaleWriter writer) { - if (Functions is null && (writer.undertaleData.IsVersionAtLeast(2024, 8) || CodeLocals is null)) + if (Functions is null && CodeLocals is null) return; UndertaleInstruction.Reference.SerializeReferenceChain(writer, writer.undertaleData.Code, Functions); From bd77325f11165fcc17d89dcae1d4f0041075b2dc Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Sat, 5 Oct 2024 00:08:42 +0800 Subject: [PATCH 05/11] Fix adding new code entries --- UndertaleModTool/MainWindow.xaml.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/UndertaleModTool/MainWindow.xaml.cs b/UndertaleModTool/MainWindow.xaml.cs index b05ccd15c..7c85aa001 100644 --- a/UndertaleModTool/MainWindow.xaml.cs +++ b/UndertaleModTool/MainWindow.xaml.cs @@ -2141,7 +2141,7 @@ private void MenuItem_Add_Click(object sender, RoutedEventArgs e) } (obj as UndertaleScript).Code = code; } - if ((obj is UndertaleCode) && (Data.GeneralInfo.BytecodeVersion > 14)) + if (obj is UndertaleCode && Data.CodeLocals is not null) { UndertaleCodeLocals locals = new UndertaleCodeLocals(); locals.Name = (obj as UndertaleCode).Name; From add0393088fc557d15c0599f8b8496c9b153facb Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Sat, 5 Oct 2024 00:09:21 +0800 Subject: [PATCH 06/11] Addendum (what) --- UndertaleModTool/MainWindow.xaml.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/UndertaleModTool/MainWindow.xaml.cs b/UndertaleModTool/MainWindow.xaml.cs index 7c85aa001..0c4304ec8 100644 --- a/UndertaleModTool/MainWindow.xaml.cs +++ b/UndertaleModTool/MainWindow.xaml.cs @@ -2128,7 +2128,7 @@ private void MenuItem_Add_Click(object sender, RoutedEventArgs e) string prefix = Data.IsVersionAtLeast(2, 3) ? "gml_GlobalScript_" : "gml_Script_"; code.Name = Data.Strings.MakeString(prefix + newName); Data.Code.Add(code); - if (Data?.GeneralInfo.BytecodeVersion > 14) + if (Data.CodeLocals is not null) { UndertaleCodeLocals locals = new UndertaleCodeLocals(); locals.Name = code.Name; From 0454f070342f82cefc17f5d3ee0fb8032fb0fe4f Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Sat, 5 Oct 2024 00:37:28 +0800 Subject: [PATCH 07/11] Fix cases of code compilation changing GMS2 version to 2023.8 --- UndertaleModLib/UndertaleChunks.cs | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/UndertaleModLib/UndertaleChunks.cs b/UndertaleModLib/UndertaleChunks.cs index 4b49062df..cb2a4c642 100644 --- a/UndertaleModLib/UndertaleChunks.cs +++ b/UndertaleModLib/UndertaleChunks.cs @@ -2237,7 +2237,8 @@ private void CheckPsemVersion(UndertaleReader reader) // Fortunately, consistent padding means we need no parsing here if (Length == 0xF8) { - reader.undertaleData.SetGMS2Version(2023, 8); + if (!reader.undertaleData.IsVersionAtLeast(2023, 8)) + reader.undertaleData.SetGMS2Version(2023, 8); } else if (Length == 0xD8) { @@ -2267,7 +2268,8 @@ private void CheckPsemVersion(UndertaleReader reader) uint secondPtr = reader.ReadUInt32(); if (secondPtr - firstPtr == 0xEC) { - reader.undertaleData.SetGMS2Version(2023, 8); + if (!reader.undertaleData.IsVersionAtLeast(2023, 8)) + reader.undertaleData.SetGMS2Version(2023, 8); } else if (secondPtr - firstPtr == 0xC0) { From a950d149715539f0d8c5f1c10feba1017ec7a978 Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Sat, 5 Oct 2024 01:28:04 +0800 Subject: [PATCH 08/11] Make the 2024.8 detection code look worse --- UndertaleModLib/UndertaleChunks.cs | 35 +++++++++++++++++++++++++----- 1 file changed, 29 insertions(+), 6 deletions(-) diff --git a/UndertaleModLib/UndertaleChunks.cs b/UndertaleModLib/UndertaleChunks.cs index cb2a4c642..af53b818e 100644 --- a/UndertaleModLib/UndertaleChunks.cs +++ b/UndertaleModLib/UndertaleChunks.cs @@ -1380,13 +1380,36 @@ private void CheckFor2024_8(UndertaleReader reader) long returnPos = reader.Position; - // The CodeLocals list was removed in 2024.8, so we check for its absence here - + // The CodeLocals list was removed in 2024.8, so we check if Functions + // is the only thing in here. uint funcCount = reader.ReadUInt32(); - // If this is 2024.8, the chunk should only contain the Functions list. - // Considering Functions is a SimpleList, this would make the chunk size - // sizeof funcCount + sizeof UndertaleFunction * funcCount - if (Length == 4 + 12 * funcCount) + // Skip over the (Simple)List + // (3*4 is the size of an UndertaleFunction object) + reader.Position += 3 * 4 * funcCount; + if (reader.Position == returnPos + Length) + { + // Whatever, let's consider this a win + reader.undertaleData.SetGMS2Version(2024, 8); + reader.Position = returnPos; + checkedFor2024_8 = true; + return; + } + + // Then align the position + int specAlign = reader.undertaleData.PaddingAlignException; + while ((reader.AbsPosition & ((specAlign == -1 ? 16 : specAlign) - 1)) != 0) + { + if (reader.ReadByte() != 0) + { + // If we hit a non-zero byte, it can't be padding + reader.Position = returnPos; + checkedFor2024_8 = true; + return; + } + } + + // Then check if we're at the end of the chunk + if (reader.Position == returnPos + Length) reader.undertaleData.SetGMS2Version(2024, 8); reader.Position = returnPos; From 027d01b4e9d6e1d2a53526bed056ce1bfcbee971 Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Sun, 13 Oct 2024 12:43:25 +0800 Subject: [PATCH 09/11] Hack around failing to find local vars --- UndertaleModLib/Decompiler/Assembler.cs | 4 +++- UndertaleModLib/UndertaleDataExtensionMethods.cs | 8 +++++--- 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/UndertaleModLib/Decompiler/Assembler.cs b/UndertaleModLib/Decompiler/Assembler.cs index 2d811dfb3..0019f9683 100644 --- a/UndertaleModLib/Decompiler/Assembler.cs +++ b/UndertaleModLib/Decompiler/Assembler.cs @@ -341,6 +341,8 @@ public static List Assemble(string source, IList= 4) { var varii = vars[Int32.Parse(aaa[3])]; @@ -526,7 +528,7 @@ private static UndertaleInstruction.Reference ParseVariableRe // Locate variable from either local variables, or VARI chunk UndertaleVariable locatedVariable; string variableName = str[strPosition..].ToString(); - if (variInstanceType == UndertaleInstruction.InstanceType.Local) + if (variInstanceType == UndertaleInstruction.InstanceType.Local && data?.CodeLocals is not null) { locatedVariable = localvars.ContainsKey(variableName) ? localvars[variableName] : null; } diff --git a/UndertaleModLib/UndertaleDataExtensionMethods.cs b/UndertaleModLib/UndertaleDataExtensionMethods.cs index 0e0bb9a1e..7667d7add 100644 --- a/UndertaleModLib/UndertaleDataExtensionMethods.cs +++ b/UndertaleModLib/UndertaleDataExtensionMethods.cs @@ -174,10 +174,12 @@ public static UndertaleVariable EnsureDefined(this IList list public static UndertaleVariable DefineLocal(this IList list, IList originalReferencedLocalVars, int localId, string name, IList strg, UndertaleData data) { - bool bytecode14 = (data?.GeneralInfo?.BytecodeVersion <= 14); - if (bytecode14) + bool bytecode14 = data?.GeneralInfo?.BytecodeVersion <= 14; + if (bytecode14 || data?.CodeLocals is null) { - UndertaleVariable search = list.Where((x) => x.Name.Content == name).FirstOrDefault(); + UndertaleVariable search = list.Where((x) => + x.Name.Content == name && (bytecode14 || x.InstanceType == UndertaleInstruction.InstanceType.Local) + ).FirstOrDefault(); if (search != null) return search; } From 0ba0f09f6ec742e22d1523ff177d36606127dd12 Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Sun, 13 Oct 2024 12:47:29 +0800 Subject: [PATCH 10/11] [skip ci] Ok that doesn't make much sense --- UndertaleModLib/Decompiler/Assembler.cs | 2 -- 1 file changed, 2 deletions(-) diff --git a/UndertaleModLib/Decompiler/Assembler.cs b/UndertaleModLib/Decompiler/Assembler.cs index 0019f9683..5f38a3ed0 100644 --- a/UndertaleModLib/Decompiler/Assembler.cs +++ b/UndertaleModLib/Decompiler/Assembler.cs @@ -341,8 +341,6 @@ public static List Assemble(string source, IList= 4) { var varii = vars[Int32.Parse(aaa[3])]; From a291e8f74cce599f0e26b0cd9c965bbda2727c4a Mon Sep 17 00:00:00 2001 From: Liu Wenyuan <15816141883@163.com> Date: Sun, 13 Oct 2024 20:53:55 +0800 Subject: [PATCH 11/11] Don't let CodeLocals be nullable in reference finding code Co-authored-by: zivmaor <63353113+zivmaor@users.noreply.github.com> --- .../UndertaleResourceReferenceMethodsMap.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs b/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs index e0de674a1..269fbb0d0 100644 --- a/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs +++ b/UndertaleModTool/Windows/FindReferencesTypesDialog/UndertaleResourceReferenceMethodsMap.cs @@ -617,7 +617,7 @@ IEnumerable GetExtnFunctions() if (objSrc is not UndertaleString obj) return null; - var codeLocals = data.CodeLocals?.Where(x => x.Name == obj || x.Locals.Any(l => l.Name == obj)); + var codeLocals = data.CodeLocals.Where(x => x.Name == obj || x.Locals.Any(l => l.Name == obj)); if (codeLocals.Any()) return new() { { "Code locals", checkOne ? codeLocals.ToEmptyArray() : codeLocals.ToArray() } }; else