#ifndef SRC_INSPECTOR_IO_H_ #define SRC_INSPECTOR_IO_H_ #include "inspector_socket_server.h" #include "node_debug_options.h" #include "node_mutex.h" #include "uv.h" #include #include #include #if !HAVE_INSPECTOR #error("This header can only be used when inspector is enabled") #endif // Forward declaration to break recursive dependency chain with src/env.h. namespace node { class Environment; } // namespace node namespace v8_inspector { class StringBuffer; class StringView; } // namespace v8_inspector namespace node { namespace inspector { class InspectorIoDelegate; enum class InspectorAction { kStartSession, kEndSession, kSendMessage }; // kKill closes connections and stops the server, kStop only stops the server enum class TransportAction { kKill, kSendMessage, kStop }; class InspectorIo { public: InspectorIo(node::Environment* env, v8::Platform* platform, const std::string& path, const DebugOptions& options, bool wait_for_connect); ~InspectorIo(); // Start the inspector agent thread, waiting for it to initialize, // and waiting as well for a connection if wait_for_connect. bool Start(); // Stop the inspector agent thread. void Stop(); bool IsStarted(); bool IsConnected(); void WaitForDisconnect(); // Called from thread to queue an incoming message and trigger // DispatchMessages() on the main thread. void PostIncomingMessage(InspectorAction action, int session_id, const std::string& message); void ResumeStartup() { uv_sem_post(&thread_start_sem_); } void ServerDone() { uv_close(reinterpret_cast(&thread_req_), nullptr); } int port() const { return port_; } private: template using MessageQueue = std::vector>>; enum class State { kNew, kAccepting, kConnected, kDone, kError, kShutDown }; // Callback for main_thread_req_'s uv_async_t static void MainThreadReqAsyncCb(uv_async_t* req); // Wrapper for agent->ThreadMain() static void ThreadMain(void* agent); // Runs a uv_loop_t template void ThreadMain(); // Called by ThreadMain's loop when triggered by thread_req_, writes // messages from outgoing_message_queue to the InspectorSockerServer template static void IoThreadAsyncCb(uv_async_t* async); void SetConnected(bool connected); void DispatchMessages(); // Write action to outgoing_message_queue, and wake the thread void Write(TransportAction action, int session_id, const v8_inspector::StringView& message); // Thread-safe append of message to a queue. Return true if the queue // used to be empty. template bool AppendMessage(MessageQueue* vector, ActionType action, int session_id, std::unique_ptr buffer); // Used as equivalent of a thread-safe "pop" of an entire queue's content. template void SwapBehindLock(MessageQueue* vector1, MessageQueue* vector2); // Wait on incoming_message_cond_ void WaitForIncomingMessage(); // Broadcast incoming_message_cond_ void NotifyMessageReceived(); const DebugOptions options_; // The IO thread runs its own uv_loop to implement the TCP server off // the main thread. uv_thread_t thread_; // Used by Start() to wait for thread to initialize, or for it to initialize // and receive a connection if wait_for_connect was requested. uv_sem_t thread_start_sem_; InspectorIoDelegate* delegate_; State state_; node::Environment* parent_env_; // Attached to the uv_loop in ThreadMain() uv_async_t thread_req_; // Note that this will live while the async is being closed - likely, past // the parent object lifespan std::pair* main_thread_req_; std::unique_ptr session_delegate_; v8::Platform* platform_; // Message queues ConditionVariable incoming_message_cond_; Mutex state_lock_; // Locked before mutating either queue. MessageQueue incoming_message_queue_; MessageQueue outgoing_message_queue_; bool dispatching_messages_; int session_id_; std::string script_name_; std::string script_path_; const std::string id_; const bool wait_for_connect_; int port_; friend class DispatchMessagesTask; friend class IoSessionDelegate; friend void InterruptCallback(v8::Isolate*, void* agent); }; std::unique_ptr Utf8ToStringView( const std::string& message); } // namespace inspector } // namespace node #endif // SRC_INSPECTOR_IO_H_