| //===-- 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 <cstdint> |
| #include <optional> |
| #include <string> |
| #include <unordered_map> |
| #include <utility> |
| #include <vector> |
| |
| 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<llvm::StringRef> GetString(const llvm::json::Object &obj, |
| llvm::StringRef key); |
| std::optional<llvm::StringRef> 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 <typename T> |
| std::optional<T> GetInteger(const llvm::json::Object &obj, |
| llvm::StringRef key) { |
| return obj.getInteger(key); |
| } |
| |
| template <typename T> |
| std::optional<T> GetInteger(const llvm::json::Object *obj, |
| llvm::StringRef key) { |
| if (obj != nullptr) |
| return GetInteger<T>(*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<bool> GetBoolean(const llvm::json::Object &obj, |
| llvm::StringRef key); |
| std::optional<bool> 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<lldb::addr_t> |
| 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<std::string> 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<std::string, std::string> |
| 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<std::string> 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<std::string> 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<std::string> 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<int64_t, bool> 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<std::string> &args, |
| const llvm::StringMap<std::string> &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 |