 521751795a
			
		
	
	
		521751795a
		
			
		
	
	
	
	
		
			
			* Some style fixes and nits on ITimeZoneService * Remove some unneeded usings * Remove the Ryujinx.HLE.OsHle.Handles namespace * Remove hbmenu automatic load on process exit * Rename Ns to Device, rename Os to System, rename SystemState to State * Move Exceptions and Utilities out of OsHle * Rename OsHle to HOS * Rename OsHle folder to HOS * IManagerDisplayService and ISystemDisplayService style fixes * BsdError shouldn't be public * Add a empty new line before using static * Remove unused file * Some style fixes on NPDM * Exit gracefully when the application is closed * Code style fixes on IGeneralService * Add 0x prefix on values printed as hex * Small improvements on finalization code * Move ProcessId and ThreadId out of AThreadState * Rename VFs to FileSystem * FsAccessHeader shouldn't be public. Also fix file names casing * More case changes on NPDM * Remove unused files * Move using to the correct place on NPDM * Use properties on KernelAccessControlMmio * Address PR feedback
		
			
				
	
	
		
			398 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			398 lines
		
	
	
	
		
			12 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using ChocolArm64.Memory;
 | |
| using Ryujinx.HLE.Logging;
 | |
| using System;
 | |
| using System.Collections.Concurrent;
 | |
| using System.Text;
 | |
| using System.Threading;
 | |
| 
 | |
| namespace Ryujinx.HLE.HOS.Services.Nv.NvHostCtrl
 | |
| {
 | |
|     class NvHostCtrlIoctl
 | |
|     {
 | |
|         private static ConcurrentDictionary<Process, NvHostCtrlUserCtx> UserCtxs;
 | |
| 
 | |
|         private static bool IsProductionMode = true;
 | |
| 
 | |
|         static NvHostCtrlIoctl()
 | |
|         {
 | |
|             UserCtxs = new ConcurrentDictionary<Process, NvHostCtrlUserCtx>();
 | |
| 
 | |
|             if (Set.NxSettings.Settings.TryGetValue("nv!rmos_set_production_mode", out object ProductionModeSetting))
 | |
|             {
 | |
|                 IsProductionMode = ((string)ProductionModeSetting) != "0"; // Default value is ""
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public static int ProcessIoctl(ServiceCtx Context, int Cmd)
 | |
|         {
 | |
|             switch (Cmd & 0xffff)
 | |
|             {
 | |
|                 case 0x0014: return SyncptRead    (Context);
 | |
|                 case 0x0015: return SyncptIncr    (Context);
 | |
|                 case 0x0016: return SyncptWait    (Context);
 | |
|                 case 0x0019: return SyncptWaitEx  (Context);
 | |
|                 case 0x001a: return SyncptReadMax (Context);
 | |
|                 case 0x001b: return GetConfig     (Context);
 | |
|                 case 0x001d: return EventWait     (Context);
 | |
|                 case 0x001e: return EventWaitAsync(Context);
 | |
|                 case 0x001f: return EventRegister (Context);
 | |
|             }
 | |
| 
 | |
|             throw new NotImplementedException(Cmd.ToString("x8"));
 | |
|         }
 | |
| 
 | |
|         private static int SyncptRead(ServiceCtx Context)
 | |
|         {
 | |
|             return SyncptReadMinOrMax(Context, Max: false);
 | |
|         }
 | |
| 
 | |
|         private static int SyncptIncr(ServiceCtx Context)
 | |
|         {
 | |
|             long InputPosition = Context.Request.GetBufferType0x21().Position;
 | |
| 
 | |
|             int Id = Context.Memory.ReadInt32(InputPosition);
 | |
| 
 | |
|             if ((uint)Id >= NvHostSyncpt.SyncptsCount)
 | |
|             {
 | |
|                 return NvResult.InvalidInput;
 | |
|             }
 | |
| 
 | |
|             GetUserCtx(Context).Syncpt.Increment(Id);
 | |
| 
 | |
|             return NvResult.Success;
 | |
|         }
 | |
| 
 | |
|         private static int SyncptWait(ServiceCtx Context)
 | |
|         {
 | |
|             return SyncptWait(Context, Extended: false);
 | |
|         }
 | |
| 
 | |
|         private static int SyncptWaitEx(ServiceCtx Context)
 | |
|         {
 | |
|             return SyncptWait(Context, Extended: true);
 | |
|         }
 | |
| 
 | |
|         private static int SyncptReadMax(ServiceCtx Context)
 | |
|         {
 | |
|             return SyncptReadMinOrMax(Context, Max: true);
 | |
|         }
 | |
| 
 | |
|         private static int GetConfig(ServiceCtx Context)
 | |
|         {
 | |
|             if (!IsProductionMode)
 | |
|             {
 | |
|                 long InputPosition  = Context.Request.GetBufferType0x21().Position;
 | |
|                 long OutputPosition = Context.Request.GetBufferType0x22().Position;
 | |
| 
 | |
|                 string Domain = AMemoryHelper.ReadAsciiString(Context.Memory, InputPosition + 0, 0x41);
 | |
|                 string Name   = AMemoryHelper.ReadAsciiString(Context.Memory, InputPosition + 0x41, 0x41);
 | |
| 
 | |
|                 if (Set.NxSettings.Settings.TryGetValue($"{Domain}!{Name}", out object NvSetting))
 | |
|                 {
 | |
|                     byte[] SettingBuffer = new byte[0x101];
 | |
| 
 | |
|                     if (NvSetting is string StringValue)
 | |
|                     {
 | |
|                         if (StringValue.Length > 0x100)
 | |
|                         {
 | |
|                             Context.Device.Log.PrintError(Logging.LogClass.ServiceNv, $"{Domain}!{Name} String value size is too big!");
 | |
|                         }
 | |
|                         else
 | |
|                         {
 | |
|                             SettingBuffer = Encoding.ASCII.GetBytes(StringValue + "\0");
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     if (NvSetting is int IntValue)
 | |
|                     {
 | |
|                         SettingBuffer = BitConverter.GetBytes(IntValue);
 | |
|                     }
 | |
|                     else if (NvSetting is bool BoolValue)
 | |
|                     {
 | |
|                         SettingBuffer[0] = BoolValue ? (byte)1 : (byte)0;
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         throw new NotImplementedException(NvSetting.GetType().Name);
 | |
|                     }
 | |
| 
 | |
|                     Context.Memory.WriteBytes(OutputPosition + 0x82, SettingBuffer);
 | |
| 
 | |
|                     Context.Device.Log.PrintDebug(Logging.LogClass.ServiceNv, $"Got setting {Domain}!{Name}");
 | |
|                 }
 | |
| 
 | |
|                 return NvResult.Success;
 | |
|             }
 | |
| 
 | |
|             return NvResult.NotAvailableInProduction;
 | |
|         }
 | |
| 
 | |
|         private static int EventWait(ServiceCtx Context)
 | |
|         {
 | |
|             return EventWait(Context, Async: false);
 | |
|         }
 | |
| 
 | |
|         private static int EventWaitAsync(ServiceCtx Context)
 | |
|         {
 | |
|             return EventWait(Context, Async: true);
 | |
|         }
 | |
| 
 | |
|         private static int EventRegister(ServiceCtx Context)
 | |
|         {
 | |
|             long InputPosition  = Context.Request.GetBufferType0x21().Position;
 | |
|             long OutputPosition = Context.Request.GetBufferType0x22().Position;
 | |
| 
 | |
|             int EventId = Context.Memory.ReadInt32(InputPosition);
 | |
| 
 | |
|             Context.Device.Log.PrintStub(LogClass.ServiceNv, "Stubbed.");
 | |
| 
 | |
|             return NvResult.Success;
 | |
|         }
 | |
| 
 | |
|         private static int SyncptReadMinOrMax(ServiceCtx Context, bool Max)
 | |
|         {
 | |
|             long InputPosition  = Context.Request.GetBufferType0x21().Position;
 | |
|             long OutputPosition = Context.Request.GetBufferType0x22().Position;
 | |
| 
 | |
|             NvHostCtrlSyncptRead Args = AMemoryHelper.Read<NvHostCtrlSyncptRead>(Context.Memory, InputPosition);
 | |
| 
 | |
|             if ((uint)Args.Id >= NvHostSyncpt.SyncptsCount)
 | |
|             {
 | |
|                 return NvResult.InvalidInput;
 | |
|             }
 | |
| 
 | |
|             if (Max)
 | |
|             {
 | |
|                 Args.Value = GetUserCtx(Context).Syncpt.GetMax(Args.Id);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 Args.Value = GetUserCtx(Context).Syncpt.GetMin(Args.Id);
 | |
|             }
 | |
| 
 | |
|             AMemoryHelper.Write(Context.Memory, OutputPosition, Args);
 | |
| 
 | |
|             return NvResult.Success;
 | |
|         }
 | |
| 
 | |
|         private static int SyncptWait(ServiceCtx Context, bool Extended)
 | |
|         {
 | |
|             long InputPosition  = Context.Request.GetBufferType0x21().Position;
 | |
|             long OutputPosition = Context.Request.GetBufferType0x22().Position;
 | |
| 
 | |
|             NvHostCtrlSyncptWait Args = AMemoryHelper.Read<NvHostCtrlSyncptWait>(Context.Memory, InputPosition);
 | |
| 
 | |
|             NvHostSyncpt Syncpt = GetUserCtx(Context).Syncpt;
 | |
| 
 | |
|             if ((uint)Args.Id >= NvHostSyncpt.SyncptsCount)
 | |
|             {
 | |
|                 return NvResult.InvalidInput;
 | |
|             }
 | |
| 
 | |
|             int Result;
 | |
| 
 | |
|             if (Syncpt.MinCompare(Args.Id, Args.Thresh))
 | |
|             {
 | |
|                 Result = NvResult.Success;
 | |
|             }
 | |
|             else if (Args.Timeout == 0)
 | |
|             {
 | |
|                 Result = NvResult.TryAgain;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 Context.Device.Log.PrintDebug(LogClass.ServiceNv, "Waiting syncpt with timeout of " + Args.Timeout + "ms...");
 | |
| 
 | |
|                 using (ManualResetEvent WaitEvent = new ManualResetEvent(false))
 | |
|                 {
 | |
|                     Syncpt.AddWaiter(Args.Thresh, WaitEvent);
 | |
| 
 | |
|                     //Note: Negative (> INT_MAX) timeouts aren't valid on .NET,
 | |
|                     //in this case we just use the maximum timeout possible.
 | |
|                     int Timeout = Args.Timeout;
 | |
| 
 | |
|                     if (Timeout < -1)
 | |
|                     {
 | |
|                         Timeout = int.MaxValue;
 | |
|                     }
 | |
| 
 | |
|                     if (Timeout == -1)
 | |
|                     {
 | |
|                         WaitEvent.WaitOne();
 | |
| 
 | |
|                         Result = NvResult.Success;
 | |
|                     }
 | |
|                     else if (WaitEvent.WaitOne(Timeout))
 | |
|                     {
 | |
|                         Result = NvResult.Success;
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         Result = NvResult.TimedOut;
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 Context.Device.Log.PrintDebug(LogClass.ServiceNv, "Resuming...");
 | |
|             }
 | |
| 
 | |
|             if (Extended)
 | |
|             {
 | |
|                 Context.Memory.WriteInt32(OutputPosition + 0xc, Syncpt.GetMin(Args.Id));
 | |
|             }
 | |
| 
 | |
|             return Result;
 | |
|         }
 | |
| 
 | |
|         private static int EventWait(ServiceCtx Context, bool Async)
 | |
|         {
 | |
|             long InputPosition  = Context.Request.GetBufferType0x21().Position;
 | |
|             long OutputPosition = Context.Request.GetBufferType0x22().Position;
 | |
| 
 | |
|             NvHostCtrlSyncptWaitEx Args = AMemoryHelper.Read<NvHostCtrlSyncptWaitEx>(Context.Memory, InputPosition);
 | |
| 
 | |
|             if ((uint)Args.Id >= NvHostSyncpt.SyncptsCount)
 | |
|             {
 | |
|                 return NvResult.InvalidInput;
 | |
|             }
 | |
| 
 | |
|             void WriteArgs()
 | |
|             {
 | |
|                 AMemoryHelper.Write(Context.Memory, OutputPosition, Args);
 | |
|             }
 | |
| 
 | |
|             NvHostSyncpt Syncpt = GetUserCtx(Context).Syncpt;
 | |
| 
 | |
|             if (Syncpt.MinCompare(Args.Id, Args.Thresh))
 | |
|             {
 | |
|                 Args.Value = Syncpt.GetMin(Args.Id);
 | |
| 
 | |
|                 WriteArgs();
 | |
| 
 | |
|                 return NvResult.Success;
 | |
|             }
 | |
| 
 | |
|             if (!Async)
 | |
|             {
 | |
|                 Args.Value = 0;
 | |
|             }
 | |
| 
 | |
|             if (Args.Timeout == 0)
 | |
|             {
 | |
|                 WriteArgs();
 | |
| 
 | |
|                 return NvResult.TryAgain;
 | |
|             }
 | |
| 
 | |
|             NvHostEvent Event;
 | |
| 
 | |
|             int Result, EventIndex;
 | |
| 
 | |
|             if (Async)
 | |
|             {
 | |
|                 EventIndex = Args.Value;
 | |
| 
 | |
|                 if ((uint)EventIndex >= NvHostCtrlUserCtx.EventsCount)
 | |
|                 {
 | |
|                     return NvResult.InvalidInput;
 | |
|                 }
 | |
| 
 | |
|                 Event = GetUserCtx(Context).Events[EventIndex];
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 Event = GetFreeEvent(Context, Syncpt, Args.Id, out EventIndex);
 | |
|             }
 | |
| 
 | |
|             if (Event != null &&
 | |
|                (Event.State == NvHostEventState.Registered ||
 | |
|                 Event.State == NvHostEventState.Free))
 | |
|             {
 | |
|                 Event.Id     = Args.Id;
 | |
|                 Event.Thresh = Args.Thresh;
 | |
| 
 | |
|                 Event.State = NvHostEventState.Waiting;
 | |
| 
 | |
|                 if (!Async)
 | |
|                 {
 | |
|                     Args.Value = ((Args.Id & 0xfff) << 16) | 0x10000000;
 | |
|                 }
 | |
|                 else
 | |
|                 {
 | |
|                     Args.Value = Args.Id << 4;
 | |
|                 }
 | |
| 
 | |
|                 Args.Value |= EventIndex;
 | |
| 
 | |
|                 Result = NvResult.TryAgain;
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 Result = NvResult.InvalidInput;
 | |
|             }
 | |
| 
 | |
|             WriteArgs();
 | |
| 
 | |
|             return Result;
 | |
|         }
 | |
| 
 | |
|         private static NvHostEvent GetFreeEvent(
 | |
|             ServiceCtx   Context,
 | |
|             NvHostSyncpt Syncpt,
 | |
|             int          Id,
 | |
|             out int      EventIndex)
 | |
|         {
 | |
|             NvHostEvent[] Events = GetUserCtx(Context).Events;
 | |
| 
 | |
|             EventIndex = NvHostCtrlUserCtx.EventsCount;
 | |
| 
 | |
|             int NullIndex = NvHostCtrlUserCtx.EventsCount;
 | |
| 
 | |
|             for (int Index = 0; Index < NvHostCtrlUserCtx.EventsCount; Index++)
 | |
|             {
 | |
|                 NvHostEvent Event = Events[Index];
 | |
| 
 | |
|                 if (Event != null)
 | |
|                 {
 | |
|                     if (Event.State == NvHostEventState.Registered ||
 | |
|                         Event.State == NvHostEventState.Free)
 | |
|                     {
 | |
|                         EventIndex = Index;
 | |
| 
 | |
|                         if (Event.Id == Id)
 | |
|                         {
 | |
|                             return Event;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 else if (NullIndex == NvHostCtrlUserCtx.EventsCount)
 | |
|                 {
 | |
|                     NullIndex = Index;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             if (NullIndex < NvHostCtrlUserCtx.EventsCount)
 | |
|             {
 | |
|                 EventIndex = NullIndex;
 | |
| 
 | |
|                 return Events[NullIndex] = new NvHostEvent();
 | |
|             }
 | |
| 
 | |
|             if (EventIndex < NvHostCtrlUserCtx.EventsCount)
 | |
|             {
 | |
|                 return Events[EventIndex];
 | |
|             }
 | |
| 
 | |
|             return null;
 | |
|         }
 | |
| 
 | |
|         public static NvHostCtrlUserCtx GetUserCtx(ServiceCtx Context)
 | |
|         {
 | |
|             return UserCtxs.GetOrAdd(Context.Process, (Key) => new NvHostCtrlUserCtx());
 | |
|         }
 | |
| 
 | |
|         public static void UnloadProcess(Process Process)
 | |
|         {
 | |
|             UserCtxs.TryRemove(Process, out _);
 | |
|         }
 | |
|     }
 | |
| }
 |