| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106 |
- using ChocolArm64.Decoder;
- using ChocolArm64.Instruction;
- using ChocolArm64.Translation;
- using System.Collections.Generic;
- using System.Reflection.Emit;
- namespace ChocolArm64
- {
- class ATranslator
- {
- public AThread Thread { get; private set; }
- private Dictionary<long, ATranslatedSub> CachedSubs;
- private bool KeepRunning;
- public ATranslator(AThread Parent)
- {
- this.Thread = Parent;
- CachedSubs = new Dictionary<long, ATranslatedSub>();
- KeepRunning = true;
- }
- public void StopExecution() => KeepRunning = false;
- public void ExecuteSubroutine(long Position)
- {
- do
- {
- if (CachedSubs.TryGetValue(Position, out ATranslatedSub Sub) && !Sub.NeedsReJit)
- {
- Position = Sub.Execute(Thread.ThreadState, Thread.Memory);
- }
- else
- {
- Position = TranslateSubroutine(Position).Execute(Thread.ThreadState, Thread.Memory);
- }
- }
- while (Position != 0 && KeepRunning);
- }
- public bool TryGetCachedSub(AOpCode OpCode, out ATranslatedSub Sub)
- {
- if (OpCode.Emitter != AInstEmit.Bl)
- {
- Sub = null;
- return false;
- }
- return TryGetCachedSub(((AOpCodeBImmAl)OpCode).Imm, out Sub);
- }
- public bool TryGetCachedSub(long Position, out ATranslatedSub Sub)
- {
- return CachedSubs.TryGetValue(Position, out Sub);
- }
- public bool HasCachedSub(long Position)
- {
- return CachedSubs.ContainsKey(Position);
- }
- private ATranslatedSub TranslateSubroutine(long Position)
- {
- (ABlock[] Graph, ABlock Root) Cfg = ADecoder.DecodeSubroutine(this, Position);
- AILEmitterCtx Context = new AILEmitterCtx(
- this,
- Cfg.Graph,
- Cfg.Root);
- if (Context.CurrBlock.Position != Position)
- {
- Context.Emit(OpCodes.Br, Context.GetLabel(Position));
- }
- do
- {
- Context.EmitOpCode();
- }
- while (Context.AdvanceOpCode());
- //Mark all methods that calls this method for ReJiting,
- //since we can now call it directly which is faster.
- foreach (ATranslatedSub TS in CachedSubs.Values)
- {
- if (TS.SubCalls.Contains(Position))
- {
- TS.MarkForReJit();
- }
- }
- ATranslatedSub Subroutine = Context.GetSubroutine();
- if (!CachedSubs.TryAdd(Position, Subroutine))
- {
- CachedSubs[Position] = Subroutine;
- }
- return Subroutine;
- }
- }
- }
|