Starting revival of debugger system. Work on #41.
This commit is contained in:
parent
c17122e022
commit
d548e7f770
|
@ -16,3 +16,6 @@
|
||||||
[submodule "third_party/beaengine"]
|
[submodule "third_party/beaengine"]
|
||||||
path = third_party/beaengine
|
path = third_party/beaengine
|
||||||
url = https://github.com/benvanik/beaengine.git
|
url = https://github.com/benvanik/beaengine.git
|
||||||
|
[submodule "third_party/wslay"]
|
||||||
|
path = third_party/wslay
|
||||||
|
url = https://github.com/benvanik/wslay.git
|
||||||
|
|
|
@ -1,13 +0,0 @@
|
||||||
# Copyright 2013 Ben Vanik. All Rights Reserved.
|
|
||||||
{
|
|
||||||
'sources': [
|
|
||||||
'client.cc',
|
|
||||||
'client.h',
|
|
||||||
'content_source.cc',
|
|
||||||
'content_source.h',
|
|
||||||
'debugger.cc',
|
|
||||||
'debugger.h',
|
|
||||||
'listener.cc',
|
|
||||||
'listener.h',
|
|
||||||
],
|
|
||||||
}
|
|
|
@ -0,0 +1,26 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* 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. *
|
||||||
|
******************************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <xenia/debug/debug_client.h>
|
||||||
|
|
||||||
|
#include <xenia/debug/debug_server.h>
|
||||||
|
|
||||||
|
|
||||||
|
using namespace xe;
|
||||||
|
using namespace xe::debug;
|
||||||
|
|
||||||
|
|
||||||
|
DebugClient::DebugClient(DebugServer* debug_server) :
|
||||||
|
debug_server_(debug_server) {
|
||||||
|
debug_server_->AddClient(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
DebugClient::~DebugClient() {
|
||||||
|
debug_server_->RemoveClient(this);
|
||||||
|
}
|
|
@ -0,0 +1,40 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* 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_DEBUG_DEBUG_CLIENT_H_
|
||||||
|
#define XENIA_DEBUG_DEBUG_CLIENT_H_
|
||||||
|
|
||||||
|
#include <xenia/common.h>
|
||||||
|
#include <xenia/core.h>
|
||||||
|
|
||||||
|
|
||||||
|
XEDECLARECLASS2(xe, debug, DebugServer);
|
||||||
|
|
||||||
|
|
||||||
|
namespace xe {
|
||||||
|
namespace debug {
|
||||||
|
|
||||||
|
|
||||||
|
class DebugClient {
|
||||||
|
public:
|
||||||
|
DebugClient(DebugServer* debug_server);
|
||||||
|
virtual ~DebugClient();
|
||||||
|
|
||||||
|
virtual int Setup() = 0;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
DebugServer* debug_server_;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace debug
|
||||||
|
} // namespace xe
|
||||||
|
|
||||||
|
|
||||||
|
#endif // XENIA_DEBUG_DEBUG_CLIENT_H_
|
|
@ -0,0 +1,97 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* 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. *
|
||||||
|
******************************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <xenia/debug/debug_server.h>
|
||||||
|
|
||||||
|
#include <gflags/gflags.h>
|
||||||
|
|
||||||
|
#include <xenia/emulator.h>
|
||||||
|
#include <xenia/debug/debug_client.h>
|
||||||
|
#include <xenia/debug/protocol.h>
|
||||||
|
#include <xenia/debug/protocols/gdb/gdb_protocol.h>
|
||||||
|
#include <xenia/debug/protocols/ws/ws_protocol.h>
|
||||||
|
|
||||||
|
|
||||||
|
using namespace xe;
|
||||||
|
using namespace xe::debug;
|
||||||
|
|
||||||
|
|
||||||
|
DEFINE_bool(wait_for_debugger, false,
|
||||||
|
"Whether to wait for the debugger to attach before launching.");
|
||||||
|
|
||||||
|
|
||||||
|
DebugServer::DebugServer(Emulator* emulator) :
|
||||||
|
emulator_(emulator) {
|
||||||
|
//protocols_.push_back(
|
||||||
|
// new protocols::gdb::GDBProtocol(this, gdb_port));
|
||||||
|
protocols_.push_back(
|
||||||
|
new protocols::ws::WSProtocol(this));
|
||||||
|
}
|
||||||
|
|
||||||
|
DebugServer::~DebugServer() {
|
||||||
|
Shutdown();
|
||||||
|
}
|
||||||
|
|
||||||
|
int DebugServer::Startup() {
|
||||||
|
// HACK(benvanik): say we are ok even if we have no listener.
|
||||||
|
if (!protocols_.size()) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start listeners.
|
||||||
|
// This may launch threads and such.
|
||||||
|
for (std::vector<Protocol*>::iterator it = protocols_.begin();
|
||||||
|
it != protocols_.end(); ++it) {
|
||||||
|
Protocol* protocol = *it;
|
||||||
|
if (protocol->Setup()) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// If desired, wait until the first client connects.
|
||||||
|
if (FLAGS_wait_for_debugger) {
|
||||||
|
//XELOGI("Waiting for debugger on port %d...", FLAGS_remote_debug_port);
|
||||||
|
//if (listener_->WaitForClient()) {
|
||||||
|
// return 1;
|
||||||
|
//}
|
||||||
|
XELOGI("Debugger attached, continuing...");
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void DebugServer::Shutdown() {
|
||||||
|
std::vector<DebugClient*> clients(clients_.begin(), clients_.end());
|
||||||
|
clients_.clear();
|
||||||
|
for (std::vector<DebugClient*>::iterator it = clients.begin();
|
||||||
|
it != clients.end(); ++it) {
|
||||||
|
delete *it;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<Protocol*> protocols(protocols_.begin(), protocols_.end());
|
||||||
|
protocols_.clear();
|
||||||
|
for (std::vector<Protocol*>::iterator it = protocols.begin();
|
||||||
|
it != protocols.end(); ++it) {
|
||||||
|
delete *it;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void DebugServer::AddClient(DebugClient* debug_client) {
|
||||||
|
clients_.push_back(debug_client);
|
||||||
|
}
|
||||||
|
|
||||||
|
void DebugServer::RemoveClient(DebugClient* debug_client) {
|
||||||
|
for (std::vector<DebugClient*>::iterator it = clients_.begin();
|
||||||
|
it != clients_.end(); ++it) {
|
||||||
|
if (*it == debug_client) {
|
||||||
|
clients_.erase(it);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,53 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* 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_DEBUG_DEBUG_SERVER_H_
|
||||||
|
#define XENIA_DEBUG_DEBUG_SERVER_H_
|
||||||
|
|
||||||
|
#include <xenia/common.h>
|
||||||
|
#include <xenia/core.h>
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
|
||||||
|
XEDECLARECLASS1(xe, Emulator);
|
||||||
|
XEDECLARECLASS2(xe, debug, DebugClient);
|
||||||
|
XEDECLARECLASS2(xe, debug, Protocol);
|
||||||
|
|
||||||
|
|
||||||
|
namespace xe {
|
||||||
|
namespace debug {
|
||||||
|
|
||||||
|
|
||||||
|
class DebugServer {
|
||||||
|
public:
|
||||||
|
DebugServer(Emulator* emulator);
|
||||||
|
virtual ~DebugServer();
|
||||||
|
|
||||||
|
int Startup();
|
||||||
|
void Shutdown();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void AddClient(DebugClient* debug_client);
|
||||||
|
void RemoveClient(DebugClient* debug_client);
|
||||||
|
|
||||||
|
friend class DebugClient;
|
||||||
|
|
||||||
|
private:
|
||||||
|
Emulator* emulator_;
|
||||||
|
std::vector<Protocol*> protocols_;
|
||||||
|
std::vector<DebugClient*> clients_;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace debug
|
||||||
|
} // namespace xe
|
||||||
|
|
||||||
|
|
||||||
|
#endif // XENIA_DEBUG_DEBUG_SERVER_H_
|
|
@ -7,16 +7,16 @@
|
||||||
******************************************************************************
|
******************************************************************************
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <xenia/dbg/listener.h>
|
#include <xenia/debug/protocol.h>
|
||||||
|
|
||||||
|
|
||||||
using namespace xe;
|
using namespace xe;
|
||||||
using namespace xe::dbg;
|
using namespace xe::debug;
|
||||||
|
|
||||||
|
|
||||||
Listener::Listener(Debugger* debugger) :
|
Protocol::Protocol(DebugServer* debug_server) :
|
||||||
debugger_(debugger) {
|
debug_server_(debug_server) {
|
||||||
}
|
}
|
||||||
|
|
||||||
Listener::~Listener() {
|
Protocol::~Protocol() {
|
||||||
}
|
}
|
|
@ -7,35 +7,35 @@
|
||||||
******************************************************************************
|
******************************************************************************
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef XENIA_DBG_LISTENER_H_
|
#ifndef XENIA_DEBUG_PROTOCOL_H_
|
||||||
#define XENIA_DBG_LISTENER_H_
|
#define XENIA_DEBUG_PROTOCOL_H_
|
||||||
|
|
||||||
#include <xenia/common.h>
|
#include <xenia/common.h>
|
||||||
#include <xenia/core.h>
|
#include <xenia/core.h>
|
||||||
|
|
||||||
|
|
||||||
|
XEDECLARECLASS2(xe, debug, DebugServer);
|
||||||
|
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
namespace dbg {
|
namespace debug {
|
||||||
|
|
||||||
|
|
||||||
class Debugger;
|
class Protocol {
|
||||||
|
|
||||||
|
|
||||||
class Listener {
|
|
||||||
public:
|
public:
|
||||||
Listener(Debugger* debugger);
|
Protocol(DebugServer* debug_server);
|
||||||
virtual ~Listener();
|
virtual ~Protocol();
|
||||||
|
|
||||||
virtual int Setup() = 0;
|
virtual int Setup() = 0;
|
||||||
virtual int WaitForClient() = 0;
|
virtual int WaitForClient() = 0;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
Debugger* debugger_;
|
DebugServer* debug_server_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
} // namespace dbg
|
} // namespace debug
|
||||||
} // namespace xe
|
} // namespace xe
|
||||||
|
|
||||||
|
|
||||||
#endif // XENIA_DBG_LISTENER_H_
|
#endif // XENIA_DEBUG_PROTOCOL_H_
|
|
@ -0,0 +1,350 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* 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. *
|
||||||
|
******************************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <xenia/debug/protocols/gdb/gdb_client.h>
|
||||||
|
|
||||||
|
#include <xenia/debug/debug_server.h>
|
||||||
|
|
||||||
|
|
||||||
|
using namespace xe;
|
||||||
|
using namespace xe::debug;
|
||||||
|
using namespace xe::debug::protocols::gdb;
|
||||||
|
|
||||||
|
#if 0
|
||||||
|
GDBClient::GDBClient(DebugServer* debug_server, int socket_id) :
|
||||||
|
DebugClient(debug_server),
|
||||||
|
thread_(NULL),
|
||||||
|
socket_id_(socket_id) {
|
||||||
|
mutex_ = xe_mutex_alloc(1000);
|
||||||
|
|
||||||
|
loop_ = xe_socket_loop_create(socket_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
GDBClient::~GDBClient() {
|
||||||
|
xe_mutex_t* mutex = mutex_;
|
||||||
|
xe_mutex_lock(mutex);
|
||||||
|
|
||||||
|
mutex_ = NULL;
|
||||||
|
|
||||||
|
xe_socket_close(socket_id_);
|
||||||
|
socket_id_ = 0;
|
||||||
|
|
||||||
|
xe_socket_loop_destroy(loop_);
|
||||||
|
loop_ = NULL;
|
||||||
|
|
||||||
|
xe_mutex_unlock(mutex);
|
||||||
|
xe_mutex_free(mutex);
|
||||||
|
|
||||||
|
xe_thread_release(thread_);
|
||||||
|
}
|
||||||
|
|
||||||
|
int GDBClient::socket_id() {
|
||||||
|
return socket_id_;
|
||||||
|
}
|
||||||
|
|
||||||
|
int GDBClient::Setup() {
|
||||||
|
// Prep the socket.
|
||||||
|
xe_socket_set_keepalive(socket_id_, true);
|
||||||
|
xe_socket_set_nodelay(socket_id_, true);
|
||||||
|
|
||||||
|
thread_ = xe_thread_create("Debugger Client", StartCallback, this);
|
||||||
|
return xe_thread_start(thread_);
|
||||||
|
}
|
||||||
|
|
||||||
|
void GDBClient::StartCallback(void* param) {
|
||||||
|
GDBClient* client = reinterpret_cast<GDBClient*>(param);
|
||||||
|
client->EventThread();
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
int64_t WsClientSendCallback(wslay_event_context_ptr ctx,
|
||||||
|
const uint8_t* data, size_t len, int flags,
|
||||||
|
void* user_data) {
|
||||||
|
GDBClient* client = reinterpret_cast<GDBClient*>(user_data);
|
||||||
|
|
||||||
|
int error_code = 0;
|
||||||
|
int64_t r;
|
||||||
|
while ((r = xe_socket_send(client->socket_id(), data, len, 0,
|
||||||
|
&error_code)) == -1 && error_code == EINTR);
|
||||||
|
if (r == -1) {
|
||||||
|
if (error_code == EAGAIN || error_code == EWOULDBLOCK) {
|
||||||
|
wslay_event_set_error(ctx, GDBLAY_ERR_WOULDBLOCK);
|
||||||
|
} else {
|
||||||
|
wslay_event_set_error(ctx, GDBLAY_ERR_CALLBACK_FAILURE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
int64_t WsClientRecvCallback(wslay_event_context_ptr ctx,
|
||||||
|
uint8_t* data, size_t len, int flags,
|
||||||
|
void* user_data) {
|
||||||
|
GDBClient* client = reinterpret_cast<GDBClient*>(user_data);
|
||||||
|
|
||||||
|
int error_code = 0;
|
||||||
|
int64_t r;
|
||||||
|
while ((r = xe_socket_recv(client->socket_id(), data, len, 0,
|
||||||
|
&error_code)) == -1 && error_code == EINTR);
|
||||||
|
if (r == -1) {
|
||||||
|
if (error_code == EAGAIN || error_code == EWOULDBLOCK) {
|
||||||
|
wslay_event_set_error(ctx, GDBLAY_ERR_WOULDBLOCK);
|
||||||
|
} else {
|
||||||
|
wslay_event_set_error(ctx, GDBLAY_ERR_CALLBACK_FAILURE);
|
||||||
|
}
|
||||||
|
} else if (r == 0) {
|
||||||
|
wslay_event_set_error(ctx, GDBLAY_ERR_CALLBACK_FAILURE);
|
||||||
|
r = -1;
|
||||||
|
}
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
void GDBClientOnMsgCallback(wslay_event_context_ptr ctx,
|
||||||
|
const struct wslay_event_on_msg_recv_arg* arg,
|
||||||
|
void* user_data) {
|
||||||
|
if (wslay_is_ctrl_frame(arg->opcode)) {
|
||||||
|
// Ignore control frames.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
GDBClient* client = reinterpret_cast<GDBClient*>(user_data);
|
||||||
|
switch (arg->opcode) {
|
||||||
|
case GDBLAY_TEXT_FRAME:
|
||||||
|
XELOGW("Text frame ignored; use binary messages");
|
||||||
|
break;
|
||||||
|
case GDBLAY_BINARY_FRAME:
|
||||||
|
client->OnMessage(arg->msg, arg->msg_length);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
// Unknown opcode - some frame stuff?
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string EncodeBase64(const uint8_t* input, size_t length) {
|
||||||
|
static const char b64[] =
|
||||||
|
"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
|
||||||
|
std::string result;
|
||||||
|
size_t remaining = length;
|
||||||
|
size_t n = 0;
|
||||||
|
while (remaining) {
|
||||||
|
result.push_back(b64[input[n] >> 2]);
|
||||||
|
result.push_back(b64[((input[n] & 0x03) << 4) |
|
||||||
|
((input[n + 1] & 0xf0) >> 4)]);
|
||||||
|
remaining--;
|
||||||
|
if (remaining) {
|
||||||
|
result.push_back(b64[((input[n + 1] & 0x0f) << 2) |
|
||||||
|
((input[n + 2] & 0xc0) >> 6)]);
|
||||||
|
remaining--;
|
||||||
|
} else {
|
||||||
|
result.push_back('=');
|
||||||
|
}
|
||||||
|
if (remaining) {
|
||||||
|
result.push_back(b64[input[n + 2] & 0x3f]);
|
||||||
|
remaining--;
|
||||||
|
} else {
|
||||||
|
result.push_back('=');
|
||||||
|
}
|
||||||
|
n += 3;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
int GDBClient::PerformHandshake() {
|
||||||
|
std::string headers;
|
||||||
|
uint8_t buffer[4096];
|
||||||
|
int error_code = 0;
|
||||||
|
int64_t r;
|
||||||
|
while (true) {
|
||||||
|
while ((r = xe_socket_recv(socket_id_, buffer, sizeof(buffer), 0,
|
||||||
|
&error_code)) == -1 && error_code == EINTR);
|
||||||
|
if (r == -1) {
|
||||||
|
if (error_code == EWOULDBLOCK || error_code == EAGAIN) {
|
||||||
|
if (!headers.size()) {
|
||||||
|
// Nothing read yet - spin.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
XELOGE("HTTP header read failure");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
} else if (r == 0) {
|
||||||
|
// EOF.
|
||||||
|
XELOGE("HTTP header EOF");
|
||||||
|
return 2;
|
||||||
|
} else {
|
||||||
|
headers.append(buffer, buffer + r);
|
||||||
|
if (headers.size() > 8192) {
|
||||||
|
XELOGE("HTTP headers exceeded max buffer size");
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (headers.find("\r\n\r\n") == std::string::npos) {
|
||||||
|
XELOGE("Incomplete HTTP headers: %s", headers.c_str());
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Parse the headers to verify its a websocket request.
|
||||||
|
std::string::size_type keyhdstart;
|
||||||
|
if (headers.find("Upgrade: websocket\r\n") == std::string::npos ||
|
||||||
|
headers.find("Connection: Upgrade\r\n") == std::string::npos ||
|
||||||
|
(keyhdstart = headers.find("Sec-WebSocket-Key: ")) ==
|
||||||
|
std::string::npos) {
|
||||||
|
XELOGW("HTTP connection does not contain websocket headers");
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
keyhdstart += 19;
|
||||||
|
std::string::size_type keyhdend = headers.find("\r\n", keyhdstart);
|
||||||
|
std::string client_key = headers.substr(keyhdstart, keyhdend - keyhdstart);
|
||||||
|
std::string accept_key = client_key + "258EAFA5-E914-47DA-95CA-C5AB0DC85B11";
|
||||||
|
uint8_t accept_sha[20];
|
||||||
|
SHA1((uint8_t*)accept_key.c_str(), accept_key.size(), accept_sha);
|
||||||
|
accept_key = EncodeBase64(accept_sha, sizeof(accept_sha));
|
||||||
|
|
||||||
|
// Write the response to upgrade the connection.
|
||||||
|
std::string response =
|
||||||
|
"HTTP/1.1 101 Switching Protocols\r\n"
|
||||||
|
"Upgrade: websocket\r\n"
|
||||||
|
"Connection: Upgrade\r\n"
|
||||||
|
"Sec-WebSocket-Accept: " + accept_key + "\r\n"
|
||||||
|
"\r\n";
|
||||||
|
size_t write_offset = 0;
|
||||||
|
size_t write_length = response.size();
|
||||||
|
while (true) {
|
||||||
|
while ((r = xe_socket_send(socket_id_,
|
||||||
|
(uint8_t*)response.c_str() + write_offset,
|
||||||
|
write_length, 0, &error_code)) == -1 &&
|
||||||
|
error_code == EINTR);
|
||||||
|
if (r == -1) {
|
||||||
|
if (error_code == EAGAIN || error_code == EWOULDBLOCK) {
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
XELOGE("HTTP response write failure");
|
||||||
|
return 4;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
write_offset += r;
|
||||||
|
write_length -= r;
|
||||||
|
if (!write_length) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void GDBClient::EventThread() {
|
||||||
|
// Enable non-blocking IO on the socket.
|
||||||
|
xe_socket_set_nonblock(socket_id_, true);
|
||||||
|
|
||||||
|
// First run the HTTP handshake.
|
||||||
|
// This will fail if the connection is not for websockets.
|
||||||
|
if (PerformHandshake()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Prep callbacks.
|
||||||
|
struct wslay_event_callbacks callbacks = {
|
||||||
|
(wslay_event_recv_callback)WsClientRecvCallback,
|
||||||
|
(wslay_event_send_callback)WsClientSendCallback,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
GDBClientOnMsgCallback,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Prep the websocket server context.
|
||||||
|
wslay_event_context_ptr ctx;
|
||||||
|
wslay_event_context_server_init(&ctx, &callbacks, this);
|
||||||
|
|
||||||
|
// Loop forever.
|
||||||
|
while (wslay_event_want_read(ctx) || wslay_event_want_write(ctx)) {
|
||||||
|
// Wait on the event.
|
||||||
|
if (xe_socket_loop_poll(loop_,
|
||||||
|
!!wslay_event_want_read(ctx),
|
||||||
|
!!wslay_event_want_write(ctx))) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle any self-generated events to queue messages.
|
||||||
|
if (xe_socket_loop_check_queued_write(loop_)) {
|
||||||
|
xe_mutex_lock(mutex_);
|
||||||
|
for (std::vector<struct wslay_event_msg>::iterator it =
|
||||||
|
pending_messages_.begin(); it != pending_messages_.end(); it++) {
|
||||||
|
struct wslay_event_msg* msg = &*it;
|
||||||
|
wslay_event_queue_msg(ctx, msg);
|
||||||
|
}
|
||||||
|
pending_messages_.clear();
|
||||||
|
xe_mutex_unlock(mutex_);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle websocket messages.
|
||||||
|
if ((xe_socket_loop_check_socket_recv(loop_) && wslay_event_recv(ctx)) ||
|
||||||
|
(xe_socket_loop_check_socket_send(loop_) && wslay_event_send(ctx))) {
|
||||||
|
// Error handling the event.
|
||||||
|
XELOGE("Error handling WebSocket data");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wslay_event_context_free(ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
void GDBClient::Write(uint8_t** buffers, size_t* lengths, size_t count) {
|
||||||
|
if (!count) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
size_t combined_length;
|
||||||
|
uint8_t* combined_buffer;
|
||||||
|
if (count == 1) {
|
||||||
|
// Single buffer, just copy.
|
||||||
|
combined_length = lengths[0];
|
||||||
|
combined_buffer = (uint8_t*)xe_malloc(lengths[0]);
|
||||||
|
XEIGNORE(xe_copy_memory(combined_buffer, combined_length,
|
||||||
|
buffers[0], lengths[0]));
|
||||||
|
} else {
|
||||||
|
// Multiple buffers, merge.
|
||||||
|
combined_length = 0;
|
||||||
|
for (size_t n = 0; n < count; n++) {
|
||||||
|
combined_length += lengths[n];
|
||||||
|
}
|
||||||
|
combined_buffer = (uint8_t*)xe_malloc(combined_length);
|
||||||
|
for (size_t n = 0, offset = 0; n < count; n++) {
|
||||||
|
XEIGNORE(xe_copy_memory(
|
||||||
|
combined_buffer + offset, combined_length - offset,
|
||||||
|
buffers[n], lengths[n]));
|
||||||
|
offset += lengths[n];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct wslay_event_msg msg = {
|
||||||
|
GDBLAY_BINARY_FRAME,
|
||||||
|
combined_buffer,
|
||||||
|
combined_length,
|
||||||
|
};
|
||||||
|
|
||||||
|
xe_mutex_lock(mutex_);
|
||||||
|
pending_messages_.push_back(msg);
|
||||||
|
bool needs_signal = pending_messages_.size() == 1;
|
||||||
|
xe_mutex_unlock(mutex_);
|
||||||
|
|
||||||
|
if (needs_signal) {
|
||||||
|
// Notify the poll().
|
||||||
|
xe_socket_loop_set_queued_write(loop_);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
|
@ -0,0 +1,58 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* 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_DEBUG_PROTOCOLS_GDB_GDB_CLIENT_H_
|
||||||
|
#define XENIA_DEBUG_PROTOCOLS_GDB_GDB_CLIENT_H_
|
||||||
|
|
||||||
|
#include <xenia/common.h>
|
||||||
|
#include <xenia/core.h>
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include <xenia/debug/debug_client.h>
|
||||||
|
|
||||||
|
|
||||||
|
namespace xe {
|
||||||
|
namespace debug {
|
||||||
|
namespace protocols {
|
||||||
|
namespace gdb {
|
||||||
|
|
||||||
|
|
||||||
|
class GDBClient : public DebugClient {
|
||||||
|
public:
|
||||||
|
GDBClient(DebugServer* debug_server, int socket_id);
|
||||||
|
virtual ~GDBClient();
|
||||||
|
|
||||||
|
socket_t socket_id();
|
||||||
|
|
||||||
|
virtual int Setup();
|
||||||
|
|
||||||
|
void Write(uint8_t** buffers, size_t* lengths, size_t count);
|
||||||
|
|
||||||
|
private:
|
||||||
|
static void StartCallback(void* param);
|
||||||
|
|
||||||
|
int PerformHandshake();
|
||||||
|
void EventThread();
|
||||||
|
|
||||||
|
xe_thread_ref thread_;
|
||||||
|
|
||||||
|
socket_t socket_id_;
|
||||||
|
xe_socket_loop_t* loop_;
|
||||||
|
xe_mutex_t* mutex_;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace gdb
|
||||||
|
} // namespace protocols
|
||||||
|
} // namespace debug
|
||||||
|
} // namespace xe
|
||||||
|
|
||||||
|
|
||||||
|
#endif // XENIA_DEBUG_PROTOCOLS_GDB_GDB_CLIENT_H_
|
|
@ -0,0 +1,88 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* 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. *
|
||||||
|
******************************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <xenia/debug/protocols/gdb/gdb_protocol.h>
|
||||||
|
|
||||||
|
#include <xenia/debug/protocols/gdb/gdb_client.h>
|
||||||
|
|
||||||
|
#include <gflags/gflags.h>
|
||||||
|
|
||||||
|
|
||||||
|
DEFINE_int32(gdb_debug_port, 6201,
|
||||||
|
"Remote debugging port for GDB TCP connections.");
|
||||||
|
|
||||||
|
|
||||||
|
using namespace xe;
|
||||||
|
using namespace xe::debug;
|
||||||
|
using namespace xe::debug::protocols::gdb;
|
||||||
|
|
||||||
|
|
||||||
|
GDBProtocol::GDBProtocol(DebugServer* debug_server) :
|
||||||
|
Protocol(debug_server) {
|
||||||
|
port_ = FLAGS_gdb_debug_port;
|
||||||
|
}
|
||||||
|
|
||||||
|
GDBProtocol::~GDBProtocol() {
|
||||||
|
if (socket_id_) {
|
||||||
|
xe_socket_close(socket_id_);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
int GDBProtocol::Setup() {
|
||||||
|
if (port_ == 0 || port_ == -1) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
xe_socket_init();
|
||||||
|
|
||||||
|
socket_id_ = xe_socket_create_tcp();
|
||||||
|
if (socket_id_ == XE_INVALID_SOCKET) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
xe_socket_set_keepalive(socket_id_, true);
|
||||||
|
xe_socket_set_reuseaddr(socket_id_, true);
|
||||||
|
xe_socket_set_nodelay(socket_id_, true);
|
||||||
|
|
||||||
|
if (xe_socket_bind(socket_id_, port_)) {
|
||||||
|
XELOGE("Could not bind listen socket: %d", errno);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (xe_socket_listen(socket_id_)) {
|
||||||
|
xe_socket_close(socket_id_);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
int GDBProtocol::WaitForClient() {
|
||||||
|
if (!socket_id_) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Accept the first connection we get.
|
||||||
|
xe_socket_connection_t client_info;
|
||||||
|
if (xe_socket_accept(socket_id_, &client_info)) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
XELOGI("GDB debugger connected from %s", client_info.addr);
|
||||||
|
|
||||||
|
// Create the client object.
|
||||||
|
// Note that the client will delete itself when done.
|
||||||
|
GDBClient* client = new GDBClient(debug_server_, client_info.socket);
|
||||||
|
if (client->Setup()) {
|
||||||
|
// Client failed to setup - abort.
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
|
@ -0,0 +1,49 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* 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_DEBUG_PROTOCOLS_GDB_GDB_PROTOCOL_H_
|
||||||
|
#define XENIA_DEBUG_PROTOCOLS_GDB_GDB_PROTOCOL_H_
|
||||||
|
|
||||||
|
#include <xenia/common.h>
|
||||||
|
#include <xenia/core.h>
|
||||||
|
|
||||||
|
#include <xenia/debug/protocol.h>
|
||||||
|
|
||||||
|
|
||||||
|
XEDECLARECLASS2(xe, debug, DebugServer);
|
||||||
|
|
||||||
|
|
||||||
|
namespace xe {
|
||||||
|
namespace debug {
|
||||||
|
namespace protocols {
|
||||||
|
namespace gdb {
|
||||||
|
|
||||||
|
|
||||||
|
class GDBProtocol : public Protocol {
|
||||||
|
public:
|
||||||
|
GDBProtocol(DebugServer* debug_server);
|
||||||
|
virtual ~GDBProtocol();
|
||||||
|
|
||||||
|
virtual int Setup();
|
||||||
|
virtual int WaitForClient();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
uint32_t port_;
|
||||||
|
|
||||||
|
socket_t socket_id_;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace gdb
|
||||||
|
} // namespace protocols
|
||||||
|
} // namespace debug
|
||||||
|
} // namespace xe
|
||||||
|
|
||||||
|
|
||||||
|
#endif // XENIA_DEBUG_PROTOCOLS_GDB_GDB_PROTOCOL_H_
|
|
@ -0,0 +1,9 @@
|
||||||
|
# Copyright 2013 Ben Vanik. All Rights Reserved.
|
||||||
|
{
|
||||||
|
'sources': [
|
||||||
|
'gdb_client.cc',
|
||||||
|
'gdb_client.h',
|
||||||
|
'gdb_protocol.cc',
|
||||||
|
'gdb_protocol.h',
|
||||||
|
],
|
||||||
|
}
|
|
@ -0,0 +1,7 @@
|
||||||
|
# Copyright 2013 Ben Vanik. All Rights Reserved.
|
||||||
|
{
|
||||||
|
'includes': [
|
||||||
|
'gdb/sources.gypi',
|
||||||
|
'ws/sources.gypi',
|
||||||
|
],
|
||||||
|
}
|
|
@ -0,0 +1,235 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* 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. *
|
||||||
|
******************************************************************************
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <xenia/debug/protocols/ws/simple_sha1.h>
|
||||||
|
|
||||||
|
#if XE_PLATFORM(WIN32)
|
||||||
|
#include <winsock2.h>
|
||||||
|
#else
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
#endif // WIN32
|
||||||
|
|
||||||
|
|
||||||
|
using namespace xe;
|
||||||
|
using namespace xe::debug::protocols::ws;
|
||||||
|
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
|
||||||
|
// http://git.kernel.org/?p=git/git.git;a=blob;f=block-sha1/sha1.c
|
||||||
|
|
||||||
|
/*
|
||||||
|
* SHA1 routine optimized to do word accesses rather than byte accesses,
|
||||||
|
* and to avoid unnecessary copies into the context array.
|
||||||
|
*
|
||||||
|
* This was initially based on the Mozilla SHA1 implementation, although
|
||||||
|
* none of the original Mozilla code remains.
|
||||||
|
*/
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
size_t size;
|
||||||
|
uint32_t H[5];
|
||||||
|
uint32_t W[16];
|
||||||
|
} SHA_CTX;
|
||||||
|
|
||||||
|
#define SHA_ROT(X,l,r) (((X) << (l)) | ((X) >> (r)))
|
||||||
|
#define SHA_ROL(X,n) SHA_ROT(X,n,32-(n))
|
||||||
|
#define SHA_ROR(X,n) SHA_ROT(X,32-(n),n)
|
||||||
|
|
||||||
|
#define setW(x, val) (*(volatile unsigned int *)&W(x) = (val))
|
||||||
|
|
||||||
|
#define get_be32(p) ntohl(*(unsigned int *)(p))
|
||||||
|
#define put_be32(p, v) do { *(unsigned int *)(p) = htonl(v); } while (0)
|
||||||
|
|
||||||
|
#define W(x) (array[(x)&15])
|
||||||
|
|
||||||
|
#define SHA_SRC(t) get_be32((unsigned char *) block + (t)*4)
|
||||||
|
#define SHA_MIX(t) SHA_ROL(W((t)+13) ^ W((t)+8) ^ W((t)+2) ^ W(t), 1);
|
||||||
|
|
||||||
|
#define SHA_ROUND(t, input, fn, constant, A, B, C, D, E) do { \
|
||||||
|
unsigned int TEMP = input(t); setW(t, TEMP); \
|
||||||
|
E += TEMP + SHA_ROL(A,5) + (fn) + (constant); \
|
||||||
|
B = SHA_ROR(B, 2); } while (0)
|
||||||
|
|
||||||
|
#define T_0_15(t, A, B, C, D, E) SHA_ROUND(t, SHA_SRC, (((C^D)&B)^D) , 0x5a827999, A, B, C, D, E )
|
||||||
|
#define T_16_19(t, A, B, C, D, E) SHA_ROUND(t, SHA_MIX, (((C^D)&B)^D) , 0x5a827999, A, B, C, D, E )
|
||||||
|
#define T_20_39(t, A, B, C, D, E) SHA_ROUND(t, SHA_MIX, (B^C^D) , 0x6ed9eba1, A, B, C, D, E )
|
||||||
|
#define T_40_59(t, A, B, C, D, E) SHA_ROUND(t, SHA_MIX, ((B&C)+(D&(B^C))) , 0x8f1bbcdc, A, B, C, D, E )
|
||||||
|
#define T_60_79(t, A, B, C, D, E) SHA_ROUND(t, SHA_MIX, (B^C^D) , 0xca62c1d6, A, B, C, D, E )
|
||||||
|
|
||||||
|
static void SHA1_Block(SHA_CTX *ctx, const void *block) {
|
||||||
|
uint32_t A = ctx->H[0];
|
||||||
|
uint32_t B = ctx->H[1];
|
||||||
|
uint32_t C = ctx->H[2];
|
||||||
|
uint32_t D = ctx->H[3];
|
||||||
|
uint32_t E = ctx->H[4];
|
||||||
|
|
||||||
|
uint32_t array[16];
|
||||||
|
|
||||||
|
/* Round 1 - iterations 0-16 take their input from 'block' */
|
||||||
|
T_0_15( 0, A, B, C, D, E);
|
||||||
|
T_0_15( 1, E, A, B, C, D);
|
||||||
|
T_0_15( 2, D, E, A, B, C);
|
||||||
|
T_0_15( 3, C, D, E, A, B);
|
||||||
|
T_0_15( 4, B, C, D, E, A);
|
||||||
|
T_0_15( 5, A, B, C, D, E);
|
||||||
|
T_0_15( 6, E, A, B, C, D);
|
||||||
|
T_0_15( 7, D, E, A, B, C);
|
||||||
|
T_0_15( 8, C, D, E, A, B);
|
||||||
|
T_0_15( 9, B, C, D, E, A);
|
||||||
|
T_0_15(10, A, B, C, D, E);
|
||||||
|
T_0_15(11, E, A, B, C, D);
|
||||||
|
T_0_15(12, D, E, A, B, C);
|
||||||
|
T_0_15(13, C, D, E, A, B);
|
||||||
|
T_0_15(14, B, C, D, E, A);
|
||||||
|
T_0_15(15, A, B, C, D, E);
|
||||||
|
|
||||||
|
/* Round 1 - tail. Input from 512-bit mixing array */
|
||||||
|
T_16_19(16, E, A, B, C, D);
|
||||||
|
T_16_19(17, D, E, A, B, C);
|
||||||
|
T_16_19(18, C, D, E, A, B);
|
||||||
|
T_16_19(19, B, C, D, E, A);
|
||||||
|
|
||||||
|
/* Round 2 */
|
||||||
|
T_20_39(20, A, B, C, D, E);
|
||||||
|
T_20_39(21, E, A, B, C, D);
|
||||||
|
T_20_39(22, D, E, A, B, C);
|
||||||
|
T_20_39(23, C, D, E, A, B);
|
||||||
|
T_20_39(24, B, C, D, E, A);
|
||||||
|
T_20_39(25, A, B, C, D, E);
|
||||||
|
T_20_39(26, E, A, B, C, D);
|
||||||
|
T_20_39(27, D, E, A, B, C);
|
||||||
|
T_20_39(28, C, D, E, A, B);
|
||||||
|
T_20_39(29, B, C, D, E, A);
|
||||||
|
T_20_39(30, A, B, C, D, E);
|
||||||
|
T_20_39(31, E, A, B, C, D);
|
||||||
|
T_20_39(32, D, E, A, B, C);
|
||||||
|
T_20_39(33, C, D, E, A, B);
|
||||||
|
T_20_39(34, B, C, D, E, A);
|
||||||
|
T_20_39(35, A, B, C, D, E);
|
||||||
|
T_20_39(36, E, A, B, C, D);
|
||||||
|
T_20_39(37, D, E, A, B, C);
|
||||||
|
T_20_39(38, C, D, E, A, B);
|
||||||
|
T_20_39(39, B, C, D, E, A);
|
||||||
|
|
||||||
|
/* Round 3 */
|
||||||
|
T_40_59(40, A, B, C, D, E);
|
||||||
|
T_40_59(41, E, A, B, C, D);
|
||||||
|
T_40_59(42, D, E, A, B, C);
|
||||||
|
T_40_59(43, C, D, E, A, B);
|
||||||
|
T_40_59(44, B, C, D, E, A);
|
||||||
|
T_40_59(45, A, B, C, D, E);
|
||||||
|
T_40_59(46, E, A, B, C, D);
|
||||||
|
T_40_59(47, D, E, A, B, C);
|
||||||
|
T_40_59(48, C, D, E, A, B);
|
||||||
|
T_40_59(49, B, C, D, E, A);
|
||||||
|
T_40_59(50, A, B, C, D, E);
|
||||||
|
T_40_59(51, E, A, B, C, D);
|
||||||
|
T_40_59(52, D, E, A, B, C);
|
||||||
|
T_40_59(53, C, D, E, A, B);
|
||||||
|
T_40_59(54, B, C, D, E, A);
|
||||||
|
T_40_59(55, A, B, C, D, E);
|
||||||
|
T_40_59(56, E, A, B, C, D);
|
||||||
|
T_40_59(57, D, E, A, B, C);
|
||||||
|
T_40_59(58, C, D, E, A, B);
|
||||||
|
T_40_59(59, B, C, D, E, A);
|
||||||
|
|
||||||
|
/* Round 4 */
|
||||||
|
T_60_79(60, A, B, C, D, E);
|
||||||
|
T_60_79(61, E, A, B, C, D);
|
||||||
|
T_60_79(62, D, E, A, B, C);
|
||||||
|
T_60_79(63, C, D, E, A, B);
|
||||||
|
T_60_79(64, B, C, D, E, A);
|
||||||
|
T_60_79(65, A, B, C, D, E);
|
||||||
|
T_60_79(66, E, A, B, C, D);
|
||||||
|
T_60_79(67, D, E, A, B, C);
|
||||||
|
T_60_79(68, C, D, E, A, B);
|
||||||
|
T_60_79(69, B, C, D, E, A);
|
||||||
|
T_60_79(70, A, B, C, D, E);
|
||||||
|
T_60_79(71, E, A, B, C, D);
|
||||||
|
T_60_79(72, D, E, A, B, C);
|
||||||
|
T_60_79(73, C, D, E, A, B);
|
||||||
|
T_60_79(74, B, C, D, E, A);
|
||||||
|
T_60_79(75, A, B, C, D, E);
|
||||||
|
T_60_79(76, E, A, B, C, D);
|
||||||
|
T_60_79(77, D, E, A, B, C);
|
||||||
|
T_60_79(78, C, D, E, A, B);
|
||||||
|
T_60_79(79, B, C, D, E, A);
|
||||||
|
|
||||||
|
ctx->H[0] += A;
|
||||||
|
ctx->H[1] += B;
|
||||||
|
ctx->H[2] += C;
|
||||||
|
ctx->H[3] += D;
|
||||||
|
ctx->H[4] += E;
|
||||||
|
}
|
||||||
|
|
||||||
|
void SHA1_Update(SHA_CTX *ctx, const void *data, unsigned long len)
|
||||||
|
{
|
||||||
|
uint32_t lenW = ctx->size & 63;
|
||||||
|
ctx->size += len;
|
||||||
|
|
||||||
|
if (lenW) {
|
||||||
|
uint32_t left = 64 - lenW;
|
||||||
|
if (len < left) {
|
||||||
|
left = len;
|
||||||
|
}
|
||||||
|
memcpy(lenW + (char *)ctx->W, data, left);
|
||||||
|
lenW = (lenW + left) & 63;
|
||||||
|
len -= left;
|
||||||
|
data = ((const char *)data + left);
|
||||||
|
if (lenW) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
SHA1_Block(ctx, ctx->W);
|
||||||
|
}
|
||||||
|
while (len >= 64) {
|
||||||
|
SHA1_Block(ctx, data);
|
||||||
|
data = ((const char *)data + 64);
|
||||||
|
len -= 64;
|
||||||
|
}
|
||||||
|
if (len) {
|
||||||
|
memcpy(ctx->W, data, len);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void xe::debug::protocols::ws::SHA1(
|
||||||
|
const uint8_t* data, size_t length, uint8_t out_hash[20]) {
|
||||||
|
static const uint8_t pad[64] = { 0x80 };
|
||||||
|
|
||||||
|
SHA_CTX ctx = {
|
||||||
|
0,
|
||||||
|
{
|
||||||
|
0x67452301,
|
||||||
|
0xefcdab89,
|
||||||
|
0x98badcfe,
|
||||||
|
0x10325476,
|
||||||
|
0xc3d2e1f0,
|
||||||
|
},
|
||||||
|
{ 0 }
|
||||||
|
};
|
||||||
|
|
||||||
|
SHA1_Update(&ctx, data, (unsigned long)length);
|
||||||
|
|
||||||
|
uint32_t padlen[2] = {
|
||||||
|
htonl((uint32_t)(ctx.size >> 29)),
|
||||||
|
htonl((uint32_t)(ctx.size << 3)),
|
||||||
|
};
|
||||||
|
|
||||||
|
size_t i = ctx.size & 63;
|
||||||
|
SHA1_Update(&ctx, pad, 1 + (63 & (55 - i)));
|
||||||
|
SHA1_Update(&ctx, padlen, 8);
|
||||||
|
|
||||||
|
for (size_t n = 0; n < 5; n++) {
|
||||||
|
put_be32(out_hash + n * 4, ctx.H[n]);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,35 @@
|
||||||
|
/**
|
||||||
|
******************************************************************************
|
||||||
|
* 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_DEBUG_PROTOCOLS_WS_SIMPLE_SHA1_H_
|
||||||
|
#define XENIA_DEBUG_PROTOCOLS_WS_SIMPLE_SHA1_H_
|
||||||
|
|
||||||
|
#include <xenia/common.h>
|
||||||
|
#include <xenia/core.h>
|
||||||
|
|
||||||
|
|
||||||
|
namespace xe {
|
||||||
|
namespace debug {
|
||||||
|
namespace protocols {
|
||||||
|
namespace ws {
|
||||||
|
|
||||||
|
|
||||||
|
// This is a (likely) slow SHA1 designed for use on small values such as
|
||||||
|
// Websocket security keys. If we need something more complex it'd be best
|
||||||
|
// to use a real library.
|
||||||
|
void SHA1(const uint8_t* data, size_t length, uint8_t out_hash[20]);
|
||||||
|
|
||||||
|
|
||||||
|
} // namespace ws
|
||||||
|
} // namespace protocols
|
||||||
|
} // namespace debug
|
||||||
|
} // namespace xe
|
||||||
|
|
||||||
|
|
||||||
|
#endif // XENIA_DEBUG_PROTOCOLS_WS_SIMPLE_SHA1_H_
|
|
@ -0,0 +1,11 @@
|
||||||
|
# Copyright 2013 Ben Vanik. All Rights Reserved.
|
||||||
|
{
|
||||||
|
'sources': [
|
||||||
|
'simple_sha1.cc',
|
||||||
|
'simple_sha1.h',
|
||||||
|
'ws_client.cc',
|
||||||
|
'ws_client.h',
|
||||||
|
'ws_protocol.cc',
|
||||||
|
'ws_protocol.h',
|
||||||
|
],
|
||||||
|
}
|
|
@ -7,10 +7,10 @@
|
||||||
******************************************************************************
|
******************************************************************************
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <xenia/dbg/ws_client.h>
|
#include <xenia/debug/protocols/ws/ws_client.h>
|
||||||
|
|
||||||
#include <xenia/dbg/debugger.h>
|
#include <xenia/debug/debug_server.h>
|
||||||
#include <xenia/dbg/simple_sha1.h>
|
#include <xenia/debug/protocols/ws/simple_sha1.h>
|
||||||
|
|
||||||
#if XE_PLATFORM(WIN32)
|
#if XE_PLATFORM(WIN32)
|
||||||
// Required for wslay.
|
// Required for wslay.
|
||||||
|
@ -21,11 +21,12 @@ typedef SSIZE_T ssize_t;
|
||||||
|
|
||||||
|
|
||||||
using namespace xe;
|
using namespace xe;
|
||||||
using namespace xe::dbg;
|
using namespace xe::debug;
|
||||||
|
using namespace xe::debug::protocols::ws;
|
||||||
|
|
||||||
|
|
||||||
WsClient::WsClient(Debugger* debugger, int socket_id) :
|
WSClient::WSClient(DebugServer* debug_server, int socket_id) :
|
||||||
Client(debugger),
|
DebugClient(debug_server),
|
||||||
thread_(NULL),
|
thread_(NULL),
|
||||||
socket_id_(socket_id) {
|
socket_id_(socket_id) {
|
||||||
mutex_ = xe_mutex_alloc(1000);
|
mutex_ = xe_mutex_alloc(1000);
|
||||||
|
@ -33,7 +34,7 @@ WsClient::WsClient(Debugger* debugger, int socket_id) :
|
||||||
loop_ = xe_socket_loop_create(socket_id);
|
loop_ = xe_socket_loop_create(socket_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
WsClient::~WsClient() {
|
WSClient::~WSClient() {
|
||||||
xe_mutex_t* mutex = mutex_;
|
xe_mutex_t* mutex = mutex_;
|
||||||
xe_mutex_lock(mutex);
|
xe_mutex_lock(mutex);
|
||||||
|
|
||||||
|
@ -51,30 +52,30 @@ WsClient::~WsClient() {
|
||||||
xe_thread_release(thread_);
|
xe_thread_release(thread_);
|
||||||
}
|
}
|
||||||
|
|
||||||
int WsClient::socket_id() {
|
int WSClient::socket_id() {
|
||||||
return socket_id_;
|
return socket_id_;
|
||||||
}
|
}
|
||||||
|
|
||||||
int WsClient::Setup() {
|
int WSClient::Setup() {
|
||||||
// Prep the socket.
|
// Prep the socket.
|
||||||
xe_socket_set_keepalive(socket_id_, true);
|
xe_socket_set_keepalive(socket_id_, true);
|
||||||
xe_socket_set_nodelay(socket_id_, true);
|
xe_socket_set_nodelay(socket_id_, true);
|
||||||
|
|
||||||
thread_ = xe_thread_create("Debugger Client", StartCallback, this);
|
thread_ = xe_thread_create("WS Debugger Client", StartCallback, this);
|
||||||
return xe_thread_start(thread_);
|
return xe_thread_start(thread_);
|
||||||
}
|
}
|
||||||
|
|
||||||
void WsClient::StartCallback(void* param) {
|
void WSClient::StartCallback(void* param) {
|
||||||
WsClient* client = reinterpret_cast<WsClient*>(param);
|
WSClient* client = reinterpret_cast<WSClient*>(param);
|
||||||
client->EventThread();
|
client->EventThread();
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
int64_t WsClientSendCallback(wslay_event_context_ptr ctx,
|
int64_t WSClientSendCallback(wslay_event_context_ptr ctx,
|
||||||
const uint8_t* data, size_t len, int flags,
|
const uint8_t* data, size_t len, int flags,
|
||||||
void* user_data) {
|
void* user_data) {
|
||||||
WsClient* client = reinterpret_cast<WsClient*>(user_data);
|
WSClient* client = reinterpret_cast<WSClient*>(user_data);
|
||||||
|
|
||||||
int error_code = 0;
|
int error_code = 0;
|
||||||
int64_t r;
|
int64_t r;
|
||||||
|
@ -90,10 +91,10 @@ int64_t WsClientSendCallback(wslay_event_context_ptr ctx,
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
int64_t WsClientRecvCallback(wslay_event_context_ptr ctx,
|
int64_t WSClientRecvCallback(wslay_event_context_ptr ctx,
|
||||||
uint8_t* data, size_t len, int flags,
|
uint8_t* data, size_t len, int flags,
|
||||||
void* user_data) {
|
void* user_data) {
|
||||||
WsClient* client = reinterpret_cast<WsClient*>(user_data);
|
WSClient* client = reinterpret_cast<WSClient*>(user_data);
|
||||||
|
|
||||||
int error_code = 0;
|
int error_code = 0;
|
||||||
int64_t r;
|
int64_t r;
|
||||||
|
@ -112,7 +113,7 @@ int64_t WsClientRecvCallback(wslay_event_context_ptr ctx,
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
void WsClientOnMsgCallback(wslay_event_context_ptr ctx,
|
void WSClientOnMsgCallback(wslay_event_context_ptr ctx,
|
||||||
const struct wslay_event_on_msg_recv_arg* arg,
|
const struct wslay_event_on_msg_recv_arg* arg,
|
||||||
void* user_data) {
|
void* user_data) {
|
||||||
if (wslay_is_ctrl_frame(arg->opcode)) {
|
if (wslay_is_ctrl_frame(arg->opcode)) {
|
||||||
|
@ -120,13 +121,13 @@ void WsClientOnMsgCallback(wslay_event_context_ptr ctx,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
WsClient* client = reinterpret_cast<WsClient*>(user_data);
|
WSClient* client = reinterpret_cast<WSClient*>(user_data);
|
||||||
switch (arg->opcode) {
|
switch (arg->opcode) {
|
||||||
case WSLAY_TEXT_FRAME:
|
case WSLAY_TEXT_FRAME:
|
||||||
XELOGW("Text frame ignored; use binary messages");
|
XELOGW("Text frame ignored; use binary messages");
|
||||||
break;
|
break;
|
||||||
case WSLAY_BINARY_FRAME:
|
case WSLAY_BINARY_FRAME:
|
||||||
client->OnMessage(arg->msg, arg->msg_length);
|
//client->OnMessage(arg->msg, arg->msg_length);
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
// Unknown opcode - some frame stuff?
|
// Unknown opcode - some frame stuff?
|
||||||
|
@ -165,7 +166,7 @@ std::string EncodeBase64(const uint8_t* input, size_t length) {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int WsClient::PerformHandshake() {
|
int WSClient::PerformHandshake() {
|
||||||
std::string headers;
|
std::string headers;
|
||||||
uint8_t buffer[4096];
|
uint8_t buffer[4096];
|
||||||
int error_code = 0;
|
int error_code = 0;
|
||||||
|
@ -252,7 +253,7 @@ int WsClient::PerformHandshake() {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void WsClient::EventThread() {
|
void WSClient::EventThread() {
|
||||||
// Enable non-blocking IO on the socket.
|
// Enable non-blocking IO on the socket.
|
||||||
xe_socket_set_nonblock(socket_id_, true);
|
xe_socket_set_nonblock(socket_id_, true);
|
||||||
|
|
||||||
|
@ -264,13 +265,13 @@ void WsClient::EventThread() {
|
||||||
|
|
||||||
// Prep callbacks.
|
// Prep callbacks.
|
||||||
struct wslay_event_callbacks callbacks = {
|
struct wslay_event_callbacks callbacks = {
|
||||||
(wslay_event_recv_callback)WsClientRecvCallback,
|
(wslay_event_recv_callback)WSClientRecvCallback,
|
||||||
(wslay_event_send_callback)WsClientSendCallback,
|
(wslay_event_send_callback)WSClientSendCallback,
|
||||||
NULL,
|
NULL,
|
||||||
NULL,
|
NULL,
|
||||||
NULL,
|
NULL,
|
||||||
NULL,
|
NULL,
|
||||||
WsClientOnMsgCallback,
|
WSClientOnMsgCallback,
|
||||||
};
|
};
|
||||||
|
|
||||||
// Prep the websocket server context.
|
// Prep the websocket server context.
|
||||||
|
@ -310,7 +311,7 @@ void WsClient::EventThread() {
|
||||||
wslay_event_context_free(ctx);
|
wslay_event_context_free(ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
void WsClient::Write(uint8_t** buffers, size_t* lengths, size_t count) {
|
void WSClient::Write(uint8_t** buffers, size_t* lengths, size_t count) {
|
||||||
if (!count) {
|
if (!count) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
|
@ -7,34 +7,36 @@
|
||||||
******************************************************************************
|
******************************************************************************
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef XENIA_DBG_WS_CLIENT_H_
|
#ifndef XENIA_DEBUG_PROTOCOLS_WS_WS_CLIENT_H_
|
||||||
#define XENIA_DBG_WS_CLIENT_H_
|
#define XENIA_DEBUG_PROTOCOLS_WS_WS_CLIENT_H_
|
||||||
|
|
||||||
#include <xenia/common.h>
|
#include <xenia/common.h>
|
||||||
#include <xenia/core.h>
|
#include <xenia/core.h>
|
||||||
|
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
|
||||||
#include <xenia/dbg/client.h>
|
#include <xenia/debug/debug_client.h>
|
||||||
|
|
||||||
|
|
||||||
struct wslay_event_msg;
|
struct wslay_event_msg;
|
||||||
|
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
namespace dbg {
|
namespace debug {
|
||||||
|
namespace protocols {
|
||||||
|
namespace ws {
|
||||||
|
|
||||||
|
|
||||||
class WsClient : public Client {
|
class WSClient : public DebugClient {
|
||||||
public:
|
public:
|
||||||
WsClient(Debugger* debugger, int socket_id);
|
WSClient(DebugServer* debug_server, int socket_id);
|
||||||
virtual ~WsClient();
|
virtual ~WSClient();
|
||||||
|
|
||||||
socket_t socket_id();
|
socket_t socket_id();
|
||||||
|
|
||||||
virtual int Setup();
|
virtual int Setup();
|
||||||
|
|
||||||
virtual void Write(uint8_t** buffers, size_t* lengths, size_t count);
|
void Write(uint8_t** buffers, size_t* lengths, size_t count);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static void StartCallback(void* param);
|
static void StartCallback(void* param);
|
||||||
|
@ -42,7 +44,7 @@ private:
|
||||||
int PerformHandshake();
|
int PerformHandshake();
|
||||||
void EventThread();
|
void EventThread();
|
||||||
|
|
||||||
xe_thread_ref thread_;
|
xe_thread_ref thread_;
|
||||||
|
|
||||||
socket_t socket_id_;
|
socket_t socket_id_;
|
||||||
xe_socket_loop_t* loop_;
|
xe_socket_loop_t* loop_;
|
||||||
|
@ -51,8 +53,10 @@ private:
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
} // namespace dbg
|
} // namespace ws
|
||||||
|
} // namespace protocols
|
||||||
|
} // namespace debug
|
||||||
} // namespace xe
|
} // namespace xe
|
||||||
|
|
||||||
|
|
||||||
#endif // XENIA_DBG_WS_CLIENT_H_
|
#endif // XENIA_DEBUG_PROTOCOLS_WS_WS_CLIENT_H_
|
|
@ -7,28 +7,38 @@
|
||||||
******************************************************************************
|
******************************************************************************
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#include <xenia/dbg/ws_listener.h>
|
#include <xenia/debug/protocols/ws/ws_protocol.h>
|
||||||
|
|
||||||
#include <xenia/dbg/ws_client.h>
|
#include <xenia/debug/protocols/ws/ws_client.h>
|
||||||
|
|
||||||
|
#include <gflags/gflags.h>
|
||||||
|
|
||||||
|
|
||||||
using namespace xe;
|
using namespace xe;
|
||||||
using namespace xe::dbg;
|
using namespace xe::debug;
|
||||||
|
using namespace xe::debug::protocols::ws;
|
||||||
|
|
||||||
|
|
||||||
WsListener::WsListener(Debugger* debugger, uint32_t port) :
|
DEFINE_int32(ws_debug_port, 6200,
|
||||||
Listener(debugger),
|
"Remote debugging port for ws:// connections.");
|
||||||
port_(port) {
|
|
||||||
|
|
||||||
|
|
||||||
|
WSProtocol::WSProtocol(DebugServer* debug_server) :
|
||||||
|
Protocol(debug_server) {
|
||||||
|
port_ = FLAGS_ws_debug_port;
|
||||||
}
|
}
|
||||||
|
|
||||||
WsListener::~WsListener() {
|
WSProtocol::~WSProtocol() {
|
||||||
if (socket_id_) {
|
if (socket_id_) {
|
||||||
xe_socket_close(socket_id_);
|
xe_socket_close(socket_id_);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
int WsListener::Setup() {
|
int WSProtocol::Setup() {
|
||||||
|
if (port_ == 0 || port_ == -1) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
xe_socket_init();
|
xe_socket_init();
|
||||||
|
|
||||||
socket_id_ = xe_socket_create_tcp();
|
socket_id_ = xe_socket_create_tcp();
|
||||||
|
@ -53,18 +63,22 @@ int WsListener::Setup() {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int WsListener::WaitForClient() {
|
int WSProtocol::WaitForClient() {
|
||||||
|
if (!socket_id_) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
// Accept the first connection we get.
|
// Accept the first connection we get.
|
||||||
xe_socket_connection_t client_info;
|
xe_socket_connection_t client_info;
|
||||||
if (xe_socket_accept(socket_id_, &client_info)) {
|
if (xe_socket_accept(socket_id_, &client_info)) {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
XELOGI("Debugger connected from %s", client_info.addr);
|
XELOGI("WS debugger connected from %s", client_info.addr);
|
||||||
|
|
||||||
// Create the client object.
|
// Create the client object.
|
||||||
// Note that the client will delete itself when done.
|
// Note that the client will delete itself when done.
|
||||||
WsClient* client = new WsClient(debugger_, client_info.socket);
|
WSClient* client = new WSClient(debug_server_, client_info.socket);
|
||||||
if (client->Setup()) {
|
if (client->Setup()) {
|
||||||
// Client failed to setup - abort.
|
// Client failed to setup - abort.
|
||||||
return 1;
|
return 1;
|
|
@ -7,26 +7,28 @@
|
||||||
******************************************************************************
|
******************************************************************************
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef XENIA_DBG_WS_LISTENER_H_
|
#ifndef XENIA_DEBUG_PROTOCOLS_WS_WS_PROTOCOL_H_
|
||||||
#define XENIA_DBG_WS_LISTENER_H_
|
#define XENIA_DEBUG_PROTOCOLS_WS_WS_PROTOCOL_H_
|
||||||
|
|
||||||
#include <xenia/common.h>
|
#include <xenia/common.h>
|
||||||
#include <xenia/core.h>
|
#include <xenia/core.h>
|
||||||
|
|
||||||
#include <xenia/dbg/listener.h>
|
#include <xenia/debug/protocol.h>
|
||||||
|
|
||||||
|
|
||||||
|
XEDECLARECLASS2(xe, debug, DebugServer);
|
||||||
|
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
namespace dbg {
|
namespace debug {
|
||||||
|
namespace protocols {
|
||||||
|
namespace ws {
|
||||||
|
|
||||||
|
|
||||||
class Debugger;
|
class WSProtocol : public Protocol {
|
||||||
|
|
||||||
|
|
||||||
class WsListener : public Listener {
|
|
||||||
public:
|
public:
|
||||||
WsListener(Debugger* debugger, uint32_t port);
|
WSProtocol(DebugServer* debug_server);
|
||||||
virtual ~WsListener();
|
virtual ~WSProtocol();
|
||||||
|
|
||||||
virtual int Setup();
|
virtual int Setup();
|
||||||
virtual int WaitForClient();
|
virtual int WaitForClient();
|
||||||
|
@ -38,8 +40,10 @@ protected:
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
} // namespace dbg
|
} // namespace ws
|
||||||
|
} // namespace protocols
|
||||||
|
} // namespace debug
|
||||||
} // namespace xe
|
} // namespace xe
|
||||||
|
|
||||||
|
|
||||||
#endif // XENIA_DBG_WS_LISTENER_H_
|
#endif // XENIA_DEBUG_PROTOCOLS_WS_WS_PROTOCOL_H_
|
|
@ -0,0 +1,15 @@
|
||||||
|
# Copyright 2013 Ben Vanik. All Rights Reserved.
|
||||||
|
{
|
||||||
|
'sources': [
|
||||||
|
'debug_client.cc',
|
||||||
|
'debug_client.h',
|
||||||
|
'debug_server.cc',
|
||||||
|
'debug_server.h',
|
||||||
|
'protocol.cc',
|
||||||
|
'protocol.h',
|
||||||
|
],
|
||||||
|
|
||||||
|
'includes': [
|
||||||
|
'protocols/sources.gypi',
|
||||||
|
],
|
||||||
|
}
|
|
@ -12,7 +12,7 @@
|
||||||
#include <xenia/apu/apu.h>
|
#include <xenia/apu/apu.h>
|
||||||
#include <xenia/cpu/cpu.h>
|
#include <xenia/cpu/cpu.h>
|
||||||
#include <xenia/cpu/xenon_memory.h>
|
#include <xenia/cpu/xenon_memory.h>
|
||||||
#include <xenia/dbg/debugger.h>
|
#include <xenia/debug/debug_server.h>
|
||||||
#include <xenia/gpu/gpu.h>
|
#include <xenia/gpu/gpu.h>
|
||||||
#include <xenia/hid/hid.h>
|
#include <xenia/hid/hid.h>
|
||||||
#include <xenia/kernel/kernel.h>
|
#include <xenia/kernel/kernel.h>
|
||||||
|
@ -23,7 +23,7 @@
|
||||||
using namespace xe;
|
using namespace xe;
|
||||||
using namespace xe::apu;
|
using namespace xe::apu;
|
||||||
using namespace xe::cpu;
|
using namespace xe::cpu;
|
||||||
using namespace xe::dbg;
|
using namespace xe::debug;
|
||||||
using namespace xe::gpu;
|
using namespace xe::gpu;
|
||||||
using namespace xe::hid;
|
using namespace xe::hid;
|
||||||
using namespace xe::kernel;
|
using namespace xe::kernel;
|
||||||
|
@ -34,7 +34,7 @@ using namespace xe::kernel::xboxkrnl::fs;
|
||||||
|
|
||||||
Emulator::Emulator(const xechar_t* command_line) :
|
Emulator::Emulator(const xechar_t* command_line) :
|
||||||
memory_(0),
|
memory_(0),
|
||||||
debugger_(0),
|
debug_server_(0),
|
||||||
processor_(0),
|
processor_(0),
|
||||||
audio_system_(0), graphics_system_(0), input_system_(0),
|
audio_system_(0), graphics_system_(0), input_system_(0),
|
||||||
export_resolver_(0), file_system_(0),
|
export_resolver_(0), file_system_(0),
|
||||||
|
@ -45,6 +45,11 @@ Emulator::Emulator(const xechar_t* command_line) :
|
||||||
Emulator::~Emulator() {
|
Emulator::~Emulator() {
|
||||||
// Note that we delete things in the reverse order they were initialized.
|
// Note that we delete things in the reverse order they were initialized.
|
||||||
|
|
||||||
|
// Disconnect all debug clients first.
|
||||||
|
if (debug_server_) {
|
||||||
|
debug_server_->Shutdown();
|
||||||
|
}
|
||||||
|
|
||||||
delete xam_;
|
delete xam_;
|
||||||
delete xboxkrnl_;
|
delete xboxkrnl_;
|
||||||
|
|
||||||
|
@ -55,7 +60,7 @@ Emulator::~Emulator() {
|
||||||
delete audio_system_;
|
delete audio_system_;
|
||||||
delete processor_;
|
delete processor_;
|
||||||
|
|
||||||
delete debugger_;
|
delete debug_server_;
|
||||||
|
|
||||||
delete export_resolver_;
|
delete export_resolver_;
|
||||||
}
|
}
|
||||||
|
@ -74,8 +79,8 @@ X_STATUS Emulator::Setup() {
|
||||||
XEEXPECTNOTNULL(export_resolver_);
|
XEEXPECTNOTNULL(export_resolver_);
|
||||||
|
|
||||||
// Create the debugger.
|
// Create the debugger.
|
||||||
debugger_ = new Debugger(this);
|
debug_server_ = new DebugServer(this);
|
||||||
XEEXPECTNOTNULL(debugger_);
|
XEEXPECTNOTNULL(debug_server_);
|
||||||
|
|
||||||
// Initialize the CPU.
|
// Initialize the CPU.
|
||||||
processor_ = new Processor(this);
|
processor_ = new Processor(this);
|
||||||
|
@ -115,7 +120,7 @@ X_STATUS Emulator::Setup() {
|
||||||
|
|
||||||
// Prepare the debugger.
|
// Prepare the debugger.
|
||||||
// This may pause waiting for connections.
|
// This may pause waiting for connections.
|
||||||
result = debugger_->Startup();
|
result = debug_server_->Startup();
|
||||||
XEEXPECTZERO(result);
|
XEEXPECTZERO(result);
|
||||||
|
|
||||||
return result;
|
return result;
|
||||||
|
|
|
@ -18,7 +18,7 @@
|
||||||
XEDECLARECLASS1(xe, ExportResolver);
|
XEDECLARECLASS1(xe, ExportResolver);
|
||||||
XEDECLARECLASS2(xe, apu, AudioSystem);
|
XEDECLARECLASS2(xe, apu, AudioSystem);
|
||||||
XEDECLARECLASS2(xe, cpu, Processor);
|
XEDECLARECLASS2(xe, cpu, Processor);
|
||||||
XEDECLARECLASS2(xe, dbg, Debugger);
|
XEDECLARECLASS2(xe, debug, DebugServer);
|
||||||
XEDECLARECLASS2(xe, gpu, GraphicsSystem);
|
XEDECLARECLASS2(xe, gpu, GraphicsSystem);
|
||||||
XEDECLARECLASS2(xe, hid, InputSystem);
|
XEDECLARECLASS2(xe, hid, InputSystem);
|
||||||
XEDECLARECLASS3(xe, kernel, xam, XamModule);
|
XEDECLARECLASS3(xe, kernel, xam, XamModule);
|
||||||
|
@ -37,7 +37,7 @@ public:
|
||||||
const xechar_t* command_line() const { return command_line_; }
|
const xechar_t* command_line() const { return command_line_; }
|
||||||
Memory* memory() const { return memory_; }
|
Memory* memory() const { return memory_; }
|
||||||
|
|
||||||
dbg::Debugger* debugger() const { return debugger_; }
|
debug::DebugServer* debug_server() const { return debug_server_; }
|
||||||
|
|
||||||
cpu::Processor* processor() const { return processor_; }
|
cpu::Processor* processor() const { return processor_; }
|
||||||
apu::AudioSystem* audio_system() const { return audio_system_; }
|
apu::AudioSystem* audio_system() const { return audio_system_; }
|
||||||
|
@ -57,7 +57,7 @@ private:
|
||||||
xechar_t command_line_[XE_MAX_PATH];
|
xechar_t command_line_[XE_MAX_PATH];
|
||||||
Memory* memory_;
|
Memory* memory_;
|
||||||
|
|
||||||
dbg::Debugger* debugger_;
|
debug::DebugServer* debug_server_;
|
||||||
|
|
||||||
cpu::Processor* processor_;
|
cpu::Processor* processor_;
|
||||||
apu::AudioSystem* audio_system_;
|
apu::AudioSystem* audio_system_;
|
||||||
|
|
|
@ -29,7 +29,7 @@
|
||||||
'apu/sources.gypi',
|
'apu/sources.gypi',
|
||||||
'core/sources.gypi',
|
'core/sources.gypi',
|
||||||
'cpu/sources.gypi',
|
'cpu/sources.gypi',
|
||||||
'dbg/sources.gypi',
|
'debug/sources.gypi',
|
||||||
'gpu/sources.gypi',
|
'gpu/sources.gypi',
|
||||||
'hid/sources.gypi',
|
'hid/sources.gypi',
|
||||||
'kernel/sources.gypi',
|
'kernel/sources.gypi',
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit b3571f1ed27093e910ab8675d0c8333e5a2818ea
|
|
@ -0,0 +1,59 @@
|
||||||
|
# Copyright 2013 Ben Vanik. All Rights Reserved.
|
||||||
|
{
|
||||||
|
'targets': [
|
||||||
|
{
|
||||||
|
'target_name': 'wslay',
|
||||||
|
'type': '<(library)',
|
||||||
|
|
||||||
|
'direct_dependent_settings': {
|
||||||
|
'include_dirs': [
|
||||||
|
'wslay/lib/includes/',
|
||||||
|
],
|
||||||
|
|
||||||
|
'defines': [
|
||||||
|
'WSLAY_VERSION=1',
|
||||||
|
],
|
||||||
|
|
||||||
|
# libraries: ws2_32 on windows
|
||||||
|
},
|
||||||
|
|
||||||
|
'defines': [
|
||||||
|
'WSLAY_VERSION="1"',
|
||||||
|
],
|
||||||
|
|
||||||
|
'conditions': [
|
||||||
|
['OS != "win"', {
|
||||||
|
'defines': [
|
||||||
|
'HAVE_ARPA_INET_H=1',
|
||||||
|
'HAVE_NETINET_IN_H=1',
|
||||||
|
],
|
||||||
|
}],
|
||||||
|
['OS == "win"', {
|
||||||
|
'defines': [
|
||||||
|
'HAVE_WINSOCK2_H=1',
|
||||||
|
'ssize_t=long long',
|
||||||
|
],
|
||||||
|
}],
|
||||||
|
],
|
||||||
|
|
||||||
|
'include_dirs': [
|
||||||
|
'wslay/lib/',
|
||||||
|
'wslay/lib/includes/',
|
||||||
|
],
|
||||||
|
|
||||||
|
'sources': [
|
||||||
|
'wslay/lib/includes/wslay/wslay.h',
|
||||||
|
'wslay/lib/wslay_event.c',
|
||||||
|
'wslay/lib/wslay_event.h',
|
||||||
|
'wslay/lib/wslay_frame.c',
|
||||||
|
'wslay/lib/wslay_frame.h',
|
||||||
|
'wslay/lib/wslay_net.c',
|
||||||
|
'wslay/lib/wslay_net.h',
|
||||||
|
'wslay/lib/wslay_queue.c',
|
||||||
|
'wslay/lib/wslay_queue.h',
|
||||||
|
'wslay/lib/wslay_stack.c',
|
||||||
|
'wslay/lib/wslay_stack.h',
|
||||||
|
],
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
|
@ -6,6 +6,7 @@
|
||||||
'third_party/beaengine.gypi',
|
'third_party/beaengine.gypi',
|
||||||
'third_party/gflags.gypi',
|
'third_party/gflags.gypi',
|
||||||
'third_party/sparsehash.gypi',
|
'third_party/sparsehash.gypi',
|
||||||
|
'third_party/wslay.gypi',
|
||||||
],
|
],
|
||||||
|
|
||||||
'default_configuration': 'release',
|
'default_configuration': 'release',
|
||||||
|
@ -248,12 +249,14 @@
|
||||||
'asmjit',
|
'asmjit',
|
||||||
'beaengine',
|
'beaengine',
|
||||||
'gflags',
|
'gflags',
|
||||||
|
'wslay',
|
||||||
'alloy',
|
'alloy',
|
||||||
],
|
],
|
||||||
'export_dependent_settings': [
|
'export_dependent_settings': [
|
||||||
'asmjit',
|
'asmjit',
|
||||||
'beaengine',
|
'beaengine',
|
||||||
'gflags',
|
'gflags',
|
||||||
|
'wslay',
|
||||||
'alloy',
|
'alloy',
|
||||||
],
|
],
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue