2019-05-30 20:27:43 +00:00
|
|
|
|
using DiscordRPC;
|
|
|
|
|
using Ryujinx.Audio;
|
2018-10-17 17:15:50 +00:00
|
|
|
|
using Ryujinx.Common.Logging;
|
2018-02-20 20:09:23 +00:00
|
|
|
|
using Ryujinx.Graphics.Gal;
|
|
|
|
|
using Ryujinx.Graphics.Gal.OpenGL;
|
2018-06-11 00:46:42 +00:00
|
|
|
|
using Ryujinx.HLE;
|
2019-04-26 04:53:10 +00:00
|
|
|
|
using Ryujinx.Profiler;
|
2018-02-04 23:08:20 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.IO;
|
2019-05-30 20:27:43 +00:00
|
|
|
|
using System.Linq;
|
2018-02-04 23:08:20 +00:00
|
|
|
|
|
|
|
|
|
namespace Ryujinx
|
|
|
|
|
{
|
|
|
|
|
class Program
|
|
|
|
|
{
|
2019-05-30 20:27:43 +00:00
|
|
|
|
public static DiscordRpcClient DiscordClient;
|
|
|
|
|
|
|
|
|
|
public static RichPresence DiscordPresence;
|
|
|
|
|
|
2019-02-12 23:24:11 +00:00
|
|
|
|
public static string ApplicationDirectory => AppDomain.CurrentDomain.BaseDirectory;
|
|
|
|
|
|
2018-02-04 23:08:20 +00:00
|
|
|
|
static void Main(string[] args)
|
|
|
|
|
{
|
2018-02-09 00:43:22 +00:00
|
|
|
|
Console.Title = "Ryujinx Console";
|
|
|
|
|
|
2019-03-04 01:45:25 +00:00
|
|
|
|
IGalRenderer renderer = new OglRenderer();
|
2018-02-04 23:08:20 +00:00
|
|
|
|
|
2018-11-15 02:22:50 +00:00
|
|
|
|
IAalOutput audioOut = InitializeAudioEngine();
|
2018-03-16 00:06:24 +00:00
|
|
|
|
|
2018-10-31 01:43:02 +00:00
|
|
|
|
Switch device = new Switch(renderer, audioOut);
|
2018-02-04 23:08:20 +00:00
|
|
|
|
|
2019-02-12 23:24:11 +00:00
|
|
|
|
Configuration.Load(Path.Combine(ApplicationDirectory, "Config.jsonc"));
|
2019-02-11 12:00:32 +00:00
|
|
|
|
Configuration.Configure(device);
|
2019-01-31 02:49:15 +00:00
|
|
|
|
|
2019-07-02 02:39:22 +00:00
|
|
|
|
Profile.Initialize();
|
2019-04-26 04:53:10 +00:00
|
|
|
|
|
2019-01-31 02:49:15 +00:00
|
|
|
|
AppDomain.CurrentDomain.UnhandledException += CurrentDomain_UnhandledException;
|
|
|
|
|
AppDomain.CurrentDomain.ProcessExit += CurrentDomain_ProcessExit;
|
2018-04-24 18:57:39 +00:00
|
|
|
|
|
2019-08-19 20:28:01 +00:00
|
|
|
|
if (device.System.State.DiscordIntegrationEnabled)
|
2019-05-30 20:27:43 +00:00
|
|
|
|
{
|
|
|
|
|
DiscordClient = new DiscordRpcClient("568815339807309834");
|
|
|
|
|
DiscordPresence = new RichPresence
|
|
|
|
|
{
|
|
|
|
|
Assets = new Assets
|
|
|
|
|
{
|
|
|
|
|
LargeImageKey = "ryujinx",
|
|
|
|
|
LargeImageText = "Ryujinx is an emulator for the Nintendo Switch"
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
DiscordClient.Initialize();
|
|
|
|
|
DiscordClient.SetPresence(DiscordPresence);
|
|
|
|
|
}
|
|
|
|
|
|
2018-02-04 23:08:20 +00:00
|
|
|
|
if (args.Length == 1)
|
|
|
|
|
{
|
|
|
|
|
if (Directory.Exists(args[0]))
|
|
|
|
|
{
|
2018-10-31 01:43:02 +00:00
|
|
|
|
string[] romFsFiles = Directory.GetFiles(args[0], "*.istorage");
|
2018-02-04 23:08:20 +00:00
|
|
|
|
|
2018-10-31 01:43:02 +00:00
|
|
|
|
if (romFsFiles.Length == 0)
|
2018-04-06 05:02:13 +00:00
|
|
|
|
{
|
2018-10-31 01:43:02 +00:00
|
|
|
|
romFsFiles = Directory.GetFiles(args[0], "*.romfs");
|
2018-04-06 05:02:13 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-31 01:43:02 +00:00
|
|
|
|
if (romFsFiles.Length > 0)
|
2018-02-04 23:08:20 +00:00
|
|
|
|
{
|
2019-02-11 12:00:32 +00:00
|
|
|
|
Logger.PrintInfo(LogClass.Application, "Loading as cart with RomFS.");
|
2018-10-31 01:43:02 +00:00
|
|
|
|
device.LoadCart(args[0], romFsFiles[0]);
|
2018-02-04 23:08:20 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2019-02-11 12:00:32 +00:00
|
|
|
|
Logger.PrintInfo(LogClass.Application, "Loading as cart WITHOUT RomFS.");
|
2018-10-31 01:43:02 +00:00
|
|
|
|
device.LoadCart(args[0]);
|
2018-02-04 23:08:20 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (File.Exists(args[0]))
|
|
|
|
|
{
|
2018-09-08 18:33:27 +00:00
|
|
|
|
switch (Path.GetExtension(args[0]).ToLowerInvariant())
|
|
|
|
|
{
|
|
|
|
|
case ".xci":
|
2019-02-11 12:00:32 +00:00
|
|
|
|
Logger.PrintInfo(LogClass.Application, "Loading as XCI.");
|
2018-10-31 01:43:02 +00:00
|
|
|
|
device.LoadXci(args[0]);
|
2018-09-08 18:33:27 +00:00
|
|
|
|
break;
|
|
|
|
|
case ".nca":
|
2019-02-11 12:00:32 +00:00
|
|
|
|
Logger.PrintInfo(LogClass.Application, "Loading as NCA.");
|
2018-10-31 01:43:02 +00:00
|
|
|
|
device.LoadNca(args[0]);
|
2018-09-08 18:33:27 +00:00
|
|
|
|
break;
|
|
|
|
|
case ".nsp":
|
2019-01-25 01:51:28 +00:00
|
|
|
|
case ".pfs0":
|
2019-02-11 12:00:32 +00:00
|
|
|
|
Logger.PrintInfo(LogClass.Application, "Loading as NSP.");
|
2018-10-31 01:43:02 +00:00
|
|
|
|
device.LoadNsp(args[0]);
|
2018-09-08 18:33:27 +00:00
|
|
|
|
break;
|
|
|
|
|
default:
|
2019-02-11 12:00:32 +00:00
|
|
|
|
Logger.PrintInfo(LogClass.Application, "Loading as homebrew.");
|
2018-10-31 01:43:02 +00:00
|
|
|
|
device.LoadProgram(args[0]);
|
2018-09-08 18:33:27 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
2018-02-04 23:08:20 +00:00
|
|
|
|
}
|
New shader translator implementation (#654)
* Start implementing a new shader translator
* Fix shift instructions and a typo
* Small refactoring on StructuredProgram, move RemovePhis method to a separate class
* Initial geometry shader support
* Implement TLD4
* Fix -- There's no negation on FMUL32I
* Add constant folding and algebraic simplification optimizations, nits
* Some leftovers from constant folding
* Avoid cast for constant assignments
* Add a branch elimination pass, and misc small fixes
* Remove redundant branches, add expression propagation and other improvements on the code
* Small leftovers -- add missing break and continue, remove unused properties, other improvements
* Add null check to handle empty block cases on block visitor
* Add HADD2 and HMUL2 half float shader instructions
* Optimize pack/unpack sequences, some fixes related to half float instructions
* Add TXQ, TLD, TLDS and TLD4S shader texture instructions, and some support for bindless textures, some refactoring on codegen
* Fix copy paste mistake that caused RZ to be ignored on the AST instruction
* Add workaround for conditional exit, and fix half float instruction with constant buffer
* Add missing 0.0 source for TLDS.LZ variants
* Simplify the switch for TLDS.LZ
* Texture instructions related fixes
* Implement the HFMA instruction, and some misc. fixes
* Enable constant folding on UnpackHalf2x16 instructions
* Refactor HFMA to use OpCode* for opcode decoding rather than on the helper methods
* Remove the old shader translator
* Remove ShaderDeclInfo and other unused things
* Add dual vertex shader support
* Add ShaderConfig, used to pass shader type and maximum cbuffer size
* Move and rename some instruction enums
* Move texture instructions into a separate file
* Move operand GetExpression and locals management to OperandManager
* Optimize opcode decoding using a simple list and binary search
* Add missing condition for do-while on goto elimination
* Misc. fixes on texture instructions
* Simplify TLDS switch
* Address PR feedback, and a nit
2019-04-17 23:57:08 +00:00
|
|
|
|
else
|
2019-02-17 09:52:16 +00:00
|
|
|
|
{
|
|
|
|
|
Logger.PrintWarning(LogClass.Application, "Please specify a valid XCI/NCA/NSP/PFS0/NRO file");
|
|
|
|
|
}
|
2018-02-04 23:08:20 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2019-02-17 09:52:16 +00:00
|
|
|
|
Logger.PrintWarning(LogClass.Application, "Please specify the folder with the NSOs/IStorage or a NSO/NRO.");
|
2018-02-04 23:08:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-08-19 20:28:01 +00:00
|
|
|
|
if (device.System.State.DiscordIntegrationEnabled)
|
2019-05-30 20:27:43 +00:00
|
|
|
|
{
|
|
|
|
|
if (File.ReadAllLines(Path.Combine(ApplicationDirectory, "RPsupported.dat")).Contains(device.System.TitleID))
|
|
|
|
|
{
|
|
|
|
|
DiscordPresence.Assets.LargeImageKey = device.System.TitleID;
|
|
|
|
|
}
|
|
|
|
|
|
2019-08-19 20:28:01 +00:00
|
|
|
|
string state = device.System.TitleID;
|
|
|
|
|
|
|
|
|
|
if (state == null)
|
|
|
|
|
{
|
|
|
|
|
state = "Ryujinx";
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
state = state.ToUpper();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
string details = "Idling";
|
|
|
|
|
|
|
|
|
|
if (device.System.TitleName != null)
|
|
|
|
|
{
|
|
|
|
|
details = $"Playing {device.System.TitleName}";
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
DiscordPresence.Details = details;
|
|
|
|
|
DiscordPresence.State = state;
|
2019-05-30 20:27:43 +00:00
|
|
|
|
DiscordPresence.Assets.LargeImageText = device.System.TitleName;
|
|
|
|
|
DiscordPresence.Assets.SmallImageKey = "ryujinx";
|
|
|
|
|
DiscordPresence.Assets.SmallImageText = "Ryujinx is an emulator for the Nintendo Switch";
|
|
|
|
|
DiscordPresence.Timestamps = new Timestamps(DateTime.UtcNow);
|
|
|
|
|
|
|
|
|
|
DiscordClient.SetPresence(DiscordPresence);
|
|
|
|
|
}
|
|
|
|
|
|
2018-10-31 01:43:02 +00:00
|
|
|
|
using (GlScreen screen = new GlScreen(device, renderer))
|
2018-02-04 23:08:20 +00:00
|
|
|
|
{
|
2018-10-31 01:43:02 +00:00
|
|
|
|
screen.MainLoop();
|
2018-08-16 23:47:36 +00:00
|
|
|
|
|
2019-04-26 04:53:10 +00:00
|
|
|
|
Profile.FinishProfiling();
|
|
|
|
|
|
2018-10-31 01:43:02 +00:00
|
|
|
|
device.Dispose();
|
2018-02-04 23:08:20 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-10-31 01:43:02 +00:00
|
|
|
|
audioOut.Dispose();
|
2019-02-11 12:00:32 +00:00
|
|
|
|
|
|
|
|
|
Logger.Shutdown();
|
2019-05-30 20:27:43 +00:00
|
|
|
|
|
|
|
|
|
DiscordClient.Dispose();
|
2018-02-04 23:08:20 +00:00
|
|
|
|
}
|
2018-11-15 02:22:50 +00:00
|
|
|
|
|
2019-01-31 02:49:15 +00:00
|
|
|
|
private static void CurrentDomain_ProcessExit(object sender, EventArgs e)
|
|
|
|
|
{
|
2019-02-11 12:00:32 +00:00
|
|
|
|
Logger.Shutdown();
|
2019-05-30 20:27:43 +00:00
|
|
|
|
|
|
|
|
|
DiscordClient.Dispose();
|
2019-01-31 02:49:15 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static void CurrentDomain_UnhandledException(object sender, UnhandledExceptionEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
var exception = e.ExceptionObject as Exception;
|
|
|
|
|
|
|
|
|
|
Logger.PrintError(LogClass.Emulation, $"Unhandled exception caught: {exception}");
|
|
|
|
|
|
|
|
|
|
if (e.IsTerminating)
|
|
|
|
|
{
|
2019-02-11 12:00:32 +00:00
|
|
|
|
Logger.Shutdown();
|
2019-05-30 20:27:43 +00:00
|
|
|
|
|
|
|
|
|
DiscordClient.Dispose();
|
2019-01-31 02:49:15 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2018-11-15 02:22:50 +00:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Picks an <see cref="IAalOutput"/> audio output renderer supported on this machine
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <returns>An <see cref="IAalOutput"/> supported by this machine</returns>
|
|
|
|
|
private static IAalOutput InitializeAudioEngine()
|
|
|
|
|
{
|
|
|
|
|
if (SoundIoAudioOut.IsSupported)
|
|
|
|
|
{
|
|
|
|
|
return new SoundIoAudioOut();
|
|
|
|
|
}
|
|
|
|
|
else if (OpenALAudioOut.IsSupported)
|
|
|
|
|
{
|
|
|
|
|
return new OpenALAudioOut();
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return new DummyAudioOut();
|
|
|
|
|
}
|
|
|
|
|
}
|
2018-02-04 23:08:20 +00:00
|
|
|
|
}
|
|
|
|
|
}
|