///////////////////////////////////////////////////////////////////////////////
// LameXP - Audio Encoder Front-End
-// Copyright (C) 2004-2010 LoRd_MuldeR <MuldeR2@GMX.de>
+// Copyright (C) 2004-2016 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
// the Free Software Foundation; either version 2 of the License, or
-// (at your option) any later version.
+// (at your option) any later version, but always including the *additional*
+// restrictions defined in the "License.txt" file.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
#include "Global.h"
#include "Dialog_SplashScreen.h"
#include "Dialog_MainWindow.h"
+#include "Dialog_Processing.h"
#include "Thread_Initialization.h"
#include "Thread_MessageProducer.h"
#include "Model_Settings.h"
+#include "Model_FileList.h"
+#include "Model_AudioFile.h"
+#include "Encoder_Abstract.h"
+#include "ShellIntegration.h"
+
+//MUitls
+#include <MUtils/Global.h>
+#include <MUtils/OSSupport.h>
+#include <MUtils/Version.h>
+#include <MUtils/CPUFeatures.h>
+#include <MUtils/Terminal.h>
+#include <MUtils/Startup.h>
+#include <MUtils/IPCChannel.h>
//Qt includes
+#include <QMutex>
#include <QApplication>
#include <QMessageBox>
#include <QDate>
+#include <QDir>
+
+//VLD
+#ifdef _MSC_VER
+#include <vld.h>
+#endif
///////////////////////////////////////////////////////////////////////////////
-// Main function
+// Helper functions
///////////////////////////////////////////////////////////////////////////////
-int lamexp_main(int argc, char* argv[])
+static void lamexp_print_logo(void)
{
- int iResult = -1;
-
- //Init console
- lamexp_init_console(argc, argv);
-
//Print version info
- qDebug("LameXP - Audio Encoder Front-End");
- qDebug("Version %d.%02d %s, Build %d [%s], MSVC compiler v%02d.%02d", lamexp_version_major(), lamexp_version_minor(), lamexp_version_release(), lamexp_version_build(), lamexp_version_date().toString(Qt::ISODate).toLatin1().constData(), _MSC_VER / 100, _MSC_VER % 100);
- qDebug("Copyright (C) 2004-%04d LoRd_MuldeR <MuldeR2@GMX.de>\n", max(lamexp_version_date().year(),QDate::currentDate().year()));
+ qDebug("LameXP - Audio Encoder Front-End v%d.%02d %s (Build #%03d)", lamexp_version_major(), lamexp_version_minor(), lamexp_version_release(), lamexp_version_build());
+ qDebug("Copyright (c) 2004-%04d LoRd_MuldeR <mulder2@gmx.de>. Some rights reserved.", qMax(MUtils::Version::app_build_date().year(), MUtils::OS::current_date().year()));
+ qDebug("Built on %s at %s with %s for Win-%s.\n", MUTILS_UTF8(MUtils::Version::app_build_date().toString(Qt::ISODate)), MUTILS_UTF8(MUtils::Version::app_build_time().toString(Qt::ISODate)), MUtils::Version::compiler_version(), MUtils::Version::compiler_arch());
//print license info
qDebug("This program is free software: you can redistribute it and/or modify");
qDebug("it under the terms of the GNU General Public License <http://www.gnu.org/>.");
- qDebug("This program comes with ABSOLUTELY NO WARRANTY.\n");
+ qDebug("Note that this program is distributed with ABSOLUTELY NO WARRANTY.\n");
+
+ //Print library version
+ qDebug("This application is powerd by MUtils library v%u.%02u (%s, %s).\n", MUtils::Version::lib_version_major(), MUtils::Version::lib_version_minor(), MUTILS_UTF8(MUtils::Version::lib_build_date().toString(Qt::ISODate)), MUTILS_UTF8(MUtils::Version::lib_build_time().toString(Qt::ISODate)));
//Print warning, if this is a "debug" build
- LAMEXP_CHECK_DEBUG_BUILD;
-
- //Initialize Qt
- lamexp_init_qt(argc, argv);
-
- //Check for expiration
- if(lamexp_version_demo())
+ if(MUTILS_DEBUG)
{
- QDate expireDate = lamexp_version_date().addDays(14);
- qWarning(QString("Note: This demo (pre-release) version of LameXP will expire at %1.\n").arg(expireDate.toString(Qt::ISODate)).toLatin1().constData());
- if(QDate::currentDate() >= expireDate)
- {
- qWarning("Binary has expired !!!");
- QMessageBox::warning(NULL, "LameXP - Expired", QString("This demo (pre-release) version of LameXP has expired at %1.\nLameXP is free software and release versions won't expire.").arg(expireDate.toString()), "Exit Program");
- return 0;
- }
+ qWarning("---------------------------------------------------------");
+ qWarning("DEBUG BUILD: DO NOT RELEASE THIS BINARY TO THE PUBLIC !!!");
+ qWarning("---------------------------------------------------------\n");
}
+}
- //Check for multiple instances of LameXP
- if((iResult = lamexp_init_ipc()) != 0)
+static int lamexp_initialize_ipc(MUtils::IPCChannel *const ipcChannel)
+{
+ int iResult = 0;
+
+ if((iResult = ipcChannel->initialize()) != MUtils::IPCChannel::RET_SUCCESS_MASTER)
{
- qDebug("LameXP is already running, connecting to running instance...");
- if(iResult == 1)
+ if(iResult == MUtils::IPCChannel::RET_SUCCESS_SLAVE)
{
- MessageProducerThread *messageProducerThread = new MessageProducerThread();
+ qDebug("LameXP is already running, connecting to running instance...");
+ QScopedPointer<MessageProducerThread> messageProducerThread(new MessageProducerThread(ipcChannel));
messageProducerThread->start();
if(!messageProducerThread->wait(30000))
{
+ qWarning("MessageProducer thread has encountered timeout -> going to kill!");
messageProducerThread->terminate();
- QMessageBox messageBox(QMessageBox::Critical, "LameXP", "LameXP is already running, but the running instance doesn't respond!", QMessageBox::NoButton, NULL, Qt::Dialog | Qt::MSWindowsFixedSizeDialogHint | Qt::WindowStaysOnTopHint);
- messageBox.exec();
messageProducerThread->wait();
- LAMEXP_DELETE(messageProducerThread);
+ MUtils::OS::system_message_err(L"LameXP", L"LameXP is already running, but the running instance doesn't respond!");
return -1;
}
- LAMEXP_DELETE(messageProducerThread);
+ return 0;
+ }
+ else
+ {
+ qFatal("The IPC initialization has failed!");
+ return -1;
}
- return 0;
}
- //Kill application?
- for(int i = 0; i < argc; i++)
+ return 1;
+}
+
+static void lamexp_show_splash(const MUtils::CPUFetaures::cpu_info_t &cpuFeatures, SettingsModel *const settingsModel)
+{
+ QScopedPointer<InitializationThread> poInitializationThread(new InitializationThread(cpuFeatures));
+ SplashScreen::showSplash(poInitializationThread.data());
+ settingsModel->slowStartup(poInitializationThread->getSlowIndicator());
+}
+
+static int lamexp_main_loop(const MUtils::CPUFetaures::cpu_info_t &cpuFeatures, MUtils::IPCChannel *const ipcChannel, int &iShutdown)
+{
+ int iResult = -1;
+ bool bAccepted = true;
+
+ //Create models
+ QScopedPointer<FileListModel> fileListModel(new FileListModel() );
+ QScopedPointer<AudioFileModel_MetaInfo> metaInfo (new AudioFileModel_MetaInfo());
+ QScopedPointer<SettingsModel> settingsModel(new SettingsModel() );
+
+ //Show splash screen
+ lamexp_show_splash(cpuFeatures, settingsModel.data());
+
+ //Validate settings
+ settingsModel->validate();
+
+ //Create main window
+ QScopedPointer<MainWindow> poMainWindow(new MainWindow(ipcChannel, fileListModel.data(), metaInfo.data(), settingsModel.data()));
+
+ //Main application loop
+ while(bAccepted && (iShutdown <= SHUTDOWN_FLAG_NONE))
{
- if(!_stricmp("--kill", argv[i]) || !_stricmp("--force-kill", argv[i]))
+ //Show main window
+ poMainWindow->show();
+ iResult = qApp->exec();
+ bAccepted = poMainWindow->isAccepted();
+
+ //Sync settings
+ settingsModel->syncNow();
+
+ //Show processing dialog
+ if(bAccepted && (fileListModel->rowCount() > 0))
{
- return 0;
+ ProcessingDialog *processingDialog = new ProcessingDialog(fileListModel.data(), metaInfo.data(), settingsModel.data());
+ processingDialog->exec();
+ iShutdown = processingDialog->getShutdownFlag();
+ MUTILS_DELETE(processingDialog);
}
}
-
- //Show splash screen
- InitializationThread *poInitializationThread = new InitializationThread();
- SplashScreen::showSplash(poInitializationThread);
- LAMEXP_DELETE(poInitializationThread);
-
- //Show main window
- MainWindow *poMainWindow = new MainWindow();
- poMainWindow->show();
- iResult = QApplication::instance()->exec();
- LAMEXP_DELETE(poMainWindow);
-
- //Final clean-up
- qDebug("Shutting down, please wait...\n");
- //Terminate
return iResult;
}
///////////////////////////////////////////////////////////////////////////////
-// Message Handler
+// Main function
///////////////////////////////////////////////////////////////////////////////
-static void lamexp_message_handler(QtMsgType type, const char *msg)
+static int lamexp_main(int &argc, char **argv)
{
- static HANDLE hConsole = NULL;
-
- if(!hConsole)
+ int iResult = -1;
+ int iShutdown = SHUTDOWN_FLAG_NONE;
+
+ //Print logo
+ lamexp_print_logo();
+
+ //Get CLI arguments
+ const MUtils::OS::ArgumentMap &arguments = MUtils::OS::arguments();
+
+ //Enumerate CLI arguments
+ if(!arguments.isEmpty())
{
- hConsole = CreateFile(L"CONOUT$", GENERIC_WRITE, FILE_SHARE_WRITE | FILE_SHARE_READ, NULL, OPEN_EXISTING, NULL, NULL);
+ qDebug("Command-Line Arguments:");
+ foreach(const QString &key, arguments.uniqueKeys())
+ {
+ foreach(const QString &val, arguments.values(key))
+ {
+ if(!val.isEmpty())
+ {
+ qDebug("--%s = \"%s\"", MUTILS_UTF8(key), MUTILS_UTF8(val));
+ continue;
+ }
+ qDebug("--%s", MUTILS_UTF8(key));
+ }
+ }
+ qDebug(" ");
}
- CONSOLE_SCREEN_BUFFER_INFO bufferInfo;
- GetConsoleScreenBufferInfo(hConsole, &bufferInfo);
-
- switch(type)
+ //Uninstall?
+ if(arguments.contains("uninstall"))
{
- case QtCriticalMsg:
- case QtFatalMsg:
- fflush(stdout);
- fflush(stderr);
- SetConsoleTextAttribute(hConsole, FOREGROUND_RED | FOREGROUND_INTENSITY);
- fprintf(stderr, "\nCRITICAL ERROR !!!\n%s\n\n", msg);
- MessageBoxA(NULL, msg, "LameXP - CRITICAL ERROR", MB_ICONERROR | MB_TOPMOST | MB_TASKMODAL);
- break;
- case QtWarningMsg:
- SetConsoleTextAttribute(hConsole, FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_INTENSITY);
- fprintf(stderr, "%s\n", msg);
- fflush(stderr);
- break;
- default:
- SetConsoleTextAttribute(hConsole, FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED | FOREGROUND_INTENSITY);
- fprintf(stderr, "%s\n", msg);
- fflush(stderr);
- break;
+ qWarning("Un-install: Removing LameXP shell integration...");
+ ShellIntegration::remove(false);
+ return EXIT_SUCCESS;
}
- SetConsoleTextAttribute(hConsole, FOREGROUND_BLUE | FOREGROUND_GREEN | FOREGROUND_RED);
+ //Detect CPU capabilities
+ const MUtils::CPUFetaures::cpu_info_t cpuFeatures = MUtils::CPUFetaures::detect();
+ qDebug(" CPU vendor id : %s (Intel=%s)", cpuFeatures.vendor, MUTILS_BOOL2STR(cpuFeatures.intel));
+ qDebug("CPU brand string : %s", cpuFeatures.brand);
+ qDebug(" CPU signature : Family=%d Model=%d Stepping=%d", cpuFeatures.family, cpuFeatures.model, cpuFeatures.stepping);
+ qDebug("CPU capabilities : MMX=%s SSE=%s SSE2=%s SSE3=%s SSSE3=%s SSE4=%s SSE4.2=%s x64=%s", MUTILS_BOOL2STR(cpuFeatures.features & MUtils::CPUFetaures::FLAG_MMX), MUTILS_BOOL2STR(cpuFeatures.features & MUtils::CPUFetaures::FLAG_SSE), MUTILS_BOOL2STR(cpuFeatures.features & MUtils::CPUFetaures::FLAG_SSE2), MUTILS_BOOL2STR(cpuFeatures.features & MUtils::CPUFetaures::FLAG_SSE3), MUTILS_BOOL2STR(cpuFeatures.features & MUtils::CPUFetaures::FLAG_SSSE3), MUTILS_BOOL2STR(cpuFeatures.features & MUtils::CPUFetaures::FLAG_SSE4), MUTILS_BOOL2STR(cpuFeatures.features & MUtils::CPUFetaures::FLAG_SSE42), MUTILS_BOOL2STR(cpuFeatures.x64));
+ qDebug(" Number of CPU's : %d\n", cpuFeatures.count);
- if(type == QtCriticalMsg || type == QtFatalMsg)
+ //Initialize Qt
+ QScopedPointer<QApplication> application(MUtils::Startup::create_qt(argc, argv, QLatin1String("LameXP - Audio Encoder Front-End")));
+ if(application.isNull())
{
- FatalAppExit(0, L"The application has encountered a critical error and will exit now!");
- TerminateProcess(GetCurrentProcess(), -1);
+ return EXIT_FAILURE;
}
- }
+ //Initialize application
+ application->setWindowIcon(lamexp_app_icon());
+ application->setApplicationVersion(QString().sprintf("%d.%02d.%04d", lamexp_version_major(), lamexp_version_minor(), lamexp_version_build()));
-///////////////////////////////////////////////////////////////////////////////
-// Applicaton entry point
-///////////////////////////////////////////////////////////////////////////////
+ //Check for expiration
+ if(lamexp_version_demo())
+ {
+ const QDate currentDate = MUtils::OS::current_date();
+ if(currentDate.addDays(1) < MUtils::Version::app_build_date())
+ {
+ qFatal("System's date (%s) is before LameXP build date (%s). Huh?", currentDate.toString(Qt::ISODate).toLatin1().constData(), MUtils::Version::app_build_date().toString(Qt::ISODate).toLatin1().constData());
+ }
+ qWarning(QString("Note: This demo (pre-release) version of LameXP will expire at %1.\n").arg(lamexp_version_expires().toString(Qt::ISODate)).toLatin1().constData());
+ }
-int main(int argc, char* argv[])
-{
- try
+ //Initialize IPC
+ QScopedPointer<MUtils::IPCChannel> ipcChannel(new MUtils::IPCChannel("lamexp-v4", lamexp_version_build(), "instance"));
+ if((iResult = lamexp_initialize_ipc(ipcChannel.data())) < 1)
{
- int iResult;
- qInstallMsgHandler(lamexp_message_handler);
- LAMEXP_MEMORY_CHECK(iResult = lamexp_main(argc, argv));
- lamexp_finalization();
- return iResult;
+ return (iResult == 0) ? EXIT_SUCCESS : EXIT_FAILURE;
}
- catch(char *error)
+
+ //Kill application?
+ if(arguments.contains("kill") || arguments.contains("force-kill"))
{
- fflush(stdout);
- fflush(stderr);
- fprintf(stderr, "\nEXCEPTION ERROR: %s\n", error);
- FatalAppExit(0, L"Unhandeled exception error, application will exit!");
- TerminateProcess(GetCurrentProcess(), -1);
+ return EXIT_SUCCESS;
}
- catch(int error)
+
+ //Self-test
+ if(MUTILS_DEBUG)
{
- fflush(stdout);
- fflush(stderr);
- fprintf(stderr, "\nEXCEPTION ERROR: Error code 0x%X\n", error);
- FatalAppExit(0, L"Unhandeled exception error, application will exit!");
- TerminateProcess(GetCurrentProcess(), -1);
+ InitializationThread::selfTest();
}
- catch(...)
+
+ //Main application loop
+ iResult = lamexp_main_loop(cpuFeatures, ipcChannel.data(), iShutdown);
+
+ //Final clean-up
+ qDebug("Shutting down, please wait...\n");
+
+ //Shotdown computer
+ if(iShutdown > SHUTDOWN_FLAG_NONE)
{
- fflush(stdout);
- fflush(stderr);
- fprintf(stderr, "\nEXCEPTION ERROR !!!\n");
- FatalAppExit(0, L"Unhandeled exception error, application will exit!");
- TerminateProcess(GetCurrentProcess(), -1);
+ if(!MUtils::OS::shutdown_computer(QApplication::applicationFilePath(), 12, true, (iShutdown == SHUTDOWN_FLAG_HIBERNATE)))
+ {
+ QMessageBox messageBox(QMessageBox::Critical, "LameXP", "Sorry, LameXP was unable to shutdown your computer!", QMessageBox::NoButton, NULL, Qt::Dialog | Qt::MSWindowsFixedSizeDialogHint | Qt::WindowStaysOnTopHint);
+ }
}
+
+ //Terminate
+ return iResult;
+}
+
+///////////////////////////////////////////////////////////////////////////////
+// Applicaton entry point
+///////////////////////////////////////////////////////////////////////////////
+
+int main(int argc, char* argv[])
+{
+ return MUtils::Startup::startup(argc, argv, lamexp_main, "LameXP", lamexp_version_demo());
}