Update TamperMachine and disable write-to-code prevention (#2506)
* Enable write to memory and improve logging * Update tamper machine opcodes and improve reporting * Add Else support * Add missing private statement
This commit is contained in:
		
							parent
							
								
									a27986c311
								
							
						
					
					
						commit
						ff8849671a
					
				
					 15 changed files with 156 additions and 66 deletions
				
			
		|  | @ -10,32 +10,73 @@ namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters | |||
|     /// </summary> | ||||
|     class EndConditionalBlock | ||||
|     { | ||||
|         const int TerminationTypeIndex = 1; | ||||
| 
 | ||||
|         private const byte End  = 0; // True end of the conditional. | ||||
|         private const byte Else = 1; // End of the 'then' block and beginning of 'else' block. | ||||
| 
 | ||||
|         public static void Emit(byte[] instruction, CompilationContext context) | ||||
|         { | ||||
|             // 20000000 | ||||
|             Emit(instruction, context, null); | ||||
|         } | ||||
| 
 | ||||
|         private static void Emit(byte[] instruction, CompilationContext context, IEnumerable<IOperation> operationsElse) | ||||
|         { | ||||
|             // 2X000000 | ||||
|             // X: End type (0 = End, 1 = Else). | ||||
| 
 | ||||
|             byte terminationType = instruction[TerminationTypeIndex]; | ||||
| 
 | ||||
|             switch (terminationType) | ||||
|             { | ||||
|                 case End: | ||||
|                     break; | ||||
|                 case Else: | ||||
|                     // Start a new operation block with the 'else' instruction to signal that there is the 'then' block just above it. | ||||
|                     context.BlockStack.Push(new OperationBlock(instruction)); | ||||
|                     return; | ||||
|                 default: | ||||
|                     throw new TamperCompilationException($"Unknown conditional termination type {terminationType}"); | ||||
|             } | ||||
| 
 | ||||
|             // Use the conditional begin instruction stored in the stack. | ||||
|             instruction = context.CurrentBlock.BaseInstruction; | ||||
|             CodeType codeType = InstructionHelper.GetCodeType(instruction); | ||||
|             var upperInstruction = context.CurrentBlock.BaseInstruction; | ||||
|             CodeType codeType = InstructionHelper.GetCodeType(upperInstruction); | ||||
| 
 | ||||
|             // Pop the current block of operations from the stack so control instructions | ||||
|             // for the conditional can be emitted in the upper block. | ||||
|             IEnumerable<IOperation> operations = context.CurrentOperations; | ||||
|             context.BlockStack.Pop(); | ||||
| 
 | ||||
|             // If the else operations are already set, then the upper block must not be another end. | ||||
|             if (operationsElse != null && codeType == CodeType.EndConditionalBlock) | ||||
|             { | ||||
|                 throw new TamperCompilationException($"Expected an upper 'if' conditional instead of 'end conditional'"); | ||||
|             } | ||||
| 
 | ||||
|             ICondition condition; | ||||
| 
 | ||||
|             switch (codeType) | ||||
|             { | ||||
|                 case CodeType.BeginMemoryConditionalBlock: | ||||
|                     condition = MemoryConditional.Emit(instruction, context); | ||||
|                     condition = MemoryConditional.Emit(upperInstruction, context); | ||||
|                     break; | ||||
|                 case CodeType.BeginKeypressConditionalBlock: | ||||
|                     condition = KeyPressConditional.Emit(instruction, context); | ||||
|                     condition = KeyPressConditional.Emit(upperInstruction, context); | ||||
|                     break; | ||||
|                 case CodeType.BeginRegisterConditionalBlock: | ||||
|                     condition = RegisterConditional.Emit(instruction, context); | ||||
|                     condition = RegisterConditional.Emit(upperInstruction, context); | ||||
|                     break; | ||||
|                 case CodeType.EndConditionalBlock: | ||||
|                     terminationType = upperInstruction[TerminationTypeIndex]; | ||||
|                     // If there is an end instruction above then it must be an else. | ||||
|                     if (terminationType != Else) | ||||
|                     { | ||||
|                         throw new TamperCompilationException($"Expected an upper 'else' conditional instead of {terminationType}"); | ||||
|                     } | ||||
|                     // Re-run the Emit with the else operations set. | ||||
|                     Emit(instruction, context, operations); | ||||
|                     return; | ||||
|                 default: | ||||
|                     throw new TamperCompilationException($"Conditional end does not match code type {codeType} in Atmosphere cheat"); | ||||
|             } | ||||
|  | @ -43,7 +84,7 @@ namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters | |||
|             // Create a conditional block with the current operations and nest it in the upper | ||||
|             // block of the stack. | ||||
| 
 | ||||
|             IfBlock block = new IfBlock(condition, operations); | ||||
|             IfBlock block = new IfBlock(condition, operations, operationsElse); | ||||
|             context.CurrentOperations.Add(block); | ||||
|         } | ||||
|     } | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Caian Benedicto
						Caian Benedicto