SharedMemoryStorage.cs 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103
  1. using Ryujinx.HLE.HOS.Kernel.Process;
  2. using Ryujinx.Memory;
  3. using Ryujinx.Memory.Range;
  4. using System;
  5. using System.Collections.Generic;
  6. namespace Ryujinx.HLE.HOS.Kernel.Memory
  7. {
  8. class SharedMemoryStorage
  9. {
  10. private readonly KernelContext _context;
  11. private readonly KPageList _pageList;
  12. private readonly ulong _size;
  13. private IVirtualMemoryManager _borrowerMemory;
  14. private ulong _borrowerVa;
  15. public SharedMemoryStorage(KernelContext context, KPageList pageList)
  16. {
  17. _context = context;
  18. _pageList = pageList;
  19. _size = pageList.GetPagesCount() * KPageTableBase.PageSize;
  20. foreach (KPageNode pageNode in pageList)
  21. {
  22. ulong address = pageNode.Address - DramMemoryMap.DramBase;
  23. ulong size = pageNode.PagesCount * KPageTableBase.PageSize;
  24. context.Memory.Commit(address, size);
  25. }
  26. }
  27. public void Borrow(KProcess dstProcess, ulong va)
  28. {
  29. ulong currentOffset = 0;
  30. foreach (KPageNode pageNode in _pageList)
  31. {
  32. ulong address = pageNode.Address - DramMemoryMap.DramBase;
  33. ulong size = pageNode.PagesCount * KPageTableBase.PageSize;
  34. dstProcess.CpuMemory.Write(va + currentOffset, _context.Memory.GetSpan(address + currentOffset, (int)size));
  35. currentOffset += size;
  36. }
  37. _borrowerMemory = dstProcess.CpuMemory;
  38. _borrowerVa = va;
  39. }
  40. public void ZeroFill()
  41. {
  42. for (ulong offset = 0; offset < _size; offset += sizeof(ulong))
  43. {
  44. GetRef<ulong>(offset) = 0;
  45. }
  46. }
  47. public ref T GetRef<T>(ulong offset) where T : unmanaged
  48. {
  49. if (_borrowerMemory == null)
  50. {
  51. if (_pageList.Nodes.Count == 1)
  52. {
  53. ulong address = _pageList.Nodes.First.Value.Address - DramMemoryMap.DramBase;
  54. return ref _context.Memory.GetRef<T>(address + offset);
  55. }
  56. throw new NotImplementedException("Non-contiguous shared memory is not yet supported.");
  57. }
  58. else
  59. {
  60. return ref _borrowerMemory.GetRef<T>(_borrowerVa + offset);
  61. }
  62. }
  63. public IEnumerable<HostMemoryRange> GetRanges()
  64. {
  65. if (_borrowerMemory == null)
  66. {
  67. var ranges = new List<HostMemoryRange>();
  68. foreach (KPageNode pageNode in _pageList)
  69. {
  70. ulong address = pageNode.Address - DramMemoryMap.DramBase;
  71. ulong size = pageNode.PagesCount * KPageTableBase.PageSize;
  72. ranges.Add(new HostMemoryRange(_context.Memory.GetPointer(address, size), size));
  73. }
  74. return ranges;
  75. }
  76. else
  77. {
  78. return _borrowerMemory.GetPhysicalRegions(_borrowerVa, _size);
  79. }
  80. }
  81. public KPageList GetPageList()
  82. {
  83. return _pageList;
  84. }
  85. }
  86. }