EventFileDescriptor.cs 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. using System;
  2. using System.Runtime.InteropServices;
  3. using System.Threading;
  4. namespace Ryujinx.HLE.HOS.Services.Sockets.Bsd
  5. {
  6. class EventFileDescriptor : IFileDescriptor
  7. {
  8. private ulong _value;
  9. private readonly EventFdFlags _flags;
  10. private object _lock = new object();
  11. public bool Blocking { get => !_flags.HasFlag(EventFdFlags.NonBlocking); set => throw new NotSupportedException(); }
  12. public ManualResetEvent WriteEvent { get; }
  13. public ManualResetEvent ReadEvent { get; }
  14. public EventFileDescriptor(ulong value, EventFdFlags flags)
  15. {
  16. _value = value;
  17. _flags = flags;
  18. WriteEvent = new ManualResetEvent(true);
  19. ReadEvent = new ManualResetEvent(true);
  20. }
  21. public int Refcount { get; set; }
  22. public void Dispose()
  23. {
  24. WriteEvent.Dispose();
  25. ReadEvent.Dispose();
  26. }
  27. public LinuxError Read(out int readSize, Span<byte> buffer)
  28. {
  29. if (buffer.Length < sizeof(ulong))
  30. {
  31. readSize = 0;
  32. return LinuxError.EINVAL;
  33. }
  34. ReadEvent.Reset();
  35. lock (_lock)
  36. {
  37. ref ulong count = ref MemoryMarshal.Cast<byte, ulong>(buffer)[0];
  38. if (_value == 0)
  39. {
  40. if (Blocking)
  41. {
  42. while (_value == 0)
  43. {
  44. Monitor.Wait(_lock);
  45. }
  46. }
  47. else
  48. {
  49. readSize = 0;
  50. return LinuxError.EAGAIN;
  51. }
  52. }
  53. readSize = sizeof(ulong);
  54. if (_flags.HasFlag(EventFdFlags.Semaphore))
  55. {
  56. --_value;
  57. count = 1;
  58. }
  59. else
  60. {
  61. count = _value;
  62. _value = 0;
  63. }
  64. ReadEvent.Set();
  65. return LinuxError.SUCCESS;
  66. }
  67. }
  68. public LinuxError Write(out int writeSize, ReadOnlySpan<byte> buffer)
  69. {
  70. if (!MemoryMarshal.TryRead(buffer, out ulong count) || count == ulong.MaxValue)
  71. {
  72. writeSize = 0;
  73. return LinuxError.EINVAL;
  74. }
  75. WriteEvent.Reset();
  76. lock (_lock)
  77. {
  78. if (_value > _value + count)
  79. {
  80. if (Blocking)
  81. {
  82. Monitor.Wait(_lock);
  83. }
  84. else
  85. {
  86. writeSize = 0;
  87. return LinuxError.EAGAIN;
  88. }
  89. }
  90. writeSize = sizeof(ulong);
  91. _value += count;
  92. Monitor.Pulse(_lock);
  93. WriteEvent.Set();
  94. return LinuxError.SUCCESS;
  95. }
  96. }
  97. }
  98. }