//===-- JSONUtils.h ---------------------------------------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #ifndef LLDB_TOOLS_LLDB_DAP_JSONUTILS_H #define LLDB_TOOLS_LLDB_DAP_JSONUTILS_H #include "DAPForward.h" #include "Protocol/ProtocolTypes.h" #include "lldb/API/SBCompileUnit.h" #include "lldb/API/SBFormat.h" #include "lldb/API/SBType.h" #include "lldb/API/SBValue.h" #include "lldb/lldb-types.h" #include "llvm/ADT/StringMap.h" #include "llvm/ADT/StringRef.h" #include "llvm/Support/JSON.h" #include #include #include #include #include #include namespace lldb_dap { /// Emplace a StringRef in a json::Object after enusring that the /// string is valid UTF8. If not, first call llvm::json::fixUTF8 /// before emplacing. /// /// \param[in] obj /// A JSON object that we will attempt to emplace the value in /// /// \param[in] key /// The key to use when emplacing the value /// /// \param[in] str /// The string to emplace void EmplaceSafeString(llvm::json::Object &obj, llvm::StringRef key, llvm::StringRef str); /// Extract simple values as a string. /// /// \param[in] value /// A JSON value to extract the string from. /// /// \return /// A llvm::StringRef that contains the string value, or an empty /// string if \a value isn't a string. llvm::StringRef GetAsString(const llvm::json::Value &value); /// Extract the string value for the specified key from the /// specified object. /// /// \param[in] obj /// A JSON object that we will attempt to extract the value from /// /// \param[in] key /// The key to use when extracting the value /// /// \return /// A llvm::StringRef that contains the string value for the /// specified \a key, or \a std::nullopt if there is no key that /// matches or if the value is not a string. std::optional GetString(const llvm::json::Object &obj, llvm::StringRef key); std::optional GetString(const llvm::json::Object *obj, llvm::StringRef key); /// Extract the integer value for the specified key from the specified object /// and return it as the specified integer type T. /// /// \param[in] obj /// A JSON object that we will attempt to extract the value from /// /// \param[in] key /// The key to use when extracting the value /// /// \return /// The integer value for the specified \a key, or std::nullopt if there is /// no key that matches or if the value is not an integer. /// @{ template std::optional GetInteger(const llvm::json::Object &obj, llvm::StringRef key) { return obj.getInteger(key); } template std::optional GetInteger(const llvm::json::Object *obj, llvm::StringRef key) { if (obj != nullptr) return GetInteger(*obj, key); return std::nullopt; } /// @} /// Extract the boolean value for the specified key from the /// specified object. /// /// \param[in] obj /// A JSON object that we will attempt to extract the value from /// /// \param[in] key /// The key to use when extracting the value /// /// \return /// The boolean value for the specified \a key, or std::nullopt /// if there is no key that matches or if the value is not a /// boolean value of an integer. /// @{ std::optional GetBoolean(const llvm::json::Object &obj, llvm::StringRef key); std::optional GetBoolean(const llvm::json::Object *obj, llvm::StringRef key); /// @} /// Check if the specified key exists in the specified object. /// /// \param[in] obj /// A JSON object that we will attempt to extract the value from /// /// \param[in] key /// The key to check for /// /// \return /// \b True if the key exists in the \a obj, \b False otherwise. bool ObjectContainsKey(const llvm::json::Object &obj, llvm::StringRef key); /// Encodes a memory reference std::string EncodeMemoryReference(lldb::addr_t addr); /// Decodes a memory reference std::optional DecodeMemoryReference(llvm::StringRef memoryReference); /// Decodes a memory reference from the given json value. /// /// \param[in] v /// A JSON value that we expected to contain the memory reference. /// /// \param[in] key /// The key of the memory reference. /// /// \param[out] out /// The memory address, if successfully decoded. /// /// \param[in] path /// The path for reporting errors. /// /// \param[in] required /// Indicates if the key is required to be present, otherwise report an error /// if the key is missing. /// /// \return /// Returns \b true if the address was decoded successfully. bool DecodeMemoryReference(const llvm::json::Value &v, llvm::StringLiteral key, lldb::addr_t &out, llvm::json::Path path, bool required); /// Extract an array of strings for the specified key from an object. /// /// String values in the array will be extracted without any quotes /// around them. Numbers and Booleans will be converted into /// strings. Any NULL, array or objects values in the array will be /// ignored. /// /// \param[in] obj /// A JSON object that we will attempt to extract the array from /// /// \param[in] key /// The key to use when extracting the value /// /// \return /// An array of string values for the specified \a key, or /// \a fail_value if there is no key that matches or if the /// value is not an array or all items in the array are not /// strings, numbers or booleans. std::vector GetStrings(const llvm::json::Object *obj, llvm::StringRef key); /// Extract an object of key value strings for the specified key from an object. /// /// String values in the object will be extracted without any quotes /// around them. Numbers and Booleans will be converted into /// strings. Any NULL, array or objects values in the array will be /// ignored. /// /// \param[in] obj /// A JSON object that we will attempt to extract the array from /// /// \param[in] key /// The key to use when extracting the value /// /// \return /// An object of key value strings for the specified \a key, or /// \a fail_value if there is no key that matches or if the /// value is not an object or key and values in the object are not /// strings, numbers or booleans. std::unordered_map GetStringMap(const llvm::json::Object &obj, llvm::StringRef key); /// Fill a response object given the request object. /// /// The \a response object will get its "type" set to "response", /// the "seq" set to zero, "response_seq" set to the "seq" value from /// \a request, "command" set to the "command" from \a request, /// and "success" set to true. /// /// \param[in] request /// The request object received from a call to DAP::ReadJSON(). /// /// \param[in,out] response /// An empty llvm::json::Object object that will be filled /// in as noted in description. void FillResponse(const llvm::json::Object &request, llvm::json::Object &response); /// Create a "Event" JSON object using \a event_name as the event name /// /// \param[in] event_name /// The string value to use for the "event" key in the JSON object. /// /// \return /// A "Event" JSON object with that follows the formal JSON /// definition outlined by Microsoft. llvm::json::Object CreateEventObject(const llvm::StringRef event_name); /// Create a "StackFrame" object for a LLDB frame object. /// /// This function will fill in the following keys in the returned /// object: /// "id" - the stack frame ID as an integer /// "name" - the function name as a string /// "source" - source file information as a "Source" DAP object /// "line" - the source file line number as an integer /// "column" - the source file column number as an integer /// /// \param[in] dap /// The DAP session associated with the stopped thread. /// /// \param[in] frame /// The LLDB stack frame to use when populating out the "StackFrame" /// object. /// /// \param[in] format /// The LLDB format to use when populating out the "StackFrame" /// object. /// /// \return /// A "StackFrame" JSON object with that follows the formal JSON /// definition outlined by Microsoft. llvm::json::Value CreateStackFrame(DAP &dap, lldb::SBFrame &frame, lldb::SBFormat &format); /// Create a "StackFrame" label object for a LLDB thread. /// /// This function will fill in the following keys in the returned /// object: /// "id" - the thread ID as an integer /// "name" - the thread name as a string which combines the LLDB /// thread index ID along with the string name of the thread /// from the OS if it has a name. /// "presentationHint" - "label" /// /// \param[in] thread /// The LLDB thread to use when populating out the "Thread" /// object. /// /// \param[in] format /// The configured formatter for the DAP session. /// /// \return /// A "StackFrame" JSON object with that follows the formal JSON /// definition outlined by Microsoft. llvm::json::Value CreateExtendedStackFrameLabel(lldb::SBThread &thread, lldb::SBFormat &format); /// Create a "StoppedEvent" object for a LLDB thread object. /// /// This function will fill in the following keys in the returned /// object's "body" object: /// "reason" - With a valid stop reason enumeration string value /// that Microsoft specifies /// "threadId" - The thread ID as an integer /// "description" - a stop description (like "breakpoint 12.3") as a /// string /// "preserveFocusHint" - a boolean value that states if this thread /// should keep the focus in the GUI. /// "allThreadsStopped" - set to True to indicate that all threads /// stop when any thread stops. /// /// \param[in] dap /// The DAP session associated with the stopped thread. /// /// \param[in] thread /// The LLDB thread to use when populating out the "StoppedEvent" /// object. /// /// \param[in] stop_id /// The stop id for this event. /// /// \return /// A "StoppedEvent" JSON object with that follows the formal JSON /// definition outlined by Microsoft. llvm::json::Value CreateThreadStopped(DAP &dap, lldb::SBThread &thread, uint32_t stop_id); /// \return /// The variable name of \a value or a default placeholder. const char *GetNonNullVariableName(lldb::SBValue &value); /// VSCode can't display two variables with the same name, so we need to /// distinguish them by using a suffix. /// /// If the source and line information is present, we use it as the suffix. /// Otherwise, we fallback to the variable address or register location. std::string CreateUniqueVariableNameForDisplay(lldb::SBValue &v, bool is_name_duplicated); /// Helper struct that parses the metadata of an \a lldb::SBValue and produces /// a canonical set of properties that can be sent to DAP clients. struct VariableDescription { // The error message if SBValue.GetValue() fails. std::optional error; // The display description to show on the IDE. std::string display_value; // The display name to show on the IDE. std::string name; // The variable path for this variable. std::string evaluate_name; // The output of SBValue.GetValue() if it doesn't fail. It might be empty. std::string value; // The summary string of this variable. It might be empty. std::string summary; // The auto summary if using `enableAutoVariableSummaries`. std::optional auto_summary; // The type of this variable. lldb::SBType type_obj; // The display type name of this variable. std::string display_type_name; /// The SBValue for this variable. lldb::SBValue v; VariableDescription(lldb::SBValue v, bool auto_variable_summaries, bool format_hex = false, bool is_name_duplicated = false, std::optional custom_name = {}); /// Returns a description of the value appropriate for the specified context. std::string GetResult(llvm::StringRef context); }; /// Does the given variable have an associated value location? bool ValuePointsToCode(lldb::SBValue v); /// Pack a location into a single integer which we can send via /// the debug adapter protocol. int64_t PackLocation(int64_t var_ref, bool is_value_location); /// Reverse of `PackLocation` std::pair UnpackLocation(int64_t location_id); llvm::json::Value CreateCompileUnit(lldb::SBCompileUnit &unit); /// Create a runInTerminal reverse request object /// /// \param[in] program /// Path to the program to run in the terminal. /// /// \param[in] args /// The arguments for the program. /// /// \param[in] env /// The environment variables to set in the terminal. /// /// \param[in] cwd /// The working directory for the run in terminal request. /// /// \param[in] comm_file /// The fifo file used to communicate the with the target launcher. /// /// \param[in] debugger_pid /// The PID of the lldb-dap instance that will attach to the target. The /// launcher uses it on Linux tell the kernel that it should allow the /// debugger process to attach. /// /// \param[in] external /// If set to true, the program will run in an external terminal window /// instead of IDE's integrated terminal. /// /// \return /// A "runInTerminal" JSON object that follows the specification outlined by /// Microsoft. llvm::json::Object CreateRunInTerminalReverseRequest( llvm::StringRef program, const std::vector &args, const llvm::StringMap &env, llvm::StringRef cwd, llvm::StringRef comm_file, lldb::pid_t debugger_pid, bool external); /// Create a "Terminated" JSON object that contains statistics /// /// \return /// A body JSON object with debug info and breakpoint info llvm::json::Object CreateTerminatedEventObject(lldb::SBTarget &target); /// Convert a given JSON object to a string. std::string JSONToString(const llvm::json::Value &json); } // namespace lldb_dap #endif