mirror of
https://github.com/ksherlock/mpw.git
synced 2024-11-22 15:31:50 +00:00
735 lines
15 KiB
C++
735 lines
15 KiB
C++
// clang++ -c -std=c++11 -stdlib=libc++ -Wno-deprecated-declarations loader.cpp
|
|
/*
|
|
* Copyright (c) 2013, Kelvin W Sherlock
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, are permitted provided that the following conditions are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright notice, this
|
|
* list of conditions and the following disclaimer.
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice,
|
|
* this list of conditions and the following disclaimer in the documentation
|
|
* and/or other materials provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
* ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
*/
|
|
|
|
#include <cstdint>
|
|
#include <cctype>
|
|
#include <cstring>
|
|
#include <string>
|
|
#include <vector>
|
|
#include <chrono>
|
|
|
|
#include <sysexits.h>
|
|
#include <getopt.h>
|
|
#include <libgen.h>
|
|
#include <unistd.h>
|
|
#include <stdlib.h>
|
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <cxx/string_splitter.h>
|
|
|
|
#include <cpu/defs.h>
|
|
#include <cpu/CpuModule.h>
|
|
#include <cpu/fmem.h>
|
|
|
|
#include <toolbox/toolbox.h>
|
|
#include <toolbox/mm.h>
|
|
#include <toolbox/os.h>
|
|
#include <toolbox/loader.h>
|
|
|
|
#include <mpw/mpw.h>
|
|
|
|
#include <mplite/mplite.h>
|
|
|
|
#include <macos/sysequ.h>
|
|
#include <macos/traps.h>
|
|
|
|
#include "main.h"
|
|
#include "debugger.h"
|
|
|
|
|
|
|
|
Settings Flags;
|
|
|
|
const uint32_t kGlobalSize = 0x10000;
|
|
// retained to make debugging easier.
|
|
uint8_t *Memory = nullptr;
|
|
uint32_t MemorySize = 0;
|
|
|
|
|
|
uint8_t ReadByte(const void *data, uint32_t offset)
|
|
{
|
|
offset &= 0xffffff;
|
|
return ((uint8_t *)data)[offset];
|
|
}
|
|
|
|
uint16_t ReadWord(const void *data, uint32_t offset)
|
|
{
|
|
offset &= 0xffffff;
|
|
return (ReadByte(data, offset) << 8) | ReadByte(data, offset+1);
|
|
}
|
|
|
|
uint32_t ReadLong(const void *data, uint32_t offset)
|
|
{
|
|
offset &= 0xffffff;
|
|
return (ReadWord(data, offset) << 16) | ReadWord(data, offset+2);
|
|
}
|
|
|
|
void WriteByte(void *data, uint32_t offset, uint8_t value)
|
|
{
|
|
offset &= 0xffffff;
|
|
((uint8_t *)data)[offset] = value;
|
|
}
|
|
|
|
void WriteWord(void *data, uint32_t offset, uint16_t value)
|
|
{
|
|
offset &= 0xffffff;
|
|
|
|
((uint8_t *)data)[offset++] = value >> 8;
|
|
((uint8_t *)data)[offset++] = value;
|
|
}
|
|
|
|
void WriteLong(void *data, uint32_t offset, uint32_t value)
|
|
{
|
|
offset &= 0xffffff;
|
|
|
|
((uint8_t *)data)[offset++] = value >> 24;
|
|
((uint8_t *)data)[offset++] = value >> 16;
|
|
((uint8_t *)data)[offset++] = value >> 8;
|
|
((uint8_t *)data)[offset++] = value;
|
|
}
|
|
|
|
|
|
|
|
|
|
void GlobalInit()
|
|
{
|
|
// todo -- move this somewhere better.
|
|
|
|
|
|
// 0x031a - Lo3Bytes
|
|
memoryWriteLong(0x00ffffff, MacOS::Lo3Bytes);
|
|
|
|
// 0x0a02 - OneOne
|
|
memoryWriteLong(0x00010001, MacOS::OneOne);
|
|
|
|
// 0x0a06 - MinusOne
|
|
memoryWriteLong(0xffffffff, MacOS::MinusOne);
|
|
|
|
|
|
// 0x0130 -- ApplLimit
|
|
memoryWriteLong(Flags.memorySize - Flags.stackSize - 1, MacOS::ApplLimit);
|
|
memoryWriteLong(kGlobalSize, MacOS::ApplZone);
|
|
memoryWriteLong(Flags.memorySize - 1, MacOS::BufPtr);
|
|
|
|
//
|
|
memoryWriteLong(Flags.stackRange.first, MacOS::CurStackBase);
|
|
|
|
}
|
|
|
|
|
|
void CreateStack()
|
|
{
|
|
// allocate stack, set A7...
|
|
|
|
uint32_t address;
|
|
//uint16_t error;
|
|
|
|
// stack is at top of memory because some apps assume stack overflow if elsewhere.
|
|
|
|
address = Flags.memorySize - Flags.stackSize;
|
|
|
|
Flags.stackRange.first = address;
|
|
Flags.stackRange.second = address + Flags.stackSize;
|
|
|
|
// TODO -- is there a global for the max (min) stack pointer?
|
|
|
|
// address grows down
|
|
// -4 is for the return address.
|
|
cpuSetAReg(7, Flags.stackRange.second - 4);
|
|
// return address.
|
|
memoryWriteLong(MacOS::MinusOne, Flags.stackRange.second - 4); // MinusOne Global -- 0xffff ffff
|
|
}
|
|
|
|
|
|
|
|
void LogToolBox(uint32_t pc, uint16_t trap)
|
|
{
|
|
const char *name;
|
|
|
|
name = TrapName(trap);
|
|
|
|
if (name)
|
|
{
|
|
fprintf(stderr, "$%08X %-51s ; %04X\n", pc, name, trap);
|
|
}
|
|
else
|
|
{
|
|
fprintf(stderr, "$%08X Tool #$%04X ; %04X\n", pc, trap, trap);
|
|
}
|
|
}
|
|
|
|
void InstructionLogger()
|
|
{
|
|
|
|
static char strings[4][256];
|
|
for (unsigned j = 0; j < 4; ++j) strings[j][0] = 0;
|
|
|
|
uint32_t pc = cpuGetPC();
|
|
uint16_t opcode = ReadWord(Memory, pc);
|
|
|
|
if ((opcode & 0xf000) == 0xa000)
|
|
{
|
|
LogToolBox(pc, opcode);
|
|
return;
|
|
}
|
|
|
|
|
|
if (Flags.traceCPU)
|
|
{
|
|
cpuDisOpcode(pc, strings[0], strings[1], strings[2], strings[3]);
|
|
|
|
// address, data, instruction, operand
|
|
fprintf(stderr, "%s %-10s %-40s ; %s\n", strings[0], strings[2], strings[3], strings[1]);
|
|
|
|
// todo -- trace registers (only print changed registers?)
|
|
|
|
#if 0
|
|
if (pc >= 0x00010E94 && pc <= 0x00010FC0)
|
|
{
|
|
fprintf(stderr, "d7 = %08x\n", cpuGetDReg(7));
|
|
}
|
|
#endif
|
|
}
|
|
|
|
int mboffset = 0;
|
|
switch (opcode)
|
|
{
|
|
case 0x4E75: // rts
|
|
case 0x4ED0: // jmp (a0)
|
|
mboffset = 2;
|
|
break;
|
|
case 0x4E74: // rtd #
|
|
mboffset = 4;
|
|
break;
|
|
}
|
|
|
|
if (mboffset) // RTS or JMP (A0)
|
|
{
|
|
pc += mboffset;
|
|
// check for MacsBug name after rts.
|
|
std::string s;
|
|
unsigned b = memoryReadByte(pc);
|
|
if (b >= 0x80 && b <= 0x9f)
|
|
{
|
|
b -= 0x80;
|
|
pc++;
|
|
|
|
if (!b) b = memoryReadByte(pc++);
|
|
|
|
s.reserve(b);
|
|
for (unsigned i = 0; i < b; ++i)
|
|
{
|
|
s.push_back(memoryReadByte(pc++));
|
|
}
|
|
fprintf(stderr, "%s\n\n", s.c_str());
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
void MemoryLogger(uint32_t address, int size, int readWrite, uint32_t value)
|
|
{
|
|
if (address < kGlobalSize)
|
|
{
|
|
const char *name = GlobalName(address);
|
|
if (!name) name = "unknown";
|
|
|
|
fprintf(stderr, "%-20s %08x - ", name, address);
|
|
|
|
if (!readWrite)
|
|
{
|
|
switch(size)
|
|
{
|
|
case 1:
|
|
value = ReadByte(Memory, address);
|
|
break;
|
|
case 2:
|
|
value = ReadWord(Memory, address);
|
|
break;
|
|
case 4:
|
|
value = ReadLong(Memory, address);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
// todo -- for write, display previous value?
|
|
fprintf(stderr, " %s %d bytes", readWrite ? "write" : "read ", size);
|
|
switch(size)
|
|
{
|
|
case 1:
|
|
fprintf(stderr, " [%02x]\n", value);
|
|
break;
|
|
case 2:
|
|
fprintf(stderr, " [%04x]\n", value);
|
|
break;
|
|
case 3:
|
|
fprintf(stderr, " [%06x]\n", value);
|
|
break;
|
|
case 4:
|
|
fprintf(stderr, " [%08x]\n", value);
|
|
break;
|
|
default:
|
|
fprintf(stderr, "\n");
|
|
break;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
void MidInstructionExceptionFunc()
|
|
{
|
|
// todo - cpu exception?
|
|
fprintf(stderr, "Mid Instruction Exception!\n");
|
|
//throw std::runtime_error::runtime_error("mid instruction exception");
|
|
}
|
|
|
|
|
|
#define MPW_VERSION "0.8.0"
|
|
void help()
|
|
{
|
|
printf("MPW " MPW_VERSION "\n");
|
|
printf("Usage: mpw [options] utility ...\n");
|
|
printf("\n");
|
|
printf(" --help display usage information\n");
|
|
printf(" --trace-cpu print cpu information\n");
|
|
printf(" --trace-macsbug print macsbug names\n");
|
|
printf(" --trace-toolbox print toolbox calls\n");
|
|
printf(" --trace-mpw print mpw calls\n");
|
|
printf(" --memory-stats print memory usage information\n");
|
|
printf(" --ram=<number> set the ram size. Default=16M\n");
|
|
printf(" --stack=<number> set the stack size. Default=8K\n");
|
|
printf("\n");
|
|
}
|
|
|
|
bool parse_number(const char *input, uint32_t *dest)
|
|
{
|
|
char *end;
|
|
long value;
|
|
int base = 0;
|
|
|
|
// octal is dumb so don't allow it.
|
|
|
|
while (isspace(*input)) ++input;
|
|
if (*input == '0' && isdigit(input[1])) base = 10;
|
|
|
|
errno = 0;
|
|
value = strtol(input, &end, base);
|
|
|
|
if (errno || value < 0 || input == end)
|
|
{
|
|
fprintf(stderr, "%s - invalid input\n", input);
|
|
return false;
|
|
}
|
|
|
|
// M/K
|
|
if (*end)
|
|
{
|
|
int old = value;
|
|
if (strcasecmp(end, "M") == 0)
|
|
value *= 1024 * 1024;
|
|
else if (strcasecmp(end, "K") == 0)
|
|
value *= 1024;
|
|
else
|
|
{
|
|
fprintf(stderr, "%s - invalid input\n", input);
|
|
return false;
|
|
}
|
|
if (value < old)
|
|
{
|
|
// overflow
|
|
fprintf(stderr, "%s - invalid input\n", input);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
if (dest) *dest = value;
|
|
return true;
|
|
}
|
|
|
|
bool file_exists(const std::string & name)
|
|
{
|
|
struct stat st;
|
|
|
|
return ::stat(name.c_str(), &st) == 0 && S_ISREG(st.st_mode);
|
|
}
|
|
|
|
std::string old_find_exe(const std::string &name)
|
|
{
|
|
if (file_exists(name)) return name;
|
|
|
|
// if name is a path, then it doesn't exist.
|
|
if (name.find('/') != name.npos) return std::string();
|
|
|
|
std::string path = MPW::RootDir();
|
|
if (path.empty()) return path;
|
|
|
|
|
|
if (path.back() != '/') path.push_back('/');
|
|
path.append("Tools/");
|
|
path.append(name);
|
|
|
|
if (file_exists(path)) return path;
|
|
|
|
return std::string();
|
|
|
|
}
|
|
|
|
|
|
// this needs to run *after* the MPW environment variables are loaded.
|
|
std::string find_exe(const std::string &name)
|
|
{
|
|
|
|
// if this is an absolute or relative name, return as-is.
|
|
|
|
if (name.find(':') != name.npos) {
|
|
std::string path = ToolBox::MacToUnix(name);
|
|
if (file_exists(path)) return path;
|
|
return "";
|
|
}
|
|
|
|
if (name.find('/') != name.npos) {
|
|
if (file_exists(name)) return name;
|
|
return "";
|
|
}
|
|
|
|
// otherwise, check the Commands variable for locations.
|
|
std::string commands = MPW::GetEnv("Commands");
|
|
if (commands.empty()) return old_find_exe(name);
|
|
|
|
|
|
// string is , separated, possibly in MacOS format.
|
|
|
|
for (auto iter = string_splitter(commands, ','); iter; ++iter)
|
|
{
|
|
if (iter->empty()) continue;
|
|
std::string path = *iter;
|
|
|
|
// convert to unix.
|
|
path = ToolBox::MacToUnix(path);
|
|
// should always have a length...
|
|
if (path.length() && path.back() != '/') path.push_back('/');
|
|
path.append(name);
|
|
if (file_exists(path)) return path;
|
|
}
|
|
|
|
return "";
|
|
}
|
|
|
|
|
|
void MainLoop()
|
|
{
|
|
#if 0
|
|
auto begin_emu_time = std::chrono::high_resolution_clock::now();
|
|
fprintf(stderr, "Begin Emulation Time: %20lld\n", (begin_emu_time - start_time).count());
|
|
#endif
|
|
|
|
|
|
uint64_t cycles = 0;
|
|
for (;;)
|
|
{
|
|
uint32_t pc = cpuGetPC();
|
|
uint32_t sp = cpuGetAReg(7);
|
|
|
|
if (pc == 0x00000000)
|
|
{
|
|
fprintf(stderr, "Exiting - PC = 0\n");
|
|
exit(EX_SOFTWARE);
|
|
}
|
|
|
|
if (sp < Flags.stackRange.first)
|
|
{
|
|
fprintf(stderr, "Stack overflow error - please increase the stack size (--stack=size)\n");
|
|
fprintf(stderr, "Current stack size is 0x%06x\n", Flags.stackSize);
|
|
exit(EX_SOFTWARE);
|
|
}
|
|
|
|
if (sp > Flags.stackRange.second)
|
|
{
|
|
fprintf(stderr, "Stack underflow error\n");
|
|
exit(EX_SOFTWARE);
|
|
}
|
|
|
|
|
|
if (cpuGetStop()) break; // will this also be set by an interrupt?
|
|
|
|
|
|
#ifndef CPU_INSTRUCTION_LOGGING
|
|
if (Flags.traceCPU || Flags.traceMacsbug)
|
|
{
|
|
InstructionLogger();
|
|
}
|
|
#endif
|
|
|
|
cycles += cpuExecuteInstruction();
|
|
}
|
|
|
|
#if 0
|
|
auto end_emu_time = std::chrono::high_resolution_clock::now();
|
|
fprintf(stderr, " End Emulation Time: %20lld\n", (end_emu_time - start_time).count());
|
|
fprintf(stderr, " Cycles: %20lld\n", cycles);
|
|
#endif
|
|
|
|
|
|
}
|
|
|
|
int main(int argc, char **argv)
|
|
{
|
|
// getopt...
|
|
|
|
enum
|
|
{
|
|
kTraceCPU = 1,
|
|
kTraceMacsBug,
|
|
kTraceGlobals,
|
|
kTraceToolBox,
|
|
kTraceMPW,
|
|
kDebugger,
|
|
kMemoryStats,
|
|
kShell,
|
|
};
|
|
static struct option LongOpts[] =
|
|
{
|
|
{ "ram",required_argument, NULL, 'r' },
|
|
{ "stack", required_argument, NULL, 's' },
|
|
{ "machine", required_argument, NULL, 'm' },
|
|
{ "trace-cpu", no_argument, NULL, kTraceCPU },
|
|
{ "trace-macsbug", no_argument, NULL, kTraceMacsBug },
|
|
{ "trace-globals", no_argument, NULL, kTraceGlobals },
|
|
{ "trace-toolbox", no_argument, NULL, kTraceToolBox },
|
|
{ "trace-tools", no_argument, NULL, kTraceToolBox },
|
|
{ "trace-mpw", no_argument, NULL, kTraceMPW },
|
|
|
|
{ "debug", no_argument, NULL, kDebugger },
|
|
{ "debugger", no_argument, NULL, kDebugger },
|
|
|
|
{ "memory-stats", no_argument, NULL, kMemoryStats },
|
|
|
|
{ "help", no_argument, NULL, 'h' },
|
|
{ "version", no_argument, NULL, 'V' },
|
|
{ "shell", no_argument, NULL, kShell },
|
|
{ NULL, 0, NULL, 0 }
|
|
};
|
|
|
|
//auto start_time = std::chrono::high_resolution_clock::now();
|
|
|
|
std::vector<std::string> defines;
|
|
|
|
int c;
|
|
while ((c = getopt_long(argc, argv, "+hVm:r:s:D:", LongOpts, NULL)) != -1)
|
|
{
|
|
switch(c)
|
|
{
|
|
case kTraceCPU:
|
|
Flags.traceCPU = true;
|
|
break;
|
|
|
|
case kTraceMacsBug:
|
|
Flags.traceMacsbug = true;
|
|
break;
|
|
|
|
case kTraceGlobals:
|
|
Flags.traceGlobals = true;
|
|
break;
|
|
|
|
case kTraceToolBox:
|
|
Flags.traceToolBox = true;
|
|
break;
|
|
|
|
case kTraceMPW:
|
|
Flags.traceMPW = true;
|
|
break;
|
|
|
|
case kMemoryStats:
|
|
Flags.memoryStats = true;
|
|
break;
|
|
|
|
case kDebugger:
|
|
Flags.debugger = true;
|
|
break;
|
|
|
|
case kShell:
|
|
break;
|
|
|
|
case 'm':
|
|
if (!parse_number(optarg, &Flags.machine))
|
|
exit(EX_CONFIG);
|
|
break;
|
|
|
|
case 'r':
|
|
if (!parse_number(optarg, &Flags.memorySize))
|
|
exit(EX_CONFIG);
|
|
break;
|
|
|
|
case 's':
|
|
if (!parse_number(optarg, &Flags.stackSize))
|
|
exit(EX_CONFIG);
|
|
break;
|
|
|
|
case 'D':
|
|
defines.push_back(optarg);
|
|
break;
|
|
|
|
case ':':
|
|
case '?':
|
|
help();
|
|
exit(EX_USAGE);
|
|
break;
|
|
|
|
case 'h':
|
|
help();
|
|
exit(EX_OK);
|
|
break;
|
|
|
|
case 'V':
|
|
printf("mpw version " MPW_VERSION "\n");
|
|
exit(EX_OK);
|
|
break;
|
|
}
|
|
|
|
}
|
|
|
|
argc -= optind;
|
|
argv += optind;
|
|
|
|
if (!argc)
|
|
{
|
|
help();
|
|
exit(EX_USAGE);
|
|
}
|
|
|
|
Flags.stackSize = (Flags.stackSize + 0xff) & ~0xff;
|
|
Flags.memorySize = (Flags.memorySize + 0xff) & ~0xff;
|
|
|
|
if (Flags.stackSize < 0x100)
|
|
{
|
|
fprintf(stderr, "Invalid stack size\n");
|
|
exit(EX_CONFIG);
|
|
}
|
|
|
|
if (Flags.memorySize < 0x200)
|
|
{
|
|
fprintf(stderr, "Invalid ram size\n");
|
|
exit(EX_CONFIG);
|
|
}
|
|
|
|
if (Flags.stackSize >= Flags.memorySize)
|
|
{
|
|
fprintf(stderr, "Invalid stack/ram size\n");
|
|
exit(EX_CONFIG);
|
|
}
|
|
|
|
|
|
|
|
MPW::InitEnvironment(defines);
|
|
|
|
std::string command(argv[0]); // InitMPW updates argv...
|
|
command = find_exe(command);
|
|
if (command.empty())
|
|
{
|
|
std::string mpw = MPW::RootDir();
|
|
fprintf(stderr, "Unable to find command %s\n", argv[0]);
|
|
fprintf(stderr, "$MPW = %s\n", mpw.c_str());
|
|
fprintf(stderr, "$Commands = %s\n", MPW::GetEnv("Commands").c_str());
|
|
exit(EX_USAGE);
|
|
}
|
|
argv[0] = ::strdup(command.c_str()); // hmm.. could setenv(mpw_command) instead.
|
|
|
|
|
|
// move to CreateRam()
|
|
Memory = Flags.memory = new uint8_t[Flags.memorySize];
|
|
MemorySize = Flags.memorySize;
|
|
|
|
|
|
/// ahhh... need to set PC after memory.
|
|
// for pre-fetch.
|
|
memorySetMemory(Memory, MemorySize);
|
|
|
|
|
|
MM::Init(Memory, MemorySize, kGlobalSize, Flags.stackSize);
|
|
OS::Init();
|
|
ToolBox::Init();
|
|
MPW::Init(argc, argv);
|
|
|
|
|
|
cpuStartup();
|
|
cpuSetModel(3,0);
|
|
|
|
CreateStack();
|
|
|
|
{
|
|
auto rv = Loader::Native::LoadFile(command);
|
|
if (!rv) {
|
|
fprintf(stderr, "### Unable to load %s\n", command.c_str());
|
|
fprintf(stderr, "# %s (OS error %d)\n", ErrorName(rv.error()), rv.error());
|
|
exit(EX_CONFIG);
|
|
}
|
|
}
|
|
|
|
GlobalInit();
|
|
|
|
|
|
cpuSetALineExceptionFunc(ToolBox::dispatch);
|
|
cpuSetFLineExceptionFunc(MPW::dispatch);
|
|
|
|
cpuSetMidInstructionExceptionFunc(MidInstructionExceptionFunc);
|
|
|
|
|
|
if (Flags.traceGlobals) //memorySetGlobalLog(kGlobalSize);
|
|
memorySetLoggingFunc(MemoryLogger);
|
|
|
|
MPW::Trace = Flags.traceMPW;
|
|
ToolBox::Trace = Flags.traceToolBox;
|
|
|
|
|
|
if (Flags.traceCPU || Flags.traceMacsbug)
|
|
{
|
|
#ifdef CPU_INSTRUCTION_LOGGING
|
|
cpuSetInstructionLoggingFunc(InstructionLogger);
|
|
#endif
|
|
// else do it manually below.
|
|
}
|
|
|
|
if (Flags.debugger) Debug::Shell();
|
|
else MainLoop();
|
|
|
|
|
|
|
|
if (Flags.memoryStats)
|
|
{
|
|
MM::Native::PrintMemoryStats();
|
|
}
|
|
|
|
uint32_t rv = MPW::ExitStatus();
|
|
if (rv > 0xff) rv = 0xff;
|
|
|
|
|
|
|
|
exit(rv);
|
|
}
|