2018-09-08 20:33:27 +02:00
|
|
|
using LibHac;
|
2018-08-17 01:47:36 +02:00
|
|
|
using Ryujinx.HLE.HOS.Font;
|
|
|
|
using Ryujinx.HLE.HOS.Kernel;
|
|
|
|
using Ryujinx.HLE.HOS.SystemState;
|
2018-06-11 02:46:42 +02:00
|
|
|
using Ryujinx.HLE.Loaders.Executables;
|
2018-07-20 23:53:06 +02:00
|
|
|
using Ryujinx.HLE.Loaders.Npdm;
|
2018-06-11 02:46:42 +02:00
|
|
|
using Ryujinx.HLE.Logging;
|
2018-02-20 21:09:23 +01:00
|
|
|
using System;
|
2018-02-05 00:08:20 +01:00
|
|
|
using System.Collections.Concurrent;
|
|
|
|
using System.IO;
|
2018-09-08 20:33:27 +02:00
|
|
|
using System.Linq;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
namespace Ryujinx.HLE.HOS
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-03-12 05:04:52 +01:00
|
|
|
public class Horizon : IDisposable
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
|
|
|
internal const int HidSize = 0x40000;
|
2018-08-15 20:59:51 +02:00
|
|
|
internal const int FontSize = 0x1100000;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
private Switch Device;
|
2018-04-24 22:14:26 +02:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
private KProcessScheduler Scheduler;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
|
|
|
private ConcurrentDictionary<int, Process> Processes;
|
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
public SystemStateMgr State { get; private set; }
|
2018-04-24 22:14:26 +02:00
|
|
|
|
2018-08-15 20:59:51 +02:00
|
|
|
internal KSharedMemory HidSharedMem { get; private set; }
|
|
|
|
internal KSharedMemory FontSharedMem { get; private set; }
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
|
2018-08-15 20:59:51 +02:00
|
|
|
internal SharedFontManager Font { get; private set; }
|
2018-03-19 19:58:46 +01:00
|
|
|
|
|
|
|
internal KEvent VsyncEvent { get; private set; }
|
2018-02-17 22:36:08 +01:00
|
|
|
|
2018-09-08 20:33:27 +02:00
|
|
|
internal Keyset KeySet { get; private set; }
|
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
public Horizon(Switch Device)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-08-17 01:47:36 +02:00
|
|
|
this.Device = Device;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
Scheduler = new KProcessScheduler(Device.Log);
|
2018-02-05 00:08:20 +01:00
|
|
|
|
|
|
|
Processes = new ConcurrentDictionary<int, Process>();
|
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
State = new SystemStateMgr();
|
2018-04-24 22:14:26 +02:00
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
if (!Device.Memory.Allocator.TryAllocate(HidSize, out long HidPA) ||
|
|
|
|
!Device.Memory.Allocator.TryAllocate(FontSize, out long FontPA))
|
2018-08-15 20:59:51 +02:00
|
|
|
{
|
|
|
|
throw new InvalidOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
HidSharedMem = new KSharedMemory(HidPA, HidSize);
|
|
|
|
FontSharedMem = new KSharedMemory(FontPA, FontSize);
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 20:53:23 +02:00
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
Font = new SharedFontManager(Device, FontSharedMem.PA);
|
2018-03-19 19:58:46 +01:00
|
|
|
|
|
|
|
VsyncEvent = new KEvent();
|
2018-09-08 20:33:27 +02:00
|
|
|
|
|
|
|
LoadKeySet();
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
public void LoadCart(string ExeFsDir, string RomFsFile = null)
|
|
|
|
{
|
|
|
|
if (RomFsFile != null)
|
|
|
|
{
|
2018-08-17 01:47:36 +02:00
|
|
|
Device.FileSystem.LoadRomFs(RomFsFile);
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
2018-08-15 20:59:51 +02:00
|
|
|
string NpdmFileName = Path.Combine(ExeFsDir, "main.npdm");
|
|
|
|
|
|
|
|
Npdm MetaData = null;
|
|
|
|
|
|
|
|
if (File.Exists(NpdmFileName))
|
|
|
|
{
|
2018-08-17 01:47:36 +02:00
|
|
|
Device.Log.PrintInfo(LogClass.Loader, $"Loading main.npdm...");
|
2018-08-15 20:59:51 +02:00
|
|
|
|
|
|
|
using (FileStream Input = new FileStream(NpdmFileName, FileMode.Open))
|
|
|
|
{
|
|
|
|
MetaData = new Npdm(Input);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-08-17 01:47:36 +02:00
|
|
|
Device.Log.PrintWarning(LogClass.Loader, $"NPDM file not found, using default values!");
|
2018-08-15 20:59:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
Process MainProcess = MakeProcess(MetaData);
|
2018-02-05 00:08:20 +01:00
|
|
|
|
|
|
|
void LoadNso(string FileName)
|
|
|
|
{
|
|
|
|
foreach (string File in Directory.GetFiles(ExeFsDir, FileName))
|
|
|
|
{
|
|
|
|
if (Path.GetExtension(File) != string.Empty)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
Device.Log.PrintInfo(LogClass.Loader, $"Loading {Path.GetFileNameWithoutExtension(File)}...");
|
2018-02-17 22:06:11 +01:00
|
|
|
|
2018-02-05 00:08:20 +01:00
|
|
|
using (FileStream Input = new FileStream(File, FileMode.Open))
|
|
|
|
{
|
2018-04-22 06:21:49 +02:00
|
|
|
string Name = Path.GetFileNameWithoutExtension(File);
|
|
|
|
|
|
|
|
Nso Program = new Nso(Input, Name);
|
2018-02-05 00:08:20 +01:00
|
|
|
|
|
|
|
MainProcess.LoadProgram(Program);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-09 00:04:26 +02:00
|
|
|
if (!(MainProcess.MetaData?.Is64Bits ?? true))
|
2018-07-20 23:53:06 +02:00
|
|
|
{
|
2018-08-03 02:58:48 +02:00
|
|
|
throw new NotImplementedException("32-bit titles are unsupported!");
|
2018-07-20 23:53:06 +02:00
|
|
|
}
|
|
|
|
|
2018-02-05 00:08:20 +01:00
|
|
|
LoadNso("rtld");
|
|
|
|
|
|
|
|
MainProcess.SetEmptyArgs();
|
|
|
|
|
|
|
|
LoadNso("main");
|
|
|
|
LoadNso("subsdk*");
|
|
|
|
LoadNso("sdk");
|
|
|
|
|
|
|
|
MainProcess.Run();
|
|
|
|
}
|
|
|
|
|
2018-09-08 20:33:27 +02:00
|
|
|
public void LoadXci(string XciFile)
|
|
|
|
{
|
|
|
|
FileStream File = new FileStream(XciFile, FileMode.Open, FileAccess.Read);
|
|
|
|
|
|
|
|
Xci Xci = new Xci(KeySet, File);
|
|
|
|
|
|
|
|
Nca Nca = GetXciMainNca(Xci);
|
|
|
|
|
|
|
|
if (Nca == null)
|
|
|
|
{
|
|
|
|
Device.Log.PrintError(LogClass.Loader, "Unable to load XCI");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
LoadNca(Nca);
|
|
|
|
}
|
|
|
|
|
|
|
|
private Nca GetXciMainNca(Xci Xci)
|
|
|
|
{
|
|
|
|
if (Xci.SecurePartition == null)
|
|
|
|
{
|
|
|
|
throw new InvalidDataException("Could not find XCI secure partition");
|
|
|
|
}
|
|
|
|
|
|
|
|
Nca MainNca = null;
|
|
|
|
Nca PatchNca = null;
|
|
|
|
|
|
|
|
foreach (PfsFileEntry FileEntry in Xci.SecurePartition.Files.Where(x => x.Name.EndsWith(".nca")))
|
|
|
|
{
|
|
|
|
Stream NcaStream = Xci.SecurePartition.OpenFile(FileEntry);
|
|
|
|
|
|
|
|
Nca Nca = new Nca(KeySet, NcaStream, true);
|
|
|
|
|
|
|
|
if (Nca.Header.ContentType == ContentType.Program)
|
|
|
|
{
|
|
|
|
if (Nca.Sections.Any(x => x?.Type == SectionType.Romfs))
|
|
|
|
{
|
|
|
|
MainNca = Nca;
|
|
|
|
}
|
|
|
|
else if (Nca.Sections.Any(x => x?.Type == SectionType.Bktr))
|
|
|
|
{
|
|
|
|
PatchNca = Nca;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (MainNca == null)
|
|
|
|
{
|
|
|
|
Device.Log.PrintError(LogClass.Loader, "Could not find an Application NCA in the provided XCI file");
|
|
|
|
}
|
|
|
|
|
|
|
|
MainNca.SetBaseNca(PatchNca);
|
|
|
|
|
|
|
|
return MainNca;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void LoadNca(string NcaFile)
|
|
|
|
{
|
|
|
|
FileStream File = new FileStream(NcaFile, FileMode.Open, FileAccess.Read);
|
|
|
|
|
|
|
|
Nca Nca = new Nca(KeySet, File, true);
|
|
|
|
|
|
|
|
LoadNca(Nca);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void LoadNsp(string NspFile)
|
|
|
|
{
|
|
|
|
FileStream File = new FileStream(NspFile, FileMode.Open, FileAccess.Read);
|
|
|
|
|
|
|
|
Pfs Nsp = new Pfs(File);
|
|
|
|
|
|
|
|
PfsFileEntry TicketFile = Nsp.Files.FirstOrDefault(x => x.Name.EndsWith(".tik"));
|
|
|
|
|
|
|
|
// Load title key from the NSP's ticket in case the user doesn't have a title key file
|
|
|
|
if (TicketFile != null)
|
|
|
|
{
|
|
|
|
// todo Change when Ticket(Stream) overload is added
|
|
|
|
Ticket Ticket = new Ticket(new BinaryReader(Nsp.OpenFile(TicketFile)));
|
|
|
|
|
|
|
|
KeySet.TitleKeys[Ticket.RightsId] = Ticket.GetTitleKey(KeySet);
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach (PfsFileEntry NcaFile in Nsp.Files.Where(x => x.Name.EndsWith(".nca")))
|
|
|
|
{
|
|
|
|
Nca Nca = new Nca(KeySet, Nsp.OpenFile(NcaFile), true);
|
|
|
|
|
|
|
|
if (Nca.Header.ContentType == ContentType.Program)
|
|
|
|
{
|
|
|
|
LoadNca(Nca);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Device.Log.PrintError(LogClass.Loader, "Could not find an Application NCA in the provided NSP file");
|
|
|
|
}
|
|
|
|
|
|
|
|
public void LoadNca(Nca Nca)
|
|
|
|
{
|
|
|
|
NcaSection RomfsSection = Nca.Sections.FirstOrDefault(x => x?.Type == SectionType.Romfs);
|
|
|
|
NcaSection ExefsSection = Nca.Sections.FirstOrDefault(x => x?.IsExefs == true);
|
|
|
|
|
|
|
|
if (ExefsSection == null)
|
|
|
|
{
|
|
|
|
Device.Log.PrintError(LogClass.Loader, "No ExeFS found in NCA");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (RomfsSection == null)
|
|
|
|
{
|
|
|
|
Device.Log.PrintError(LogClass.Loader, "No RomFS found in NCA");
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Stream RomfsStream = Nca.OpenSection(RomfsSection.SectionNum, false);
|
|
|
|
Device.FileSystem.SetRomFs(RomfsStream);
|
|
|
|
|
|
|
|
Stream ExefsStream = Nca.OpenSection(ExefsSection.SectionNum, false);
|
|
|
|
Pfs Exefs = new Pfs(ExefsStream);
|
|
|
|
|
|
|
|
Npdm MetaData = null;
|
|
|
|
|
|
|
|
if (Exefs.FileExists("main.npdm"))
|
|
|
|
{
|
|
|
|
Device.Log.PrintInfo(LogClass.Loader, "Loading main.npdm...");
|
|
|
|
|
|
|
|
MetaData = new Npdm(Exefs.OpenFile("main.npdm"));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Device.Log.PrintWarning(LogClass.Loader, $"NPDM file not found, using default values!");
|
|
|
|
}
|
|
|
|
|
|
|
|
Process MainProcess = MakeProcess(MetaData);
|
|
|
|
|
|
|
|
void LoadNso(string Filename)
|
|
|
|
{
|
|
|
|
foreach (PfsFileEntry File in Exefs.Files.Where(x => x.Name.StartsWith(Filename)))
|
|
|
|
{
|
|
|
|
if (Path.GetExtension(File.Name) != string.Empty)
|
|
|
|
{
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
Device.Log.PrintInfo(LogClass.Loader, $"Loading {Filename}...");
|
|
|
|
|
|
|
|
string Name = Path.GetFileNameWithoutExtension(File.Name);
|
|
|
|
|
|
|
|
Nso Program = new Nso(Exefs.OpenFile(File), Name);
|
|
|
|
|
|
|
|
MainProcess.LoadProgram(Program);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!MainProcess.MetaData.Is64Bits)
|
|
|
|
{
|
|
|
|
throw new NotImplementedException("32-bit titles are unsupported!");
|
|
|
|
}
|
|
|
|
|
|
|
|
LoadNso("rtld");
|
|
|
|
|
|
|
|
MainProcess.SetEmptyArgs();
|
|
|
|
|
|
|
|
LoadNso("main");
|
|
|
|
LoadNso("subsdk");
|
|
|
|
LoadNso("sdk");
|
|
|
|
|
|
|
|
MainProcess.Run();
|
|
|
|
}
|
|
|
|
|
2018-07-17 21:14:27 +02:00
|
|
|
public void LoadProgram(string FilePath)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-07-17 21:14:27 +02:00
|
|
|
bool IsNro = Path.GetExtension(FilePath).ToLower() == ".nro";
|
2018-02-24 01:59:38 +01:00
|
|
|
|
2018-07-17 21:14:27 +02:00
|
|
|
string Name = Path.GetFileNameWithoutExtension(FilePath);
|
2018-08-17 01:47:36 +02:00
|
|
|
string SwitchFilePath = Device.FileSystem.SystemPathToSwitchPath(FilePath);
|
2018-07-17 21:14:27 +02:00
|
|
|
|
|
|
|
if (IsNro && (SwitchFilePath == null || !SwitchFilePath.StartsWith("sdmc:/")))
|
|
|
|
{
|
|
|
|
string SwitchPath = $"sdmc:/switch/{Name}{Homebrew.TemporaryNroSuffix}";
|
2018-08-17 01:47:36 +02:00
|
|
|
string TempPath = Device.FileSystem.SwitchPathToSystemPath(SwitchPath);
|
2018-07-17 21:14:27 +02:00
|
|
|
|
|
|
|
string SwitchDir = Path.GetDirectoryName(TempPath);
|
2018-08-17 01:47:36 +02:00
|
|
|
|
2018-07-17 21:14:27 +02:00
|
|
|
if (!Directory.Exists(SwitchDir))
|
|
|
|
{
|
|
|
|
Directory.CreateDirectory(SwitchDir);
|
|
|
|
}
|
2018-08-17 01:47:36 +02:00
|
|
|
|
2018-07-17 21:14:27 +02:00
|
|
|
File.Copy(FilePath, TempPath, true);
|
|
|
|
|
|
|
|
FilePath = TempPath;
|
|
|
|
}
|
2018-04-22 06:21:49 +02:00
|
|
|
|
2018-03-12 05:04:52 +01:00
|
|
|
Process MainProcess = MakeProcess();
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-07-17 21:14:27 +02:00
|
|
|
using (FileStream Input = new FileStream(FilePath, FileMode.Open))
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-02-24 01:59:38 +01:00
|
|
|
MainProcess.LoadProgram(IsNro
|
2018-07-17 21:14:27 +02:00
|
|
|
? (IExecutable)new Nro(Input, FilePath)
|
|
|
|
: (IExecutable)new Nso(Input, FilePath));
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
MainProcess.SetEmptyArgs();
|
2018-02-24 01:59:38 +01:00
|
|
|
MainProcess.Run(IsNro);
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
2018-09-08 20:33:27 +02:00
|
|
|
public void LoadKeySet()
|
|
|
|
{
|
|
|
|
string KeyFile = null;
|
|
|
|
string TitleKeyFile = null;
|
|
|
|
string ConsoleKeyFile = null;
|
|
|
|
|
|
|
|
string Home = Environment.GetFolderPath(Environment.SpecialFolder.UserProfile);
|
|
|
|
|
|
|
|
LoadSetAtPath(Path.Combine(Home, ".switch"));
|
|
|
|
LoadSetAtPath(Device.FileSystem.GetSystemPath());
|
|
|
|
|
|
|
|
KeySet = ExternalKeys.ReadKeyFile(KeyFile, TitleKeyFile, ConsoleKeyFile);
|
|
|
|
|
|
|
|
void LoadSetAtPath(string BasePath)
|
|
|
|
{
|
|
|
|
string LocalKeyFile = Path.Combine(BasePath, "prod.keys");
|
|
|
|
string LocalTitleKeyFile = Path.Combine(BasePath, "title.keys");
|
|
|
|
string LocalConsoleKeyFile = Path.Combine(BasePath, "console.keys");
|
|
|
|
|
|
|
|
if (File.Exists(LocalKeyFile))
|
|
|
|
{
|
|
|
|
KeyFile = LocalKeyFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (File.Exists(LocalTitleKeyFile))
|
|
|
|
{
|
|
|
|
TitleKeyFile = LocalTitleKeyFile;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (File.Exists(LocalConsoleKeyFile))
|
|
|
|
{
|
|
|
|
ConsoleKeyFile = LocalConsoleKeyFile;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
public void SignalVsync() => VsyncEvent.WaitEvent.Set();
|
2018-03-19 19:58:46 +01:00
|
|
|
|
2018-08-15 20:59:51 +02:00
|
|
|
private Process MakeProcess(Npdm MetaData = null)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-03-12 05:04:52 +01:00
|
|
|
Process Process;
|
|
|
|
|
|
|
|
lock (Processes)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-03-12 05:04:52 +01:00
|
|
|
int ProcessId = 0;
|
|
|
|
|
|
|
|
while (Processes.ContainsKey(ProcessId))
|
|
|
|
{
|
|
|
|
ProcessId++;
|
|
|
|
}
|
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
Process = new Process(Device, Scheduler, ProcessId, MetaData);
|
2018-03-12 05:04:52 +01:00
|
|
|
|
|
|
|
Processes.TryAdd(ProcessId, Process);
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
2018-03-12 05:04:52 +01:00
|
|
|
|
2018-03-19 19:58:46 +01:00
|
|
|
InitializeProcess(Process);
|
|
|
|
|
2018-03-12 05:04:52 +01:00
|
|
|
return Process;
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
2018-03-19 19:58:46 +01:00
|
|
|
private void InitializeProcess(Process Process)
|
|
|
|
{
|
|
|
|
Process.AppletState.SetFocus(true);
|
|
|
|
}
|
|
|
|
|
2018-03-12 05:04:52 +01:00
|
|
|
internal void ExitProcess(int ProcessId)
|
2018-02-17 22:36:08 +01:00
|
|
|
{
|
2018-08-17 01:47:36 +02:00
|
|
|
if (Processes.TryRemove(ProcessId, out Process Process))
|
2018-02-17 22:36:08 +01:00
|
|
|
{
|
2018-03-12 05:04:52 +01:00
|
|
|
Process.Dispose();
|
2018-02-17 22:36:08 +01:00
|
|
|
|
2018-03-12 05:04:52 +01:00
|
|
|
if (Processes.Count == 0)
|
|
|
|
{
|
2018-08-17 01:47:36 +02:00
|
|
|
Unload();
|
|
|
|
|
|
|
|
Device.Unload();
|
2018-03-12 05:04:52 +01:00
|
|
|
}
|
|
|
|
}
|
2018-02-15 13:16:16 +01:00
|
|
|
}
|
2018-02-17 22:36:08 +01:00
|
|
|
|
2018-08-17 01:47:36 +02:00
|
|
|
private void Unload()
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-08-17 01:47:36 +02:00
|
|
|
VsyncEvent.Dispose();
|
|
|
|
|
|
|
|
Scheduler.Dispose();
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
2018-03-12 05:04:52 +01:00
|
|
|
public void Dispose()
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-03-12 05:04:52 +01:00
|
|
|
Dispose(true);
|
|
|
|
}
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-03-12 05:04:52 +01:00
|
|
|
protected virtual void Dispose(bool Disposing)
|
|
|
|
{
|
|
|
|
if (Disposing)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-03-12 05:04:52 +01:00
|
|
|
foreach (Process Process in Processes.Values)
|
|
|
|
{
|
|
|
|
Process.Dispose();
|
|
|
|
}
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-08-13 23:31:09 +02:00
|
|
|
}
|