cemu-DS4Windows/DS4Windows/Program.cs

175 lines
6.4 KiB
C#
Raw Normal View History

using System;
using System.Windows.Forms;
using System.Threading;
using System.Runtime.InteropServices;
using System.Diagnostics;
using System.ComponentModel;
using System.Globalization;
using Microsoft.Win32.TaskScheduler;
namespace DS4Windows
{
static class Program
{
[DllImport("user32.dll")]
[return: MarshalAs(UnmanagedType.Bool)]
static extern bool SetForegroundWindow(IntPtr hWnd);
// Add "global\" in front of the EventName, then only one instance is allowed on the
// whole system, including other users. But the application can not be brought
// into view, of course.
2017-07-03 17:31:58 +02:00
private static string SingleAppComEventName = "{a52b5b20-d9ee-4f32-8518-307fa14aa0c6}";
private static BackgroundWorker singleAppComThread = null;
private static EventWaitHandle threadComEvent = null;
2017-07-03 17:31:58 +02:00
private static bool exitComThread = false;
public static ControlService rootHub;
/// <summary>
/// The main entry point for the application.
/// </summary>
[STAThread]
static void Main(string[] args)
{
//Thread.CurrentThread.CurrentUICulture = CultureInfo.GetCultureInfo("he");
for (int i = 0, argsLen = args.Length; i < argsLen; i++)
{
string s = args[i];
if (s == "driverinstall" || s == "-driverinstall")
{
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new WelcomeDialog());
return;
}
else if (s == "re-enabledevice" || s == "-re-enabledevice")
{
try
{
i++;
string deviceInstanceId = args[i];
DS4Devices.reEnableDevice(deviceInstanceId);
Environment.ExitCode = 0;
return;
}
catch (Exception)
{
Environment.ExitCode = Marshal.GetLastWin32Error();
return;
}
}
else if (s == "runtask" || s == "-runtask")
{
TaskService ts = new TaskService();
Task tasker = ts.FindTask("RunDS4Windows");
if (tasker != null)
{
tasker.Run("");
}
Environment.ExitCode = 0;
return;
}
}
2017-06-20 06:37:08 +02:00
System.Runtime.GCSettings.LatencyMode = System.Runtime.GCLatencyMode.LowLatency;
2017-04-21 05:09:08 +02:00
try
{
2017-07-03 17:31:58 +02:00
Process.GetCurrentProcess().PriorityClass =
ProcessPriorityClass.High;
}
catch
{
// Ignore problems raising the priority.
}
2017-04-21 05:09:08 +02:00
try
{
// another instance is already running if OpenExsting succeeds.
threadComEvent = EventWaitHandle.OpenExisting(SingleAppComEventName);
threadComEvent.Set(); // signal the other instance.
threadComEvent.Close();
return; // return immediatly.
}
catch { /* don't care about errors */ }
2017-04-21 05:09:08 +02:00
// Create the Event handle
2017-07-03 17:31:58 +02:00
threadComEvent = new EventWaitHandle(false, EventResetMode.ManualReset, SingleAppComEventName);
CreateInterAppComThread();
2017-07-03 10:24:54 +02:00
//if (mutex.WaitOne(TimeSpan.Zero, true))
//{
rootHub = new ControlService();
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
Application.Run(new DS4Form(args));
2017-07-03 17:31:58 +02:00
//mutex.ReleaseMutex();
2017-07-03 10:24:54 +02:00
//}
2017-07-03 17:31:58 +02:00
exitComThread = true;
threadComEvent.Set(); // signal the other instance.
while (singleAppComThread.IsBusy)
Thread.Sleep(50);
threadComEvent.Close();
}
static private void CreateInterAppComThread()
{
singleAppComThread = new BackgroundWorker();
singleAppComThread.DoWork += new DoWorkEventHandler(singleAppComThread_DoWork);
2017-07-03 17:31:58 +02:00
singleAppComThread.RunWorkerAsync();
}
static private void singleAppComThread_DoWork(object sender, DoWorkEventArgs e)
{
BackgroundWorker worker = sender as BackgroundWorker;
WaitHandle[] waitHandles = new WaitHandle[] { threadComEvent };
2017-06-20 06:37:08 +02:00
Thread.CurrentThread.Priority = ThreadPriority.Lowest;
2017-07-03 17:31:58 +02:00
while (!exitComThread)
{
// check every second for a signal.
2017-07-03 10:24:54 +02:00
if (WaitHandle.WaitAny(waitHandles) == 0)
{
2017-07-03 17:31:58 +02:00
threadComEvent.Reset();
// The user tried to start another instance. We can't allow that,
// so bring the other instance back into view and enable that one.
// That form is created in another thread, so we need some thread sync magic.
2017-07-03 17:31:58 +02:00
if (!exitComThread && Application.OpenForms.Count > 0)
{
Form mainForm = Application.OpenForms[0];
2017-08-17 01:00:44 +02:00
mainForm?.Invoke(new SetFormVisableDelegate(ThreadFormVisable), mainForm);
}
}
}
}
/// <summary>
/// When this method is called using a Invoke then this runs in the thread
/// that created the form, which is nice.
/// </summary>
/// <param name="frm"></param>
private delegate void SetFormVisableDelegate(Form frm);
static private void ThreadFormVisable(Form frm)
{
2014-11-18 23:07:27 +01:00
if (frm != null)
{
if (frm is DS4Form)
{
// display the form and bring to foreground.
frm.WindowState = FormWindowState.Normal;
2014-11-18 23:07:27 +01:00
frm.Focus();
}
else
{
WinProgs wp = (WinProgs)frm;
wp.form.mAllowVisible = true;
wp.ShowMainWindow();
SetForegroundWindow(wp.form.Handle);
}
}
2017-06-20 06:37:08 +02:00
SetForegroundWindow(frm.Handle);
}
}
}