 ba5c0cf5d8
			
		
	
	
		ba5c0cf5d8
		
			
		
	
	
	
	
		
			
			* bsd: Add gdkchan's Select implementation Co-authored-by: TSRBerry <20988865+tsrberry@users.noreply.github.com> * bsd: Fix Select() causing a crash with an ArgumentException .NET Sockets have to be used for the Select() call * bsd: Make Select more generic * bsd: Adjust namespaces and remove unused imports * bsd: Fix NullReferenceException in Select Co-authored-by: gdkchan <gab.dark.100@gmail.com>
		
			
				
	
	
		
			177 lines
		
	
	
		
			No EOL
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			177 lines
		
	
	
		
			No EOL
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using Ryujinx.Common.Logging;
 | |
| using Ryujinx.HLE.HOS.Services.Sockets.Bsd.Types;
 | |
| using System.Collections.Generic;
 | |
| using System.Net.Sockets;
 | |
| 
 | |
| namespace Ryujinx.HLE.HOS.Services.Sockets.Bsd.Impl
 | |
| {
 | |
|     class ManagedSocketPollManager : IPollManager
 | |
|     {
 | |
|         private static ManagedSocketPollManager _instance;
 | |
| 
 | |
|         public static ManagedSocketPollManager Instance
 | |
|         {
 | |
|             get
 | |
|             {
 | |
|                 if (_instance == null)
 | |
|                 {
 | |
|                     _instance = new ManagedSocketPollManager();
 | |
|                 }
 | |
| 
 | |
|                 return _instance;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public bool IsCompatible(PollEvent evnt)
 | |
|         {
 | |
|             return evnt.FileDescriptor is ManagedSocket;
 | |
|         }
 | |
| 
 | |
|         public LinuxError Poll(List<PollEvent> events, int timeoutMilliseconds, out int updatedCount)
 | |
|         {
 | |
|             List<Socket> readEvents = new List<Socket>();
 | |
|             List<Socket> writeEvents = new List<Socket>();
 | |
|             List<Socket> errorEvents = new List<Socket>();
 | |
| 
 | |
|             updatedCount = 0;
 | |
| 
 | |
|             foreach (PollEvent evnt in events)
 | |
|             {
 | |
|                 ManagedSocket socket = (ManagedSocket)evnt.FileDescriptor;
 | |
| 
 | |
|                 bool isValidEvent = evnt.Data.InputEvents == 0;
 | |
| 
 | |
|                 errorEvents.Add(socket.Socket);
 | |
| 
 | |
|                 if ((evnt.Data.InputEvents & PollEventTypeMask.Input) != 0)
 | |
|                 {
 | |
|                     readEvents.Add(socket.Socket);
 | |
| 
 | |
|                     isValidEvent = true;
 | |
|                 }
 | |
| 
 | |
|                 if ((evnt.Data.InputEvents & PollEventTypeMask.UrgentInput) != 0)
 | |
|                 {
 | |
|                     readEvents.Add(socket.Socket);
 | |
| 
 | |
|                     isValidEvent = true;
 | |
|                 }
 | |
| 
 | |
|                 if ((evnt.Data.InputEvents & PollEventTypeMask.Output) != 0)
 | |
|                 {
 | |
|                     writeEvents.Add(socket.Socket);
 | |
| 
 | |
|                     isValidEvent = true;
 | |
|                 }
 | |
| 
 | |
|                 if (!isValidEvent)
 | |
|                 {
 | |
|                     Logger.Warning?.Print(LogClass.ServiceBsd, $"Unsupported Poll input event type: {evnt.Data.InputEvents}");
 | |
|                     return LinuxError.EINVAL;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             try
 | |
|             {
 | |
|                 int actualTimeoutMicroseconds = timeoutMilliseconds == -1 ? -1 : timeoutMilliseconds * 1000;
 | |
| 
 | |
|                 Socket.Select(readEvents, writeEvents, errorEvents, actualTimeoutMicroseconds);
 | |
|             }
 | |
|             catch (SocketException exception)
 | |
|             {
 | |
|                 return WinSockHelper.ConvertError((WsaError)exception.ErrorCode);
 | |
|             }
 | |
| 
 | |
|             foreach (PollEvent evnt in events)
 | |
|             {
 | |
|                 Socket socket = ((ManagedSocket)evnt.FileDescriptor).Socket;
 | |
| 
 | |
|                 PollEventTypeMask outputEvents = evnt.Data.OutputEvents & ~evnt.Data.InputEvents;
 | |
| 
 | |
|                 if (errorEvents.Contains(socket))
 | |
|                 {
 | |
|                     outputEvents |= PollEventTypeMask.Error;
 | |
| 
 | |
|                     if (!socket.Connected || !socket.IsBound)
 | |
|                     {
 | |
|                         outputEvents |= PollEventTypeMask.Disconnected;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (readEvents.Contains(socket))
 | |
|                 {
 | |
|                     if ((evnt.Data.InputEvents & PollEventTypeMask.Input) != 0)
 | |
|                     {
 | |
|                         outputEvents |= PollEventTypeMask.Input;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 if (writeEvents.Contains(socket))
 | |
|                 {
 | |
|                     outputEvents |= PollEventTypeMask.Output;
 | |
|                 }
 | |
| 
 | |
|                 evnt.Data.OutputEvents = outputEvents;
 | |
|             }
 | |
| 
 | |
|             updatedCount = readEvents.Count + writeEvents.Count + errorEvents.Count;
 | |
| 
 | |
|             return LinuxError.SUCCESS;
 | |
|         }
 | |
| 
 | |
|         public LinuxError Select(List<PollEvent> events, int timeout, out int updatedCount)
 | |
|         {
 | |
|             List<Socket> readEvents = new();
 | |
|             List<Socket> writeEvents = new();
 | |
|             List<Socket> errorEvents = new();
 | |
| 
 | |
|             updatedCount = 0;
 | |
| 
 | |
|             foreach (PollEvent pollEvent in events)
 | |
|             {
 | |
|                 ManagedSocket socket = (ManagedSocket)pollEvent.FileDescriptor;
 | |
| 
 | |
|                 if (pollEvent.Data.InputEvents.HasFlag(PollEventTypeMask.Input))
 | |
|                 {
 | |
|                     readEvents.Add(socket.Socket);
 | |
|                 }
 | |
| 
 | |
|                 if (pollEvent.Data.InputEvents.HasFlag(PollEventTypeMask.Output))
 | |
|                 {
 | |
|                     writeEvents.Add(socket.Socket);
 | |
|                 }
 | |
| 
 | |
|                 if (pollEvent.Data.InputEvents.HasFlag(PollEventTypeMask.Error))
 | |
|                 {
 | |
|                     errorEvents.Add(socket.Socket);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             Socket.Select(readEvents, writeEvents, errorEvents, timeout);
 | |
| 
 | |
|             updatedCount = readEvents.Count + writeEvents.Count + errorEvents.Count;
 | |
| 
 | |
|             foreach (PollEvent pollEvent in events)
 | |
|             {
 | |
|                 ManagedSocket socket = (ManagedSocket)pollEvent.FileDescriptor;
 | |
| 
 | |
|                 if (readEvents.Contains(socket.Socket))
 | |
|                 {
 | |
|                     pollEvent.Data.OutputEvents |= PollEventTypeMask.Input;
 | |
|                 }
 | |
| 
 | |
|                 if (writeEvents.Contains(socket.Socket))
 | |
|                 {
 | |
|                     pollEvent.Data.OutputEvents |= PollEventTypeMask.Output;
 | |
|                 }
 | |
| 
 | |
|                 if (errorEvents.Contains(socket.Socket))
 | |
|                 {
 | |
|                     pollEvent.Data.OutputEvents |= PollEventTypeMask.Error;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             return LinuxError.SUCCESS;
 | |
|         }
 | |
|     }
 | |
| } |