Initial the websocket implementation.
Not yet running on other threads, but can read/write to a websocket from a browser.
This commit is contained in:
parent
c2fbafdc28
commit
5650cf92ab
|
@ -16,3 +16,9 @@
|
||||||
[submodule "third_party/sparsehash"]
|
[submodule "third_party/sparsehash"]
|
||||||
path = third_party/sparsehash
|
path = third_party/sparsehash
|
||||||
url = https://github.com/benvanik/sparsehash.git
|
url = https://github.com/benvanik/sparsehash.git
|
||||||
|
[submodule "third_party/wslay"]
|
||||||
|
path = third_party/wslay
|
||||||
|
url = https://github.com/benvanik/wslay.git
|
||||||
|
[submodule "third_party/openssl"]
|
||||||
|
path = third_party/openssl
|
||||||
|
url = https://github.com/benvanik/openssl.git
|
||||||
|
|
19
common.gypi
19
common.gypi
|
@ -2,6 +2,14 @@
|
||||||
{
|
{
|
||||||
'default_configuration': 'release',
|
'default_configuration': 'release',
|
||||||
|
|
||||||
|
'conditions': [
|
||||||
|
['OS == "mac"', {
|
||||||
|
'variables': {
|
||||||
|
'is_clang': 1
|
||||||
|
}
|
||||||
|
}],
|
||||||
|
],
|
||||||
|
|
||||||
'variables': {
|
'variables': {
|
||||||
'configurations': {
|
'configurations': {
|
||||||
'debug': {
|
'debug': {
|
||||||
|
@ -10,6 +18,9 @@
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
||||||
|
'library%': 'static_library',
|
||||||
|
'target_arch%': 'x64',
|
||||||
|
|
||||||
# LLVM paths.
|
# LLVM paths.
|
||||||
# TODO(benvanik): switch based on configuration.
|
# TODO(benvanik): switch based on configuration.
|
||||||
'llvm_path': 'build/llvm/release/',
|
'llvm_path': 'build/llvm/release/',
|
||||||
|
@ -64,7 +75,6 @@
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
|
||||||
|
|
||||||
'target_defaults': {
|
'target_defaults': {
|
||||||
'include_dirs': [
|
'include_dirs': [
|
||||||
'include/',
|
'include/',
|
||||||
|
@ -74,6 +84,9 @@
|
||||||
'__STDC_LIMIT_MACROS=1',
|
'__STDC_LIMIT_MACROS=1',
|
||||||
'__STDC_CONSTANT_MACROS=1',
|
'__STDC_CONSTANT_MACROS=1',
|
||||||
'_ISOC99_SOURCE=1',
|
'_ISOC99_SOURCE=1',
|
||||||
|
|
||||||
|
'OPENSSL_NO_INLINE_ASM',
|
||||||
|
'OPENSSL_NO_NEXTPROTONEG',
|
||||||
],
|
],
|
||||||
'cflags': [
|
'cflags': [
|
||||||
'-std=c99',
|
'-std=c99',
|
||||||
|
@ -134,9 +147,9 @@
|
||||||
'ARCHS': ['x86_64'],
|
'ARCHS': ['x86_64'],
|
||||||
#'CLANG_CXX_LANGUAGE_STANDARD': 'c++0x',
|
#'CLANG_CXX_LANGUAGE_STANDARD': 'c++0x',
|
||||||
'COMBINE_HIDPI_IMAGES': 'YES',
|
'COMBINE_HIDPI_IMAGES': 'YES',
|
||||||
'GCC_C_LANGUAGE_STANDARD': 'c99',
|
'GCC_C_LANGUAGE_STANDARD': 'gnu99',
|
||||||
'GCC_SYMBOLS_PRIVATE_EXTERN': 'YES',
|
'GCC_SYMBOLS_PRIVATE_EXTERN': 'YES',
|
||||||
'GCC_TREAT_WARNINGS_AS_ERRORS': 'YES',
|
#'GCC_TREAT_WARNINGS_AS_ERRORS': 'YES',
|
||||||
'GCC_WARN_ABOUT_MISSING_NEWLINE': 'YES',
|
'GCC_WARN_ABOUT_MISSING_NEWLINE': 'YES',
|
||||||
'GCC_VERSION': 'com.apple.compilers.llvm.clang.1_0',
|
'GCC_VERSION': 'com.apple.compilers.llvm.clang.1_0',
|
||||||
'WARNING_CFLAGS': ['-Wall', '-Wendif-labels'],
|
'WARNING_CFLAGS': ['-Wall', '-Wendif-labels'],
|
||||||
|
|
|
@ -23,9 +23,10 @@ public:
|
||||||
Client();
|
Client();
|
||||||
virtual ~Client();
|
virtual ~Client();
|
||||||
|
|
||||||
void Write(const uint8_t* buffer, const size_t length);
|
virtual int Setup() = 0;
|
||||||
virtual void Write(const uint8_t** buffers, const size_t* lengths,
|
|
||||||
size_t count) = 0;
|
void Write(uint8_t* buffer, size_t length);
|
||||||
|
virtual void Write(uint8_t** buffers, size_t* lengths, size_t count) = 0;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
};
|
};
|
||||||
|
|
|
@ -20,8 +20,8 @@ Client::Client() {
|
||||||
Client::~Client() {
|
Client::~Client() {
|
||||||
}
|
}
|
||||||
|
|
||||||
void Client::Write(const uint8_t* buffer, const size_t length) {
|
void Client::Write(uint8_t* buffer, size_t length) {
|
||||||
const uint8_t* buffers[] = {buffer};
|
uint8_t* buffers[] = {buffer};
|
||||||
const size_t lengths[] = {length};
|
size_t lengths[] = {length};
|
||||||
Write(buffers, lengths, 1);
|
Write(buffers, lengths, 1);
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,6 +9,18 @@
|
||||||
|
|
||||||
#include "dbg/ws_client.h"
|
#include "dbg/ws_client.h"
|
||||||
|
|
||||||
|
#include <fcntl.h>
|
||||||
|
#include <poll.h>
|
||||||
|
#include <arpa/inet.h>
|
||||||
|
#include <netinet/tcp.h>
|
||||||
|
#include <sys/socket.h>
|
||||||
|
#include <wslay/wslay.h>
|
||||||
|
|
||||||
|
#include <openssl/bio.h>
|
||||||
|
#include <openssl/buffer.h>
|
||||||
|
#include <openssl/hmac.h>
|
||||||
|
#include <openssl/sha.h>
|
||||||
|
|
||||||
|
|
||||||
using namespace xe;
|
using namespace xe;
|
||||||
using namespace xe::dbg;
|
using namespace xe::dbg;
|
||||||
|
@ -17,13 +29,361 @@ using namespace xe::dbg;
|
||||||
WsClient::WsClient(int socket_id) :
|
WsClient::WsClient(int socket_id) :
|
||||||
Client(),
|
Client(),
|
||||||
socket_id_(socket_id) {
|
socket_id_(socket_id) {
|
||||||
|
mutex_ = xe_mutex_alloc(1000);
|
||||||
|
|
||||||
|
int notify_ids[2];
|
||||||
|
socketpair(PF_LOCAL, SOCK_STREAM, 0, notify_ids);
|
||||||
|
notify_rd_id_ = notify_ids[0];
|
||||||
|
notify_wr_id_ = notify_ids[1];
|
||||||
}
|
}
|
||||||
|
|
||||||
WsClient::~WsClient() {
|
WsClient::~WsClient() {
|
||||||
|
xe_mutex_t* mutex = mutex_;
|
||||||
|
xe_mutex_lock(mutex);
|
||||||
|
mutex_ = NULL;
|
||||||
|
shutdown(socket_id_, SHUT_WR);
|
||||||
close(socket_id_);
|
close(socket_id_);
|
||||||
|
socket_id_ = 0;
|
||||||
|
xe_mutex_unlock(mutex);
|
||||||
|
xe_mutex_free(mutex);
|
||||||
|
|
||||||
|
close(notify_rd_id_);
|
||||||
|
close(notify_wr_id_);
|
||||||
}
|
}
|
||||||
|
|
||||||
void WsClient::Write(const uint8_t** buffers, const size_t* lengths,
|
int WsClient::socket_id() {
|
||||||
size_t count) {
|
return socket_id_;
|
||||||
//
|
}
|
||||||
|
|
||||||
|
int WsClient::Setup() {
|
||||||
|
// Prep the socket.
|
||||||
|
int opt_value;
|
||||||
|
opt_value = 1;
|
||||||
|
setsockopt(socket_id_, SOL_SOCKET, SO_KEEPALIVE,
|
||||||
|
&opt_value, sizeof(opt_value));
|
||||||
|
opt_value = 1;
|
||||||
|
setsockopt(socket_id_, IPPROTO_TCP, TCP_NODELAY,
|
||||||
|
&opt_value, sizeof(opt_value));
|
||||||
|
|
||||||
|
// launch thread/etc
|
||||||
|
EventThread();
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
ssize_t WsClientSendCallback(wslay_event_context_ptr ctx,
|
||||||
|
const uint8_t* data, size_t len, int flags,
|
||||||
|
void* user_data) {
|
||||||
|
WsClient* client = reinterpret_cast<WsClient*>(user_data);
|
||||||
|
|
||||||
|
int sflags = 0;
|
||||||
|
#ifdef MSG_MORE
|
||||||
|
if (flags & WSLAY_MSG_MORE) {
|
||||||
|
sflags |= MSG_MORE;
|
||||||
|
}
|
||||||
|
#endif // MSG_MORE
|
||||||
|
|
||||||
|
ssize_t r;
|
||||||
|
while ((r = send(client->socket_id(), data, len, sflags)) == -1 &&
|
||||||
|
errno == EINTR);
|
||||||
|
if (r == -1) {
|
||||||
|
if (errno == EAGAIN || errno == EWOULDBLOCK) {
|
||||||
|
wslay_event_set_error(ctx, WSLAY_ERR_WOULDBLOCK);
|
||||||
|
} else {
|
||||||
|
wslay_event_set_error(ctx, WSLAY_ERR_CALLBACK_FAILURE);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
ssize_t WsClientRecvCallback(wslay_event_context_ptr ctx,
|
||||||
|
uint8_t* data, size_t len, int flags,
|
||||||
|
void* user_data) {
|
||||||
|
WsClient* client = reinterpret_cast<WsClient*>(user_data);
|
||||||
|
ssize_t r;
|
||||||
|
while ((r = recv(client->socket_id(), data, len, 0)) == -1 &&
|
||||||
|
errno == EINTR);
|
||||||
|
if (r == -1) {
|
||||||
|
if (errno == EAGAIN || errno == EWOULDBLOCK) {
|
||||||
|
wslay_event_set_error(ctx, WSLAY_ERR_WOULDBLOCK);
|
||||||
|
} else {
|
||||||
|
wslay_event_set_error(ctx, WSLAY_ERR_CALLBACK_FAILURE);
|
||||||
|
}
|
||||||
|
} else if (r == 0) {
|
||||||
|
wslay_event_set_error(ctx, WSLAY_ERR_CALLBACK_FAILURE);
|
||||||
|
r = -1;
|
||||||
|
}
|
||||||
|
return r;
|
||||||
|
}
|
||||||
|
|
||||||
|
void WsClientOnMsgCallback(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;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (arg->opcode) {
|
||||||
|
case WSLAY_TEXT_FRAME:
|
||||||
|
// arg->msg, arg->msg_length
|
||||||
|
break;
|
||||||
|
case WSLAY_BINARY_FRAME:
|
||||||
|
// arg->msg, arg->msg_length
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
// Unknown opcode - some frame stuff?
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO(benvanik): dispatch
|
||||||
|
|
||||||
|
// WsClient* client = reinterpret_cast<WsClient*>(user_data);
|
||||||
|
// char hello[] = "HELLO";
|
||||||
|
// size_t len = sizeof(hello);
|
||||||
|
// uint8_t* bs[] = {(uint8_t*)&hello};
|
||||||
|
// size_t lens[] = {len};
|
||||||
|
// client->Write((uint8_t**)bs, lens, 1);
|
||||||
|
// printf("on msg %d, %ld\n", arg->opcode, arg->msg_length);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string EncodeBase64(const uint8_t* input, size_t length) {
|
||||||
|
// Good god what a horrible API.
|
||||||
|
BIO* b64 = BIO_new(BIO_f_base64());
|
||||||
|
BIO* bmem = BIO_new(BIO_s_mem());
|
||||||
|
b64 = BIO_push(b64, bmem);
|
||||||
|
BIO_write(b64, input, length);
|
||||||
|
XEIGNORE(BIO_flush(b64));
|
||||||
|
BUF_MEM* bptr;
|
||||||
|
BIO_get_mem_ptr(b64, &bptr);
|
||||||
|
std::string result(bptr->data, bptr->length);
|
||||||
|
BIO_free_all(b64);
|
||||||
|
// Strip the last character, which is a \n.
|
||||||
|
result.erase(result.size() - 1);
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
int WsClient::PerformHandshake() {
|
||||||
|
std::string headers;
|
||||||
|
char buffer[4096];
|
||||||
|
ssize_t r;
|
||||||
|
while (true) {
|
||||||
|
while ((r = read(socket_id_, buffer, sizeof(buffer))) == -1 &&
|
||||||
|
errno == EINTR);
|
||||||
|
if (r == -1) {
|
||||||
|
if (errno == EWOULDBLOCK || errno == EAGAIN) {
|
||||||
|
if (!headers.size()) {
|
||||||
|
// Nothing read yet - spin.
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
XELOGE(XT("HTTP header read failure"));
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
} else if (r == 0) {
|
||||||
|
// EOF.
|
||||||
|
XELOGE(XT("HTTP header EOF"));
|
||||||
|
return 2;
|
||||||
|
} else {
|
||||||
|
headers.append(buffer, buffer + r);
|
||||||
|
if (headers.size() > 8192) {
|
||||||
|
XELOGE(XT("HTTP headers exceeded max buffer size"));
|
||||||
|
return 3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (headers.find("\r\n\r\n") == std::string::npos) {
|
||||||
|
XELOGE(XT("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(XT("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 = write(socket_id_, response.c_str() + write_offset,
|
||||||
|
write_length)) == -1 && errno == EINTR);
|
||||||
|
if (r == -1) {
|
||||||
|
if (errno == EAGAIN || errno == EWOULDBLOCK) {
|
||||||
|
break;
|
||||||
|
} else {
|
||||||
|
XELOGE(XT("HTTP response write failure"));
|
||||||
|
return 4;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
write_offset += r;
|
||||||
|
write_length -= r;
|
||||||
|
if (!write_length) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void WsClient::EventThread() {
|
||||||
|
int r;
|
||||||
|
|
||||||
|
// Enable non-blocking IO on the socket.
|
||||||
|
int flags;
|
||||||
|
while ((flags = fcntl(socket_id_, F_GETFL, 0)) == -1 &&
|
||||||
|
errno == EINTR);
|
||||||
|
if (flags == -1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
while ((r = fcntl(socket_id_, F_SETFL, flags | O_NONBLOCK)) == -1 &&
|
||||||
|
errno == EINTR);
|
||||||
|
if (r == -1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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 = {
|
||||||
|
WsClientRecvCallback,
|
||||||
|
WsClientSendCallback,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
NULL,
|
||||||
|
WsClientOnMsgCallback,
|
||||||
|
};
|
||||||
|
|
||||||
|
// Prep the websocket server context.
|
||||||
|
wslay_event_context_ptr ctx;
|
||||||
|
wslay_event_context_server_init(&ctx, &callbacks, this);
|
||||||
|
|
||||||
|
// Event for waiting on input.
|
||||||
|
struct pollfd events[2];
|
||||||
|
xe_zero_struct(&events, sizeof(events));
|
||||||
|
events[0].fd = socket_id_;
|
||||||
|
events[0].events = POLLIN;
|
||||||
|
events[1].fd = notify_rd_id_;
|
||||||
|
events[1].events = POLLIN;
|
||||||
|
|
||||||
|
// Loop forever.
|
||||||
|
while(wslay_event_want_read(ctx) || wslay_event_want_write(ctx)) {
|
||||||
|
// Wait on the event.
|
||||||
|
while ((r = poll(events, XECOUNT(events), -1)) == -1 && errno == EINTR);
|
||||||
|
if (r == -1) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle any self-generated events to queue messages.
|
||||||
|
if (events[1].revents) {
|
||||||
|
uint8_t dummy;
|
||||||
|
XEIGNORE(recv(notify_rd_id_, &dummy, 1, 0));
|
||||||
|
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_);
|
||||||
|
events[1].revents = 0;
|
||||||
|
events[1].events = POLL_IN;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle websocket messages.
|
||||||
|
struct pollfd* event = &events[0];
|
||||||
|
if (((event->revents & POLLIN) && wslay_event_recv(ctx)) ||
|
||||||
|
((event->revents & POLLOUT) && wslay_event_send(ctx)) ||
|
||||||
|
((event->revents & (POLLERR | POLLHUP | POLLNVAL)))) {
|
||||||
|
// Error handling the event.
|
||||||
|
XELOGE(XT("Error handling WebSocket data"));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
event->revents = 0;
|
||||||
|
event->events = 0;
|
||||||
|
if (wslay_event_want_read(ctx)) {
|
||||||
|
event->events |= POLLIN;
|
||||||
|
}
|
||||||
|
if (wslay_event_want_write(ctx)) {
|
||||||
|
event->events |= POLLOUT;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
wslay_event_context_free(ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WsClient::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 = {
|
||||||
|
WSLAY_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().
|
||||||
|
uint8_t b = 0xFF;
|
||||||
|
write(notify_wr_id_, &b, 1);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,9 +13,14 @@
|
||||||
#include <xenia/common.h>
|
#include <xenia/common.h>
|
||||||
#include <xenia/core.h>
|
#include <xenia/core.h>
|
||||||
|
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
#include <xenia/dbg/client.h>
|
#include <xenia/dbg/client.h>
|
||||||
|
|
||||||
|
|
||||||
|
struct wslay_event_msg;
|
||||||
|
|
||||||
|
|
||||||
namespace xe {
|
namespace xe {
|
||||||
namespace dbg {
|
namespace dbg {
|
||||||
|
|
||||||
|
@ -25,11 +30,23 @@ public:
|
||||||
WsClient(int socket_id);
|
WsClient(int socket_id);
|
||||||
virtual ~WsClient();
|
virtual ~WsClient();
|
||||||
|
|
||||||
virtual void Write(const uint8_t** buffers, const size_t* lengths,
|
int socket_id();
|
||||||
size_t count);
|
|
||||||
|
virtual int Setup();
|
||||||
|
|
||||||
|
virtual void Write(uint8_t** buffers, size_t* lengths, size_t count);
|
||||||
|
|
||||||
|
private:
|
||||||
|
int PerformHandshake();
|
||||||
|
void EventThread();
|
||||||
|
|
||||||
protected:
|
|
||||||
int socket_id_;
|
int socket_id_;
|
||||||
|
|
||||||
|
int notify_rd_id_;
|
||||||
|
int notify_wr_id_;
|
||||||
|
xe_mutex_t* mutex_;
|
||||||
|
|
||||||
|
std::vector<struct wslay_event_msg> pending_messages_;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -42,6 +42,9 @@ int WsListener::Setup() {
|
||||||
opt_value = 1;
|
opt_value = 1;
|
||||||
setsockopt(socket_id_, SOL_SOCKET, SO_KEEPALIVE,
|
setsockopt(socket_id_, SOL_SOCKET, SO_KEEPALIVE,
|
||||||
&opt_value, sizeof(opt_value));
|
&opt_value, sizeof(opt_value));
|
||||||
|
opt_value = 1;
|
||||||
|
setsockopt(socket_id_, SOL_SOCKET, SO_REUSEADDR,
|
||||||
|
&opt_value, sizeof(opt_value));
|
||||||
opt_value = 0;
|
opt_value = 0;
|
||||||
setsockopt(socket_id_, IPPROTO_TCP, TCP_NODELAY,
|
setsockopt(socket_id_, IPPROTO_TCP, TCP_NODELAY,
|
||||||
&opt_value, sizeof(opt_value));
|
&opt_value, sizeof(opt_value));
|
||||||
|
@ -50,8 +53,10 @@ int WsListener::Setup() {
|
||||||
socket_addr.sin_family = AF_INET;
|
socket_addr.sin_family = AF_INET;
|
||||||
socket_addr.sin_addr.s_addr = htonl(INADDR_ANY);
|
socket_addr.sin_addr.s_addr = htonl(INADDR_ANY);
|
||||||
socket_addr.sin_port = htons(port_);
|
socket_addr.sin_port = htons(port_);
|
||||||
if (bind(socket_id_, (struct sockaddr*)&socket_addr,
|
int r = bind(socket_id_, (struct sockaddr*)&socket_addr,
|
||||||
sizeof(socket_addr)) < 0) {
|
sizeof(socket_addr));
|
||||||
|
if (r < 0) {
|
||||||
|
XELOGE(XT("Could not bind listen socket: %d"), errno);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -78,9 +83,13 @@ int WsListener::WaitForClient() {
|
||||||
inet_ntop(AF_INET, &client_ip, client_ip_str, XECOUNT(client_ip_str));
|
inet_ntop(AF_INET, &client_ip, client_ip_str, XECOUNT(client_ip_str));
|
||||||
XELOGI(XT("Debugger connected from %s"), client_ip_str);
|
XELOGI(XT("Debugger connected from %s"), client_ip_str);
|
||||||
|
|
||||||
//WsClient* client = new WsClient(client_socket_id);
|
// Create the client object.
|
||||||
|
// Note that the client will delete itself when done.
|
||||||
// TODO(benvanik): add to list for cleanup
|
WsClient* client = new WsClient(client_socket_id);
|
||||||
|
if (client->Setup()) {
|
||||||
|
// Client failed to setup - abort.
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
'targets': [
|
'targets': [
|
||||||
{
|
{
|
||||||
'target_name': 'gflags',
|
'target_name': 'gflags',
|
||||||
'type': 'static_library',
|
'type': '<(library)',
|
||||||
|
|
||||||
'direct_dependent_settings': {
|
'direct_dependent_settings': {
|
||||||
'conditions': [
|
'conditions': [
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit d1b1f821ba7b0203e065c26a12b341eac382514d
|
|
@ -3,7 +3,7 @@
|
||||||
'targets': [
|
'targets': [
|
||||||
{
|
{
|
||||||
'target_name': 'sparsehash',
|
'target_name': 'sparsehash',
|
||||||
'type': 'static_library',
|
'type': '<(library)',
|
||||||
|
|
||||||
'direct_dependent_settings': {
|
'direct_dependent_settings': {
|
||||||
'include_dirs': [
|
'include_dirs': [
|
||||||
|
|
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit d2caaf13db958249d3878926238f1096a459a11b
|
|
@ -0,0 +1,52 @@
|
||||||
|
# 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',
|
||||||
|
],
|
||||||
|
}],
|
||||||
|
],
|
||||||
|
|
||||||
|
'include_dirs': [
|
||||||
|
'wslay/lib/',
|
||||||
|
'wslay/lib/includes/',
|
||||||
|
],
|
||||||
|
|
||||||
|
'sources': [
|
||||||
|
'wslay/lib/wslay_event.c',
|
||||||
|
'wslay/lib/wslay_frame.c',
|
||||||
|
'wslay/lib/wslay_net.c',
|
||||||
|
'wslay/lib/wslay_queue.c',
|
||||||
|
'wslay/lib/wslay_stack.c',
|
||||||
|
],
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
|
@ -348,6 +348,7 @@ def run_gyp(format):
|
||||||
"""
|
"""
|
||||||
shell_call(' '.join([
|
shell_call(' '.join([
|
||||||
'gyp',
|
'gyp',
|
||||||
|
'--include=common.gypi',
|
||||||
'-f %s' % (format),
|
'-f %s' % (format),
|
||||||
# Set the VS version.
|
# Set the VS version.
|
||||||
# TODO(benvanik): allow user to set?
|
# TODO(benvanik): allow user to set?
|
||||||
|
|
13
xenia.gyp
13
xenia.gyp
|
@ -1,23 +1,26 @@
|
||||||
# Copyright 2013 Ben Vanik. All Rights Reserved.
|
# Copyright 2013 Ben Vanik. All Rights Reserved.
|
||||||
{
|
{
|
||||||
'includes': [
|
'includes': [
|
||||||
'common.gypi',
|
|
||||||
'tools/tools.gypi',
|
'tools/tools.gypi',
|
||||||
'third_party/gflags.gypi',
|
'third_party/gflags.gypi',
|
||||||
'third_party/sparsehash.gypi',
|
'third_party/sparsehash.gypi',
|
||||||
|
'third_party/wslay.gypi',
|
||||||
],
|
],
|
||||||
|
|
||||||
'targets': [
|
'targets': [
|
||||||
{
|
{
|
||||||
'target_name': 'xeniacore',
|
'target_name': 'xeniacore',
|
||||||
'product_name': 'xeniacore',
|
'product_name': 'xeniacore',
|
||||||
'type': 'static_library',
|
'type': '<(library)',
|
||||||
|
|
||||||
'dependencies': [
|
'dependencies': [
|
||||||
'gflags',
|
'gflags',
|
||||||
|
'wslay',
|
||||||
|
'./third_party/openssl/openssl.gyp:openssl',
|
||||||
],
|
],
|
||||||
'export_dependent_settings': [
|
'export_dependent_settings': [
|
||||||
'gflags',
|
'gflags',
|
||||||
|
'wslay',
|
||||||
],
|
],
|
||||||
|
|
||||||
'direct_dependent_settings': {
|
'direct_dependent_settings': {
|
||||||
|
@ -41,7 +44,7 @@
|
||||||
{
|
{
|
||||||
'target_name': 'xeniakernel',
|
'target_name': 'xeniakernel',
|
||||||
'product_name': 'xeniakernel',
|
'product_name': 'xeniakernel',
|
||||||
'type': 'static_library',
|
'type': '<(library)',
|
||||||
|
|
||||||
'dependencies': [
|
'dependencies': [
|
||||||
'xeniacore',
|
'xeniacore',
|
||||||
|
@ -63,7 +66,7 @@
|
||||||
{
|
{
|
||||||
'target_name': 'xeniacpu',
|
'target_name': 'xeniacpu',
|
||||||
'product_name': 'xeniacpu',
|
'product_name': 'xeniacpu',
|
||||||
'type': 'static_library',
|
'type': '<(library)',
|
||||||
|
|
||||||
'dependencies': [
|
'dependencies': [
|
||||||
'xeniacore',
|
'xeniacore',
|
||||||
|
@ -121,7 +124,7 @@
|
||||||
{
|
{
|
||||||
'target_name': 'xeniagpu',
|
'target_name': 'xeniagpu',
|
||||||
'product_name': 'xeniagpu',
|
'product_name': 'xeniagpu',
|
||||||
'type': 'static_library',
|
'type': '<(library)',
|
||||||
|
|
||||||
'dependencies': [
|
'dependencies': [
|
||||||
'xeniacore',
|
'xeniacore',
|
||||||
|
|
Loading…
Reference in New Issue