2013-01-20 09:13:59 +00:00
|
|
|
/**
|
|
|
|
******************************************************************************
|
|
|
|
* Xenia : Xbox 360 Emulator Research Project *
|
|
|
|
******************************************************************************
|
|
|
|
* Copyright 2013 Ben Vanik. All rights reserved. *
|
|
|
|
* Released under the BSD license - see LICENSE in the root for more details. *
|
|
|
|
******************************************************************************
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef XENIA_CPU_PROCESSOR_H_
|
|
|
|
#define XENIA_CPU_PROCESSOR_H_
|
|
|
|
|
2016-01-18 19:48:21 +00:00
|
|
|
#include <gflags/gflags.h>
|
|
|
|
|
|
|
|
#include <map>
|
2015-07-29 07:15:52 +00:00
|
|
|
#include <memory>
|
2015-08-07 03:17:01 +00:00
|
|
|
#include <string>
|
2014-08-16 07:56:50 +00:00
|
|
|
#include <vector>
|
|
|
|
|
2016-01-18 19:48:21 +00:00
|
|
|
#include "xenia/base/mapped_memory.h"
|
2015-05-25 06:16:43 +00:00
|
|
|
#include "xenia/base/mutex.h"
|
2015-05-04 05:28:25 +00:00
|
|
|
#include "xenia/cpu/backend/backend.h"
|
2016-01-18 19:48:21 +00:00
|
|
|
#include "xenia/cpu/debug_listener.h"
|
2015-05-04 05:28:25 +00:00
|
|
|
#include "xenia/cpu/entry_table.h"
|
2015-05-02 09:11:11 +00:00
|
|
|
#include "xenia/cpu/export_resolver.h"
|
2015-05-04 05:28:25 +00:00
|
|
|
#include "xenia/cpu/function.h"
|
|
|
|
#include "xenia/cpu/module.h"
|
2015-12-15 05:17:55 +00:00
|
|
|
#include "xenia/cpu/ppc/ppc_frontend.h"
|
2016-01-18 19:48:21 +00:00
|
|
|
#include "xenia/cpu/thread_debug_info.h"
|
2015-05-04 05:28:25 +00:00
|
|
|
#include "xenia/cpu/thread_state.h"
|
2015-02-01 06:49:47 +00:00
|
|
|
#include "xenia/memory.h"
|
2013-06-01 04:22:00 +00:00
|
|
|
|
2016-01-18 19:48:21 +00:00
|
|
|
DECLARE_bool(debug);
|
2015-07-22 07:01:36 +00:00
|
|
|
|
2013-01-20 09:13:59 +00:00
|
|
|
namespace xe {
|
|
|
|
namespace cpu {
|
|
|
|
|
2015-11-26 03:35:05 +00:00
|
|
|
class Breakpoint;
|
2015-07-29 07:15:52 +00:00
|
|
|
class StackWalker;
|
2014-08-21 05:50:10 +00:00
|
|
|
class XexModule;
|
2013-01-20 09:13:59 +00:00
|
|
|
|
2014-08-04 22:39:42 +00:00
|
|
|
enum class Irql : uint32_t {
|
|
|
|
PASSIVE = 0,
|
|
|
|
APC = 1,
|
|
|
|
DISPATCH = 2,
|
|
|
|
DPC = 3,
|
|
|
|
};
|
|
|
|
|
2016-01-18 19:48:21 +00:00
|
|
|
// Describes the current state of the emulator as known to the debugger.
|
|
|
|
// This determines which state the debugger is in and what operations are
|
|
|
|
// allowed.
|
|
|
|
enum class ExecutionState {
|
|
|
|
// Target is running; the debugger is not waiting for any events.
|
|
|
|
kRunning,
|
|
|
|
// Target is running in stepping mode with the debugger waiting for feedback.
|
|
|
|
kStepping,
|
|
|
|
// Target is paused for debugging.
|
|
|
|
kPaused,
|
|
|
|
// Target has been stopped and cannot be restarted (crash, etc).
|
|
|
|
kEnded,
|
|
|
|
};
|
|
|
|
|
2014-07-10 04:21:40 +00:00
|
|
|
class Processor {
|
2014-08-21 05:50:10 +00:00
|
|
|
public:
|
2016-01-18 19:48:21 +00:00
|
|
|
Processor(Memory* memory, ExportResolver* export_resolver);
|
2013-01-20 09:13:59 +00:00
|
|
|
~Processor();
|
|
|
|
|
2013-12-07 06:57:16 +00:00
|
|
|
Memory* memory() const { return memory_; }
|
2015-07-29 07:15:52 +00:00
|
|
|
StackWalker* stack_walker() const { return stack_walker_.get(); }
|
2015-12-15 05:17:55 +00:00
|
|
|
ppc::PPCFrontend* frontend() const { return frontend_.get(); }
|
2015-05-04 05:28:25 +00:00
|
|
|
backend::Backend* backend() const { return backend_.get(); }
|
|
|
|
ExportResolver* export_resolver() const { return export_resolver_; }
|
2013-01-20 09:13:59 +00:00
|
|
|
|
2017-02-11 05:54:10 +00:00
|
|
|
bool Setup(std::unique_ptr<backend::Backend> backend);
|
2013-01-20 09:13:59 +00:00
|
|
|
|
2016-01-18 19:48:21 +00:00
|
|
|
// Runs any pre-launch logic once the module and thread have been setup.
|
|
|
|
void PreLaunch();
|
|
|
|
|
|
|
|
// The current execution state of the emulator.
|
|
|
|
ExecutionState execution_state() const { return execution_state_; }
|
|
|
|
|
|
|
|
// True if a debug listener is attached and the debugger is active.
|
|
|
|
bool is_debugger_attached() const { return !!debug_listener_; }
|
|
|
|
// Gets the active debug listener, if any.
|
|
|
|
DebugListener* debug_listener() const { return debug_listener_; }
|
|
|
|
// Sets the active debug listener, if any.
|
|
|
|
// This can be used to detach the listener.
|
|
|
|
void set_debug_listener(DebugListener* debug_listener);
|
|
|
|
// Sets a handler that will be called from a random thread when a debugger
|
|
|
|
// listener is required (such as on a breakpoint hit/etc).
|
|
|
|
// Will only be called if the debug listener has not already been specified
|
|
|
|
// with set_debug_listener.
|
|
|
|
void set_debug_listener_request_handler(
|
|
|
|
std::function<DebugListener*(Processor*)> handler) {
|
|
|
|
debug_listener_handler_ = std::move(handler);
|
|
|
|
}
|
|
|
|
|
2015-06-06 17:59:22 +00:00
|
|
|
void set_debug_info_flags(uint32_t debug_info_flags) {
|
|
|
|
debug_info_flags_ = debug_info_flags;
|
|
|
|
}
|
|
|
|
|
2015-05-06 00:21:08 +00:00
|
|
|
bool AddModule(std::unique_ptr<Module> module);
|
2015-05-04 05:28:25 +00:00
|
|
|
Module* GetModule(const char* name);
|
|
|
|
Module* GetModule(const std::string& name) { return GetModule(name.c_str()); }
|
|
|
|
std::vector<Module*> GetModules();
|
|
|
|
|
|
|
|
Module* builtin_module() const { return builtin_module_; }
|
2015-08-06 04:50:02 +00:00
|
|
|
Function* DefineBuiltin(const std::string& name,
|
|
|
|
BuiltinFunction::Handler handler, void* arg0,
|
|
|
|
void* arg1);
|
2015-05-04 05:28:25 +00:00
|
|
|
|
2015-06-09 06:44:38 +00:00
|
|
|
Function* QueryFunction(uint32_t address);
|
2015-05-04 05:28:25 +00:00
|
|
|
std::vector<Function*> FindFunctionsWithAddress(uint32_t address);
|
|
|
|
|
2015-08-06 04:50:02 +00:00
|
|
|
Function* LookupFunction(uint32_t address);
|
|
|
|
Function* LookupFunction(Module* module, uint32_t address);
|
|
|
|
Function* ResolveFunction(uint32_t address);
|
2015-05-04 05:28:25 +00:00
|
|
|
|
2015-05-06 00:21:08 +00:00
|
|
|
bool Execute(ThreadState* thread_state, uint32_t address);
|
2015-11-28 04:28:21 +00:00
|
|
|
bool ExecuteRaw(ThreadState* thread_state, uint32_t address);
|
2015-03-25 02:41:29 +00:00
|
|
|
uint64_t Execute(ThreadState* thread_state, uint32_t address, uint64_t args[],
|
2015-03-24 15:25:58 +00:00
|
|
|
size_t arg_count);
|
2015-09-01 16:45:14 +00:00
|
|
|
uint64_t ExecuteInterrupt(ThreadState* thread_state, uint32_t address,
|
|
|
|
uint64_t args[], size_t arg_count);
|
2013-09-25 07:46:09 +00:00
|
|
|
|
2014-08-04 22:39:42 +00:00
|
|
|
Irql RaiseIrql(Irql new_value);
|
|
|
|
void LowerIrql(Irql old_value);
|
|
|
|
|
2016-01-18 19:48:21 +00:00
|
|
|
bool Save(ByteStream* stream);
|
|
|
|
bool Restore(ByteStream* stream);
|
|
|
|
|
|
|
|
// Returns a list of debugger info for all threads that have ever existed.
|
|
|
|
// This is the preferred way to sample thread state vs. attempting to ask
|
|
|
|
// the kernel.
|
|
|
|
std::vector<ThreadDebugInfo*> QueryThreadDebugInfos();
|
|
|
|
|
|
|
|
// Returns the debugger info for the given thread.
|
|
|
|
ThreadDebugInfo* QueryThreadDebugInfo(uint32_t thread_id);
|
|
|
|
|
|
|
|
// Adds a breakpoint to the debugger and activates it (if enabled).
|
|
|
|
// The given breakpoint will not be owned by the debugger and must remain
|
|
|
|
// allocated so long as it is added.
|
|
|
|
void AddBreakpoint(Breakpoint* breakpoint);
|
|
|
|
|
|
|
|
// Removes a breakpoint from the debugger and deactivates it.
|
|
|
|
void RemoveBreakpoint(Breakpoint* breakpoint);
|
|
|
|
|
|
|
|
// Finds a breakpoint that may be registered at the given address.
|
2015-11-26 03:35:05 +00:00
|
|
|
Breakpoint* FindBreakpoint(uint32_t address);
|
2016-01-18 19:48:21 +00:00
|
|
|
|
|
|
|
// Returns all currently registered breakpoints.
|
|
|
|
std::vector<Breakpoint*> breakpoints() const;
|
|
|
|
|
|
|
|
// Shows the debug listener, focusing it if it already exists.
|
|
|
|
void ShowDebugger();
|
|
|
|
|
|
|
|
// Pauses target execution by suspending all threads.
|
|
|
|
// The debug listener will be requested if it has not been attached.
|
|
|
|
void Pause();
|
|
|
|
|
|
|
|
// Continues target execution from wherever it is.
|
|
|
|
// This will cancel any active step operations and resume all threads.
|
|
|
|
void Continue();
|
|
|
|
|
|
|
|
// Steps the given thread a single x64 host instruction.
|
|
|
|
// If the step is over a branch the branch will be followed.
|
|
|
|
void StepHostInstruction(uint32_t thread_id);
|
|
|
|
|
|
|
|
// Steps the given thread a single PPC guest instruction.
|
|
|
|
// If the step is over a branch the branch will be followed.
|
|
|
|
void StepGuestInstruction(uint32_t thread_id);
|
|
|
|
|
|
|
|
// Steps the given thread until the guest address is hit.
|
|
|
|
// Returns false if the step could not be completed (invalid target address).
|
|
|
|
bool StepToGuestAddress(uint32_t thread_id, uint32_t pc);
|
|
|
|
|
|
|
|
// Steps the given thread to the target of the branch at the specified guest
|
|
|
|
// address. The address must specify a branch instruction.
|
|
|
|
// Returns the new PC guest address.
|
|
|
|
uint32_t StepIntoGuestBranchTarget(uint32_t thread_id, uint32_t pc);
|
|
|
|
|
|
|
|
// Steps the thread to a point where it's safe to terminate or read its
|
|
|
|
// context. Returns the PC after we've finished stepping.
|
|
|
|
// Pass true for ignore_host if you've stopped the thread yourself
|
|
|
|
// in host code you want to ignore.
|
|
|
|
// Returns the new PC guest address.
|
|
|
|
uint32_t StepToGuestSafePoint(uint32_t thread_id, bool ignore_host = false);
|
|
|
|
|
|
|
|
public:
|
|
|
|
// TODO(benvanik): hide.
|
|
|
|
void OnThreadCreated(uint32_t handle, ThreadState* thread_state,
|
2017-02-07 03:04:00 +00:00
|
|
|
Thread* thread);
|
2016-01-18 19:48:21 +00:00
|
|
|
void OnThreadExit(uint32_t thread_id);
|
|
|
|
void OnThreadDestroyed(uint32_t thread_id);
|
|
|
|
void OnThreadEnteringWait(uint32_t thread_id);
|
|
|
|
void OnThreadLeavingWait(uint32_t thread_id);
|
|
|
|
|
|
|
|
bool OnUnhandledException(Exception* ex);
|
|
|
|
bool OnThreadBreakpointHit(Exception* ex);
|
|
|
|
|
|
|
|
uint8_t* AllocateFunctionTraceData(size_t size);
|
2015-11-26 03:35:05 +00:00
|
|
|
|
2014-08-21 05:50:10 +00:00
|
|
|
private:
|
2016-01-18 19:48:21 +00:00
|
|
|
// Synchronously demands a debug listener.
|
|
|
|
void DemandDebugListener();
|
|
|
|
|
|
|
|
// Suspends all known threads (except the caller).
|
|
|
|
bool SuspendAllThreads();
|
|
|
|
// Resumes the given thread.
|
|
|
|
bool ResumeThread(uint32_t thread_id);
|
|
|
|
// Resumes all known threads (except the caller).
|
|
|
|
bool ResumeAllThreads();
|
|
|
|
// Updates all cached thread execution info (state, call stacks, etc).
|
|
|
|
// The given override thread handle and context will be used in place of
|
|
|
|
// sampled values for that thread.
|
|
|
|
void UpdateThreadExecutionStates(uint32_t override_handle = 0,
|
|
|
|
X64Context* override_context = nullptr);
|
|
|
|
|
|
|
|
// Suspends all breakpoints, uninstalling them as required.
|
|
|
|
// No breakpoints will be triggered until they are resumed.
|
|
|
|
void SuspendAllBreakpoints();
|
|
|
|
// Resumes all breakpoints, re-installing them if required.
|
|
|
|
void ResumeAllBreakpoints();
|
|
|
|
|
|
|
|
void OnFunctionDefined(Function* function);
|
|
|
|
|
|
|
|
static bool ExceptionCallbackThunk(Exception* ex, void* data);
|
|
|
|
bool ExceptionCallback(Exception* ex);
|
|
|
|
void OnStepCompleted(ThreadDebugInfo* thread_info);
|
|
|
|
void OnBreakpointHit(ThreadDebugInfo* thread_info, Breakpoint* breakpoint);
|
|
|
|
|
|
|
|
// Calculates the next guest instruction based on the current thread state and
|
|
|
|
// current PC. This will look for branches and other control flow
|
|
|
|
// instructions.
|
|
|
|
uint32_t CalculateNextGuestInstruction(ThreadDebugInfo* thread_info,
|
|
|
|
uint32_t current_pc);
|
2015-11-26 03:35:05 +00:00
|
|
|
|
2015-08-06 04:50:02 +00:00
|
|
|
bool DemandFunction(Function* function);
|
2013-12-07 06:57:16 +00:00
|
|
|
|
2015-07-16 06:26:58 +00:00
|
|
|
Memory* memory_ = nullptr;
|
2015-07-29 07:15:52 +00:00
|
|
|
std::unique_ptr<StackWalker> stack_walker_;
|
2013-12-07 06:57:16 +00:00
|
|
|
|
2016-01-18 19:48:21 +00:00
|
|
|
std::function<DebugListener*(Processor*)> debug_listener_handler_;
|
|
|
|
DebugListener* debug_listener_ = nullptr;
|
|
|
|
|
|
|
|
// Which debug features are enabled in generated code.
|
2015-07-16 06:26:58 +00:00
|
|
|
uint32_t debug_info_flags_ = 0;
|
2016-01-18 19:48:21 +00:00
|
|
|
// If specified, the file trace data gets written to when running.
|
|
|
|
std::wstring functions_trace_path_;
|
|
|
|
std::unique_ptr<ChunkedMappedMemoryWriter> functions_trace_file_;
|
2015-05-04 05:28:25 +00:00
|
|
|
|
2015-12-15 05:17:55 +00:00
|
|
|
std::unique_ptr<ppc::PPCFrontend> frontend_;
|
2015-05-04 05:28:25 +00:00
|
|
|
std::unique_ptr<backend::Backend> backend_;
|
2015-07-16 06:26:58 +00:00
|
|
|
ExportResolver* export_resolver_ = nullptr;
|
2015-05-04 05:28:25 +00:00
|
|
|
|
|
|
|
EntryTable entry_table_;
|
2015-09-06 16:30:54 +00:00
|
|
|
xe::global_critical_region global_critical_region_;
|
2016-01-18 19:48:21 +00:00
|
|
|
ExecutionState execution_state_ = ExecutionState::kPaused;
|
2015-05-04 05:28:25 +00:00
|
|
|
std::vector<std::unique_ptr<Module>> modules_;
|
2015-07-16 06:26:58 +00:00
|
|
|
Module* builtin_module_ = nullptr;
|
|
|
|
uint32_t next_builtin_address_ = 0xFFFF0000u;
|
2015-05-04 05:28:25 +00:00
|
|
|
|
2016-01-18 19:48:21 +00:00
|
|
|
// Maps thread ID to state. Updated on thread create, and threads are never
|
|
|
|
// removed. Must be guarded with the global lock.
|
|
|
|
std::map<uint32_t, std::unique_ptr<ThreadDebugInfo>> thread_debug_infos_;
|
|
|
|
|
|
|
|
// TODO(benvanik): cleanup/change structures.
|
2015-11-26 03:35:05 +00:00
|
|
|
std::vector<Breakpoint*> breakpoints_;
|
|
|
|
|
2014-08-21 05:50:10 +00:00
|
|
|
Irql irql_;
|
2013-01-20 09:13:59 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
} // namespace cpu
|
|
|
|
} // namespace xe
|
|
|
|
|
|
|
|
#endif // XENIA_CPU_PROCESSOR_H_
|