| //===-- SourceBreakpoint.cpp ------------------------------------*- 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 |
| // |
| //===----------------------------------------------------------------------===// |
| |
| #include "SourceBreakpoint.h" |
| #include "BreakpointBase.h" |
| #include "DAP.h" |
| #include "JSONUtils.h" |
| #include "lldb/API/SBBreakpoint.h" |
| #include "lldb/API/SBFileSpecList.h" |
| #include "lldb/API/SBFrame.h" |
| #include "lldb/API/SBTarget.h" |
| #include "lldb/API/SBThread.h" |
| #include "lldb/API/SBValue.h" |
| #include "lldb/lldb-enumerations.h" |
| #include <cassert> |
| #include <cctype> |
| #include <cstdlib> |
| #include <utility> |
| |
| namespace lldb_dap { |
| |
| SourceBreakpoint::SourceBreakpoint(DAP &dap, const llvm::json::Object &obj) |
| : Breakpoint(dap, obj), |
| logMessage(std::string(GetString(obj, "logMessage"))), |
| line(GetUnsigned(obj, "line", 0)), column(GetUnsigned(obj, "column", 0)) { |
| } |
| |
| void SourceBreakpoint::SetBreakpoint(const llvm::StringRef source_path) { |
| lldb::SBFileSpecList module_list; |
| bp = dap.target.BreakpointCreateByLocation(source_path.str().c_str(), line, |
| column, 0, module_list); |
| if (!logMessage.empty()) |
| SetLogMessage(); |
| Breakpoint::SetBreakpoint(); |
| } |
| |
| void SourceBreakpoint::UpdateBreakpoint(const SourceBreakpoint &request_bp) { |
| if (logMessage != request_bp.logMessage) { |
| logMessage = request_bp.logMessage; |
| SetLogMessage(); |
| } |
| BreakpointBase::UpdateBreakpoint(request_bp); |
| } |
| |
| lldb::SBError SourceBreakpoint::AppendLogMessagePart(llvm::StringRef part, |
| bool is_expr) { |
| if (is_expr) { |
| logMessageParts.emplace_back(part, is_expr); |
| } else { |
| std::string formatted; |
| lldb::SBError error = FormatLogText(part, formatted); |
| if (error.Fail()) |
| return error; |
| logMessageParts.emplace_back(formatted, is_expr); |
| } |
| return lldb::SBError(); |
| } |
| |
| // TODO: consolidate this code with the implementation in |
| // FormatEntity::ParseInternal(). |
| lldb::SBError SourceBreakpoint::FormatLogText(llvm::StringRef text, |
| std::string &formatted) { |
| lldb::SBError error; |
| while (!text.empty()) { |
| size_t backslash_pos = text.find_first_of('\\'); |
| if (backslash_pos == std::string::npos) { |
| formatted += text.str(); |
| return error; |
| } |
| |
| formatted += text.substr(0, backslash_pos).str(); |
| // Skip the characters before and including '\'. |
| text = text.drop_front(backslash_pos + 1); |
| |
| if (text.empty()) { |
| error.SetErrorString( |
| "'\\' character was not followed by another character"); |
| return error; |
| } |
| |
| const char desens_char = text[0]; |
| text = text.drop_front(); // Skip the desensitized char character |
| switch (desens_char) { |
| case 'a': |
| formatted.push_back('\a'); |
| break; |
| case 'b': |
| formatted.push_back('\b'); |
| break; |
| case 'f': |
| formatted.push_back('\f'); |
| break; |
| case 'n': |
| formatted.push_back('\n'); |
| break; |
| case 'r': |
| formatted.push_back('\r'); |
| break; |
| case 't': |
| formatted.push_back('\t'); |
| break; |
| case 'v': |
| formatted.push_back('\v'); |
| break; |
| case '\'': |
| formatted.push_back('\''); |
| break; |
| case '\\': |
| formatted.push_back('\\'); |
| break; |
| case '0': |
| // 1 to 3 octal chars |
| { |
| if (text.empty()) { |
| error.SetErrorString("missing octal number following '\\0'"); |
| return error; |
| } |
| |
| // Make a string that can hold onto the initial zero char, up to 3 |
| // octal digits, and a terminating NULL. |
| char oct_str[5] = {0, 0, 0, 0, 0}; |
| |
| size_t i; |
| for (i = 0; |
| i < text.size() && i < 4 && (text[i] >= '0' && text[i] <= '7'); |
| ++i) { |
| oct_str[i] = text[i]; |
| } |
| |
| text = text.drop_front(i); |
| unsigned long octal_value = ::strtoul(oct_str, nullptr, 8); |
| if (octal_value <= UINT8_MAX) { |
| formatted.push_back((char)octal_value); |
| } else { |
| error.SetErrorString("octal number is larger than a single byte"); |
| return error; |
| } |
| } |
| break; |
| |
| case 'x': { |
| if (text.empty()) { |
| error.SetErrorString("missing hex number following '\\x'"); |
| return error; |
| } |
| // hex number in the text |
| if (std::isxdigit(text[0])) { |
| // Make a string that can hold onto two hex chars plus a |
| // NULL terminator |
| char hex_str[3] = {0, 0, 0}; |
| hex_str[0] = text[0]; |
| |
| text = text.drop_front(); |
| |
| if (!text.empty() && std::isxdigit(text[0])) { |
| hex_str[1] = text[0]; |
| text = text.drop_front(); |
| } |
| |
| unsigned long hex_value = strtoul(hex_str, nullptr, 16); |
| if (hex_value <= UINT8_MAX) { |
| formatted.push_back((char)hex_value); |
| } else { |
| error.SetErrorString("hex number is larger than a single byte"); |
| return error; |
| } |
| } else { |
| formatted.push_back(desens_char); |
| } |
| break; |
| } |
| |
| default: |
| // Just desensitize any other character by just printing what came |
| // after the '\' |
| formatted.push_back(desens_char); |
| break; |
| } |
| } |
| return error; |
| } |
| |
| // logMessage will be divided into array of LogMessagePart as two kinds: |
| // 1. raw print text message, and |
| // 2. interpolated expression for evaluation which is inside matching curly |
| // braces. |
| // |
| // The function tries to parse logMessage into a list of LogMessageParts |
| // for easy later access in BreakpointHitCallback. |
| void SourceBreakpoint::SetLogMessage() { |
| logMessageParts.clear(); |
| |
| // Contains unmatched open curly braces indices. |
| std::vector<int> unmatched_curly_braces; |
| |
| // Contains all matched curly braces in logMessage. |
| // Loop invariant: matched_curly_braces_ranges are sorted by start index in |
| // ascending order without any overlap between them. |
| std::vector<std::pair<int, int>> matched_curly_braces_ranges; |
| |
| lldb::SBError error; |
| // Part1 - parse matched_curly_braces_ranges. |
| // locating all curly braced expression ranges in logMessage. |
| // The algorithm takes care of nested and imbalanced curly braces. |
| for (size_t i = 0; i < logMessage.size(); ++i) { |
| if (logMessage[i] == '{') { |
| unmatched_curly_braces.push_back(i); |
| } else if (logMessage[i] == '}') { |
| if (unmatched_curly_braces.empty()) |
| // Nothing to match. |
| continue; |
| |
| int last_unmatched_index = unmatched_curly_braces.back(); |
| unmatched_curly_braces.pop_back(); |
| |
| // Erase any matched ranges included in the new match. |
| while (!matched_curly_braces_ranges.empty()) { |
| assert(matched_curly_braces_ranges.back().first != |
| last_unmatched_index && |
| "How can a curley brace be matched twice?"); |
| if (matched_curly_braces_ranges.back().first < last_unmatched_index) |
| break; |
| |
| // This is a nested range let's earse it. |
| assert((size_t)matched_curly_braces_ranges.back().second < i); |
| matched_curly_braces_ranges.pop_back(); |
| } |
| |
| // Assert invariant. |
| assert(matched_curly_braces_ranges.empty() || |
| matched_curly_braces_ranges.back().first < last_unmatched_index); |
| matched_curly_braces_ranges.emplace_back(last_unmatched_index, i); |
| } |
| } |
| |
| // Part2 - parse raw text and expresions parts. |
| // All expression ranges have been parsed in matched_curly_braces_ranges. |
| // The code below uses matched_curly_braces_ranges to divide logMessage |
| // into raw text parts and expression parts. |
| int last_raw_text_start = 0; |
| for (const std::pair<int, int> &curly_braces_range : |
| matched_curly_braces_ranges) { |
| // Raw text before open curly brace. |
| assert(curly_braces_range.first >= last_raw_text_start); |
| size_t raw_text_len = curly_braces_range.first - last_raw_text_start; |
| if (raw_text_len > 0) { |
| error = AppendLogMessagePart( |
| llvm::StringRef(logMessage.c_str() + last_raw_text_start, |
| raw_text_len), |
| /*is_expr=*/false); |
| if (error.Fail()) { |
| NotifyLogMessageError(error.GetCString()); |
| return; |
| } |
| } |
| |
| // Expression between curly braces. |
| assert(curly_braces_range.second > curly_braces_range.first); |
| size_t expr_len = curly_braces_range.second - curly_braces_range.first - 1; |
| error = AppendLogMessagePart( |
| llvm::StringRef(logMessage.c_str() + curly_braces_range.first + 1, |
| expr_len), |
| /*is_expr=*/true); |
| if (error.Fail()) { |
| NotifyLogMessageError(error.GetCString()); |
| return; |
| } |
| |
| last_raw_text_start = curly_braces_range.second + 1; |
| } |
| // Trailing raw text after close curly brace. |
| assert(last_raw_text_start >= 0); |
| if (logMessage.size() > (size_t)last_raw_text_start) { |
| error = AppendLogMessagePart( |
| llvm::StringRef(logMessage.c_str() + last_raw_text_start, |
| logMessage.size() - last_raw_text_start), |
| /*is_expr=*/false); |
| if (error.Fail()) { |
| NotifyLogMessageError(error.GetCString()); |
| return; |
| } |
| } |
| |
| bp.SetCallback(BreakpointHitCallback, this); |
| } |
| |
| void SourceBreakpoint::NotifyLogMessageError(llvm::StringRef error) { |
| std::string message = "Log message has error: "; |
| message += error; |
| dap.SendOutput(OutputType::Console, message); |
| } |
| |
| /*static*/ |
| bool SourceBreakpoint::BreakpointHitCallback( |
| void *baton, lldb::SBProcess &process, lldb::SBThread &thread, |
| lldb::SBBreakpointLocation &location) { |
| if (!baton) |
| return true; |
| |
| SourceBreakpoint *bp = (SourceBreakpoint *)baton; |
| lldb::SBFrame frame = thread.GetSelectedFrame(); |
| |
| std::string output; |
| for (const SourceBreakpoint::LogMessagePart &messagePart : |
| bp->logMessageParts) { |
| if (messagePart.is_expr) { |
| // Try local frame variables first before fall back to expression |
| // evaluation |
| const std::string &expr_str = messagePart.text; |
| const char *expr = expr_str.c_str(); |
| lldb::SBValue value = |
| frame.GetValueForVariablePath(expr, lldb::eDynamicDontRunTarget); |
| if (value.GetError().Fail()) |
| value = frame.EvaluateExpression(expr); |
| output += |
| VariableDescription(value, bp->dap.enable_auto_variable_summaries) |
| .display_value; |
| } else { |
| output += messagePart.text; |
| } |
| } |
| if (!output.empty() && output.back() != '\n') |
| output.push_back('\n'); // Ensure log message has line break. |
| bp->dap.SendOutput(OutputType::Console, output.c_str()); |
| |
| // Do not stop. |
| return false; |
| } |
| |
| } // namespace lldb_dap |