384 lines
13 KiB
C++
384 lines
13 KiB
C++
/*
|
|
* 86Box A hypervisor and IBM PC system emulator that specializes in
|
|
* running old operating systems and software designed for IBM
|
|
* PC systems and compatibles from 1981 through fairly recent
|
|
* system designs based on the PCI bus.
|
|
*
|
|
* This file is part of the 86Box distribution.
|
|
*
|
|
* Main entry point module
|
|
*
|
|
*
|
|
* Authors: Joakim L. Gilje <jgilje@jgilje.net>
|
|
* Cacodemon345
|
|
* Teemu Korhonen
|
|
*
|
|
* Copyright 2021 Joakim L. Gilje
|
|
* Copyright 2021-2022 Cacodemon345
|
|
* Copyright 2021-2022 Teemu Korhonen
|
|
*/
|
|
#include <QApplication>
|
|
#include <QSurfaceFormat>
|
|
#include <QDebug>
|
|
#include <QElapsedTimer>
|
|
#include <QThread>
|
|
#include <QTimer>
|
|
#include <QTranslator>
|
|
#include <QDirIterator>
|
|
#include <QLibraryInfo>
|
|
#include <QString>
|
|
#include <QFont>
|
|
#include <QDialog>
|
|
#include <QMessageBox>
|
|
|
|
#ifdef QT_STATIC
|
|
/* Static builds need plugin imports */
|
|
# include <QtPlugin>
|
|
Q_IMPORT_PLUGIN(QICOPlugin)
|
|
# ifdef Q_OS_WINDOWS
|
|
Q_IMPORT_PLUGIN(QWindowsIntegrationPlugin)
|
|
Q_IMPORT_PLUGIN(QWindowsVistaStylePlugin)
|
|
# endif
|
|
#endif
|
|
|
|
#ifdef Q_OS_WINDOWS
|
|
# include "qt_winrawinputfilter.hpp"
|
|
# include "qt_winmanagerfilter.hpp"
|
|
# include <86box/win.h>
|
|
# include <shobjidl.h>
|
|
#endif
|
|
|
|
extern "C" {
|
|
#include <86box/86box.h>
|
|
#include <86box/config.h>
|
|
#include <86box/plat.h>
|
|
#include <86box/ui.h>
|
|
#include <86box/video.h>
|
|
#ifdef DISCORD
|
|
# include <86box/discord.h>
|
|
#endif
|
|
#include <86box/gdbstub.h>
|
|
#include <86box/version.h>
|
|
}
|
|
|
|
#include <thread>
|
|
#include <iostream>
|
|
#include <memory>
|
|
|
|
#include "qt_mainwindow.hpp"
|
|
#include "qt_progsettings.hpp"
|
|
#include "qt_settings.hpp"
|
|
#include "cocoa_mouse.hpp"
|
|
#include "qt_styleoverride.hpp"
|
|
#include "qt_unixmanagerfilter.hpp"
|
|
|
|
// Void Cast
|
|
#define VC(x) const_cast<wchar_t *>(x)
|
|
|
|
extern QElapsedTimer elapsed_timer;
|
|
extern MainWindow *main_window;
|
|
|
|
extern "C" {
|
|
#include <86box/timer.h>
|
|
#include <86box/nvr.h>
|
|
extern int qt_nvr_save(void);
|
|
}
|
|
|
|
void qt_set_sequence_auto_mnemonic(bool b);
|
|
|
|
void
|
|
main_thread_fn()
|
|
{
|
|
uint64_t old_time;
|
|
uint64_t new_time;
|
|
int drawits;
|
|
int frames;
|
|
|
|
QThread::currentThread()->setPriority(QThread::HighestPriority);
|
|
plat_set_thread_name(NULL, "main_thread_fn");
|
|
framecountx = 0;
|
|
// title_update = 1;
|
|
old_time = elapsed_timer.elapsed();
|
|
drawits = frames = 0;
|
|
while (!is_quit && cpu_thread_run) {
|
|
/* See if it is time to run a frame of code. */
|
|
new_time = elapsed_timer.elapsed();
|
|
#ifdef USE_GDBSTUB
|
|
if (gdbstub_next_asap && (drawits <= 0))
|
|
drawits = 10;
|
|
else
|
|
#endif
|
|
drawits += (new_time - old_time);
|
|
old_time = new_time;
|
|
if (drawits > 0 && !dopause) {
|
|
/* Yes, so do one frame now. */
|
|
drawits -= 10;
|
|
if (drawits > 50)
|
|
drawits = 0;
|
|
|
|
#ifdef USE_INSTRUMENT
|
|
uint64_t start_time = elapsed_timer.nsecsElapsed();
|
|
#endif
|
|
/* Run a block of code. */
|
|
pc_run();
|
|
|
|
#ifdef USE_INSTRUMENT
|
|
if (instru_enabled) {
|
|
uint64_t elapsed_us = (elapsed_timer.nsecsElapsed() - start_time) / 1000;
|
|
uint64_t total_elapsed_ms = (uint64_t) ((double) tsc / cpu_s->rspeed * 1000);
|
|
printf("[instrument] %llu, %llu\n", total_elapsed_ms, elapsed_us);
|
|
if (instru_run_ms && total_elapsed_ms >= instru_run_ms)
|
|
break;
|
|
}
|
|
#endif
|
|
/* Every 200 frames we save the machine status. */
|
|
if (++frames >= 200 && nvr_dosave) {
|
|
qt_nvr_save();
|
|
nvr_dosave = 0;
|
|
frames = 0;
|
|
}
|
|
} else {
|
|
/* Just so we dont overload the host OS. */
|
|
if (dopause)
|
|
ack_pause();
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(1));
|
|
}
|
|
}
|
|
|
|
is_quit = 1;
|
|
if (gfxcard[1]) {
|
|
ui_deinit_monitor(1);
|
|
std::this_thread::sleep_for(std::chrono::milliseconds(500));
|
|
}
|
|
QTimer::singleShot(0, QApplication::instance(), []() { QApplication::processEvents(); QApplication::instance()->quit(); });
|
|
}
|
|
|
|
static std::thread *main_thread;
|
|
|
|
int
|
|
main(int argc, char *argv[])
|
|
{
|
|
#if QT_VERSION < QT_VERSION_CHECK(6, 0, 0)
|
|
QApplication::setAttribute(Qt::AA_DisableHighDpiScaling, false);
|
|
QApplication::setAttribute(Qt::AA_UseHighDpiPixmaps);
|
|
QApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
|
|
#endif
|
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
|
QApplication::setHighDpiScaleFactorRoundingPolicy(Qt::HighDpiScaleFactorRoundingPolicy::PassThrough);
|
|
#endif
|
|
QApplication app(argc, argv);
|
|
QLocale::setDefault(QLocale::C);
|
|
|
|
qt_set_sequence_auto_mnemonic(false);
|
|
Q_INIT_RESOURCE(qt_resources);
|
|
Q_INIT_RESOURCE(qt_translations);
|
|
QSurfaceFormat fmt = QSurfaceFormat::defaultFormat();
|
|
fmt.setSwapInterval(0);
|
|
QSurfaceFormat::setDefaultFormat(fmt);
|
|
app.setStyle(new StyleOverride());
|
|
|
|
#ifdef __APPLE__
|
|
CocoaEventFilter cocoafilter;
|
|
app.installNativeEventFilter(&cocoafilter);
|
|
#endif
|
|
elapsed_timer.start();
|
|
|
|
if (!pc_init(argc, argv)) {
|
|
return 0;
|
|
}
|
|
|
|
bool startMaximized = window_remember && monitor_settings[0].mon_window_maximized;
|
|
fprintf(stderr, "Qt: version %s, platform \"%s\"\n", qVersion(), QApplication::platformName().toUtf8().data());
|
|
ProgSettings::loadTranslators(&app);
|
|
#ifdef Q_OS_WINDOWS
|
|
auto font_name = QObject::tr("FONT_NAME");
|
|
auto font_size = QObject::tr("FONT_SIZE");
|
|
QApplication::setFont(QFont(font_name, font_size.toInt()));
|
|
SetCurrentProcessExplicitAppUserModelID(L"86Box.86Box");
|
|
#endif
|
|
|
|
#ifndef Q_OS_MACOS
|
|
# ifdef RELEASE_BUILD
|
|
app.setWindowIcon(QIcon(":/settings/qt/icons/86Box-green.ico"));
|
|
# elif defined ALPHA_BUILD
|
|
app.setWindowIcon(QIcon(":/settings/qt/icons/86Box-red.ico"));
|
|
# elif defined BETA_BUILD
|
|
app.setWindowIcon(QIcon(":/settings/qt/icons/86Box-yellow.ico"));
|
|
# else
|
|
app.setWindowIcon(QIcon(":/settings/qt/icons/86Box-gray.ico"));
|
|
# endif
|
|
|
|
# ifdef Q_OS_UNIX
|
|
app.setDesktopFileName("net.86box.86Box");
|
|
# endif
|
|
#endif
|
|
|
|
if (!pc_init_modules()) {
|
|
ui_msgbox_header(MBX_FATAL, (void *) IDS_2121, (void *) IDS_2056);
|
|
return 6;
|
|
}
|
|
|
|
#ifdef Q_OS_WINDOWS
|
|
# if !defined(EMU_BUILD_NUM) || (EMU_BUILD_NUM != 5624)
|
|
HWND winbox = FindWindowW(L"TWinBoxMain", NULL);
|
|
if (winbox &&
|
|
FindWindowExW(winbox, NULL, L"TToolBar", NULL) &&
|
|
FindWindowExW(winbox, NULL, L"TListBox", NULL) &&
|
|
FindWindowExW(winbox, NULL, L"TStatusBar", NULL) &&
|
|
(winbox = FindWindowExW(winbox, NULL, L"TPageControl", NULL)) && /* holds a TTabSheet even on VM pages */
|
|
FindWindowExW(winbox, NULL, L"TTabSheet", NULL))
|
|
# endif
|
|
{
|
|
QMessageBox warningbox(QMessageBox::Icon::Warning, QObject::tr("WinBox is no longer supported"),
|
|
QObject::tr("Development of the WinBox manager stopped in 2022 due to a lack of maintainers. As we direct our efforts towards making 86Box even better, we have made the decision to no longer support WinBox as a manager.\n\nNo further updates will be provided through WinBox, and you may encounter incorrect behavior should you continue using it with newer versions of 86Box. Any bug reports related to WinBox behavior will be closed as invalid.\n\nGo to 86box.net for a list of other managers you can use."),
|
|
QMessageBox::NoButton);
|
|
warningbox.addButton(QObject::tr("Continue"), QMessageBox::AcceptRole);
|
|
warningbox.addButton(QObject::tr("Exit"), QMessageBox::RejectRole);
|
|
warningbox.exec();
|
|
if (warningbox.result() == QDialog::Accepted)
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
if (settings_only) {
|
|
Settings settings;
|
|
if (settings.exec() == QDialog::Accepted) {
|
|
settings.save();
|
|
config_save();
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* Warn the user about unsupported configs */
|
|
if (cpu_override) {
|
|
QMessageBox warningbox(QMessageBox::Icon::Warning, QObject::tr("You are loading an unsupported configuration"),
|
|
QObject::tr("CPU type filtering based on selected machine is disabled for this emulated machine.\n\nThis makes it possible to choose a CPU that is otherwise incompatible with the selected machine. However, you may run into incompatibilities with the machine BIOS or other software.\n\nEnabling this setting is not officially supported and any bug reports filed may be closed as invalid."),
|
|
QMessageBox::NoButton);
|
|
warningbox.addButton(QObject::tr("Continue"), QMessageBox::AcceptRole);
|
|
warningbox.addButton(QObject::tr("Exit"), QMessageBox::RejectRole);
|
|
warningbox.exec();
|
|
if (warningbox.result() == QDialog::Accepted)
|
|
return 0;
|
|
}
|
|
|
|
#ifdef DISCORD
|
|
discord_load();
|
|
#endif
|
|
|
|
main_window = new MainWindow();
|
|
if (startMaximized) {
|
|
main_window->showMaximized();
|
|
} else {
|
|
main_window->show();
|
|
}
|
|
|
|
app.installEventFilter(main_window);
|
|
|
|
#ifdef Q_OS_WINDOWS
|
|
/* Setup VM-manager messages */
|
|
std::unique_ptr<WindowsManagerFilter> wmfilter;
|
|
if (source_hwnd) {
|
|
HWND main_hwnd = (HWND) main_window->winId();
|
|
|
|
wmfilter.reset(new WindowsManagerFilter());
|
|
QObject::connect(wmfilter.get(), &WindowsManagerFilter::showsettings, main_window, &MainWindow::showSettings);
|
|
QObject::connect(wmfilter.get(), &WindowsManagerFilter::pause, main_window, &MainWindow::togglePause);
|
|
QObject::connect(wmfilter.get(), &WindowsManagerFilter::reset, main_window, &MainWindow::hardReset);
|
|
QObject::connect(wmfilter.get(), &WindowsManagerFilter::request_shutdown, main_window, &MainWindow::close);
|
|
QObject::connect(wmfilter.get(), &WindowsManagerFilter::force_shutdown, []() {
|
|
do_stop();
|
|
emit main_window->close();
|
|
});
|
|
QObject::connect(wmfilter.get(), &WindowsManagerFilter::ctrlaltdel, []() { pc_send_cad(); });
|
|
QObject::connect(wmfilter.get(), &WindowsManagerFilter::dialogstatus, [main_hwnd](bool open) {
|
|
PostMessage((HWND) (uintptr_t) source_hwnd, WM_SENDDLGSTATUS, (WPARAM) (open ? 1 : 0), (LPARAM) main_hwnd);
|
|
});
|
|
|
|
/* Native filter to catch VM-managers commands */
|
|
app.installNativeEventFilter(wmfilter.get());
|
|
|
|
/* Filter to catch main window being blocked (by modal dialog) */
|
|
main_window->installEventFilter(wmfilter.get());
|
|
|
|
/* Send main window HWND to manager */
|
|
PostMessage((HWND) (uintptr_t) source_hwnd, WM_SENDHWND, (WPARAM) unique_id, (LPARAM) main_hwnd);
|
|
|
|
/* Send shutdown message to manager */
|
|
QObject::connect(&app, &QApplication::destroyed, [main_hwnd](QObject *) {
|
|
PostMessage((HWND) (uintptr_t) source_hwnd, WM_HAS_SHUTDOWN, (WPARAM) 0, (LPARAM) main_hwnd);
|
|
});
|
|
}
|
|
|
|
/* Setup raw input */
|
|
auto rawInputFilter = WindowsRawInputFilter::Register(main_window);
|
|
if (rawInputFilter) {
|
|
app.installNativeEventFilter(rawInputFilter.get());
|
|
main_window->setSendKeyboardInput(false);
|
|
}
|
|
#endif
|
|
|
|
UnixManagerSocket socket;
|
|
if (qgetenv("86BOX_MANAGER_SOCKET").size()) {
|
|
QObject::connect(&socket, &UnixManagerSocket::showsettings, main_window, &MainWindow::showSettings);
|
|
QObject::connect(&socket, &UnixManagerSocket::pause, main_window, &MainWindow::togglePause);
|
|
QObject::connect(&socket, &UnixManagerSocket::resetVM, main_window, &MainWindow::hardReset);
|
|
QObject::connect(&socket, &UnixManagerSocket::request_shutdown, main_window, &MainWindow::close);
|
|
QObject::connect(&socket, &UnixManagerSocket::force_shutdown, []() {
|
|
do_stop();
|
|
emit main_window->close();
|
|
});
|
|
QObject::connect(&socket, &UnixManagerSocket::ctrlaltdel, []() { pc_send_cad(); });
|
|
main_window->installEventFilter(&socket);
|
|
socket.connectToServer(qgetenv("86BOX_MANAGER_SOCKET"));
|
|
}
|
|
|
|
// pc_reset_hard_init();
|
|
|
|
QTimer onesec;
|
|
QObject::connect(&onesec, &QTimer::timeout, &app, [] {
|
|
pc_onesec();
|
|
});
|
|
onesec.setTimerType(Qt::PreciseTimer);
|
|
onesec.start(1000);
|
|
|
|
#ifdef DISCORD
|
|
QTimer discordupdate;
|
|
if (discord_loaded) {
|
|
QTimer::singleShot(1000, &app, [] {
|
|
if (enable_discord) {
|
|
discord_init();
|
|
discord_update_activity(dopause);
|
|
} else
|
|
discord_close();
|
|
});
|
|
QObject::connect(&discordupdate, &QTimer::timeout, &app, [] {
|
|
discord_run_callbacks();
|
|
});
|
|
discordupdate.start(1000);
|
|
}
|
|
#endif
|
|
|
|
/* Initialize the rendering window, or fullscreen. */
|
|
QTimer::singleShot(0, &app, [] {
|
|
pc_reset_hard_init();
|
|
main_thread = new std::thread(main_thread_fn);
|
|
|
|
/* Set the PAUSE mode depending on the renderer. */
|
|
#ifdef USE_VNC
|
|
if (vnc_enabled && vid_api != 5)
|
|
plat_pause(1);
|
|
else
|
|
#endif
|
|
plat_pause(0);
|
|
});
|
|
|
|
auto ret = app.exec();
|
|
cpu_thread_run = 0;
|
|
main_thread->join();
|
|
pc_close(nullptr);
|
|
endblit();
|
|
|
|
socket.close();
|
|
return ret;
|
|
}
|