 ff8849671a
			
		
	
	
		ff8849671a
		
			
		
	
	
	
	
		
			
			* Enable write to memory and improve logging * Update tamper machine opcodes and improve reporting * Add Else support * Add missing private statement
		
			
				
	
	
		
			91 lines
		
	
	
	
		
			3.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			91 lines
		
	
	
	
		
			3.9 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Ryujinx.HLE.Exceptions;
 | |
| using Ryujinx.HLE.HOS.Tamper.Conditions;
 | |
| using Ryujinx.HLE.HOS.Tamper.Operations;
 | |
| using System.Collections.Generic;
 | |
| 
 | |
| namespace Ryujinx.HLE.HOS.Tamper.CodeEmitters
 | |
| {
 | |
|     /// <summary>
 | |
|     /// Code type 2 marks the end of a conditional block (started by Code Type 1, Code Type 8 or Code Type C0).
 | |
|     /// </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)
 | |
|         {
 | |
|             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.
 | |
|             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(upperInstruction, context);
 | |
|                     break;
 | |
|                 case CodeType.BeginKeypressConditionalBlock:
 | |
|                     condition = KeyPressConditional.Emit(upperInstruction, context);
 | |
|                     break;
 | |
|                 case CodeType.BeginRegisterConditionalBlock:
 | |
|                     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");
 | |
|             }
 | |
| 
 | |
|             // Create a conditional block with the current operations and nest it in the upper
 | |
|             // block of the stack.
 | |
| 
 | |
|             IfBlock block = new IfBlock(condition, operations, operationsElse);
 | |
|             context.CurrentOperations.Add(block);
 | |
|         }
 | |
|     }
 | |
| }
 |