180814287SRaphael Isemann //===-- DataExtractor.cpp -------------------------------------------------===// 2666cc0b2SZachary Turner // 32946cd70SChandler Carruth // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 42946cd70SChandler Carruth // See https://llvm.org/LICENSE.txt for license information. 52946cd70SChandler Carruth // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6666cc0b2SZachary Turner // 7666cc0b2SZachary Turner //===----------------------------------------------------------------------===// 8666cc0b2SZachary Turner 94479ac15SZachary Turner #include "lldb/Utility/DataExtractor.h" 10666cc0b2SZachary Turner 11672d2c12SJonas Devlieghere #include "lldb/lldb-defines.h" 12672d2c12SJonas Devlieghere #include "lldb/lldb-enumerations.h" 13672d2c12SJonas Devlieghere #include "lldb/lldb-forward.h" 14672d2c12SJonas Devlieghere #include "lldb/lldb-types.h" 154479ac15SZachary Turner 16666cc0b2SZachary Turner #include "lldb/Utility/DataBuffer.h" 17666cc0b2SZachary Turner #include "lldb/Utility/DataBufferHeap.h" 18666cc0b2SZachary Turner #include "lldb/Utility/Endian.h" 19dbd7c338SPetr Pavlu #include "lldb/Utility/LLDBAssert.h" 20666cc0b2SZachary Turner #include "lldb/Utility/Log.h" 21666cc0b2SZachary Turner #include "lldb/Utility/Stream.h" 22666cc0b2SZachary Turner #include "lldb/Utility/StreamString.h" 23666cc0b2SZachary Turner #include "lldb/Utility/UUID.h" 24666cc0b2SZachary Turner 25666cc0b2SZachary Turner #include "llvm/ADT/ArrayRef.h" 26666cc0b2SZachary Turner #include "llvm/ADT/SmallVector.h" 27*bb9d93f4SRaphael Isemann #include "llvm/Support/LEB128.h" 28666cc0b2SZachary Turner #include "llvm/Support/MD5.h" 29666cc0b2SZachary Turner #include "llvm/Support/MathExtras.h" 30666cc0b2SZachary Turner 31672d2c12SJonas Devlieghere #include <algorithm> 32672d2c12SJonas Devlieghere #include <array> 334479ac15SZachary Turner #include <cassert> 34672d2c12SJonas Devlieghere #include <cstdint> 354479ac15SZachary Turner #include <string> 364479ac15SZachary Turner 37672d2c12SJonas Devlieghere #include <ctype.h> 38672d2c12SJonas Devlieghere #include <inttypes.h> 39672d2c12SJonas Devlieghere #include <string.h> 404479ac15SZachary Turner 41666cc0b2SZachary Turner using namespace lldb; 42666cc0b2SZachary Turner using namespace lldb_private; 43666cc0b2SZachary Turner 44666cc0b2SZachary Turner static inline uint16_t ReadInt16(const unsigned char *ptr, offset_t offset) { 45666cc0b2SZachary Turner uint16_t value; 46666cc0b2SZachary Turner memcpy(&value, ptr + offset, 2); 47666cc0b2SZachary Turner return value; 48666cc0b2SZachary Turner } 49666cc0b2SZachary Turner 50666cc0b2SZachary Turner static inline uint32_t ReadInt32(const unsigned char *ptr, 51666cc0b2SZachary Turner offset_t offset = 0) { 52666cc0b2SZachary Turner uint32_t value; 53666cc0b2SZachary Turner memcpy(&value, ptr + offset, 4); 54666cc0b2SZachary Turner return value; 55666cc0b2SZachary Turner } 56666cc0b2SZachary Turner 57666cc0b2SZachary Turner static inline uint64_t ReadInt64(const unsigned char *ptr, 58666cc0b2SZachary Turner offset_t offset = 0) { 59666cc0b2SZachary Turner uint64_t value; 60666cc0b2SZachary Turner memcpy(&value, ptr + offset, 8); 61666cc0b2SZachary Turner return value; 62666cc0b2SZachary Turner } 63666cc0b2SZachary Turner 64666cc0b2SZachary Turner static inline uint16_t ReadInt16(const void *ptr) { 65666cc0b2SZachary Turner uint16_t value; 66666cc0b2SZachary Turner memcpy(&value, ptr, 2); 67666cc0b2SZachary Turner return value; 68666cc0b2SZachary Turner } 69666cc0b2SZachary Turner 70666cc0b2SZachary Turner static inline uint16_t ReadSwapInt16(const unsigned char *ptr, 71666cc0b2SZachary Turner offset_t offset) { 72666cc0b2SZachary Turner uint16_t value; 73666cc0b2SZachary Turner memcpy(&value, ptr + offset, 2); 74666cc0b2SZachary Turner return llvm::ByteSwap_16(value); 75666cc0b2SZachary Turner } 76666cc0b2SZachary Turner 77666cc0b2SZachary Turner static inline uint32_t ReadSwapInt32(const unsigned char *ptr, 78666cc0b2SZachary Turner offset_t offset) { 79666cc0b2SZachary Turner uint32_t value; 80666cc0b2SZachary Turner memcpy(&value, ptr + offset, 4); 81666cc0b2SZachary Turner return llvm::ByteSwap_32(value); 82666cc0b2SZachary Turner } 83666cc0b2SZachary Turner 84666cc0b2SZachary Turner static inline uint64_t ReadSwapInt64(const unsigned char *ptr, 85666cc0b2SZachary Turner offset_t offset) { 86666cc0b2SZachary Turner uint64_t value; 87666cc0b2SZachary Turner memcpy(&value, ptr + offset, 8); 88666cc0b2SZachary Turner return llvm::ByteSwap_64(value); 89666cc0b2SZachary Turner } 90666cc0b2SZachary Turner 91666cc0b2SZachary Turner static inline uint16_t ReadSwapInt16(const void *ptr) { 92666cc0b2SZachary Turner uint16_t value; 93666cc0b2SZachary Turner memcpy(&value, ptr, 2); 94666cc0b2SZachary Turner return llvm::ByteSwap_16(value); 95666cc0b2SZachary Turner } 96666cc0b2SZachary Turner 97666cc0b2SZachary Turner static inline uint32_t ReadSwapInt32(const void *ptr) { 98666cc0b2SZachary Turner uint32_t value; 99666cc0b2SZachary Turner memcpy(&value, ptr, 4); 100666cc0b2SZachary Turner return llvm::ByteSwap_32(value); 101666cc0b2SZachary Turner } 102666cc0b2SZachary Turner 103666cc0b2SZachary Turner static inline uint64_t ReadSwapInt64(const void *ptr) { 104666cc0b2SZachary Turner uint64_t value; 105666cc0b2SZachary Turner memcpy(&value, ptr, 8); 106666cc0b2SZachary Turner return llvm::ByteSwap_64(value); 107666cc0b2SZachary Turner } 108666cc0b2SZachary Turner 109dbd7c338SPetr Pavlu static inline uint64_t ReadMaxInt64(const uint8_t *data, size_t byte_size, 110dbd7c338SPetr Pavlu ByteOrder byte_order) { 111dbd7c338SPetr Pavlu uint64_t res = 0; 112dbd7c338SPetr Pavlu if (byte_order == eByteOrderBig) 113dbd7c338SPetr Pavlu for (size_t i = 0; i < byte_size; ++i) 114dbd7c338SPetr Pavlu res = (res << 8) | data[i]; 115dbd7c338SPetr Pavlu else { 116dbd7c338SPetr Pavlu assert(byte_order == eByteOrderLittle); 117dbd7c338SPetr Pavlu for (size_t i = 0; i < byte_size; ++i) 118dbd7c338SPetr Pavlu res = (res << 8) | data[byte_size - 1 - i]; 119dbd7c338SPetr Pavlu } 120dbd7c338SPetr Pavlu return res; 121dbd7c338SPetr Pavlu } 122dbd7c338SPetr Pavlu 123666cc0b2SZachary Turner DataExtractor::DataExtractor() 124666cc0b2SZachary Turner : m_start(nullptr), m_end(nullptr), 125666cc0b2SZachary Turner m_byte_order(endian::InlHostByteOrder()), m_addr_size(sizeof(void *)), 126666cc0b2SZachary Turner m_data_sp(), m_target_byte_size(1) {} 127666cc0b2SZachary Turner 12805097246SAdrian Prantl // This constructor allows us to use data that is owned by someone else. The 12905097246SAdrian Prantl // data must stay around as long as this object is valid. 130666cc0b2SZachary Turner DataExtractor::DataExtractor(const void *data, offset_t length, 131666cc0b2SZachary Turner ByteOrder endian, uint32_t addr_size, 132666cc0b2SZachary Turner uint32_t target_byte_size /*=1*/) 13365fdb342SRaphael Isemann : m_start(const_cast<uint8_t *>(static_cast<const uint8_t *>(data))), 13465fdb342SRaphael Isemann m_end(const_cast<uint8_t *>(static_cast<const uint8_t *>(data)) + length), 135666cc0b2SZachary Turner m_byte_order(endian), m_addr_size(addr_size), m_data_sp(), 136666cc0b2SZachary Turner m_target_byte_size(target_byte_size) { 137ec1efe71SAyke van Laethem assert(addr_size >= 1 && addr_size <= 8); 138666cc0b2SZachary Turner } 139666cc0b2SZachary Turner 14005097246SAdrian Prantl // Make a shared pointer reference to the shared data in "data_sp" and set the 14105097246SAdrian Prantl // endian swapping setting to "swap", and the address size to "addr_size". The 14205097246SAdrian Prantl // shared data reference will ensure the data lives as long as any 14305097246SAdrian Prantl // DataExtractor objects exist that have a reference to this data. 144666cc0b2SZachary Turner DataExtractor::DataExtractor(const DataBufferSP &data_sp, ByteOrder endian, 145666cc0b2SZachary Turner uint32_t addr_size, 146666cc0b2SZachary Turner uint32_t target_byte_size /*=1*/) 147666cc0b2SZachary Turner : m_start(nullptr), m_end(nullptr), m_byte_order(endian), 148666cc0b2SZachary Turner m_addr_size(addr_size), m_data_sp(), 149666cc0b2SZachary Turner m_target_byte_size(target_byte_size) { 150ec1efe71SAyke van Laethem assert(addr_size >= 1 && addr_size <= 8); 151666cc0b2SZachary Turner SetData(data_sp); 152666cc0b2SZachary Turner } 153666cc0b2SZachary Turner 15405097246SAdrian Prantl // Initialize this object with a subset of the data bytes in "data". If "data" 15505097246SAdrian Prantl // contains shared data, then a reference to this shared data will added and 15605097246SAdrian Prantl // the shared data will stay around as long as any object contains a reference 15705097246SAdrian Prantl // to that data. The endian swap and address size settings are copied from 15805097246SAdrian Prantl // "data". 159666cc0b2SZachary Turner DataExtractor::DataExtractor(const DataExtractor &data, offset_t offset, 160666cc0b2SZachary Turner offset_t length, uint32_t target_byte_size /*=1*/) 161666cc0b2SZachary Turner : m_start(nullptr), m_end(nullptr), m_byte_order(data.m_byte_order), 162666cc0b2SZachary Turner m_addr_size(data.m_addr_size), m_data_sp(), 163666cc0b2SZachary Turner m_target_byte_size(target_byte_size) { 164ec1efe71SAyke van Laethem assert(m_addr_size >= 1 && m_addr_size <= 8); 165666cc0b2SZachary Turner if (data.ValidOffset(offset)) { 166666cc0b2SZachary Turner offset_t bytes_available = data.GetByteSize() - offset; 167666cc0b2SZachary Turner if (length > bytes_available) 168666cc0b2SZachary Turner length = bytes_available; 169666cc0b2SZachary Turner SetData(data, offset, length); 170666cc0b2SZachary Turner } 171666cc0b2SZachary Turner } 172666cc0b2SZachary Turner 173666cc0b2SZachary Turner DataExtractor::DataExtractor(const DataExtractor &rhs) 174666cc0b2SZachary Turner : m_start(rhs.m_start), m_end(rhs.m_end), m_byte_order(rhs.m_byte_order), 175666cc0b2SZachary Turner m_addr_size(rhs.m_addr_size), m_data_sp(rhs.m_data_sp), 176666cc0b2SZachary Turner m_target_byte_size(rhs.m_target_byte_size) { 177ec1efe71SAyke van Laethem assert(m_addr_size >= 1 && m_addr_size <= 8); 178666cc0b2SZachary Turner } 179666cc0b2SZachary Turner 180666cc0b2SZachary Turner // Assignment operator 181666cc0b2SZachary Turner const DataExtractor &DataExtractor::operator=(const DataExtractor &rhs) { 182666cc0b2SZachary Turner if (this != &rhs) { 183666cc0b2SZachary Turner m_start = rhs.m_start; 184666cc0b2SZachary Turner m_end = rhs.m_end; 185666cc0b2SZachary Turner m_byte_order = rhs.m_byte_order; 186666cc0b2SZachary Turner m_addr_size = rhs.m_addr_size; 187666cc0b2SZachary Turner m_data_sp = rhs.m_data_sp; 188666cc0b2SZachary Turner } 189666cc0b2SZachary Turner return *this; 190666cc0b2SZachary Turner } 191666cc0b2SZachary Turner 192666cc0b2SZachary Turner DataExtractor::~DataExtractor() = default; 193666cc0b2SZachary Turner 19405097246SAdrian Prantl // Clears the object contents back to a default invalid state, and release any 19505097246SAdrian Prantl // references to shared data that this object may contain. 196666cc0b2SZachary Turner void DataExtractor::Clear() { 197666cc0b2SZachary Turner m_start = nullptr; 198666cc0b2SZachary Turner m_end = nullptr; 199666cc0b2SZachary Turner m_byte_order = endian::InlHostByteOrder(); 200666cc0b2SZachary Turner m_addr_size = sizeof(void *); 201666cc0b2SZachary Turner m_data_sp.reset(); 202666cc0b2SZachary Turner } 203666cc0b2SZachary Turner 20405097246SAdrian Prantl // If this object contains shared data, this function returns the offset into 20505097246SAdrian Prantl // that shared data. Else zero is returned. 206666cc0b2SZachary Turner size_t DataExtractor::GetSharedDataOffset() const { 207666cc0b2SZachary Turner if (m_start != nullptr) { 208666cc0b2SZachary Turner const DataBuffer *data = m_data_sp.get(); 209666cc0b2SZachary Turner if (data != nullptr) { 210666cc0b2SZachary Turner const uint8_t *data_bytes = data->GetBytes(); 211666cc0b2SZachary Turner if (data_bytes != nullptr) { 212666cc0b2SZachary Turner assert(m_start >= data_bytes); 213666cc0b2SZachary Turner return m_start - data_bytes; 214666cc0b2SZachary Turner } 215666cc0b2SZachary Turner } 216666cc0b2SZachary Turner } 217666cc0b2SZachary Turner return 0; 218666cc0b2SZachary Turner } 219666cc0b2SZachary Turner 22005097246SAdrian Prantl // Set the data with which this object will extract from to data starting at 22105097246SAdrian Prantl // BYTES and set the length of the data to LENGTH bytes long. The data is 22205097246SAdrian Prantl // externally owned must be around at least as long as this object points to 22305097246SAdrian Prantl // the data. No copy of the data is made, this object just refers to this data 22405097246SAdrian Prantl // and can extract from it. If this object refers to any shared data upon 22505097246SAdrian Prantl // entry, the reference to that data will be released. Is SWAP is set to true, 226666cc0b2SZachary Turner // any data extracted will be endian swapped. 227666cc0b2SZachary Turner lldb::offset_t DataExtractor::SetData(const void *bytes, offset_t length, 228666cc0b2SZachary Turner ByteOrder endian) { 229666cc0b2SZachary Turner m_byte_order = endian; 230666cc0b2SZachary Turner m_data_sp.reset(); 231666cc0b2SZachary Turner if (bytes == nullptr || length == 0) { 232666cc0b2SZachary Turner m_start = nullptr; 233666cc0b2SZachary Turner m_end = nullptr; 234666cc0b2SZachary Turner } else { 23565fdb342SRaphael Isemann m_start = const_cast<uint8_t *>(static_cast<const uint8_t *>(bytes)); 236666cc0b2SZachary Turner m_end = m_start + length; 237666cc0b2SZachary Turner } 238666cc0b2SZachary Turner return GetByteSize(); 239666cc0b2SZachary Turner } 240666cc0b2SZachary Turner 24105097246SAdrian Prantl // Assign the data for this object to be a subrange in "data" starting 24205097246SAdrian Prantl // "data_offset" bytes into "data" and ending "data_length" bytes later. If 24305097246SAdrian Prantl // "data_offset" is not a valid offset into "data", then this object will 24405097246SAdrian Prantl // contain no bytes. If "data_offset" is within "data" yet "data_length" is too 24505097246SAdrian Prantl // large, the length will be capped at the number of bytes remaining in "data". 24605097246SAdrian Prantl // If "data" contains a shared pointer to other data, then a ref counted 24705097246SAdrian Prantl // pointer to that data will be made in this object. If "data" doesn't contain 24805097246SAdrian Prantl // a shared pointer to data, then the bytes referred to in "data" will need to 24905097246SAdrian Prantl // exist at least as long as this object refers to those bytes. The address 25005097246SAdrian Prantl // size and endian swap settings are copied from the current values in "data". 251666cc0b2SZachary Turner lldb::offset_t DataExtractor::SetData(const DataExtractor &data, 252666cc0b2SZachary Turner offset_t data_offset, 253666cc0b2SZachary Turner offset_t data_length) { 254666cc0b2SZachary Turner m_addr_size = data.m_addr_size; 255ec1efe71SAyke van Laethem assert(m_addr_size >= 1 && m_addr_size <= 8); 256666cc0b2SZachary Turner // If "data" contains shared pointer to data, then we can use that 257666cc0b2SZachary Turner if (data.m_data_sp) { 258666cc0b2SZachary Turner m_byte_order = data.m_byte_order; 259666cc0b2SZachary Turner return SetData(data.m_data_sp, data.GetSharedDataOffset() + data_offset, 260666cc0b2SZachary Turner data_length); 261666cc0b2SZachary Turner } 262666cc0b2SZachary Turner 263666cc0b2SZachary Turner // We have a DataExtractor object that just has a pointer to bytes 264666cc0b2SZachary Turner if (data.ValidOffset(data_offset)) { 265666cc0b2SZachary Turner if (data_length > data.GetByteSize() - data_offset) 266666cc0b2SZachary Turner data_length = data.GetByteSize() - data_offset; 267666cc0b2SZachary Turner return SetData(data.GetDataStart() + data_offset, data_length, 268666cc0b2SZachary Turner data.GetByteOrder()); 269666cc0b2SZachary Turner } 270666cc0b2SZachary Turner return 0; 271666cc0b2SZachary Turner } 272666cc0b2SZachary Turner 27305097246SAdrian Prantl // Assign the data for this object to be a subrange of the shared data in 27405097246SAdrian Prantl // "data_sp" starting "data_offset" bytes into "data_sp" and ending 27505097246SAdrian Prantl // "data_length" bytes later. If "data_offset" is not a valid offset into 27605097246SAdrian Prantl // "data_sp", then this object will contain no bytes. If "data_offset" is 27705097246SAdrian Prantl // within "data_sp" yet "data_length" is too large, the length will be capped 27805097246SAdrian Prantl // at the number of bytes remaining in "data_sp". A ref counted pointer to the 27905097246SAdrian Prantl // data in "data_sp" will be made in this object IF the number of bytes this 28005097246SAdrian Prantl // object refers to in greater than zero (if at least one byte was available 28105097246SAdrian Prantl // starting at "data_offset") to ensure the data stays around as long as it is 28205097246SAdrian Prantl // needed. The address size and endian swap settings will remain unchanged from 28305097246SAdrian Prantl // their current settings. 284666cc0b2SZachary Turner lldb::offset_t DataExtractor::SetData(const DataBufferSP &data_sp, 285666cc0b2SZachary Turner offset_t data_offset, 286666cc0b2SZachary Turner offset_t data_length) { 287666cc0b2SZachary Turner m_start = m_end = nullptr; 288666cc0b2SZachary Turner 289666cc0b2SZachary Turner if (data_length > 0) { 290666cc0b2SZachary Turner m_data_sp = data_sp; 291666cc0b2SZachary Turner if (data_sp) { 292666cc0b2SZachary Turner const size_t data_size = data_sp->GetByteSize(); 293666cc0b2SZachary Turner if (data_offset < data_size) { 294666cc0b2SZachary Turner m_start = data_sp->GetBytes() + data_offset; 295666cc0b2SZachary Turner const size_t bytes_left = data_size - data_offset; 296666cc0b2SZachary Turner // Cap the length of we asked for too many 297666cc0b2SZachary Turner if (data_length <= bytes_left) 298666cc0b2SZachary Turner m_end = m_start + data_length; // We got all the bytes we wanted 299666cc0b2SZachary Turner else 300666cc0b2SZachary Turner m_end = m_start + bytes_left; // Not all the bytes requested were 301666cc0b2SZachary Turner // available in the shared data 302666cc0b2SZachary Turner } 303666cc0b2SZachary Turner } 304666cc0b2SZachary Turner } 305666cc0b2SZachary Turner 306666cc0b2SZachary Turner size_t new_size = GetByteSize(); 307666cc0b2SZachary Turner 30805097246SAdrian Prantl // Don't hold a shared pointer to the data buffer if we don't share any valid 30905097246SAdrian Prantl // bytes in the shared buffer. 310666cc0b2SZachary Turner if (new_size == 0) 311666cc0b2SZachary Turner m_data_sp.reset(); 312666cc0b2SZachary Turner 313666cc0b2SZachary Turner return new_size; 314666cc0b2SZachary Turner } 315666cc0b2SZachary Turner 31605097246SAdrian Prantl // Extract a single unsigned char from the binary data and update the offset 31705097246SAdrian Prantl // pointed to by "offset_ptr". 318666cc0b2SZachary Turner // 319666cc0b2SZachary Turner // RETURNS the byte that was extracted, or zero on failure. 320666cc0b2SZachary Turner uint8_t DataExtractor::GetU8(offset_t *offset_ptr) const { 32124374aefSJonas Devlieghere const uint8_t *data = static_cast<const uint8_t *>(GetData(offset_ptr, 1)); 322666cc0b2SZachary Turner if (data) 323666cc0b2SZachary Turner return *data; 324666cc0b2SZachary Turner return 0; 325666cc0b2SZachary Turner } 326666cc0b2SZachary Turner 32705097246SAdrian Prantl // Extract "count" unsigned chars from the binary data and update the offset 32805097246SAdrian Prantl // pointed to by "offset_ptr". The extracted data is copied into "dst". 329666cc0b2SZachary Turner // 330666cc0b2SZachary Turner // RETURNS the non-nullptr buffer pointer upon successful extraction of 33105097246SAdrian Prantl // all the requested bytes, or nullptr when the data is not available in the 33205097246SAdrian Prantl // buffer due to being out of bounds, or insufficient data. 333666cc0b2SZachary Turner void *DataExtractor::GetU8(offset_t *offset_ptr, void *dst, 334666cc0b2SZachary Turner uint32_t count) const { 33524374aefSJonas Devlieghere const uint8_t *data = 33624374aefSJonas Devlieghere static_cast<const uint8_t *>(GetData(offset_ptr, count)); 337666cc0b2SZachary Turner if (data) { 338666cc0b2SZachary Turner // Copy the data into the buffer 339666cc0b2SZachary Turner memcpy(dst, data, count); 340666cc0b2SZachary Turner // Return a non-nullptr pointer to the converted data as an indicator of 341666cc0b2SZachary Turner // success 342666cc0b2SZachary Turner return dst; 343666cc0b2SZachary Turner } 344666cc0b2SZachary Turner return nullptr; 345666cc0b2SZachary Turner } 346666cc0b2SZachary Turner 34705097246SAdrian Prantl // Extract a single uint16_t from the data and update the offset pointed to by 34805097246SAdrian Prantl // "offset_ptr". 349666cc0b2SZachary Turner // 350666cc0b2SZachary Turner // RETURNS the uint16_t that was extracted, or zero on failure. 351666cc0b2SZachary Turner uint16_t DataExtractor::GetU16(offset_t *offset_ptr) const { 352666cc0b2SZachary Turner uint16_t val = 0; 35324374aefSJonas Devlieghere const uint8_t *data = 35424374aefSJonas Devlieghere static_cast<const uint8_t *>(GetData(offset_ptr, sizeof(val))); 355666cc0b2SZachary Turner if (data) { 356666cc0b2SZachary Turner if (m_byte_order != endian::InlHostByteOrder()) 357666cc0b2SZachary Turner val = ReadSwapInt16(data); 358666cc0b2SZachary Turner else 359666cc0b2SZachary Turner val = ReadInt16(data); 360666cc0b2SZachary Turner } 361666cc0b2SZachary Turner return val; 362666cc0b2SZachary Turner } 363666cc0b2SZachary Turner 364666cc0b2SZachary Turner uint16_t DataExtractor::GetU16_unchecked(offset_t *offset_ptr) const { 365666cc0b2SZachary Turner uint16_t val; 366666cc0b2SZachary Turner if (m_byte_order == endian::InlHostByteOrder()) 367666cc0b2SZachary Turner val = ReadInt16(m_start, *offset_ptr); 368666cc0b2SZachary Turner else 369666cc0b2SZachary Turner val = ReadSwapInt16(m_start, *offset_ptr); 370666cc0b2SZachary Turner *offset_ptr += sizeof(val); 371666cc0b2SZachary Turner return val; 372666cc0b2SZachary Turner } 373666cc0b2SZachary Turner 374666cc0b2SZachary Turner uint32_t DataExtractor::GetU32_unchecked(offset_t *offset_ptr) const { 375666cc0b2SZachary Turner uint32_t val; 376666cc0b2SZachary Turner if (m_byte_order == endian::InlHostByteOrder()) 377666cc0b2SZachary Turner val = ReadInt32(m_start, *offset_ptr); 378666cc0b2SZachary Turner else 379666cc0b2SZachary Turner val = ReadSwapInt32(m_start, *offset_ptr); 380666cc0b2SZachary Turner *offset_ptr += sizeof(val); 381666cc0b2SZachary Turner return val; 382666cc0b2SZachary Turner } 383666cc0b2SZachary Turner 384666cc0b2SZachary Turner uint64_t DataExtractor::GetU64_unchecked(offset_t *offset_ptr) const { 385666cc0b2SZachary Turner uint64_t val; 386666cc0b2SZachary Turner if (m_byte_order == endian::InlHostByteOrder()) 387666cc0b2SZachary Turner val = ReadInt64(m_start, *offset_ptr); 388666cc0b2SZachary Turner else 389666cc0b2SZachary Turner val = ReadSwapInt64(m_start, *offset_ptr); 390666cc0b2SZachary Turner *offset_ptr += sizeof(val); 391666cc0b2SZachary Turner return val; 392666cc0b2SZachary Turner } 393666cc0b2SZachary Turner 39405097246SAdrian Prantl // Extract "count" uint16_t values from the binary data and update the offset 39505097246SAdrian Prantl // pointed to by "offset_ptr". The extracted data is copied into "dst". 396666cc0b2SZachary Turner // 397666cc0b2SZachary Turner // RETURNS the non-nullptr buffer pointer upon successful extraction of 39805097246SAdrian Prantl // all the requested bytes, or nullptr when the data is not available in the 39905097246SAdrian Prantl // buffer due to being out of bounds, or insufficient data. 400666cc0b2SZachary Turner void *DataExtractor::GetU16(offset_t *offset_ptr, void *void_dst, 401666cc0b2SZachary Turner uint32_t count) const { 402666cc0b2SZachary Turner const size_t src_size = sizeof(uint16_t) * count; 40324374aefSJonas Devlieghere const uint16_t *src = 40424374aefSJonas Devlieghere static_cast<const uint16_t *>(GetData(offset_ptr, src_size)); 405666cc0b2SZachary Turner if (src) { 406666cc0b2SZachary Turner if (m_byte_order != endian::InlHostByteOrder()) { 40724374aefSJonas Devlieghere uint16_t *dst_pos = static_cast<uint16_t *>(void_dst); 408666cc0b2SZachary Turner uint16_t *dst_end = dst_pos + count; 409666cc0b2SZachary Turner const uint16_t *src_pos = src; 410666cc0b2SZachary Turner while (dst_pos < dst_end) { 411666cc0b2SZachary Turner *dst_pos = ReadSwapInt16(src_pos); 412666cc0b2SZachary Turner ++dst_pos; 413666cc0b2SZachary Turner ++src_pos; 414666cc0b2SZachary Turner } 415666cc0b2SZachary Turner } else { 416666cc0b2SZachary Turner memcpy(void_dst, src, src_size); 417666cc0b2SZachary Turner } 418666cc0b2SZachary Turner // Return a non-nullptr pointer to the converted data as an indicator of 419666cc0b2SZachary Turner // success 420666cc0b2SZachary Turner return void_dst; 421666cc0b2SZachary Turner } 422666cc0b2SZachary Turner return nullptr; 423666cc0b2SZachary Turner } 424666cc0b2SZachary Turner 42505097246SAdrian Prantl // Extract a single uint32_t from the data and update the offset pointed to by 42605097246SAdrian Prantl // "offset_ptr". 427666cc0b2SZachary Turner // 428666cc0b2SZachary Turner // RETURNS the uint32_t that was extracted, or zero on failure. 429666cc0b2SZachary Turner uint32_t DataExtractor::GetU32(offset_t *offset_ptr) const { 430666cc0b2SZachary Turner uint32_t val = 0; 43124374aefSJonas Devlieghere const uint8_t *data = 43224374aefSJonas Devlieghere static_cast<const uint8_t *>(GetData(offset_ptr, sizeof(val))); 433666cc0b2SZachary Turner if (data) { 434666cc0b2SZachary Turner if (m_byte_order != endian::InlHostByteOrder()) { 435666cc0b2SZachary Turner val = ReadSwapInt32(data); 436666cc0b2SZachary Turner } else { 437666cc0b2SZachary Turner memcpy(&val, data, 4); 438666cc0b2SZachary Turner } 439666cc0b2SZachary Turner } 440666cc0b2SZachary Turner return val; 441666cc0b2SZachary Turner } 442666cc0b2SZachary Turner 44305097246SAdrian Prantl // Extract "count" uint32_t values from the binary data and update the offset 44405097246SAdrian Prantl // pointed to by "offset_ptr". The extracted data is copied into "dst". 445666cc0b2SZachary Turner // 446666cc0b2SZachary Turner // RETURNS the non-nullptr buffer pointer upon successful extraction of 44705097246SAdrian Prantl // all the requested bytes, or nullptr when the data is not available in the 44805097246SAdrian Prantl // buffer due to being out of bounds, or insufficient data. 449666cc0b2SZachary Turner void *DataExtractor::GetU32(offset_t *offset_ptr, void *void_dst, 450666cc0b2SZachary Turner uint32_t count) const { 451666cc0b2SZachary Turner const size_t src_size = sizeof(uint32_t) * count; 45224374aefSJonas Devlieghere const uint32_t *src = 45324374aefSJonas Devlieghere static_cast<const uint32_t *>(GetData(offset_ptr, src_size)); 454666cc0b2SZachary Turner if (src) { 455666cc0b2SZachary Turner if (m_byte_order != endian::InlHostByteOrder()) { 45624374aefSJonas Devlieghere uint32_t *dst_pos = static_cast<uint32_t *>(void_dst); 457666cc0b2SZachary Turner uint32_t *dst_end = dst_pos + count; 458666cc0b2SZachary Turner const uint32_t *src_pos = src; 459666cc0b2SZachary Turner while (dst_pos < dst_end) { 460666cc0b2SZachary Turner *dst_pos = ReadSwapInt32(src_pos); 461666cc0b2SZachary Turner ++dst_pos; 462666cc0b2SZachary Turner ++src_pos; 463666cc0b2SZachary Turner } 464666cc0b2SZachary Turner } else { 465666cc0b2SZachary Turner memcpy(void_dst, src, src_size); 466666cc0b2SZachary Turner } 467666cc0b2SZachary Turner // Return a non-nullptr pointer to the converted data as an indicator of 468666cc0b2SZachary Turner // success 469666cc0b2SZachary Turner return void_dst; 470666cc0b2SZachary Turner } 471666cc0b2SZachary Turner return nullptr; 472666cc0b2SZachary Turner } 473666cc0b2SZachary Turner 47405097246SAdrian Prantl // Extract a single uint64_t from the data and update the offset pointed to by 47505097246SAdrian Prantl // "offset_ptr". 476666cc0b2SZachary Turner // 477666cc0b2SZachary Turner // RETURNS the uint64_t that was extracted, or zero on failure. 478666cc0b2SZachary Turner uint64_t DataExtractor::GetU64(offset_t *offset_ptr) const { 479666cc0b2SZachary Turner uint64_t val = 0; 48024374aefSJonas Devlieghere const uint8_t *data = 48124374aefSJonas Devlieghere static_cast<const uint8_t *>(GetData(offset_ptr, sizeof(val))); 482666cc0b2SZachary Turner if (data) { 483666cc0b2SZachary Turner if (m_byte_order != endian::InlHostByteOrder()) { 484666cc0b2SZachary Turner val = ReadSwapInt64(data); 485666cc0b2SZachary Turner } else { 486666cc0b2SZachary Turner memcpy(&val, data, 8); 487666cc0b2SZachary Turner } 488666cc0b2SZachary Turner } 489666cc0b2SZachary Turner return val; 490666cc0b2SZachary Turner } 491666cc0b2SZachary Turner 492666cc0b2SZachary Turner // GetU64 493666cc0b2SZachary Turner // 49405097246SAdrian Prantl // Get multiple consecutive 64 bit values. Return true if the entire read 49505097246SAdrian Prantl // succeeds and increment the offset pointed to by offset_ptr, else return 49605097246SAdrian Prantl // false and leave the offset pointed to by offset_ptr unchanged. 497666cc0b2SZachary Turner void *DataExtractor::GetU64(offset_t *offset_ptr, void *void_dst, 498666cc0b2SZachary Turner uint32_t count) const { 499666cc0b2SZachary Turner const size_t src_size = sizeof(uint64_t) * count; 50024374aefSJonas Devlieghere const uint64_t *src = 50124374aefSJonas Devlieghere static_cast<const uint64_t *>(GetData(offset_ptr, src_size)); 502666cc0b2SZachary Turner if (src) { 503666cc0b2SZachary Turner if (m_byte_order != endian::InlHostByteOrder()) { 50424374aefSJonas Devlieghere uint64_t *dst_pos = static_cast<uint64_t *>(void_dst); 505666cc0b2SZachary Turner uint64_t *dst_end = dst_pos + count; 506666cc0b2SZachary Turner const uint64_t *src_pos = src; 507666cc0b2SZachary Turner while (dst_pos < dst_end) { 508666cc0b2SZachary Turner *dst_pos = ReadSwapInt64(src_pos); 509666cc0b2SZachary Turner ++dst_pos; 510666cc0b2SZachary Turner ++src_pos; 511666cc0b2SZachary Turner } 512666cc0b2SZachary Turner } else { 513666cc0b2SZachary Turner memcpy(void_dst, src, src_size); 514666cc0b2SZachary Turner } 515666cc0b2SZachary Turner // Return a non-nullptr pointer to the converted data as an indicator of 516666cc0b2SZachary Turner // success 517666cc0b2SZachary Turner return void_dst; 518666cc0b2SZachary Turner } 519666cc0b2SZachary Turner return nullptr; 520666cc0b2SZachary Turner } 521666cc0b2SZachary Turner 522666cc0b2SZachary Turner uint32_t DataExtractor::GetMaxU32(offset_t *offset_ptr, 523666cc0b2SZachary Turner size_t byte_size) const { 524dbd7c338SPetr Pavlu lldbassert(byte_size > 0 && byte_size <= 4 && "GetMaxU32 invalid byte_size!"); 525dbd7c338SPetr Pavlu return GetMaxU64(offset_ptr, byte_size); 526dbd7c338SPetr Pavlu } 527dbd7c338SPetr Pavlu 528dbd7c338SPetr Pavlu uint64_t DataExtractor::GetMaxU64(offset_t *offset_ptr, 529dbd7c338SPetr Pavlu size_t byte_size) const { 530dbd7c338SPetr Pavlu lldbassert(byte_size > 0 && byte_size <= 8 && "GetMaxU64 invalid byte_size!"); 531666cc0b2SZachary Turner switch (byte_size) { 532666cc0b2SZachary Turner case 1: 533666cc0b2SZachary Turner return GetU8(offset_ptr); 534666cc0b2SZachary Turner case 2: 535666cc0b2SZachary Turner return GetU16(offset_ptr); 536666cc0b2SZachary Turner case 4: 537666cc0b2SZachary Turner return GetU32(offset_ptr); 538666cc0b2SZachary Turner case 8: 539666cc0b2SZachary Turner return GetU64(offset_ptr); 540dbd7c338SPetr Pavlu default: { 541dbd7c338SPetr Pavlu // General case. 542dbd7c338SPetr Pavlu const uint8_t *data = 543dbd7c338SPetr Pavlu static_cast<const uint8_t *>(GetData(offset_ptr, byte_size)); 544dbd7c338SPetr Pavlu if (data == nullptr) 545dbd7c338SPetr Pavlu return 0; 546dbd7c338SPetr Pavlu return ReadMaxInt64(data, byte_size, m_byte_order); 547dbd7c338SPetr Pavlu } 548666cc0b2SZachary Turner } 549666cc0b2SZachary Turner return 0; 550666cc0b2SZachary Turner } 551666cc0b2SZachary Turner 552666cc0b2SZachary Turner uint64_t DataExtractor::GetMaxU64_unchecked(offset_t *offset_ptr, 553dbd7c338SPetr Pavlu size_t byte_size) const { 554dbd7c338SPetr Pavlu switch (byte_size) { 555666cc0b2SZachary Turner case 1: 556666cc0b2SZachary Turner return GetU8_unchecked(offset_ptr); 557666cc0b2SZachary Turner case 2: 558666cc0b2SZachary Turner return GetU16_unchecked(offset_ptr); 559666cc0b2SZachary Turner case 4: 560666cc0b2SZachary Turner return GetU32_unchecked(offset_ptr); 561666cc0b2SZachary Turner case 8: 562666cc0b2SZachary Turner return GetU64_unchecked(offset_ptr); 563dbd7c338SPetr Pavlu default: { 564dbd7c338SPetr Pavlu uint64_t res = ReadMaxInt64(&m_start[*offset_ptr], byte_size, m_byte_order); 565dbd7c338SPetr Pavlu *offset_ptr += byte_size; 566dbd7c338SPetr Pavlu return res; 567dbd7c338SPetr Pavlu } 568666cc0b2SZachary Turner } 569666cc0b2SZachary Turner return 0; 570666cc0b2SZachary Turner } 571666cc0b2SZachary Turner 572dbd7c338SPetr Pavlu int64_t DataExtractor::GetMaxS64(offset_t *offset_ptr, size_t byte_size) const { 573dbd7c338SPetr Pavlu uint64_t u64 = GetMaxU64(offset_ptr, byte_size); 574dbd7c338SPetr Pavlu return llvm::SignExtend64(u64, 8 * byte_size); 575666cc0b2SZachary Turner } 576666cc0b2SZachary Turner 577666cc0b2SZachary Turner uint64_t DataExtractor::GetMaxU64Bitfield(offset_t *offset_ptr, size_t size, 578666cc0b2SZachary Turner uint32_t bitfield_bit_size, 579666cc0b2SZachary Turner uint32_t bitfield_bit_offset) const { 580fffd7029Sshafik assert(bitfield_bit_size <= 64); 581666cc0b2SZachary Turner uint64_t uval64 = GetMaxU64(offset_ptr, size); 582fffd7029Sshafik 583fffd7029Sshafik if (bitfield_bit_size == 0) 584fffd7029Sshafik return uval64; 585fffd7029Sshafik 586666cc0b2SZachary Turner int32_t lsbcount = bitfield_bit_offset; 587666cc0b2SZachary Turner if (m_byte_order == eByteOrderBig) 588666cc0b2SZachary Turner lsbcount = size * 8 - bitfield_bit_offset - bitfield_bit_size; 589fffd7029Sshafik 590666cc0b2SZachary Turner if (lsbcount > 0) 591666cc0b2SZachary Turner uval64 >>= lsbcount; 592fffd7029Sshafik 593fffd7029Sshafik uint64_t bitfield_mask = 594fffd7029Sshafik (bitfield_bit_size == 64 595fffd7029Sshafik ? std::numeric_limits<uint64_t>::max() 596fffd7029Sshafik : ((static_cast<uint64_t>(1) << bitfield_bit_size) - 1)); 597666cc0b2SZachary Turner if (!bitfield_mask && bitfield_bit_offset == 0 && bitfield_bit_size == 64) 598666cc0b2SZachary Turner return uval64; 599fffd7029Sshafik 600666cc0b2SZachary Turner uval64 &= bitfield_mask; 601fffd7029Sshafik 602666cc0b2SZachary Turner return uval64; 603666cc0b2SZachary Turner } 604666cc0b2SZachary Turner 605666cc0b2SZachary Turner int64_t DataExtractor::GetMaxS64Bitfield(offset_t *offset_ptr, size_t size, 606666cc0b2SZachary Turner uint32_t bitfield_bit_size, 607666cc0b2SZachary Turner uint32_t bitfield_bit_offset) const { 6087b90cdedSVedant Kumar assert(size >= 1 && "GetMaxS64Bitfield size must be >= 1"); 6097b90cdedSVedant Kumar assert(size <= 8 && "GetMaxS64Bitfield size must be <= 8"); 610666cc0b2SZachary Turner int64_t sval64 = GetMaxS64(offset_ptr, size); 611bb6646ceSVedant Kumar if (bitfield_bit_size == 0) 612bb6646ceSVedant Kumar return sval64; 613666cc0b2SZachary Turner int32_t lsbcount = bitfield_bit_offset; 614666cc0b2SZachary Turner if (m_byte_order == eByteOrderBig) 615666cc0b2SZachary Turner lsbcount = size * 8 - bitfield_bit_offset - bitfield_bit_size; 616666cc0b2SZachary Turner if (lsbcount > 0) 617666cc0b2SZachary Turner sval64 >>= lsbcount; 618bb6646ceSVedant Kumar uint64_t bitfield_mask = llvm::maskTrailingOnes<uint64_t>(bitfield_bit_size); 619666cc0b2SZachary Turner sval64 &= bitfield_mask; 620666cc0b2SZachary Turner // sign extend if needed 62124374aefSJonas Devlieghere if (sval64 & ((static_cast<uint64_t>(1)) << (bitfield_bit_size - 1))) 622666cc0b2SZachary Turner sval64 |= ~bitfield_mask; 623666cc0b2SZachary Turner return sval64; 624666cc0b2SZachary Turner } 625666cc0b2SZachary Turner 626666cc0b2SZachary Turner float DataExtractor::GetFloat(offset_t *offset_ptr) const { 627666cc0b2SZachary Turner typedef float float_type; 628666cc0b2SZachary Turner float_type val = 0.0; 629666cc0b2SZachary Turner const size_t src_size = sizeof(float_type); 63024374aefSJonas Devlieghere const float_type *src = 63124374aefSJonas Devlieghere static_cast<const float_type *>(GetData(offset_ptr, src_size)); 632666cc0b2SZachary Turner if (src) { 633666cc0b2SZachary Turner if (m_byte_order != endian::InlHostByteOrder()) { 63424374aefSJonas Devlieghere const uint8_t *src_data = reinterpret_cast<const uint8_t *>(src); 63524374aefSJonas Devlieghere uint8_t *dst_data = reinterpret_cast<uint8_t *>(&val); 636666cc0b2SZachary Turner for (size_t i = 0; i < sizeof(float_type); ++i) 637666cc0b2SZachary Turner dst_data[sizeof(float_type) - 1 - i] = src_data[i]; 638666cc0b2SZachary Turner } else { 639666cc0b2SZachary Turner val = *src; 640666cc0b2SZachary Turner } 641666cc0b2SZachary Turner } 642666cc0b2SZachary Turner return val; 643666cc0b2SZachary Turner } 644666cc0b2SZachary Turner 645666cc0b2SZachary Turner double DataExtractor::GetDouble(offset_t *offset_ptr) const { 646666cc0b2SZachary Turner typedef double float_type; 647666cc0b2SZachary Turner float_type val = 0.0; 648666cc0b2SZachary Turner const size_t src_size = sizeof(float_type); 64924374aefSJonas Devlieghere const float_type *src = 65024374aefSJonas Devlieghere static_cast<const float_type *>(GetData(offset_ptr, src_size)); 651666cc0b2SZachary Turner if (src) { 652666cc0b2SZachary Turner if (m_byte_order != endian::InlHostByteOrder()) { 65324374aefSJonas Devlieghere const uint8_t *src_data = reinterpret_cast<const uint8_t *>(src); 65424374aefSJonas Devlieghere uint8_t *dst_data = reinterpret_cast<uint8_t *>(&val); 655666cc0b2SZachary Turner for (size_t i = 0; i < sizeof(float_type); ++i) 656666cc0b2SZachary Turner dst_data[sizeof(float_type) - 1 - i] = src_data[i]; 657666cc0b2SZachary Turner } else { 658666cc0b2SZachary Turner val = *src; 659666cc0b2SZachary Turner } 660666cc0b2SZachary Turner } 661666cc0b2SZachary Turner return val; 662666cc0b2SZachary Turner } 663666cc0b2SZachary Turner 664666cc0b2SZachary Turner long double DataExtractor::GetLongDouble(offset_t *offset_ptr) const { 665666cc0b2SZachary Turner long double val = 0.0; 666666cc0b2SZachary Turner #if defined(__i386__) || defined(__amd64__) || defined(__x86_64__) || \ 667666cc0b2SZachary Turner defined(_M_IX86) || defined(_M_IA64) || defined(_M_X64) 668666cc0b2SZachary Turner *offset_ptr += CopyByteOrderedData(*offset_ptr, 10, &val, sizeof(val), 669666cc0b2SZachary Turner endian::InlHostByteOrder()); 670666cc0b2SZachary Turner #else 671666cc0b2SZachary Turner *offset_ptr += CopyByteOrderedData(*offset_ptr, sizeof(val), &val, 672666cc0b2SZachary Turner sizeof(val), endian::InlHostByteOrder()); 673666cc0b2SZachary Turner #endif 674666cc0b2SZachary Turner return val; 675666cc0b2SZachary Turner } 676666cc0b2SZachary Turner 67705097246SAdrian Prantl // Extract a single address from the data and update the offset pointed to by 67805097246SAdrian Prantl // "offset_ptr". The size of the extracted address comes from the 67905097246SAdrian Prantl // "this->m_addr_size" member variable and should be set correctly prior to 68005097246SAdrian Prantl // extracting any address values. 681666cc0b2SZachary Turner // 682666cc0b2SZachary Turner // RETURNS the address that was extracted, or zero on failure. 683666cc0b2SZachary Turner uint64_t DataExtractor::GetAddress(offset_t *offset_ptr) const { 684ec1efe71SAyke van Laethem assert(m_addr_size >= 1 && m_addr_size <= 8); 685666cc0b2SZachary Turner return GetMaxU64(offset_ptr, m_addr_size); 686666cc0b2SZachary Turner } 687666cc0b2SZachary Turner 688666cc0b2SZachary Turner uint64_t DataExtractor::GetAddress_unchecked(offset_t *offset_ptr) const { 689ec1efe71SAyke van Laethem assert(m_addr_size >= 1 && m_addr_size <= 8); 690666cc0b2SZachary Turner return GetMaxU64_unchecked(offset_ptr, m_addr_size); 691666cc0b2SZachary Turner } 692666cc0b2SZachary Turner 693666cc0b2SZachary Turner size_t DataExtractor::ExtractBytes(offset_t offset, offset_t length, 694666cc0b2SZachary Turner ByteOrder dst_byte_order, void *dst) const { 695666cc0b2SZachary Turner const uint8_t *src = PeekData(offset, length); 696666cc0b2SZachary Turner if (src) { 697666cc0b2SZachary Turner if (dst_byte_order != GetByteOrder()) { 698666cc0b2SZachary Turner // Validate that only a word- or register-sized dst is byte swapped 699666cc0b2SZachary Turner assert(length == 1 || length == 2 || length == 4 || length == 8 || 700666cc0b2SZachary Turner length == 10 || length == 16 || length == 32); 701666cc0b2SZachary Turner 702666cc0b2SZachary Turner for (uint32_t i = 0; i < length; ++i) 70324374aefSJonas Devlieghere (static_cast<uint8_t *>(dst))[i] = src[length - i - 1]; 704666cc0b2SZachary Turner } else 705666cc0b2SZachary Turner ::memcpy(dst, src, length); 706666cc0b2SZachary Turner return length; 707666cc0b2SZachary Turner } 708666cc0b2SZachary Turner return 0; 709666cc0b2SZachary Turner } 710666cc0b2SZachary Turner 711666cc0b2SZachary Turner // Extract data as it exists in target memory 712666cc0b2SZachary Turner lldb::offset_t DataExtractor::CopyData(offset_t offset, offset_t length, 713666cc0b2SZachary Turner void *dst) const { 714666cc0b2SZachary Turner const uint8_t *src = PeekData(offset, length); 715666cc0b2SZachary Turner if (src) { 716666cc0b2SZachary Turner ::memcpy(dst, src, length); 717666cc0b2SZachary Turner return length; 718666cc0b2SZachary Turner } 719666cc0b2SZachary Turner return 0; 720666cc0b2SZachary Turner } 721666cc0b2SZachary Turner 722666cc0b2SZachary Turner // Extract data and swap if needed when doing the copy 723666cc0b2SZachary Turner lldb::offset_t 724666cc0b2SZachary Turner DataExtractor::CopyByteOrderedData(offset_t src_offset, offset_t src_len, 725666cc0b2SZachary Turner void *dst_void_ptr, offset_t dst_len, 726666cc0b2SZachary Turner ByteOrder dst_byte_order) const { 727666cc0b2SZachary Turner // Validate the source info 728666cc0b2SZachary Turner if (!ValidOffsetForDataOfSize(src_offset, src_len)) 729666cc0b2SZachary Turner assert(ValidOffsetForDataOfSize(src_offset, src_len)); 730666cc0b2SZachary Turner assert(src_len > 0); 731666cc0b2SZachary Turner assert(m_byte_order == eByteOrderBig || m_byte_order == eByteOrderLittle); 732666cc0b2SZachary Turner 733666cc0b2SZachary Turner // Validate the destination info 734666cc0b2SZachary Turner assert(dst_void_ptr != nullptr); 735666cc0b2SZachary Turner assert(dst_len > 0); 736666cc0b2SZachary Turner assert(dst_byte_order == eByteOrderBig || dst_byte_order == eByteOrderLittle); 737666cc0b2SZachary Turner 738666cc0b2SZachary Turner // Validate that only a word- or register-sized dst is byte swapped 739666cc0b2SZachary Turner assert(dst_byte_order == m_byte_order || dst_len == 1 || dst_len == 2 || 740666cc0b2SZachary Turner dst_len == 4 || dst_len == 8 || dst_len == 10 || dst_len == 16 || 741666cc0b2SZachary Turner dst_len == 32); 742666cc0b2SZachary Turner 743666cc0b2SZachary Turner // Must have valid byte orders set in this object and for destination 744666cc0b2SZachary Turner if (!(dst_byte_order == eByteOrderBig || 745666cc0b2SZachary Turner dst_byte_order == eByteOrderLittle) || 746666cc0b2SZachary Turner !(m_byte_order == eByteOrderBig || m_byte_order == eByteOrderLittle)) 747666cc0b2SZachary Turner return 0; 748666cc0b2SZachary Turner 74924374aefSJonas Devlieghere uint8_t *dst = static_cast<uint8_t *>(dst_void_ptr); 75024374aefSJonas Devlieghere const uint8_t *src = PeekData(src_offset, src_len); 751666cc0b2SZachary Turner if (src) { 752666cc0b2SZachary Turner if (dst_len >= src_len) { 75305097246SAdrian Prantl // We are copying the entire value from src into dst. Calculate how many, 75405097246SAdrian Prantl // if any, zeroes we need for the most significant bytes if "dst_len" is 75505097246SAdrian Prantl // greater than "src_len"... 756666cc0b2SZachary Turner const size_t num_zeroes = dst_len - src_len; 757666cc0b2SZachary Turner if (dst_byte_order == eByteOrderBig) { 758666cc0b2SZachary Turner // Big endian, so we lead with zeroes... 759666cc0b2SZachary Turner if (num_zeroes > 0) 760666cc0b2SZachary Turner ::memset(dst, 0, num_zeroes); 761666cc0b2SZachary Turner // Then either copy or swap the rest 762666cc0b2SZachary Turner if (m_byte_order == eByteOrderBig) { 763666cc0b2SZachary Turner ::memcpy(dst + num_zeroes, src, src_len); 764666cc0b2SZachary Turner } else { 765666cc0b2SZachary Turner for (uint32_t i = 0; i < src_len; ++i) 766666cc0b2SZachary Turner dst[i + num_zeroes] = src[src_len - 1 - i]; 767666cc0b2SZachary Turner } 768666cc0b2SZachary Turner } else { 769666cc0b2SZachary Turner // Little endian destination, so we lead the value bytes 770666cc0b2SZachary Turner if (m_byte_order == eByteOrderBig) { 771666cc0b2SZachary Turner for (uint32_t i = 0; i < src_len; ++i) 772666cc0b2SZachary Turner dst[i] = src[src_len - 1 - i]; 773666cc0b2SZachary Turner } else { 774666cc0b2SZachary Turner ::memcpy(dst, src, src_len); 775666cc0b2SZachary Turner } 776666cc0b2SZachary Turner // And zero the rest... 777666cc0b2SZachary Turner if (num_zeroes > 0) 778666cc0b2SZachary Turner ::memset(dst + src_len, 0, num_zeroes); 779666cc0b2SZachary Turner } 780666cc0b2SZachary Turner return src_len; 781666cc0b2SZachary Turner } else { 782666cc0b2SZachary Turner // We are only copying some of the value from src into dst.. 783666cc0b2SZachary Turner 784666cc0b2SZachary Turner if (dst_byte_order == eByteOrderBig) { 785666cc0b2SZachary Turner // Big endian dst 786666cc0b2SZachary Turner if (m_byte_order == eByteOrderBig) { 787666cc0b2SZachary Turner // Big endian dst, with big endian src 788666cc0b2SZachary Turner ::memcpy(dst, src + (src_len - dst_len), dst_len); 789666cc0b2SZachary Turner } else { 790666cc0b2SZachary Turner // Big endian dst, with little endian src 791666cc0b2SZachary Turner for (uint32_t i = 0; i < dst_len; ++i) 792666cc0b2SZachary Turner dst[i] = src[dst_len - 1 - i]; 793666cc0b2SZachary Turner } 794666cc0b2SZachary Turner } else { 795666cc0b2SZachary Turner // Little endian dst 796666cc0b2SZachary Turner if (m_byte_order == eByteOrderBig) { 797666cc0b2SZachary Turner // Little endian dst, with big endian src 798666cc0b2SZachary Turner for (uint32_t i = 0; i < dst_len; ++i) 799666cc0b2SZachary Turner dst[i] = src[src_len - 1 - i]; 800666cc0b2SZachary Turner } else { 801666cc0b2SZachary Turner // Little endian dst, with big endian src 802666cc0b2SZachary Turner ::memcpy(dst, src, dst_len); 803666cc0b2SZachary Turner } 804666cc0b2SZachary Turner } 805666cc0b2SZachary Turner return dst_len; 806666cc0b2SZachary Turner } 807666cc0b2SZachary Turner } 808666cc0b2SZachary Turner return 0; 809666cc0b2SZachary Turner } 810666cc0b2SZachary Turner 81105097246SAdrian Prantl // Extracts a variable length NULL terminated C string from the data at the 81205097246SAdrian Prantl // offset pointed to by "offset_ptr". The "offset_ptr" will be updated with 81305097246SAdrian Prantl // the offset of the byte that follows the NULL terminator byte. 814666cc0b2SZachary Turner // 81505097246SAdrian Prantl // If the offset pointed to by "offset_ptr" is out of bounds, or if "length" is 81605097246SAdrian Prantl // non-zero and there aren't enough available bytes, nullptr will be returned 81705097246SAdrian Prantl // and "offset_ptr" will not be updated. 818666cc0b2SZachary Turner const char *DataExtractor::GetCStr(offset_t *offset_ptr) const { 819067bb1f5SRaphael Isemann const char *start = reinterpret_cast<const char *>(PeekData(*offset_ptr, 1)); 820067bb1f5SRaphael Isemann // Already at the end of the data. 821067bb1f5SRaphael Isemann if (!start) 822666cc0b2SZachary Turner return nullptr; 823067bb1f5SRaphael Isemann 824067bb1f5SRaphael Isemann const char *end = reinterpret_cast<const char *>(m_end); 825067bb1f5SRaphael Isemann 826067bb1f5SRaphael Isemann // Check all bytes for a null terminator that terminates a C string. 827067bb1f5SRaphael Isemann const char *terminator_or_end = std::find(start, end, '\0'); 828067bb1f5SRaphael Isemann 829067bb1f5SRaphael Isemann // We didn't find a null terminator, so return nullptr to indicate that there 830067bb1f5SRaphael Isemann // is no valid C string at that offset. 831067bb1f5SRaphael Isemann if (terminator_or_end == end) 832067bb1f5SRaphael Isemann return nullptr; 833067bb1f5SRaphael Isemann 834067bb1f5SRaphael Isemann // Update offset_ptr for the caller to point to the data behind the 835067bb1f5SRaphael Isemann // terminator (which is 1 byte long). 836067bb1f5SRaphael Isemann *offset_ptr += (terminator_or_end - start + 1UL); 837067bb1f5SRaphael Isemann return start; 838666cc0b2SZachary Turner } 839666cc0b2SZachary Turner 84005097246SAdrian Prantl // Extracts a NULL terminated C string from the fixed length field of length 84105097246SAdrian Prantl // "len" at the offset pointed to by "offset_ptr". The "offset_ptr" will be 84205097246SAdrian Prantl // updated with the offset of the byte that follows the fixed length field. 843666cc0b2SZachary Turner // 84405097246SAdrian Prantl // If the offset pointed to by "offset_ptr" is out of bounds, or if the offset 84505097246SAdrian Prantl // plus the length of the field is out of bounds, or if the field does not 84605097246SAdrian Prantl // contain a NULL terminator byte, nullptr will be returned and "offset_ptr" 84705097246SAdrian Prantl // will not be updated. 848666cc0b2SZachary Turner const char *DataExtractor::GetCStr(offset_t *offset_ptr, offset_t len) const { 84924374aefSJonas Devlieghere const char *cstr = reinterpret_cast<const char *>(PeekData(*offset_ptr, len)); 850666cc0b2SZachary Turner if (cstr != nullptr) { 851666cc0b2SZachary Turner if (memchr(cstr, '\0', len) == nullptr) { 852666cc0b2SZachary Turner return nullptr; 853666cc0b2SZachary Turner } 854666cc0b2SZachary Turner *offset_ptr += len; 855666cc0b2SZachary Turner return cstr; 856666cc0b2SZachary Turner } 857666cc0b2SZachary Turner return nullptr; 858666cc0b2SZachary Turner } 859666cc0b2SZachary Turner 86005097246SAdrian Prantl // Peeks at a string in the contained data. No verification is done to make 86105097246SAdrian Prantl // sure the entire string lies within the bounds of this object's data, only 86205097246SAdrian Prantl // "offset" is verified to be a valid offset. 863666cc0b2SZachary Turner // 86405097246SAdrian Prantl // Returns a valid C string pointer if "offset" is a valid offset in this 86505097246SAdrian Prantl // object's data, else nullptr is returned. 866666cc0b2SZachary Turner const char *DataExtractor::PeekCStr(offset_t offset) const { 86724374aefSJonas Devlieghere return reinterpret_cast<const char *>(PeekData(offset, 1)); 868666cc0b2SZachary Turner } 869666cc0b2SZachary Turner 87005097246SAdrian Prantl // Extracts an unsigned LEB128 number from this object's data starting at the 87105097246SAdrian Prantl // offset pointed to by "offset_ptr". The offset pointed to by "offset_ptr" 87205097246SAdrian Prantl // will be updated with the offset of the byte following the last extracted 87305097246SAdrian Prantl // byte. 874666cc0b2SZachary Turner // 875666cc0b2SZachary Turner // Returned the extracted integer value. 876666cc0b2SZachary Turner uint64_t DataExtractor::GetULEB128(offset_t *offset_ptr) const { 87724374aefSJonas Devlieghere const uint8_t *src = PeekData(*offset_ptr, 1); 878666cc0b2SZachary Turner if (src == nullptr) 879666cc0b2SZachary Turner return 0; 880666cc0b2SZachary Turner 881*bb9d93f4SRaphael Isemann unsigned byte_count = 0; 882*bb9d93f4SRaphael Isemann uint64_t result = llvm::decodeULEB128(src, &byte_count, m_end); 883*bb9d93f4SRaphael Isemann *offset_ptr += byte_count; 884666cc0b2SZachary Turner return result; 885666cc0b2SZachary Turner } 886666cc0b2SZachary Turner 88705097246SAdrian Prantl // Extracts an signed LEB128 number from this object's data starting at the 88805097246SAdrian Prantl // offset pointed to by "offset_ptr". The offset pointed to by "offset_ptr" 88905097246SAdrian Prantl // will be updated with the offset of the byte following the last extracted 89005097246SAdrian Prantl // byte. 891666cc0b2SZachary Turner // 892666cc0b2SZachary Turner // Returned the extracted integer value. 893666cc0b2SZachary Turner int64_t DataExtractor::GetSLEB128(offset_t *offset_ptr) const { 89424374aefSJonas Devlieghere const uint8_t *src = PeekData(*offset_ptr, 1); 895666cc0b2SZachary Turner if (src == nullptr) 896666cc0b2SZachary Turner return 0; 897666cc0b2SZachary Turner 898*bb9d93f4SRaphael Isemann unsigned byte_count = 0; 899*bb9d93f4SRaphael Isemann int64_t result = llvm::decodeSLEB128(src, &byte_count, m_end); 900*bb9d93f4SRaphael Isemann *offset_ptr += byte_count; 901666cc0b2SZachary Turner return result; 902666cc0b2SZachary Turner } 903666cc0b2SZachary Turner 90405097246SAdrian Prantl // Skips a ULEB128 number (signed or unsigned) from this object's data starting 90505097246SAdrian Prantl // at the offset pointed to by "offset_ptr". The offset pointed to by 90605097246SAdrian Prantl // "offset_ptr" will be updated with the offset of the byte following the last 90705097246SAdrian Prantl // extracted byte. 908666cc0b2SZachary Turner // 909666cc0b2SZachary Turner // Returns the number of bytes consumed during the extraction. 910666cc0b2SZachary Turner uint32_t DataExtractor::Skip_LEB128(offset_t *offset_ptr) const { 911666cc0b2SZachary Turner uint32_t bytes_consumed = 0; 91224374aefSJonas Devlieghere const uint8_t *src = PeekData(*offset_ptr, 1); 913666cc0b2SZachary Turner if (src == nullptr) 914666cc0b2SZachary Turner return 0; 915666cc0b2SZachary Turner 916666cc0b2SZachary Turner const uint8_t *end = m_end; 917666cc0b2SZachary Turner 918666cc0b2SZachary Turner if (src < end) { 919666cc0b2SZachary Turner const uint8_t *src_pos = src; 920666cc0b2SZachary Turner while ((src_pos < end) && (*src_pos++ & 0x80)) 921666cc0b2SZachary Turner ++bytes_consumed; 922666cc0b2SZachary Turner *offset_ptr += src_pos - src; 923666cc0b2SZachary Turner } 924666cc0b2SZachary Turner return bytes_consumed; 925666cc0b2SZachary Turner } 926666cc0b2SZachary Turner 927666cc0b2SZachary Turner // Dumps bytes from this object's data to the stream "s" starting 92805097246SAdrian Prantl // "start_offset" bytes into this data, and ending with the byte before 92905097246SAdrian Prantl // "end_offset". "base_addr" will be added to the offset into the dumped data 93005097246SAdrian Prantl // when showing the offset into the data in the output information. 93105097246SAdrian Prantl // "num_per_line" objects of type "type" will be dumped with the option to 93205097246SAdrian Prantl // override the format for each object with "type_format". "type_format" is a 93305097246SAdrian Prantl // printf style formatting string. If "type_format" is nullptr, then an 93405097246SAdrian Prantl // appropriate format string will be used for the supplied "type". If the 93505097246SAdrian Prantl // stream "s" is nullptr, then the output will be send to Log(). 936666cc0b2SZachary Turner lldb::offset_t DataExtractor::PutToLog(Log *log, offset_t start_offset, 937666cc0b2SZachary Turner offset_t length, uint64_t base_addr, 938666cc0b2SZachary Turner uint32_t num_per_line, 939b6e2cf32SRaphael Isemann DataExtractor::Type type) const { 940666cc0b2SZachary Turner if (log == nullptr) 941666cc0b2SZachary Turner return start_offset; 942666cc0b2SZachary Turner 943666cc0b2SZachary Turner offset_t offset; 944666cc0b2SZachary Turner offset_t end_offset; 945666cc0b2SZachary Turner uint32_t count; 946666cc0b2SZachary Turner StreamString sstr; 947666cc0b2SZachary Turner for (offset = start_offset, end_offset = offset + length, count = 0; 948666cc0b2SZachary Turner ValidOffset(offset) && offset < end_offset; ++count) { 949666cc0b2SZachary Turner if ((count % num_per_line) == 0) { 950666cc0b2SZachary Turner // Print out any previous string 951666cc0b2SZachary Turner if (sstr.GetSize() > 0) { 952666cc0b2SZachary Turner log->PutString(sstr.GetString()); 953666cc0b2SZachary Turner sstr.Clear(); 954666cc0b2SZachary Turner } 955666cc0b2SZachary Turner // Reset string offset and fill the current line string with address: 956666cc0b2SZachary Turner if (base_addr != LLDB_INVALID_ADDRESS) 957666cc0b2SZachary Turner sstr.Printf("0x%8.8" PRIx64 ":", 95824374aefSJonas Devlieghere static_cast<uint64_t>(base_addr + (offset - start_offset))); 959666cc0b2SZachary Turner } 960666cc0b2SZachary Turner 961666cc0b2SZachary Turner switch (type) { 962666cc0b2SZachary Turner case TypeUInt8: 963b6e2cf32SRaphael Isemann sstr.Printf(" %2.2x", GetU8(&offset)); 964666cc0b2SZachary Turner break; 965666cc0b2SZachary Turner case TypeChar: { 966666cc0b2SZachary Turner char ch = GetU8(&offset); 967b6e2cf32SRaphael Isemann sstr.Printf(" %c", isprint(ch) ? ch : ' '); 968666cc0b2SZachary Turner } break; 969666cc0b2SZachary Turner case TypeUInt16: 970b6e2cf32SRaphael Isemann sstr.Printf(" %4.4x", GetU16(&offset)); 971666cc0b2SZachary Turner break; 972666cc0b2SZachary Turner case TypeUInt32: 973b6e2cf32SRaphael Isemann sstr.Printf(" %8.8x", GetU32(&offset)); 974666cc0b2SZachary Turner break; 975666cc0b2SZachary Turner case TypeUInt64: 976b6e2cf32SRaphael Isemann sstr.Printf(" %16.16" PRIx64, GetU64(&offset)); 977666cc0b2SZachary Turner break; 978666cc0b2SZachary Turner case TypePointer: 979b6e2cf32SRaphael Isemann sstr.Printf(" 0x%" PRIx64, GetAddress(&offset)); 980666cc0b2SZachary Turner break; 981666cc0b2SZachary Turner case TypeULEB128: 982b6e2cf32SRaphael Isemann sstr.Printf(" 0x%" PRIx64, GetULEB128(&offset)); 983666cc0b2SZachary Turner break; 984666cc0b2SZachary Turner case TypeSLEB128: 985b6e2cf32SRaphael Isemann sstr.Printf(" %" PRId64, GetSLEB128(&offset)); 986666cc0b2SZachary Turner break; 987666cc0b2SZachary Turner } 988666cc0b2SZachary Turner } 989666cc0b2SZachary Turner 990666cc0b2SZachary Turner if (!sstr.Empty()) 991666cc0b2SZachary Turner log->PutString(sstr.GetString()); 992666cc0b2SZachary Turner 993666cc0b2SZachary Turner return offset; // Return the offset at which we ended up 994666cc0b2SZachary Turner } 995666cc0b2SZachary Turner 996666cc0b2SZachary Turner size_t DataExtractor::Copy(DataExtractor &dest_data) const { 997666cc0b2SZachary Turner if (m_data_sp) { 998666cc0b2SZachary Turner // we can pass along the SP to the data 999666cc0b2SZachary Turner dest_data.SetData(m_data_sp); 1000666cc0b2SZachary Turner } else { 1001666cc0b2SZachary Turner const uint8_t *base_ptr = m_start; 1002666cc0b2SZachary Turner size_t data_size = GetByteSize(); 1003666cc0b2SZachary Turner dest_data.SetData(DataBufferSP(new DataBufferHeap(base_ptr, data_size))); 1004666cc0b2SZachary Turner } 1005666cc0b2SZachary Turner return GetByteSize(); 1006666cc0b2SZachary Turner } 1007666cc0b2SZachary Turner 1008666cc0b2SZachary Turner bool DataExtractor::Append(DataExtractor &rhs) { 1009666cc0b2SZachary Turner if (rhs.GetByteOrder() != GetByteOrder()) 1010666cc0b2SZachary Turner return false; 1011666cc0b2SZachary Turner 1012666cc0b2SZachary Turner if (rhs.GetByteSize() == 0) 1013666cc0b2SZachary Turner return true; 1014666cc0b2SZachary Turner 1015666cc0b2SZachary Turner if (GetByteSize() == 0) 1016666cc0b2SZachary Turner return (rhs.Copy(*this) > 0); 1017666cc0b2SZachary Turner 1018666cc0b2SZachary Turner size_t bytes = GetByteSize() + rhs.GetByteSize(); 1019666cc0b2SZachary Turner 1020666cc0b2SZachary Turner DataBufferHeap *buffer_heap_ptr = nullptr; 1021666cc0b2SZachary Turner DataBufferSP buffer_sp(buffer_heap_ptr = new DataBufferHeap(bytes, 0)); 1022666cc0b2SZachary Turner 1023666cc0b2SZachary Turner if (!buffer_sp || buffer_heap_ptr == nullptr) 1024666cc0b2SZachary Turner return false; 1025666cc0b2SZachary Turner 1026666cc0b2SZachary Turner uint8_t *bytes_ptr = buffer_heap_ptr->GetBytes(); 1027666cc0b2SZachary Turner 1028666cc0b2SZachary Turner memcpy(bytes_ptr, GetDataStart(), GetByteSize()); 1029666cc0b2SZachary Turner memcpy(bytes_ptr + GetByteSize(), rhs.GetDataStart(), rhs.GetByteSize()); 1030666cc0b2SZachary Turner 1031666cc0b2SZachary Turner SetData(buffer_sp); 1032666cc0b2SZachary Turner 1033666cc0b2SZachary Turner return true; 1034666cc0b2SZachary Turner } 1035666cc0b2SZachary Turner 1036666cc0b2SZachary Turner bool DataExtractor::Append(void *buf, offset_t length) { 1037666cc0b2SZachary Turner if (buf == nullptr) 1038666cc0b2SZachary Turner return false; 1039666cc0b2SZachary Turner 1040666cc0b2SZachary Turner if (length == 0) 1041666cc0b2SZachary Turner return true; 1042666cc0b2SZachary Turner 1043666cc0b2SZachary Turner size_t bytes = GetByteSize() + length; 1044666cc0b2SZachary Turner 1045666cc0b2SZachary Turner DataBufferHeap *buffer_heap_ptr = nullptr; 1046666cc0b2SZachary Turner DataBufferSP buffer_sp(buffer_heap_ptr = new DataBufferHeap(bytes, 0)); 1047666cc0b2SZachary Turner 1048666cc0b2SZachary Turner if (!buffer_sp || buffer_heap_ptr == nullptr) 1049666cc0b2SZachary Turner return false; 1050666cc0b2SZachary Turner 1051666cc0b2SZachary Turner uint8_t *bytes_ptr = buffer_heap_ptr->GetBytes(); 1052666cc0b2SZachary Turner 1053666cc0b2SZachary Turner if (GetByteSize() > 0) 1054666cc0b2SZachary Turner memcpy(bytes_ptr, GetDataStart(), GetByteSize()); 1055666cc0b2SZachary Turner 1056666cc0b2SZachary Turner memcpy(bytes_ptr + GetByteSize(), buf, length); 1057666cc0b2SZachary Turner 1058666cc0b2SZachary Turner SetData(buffer_sp); 1059666cc0b2SZachary Turner 1060666cc0b2SZachary Turner return true; 1061666cc0b2SZachary Turner } 1062666cc0b2SZachary Turner 1063666cc0b2SZachary Turner void DataExtractor::Checksum(llvm::SmallVectorImpl<uint8_t> &dest, 1064666cc0b2SZachary Turner uint64_t max_data) { 1065666cc0b2SZachary Turner if (max_data == 0) 1066666cc0b2SZachary Turner max_data = GetByteSize(); 1067666cc0b2SZachary Turner else 1068666cc0b2SZachary Turner max_data = std::min(max_data, GetByteSize()); 1069666cc0b2SZachary Turner 1070666cc0b2SZachary Turner llvm::MD5 md5; 1071666cc0b2SZachary Turner 1072666cc0b2SZachary Turner const llvm::ArrayRef<uint8_t> data(GetDataStart(), max_data); 1073666cc0b2SZachary Turner md5.update(data); 1074666cc0b2SZachary Turner 1075666cc0b2SZachary Turner llvm::MD5::MD5Result result; 1076666cc0b2SZachary Turner md5.final(result); 1077666cc0b2SZachary Turner 107882a0c97bSZachary Turner dest.clear(); 107982a0c97bSZachary Turner dest.append(result.Bytes.begin(), result.Bytes.end()); 1080666cc0b2SZachary Turner } 1081