OSDN Git Service

Moved more functions into MUtils library, especially all the Qt initialization code...
[lamexp/LameXP.git] / src / Global_Win32.cpp
index bfd20ba..a1a74b9 100644 (file)
@@ -1,6 +1,6 @@
 ///////////////////////////////////////////////////////////////////////////////
 // LameXP - Audio Encoder Front-End
-// Copyright (C) 2004-2013 LoRd_MuldeR <MuldeR2@GMX.de>
+// Copyright (C) 2004-2014 LoRd_MuldeR <MuldeR2@GMX.de>
 //
 // This program is free software; you can redistribute it and/or modify
 // it under the terms of the GNU General Public License as published by
@@ -51,7 +51,6 @@
 #include <QLibraryInfo>
 #include <QMap>
 #include <QMessageBox>
-#include <QMutex>
 #include <QPlastiqueStyle>
 #include <QProcess>
 #include <QReadWriteLock>
 #include <QTimer>
 #include <QTranslator>
 #include <QUuid>
+#include <QResource>
 
 //LameXP includes
 #define LAMEXP_INC_CONFIG
 #include "Resource.h"
 #include "Config.h"
 
+//MUtils
+#include <MUtils/Global.h>
+#include <MUtils/OSSupport.h>
+#include <MUtils/Terminal.h>
+
 //CRT includes
 #include <cstdio>
 #include <iostream>
@@ -97,60 +102,10 @@ Q_IMPORT_PLUGIN(QICOPlugin)
 #define _LAMEXP_MAKE_STR(STR) #STR
 #define LAMEXP_MAKE_STR(STR) _LAMEXP_MAKE_STR(STR)
 
-//String helper
-#define CLEAN_OUTPUT_STRING(STR) do \
-{ \
-       const char CTRL_CHARS[3] = { '\r', '\n', '\t' }; \
-       for(size_t i = 0; i < 3; i++) \
-       { \
-               while(char *pos = strchr((STR), CTRL_CHARS[i])) *pos = char(0x20); \
-       } \
-} \
-while(0)
-
-//String helper
-#define TRIM_LEFT(STR) do \
-{ \
-       const char WHITE_SPACE[4] = { char(0x20), '\r', '\n', '\t' }; \
-       for(size_t i = 0; i < 4; i++) \
-       { \
-               while(*(STR) == WHITE_SPACE[i]) (STR)++; \
-       } \
-} \
-while(0)
-
 ///////////////////////////////////////////////////////////////////////////////
 // GLOBAL VARS
 ///////////////////////////////////////////////////////////////////////////////
 
-//Console attached flag
-static bool g_lamexp_console_attached = false;
-
-//Special folders
-static struct
-{
-       QMap<size_t, QString> *knownFolders;
-       QReadWriteLock lock;
-}
-g_lamexp_known_folder;
-
-//CLI Arguments
-static struct
-{
-       QStringList *list;
-       QReadWriteLock lock;
-}
-g_lamexp_argv;
-
-//OS Version
-static struct
-{
-       bool bInitialized;
-       lamexp_os_version_t version;
-       QReadWriteLock lock;
-}
-g_lamexp_os_version;
-
 //Wine detection
 static struct
 {
@@ -161,1158 +116,113 @@ static struct
 g_lamexp_wine;
 
 //Win32 Theme support
-static struct
-{
-       bool bInitialized;
-       bool bThemesEnabled;
-       QReadWriteLock lock;
-}
-g_lamexp_themes_enabled;
-
-//Win32 DWM API functions
-static struct
-{
-       bool bInitialized;
-       HRESULT (__stdcall *dwmIsCompositionEnabled)(BOOL *bEnabled);
-       HRESULT (__stdcall *dwmExtendFrameIntoClientArea)(HWND hWnd, const MARGINS* pMarInset);
-       HRESULT (__stdcall *dwmEnableBlurBehindWindow)(HWND hWnd, const DWM_BLURBEHIND* pBlurBehind);
-       QLibrary *dwmapi_dll;
-       QReadWriteLock lock;
-}
-g_lamexp_dwmapi;
-
-//Image formats
-static const char *g_lamexp_imageformats[] = {"bmp", "png", "jpg", "gif", "ico", "xpm", NULL}; //"svg"
-
-//Global locks
-static QMutex g_lamexp_message_mutex;
-
-//Main thread ID
-static const DWORD g_main_thread_id = GetCurrentThreadId();
-
-//Log file
-static FILE *g_lamexp_log_file = NULL;
-
-//Localization
-const char* LAMEXP_DEFAULT_LANGID = "en";
-const char* LAMEXP_DEFAULT_TRANSLATION = "LameXP_EN.qm";
-
-//Known Windows versions - maps marketing names to the actual Windows NT versions
-const lamexp_os_version_t lamexp_winver_win2k = {5,0};
-const lamexp_os_version_t lamexp_winver_winxp = {5,1};
-const lamexp_os_version_t lamexp_winver_xpx64 = {5,2};
-const lamexp_os_version_t lamexp_winver_vista = {6,0};
-const lamexp_os_version_t lamexp_winver_win70 = {6,1};
-const lamexp_os_version_t lamexp_winver_win80 = {6,2};
-const lamexp_os_version_t lamexp_winver_win81 = {6,3};
-
-//GURU MEDITATION
-static const char *GURU_MEDITATION = "\n\nGURU MEDITATION !!!\n\n";
-
-///////////////////////////////////////////////////////////////////////////////
-// GLOBAL FUNCTIONS
-///////////////////////////////////////////////////////////////////////////////
-
-/*
- * Verify a specific Windows version
- */
-static bool lamexp_verify_os_version(const DWORD major, const DWORD minor)
-{
-       OSVERSIONINFOEXW osvi;
-       DWORDLONG dwlConditionMask = 0;
-
-       //Initialize the OSVERSIONINFOEX structure
-       memset(&osvi, 0, sizeof(OSVERSIONINFOEXW));
-       osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEXW);
-       osvi.dwMajorVersion = major;
-       osvi.dwMinorVersion = minor;
-       osvi.dwPlatformId = VER_PLATFORM_WIN32_NT;
-
-       //Initialize the condition mask
-       VER_SET_CONDITION(dwlConditionMask, VER_MAJORVERSION, VER_GREATER_EQUAL);
-       VER_SET_CONDITION(dwlConditionMask, VER_MINORVERSION, VER_GREATER_EQUAL);
-       VER_SET_CONDITION(dwlConditionMask, VER_PLATFORMID, VER_EQUAL);
-
-       // Perform the test
-       const BOOL ret = VerifyVersionInfoW(&osvi, VER_MAJORVERSION | VER_MINORVERSION | VER_PLATFORMID, dwlConditionMask);
-
-       //Error checking
-       if(!ret)
-       {
-               if(GetLastError() != ERROR_OLD_WIN_VERSION)
-               {
-                       qWarning("VerifyVersionInfo() system call has failed!");
-               }
-       }
-
-       return (ret != FALSE);
-}
-
-/*
- * Determine the *real* Windows version
- */
-static bool lamexp_get_real_os_version(unsigned int *major, unsigned int *minor, bool *pbOverride)
-{
-       *major = *minor = 0;
-       *pbOverride = false;
-       
-       //Initialize local variables
-       OSVERSIONINFOEXW osvi;
-       memset(&osvi, 0, sizeof(OSVERSIONINFOEXW));
-       osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFOEXW);
-
-       //Try GetVersionEx() first
-       if(GetVersionExW((LPOSVERSIONINFOW)&osvi) == FALSE)
-       {
-               qWarning("GetVersionEx() has failed, cannot detect Windows version!");
-               return false;
-       }
-
-       //Make sure we are running on NT
-       if(osvi.dwPlatformId == VER_PLATFORM_WIN32_NT)
-       {
-               *major = osvi.dwMajorVersion;
-               *minor = osvi.dwMinorVersion;
-       }
-       else
-       {
-               qWarning("Not running on Windows NT, unsupported operating system!");
-               return false;
-       }
-
-       //Determine the real *major* version first
-       forever
-       {
-               const DWORD nextMajor = (*major) + 1;
-               if(lamexp_verify_os_version(nextMajor, 0))
-               {
-                       *pbOverride = true;
-                       *major = nextMajor;
-                       *minor = 0;
-                       continue;
-               }
-               break;
-       }
-
-       //Now also determine the real *minor* version
-       forever
-       {
-               const DWORD nextMinor = (*minor) + 1;
-               if(lamexp_verify_os_version((*major), nextMinor))
-               {
-                       *pbOverride = true;
-                       *minor = nextMinor;
-                       continue;
-               }
-               break;
-       }
-
-       return true;
-}
-
-/*
- * Get the native operating system version
- */
-const lamexp_os_version_t &lamexp_get_os_version(void)
-{
-       QReadLocker readLock(&g_lamexp_os_version.lock);
-
-       //Already initialized?
-       if(g_lamexp_os_version.bInitialized)
-       {
-               return g_lamexp_os_version.version;
-       }
-       
-       readLock.unlock();
-       QWriteLocker writeLock(&g_lamexp_os_version.lock);
-
-       //Detect OS version
-       if(!g_lamexp_os_version.bInitialized)
-       {
-               unsigned int major, minor; bool oflag;
-               if(lamexp_get_real_os_version(&major, &minor, &oflag))
-               {
-                       g_lamexp_os_version.version.versionMajor = major;
-                       g_lamexp_os_version.version.versionMinor = minor;
-                       g_lamexp_os_version.version.overrideFlag = oflag;
-                       g_lamexp_os_version.bInitialized = true;
-               }
-               else
-               {
-                       qWarning("Failed to determin the operating system version!");
-               }
-       }
-
-       return g_lamexp_os_version.version;
-}
-
-/*
- * Check if we are running under wine
- */
-bool lamexp_detect_wine(void)
-{
-       QReadLocker readLock(&g_lamexp_wine.lock);
-
-       //Already initialized?
-       if(g_lamexp_wine.bInitialized)
-       {
-               return g_lamexp_wine.bIsWine;
-       }
-       
-       readLock.unlock();
-       QWriteLocker writeLock(&g_lamexp_wine.lock);
-
-       if(!g_lamexp_wine.bInitialized)
-       {
-               g_lamexp_wine.bIsWine = false;
-               QLibrary ntdll("ntdll.dll");
-               if(ntdll.load())
-               {
-                       if(ntdll.resolve("wine_nt_to_unix_file_name") != NULL) g_lamexp_wine.bIsWine = true;
-                       if(ntdll.resolve("wine_get_version") != NULL) g_lamexp_wine.bIsWine = true;
-                       ntdll.unload();
-               }
-               g_lamexp_wine.bInitialized = true;
-       }
-
-       return g_lamexp_wine.bIsWine;
-}
-
-/*
- * Change console text color
- */
-static void lamexp_console_color(FILE* file, WORD attributes)
-{
-       const HANDLE hConsole = (HANDLE)(_get_osfhandle(_fileno(file)));
-       if((hConsole != NULL) && (hConsole != INVALID_HANDLE_VALUE))
-       {
-               SetConsoleTextAttribute(hConsole, attributes);
-       }
-}
-
-/*
- * Output logging message to console
- */
-static void lamexp_write_console(const int type, const char *msg)
-{      
-       __try
-       {
-               if(_isatty(_fileno(stderr)))
-               {
-                       UINT oldOutputCP = GetConsoleOutputCP();
-                       if(oldOutputCP != CP_UTF8) SetConsoleOutputCP(CP_UTF8);
-
-                       switch(type)
-                       {
-                       case QtCriticalMsg:
-                       case QtFatalMsg:
-                               lamexp_console_color(stderr, FOREGROUND_RED | FOREGROUND_INTENSITY);
-                               fprintf(stderr, GURU_MEDITATION);
-                               fprintf(stderr, "%s\n", msg);
-                               fflush(stderr);
-                               break;
-                       case QtWarningMsg:
-                               lamexp_console_color(stderr, FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_INTENSITY);
-                               fprintf(stderr, "%s\n", msg);
-                               fflush(stderr);
-                               break;
-                       default:
-                               lamexp_console_color(stderr, FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_INTENSITY);
-                               fprintf(stderr, "%s\n", msg);
-                               fflush(stderr);
-                               break;
-                       }
-       
-                       lamexp_console_color(stderr, FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED);
-                       if(oldOutputCP != CP_UTF8) SetConsoleOutputCP(oldOutputCP);
-               }
-       }
-       __except(1)
-       {
-               /*ignore any exception that might occur here!*/
-       }
-}
-
-/*
- * Output logging message to debugger
- */
-static void lamexp_write_dbg_out(const int type, const char *msg)
-{      
-       const char *FORMAT = "[LameXP][%c] %s\n";
-
-       __try
-       {
-               char buffer[512];
-               const char* input = msg;
-               TRIM_LEFT(input);
-               
-               switch(type)
-               {
-               case QtCriticalMsg:
-               case QtFatalMsg:
-                       _snprintf_s(buffer, 512, _TRUNCATE, FORMAT, 'C', input);
-                       break;
-               case QtWarningMsg:
-                       _snprintf_s(buffer, 512, _TRUNCATE, FORMAT, 'W', input);
-                       break;
-               default:
-                       _snprintf_s(buffer, 512, _TRUNCATE, FORMAT, 'I', input);
-                       break;
-               }
-
-               char *temp = &buffer[0];
-               CLEAN_OUTPUT_STRING(temp);
-               OutputDebugStringA(temp);
-       }
-       __except(1)
-       {
-               /*ignore any exception that might occur here!*/
-       }
-}
-
-/*
- * Output logging message to logfile
- */
-static void lamexp_write_logfile(const int type, const char *msg)
-{      
-       const char *FORMAT = "[%c][%04u] %s\r\n";
-
-       __try
-       {
-               if(g_lamexp_log_file)
-               {
-                       char buffer[512];
-                       strncpy_s(buffer, 512, msg, _TRUNCATE);
-
-                       char *temp = &buffer[0];
-                       TRIM_LEFT(temp);
-                       CLEAN_OUTPUT_STRING(temp);
-                       
-                       const unsigned int timestamp = static_cast<unsigned int>(_time64(NULL) % 3600I64);
-
-                       switch(type)
-                       {
-                       case QtCriticalMsg:
-                       case QtFatalMsg:
-                               fprintf(g_lamexp_log_file, FORMAT, 'C', timestamp, temp);
-                               break;
-                       case QtWarningMsg:
-                               fprintf(g_lamexp_log_file, FORMAT, 'W', timestamp, temp);
-                               break;
-                       default:
-                               fprintf(g_lamexp_log_file, FORMAT, 'I', timestamp, temp);
-                               break;
-                       }
-
-                       fflush(g_lamexp_log_file);
-               }
-       }
-       __except(1)
-       {
-               /*ignore any exception that might occur here!*/
-       }
-}
-
-/*
- * Qt message handler
- */
-void lamexp_message_handler(QtMsgType type, const char *msg)
-{
-       if((!msg) || (!(msg[0])))
-       {
-               return;
-       }
-
-       QMutexLocker lock(&g_lamexp_message_mutex);
-
-       if(g_lamexp_log_file)
-       {
-               lamexp_write_logfile(type, msg);
-       }
-
-       if(g_lamexp_console_attached)
-       {
-               lamexp_write_console(type, msg);
-       }
-       else
-       {
-               lamexp_write_dbg_out(type, msg);
-       }
-
-       if((type == QtCriticalMsg) || (type == QtFatalMsg))
-       {
-               lock.unlock();
-               lamexp_fatal_exit(L"The application has encountered a critical error and will exit now!", QWCHAR(QString::fromUtf8(msg)));
-       }
-}
-
-/*
- * Invalid parameters handler
- */
-static void lamexp_invalid_param_handler(const wchar_t* exp, const wchar_t* fun, const wchar_t* fil, unsigned int, uintptr_t)
-{
-       lamexp_fatal_exit(L"Invalid parameter handler invoked, application will exit!");
-}
-
-/*
- * Signal handler
- */
-static void lamexp_signal_handler(int signal_num)
-{
-       signal(signal_num, lamexp_signal_handler);
-       lamexp_fatal_exit(L"Signal handler invoked, application will exit!");
-}
-
-/*
- * Global exception handler
- */
-static LONG WINAPI lamexp_exception_handler(struct _EXCEPTION_POINTERS *ExceptionInfo)
-{
-       lamexp_fatal_exit(L"Unhandeled exception handler invoked, application will exit!");
-       return LONG_MAX;
-}
-
-/*
- * Initialize error handlers
- */
-void lamexp_init_error_handlers(void)
-{
-       SetErrorMode(SEM_FAILCRITICALERRORS | SEM_NOOPENFILEERRORBOX);
-       SetUnhandledExceptionFilter(lamexp_exception_handler);
-       SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_ABOVE_NORMAL);
-       _set_invalid_parameter_handler(lamexp_invalid_param_handler);
-       
-       static const int signal_num[6] = { SIGABRT, SIGFPE, SIGILL, SIGINT, SIGSEGV, SIGTERM };
-
-       for(size_t i = 0; i < 6; i++)
-       {
-               signal(signal_num[i], lamexp_signal_handler);
-       }
-}
-
-/*
- * Initialize the console
- */
-void lamexp_init_console(const QStringList &argv)
-{
-       bool enableConsole = (LAMEXP_DEBUG) || ((VER_LAMEXP_CONSOLE_ENABLED) && lamexp_version_demo());
-
-       if(_environ)
-       {
-               wchar_t *logfile = NULL;
-               size_t logfile_len = 0;
-               if(!_wdupenv_s(&logfile, &logfile_len, L"LAMEXP_LOGFILE"))
-               {
-                       if(logfile && (logfile_len > 0))
-                       {
-                               FILE *temp = NULL;
-                               if(!_wfopen_s(&temp, logfile, L"wb"))
-                               {
-                                       fprintf(temp, "%c%c%c", char(0xEF), char(0xBB), char(0xBF));
-                                       g_lamexp_log_file = temp;
-                               }
-                               free(logfile);
-                       }
-               }
-       }
-
-       if(!LAMEXP_DEBUG)
-       {
-               for(int i = 0; i < argv.count(); i++)
-               {
-                       if(!argv.at(i).compare("--console", Qt::CaseInsensitive))
-                       {
-                               enableConsole = true;
-                       }
-                       else if(!argv.at(i).compare("--no-console", Qt::CaseInsensitive))
-                       {
-                               enableConsole = false;
-                       }
-               }
-       }
-
-       if(enableConsole)
-       {
-               if(!g_lamexp_console_attached)
-               {
-                       if(AllocConsole() != FALSE)
-                       {
-                               SetConsoleCtrlHandler(NULL, TRUE);
-                               SetConsoleTitle(L"LameXP - Audio Encoder Front-End | Debug Console");
-                               SetConsoleOutputCP(CP_UTF8);
-                               g_lamexp_console_attached = true;
-                       }
-               }
-               
-               if(g_lamexp_console_attached)
-               {
-                       //-------------------------------------------------------------------
-                       //See: http://support.microsoft.com/default.aspx?scid=kb;en-us;105305
-                       //-------------------------------------------------------------------
-                       const int flags = _O_WRONLY | _O_U8TEXT;
-                       int hCrtStdOut = _open_osfhandle((intptr_t) GetStdHandle(STD_OUTPUT_HANDLE), flags);
-                       int hCrtStdErr = _open_osfhandle((intptr_t) GetStdHandle(STD_ERROR_HANDLE),  flags);
-                       FILE *hfStdOut = (hCrtStdOut >= 0) ? _fdopen(hCrtStdOut, "wb") : NULL;
-                       FILE *hfStdErr = (hCrtStdErr >= 0) ? _fdopen(hCrtStdErr, "wb") : NULL;
-                       if(hfStdOut) { *stdout = *hfStdOut; std::cout.rdbuf(new std::filebuf(hfStdOut)); }
-                       if(hfStdErr) { *stderr = *hfStdErr; std::cerr.rdbuf(new std::filebuf(hfStdErr)); }
-               }
-
-               HWND hwndConsole = GetConsoleWindow();
-
-               if((hwndConsole != NULL) && (hwndConsole != INVALID_HANDLE_VALUE))
-               {
-                       HMENU hMenu = GetSystemMenu(hwndConsole, 0);
-                       EnableMenuItem(hMenu, SC_CLOSE, MF_BYCOMMAND | MF_GRAYED);
-                       RemoveMenu(hMenu, SC_CLOSE, MF_BYCOMMAND);
-
-                       SetWindowPos(hwndConsole, HWND_TOP, 0, 0, 0, 0, SWP_NOMOVE|SWP_NOSIZE|SWP_NOZORDER|SWP_FRAMECHANGED);
-                       SetWindowLong(hwndConsole, GWL_STYLE, GetWindowLong(hwndConsole, GWL_STYLE) & (~WS_MAXIMIZEBOX) & (~WS_MINIMIZEBOX));
-                       SetWindowPos(hwndConsole, HWND_TOP, 0, 0, 0, 0, SWP_NOMOVE|SWP_NOSIZE|SWP_NOZORDER|SWP_FRAMECHANGED);
-               }
-       }
-}
-
-/*
- * Detect CPU features
- */
-lamexp_cpu_t lamexp_detect_cpu_features(const QStringList &argv)
-{
-       typedef BOOL (WINAPI *IsWow64ProcessFun)(__in HANDLE hProcess, __out PBOOL Wow64Process);
-
-       lamexp_cpu_t features;
-       SYSTEM_INFO systemInfo;
-       int CPUInfo[4] = {-1};
-       char CPUIdentificationString[0x40];
-       char CPUBrandString[0x40];
-
-       memset(&features, 0, sizeof(lamexp_cpu_t));
-       memset(&systemInfo, 0, sizeof(SYSTEM_INFO));
-       memset(CPUIdentificationString, 0, sizeof(CPUIdentificationString));
-       memset(CPUBrandString, 0, sizeof(CPUBrandString));
-       
-       __cpuid(CPUInfo, 0);
-       memcpy(CPUIdentificationString, &CPUInfo[1], sizeof(int));
-       memcpy(CPUIdentificationString + 4, &CPUInfo[3], sizeof(int));
-       memcpy(CPUIdentificationString + 8, &CPUInfo[2], sizeof(int));
-       features.intel = (_stricmp(CPUIdentificationString, "GenuineIntel") == 0);
-       strncpy_s(features.vendor, 0x40, CPUIdentificationString, _TRUNCATE);
-
-       if(CPUInfo[0] >= 1)
-       {
-               __cpuid(CPUInfo, 1);
-               features.mmx = (CPUInfo[3] & 0x800000) || false;
-               features.sse = (CPUInfo[3] & 0x2000000) || false;
-               features.sse2 = (CPUInfo[3] & 0x4000000) || false;
-               features.ssse3 = (CPUInfo[2] & 0x200) || false;
-               features.sse3 = (CPUInfo[2] & 0x1) || false;
-               features.ssse3 = (CPUInfo[2] & 0x200) || false;
-               features.stepping = CPUInfo[0] & 0xf;
-               features.model = ((CPUInfo[0] >> 4) & 0xf) + (((CPUInfo[0] >> 16) & 0xf) << 4);
-               features.family = ((CPUInfo[0] >> 8) & 0xf) + ((CPUInfo[0] >> 20) & 0xff);
-       }
-
-       __cpuid(CPUInfo, 0x80000000);
-       int nExIds = qMax<int>(qMin<int>(CPUInfo[0], 0x80000004), 0x80000000);
-
-       for(int i = 0x80000002; i <= nExIds; ++i)
-       {
-               __cpuid(CPUInfo, i);
-               switch(i)
-               {
-               case 0x80000002:
-                       memcpy(CPUBrandString, CPUInfo, sizeof(CPUInfo));
-                       break;
-               case 0x80000003:
-                       memcpy(CPUBrandString + 16, CPUInfo, sizeof(CPUInfo));
-                       break;
-               case 0x80000004:
-                       memcpy(CPUBrandString + 32, CPUInfo, sizeof(CPUInfo));
-                       break;
-               }
-       }
-
-       strncpy_s(features.brand, 0x40, CPUBrandString, _TRUNCATE);
-
-       if(strlen(features.brand) < 1) strncpy_s(features.brand, 0x40, "Unknown", _TRUNCATE);
-       if(strlen(features.vendor) < 1) strncpy_s(features.vendor, 0x40, "Unknown", _TRUNCATE);
-
-#if (!(defined(_M_X64) || defined(_M_IA64)))
-       QLibrary Kernel32Lib("kernel32.dll");
-       if(IsWow64ProcessFun IsWow64ProcessPtr = (IsWow64ProcessFun) Kernel32Lib.resolve("IsWow64Process"))
-       {
-               BOOL x64flag = FALSE;
-               if(IsWow64ProcessPtr(GetCurrentProcess(), &x64flag))
-               {
-                       features.x64 = (x64flag == TRUE);
-               }
-       }
-#else
-       features.x64 = true;
-#endif
-
-       DWORD_PTR procAffinity, sysAffinity;
-       if(GetProcessAffinityMask(GetCurrentProcess(), &procAffinity, &sysAffinity))
-       {
-               for(DWORD_PTR mask = 1; mask; mask <<= 1)
-               {
-                       features.count += ((sysAffinity & mask) ? (1) : (0));
-               }
-       }
-       if(features.count < 1)
-       {
-               GetNativeSystemInfo(&systemInfo);
-               features.count = qBound(1UL, systemInfo.dwNumberOfProcessors, 64UL);
-       }
-
-       if(argv.count() > 0)
-       {
-               bool flag = false;
-               for(int i = 0; i < argv.count(); i++)
-               {
-                       if(!argv[i].compare("--force-cpu-no-64bit", Qt::CaseInsensitive)) { flag = true; features.x64 = false; }
-                       if(!argv[i].compare("--force-cpu-no-sse", Qt::CaseInsensitive)) { flag = true; features.sse = features.sse2 = features.sse3 = features.ssse3 = false; }
-                       if(!argv[i].compare("--force-cpu-no-intel", Qt::CaseInsensitive)) { flag = true; features.intel = false; }
-               }
-               if(flag) qWarning("CPU flags overwritten by user-defined parameters. Take care!\n");
-       }
-
-       return features;
-}
-
-/*
- * Check for debugger (detect routine)
- */
-static __forceinline bool lamexp_check_for_debugger(void)
-{
-       __try
-       {
-               CloseHandle((HANDLE)((DWORD_PTR)-3));
-       }
-       __except(1)
-       {
-               return true;
-       }
-       __try 
-       {
-               __debugbreak();
-       }
-       __except(1) 
-       {
-               return IsDebuggerPresent();
-       }
-       return true;
-}
-
-/*
- * Check for debugger (thread proc)
- */
-static unsigned int __stdcall lamexp_debug_thread_proc(LPVOID lpParameter)
-{
-       SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST);
-       forever
-       {
-               if(lamexp_check_for_debugger())
-               {
-                       lamexp_fatal_exit(L"Not a debug build. Please unload debugger and try again!");
-                       return 666;
-               }
-               lamexp_sleep(100);
-       }
-}
-
-/*
- * Check for debugger (startup routine)
- */
-static HANDLE lamexp_debug_thread_init()
-{
-       if(lamexp_check_for_debugger())
-       {
-               lamexp_fatal_exit(L"Not a debug build. Please unload debugger and try again!");
-       }
-       const uintptr_t h = _beginthreadex(NULL, 0, lamexp_debug_thread_proc, NULL, 0, NULL);
-       return (HANDLE)(h^0xdeadbeef);
-}
-
-/*
- * Qt event filter
- */
-static bool lamexp_event_filter(void *message, long *result)
-{ 
-       if((!(LAMEXP_DEBUG)) && lamexp_check_for_debugger())
-       {
-               lamexp_fatal_exit(L"Not a debug build. Please unload debugger and try again!");
-       }
-       
-       switch(reinterpret_cast<MSG*>(message)->message)
-       {
-       case WM_QUERYENDSESSION:
-               qWarning("WM_QUERYENDSESSION message received!");
-               *result = lamexp_broadcast(lamexp_event_queryendsession, false) ? TRUE : FALSE;
-               return true;
-       case WM_ENDSESSION:
-               qWarning("WM_ENDSESSION message received!");
-               if(reinterpret_cast<MSG*>(message)->wParam == TRUE)
-               {
-                       lamexp_broadcast(lamexp_event_endsession, false);
-                       if(QApplication *app = reinterpret_cast<QApplication*>(QApplication::instance()))
-                       {
-                               app->closeAllWindows();
-                               app->quit();
-                       }
-                       lamexp_finalization();
-                       exit(1);
-               }
-               *result = 0;
-               return true;
-       default:
-               /*ignore this message and let Qt handle it*/
-               return false;
-       }
-}
-
-/*
- * Check for process elevation
- */
-static bool lamexp_check_elevation(void)
-{
-       typedef enum { lamexp_token_elevationType_class = 18, lamexp_token_elevation_class = 20 } LAMEXP_TOKEN_INFORMATION_CLASS;
-       typedef enum { lamexp_elevationType_default = 1, lamexp_elevationType_full, lamexp_elevationType_limited } LAMEXP_TOKEN_ELEVATION_TYPE;
-
-       HANDLE hToken = NULL;
-       bool bIsProcessElevated = false;
-       
-       if(OpenProcessToken(GetCurrentProcess(), TOKEN_QUERY, &hToken))
-       {
-               LAMEXP_TOKEN_ELEVATION_TYPE tokenElevationType;
-               DWORD returnLength;
-               if(GetTokenInformation(hToken, (TOKEN_INFORMATION_CLASS) lamexp_token_elevationType_class, &tokenElevationType, sizeof(LAMEXP_TOKEN_ELEVATION_TYPE), &returnLength))
-               {
-                       if(returnLength == sizeof(LAMEXP_TOKEN_ELEVATION_TYPE))
-                       {
-                               switch(tokenElevationType)
-                               {
-                               case lamexp_elevationType_default:
-                                       qDebug("Process token elevation type: Default -> UAC is disabled.\n");
-                                       break;
-                               case lamexp_elevationType_full:
-                                       qWarning("Process token elevation type: Full -> potential security risk!\n");
-                                       bIsProcessElevated = true;
-                                       break;
-                               case lamexp_elevationType_limited:
-                                       qDebug("Process token elevation type: Limited -> not elevated.\n");
-                                       break;
-                               }
-                       }
-               }
-               CloseHandle(hToken);
-       }
-       else
-       {
-               qWarning("Failed to open process token!");
-       }
-
-       return !bIsProcessElevated;
-}
-
-/*
- * Convert QIcon to HICON -> caller is responsible for destroying the HICON!
- */
-static HICON lamexp_qicon2hicon(const QIcon &icon, const int w, const int h)
-{
-       if(!icon.isNull())
-       {
-               QPixmap pixmap = icon.pixmap(w, h);
-               if(!pixmap.isNull())
-               {
-                       return pixmap.toWinHICON();
-               }
-       }
-       return NULL;
-}
-
-/*
- * Initialize Qt framework
- */
-bool lamexp_init_qt(int argc, char* argv[])
-{
-       static bool qt_initialized = false;
-       typedef BOOL (WINAPI *SetDllDirectoryProc)(WCHAR *lpPathName);
-       const QStringList &arguments = lamexp_arguments();
-
-       //Don't initialized again, if done already
-       if(qt_initialized)
-       {
-               return true;
-       }
-
-       //Secure DLL loading
-       QLibrary kernel32("kernel32.dll");
-       if(kernel32.load())
-       {
-               SetDllDirectoryProc pSetDllDirectory = (SetDllDirectoryProc) kernel32.resolve("SetDllDirectoryW");
-               if(pSetDllDirectory != NULL) pSetDllDirectory(L"");
-       }
-
-       //Extract executable name from argv[] array
-       QString executableName = QLatin1String("LameXP.exe");
-       if(arguments.count() > 0)
-       {
-               static const char *delimiters = "\\/:?";
-               executableName = arguments[0].trimmed();
-               for(int i = 0; delimiters[i]; i++)
-               {
-                       int temp = executableName.lastIndexOf(QChar(delimiters[i]));
-                       if(temp >= 0) executableName = executableName.mid(temp + 1);
-               }
-               executableName = executableName.trimmed();
-               if(executableName.isEmpty())
-               {
-                       executableName = QLatin1String("LameXP.exe");
-               }
-       }
-
-       //Check Qt version
-#ifdef QT_BUILD_KEY
-       qDebug("Using Qt v%s [%s], %s, %s", qVersion(), QLibraryInfo::buildDate().toString(Qt::ISODate).toLatin1().constData(), (qSharedBuild() ? "DLL" : "Static"), QLibraryInfo::buildKey().toLatin1().constData());
-       qDebug("Compiled with Qt v%s [%s], %s\n", QT_VERSION_STR, QT_PACKAGEDATE_STR, QT_BUILD_KEY);
-       if(_stricmp(qVersion(), QT_VERSION_STR))
-       {
-               qFatal("%s", QApplication::tr("Executable '%1' requires Qt v%2, but found Qt v%3.").arg(executableName, QString::fromLatin1(QT_VERSION_STR), QString::fromLatin1(qVersion())).toLatin1().constData());
-               return false;
-       }
-       if(QLibraryInfo::buildKey().compare(QString::fromLatin1(QT_BUILD_KEY), Qt::CaseInsensitive))
-       {
-               qFatal("%s", QApplication::tr("Executable '%1' was built for Qt '%2', but found Qt '%3'.").arg(executableName, QString::fromLatin1(QT_BUILD_KEY), QLibraryInfo::buildKey()).toLatin1().constData());
-               return false;
-       }
-#else
-       qDebug("Using Qt v%s [%s], %s", qVersion(), QLibraryInfo::buildDate().toString(Qt::ISODate).toLatin1().constData(), (qSharedBuild() ? "DLL" : "Static"));
-       qDebug("Compiled with Qt v%s [%s]\n", QT_VERSION_STR, QT_PACKAGEDATE_STR);
-#endif
-
-       //Check the Windows version
-       const lamexp_os_version_t &osVersionNo = lamexp_get_os_version();
-       if(osVersionNo < lamexp_winver_winxp)
-       {
-               qFatal("%s", QApplication::tr("Executable '%1' requires Windows XP or later.").arg(executableName).toLatin1().constData());
-       }
-
-       //Supported Windows version?
-       if(osVersionNo == lamexp_winver_winxp)
-       {
-               qDebug("Running on Windows XP or Windows XP Media Center Edition.\n");                                          //lamexp_check_compatibility_mode("GetLargePageMinimum", executableName);
-       }
-       else if(osVersionNo == lamexp_winver_xpx64)
-       {
-               qDebug("Running on Windows Server 2003, Windows Server 2003 R2 or Windows XP x64.\n");          //lamexp_check_compatibility_mode("GetLocaleInfoEx", executableName);
-       }
-       else if(osVersionNo == lamexp_winver_vista)
-       {
-               qDebug("Running on Windows Vista or Windows Server 2008.\n");                                                           //lamexp_check_compatibility_mode("CreateRemoteThreadEx", executableName*/);
-       }
-       else if(osVersionNo == lamexp_winver_win70)
-       {
-               qDebug("Running on Windows 7 or Windows Server 2008 R2.\n");                                                            //lamexp_check_compatibility_mode("CreateFile2", executableName);
-       }
-       else if(osVersionNo == lamexp_winver_win80)
-       {
-               qDebug("Running on Windows 8 or Windows Server 2012.\n");                                                                       //lamexp_check_compatibility_mode("FindPackagesByPackageFamily", executableName);
-       }
-       else if(osVersionNo == lamexp_winver_win81)
-       {
-               qDebug("Running on Windows 8.1 or Windows Server 2012 R2.\n");                                                          //lamexp_check_compatibility_mode(NULL, executableName);
-       }
-       else
-       {
-               const QString message = QString().sprintf("Running on an unknown WindowsNT-based system (v%u.%u).", osVersionNo.versionMajor, osVersionNo.versionMinor);
-               qWarning("%s\n", QUTF8(message));
-               MessageBoxW(NULL, QWCHAR(message), L"LameXP", MB_OK | MB_TOPMOST | MB_ICONWARNING);
-       }
-
-       //Check for compat mode
-       if(osVersionNo.overrideFlag && (osVersionNo <= lamexp_winver_win81))
-       {
-               qWarning("Windows compatibility mode detected!");
-               if(!arguments.contains("--ignore-compat-mode", Qt::CaseInsensitive))
-               {
-                       qFatal("%s", QApplication::tr("Executable '%1' doesn't support Windows compatibility mode.").arg(executableName).toLatin1().constData());
-                       return false;
-               }
-       }
-
-       //Check for Wine
-       if(lamexp_detect_wine())
-       {
-               qWarning("It appears we are running under Wine, unexpected things might happen!\n");
-       }
-
-       //Set text Codec for locale
-       QTextCodec::setCodecForLocale(QTextCodec::codecForName("UTF-8"));
-
-       //Create Qt application instance
-       QApplication *application = new QApplication(argc, argv);
-
-       //Load plugins from application directory
-       QCoreApplication::setLibraryPaths(QStringList() << QApplication::applicationDirPath());
-       qDebug("Library Path:\n%s\n", QUTF8(QApplication::libraryPaths().first()));
-
-       //Set application properties
-       application->setApplicationName("LameXP - Audio Encoder Front-End");
-       application->setApplicationVersion(QString().sprintf("%d.%02d.%04d", lamexp_version_major(), lamexp_version_minor(), lamexp_version_build())); 
-       application->setOrganizationName("LoRd_MuldeR");
-       application->setOrganizationDomain("mulder.at.gg");
-       application->setWindowIcon(lamexp_app_icon());
-       application->setEventFilter(lamexp_event_filter);
-
-       //Check for supported image formats
-       QList<QByteArray> supportedFormats = QImageReader::supportedImageFormats();
-       for(int i = 0; g_lamexp_imageformats[i]; i++)
-       {
-               if(!supportedFormats.contains(g_lamexp_imageformats[i]))
-               {
-                       qFatal("Qt initialization error: QImageIOHandler for '%s' missing!", g_lamexp_imageformats[i]);
-                       return false;
-               }
-       }
-       
-       //Add the default translations
-       lamexp_translation_init();
-
-       //Check for process elevation
-       if((!lamexp_check_elevation()) && (!lamexp_detect_wine()))
-       {
-               QMessageBox messageBox(QMessageBox::Warning, "LameXP", "<nobr>LameXP was started with 'elevated' rights, altough LameXP does not need these rights.<br>Running an applications with unnecessary rights is a potential security risk!</nobr>", QMessageBox::NoButton, NULL, Qt::Dialog | Qt::MSWindowsFixedSizeDialogHint | Qt::WindowStaysOnTopHint);
-               messageBox.addButton("Quit Program (Recommended)", QMessageBox::NoRole);
-               messageBox.addButton("Ignore", QMessageBox::NoRole);
-               if(messageBox.exec() == 0)
-               {
-                       return false;
-               }
-       }
-
-       //Update console icon, if a console is attached
-#if QT_VERSION < QT_VERSION_CHECK(5,0,0)
-       if(g_lamexp_console_attached && (!lamexp_detect_wine()))
-       {
-               QLibrary kernel32("kernel32.dll");
-               if(kernel32.load())
-               {
-                       typedef DWORD (__stdcall *SetConsoleIconFun)(HICON);
-                       if(SetConsoleIconFun SetConsoleIconPtr = (SetConsoleIconFun) kernel32.resolve("SetConsoleIcon"))
-                       {
-                               if(HICON hIcon = lamexp_qicon2hicon(QIcon(":/icons/sound.png"), 16, 16))
-                               {
-                                       SetConsoleIconPtr(hIcon);
-                                       DestroyIcon(hIcon);
-                               }
-                       }
-               }
-       }
-#endif
-
-       //Done
-       qt_initialized = true;
-       return true;
+static struct
+{
+       bool bInitialized;
+       bool bThemesEnabled;
+       QReadWriteLock lock;
 }
+g_lamexp_themes_enabled;
 
-const QStringList &lamexp_arguments(void)
+//Win32 DWM API functions
+static struct
 {
-       QReadLocker readLock(&g_lamexp_argv.lock);
-
-       if(!g_lamexp_argv.list)
-       {
-               readLock.unlock();
-               QWriteLocker writeLock(&g_lamexp_argv.lock);
+       bool bInitialized;
+       HRESULT (__stdcall *dwmIsCompositionEnabled)(BOOL *bEnabled);
+       HRESULT (__stdcall *dwmExtendFrameIntoClientArea)(HWND hWnd, const MARGINS* pMarInset);
+       HRESULT (__stdcall *dwmEnableBlurBehindWindow)(HWND hWnd, const DWM_BLURBEHIND* pBlurBehind);
+       QLibrary *dwmapi_dll;
+       QReadWriteLock lock;
+}
+g_lamexp_dwmapi;
 
-               g_lamexp_argv.list = new QStringList;
+//Sound file cache
+static struct
+{
+       QHash<const QString, const unsigned char*> *sound_db;
+       QReadWriteLock lock;
+}
+g_lamexp_sounds;
 
-               int nArgs = 0;
-               LPWSTR *szArglist = CommandLineToArgvW(GetCommandLineW(), &nArgs);
+//Main thread ID
+static const DWORD g_main_thread_id = GetCurrentThreadId();
 
-               if(NULL != szArglist)
-               {
-                       for(int i = 0; i < nArgs; i++)
-                       {
-                               (*g_lamexp_argv.list) << WCHAR2QSTR(szArglist[i]);
-                       }
-                       LocalFree(szArglist);
-               }
-               else
-               {
-                       qWarning("CommandLineToArgvW() has failed !!!");
-               }
-       }
+//Localization
+const char* LAMEXP_DEFAULT_LANGID = "en";
+const char* LAMEXP_DEFAULT_TRANSLATION = "LameXP_EN.qm";
 
-       return (*g_lamexp_argv.list);
-}
+///////////////////////////////////////////////////////////////////////////////
+// GLOBAL FUNCTIONS
+///////////////////////////////////////////////////////////////////////////////
 
 /*
- * Locate known folder on local system
+ * Check for debugger (detect routine)
  */
-const QString &lamexp_known_folder(lamexp_known_folder_t folder_id)
+static __forceinline bool lamexp_check_for_debugger(void)
 {
-       typedef HRESULT (WINAPI *SHGetKnownFolderPathFun)(__in const GUID &rfid, __in DWORD dwFlags, __in HANDLE hToken, __out PWSTR *ppszPath);
-       typedef HRESULT (WINAPI *SHGetFolderPathFun)(__in HWND hwndOwner, __in int nFolder, __in HANDLE hToken, __in DWORD dwFlags, __out LPWSTR pszPath);
-
-       static const int CSIDL_LOCAL_APPDATA = 0x001c;
-       static const int CSIDL_PROGRAM_FILES = 0x0026;
-       static const int CSIDL_SYSTEM_FOLDER = 0x0025;
-       static const GUID GUID_LOCAL_APPDATA = {0xF1B32785,0x6FBA,0x4FCF,{0x9D,0x55,0x7B,0x8E,0x7F,0x15,0x70,0x91}};
-       static const GUID GUID_LOCAL_APPDATA_LOW = {0xA520A1A4,0x1780,0x4FF6,{0xBD,0x18,0x16,0x73,0x43,0xC5,0xAF,0x16}};
-       static const GUID GUID_PROGRAM_FILES = {0x905e63b6,0xc1bf,0x494e,{0xb2,0x9c,0x65,0xb7,0x32,0xd3,0xd2,0x1a}};
-       static const GUID GUID_SYSTEM_FOLDER = {0x1AC14E77,0x02E7,0x4E5D,{0xB7,0x44,0x2E,0xB1,0xAE,0x51,0x98,0xB7}};
-
-       QReadLocker readLock(&g_lamexp_known_folder.lock);
-
-       int folderCSIDL = -1;
-       GUID folderGUID = {0x0000,0x0000,0x0000,{0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00}};
-       size_t folderCacheId = size_t(-1);
-
-       switch(folder_id)
-       {
-       case lamexp_folder_localappdata:
-               folderCacheId = 0;
-               folderCSIDL = CSIDL_LOCAL_APPDATA;
-               folderGUID = GUID_LOCAL_APPDATA;
-               break;
-       case lamexp_folder_programfiles:
-               folderCacheId = 1;
-               folderCSIDL = CSIDL_PROGRAM_FILES;
-               folderGUID = GUID_PROGRAM_FILES;
-               break;
-       case lamexp_folder_systemfolder:
-               folderCacheId = 2;
-               folderCSIDL = CSIDL_SYSTEM_FOLDER;
-               folderGUID = GUID_SYSTEM_FOLDER;
-               break;
-       default:
-               qWarning("Invalid 'known' folder was requested!");
-               return *reinterpret_cast<QString*>(NULL);
-               break;
-       }
-
-       //Already in cache?
-       if(g_lamexp_known_folder.knownFolders)
+       return false;
+       __try
        {
-               if(g_lamexp_known_folder.knownFolders->contains(folderCacheId))
-               {
-                       return (*g_lamexp_known_folder.knownFolders)[folderCacheId];
-               }
+               CloseHandle((HANDLE)((DWORD_PTR)-3));
        }
-
-       //Obtain write lock to initialize
-       readLock.unlock();
-       QWriteLocker writeLock(&g_lamexp_known_folder.lock);
-
-       //Still not in cache?
-       if(g_lamexp_known_folder.knownFolders)
+       __except(1)
        {
-               if(g_lamexp_known_folder.knownFolders->contains(folderCacheId))
-               {
-                       return (*g_lamexp_known_folder.knownFolders)[folderCacheId];
-               }
+               return true;
        }
 
-       static SHGetKnownFolderPathFun SHGetKnownFolderPathPtr = NULL;
-       static SHGetFolderPathFun SHGetFolderPathPtr = NULL;
-
-       //Lookup functions
-       if((!SHGetKnownFolderPathPtr) && (!SHGetFolderPathPtr))
+       BOOL bHaveDebugger = FALSE;
+       if(CheckRemoteDebuggerPresent(GetCurrentProcess(), &bHaveDebugger))
        {
-               QLibrary kernel32Lib("shell32.dll");
-               if(kernel32Lib.load())
-               {
-                       SHGetKnownFolderPathPtr = (SHGetKnownFolderPathFun) kernel32Lib.resolve("SHGetKnownFolderPath");
-                       SHGetFolderPathPtr = (SHGetFolderPathFun) kernel32Lib.resolve("SHGetFolderPathW");
-               }
+               if(bHaveDebugger) return true;
        }
 
-       QString folder;
+       return IsDebuggerPresent();
+}
 
-       //Now try to get the folder path!
-       if(SHGetKnownFolderPathPtr)
-       {
-               WCHAR *path = NULL;
-               if(SHGetKnownFolderPathPtr(folderGUID, 0x00008000, NULL, &path) == S_OK)
-               {
-                       //MessageBoxW(0, path, L"SHGetKnownFolderPath", MB_TOPMOST);
-                       QDir folderTemp = QDir(QDir::fromNativeSeparators(QString::fromUtf16(reinterpret_cast<const unsigned short*>(path))));
-                       if(!folderTemp.exists())
-                       {
-                               folderTemp.mkpath(".");
-                       }
-                       if(folderTemp.exists())
-                       {
-                               folder = folderTemp.canonicalPath();
-                       }
-                       CoTaskMemFree(path);
-               }
-       }
-       else if(SHGetFolderPathPtr)
+/*
+ * Check for debugger (thread proc)
+ */
+static unsigned int __stdcall lamexp_debug_thread_proc(LPVOID lpParameter)
+{
+       SetThreadPriority(GetCurrentThread(), THREAD_PRIORITY_LOWEST);
+       forever
        {
-               WCHAR *path = new WCHAR[4096];
-               if(SHGetFolderPathPtr(NULL, folderCSIDL, NULL, NULL, path) == S_OK)
+               if(lamexp_check_for_debugger())
                {
-                       //MessageBoxW(0, path, L"SHGetFolderPathW", MB_TOPMOST);
-                       QDir folderTemp = QDir(QDir::fromNativeSeparators(QString::fromUtf16(reinterpret_cast<const unsigned short*>(path))));
-                       if(!folderTemp.exists())
-                       {
-                               folderTemp.mkpath(".");
-                       }
-                       if(folderTemp.exists())
-                       {
-                               folder = folderTemp.canonicalPath();
-                       }
+                       MUtils::OS::fatal_exit(L"Not a debug build. Please unload debugger and try again!");
+                       return 666;
                }
-               delete [] path;
+               lamexp_sleep(100);
        }
+}
 
-       //Create cache
-       if(!g_lamexp_known_folder.knownFolders)
+/*
+ * Check for debugger (startup routine)
+ */
+static HANDLE lamexp_debug_thread_init()
+{
+       if(lamexp_check_for_debugger())
        {
-               g_lamexp_known_folder.knownFolders = new QMap<size_t, QString>();
+               MUtils::OS::fatal_exit(L"Not a debug build. Please unload debugger and try again!");
        }
-
-       //Update cache
-       g_lamexp_known_folder.knownFolders->insert(folderCacheId, folder);
-       return (*g_lamexp_known_folder.knownFolders)[folderCacheId];
+       const uintptr_t h = _beginthreadex(NULL, 0, lamexp_debug_thread_proc, NULL, 0, NULL);
+       return (HANDLE)(h^0xdeadbeef);
 }
 
 /*
- * Safely remove a file
+ * Convert QIcon to HICON -> caller is responsible for destroying the HICON!
  */
-bool lamexp_remove_file(const QString &filename)
+static HICON lamexp_qicon2hicon(const QIcon &icon, const int w, const int h)
 {
-       if(!QFileInfo(filename).exists() || !QFileInfo(filename).isFile())
-       {
-               return true;
-       }
-       else
+       if(!icon.isNull())
        {
-               if(!QFile::remove(filename))
-               {
-                       static const DWORD attrMask = FILE_ATTRIBUTE_READONLY | FILE_ATTRIBUTE_HIDDEN | FILE_ATTRIBUTE_SYSTEM;
-                       const DWORD attributes = GetFileAttributesW(QWCHAR(filename));
-                       if(attributes & attrMask)
-                       {
-                               SetFileAttributesW(QWCHAR(filename), FILE_ATTRIBUTE_NORMAL);
-                       }
-                       if(!QFile::remove(filename))
-                       {
-                               qWarning("Could not delete \"%s\"", filename.toLatin1().constData());
-                               return false;
-                       }
-                       else
-                       {
-                               return true;
-                       }
-               }
-               else
+               QPixmap pixmap = icon.pixmap(w, h);
+               if(!pixmap.isNull())
                {
-                       return true;
+                       return pixmap.toWinHICON();
                }
        }
+       return NULL;
 }
 
 /*
@@ -1334,11 +244,11 @@ bool lamexp_themes_enabled(void)
        if(!g_lamexp_themes_enabled.bInitialized)
        {
                g_lamexp_themes_enabled.bThemesEnabled = false;
-               const lamexp_os_version_t &osVersion = lamexp_get_os_version();
-               if(osVersion >= lamexp_winver_winxp)
+               const MUtils::OS::Version::os_version_t &osVersion = MUtils::OS::os_version();
+               if(osVersion >= MUtils::OS::Version::WINDOWS_WINXP)
                {
                        IsAppThemedFun IsAppThemedPtr = NULL;
-                       QLibrary uxTheme(QString("%1/UxTheme.dll").arg(lamexp_known_folder(lamexp_folder_systemfolder)));
+                       QLibrary uxTheme(QString("%1/UxTheme.dll").arg(MUtils::OS::known_folder(MUtils::OS::FOLDER_SYSTEMFOLDER)));
                        if(uxTheme.load())
                        {
                                IsAppThemedPtr = (IsAppThemedFun) uxTheme.resolve("IsAppThemed");
@@ -1420,7 +330,7 @@ bool lamexp_shutdown_computer(const QString &message, const unsigned long timeou
                                        }
                                }
                                const DWORD reason = SHTDN_REASON_MAJOR_APPLICATION | SHTDN_REASON_FLAG_PLANNED;
-                               return InitiateSystemShutdownEx(NULL, const_cast<wchar_t*>(QWCHAR(message)), timeout, forceShutdown ? TRUE : FALSE, FALSE, reason);
+                               return InitiateSystemShutdownEx(NULL, const_cast<wchar_t*>(MUTILS_WCHR(message)), timeout, forceShutdown ? TRUE : FALSE, FALSE, reason);
                        }
                }
        }
@@ -1429,83 +339,23 @@ bool lamexp_shutdown_computer(const QString &message, const unsigned long timeou
 }
 
 /*
- * Determines the current date, resistant against certain manipulations
+ * Block window "move" message
  */
-QDate lamexp_current_date_safe(void)
+bool lamexp_block_window_move(void *message)
 {
-       const DWORD MAX_PROC = 1024;
-       DWORD *processes = new DWORD[MAX_PROC];
-       DWORD bytesReturned = 0;
-       
-       if(!EnumProcesses(processes, sizeof(DWORD) * MAX_PROC, &bytesReturned))
-       {
-               LAMEXP_DELETE_ARRAY(processes);
-               return QDate::currentDate();
-       }
-
-       const DWORD procCount = bytesReturned / sizeof(DWORD);
-       ULARGE_INTEGER lastStartTime;
-       memset(&lastStartTime, 0, sizeof(ULARGE_INTEGER));
-
-       for(DWORD i = 0; i < procCount; i++)
+       if(message)
        {
-               HANDLE hProc = OpenProcess(PROCESS_QUERY_INFORMATION, FALSE, processes[i]);
-               if(hProc)
+               MSG *msg = reinterpret_cast<MSG*>(message);
+               if((msg->message == WM_SYSCOMMAND) && (msg->wParam == SC_MOVE))
                {
-                       FILETIME processTime[4];
-                       if(GetProcessTimes(hProc, &processTime[0], &processTime[1], &processTime[2], &processTime[3]))
-                       {
-                               ULARGE_INTEGER timeCreation;
-                               timeCreation.LowPart = processTime[0].dwLowDateTime;
-                               timeCreation.HighPart = processTime[0].dwHighDateTime;
-                               if(timeCreation.QuadPart > lastStartTime.QuadPart)
-                               {
-                                       lastStartTime.QuadPart = timeCreation.QuadPart;
-                               }
-                       }
-                       CloseHandle(hProc);
+                       return true;
+               }
+               if((msg->message == WM_NCLBUTTONDOWN) && (msg->wParam == HTCAPTION))
+               {
+                       return true;
                }
        }
-
-       LAMEXP_DELETE_ARRAY(processes);
-       
-       FILETIME lastStartTime_fileTime;
-       lastStartTime_fileTime.dwHighDateTime = lastStartTime.HighPart;
-       lastStartTime_fileTime.dwLowDateTime = lastStartTime.LowPart;
-
-       FILETIME lastStartTime_localTime;
-       if(!FileTimeToLocalFileTime(&lastStartTime_fileTime, &lastStartTime_localTime))
-       {
-               memcpy(&lastStartTime_localTime, &lastStartTime_fileTime, sizeof(FILETIME));
-       }
-       
-       SYSTEMTIME lastStartTime_system;
-       if(!FileTimeToSystemTime(&lastStartTime_localTime, &lastStartTime_system))
-       {
-               memset(&lastStartTime_system, 0, sizeof(SYSTEMTIME));
-               lastStartTime_system.wYear = 1970; lastStartTime_system.wMonth = lastStartTime_system.wDay = 1;
-       }
-
-       const QDate currentDate = QDate::currentDate();
-       const QDate processDate = QDate(lastStartTime_system.wYear, lastStartTime_system.wMonth, lastStartTime_system.wDay);
-       return (currentDate >= processDate) ? currentDate : processDate;
-}
-
-/*
- * Show system message box
- */
-int lamexp_system_message(const wchar_t *text, int beepType)
-{
-       UINT flags = MB_OK | MB_TOPMOST;
-
-       switch(beepType)
-       {
-               case lamexp_beep_info:    flags = flags | MB_ICONASTERISK;
-               case lamexp_beep_warning: flags = flags | MB_ICONEXCLAMATION;
-               case lamexp_beep_error:   flags = flags | MB_ICONHAND;
-       }
-
-       return MessageBoxW(NULL, text, L"LameXP", flags);
+       return false;
 }
 
 /*
@@ -1530,44 +380,84 @@ bool lamexp_beep(int beepType)
 /*
  * Play a sound (from resources)
  */
-bool lamexp_play_sound(const unsigned short uiSoundIdx, const bool bAsync, const wchar_t *alias)
+bool lamexp_play_sound(const QString &name, const bool bAsync)
 {
-       if(alias)
+       const unsigned char *data = NULL;
+       
+       //Try to look-up the sound in the cache first
+       if(!name.isEmpty())
        {
-               return PlaySound(alias, GetModuleHandle(NULL), (SND_ALIAS | (bAsync ? SND_ASYNC : SND_SYNC))) == TRUE;
+               QReadLocker readLock(&g_lamexp_sounds.lock);
+               if(g_lamexp_sounds.sound_db && g_lamexp_sounds.sound_db->contains(name))
+               {
+                       data = g_lamexp_sounds.sound_db->value(name);
+               }
        }
-       else
+       
+       //If data not found in cache, try to load from resource!
+       if((!data) && (!name.isEmpty()))
+       {
+               QResource resource(QString(":/sounds/%1.wav").arg(name));
+               if(resource.isValid() && (data = resource.data()))
+               {
+                       QWriteLocker writeLock(&g_lamexp_sounds.lock);
+                       if(!g_lamexp_sounds.sound_db)
+                       {
+                               g_lamexp_sounds.sound_db = new QHash<const QString, const unsigned char*>();
+                       }
+                       g_lamexp_sounds.sound_db->insert(name, data);
+               }
+               else
+               {
+                       qWarning("Sound effect \"%s\" not found!", MUTILS_UTF8(name));
+               }
+       }
+
+       //Play the sound, if availbale
+       if(data)
        {
-               return PlaySound(MAKEINTRESOURCE(uiSoundIdx), GetModuleHandle(NULL), (SND_RESOURCE | (bAsync ? SND_ASYNC : SND_SYNC))) == TRUE;
+               return PlaySound(LPCWSTR(data), NULL, (SND_MEMORY | (bAsync ? SND_ASYNC : SND_SYNC))) != FALSE;
        }
+
+       return false;
 }
 
 /*
- * Play a sound (from resources)
+ * Play a sound (system alias)
+ */
+bool lamexp_play_sound_alias(const QString &alias, const bool bAsync)
+{
+       return PlaySound(MUTILS_WCHR(alias), GetModuleHandle(NULL), (SND_ALIAS | (bAsync ? SND_ASYNC : SND_SYNC))) != FALSE;
+}
+
+/*
+ * Play a sound (from external DLL)
  */
 bool lamexp_play_sound_file(const QString &library, const unsigned short uiSoundIdx, const bool bAsync)
 {
        bool result = false;
-       HMODULE module = NULL;
 
        QFileInfo libraryFile(library);
        if(!libraryFile.isAbsolute())
        {
-               unsigned int buffSize = GetSystemDirectoryW(NULL, NULL) + 1;
-               wchar_t *buffer = (wchar_t*) _malloca(buffSize * sizeof(wchar_t));
-               unsigned int result = GetSystemDirectory(buffer, buffSize);
-               if(result > 0 && result < buffSize)
+               const QString &systemDir = MUtils::OS::known_folder(MUtils::OS::FOLDER_SYSTEMFOLDER);
+               if(!systemDir.isEmpty())
                {
-                       libraryFile.setFile(QString("%1/%2").arg(QDir::fromNativeSeparators(QString::fromUtf16(reinterpret_cast<const unsigned short*>(buffer))), library));
+                       libraryFile.setFile(QDir(systemDir), libraryFile.fileName());
                }
-               _freea(buffer);
        }
 
-       module = LoadLibraryW(QWCHAR(QDir::toNativeSeparators(libraryFile.absoluteFilePath())));
-       if(module)
+       if(libraryFile.exists() && libraryFile.isFile())
+       {
+               if(HMODULE module = LoadLibraryW(MUTILS_WCHR(QDir::toNativeSeparators(libraryFile.canonicalFilePath()))))
+               {
+                       result = (PlaySound(MAKEINTRESOURCE(uiSoundIdx), module, (SND_RESOURCE | (bAsync ? SND_ASYNC : SND_SYNC))) != FALSE);
+                       FreeLibrary(module);
+               }
+       }
+       else
        {
-               result = (PlaySound(MAKEINTRESOURCE(uiSoundIdx), module, (SND_RESOURCE | (bAsync ? SND_ASYNC : SND_SYNC))) == TRUE);
-               FreeLibrary(module);
+               qWarning("PlaySound: File \"%s\" could not be found!", MUTILS_UTF8(libraryFile.absoluteFilePath()));
        }
 
        return result;
@@ -1586,7 +476,7 @@ bool lamexp_exec_shell(const QWidget *win, const QString &url, const bool explor
  */
 bool lamexp_exec_shell(const QWidget *win, const QString &url, const QString &parameters, const QString &directory, const bool explore)
 {
-       return ((int) ShellExecuteW(((win) ? win->winId() : NULL), (explore ? L"explore" : L"open"), QWCHAR(url), ((!parameters.isEmpty()) ? QWCHAR(parameters) : NULL), ((!directory.isEmpty()) ? QWCHAR(directory) : NULL), SW_SHOW)) > 32;
+       return ((int) ShellExecuteW(((win) ? win->winId() : NULL), (explore ? L"explore" : L"open"), MUTILS_WCHR(url), ((!parameters.isEmpty()) ? MUTILS_WCHR(parameters) : NULL), ((!directory.isEmpty()) ? MUTILS_WCHR(directory) : NULL), SW_SHOW)) > 32;
 }
 
        /*
@@ -1625,7 +515,7 @@ bool lamexp_append_sysmenu(const QWidget *win, const unsigned int identifier, co
        if(HMENU hMenu = GetSystemMenu(win->winId(), FALSE))
        {
                ok = (AppendMenuW(hMenu, MF_SEPARATOR, 0, 0) == TRUE);
-               ok = (AppendMenuW(hMenu, MF_STRING, identifier, QWCHAR(text)) == TRUE);
+               ok = (AppendMenuW(hMenu, MF_STRING, identifier, MUTILS_WCHR(text)) == TRUE);
        }
 
        return ok;
@@ -1648,7 +538,7 @@ bool lamexp_update_sysmenu(const QWidget *win, const unsigned int identifier, co
        
        if(HMENU hMenu = ::GetSystemMenu(win->winId(), FALSE))
        {
-               ok = (ModifyMenu(hMenu, identifier, MF_STRING | MF_BYCOMMAND, identifier, QWCHAR(text)) == TRUE);
+               ok = (ModifyMenu(hMenu, identifier, MF_STRING | MF_BYCOMMAND, identifier, MUTILS_WCHR(text)) == TRUE);
        }
        return ok;
 }
@@ -1758,8 +648,12 @@ bool lamexp_bring_to_front(const QWidget *window)
        
        if(window)
        {
-               ret = (SetForegroundWindow(window->winId()) == TRUE);
-               SwitchToThisWindow(window->winId(), TRUE);
+               for(int i = 0; (i < 5) && (!ret); i++)
+               {
+                       ret = (SetForegroundWindow(window->winId()) != FALSE);
+                       SwitchToThisWindow(window->winId(), TRUE);
+               }
+               LockSetForegroundWindow(LSFW_LOCK);
        }
 
        return ret;
@@ -1792,20 +686,6 @@ bool lamexp_bring_process_to_front(const unsigned long pid)
 }
 
 /*
- * Check the network connection status
- */
-int lamexp_network_status(void)
-{
-       DWORD dwFlags;
-       const BOOL ret = (IsNetworkAlive(&dwFlags) == TRUE);
-       if(GetLastError() == 0)
-       {
-               return (ret == TRUE) ? lamexp_network_yes : lamexp_network_non;
-       }
-       return lamexp_network_err;
-}
-
-/*
  * Retrun the process ID of the given QProcess
  */
 unsigned long lamexp_process_id(const QProcess *proc)
@@ -1860,8 +740,8 @@ bool lamexp_open_media_file(const QString &mediaFilePath)
                        QString mplayerPath;
                        HKEY registryKeyHandle = NULL;
 
-                       const QString currentKey = WCHAR2QSTR(registryPrefix[j]).append(WCHAR2QSTR(registryKeys[i]));
-                       if(RegOpenKeyExW(HKEY_LOCAL_MACHINE, QWCHAR(currentKey), 0, KEY_READ, &registryKeyHandle) == ERROR_SUCCESS)
+                       const QString currentKey = MUTILS_QSTR(registryPrefix[j]).append(MUTILS_QSTR(registryKeys[i]));
+                       if(RegOpenKeyExW(HKEY_LOCAL_MACHINE, MUTILS_WCHR(currentKey), 0, KEY_READ, &registryKeyHandle) == ERROR_SUCCESS)
                        {
                                for(size_t k = 0; k < 2; k++)
                                {
@@ -1872,7 +752,7 @@ bool lamexp_open_media_file(const QString &mediaFilePath)
                                        {
                                                if((DataType == REG_SZ) || (DataType == REG_EXPAND_SZ) || (DataType == REG_LINK))
                                                {
-                                                       mplayerPath = WCHAR2QSTR(Buffer);
+                                                       mplayerPath = MUTILS_QSTR(Buffer);
                                                        break;
                                                }
                                        }
@@ -1887,10 +767,10 @@ bool lamexp_open_media_file(const QString &mediaFilePath)
                                {
                                        for(size_t k = 0; k < 4; k++)
                                        {
-                                               if(mplayerDir.exists(WCHAR2QSTR(appNames[k])))
+                                               if(mplayerDir.exists(MUTILS_QSTR(appNames[k])))
                                                {
-                                                       qDebug("Player found at:\n%s\n", QUTF8(mplayerDir.absoluteFilePath(WCHAR2QSTR(appNames[k]))));
-                                                       QProcess::startDetached(mplayerDir.absoluteFilePath(WCHAR2QSTR(appNames[k])), QStringList() << QDir::toNativeSeparators(mediaFilePath));
+                                                       qDebug("Player found at:\n%s\n", MUTILS_UTF8(mplayerDir.absoluteFilePath(MUTILS_QSTR(appNames[k]))));
+                                                       QProcess::startDetached(mplayerDir.absoluteFilePath(MUTILS_QSTR(appNames[k])), QStringList() << QDir::toNativeSeparators(mediaFilePath));
                                                        return true;
                                                }
                                        }
@@ -1917,7 +797,8 @@ static void lamexp_init_dwmapi(void)
        g_lamexp_dwmapi.dwmEnableBlurBehindWindow = NULL;
                        
        //Does OS support DWM?
-       if(lamexp_get_os_version() >= lamexp_winver_vista)
+       const MUtils::OS::Version::os_version_t &osVersion = MUtils::OS::os_version();
+       if(osVersion >= MUtils::OS::Version::WINDOWS_VISTA)
        {
                //Load DWMAPI.DLL
                g_lamexp_dwmapi.dwmapi_dll = new QLibrary("dwmapi.dll");
@@ -1930,7 +811,7 @@ static void lamexp_init_dwmapi(void)
                }
                else
                {
-                       LAMEXP_DELETE(g_lamexp_dwmapi.dwmapi_dll);
+                       MUTILS_DELETE(g_lamexp_dwmapi.dwmapi_dll);
                        qWarning("Failed to load DWMAPI.DLL on a DWM-enabled system!");
                }
        }
@@ -2104,41 +985,115 @@ QColor lamexp_system_color(const int color_id)
 }
 
 /*
- * Fatal application exit
+ * Check if the current user is an administartor (helper function)
  */
-#pragma intrinsic(_InterlockedExchange)
-void lamexp_fatal_exit(const wchar_t* exitMessage, const wchar_t* errorBoxMessage)
+static bool lamexp_user_is_admin_helper(void)
 {
-       static volatile long bFatalFlag = 0L;
+       HANDLE hToken = NULL;
+       if(!OpenProcessToken(GetCurrentProcess(), TOKEN_QUERY, &hToken))
+       {
+               return false;
+       }
 
-       if(_InterlockedExchange(&bFatalFlag, 1L) == 0L)
+       DWORD dwSize = 0;
+       if(!GetTokenInformation(hToken, TokenGroups, NULL, 0, &dwSize))
        {
-               if(GetCurrentThreadId() != g_main_thread_id)
-               {
-                       HANDLE mainThread = OpenThread(THREAD_TERMINATE, FALSE, g_main_thread_id);
-                       if(mainThread) TerminateThread(mainThread, ULONG_MAX);
-               }
-       
-               if(errorBoxMessage)
+               if(GetLastError() != ERROR_INSUFFICIENT_BUFFER)
                {
-                       MessageBoxW(NULL, errorBoxMessage, L"LameXP - GURU MEDITATION", MB_ICONERROR | MB_TOPMOST | MB_TASKMODAL);
+                       CloseHandle(hToken);
+                       return false;
                }
+       }
+
+       PTOKEN_GROUPS lpGroups = (PTOKEN_GROUPS) malloc(dwSize);
+       if(!lpGroups)
+       {
+               CloseHandle(hToken);
+               return false;
+       }
 
-               for(;;)
+       if(!GetTokenInformation(hToken, TokenGroups, lpGroups, dwSize, &dwSize))
+       {
+               free(lpGroups);
+               CloseHandle(hToken);
+               return false;
+       }
+
+       PSID lpSid = NULL; SID_IDENTIFIER_AUTHORITY Authority = {SECURITY_NT_AUTHORITY};
+       if(!AllocateAndInitializeSid(&Authority, 2, SECURITY_BUILTIN_DOMAIN_RID, DOMAIN_ALIAS_RID_ADMINS, 0, 0, 0, 0, 0, 0, &lpSid))
+       {
+               free(lpGroups);
+               CloseHandle(hToken);
+               return false;
+       }
+
+       bool bResult = false;
+       for(DWORD i = 0; i < lpGroups->GroupCount; i++)
+       {
+               if(EqualSid(lpSid, lpGroups->Groups[i].Sid))
                {
-                       FatalAppExit(0, exitMessage);
-                       TerminateProcess(GetCurrentProcess(), -1);
+                       bResult = true;
+                       break;
                }
        }
 
-       TerminateThread(GetCurrentThread(), -1);
-       Sleep(INFINITE);
+       FreeSid(lpSid);
+       free(lpGroups);
+       CloseHandle(hToken);
+       return bResult;
+}
+
+/*
+ * Check if the current user is an administartor
+ */
+bool lamexp_user_is_admin(void)
+{
+       bool isAdmin = false;
+
+       //Check for process elevation and UAC support first!
+       if(MUtils::OS::is_elevated(&isAdmin))
+       {
+               qWarning("Process is elevated -> user is admin!");
+               return true;
+       }
+       
+       //If not elevated and UAC is not available -> user must be in admin group!
+       if(!isAdmin)
+       {
+               qDebug("UAC is disabled/unavailable -> checking for Administrators group");
+               isAdmin = lamexp_user_is_admin_helper();
+       }
+
+       return isAdmin;
+}
+
+/*
+ * Check if file is a valid Win32/Win64 executable
+ */
+bool lamexp_is_executable(const QString &path)
+{
+       bool bIsExecutable = false;
+       DWORD binaryType;
+       if(GetBinaryType(MUTILS_WCHR(QDir::toNativeSeparators(path)), &binaryType))
+       {
+               bIsExecutable = (binaryType == SCS_32BIT_BINARY || binaryType == SCS_64BIT_BINARY);
+       }
+       return bIsExecutable;
+}
+
+/*
+ * Fatal application exit - helper
+ */
+static DWORD WINAPI lamexp_fatal_exit_helper(LPVOID lpParameter)
+{
+       MessageBoxA(NULL, ((LPCSTR) lpParameter), "LameXP - Guru Meditation", MB_OK | MB_ICONERROR | MB_TASKMODAL | MB_TOPMOST | MB_SETFOREGROUND);
+       return 0;
 }
 
 /*
  * Initialize debug thread
  */
-static const HANDLE g_debug_thread1 = LAMEXP_DEBUG ? NULL : lamexp_debug_thread_init();
+static const HANDLE g_debug_thread1 = MUTILS_DEBUG ? NULL : lamexp_debug_thread_init();
 
 /*
  * Get number private bytes [debug only]
@@ -2179,18 +1134,16 @@ void lamexp_dbg_dbg_output_string(const char* format, ...)
 
 extern "C" void _lamexp_global_init_win32(void)
 {
-       if((!LAMEXP_DEBUG) && lamexp_check_for_debugger())
+       if((!MUTILS_DEBUG) && lamexp_check_for_debugger())
        {
-               lamexp_fatal_exit(L"Not a debug build. Please unload debugger and try again!");
+               MUtils::OS::fatal_exit(L"Not a debug build. Please unload debugger and try again!");
        }
 
        //Zero *before* constructors are called
-       LAMEXP_ZERO_MEMORY(g_lamexp_argv);
-       LAMEXP_ZERO_MEMORY(g_lamexp_known_folder);
-       LAMEXP_ZERO_MEMORY(g_lamexp_os_version);
        LAMEXP_ZERO_MEMORY(g_lamexp_wine);
        LAMEXP_ZERO_MEMORY(g_lamexp_themes_enabled);
        LAMEXP_ZERO_MEMORY(g_lamexp_dwmapi);
+       LAMEXP_ZERO_MEMORY(g_lamexp_sounds);
 }
 
 ///////////////////////////////////////////////////////////////////////////////
@@ -2199,41 +1152,16 @@ extern "C" void _lamexp_global_init_win32(void)
 
 extern "C" void _lamexp_global_free_win32(void)
 {
-       //Clear folder cache
-       LAMEXP_DELETE(g_lamexp_known_folder.knownFolders);
-
        //Destroy Qt application object
        QApplication *application = dynamic_cast<QApplication*>(QApplication::instance());
-       LAMEXP_DELETE(application);
+       MUTILS_DELETE(application);
 
        //Release DWM API
        g_lamexp_dwmapi.dwmIsCompositionEnabled = NULL;
        g_lamexp_dwmapi.dwmExtendFrameIntoClientArea = NULL;
        g_lamexp_dwmapi.dwmEnableBlurBehindWindow = NULL;
-       LAMEXP_DELETE(g_lamexp_dwmapi.dwmapi_dll);
-
-       //Free STDOUT and STDERR buffers
-       if(g_lamexp_console_attached)
-       {
-               if(std::filebuf *tmp = dynamic_cast<std::filebuf*>(std::cout.rdbuf()))
-               {
-                       std::cout.rdbuf(NULL);
-                       LAMEXP_DELETE(tmp);
-               }
-               if(std::filebuf *tmp = dynamic_cast<std::filebuf*>(std::cerr.rdbuf()))
-               {
-                       std::cerr.rdbuf(NULL);
-                       LAMEXP_DELETE(tmp);
-               }
-       }
-
-       //Close log file
-       if(g_lamexp_log_file)
-       {
-               fclose(g_lamexp_log_file);
-               g_lamexp_log_file = NULL;
-       }
+       MUTILS_DELETE(g_lamexp_dwmapi.dwmapi_dll);
 
-       //Free CLI Arguments
-       LAMEXP_DELETE(g_lamexp_argv.list);
+       //Clear sound cache
+       MUTILS_DELETE(g_lamexp_sounds.sound_db);
 }