| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091 |
- 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);
- }
- }
- }
|