/* DingusPPC - The Experimental PowerPC Macintosh emulator Copyright (C) 2018-24 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 . */ #include #include #include "cpu/ppc/ppcemu.h" #include "cpu/ppc/ppcmmu.h" #include "devices/memctrl/mpc106.h" #include #include #if defined(PPC_BENCHMARKS) void ppc_exception_handler(Except_Type exception_type, uint32_t srr1_bits) { power_on = false; } #endif uint32_t cs_code[] = { 0x3863FFFC, 0x7C861671, 0x41820090, 0x70600002, 0x41E2001C, 0xA0030004, 0x3884FFFE, 0x38630002, 0x5486F0BF, 0x7CA50114, 0x41820070, 0x70C60003, 0x41820014, 0x7CC903A6, 0x84030004, 0x7CA50114, 0x4200FFF8, 0x5486E13F, 0x41820050, 0x80030004, 0x7CC903A6, 0x80C30008, 0x7CA50114, 0x80E3000C, 0x7CA53114, 0x85030010, 0x7CA53914, 0x42400028, 0x80030004, 0x7CA54114, 0x80C30008, 0x7CA50114, 0x80E3000C, 0x7CA53114, 0x85030010, 0x7CA53914, 0x4200FFE0, 0x7CA54114, 0x70800002, 0x41E20010, 0xA0030004, 0x38630002, 0x7CA50114, 0x70800001, 0x41E20010, 0x88030004, 0x5400402E, 0x7CA50114, 0x7C650194, /* 0x4E800020 */ 0x00005AF0 }; constexpr uint32_t test_size = 0x8000; // 0x7FFFFFFC is the max constexpr uint32_t test_samples = 200; constexpr uint32_t test_iterations = 5; int main(int argc, char** argv) { int i, j; /* initialize logging */ loguru::g_preamble_date = false; loguru::g_preamble_time = false; loguru::g_preamble_thread = false; loguru::g_stderr_verbosity = 0; loguru::init(argc, argv); MPC106* grackle_obj = new MPC106; /* we need some RAM */ if (!grackle_obj->add_ram_region(0, 0x1000 + test_size)) { LOG_F(ERROR, "Could not create RAM region"); delete(grackle_obj); return -1; } constexpr uint64_t tbr_freq = 16705000; ppc_cpu_init(grackle_obj, PPC_VER::MPC750, false, tbr_freq); /* load executable code into RAM at address 0 */ for (i = 0; i < sizeof(cs_code) / sizeof(cs_code[0]); i++) { mmu_write_vmem(i*4, cs_code[i]); } srand(0xCAFEBABE); LOG_F(INFO, "Test size: 0x%X", test_size); LOG_F(INFO, "First few bytes:"); bool did_lf = false; for (i = 0; i < test_size; i++) { uint8_t val = rand() % 256; mmu_write_vmem(0x1000+i, val); if (i < 64) { printf("%02x", val); did_lf = false; if (i % 32 == 31) { printf("\n"); did_lf = true; } } } if (!did_lf) printf("\n"); #if 0 /* prepare benchmark code execution */ ppc_state.pc = 0; ppc_state.gpr[3] = 0x1000; // buf ppc_state.gpr[4] = test_size; // len ppc_state.gpr[5] = 0; // sum enter_debugger(); #endif ppc_state.pc = 0; ppc_state.gpr[3] = 0x1000; // buf ppc_state.gpr[4] = test_size; // len ppc_state.gpr[5] = 0; // sum power_on = true; ppc_exec_until(0xC4); LOG_F(INFO, "Checksum: 0x%08X", ppc_state.gpr[3]); uint32_t checksum = ppc_state.gpr[3]; // run the clock once for cache fill etc. uint64_t overhead = -1; for (j = 0; j < test_samples; j++) { auto start_time = std::chrono::steady_clock::now(); auto end_time = std::chrono::steady_clock::now(); auto time_elapsed = std::chrono::duration_cast(end_time - start_time); if (time_elapsed.count() < overhead) overhead = time_elapsed.count(); } LOG_F(INFO, "Overhead Time: %lld ns", overhead); for (int theproc = 0; theproc < 2; theproc++) { LOG_F(INFO, "Doing %s", theproc ? "ppc_exec_until" : "ppc_exec"); for (i = 0; i < test_iterations; i++) { uint64_t best_sample = -1; for (j = 0; j < test_samples; j++) { ppc_state.pc = 0; ppc_state.gpr[3] = 0x1000; // buf ppc_state.gpr[4] = test_size; // len ppc_state.gpr[5] = 0; // sum power_on = true; auto start_time = std::chrono::steady_clock::now(); (theproc) ? ppc_exec_until(0xC4) : ppc_exec(); auto end_time = std::chrono::steady_clock::now(); auto time_elapsed = std::chrono::duration_cast(end_time - start_time); if (time_elapsed.count() < best_sample) best_sample = time_elapsed.count(); } if (ppc_state.gpr[3] != checksum) LOG_F(INFO, "Checksum: 0x%08X", ppc_state.gpr[3]); best_sample -= overhead; LOG_F(INFO, "(%d) %lld ns, %.4lf MiB/s", i+1, best_sample, 1E9 * test_size / (best_sample * 1024 * 1024)); } } delete(grackle_obj); return 0; }