2015-10-20 09:10:59 +08:00
|
|
|
//===-- NSIndexPath.cpp -----------------------------------------*- C++ -*-===//
|
2014-10-16 05:38:32 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2015-10-20 09:10:59 +08:00
|
|
|
// C Includes
|
|
|
|
// C++ Includes
|
|
|
|
// Other libraries and framework includes
|
|
|
|
// Project includes
|
2015-09-15 06:18:32 +08:00
|
|
|
#include "Cocoa.h"
|
2014-10-16 05:38:32 +08:00
|
|
|
|
|
|
|
#include "lldb/Core/ValueObject.h"
|
|
|
|
#include "lldb/Core/ValueObjectConstResult.h"
|
2015-09-04 08:33:51 +08:00
|
|
|
#include "lldb/DataFormatters/FormattersHelpers.h"
|
2014-10-16 05:38:32 +08:00
|
|
|
#include "lldb/DataFormatters/TypeSynthetic.h"
|
2015-10-20 12:50:09 +08:00
|
|
|
#include "lldb/Symbol/ClangASTContext.h"
|
2014-10-16 05:38:32 +08:00
|
|
|
#include "lldb/Target/ObjCLanguageRuntime.h"
|
|
|
|
#include "lldb/Target/Process.h"
|
2015-10-20 12:50:09 +08:00
|
|
|
#include "lldb/Target/Target.h"
|
2014-10-16 05:38:32 +08:00
|
|
|
|
|
|
|
using namespace lldb;
|
|
|
|
using namespace lldb_private;
|
|
|
|
using namespace lldb_private::formatters;
|
|
|
|
|
|
|
|
class NSIndexPathSyntheticFrontEnd : public SyntheticChildrenFrontEnd
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
NSIndexPathSyntheticFrontEnd (lldb::ValueObjectSP valobj_sp) :
|
|
|
|
SyntheticChildrenFrontEnd (*valobj_sp.get()),
|
2016-07-13 02:33:52 +08:00
|
|
|
m_descriptor_sp(nullptr),
|
|
|
|
m_impl(),
|
2014-10-16 05:38:32 +08:00
|
|
|
m_ptr_size(0),
|
|
|
|
m_uint_star_type()
|
|
|
|
{
|
|
|
|
m_ptr_size = m_backend.GetTargetSP()->GetArchitecture().GetAddressByteSize();
|
|
|
|
}
|
2015-10-20 09:10:59 +08:00
|
|
|
|
|
|
|
~NSIndexPathSyntheticFrontEnd() override = default;
|
|
|
|
|
|
|
|
size_t
|
|
|
|
CalculateNumChildren() override
|
2014-10-16 05:38:32 +08:00
|
|
|
{
|
|
|
|
return m_impl.GetNumIndexes();
|
|
|
|
}
|
|
|
|
|
2015-10-20 09:10:59 +08:00
|
|
|
lldb::ValueObjectSP
|
|
|
|
GetChildAtIndex(size_t idx) override
|
2014-10-16 05:38:32 +08:00
|
|
|
{
|
|
|
|
return m_impl.GetIndexAtIndex(idx, m_uint_star_type);
|
|
|
|
}
|
|
|
|
|
2015-10-20 09:10:59 +08:00
|
|
|
bool
|
|
|
|
Update() override
|
2014-10-16 05:38:32 +08:00
|
|
|
{
|
2015-06-17 04:48:49 +08:00
|
|
|
m_impl.Clear();
|
2014-10-16 05:38:32 +08:00
|
|
|
|
2015-08-25 07:46:31 +08:00
|
|
|
TypeSystem* type_system = m_backend.GetCompilerType().GetTypeSystem();
|
2015-08-12 05:38:15 +08:00
|
|
|
if (!type_system)
|
|
|
|
return false;
|
2015-09-09 02:15:05 +08:00
|
|
|
|
|
|
|
ClangASTContext *ast = m_backend.GetExecutionContextRef().GetTargetSP()->GetScratchClangASTContext();
|
|
|
|
if (!ast)
|
2014-10-16 05:38:32 +08:00
|
|
|
return false;
|
2015-09-09 02:15:05 +08:00
|
|
|
|
|
|
|
m_uint_star_type = ast->GetPointerSizedIntType(false);
|
2014-10-16 05:38:32 +08:00
|
|
|
|
|
|
|
static ConstString g__indexes("_indexes");
|
|
|
|
static ConstString g__length("_length");
|
|
|
|
|
|
|
|
ProcessSP process_sp = m_backend.GetProcessSP();
|
|
|
|
if (!process_sp)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ObjCLanguageRuntime* runtime = (ObjCLanguageRuntime*)process_sp->GetLanguageRuntime(lldb::eLanguageTypeObjC);
|
|
|
|
|
|
|
|
if (!runtime)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ObjCLanguageRuntime::ClassDescriptorSP descriptor(runtime->GetClassDescriptor(m_backend));
|
|
|
|
|
|
|
|
if (!descriptor.get() || !descriptor->IsValid())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
uint64_t info_bits(0),value_bits(0),payload(0);
|
|
|
|
|
|
|
|
if (descriptor->GetTaggedPointerInfo(&info_bits, &value_bits, &payload))
|
|
|
|
{
|
|
|
|
m_impl.m_inlined.SetIndexes(payload, *process_sp);
|
2015-06-17 04:48:49 +08:00
|
|
|
m_impl.m_mode = Mode::Inlined;
|
2014-10-16 05:38:32 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ObjCLanguageRuntime::ClassDescriptor::iVarDescriptor _indexes_id;
|
|
|
|
ObjCLanguageRuntime::ClassDescriptor::iVarDescriptor _length_id;
|
|
|
|
|
|
|
|
bool has_indexes(false),has_length(false);
|
|
|
|
|
2015-01-10 04:15:03 +08:00
|
|
|
for (size_t x = 0;
|
2014-10-16 05:38:32 +08:00
|
|
|
x < descriptor->GetNumIVars();
|
|
|
|
x++)
|
|
|
|
{
|
|
|
|
const auto& ivar = descriptor->GetIVarAtIndex(x);
|
|
|
|
if (ivar.m_name == g__indexes)
|
|
|
|
{
|
|
|
|
_indexes_id = ivar;
|
|
|
|
has_indexes = true;
|
|
|
|
}
|
|
|
|
else if (ivar.m_name == g__length)
|
|
|
|
{
|
|
|
|
_length_id = ivar;
|
|
|
|
has_length = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (has_length && has_indexes)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (has_length && has_indexes)
|
|
|
|
{
|
|
|
|
m_impl.m_outsourced.m_indexes = m_backend.GetSyntheticChildAtOffset(_indexes_id.m_offset,
|
|
|
|
m_uint_star_type.GetPointerType(),
|
|
|
|
true).get();
|
|
|
|
ValueObjectSP length_sp(m_backend.GetSyntheticChildAtOffset(_length_id.m_offset,
|
|
|
|
m_uint_star_type,
|
|
|
|
true));
|
|
|
|
if (length_sp)
|
|
|
|
{
|
|
|
|
m_impl.m_outsourced.m_count = length_sp->GetValueAsUnsigned(0);
|
|
|
|
if (m_impl.m_outsourced.m_indexes)
|
|
|
|
m_impl.m_mode = Mode::Outsourced;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2015-10-20 09:10:59 +08:00
|
|
|
bool
|
|
|
|
MightHaveChildren() override
|
2014-10-16 05:38:32 +08:00
|
|
|
{
|
|
|
|
if (m_impl.m_mode == Mode::Invalid)
|
|
|
|
return false;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2015-10-20 09:10:59 +08:00
|
|
|
size_t
|
|
|
|
GetIndexOfChildWithName(const ConstString &name) override
|
2014-10-16 05:38:32 +08:00
|
|
|
{
|
|
|
|
const char* item_name = name.GetCString();
|
|
|
|
uint32_t idx = ExtractIndexFromString(item_name);
|
|
|
|
if (idx < UINT32_MAX && idx >= CalculateNumChildren())
|
|
|
|
return UINT32_MAX;
|
|
|
|
return idx;
|
|
|
|
}
|
|
|
|
|
2015-10-20 09:10:59 +08:00
|
|
|
lldb::ValueObjectSP
|
|
|
|
GetSyntheticValue() override
|
|
|
|
{
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
2014-10-16 05:38:32 +08:00
|
|
|
protected:
|
|
|
|
ObjCLanguageRuntime::ClassDescriptorSP m_descriptor_sp;
|
|
|
|
|
|
|
|
enum class Mode {
|
|
|
|
Inlined,
|
|
|
|
Outsourced,
|
|
|
|
Invalid
|
|
|
|
};
|
|
|
|
|
2016-07-13 02:33:52 +08:00
|
|
|
struct Impl
|
|
|
|
{
|
2014-10-16 05:38:32 +08:00
|
|
|
size_t
|
|
|
|
GetNumIndexes ()
|
|
|
|
{
|
|
|
|
switch (m_mode)
|
|
|
|
{
|
|
|
|
case Mode::Inlined:
|
|
|
|
return m_inlined.GetNumIndexes();
|
|
|
|
case Mode::Outsourced:
|
|
|
|
return m_outsourced.m_count;
|
|
|
|
default:
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
lldb::ValueObjectSP
|
2015-08-12 06:53:00 +08:00
|
|
|
GetIndexAtIndex (size_t idx, const CompilerType& desired_type)
|
2014-10-16 05:38:32 +08:00
|
|
|
{
|
|
|
|
if (idx >= GetNumIndexes())
|
|
|
|
return nullptr;
|
|
|
|
switch (m_mode)
|
|
|
|
{
|
|
|
|
default: return nullptr;
|
|
|
|
case Mode::Inlined:
|
|
|
|
return m_inlined.GetIndexAtIndex (idx, desired_type);
|
|
|
|
case Mode::Outsourced:
|
|
|
|
return m_outsourced.GetIndexAtIndex (idx);
|
|
|
|
}
|
|
|
|
}
|
2016-07-13 02:33:52 +08:00
|
|
|
|
|
|
|
struct InlinedIndexes
|
|
|
|
{
|
2015-06-17 04:48:49 +08:00
|
|
|
public:
|
2016-07-13 02:33:52 +08:00
|
|
|
void SetIndexes(uint64_t value, Process& p)
|
|
|
|
{
|
|
|
|
m_indexes = value;
|
|
|
|
_lengthForInlinePayload(p.GetAddressByteSize());
|
|
|
|
m_process = &p;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t
|
|
|
|
GetNumIndexes ()
|
|
|
|
{
|
|
|
|
return m_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
lldb::ValueObjectSP
|
|
|
|
GetIndexAtIndex (size_t idx, const CompilerType& desired_type)
|
|
|
|
{
|
|
|
|
if (!m_process)
|
|
|
|
return nullptr;
|
2015-06-17 04:48:49 +08:00
|
|
|
|
2016-07-13 02:33:52 +08:00
|
|
|
std::pair<uint64_t, bool> value(_indexAtPositionForInlinePayload(idx));
|
|
|
|
if (!value.second)
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
Value v;
|
|
|
|
if (m_ptr_size == 8)
|
|
|
|
{
|
|
|
|
Scalar scalar( (unsigned long long)value.first );
|
|
|
|
v = Value(scalar);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Scalar scalar( (unsigned int)value.first );
|
|
|
|
v = Value(scalar);
|
|
|
|
}
|
|
|
|
|
|
|
|
v.SetCompilerType(desired_type);
|
|
|
|
|
|
|
|
StreamString idx_name;
|
|
|
|
idx_name.Printf("[%" PRIu64 "]", (uint64_t)idx);
|
|
|
|
|
|
|
|
return ValueObjectConstResult::Create(m_process, v, ConstString(idx_name.GetData()));
|
|
|
|
}
|
2015-06-17 04:48:49 +08:00
|
|
|
|
|
|
|
void
|
|
|
|
Clear ()
|
|
|
|
{
|
|
|
|
m_indexes = 0;
|
|
|
|
m_count = 0;
|
|
|
|
m_ptr_size = 0;
|
|
|
|
m_process = nullptr;
|
|
|
|
}
|
2016-07-13 02:33:52 +08:00
|
|
|
|
|
|
|
InlinedIndexes () :
|
|
|
|
m_indexes(0),
|
|
|
|
m_count(0),
|
|
|
|
m_ptr_size(0),
|
|
|
|
m_process(nullptr)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2015-10-20 09:10:59 +08:00
|
|
|
private:
|
2016-07-13 02:33:52 +08:00
|
|
|
uint64_t m_indexes;
|
|
|
|
size_t m_count;
|
|
|
|
uint32_t m_ptr_size;
|
|
|
|
Process *m_process;
|
|
|
|
|
|
|
|
// cfr. Foundation for the details of this code
|
|
|
|
size_t _lengthForInlinePayload(uint32_t ptr_size) {
|
|
|
|
m_ptr_size = ptr_size;
|
|
|
|
if (m_ptr_size == 8)
|
|
|
|
m_count = ((m_indexes >> 3) & 0x7);
|
|
|
|
else
|
|
|
|
m_count = ((m_indexes >> 3) & 0x3);
|
|
|
|
return m_count;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<uint64_t, bool>
|
|
|
|
_indexAtPositionForInlinePayload(size_t pos)
|
|
|
|
{
|
|
|
|
if (m_ptr_size == 8)
|
|
|
|
{
|
|
|
|
switch (pos) {
|
|
|
|
case 5: return {((m_indexes >> 51) & 0x1ff),true};
|
|
|
|
case 4: return {((m_indexes >> 42) & 0x1ff),true};
|
|
|
|
case 3: return {((m_indexes >> 33) & 0x1ff),true};
|
|
|
|
case 2: return {((m_indexes >> 24) & 0x1ff),true};
|
|
|
|
case 1: return {((m_indexes >> 15) & 0x1ff),true};
|
|
|
|
case 0: return {((m_indexes >> 6) & 0x1ff),true};
|
|
|
|
}
|
2015-06-17 04:48:49 +08:00
|
|
|
}
|
2016-07-13 02:33:52 +08:00
|
|
|
else
|
|
|
|
{
|
|
|
|
switch (pos) {
|
|
|
|
case 2: return {((m_indexes >> 23) & 0x1ff),true};
|
|
|
|
case 1: return {((m_indexes >> 14) & 0x1ff),true};
|
|
|
|
case 0: return {((m_indexes >> 5) & 0x1ff),true};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return {0,false};
|
|
|
|
}
|
|
|
|
|
2015-06-17 04:48:49 +08:00
|
|
|
};
|
2016-07-13 02:33:52 +08:00
|
|
|
|
|
|
|
struct OutsourcedIndexes
|
|
|
|
{
|
2015-06-17 04:48:49 +08:00
|
|
|
lldb::ValueObjectSP
|
|
|
|
GetIndexAtIndex (size_t idx)
|
|
|
|
{
|
|
|
|
if (m_indexes)
|
|
|
|
{
|
|
|
|
ValueObjectSP index_sp(m_indexes->GetSyntheticArrayMember(idx, true));
|
|
|
|
return index_sp;
|
|
|
|
}
|
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Clear ()
|
|
|
|
{
|
|
|
|
m_indexes = nullptr;
|
|
|
|
m_count = 0;
|
|
|
|
}
|
2016-07-13 02:33:52 +08:00
|
|
|
|
|
|
|
OutsourcedIndexes () :
|
|
|
|
m_indexes(nullptr),
|
|
|
|
m_count(0)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
ValueObject *m_indexes;
|
|
|
|
size_t m_count;
|
2015-06-17 04:48:49 +08:00
|
|
|
};
|
2016-07-13 02:33:52 +08:00
|
|
|
|
|
|
|
union
|
|
|
|
{
|
2015-06-17 04:48:49 +08:00
|
|
|
struct InlinedIndexes m_inlined;
|
|
|
|
struct OutsourcedIndexes m_outsourced;
|
2014-10-16 05:38:32 +08:00
|
|
|
};
|
2015-06-17 04:48:49 +08:00
|
|
|
|
|
|
|
void
|
|
|
|
Clear ()
|
|
|
|
{
|
|
|
|
m_mode = Mode::Invalid;
|
|
|
|
m_inlined.Clear();
|
|
|
|
m_outsourced.Clear();
|
|
|
|
}
|
2016-07-13 02:33:52 +08:00
|
|
|
|
|
|
|
Impl() :
|
|
|
|
m_mode(Mode::Invalid)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
Mode m_mode;
|
2014-10-16 05:38:32 +08:00
|
|
|
} m_impl;
|
|
|
|
|
|
|
|
uint32_t m_ptr_size;
|
2015-08-12 06:53:00 +08:00
|
|
|
CompilerType m_uint_star_type;
|
2014-10-16 05:38:32 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
namespace lldb_private {
|
|
|
|
namespace formatters {
|
|
|
|
|
|
|
|
SyntheticChildrenFrontEnd* NSIndexPathSyntheticFrontEndCreator (CXXSyntheticChildren*, lldb::ValueObjectSP valobj_sp)
|
|
|
|
{
|
|
|
|
if (valobj_sp)
|
|
|
|
return new NSIndexPathSyntheticFrontEnd(valobj_sp);
|
|
|
|
return nullptr;
|
|
|
|
}
|
2015-10-20 09:10:59 +08:00
|
|
|
|
|
|
|
} // namespace formatters
|
|
|
|
} // namespace lldb_private
|