summaryrefslogtreecommitdiff
path: root/editor/debugger/editor_debugger_server.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'editor/debugger/editor_debugger_server.cpp')
-rw-r--r--editor/debugger/editor_debugger_server.cpp216
1 files changed, 216 insertions, 0 deletions
diff --git a/editor/debugger/editor_debugger_server.cpp b/editor/debugger/editor_debugger_server.cpp
new file mode 100644
index 0000000000..bde0921ced
--- /dev/null
+++ b/editor/debugger/editor_debugger_server.cpp
@@ -0,0 +1,216 @@
+/*************************************************************************/
+/* editor_debugger_server.cpp */
+/*************************************************************************/
+/* This file is part of: */
+/* GODOT ENGINE */
+/* https://godotengine.org */
+/*************************************************************************/
+/* Copyright (c) 2007-2020 Juan Linietsky, Ariel Manzur. */
+/* Copyright (c) 2014-2020 Godot Engine contributors (cf. AUTHORS.md). */
+/* */
+/* Permission is hereby granted, free of charge, to any person obtaining */
+/* a copy of this software and associated documentation files (the */
+/* "Software"), to deal in the Software without restriction, including */
+/* without limitation the rights to use, copy, modify, merge, publish, */
+/* distribute, sublicense, and/or sell copies of the Software, and to */
+/* permit persons to whom the Software is furnished to do so, subject to */
+/* the following conditions: */
+/* */
+/* The above copyright notice and this permission notice shall be */
+/* included in all copies or substantial portions of the Software. */
+/* */
+/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
+/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
+/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
+/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
+/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
+/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
+/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
+/*************************************************************************/
+
+#include "editor_debugger_server.h"
+
+#include "core/io/packet_peer.h"
+#include "core/io/tcp_server.h"
+#include "core/os/mutex.h"
+#include "core/os/thread.h"
+#include "editor/editor_log.h"
+#include "editor/editor_node.h"
+#include "editor/editor_settings.h"
+
+class EditorDebuggerPeerTCP : public EditorDebuggerPeer {
+
+private:
+ enum {
+ QUEUE_MAX = 2048,
+ POLL_USEC_MAX = 100,
+ };
+ Ref<StreamPeerTCP> tcp;
+ Ref<PacketPeerStream> packet_peer;
+ List<Array> out_queue;
+ List<Array> in_queue;
+ Mutex mutex;
+ bool connected = false;
+
+public:
+ Error poll() {
+ MutexLock lock(mutex);
+ connected = tcp->get_status() == StreamPeerTCP::STATUS_CONNECTED;
+ Error err = OK;
+ uint64_t ticks = OS::get_singleton()->get_ticks_usec();
+ while (connected && packet_peer->get_available_packet_count() > 0 && in_queue.size() < QUEUE_MAX && OS::get_singleton()->get_ticks_usec() - ticks < POLL_USEC_MAX) {
+ Variant var;
+ err = packet_peer->get_var(var);
+ connected = tcp->get_status() == StreamPeerTCP::STATUS_CONNECTED;
+ if (err != OK) {
+ ERR_PRINT("Error reading variant from peer");
+ break;
+ }
+ ERR_CONTINUE_MSG(var.get_type() != Variant::ARRAY, "Malformed packet received, not an Array.");
+ in_queue.push_back(var);
+ }
+ ticks = OS::get_singleton()->get_ticks_usec();
+ while (connected && out_queue.size() > 0 && OS::get_singleton()->get_ticks_usec() - ticks < POLL_USEC_MAX) {
+ Array arr = out_queue[0];
+ out_queue.pop_front();
+ packet_peer->put_var(arr);
+ connected = tcp->get_status() == StreamPeerTCP::STATUS_CONNECTED;
+ }
+ return err;
+ }
+
+ bool has_message() {
+ return in_queue.size() > 0;
+ }
+
+ Array get_message() {
+ MutexLock lock(mutex);
+ ERR_FAIL_COND_V(!has_message(), Array());
+ Array out = in_queue[0];
+ in_queue.pop_front();
+ return out;
+ }
+
+ Error put_message(const Array p_arr) {
+ MutexLock lock(mutex);
+ if (out_queue.size() > QUEUE_MAX) {
+ return ERR_OUT_OF_MEMORY;
+ }
+ out_queue.push_back(p_arr);
+ return OK;
+ }
+
+ int get_max_message_size() const {
+ return 8 << 20; // 8 MiB
+ }
+
+ bool is_peer_connected() {
+ return connected;
+ }
+
+ void close() {
+ MutexLock lock(mutex);
+ connected = false;
+ tcp->disconnect_from_host();
+ }
+
+ EditorDebuggerPeerTCP(Ref<StreamPeerTCP> p_stream) {
+ packet_peer.instance();
+ tcp = p_stream;
+ if (tcp.is_null()) {
+ tcp.instance(); // Bug?
+ }
+ packet_peer->set_stream_peer(tcp);
+ }
+
+ ~EditorDebuggerPeerTCP() {
+ close();
+ packet_peer->set_stream_peer(Ref<StreamPeer>());
+ }
+};
+
+class EditorDebuggerServerTCP : public EditorDebuggerServer {
+
+private:
+ Ref<TCP_Server> server;
+ List<Ref<EditorDebuggerPeer> > peers;
+ Thread *thread = NULL;
+ Mutex mutex;
+ bool running = false;
+
+ static void _poll_func(void *p_ud);
+
+public:
+ virtual Error start();
+ virtual void stop();
+ virtual bool is_active() const;
+ virtual bool is_connection_available() const;
+ virtual Ref<EditorDebuggerPeer> take_connection();
+
+ EditorDebuggerServerTCP();
+};
+
+EditorDebuggerServerTCP::EditorDebuggerServerTCP() {
+ server.instance();
+}
+
+Error EditorDebuggerServerTCP::start() {
+ int remote_port = (int)EditorSettings::get_singleton()->get("network/debug/remote_port");
+ const Error err = server->listen(remote_port);
+ if (err != OK) {
+ EditorNode::get_log()->add_message(String("Error listening on port ") + itos(remote_port), EditorLog::MSG_TYPE_ERROR);
+ return err;
+ }
+ running = true;
+ thread = Thread::create(_poll_func, this);
+ return err;
+}
+
+void EditorDebuggerServerTCP::stop() {
+ server->stop();
+ if (thread != NULL) {
+ running = false;
+ Thread::wait_to_finish(thread);
+ memdelete(thread);
+ thread = NULL;
+ }
+}
+
+bool EditorDebuggerServerTCP::is_active() const {
+ return server->is_listening();
+}
+
+bool EditorDebuggerServerTCP::is_connection_available() const {
+ return server->is_listening() && server->is_connection_available();
+}
+
+Ref<EditorDebuggerPeer> EditorDebuggerServerTCP::take_connection() {
+ ERR_FAIL_COND_V(!is_connection_available(), Ref<EditorDebuggerPeer>());
+ MutexLock lock(mutex);
+ Ref<EditorDebuggerPeerTCP> peer = memnew(EditorDebuggerPeerTCP(server->take_connection()));
+ peers.push_back(peer);
+ return peer;
+}
+
+void EditorDebuggerServerTCP::_poll_func(void *p_ud) {
+ EditorDebuggerServerTCP *me = (EditorDebuggerServerTCP *)p_ud;
+ while (me->running) {
+ me->mutex.lock();
+ List<Ref<EditorDebuggerPeer> > remove;
+ for (int i = 0; i < me->peers.size(); i++) {
+ Ref<EditorDebuggerPeer> peer = me->peers[i];
+ Error err = ((EditorDebuggerPeerTCP *)peer.ptr())->poll();
+ if (err != OK || !peer->is_peer_connected())
+ remove.push_back(peer);
+ }
+ for (List<Ref<EditorDebuggerPeer> >::Element *E = remove.front(); E; E = E->next()) {
+ me->peers.erase(E->get());
+ }
+ me->mutex.unlock();
+ OS::get_singleton()->delay_usec(50);
+ }
+}
+
+EditorDebuggerServer *EditorDebuggerServer::create_default() {
+ return memnew(EditorDebuggerServerTCP);
+}