Merge pull request #436 from DrChat/misc_changes

Misc changes
This commit is contained in:
Ben Vanik 2015-10-17 01:15:20 -07:00
commit 91dfb6a8a0
13 changed files with 132 additions and 26 deletions

View File

@ -294,6 +294,11 @@ void XmaContext::DecodePackets(XMA_CONTEXT_DATA* data) {
return; return;
} }
// No available data.
if (!data->input_buffer_0_valid && !data->input_buffer_1_valid) {
return;
}
assert_zero(data->unk_dword_9); assert_zero(data->unk_dword_9);
// XAudio Loops // XAudio Loops

View File

@ -189,6 +189,7 @@ Function* Processor::ResolveFunction(uint32_t address) {
// Grab symbol declaration. // Grab symbol declaration.
auto function = LookupFunction(address); auto function = LookupFunction(address);
if (!function) { if (!function) {
entry->status = Entry::STATUS_FAILED;
return nullptr; return nullptr;
} }
@ -317,10 +318,21 @@ uint64_t Processor::Execute(ThreadState* thread_state, uint32_t address,
SCOPE_profile_cpu_f("cpu"); SCOPE_profile_cpu_f("cpu");
PPCContext* context = thread_state->context(); PPCContext* context = thread_state->context();
assert_true(arg_count <= 5); for (size_t i = 0; i < std::min(arg_count, 8ull); ++i) {
for (size_t i = 0; i < arg_count; ++i) {
context->r[3 + i] = args[i]; context->r[3 + i] = args[i];
} }
if (arg_count > 7) {
// Rest of the arguments go on the stack.
// FIXME: This assumes arguments are 32 bits!
auto stack_arg_base =
memory()->TranslateVirtual((uint32_t)context->r[1] + 0x54 - (64 + 112));
for (size_t i = 0; i < arg_count - 8; i++) {
xe::store_and_swap<uint32_t>(stack_arg_base + (i * 8),
(uint32_t)args[i + 8]);
}
}
if (!Execute(thread_state, address)) { if (!Execute(thread_state, address)) {
return 0xDEADBABE; return 0xDEADBABE;
} }

View File

@ -55,6 +55,9 @@ bool RawModule::LoadFile(uint32_t base_address, const std::wstring& path) {
low_address_ = base_address; low_address_ = base_address;
high_address_ = base_address + file_length; high_address_ = base_address + file_length;
// Notify backend about executable code.
processor_->backend()->CommitExecutableRange(low_address_, high_address_);
return true; return true;
} }

View File

@ -24,6 +24,9 @@
#include "third_party/xxhash/xxhash.h" #include "third_party/xxhash/xxhash.h"
DEFINE_bool(draw_all_framebuffers, false,
"Copy all render targets to screen on swap");
namespace xe { namespace xe {
namespace gpu { namespace gpu {
namespace gl4 { namespace gl4 {
@ -641,6 +644,53 @@ void CommandProcessor::IssueSwap(uint32_t frontbuffer_ptr,
swap_state_.back_buffer_texture, dest_rect, swap_state_.back_buffer_texture, dest_rect,
GL_LINEAR); GL_LINEAR);
if (FLAGS_draw_all_framebuffers) {
int32_t offsetx = (1280 - (1280 / 5));
int32_t offsety = 0;
int32_t doffsetx = 0;
for (int i = 0; i < cached_framebuffers_.size(); i++) {
bool has_colortargets = false;
// Copy color targets to top right corner
for (int j = 0; j < 4; j++) {
GLuint tex = cached_framebuffers_[i].color_targets[j];
if (!tex) {
continue;
}
has_colortargets = true;
dest_rect = {offsetx, offsety, 1280 / 5, 720 / 5};
reinterpret_cast<ui::gl::GLContext*>(context_.get())
->blitter()
->CopyColorTexture2D(tex, src_rect, swap_state_.back_buffer_texture,
dest_rect, GL_LINEAR);
offsety += 720 / 5;
}
if (has_colortargets) {
offsetx -= 1280 / 5;
}
offsety = 0;
GLuint tex = cached_framebuffers_[i].depth_target;
if (!tex) {
continue;
}
// Copy depth targets to bottom left corner of screen
dest_rect = {doffsetx, (int32_t)swap_state_.height - (720 / 5), 1280 / 5,
720 / 5};
reinterpret_cast<ui::gl::GLContext*>(context_.get())
->blitter()
->CopyColorTexture2D(tex, src_rect, swap_state_.back_buffer_texture,
dest_rect, GL_LINEAR);
doffsetx += 1280 / 5;
}
}
// Need to finish to be sure the other context sees the right data. // Need to finish to be sure the other context sees the right data.
// TODO(benvanik): prevent this? fences? // TODO(benvanik): prevent this? fences?
glFinish(); glFinish();

View File

@ -142,12 +142,12 @@ X_STATUS UserModule::LoadFromMemory(const void* addr, const size_t length) {
} }
X_STATUS UserModule::Unload() { X_STATUS UserModule::Unload() {
if (!xex_module()->loaded()) { if (module_format_ == kModuleFormatXex && !xex_module()->loaded()) {
// Quick abort. // Quick abort.
return X_STATUS_SUCCESS; return X_STATUS_SUCCESS;
} }
if (xex_module()->Unload()) { if (module_format_ == kModuleFormatXex && xex_module()->Unload()) {
OnUnload(); OnUnload();
return X_STATUS_SUCCESS; return X_STATUS_SUCCESS;
} }
@ -505,7 +505,14 @@ void UserModule::Dump() {
sb.AppendFormat(" XEX_HEADER_GAME_RATINGS (TODO):\n"); sb.AppendFormat(" XEX_HEADER_GAME_RATINGS (TODO):\n");
} break; } break;
case XEX_HEADER_LAN_KEY: { case XEX_HEADER_LAN_KEY: {
sb.AppendFormat(" XEX_HEADER_LAN_KEY (TODO):\n"); sb.AppendFormat(" XEX_HEADER_LAN_KEY:");
auto opt_lan_key =
reinterpret_cast<const xex2_opt_lan_key*>(opt_header_ptr);
for (int l = 0; l < 16; l++) {
sb.AppendFormat(" %.2X", opt_lan_key->key[l]);
}
sb.Append("\n");
} break; } break;
case XEX_HEADER_XBOX360_LOGO: { case XEX_HEADER_XBOX360_LOGO: {
sb.AppendFormat(" XEX_HEADER_XBOX360_LOGO (TODO):\n"); sb.AppendFormat(" XEX_HEADER_XBOX360_LOGO (TODO):\n");

View File

@ -297,6 +297,7 @@ class Result {
} // namespace shim } // namespace shim
using int_t = const shim::ParamBase<int32_t>&; using int_t = const shim::ParamBase<int32_t>&;
using word_t = const shim::ParamBase<uint16_t>&;
using dword_t = const shim::ParamBase<uint32_t>&; using dword_t = const shim::ParamBase<uint32_t>&;
using qword_t = const shim::ParamBase<uint64_t>&; using qword_t = const shim::ParamBase<uint64_t>&;
using float_t = const shim::ParamBase<float>&; using float_t = const shim::ParamBase<float>&;
@ -322,6 +323,9 @@ namespace shim {
inline void AppendParam(StringBuffer* string_buffer, int_t param) { inline void AppendParam(StringBuffer* string_buffer, int_t param) {
string_buffer->AppendFormat("%d", int32_t(param)); string_buffer->AppendFormat("%d", int32_t(param));
} }
inline void AppendParam(StringBuffer* string_buffer, word_t param) {
string_buffer->AppendFormat("%.4X", uint16_t(param));
}
inline void AppendParam(StringBuffer* string_buffer, dword_t param) { inline void AppendParam(StringBuffer* string_buffer, dword_t param) {
string_buffer->AppendFormat("%.8X", uint32_t(param)); string_buffer->AppendFormat("%.8X", uint32_t(param));
} }

View File

@ -468,6 +468,10 @@ union xex2_version {
}; };
}; };
struct xex2_opt_lan_key {
uint8_t key[0x10];
};
struct xex2_opt_bound_path { struct xex2_opt_bound_path {
xe::be<uint32_t> size; xe::be<uint32_t> size;
char path[1]; char path[1];

View File

@ -333,6 +333,13 @@ SHIM_CALL XamContentCreateEx_shim(PPCContext* ppc_context,
cache_size, content_size, overlapped_ptr); cache_size, content_size, overlapped_ptr);
} }
dword_result_t XamContentOpenFile(dword_t r3, lpstring_t r4, lpstring_t r5,
dword_t r6, dword_t r7, dword_t r8,
dword_t r9) {
return X_ERROR_FILE_NOT_FOUND;
}
DECLARE_XAM_EXPORT(XamContentOpenFile, ExportTag::kStub);
SHIM_CALL XamContentFlush_shim(PPCContext* ppc_context, SHIM_CALL XamContentFlush_shim(PPCContext* ppc_context,
KernelState* kernel_state) { KernelState* kernel_state) {
uint32_t root_name_ptr = SHIM_GET_ARG_32(0); uint32_t root_name_ptr = SHIM_GET_ARG_32(0);

View File

@ -117,6 +117,13 @@ SHIM_CALL NtAllocateVirtualMemory_shim(PPCContext* ppc_context,
} }
uint32_t adjusted_size = xe::round_up(region_size_value, page_size); uint32_t adjusted_size = xe::round_up(region_size_value, page_size);
// Some games (BF1943) do this, but then if we return an error code it'll
// allocate with a smaller page size.
if (base_addr_value % page_size != 0) {
SHIM_SET_RETURN_32(X_STATUS_MAPPED_ALIGNMENT);
return;
}
// Allocate. // Allocate.
uint32_t allocation_type = 0; uint32_t allocation_type = 0;
if (alloc_type & X_MEM_RESERVE) { if (alloc_type & X_MEM_RESERVE) {
@ -480,6 +487,7 @@ SHIM_CALL MmGetPhysicalAddress_shim(PPCContext* ppc_context,
// _In_ PVOID BaseAddress // _In_ PVOID BaseAddress
// ); // );
// base_address = result of MmAllocatePhysicalMemory. // base_address = result of MmAllocatePhysicalMemory.
assert_true(base_address >= 0xA0000000);
uint32_t physical_address = base_address & 0x1FFFFFFF; uint32_t physical_address = base_address & 0x1FFFFFFF;
if (base_address >= 0xE0000000) { if (base_address >= 0xE0000000) {
@ -514,7 +522,7 @@ SHIM_CALL ExAllocatePoolTypeWithTag_shim(PPCContext* ppc_context,
uint32_t tag = SHIM_GET_ARG_32(1); uint32_t tag = SHIM_GET_ARG_32(1);
uint32_t zero = SHIM_GET_ARG_32(2); uint32_t zero = SHIM_GET_ARG_32(2);
XELOGD("ExAllocatePoolTypeWithTag(%d, %.8X, %d)", size, tag, zero); XELOGD("ExAllocatePoolTypeWithTag(%d, %.4s, %d)", size, &tag, zero);
uint32_t alignment = 8; uint32_t alignment = 8;
uint32_t adjusted_size = size; uint32_t adjusted_size = size;

View File

@ -494,33 +494,25 @@ dword_result_t KeResetEvent(pointer_t<X_KEVENT> event_ptr) {
DECLARE_XBOXKRNL_EXPORT(KeResetEvent, DECLARE_XBOXKRNL_EXPORT(KeResetEvent,
ExportTag::kImplemented | ExportTag::kThreading); ExportTag::kImplemented | ExportTag::kThreading);
SHIM_CALL NtCreateEvent_shim(PPCContext* ppc_context, dword_result_t NtCreateEvent(lpdword_t handle_ptr,
KernelState* kernel_state) { pointer_t<X_OBJECT_ATTRIBUTES> obj_attributes_ptr,
uint32_t handle_ptr = SHIM_GET_ARG_32(0); dword_t event_type, dword_t initial_state) {
uint32_t obj_attributes_ptr = SHIM_GET_ARG_32(1);
uint32_t event_type = SHIM_GET_ARG_32(2);
uint32_t initial_state = SHIM_GET_ARG_32(3);
XELOGD("NtCreateEvent(%.8X, %.8X, %d, %d)", handle_ptr, obj_attributes_ptr,
event_type, initial_state);
// Check for an existing timer with the same name. // Check for an existing timer with the same name.
auto existing_object = auto existing_object =
LookupNamedObject<XEvent>(kernel_state, obj_attributes_ptr); LookupNamedObject<XEvent>(kernel_state(), obj_attributes_ptr);
if (existing_object) { if (existing_object) {
if (existing_object->type() == XObject::kTypeEvent) { if (existing_object->type() == XObject::kTypeEvent) {
if (handle_ptr) { if (handle_ptr) {
existing_object->RetainHandle(); existing_object->RetainHandle();
SHIM_SET_MEM_32(handle_ptr, existing_object->handle()); *handle_ptr = existing_object->handle();
} }
SHIM_SET_RETURN_32(X_STATUS_SUCCESS); return X_STATUS_SUCCESS;
} else { } else {
SHIM_SET_RETURN_32(X_STATUS_INVALID_HANDLE); return X_STATUS_INVALID_HANDLE;
} }
return;
} }
XEvent* ev = new XEvent(kernel_state); XEvent* ev = new XEvent(kernel_state());
ev->Initialize(!event_type, !!initial_state); ev->Initialize(!event_type, !!initial_state);
// obj_attributes may have a name inside of it, if != NULL. // obj_attributes may have a name inside of it, if != NULL.
@ -529,10 +521,12 @@ SHIM_CALL NtCreateEvent_shim(PPCContext* ppc_context,
} }
if (handle_ptr) { if (handle_ptr) {
SHIM_SET_MEM_32(handle_ptr, ev->handle()); *handle_ptr = ev->handle();
} }
SHIM_SET_RETURN_32(X_STATUS_SUCCESS); return X_STATUS_SUCCESS;
} }
DECLARE_XBOXKRNL_EXPORT(NtCreateEvent,
ExportTag::kImplemented | ExportTag::kThreading);
dword_result_t NtSetEvent(dword_t handle, lpdword_t previous_state_ptr) { dword_result_t NtSetEvent(dword_t handle, lpdword_t previous_state_ptr) {
X_STATUS result = X_STATUS_SUCCESS; X_STATUS result = X_STATUS_SUCCESS;
@ -1421,8 +1415,6 @@ void RegisterThreadingExports(xe::cpu::ExportResolver* export_resolver,
SHIM_SET_MAPPING("xboxkrnl.exe", KeTlsGetValue, state); SHIM_SET_MAPPING("xboxkrnl.exe", KeTlsGetValue, state);
SHIM_SET_MAPPING("xboxkrnl.exe", KeTlsSetValue, state); SHIM_SET_MAPPING("xboxkrnl.exe", KeTlsSetValue, state);
SHIM_SET_MAPPING("xboxkrnl.exe", NtCreateEvent, state);
SHIM_SET_MAPPING("xboxkrnl.exe", NtCreateSemaphore, state); SHIM_SET_MAPPING("xboxkrnl.exe", NtCreateSemaphore, state);
SHIM_SET_MAPPING("xboxkrnl.exe", NtReleaseSemaphore, state); SHIM_SET_MAPPING("xboxkrnl.exe", NtReleaseSemaphore, state);

View File

@ -107,6 +107,16 @@ uint32_t XThread::GetCurrentThreadId() {
return thread->guest_object<X_KTHREAD>()->thread_id; return thread->guest_object<X_KTHREAD>()->thread_id;
} }
uint32_t XThread::GetLastError() {
XThread* thread = XThread::GetCurrentThread();
return thread->last_error();
}
void XThread::SetLastError(uint32_t error_code) {
XThread* thread = XThread::GetCurrentThread();
thread->set_last_error(error_code);
}
uint32_t XThread::last_error() { return guest_object<X_KTHREAD>()->last_error; } uint32_t XThread::last_error() { return guest_object<X_KTHREAD>()->last_error; }
void XThread::set_last_error(uint32_t error_code) { void XThread::set_last_error(uint32_t error_code) {

View File

@ -122,6 +122,9 @@ class XThread : public XObject {
static uint32_t GetCurrentThreadHandle(); static uint32_t GetCurrentThreadHandle();
static uint32_t GetCurrentThreadId(); static uint32_t GetCurrentThreadId();
static uint32_t GetLastError();
static void SetLastError(uint32_t error_code);
const CreationParams* creation_params() const { return &creation_params_; } const CreationParams* creation_params() const { return &creation_params_; }
uint32_t tls_ptr() const { return tls_address_; } uint32_t tls_ptr() const { return tls_address_; }
uint32_t pcr_ptr() const { return pcr_address_; } uint32_t pcr_ptr() const { return pcr_address_; }

View File

@ -66,6 +66,7 @@ typedef uint32_t X_STATUS;
#define X_STATUS_INVALID_PARAMETER_2 ((X_STATUS)0xC00000F0L) #define X_STATUS_INVALID_PARAMETER_2 ((X_STATUS)0xC00000F0L)
#define X_STATUS_INVALID_PARAMETER_3 ((X_STATUS)0xC00000F1L) #define X_STATUS_INVALID_PARAMETER_3 ((X_STATUS)0xC00000F1L)
#define X_STATUS_DLL_NOT_FOUND ((X_STATUS)0xC0000135L) #define X_STATUS_DLL_NOT_FOUND ((X_STATUS)0xC0000135L)
#define X_STATUS_MAPPED_ALIGNMENT ((X_STATUS)0xC0000220L)
#define X_STATUS_NOT_FOUND ((X_STATUS)0xC0000225L) #define X_STATUS_NOT_FOUND ((X_STATUS)0xC0000225L)
#define X_STATUS_DRIVER_ORDINAL_NOT_FOUND ((X_STATUS)0xC0000262L) #define X_STATUS_DRIVER_ORDINAL_NOT_FOUND ((X_STATUS)0xC0000262L)
#define X_STATUS_DRIVER_ENTRYPOINT_NOT_FOUND ((X_STATUS)0xC0000263L) #define X_STATUS_DRIVER_ENTRYPOINT_NOT_FOUND ((X_STATUS)0xC0000263L)