Fixed instance checking failed in some cases due to 2c6ed8e

This commit is contained in:
Birunthan Mohanathas 2012-05-24 09:19:52 +03:00
parent 73dba004e8
commit 7cee196f11
2 changed files with 81 additions and 41 deletions

View File

@ -50,45 +50,25 @@ CRainmeter* Rainmeter; // The module
*/ */
int RainmeterMain(LPWSTR cmdLine) int RainmeterMain(LPWSTR cmdLine)
{ {
HWND wnd = NULL;
while (wnd = FindWindowEx(NULL, wnd, RAINMETER_CLASS_NAME, RAINMETER_WINDOW_NAME))
{
COPYDATASTRUCT cds;
if (cmdLine[0] == L'!' || cmdLine[0] == L'[') if (cmdLine[0] == L'!' || cmdLine[0] == L'[')
{
HWND wnd = FindWindow(RAINMETER_CLASS_NAME, RAINMETER_WINDOW_NAME);
if (wnd)
{ {
// Deliver bang to existing Rainmeter instance // Deliver bang to existing Rainmeter instance
COPYDATASTRUCT cds;
cds.dwData = 1; cds.dwData = 1;
cds.cbData = (DWORD)((wcslen(cmdLine) + 1) * sizeof(WCHAR)); cds.cbData = (DWORD)((wcslen(cmdLine) + 1) * sizeof(WCHAR));
cds.lpData = (PVOID)cmdLine; cds.lpData = (PVOID)cmdLine;
SendMessage(wnd, WM_COPYDATA, NULL, (LPARAM)&cds); SendMessage(wnd, WM_COPYDATA, NULL, (LPARAM)&cds);
return 0; return 0;
} }
else else if (_wcsicmp(L"!RainmeterQuit", cmdLine) != 0 &&
{
const WCHAR* fullCmdLine = GetCommandLine();
COPYDATASTRUCT cds;
cds.dwData = SIZE_MAX;
cds.cbData = (DWORD)((wcslen(fullCmdLine) + 1) * sizeof(WCHAR));
cds.lpData = (PVOID)fullCmdLine;
if (SendMessage(wnd, WM_COPYDATA, NULL, (LPARAM)&cds) == SIZE_MAX)
{
// An instance of Rainmeter with same command-line arguments already exists
return 1;
}
}
}
if ((cmdLine[0] == L'!' || cmdLine[0] == L'[') &&
_wcsicmp(L"!RainmeterQuit", cmdLine) != 0 &&
_wcsicmp(L"!Quit", cmdLine) != 0) _wcsicmp(L"!Quit", cmdLine) != 0)
{ {
MessageBox(NULL, L"Unable to send bang: Rainmeter is not running.", L"Rainmeter", MB_OK | MB_TOPMOST | MB_ICONERROR);
return 1; return 1;
} }
}
// Avoid loading a dll from current directory // Avoid loading a dll from current directory
SetDllDirectory(L""); SetDllDirectory(L"");
@ -96,6 +76,8 @@ int RainmeterMain(LPWSTR cmdLine)
int ret = 1; int ret = 1;
Rainmeter = new CRainmeter; Rainmeter = new CRainmeter;
if (Rainmeter) if (Rainmeter)
{
if (!Rainmeter->IsAlreadyRunning())
{ {
try try
{ {
@ -110,6 +92,7 @@ int RainmeterMain(LPWSTR cmdLine)
{ {
ret = Rainmeter->MessagePump(); ret = Rainmeter->MessagePump();
} }
}
delete Rainmeter; delete Rainmeter;
Rainmeter = NULL; Rainmeter = NULL;
@ -682,6 +665,7 @@ CRainmeter::CRainmeter() :
m_Logging(false), m_Logging(false),
m_CurrentParser(), m_CurrentParser(),
m_Window(), m_Window(),
m_Mutex(),
m_Instance(), m_Instance(),
m_ResourceInstance(), m_ResourceInstance(),
m_ResourceLCID(), m_ResourceLCID(),
@ -727,6 +711,7 @@ CRainmeter::~CRainmeter()
FinalizeLitestep(); FinalizeLitestep();
if (m_Mutex) ReleaseMutex(m_Mutex);
if (m_ResourceInstance) FreeLibrary(m_ResourceInstance); if (m_ResourceInstance) FreeLibrary(m_ResourceInstance);
CoUninitialize(); CoUninitialize();
@ -1053,6 +1038,63 @@ int CRainmeter::Initialize(LPCWSTR szPath)
return result; // All is OK return result; // All is OK
} }
bool CRainmeter::IsAlreadyRunning()
{
typedef struct
{
ULONG i[2];
ULONG buf[4];
unsigned char in[64];
unsigned char digest[16];
} MD5_CTX;
typedef void (WINAPI * FPMD5INIT)(MD5_CTX* context);
typedef void (WINAPI * FPMD5UPDATE)(MD5_CTX* context, const unsigned char* input, unsigned int inlen);
typedef void (WINAPI * FPMD5FINAL)(MD5_CTX* context);
// Create MD5 digest from command line
HMODULE cryptDll = CSystem::RmLoadLibrary(L"cryptdll.dll");
if (cryptDll)
{
FPMD5INIT MD5Init = (FPMD5INIT)GetProcAddress(cryptDll, "MD5Init");
FPMD5UPDATE MD5Update = (FPMD5UPDATE)GetProcAddress(cryptDll, "MD5Update");
FPMD5FINAL MD5Final = (FPMD5FINAL)GetProcAddress(cryptDll, "MD5Final");
if (MD5Init && MD5Update && MD5Final)
{
std::wstring cmdLine = GetCommandLine();
_wcsupr(&cmdLine[0]);
MD5_CTX ctx = {0};
MD5Init(&ctx);
MD5Update(&ctx, (LPBYTE)&cmdLine[0], cmdLine.length() * sizeof(WCHAR));
MD5Final(&ctx);
FreeLibrary(cryptDll);
// Convert MD5 digest to mutex string (e.g. "Rainmeter0123456789abcdef0123456789abcdef")
const WCHAR hexChars[] = L"0123456789abcdef";
WCHAR mutexName[64] = L"Rainmeter";
WCHAR* pos = mutexName + (_countof(L"Rainmeter") - 1);
for (size_t i = 0; i < 16; ++i)
{
*(pos++) = hexChars[ctx.digest[i] >> 4];
*(pos++) = hexChars[ctx.digest[i] & 0xF];
}
*pos = L'\0';
m_Mutex = CreateMutex(NULL, FALSE, mutexName);
if (GetLastError() == ERROR_ALREADY_EXISTS)
{
m_Mutex = NULL;
return true;
}
}
FreeLibrary(cryptDll);
}
return false;
}
int CRainmeter::MessagePump() int CRainmeter::MessagePump()
{ {
MSG msg; MSG msg;
@ -1097,10 +1139,6 @@ LRESULT CALLBACK CRainmeter::MainWndProc(HWND hWnd, UINT uMsg, WPARAM wParam, LP
{ {
Rainmeter->DelayedExecuteCommand(data); Rainmeter->DelayedExecuteCommand(data);
} }
else if (cds->dwData == SIZE_MAX && _wcsicmp(GetCommandLine(), data) == 0)
{
return SIZE_MAX;
}
} }
} }
break; break;

View File

@ -122,6 +122,7 @@ public:
~CRainmeter(); ~CRainmeter();
int Initialize(LPCWSTR szPath); int Initialize(LPCWSTR szPath);
bool IsAlreadyRunning();
int MessagePump(); int MessagePump();
void SetNetworkStatisticsTimer(); void SetNetworkStatisticsTimer();
@ -322,6 +323,7 @@ private:
HWND m_Window; HWND m_Window;
HANDLE m_Mutex;
HINSTANCE m_Instance; HINSTANCE m_Instance;
HMODULE m_ResourceInstance; HMODULE m_ResourceInstance;
LCID m_ResourceLCID; LCID m_ResourceLCID;