dingusppc/main.cpp

356 lines
12 KiB
C++
Raw Normal View History

/*
DingusPPC - The Experimental PowerPC Macintosh emulator
2023-04-01 17:20:53 +02:00
Copyright (C) 2018-23 divingkatae and maximum
(theweirdo) spatium
(Contact divingkatae#1017 or powermax#2286 on Discord for more info)
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 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
2020-05-12 23:55:45 +05:00
// The main runfile - main.cpp
// This is where the magic begins
2023-04-01 17:20:53 +02:00
#include <core/hostevents.h>
#include <core/timermanager.h>
#include <cpu/ppc/ppcdisasm.h>
#include <cpu/ppc/ppcemu.h>
#include <cpu/ppc/ppcmmu.h>
2023-04-01 17:20:53 +02:00
#include <debugger/debugger.h>
#include <machines/machinebase.h>
2023-04-01 17:20:53 +02:00
#include <machines/machinefactory.h>
#include <utils/profiler.h>
#include <main.h>
2023-04-01 17:20:53 +02:00
2020-05-12 23:55:45 +05:00
#include <cinttypes>
2021-08-19 11:29:44 +02:00
#include <csignal>
2020-05-12 23:55:45 +05:00
#include <cstring>
#include <iostream>
2021-09-16 00:46:38 +02:00
#include <CLI11.hpp>
#include <loguru.hpp>
#ifdef _WIN32
#include <direct.h>
#else
#include <unistd.h>
#endif
using namespace std;
2024-05-08 07:07:32 -07:00
static void sigint_handler(int signum) {
power_on = false;
power_off_reason = po_signal_interrupt;
2021-08-19 11:29:44 +02:00
}
2024-05-08 07:07:32 -07:00
static void sigabrt_handler(int signum) {
LOG_F(INFO, "Shutting down...");
delete gMachineObj.release();
cleanup();
}
static string appDescription = string(
"\nDingusPPC - Alpha 1.01 (10/31/2024) "
2024-04-15 07:01:51 -07:00
"\nWritten by divingkatae, maximumspatium, "
"\njoevt, mihaip, kkaisershot, et. al. "
2024-04-15 07:01:51 -07:00
"\n(c) 2018-2024 The DingusPPC Dev Team. "
"\nThis is a build intended for testing. "
"\nUse at your own discretion. "
"\n"
);
2020-08-25 20:07:02 -07:00
/// Check for an existing directory (returns error message if check fails)
class WorkingDirectoryValidator : public CLI::detail::ExistingDirectoryValidator {
public:
WorkingDirectoryValidator() {
func_ = [](std::string& filename) {
std::string result = CLI::ExistingDirectory.operator()(filename);
if (result.empty()) {
#ifdef _WIN32
_chdir(filename.c_str());
#else
chdir(filename.c_str());
#endif
}
return result;
};
}
};
const WorkingDirectoryValidator WorkingDirectory;
void run_machine(std::string machine_str, char *rom_data, size_t rom_size, uint32_t execution_mode, uint32_t profiling_interval_ms);
2020-05-12 23:55:45 +05:00
int main(int argc, char** argv) {
2020-08-22 11:05:08 -07:00
2023-06-19 22:36:27 -07:00
uint32_t execution_mode = interpreter;
2020-08-22 11:05:08 -07:00
CLI::App app(appDescription);
2020-09-20 23:25:29 +02:00
app.allow_windows_style_options(); /* we want Windows-style options */
app.allow_extras();
bool realtime_enabled = false;
bool debugger_enabled = false;
2020-09-20 23:25:29 +02:00
string bootrom_path("bootrom.bin");
string working_directory_path(".");
2020-09-20 23:25:29 +02:00
auto execution_mode_group = app.add_option_group("execution mode")
->require_option(-1);
execution_mode_group->add_flag("-r,--realtime", realtime_enabled,
2020-09-20 23:25:29 +02:00
"Run the emulator in real-time");
execution_mode_group->add_flag("-d,--debugger", debugger_enabled,
2020-09-20 23:25:29 +02:00
"Enter the built-in debugger");
app.add_option("-w,--workingdir", working_directory_path, "Specifies working directory")
->check(WorkingDirectory);
2020-09-20 23:25:29 +02:00
app.add_option("-b,--bootrom", bootrom_path, "Specifies BootROM path")
->check(CLI::ExistingFile);
app.add_flag("--deterministic", is_deterministic,
"Make execution deterministic");
2020-09-20 23:25:29 +02:00
bool log_to_stderr = false;
loguru::Verbosity log_verbosity = loguru::Verbosity_INFO;
bool log_no_uptime = false;
app.add_flag("--log-to-stderr", log_to_stderr,
"Send internal logging to stderr (instead of dingusppc.log)");
app.add_flag("--log-verbosity", log_verbosity,
"Adjust logging verbosity (default is 0 a.k.a. INFO)")
->check(CLI::Number);
app.add_flag("--log-no-uptime", log_no_uptime,
"Disable the uptime preamble of logged messages");
uint32_t profiling_interval_ms = 0;
#ifdef CPU_PROFILING
app.add_option("--profiling-interval-ms", profiling_interval_ms,
"Specifies periodic interval (in ms) at which to output CPU profiling information");
#endif
string machine_str;
2020-09-20 23:25:29 +02:00
CLI::Option* machine_opt = app.add_option("-m,--machine",
machine_str, "Specify machine ID");
auto list_cmd = app.add_subcommand("list",
2020-09-20 23:25:29 +02:00
"Display available machine configurations and exit");
string sub_arg;
list_cmd->add_option("machines", sub_arg, "List supported machines");
list_cmd->add_option("properties", sub_arg, "List available properties");
2020-09-20 23:25:29 +02:00
CLI11_PARSE(app, argc, argv);
if (*list_cmd) {
2020-10-13 04:24:54 +02:00
if (sub_arg == "machines") {
MachineFactory::list_machines();
2020-10-13 04:24:54 +02:00
} else if (sub_arg == "properties") {
MachineFactory::list_properties();
2020-10-13 04:24:54 +02:00
} else {
cout << "Unknown list subcommand " << sub_arg << endl;
}
return 0;
}
2020-09-20 23:25:29 +02:00
if (debugger_enabled) {
execution_mode = debugger;
2020-09-20 23:25:29 +02:00
}
/* initialize logging */
loguru::g_preamble_date = false;
loguru::g_preamble_time = false;
loguru::g_preamble_thread = false;
loguru::g_preamble_uptime = !log_no_uptime;
2020-09-20 23:25:29 +02:00
if (execution_mode == interpreter && !log_to_stderr) {
2020-09-20 23:25:29 +02:00
loguru::g_stderr_verbosity = loguru::Verbosity_OFF;
loguru::init(argc, argv);
loguru::add_file("dingusppc.log", loguru::Append, log_verbosity);
2020-09-20 23:25:29 +02:00
} else {
loguru::g_stderr_verbosity = log_verbosity;
2020-09-20 23:25:29 +02:00
loguru::init(argc, argv);
}
auto rom_data = std::unique_ptr<char[]>(new char[4 * 1024 * 1024]);
memset(&rom_data[0], 0, 4 * 1024 * 1024);
size_t rom_size = MachineFactory::read_boot_rom(bootrom_path, &rom_data[0]);
if (!rom_size) {
return 1;
}
string machine_str_from_rom = MachineFactory::machine_name_from_rom(&rom_data[0], rom_size);
if (machine_str_from_rom.empty()) {
LOG_F(ERROR, "Could not autodetect machine from ROM.");
} else {
LOG_F(INFO, "Machine detected from ROM as: %s", machine_str_from_rom.c_str());
}
2020-09-20 23:25:29 +02:00
if (*machine_opt) {
LOG_F(INFO, "Machine option was passed in: %s", machine_str.c_str());
} else {
machine_str = machine_str_from_rom;
}
2020-09-20 23:25:29 +02:00
if (machine_str.empty()) {
LOG_F(ERROR, "Must specificy a machine or provide a supported ROM.");
return 1;
2020-09-20 23:25:29 +02:00
}
// Hook to allow properties to be read from the command-line, regardless
// of when they are registered.
MachineFactory::get_setting_value = [&](const std::string& name) -> std::optional<std::string> {
CLI::App sa;
sa.allow_extras();
std::string value;
sa.add_option("--" + name, value);
try {
sa.parse(app.remaining_for_passthrough());
} catch (const CLI::Error& e) {
ABORT_F("Cannot parse CLI: %s", e.get_name().c_str());
}
2020-09-20 23:25:29 +02:00
if (sa.count("--" + name) > 0) {
return value;
} else {
return std::nullopt;
}
};
2020-09-20 23:25:29 +02:00
if (MachineFactory::register_machine_settings(machine_str) < 0) {
return 1;
2020-09-20 23:25:29 +02:00
}
cout << "BootROM path: " << bootrom_path << endl;
cout << "Execution mode: " << execution_mode << endl;
if (is_deterministic) {
cout << "Using deterministic execution mode, input will be ignored." << endl;
}
2020-09-20 23:25:29 +02:00
if (!init()) {
LOG_F(ERROR, "Cannot initialize");
return 1;
}
2021-04-14 01:27:07 +02:00
// initialize global profiler object
gProfilerObj.reset(new Profiler());
// graceful handling of fatal errors
loguru::set_fatal_handler([](const loguru::Message& message) {
// Make sure the reason for the failure is visible (it may have been
// sent to the logfile only).
cerr << message.preamble << message.indentation << message.prefix << message.message << endl;
power_off_reason = po_enter_debugger;
2024-11-30 12:04:23 +01:00
DppcDebugger::get_instance()->enter_debugger();
// Ensure that NVRAM and other state is persisted before we terminate.
delete gMachineObj.release();
});
2021-08-19 11:29:44 +02:00
// redirect SIGINT to our own handler
signal(SIGINT, sigint_handler);
// redirect SIGABRT to our own handler
signal(SIGABRT, sigabrt_handler);
while (true) {
run_machine(machine_str, &rom_data[0], rom_size, execution_mode, profiling_interval_ms);
if (power_off_reason == po_restarting) {
LOG_F(INFO, "Restarting...");
power_on = true;
continue;
}
if (power_off_reason == po_shutting_down) {
if (execution_mode != debugger) {
LOG_F(INFO, "Shutdown.");
break;
}
LOG_F(INFO, "Shutdown...");
power_on = true;
continue;
}
break;
}
cleanup();
return 0;
}
void run_machine(std::string machine_str, char *rom_data, size_t rom_size, uint32_t execution_mode,
uint32_t
#ifdef CPU_PROFILING
profiling_interval_ms
#endif
) {
if (MachineFactory::create_machine_for_id(machine_str, rom_data, rom_size) < 0) {
return;
}
uint32_t deterministic_timer;
if (is_deterministic) {
EventManager::get_instance()->disable_input_handlers();
// Log the PC and instruction every second to make it easier to validate
// that execution is the same every time.
deterministic_timer = TimerManager::get_instance()->add_cyclic_timer(MSECS_TO_NSECS(1000), [] {
PPCDisasmContext ctx;
ctx.instr_code = ppc_read_instruction(mmu_translate_imem(ppc_state.pc));
ctx.instr_addr = ppc_state.pc;
ctx.simplified = false;
auto op_name = disassemble_single(&ctx);
LOG_F(INFO, "TS=%016llu PC=0x%08x executing %s", get_virt_time_ns(), ppc_state.pc, op_name.c_str());
});
}
2023-04-01 17:20:53 +02:00
// set up system wide event polling using
// default Macintosh polling rate of 11 ms
uint32_t event_timer = TimerManager::get_instance()->add_cyclic_timer(MSECS_TO_NSECS(11), [] {
2023-04-01 17:20:53 +02:00
EventManager::get_instance()->poll_events();
});
#ifdef CPU_PROFILING
uint32_t profiling_timer;
if (profiling_interval_ms > 0) {
profiling_timer = TimerManager::get_instance()->add_cyclic_timer(MSECS_TO_NSECS(profiling_interval_ms), [] {
gProfilerObj->print_profile("PPC_CPU");
});
}
#endif
2020-10-04 09:58:21 -07:00
switch (execution_mode) {
2023-06-19 22:36:27 -07:00
case interpreter:
power_off_reason = po_starting_up;
2024-11-30 12:04:23 +01:00
DppcDebugger::get_instance()->enter_debugger();
break;
2024-09-15 18:31:36 -07:00
case threaded_int:
power_off_reason = po_starting_up;
2024-11-30 12:04:23 +01:00
DppcDebugger::get_instance()->enter_debugger();
2024-09-15 18:31:36 -07:00
break;
2023-06-19 22:36:27 -07:00
case debugger:
power_off_reason = po_enter_debugger;
2024-11-30 12:04:23 +01:00
DppcDebugger::get_instance()->enter_debugger();
break;
default:
LOG_F(ERROR, "Invalid EXECUTION MODE");
return;
2020-09-20 23:25:29 +02:00
}
LOG_F(INFO, "Cleaning up...");
TimerManager::get_instance()->cancel_timer(event_timer);
#ifdef CPU_PROFILING
if (profiling_interval_ms > 0) {
TimerManager::get_instance()->cancel_timer(profiling_timer);
}
#endif
if (is_deterministic) {
TimerManager::get_instance()->cancel_timer(deterministic_timer);
}
EventManager::get_instance()->disconnect_handlers();
delete gMachineObj.release();
}