ArmProcessContextFactory.cs 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. using Ryujinx.Common.Configuration;
  2. using Ryujinx.Common.Logging;
  3. using Ryujinx.Cpu;
  4. using Ryujinx.Cpu.AppleHv;
  5. using Ryujinx.Cpu.Jit;
  6. using Ryujinx.Cpu.LightningJit;
  7. using Ryujinx.Graphics.Gpu;
  8. using Ryujinx.HLE.HOS.Kernel;
  9. using Ryujinx.HLE.HOS.Kernel.Process;
  10. using Ryujinx.Memory;
  11. using System;
  12. using System.Runtime.InteropServices;
  13. namespace Ryujinx.HLE.HOS
  14. {
  15. class ArmProcessContextFactory : IProcessContextFactory
  16. {
  17. private readonly ITickSource _tickSource;
  18. private readonly GpuContext _gpu;
  19. private readonly string _titleIdText;
  20. private readonly string _displayVersion;
  21. private readonly bool _diskCacheEnabled;
  22. private readonly ulong _codeAddress;
  23. private readonly ulong _codeSize;
  24. public IDiskCacheLoadState DiskCacheLoadState { get; private set; }
  25. public ArmProcessContextFactory(
  26. ITickSource tickSource,
  27. GpuContext gpu,
  28. string titleIdText,
  29. string displayVersion,
  30. bool diskCacheEnabled,
  31. ulong codeAddress,
  32. ulong codeSize)
  33. {
  34. _tickSource = tickSource;
  35. _gpu = gpu;
  36. _titleIdText = titleIdText;
  37. _displayVersion = displayVersion;
  38. _diskCacheEnabled = diskCacheEnabled;
  39. _codeAddress = codeAddress;
  40. _codeSize = codeSize;
  41. }
  42. public IProcessContext Create(KernelContext context, ulong pid, ulong addressSpaceSize, InvalidAccessHandler invalidAccessHandler, bool for64Bit)
  43. {
  44. IArmProcessContext processContext;
  45. bool isArm64Host = RuntimeInformation.ProcessArchitecture == Architecture.Arm64;
  46. if (OperatingSystem.IsMacOS() && isArm64Host && for64Bit && context.Device.Configuration.UseHypervisor)
  47. {
  48. var cpuEngine = new HvEngine(_tickSource);
  49. var memoryManager = new HvMemoryManager(context.Memory, addressSpaceSize, invalidAccessHandler);
  50. processContext = new ArmProcessContext<HvMemoryManager>(pid, cpuEngine, _gpu, memoryManager, addressSpaceSize, for64Bit);
  51. }
  52. else
  53. {
  54. MemoryManagerMode mode = context.Device.Configuration.MemoryManagerMode;
  55. if (!MemoryBlock.SupportsFlags(MemoryAllocationFlags.ViewCompatible))
  56. {
  57. Logger.Warning?.Print(LogClass.Cpu, "Host system doesn't support views, falling back to software page table");
  58. mode = MemoryManagerMode.SoftwarePageTable;
  59. }
  60. ICpuEngine cpuEngine = isArm64Host && (mode == MemoryManagerMode.HostMapped || mode == MemoryManagerMode.HostMappedUnsafe)
  61. ? new LightningJitEngine(_tickSource)
  62. : new JitEngine(_tickSource);
  63. AddressSpace addressSpace = null;
  64. // We want to use host tracked mode if the host page size is > 4KB.
  65. if ((mode == MemoryManagerMode.HostMapped || mode == MemoryManagerMode.HostMappedUnsafe) && MemoryBlock.GetPageSize() <= 0x1000)
  66. {
  67. if (!AddressSpace.TryCreate(context.Memory, addressSpaceSize, out addressSpace))
  68. {
  69. Logger.Warning?.Print(LogClass.Cpu, "Address space creation failed, falling back to software page table");
  70. mode = MemoryManagerMode.SoftwarePageTable;
  71. }
  72. }
  73. switch (mode)
  74. {
  75. case MemoryManagerMode.SoftwarePageTable:
  76. var memoryManager = new MemoryManager(context.Memory, addressSpaceSize, invalidAccessHandler);
  77. processContext = new ArmProcessContext<MemoryManager>(pid, cpuEngine, _gpu, memoryManager, addressSpaceSize, for64Bit);
  78. break;
  79. case MemoryManagerMode.HostMapped:
  80. case MemoryManagerMode.HostMappedUnsafe:
  81. if (addressSpace == null)
  82. {
  83. var memoryManagerHostTracked = new MemoryManagerHostTracked(context.Memory, addressSpaceSize, mode == MemoryManagerMode.HostMappedUnsafe, invalidAccessHandler);
  84. processContext = new ArmProcessContext<MemoryManagerHostTracked>(pid, cpuEngine, _gpu, memoryManagerHostTracked, addressSpaceSize, for64Bit);
  85. }
  86. else
  87. {
  88. if (addressSpaceSize != addressSpace.AddressSpaceSize)
  89. {
  90. Logger.Warning?.Print(LogClass.Emulation, $"Allocated address space (0x{addressSpace.AddressSpaceSize:X}) is smaller than guest application requirements (0x{addressSpaceSize:X})");
  91. }
  92. var memoryManagerHostMapped = new MemoryManagerHostMapped(addressSpace, mode == MemoryManagerMode.HostMappedUnsafe, invalidAccessHandler);
  93. processContext = new ArmProcessContext<MemoryManagerHostMapped>(pid, cpuEngine, _gpu, memoryManagerHostMapped, addressSpace.AddressSpaceSize, for64Bit);
  94. }
  95. break;
  96. default:
  97. throw new InvalidOperationException($"{nameof(mode)} contains an invalid value: {mode}");
  98. }
  99. }
  100. DiskCacheLoadState = processContext.Initialize(_titleIdText, _displayVersion, _diskCacheEnabled, _codeAddress, _codeSize);
  101. return processContext;
  102. }
  103. }
  104. }