|
86 | 86 | #include "lldb/Host/Host.h" |
87 | 87 | #include "lldb/Utility/StringExtractorGDBRemote.h" |
88 | 88 |
|
| 89 | +#include "llvm/ADT/STLExtras.h" |
89 | 90 | #include "llvm/ADT/ScopeExit.h" |
90 | 91 | #include "llvm/ADT/StringMap.h" |
91 | 92 | #include "llvm/ADT/StringSwitch.h" |
@@ -2735,6 +2736,108 @@ size_t ProcessGDBRemote::DoReadMemory(addr_t addr, void *buf, size_t size, |
2735 | 2736 | return 0; |
2736 | 2737 | } |
2737 | 2738 |
|
| 2739 | +llvm::SmallVector<llvm::MutableArrayRef<uint8_t>> |
| 2740 | +ProcessGDBRemote::ReadMemoryRanges( |
| 2741 | + llvm::ArrayRef<Range<lldb::addr_t, size_t>> ranges, |
| 2742 | + llvm::MutableArrayRef<uint8_t> buffer) { |
| 2743 | + if (!m_gdb_comm.GetMultiMemReadSupported()) |
| 2744 | + return Process::ReadMemoryRanges(ranges, buffer); |
| 2745 | + |
| 2746 | + llvm::Expected<StringExtractorGDBRemote> response = |
| 2747 | + SendMultiMemReadPacket(ranges); |
| 2748 | + if (!response) { |
| 2749 | + LLDB_LOG_ERROR(GetLog(GDBRLog::Process), response.takeError(), |
| 2750 | + "MultiMemRead error response: {0}"); |
| 2751 | + return Process::ReadMemoryRanges(ranges, buffer); |
| 2752 | + } |
| 2753 | + |
| 2754 | + llvm::StringRef response_str = response->GetStringRef(); |
| 2755 | + const unsigned expected_num_ranges = ranges.size(); |
| 2756 | + llvm::Expected<llvm::SmallVector<llvm::MutableArrayRef<uint8_t>>> |
| 2757 | + parsed_response = |
| 2758 | + ParseMultiMemReadPacket(response_str, buffer, expected_num_ranges); |
| 2759 | + if (!parsed_response) { |
| 2760 | + LLDB_LOG_ERROR(GetLog(GDBRLog::Process), parsed_response.takeError(), |
| 2761 | + "MultiMemRead error parsing response: {0}"); |
| 2762 | + return Process::ReadMemoryRanges(ranges, buffer); |
| 2763 | + } |
| 2764 | + return std::move(*parsed_response); |
| 2765 | +} |
| 2766 | + |
| 2767 | +llvm::Expected<StringExtractorGDBRemote> |
| 2768 | +ProcessGDBRemote::SendMultiMemReadPacket( |
| 2769 | + llvm::ArrayRef<Range<lldb::addr_t, size_t>> ranges) { |
| 2770 | + std::string packet_str; |
| 2771 | + llvm::raw_string_ostream stream(packet_str); |
| 2772 | + stream << "MultiMemRead:ranges:"; |
| 2773 | + |
| 2774 | + auto range_to_stream = [&](auto range) { |
| 2775 | + // the "-" marker omits the '0x' prefix. |
| 2776 | + stream << llvm::formatv("{0:x-},{1:x-}", range.base, range.size); |
| 2777 | + }; |
| 2778 | + llvm::interleave(ranges, stream, range_to_stream, ","); |
| 2779 | + stream << ";"; |
| 2780 | + |
| 2781 | + StringExtractorGDBRemote response; |
| 2782 | + GDBRemoteCommunication::PacketResult packet_result = |
| 2783 | + m_gdb_comm.SendPacketAndWaitForResponse(packet_str.data(), response, |
| 2784 | + GetInterruptTimeout()); |
| 2785 | + if (packet_result != GDBRemoteCommunication::PacketResult::Success) |
| 2786 | + return llvm::createStringError( |
| 2787 | + llvm::formatv("MultiMemRead failed to send packet: '{0}'", packet_str)); |
| 2788 | + |
| 2789 | + if (response.IsErrorResponse()) |
| 2790 | + return llvm::createStringError( |
| 2791 | + llvm::formatv("MultiMemRead failed: '{0}'", response.GetStringRef())); |
| 2792 | + |
| 2793 | + if (!response.IsNormalResponse()) |
| 2794 | + return llvm::createStringError(llvm::formatv( |
| 2795 | + "MultiMemRead unexpected response: '{0}'", response.GetStringRef())); |
| 2796 | + |
| 2797 | + return response; |
| 2798 | +} |
| 2799 | + |
| 2800 | +llvm::Expected<llvm::SmallVector<llvm::MutableArrayRef<uint8_t>>> |
| 2801 | +ProcessGDBRemote::ParseMultiMemReadPacket(llvm::StringRef response_str, |
| 2802 | + llvm::MutableArrayRef<uint8_t> buffer, |
| 2803 | + unsigned expected_num_ranges) { |
| 2804 | + // The sizes and the data are separated by a `;`. |
| 2805 | + auto [sizes_str, memory_data] = response_str.split(';'); |
| 2806 | + if (sizes_str.size() == response_str.size()) |
| 2807 | + return llvm::createStringError(llvm::formatv( |
| 2808 | + "MultiMemRead response missing field separator ';' in: '{0}'", |
| 2809 | + response_str)); |
| 2810 | + |
| 2811 | + llvm::SmallVector<llvm::MutableArrayRef<uint8_t>> read_results; |
| 2812 | + |
| 2813 | + // Sizes are separated by a `,`. |
| 2814 | + for (llvm::StringRef size_str : llvm::split(sizes_str, ',')) { |
| 2815 | + uint64_t read_size; |
| 2816 | + if (size_str.getAsInteger(16, read_size)) |
| 2817 | + return llvm::createStringError(llvm::formatv( |
| 2818 | + "MultiMemRead response has invalid size string: {0}", size_str)); |
| 2819 | + |
| 2820 | + if (memory_data.size() < read_size) |
| 2821 | + return llvm::createStringError( |
| 2822 | + llvm::formatv("MultiMemRead response did not have enough data, " |
| 2823 | + "requested sizes: {0}", |
| 2824 | + sizes_str)); |
| 2825 | + |
| 2826 | + llvm::StringRef region_to_read = memory_data.take_front(read_size); |
| 2827 | + memory_data = memory_data.drop_front(read_size); |
| 2828 | + |
| 2829 | + assert(buffer.size() >= read_size); |
| 2830 | + llvm::MutableArrayRef<uint8_t> region_to_write = |
| 2831 | + buffer.take_front(read_size); |
| 2832 | + buffer = buffer.drop_front(read_size); |
| 2833 | + |
| 2834 | + memcpy(region_to_write.data(), region_to_read.data(), read_size); |
| 2835 | + read_results.push_back(region_to_write); |
| 2836 | + } |
| 2837 | + |
| 2838 | + return read_results; |
| 2839 | +} |
| 2840 | + |
2738 | 2841 | bool ProcessGDBRemote::SupportsMemoryTagging() { |
2739 | 2842 | return m_gdb_comm.GetMemoryTaggingSupported(); |
2740 | 2843 | } |
|
0 commit comments