blob: ade29c0a7cf7fd5c655242da0284c50a48c0fb14 [file] [log] [blame]
//===---------------------ProcessStructReader.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_TARGET_PROCESSSTRUCTREADER_H
#define LLDB_TARGET_PROCESSSTRUCTREADER_H
#include "lldb/lldb-defines.h"
#include "lldb/lldb-types.h"
#include "lldb/Symbol/CompilerType.h"
#include "lldb/Target/Process.h"
#include "lldb/Utility/ConstString.h"
#include "lldb/Utility/DataBufferHeap.h"
#include "lldb/Utility/DataExtractor.h"
#include "lldb/Utility/Status.h"
#include <initializer_list>
#include <map>
#include <string>
namespace lldb_private {
class ProcessStructReader {
protected:
struct FieldImpl {
CompilerType type;
size_t offset;
size_t size;
};
std::map<ConstString, FieldImpl> m_fields;
DataExtractor m_data;
lldb::ByteOrder m_byte_order;
size_t m_addr_byte_size;
public:
ProcessStructReader(Process *process, lldb::addr_t base_addr,
CompilerType struct_type) {
if (!process)
return;
if (base_addr == 0 || base_addr == LLDB_INVALID_ADDRESS)
return;
m_byte_order = process->GetByteOrder();
m_addr_byte_size = process->GetAddressByteSize();
for (size_t idx = 0; idx < struct_type.GetNumFields(); idx++) {
std::string name;
uint64_t bit_offset;
uint32_t bitfield_bit_size;
bool is_bitfield;
CompilerType field_type = struct_type.GetFieldAtIndex(
idx, name, &bit_offset, &bitfield_bit_size, &is_bitfield);
// no support for bitfields in here (yet)
if (is_bitfield)
return;
auto size = field_type.GetByteSize(nullptr);
// no support for things larger than a uint64_t (yet)
if (!size || *size > 8)
return;
ConstString const_name = ConstString(name.c_str());
size_t byte_index = static_cast<size_t>(bit_offset / 8);
m_fields[const_name] =
FieldImpl{field_type, byte_index, static_cast<size_t>(*size)};
}
auto total_size = struct_type.GetByteSize(nullptr);
if (!total_size)
return;
lldb::DataBufferSP buffer_sp(new DataBufferHeap(*total_size, 0));
Status error;
process->ReadMemoryFromInferior(base_addr, buffer_sp->GetBytes(),
*total_size, error);
if (error.Fail())
return;
m_data = DataExtractor(buffer_sp, m_byte_order, m_addr_byte_size);
}
template <typename RetType>
RetType GetField(ConstString name, RetType fail_value = RetType()) {
auto iter = m_fields.find(name), end = m_fields.end();
if (iter == end)
return fail_value;
auto size = iter->second.size;
if (sizeof(RetType) < size)
return fail_value;
lldb::offset_t offset = iter->second.offset;
if (offset + size > m_data.GetByteSize())
return fail_value;
return (RetType)(m_data.GetMaxU64(&offset, size));
}
size_t GetOffsetOf(ConstString name, size_t fail_value = SIZE_MAX) {
auto iter = m_fields.find(name), end = m_fields.end();
if (iter == end)
return fail_value;
return iter->second.offset;
}
};
}
#endif // utility_ProcessStructReader_h_